2008-08-04 00:16:41 -07:00
/*
* Copyright ( c ) 2004 Video54 Technologies , Inc .
* Copyright ( c ) 2004 - 2008 Atheros Communications , Inc .
*
* Permission to use , copy , modify , and / or distribute this software for any
* purpose with or without fee is hereby granted , provided that the above
* copyright notice and this permission notice appear in all copies .
*
* THE SOFTWARE IS PROVIDED " AS IS " AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS . IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL , DIRECT , INDIRECT , OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE , DATA OR PROFITS , WHETHER IN AN
* ACTION OF CONTRACT , NEGLIGENCE OR OTHER TORTIOUS ACTION , ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
*/
/*
* Atheros rate control algorithm
*/
# include "core.h"
2008-09-18 18:14:18 +02:00
/* FIXME: remove this include! */
2008-08-04 00:16:41 -07:00
# include "../net/mac80211/rate.h"
static struct ath_rate_table ar5416_11na_ratetable = {
42 ,
{
{ TRUE , TRUE , WLAN_PHY_OFDM , 6000 , /* 6 Mb */
5400 , 0x0b , 0x00 , 12 ,
0 , 2 , 1 , 0 , 0 , 0 , 0 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 9000 , /* 9 Mb */
7800 , 0x0f , 0x00 , 18 ,
0 , 3 , 1 , 1 , 1 , 1 , 1 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 12000 , /* 12 Mb */
10000 , 0x0a , 0x00 , 24 ,
2 , 4 , 2 , 2 , 2 , 2 , 2 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 18000 , /* 18 Mb */
13900 , 0x0e , 0x00 , 36 ,
2 , 6 , 2 , 3 , 3 , 3 , 3 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 24000 , /* 24 Mb */
17300 , 0x09 , 0x00 , 48 ,
4 , 10 , 3 , 4 , 4 , 4 , 4 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 36000 , /* 36 Mb */
23000 , 0x0d , 0x00 , 72 ,
4 , 14 , 3 , 5 , 5 , 5 , 5 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 48000 , /* 48 Mb */
27400 , 0x08 , 0x00 , 96 ,
4 , 20 , 3 , 6 , 6 , 6 , 6 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 54000 , /* 54 Mb */
29300 , 0x0c , 0x00 , 108 ,
4 , 23 , 3 , 7 , 7 , 7 , 7 , 0 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 6500 , /* 6.5 Mb */
6400 , 0x80 , 0x00 , 0 ,
0 , 2 , 3 , 8 , 24 , 8 , 24 , 3216 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 13000 , /* 13 Mb */
12700 , 0x81 , 0x00 , 1 ,
2 , 4 , 3 , 9 , 25 , 9 , 25 , 6434 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 19500 , /* 19.5 Mb */
18800 , 0x82 , 0x00 , 2 ,
2 , 6 , 3 , 10 , 26 , 10 , 26 , 9650 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 26000 , /* 26 Mb */
25000 , 0x83 , 0x00 , 3 ,
4 , 10 , 3 , 11 , 27 , 11 , 27 , 12868 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 39000 , /* 39 Mb */
36700 , 0x84 , 0x00 , 4 ,
4 , 14 , 3 , 12 , 28 , 12 , 28 , 19304 } ,
{ FALSE , TRUE_20 , WLAN_PHY_HT_20_SS , 52000 , /* 52 Mb */
48100 , 0x85 , 0x00 , 5 ,
4 , 20 , 3 , 13 , 29 , 13 , 29 , 25740 } ,
{ FALSE , TRUE_20 , WLAN_PHY_HT_20_SS , 58500 , /* 58.5 Mb */
53500 , 0x86 , 0x00 , 6 ,
4 , 23 , 3 , 14 , 30 , 14 , 30 , 28956 } ,
{ FALSE , TRUE_20 , WLAN_PHY_HT_20_SS , 65000 , /* 65 Mb */
59000 , 0x87 , 0x00 , 7 ,
4 , 25 , 3 , 15 , 31 , 15 , 32 , 32180 } ,
{ FALSE , FALSE , WLAN_PHY_HT_20_DS , 13000 , /* 13 Mb */
12700 , 0x88 , 0x00 ,
8 , 0 , 2 , 3 , 16 , 33 , 16 , 33 , 6430 } ,
{ FALSE , FALSE , WLAN_PHY_HT_20_DS , 26000 , /* 26 Mb */
24800 , 0x89 , 0x00 , 9 ,
2 , 4 , 3 , 17 , 34 , 17 , 34 , 12860 } ,
{ FALSE , FALSE , WLAN_PHY_HT_20_DS , 39000 , /* 39 Mb */
36600 , 0x8a , 0x00 , 10 ,
2 , 6 , 3 , 18 , 35 , 18 , 35 , 19300 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 52000 , /* 52 Mb */
48100 , 0x8b , 0x00 , 11 ,
4 , 10 , 3 , 19 , 36 , 19 , 36 , 25736 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 78000 , /* 78 Mb */
69500 , 0x8c , 0x00 , 12 ,
4 , 14 , 3 , 20 , 37 , 20 , 37 , 38600 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 104000 , /* 104 Mb */
89500 , 0x8d , 0x00 , 13 ,
4 , 20 , 3 , 21 , 38 , 21 , 38 , 51472 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 117000 , /* 117 Mb */
98900 , 0x8e , 0x00 , 14 ,
4 , 23 , 3 , 22 , 39 , 22 , 39 , 57890 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 130000 , /* 130 Mb */
108300 , 0x8f , 0x00 , 15 ,
4 , 25 , 3 , 23 , 40 , 23 , 41 , 64320 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 13500 , /* 13.5 Mb */
13200 , 0x80 , 0x00 , 0 ,
0 , 2 , 3 , 8 , 24 , 24 , 24 , 6684 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 27500 , /* 27.0 Mb */
25900 , 0x81 , 0x00 , 1 ,
2 , 4 , 3 , 9 , 25 , 25 , 25 , 13368 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 40500 , /* 40.5 Mb */
38600 , 0x82 , 0x00 , 2 ,
2 , 6 , 3 , 10 , 26 , 26 , 26 , 20052 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 54000 , /* 54 Mb */
49800 , 0x83 , 0x00 , 3 ,
4 , 10 , 3 , 11 , 27 , 27 , 27 , 26738 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 81500 , /* 81 Mb */
72200 , 0x84 , 0x00 , 4 ,
4 , 14 , 3 , 12 , 28 , 28 , 28 , 40104 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS , 108000 , /* 108 Mb */
92900 , 0x85 , 0x00 , 5 ,
4 , 20 , 3 , 13 , 29 , 29 , 29 , 53476 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS , 121500 , /* 121.5 Mb */
102700 , 0x86 , 0x00 , 6 ,
4 , 23 , 3 , 14 , 30 , 30 , 30 , 60156 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS , 135000 , /* 135 Mb */
112000 , 0x87 , 0x00 , 7 ,
4 , 25 , 3 , 15 , 31 , 32 , 32 , 66840 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS_HGI , 150000 , /* 150 Mb */
122000 , 0x87 , 0x00 , 7 ,
4 , 25 , 3 , 15 , 31 , 32 , 32 , 74200 } ,
{ FALSE , FALSE , WLAN_PHY_HT_40_DS , 27000 , /* 27 Mb */
25800 , 0x88 , 0x00 , 8 ,
0 , 2 , 3 , 16 , 33 , 33 , 33 , 13360 } ,
{ FALSE , FALSE , WLAN_PHY_HT_40_DS , 54000 , /* 54 Mb */
49800 , 0x89 , 0x00 , 9 ,
2 , 4 , 3 , 17 , 34 , 34 , 34 , 26720 } ,
{ FALSE , FALSE , WLAN_PHY_HT_40_DS , 81000 , /* 81 Mb */
71900 , 0x8a , 0x00 , 10 ,
2 , 6 , 3 , 18 , 35 , 35 , 35 , 40080 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 108000 , /* 108 Mb */
92500 , 0x8b , 0x00 , 11 ,
4 , 10 , 3 , 19 , 36 , 36 , 36 , 53440 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 162000 , /* 162 Mb */
130300 , 0x8c , 0x00 , 12 ,
4 , 14 , 3 , 20 , 37 , 37 , 37 , 80160 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 216000 , /* 216 Mb */
162800 , 0x8d , 0x00 , 13 ,
4 , 20 , 3 , 21 , 38 , 38 , 38 , 106880 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 243000 , /* 243 Mb */
178200 , 0x8e , 0x00 , 14 ,
4 , 23 , 3 , 22 , 39 , 39 , 39 , 120240 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 270000 , /* 270 Mb */
192100 , 0x8f , 0x00 , 15 ,
4 , 25 , 3 , 23 , 40 , 41 , 41 , 133600 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS_HGI , 300000 , /* 300 Mb */
207000 , 0x8f , 0x00 , 15 ,
4 , 25 , 3 , 23 , 40 , 41 , 41 , 148400 } ,
} ,
50 , /* probe interval */
50 , /* rssi reduce interval */
WLAN_RC_HT_FLAG , /* Phy rates allowed initially */
} ;
/* TRUE_ALL - valid for 20/40/Legacy,
* TRUE - Legacy only ,
* TRUE_20 - HT 20 only ,
* TRUE_40 - HT 40 only */
/* 4ms frame limit not used for NG mode. The values filled
* for HT are the 64 K max aggregate limit */
static struct ath_rate_table ar5416_11ng_ratetable = {
46 ,
{
{ TRUE_ALL , TRUE_ALL , WLAN_PHY_CCK , 1000 , /* 1 Mb */
900 , 0x1b , 0x00 , 2 ,
0 , 0 , 1 , 0 , 0 , 0 , 0 , 0 } ,
{ TRUE_ALL , TRUE_ALL , WLAN_PHY_CCK , 2000 , /* 2 Mb */
1900 , 0x1a , 0x04 , 4 ,
1 , 1 , 1 , 1 , 1 , 1 , 1 , 0 } ,
{ TRUE_ALL , TRUE_ALL , WLAN_PHY_CCK , 5500 , /* 5.5 Mb */
4900 , 0x19 , 0x04 , 11 ,
2 , 2 , 2 , 2 , 2 , 2 , 2 , 0 } ,
{ TRUE_ALL , TRUE_ALL , WLAN_PHY_CCK , 11000 , /* 11 Mb */
8100 , 0x18 , 0x04 , 22 ,
3 , 3 , 2 , 3 , 3 , 3 , 3 , 0 } ,
{ FALSE , FALSE , WLAN_PHY_OFDM , 6000 , /* 6 Mb */
5400 , 0x0b , 0x00 , 12 ,
4 , 2 , 1 , 4 , 4 , 4 , 4 , 0 } ,
{ FALSE , FALSE , WLAN_PHY_OFDM , 9000 , /* 9 Mb */
7800 , 0x0f , 0x00 , 18 ,
4 , 3 , 1 , 5 , 5 , 5 , 5 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 12000 , /* 12 Mb */
10100 , 0x0a , 0x00 , 24 ,
6 , 4 , 1 , 6 , 6 , 6 , 6 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 18000 , /* 18 Mb */
14100 , 0x0e , 0x00 , 36 ,
6 , 6 , 2 , 7 , 7 , 7 , 7 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 24000 , /* 24 Mb */
17700 , 0x09 , 0x00 , 48 ,
8 , 10 , 3 , 8 , 8 , 8 , 8 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 36000 , /* 36 Mb */
23700 , 0x0d , 0x00 , 72 ,
8 , 14 , 3 , 9 , 9 , 9 , 9 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 48000 , /* 48 Mb */
27400 , 0x08 , 0x00 , 96 ,
8 , 20 , 3 , 10 , 10 , 10 , 10 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 54000 , /* 54 Mb */
30900 , 0x0c , 0x00 , 108 ,
8 , 23 , 3 , 11 , 11 , 11 , 11 , 0 } ,
{ FALSE , FALSE , WLAN_PHY_HT_20_SS , 6500 , /* 6.5 Mb */
6400 , 0x80 , 0x00 , 0 ,
4 , 2 , 3 , 12 , 28 , 12 , 28 , 3216 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 13000 , /* 13 Mb */
12700 , 0x81 , 0x00 , 1 ,
6 , 4 , 3 , 13 , 29 , 13 , 29 , 6434 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 19500 , /* 19.5 Mb */
18800 , 0x82 , 0x00 , 2 ,
6 , 6 , 3 , 14 , 30 , 14 , 30 , 9650 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 26000 , /* 26 Mb */
25000 , 0x83 , 0x00 , 3 ,
8 , 10 , 3 , 15 , 31 , 15 , 31 , 12868 } ,
{ TRUE_20 , TRUE_20 , WLAN_PHY_HT_20_SS , 39000 , /* 39 Mb */
36700 , 0x84 , 0x00 , 4 ,
8 , 14 , 3 , 16 , 32 , 16 , 32 , 19304 } ,
{ FALSE , TRUE_20 , WLAN_PHY_HT_20_SS , 52000 , /* 52 Mb */
48100 , 0x85 , 0x00 , 5 ,
8 , 20 , 3 , 17 , 33 , 17 , 33 , 25740 } ,
{ FALSE , TRUE_20 , WLAN_PHY_HT_20_SS , 58500 , /* 58.5 Mb */
53500 , 0x86 , 0x00 , 6 ,
8 , 23 , 3 , 18 , 34 , 18 , 34 , 28956 } ,
{ FALSE , TRUE_20 , WLAN_PHY_HT_20_SS , 65000 , /* 65 Mb */
59000 , 0x87 , 0x00 , 7 ,
8 , 25 , 3 , 19 , 35 , 19 , 36 , 32180 } ,
{ FALSE , FALSE , WLAN_PHY_HT_20_DS , 13000 , /* 13 Mb */
12700 , 0x88 , 0x00 , 8 ,
4 , 2 , 3 , 20 , 37 , 20 , 37 , 6430 } ,
{ FALSE , FALSE , WLAN_PHY_HT_20_DS , 26000 , /* 26 Mb */
24800 , 0x89 , 0x00 , 9 ,
6 , 4 , 3 , 21 , 38 , 21 , 38 , 12860 } ,
{ FALSE , FALSE , WLAN_PHY_HT_20_DS , 39000 , /* 39 Mb */
36600 , 0x8a , 0x00 , 10 ,
6 , 6 , 3 , 22 , 39 , 22 , 39 , 19300 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 52000 , /* 52 Mb */
48100 , 0x8b , 0x00 , 11 ,
8 , 10 , 3 , 23 , 40 , 23 , 40 , 25736 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 78000 , /* 78 Mb */
69500 , 0x8c , 0x00 , 12 ,
8 , 14 , 3 , 24 , 41 , 24 , 41 , 38600 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 104000 , /* 104 Mb */
89500 , 0x8d , 0x00 , 13 ,
8 , 20 , 3 , 25 , 42 , 25 , 42 , 51472 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 117000 , /* 117 Mb */
98900 , 0x8e , 0x00 , 14 ,
8 , 23 , 3 , 26 , 43 , 26 , 44 , 57890 } ,
{ TRUE_20 , FALSE , WLAN_PHY_HT_20_DS , 130000 , /* 130 Mb */
108300 , 0x8f , 0x00 , 15 ,
8 , 25 , 3 , 27 , 44 , 27 , 45 , 64320 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 13500 , /* 13.5 Mb */
13200 , 0x80 , 0x00 , 0 ,
8 , 2 , 3 , 12 , 28 , 28 , 28 , 6684 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 27500 , /* 27.0 Mb */
25900 , 0x81 , 0x00 , 1 ,
8 , 4 , 3 , 13 , 29 , 29 , 29 , 13368 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 40500 , /* 40.5 Mb */
38600 , 0x82 , 0x00 , 2 ,
8 , 6 , 3 , 14 , 30 , 30 , 30 , 20052 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 54000 , /* 54 Mb */
49800 , 0x83 , 0x00 , 3 ,
8 , 10 , 3 , 15 , 31 , 31 , 31 , 26738 } ,
{ TRUE_40 , TRUE_40 , WLAN_PHY_HT_40_SS , 81500 , /* 81 Mb */
72200 , 0x84 , 0x00 , 4 ,
8 , 14 , 3 , 16 , 32 , 32 , 32 , 40104 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS , 108000 , /* 108 Mb */
92900 , 0x85 , 0x00 , 5 ,
8 , 20 , 3 , 17 , 33 , 33 , 33 , 53476 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS , 121500 , /* 121.5 Mb */
102700 , 0x86 , 0x00 , 6 ,
8 , 23 , 3 , 18 , 34 , 34 , 34 , 60156 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS , 135000 , /* 135 Mb */
112000 , 0x87 , 0x00 , 7 ,
8 , 23 , 3 , 19 , 35 , 36 , 36 , 66840 } ,
{ FALSE , TRUE_40 , WLAN_PHY_HT_40_SS_HGI , 150000 , /* 150 Mb */
122000 , 0x87 , 0x00 , 7 ,
8 , 25 , 3 , 19 , 35 , 36 , 36 , 74200 } ,
{ FALSE , FALSE , WLAN_PHY_HT_40_DS , 27000 , /* 27 Mb */
25800 , 0x88 , 0x00 , 8 ,
8 , 2 , 3 , 20 , 37 , 37 , 37 , 13360 } ,
{ FALSE , FALSE , WLAN_PHY_HT_40_DS , 54000 , /* 54 Mb */
49800 , 0x89 , 0x00 , 9 ,
8 , 4 , 3 , 21 , 38 , 38 , 38 , 26720 } ,
{ FALSE , FALSE , WLAN_PHY_HT_40_DS , 81000 , /* 81 Mb */
71900 , 0x8a , 0x00 , 10 ,
8 , 6 , 3 , 22 , 39 , 39 , 39 , 40080 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 108000 , /* 108 Mb */
92500 , 0x8b , 0x00 , 11 ,
8 , 10 , 3 , 23 , 40 , 40 , 40 , 53440 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 162000 , /* 162 Mb */
130300 , 0x8c , 0x00 , 12 ,
8 , 14 , 3 , 24 , 41 , 41 , 41 , 80160 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 216000 , /* 216 Mb */
162800 , 0x8d , 0x00 , 13 ,
8 , 20 , 3 , 25 , 42 , 42 , 42 , 106880 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 243000 , /* 243 Mb */
178200 , 0x8e , 0x00 , 14 ,
8 , 23 , 3 , 26 , 43 , 43 , 43 , 120240 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS , 270000 , /* 270 Mb */
192100 , 0x8f , 0x00 , 15 ,
8 , 23 , 3 , 27 , 44 , 45 , 45 , 133600 } ,
{ TRUE_40 , FALSE , WLAN_PHY_HT_40_DS_HGI , 300000 , /* 300 Mb */
207000 , 0x8f , 0x00 , 15 ,
8 , 25 , 3 , 27 , 44 , 45 , 45 , 148400 } ,
} ,
50 , /* probe interval */
50 , /* rssi reduce interval */
WLAN_RC_HT_FLAG , /* Phy rates allowed initially */
} ;
static struct ath_rate_table ar5416_11a_ratetable = {
8 ,
{
{ TRUE , TRUE , WLAN_PHY_OFDM , 6000 , /* 6 Mb */
5400 , 0x0b , 0x00 , ( 0x80 | 12 ) ,
0 , 2 , 1 , 0 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 9000 , /* 9 Mb */
7800 , 0x0f , 0x00 , 18 ,
0 , 3 , 1 , 1 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 12000 , /* 12 Mb */
10000 , 0x0a , 0x00 , ( 0x80 | 24 ) ,
2 , 4 , 2 , 2 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 18000 , /* 18 Mb */
13900 , 0x0e , 0x00 , 36 ,
2 , 6 , 2 , 3 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 24000 , /* 24 Mb */
17300 , 0x09 , 0x00 , ( 0x80 | 48 ) ,
4 , 10 , 3 , 4 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 36000 , /* 36 Mb */
23000 , 0x0d , 0x00 , 72 ,
4 , 14 , 3 , 5 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 48000 , /* 48 Mb */
27400 , 0x08 , 0x00 , 96 ,
4 , 19 , 3 , 6 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 54000 , /* 54 Mb */
29300 , 0x0c , 0x00 , 108 ,
4 , 23 , 3 , 7 , 0 } ,
} ,
50 , /* probe interval */
50 , /* rssi reduce interval */
0 , /* Phy rates allowed initially */
} ;
static struct ath_rate_table ar5416_11g_ratetable = {
12 ,
{
{ TRUE , TRUE , WLAN_PHY_CCK , 1000 , /* 1 Mb */
900 , 0x1b , 0x00 , 2 ,
0 , 0 , 1 , 0 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_CCK , 2000 , /* 2 Mb */
1900 , 0x1a , 0x04 , 4 ,
1 , 1 , 1 , 1 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_CCK , 5500 , /* 5.5 Mb */
4900 , 0x19 , 0x04 , 11 ,
2 , 2 , 2 , 2 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_CCK , 11000 , /* 11 Mb */
8100 , 0x18 , 0x04 , 22 ,
3 , 3 , 2 , 3 , 0 } ,
{ FALSE , FALSE , WLAN_PHY_OFDM , 6000 , /* 6 Mb */
5400 , 0x0b , 0x00 , 12 ,
4 , 2 , 1 , 4 , 0 } ,
{ FALSE , FALSE , WLAN_PHY_OFDM , 9000 , /* 9 Mb */
7800 , 0x0f , 0x00 , 18 ,
4 , 3 , 1 , 5 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 12000 , /* 12 Mb */
10000 , 0x0a , 0x00 , 24 ,
6 , 4 , 1 , 6 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 18000 , /* 18 Mb */
13900 , 0x0e , 0x00 , 36 ,
6 , 6 , 2 , 7 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 24000 , /* 24 Mb */
17300 , 0x09 , 0x00 , 48 ,
8 , 10 , 3 , 8 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 36000 , /* 36 Mb */
23000 , 0x0d , 0x00 , 72 ,
8 , 14 , 3 , 9 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 48000 , /* 48 Mb */
27400 , 0x08 , 0x00 , 96 ,
8 , 19 , 3 , 10 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_OFDM , 54000 , /* 54 Mb */
29300 , 0x0c , 0x00 , 108 ,
8 , 23 , 3 , 11 , 0 } ,
} ,
50 , /* probe interval */
50 , /* rssi reduce interval */
0 , /* Phy rates allowed initially */
} ;
static struct ath_rate_table ar5416_11b_ratetable = {
4 ,
{
{ TRUE , TRUE , WLAN_PHY_CCK , 1000 , /* 1 Mb */
900 , 0x1b , 0x00 , ( 0x80 | 2 ) ,
0 , 0 , 1 , 0 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_CCK , 2000 , /* 2 Mb */
1800 , 0x1a , 0x04 , ( 0x80 | 4 ) ,
1 , 1 , 1 , 1 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_CCK , 5500 , /* 5.5 Mb */
4300 , 0x19 , 0x04 , ( 0x80 | 11 ) ,
1 , 2 , 2 , 2 , 0 } ,
{ TRUE , TRUE , WLAN_PHY_CCK , 11000 , /* 11 Mb */
7100 , 0x18 , 0x04 , ( 0x80 | 22 ) ,
1 , 4 , 100 , 3 , 0 } ,
} ,
100 , /* probe interval */
100 , /* rssi reduce interval */
0 , /* Phy rates allowed initially */
} ;
/*
* Return the median of three numbers
*/
static inline int8_t median ( int8_t a , int8_t b , int8_t c )
{
if ( a > = b ) {
if ( b > = c )
return b ;
else if ( a > c )
return c ;
else
return a ;
} else {
if ( a > = c )
return a ;
else if ( b > = c )
return c ;
else
return b ;
}
}
2008-11-18 09:07:06 +05:30
static void ath_rc_sort_validrates ( struct ath_rate_table * rate_table ,
2008-11-18 09:03:12 +05:30
struct ath_rate_node * ath_rc_priv )
2008-08-04 00:16:41 -07:00
{
u8 i , j , idx , idx_next ;
2008-11-18 09:03:12 +05:30
for ( i = ath_rc_priv - > max_valid_rate - 1 ; i > 0 ; i - - ) {
2008-08-04 00:16:41 -07:00
for ( j = 0 ; j < = i - 1 ; j + + ) {
2008-11-18 09:03:12 +05:30
idx = ath_rc_priv - > valid_rate_index [ j ] ;
idx_next = ath_rc_priv - > valid_rate_index [ j + 1 ] ;
2008-08-04 00:16:41 -07:00
if ( rate_table - > info [ idx ] . ratekbps >
rate_table - > info [ idx_next ] . ratekbps ) {
2008-11-18 09:03:12 +05:30
ath_rc_priv - > valid_rate_index [ j ] = idx_next ;
ath_rc_priv - > valid_rate_index [ j + 1 ] = idx ;
2008-08-04 00:16:41 -07:00
}
}
}
}
/* Access functions for valid_txrate_mask */
2008-11-18 09:03:12 +05:30
static void ath_rc_init_valid_txmask ( struct ath_rate_node * ath_rc_priv )
2008-08-04 00:16:41 -07:00
{
u8 i ;
2008-11-18 09:03:12 +05:30
for ( i = 0 ; i < ath_rc_priv - > rate_table_size ; i + + )
ath_rc_priv - > valid_rate_index [ i ] = FALSE ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:03:12 +05:30
static inline void ath_rc_set_valid_txmask ( struct ath_rate_node * ath_rc_priv ,
2008-08-04 00:16:41 -07:00
u8 index , int valid_tx_rate )
{
2008-11-18 09:03:12 +05:30
ASSERT ( index < = ath_rc_priv - > rate_table_size ) ;
ath_rc_priv - > valid_rate_index [ index ] = valid_tx_rate ? TRUE : FALSE ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:03:12 +05:30
static inline int ath_rc_isvalid_txmask ( struct ath_rate_node * ath_rc_priv ,
2008-08-04 00:16:41 -07:00
u8 index )
{
2008-11-18 09:03:12 +05:30
ASSERT ( index < = ath_rc_priv - > rate_table_size ) ;
return ath_rc_priv - > valid_rate_index [ index ] ;
2008-08-04 00:16:41 -07:00
}
/* Iterators for valid_txrate_mask */
static inline int
2008-11-18 09:07:06 +05:30
ath_rc_get_nextvalid_txrate ( struct ath_rate_table * rate_table ,
2008-11-18 09:03:12 +05:30
struct ath_rate_node * ath_rc_priv ,
2008-08-04 00:16:41 -07:00
u8 cur_valid_txrate ,
u8 * next_idx )
{
u8 i ;
2008-11-18 09:03:12 +05:30
for ( i = 0 ; i < ath_rc_priv - > max_valid_rate - 1 ; i + + ) {
if ( ath_rc_priv - > valid_rate_index [ i ] = = cur_valid_txrate ) {
* next_idx = ath_rc_priv - > valid_rate_index [ i + 1 ] ;
2008-08-04 00:16:41 -07:00
return TRUE ;
}
}
/* No more valid rates */
* next_idx = 0 ;
return FALSE ;
}
/* Return true only for single stream */
static int ath_rc_valid_phyrate ( u32 phy , u32 capflag , int ignore_cw )
{
if ( WLAN_RC_PHY_HT ( phy ) & ! ( capflag & WLAN_RC_HT_FLAG ) )
return FALSE ;
if ( WLAN_RC_PHY_DS ( phy ) & & ! ( capflag & WLAN_RC_DS_FLAG ) )
return FALSE ;
if ( WLAN_RC_PHY_SGI ( phy ) & & ! ( capflag & WLAN_RC_SGI_FLAG ) )
return FALSE ;
if ( ! ignore_cw & & WLAN_RC_PHY_HT ( phy ) )
if ( WLAN_RC_PHY_40 ( phy ) & & ! ( capflag & WLAN_RC_40_FLAG ) )
return FALSE ;
if ( ! WLAN_RC_PHY_40 ( phy ) & & ( capflag & WLAN_RC_40_FLAG ) )
return FALSE ;
return TRUE ;
}
static inline int
2008-11-18 09:07:06 +05:30
ath_rc_get_nextlowervalid_txrate ( struct ath_rate_table * rate_table ,
2008-11-18 09:03:12 +05:30
struct ath_rate_node * ath_rc_priv ,
2008-08-04 00:16:41 -07:00
u8 cur_valid_txrate , u8 * next_idx )
{
int8_t i ;
2008-11-18 09:03:12 +05:30
for ( i = 1 ; i < ath_rc_priv - > max_valid_rate ; i + + ) {
if ( ath_rc_priv - > valid_rate_index [ i ] = = cur_valid_txrate ) {
* next_idx = ath_rc_priv - > valid_rate_index [ i - 1 ] ;
2008-08-04 00:16:41 -07:00
return TRUE ;
}
}
return FALSE ;
}
/*
* Initialize the Valid Rate Index from valid entries in Rate Table
*/
static u8
ath_rc_sib_init_validrates ( struct ath_rate_node * ath_rc_priv ,
2008-11-18 09:07:06 +05:30
struct ath_rate_table * rate_table ,
2008-08-04 00:16:41 -07:00
u32 capflag )
{
u8 i , hi = 0 ;
u32 valid ;
for ( i = 0 ; i < rate_table - > rate_cnt ; i + + ) {
valid = ( ath_rc_priv - > single_stream ?
2008-08-14 13:26:55 +05:30
rate_table - > info [ i ] . valid_single_stream :
rate_table - > info [ i ] . valid ) ;
2008-08-04 00:16:41 -07:00
if ( valid = = TRUE ) {
u32 phy = rate_table - > info [ i ] . phy ;
u8 valid_rate_count = 0 ;
if ( ! ath_rc_valid_phyrate ( phy , capflag , FALSE ) )
continue ;
2008-11-18 09:03:12 +05:30
valid_rate_count = ath_rc_priv - > valid_phy_ratecnt [ phy ] ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:03:12 +05:30
ath_rc_priv - > valid_phy_rateidx [ phy ] [ valid_rate_count ] = i ;
ath_rc_priv - > valid_phy_ratecnt [ phy ] + = 1 ;
ath_rc_set_valid_txmask ( ath_rc_priv , i , TRUE ) ;
2008-08-04 00:16:41 -07:00
hi = A_MAX ( hi , i ) ;
}
}
return hi ;
}
/*
* Initialize the Valid Rate Index from Rate Set
*/
static u8
ath_rc_sib_setvalid_rates ( struct ath_rate_node * ath_rc_priv ,
2008-11-18 09:07:06 +05:30
struct ath_rate_table * rate_table ,
2008-08-04 00:16:41 -07:00
struct ath_rateset * rateset ,
u32 capflag )
{
/* XXX: Clean me up and make identation friendly */
u8 i , j , hi = 0 ;
/* Use intersection of working rates and valid rates */
for ( i = 0 ; i < rateset - > rs_nrates ; i + + ) {
for ( j = 0 ; j < rate_table - > rate_cnt ; j + + ) {
u32 phy = rate_table - > info [ j ] . phy ;
u32 valid = ( ath_rc_priv - > single_stream ?
rate_table - > info [ j ] . valid_single_stream :
rate_table - > info [ j ] . valid ) ;
/* We allow a rate only if its valid and the
* capflag matches one of the validity
* ( TRUE / TRUE_20 / TRUE_40 ) flags */
/* XXX: catch the negative of this branch
* first and then continue */
if ( ( ( rateset - > rs_rates [ i ] & 0x7F ) = =
( rate_table - > info [ j ] . dot11rate & 0x7F ) ) & &
( ( valid & WLAN_RC_CAP_MODE ( capflag ) ) = =
WLAN_RC_CAP_MODE ( capflag ) ) & &
! WLAN_RC_PHY_HT ( phy ) ) {
u8 valid_rate_count = 0 ;
if ( ! ath_rc_valid_phyrate ( phy , capflag , FALSE ) )
continue ;
valid_rate_count =
2008-11-18 09:03:12 +05:30
ath_rc_priv - > valid_phy_ratecnt [ phy ] ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:03:12 +05:30
ath_rc_priv - > valid_phy_rateidx [ phy ]
2008-08-04 00:16:41 -07:00
[ valid_rate_count ] = j ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > valid_phy_ratecnt [ phy ] + = 1 ;
ath_rc_set_valid_txmask ( ath_rc_priv , j , TRUE ) ;
2008-08-04 00:16:41 -07:00
hi = A_MAX ( hi , j ) ;
}
}
}
return hi ;
}
static u8
ath_rc_sib_setvalid_htrates ( struct ath_rate_node * ath_rc_priv ,
2008-11-18 09:07:06 +05:30
struct ath_rate_table * rate_table ,
2008-08-04 00:16:41 -07:00
u8 * mcs_set , u32 capflag )
{
u8 i , j , hi = 0 ;
/* Use intersection of working rates and valid rates */
for ( i = 0 ; i < ( ( struct ath_rateset * ) mcs_set ) - > rs_nrates ; i + + ) {
for ( j = 0 ; j < rate_table - > rate_cnt ; j + + ) {
u32 phy = rate_table - > info [ j ] . phy ;
u32 valid = ( ath_rc_priv - > single_stream ?
2008-08-14 13:26:55 +05:30
rate_table - > info [ j ] . valid_single_stream :
rate_table - > info [ j ] . valid ) ;
2008-08-04 00:16:41 -07:00
if ( ( ( ( ( struct ath_rateset * )
2008-08-14 13:26:55 +05:30
mcs_set ) - > rs_rates [ i ] & 0x7F ) ! =
( rate_table - > info [ j ] . dot11rate & 0x7F ) ) | |
! WLAN_RC_PHY_HT ( phy ) | |
! WLAN_RC_PHY_HT_VALID ( valid , capflag ) )
2008-08-04 00:16:41 -07:00
continue ;
if ( ! ath_rc_valid_phyrate ( phy , capflag , FALSE ) )
continue ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > valid_phy_rateidx [ phy ]
[ ath_rc_priv - > valid_phy_ratecnt [ phy ] ] = j ;
ath_rc_priv - > valid_phy_ratecnt [ phy ] + = 1 ;
ath_rc_set_valid_txmask ( ath_rc_priv , j , TRUE ) ;
2008-08-04 00:16:41 -07:00
hi = A_MAX ( hi , j ) ;
}
}
return hi ;
}
u8 ath_rate_findrateix ( struct ath_softc * sc ,
2008-10-29 10:15:16 +05:30
u8 dot11rate )
2008-08-04 00:16:41 -07:00
{
2008-11-18 09:07:06 +05:30
struct ath_rate_table * ratetable ;
2008-08-04 00:16:41 -07:00
int i ;
2008-11-18 09:07:06 +05:30
ratetable = sc - > hw_rate_table [ sc - > sc_curmode ] ;
2008-08-04 00:16:41 -07:00
if ( WARN_ON ( ! ratetable ) )
return 0 ;
for ( i = 0 ; i < ratetable - > rate_cnt ; i + + ) {
if ( ( ratetable - > info [ i ] . dot11rate & 0x7f ) = = ( dot11rate & 0x7f ) )
return i ;
}
return 0 ;
}
static u8 ath_rc_ratefind_ht ( struct ath_softc * sc ,
2008-08-14 13:26:55 +05:30
struct ath_rate_node * ath_rc_priv ,
2008-11-18 09:07:06 +05:30
struct ath_rate_table * rate_table ,
2008-08-14 13:26:55 +05:30
int probe_allowed , int * is_probing ,
int is_retry )
2008-08-04 00:16:41 -07:00
{
u32 dt , best_thruput , this_thruput , now_msec ;
u8 rate , next_rate , best_rate , maxindex , minindex ;
int8_t rssi_last , rssi_reduce = 0 , index = 0 ;
* is_probing = FALSE ;
2008-11-18 09:03:12 +05:30
rssi_last = median ( ath_rc_priv - > rssi_last ,
ath_rc_priv - > rssi_last_prev ,
ath_rc_priv - > rssi_last_prev2 ) ;
2008-08-04 00:16:41 -07:00
/*
* Age ( reduce ) last ack rssi based on how old it is .
* The bizarre numbers are so the delta is 160 msec ,
* meaning we divide by 16.
* 0 msec < = dt < = 25 msec : don ' t derate
* 25 msec < = dt < = 185 msec : derate linearly from 0 to 10 dB
* 185 msec < = dt : derate by 10 dB
*/
now_msec = jiffies_to_msecs ( jiffies ) ;
2008-11-18 09:03:12 +05:30
dt = now_msec - ath_rc_priv - > rssi_time ;
2008-08-04 00:16:41 -07:00
if ( dt > = 185 )
rssi_reduce = 10 ;
else if ( dt > = 25 )
rssi_reduce = ( u8 ) ( ( dt - 25 ) > > 4 ) ;
/* Now reduce rssi_last by rssi_reduce */
if ( rssi_last < rssi_reduce )
rssi_last = 0 ;
else
rssi_last - = rssi_reduce ;
/*
* Now look up the rate in the rssi table and return it .
* If no rates match then we return 0 ( lowest rate )
*/
best_thruput = 0 ;
2008-11-18 09:03:12 +05:30
maxindex = ath_rc_priv - > max_valid_rate - 1 ;
2008-08-04 00:16:41 -07:00
minindex = 0 ;
best_rate = minindex ;
/*
* Try the higher rate first . It will reduce memory moving time
* if we have very good channel characteristics .
*/
for ( index = maxindex ; index > = minindex ; index - - ) {
u8 per_thres ;
2008-11-18 09:03:12 +05:30
rate = ath_rc_priv - > valid_rate_index [ index ] ;
if ( rate > ath_rc_priv - > rate_max_phy )
2008-08-04 00:16:41 -07:00
continue ;
/*
* For TCP the average collision rate is around 11 % ,
* so we ignore PERs less than this . This is to
* prevent the rate we are currently using ( whose
* PER might be in the 10 - 15 range because of TCP
* collisions ) looking worse than the next lower
* rate whose PER has decayed close to 0. If we
* used to next lower rate , its PER would grow to
* 10 - 15 and we would be worse off then staying
* at the current rate .
*/
2008-11-18 09:03:12 +05:30
per_thres = ath_rc_priv - > state [ rate ] . per ;
2008-08-04 00:16:41 -07:00
if ( per_thres < 12 )
per_thres = 12 ;
this_thruput = rate_table - > info [ rate ] . user_ratekbps *
( 100 - per_thres ) ;
if ( best_thruput < = this_thruput ) {
best_thruput = this_thruput ;
best_rate = rate ;
}
}
rate = best_rate ;
/* if we are retrying for more than half the number
* of max retries , use the min rate for the next retry
*/
if ( is_retry )
2008-11-18 09:03:12 +05:30
rate = ath_rc_priv - > valid_rate_index [ minindex ] ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rssi_last_lookup = rssi_last ;
2008-08-04 00:16:41 -07:00
/*
* Must check the actual rate ( ratekbps ) to account for
* non - monoticity of 11 g ' s rate table
*/
2008-11-18 09:03:12 +05:30
if ( rate > = ath_rc_priv - > rate_max_phy & & probe_allowed ) {
rate = ath_rc_priv - > rate_max_phy ;
2008-08-04 00:16:41 -07:00
/* Probe the next allowed phy state */
/* FIXME:XXXX Check to make sure ratMax is checked properly */
if ( ath_rc_get_nextvalid_txrate ( rate_table ,
2008-11-18 09:03:12 +05:30
ath_rc_priv , rate , & next_rate ) & &
( now_msec - ath_rc_priv - > probe_time >
2008-08-04 00:16:41 -07:00
rate_table - > probe_interval ) & &
2008-11-18 09:03:12 +05:30
( ath_rc_priv - > hw_maxretry_pktcnt > = 1 ) ) {
2008-08-04 00:16:41 -07:00
rate = next_rate ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > probe_rate = rate ;
ath_rc_priv - > probe_time = now_msec ;
ath_rc_priv - > hw_maxretry_pktcnt = 0 ;
2008-08-04 00:16:41 -07:00
* is_probing = TRUE ;
}
}
/*
* Make sure rate is not higher than the allowed maximum .
* We should also enforce the min , but I suspect the min is
* normally 1 rather than 0 because of the rate 9 vs 6 issue
* in the old code .
*/
2008-11-18 09:03:12 +05:30
if ( rate > ( ath_rc_priv - > rate_table_size - 1 ) )
rate = ath_rc_priv - > rate_table_size - 1 ;
2008-08-04 00:16:41 -07:00
ASSERT ( ( rate_table - > info [ rate ] . valid & & ! ath_rc_priv - > single_stream ) | |
2008-08-14 13:26:55 +05:30
( rate_table - > info [ rate ] . valid_single_stream & &
ath_rc_priv - > single_stream ) ) ;
2008-08-04 00:16:41 -07:00
return rate ;
}
2008-11-18 09:07:06 +05:30
static void ath_rc_rate_set_series ( struct ath_rate_table * rate_table ,
2008-11-18 09:07:30 +05:30
struct ieee80211_tx_rate * rate ,
2008-08-04 00:16:41 -07:00
u8 tries ,
u8 rix ,
int rtsctsenable )
{
2008-11-18 09:07:30 +05:30
rate - > count = tries ;
rate - > idx = rix ;
if ( rtsctsenable )
rate - > flags | = IEEE80211_TX_RC_USE_RTS_CTS ;
if ( WLAN_RC_PHY_40 ( rate_table - > info [ rix ] . phy ) )
rate - > flags | = IEEE80211_TX_RC_40_MHZ_WIDTH ;
if ( WLAN_RC_PHY_SGI ( rate_table - > info [ rix ] . phy ) )
rate - > flags | = IEEE80211_TX_RC_SHORT_GI ;
if ( WLAN_RC_PHY_HT ( rate_table - > info [ rix ] . phy ) )
rate - > flags | = IEEE80211_TX_RC_MCS ;
2008-08-04 00:16:41 -07:00
}
static u8 ath_rc_rate_getidx ( struct ath_softc * sc ,
2008-08-14 13:26:55 +05:30
struct ath_rate_node * ath_rc_priv ,
2008-11-18 09:07:06 +05:30
struct ath_rate_table * rate_table ,
2008-08-14 13:26:55 +05:30
u8 rix , u16 stepdown ,
u16 min_rate )
2008-08-04 00:16:41 -07:00
{
u32 j ;
u8 nextindex ;
if ( min_rate ) {
for ( j = RATE_TABLE_SIZE ; j > 0 ; j - - ) {
if ( ath_rc_get_nextlowervalid_txrate ( rate_table ,
2008-11-18 09:03:12 +05:30
ath_rc_priv , rix , & nextindex ) )
2008-08-04 00:16:41 -07:00
rix = nextindex ;
else
break ;
}
} else {
for ( j = stepdown ; j > 0 ; j - - ) {
if ( ath_rc_get_nextlowervalid_txrate ( rate_table ,
2008-11-18 09:03:12 +05:30
ath_rc_priv , rix , & nextindex ) )
2008-08-04 00:16:41 -07:00
rix = nextindex ;
else
break ;
}
}
return rix ;
}
static void ath_rc_ratefind ( struct ath_softc * sc ,
struct ath_rate_node * ath_rc_priv ,
int num_tries , int num_rates , unsigned int rcflag ,
2008-11-18 09:07:30 +05:30
struct ieee80211_tx_info * tx_info , int * is_probe ,
2008-08-04 00:16:41 -07:00
int is_retry )
{
u8 try_per_rate = 0 , i = 0 , rix , nrix ;
struct ath_rate_table * rate_table ;
2008-11-18 09:07:30 +05:30
struct ieee80211_tx_rate * rates = tx_info - > control . rates ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:07:06 +05:30
rate_table = sc - > hw_rate_table [ sc - > sc_curmode ] ;
2008-08-04 00:16:41 -07:00
rix = ath_rc_ratefind_ht ( sc , ath_rc_priv , rate_table ,
2008-08-14 13:26:55 +05:30
( rcflag & ATH_RC_PROBE_ALLOWED ) ? 1 : 0 ,
is_probe , is_retry ) ;
2008-08-04 00:16:41 -07:00
nrix = rix ;
if ( ( rcflag & ATH_RC_PROBE_ALLOWED ) & & ( * is_probe ) ) {
/* set one try for probe rates. For the
* probes don ' t enable rts */
ath_rc_rate_set_series ( rate_table ,
2008-11-18 09:07:30 +05:30
& rates [ i + + ] , 1 , nrix , FALSE ) ;
2008-08-04 00:16:41 -07:00
try_per_rate = ( num_tries / num_rates ) ;
/* Get the next tried/allowed rate. No RTS for the next series
* after the probe rate
*/
nrix = ath_rc_rate_getidx ( sc ,
ath_rc_priv , rate_table , nrix , 1 , FALSE ) ;
ath_rc_rate_set_series ( rate_table ,
2008-11-18 09:07:30 +05:30
& rates [ i + + ] , try_per_rate , nrix , 0 ) ;
2008-08-04 00:16:41 -07:00
} else {
try_per_rate = ( num_tries / num_rates ) ;
/* Set the choosen rate. No RTS for first series entry. */
ath_rc_rate_set_series ( rate_table ,
2008-11-18 09:07:30 +05:30
& rates [ i + + ] , try_per_rate , nrix , FALSE ) ;
2008-08-04 00:16:41 -07:00
}
/* Fill in the other rates for multirate retry */
for ( ; i < num_rates ; i + + ) {
u8 try_num ;
u8 min_rate ;
try_num = ( ( i + 1 ) = = num_rates ) ?
num_tries - ( try_per_rate * i ) : try_per_rate ;
min_rate = ( ( ( i + 1 ) = = num_rates ) & &
2008-08-14 13:26:55 +05:30
( rcflag & ATH_RC_MINRATE_LASTRATE ) ) ? 1 : 0 ;
2008-08-04 00:16:41 -07:00
nrix = ath_rc_rate_getidx ( sc , ath_rc_priv ,
2008-08-14 13:26:55 +05:30
rate_table , nrix , 1 , min_rate ) ;
2008-08-04 00:16:41 -07:00
/* All other rates in the series have RTS enabled */
ath_rc_rate_set_series ( rate_table ,
2008-11-18 09:07:30 +05:30
& rates [ i ] , try_num , nrix , TRUE ) ;
2008-08-04 00:16:41 -07:00
}
/*
* NB : Change rate series to enable aggregation when operating
* at lower MCS rates . When first rate in series is MCS2
* in HT40 @ 2.4 GHz , series should look like :
*
* { MCS2 , MCS1 , MCS0 , MCS0 } .
*
* When first rate in series is MCS3 in HT20 @ 2.4 GHz , series should
* look like :
*
* { MCS3 , MCS2 , MCS1 , MCS1 }
*
* So , set fourth rate in series to be same as third one for
* above conditions .
*/
2008-08-07 10:54:57 +05:30
if ( ( sc - > sc_curmode = = ATH9K_MODE_11NG_HT20 ) | |
2008-08-14 13:26:55 +05:30
( sc - > sc_curmode = = ATH9K_MODE_11NG_HT40PLUS ) | |
( sc - > sc_curmode = = ATH9K_MODE_11NG_HT40MINUS ) ) {
2008-08-04 00:16:41 -07:00
u8 dot11rate = rate_table - > info [ rix ] . dot11rate ;
u8 phy = rate_table - > info [ rix ] . phy ;
if ( i = = 4 & &
( ( dot11rate = = 2 & & phy = = WLAN_RC_PHY_HT_40_SS ) | |
2008-08-14 13:26:55 +05:30
( dot11rate = = 3 & & phy = = WLAN_RC_PHY_HT_20_SS ) ) ) {
2008-11-18 09:07:30 +05:30
rates [ 3 ] . idx = rates [ 2 ] . idx ;
rates [ 3 ] . flags = rates [ 2 ] . flags ;
2008-08-04 00:16:41 -07:00
}
}
}
/*
* Return the Tx rate series .
*/
2008-08-26 08:11:26 +05:30
static void ath_rate_findrate ( struct ath_softc * sc ,
struct ath_rate_node * ath_rc_priv ,
int num_tries ,
int num_rates ,
unsigned int rcflag ,
2008-11-18 09:07:30 +05:30
struct ieee80211_tx_info * tx_info ,
2008-08-26 08:11:26 +05:30
int * is_probe ,
int is_retry )
2008-08-04 00:16:41 -07:00
{
if ( ! num_rates | | ! num_tries )
return ;
2008-11-18 09:05:35 +05:30
ath_rc_ratefind ( sc , ath_rc_priv , num_tries , num_rates ,
2008-11-18 09:07:30 +05:30
rcflag , tx_info , is_probe , is_retry ) ;
2008-08-04 00:16:41 -07:00
}
static void ath_rc_update_ht ( struct ath_softc * sc ,
struct ath_rate_node * ath_rc_priv ,
struct ath_tx_info_priv * info_priv ,
int tx_rate , int xretries , int retries )
{
u32 now_msec = jiffies_to_msecs ( jiffies ) ;
int state_change = FALSE , rate , count ;
u8 last_per ;
2008-11-18 09:07:06 +05:30
struct ath_rate_table * rate_table = sc - > hw_rate_table [ sc - > sc_curmode ] ;
2008-08-04 00:16:41 -07:00
static u32 nretry_to_per_lookup [ 10 ] = {
100 * 0 / 1 ,
100 * 1 / 4 ,
100 * 1 / 2 ,
100 * 3 / 4 ,
100 * 4 / 5 ,
100 * 5 / 6 ,
100 * 6 / 7 ,
100 * 7 / 8 ,
100 * 8 / 9 ,
100 * 9 / 10
} ;
if ( ! ath_rc_priv )
return ;
ASSERT ( tx_rate > = 0 ) ;
if ( tx_rate < 0 )
return ;
/* To compensate for some imbalance between ctrl and ext. channel */
if ( WLAN_RC_PHY_40 ( rate_table - > info [ tx_rate ] . phy ) )
info_priv - > tx . ts_rssi =
info_priv - > tx . ts_rssi < 3 ? 0 :
info_priv - > tx . ts_rssi - 3 ;
2008-11-18 09:03:12 +05:30
last_per = ath_rc_priv - > state [ tx_rate ] . per ;
2008-08-04 00:16:41 -07:00
if ( xretries ) {
/* Update the PER. */
if ( xretries = = 1 ) {
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ tx_rate ] . per + = 30 ;
if ( ath_rc_priv - > state [ tx_rate ] . per > 100 )
ath_rc_priv - > state [ tx_rate ] . per = 100 ;
2008-08-04 00:16:41 -07:00
} else {
/* xretries == 2 */
2008-11-09 17:56:10 +01:00
count = ARRAY_SIZE ( nretry_to_per_lookup ) ;
2008-08-04 00:16:41 -07:00
if ( retries > = count )
retries = count - 1 ;
/* new_PER = 7/8*old_PER + 1/8*(currentPER) */
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ tx_rate ] . per =
( u8 ) ( ath_rc_priv - > state [ tx_rate ] . per -
( ath_rc_priv - > state [ tx_rate ] . per > > 3 ) +
2008-08-14 13:26:55 +05:30
( ( 100 ) > > 3 ) ) ;
2008-08-04 00:16:41 -07:00
}
/* xretries == 1 or 2 */
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > probe_rate = = tx_rate )
ath_rc_priv - > probe_rate = 0 ;
2008-08-04 00:16:41 -07:00
} else { /* xretries == 0 */
/* Update the PER. */
/* Make sure it doesn't index out of array's bounds. */
2008-11-09 17:56:10 +01:00
count = ARRAY_SIZE ( nretry_to_per_lookup ) ;
2008-08-04 00:16:41 -07:00
if ( retries > = count )
retries = count - 1 ;
if ( info_priv - > n_bad_frames ) {
2008-08-14 13:26:55 +05:30
/* new_PER = 7/8*old_PER + 1/8*(currentPER)
2008-08-04 00:16:41 -07:00
* Assuming that n_frames is not 0. The current PER
* from the retries is 100 * retries / ( retries + 1 ) ,
* since the first retries attempts failed , and the
* next one worked . For the one that worked ,
* n_bad_frames subframes out of n_frames wored ,
* so the PER for that part is
* 100 * n_bad_frames / n_frames , and it contributes
* 100 * n_bad_frames / ( n_frames * ( retries + 1 ) ) to
* the above PER . The expression below is a
* simplified version of the sum of these two terms .
*/
if ( info_priv - > n_frames > 0 )
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ tx_rate ] . per
2008-08-04 00:16:41 -07:00
= ( u8 )
2008-11-18 09:03:12 +05:30
( ath_rc_priv - > state [ tx_rate ] . per -
( ath_rc_priv - > state [ tx_rate ] . per > > 3 ) +
2008-08-04 00:16:41 -07:00
( ( 100 * ( retries * info_priv - > n_frames +
info_priv - > n_bad_frames ) /
( info_priv - > n_frames *
( retries + 1 ) ) ) > > 3 ) ) ;
} else {
/* new_PER = 7/8*old_PER + 1/8*(currentPER) */
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ tx_rate ] . per = ( u8 )
( ath_rc_priv - > state [ tx_rate ] . per -
( ath_rc_priv - > state [ tx_rate ] . per > > 3 ) +
2008-08-04 00:16:41 -07:00
( nretry_to_per_lookup [ retries ] > > 3 ) ) ;
}
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rssi_last_prev2 = ath_rc_priv - > rssi_last_prev ;
ath_rc_priv - > rssi_last_prev = ath_rc_priv - > rssi_last ;
ath_rc_priv - > rssi_last = info_priv - > tx . ts_rssi ;
ath_rc_priv - > rssi_time = now_msec ;
2008-08-04 00:16:41 -07:00
/*
* If we got at most one retry then increase the max rate if
* this was a probe . Otherwise , ignore the probe .
*/
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > probe_rate & & ath_rc_priv - > probe_rate = = tx_rate ) {
2008-08-04 00:16:41 -07:00
if ( retries > 0 | | 2 * info_priv - > n_bad_frames >
info_priv - > n_frames ) {
/*
* Since we probed with just a single attempt ,
* any retries means the probe failed . Also ,
* if the attempt worked , but more than half
* the subframes were bad then also consider
* the probe a failure .
*/
2008-11-18 09:03:12 +05:30
ath_rc_priv - > probe_rate = 0 ;
2008-08-04 00:16:41 -07:00
} else {
u8 probe_rate = 0 ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rate_max_phy = ath_rc_priv - > probe_rate ;
probe_rate = ath_rc_priv - > probe_rate ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > state [ probe_rate ] . per > 30 )
ath_rc_priv - > state [ probe_rate ] . per = 20 ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:03:12 +05:30
ath_rc_priv - > probe_rate = 0 ;
2008-08-04 00:16:41 -07:00
/*
* Since this probe succeeded , we allow the next
* probe twice as soon . This allows the maxRate
* to move up faster if the probes are
* succesful .
*/
2008-11-18 09:03:12 +05:30
ath_rc_priv - > probe_time = now_msec -
2008-08-04 00:16:41 -07:00
rate_table - > probe_interval / 2 ;
}
}
if ( retries > 0 ) {
/*
* Don ' t update anything . We don ' t know if
* this was because of collisions or poor signal .
*
* Later : if rssi_ack is close to
2008-11-18 09:03:12 +05:30
* ath_rc_priv - > state [ txRate ] . rssi_thres and we see lots
2008-08-04 00:16:41 -07:00
* of retries , then we could increase
2008-11-18 09:03:12 +05:30
* ath_rc_priv - > state [ txRate ] . rssi_thres .
2008-08-04 00:16:41 -07:00
*/
2008-11-18 09:03:12 +05:30
ath_rc_priv - > hw_maxretry_pktcnt = 0 ;
2008-08-04 00:16:41 -07:00
} else {
/*
* It worked with no retries . First ignore bogus ( small )
* rssi_ack values .
*/
2008-11-18 09:03:12 +05:30
if ( tx_rate = = ath_rc_priv - > rate_max_phy & &
ath_rc_priv - > hw_maxretry_pktcnt < 255 ) {
ath_rc_priv - > hw_maxretry_pktcnt + + ;
2008-08-04 00:16:41 -07:00
}
if ( info_priv - > tx . ts_rssi > =
rate_table - > info [ tx_rate ] . rssi_ack_validmin ) {
/* Average the rssi */
2008-11-18 09:03:12 +05:30
if ( tx_rate ! = ath_rc_priv - > rssi_sum_rate ) {
ath_rc_priv - > rssi_sum_rate = tx_rate ;
ath_rc_priv - > rssi_sum =
ath_rc_priv - > rssi_sum_cnt = 0 ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rssi_sum + = info_priv - > tx . ts_rssi ;
ath_rc_priv - > rssi_sum_cnt + + ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > rssi_sum_cnt > 4 ) {
2008-08-04 00:16:41 -07:00
int32_t rssi_ackAvg =
2008-11-18 09:03:12 +05:30
( ath_rc_priv - > rssi_sum + 2 ) / 4 ;
2008-08-04 00:16:41 -07:00
int8_t rssi_thres =
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ tx_rate ] .
2008-08-04 00:16:41 -07:00
rssi_thres ;
int8_t rssi_ack_vmin =
rate_table - > info [ tx_rate ] .
rssi_ack_validmin ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rssi_sum =
ath_rc_priv - > rssi_sum_cnt = 0 ;
2008-08-04 00:16:41 -07:00
/* Now reduce the current
* rssi threshold . */
if ( ( rssi_ackAvg < rssi_thres + 2 ) & &
2008-08-14 13:26:55 +05:30
( rssi_thres > rssi_ack_vmin ) ) {
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ tx_rate ] .
2008-08-04 00:16:41 -07:00
rssi_thres - - ;
}
state_change = TRUE ;
}
}
}
}
/* For all cases */
/*
* If this rate looks bad ( high PER ) then stop using it for
* a while ( except if we are probing ) .
*/
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > state [ tx_rate ] . per > = 55 & & tx_rate > 0 & &
2008-08-14 13:26:55 +05:30
rate_table - > info [ tx_rate ] . ratekbps < =
2008-11-18 09:03:12 +05:30
rate_table - > info [ ath_rc_priv - > rate_max_phy ] . ratekbps ) {
ath_rc_get_nextlowervalid_txrate ( rate_table , ath_rc_priv ,
( u8 ) tx_rate , & ath_rc_priv - > rate_max_phy ) ;
2008-08-04 00:16:41 -07:00
/* Don't probe for a little while. */
2008-11-18 09:03:12 +05:30
ath_rc_priv - > probe_time = now_msec ;
2008-08-04 00:16:41 -07:00
}
if ( state_change ) {
/*
* Make sure the rates above this have higher rssi thresholds .
* ( Note : Monotonicity is kept within the OFDM rates and
* within the CCK rates . However , no adjustment is
* made to keep the rssi thresholds monotonically
* increasing between the CCK and OFDM rates . )
*/
for ( rate = tx_rate ; rate <
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rate_table_size - 1 ; rate + + ) {
2008-08-04 00:16:41 -07:00
if ( rate_table - > info [ rate + 1 ] . phy ! =
rate_table - > info [ tx_rate ] . phy )
break ;
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > state [ rate ] . rssi_thres +
2008-08-14 13:26:55 +05:30
rate_table - > info [ rate ] . rssi_ack_deltamin >
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ rate + 1 ] . rssi_thres ) {
ath_rc_priv - > state [ rate + 1 ] . rssi_thres =
ath_rc_priv - > state [ rate ] .
2008-08-14 13:26:55 +05:30
rssi_thres +
2008-08-04 00:16:41 -07:00
rate_table - > info [ rate ] .
2008-08-14 13:26:55 +05:30
rssi_ack_deltamin ;
2008-08-04 00:16:41 -07:00
}
}
/* Make sure the rates below this have lower rssi thresholds. */
for ( rate = tx_rate - 1 ; rate > = 0 ; rate - - ) {
if ( rate_table - > info [ rate ] . phy ! =
2008-08-14 13:26:55 +05:30
rate_table - > info [ tx_rate ] . phy )
2008-08-04 00:16:41 -07:00
break ;
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > state [ rate ] . rssi_thres +
2008-08-14 13:26:55 +05:30
rate_table - > info [ rate ] . rssi_ack_deltamin >
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ rate + 1 ] . rssi_thres ) {
if ( ath_rc_priv - > state [ rate + 1 ] . rssi_thres <
2008-08-14 13:26:55 +05:30
rate_table - > info [ rate ] .
rssi_ack_deltamin )
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ rate ] . rssi_thres = 0 ;
2008-08-04 00:16:41 -07:00
else {
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ rate ] . rssi_thres =
ath_rc_priv - > state [ rate + 1 ] .
2008-08-14 13:26:55 +05:30
rssi_thres -
rate_table - > info [ rate ] .
rssi_ack_deltamin ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > state [ rate ] . rssi_thres <
2008-08-14 13:26:55 +05:30
rate_table - > info [ rate ] .
rssi_ack_validmin ) {
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ rate ] . rssi_thres =
2008-08-04 00:16:41 -07:00
rate_table - > info [ rate ] .
2008-08-14 13:26:55 +05:30
rssi_ack_validmin ;
2008-08-04 00:16:41 -07:00
}
}
}
}
/* Make sure the rates below this have lower PER */
/* Monotonicity is kept only for rates below the current rate. */
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > state [ tx_rate ] . per < last_per ) {
2008-08-04 00:16:41 -07:00
for ( rate = tx_rate - 1 ; rate > = 0 ; rate - - ) {
if ( rate_table - > info [ rate ] . phy ! =
2008-08-14 13:26:55 +05:30
rate_table - > info [ tx_rate ] . phy )
2008-08-04 00:16:41 -07:00
break ;
2008-11-18 09:03:12 +05:30
if ( ath_rc_priv - > state [ rate ] . per >
ath_rc_priv - > state [ rate + 1 ] . per ) {
ath_rc_priv - > state [ rate ] . per =
ath_rc_priv - > state [ rate + 1 ] . per ;
2008-08-04 00:16:41 -07:00
}
}
}
/* Maintain monotonicity for rates above the current rate */
2008-11-18 09:03:12 +05:30
for ( rate = tx_rate ; rate < ath_rc_priv - > rate_table_size - 1 ; rate + + ) {
if ( ath_rc_priv - > state [ rate + 1 ] . per < ath_rc_priv - > state [ rate ] . per )
ath_rc_priv - > state [ rate + 1 ] . per =
ath_rc_priv - > state [ rate ] . per ;
2008-08-04 00:16:41 -07:00
}
/* Every so often, we reduce the thresholds and
* PER ( different for CCK and OFDM ) . */
2008-11-18 09:03:12 +05:30
if ( now_msec - ath_rc_priv - > rssi_down_time > =
2008-08-14 13:26:55 +05:30
rate_table - > rssi_reduce_interval ) {
2008-08-04 00:16:41 -07:00
2008-11-18 09:03:12 +05:30
for ( rate = 0 ; rate < ath_rc_priv - > rate_table_size ; rate + + ) {
if ( ath_rc_priv - > state [ rate ] . rssi_thres >
2008-08-14 13:26:55 +05:30
rate_table - > info [ rate ] . rssi_ack_validmin )
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ rate ] . rssi_thres - = 1 ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rssi_down_time = now_msec ;
2008-08-04 00:16:41 -07:00
}
/* Every so often, we reduce the thresholds
* and PER ( different for CCK and OFDM ) . */
2008-11-18 09:03:12 +05:30
if ( now_msec - ath_rc_priv - > per_down_time > =
2008-08-14 13:26:55 +05:30
rate_table - > rssi_reduce_interval ) {
2008-11-18 09:03:12 +05:30
for ( rate = 0 ; rate < ath_rc_priv - > rate_table_size ; rate + + ) {
ath_rc_priv - > state [ rate ] . per =
7 * ath_rc_priv - > state [ rate ] . per / 8 ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:03:12 +05:30
ath_rc_priv - > per_down_time = now_msec ;
2008-08-04 00:16:41 -07:00
}
}
/*
* This routine is called in rate control callback tx_status ( ) to give
* the status of previous frames .
*/
static void ath_rc_update ( struct ath_softc * sc ,
struct ath_rate_node * ath_rc_priv ,
2008-11-18 09:07:30 +05:30
struct ieee80211_tx_info * tx_info , int final_ts_idx ,
2008-08-04 00:16:41 -07:00
int xretries , int long_retry )
{
2008-11-18 09:07:30 +05:30
struct ath_tx_info_priv * info_priv =
( struct ath_tx_info_priv * ) tx_info - > rate_driver_data [ 0 ] ;
2008-08-04 00:16:41 -07:00
struct ath_rate_table * rate_table ;
2008-11-18 09:07:30 +05:30
struct ieee80211_tx_rate * rates = tx_info - > status . rates ;
2008-08-04 00:16:41 -07:00
u8 flags ;
u32 series = 0 , rix ;
2008-11-18 09:07:06 +05:30
rate_table = sc - > hw_rate_table [ sc - > sc_curmode ] ;
2008-08-04 00:16:41 -07:00
/*
* If the first rate is not the final index , there
* are intermediate rate failures to be processed .
*/
if ( final_ts_idx ! = 0 ) {
/* Process intermediate rates that failed.*/
for ( series = 0 ; series < final_ts_idx ; series + + ) {
2008-11-18 09:07:30 +05:30
if ( rates [ series ] . count ! = 0 ) {
flags = rates [ series ] . flags ;
2008-08-04 00:16:41 -07:00
/* If HT40 and we have switched mode from
* 40 to 20 = > don ' t update */
2008-11-18 09:07:30 +05:30
if ( ( flags & IEEE80211_TX_RC_40_MHZ_WIDTH ) & &
2008-11-18 09:03:12 +05:30
( ath_rc_priv - > rc_phy_mode ! =
2008-11-18 09:07:30 +05:30
( flags & IEEE80211_TX_RC_40_MHZ_WIDTH ) ) )
2008-08-04 00:16:41 -07:00
return ;
2008-11-18 09:07:30 +05:30
if ( ( flags & IEEE80211_TX_RC_40_MHZ_WIDTH ) & &
( flags & IEEE80211_TX_RC_SHORT_GI ) )
2008-08-04 00:16:41 -07:00
rix = rate_table - > info [
2008-11-18 09:07:30 +05:30
rates [ series ] . idx ] . ht_index ;
else if ( flags & IEEE80211_TX_RC_SHORT_GI )
2008-08-04 00:16:41 -07:00
rix = rate_table - > info [
2008-11-18 09:07:30 +05:30
rates [ series ] . idx ] . sgi_index ;
else if ( flags & IEEE80211_TX_RC_40_MHZ_WIDTH )
2008-08-04 00:16:41 -07:00
rix = rate_table - > info [
2008-11-18 09:07:30 +05:30
rates [ series ] . idx ] . cw40index ;
2008-08-04 00:16:41 -07:00
else
rix = rate_table - > info [
2008-11-18 09:07:30 +05:30
rates [ series ] . idx ] . base_index ;
2008-08-04 00:16:41 -07:00
ath_rc_update_ht ( sc , ath_rc_priv ,
info_priv , rix ,
xretries ? 1 : 2 ,
2008-11-18 09:07:30 +05:30
rates [ series ] . count ) ;
2008-08-04 00:16:41 -07:00
}
}
} else {
/*
* Handle the special case of MIMO PS burst , where the second
* aggregate is sent out with only one rate and one try .
* Treating it as an excessive retry penalizes the rate
* inordinately .
*/
2008-11-18 09:07:30 +05:30
if ( rates [ 0 ] . count = = 1 & & xretries = = 1 )
2008-08-04 00:16:41 -07:00
xretries = 2 ;
}
2008-11-18 09:07:30 +05:30
flags = rates [ series ] . flags ;
2008-08-04 00:16:41 -07:00
/* If HT40 and we have switched mode from 40 to 20 => don't update */
2008-11-18 09:07:30 +05:30
if ( ( flags & IEEE80211_TX_RC_40_MHZ_WIDTH ) & &
( ath_rc_priv - > rc_phy_mode ! = ( flags & IEEE80211_TX_RC_40_MHZ_WIDTH ) ) )
2008-08-04 00:16:41 -07:00
return ;
2008-11-18 09:07:30 +05:30
if ( ( flags & IEEE80211_TX_RC_40_MHZ_WIDTH ) & & ( flags & IEEE80211_TX_RC_SHORT_GI ) )
rix = rate_table - > info [ rates [ series ] . idx ] . ht_index ;
else if ( flags & IEEE80211_TX_RC_SHORT_GI )
rix = rate_table - > info [ rates [ series ] . idx ] . sgi_index ;
else if ( flags & IEEE80211_TX_RC_40_MHZ_WIDTH )
rix = rate_table - > info [ rates [ series ] . idx ] . cw40index ;
2008-08-04 00:16:41 -07:00
else
2008-11-18 09:07:30 +05:30
rix = rate_table - > info [ rates [ series ] . idx ] . base_index ;
2008-08-04 00:16:41 -07:00
ath_rc_update_ht ( sc , ath_rc_priv , info_priv , rix ,
xretries , long_retry ) ;
}
/*
* Process a tx descriptor for a completed transmit ( success or failure ) .
*/
static void ath_rate_tx_complete ( struct ath_softc * sc ,
struct ath_node * an ,
struct ath_rate_node * rc_priv ,
2008-11-18 09:07:30 +05:30
struct ieee80211_tx_info * tx_info )
2008-08-04 00:16:41 -07:00
{
2008-11-18 09:07:30 +05:30
struct ath_tx_info_priv * info_priv =
( struct ath_tx_info_priv * ) tx_info - > rate_driver_data [ 0 ] ;
2008-08-04 00:16:41 -07:00
int final_ts_idx = info_priv - > tx . ts_rateindex ;
int tx_status = 0 , is_underrun = 0 ;
2008-11-18 09:05:35 +05:30
if ( info_priv - > tx . ts_status & ATH9K_TXERR_FILT )
2008-08-04 00:16:41 -07:00
return ;
if ( info_priv - > tx . ts_rssi > 0 ) {
ATH_RSSI_LPF ( an - > an_chainmask_sel . tx_avgrssi ,
2008-08-14 13:26:55 +05:30
info_priv - > tx . ts_rssi ) ;
2008-08-04 00:16:41 -07:00
}
/*
* If underrun error is seen assume it as an excessive retry only
* if prefetch trigger level have reached the max ( 0x3f for 5416 )
* Adjust the long retry as if the frame was tried ATH_11N_TXMAXTRY
* times . This affects how ratectrl updates PER for the failed rate .
*/
if ( info_priv - > tx . ts_flags &
( ATH9K_TX_DATA_UNDERRUN | ATH9K_TX_DELIM_UNDERRUN ) & &
2008-11-18 09:07:06 +05:30
( ( sc - > sc_ah - > ah_txTrigLevel ) > = rc_priv - > tx_triglevel_max ) ) {
2008-08-04 00:16:41 -07:00
tx_status = 1 ;
is_underrun = 1 ;
}
if ( ( info_priv - > tx . ts_status & ATH9K_TXERR_XRETRY ) | |
( info_priv - > tx . ts_status & ATH9K_TXERR_FIFO ) )
tx_status = 1 ;
2008-11-18 09:07:30 +05:30
ath_rc_update ( sc , rc_priv , tx_info , final_ts_idx , tx_status ,
2008-08-04 00:16:41 -07:00
( is_underrun ) ? ATH_11N_TXMAXTRY :
info_priv - > tx . ts_longretry ) ;
}
2008-11-18 09:06:44 +05:30
static void ath_rc_init ( struct ath_softc * sc ,
struct ath_rate_node * ath_rc_priv ,
struct ieee80211_supported_band * sband ,
struct ieee80211_sta * sta )
2008-08-04 00:16:41 -07:00
{
struct ath_rate_table * rate_table = NULL ;
2008-11-18 09:06:44 +05:30
struct ath_rateset * rateset = & ath_rc_priv - > neg_rates ;
u8 * ht_mcs = ( u8 * ) & ath_rc_priv - > neg_ht_rates ;
2008-08-04 00:16:41 -07:00
u8 i , j , k , hi = 0 , hthi = 0 ;
2008-11-18 09:07:06 +05:30
rate_table = sc - > hw_rate_table [ sc - > sc_curmode ] ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:06:44 +05:30
if ( sta - > ht_cap . ht_supported ) {
if ( sband - > band = = IEEE80211_BAND_2GHZ )
2008-11-18 09:07:06 +05:30
rate_table = sc - > hw_rate_table [ ATH9K_MODE_11NG_HT20 ] ;
2008-11-18 09:06:44 +05:30
else
2008-11-18 09:07:06 +05:30
rate_table = sc - > hw_rate_table [ ATH9K_MODE_11NA_HT20 ] ;
2008-11-18 09:06:44 +05:30
ath_rc_priv - > ht_cap = ( WLAN_RC_HT_FLAG | WLAN_RC_DS_FLAG ) ;
if ( sta - > ht_cap . cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40 )
ath_rc_priv - > ht_cap | = WLAN_RC_40_FLAG ;
}
2008-08-04 00:16:41 -07:00
/* Initial rate table size. Will change depending
* on the working rate set */
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rate_table_size = MAX_TX_RATE_TBL ;
2008-08-04 00:16:41 -07:00
/* Initialize thresholds according to the global rate table */
2008-11-18 09:06:44 +05:30
for ( i = 0 ; i < ath_rc_priv - > rate_table_size ; i + + ) {
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ i ] . rssi_thres =
2008-08-04 00:16:41 -07:00
rate_table - > info [ i ] . rssi_ack_validmin ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > state [ i ] . per = 0 ;
2008-08-04 00:16:41 -07:00
}
/* Determine the valid rates */
2008-11-18 09:03:12 +05:30
ath_rc_init_valid_txmask ( ath_rc_priv ) ;
2008-08-04 00:16:41 -07:00
for ( i = 0 ; i < WLAN_RC_PHY_MAX ; i + + ) {
for ( j = 0 ; j < MAX_TX_RATE_PHY ; j + + )
2008-11-18 09:03:12 +05:30
ath_rc_priv - > valid_phy_rateidx [ i ] [ j ] = 0 ;
ath_rc_priv - > valid_phy_ratecnt [ i ] = 0 ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:06:44 +05:30
ath_rc_priv - > rc_phy_mode = ( ath_rc_priv - > ht_cap & WLAN_RC_40_FLAG ) ;
2008-08-04 00:16:41 -07:00
/* Set stream capability */
2008-11-18 09:06:44 +05:30
ath_rc_priv - > single_stream = ( ath_rc_priv - > ht_cap & WLAN_RC_DS_FLAG ) ? 0 : 1 ;
2008-08-04 00:16:41 -07:00
if ( ! rateset - > rs_nrates ) {
/* No working rate, just initialize valid rates */
hi = ath_rc_sib_init_validrates ( ath_rc_priv , rate_table ,
2008-11-18 09:06:44 +05:30
ath_rc_priv - > ht_cap ) ;
2008-08-04 00:16:41 -07:00
} else {
/* Use intersection of working rates and valid rates */
hi = ath_rc_sib_setvalid_rates ( ath_rc_priv , rate_table ,
2008-11-18 09:06:44 +05:30
rateset , ath_rc_priv - > ht_cap ) ;
if ( ath_rc_priv - > ht_cap & WLAN_RC_HT_FLAG ) {
2008-08-04 00:16:41 -07:00
hthi = ath_rc_sib_setvalid_htrates ( ath_rc_priv ,
rate_table ,
ht_mcs ,
2008-11-18 09:06:44 +05:30
ath_rc_priv - > ht_cap ) ;
2008-08-04 00:16:41 -07:00
}
hi = A_MAX ( hi , hthi ) ;
}
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rate_table_size = hi + 1 ;
ath_rc_priv - > rate_max_phy = 0 ;
ASSERT ( ath_rc_priv - > rate_table_size < = MAX_TX_RATE_TBL ) ;
2008-08-04 00:16:41 -07:00
for ( i = 0 , k = 0 ; i < WLAN_RC_PHY_MAX ; i + + ) {
2008-11-18 09:03:12 +05:30
for ( j = 0 ; j < ath_rc_priv - > valid_phy_ratecnt [ i ] ; j + + ) {
ath_rc_priv - > valid_rate_index [ k + + ] =
ath_rc_priv - > valid_phy_rateidx [ i ] [ j ] ;
2008-08-04 00:16:41 -07:00
}
if ( ! ath_rc_valid_phyrate ( i , rate_table - > initial_ratemax , TRUE )
2008-11-18 09:03:12 +05:30
| | ! ath_rc_priv - > valid_phy_ratecnt [ i ] )
2008-08-04 00:16:41 -07:00
continue ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > rate_max_phy = ath_rc_priv - > valid_phy_rateidx [ i ] [ j - 1 ] ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:03:12 +05:30
ASSERT ( ath_rc_priv - > rate_table_size < = MAX_TX_RATE_TBL ) ;
2008-08-04 00:16:41 -07:00
ASSERT ( k < = MAX_TX_RATE_TBL ) ;
2008-11-18 09:03:12 +05:30
ath_rc_priv - > max_valid_rate = k ;
2008-08-04 00:16:41 -07:00
/*
* Some third party vendors don ' t send the supported rate series in
* order . So sorting to make sure its in order , otherwise our RateFind
* Algo will select wrong rates
*/
2008-11-18 09:03:12 +05:30
ath_rc_sort_validrates ( rate_table , ath_rc_priv ) ;
ath_rc_priv - > rate_max_phy = ath_rc_priv - > valid_rate_index [ k - 4 ] ;
2008-08-04 00:16:41 -07:00
}
/* Rate Control callbacks */
2008-09-18 18:14:18 +02:00
static void ath_tx_status ( void * priv , struct ieee80211_supported_band * sband ,
struct ieee80211_sta * sta , void * priv_sta ,
2008-08-04 00:16:41 -07:00
struct sk_buff * skb )
{
struct ath_softc * sc = priv ;
struct ath_tx_info_priv * tx_info_priv ;
struct ath_node * an ;
struct ieee80211_tx_info * tx_info = IEEE80211_SKB_CB ( skb ) ;
struct ieee80211_hdr * hdr ;
__le16 fc ;
hdr = ( struct ieee80211_hdr * ) skb - > data ;
fc = hdr - > frame_control ;
2008-11-18 09:07:30 +05:30
tx_info_priv = ( struct ath_tx_info_priv * ) tx_info - > rate_driver_data [ 0 ] ;
2008-08-04 00:16:41 -07:00
2008-10-29 10:13:31 +05:30
an = ( struct ath_node * ) sta - > drv_priv ;
2008-08-04 00:16:41 -07:00
2008-10-21 12:40:02 +02:00
if ( tx_info_priv = = NULL )
2008-08-04 00:16:41 -07:00
return ;
2008-10-21 12:40:02 +02:00
if ( an & & priv_sta & & ieee80211_is_data ( fc ) )
2008-11-18 09:07:30 +05:30
ath_rate_tx_complete ( sc , an , priv_sta , tx_info ) ;
2008-10-21 12:40:02 +02:00
kfree ( tx_info_priv ) ;
2008-08-04 00:16:41 -07:00
}
2008-10-21 12:40:02 +02:00
static void ath_get_rate ( void * priv , struct ieee80211_sta * sta , void * priv_sta ,
struct ieee80211_tx_rate_control * txrc )
2008-08-04 00:16:41 -07:00
{
2008-10-21 12:40:02 +02:00
struct ieee80211_supported_band * sband = txrc - > sband ;
struct sk_buff * skb = txrc - > skb ;
2008-08-04 00:16:41 -07:00
struct ieee80211_hdr * hdr = ( struct ieee80211_hdr * ) skb - > data ;
2008-09-18 18:14:18 +02:00
struct ath_softc * sc = priv ;
2008-08-04 00:16:41 -07:00
struct ieee80211_hw * hw = sc - > hw ;
2008-09-18 18:14:18 +02:00
struct ath_rate_node * ath_rc_priv = priv_sta ;
2008-08-04 00:16:41 -07:00
struct ath_node * an ;
struct ieee80211_tx_info * tx_info = IEEE80211_SKB_CB ( skb ) ;
2008-10-29 10:18:14 +05:30
int is_probe = FALSE ;
2008-08-04 00:16:41 -07:00
__le16 fc = hdr - > frame_control ;
u8 * qc , tid ;
/* lowest rate for management and multicast/broadcast frames */
2008-11-18 09:07:30 +05:30
if ( ! ieee80211_is_data ( fc ) | | is_multicast_ether_addr ( hdr - > addr1 ) ) {
tx_info - > control . rates [ 0 ] . idx = rate_lowest_index ( sband , sta ) ;
tx_info - > control . rates [ 0 ] . count =
is_multicast_ether_addr ( hdr - > addr1 ) ? 1 : ATH_MGT_TXMAXTRY ;
2008-08-04 00:16:41 -07:00
return ;
}
/* Find tx rate for unicast frames */
ath_rate_findrate ( sc , ath_rc_priv ,
ATH_11N_TXMAXTRY , 4 ,
ATH_RC_PROBE_ALLOWED ,
2008-11-18 09:07:30 +05:30
tx_info ,
2008-08-04 00:16:41 -07:00
& is_probe ,
false ) ;
/* Check if aggregation has to be enabled for this tid */
2008-10-14 16:58:37 +02:00
if ( hw - > conf . ht . enabled ) {
2008-08-04 00:16:41 -07:00
if ( ieee80211_is_data_qos ( fc ) ) {
qc = ieee80211_get_qos_ctl ( hdr ) ;
tid = qc [ 0 ] & 0xf ;
2008-10-29 10:13:31 +05:30
an = ( struct ath_node * ) sta - > drv_priv ;
2008-08-04 00:16:41 -07:00
2008-10-29 10:18:14 +05:30
if ( ath_tx_aggr_check ( sc , an , tid ) )
ieee80211_start_tx_ba_session ( hw , hdr - > addr1 , tid ) ;
2008-08-04 00:16:41 -07:00
}
}
}
2008-09-18 18:14:18 +02:00
static void ath_rate_init ( void * priv , struct ieee80211_supported_band * sband ,
struct ieee80211_sta * sta , void * priv_sta )
2008-08-04 00:16:41 -07:00
{
2008-09-18 18:14:18 +02:00
struct ath_softc * sc = priv ;
2008-08-14 13:26:55 +05:30
struct ath_rate_node * ath_rc_priv = priv_sta ;
2008-08-04 00:16:41 -07:00
int i , j = 0 ;
2008-11-18 09:04:00 +05:30
for ( i = 0 ; i < sband - > n_bitrates ; i + + ) {
if ( sta - > supp_rates [ sband - > band ] & BIT ( i ) ) {
ath_rc_priv - > neg_rates . rs_rates [ j ]
= ( sband - > bitrates [ i ] . bitrate * 2 ) / 10 ;
j + + ;
}
}
ath_rc_priv - > neg_rates . rs_nrates = j ;
2008-08-04 00:16:41 -07:00
2008-10-23 12:15:19 +05:30
if ( sta - > ht_cap . ht_supported ) {
2008-11-18 09:04:00 +05:30
for ( i = 0 , j = 0 ; i < 77 ; i + + ) {
2008-10-14 16:58:37 +02:00
if ( sta - > ht_cap . mcs . rx_mask [ i / 8 ] & ( 1 < < ( i % 8 ) ) )
2008-08-14 13:26:55 +05:30
ath_rc_priv - > neg_ht_rates . rs_rates [ j + + ] = i ;
2008-08-04 00:16:41 -07:00
if ( j = = ATH_RATE_MAX )
break ;
}
2008-08-14 13:26:55 +05:30
ath_rc_priv - > neg_ht_rates . rs_nrates = j ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:04:00 +05:30
2008-11-18 09:06:44 +05:30
ath_rc_init ( sc , priv_sta , sband , sta ) ;
2008-08-04 00:16:41 -07:00
}
2008-09-18 18:14:18 +02:00
static void * ath_rate_alloc ( struct ieee80211_hw * hw , struct dentry * debugfsdir )
2008-08-04 00:16:41 -07:00
{
2008-09-18 18:14:18 +02:00
return hw - > priv ;
2008-08-04 00:16:41 -07:00
}
static void ath_rate_free ( void * priv )
{
return ;
}
2008-09-18 18:14:18 +02:00
static void * ath_rate_alloc_sta ( void * priv , struct ieee80211_sta * sta , gfp_t gfp )
2008-08-04 00:16:41 -07:00
{
struct ath_softc * sc = priv ;
struct ath_rate_node * rate_priv ;
2008-11-18 09:03:36 +05:30
rate_priv = kzalloc ( sizeof ( struct ath_rate_node ) , gfp ) ;
2008-08-04 00:16:41 -07:00
if ( ! rate_priv ) {
2008-08-14 13:26:55 +05:30
DPRINTF ( sc , ATH_DBG_FATAL ,
" %s: Unable to allocate private rc structure \n " ,
__func__ ) ;
2008-08-04 00:16:41 -07:00
return NULL ;
}
2008-11-18 09:03:36 +05:30
rate_priv - > rssi_down_time = jiffies_to_msecs ( jiffies ) ;
2008-11-18 09:07:06 +05:30
rate_priv - > tx_triglevel_max = sc - > sc_ah - > ah_caps . tx_triglevel_max ;
2008-08-14 13:26:55 +05:30
2008-08-04 00:16:41 -07:00
return rate_priv ;
}
2008-09-18 18:14:18 +02:00
static void ath_rate_free_sta ( void * priv , struct ieee80211_sta * sta ,
void * priv_sta )
2008-08-04 00:16:41 -07:00
{
struct ath_rate_node * rate_priv = priv_sta ;
2008-11-18 09:03:36 +05:30
kfree ( rate_priv ) ;
2008-08-04 00:16:41 -07:00
}
static struct rate_control_ops ath_rate_ops = {
. module = NULL ,
. name = " ath9k_rate_control " ,
. tx_status = ath_tx_status ,
. get_rate = ath_get_rate ,
. rate_init = ath_rate_init ,
. alloc = ath_rate_alloc ,
. free = ath_rate_free ,
. alloc_sta = ath_rate_alloc_sta ,
2008-09-18 18:14:18 +02:00
. free_sta = ath_rate_free_sta ,
2008-08-04 00:16:41 -07:00
} ;
2008-11-18 09:07:06 +05:30
void ath_rate_attach ( struct ath_softc * sc )
{
sc - > hw_rate_table [ ATH9K_MODE_11B ] =
& ar5416_11b_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11A ] =
& ar5416_11a_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11G ] =
& ar5416_11g_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11NA_HT20 ] =
& ar5416_11na_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11NG_HT20 ] =
& ar5416_11ng_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11NA_HT40PLUS ] =
& ar5416_11na_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11NA_HT40MINUS ] =
& ar5416_11na_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11NG_HT40PLUS ] =
& ar5416_11ng_ratetable ;
sc - > hw_rate_table [ ATH9K_MODE_11NG_HT40MINUS ] =
& ar5416_11ng_ratetable ;
}
2008-08-04 00:16:41 -07:00
int ath_rate_control_register ( void )
{
return ieee80211_rate_control_register ( & ath_rate_ops ) ;
}
void ath_rate_control_unregister ( void )
{
ieee80211_rate_control_unregister ( & ath_rate_ops ) ;
}