2009-08-06 21:25:28 +08:00
|
|
|
/*
|
|
|
|
* This file is part of wl1271
|
|
|
|
*
|
2010-02-18 19:25:51 +08:00
|
|
|
* Copyright (C) 2008-2010 Nokia Corporation
|
2009-08-06 21:25:28 +08:00
|
|
|
*
|
|
|
|
* Contact: Luciano Coelho <luciano.coelho@nokia.com>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* version 2 as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
|
|
* 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/firmware.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/spi/spi.h>
|
|
|
|
#include <linux/crc32.h>
|
|
|
|
#include <linux/etherdevice.h>
|
2009-10-09 02:56:32 +08:00
|
|
|
#include <linux/vmalloc.h>
|
2010-03-18 18:26:31 +08:00
|
|
|
#include <linux/platform_device.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2011-03-31 16:07:01 +08:00
|
|
|
#include <linux/wl12xx.h>
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-11-08 19:20:10 +08:00
|
|
|
#include "wl12xx.h"
|
2009-08-06 21:25:28 +08:00
|
|
|
#include "wl12xx_80211.h"
|
2010-11-08 19:20:10 +08:00
|
|
|
#include "reg.h"
|
|
|
|
#include "io.h"
|
|
|
|
#include "event.h"
|
|
|
|
#include "tx.h"
|
|
|
|
#include "rx.h"
|
|
|
|
#include "ps.h"
|
|
|
|
#include "init.h"
|
|
|
|
#include "debugfs.h"
|
|
|
|
#include "cmd.h"
|
|
|
|
#include "boot.h"
|
|
|
|
#include "testmode.h"
|
|
|
|
#include "scan.h"
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
#define WL1271_BOOT_RETRIES 3
|
|
|
|
|
2009-10-13 17:47:44 +08:00
|
|
|
static struct conf_drv_settings default_conf = {
|
|
|
|
.sg = {
|
2010-03-18 18:26:30 +08:00
|
|
|
.params = {
|
|
|
|
[CONF_SG_BT_PER_THRESHOLD] = 7500,
|
|
|
|
[CONF_SG_HV3_MAX_OVERRIDE] = 0,
|
|
|
|
[CONF_SG_BT_NFS_SAMPLE_INTERVAL] = 400,
|
2011-03-21 23:58:32 +08:00
|
|
|
[CONF_SG_BT_LOAD_RATIO] = 200,
|
2010-07-08 22:50:03 +08:00
|
|
|
[CONF_SG_AUTO_PS_MODE] = 1,
|
2010-03-18 18:26:30 +08:00
|
|
|
[CONF_SG_AUTO_SCAN_PROBE_REQ] = 170,
|
|
|
|
[CONF_SG_ACTIVE_SCAN_DURATION_FACTOR_HV3] = 50,
|
|
|
|
[CONF_SG_ANTENNA_CONFIGURATION] = 0,
|
|
|
|
[CONF_SG_BEACON_MISS_PERCENT] = 60,
|
|
|
|
[CONF_SG_RATE_ADAPT_THRESH] = 12,
|
|
|
|
[CONF_SG_RATE_ADAPT_SNR] = 0,
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_MASTER_MIN_BR] = 10,
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_MASTER_MAX_BR] = 30,
|
|
|
|
[CONF_SG_WLAN_PS_MAX_BT_ACL_MASTER_BR] = 8,
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_SLAVE_MIN_BR] = 20,
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_SLAVE_MAX_BR] = 50,
|
|
|
|
/* Note: with UPSD, this should be 4 */
|
|
|
|
[CONF_SG_WLAN_PS_MAX_BT_ACL_SLAVE_BR] = 8,
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_MASTER_MIN_EDR] = 7,
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_MASTER_MAX_EDR] = 25,
|
|
|
|
[CONF_SG_WLAN_PS_MAX_BT_ACL_MASTER_EDR] = 20,
|
|
|
|
/* Note: with UPDS, this should be 15 */
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_SLAVE_MIN_EDR] = 8,
|
|
|
|
/* Note: with UPDS, this should be 50 */
|
|
|
|
[CONF_SG_WLAN_PS_BT_ACL_SLAVE_MAX_EDR] = 40,
|
|
|
|
/* Note: with UPDS, this should be 10 */
|
|
|
|
[CONF_SG_WLAN_PS_MAX_BT_ACL_SLAVE_EDR] = 20,
|
|
|
|
[CONF_SG_RXT] = 1200,
|
|
|
|
[CONF_SG_TXT] = 1000,
|
|
|
|
[CONF_SG_ADAPTIVE_RXT_TXT] = 1,
|
|
|
|
[CONF_SG_PS_POLL_TIMEOUT] = 10,
|
|
|
|
[CONF_SG_UPSD_TIMEOUT] = 10,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_BT_ACL_MASTER_MIN_EDR] = 7,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_BT_ACL_MASTER_MAX_EDR] = 15,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_MAX_BT_ACL_MASTER_EDR] = 15,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_BT_ACL_SLAVE_MIN_EDR] = 8,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_BT_ACL_SLAVE_MAX_EDR] = 20,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_MAX_BT_ACL_SLAVE_EDR] = 15,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_BT_ACL_MIN_BR] = 20,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_BT_ACL_MAX_BR] = 50,
|
|
|
|
[CONF_SG_WLAN_ACTIVE_MAX_BT_ACL_BR] = 10,
|
|
|
|
[CONF_SG_PASSIVE_SCAN_DURATION_FACTOR_HV3] = 200,
|
|
|
|
[CONF_SG_PASSIVE_SCAN_DURATION_FACTOR_A2DP] = 800,
|
|
|
|
[CONF_SG_PASSIVE_SCAN_A2DP_BT_TIME] = 75,
|
|
|
|
[CONF_SG_PASSIVE_SCAN_A2DP_WLAN_TIME] = 15,
|
|
|
|
[CONF_SG_HV3_MAX_SERVED] = 6,
|
|
|
|
[CONF_SG_DHCP_TIME] = 5000,
|
|
|
|
[CONF_SG_ACTIVE_SCAN_DURATION_FACTOR_A2DP] = 100,
|
|
|
|
},
|
|
|
|
.state = CONF_SG_PROTECTIVE,
|
2009-10-13 17:47:44 +08:00
|
|
|
},
|
|
|
|
.rx = {
|
|
|
|
.rx_msdu_life_time = 512000,
|
|
|
|
.packet_detection_threshold = 0,
|
|
|
|
.ps_poll_timeout = 15,
|
|
|
|
.upsd_timeout = 15,
|
|
|
|
.rts_threshold = 2347,
|
2009-12-11 21:40:54 +08:00
|
|
|
.rx_cca_threshold = 0,
|
|
|
|
.irq_blk_threshold = 0xFFFF,
|
|
|
|
.irq_pkt_threshold = 0,
|
|
|
|
.irq_timeout = 600,
|
2009-10-13 17:47:44 +08:00
|
|
|
.queue_type = CONF_RX_QUEUE_TYPE_LOW_PRIORITY,
|
|
|
|
},
|
|
|
|
.tx = {
|
|
|
|
.tx_energy_detection = 0,
|
2010-10-16 23:44:51 +08:00
|
|
|
.sta_rc_conf = {
|
2010-04-01 16:38:20 +08:00
|
|
|
.enabled_rates = 0,
|
2009-10-13 17:47:44 +08:00
|
|
|
.short_retry_limit = 10,
|
|
|
|
.long_retry_limit = 10,
|
2010-10-16 23:44:51 +08:00
|
|
|
.aflags = 0,
|
2009-10-13 17:47:41 +08:00
|
|
|
},
|
2009-10-13 17:47:44 +08:00
|
|
|
.ac_conf_count = 4,
|
|
|
|
.ac_conf = {
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_BE] = {
|
2009-10-13 17:47:44 +08:00
|
|
|
.ac = CONF_TX_AC_BE,
|
|
|
|
.cw_min = 15,
|
|
|
|
.cw_max = 63,
|
|
|
|
.aifsn = 3,
|
|
|
|
.tx_op_limit = 0,
|
2009-10-13 17:47:41 +08:00
|
|
|
},
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_BK] = {
|
2009-10-13 17:47:44 +08:00
|
|
|
.ac = CONF_TX_AC_BK,
|
|
|
|
.cw_min = 15,
|
|
|
|
.cw_max = 63,
|
|
|
|
.aifsn = 7,
|
|
|
|
.tx_op_limit = 0,
|
2009-10-13 17:47:41 +08:00
|
|
|
},
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_VI] = {
|
2009-10-13 17:47:44 +08:00
|
|
|
.ac = CONF_TX_AC_VI,
|
|
|
|
.cw_min = 15,
|
|
|
|
.cw_max = 63,
|
|
|
|
.aifsn = CONF_TX_AIFS_PIFS,
|
|
|
|
.tx_op_limit = 3008,
|
|
|
|
},
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_VO] = {
|
2009-10-13 17:47:44 +08:00
|
|
|
.ac = CONF_TX_AC_VO,
|
|
|
|
.cw_min = 15,
|
|
|
|
.cw_max = 63,
|
|
|
|
.aifsn = CONF_TX_AIFS_PIFS,
|
|
|
|
.tx_op_limit = 1504,
|
2009-10-13 17:47:41 +08:00
|
|
|
},
|
2009-10-13 17:47:42 +08:00
|
|
|
},
|
2010-10-16 23:44:51 +08:00
|
|
|
.ap_rc_conf = {
|
|
|
|
[0] = {
|
|
|
|
.enabled_rates = CONF_TX_AP_ENABLED_RATES,
|
|
|
|
.short_retry_limit = 10,
|
|
|
|
.long_retry_limit = 10,
|
|
|
|
.aflags = 0,
|
|
|
|
},
|
|
|
|
[1] = {
|
|
|
|
.enabled_rates = CONF_TX_AP_ENABLED_RATES,
|
|
|
|
.short_retry_limit = 10,
|
|
|
|
.long_retry_limit = 10,
|
|
|
|
.aflags = 0,
|
|
|
|
},
|
|
|
|
[2] = {
|
|
|
|
.enabled_rates = CONF_TX_AP_ENABLED_RATES,
|
|
|
|
.short_retry_limit = 10,
|
|
|
|
.long_retry_limit = 10,
|
|
|
|
.aflags = 0,
|
|
|
|
},
|
|
|
|
[3] = {
|
|
|
|
.enabled_rates = CONF_TX_AP_ENABLED_RATES,
|
|
|
|
.short_retry_limit = 10,
|
|
|
|
.long_retry_limit = 10,
|
|
|
|
.aflags = 0,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
.ap_mgmt_conf = {
|
|
|
|
.enabled_rates = CONF_TX_AP_DEFAULT_MGMT_RATES,
|
|
|
|
.short_retry_limit = 10,
|
|
|
|
.long_retry_limit = 10,
|
|
|
|
.aflags = 0,
|
|
|
|
},
|
|
|
|
.ap_bcst_conf = {
|
|
|
|
.enabled_rates = CONF_HW_BIT_RATE_1MBPS,
|
|
|
|
.short_retry_limit = 10,
|
|
|
|
.long_retry_limit = 10,
|
|
|
|
.aflags = 0,
|
|
|
|
},
|
2011-04-27 04:21:51 +08:00
|
|
|
.max_tx_retries = 100,
|
|
|
|
.ap_aging_period = 300,
|
2010-09-01 17:31:12 +08:00
|
|
|
.tid_conf_count = 4,
|
2009-10-13 17:47:44 +08:00
|
|
|
.tid_conf = {
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_BE] = {
|
|
|
|
.queue_id = CONF_TX_AC_BE,
|
|
|
|
.channel_type = CONF_CHANNEL_TYPE_EDCF,
|
2009-10-13 17:47:44 +08:00
|
|
|
.tsid = CONF_TX_AC_BE,
|
|
|
|
.ps_scheme = CONF_PS_SCHEME_LEGACY,
|
|
|
|
.ack_policy = CONF_ACK_POLICY_LEGACY,
|
|
|
|
.apsd_conf = {0, 0},
|
2009-10-13 17:47:42 +08:00
|
|
|
},
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_BK] = {
|
|
|
|
.queue_id = CONF_TX_AC_BK,
|
|
|
|
.channel_type = CONF_CHANNEL_TYPE_EDCF,
|
|
|
|
.tsid = CONF_TX_AC_BK,
|
2009-10-13 17:47:44 +08:00
|
|
|
.ps_scheme = CONF_PS_SCHEME_LEGACY,
|
|
|
|
.ack_policy = CONF_ACK_POLICY_LEGACY,
|
|
|
|
.apsd_conf = {0, 0},
|
|
|
|
},
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_VI] = {
|
|
|
|
.queue_id = CONF_TX_AC_VI,
|
|
|
|
.channel_type = CONF_CHANNEL_TYPE_EDCF,
|
|
|
|
.tsid = CONF_TX_AC_VI,
|
2009-10-13 17:47:44 +08:00
|
|
|
.ps_scheme = CONF_PS_SCHEME_LEGACY,
|
|
|
|
.ack_policy = CONF_ACK_POLICY_LEGACY,
|
|
|
|
.apsd_conf = {0, 0},
|
|
|
|
},
|
2010-09-01 17:31:12 +08:00
|
|
|
[CONF_TX_AC_VO] = {
|
|
|
|
.queue_id = CONF_TX_AC_VO,
|
|
|
|
.channel_type = CONF_CHANNEL_TYPE_EDCF,
|
|
|
|
.tsid = CONF_TX_AC_VO,
|
2009-10-13 17:47:44 +08:00
|
|
|
.ps_scheme = CONF_PS_SCHEME_LEGACY,
|
|
|
|
.ack_policy = CONF_ACK_POLICY_LEGACY,
|
|
|
|
.apsd_conf = {0, 0},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
.frag_threshold = IEEE80211_MAX_FRAG_THRESHOLD,
|
2009-12-11 21:40:54 +08:00
|
|
|
.tx_compl_timeout = 700,
|
2010-04-01 16:38:20 +08:00
|
|
|
.tx_compl_threshold = 4,
|
|
|
|
.basic_rate = CONF_HW_BIT_RATE_1MBPS,
|
|
|
|
.basic_rate_5 = CONF_HW_BIT_RATE_6MBPS,
|
2010-10-16 23:44:51 +08:00
|
|
|
.tmpl_short_retry_limit = 10,
|
|
|
|
.tmpl_long_retry_limit = 10,
|
2009-10-13 17:47:44 +08:00
|
|
|
},
|
|
|
|
.conn = {
|
|
|
|
.wake_up_event = CONF_WAKE_UP_EVENT_DTIM,
|
2010-04-01 16:38:22 +08:00
|
|
|
.listen_interval = 1,
|
2009-10-13 17:47:44 +08:00
|
|
|
.bcn_filt_mode = CONF_BCN_FILT_MODE_ENABLED,
|
|
|
|
.bcn_filt_ie_count = 1,
|
|
|
|
.bcn_filt_ie = {
|
|
|
|
[0] = {
|
|
|
|
.ie = WLAN_EID_CHANNEL_SWITCH,
|
|
|
|
.rule = CONF_BCN_RULE_PASS_ON_APPEARANCE,
|
2009-10-13 17:47:42 +08:00
|
|
|
}
|
2009-10-13 17:47:43 +08:00
|
|
|
},
|
2009-12-11 21:40:54 +08:00
|
|
|
.synch_fail_thold = 10,
|
2009-10-13 17:47:44 +08:00
|
|
|
.bss_lose_timeout = 100,
|
|
|
|
.beacon_rx_timeout = 10000,
|
|
|
|
.broadcast_timeout = 20000,
|
|
|
|
.rx_broadcast_in_ps = 1,
|
2010-07-08 22:50:00 +08:00
|
|
|
.ps_poll_threshold = 10,
|
|
|
|
.ps_poll_recovery_period = 700,
|
2009-10-13 17:47:46 +08:00
|
|
|
.bet_enable = CONF_BET_MODE_ENABLE,
|
2011-03-15 00:53:10 +08:00
|
|
|
.bet_max_consecutive = 50,
|
2010-09-24 09:10:11 +08:00
|
|
|
.psm_entry_retries = 5,
|
2011-02-02 15:59:34 +08:00
|
|
|
.psm_exit_retries = 255,
|
2010-09-24 09:10:11 +08:00
|
|
|
.psm_entry_nullfunc_retries = 3,
|
|
|
|
.psm_entry_hangover_period = 1,
|
2010-04-01 16:38:22 +08:00
|
|
|
.keep_alive_interval = 55000,
|
|
|
|
.max_listen_interval = 20,
|
2009-10-13 17:47:44 +08:00
|
|
|
},
|
2009-12-11 21:40:50 +08:00
|
|
|
.itrim = {
|
|
|
|
.enable = false,
|
|
|
|
.timeout = 50000,
|
2009-12-11 21:41:08 +08:00
|
|
|
},
|
|
|
|
.pm_config = {
|
|
|
|
.host_clk_settling_time = 5000,
|
|
|
|
.host_fast_wakeup_support = false
|
2010-04-09 16:07:30 +08:00
|
|
|
},
|
|
|
|
.roam_trigger = {
|
|
|
|
.trigger_pacing = 1,
|
|
|
|
.avg_weight_rssi_beacon = 20,
|
|
|
|
.avg_weight_rssi_data = 10,
|
|
|
|
.avg_weight_snr_beacon = 20,
|
2011-01-23 14:27:22 +08:00
|
|
|
.avg_weight_snr_data = 10,
|
2010-09-21 14:14:31 +08:00
|
|
|
},
|
|
|
|
.scan = {
|
|
|
|
.min_dwell_time_active = 7500,
|
|
|
|
.max_dwell_time_active = 30000,
|
2011-01-24 14:01:54 +08:00
|
|
|
.min_dwell_time_passive = 100000,
|
|
|
|
.max_dwell_time_passive = 100000,
|
2010-09-21 14:14:31 +08:00
|
|
|
.num_probe_reqs = 2,
|
|
|
|
},
|
2010-10-05 19:11:56 +08:00
|
|
|
.rf = {
|
|
|
|
.tx_per_channel_power_compensation_2 = {
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
.tx_per_channel_power_compensation_5 = {
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
},
|
2011-01-23 14:27:22 +08:00
|
|
|
.ht = {
|
|
|
|
.tx_ba_win_size = 64,
|
|
|
|
.inactivity_timeout = 10000,
|
|
|
|
},
|
2011-03-06 22:32:12 +08:00
|
|
|
.mem_wl127x = {
|
2011-02-02 15:59:36 +08:00
|
|
|
.num_stations = 1,
|
|
|
|
.ssid_profiles = 1,
|
|
|
|
.rx_block_num = 70,
|
|
|
|
.tx_min_block_num = 40,
|
2011-02-02 15:59:35 +08:00
|
|
|
.dynamic_memory = 0,
|
2011-03-01 21:14:44 +08:00
|
|
|
.min_req_tx_blocks = 100,
|
2011-02-02 15:59:35 +08:00
|
|
|
.min_req_rx_blocks = 22,
|
|
|
|
.tx_min = 27,
|
2011-03-06 22:32:12 +08:00
|
|
|
},
|
|
|
|
.mem_wl128x = {
|
|
|
|
.num_stations = 1,
|
|
|
|
.ssid_profiles = 1,
|
|
|
|
.rx_block_num = 40,
|
|
|
|
.tx_min_block_num = 40,
|
|
|
|
.dynamic_memory = 1,
|
|
|
|
.min_req_tx_blocks = 45,
|
|
|
|
.min_req_rx_blocks = 22,
|
|
|
|
.tx_min = 27,
|
|
|
|
},
|
2011-04-02 01:48:02 +08:00
|
|
|
.hci_io_ds = HCI_IO_DS_6MA,
|
2009-10-13 17:47:44 +08:00
|
|
|
};
|
|
|
|
|
2010-09-21 12:23:31 +08:00
|
|
|
static void __wl1271_op_remove_interface(struct wl1271 *wl);
|
2010-10-17 03:39:06 +08:00
|
|
|
static void wl1271_free_ap_keys(struct wl1271 *wl);
|
2010-09-21 12:23:31 +08:00
|
|
|
|
|
|
|
|
2010-03-18 18:26:31 +08:00
|
|
|
static void wl1271_device_release(struct device *dev)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct platform_device wl1271_device = {
|
|
|
|
.name = "wl1271",
|
|
|
|
.id = -1,
|
|
|
|
|
|
|
|
/* device model insists to have a release function */
|
|
|
|
.dev = {
|
|
|
|
.release = wl1271_device_release,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2011-03-21 16:43:36 +08:00
|
|
|
static DEFINE_MUTEX(wl_list_mutex);
|
2009-10-13 17:47:55 +08:00
|
|
|
static LIST_HEAD(wl_list);
|
|
|
|
|
2010-07-27 08:30:09 +08:00
|
|
|
static int wl1271_dev_notify(struct notifier_block *me, unsigned long what,
|
|
|
|
void *arg)
|
|
|
|
{
|
|
|
|
struct net_device *dev = arg;
|
|
|
|
struct wireless_dev *wdev;
|
|
|
|
struct wiphy *wiphy;
|
|
|
|
struct ieee80211_hw *hw;
|
|
|
|
struct wl1271 *wl;
|
|
|
|
struct wl1271 *wl_temp;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* Check that this notification is for us. */
|
|
|
|
if (what != NETDEV_CHANGE)
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
|
|
|
|
wdev = dev->ieee80211_ptr;
|
|
|
|
if (wdev == NULL)
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
|
|
|
|
wiphy = wdev->wiphy;
|
|
|
|
if (wiphy == NULL)
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
|
|
|
|
hw = wiphy_priv(wiphy);
|
|
|
|
if (hw == NULL)
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
|
|
|
|
wl_temp = hw->priv;
|
2011-03-21 16:43:36 +08:00
|
|
|
mutex_lock(&wl_list_mutex);
|
2010-07-27 08:30:09 +08:00
|
|
|
list_for_each_entry(wl, &wl_list, list) {
|
|
|
|
if (wl == wl_temp)
|
|
|
|
break;
|
|
|
|
}
|
2011-03-21 16:43:36 +08:00
|
|
|
mutex_unlock(&wl_list_mutex);
|
2010-07-27 08:30:09 +08:00
|
|
|
if (wl != wl_temp)
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
if (wl->state == WL1271_STATE_OFF)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (!test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
|
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2010-07-27 08:30:09 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if ((dev->operstate == IF_OPER_UP) &&
|
|
|
|
!test_and_set_bit(WL1271_FLAG_STA_STATE_SENT, &wl->flags)) {
|
|
|
|
wl1271_cmd_set_sta_state(wl);
|
|
|
|
wl1271_info("Association completed.");
|
|
|
|
}
|
|
|
|
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return NOTIFY_OK;
|
|
|
|
}
|
|
|
|
|
2010-11-10 18:27:19 +08:00
|
|
|
static int wl1271_reg_notify(struct wiphy *wiphy,
|
2010-11-26 19:44:59 +08:00
|
|
|
struct regulatory_request *request)
|
|
|
|
{
|
2010-11-10 18:27:19 +08:00
|
|
|
struct ieee80211_supported_band *band;
|
|
|
|
struct ieee80211_channel *ch;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
band = wiphy->bands[IEEE80211_BAND_5GHZ];
|
|
|
|
for (i = 0; i < band->n_channels; i++) {
|
|
|
|
ch = &band->channels[i];
|
|
|
|
if (ch->flags & IEEE80211_CHAN_DISABLED)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (ch->flags & IEEE80211_CHAN_RADAR)
|
|
|
|
ch->flags |= IEEE80211_CHAN_NO_IBSS |
|
|
|
|
IEEE80211_CHAN_PASSIVE_SCAN;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-13 17:47:44 +08:00
|
|
|
static void wl1271_conf_init(struct wl1271 *wl)
|
|
|
|
{
|
2009-10-13 17:47:39 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This function applies the default configuration to the driver. This
|
|
|
|
* function is invoked upon driver load (spi probe.)
|
|
|
|
*
|
|
|
|
* The configuration is stored in a run-time structure in order to
|
|
|
|
* facilitate for run-time adjustment of any of the parameters. Making
|
|
|
|
* changes to the configuration structure will apply the new values on
|
|
|
|
* the next interface up (wl1271_op_start.)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* apply driver default configuration */
|
2009-10-13 17:47:44 +08:00
|
|
|
memcpy(&wl->conf, &default_conf, sizeof(default_conf));
|
2009-10-13 17:47:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static int wl1271_plt_init(struct wl1271 *wl)
|
|
|
|
{
|
2010-02-18 19:25:44 +08:00
|
|
|
struct conf_tx_ac_category *conf_ac;
|
|
|
|
struct conf_tx_tid *conf_tid;
|
|
|
|
int ret, i;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2011-03-06 22:32:09 +08:00
|
|
|
if (wl->chip.id == CHIP_ID_1283_PG20)
|
|
|
|
ret = wl128x_cmd_general_parms(wl);
|
|
|
|
else
|
|
|
|
ret = wl1271_cmd_general_parms(wl);
|
2009-11-24 05:22:18 +08:00
|
|
|
if (ret < 0)
|
2009-11-24 05:22:16 +08:00
|
|
|
return ret;
|
|
|
|
|
2011-03-06 22:32:09 +08:00
|
|
|
if (wl->chip.id == CHIP_ID_1283_PG20)
|
|
|
|
ret = wl128x_cmd_radio_parms(wl);
|
|
|
|
else
|
|
|
|
ret = wl1271_cmd_radio_parms(wl);
|
2009-11-24 05:22:18 +08:00
|
|
|
if (ret < 0)
|
2009-11-24 05:22:16 +08:00
|
|
|
return ret;
|
|
|
|
|
2011-03-06 22:32:09 +08:00
|
|
|
if (wl->chip.id != CHIP_ID_1283_PG20) {
|
|
|
|
ret = wl1271_cmd_ext_radio_parms(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
2010-10-05 19:11:56 +08:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2011-03-06 22:32:08 +08:00
|
|
|
/* Chip-specific initializations */
|
|
|
|
ret = wl1271_chip_specific_init(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2010-10-17 00:19:53 +08:00
|
|
|
ret = wl1271_sta_init_templates_config(wl);
|
2010-02-18 19:25:44 +08:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
ret = wl1271_acx_init_mem_config(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2010-02-18 19:25:44 +08:00
|
|
|
/* PHY layer config */
|
|
|
|
ret = wl1271_init_phy_config(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
|
|
|
ret = wl1271_acx_dco_itrim_params(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
|
|
|
/* Initialize connection monitoring thresholds */
|
2010-03-26 18:53:23 +08:00
|
|
|
ret = wl1271_acx_conn_monit_params(wl, false);
|
2010-02-18 19:25:44 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
|
|
|
/* Bluetooth WLAN coexistence */
|
|
|
|
ret = wl1271_init_pta(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
|
|
|
/* Energy detection */
|
|
|
|
ret = wl1271_init_energy_detection(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
2011-02-01 17:03:08 +08:00
|
|
|
ret = wl1271_acx_sta_mem_cfg(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
2010-02-18 19:25:44 +08:00
|
|
|
/* Default fragmentation threshold */
|
2010-11-08 17:51:07 +08:00
|
|
|
ret = wl1271_acx_frag_threshold(wl, wl->conf.tx.frag_threshold);
|
2010-02-18 19:25:44 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
2010-09-01 17:31:12 +08:00
|
|
|
/* Default TID/AC configuration */
|
|
|
|
BUG_ON(wl->conf.tx.tid_conf_count != wl->conf.tx.ac_conf_count);
|
2010-02-18 19:25:44 +08:00
|
|
|
for (i = 0; i < wl->conf.tx.tid_conf_count; i++) {
|
2010-09-01 17:31:12 +08:00
|
|
|
conf_ac = &wl->conf.tx.ac_conf[i];
|
|
|
|
ret = wl1271_acx_ac_cfg(wl, conf_ac->ac, conf_ac->cw_min,
|
|
|
|
conf_ac->cw_max, conf_ac->aifsn,
|
|
|
|
conf_ac->tx_op_limit);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
2010-02-18 19:25:44 +08:00
|
|
|
conf_tid = &wl->conf.tx.tid_conf[i];
|
|
|
|
ret = wl1271_acx_tid_cfg(wl, conf_tid->queue_id,
|
|
|
|
conf_tid->channel_type,
|
|
|
|
conf_tid->tsid,
|
|
|
|
conf_tid->ps_scheme,
|
|
|
|
conf_tid->ack_policy,
|
|
|
|
conf_tid->apsd_conf[0],
|
|
|
|
conf_tid->apsd_conf[1]);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable data path */
|
2009-12-11 21:40:55 +08:00
|
|
|
ret = wl1271_cmd_data_path(wl, 1);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
2010-02-18 19:25:44 +08:00
|
|
|
goto out_free_memmap;
|
|
|
|
|
|
|
|
/* Configure for CAM power saving (ie. always active) */
|
|
|
|
ret = wl1271_acx_sleep_auth(wl, WL1271_PSM_CAM);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
|
|
|
|
|
|
|
/* configure PM */
|
|
|
|
ret = wl1271_acx_pm_config(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_free_memmap;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
return 0;
|
2010-02-18 19:25:44 +08:00
|
|
|
|
|
|
|
out_free_memmap:
|
|
|
|
kfree(wl->target_mem_map);
|
|
|
|
wl->target_mem_map = NULL;
|
|
|
|
|
|
|
|
return ret;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2011-02-23 06:22:31 +08:00
|
|
|
static void wl1271_irq_ps_regulate_link(struct wl1271 *wl, u8 hlid, u8 tx_blks)
|
|
|
|
{
|
|
|
|
bool fw_ps;
|
|
|
|
|
|
|
|
/* only regulate station links */
|
|
|
|
if (hlid < WL1271_AP_STA_HLID_START)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fw_ps = test_bit(hlid, (unsigned long *)&wl->ap_fw_ps_map);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wake up from high level PS if the STA is asleep with too little
|
|
|
|
* blocks in FW or if the STA is awake.
|
|
|
|
*/
|
|
|
|
if (!fw_ps || tx_blks < WL1271_PS_STA_MAX_BLOCKS)
|
|
|
|
wl1271_ps_link_end(wl, hlid);
|
|
|
|
|
|
|
|
/* Start high-level PS if the STA is asleep with enough blocks in FW */
|
|
|
|
else if (fw_ps && tx_blks >= WL1271_PS_STA_MAX_BLOCKS)
|
|
|
|
wl1271_ps_link_start(wl, hlid, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wl1271_irq_update_links_status(struct wl1271 *wl,
|
|
|
|
struct wl1271_fw_ap_status *status)
|
|
|
|
{
|
|
|
|
u32 cur_fw_ps_map;
|
|
|
|
u8 hlid;
|
|
|
|
|
|
|
|
cur_fw_ps_map = le32_to_cpu(status->link_ps_bitmap);
|
|
|
|
if (wl->ap_fw_ps_map != cur_fw_ps_map) {
|
|
|
|
wl1271_debug(DEBUG_PSM,
|
|
|
|
"link ps prev 0x%x cur 0x%x changed 0x%x",
|
|
|
|
wl->ap_fw_ps_map, cur_fw_ps_map,
|
|
|
|
wl->ap_fw_ps_map ^ cur_fw_ps_map);
|
|
|
|
|
|
|
|
wl->ap_fw_ps_map = cur_fw_ps_map;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (hlid = WL1271_AP_STA_HLID_START; hlid < AP_MAX_LINKS; hlid++) {
|
|
|
|
u8 cnt = status->tx_lnk_free_blks[hlid] -
|
|
|
|
wl->links[hlid].prev_freed_blks;
|
|
|
|
|
|
|
|
wl->links[hlid].prev_freed_blks =
|
|
|
|
status->tx_lnk_free_blks[hlid];
|
|
|
|
wl->links[hlid].allocated_blks -= cnt;
|
|
|
|
|
|
|
|
wl1271_irq_ps_regulate_link(wl, hlid,
|
|
|
|
wl->links[hlid].allocated_blks);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-12 20:08:50 +08:00
|
|
|
static void wl1271_fw_status(struct wl1271 *wl,
|
2011-02-02 15:59:35 +08:00
|
|
|
struct wl1271_fw_full_status *full_status)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
2011-02-02 15:59:35 +08:00
|
|
|
struct wl1271_fw_common_status *status = &full_status->common;
|
2010-02-22 14:38:38 +08:00
|
|
|
struct timespec ts;
|
2011-03-06 22:32:12 +08:00
|
|
|
u32 old_tx_blk_count = wl->tx_blocks_available;
|
2011-03-31 16:07:00 +08:00
|
|
|
u32 freed_blocks = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
int i;
|
|
|
|
|
2011-03-06 22:32:12 +08:00
|
|
|
if (wl->bss_type == BSS_TYPE_AP_BSS) {
|
2011-02-02 15:59:35 +08:00
|
|
|
wl1271_raw_read(wl, FW_STATUS_ADDR, status,
|
|
|
|
sizeof(struct wl1271_fw_ap_status), false);
|
2011-03-06 22:32:12 +08:00
|
|
|
} else {
|
2011-02-02 15:59:35 +08:00
|
|
|
wl1271_raw_read(wl, FW_STATUS_ADDR, status,
|
|
|
|
sizeof(struct wl1271_fw_sta_status), false);
|
2011-03-06 22:32:12 +08:00
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
wl1271_debug(DEBUG_IRQ, "intr: 0x%x (fw_rx_counter = %d, "
|
|
|
|
"drv_rx_counter = %d, tx_results_counter = %d)",
|
|
|
|
status->intr,
|
|
|
|
status->fw_rx_counter,
|
|
|
|
status->drv_rx_counter,
|
|
|
|
status->tx_results_counter);
|
|
|
|
|
|
|
|
/* update number of available TX blocks */
|
|
|
|
for (i = 0; i < NUM_TX_QUEUES; i++) {
|
2011-03-31 16:07:00 +08:00
|
|
|
freed_blocks += le32_to_cpu(status->tx_released_blks[i]) -
|
|
|
|
wl->tx_blocks_freed[i];
|
2009-10-15 15:33:29 +08:00
|
|
|
|
|
|
|
wl->tx_blocks_freed[i] =
|
|
|
|
le32_to_cpu(status->tx_released_blks[i]);
|
2011-03-06 22:32:12 +08:00
|
|
|
}
|
|
|
|
|
2011-03-31 16:07:00 +08:00
|
|
|
wl->tx_allocated_blocks -= freed_blocks;
|
|
|
|
|
|
|
|
if (wl->bss_type == BSS_TYPE_AP_BSS) {
|
|
|
|
/* Update num of allocated TX blocks per link and ps status */
|
|
|
|
wl1271_irq_update_links_status(wl, &full_status->ap);
|
|
|
|
wl->tx_blocks_available += freed_blocks;
|
|
|
|
} else {
|
|
|
|
int avail = full_status->sta.tx_total - wl->tx_allocated_blocks;
|
2011-03-06 22:32:12 +08:00
|
|
|
|
2011-03-31 16:07:00 +08:00
|
|
|
/*
|
|
|
|
* The FW might change the total number of TX memblocks before
|
|
|
|
* we get a notification about blocks being released. Thus, the
|
|
|
|
* available blocks calculation might yield a temporary result
|
|
|
|
* which is lower than the actual available blocks. Keeping in
|
|
|
|
* mind that only blocks that were allocated can be moved from
|
|
|
|
* TX to RX, tx_blocks_available should never decrease here.
|
|
|
|
*/
|
|
|
|
wl->tx_blocks_available = max((int)wl->tx_blocks_available,
|
|
|
|
avail);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2010-10-12 20:49:10 +08:00
|
|
|
/* if more blocks are available now, tx work can be scheduled */
|
2011-03-06 22:32:12 +08:00
|
|
|
if (wl->tx_blocks_available > old_tx_blk_count)
|
2010-10-12 20:49:10 +08:00
|
|
|
clear_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/* update the host-chipset time offset */
|
2010-02-22 14:38:38 +08:00
|
|
|
getnstimeofday(&ts);
|
|
|
|
wl->time_offset = (timespec_to_ns(&ts) >> 10) -
|
|
|
|
(s64)le32_to_cpu(status->fw_localtime);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
static void wl1271_flush_deferred_work(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
struct sk_buff *skb;
|
|
|
|
|
|
|
|
/* Pass all received frames to the network stack */
|
|
|
|
while ((skb = skb_dequeue(&wl->deferred_rx_queue)))
|
|
|
|
ieee80211_rx_ni(wl->hw, skb);
|
|
|
|
|
|
|
|
/* Return sent skbs to the network stack */
|
|
|
|
while ((skb = skb_dequeue(&wl->deferred_tx_queue)))
|
|
|
|
ieee80211_tx_status(wl->hw, skb);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wl1271_netstack_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl =
|
|
|
|
container_of(work, struct wl1271, netstack_work);
|
|
|
|
|
|
|
|
do {
|
|
|
|
wl1271_flush_deferred_work(wl);
|
|
|
|
} while (skb_queue_len(&wl->deferred_rx_queue));
|
|
|
|
}
|
2010-02-22 14:38:37 +08:00
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
#define WL1271_IRQ_MAX_LOOPS 256
|
|
|
|
|
|
|
|
irqreturn_t wl1271_irq(int irq, void *cookie)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
|
|
|
int ret;
|
2009-10-12 20:08:50 +08:00
|
|
|
u32 intr;
|
2010-02-22 14:38:37 +08:00
|
|
|
int loopcount = WL1271_IRQ_MAX_LOOPS;
|
2011-03-01 21:14:41 +08:00
|
|
|
struct wl1271 *wl = (struct wl1271 *)cookie;
|
|
|
|
bool done = false;
|
|
|
|
unsigned int defer_count;
|
2011-03-01 21:14:43 +08:00
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
/* TX might be handled here, avoid redundant work */
|
|
|
|
set_bit(WL1271_FLAG_TX_PENDING, &wl->flags);
|
|
|
|
cancel_work_sync(&wl->tx_work);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2011-03-31 16:07:01 +08:00
|
|
|
/*
|
|
|
|
* In case edge triggered interrupt must be used, we cannot iterate
|
|
|
|
* more than once without introducing race conditions with the hardirq.
|
|
|
|
*/
|
|
|
|
if (wl->platform_quirks & WL12XX_PLATFORM_QUIRK_EDGE_IRQ)
|
|
|
|
loopcount = 1;
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_IRQ, "IRQ work");
|
|
|
|
|
2010-02-22 14:38:37 +08:00
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF))
|
2009-08-06 21:25:28 +08:00
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
while (!done && loopcount--) {
|
|
|
|
/*
|
|
|
|
* In order to avoid a race with the hardirq, clear the flag
|
|
|
|
* before acknowledging the chip. Since the mutex is held,
|
|
|
|
* wl1271_ps_elp_wakeup cannot be called concurrently.
|
|
|
|
*/
|
|
|
|
clear_bit(WL1271_FLAG_IRQ_RUNNING, &wl->flags);
|
|
|
|
smp_mb__after_clear_bit();
|
2010-02-22 14:38:37 +08:00
|
|
|
|
|
|
|
wl1271_fw_status(wl, wl->fw_status);
|
2011-02-02 15:59:35 +08:00
|
|
|
intr = le32_to_cpu(wl->fw_status->common.intr);
|
2011-03-01 21:14:41 +08:00
|
|
|
intr &= WL1271_INTR_MASK;
|
2010-02-22 14:38:37 +08:00
|
|
|
if (!intr) {
|
2011-03-01 21:14:41 +08:00
|
|
|
done = true;
|
2010-02-22 14:38:37 +08:00
|
|
|
continue;
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-27 20:09:57 +08:00
|
|
|
if (unlikely(intr & WL1271_ACX_INTR_WATCHDOG)) {
|
|
|
|
wl1271_error("watchdog interrupt received! "
|
|
|
|
"starting recovery.");
|
|
|
|
ieee80211_queue_work(wl->hw, &wl->recovery_work);
|
|
|
|
|
|
|
|
/* restarting the chip. ignore any other interrupt. */
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
if (likely(intr & WL1271_ACX_INTR_DATA)) {
|
2010-02-22 14:38:37 +08:00
|
|
|
wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_DATA");
|
2009-10-13 17:47:54 +08:00
|
|
|
|
2011-03-01 21:14:38 +08:00
|
|
|
wl1271_rx(wl, &wl->fw_status->common);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-12 20:49:10 +08:00
|
|
|
/* Check if any tx blocks were freed */
|
2011-03-01 21:14:43 +08:00
|
|
|
spin_lock_irqsave(&wl->wl_lock, flags);
|
2010-10-12 20:49:10 +08:00
|
|
|
if (!test_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags) &&
|
2010-12-13 15:52:37 +08:00
|
|
|
wl->tx_queue_count) {
|
2011-03-01 21:14:43 +08:00
|
|
|
spin_unlock_irqrestore(&wl->wl_lock, flags);
|
2010-10-12 20:49:10 +08:00
|
|
|
/*
|
|
|
|
* In order to avoid starvation of the TX path,
|
|
|
|
* call the work function directly.
|
|
|
|
*/
|
|
|
|
wl1271_tx_work_locked(wl);
|
2011-03-01 21:14:43 +08:00
|
|
|
} else {
|
|
|
|
spin_unlock_irqrestore(&wl->wl_lock, flags);
|
2010-10-12 20:49:10 +08:00
|
|
|
}
|
|
|
|
|
2011-03-01 21:14:38 +08:00
|
|
|
/* check for tx results */
|
|
|
|
if (wl->fw_status->common.tx_results_counter !=
|
|
|
|
(wl->tx_results_count & 0xff))
|
|
|
|
wl1271_tx_complete(wl);
|
2011-03-01 21:14:41 +08:00
|
|
|
|
|
|
|
/* Make sure the deferred queues don't get too long */
|
|
|
|
defer_count = skb_queue_len(&wl->deferred_tx_queue) +
|
|
|
|
skb_queue_len(&wl->deferred_rx_queue);
|
|
|
|
if (defer_count > WL1271_DEFERRED_QUEUE_LIMIT)
|
|
|
|
wl1271_flush_deferred_work(wl);
|
2010-02-22 14:38:37 +08:00
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-02-22 14:38:37 +08:00
|
|
|
if (intr & WL1271_ACX_INTR_EVENT_A) {
|
|
|
|
wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_EVENT_A");
|
|
|
|
wl1271_event_handle(wl, 0);
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-02-22 14:38:37 +08:00
|
|
|
if (intr & WL1271_ACX_INTR_EVENT_B) {
|
|
|
|
wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_EVENT_B");
|
|
|
|
wl1271_event_handle(wl, 1);
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-02-22 14:38:37 +08:00
|
|
|
if (intr & WL1271_ACX_INTR_INIT_COMPLETE)
|
|
|
|
wl1271_debug(DEBUG_IRQ,
|
|
|
|
"WL1271_ACX_INTR_INIT_COMPLETE");
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-02-22 14:38:37 +08:00
|
|
|
if (intr & WL1271_ACX_INTR_HW_AVAILABLE)
|
|
|
|
wl1271_debug(DEBUG_IRQ, "WL1271_ACX_INTR_HW_AVAILABLE");
|
2009-10-12 20:08:50 +08:00
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
2011-03-01 21:14:43 +08:00
|
|
|
spin_lock_irqsave(&wl->wl_lock, flags);
|
|
|
|
/* In case TX was not handled here, queue TX work */
|
|
|
|
clear_bit(WL1271_FLAG_TX_PENDING, &wl->flags);
|
|
|
|
if (!test_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags) &&
|
|
|
|
wl->tx_queue_count)
|
|
|
|
ieee80211_queue_work(wl->hw, &wl->tx_work);
|
|
|
|
spin_unlock_irqrestore(&wl->wl_lock, flags);
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
2011-03-01 21:14:41 +08:00
|
|
|
|
|
|
|
return IRQ_HANDLED;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
2011-03-01 21:14:41 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1271_irq);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
static int wl1271_fetch_firmware(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
const struct firmware *fw;
|
2010-10-17 03:44:57 +08:00
|
|
|
const char *fw_name;
|
2009-08-06 21:25:28 +08:00
|
|
|
int ret;
|
|
|
|
|
2010-10-17 03:44:57 +08:00
|
|
|
switch (wl->bss_type) {
|
|
|
|
case BSS_TYPE_AP_BSS:
|
2011-03-06 22:32:18 +08:00
|
|
|
if (wl->chip.id == CHIP_ID_1283_PG20)
|
|
|
|
fw_name = WL128X_AP_FW_NAME;
|
|
|
|
else
|
|
|
|
fw_name = WL127X_AP_FW_NAME;
|
2010-10-17 03:44:57 +08:00
|
|
|
break;
|
|
|
|
case BSS_TYPE_IBSS:
|
|
|
|
case BSS_TYPE_STA_BSS:
|
2011-03-06 22:32:10 +08:00
|
|
|
if (wl->chip.id == CHIP_ID_1283_PG20)
|
|
|
|
fw_name = WL128X_FW_NAME;
|
|
|
|
else
|
|
|
|
fw_name = WL1271_FW_NAME;
|
2010-10-17 03:44:57 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
wl1271_error("no compatible firmware for bss_type %d",
|
|
|
|
wl->bss_type);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_BOOT, "booting firmware %s", fw_name);
|
|
|
|
|
|
|
|
ret = request_firmware(&fw, fw_name, wl1271_wl_to_dev(wl));
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("could not get firmware: %d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fw->size % 4) {
|
|
|
|
wl1271_error("firmware size is not multiple of 32 bits: %zu",
|
|
|
|
fw->size);
|
|
|
|
ret = -EILSEQ;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2010-10-17 03:44:57 +08:00
|
|
|
vfree(wl->fw);
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->fw_len = fw->size;
|
2009-10-09 02:56:32 +08:00
|
|
|
wl->fw = vmalloc(wl->fw_len);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
if (!wl->fw) {
|
|
|
|
wl1271_error("could not allocate memory for the firmware");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(wl->fw, fw->data, wl->fw_len);
|
2010-10-17 03:44:57 +08:00
|
|
|
wl->fw_bss_type = wl->bss_type;
|
2009-08-06 21:25:28 +08:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
release_firmware(fw);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_fetch_nvs(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
const struct firmware *fw;
|
|
|
|
int ret;
|
|
|
|
|
2011-03-06 22:32:07 +08:00
|
|
|
ret = request_firmware(&fw, WL12XX_NVS_NAME, wl1271_wl_to_dev(wl));
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("could not get nvs file: %d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-03-06 22:32:10 +08:00
|
|
|
wl->nvs = kmemdup(fw->data, fw->size, GFP_KERNEL);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
if (!wl->nvs) {
|
|
|
|
wl1271_error("could not allocate memory for the nvs file");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2010-08-19 10:41:15 +08:00
|
|
|
wl->nvs_len = fw->size;
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
out:
|
|
|
|
release_firmware(fw);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-09-21 12:23:31 +08:00
|
|
|
static void wl1271_recovery_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl =
|
|
|
|
container_of(work, struct wl1271, recovery_work);
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
if (wl->state != WL1271_STATE_ON)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
wl1271_info("Hardware recovery in progress.");
|
|
|
|
|
2010-09-30 16:43:27 +08:00
|
|
|
if (test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
|
|
|
|
ieee80211_connection_loss(wl->vif);
|
|
|
|
|
2010-09-21 12:23:31 +08:00
|
|
|
/* reboot the chipset */
|
|
|
|
__wl1271_op_remove_interface(wl);
|
|
|
|
ieee80211_restart_hw(wl->hw);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static void wl1271_fw_wakeup(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
u32 elp_reg;
|
|
|
|
|
|
|
|
elp_reg = ELPCTRL_WAKE_UP;
|
2009-10-12 20:08:54 +08:00
|
|
|
wl1271_raw_write32(wl, HW_ACCESS_ELP_CTRL_REG_ADDR, elp_reg);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_setup(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
wl->fw_status = kmalloc(sizeof(*wl->fw_status), GFP_KERNEL);
|
|
|
|
if (!wl->fw_status)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
wl->tx_res_if = kmalloc(sizeof(*wl->tx_res_if), GFP_KERNEL);
|
|
|
|
if (!wl->tx_res_if) {
|
|
|
|
kfree(wl->fw_status);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_chip_wakeup(struct wl1271 *wl)
|
|
|
|
{
|
2009-10-12 20:08:46 +08:00
|
|
|
struct wl1271_partition_set partition;
|
2009-08-06 21:25:28 +08:00
|
|
|
int ret = 0;
|
|
|
|
|
2009-12-11 21:41:02 +08:00
|
|
|
msleep(WL1271_PRE_POWER_ON_SLEEP);
|
2010-09-16 07:22:04 +08:00
|
|
|
ret = wl1271_power_on(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
msleep(WL1271_POWER_ON_SLEEP);
|
2010-02-18 19:25:56 +08:00
|
|
|
wl1271_io_reset(wl);
|
|
|
|
wl1271_io_init(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/* We don't need a real memory partition here, because we only want
|
|
|
|
* to use the registers at this point. */
|
2009-10-12 20:08:46 +08:00
|
|
|
memset(&partition, 0, sizeof(partition));
|
|
|
|
partition.reg.start = REGISTERS_BASE;
|
|
|
|
partition.reg.size = REGISTERS_DOWN_SIZE;
|
|
|
|
wl1271_set_partition(wl, &partition);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/* ELP module wake up */
|
|
|
|
wl1271_fw_wakeup(wl);
|
|
|
|
|
|
|
|
/* whal_FwCtrl_BootSm() */
|
|
|
|
|
|
|
|
/* 0. read chip id from CHIP_ID */
|
2010-02-18 19:25:55 +08:00
|
|
|
wl->chip.id = wl1271_read32(wl, CHIP_ID_B);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/* 1. check if chip id is valid */
|
|
|
|
|
|
|
|
switch (wl->chip.id) {
|
|
|
|
case CHIP_ID_1271_PG10:
|
|
|
|
wl1271_warning("chip id 0x%x (1271 PG10) support is obsolete",
|
|
|
|
wl->chip.id);
|
|
|
|
|
|
|
|
ret = wl1271_setup(wl);
|
|
|
|
if (ret < 0)
|
2009-12-11 21:41:01 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
break;
|
|
|
|
case CHIP_ID_1271_PG20:
|
|
|
|
wl1271_debug(DEBUG_BOOT, "chip id 0x%x (1271 PG20)",
|
|
|
|
wl->chip.id);
|
|
|
|
|
|
|
|
ret = wl1271_setup(wl);
|
|
|
|
if (ret < 0)
|
2009-12-11 21:41:01 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
break;
|
2011-03-06 22:32:20 +08:00
|
|
|
case CHIP_ID_1283_PG20:
|
|
|
|
wl1271_debug(DEBUG_BOOT, "chip id 0x%x (1283 PG20)",
|
|
|
|
wl->chip.id);
|
|
|
|
|
|
|
|
ret = wl1271_setup(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2011-03-31 16:06:58 +08:00
|
|
|
if (wl1271_set_block_size(wl))
|
|
|
|
wl->quirks |= WL12XX_QUIRK_BLOCKSIZE_ALIGNMENT;
|
2011-03-06 22:32:20 +08:00
|
|
|
break;
|
|
|
|
case CHIP_ID_1283_PG10:
|
2009-08-06 21:25:28 +08:00
|
|
|
default:
|
2009-12-11 21:41:01 +08:00
|
|
|
wl1271_warning("unsupported chip id: 0x%x", wl->chip.id);
|
2009-08-06 21:25:28 +08:00
|
|
|
ret = -ENODEV;
|
2009-12-11 21:41:01 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2010-10-17 03:44:57 +08:00
|
|
|
/* Make sure the firmware type matches the BSS type */
|
|
|
|
if (wl->fw == NULL || wl->fw_bss_type != wl->bss_type) {
|
2009-08-06 21:25:28 +08:00
|
|
|
ret = wl1271_fetch_firmware(wl);
|
|
|
|
if (ret < 0)
|
2009-12-11 21:41:01 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* No NVS from netlink, try to get it from the filesystem */
|
|
|
|
if (wl->nvs == NULL) {
|
|
|
|
ret = wl1271_fetch_nvs(wl);
|
|
|
|
if (ret < 0)
|
2009-12-11 21:41:01 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-03-10 21:24:57 +08:00
|
|
|
static unsigned int wl1271_get_fw_ver_quirks(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
unsigned int quirks = 0;
|
|
|
|
unsigned int *fw_ver = wl->chip.fw_ver;
|
|
|
|
|
|
|
|
/* Only for wl127x */
|
|
|
|
if ((fw_ver[FW_VER_CHIP] == FW_VER_CHIP_WL127X) &&
|
|
|
|
/* Check STA version */
|
|
|
|
(((fw_ver[FW_VER_IF_TYPE] == FW_VER_IF_TYPE_STA) &&
|
|
|
|
(fw_ver[FW_VER_MINOR] < FW_VER_MINOR_1_SPARE_STA_MIN)) ||
|
|
|
|
/* Check AP version */
|
|
|
|
((fw_ver[FW_VER_IF_TYPE] == FW_VER_IF_TYPE_AP) &&
|
|
|
|
(fw_ver[FW_VER_MINOR] < FW_VER_MINOR_1_SPARE_AP_MIN))))
|
|
|
|
quirks |= WL12XX_QUIRK_USE_2_SPARE_BLOCKS;
|
|
|
|
|
|
|
|
return quirks;
|
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
int wl1271_plt_start(struct wl1271 *wl)
|
|
|
|
{
|
2009-12-11 21:41:01 +08:00
|
|
|
int retries = WL1271_BOOT_RETRIES;
|
2009-08-06 21:25:28 +08:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
wl1271_notice("power up");
|
|
|
|
|
|
|
|
if (wl->state != WL1271_STATE_OFF) {
|
|
|
|
wl1271_error("cannot go into PLT state because not "
|
|
|
|
"in off state: %d", wl->state);
|
|
|
|
ret = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2010-10-17 03:44:57 +08:00
|
|
|
wl->bss_type = BSS_TYPE_STA_BSS;
|
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
while (retries) {
|
|
|
|
retries--;
|
|
|
|
ret = wl1271_chip_wakeup(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto power_off;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
ret = wl1271_boot(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto power_off;
|
2009-10-13 17:47:45 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
ret = wl1271_plt_init(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto irq_disable;
|
2009-10-13 17:47:58 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
wl->state = WL1271_STATE_PLT;
|
|
|
|
wl1271_notice("firmware booted in PLT mode (%s)",
|
2011-01-23 14:27:22 +08:00
|
|
|
wl->chip.fw_ver_str);
|
2011-03-10 21:24:57 +08:00
|
|
|
|
|
|
|
/* Check if any quirks are needed with older fw versions */
|
|
|
|
wl->quirks |= wl1271_get_fw_ver_quirks(wl);
|
2009-12-11 21:41:01 +08:00
|
|
|
goto out;
|
2009-10-13 17:47:45 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
irq_disable:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
/* Unlocking the mutex in the middle of handling is
|
|
|
|
inherently unsafe. In this case we deem it safe to do,
|
|
|
|
because we need to let any possibly pending IRQ out of
|
|
|
|
the system (and while we are WL1271_STATE_OFF the IRQ
|
|
|
|
work function will not do anything.) Also, any other
|
|
|
|
possible concurrent operations will fail due to the
|
|
|
|
current state, hence the wl1271 struct should be safe. */
|
2011-03-01 21:14:41 +08:00
|
|
|
wl1271_disable_interrupts(wl);
|
|
|
|
wl1271_flush_deferred_work(wl);
|
|
|
|
cancel_work_sync(&wl->netstack_work);
|
2009-12-11 21:41:01 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
power_off:
|
|
|
|
wl1271_power_off(wl);
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
wl1271_error("firmware boot in PLT mode failed despite %d retries",
|
|
|
|
WL1271_BOOT_RETRIES);
|
2009-08-06 21:25:28 +08:00
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-03-22 01:26:41 +08:00
|
|
|
static int __wl1271_plt_stop(struct wl1271 *wl)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
wl1271_notice("power down");
|
|
|
|
|
|
|
|
if (wl->state != WL1271_STATE_PLT) {
|
|
|
|
wl1271_error("cannot power down because not in PLT "
|
|
|
|
"state: %d", wl->state);
|
|
|
|
ret = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl1271_power_off(wl);
|
|
|
|
|
|
|
|
wl->state = WL1271_STATE_OFF;
|
2009-10-13 17:47:58 +08:00
|
|
|
wl->rx_counter = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
mutex_unlock(&wl->mutex);
|
2011-03-01 21:14:41 +08:00
|
|
|
wl1271_disable_interrupts(wl);
|
|
|
|
wl1271_flush_deferred_work(wl);
|
|
|
|
cancel_work_sync(&wl->netstack_work);
|
2010-09-21 12:23:31 +08:00
|
|
|
cancel_work_sync(&wl->recovery_work);
|
2011-01-14 19:48:46 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int wl1271_plt_stop(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
int ret;
|
2010-09-21 12:23:29 +08:00
|
|
|
|
2011-01-14 19:48:46 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
ret = __wl1271_plt_stop(wl);
|
|
|
|
mutex_unlock(&wl->mutex);
|
2009-08-06 21:25:28 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-02-24 21:42:06 +08:00
|
|
|
static void wl1271_op_tx(struct ieee80211_hw *hw, struct sk_buff *skb)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
2009-12-11 21:41:06 +08:00
|
|
|
unsigned long flags;
|
2010-12-13 15:52:37 +08:00
|
|
|
int q;
|
2011-02-23 06:22:26 +08:00
|
|
|
u8 hlid = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2011-03-01 21:14:43 +08:00
|
|
|
q = wl1271_tx_get_queue(skb_get_queue_mapping(skb));
|
|
|
|
|
|
|
|
if (wl->bss_type == BSS_TYPE_AP_BSS)
|
|
|
|
hlid = wl1271_tx_get_hlid(skb);
|
|
|
|
|
2009-12-11 21:41:06 +08:00
|
|
|
spin_lock_irqsave(&wl->wl_lock, flags);
|
2011-03-01 21:14:43 +08:00
|
|
|
|
2010-12-13 15:52:37 +08:00
|
|
|
wl->tx_queue_count++;
|
2011-02-23 06:22:24 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The workqueue is slow to process the tx_queue and we need stop
|
|
|
|
* the queue here, otherwise the queue will get too long.
|
|
|
|
*/
|
|
|
|
if (wl->tx_queue_count >= WL1271_TX_QUEUE_HIGH_WATERMARK) {
|
|
|
|
wl1271_debug(DEBUG_TX, "op_tx: stopping queues");
|
|
|
|
ieee80211_stop_queues(wl->hw);
|
|
|
|
set_bit(WL1271_FLAG_TX_QUEUE_STOPPED, &wl->flags);
|
|
|
|
}
|
|
|
|
|
2009-12-11 21:41:06 +08:00
|
|
|
/* queue the packet */
|
2011-02-23 06:22:26 +08:00
|
|
|
if (wl->bss_type == BSS_TYPE_AP_BSS) {
|
|
|
|
wl1271_debug(DEBUG_TX, "queue skb hlid %d q %d", hlid, q);
|
|
|
|
skb_queue_tail(&wl->links[hlid].tx_queue[q], skb);
|
|
|
|
} else {
|
|
|
|
skb_queue_tail(&wl->tx_queue[q], skb);
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The chip specific setup must run before the first TX packet -
|
|
|
|
* before that, the tx_work will not be initialized!
|
|
|
|
*/
|
|
|
|
|
2011-03-01 21:14:43 +08:00
|
|
|
if (!test_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags) &&
|
|
|
|
!test_bit(WL1271_FLAG_TX_PENDING, &wl->flags))
|
2010-10-12 20:49:10 +08:00
|
|
|
ieee80211_queue_work(wl->hw, &wl->tx_work);
|
2011-03-01 21:14:43 +08:00
|
|
|
|
|
|
|
spin_unlock_irqrestore(&wl->wl_lock, flags);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2011-03-06 22:32:14 +08:00
|
|
|
int wl1271_tx_dummy_packet(struct wl1271 *wl)
|
|
|
|
{
|
2011-03-31 16:06:59 +08:00
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&wl->wl_lock, flags);
|
|
|
|
set_bit(WL1271_FLAG_DUMMY_PACKET_PENDING, &wl->flags);
|
|
|
|
wl->tx_queue_count++;
|
|
|
|
spin_unlock_irqrestore(&wl->wl_lock, flags);
|
|
|
|
|
|
|
|
/* The FW is low on RX memory blocks, so send the dummy packet asap */
|
|
|
|
if (!test_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags))
|
|
|
|
wl1271_tx_work_locked(wl);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the FW TX is busy, TX work will be scheduled by the threaded
|
|
|
|
* interrupt handler function
|
|
|
|
*/
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The size of the dummy packet should be at least 1400 bytes. However, in
|
|
|
|
* order to minimize the number of bus transactions, aligning it to 512 bytes
|
|
|
|
* boundaries could be beneficial, performance wise
|
|
|
|
*/
|
|
|
|
#define TOTAL_TX_DUMMY_PACKET_SIZE (ALIGN(1400, 512))
|
|
|
|
|
2011-04-02 02:08:23 +08:00
|
|
|
static struct sk_buff *wl12xx_alloc_dummy_packet(struct wl1271 *wl)
|
2011-03-31 16:06:59 +08:00
|
|
|
{
|
|
|
|
struct sk_buff *skb;
|
2011-03-06 22:32:14 +08:00
|
|
|
struct ieee80211_hdr_3addr *hdr;
|
2011-03-31 16:06:59 +08:00
|
|
|
unsigned int dummy_packet_size;
|
|
|
|
|
|
|
|
dummy_packet_size = TOTAL_TX_DUMMY_PACKET_SIZE -
|
|
|
|
sizeof(struct wl1271_tx_hw_descr) - sizeof(*hdr);
|
2011-03-06 22:32:14 +08:00
|
|
|
|
2011-03-31 16:06:59 +08:00
|
|
|
skb = dev_alloc_skb(TOTAL_TX_DUMMY_PACKET_SIZE);
|
2011-03-06 22:32:14 +08:00
|
|
|
if (!skb) {
|
2011-03-31 16:06:59 +08:00
|
|
|
wl1271_warning("Failed to allocate a dummy packet skb");
|
|
|
|
return NULL;
|
2011-03-06 22:32:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
skb_reserve(skb, sizeof(struct wl1271_tx_hw_descr));
|
|
|
|
|
|
|
|
hdr = (struct ieee80211_hdr_3addr *) skb_put(skb, sizeof(*hdr));
|
|
|
|
memset(hdr, 0, sizeof(*hdr));
|
|
|
|
hdr->frame_control = cpu_to_le16(IEEE80211_FTYPE_DATA |
|
2011-03-31 16:06:59 +08:00
|
|
|
IEEE80211_STYPE_NULLFUNC |
|
|
|
|
IEEE80211_FCTL_TODS);
|
2011-03-06 22:32:14 +08:00
|
|
|
|
2011-03-31 16:06:59 +08:00
|
|
|
memset(skb_put(skb, dummy_packet_size), 0, dummy_packet_size);
|
2011-03-06 22:32:14 +08:00
|
|
|
|
2011-03-21 22:35:21 +08:00
|
|
|
/* Dummy packets require the TID to be management */
|
|
|
|
skb->priority = WL1271_TID_MGMT;
|
2011-03-06 22:32:14 +08:00
|
|
|
|
2011-03-31 16:06:59 +08:00
|
|
|
/* Initialize all fields that might be used */
|
|
|
|
skb->queue_mapping = 0;
|
|
|
|
memset(IEEE80211_SKB_CB(skb), 0, sizeof(struct ieee80211_tx_info));
|
2011-03-06 22:32:14 +08:00
|
|
|
|
2011-03-31 16:06:59 +08:00
|
|
|
return skb;
|
2011-03-06 22:32:14 +08:00
|
|
|
}
|
|
|
|
|
2011-03-31 16:06:59 +08:00
|
|
|
|
2010-07-27 08:30:09 +08:00
|
|
|
static struct notifier_block wl1271_dev_notifier = {
|
|
|
|
.notifier_call = wl1271_dev_notify,
|
|
|
|
};
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static int wl1271_op_start(struct ieee80211_hw *hw)
|
2010-03-18 18:26:39 +08:00
|
|
|
{
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 start");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We have to delay the booting of the hardware because
|
|
|
|
* we need to know the local MAC address before downloading and
|
|
|
|
* initializing the firmware. The MAC address cannot be changed
|
|
|
|
* after boot, and without the proper MAC address, the firmware
|
|
|
|
* will not function properly.
|
|
|
|
*
|
|
|
|
* The MAC address is first known when the corresponding interface
|
|
|
|
* is added. That is where we will initialize the hardware.
|
2010-10-17 03:44:57 +08:00
|
|
|
*
|
|
|
|
* In addition, we currently have different firmwares for AP and managed
|
|
|
|
* operation. We will know which to boot according to interface type.
|
2010-03-18 18:26:39 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wl1271_op_stop(struct ieee80211_hw *hw)
|
|
|
|
{
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 stop");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_op_add_interface(struct ieee80211_hw *hw,
|
|
|
|
struct ieee80211_vif *vif)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
2010-07-29 05:09:41 +08:00
|
|
|
struct wiphy *wiphy = hw->wiphy;
|
2009-12-11 21:41:01 +08:00
|
|
|
int retries = WL1271_BOOT_RETRIES;
|
2009-08-06 21:25:28 +08:00
|
|
|
int ret = 0;
|
2010-10-29 03:46:43 +08:00
|
|
|
bool booted = false;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-03-18 18:26:39 +08:00
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 add interface type %d mac %pM",
|
|
|
|
vif->type, vif->addr);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
2010-03-18 18:26:39 +08:00
|
|
|
if (wl->vif) {
|
2010-10-29 03:46:43 +08:00
|
|
|
wl1271_debug(DEBUG_MAC80211,
|
|
|
|
"multiple vifs are not supported yet");
|
2010-03-18 18:26:39 +08:00
|
|
|
ret = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2011-03-29 21:43:50 +08:00
|
|
|
/*
|
|
|
|
* in some very corner case HW recovery scenarios its possible to
|
|
|
|
* get here before __wl1271_op_remove_interface is complete, so
|
|
|
|
* opt out if that is the case.
|
|
|
|
*/
|
|
|
|
if (test_bit(WL1271_FLAG_IF_INITIALIZED, &wl->flags)) {
|
|
|
|
ret = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2010-03-18 18:26:39 +08:00
|
|
|
switch (vif->type) {
|
|
|
|
case NL80211_IFTYPE_STATION:
|
|
|
|
wl->bss_type = BSS_TYPE_STA_BSS;
|
2010-03-26 18:53:24 +08:00
|
|
|
wl->set_bss_type = BSS_TYPE_STA_BSS;
|
2010-03-18 18:26:39 +08:00
|
|
|
break;
|
|
|
|
case NL80211_IFTYPE_ADHOC:
|
|
|
|
wl->bss_type = BSS_TYPE_IBSS;
|
2010-03-26 18:53:24 +08:00
|
|
|
wl->set_bss_type = BSS_TYPE_STA_BSS;
|
2010-03-18 18:26:39 +08:00
|
|
|
break;
|
2010-10-17 03:53:24 +08:00
|
|
|
case NL80211_IFTYPE_AP:
|
|
|
|
wl->bss_type = BSS_TYPE_AP_BSS;
|
|
|
|
break;
|
2010-03-18 18:26:39 +08:00
|
|
|
default:
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(wl->mac_addr, vif->addr, ETH_ALEN);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
if (wl->state != WL1271_STATE_OFF) {
|
|
|
|
wl1271_error("cannot start because not in off state: %d",
|
|
|
|
wl->state);
|
|
|
|
ret = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
while (retries) {
|
|
|
|
retries--;
|
|
|
|
ret = wl1271_chip_wakeup(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto power_off;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
ret = wl1271_boot(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto power_off;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
ret = wl1271_hw_init(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto irq_disable;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-29 03:46:43 +08:00
|
|
|
booted = true;
|
|
|
|
break;
|
2009-10-13 17:47:45 +08:00
|
|
|
|
2009-12-11 21:41:01 +08:00
|
|
|
irq_disable:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
/* Unlocking the mutex in the middle of handling is
|
|
|
|
inherently unsafe. In this case we deem it safe to do,
|
|
|
|
because we need to let any possibly pending IRQ out of
|
|
|
|
the system (and while we are WL1271_STATE_OFF the IRQ
|
|
|
|
work function will not do anything.) Also, any other
|
|
|
|
possible concurrent operations will fail due to the
|
|
|
|
current state, hence the wl1271 struct should be safe. */
|
2011-03-01 21:14:41 +08:00
|
|
|
wl1271_disable_interrupts(wl);
|
|
|
|
wl1271_flush_deferred_work(wl);
|
|
|
|
cancel_work_sync(&wl->netstack_work);
|
2009-12-11 21:41:01 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
power_off:
|
|
|
|
wl1271_power_off(wl);
|
|
|
|
}
|
2009-10-13 17:47:45 +08:00
|
|
|
|
2010-10-29 03:46:43 +08:00
|
|
|
if (!booted) {
|
|
|
|
wl1271_error("firmware boot failed despite %d retries",
|
|
|
|
WL1271_BOOT_RETRIES);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->vif = vif;
|
|
|
|
wl->state = WL1271_STATE_ON;
|
2011-03-29 21:43:50 +08:00
|
|
|
set_bit(WL1271_FLAG_IF_INITIALIZED, &wl->flags);
|
2011-01-23 14:27:22 +08:00
|
|
|
wl1271_info("firmware booted (%s)", wl->chip.fw_ver_str);
|
2010-10-29 03:46:43 +08:00
|
|
|
|
|
|
|
/* update hw/fw version info in wiphy struct */
|
|
|
|
wiphy->hw_version = wl->chip.id;
|
2011-01-23 14:27:22 +08:00
|
|
|
strncpy(wiphy->fw_version, wl->chip.fw_ver_str,
|
2010-10-29 03:46:43 +08:00
|
|
|
sizeof(wiphy->fw_version));
|
|
|
|
|
2011-03-10 21:24:57 +08:00
|
|
|
/* Check if any quirks are needed with older fw versions */
|
|
|
|
wl->quirks |= wl1271_get_fw_ver_quirks(wl);
|
|
|
|
|
2010-12-03 23:05:40 +08:00
|
|
|
/*
|
|
|
|
* Now we know if 11a is supported (info from the NVS), so disable
|
|
|
|
* 11a channels if not supported
|
|
|
|
*/
|
|
|
|
if (!wl->enable_11a)
|
|
|
|
wiphy->bands[IEEE80211_BAND_5GHZ]->n_channels = 0;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "11a is %ssupported",
|
|
|
|
wl->enable_11a ? "" : "not ");
|
|
|
|
|
2009-10-13 17:47:45 +08:00
|
|
|
out:
|
2009-08-06 21:25:28 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
2011-03-21 16:43:36 +08:00
|
|
|
mutex_lock(&wl_list_mutex);
|
2010-07-08 22:49:58 +08:00
|
|
|
if (!ret)
|
2009-10-13 17:47:55 +08:00
|
|
|
list_add(&wl->list, &wl_list);
|
2011-03-21 16:43:36 +08:00
|
|
|
mutex_unlock(&wl_list_mutex);
|
2009-10-13 17:47:55 +08:00
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-09-21 12:23:30 +08:00
|
|
|
static void __wl1271_op_remove_interface(struct wl1271 *wl)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2010-03-18 18:26:39 +08:00
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 remove interface");
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2011-03-29 21:43:50 +08:00
|
|
|
/* because of hardware recovery, we may get here twice */
|
|
|
|
if (wl->state != WL1271_STATE_ON)
|
|
|
|
return;
|
|
|
|
|
2010-03-18 18:26:39 +08:00
|
|
|
wl1271_info("down");
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2011-03-21 16:43:36 +08:00
|
|
|
mutex_lock(&wl_list_mutex);
|
2009-10-13 17:47:55 +08:00
|
|
|
list_del(&wl->list);
|
2011-03-21 16:43:36 +08:00
|
|
|
mutex_unlock(&wl_list_mutex);
|
2009-10-13 17:47:55 +08:00
|
|
|
|
2010-07-08 22:50:03 +08:00
|
|
|
/* enable dyn ps just in case (if left on due to fw crash etc) */
|
2010-07-08 22:50:04 +08:00
|
|
|
if (wl->bss_type == BSS_TYPE_STA_BSS)
|
2010-07-08 22:50:05 +08:00
|
|
|
ieee80211_enable_dyn_ps(wl->vif);
|
2010-07-08 22:50:03 +08:00
|
|
|
|
2010-07-08 22:50:07 +08:00
|
|
|
if (wl->scan.state != WL1271_SCAN_STATE_IDLE) {
|
|
|
|
wl->scan.state = WL1271_SCAN_STATE_IDLE;
|
2011-03-22 05:16:14 +08:00
|
|
|
memset(wl->scan.scanned_ch, 0, sizeof(wl->scan.scanned_ch));
|
2010-10-26 19:24:38 +08:00
|
|
|
wl->scan.req = NULL;
|
2010-07-29 09:54:45 +08:00
|
|
|
ieee80211_scan_completed(wl->hw, true);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2011-03-29 21:43:50 +08:00
|
|
|
/*
|
|
|
|
* this must be before the cancel_work calls below, so that the work
|
|
|
|
* functions don't perform further work.
|
|
|
|
*/
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->state = WL1271_STATE_OFF;
|
|
|
|
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
wl1271_disable_interrupts(wl);
|
|
|
|
wl1271_flush_deferred_work(wl);
|
2010-09-21 12:23:32 +08:00
|
|
|
cancel_delayed_work_sync(&wl->scan_complete_work);
|
2011-03-01 21:14:41 +08:00
|
|
|
cancel_work_sync(&wl->netstack_work);
|
2009-08-06 21:25:28 +08:00
|
|
|
cancel_work_sync(&wl->tx_work);
|
2010-07-08 22:50:00 +08:00
|
|
|
cancel_delayed_work_sync(&wl->pspoll_work);
|
2010-09-21 12:23:29 +08:00
|
|
|
cancel_delayed_work_sync(&wl->elp_work);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
/* let's notify MAC80211 about the remaining pending TX frames */
|
2010-05-24 16:18:17 +08:00
|
|
|
wl1271_tx_reset(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
wl1271_power_off(wl);
|
|
|
|
|
|
|
|
memset(wl->bssid, 0, ETH_ALEN);
|
|
|
|
memset(wl->ssid, 0, IW_ESSID_MAX_SIZE + 1);
|
|
|
|
wl->ssid_len = 0;
|
|
|
|
wl->bss_type = MAX_BSS_TYPE;
|
2010-03-26 18:53:24 +08:00
|
|
|
wl->set_bss_type = MAX_BSS_TYPE;
|
2009-10-09 02:56:24 +08:00
|
|
|
wl->band = IEEE80211_BAND_2GHZ;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
wl->rx_counter = 0;
|
2009-11-03 02:22:11 +08:00
|
|
|
wl->psm_entry_retry = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->power_level = WL1271_DEFAULT_POWER_LEVEL;
|
|
|
|
wl->tx_blocks_available = 0;
|
2011-03-31 16:07:00 +08:00
|
|
|
wl->tx_allocated_blocks = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->tx_results_count = 0;
|
|
|
|
wl->tx_packets_count = 0;
|
2009-10-09 02:56:19 +08:00
|
|
|
wl->tx_security_last_seq = 0;
|
2010-02-22 14:38:40 +08:00
|
|
|
wl->tx_security_seq = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->time_offset = 0;
|
|
|
|
wl->session_counter = 0;
|
2009-12-11 21:41:06 +08:00
|
|
|
wl->rate_set = CONF_TX_RATE_MASK_BASIC;
|
2010-03-18 18:26:39 +08:00
|
|
|
wl->vif = NULL;
|
2010-03-18 18:26:43 +08:00
|
|
|
wl->filters = 0;
|
2010-10-17 03:39:06 +08:00
|
|
|
wl1271_free_ap_keys(wl);
|
2010-10-17 02:21:23 +08:00
|
|
|
memset(wl->ap_hlid_map, 0, sizeof(wl->ap_hlid_map));
|
2011-02-23 06:22:31 +08:00
|
|
|
wl->ap_fw_ps_map = 0;
|
|
|
|
wl->ap_ps_map = 0;
|
2009-10-12 20:08:43 +08:00
|
|
|
|
2011-03-29 21:43:50 +08:00
|
|
|
/*
|
|
|
|
* this is performed after the cancel_work calls and the associated
|
|
|
|
* mutex_lock, so that wl1271_op_add_interface does not accidentally
|
|
|
|
* get executed before all these vars have been reset.
|
|
|
|
*/
|
|
|
|
wl->flags = 0;
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
for (i = 0; i < NUM_TX_QUEUES; i++)
|
|
|
|
wl->tx_blocks_freed[i] = 0;
|
|
|
|
|
|
|
|
wl1271_debugfs_reset(wl);
|
2010-04-09 16:07:26 +08:00
|
|
|
|
|
|
|
kfree(wl->fw_status);
|
|
|
|
wl->fw_status = NULL;
|
|
|
|
kfree(wl->tx_res_if);
|
|
|
|
wl->tx_res_if = NULL;
|
|
|
|
kfree(wl->target_mem_map);
|
|
|
|
wl->target_mem_map = NULL;
|
2010-09-21 12:23:30 +08:00
|
|
|
}
|
2010-04-09 16:07:26 +08:00
|
|
|
|
2010-09-21 12:23:30 +08:00
|
|
|
static void wl1271_op_remove_interface(struct ieee80211_hw *hw,
|
|
|
|
struct ieee80211_vif *vif)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
2010-11-18 21:19:02 +08:00
|
|
|
/*
|
|
|
|
* wl->vif can be null here if someone shuts down the interface
|
|
|
|
* just when hardware recovery has been started.
|
|
|
|
*/
|
|
|
|
if (wl->vif) {
|
|
|
|
WARN_ON(wl->vif != vif);
|
|
|
|
__wl1271_op_remove_interface(wl);
|
|
|
|
}
|
2010-09-21 12:23:31 +08:00
|
|
|
|
2010-11-18 21:19:02 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
2010-09-21 12:23:31 +08:00
|
|
|
cancel_work_sync(&wl->recovery_work);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2011-03-30 22:35:59 +08:00
|
|
|
void wl1271_configure_filters(struct wl1271 *wl, unsigned int filters)
|
2010-03-18 18:26:43 +08:00
|
|
|
{
|
2010-10-17 00:45:07 +08:00
|
|
|
wl1271_set_default_filters(wl);
|
2010-03-18 18:26:43 +08:00
|
|
|
|
|
|
|
/* combine requested filters with current filter config */
|
|
|
|
filters = wl->filters | filters;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_FILTERS, "RX filters set: ");
|
|
|
|
|
|
|
|
if (filters & FIF_PROMISC_IN_BSS) {
|
|
|
|
wl1271_debug(DEBUG_FILTERS, " - FIF_PROMISC_IN_BSS");
|
|
|
|
wl->rx_config &= ~CFG_UNI_FILTER_EN;
|
|
|
|
wl->rx_config |= CFG_BSSID_FILTER_EN;
|
|
|
|
}
|
|
|
|
if (filters & FIF_BCN_PRBRESP_PROMISC) {
|
|
|
|
wl1271_debug(DEBUG_FILTERS, " - FIF_BCN_PRBRESP_PROMISC");
|
|
|
|
wl->rx_config &= ~CFG_BSSID_FILTER_EN;
|
|
|
|
wl->rx_config &= ~CFG_SSID_FILTER_EN;
|
|
|
|
}
|
|
|
|
if (filters & FIF_OTHER_BSS) {
|
|
|
|
wl1271_debug(DEBUG_FILTERS, " - FIF_OTHER_BSS");
|
|
|
|
wl->rx_config &= ~CFG_BSSID_FILTER_EN;
|
|
|
|
}
|
|
|
|
if (filters & FIF_CONTROL) {
|
|
|
|
wl1271_debug(DEBUG_FILTERS, " - FIF_CONTROL");
|
|
|
|
wl->rx_filter |= CFG_RX_CTL_EN;
|
|
|
|
}
|
|
|
|
if (filters & FIF_FCSFAIL) {
|
|
|
|
wl1271_debug(DEBUG_FILTERS, " - FIF_FCSFAIL");
|
|
|
|
wl->rx_filter |= CFG_RX_FCS_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-28 14:50:01 +08:00
|
|
|
static int wl1271_dummy_join(struct wl1271 *wl)
|
2009-12-11 21:40:44 +08:00
|
|
|
{
|
2009-12-11 21:41:04 +08:00
|
|
|
int ret = 0;
|
2009-12-11 21:40:44 +08:00
|
|
|
/* we need to use a dummy BSSID for now */
|
|
|
|
static const u8 dummy_bssid[ETH_ALEN] = { 0x0b, 0xad, 0xde,
|
|
|
|
0xad, 0xbe, 0xef };
|
|
|
|
|
|
|
|
memcpy(wl->bssid, dummy_bssid, ETH_ALEN);
|
|
|
|
|
2010-03-18 18:26:43 +08:00
|
|
|
/* pass through frames from all BSS */
|
|
|
|
wl1271_configure_filters(wl, FIF_OTHER_BSS);
|
|
|
|
|
2010-03-26 18:53:24 +08:00
|
|
|
ret = wl1271_cmd_join(wl, wl->set_bss_type);
|
2009-12-11 21:40:44 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2009-12-11 21:41:07 +08:00
|
|
|
set_bit(WL1271_FLAG_JOINED, &wl->flags);
|
2009-12-11 21:40:44 +08:00
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-05-07 16:39:00 +08:00
|
|
|
static int wl1271_join(struct wl1271 *wl, bool set_assoc)
|
2010-04-28 14:50:01 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2010-05-07 16:39:00 +08:00
|
|
|
/*
|
|
|
|
* One of the side effects of the JOIN command is that is clears
|
|
|
|
* WPA/WPA2 keys from the chipset. Performing a JOIN while associated
|
|
|
|
* to a WPA/WPA2 access point will therefore kill the data-path.
|
2011-03-31 01:18:31 +08:00
|
|
|
* Currently the only valid scenario for JOIN during association
|
|
|
|
* is on roaming, in which case we will also be given new keys.
|
|
|
|
* Keep the below message for now, unless it starts bothering
|
|
|
|
* users who really like to roam a lot :)
|
2010-05-07 16:39:00 +08:00
|
|
|
*/
|
|
|
|
if (test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
|
|
|
|
wl1271_info("JOIN while associated.");
|
|
|
|
|
|
|
|
if (set_assoc)
|
|
|
|
set_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags);
|
|
|
|
|
2010-04-28 14:50:01 +08:00
|
|
|
ret = wl1271_cmd_join(wl, wl->set_bss_type);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
set_bit(WL1271_FLAG_JOINED, &wl->flags);
|
|
|
|
|
|
|
|
if (!test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The join command disable the keep-alive mode, shut down its process,
|
|
|
|
* and also clear the template config, so we need to reset it all after
|
|
|
|
* the join. The acx_aid starts the keep-alive process, and the order
|
|
|
|
* of the commands below is relevant.
|
|
|
|
*/
|
|
|
|
ret = wl1271_acx_keep_alive_mode(wl, true);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1271_acx_aid(wl, wl->aid);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1271_cmd_build_klv_null_data(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1271_acx_keep_alive_config(wl, CMD_TEMPL_KLV_IDX_NULL_DATA,
|
|
|
|
ACX_KEEP_ALIVE_TPL_VALID);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_unjoin(struct wl1271 *wl)
|
2009-12-11 21:40:44 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* to stop listening to a channel, we disconnect */
|
|
|
|
ret = wl1271_cmd_disconnect(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2009-12-11 21:41:07 +08:00
|
|
|
clear_bit(WL1271_FLAG_JOINED, &wl->flags);
|
2009-12-11 21:40:44 +08:00
|
|
|
memset(wl->bssid, 0, ETH_ALEN);
|
2010-03-18 18:26:43 +08:00
|
|
|
|
2011-03-30 22:35:59 +08:00
|
|
|
/* stop filtering packets based on bssid */
|
2010-03-18 18:26:43 +08:00
|
|
|
wl1271_configure_filters(wl, FIF_OTHER_BSS);
|
2009-12-11 21:40:44 +08:00
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-04-01 16:38:20 +08:00
|
|
|
static void wl1271_set_band_rate(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
if (wl->band == IEEE80211_BAND_2GHZ)
|
|
|
|
wl->basic_rate_set = wl->conf.tx.basic_rate;
|
|
|
|
else
|
|
|
|
wl->basic_rate_set = wl->conf.tx.basic_rate_5;
|
|
|
|
}
|
|
|
|
|
2010-10-17 01:17:02 +08:00
|
|
|
static int wl1271_sta_handle_idle(struct wl1271 *wl, bool idle)
|
2010-05-24 16:18:16 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (idle) {
|
|
|
|
if (test_bit(WL1271_FLAG_JOINED, &wl->flags)) {
|
|
|
|
ret = wl1271_unjoin(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
}
|
2010-10-17 00:19:53 +08:00
|
|
|
wl->rate_set = wl1271_tx_min_rate_get(wl);
|
2010-10-16 23:52:59 +08:00
|
|
|
ret = wl1271_acx_sta_rate_policies(wl);
|
2010-05-24 16:18:16 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
ret = wl1271_acx_keep_alive_config(
|
|
|
|
wl, CMD_TEMPL_KLV_IDX_NULL_DATA,
|
|
|
|
ACX_KEEP_ALIVE_TPL_INVALID);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
set_bit(WL1271_FLAG_IDLE, &wl->flags);
|
|
|
|
} else {
|
|
|
|
/* increment the session counter */
|
|
|
|
wl->session_counter++;
|
|
|
|
if (wl->session_counter >= SESSION_COUNTER_MAX)
|
|
|
|
wl->session_counter = 0;
|
|
|
|
ret = wl1271_dummy_join(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
clear_bit(WL1271_FLAG_IDLE, &wl->flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static int wl1271_op_config(struct ieee80211_hw *hw, u32 changed)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
struct ieee80211_conf *conf = &hw->conf;
|
|
|
|
int channel, ret = 0;
|
2010-10-17 01:17:02 +08:00
|
|
|
bool is_ap;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
channel = ieee80211_frequency_to_channel(conf->channel->center_freq);
|
|
|
|
|
2010-10-17 01:17:02 +08:00
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 config ch %d psm %s power %d %s"
|
|
|
|
" changed 0x%x",
|
2009-08-06 21:25:28 +08:00
|
|
|
channel,
|
|
|
|
conf->flags & IEEE80211_CONF_PS ? "on" : "off",
|
2009-12-11 21:40:44 +08:00
|
|
|
conf->power_level,
|
2010-10-17 01:17:02 +08:00
|
|
|
conf->flags & IEEE80211_CONF_IDLE ? "idle" : "in use",
|
|
|
|
changed);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-05-24 16:18:17 +08:00
|
|
|
/*
|
|
|
|
* mac80211 will go to idle nearly immediately after transmitting some
|
|
|
|
* frames, such as the deauth. To make sure those frames reach the air,
|
|
|
|
* wait here until the TX queue is fully flushed.
|
|
|
|
*/
|
|
|
|
if ((changed & IEEE80211_CONF_CHANGE_IDLE) &&
|
|
|
|
(conf->flags & IEEE80211_CONF_IDLE))
|
|
|
|
wl1271_tx_flush(wl);
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2010-10-26 19:24:39 +08:00
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF)) {
|
2011-03-22 16:07:47 +08:00
|
|
|
/* we support configuring the channel and band while off */
|
|
|
|
if ((changed & IEEE80211_CONF_CHANGE_CHANNEL)) {
|
|
|
|
wl->band = conf->channel->band;
|
|
|
|
wl->channel = channel;
|
|
|
|
}
|
|
|
|
|
2010-04-09 16:07:27 +08:00
|
|
|
goto out;
|
2010-10-26 19:24:39 +08:00
|
|
|
}
|
2009-10-09 02:56:24 +08:00
|
|
|
|
2010-10-17 01:17:02 +08:00
|
|
|
is_ap = (wl->bss_type == BSS_TYPE_AP_BSS);
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2010-04-01 16:38:20 +08:00
|
|
|
/* if the channel changes while joined, join again */
|
2010-05-07 16:39:00 +08:00
|
|
|
if (changed & IEEE80211_CONF_CHANGE_CHANNEL &&
|
|
|
|
((wl->band != conf->channel->band) ||
|
|
|
|
(wl->channel != channel))) {
|
2010-04-01 16:38:20 +08:00
|
|
|
wl->band = conf->channel->band;
|
|
|
|
wl->channel = channel;
|
|
|
|
|
2010-10-17 01:17:02 +08:00
|
|
|
if (!is_ap) {
|
|
|
|
/*
|
|
|
|
* FIXME: the mac80211 should really provide a fixed
|
|
|
|
* rate to use here. for now, just use the smallest
|
|
|
|
* possible rate for the band as a fixed rate for
|
|
|
|
* association frames and other control messages.
|
|
|
|
*/
|
|
|
|
if (!test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags))
|
|
|
|
wl1271_set_band_rate(wl);
|
|
|
|
|
|
|
|
wl->basic_rate = wl1271_tx_min_rate_get(wl);
|
|
|
|
ret = wl1271_acx_sta_rate_policies(wl);
|
2010-04-01 16:38:20 +08:00
|
|
|
if (ret < 0)
|
2010-10-17 01:17:02 +08:00
|
|
|
wl1271_warning("rate policy for channel "
|
2010-04-01 16:38:20 +08:00
|
|
|
"failed %d", ret);
|
2010-10-17 01:17:02 +08:00
|
|
|
|
|
|
|
if (test_bit(WL1271_FLAG_JOINED, &wl->flags)) {
|
|
|
|
ret = wl1271_join(wl, false);
|
|
|
|
if (ret < 0)
|
|
|
|
wl1271_warning("cmd join on channel "
|
|
|
|
"failed %d", ret);
|
|
|
|
}
|
2010-04-01 16:38:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-17 01:17:02 +08:00
|
|
|
if (changed & IEEE80211_CONF_CHANGE_IDLE && !is_ap) {
|
|
|
|
ret = wl1271_sta_handle_idle(wl,
|
|
|
|
conf->flags & IEEE80211_CONF_IDLE);
|
2010-05-24 16:18:16 +08:00
|
|
|
if (ret < 0)
|
|
|
|
wl1271_warning("idle mode change failed %d", ret);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2010-07-08 22:50:00 +08:00
|
|
|
/*
|
|
|
|
* if mac80211 changes the PSM mode, make sure the mode is not
|
|
|
|
* incorrectly changed after the pspoll failure active window.
|
|
|
|
*/
|
|
|
|
if (changed & IEEE80211_CONF_CHANGE_PS)
|
|
|
|
clear_bit(WL1271_FLAG_PSPOLL_FAILURE, &wl->flags);
|
|
|
|
|
2009-12-11 21:41:07 +08:00
|
|
|
if (conf->flags & IEEE80211_CONF_PS &&
|
|
|
|
!test_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags)) {
|
|
|
|
set_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We enter PSM only if we're already associated.
|
|
|
|
* If we're not, we'll enter it when joining an SSID,
|
|
|
|
* through the bss_info_changed() hook.
|
|
|
|
*/
|
2009-12-11 21:41:06 +08:00
|
|
|
if (test_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags)) {
|
2010-02-22 14:38:34 +08:00
|
|
|
wl1271_debug(DEBUG_PSM, "psm enabled");
|
2010-02-18 19:25:36 +08:00
|
|
|
ret = wl1271_ps_set_mode(wl, STATION_POWER_SAVE_MODE,
|
2010-09-24 09:10:11 +08:00
|
|
|
wl->basic_rate, true);
|
2009-12-11 21:41:00 +08:00
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
} else if (!(conf->flags & IEEE80211_CONF_PS) &&
|
2009-12-11 21:41:07 +08:00
|
|
|
test_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags)) {
|
2010-02-22 14:38:34 +08:00
|
|
|
wl1271_debug(DEBUG_PSM, "psm disabled");
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-12-11 21:41:07 +08:00
|
|
|
clear_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-12-11 21:41:07 +08:00
|
|
|
if (test_bit(WL1271_FLAG_PSM, &wl->flags))
|
2010-02-18 19:25:36 +08:00
|
|
|
ret = wl1271_ps_set_mode(wl, STATION_ACTIVE_MODE,
|
2010-09-24 09:10:11 +08:00
|
|
|
wl->basic_rate, true);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (conf->power_level != wl->power_level) {
|
|
|
|
ret = wl1271_acx_tx_power(wl, conf->power_level);
|
|
|
|
if (ret < 0)
|
2009-11-03 02:22:08 +08:00
|
|
|
goto out_sleep;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
wl->power_level = conf->power_level;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_sleep:
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-10-13 17:47:59 +08:00
|
|
|
struct wl1271_filter_params {
|
|
|
|
bool enabled;
|
|
|
|
int mc_list_length;
|
|
|
|
u8 mc_list[ACX_MC_ADDRESS_GROUP_MAX][ETH_ALEN];
|
|
|
|
};
|
|
|
|
|
2010-04-02 05:22:57 +08:00
|
|
|
static u64 wl1271_op_prepare_multicast(struct ieee80211_hw *hw,
|
|
|
|
struct netdev_hw_addr_list *mc_list)
|
2009-10-09 02:56:31 +08:00
|
|
|
{
|
|
|
|
struct wl1271_filter_params *fp;
|
2010-04-02 05:22:57 +08:00
|
|
|
struct netdev_hw_addr *ha;
|
2010-04-09 16:07:27 +08:00
|
|
|
struct wl1271 *wl = hw->priv;
|
2009-10-09 02:56:31 +08:00
|
|
|
|
2010-04-09 16:07:27 +08:00
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF))
|
|
|
|
return 0;
|
2009-10-09 02:56:31 +08:00
|
|
|
|
2009-10-13 17:47:53 +08:00
|
|
|
fp = kzalloc(sizeof(*fp), GFP_ATOMIC);
|
2009-10-09 02:56:31 +08:00
|
|
|
if (!fp) {
|
|
|
|
wl1271_error("Out of memory setting filters.");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* update multicast filtering parameters */
|
|
|
|
fp->mc_list_length = 0;
|
2010-04-02 05:22:57 +08:00
|
|
|
if (netdev_hw_addr_list_count(mc_list) > ACX_MC_ADDRESS_GROUP_MAX) {
|
|
|
|
fp->enabled = false;
|
|
|
|
} else {
|
|
|
|
fp->enabled = true;
|
|
|
|
netdev_hw_addr_list_for_each(ha, mc_list) {
|
2009-10-09 02:56:31 +08:00
|
|
|
memcpy(fp->mc_list[fp->mc_list_length],
|
2010-04-02 05:22:57 +08:00
|
|
|
ha->addr, ETH_ALEN);
|
2009-10-09 02:56:31 +08:00
|
|
|
fp->mc_list_length++;
|
2010-04-02 05:22:57 +08:00
|
|
|
}
|
2009-10-09 02:56:31 +08:00
|
|
|
}
|
|
|
|
|
2009-10-13 17:47:59 +08:00
|
|
|
return (u64)(unsigned long)fp;
|
2009-10-09 02:56:31 +08:00
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-10-13 17:47:59 +08:00
|
|
|
#define WL1271_SUPPORTED_FILTERS (FIF_PROMISC_IN_BSS | \
|
|
|
|
FIF_ALLMULTI | \
|
|
|
|
FIF_FCSFAIL | \
|
|
|
|
FIF_BCN_PRBRESP_PROMISC | \
|
|
|
|
FIF_CONTROL | \
|
|
|
|
FIF_OTHER_BSS)
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static void wl1271_op_configure_filter(struct ieee80211_hw *hw,
|
|
|
|
unsigned int changed,
|
2009-10-09 02:56:31 +08:00
|
|
|
unsigned int *total, u64 multicast)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
2009-10-13 17:47:59 +08:00
|
|
|
struct wl1271_filter_params *fp = (void *)(unsigned long)multicast;
|
2009-08-06 21:25:28 +08:00
|
|
|
struct wl1271 *wl = hw->priv;
|
2009-10-13 17:47:59 +08:00
|
|
|
int ret;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-17 01:25:35 +08:00
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 configure filter changed %x"
|
|
|
|
" total %x", changed, *total);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-10-13 17:47:59 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2010-04-09 16:07:27 +08:00
|
|
|
*total &= WL1271_SUPPORTED_FILTERS;
|
|
|
|
changed &= WL1271_SUPPORTED_FILTERS;
|
|
|
|
|
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF))
|
2009-10-13 17:47:59 +08:00
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2009-10-13 17:47:59 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2010-10-17 01:25:35 +08:00
|
|
|
if (wl->bss_type != BSS_TYPE_AP_BSS) {
|
|
|
|
if (*total & FIF_ALLMULTI)
|
|
|
|
ret = wl1271_acx_group_address_tbl(wl, false, NULL, 0);
|
|
|
|
else if (fp)
|
|
|
|
ret = wl1271_acx_group_address_tbl(wl, fp->enabled,
|
|
|
|
fp->mc_list,
|
|
|
|
fp->mc_list_length);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2009-10-13 17:47:59 +08:00
|
|
|
/* determine, whether supported filter values have changed */
|
|
|
|
if (changed == 0)
|
|
|
|
goto out_sleep;
|
2009-10-09 02:56:31 +08:00
|
|
|
|
2010-03-18 18:26:43 +08:00
|
|
|
/* configure filters */
|
|
|
|
wl->filters = *total;
|
|
|
|
wl1271_configure_filters(wl, 0);
|
|
|
|
|
2009-10-13 17:47:59 +08:00
|
|
|
/* apply configured filters */
|
|
|
|
ret = wl1271_acx_rx_config(wl, wl->rx_config, wl->rx_filter);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
|
|
|
|
out_sleep:
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
2010-03-18 18:26:43 +08:00
|
|
|
kfree(fp);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
2010-10-17 03:39:06 +08:00
|
|
|
static int wl1271_record_ap_key(struct wl1271 *wl, u8 id, u8 key_type,
|
|
|
|
u8 key_size, const u8 *key, u8 hlid, u32 tx_seq_32,
|
|
|
|
u16 tx_seq_16)
|
|
|
|
{
|
|
|
|
struct wl1271_ap_key *ap_key;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_CRYPT, "record ap key id %d", (int)id);
|
|
|
|
|
|
|
|
if (key_size > MAX_KEY_SIZE)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find next free entry in ap_keys. Also check we are not replacing
|
|
|
|
* an existing key.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < MAX_NUM_KEYS; i++) {
|
|
|
|
if (wl->recorded_ap_keys[i] == NULL)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (wl->recorded_ap_keys[i]->id == id) {
|
|
|
|
wl1271_warning("trying to record key replacement");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == MAX_NUM_KEYS)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
ap_key = kzalloc(sizeof(*ap_key), GFP_KERNEL);
|
|
|
|
if (!ap_key)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
ap_key->id = id;
|
|
|
|
ap_key->key_type = key_type;
|
|
|
|
ap_key->key_size = key_size;
|
|
|
|
memcpy(ap_key->key, key, key_size);
|
|
|
|
ap_key->hlid = hlid;
|
|
|
|
ap_key->tx_seq_32 = tx_seq_32;
|
|
|
|
ap_key->tx_seq_16 = tx_seq_16;
|
|
|
|
|
|
|
|
wl->recorded_ap_keys[i] = ap_key;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wl1271_free_ap_keys(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_NUM_KEYS; i++) {
|
|
|
|
kfree(wl->recorded_ap_keys[i]);
|
|
|
|
wl->recorded_ap_keys[i] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_ap_init_hwenc(struct wl1271 *wl)
|
|
|
|
{
|
|
|
|
int i, ret = 0;
|
|
|
|
struct wl1271_ap_key *key;
|
|
|
|
bool wep_key_added = false;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_NUM_KEYS; i++) {
|
|
|
|
if (wl->recorded_ap_keys[i] == NULL)
|
|
|
|
break;
|
|
|
|
|
|
|
|
key = wl->recorded_ap_keys[i];
|
|
|
|
ret = wl1271_cmd_set_ap_key(wl, KEY_ADD_OR_REPLACE,
|
|
|
|
key->id, key->key_type,
|
|
|
|
key->key_size, key->key,
|
|
|
|
key->hlid, key->tx_seq_32,
|
|
|
|
key->tx_seq_16);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (key->key_type == KEY_WEP)
|
|
|
|
wep_key_added = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wep_key_added) {
|
|
|
|
ret = wl1271_cmd_set_ap_default_wep_key(wl, wl->default_key);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
wl1271_free_ap_keys(wl);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_set_key(struct wl1271 *wl, u16 action, u8 id, u8 key_type,
|
|
|
|
u8 key_size, const u8 *key, u32 tx_seq_32,
|
|
|
|
u16 tx_seq_16, struct ieee80211_sta *sta)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
bool is_ap = (wl->bss_type == BSS_TYPE_AP_BSS);
|
|
|
|
|
|
|
|
if (is_ap) {
|
|
|
|
struct wl1271_station *wl_sta;
|
|
|
|
u8 hlid;
|
|
|
|
|
|
|
|
if (sta) {
|
|
|
|
wl_sta = (struct wl1271_station *)sta->drv_priv;
|
|
|
|
hlid = wl_sta->hlid;
|
|
|
|
} else {
|
|
|
|
hlid = WL1271_AP_BROADCAST_HLID;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!test_bit(WL1271_FLAG_AP_STARTED, &wl->flags)) {
|
|
|
|
/*
|
|
|
|
* We do not support removing keys after AP shutdown.
|
|
|
|
* Pretend we do to make mac80211 happy.
|
|
|
|
*/
|
|
|
|
if (action != KEY_ADD_OR_REPLACE)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = wl1271_record_ap_key(wl, id,
|
|
|
|
key_type, key_size,
|
|
|
|
key, hlid, tx_seq_32,
|
|
|
|
tx_seq_16);
|
|
|
|
} else {
|
|
|
|
ret = wl1271_cmd_set_ap_key(wl, action,
|
|
|
|
id, key_type, key_size,
|
|
|
|
key, hlid, tx_seq_32,
|
|
|
|
tx_seq_16);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
} else {
|
|
|
|
const u8 *addr;
|
|
|
|
static const u8 bcast_addr[ETH_ALEN] = {
|
|
|
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff
|
|
|
|
};
|
|
|
|
|
|
|
|
addr = sta ? sta->addr : bcast_addr;
|
|
|
|
|
|
|
|
if (is_zero_ether_addr(addr)) {
|
|
|
|
/* We dont support TX only encryption */
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The wl1271 does not allow to remove unicast keys - they
|
|
|
|
will be cleared automatically on next CMD_JOIN. Ignore the
|
|
|
|
request silently, as we dont want the mac80211 to emit
|
|
|
|
an error message. */
|
|
|
|
if (action == KEY_REMOVE && !is_broadcast_ether_addr(addr))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = wl1271_cmd_set_sta_key(wl, action,
|
|
|
|
id, key_type, key_size,
|
|
|
|
key, addr, tx_seq_32,
|
|
|
|
tx_seq_16);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* the default WEP key needs to be configured at least once */
|
|
|
|
if (key_type == KEY_WEP) {
|
|
|
|
ret = wl1271_cmd_set_sta_default_wep_key(wl,
|
|
|
|
wl->default_key);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static int wl1271_op_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_sta *sta,
|
|
|
|
struct ieee80211_key_conf *key_conf)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
int ret;
|
2009-10-09 02:56:19 +08:00
|
|
|
u32 tx_seq_32 = 0;
|
|
|
|
u16 tx_seq_16 = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
u8 key_type;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 set key");
|
|
|
|
|
2010-10-17 03:39:06 +08:00
|
|
|
wl1271_debug(DEBUG_CRYPT, "CMD: 0x%x sta: %p", cmd, sta);
|
2009-08-06 21:25:28 +08:00
|
|
|
wl1271_debug(DEBUG_CRYPT, "Key: algo:0x%x, id:%d, len:%d flags 0x%x",
|
2010-08-10 15:46:38 +08:00
|
|
|
key_conf->cipher, key_conf->keyidx,
|
2009-08-06 21:25:28 +08:00
|
|
|
key_conf->keylen, key_conf->flags);
|
|
|
|
wl1271_dump(DEBUG_CRYPT, "KEY: ", key_conf->key, key_conf->keylen);
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2010-10-26 19:24:39 +08:00
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF)) {
|
|
|
|
ret = -EAGAIN;
|
|
|
|
goto out_unlock;
|
|
|
|
}
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_unlock;
|
|
|
|
|
2010-08-10 15:46:38 +08:00
|
|
|
switch (key_conf->cipher) {
|
|
|
|
case WLAN_CIPHER_SUITE_WEP40:
|
|
|
|
case WLAN_CIPHER_SUITE_WEP104:
|
2009-08-06 21:25:28 +08:00
|
|
|
key_type = KEY_WEP;
|
|
|
|
|
|
|
|
key_conf->hw_key_idx = key_conf->keyidx;
|
|
|
|
break;
|
2010-08-10 15:46:38 +08:00
|
|
|
case WLAN_CIPHER_SUITE_TKIP:
|
2009-08-06 21:25:28 +08:00
|
|
|
key_type = KEY_TKIP;
|
|
|
|
|
|
|
|
key_conf->hw_key_idx = key_conf->keyidx;
|
2010-02-22 14:38:40 +08:00
|
|
|
tx_seq_32 = WL1271_TX_SECURITY_HI32(wl->tx_security_seq);
|
|
|
|
tx_seq_16 = WL1271_TX_SECURITY_LO16(wl->tx_security_seq);
|
2009-08-06 21:25:28 +08:00
|
|
|
break;
|
2010-08-10 15:46:38 +08:00
|
|
|
case WLAN_CIPHER_SUITE_CCMP:
|
2009-08-06 21:25:28 +08:00
|
|
|
key_type = KEY_AES;
|
|
|
|
|
|
|
|
key_conf->flags |= IEEE80211_KEY_FLAG_GENERATE_IV;
|
2010-02-22 14:38:40 +08:00
|
|
|
tx_seq_32 = WL1271_TX_SECURITY_HI32(wl->tx_security_seq);
|
|
|
|
tx_seq_16 = WL1271_TX_SECURITY_LO16(wl->tx_security_seq);
|
2009-08-06 21:25:28 +08:00
|
|
|
break;
|
2010-09-27 18:42:07 +08:00
|
|
|
case WL1271_CIPHER_SUITE_GEM:
|
|
|
|
key_type = KEY_GEM;
|
|
|
|
tx_seq_32 = WL1271_TX_SECURITY_HI32(wl->tx_security_seq);
|
|
|
|
tx_seq_16 = WL1271_TX_SECURITY_LO16(wl->tx_security_seq);
|
|
|
|
break;
|
2009-08-06 21:25:28 +08:00
|
|
|
default:
|
2010-08-10 15:46:38 +08:00
|
|
|
wl1271_error("Unknown key algo 0x%x", key_conf->cipher);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
case SET_KEY:
|
2010-10-17 03:39:06 +08:00
|
|
|
ret = wl1271_set_key(wl, KEY_ADD_OR_REPLACE,
|
|
|
|
key_conf->keyidx, key_type,
|
|
|
|
key_conf->keylen, key_conf->key,
|
|
|
|
tx_seq_32, tx_seq_16, sta);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("Could not add or replace key");
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DISABLE_KEY:
|
2010-10-17 03:39:06 +08:00
|
|
|
ret = wl1271_set_key(wl, KEY_REMOVE,
|
|
|
|
key_conf->keyidx, key_type,
|
|
|
|
key_conf->keylen, key_conf->key,
|
|
|
|
0, 0, sta);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("Could not remove key");
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
wl1271_error("Unsupported key cmd 0x%x", cmd);
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_sleep:
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out_unlock:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_op_hw_scan(struct ieee80211_hw *hw,
|
2010-04-27 17:59:34 +08:00
|
|
|
struct ieee80211_vif *vif,
|
2009-08-06 21:25:28 +08:00
|
|
|
struct cfg80211_scan_request *req)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
int ret;
|
|
|
|
u8 *ssid = NULL;
|
2009-10-13 17:47:50 +08:00
|
|
|
size_t len = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 hw scan");
|
|
|
|
|
|
|
|
if (req->n_ssids) {
|
|
|
|
ssid = req->ssids[0].ssid;
|
2009-10-13 17:47:50 +08:00
|
|
|
len = req->ssids[0].ssid_len;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2010-10-26 19:24:38 +08:00
|
|
|
if (wl->state == WL1271_STATE_OFF) {
|
|
|
|
/*
|
|
|
|
* We cannot return -EBUSY here because cfg80211 will expect
|
|
|
|
* a call to ieee80211_scan_completed if we do - in this case
|
|
|
|
* there won't be any call.
|
|
|
|
*/
|
|
|
|
ret = -EAGAIN;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2010-08-04 08:46:22 +08:00
|
|
|
ret = wl1271_scan(hw->priv, ssid, len, req);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-11-08 17:51:07 +08:00
|
|
|
static int wl1271_op_set_frag_threshold(struct ieee80211_hw *hw, u32 value)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF)) {
|
|
|
|
ret = -EAGAIN;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2010-11-08 17:51:07 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1271_acx_frag_threshold(wl, (u16)value);
|
|
|
|
if (ret < 0)
|
|
|
|
wl1271_warning("wl1271_op_set_frag_threshold failed: %d", ret);
|
|
|
|
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static int wl1271_op_set_rts_threshold(struct ieee80211_hw *hw, u32 value)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
2010-04-09 16:07:28 +08:00
|
|
|
int ret = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2010-10-26 19:24:39 +08:00
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF)) {
|
|
|
|
ret = -EAGAIN;
|
2010-04-09 16:07:28 +08:00
|
|
|
goto out;
|
2010-10-26 19:24:39 +08:00
|
|
|
}
|
2010-04-09 16:07:28 +08:00
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1271_acx_rts_threshold(wl, (u16) value);
|
|
|
|
if (ret < 0)
|
|
|
|
wl1271_warning("wl1271_op_set_rts_threshold failed: %d", ret);
|
|
|
|
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
static int wl1271_ssid_set(struct wl1271 *wl, struct sk_buff *skb,
|
2010-11-24 14:16:57 +08:00
|
|
|
int offset)
|
2010-02-18 19:25:38 +08:00
|
|
|
{
|
2010-11-24 14:16:57 +08:00
|
|
|
u8 *ptr = skb->data + offset;
|
2010-02-18 19:25:38 +08:00
|
|
|
|
|
|
|
/* find the location of the ssid in the beacon */
|
2010-11-24 14:16:57 +08:00
|
|
|
while (ptr < skb->data + skb->len) {
|
2010-02-18 19:25:38 +08:00
|
|
|
if (ptr[0] == WLAN_EID_SSID) {
|
|
|
|
wl->ssid_len = ptr[1];
|
|
|
|
memcpy(wl->ssid, ptr+2, wl->ssid_len);
|
2010-10-17 01:07:21 +08:00
|
|
|
return 0;
|
2010-02-18 19:25:38 +08:00
|
|
|
}
|
2010-11-24 14:16:57 +08:00
|
|
|
ptr += (ptr[1] + 2);
|
2010-02-18 19:25:38 +08:00
|
|
|
}
|
2010-10-17 01:07:21 +08:00
|
|
|
|
2010-11-24 14:16:57 +08:00
|
|
|
wl1271_error("No SSID in IEs!\n");
|
2010-10-17 01:07:21 +08:00
|
|
|
return -ENOENT;
|
2010-02-18 19:25:38 +08:00
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
static int wl1271_bss_erp_info_changed(struct wl1271 *wl,
|
2009-08-06 21:25:28 +08:00
|
|
|
struct ieee80211_bss_conf *bss_conf,
|
|
|
|
u32 changed)
|
|
|
|
{
|
2010-10-17 01:07:21 +08:00
|
|
|
int ret = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if (changed & BSS_CHANGED_ERP_SLOT) {
|
|
|
|
if (bss_conf->use_short_slot)
|
|
|
|
ret = wl1271_acx_slot(wl, SLOT_TIME_SHORT);
|
|
|
|
else
|
|
|
|
ret = wl1271_acx_slot(wl, SLOT_TIME_LONG);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("Set slot time failed %d", ret);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if (changed & BSS_CHANGED_ERP_PREAMBLE) {
|
|
|
|
if (bss_conf->use_short_preamble)
|
|
|
|
wl1271_acx_set_preamble(wl, ACX_PREAMBLE_SHORT);
|
|
|
|
else
|
|
|
|
wl1271_acx_set_preamble(wl, ACX_PREAMBLE_LONG);
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if (changed & BSS_CHANGED_ERP_CTS_PROT) {
|
|
|
|
if (bss_conf->use_cts_prot)
|
|
|
|
ret = wl1271_acx_cts_protect(wl, CTSPROTECT_ENABLE);
|
|
|
|
else
|
|
|
|
ret = wl1271_acx_cts_protect(wl, CTSPROTECT_DISABLE);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("Set ctsprotect failed %d", ret);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
2010-10-26 19:24:39 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
static int wl1271_bss_beacon_info_changed(struct wl1271 *wl,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_bss_conf *bss_conf,
|
|
|
|
u32 changed)
|
|
|
|
{
|
|
|
|
bool is_ap = (wl->bss_type == BSS_TYPE_AP_BSS);
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if ((changed & BSS_CHANGED_BEACON_INT)) {
|
|
|
|
wl1271_debug(DEBUG_MASTER, "beacon interval updated: %d",
|
2010-03-26 18:53:25 +08:00
|
|
|
bss_conf->beacon_int);
|
|
|
|
|
|
|
|
wl->beacon_int = bss_conf->beacon_int;
|
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if ((changed & BSS_CHANGED_BEACON)) {
|
|
|
|
struct ieee80211_hdr *hdr;
|
|
|
|
int ieoffset = offsetof(struct ieee80211_mgmt,
|
|
|
|
u.beacon.variable);
|
|
|
|
struct sk_buff *beacon = ieee80211_beacon_get(wl->hw, vif);
|
|
|
|
u16 tmpl_id;
|
|
|
|
|
|
|
|
if (!beacon)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MASTER, "beacon updated");
|
|
|
|
|
|
|
|
ret = wl1271_ssid_set(wl, beacon, ieoffset);
|
|
|
|
if (ret < 0) {
|
|
|
|
dev_kfree_skb(beacon);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
tmpl_id = is_ap ? CMD_TEMPL_AP_BEACON :
|
|
|
|
CMD_TEMPL_BEACON;
|
|
|
|
ret = wl1271_cmd_template_set(wl, tmpl_id,
|
|
|
|
beacon->data,
|
|
|
|
beacon->len, 0,
|
|
|
|
wl1271_tx_min_rate_get(wl));
|
|
|
|
if (ret < 0) {
|
|
|
|
dev_kfree_skb(beacon);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
hdr = (struct ieee80211_hdr *) beacon->data;
|
|
|
|
hdr->frame_control = cpu_to_le16(IEEE80211_FTYPE_MGMT |
|
|
|
|
IEEE80211_STYPE_PROBE_RESP);
|
|
|
|
|
|
|
|
tmpl_id = is_ap ? CMD_TEMPL_AP_PROBE_RESPONSE :
|
|
|
|
CMD_TEMPL_PROBE_RESPONSE;
|
|
|
|
ret = wl1271_cmd_template_set(wl,
|
|
|
|
tmpl_id,
|
|
|
|
beacon->data,
|
|
|
|
beacon->len, 0,
|
|
|
|
wl1271_tx_min_rate_get(wl));
|
|
|
|
dev_kfree_skb(beacon);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* AP mode changes */
|
|
|
|
static void wl1271_bss_info_changed_ap(struct wl1271 *wl,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_bss_conf *bss_conf,
|
|
|
|
u32 changed)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
2009-12-11 21:41:04 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if ((changed & BSS_CHANGED_BASIC_RATES)) {
|
|
|
|
u32 rates = bss_conf->basic_rates;
|
|
|
|
struct conf_tx_rate_class mgmt_rc;
|
2010-03-26 18:53:24 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
wl->basic_rate_set = wl1271_tx_enabled_rates_get(wl, rates);
|
|
|
|
wl->basic_rate = wl1271_tx_min_rate_get(wl);
|
|
|
|
wl1271_debug(DEBUG_AP, "basic rates: 0x%x",
|
|
|
|
wl->basic_rate_set);
|
|
|
|
|
|
|
|
/* update the AP management rate policy with the new rates */
|
|
|
|
mgmt_rc.enabled_rates = wl->basic_rate_set;
|
|
|
|
mgmt_rc.long_retry_limit = 10;
|
|
|
|
mgmt_rc.short_retry_limit = 10;
|
|
|
|
mgmt_rc.aflags = 0;
|
|
|
|
ret = wl1271_acx_ap_rate_policy(wl, &mgmt_rc,
|
|
|
|
ACX_TX_AP_MODE_MGMT_RATE);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("AP mgmt policy change failed %d", ret);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
2010-11-24 14:16:57 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
ret = wl1271_bss_beacon_info_changed(wl, vif, bss_conf, changed);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2010-02-18 19:25:38 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if ((changed & BSS_CHANGED_BEACON_ENABLED)) {
|
|
|
|
if (bss_conf->enable_beacon) {
|
|
|
|
if (!test_bit(WL1271_FLAG_AP_STARTED, &wl->flags)) {
|
|
|
|
ret = wl1271_cmd_start_bss(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-12-11 21:41:04 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
set_bit(WL1271_FLAG_AP_STARTED, &wl->flags);
|
|
|
|
wl1271_debug(DEBUG_AP, "started AP");
|
2010-10-17 03:39:06 +08:00
|
|
|
|
|
|
|
ret = wl1271_ap_init_hwenc(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-12-11 21:41:04 +08:00
|
|
|
}
|
2010-10-17 01:07:21 +08:00
|
|
|
} else {
|
|
|
|
if (test_bit(WL1271_FLAG_AP_STARTED, &wl->flags)) {
|
|
|
|
ret = wl1271_cmd_stop_bss(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-12-11 21:41:04 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
clear_bit(WL1271_FLAG_AP_STARTED, &wl->flags);
|
|
|
|
wl1271_debug(DEBUG_AP, "stopped AP");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-12-11 21:41:04 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
ret = wl1271_bss_erp_info_changed(wl, bss_conf, changed);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
out:
|
|
|
|
return;
|
|
|
|
}
|
2010-02-18 19:25:51 +08:00
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
/* STA/IBSS mode changes */
|
|
|
|
static void wl1271_bss_info_changed_sta(struct wl1271 *wl,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_bss_conf *bss_conf,
|
|
|
|
u32 changed)
|
|
|
|
{
|
|
|
|
bool do_join = false, set_assoc = false;
|
|
|
|
bool is_ibss = (wl->bss_type == BSS_TYPE_IBSS);
|
2011-02-02 15:59:37 +08:00
|
|
|
u32 sta_rate_set = 0;
|
2010-10-17 01:07:21 +08:00
|
|
|
int ret;
|
2011-01-12 02:07:21 +08:00
|
|
|
struct ieee80211_sta *sta;
|
2011-02-13 05:24:20 +08:00
|
|
|
bool sta_exists = false;
|
|
|
|
struct ieee80211_sta_ht_cap sta_ht_cap;
|
2010-10-17 01:07:21 +08:00
|
|
|
|
|
|
|
if (is_ibss) {
|
|
|
|
ret = wl1271_bss_beacon_info_changed(wl, vif, bss_conf,
|
|
|
|
changed);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-12-11 21:41:04 +08:00
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if ((changed & BSS_CHANGED_BEACON_INT) && is_ibss)
|
|
|
|
do_join = true;
|
|
|
|
|
|
|
|
/* Need to update the SSID (for filtering etc) */
|
|
|
|
if ((changed & BSS_CHANGED_BEACON) && is_ibss)
|
|
|
|
do_join = true;
|
|
|
|
|
|
|
|
if ((changed & BSS_CHANGED_BEACON_ENABLED) && is_ibss) {
|
2010-03-26 18:53:24 +08:00
|
|
|
wl1271_debug(DEBUG_ADHOC, "ad-hoc beaconing: %s",
|
|
|
|
bss_conf->enable_beacon ? "enabled" : "disabled");
|
|
|
|
|
|
|
|
if (bss_conf->enable_beacon)
|
|
|
|
wl->set_bss_type = BSS_TYPE_IBSS;
|
|
|
|
else
|
|
|
|
wl->set_bss_type = BSS_TYPE_STA_BSS;
|
|
|
|
do_join = true;
|
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if ((changed & BSS_CHANGED_CQM)) {
|
2010-04-09 16:07:30 +08:00
|
|
|
bool enable = false;
|
|
|
|
if (bss_conf->cqm_rssi_thold)
|
|
|
|
enable = true;
|
|
|
|
ret = wl1271_acx_rssi_snr_trigger(wl, enable,
|
|
|
|
bss_conf->cqm_rssi_thold,
|
|
|
|
bss_conf->cqm_rssi_hyst);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
wl->rssi_thold = bss_conf->cqm_rssi_thold;
|
|
|
|
}
|
|
|
|
|
2010-02-18 19:25:38 +08:00
|
|
|
if ((changed & BSS_CHANGED_BSSID) &&
|
|
|
|
/*
|
|
|
|
* Now we know the correct bssid, so we send a new join command
|
|
|
|
* and enable the BSSID filter
|
|
|
|
*/
|
|
|
|
memcmp(wl->bssid, bss_conf->bssid, ETH_ALEN)) {
|
2010-10-17 01:07:21 +08:00
|
|
|
memcpy(wl->bssid, bss_conf->bssid, ETH_ALEN);
|
2010-03-18 18:26:44 +08:00
|
|
|
|
2010-12-26 16:27:50 +08:00
|
|
|
if (!is_zero_ether_addr(wl->bssid)) {
|
|
|
|
ret = wl1271_cmd_build_null_data(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2010-02-18 19:25:38 +08:00
|
|
|
|
2010-12-26 16:27:50 +08:00
|
|
|
ret = wl1271_build_qos_null_data(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2010-02-18 19:25:38 +08:00
|
|
|
|
2010-12-26 16:27:50 +08:00
|
|
|
/* filter out all packets not from this BSSID */
|
|
|
|
wl1271_configure_filters(wl, 0);
|
2010-03-18 18:26:43 +08:00
|
|
|
|
2010-12-26 16:27:50 +08:00
|
|
|
/* Need to update the BSSID (for filtering etc) */
|
|
|
|
do_join = true;
|
|
|
|
}
|
2010-02-18 19:25:38 +08:00
|
|
|
}
|
|
|
|
|
2011-02-02 15:59:37 +08:00
|
|
|
rcu_read_lock();
|
|
|
|
sta = ieee80211_find_sta(vif, bss_conf->bssid);
|
|
|
|
if (sta) {
|
|
|
|
/* save the supp_rates of the ap */
|
|
|
|
sta_rate_set = sta->supp_rates[wl->hw->conf.channel->band];
|
|
|
|
if (sta->ht_cap.ht_supported)
|
|
|
|
sta_rate_set |=
|
|
|
|
(sta->ht_cap.mcs.rx_mask[0] << HW_HT_RATES_OFFSET);
|
2011-02-13 05:24:20 +08:00
|
|
|
sta_ht_cap = sta->ht_cap;
|
|
|
|
sta_exists = true;
|
|
|
|
}
|
|
|
|
rcu_read_unlock();
|
2011-02-02 15:59:37 +08:00
|
|
|
|
2011-02-13 05:24:20 +08:00
|
|
|
if (sta_exists) {
|
2011-02-02 15:59:37 +08:00
|
|
|
/* handle new association with HT and HT information change */
|
|
|
|
if ((changed & BSS_CHANGED_HT) &&
|
|
|
|
(bss_conf->channel_type != NL80211_CHAN_NO_HT)) {
|
2011-02-13 05:24:20 +08:00
|
|
|
ret = wl1271_acx_set_ht_capabilities(wl, &sta_ht_cap,
|
2011-02-02 15:59:37 +08:00
|
|
|
true);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("Set ht cap true failed %d",
|
|
|
|
ret);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
ret = wl1271_acx_set_ht_information(wl,
|
|
|
|
bss_conf->ht_operation_mode);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("Set ht information failed %d",
|
|
|
|
ret);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* handle new association without HT and disassociation */
|
|
|
|
else if (changed & BSS_CHANGED_ASSOC) {
|
2011-02-13 05:24:20 +08:00
|
|
|
ret = wl1271_acx_set_ht_capabilities(wl, &sta_ht_cap,
|
2011-02-02 15:59:37 +08:00
|
|
|
false);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("Set ht cap false failed %d",
|
|
|
|
ret);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
if ((changed & BSS_CHANGED_ASSOC)) {
|
2009-08-06 21:25:28 +08:00
|
|
|
if (bss_conf->assoc) {
|
2010-04-01 16:38:20 +08:00
|
|
|
u32 rates;
|
2010-11-24 14:16:57 +08:00
|
|
|
int ieoffset;
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->aid = bss_conf->aid;
|
2010-05-07 16:39:00 +08:00
|
|
|
set_assoc = true;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-07-08 22:50:00 +08:00
|
|
|
wl->ps_poll_failures = 0;
|
|
|
|
|
2010-04-01 16:38:20 +08:00
|
|
|
/*
|
|
|
|
* use basic rates from AP, and determine lowest rate
|
|
|
|
* to use with control frames.
|
|
|
|
*/
|
|
|
|
rates = bss_conf->basic_rates;
|
|
|
|
wl->basic_rate_set = wl1271_tx_enabled_rates_get(wl,
|
|
|
|
rates);
|
2010-10-17 00:19:53 +08:00
|
|
|
wl->basic_rate = wl1271_tx_min_rate_get(wl);
|
2011-02-02 15:59:37 +08:00
|
|
|
if (sta_rate_set)
|
|
|
|
wl->rate_set = wl1271_tx_enabled_rates_get(wl,
|
|
|
|
sta_rate_set);
|
2010-10-16 23:52:59 +08:00
|
|
|
ret = wl1271_acx_sta_rate_policies(wl);
|
2010-04-01 16:38:20 +08:00
|
|
|
if (ret < 0)
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2010-04-01 16:38:20 +08:00
|
|
|
|
2009-10-12 20:08:57 +08:00
|
|
|
/*
|
|
|
|
* with wl1271, we don't need to update the
|
|
|
|
* beacon_int and dtim_period, because the firmware
|
|
|
|
* updates it by itself when the first beacon is
|
|
|
|
* received after a join.
|
|
|
|
*/
|
2009-08-06 21:25:28 +08:00
|
|
|
ret = wl1271_cmd_build_ps_poll(wl, wl->aid);
|
|
|
|
if (ret < 0)
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-03-26 18:53:28 +08:00
|
|
|
/*
|
2010-11-24 14:16:57 +08:00
|
|
|
* Get a template for hardware connection maintenance
|
2010-03-26 18:53:28 +08:00
|
|
|
*/
|
2010-11-24 14:16:57 +08:00
|
|
|
dev_kfree_skb(wl->probereq);
|
|
|
|
wl->probereq = wl1271_cmd_build_ap_probe_req(wl, NULL);
|
|
|
|
ieoffset = offsetof(struct ieee80211_mgmt,
|
|
|
|
u.probe_req.variable);
|
|
|
|
wl1271_ssid_set(wl, wl->probereq, ieoffset);
|
2010-03-26 18:53:28 +08:00
|
|
|
|
2010-03-26 18:53:23 +08:00
|
|
|
/* enable the connection monitoring feature */
|
|
|
|
ret = wl1271_acx_conn_monit_params(wl, true);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/* If we want to go in PSM but we're not there yet */
|
2009-12-11 21:41:07 +08:00
|
|
|
if (test_bit(WL1271_FLAG_PSM_REQUESTED, &wl->flags) &&
|
|
|
|
!test_bit(WL1271_FLAG_PSM, &wl->flags)) {
|
2010-10-17 01:07:21 +08:00
|
|
|
enum wl1271_cmd_ps_mode mode;
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
mode = STATION_POWER_SAVE_MODE;
|
2010-08-24 11:28:03 +08:00
|
|
|
ret = wl1271_ps_set_mode(wl, mode,
|
2010-09-24 09:10:11 +08:00
|
|
|
wl->basic_rate,
|
2010-08-24 11:28:03 +08:00
|
|
|
true);
|
2009-08-06 21:25:28 +08:00
|
|
|
if (ret < 0)
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
2009-10-09 02:56:25 +08:00
|
|
|
} else {
|
|
|
|
/* use defaults when not associated */
|
2010-07-27 08:30:09 +08:00
|
|
|
clear_bit(WL1271_FLAG_STA_STATE_SENT, &wl->flags);
|
2009-12-11 21:41:06 +08:00
|
|
|
clear_bit(WL1271_FLAG_STA_ASSOCIATED, &wl->flags);
|
2009-10-09 02:56:25 +08:00
|
|
|
wl->aid = 0;
|
2010-03-26 18:53:23 +08:00
|
|
|
|
2010-11-24 14:16:57 +08:00
|
|
|
/* free probe-request template */
|
|
|
|
dev_kfree_skb(wl->probereq);
|
|
|
|
wl->probereq = NULL;
|
|
|
|
|
2010-07-08 22:50:03 +08:00
|
|
|
/* re-enable dynamic ps - just in case */
|
2010-07-08 22:50:05 +08:00
|
|
|
ieee80211_enable_dyn_ps(wl->vif);
|
2010-07-08 22:50:03 +08:00
|
|
|
|
2010-04-01 16:38:20 +08:00
|
|
|
/* revert back to minimum rates for the current band */
|
|
|
|
wl1271_set_band_rate(wl);
|
2010-10-17 00:19:53 +08:00
|
|
|
wl->basic_rate = wl1271_tx_min_rate_get(wl);
|
2010-10-16 23:52:59 +08:00
|
|
|
ret = wl1271_acx_sta_rate_policies(wl);
|
2010-04-01 16:38:20 +08:00
|
|
|
if (ret < 0)
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2010-04-01 16:38:20 +08:00
|
|
|
|
2010-03-26 18:53:23 +08:00
|
|
|
/* disable connection monitor features */
|
|
|
|
ret = wl1271_acx_conn_monit_params(wl, false);
|
2010-03-26 18:53:32 +08:00
|
|
|
|
|
|
|
/* Disable the keep-alive feature */
|
|
|
|
ret = wl1271_acx_keep_alive_mode(wl, false);
|
2010-03-26 18:53:23 +08:00
|
|
|
if (ret < 0)
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2010-11-22 18:59:08 +08:00
|
|
|
|
|
|
|
/* restore the bssid filter and go to dummy bssid */
|
|
|
|
wl1271_unjoin(wl);
|
|
|
|
wl1271_dummy_join(wl);
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
ret = wl1271_bss_erp_info_changed(wl, bss_conf, changed);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-07-08 22:50:02 +08:00
|
|
|
if (changed & BSS_CHANGED_ARP_FILTER) {
|
|
|
|
__be32 addr = bss_conf->arp_addr_list[0];
|
|
|
|
WARN_ON(wl->bss_type != BSS_TYPE_STA_BSS);
|
|
|
|
|
2010-12-09 17:31:27 +08:00
|
|
|
if (bss_conf->arp_addr_cnt == 1 &&
|
|
|
|
bss_conf->arp_filter_enabled) {
|
|
|
|
/*
|
|
|
|
* The template should have been configured only upon
|
|
|
|
* association. however, it seems that the correct ip
|
|
|
|
* isn't being set (when sending), so we have to
|
|
|
|
* reconfigure the template upon every ip change.
|
|
|
|
*/
|
|
|
|
ret = wl1271_cmd_build_arp_rsp(wl, addr);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("build arp rsp failed: %d", ret);
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2010-12-09 17:31:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = wl1271_acx_arp_ip_filter(wl,
|
2011-01-25 02:19:03 +08:00
|
|
|
ACX_ARP_FILTER_ARP_FILTERING,
|
2010-12-09 17:31:27 +08:00
|
|
|
addr);
|
|
|
|
} else
|
|
|
|
ret = wl1271_acx_arp_ip_filter(wl, 0, addr);
|
2010-07-08 22:50:02 +08:00
|
|
|
|
|
|
|
if (ret < 0)
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2010-07-08 22:50:02 +08:00
|
|
|
}
|
|
|
|
|
2010-02-18 19:25:51 +08:00
|
|
|
if (do_join) {
|
2010-05-07 16:39:00 +08:00
|
|
|
ret = wl1271_join(wl, set_assoc);
|
2010-02-18 19:25:51 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("cmd join failed %d", ret);
|
2010-10-17 01:07:21 +08:00
|
|
|
goto out;
|
2010-02-18 19:25:51 +08:00
|
|
|
}
|
2010-03-26 18:53:32 +08:00
|
|
|
}
|
|
|
|
|
2010-10-17 01:07:21 +08:00
|
|
|
out:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wl1271_op_bss_info_changed(struct ieee80211_hw *hw,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_bss_conf *bss_conf,
|
|
|
|
u32 changed)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
bool is_ap = (wl->bss_type == BSS_TYPE_AP_BSS);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 bss info changed 0x%x",
|
|
|
|
(int)changed);
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF))
|
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2010-10-17 01:07:21 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (is_ap)
|
|
|
|
wl1271_bss_info_changed_ap(wl, vif, bss_conf, changed);
|
|
|
|
else
|
|
|
|
wl1271_bss_info_changed_sta(wl, vif, bss_conf, changed);
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
}
|
|
|
|
|
2010-02-18 19:25:41 +08:00
|
|
|
static int wl1271_op_conf_tx(struct ieee80211_hw *hw, u16 queue,
|
|
|
|
const struct ieee80211_tx_queue_params *params)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
2010-03-18 18:26:38 +08:00
|
|
|
u8 ps_scheme;
|
2010-10-17 02:33:45 +08:00
|
|
|
int ret = 0;
|
2010-02-18 19:25:41 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 conf tx %d", queue);
|
|
|
|
|
2010-03-18 18:26:38 +08:00
|
|
|
if (params->uapsd)
|
|
|
|
ps_scheme = CONF_PS_SCHEME_UPSD_TRIGGER;
|
|
|
|
else
|
|
|
|
ps_scheme = CONF_PS_SCHEME_LEGACY;
|
|
|
|
|
2010-10-17 02:33:45 +08:00
|
|
|
if (wl->state == WL1271_STATE_OFF) {
|
|
|
|
/*
|
|
|
|
* If the state is off, the parameters will be recorded and
|
|
|
|
* configured on init. This happens in AP-mode.
|
|
|
|
*/
|
|
|
|
struct conf_tx_ac_category *conf_ac =
|
|
|
|
&wl->conf.tx.ac_conf[wl1271_tx_get_queue(queue)];
|
|
|
|
struct conf_tx_tid *conf_tid =
|
|
|
|
&wl->conf.tx.tid_conf[wl1271_tx_get_queue(queue)];
|
|
|
|
|
|
|
|
conf_ac->ac = wl1271_tx_get_queue(queue);
|
|
|
|
conf_ac->cw_min = (u8)params->cw_min;
|
|
|
|
conf_ac->cw_max = params->cw_max;
|
|
|
|
conf_ac->aifsn = params->aifs;
|
|
|
|
conf_ac->tx_op_limit = params->txop << 5;
|
|
|
|
|
|
|
|
conf_tid->queue_id = wl1271_tx_get_queue(queue);
|
|
|
|
conf_tid->channel_type = CONF_CHANNEL_TYPE_EDCF;
|
|
|
|
conf_tid->tsid = wl1271_tx_get_queue(queue);
|
|
|
|
conf_tid->ps_scheme = ps_scheme;
|
|
|
|
conf_tid->ack_policy = CONF_ACK_POLICY_LEGACY;
|
|
|
|
conf_tid->apsd_conf[0] = 0;
|
|
|
|
conf_tid->apsd_conf[1] = 0;
|
2011-03-24 04:22:15 +08:00
|
|
|
goto out;
|
|
|
|
}
|
2010-10-17 02:33:45 +08:00
|
|
|
|
2011-03-24 04:22:15 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2010-10-17 02:33:45 +08:00
|
|
|
|
2011-03-24 04:22:15 +08:00
|
|
|
/*
|
|
|
|
* the txop is confed in units of 32us by the mac80211,
|
|
|
|
* we need us
|
|
|
|
*/
|
|
|
|
ret = wl1271_acx_ac_cfg(wl, wl1271_tx_get_queue(queue),
|
|
|
|
params->cw_min, params->cw_max,
|
|
|
|
params->aifs, params->txop << 5);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
|
|
|
|
ret = wl1271_acx_tid_cfg(wl, wl1271_tx_get_queue(queue),
|
|
|
|
CONF_CHANNEL_TYPE_EDCF,
|
|
|
|
wl1271_tx_get_queue(queue),
|
|
|
|
ps_scheme, CONF_ACK_POLICY_LEGACY,
|
|
|
|
0, 0);
|
2010-02-18 19:25:47 +08:00
|
|
|
|
|
|
|
out_sleep:
|
2011-03-24 04:22:15 +08:00
|
|
|
wl1271_ps_elp_sleep(wl);
|
2010-02-18 19:25:41 +08:00
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-07-08 22:49:57 +08:00
|
|
|
static u64 wl1271_op_get_tsf(struct ieee80211_hw *hw)
|
|
|
|
{
|
|
|
|
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
u64 mactime = ULLONG_MAX;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 get tsf");
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2010-10-26 19:24:39 +08:00
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF))
|
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2010-07-08 22:49:57 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1271_acx_tsf_info(wl, &mactime);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
|
|
|
|
out_sleep:
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
return mactime;
|
|
|
|
}
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-07-29 04:41:06 +08:00
|
|
|
static int wl1271_op_get_survey(struct ieee80211_hw *hw, int idx,
|
|
|
|
struct survey_info *survey)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
struct ieee80211_conf *conf = &hw->conf;
|
2010-10-26 19:24:38 +08:00
|
|
|
|
2010-07-29 04:41:06 +08:00
|
|
|
if (idx != 0)
|
|
|
|
return -ENOENT;
|
2010-10-26 19:24:38 +08:00
|
|
|
|
2010-07-29 04:41:06 +08:00
|
|
|
survey->channel = conf->channel;
|
|
|
|
survey->filled = SURVEY_INFO_NOISE_DBM;
|
|
|
|
survey->noise = wl->noise;
|
2010-10-26 19:24:38 +08:00
|
|
|
|
2010-07-29 04:41:06 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-23 06:22:29 +08:00
|
|
|
static int wl1271_allocate_sta(struct wl1271 *wl,
|
2010-10-17 02:21:23 +08:00
|
|
|
struct ieee80211_sta *sta,
|
|
|
|
u8 *hlid)
|
|
|
|
{
|
|
|
|
struct wl1271_station *wl_sta;
|
|
|
|
int id;
|
|
|
|
|
|
|
|
id = find_first_zero_bit(wl->ap_hlid_map, AP_MAX_STATIONS);
|
|
|
|
if (id >= AP_MAX_STATIONS) {
|
|
|
|
wl1271_warning("could not allocate HLID - too much stations");
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl_sta = (struct wl1271_station *)sta->drv_priv;
|
|
|
|
__set_bit(id, wl->ap_hlid_map);
|
|
|
|
wl_sta->hlid = WL1271_AP_STA_HLID_START + id;
|
|
|
|
*hlid = wl_sta->hlid;
|
2011-02-23 06:22:31 +08:00
|
|
|
memcpy(wl->links[wl_sta->hlid].addr, sta->addr, ETH_ALEN);
|
2010-10-17 02:21:23 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-23 06:22:29 +08:00
|
|
|
static void wl1271_free_sta(struct wl1271 *wl, u8 hlid)
|
2010-10-17 02:21:23 +08:00
|
|
|
{
|
|
|
|
int id = hlid - WL1271_AP_STA_HLID_START;
|
|
|
|
|
2011-02-23 06:22:29 +08:00
|
|
|
if (WARN_ON(!test_bit(id, wl->ap_hlid_map)))
|
|
|
|
return;
|
|
|
|
|
2010-10-17 02:21:23 +08:00
|
|
|
__clear_bit(id, wl->ap_hlid_map);
|
2011-02-23 06:22:31 +08:00
|
|
|
memset(wl->links[hlid].addr, 0, ETH_ALEN);
|
2011-02-23 06:22:26 +08:00
|
|
|
wl1271_tx_reset_link_queues(wl, hlid);
|
2011-02-23 06:22:31 +08:00
|
|
|
__clear_bit(hlid, &wl->ap_ps_map);
|
|
|
|
__clear_bit(hlid, (unsigned long *)&wl->ap_fw_ps_map);
|
2010-10-17 02:21:23 +08:00
|
|
|
}
|
|
|
|
|
2011-04-27 04:21:51 +08:00
|
|
|
bool wl1271_is_active_sta(struct wl1271 *wl, u8 hlid)
|
|
|
|
{
|
|
|
|
int id = hlid - WL1271_AP_STA_HLID_START;
|
|
|
|
return test_bit(id, wl->ap_hlid_map);
|
|
|
|
}
|
|
|
|
|
2010-10-17 02:21:23 +08:00
|
|
|
static int wl1271_op_sta_add(struct ieee80211_hw *hw,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_sta *sta)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
int ret = 0;
|
|
|
|
u8 hlid;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (wl->bss_type != BSS_TYPE_AP_BSS)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 add sta %d", (int)sta->aid);
|
|
|
|
|
2011-02-23 06:22:29 +08:00
|
|
|
ret = wl1271_allocate_sta(wl, sta, &hlid);
|
2010-10-17 02:21:23 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2010-10-17 02:21:23 +08:00
|
|
|
if (ret < 0)
|
2011-02-23 06:22:29 +08:00
|
|
|
goto out_free_sta;
|
2010-10-17 02:21:23 +08:00
|
|
|
|
|
|
|
ret = wl1271_cmd_add_sta(wl, sta, hlid);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
|
|
|
|
out_sleep:
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
2011-02-23 06:22:29 +08:00
|
|
|
out_free_sta:
|
|
|
|
if (ret < 0)
|
|
|
|
wl1271_free_sta(wl, hlid);
|
|
|
|
|
2010-10-17 02:21:23 +08:00
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1271_op_sta_remove(struct ieee80211_hw *hw,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_sta *sta)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
struct wl1271_station *wl_sta;
|
|
|
|
int ret = 0, id;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (wl->bss_type != BSS_TYPE_AP_BSS)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
wl1271_debug(DEBUG_MAC80211, "mac80211 remove sta %d", (int)sta->aid);
|
|
|
|
|
|
|
|
wl_sta = (struct wl1271_station *)sta->drv_priv;
|
|
|
|
id = wl_sta->hlid - WL1271_AP_STA_HLID_START;
|
|
|
|
if (WARN_ON(!test_bit(id, wl->ap_hlid_map)))
|
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2010-10-17 02:21:23 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1271_cmd_remove_sta(wl, wl_sta->hlid);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
|
2011-02-23 06:22:29 +08:00
|
|
|
wl1271_free_sta(wl, wl_sta->hlid);
|
2010-10-17 02:21:23 +08:00
|
|
|
|
|
|
|
out_sleep:
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-03-22 01:26:41 +08:00
|
|
|
static int wl1271_op_ampdu_action(struct ieee80211_hw *hw,
|
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
enum ieee80211_ampdu_mlme_action action,
|
|
|
|
struct ieee80211_sta *sta, u16 tid, u16 *ssn,
|
|
|
|
u8 buf_size)
|
2011-01-23 14:27:23 +08:00
|
|
|
{
|
|
|
|
struct wl1271 *wl = hw->priv;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
if (unlikely(wl->state == WL1271_STATE_OFF)) {
|
|
|
|
ret = -EAGAIN;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2011-01-23 14:27:23 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
switch (action) {
|
|
|
|
case IEEE80211_AMPDU_RX_START:
|
|
|
|
if (wl->ba_support) {
|
|
|
|
ret = wl1271_acx_set_ba_receiver_session(wl, tid, *ssn,
|
|
|
|
true);
|
|
|
|
if (!ret)
|
|
|
|
wl->ba_rx_bitmap |= BIT(tid);
|
|
|
|
} else {
|
|
|
|
ret = -ENOTSUPP;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IEEE80211_AMPDU_RX_STOP:
|
|
|
|
ret = wl1271_acx_set_ba_receiver_session(wl, tid, 0, false);
|
|
|
|
if (!ret)
|
|
|
|
wl->ba_rx_bitmap &= ~BIT(tid);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The BA initiator session management in FW independently.
|
|
|
|
* Falling break here on purpose for all TX APDU commands.
|
|
|
|
*/
|
|
|
|
case IEEE80211_AMPDU_TX_START:
|
|
|
|
case IEEE80211_AMPDU_TX_STOP:
|
|
|
|
case IEEE80211_AMPDU_TX_OPERATIONAL:
|
|
|
|
ret = -EINVAL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
wl1271_error("Incorrect ampdu action id=%x\n", action);
|
|
|
|
ret = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
/* can't be const, mac80211 writes to this */
|
|
|
|
static struct ieee80211_rate wl1271_rates[] = {
|
|
|
|
{ .bitrate = 10,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_1MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_1MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 20,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_2MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_2MBPS,
|
2009-08-06 21:25:28 +08:00
|
|
|
.flags = IEEE80211_RATE_SHORT_PREAMBLE },
|
|
|
|
{ .bitrate = 55,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_5_5MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_5_5MBPS,
|
2009-08-06 21:25:28 +08:00
|
|
|
.flags = IEEE80211_RATE_SHORT_PREAMBLE },
|
|
|
|
{ .bitrate = 110,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_11MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_11MBPS,
|
2009-08-06 21:25:28 +08:00
|
|
|
.flags = IEEE80211_RATE_SHORT_PREAMBLE },
|
|
|
|
{ .bitrate = 60,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_6MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_6MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 90,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_9MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_9MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 120,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_12MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_12MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 180,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_18MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_18MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 240,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_24MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_24MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 360,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_36MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_36MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 480,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_48MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_48MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
{ .bitrate = 540,
|
2009-10-13 17:47:39 +08:00
|
|
|
.hw_value = CONF_HW_BIT_RATE_54MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_54MBPS, },
|
2009-08-06 21:25:28 +08:00
|
|
|
};
|
|
|
|
|
2010-11-10 18:27:20 +08:00
|
|
|
/* can't be const, mac80211 writes to this */
|
2009-08-06 21:25:28 +08:00
|
|
|
static struct ieee80211_channel wl1271_channels[] = {
|
2009-12-11 21:40:46 +08:00
|
|
|
{ .hw_value = 1, .center_freq = 2412, .max_power = 25 },
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 2, .center_freq = 2417, .max_power = 25 },
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 3, .center_freq = 2422, .max_power = 25 },
|
|
|
|
{ .hw_value = 4, .center_freq = 2427, .max_power = 25 },
|
|
|
|
{ .hw_value = 5, .center_freq = 2432, .max_power = 25 },
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 6, .center_freq = 2437, .max_power = 25 },
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 7, .center_freq = 2442, .max_power = 25 },
|
|
|
|
{ .hw_value = 8, .center_freq = 2447, .max_power = 25 },
|
|
|
|
{ .hw_value = 9, .center_freq = 2452, .max_power = 25 },
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 10, .center_freq = 2457, .max_power = 25 },
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 11, .center_freq = 2462, .max_power = 25 },
|
|
|
|
{ .hw_value = 12, .center_freq = 2467, .max_power = 25 },
|
|
|
|
{ .hw_value = 13, .center_freq = 2472, .max_power = 25 },
|
2011-01-19 03:39:52 +08:00
|
|
|
{ .hw_value = 14, .center_freq = 2484, .max_power = 25 },
|
2009-08-06 21:25:28 +08:00
|
|
|
};
|
|
|
|
|
2010-03-26 18:53:11 +08:00
|
|
|
/* mapping to indexes for wl1271_rates */
|
2010-05-20 16:38:11 +08:00
|
|
|
static const u8 wl1271_rate_to_idx_2ghz[] = {
|
2010-03-26 18:53:11 +08:00
|
|
|
/* MCS rates are used only with 11n */
|
2010-10-13 22:09:41 +08:00
|
|
|
7, /* CONF_HW_RXTX_RATE_MCS7 */
|
|
|
|
6, /* CONF_HW_RXTX_RATE_MCS6 */
|
|
|
|
5, /* CONF_HW_RXTX_RATE_MCS5 */
|
|
|
|
4, /* CONF_HW_RXTX_RATE_MCS4 */
|
|
|
|
3, /* CONF_HW_RXTX_RATE_MCS3 */
|
|
|
|
2, /* CONF_HW_RXTX_RATE_MCS2 */
|
|
|
|
1, /* CONF_HW_RXTX_RATE_MCS1 */
|
|
|
|
0, /* CONF_HW_RXTX_RATE_MCS0 */
|
2010-03-26 18:53:11 +08:00
|
|
|
|
|
|
|
11, /* CONF_HW_RXTX_RATE_54 */
|
|
|
|
10, /* CONF_HW_RXTX_RATE_48 */
|
|
|
|
9, /* CONF_HW_RXTX_RATE_36 */
|
|
|
|
8, /* CONF_HW_RXTX_RATE_24 */
|
|
|
|
|
|
|
|
/* TI-specific rate */
|
|
|
|
CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_22 */
|
|
|
|
|
|
|
|
7, /* CONF_HW_RXTX_RATE_18 */
|
|
|
|
6, /* CONF_HW_RXTX_RATE_12 */
|
|
|
|
3, /* CONF_HW_RXTX_RATE_11 */
|
|
|
|
5, /* CONF_HW_RXTX_RATE_9 */
|
|
|
|
4, /* CONF_HW_RXTX_RATE_6 */
|
|
|
|
2, /* CONF_HW_RXTX_RATE_5_5 */
|
|
|
|
1, /* CONF_HW_RXTX_RATE_2 */
|
|
|
|
0 /* CONF_HW_RXTX_RATE_1 */
|
|
|
|
};
|
|
|
|
|
2010-10-13 22:09:39 +08:00
|
|
|
/* 11n STA capabilities */
|
|
|
|
#define HW_RX_HIGHEST_RATE 72
|
|
|
|
|
2010-11-08 19:20:10 +08:00
|
|
|
#ifdef CONFIG_WL12XX_HT
|
|
|
|
#define WL12XX_HT_CAP { \
|
2011-03-13 17:24:40 +08:00
|
|
|
.cap = IEEE80211_HT_CAP_GRN_FLD | IEEE80211_HT_CAP_SGI_20 | \
|
|
|
|
(1 << IEEE80211_HT_CAP_RX_STBC_SHIFT), \
|
2010-10-13 22:09:39 +08:00
|
|
|
.ht_supported = true, \
|
|
|
|
.ampdu_factor = IEEE80211_HT_MAX_AMPDU_8K, \
|
|
|
|
.ampdu_density = IEEE80211_HT_MPDU_DENSITY_8, \
|
|
|
|
.mcs = { \
|
|
|
|
.rx_mask = { 0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, }, \
|
|
|
|
.rx_highest = cpu_to_le16(HW_RX_HIGHEST_RATE), \
|
|
|
|
.tx_params = IEEE80211_HT_MCS_TX_DEFINED, \
|
|
|
|
}, \
|
|
|
|
}
|
2010-10-13 22:09:41 +08:00
|
|
|
#else
|
2010-11-08 19:20:10 +08:00
|
|
|
#define WL12XX_HT_CAP { \
|
2010-10-13 22:09:41 +08:00
|
|
|
.ht_supported = false, \
|
|
|
|
}
|
|
|
|
#endif
|
2010-10-13 22:09:39 +08:00
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
/* can't be const, mac80211 writes to this */
|
|
|
|
static struct ieee80211_supported_band wl1271_band_2ghz = {
|
|
|
|
.channels = wl1271_channels,
|
|
|
|
.n_channels = ARRAY_SIZE(wl1271_channels),
|
|
|
|
.bitrates = wl1271_rates,
|
|
|
|
.n_bitrates = ARRAY_SIZE(wl1271_rates),
|
2010-11-08 19:20:10 +08:00
|
|
|
.ht_cap = WL12XX_HT_CAP,
|
2009-08-06 21:25:28 +08:00
|
|
|
};
|
|
|
|
|
2009-10-13 17:47:48 +08:00
|
|
|
/* 5 GHz data rates for WL1273 */
|
|
|
|
static struct ieee80211_rate wl1271_rates_5ghz[] = {
|
|
|
|
{ .bitrate = 60,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_6MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_6MBPS, },
|
|
|
|
{ .bitrate = 90,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_9MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_9MBPS, },
|
|
|
|
{ .bitrate = 120,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_12MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_12MBPS, },
|
|
|
|
{ .bitrate = 180,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_18MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_18MBPS, },
|
|
|
|
{ .bitrate = 240,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_24MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_24MBPS, },
|
|
|
|
{ .bitrate = 360,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_36MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_36MBPS, },
|
|
|
|
{ .bitrate = 480,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_48MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_48MBPS, },
|
|
|
|
{ .bitrate = 540,
|
|
|
|
.hw_value = CONF_HW_BIT_RATE_54MBPS,
|
|
|
|
.hw_value_short = CONF_HW_BIT_RATE_54MBPS, },
|
|
|
|
};
|
|
|
|
|
2010-11-10 18:27:20 +08:00
|
|
|
/* 5 GHz band channels for WL1273 */
|
2009-10-13 17:47:48 +08:00
|
|
|
static struct ieee80211_channel wl1271_channels_5ghz[] = {
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 7, .center_freq = 5035},
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 8, .center_freq = 5040},
|
|
|
|
{ .hw_value = 9, .center_freq = 5045},
|
|
|
|
{ .hw_value = 11, .center_freq = 5055},
|
|
|
|
{ .hw_value = 12, .center_freq = 5060},
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 16, .center_freq = 5080},
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 34, .center_freq = 5170},
|
|
|
|
{ .hw_value = 36, .center_freq = 5180},
|
|
|
|
{ .hw_value = 38, .center_freq = 5190},
|
|
|
|
{ .hw_value = 40, .center_freq = 5200},
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 42, .center_freq = 5210},
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 44, .center_freq = 5220},
|
|
|
|
{ .hw_value = 46, .center_freq = 5230},
|
|
|
|
{ .hw_value = 48, .center_freq = 5240},
|
|
|
|
{ .hw_value = 52, .center_freq = 5260},
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 56, .center_freq = 5280},
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 60, .center_freq = 5300},
|
|
|
|
{ .hw_value = 64, .center_freq = 5320},
|
|
|
|
{ .hw_value = 100, .center_freq = 5500},
|
|
|
|
{ .hw_value = 104, .center_freq = 5520},
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 108, .center_freq = 5540},
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 112, .center_freq = 5560},
|
|
|
|
{ .hw_value = 116, .center_freq = 5580},
|
|
|
|
{ .hw_value = 120, .center_freq = 5600},
|
|
|
|
{ .hw_value = 124, .center_freq = 5620},
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 128, .center_freq = 5640},
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 132, .center_freq = 5660},
|
|
|
|
{ .hw_value = 136, .center_freq = 5680},
|
|
|
|
{ .hw_value = 140, .center_freq = 5700},
|
|
|
|
{ .hw_value = 149, .center_freq = 5745},
|
2010-08-10 14:22:02 +08:00
|
|
|
{ .hw_value = 153, .center_freq = 5765},
|
2010-11-10 18:27:20 +08:00
|
|
|
{ .hw_value = 157, .center_freq = 5785},
|
|
|
|
{ .hw_value = 161, .center_freq = 5805},
|
2009-10-13 17:47:48 +08:00
|
|
|
{ .hw_value = 165, .center_freq = 5825},
|
|
|
|
};
|
|
|
|
|
2010-03-26 18:53:11 +08:00
|
|
|
/* mapping to indexes for wl1271_rates_5ghz */
|
2010-05-20 16:38:11 +08:00
|
|
|
static const u8 wl1271_rate_to_idx_5ghz[] = {
|
2010-03-26 18:53:11 +08:00
|
|
|
/* MCS rates are used only with 11n */
|
2010-10-13 22:09:41 +08:00
|
|
|
7, /* CONF_HW_RXTX_RATE_MCS7 */
|
|
|
|
6, /* CONF_HW_RXTX_RATE_MCS6 */
|
|
|
|
5, /* CONF_HW_RXTX_RATE_MCS5 */
|
|
|
|
4, /* CONF_HW_RXTX_RATE_MCS4 */
|
|
|
|
3, /* CONF_HW_RXTX_RATE_MCS3 */
|
|
|
|
2, /* CONF_HW_RXTX_RATE_MCS2 */
|
|
|
|
1, /* CONF_HW_RXTX_RATE_MCS1 */
|
|
|
|
0, /* CONF_HW_RXTX_RATE_MCS0 */
|
2010-03-26 18:53:11 +08:00
|
|
|
|
|
|
|
7, /* CONF_HW_RXTX_RATE_54 */
|
|
|
|
6, /* CONF_HW_RXTX_RATE_48 */
|
|
|
|
5, /* CONF_HW_RXTX_RATE_36 */
|
|
|
|
4, /* CONF_HW_RXTX_RATE_24 */
|
|
|
|
|
|
|
|
/* TI-specific rate */
|
|
|
|
CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_22 */
|
|
|
|
|
|
|
|
3, /* CONF_HW_RXTX_RATE_18 */
|
|
|
|
2, /* CONF_HW_RXTX_RATE_12 */
|
|
|
|
CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_11 */
|
|
|
|
1, /* CONF_HW_RXTX_RATE_9 */
|
|
|
|
0, /* CONF_HW_RXTX_RATE_6 */
|
|
|
|
CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_5_5 */
|
|
|
|
CONF_HW_RXTX_RATE_UNSUPPORTED, /* CONF_HW_RXTX_RATE_2 */
|
|
|
|
CONF_HW_RXTX_RATE_UNSUPPORTED /* CONF_HW_RXTX_RATE_1 */
|
|
|
|
};
|
2009-10-13 17:47:48 +08:00
|
|
|
|
|
|
|
static struct ieee80211_supported_band wl1271_band_5ghz = {
|
|
|
|
.channels = wl1271_channels_5ghz,
|
|
|
|
.n_channels = ARRAY_SIZE(wl1271_channels_5ghz),
|
|
|
|
.bitrates = wl1271_rates_5ghz,
|
|
|
|
.n_bitrates = ARRAY_SIZE(wl1271_rates_5ghz),
|
2010-11-08 19:20:10 +08:00
|
|
|
.ht_cap = WL12XX_HT_CAP,
|
2009-10-13 17:47:48 +08:00
|
|
|
};
|
|
|
|
|
2010-05-20 16:38:11 +08:00
|
|
|
static const u8 *wl1271_band_rate_to_idx[] = {
|
2010-03-26 18:53:11 +08:00
|
|
|
[IEEE80211_BAND_2GHZ] = wl1271_rate_to_idx_2ghz,
|
|
|
|
[IEEE80211_BAND_5GHZ] = wl1271_rate_to_idx_5ghz
|
|
|
|
};
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
static const struct ieee80211_ops wl1271_ops = {
|
|
|
|
.start = wl1271_op_start,
|
|
|
|
.stop = wl1271_op_stop,
|
|
|
|
.add_interface = wl1271_op_add_interface,
|
|
|
|
.remove_interface = wl1271_op_remove_interface,
|
|
|
|
.config = wl1271_op_config,
|
2009-10-09 02:56:31 +08:00
|
|
|
.prepare_multicast = wl1271_op_prepare_multicast,
|
2009-08-06 21:25:28 +08:00
|
|
|
.configure_filter = wl1271_op_configure_filter,
|
|
|
|
.tx = wl1271_op_tx,
|
|
|
|
.set_key = wl1271_op_set_key,
|
|
|
|
.hw_scan = wl1271_op_hw_scan,
|
|
|
|
.bss_info_changed = wl1271_op_bss_info_changed,
|
2010-11-08 17:51:07 +08:00
|
|
|
.set_frag_threshold = wl1271_op_set_frag_threshold,
|
2009-08-06 21:25:28 +08:00
|
|
|
.set_rts_threshold = wl1271_op_set_rts_threshold,
|
2010-02-18 19:25:41 +08:00
|
|
|
.conf_tx = wl1271_op_conf_tx,
|
2010-07-08 22:49:57 +08:00
|
|
|
.get_tsf = wl1271_op_get_tsf,
|
2010-07-29 04:41:06 +08:00
|
|
|
.get_survey = wl1271_op_get_survey,
|
2010-10-17 02:21:23 +08:00
|
|
|
.sta_add = wl1271_op_sta_add,
|
|
|
|
.sta_remove = wl1271_op_sta_remove,
|
2011-01-23 14:27:23 +08:00
|
|
|
.ampdu_action = wl1271_op_ampdu_action,
|
2010-02-18 19:25:53 +08:00
|
|
|
CFG80211_TESTMODE_CMD(wl1271_tm_cmd)
|
2009-08-06 21:25:28 +08:00
|
|
|
};
|
|
|
|
|
2010-03-26 18:53:11 +08:00
|
|
|
|
2010-10-14 17:00:04 +08:00
|
|
|
u8 wl1271_rate_to_idx(int rate, enum ieee80211_band band)
|
2010-03-26 18:53:11 +08:00
|
|
|
{
|
|
|
|
u8 idx;
|
|
|
|
|
2010-10-14 17:00:04 +08:00
|
|
|
BUG_ON(band >= sizeof(wl1271_band_rate_to_idx)/sizeof(u8 *));
|
2010-03-26 18:53:11 +08:00
|
|
|
|
|
|
|
if (unlikely(rate >= CONF_HW_RXTX_RATE_MAX)) {
|
|
|
|
wl1271_error("Illegal RX rate from HW: %d", rate);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-10-14 17:00:04 +08:00
|
|
|
idx = wl1271_band_rate_to_idx[band][rate];
|
2010-03-26 18:53:11 +08:00
|
|
|
if (unlikely(idx == CONF_HW_RXTX_RATE_UNSUPPORTED)) {
|
|
|
|
wl1271_error("Unsupported RX rate from HW: %d", rate);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2010-03-18 18:26:32 +08:00
|
|
|
static ssize_t wl1271_sysfs_show_bt_coex_state(struct device *dev,
|
|
|
|
struct device_attribute *attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = dev_get_drvdata(dev);
|
|
|
|
ssize_t len;
|
|
|
|
|
2010-08-10 12:38:35 +08:00
|
|
|
len = PAGE_SIZE;
|
2010-03-18 18:26:32 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
len = snprintf(buf, len, "%d\n\n0 - off\n1 - on\n",
|
|
|
|
wl->sg_enabled);
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return len;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t wl1271_sysfs_store_bt_coex_state(struct device *dev,
|
|
|
|
struct device_attribute *attr,
|
|
|
|
const char *buf, size_t count)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = dev_get_drvdata(dev);
|
|
|
|
unsigned long res;
|
|
|
|
int ret;
|
|
|
|
|
2011-04-01 22:49:54 +08:00
|
|
|
ret = kstrtoul(buf, 10, &res);
|
2010-03-18 18:26:32 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_warning("incorrect value written to bt_coex_mode");
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
res = !!res;
|
|
|
|
|
|
|
|
if (res == wl->sg_enabled)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
wl->sg_enabled = res;
|
|
|
|
|
|
|
|
if (wl->state == WL1271_STATE_OFF)
|
|
|
|
goto out;
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
ret = wl1271_ps_elp_wakeup(wl);
|
2010-03-18 18:26:32 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
wl1271_acx_sg_enable(wl, wl->sg_enabled);
|
|
|
|
wl1271_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DEVICE_ATTR(bt_coex_state, S_IRUGO | S_IWUSR,
|
|
|
|
wl1271_sysfs_show_bt_coex_state,
|
|
|
|
wl1271_sysfs_store_bt_coex_state);
|
|
|
|
|
2010-05-07 16:38:58 +08:00
|
|
|
static ssize_t wl1271_sysfs_show_hw_pg_ver(struct device *dev,
|
|
|
|
struct device_attribute *attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct wl1271 *wl = dev_get_drvdata(dev);
|
|
|
|
ssize_t len;
|
|
|
|
|
2010-08-10 12:38:35 +08:00
|
|
|
len = PAGE_SIZE;
|
2010-05-07 16:38:58 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
if (wl->hw_pg_ver >= 0)
|
|
|
|
len = snprintf(buf, len, "%d\n", wl->hw_pg_ver);
|
|
|
|
else
|
|
|
|
len = snprintf(buf, len, "n/a\n");
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DEVICE_ATTR(hw_pg_ver, S_IRUGO | S_IWUSR,
|
|
|
|
wl1271_sysfs_show_hw_pg_ver, NULL);
|
|
|
|
|
2010-02-22 14:38:21 +08:00
|
|
|
int wl1271_register_hw(struct wl1271 *wl)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (wl->mac80211_registered)
|
|
|
|
return 0;
|
|
|
|
|
2010-10-17 03:49:52 +08:00
|
|
|
ret = wl1271_fetch_nvs(wl);
|
|
|
|
if (ret == 0) {
|
2011-03-06 22:32:10 +08:00
|
|
|
/* NOTE: The wl->nvs->nvs element must be first, in
|
|
|
|
* order to simplify the casting, we assume it is at
|
|
|
|
* the beginning of the wl->nvs structure.
|
|
|
|
*/
|
|
|
|
u8 *nvs_ptr = (u8 *)wl->nvs;
|
2010-10-17 03:49:52 +08:00
|
|
|
|
|
|
|
wl->mac_addr[0] = nvs_ptr[11];
|
|
|
|
wl->mac_addr[1] = nvs_ptr[10];
|
|
|
|
wl->mac_addr[2] = nvs_ptr[6];
|
|
|
|
wl->mac_addr[3] = nvs_ptr[5];
|
|
|
|
wl->mac_addr[4] = nvs_ptr[4];
|
|
|
|
wl->mac_addr[5] = nvs_ptr[3];
|
|
|
|
}
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
SET_IEEE80211_PERM_ADDR(wl->hw, wl->mac_addr);
|
|
|
|
|
|
|
|
ret = ieee80211_register_hw(wl->hw);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("unable to register mac80211 hw: %d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->mac80211_registered = true;
|
|
|
|
|
2010-11-24 18:53:16 +08:00
|
|
|
wl1271_debugfs_init(wl);
|
|
|
|
|
2010-07-27 08:30:09 +08:00
|
|
|
register_netdevice_notifier(&wl1271_dev_notifier);
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
wl1271_notice("loaded");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2010-02-22 14:38:26 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1271_register_hw);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-03-18 18:26:46 +08:00
|
|
|
void wl1271_unregister_hw(struct wl1271 *wl)
|
|
|
|
{
|
2011-01-14 19:48:46 +08:00
|
|
|
if (wl->state == WL1271_STATE_PLT)
|
|
|
|
__wl1271_plt_stop(wl);
|
|
|
|
|
2010-07-27 08:30:09 +08:00
|
|
|
unregister_netdevice_notifier(&wl1271_dev_notifier);
|
2010-03-18 18:26:46 +08:00
|
|
|
ieee80211_unregister_hw(wl->hw);
|
|
|
|
wl->mac80211_registered = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(wl1271_unregister_hw);
|
|
|
|
|
2010-02-22 14:38:21 +08:00
|
|
|
int wl1271_init_ieee80211(struct wl1271 *wl)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
2010-09-27 18:42:07 +08:00
|
|
|
static const u32 cipher_suites[] = {
|
|
|
|
WLAN_CIPHER_SUITE_WEP40,
|
|
|
|
WLAN_CIPHER_SUITE_WEP104,
|
|
|
|
WLAN_CIPHER_SUITE_TKIP,
|
|
|
|
WLAN_CIPHER_SUITE_CCMP,
|
|
|
|
WL1271_CIPHER_SUITE_GEM,
|
|
|
|
};
|
|
|
|
|
2009-10-09 02:56:20 +08:00
|
|
|
/* The tx descriptor buffer and the TKIP space. */
|
|
|
|
wl->hw->extra_tx_headroom = WL1271_TKIP_IV_SPACE +
|
|
|
|
sizeof(struct wl1271_tx_hw_descr);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
/* unit us */
|
|
|
|
/* FIXME: find a proper value */
|
|
|
|
wl->hw->channel_change_time = 10000;
|
2010-04-01 16:38:22 +08:00
|
|
|
wl->hw->max_listen_interval = wl->conf.conn.max_listen_interval;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
wl->hw->flags = IEEE80211_HW_SIGNAL_DBM |
|
2009-11-24 05:22:14 +08:00
|
|
|
IEEE80211_HW_BEACON_FILTER |
|
2010-02-22 14:38:41 +08:00
|
|
|
IEEE80211_HW_SUPPORTS_PS |
|
2010-03-18 18:26:38 +08:00
|
|
|
IEEE80211_HW_SUPPORTS_UAPSD |
|
2010-03-26 18:53:30 +08:00
|
|
|
IEEE80211_HW_HAS_RATE_CONTROL |
|
2010-04-09 16:07:30 +08:00
|
|
|
IEEE80211_HW_CONNECTION_MONITOR |
|
2011-02-02 17:20:05 +08:00
|
|
|
IEEE80211_HW_SUPPORTS_CQM_RSSI |
|
2011-02-23 06:22:28 +08:00
|
|
|
IEEE80211_HW_AP_LINK_PS;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-09-27 18:42:07 +08:00
|
|
|
wl->hw->wiphy->cipher_suites = cipher_suites;
|
|
|
|
wl->hw->wiphy->n_cipher_suites = ARRAY_SIZE(cipher_suites);
|
|
|
|
|
2009-12-11 21:41:04 +08:00
|
|
|
wl->hw->wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
|
2010-10-17 03:53:24 +08:00
|
|
|
BIT(NL80211_IFTYPE_ADHOC) | BIT(NL80211_IFTYPE_AP);
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->hw->wiphy->max_scan_ssids = 1;
|
2010-12-09 22:54:59 +08:00
|
|
|
/*
|
|
|
|
* Maximum length of elements in scanning probe request templates
|
|
|
|
* should be the maximum length possible for a template, without
|
|
|
|
* the IEEE80211 header of the template
|
|
|
|
*/
|
|
|
|
wl->hw->wiphy->max_scan_ie_len = WL1271_CMD_TEMPL_MAX_SIZE -
|
|
|
|
sizeof(struct ieee80211_header);
|
2011-01-12 01:25:18 +08:00
|
|
|
|
2011-03-22 05:16:14 +08:00
|
|
|
/* make sure all our channels fit in the scanned_ch bitmask */
|
|
|
|
BUILD_BUG_ON(ARRAY_SIZE(wl1271_channels) +
|
|
|
|
ARRAY_SIZE(wl1271_channels_5ghz) >
|
|
|
|
WL1271_MAX_CHANNELS);
|
2011-01-12 01:25:18 +08:00
|
|
|
/*
|
|
|
|
* We keep local copies of the band structs because we need to
|
|
|
|
* modify them on a per-device basis.
|
|
|
|
*/
|
|
|
|
memcpy(&wl->bands[IEEE80211_BAND_2GHZ], &wl1271_band_2ghz,
|
|
|
|
sizeof(wl1271_band_2ghz));
|
|
|
|
memcpy(&wl->bands[IEEE80211_BAND_5GHZ], &wl1271_band_5ghz,
|
|
|
|
sizeof(wl1271_band_5ghz));
|
|
|
|
|
|
|
|
wl->hw->wiphy->bands[IEEE80211_BAND_2GHZ] =
|
|
|
|
&wl->bands[IEEE80211_BAND_2GHZ];
|
|
|
|
wl->hw->wiphy->bands[IEEE80211_BAND_5GHZ] =
|
|
|
|
&wl->bands[IEEE80211_BAND_5GHZ];
|
2009-10-13 17:47:48 +08:00
|
|
|
|
2010-03-18 18:26:33 +08:00
|
|
|
wl->hw->queues = 4;
|
2010-03-26 18:53:12 +08:00
|
|
|
wl->hw->max_rates = 1;
|
2010-03-18 18:26:33 +08:00
|
|
|
|
2010-11-10 18:27:19 +08:00
|
|
|
wl->hw->wiphy->reg_notifier = wl1271_reg_notify;
|
|
|
|
|
2010-02-22 14:38:23 +08:00
|
|
|
SET_IEEE80211_DEV(wl->hw, wl1271_wl_to_dev(wl));
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-17 02:21:23 +08:00
|
|
|
wl->hw->sta_data_size = sizeof(struct wl1271_station);
|
|
|
|
|
2011-01-12 21:27:03 +08:00
|
|
|
wl->hw->max_rx_aggregation_subframes = 8;
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2010-02-22 14:38:26 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1271_init_ieee80211);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
#define WL1271_DEFAULT_CHANNEL 0
|
2010-02-18 19:25:57 +08:00
|
|
|
|
2010-02-22 14:38:21 +08:00
|
|
|
struct ieee80211_hw *wl1271_alloc_hw(void)
|
2009-08-06 21:25:28 +08:00
|
|
|
{
|
|
|
|
struct ieee80211_hw *hw;
|
2010-03-18 18:26:46 +08:00
|
|
|
struct platform_device *plat_dev = NULL;
|
2009-08-06 21:25:28 +08:00
|
|
|
struct wl1271 *wl;
|
2011-02-23 06:22:26 +08:00
|
|
|
int i, j, ret;
|
2010-09-30 19:28:27 +08:00
|
|
|
unsigned int order;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
|
|
|
hw = ieee80211_alloc_hw(sizeof(*wl), &wl1271_ops);
|
|
|
|
if (!hw) {
|
|
|
|
wl1271_error("could not alloc ieee80211_hw");
|
2010-03-18 18:26:31 +08:00
|
|
|
ret = -ENOMEM;
|
2010-03-18 18:26:46 +08:00
|
|
|
goto err_hw_alloc;
|
|
|
|
}
|
|
|
|
|
2010-05-16 05:16:39 +08:00
|
|
|
plat_dev = kmemdup(&wl1271_device, sizeof(wl1271_device), GFP_KERNEL);
|
2010-03-18 18:26:46 +08:00
|
|
|
if (!plat_dev) {
|
|
|
|
wl1271_error("could not allocate platform_device");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto err_plat_alloc;
|
2009-08-06 21:25:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
wl = hw->priv;
|
|
|
|
memset(wl, 0, sizeof(*wl));
|
|
|
|
|
2009-10-13 17:47:55 +08:00
|
|
|
INIT_LIST_HEAD(&wl->list);
|
|
|
|
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->hw = hw;
|
2010-03-18 18:26:46 +08:00
|
|
|
wl->plat_dev = plat_dev;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-12-13 15:52:37 +08:00
|
|
|
for (i = 0; i < NUM_TX_QUEUES; i++)
|
|
|
|
skb_queue_head_init(&wl->tx_queue[i]);
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2011-02-23 06:22:26 +08:00
|
|
|
for (i = 0; i < NUM_TX_QUEUES; i++)
|
|
|
|
for (j = 0; j < AP_MAX_LINKS; j++)
|
|
|
|
skb_queue_head_init(&wl->links[j].tx_queue[i]);
|
|
|
|
|
2011-03-01 21:14:41 +08:00
|
|
|
skb_queue_head_init(&wl->deferred_rx_queue);
|
|
|
|
skb_queue_head_init(&wl->deferred_tx_queue);
|
|
|
|
|
2009-10-09 02:56:21 +08:00
|
|
|
INIT_DELAYED_WORK(&wl->elp_work, wl1271_elp_work);
|
2010-07-08 22:50:00 +08:00
|
|
|
INIT_DELAYED_WORK(&wl->pspoll_work, wl1271_pspoll_work);
|
2011-03-01 21:14:41 +08:00
|
|
|
INIT_WORK(&wl->netstack_work, wl1271_netstack_work);
|
2010-09-30 16:43:28 +08:00
|
|
|
INIT_WORK(&wl->tx_work, wl1271_tx_work);
|
|
|
|
INIT_WORK(&wl->recovery_work, wl1271_recovery_work);
|
|
|
|
INIT_DELAYED_WORK(&wl->scan_complete_work, wl1271_scan_complete_work);
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->channel = WL1271_DEFAULT_CHANNEL;
|
2010-03-26 18:53:25 +08:00
|
|
|
wl->beacon_int = WL1271_DEFAULT_BEACON_INT;
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->default_key = 0;
|
|
|
|
wl->rx_counter = 0;
|
2010-10-17 00:45:07 +08:00
|
|
|
wl->rx_config = WL1271_DEFAULT_STA_RX_CONFIG;
|
|
|
|
wl->rx_filter = WL1271_DEFAULT_STA_RX_FILTER;
|
2009-11-03 02:22:11 +08:00
|
|
|
wl->psm_entry_retry = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->power_level = WL1271_DEFAULT_POWER_LEVEL;
|
2009-12-11 21:40:58 +08:00
|
|
|
wl->basic_rate_set = CONF_TX_RATE_MASK_BASIC;
|
2010-04-01 16:38:20 +08:00
|
|
|
wl->basic_rate = CONF_TX_RATE_MASK_BASIC;
|
2009-12-11 21:41:06 +08:00
|
|
|
wl->rate_set = CONF_TX_RATE_MASK_BASIC;
|
2009-10-09 02:56:24 +08:00
|
|
|
wl->band = IEEE80211_BAND_2GHZ;
|
2009-10-09 02:56:34 +08:00
|
|
|
wl->vif = NULL;
|
2009-12-11 21:41:06 +08:00
|
|
|
wl->flags = 0;
|
2010-03-18 18:26:32 +08:00
|
|
|
wl->sg_enabled = true;
|
2010-05-07 16:38:58 +08:00
|
|
|
wl->hw_pg_ver = -1;
|
2010-10-17 03:44:57 +08:00
|
|
|
wl->bss_type = MAX_BSS_TYPE;
|
|
|
|
wl->set_bss_type = MAX_BSS_TYPE;
|
|
|
|
wl->fw_bss_type = MAX_BSS_TYPE;
|
2011-02-23 06:22:26 +08:00
|
|
|
wl->last_tx_hlid = 0;
|
2011-02-23 06:22:31 +08:00
|
|
|
wl->ap_ps_map = 0;
|
|
|
|
wl->ap_fw_ps_map = 0;
|
2011-03-01 21:14:39 +08:00
|
|
|
wl->quirks = 0;
|
2011-03-31 16:07:01 +08:00
|
|
|
wl->platform_quirks = 0;
|
2009-08-06 21:25:28 +08:00
|
|
|
|
2010-10-12 22:20:06 +08:00
|
|
|
memset(wl->tx_frames_map, 0, sizeof(wl->tx_frames_map));
|
2009-10-09 02:56:26 +08:00
|
|
|
for (i = 0; i < ACX_TX_DESCRIPTORS; i++)
|
2009-08-06 21:25:28 +08:00
|
|
|
wl->tx_frames[i] = NULL;
|
|
|
|
|
|
|
|
spin_lock_init(&wl->wl_lock);
|
|
|
|
|
|
|
|
wl->state = WL1271_STATE_OFF;
|
|
|
|
mutex_init(&wl->mutex);
|
|
|
|
|
2010-02-18 19:25:57 +08:00
|
|
|
/* Apply default driver configuration. */
|
|
|
|
wl1271_conf_init(wl);
|
|
|
|
|
2010-09-30 19:28:27 +08:00
|
|
|
order = get_order(WL1271_AGGR_BUFFER_SIZE);
|
|
|
|
wl->aggr_buf = (u8 *)__get_free_pages(GFP_KERNEL, order);
|
|
|
|
if (!wl->aggr_buf) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto err_hw;
|
|
|
|
}
|
|
|
|
|
2011-03-31 16:06:59 +08:00
|
|
|
wl->dummy_packet = wl12xx_alloc_dummy_packet(wl);
|
|
|
|
if (!wl->dummy_packet) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto err_aggr;
|
|
|
|
}
|
|
|
|
|
2010-03-18 18:26:31 +08:00
|
|
|
/* Register platform device */
|
2010-03-18 18:26:46 +08:00
|
|
|
ret = platform_device_register(wl->plat_dev);
|
2010-03-18 18:26:31 +08:00
|
|
|
if (ret) {
|
|
|
|
wl1271_error("couldn't register platform device");
|
2011-03-31 16:06:59 +08:00
|
|
|
goto err_dummy_packet;
|
2010-03-18 18:26:31 +08:00
|
|
|
}
|
2010-03-18 18:26:46 +08:00
|
|
|
dev_set_drvdata(&wl->plat_dev->dev, wl);
|
2010-03-18 18:26:31 +08:00
|
|
|
|
2010-03-18 18:26:32 +08:00
|
|
|
/* Create sysfs file to control bt coex state */
|
2010-03-18 18:26:46 +08:00
|
|
|
ret = device_create_file(&wl->plat_dev->dev, &dev_attr_bt_coex_state);
|
2010-03-18 18:26:32 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("failed to create sysfs file bt_coex_state");
|
|
|
|
goto err_platform;
|
|
|
|
}
|
2010-03-18 18:26:31 +08:00
|
|
|
|
2010-05-07 16:38:58 +08:00
|
|
|
/* Create sysfs file to get HW PG version */
|
|
|
|
ret = device_create_file(&wl->plat_dev->dev, &dev_attr_hw_pg_ver);
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1271_error("failed to create sysfs file hw_pg_ver");
|
|
|
|
goto err_bt_coex_state;
|
|
|
|
}
|
|
|
|
|
2010-02-18 19:25:57 +08:00
|
|
|
return hw;
|
2010-03-18 18:26:31 +08:00
|
|
|
|
2010-05-07 16:38:58 +08:00
|
|
|
err_bt_coex_state:
|
|
|
|
device_remove_file(&wl->plat_dev->dev, &dev_attr_bt_coex_state);
|
|
|
|
|
2010-03-18 18:26:32 +08:00
|
|
|
err_platform:
|
2010-03-18 18:26:46 +08:00
|
|
|
platform_device_unregister(wl->plat_dev);
|
2010-03-18 18:26:32 +08:00
|
|
|
|
2011-03-31 16:06:59 +08:00
|
|
|
err_dummy_packet:
|
|
|
|
dev_kfree_skb(wl->dummy_packet);
|
|
|
|
|
2010-09-30 19:28:27 +08:00
|
|
|
err_aggr:
|
|
|
|
free_pages((unsigned long)wl->aggr_buf, order);
|
|
|
|
|
2010-03-18 18:26:31 +08:00
|
|
|
err_hw:
|
2010-03-18 18:26:46 +08:00
|
|
|
wl1271_debugfs_exit(wl);
|
|
|
|
kfree(plat_dev);
|
|
|
|
|
|
|
|
err_plat_alloc:
|
|
|
|
ieee80211_free_hw(hw);
|
|
|
|
|
|
|
|
err_hw_alloc:
|
2010-03-18 18:26:31 +08:00
|
|
|
|
|
|
|
return ERR_PTR(ret);
|
2010-02-18 19:25:57 +08:00
|
|
|
}
|
2010-02-22 14:38:26 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1271_alloc_hw);
|
2010-02-18 19:25:57 +08:00
|
|
|
|
|
|
|
int wl1271_free_hw(struct wl1271 *wl)
|
|
|
|
{
|
2010-03-18 18:26:46 +08:00
|
|
|
platform_device_unregister(wl->plat_dev);
|
2011-03-31 16:06:59 +08:00
|
|
|
dev_kfree_skb(wl->dummy_packet);
|
2010-09-30 19:28:27 +08:00
|
|
|
free_pages((unsigned long)wl->aggr_buf,
|
|
|
|
get_order(WL1271_AGGR_BUFFER_SIZE));
|
2010-03-18 18:26:46 +08:00
|
|
|
kfree(wl->plat_dev);
|
2010-02-18 19:25:57 +08:00
|
|
|
|
|
|
|
wl1271_debugfs_exit(wl);
|
|
|
|
|
|
|
|
vfree(wl->fw);
|
|
|
|
wl->fw = NULL;
|
|
|
|
kfree(wl->nvs);
|
|
|
|
wl->nvs = NULL;
|
|
|
|
|
|
|
|
kfree(wl->fw_status);
|
|
|
|
kfree(wl->tx_res_if);
|
|
|
|
|
|
|
|
ieee80211_free_hw(wl->hw);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2010-02-22 14:38:26 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1271_free_hw);
|
|
|
|
|
2011-01-12 17:33:29 +08:00
|
|
|
u32 wl12xx_debug_level = DEBUG_NONE;
|
2010-12-12 18:15:35 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl12xx_debug_level);
|
2011-01-12 17:33:29 +08:00
|
|
|
module_param_named(debug_level, wl12xx_debug_level, uint, S_IRUSR | S_IWUSR);
|
2010-12-12 18:15:35 +08:00
|
|
|
MODULE_PARM_DESC(debug_level, "wl12xx debugging level");
|
|
|
|
|
2010-02-22 14:38:26 +08:00
|
|
|
MODULE_LICENSE("GPL");
|
2011-02-22 20:19:28 +08:00
|
|
|
MODULE_AUTHOR("Luciano Coelho <coelho@ti.com>");
|
2010-02-22 14:38:26 +08:00
|
|
|
MODULE_AUTHOR("Juuso Oikarinen <juuso.oikarinen@nokia.com>");
|