iwl-3945-rs.c 25.1 KB
Newer Older
Z
Zhu Yi 已提交
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
/******************************************************************************
 *
 * Copyright(c) 2005 - 2007 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.
 *
 * Contact Information:
 * James P. Ketrenos <ipw2100-admin@linux.intel.com>
 * Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
 *
 *****************************************************************************/

#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/skbuff.h>
#include <linux/wireless.h>
#include <net/mac80211.h>
#include <net/ieee80211.h>

#include <linux/netdevice.h>
#include <linux/etherdevice.h>
#include <linux/delay.h>

#include <linux/workqueue.h>

#include "../net/mac80211/ieee80211_rate.h"

42
#include "iwl-3945.h"
Z
Zhu Yi 已提交
43 44 45

#define RS_NAME "iwl-3945-rs"

C
Christoph Hellwig 已提交
46
struct iwl3945_rate_scale_data {
Z
Zhu Yi 已提交
47 48 49 50 51 52 53 54
	u64 data;
	s32 success_counter;
	s32 success_ratio;
	s32 counter;
	s32 average_tpt;
	unsigned long stamp;
};

55
struct iwl3945_rs_sta {
Z
Zhu Yi 已提交
56 57 58 59 60 61 62 63 64 65 66 67
	spinlock_t lock;
	s32 *expected_tpt;
	unsigned long last_partial_flush;
	unsigned long last_flush;
	u32 flush_time;
	u32 last_tx_packets;
	u32 tx_packets;
	u8 tgg;
	u8 flush_pending;
	u8 start_rate;
	u8 ibss_sta_added;
	struct timer_list rate_scale_flush;
C
Christoph Hellwig 已提交
68
	struct iwl3945_rate_scale_data win[IWL_RATE_COUNT];
Z
Zhu Yi 已提交
69 70
};

C
Christoph Hellwig 已提交
71
static s32 iwl3945_expected_tpt_g[IWL_RATE_COUNT] = {
72
	7, 13, 35, 58, 0, 0, 76, 104, 130, 168, 191, 202
Z
Zhu Yi 已提交
73 74
};

C
Christoph Hellwig 已提交
75
static s32 iwl3945_expected_tpt_g_prot[IWL_RATE_COUNT] = {
76
	7, 13, 35, 58, 0, 0, 0, 80, 93, 113, 123, 125
Z
Zhu Yi 已提交
77 78
};

C
Christoph Hellwig 已提交
79
static s32 iwl3945_expected_tpt_a[IWL_RATE_COUNT] = {
80
	0, 0, 0, 0, 40, 57, 72, 98, 121, 154, 177, 186
Z
Zhu Yi 已提交
81 82
};

C
Christoph Hellwig 已提交
83
static s32 iwl3945_expected_tpt_b[IWL_RATE_COUNT] = {
84
	7, 13, 35, 58, 0, 0, 0, 0, 0, 0, 0, 0
Z
Zhu Yi 已提交
85 86
};

C
Christoph Hellwig 已提交
87
struct iwl3945_tpt_entry {
Z
Zhu Yi 已提交
88 89 90 91
	s8 min_rssi;
	u8 index;
};

C
Christoph Hellwig 已提交
92
static struct iwl3945_tpt_entry iwl3945_tpt_table_a[] = {
Z
Zhu Yi 已提交
93 94 95 96 97 98 99 100 101 102
	{-60, IWL_RATE_54M_INDEX},
	{-64, IWL_RATE_48M_INDEX},
	{-72, IWL_RATE_36M_INDEX},
	{-80, IWL_RATE_24M_INDEX},
	{-84, IWL_RATE_18M_INDEX},
	{-85, IWL_RATE_12M_INDEX},
	{-87, IWL_RATE_9M_INDEX},
	{-89, IWL_RATE_6M_INDEX}
};

C
Christoph Hellwig 已提交
103
static struct iwl3945_tpt_entry iwl3945_tpt_table_b[] = {
Z
Zhu Yi 已提交
104 105 106 107 108 109 110
	{-86, IWL_RATE_11M_INDEX},
	{-88, IWL_RATE_5M_INDEX},
	{-90, IWL_RATE_2M_INDEX},
	{-92, IWL_RATE_1M_INDEX}

};

C
Christoph Hellwig 已提交
111
static struct iwl3945_tpt_entry iwl3945_tpt_table_g[] = {
Z
Zhu Yi 已提交
112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
	{-60, IWL_RATE_54M_INDEX},
	{-64, IWL_RATE_48M_INDEX},
	{-68, IWL_RATE_36M_INDEX},
	{-80, IWL_RATE_24M_INDEX},
	{-84, IWL_RATE_18M_INDEX},
	{-85, IWL_RATE_12M_INDEX},
	{-86, IWL_RATE_11M_INDEX},
	{-88, IWL_RATE_5M_INDEX},
	{-90, IWL_RATE_2M_INDEX},
	{-92, IWL_RATE_1M_INDEX}
};

#define IWL_RATE_MAX_WINDOW          62
#define IWL_RATE_FLUSH        (3*HZ/10)
#define IWL_RATE_WIN_FLUSH       (HZ/2)
#define IWL_RATE_HIGH_TH          11520
#define IWL_RATE_MIN_FAILURE_TH       8
#define IWL_RATE_MIN_SUCCESS_TH       8
#define IWL_RATE_DECREASE_TH       1920

C
Christoph Hellwig 已提交
132
static u8 iwl3945_get_rate_index_by_rssi(s32 rssi, u8 mode)
Z
Zhu Yi 已提交
133 134 135
{
	u32 index = 0;
	u32 table_size = 0;
C
Christoph Hellwig 已提交
136
	struct iwl3945_tpt_entry *tpt_table = NULL;
Z
Zhu Yi 已提交
137 138 139 140 141 142

	if ((rssi < IWL_MIN_RSSI_VAL) || (rssi > IWL_MAX_RSSI_VAL))
		rssi = IWL_MIN_RSSI_VAL;

	switch (mode) {
	case MODE_IEEE80211G:
C
Christoph Hellwig 已提交
143 144
		tpt_table = iwl3945_tpt_table_g;
		table_size = ARRAY_SIZE(iwl3945_tpt_table_g);
Z
Zhu Yi 已提交
145 146 147
		break;

	case MODE_IEEE80211A:
C
Christoph Hellwig 已提交
148 149
		tpt_table = iwl3945_tpt_table_a;
		table_size = ARRAY_SIZE(iwl3945_tpt_table_a);
Z
Zhu Yi 已提交
150 151 152 153
		break;

	default:
	case MODE_IEEE80211B:
C
Christoph Hellwig 已提交
154 155
		tpt_table = iwl3945_tpt_table_b;
		table_size = ARRAY_SIZE(iwl3945_tpt_table_b);
Z
Zhu Yi 已提交
156 157 158 159 160 161 162 163 164 165 166
		break;
	}

	while ((index < table_size) && (rssi < tpt_table[index].min_rssi))
		index++;

	index = min(index, (table_size - 1));

	return tpt_table[index].index;
}

C
Christoph Hellwig 已提交
167
static void iwl3945_clear_window(struct iwl3945_rate_scale_data *window)
Z
Zhu Yi 已提交
168 169 170 171 172 173 174 175 176 177
{
	window->data = 0;
	window->success_counter = 0;
	window->success_ratio = IWL_INVALID_VALUE;
	window->counter = 0;
	window->average_tpt = IWL_INVALID_VALUE;
	window->stamp = 0;
}

/**
C
Christoph Hellwig 已提交
178
 * iwl3945_rate_scale_flush_windows - flush out the rate scale windows
Z
Zhu Yi 已提交
179 180 181 182 183
 *
 * Returns the number of windows that have gathered data but were
 * not flushed.  If there were any that were not flushed, then
 * reschedule the rate flushing routine.
 */
184
static int iwl3945_rate_scale_flush_windows(struct iwl3945_rs_sta *rs_sta)
Z
Zhu Yi 已提交
185 186 187 188 189 190 191 192 193 194 195
{
	int unflushed = 0;
	int i;
	unsigned long flags;

	/*
	 * For each rate, if we have collected data on that rate
	 * and it has been more than IWL_RATE_WIN_FLUSH
	 * since we flushed, clear out the gathered statistics
	 */
	for (i = 0; i < IWL_RATE_COUNT; i++) {
196
		if (!rs_sta->win[i].counter)
Z
Zhu Yi 已提交
197 198
			continue;

199 200
		spin_lock_irqsave(&rs_sta->lock, flags);
		if (time_after(jiffies, rs_sta->win[i].stamp +
Z
Zhu Yi 已提交
201 202 203
			       IWL_RATE_WIN_FLUSH)) {
			IWL_DEBUG_RATE("flushing %d samples of rate "
				       "index %d\n",
204 205
				       rs_sta->win[i].counter, i);
			iwl3945_clear_window(&rs_sta->win[i]);
Z
Zhu Yi 已提交
206 207
		} else
			unflushed++;
208
		spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
209 210 211 212 213 214 215 216
	}

	return unflushed;
}

#define IWL_RATE_FLUSH_MAX              5000	/* msec */
#define IWL_RATE_FLUSH_MIN              50	/* msec */

C
Christoph Hellwig 已提交
217
static void iwl3945_bg_rate_scale_flush(unsigned long data)
Z
Zhu Yi 已提交
218
{
219
	struct iwl3945_rs_sta *rs_sta = (void *)data;
Z
Zhu Yi 已提交
220 221 222 223 224 225
	int unflushed = 0;
	unsigned long flags;
	u32 packet_count, duration, pps;

	IWL_DEBUG_RATE("enter\n");

226
	unflushed = iwl3945_rate_scale_flush_windows(rs_sta);
Z
Zhu Yi 已提交
227

228
	spin_lock_irqsave(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
229

230
	rs_sta->flush_pending = 0;
Z
Zhu Yi 已提交
231 232

	/* Number of packets Rx'd since last time this timer ran */
233
	packet_count = (rs_sta->tx_packets - rs_sta->last_tx_packets) + 1;
Z
Zhu Yi 已提交
234

235
	rs_sta->last_tx_packets = rs_sta->tx_packets + 1;
Z
Zhu Yi 已提交
236 237 238

	if (unflushed) {
		duration =
239 240
		    jiffies_to_msecs(jiffies - rs_sta->last_partial_flush);
/*              duration = jiffies_to_msecs(rs_sta->flush_time); */
Z
Zhu Yi 已提交
241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257

		IWL_DEBUG_RATE("Tx'd %d packets in %dms\n",
			       packet_count, duration);

		/* Determine packets per second */
		if (duration)
			pps = (packet_count * 1000) / duration;
		else
			pps = 0;

		if (pps) {
			duration = IWL_RATE_FLUSH_MAX / pps;
			if (duration < IWL_RATE_FLUSH_MIN)
				duration = IWL_RATE_FLUSH_MIN;
		} else
			duration = IWL_RATE_FLUSH_MAX;

258
		rs_sta->flush_time = msecs_to_jiffies(duration);
Z
Zhu Yi 已提交
259 260 261 262

		IWL_DEBUG_RATE("new flush period: %d msec ave %d\n",
			       duration, packet_count);

263 264
		mod_timer(&rs_sta->rate_scale_flush, jiffies +
			  rs_sta->flush_time);
Z
Zhu Yi 已提交
265

266
		rs_sta->last_partial_flush = jiffies;
Z
Zhu Yi 已提交
267 268 269 270 271
	}

	/* If there weren't any unflushed entries, we don't schedule the timer
	 * to run again */

272
	rs_sta->last_flush = jiffies;
Z
Zhu Yi 已提交
273

274
	spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
275 276 277 278 279

	IWL_DEBUG_RATE("leave\n");
}

/**
C
Christoph Hellwig 已提交
280
 * iwl3945_collect_tx_data - Update the success/failure sliding window
Z
Zhu Yi 已提交
281 282 283 284 285
 *
 * We keep a sliding window of the last 64 packets transmitted
 * at this rate.  window->data contains the bitmask of successful
 * packets.
 */
286
static void iwl3945_collect_tx_data(struct iwl3945_rs_sta *rs_sta,
C
Christoph Hellwig 已提交
287
				struct iwl3945_rate_scale_data *window,
Z
Zhu Yi 已提交
288 289 290 291 292 293 294 295 296 297
				int success, int retries)
{
	unsigned long flags;

	if (!retries) {
		IWL_DEBUG_RATE("leave: retries == 0 -- should be at least 1\n");
		return;
	}

	while (retries--) {
298
		spin_lock_irqsave(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325

		/* If we have filled up the window then subtract one from the
		 * success counter if the high-bit is counting toward
		 * success */
		if (window->counter == IWL_RATE_MAX_WINDOW) {
			if (window->data & (1ULL << (IWL_RATE_MAX_WINDOW - 1)))
				window->success_counter--;
		} else
			window->counter++;

		/* Slide the window to the left one bit */
		window->data = (window->data << 1);

		/* If this packet was a success then set the low bit high */
		if (success) {
			window->success_counter++;
			window->data |= 1;
		}

		/* window->counter can't be 0 -- it is either >0 or
		 * IWL_RATE_MAX_WINDOW */
		window->success_ratio = 12800 * window->success_counter /
		    window->counter;

		/* Tag this window as having been updated */
		window->stamp = jiffies;

326
		spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350
	}
}

static void rs_rate_init(void *priv_rate, void *priv_sta,
			 struct ieee80211_local *local, struct sta_info *sta)
{
	int i;

	IWL_DEBUG_RATE("enter\n");

	/* TODO: what is a good starting rate for STA? About middle? Maybe not
	 * the lowest or the highest rate.. Could consider using RSSI from
	 * previous packets? Need to have IEEE 802.1X auth succeed immediately
	 * after assoc.. */

	for (i = IWL_RATE_COUNT - 1; i >= 0; i--) {
		if (sta->supp_rates & (1 << i)) {
			sta->txrate = i;
			break;
		}
	}

	sta->last_txrate = sta->txrate;

351 352 353 354
	/* For MODE_IEEE80211A mode it start at IWL_FIRST_OFDM_RATE */
        if (local->hw.conf.phymode == MODE_IEEE80211A)
                sta->last_txrate += IWL_FIRST_OFDM_RATE;

Z
Zhu Yi 已提交
355 356 357 358 359 360 361 362
	IWL_DEBUG_RATE("leave\n");
}

static void *rs_alloc(struct ieee80211_local *local)
{
	return local->hw.priv;
}

363
/* rate scale requires free function to be implemented */
Z
Zhu Yi 已提交
364 365 366 367 368 369 370 371 372 373 374 375
static void rs_free(void *priv)
{
	return;
}
static void rs_clear(void *priv)
{
	return;
}


static void *rs_alloc_sta(void *priv, gfp_t gfp)
{
376
	struct iwl3945_rs_sta *rs_sta;
Z
Zhu Yi 已提交
377 378 379 380
	int i;

	IWL_DEBUG_RATE("enter\n");

381 382
	rs_sta = kzalloc(sizeof(struct iwl3945_rs_sta), gfp);
	if (!rs_sta) {
Z
Zhu Yi 已提交
383 384 385 386
		IWL_DEBUG_RATE("leave: ENOMEM\n");
		return NULL;
	}

387
	spin_lock_init(&rs_sta->lock);
Z
Zhu Yi 已提交
388

389
	rs_sta->start_rate = IWL_RATE_INVALID;
Z
Zhu Yi 已提交
390 391

	/* default to just 802.11b */
392
	rs_sta->expected_tpt = iwl3945_expected_tpt_b;
Z
Zhu Yi 已提交
393

394 395 396 397 398
	rs_sta->last_partial_flush = jiffies;
	rs_sta->last_flush = jiffies;
	rs_sta->flush_time = IWL_RATE_FLUSH;
	rs_sta->last_tx_packets = 0;
	rs_sta->ibss_sta_added = 0;
Z
Zhu Yi 已提交
399

400 401 402
	init_timer(&rs_sta->rate_scale_flush);
	rs_sta->rate_scale_flush.data = (unsigned long)rs_sta;
	rs_sta->rate_scale_flush.function = &iwl3945_bg_rate_scale_flush;
Z
Zhu Yi 已提交
403 404

	for (i = 0; i < IWL_RATE_COUNT; i++)
405
		iwl3945_clear_window(&rs_sta->win[i]);
Z
Zhu Yi 已提交
406 407 408

	IWL_DEBUG_RATE("leave\n");

409
	return rs_sta;
Z
Zhu Yi 已提交
410 411 412 413
}

static void rs_free_sta(void *priv, void *priv_sta)
{
414
	struct iwl3945_rs_sta *rs_sta = priv_sta;
Z
Zhu Yi 已提交
415 416

	IWL_DEBUG_RATE("enter\n");
417 418
	del_timer_sync(&rs_sta->rate_scale_flush);
	kfree(rs_sta);
Z
Zhu Yi 已提交
419 420 421
	IWL_DEBUG_RATE("leave\n");
}

422 423 424 425 426 427

/*
 * get ieee prev rate from rate scale table.
 * for A and B mode we need to overright prev
 * value
 */
C
Christoph Hellwig 已提交
428
static int rs_adjust_next_rate(struct iwl3945_priv *priv, int rate)
429
{
C
Christoph Hellwig 已提交
430
	int next_rate = iwl3945_get_prev_ieee_rate(rate);
431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448

	switch (priv->phymode) {
	case MODE_IEEE80211A:
		if (rate == IWL_RATE_12M_INDEX)
			next_rate = IWL_RATE_9M_INDEX;
		else if (rate == IWL_RATE_6M_INDEX)
			next_rate = IWL_RATE_6M_INDEX;
		break;
	case MODE_IEEE80211B:
		if (rate == IWL_RATE_11M_INDEX_TABLE)
			next_rate = IWL_RATE_5M_INDEX_TABLE;
		break;
	default:
		break;
	}

	return next_rate;
}
Z
Zhu Yi 已提交
449 450 451
/**
 * rs_tx_status - Update rate control values based on Tx results
 *
C
Christoph Hellwig 已提交
452
 * NOTE: Uses iwl3945_priv->retry_rate for the # of retries attempted by
Z
Zhu Yi 已提交
453 454 455 456 457 458 459 460 461 462 463 464
 * the hardware for each rate.
 */
static void rs_tx_status(void *priv_rate,
			 struct net_device *dev,
			 struct sk_buff *skb,
			 struct ieee80211_tx_status *tx_resp)
{
	u8 retries, current_count;
	int scale_rate_index, first_index, last_index;
	unsigned long flags;
	struct sta_info *sta;
	struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
C
Christoph Hellwig 已提交
465
	struct iwl3945_priv *priv = (struct iwl3945_priv *)priv_rate;
Z
Zhu Yi 已提交
466
	struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr);
467
	struct iwl3945_rs_sta *rs_sta;
Z
Zhu Yi 已提交
468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487

	IWL_DEBUG_RATE("enter\n");

	retries = tx_resp->retry_count;

	first_index = tx_resp->control.tx_rate;
	if ((first_index < 0) || (first_index >= IWL_RATE_COUNT)) {
		IWL_DEBUG_RATE("leave: Rate out of bounds: %0x for %d\n",
			       tx_resp->control.tx_rate, first_index);
		return;
	}

	sta = sta_info_get(local, hdr->addr1);
	if (!sta || !sta->rate_ctrl_priv) {
		if (sta)
			sta_info_put(sta);
		IWL_DEBUG_RATE("leave: No STA priv data to update!\n");
		return;
	}

488
	rs_sta = (void *)sta->rate_ctrl_priv;
Z
Zhu Yi 已提交
489

490
	rs_sta->tx_packets++;
Z
Zhu Yi 已提交
491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510

	scale_rate_index = first_index;
	last_index = first_index;

	/*
	 * Update the window for each rate.  We determine which rates
	 * were Tx'd based on the total number of retries vs. the number
	 * of retries configured for each rate -- currently set to the
	 * priv value 'retry_rate' vs. rate specific
	 *
	 * On exit from this while loop last_index indicates the rate
	 * at which the frame was finally transmitted (or failed if no
	 * ACK)
	 */
	while (retries > 0) {
		if (retries < priv->retry_rate) {
			current_count = retries;
			last_index = scale_rate_index;
		} else {
			current_count = priv->retry_rate;
511 512
			last_index = rs_adjust_next_rate(priv,
							 scale_rate_index);
Z
Zhu Yi 已提交
513 514 515 516
		}

		/* Update this rate accounting for as many retries
		 * as was used for it (per current_count) */
517 518
		iwl3945_collect_tx_data(rs_sta,
				    &rs_sta->win[scale_rate_index],
Z
Zhu Yi 已提交
519 520 521 522 523 524 525 526
				    0, current_count);
		IWL_DEBUG_RATE("Update rate %d for %d retries.\n",
			       scale_rate_index, current_count);

		retries -= current_count;

		if (retries)
			scale_rate_index =
527
			    rs_adjust_next_rate(priv, scale_rate_index);
Z
Zhu Yi 已提交
528 529
	}

530

Z
Zhu Yi 已提交
531 532 533 534 535
	/* Update the last index window with success/failure based on ACK */
	IWL_DEBUG_RATE("Update rate %d with %s.\n",
		       last_index,
		       (tx_resp->flags & IEEE80211_TX_STATUS_ACK) ?
		       "success" : "failure");
536 537
	iwl3945_collect_tx_data(rs_sta,
			    &rs_sta->win[last_index],
Z
Zhu Yi 已提交
538 539 540 541 542
			    tx_resp->flags & IEEE80211_TX_STATUS_ACK, 1);

	/* We updated the rate scale window -- if its been more than
	 * flush_time since the last run, schedule the flush
	 * again */
543
	spin_lock_irqsave(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
544

545 546 547
	if (!rs_sta->flush_pending &&
	    time_after(jiffies, rs_sta->last_partial_flush +
		       rs_sta->flush_time)) {
Z
Zhu Yi 已提交
548

549 550 551
		rs_sta->flush_pending = 1;
		mod_timer(&rs_sta->rate_scale_flush,
			  jiffies + rs_sta->flush_time);
Z
Zhu Yi 已提交
552 553
	}

554
	spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
555 556 557 558 559 560 561 562

	sta_info_put(sta);

	IWL_DEBUG_RATE("leave\n");

	return;
}

563 564
static u16 iwl3945_get_adjacent_rate(struct iwl3945_rs_sta *rs_sta,
				 u8 index, u16 rate_mask, int phymode)
Z
Zhu Yi 已提交
565 566 567 568
{
	u8 high = IWL_RATE_INVALID;
	u8 low = IWL_RATE_INVALID;

569
	/* 802.11A walks to the next literal adjacent rate in
Z
Zhu Yi 已提交
570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597
	 * the rate table */
	if (unlikely(phymode == MODE_IEEE80211A)) {
		int i;
		u32 mask;

		/* Find the previous rate that is in the rate mask */
		i = index - 1;
		for (mask = (1 << i); i >= 0; i--, mask >>= 1) {
			if (rate_mask & mask) {
				low = i;
				break;
			}
		}

		/* Find the next rate that is in the rate mask */
		i = index + 1;
		for (mask = (1 << i); i < IWL_RATE_COUNT; i++, mask <<= 1) {
			if (rate_mask & mask) {
				high = i;
				break;
			}
		}

		return (high << 8) | low;
	}

	low = index;
	while (low != IWL_RATE_INVALID) {
598
		if (rs_sta->tgg)
C
Christoph Hellwig 已提交
599
			low = iwl3945_rates[low].prev_rs_tgg;
Z
Zhu Yi 已提交
600
		else
C
Christoph Hellwig 已提交
601
			low = iwl3945_rates[low].prev_rs;
Z
Zhu Yi 已提交
602 603 604 605 606 607 608 609 610
		if (low == IWL_RATE_INVALID)
			break;
		if (rate_mask & (1 << low))
			break;
		IWL_DEBUG_RATE("Skipping masked lower rate: %d\n", low);
	}

	high = index;
	while (high != IWL_RATE_INVALID) {
611
		if (rs_sta->tgg)
C
Christoph Hellwig 已提交
612
			high = iwl3945_rates[high].next_rs_tgg;
Z
Zhu Yi 已提交
613
		else
C
Christoph Hellwig 已提交
614
			high = iwl3945_rates[high].next_rs;
Z
Zhu Yi 已提交
615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640
		if (high == IWL_RATE_INVALID)
			break;
		if (rate_mask & (1 << high))
			break;
		IWL_DEBUG_RATE("Skipping masked higher rate: %d\n", high);
	}

	return (high << 8) | low;
}

/**
 * rs_get_rate - find the rate for the requested packet
 *
 * Returns the ieee80211_rate structure allocated by the driver.
 *
 * The rate control algorithm has no internal mapping between hw_mode's
 * rate ordering and the rate ordering used by the rate control algorithm.
 *
 * The rate control algorithm uses a single table of rates that goes across
 * the entire A/B/G spectrum vs. being limited to just one particular
 * hw_mode.
 *
 * As such, we can't convert the index obtained below into the hw_mode's
 * rate table and must reference the driver allocated rate table
 *
 */
641 642 643
static void rs_get_rate(void *priv_rate, struct net_device *dev,
			struct ieee80211_hw_mode *mode, struct sk_buff *skb,
			struct rate_selection *sel)
Z
Zhu Yi 已提交
644 645 646 647 648
{
	u8 low = IWL_RATE_INVALID;
	u8 high = IWL_RATE_INVALID;
	u16 high_low;
	int index;
649
	struct iwl3945_rs_sta *rs_sta;
C
Christoph Hellwig 已提交
650
	struct iwl3945_rate_scale_data *window = NULL;
Z
Zhu Yi 已提交
651 652 653 654 655 656 657 658 659
	int current_tpt = IWL_INVALID_VALUE;
	int low_tpt = IWL_INVALID_VALUE;
	int high_tpt = IWL_INVALID_VALUE;
	u32 fail_count;
	s8 scale_action = 0;
	unsigned long flags;
	struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr);
	struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
	struct sta_info *sta;
660
	u16 fc, rate_mask;
C
Christoph Hellwig 已提交
661
	struct iwl3945_priv *priv = (struct iwl3945_priv *)priv_rate;
662
	DECLARE_MAC_BUF(mac);
Z
Zhu Yi 已提交
663 664 665 666

	IWL_DEBUG_RATE("enter\n");

	sta = sta_info_get(local, hdr->addr1);
667 668 669 670 671 672 673

	/* Send management frames and broadcast/multicast data using lowest
	 * rate. */
	fc = le16_to_cpu(hdr->frame_control);
	if ((fc & IEEE80211_FCTL_FTYPE) != IEEE80211_FTYPE_DATA ||
	    is_multicast_ether_addr(hdr->addr1) ||
	    !sta || !sta->rate_ctrl_priv) {
Z
Zhu Yi 已提交
674
		IWL_DEBUG_RATE("leave: No STA priv data to update!\n");
675
		sel->rate = rate_lowest(local, local->oper_hw_mode, sta);
Z
Zhu Yi 已提交
676 677
		if (sta)
			sta_info_put(sta);
678
		return;
Z
Zhu Yi 已提交
679 680 681
	}

	rate_mask = sta->supp_rates;
682 683 684 685
	index = min(sta->last_txrate & 0xffff, IWL_RATE_COUNT - 1);

	if (priv->phymode == (u8) MODE_IEEE80211A)
		rate_mask = rate_mask << IWL_FIRST_OFDM_RATE;
Z
Zhu Yi 已提交
686

687
	rs_sta = (void *)sta->rate_ctrl_priv;
Z
Zhu Yi 已提交
688 689

	if ((priv->iw_mode == IEEE80211_IF_TYPE_IBSS) &&
690
	    !rs_sta->ibss_sta_added) {
C
Christoph Hellwig 已提交
691
		u8 sta_id = iwl3945_hw_find_station(priv, hdr->addr1);
Z
Zhu Yi 已提交
692 693

		if (sta_id == IWL_INVALID_STATION) {
694 695
			IWL_DEBUG_RATE("LQ: ADD station %s\n",
				       print_mac(mac, hdr->addr1));
C
Christoph Hellwig 已提交
696
			sta_id = iwl3945_add_station(priv,
Z
Zhu Yi 已提交
697 698 699
				    hdr->addr1, 0, CMD_ASYNC);
		}
		if (sta_id != IWL_INVALID_STATION)
700
			rs_sta->ibss_sta_added = 1;
Z
Zhu Yi 已提交
701 702
	}

703
	spin_lock_irqsave(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
704

705 706 707
	if (rs_sta->start_rate != IWL_RATE_INVALID) {
		index = rs_sta->start_rate;
		rs_sta->start_rate = IWL_RATE_INVALID;
Z
Zhu Yi 已提交
708 709
	}

710
	window = &(rs_sta->win[index]);
Z
Zhu Yi 已提交
711 712 713 714 715 716

	fail_count = window->counter - window->success_counter;

	if (((fail_count <= IWL_RATE_MIN_FAILURE_TH) &&
	     (window->success_counter < IWL_RATE_MIN_SUCCESS_TH))) {
		window->average_tpt = IWL_INVALID_VALUE;
717
		spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
718 719 720 721 722 723 724

		IWL_DEBUG_RATE("Invalid average_tpt on rate %d: "
			       "counter: %d, success_counter: %d, "
			       "expected_tpt is %sNULL\n",
			       index,
			       window->counter,
			       window->success_counter,
725
			       rs_sta->expected_tpt ? "not " : "");
Z
Zhu Yi 已提交
726 727 728 729 730
		goto out;

	}

	window->average_tpt = ((window->success_ratio *
731
				rs_sta->expected_tpt[index] + 64) / 128);
Z
Zhu Yi 已提交
732 733
	current_tpt = window->average_tpt;

734
	high_low = iwl3945_get_adjacent_rate(rs_sta, index, rate_mask,
Z
Zhu Yi 已提交
735 736 737 738 739
					 local->hw.conf.phymode);
	low = high_low & 0xff;
	high = (high_low >> 8) & 0xff;

	if (low != IWL_RATE_INVALID)
740
		low_tpt = rs_sta->win[low].average_tpt;
Z
Zhu Yi 已提交
741 742

	if (high != IWL_RATE_INVALID)
743
		high_tpt = rs_sta->win[high].average_tpt;
Z
Zhu Yi 已提交
744

745
	spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813

	scale_action = 1;

	if ((window->success_ratio < IWL_RATE_DECREASE_TH) || !current_tpt) {
		IWL_DEBUG_RATE("decrease rate because of low success_ratio\n");
		scale_action = -1;
	} else if ((low_tpt == IWL_INVALID_VALUE) &&
		   (high_tpt == IWL_INVALID_VALUE))
		scale_action = 1;
	else if ((low_tpt != IWL_INVALID_VALUE) &&
		   (high_tpt != IWL_INVALID_VALUE)
		   && (low_tpt < current_tpt)
		   && (high_tpt < current_tpt)) {
		IWL_DEBUG_RATE("No action -- low [%d] & high [%d] < "
			       "current_tpt [%d]\n",
			       low_tpt, high_tpt, current_tpt);
		scale_action = 0;
	} else {
		if (high_tpt != IWL_INVALID_VALUE) {
			if (high_tpt > current_tpt)
				scale_action = 1;
			else {
				IWL_DEBUG_RATE
				    ("decrease rate because of high tpt\n");
				scale_action = -1;
			}
		} else if (low_tpt != IWL_INVALID_VALUE) {
			if (low_tpt > current_tpt) {
				IWL_DEBUG_RATE
				    ("decrease rate because of low tpt\n");
				scale_action = -1;
			} else
				scale_action = 1;
		}
	}

	if ((window->success_ratio > IWL_RATE_HIGH_TH) ||
	    (current_tpt > window->average_tpt)) {
		IWL_DEBUG_RATE("No action -- success_ratio [%d] > HIGH_TH or "
			       "current_tpt [%d] > average_tpt [%d]\n",
			       window->success_ratio,
			       current_tpt, window->average_tpt);
		scale_action = 0;
	}

	switch (scale_action) {
	case -1:
		if (low != IWL_RATE_INVALID)
			index = low;
		break;

	case 1:
		if (high != IWL_RATE_INVALID)
			index = high;

		break;

	case 0:
	default:
		break;
	}

	IWL_DEBUG_RATE("Selected %d (action %d) - low %d high %d\n",
		       index, scale_action, low, high);

 out:

	sta->last_txrate = index;
814 815 816 817 818
	if (priv->phymode == (u8) MODE_IEEE80211A)
		sta->txrate = sta->last_txrate - IWL_FIRST_OFDM_RATE;
	else
		sta->txrate = sta->last_txrate;

Z
Zhu Yi 已提交
819 820 821 822
	sta_info_put(sta);

	IWL_DEBUG_RATE("leave: %d\n", index);

823
	sel->rate = &priv->ieee_rates[index];
Z
Zhu Yi 已提交
824 825 826 827 828 829 830 831 832 833 834 835 836 837 838
}

static struct rate_control_ops rs_ops = {
	.module = NULL,
	.name = RS_NAME,
	.tx_status = rs_tx_status,
	.get_rate = rs_get_rate,
	.rate_init = rs_rate_init,
	.clear = rs_clear,
	.alloc = rs_alloc,
	.free = rs_free,
	.alloc_sta = rs_alloc_sta,
	.free_sta = rs_free_sta,
};

C
Christoph Hellwig 已提交
839
int iwl3945_fill_rs_info(struct ieee80211_hw *hw, char *buf, u8 sta_id)
Z
Zhu Yi 已提交
840 841
{
	struct ieee80211_local *local = hw_to_local(hw);
C
Christoph Hellwig 已提交
842
	struct iwl3945_priv *priv = hw->priv;
843
	struct iwl3945_rs_sta *rs_sta;
Z
Zhu Yi 已提交
844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860
	struct sta_info *sta;
	unsigned long flags;
	int count = 0, i;
	u32 samples = 0, success = 0, good = 0;
	unsigned long now = jiffies;
	u32 max_time = 0;

	sta = sta_info_get(local, priv->stations[sta_id].sta.sta.addr);
	if (!sta || !sta->rate_ctrl_priv) {
		if (sta) {
			sta_info_put(sta);
			IWL_DEBUG_RATE("leave - no private rate data!\n");
		} else
			IWL_DEBUG_RATE("leave - no station!\n");
		return sprintf(buf, "station %d not found\n", sta_id);
	}

861 862
	rs_sta = (void *)sta->rate_ctrl_priv;
	spin_lock_irqsave(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
863 864 865 866 867 868
	i = IWL_RATE_54M_INDEX;
	while (1) {
		u64 mask;
		int j;

		count +=
C
Christoph Hellwig 已提交
869
		    sprintf(&buf[count], " %2dMbs: ", iwl3945_rates[i].ieee / 2);
Z
Zhu Yi 已提交
870 871 872 873

		mask = (1ULL << (IWL_RATE_MAX_WINDOW - 1));
		for (j = 0; j < IWL_RATE_MAX_WINDOW; j++, mask >>= 1)
			buf[count++] =
874
			    (rs_sta->win[i].data & mask) ? '1' : '0';
Z
Zhu Yi 已提交
875

876 877 878 879
		samples += rs_sta->win[i].counter;
		good += rs_sta->win[i].success_counter;
		success += rs_sta->win[i].success_counter *
						iwl3945_rates[i].ieee;
Z
Zhu Yi 已提交
880

881
		if (rs_sta->win[i].stamp) {
Z
Zhu Yi 已提交
882
			int delta =
883
			    jiffies_to_msecs(now - rs_sta->win[i].stamp);
Z
Zhu Yi 已提交
884 885 886 887 888 889 890 891

			if (delta > max_time)
				max_time = delta;

			count += sprintf(&buf[count], "%5dms\n", delta);
		} else
			buf[count++] = '\n';

C
Christoph Hellwig 已提交
892
		j = iwl3945_get_prev_ieee_rate(i);
Z
Zhu Yi 已提交
893 894 895 896
		if (j == i)
			break;
		i = j;
	}
897
	spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
898 899 900 901 902
	sta_info_put(sta);

	/* Display the average rate of all samples taken.
	 *
	 * NOTE:  We multiple # of samples by 2 since the IEEE measurement
C
Christoph Hellwig 已提交
903
	 * added from iwl3945_rates is actually 2X the rate */
Z
Zhu Yi 已提交
904 905 906 907 908 909 910 911 912 913 914 915 916
	if (samples)
		count += sprintf(
			&buf[count],
			"\nAverage rate is %3d.%02dMbs over last %4dms\n"
			"%3d%% success (%d good packets over %d tries)\n",
			success / (2 * samples), (success * 5 / samples) % 10,
			max_time, good * 100 / samples, good, samples);
	else
		count += sprintf(&buf[count], "\nAverage rate: 0Mbs\n");

	return count;
}

C
Christoph Hellwig 已提交
917
void iwl3945_rate_scale_init(struct ieee80211_hw *hw, s32 sta_id)
Z
Zhu Yi 已提交
918
{
C
Christoph Hellwig 已提交
919
	struct iwl3945_priv *priv = hw->priv;
Z
Zhu Yi 已提交
920 921 922
	s32 rssi = 0;
	unsigned long flags;
	struct ieee80211_local *local = hw_to_local(hw);
923
	struct iwl3945_rs_sta *rs_sta;
Z
Zhu Yi 已提交
924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942
	struct sta_info *sta;

	IWL_DEBUG_RATE("enter\n");

	if (!local->rate_ctrl->ops->name ||
	    strcmp(local->rate_ctrl->ops->name, RS_NAME)) {
		IWL_WARNING("iwl-3945-rs not selected as rate control algo!\n");
		IWL_DEBUG_RATE("leave - mac80211 picked the wrong RC algo.\n");
		return;
	}

	sta = sta_info_get(local, priv->stations[sta_id].sta.sta.addr);
	if (!sta || !sta->rate_ctrl_priv) {
		if (sta)
			sta_info_put(sta);
		IWL_DEBUG_RATE("leave - no private rate data!\n");
		return;
	}

943
	rs_sta = (void *)sta->rate_ctrl_priv;
Z
Zhu Yi 已提交
944

945
	spin_lock_irqsave(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
946

947
	rs_sta->tgg = 0;
Z
Zhu Yi 已提交
948 949 950
	switch (priv->phymode) {
	case MODE_IEEE80211G:
		if (priv->active_rxon.flags & RXON_FLG_TGG_PROTECT_MSK) {
951 952
			rs_sta->tgg = 1;
			rs_sta->expected_tpt = iwl3945_expected_tpt_g_prot;
Z
Zhu Yi 已提交
953
		} else
954
			rs_sta->expected_tpt = iwl3945_expected_tpt_g;
Z
Zhu Yi 已提交
955 956 957
		break;

	case MODE_IEEE80211A:
958
		rs_sta->expected_tpt = iwl3945_expected_tpt_a;
Z
Zhu Yi 已提交
959 960 961 962 963
		break;

	default:
		IWL_WARNING("Invalid phymode.  Defaulting to 802.11b\n");
	case MODE_IEEE80211B:
964
		rs_sta->expected_tpt = iwl3945_expected_tpt_b;
Z
Zhu Yi 已提交
965 966 967 968
		break;
	}

	sta_info_put(sta);
969
	spin_unlock_irqrestore(&rs_sta->lock, flags);
Z
Zhu Yi 已提交
970 971 972 973 974 975 976

	rssi = priv->last_rx_rssi;
	if (rssi == 0)
		rssi = IWL_MIN_RSSI_VAL;

	IWL_DEBUG(IWL_DL_INFO | IWL_DL_RATE, "Network RSSI: %d\n", rssi);

977 978
	rs_sta->start_rate =
			iwl3945_get_rate_index_by_rssi(rssi, priv->phymode);
Z
Zhu Yi 已提交
979 980

	IWL_DEBUG_RATE("leave: rssi %d assign rate index: "
981 982
		       "%d (plcp 0x%x)\n", rssi, rs_sta->start_rate,
		       iwl3945_rates[rs_sta->start_rate].plcp);
Z
Zhu Yi 已提交
983 984
}

C
Christoph Hellwig 已提交
985
void iwl3945_rate_control_register(struct ieee80211_hw *hw)
Z
Zhu Yi 已提交
986 987 988 989
{
	ieee80211_rate_control_register(&rs_ops);
}

C
Christoph Hellwig 已提交
990
void iwl3945_rate_control_unregister(struct ieee80211_hw *hw)
Z
Zhu Yi 已提交
991 992 993 994 995
{
	ieee80211_rate_control_unregister(&rs_ops);
}