2009-04-30 04:33:31 +08:00
|
|
|
/*
|
2009-06-12 19:17:39 +08:00
|
|
|
* This file is part of wl1251
|
2009-04-30 04:33:31 +08:00
|
|
|
*
|
|
|
|
* Copyright (C) 2008-2009 Nokia Corporation
|
|
|
|
*
|
|
|
|
* 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/interrupt.h>
|
|
|
|
#include <linux/firmware.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/irq.h>
|
|
|
|
#include <linux/crc32.h>
|
|
|
|
#include <linux/etherdevice.h>
|
2009-11-18 00:48:45 +08:00
|
|
|
#include <linux/vmalloc.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>
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:25 +08:00
|
|
|
#include "wl1251.h"
|
2009-04-30 04:33:31 +08:00
|
|
|
#include "wl12xx_80211.h"
|
2010-10-10 16:28:32 +08:00
|
|
|
#include "reg.h"
|
|
|
|
#include "io.h"
|
|
|
|
#include "cmd.h"
|
|
|
|
#include "event.h"
|
|
|
|
#include "tx.h"
|
|
|
|
#include "rx.h"
|
|
|
|
#include "ps.h"
|
|
|
|
#include "init.h"
|
|
|
|
#include "debugfs.h"
|
|
|
|
#include "boot.h"
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:33:49 +08:00
|
|
|
void wl1251_enable_interrupts(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-08-07 18:33:49 +08:00
|
|
|
wl->if_ops->enable_irq(wl);
|
|
|
|
}
|
|
|
|
|
|
|
|
void wl1251_disable_interrupts(struct wl1251 *wl)
|
|
|
|
{
|
|
|
|
wl->if_ops->disable_irq(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2010-11-04 06:13:47 +08:00
|
|
|
static int wl1251_power_off(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2010-11-04 06:13:47 +08:00
|
|
|
return wl->if_ops->power(wl, false);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2010-11-04 06:13:47 +08:00
|
|
|
static int wl1251_power_on(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2010-11-04 06:13:47 +08:00
|
|
|
return wl->if_ops->power(wl, true);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_fetch_firmware(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
|
|
|
const struct firmware *fw;
|
2009-08-07 18:33:04 +08:00
|
|
|
struct device *dev = wiphy_dev(wl->hw->wiphy);
|
2009-04-30 04:33:31 +08:00
|
|
|
int ret;
|
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
ret = request_firmware(&fw, WL1251_FW_NAME, dev);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (ret < 0) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("could not get firmware: %d", ret);
|
2009-04-30 04:33:31 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fw->size % 4) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("firmware size is not multiple of 32 bits: %zu",
|
2009-04-30 04:33:31 +08:00
|
|
|
fw->size);
|
|
|
|
ret = -EILSEQ;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->fw_len = fw->size;
|
2009-11-18 00:48:45 +08:00
|
|
|
wl->fw = vmalloc(wl->fw_len);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (!wl->fw) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("could not allocate memory for the firmware");
|
2009-04-30 04:33:31 +08:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(wl->fw, fw->data, wl->fw_len);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
release_firmware(fw);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_fetch_nvs(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
|
|
|
const struct firmware *fw;
|
2009-08-07 18:33:04 +08:00
|
|
|
struct device *dev = wiphy_dev(wl->hw->wiphy);
|
2009-04-30 04:33:31 +08:00
|
|
|
int ret;
|
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
ret = request_firmware(&fw, WL1251_NVS_NAME, dev);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (ret < 0) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("could not get nvs file: %d", ret);
|
2009-04-30 04:33:31 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fw->size % 4) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("nvs size is not multiple of 32 bits: %zu",
|
2009-04-30 04:33:31 +08:00
|
|
|
fw->size);
|
|
|
|
ret = -EILSEQ;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->nvs_len = fw->size;
|
2010-05-16 05:15:10 +08:00
|
|
|
wl->nvs = kmemdup(fw->data, wl->nvs_len, GFP_KERNEL);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (!wl->nvs) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("could not allocate memory for the nvs file");
|
2009-04-30 04:33:31 +08:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
release_firmware(fw);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static void wl1251_fw_wakeup(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
|
|
|
u32 elp_reg;
|
|
|
|
|
|
|
|
elp_reg = ELPCTRL_WAKE_UP;
|
2010-03-11 23:44:57 +08:00
|
|
|
wl1251_write_elp(wl, HW_ACCESS_ELP_CTRL_REG_ADDR, elp_reg);
|
|
|
|
elp_reg = wl1251_read_elp(wl, HW_ACCESS_ELP_CTRL_REG_ADDR);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:47 +08:00
|
|
|
if (!(elp_reg & ELPCTRL_WLAN_READY))
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_warning("WLAN not ready");
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_chip_wakeup(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2010-11-04 06:13:47 +08:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = wl1251_power_on(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
msleep(WL1251_POWER_ON_SLEEP);
|
2009-08-07 18:33:11 +08:00
|
|
|
wl->if_ops->reset(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
/* We don't need a real memory partition here, because we only want
|
|
|
|
* to use the registers at this point. */
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_set_partition(wl,
|
2009-04-30 04:33:31 +08:00
|
|
|
0x00000000,
|
|
|
|
0x00000000,
|
|
|
|
REGISTERS_BASE,
|
|
|
|
REGISTERS_DOWN_SIZE);
|
|
|
|
|
|
|
|
/* ELP module wake up */
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_fw_wakeup(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
/* whal_FwCtrl_BootSm() */
|
|
|
|
|
|
|
|
/* 0. read chip id from CHIP_ID */
|
2009-08-07 18:33:57 +08:00
|
|
|
wl->chip_id = wl1251_reg_read32(wl, CHIP_ID_B);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
/* 1. check if chip id is valid */
|
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
switch (wl->chip_id) {
|
2009-04-30 04:33:31 +08:00
|
|
|
case CHIP_ID_1251_PG12:
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_BOOT, "chip id 0x%x (1251 PG12)",
|
2009-08-07 18:33:57 +08:00
|
|
|
wl->chip_id);
|
2009-04-30 04:33:31 +08:00
|
|
|
break;
|
|
|
|
case CHIP_ID_1251_PG11:
|
2009-10-15 21:38:16 +08:00
|
|
|
wl1251_debug(DEBUG_BOOT, "chip id 0x%x (1251 PG11)",
|
|
|
|
wl->chip_id);
|
|
|
|
break;
|
2009-10-28 03:15:05 +08:00
|
|
|
case CHIP_ID_1251_PG10:
|
2009-04-30 04:33:31 +08:00
|
|
|
default:
|
2009-08-07 18:33:57 +08:00
|
|
|
wl1251_error("unsupported chip id: 0x%x", wl->chip_id);
|
2009-04-30 04:33:31 +08:00
|
|
|
ret = -ENODEV;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wl->fw == NULL) {
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_fetch_firmware(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2010-04-13 06:21:40 +08:00
|
|
|
if (wl->nvs == NULL && !wl->use_eeprom) {
|
|
|
|
/* No NVS from netlink, try to get it from the filesystem */
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_fetch_nvs(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
#define WL1251_IRQ_LOOP_COUNT 10
|
2009-08-07 18:33:57 +08:00
|
|
|
static void wl1251_irq_work(struct work_struct *work)
|
|
|
|
{
|
2009-11-18 00:49:31 +08:00
|
|
|
u32 intr, ctr = WL1251_IRQ_LOOP_COUNT;
|
2009-08-07 18:33:57 +08:00
|
|
|
struct wl1251 *wl =
|
|
|
|
container_of(work, struct wl1251, irq_work);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
wl1251_debug(DEBUG_IRQ, "IRQ work");
|
|
|
|
|
|
|
|
if (wl->state == WL1251_STATE_OFF)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
wl1251_reg_write32(wl, ACX_REG_INTERRUPT_MASK, WL1251_ACX_INTR_ALL);
|
|
|
|
|
|
|
|
intr = wl1251_reg_read32(wl, ACX_REG_INTERRUPT_CLEAR);
|
|
|
|
wl1251_debug(DEBUG_IRQ, "intr: 0x%x", intr);
|
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
do {
|
|
|
|
if (wl->data_path) {
|
|
|
|
wl->rx_counter = wl1251_mem_read32(
|
|
|
|
wl, wl->data_path->rx_control_addr);
|
|
|
|
|
|
|
|
/* We handle a frmware bug here */
|
|
|
|
switch ((wl->rx_counter - wl->rx_handled) & 0xf) {
|
|
|
|
case 0:
|
|
|
|
wl1251_debug(DEBUG_IRQ,
|
|
|
|
"RX: FW and host in sync");
|
|
|
|
intr &= ~WL1251_ACX_INTR_RX0_DATA;
|
|
|
|
intr &= ~WL1251_ACX_INTR_RX1_DATA;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
wl1251_debug(DEBUG_IRQ, "RX: FW +1");
|
|
|
|
intr |= WL1251_ACX_INTR_RX0_DATA;
|
|
|
|
intr &= ~WL1251_ACX_INTR_RX1_DATA;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
wl1251_debug(DEBUG_IRQ, "RX: FW +2");
|
|
|
|
intr |= WL1251_ACX_INTR_RX0_DATA;
|
|
|
|
intr |= WL1251_ACX_INTR_RX1_DATA;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
wl1251_warning(
|
|
|
|
"RX: FW and host out of sync: %d",
|
|
|
|
wl->rx_counter - wl->rx_handled);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->rx_handled = wl->rx_counter;
|
|
|
|
|
|
|
|
wl1251_debug(DEBUG_IRQ, "RX counter: %d",
|
|
|
|
wl->rx_counter);
|
2009-08-07 18:33:57 +08:00
|
|
|
}
|
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
intr &= wl->intr_mask;
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
if (intr == 0) {
|
|
|
|
wl1251_debug(DEBUG_IRQ, "INTR is 0");
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
if (intr & WL1251_ACX_INTR_RX0_DATA) {
|
|
|
|
wl1251_debug(DEBUG_IRQ, "WL1251_ACX_INTR_RX0_DATA");
|
|
|
|
wl1251_rx(wl);
|
|
|
|
}
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
if (intr & WL1251_ACX_INTR_RX1_DATA) {
|
|
|
|
wl1251_debug(DEBUG_IRQ, "WL1251_ACX_INTR_RX1_DATA");
|
|
|
|
wl1251_rx(wl);
|
|
|
|
}
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
if (intr & WL1251_ACX_INTR_TX_RESULT) {
|
|
|
|
wl1251_debug(DEBUG_IRQ, "WL1251_ACX_INTR_TX_RESULT");
|
|
|
|
wl1251_tx_complete(wl);
|
|
|
|
}
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2010-08-18 03:46:53 +08:00
|
|
|
if (intr & WL1251_ACX_INTR_EVENT_A) {
|
|
|
|
wl1251_debug(DEBUG_IRQ, "WL1251_ACX_INTR_EVENT_A");
|
|
|
|
wl1251_event_handle(wl, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (intr & WL1251_ACX_INTR_EVENT_B) {
|
|
|
|
wl1251_debug(DEBUG_IRQ, "WL1251_ACX_INTR_EVENT_B");
|
|
|
|
wl1251_event_handle(wl, 1);
|
2009-11-18 00:49:31 +08:00
|
|
|
}
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2009-11-18 00:49:31 +08:00
|
|
|
if (intr & WL1251_ACX_INTR_INIT_COMPLETE)
|
|
|
|
wl1251_debug(DEBUG_IRQ,
|
|
|
|
"WL1251_ACX_INTR_INIT_COMPLETE");
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2009-11-18 00:49:46 +08:00
|
|
|
if (--ctr == 0)
|
|
|
|
break;
|
2009-08-07 18:33:57 +08:00
|
|
|
|
2009-11-18 00:49:46 +08:00
|
|
|
intr = wl1251_reg_read32(wl, ACX_REG_INTERRUPT_CLEAR);
|
|
|
|
} while (intr);
|
2009-08-07 18:33:57 +08:00
|
|
|
|
|
|
|
out_sleep:
|
2009-11-18 00:49:31 +08:00
|
|
|
wl1251_reg_write32(wl, ACX_REG_INTERRUPT_MASK, ~(wl->intr_mask));
|
2009-08-07 18:33:57 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
}
|
|
|
|
|
2009-08-07 18:34:42 +08:00
|
|
|
static int wl1251_join(struct wl1251 *wl, u8 bss_type, u8 channel,
|
|
|
|
u16 beacon_interval, u8 dtim_period)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = wl1251_acx_frame_rates(wl, DEFAULT_HW_GEN_TX_RATE,
|
|
|
|
DEFAULT_HW_GEN_MODULATION_TYPE,
|
|
|
|
wl->tx_mgmt_frm_rate,
|
|
|
|
wl->tx_mgmt_frm_mod);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
|
|
ret = wl1251_cmd_join(wl, bss_type, channel, beacon_interval,
|
|
|
|
dtim_period);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2010-08-18 03:46:55 +08:00
|
|
|
ret = wl1251_event_wait(wl, JOIN_EVENT_COMPLETE_ID, 100);
|
|
|
|
if (ret < 0)
|
|
|
|
wl1251_warning("join timeout");
|
2009-08-07 18:34:42 +08:00
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static void wl1251_filter_work(struct work_struct *work)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl =
|
|
|
|
container_of(work, struct wl1251, filter_work);
|
2009-04-30 04:33:31 +08:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
if (wl->state == WL1251_STATE_OFF)
|
2009-04-30 04:33:31 +08:00
|
|
|
goto out;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2009-08-07 18:34:42 +08:00
|
|
|
ret = wl1251_join(wl, wl->bss_type, wl->channel, wl->beacon_int,
|
|
|
|
wl->dtim_period);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
2009-06-12 19:16:32 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:16:32 +08:00
|
|
|
out_sleep:
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
|
|
|
|
out:
|
2009-04-30 04:33:31 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
}
|
|
|
|
|
2011-02-24 21:42:06 +08:00
|
|
|
static void wl1251_op_tx(struct ieee80211_hw *hw, struct sk_buff *skb)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2010-08-28 05:48:19 +08:00
|
|
|
unsigned long flags;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
skb_queue_tail(&wl->tx_queue, skb);
|
|
|
|
|
2009-06-12 19:15:54 +08:00
|
|
|
/*
|
|
|
|
* The chip specific setup must run before the first TX packet -
|
|
|
|
* before that, the tx_work will not be initialized!
|
|
|
|
*/
|
|
|
|
|
2009-08-07 18:35:04 +08:00
|
|
|
ieee80211_queue_work(wl->hw, &wl->tx_work);
|
2009-04-30 04:33:31 +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.
|
|
|
|
*/
|
2010-08-28 05:48:19 +08:00
|
|
|
if (skb_queue_len(&wl->tx_queue) >= WL1251_TX_QUEUE_HIGH_WATERMARK) {
|
2009-11-30 16:17:45 +08:00
|
|
|
wl1251_debug(DEBUG_TX, "op_tx: tx_queue full, stop queues");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2010-08-28 05:48:19 +08:00
|
|
|
spin_lock_irqsave(&wl->wl_lock, flags);
|
|
|
|
ieee80211_stop_queues(wl->hw);
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->tx_queue_stopped = true;
|
2010-08-28 05:48:19 +08:00
|
|
|
spin_unlock_irqrestore(&wl->wl_lock, flags);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_op_start(struct ieee80211_hw *hw)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2010-07-29 04:59:41 +08:00
|
|
|
struct wiphy *wiphy = hw->wiphy;
|
2009-04-30 04:33:31 +08:00
|
|
|
int ret = 0;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 start");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
if (wl->state != WL1251_STATE_OFF) {
|
|
|
|
wl1251_error("cannot start because not in off state: %d",
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->state);
|
|
|
|
ret = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_chip_wakeup(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
2009-07-15 04:37:13 +08:00
|
|
|
goto out;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
ret = wl1251_boot(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
ret = wl1251_hw_init(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_station_id(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->state = WL1251_STATE_ON;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
wl1251_info("firmware booted (%s)", wl->fw_ver);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2010-07-29 04:59:41 +08:00
|
|
|
/* update hw/fw version info in wiphy struct */
|
|
|
|
wiphy->hw_version = wl->chip_id;
|
|
|
|
strncpy(wiphy->fw_version, wl->fw_ver, sizeof(wiphy->fw_version));
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
out:
|
|
|
|
if (ret < 0)
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_power_off(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static void wl1251_op_stop(struct ieee80211_hw *hw)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_info("down");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 stop");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
WARN_ON(wl->state != WL1251_STATE_ON);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (wl->scanning) {
|
|
|
|
ieee80211_scan_completed(wl->hw, true);
|
|
|
|
wl->scanning = false;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->state = WL1251_STATE_OFF;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_disable_interrupts(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
cancel_work_sync(&wl->irq_work);
|
|
|
|
cancel_work_sync(&wl->tx_work);
|
|
|
|
cancel_work_sync(&wl->filter_work);
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
/* let's notify MAC80211 about the remaining pending TX frames */
|
2009-08-07 18:33:57 +08:00
|
|
|
wl1251_tx_flush(wl);
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_power_off(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
memset(wl->bssid, 0, ETH_ALEN);
|
|
|
|
wl->listen_int = 1;
|
|
|
|
wl->bss_type = MAX_BSS_TYPE;
|
|
|
|
|
|
|
|
wl->data_in_count = 0;
|
|
|
|
wl->rx_counter = 0;
|
|
|
|
wl->rx_handled = 0;
|
|
|
|
wl->rx_current_buffer = 0;
|
|
|
|
wl->rx_last_id = 0;
|
|
|
|
wl->next_tx_complete = 0;
|
|
|
|
wl->elp = false;
|
2011-04-04 16:04:57 +08:00
|
|
|
wl->station_mode = STATION_ACTIVE_MODE;
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->tx_queue_stopped = false;
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->power_level = WL1251_DEFAULT_POWER_LEVEL;
|
2011-01-31 03:11:00 +08:00
|
|
|
wl->rssi_thold = 0;
|
2009-08-07 18:34:27 +08:00
|
|
|
wl->channel = WL1251_DEFAULT_CHANNEL;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debugfs_reset(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_op_add_interface(struct ieee80211_hw *hw,
|
2009-12-23 20:15:45 +08:00
|
|
|
struct ieee80211_vif *vif)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2009-04-30 04:33:31 +08:00
|
|
|
int ret = 0;
|
|
|
|
|
2009-07-15 23:21:41 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 add interface type %d mac %pM",
|
2009-12-23 20:15:45 +08:00
|
|
|
vif->type, vif->addr);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
2009-11-18 00:48:23 +08:00
|
|
|
if (wl->vif) {
|
|
|
|
ret = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2009-12-23 20:15:45 +08:00
|
|
|
wl->vif = vif;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-12-23 20:15:45 +08:00
|
|
|
switch (vif->type) {
|
2009-04-30 04:33:31 +08:00
|
|
|
case NL80211_IFTYPE_STATION:
|
|
|
|
wl->bss_type = BSS_TYPE_STA_BSS;
|
|
|
|
break;
|
|
|
|
case NL80211_IFTYPE_ADHOC:
|
|
|
|
wl->bss_type = BSS_TYPE_IBSS;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2009-12-23 20:15:45 +08:00
|
|
|
if (memcmp(wl->mac_addr, vif->addr, ETH_ALEN)) {
|
|
|
|
memcpy(wl->mac_addr, vif->addr, ETH_ALEN);
|
2009-04-30 04:33:31 +08:00
|
|
|
SET_IEEE80211_PERM_ADDR(wl->hw, wl->mac_addr);
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_station_id(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static void wl1251_op_remove_interface(struct ieee80211_hw *hw,
|
2009-12-23 20:15:45 +08:00
|
|
|
struct ieee80211_vif *vif)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-11-18 00:48:23 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 remove interface");
|
2009-11-18 00:48:23 +08:00
|
|
|
wl->vif = NULL;
|
|
|
|
mutex_unlock(&wl->mutex);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2010-01-12 16:43:07 +08:00
|
|
|
static int wl1251_build_qos_null_data(struct wl1251 *wl)
|
|
|
|
{
|
|
|
|
struct ieee80211_qos_hdr template;
|
|
|
|
|
|
|
|
memset(&template, 0, sizeof(template));
|
|
|
|
|
|
|
|
memcpy(template.addr1, wl->bssid, ETH_ALEN);
|
|
|
|
memcpy(template.addr2, wl->mac_addr, ETH_ALEN);
|
|
|
|
memcpy(template.addr3, wl->bssid, ETH_ALEN);
|
|
|
|
|
|
|
|
template.frame_control = cpu_to_le16(IEEE80211_FTYPE_DATA |
|
|
|
|
IEEE80211_STYPE_QOS_NULLFUNC |
|
|
|
|
IEEE80211_FCTL_TODS);
|
|
|
|
|
|
|
|
/* FIXME: not sure what priority to use here */
|
|
|
|
template.qos_ctrl = cpu_to_le16(0);
|
|
|
|
|
|
|
|
return wl1251_cmd_template_set(wl, CMD_QOS_NULL_DATA, &template,
|
|
|
|
sizeof(template));
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_op_config(struct ieee80211_hw *hw, u32 changed)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2009-04-30 04:33:31 +08:00
|
|
|
struct ieee80211_conf *conf = &hw->conf;
|
|
|
|
int channel, ret = 0;
|
|
|
|
|
|
|
|
channel = ieee80211_frequency_to_channel(conf->channel->center_freq);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 config ch %d psm %s power %d",
|
2009-04-30 04:33:31 +08:00
|
|
|
channel,
|
|
|
|
conf->flags & IEEE80211_CONF_PS ? "on" : "off",
|
|
|
|
conf->power_level);
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
if (channel != wl->channel) {
|
2009-08-07 18:34:49 +08:00
|
|
|
wl->channel = channel;
|
|
|
|
|
2009-08-07 18:34:42 +08:00
|
|
|
ret = wl1251_join(wl, wl->bss_type, wl->channel,
|
|
|
|
wl->beacon_int, wl->dtim_period);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
2009-06-12 19:16:32 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (conf->flags & IEEE80211_CONF_PS && !wl->psm_requested) {
|
2009-06-12 19:17:53 +08:00
|
|
|
wl1251_debug(DEBUG_PSM, "psm enabled");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
wl->psm_requested = true;
|
|
|
|
|
2010-01-26 21:19:52 +08:00
|
|
|
wl->dtim_period = conf->ps_dtim_period;
|
|
|
|
|
|
|
|
ret = wl1251_acx_wr_tbtt_and_dtim(wl, wl->beacon_int,
|
|
|
|
wl->dtim_period);
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
/*
|
2010-01-26 21:19:52 +08:00
|
|
|
* mac80211 enables PSM only if we're already associated.
|
2009-04-30 04:33:31 +08:00
|
|
|
*/
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_set_mode(wl, STATION_POWER_SAVE_MODE);
|
2009-11-30 16:17:52 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
} else if (!(conf->flags & IEEE80211_CONF_PS) &&
|
|
|
|
wl->psm_requested) {
|
2009-06-12 19:17:53 +08:00
|
|
|
wl1251_debug(DEBUG_PSM, "psm disabled");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
wl->psm_requested = false;
|
|
|
|
|
2011-04-04 16:04:57 +08:00
|
|
|
if (wl->station_mode != STATION_ACTIVE_MODE) {
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_set_mode(wl, STATION_ACTIVE_MODE);
|
2009-11-30 16:17:52 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2011-04-04 16:04:58 +08:00
|
|
|
if (changed & IEEE80211_CONF_CHANGE_IDLE) {
|
|
|
|
if (conf->flags & IEEE80211_CONF_IDLE) {
|
|
|
|
ret = wl1251_ps_set_mode(wl, STATION_IDLE);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
} else {
|
|
|
|
ret = wl1251_ps_set_mode(wl, STATION_ACTIVE_MODE);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
ret = wl1251_join(wl, wl->bss_type, wl->channel,
|
|
|
|
wl->beacon_int, wl->dtim_period);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
if (conf->power_level != wl->power_level) {
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_tx_power(wl, conf->power_level);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
2009-11-30 16:17:52 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
wl->power_level = conf->power_level;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:16:32 +08:00
|
|
|
out_sleep:
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
|
|
|
|
out:
|
2009-04-30 04:33:31 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
2009-06-12 19:16:32 +08:00
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
#define WL1251_SUPPORTED_FILTERS (FIF_PROMISC_IN_BSS | \
|
2009-04-30 04:33:31 +08:00
|
|
|
FIF_ALLMULTI | \
|
|
|
|
FIF_FCSFAIL | \
|
|
|
|
FIF_BCN_PRBRESP_PROMISC | \
|
|
|
|
FIF_CONTROL | \
|
|
|
|
FIF_OTHER_BSS)
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static void wl1251_op_configure_filter(struct ieee80211_hw *hw,
|
2009-04-30 04:33:31 +08:00
|
|
|
unsigned int changed,
|
2009-08-17 22:16:53 +08:00
|
|
|
unsigned int *total,u64 multicast)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 configure filter");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
*total &= WL1251_SUPPORTED_FILTERS;
|
|
|
|
changed &= WL1251_SUPPORTED_FILTERS;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (changed == 0)
|
|
|
|
/* no filters which we support changed */
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* FIXME: wl->rx_config and wl->rx_filter are not protected */
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->rx_config = WL1251_DEFAULT_RX_CONFIG;
|
|
|
|
wl->rx_filter = WL1251_DEFAULT_RX_FILTER;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (*total & FIF_PROMISC_IN_BSS) {
|
|
|
|
wl->rx_config |= CFG_BSSID_FILTER_EN;
|
|
|
|
wl->rx_config |= CFG_RX_ALL_GOOD;
|
|
|
|
}
|
|
|
|
if (*total & FIF_ALLMULTI)
|
|
|
|
/*
|
|
|
|
* CFG_MC_FILTER_EN in rx_config needs to be 0 to receive
|
|
|
|
* all multicast frames
|
|
|
|
*/
|
|
|
|
wl->rx_config &= ~CFG_MC_FILTER_EN;
|
|
|
|
if (*total & FIF_FCSFAIL)
|
|
|
|
wl->rx_filter |= CFG_RX_FCS_ERROR;
|
|
|
|
if (*total & FIF_BCN_PRBRESP_PROMISC) {
|
|
|
|
wl->rx_config &= ~CFG_BSSID_FILTER_EN;
|
|
|
|
wl->rx_config &= ~CFG_SSID_FILTER_EN;
|
|
|
|
}
|
|
|
|
if (*total & FIF_CONTROL)
|
|
|
|
wl->rx_filter |= CFG_RX_CTL_EN;
|
|
|
|
if (*total & FIF_OTHER_BSS)
|
|
|
|
wl->rx_filter &= ~CFG_BSSID_FILTER_EN;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* FIXME: workqueues need to be properly cancelled on stop(), for
|
|
|
|
* now let's just disable changing the filter settings. They will
|
|
|
|
* be updated any on config().
|
|
|
|
*/
|
|
|
|
/* schedule_work(&wl->filter_work); */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* HW encryption */
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_set_key_type(struct wl1251 *wl,
|
|
|
|
struct wl1251_cmd_set_keys *key,
|
2009-04-30 04:33:31 +08:00
|
|
|
enum set_key_cmd cmd,
|
|
|
|
struct ieee80211_key_conf *mac80211_key,
|
|
|
|
const u8 *addr)
|
|
|
|
{
|
2010-08-10 15:46:38 +08:00
|
|
|
switch (mac80211_key->cipher) {
|
|
|
|
case WLAN_CIPHER_SUITE_WEP40:
|
|
|
|
case WLAN_CIPHER_SUITE_WEP104:
|
2009-04-30 04:33:31 +08:00
|
|
|
if (is_broadcast_ether_addr(addr))
|
|
|
|
key->key_type = KEY_WEP_DEFAULT;
|
|
|
|
else
|
|
|
|
key->key_type = KEY_WEP_ADDR;
|
|
|
|
|
|
|
|
mac80211_key->hw_key_idx = mac80211_key->keyidx;
|
|
|
|
break;
|
2010-08-10 15:46:38 +08:00
|
|
|
case WLAN_CIPHER_SUITE_TKIP:
|
2009-04-30 04:33:31 +08:00
|
|
|
if (is_broadcast_ether_addr(addr))
|
|
|
|
key->key_type = KEY_TKIP_MIC_GROUP;
|
|
|
|
else
|
|
|
|
key->key_type = KEY_TKIP_MIC_PAIRWISE;
|
|
|
|
|
|
|
|
mac80211_key->hw_key_idx = mac80211_key->keyidx;
|
|
|
|
break;
|
2010-08-10 15:46:38 +08:00
|
|
|
case WLAN_CIPHER_SUITE_CCMP:
|
2009-04-30 04:33:31 +08:00
|
|
|
if (is_broadcast_ether_addr(addr))
|
|
|
|
key->key_type = KEY_AES_GROUP;
|
|
|
|
else
|
|
|
|
key->key_type = KEY_AES_PAIRWISE;
|
|
|
|
mac80211_key->flags |= IEEE80211_KEY_FLAG_GENERATE_IV;
|
|
|
|
break;
|
|
|
|
default:
|
2010-08-10 15:46:38 +08:00
|
|
|
wl1251_error("Unknown key cipher 0x%x", mac80211_key->cipher);
|
2009-04-30 04:33:31 +08:00
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_op_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
|
2009-04-30 04:33:31 +08:00
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_sta *sta,
|
|
|
|
struct ieee80211_key_conf *key)
|
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
|
|
|
struct wl1251_cmd_set_keys *wl_cmd;
|
2009-04-30 04:33:31 +08:00
|
|
|
const u8 *addr;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
static const u8 bcast_addr[ETH_ALEN] =
|
|
|
|
{ 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 set key");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:14:19 +08:00
|
|
|
wl_cmd = kzalloc(sizeof(*wl_cmd), GFP_KERNEL);
|
|
|
|
if (!wl_cmd) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
addr = sta ? sta->addr : bcast_addr;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_CRYPT, "CMD: 0x%x", cmd);
|
|
|
|
wl1251_dump(DEBUG_CRYPT, "ADDR: ", addr, ETH_ALEN);
|
|
|
|
wl1251_debug(DEBUG_CRYPT, "Key: algo:0x%x, id:%d, len:%d flags 0x%x",
|
2010-08-10 15:46:38 +08:00
|
|
|
key->cipher, key->keyidx, key->keylen, key->flags);
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_dump(DEBUG_CRYPT, "KEY: ", key->key, key->keylen);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:14:19 +08:00
|
|
|
if (is_zero_ether_addr(addr)) {
|
|
|
|
/* We dont support TX only encryption */
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_unlock;
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
switch (cmd) {
|
|
|
|
case SET_KEY:
|
2009-06-12 19:14:19 +08:00
|
|
|
wl_cmd->key_action = KEY_ADD_OR_REPLACE;
|
2009-04-30 04:33:31 +08:00
|
|
|
break;
|
|
|
|
case DISABLE_KEY:
|
2009-06-12 19:14:19 +08:00
|
|
|
wl_cmd->key_action = KEY_REMOVE;
|
2009-04-30 04:33:31 +08:00
|
|
|
break;
|
|
|
|
default:
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("Unsupported key cmd 0x%x", cmd);
|
2009-04-30 04:33:31 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_set_key_type(wl, wl_cmd, cmd, key, addr);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("Set KEY type failed");
|
2009-06-12 19:16:32 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2009-06-12 19:14:19 +08:00
|
|
|
if (wl_cmd->key_type != KEY_WEP_DEFAULT)
|
|
|
|
memcpy(wl_cmd->addr, addr, ETH_ALEN);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:14:19 +08:00
|
|
|
if ((wl_cmd->key_type == KEY_TKIP_MIC_GROUP) ||
|
|
|
|
(wl_cmd->key_type == KEY_TKIP_MIC_PAIRWISE)) {
|
2009-04-30 04:33:31 +08:00
|
|
|
/*
|
|
|
|
* We get the key in the following form:
|
|
|
|
* TKIP (16 bytes) - TX MIC (8 bytes) - RX MIC (8 bytes)
|
|
|
|
* but the target is expecting:
|
|
|
|
* TKIP - RX MIC - TX MIC
|
|
|
|
*/
|
2009-06-12 19:14:19 +08:00
|
|
|
memcpy(wl_cmd->key, key->key, 16);
|
|
|
|
memcpy(wl_cmd->key + 16, key->key + 24, 8);
|
|
|
|
memcpy(wl_cmd->key + 24, key->key + 16, 8);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
} else {
|
2009-06-12 19:14:19 +08:00
|
|
|
memcpy(wl_cmd->key, key->key, key->keylen);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
2009-06-12 19:14:19 +08:00
|
|
|
wl_cmd->key_size = key->keylen;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:14:19 +08:00
|
|
|
wl_cmd->id = key->keyidx;
|
|
|
|
wl_cmd->ssid_profile = 0;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_dump(DEBUG_CRYPT, "TARGET KEY: ", wl_cmd, sizeof(*wl_cmd));
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_cmd_send(wl, CMD_SET_KEYS, wl_cmd, sizeof(*wl_cmd));
|
2009-06-12 19:14:19 +08:00
|
|
|
if (ret < 0) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_warning("could not set keys");
|
2009-06-12 19:16:32 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2009-06-12 19:16:32 +08:00
|
|
|
out_sleep:
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
|
|
|
|
out_unlock:
|
2009-04-30 04:33:31 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
2009-06-12 19:14:19 +08:00
|
|
|
|
|
|
|
out:
|
|
|
|
kfree(wl_cmd);
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_op_hw_scan(struct ieee80211_hw *hw,
|
2010-04-27 17:59:34 +08:00
|
|
|
struct ieee80211_vif *vif,
|
2009-04-30 04:33:31 +08:00
|
|
|
struct cfg80211_scan_request *req)
|
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2010-01-06 02:16:57 +08:00
|
|
|
struct sk_buff *skb;
|
2009-04-30 04:33:31 +08:00
|
|
|
size_t ssid_len = 0;
|
2010-01-06 02:16:51 +08:00
|
|
|
u8 *ssid = NULL;
|
|
|
|
int ret;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 hw scan");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (req->n_ssids) {
|
|
|
|
ssid = req->ssids[0].ssid;
|
|
|
|
ssid_len = req->ssids[0].ssid_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
2009-06-12 19:16:32 +08:00
|
|
|
|
2010-01-06 02:16:51 +08:00
|
|
|
if (wl->scanning) {
|
|
|
|
wl1251_debug(DEBUG_SCAN, "scan already in progress");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2010-01-06 02:16:57 +08:00
|
|
|
skb = ieee80211_probereq_get(wl->hw, wl->vif, ssid, ssid_len,
|
|
|
|
req->ie, req->ie_len);
|
|
|
|
if (!skb) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = wl1251_cmd_template_set(wl, CMD_PROBE_REQ, skb->data,
|
|
|
|
skb->len);
|
|
|
|
dev_kfree_skb(skb);
|
2010-01-06 02:16:51 +08:00
|
|
|
if (ret < 0)
|
2010-01-06 02:16:57 +08:00
|
|
|
goto out_sleep;
|
2010-01-06 02:16:51 +08:00
|
|
|
|
|
|
|
ret = wl1251_cmd_trigger_scan_to(wl, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
|
|
|
|
wl->scanning = true;
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2010-01-06 02:17:03 +08:00
|
|
|
ret = wl1251_cmd_scan(wl, ssid, ssid_len, req->channels,
|
|
|
|
req->n_channels, WL1251_SCAN_NUM_PROBES);
|
2010-01-06 02:16:51 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
wl->scanning = false;
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_sleep:
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
|
|
|
|
out:
|
2009-04-30 04:33:31 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_op_set_rts_threshold(struct ieee80211_hw *hw, u32 value)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2009-04-30 04:33:31 +08:00
|
|
|
int ret;
|
|
|
|
|
2009-06-12 19:16:20 +08:00
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_rts_threshold(wl, (u16) value);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_warning("wl1251_op_set_rts_threshold failed: %d", ret);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2009-06-12 19:16:32 +08:00
|
|
|
out:
|
2009-06-12 19:16:20 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static void wl1251_op_bss_info_changed(struct ieee80211_hw *hw,
|
2009-04-30 04:33:31 +08:00
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_bss_conf *bss_conf,
|
|
|
|
u32 changed)
|
|
|
|
{
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
2010-01-06 02:16:32 +08:00
|
|
|
struct sk_buff *beacon, *skb;
|
2009-04-30 04:33:31 +08:00
|
|
|
int ret;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 bss info changed");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
2009-06-12 19:16:26 +08:00
|
|
|
|
2011-01-31 03:11:00 +08:00
|
|
|
if (changed & BSS_CHANGED_CQM) {
|
|
|
|
ret = wl1251_acx_low_rssi(wl, bss_conf->cqm_rssi_thold,
|
|
|
|
WL1251_DEFAULT_LOW_RSSI_WEIGHT,
|
|
|
|
WL1251_DEFAULT_LOW_RSSI_DEPTH,
|
|
|
|
WL1251_ACX_LOW_RSSI_TYPE_EDGE);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
wl->rssi_thold = bss_conf->cqm_rssi_thold;
|
|
|
|
}
|
|
|
|
|
2009-11-26 16:56:06 +08:00
|
|
|
if (changed & BSS_CHANGED_BSSID) {
|
|
|
|
memcpy(wl->bssid, bss_conf->bssid, ETH_ALEN);
|
|
|
|
|
2010-01-06 02:16:32 +08:00
|
|
|
skb = ieee80211_nullfunc_get(wl->hw, wl->vif);
|
|
|
|
if (!skb)
|
|
|
|
goto out_sleep;
|
|
|
|
|
|
|
|
ret = wl1251_cmd_template_set(wl, CMD_NULL_DATA,
|
|
|
|
skb->data, skb->len);
|
|
|
|
dev_kfree_skb(skb);
|
2009-11-26 16:56:06 +08:00
|
|
|
if (ret < 0)
|
2010-01-06 02:17:10 +08:00
|
|
|
goto out_sleep;
|
2009-11-26 16:56:06 +08:00
|
|
|
|
2010-01-12 16:43:07 +08:00
|
|
|
ret = wl1251_build_qos_null_data(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2009-11-26 16:56:06 +08:00
|
|
|
if (wl->bss_type != BSS_TYPE_IBSS) {
|
|
|
|
ret = wl1251_join(wl, wl->bss_type, wl->channel,
|
|
|
|
wl->beacon_int, wl->dtim_period);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
if (changed & BSS_CHANGED_ASSOC) {
|
|
|
|
if (bss_conf->assoc) {
|
2009-08-07 18:34:12 +08:00
|
|
|
wl->beacon_int = bss_conf->beacon_int;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2010-01-06 02:16:32 +08:00
|
|
|
skb = ieee80211_pspoll_get(wl->hw, wl->vif);
|
|
|
|
if (!skb)
|
|
|
|
goto out_sleep;
|
|
|
|
|
|
|
|
ret = wl1251_cmd_template_set(wl, CMD_PS_POLL,
|
|
|
|
skb->data,
|
|
|
|
skb->len);
|
|
|
|
dev_kfree_skb(skb);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
2009-06-12 19:16:32 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2010-01-26 21:19:52 +08:00
|
|
|
ret = wl1251_acx_aid(wl, bss_conf->aid);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0)
|
2009-06-12 19:16:32 +08:00
|
|
|
goto out_sleep;
|
2009-08-07 18:34:12 +08:00
|
|
|
} else {
|
|
|
|
/* use defaults when not associated */
|
|
|
|
wl->beacon_int = WL1251_DEFAULT_BEACON_INT;
|
|
|
|
wl->dtim_period = WL1251_DEFAULT_DTIM_PERIOD;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (changed & BSS_CHANGED_ERP_SLOT) {
|
|
|
|
if (bss_conf->use_short_slot)
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_slot(wl, SLOT_TIME_SHORT);
|
2009-04-30 04:33:31 +08:00
|
|
|
else
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_slot(wl, SLOT_TIME_LONG);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_warning("Set slot time failed %d", ret);
|
2009-06-12 19:16:32 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (changed & BSS_CHANGED_ERP_PREAMBLE) {
|
|
|
|
if (bss_conf->use_short_preamble)
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_acx_set_preamble(wl, ACX_PREAMBLE_SHORT);
|
2009-04-30 04:33:31 +08:00
|
|
|
else
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_acx_set_preamble(wl, ACX_PREAMBLE_LONG);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (changed & BSS_CHANGED_ERP_CTS_PROT) {
|
|
|
|
if (bss_conf->use_cts_prot)
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_cts_protect(wl, CTSPROTECT_ENABLE);
|
2009-04-30 04:33:31 +08:00
|
|
|
else
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_acx_cts_protect(wl, CTSPROTECT_DISABLE);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (ret < 0) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_warning("Set ctsprotect failed %d", ret);
|
2010-01-06 02:17:10 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (changed & BSS_CHANGED_BEACON) {
|
|
|
|
beacon = ieee80211_beacon_get(hw, vif);
|
2011-02-04 04:14:01 +08:00
|
|
|
if (!beacon)
|
|
|
|
goto out_sleep;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_cmd_template_set(wl, CMD_BEACON, beacon->data,
|
2009-04-30 04:33:31 +08:00
|
|
|
beacon->len);
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
dev_kfree_skb(beacon);
|
2010-01-06 02:17:10 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
ret = wl1251_cmd_template_set(wl, CMD_PROBE_RESP, beacon->data,
|
2009-04-30 04:33:31 +08:00
|
|
|
beacon->len);
|
|
|
|
|
|
|
|
dev_kfree_skb(beacon);
|
|
|
|
|
|
|
|
if (ret < 0)
|
2010-01-06 02:17:10 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:34:42 +08:00
|
|
|
ret = wl1251_join(wl, wl->bss_type, wl->beacon_int,
|
|
|
|
wl->channel, wl->dtim_period);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
if (ret < 0)
|
2010-01-06 02:17:10 +08:00
|
|
|
goto out_sleep;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
2009-06-12 19:16:32 +08:00
|
|
|
out_sleep:
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
2009-06-12 19:16:32 +08:00
|
|
|
|
|
|
|
out:
|
2009-04-30 04:33:31 +08:00
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* can't be const, mac80211 writes to this */
|
2009-06-12 19:17:39 +08:00
|
|
|
static struct ieee80211_rate wl1251_rates[] = {
|
2009-04-30 04:33:31 +08:00
|
|
|
{ .bitrate = 10,
|
|
|
|
.hw_value = 0x1,
|
|
|
|
.hw_value_short = 0x1, },
|
|
|
|
{ .bitrate = 20,
|
|
|
|
.hw_value = 0x2,
|
|
|
|
.hw_value_short = 0x2,
|
|
|
|
.flags = IEEE80211_RATE_SHORT_PREAMBLE },
|
|
|
|
{ .bitrate = 55,
|
|
|
|
.hw_value = 0x4,
|
|
|
|
.hw_value_short = 0x4,
|
|
|
|
.flags = IEEE80211_RATE_SHORT_PREAMBLE },
|
|
|
|
{ .bitrate = 110,
|
|
|
|
.hw_value = 0x20,
|
|
|
|
.hw_value_short = 0x20,
|
|
|
|
.flags = IEEE80211_RATE_SHORT_PREAMBLE },
|
|
|
|
{ .bitrate = 60,
|
|
|
|
.hw_value = 0x8,
|
|
|
|
.hw_value_short = 0x8, },
|
|
|
|
{ .bitrate = 90,
|
|
|
|
.hw_value = 0x10,
|
|
|
|
.hw_value_short = 0x10, },
|
|
|
|
{ .bitrate = 120,
|
|
|
|
.hw_value = 0x40,
|
|
|
|
.hw_value_short = 0x40, },
|
|
|
|
{ .bitrate = 180,
|
|
|
|
.hw_value = 0x80,
|
|
|
|
.hw_value_short = 0x80, },
|
|
|
|
{ .bitrate = 240,
|
|
|
|
.hw_value = 0x200,
|
|
|
|
.hw_value_short = 0x200, },
|
|
|
|
{ .bitrate = 360,
|
|
|
|
.hw_value = 0x400,
|
|
|
|
.hw_value_short = 0x400, },
|
|
|
|
{ .bitrate = 480,
|
|
|
|
.hw_value = 0x800,
|
|
|
|
.hw_value_short = 0x800, },
|
|
|
|
{ .bitrate = 540,
|
|
|
|
.hw_value = 0x1000,
|
|
|
|
.hw_value_short = 0x1000, },
|
|
|
|
};
|
|
|
|
|
|
|
|
/* can't be const, mac80211 writes to this */
|
2009-06-12 19:17:39 +08:00
|
|
|
static struct ieee80211_channel wl1251_channels[] = {
|
2009-04-30 04:33:31 +08:00
|
|
|
{ .hw_value = 1, .center_freq = 2412},
|
|
|
|
{ .hw_value = 2, .center_freq = 2417},
|
|
|
|
{ .hw_value = 3, .center_freq = 2422},
|
|
|
|
{ .hw_value = 4, .center_freq = 2427},
|
|
|
|
{ .hw_value = 5, .center_freq = 2432},
|
|
|
|
{ .hw_value = 6, .center_freq = 2437},
|
|
|
|
{ .hw_value = 7, .center_freq = 2442},
|
|
|
|
{ .hw_value = 8, .center_freq = 2447},
|
|
|
|
{ .hw_value = 9, .center_freq = 2452},
|
|
|
|
{ .hw_value = 10, .center_freq = 2457},
|
|
|
|
{ .hw_value = 11, .center_freq = 2462},
|
|
|
|
{ .hw_value = 12, .center_freq = 2467},
|
|
|
|
{ .hw_value = 13, .center_freq = 2472},
|
|
|
|
};
|
|
|
|
|
2009-11-30 16:18:19 +08:00
|
|
|
static int wl1251_op_conf_tx(struct ieee80211_hw *hw, u16 queue,
|
|
|
|
const struct ieee80211_tx_queue_params *params)
|
|
|
|
{
|
2010-01-12 16:43:15 +08:00
|
|
|
enum wl1251_acx_ps_scheme ps_scheme;
|
2009-11-30 16:18:19 +08:00
|
|
|
struct wl1251 *wl = hw->priv;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&wl->mutex);
|
|
|
|
|
|
|
|
wl1251_debug(DEBUG_MAC80211, "mac80211 conf tx %d", queue);
|
|
|
|
|
|
|
|
ret = wl1251_ps_elp_wakeup(wl);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2010-02-04 21:33:25 +08:00
|
|
|
/* mac80211 uses units of 32 usec */
|
2009-11-30 16:18:19 +08:00
|
|
|
ret = wl1251_acx_ac_cfg(wl, wl1251_tx_get_queue(queue),
|
|
|
|
params->cw_min, params->cw_max,
|
2010-02-04 21:33:25 +08:00
|
|
|
params->aifs, params->txop * 32);
|
2009-11-30 16:18:27 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
|
|
|
|
2010-01-12 16:43:15 +08:00
|
|
|
if (params->uapsd)
|
|
|
|
ps_scheme = WL1251_ACX_PS_SCHEME_UPSD_TRIGGER;
|
|
|
|
else
|
|
|
|
ps_scheme = WL1251_ACX_PS_SCHEME_LEGACY;
|
|
|
|
|
2009-11-30 16:18:27 +08:00
|
|
|
ret = wl1251_acx_tid_cfg(wl, wl1251_tx_get_queue(queue),
|
2009-11-30 16:18:33 +08:00
|
|
|
CHANNEL_TYPE_EDCF,
|
2010-01-12 16:43:15 +08:00
|
|
|
wl1251_tx_get_queue(queue), ps_scheme,
|
2009-11-30 16:18:27 +08:00
|
|
|
WL1251_ACX_ACK_POLICY_LEGACY);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out_sleep;
|
2009-11-30 16:18:19 +08:00
|
|
|
|
2009-11-30 16:18:27 +08:00
|
|
|
out_sleep:
|
2009-11-30 16:18:19 +08:00
|
|
|
wl1251_ps_elp_sleep(wl);
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&wl->mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-07-29 03:23:30 +08:00
|
|
|
static int wl1251_op_get_survey(struct ieee80211_hw *hw, int idx,
|
|
|
|
struct survey_info *survey)
|
|
|
|
{
|
|
|
|
struct wl1251 *wl = hw->priv;
|
|
|
|
struct ieee80211_conf *conf = &hw->conf;
|
|
|
|
|
|
|
|
if (idx != 0)
|
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
survey->channel = conf->channel;
|
|
|
|
survey->filled = SURVEY_INFO_NOISE_DBM;
|
|
|
|
survey->noise = wl->noise;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
/* can't be const, mac80211 writes to this */
|
2009-06-12 19:17:39 +08:00
|
|
|
static struct ieee80211_supported_band wl1251_band_2ghz = {
|
|
|
|
.channels = wl1251_channels,
|
|
|
|
.n_channels = ARRAY_SIZE(wl1251_channels),
|
|
|
|
.bitrates = wl1251_rates,
|
|
|
|
.n_bitrates = ARRAY_SIZE(wl1251_rates),
|
2009-04-30 04:33:31 +08:00
|
|
|
};
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static const struct ieee80211_ops wl1251_ops = {
|
|
|
|
.start = wl1251_op_start,
|
|
|
|
.stop = wl1251_op_stop,
|
|
|
|
.add_interface = wl1251_op_add_interface,
|
|
|
|
.remove_interface = wl1251_op_remove_interface,
|
|
|
|
.config = wl1251_op_config,
|
|
|
|
.configure_filter = wl1251_op_configure_filter,
|
|
|
|
.tx = wl1251_op_tx,
|
|
|
|
.set_key = wl1251_op_set_key,
|
|
|
|
.hw_scan = wl1251_op_hw_scan,
|
|
|
|
.bss_info_changed = wl1251_op_bss_info_changed,
|
|
|
|
.set_rts_threshold = wl1251_op_set_rts_threshold,
|
2009-11-30 16:18:19 +08:00
|
|
|
.conf_tx = wl1251_op_conf_tx,
|
2010-07-29 03:23:30 +08:00
|
|
|
.get_survey = wl1251_op_get_survey,
|
2009-04-30 04:33:31 +08:00
|
|
|
};
|
|
|
|
|
2010-04-14 18:05:48 +08:00
|
|
|
static int wl1251_read_eeprom_byte(struct wl1251 *wl, off_t offset, u8 *data)
|
|
|
|
{
|
|
|
|
unsigned long timeout;
|
|
|
|
|
|
|
|
wl1251_reg_write32(wl, EE_ADDR, offset);
|
|
|
|
wl1251_reg_write32(wl, EE_CTL, EE_CTL_READ);
|
|
|
|
|
|
|
|
/* EE_CTL_READ clears when data is ready */
|
|
|
|
timeout = jiffies + msecs_to_jiffies(100);
|
|
|
|
while (1) {
|
|
|
|
if (!(wl1251_reg_read32(wl, EE_CTL) & EE_CTL_READ))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (time_after(jiffies, timeout))
|
|
|
|
return -ETIMEDOUT;
|
|
|
|
|
|
|
|
msleep(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
*data = wl1251_reg_read32(wl, EE_DATA);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1251_read_eeprom(struct wl1251 *wl, off_t offset,
|
|
|
|
u8 *data, size_t len)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
wl1251_reg_write32(wl, EE_START, 0);
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
ret = wl1251_read_eeprom_byte(wl, offset + i, &data[i]);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int wl1251_read_eeprom_mac(struct wl1251 *wl)
|
|
|
|
{
|
|
|
|
u8 mac[ETH_ALEN];
|
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
wl1251_set_partition(wl, 0, 0, REGISTERS_BASE, REGISTERS_DOWN_SIZE);
|
|
|
|
|
|
|
|
ret = wl1251_read_eeprom(wl, 0x1c, mac, sizeof(mac));
|
|
|
|
if (ret < 0) {
|
|
|
|
wl1251_warning("failed to read MAC address from EEPROM");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* MAC is stored in reverse order */
|
|
|
|
for (i = 0; i < ETH_ALEN; i++)
|
|
|
|
wl->mac_addr[i] = mac[ETH_ALEN - i - 1];
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
static int wl1251_register_hw(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (wl->mac80211_registered)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
SET_IEEE80211_PERM_ADDR(wl->hw, wl->mac_addr);
|
|
|
|
|
|
|
|
ret = ieee80211_register_hw(wl->hw);
|
|
|
|
if (ret < 0) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("unable to register mac80211 hw: %d", ret);
|
2009-04-30 04:33:31 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->mac80211_registered = true;
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_notice("loaded");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-08-07 18:33:26 +08:00
|
|
|
int wl1251_init_ieee80211(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
2009-08-07 18:33:26 +08:00
|
|
|
int ret;
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
/* The tx descriptor buffer and the TKIP space */
|
|
|
|
wl->hw->extra_tx_headroom = sizeof(struct tx_double_buffer_desc)
|
2009-06-12 19:15:54 +08:00
|
|
|
+ WL1251_TKIP_IV_SPACE;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
/* unit us */
|
|
|
|
/* FIXME: find a proper value */
|
|
|
|
wl->hw->channel_change_time = 10000;
|
|
|
|
|
|
|
|
wl->hw->flags = IEEE80211_HW_SIGNAL_DBM |
|
2009-11-18 00:48:30 +08:00
|
|
|
IEEE80211_HW_SUPPORTS_PS |
|
2010-01-12 16:43:15 +08:00
|
|
|
IEEE80211_HW_BEACON_FILTER |
|
2011-01-31 03:11:00 +08:00
|
|
|
IEEE80211_HW_SUPPORTS_UAPSD |
|
|
|
|
IEEE80211_HW_SUPPORTS_CQM_RSSI;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2011-01-31 03:11:04 +08:00
|
|
|
wl->hw->wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
|
|
|
|
BIT(NL80211_IFTYPE_ADHOC);
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->hw->wiphy->max_scan_ssids = 1;
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->hw->wiphy->bands[IEEE80211_BAND_2GHZ] = &wl1251_band_2ghz;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-11-30 16:18:47 +08:00
|
|
|
wl->hw->queues = 4;
|
|
|
|
|
2010-04-14 18:05:48 +08:00
|
|
|
if (wl->use_eeprom)
|
|
|
|
wl1251_read_eeprom_mac(wl);
|
|
|
|
|
2009-08-07 18:33:26 +08:00
|
|
|
ret = wl1251_register_hw(wl);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:33:26 +08:00
|
|
|
wl1251_debugfs_init(wl);
|
|
|
|
wl1251_notice("initialized");
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:33:26 +08:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
2009-08-07 18:33:34 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1251_init_ieee80211);
|
2009-08-07 18:33:11 +08:00
|
|
|
|
2009-08-07 18:33:26 +08:00
|
|
|
struct ieee80211_hw *wl1251_alloc_hw(void)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
|
|
|
struct ieee80211_hw *hw;
|
2009-06-12 19:17:39 +08:00
|
|
|
struct wl1251 *wl;
|
2009-08-07 18:33:26 +08:00
|
|
|
int i;
|
2009-04-30 04:33:31 +08:00
|
|
|
static const u8 nokia_oui[3] = {0x00, 0x1f, 0xdf};
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
hw = ieee80211_alloc_hw(sizeof(*wl), &wl1251_ops);
|
2009-04-30 04:33:31 +08:00
|
|
|
if (!hw) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("could not alloc ieee80211_hw");
|
2009-08-07 18:33:26 +08:00
|
|
|
return ERR_PTR(-ENOMEM);
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
wl = hw->priv;
|
|
|
|
memset(wl, 0, sizeof(*wl));
|
|
|
|
|
|
|
|
wl->hw = hw;
|
|
|
|
|
|
|
|
wl->data_in_count = 0;
|
|
|
|
|
|
|
|
skb_queue_head_init(&wl->tx_queue);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
INIT_WORK(&wl->filter_work, wl1251_filter_work);
|
2009-11-18 00:48:37 +08:00
|
|
|
INIT_DELAYED_WORK(&wl->elp_work, wl1251_elp_work);
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->channel = WL1251_DEFAULT_CHANNEL;
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->scanning = false;
|
|
|
|
wl->default_key = 0;
|
|
|
|
wl->listen_int = 1;
|
|
|
|
wl->rx_counter = 0;
|
|
|
|
wl->rx_handled = 0;
|
|
|
|
wl->rx_current_buffer = 0;
|
|
|
|
wl->rx_last_id = 0;
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->rx_config = WL1251_DEFAULT_RX_CONFIG;
|
|
|
|
wl->rx_filter = WL1251_DEFAULT_RX_FILTER;
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->elp = false;
|
2011-04-04 16:04:57 +08:00
|
|
|
wl->station_mode = STATION_ACTIVE_MODE;
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->psm_requested = false;
|
|
|
|
wl->tx_queue_stopped = false;
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->power_level = WL1251_DEFAULT_POWER_LEVEL;
|
2011-01-31 03:11:00 +08:00
|
|
|
wl->rssi_thold = 0;
|
2009-08-07 18:34:12 +08:00
|
|
|
wl->beacon_int = WL1251_DEFAULT_BEACON_INT;
|
|
|
|
wl->dtim_period = WL1251_DEFAULT_DTIM_PERIOD;
|
2009-11-18 00:48:23 +08:00
|
|
|
wl->vif = NULL;
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
for (i = 0; i < FW_TX_CMPLT_BLOCK_SIZE; i++)
|
|
|
|
wl->tx_frames[i] = NULL;
|
|
|
|
|
|
|
|
wl->next_tx_complete = 0;
|
|
|
|
|
2009-08-07 18:33:57 +08:00
|
|
|
INIT_WORK(&wl->irq_work, wl1251_irq_work);
|
|
|
|
INIT_WORK(&wl->tx_work, wl1251_tx_work);
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
/*
|
|
|
|
* In case our MAC address is not correctly set,
|
|
|
|
* we use a random but Nokia MAC.
|
|
|
|
*/
|
|
|
|
memcpy(wl->mac_addr, nokia_oui, 3);
|
|
|
|
get_random_bytes(wl->mac_addr + 3, 3);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl->state = WL1251_STATE_OFF;
|
2009-04-30 04:33:31 +08:00
|
|
|
mutex_init(&wl->mutex);
|
|
|
|
|
|
|
|
wl->tx_mgmt_frm_rate = DEFAULT_HW_GEN_TX_RATE;
|
|
|
|
wl->tx_mgmt_frm_mod = DEFAULT_HW_GEN_MODULATION_TYPE;
|
|
|
|
|
2009-06-12 19:15:08 +08:00
|
|
|
wl->rx_descriptor = kmalloc(sizeof(*wl->rx_descriptor), GFP_KERNEL);
|
|
|
|
if (!wl->rx_descriptor) {
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_error("could not allocate memory for rx descriptor");
|
2009-08-07 18:33:26 +08:00
|
|
|
ieee80211_free_hw(hw);
|
|
|
|
return ERR_PTR(-ENOMEM);
|
2009-06-12 19:15:08 +08:00
|
|
|
}
|
|
|
|
|
2009-08-07 18:33:26 +08:00
|
|
|
return hw;
|
2009-04-30 04:33:31 +08:00
|
|
|
}
|
2009-08-07 18:33:34 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1251_alloc_hw);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
2009-08-07 18:33:26 +08:00
|
|
|
int wl1251_free_hw(struct wl1251 *wl)
|
2009-04-30 04:33:31 +08:00
|
|
|
{
|
|
|
|
ieee80211_unregister_hw(wl->hw);
|
|
|
|
|
2009-06-12 19:17:39 +08:00
|
|
|
wl1251_debugfs_exit(wl);
|
2009-04-30 04:33:31 +08:00
|
|
|
|
|
|
|
kfree(wl->target_mem_map);
|
|
|
|
kfree(wl->data_path);
|
2009-11-18 00:48:45 +08:00
|
|
|
vfree(wl->fw);
|
2009-04-30 04:33:31 +08:00
|
|
|
wl->fw = NULL;
|
|
|
|
kfree(wl->nvs);
|
|
|
|
wl->nvs = NULL;
|
2009-06-12 19:15:08 +08:00
|
|
|
|
|
|
|
kfree(wl->rx_descriptor);
|
|
|
|
wl->rx_descriptor = NULL;
|
|
|
|
|
2009-04-30 04:33:31 +08:00
|
|
|
ieee80211_free_hw(wl->hw);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2009-08-07 18:33:34 +08:00
|
|
|
EXPORT_SYMBOL_GPL(wl1251_free_hw);
|
|
|
|
|
|
|
|
MODULE_DESCRIPTION("TI wl1251 Wireles LAN Driver Core");
|
|
|
|
MODULE_LICENSE("GPL");
|
2010-08-23 03:46:15 +08:00
|
|
|
MODULE_AUTHOR("Kalle Valo <kvalo@adurom.com>");
|
2009-11-08 06:02:15 +08:00
|
|
|
MODULE_FIRMWARE(WL1251_FW_NAME);
|