blob: 8c50d3b3fabbd8a2de3635d620c7a12b59206e26 [file] [log] [blame]
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001/*
2 * This file is part of wl1271
3 *
Juuso Oikarinen8bf29b02010-02-18 13:25:51 +02004 * Copyright (C) 2008-2010 Nokia Corporation
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03005 *
6 * Contact: Luciano Coelho <luciano.coelho@nokia.com>
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * version 2 as published by the Free Software Foundation.
11 *
12 * This program is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
20 * 02110-1301 USA
21 *
22 */
23
24#include <linux/module.h>
Luciano Coelhof5fc0f82009-08-06 16:25:28 +030025#include <linux/firmware.h>
26#include <linux/delay.h>
Luciano Coelhof5fc0f82009-08-06 16:25:28 +030027#include <linux/spi/spi.h>
28#include <linux/crc32.h>
29#include <linux/etherdevice.h>
Juuso Oikarinen1fba4972009-10-08 21:56:32 +030030#include <linux/vmalloc.h>
Juuso Oikarinena1dd8182010-03-18 12:26:31 +020031#include <linux/platform_device.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090032#include <linux/slab.h>
Luciano Coelhof5fc0f82009-08-06 16:25:28 +030033
Shahar Levi00d20102010-11-08 11:20:10 +000034#include "wl12xx.h"
Luciano Coelhof5fc0f82009-08-06 16:25:28 +030035#include "wl12xx_80211.h"
Shahar Levi00d20102010-11-08 11:20:10 +000036#include "reg.h"
37#include "io.h"
38#include "event.h"
39#include "tx.h"
40#include "rx.h"
41#include "ps.h"
42#include "init.h"
43#include "debugfs.h"
44#include "cmd.h"
45#include "boot.h"
46#include "testmode.h"
47#include "scan.h"
Luciano Coelhof5fc0f82009-08-06 16:25:28 +030048
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +020049#define WL1271_BOOT_RETRIES 3
50
Juuso Oikarinen8a080482009-10-13 12:47:44 +030051static struct conf_drv_settings default_conf = {
52 .sg = {
Juuso Oikarinen1b00f542010-03-18 12:26:30 +020053 .params = {
54 [CONF_SG_BT_PER_THRESHOLD] = 7500,
55 [CONF_SG_HV3_MAX_OVERRIDE] = 0,
56 [CONF_SG_BT_NFS_SAMPLE_INTERVAL] = 400,
57 [CONF_SG_BT_LOAD_RATIO] = 50,
Juuso Oikarinen8d2ef7b2010-07-08 17:50:03 +030058 [CONF_SG_AUTO_PS_MODE] = 1,
Juuso Oikarinen1b00f542010-03-18 12:26:30 +020059 [CONF_SG_AUTO_SCAN_PROBE_REQ] = 170,
60 [CONF_SG_ACTIVE_SCAN_DURATION_FACTOR_HV3] = 50,
61 [CONF_SG_ANTENNA_CONFIGURATION] = 0,
62 [CONF_SG_BEACON_MISS_PERCENT] = 60,
63 [CONF_SG_RATE_ADAPT_THRESH] = 12,
64 [CONF_SG_RATE_ADAPT_SNR] = 0,
65 [CONF_SG_WLAN_PS_BT_ACL_MASTER_MIN_BR] = 10,
66 [CONF_SG_WLAN_PS_BT_ACL_MASTER_MAX_BR] = 30,
67 [CONF_SG_WLAN_PS_MAX_BT_ACL_MASTER_BR] = 8,
68 [CONF_SG_WLAN_PS_BT_ACL_SLAVE_MIN_BR] = 20,
69 [CONF_SG_WLAN_PS_BT_ACL_SLAVE_MAX_BR] = 50,
70 /* Note: with UPSD, this should be 4 */
71 [CONF_SG_WLAN_PS_MAX_BT_ACL_SLAVE_BR] = 8,
72 [CONF_SG_WLAN_PS_BT_ACL_MASTER_MIN_EDR] = 7,
73 [CONF_SG_WLAN_PS_BT_ACL_MASTER_MAX_EDR] = 25,
74 [CONF_SG_WLAN_PS_MAX_BT_ACL_MASTER_EDR] = 20,
75 /* Note: with UPDS, this should be 15 */
76 [CONF_SG_WLAN_PS_BT_ACL_SLAVE_MIN_EDR] = 8,
77 /* Note: with UPDS, this should be 50 */
78 [CONF_SG_WLAN_PS_BT_ACL_SLAVE_MAX_EDR] = 40,
79 /* Note: with UPDS, this should be 10 */
80 [CONF_SG_WLAN_PS_MAX_BT_ACL_SLAVE_EDR] = 20,
81 [CONF_SG_RXT] = 1200,
82 [CONF_SG_TXT] = 1000,
83 [CONF_SG_ADAPTIVE_RXT_TXT] = 1,
84 [CONF_SG_PS_POLL_TIMEOUT] = 10,
85 [CONF_SG_UPSD_TIMEOUT] = 10,
86 [CONF_SG_WLAN_ACTIVE_BT_ACL_MASTER_MIN_EDR] = 7,
87 [CONF_SG_WLAN_ACTIVE_BT_ACL_MASTER_MAX_EDR] = 15,
88 [CONF_SG_WLAN_ACTIVE_MAX_BT_ACL_MASTER_EDR] = 15,
89 [CONF_SG_WLAN_ACTIVE_BT_ACL_SLAVE_MIN_EDR] = 8,
90 [CONF_SG_WLAN_ACTIVE_BT_ACL_SLAVE_MAX_EDR] = 20,
91 [CONF_SG_WLAN_ACTIVE_MAX_BT_ACL_SLAVE_EDR] = 15,
92 [CONF_SG_WLAN_ACTIVE_BT_ACL_MIN_BR] = 20,
93 [CONF_SG_WLAN_ACTIVE_BT_ACL_MAX_BR] = 50,
94 [CONF_SG_WLAN_ACTIVE_MAX_BT_ACL_BR] = 10,
95 [CONF_SG_PASSIVE_SCAN_DURATION_FACTOR_HV3] = 200,
96 [CONF_SG_PASSIVE_SCAN_DURATION_FACTOR_A2DP] = 800,
97 [CONF_SG_PASSIVE_SCAN_A2DP_BT_TIME] = 75,
98 [CONF_SG_PASSIVE_SCAN_A2DP_WLAN_TIME] = 15,
99 [CONF_SG_HV3_MAX_SERVED] = 6,
100 [CONF_SG_DHCP_TIME] = 5000,
101 [CONF_SG_ACTIVE_SCAN_DURATION_FACTOR_A2DP] = 100,
102 },
103 .state = CONF_SG_PROTECTIVE,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300104 },
105 .rx = {
106 .rx_msdu_life_time = 512000,
107 .packet_detection_threshold = 0,
108 .ps_poll_timeout = 15,
109 .upsd_timeout = 15,
110 .rts_threshold = 2347,
Luciano Coelho3ed8f2c2009-12-11 15:40:54 +0200111 .rx_cca_threshold = 0,
112 .irq_blk_threshold = 0xFFFF,
113 .irq_pkt_threshold = 0,
114 .irq_timeout = 600,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300115 .queue_type = CONF_RX_QUEUE_TYPE_LOW_PRIORITY,
116 },
117 .tx = {
118 .tx_energy_detection = 0,
119 .rc_conf = {
Juuso Oikarinenebba60c2010-04-01 11:38:20 +0300120 .enabled_rates = 0,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300121 .short_retry_limit = 10,
122 .long_retry_limit = 10,
123 .aflags = 0
124 },
125 .ac_conf_count = 4,
126 .ac_conf = {
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200127 [CONF_TX_AC_BE] = {
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300128 .ac = CONF_TX_AC_BE,
129 .cw_min = 15,
130 .cw_max = 63,
131 .aifsn = 3,
132 .tx_op_limit = 0,
133 },
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200134 [CONF_TX_AC_BK] = {
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300135 .ac = CONF_TX_AC_BK,
136 .cw_min = 15,
137 .cw_max = 63,
138 .aifsn = 7,
139 .tx_op_limit = 0,
140 },
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200141 [CONF_TX_AC_VI] = {
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300142 .ac = CONF_TX_AC_VI,
143 .cw_min = 15,
144 .cw_max = 63,
145 .aifsn = CONF_TX_AIFS_PIFS,
146 .tx_op_limit = 3008,
147 },
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200148 [CONF_TX_AC_VO] = {
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300149 .ac = CONF_TX_AC_VO,
150 .cw_min = 15,
151 .cw_max = 63,
152 .aifsn = CONF_TX_AIFS_PIFS,
153 .tx_op_limit = 1504,
154 },
155 },
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200156 .tid_conf_count = 4,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300157 .tid_conf = {
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200158 [CONF_TX_AC_BE] = {
159 .queue_id = CONF_TX_AC_BE,
160 .channel_type = CONF_CHANNEL_TYPE_EDCF,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300161 .tsid = CONF_TX_AC_BE,
162 .ps_scheme = CONF_PS_SCHEME_LEGACY,
163 .ack_policy = CONF_ACK_POLICY_LEGACY,
164 .apsd_conf = {0, 0},
165 },
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200166 [CONF_TX_AC_BK] = {
167 .queue_id = CONF_TX_AC_BK,
168 .channel_type = CONF_CHANNEL_TYPE_EDCF,
169 .tsid = CONF_TX_AC_BK,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300170 .ps_scheme = CONF_PS_SCHEME_LEGACY,
171 .ack_policy = CONF_ACK_POLICY_LEGACY,
172 .apsd_conf = {0, 0},
173 },
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200174 [CONF_TX_AC_VI] = {
175 .queue_id = CONF_TX_AC_VI,
176 .channel_type = CONF_CHANNEL_TYPE_EDCF,
177 .tsid = CONF_TX_AC_VI,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300178 .ps_scheme = CONF_PS_SCHEME_LEGACY,
179 .ack_policy = CONF_ACK_POLICY_LEGACY,
180 .apsd_conf = {0, 0},
181 },
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200182 [CONF_TX_AC_VO] = {
183 .queue_id = CONF_TX_AC_VO,
184 .channel_type = CONF_CHANNEL_TYPE_EDCF,
185 .tsid = CONF_TX_AC_VO,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300186 .ps_scheme = CONF_PS_SCHEME_LEGACY,
187 .ack_policy = CONF_ACK_POLICY_LEGACY,
188 .apsd_conf = {0, 0},
189 },
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300190 },
191 .frag_threshold = IEEE80211_MAX_FRAG_THRESHOLD,
Luciano Coelho3ed8f2c2009-12-11 15:40:54 +0200192 .tx_compl_timeout = 700,
Juuso Oikarinenebba60c2010-04-01 11:38:20 +0300193 .tx_compl_threshold = 4,
194 .basic_rate = CONF_HW_BIT_RATE_1MBPS,
195 .basic_rate_5 = CONF_HW_BIT_RATE_6MBPS,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300196 },
197 .conn = {
198 .wake_up_event = CONF_WAKE_UP_EVENT_DTIM,
Juuso Oikarinen50c500a2010-04-01 11:38:22 +0300199 .listen_interval = 1,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300200 .bcn_filt_mode = CONF_BCN_FILT_MODE_ENABLED,
201 .bcn_filt_ie_count = 1,
202 .bcn_filt_ie = {
203 [0] = {
204 .ie = WLAN_EID_CHANNEL_SWITCH,
205 .rule = CONF_BCN_RULE_PASS_ON_APPEARANCE,
206 }
207 },
Luciano Coelho3ed8f2c2009-12-11 15:40:54 +0200208 .synch_fail_thold = 10,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300209 .bss_lose_timeout = 100,
210 .beacon_rx_timeout = 10000,
211 .broadcast_timeout = 20000,
212 .rx_broadcast_in_ps = 1,
Juuso Oikarinen90494a92010-07-08 17:50:00 +0300213 .ps_poll_threshold = 10,
214 .ps_poll_recovery_period = 700,
Juuso Oikarinen11f70f92009-10-13 12:47:46 +0300215 .bet_enable = CONF_BET_MODE_ENABLE,
Juuso Oikarinen84502562009-11-23 23:22:12 +0200216 .bet_max_consecutive = 10,
Juuso Oikarinen8eab7b42010-09-24 03:10:11 +0200217 .psm_entry_retries = 5,
218 .psm_entry_nullfunc_retries = 3,
219 .psm_entry_hangover_period = 1,
Juuso Oikarinen50c500a2010-04-01 11:38:22 +0300220 .keep_alive_interval = 55000,
221 .max_listen_interval = 20,
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300222 },
Luciano Coelho6e92b412009-12-11 15:40:50 +0200223 .itrim = {
224 .enable = false,
225 .timeout = 50000,
Juuso Oikarinen38ad2d82009-12-11 15:41:08 +0200226 },
227 .pm_config = {
228 .host_clk_settling_time = 5000,
229 .host_fast_wakeup_support = false
Juuso Oikarinen00236aed2010-04-09 11:07:30 +0300230 },
231 .roam_trigger = {
Juuso Oikarinen00236aed2010-04-09 11:07:30 +0300232 .trigger_pacing = 1,
233 .avg_weight_rssi_beacon = 20,
234 .avg_weight_rssi_data = 10,
235 .avg_weight_snr_beacon = 20,
236 .avg_weight_snr_data = 10
Juuso Oikarinenbea39d62010-09-21 08:14:31 +0200237 },
238 .scan = {
239 .min_dwell_time_active = 7500,
240 .max_dwell_time_active = 30000,
241 .min_dwell_time_passive = 30000,
242 .max_dwell_time_passive = 60000,
243 .num_probe_reqs = 2,
244 },
Juuso Oikarinen644a4862010-10-05 13:11:56 +0200245 .rf = {
246 .tx_per_channel_power_compensation_2 = {
247 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
248 },
249 .tx_per_channel_power_compensation_5 = {
250 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
251 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
252 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
253 },
254 },
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300255};
256
Juuso Oikarinen52b0e7a2010-09-21 06:23:31 +0200257static void __wl1271_op_remove_interface(struct wl1271 *wl);
258
259
Juuso Oikarinena1dd8182010-03-18 12:26:31 +0200260static void wl1271_device_release(struct device *dev)
261{
262
263}
264
265static struct platform_device wl1271_device = {
266 .name = "wl1271",
267 .id = -1,
268
269 /* device model insists to have a release function */
270 .dev = {
271 .release = wl1271_device_release,
272 },
273};
274
Juuso Oikarinen01c09162009-10-13 12:47:55 +0300275static LIST_HEAD(wl_list);
276
Juuso Oikarinenc2c192a2010-07-27 03:30:09 +0300277static int wl1271_dev_notify(struct notifier_block *me, unsigned long what,
278 void *arg)
279{
280 struct net_device *dev = arg;
281 struct wireless_dev *wdev;
282 struct wiphy *wiphy;
283 struct ieee80211_hw *hw;
284 struct wl1271 *wl;
285 struct wl1271 *wl_temp;
286 int ret = 0;
287
288 /* Check that this notification is for us. */
289 if (what != NETDEV_CHANGE)
290 return NOTIFY_DONE;
291
292 wdev = dev->ieee80211_ptr;
293 if (wdev == NULL)
294 return NOTIFY_DONE;
295
296 wiphy = wdev->wiphy;
297 if (wiphy == NULL)
298 return NOTIFY_DONE;
299
300 hw = wiphy_priv(wiphy);
301 if (hw == NULL)
302 return NOTIFY_DONE;
303
304 wl_temp = hw->priv;
305 list_for_each_entry(wl, &wl_list, list) {
306 if (wl == wl_temp)
307 break;
308 }
309 if (wl != wl_temp)
310 return NOTIFY_DONE;
311
312 mutex_lock(&wl->mutex);
313
314 if (wl->state == WL1271_STATE_OFF)
315 goto out;
316
317 if (!test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
318 goto out;
319
320 ret = wl1271_ps_elp_wakeup(wl, false);
321 if (ret < 0)
322 goto out;
323
324 if ((dev->operstate == IF_OPER_UP) &&
325 !test_and_set_bit(WL1271_FLAG_STA_STATE_SENT, &wl->flags)) {
326 wl1271_cmd_set_sta_state(wl);
327 wl1271_info("Association completed.");
328 }
329
330 wl1271_ps_elp_sleep(wl);
331
332out:
333 mutex_unlock(&wl->mutex);
334
335 return NOTIFY_OK;
336}
337
Juuso Oikarinenb7417d92010-11-10 11:27:19 +0100338static int wl1271_reg_notify(struct wiphy *wiphy,
Luciano Coelho573c67c2010-11-26 13:44:59 +0200339 struct regulatory_request *request)
340{
Juuso Oikarinenb7417d92010-11-10 11:27:19 +0100341 struct ieee80211_supported_band *band;
342 struct ieee80211_channel *ch;
343 int i;
344
345 band = wiphy->bands[IEEE80211_BAND_5GHZ];
346 for (i = 0; i < band->n_channels; i++) {
347 ch = &band->channels[i];
348 if (ch->flags & IEEE80211_CHAN_DISABLED)
349 continue;
350
351 if (ch->flags & IEEE80211_CHAN_RADAR)
352 ch->flags |= IEEE80211_CHAN_NO_IBSS |
353 IEEE80211_CHAN_PASSIVE_SCAN;
354
355 }
356
357 return 0;
358}
359
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +0300360static void wl1271_conf_init(struct wl1271 *wl)
361{
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +0300362
363 /*
364 * This function applies the default configuration to the driver. This
365 * function is invoked upon driver load (spi probe.)
366 *
367 * The configuration is stored in a run-time structure in order to
368 * facilitate for run-time adjustment of any of the parameters. Making
369 * changes to the configuration structure will apply the new values on
370 * the next interface up (wl1271_op_start.)
371 */
372
373 /* apply driver default configuration */
Juuso Oikarinen8a080482009-10-13 12:47:44 +0300374 memcpy(&wl->conf, &default_conf, sizeof(default_conf));
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +0300375}
376
377
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300378static int wl1271_plt_init(struct wl1271 *wl)
379{
Luciano Coelho12419cc2010-02-18 13:25:44 +0200380 struct conf_tx_ac_category *conf_ac;
381 struct conf_tx_tid *conf_tid;
382 int ret, i;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300383
Luciano Coelho98b5dd52009-11-23 23:22:17 +0200384 ret = wl1271_cmd_general_parms(wl);
Luciano Coelho4a904062009-11-23 23:22:18 +0200385 if (ret < 0)
Luciano Coelhocc7defa2009-11-23 23:22:16 +0200386 return ret;
387
Luciano Coelho98b5dd52009-11-23 23:22:17 +0200388 ret = wl1271_cmd_radio_parms(wl);
Luciano Coelho4a904062009-11-23 23:22:18 +0200389 if (ret < 0)
Luciano Coelhocc7defa2009-11-23 23:22:16 +0200390 return ret;
391
Juuso Oikarinen644a4862010-10-05 13:11:56 +0200392 ret = wl1271_cmd_ext_radio_parms(wl);
393 if (ret < 0)
394 return ret;
395
Luciano Coelho12419cc2010-02-18 13:25:44 +0200396 ret = wl1271_init_templates_config(wl);
397 if (ret < 0)
398 return ret;
399
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300400 ret = wl1271_acx_init_mem_config(wl);
401 if (ret < 0)
402 return ret;
403
Luciano Coelho12419cc2010-02-18 13:25:44 +0200404 /* PHY layer config */
405 ret = wl1271_init_phy_config(wl);
406 if (ret < 0)
407 goto out_free_memmap;
408
409 ret = wl1271_acx_dco_itrim_params(wl);
410 if (ret < 0)
411 goto out_free_memmap;
412
413 /* Initialize connection monitoring thresholds */
Juuso Oikarinen6ccbb922010-03-26 12:53:23 +0200414 ret = wl1271_acx_conn_monit_params(wl, false);
Luciano Coelho12419cc2010-02-18 13:25:44 +0200415 if (ret < 0)
416 goto out_free_memmap;
417
418 /* Bluetooth WLAN coexistence */
419 ret = wl1271_init_pta(wl);
420 if (ret < 0)
421 goto out_free_memmap;
422
423 /* Energy detection */
424 ret = wl1271_init_energy_detection(wl);
425 if (ret < 0)
426 goto out_free_memmap;
427
428 /* Default fragmentation threshold */
Arik Nemtsov68d069c2010-11-08 10:51:07 +0100429 ret = wl1271_acx_frag_threshold(wl, wl->conf.tx.frag_threshold);
Luciano Coelho12419cc2010-02-18 13:25:44 +0200430 if (ret < 0)
431 goto out_free_memmap;
432
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200433 /* Default TID/AC configuration */
434 BUG_ON(wl->conf.tx.tid_conf_count != wl->conf.tx.ac_conf_count);
Luciano Coelho12419cc2010-02-18 13:25:44 +0200435 for (i = 0; i < wl->conf.tx.tid_conf_count; i++) {
Juuso Oikarinen9987a9d2010-09-01 11:31:12 +0200436 conf_ac = &wl->conf.tx.ac_conf[i];
437 ret = wl1271_acx_ac_cfg(wl, conf_ac->ac, conf_ac->cw_min,
438 conf_ac->cw_max, conf_ac->aifsn,
439 conf_ac->tx_op_limit);
440 if (ret < 0)
441 goto out_free_memmap;
442
Luciano Coelho12419cc2010-02-18 13:25:44 +0200443 conf_tid = &wl->conf.tx.tid_conf[i];
444 ret = wl1271_acx_tid_cfg(wl, conf_tid->queue_id,
445 conf_tid->channel_type,
446 conf_tid->tsid,
447 conf_tid->ps_scheme,
448 conf_tid->ack_policy,
449 conf_tid->apsd_conf[0],
450 conf_tid->apsd_conf[1]);
451 if (ret < 0)
452 goto out_free_memmap;
453 }
454
Luciano Coelho12419cc2010-02-18 13:25:44 +0200455 /* Enable data path */
Luciano Coelho94210892009-12-11 15:40:55 +0200456 ret = wl1271_cmd_data_path(wl, 1);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300457 if (ret < 0)
Luciano Coelho12419cc2010-02-18 13:25:44 +0200458 goto out_free_memmap;
459
460 /* Configure for CAM power saving (ie. always active) */
461 ret = wl1271_acx_sleep_auth(wl, WL1271_PSM_CAM);
462 if (ret < 0)
463 goto out_free_memmap;
464
465 /* configure PM */
466 ret = wl1271_acx_pm_config(wl);
467 if (ret < 0)
468 goto out_free_memmap;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300469
470 return 0;
Luciano Coelho12419cc2010-02-18 13:25:44 +0200471
472 out_free_memmap:
473 kfree(wl->target_mem_map);
474 wl->target_mem_map = NULL;
475
476 return ret;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300477}
478
Juuso Oikarinenc15f63b2009-10-12 15:08:50 +0300479static void wl1271_fw_status(struct wl1271 *wl,
480 struct wl1271_fw_status *status)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300481{
Juuso Oikarinenac5e1e32010-02-22 08:38:38 +0200482 struct timespec ts;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300483 u32 total = 0;
484 int i;
485
Teemu Paasikivi09a9c2b2010-02-22 08:38:28 +0200486 wl1271_raw_read(wl, FW_STATUS_ADDR, status, sizeof(*status), false);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300487
488 wl1271_debug(DEBUG_IRQ, "intr: 0x%x (fw_rx_counter = %d, "
489 "drv_rx_counter = %d, tx_results_counter = %d)",
490 status->intr,
491 status->fw_rx_counter,
492 status->drv_rx_counter,
493 status->tx_results_counter);
494
495 /* update number of available TX blocks */
496 for (i = 0; i < NUM_TX_QUEUES; i++) {
Luciano Coelhod0f63b22009-10-15 10:33:29 +0300497 u32 cnt = le32_to_cpu(status->tx_released_blks[i]) -
498 wl->tx_blocks_freed[i];
499
500 wl->tx_blocks_freed[i] =
501 le32_to_cpu(status->tx_released_blks[i]);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300502 wl->tx_blocks_available += cnt;
503 total += cnt;
504 }
505
Ido Yariva5225502010-10-12 14:49:10 +0200506 /* if more blocks are available now, tx work can be scheduled */
507 if (total)
508 clear_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300509
510 /* update the host-chipset time offset */
Juuso Oikarinenac5e1e32010-02-22 08:38:38 +0200511 getnstimeofday(&ts);
512 wl->time_offset = (timespec_to_ns(&ts) >> 10) -
513 (s64)le32_to_cpu(status->fw_localtime);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300514}
515
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200516#define WL1271_IRQ_MAX_LOOPS 10
517
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300518static void wl1271_irq_work(struct work_struct *work)
519{
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300520 int ret;
Juuso Oikarinenc15f63b2009-10-12 15:08:50 +0300521 u32 intr;
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200522 int loopcount = WL1271_IRQ_MAX_LOOPS;
523 unsigned long flags;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300524 struct wl1271 *wl =
525 container_of(work, struct wl1271, irq_work);
526
527 mutex_lock(&wl->mutex);
528
529 wl1271_debug(DEBUG_IRQ, "IRQ work");
530
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200531 if (unlikely(wl->state == WL1271_STATE_OFF))
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300532 goto out;
533
534 ret = wl1271_ps_elp_wakeup(wl, true);
535 if (ret < 0)
536 goto out;
537
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200538 spin_lock_irqsave(&wl->wl_lock, flags);
539 while (test_bit(WL1271_FLAG_IRQ_PENDING, &wl->flags) && loopcount) {
540 clear_bit(WL1271_FLAG_IRQ_PENDING, &wl->flags);
541 spin_unlock_irqrestore(&wl->wl_lock, flags);
542 loopcount--;
543
544 wl1271_fw_status(wl, wl->fw_status);
545 intr = le32_to_cpu(wl->fw_status->intr);
546 if (!intr) {
547 wl1271_debug(DEBUG_IRQ, "Zero interrupt received.");
Dan Carpentercdd08642010-05-08 18:25:17 +0200548 spin_lock_irqsave(&wl->wl_lock, flags);
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200549 continue;
550 }
551
552 intr &= WL1271_INTR_MASK;
553
Eliad Pellerccc83b02010-10-27 14:09:57 +0200554 if (unlikely(intr & WL1271_ACX_INTR_WATCHDOG)) {
555 wl1271_error("watchdog interrupt received! "
556 "starting recovery.");
557 ieee80211_queue_work(wl->hw, &wl->recovery_work);
558
559 /* restarting the chip. ignore any other interrupt. */
560 goto out;
561 }
562
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200563 if (intr & WL1271_ACX_INTR_DATA) {
564 wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_DATA");
565
566 /* check for tx results */
567 if (wl->fw_status->tx_results_counter !=
568 (wl->tx_results_count & 0xff))
569 wl1271_tx_complete(wl);
570
Ido Yariva5225502010-10-12 14:49:10 +0200571 /* Check if any tx blocks were freed */
572 if (!test_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags) &&
573 !skb_queue_empty(&wl->tx_queue)) {
574 /*
575 * In order to avoid starvation of the TX path,
576 * call the work function directly.
577 */
578 wl1271_tx_work_locked(wl);
579 }
580
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200581 wl1271_rx(wl, wl->fw_status);
582 }
583
584 if (intr & WL1271_ACX_INTR_EVENT_A) {
585 wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_EVENT_A");
586 wl1271_event_handle(wl, 0);
587 }
588
589 if (intr & WL1271_ACX_INTR_EVENT_B) {
590 wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_EVENT_B");
591 wl1271_event_handle(wl, 1);
592 }
593
594 if (intr & WL1271_ACX_INTR_INIT_COMPLETE)
595 wl1271_debug(DEBUG_IRQ,
596 "WL1271_ACX_INTR_INIT_COMPLETE");
597
598 if (intr & WL1271_ACX_INTR_HW_AVAILABLE)
599 wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_HW_AVAILABLE");
600
601 spin_lock_irqsave(&wl->wl_lock, flags);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300602 }
603
Juuso Oikarinen1e73eb62010-02-22 08:38:37 +0200604 if (test_bit(WL1271_FLAG_IRQ_PENDING, &wl->flags))
605 ieee80211_queue_work(wl->hw, &wl->irq_work);
606 else
607 clear_bit(WL1271_FLAG_IRQ_RUNNING, &wl->flags);
608 spin_unlock_irqrestore(&wl->wl_lock, flags);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300609
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300610 wl1271_ps_elp_sleep(wl);
611
612out:
613 mutex_unlock(&wl->mutex);
614}
615
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300616static int wl1271_fetch_firmware(struct wl1271 *wl)
617{
618 const struct firmware *fw;
619 int ret;
620
Teemu Paasikivi8197b712010-02-22 08:38:23 +0200621 ret = request_firmware(&fw, WL1271_FW_NAME, wl1271_wl_to_dev(wl));
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300622
623 if (ret < 0) {
624 wl1271_error("could not get firmware: %d", ret);
625 return ret;
626 }
627
628 if (fw->size % 4) {
629 wl1271_error("firmware size is not multiple of 32 bits: %zu",
630 fw->size);
631 ret = -EILSEQ;
632 goto out;
633 }
634
635 wl->fw_len = fw->size;
Juuso Oikarinen1fba4972009-10-08 21:56:32 +0300636 wl->fw = vmalloc(wl->fw_len);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300637
638 if (!wl->fw) {
639 wl1271_error("could not allocate memory for the firmware");
640 ret = -ENOMEM;
641 goto out;
642 }
643
644 memcpy(wl->fw, fw->data, wl->fw_len);
645
646 ret = 0;
647
648out:
649 release_firmware(fw);
650
651 return ret;
652}
653
654static int wl1271_fetch_nvs(struct wl1271 *wl)
655{
656 const struct firmware *fw;
657 int ret;
658
Teemu Paasikivi8197b712010-02-22 08:38:23 +0200659 ret = request_firmware(&fw, WL1271_NVS_NAME, wl1271_wl_to_dev(wl));
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300660
661 if (ret < 0) {
662 wl1271_error("could not get nvs file: %d", ret);
663 return ret;
664 }
665
Julia Lawall929ebd32010-05-15 23:16:39 +0200666 wl->nvs = kmemdup(fw->data, sizeof(struct wl1271_nvs_file), GFP_KERNEL);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300667
668 if (!wl->nvs) {
669 wl1271_error("could not allocate memory for the nvs file");
670 ret = -ENOMEM;
671 goto out;
672 }
673
Juuso Oikarinen02fabb02010-08-19 04:41:15 +0200674 wl->nvs_len = fw->size;
675
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300676out:
677 release_firmware(fw);
678
679 return ret;
680}
681
Juuso Oikarinen52b0e7a2010-09-21 06:23:31 +0200682static void wl1271_recovery_work(struct work_struct *work)
683{
684 struct wl1271 *wl =
685 container_of(work, struct wl1271, recovery_work);
686
687 mutex_lock(&wl->mutex);
688
689 if (wl->state != WL1271_STATE_ON)
690 goto out;
691
692 wl1271_info("Hardware recovery in progress.");
693
Juuso Oikarinend25611d2010-09-30 10:43:27 +0200694 if (test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
695 ieee80211_connection_loss(wl->vif);
696
Juuso Oikarinen52b0e7a2010-09-21 06:23:31 +0200697 /* reboot the chipset */
698 __wl1271_op_remove_interface(wl);
699 ieee80211_restart_hw(wl->hw);
700
701out:
702 mutex_unlock(&wl->mutex);
703}
704
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300705static void wl1271_fw_wakeup(struct wl1271 *wl)
706{
707 u32 elp_reg;
708
709 elp_reg = ELPCTRL_WAKE_UP;
Juuso Oikarinen74621412009-10-12 15:08:54 +0300710 wl1271_raw_write32(wl, HW_ACCESS_ELP_CTRL_REG_ADDR, elp_reg);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300711}
712
713static int wl1271_setup(struct wl1271 *wl)
714{
715 wl->fw_status = kmalloc(sizeof(*wl->fw_status), GFP_KERNEL);
716 if (!wl->fw_status)
717 return -ENOMEM;
718
719 wl->tx_res_if = kmalloc(sizeof(*wl->tx_res_if), GFP_KERNEL);
720 if (!wl->tx_res_if) {
721 kfree(wl->fw_status);
722 return -ENOMEM;
723 }
724
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300725 return 0;
726}
727
728static int wl1271_chip_wakeup(struct wl1271 *wl)
729{
Juuso Oikarinen451de972009-10-12 15:08:46 +0300730 struct wl1271_partition_set partition;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300731 int ret = 0;
732
Juuso Oikarinen01ac17e2009-12-11 15:41:02 +0200733 msleep(WL1271_PRE_POWER_ON_SLEEP);
Ohad Ben-Cohen2cc78ff2010-09-16 01:22:04 +0200734 ret = wl1271_power_on(wl);
735 if (ret < 0)
736 goto out;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300737 msleep(WL1271_POWER_ON_SLEEP);
Teemu Paasikivi9b280722010-02-18 13:25:56 +0200738 wl1271_io_reset(wl);
739 wl1271_io_init(wl);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300740
741 /* We don't need a real memory partition here, because we only want
742 * to use the registers at this point. */
Juuso Oikarinen451de972009-10-12 15:08:46 +0300743 memset(&partition, 0, sizeof(partition));
744 partition.reg.start = REGISTERS_BASE;
745 partition.reg.size = REGISTERS_DOWN_SIZE;
746 wl1271_set_partition(wl, &partition);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300747
748 /* ELP module wake up */
749 wl1271_fw_wakeup(wl);
750
751 /* whal_FwCtrl_BootSm() */
752
753 /* 0. read chip id from CHIP_ID */
Teemu Paasikivi7b048c52010-02-18 13:25:55 +0200754 wl->chip.id = wl1271_read32(wl, CHIP_ID_B);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300755
756 /* 1. check if chip id is valid */
757
758 switch (wl->chip.id) {
759 case CHIP_ID_1271_PG10:
760 wl1271_warning("chip id 0x%x (1271 PG10) support is obsolete",
761 wl->chip.id);
762
763 ret = wl1271_setup(wl);
764 if (ret < 0)
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200765 goto out;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300766 break;
767 case CHIP_ID_1271_PG20:
768 wl1271_debug(DEBUG_BOOT, "chip id 0x%x (1271 PG20)",
769 wl->chip.id);
770
771 ret = wl1271_setup(wl);
772 if (ret < 0)
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200773 goto out;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300774 break;
775 default:
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200776 wl1271_warning("unsupported chip id: 0x%x", wl->chip.id);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300777 ret = -ENODEV;
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200778 goto out;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300779 }
780
781 if (wl->fw == NULL) {
782 ret = wl1271_fetch_firmware(wl);
783 if (ret < 0)
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200784 goto out;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300785 }
786
787 /* No NVS from netlink, try to get it from the filesystem */
788 if (wl->nvs == NULL) {
789 ret = wl1271_fetch_nvs(wl);
790 if (ret < 0)
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200791 goto out;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300792 }
793
794out:
795 return ret;
796}
797
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300798int wl1271_plt_start(struct wl1271 *wl)
799{
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200800 int retries = WL1271_BOOT_RETRIES;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300801 int ret;
802
803 mutex_lock(&wl->mutex);
804
805 wl1271_notice("power up");
806
807 if (wl->state != WL1271_STATE_OFF) {
808 wl1271_error("cannot go into PLT state because not "
809 "in off state: %d", wl->state);
810 ret = -EBUSY;
811 goto out;
812 }
813
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200814 while (retries) {
815 retries--;
816 ret = wl1271_chip_wakeup(wl);
817 if (ret < 0)
818 goto power_off;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300819
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200820 ret = wl1271_boot(wl);
821 if (ret < 0)
822 goto power_off;
823
824 ret = wl1271_plt_init(wl);
825 if (ret < 0)
826 goto irq_disable;
827
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200828 wl->state = WL1271_STATE_PLT;
829 wl1271_notice("firmware booted in PLT mode (%s)",
830 wl->chip.fw_ver);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300831 goto out;
832
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200833irq_disable:
834 wl1271_disable_interrupts(wl);
835 mutex_unlock(&wl->mutex);
836 /* Unlocking the mutex in the middle of handling is
837 inherently unsafe. In this case we deem it safe to do,
838 because we need to let any possibly pending IRQ out of
839 the system (and while we are WL1271_STATE_OFF the IRQ
840 work function will not do anything.) Also, any other
841 possible concurrent operations will fail due to the
842 current state, hence the wl1271 struct should be safe. */
843 cancel_work_sync(&wl->irq_work);
844 mutex_lock(&wl->mutex);
845power_off:
846 wl1271_power_off(wl);
847 }
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300848
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200849 wl1271_error("firmware boot in PLT mode failed despite %d retries",
850 WL1271_BOOT_RETRIES);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300851out:
852 mutex_unlock(&wl->mutex);
853
854 return ret;
855}
856
857int wl1271_plt_stop(struct wl1271 *wl)
858{
859 int ret = 0;
860
861 mutex_lock(&wl->mutex);
862
863 wl1271_notice("power down");
864
865 if (wl->state != WL1271_STATE_PLT) {
866 wl1271_error("cannot power down because not in PLT "
867 "state: %d", wl->state);
868 ret = -EBUSY;
869 goto out;
870 }
871
872 wl1271_disable_interrupts(wl);
873 wl1271_power_off(wl);
874
875 wl->state = WL1271_STATE_OFF;
Luciano Coelhobd5ea182009-10-13 12:47:58 +0300876 wl->rx_counter = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300877
878out:
879 mutex_unlock(&wl->mutex);
880
Juuso Oikarinen8c7f4f32010-09-21 06:23:29 +0200881 cancel_work_sync(&wl->irq_work);
Juuso Oikarinen52b0e7a2010-09-21 06:23:31 +0200882 cancel_work_sync(&wl->recovery_work);
Juuso Oikarinen8c7f4f32010-09-21 06:23:29 +0200883
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300884 return ret;
885}
886
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300887static int wl1271_op_tx(struct ieee80211_hw *hw, struct sk_buff *skb)
888{
889 struct wl1271 *wl = hw->priv;
Juuso Oikarinen830fb672009-12-11 15:41:06 +0200890 struct ieee80211_conf *conf = &hw->conf;
891 struct ieee80211_tx_info *txinfo = IEEE80211_SKB_CB(skb);
892 struct ieee80211_sta *sta = txinfo->control.sta;
893 unsigned long flags;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300894
Shahar Levi18357852010-10-13 16:09:41 +0200895 /*
896 * peek into the rates configured in the STA entry.
897 * The rates set after connection stage, The first block only BG sets:
898 * the compare is for bit 0-16 of sta_rate_set. The second block add
899 * HT rates in case of HT supported.
900 */
Juuso Oikarinen830fb672009-12-11 15:41:06 +0200901 spin_lock_irqsave(&wl->wl_lock, flags);
Shahar Levi18357852010-10-13 16:09:41 +0200902 if (sta &&
903 (sta->supp_rates[conf->channel->band] !=
904 (wl->sta_rate_set & HW_BG_RATES_MASK))) {
Juuso Oikarinen830fb672009-12-11 15:41:06 +0200905 wl->sta_rate_set = sta->supp_rates[conf->channel->band];
906 set_bit(WL1271_FLAG_STA_RATES_CHANGED, &wl->flags);
907 }
Shahar Levi18357852010-10-13 16:09:41 +0200908
Shahar Levi00d20102010-11-08 11:20:10 +0000909#ifdef CONFIG_WL12XX_HT
Shahar Levi18357852010-10-13 16:09:41 +0200910 if (sta &&
911 sta->ht_cap.ht_supported &&
912 ((wl->sta_rate_set >> HW_HT_RATES_OFFSET) !=
913 sta->ht_cap.mcs.rx_mask[0])) {
914 /* Clean MCS bits before setting them */
915 wl->sta_rate_set &= HW_BG_RATES_MASK;
916 wl->sta_rate_set |=
917 (sta->ht_cap.mcs.rx_mask[0] << HW_HT_RATES_OFFSET);
918 set_bit(WL1271_FLAG_STA_RATES_CHANGED, &wl->flags);
919 }
920#endif
Juuso Oikarinen830fb672009-12-11 15:41:06 +0200921 spin_unlock_irqrestore(&wl->wl_lock, flags);
922
923 /* queue the packet */
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300924 skb_queue_tail(&wl->tx_queue, skb);
925
926 /*
927 * The chip specific setup must run before the first TX packet -
928 * before that, the tx_work will not be initialized!
929 */
930
Ido Yariva5225502010-10-12 14:49:10 +0200931 if (!test_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags))
932 ieee80211_queue_work(wl->hw, &wl->tx_work);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300933
934 /*
935 * The workqueue is slow to process the tx_queue and we need stop
936 * the queue here, otherwise the queue will get too long.
937 */
Juuso Oikarinen06f7bc72010-02-22 08:38:33 +0200938 if (skb_queue_len(&wl->tx_queue) >= WL1271_TX_QUEUE_HIGH_WATERMARK) {
939 wl1271_debug(DEBUG_TX, "op_tx: stopping queues");
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300940
Juuso Oikarinen06f7bc72010-02-22 08:38:33 +0200941 spin_lock_irqsave(&wl->wl_lock, flags);
942 ieee80211_stop_queues(wl->hw);
Juuso Oikarinen71449f82009-12-11 15:41:07 +0200943 set_bit(WL1271_FLAG_TX_QUEUE_STOPPED, &wl->flags);
Juuso Oikarinen06f7bc72010-02-22 08:38:33 +0200944 spin_unlock_irqrestore(&wl->wl_lock, flags);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300945 }
946
947 return NETDEV_TX_OK;
948}
949
Juuso Oikarinenc2c192a2010-07-27 03:30:09 +0300950static struct notifier_block wl1271_dev_notifier = {
951 .notifier_call = wl1271_dev_notify,
952};
953
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300954static int wl1271_op_start(struct ieee80211_hw *hw)
955{
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +0200956 wl1271_debug(DEBUG_MAC80211, "mac80211 start");
957
958 /*
959 * We have to delay the booting of the hardware because
960 * we need to know the local MAC address before downloading and
961 * initializing the firmware. The MAC address cannot be changed
962 * after boot, and without the proper MAC address, the firmware
963 * will not function properly.
964 *
965 * The MAC address is first known when the corresponding interface
966 * is added. That is where we will initialize the hardware.
967 */
968
969 return 0;
970}
971
972static void wl1271_op_stop(struct ieee80211_hw *hw)
973{
974 wl1271_debug(DEBUG_MAC80211, "mac80211 stop");
975}
976
977static int wl1271_op_add_interface(struct ieee80211_hw *hw,
978 struct ieee80211_vif *vif)
979{
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300980 struct wl1271 *wl = hw->priv;
John W. Linvilleac01e942010-07-28 17:09:41 -0400981 struct wiphy *wiphy = hw->wiphy;
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +0200982 int retries = WL1271_BOOT_RETRIES;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300983 int ret = 0;
Eliad Peller71125ab2010-10-28 21:46:43 +0200984 bool booted = false;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300985
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +0200986 wl1271_debug(DEBUG_MAC80211, "mac80211 add interface type %d mac %pM",
987 vif->type, vif->addr);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +0300988
989 mutex_lock(&wl->mutex);
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +0200990 if (wl->vif) {
Eliad Peller71125ab2010-10-28 21:46:43 +0200991 wl1271_debug(DEBUG_MAC80211,
992 "multiple vifs are not supported yet");
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +0200993 ret = -EBUSY;
994 goto out;
995 }
996
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +0200997 switch (vif->type) {
998 case NL80211_IFTYPE_STATION:
999 wl->bss_type = BSS_TYPE_STA_BSS;
Juuso Oikarinen5da11dc2010-03-26 12:53:24 +02001000 wl->set_bss_type = BSS_TYPE_STA_BSS;
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +02001001 break;
1002 case NL80211_IFTYPE_ADHOC:
1003 wl->bss_type = BSS_TYPE_IBSS;
Juuso Oikarinen5da11dc2010-03-26 12:53:24 +02001004 wl->set_bss_type = BSS_TYPE_STA_BSS;
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +02001005 break;
1006 default:
1007 ret = -EOPNOTSUPP;
1008 goto out;
1009 }
1010
1011 memcpy(wl->mac_addr, vif->addr, ETH_ALEN);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001012
1013 if (wl->state != WL1271_STATE_OFF) {
1014 wl1271_error("cannot start because not in off state: %d",
1015 wl->state);
1016 ret = -EBUSY;
1017 goto out;
1018 }
1019
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +02001020 while (retries) {
1021 retries--;
1022 ret = wl1271_chip_wakeup(wl);
1023 if (ret < 0)
1024 goto power_off;
1025
1026 ret = wl1271_boot(wl);
1027 if (ret < 0)
1028 goto power_off;
1029
1030 ret = wl1271_hw_init(wl);
1031 if (ret < 0)
1032 goto irq_disable;
1033
Eliad Peller71125ab2010-10-28 21:46:43 +02001034 booted = true;
1035 break;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001036
Juuso Oikarinen9ccd9212009-12-11 15:41:01 +02001037irq_disable:
1038 wl1271_disable_interrupts(wl);
1039 mutex_unlock(&wl->mutex);
1040 /* Unlocking the mutex in the middle of handling is
1041 inherently unsafe. In this case we deem it safe to do,
1042 because we need to let any possibly pending IRQ out of
1043 the system (and while we are WL1271_STATE_OFF the IRQ
1044 work function will not do anything.) Also, any other
1045 possible concurrent operations will fail due to the
1046 current state, hence the wl1271 struct should be safe. */
1047 cancel_work_sync(&wl->irq_work);
1048 mutex_lock(&wl->mutex);
1049power_off:
1050 wl1271_power_off(wl);
1051 }
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001052
Eliad Peller71125ab2010-10-28 21:46:43 +02001053 if (!booted) {
1054 wl1271_error("firmware boot failed despite %d retries",
1055 WL1271_BOOT_RETRIES);
1056 goto out;
1057 }
1058
1059 wl->vif = vif;
1060 wl->state = WL1271_STATE_ON;
1061 wl1271_info("firmware booted (%s)", wl->chip.fw_ver);
1062
1063 /* update hw/fw version info in wiphy struct */
1064 wiphy->hw_version = wl->chip.id;
1065 strncpy(wiphy->fw_version, wl->chip.fw_ver,
1066 sizeof(wiphy->fw_version));
1067
Luciano Coelhofb6a6812010-12-03 17:05:40 +02001068 /*
1069 * Now we know if 11a is supported (info from the NVS), so disable
1070 * 11a channels if not supported
1071 */
1072 if (!wl->enable_11a)
1073 wiphy->bands[IEEE80211_BAND_5GHZ]->n_channels = 0;
1074
1075 wl1271_debug(DEBUG_MAC80211, "11a is %ssupported",
1076 wl->enable_11a ? "" : "not ");
1077
Juuso Oikarineneb5b28d2009-10-13 12:47:45 +03001078out:
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001079 mutex_unlock(&wl->mutex);
1080
Juuso Oikarineneb887df2010-07-08 17:49:58 +03001081 if (!ret)
Juuso Oikarinen01c09162009-10-13 12:47:55 +03001082 list_add(&wl->list, &wl_list);
Juuso Oikarinen01c09162009-10-13 12:47:55 +03001083
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001084 return ret;
1085}
1086
Juuso Oikarinen52a2a372010-09-21 06:23:30 +02001087static void __wl1271_op_remove_interface(struct wl1271 *wl)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001088{
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001089 int i;
1090
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +02001091 wl1271_debug(DEBUG_MAC80211, "mac80211 remove interface");
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001092
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +02001093 wl1271_info("down");
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001094
Juuso Oikarinen01c09162009-10-13 12:47:55 +03001095 list_del(&wl->list);
1096
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001097 WARN_ON(wl->state != WL1271_STATE_ON);
1098
Juuso Oikarinen8d2ef7b2010-07-08 17:50:03 +03001099 /* enable dyn ps just in case (if left on due to fw crash etc) */
Juuso Oikarinen9a547bf2010-07-08 17:50:04 +03001100 if (wl->bss_type == BSS_TYPE_STA_BSS)
Juuso Oikarinenf532be62010-07-08 17:50:05 +03001101 ieee80211_enable_dyn_ps(wl->vif);
Juuso Oikarinen8d2ef7b2010-07-08 17:50:03 +03001102
Luciano Coelho08688d62010-07-08 17:50:07 +03001103 if (wl->scan.state != WL1271_SCAN_STATE_IDLE) {
Luciano Coelho08688d62010-07-08 17:50:07 +03001104 wl->scan.state = WL1271_SCAN_STATE_IDLE;
1105 kfree(wl->scan.scanned_ch);
1106 wl->scan.scanned_ch = NULL;
Juuso Oikarinenb739a422010-10-26 13:24:38 +02001107 wl->scan.req = NULL;
Juuso Oikarinen76a029f2010-07-29 04:54:45 +03001108 ieee80211_scan_completed(wl->hw, true);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001109 }
1110
1111 wl->state = WL1271_STATE_OFF;
1112
1113 wl1271_disable_interrupts(wl);
1114
1115 mutex_unlock(&wl->mutex);
1116
Juuso Oikarinen78abd322010-09-21 06:23:32 +02001117 cancel_delayed_work_sync(&wl->scan_complete_work);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001118 cancel_work_sync(&wl->irq_work);
1119 cancel_work_sync(&wl->tx_work);
Juuso Oikarinen90494a92010-07-08 17:50:00 +03001120 cancel_delayed_work_sync(&wl->pspoll_work);
Juuso Oikarinen8c7f4f32010-09-21 06:23:29 +02001121 cancel_delayed_work_sync(&wl->elp_work);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001122
1123 mutex_lock(&wl->mutex);
1124
1125 /* let's notify MAC80211 about the remaining pending TX frames */
Juuso Oikarinen781608c2010-05-24 11:18:17 +03001126 wl1271_tx_reset(wl);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001127 wl1271_power_off(wl);
1128
1129 memset(wl->bssid, 0, ETH_ALEN);
1130 memset(wl->ssid, 0, IW_ESSID_MAX_SIZE + 1);
1131 wl->ssid_len = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001132 wl->bss_type = MAX_BSS_TYPE;
Juuso Oikarinen5da11dc2010-03-26 12:53:24 +02001133 wl->set_bss_type = MAX_BSS_TYPE;
Juuso Oikarinen8a5a37a2009-10-08 21:56:24 +03001134 wl->band = IEEE80211_BAND_2GHZ;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001135
1136 wl->rx_counter = 0;
Juuso Oikarinen19ad0712009-11-02 20:22:11 +02001137 wl->psm_entry_retry = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001138 wl->power_level = WL1271_DEFAULT_POWER_LEVEL;
1139 wl->tx_blocks_available = 0;
1140 wl->tx_results_count = 0;
1141 wl->tx_packets_count = 0;
Juuso Oikarinenac4e4ce2009-10-08 21:56:19 +03001142 wl->tx_security_last_seq = 0;
Juuso Oikarinen04e36fc2010-02-22 08:38:40 +02001143 wl->tx_security_seq = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001144 wl->time_offset = 0;
1145 wl->session_counter = 0;
Juuso Oikarinen830fb672009-12-11 15:41:06 +02001146 wl->rate_set = CONF_TX_RATE_MASK_BASIC;
1147 wl->sta_rate_set = 0;
1148 wl->flags = 0;
Juuso Oikarinen1b72aec2010-03-18 12:26:39 +02001149 wl->vif = NULL;
Juuso Oikarinen14b228a2010-03-18 12:26:43 +02001150 wl->filters = 0;
Luciano Coelhod6e19d132009-10-12 15:08:43 +03001151
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001152 for (i = 0; i < NUM_TX_QUEUES; i++)
1153 wl->tx_blocks_freed[i] = 0;
1154
1155 wl1271_debugfs_reset(wl);
Juuso Oikarinenbd9dc492010-04-09 11:07:26 +03001156
1157 kfree(wl->fw_status);
1158 wl->fw_status = NULL;
1159 kfree(wl->tx_res_if);
1160 wl->tx_res_if = NULL;
1161 kfree(wl->target_mem_map);
1162 wl->target_mem_map = NULL;
Juuso Oikarinen52a2a372010-09-21 06:23:30 +02001163}
Juuso Oikarinenbd9dc492010-04-09 11:07:26 +03001164
Juuso Oikarinen52a2a372010-09-21 06:23:30 +02001165static void wl1271_op_remove_interface(struct ieee80211_hw *hw,
1166 struct ieee80211_vif *vif)
1167{
1168 struct wl1271 *wl = hw->priv;
1169
1170 mutex_lock(&wl->mutex);
Juuso Oikarinen67353292010-11-18 15:19:02 +02001171 /*
1172 * wl->vif can be null here if someone shuts down the interface
1173 * just when hardware recovery has been started.
1174 */
1175 if (wl->vif) {
1176 WARN_ON(wl->vif != vif);
1177 __wl1271_op_remove_interface(wl);
1178 }
Juuso Oikarinen52b0e7a2010-09-21 06:23:31 +02001179
Juuso Oikarinen67353292010-11-18 15:19:02 +02001180 mutex_unlock(&wl->mutex);
Juuso Oikarinen52b0e7a2010-09-21 06:23:31 +02001181 cancel_work_sync(&wl->recovery_work);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001182}
1183
Juuso Oikarinen14b228a2010-03-18 12:26:43 +02001184static void wl1271_configure_filters(struct wl1271 *wl, unsigned int filters)
1185{
1186 wl->rx_config = WL1271_DEFAULT_RX_CONFIG;
1187 wl->rx_filter = WL1271_DEFAULT_RX_FILTER;
1188
1189 /* combine requested filters with current filter config */
1190 filters = wl->filters | filters;
1191
1192 wl1271_debug(DEBUG_FILTERS, "RX filters set: ");
1193
1194 if (filters & FIF_PROMISC_IN_BSS) {
1195 wl1271_debug(DEBUG_FILTERS, " - FIF_PROMISC_IN_BSS");
1196 wl->rx_config &= ~CFG_UNI_FILTER_EN;
1197 wl->rx_config |= CFG_BSSID_FILTER_EN;
1198 }
1199 if (filters & FIF_BCN_PRBRESP_PROMISC) {
1200 wl1271_debug(DEBUG_FILTERS, " - FIF_BCN_PRBRESP_PROMISC");
1201 wl->rx_config &= ~CFG_BSSID_FILTER_EN;
1202 wl->rx_config &= ~CFG_SSID_FILTER_EN;
1203 }
1204 if (filters & FIF_OTHER_BSS) {
1205 wl1271_debug(DEBUG_FILTERS, " - FIF_OTHER_BSS");
1206 wl->rx_config &= ~CFG_BSSID_FILTER_EN;
1207 }
1208 if (filters & FIF_CONTROL) {
1209 wl1271_debug(DEBUG_FILTERS, " - FIF_CONTROL");
1210 wl->rx_filter |= CFG_RX_CTL_EN;
1211 }
1212 if (filters & FIF_FCSFAIL) {
1213 wl1271_debug(DEBUG_FILTERS, " - FIF_FCSFAIL");
1214 wl->rx_filter |= CFG_RX_FCS_ERROR;
1215 }
1216}
1217
Juuso Oikarinen82429d32010-04-28 09:50:01 +03001218static int wl1271_dummy_join(struct wl1271 *wl)
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001219{
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02001220 int ret = 0;
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001221 /* we need to use a dummy BSSID for now */
1222 static const u8 dummy_bssid[ETH_ALEN] = { 0x0b, 0xad, 0xde,
1223 0xad, 0xbe, 0xef };
1224
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001225 memcpy(wl->bssid, dummy_bssid, ETH_ALEN);
1226
Juuso Oikarinen14b228a2010-03-18 12:26:43 +02001227 /* pass through frames from all BSS */
1228 wl1271_configure_filters(wl, FIF_OTHER_BSS);
1229
Juuso Oikarinen5da11dc2010-03-26 12:53:24 +02001230 ret = wl1271_cmd_join(wl, wl->set_bss_type);
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001231 if (ret < 0)
1232 goto out;
1233
Juuso Oikarinen71449f82009-12-11 15:41:07 +02001234 set_bit(WL1271_FLAG_JOINED, &wl->flags);
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001235
1236out:
1237 return ret;
1238}
1239
Juuso Oikarinen69e54342010-05-07 11:39:00 +03001240static int wl1271_join(struct wl1271 *wl, bool set_assoc)
Juuso Oikarinen82429d32010-04-28 09:50:01 +03001241{
1242 int ret;
1243
Juuso Oikarinen69e54342010-05-07 11:39:00 +03001244 /*
1245 * One of the side effects of the JOIN command is that is clears
1246 * WPA/WPA2 keys from the chipset. Performing a JOIN while associated
1247 * to a WPA/WPA2 access point will therefore kill the data-path.
1248 * Currently there is no supported scenario for JOIN during
1249 * association - if it becomes a supported scenario, the WPA/WPA2 keys
1250 * must be handled somehow.
1251 *
1252 */
1253 if (test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
1254 wl1271_info("JOIN while associated.");
1255
1256 if (set_assoc)
1257 set_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags);
1258
Juuso Oikarinen82429d32010-04-28 09:50:01 +03001259 ret = wl1271_cmd_join(wl, wl->set_bss_type);
1260 if (ret < 0)
1261 goto out;
1262
1263 set_bit(WL1271_FLAG_JOINED, &wl->flags);
1264
1265 if (!test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
1266 goto out;
1267
1268 /*
1269 * The join command disable the keep-alive mode, shut down its process,
1270 * and also clear the template config, so we need to reset it all after
1271 * the join. The acx_aid starts the keep-alive process, and the order
1272 * of the commands below is relevant.
1273 */
1274 ret = wl1271_acx_keep_alive_mode(wl, true);
1275 if (ret < 0)
1276 goto out;
1277
1278 ret = wl1271_acx_aid(wl, wl->aid);
1279 if (ret < 0)
1280 goto out;
1281
1282 ret = wl1271_cmd_build_klv_null_data(wl);
1283 if (ret < 0)
1284 goto out;
1285
1286 ret = wl1271_acx_keep_alive_config(wl, CMD_TEMPL_KLV_IDX_NULL_DATA,
1287 ACX_KEEP_ALIVE_TPL_VALID);
1288 if (ret < 0)
1289 goto out;
1290
1291out:
1292 return ret;
1293}
1294
1295static int wl1271_unjoin(struct wl1271 *wl)
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001296{
1297 int ret;
1298
1299 /* to stop listening to a channel, we disconnect */
1300 ret = wl1271_cmd_disconnect(wl);
1301 if (ret < 0)
1302 goto out;
1303
Juuso Oikarinen71449f82009-12-11 15:41:07 +02001304 clear_bit(WL1271_FLAG_JOINED, &wl->flags);
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001305 memset(wl->bssid, 0, ETH_ALEN);
Juuso Oikarinen14b228a2010-03-18 12:26:43 +02001306
1307 /* stop filterting packets based on bssid */
1308 wl1271_configure_filters(wl, FIF_OTHER_BSS);
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001309
1310out:
1311 return ret;
1312}
1313
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03001314static void wl1271_set_band_rate(struct wl1271 *wl)
1315{
1316 if (wl->band == IEEE80211_BAND_2GHZ)
1317 wl->basic_rate_set = wl->conf.tx.basic_rate;
1318 else
1319 wl->basic_rate_set = wl->conf.tx.basic_rate_5;
1320}
1321
1322static u32 wl1271_min_rate_get(struct wl1271 *wl)
1323{
1324 int i;
1325 u32 rate = 0;
1326
1327 if (!wl->basic_rate_set) {
1328 WARN_ON(1);
1329 wl->basic_rate_set = wl->conf.tx.basic_rate;
1330 }
1331
1332 for (i = 0; !rate; i++) {
1333 if ((wl->basic_rate_set >> i) & 0x1)
1334 rate = 1 << i;
1335 }
1336
1337 return rate;
1338}
1339
Juuso Oikarinen0d58cbf2010-05-24 11:18:16 +03001340static int wl1271_handle_idle(struct wl1271 *wl, bool idle)
1341{
1342 int ret;
1343
1344 if (idle) {
1345 if (test_bit(WL1271_FLAG_JOINED, &wl->flags)) {
1346 ret = wl1271_unjoin(wl);
1347 if (ret < 0)
1348 goto out;
1349 }
1350 wl->rate_set = wl1271_min_rate_get(wl);
1351 wl->sta_rate_set = 0;
1352 ret = wl1271_acx_rate_policies(wl);
1353 if (ret < 0)
1354 goto out;
1355 ret = wl1271_acx_keep_alive_config(
1356 wl, CMD_TEMPL_KLV_IDX_NULL_DATA,
1357 ACX_KEEP_ALIVE_TPL_INVALID);
1358 if (ret < 0)
1359 goto out;
1360 set_bit(WL1271_FLAG_IDLE, &wl->flags);
1361 } else {
1362 /* increment the session counter */
1363 wl->session_counter++;
1364 if (wl->session_counter >= SESSION_COUNTER_MAX)
1365 wl->session_counter = 0;
1366 ret = wl1271_dummy_join(wl);
1367 if (ret < 0)
1368 goto out;
1369 clear_bit(WL1271_FLAG_IDLE, &wl->flags);
1370 }
1371
1372out:
1373 return ret;
1374}
1375
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001376static int wl1271_op_config(struct ieee80211_hw *hw, u32 changed)
1377{
1378 struct wl1271 *wl = hw->priv;
1379 struct ieee80211_conf *conf = &hw->conf;
1380 int channel, ret = 0;
1381
1382 channel = ieee80211_frequency_to_channel(conf->channel->center_freq);
1383
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001384 wl1271_debug(DEBUG_MAC80211, "mac80211 config ch %d psm %s power %d %s",
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001385 channel,
1386 conf->flags & IEEE80211_CONF_PS ? "on" : "off",
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001387 conf->power_level,
1388 conf->flags & IEEE80211_CONF_IDLE ? "idle" : "in use");
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001389
Juuso Oikarinen781608c2010-05-24 11:18:17 +03001390 /*
1391 * mac80211 will go to idle nearly immediately after transmitting some
1392 * frames, such as the deauth. To make sure those frames reach the air,
1393 * wait here until the TX queue is fully flushed.
1394 */
1395 if ((changed & IEEE80211_CONF_CHANGE_IDLE) &&
1396 (conf->flags & IEEE80211_CONF_IDLE))
1397 wl1271_tx_flush(wl);
1398
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001399 mutex_lock(&wl->mutex);
1400
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02001401 if (unlikely(wl->state == WL1271_STATE_OFF)) {
1402 ret = -EAGAIN;
Saravanan Dhanabal2c10bb92010-04-09 11:07:27 +03001403 goto out;
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02001404 }
Juuso Oikarinen8a5a37a2009-10-08 21:56:24 +03001405
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001406 ret = wl1271_ps_elp_wakeup(wl, false);
1407 if (ret < 0)
1408 goto out;
1409
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03001410 /* if the channel changes while joined, join again */
Juuso Oikarinen69e54342010-05-07 11:39:00 +03001411 if (changed & IEEE80211_CONF_CHANGE_CHANNEL &&
1412 ((wl->band != conf->channel->band) ||
1413 (wl->channel != channel))) {
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03001414 wl->band = conf->channel->band;
1415 wl->channel = channel;
1416
1417 /*
1418 * FIXME: the mac80211 should really provide a fixed rate
1419 * to use here. for now, just use the smallest possible rate
1420 * for the band as a fixed rate for association frames and
1421 * other control messages.
1422 */
1423 if (!test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
1424 wl1271_set_band_rate(wl);
1425
1426 wl->basic_rate = wl1271_min_rate_get(wl);
1427 ret = wl1271_acx_rate_policies(wl);
1428 if (ret < 0)
1429 wl1271_warning("rate policy for update channel "
1430 "failed %d", ret);
1431
1432 if (test_bit(WL1271_FLAG_JOINED, &wl->flags)) {
Juuso Oikarinen69e54342010-05-07 11:39:00 +03001433 ret = wl1271_join(wl, false);
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03001434 if (ret < 0)
1435 wl1271_warning("cmd join to update channel "
1436 "failed %d", ret);
1437 }
1438 }
1439
Luciano Coelhoc7f43e42009-12-11 15:40:44 +02001440 if (changed & IEEE80211_CONF_CHANGE_IDLE) {
Juuso Oikarinen0d58cbf2010-05-24 11:18:16 +03001441 ret = wl1271_handle_idle(wl, conf->flags & IEEE80211_CONF_IDLE);
1442 if (ret < 0)
1443 wl1271_warning("idle mode change failed %d", ret);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001444 }
1445
Juuso Oikarinen90494a92010-07-08 17:50:00 +03001446 /*
1447 * if mac80211 changes the PSM mode, make sure the mode is not
1448 * incorrectly changed after the pspoll failure active window.
1449 */
1450 if (changed & IEEE80211_CONF_CHANGE_PS)
1451 clear_bit(WL1271_FLAG_PSPOLL_FAILURE, &wl->flags);
1452
Juuso Oikarinen71449f82009-12-11 15:41:07 +02001453 if (conf->flags & IEEE80211_CONF_PS &&
1454 !test_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags)) {
1455 set_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001456
1457 /*
1458 * We enter PSM only if we're already associated.
1459 * If we're not, we'll enter it when joining an SSID,
1460 * through the bss_info_changed() hook.
1461 */
Juuso Oikarinen830fb672009-12-11 15:41:06 +02001462 if (test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags)) {
Juuso Oikarinen18f8d462010-02-22 08:38:34 +02001463 wl1271_debug(DEBUG_PSM, "psm enabled");
Juuso Oikarinend8c42c02010-02-18 13:25:36 +02001464 ret = wl1271_ps_set_mode(wl, STATION_POWER_SAVE_MODE,
Juuso Oikarinen8eab7b42010-09-24 03:10:11 +02001465 wl->basic_rate, true);
Juuso Oikarinenaf5e0842009-12-11 15:41:00 +02001466 }
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001467 } else if (!(conf->flags & IEEE80211_CONF_PS) &&
Juuso Oikarinen71449f82009-12-11 15:41:07 +02001468 test_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags)) {
Juuso Oikarinen18f8d462010-02-22 08:38:34 +02001469 wl1271_debug(DEBUG_PSM, "psm disabled");
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001470
Juuso Oikarinen71449f82009-12-11 15:41:07 +02001471 clear_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001472
Juuso Oikarinen71449f82009-12-11 15:41:07 +02001473 if (test_bit(WL1271_FLAG_PSM, &wl->flags))
Juuso Oikarinend8c42c02010-02-18 13:25:36 +02001474 ret = wl1271_ps_set_mode(wl, STATION_ACTIVE_MODE,
Juuso Oikarinen8eab7b42010-09-24 03:10:11 +02001475 wl->basic_rate, true);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001476 }
1477
1478 if (conf->power_level != wl->power_level) {
1479 ret = wl1271_acx_tx_power(wl, conf->power_level);
1480 if (ret < 0)
Juuso Oikarinenc6317a52009-11-02 20:22:08 +02001481 goto out_sleep;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001482
1483 wl->power_level = conf->power_level;
1484 }
1485
1486out_sleep:
1487 wl1271_ps_elp_sleep(wl);
1488
1489out:
1490 mutex_unlock(&wl->mutex);
1491
1492 return ret;
1493}
1494
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001495struct wl1271_filter_params {
1496 bool enabled;
1497 int mc_list_length;
1498 u8 mc_list[ACX_MC_ADDRESS_GROUP_MAX][ETH_ALEN];
1499};
1500
Jiri Pirko22bedad32010-04-01 21:22:57 +00001501static u64 wl1271_op_prepare_multicast(struct ieee80211_hw *hw,
1502 struct netdev_hw_addr_list *mc_list)
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001503{
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001504 struct wl1271_filter_params *fp;
Jiri Pirko22bedad32010-04-01 21:22:57 +00001505 struct netdev_hw_addr *ha;
Saravanan Dhanabal2c10bb92010-04-09 11:07:27 +03001506 struct wl1271 *wl = hw->priv;
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001507
Saravanan Dhanabal2c10bb92010-04-09 11:07:27 +03001508 if (unlikely(wl->state == WL1271_STATE_OFF))
1509 return 0;
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001510
Juuso Oikarinen74441132009-10-13 12:47:53 +03001511 fp = kzalloc(sizeof(*fp), GFP_ATOMIC);
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001512 if (!fp) {
1513 wl1271_error("Out of memory setting filters.");
1514 return 0;
1515 }
1516
1517 /* update multicast filtering parameters */
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001518 fp->mc_list_length = 0;
Jiri Pirko22bedad32010-04-01 21:22:57 +00001519 if (netdev_hw_addr_list_count(mc_list) > ACX_MC_ADDRESS_GROUP_MAX) {
1520 fp->enabled = false;
1521 } else {
1522 fp->enabled = true;
1523 netdev_hw_addr_list_for_each(ha, mc_list) {
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001524 memcpy(fp->mc_list[fp->mc_list_length],
Jiri Pirko22bedad32010-04-01 21:22:57 +00001525 ha->addr, ETH_ALEN);
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001526 fp->mc_list_length++;
Jiri Pirko22bedad32010-04-01 21:22:57 +00001527 }
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001528 }
1529
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001530 return (u64)(unsigned long)fp;
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001531}
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001532
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001533#define WL1271_SUPPORTED_FILTERS (FIF_PROMISC_IN_BSS | \
1534 FIF_ALLMULTI | \
1535 FIF_FCSFAIL | \
1536 FIF_BCN_PRBRESP_PROMISC | \
1537 FIF_CONTROL | \
1538 FIF_OTHER_BSS)
1539
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001540static void wl1271_op_configure_filter(struct ieee80211_hw *hw,
1541 unsigned int changed,
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03001542 unsigned int *total, u64 multicast)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001543{
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001544 struct wl1271_filter_params *fp = (void *)(unsigned long)multicast;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001545 struct wl1271 *wl = hw->priv;
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001546 int ret;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001547
1548 wl1271_debug(DEBUG_MAC80211, "mac80211 configure filter");
1549
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001550 mutex_lock(&wl->mutex);
1551
Saravanan Dhanabal2c10bb92010-04-09 11:07:27 +03001552 *total &= WL1271_SUPPORTED_FILTERS;
1553 changed &= WL1271_SUPPORTED_FILTERS;
1554
1555 if (unlikely(wl->state == WL1271_STATE_OFF))
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001556 goto out;
1557
1558 ret = wl1271_ps_elp_wakeup(wl, false);
1559 if (ret < 0)
1560 goto out;
1561
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001562
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001563 if (*total & FIF_ALLMULTI)
1564 ret = wl1271_acx_group_address_tbl(wl, false, NULL, 0);
1565 else if (fp)
1566 ret = wl1271_acx_group_address_tbl(wl, fp->enabled,
1567 fp->mc_list,
1568 fp->mc_list_length);
1569 if (ret < 0)
1570 goto out_sleep;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001571
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001572 /* determine, whether supported filter values have changed */
1573 if (changed == 0)
1574 goto out_sleep;
1575
Juuso Oikarinen14b228a2010-03-18 12:26:43 +02001576 /* configure filters */
1577 wl->filters = *total;
1578 wl1271_configure_filters(wl, 0);
1579
Juuso Oikarinenb54853f2009-10-13 12:47:59 +03001580 /* apply configured filters */
1581 ret = wl1271_acx_rx_config(wl, wl->rx_config, wl->rx_filter);
1582 if (ret < 0)
1583 goto out_sleep;
1584
1585out_sleep:
1586 wl1271_ps_elp_sleep(wl);
1587
1588out:
1589 mutex_unlock(&wl->mutex);
Juuso Oikarinen14b228a2010-03-18 12:26:43 +02001590 kfree(fp);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001591}
1592
1593static int wl1271_op_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
1594 struct ieee80211_vif *vif,
1595 struct ieee80211_sta *sta,
1596 struct ieee80211_key_conf *key_conf)
1597{
1598 struct wl1271 *wl = hw->priv;
1599 const u8 *addr;
1600 int ret;
Juuso Oikarinenac4e4ce2009-10-08 21:56:19 +03001601 u32 tx_seq_32 = 0;
1602 u16 tx_seq_16 = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001603 u8 key_type;
1604
1605 static const u8 bcast_addr[ETH_ALEN] =
1606 { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
1607
1608 wl1271_debug(DEBUG_MAC80211, "mac80211 set key");
1609
1610 addr = sta ? sta->addr : bcast_addr;
1611
1612 wl1271_debug(DEBUG_CRYPT, "CMD: 0x%x", cmd);
1613 wl1271_dump(DEBUG_CRYPT, "ADDR: ", addr, ETH_ALEN);
1614 wl1271_debug(DEBUG_CRYPT, "Key: algo:0x%x, id:%d, len:%d flags 0x%x",
Johannes Berg97359d12010-08-10 09:46:38 +02001615 key_conf->cipher, key_conf->keyidx,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001616 key_conf->keylen, key_conf->flags);
1617 wl1271_dump(DEBUG_CRYPT, "KEY: ", key_conf->key, key_conf->keylen);
1618
1619 if (is_zero_ether_addr(addr)) {
1620 /* We dont support TX only encryption */
1621 ret = -EOPNOTSUPP;
1622 goto out;
1623 }
1624
1625 mutex_lock(&wl->mutex);
1626
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02001627 if (unlikely(wl->state == WL1271_STATE_OFF)) {
1628 ret = -EAGAIN;
1629 goto out_unlock;
1630 }
1631
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001632 ret = wl1271_ps_elp_wakeup(wl, false);
1633 if (ret < 0)
1634 goto out_unlock;
1635
Johannes Berg97359d12010-08-10 09:46:38 +02001636 switch (key_conf->cipher) {
1637 case WLAN_CIPHER_SUITE_WEP40:
1638 case WLAN_CIPHER_SUITE_WEP104:
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001639 key_type = KEY_WEP;
1640
1641 key_conf->hw_key_idx = key_conf->keyidx;
1642 break;
Johannes Berg97359d12010-08-10 09:46:38 +02001643 case WLAN_CIPHER_SUITE_TKIP:
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001644 key_type = KEY_TKIP;
1645
1646 key_conf->hw_key_idx = key_conf->keyidx;
Juuso Oikarinen04e36fc2010-02-22 08:38:40 +02001647 tx_seq_32 = WL1271_TX_SECURITY_HI32(wl->tx_security_seq);
1648 tx_seq_16 = WL1271_TX_SECURITY_LO16(wl->tx_security_seq);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001649 break;
Johannes Berg97359d12010-08-10 09:46:38 +02001650 case WLAN_CIPHER_SUITE_CCMP:
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001651 key_type = KEY_AES;
1652
1653 key_conf->flags |= IEEE80211_KEY_FLAG_GENERATE_IV;
Juuso Oikarinen04e36fc2010-02-22 08:38:40 +02001654 tx_seq_32 = WL1271_TX_SECURITY_HI32(wl->tx_security_seq);
1655 tx_seq_16 = WL1271_TX_SECURITY_LO16(wl->tx_security_seq);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001656 break;
Juuso Oikarinen7a557242010-09-27 12:42:07 +02001657 case WL1271_CIPHER_SUITE_GEM:
1658 key_type = KEY_GEM;
1659 tx_seq_32 = WL1271_TX_SECURITY_HI32(wl->tx_security_seq);
1660 tx_seq_16 = WL1271_TX_SECURITY_LO16(wl->tx_security_seq);
1661 break;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001662 default:
Johannes Berg97359d12010-08-10 09:46:38 +02001663 wl1271_error("Unknown key algo 0x%x", key_conf->cipher);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001664
1665 ret = -EOPNOTSUPP;
1666 goto out_sleep;
1667 }
1668
1669 switch (cmd) {
1670 case SET_KEY:
1671 ret = wl1271_cmd_set_key(wl, KEY_ADD_OR_REPLACE,
1672 key_conf->keyidx, key_type,
1673 key_conf->keylen, key_conf->key,
Juuso Oikarinenac4e4ce2009-10-08 21:56:19 +03001674 addr, tx_seq_32, tx_seq_16);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001675 if (ret < 0) {
1676 wl1271_error("Could not add or replace key");
1677 goto out_sleep;
1678 }
Juuso Oikarinenee444cf2010-02-18 13:25:50 +02001679
1680 /* the default WEP key needs to be configured at least once */
1681 if (key_type == KEY_WEP) {
1682 ret = wl1271_cmd_set_default_wep_key(wl,
1683 wl->default_key);
1684 if (ret < 0)
1685 goto out_sleep;
1686 }
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001687 break;
1688
1689 case DISABLE_KEY:
Juuso Oikarinenfddc7dd2010-02-18 13:25:49 +02001690 /* The wl1271 does not allow to remove unicast keys - they
1691 will be cleared automatically on next CMD_JOIN. Ignore the
1692 request silently, as we dont want the mac80211 to emit
1693 an error message. */
1694 if (!is_broadcast_ether_addr(addr))
1695 break;
1696
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001697 ret = wl1271_cmd_set_key(wl, KEY_REMOVE,
1698 key_conf->keyidx, key_type,
1699 key_conf->keylen, key_conf->key,
Juuso Oikarinenac4e4ce2009-10-08 21:56:19 +03001700 addr, 0, 0);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001701 if (ret < 0) {
1702 wl1271_error("Could not remove key");
1703 goto out_sleep;
1704 }
1705 break;
1706
1707 default:
1708 wl1271_error("Unsupported key cmd 0x%x", cmd);
1709 ret = -EOPNOTSUPP;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001710 break;
1711 }
1712
1713out_sleep:
1714 wl1271_ps_elp_sleep(wl);
1715
1716out_unlock:
1717 mutex_unlock(&wl->mutex);
1718
1719out:
1720 return ret;
1721}
1722
1723static int wl1271_op_hw_scan(struct ieee80211_hw *hw,
Johannes Berga060bbf2010-04-27 11:59:34 +02001724 struct ieee80211_vif *vif,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001725 struct cfg80211_scan_request *req)
1726{
1727 struct wl1271 *wl = hw->priv;
1728 int ret;
1729 u8 *ssid = NULL;
Teemu Paasikiviabb0b3b2009-10-13 12:47:50 +03001730 size_t len = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001731
1732 wl1271_debug(DEBUG_MAC80211, "mac80211 hw scan");
1733
1734 if (req->n_ssids) {
1735 ssid = req->ssids[0].ssid;
Teemu Paasikiviabb0b3b2009-10-13 12:47:50 +03001736 len = req->ssids[0].ssid_len;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001737 }
1738
1739 mutex_lock(&wl->mutex);
1740
Juuso Oikarinenb739a422010-10-26 13:24:38 +02001741 if (wl->state == WL1271_STATE_OFF) {
1742 /*
1743 * We cannot return -EBUSY here because cfg80211 will expect
1744 * a call to ieee80211_scan_completed if we do - in this case
1745 * there won't be any call.
1746 */
1747 ret = -EAGAIN;
1748 goto out;
1749 }
1750
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001751 ret = wl1271_ps_elp_wakeup(wl, false);
1752 if (ret < 0)
1753 goto out;
1754
Luciano Coelho5924f892010-08-04 03:46:22 +03001755 ret = wl1271_scan(hw->priv, ssid, len, req);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001756
1757 wl1271_ps_elp_sleep(wl);
1758
1759out:
1760 mutex_unlock(&wl->mutex);
1761
1762 return ret;
1763}
1764
Arik Nemtsov68d069c2010-11-08 10:51:07 +01001765static int wl1271_op_set_frag_threshold(struct ieee80211_hw *hw, u32 value)
1766{
1767 struct wl1271 *wl = hw->priv;
1768 int ret = 0;
1769
1770 mutex_lock(&wl->mutex);
1771
1772 if (unlikely(wl->state == WL1271_STATE_OFF)) {
1773 ret = -EAGAIN;
1774 goto out;
1775 }
1776
1777 ret = wl1271_ps_elp_wakeup(wl, false);
1778 if (ret < 0)
1779 goto out;
1780
1781 ret = wl1271_acx_frag_threshold(wl, (u16)value);
1782 if (ret < 0)
1783 wl1271_warning("wl1271_op_set_frag_threshold failed: %d", ret);
1784
1785 wl1271_ps_elp_sleep(wl);
1786
1787out:
1788 mutex_unlock(&wl->mutex);
1789
1790 return ret;
1791}
1792
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001793static int wl1271_op_set_rts_threshold(struct ieee80211_hw *hw, u32 value)
1794{
1795 struct wl1271 *wl = hw->priv;
Saravanan Dhanabalaecb0562010-04-09 11:07:28 +03001796 int ret = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001797
1798 mutex_lock(&wl->mutex);
1799
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02001800 if (unlikely(wl->state == WL1271_STATE_OFF)) {
1801 ret = -EAGAIN;
Saravanan Dhanabalaecb0562010-04-09 11:07:28 +03001802 goto out;
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02001803 }
Saravanan Dhanabalaecb0562010-04-09 11:07:28 +03001804
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001805 ret = wl1271_ps_elp_wakeup(wl, false);
1806 if (ret < 0)
1807 goto out;
1808
1809 ret = wl1271_acx_rts_threshold(wl, (u16) value);
1810 if (ret < 0)
1811 wl1271_warning("wl1271_op_set_rts_threshold failed: %d", ret);
1812
1813 wl1271_ps_elp_sleep(wl);
1814
1815out:
1816 mutex_unlock(&wl->mutex);
1817
1818 return ret;
1819}
1820
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001821static void wl1271_ssid_set(struct wl1271 *wl, struct sk_buff *skb,
1822 int offset)
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001823{
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001824 u8 *ptr = skb->data + offset;
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001825
1826 /* find the location of the ssid in the beacon */
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001827 while (ptr < skb->data + skb->len) {
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001828 if (ptr[0] == WLAN_EID_SSID) {
1829 wl->ssid_len = ptr[1];
1830 memcpy(wl->ssid, ptr+2, wl->ssid_len);
1831 return;
1832 }
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001833 ptr += (ptr[1] + 2);
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001834 }
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001835 wl1271_error("No SSID in IEs!\n");
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001836}
1837
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001838static void wl1271_op_bss_info_changed(struct ieee80211_hw *hw,
1839 struct ieee80211_vif *vif,
1840 struct ieee80211_bss_conf *bss_conf,
1841 u32 changed)
1842{
1843 enum wl1271_cmd_ps_mode mode;
1844 struct wl1271 *wl = hw->priv;
Shahar Levi18357852010-10-13 16:09:41 +02001845 struct ieee80211_sta *sta = ieee80211_find_sta(vif, bss_conf->bssid);
Juuso Oikarinen8bf29b02010-02-18 13:25:51 +02001846 bool do_join = false;
Juuso Oikarinen69e54342010-05-07 11:39:00 +03001847 bool set_assoc = false;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001848 int ret;
1849
1850 wl1271_debug(DEBUG_MAC80211, "mac80211 bss info changed");
1851
1852 mutex_lock(&wl->mutex);
1853
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02001854 if (unlikely(wl->state == WL1271_STATE_OFF))
1855 goto out;
1856
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001857 ret = wl1271_ps_elp_wakeup(wl, false);
1858 if (ret < 0)
1859 goto out;
1860
Eliad Peller9ee82d52010-09-19 18:55:09 +02001861 if ((changed & BSS_CHANGED_BEACON_INT) &&
Juuso Oikarinen60e84c22010-03-26 12:53:25 +02001862 (wl->bss_type == BSS_TYPE_IBSS)) {
1863 wl1271_debug(DEBUG_ADHOC, "ad-hoc beacon interval updated: %d",
1864 bss_conf->beacon_int);
1865
1866 wl->beacon_int = bss_conf->beacon_int;
1867 do_join = true;
1868 }
1869
Eliad Peller9ee82d52010-09-19 18:55:09 +02001870 if ((changed & BSS_CHANGED_BEACON) &&
Juuso Oikarinen5da11dc2010-03-26 12:53:24 +02001871 (wl->bss_type == BSS_TYPE_IBSS)) {
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02001872 struct sk_buff *beacon = ieee80211_beacon_get(hw, vif);
1873
Juuso Oikarinen5da11dc2010-03-26 12:53:24 +02001874 wl1271_debug(DEBUG_ADHOC, "ad-hoc beacon updated");
1875
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02001876 if (beacon) {
1877 struct ieee80211_hdr *hdr;
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001878 int ieoffset = offsetof(struct ieee80211_mgmt,
1879 u.beacon.variable);
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001880
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001881 wl1271_ssid_set(wl, beacon, ieoffset);
1882
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02001883 ret = wl1271_cmd_template_set(wl, CMD_TEMPL_BEACON,
1884 beacon->data,
Juuso Oikarinen606c1482010-04-01 11:38:21 +03001885 beacon->len, 0,
1886 wl1271_min_rate_get(wl));
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02001887
1888 if (ret < 0) {
1889 dev_kfree_skb(beacon);
1890 goto out_sleep;
1891 }
1892
1893 hdr = (struct ieee80211_hdr *) beacon->data;
1894 hdr->frame_control = cpu_to_le16(
1895 IEEE80211_FTYPE_MGMT |
1896 IEEE80211_STYPE_PROBE_RESP);
1897
1898 ret = wl1271_cmd_template_set(wl,
1899 CMD_TEMPL_PROBE_RESPONSE,
1900 beacon->data,
Juuso Oikarinen606c1482010-04-01 11:38:21 +03001901 beacon->len, 0,
1902 wl1271_min_rate_get(wl));
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02001903 dev_kfree_skb(beacon);
1904 if (ret < 0)
1905 goto out_sleep;
Juuso Oikarinen8bf29b02010-02-18 13:25:51 +02001906
1907 /* Need to update the SSID (for filtering etc) */
1908 do_join = true;
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02001909 }
1910 }
1911
Juuso Oikarinen5da11dc2010-03-26 12:53:24 +02001912 if ((changed & BSS_CHANGED_BEACON_ENABLED) &&
1913 (wl->bss_type == BSS_TYPE_IBSS)) {
1914 wl1271_debug(DEBUG_ADHOC, "ad-hoc beaconing: %s",
1915 bss_conf->enable_beacon ? "enabled" : "disabled");
1916
1917 if (bss_conf->enable_beacon)
1918 wl->set_bss_type = BSS_TYPE_IBSS;
1919 else
1920 wl->set_bss_type = BSS_TYPE_STA_BSS;
1921 do_join = true;
1922 }
1923
Juuso Oikarinen00236aed2010-04-09 11:07:30 +03001924 if (changed & BSS_CHANGED_CQM) {
1925 bool enable = false;
1926 if (bss_conf->cqm_rssi_thold)
1927 enable = true;
1928 ret = wl1271_acx_rssi_snr_trigger(wl, enable,
1929 bss_conf->cqm_rssi_thold,
1930 bss_conf->cqm_rssi_hyst);
1931 if (ret < 0)
1932 goto out;
1933 wl->rssi_thold = bss_conf->cqm_rssi_thold;
1934 }
1935
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001936 if ((changed & BSS_CHANGED_BSSID) &&
1937 /*
1938 * Now we know the correct bssid, so we send a new join command
1939 * and enable the BSSID filter
1940 */
1941 memcmp(wl->bssid, bss_conf->bssid, ETH_ALEN)) {
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001942 memcpy(wl->bssid, bss_conf->bssid, ETH_ALEN);
Juuso Oikarinena0cb7be2010-03-18 12:26:44 +02001943
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001944 ret = wl1271_cmd_build_null_data(wl);
Juuso Oikarinena0cb7be2010-03-18 12:26:44 +02001945 if (ret < 0)
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001946 goto out_sleep;
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001947
Saravanan Dhanabal141418c2010-04-28 09:50:00 +03001948 ret = wl1271_build_qos_null_data(wl);
1949 if (ret < 0)
1950 goto out_sleep;
1951
Juuso Oikarinen14b228a2010-03-18 12:26:43 +02001952 /* filter out all packets not from this BSSID */
1953 wl1271_configure_filters(wl, 0);
1954
Juuso Oikarinen8bf29b02010-02-18 13:25:51 +02001955 /* Need to update the BSSID (for filtering etc) */
1956 do_join = true;
Juuso Oikarinen30240fc2010-02-18 13:25:38 +02001957 }
1958
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001959 if (changed & BSS_CHANGED_ASSOC) {
1960 if (bss_conf->assoc) {
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03001961 u32 rates;
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001962 int ieoffset;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001963 wl->aid = bss_conf->aid;
Juuso Oikarinen69e54342010-05-07 11:39:00 +03001964 set_assoc = true;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001965
Juuso Oikarinen90494a92010-07-08 17:50:00 +03001966 wl->ps_poll_failures = 0;
1967
Luciano Coelhoae751ba2009-10-12 15:08:57 +03001968 /*
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03001969 * use basic rates from AP, and determine lowest rate
1970 * to use with control frames.
1971 */
1972 rates = bss_conf->basic_rates;
1973 wl->basic_rate_set = wl1271_tx_enabled_rates_get(wl,
1974 rates);
1975 wl->basic_rate = wl1271_min_rate_get(wl);
1976 ret = wl1271_acx_rate_policies(wl);
1977 if (ret < 0)
1978 goto out_sleep;
1979
1980 /*
Luciano Coelhoae751ba2009-10-12 15:08:57 +03001981 * with wl1271, we don't need to update the
1982 * beacon_int and dtim_period, because the firmware
1983 * updates it by itself when the first beacon is
1984 * received after a join.
1985 */
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03001986 ret = wl1271_cmd_build_ps_poll(wl, wl->aid);
1987 if (ret < 0)
1988 goto out_sleep;
1989
Juuso Oikarinenc2b2d992010-03-26 12:53:28 +02001990 /*
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001991 * Get a template for hardware connection maintenance
Juuso Oikarinenc2b2d992010-03-26 12:53:28 +02001992 */
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02001993 dev_kfree_skb(wl->probereq);
1994 wl->probereq = wl1271_cmd_build_ap_probe_req(wl, NULL);
1995 ieoffset = offsetof(struct ieee80211_mgmt,
1996 u.probe_req.variable);
1997 wl1271_ssid_set(wl, wl->probereq, ieoffset);
Juuso Oikarinenc2b2d992010-03-26 12:53:28 +02001998
Juuso Oikarinen6ccbb922010-03-26 12:53:23 +02001999 /* enable the connection monitoring feature */
2000 ret = wl1271_acx_conn_monit_params(wl, true);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002001 if (ret < 0)
2002 goto out_sleep;
2003
2004 /* If we want to go in PSM but we're not there yet */
Juuso Oikarinen71449f82009-12-11 15:41:07 +02002005 if (test_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags) &&
2006 !test_bit(WL1271_FLAG_PSM, &wl->flags)) {
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002007 mode = STATION_POWER_SAVE_MODE;
Juuso Oikarinen65cddbf2010-08-24 06:28:03 +03002008 ret = wl1271_ps_set_mode(wl, mode,
Juuso Oikarinen8eab7b42010-09-24 03:10:11 +02002009 wl->basic_rate,
Juuso Oikarinen65cddbf2010-08-24 06:28:03 +03002010 true);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002011 if (ret < 0)
2012 goto out_sleep;
2013 }
Juuso Oikarinend94cd292009-10-08 21:56:25 +03002014 } else {
2015 /* use defaults when not associated */
Juuso Oikarinenc2c192a2010-07-27 03:30:09 +03002016 clear_bit(WL1271_FLAG_STA_STATE_SENT, &wl->flags);
Juuso Oikarinen830fb672009-12-11 15:41:06 +02002017 clear_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags);
Juuso Oikarinend94cd292009-10-08 21:56:25 +03002018 wl->aid = 0;
Juuso Oikarinen6ccbb922010-03-26 12:53:23 +02002019
Juuso Oikarinen2f6724b2010-11-24 08:16:57 +02002020 /* free probe-request template */
2021 dev_kfree_skb(wl->probereq);
2022 wl->probereq = NULL;
2023
Juuso Oikarinen8d2ef7b2010-07-08 17:50:03 +03002024 /* re-enable dynamic ps - just in case */
Juuso Oikarinenf532be62010-07-08 17:50:05 +03002025 ieee80211_enable_dyn_ps(wl->vif);
Juuso Oikarinen8d2ef7b2010-07-08 17:50:03 +03002026
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03002027 /* revert back to minimum rates for the current band */
2028 wl1271_set_band_rate(wl);
2029 wl->basic_rate = wl1271_min_rate_get(wl);
2030 ret = wl1271_acx_rate_policies(wl);
2031 if (ret < 0)
2032 goto out_sleep;
2033
Juuso Oikarinen6ccbb922010-03-26 12:53:23 +02002034 /* disable connection monitor features */
2035 ret = wl1271_acx_conn_monit_params(wl, false);
Juuso Oikarinenc1899552010-03-26 12:53:32 +02002036
2037 /* Disable the keep-alive feature */
2038 ret = wl1271_acx_keep_alive_mode(wl, false);
Juuso Oikarinen6ccbb922010-03-26 12:53:23 +02002039 if (ret < 0)
2040 goto out_sleep;
Juuso Oikarinenb84a7d32010-11-22 12:59:08 +02002041
2042 /* restore the bssid filter and go to dummy bssid */
2043 wl1271_unjoin(wl);
2044 wl1271_dummy_join(wl);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002045 }
Juuso Oikarinend94cd292009-10-08 21:56:25 +03002046
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002047 }
Juuso Oikarinen8a5a37a2009-10-08 21:56:24 +03002048
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002049 if (changed & BSS_CHANGED_ERP_SLOT) {
2050 if (bss_conf->use_short_slot)
2051 ret = wl1271_acx_slot(wl, SLOT_TIME_SHORT);
2052 else
2053 ret = wl1271_acx_slot(wl, SLOT_TIME_LONG);
2054 if (ret < 0) {
2055 wl1271_warning("Set slot time failed %d", ret);
2056 goto out_sleep;
2057 }
2058 }
2059
2060 if (changed & BSS_CHANGED_ERP_PREAMBLE) {
2061 if (bss_conf->use_short_preamble)
2062 wl1271_acx_set_preamble(wl, ACX_PREAMBLE_SHORT);
2063 else
2064 wl1271_acx_set_preamble(wl, ACX_PREAMBLE_LONG);
2065 }
2066
2067 if (changed & BSS_CHANGED_ERP_CTS_PROT) {
2068 if (bss_conf->use_cts_prot)
2069 ret = wl1271_acx_cts_protect(wl, CTSPROTECT_ENABLE);
2070 else
2071 ret = wl1271_acx_cts_protect(wl, CTSPROTECT_DISABLE);
2072 if (ret < 0) {
2073 wl1271_warning("Set ctsprotect failed %d", ret);
2074 goto out_sleep;
2075 }
2076 }
2077
Shahar Levi18357852010-10-13 16:09:41 +02002078 /*
2079 * Takes care of: New association with HT enable,
2080 * HT information change in beacon.
2081 */
2082 if (sta &&
2083 (changed & BSS_CHANGED_HT) &&
2084 (bss_conf->channel_type != NL80211_CHAN_NO_HT)) {
2085 ret = wl1271_acx_set_ht_capabilities(wl, &sta->ht_cap, true);
2086 if (ret < 0) {
2087 wl1271_warning("Set ht cap true failed %d", ret);
2088 goto out_sleep;
2089 }
2090 ret = wl1271_acx_set_ht_information(wl,
2091 bss_conf->ht_operation_mode);
2092 if (ret < 0) {
2093 wl1271_warning("Set ht information failed %d", ret);
2094 goto out_sleep;
2095 }
2096 }
2097 /*
2098 * Takes care of: New association without HT,
2099 * Disassociation.
2100 */
2101 else if (sta && (changed & BSS_CHANGED_ASSOC)) {
2102 ret = wl1271_acx_set_ht_capabilities(wl, &sta->ht_cap, false);
2103 if (ret < 0) {
2104 wl1271_warning("Set ht cap false failed %d", ret);
2105 goto out_sleep;
2106 }
2107 }
2108
Juuso Oikarinenca52a5e2010-07-08 17:50:02 +03002109 if (changed & BSS_CHANGED_ARP_FILTER) {
2110 __be32 addr = bss_conf->arp_addr_list[0];
2111 WARN_ON(wl->bss_type != BSS_TYPE_STA_BSS);
2112
Eliad Pellerc5312772010-12-09 11:31:27 +02002113 if (bss_conf->arp_addr_cnt == 1 &&
2114 bss_conf->arp_filter_enabled) {
2115 /*
2116 * The template should have been configured only upon
2117 * association. however, it seems that the correct ip
2118 * isn't being set (when sending), so we have to
2119 * reconfigure the template upon every ip change.
2120 */
2121 ret = wl1271_cmd_build_arp_rsp(wl, addr);
2122 if (ret < 0) {
2123 wl1271_warning("build arp rsp failed: %d", ret);
2124 goto out_sleep;
2125 }
2126
2127 ret = wl1271_acx_arp_ip_filter(wl,
2128 (ACX_ARP_FILTER_ARP_FILTERING |
2129 ACX_ARP_FILTER_AUTO_ARP),
2130 addr);
2131 } else
2132 ret = wl1271_acx_arp_ip_filter(wl, 0, addr);
Juuso Oikarinenca52a5e2010-07-08 17:50:02 +03002133
2134 if (ret < 0)
2135 goto out_sleep;
2136 }
2137
Juuso Oikarinen8bf29b02010-02-18 13:25:51 +02002138 if (do_join) {
Juuso Oikarinen69e54342010-05-07 11:39:00 +03002139 ret = wl1271_join(wl, set_assoc);
Juuso Oikarinen8bf29b02010-02-18 13:25:51 +02002140 if (ret < 0) {
2141 wl1271_warning("cmd join failed %d", ret);
2142 goto out_sleep;
2143 }
Juuso Oikarinenc1899552010-03-26 12:53:32 +02002144 }
2145
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002146out_sleep:
2147 wl1271_ps_elp_sleep(wl);
2148
2149out:
2150 mutex_unlock(&wl->mutex);
2151}
2152
Kalle Valoc6999d82010-02-18 13:25:41 +02002153static int wl1271_op_conf_tx(struct ieee80211_hw *hw, u16 queue,
2154 const struct ieee80211_tx_queue_params *params)
2155{
2156 struct wl1271 *wl = hw->priv;
Kalle Valo4695dc92010-03-18 12:26:38 +02002157 u8 ps_scheme;
Kalle Valoc6999d82010-02-18 13:25:41 +02002158 int ret;
2159
2160 mutex_lock(&wl->mutex);
2161
2162 wl1271_debug(DEBUG_MAC80211, "mac80211 conf tx %d", queue);
2163
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02002164 if (unlikely(wl->state == WL1271_STATE_OFF)) {
2165 ret = -EAGAIN;
2166 goto out;
2167 }
2168
Kalle Valoc82c1dd2010-02-18 13:25:47 +02002169 ret = wl1271_ps_elp_wakeup(wl, false);
2170 if (ret < 0)
2171 goto out;
2172
Juuso Oikarinenb43316d2010-03-18 12:26:26 +02002173 /* the txop is confed in units of 32us by the mac80211, we need us */
Kalle Valoc6999d82010-02-18 13:25:41 +02002174 ret = wl1271_acx_ac_cfg(wl, wl1271_tx_get_queue(queue),
2175 params->cw_min, params->cw_max,
Juuso Oikarinenb43316d2010-03-18 12:26:26 +02002176 params->aifs, params->txop << 5);
Kalle Valoc6999d82010-02-18 13:25:41 +02002177 if (ret < 0)
Kalle Valoc82c1dd2010-02-18 13:25:47 +02002178 goto out_sleep;
Kalle Valoc6999d82010-02-18 13:25:41 +02002179
Kalle Valo4695dc92010-03-18 12:26:38 +02002180 if (params->uapsd)
2181 ps_scheme = CONF_PS_SCHEME_UPSD_TRIGGER;
2182 else
2183 ps_scheme = CONF_PS_SCHEME_LEGACY;
2184
Kalle Valoc6999d82010-02-18 13:25:41 +02002185 ret = wl1271_acx_tid_cfg(wl, wl1271_tx_get_queue(queue),
2186 CONF_CHANNEL_TYPE_EDCF,
2187 wl1271_tx_get_queue(queue),
Kalle Valo4695dc92010-03-18 12:26:38 +02002188 ps_scheme, CONF_ACK_POLICY_LEGACY, 0, 0);
Kalle Valoc6999d82010-02-18 13:25:41 +02002189 if (ret < 0)
Kalle Valoc82c1dd2010-02-18 13:25:47 +02002190 goto out_sleep;
2191
2192out_sleep:
2193 wl1271_ps_elp_sleep(wl);
Kalle Valoc6999d82010-02-18 13:25:41 +02002194
2195out:
2196 mutex_unlock(&wl->mutex);
2197
2198 return ret;
2199}
2200
Juuso Oikarinenbbbb5382010-07-08 17:49:57 +03002201static u64 wl1271_op_get_tsf(struct ieee80211_hw *hw)
2202{
2203
2204 struct wl1271 *wl = hw->priv;
2205 u64 mactime = ULLONG_MAX;
2206 int ret;
2207
2208 wl1271_debug(DEBUG_MAC80211, "mac80211 get tsf");
2209
2210 mutex_lock(&wl->mutex);
2211
Juuso Oikarinenf8d98022010-10-26 13:24:39 +02002212 if (unlikely(wl->state == WL1271_STATE_OFF))
2213 goto out;
2214
Juuso Oikarinenbbbb5382010-07-08 17:49:57 +03002215 ret = wl1271_ps_elp_wakeup(wl, false);
2216 if (ret < 0)
2217 goto out;
2218
2219 ret = wl1271_acx_tsf_info(wl, &mactime);
2220 if (ret < 0)
2221 goto out_sleep;
2222
2223out_sleep:
2224 wl1271_ps_elp_sleep(wl);
2225
2226out:
2227 mutex_unlock(&wl->mutex);
2228 return mactime;
2229}
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002230
John W. Linvilleece550d2010-07-28 16:41:06 -04002231static int wl1271_op_get_survey(struct ieee80211_hw *hw, int idx,
2232 struct survey_info *survey)
2233{
2234 struct wl1271 *wl = hw->priv;
2235 struct ieee80211_conf *conf = &hw->conf;
Juuso Oikarinenb739a422010-10-26 13:24:38 +02002236
John W. Linvilleece550d2010-07-28 16:41:06 -04002237 if (idx != 0)
2238 return -ENOENT;
Juuso Oikarinenb739a422010-10-26 13:24:38 +02002239
John W. Linvilleece550d2010-07-28 16:41:06 -04002240 survey->channel = conf->channel;
2241 survey->filled = SURVEY_INFO_NOISE_DBM;
2242 survey->noise = wl->noise;
Juuso Oikarinenb739a422010-10-26 13:24:38 +02002243
John W. Linvilleece550d2010-07-28 16:41:06 -04002244 return 0;
2245}
2246
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002247/* can't be const, mac80211 writes to this */
2248static struct ieee80211_rate wl1271_rates[] = {
2249 { .bitrate = 10,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002250 .hw_value = CONF_HW_BIT_RATE_1MBPS,
2251 .hw_value_short = CONF_HW_BIT_RATE_1MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002252 { .bitrate = 20,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002253 .hw_value = CONF_HW_BIT_RATE_2MBPS,
2254 .hw_value_short = CONF_HW_BIT_RATE_2MBPS,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002255 .flags = IEEE80211_RATE_SHORT_PREAMBLE },
2256 { .bitrate = 55,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002257 .hw_value = CONF_HW_BIT_RATE_5_5MBPS,
2258 .hw_value_short = CONF_HW_BIT_RATE_5_5MBPS,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002259 .flags = IEEE80211_RATE_SHORT_PREAMBLE },
2260 { .bitrate = 110,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002261 .hw_value = CONF_HW_BIT_RATE_11MBPS,
2262 .hw_value_short = CONF_HW_BIT_RATE_11MBPS,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002263 .flags = IEEE80211_RATE_SHORT_PREAMBLE },
2264 { .bitrate = 60,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002265 .hw_value = CONF_HW_BIT_RATE_6MBPS,
2266 .hw_value_short = CONF_HW_BIT_RATE_6MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002267 { .bitrate = 90,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002268 .hw_value = CONF_HW_BIT_RATE_9MBPS,
2269 .hw_value_short = CONF_HW_BIT_RATE_9MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002270 { .bitrate = 120,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002271 .hw_value = CONF_HW_BIT_RATE_12MBPS,
2272 .hw_value_short = CONF_HW_BIT_RATE_12MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002273 { .bitrate = 180,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002274 .hw_value = CONF_HW_BIT_RATE_18MBPS,
2275 .hw_value_short = CONF_HW_BIT_RATE_18MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002276 { .bitrate = 240,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002277 .hw_value = CONF_HW_BIT_RATE_24MBPS,
2278 .hw_value_short = CONF_HW_BIT_RATE_24MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002279 { .bitrate = 360,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002280 .hw_value = CONF_HW_BIT_RATE_36MBPS,
2281 .hw_value_short = CONF_HW_BIT_RATE_36MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002282 { .bitrate = 480,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002283 .hw_value = CONF_HW_BIT_RATE_48MBPS,
2284 .hw_value_short = CONF_HW_BIT_RATE_48MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002285 { .bitrate = 540,
Juuso Oikarinen2b60100b2009-10-13 12:47:39 +03002286 .hw_value = CONF_HW_BIT_RATE_54MBPS,
2287 .hw_value_short = CONF_HW_BIT_RATE_54MBPS, },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002288};
2289
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002290/* can't be const, mac80211 writes to this */
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002291static struct ieee80211_channel wl1271_channels[] = {
Luciano Coelhoa2d0e3f2009-12-11 15:40:46 +02002292 { .hw_value = 1, .center_freq = 2412, .max_power = 25 },
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002293 { .hw_value = 2, .center_freq = 2417, .max_power = 25 },
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002294 { .hw_value = 3, .center_freq = 2422, .max_power = 25 },
2295 { .hw_value = 4, .center_freq = 2427, .max_power = 25 },
2296 { .hw_value = 5, .center_freq = 2432, .max_power = 25 },
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002297 { .hw_value = 6, .center_freq = 2437, .max_power = 25 },
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002298 { .hw_value = 7, .center_freq = 2442, .max_power = 25 },
2299 { .hw_value = 8, .center_freq = 2447, .max_power = 25 },
2300 { .hw_value = 9, .center_freq = 2452, .max_power = 25 },
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002301 { .hw_value = 10, .center_freq = 2457, .max_power = 25 },
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002302 { .hw_value = 11, .center_freq = 2462, .max_power = 25 },
2303 { .hw_value = 12, .center_freq = 2467, .max_power = 25 },
2304 { .hw_value = 13, .center_freq = 2472, .max_power = 25 },
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002305};
2306
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002307/* mapping to indexes for wl1271_rates */
Tobias Klausera0ea9492010-05-20 10:38:11 +02002308static const u8 wl1271_rate_to_idx_2ghz[] = {
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002309 /* MCS rates are used only with 11n */
Shahar Levi18357852010-10-13 16:09:41 +02002310 7, /* CONF_HW_RXTX_RATE_MCS7 */
2311 6, /* CONF_HW_RXTX_RATE_MCS6 */
2312 5, /* CONF_HW_RXTX_RATE_MCS5 */
2313 4, /* CONF_HW_RXTX_RATE_MCS4 */
2314 3, /* CONF_HW_RXTX_RATE_MCS3 */
2315 2, /* CONF_HW_RXTX_RATE_MCS2 */
2316 1, /* CONF_HW_RXTX_RATE_MCS1 */
2317 0, /* CONF_HW_RXTX_RATE_MCS0 */
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002318
2319 11, /* CONF_HW_RXTX_RATE_54 */
2320 10, /* CONF_HW_RXTX_RATE_48 */
2321 9, /* CONF_HW_RXTX_RATE_36 */
2322 8, /* CONF_HW_RXTX_RATE_24 */
2323
2324 /* TI-specific rate */
2325 CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_22 */
2326
2327 7, /* CONF_HW_RXTX_RATE_18 */
2328 6, /* CONF_HW_RXTX_RATE_12 */
2329 3, /* CONF_HW_RXTX_RATE_11 */
2330 5, /* CONF_HW_RXTX_RATE_9 */
2331 4, /* CONF_HW_RXTX_RATE_6 */
2332 2, /* CONF_HW_RXTX_RATE_5_5 */
2333 1, /* CONF_HW_RXTX_RATE_2 */
2334 0 /* CONF_HW_RXTX_RATE_1 */
2335};
2336
Shahar Levie8b03a22010-10-13 16:09:39 +02002337/* 11n STA capabilities */
2338#define HW_RX_HIGHEST_RATE 72
2339
Shahar Levi00d20102010-11-08 11:20:10 +00002340#ifdef CONFIG_WL12XX_HT
2341#define WL12XX_HT_CAP { \
Shahar Levie8b03a22010-10-13 16:09:39 +02002342 .cap = IEEE80211_HT_CAP_GRN_FLD | IEEE80211_HT_CAP_SGI_20, \
2343 .ht_supported = true, \
2344 .ampdu_factor = IEEE80211_HT_MAX_AMPDU_8K, \
2345 .ampdu_density = IEEE80211_HT_MPDU_DENSITY_8, \
2346 .mcs = { \
2347 .rx_mask = { 0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, }, \
2348 .rx_highest = cpu_to_le16(HW_RX_HIGHEST_RATE), \
2349 .tx_params = IEEE80211_HT_MCS_TX_DEFINED, \
2350 }, \
2351}
Shahar Levi18357852010-10-13 16:09:41 +02002352#else
Shahar Levi00d20102010-11-08 11:20:10 +00002353#define WL12XX_HT_CAP { \
Shahar Levi18357852010-10-13 16:09:41 +02002354 .ht_supported = false, \
2355}
2356#endif
Shahar Levie8b03a22010-10-13 16:09:39 +02002357
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002358/* can't be const, mac80211 writes to this */
2359static struct ieee80211_supported_band wl1271_band_2ghz = {
2360 .channels = wl1271_channels,
2361 .n_channels = ARRAY_SIZE(wl1271_channels),
2362 .bitrates = wl1271_rates,
2363 .n_bitrates = ARRAY_SIZE(wl1271_rates),
Shahar Levi00d20102010-11-08 11:20:10 +00002364 .ht_cap = WL12XX_HT_CAP,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002365};
2366
Teemu Paasikivi1ebec3d2009-10-13 12:47:48 +03002367/* 5 GHz data rates for WL1273 */
2368static struct ieee80211_rate wl1271_rates_5ghz[] = {
2369 { .bitrate = 60,
2370 .hw_value = CONF_HW_BIT_RATE_6MBPS,
2371 .hw_value_short = CONF_HW_BIT_RATE_6MBPS, },
2372 { .bitrate = 90,
2373 .hw_value = CONF_HW_BIT_RATE_9MBPS,
2374 .hw_value_short = CONF_HW_BIT_RATE_9MBPS, },
2375 { .bitrate = 120,
2376 .hw_value = CONF_HW_BIT_RATE_12MBPS,
2377 .hw_value_short = CONF_HW_BIT_RATE_12MBPS, },
2378 { .bitrate = 180,
2379 .hw_value = CONF_HW_BIT_RATE_18MBPS,
2380 .hw_value_short = CONF_HW_BIT_RATE_18MBPS, },
2381 { .bitrate = 240,
2382 .hw_value = CONF_HW_BIT_RATE_24MBPS,
2383 .hw_value_short = CONF_HW_BIT_RATE_24MBPS, },
2384 { .bitrate = 360,
2385 .hw_value = CONF_HW_BIT_RATE_36MBPS,
2386 .hw_value_short = CONF_HW_BIT_RATE_36MBPS, },
2387 { .bitrate = 480,
2388 .hw_value = CONF_HW_BIT_RATE_48MBPS,
2389 .hw_value_short = CONF_HW_BIT_RATE_48MBPS, },
2390 { .bitrate = 540,
2391 .hw_value = CONF_HW_BIT_RATE_54MBPS,
2392 .hw_value_short = CONF_HW_BIT_RATE_54MBPS, },
2393};
2394
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002395/* 5 GHz band channels for WL1273 */
Teemu Paasikivi1ebec3d2009-10-13 12:47:48 +03002396static struct ieee80211_channel wl1271_channels_5ghz[] = {
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002397 { .hw_value = 7, .center_freq = 5035},
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002398 { .hw_value = 8, .center_freq = 5040},
2399 { .hw_value = 9, .center_freq = 5045},
2400 { .hw_value = 11, .center_freq = 5055},
2401 { .hw_value = 12, .center_freq = 5060},
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002402 { .hw_value = 16, .center_freq = 5080},
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002403 { .hw_value = 34, .center_freq = 5170},
2404 { .hw_value = 36, .center_freq = 5180},
2405 { .hw_value = 38, .center_freq = 5190},
2406 { .hw_value = 40, .center_freq = 5200},
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002407 { .hw_value = 42, .center_freq = 5210},
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002408 { .hw_value = 44, .center_freq = 5220},
2409 { .hw_value = 46, .center_freq = 5230},
2410 { .hw_value = 48, .center_freq = 5240},
2411 { .hw_value = 52, .center_freq = 5260},
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002412 { .hw_value = 56, .center_freq = 5280},
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002413 { .hw_value = 60, .center_freq = 5300},
2414 { .hw_value = 64, .center_freq = 5320},
2415 { .hw_value = 100, .center_freq = 5500},
2416 { .hw_value = 104, .center_freq = 5520},
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002417 { .hw_value = 108, .center_freq = 5540},
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002418 { .hw_value = 112, .center_freq = 5560},
2419 { .hw_value = 116, .center_freq = 5580},
2420 { .hw_value = 120, .center_freq = 5600},
2421 { .hw_value = 124, .center_freq = 5620},
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002422 { .hw_value = 128, .center_freq = 5640},
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002423 { .hw_value = 132, .center_freq = 5660},
2424 { .hw_value = 136, .center_freq = 5680},
2425 { .hw_value = 140, .center_freq = 5700},
2426 { .hw_value = 149, .center_freq = 5745},
Juuso Oikarinenfa21c7a2010-08-10 08:22:02 +02002427 { .hw_value = 153, .center_freq = 5765},
Juuso Oikarinenfa97f462010-11-10 11:27:20 +01002428 { .hw_value = 157, .center_freq = 5785},
2429 { .hw_value = 161, .center_freq = 5805},
Teemu Paasikivi1ebec3d2009-10-13 12:47:48 +03002430 { .hw_value = 165, .center_freq = 5825},
2431};
2432
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002433/* mapping to indexes for wl1271_rates_5ghz */
Tobias Klausera0ea9492010-05-20 10:38:11 +02002434static const u8 wl1271_rate_to_idx_5ghz[] = {
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002435 /* MCS rates are used only with 11n */
Shahar Levi18357852010-10-13 16:09:41 +02002436 7, /* CONF_HW_RXTX_RATE_MCS7 */
2437 6, /* CONF_HW_RXTX_RATE_MCS6 */
2438 5, /* CONF_HW_RXTX_RATE_MCS5 */
2439 4, /* CONF_HW_RXTX_RATE_MCS4 */
2440 3, /* CONF_HW_RXTX_RATE_MCS3 */
2441 2, /* CONF_HW_RXTX_RATE_MCS2 */
2442 1, /* CONF_HW_RXTX_RATE_MCS1 */
2443 0, /* CONF_HW_RXTX_RATE_MCS0 */
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002444
2445 7, /* CONF_HW_RXTX_RATE_54 */
2446 6, /* CONF_HW_RXTX_RATE_48 */
2447 5, /* CONF_HW_RXTX_RATE_36 */
2448 4, /* CONF_HW_RXTX_RATE_24 */
2449
2450 /* TI-specific rate */
2451 CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_22 */
2452
2453 3, /* CONF_HW_RXTX_RATE_18 */
2454 2, /* CONF_HW_RXTX_RATE_12 */
2455 CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_11 */
2456 1, /* CONF_HW_RXTX_RATE_9 */
2457 0, /* CONF_HW_RXTX_RATE_6 */
2458 CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_5_5 */
2459 CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_2 */
2460 CONF_HW_RXTX_RATE_UNSUPPORTED /* CONF_HW_RXTX_RATE_1 */
2461};
Teemu Paasikivi1ebec3d2009-10-13 12:47:48 +03002462
2463static struct ieee80211_supported_band wl1271_band_5ghz = {
2464 .channels = wl1271_channels_5ghz,
2465 .n_channels = ARRAY_SIZE(wl1271_channels_5ghz),
2466 .bitrates = wl1271_rates_5ghz,
2467 .n_bitrates = ARRAY_SIZE(wl1271_rates_5ghz),
Shahar Levi00d20102010-11-08 11:20:10 +00002468 .ht_cap = WL12XX_HT_CAP,
Teemu Paasikivi1ebec3d2009-10-13 12:47:48 +03002469};
2470
Tobias Klausera0ea9492010-05-20 10:38:11 +02002471static const u8 *wl1271_band_rate_to_idx[] = {
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002472 [IEEE80211_BAND_2GHZ] = wl1271_rate_to_idx_2ghz,
2473 [IEEE80211_BAND_5GHZ] = wl1271_rate_to_idx_5ghz
2474};
2475
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002476static const struct ieee80211_ops wl1271_ops = {
2477 .start = wl1271_op_start,
2478 .stop = wl1271_op_stop,
2479 .add_interface = wl1271_op_add_interface,
2480 .remove_interface = wl1271_op_remove_interface,
2481 .config = wl1271_op_config,
Juuso Oikarinenc87dec92009-10-08 21:56:31 +03002482 .prepare_multicast = wl1271_op_prepare_multicast,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002483 .configure_filter = wl1271_op_configure_filter,
2484 .tx = wl1271_op_tx,
2485 .set_key = wl1271_op_set_key,
2486 .hw_scan = wl1271_op_hw_scan,
2487 .bss_info_changed = wl1271_op_bss_info_changed,
Arik Nemtsov68d069c2010-11-08 10:51:07 +01002488 .set_frag_threshold = wl1271_op_set_frag_threshold,
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002489 .set_rts_threshold = wl1271_op_set_rts_threshold,
Kalle Valoc6999d82010-02-18 13:25:41 +02002490 .conf_tx = wl1271_op_conf_tx,
Juuso Oikarinenbbbb5382010-07-08 17:49:57 +03002491 .get_tsf = wl1271_op_get_tsf,
John W. Linvilleece550d2010-07-28 16:41:06 -04002492 .get_survey = wl1271_op_get_survey,
Kalle Valoc8c90872010-02-18 13:25:53 +02002493 CFG80211_TESTMODE_CMD(wl1271_tm_cmd)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002494};
2495
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002496
Teemu Paasikivi6a2de932010-10-14 11:00:04 +02002497u8 wl1271_rate_to_idx(int rate, enum ieee80211_band band)
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002498{
2499 u8 idx;
2500
Teemu Paasikivi6a2de932010-10-14 11:00:04 +02002501 BUG_ON(band >= sizeof(wl1271_band_rate_to_idx)/sizeof(u8 *));
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002502
2503 if (unlikely(rate >= CONF_HW_RXTX_RATE_MAX)) {
2504 wl1271_error("Illegal RX rate from HW: %d", rate);
2505 return 0;
2506 }
2507
Teemu Paasikivi6a2de932010-10-14 11:00:04 +02002508 idx = wl1271_band_rate_to_idx[band][rate];
Juuso Oikarinenf876bb92010-03-26 12:53:11 +02002509 if (unlikely(idx == CONF_HW_RXTX_RATE_UNSUPPORTED)) {
2510 wl1271_error("Unsupported RX rate from HW: %d", rate);
2511 return 0;
2512 }
2513
2514 return idx;
2515}
2516
Juuso Oikarinen7fc3a862010-03-18 12:26:32 +02002517static ssize_t wl1271_sysfs_show_bt_coex_state(struct device *dev,
2518 struct device_attribute *attr,
2519 char *buf)
2520{
2521 struct wl1271 *wl = dev_get_drvdata(dev);
2522 ssize_t len;
2523
Juuso Oikarinen2f63b012010-08-10 06:38:35 +02002524 len = PAGE_SIZE;
Juuso Oikarinen7fc3a862010-03-18 12:26:32 +02002525
2526 mutex_lock(&wl->mutex);
2527 len = snprintf(buf, len, "%d\n\n0 - off\n1 - on\n",
2528 wl->sg_enabled);
2529 mutex_unlock(&wl->mutex);
2530
2531 return len;
2532
2533}
2534
2535static ssize_t wl1271_sysfs_store_bt_coex_state(struct device *dev,
2536 struct device_attribute *attr,
2537 const char *buf, size_t count)
2538{
2539 struct wl1271 *wl = dev_get_drvdata(dev);
2540 unsigned long res;
2541 int ret;
2542
2543 ret = strict_strtoul(buf, 10, &res);
2544
2545 if (ret < 0) {
2546 wl1271_warning("incorrect value written to bt_coex_mode");
2547 return count;
2548 }
2549
2550 mutex_lock(&wl->mutex);
2551
2552 res = !!res;
2553
2554 if (res == wl->sg_enabled)
2555 goto out;
2556
2557 wl->sg_enabled = res;
2558
2559 if (wl->state == WL1271_STATE_OFF)
2560 goto out;
2561
2562 ret = wl1271_ps_elp_wakeup(wl, false);
2563 if (ret < 0)
2564 goto out;
2565
2566 wl1271_acx_sg_enable(wl, wl->sg_enabled);
2567 wl1271_ps_elp_sleep(wl);
2568
2569 out:
2570 mutex_unlock(&wl->mutex);
2571 return count;
2572}
2573
2574static DEVICE_ATTR(bt_coex_state, S_IRUGO | S_IWUSR,
2575 wl1271_sysfs_show_bt_coex_state,
2576 wl1271_sysfs_store_bt_coex_state);
2577
Juuso Oikarinend717fd62010-05-07 11:38:58 +03002578static ssize_t wl1271_sysfs_show_hw_pg_ver(struct device *dev,
2579 struct device_attribute *attr,
2580 char *buf)
2581{
2582 struct wl1271 *wl = dev_get_drvdata(dev);
2583 ssize_t len;
2584
Juuso Oikarinen2f63b012010-08-10 06:38:35 +02002585 len = PAGE_SIZE;
Juuso Oikarinend717fd62010-05-07 11:38:58 +03002586
2587 mutex_lock(&wl->mutex);
2588 if (wl->hw_pg_ver >= 0)
2589 len = snprintf(buf, len, "%d\n", wl->hw_pg_ver);
2590 else
2591 len = snprintf(buf, len, "n/a\n");
2592 mutex_unlock(&wl->mutex);
2593
2594 return len;
2595}
2596
2597static DEVICE_ATTR(hw_pg_ver, S_IRUGO | S_IWUSR,
2598 wl1271_sysfs_show_hw_pg_ver, NULL);
2599
Teemu Paasikivi2d5e82b2010-02-22 08:38:21 +02002600int wl1271_register_hw(struct wl1271 *wl)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002601{
2602 int ret;
2603
2604 if (wl->mac80211_registered)
2605 return 0;
2606
2607 SET_IEEE80211_PERM_ADDR(wl->hw, wl->mac_addr);
2608
2609 ret = ieee80211_register_hw(wl->hw);
2610 if (ret < 0) {
2611 wl1271_error("unable to register mac80211 hw: %d", ret);
2612 return ret;
2613 }
2614
2615 wl->mac80211_registered = true;
2616
Eliad Pellerd60080a2010-11-24 12:53:16 +02002617 wl1271_debugfs_init(wl);
2618
Juuso Oikarinenc2c192a2010-07-27 03:30:09 +03002619 register_netdevice_notifier(&wl1271_dev_notifier);
2620
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002621 wl1271_notice("loaded");
2622
2623 return 0;
2624}
Teemu Paasikivi50b3eb42010-02-22 08:38:26 +02002625EXPORT_SYMBOL_GPL(wl1271_register_hw);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002626
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002627void wl1271_unregister_hw(struct wl1271 *wl)
2628{
Juuso Oikarinenc2c192a2010-07-27 03:30:09 +03002629 unregister_netdevice_notifier(&wl1271_dev_notifier);
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002630 ieee80211_unregister_hw(wl->hw);
2631 wl->mac80211_registered = false;
2632
2633}
2634EXPORT_SYMBOL_GPL(wl1271_unregister_hw);
2635
Teemu Paasikivi2d5e82b2010-02-22 08:38:21 +02002636int wl1271_init_ieee80211(struct wl1271 *wl)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002637{
Juuso Oikarinen7a557242010-09-27 12:42:07 +02002638 static const u32 cipher_suites[] = {
2639 WLAN_CIPHER_SUITE_WEP40,
2640 WLAN_CIPHER_SUITE_WEP104,
2641 WLAN_CIPHER_SUITE_TKIP,
2642 WLAN_CIPHER_SUITE_CCMP,
2643 WL1271_CIPHER_SUITE_GEM,
2644 };
2645
Juuso Oikarinen1e2b7972009-10-08 21:56:20 +03002646 /* The tx descriptor buffer and the TKIP space. */
2647 wl->hw->extra_tx_headroom = WL1271_TKIP_IV_SPACE +
2648 sizeof(struct wl1271_tx_hw_descr);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002649
2650 /* unit us */
2651 /* FIXME: find a proper value */
2652 wl->hw->channel_change_time = 10000;
Juuso Oikarinen50c500a2010-04-01 11:38:22 +03002653 wl->hw->max_listen_interval = wl->conf.conn.max_listen_interval;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002654
2655 wl->hw->flags = IEEE80211_HW_SIGNAL_DBM |
Juuso Oikarinen03442a32009-11-23 23:22:14 +02002656 IEEE80211_HW_BEACON_FILTER |
Juuso Oikarinen0a343322010-02-22 08:38:41 +02002657 IEEE80211_HW_SUPPORTS_PS |
Kalle Valo4695dc92010-03-18 12:26:38 +02002658 IEEE80211_HW_SUPPORTS_UAPSD |
Juuso Oikarinena9af0922010-03-26 12:53:30 +02002659 IEEE80211_HW_HAS_RATE_CONTROL |
Juuso Oikarinen00236aed2010-04-09 11:07:30 +03002660 IEEE80211_HW_CONNECTION_MONITOR |
2661 IEEE80211_HW_SUPPORTS_CQM_RSSI;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002662
Juuso Oikarinen7a557242010-09-27 12:42:07 +02002663 wl->hw->wiphy->cipher_suites = cipher_suites;
2664 wl->hw->wiphy->n_cipher_suites = ARRAY_SIZE(cipher_suites);
2665
Juuso Oikarinene0d8bbf2009-12-11 15:41:04 +02002666 wl->hw->wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
2667 BIT(NL80211_IFTYPE_ADHOC);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002668 wl->hw->wiphy->max_scan_ssids = 1;
Guy Eilamea559b42010-12-09 16:54:59 +02002669 /*
2670 * Maximum length of elements in scanning probe request templates
2671 * should be the maximum length possible for a template, without
2672 * the IEEE80211 header of the template
2673 */
2674 wl->hw->wiphy->max_scan_ie_len = WL1271_CMD_TEMPL_MAX_SIZE -
2675 sizeof(struct ieee80211_header);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002676 wl->hw->wiphy->bands[IEEE80211_BAND_2GHZ] = &wl1271_band_2ghz;
Juuso Oikarinen11eb5422010-08-24 06:28:03 +03002677 wl->hw->wiphy->bands[IEEE80211_BAND_5GHZ] = &wl1271_band_5ghz;
Teemu Paasikivi1ebec3d2009-10-13 12:47:48 +03002678
Kalle Valo12bd8942010-03-18 12:26:33 +02002679 wl->hw->queues = 4;
Juuso Oikarinen31627dc2010-03-26 12:53:12 +02002680 wl->hw->max_rates = 1;
Kalle Valo12bd8942010-03-18 12:26:33 +02002681
Juuso Oikarinenb7417d92010-11-10 11:27:19 +01002682 wl->hw->wiphy->reg_notifier = wl1271_reg_notify;
2683
Teemu Paasikivi8197b712010-02-22 08:38:23 +02002684 SET_IEEE80211_DEV(wl->hw, wl1271_wl_to_dev(wl));
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002685
2686 return 0;
2687}
Teemu Paasikivi50b3eb42010-02-22 08:38:26 +02002688EXPORT_SYMBOL_GPL(wl1271_init_ieee80211);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002689
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002690#define WL1271_DEFAULT_CHANNEL 0
Teemu Paasikivic332a4b2010-02-18 13:25:57 +02002691
Teemu Paasikivi2d5e82b2010-02-22 08:38:21 +02002692struct ieee80211_hw *wl1271_alloc_hw(void)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002693{
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002694 struct ieee80211_hw *hw;
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002695 struct platform_device *plat_dev = NULL;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002696 struct wl1271 *wl;
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002697 int i, ret;
Ido Yariv1f37cbc2010-09-30 13:28:27 +02002698 unsigned int order;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002699
2700 hw = ieee80211_alloc_hw(sizeof(*wl), &wl1271_ops);
2701 if (!hw) {
2702 wl1271_error("could not alloc ieee80211_hw");
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002703 ret = -ENOMEM;
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002704 goto err_hw_alloc;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002705 }
2706
Julia Lawall929ebd32010-05-15 23:16:39 +02002707 plat_dev = kmemdup(&wl1271_device, sizeof(wl1271_device), GFP_KERNEL);
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002708 if (!plat_dev) {
2709 wl1271_error("could not allocate platform_device");
2710 ret = -ENOMEM;
2711 goto err_plat_alloc;
2712 }
2713
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002714 wl = hw->priv;
2715 memset(wl, 0, sizeof(*wl));
2716
Juuso Oikarinen01c09162009-10-13 12:47:55 +03002717 INIT_LIST_HEAD(&wl->list);
2718
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002719 wl->hw = hw;
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002720 wl->plat_dev = plat_dev;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002721
2722 skb_queue_head_init(&wl->tx_queue);
2723
Juuso Oikarinen37b70a82009-10-08 21:56:21 +03002724 INIT_DELAYED_WORK(&wl->elp_work, wl1271_elp_work);
Juuso Oikarinen90494a92010-07-08 17:50:00 +03002725 INIT_DELAYED_WORK(&wl->pspoll_work, wl1271_pspoll_work);
Juuso Oikarinen117b38d2010-09-30 10:43:28 +02002726 INIT_WORK(&wl->irq_work, wl1271_irq_work);
2727 INIT_WORK(&wl->tx_work, wl1271_tx_work);
2728 INIT_WORK(&wl->recovery_work, wl1271_recovery_work);
2729 INIT_DELAYED_WORK(&wl->scan_complete_work, wl1271_scan_complete_work);
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002730 wl->channel = WL1271_DEFAULT_CHANNEL;
Juuso Oikarinen60e84c22010-03-26 12:53:25 +02002731 wl->beacon_int = WL1271_DEFAULT_BEACON_INT;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002732 wl->default_key = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002733 wl->rx_counter = 0;
2734 wl->rx_config = WL1271_DEFAULT_RX_CONFIG;
2735 wl->rx_filter = WL1271_DEFAULT_RX_FILTER;
Juuso Oikarinen19ad0712009-11-02 20:22:11 +02002736 wl->psm_entry_retry = 0;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002737 wl->power_level = WL1271_DEFAULT_POWER_LEVEL;
Juuso Oikarinena6fe2312009-12-11 15:40:58 +02002738 wl->basic_rate_set = CONF_TX_RATE_MASK_BASIC;
Juuso Oikarinenebba60c2010-04-01 11:38:20 +03002739 wl->basic_rate = CONF_TX_RATE_MASK_BASIC;
Juuso Oikarinen830fb672009-12-11 15:41:06 +02002740 wl->rate_set = CONF_TX_RATE_MASK_BASIC;
2741 wl->sta_rate_set = 0;
Juuso Oikarinen8a5a37a2009-10-08 21:56:24 +03002742 wl->band = IEEE80211_BAND_2GHZ;
Juuso Oikarinenb771eee2009-10-08 21:56:34 +03002743 wl->vif = NULL;
Juuso Oikarinen830fb672009-12-11 15:41:06 +02002744 wl->flags = 0;
Juuso Oikarinen7fc3a862010-03-18 12:26:32 +02002745 wl->sg_enabled = true;
Juuso Oikarinend717fd62010-05-07 11:38:58 +03002746 wl->hw_pg_ver = -1;
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002747
Ido Yariv25eeb9e2010-10-12 16:20:06 +02002748 memset(wl->tx_frames_map, 0, sizeof(wl->tx_frames_map));
Juuso Oikarinenbe7078c2009-10-08 21:56:26 +03002749 for (i = 0; i < ACX_TX_DESCRIPTORS; i++)
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002750 wl->tx_frames[i] = NULL;
2751
2752 spin_lock_init(&wl->wl_lock);
2753
Luciano Coelhof5fc0f82009-08-06 16:25:28 +03002754 wl->state = WL1271_STATE_OFF;
2755 mutex_init(&wl->mutex);
2756
Teemu Paasikivic332a4b2010-02-18 13:25:57 +02002757 /* Apply default driver configuration. */
2758 wl1271_conf_init(wl);
2759
Ido Yariv1f37cbc2010-09-30 13:28:27 +02002760 order = get_order(WL1271_AGGR_BUFFER_SIZE);
2761 wl->aggr_buf = (u8 *)__get_free_pages(GFP_KERNEL, order);
2762 if (!wl->aggr_buf) {
2763 ret = -ENOMEM;
2764 goto err_hw;
2765 }
2766
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002767 /* Register platform device */
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002768 ret = platform_device_register(wl->plat_dev);
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002769 if (ret) {
2770 wl1271_error("couldn't register platform device");
Ido Yariv1f37cbc2010-09-30 13:28:27 +02002771 goto err_aggr;
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002772 }
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002773 dev_set_drvdata(&wl->plat_dev->dev, wl);
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002774
Juuso Oikarinen7fc3a862010-03-18 12:26:32 +02002775 /* Create sysfs file to control bt coex state */
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002776 ret = device_create_file(&wl->plat_dev->dev, &dev_attr_bt_coex_state);
Juuso Oikarinen7fc3a862010-03-18 12:26:32 +02002777 if (ret < 0) {
2778 wl1271_error("failed to create sysfs file bt_coex_state");
2779 goto err_platform;
2780 }
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002781
Juuso Oikarinend717fd62010-05-07 11:38:58 +03002782 /* Create sysfs file to get HW PG version */
2783 ret = device_create_file(&wl->plat_dev->dev, &dev_attr_hw_pg_ver);
2784 if (ret < 0) {
2785 wl1271_error("failed to create sysfs file hw_pg_ver");
2786 goto err_bt_coex_state;
2787 }
2788
Teemu Paasikivic332a4b2010-02-18 13:25:57 +02002789 return hw;
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002790
Juuso Oikarinend717fd62010-05-07 11:38:58 +03002791err_bt_coex_state:
2792 device_remove_file(&wl->plat_dev->dev, &dev_attr_bt_coex_state);
2793
Juuso Oikarinen7fc3a862010-03-18 12:26:32 +02002794err_platform:
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002795 platform_device_unregister(wl->plat_dev);
Juuso Oikarinen7fc3a862010-03-18 12:26:32 +02002796
Ido Yariv1f37cbc2010-09-30 13:28:27 +02002797err_aggr:
2798 free_pages((unsigned long)wl->aggr_buf, order);
2799
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002800err_hw:
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002801 wl1271_debugfs_exit(wl);
2802 kfree(plat_dev);
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002803
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002804err_plat_alloc:
2805 ieee80211_free_hw(hw);
2806
2807err_hw_alloc:
2808
Juuso Oikarinena1dd8182010-03-18 12:26:31 +02002809 return ERR_PTR(ret);
Teemu Paasikivic332a4b2010-02-18 13:25:57 +02002810}
Teemu Paasikivi50b3eb42010-02-22 08:38:26 +02002811EXPORT_SYMBOL_GPL(wl1271_alloc_hw);
Teemu Paasikivic332a4b2010-02-18 13:25:57 +02002812
2813int wl1271_free_hw(struct wl1271 *wl)
2814{
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002815 platform_device_unregister(wl->plat_dev);
Ido Yariv1f37cbc2010-09-30 13:28:27 +02002816 free_pages((unsigned long)wl->aggr_buf,
2817 get_order(WL1271_AGGR_BUFFER_SIZE));
Teemu Paasikivi3b56dd62010-03-18 12:26:46 +02002818 kfree(wl->plat_dev);
Teemu Paasikivic332a4b2010-02-18 13:25:57 +02002819
2820 wl1271_debugfs_exit(wl);
2821
Teemu Paasikivic332a4b2010-02-18 13:25:57 +02002822 vfree(wl->fw);
2823 wl->fw = NULL;
2824 kfree(wl->nvs);
2825 wl->nvs = NULL;
2826
2827 kfree(wl->fw_status);
2828 kfree(wl->tx_res_if);
2829
2830 ieee80211_free_hw(wl->hw);
2831
2832 return 0;
2833}
Teemu Paasikivi50b3eb42010-02-22 08:38:26 +02002834EXPORT_SYMBOL_GPL(wl1271_free_hw);
2835
Eliad Peller17c17552010-12-12 12:15:35 +02002836u32 wl12xx_debug_level;
2837EXPORT_SYMBOL_GPL(wl12xx_debug_level);
2838module_param_named(debug_level, wl12xx_debug_level, uint, DEBUG_NONE);
2839MODULE_PARM_DESC(debug_level, "wl12xx debugging level");
2840
Teemu Paasikivi50b3eb42010-02-22 08:38:26 +02002841MODULE_LICENSE("GPL");
2842MODULE_AUTHOR("Luciano Coelho <luciano.coelho@nokia.com>");
2843MODULE_AUTHOR("Juuso Oikarinen <juuso.oikarinen@nokia.com>");