iwl-4965-tx.c 39.3 KB
Newer Older
1 2 3 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 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
/******************************************************************************
 *
 * GPL LICENSE SUMMARY
 *
 * Copyright(c) 2008 - 2011 Intel Corporation. All rights reserved.
 *
 * 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:
 *  Intel Linux Wireless <ilw@linux.intel.com>
 * 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/sched.h>

#include "iwl-dev.h"
#include "iwl-core.h"
#include "iwl-sta.h"
#include "iwl-io.h"
#include "iwl-helpers.h"
#include "iwl-4965-hw.h"
#include "iwl-4965.h"

/*
 * mac80211 queues, ACs, hardware queues, FIFOs.
 *
 * Cf. http://wireless.kernel.org/en/developers/Documentation/mac80211/queues
 *
 * Mac80211 uses the following numbers, which we get as from it
 * by way of skb_get_queue_mapping(skb):
 *
 *	VO	0
 *	VI	1
 *	BE	2
 *	BK	3
 *
 *
 * Regular (not A-MPDU) frames are put into hardware queues corresponding
 * to the FIFOs, see comments in iwl-prph.h. Aggregated frames get their
 * own queue per aggregation session (RA/TID combination), such queues are
 * set up to map into FIFOs too, for which we need an AC->FIFO mapping. In
 * order to map frames to the right queue, we also need an AC->hw queue
 * mapping. This is implemented here.
 *
 * Due to the way hw queues are set up (by the hw specific modules like
 * iwl-4965.c), the AC->hw queue mapping is the identity
 * mapping.
 */

static const u8 tid_to_ac[] = {
	IEEE80211_AC_BE,
	IEEE80211_AC_BK,
	IEEE80211_AC_BK,
	IEEE80211_AC_BE,
	IEEE80211_AC_VI,
	IEEE80211_AC_VI,
	IEEE80211_AC_VO,
	IEEE80211_AC_VO
};

S
Stanislaw Gruszka 已提交
80
static inline int il4965_get_ac_from_tid(u16 tid)
81 82 83 84 85 86 87 88 89
{
	if (likely(tid < ARRAY_SIZE(tid_to_ac)))
		return tid_to_ac[tid];

	/* no support for TIDs 8-15 yet */
	return -EINVAL;
}

static inline int
S
Stanislaw Gruszka 已提交
90
il4965_get_fifo_from_tid(struct il_rxon_context *ctx, u16 tid)
91 92 93 94 95 96 97 98 99 100 101
{
	if (likely(tid < ARRAY_SIZE(tid_to_ac)))
		return ctx->ac_to_fifo[tid_to_ac[tid]];

	/* no support for TIDs 8-15 yet */
	return -EINVAL;
}

/*
 * handle build REPLY_TX command notification.
 */
S
Stanislaw Gruszka 已提交
102
static void il4965_tx_cmd_build_basic(struct il_priv *priv,
103
					struct sk_buff *skb,
S
Stanislaw Gruszka 已提交
104
					struct il_tx_cmd *tx_cmd,
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
					struct ieee80211_tx_info *info,
					struct ieee80211_hdr *hdr,
					u8 std_id)
{
	__le16 fc = hdr->frame_control;
	__le32 tx_flags = tx_cmd->tx_flags;

	tx_cmd->stop_time.life_time = TX_CMD_LIFE_TIME_INFINITE;
	if (!(info->flags & IEEE80211_TX_CTL_NO_ACK)) {
		tx_flags |= TX_CMD_FLG_ACK_MSK;
		if (ieee80211_is_mgmt(fc))
			tx_flags |= TX_CMD_FLG_SEQ_CTL_MSK;
		if (ieee80211_is_probe_resp(fc) &&
		    !(le16_to_cpu(hdr->seq_ctrl) & 0xf))
			tx_flags |= TX_CMD_FLG_TSF_MSK;
	} else {
		tx_flags &= (~TX_CMD_FLG_ACK_MSK);
		tx_flags |= TX_CMD_FLG_SEQ_CTL_MSK;
	}

	if (ieee80211_is_back_req(fc))
		tx_flags |= TX_CMD_FLG_ACK_MSK | TX_CMD_FLG_IMM_BA_RSP_MASK;

	tx_cmd->sta_id = std_id;
	if (ieee80211_has_morefrags(fc))
		tx_flags |= TX_CMD_FLG_MORE_FRAG_MSK;

	if (ieee80211_is_data_qos(fc)) {
		u8 *qc = ieee80211_get_qos_ctl(hdr);
		tx_cmd->tid_tspec = qc[0] & 0xf;
		tx_flags &= ~TX_CMD_FLG_SEQ_CTL_MSK;
	} else {
		tx_flags |= TX_CMD_FLG_SEQ_CTL_MSK;
	}

S
Stanislaw Gruszka 已提交
140
	il_tx_cmd_protection(priv, info, fc, &tx_flags);
141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158

	tx_flags &= ~(TX_CMD_FLG_ANT_SEL_MSK);
	if (ieee80211_is_mgmt(fc)) {
		if (ieee80211_is_assoc_req(fc) || ieee80211_is_reassoc_req(fc))
			tx_cmd->timeout.pm_frame_timeout = cpu_to_le16(3);
		else
			tx_cmd->timeout.pm_frame_timeout = cpu_to_le16(2);
	} else {
		tx_cmd->timeout.pm_frame_timeout = 0;
	}

	tx_cmd->driver_txop = 0;
	tx_cmd->tx_flags = tx_flags;
	tx_cmd->next_frame_len = 0;
}

#define RTS_DFAULT_RETRY_LIMIT		60

S
Stanislaw Gruszka 已提交
159 160
static void il4965_tx_cmd_build_rate(struct il_priv *priv,
			      struct il_tx_cmd *tx_cmd,
161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198
			      struct ieee80211_tx_info *info,
			      __le16 fc)
{
	u32 rate_flags;
	int rate_idx;
	u8 rts_retry_limit;
	u8 data_retry_limit;
	u8 rate_plcp;

	/* Set retry limit on DATA packets and Probe Responses*/
	if (ieee80211_is_probe_resp(fc))
		data_retry_limit = 3;
	else
		data_retry_limit = IWL4965_DEFAULT_TX_RETRY;
	tx_cmd->data_retry_limit = data_retry_limit;

	/* Set retry limit on RTS packets */
	rts_retry_limit = RTS_DFAULT_RETRY_LIMIT;
	if (data_retry_limit < rts_retry_limit)
		rts_retry_limit = data_retry_limit;
	tx_cmd->rts_retry_limit = rts_retry_limit;

	/* DATA packets will use the uCode station table for rate/antenna
	 * selection */
	if (ieee80211_is_data(fc)) {
		tx_cmd->initial_rate_index = 0;
		tx_cmd->tx_flags |= TX_CMD_FLG_STA_RATE_MSK;
		return;
	}

	/**
	 * If the current TX rate stored in mac80211 has the MCS bit set, it's
	 * not really a TX rate.  Thus, we use the lowest supported rate for
	 * this band.  Also use the lowest supported rate if the stored rate
	 * index is invalid.
	 */
	rate_idx = info->control.rates[0].idx;
	if (info->control.rates[0].flags & IEEE80211_TX_RC_MCS ||
S
Stanislaw Gruszka 已提交
199
			(rate_idx < 0) || (rate_idx > IL_RATE_COUNT_LEGACY))
200 201 202 203
		rate_idx = rate_lowest_index(&priv->bands[info->band],
				info->control.sta);
	/* For 5 GHZ band, remap mac80211 rate indices into driver indices */
	if (info->band == IEEE80211_BAND_5GHZ)
S
Stanislaw Gruszka 已提交
204
		rate_idx += IL_FIRST_OFDM_RATE;
205
	/* Get PLCP rate for tx_cmd->rate_n_flags */
206
	rate_plcp = iwlegacy_rates[rate_idx].plcp;
207 208 209 210
	/* Zero out flags for this packet */
	rate_flags = 0;

	/* Set CCK flag as needed */
S
Stanislaw Gruszka 已提交
211
	if ((rate_idx >= IL_FIRST_CCK_RATE) && (rate_idx <= IL_LAST_CCK_RATE))
212 213 214
		rate_flags |= RATE_MCS_CCK_MSK;

	/* Set up antennas */
S
Stanislaw Gruszka 已提交
215
	priv->mgmt_tx_ant = il4965_toggle_tx_ant(priv, priv->mgmt_tx_ant,
216 217
				      priv->hw_params.valid_tx_ant);

S
Stanislaw Gruszka 已提交
218
	rate_flags |= il4965_ant_idx_to_flags(priv->mgmt_tx_ant);
219 220

	/* Set the rate in the TX cmd */
S
Stanislaw Gruszka 已提交
221
	tx_cmd->rate_n_flags = il4965_hw_set_rate_n_flags(rate_plcp, rate_flags);
222 223
}

S
Stanislaw Gruszka 已提交
224
static void il4965_tx_cmd_build_hwcrypto(struct il_priv *priv,
225
				      struct ieee80211_tx_info *info,
S
Stanislaw Gruszka 已提交
226
				      struct il_tx_cmd *tx_cmd,
227 228 229 230 231 232 233 234 235 236 237
				      struct sk_buff *skb_frag,
				      int sta_id)
{
	struct ieee80211_key_conf *keyconf = info->control.hw_key;

	switch (keyconf->cipher) {
	case WLAN_CIPHER_SUITE_CCMP:
		tx_cmd->sec_ctl = TX_CMD_SEC_CCM;
		memcpy(tx_cmd->key, keyconf->key, keyconf->keylen);
		if (info->flags & IEEE80211_TX_CTL_AMPDU)
			tx_cmd->tx_flags |= TX_CMD_FLG_AGG_CCMP_MSK;
S
Stanislaw Gruszka 已提交
238
		IL_DEBUG_TX(priv, "tx_cmd with AES hwcrypto\n");
239 240 241 242
		break;

	case WLAN_CIPHER_SUITE_TKIP:
		tx_cmd->sec_ctl = TX_CMD_SEC_TKIP;
243
		ieee80211_get_tkip_p2k(keyconf, skb_frag, tx_cmd->key);
S
Stanislaw Gruszka 已提交
244
		IL_DEBUG_TX(priv, "tx_cmd with tkip hwcrypto\n");
245 246 247 248 249 250 251 252 253 254 255
		break;

	case WLAN_CIPHER_SUITE_WEP104:
		tx_cmd->sec_ctl |= TX_CMD_SEC_KEY128;
		/* fall through */
	case WLAN_CIPHER_SUITE_WEP40:
		tx_cmd->sec_ctl |= (TX_CMD_SEC_WEP |
			(keyconf->keyidx & TX_CMD_SEC_MSK) << TX_CMD_SEC_SHIFT);

		memcpy(&tx_cmd->key[3], keyconf->key, keyconf->keylen);

S
Stanislaw Gruszka 已提交
256
		IL_DEBUG_TX(priv, "Configuring packet for WEP encryption "
257 258 259 260
			     "with key %d\n", keyconf->keyidx);
		break;

	default:
S
Stanislaw Gruszka 已提交
261
		IL_ERR(priv, "Unknown encode cipher %x\n", keyconf->cipher);
262 263 264 265 266 267 268
		break;
	}
}

/*
 * start REPLY_TX command process
 */
S
Stanislaw Gruszka 已提交
269
int il4965_tx_skb(struct il_priv *priv, struct sk_buff *skb)
270 271 272 273
{
	struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
	struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
	struct ieee80211_sta *sta = info->control.sta;
S
Stanislaw Gruszka 已提交
274 275 276 277 278 279 280
	struct il_station_priv *sta_priv = NULL;
	struct il_tx_queue *txq;
	struct il_queue *q;
	struct il_device_cmd *out_cmd;
	struct il_cmd_meta *out_meta;
	struct il_tx_cmd *tx_cmd;
	struct il_rxon_context *ctx = &priv->contexts[IL_RXON_CTX_BSS];
281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296
	int txq_id;
	dma_addr_t phys_addr;
	dma_addr_t txcmd_phys;
	dma_addr_t scratch_phys;
	u16 len, firstlen, secondlen;
	u16 seq_number = 0;
	__le16 fc;
	u8 hdr_len;
	u8 sta_id;
	u8 wait_write_ptr = 0;
	u8 tid = 0;
	u8 *qc = NULL;
	unsigned long flags;
	bool is_agg = false;

	if (info->control.vif)
S
Stanislaw Gruszka 已提交
297
		ctx = il_rxon_ctx_from_vif(info->control.vif);
298 299

	spin_lock_irqsave(&priv->lock, flags);
S
Stanislaw Gruszka 已提交
300 301
	if (il_is_rfkill(priv)) {
		IL_DEBUG_DROP(priv, "Dropping - RF KILL\n");
302 303 304 305 306 307 308
		goto drop_unlock;
	}

	fc = hdr->frame_control;

#ifdef CONFIG_IWLWIFI_LEGACY_DEBUG
	if (ieee80211_is_auth(fc))
S
Stanislaw Gruszka 已提交
309
		IL_DEBUG_TX(priv, "Sending AUTH frame\n");
310
	else if (ieee80211_is_assoc_req(fc))
S
Stanislaw Gruszka 已提交
311
		IL_DEBUG_TX(priv, "Sending ASSOC frame\n");
312
	else if (ieee80211_is_reassoc_req(fc))
S
Stanislaw Gruszka 已提交
313
		IL_DEBUG_TX(priv, "Sending REASSOC frame\n");
314 315 316 317
#endif

	hdr_len = ieee80211_hdrlen(fc);

318 319 320 321 322
	/* For management frames use broadcast id to do not break aggregation */
	if (!ieee80211_is_data(fc))
		sta_id = ctx->bcast_sta_id;
	else {
		/* Find index into station table for destination station */
S
Stanislaw Gruszka 已提交
323
		sta_id = il_sta_id_or_broadcast(priv, ctx, info->control.sta);
324

S
Stanislaw Gruszka 已提交
325 326
		if (sta_id == IL_INVALID_STATION) {
			IL_DEBUG_DROP(priv, "Dropping - INVALID STATION: %pM\n",
327 328 329
				       hdr->addr1);
			goto drop_unlock;
		}
330 331
	}

S
Stanislaw Gruszka 已提交
332
	IL_DEBUG_TX(priv, "station Id %d\n", sta_id);
333 334 335 336 337

	if (sta)
		sta_priv = (void *)sta->drv_priv;

	if (sta_priv && sta_priv->asleep &&
J
Johannes Berg 已提交
338
	    (info->flags & IEEE80211_TX_CTL_POLL_RESPONSE)) {
339 340 341 342 343 344 345 346 347
		/*
		 * This sends an asynchronous command to the device,
		 * but we can rely on it being processed before the
		 * next frame is processed -- and the next frame to
		 * this station is the one that will consume this
		 * counter.
		 * For now set the counter to just 1 since we do not
		 * support uAPSD yet.
		 */
S
Stanislaw Gruszka 已提交
348
		il4965_sta_modify_sleep_tx_count(priv, sta_id, 1);
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383
	}

	/*
	 * Send this frame after DTIM -- there's a special queue
	 * reserved for this for contexts that support AP mode.
	 */
	if (info->flags & IEEE80211_TX_CTL_SEND_AFTER_DTIM) {
		txq_id = ctx->mcast_queue;
		/*
		 * The microcode will clear the more data
		 * bit in the last frame it transmits.
		 */
		hdr->frame_control |=
			cpu_to_le16(IEEE80211_FCTL_MOREDATA);
	} else
		txq_id = ctx->ac_to_queue[skb_get_queue_mapping(skb)];

	/* irqs already disabled/saved above when locking priv->lock */
	spin_lock(&priv->sta_lock);

	if (ieee80211_is_data_qos(fc)) {
		qc = ieee80211_get_qos_ctl(hdr);
		tid = qc[0] & IEEE80211_QOS_CTL_TID_MASK;
		if (WARN_ON_ONCE(tid >= MAX_TID_COUNT)) {
			spin_unlock(&priv->sta_lock);
			goto drop_unlock;
		}
		seq_number = priv->stations[sta_id].tid[tid].seq_number;
		seq_number &= IEEE80211_SCTL_SEQ;
		hdr->seq_ctrl = hdr->seq_ctrl &
				cpu_to_le16(IEEE80211_SCTL_FRAG);
		hdr->seq_ctrl |= cpu_to_le16(seq_number);
		seq_number += 0x10;
		/* aggregation is on for this <sta,tid> */
		if (info->flags & IEEE80211_TX_CTL_AMPDU &&
S
Stanislaw Gruszka 已提交
384
		    priv->stations[sta_id].tid[tid].agg.state == IL_AGG_ON) {
385 386 387 388 389 390 391 392
			txq_id = priv->stations[sta_id].tid[tid].agg.txq_id;
			is_agg = true;
		}
	}

	txq = &priv->txq[txq_id];
	q = &txq->q;

S
Stanislaw Gruszka 已提交
393
	if (unlikely(il_queue_space(q) < q->high_mark)) {
394 395 396 397 398 399 400 401 402 403 404 405 406
		spin_unlock(&priv->sta_lock);
		goto drop_unlock;
	}

	if (ieee80211_is_data_qos(fc)) {
		priv->stations[sta_id].tid[tid].tfds_in_queue++;
		if (!ieee80211_has_morefrags(fc))
			priv->stations[sta_id].tid[tid].seq_number = seq_number;
	}

	spin_unlock(&priv->sta_lock);

	/* Set up driver data for this TFD */
S
Stanislaw Gruszka 已提交
407
	memset(&(txq->txb[q->write_ptr]), 0, sizeof(struct il_tx_info));
408 409 410 411 412 413 414 415
	txq->txb[q->write_ptr].skb = skb;
	txq->txb[q->write_ptr].ctx = ctx;

	/* Set up first empty entry in queue's array of Tx/cmd buffers */
	out_cmd = txq->cmd[q->write_ptr];
	out_meta = &txq->meta[q->write_ptr];
	tx_cmd = &out_cmd->cmd.tx;
	memset(&out_cmd->hdr, 0, sizeof(out_cmd->hdr));
S
Stanislaw Gruszka 已提交
416
	memset(tx_cmd, 0, sizeof(struct il_tx_cmd));
417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436

	/*
	 * Set up the Tx-command (not MAC!) header.
	 * Store the chosen Tx queue and TFD index within the sequence field;
	 * after Tx, uCode's Tx response will return this value so driver can
	 * locate the frame within the tx queue and do post-tx processing.
	 */
	out_cmd->hdr.cmd = REPLY_TX;
	out_cmd->hdr.sequence = cpu_to_le16((u16)(QUEUE_TO_SEQ(txq_id) |
				INDEX_TO_SEQ(q->write_ptr)));

	/* Copy MAC header from skb into command buffer */
	memcpy(tx_cmd->hdr, hdr, hdr_len);


	/* Total # bytes to be transmitted */
	len = (u16)skb->len;
	tx_cmd->len = cpu_to_le16(len);

	if (info->control.hw_key)
S
Stanislaw Gruszka 已提交
437
		il4965_tx_cmd_build_hwcrypto(priv, info, tx_cmd, skb, sta_id);
438 439

	/* TODO need this for burst mode later on */
S
Stanislaw Gruszka 已提交
440 441
	il4965_tx_cmd_build_basic(priv, skb, tx_cmd, info, hdr, sta_id);
	il_dbg_log_tx_data_frame(priv, len, hdr);
442

S
Stanislaw Gruszka 已提交
443
	il4965_tx_cmd_build_rate(priv, tx_cmd, info, fc);
444

S
Stanislaw Gruszka 已提交
445
	il_update_stats(priv, true, fc, len);
446 447 448 449 450 451 452 453 454
	/*
	 * Use the first empty entry in this queue's command buffer array
	 * to contain the Tx command and MAC header concatenated together
	 * (payload data will be in another buffer).
	 * Size of this varies, due to varying MAC header length.
	 * If end is not dword aligned, we'll have 2 extra bytes at the end
	 * of the MAC header (device reads on dword boundaries).
	 * We'll tell device about this padding later.
	 */
S
Stanislaw Gruszka 已提交
455 456
	len = sizeof(struct il_tx_cmd) +
		sizeof(struct il_cmd_header) + hdr_len;
457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492
	firstlen = (len + 3) & ~3;

	/* Tell NIC about any 2-byte padding after MAC header */
	if (firstlen != len)
		tx_cmd->tx_flags |= TX_CMD_FLG_MH_PAD_MSK;

	/* Physical address of this Tx command's header (not MAC header!),
	 * within command buffer array. */
	txcmd_phys = pci_map_single(priv->pci_dev,
				    &out_cmd->hdr, firstlen,
				    PCI_DMA_BIDIRECTIONAL);
	dma_unmap_addr_set(out_meta, mapping, txcmd_phys);
	dma_unmap_len_set(out_meta, len, firstlen);
	/* Add buffer containing Tx command and MAC(!) header to TFD's
	 * first entry */
	priv->cfg->ops->lib->txq_attach_buf_to_tfd(priv, txq,
						   txcmd_phys, firstlen, 1, 0);

	if (!ieee80211_has_morefrags(hdr->frame_control)) {
		txq->need_update = 1;
	} else {
		wait_write_ptr = 1;
		txq->need_update = 0;
	}

	/* Set up TFD's 2nd entry to point directly to remainder of skb,
	 * if any (802.11 null frames have no payload). */
	secondlen = skb->len - hdr_len;
	if (secondlen > 0) {
		phys_addr = pci_map_single(priv->pci_dev, skb->data + hdr_len,
					   secondlen, PCI_DMA_TODEVICE);
		priv->cfg->ops->lib->txq_attach_buf_to_tfd(priv, txq,
							   phys_addr, secondlen,
							   0, 0);
	}

S
Stanislaw Gruszka 已提交
493 494
	scratch_phys = txcmd_phys + sizeof(struct il_cmd_header) +
				offsetof(struct il_tx_cmd, scratch);
495 496 497 498 499

	/* take back ownership of DMA buffer to enable update */
	pci_dma_sync_single_for_cpu(priv->pci_dev, txcmd_phys,
				    firstlen, PCI_DMA_BIDIRECTIONAL);
	tx_cmd->dram_lsb_ptr = cpu_to_le32(scratch_phys);
S
Stanislaw Gruszka 已提交
500
	tx_cmd->dram_msb_ptr = il_get_dma_hi_addr(scratch_phys);
501

S
Stanislaw Gruszka 已提交
502
	IL_DEBUG_TX(priv, "sequence nr = 0X%x\n",
503
		     le16_to_cpu(out_cmd->hdr.sequence));
S
Stanislaw Gruszka 已提交
504 505 506
	IL_DEBUG_TX(priv, "tx_flags = 0X%x\n", le32_to_cpu(tx_cmd->tx_flags));
	il_print_hex_dump(priv, IL_DL_TX, (u8 *)tx_cmd, sizeof(*tx_cmd));
	il_print_hex_dump(priv, IL_DL_TX, (u8 *)tx_cmd->hdr, hdr_len);
507 508 509 510 511 512 513 514 515 516

	/* Set up entry for this TFD in Tx byte-count array */
	if (info->flags & IEEE80211_TX_CTL_AMPDU)
		priv->cfg->ops->lib->txq_update_byte_cnt_tbl(priv, txq,
						     le16_to_cpu(tx_cmd->len));

	pci_dma_sync_single_for_device(priv->pci_dev, txcmd_phys,
				       firstlen, PCI_DMA_BIDIRECTIONAL);

	/* Tell device the write index *just past* this latest filled TFD */
S
Stanislaw Gruszka 已提交
517 518
	q->write_ptr = il_queue_inc_wrap(q->write_ptr, q->n_bd);
	il_txq_update_write_ptr(priv, txq);
519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537
	spin_unlock_irqrestore(&priv->lock, flags);

	/*
	 * At this point the frame is "transmitted" successfully
	 * and we will get a TX status notification eventually,
	 * regardless of the value of ret. "ret" only indicates
	 * whether or not we should update the write pointer.
	 */

	/*
	 * Avoid atomic ops if it isn't an associated client.
	 * Also, if this is a packet for aggregation, don't
	 * increase the counter because the ucode will stop
	 * aggregation queues when their respective station
	 * goes to sleep.
	 */
	if (sta_priv && sta_priv->client && !is_agg)
		atomic_inc(&sta_priv->pending_frames);

S
Stanislaw Gruszka 已提交
538
	if ((il_queue_space(q) < q->high_mark) &&
539 540 541 542
			priv->mac80211_registered) {
		if (wait_write_ptr) {
			spin_lock_irqsave(&priv->lock, flags);
			txq->need_update = 1;
S
Stanislaw Gruszka 已提交
543
			il_txq_update_write_ptr(priv, txq);
544 545
			spin_unlock_irqrestore(&priv->lock, flags);
		} else {
S
Stanislaw Gruszka 已提交
546
			il_stop_queue(priv, txq);
547 548 549 550 551 552 553 554 555 556
		}
	}

	return 0;

drop_unlock:
	spin_unlock_irqrestore(&priv->lock, flags);
	return -1;
}

S
Stanislaw Gruszka 已提交
557 558
static inline int il4965_alloc_dma_ptr(struct il_priv *priv,
				    struct il_dma_ptr *ptr, size_t size)
559 560 561 562 563 564 565 566 567
{
	ptr->addr = dma_alloc_coherent(&priv->pci_dev->dev, size, &ptr->dma,
				       GFP_KERNEL);
	if (!ptr->addr)
		return -ENOMEM;
	ptr->size = size;
	return 0;
}

S
Stanislaw Gruszka 已提交
568 569
static inline void il4965_free_dma_ptr(struct il_priv *priv,
				    struct il_dma_ptr *ptr)
570 571 572 573 574 575 576 577 578
{
	if (unlikely(!ptr->addr))
		return;

	dma_free_coherent(&priv->pci_dev->dev, ptr->size, ptr->addr, ptr->dma);
	memset(ptr, 0, sizeof(*ptr));
}

/**
S
Stanislaw Gruszka 已提交
579
 * il4965_hw_txq_ctx_free - Free TXQ Context
580 581 582
 *
 * Destroy all TX DMA queues and structures
 */
S
Stanislaw Gruszka 已提交
583
void il4965_hw_txq_ctx_free(struct il_priv *priv)
584 585 586 587 588 589 590
{
	int txq_id;

	/* Tx queues */
	if (priv->txq) {
		for (txq_id = 0; txq_id < priv->hw_params.max_txq_num; txq_id++)
			if (txq_id == priv->cmd_queue)
S
Stanislaw Gruszka 已提交
591
				il_cmd_queue_free(priv);
592
			else
S
Stanislaw Gruszka 已提交
593
				il_tx_queue_free(priv, txq_id);
594
	}
S
Stanislaw Gruszka 已提交
595
	il4965_free_dma_ptr(priv, &priv->kw);
596

S
Stanislaw Gruszka 已提交
597
	il4965_free_dma_ptr(priv, &priv->scd_bc_tbls);
598 599

	/* free tx queue structure */
S
Stanislaw Gruszka 已提交
600
	il_txq_mem(priv);
601 602 603
}

/**
S
Stanislaw Gruszka 已提交
604
 * il4965_txq_ctx_alloc - allocate TX queue context
605 606 607 608 609
 * Allocate all Tx DMA structures and initialize them
 *
 * @param priv
 * @return error code
 */
S
Stanislaw Gruszka 已提交
610
int il4965_txq_ctx_alloc(struct il_priv *priv)
611 612 613 614 615 616
{
	int ret;
	int txq_id, slots_num;
	unsigned long flags;

	/* Free all tx/cmd queues and keep-warm buffer */
S
Stanislaw Gruszka 已提交
617
	il4965_hw_txq_ctx_free(priv);
618

S
Stanislaw Gruszka 已提交
619
	ret = il4965_alloc_dma_ptr(priv, &priv->scd_bc_tbls,
620 621
				priv->hw_params.scd_bc_tbls_size);
	if (ret) {
S
Stanislaw Gruszka 已提交
622
		IL_ERR(priv, "Scheduler BC Table allocation failed\n");
623 624 625
		goto error_bc_tbls;
	}
	/* Alloc keep-warm buffer */
S
Stanislaw Gruszka 已提交
626
	ret = il4965_alloc_dma_ptr(priv, &priv->kw, IL_KW_SIZE);
627
	if (ret) {
S
Stanislaw Gruszka 已提交
628
		IL_ERR(priv, "Keep Warm allocation failed\n");
629 630 631 632
		goto error_kw;
	}

	/* allocate tx queue structure */
S
Stanislaw Gruszka 已提交
633
	ret = il_alloc_txq_mem(priv);
634 635 636 637 638 639
	if (ret)
		goto error;

	spin_lock_irqsave(&priv->lock, flags);

	/* Turn off all Tx DMA fifos */
S
Stanislaw Gruszka 已提交
640
	il4965_txq_set_sched(priv, 0);
641 642

	/* Tell NIC where to find the "keep warm" buffer */
S
Stanislaw Gruszka 已提交
643
	il_write_direct32(priv, FH_KW_MEM_ADDR_REG, priv->kw.dma >> 4);
644 645 646 647 648 649 650

	spin_unlock_irqrestore(&priv->lock, flags);

	/* Alloc and init all Tx queues, including the command queue (#4/#9) */
	for (txq_id = 0; txq_id < priv->hw_params.max_txq_num; txq_id++) {
		slots_num = (txq_id == priv->cmd_queue) ?
					TFD_CMD_SLOTS : TFD_TX_CMD_SLOTS;
S
Stanislaw Gruszka 已提交
651
		ret = il_tx_queue_init(priv,
652 653 654
					&priv->txq[txq_id], slots_num,
				       txq_id);
		if (ret) {
S
Stanislaw Gruszka 已提交
655
			IL_ERR(priv, "Tx %d queue init failed\n", txq_id);
656 657 658 659 660 661 662
			goto error;
		}
	}

	return ret;

 error:
S
Stanislaw Gruszka 已提交
663 664
	il4965_hw_txq_ctx_free(priv);
	il4965_free_dma_ptr(priv, &priv->kw);
665
 error_kw:
S
Stanislaw Gruszka 已提交
666
	il4965_free_dma_ptr(priv, &priv->scd_bc_tbls);
667 668 669 670
 error_bc_tbls:
	return ret;
}

S
Stanislaw Gruszka 已提交
671
void il4965_txq_ctx_reset(struct il_priv *priv)
672 673 674 675 676 677 678
{
	int txq_id, slots_num;
	unsigned long flags;

	spin_lock_irqsave(&priv->lock, flags);

	/* Turn off all Tx DMA fifos */
S
Stanislaw Gruszka 已提交
679
	il4965_txq_set_sched(priv, 0);
680 681

	/* Tell NIC where to find the "keep warm" buffer */
S
Stanislaw Gruszka 已提交
682
	il_write_direct32(priv, FH_KW_MEM_ADDR_REG, priv->kw.dma >> 4);
683 684 685 686 687 688 689

	spin_unlock_irqrestore(&priv->lock, flags);

	/* Alloc and init all Tx queues, including the command queue (#4) */
	for (txq_id = 0; txq_id < priv->hw_params.max_txq_num; txq_id++) {
		slots_num = txq_id == priv->cmd_queue ?
			    TFD_CMD_SLOTS : TFD_TX_CMD_SLOTS;
S
Stanislaw Gruszka 已提交
690
		il_tx_queue_reset(priv, &priv->txq[txq_id],
691 692 693 694 695
						slots_num, txq_id);
	}
}

/**
S
Stanislaw Gruszka 已提交
696
 * il4965_txq_ctx_stop - Stop all Tx DMA channels
697
 */
S
Stanislaw Gruszka 已提交
698
void il4965_txq_ctx_stop(struct il_priv *priv)
699
{
700
	int ch, txq_id;
701 702 703 704 705
	unsigned long flags;

	/* Turn off all Tx DMA fifos */
	spin_lock_irqsave(&priv->lock, flags);

S
Stanislaw Gruszka 已提交
706
	il4965_txq_set_sched(priv, 0);
707 708 709

	/* Stop each Tx DMA channel, and wait for it to be idle */
	for (ch = 0; ch < priv->hw_params.dma_chnl_num; ch++) {
S
Stanislaw Gruszka 已提交
710
		il_write_direct32(priv,
711
				FH_TCSR_CHNL_TX_CONFIG_REG(ch), 0x0);
S
Stanislaw Gruszka 已提交
712
		if (il_poll_direct_bit(priv, FH_TSSR_TX_STATUS_REG,
713 714
				    FH_TSSR_TX_STATUS_REG_MSK_CHNL_IDLE(ch),
				    1000))
S
Stanislaw Gruszka 已提交
715
			IL_ERR(priv, "Failing on timeout while stopping"
716
			    " DMA channel %d [0x%08x]", ch,
S
Stanislaw Gruszka 已提交
717
			    il_read_direct32(priv,
718 719 720
					FH_TSSR_TX_STATUS_REG));
	}
	spin_unlock_irqrestore(&priv->lock, flags);
721 722 723 724 725 726 727

	if (!priv->txq)
		return;

	/* Unmap DMA from host system and free skb's */
	for (txq_id = 0; txq_id < priv->hw_params.max_txq_num; txq_id++)
		if (txq_id == priv->cmd_queue)
S
Stanislaw Gruszka 已提交
728
			il_cmd_queue_unmap(priv);
729
		else
S
Stanislaw Gruszka 已提交
730
			il_tx_queue_unmap(priv, txq_id);
731 732 733 734 735 736 737 738
}

/*
 * Find first available (lowest unused) Tx Queue, mark it "active".
 * Called only when finding queue for aggregation.
 * Should never return anything < 7, because they should already
 * be in use as EDCA AC (0-3), Command (4), reserved (5, 6)
 */
S
Stanislaw Gruszka 已提交
739
static int il4965_txq_ctx_activate_free(struct il_priv *priv)
740 741 742 743 744 745 746 747 748 749
{
	int txq_id;

	for (txq_id = 0; txq_id < priv->hw_params.max_txq_num; txq_id++)
		if (!test_and_set_bit(txq_id, &priv->txq_ctx_active_msk))
			return txq_id;
	return -1;
}

/**
S
Stanislaw Gruszka 已提交
750
 * il4965_tx_queue_stop_scheduler - Stop queue, but keep configuration
751
 */
S
Stanislaw Gruszka 已提交
752
static void il4965_tx_queue_stop_scheduler(struct il_priv *priv,
753 754 755 756
					    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. */
S
Stanislaw Gruszka 已提交
757
	il_write_prph(priv,
758 759 760 761 762 763
		IWL49_SCD_QUEUE_STATUS_BITS(txq_id),
		(0 << IWL49_SCD_QUEUE_STTS_REG_POS_ACTIVE)|
		(1 << IWL49_SCD_QUEUE_STTS_REG_POS_SCD_ACT_EN));
}

/**
S
Stanislaw Gruszka 已提交
764
 * il4965_tx_queue_set_q2ratid - Map unique receiver/tid combination to a queue
765
 */
S
Stanislaw Gruszka 已提交
766
static int il4965_tx_queue_set_q2ratid(struct il_priv *priv, u16 ra_tid,
767 768 769 770 771 772
					u16 txq_id)
{
	u32 tbl_dw_addr;
	u32 tbl_dw;
	u16 scd_q2ratid;

S
Stanislaw Gruszka 已提交
773
	scd_q2ratid = ra_tid & IL_SCD_QUEUE_RA_TID_MAP_RATID_MSK;
774 775 776 777

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

S
Stanislaw Gruszka 已提交
778
	tbl_dw = il_read_targ_mem(priv, tbl_dw_addr);
779 780 781 782 783 784

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

S
Stanislaw Gruszka 已提交
785
	il_write_targ_mem(priv, tbl_dw_addr, tbl_dw);
786 787 788 789 790

	return 0;
}

/**
S
Stanislaw Gruszka 已提交
791
 * il4965_tx_queue_agg_enable - Set up & enable aggregation for selected queue
792 793 794 795
 *
 * NOTE:  txq_id must be greater than IWL49_FIRST_AMPDU_QUEUE,
 *        i.e. it must be one of the higher queues used for aggregation
 */
S
Stanislaw Gruszka 已提交
796
static int il4965_txq_agg_enable(struct il_priv *priv, int txq_id,
797 798 799 800 801 802 803 804 805
				  int tx_fifo, int sta_id, int tid, u16 ssn_idx)
{
	unsigned long flags;
	u16 ra_tid;
	int ret;

	if ((IWL49_FIRST_AMPDU_QUEUE > txq_id) ||
	    (IWL49_FIRST_AMPDU_QUEUE +
		priv->cfg->base_params->num_of_ampdu_queues <= txq_id)) {
S
Stanislaw Gruszka 已提交
806
		IL_WARN(priv,
807 808 809 810 811 812 813 814 815 816
			"queue number out of range: %d, must be %d to %d\n",
			txq_id, IWL49_FIRST_AMPDU_QUEUE,
			IWL49_FIRST_AMPDU_QUEUE +
			priv->cfg->base_params->num_of_ampdu_queues - 1);
		return -EINVAL;
	}

	ra_tid = BUILD_RAxTID(sta_id, tid);

	/* Modify device's station table to Tx this TID */
S
Stanislaw Gruszka 已提交
817
	ret = il4965_sta_tx_modify_enable_tid(priv, sta_id, tid);
818 819 820 821 822 823
	if (ret)
		return ret;

	spin_lock_irqsave(&priv->lock, flags);

	/* Stop this Tx queue before configuring it */
S
Stanislaw Gruszka 已提交
824
	il4965_tx_queue_stop_scheduler(priv, txq_id);
825 826

	/* Map receiver-address / traffic-ID to this queue */
S
Stanislaw Gruszka 已提交
827
	il4965_tx_queue_set_q2ratid(priv, ra_tid, txq_id);
828 829

	/* Set this queue as a chain-building queue */
S
Stanislaw Gruszka 已提交
830
	il_set_bits_prph(priv, IWL49_SCD_QUEUECHAIN_SEL, (1 << txq_id));
831 832 833 834 835

	/* 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);
S
Stanislaw Gruszka 已提交
836
	il4965_set_wr_ptrs(priv, txq_id, ssn_idx);
837 838

	/* Set up Tx window size and frame limit for this queue */
S
Stanislaw Gruszka 已提交
839
	il_write_targ_mem(priv,
840 841 842 843
		priv->scd_base_addr + IWL49_SCD_CONTEXT_QUEUE_OFFSET(txq_id),
		(SCD_WIN_SIZE << IWL49_SCD_QUEUE_CTX_REG1_WIN_SIZE_POS) &
		IWL49_SCD_QUEUE_CTX_REG1_WIN_SIZE_MSK);

S
Stanislaw Gruszka 已提交
844
	il_write_targ_mem(priv, priv->scd_base_addr +
845 846 847 848
		IWL49_SCD_CONTEXT_QUEUE_OFFSET(txq_id) + sizeof(u32),
		(SCD_FRAME_LIMIT << IWL49_SCD_QUEUE_CTX_REG2_FRAME_LIMIT_POS)
		& IWL49_SCD_QUEUE_CTX_REG2_FRAME_LIMIT_MSK);

S
Stanislaw Gruszka 已提交
849
	il_set_bits_prph(priv, IWL49_SCD_INTERRUPT_MASK, (1 << txq_id));
850 851

	/* Set up Status area in SRAM, map to Tx DMA/FIFO, activate the queue */
S
Stanislaw Gruszka 已提交
852
	il4965_tx_queue_set_status(priv, &priv->txq[txq_id], tx_fifo, 1);
853 854 855 856 857 858 859

	spin_unlock_irqrestore(&priv->lock, flags);

	return 0;
}


S
Stanislaw Gruszka 已提交
860
int il4965_tx_agg_start(struct il_priv *priv, struct ieee80211_vif *vif,
861 862 863 864 865 866 867
			struct ieee80211_sta *sta, u16 tid, u16 *ssn)
{
	int sta_id;
	int tx_fifo;
	int txq_id;
	int ret;
	unsigned long flags;
S
Stanislaw Gruszka 已提交
868
	struct il_tid_data *tid_data;
869

S
Stanislaw Gruszka 已提交
870
	tx_fifo = il4965_get_fifo_from_tid(il_rxon_ctx_from_vif(vif), tid);
871 872 873
	if (unlikely(tx_fifo < 0))
		return tx_fifo;

S
Stanislaw Gruszka 已提交
874
	IL_WARN(priv, "%s on ra = %pM tid = %d\n",
875 876
			__func__, sta->addr, tid);

S
Stanislaw Gruszka 已提交
877 878 879
	sta_id = il_sta_id(sta);
	if (sta_id == IL_INVALID_STATION) {
		IL_ERR(priv, "Start AGG on invalid station\n");
880 881 882 883 884
		return -ENXIO;
	}
	if (unlikely(tid >= MAX_TID_COUNT))
		return -EINVAL;

S
Stanislaw Gruszka 已提交
885 886
	if (priv->stations[sta_id].tid[tid].agg.state != IL_AGG_OFF) {
		IL_ERR(priv, "Start AGG when state is not IL_AGG_OFF !\n");
887 888 889
		return -ENXIO;
	}

S
Stanislaw Gruszka 已提交
890
	txq_id = il4965_txq_ctx_activate_free(priv);
891
	if (txq_id == -1) {
S
Stanislaw Gruszka 已提交
892
		IL_ERR(priv, "No free aggregation queue available\n");
893 894 895 896 897 898 899
		return -ENXIO;
	}

	spin_lock_irqsave(&priv->sta_lock, flags);
	tid_data = &priv->stations[sta_id].tid[tid];
	*ssn = SEQ_TO_SN(tid_data->seq_number);
	tid_data->agg.txq_id = txq_id;
S
Stanislaw Gruszka 已提交
900 901
	il_set_swq_id(&priv->txq[txq_id],
				il4965_get_ac_from_tid(tid), txq_id);
902 903
	spin_unlock_irqrestore(&priv->sta_lock, flags);

S
Stanislaw Gruszka 已提交
904
	ret = il4965_txq_agg_enable(priv, txq_id, tx_fifo,
905 906 907 908 909 910 911
						  sta_id, tid, *ssn);
	if (ret)
		return ret;

	spin_lock_irqsave(&priv->sta_lock, flags);
	tid_data = &priv->stations[sta_id].tid[tid];
	if (tid_data->tfds_in_queue == 0) {
S
Stanislaw Gruszka 已提交
912 913
		IL_DEBUG_HT(priv, "HW queue is empty\n");
		tid_data->agg.state = IL_AGG_ON;
914 915
		ieee80211_start_tx_ba_cb_irqsafe(vif, sta->addr, tid);
	} else {
S
Stanislaw Gruszka 已提交
916
		IL_DEBUG_HT(priv,
917 918
			"HW queue is NOT empty: %d packets in HW queue\n",
			     tid_data->tfds_in_queue);
S
Stanislaw Gruszka 已提交
919
		tid_data->agg.state = IL_EMPTYING_HW_QUEUE_ADDBA;
920 921 922 923 924 925 926 927 928
	}
	spin_unlock_irqrestore(&priv->sta_lock, flags);
	return ret;
}

/**
 * txq_id must be greater than IWL49_FIRST_AMPDU_QUEUE
 * priv->lock must be held by the caller
 */
S
Stanislaw Gruszka 已提交
929
static int il4965_txq_agg_disable(struct il_priv *priv, u16 txq_id,
930 931 932 933 934
				   u16 ssn_idx, u8 tx_fifo)
{
	if ((IWL49_FIRST_AMPDU_QUEUE > txq_id) ||
	    (IWL49_FIRST_AMPDU_QUEUE +
		priv->cfg->base_params->num_of_ampdu_queues <= txq_id)) {
S
Stanislaw Gruszka 已提交
935
		IL_WARN(priv,
936 937 938 939 940 941 942
			"queue number out of range: %d, must be %d to %d\n",
			txq_id, IWL49_FIRST_AMPDU_QUEUE,
			IWL49_FIRST_AMPDU_QUEUE +
			priv->cfg->base_params->num_of_ampdu_queues - 1);
		return -EINVAL;
	}

S
Stanislaw Gruszka 已提交
943
	il4965_tx_queue_stop_scheduler(priv, txq_id);
944

S
Stanislaw Gruszka 已提交
945
	il_clear_bits_prph(priv,
946 947 948 949 950
			IWL49_SCD_QUEUECHAIN_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) */
S
Stanislaw Gruszka 已提交
951
	il4965_set_wr_ptrs(priv, txq_id, ssn_idx);
952

S
Stanislaw Gruszka 已提交
953
	il_clear_bits_prph(priv,
954
			 IWL49_SCD_INTERRUPT_MASK, (1 << txq_id));
S
Stanislaw Gruszka 已提交
955 956
	il_txq_ctx_deactivate(priv, txq_id);
	il4965_tx_queue_set_status(priv, &priv->txq[txq_id], tx_fifo, 0);
957 958 959 960

	return 0;
}

S
Stanislaw Gruszka 已提交
961
int il4965_tx_agg_stop(struct il_priv *priv, struct ieee80211_vif *vif,
962 963 964
		       struct ieee80211_sta *sta, u16 tid)
{
	int tx_fifo_id, txq_id, sta_id, ssn;
S
Stanislaw Gruszka 已提交
965
	struct il_tid_data *tid_data;
966 967 968
	int write_ptr, read_ptr;
	unsigned long flags;

S
Stanislaw Gruszka 已提交
969
	tx_fifo_id = il4965_get_fifo_from_tid(il_rxon_ctx_from_vif(vif), tid);
970 971 972
	if (unlikely(tx_fifo_id < 0))
		return tx_fifo_id;

S
Stanislaw Gruszka 已提交
973
	sta_id = il_sta_id(sta);
974

S
Stanislaw Gruszka 已提交
975 976
	if (sta_id == IL_INVALID_STATION) {
		IL_ERR(priv, "Invalid station for AGG tid %d\n", tid);
977 978 979 980 981 982 983 984 985 986
		return -ENXIO;
	}

	spin_lock_irqsave(&priv->sta_lock, flags);

	tid_data = &priv->stations[sta_id].tid[tid];
	ssn = (tid_data->seq_number & IEEE80211_SCTL_SEQ) >> 4;
	txq_id = tid_data->agg.txq_id;

	switch (priv->stations[sta_id].tid[tid].agg.state) {
S
Stanislaw Gruszka 已提交
987
	case IL_EMPTYING_HW_QUEUE_ADDBA:
988 989 990 991 992 993
		/*
		 * This can happen if the peer stops aggregation
		 * again before we've had a chance to drain the
		 * queue we selected previously, i.e. before the
		 * session was really started completely.
		 */
S
Stanislaw Gruszka 已提交
994
		IL_DEBUG_HT(priv, "AGG stop before setup done\n");
995
		goto turn_off;
S
Stanislaw Gruszka 已提交
996
	case IL_AGG_ON:
997 998
		break;
	default:
S
Stanislaw Gruszka 已提交
999
		IL_WARN(priv, "Stopping AGG while state not ON or starting\n");
1000 1001 1002 1003 1004 1005 1006
	}

	write_ptr = priv->txq[txq_id].q.write_ptr;
	read_ptr = priv->txq[txq_id].q.read_ptr;

	/* The queue is not empty */
	if (write_ptr != read_ptr) {
S
Stanislaw Gruszka 已提交
1007
		IL_DEBUG_HT(priv, "Stopping a non empty AGG HW QUEUE\n");
1008
		priv->stations[sta_id].tid[tid].agg.state =
S
Stanislaw Gruszka 已提交
1009
				IL_EMPTYING_HW_QUEUE_DELBA;
1010 1011 1012 1013
		spin_unlock_irqrestore(&priv->sta_lock, flags);
		return 0;
	}

S
Stanislaw Gruszka 已提交
1014
	IL_DEBUG_HT(priv, "HW queue is empty\n");
1015
 turn_off:
S
Stanislaw Gruszka 已提交
1016
	priv->stations[sta_id].tid[tid].agg.state = IL_AGG_OFF;
1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028

	/* do not restore/save irqs */
	spin_unlock(&priv->sta_lock);
	spin_lock(&priv->lock);

	/*
	 * the only reason this call can fail is queue number out of range,
	 * which can happen if uCode is reloaded and all the station
	 * information are lost. if it is outside the range, there is no need
	 * to deactivate the uCode queue, just return "success" to allow
	 *  mac80211 to clean up it own data.
	 */
S
Stanislaw Gruszka 已提交
1029
	il4965_txq_agg_disable(priv, txq_id, ssn, tx_fifo_id);
1030 1031 1032 1033 1034 1035 1036
	spin_unlock_irqrestore(&priv->lock, flags);

	ieee80211_stop_tx_ba_cb_irqsafe(vif, sta->addr, tid);

	return 0;
}

S
Stanislaw Gruszka 已提交
1037
int il4965_txq_check_empty(struct il_priv *priv,
1038 1039
			   int sta_id, u8 tid, int txq_id)
{
S
Stanislaw Gruszka 已提交
1040
	struct il_queue *q = &priv->txq[txq_id].q;
1041
	u8 *addr = priv->stations[sta_id].sta.sta.addr;
S
Stanislaw Gruszka 已提交
1042 1043
	struct il_tid_data *tid_data = &priv->stations[sta_id].tid[tid];
	struct il_rxon_context *ctx;
1044 1045 1046 1047 1048 1049

	ctx = &priv->contexts[priv->stations[sta_id].ctxid];

	lockdep_assert_held(&priv->sta_lock);

	switch (priv->stations[sta_id].tid[tid].agg.state) {
S
Stanislaw Gruszka 已提交
1050
	case IL_EMPTYING_HW_QUEUE_DELBA:
1051 1052 1053 1054 1055
		/* We are reclaiming the last packet of the */
		/* aggregated HW queue */
		if ((txq_id  == tid_data->agg.txq_id) &&
		    (q->read_ptr == q->write_ptr)) {
			u16 ssn = SEQ_TO_SN(tid_data->seq_number);
S
Stanislaw Gruszka 已提交
1056 1057
			int tx_fifo = il4965_get_fifo_from_tid(ctx, tid);
			IL_DEBUG_HT(priv,
1058
				"HW queue empty: continue DELBA flow\n");
S
Stanislaw Gruszka 已提交
1059 1060
			il4965_txq_agg_disable(priv, txq_id, ssn, tx_fifo);
			tid_data->agg.state = IL_AGG_OFF;
1061 1062 1063
			ieee80211_stop_tx_ba_cb_irqsafe(ctx->vif, addr, tid);
		}
		break;
S
Stanislaw Gruszka 已提交
1064
	case IL_EMPTYING_HW_QUEUE_ADDBA:
1065 1066
		/* We are reclaiming the last packet of the queue */
		if (tid_data->tfds_in_queue == 0) {
S
Stanislaw Gruszka 已提交
1067
			IL_DEBUG_HT(priv,
1068
				"HW queue empty: continue ADDBA flow\n");
S
Stanislaw Gruszka 已提交
1069
			tid_data->agg.state = IL_AGG_ON;
1070 1071 1072 1073 1074 1075 1076 1077
			ieee80211_start_tx_ba_cb_irqsafe(ctx->vif, addr, tid);
		}
		break;
	}

	return 0;
}

S
Stanislaw Gruszka 已提交
1078 1079
static void il4965_non_agg_tx_status(struct il_priv *priv,
				     struct il_rxon_context *ctx,
1080 1081 1082
				     const u8 *addr1)
{
	struct ieee80211_sta *sta;
S
Stanislaw Gruszka 已提交
1083
	struct il_station_priv *sta_priv;
1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097

	rcu_read_lock();
	sta = ieee80211_find_sta(ctx->vif, addr1);
	if (sta) {
		sta_priv = (void *)sta->drv_priv;
		/* avoid atomic ops if this isn't a client */
		if (sta_priv->client &&
		    atomic_dec_return(&sta_priv->pending_frames) == 0)
			ieee80211_sta_block_awake(priv->hw, sta, false);
	}
	rcu_read_unlock();
}

static void
S
Stanislaw Gruszka 已提交
1098
il4965_tx_status(struct il_priv *priv, struct il_tx_info *tx_info,
1099 1100 1101 1102 1103
			     bool is_agg)
{
	struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) tx_info->skb->data;

	if (!is_agg)
S
Stanislaw Gruszka 已提交
1104
		il4965_non_agg_tx_status(priv, tx_info->ctx, hdr->addr1);
1105 1106 1107 1108

	ieee80211_tx_status_irqsafe(priv->hw, tx_info->skb);
}

S
Stanislaw Gruszka 已提交
1109
int il4965_tx_queue_reclaim(struct il_priv *priv, int txq_id, int index)
1110
{
S
Stanislaw Gruszka 已提交
1111 1112 1113
	struct il_tx_queue *txq = &priv->txq[txq_id];
	struct il_queue *q = &txq->q;
	struct il_tx_info *tx_info;
1114 1115 1116
	int nfreed = 0;
	struct ieee80211_hdr *hdr;

S
Stanislaw Gruszka 已提交
1117 1118
	if ((index >= q->n_bd) || (il_queue_used(q, index) == 0)) {
		IL_ERR(priv, "Read index for DMA queue txq id (%d), index %d, "
1119 1120 1121 1122 1123
			  "is out of range [0-%d] %d %d.\n", txq_id,
			  index, q->n_bd, q->write_ptr, q->read_ptr);
		return 0;
	}

S
Stanislaw Gruszka 已提交
1124
	for (index = il_queue_inc_wrap(index, q->n_bd);
1125
	     q->read_ptr != index;
S
Stanislaw Gruszka 已提交
1126
	     q->read_ptr = il_queue_inc_wrap(q->read_ptr, q->n_bd)) {
1127 1128

		tx_info = &txq->txb[txq->q.read_ptr];
1129 1130 1131

		if (WARN_ON_ONCE(tx_info->skb == NULL))
			continue;
1132 1133

		hdr = (struct ieee80211_hdr *)tx_info->skb->data;
1134
		if (ieee80211_is_data_qos(hdr->frame_control))
1135
			nfreed++;
1136

S
Stanislaw Gruszka 已提交
1137
		il4965_tx_status(priv, tx_info,
1138
				 txq_id >= IWL4965_FIRST_AMPDU_QUEUE);
1139 1140 1141 1142 1143 1144 1145 1146
		tx_info->skb = NULL;

		priv->cfg->ops->lib->txq_free_tfd(priv, txq);
	}
	return nfreed;
}

/**
S
Stanislaw Gruszka 已提交
1147
 * il4965_tx_status_reply_compressed_ba - Update tx status from block-ack
1148 1149 1150 1151
 *
 * Go through block-ack's bitmap of ACK'd frames, update driver's record of
 * ACK vs. not.  This gets sent to mac80211, then to rate scaling algo.
 */
S
Stanislaw Gruszka 已提交
1152 1153 1154
static int il4965_tx_status_reply_compressed_ba(struct il_priv *priv,
				 struct il_ht_agg *agg,
				 struct il_compressed_ba_resp *ba_resp)
1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165

{
	int i, sh, ack;
	u16 seq_ctl = le16_to_cpu(ba_resp->seq_ctl);
	u16 scd_flow = le16_to_cpu(ba_resp->scd_flow);
	int successes = 0;
	struct ieee80211_tx_info *info;
	u64 bitmap, sent_bitmap;

	if (unlikely(!agg->wait_for_ba))  {
		if (unlikely(ba_resp->bitmap))
S
Stanislaw Gruszka 已提交
1166
			IL_ERR(priv, "Received BA when not expected\n");
1167 1168 1169 1170 1171
		return -EINVAL;
	}

	/* Mark that the expected block-ack response arrived */
	agg->wait_for_ba = 0;
S
Stanislaw Gruszka 已提交
1172
	IL_DEBUG_TX_REPLY(priv, "BA %d %d\n", agg->start_idx,
1173 1174 1175 1176 1177 1178 1179 1180
							ba_resp->seq_ctl);

	/* Calculate shift to align block-ack bits with our Tx window bits */
	sh = agg->start_idx - SEQ_TO_INDEX(seq_ctl >> 4);
	if (sh < 0) /* tbw something is wrong with indices */
		sh += 0x100;

	if (agg->frame_count > (64 - sh)) {
S
Stanislaw Gruszka 已提交
1181
		IL_DEBUG_TX_REPLY(priv, "more frames than bitmap size");
1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197
		return -1;
	}

	/* don't use 64-bit values for now */
	bitmap = le64_to_cpu(ba_resp->bitmap) >> sh;

	/* check for success or failure according to the
	 * transmitted bitmap and block-ack bitmap */
	sent_bitmap = bitmap & agg->bitmap;

	/* For each frame attempted in aggregation,
	 * update driver's record of tx frame's status. */
	i = 0;
	while (sent_bitmap) {
		ack = sent_bitmap & 1ULL;
		successes += ack;
S
Stanislaw Gruszka 已提交
1198
		IL_DEBUG_TX_REPLY(priv, "%s ON i=%d idx=%d raw=%d\n",
1199 1200 1201 1202 1203 1204 1205
			ack ? "ACK" : "NACK", i,
			(agg->start_idx + i) & 0xff,
			agg->start_idx + i);
		sent_bitmap >>= 1;
		++i;
	}

S
Stanislaw Gruszka 已提交
1206
	IL_DEBUG_TX_REPLY(priv, "Bitmap %llx\n",
1207 1208 1209 1210 1211 1212 1213 1214
				   (unsigned long long)bitmap);

	info = IEEE80211_SKB_CB(priv->txq[scd_flow].txb[agg->start_idx].skb);
	memset(&info->status, 0, sizeof(info->status));
	info->flags |= IEEE80211_TX_STAT_ACK;
	info->flags |= IEEE80211_TX_STAT_AMPDU;
	info->status.ampdu_ack_len = successes;
	info->status.ampdu_len = agg->frame_count;
S
Stanislaw Gruszka 已提交
1215
	il4965_hwrate_to_tx_control(priv, agg->rate_n_flags, info);
1216 1217 1218 1219 1220 1221 1222

	return 0;
}

/**
 * translate ucode response to mac80211 tx status control values
 */
S
Stanislaw Gruszka 已提交
1223
void il4965_hwrate_to_tx_control(struct il_priv *priv, u32 rate_n_flags,
1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239
				  struct ieee80211_tx_info *info)
{
	struct ieee80211_tx_rate *r = &info->control.rates[0];

	info->antenna_sel_tx =
		((rate_n_flags & RATE_MCS_ANT_ABC_MSK) >> RATE_MCS_ANT_POS);
	if (rate_n_flags & RATE_MCS_HT_MSK)
		r->flags |= IEEE80211_TX_RC_MCS;
	if (rate_n_flags & RATE_MCS_GF_MSK)
		r->flags |= IEEE80211_TX_RC_GREEN_FIELD;
	if (rate_n_flags & RATE_MCS_HT40_MSK)
		r->flags |= IEEE80211_TX_RC_40_MHZ_WIDTH;
	if (rate_n_flags & RATE_MCS_DUP_MSK)
		r->flags |= IEEE80211_TX_RC_DUP_DATA;
	if (rate_n_flags & RATE_MCS_SGI_MSK)
		r->flags |= IEEE80211_TX_RC_SHORT_GI;
S
Stanislaw Gruszka 已提交
1240
	r->idx = il4965_hwrate_to_mac80211_idx(rate_n_flags, info->band);
1241 1242 1243
}

/**
S
Stanislaw Gruszka 已提交
1244
 * il4965_rx_reply_compressed_ba - Handler for REPLY_COMPRESSED_BA
1245 1246 1247 1248
 *
 * Handles block-acknowledge notification from device, which reports success
 * of frames sent via aggregation.
 */
S
Stanislaw Gruszka 已提交
1249 1250
void il4965_rx_reply_compressed_ba(struct il_priv *priv,
					   struct il_rx_mem_buffer *rxb)
1251
{
S
Stanislaw Gruszka 已提交
1252 1253 1254 1255
	struct il_rx_packet *pkt = rxb_addr(rxb);
	struct il_compressed_ba_resp *ba_resp = &pkt->u.compressed_ba;
	struct il_tx_queue *txq = NULL;
	struct il_ht_agg *agg;
1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268
	int index;
	int sta_id;
	int tid;
	unsigned long flags;

	/* "flow" corresponds to Tx queue */
	u16 scd_flow = le16_to_cpu(ba_resp->scd_flow);

	/* "ssn" is start of block-ack Tx window, corresponds to index
	 * (in Tx queue's circular buffer) of first TFD/frame in window */
	u16 ba_resp_scd_ssn = le16_to_cpu(ba_resp->scd_ssn);

	if (scd_flow >= priv->hw_params.max_txq_num) {
S
Stanislaw Gruszka 已提交
1269
		IL_ERR(priv,
1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284
			"BUG_ON scd_flow is bigger than number of queues\n");
		return;
	}

	txq = &priv->txq[scd_flow];
	sta_id = ba_resp->sta_id;
	tid = ba_resp->tid;
	agg = &priv->stations[sta_id].tid[tid].agg;
	if (unlikely(agg->txq_id != scd_flow)) {
		/*
		 * FIXME: this is a uCode bug which need to be addressed,
		 * log the information and return for now!
		 * since it is possible happen very often and in order
		 * not to fill the syslog, don't enable the logging by default
		 */
S
Stanislaw Gruszka 已提交
1285
		IL_DEBUG_TX_REPLY(priv,
1286 1287 1288 1289 1290 1291
			"BA scd_flow %d does not match txq_id %d\n",
			scd_flow, agg->txq_id);
		return;
	}

	/* Find index just before block-ack window */
S
Stanislaw Gruszka 已提交
1292
	index = il_queue_dec_wrap(ba_resp_scd_ssn & 0xff, txq->q.n_bd);
1293 1294 1295

	spin_lock_irqsave(&priv->sta_lock, flags);

S
Stanislaw Gruszka 已提交
1296
	IL_DEBUG_TX_REPLY(priv, "REPLY_COMPRESSED_BA [%d] Received from %pM, "
1297 1298 1299 1300
			   "sta_id = %d\n",
			   agg->wait_for_ba,
			   (u8 *) &ba_resp->sta_addr_lo32,
			   ba_resp->sta_id);
S
Stanislaw Gruszka 已提交
1301
	IL_DEBUG_TX_REPLY(priv, "TID = %d, SeqCtl = %d, bitmap = 0x%llx,"
1302 1303 1304 1305 1306 1307 1308
			"scd_flow = "
			   "%d, scd_ssn = %d\n",
			   ba_resp->tid,
			   ba_resp->seq_ctl,
			   (unsigned long long)le64_to_cpu(ba_resp->bitmap),
			   ba_resp->scd_flow,
			   ba_resp->scd_ssn);
S
Stanislaw Gruszka 已提交
1309
	IL_DEBUG_TX_REPLY(priv, "DAT start_idx = %d, bitmap = 0x%llx\n",
1310 1311 1312 1313
			   agg->start_idx,
			   (unsigned long long)agg->bitmap);

	/* Update driver's record of ACK vs. not for each frame in window */
S
Stanislaw Gruszka 已提交
1314
	il4965_tx_status_reply_compressed_ba(priv, agg, ba_resp);
1315 1316 1317 1318 1319 1320

	/* Release all TFDs before the SSN, i.e. all TFDs in front of
	 * block-ack window (we assume that they've been successfully
	 * transmitted ... if not, it's too late anyway). */
	if (txq->q.read_ptr != (ba_resp_scd_ssn & 0xff)) {
		/* calculate mac80211 ampdu sw queue to wake */
S
Stanislaw Gruszka 已提交
1321 1322
		int freed = il4965_tx_queue_reclaim(priv, scd_flow, index);
		il4965_free_tfds_in_queue(priv, sta_id, tid, freed);
1323

S
Stanislaw Gruszka 已提交
1324
		if ((il_queue_space(&txq->q) > txq->q.low_mark) &&
1325
		    priv->mac80211_registered &&
S
Stanislaw Gruszka 已提交
1326 1327
		    (agg->state != IL_EMPTYING_HW_QUEUE_DELBA))
			il_wake_queue(priv, txq);
1328

S
Stanislaw Gruszka 已提交
1329
		il4965_txq_check_empty(priv, sta_id, tid, scd_flow);
1330 1331 1332 1333 1334 1335
	}

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

#ifdef CONFIG_IWLWIFI_LEGACY_DEBUG
S
Stanislaw Gruszka 已提交
1336
const char *il4965_get_tx_fail_reason(u32 status)
1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372
{
#define TX_STATUS_FAIL(x) case TX_STATUS_FAIL_ ## x: return #x
#define TX_STATUS_POSTPONE(x) case TX_STATUS_POSTPONE_ ## x: return #x

	switch (status & TX_STATUS_MSK) {
	case TX_STATUS_SUCCESS:
		return "SUCCESS";
	TX_STATUS_POSTPONE(DELAY);
	TX_STATUS_POSTPONE(FEW_BYTES);
	TX_STATUS_POSTPONE(QUIET_PERIOD);
	TX_STATUS_POSTPONE(CALC_TTAK);
	TX_STATUS_FAIL(INTERNAL_CROSSED_RETRY);
	TX_STATUS_FAIL(SHORT_LIMIT);
	TX_STATUS_FAIL(LONG_LIMIT);
	TX_STATUS_FAIL(FIFO_UNDERRUN);
	TX_STATUS_FAIL(DRAIN_FLOW);
	TX_STATUS_FAIL(RFKILL_FLUSH);
	TX_STATUS_FAIL(LIFE_EXPIRE);
	TX_STATUS_FAIL(DEST_PS);
	TX_STATUS_FAIL(HOST_ABORTED);
	TX_STATUS_FAIL(BT_RETRY);
	TX_STATUS_FAIL(STA_INVALID);
	TX_STATUS_FAIL(FRAG_DROPPED);
	TX_STATUS_FAIL(TID_DISABLE);
	TX_STATUS_FAIL(FIFO_FLUSHED);
	TX_STATUS_FAIL(INSUFFICIENT_CF_POLL);
	TX_STATUS_FAIL(PASSIVE_NO_RX);
	TX_STATUS_FAIL(NO_BEACON_ON_RADAR);
	}

	return "UNKNOWN";

#undef TX_STATUS_FAIL
#undef TX_STATUS_POSTPONE
}
#endif /* CONFIG_IWLWIFI_LEGACY_DEBUG */