iwl-5000.c 48.0 KB
Newer Older
1 2
/******************************************************************************
 *
3
 * Copyright(c) 2007 - 2009 Intel Corporation. All rights reserved.
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
 *
 * 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.
 *
 * Contact Information:
 * Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
 *
 *****************************************************************************/

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/pci.h>
#include <linux/dma-mapping.h>
#include <linux/delay.h>
#include <linux/skbuff.h>
#include <linux/netdevice.h>
#include <linux/wireless.h>
#include <net/mac80211.h>
#include <linux/etherdevice.h>
#include <asm/unaligned.h>

#include "iwl-eeprom.h"
40
#include "iwl-dev.h"
41 42
#include "iwl-core.h"
#include "iwl-io.h"
43
#include "iwl-sta.h"
44 45
#include "iwl-helpers.h"
#include "iwl-5000-hw.h"
46
#include "iwl-6000-hw.h"
47

48 49 50
/* Highest firmware API version supported */
#define IWL5000_UCODE_API_MAX 1
#define IWL5150_UCODE_API_MAX 1
51

52 53 54 55 56 57 58 59 60 61 62
/* Lowest firmware API version supported */
#define IWL5000_UCODE_API_MIN 1
#define IWL5150_UCODE_API_MIN 1

#define IWL5000_FW_PRE "iwlwifi-5000-"
#define _IWL5000_MODULE_FIRMWARE(api) IWL5000_FW_PRE #api ".ucode"
#define IWL5000_MODULE_FIRMWARE(api) _IWL5000_MODULE_FIRMWARE(api)

#define IWL5150_FW_PRE "iwlwifi-5150-"
#define _IWL5150_MODULE_FIRMWARE(api) IWL5150_FW_PRE #api ".ucode"
#define IWL5150_MODULE_FIRMWARE(api) _IWL5150_MODULE_FIRMWARE(api)
63

64 65 66 67 68 69 70 71 72 73
static const u16 iwl5000_default_queue_to_tx_fifo[] = {
	IWL_TX_FIFO_AC3,
	IWL_TX_FIFO_AC2,
	IWL_TX_FIFO_AC1,
	IWL_TX_FIFO_AC0,
	IWL50_CMD_FIFO_NUM,
	IWL_TX_FIFO_HCCA_1,
	IWL_TX_FIFO_HCCA_2
};

74 75 76 77 78 79 80 81 82 83
/* FIXME: same implementation as 4965 */
static int iwl5000_apm_stop_master(struct iwl_priv *priv)
{
	unsigned long flags;

	spin_lock_irqsave(&priv->lock, flags);

	/* set stop master bit */
	iwl_set_bit(priv, CSR_RESET, CSR_RESET_REG_FLAG_STOP_MASTER);

W
Wu Fengguang 已提交
84
	iwl_poll_direct_bit(priv, CSR_RESET,
85 86 87
				  CSR_RESET_REG_FLAG_MASTER_DISABLED, 100);

	spin_unlock_irqrestore(&priv->lock, flags);
88
	IWL_DEBUG_INFO(priv, "stop master\n");
89

W
Wu Fengguang 已提交
90
	return 0;
91 92 93
}


94 95 96 97 98 99 100
static int iwl5000_apm_init(struct iwl_priv *priv)
{
	int ret = 0;

	iwl_set_bit(priv, CSR_GIO_CHICKEN_BITS,
		    CSR_GIO_CHICKEN_BITS_REG_BIT_DIS_L0S_EXIT_TIMER);

101 102 103 104
	/* disable L0s without affecting L1 :don't wait for ICH L0s bug W/A) */
	iwl_set_bit(priv, CSR_GIO_CHICKEN_BITS,
		    CSR_GIO_CHICKEN_BITS_REG_BIT_L1A_NO_L0S_RX);

T
Tomas Winkler 已提交
105
	/* Set FH wait threshold to maximum (HW error during stress W/A) */
T
Tomas Winkler 已提交
106 107 108 109 110 111
	iwl_set_bit(priv, CSR_DBG_HPET_MEM_REG, CSR_DBG_HPET_MEM_REG_VAL);

	/* enable HAP INTA to move device L1a -> L0s */
	iwl_set_bit(priv, CSR_HW_IF_CONFIG_REG,
		    CSR_HW_IF_CONFIG_REG_BIT_HAP_WAKE_L1A);

112 113
	if (priv->cfg->need_pll_cfg)
		iwl_set_bit(priv, CSR_ANA_PLL_CFG, CSR50_ANA_PLL_CFG_VAL);
114 115 116 117 118 119

	/* set "initialization complete" bit to move adapter
	 * D0U* --> D0A* state */
	iwl_set_bit(priv, CSR_GP_CNTRL, CSR_GP_CNTRL_REG_FLAG_INIT_DONE);

	/* wait for clock stabilization */
Z
Zhu, Yi 已提交
120 121
	ret = iwl_poll_direct_bit(priv, CSR_GP_CNTRL,
			CSR_GP_CNTRL_REG_FLAG_MAC_CLOCK_READY, 25000);
122
	if (ret < 0) {
123
		IWL_DEBUG_INFO(priv, "Failed to init the card\n");
124 125 126 127 128 129 130 131
		return ret;
	}

	ret = iwl_grab_nic_access(priv);
	if (ret)
		return ret;

	/* enable DMA */
132
	iwl_write_prph(priv, APMG_CLK_EN_REG, APMG_CLK_VAL_DMA_CLK_RQT);
133 134 135

	udelay(20);

136
	/* disable L1-Active */
137
	iwl_set_bits_prph(priv, APMG_PCIDEV_STT_REG,
138
			  APMG_PCIDEV_STT_VAL_L1_ACT_DIS);
139 140 141 142 143 144

	iwl_release_nic_access(priv);

	return ret;
}

T
Tomas Winkler 已提交
145
/* FIXME: this is identical to 4965 */
146 147 148 149
static void iwl5000_apm_stop(struct iwl_priv *priv)
{
	unsigned long flags;

150
	iwl5000_apm_stop_master(priv);
151 152 153 154 155 156 157

	spin_lock_irqsave(&priv->lock, flags);

	iwl_set_bit(priv, CSR_RESET, CSR_RESET_REG_FLAG_SW_RESET);

	udelay(10);

158 159
	/* clear "init complete"  move adapter D0A* --> D0U state */
	iwl_clear_bit(priv, CSR_GP_CNTRL, CSR_GP_CNTRL_REG_FLAG_INIT_DONE);
160 161 162 163 164

	spin_unlock_irqrestore(&priv->lock, flags);
}


165 166 167 168 169
static int iwl5000_apm_reset(struct iwl_priv *priv)
{
	int ret = 0;
	unsigned long flags;

170
	iwl5000_apm_stop_master(priv);
171 172 173 174 175 176 177 178 179 180

	spin_lock_irqsave(&priv->lock, flags);

	iwl_set_bit(priv, CSR_RESET, CSR_RESET_REG_FLAG_SW_RESET);

	udelay(10);


	/* FIXME: put here L1A -L0S w/a */

181 182
	if (priv->cfg->need_pll_cfg)
		iwl_set_bit(priv, CSR_ANA_PLL_CFG, CSR50_ANA_PLL_CFG_VAL);
183 184 185 186 187 188

	/* set "initialization complete" bit to move adapter
	 * D0U* --> D0A* state */
	iwl_set_bit(priv, CSR_GP_CNTRL, CSR_GP_CNTRL_REG_FLAG_INIT_DONE);

	/* wait for clock stabilization */
Z
Zhu, Yi 已提交
189 190
	ret = iwl_poll_direct_bit(priv, CSR_GP_CNTRL,
			CSR_GP_CNTRL_REG_FLAG_MAC_CLOCK_READY, 25000);
191
	if (ret < 0) {
192
		IWL_DEBUG_INFO(priv, "Failed to init the card\n");
193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217
		goto out;
	}

	ret = iwl_grab_nic_access(priv);
	if (ret)
		goto out;

	/* enable DMA */
	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);

	iwl_release_nic_access(priv);

out:
	spin_unlock_irqrestore(&priv->lock, flags);

	return ret;
}


218
static void iwl5000_nic_config(struct iwl_priv *priv)
219 220 221
{
	unsigned long flags;
	u16 radio_cfg;
222
	u16 link;
223 224 225

	spin_lock_irqsave(&priv->lock, flags);

226
	pci_read_config_word(priv->pci_dev, PCI_CFG_LINK_CTRL, &link);
227

228
	/* L1 is enabled by BIOS */
229
	if ((link & PCI_CFG_LINK_CTRL_VAL_L1_EN) == PCI_CFG_LINK_CTRL_VAL_L1_EN)
T
Tomas Winkler 已提交
230
		/* disable L0S disabled L1A enabled */
231 232 233 234
		iwl_set_bit(priv, CSR_GIO_REG, CSR_GIO_REG_VAL_L0S_ENABLED);
	else
		/* L0S enabled L1A disabled */
		iwl_clear_bit(priv, CSR_GIO_REG, CSR_GIO_REG_VAL_L0S_ENABLED);
235 236 237 238 239 240 241 242 243 244 245 246 247 248 249

	radio_cfg = iwl_eeprom_query16(priv, EEPROM_RADIO_CONFIG);

	/* write radio config values to register */
	if (EEPROM_RF_CFG_TYPE_MSK(radio_cfg) < EEPROM_5000_RF_CFG_TYPE_MAX)
		iwl_set_bit(priv, CSR_HW_IF_CONFIG_REG,
			    EEPROM_RF_CFG_TYPE_MSK(radio_cfg) |
			    EEPROM_RF_CFG_STEP_MSK(radio_cfg) |
			    EEPROM_RF_CFG_DASH_MSK(radio_cfg));

	/* set CSR_HW_CONFIG_REG for uCode use */
	iwl_set_bit(priv, CSR_HW_IF_CONFIG_REG,
		    CSR_HW_IF_CONFIG_REG_BIT_RADIO_SI |
		    CSR_HW_IF_CONFIG_REG_BIT_MAC_SI);

T
Tomas Winkler 已提交
250 251 252 253
	/* W/A : NIC is stuck in a reset state after Early PCIe power off
	 * (PCIe power is lost before PERST# is asserted),
	 * causing ME FW to lose ownership and not being able to obtain it back.
	 */
254 255
	iwl_grab_nic_access(priv);
	iwl_set_bits_mask_prph(priv, APMG_PS_CTRL_REG,
T
Tomas Winkler 已提交
256 257
				APMG_PS_CTRL_EARLY_PWR_OFF_RESET_DIS,
				~APMG_PS_CTRL_EARLY_PWR_OFF_RESET_DIS);
258
	iwl_release_nic_access(priv);
T
Tomas Winkler 已提交
259

260 261 262 263 264
	spin_unlock_irqrestore(&priv->lock, flags);
}



265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294
/*
 * EEPROM
 */
static u32 eeprom_indirect_address(const struct iwl_priv *priv, u32 address)
{
	u16 offset = 0;

	if ((address & INDIRECT_ADDRESS) == 0)
		return address;

	switch (address & INDIRECT_TYPE_MSK) {
	case INDIRECT_HOST:
		offset = iwl_eeprom_query16(priv, EEPROM_5000_LINK_HOST);
		break;
	case INDIRECT_GENERAL:
		offset = iwl_eeprom_query16(priv, EEPROM_5000_LINK_GENERAL);
		break;
	case INDIRECT_REGULATORY:
		offset = iwl_eeprom_query16(priv, EEPROM_5000_LINK_REGULATORY);
		break;
	case INDIRECT_CALIBRATION:
		offset = iwl_eeprom_query16(priv, EEPROM_5000_LINK_CALIBRATION);
		break;
	case INDIRECT_PROCESS_ADJST:
		offset = iwl_eeprom_query16(priv, EEPROM_5000_LINK_PROCESS_ADJST);
		break;
	case INDIRECT_OTHERS:
		offset = iwl_eeprom_query16(priv, EEPROM_5000_LINK_OTHERS);
		break;
	default:
295
		IWL_ERR(priv, "illegal indirect type: 0x%X\n",
296 297 298 299 300 301 302 303
		address & INDIRECT_TYPE_MSK);
		break;
	}

	/* translate the offset from words to byte */
	return (address & ADDRESS_MSK) + (offset << 1);
}

304
static u16 iwl5000_eeprom_calib_version(struct iwl_priv *priv)
305 306 307 308 309 310 311 312 313
{
	struct iwl_eeprom_calib_hdr {
		u8 version;
		u8 pa_type;
		u16 voltage;
	} *hdr;

	hdr = (struct iwl_eeprom_calib_hdr *)iwl_eeprom_query_addr(priv,
							EEPROM_5000_CALIB_ALL);
314
	return hdr->version;
315 316 317

}

318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343
static void iwl5000_gain_computation(struct iwl_priv *priv,
		u32 average_noise[NUM_RX_CHAINS],
		u16 min_average_noise_antenna_i,
		u32 min_average_noise)
{
	int i;
	s32 delta_g;
	struct iwl_chain_noise_data *data = &priv->chain_noise_data;

	/* Find Gain Code for the antennas B and C */
	for (i = 1; i < NUM_RX_CHAINS; i++) {
		if ((data->disconn_array[i])) {
			data->delta_gain_code[i] = 0;
			continue;
		}
		delta_g = (1000 * ((s32)average_noise[0] -
			(s32)average_noise[i])) / 1500;
		/* bound gain by 2 bits value max, 3rd bit is sign */
		data->delta_gain_code[i] =
			min(abs(delta_g), CHAIN_NOISE_MAX_DELTA_GAIN_CODE);

		if (delta_g < 0)
			/* set negative sign */
			data->delta_gain_code[i] |= (1 << 2);
	}

344
	IWL_DEBUG_CALIB(priv, "Delta gains: ANT_B = %d  ANT_C = %d\n",
345 346 347
			data->delta_gain_code[1], data->delta_gain_code[2]);

	if (!data->radio_write) {
348
		struct iwl_calib_chain_noise_gain_cmd cmd;
349

350 351
		memset(&cmd, 0, sizeof(cmd));

352 353 354 355
		cmd.hdr.op_code = IWL_PHY_CALIBRATE_CHAIN_NOISE_GAIN_CMD;
		cmd.hdr.first_group = 0;
		cmd.hdr.groups_num = 1;
		cmd.hdr.data_valid = 1;
356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376
		cmd.delta_gain_1 = data->delta_gain_code[1];
		cmd.delta_gain_2 = data->delta_gain_code[2];
		iwl_send_cmd_pdu_async(priv, REPLY_PHY_CALIBRATION_CMD,
			sizeof(cmd), &cmd, NULL);

		data->radio_write = 1;
		data->state = IWL_CHAIN_NOISE_CALIBRATED;
	}

	data->chain_noise_a = 0;
	data->chain_noise_b = 0;
	data->chain_noise_c = 0;
	data->chain_signal_a = 0;
	data->chain_signal_b = 0;
	data->chain_signal_c = 0;
	data->beacon_count = 0;
}

static void iwl5000_chain_noise_reset(struct iwl_priv *priv)
{
	struct iwl_chain_noise_data *data = &priv->chain_noise_data;
377
	int ret;
378 379

	if ((data->state == IWL_CHAIN_NOISE_ALIVE) && iwl_is_associated(priv)) {
380
		struct iwl_calib_chain_noise_reset_cmd cmd;
381
		memset(&cmd, 0, sizeof(cmd));
382 383 384 385 386 387 388 389

		cmd.hdr.op_code = IWL_PHY_CALIBRATE_CHAIN_NOISE_RESET_CMD;
		cmd.hdr.first_group = 0;
		cmd.hdr.groups_num = 1;
		cmd.hdr.data_valid = 1;
		ret = iwl_send_cmd_pdu(priv, REPLY_PHY_CALIBRATION_CMD,
					sizeof(cmd), &cmd);
		if (ret)
390 391
			IWL_ERR(priv,
				"Could not send REPLY_PHY_CALIBRATION_CMD\n");
392
		data->state = IWL_CHAIN_NOISE_ACCUMULATE;
393
		IWL_DEBUG_CALIB(priv, "Run chain_noise_calibrate\n");
394 395 396
	}
}

397 398 399
static void iwl5000_rts_tx_cmd_flag(struct ieee80211_tx_info *info,
			__le32 *tx_flags)
{
400 401
	if ((info->control.rates[0].flags & IEEE80211_TX_RC_USE_RTS_CTS) ||
	    (info->control.rates[0].flags & IEEE80211_TX_RC_USE_CTS_PROTECT))
402 403 404 405 406
		*tx_flags |= TX_CMD_FLG_RTS_CTS_MSK;
	else
		*tx_flags &= ~TX_CMD_FLG_RTS_CTS_MSK;
}

407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427
static struct iwl_sensitivity_ranges iwl5000_sensitivity = {
	.min_nrg_cck = 95,
	.max_nrg_cck = 0,
	.auto_corr_min_ofdm = 90,
	.auto_corr_min_ofdm_mrc = 170,
	.auto_corr_min_ofdm_x1 = 120,
	.auto_corr_min_ofdm_mrc_x1 = 240,

	.auto_corr_max_ofdm = 120,
	.auto_corr_max_ofdm_mrc = 210,
	.auto_corr_max_ofdm_x1 = 155,
	.auto_corr_max_ofdm_mrc_x1 = 290,

	.auto_corr_min_cck = 125,
	.auto_corr_max_cck = 200,
	.auto_corr_min_cck_mrc = 170,
	.auto_corr_max_cck_mrc = 400,
	.nrg_th_cck = 95,
	.nrg_th_ofdm = 95,
};

428 429 430 431 432 433 434 435
static const u8 *iwl5000_eeprom_query_addr(const struct iwl_priv *priv,
					   size_t offset)
{
	u32 address = eeprom_indirect_address(priv, offset);
	BUG_ON(address >= priv->cfg->eeprom_size);
	return &priv->eeprom[address];
}

436 437 438 439 440 441 442 443 444 445 446
static s32 iwl5150_get_ct_threshold(struct iwl_priv *priv)
{
	const s32 volt2temp_coef = -5;
	u16 *temp_calib = (u16 *)iwl_eeprom_query_addr(priv,
						EEPROM_5000_TEMPERATURE);
	/* offset =  temperate -  voltage / coef */
	s32 offset = temp_calib[0] - temp_calib[1] / volt2temp_coef;
	s32 threshold = (s32)CELSIUS_TO_KELVIN(CT_KILL_THRESHOLD) - offset;
	return threshold * volt2temp_coef;
}

447 448 449
/*
 *  Calibration
 */
450
static int iwl5000_set_Xtal_calib(struct iwl_priv *priv)
451
{
452
	struct iwl_calib_xtal_freq_cmd cmd;
453 454
	u16 *xtal_calib = (u16 *)iwl_eeprom_query_addr(priv, EEPROM_5000_XTAL);

455 456 457 458 459 460
	cmd.hdr.op_code = IWL_PHY_CALIBRATE_CRYSTAL_FRQ_CMD;
	cmd.hdr.first_group = 0;
	cmd.hdr.groups_num = 1;
	cmd.hdr.data_valid = 1;
	cmd.cap_pin1 = (u8)xtal_calib[0];
	cmd.cap_pin2 = (u8)xtal_calib[1];
461
	return iwl_calib_set(&priv->calib_results[IWL_CALIB_XTAL],
462
			     (u8 *)&cmd, sizeof(cmd));
463 464 465 466
}

static int iwl5000_send_calib_cfg(struct iwl_priv *priv)
{
467
	struct iwl_calib_cfg_cmd calib_cfg_cmd;
468 469
	struct iwl_host_cmd cmd = {
		.id = CALIBRATION_CFG_CMD,
470
		.len = sizeof(struct iwl_calib_cfg_cmd),
471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486
		.data = &calib_cfg_cmd,
	};

	memset(&calib_cfg_cmd, 0, sizeof(calib_cfg_cmd));
	calib_cfg_cmd.ucd_calib_cfg.once.is_enable = IWL_CALIB_INIT_CFG_ALL;
	calib_cfg_cmd.ucd_calib_cfg.once.start = IWL_CALIB_INIT_CFG_ALL;
	calib_cfg_cmd.ucd_calib_cfg.once.send_res = IWL_CALIB_INIT_CFG_ALL;
	calib_cfg_cmd.ucd_calib_cfg.flags = IWL_CALIB_INIT_CFG_ALL;

	return iwl_send_cmd(priv, &cmd);
}

static void iwl5000_rx_calib_result(struct iwl_priv *priv,
			     struct iwl_rx_mem_buffer *rxb)
{
	struct iwl_rx_packet *pkt = (void *)rxb->skb->data;
487
	struct iwl_calib_hdr *hdr = (struct iwl_calib_hdr *)pkt->u.raw;
488
	int len = le32_to_cpu(pkt->len) & FH_RSCSR_FRAME_SIZE_MSK;
489
	int index;
490 491 492 493

	/* reduce the size of the length field itself */
	len -= 4;

494 495 496
	/* Define the order in which the results will be sent to the runtime
	 * uCode. iwl_send_calib_results sends them in a row according to their
	 * index. We sort them here */
497
	switch (hdr->op_code) {
498 499 500
	case IWL_PHY_CALIBRATE_DC_CMD:
		index = IWL_CALIB_DC;
		break;
501 502
	case IWL_PHY_CALIBRATE_LO_CMD:
		index = IWL_CALIB_LO;
503
		break;
504 505
	case IWL_PHY_CALIBRATE_TX_IQ_CMD:
		index = IWL_CALIB_TX_IQ;
506
		break;
507 508
	case IWL_PHY_CALIBRATE_TX_IQ_PERD_CMD:
		index = IWL_CALIB_TX_IQ_PERD;
509
		break;
510 511 512
	case IWL_PHY_CALIBRATE_BASE_BAND_CMD:
		index = IWL_CALIB_BASE_BAND;
		break;
513
	default:
514
		IWL_ERR(priv, "Unknown calibration notification %d\n",
515 516 517
			  hdr->op_code);
		return;
	}
518
	iwl_calib_set(&priv->calib_results[index], pkt->u.raw, len);
519 520 521 522 523
}

static void iwl5000_rx_calib_complete(struct iwl_priv *priv,
			       struct iwl_rx_mem_buffer *rxb)
{
524
	IWL_DEBUG_INFO(priv, "Init. calibration is completed, restarting fw.\n");
525 526 527
	queue_work(priv->workqueue, &priv->restart);
}

528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559
/*
 * ucode
 */
static int iwl5000_load_section(struct iwl_priv *priv,
				struct fw_desc *image,
				u32 dst_addr)
{
	int ret = 0;
	unsigned long flags;

	dma_addr_t phy_addr = image->p_addr;
	u32 byte_cnt = image->len;

	spin_lock_irqsave(&priv->lock, flags);
	ret = iwl_grab_nic_access(priv);
	if (ret) {
		spin_unlock_irqrestore(&priv->lock, flags);
		return ret;
	}

	iwl_write_direct32(priv,
		FH_TCSR_CHNL_TX_CONFIG_REG(FH_SRVC_CHNL),
		FH_TCSR_TX_CONFIG_REG_VAL_DMA_CHNL_PAUSE);

	iwl_write_direct32(priv,
		FH_SRVC_CHNL_SRAM_ADDR_REG(FH_SRVC_CHNL), dst_addr);

	iwl_write_direct32(priv,
		FH_TFDIB_CTRL0_REG(FH_SRVC_CHNL),
		phy_addr & FH_MEM_TFDIB_DRAM_ADDR_LSB_MSK);

	iwl_write_direct32(priv,
560
		FH_TFDIB_CTRL1_REG(FH_SRVC_CHNL),
T
Tomas Winkler 已提交
561
		(iwl_get_dma_hi_addr(phy_addr)
562 563
			<< FH_MEM_TFDIB_REG1_ADDR_BITSHIFT) | byte_cnt);

564 565 566 567 568 569 570 571 572
	iwl_write_direct32(priv,
		FH_TCSR_CHNL_TX_BUF_STS_REG(FH_SRVC_CHNL),
		1 << FH_TCSR_CHNL_TX_BUF_STS_REG_POS_TB_NUM |
		1 << FH_TCSR_CHNL_TX_BUF_STS_REG_POS_TB_IDX |
		FH_TCSR_CHNL_TX_BUF_STS_REG_VAL_TFDB_VALID);

	iwl_write_direct32(priv,
		FH_TCSR_CHNL_TX_CONFIG_REG(FH_SRVC_CHNL),
		FH_TCSR_TX_CONFIG_REG_VAL_DMA_CHNL_ENABLE	|
573
		FH_TCSR_TX_CONFIG_REG_VAL_DMA_CREDIT_DISABLE	|
574 575 576 577 578 579 580 581 582 583 584 585 586
		FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_HOST_ENDTFD);

	iwl_release_nic_access(priv);
	spin_unlock_irqrestore(&priv->lock, flags);
	return 0;
}

static int iwl5000_load_given_ucode(struct iwl_priv *priv,
		struct fw_desc *inst_image,
		struct fw_desc *data_image)
{
	int ret = 0;

587 588
	ret = iwl5000_load_section(priv, inst_image,
				   IWL50_RTC_INST_LOWER_BOUND);
589 590 591
	if (ret)
		return ret;

592
	IWL_DEBUG_INFO(priv, "INST uCode section being loaded...\n");
593
	ret = wait_event_interruptible_timeout(priv->wait_command_queue,
594
					priv->ucode_write_complete, 5 * HZ);
595
	if (ret == -ERESTARTSYS) {
596
		IWL_ERR(priv, "Could not load the INST uCode section due "
597 598 599 600
			"to interrupt\n");
		return ret;
	}
	if (!ret) {
601
		IWL_ERR(priv, "Could not load the INST uCode section\n");
602 603 604 605 606 607
		return -ETIMEDOUT;
	}

	priv->ucode_write_complete = 0;

	ret = iwl5000_load_section(
608
		priv, data_image, IWL50_RTC_DATA_LOWER_BOUND);
609 610 611
	if (ret)
		return ret;

612
	IWL_DEBUG_INFO(priv, "DATA uCode section being loaded...\n");
613 614 615 616

	ret = wait_event_interruptible_timeout(priv->wait_command_queue,
				priv->ucode_write_complete, 5 * HZ);
	if (ret == -ERESTARTSYS) {
617
		IWL_ERR(priv, "Could not load the INST uCode section due "
618 619 620
			"to interrupt\n");
		return ret;
	} else if (!ret) {
621
		IWL_ERR(priv, "Could not load the DATA uCode section\n");
622 623 624 625 626 627 628 629 630 631 632 633 634 635 636
		return -ETIMEDOUT;
	} else
		ret = 0;

	priv->ucode_write_complete = 0;

	return ret;
}

static int iwl5000_load_ucode(struct iwl_priv *priv)
{
	int ret = 0;

	/* check whether init ucode should be loaded, or rather runtime ucode */
	if (priv->ucode_init.len && (priv->ucode_type == UCODE_NONE)) {
637
		IWL_DEBUG_INFO(priv, "Init ucode found. Loading init ucode...\n");
638 639 640
		ret = iwl5000_load_given_ucode(priv,
			&priv->ucode_init, &priv->ucode_init_data);
		if (!ret) {
641
			IWL_DEBUG_INFO(priv, "Init ucode load complete.\n");
642 643 644
			priv->ucode_type = UCODE_INIT;
		}
	} else {
645
		IWL_DEBUG_INFO(priv, "Init ucode not found, or already loaded. "
646 647 648 649
			"Loading runtime ucode...\n");
		ret = iwl5000_load_given_ucode(priv,
			&priv->ucode_code, &priv->ucode_data);
		if (!ret) {
650
			IWL_DEBUG_INFO(priv, "Runtime ucode load complete.\n");
651 652 653 654 655 656 657
			priv->ucode_type = UCODE_RT;
		}
	}

	return ret;
}

658 659 660 661 662 663 664 665
static void iwl5000_init_alive_start(struct iwl_priv *priv)
{
	int ret = 0;

	/* Check alive response for "valid" sign from uCode */
	if (priv->card_alive_init.is_valid != UCODE_VALID_OK) {
		/* We had an error bringing up the hardware, so take it
		 * all the way back down so we can try again */
666
		IWL_DEBUG_INFO(priv, "Initialize Alive failed.\n");
667 668 669 670 671 672 673 674 675
		goto restart;
	}

	/* initialize uCode was loaded... verify inst image.
	 * This is a paranoid check, because we would not have gotten the
	 * "initialize" alive if code weren't properly loaded.  */
	if (iwl_verify_ucode(priv)) {
		/* Runtime instruction load was bad;
		 * take it all the way back down so we can try again */
676
		IWL_DEBUG_INFO(priv, "Bad \"initialize\" uCode load.\n");
677 678 679
		goto restart;
	}

680
	iwl_clear_stations_table(priv);
681 682
	ret = priv->cfg->ops->lib->alive_notify(priv);
	if (ret) {
683 684
		IWL_WARN(priv,
			"Could not complete ALIVE transition: %d\n", ret);
685 686 687
		goto restart;
	}

688
	iwl5000_send_calib_cfg(priv);
689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708
	return;

restart:
	/* real restart (first load init_ucode) */
	queue_work(priv->workqueue, &priv->restart);
}

static void iwl5000_set_wr_ptrs(struct iwl_priv *priv,
				int txq_id, u32 index)
{
	iwl_write_direct32(priv, HBUS_TARG_WRPTR,
			(index & 0xff) | (txq_id << 8));
	iwl_write_prph(priv, IWL50_SCD_QUEUE_RDPTR(txq_id), index);
}

static void iwl5000_tx_queue_set_status(struct iwl_priv *priv,
					struct iwl_tx_queue *txq,
					int tx_fifo_id, int scd_retry)
{
	int txq_id = txq->q.id;
T
Tomas Winkler 已提交
709
	int active = test_bit(txq_id, &priv->txq_ctx_active_msk) ? 1 : 0;
710 711 712 713 714 715 716 717 718

	iwl_write_prph(priv, IWL50_SCD_QUEUE_STATUS_BITS(txq_id),
			(active << IWL50_SCD_QUEUE_STTS_REG_POS_ACTIVE) |
			(tx_fifo_id << IWL50_SCD_QUEUE_STTS_REG_POS_TXF) |
			(1 << IWL50_SCD_QUEUE_STTS_REG_POS_WSL) |
			IWL50_SCD_QUEUE_STTS_REG_MSK);

	txq->sched_retry = scd_retry;

719
	IWL_DEBUG_INFO(priv, "%s %s Queue %d on AC %d\n",
720 721 722 723
		       active ? "Activate" : "Deactivate",
		       scd_retry ? "BA" : "AC", txq_id, tx_fifo_id);
}

724 725 726 727 728 729 730 731 732 733
static int iwl5000_send_wimax_coex(struct iwl_priv *priv)
{
	struct iwl_wimax_coex_cmd coex_cmd;

	memset(&coex_cmd, 0, sizeof(coex_cmd));

	return iwl_send_cmd_pdu(priv, COEX_PRIORITY_TABLE_CMD,
				sizeof(coex_cmd), &coex_cmd);
}

734 735 736 737 738
static int iwl5000_alive_notify(struct iwl_priv *priv)
{
	u32 a;
	unsigned long flags;
	int ret;
739
	int i, chan;
W
Winkler, Tomas 已提交
740
	u32 reg_val;
741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761

	spin_lock_irqsave(&priv->lock, flags);

	ret = iwl_grab_nic_access(priv);
	if (ret) {
		spin_unlock_irqrestore(&priv->lock, flags);
		return ret;
	}

	priv->scd_base_addr = iwl_read_prph(priv, IWL50_SCD_SRAM_BASE_ADDR);
	a = priv->scd_base_addr + IWL50_SCD_CONTEXT_DATA_OFFSET;
	for (; a < priv->scd_base_addr + IWL50_SCD_TX_STTS_BITMAP_OFFSET;
		a += 4)
		iwl_write_targ_mem(priv, a, 0);
	for (; a < priv->scd_base_addr + IWL50_SCD_TRANSLATE_TBL_OFFSET;
		a += 4)
		iwl_write_targ_mem(priv, a, 0);
	for (; a < sizeof(u16) * priv->hw_params.max_txq_num; a += 4)
		iwl_write_targ_mem(priv, a, 0);

	iwl_write_prph(priv, IWL50_SCD_DRAM_BASE_ADDR,
762
		       priv->scd_bc_tbls.dma >> 10);
763 764 765 766 767 768 769

	/* Enable DMA channel */
	for (chan = 0; chan < FH50_TCSR_CHNL_NUM ; chan++)
		iwl_write_direct32(priv, FH_TCSR_CHNL_TX_CONFIG_REG(chan),
				FH_TCSR_TX_CONFIG_REG_VAL_DMA_CHNL_ENABLE |
				FH_TCSR_TX_CONFIG_REG_VAL_DMA_CREDIT_ENABLE);

W
Winkler, Tomas 已提交
770 771 772 773 774
	/* Update FH chicken bits */
	reg_val = iwl_read_direct32(priv, FH_TX_CHICKEN_BITS_REG);
	iwl_write_direct32(priv, FH_TX_CHICKEN_BITS_REG,
			   reg_val | FH_TX_CHICKEN_BITS_SCD_AUTO_RETRY_EN);

775
	iwl_write_prph(priv, IWL50_SCD_QUEUECHAIN_SEL,
776
		IWL50_SCD_QUEUECHAIN_SEL_ALL(priv->hw_params.max_txq_num));
777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796
	iwl_write_prph(priv, IWL50_SCD_AGGR_SEL, 0);

	/* initiate the queues */
	for (i = 0; i < priv->hw_params.max_txq_num; i++) {
		iwl_write_prph(priv, IWL50_SCD_QUEUE_RDPTR(i), 0);
		iwl_write_direct32(priv, HBUS_TARG_WRPTR, 0 | (i << 8));
		iwl_write_targ_mem(priv, priv->scd_base_addr +
				IWL50_SCD_CONTEXT_QUEUE_OFFSET(i), 0);
		iwl_write_targ_mem(priv, priv->scd_base_addr +
				IWL50_SCD_CONTEXT_QUEUE_OFFSET(i) +
				sizeof(u32),
				((SCD_WIN_SIZE <<
				IWL50_SCD_QUEUE_CTX_REG2_WIN_SIZE_POS) &
				IWL50_SCD_QUEUE_CTX_REG2_WIN_SIZE_MSK) |
				((SCD_FRAME_LIMIT <<
				IWL50_SCD_QUEUE_CTX_REG2_FRAME_LIMIT_POS) &
				IWL50_SCD_QUEUE_CTX_REG2_FRAME_LIMIT_MSK));
	}

	iwl_write_prph(priv, IWL50_SCD_INTERRUPT_MASK,
797
			IWL_MASK(0, priv->hw_params.max_txq_num));
798

799 800
	/* Activate all Tx DMA/FIFO channels */
	priv->cfg->ops->lib->txq_set_sched(priv, IWL_MASK(0, 7));
801 802

	iwl5000_set_wr_ptrs(priv, IWL_CMD_QUEUE_NUM, 0);
803

804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819
	/* map qos queues to fifos one-to-one */
	for (i = 0; i < ARRAY_SIZE(iwl5000_default_queue_to_tx_fifo); i++) {
		int ac = iwl5000_default_queue_to_tx_fifo[i];
		iwl_txq_ctx_activate(priv, i);
		iwl5000_tx_queue_set_status(priv, &priv->txq[i], ac, 0);
	}
	/* TODO - need to initialize those FIFOs inside the loop above,
	 * not only mark them as active */
	iwl_txq_ctx_activate(priv, 4);
	iwl_txq_ctx_activate(priv, 7);
	iwl_txq_ctx_activate(priv, 8);
	iwl_txq_ctx_activate(priv, 9);

	iwl_release_nic_access(priv);
	spin_unlock_irqrestore(&priv->lock, flags);

820

821 822
	iwl5000_send_wimax_coex(priv);

823 824
	iwl5000_set_Xtal_calib(priv);
	iwl_send_calib_results(priv);
825

826 827 828
	return 0;
}

829 830 831 832
static int iwl5000_hw_set_hw_params(struct iwl_priv *priv)
{
	if ((priv->cfg->mod_params->num_of_queues > IWL50_NUM_QUEUES) ||
	    (priv->cfg->mod_params->num_of_queues < IWL_MIN_NUM_QUEUES)) {
833 834 835
		IWL_ERR(priv,
			"invalid queues_num, should be between %d and %d\n",
			IWL_MIN_NUM_QUEUES, IWL50_NUM_QUEUES);
836 837
		return -EINVAL;
	}
838

839
	priv->hw_params.max_txq_num = priv->cfg->mod_params->num_of_queues;
840
	priv->hw_params.dma_chnl_num = FH50_TCSR_CHNL_NUM;
841 842
	priv->hw_params.scd_bc_tbls_size =
			IWL50_NUM_QUEUES * sizeof(struct iwl5000_scd_bc_tbl);
843
	priv->hw_params.tfd_size = sizeof(struct iwl_tfd);
844 845
	priv->hw_params.max_stations = IWL5000_STATION_COUNT;
	priv->hw_params.bcast_sta_id = IWL5000_BROADCAST_ID;
846 847 848 849 850 851 852 853 854 855 856 857

	switch (priv->hw_rev & CSR_HW_REV_TYPE_MSK) {
	case CSR_HW_REV_TYPE_6x00:
	case CSR_HW_REV_TYPE_6x50:
		priv->hw_params.max_data_size = IWL60_RTC_DATA_SIZE;
		priv->hw_params.max_inst_size = IWL60_RTC_INST_SIZE;
		break;
	default:
		priv->hw_params.max_data_size = IWL50_RTC_DATA_SIZE;
		priv->hw_params.max_inst_size = IWL50_RTC_INST_SIZE;
	}

858
	priv->hw_params.max_bsm_size = 0;
859 860
	priv->hw_params.fat_channel =  BIT(IEEE80211_BAND_2GHZ) |
					BIT(IEEE80211_BAND_5GHZ);
861 862
	priv->hw_params.rx_wrt_ptr_reg = FH_RSCSR_CHNL0_WPTR;

863
	priv->hw_params.sens = &iwl5000_sensitivity;
864

865 866 867 868
	priv->hw_params.tx_chains_num = num_of_ant(priv->cfg->valid_tx_ant);
	priv->hw_params.rx_chains_num = num_of_ant(priv->cfg->valid_rx_ant);
	priv->hw_params.valid_tx_ant = priv->cfg->valid_tx_ant;
	priv->hw_params.valid_rx_ant = priv->cfg->valid_rx_ant;
869 870 871

	switch (priv->hw_rev & CSR_HW_REV_TYPE_MSK) {
	case CSR_HW_REV_TYPE_5150:
872
		/* 5150 wants in Kelvin */
873
		priv->hw_params.ct_kill_threshold =
874
				iwl5150_get_ct_threshold(priv);
875
		break;
876 877 878 879
	default:
		/* all others want Celsius */
		priv->hw_params.ct_kill_threshold = CT_KILL_THRESHOLD;
		break;
880 881
	}

882 883
	/* Set initial calibration set */
	switch (priv->hw_rev & CSR_HW_REV_TYPE_MSK) {
884
	case CSR_HW_REV_TYPE_5150:
885
		priv->hw_params.calib_init_cfg =
886
			BIT(IWL_CALIB_DC)		|
887
			BIT(IWL_CALIB_LO)		|
888 889
			BIT(IWL_CALIB_TX_IQ) 		|
			BIT(IWL_CALIB_BASE_BAND);
890

891
		break;
892
	default:
893
		priv->hw_params.calib_init_cfg =
894
			BIT(IWL_CALIB_XTAL)		|
895 896
			BIT(IWL_CALIB_LO)		|
			BIT(IWL_CALIB_TX_IQ) 		|
897
			BIT(IWL_CALIB_TX_IQ_PERD)	|
898
			BIT(IWL_CALIB_BASE_BAND);
899 900 901 902
		break;
	}


903 904
	return 0;
}
905

906 907 908 909
/**
 * iwl5000_txq_update_byte_cnt_tbl - Set up entry in Tx byte-count array
 */
static void iwl5000_txq_update_byte_cnt_tbl(struct iwl_priv *priv,
910
					    struct iwl_tx_queue *txq,
911 912
					    u16 byte_cnt)
{
913
	struct iwl5000_scd_bc_tbl *scd_bc_tbl = priv->scd_bc_tbls.addr;
914
	int write_ptr = txq->q.write_ptr;
915 916
	int txq_id = txq->q.id;
	u8 sec_ctl = 0;
917 918 919
	u8 sta_id = 0;
	u16 len = byte_cnt + IWL_TX_CRC_SIZE + IWL_TX_DELIMITER_SIZE;
	__le16 bc_ent;
920

921
	WARN_ON(len > 0xFFF || write_ptr >= TFD_QUEUE_SIZE_MAX);
922 923

	if (txq_id != IWL_CMD_QUEUE_NUM) {
924
		sta_id = txq->cmd[txq->q.write_ptr]->cmd.tx.sta_id;
925
		sec_ctl = txq->cmd[txq->q.write_ptr]->cmd.tx.sec_ctl;
926 927 928 929 930 931 932 933 934 935 936 937 938 939

		switch (sec_ctl & TX_CMD_SEC_MSK) {
		case TX_CMD_SEC_CCM:
			len += CCMP_MIC_LEN;
			break;
		case TX_CMD_SEC_TKIP:
			len += TKIP_ICV_LEN;
			break;
		case TX_CMD_SEC_WEP:
			len += WEP_IV_LEN + WEP_ICV_LEN;
			break;
		}
	}

940
	bc_ent = cpu_to_le16((len & 0xFFF) | (sta_id << 12));
941

942
	scd_bc_tbl[txq_id].tfd_offset[write_ptr] = bc_ent;
943

944
	if (txq->q.write_ptr < TFD_QUEUE_SIZE_BC_DUP)
945
		scd_bc_tbl[txq_id].
946
			tfd_offset[TFD_QUEUE_SIZE_MAX + write_ptr] = bc_ent;
947 948
}

949 950 951
static void iwl5000_txq_inval_byte_cnt_tbl(struct iwl_priv *priv,
					   struct iwl_tx_queue *txq)
{
952
	struct iwl5000_scd_bc_tbl *scd_bc_tbl = priv->scd_bc_tbls.addr;
953 954 955 956 957 958
	int txq_id = txq->q.id;
	int read_ptr = txq->q.read_ptr;
	u8 sta_id = 0;
	__le16 bc_ent;

	WARN_ON(read_ptr >= TFD_QUEUE_SIZE_MAX);
959 960

	if (txq_id != IWL_CMD_QUEUE_NUM)
961
		sta_id = txq->cmd[read_ptr]->cmd.tx.sta_id;
962

963
	bc_ent =  cpu_to_le16(1 | (sta_id << 12));
964
	scd_bc_tbl[txq_id].tfd_offset[read_ptr] = bc_ent;
965

966
	if (txq->q.write_ptr < TFD_QUEUE_SIZE_BC_DUP)
967
		scd_bc_tbl[txq_id].
968
			tfd_offset[TFD_QUEUE_SIZE_MAX + read_ptr] =  bc_ent;
969 970
}

971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010
static int iwl5000_tx_queue_set_q2ratid(struct iwl_priv *priv, u16 ra_tid,
					u16 txq_id)
{
	u32 tbl_dw_addr;
	u32 tbl_dw;
	u16 scd_q2ratid;

	scd_q2ratid = ra_tid & IWL_SCD_QUEUE_RA_TID_MAP_RATID_MSK;

	tbl_dw_addr = priv->scd_base_addr +
			IWL50_SCD_TRANSLATE_TBL_OFFSET_QUEUE(txq_id);

	tbl_dw = iwl_read_targ_mem(priv, tbl_dw_addr);

	if (txq_id & 0x1)
		tbl_dw = (scd_q2ratid << 16) | (tbl_dw & 0x0000FFFF);
	else
		tbl_dw = scd_q2ratid | (tbl_dw & 0xFFFF0000);

	iwl_write_targ_mem(priv, tbl_dw_addr, tbl_dw);

	return 0;
}
static void iwl5000_tx_queue_stop_scheduler(struct iwl_priv *priv, u16 txq_id)
{
	/* Simply stop the queue, but don't change any configuration;
	 * the SCD_ACT_EN bit is the write-enable mask for the ACTIVE bit. */
	iwl_write_prph(priv,
		IWL50_SCD_QUEUE_STATUS_BITS(txq_id),
		(0 << IWL50_SCD_QUEUE_STTS_REG_POS_ACTIVE)|
		(1 << IWL50_SCD_QUEUE_STTS_REG_POS_SCD_ACT_EN));
}

static int iwl5000_txq_agg_enable(struct iwl_priv *priv, int txq_id,
				  int tx_fifo, int sta_id, int tid, u16 ssn_idx)
{
	unsigned long flags;
	int ret;
	u16 ra_tid;

1011 1012
	if ((IWL50_FIRST_AMPDU_QUEUE > txq_id) ||
	    (IWL50_FIRST_AMPDU_QUEUE + IWL50_NUM_AMPDU_QUEUES <= txq_id)) {
1013 1014
		IWL_WARN(priv,
			"queue number out of range: %d, must be %d to %d\n",
1015 1016 1017 1018
			txq_id, IWL50_FIRST_AMPDU_QUEUE,
			IWL50_FIRST_AMPDU_QUEUE + IWL50_NUM_AMPDU_QUEUES - 1);
		return -EINVAL;
	}
1019 1020 1021 1022

	ra_tid = BUILD_RAxTID(sta_id, tid);

	/* Modify device's station table to Tx this TID */
1023
	iwl_sta_tx_modify_enable_tid(priv, sta_id, tid);
1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076

	spin_lock_irqsave(&priv->lock, flags);
	ret = iwl_grab_nic_access(priv);
	if (ret) {
		spin_unlock_irqrestore(&priv->lock, flags);
		return ret;
	}

	/* Stop this Tx queue before configuring it */
	iwl5000_tx_queue_stop_scheduler(priv, txq_id);

	/* Map receiver-address / traffic-ID to this queue */
	iwl5000_tx_queue_set_q2ratid(priv, ra_tid, txq_id);

	/* Set this queue as a chain-building queue */
	iwl_set_bits_prph(priv, IWL50_SCD_QUEUECHAIN_SEL, (1<<txq_id));

	/* enable aggregations for the queue */
	iwl_set_bits_prph(priv, IWL50_SCD_AGGR_SEL, (1<<txq_id));

	/* Place first TFD at index corresponding to start sequence number.
	 * Assumes that ssn_idx is valid (!= 0xFFF) */
	priv->txq[txq_id].q.read_ptr = (ssn_idx & 0xff);
	priv->txq[txq_id].q.write_ptr = (ssn_idx & 0xff);
	iwl5000_set_wr_ptrs(priv, txq_id, ssn_idx);

	/* Set up Tx window size and frame limit for this queue */
	iwl_write_targ_mem(priv, priv->scd_base_addr +
			IWL50_SCD_CONTEXT_QUEUE_OFFSET(txq_id) +
			sizeof(u32),
			((SCD_WIN_SIZE <<
			IWL50_SCD_QUEUE_CTX_REG2_WIN_SIZE_POS) &
			IWL50_SCD_QUEUE_CTX_REG2_WIN_SIZE_MSK) |
			((SCD_FRAME_LIMIT <<
			IWL50_SCD_QUEUE_CTX_REG2_FRAME_LIMIT_POS) &
			IWL50_SCD_QUEUE_CTX_REG2_FRAME_LIMIT_MSK));

	iwl_set_bits_prph(priv, IWL50_SCD_INTERRUPT_MASK, (1 << txq_id));

	/* Set up Status area in SRAM, map to Tx DMA/FIFO, activate the queue */
	iwl5000_tx_queue_set_status(priv, &priv->txq[txq_id], tx_fifo, 1);

	iwl_release_nic_access(priv);
	spin_unlock_irqrestore(&priv->lock, flags);

	return 0;
}

static int iwl5000_txq_agg_disable(struct iwl_priv *priv, u16 txq_id,
				   u16 ssn_idx, u8 tx_fifo)
{
	int ret;

1077 1078
	if ((IWL50_FIRST_AMPDU_QUEUE > txq_id) ||
	    (IWL50_FIRST_AMPDU_QUEUE + IWL50_NUM_AMPDU_QUEUES <= txq_id)) {
1079 1080
		IWL_WARN(priv,
			"queue number out of range: %d, must be %d to %d\n",
1081 1082
			txq_id, IWL50_FIRST_AMPDU_QUEUE,
			IWL50_FIRST_AMPDU_QUEUE + IWL50_NUM_AMPDU_QUEUES - 1);
1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107
		return -EINVAL;
	}

	ret = iwl_grab_nic_access(priv);
	if (ret)
		return ret;

	iwl5000_tx_queue_stop_scheduler(priv, txq_id);

	iwl_clear_bits_prph(priv, IWL50_SCD_AGGR_SEL, (1 << txq_id));

	priv->txq[txq_id].q.read_ptr = (ssn_idx & 0xff);
	priv->txq[txq_id].q.write_ptr = (ssn_idx & 0xff);
	/* supposes that ssn_idx is valid (!= 0xFFF) */
	iwl5000_set_wr_ptrs(priv, txq_id, ssn_idx);

	iwl_clear_bits_prph(priv, IWL50_SCD_INTERRUPT_MASK, (1 << txq_id));
	iwl_txq_ctx_deactivate(priv, txq_id);
	iwl5000_tx_queue_set_status(priv, &priv->txq[txq_id], tx_fifo, 0);

	iwl_release_nic_access(priv);

	return 0;
}

1108 1109 1110 1111 1112 1113 1114 1115
static u16 iwl5000_build_addsta_hcmd(const struct iwl_addsta_cmd *cmd, u8 *data)
{
	u16 size = (u16)sizeof(struct iwl_addsta_cmd);
	memcpy(data, cmd, size);
	return size;
}


1116
/*
T
Tomas Winkler 已提交
1117
 * Activate/Deactivate Tx DMA/FIFO channels according tx fifos mask
1118 1119 1120
 * must be called under priv->lock and mac access
 */
static void iwl5000_txq_set_sched(struct iwl_priv *priv, u32 mask)
1121
{
1122
	iwl_write_prph(priv, IWL50_SCD_TXFACT, mask);
1123 1124
}

1125 1126 1127

static inline u32 iwl5000_get_scd_ssn(struct iwl5000_tx_resp *tx_resp)
{
1128
	return le32_to_cpup((__le32 *)&tx_resp->status +
1129
			    tx_resp->frame_count) & MAX_SN;
1130 1131 1132 1133 1134
}

static int iwl5000_tx_status_reply_tx(struct iwl_priv *priv,
				      struct iwl_ht_agg *agg,
				      struct iwl5000_tx_resp *tx_resp,
1135
				      int txq_id, u16 start_idx)
1136 1137 1138 1139 1140
{
	u16 status;
	struct agg_tx_status *frame_status = &tx_resp->status;
	struct ieee80211_tx_info *info = NULL;
	struct ieee80211_hdr *hdr = NULL;
1141
	u32 rate_n_flags = le32_to_cpu(tx_resp->rate_n_flags);
1142
	int i, sh, idx;
1143 1144 1145
	u16 seq;

	if (agg->wait_for_ba)
1146
		IWL_DEBUG_TX_REPLY(priv, "got tx response w/o block-ack\n");
1147 1148 1149

	agg->frame_count = tx_resp->frame_count;
	agg->start_idx = start_idx;
1150
	agg->rate_n_flags = rate_n_flags;
1151 1152 1153 1154 1155 1156
	agg->bitmap = 0;

	/* # frames attempted by Tx command */
	if (agg->frame_count == 1) {
		/* Only one frame was attempted; no block-ack will arrive */
		status = le16_to_cpu(frame_status[0].status);
1157
		idx = start_idx;
1158 1159

		/* FIXME: code repetition */
1160
		IWL_DEBUG_TX_REPLY(priv, "FrameCnt = %d, StartIdx=%d idx=%d\n",
1161 1162 1163
				   agg->frame_count, agg->start_idx, idx);

		info = IEEE80211_SKB_CB(priv->txq[txq_id].txb[idx].skb[0]);
1164
		info->status.rates[0].count = tx_resp->failure_frame + 1;
1165
		info->flags &= ~IEEE80211_TX_CTL_AMPDU;
1166
		info->flags |= iwl_is_tx_success(status) ?
T
Tomas Winkler 已提交
1167
					IEEE80211_TX_STAT_ACK : 0;
1168 1169
		iwl_hwrate_to_tx_control(priv, rate_n_flags, info);

1170 1171
		/* FIXME: code repetition end */

1172
		IWL_DEBUG_TX_REPLY(priv, "1 Frame 0x%x failure :%d\n",
1173
				    status & 0xff, tx_resp->failure_frame);
1174
		IWL_DEBUG_TX_REPLY(priv, "Rate Info rate_n_flags=%x\n", rate_n_flags);
1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193

		agg->wait_for_ba = 0;
	} else {
		/* Two or more frames were attempted; expect block-ack */
		u64 bitmap = 0;
		int start = agg->start_idx;

		/* Construct bit-map of pending frames within Tx window */
		for (i = 0; i < agg->frame_count; i++) {
			u16 sc;
			status = le16_to_cpu(frame_status[i].status);
			seq  = le16_to_cpu(frame_status[i].sequence);
			idx = SEQ_TO_INDEX(seq);
			txq_id = SEQ_TO_QUEUE(seq);

			if (status & (AGG_TX_STATE_FEW_BYTES_MSK |
				      AGG_TX_STATE_ABORT_MSK))
				continue;

1194
			IWL_DEBUG_TX_REPLY(priv, "FrameCnt = %d, txq_id=%d idx=%d\n",
1195 1196 1197 1198 1199 1200
					   agg->frame_count, txq_id, idx);

			hdr = iwl_tx_queue_get_hdr(priv, txq_id, idx);

			sc = le16_to_cpu(hdr->seq_ctrl);
			if (idx != (SEQ_TO_SN(sc) & 0xff)) {
1201 1202 1203
				IWL_ERR(priv,
					"BUG_ON idx doesn't match seq control"
					" idx=%d, seq_idx=%d, seq=%d\n",
1204 1205 1206 1207 1208
					  idx, SEQ_TO_SN(sc),
					  hdr->seq_ctrl);
				return -1;
			}

1209
			IWL_DEBUG_TX_REPLY(priv, "AGG Frame i=%d idx %d seq=%d\n",
1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225
					   i, idx, SEQ_TO_SN(sc));

			sh = idx - start;
			if (sh > 64) {
				sh = (start - idx) + 0xff;
				bitmap = bitmap << sh;
				sh = 0;
				start = idx;
			} else if (sh < -64)
				sh  = 0xff - (start - idx);
			else if (sh < 0) {
				sh = start - idx;
				start = idx;
				bitmap = bitmap << sh;
				sh = 0;
			}
1226
			bitmap |= 1ULL << sh;
1227
			IWL_DEBUG_TX_REPLY(priv, "start=%d bitmap=0x%llx\n",
1228
					   start, (unsigned long long)bitmap);
1229 1230 1231 1232
		}

		agg->bitmap = bitmap;
		agg->start_idx = start;
1233
		IWL_DEBUG_TX_REPLY(priv, "Frames %d start_idx=%d bitmap=0x%llx\n",
1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253
				   agg->frame_count, agg->start_idx,
				   (unsigned long long)agg->bitmap);

		if (bitmap)
			agg->wait_for_ba = 1;
	}
	return 0;
}

static void iwl5000_rx_reply_tx(struct iwl_priv *priv,
				struct iwl_rx_mem_buffer *rxb)
{
	struct iwl_rx_packet *pkt = (struct iwl_rx_packet *)rxb->skb->data;
	u16 sequence = le16_to_cpu(pkt->hdr.sequence);
	int txq_id = SEQ_TO_QUEUE(sequence);
	int index = SEQ_TO_INDEX(sequence);
	struct iwl_tx_queue *txq = &priv->txq[txq_id];
	struct ieee80211_tx_info *info;
	struct iwl5000_tx_resp *tx_resp = (void *)&pkt->u.raw[0];
	u32  status = le16_to_cpu(tx_resp->status.status);
T
Tomas Winkler 已提交
1254 1255 1256
	int tid;
	int sta_id;
	int freed;
1257 1258

	if ((index >= txq->q.n_bd) || (iwl_queue_used(&txq->q, index) == 0)) {
1259
		IWL_ERR(priv, "Read index for DMA queue txq_id (%d) index %d "
1260 1261 1262 1263 1264 1265 1266 1267 1268
			  "is out of range [0-%d] %d %d\n", txq_id,
			  index, txq->q.n_bd, txq->q.write_ptr,
			  txq->q.read_ptr);
		return;
	}

	info = IEEE80211_SKB_CB(txq->txb[txq->q.read_ptr].skb[0]);
	memset(&info->status, 0, sizeof(info->status));

T
Tomas Winkler 已提交
1269 1270
	tid = (tx_resp->ra_tid & IWL50_TX_RES_TID_MSK) >> IWL50_TX_RES_TID_POS;
	sta_id = (tx_resp->ra_tid & IWL50_TX_RES_RA_MSK) >> IWL50_TX_RES_RA_POS;
1271 1272 1273 1274 1275 1276 1277

	if (txq->sched_retry) {
		const u32 scd_ssn = iwl5000_get_scd_ssn(tx_resp);
		struct iwl_ht_agg *agg = NULL;

		agg = &priv->stations[sta_id].tid[tid].agg;

1278
		iwl5000_tx_status_reply_tx(priv, agg, tx_resp, txq_id, index);
1279

1280 1281 1282
		/* check if BAR is needed */
		if ((tx_resp->frame_count == 1) && !iwl_is_tx_success(status))
			info->flags |= IEEE80211_TX_STAT_AMPDU_NO_BACK;
1283 1284 1285

		if (txq->q.read_ptr != (scd_ssn & 0xff)) {
			index = iwl_queue_dec_wrap(scd_ssn & 0xff, txq->q.n_bd);
1286
			IWL_DEBUG_TX_REPLY(priv, "Retry scheduler reclaim "
T
Tomas Winkler 已提交
1287 1288 1289
					"scd_ssn=%d idx=%d txq=%d swq=%d\n",
					scd_ssn , index, txq_id, txq->swq_id);

1290
			freed = iwl_tx_queue_reclaim(priv, txq_id, index);
1291 1292
			priv->stations[sta_id].tid[tid].tfds_in_queue -= freed;

T
Tomas Winkler 已提交
1293 1294 1295
			if (priv->mac80211_registered &&
			    (iwl_queue_space(&txq->q) > txq->q.low_mark) &&
			    (agg->state != IWL_EMPTYING_HW_QUEUE_DELBA)) {
1296 1297 1298
				if (agg->state == IWL_AGG_OFF)
					ieee80211_wake_queue(priv->hw, txq_id);
				else
T
Tomas Winkler 已提交
1299 1300
					ieee80211_wake_queue(priv->hw,
							     txq->swq_id);
1301 1302 1303
			}
		}
	} else {
T
Tomas Winkler 已提交
1304 1305
		BUG_ON(txq_id != txq->swq_id);

1306
		info->status.rates[0].count = tx_resp->failure_frame + 1;
T
Tomas Winkler 已提交
1307 1308
		info->flags |= iwl_is_tx_success(status) ?
					IEEE80211_TX_STAT_ACK : 0;
1309
		iwl_hwrate_to_tx_control(priv,
1310 1311 1312
					le32_to_cpu(tx_resp->rate_n_flags),
					info);

1313
		IWL_DEBUG_TX_REPLY(priv, "TXQ %d status %s (0x%08x) rate_n_flags "
T
Tomas Winkler 已提交
1314 1315 1316 1317 1318
				   "0x%x retries %d\n",
				   txq_id,
				   iwl_get_tx_fail_reason(status), status,
				   le32_to_cpu(tx_resp->rate_n_flags),
				   tx_resp->failure_frame);
1319

T
Tomas Winkler 已提交
1320 1321
		freed = iwl_tx_queue_reclaim(priv, txq_id, index);
		if (ieee80211_is_data_qos(tx_resp->frame_ctrl))
1322
			priv->stations[sta_id].tid[tid].tfds_in_queue -= freed;
T
Tomas Winkler 已提交
1323 1324 1325

		if (priv->mac80211_registered &&
		    (iwl_queue_space(&txq->q) > txq->q.low_mark))
1326 1327 1328
			ieee80211_wake_queue(priv->hw, txq_id);
	}

T
Tomas Winkler 已提交
1329 1330 1331
	if (ieee80211_is_data_qos(tx_resp->frame_ctrl))
		iwl_txq_check_empty(priv, sta_id, tid, txq_id);

1332
	if (iwl_check_bits(status, TX_ABORT_REQUIRED_MSK))
1333
		IWL_ERR(priv, "TODO:  Implement Tx ABORT REQUIRED!!!\n");
1334 1335
}

T
Tomas Winkler 已提交
1336
/* Currently 5000 is the superset of everything */
G
Gregory Greenman 已提交
1337 1338 1339 1340 1341
static u16 iwl5000_get_hcmd_size(u8 cmd_id, u16 len)
{
	return len;
}

1342 1343 1344 1345 1346 1347
static void iwl5000_setup_deferred_work(struct iwl_priv *priv)
{
	/* in 5000 the tx power calibration is done in uCode */
	priv->disable_tx_power_cal = 1;
}

1348 1349
static void iwl5000_rx_handler_setup(struct iwl_priv *priv)
{
1350 1351 1352 1353 1354
	/* init calibration handlers */
	priv->rx_handlers[CALIBRATION_RES_NOTIFICATION] =
					iwl5000_rx_calib_result;
	priv->rx_handlers[CALIBRATION_COMPLETE_NOTIFICATION] =
					iwl5000_rx_calib_complete;
1355
	priv->rx_handlers[REPLY_TX] = iwl5000_rx_reply_tx;
1356 1357
}

1358

1359 1360
static int iwl5000_hw_valid_rtc_data_addr(u32 addr)
{
1361
	return (addr >= IWL50_RTC_DATA_LOWER_BOUND) &&
1362 1363 1364
		(addr < IWL50_RTC_DATA_UPPER_BOUND);
}

1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383
static int iwl5000_send_rxon_assoc(struct iwl_priv *priv)
{
	int ret = 0;
	struct iwl5000_rxon_assoc_cmd rxon_assoc;
	const struct iwl_rxon_cmd *rxon1 = &priv->staging_rxon;
	const struct iwl_rxon_cmd *rxon2 = &priv->active_rxon;

	if ((rxon1->flags == rxon2->flags) &&
	    (rxon1->filter_flags == rxon2->filter_flags) &&
	    (rxon1->cck_basic_rates == rxon2->cck_basic_rates) &&
	    (rxon1->ofdm_ht_single_stream_basic_rates ==
	     rxon2->ofdm_ht_single_stream_basic_rates) &&
	    (rxon1->ofdm_ht_dual_stream_basic_rates ==
	     rxon2->ofdm_ht_dual_stream_basic_rates) &&
	    (rxon1->ofdm_ht_triple_stream_basic_rates ==
	     rxon2->ofdm_ht_triple_stream_basic_rates) &&
	    (rxon1->acquisition_data == rxon2->acquisition_data) &&
	    (rxon1->rx_chain == rxon2->rx_chain) &&
	    (rxon1->ofdm_basic_rates == rxon2->ofdm_basic_rates)) {
1384
		IWL_DEBUG_INFO(priv, "Using current RXON_ASSOC.  Not resending.\n");
1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410
		return 0;
	}

	rxon_assoc.flags = priv->staging_rxon.flags;
	rxon_assoc.filter_flags = priv->staging_rxon.filter_flags;
	rxon_assoc.ofdm_basic_rates = priv->staging_rxon.ofdm_basic_rates;
	rxon_assoc.cck_basic_rates = priv->staging_rxon.cck_basic_rates;
	rxon_assoc.reserved1 = 0;
	rxon_assoc.reserved2 = 0;
	rxon_assoc.reserved3 = 0;
	rxon_assoc.ofdm_ht_single_stream_basic_rates =
	    priv->staging_rxon.ofdm_ht_single_stream_basic_rates;
	rxon_assoc.ofdm_ht_dual_stream_basic_rates =
	    priv->staging_rxon.ofdm_ht_dual_stream_basic_rates;
	rxon_assoc.rx_chain_select_flags = priv->staging_rxon.rx_chain;
	rxon_assoc.ofdm_ht_triple_stream_basic_rates =
		 priv->staging_rxon.ofdm_ht_triple_stream_basic_rates;
	rxon_assoc.acquisition_data = priv->staging_rxon.acquisition_data;

	ret = iwl_send_cmd_pdu_async(priv, REPLY_RXON_ASSOC,
				     sizeof(rxon_assoc), &rxon_assoc, NULL);
	if (ret)
		return ret;

	return ret;
}
1411 1412 1413 1414 1415 1416
static int  iwl5000_send_tx_power(struct iwl_priv *priv)
{
	struct iwl5000_tx_power_dbm_cmd tx_power_cmd;

	/* half dBm need to multiply */
	tx_power_cmd.global_lmt = (s8)(2 * priv->tx_power_user_lmt);
1417
	tx_power_cmd.flags = IWL50_TX_POWER_NO_CLOSED;
1418 1419 1420 1421 1422 1423
	tx_power_cmd.srv_chan_lmt = IWL50_TX_POWER_AUTO;
	return  iwl_send_cmd_pdu_async(priv, REPLY_TX_POWER_DBM_CMD,
				       sizeof(tx_power_cmd), &tx_power_cmd,
				       NULL);
}

1424
static void iwl5000_temperature(struct iwl_priv *priv)
1425 1426
{
	/* store temperature from statistics (in Celsius) */
1427
	priv->temperature = le32_to_cpu(priv->statistics.general.temperature);
1428
}
1429

1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459
/* Calc max signal level (dBm) among 3 possible receivers */
static int iwl5000_calc_rssi(struct iwl_priv *priv,
			     struct iwl_rx_phy_res *rx_resp)
{
	/* data from PHY/DSP regarding signal strength, etc.,
	 *   contents are always there, not configurable by host
	 */
	struct iwl5000_non_cfg_phy *ncphy =
		(struct iwl5000_non_cfg_phy *)rx_resp->non_cfg_phy_buf;
	u32 val, rssi_a, rssi_b, rssi_c, max_rssi;
	u8 agc;

	val  = le32_to_cpu(ncphy->non_cfg_phy[IWL50_RX_RES_AGC_IDX]);
	agc = (val & IWL50_OFDM_AGC_MSK) >> IWL50_OFDM_AGC_BIT_POS;

	/* Find max rssi among 3 possible receivers.
	 * These values are measured by the digital signal processor (DSP).
	 * They should stay fairly constant even as the signal strength varies,
	 *   if the radio's automatic gain control (AGC) is working right.
	 * AGC value (see below) will provide the "interesting" info.
	 */
	val = le32_to_cpu(ncphy->non_cfg_phy[IWL50_RX_RES_RSSI_AB_IDX]);
	rssi_a = (val & IWL50_OFDM_RSSI_A_MSK) >> IWL50_OFDM_RSSI_A_BIT_POS;
	rssi_b = (val & IWL50_OFDM_RSSI_B_MSK) >> IWL50_OFDM_RSSI_B_BIT_POS;
	val = le32_to_cpu(ncphy->non_cfg_phy[IWL50_RX_RES_RSSI_C_IDX]);
	rssi_c = (val & IWL50_OFDM_RSSI_C_MSK) >> IWL50_OFDM_RSSI_C_BIT_POS;

	max_rssi = max_t(u32, rssi_a, rssi_b);
	max_rssi = max_t(u32, max_rssi, rssi_c);

1460
	IWL_DEBUG_STATS(priv, "Rssi In A %d B %d C %d Max %d AGC dB %d\n",
1461 1462 1463 1464
		rssi_a, rssi_b, rssi_c, max_rssi, agc);

	/* dBm = max_rssi dB - agc dB - constant.
	 * Higher AGC (higher radio gain) means lower signal. */
1465
	return max_rssi - agc - IWL49_RSSI_OFFSET;
1466 1467
}

1468
static struct iwl_hcmd_ops iwl5000_hcmd = {
1469
	.rxon_assoc = iwl5000_send_rxon_assoc,
1470 1471 1472
};

static struct iwl_hcmd_utils_ops iwl5000_hcmd_utils = {
G
Gregory Greenman 已提交
1473
	.get_hcmd_size = iwl5000_get_hcmd_size,
1474
	.build_addsta_hcmd = iwl5000_build_addsta_hcmd,
1475 1476
	.gain_computation = iwl5000_gain_computation,
	.chain_noise_reset = iwl5000_chain_noise_reset,
1477
	.rts_tx_cmd_flag = iwl5000_rts_tx_cmd_flag,
1478
	.calc_rssi = iwl5000_calc_rssi,
1479 1480 1481
};

static struct iwl_lib_ops iwl5000_lib = {
1482
	.set_hw_params = iwl5000_hw_set_hw_params,
1483
	.txq_update_byte_cnt_tbl = iwl5000_txq_update_byte_cnt_tbl,
1484
	.txq_inval_byte_cnt_tbl = iwl5000_txq_inval_byte_cnt_tbl,
1485
	.txq_set_sched = iwl5000_txq_set_sched,
1486 1487
	.txq_agg_enable = iwl5000_txq_agg_enable,
	.txq_agg_disable = iwl5000_txq_agg_disable,
1488 1489
	.txq_attach_buf_to_tfd = iwl_hw_txq_attach_buf_to_tfd,
	.txq_free_tfd = iwl_hw_txq_free_tfd,
1490
	.txq_init = iwl_hw_tx_queue_init,
1491
	.rx_handler_setup = iwl5000_rx_handler_setup,
1492
	.setup_deferred_work = iwl5000_setup_deferred_work,
1493
	.is_valid_rtc_data_addr = iwl5000_hw_valid_rtc_data_addr,
1494
	.load_ucode = iwl5000_load_ucode,
1495 1496
	.init_alive_start = iwl5000_init_alive_start,
	.alive_notify = iwl5000_alive_notify,
1497
	.send_tx_power = iwl5000_send_tx_power,
1498
	.temperature = iwl5000_temperature,
1499
	.update_chain_flags = iwl_update_chain_flags,
1500 1501
	.apm_ops = {
		.init =	iwl5000_apm_init,
1502
		.reset = iwl5000_apm_reset,
1503
		.stop = iwl5000_apm_stop,
1504
		.config = iwl5000_nic_config,
1505
		.set_pwr_src = iwl_set_pwr_src,
1506
	},
1507
	.eeprom_ops = {
1508 1509 1510 1511 1512 1513 1514 1515 1516
		.regulatory_bands = {
			EEPROM_5000_REG_BAND_1_CHANNELS,
			EEPROM_5000_REG_BAND_2_CHANNELS,
			EEPROM_5000_REG_BAND_3_CHANNELS,
			EEPROM_5000_REG_BAND_4_CHANNELS,
			EEPROM_5000_REG_BAND_5_CHANNELS,
			EEPROM_5000_REG_BAND_24_FAT_CHANNELS,
			EEPROM_5000_REG_BAND_52_FAT_CHANNELS
		},
1517 1518 1519
		.verify_signature  = iwlcore_eeprom_verify_signature,
		.acquire_semaphore = iwlcore_eeprom_acquire_semaphore,
		.release_semaphore = iwlcore_eeprom_release_semaphore,
1520
		.calib_version	= iwl5000_eeprom_calib_version,
1521
		.query_addr = iwl5000_eeprom_query_addr,
1522 1523 1524
	},
};

1525
struct iwl_ops iwl5000_ops = {
1526 1527 1528 1529 1530
	.lib = &iwl5000_lib,
	.hcmd = &iwl5000_hcmd,
	.utils = &iwl5000_hcmd_utils,
};

1531
struct iwl_mod_params iwl50_mod_params = {
1532
	.num_of_queues = IWL50_NUM_QUEUES,
1533
	.num_of_ampdu_queues = IWL50_NUM_AMPDU_QUEUES,
1534
	.amsdu_size_8K = 1,
1535
	.restart_fw = 1,
1536 1537 1538 1539 1540 1541
	/* the rest are 0 by default */
};


struct iwl_cfg iwl5300_agn_cfg = {
	.name = "5300AGN",
1542 1543 1544
	.fw_name_pre = IWL5000_FW_PRE,
	.ucode_api_max = IWL5000_UCODE_API_MAX,
	.ucode_api_min = IWL5000_UCODE_API_MIN,
1545
	.sku = IWL_SKU_A|IWL_SKU_G|IWL_SKU_N,
1546
	.ops = &iwl5000_ops,
1547
	.eeprom_size = IWL_5000_EEPROM_IMG_SIZE,
1548 1549
	.eeprom_ver = EEPROM_5000_EEPROM_VERSION,
	.eeprom_calib_ver = EEPROM_5000_TX_POWER_VERSION,
1550
	.mod_params = &iwl50_mod_params,
1551 1552
	.valid_tx_ant = ANT_ABC,
	.valid_rx_ant = ANT_ABC,
1553
	.need_pll_cfg = true,
1554 1555
};

1556 1557
struct iwl_cfg iwl5100_bg_cfg = {
	.name = "5100BG",
1558 1559 1560
	.fw_name_pre = IWL5000_FW_PRE,
	.ucode_api_max = IWL5000_UCODE_API_MAX,
	.ucode_api_min = IWL5000_UCODE_API_MIN,
1561 1562 1563
	.sku = IWL_SKU_G,
	.ops = &iwl5000_ops,
	.eeprom_size = IWL_5000_EEPROM_IMG_SIZE,
1564 1565
	.eeprom_ver = EEPROM_5000_EEPROM_VERSION,
	.eeprom_calib_ver = EEPROM_5000_TX_POWER_VERSION,
1566
	.mod_params = &iwl50_mod_params,
1567 1568
	.valid_tx_ant = ANT_B,
	.valid_rx_ant = ANT_AB,
1569
	.need_pll_cfg = true,
1570 1571 1572 1573
};

struct iwl_cfg iwl5100_abg_cfg = {
	.name = "5100ABG",
1574 1575 1576
	.fw_name_pre = IWL5000_FW_PRE,
	.ucode_api_max = IWL5000_UCODE_API_MAX,
	.ucode_api_min = IWL5000_UCODE_API_MIN,
1577 1578 1579
	.sku = IWL_SKU_A|IWL_SKU_G,
	.ops = &iwl5000_ops,
	.eeprom_size = IWL_5000_EEPROM_IMG_SIZE,
1580 1581
	.eeprom_ver = EEPROM_5000_EEPROM_VERSION,
	.eeprom_calib_ver = EEPROM_5000_TX_POWER_VERSION,
1582
	.mod_params = &iwl50_mod_params,
1583 1584
	.valid_tx_ant = ANT_B,
	.valid_rx_ant = ANT_AB,
1585
	.need_pll_cfg = true,
1586 1587
};

1588 1589
struct iwl_cfg iwl5100_agn_cfg = {
	.name = "5100AGN",
1590 1591 1592
	.fw_name_pre = IWL5000_FW_PRE,
	.ucode_api_max = IWL5000_UCODE_API_MAX,
	.ucode_api_min = IWL5000_UCODE_API_MIN,
1593
	.sku = IWL_SKU_A|IWL_SKU_G|IWL_SKU_N,
1594
	.ops = &iwl5000_ops,
1595
	.eeprom_size = IWL_5000_EEPROM_IMG_SIZE,
1596 1597
	.eeprom_ver = EEPROM_5000_EEPROM_VERSION,
	.eeprom_calib_ver = EEPROM_5000_TX_POWER_VERSION,
1598
	.mod_params = &iwl50_mod_params,
1599 1600
	.valid_tx_ant = ANT_B,
	.valid_rx_ant = ANT_AB,
1601
	.need_pll_cfg = true,
1602 1603 1604 1605
};

struct iwl_cfg iwl5350_agn_cfg = {
	.name = "5350AGN",
1606 1607 1608
	.fw_name_pre = IWL5000_FW_PRE,
	.ucode_api_max = IWL5000_UCODE_API_MAX,
	.ucode_api_min = IWL5000_UCODE_API_MIN,
1609
	.sku = IWL_SKU_A|IWL_SKU_G|IWL_SKU_N,
1610
	.ops = &iwl5000_ops,
1611
	.eeprom_size = IWL_5000_EEPROM_IMG_SIZE,
1612 1613
	.eeprom_ver = EEPROM_5050_EEPROM_VERSION,
	.eeprom_calib_ver = EEPROM_5050_TX_POWER_VERSION,
1614
	.mod_params = &iwl50_mod_params,
1615 1616
	.valid_tx_ant = ANT_ABC,
	.valid_rx_ant = ANT_ABC,
1617
	.need_pll_cfg = true,
1618 1619
};

T
Tomas Winkler 已提交
1620 1621
struct iwl_cfg iwl5150_agn_cfg = {
	.name = "5150AGN",
1622 1623 1624
	.fw_name_pre = IWL5150_FW_PRE,
	.ucode_api_max = IWL5150_UCODE_API_MAX,
	.ucode_api_min = IWL5150_UCODE_API_MIN,
T
Tomas Winkler 已提交
1625 1626 1627
	.sku = IWL_SKU_A|IWL_SKU_G|IWL_SKU_N,
	.ops = &iwl5000_ops,
	.eeprom_size = IWL_5000_EEPROM_IMG_SIZE,
1628 1629
	.eeprom_ver = EEPROM_5050_EEPROM_VERSION,
	.eeprom_calib_ver = EEPROM_5050_TX_POWER_VERSION,
T
Tomas Winkler 已提交
1630
	.mod_params = &iwl50_mod_params,
1631 1632
	.valid_tx_ant = ANT_A,
	.valid_rx_ant = ANT_AB,
1633
	.need_pll_cfg = true,
T
Tomas Winkler 已提交
1634 1635
};

1636 1637
MODULE_FIRMWARE(IWL5000_MODULE_FIRMWARE(IWL5000_UCODE_API_MAX));
MODULE_FIRMWARE(IWL5150_MODULE_FIRMWARE(IWL5150_UCODE_API_MAX));
1638

1639 1640 1641 1642 1643 1644
module_param_named(disable50, iwl50_mod_params.disable, int, 0444);
MODULE_PARM_DESC(disable50,
		  "manually disable the 50XX radio (default 0 [radio on])");
module_param_named(swcrypto50, iwl50_mod_params.sw_crypto, bool, 0444);
MODULE_PARM_DESC(swcrypto50,
		  "using software crypto engine (default 0 [hardware])\n");
1645
module_param_named(debug50, iwl50_mod_params.debug, uint, 0444);
1646 1647 1648
MODULE_PARM_DESC(debug50, "50XX debug output mask");
module_param_named(queues_num50, iwl50_mod_params.num_of_queues, int, 0444);
MODULE_PARM_DESC(queues_num50, "number of hw queues in 50xx series");
1649 1650
module_param_named(11n_disable50, iwl50_mod_params.disable_11n, int, 0444);
MODULE_PARM_DESC(11n_disable50, "disable 50XX 11n functionality");
1651 1652
module_param_named(amsdu_size_8K50, iwl50_mod_params.amsdu_size_8K, int, 0444);
MODULE_PARM_DESC(amsdu_size_8K50, "enable 8K amsdu size in 50XX series");
1653 1654
module_param_named(fw_restart50, iwl50_mod_params.restart_fw, int, 0444);
MODULE_PARM_DESC(fw_restart50, "restart firmware in case of error");