2008-03-07 02:40:19 +08:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* GPL LICENSE SUMMARY
|
|
|
|
*
|
2010-01-16 05:43:41 +08:00
|
|
|
* Copyright(c) 2008 - 2010 Intel Corporation. All rights reserved.
|
2008-03-07 02:40:19 +08:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of version 2 of the GNU General Public License 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 Street, Fifth Floor, Boston, MA 02110,
|
|
|
|
* USA
|
|
|
|
*
|
|
|
|
* The full GNU General Public License is included in this distribution
|
|
|
|
* in the file called LICENSE.GPL.
|
|
|
|
*
|
|
|
|
* Contact Information:
|
2008-12-10 03:28:58 +08:00
|
|
|
* Intel Linux Wireless <ilw@linux.intel.com>
|
2008-03-07 02:40:19 +08:00
|
|
|
* Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
2009-01-28 06:27:52 +08:00
|
|
|
#include <linux/etherdevice.h>
|
2009-10-07 21:09:06 +08:00
|
|
|
#include <linux/sched.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>
|
2008-03-15 01:38:48 +08:00
|
|
|
#include <net/mac80211.h>
|
2008-03-07 02:40:19 +08:00
|
|
|
|
2008-03-12 07:17:18 +08:00
|
|
|
#include "iwl-eeprom.h"
|
2008-04-25 02:55:38 +08:00
|
|
|
#include "iwl-dev.h" /* FIXME: remove */
|
2008-11-08 01:58:36 +08:00
|
|
|
#include "iwl-debug.h"
|
2008-03-07 02:40:19 +08:00
|
|
|
#include "iwl-core.h"
|
2008-04-24 08:14:54 +08:00
|
|
|
#include "iwl-io.h"
|
2008-04-22 06:41:51 +08:00
|
|
|
#include "iwl-power.h"
|
2008-11-20 07:32:23 +08:00
|
|
|
#include "iwl-sta.h"
|
2009-05-23 02:01:50 +08:00
|
|
|
#include "iwl-helpers.h"
|
2008-03-07 02:40:19 +08:00
|
|
|
|
2008-03-15 01:38:48 +08:00
|
|
|
|
2008-03-07 02:40:19 +08:00
|
|
|
MODULE_DESCRIPTION("iwl core");
|
|
|
|
MODULE_VERSION(IWLWIFI_VERSION);
|
2008-12-12 02:33:41 +08:00
|
|
|
MODULE_AUTHOR(DRV_COPYRIGHT " " DRV_AUTHOR);
|
2008-03-13 07:58:52 +08:00
|
|
|
MODULE_LICENSE("GPL");
|
2008-03-07 02:40:19 +08:00
|
|
|
|
2010-01-23 06:22:51 +08:00
|
|
|
/*
|
|
|
|
* set bt_coex_active to true, uCode will do kill/defer
|
|
|
|
* every time the priority line is asserted (BT is sending signals on the
|
|
|
|
* priority line in the PCIx).
|
|
|
|
* set bt_coex_active to false, uCode will ignore the BT activity and
|
|
|
|
* perform the normal operation
|
|
|
|
*
|
|
|
|
* User might experience transmit issue on some platform due to WiFi/BT
|
|
|
|
* co-exist problem. The possible behaviors are:
|
|
|
|
* Able to scan and finding all the available AP
|
|
|
|
* Not able to associate with any AP
|
|
|
|
* On those platforms, WiFi communication can be restored by set
|
|
|
|
* "bt_coex_active" module parameter to "false"
|
|
|
|
*
|
|
|
|
* default: bt_coex_active = true (BT_COEX_ENABLE)
|
|
|
|
*/
|
2010-08-23 22:56:55 +08:00
|
|
|
bool bt_coex_active = true;
|
|
|
|
EXPORT_SYMBOL_GPL(bt_coex_active);
|
2010-01-23 06:22:51 +08:00
|
|
|
module_param(bt_coex_active, bool, S_IRUGO);
|
2010-03-19 06:37:49 +08:00
|
|
|
MODULE_PARM_DESC(bt_coex_active, "enable wifi/bluetooth co-exist");
|
2010-01-23 06:22:51 +08:00
|
|
|
|
2009-07-18 00:30:24 +08:00
|
|
|
u32 iwl_debug_level;
|
|
|
|
EXPORT_SYMBOL(iwl_debug_level);
|
|
|
|
|
2008-05-15 13:54:03 +08:00
|
|
|
const u8 iwl_bcast_addr[ETH_ALEN] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
|
|
|
|
EXPORT_SYMBOL(iwl_bcast_addr);
|
|
|
|
|
|
|
|
|
2008-03-15 01:38:48 +08:00
|
|
|
/* This function both allocates and initializes hw and priv. */
|
2010-10-24 00:15:39 +08:00
|
|
|
struct ieee80211_hw *iwl_alloc_all(struct iwl_cfg *cfg)
|
2008-03-15 01:38:48 +08:00
|
|
|
{
|
|
|
|
struct iwl_priv *priv;
|
|
|
|
/* mac80211 allocates memory for this device instance, including
|
|
|
|
* space for this driver's private structure */
|
2010-10-24 00:15:39 +08:00
|
|
|
struct ieee80211_hw *hw;
|
|
|
|
|
|
|
|
hw = ieee80211_alloc_hw(sizeof(struct iwl_priv),
|
|
|
|
cfg->ops->ieee80211_ops);
|
2008-03-15 01:38:48 +08:00
|
|
|
if (hw == NULL) {
|
2010-07-27 05:39:58 +08:00
|
|
|
pr_err("%s: Can not allocate network device\n",
|
2008-12-19 10:37:09 +08:00
|
|
|
cfg->name);
|
2008-03-15 01:38:48 +08:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
priv = hw->priv;
|
|
|
|
priv->hw = hw;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return hw;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_alloc_all);
|
|
|
|
|
2009-02-28 08:21:24 +08:00
|
|
|
/*
|
|
|
|
* QoS support
|
|
|
|
*/
|
2010-08-23 16:46:38 +08:00
|
|
|
static void iwl_update_qos(struct iwl_priv *priv, struct iwl_rxon_context *ctx)
|
2009-02-28 08:21:24 +08:00
|
|
|
{
|
|
|
|
if (test_bit(STATUS_EXIT_PENDING, &priv->status))
|
|
|
|
return;
|
|
|
|
|
2010-09-03 21:32:21 +08:00
|
|
|
if (!ctx->is_active)
|
|
|
|
return;
|
|
|
|
|
2010-08-23 16:46:38 +08:00
|
|
|
ctx->qos_data.def_qos_parm.qos_flags = 0;
|
2009-02-28 08:21:24 +08:00
|
|
|
|
2010-08-23 16:46:38 +08:00
|
|
|
if (ctx->qos_data.qos_active)
|
|
|
|
ctx->qos_data.def_qos_parm.qos_flags |=
|
2009-02-28 08:21:24 +08:00
|
|
|
QOS_PARAM_FLG_UPDATE_EDCA_MSK;
|
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
if (ctx->ht.enabled)
|
2010-08-23 16:46:38 +08:00
|
|
|
ctx->qos_data.def_qos_parm.qos_flags |= QOS_PARAM_FLG_TGN_MSK;
|
2009-02-28 08:21:24 +08:00
|
|
|
|
2010-03-29 18:18:35 +08:00
|
|
|
IWL_DEBUG_QOS(priv, "send QoS cmd with Qos active=%d FLAGS=0x%X\n",
|
2010-08-23 16:46:38 +08:00
|
|
|
ctx->qos_data.qos_active,
|
|
|
|
ctx->qos_data.def_qos_parm.qos_flags);
|
2009-02-28 08:21:24 +08:00
|
|
|
|
2010-08-23 16:46:38 +08:00
|
|
|
iwl_send_cmd_pdu_async(priv, ctx->qos_cmd,
|
2010-03-29 18:18:35 +08:00
|
|
|
sizeof(struct iwl_qosparam_cmd),
|
2010-08-23 16:46:38 +08:00
|
|
|
&ctx->qos_data.def_qos_parm, NULL);
|
2009-02-28 08:21:24 +08:00
|
|
|
}
|
2008-04-24 08:15:05 +08:00
|
|
|
|
2008-10-09 18:13:49 +08:00
|
|
|
#define MAX_BIT_RATE_40_MHZ 150 /* Mbps */
|
|
|
|
#define MAX_BIT_RATE_20_MHZ 72 /* Mbps */
|
2008-04-24 08:15:05 +08:00
|
|
|
static void iwlcore_init_ht_hw_capab(const struct iwl_priv *priv,
|
2008-10-09 18:13:49 +08:00
|
|
|
struct ieee80211_sta_ht_cap *ht_info,
|
2008-04-24 08:15:05 +08:00
|
|
|
enum ieee80211_band band)
|
|
|
|
{
|
2008-05-15 13:53:56 +08:00
|
|
|
u16 max_bit_rate = 0;
|
|
|
|
u8 rx_chains_num = priv->hw_params.rx_chains_num;
|
|
|
|
u8 tx_chains_num = priv->hw_params.tx_chains_num;
|
|
|
|
|
2008-04-24 08:15:05 +08:00
|
|
|
ht_info->cap = 0;
|
2008-10-09 18:13:49 +08:00
|
|
|
memset(&ht_info->mcs, 0, sizeof(ht_info->mcs));
|
2008-04-24 08:15:05 +08:00
|
|
|
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->ht_supported = true;
|
2008-04-24 08:15:05 +08:00
|
|
|
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->ht_params &&
|
|
|
|
priv->cfg->ht_params->ht_greenfield_support)
|
2009-08-14 04:30:59 +08:00
|
|
|
ht_info->cap |= IEEE80211_HT_CAP_GRN_FLD;
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->cap |= IEEE80211_HT_CAP_SGI_20;
|
2008-05-15 13:53:56 +08:00
|
|
|
max_bit_rate = MAX_BIT_RATE_20_MHZ;
|
2009-08-08 06:41:38 +08:00
|
|
|
if (priv->hw_params.ht40_channel & BIT(band)) {
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->cap |= IEEE80211_HT_CAP_SUP_WIDTH_20_40;
|
|
|
|
ht_info->cap |= IEEE80211_HT_CAP_SGI_40;
|
|
|
|
ht_info->mcs.rx_mask[4] = 0x01;
|
2008-05-15 13:53:56 +08:00
|
|
|
max_bit_rate = MAX_BIT_RATE_40_MHZ;
|
2008-04-24 08:15:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (priv->cfg->mod_params->amsdu_size_8K)
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->cap |= IEEE80211_HT_CAP_MAX_AMSDU;
|
2008-04-24 08:15:05 +08:00
|
|
|
|
|
|
|
ht_info->ampdu_factor = CFG_HT_RX_AMPDU_FACTOR_DEF;
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->bt_params && priv->cfg->bt_params->ampdu_factor)
|
|
|
|
ht_info->ampdu_factor = priv->cfg->bt_params->ampdu_factor;
|
2008-04-24 08:15:05 +08:00
|
|
|
ht_info->ampdu_density = CFG_HT_MPDU_DENSITY_DEF;
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->bt_params && priv->cfg->bt_params->ampdu_density)
|
|
|
|
ht_info->ampdu_density = priv->cfg->bt_params->ampdu_density;
|
2008-04-24 08:15:05 +08:00
|
|
|
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->mcs.rx_mask[0] = 0xFF;
|
2008-05-15 13:53:56 +08:00
|
|
|
if (rx_chains_num >= 2)
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->mcs.rx_mask[1] = 0xFF;
|
2008-05-15 13:53:56 +08:00
|
|
|
if (rx_chains_num >= 3)
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->mcs.rx_mask[2] = 0xFF;
|
2008-05-15 13:53:56 +08:00
|
|
|
|
|
|
|
/* Highest supported Rx data rate */
|
|
|
|
max_bit_rate *= rx_chains_num;
|
2008-10-09 18:13:49 +08:00
|
|
|
WARN_ON(max_bit_rate & ~IEEE80211_HT_MCS_RX_HIGHEST_MASK);
|
|
|
|
ht_info->mcs.rx_highest = cpu_to_le16(max_bit_rate);
|
2008-05-15 13:53:56 +08:00
|
|
|
|
|
|
|
/* Tx MCS capabilities */
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->mcs.tx_params = IEEE80211_HT_MCS_TX_DEFINED;
|
2008-05-15 13:53:56 +08:00
|
|
|
if (tx_chains_num != rx_chains_num) {
|
2008-10-09 18:13:49 +08:00
|
|
|
ht_info->mcs.tx_params |= IEEE80211_HT_MCS_TX_RX_DIFF;
|
|
|
|
ht_info->mcs.tx_params |= ((tx_chains_num - 1) <<
|
|
|
|
IEEE80211_HT_MCS_TX_MAX_STREAMS_SHIFT);
|
2008-05-15 13:53:56 +08:00
|
|
|
}
|
2008-04-24 08:15:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* iwlcore_init_geos - Initialize mac80211's geo/channel info based from eeprom
|
|
|
|
*/
|
2009-01-24 05:45:16 +08:00
|
|
|
int iwlcore_init_geos(struct iwl_priv *priv)
|
2008-04-24 08:15:05 +08:00
|
|
|
{
|
|
|
|
struct iwl_channel_info *ch;
|
|
|
|
struct ieee80211_supported_band *sband;
|
|
|
|
struct ieee80211_channel *channels;
|
|
|
|
struct ieee80211_channel *geo_ch;
|
|
|
|
struct ieee80211_rate *rates;
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
if (priv->bands[IEEE80211_BAND_2GHZ].n_bitrates ||
|
|
|
|
priv->bands[IEEE80211_BAND_5GHZ].n_bitrates) {
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG_INFO(priv, "Geography modes already initialized.\n");
|
2008-04-24 08:15:05 +08:00
|
|
|
set_bit(STATUS_GEO_CONFIGURED, &priv->status);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
channels = kzalloc(sizeof(struct ieee80211_channel) *
|
|
|
|
priv->channel_count, GFP_KERNEL);
|
|
|
|
if (!channels)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2009-08-29 00:44:45 +08:00
|
|
|
rates = kzalloc((sizeof(struct ieee80211_rate) * IWL_RATE_COUNT_LEGACY),
|
2008-04-24 08:15:05 +08:00
|
|
|
GFP_KERNEL);
|
|
|
|
if (!rates) {
|
|
|
|
kfree(channels);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 5.2GHz channels start after the 2.4GHz channels */
|
|
|
|
sband = &priv->bands[IEEE80211_BAND_5GHZ];
|
|
|
|
sband->channels = &channels[ARRAY_SIZE(iwl_eeprom_band_1)];
|
|
|
|
/* just OFDM */
|
|
|
|
sband->bitrates = &rates[IWL_FIRST_OFDM_RATE];
|
2009-08-29 00:44:45 +08:00
|
|
|
sband->n_bitrates = IWL_RATE_COUNT_LEGACY - IWL_FIRST_OFDM_RATE;
|
2008-04-24 08:15:05 +08:00
|
|
|
|
2008-06-30 17:23:21 +08:00
|
|
|
if (priv->cfg->sku & IWL_SKU_N)
|
2008-10-09 18:13:49 +08:00
|
|
|
iwlcore_init_ht_hw_capab(priv, &sband->ht_cap,
|
2008-06-30 17:23:21 +08:00
|
|
|
IEEE80211_BAND_5GHZ);
|
2008-04-24 08:15:05 +08:00
|
|
|
|
|
|
|
sband = &priv->bands[IEEE80211_BAND_2GHZ];
|
|
|
|
sband->channels = channels;
|
|
|
|
/* OFDM & CCK */
|
|
|
|
sband->bitrates = rates;
|
2009-08-29 00:44:45 +08:00
|
|
|
sband->n_bitrates = IWL_RATE_COUNT_LEGACY;
|
2008-04-24 08:15:05 +08:00
|
|
|
|
2008-06-30 17:23:21 +08:00
|
|
|
if (priv->cfg->sku & IWL_SKU_N)
|
2008-10-09 18:13:49 +08:00
|
|
|
iwlcore_init_ht_hw_capab(priv, &sband->ht_cap,
|
2008-06-30 17:23:21 +08:00
|
|
|
IEEE80211_BAND_2GHZ);
|
2008-04-24 08:15:05 +08:00
|
|
|
|
|
|
|
priv->ieee_channels = channels;
|
|
|
|
priv->ieee_rates = rates;
|
|
|
|
|
|
|
|
for (i = 0; i < priv->channel_count; i++) {
|
|
|
|
ch = &priv->channel_info[i];
|
|
|
|
|
|
|
|
/* FIXME: might be removed if scan is OK */
|
|
|
|
if (!is_channel_valid(ch))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (is_channel_a_band(ch))
|
|
|
|
sband = &priv->bands[IEEE80211_BAND_5GHZ];
|
|
|
|
else
|
|
|
|
sband = &priv->bands[IEEE80211_BAND_2GHZ];
|
|
|
|
|
|
|
|
geo_ch = &sband->channels[sband->n_channels++];
|
|
|
|
|
|
|
|
geo_ch->center_freq =
|
|
|
|
ieee80211_channel_to_frequency(ch->channel);
|
|
|
|
geo_ch->max_power = ch->max_power_avg;
|
|
|
|
geo_ch->max_antenna_gain = 0xff;
|
|
|
|
geo_ch->hw_value = ch->channel;
|
|
|
|
|
|
|
|
if (is_channel_valid(ch)) {
|
|
|
|
if (!(ch->flags & EEPROM_CHANNEL_IBSS))
|
|
|
|
geo_ch->flags |= IEEE80211_CHAN_NO_IBSS;
|
|
|
|
|
|
|
|
if (!(ch->flags & EEPROM_CHANNEL_ACTIVE))
|
|
|
|
geo_ch->flags |= IEEE80211_CHAN_PASSIVE_SCAN;
|
|
|
|
|
|
|
|
if (ch->flags & EEPROM_CHANNEL_RADAR)
|
|
|
|
geo_ch->flags |= IEEE80211_CHAN_RADAR;
|
|
|
|
|
2009-08-08 06:41:38 +08:00
|
|
|
geo_ch->flags |= ch->ht40_extension_channel;
|
2008-05-29 16:35:24 +08:00
|
|
|
|
2009-08-22 04:34:16 +08:00
|
|
|
if (ch->max_power_avg > priv->tx_power_device_lmt)
|
|
|
|
priv->tx_power_device_lmt = ch->max_power_avg;
|
2008-04-24 08:15:05 +08:00
|
|
|
} else {
|
|
|
|
geo_ch->flags |= IEEE80211_CHAN_DISABLED;
|
|
|
|
}
|
|
|
|
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG_INFO(priv, "Channel %d Freq=%d[%sGHz] %s flag=0x%X\n",
|
2008-04-24 08:15:05 +08:00
|
|
|
ch->channel, geo_ch->center_freq,
|
|
|
|
is_channel_a_band(ch) ? "5.2" : "2.4",
|
|
|
|
geo_ch->flags & IEEE80211_CHAN_DISABLED ?
|
|
|
|
"restricted" : "valid",
|
|
|
|
geo_ch->flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((priv->bands[IEEE80211_BAND_5GHZ].n_channels == 0) &&
|
|
|
|
priv->cfg->sku & IWL_SKU_A) {
|
2008-12-19 10:37:31 +08:00
|
|
|
IWL_INFO(priv, "Incorrectly detected BG card as ABG. "
|
|
|
|
"Please send your PCI ID 0x%04X:0x%04X to maintainer.\n",
|
2008-12-19 10:37:09 +08:00
|
|
|
priv->pci_dev->device,
|
|
|
|
priv->pci_dev->subsystem_device);
|
2008-04-24 08:15:05 +08:00
|
|
|
priv->cfg->sku &= ~IWL_SKU_A;
|
|
|
|
}
|
|
|
|
|
2008-12-19 10:37:31 +08:00
|
|
|
IWL_INFO(priv, "Tunable channels: %d 802.11bg, %d 802.11a channels\n",
|
2008-12-19 10:37:09 +08:00
|
|
|
priv->bands[IEEE80211_BAND_2GHZ].n_channels,
|
|
|
|
priv->bands[IEEE80211_BAND_5GHZ].n_channels);
|
2008-04-24 08:15:05 +08:00
|
|
|
|
|
|
|
set_bit(STATUS_GEO_CONFIGURED, &priv->status);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2009-01-24 05:45:16 +08:00
|
|
|
EXPORT_SYMBOL(iwlcore_init_geos);
|
2008-04-24 08:15:05 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* iwlcore_free_geos - undo allocations in iwlcore_init_geos
|
|
|
|
*/
|
2009-01-24 05:45:16 +08:00
|
|
|
void iwlcore_free_geos(struct iwl_priv *priv)
|
2008-04-24 08:15:05 +08:00
|
|
|
{
|
|
|
|
kfree(priv->ieee_channels);
|
|
|
|
kfree(priv->ieee_rates);
|
|
|
|
clear_bit(STATUS_GEO_CONFIGURED, &priv->status);
|
|
|
|
}
|
2009-01-24 05:45:16 +08:00
|
|
|
EXPORT_SYMBOL(iwlcore_free_geos);
|
2008-04-24 08:15:05 +08:00
|
|
|
|
2009-10-10 04:20:30 +08:00
|
|
|
/*
|
iwlagn: fix rts cts protection
Currently the driver will try to protect all frames,
which leads to a lot of odd things like sending an
RTS with a zeroed RA before multicast frames, which
is clearly bogus.
In order to fix all of this, we need to take a step
back and see what we need to achieve:
* we need RTS/CTS protection if requested by
the AP for the BSS, mac80211 tells us this
* in that case, CTS-to-self should only be
enabled when mac80211 tells us
* additionally, as a hardware workaround, on
some devices we have to protect aggregated
frames with RTS
To achieve the first two items, set up the RXON
accordingly and set the protection required flag
in the transmit command when mac80211 requests
protection for the frame.
To achieve the last item, set the rate-control
RTS-requested flag for all stations that we have
aggregation sessions with, and set the protection
required flag when sending aggregated frames (on
those devices where this is required).
Since otherwise bugs can occur, do not allow the
user to override the RTS-for-aggregation setting
from sysfs any more.
Finally, also clean up the way all these flags get
set in the driver and move everything into the
device-specific functions.
Cc: stable@kernel.org [2.6.35]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2010-08-10 01:57:02 +08:00
|
|
|
* iwlcore_tx_cmd_protection: Set rts/cts. 3945 and 4965 only share this
|
2009-10-10 04:20:30 +08:00
|
|
|
* function.
|
|
|
|
*/
|
iwlagn: fix rts cts protection
Currently the driver will try to protect all frames,
which leads to a lot of odd things like sending an
RTS with a zeroed RA before multicast frames, which
is clearly bogus.
In order to fix all of this, we need to take a step
back and see what we need to achieve:
* we need RTS/CTS protection if requested by
the AP for the BSS, mac80211 tells us this
* in that case, CTS-to-self should only be
enabled when mac80211 tells us
* additionally, as a hardware workaround, on
some devices we have to protect aggregated
frames with RTS
To achieve the first two items, set up the RXON
accordingly and set the protection required flag
in the transmit command when mac80211 requests
protection for the frame.
To achieve the last item, set the rate-control
RTS-requested flag for all stations that we have
aggregation sessions with, and set the protection
required flag when sending aggregated frames (on
those devices where this is required).
Since otherwise bugs can occur, do not allow the
user to override the RTS-for-aggregation setting
from sysfs any more.
Finally, also clean up the way all these flags get
set in the driver and move everything into the
device-specific functions.
Cc: stable@kernel.org [2.6.35]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2010-08-10 01:57:02 +08:00
|
|
|
void iwlcore_tx_cmd_protection(struct iwl_priv *priv,
|
|
|
|
struct ieee80211_tx_info *info,
|
|
|
|
__le16 fc, __le32 *tx_flags)
|
2009-10-10 04:20:30 +08:00
|
|
|
{
|
|
|
|
if (info->control.rates[0].flags & IEEE80211_TX_RC_USE_RTS_CTS) {
|
|
|
|
*tx_flags |= TX_CMD_FLG_RTS_MSK;
|
|
|
|
*tx_flags &= ~TX_CMD_FLG_CTS_MSK;
|
iwlagn: fix rts cts protection
Currently the driver will try to protect all frames,
which leads to a lot of odd things like sending an
RTS with a zeroed RA before multicast frames, which
is clearly bogus.
In order to fix all of this, we need to take a step
back and see what we need to achieve:
* we need RTS/CTS protection if requested by
the AP for the BSS, mac80211 tells us this
* in that case, CTS-to-self should only be
enabled when mac80211 tells us
* additionally, as a hardware workaround, on
some devices we have to protect aggregated
frames with RTS
To achieve the first two items, set up the RXON
accordingly and set the protection required flag
in the transmit command when mac80211 requests
protection for the frame.
To achieve the last item, set the rate-control
RTS-requested flag for all stations that we have
aggregation sessions with, and set the protection
required flag when sending aggregated frames (on
those devices where this is required).
Since otherwise bugs can occur, do not allow the
user to override the RTS-for-aggregation setting
from sysfs any more.
Finally, also clean up the way all these flags get
set in the driver and move everything into the
device-specific functions.
Cc: stable@kernel.org [2.6.35]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2010-08-10 01:57:02 +08:00
|
|
|
*tx_flags |= TX_CMD_FLG_FULL_TXOP_PROT_MSK;
|
|
|
|
|
|
|
|
if (!ieee80211_is_mgmt(fc))
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (fc & cpu_to_le16(IEEE80211_FCTL_STYPE)) {
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_AUTH):
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_DEAUTH):
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_ASSOC_REQ):
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_REASSOC_REQ):
|
|
|
|
*tx_flags &= ~TX_CMD_FLG_RTS_MSK;
|
|
|
|
*tx_flags |= TX_CMD_FLG_CTS_MSK;
|
|
|
|
break;
|
|
|
|
}
|
2009-10-10 04:20:30 +08:00
|
|
|
} else if (info->control.rates[0].flags & IEEE80211_TX_RC_USE_CTS_PROTECT) {
|
|
|
|
*tx_flags &= ~TX_CMD_FLG_RTS_MSK;
|
|
|
|
*tx_flags |= TX_CMD_FLG_CTS_MSK;
|
iwlagn: fix rts cts protection
Currently the driver will try to protect all frames,
which leads to a lot of odd things like sending an
RTS with a zeroed RA before multicast frames, which
is clearly bogus.
In order to fix all of this, we need to take a step
back and see what we need to achieve:
* we need RTS/CTS protection if requested by
the AP for the BSS, mac80211 tells us this
* in that case, CTS-to-self should only be
enabled when mac80211 tells us
* additionally, as a hardware workaround, on
some devices we have to protect aggregated
frames with RTS
To achieve the first two items, set up the RXON
accordingly and set the protection required flag
in the transmit command when mac80211 requests
protection for the frame.
To achieve the last item, set the rate-control
RTS-requested flag for all stations that we have
aggregation sessions with, and set the protection
required flag when sending aggregated frames (on
those devices where this is required).
Since otherwise bugs can occur, do not allow the
user to override the RTS-for-aggregation setting
from sysfs any more.
Finally, also clean up the way all these flags get
set in the driver and move everything into the
device-specific functions.
Cc: stable@kernel.org [2.6.35]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2010-08-10 01:57:02 +08:00
|
|
|
*tx_flags |= TX_CMD_FLG_FULL_TXOP_PROT_MSK;
|
2009-10-10 04:20:30 +08:00
|
|
|
}
|
|
|
|
}
|
iwlagn: fix rts cts protection
Currently the driver will try to protect all frames,
which leads to a lot of odd things like sending an
RTS with a zeroed RA before multicast frames, which
is clearly bogus.
In order to fix all of this, we need to take a step
back and see what we need to achieve:
* we need RTS/CTS protection if requested by
the AP for the BSS, mac80211 tells us this
* in that case, CTS-to-self should only be
enabled when mac80211 tells us
* additionally, as a hardware workaround, on
some devices we have to protect aggregated
frames with RTS
To achieve the first two items, set up the RXON
accordingly and set the protection required flag
in the transmit command when mac80211 requests
protection for the frame.
To achieve the last item, set the rate-control
RTS-requested flag for all stations that we have
aggregation sessions with, and set the protection
required flag when sending aggregated frames (on
those devices where this is required).
Since otherwise bugs can occur, do not allow the
user to override the RTS-for-aggregation setting
from sysfs any more.
Finally, also clean up the way all these flags get
set in the driver and move everything into the
device-specific functions.
Cc: stable@kernel.org [2.6.35]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2010-08-10 01:57:02 +08:00
|
|
|
EXPORT_SYMBOL(iwlcore_tx_cmd_protection);
|
|
|
|
|
2009-10-10 04:20:30 +08:00
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
static bool iwl_is_channel_extension(struct iwl_priv *priv,
|
|
|
|
enum ieee80211_band band,
|
|
|
|
u16 channel, u8 extension_chan_offset)
|
2008-05-05 10:22:41 +08:00
|
|
|
{
|
|
|
|
const struct iwl_channel_info *ch_info;
|
|
|
|
|
|
|
|
ch_info = iwl_get_channel_info(priv, band, channel);
|
|
|
|
if (!is_channel_valid(ch_info))
|
2010-08-23 16:46:46 +08:00
|
|
|
return false;
|
2008-05-05 10:22:41 +08:00
|
|
|
|
2008-10-09 18:13:49 +08:00
|
|
|
if (extension_chan_offset == IEEE80211_HT_PARAM_CHA_SEC_ABOVE)
|
2009-08-08 06:41:38 +08:00
|
|
|
return !(ch_info->ht40_extension_channel &
|
2009-05-02 12:37:18 +08:00
|
|
|
IEEE80211_CHAN_NO_HT40PLUS);
|
2008-10-09 18:13:49 +08:00
|
|
|
else if (extension_chan_offset == IEEE80211_HT_PARAM_CHA_SEC_BELOW)
|
2009-08-08 06:41:38 +08:00
|
|
|
return !(ch_info->ht40_extension_channel &
|
2009-05-02 12:37:18 +08:00
|
|
|
IEEE80211_CHAN_NO_HT40MINUS);
|
2008-05-05 10:22:41 +08:00
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
return false;
|
2008-05-05 10:22:41 +08:00
|
|
|
}
|
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
bool iwl_is_ht40_tx_allowed(struct iwl_priv *priv,
|
|
|
|
struct iwl_rxon_context *ctx,
|
|
|
|
struct ieee80211_sta_ht_cap *ht_cap)
|
2008-05-05 10:22:41 +08:00
|
|
|
{
|
2010-08-23 16:46:46 +08:00
|
|
|
if (!ctx->ht.enabled || !ctx->ht.is_40mhz)
|
|
|
|
return false;
|
2008-05-05 10:22:41 +08:00
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
/*
|
|
|
|
* We do not check for IEEE80211_HT_CAP_SUP_WIDTH_20_40
|
2009-05-23 02:01:49 +08:00
|
|
|
* the bit will not set if it is pure 40MHz case
|
|
|
|
*/
|
2010-08-23 16:46:46 +08:00
|
|
|
if (ht_cap && !ht_cap->ht_supported)
|
|
|
|
return false;
|
|
|
|
|
2010-05-07 03:18:41 +08:00
|
|
|
#ifdef CONFIG_IWLWIFI_DEBUGFS
|
2009-07-28 04:50:15 +08:00
|
|
|
if (priv->disable_ht40)
|
2010-08-23 16:46:46 +08:00
|
|
|
return false;
|
2009-07-28 04:50:15 +08:00
|
|
|
#endif
|
2010-08-23 16:46:46 +08:00
|
|
|
|
2009-06-13 04:22:51 +08:00
|
|
|
return iwl_is_channel_extension(priv, priv->band,
|
2010-08-23 16:46:32 +08:00
|
|
|
le16_to_cpu(ctx->staging.channel),
|
2010-08-23 16:46:46 +08:00
|
|
|
ctx->ht.extension_chan_offset);
|
2008-05-05 10:22:41 +08:00
|
|
|
}
|
2009-08-08 06:41:38 +08:00
|
|
|
EXPORT_SYMBOL(iwl_is_ht40_tx_allowed);
|
2008-05-05 10:22:41 +08:00
|
|
|
|
2009-06-20 04:52:45 +08:00
|
|
|
static u16 iwl_adjust_beacon_interval(u16 beacon_val, u16 max_beacon_val)
|
|
|
|
{
|
2010-09-03 21:30:55 +08:00
|
|
|
u16 new_val;
|
|
|
|
u16 beacon_factor;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If mac80211 hasn't given us a beacon interval, program
|
|
|
|
* the default into the device (not checking this here
|
|
|
|
* would cause the adjustment below to return the maximum
|
|
|
|
* value, which may break PAN.)
|
|
|
|
*/
|
|
|
|
if (!beacon_val)
|
|
|
|
return DEFAULT_BEACON_INTERVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the beacon interval we obtained from the peer
|
|
|
|
* is too large, we'll have to wake up more often
|
|
|
|
* (and in IBSS case, we'll beacon too much)
|
|
|
|
*
|
|
|
|
* For example, if max_beacon_val is 4096, and the
|
|
|
|
* requested beacon interval is 7000, we'll have to
|
|
|
|
* use 3500 to be able to wake up on the beacons.
|
|
|
|
*
|
|
|
|
* This could badly influence beacon detection stats.
|
|
|
|
*/
|
2009-06-20 04:52:45 +08:00
|
|
|
|
|
|
|
beacon_factor = (beacon_val + max_beacon_val) / max_beacon_val;
|
|
|
|
new_val = beacon_val / beacon_factor;
|
|
|
|
|
|
|
|
if (!new_val)
|
|
|
|
new_val = max_beacon_val;
|
|
|
|
|
|
|
|
return new_val;
|
|
|
|
}
|
|
|
|
|
2010-08-23 16:46:55 +08:00
|
|
|
int iwl_send_rxon_timing(struct iwl_priv *priv, struct iwl_rxon_context *ctx)
|
2009-06-20 04:52:45 +08:00
|
|
|
{
|
|
|
|
u64 tsf;
|
|
|
|
s32 interval_tm, rem;
|
|
|
|
struct ieee80211_conf *conf = NULL;
|
|
|
|
u16 beacon_int;
|
2010-08-23 16:46:55 +08:00
|
|
|
struct ieee80211_vif *vif = ctx->vif;
|
2009-06-20 04:52:45 +08:00
|
|
|
|
|
|
|
conf = ieee80211_get_hw_conf(priv->hw);
|
|
|
|
|
2010-07-29 22:07:51 +08:00
|
|
|
lockdep_assert_held(&priv->mutex);
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
memset(&ctx->timing, 0, sizeof(struct iwl_rxon_time_cmd));
|
2010-07-29 22:07:51 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->timing.timestamp = cpu_to_le64(priv->timestamp);
|
|
|
|
ctx->timing.listen_interval = cpu_to_le16(conf->listen_interval);
|
2009-06-20 04:52:45 +08:00
|
|
|
|
2010-08-23 16:46:55 +08:00
|
|
|
beacon_int = vif ? vif->bss_conf.beacon_int : 0;
|
2009-06-20 04:52:45 +08:00
|
|
|
|
2010-08-23 16:46:55 +08:00
|
|
|
/*
|
|
|
|
* TODO: For IBSS we need to get atim_window from mac80211,
|
|
|
|
* for now just always use 0
|
|
|
|
*/
|
|
|
|
ctx->timing.atim_window = 0;
|
2009-06-20 04:52:45 +08:00
|
|
|
|
2010-08-23 16:46:57 +08:00
|
|
|
if (ctx->ctxid == IWL_RXON_CTX_PAN &&
|
2010-09-03 21:31:26 +08:00
|
|
|
(!ctx->vif || ctx->vif->type != NL80211_IFTYPE_STATION) &&
|
|
|
|
iwl_is_associated(priv, IWL_RXON_CTX_BSS) &&
|
|
|
|
priv->contexts[IWL_RXON_CTX_BSS].vif &&
|
|
|
|
priv->contexts[IWL_RXON_CTX_BSS].vif->bss_conf.beacon_int) {
|
2010-08-23 16:46:57 +08:00
|
|
|
ctx->timing.beacon_interval =
|
|
|
|
priv->contexts[IWL_RXON_CTX_BSS].timing.beacon_interval;
|
|
|
|
beacon_int = le16_to_cpu(ctx->timing.beacon_interval);
|
2010-09-03 21:31:26 +08:00
|
|
|
} else if (ctx->ctxid == IWL_RXON_CTX_BSS &&
|
|
|
|
iwl_is_associated(priv, IWL_RXON_CTX_PAN) &&
|
|
|
|
priv->contexts[IWL_RXON_CTX_PAN].vif &&
|
|
|
|
priv->contexts[IWL_RXON_CTX_PAN].vif->bss_conf.beacon_int &&
|
|
|
|
(!iwl_is_associated_ctx(ctx) || !ctx->vif ||
|
|
|
|
!ctx->vif->bss_conf.beacon_int)) {
|
|
|
|
ctx->timing.beacon_interval =
|
|
|
|
priv->contexts[IWL_RXON_CTX_PAN].timing.beacon_interval;
|
|
|
|
beacon_int = le16_to_cpu(ctx->timing.beacon_interval);
|
2010-08-23 16:46:57 +08:00
|
|
|
} else {
|
|
|
|
beacon_int = iwl_adjust_beacon_interval(beacon_int,
|
2010-05-06 02:31:38 +08:00
|
|
|
priv->hw_params.max_beacon_itrvl * TIME_UNIT);
|
2010-08-23 16:46:57 +08:00
|
|
|
ctx->timing.beacon_interval = cpu_to_le16(beacon_int);
|
|
|
|
}
|
2009-06-20 04:52:45 +08:00
|
|
|
|
|
|
|
tsf = priv->timestamp; /* tsf is modifed by do_div: copy it */
|
2010-05-06 02:31:38 +08:00
|
|
|
interval_tm = beacon_int * TIME_UNIT;
|
2009-06-20 04:52:45 +08:00
|
|
|
rem = do_div(tsf, interval_tm);
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->timing.beacon_init_val = cpu_to_le32(interval_tm - rem);
|
2009-06-20 04:52:45 +08:00
|
|
|
|
2010-08-23 16:46:55 +08:00
|
|
|
ctx->timing.dtim_period = vif ? (vif->bss_conf.dtim_period ?: 1) : 1;
|
2010-08-23 16:46:52 +08:00
|
|
|
|
2009-06-20 04:52:45 +08:00
|
|
|
IWL_DEBUG_ASSOC(priv,
|
|
|
|
"beacon interval %d beacon timer %d beacon tim %d\n",
|
2010-08-23 16:46:32 +08:00
|
|
|
le16_to_cpu(ctx->timing.beacon_interval),
|
|
|
|
le32_to_cpu(ctx->timing.beacon_init_val),
|
|
|
|
le16_to_cpu(ctx->timing.atim_window));
|
2010-07-29 22:07:51 +08:00
|
|
|
|
2010-08-23 16:46:37 +08:00
|
|
|
return iwl_send_cmd_pdu(priv, ctx->rxon_timing_cmd,
|
2010-08-23 16:46:32 +08:00
|
|
|
sizeof(ctx->timing), &ctx->timing);
|
2009-06-20 04:52:45 +08:00
|
|
|
}
|
2010-07-29 22:07:51 +08:00
|
|
|
EXPORT_SYMBOL(iwl_send_rxon_timing);
|
2009-06-20 04:52:45 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
void iwl_set_rxon_hwcrypto(struct iwl_priv *priv, struct iwl_rxon_context *ctx,
|
|
|
|
int hw_decrypt)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_cmd *rxon = &ctx->staging;
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
if (hw_decrypt)
|
|
|
|
rxon->filter_flags &= ~RXON_FILTER_DIS_DECRYPT_MSK;
|
|
|
|
else
|
|
|
|
rxon->filter_flags |= RXON_FILTER_DIS_DECRYPT_MSK;
|
|
|
|
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_set_rxon_hwcrypto);
|
|
|
|
|
2010-10-07 19:02:03 +08:00
|
|
|
/* validate RXON structure is valid */
|
2010-08-23 16:46:32 +08:00
|
|
|
int iwl_check_rxon_cmd(struct iwl_priv *priv, struct iwl_rxon_context *ctx)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_cmd *rxon = &ctx->staging;
|
2010-10-07 19:02:03 +08:00
|
|
|
bool error = false;
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
if (rxon->flags & RXON_FLG_BAND_24G_MSK) {
|
2010-10-07 19:02:03 +08:00
|
|
|
if (rxon->flags & RXON_FLG_TGJ_NARROW_BAND_MSK) {
|
|
|
|
IWL_WARN(priv, "check 2.4G: wrong narrow\n");
|
|
|
|
error = true;
|
|
|
|
}
|
|
|
|
if (rxon->flags & RXON_FLG_RADAR_DETECT_MSK) {
|
|
|
|
IWL_WARN(priv, "check 2.4G: wrong radar\n");
|
|
|
|
error = true;
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
} else {
|
2010-10-07 19:02:03 +08:00
|
|
|
if (!(rxon->flags & RXON_FLG_SHORT_SLOT_MSK)) {
|
|
|
|
IWL_WARN(priv, "check 5.2G: not short slot!\n");
|
|
|
|
error = true;
|
|
|
|
}
|
|
|
|
if (rxon->flags & RXON_FLG_CCK_MSK) {
|
|
|
|
IWL_WARN(priv, "check 5.2G: CCK!\n");
|
|
|
|
error = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((rxon->node_addr[0] | rxon->bssid_addr[0]) & 0x1) {
|
|
|
|
IWL_WARN(priv, "mac/bssid mcast!\n");
|
|
|
|
error = true;
|
2009-01-28 06:27:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* make sure basic rates 6Mbps and 1Mbps are supported */
|
2010-10-07 19:02:03 +08:00
|
|
|
if ((rxon->ofdm_basic_rates & IWL_RATE_6M_MASK) == 0 &&
|
|
|
|
(rxon->cck_basic_rates & IWL_RATE_1M_MASK) == 0) {
|
|
|
|
IWL_WARN(priv, "neither 1 nor 6 are basic\n");
|
|
|
|
error = true;
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-10-07 19:02:03 +08:00
|
|
|
if (le16_to_cpu(rxon->assoc_id) > 2007) {
|
|
|
|
IWL_WARN(priv, "aid > 2007\n");
|
|
|
|
error = true;
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-10-07 19:02:03 +08:00
|
|
|
if ((rxon->flags & (RXON_FLG_CCK_MSK | RXON_FLG_SHORT_SLOT_MSK))
|
|
|
|
== (RXON_FLG_CCK_MSK | RXON_FLG_SHORT_SLOT_MSK)) {
|
|
|
|
IWL_WARN(priv, "CCK and short slot\n");
|
|
|
|
error = true;
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-10-07 19:02:03 +08:00
|
|
|
if ((rxon->flags & (RXON_FLG_CCK_MSK | RXON_FLG_AUTO_DETECT_MSK))
|
|
|
|
== (RXON_FLG_CCK_MSK | RXON_FLG_AUTO_DETECT_MSK)) {
|
|
|
|
IWL_WARN(priv, "CCK and auto detect");
|
|
|
|
error = true;
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-10-07 19:02:03 +08:00
|
|
|
if ((rxon->flags & (RXON_FLG_AUTO_DETECT_MSK |
|
|
|
|
RXON_FLG_TGG_PROTECT_MSK)) ==
|
|
|
|
RXON_FLG_TGG_PROTECT_MSK) {
|
|
|
|
IWL_WARN(priv, "TGg but no auto-detect\n");
|
|
|
|
error = true;
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
if (error)
|
|
|
|
IWL_WARN(priv, "Tuning to channel %d\n",
|
|
|
|
le16_to_cpu(rxon->channel));
|
|
|
|
|
|
|
|
if (error) {
|
2010-10-07 19:02:03 +08:00
|
|
|
IWL_ERR(priv, "Invalid RXON\n");
|
|
|
|
return -EINVAL;
|
2009-01-28 06:27:52 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_check_rxon_cmd);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* iwl_full_rxon_required - check if full RXON (vs RXON_ASSOC) cmd is needed
|
|
|
|
* @priv: staging_rxon is compared to active_rxon
|
|
|
|
*
|
|
|
|
* If the RXON structure is changing enough to require a new tune,
|
|
|
|
* or is clearing the RXON_FILTER_ASSOC_MSK, then return 1 to indicate that
|
|
|
|
* a new tune (full RXON command, rather than RXON_ASSOC cmd) is required.
|
|
|
|
*/
|
2010-08-23 16:46:32 +08:00
|
|
|
int iwl_full_rxon_required(struct iwl_priv *priv,
|
|
|
|
struct iwl_rxon_context *ctx)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
2010-08-23 16:46:32 +08:00
|
|
|
const struct iwl_rxon_cmd *staging = &ctx->staging;
|
|
|
|
const struct iwl_rxon_cmd *active = &ctx->active;
|
|
|
|
|
|
|
|
#define CHK(cond) \
|
|
|
|
if ((cond)) { \
|
|
|
|
IWL_DEBUG_INFO(priv, "need full RXON - " #cond "\n"); \
|
|
|
|
return 1; \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define CHK_NEQ(c1, c2) \
|
|
|
|
if ((c1) != (c2)) { \
|
|
|
|
IWL_DEBUG_INFO(priv, "need full RXON - " \
|
|
|
|
#c1 " != " #c2 " - %d != %d\n", \
|
|
|
|
(c1), (c2)); \
|
|
|
|
return 1; \
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
/* These items are only settable from the full RXON command */
|
2010-08-23 16:46:32 +08:00
|
|
|
CHK(!iwl_is_associated_ctx(ctx));
|
|
|
|
CHK(compare_ether_addr(staging->bssid_addr, active->bssid_addr));
|
|
|
|
CHK(compare_ether_addr(staging->node_addr, active->node_addr));
|
|
|
|
CHK(compare_ether_addr(staging->wlap_bssid_addr,
|
|
|
|
active->wlap_bssid_addr));
|
|
|
|
CHK_NEQ(staging->dev_type, active->dev_type);
|
|
|
|
CHK_NEQ(staging->channel, active->channel);
|
|
|
|
CHK_NEQ(staging->air_propagation, active->air_propagation);
|
|
|
|
CHK_NEQ(staging->ofdm_ht_single_stream_basic_rates,
|
|
|
|
active->ofdm_ht_single_stream_basic_rates);
|
|
|
|
CHK_NEQ(staging->ofdm_ht_dual_stream_basic_rates,
|
|
|
|
active->ofdm_ht_dual_stream_basic_rates);
|
|
|
|
CHK_NEQ(staging->ofdm_ht_triple_stream_basic_rates,
|
|
|
|
active->ofdm_ht_triple_stream_basic_rates);
|
|
|
|
CHK_NEQ(staging->assoc_id, active->assoc_id);
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
/* flags, filter_flags, ofdm_basic_rates, and cck_basic_rates can
|
|
|
|
* be updated with the RXON_ASSOC command -- however only some
|
|
|
|
* flag transitions are allowed using RXON_ASSOC */
|
|
|
|
|
|
|
|
/* Check if we are not switching bands */
|
2010-08-23 16:46:32 +08:00
|
|
|
CHK_NEQ(staging->flags & RXON_FLG_BAND_24G_MSK,
|
|
|
|
active->flags & RXON_FLG_BAND_24G_MSK);
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
/* Check if we are switching association toggle */
|
2010-08-23 16:46:32 +08:00
|
|
|
CHK_NEQ(staging->filter_flags & RXON_FILTER_ASSOC_MSK,
|
|
|
|
active->filter_flags & RXON_FILTER_ASSOC_MSK);
|
|
|
|
|
|
|
|
#undef CHK
|
|
|
|
#undef CHK_NEQ
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_full_rxon_required);
|
|
|
|
|
2010-08-23 16:46:47 +08:00
|
|
|
u8 iwl_rate_get_lowest_plcp(struct iwl_priv *priv,
|
|
|
|
struct iwl_rxon_context *ctx)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
2010-01-22 03:33:19 +08:00
|
|
|
/*
|
|
|
|
* Assign the lowest rate -- should really get this from
|
|
|
|
* the beacon skb from mac80211.
|
|
|
|
*/
|
2010-08-23 16:46:32 +08:00
|
|
|
if (ctx->staging.flags & RXON_FLG_BAND_24G_MSK)
|
2009-01-28 06:27:52 +08:00
|
|
|
return IWL_RATE_1M_PLCP;
|
|
|
|
else
|
|
|
|
return IWL_RATE_6M_PLCP;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_rate_get_lowest_plcp);
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
static void _iwl_set_rxon_ht(struct iwl_priv *priv,
|
|
|
|
struct iwl_ht_config *ht_conf,
|
|
|
|
struct iwl_rxon_context *ctx)
|
2008-05-05 10:22:41 +08:00
|
|
|
{
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_cmd *rxon = &ctx->staging;
|
2008-05-05 10:22:41 +08:00
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
if (!ctx->ht.enabled) {
|
2009-05-23 02:01:49 +08:00
|
|
|
rxon->flags &= ~(RXON_FLG_CHANNEL_MODE_MSK |
|
2008-09-17 10:10:05 +08:00
|
|
|
RXON_FLG_CTRL_CHANNEL_LOC_HI_MSK |
|
2009-08-08 06:41:38 +08:00
|
|
|
RXON_FLG_HT40_PROT_MSK |
|
2008-09-17 10:10:05 +08:00
|
|
|
RXON_FLG_HT_PROT_MSK);
|
2008-05-05 10:22:41 +08:00
|
|
|
return;
|
2008-09-17 10:10:05 +08:00
|
|
|
}
|
2008-05-05 10:22:41 +08:00
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
/* FIXME: if the definition of ht.protection changed, the "translation"
|
2009-05-23 02:01:49 +08:00
|
|
|
* will be needed for rxon->flags
|
|
|
|
*/
|
2010-08-23 16:46:46 +08:00
|
|
|
rxon->flags |= cpu_to_le32(ctx->ht.protection << RXON_FLG_HT_OPERATING_MODE_POS);
|
2009-05-23 02:01:49 +08:00
|
|
|
|
|
|
|
/* Set up channel bandwidth:
|
2009-08-08 06:41:38 +08:00
|
|
|
* 20 MHz only, 20/40 mixed or pure 40 if ht40 ok */
|
2009-05-23 02:01:49 +08:00
|
|
|
/* clear the HT channel mode before set the mode */
|
|
|
|
rxon->flags &= ~(RXON_FLG_CHANNEL_MODE_MSK |
|
|
|
|
RXON_FLG_CTRL_CHANNEL_LOC_HI_MSK);
|
2010-08-23 16:46:46 +08:00
|
|
|
if (iwl_is_ht40_tx_allowed(priv, ctx, NULL)) {
|
2009-08-08 06:41:38 +08:00
|
|
|
/* pure ht40 */
|
2010-08-23 16:46:46 +08:00
|
|
|
if (ctx->ht.protection == IEEE80211_HT_OP_MODE_PROTECTION_20MHZ) {
|
2009-05-23 02:01:49 +08:00
|
|
|
rxon->flags |= RXON_FLG_CHANNEL_MODE_PURE_40;
|
2009-06-13 04:22:50 +08:00
|
|
|
/* Note: control channel is opposite of extension channel */
|
2010-08-23 16:46:46 +08:00
|
|
|
switch (ctx->ht.extension_chan_offset) {
|
2009-06-13 04:22:50 +08:00
|
|
|
case IEEE80211_HT_PARAM_CHA_SEC_ABOVE:
|
|
|
|
rxon->flags &= ~RXON_FLG_CTRL_CHANNEL_LOC_HI_MSK;
|
|
|
|
break;
|
|
|
|
case IEEE80211_HT_PARAM_CHA_SEC_BELOW:
|
|
|
|
rxon->flags |= RXON_FLG_CTRL_CHANNEL_LOC_HI_MSK;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
2009-05-23 02:01:49 +08:00
|
|
|
/* Note: control channel is opposite of extension channel */
|
2010-08-23 16:46:46 +08:00
|
|
|
switch (ctx->ht.extension_chan_offset) {
|
2009-05-23 02:01:49 +08:00
|
|
|
case IEEE80211_HT_PARAM_CHA_SEC_ABOVE:
|
|
|
|
rxon->flags &= ~(RXON_FLG_CTRL_CHANNEL_LOC_HI_MSK);
|
|
|
|
rxon->flags |= RXON_FLG_CHANNEL_MODE_MIXED;
|
|
|
|
break;
|
|
|
|
case IEEE80211_HT_PARAM_CHA_SEC_BELOW:
|
|
|
|
rxon->flags |= RXON_FLG_CTRL_CHANNEL_LOC_HI_MSK;
|
|
|
|
rxon->flags |= RXON_FLG_CHANNEL_MODE_MIXED;
|
|
|
|
break;
|
|
|
|
case IEEE80211_HT_PARAM_CHA_SEC_NONE:
|
|
|
|
default:
|
|
|
|
/* channel location only valid if in Mixed mode */
|
|
|
|
IWL_ERR(priv, "invalid extension channel offset\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
rxon->flags |= RXON_FLG_CHANNEL_MODE_LEGACY;
|
2008-05-05 10:22:41 +08:00
|
|
|
}
|
|
|
|
|
2009-04-09 02:26:44 +08:00
|
|
|
if (priv->cfg->ops->hcmd->set_rxon_chain)
|
2010-08-23 16:46:32 +08:00
|
|
|
priv->cfg->ops->hcmd->set_rxon_chain(priv, ctx);
|
2008-05-05 10:22:41 +08:00
|
|
|
|
iwlwifi: clarify and clean up chain settings
The chain settings we currently use in iwlwifi are
rather confusing -- and we also go by the wrong
settings entirely under certain circumstances. To
clean it up, create a new variable in the current
HT config -- single_chain_sufficient -- that tells
us whether we need more than one chain. Calculate
that based on the AP and operating mode (no IBSS
HT implemented -- so no need for multiple chains,
for station mode we use the AP's capabilities).
Additionally, since APs always send disabled SM PS
mode, keeping track of their sm_ps mode isn't very
useful -- doubly not so for our _own_ RX config
since that should depend on our, not the AP's, SM
PS mode.
Finally, document that our configuration of the
number of RX chains used is currently wrong when
in powersave (by adding a comment).
All together this removes the two remaining items
in struct iwl_ht_config that were done wrong there.
For the future, the number of RX chains and some
SM PS handshaking needs to be added to mac80211,
which then needs to tell us, and the new variable
current_ht_config.single_chain_sufficient should
also be calculated by mac80211.
Signed-off-by: Johannes Berg <johannes@sipsolutions.net>
Acked-by: Daniel C Halperin <daniel.c.halperin@intel.com>
Signed-off-by: Reinette Chatre <reinette.chatre@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2009-09-12 01:38:17 +08:00
|
|
|
IWL_DEBUG_ASSOC(priv, "rxon flags 0x%X operation mode :0x%X "
|
2008-10-14 22:58:37 +08:00
|
|
|
"extension channel offset 0x%x\n",
|
2010-08-23 16:46:46 +08:00
|
|
|
le32_to_cpu(rxon->flags), ctx->ht.protection,
|
|
|
|
ctx->ht.extension_chan_offset);
|
2008-05-05 10:22:41 +08:00
|
|
|
}
|
2010-08-23 16:46:32 +08:00
|
|
|
|
|
|
|
void iwl_set_rxon_ht(struct iwl_priv *priv, struct iwl_ht_config *ht_conf)
|
|
|
|
{
|
|
|
|
struct iwl_rxon_context *ctx;
|
|
|
|
|
|
|
|
for_each_context(priv, ctx)
|
|
|
|
_iwl_set_rxon_ht(priv, ht_conf, ctx);
|
|
|
|
}
|
2008-05-05 10:22:41 +08:00
|
|
|
EXPORT_SYMBOL(iwl_set_rxon_ht);
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
/* Return valid, unused, channel for a passive scan to reset the RF */
|
2010-06-03 12:15:10 +08:00
|
|
|
u8 iwl_get_single_channel_number(struct iwl_priv *priv,
|
2010-08-23 16:46:32 +08:00
|
|
|
enum ieee80211_band band)
|
2010-06-03 12:15:10 +08:00
|
|
|
{
|
|
|
|
const struct iwl_channel_info *ch_info;
|
|
|
|
int i;
|
|
|
|
u8 channel = 0;
|
2010-08-23 16:46:32 +08:00
|
|
|
u8 min, max;
|
|
|
|
struct iwl_rxon_context *ctx;
|
2010-06-03 12:15:10 +08:00
|
|
|
|
|
|
|
if (band == IEEE80211_BAND_5GHZ) {
|
2010-08-23 16:46:32 +08:00
|
|
|
min = 14;
|
|
|
|
max = priv->channel_count;
|
2010-06-03 12:15:10 +08:00
|
|
|
} else {
|
2010-08-23 16:46:32 +08:00
|
|
|
min = 0;
|
|
|
|
max = 14;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = min; i < max; i++) {
|
|
|
|
bool busy = false;
|
|
|
|
|
|
|
|
for_each_context(priv, ctx) {
|
|
|
|
busy = priv->channel_info[i].channel ==
|
|
|
|
le16_to_cpu(ctx->staging.channel);
|
|
|
|
if (busy)
|
|
|
|
break;
|
2010-06-03 12:15:10 +08:00
|
|
|
}
|
2010-08-23 16:46:32 +08:00
|
|
|
|
|
|
|
if (busy)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
channel = priv->channel_info[i].channel;
|
|
|
|
ch_info = iwl_get_channel_info(priv, band, channel);
|
|
|
|
if (is_channel_valid(ch_info))
|
|
|
|
break;
|
2010-06-03 12:15:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return channel;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_get_single_channel_number);
|
|
|
|
|
2008-03-15 01:38:49 +08:00
|
|
|
/**
|
2010-07-28 11:45:15 +08:00
|
|
|
* iwl_set_rxon_channel - Set the band and channel values in staging RXON
|
|
|
|
* @ch: requested channel as a pointer to struct ieee80211_channel
|
2008-03-15 01:38:49 +08:00
|
|
|
|
|
|
|
* NOTE: Does not commit to the hardware; it sets appropriate bit fields
|
2010-07-28 11:45:15 +08:00
|
|
|
* in the staging RXON flag structure based on the ch->band
|
2008-03-15 01:38:49 +08:00
|
|
|
*/
|
2010-08-23 16:46:32 +08:00
|
|
|
int iwl_set_rxon_channel(struct iwl_priv *priv, struct ieee80211_channel *ch,
|
|
|
|
struct iwl_rxon_context *ctx)
|
2008-03-15 01:38:49 +08:00
|
|
|
{
|
2008-09-03 11:26:26 +08:00
|
|
|
enum ieee80211_band band = ch->band;
|
2010-07-29 04:40:27 +08:00
|
|
|
u16 channel = ch->hw_value;
|
2008-09-03 11:26:26 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
if ((le16_to_cpu(ctx->staging.channel) == channel) &&
|
2008-03-15 01:38:49 +08:00
|
|
|
(priv->band == band))
|
|
|
|
return 0;
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.channel = cpu_to_le16(channel);
|
2008-03-15 01:38:49 +08:00
|
|
|
if (band == IEEE80211_BAND_5GHZ)
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &= ~RXON_FLG_BAND_24G_MSK;
|
2008-03-15 01:38:49 +08:00
|
|
|
else
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_BAND_24G_MSK;
|
2008-03-15 01:38:49 +08:00
|
|
|
|
|
|
|
priv->band = band;
|
|
|
|
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG_INFO(priv, "Staging channel set to %d [%d]\n", channel, band);
|
2008-03-15 01:38:49 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2008-04-24 08:15:05 +08:00
|
|
|
EXPORT_SYMBOL(iwl_set_rxon_channel);
|
2008-03-15 01:38:49 +08:00
|
|
|
|
2010-05-06 23:54:11 +08:00
|
|
|
void iwl_set_flags_for_band(struct iwl_priv *priv,
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx,
|
2010-05-06 23:54:11 +08:00
|
|
|
enum ieee80211_band band,
|
|
|
|
struct ieee80211_vif *vif)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
|
|
|
if (band == IEEE80211_BAND_5GHZ) {
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &=
|
2009-01-28 06:27:52 +08:00
|
|
|
~(RXON_FLG_BAND_24G_MSK | RXON_FLG_AUTO_DETECT_MSK
|
|
|
|
| RXON_FLG_CCK_MSK);
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_SHORT_SLOT_MSK;
|
2009-01-28 06:27:52 +08:00
|
|
|
} else {
|
|
|
|
/* Copied from iwl_post_associate() */
|
2010-05-07 03:21:40 +08:00
|
|
|
if (vif && vif->bss_conf.use_short_slot)
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_SHORT_SLOT_MSK;
|
2009-01-28 06:27:52 +08:00
|
|
|
else
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &= ~RXON_FLG_SHORT_SLOT_MSK;
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_BAND_24G_MSK;
|
|
|
|
ctx->staging.flags |= RXON_FLG_AUTO_DETECT_MSK;
|
|
|
|
ctx->staging.flags &= ~RXON_FLG_CCK_MSK;
|
2009-01-28 06:27:52 +08:00
|
|
|
}
|
|
|
|
}
|
2010-05-06 23:54:11 +08:00
|
|
|
EXPORT_SYMBOL(iwl_set_flags_for_band);
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* initialize rxon structure with default values from eeprom
|
|
|
|
*/
|
2010-04-29 19:43:06 +08:00
|
|
|
void iwl_connection_init_rx_config(struct iwl_priv *priv,
|
2010-08-23 16:46:58 +08:00
|
|
|
struct iwl_rxon_context *ctx)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
|
|
|
const struct iwl_channel_info *ch_info;
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
memset(&ctx->staging, 0, sizeof(ctx->staging));
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-08-23 16:46:58 +08:00
|
|
|
if (!ctx->vif) {
|
|
|
|
ctx->staging.dev_type = ctx->unused_devtype;
|
|
|
|
} else switch (ctx->vif->type) {
|
2009-01-28 06:27:52 +08:00
|
|
|
case NL80211_IFTYPE_AP:
|
2010-08-23 16:46:58 +08:00
|
|
|
ctx->staging.dev_type = ctx->ap_devtype;
|
2009-01-28 06:27:52 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_IFTYPE_STATION:
|
2010-08-23 16:46:58 +08:00
|
|
|
ctx->staging.dev_type = ctx->station_devtype;
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.filter_flags = RXON_FILTER_ACCEPT_GRP_MSK;
|
2009-01-28 06:27:52 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_IFTYPE_ADHOC:
|
2010-08-23 16:46:58 +08:00
|
|
|
ctx->staging.dev_type = ctx->ibss_devtype;
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags = RXON_FLG_SHORT_PREAMBLE_MSK;
|
|
|
|
ctx->staging.filter_flags = RXON_FILTER_BCON_AWARE_MSK |
|
2009-01-28 06:27:52 +08:00
|
|
|
RXON_FILTER_ACCEPT_GRP_MSK;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-08-23 16:46:58 +08:00
|
|
|
IWL_ERR(priv, "Unsupported interface type %d\n",
|
|
|
|
ctx->vif->type);
|
2009-01-28 06:27:52 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* TODO: Figure out when short_preamble would be set and cache from
|
|
|
|
* that */
|
|
|
|
if (!hw_to_local(priv->hw)->short_preamble)
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &= ~RXON_FLG_SHORT_PREAMBLE_MSK;
|
2009-01-28 06:27:52 +08:00
|
|
|
else
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_SHORT_PREAMBLE_MSK;
|
2009-01-28 06:27:52 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
ch_info = iwl_get_channel_info(priv, priv->band,
|
2010-08-23 16:46:32 +08:00
|
|
|
le16_to_cpu(ctx->active.channel));
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
if (!ch_info)
|
|
|
|
ch_info = &priv->channel_info[0];
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.channel = cpu_to_le16(ch_info->channel);
|
2009-01-28 06:27:52 +08:00
|
|
|
priv->band = ch_info->band;
|
|
|
|
|
2010-08-23 16:46:58 +08:00
|
|
|
iwl_set_flags_for_band(priv, ctx, priv->band, ctx->vif);
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.ofdm_basic_rates =
|
2009-01-28 06:27:52 +08:00
|
|
|
(IWL_OFDM_RATES_MASK >> IWL_FIRST_OFDM_RATE) & 0xFF;
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.cck_basic_rates =
|
2009-01-28 06:27:52 +08:00
|
|
|
(IWL_CCK_RATES_MASK >> IWL_FIRST_CCK_RATE) & 0xF;
|
|
|
|
|
2009-05-23 02:01:49 +08:00
|
|
|
/* clear both MIX and PURE40 mode flag */
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &= ~(RXON_FLG_CHANNEL_MODE_MIXED |
|
2009-05-23 02:01:49 +08:00
|
|
|
RXON_FLG_CHANNEL_MODE_PURE_40);
|
2010-08-23 16:46:58 +08:00
|
|
|
if (ctx->vif)
|
|
|
|
memcpy(ctx->staging.node_addr, ctx->vif->addr, ETH_ALEN);
|
2010-05-12 18:33:11 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.ofdm_ht_single_stream_basic_rates = 0xff;
|
|
|
|
ctx->staging.ofdm_ht_dual_stream_basic_rates = 0xff;
|
|
|
|
ctx->staging.ofdm_ht_triple_stream_basic_rates = 0xff;
|
2009-01-28 06:27:52 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_connection_init_rx_config);
|
|
|
|
|
2010-05-06 23:54:11 +08:00
|
|
|
void iwl_set_rate(struct iwl_priv *priv)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
|
|
|
const struct ieee80211_supported_band *hw = NULL;
|
|
|
|
struct ieee80211_rate *rate;
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx;
|
2009-01-28 06:27:52 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
hw = iwl_get_hw_mode(priv, priv->band);
|
|
|
|
if (!hw) {
|
|
|
|
IWL_ERR(priv, "Failed to set rate: unable to get hw mode\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
priv->active_rate = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < hw->n_bitrates; i++) {
|
|
|
|
rate = &(hw->bitrates[i]);
|
2009-08-29 00:44:45 +08:00
|
|
|
if (rate->hw_value < IWL_RATE_COUNT_LEGACY)
|
2009-01-28 06:27:52 +08:00
|
|
|
priv->active_rate |= (1 << rate->hw_value);
|
|
|
|
}
|
|
|
|
|
2010-01-22 03:33:19 +08:00
|
|
|
IWL_DEBUG_RATE(priv, "Set active_rate = %0x\n", priv->active_rate);
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
for_each_context(priv, ctx) {
|
|
|
|
ctx->staging.cck_basic_rates =
|
|
|
|
(IWL_CCK_BASIC_RATES_MASK >> IWL_FIRST_CCK_RATE) & 0xF;
|
2010-01-22 03:33:19 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.ofdm_basic_rates =
|
|
|
|
(IWL_OFDM_BASIC_RATES_MASK >> IWL_FIRST_OFDM_RATE) & 0xFF;
|
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
}
|
2010-05-06 23:54:11 +08:00
|
|
|
EXPORT_SYMBOL(iwl_set_rate);
|
|
|
|
|
|
|
|
void iwl_chswitch_done(struct iwl_priv *priv, bool is_success)
|
|
|
|
{
|
2010-08-23 16:46:40 +08:00
|
|
|
/*
|
|
|
|
* MULTI-FIXME
|
|
|
|
* See iwl_mac_channel_switch.
|
|
|
|
*/
|
|
|
|
struct iwl_rxon_context *ctx = &priv->contexts[IWL_RXON_CTX_BSS];
|
|
|
|
|
2010-05-06 23:54:11 +08:00
|
|
|
if (test_bit(STATUS_EXIT_PENDING, &priv->status))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (priv->switch_rxon.switch_in_progress) {
|
2010-08-23 16:46:40 +08:00
|
|
|
ieee80211_chswitch_done(ctx->vif, is_success);
|
2010-05-06 23:54:11 +08:00
|
|
|
mutex_lock(&priv->mutex);
|
|
|
|
priv->switch_rxon.switch_in_progress = false;
|
|
|
|
mutex_unlock(&priv->mutex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_chswitch_done);
|
2009-01-28 06:27:52 +08:00
|
|
|
|
|
|
|
void iwl_rx_csa(struct iwl_priv *priv, struct iwl_rx_mem_buffer *rxb)
|
|
|
|
{
|
2009-10-09 17:19:45 +08:00
|
|
|
struct iwl_rx_packet *pkt = rxb_addr(rxb);
|
2009-01-28 06:27:52 +08:00
|
|
|
struct iwl_csa_notification *csa = &(pkt->u.csa_notif);
|
2010-08-23 16:46:40 +08:00
|
|
|
/*
|
|
|
|
* MULTI-FIXME
|
|
|
|
* See iwl_mac_channel_switch.
|
|
|
|
*/
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx = &priv->contexts[IWL_RXON_CTX_BSS];
|
|
|
|
struct iwl_rxon_cmd *rxon = (void *)&ctx->active;
|
2009-10-24 04:42:29 +08:00
|
|
|
|
2009-11-07 06:52:54 +08:00
|
|
|
if (priv->switch_rxon.switch_in_progress) {
|
|
|
|
if (!le32_to_cpu(csa->status) &&
|
|
|
|
(csa->channel == priv->switch_rxon.channel)) {
|
|
|
|
rxon->channel = csa->channel;
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.channel = csa->channel;
|
2009-11-07 06:52:54 +08:00
|
|
|
IWL_DEBUG_11H(priv, "CSA notif: channel %d\n",
|
|
|
|
le16_to_cpu(csa->channel));
|
2010-05-06 23:54:11 +08:00
|
|
|
iwl_chswitch_done(priv, true);
|
|
|
|
} else {
|
2009-11-07 06:52:54 +08:00
|
|
|
IWL_ERR(priv, "CSA notif (fail) : channel %d\n",
|
|
|
|
le16_to_cpu(csa->channel));
|
2010-05-06 23:54:11 +08:00
|
|
|
iwl_chswitch_done(priv, false);
|
|
|
|
}
|
2009-11-07 06:52:54 +08:00
|
|
|
}
|
2009-01-28 06:27:52 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_rx_csa);
|
|
|
|
|
|
|
|
#ifdef CONFIG_IWLWIFI_DEBUG
|
2010-08-23 16:46:32 +08:00
|
|
|
void iwl_print_rx_config_cmd(struct iwl_priv *priv,
|
|
|
|
struct iwl_rxon_context *ctx)
|
2009-01-28 06:27:52 +08:00
|
|
|
{
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_cmd *rxon = &ctx->staging;
|
2009-01-28 06:27:52 +08:00
|
|
|
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG_RADIO(priv, "RX CONFIG:\n");
|
2009-08-08 06:41:37 +08:00
|
|
|
iwl_print_hex_dump(priv, IWL_DL_RADIO, (u8 *) rxon, sizeof(*rxon));
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG_RADIO(priv, "u16 channel: 0x%x\n", le16_to_cpu(rxon->channel));
|
|
|
|
IWL_DEBUG_RADIO(priv, "u32 flags: 0x%08X\n", le32_to_cpu(rxon->flags));
|
|
|
|
IWL_DEBUG_RADIO(priv, "u32 filter_flags: 0x%08x\n",
|
2009-01-28 06:27:52 +08:00
|
|
|
le32_to_cpu(rxon->filter_flags));
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG_RADIO(priv, "u8 dev_type: 0x%x\n", rxon->dev_type);
|
|
|
|
IWL_DEBUG_RADIO(priv, "u8 ofdm_basic_rates: 0x%02x\n",
|
2009-01-28 06:27:52 +08:00
|
|
|
rxon->ofdm_basic_rates);
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG_RADIO(priv, "u8 cck_basic_rates: 0x%02x\n", rxon->cck_basic_rates);
|
|
|
|
IWL_DEBUG_RADIO(priv, "u8[6] node_addr: %pM\n", rxon->node_addr);
|
|
|
|
IWL_DEBUG_RADIO(priv, "u8[6] bssid_addr: %pM\n", rxon->bssid_addr);
|
|
|
|
IWL_DEBUG_RADIO(priv, "u16 assoc_id: 0x%x\n", le16_to_cpu(rxon->assoc_id));
|
2009-01-28 06:27:52 +08:00
|
|
|
}
|
2009-11-07 06:52:46 +08:00
|
|
|
EXPORT_SYMBOL(iwl_print_rx_config_cmd);
|
2009-07-25 02:13:13 +08:00
|
|
|
#endif
|
2009-01-28 06:27:52 +08:00
|
|
|
/**
|
|
|
|
* iwl_irq_handle_error - called for HW or SW error interrupt from card
|
|
|
|
*/
|
|
|
|
void iwl_irq_handle_error(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
/* Set the FW error flag -- cleared on iwl_down */
|
|
|
|
set_bit(STATUS_FW_ERROR, &priv->status);
|
|
|
|
|
|
|
|
/* Cancel currently queued command. */
|
|
|
|
clear_bit(STATUS_HCMD_ACTIVE, &priv->status);
|
|
|
|
|
2010-04-28 13:05:15 +08:00
|
|
|
IWL_ERR(priv, "Loaded firmware version: %s\n",
|
|
|
|
priv->hw->wiphy->fw_version);
|
|
|
|
|
2009-11-21 04:05:05 +08:00
|
|
|
priv->cfg->ops->lib->dump_nic_error_log(priv);
|
2009-12-11 06:37:25 +08:00
|
|
|
if (priv->cfg->ops->lib->dump_csr)
|
|
|
|
priv->cfg->ops->lib->dump_csr(priv);
|
2010-01-16 05:43:39 +08:00
|
|
|
if (priv->cfg->ops->lib->dump_fh)
|
|
|
|
priv->cfg->ops->lib->dump_fh(priv, NULL, false);
|
2009-12-15 06:12:20 +08:00
|
|
|
priv->cfg->ops->lib->dump_nic_event_log(priv, false, NULL, false);
|
2009-01-28 06:27:52 +08:00
|
|
|
#ifdef CONFIG_IWLWIFI_DEBUG
|
2009-11-21 04:05:07 +08:00
|
|
|
if (iwl_get_debug_level(priv) & IWL_DL_FW_ERRORS)
|
2010-08-23 16:46:32 +08:00
|
|
|
iwl_print_rx_config_cmd(priv,
|
|
|
|
&priv->contexts[IWL_RXON_CTX_BSS]);
|
2009-01-28 06:27:52 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
wake_up_interruptible(&priv->wait_command_queue);
|
|
|
|
|
|
|
|
/* Keep the restart process from trying to send host
|
|
|
|
* commands by clearing the INIT status bit */
|
|
|
|
clear_bit(STATUS_READY, &priv->status);
|
|
|
|
|
|
|
|
if (!test_bit(STATUS_EXIT_PENDING, &priv->status)) {
|
2009-01-28 06:27:56 +08:00
|
|
|
IWL_DEBUG(priv, IWL_DL_FW_ERRORS,
|
2009-01-28 06:27:52 +08:00
|
|
|
"Restarting adapter due to uCode error.\n");
|
|
|
|
|
|
|
|
if (priv->cfg->mod_params->restart_fw)
|
|
|
|
queue_work(priv->workqueue, &priv->restart);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_irq_handle_error);
|
|
|
|
|
2010-02-20 03:41:32 +08:00
|
|
|
static int iwl_apm_stop_master(struct iwl_priv *priv)
|
2009-10-03 04:44:04 +08:00
|
|
|
{
|
2009-10-31 05:36:05 +08:00
|
|
|
int ret = 0;
|
2009-10-03 04:44:04 +08:00
|
|
|
|
2009-10-31 05:36:05 +08:00
|
|
|
/* stop device's busmaster DMA activity */
|
2009-10-03 04:44:04 +08:00
|
|
|
iwl_set_bit(priv, CSR_RESET, CSR_RESET_REG_FLAG_STOP_MASTER);
|
|
|
|
|
2009-10-31 05:36:05 +08:00
|
|
|
ret = iwl_poll_bit(priv, CSR_RESET, CSR_RESET_REG_FLAG_MASTER_DISABLED,
|
2009-10-03 04:44:04 +08:00
|
|
|
CSR_RESET_REG_FLAG_MASTER_DISABLED, 100);
|
2009-10-31 05:36:05 +08:00
|
|
|
if (ret)
|
|
|
|
IWL_WARN(priv, "Master Disable Timed Out, 100 usec\n");
|
2009-10-03 04:44:04 +08:00
|
|
|
|
|
|
|
IWL_DEBUG_INFO(priv, "stop master\n");
|
|
|
|
|
2009-10-31 05:36:05 +08:00
|
|
|
return ret;
|
2009-10-03 04:44:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void iwl_apm_stop(struct iwl_priv *priv)
|
|
|
|
{
|
2009-10-24 04:42:21 +08:00
|
|
|
IWL_DEBUG_INFO(priv, "Stop card, put in low power state\n");
|
|
|
|
|
2009-10-31 05:36:05 +08:00
|
|
|
/* Stop device's DMA activity */
|
2009-10-03 04:44:04 +08:00
|
|
|
iwl_apm_stop_master(priv);
|
|
|
|
|
2009-10-31 05:36:05 +08:00
|
|
|
/* Reset the entire device */
|
2009-10-03 04:44:04 +08:00
|
|
|
iwl_set_bit(priv, CSR_RESET, CSR_RESET_REG_FLAG_SW_RESET);
|
|
|
|
|
|
|
|
udelay(10);
|
2009-10-31 05:36:05 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Clear "initialization complete" bit to move adapter from
|
|
|
|
* D0A* (powered-up Active) --> D0U* (Uninitialized) state.
|
|
|
|
*/
|
2009-10-03 04:44:04 +08:00
|
|
|
iwl_clear_bit(priv, CSR_GP_CNTRL, CSR_GP_CNTRL_REG_FLAG_INIT_DONE);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_apm_stop);
|
|
|
|
|
2009-10-24 04:42:21 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Start up NIC's basic functionality after it has been reset
|
|
|
|
* (e.g. after platform boot, or shutdown via iwl_apm_stop())
|
|
|
|
* NOTE: This does not load uCode nor start the embedded processor
|
|
|
|
*/
|
|
|
|
int iwl_apm_init(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
u16 lctl;
|
|
|
|
|
|
|
|
IWL_DEBUG_INFO(priv, "Init card's basic functions\n");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Use "set_bit" below rather than "write", to preserve any hardware
|
|
|
|
* bits already set by default after reset.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Disable L0S exit timer (platform NMI Work/Around) */
|
|
|
|
iwl_set_bit(priv, CSR_GIO_CHICKEN_BITS,
|
|
|
|
CSR_GIO_CHICKEN_BITS_REG_BIT_DIS_L0S_EXIT_TIMER);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Disable L0s without affecting L1;
|
|
|
|
* don't wait for ICH L0s (ICH bug W/A)
|
|
|
|
*/
|
|
|
|
iwl_set_bit(priv, CSR_GIO_CHICKEN_BITS,
|
|
|
|
CSR_GIO_CHICKEN_BITS_REG_BIT_L1A_NO_L0S_RX);
|
|
|
|
|
|
|
|
/* Set FH wait threshold to maximum (HW error during stress W/A) */
|
|
|
|
iwl_set_bit(priv, CSR_DBG_HPET_MEM_REG, CSR_DBG_HPET_MEM_REG_VAL);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable HAP INTA (interrupt from management bus) to
|
|
|
|
* wake device's PCI Express link L1a -> L0s
|
|
|
|
* NOTE: This is no-op for 3945 (non-existant bit)
|
|
|
|
*/
|
|
|
|
iwl_set_bit(priv, CSR_HW_IF_CONFIG_REG,
|
|
|
|
CSR_HW_IF_CONFIG_REG_BIT_HAP_WAKE_L1A);
|
|
|
|
|
|
|
|
/*
|
2009-10-31 05:36:07 +08:00
|
|
|
* HW bug W/A for instability in PCIe bus L0->L0S->L1 transition.
|
|
|
|
* Check if BIOS (or OS) enabled L1-ASPM on this device.
|
|
|
|
* If so (likely), disable L0S, so device moves directly L0->L1;
|
|
|
|
* costs negligible amount of power savings.
|
|
|
|
* If not (unlikely), enable L0S, so there is at least some
|
|
|
|
* power savings, even without L1.
|
2009-10-24 04:42:21 +08:00
|
|
|
*/
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->base_params->set_l0s) {
|
2009-10-24 04:42:21 +08:00
|
|
|
lctl = iwl_pcie_link_ctl(priv);
|
|
|
|
if ((lctl & PCI_CFG_LINK_CTRL_VAL_L1_EN) ==
|
|
|
|
PCI_CFG_LINK_CTRL_VAL_L1_EN) {
|
|
|
|
/* L1-ASPM enabled; disable(!) L0S */
|
|
|
|
iwl_set_bit(priv, CSR_GIO_REG,
|
|
|
|
CSR_GIO_REG_VAL_L0S_ENABLED);
|
|
|
|
IWL_DEBUG_POWER(priv, "L1 Enabled; Disabling L0S\n");
|
|
|
|
} else {
|
|
|
|
/* L1-ASPM disabled; enable(!) L0S */
|
|
|
|
iwl_clear_bit(priv, CSR_GIO_REG,
|
|
|
|
CSR_GIO_REG_VAL_L0S_ENABLED);
|
|
|
|
IWL_DEBUG_POWER(priv, "L1 Disabled; Enabling L0S\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Configure analog phase-lock-loop before activating to D0A */
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->base_params->pll_cfg_val)
|
|
|
|
iwl_set_bit(priv, CSR_ANA_PLL_CFG,
|
|
|
|
priv->cfg->base_params->pll_cfg_val);
|
2009-10-24 04:42:21 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set "initialization complete" bit to move adapter from
|
|
|
|
* D0U* --> D0A* (powered-up active) state.
|
|
|
|
*/
|
|
|
|
iwl_set_bit(priv, CSR_GP_CNTRL, CSR_GP_CNTRL_REG_FLAG_INIT_DONE);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait for clock stabilization; once stabilized, access to
|
|
|
|
* device-internal resources is supported, e.g. iwl_write_prph()
|
|
|
|
* and accesses to uCode SRAM.
|
|
|
|
*/
|
|
|
|
ret = iwl_poll_bit(priv, CSR_GP_CNTRL,
|
|
|
|
CSR_GP_CNTRL_REG_FLAG_MAC_CLOCK_READY,
|
|
|
|
CSR_GP_CNTRL_REG_FLAG_MAC_CLOCK_READY, 25000);
|
|
|
|
if (ret < 0) {
|
|
|
|
IWL_DEBUG_INFO(priv, "Failed to init the card\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable DMA and BSM (if used) clocks, wait for them to stabilize.
|
|
|
|
* BSM (Boostrap State Machine) is only in 3945 and 4965;
|
|
|
|
* later devices (i.e. 5000 and later) have non-volatile SRAM,
|
|
|
|
* and don't need BSM to restore data after power-saving sleep.
|
|
|
|
*
|
|
|
|
* Write to "CLK_EN_REG"; "1" bits enable clocks, while "0" bits
|
|
|
|
* do not disable clocks. This preserves any hardware bits already
|
|
|
|
* set by default in "CLK_CTRL_REG" after reset.
|
|
|
|
*/
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->base_params->use_bsm)
|
2009-10-24 04:42:21 +08:00
|
|
|
iwl_write_prph(priv, APMG_CLK_EN_REG,
|
|
|
|
APMG_CLK_VAL_DMA_CLK_RQT | APMG_CLK_VAL_BSM_CLK_RQT);
|
|
|
|
else
|
|
|
|
iwl_write_prph(priv, APMG_CLK_EN_REG,
|
|
|
|
APMG_CLK_VAL_DMA_CLK_RQT);
|
|
|
|
udelay(20);
|
|
|
|
|
|
|
|
/* Disable L1-Active */
|
|
|
|
iwl_set_bits_prph(priv, APMG_PCIDEV_STT_REG,
|
|
|
|
APMG_PCIDEV_STT_VAL_L1_ACT_DIS);
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_apm_init);
|
|
|
|
|
|
|
|
|
2008-06-12 09:47:08 +08:00
|
|
|
int iwl_set_tx_power(struct iwl_priv *priv, s8 tx_power, bool force)
|
|
|
|
{
|
2010-10-22 23:04:26 +08:00
|
|
|
int ret;
|
|
|
|
s8 prev_tx_power;
|
|
|
|
|
|
|
|
lockdep_assert_held(&priv->mutex);
|
|
|
|
|
|
|
|
if (priv->tx_power_user_lmt == tx_power && !force)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!priv->cfg->ops->lib->send_tx_power)
|
|
|
|
return -EOPNOTSUPP;
|
2009-08-22 04:34:17 +08:00
|
|
|
|
2010-03-24 02:37:59 +08:00
|
|
|
if (tx_power < IWLAGN_TX_POWER_TARGET_POWER_MIN) {
|
|
|
|
IWL_WARN(priv,
|
|
|
|
"Requested user TXPOWER %d below lower limit %d.\n",
|
2009-02-14 03:51:17 +08:00
|
|
|
tx_power,
|
2010-03-24 02:37:59 +08:00
|
|
|
IWLAGN_TX_POWER_TARGET_POWER_MIN);
|
2008-06-12 09:47:08 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2009-08-22 04:34:16 +08:00
|
|
|
if (tx_power > priv->tx_power_device_lmt) {
|
2009-08-22 04:34:15 +08:00
|
|
|
IWL_WARN(priv,
|
|
|
|
"Requested user TXPOWER %d above upper limit %d.\n",
|
2009-08-22 04:34:16 +08:00
|
|
|
tx_power, priv->tx_power_device_lmt);
|
2008-06-12 09:47:08 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2010-10-22 23:04:26 +08:00
|
|
|
if (!iwl_is_ready_rf(priv))
|
|
|
|
return -EIO;
|
2008-06-12 09:47:08 +08:00
|
|
|
|
2010-10-22 23:04:26 +08:00
|
|
|
/* scan complete use tx_power_next, need to be updated */
|
|
|
|
priv->tx_power_next = tx_power;
|
|
|
|
if (test_bit(STATUS_SCANNING, &priv->status) && !force) {
|
|
|
|
IWL_DEBUG_INFO(priv, "Deferring tx power set while scanning\n");
|
|
|
|
return 0;
|
2009-08-22 04:34:17 +08:00
|
|
|
}
|
2008-06-12 09:47:08 +08:00
|
|
|
|
2010-10-22 23:04:26 +08:00
|
|
|
prev_tx_power = priv->tx_power_user_lmt;
|
|
|
|
priv->tx_power_user_lmt = tx_power;
|
|
|
|
|
|
|
|
ret = priv->cfg->ops->lib->send_tx_power(priv);
|
|
|
|
|
|
|
|
/* if fail to set tx_power, restore the orig. tx power */
|
|
|
|
if (ret) {
|
|
|
|
priv->tx_power_user_lmt = prev_tx_power;
|
|
|
|
priv->tx_power_next = prev_tx_power;
|
|
|
|
}
|
2008-06-12 09:47:08 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_set_tx_power);
|
|
|
|
|
2009-05-23 02:01:50 +08:00
|
|
|
irqreturn_t iwl_isr_legacy(int irq, void *data)
|
2009-02-19 07:54:31 +08:00
|
|
|
{
|
|
|
|
struct iwl_priv *priv = data;
|
|
|
|
u32 inta, inta_mask;
|
|
|
|
u32 inta_fh;
|
2010-03-20 01:36:09 +08:00
|
|
|
unsigned long flags;
|
2009-02-19 07:54:31 +08:00
|
|
|
if (!priv)
|
|
|
|
return IRQ_NONE;
|
|
|
|
|
2010-03-20 01:36:09 +08:00
|
|
|
spin_lock_irqsave(&priv->lock, flags);
|
2009-02-19 07:54:31 +08:00
|
|
|
|
|
|
|
/* Disable (but don't clear!) interrupts here to avoid
|
|
|
|
* back-to-back ISRs and sporadic interrupts from our NIC.
|
|
|
|
* If we have something to service, the tasklet will re-enable ints.
|
|
|
|
* If we *don't* have something, we'll re-enable before leaving here. */
|
|
|
|
inta_mask = iwl_read32(priv, CSR_INT_MASK); /* just for debug */
|
|
|
|
iwl_write32(priv, CSR_INT_MASK, 0x00000000);
|
|
|
|
|
|
|
|
/* Discover which interrupts are active/pending */
|
|
|
|
inta = iwl_read32(priv, CSR_INT);
|
|
|
|
inta_fh = iwl_read32(priv, CSR_FH_INT_STATUS);
|
|
|
|
|
|
|
|
/* Ignore interrupt if there's nothing in NIC to service.
|
|
|
|
* This may be due to IRQ shared with another device,
|
|
|
|
* or due to sporadic interrupts thrown from our NIC. */
|
|
|
|
if (!inta && !inta_fh) {
|
|
|
|
IWL_DEBUG_ISR(priv, "Ignore interrupt, inta == 0, inta_fh == 0\n");
|
|
|
|
goto none;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((inta == 0xFFFFFFFF) || ((inta & 0xFFFFFFF0) == 0xa5a5a5a0)) {
|
|
|
|
/* Hardware disappeared. It might have already raised
|
|
|
|
* an interrupt */
|
|
|
|
IWL_WARN(priv, "HARDWARE GONE?? INTA == 0x%08x\n", inta);
|
|
|
|
goto unplugged;
|
|
|
|
}
|
|
|
|
|
|
|
|
IWL_DEBUG_ISR(priv, "ISR inta 0x%08x, enabled 0x%08x, fh 0x%08x\n",
|
|
|
|
inta, inta_mask, inta_fh);
|
|
|
|
|
|
|
|
inta &= ~CSR_INT_BIT_SCD;
|
|
|
|
|
|
|
|
/* iwl_irq_tasklet() will service interrupts and re-enable them */
|
|
|
|
if (likely(inta || inta_fh))
|
|
|
|
tasklet_schedule(&priv->irq_tasklet);
|
|
|
|
|
|
|
|
unplugged:
|
2010-03-20 01:36:09 +08:00
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
2009-02-19 07:54:31 +08:00
|
|
|
return IRQ_HANDLED;
|
|
|
|
|
|
|
|
none:
|
|
|
|
/* re-enable interrupts here since we don't have anything to service. */
|
|
|
|
/* only Re-enable if diabled by irq */
|
|
|
|
if (test_bit(STATUS_INT_ENABLED, &priv->status))
|
|
|
|
iwl_enable_interrupts(priv);
|
2010-03-20 01:36:09 +08:00
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
2009-02-19 07:54:31 +08:00
|
|
|
return IRQ_NONE;
|
|
|
|
}
|
2009-05-23 02:01:50 +08:00
|
|
|
EXPORT_SYMBOL(iwl_isr_legacy);
|
2009-02-19 07:54:31 +08:00
|
|
|
|
2010-04-13 16:04:31 +08:00
|
|
|
void iwl_send_bt_config(struct iwl_priv *priv)
|
2009-01-24 05:45:20 +08:00
|
|
|
{
|
|
|
|
struct iwl_bt_cmd bt_cmd = {
|
2009-10-24 04:42:23 +08:00
|
|
|
.lead_time = BT_LEAD_TIME_DEF,
|
|
|
|
.max_kill = BT_MAX_KILL_DEF,
|
2009-01-24 05:45:20 +08:00
|
|
|
.kill_ack_mask = 0,
|
|
|
|
.kill_cts_mask = 0,
|
|
|
|
};
|
|
|
|
|
2010-01-23 06:22:51 +08:00
|
|
|
if (!bt_coex_active)
|
|
|
|
bt_cmd.flags = BT_COEX_DISABLE;
|
|
|
|
else
|
|
|
|
bt_cmd.flags = BT_COEX_ENABLE;
|
|
|
|
|
|
|
|
IWL_DEBUG_INFO(priv, "BT coex %s\n",
|
|
|
|
(bt_cmd.flags == BT_COEX_DISABLE) ? "disable" : "active");
|
|
|
|
|
2010-04-13 16:04:31 +08:00
|
|
|
if (iwl_send_cmd_pdu(priv, REPLY_BT_CONFIG,
|
|
|
|
sizeof(struct iwl_bt_cmd), &bt_cmd))
|
|
|
|
IWL_ERR(priv, "failed to send BT Coex Config\n");
|
2009-01-24 05:45:20 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_send_bt_config);
|
|
|
|
|
2009-11-14 03:56:28 +08:00
|
|
|
int iwl_send_statistics_request(struct iwl_priv *priv, u8 flags, bool clear)
|
2008-04-16 07:01:37 +08:00
|
|
|
{
|
2009-11-14 03:56:28 +08:00
|
|
|
struct iwl_statistics_cmd statistics_cmd = {
|
|
|
|
.configuration_flags =
|
|
|
|
clear ? IWL_STATS_CONF_CLEAR_STATS : 0,
|
2008-04-16 07:01:37 +08:00
|
|
|
};
|
2009-11-14 03:56:28 +08:00
|
|
|
|
|
|
|
if (flags & CMD_ASYNC)
|
|
|
|
return iwl_send_cmd_pdu_async(priv, REPLY_STATISTICS_CMD,
|
|
|
|
sizeof(struct iwl_statistics_cmd),
|
|
|
|
&statistics_cmd, NULL);
|
|
|
|
else
|
|
|
|
return iwl_send_cmd_pdu(priv, REPLY_STATISTICS_CMD,
|
|
|
|
sizeof(struct iwl_statistics_cmd),
|
|
|
|
&statistics_cmd);
|
2008-04-16 07:01:37 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_send_statistics_request);
|
2008-04-16 07:01:43 +08:00
|
|
|
|
2009-02-19 07:54:28 +08:00
|
|
|
void iwl_rx_pm_sleep_notif(struct iwl_priv *priv,
|
|
|
|
struct iwl_rx_mem_buffer *rxb)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_IWLWIFI_DEBUG
|
2009-10-09 17:19:45 +08:00
|
|
|
struct iwl_rx_packet *pkt = rxb_addr(rxb);
|
2009-02-19 07:54:28 +08:00
|
|
|
struct iwl_sleep_notification *sleep = &(pkt->u.sleep_notif);
|
|
|
|
IWL_DEBUG_RX(priv, "sleep mode: %d, src: %d\n",
|
|
|
|
sleep->pm_sleep_mode, sleep->pm_wakeup_src);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_rx_pm_sleep_notif);
|
|
|
|
|
|
|
|
void iwl_rx_pm_debug_statistics_notif(struct iwl_priv *priv,
|
|
|
|
struct iwl_rx_mem_buffer *rxb)
|
|
|
|
{
|
2009-10-09 17:19:45 +08:00
|
|
|
struct iwl_rx_packet *pkt = rxb_addr(rxb);
|
iwlwifi: fix erroneous use of iwl_rx_packet.len as a length
The field called 'len' in struct iwl_rx_packet is in fact not just a length
field but also includes some flags from the flow handler. In several places
throughout the driver, this causes incorrect values to be interpreted as
lengths when the field is improperly masked.
In most situations the improper use is for debugging output, and simply results
in an erroneous message, such as:
[551933.070224] ieee80211 phy0: I iwl_rx_statistics Statistics notification received (480 vs -1367342620).
which should read '(480 vs 484)'.
In at least one case this could case bad things to happen:
void iwl_rx_pm_debug_statistics_notif(struct iwl_priv *priv,
struct iwl_rx_mem_buffer *rxb)
{
struct iwl_rx_packet *pkt = (struct iwl_rx_packet *)rxb->skb->data;
IWL_DEBUG_RADIO(priv, "Dumping %d bytes of unhandled "
"notification for %s:\n",
le32_to_cpu(pkt->len), get_cmd_string(pkt->hdr.cmd));
iwl_print_hex_dump(priv, IWL_DL_RADIO, pkt->u.raw, le32_to_cpu(pkt->len)
);
}
EXPORT_SYMBOL(iwl_rx_pm_debug_statistics_notif);
Given the rampant misuse of this field without proper masking throughout the
driver (every use but one), this patch renames the field from 'len' to
'len_n_flags' to reduce confusion. It also adds the proper masking when
this field is used as a length value.
Signed-off-by: Daniel C Halperin <daniel.c.halperin@intel.com>
Signed-off-by: Reinette Chatre <reinette.chatre@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2009-08-14 04:31:01 +08:00
|
|
|
u32 len = le32_to_cpu(pkt->len_n_flags) & FH_RSCSR_FRAME_SIZE_MSK;
|
2009-02-19 07:54:28 +08:00
|
|
|
IWL_DEBUG_RADIO(priv, "Dumping %d bytes of unhandled "
|
iwlwifi: fix erroneous use of iwl_rx_packet.len as a length
The field called 'len' in struct iwl_rx_packet is in fact not just a length
field but also includes some flags from the flow handler. In several places
throughout the driver, this causes incorrect values to be interpreted as
lengths when the field is improperly masked.
In most situations the improper use is for debugging output, and simply results
in an erroneous message, such as:
[551933.070224] ieee80211 phy0: I iwl_rx_statistics Statistics notification received (480 vs -1367342620).
which should read '(480 vs 484)'.
In at least one case this could case bad things to happen:
void iwl_rx_pm_debug_statistics_notif(struct iwl_priv *priv,
struct iwl_rx_mem_buffer *rxb)
{
struct iwl_rx_packet *pkt = (struct iwl_rx_packet *)rxb->skb->data;
IWL_DEBUG_RADIO(priv, "Dumping %d bytes of unhandled "
"notification for %s:\n",
le32_to_cpu(pkt->len), get_cmd_string(pkt->hdr.cmd));
iwl_print_hex_dump(priv, IWL_DL_RADIO, pkt->u.raw, le32_to_cpu(pkt->len)
);
}
EXPORT_SYMBOL(iwl_rx_pm_debug_statistics_notif);
Given the rampant misuse of this field without proper masking throughout the
driver (every use but one), this patch renames the field from 'len' to
'len_n_flags' to reduce confusion. It also adds the proper masking when
this field is used as a length value.
Signed-off-by: Daniel C Halperin <daniel.c.halperin@intel.com>
Signed-off-by: Reinette Chatre <reinette.chatre@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2009-08-14 04:31:01 +08:00
|
|
|
"notification for %s:\n", len,
|
|
|
|
get_cmd_string(pkt->hdr.cmd));
|
|
|
|
iwl_print_hex_dump(priv, IWL_DL_RADIO, pkt->u.raw, len);
|
2009-02-19 07:54:28 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_rx_pm_debug_statistics_notif);
|
2009-02-19 07:54:29 +08:00
|
|
|
|
|
|
|
void iwl_rx_reply_error(struct iwl_priv *priv,
|
|
|
|
struct iwl_rx_mem_buffer *rxb)
|
|
|
|
{
|
2009-10-09 17:19:45 +08:00
|
|
|
struct iwl_rx_packet *pkt = rxb_addr(rxb);
|
2009-02-19 07:54:29 +08:00
|
|
|
|
|
|
|
IWL_ERR(priv, "Error Reply type 0x%08X cmd %s (0x%02X) "
|
|
|
|
"seq 0x%04X ser 0x%08X\n",
|
|
|
|
le32_to_cpu(pkt->u.err_resp.error_type),
|
|
|
|
get_cmd_string(pkt->u.err_resp.cmd_id),
|
|
|
|
pkt->u.err_resp.cmd_id,
|
|
|
|
le16_to_cpu(pkt->u.err_resp.bad_cmd_seq_num),
|
|
|
|
le32_to_cpu(pkt->u.err_resp.error_info));
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_rx_reply_error);
|
|
|
|
|
2009-04-09 02:39:32 +08:00
|
|
|
void iwl_clear_isr_stats(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
memset(&priv->isr_stats, 0, sizeof(priv->isr_stats));
|
|
|
|
}
|
|
|
|
|
2009-03-27 01:14:10 +08:00
|
|
|
int iwl_mac_conf_tx(struct ieee80211_hw *hw, u16 queue,
|
|
|
|
const struct ieee80211_tx_queue_params *params)
|
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
2010-08-23 16:46:38 +08:00
|
|
|
struct iwl_rxon_context *ctx;
|
2009-03-27 01:14:10 +08:00
|
|
|
unsigned long flags;
|
|
|
|
int q;
|
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "enter\n");
|
|
|
|
|
|
|
|
if (!iwl_is_ready_rf(priv)) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave - RF not ready\n");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (queue >= AC_NUM) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave - queue >= AC_NUM %d\n", queue);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
q = AC_NUM - 1 - queue;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&priv->lock, flags);
|
|
|
|
|
2010-08-23 16:46:38 +08:00
|
|
|
/*
|
|
|
|
* MULTI-FIXME
|
|
|
|
* This may need to be done per interface in nl80211/cfg80211/mac80211.
|
|
|
|
*/
|
|
|
|
for_each_context(priv, ctx) {
|
|
|
|
ctx->qos_data.def_qos_parm.ac[q].cw_min =
|
|
|
|
cpu_to_le16(params->cw_min);
|
|
|
|
ctx->qos_data.def_qos_parm.ac[q].cw_max =
|
|
|
|
cpu_to_le16(params->cw_max);
|
|
|
|
ctx->qos_data.def_qos_parm.ac[q].aifsn = params->aifs;
|
|
|
|
ctx->qos_data.def_qos_parm.ac[q].edca_txop =
|
|
|
|
cpu_to_le16((params->txop * 32));
|
|
|
|
|
|
|
|
ctx->qos_data.def_qos_parm.ac[q].reserved1 = 0;
|
|
|
|
}
|
2009-03-27 01:14:10 +08:00
|
|
|
|
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_mac_conf_tx);
|
2009-04-09 02:26:35 +08:00
|
|
|
|
2010-07-31 23:34:10 +08:00
|
|
|
int iwl_mac_tx_last_beacon(struct ieee80211_hw *hw)
|
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
|
|
|
|
|
|
|
return priv->ibss_manager == IWL_IBSS_MANAGER;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iwl_mac_tx_last_beacon);
|
|
|
|
|
2009-04-09 02:26:35 +08:00
|
|
|
static void iwl_ht_conf(struct iwl_priv *priv,
|
2010-04-29 19:43:09 +08:00
|
|
|
struct ieee80211_vif *vif)
|
2009-04-09 02:26:35 +08:00
|
|
|
{
|
2009-09-12 01:38:15 +08:00
|
|
|
struct iwl_ht_config *ht_conf = &priv->current_ht_config;
|
2009-04-09 02:26:35 +08:00
|
|
|
struct ieee80211_sta *sta;
|
2010-04-29 19:43:09 +08:00
|
|
|
struct ieee80211_bss_conf *bss_conf = &vif->bss_conf;
|
2010-08-23 16:46:46 +08:00
|
|
|
struct iwl_rxon_context *ctx = iwl_rxon_ctx_from_vif(vif);
|
2009-04-09 02:26:35 +08:00
|
|
|
|
2010-03-25 05:19:58 +08:00
|
|
|
IWL_DEBUG_MAC80211(priv, "enter:\n");
|
2009-04-09 02:26:35 +08:00
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
if (!ctx->ht.enabled)
|
2009-04-09 02:26:35 +08:00
|
|
|
return;
|
|
|
|
|
2010-08-23 16:46:46 +08:00
|
|
|
ctx->ht.protection =
|
2009-05-09 02:47:39 +08:00
|
|
|
bss_conf->ht_operation_mode & IEEE80211_HT_OP_MODE_PROTECTION;
|
2010-08-23 16:46:46 +08:00
|
|
|
ctx->ht.non_gf_sta_present =
|
2009-05-09 02:47:39 +08:00
|
|
|
!!(bss_conf->ht_operation_mode & IEEE80211_HT_OP_MODE_NON_GF_STA_PRSNT);
|
2009-04-09 02:26:35 +08:00
|
|
|
|
iwlwifi: clarify and clean up chain settings
The chain settings we currently use in iwlwifi are
rather confusing -- and we also go by the wrong
settings entirely under certain circumstances. To
clean it up, create a new variable in the current
HT config -- single_chain_sufficient -- that tells
us whether we need more than one chain. Calculate
that based on the AP and operating mode (no IBSS
HT implemented -- so no need for multiple chains,
for station mode we use the AP's capabilities).
Additionally, since APs always send disabled SM PS
mode, keeping track of their sm_ps mode isn't very
useful -- doubly not so for our _own_ RX config
since that should depend on our, not the AP's, SM
PS mode.
Finally, document that our configuration of the
number of RX chains used is currently wrong when
in powersave (by adding a comment).
All together this removes the two remaining items
in struct iwl_ht_config that were done wrong there.
For the future, the number of RX chains and some
SM PS handshaking needs to be added to mac80211,
which then needs to tell us, and the new variable
current_ht_config.single_chain_sufficient should
also be calculated by mac80211.
Signed-off-by: Johannes Berg <johannes@sipsolutions.net>
Acked-by: Daniel C Halperin <daniel.c.halperin@intel.com>
Signed-off-by: Reinette Chatre <reinette.chatre@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2009-09-12 01:38:17 +08:00
|
|
|
ht_conf->single_chain_sufficient = false;
|
|
|
|
|
2010-04-29 19:43:09 +08:00
|
|
|
switch (vif->type) {
|
iwlwifi: clarify and clean up chain settings
The chain settings we currently use in iwlwifi are
rather confusing -- and we also go by the wrong
settings entirely under certain circumstances. To
clean it up, create a new variable in the current
HT config -- single_chain_sufficient -- that tells
us whether we need more than one chain. Calculate
that based on the AP and operating mode (no IBSS
HT implemented -- so no need for multiple chains,
for station mode we use the AP's capabilities).
Additionally, since APs always send disabled SM PS
mode, keeping track of their sm_ps mode isn't very
useful -- doubly not so for our _own_ RX config
since that should depend on our, not the AP's, SM
PS mode.
Finally, document that our configuration of the
number of RX chains used is currently wrong when
in powersave (by adding a comment).
All together this removes the two remaining items
in struct iwl_ht_config that were done wrong there.
For the future, the number of RX chains and some
SM PS handshaking needs to be added to mac80211,
which then needs to tell us, and the new variable
current_ht_config.single_chain_sufficient should
also be calculated by mac80211.
Signed-off-by: Johannes Berg <johannes@sipsolutions.net>
Acked-by: Daniel C Halperin <daniel.c.halperin@intel.com>
Signed-off-by: Reinette Chatre <reinette.chatre@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2009-09-12 01:38:17 +08:00
|
|
|
case NL80211_IFTYPE_STATION:
|
|
|
|
rcu_read_lock();
|
2010-04-29 19:43:09 +08:00
|
|
|
sta = ieee80211_find_sta(vif, bss_conf->bssid);
|
iwlwifi: clarify and clean up chain settings
The chain settings we currently use in iwlwifi are
rather confusing -- and we also go by the wrong
settings entirely under certain circumstances. To
clean it up, create a new variable in the current
HT config -- single_chain_sufficient -- that tells
us whether we need more than one chain. Calculate
that based on the AP and operating mode (no IBSS
HT implemented -- so no need for multiple chains,
for station mode we use the AP's capabilities).
Additionally, since APs always send disabled SM PS
mode, keeping track of their sm_ps mode isn't very
useful -- doubly not so for our _own_ RX config
since that should depend on our, not the AP's, SM
PS mode.
Finally, document that our configuration of the
number of RX chains used is currently wrong when
in powersave (by adding a comment).
All together this removes the two remaining items
in struct iwl_ht_config that were done wrong there.
For the future, the number of RX chains and some
SM PS handshaking needs to be added to mac80211,
which then needs to tell us, and the new variable
current_ht_config.single_chain_sufficient should
also be calculated by mac80211.
Signed-off-by: Johannes Berg <johannes@sipsolutions.net>
Acked-by: Daniel C Halperin <daniel.c.halperin@intel.com>
Signed-off-by: Reinette Chatre <reinette.chatre@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2009-09-12 01:38:17 +08:00
|
|
|
if (sta) {
|
|
|
|
struct ieee80211_sta_ht_cap *ht_cap = &sta->ht_cap;
|
|
|
|
int maxstreams;
|
|
|
|
|
|
|
|
maxstreams = (ht_cap->mcs.tx_params &
|
|
|
|
IEEE80211_HT_MCS_TX_MAX_STREAMS_MASK)
|
|
|
|
>> IEEE80211_HT_MCS_TX_MAX_STREAMS_SHIFT;
|
|
|
|
maxstreams += 1;
|
|
|
|
|
|
|
|
if ((ht_cap->mcs.rx_mask[1] == 0) &&
|
|
|
|
(ht_cap->mcs.rx_mask[2] == 0))
|
|
|
|
ht_conf->single_chain_sufficient = true;
|
|
|
|
if (maxstreams <= 1)
|
|
|
|
ht_conf->single_chain_sufficient = true;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* If at all, this can only happen through a race
|
|
|
|
* when the AP disconnects us while we're still
|
|
|
|
* setting up the connection, in that case mac80211
|
|
|
|
* will soon tell us about that.
|
|
|
|
*/
|
|
|
|
ht_conf->single_chain_sufficient = true;
|
|
|
|
}
|
|
|
|
rcu_read_unlock();
|
|
|
|
break;
|
|
|
|
case NL80211_IFTYPE_ADHOC:
|
|
|
|
ht_conf->single_chain_sufficient = true;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2009-04-09 02:26:35 +08:00
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave\n");
|
|
|
|
}
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
static inline void iwl_set_no_assoc(struct iwl_priv *priv,
|
|
|
|
struct ieee80211_vif *vif)
|
2010-01-09 02:04:31 +08:00
|
|
|
{
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx = iwl_rxon_ctx_from_vif(vif);
|
|
|
|
|
2010-01-09 02:04:31 +08:00
|
|
|
iwl_led_disassociate(priv);
|
|
|
|
/*
|
|
|
|
* inform the ucode that there is no longer an
|
|
|
|
* association and that no more packets should be
|
|
|
|
* sent
|
|
|
|
*/
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.filter_flags &= ~RXON_FILTER_ASSOC_MSK;
|
|
|
|
ctx->staging.assoc_id = 0;
|
|
|
|
iwlcore_commit_rxon(priv, ctx);
|
2010-01-09 02:04:31 +08:00
|
|
|
}
|
|
|
|
|
2010-09-23 00:01:59 +08:00
|
|
|
static void iwlcore_beacon_update(struct ieee80211_hw *hw,
|
|
|
|
struct ieee80211_vif *vif)
|
2010-07-15 20:57:48 +08:00
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
|
|
|
unsigned long flags;
|
|
|
|
__le64 timestamp;
|
2010-09-23 00:01:59 +08:00
|
|
|
struct sk_buff *skb = ieee80211_beacon_get(hw, vif);
|
2010-07-15 20:57:48 +08:00
|
|
|
|
2010-09-23 00:01:59 +08:00
|
|
|
if (!skb)
|
|
|
|
return;
|
|
|
|
|
|
|
|
IWL_DEBUG_ASSOC(priv, "enter\n");
|
2010-07-15 20:57:48 +08:00
|
|
|
|
2010-08-23 16:46:47 +08:00
|
|
|
lockdep_assert_held(&priv->mutex);
|
|
|
|
|
|
|
|
if (!priv->beacon_ctx) {
|
|
|
|
IWL_ERR(priv, "update beacon but no beacon context!\n");
|
|
|
|
dev_kfree_skb(skb);
|
2010-09-23 00:01:59 +08:00
|
|
|
return;
|
2010-07-15 20:57:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock_irqsave(&priv->lock, flags);
|
|
|
|
|
2010-10-04 20:50:06 +08:00
|
|
|
if (priv->beacon_skb)
|
|
|
|
dev_kfree_skb(priv->beacon_skb);
|
2010-07-15 20:57:48 +08:00
|
|
|
|
2010-10-04 20:50:06 +08:00
|
|
|
priv->beacon_skb = skb;
|
2010-07-15 20:57:48 +08:00
|
|
|
|
|
|
|
timestamp = ((struct ieee80211_mgmt *)skb->data)->u.beacon.timestamp;
|
|
|
|
priv->timestamp = le64_to_cpu(timestamp);
|
|
|
|
|
2010-09-23 00:01:59 +08:00
|
|
|
IWL_DEBUG_ASSOC(priv, "leave\n");
|
|
|
|
|
2010-07-15 20:57:48 +08:00
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
|
|
|
|
2010-09-23 00:01:59 +08:00
|
|
|
if (!iwl_is_ready_rf(priv)) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave - RF not ready\n");
|
|
|
|
return;
|
|
|
|
}
|
2010-07-15 20:57:48 +08:00
|
|
|
|
2010-09-23 00:01:59 +08:00
|
|
|
priv->cfg->ops->lib->post_associate(priv, priv->beacon_ctx->vif);
|
2010-07-15 20:57:48 +08:00
|
|
|
}
|
|
|
|
|
2009-04-09 02:26:35 +08:00
|
|
|
void iwl_bss_info_changed(struct ieee80211_hw *hw,
|
2009-04-23 22:13:26 +08:00
|
|
|
struct ieee80211_vif *vif,
|
|
|
|
struct ieee80211_bss_conf *bss_conf,
|
|
|
|
u32 changes)
|
2009-04-09 02:26:35 +08:00
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx = iwl_rxon_ctx_from_vif(vif);
|
2009-04-09 02:39:29 +08:00
|
|
|
int ret;
|
2009-04-09 02:26:35 +08:00
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "changes = 0x%X\n", changes);
|
|
|
|
|
2009-04-23 22:13:26 +08:00
|
|
|
if (!iwl_is_alive(priv))
|
|
|
|
return;
|
|
|
|
|
|
|
|
mutex_lock(&priv->mutex);
|
|
|
|
|
2010-10-22 23:04:30 +08:00
|
|
|
if (changes & (BSS_CHANGED_BSSID | BSS_CHANGED_ASSOC |
|
|
|
|
BSS_CHANGED_BEACON_ENABLED)) {
|
|
|
|
/*
|
|
|
|
* If there is currently a HW scan going on in the
|
|
|
|
* background then we need to cancel it else the RXON
|
|
|
|
* below in post_associate or set_no_assoc can fail.
|
|
|
|
*/
|
|
|
|
if (iwl_scan_cancel_timeout(priv, 200)) {
|
|
|
|
IWL_WARN(priv, "Can not cancel scan\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-19 22:39:04 +08:00
|
|
|
if (changes & BSS_CHANGED_QOS) {
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&priv->lock, flags);
|
2010-08-23 16:46:38 +08:00
|
|
|
ctx->qos_data.qos_active = bss_conf->qos;
|
|
|
|
iwl_update_qos(priv, ctx);
|
2010-07-19 22:39:04 +08:00
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
|
|
|
}
|
|
|
|
|
2010-08-23 16:46:47 +08:00
|
|
|
if (changes & BSS_CHANGED_BEACON_ENABLED) {
|
|
|
|
/*
|
|
|
|
* the add_interface code must make sure we only ever
|
|
|
|
* have a single interface that could be beaconing at
|
|
|
|
* any time.
|
|
|
|
*/
|
|
|
|
if (vif->bss_conf.enable_beacon)
|
|
|
|
priv->beacon_ctx = ctx;
|
|
|
|
else
|
|
|
|
priv->beacon_ctx = NULL;
|
|
|
|
}
|
|
|
|
|
2010-04-27 18:36:15 +08:00
|
|
|
if (changes & BSS_CHANGED_BEACON && vif->type == NL80211_IFTYPE_AP) {
|
2010-10-04 20:50:06 +08:00
|
|
|
dev_kfree_skb(priv->beacon_skb);
|
|
|
|
priv->beacon_skb = ieee80211_beacon_get(hw, vif);
|
2009-04-23 22:13:26 +08:00
|
|
|
}
|
|
|
|
|
2010-08-30 21:13:42 +08:00
|
|
|
if (changes & BSS_CHANGED_BEACON_INT && vif->type == NL80211_IFTYPE_AP)
|
|
|
|
iwl_send_rxon_timing(priv, ctx);
|
2009-06-13 04:22:48 +08:00
|
|
|
|
|
|
|
if (changes & BSS_CHANGED_BSSID) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "BSSID %pM\n", bss_conf->bssid);
|
|
|
|
|
|
|
|
/* mac80211 only sets assoc when in STATION mode */
|
2010-04-27 18:36:15 +08:00
|
|
|
if (vif->type == NL80211_IFTYPE_ADHOC || bss_conf->assoc) {
|
2010-08-23 16:46:32 +08:00
|
|
|
memcpy(ctx->staging.bssid_addr,
|
2009-06-13 04:22:48 +08:00
|
|
|
bss_conf->bssid, ETH_ALEN);
|
2009-04-23 22:13:26 +08:00
|
|
|
|
2009-06-13 04:22:48 +08:00
|
|
|
/* currently needed in a few places */
|
|
|
|
memcpy(priv->bssid, bss_conf->bssid, ETH_ALEN);
|
|
|
|
} else {
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.filter_flags &=
|
2009-06-13 04:22:48 +08:00
|
|
|
~RXON_FILTER_ASSOC_MSK;
|
2009-04-23 22:13:26 +08:00
|
|
|
}
|
2009-06-13 04:22:48 +08:00
|
|
|
|
2009-04-23 22:13:26 +08:00
|
|
|
}
|
|
|
|
|
2009-06-13 04:22:48 +08:00
|
|
|
/*
|
|
|
|
* This needs to be after setting the BSSID in case
|
|
|
|
* mac80211 decides to do both changes at once because
|
|
|
|
* it will invoke post_associate.
|
|
|
|
*/
|
2010-09-23 00:01:59 +08:00
|
|
|
if (vif->type == NL80211_IFTYPE_ADHOC && changes & BSS_CHANGED_BEACON)
|
|
|
|
iwlcore_beacon_update(hw, vif);
|
2009-04-23 22:13:26 +08:00
|
|
|
|
2009-04-09 02:26:35 +08:00
|
|
|
if (changes & BSS_CHANGED_ERP_PREAMBLE) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "ERP_PREAMBLE %d\n",
|
|
|
|
bss_conf->use_short_preamble);
|
|
|
|
if (bss_conf->use_short_preamble)
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_SHORT_PREAMBLE_MSK;
|
2009-04-09 02:26:35 +08:00
|
|
|
else
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &= ~RXON_FLG_SHORT_PREAMBLE_MSK;
|
2009-04-09 02:26:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (changes & BSS_CHANGED_ERP_CTS_PROT) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "ERP_CTS %d\n", bss_conf->use_cts_prot);
|
|
|
|
if (bss_conf->use_cts_prot && (priv->band != IEEE80211_BAND_5GHZ))
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_TGG_PROTECT_MSK;
|
2009-04-09 02:26:35 +08:00
|
|
|
else
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &= ~RXON_FLG_TGG_PROTECT_MSK;
|
iwlagn: fix rts cts protection
Currently the driver will try to protect all frames,
which leads to a lot of odd things like sending an
RTS with a zeroed RA before multicast frames, which
is clearly bogus.
In order to fix all of this, we need to take a step
back and see what we need to achieve:
* we need RTS/CTS protection if requested by
the AP for the BSS, mac80211 tells us this
* in that case, CTS-to-self should only be
enabled when mac80211 tells us
* additionally, as a hardware workaround, on
some devices we have to protect aggregated
frames with RTS
To achieve the first two items, set up the RXON
accordingly and set the protection required flag
in the transmit command when mac80211 requests
protection for the frame.
To achieve the last item, set the rate-control
RTS-requested flag for all stations that we have
aggregation sessions with, and set the protection
required flag when sending aggregated frames (on
those devices where this is required).
Since otherwise bugs can occur, do not allow the
user to override the RTS-for-aggregation setting
from sysfs any more.
Finally, also clean up the way all these flags get
set in the driver and move everything into the
device-specific functions.
Cc: stable@kernel.org [2.6.35]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2010-08-10 01:57:02 +08:00
|
|
|
if (bss_conf->use_cts_prot)
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags |= RXON_FLG_SELF_CTS_EN;
|
iwlagn: fix rts cts protection
Currently the driver will try to protect all frames,
which leads to a lot of odd things like sending an
RTS with a zeroed RA before multicast frames, which
is clearly bogus.
In order to fix all of this, we need to take a step
back and see what we need to achieve:
* we need RTS/CTS protection if requested by
the AP for the BSS, mac80211 tells us this
* in that case, CTS-to-self should only be
enabled when mac80211 tells us
* additionally, as a hardware workaround, on
some devices we have to protect aggregated
frames with RTS
To achieve the first two items, set up the RXON
accordingly and set the protection required flag
in the transmit command when mac80211 requests
protection for the frame.
To achieve the last item, set the rate-control
RTS-requested flag for all stations that we have
aggregation sessions with, and set the protection
required flag when sending aggregated frames (on
those devices where this is required).
Since otherwise bugs can occur, do not allow the
user to override the RTS-for-aggregation setting
from sysfs any more.
Finally, also clean up the way all these flags get
set in the driver and move everything into the
device-specific functions.
Cc: stable@kernel.org [2.6.35]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2010-08-10 01:57:02 +08:00
|
|
|
else
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.flags &= ~RXON_FLG_SELF_CTS_EN;
|
2009-04-09 02:26:35 +08:00
|
|
|
}
|
|
|
|
|
2009-06-13 04:22:48 +08:00
|
|
|
if (changes & BSS_CHANGED_BASIC_RATES) {
|
|
|
|
/* XXX use this information
|
|
|
|
*
|
|
|
|
* To do that, remove code from iwl_set_rate() and put something
|
|
|
|
* like this here:
|
|
|
|
*
|
|
|
|
if (A-band)
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.ofdm_basic_rates =
|
2009-06-13 04:22:48 +08:00
|
|
|
bss_conf->basic_rates;
|
|
|
|
else
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.ofdm_basic_rates =
|
2009-06-13 04:22:48 +08:00
|
|
|
bss_conf->basic_rates >> 4;
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.cck_basic_rates =
|
2009-06-13 04:22:48 +08:00
|
|
|
bss_conf->basic_rates & 0xF;
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
2009-04-09 02:26:35 +08:00
|
|
|
if (changes & BSS_CHANGED_HT) {
|
2010-04-29 19:43:09 +08:00
|
|
|
iwl_ht_conf(priv, vif);
|
2009-04-09 02:26:44 +08:00
|
|
|
|
|
|
|
if (priv->cfg->ops->hcmd->set_rxon_chain)
|
2010-08-23 16:46:32 +08:00
|
|
|
priv->cfg->ops->hcmd->set_rxon_chain(priv, ctx);
|
2009-04-09 02:26:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (changes & BSS_CHANGED_ASSOC) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "ASSOC %d\n", bss_conf->assoc);
|
|
|
|
if (bss_conf->assoc) {
|
|
|
|
priv->timestamp = bss_conf->timestamp;
|
|
|
|
|
2009-10-03 04:44:03 +08:00
|
|
|
iwl_led_associate(priv);
|
|
|
|
|
2009-06-13 04:22:48 +08:00
|
|
|
if (!iwl_is_rfkill(priv))
|
2010-04-29 19:43:06 +08:00
|
|
|
priv->cfg->ops->lib->post_associate(priv, vif);
|
2010-01-09 02:04:31 +08:00
|
|
|
} else
|
2010-08-23 16:46:32 +08:00
|
|
|
iwl_set_no_assoc(priv, vif);
|
2009-06-13 04:22:48 +08:00
|
|
|
}
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
if (changes && iwl_is_associated_ctx(ctx) && bss_conf->aid) {
|
2009-06-13 04:22:48 +08:00
|
|
|
IWL_DEBUG_MAC80211(priv, "Changes (%#x) while associated\n",
|
|
|
|
changes);
|
2010-08-23 16:46:32 +08:00
|
|
|
ret = iwl_send_rxon_assoc(priv, ctx);
|
2009-06-13 04:22:48 +08:00
|
|
|
if (!ret) {
|
|
|
|
/* Sync active_rxon with latest change. */
|
2010-08-23 16:46:32 +08:00
|
|
|
memcpy((void *)&ctx->active,
|
|
|
|
&ctx->staging,
|
2009-06-13 04:22:48 +08:00
|
|
|
sizeof(struct iwl_rxon_cmd));
|
2009-04-09 02:26:35 +08:00
|
|
|
}
|
|
|
|
}
|
2009-06-13 04:22:48 +08:00
|
|
|
|
2010-01-09 02:04:31 +08:00
|
|
|
if (changes & BSS_CHANGED_BEACON_ENABLED) {
|
|
|
|
if (vif->bss_conf.enable_beacon) {
|
2010-08-23 16:46:32 +08:00
|
|
|
memcpy(ctx->staging.bssid_addr,
|
2010-01-09 02:04:31 +08:00
|
|
|
bss_conf->bssid, ETH_ALEN);
|
|
|
|
memcpy(priv->bssid, bss_conf->bssid, ETH_ALEN);
|
2010-10-07 18:52:35 +08:00
|
|
|
iwl_led_associate(priv);
|
2010-04-29 19:43:06 +08:00
|
|
|
iwlcore_config_ap(priv, vif);
|
2010-01-09 02:04:31 +08:00
|
|
|
} else
|
2010-08-23 16:46:32 +08:00
|
|
|
iwl_set_no_assoc(priv, vif);
|
2009-11-14 03:56:34 +08:00
|
|
|
}
|
|
|
|
|
2010-04-28 23:44:52 +08:00
|
|
|
if (changes & BSS_CHANGED_IBSS) {
|
|
|
|
ret = priv->cfg->ops->lib->manage_ibss_station(priv, vif,
|
|
|
|
bss_conf->ibss_joined);
|
|
|
|
if (ret)
|
|
|
|
IWL_ERR(priv, "failed to %s IBSS station %pM\n",
|
|
|
|
bss_conf->ibss_joined ? "add" : "remove",
|
|
|
|
bss_conf->bssid);
|
|
|
|
}
|
|
|
|
|
2010-08-28 00:44:50 +08:00
|
|
|
if (changes & BSS_CHANGED_IDLE &&
|
|
|
|
priv->cfg->ops->hcmd->set_pan_params) {
|
|
|
|
if (priv->cfg->ops->hcmd->set_pan_params(priv))
|
|
|
|
IWL_ERR(priv, "failed to update PAN params\n");
|
|
|
|
}
|
|
|
|
|
2010-10-22 23:04:30 +08:00
|
|
|
out:
|
2009-06-13 04:22:48 +08:00
|
|
|
mutex_unlock(&priv->mutex);
|
|
|
|
|
2009-04-23 22:13:26 +08:00
|
|
|
IWL_DEBUG_MAC80211(priv, "leave\n");
|
2009-04-09 02:26:35 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_bss_info_changed);
|
|
|
|
|
2010-02-17 20:54:08 +08:00
|
|
|
static int iwl_set_mode(struct iwl_priv *priv, struct ieee80211_vif *vif)
|
2009-04-09 02:26:45 +08:00
|
|
|
{
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx = iwl_rxon_ctx_from_vif(vif);
|
|
|
|
|
2010-08-23 16:46:58 +08:00
|
|
|
iwl_connection_init_rx_config(priv, ctx);
|
2009-04-09 02:26:45 +08:00
|
|
|
|
|
|
|
if (priv->cfg->ops->hcmd->set_rxon_chain)
|
2010-08-23 16:46:32 +08:00
|
|
|
priv->cfg->ops->hcmd->set_rxon_chain(priv, ctx);
|
2009-04-09 02:26:45 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
return iwlcore_commit_rxon(priv, ctx);
|
2009-04-09 02:26:45 +08:00
|
|
|
}
|
|
|
|
|
2010-02-17 20:54:08 +08:00
|
|
|
int iwl_mac_add_interface(struct ieee80211_hw *hw, struct ieee80211_vif *vif)
|
2009-04-09 02:26:46 +08:00
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_vif_priv *vif_priv = (void *)vif->drv_priv;
|
2010-08-23 16:46:58 +08:00
|
|
|
struct iwl_rxon_context *tmp, *ctx = NULL;
|
2010-01-21 03:21:06 +08:00
|
|
|
int err = 0;
|
2009-04-09 02:26:46 +08:00
|
|
|
|
2010-05-14 21:25:58 +08:00
|
|
|
IWL_DEBUG_MAC80211(priv, "enter: type %d, addr %pM\n",
|
|
|
|
vif->type, vif->addr);
|
2009-04-09 02:26:46 +08:00
|
|
|
|
2010-01-21 03:21:06 +08:00
|
|
|
mutex_lock(&priv->mutex);
|
|
|
|
|
2010-09-30 22:27:35 +08:00
|
|
|
if (!iwl_is_ready_rf(priv)) {
|
|
|
|
IWL_WARN(priv, "Try to add interface when device not ready\n");
|
2010-02-17 20:54:08 +08:00
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2010-08-23 16:46:58 +08:00
|
|
|
for_each_context(priv, tmp) {
|
|
|
|
u32 possible_modes =
|
|
|
|
tmp->interface_modes | tmp->exclusive_interface_modes;
|
|
|
|
|
|
|
|
if (tmp->vif) {
|
|
|
|
/* check if this busy context is exclusive */
|
|
|
|
if (tmp->exclusive_interface_modes &
|
|
|
|
BIT(tmp->vif->type)) {
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(possible_modes & BIT(vif->type)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* have maybe usable context w/o interface */
|
|
|
|
ctx = tmp;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ctx) {
|
2010-01-21 03:21:06 +08:00
|
|
|
err = -EOPNOTSUPP;
|
|
|
|
goto out;
|
2009-04-09 02:26:46 +08:00
|
|
|
}
|
|
|
|
|
2010-08-23 16:46:58 +08:00
|
|
|
vif_priv->ctx = ctx;
|
2010-08-23 16:46:40 +08:00
|
|
|
ctx->vif = vif;
|
2010-08-23 16:46:58 +08:00
|
|
|
/*
|
|
|
|
* This variable will be correct only when there's just
|
|
|
|
* a single context, but all code using it is for hardware
|
|
|
|
* that supports only one context.
|
|
|
|
*/
|
2009-12-23 20:15:45 +08:00
|
|
|
priv->iw_mode = vif->type;
|
2009-04-09 02:26:46 +08:00
|
|
|
|
2010-09-03 21:32:21 +08:00
|
|
|
ctx->is_active = true;
|
|
|
|
|
2010-02-17 20:54:08 +08:00
|
|
|
err = iwl_set_mode(priv, vif);
|
2010-09-03 21:32:21 +08:00
|
|
|
if (err) {
|
|
|
|
if (!ctx->always_active)
|
|
|
|
ctx->is_active = false;
|
2010-02-17 20:54:08 +08:00
|
|
|
goto out_err;
|
2010-09-03 21:32:21 +08:00
|
|
|
}
|
2010-02-19 14:58:32 +08:00
|
|
|
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->bt_params &&
|
|
|
|
priv->cfg->bt_params->advanced_bt_coexist &&
|
2010-08-23 22:57:00 +08:00
|
|
|
vif->type == NL80211_IFTYPE_ADHOC) {
|
|
|
|
/*
|
|
|
|
* pretend to have high BT traffic as long as we
|
|
|
|
* are operating in IBSS mode, as this will cause
|
|
|
|
* the rate scaling etc. to behave as intended.
|
|
|
|
*/
|
|
|
|
priv->bt_traffic_load = IWL_BT_COEX_TRAFFIC_LOAD_HIGH;
|
|
|
|
}
|
|
|
|
|
2010-02-17 20:54:08 +08:00
|
|
|
goto out;
|
2009-04-09 02:26:46 +08:00
|
|
|
|
2010-02-17 20:54:08 +08:00
|
|
|
out_err:
|
2010-08-23 16:46:40 +08:00
|
|
|
ctx->vif = NULL;
|
2010-02-17 20:54:08 +08:00
|
|
|
priv->iw_mode = NL80211_IFTYPE_STATION;
|
2010-01-21 03:21:06 +08:00
|
|
|
out:
|
2009-04-09 02:26:46 +08:00
|
|
|
mutex_unlock(&priv->mutex);
|
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave\n");
|
2010-01-21 03:21:06 +08:00
|
|
|
return err;
|
2009-04-09 02:26:46 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_mac_add_interface);
|
|
|
|
|
2009-04-09 02:26:47 +08:00
|
|
|
void iwl_mac_remove_interface(struct ieee80211_hw *hw,
|
2010-02-17 20:54:08 +08:00
|
|
|
struct ieee80211_vif *vif)
|
2009-04-09 02:26:47 +08:00
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx = iwl_rxon_ctx_from_vif(vif);
|
2009-04-09 02:26:47 +08:00
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "enter\n");
|
|
|
|
|
|
|
|
mutex_lock(&priv->mutex);
|
|
|
|
|
2010-08-23 16:46:58 +08:00
|
|
|
WARN_ON(ctx->vif != vif);
|
|
|
|
ctx->vif = NULL;
|
|
|
|
|
2010-09-13 20:46:41 +08:00
|
|
|
if (priv->scan_vif == vif) {
|
|
|
|
iwl_scan_cancel_timeout(priv, 200);
|
|
|
|
iwl_force_scan_end(priv);
|
|
|
|
}
|
2010-08-23 16:46:58 +08:00
|
|
|
iwl_set_mode(priv, vif);
|
2010-08-23 16:46:40 +08:00
|
|
|
|
2010-09-03 21:32:21 +08:00
|
|
|
if (!ctx->always_active)
|
|
|
|
ctx->is_active = false;
|
|
|
|
|
2010-08-23 22:57:00 +08:00
|
|
|
/*
|
|
|
|
* When removing the IBSS interface, overwrite the
|
|
|
|
* BT traffic load with the stored one from the last
|
|
|
|
* notification, if any. If this is a device that
|
|
|
|
* doesn't implement this, this has no effect since
|
|
|
|
* both values are the same and zero.
|
|
|
|
*/
|
|
|
|
if (vif->type == NL80211_IFTYPE_ADHOC)
|
|
|
|
priv->bt_traffic_load = priv->notif_bt_traffic_load;
|
|
|
|
|
2010-08-23 16:46:40 +08:00
|
|
|
memset(priv->bssid, 0, ETH_ALEN);
|
2009-04-09 02:26:47 +08:00
|
|
|
mutex_unlock(&priv->mutex);
|
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave\n");
|
|
|
|
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_mac_remove_interface);
|
|
|
|
|
2009-04-09 02:26:48 +08:00
|
|
|
/**
|
|
|
|
* iwl_mac_config - mac80211 config callback
|
|
|
|
*/
|
|
|
|
int iwl_mac_config(struct ieee80211_hw *hw, u32 changed)
|
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
|
|
|
const struct iwl_channel_info *ch_info;
|
|
|
|
struct ieee80211_conf *conf = &hw->conf;
|
2010-07-29 04:40:39 +08:00
|
|
|
struct ieee80211_channel *channel = conf->channel;
|
2009-09-12 01:38:15 +08:00
|
|
|
struct iwl_ht_config *ht_conf = &priv->current_ht_config;
|
2010-08-23 16:46:32 +08:00
|
|
|
struct iwl_rxon_context *ctx;
|
2009-04-09 02:26:48 +08:00
|
|
|
unsigned long flags = 0;
|
|
|
|
int ret = 0;
|
|
|
|
u16 ch;
|
|
|
|
int scan_active = 0;
|
|
|
|
|
|
|
|
mutex_lock(&priv->mutex);
|
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "enter to channel %d changed 0x%X\n",
|
2010-07-29 04:40:39 +08:00
|
|
|
channel->hw_value, changed);
|
2009-04-09 02:26:48 +08:00
|
|
|
|
|
|
|
if (unlikely(!priv->cfg->mod_params->disable_hw_scan &&
|
|
|
|
test_bit(STATUS_SCANNING, &priv->status))) {
|
|
|
|
scan_active = 1;
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave - scanning\n");
|
|
|
|
}
|
|
|
|
|
2009-12-11 06:37:27 +08:00
|
|
|
if (changed & (IEEE80211_CONF_CHANGE_SMPS |
|
|
|
|
IEEE80211_CONF_CHANGE_CHANNEL)) {
|
|
|
|
/* mac80211 uses static for non-HT which is what we want */
|
|
|
|
priv->current_ht_config.smps = conf->smps_mode;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Recalculate chain counts.
|
|
|
|
*
|
|
|
|
* If monitor mode is enabled then mac80211 will
|
|
|
|
* set up the SM PS mode to OFF if an HT channel is
|
|
|
|
* configured.
|
|
|
|
*/
|
|
|
|
if (priv->cfg->ops->hcmd->set_rxon_chain)
|
2010-08-23 16:46:32 +08:00
|
|
|
for_each_context(priv, ctx)
|
|
|
|
priv->cfg->ops->hcmd->set_rxon_chain(priv, ctx);
|
2009-12-11 06:37:27 +08:00
|
|
|
}
|
2009-04-09 02:26:48 +08:00
|
|
|
|
|
|
|
/* during scanning mac80211 will delay channel setting until
|
|
|
|
* scan finish with changed = 0
|
|
|
|
*/
|
|
|
|
if (!changed || (changed & IEEE80211_CONF_CHANGE_CHANNEL)) {
|
|
|
|
if (scan_active)
|
|
|
|
goto set_ch_out;
|
|
|
|
|
2010-07-29 04:40:39 +08:00
|
|
|
ch = channel->hw_value;
|
|
|
|
ch_info = iwl_get_channel_info(priv, channel->band, ch);
|
2009-04-09 02:26:48 +08:00
|
|
|
if (!is_channel_valid(ch_info)) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave - invalid channel\n");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto set_ch_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock_irqsave(&priv->lock, flags);
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
for_each_context(priv, ctx) {
|
2010-08-23 16:46:46 +08:00
|
|
|
/* Configure HT40 channels */
|
|
|
|
ctx->ht.enabled = conf_is_ht(conf);
|
|
|
|
if (ctx->ht.enabled) {
|
|
|
|
if (conf_is_ht40_minus(conf)) {
|
|
|
|
ctx->ht.extension_chan_offset =
|
|
|
|
IEEE80211_HT_PARAM_CHA_SEC_BELOW;
|
|
|
|
ctx->ht.is_40mhz = true;
|
|
|
|
} else if (conf_is_ht40_plus(conf)) {
|
|
|
|
ctx->ht.extension_chan_offset =
|
|
|
|
IEEE80211_HT_PARAM_CHA_SEC_ABOVE;
|
|
|
|
ctx->ht.is_40mhz = true;
|
|
|
|
} else {
|
|
|
|
ctx->ht.extension_chan_offset =
|
|
|
|
IEEE80211_HT_PARAM_CHA_SEC_NONE;
|
|
|
|
ctx->ht.is_40mhz = false;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
ctx->ht.is_40mhz = false;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Default to no protection. Protection mode will
|
|
|
|
* later be set from BSS config in iwl_ht_conf
|
|
|
|
*/
|
|
|
|
ctx->ht.protection = IEEE80211_HT_OP_MODE_PROTECTION_NONE;
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
/* if we are switching from ht to 2.4 clear flags
|
|
|
|
* from any ht related info since 2.4 does not
|
|
|
|
* support ht */
|
|
|
|
if ((le16_to_cpu(ctx->staging.channel) != ch))
|
|
|
|
ctx->staging.flags = 0;
|
2009-04-09 02:26:48 +08:00
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
iwl_set_rxon_channel(priv, channel, ctx);
|
|
|
|
iwl_set_rxon_ht(priv, ht_conf);
|
|
|
|
|
|
|
|
iwl_set_flags_for_band(priv, ctx, channel->band,
|
2010-08-23 16:46:40 +08:00
|
|
|
ctx->vif);
|
2010-08-23 16:46:32 +08:00
|
|
|
}
|
2009-04-09 02:26:48 +08:00
|
|
|
|
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
2010-05-06 23:54:11 +08:00
|
|
|
|
2010-08-27 23:53:46 +08:00
|
|
|
if (priv->cfg->ops->lib->update_bcast_stations)
|
|
|
|
ret = priv->cfg->ops->lib->update_bcast_stations(priv);
|
2010-06-15 04:10:29 +08:00
|
|
|
|
2009-04-09 02:26:48 +08:00
|
|
|
set_ch_out:
|
|
|
|
/* The list of supported rates and rate mask can be different
|
|
|
|
* for each band; since the band may have changed, reset
|
|
|
|
* the rate mask to what mac80211 lists */
|
|
|
|
iwl_set_rate(priv);
|
|
|
|
}
|
|
|
|
|
2009-09-26 05:24:27 +08:00
|
|
|
if (changed & (IEEE80211_CONF_CHANGE_PS |
|
|
|
|
IEEE80211_CONF_CHANGE_IDLE)) {
|
2009-08-08 06:41:51 +08:00
|
|
|
ret = iwl_power_update_mode(priv, false);
|
2009-04-09 02:26:48 +08:00
|
|
|
if (ret)
|
2009-08-08 06:41:51 +08:00
|
|
|
IWL_DEBUG_MAC80211(priv, "Error setting sleep level\n");
|
2009-04-09 02:26:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (changed & IEEE80211_CONF_CHANGE_POWER) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "TX Power old=%d new=%d\n",
|
|
|
|
priv->tx_power_user_lmt, conf->power_level);
|
|
|
|
|
2010-10-22 23:04:26 +08:00
|
|
|
ret = iwl_set_tx_power(priv, conf->power_level, false);
|
|
|
|
if (ret)
|
|
|
|
IWL_ERR(priv, "Error sending TX power (%d)\n", ret);
|
2009-04-09 02:26:48 +08:00
|
|
|
}
|
|
|
|
|
2009-04-21 05:37:04 +08:00
|
|
|
if (!iwl_is_ready(priv)) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave - not ready\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2009-04-09 02:26:48 +08:00
|
|
|
if (scan_active)
|
|
|
|
goto out;
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
for_each_context(priv, ctx) {
|
|
|
|
if (memcmp(&ctx->active, &ctx->staging, sizeof(ctx->staging)))
|
|
|
|
iwlcore_commit_rxon(priv, ctx);
|
|
|
|
else
|
|
|
|
IWL_DEBUG_INFO(priv,
|
|
|
|
"Not re-sending same RXON configuration.\n");
|
|
|
|
}
|
2009-04-09 02:26:48 +08:00
|
|
|
|
|
|
|
out:
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave\n");
|
|
|
|
mutex_unlock(&priv->mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_mac_config);
|
|
|
|
|
2009-04-09 02:26:52 +08:00
|
|
|
void iwl_mac_reset_tsf(struct ieee80211_hw *hw)
|
|
|
|
{
|
|
|
|
struct iwl_priv *priv = hw->priv;
|
|
|
|
unsigned long flags;
|
2010-08-23 16:46:32 +08:00
|
|
|
/* IBSS can only be the IWL_RXON_CTX_BSS context */
|
|
|
|
struct iwl_rxon_context *ctx = &priv->contexts[IWL_RXON_CTX_BSS];
|
2009-04-09 02:26:52 +08:00
|
|
|
|
|
|
|
mutex_lock(&priv->mutex);
|
|
|
|
IWL_DEBUG_MAC80211(priv, "enter\n");
|
|
|
|
|
|
|
|
spin_lock_irqsave(&priv->lock, flags);
|
2009-09-12 01:38:15 +08:00
|
|
|
memset(&priv->current_ht_config, 0, sizeof(struct iwl_ht_config));
|
2009-04-09 02:26:52 +08:00
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
|
|
|
|
|
|
|
spin_lock_irqsave(&priv->lock, flags);
|
|
|
|
|
|
|
|
/* new association get rid of ibss beacon skb */
|
2010-10-04 20:50:06 +08:00
|
|
|
if (priv->beacon_skb)
|
|
|
|
dev_kfree_skb(priv->beacon_skb);
|
2009-04-09 02:26:52 +08:00
|
|
|
|
2010-10-04 20:50:06 +08:00
|
|
|
priv->beacon_skb = NULL;
|
2009-04-09 02:26:52 +08:00
|
|
|
|
|
|
|
priv->timestamp = 0;
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&priv->lock, flags);
|
|
|
|
|
2010-09-13 20:46:39 +08:00
|
|
|
iwl_scan_cancel_timeout(priv, 100);
|
2009-04-09 02:26:52 +08:00
|
|
|
if (!iwl_is_ready_rf(priv)) {
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave - not ready\n");
|
|
|
|
mutex_unlock(&priv->mutex);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* we are restarting association process
|
|
|
|
* clear RXON_FILTER_ASSOC_MSK bit
|
|
|
|
*/
|
2010-08-23 16:46:32 +08:00
|
|
|
ctx->staging.filter_flags &= ~RXON_FILTER_ASSOC_MSK;
|
|
|
|
iwlcore_commit_rxon(priv, ctx);
|
2009-04-09 02:26:52 +08:00
|
|
|
|
|
|
|
iwl_set_rate(priv);
|
|
|
|
|
|
|
|
mutex_unlock(&priv->mutex);
|
|
|
|
|
|
|
|
IWL_DEBUG_MAC80211(priv, "leave\n");
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_mac_reset_tsf);
|
|
|
|
|
2009-10-10 04:20:28 +08:00
|
|
|
int iwl_alloc_txq_mem(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
if (!priv->txq)
|
|
|
|
priv->txq = kzalloc(
|
2010-10-06 23:10:00 +08:00
|
|
|
sizeof(struct iwl_tx_queue) *
|
|
|
|
priv->cfg->base_params->num_of_queues,
|
2009-10-10 04:20:28 +08:00
|
|
|
GFP_KERNEL);
|
|
|
|
if (!priv->txq) {
|
2010-03-25 05:19:58 +08:00
|
|
|
IWL_ERR(priv, "Not enough memory for txq\n");
|
2009-10-10 04:20:28 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_alloc_txq_mem);
|
|
|
|
|
|
|
|
void iwl_free_txq_mem(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
kfree(priv->txq);
|
|
|
|
priv->txq = NULL;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_free_txq_mem);
|
|
|
|
|
2009-08-08 06:41:39 +08:00
|
|
|
#ifdef CONFIG_IWLWIFI_DEBUGFS
|
|
|
|
|
|
|
|
#define IWL_TRAFFIC_DUMP_SIZE (IWL_TRAFFIC_ENTRY_SIZE * IWL_TRAFFIC_ENTRIES)
|
|
|
|
|
|
|
|
void iwl_reset_traffic_log(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
priv->tx_traffic_idx = 0;
|
|
|
|
priv->rx_traffic_idx = 0;
|
|
|
|
if (priv->tx_traffic)
|
|
|
|
memset(priv->tx_traffic, 0, IWL_TRAFFIC_DUMP_SIZE);
|
|
|
|
if (priv->rx_traffic)
|
|
|
|
memset(priv->rx_traffic, 0, IWL_TRAFFIC_DUMP_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
int iwl_alloc_traffic_mem(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
u32 traffic_size = IWL_TRAFFIC_DUMP_SIZE;
|
|
|
|
|
|
|
|
if (iwl_debug_level & IWL_DL_TX) {
|
|
|
|
if (!priv->tx_traffic) {
|
|
|
|
priv->tx_traffic =
|
|
|
|
kzalloc(traffic_size, GFP_KERNEL);
|
|
|
|
if (!priv->tx_traffic)
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (iwl_debug_level & IWL_DL_RX) {
|
|
|
|
if (!priv->rx_traffic) {
|
|
|
|
priv->rx_traffic =
|
|
|
|
kzalloc(traffic_size, GFP_KERNEL);
|
|
|
|
if (!priv->rx_traffic)
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
iwl_reset_traffic_log(priv);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_alloc_traffic_mem);
|
|
|
|
|
|
|
|
void iwl_free_traffic_mem(struct iwl_priv *priv)
|
|
|
|
{
|
|
|
|
kfree(priv->tx_traffic);
|
|
|
|
priv->tx_traffic = NULL;
|
|
|
|
|
|
|
|
kfree(priv->rx_traffic);
|
|
|
|
priv->rx_traffic = NULL;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_free_traffic_mem);
|
|
|
|
|
|
|
|
void iwl_dbg_log_tx_data_frame(struct iwl_priv *priv,
|
|
|
|
u16 length, struct ieee80211_hdr *header)
|
|
|
|
{
|
|
|
|
__le16 fc;
|
|
|
|
u16 len;
|
|
|
|
|
|
|
|
if (likely(!(iwl_debug_level & IWL_DL_TX)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!priv->tx_traffic)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fc = header->frame_control;
|
|
|
|
if (ieee80211_is_data(fc)) {
|
|
|
|
len = (length > IWL_TRAFFIC_ENTRY_SIZE)
|
|
|
|
? IWL_TRAFFIC_ENTRY_SIZE : length;
|
|
|
|
memcpy((priv->tx_traffic +
|
|
|
|
(priv->tx_traffic_idx * IWL_TRAFFIC_ENTRY_SIZE)),
|
|
|
|
header, len);
|
|
|
|
priv->tx_traffic_idx =
|
|
|
|
(priv->tx_traffic_idx + 1) % IWL_TRAFFIC_ENTRIES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_dbg_log_tx_data_frame);
|
|
|
|
|
|
|
|
void iwl_dbg_log_rx_data_frame(struct iwl_priv *priv,
|
|
|
|
u16 length, struct ieee80211_hdr *header)
|
|
|
|
{
|
|
|
|
__le16 fc;
|
|
|
|
u16 len;
|
|
|
|
|
|
|
|
if (likely(!(iwl_debug_level & IWL_DL_RX)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!priv->rx_traffic)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fc = header->frame_control;
|
|
|
|
if (ieee80211_is_data(fc)) {
|
|
|
|
len = (length > IWL_TRAFFIC_ENTRY_SIZE)
|
|
|
|
? IWL_TRAFFIC_ENTRY_SIZE : length;
|
|
|
|
memcpy((priv->rx_traffic +
|
|
|
|
(priv->rx_traffic_idx * IWL_TRAFFIC_ENTRY_SIZE)),
|
|
|
|
header, len);
|
|
|
|
priv->rx_traffic_idx =
|
|
|
|
(priv->rx_traffic_idx + 1) % IWL_TRAFFIC_ENTRIES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_dbg_log_rx_data_frame);
|
2009-08-08 06:41:40 +08:00
|
|
|
|
|
|
|
const char *get_mgmt_string(int cmd)
|
|
|
|
{
|
|
|
|
switch (cmd) {
|
|
|
|
IWL_CMD(MANAGEMENT_ASSOC_REQ);
|
|
|
|
IWL_CMD(MANAGEMENT_ASSOC_RESP);
|
|
|
|
IWL_CMD(MANAGEMENT_REASSOC_REQ);
|
|
|
|
IWL_CMD(MANAGEMENT_REASSOC_RESP);
|
|
|
|
IWL_CMD(MANAGEMENT_PROBE_REQ);
|
|
|
|
IWL_CMD(MANAGEMENT_PROBE_RESP);
|
|
|
|
IWL_CMD(MANAGEMENT_BEACON);
|
|
|
|
IWL_CMD(MANAGEMENT_ATIM);
|
|
|
|
IWL_CMD(MANAGEMENT_DISASSOC);
|
|
|
|
IWL_CMD(MANAGEMENT_AUTH);
|
|
|
|
IWL_CMD(MANAGEMENT_DEAUTH);
|
|
|
|
IWL_CMD(MANAGEMENT_ACTION);
|
|
|
|
default:
|
|
|
|
return "UNKNOWN";
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *get_ctrl_string(int cmd)
|
|
|
|
{
|
|
|
|
switch (cmd) {
|
|
|
|
IWL_CMD(CONTROL_BACK_REQ);
|
|
|
|
IWL_CMD(CONTROL_BACK);
|
|
|
|
IWL_CMD(CONTROL_PSPOLL);
|
|
|
|
IWL_CMD(CONTROL_RTS);
|
|
|
|
IWL_CMD(CONTROL_CTS);
|
|
|
|
IWL_CMD(CONTROL_ACK);
|
|
|
|
IWL_CMD(CONTROL_CFEND);
|
|
|
|
IWL_CMD(CONTROL_CFENDACK);
|
|
|
|
default:
|
|
|
|
return "UNKNOWN";
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-21 04:04:56 +08:00
|
|
|
void iwl_clear_traffic_stats(struct iwl_priv *priv)
|
2009-08-08 06:41:40 +08:00
|
|
|
{
|
|
|
|
memset(&priv->tx_stats, 0, sizeof(struct traffic_stats));
|
|
|
|
memset(&priv->rx_stats, 0, sizeof(struct traffic_stats));
|
2009-11-21 04:04:56 +08:00
|
|
|
priv->led_tpt = 0;
|
2009-08-08 06:41:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* if CONFIG_IWLWIFI_DEBUGFS defined, iwl_update_stats function will
|
|
|
|
* record all the MGMT, CTRL and DATA pkt for both TX and Rx pass.
|
|
|
|
* Use debugFs to display the rx/rx_statistics
|
|
|
|
* if CONFIG_IWLWIFI_DEBUGFS not being defined, then no MGMT and CTRL
|
|
|
|
* information will be recorded, but DATA pkt still will be recorded
|
|
|
|
* for the reason of iwl_led.c need to control the led blinking based on
|
|
|
|
* number of tx and rx data.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void iwl_update_stats(struct iwl_priv *priv, bool is_tx, __le16 fc, u16 len)
|
|
|
|
{
|
|
|
|
struct traffic_stats *stats;
|
|
|
|
|
|
|
|
if (is_tx)
|
|
|
|
stats = &priv->tx_stats;
|
|
|
|
else
|
|
|
|
stats = &priv->rx_stats;
|
|
|
|
|
|
|
|
if (ieee80211_is_mgmt(fc)) {
|
|
|
|
switch (fc & cpu_to_le16(IEEE80211_FCTL_STYPE)) {
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_ASSOC_REQ):
|
|
|
|
stats->mgmt[MANAGEMENT_ASSOC_REQ]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_ASSOC_RESP):
|
|
|
|
stats->mgmt[MANAGEMENT_ASSOC_RESP]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_REASSOC_REQ):
|
|
|
|
stats->mgmt[MANAGEMENT_REASSOC_REQ]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_REASSOC_RESP):
|
|
|
|
stats->mgmt[MANAGEMENT_REASSOC_RESP]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_PROBE_REQ):
|
|
|
|
stats->mgmt[MANAGEMENT_PROBE_REQ]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_PROBE_RESP):
|
|
|
|
stats->mgmt[MANAGEMENT_PROBE_RESP]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_BEACON):
|
|
|
|
stats->mgmt[MANAGEMENT_BEACON]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_ATIM):
|
|
|
|
stats->mgmt[MANAGEMENT_ATIM]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_DISASSOC):
|
|
|
|
stats->mgmt[MANAGEMENT_DISASSOC]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_AUTH):
|
|
|
|
stats->mgmt[MANAGEMENT_AUTH]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_DEAUTH):
|
|
|
|
stats->mgmt[MANAGEMENT_DEAUTH]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_ACTION):
|
|
|
|
stats->mgmt[MANAGEMENT_ACTION]++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (ieee80211_is_ctl(fc)) {
|
|
|
|
switch (fc & cpu_to_le16(IEEE80211_FCTL_STYPE)) {
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_BACK_REQ):
|
|
|
|
stats->ctrl[CONTROL_BACK_REQ]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_BACK):
|
|
|
|
stats->ctrl[CONTROL_BACK]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_PSPOLL):
|
|
|
|
stats->ctrl[CONTROL_PSPOLL]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_RTS):
|
|
|
|
stats->ctrl[CONTROL_RTS]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_CTS):
|
|
|
|
stats->ctrl[CONTROL_CTS]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_ACK):
|
|
|
|
stats->ctrl[CONTROL_ACK]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_CFEND):
|
|
|
|
stats->ctrl[CONTROL_CFEND]++;
|
|
|
|
break;
|
|
|
|
case cpu_to_le16(IEEE80211_STYPE_CFENDACK):
|
|
|
|
stats->ctrl[CONTROL_CFENDACK]++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* data */
|
|
|
|
stats->data_cnt++;
|
|
|
|
stats->data_bytes += len;
|
|
|
|
}
|
2009-11-21 04:04:54 +08:00
|
|
|
iwl_leds_background(priv);
|
2009-08-08 06:41:40 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_update_stats);
|
2009-08-08 06:41:39 +08:00
|
|
|
#endif
|
|
|
|
|
2010-02-04 03:47:19 +08:00
|
|
|
static void iwl_force_rf_reset(struct iwl_priv *priv)
|
2010-01-23 06:22:43 +08:00
|
|
|
{
|
|
|
|
if (test_bit(STATUS_EXIT_PENDING, &priv->status))
|
|
|
|
return;
|
|
|
|
|
2010-08-23 16:46:32 +08:00
|
|
|
if (!iwl_is_any_associated(priv)) {
|
2010-01-23 06:22:43 +08:00
|
|
|
IWL_DEBUG_SCAN(priv, "force reset rejected: not associated\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* There is no easy and better way to force reset the radio,
|
|
|
|
* the only known method is switching channel which will force to
|
|
|
|
* reset and tune the radio.
|
|
|
|
* Use internal short scan (single channel) operation to should
|
|
|
|
* achieve this objective.
|
|
|
|
* Driver should reset the radio when number of consecutive missed
|
|
|
|
* beacon, or any other uCode error condition detected.
|
|
|
|
*/
|
|
|
|
IWL_DEBUG_INFO(priv, "perform radio reset.\n");
|
|
|
|
iwl_internal_short_hw_scan(priv);
|
|
|
|
}
|
2010-02-04 03:47:19 +08:00
|
|
|
|
|
|
|
|
2010-06-22 07:52:55 +08:00
|
|
|
int iwl_force_reset(struct iwl_priv *priv, int mode, bool external)
|
2010-02-04 03:47:19 +08:00
|
|
|
{
|
2010-02-19 14:03:06 +08:00
|
|
|
struct iwl_force_reset *force_reset;
|
|
|
|
|
2010-02-04 03:47:19 +08:00
|
|
|
if (test_bit(STATUS_EXIT_PENDING, &priv->status))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2010-02-19 14:03:06 +08:00
|
|
|
if (mode >= IWL_MAX_FORCE_RESET) {
|
|
|
|
IWL_DEBUG_INFO(priv, "invalid reset request.\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
force_reset = &priv->force_reset[mode];
|
|
|
|
force_reset->reset_request_count++;
|
2010-06-22 07:52:55 +08:00
|
|
|
if (!external) {
|
|
|
|
if (force_reset->last_force_reset_jiffies &&
|
|
|
|
time_after(force_reset->last_force_reset_jiffies +
|
|
|
|
force_reset->reset_duration, jiffies)) {
|
|
|
|
IWL_DEBUG_INFO(priv, "force reset rejected\n");
|
|
|
|
force_reset->reset_reject_count++;
|
|
|
|
return -EAGAIN;
|
|
|
|
}
|
2010-02-04 03:47:19 +08:00
|
|
|
}
|
2010-02-19 14:03:06 +08:00
|
|
|
force_reset->reset_success_count++;
|
|
|
|
force_reset->last_force_reset_jiffies = jiffies;
|
2010-02-04 03:47:19 +08:00
|
|
|
IWL_DEBUG_INFO(priv, "perform force reset (%d)\n", mode);
|
|
|
|
switch (mode) {
|
|
|
|
case IWL_RF_RESET:
|
|
|
|
iwl_force_rf_reset(priv);
|
|
|
|
break;
|
|
|
|
case IWL_FW_RESET:
|
2010-06-22 07:52:55 +08:00
|
|
|
/*
|
|
|
|
* if the request is from external(ex: debugfs),
|
|
|
|
* then always perform the request in regardless the module
|
|
|
|
* parameter setting
|
|
|
|
* if the request is from internal (uCode error or driver
|
|
|
|
* detect failure), then fw_restart module parameter
|
|
|
|
* need to be check before performing firmware reload
|
|
|
|
*/
|
|
|
|
if (!external && !priv->cfg->mod_params->restart_fw) {
|
|
|
|
IWL_DEBUG_INFO(priv, "Cancel firmware reload based on "
|
|
|
|
"module parameter setting\n");
|
|
|
|
break;
|
|
|
|
}
|
2010-02-04 03:47:19 +08:00
|
|
|
IWL_ERR(priv, "On demand firmware reload\n");
|
|
|
|
/* Set the FW error flag -- cleared on iwl_down */
|
|
|
|
set_bit(STATUS_FW_ERROR, &priv->status);
|
|
|
|
wake_up_interruptible(&priv->wait_command_queue);
|
|
|
|
/*
|
|
|
|
* Keep the restart process from trying to send host
|
|
|
|
* commands by clearing the INIT status bit
|
|
|
|
*/
|
|
|
|
clear_bit(STATUS_READY, &priv->status);
|
|
|
|
queue_work(priv->workqueue, &priv->restart);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2010-03-02 09:23:50 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* iwl_bg_monitor_recover - Timer callback to check for stuck queue and recover
|
|
|
|
*
|
|
|
|
* During normal condition (no queue is stuck), the timer is continually set to
|
|
|
|
* execute every monitor_recover_period milliseconds after the last timer
|
|
|
|
* expired. When the queue read_ptr is at the same place, the timer is
|
|
|
|
* shorten to 100mSecs. This is
|
|
|
|
* 1) to reduce the chance that the read_ptr may wrap around (not stuck)
|
|
|
|
* 2) to detect the stuck queues quicker before the station and AP can
|
|
|
|
* disassociate each other.
|
|
|
|
*
|
|
|
|
* This function monitors all the tx queues and recover from it if any
|
|
|
|
* of the queues are stuck.
|
|
|
|
* 1. It first check the cmd queue for stuck conditions. If it is stuck,
|
|
|
|
* it will recover by resetting the firmware and return.
|
|
|
|
* 2. Then, it checks for station association. If it associates it will check
|
|
|
|
* other queues. If any queue is stuck, it will recover by resetting
|
|
|
|
* the firmware.
|
|
|
|
* Note: It the number of times the queue read_ptr to be at the same place to
|
|
|
|
* be MAX_REPEAT+1 in order to consider to be stuck.
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* The maximum number of times the read pointer of the tx queue at the
|
|
|
|
* same place without considering to be stuck.
|
|
|
|
*/
|
|
|
|
#define MAX_REPEAT (2)
|
|
|
|
static int iwl_check_stuck_queue(struct iwl_priv *priv, int cnt)
|
|
|
|
{
|
|
|
|
struct iwl_tx_queue *txq;
|
|
|
|
struct iwl_queue *q;
|
|
|
|
|
|
|
|
txq = &priv->txq[cnt];
|
|
|
|
q = &txq->q;
|
|
|
|
/* queue is empty, skip */
|
2010-10-06 23:10:00 +08:00
|
|
|
if (q->read_ptr == q->write_ptr)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (q->read_ptr == q->last_read_ptr) {
|
|
|
|
/* a queue has not been read from last time */
|
|
|
|
if (q->repeat_same_read_ptr > MAX_REPEAT) {
|
|
|
|
IWL_ERR(priv,
|
|
|
|
"queue %d stuck %d time. Fw reload.\n",
|
|
|
|
q->id, q->repeat_same_read_ptr);
|
2010-03-02 09:23:50 +08:00
|
|
|
q->repeat_same_read_ptr = 0;
|
2010-10-06 23:10:00 +08:00
|
|
|
iwl_force_reset(priv, IWL_FW_RESET, false);
|
|
|
|
} else {
|
|
|
|
q->repeat_same_read_ptr++;
|
|
|
|
IWL_DEBUG_RADIO(priv,
|
|
|
|
"queue %d, not read %d time\n",
|
|
|
|
q->id,
|
|
|
|
q->repeat_same_read_ptr);
|
2010-09-24 06:24:22 +08:00
|
|
|
mod_timer(&priv->monitor_recover,
|
|
|
|
jiffies + msecs_to_jiffies(
|
|
|
|
IWL_ONE_HUNDRED_MSECS));
|
|
|
|
return 1;
|
2010-03-02 09:23:50 +08:00
|
|
|
}
|
2010-10-06 23:10:00 +08:00
|
|
|
} else {
|
|
|
|
q->last_read_ptr = q->read_ptr;
|
|
|
|
q->repeat_same_read_ptr = 0;
|
2010-03-02 09:23:50 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void iwl_bg_monitor_recover(unsigned long data)
|
|
|
|
{
|
|
|
|
struct iwl_priv *priv = (struct iwl_priv *)data;
|
|
|
|
int cnt;
|
|
|
|
|
|
|
|
if (test_bit(STATUS_EXIT_PENDING, &priv->status))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* monitor and check for stuck cmd queue */
|
2010-08-23 16:46:33 +08:00
|
|
|
if (iwl_check_stuck_queue(priv, priv->cmd_queue))
|
2010-03-02 09:23:50 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* monitor and check for other stuck queues */
|
2010-08-23 16:46:32 +08:00
|
|
|
if (iwl_is_any_associated(priv)) {
|
2010-03-02 09:23:50 +08:00
|
|
|
for (cnt = 0; cnt < priv->hw_params.max_txq_num; cnt++) {
|
|
|
|
/* skip as we already checked the command queue */
|
2010-08-23 16:46:33 +08:00
|
|
|
if (cnt == priv->cmd_queue)
|
2010-03-02 09:23:50 +08:00
|
|
|
continue;
|
|
|
|
if (iwl_check_stuck_queue(priv, cnt))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2010-10-06 23:10:00 +08:00
|
|
|
if (priv->cfg->base_params->monitor_recover_period) {
|
2010-08-23 22:57:07 +08:00
|
|
|
/*
|
|
|
|
* Reschedule the timer to occur in
|
2010-10-06 23:10:00 +08:00
|
|
|
* priv->cfg->base_params->monitor_recover_period
|
2010-08-23 22:57:07 +08:00
|
|
|
*/
|
|
|
|
mod_timer(&priv->monitor_recover, jiffies + msecs_to_jiffies(
|
2010-10-06 23:10:00 +08:00
|
|
|
priv->cfg->base_params->monitor_recover_period));
|
2010-08-23 22:57:07 +08:00
|
|
|
}
|
2010-03-02 09:23:50 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_bg_monitor_recover);
|
2010-01-23 06:22:43 +08:00
|
|
|
|
2010-05-06 23:54:10 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* extended beacon time format
|
|
|
|
* time in usec will be changed into a 32-bit value in extended:internal format
|
|
|
|
* the extended part is the beacon counts
|
|
|
|
* the internal part is the time in usec within one beacon interval
|
|
|
|
*/
|
|
|
|
u32 iwl_usecs_to_beacons(struct iwl_priv *priv, u32 usec, u32 beacon_interval)
|
|
|
|
{
|
|
|
|
u32 quot;
|
|
|
|
u32 rem;
|
|
|
|
u32 interval = beacon_interval * TIME_UNIT;
|
|
|
|
|
|
|
|
if (!interval || !usec)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
quot = (usec / interval) &
|
|
|
|
(iwl_beacon_time_mask_high(priv,
|
|
|
|
priv->hw_params.beacon_time_tsf_bits) >>
|
|
|
|
priv->hw_params.beacon_time_tsf_bits);
|
|
|
|
rem = (usec % interval) & iwl_beacon_time_mask_low(priv,
|
|
|
|
priv->hw_params.beacon_time_tsf_bits);
|
|
|
|
|
|
|
|
return (quot << priv->hw_params.beacon_time_tsf_bits) + rem;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_usecs_to_beacons);
|
|
|
|
|
|
|
|
/* base is usually what we get from ucode with each received frame,
|
|
|
|
* the same as HW timer counter counting down
|
|
|
|
*/
|
|
|
|
__le32 iwl_add_beacon_time(struct iwl_priv *priv, u32 base,
|
|
|
|
u32 addon, u32 beacon_interval)
|
|
|
|
{
|
|
|
|
u32 base_low = base & iwl_beacon_time_mask_low(priv,
|
|
|
|
priv->hw_params.beacon_time_tsf_bits);
|
|
|
|
u32 addon_low = addon & iwl_beacon_time_mask_low(priv,
|
|
|
|
priv->hw_params.beacon_time_tsf_bits);
|
|
|
|
u32 interval = beacon_interval * TIME_UNIT;
|
|
|
|
u32 res = (base & iwl_beacon_time_mask_high(priv,
|
|
|
|
priv->hw_params.beacon_time_tsf_bits)) +
|
|
|
|
(addon & iwl_beacon_time_mask_high(priv,
|
|
|
|
priv->hw_params.beacon_time_tsf_bits));
|
|
|
|
|
|
|
|
if (base_low > addon_low)
|
|
|
|
res += base_low - addon_low;
|
|
|
|
else if (base_low < addon_low) {
|
|
|
|
res += interval + base_low - addon_low;
|
|
|
|
res += (1 << priv->hw_params.beacon_time_tsf_bits);
|
|
|
|
} else
|
|
|
|
res += (1 << priv->hw_params.beacon_time_tsf_bits);
|
|
|
|
|
|
|
|
return cpu_to_le32(res);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_add_beacon_time);
|
|
|
|
|
2009-03-27 01:14:08 +08:00
|
|
|
#ifdef CONFIG_PM
|
|
|
|
|
2010-10-26 04:12:37 +08:00
|
|
|
int iwl_pci_suspend(struct device *device)
|
2009-03-27 01:14:08 +08:00
|
|
|
{
|
2010-10-26 04:12:37 +08:00
|
|
|
struct pci_dev *pdev = to_pci_dev(device);
|
2009-03-27 01:14:08 +08:00
|
|
|
struct iwl_priv *priv = pci_get_drvdata(pdev);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function is called when system goes into suspend state
|
|
|
|
* mac80211 will call iwl_mac_stop() from the mac80211 suspend function
|
|
|
|
* first but since iwl_mac_stop() has no knowledge of who the caller is,
|
|
|
|
* it will not call apm_ops.stop() to stop the DMA operation.
|
|
|
|
* Calling apm_ops.stop here to make sure we stop the DMA.
|
|
|
|
*/
|
2010-09-23 00:02:10 +08:00
|
|
|
iwl_apm_stop(priv);
|
2009-03-27 01:14:08 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_pci_suspend);
|
|
|
|
|
2010-10-26 04:12:37 +08:00
|
|
|
int iwl_pci_resume(struct device *device)
|
2009-03-27 01:14:08 +08:00
|
|
|
{
|
2010-10-26 04:12:37 +08:00
|
|
|
struct pci_dev *pdev = to_pci_dev(device);
|
2009-03-27 01:14:08 +08:00
|
|
|
struct iwl_priv *priv = pci_get_drvdata(pdev);
|
2010-06-18 16:38:56 +08:00
|
|
|
bool hw_rfkill = false;
|
2009-03-27 01:14:08 +08:00
|
|
|
|
2010-04-15 01:27:00 +08:00
|
|
|
/*
|
|
|
|
* We disable the RETRY_TIMEOUT register (0x41) to keep
|
|
|
|
* PCI Tx retries from interfering with C3 CPU state.
|
|
|
|
*/
|
|
|
|
pci_write_config_byte(pdev, PCI_CFG_RETRY_TIMEOUT, 0x00);
|
|
|
|
|
2009-03-27 01:14:08 +08:00
|
|
|
iwl_enable_interrupts(priv);
|
|
|
|
|
2010-06-18 16:38:56 +08:00
|
|
|
if (!(iwl_read32(priv, CSR_GP_CNTRL) &
|
|
|
|
CSR_GP_CNTRL_REG_FLAG_HW_RF_KILL_SW))
|
|
|
|
hw_rfkill = true;
|
|
|
|
|
|
|
|
if (hw_rfkill)
|
|
|
|
set_bit(STATUS_RF_KILL_HW, &priv->status);
|
|
|
|
else
|
|
|
|
clear_bit(STATUS_RF_KILL_HW, &priv->status);
|
|
|
|
|
|
|
|
wiphy_rfkill_set_hw_state(priv->hw->wiphy, hw_rfkill);
|
|
|
|
|
2009-03-27 01:14:08 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iwl_pci_resume);
|
|
|
|
|
2010-10-26 04:12:37 +08:00
|
|
|
const struct dev_pm_ops iwl_pm_ops = {
|
|
|
|
.suspend = iwl_pci_suspend,
|
|
|
|
.resume = iwl_pci_resume,
|
|
|
|
.freeze = iwl_pci_suspend,
|
|
|
|
.thaw = iwl_pci_resume,
|
|
|
|
.poweroff = iwl_pci_suspend,
|
|
|
|
.restore = iwl_pci_resume,
|
|
|
|
};
|
|
|
|
EXPORT_SYMBOL(iwl_pm_ops);
|
|
|
|
|
2009-03-27 01:14:08 +08:00
|
|
|
#endif /* CONFIG_PM */
|