blob: ad1ed5084226dc943b84d08bcfd255b6b01d19cc [file] [log] [blame]
Forest Bond92b96792009-06-13 07:38:31 -04001/*
2 * Copyright (c) 1996, 2003 VIA Networking Technologies, Inc.
3 * All rights reserved.
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * File: datarate.c
20 *
21 * Purpose: Handles the auto fallback & data rates functions
22 *
23 * Author: Lyndon Chen
24 *
25 * Date: July 17, 2002
26 *
27 * Functions:
28 * RATEvParseMaxRate - Parsing the highest basic & support rate in rate field of frame
29 * RATEvTxRateFallBack - Rate fallback Algorithm Implementaion
30 * RATEuSetIE- Set rate IE field.
31 *
32 * Revision History:
33 *
34 */
35
36#if !defined(__TTYPE_H__)
37#include "ttype.h"
38#endif
39#if !defined(__TMACRO_H__)
40#include "tmacro.h"
41#endif
42#if !defined(__MAC_H__)
43#include "mac.h"
44#endif
45#if !defined(__80211MGR_H__)
46#include "80211mgr.h"
47#endif
48#if !defined(__BSSDB_H__)
49#include "bssdb.h"
50#endif
51#if !defined(__DATARATE_H__)
52#include "datarate.h"
53#endif
54#if !defined(__CARD_H__)
55#include "card.h"
56#endif
57#if !defined(__BASEBAND_H__)
58#include "baseband.h"
59#endif
60#if !defined(__SROM_H__)
61#include "srom.h"
62#endif
63#if !defined(__RF_H__)
64#include "rf.h"
65#endif
66/*--------------------- Static Definitions -------------------------*/
67
68
69
70
71/*--------------------- Static Classes ----------------------------*/
72
73
74
75/*--------------------- Static Variables --------------------------*/
76//static int msglevel =MSG_LEVEL_DEBUG;
77static int msglevel =MSG_LEVEL_INFO;
78const BYTE acbyIERate[MAX_RATE] =
79{0x02, 0x04, 0x0B, 0x16, 0x0C, 0x12, 0x18, 0x24, 0x30, 0x48, 0x60, 0x6C};
80
81#define AUTORATE_TXOK_CNT 0x0400
82#define AUTORATE_TXFAIL_CNT 0x0064
83#define AUTORATE_TIMEOUT 10
84
85/*--------------------- Static Functions --------------------------*/
86
87VOID s_vResetCounter (
88 IN PKnownNodeDB psNodeDBTable
89 );
90
91
92
93VOID
94s_vResetCounter (
95 IN PKnownNodeDB psNodeDBTable
96 )
97{
98 BYTE ii;
99
100 // clear statistic counter for auto_rate
101 for(ii=0;ii<=MAX_RATE;ii++) {
102 psNodeDBTable->uTxOk[ii] = 0;
103 psNodeDBTable->uTxFail[ii] = 0;
104 }
105}
106
107/*--------------------- Export Variables --------------------------*/
108
109
110/*--------------------- Export Functions --------------------------*/
111
112
113/*+
114 *
115 * Description:
116 * Get RateIdx from the value in SuppRates IE or ExtSuppRates IE
117 *
118 * Parameters:
119 * In:
120 * BYTE - Rate value in SuppRates IE or ExtSuppRates IE
121 * Out:
122 * none
123 *
124 * Return Value: RateIdx
125 *
126-*/
127BYTE
128DATARATEbyGetRateIdx (
129 IN BYTE byRate
130 )
131{
132 BYTE ii;
133
134 //Erase basicRate flag.
135 byRate = byRate & 0x7F;//0111 1111
136
137 for (ii = 0; ii < MAX_RATE; ii ++) {
138 if (acbyIERate[ii] == byRate)
139 return ii;
140 }
141 return 0;
142}
143
144
145
146/*+
147 *
148 * Routine Description:
149 * Rate fallback Algorithm Implementaion
150 *
151 * Parameters:
152 * In:
153 * pDevice - Pointer to the adapter
154 * psNodeDBTable - Pointer to Node Data Base
155 * Out:
156 * none
157 *
158 * Return Value: none
159 *
160-*/
161#define AUTORATE_TXCNT_THRESHOLD 20
162#define AUTORATE_INC_THRESHOLD 30
163
164
165
166
167/*+
168 *
169 * Description:
170 * Get RateIdx from the value in SuppRates IE or ExtSuppRates IE
171 *
172 * Parameters:
173 * In:
174 * BYTE - Rate value in SuppRates IE or ExtSuppRates IE
175 * Out:
176 * none
177 *
178 * Return Value: RateIdx
179 *
180-*/
181WORD
182RATEwGetRateIdx(
183 IN BYTE byRate
184 )
185{
186 WORD ii;
187
188 //Erase basicRate flag.
189 byRate = byRate & 0x7F;//0111 1111
190
191 for (ii = 0; ii < MAX_RATE; ii ++) {
192 if (acbyIERate[ii] == byRate)
193 return ii;
194 }
195 return 0;
196}
197
198/*+
199 *
200 * Description:
201 * Parsing the highest basic & support rate in rate field of frame.
202 *
203 * Parameters:
204 * In:
205 * pDevice - Pointer to the adapter
206 * pItemRates - Pointer to Rate field defined in 802.11 spec.
207 * pItemExtRates - Pointer to Extended Rate field defined in 802.11 spec.
208 * Out:
209 * pwMaxBasicRate - Maximum Basic Rate
210 * pwMaxSuppRate - Maximum Supported Rate
211 * pbyTopCCKRate - Maximum Basic Rate in CCK mode
212 * pbyTopOFDMRate - Maximum Basic Rate in OFDM mode
213 *
214 * Return Value: none
215 *
216-*/
217VOID
218RATEvParseMaxRate (
219 IN PVOID pDeviceHandler,
220 IN PWLAN_IE_SUPP_RATES pItemRates,
221 IN PWLAN_IE_SUPP_RATES pItemExtRates,
222 IN BOOL bUpdateBasicRate,
223 OUT PWORD pwMaxBasicRate,
224 OUT PWORD pwMaxSuppRate,
225 OUT PWORD pwSuppRate,
226 OUT PBYTE pbyTopCCKRate,
227 OUT PBYTE pbyTopOFDMRate
228 )
229{
230PSDevice pDevice = (PSDevice) pDeviceHandler;
231UINT ii;
232BYTE byHighSuppRate = 0;
233BYTE byRate = 0;
234WORD wOldBasicRate = pDevice->wBasicRate;
235UINT uRateLen;
236
237
238 if (pItemRates == NULL)
239 return;
240
241 *pwSuppRate = 0;
242 uRateLen = pItemRates->len;
243
244 DBG_PRT(MSG_LEVEL_DEBUG, KERN_INFO"ParseMaxRate Len: %d\n", uRateLen);
245 if (pDevice->byBBType != BB_TYPE_11B) {
246 if (uRateLen > WLAN_RATES_MAXLEN)
247 uRateLen = WLAN_RATES_MAXLEN;
248 } else {
249 if (uRateLen > WLAN_RATES_MAXLEN_11B)
250 uRateLen = WLAN_RATES_MAXLEN_11B;
251 }
252
253 for (ii = 0; ii < uRateLen; ii++) {
254 byRate = (BYTE)(pItemRates->abyRates[ii]);
255 if (WLAN_MGMT_IS_BASICRATE(byRate) &&
256 (bUpdateBasicRate == TRUE)) {
257 // Add to basic rate set, update pDevice->byTopCCKBasicRate and pDevice->byTopOFDMBasicRate
258 CARDbAddBasicRate((PVOID)pDevice, RATEwGetRateIdx(byRate));
259 DBG_PRT(MSG_LEVEL_DEBUG, KERN_INFO"ParseMaxRate AddBasicRate: %d\n", RATEwGetRateIdx(byRate));
260 }
261 byRate = (BYTE)(pItemRates->abyRates[ii]&0x7F);
262 if (byHighSuppRate == 0)
263 byHighSuppRate = byRate;
264 if (byRate > byHighSuppRate)
265 byHighSuppRate = byRate;
266 *pwSuppRate |= (1<<RATEwGetRateIdx(byRate));
267 }
268 if ((pItemExtRates != NULL) && (pItemExtRates->byElementID == WLAN_EID_EXTSUPP_RATES) &&
269 (pDevice->byBBType != BB_TYPE_11B)) {
270
271 UINT uExtRateLen = pItemExtRates->len;
272
273 if (uExtRateLen > WLAN_RATES_MAXLEN)
274 uExtRateLen = WLAN_RATES_MAXLEN;
275
276 for (ii = 0; ii < uExtRateLen ; ii++) {
277 byRate = (BYTE)(pItemExtRates->abyRates[ii]);
278 // select highest basic rate
279 if (WLAN_MGMT_IS_BASICRATE(pItemExtRates->abyRates[ii])) {
280 // Add to basic rate set, update pDevice->byTopCCKBasicRate and pDevice->byTopOFDMBasicRate
281 CARDbAddBasicRate((PVOID)pDevice, RATEwGetRateIdx(byRate));
282 DBG_PRT(MSG_LEVEL_DEBUG, KERN_INFO"ParseMaxRate AddBasicRate: %d\n", RATEwGetRateIdx(byRate));
283 }
284 byRate = (BYTE)(pItemExtRates->abyRates[ii]&0x7F);
285 if (byHighSuppRate == 0)
286 byHighSuppRate = byRate;
287 if (byRate > byHighSuppRate)
288 byHighSuppRate = byRate;
289 *pwSuppRate |= (1<<RATEwGetRateIdx(byRate));
290 //DBG_PRN_GRP09(("ParseMaxRate : HighSuppRate: %d, %X\n", RATEwGetRateIdx(byRate), byRate));
291 }
292 } //if(pItemExtRates != NULL)
293
294 if ((pDevice->byPacketType == PK_TYPE_11GB) && CARDbIsOFDMinBasicRate((PVOID)pDevice)) {
295 pDevice->byPacketType = PK_TYPE_11GA;
296 }
297
298 *pbyTopCCKRate = pDevice->byTopCCKBasicRate;
299 *pbyTopOFDMRate = pDevice->byTopOFDMBasicRate;
300 *pwMaxSuppRate = RATEwGetRateIdx(byHighSuppRate);
301 if ((pDevice->byPacketType==PK_TYPE_11B) || (pDevice->byPacketType==PK_TYPE_11GB))
302 *pwMaxBasicRate = pDevice->byTopCCKBasicRate;
303 else
304 *pwMaxBasicRate = pDevice->byTopOFDMBasicRate;
305 if (wOldBasicRate != pDevice->wBasicRate)
306 CARDvSetRSPINF((PVOID)pDevice, pDevice->byBBType);
307
308 DBG_PRT(MSG_LEVEL_DEBUG, KERN_INFO"Exit ParseMaxRate\n");
309}
310
311
312/*+
313 *
314 * Routine Description:
315 * Rate fallback Algorithm Implementaion
316 *
317 * Parameters:
318 * In:
319 * pDevice - Pointer to the adapter
320 * psNodeDBTable - Pointer to Node Data Base
321 * Out:
322 * none
323 *
324 * Return Value: none
325 *
326-*/
327#define AUTORATE_TXCNT_THRESHOLD 20
328#define AUTORATE_INC_THRESHOLD 30
329
330VOID
331RATEvTxRateFallBack (
332 IN PVOID pDeviceHandler,
333 IN PKnownNodeDB psNodeDBTable
334 )
335{
336PSDevice pDevice = (PSDevice) pDeviceHandler;
337PSMgmtObject pMgmt = &(pDevice->sMgmtObj);
338#if 1 //mike fixed old: use packet lose ratio algorithm to control rate
339WORD wIdxDownRate = 0;
340UINT ii;
341BOOL bAutoRate[MAX_RATE] = {TRUE,TRUE,TRUE,TRUE,FALSE,FALSE,TRUE,TRUE,TRUE,TRUE,TRUE,TRUE};
342DWORD dwThroughputTbl[MAX_RATE] = {10, 20, 55, 110, 60, 90, 120, 180, 240, 360, 480, 540};
343DWORD dwThroughput = 0;
344WORD wIdxUpRate = 0;
345DWORD dwTxDiff = 0;
346
347 if (pMgmt->eScanState != WMAC_NO_SCANNING) {
348 // Don't do Fallback when scanning Channel
349 return;
350 }
351 psNodeDBTable->uTimeCount ++;
352
353 if (psNodeDBTable->uTxFail[MAX_RATE] > psNodeDBTable->uTxOk[MAX_RATE])
354 dwTxDiff = psNodeDBTable->uTxFail[MAX_RATE] - psNodeDBTable->uTxOk[MAX_RATE];
355
356 if ((psNodeDBTable->uTxOk[MAX_RATE] < AUTORATE_TXOK_CNT) &&
357 (dwTxDiff < AUTORATE_TXFAIL_CNT) &&
358 (psNodeDBTable->uTimeCount < AUTORATE_TIMEOUT)) {
359 return;
360 }
361
362 if (psNodeDBTable->uTimeCount >= AUTORATE_TIMEOUT) {
363 psNodeDBTable->uTimeCount = 0;
364 }
365
366 for(ii=0;ii<MAX_RATE;ii++) {
367 if (psNodeDBTable->wSuppRate & (0x0001<<ii)) {
368 if (bAutoRate[ii] == TRUE) {
369 wIdxUpRate = (WORD) ii;
370 }
371 } else {
372 bAutoRate[ii] = FALSE;
373 }
374 }
375
376 for(ii=0;ii<=psNodeDBTable->wTxDataRate;ii++) {
377 if ( (psNodeDBTable->uTxOk[ii] != 0) ||
378 (psNodeDBTable->uTxFail[ii] != 0) ) {
379 dwThroughputTbl[ii] *= psNodeDBTable->uTxOk[ii];
380 if (ii < RATE_11M) {
381 psNodeDBTable->uTxFail[ii] *= 4;
382 }
383 dwThroughputTbl[ii] /= (psNodeDBTable->uTxOk[ii] + psNodeDBTable->uTxFail[ii]);
384 }
385 DBG_PRT(MSG_LEVEL_DEBUG, KERN_INFO"Rate %d,Ok: %d, Fail:%d, Throughput:%d\n",
386 ii, (int)psNodeDBTable->uTxOk[ii], (int)psNodeDBTable->uTxFail[ii], (int)dwThroughputTbl[ii]);
387 }
388 dwThroughput = dwThroughputTbl[psNodeDBTable->wTxDataRate];
389
390 wIdxDownRate = psNodeDBTable->wTxDataRate;
391 for(ii = psNodeDBTable->wTxDataRate; ii > 0;) {
392 ii--;
393 if ( (dwThroughputTbl[ii] > dwThroughput) &&
394 (bAutoRate[ii]==TRUE) ) {
395 dwThroughput = dwThroughputTbl[ii];
396 wIdxDownRate = (WORD) ii;
397 }
398 }
399 psNodeDBTable->wTxDataRate = wIdxDownRate;
400 if (psNodeDBTable->uTxOk[MAX_RATE]) {
401 if (psNodeDBTable->uTxOk[MAX_RATE] >
402 (psNodeDBTable->uTxFail[MAX_RATE] * 4) ) {
403 psNodeDBTable->wTxDataRate = wIdxUpRate;
404 }
405 }else { // adhoc, if uTxOk(total) =0 & uTxFail(total) = 0
406 if (psNodeDBTable->uTxFail[MAX_RATE] == 0)
407 psNodeDBTable->wTxDataRate = wIdxUpRate;
408 }
409
410 if (pDevice->byBBType == BB_TYPE_11A) {
411 if (psNodeDBTable->wTxDataRate <= RATE_11M)
412 psNodeDBTable->wTxDataRate = RATE_6M;
413 }
414 DBG_PRT(MSG_LEVEL_DEBUG, KERN_INFO"uTxOk[MAX_RATE] %d, uTxFail[MAX_RATE]:%d\n",(int)psNodeDBTable->uTxOk[MAX_RATE], (int)psNodeDBTable->uTxFail[MAX_RATE]);
415 s_vResetCounter(psNodeDBTable);
416 DBG_PRT(MSG_LEVEL_DEBUG, KERN_INFO"Rate: %d, U:%d, D:%d\n", (int)psNodeDBTable->wTxDataRate, (int)wIdxUpRate, (int)wIdxDownRate);
417 return;
418#else //mike fixed new: use differ-signal strength to control rate
419WORD wIdxUpRate = 0;
420BOOL bAutoRate[MAX_RATE] = {TRUE,TRUE,TRUE,TRUE,FALSE,FALSE,TRUE,TRUE,TRUE,TRUE,TRUE,TRUE};
421UINT ii;
422long ldBm;
423
424 if (pMgmt->eScanState != WMAC_NO_SCANNING) {
425 // Don't do Fallback when scanning Channel
426 return;
427 }
428
429 for(ii=0;ii<MAX_RATE;ii++) {
430 if (psNodeDBTable->wSuppRate & (0x0001<<ii)) {
431 if (bAutoRate[ii] == TRUE) {
432 wIdxUpRate = (WORD) ii;
433 }
434 } else {
435 bAutoRate[ii] = FALSE;
436 }
437 }
438
439 RFvRSSITodBm(pDevice, (BYTE)(pDevice->uCurrRSSI), &ldBm);
440
441 if (ldBm > -55) {
442 if ( psNodeDBTable->wSuppRate & (0x0001<<RATE_54M) ) //11a/g
443 {
444 psNodeDBTable->wTxDataRate = RATE_54M;
445 }
446 else{ //11b
447 psNodeDBTable->wTxDataRate = RATE_11M;
448 }
449 }
450
451if (wIdxUpRate == RATE_54M ) { //11a/g
452 if (ldBm > -56 )
453 psNodeDBTable->wTxDataRate = RATE_54M;
454 else if (ldBm > -61 )
455 psNodeDBTable->wTxDataRate = RATE_48M;
456 else if (ldBm > -66 )
457 psNodeDBTable->wTxDataRate = RATE_36M;
458 else if (ldBm > -72 )
459 psNodeDBTable->wTxDataRate = RATE_24M;
460 else if (ldBm > -80 )
461 psNodeDBTable->wTxDataRate = RATE_5M;
462 else {
463 psNodeDBTable->wTxDataRate = RATE_1M;
464 //increasingVGA = TRUE;
465 }
466 }
467 else { //11b
468 if (ldBm > -65 )
469 psNodeDBTable->wTxDataRate = RATE_11M;
470 else if (ldBm > -75 )
471 psNodeDBTable->wTxDataRate = RATE_5M;
472 else
473 psNodeDBTable->wTxDataRate = RATE_1M;
474 }
475
476 return;
477#endif
478}
479
480/*+
481 *
482 * Description:
483 * This routine is used to assemble available Rate IE.
484 *
485 * Parameters:
486 * In:
487 * pDevice
488 * Out:
489 *
490 * Return Value: None
491 *
492-*/
493BYTE
494RATEuSetIE (
495 IN PWLAN_IE_SUPP_RATES pSrcRates,
496 IN PWLAN_IE_SUPP_RATES pDstRates,
497 IN UINT uRateLen
498 )
499{
500 UINT ii, uu, uRateCnt = 0;
501
502 if ((pSrcRates == NULL) || (pDstRates == NULL))
503 return 0;
504
505 if (pSrcRates->len == 0)
506 return 0;
507
508 for (ii = 0; ii < uRateLen; ii++) {
509 for (uu = 0; uu < pSrcRates->len; uu++) {
510 if ((pSrcRates->abyRates[uu] & 0x7F) == acbyIERate[ii]) {
511 pDstRates->abyRates[uRateCnt ++] = pSrcRates->abyRates[uu];
512 break;
513 }
514 }
515 }
516 return (BYTE)uRateCnt;
517}
518