tx.c 29.6 KB
Newer Older
L
Luciano Coelho 已提交
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
/*
 * This file is part of wl1271
 *
 * Copyright (C) 2009 Nokia Corporation
 *
 * Contact: Luciano Coelho <luciano.coelho@nokia.com>
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * version 2 as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA
 *
 */

#include <linux/kernel.h>
#include <linux/module.h>
26
#include <linux/etherdevice.h>
L
Luciano Coelho 已提交
27

28
#include "wlcore.h"
29
#include "debug.h"
S
Shahar Levi 已提交
30 31 32
#include "io.h"
#include "ps.h"
#include "tx.h"
33
#include "event.h"
34
#include "hw_ops.h"
L
Luciano Coelho 已提交
35

36 37 38 39 40 41
/*
 * TODO: this is here just for now, it must be removed when the data
 * operations are in place.
 */
#include "../wl12xx/reg.h"

E
Eliad Peller 已提交
42 43
static int wl1271_set_default_wep_key(struct wl1271 *wl,
				      struct wl12xx_vif *wlvif, u8 id)
44 45
{
	int ret;
E
Eliad Peller 已提交
46
	bool is_ap = (wlvif->bss_type == BSS_TYPE_AP_BSS);
47 48

	if (is_ap)
E
Eliad Peller 已提交
49
		ret = wl12xx_cmd_set_default_wep_key(wl, id,
50
						     wlvif->ap.bcast_hlid);
51
	else
E
Eliad Peller 已提交
52
		ret = wl12xx_cmd_set_default_wep_key(wl, id, wlvif->sta.hlid);
53 54 55 56 57 58 59 60

	if (ret < 0)
		return ret;

	wl1271_debug(DEBUG_CRYPT, "default wep key idx: %d", (int)id);
	return 0;
}

61
static int wl1271_alloc_tx_id(struct wl1271 *wl, struct sk_buff *skb)
L
Luciano Coelho 已提交
62
{
63 64
	int id;

65 66
	id = find_first_zero_bit(wl->tx_frames_map, wl->num_tx_desc);
	if (id >= wl->num_tx_desc)
67 68 69 70 71 72 73
		return -EBUSY;

	__set_bit(id, wl->tx_frames_map);
	wl->tx_frames[id] = skb;
	wl->tx_frames_cnt++;
	return id;
}
L
Luciano Coelho 已提交
74

75
void wl1271_free_tx_id(struct wl1271 *wl, int id)
76 77
{
	if (__test_and_clear_bit(id, wl->tx_frames_map)) {
78
		if (unlikely(wl->tx_frames_cnt == wl->num_tx_desc))
79 80
			clear_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags);

81 82 83
		wl->tx_frames[id] = NULL;
		wl->tx_frames_cnt--;
	}
L
Luciano Coelho 已提交
84
}
85
EXPORT_SYMBOL(wl1271_free_tx_id);
L
Luciano Coelho 已提交
86

87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
static void wl1271_tx_ap_update_inconnection_sta(struct wl1271 *wl,
						 struct sk_buff *skb)
{
	struct ieee80211_hdr *hdr;

	/*
	 * add the station to the known list before transmitting the
	 * authentication response. this way it won't get de-authed by FW
	 * when transmitting too soon.
	 */
	hdr = (struct ieee80211_hdr *)(skb->data +
				       sizeof(struct wl1271_tx_hw_descr));
	if (ieee80211_is_auth(hdr->frame_control))
		wl1271_acx_set_inconnection_sta(wl, hdr->addr1);
}

103 104 105
static void wl1271_tx_regulate_link(struct wl1271 *wl,
				    struct wl12xx_vif *wlvif,
				    u8 hlid)
106
{
107
	bool fw_ps, single_sta;
108
	u8 tx_pkts;
109

110
	if (WARN_ON(!test_bit(hlid, wlvif->links_map)))
111 112 113
		return;

	fw_ps = test_bit(hlid, (unsigned long *)&wl->ap_fw_ps_map);
114
	tx_pkts = wl->links[hlid].allocated_pkts;
115
	single_sta = (wl->active_sta_count == 1);
116 117 118 119

	/*
	 * if in FW PS and there is enough data in FW we can put the link
	 * into high-level PS and clean out its TX queues.
120 121
	 * Make an exception if this is the only connected station. In this
	 * case FW-memory congestion is not a problem.
122
	 */
123
	if (!single_sta && fw_ps && tx_pkts >= WL1271_PS_STA_MAX_PACKETS)
124
		wl12xx_ps_link_start(wl, wlvif, hlid, true);
125 126
}

127
bool wl12xx_is_dummy_packet(struct wl1271 *wl, struct sk_buff *skb)
E
Eliad Peller 已提交
128 129 130
{
	return wl->dummy_packet == skb;
}
131
EXPORT_SYMBOL(wl12xx_is_dummy_packet);
E
Eliad Peller 已提交
132

133 134
u8 wl12xx_tx_get_hlid_ap(struct wl1271 *wl, struct wl12xx_vif *wlvif,
			 struct sk_buff *skb)
135 136 137 138 139 140 141 142 143 144 145 146
{
	struct ieee80211_tx_info *control = IEEE80211_SKB_CB(skb);

	if (control->control.sta) {
		struct wl1271_station *wl_sta;

		wl_sta = (struct wl1271_station *)
				control->control.sta->drv_priv;
		return wl_sta->hlid;
	} else {
		struct ieee80211_hdr *hdr;

147
		if (!test_bit(WLVIF_FLAG_AP_STARTED, &wlvif->flags))
E
Eliad Peller 已提交
148 149
			return wl->system_hlid;

150
		hdr = (struct ieee80211_hdr *)skb->data;
151
		if (is_multicast_ether_addr(ieee80211_get_DA(hdr)))
152
			return wlvif->ap.bcast_hlid;
153 154
		else
			return wlvif->ap.global_hlid;
155 156 157
	}
}

158 159
u8 wl12xx_tx_get_hlid(struct wl1271 *wl, struct wl12xx_vif *wlvif,
		      struct sk_buff *skb)
E
Eliad Peller 已提交
160
{
161 162
	struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;

E
Eliad Peller 已提交
163
	if (!wlvif || wl12xx_is_dummy_packet(wl, skb))
E
Eliad Peller 已提交
164 165
		return wl->system_hlid;

E
Eliad Peller 已提交
166
	if (wlvif->bss_type == BSS_TYPE_AP_BSS)
167
		return wl12xx_tx_get_hlid_ap(wl, wlvif, skb);
E
Eliad Peller 已提交
168

169
	if ((test_bit(WLVIF_FLAG_STA_ASSOCIATED, &wlvif->flags) ||
170
	     test_bit(WLVIF_FLAG_IBSS_JOINED, &wlvif->flags)) &&
171 172
	    !ieee80211_is_auth(hdr->frame_control) &&
	    !ieee80211_is_assoc_req(hdr->frame_control))
E
Eliad Peller 已提交
173
		return wlvif->sta.hlid;
E
Eliad Peller 已提交
174
	else
E
Eliad Peller 已提交
175
		return wlvif->dev_hlid;
E
Eliad Peller 已提交
176 177
}

178 179
unsigned int wlcore_calc_packet_alignment(struct wl1271 *wl,
					  unsigned int packet_length)
180
{
181
	if (wl->quirks & WLCORE_QUIRK_TX_BLOCKSIZE_ALIGN)
182
		return ALIGN(packet_length, WL12XX_BUS_BLOCK_SIZE);
183 184
	else
		return ALIGN(packet_length, WL1271_TX_ALIGN_TO);
185
}
186
EXPORT_SYMBOL(wlcore_calc_packet_alignment);
187

188
static int wl1271_tx_allocate(struct wl1271 *wl, struct wl12xx_vif *wlvif,
E
Eliad Peller 已提交
189
			      struct sk_buff *skb, u32 extra, u32 buf_offset,
190
			      u8 hlid, bool is_gem)
L
Luciano Coelho 已提交
191 192 193
{
	struct wl1271_tx_hw_descr *desc;
	u32 total_len = skb->len + sizeof(struct wl1271_tx_hw_descr) + extra;
194
	u32 total_blocks;
195
	int id, ret = -EBUSY, ac;
196
	u32 spare_blocks;
L
Luciano Coelho 已提交
197

198
	if (buf_offset + total_len > WL1271_AGGR_BUFFER_SIZE)
I
Ido Yariv 已提交
199
		return -EAGAIN;
200

201 202
	spare_blocks = wlcore_hw_get_spare_blocks(wl, is_gem);

L
Luciano Coelho 已提交
203
	/* allocate free identifier for the packet */
204
	id = wl1271_alloc_tx_id(wl, skb);
L
Luciano Coelho 已提交
205 206 207
	if (id < 0)
		return id;

208
	total_blocks = wlcore_hw_calc_tx_blocks(wl, total_len, spare_blocks);
209

L
Luciano Coelho 已提交
210 211 212 213
	if (total_blocks <= wl->tx_blocks_available) {
		desc = (struct wl1271_tx_hw_descr *)skb_push(
			skb, total_len - skb->len);

214 215
		wlcore_hw_set_tx_desc_blocks(wl, desc, total_blocks,
					     spare_blocks);
216

L
Luciano Coelho 已提交
217 218 219
		desc->id = id;

		wl->tx_blocks_available -= total_blocks;
220
		wl->tx_allocated_blocks += total_blocks;
L
Luciano Coelho 已提交
221

A
Arik Nemtsov 已提交
222 223 224 225
		/* If the FW was empty before, arm the Tx watchdog */
		if (wl->tx_allocated_blocks == total_blocks)
			wl12xx_rearm_tx_watchdog_locked(wl);

226 227
		ac = wl1271_tx_get_queue(skb_get_queue_mapping(skb));
		wl->tx_allocated_pkts[ac]++;
228

229
		if (!wl12xx_is_dummy_packet(wl, skb) && wlvif &&
E
Eliad Peller 已提交
230
		    wlvif->bss_type == BSS_TYPE_AP_BSS &&
231
		    test_bit(hlid, wlvif->ap.sta_hlid_map))
232
			wl->links[hlid].allocated_pkts++;
233

L
Luciano Coelho 已提交
234 235 236 237 238
		ret = 0;

		wl1271_debug(DEBUG_TX,
			     "tx_allocate: size: %d, blocks: %d, id: %d",
			     total_len, total_blocks, id);
239
	} else {
240
		wl1271_free_tx_id(wl, id);
241
	}
L
Luciano Coelho 已提交
242 243 244 245

	return ret;
}

246
static void wl1271_tx_fill_hdr(struct wl1271 *wl, struct wl12xx_vif *wlvif,
E
Eliad Peller 已提交
247 248
			       struct sk_buff *skb, u32 extra,
			       struct ieee80211_tx_info *control, u8 hlid)
L
Luciano Coelho 已提交
249
{
250
	struct timespec ts;
L
Luciano Coelho 已提交
251
	struct wl1271_tx_hw_descr *desc;
252
	int ac, rate_idx;
253
	s64 hosttime;
254
	u16 tx_attr = 0;
255 256 257
	__le16 frame_control;
	struct ieee80211_hdr *hdr;
	u8 *frame_start;
258
	bool is_dummy;
L
Luciano Coelho 已提交
259 260

	desc = (struct wl1271_tx_hw_descr *) skb->data;
261 262 263
	frame_start = (u8 *)(desc + 1);
	hdr = (struct ieee80211_hdr *)(frame_start + extra);
	frame_control = hdr->frame_control;
L
Luciano Coelho 已提交
264

265 266
	/* relocate space for security header */
	if (extra) {
267 268
		int hdrlen = ieee80211_hdrlen(frame_control);
		memmove(frame_start, hdr, hdrlen);
269
		skb_set_network_header(skb, skb_network_offset(skb) + extra);
270 271
	}

L
Luciano Coelho 已提交
272
	/* configure packet life time */
273 274 275
	getnstimeofday(&ts);
	hosttime = (timespec_to_ns(&ts) >> 10);
	desc->start_time = cpu_to_le32(hosttime - wl->time_offset);
276

277
	is_dummy = wl12xx_is_dummy_packet(wl, skb);
E
Eliad Peller 已提交
278
	if (is_dummy || !wlvif || wlvif->bss_type != BSS_TYPE_AP_BSS)
279 280 281
		desc->life_time = cpu_to_le16(TX_HW_MGMT_PKT_LIFETIME_TU);
	else
		desc->life_time = cpu_to_le16(TX_HW_AP_MODE_PKT_LIFETIME_TU);
L
Luciano Coelho 已提交
282

283
	/* queue */
K
Kalle Valo 已提交
284
	ac = wl1271_tx_get_queue(skb_get_queue_mapping(skb));
285
	desc->tid = skb->priority;
286

287
	if (is_dummy) {
288 289 290 291
		/*
		 * FW expects the dummy packet to have an invalid session id -
		 * any session id that is different than the one set in the join
		 */
292
		tx_attr = (SESSION_COUNTER_INVALID <<
293 294 295 296
			   TX_HW_ATTR_OFST_SESSION_COUNTER) &
			   TX_HW_ATTR_SESSION_COUNTER;

		tx_attr |= TX_HW_ATTR_TX_DUMMY_REQ;
E
Eliad Peller 已提交
297
	} else if (wlvif) {
298
		/* configure the tx attributes */
299 300
		tx_attr = wlvif->session_counter <<
			  TX_HW_ATTR_OFST_SESSION_COUNTER;
301 302
	}

E
Eliad Peller 已提交
303
	desc->hlid = hlid;
E
Eliad Peller 已提交
304
	if (is_dummy || !wlvif)
305 306
		rate_idx = 0;
	else if (wlvif->bss_type != BSS_TYPE_AP_BSS) {
307 308 309
		/* if the packets are destined for AP (have a STA entry)
		   send them with AP rate policies, otherwise use default
		   basic rates */
310 311 312
		if (control->flags & IEEE80211_TX_CTL_NO_CCK_RATE)
			rate_idx = wlvif->sta.p2p_rate_idx;
		else if (control->control.sta)
E
Eliad Peller 已提交
313
			rate_idx = wlvif->sta.ap_rate_idx;
314
		else
E
Eliad Peller 已提交
315
			rate_idx = wlvif->sta.basic_rate_idx;
316
	} else {
317
		if (hlid == wlvif->ap.global_hlid)
E
Eliad Peller 已提交
318
			rate_idx = wlvif->ap.mgmt_rate_idx;
319
		else if (hlid == wlvif->ap.bcast_hlid)
E
Eliad Peller 已提交
320
			rate_idx = wlvif->ap.bcast_rate_idx;
321
		else
E
Eliad Peller 已提交
322
			rate_idx = wlvif->ap.ucast_rate_idx[ac];
323 324 325
	}

	tx_attr |= rate_idx << TX_HW_ATTR_OFST_RATE_POLICY;
L
Luciano Coelho 已提交
326

327 328 329 330 331
	/* for WEP shared auth - no fw encryption is needed */
	if (ieee80211_is_auth(frame_control) &&
	    ieee80211_has_protected(frame_control))
		tx_attr |= TX_HW_ATTR_HOST_ENCRYPT;

L
Luciano Coelho 已提交
332
	desc->tx_attr = cpu_to_le16(tx_attr);
333

334
	wlcore_hw_set_tx_desc_csum(wl, desc, skb);
335
	wlcore_hw_set_tx_desc_data_len(wl, desc, skb);
L
Luciano Coelho 已提交
336 337 338
}

/* caller must hold wl->mutex */
339 340
static int wl1271_prepare_tx_frame(struct wl1271 *wl, struct wl12xx_vif *wlvif,
				   struct sk_buff *skb, u32 buf_offset)
L
Luciano Coelho 已提交
341 342 343 344
{
	struct ieee80211_tx_info *info;
	u32 extra = 0;
	int ret = 0;
345
	u32 total_len;
346
	u8 hlid;
E
Eliad Peller 已提交
347
	bool is_dummy;
348
	bool is_gem = false;
L
Luciano Coelho 已提交
349 350 351 352 353 354

	if (!skb)
		return -EINVAL;

	info = IEEE80211_SKB_CB(skb);

E
Eliad Peller 已提交
355 356 357
	/* TODO: handle dummy packets on multi-vifs */
	is_dummy = wl12xx_is_dummy_packet(wl, skb);

358 359
	if ((wl->quirks & WLCORE_QUIRK_TKIP_HEADER_SPACE) &&
	    info->control.hw_key &&
360
	    info->control.hw_key->cipher == WLAN_CIPHER_SUITE_TKIP)
361
		extra = WL1271_EXTRA_SPACE_TKIP;
L
Luciano Coelho 已提交
362 363

	if (info->control.hw_key) {
364 365 366 367 368 369
		bool is_wep;
		u8 idx = info->control.hw_key->hw_key_idx;
		u32 cipher = info->control.hw_key->cipher;

		is_wep = (cipher == WLAN_CIPHER_SUITE_WEP40) ||
			 (cipher == WLAN_CIPHER_SUITE_WEP104);
L
Luciano Coelho 已提交
370

E
Eliad Peller 已提交
371
		if (unlikely(is_wep && wlvif->default_key != idx)) {
E
Eliad Peller 已提交
372
			ret = wl1271_set_default_wep_key(wl, wlvif, idx);
L
Luciano Coelho 已提交
373 374
			if (ret < 0)
				return ret;
E
Eliad Peller 已提交
375
			wlvif->default_key = idx;
L
Luciano Coelho 已提交
376
		}
377 378

		is_gem = (cipher == WL1271_CIPHER_SUITE_GEM);
L
Luciano Coelho 已提交
379
	}
380
	hlid = wl12xx_tx_get_hlid(wl, wlvif, skb);
E
Eliad Peller 已提交
381
	if (hlid == WL12XX_INVALID_LINK_ID) {
382
		wl1271_error("invalid hlid. dropping skb 0x%p", skb);
E
Eliad Peller 已提交
383 384
		return -EINVAL;
	}
385

386 387
	ret = wl1271_tx_allocate(wl, wlvif, skb, extra, buf_offset, hlid,
				 is_gem);
L
Luciano Coelho 已提交
388 389 390
	if (ret < 0)
		return ret;

391
	wl1271_tx_fill_hdr(wl, wlvif, skb, extra, info, hlid);
392

E
Eliad Peller 已提交
393
	if (!is_dummy && wlvif && wlvif->bss_type == BSS_TYPE_AP_BSS) {
394
		wl1271_tx_ap_update_inconnection_sta(wl, skb);
395
		wl1271_tx_regulate_link(wl, wlvif, hlid);
396
	}
397

398
	/*
399 400 401 402 403 404
	 * The length of each packet is stored in terms of
	 * words. Thus, we must pad the skb data to make sure its
	 * length is aligned.  The number of padding bytes is computed
	 * and set in wl1271_tx_fill_hdr.
	 * In special cases, we want to align to a specific block size
	 * (eg. for wl128x with SDIO we align to 256).
405
	 */
406
	total_len = wlcore_calc_packet_alignment(wl, skb->len);
407

408 409
	memcpy(wl->aggr_buf + buf_offset, skb->data, skb->len);
	memset(wl->aggr_buf + buf_offset + skb->len, 0, total_len - skb->len);
L
Luciano Coelho 已提交
410

411
	/* Revert side effects in the dummy packet skb, so it can be reused */
E
Eliad Peller 已提交
412
	if (is_dummy)
413 414
		skb_pull(skb, sizeof(struct wl1271_tx_hw_descr));

415
	return total_len;
L
Luciano Coelho 已提交
416 417
}

418 419
u32 wl1271_tx_enabled_rates_get(struct wl1271 *wl, u32 rate_set,
				enum ieee80211_band rate_band)
420 421 422 423 424
{
	struct ieee80211_supported_band *band;
	u32 enabled_rates = 0;
	int bit;

425
	band = wl->hw->wiphy->bands[rate_band];
426 427 428 429 430 431
	for (bit = 0; bit < band->n_bitrates; bit++) {
		if (rate_set & 0x1)
			enabled_rates |= band->bitrates[bit].hw_value;
		rate_set >>= 1;
	}

A
Arik Nemtsov 已提交
432
	/* MCS rates indication are on bits 16 - 31 */
433 434
	rate_set >>= HW_HT_RATES_OFFSET - band->n_bitrates;

A
Arik Nemtsov 已提交
435
	for (bit = 0; bit < 16; bit++) {
436 437 438 439 440
		if (rate_set & 0x1)
			enabled_rates |= (CONF_HW_BIT_RATE_MCS_0 << bit);
		rate_set >>= 1;
	}

441 442 443
	return enabled_rates;
}

444
void wl1271_handle_tx_low_watermark(struct wl1271 *wl)
445
{
446
	int i;
447

448
	for (i = 0; i < NUM_TX_QUEUES; i++) {
449 450
		if (wlcore_is_queue_stopped_by_reason(wl, i,
			WLCORE_QUEUE_STOP_REASON_WATERMARK) &&
451
		    wl->tx_queue_count[i] <= WL1271_TX_QUEUE_LOW_WATERMARK) {
452
			/* firmware buffer has space, restart queues */
453 454
			wlcore_wake_queue(wl, i,
					  WLCORE_QUEUE_STOP_REASON_WATERMARK);
455
		}
456 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
static struct sk_buff_head *wl1271_select_queue(struct wl1271 *wl,
						struct sk_buff_head *queues)
{
	int i, q = -1, ac;
	u32 min_pkts = 0xffffffff;

	/*
	 * Find a non-empty ac where:
	 * 1. There are packets to transmit
	 * 2. The FW has the least allocated blocks
	 *
	 * We prioritize the ACs according to VO>VI>BE>BK
	 */
	for (i = 0; i < NUM_TX_QUEUES; i++) {
		ac = wl1271_tx_get_queue(i);
		if (!skb_queue_empty(&queues[ac]) &&
		    (wl->tx_allocated_pkts[ac] < min_pkts)) {
			q = ac;
			min_pkts = wl->tx_allocated_pkts[q];
		}
	}

	if (q == -1)
		return NULL;

	return &queues[q];
}

487 488
static struct sk_buff *wl12xx_lnk_skb_dequeue(struct wl1271 *wl,
					      struct wl1271_link *lnk)
489
{
490
	struct sk_buff *skb;
491
	unsigned long flags;
492
	struct sk_buff_head *queue;
493

494
	queue = wl1271_select_queue(wl, lnk->tx_queue);
495
	if (!queue)
496
		return NULL;
497 498

	skb = skb_dequeue(queue);
499
	if (skb) {
500
		int q = wl1271_tx_get_queue(skb_get_queue_mapping(skb));
501
		spin_lock_irqsave(&wl->wl_lock, flags);
502
		WARN_ON_ONCE(wl->tx_queue_count[q] <= 0);
503
		wl->tx_queue_count[q]--;
504 505 506 507 508 509
		spin_unlock_irqrestore(&wl->wl_lock, flags);
	}

	return skb;
}

510 511
static struct sk_buff *wl12xx_vif_skb_dequeue(struct wl1271 *wl,
					      struct wl12xx_vif *wlvif)
512 513 514 515 516
{
	struct sk_buff *skb = NULL;
	int i, h, start_hlid;

	/* start from the link after the last one */
517
	start_hlid = (wlvif->last_tx_hlid + 1) % WL12XX_MAX_LINKS;
518 519

	/* dequeue according to AC, round robin on each link */
520 521
	for (i = 0; i < WL12XX_MAX_LINKS; i++) {
		h = (start_hlid + i) % WL12XX_MAX_LINKS;
522

523
		/* only consider connected stations */
524
		if (!test_bit(h, wlvif->links_map))
525 526
			continue;

527 528
		skb = wl12xx_lnk_skb_dequeue(wl, &wl->links[h]);
		if (!skb)
529 530
			continue;

531 532
		wlvif->last_tx_hlid = h;
		break;
533 534
	}

535
	if (!skb)
536
		wlvif->last_tx_hlid = 0;
537 538 539 540

	return skb;
}

541
static struct sk_buff *wl1271_skb_dequeue(struct wl1271 *wl)
542
{
543
	unsigned long flags;
544
	struct wl12xx_vif *wlvif = wl->last_wlvif;
545 546
	struct sk_buff *skb = NULL;

547
	/* continue from last wlvif (round robin) */
548 549 550 551 552 553 554 555 556 557
	if (wlvif) {
		wl12xx_for_each_wlvif_continue(wl, wlvif) {
			skb = wl12xx_vif_skb_dequeue(wl, wlvif);
			if (skb) {
				wl->last_wlvif = wlvif;
				break;
			}
		}
	}

558 559 560 561 562
	/* dequeue from the system HLID before the restarting wlvif list */
	if (!skb)
		skb = wl12xx_lnk_skb_dequeue(wl, &wl->links[wl->system_hlid]);

	/* do a new pass over the wlvif list */
563 564 565 566 567 568 569
	if (!skb) {
		wl12xx_for_each_wlvif(wl, wlvif) {
			skb = wl12xx_vif_skb_dequeue(wl, wlvif);
			if (skb) {
				wl->last_wlvif = wlvif;
				break;
			}
570 571 572 573 574 575 576

			/*
			 * No need to continue after last_wlvif. The previous
			 * pass should have found it.
			 */
			if (wlvif == wl->last_wlvif)
				break;
577
		}
578 579
	}

580 581
	if (!skb &&
	    test_and_clear_bit(WL1271_FLAG_DUMMY_PACKET_PENDING, &wl->flags)) {
582 583
		int q;

584
		skb = wl->dummy_packet;
585
		q = wl1271_tx_get_queue(skb_get_queue_mapping(skb));
586
		spin_lock_irqsave(&wl->wl_lock, flags);
587
		WARN_ON_ONCE(wl->tx_queue_count[q] <= 0);
588
		wl->tx_queue_count[q]--;
589 590 591 592
		spin_unlock_irqrestore(&wl->wl_lock, flags);
	}

	return skb;
593 594
}

595
static void wl1271_skb_queue_head(struct wl1271 *wl, struct wl12xx_vif *wlvif,
E
Eliad Peller 已提交
596
				  struct sk_buff *skb)
597 598 599 600
{
	unsigned long flags;
	int q = wl1271_tx_get_queue(skb_get_queue_mapping(skb));

601 602
	if (wl12xx_is_dummy_packet(wl, skb)) {
		set_bit(WL1271_FLAG_DUMMY_PACKET_PENDING, &wl->flags);
603 604
	} else {
		u8 hlid = wl12xx_tx_get_hlid(wl, wlvif, skb);
605 606 607
		skb_queue_head(&wl->links[hlid].tx_queue[q], skb);

		/* make sure we dequeue the same packet next time */
608
		wlvif->last_tx_hlid = (hlid + WL12XX_MAX_LINKS - 1) %
609
				      WL12XX_MAX_LINKS;
610 611
	}

612
	spin_lock_irqsave(&wl->wl_lock, flags);
613
	wl->tx_queue_count[q]++;
614 615 616
	spin_unlock_irqrestore(&wl->wl_lock, flags);
}

617 618 619 620 621 622 623
static bool wl1271_tx_is_data_present(struct sk_buff *skb)
{
	struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)(skb->data);

	return ieee80211_is_data_present(hdr->frame_control);
}

E
Eliad Peller 已提交
624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650
void wl12xx_rearm_rx_streaming(struct wl1271 *wl, unsigned long *active_hlids)
{
	struct wl12xx_vif *wlvif;
	u32 timeout;
	u8 hlid;

	if (!wl->conf.rx_streaming.interval)
		return;

	if (!wl->conf.rx_streaming.always &&
	    !test_bit(WL1271_FLAG_SOFT_GEMINI, &wl->flags))
		return;

	timeout = wl->conf.rx_streaming.duration;
	wl12xx_for_each_wlvif_sta(wl, wlvif) {
		bool found = false;
		for_each_set_bit(hlid, active_hlids, WL12XX_MAX_LINKS) {
			if (test_bit(hlid, wlvif->links_map)) {
				found  = true;
				break;
			}
		}

		if (!found)
			continue;

		/* enable rx streaming */
651
		if (!test_bit(WLVIF_FLAG_RX_STREAMING_STARTED, &wlvif->flags))
E
Eliad Peller 已提交
652 653 654 655 656 657 658 659
			ieee80211_queue_work(wl->hw,
					     &wlvif->rx_streaming_enable_work);

		mod_timer(&wlvif->rx_streaming_timer,
			  jiffies + msecs_to_jiffies(timeout));
	}
}

660
void wl1271_tx_work_locked(struct wl1271 *wl)
L
Luciano Coelho 已提交
661
{
662
	struct wl12xx_vif *wlvif;
L
Luciano Coelho 已提交
663
	struct sk_buff *skb;
E
Eliad Peller 已提交
664
	struct wl1271_tx_hw_descr *desc;
I
Ido Yariv 已提交
665 666
	u32 buf_offset = 0;
	bool sent_packets = false;
E
Eliad Peller 已提交
667
	unsigned long active_hlids[BITS_TO_LONGS(WL12XX_MAX_LINKS)] = {0};
L
Luciano Coelho 已提交
668 669 670
	int ret;

	if (unlikely(wl->state == WL1271_STATE_OFF))
671
		return;
L
Luciano Coelho 已提交
672

673
	while ((skb = wl1271_skb_dequeue(wl))) {
E
Eliad Peller 已提交
674
		struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
E
Eliad Peller 已提交
675 676
		bool has_data = false;

677
		wlvif = NULL;
E
Eliad Peller 已提交
678 679
		if (!wl12xx_is_dummy_packet(wl, skb) && info->control.vif)
			wlvif = wl12xx_vif_to_data(info->control.vif);
680

E
Eliad Peller 已提交
681
		has_data = wlvif && wl1271_tx_is_data_present(skb);
682
		ret = wl1271_prepare_tx_frame(wl, wlvif, skb, buf_offset);
I
Ido Yariv 已提交
683
		if (ret == -EAGAIN) {
684
			/*
I
Ido Yariv 已提交
685 686 687
			 * Aggregation buffer is full.
			 * Flush buffer and try again.
			 */
688
			wl1271_skb_queue_head(wl, wlvif, skb);
689 690
			wlcore_write_data(wl, REG_SLV_MEM_DATA, wl->aggr_buf,
					  buf_offset, true);
I
Ido Yariv 已提交
691 692 693 694 695 696
			sent_packets = true;
			buf_offset = 0;
			continue;
		} else if (ret == -EBUSY) {
			/*
			 * Firmware buffer is full.
697 698
			 * Queue back last skb, and stop aggregating.
			 */
699
			wl1271_skb_queue_head(wl, wlvif, skb);
I
Ido Yariv 已提交
700 701
			/* No work left, avoid scheduling redundant tx work */
			set_bit(WL1271_FLAG_FW_TX_BUSY, &wl->flags);
702
			goto out_ack;
L
Luciano Coelho 已提交
703
		} else if (ret < 0) {
E
Eliad Peller 已提交
704 705 706 707 708 709 710 711
			if (wl12xx_is_dummy_packet(wl, skb))
				/*
				 * fw still expects dummy packet,
				 * so re-enqueue it
				 */
				wl1271_skb_queue_head(wl, wlvif, skb);
			else
				ieee80211_free_txskb(wl->hw, skb);
712
			goto out_ack;
L
Luciano Coelho 已提交
713
		}
714 715
		buf_offset += ret;
		wl->tx_packets_count++;
E
Eliad Peller 已提交
716 717 718 719
		if (has_data) {
			desc = (struct wl1271_tx_hw_descr *) skb->data;
			__set_bit(desc->hlid, active_hlids);
		}
L
Luciano Coelho 已提交
720 721
	}

722
out_ack:
723
	if (buf_offset) {
724 725
		wlcore_write_data(wl, REG_SLV_MEM_DATA, wl->aggr_buf,
				  buf_offset, true);
I
Ido Yariv 已提交
726 727 728
		sent_packets = true;
	}
	if (sent_packets) {
729 730 731 732
		/*
		 * Interrupt the firmware with the new packets. This is only
		 * required for older hardware revisions
		 */
733
		if (wl->quirks & WLCORE_QUIRK_END_OF_TRANSACTION)
734
			wl1271_write32(wl, WL12XX_HOST_WR_ACCESS,
735 736
				       wl->tx_packets_count);

737
		wl1271_handle_tx_low_watermark(wl);
738
	}
E
Eliad Peller 已提交
739
	wl12xx_rearm_rx_streaming(wl, active_hlids);
I
Ido Yariv 已提交
740
}
L
Luciano Coelho 已提交
741

I
Ido Yariv 已提交
742 743 744
void wl1271_tx_work(struct work_struct *work)
{
	struct wl1271 *wl = container_of(work, struct wl1271, tx_work);
745
	int ret;
I
Ido Yariv 已提交
746 747

	mutex_lock(&wl->mutex);
748 749 750 751
	ret = wl1271_ps_elp_wakeup(wl);
	if (ret < 0)
		goto out;

752
	wl1271_tx_work_locked(wl);
753

754
	wl1271_ps_elp_sleep(wl);
755
out:
L
Luciano Coelho 已提交
756 757 758
	mutex_unlock(&wl->mutex);
}

759 760
static u8 wl1271_tx_get_rate_flags(u8 rate_class_index)
{
761 762
	u8 flags = 0;

763 764 765 766 767
	/*
	 * TODO: use wl12xx constants when this code is moved to wl12xx, as
	 * only it uses Tx-completion.
	 */
	if (rate_class_index <= 8)
768
		flags |= IEEE80211_TX_RC_MCS;
769 770 771 772 773 774

	/*
	 * TODO: use wl12xx constants when this code is moved to wl12xx, as
	 * only it uses Tx-completion.
	 */
	if (rate_class_index == 0)
775
		flags |= IEEE80211_TX_RC_SHORT_GI;
776

777
	return flags;
778 779
}

L
Luciano Coelho 已提交
780 781 782 783
static void wl1271_tx_complete_packet(struct wl1271 *wl,
				      struct wl1271_tx_hw_res_descr *result)
{
	struct ieee80211_tx_info *info;
784 785
	struct ieee80211_vif *vif;
	struct wl12xx_vif *wlvif;
L
Luciano Coelho 已提交
786 787
	struct sk_buff *skb;
	int id = result->id;
J
Juuso Oikarinen 已提交
788
	int rate = -1;
789
	u8 rate_flags = 0;
J
Juuso Oikarinen 已提交
790
	u8 retries = 0;
L
Luciano Coelho 已提交
791 792

	/* check for id legality */
793
	if (unlikely(id >= wl->num_tx_desc || wl->tx_frames[id] == NULL)) {
L
Luciano Coelho 已提交
794 795 796 797 798 799 800
		wl1271_warning("TX result illegal id: %d", id);
		return;
	}

	skb = wl->tx_frames[id];
	info = IEEE80211_SKB_CB(skb);

801
	if (wl12xx_is_dummy_packet(wl, skb)) {
802 803 804 805
		wl1271_free_tx_id(wl, id);
		return;
	}

806 807 808 809
	/* info->control is valid as long as we don't update info->status */
	vif = info->control.vif;
	wlvif = wl12xx_vif_to_data(vif);

J
Juuso Oikarinen 已提交
810 811 812
	/* update the TX status info */
	if (result->status == TX_SUCCESS) {
		if (!(info->flags & IEEE80211_TX_CTL_NO_ACK))
L
Luciano Coelho 已提交
813
			info->flags |= IEEE80211_TX_STAT_ACK;
814
		rate = wlcore_rate_to_idx(wl, result->rate_class_index,
E
Eliad Peller 已提交
815
					  wlvif->band);
816
		rate_flags = wl1271_tx_get_rate_flags(result->rate_class_index);
J
Juuso Oikarinen 已提交
817 818 819 820
		retries = result->ack_failures;
	} else if (result->status == TX_RETRY_EXCEEDED) {
		wl->stats.excessive_retries++;
		retries = result->ack_failures;
L
Luciano Coelho 已提交
821 822
	}

J
Juuso Oikarinen 已提交
823 824
	info->status.rates[0].idx = rate;
	info->status.rates[0].count = retries;
825
	info->status.rates[0].flags = rate_flags;
J
Juuso Oikarinen 已提交
826 827
	info->status.ack_signal = -1;

L
Luciano Coelho 已提交
828 829
	wl->stats.retry_count += result->ack_failures;

830 831 832 833 834 835 836 837 838
	/*
	 * update sequence number only when relevant, i.e. only in
	 * sessions of TKIP, AES and GEM (not in open or WEP sessions)
	 */
	if (info->control.hw_key &&
	    (info->control.hw_key->cipher == WLAN_CIPHER_SUITE_TKIP ||
	     info->control.hw_key->cipher == WLAN_CIPHER_SUITE_CCMP ||
	     info->control.hw_key->cipher == WL1271_CIPHER_SUITE_GEM)) {
		u8 fw_lsb = result->tx_security_sequence_number_lsb;
839
		u8 cur_lsb = wlvif->tx_security_last_seq_lsb;
840 841 842 843 844

		/*
		 * update security sequence number, taking care of potential
		 * wrap-around
		 */
845 846
		wlvif->tx_security_seq += (fw_lsb - cur_lsb) & 0xff;
		wlvif->tx_security_last_seq_lsb = fw_lsb;
847
	}
848

849 850 851 852
	/* remove private header from packet */
	skb_pull(skb, sizeof(struct wl1271_tx_hw_descr));

	/* remove TKIP header space if present */
853 854
	if ((wl->quirks & WLCORE_QUIRK_TKIP_HEADER_SPACE) &&
	    info->control.hw_key &&
855
	    info->control.hw_key->cipher == WLAN_CIPHER_SUITE_TKIP) {
856
		int hdrlen = ieee80211_get_hdrlen_from_skb(skb);
857 858 859
		memmove(skb->data + WL1271_EXTRA_SPACE_TKIP, skb->data,
			hdrlen);
		skb_pull(skb, WL1271_EXTRA_SPACE_TKIP);
860
	}
L
Luciano Coelho 已提交
861 862 863 864 865 866 867

	wl1271_debug(DEBUG_TX, "tx status id %u skb 0x%p failures %u rate 0x%x"
		     " status 0x%x",
		     result->id, skb, result->ack_failures,
		     result->rate_class_index, result->status);

	/* return the packet to the stack */
868
	skb_queue_tail(&wl->deferred_tx_queue, skb);
869
	queue_work(wl->freezable_wq, &wl->netstack_work);
870
	wl1271_free_tx_id(wl, result->id);
L
Luciano Coelho 已提交
871 872 873
}

/* Called upon reception of a TX complete interrupt */
874
void wl1271_tx_complete(struct wl1271 *wl)
L
Luciano Coelho 已提交
875 876 877
{
	struct wl1271_acx_mem_map *memmap =
		(struct wl1271_acx_mem_map *)wl->target_mem_map;
878
	u32 count, fw_counter;
L
Luciano Coelho 已提交
879 880 881
	u32 i;

	/* read the tx results from the chipset */
T
Teemu Paasikivi 已提交
882 883
	wl1271_read(wl, le32_to_cpu(memmap->tx_result),
		    wl->tx_res_if, sizeof(*wl->tx_res_if), false);
884 885 886 887 888 889 890 891
	fw_counter = le32_to_cpu(wl->tx_res_if->tx_result_fw_counter);

	/* write host counter to chipset (to ack) */
	wl1271_write32(wl, le32_to_cpu(memmap->tx_result) +
		       offsetof(struct wl1271_tx_hw_res_if,
				tx_result_host_counter), fw_counter);

	count = fw_counter - wl->tx_results_count;
892
	wl1271_debug(DEBUG_TX, "tx_complete received, packets: %d", count);
L
Luciano Coelho 已提交
893 894

	/* verify that the result buffer is not getting overrun */
895
	if (unlikely(count > TX_HW_RESULT_QUEUE_LEN))
L
Luciano Coelho 已提交
896 897 898 899 900 901 902 903 904 905 906 907 908 909
		wl1271_warning("TX result overflow from chipset: %d", count);

	/* process the results */
	for (i = 0; i < count; i++) {
		struct wl1271_tx_hw_res_descr *result;
		u8 offset = wl->tx_results_count & TX_HW_RESULT_QUEUE_LEN_MASK;

		/* process the packet */
		result =  &(wl->tx_res_if->tx_results_queue[offset]);
		wl1271_tx_complete_packet(wl, result);

		wl->tx_results_count++;
	}
}
910
EXPORT_SYMBOL(wl1271_tx_complete);
L
Luciano Coelho 已提交
911

912 913 914
void wl1271_tx_reset_link_queues(struct wl1271 *wl, u8 hlid)
{
	struct sk_buff *skb;
915
	int i;
916
	unsigned long flags;
917
	struct ieee80211_tx_info *info;
918
	int total[NUM_TX_QUEUES];
919 920

	for (i = 0; i < NUM_TX_QUEUES; i++) {
921
		total[i] = 0;
922 923
		while ((skb = skb_dequeue(&wl->links[hlid].tx_queue[i]))) {
			wl1271_debug(DEBUG_TX, "link freeing skb 0x%p", skb);
924 925 926 927 928 929 930 931

			if (!wl12xx_is_dummy_packet(wl, skb)) {
				info = IEEE80211_SKB_CB(skb);
				info->status.rates[0].idx = -1;
				info->status.rates[0].count = 0;
				ieee80211_tx_status_ni(wl->hw, skb);
			}

932
			total[i]++;
933 934 935 936
		}
	}

	spin_lock_irqsave(&wl->wl_lock, flags);
937 938
	for (i = 0; i < NUM_TX_QUEUES; i++)
		wl->tx_queue_count[i] -= total[i];
939 940 941 942 943
	spin_unlock_irqrestore(&wl->wl_lock, flags);

	wl1271_handle_tx_low_watermark(wl);
}

944
/* caller must hold wl->mutex and TX must be stopped */
945
void wl12xx_tx_reset_wlvif(struct wl1271 *wl, struct wl12xx_vif *wlvif)
L
Luciano Coelho 已提交
946 947 948 949
{
	int i;

	/* TX failure */
950 951
	for_each_set_bit(i, wlvif->links_map, WL12XX_MAX_LINKS) {
		if (wlvif->bss_type == BSS_TYPE_AP_BSS)
952
			wl1271_free_sta(wl, wlvif, i);
953 954
		else
			wlvif->sta.ba_rx_bitmap = 0;
955

956 957
		wl->links[i].allocated_pkts = 0;
		wl->links[i].prev_freed_pkts = 0;
L
Luciano Coelho 已提交
958
	}
959 960 961 962
	wlvif->last_tx_hlid = 0;

}
/* caller must hold wl->mutex and TX must be stopped */
963
void wl12xx_tx_reset(struct wl1271 *wl)
964 965 966 967
{
	int i;
	struct sk_buff *skb;
	struct ieee80211_tx_info *info;
968

969 970 971 972 973 974 975 976
	/* only reset the queues if something bad happened */
	if (WARN_ON_ONCE(wl1271_tx_total_queue_count(wl) != 0)) {
		for (i = 0; i < WL12XX_MAX_LINKS; i++)
			wl1271_tx_reset_link_queues(wl, i);

		for (i = 0; i < NUM_TX_QUEUES; i++)
			wl->tx_queue_count[i] = 0;
	}
977

978 979 980
	/*
	 * Make sure the driver is at a consistent state, in case this
	 * function is called from a context other than interface removal.
981
	 * This call will always wake the TX queues.
982
	 */
983
	wl1271_handle_tx_low_watermark(wl);
984

985
	for (i = 0; i < wl->num_tx_desc; i++) {
986 987 988 989 990 991 992
		if (wl->tx_frames[i] == NULL)
			continue;

		skb = wl->tx_frames[i];
		wl1271_free_tx_id(wl, i);
		wl1271_debug(DEBUG_TX, "freeing skb 0x%p", skb);

993
		if (!wl12xx_is_dummy_packet(wl, skb)) {
994 995 996 997 998 999
			/*
			 * Remove private headers before passing the skb to
			 * mac80211
			 */
			info = IEEE80211_SKB_CB(skb);
			skb_pull(skb, sizeof(struct wl1271_tx_hw_descr));
1000 1001
			if ((wl->quirks & WLCORE_QUIRK_TKIP_HEADER_SPACE) &&
			    info->control.hw_key &&
1002 1003 1004
			    info->control.hw_key->cipher ==
			    WLAN_CIPHER_SUITE_TKIP) {
				int hdrlen = ieee80211_get_hdrlen_from_skb(skb);
1005
				memmove(skb->data + WL1271_EXTRA_SPACE_TKIP,
1006
					skb->data, hdrlen);
1007
				skb_pull(skb, WL1271_EXTRA_SPACE_TKIP);
1008
			}
1009

1010 1011
			info->status.rates[0].idx = -1;
			info->status.rates[0].count = 0;
1012

1013
			ieee80211_tx_status_ni(wl->hw, skb);
1014
		}
1015
	}
1016 1017 1018 1019 1020 1021 1022 1023
}

#define WL1271_TX_FLUSH_TIMEOUT 500000

/* caller must *NOT* hold wl->mutex */
void wl1271_tx_flush(struct wl1271 *wl)
{
	unsigned long timeout;
1024
	int i;
1025 1026
	timeout = jiffies + usecs_to_jiffies(WL1271_TX_FLUSH_TIMEOUT);

A
Arik Nemtsov 已提交
1027 1028 1029 1030 1031
	/* only one flush should be in progress, for consistent queue state */
	mutex_lock(&wl->flush_mutex);

	wlcore_stop_queues(wl, WLCORE_QUEUE_STOP_REASON_FLUSH);

1032 1033
	while (!time_after(jiffies, timeout)) {
		mutex_lock(&wl->mutex);
1034
		wl1271_debug(DEBUG_TX, "flushing tx buffer: %d %d",
1035 1036 1037 1038
			     wl->tx_frames_cnt,
			     wl1271_tx_total_queue_count(wl));
		if ((wl->tx_frames_cnt == 0) &&
		    (wl1271_tx_total_queue_count(wl) == 0)) {
1039
			mutex_unlock(&wl->mutex);
A
Arik Nemtsov 已提交
1040
			goto out;
1041 1042 1043 1044 1045 1046
		}
		mutex_unlock(&wl->mutex);
		msleep(1);
	}

	wl1271_warning("Unable to flush all TX buffers, timed out.");
1047 1048 1049 1050 1051 1052

	/* forcibly flush all Tx buffers on our queues */
	mutex_lock(&wl->mutex);
	for (i = 0; i < WL12XX_MAX_LINKS; i++)
		wl1271_tx_reset_link_queues(wl, i);
	mutex_unlock(&wl->mutex);
A
Arik Nemtsov 已提交
1053 1054 1055 1056

out:
	wlcore_wake_queues(wl, WLCORE_QUEUE_STOP_REASON_FLUSH);
	mutex_unlock(&wl->flush_mutex);
L
Luciano Coelho 已提交
1057
}
A
Arik Nemtsov 已提交
1058

1059
u32 wl1271_tx_min_rate_get(struct wl1271 *wl, u32 rate_set)
A
Arik Nemtsov 已提交
1060
{
1061 1062
	if (WARN_ON(!rate_set))
		return 0;
A
Arik Nemtsov 已提交
1063

1064
	return BIT(__ffs(rate_set));
A
Arik Nemtsov 已提交
1065
}
1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 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 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156

void wlcore_stop_queue_locked(struct wl1271 *wl, u8 queue,
			      enum wlcore_queue_stop_reason reason)
{
	bool stopped = !!wl->queue_stop_reasons[queue];

	/* queue should not be stopped for this reason */
	WARN_ON(test_and_set_bit(reason, &wl->queue_stop_reasons[queue]));

	if (stopped)
		return;

	ieee80211_stop_queue(wl->hw, wl1271_tx_get_mac80211_queue(queue));
}

void wlcore_stop_queue(struct wl1271 *wl, u8 queue,
		       enum wlcore_queue_stop_reason reason)
{
	unsigned long flags;

	spin_lock_irqsave(&wl->wl_lock, flags);
	wlcore_stop_queue_locked(wl, queue, reason);
	spin_unlock_irqrestore(&wl->wl_lock, flags);
}

void wlcore_wake_queue(struct wl1271 *wl, u8 queue,
		       enum wlcore_queue_stop_reason reason)
{
	unsigned long flags;

	spin_lock_irqsave(&wl->wl_lock, flags);

	/* queue should not be clear for this reason */
	WARN_ON(!test_and_clear_bit(reason, &wl->queue_stop_reasons[queue]));

	if (wl->queue_stop_reasons[queue])
		goto out;

	ieee80211_wake_queue(wl->hw, wl1271_tx_get_mac80211_queue(queue));

out:
	spin_unlock_irqrestore(&wl->wl_lock, flags);
}

void wlcore_stop_queues(struct wl1271 *wl,
			enum wlcore_queue_stop_reason reason)
{
	int i;

	for (i = 0; i < NUM_TX_QUEUES; i++)
		wlcore_stop_queue(wl, i, reason);
}

void wlcore_wake_queues(struct wl1271 *wl,
			enum wlcore_queue_stop_reason reason)
{
	int i;

	for (i = 0; i < NUM_TX_QUEUES; i++)
		wlcore_wake_queue(wl, i, reason);
}

void wlcore_reset_stopped_queues(struct wl1271 *wl)
{
	int i;
	unsigned long flags;

	spin_lock_irqsave(&wl->wl_lock, flags);

	for (i = 0; i < NUM_TX_QUEUES; i++) {
		if (!wl->queue_stop_reasons[i])
			continue;

		wl->queue_stop_reasons[i] = 0;
		ieee80211_wake_queue(wl->hw,
				     wl1271_tx_get_mac80211_queue(i));
	}

	spin_unlock_irqrestore(&wl->wl_lock, flags);
}

bool wlcore_is_queue_stopped_by_reason(struct wl1271 *wl, u8 queue,
			     enum wlcore_queue_stop_reason reason)
{
	return test_bit(reason, &wl->queue_stop_reasons[queue]);
}

bool wlcore_is_queue_stopped(struct wl1271 *wl, u8 queue)
{
	return !!wl->queue_stop_reasons[queue];
}