| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384 |
- /******************************************************************************
- *
- * Copyright(c) 2007 - 2017 Realtek Corporation.
- *
- * 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.
- *
- *****************************************************************************/
- #define _RTW_RF_C_
- #include <drv_types.h>
- #include <hal_data.h>
- u8 center_ch_2g[CENTER_CH_2G_NUM] = {
- /* G00 */1, 2,
- /* G01 */3, 4, 5,
- /* G02 */6, 7, 8,
- /* G03 */9, 10, 11,
- /* G04 */12, 13,
- /* G05 */14
- };
- u8 center_ch_2g_40m[CENTER_CH_2G_40M_NUM] = {
- 3,
- 4,
- 5,
- 6,
- 7,
- 8,
- 9,
- 10,
- 11,
- };
- u8 op_chs_of_cch_2g_40m[CENTER_CH_2G_40M_NUM][2] = {
- {1, 5}, /* 3 */
- {2, 6}, /* 4 */
- {3, 7}, /* 5 */
- {4, 8}, /* 6 */
- {5, 9}, /* 7 */
- {6, 10}, /* 8 */
- {7, 11}, /* 9 */
- {8, 12}, /* 10 */
- {9, 13}, /* 11 */
- };
- u8 center_ch_5g_all[CENTER_CH_5G_ALL_NUM] = {
- /* G00 */36, 38, 40,
- 42,
- /* G01 */44, 46, 48,
- /* 50, */
- /* G02 */52, 54, 56,
- 58,
- /* G03 */60, 62, 64,
- /* G04 */100, 102, 104,
- 106,
- /* G05 */108, 110, 112,
- /* 114, */
- /* G06 */116, 118, 120,
- 122,
- /* G07 */124, 126, 128,
- /* G08 */132, 134, 136,
- 138,
- /* G09 */140, 142, 144,
- /* G10 */149, 151, 153,
- 155,
- /* G11 */157, 159, 161,
- /* 163, */
- /* G12 */165, 167, 169,
- 171,
- /* G13 */173, 175, 177
- };
- u8 center_ch_5g_20m[CENTER_CH_5G_20M_NUM] = {
- /* G00 */36, 40,
- /* G01 */44, 48,
- /* G02 */52, 56,
- /* G03 */60, 64,
- /* G04 */100, 104,
- /* G05 */108, 112,
- /* G06 */116, 120,
- /* G07 */124, 128,
- /* G08 */132, 136,
- /* G09 */140, 144,
- /* G10 */149, 153,
- /* G11 */157, 161,
- /* G12 */165, 169,
- /* G13 */173, 177
- };
- u8 center_ch_5g_40m[CENTER_CH_5G_40M_NUM] = {
- /* G00 */38,
- /* G01 */46,
- /* G02 */54,
- /* G03 */62,
- /* G04 */102,
- /* G05 */110,
- /* G06 */118,
- /* G07 */126,
- /* G08 */134,
- /* G09 */142,
- /* G10 */151,
- /* G11 */159,
- /* G12 */167,
- /* G13 */175
- };
- u8 center_ch_5g_20m_40m[CENTER_CH_5G_20M_NUM + CENTER_CH_5G_40M_NUM] = {
- /* G00 */36, 38, 40,
- /* G01 */44, 46, 48,
- /* G02 */52, 54, 56,
- /* G03 */60, 62, 64,
- /* G04 */100, 102, 104,
- /* G05 */108, 110, 112,
- /* G06 */116, 118, 120,
- /* G07 */124, 126, 128,
- /* G08 */132, 134, 136,
- /* G09 */140, 142, 144,
- /* G10 */149, 151, 153,
- /* G11 */157, 159, 161,
- /* G12 */165, 167, 169,
- /* G13 */173, 175, 177
- };
- u8 op_chs_of_cch_5g_40m[CENTER_CH_5G_40M_NUM][2] = {
- {36, 40}, /* 38 */
- {44, 48}, /* 46 */
- {52, 56}, /* 54 */
- {60, 64}, /* 62 */
- {100, 104}, /* 102 */
- {108, 112}, /* 110 */
- {116, 120}, /* 118 */
- {124, 128}, /* 126 */
- {132, 136}, /* 134 */
- {140, 144}, /* 142 */
- {149, 153}, /* 151 */
- {157, 161}, /* 159 */
- {165, 169}, /* 167 */
- {173, 177}, /* 175 */
- };
- u8 center_ch_5g_80m[CENTER_CH_5G_80M_NUM] = {
- /* G00 ~ G01*/42,
- /* G02 ~ G03*/58,
- /* G04 ~ G05*/106,
- /* G06 ~ G07*/122,
- /* G08 ~ G09*/138,
- /* G10 ~ G11*/155,
- /* G12 ~ G13*/171
- };
- u8 op_chs_of_cch_5g_80m[CENTER_CH_5G_80M_NUM][4] = {
- {36, 40, 44, 48}, /* 42 */
- {52, 56, 60, 64}, /* 58 */
- {100, 104, 108, 112}, /* 106 */
- {116, 120, 124, 128}, /* 122 */
- {132, 136, 140, 144}, /* 138 */
- {149, 153, 157, 161}, /* 155 */
- {165, 169, 173, 177}, /* 171 */
- };
- u8 center_ch_5g_160m[CENTER_CH_5G_160M_NUM] = {
- /* G00 ~ G03*/50,
- /* G04 ~ G07*/114,
- /* G10 ~ G13*/163
- };
- u8 op_chs_of_cch_5g_160m[CENTER_CH_5G_160M_NUM][8] = {
- {36, 40, 44, 48, 52, 56, 60, 64}, /* 50 */
- {100, 104, 108, 112, 116, 120, 124, 128}, /* 114 */
- {149, 153, 157, 161, 165, 169, 173, 177}, /* 163 */
- };
- struct center_chs_ent_t {
- u8 ch_num;
- u8 *chs;
- };
- struct center_chs_ent_t center_chs_2g_by_bw[] = {
- {CENTER_CH_2G_NUM, center_ch_2g},
- {CENTER_CH_2G_40M_NUM, center_ch_2g_40m},
- };
- struct center_chs_ent_t center_chs_5g_by_bw[] = {
- {CENTER_CH_5G_20M_NUM, center_ch_5g_20m},
- {CENTER_CH_5G_40M_NUM, center_ch_5g_40m},
- {CENTER_CH_5G_80M_NUM, center_ch_5g_80m},
- {CENTER_CH_5G_160M_NUM, center_ch_5g_160m},
- };
- /*
- * Get center channel of smaller bandwidth by @param cch, @param bw, @param offset
- * @cch: the given center channel
- * @bw: the given bandwidth
- * @offset: the given primary SC offset of the given bandwidth
- *
- * return center channel of smaller bandiwdth if valid, or 0
- */
- u8 rtw_get_scch_by_cch_offset(u8 cch, u8 bw, u8 offset)
- {
- u8 t_cch = 0;
- if (bw == CHANNEL_WIDTH_20) {
- t_cch = cch;
- goto exit;
- }
- if (offset == HAL_PRIME_CHNL_OFFSET_DONT_CARE) {
- rtw_warn_on(1);
- goto exit;
- }
- /* 2.4G, 40MHz */
- if (cch >= 3 && cch <= 11 && bw == CHANNEL_WIDTH_40) {
- t_cch = (offset == HAL_PRIME_CHNL_OFFSET_UPPER) ? cch + 2 : cch - 2;
- goto exit;
- }
- /* 5G, 160MHz */
- if (cch >= 50 && cch <= 163 && bw == CHANNEL_WIDTH_160) {
- t_cch = (offset == HAL_PRIME_CHNL_OFFSET_UPPER) ? cch + 8 : cch - 8;
- goto exit;
- /* 5G, 80MHz */
- } else if (cch >= 42 && cch <= 171 && bw == CHANNEL_WIDTH_80) {
- t_cch = (offset == HAL_PRIME_CHNL_OFFSET_UPPER) ? cch + 4 : cch - 4;
- goto exit;
- /* 5G, 40MHz */
- } else if (cch >= 38 && cch <= 175 && bw == CHANNEL_WIDTH_40) {
- t_cch = (offset == HAL_PRIME_CHNL_OFFSET_UPPER) ? cch + 2 : cch - 2;
- goto exit;
- } else {
- rtw_warn_on(1);
- goto exit;
- }
- exit:
- return t_cch;
- }
- struct op_chs_ent_t {
- u8 ch_num;
- u8 *chs;
- };
- struct op_chs_ent_t op_chs_of_cch_2g_by_bw[] = {
- {1, center_ch_2g},
- {2, (u8 *)op_chs_of_cch_2g_40m},
- };
- struct op_chs_ent_t op_chs_of_cch_5g_by_bw[] = {
- {1, center_ch_5g_20m},
- {2, (u8 *)op_chs_of_cch_5g_40m},
- {4, (u8 *)op_chs_of_cch_5g_80m},
- {8, (u8 *)op_chs_of_cch_5g_160m},
- };
- inline u8 center_chs_2g_num(u8 bw)
- {
- if (bw > CHANNEL_WIDTH_40)
- return 0;
- return center_chs_2g_by_bw[bw].ch_num;
- }
- inline u8 center_chs_2g(u8 bw, u8 id)
- {
- if (bw > CHANNEL_WIDTH_40)
- return 0;
- if (id >= center_chs_2g_num(bw))
- return 0;
- return center_chs_2g_by_bw[bw].chs[id];
- }
- inline u8 center_chs_5g_num(u8 bw)
- {
- if (bw > CHANNEL_WIDTH_80)
- return 0;
- return center_chs_5g_by_bw[bw].ch_num;
- }
- inline u8 center_chs_5g(u8 bw, u8 id)
- {
- if (bw > CHANNEL_WIDTH_80)
- return 0;
- if (id >= center_chs_5g_num(bw))
- return 0;
- return center_chs_5g_by_bw[bw].chs[id];
- }
- /*
- * Get available op channels by @param cch, @param bw
- * @cch: the given center channel
- * @bw: the given bandwidth
- * @op_chs: the pointer to return pointer of op channel array
- * @op_ch_num: the pointer to return pointer of op channel number
- *
- * return valid (1) or not (0)
- */
- u8 rtw_get_op_chs_by_cch_bw(u8 cch, u8 bw, u8 **op_chs, u8 *op_ch_num)
- {
- int i;
- struct center_chs_ent_t *c_chs_ent = NULL;
- struct op_chs_ent_t *op_chs_ent = NULL;
- u8 valid = 1;
- if (cch <= 14
- && bw >= CHANNEL_WIDTH_20 && bw <= CHANNEL_WIDTH_40
- ) {
- c_chs_ent = ¢er_chs_2g_by_bw[bw];
- op_chs_ent = &op_chs_of_cch_2g_by_bw[bw];
- } else if (cch >= 36 && cch <= 177
- && bw >= CHANNEL_WIDTH_20 && bw <= CHANNEL_WIDTH_160
- ) {
- c_chs_ent = ¢er_chs_5g_by_bw[bw];
- op_chs_ent = &op_chs_of_cch_5g_by_bw[bw];
- } else {
- valid = 0;
- goto exit;
- }
- for (i = 0; i < c_chs_ent->ch_num; i++)
- if (cch == *(c_chs_ent->chs + i))
- break;
- if (i == c_chs_ent->ch_num) {
- valid = 0;
- goto exit;
- }
- *op_chs = op_chs_ent->chs + op_chs_ent->ch_num * i;
- *op_ch_num = op_chs_ent->ch_num;
- exit:
- return valid;
- }
- u8 rtw_get_ch_group(u8 ch, u8 *group, u8 *cck_group)
- {
- BAND_TYPE band = BAND_MAX;
- s8 gp = -1, cck_gp = -1;
- if (ch <= 14) {
- band = BAND_ON_2_4G;
- if (1 <= ch && ch <= 2)
- gp = 0;
- else if (3 <= ch && ch <= 5)
- gp = 1;
- else if (6 <= ch && ch <= 8)
- gp = 2;
- else if (9 <= ch && ch <= 11)
- gp = 3;
- else if (12 <= ch && ch <= 14)
- gp = 4;
- else
- band = BAND_MAX;
- if (ch == 14)
- cck_gp = 5;
- else
- cck_gp = gp;
- } else {
- band = BAND_ON_5G;
- if (36 <= ch && ch <= 42)
- gp = 0;
- else if (44 <= ch && ch <= 48)
- gp = 1;
- else if (50 <= ch && ch <= 58)
- gp = 2;
- else if (60 <= ch && ch <= 64)
- gp = 3;
- else if (100 <= ch && ch <= 106)
- gp = 4;
- else if (108 <= ch && ch <= 114)
- gp = 5;
- else if (116 <= ch && ch <= 122)
- gp = 6;
- else if (124 <= ch && ch <= 130)
- gp = 7;
- else if (132 <= ch && ch <= 138)
- gp = 8;
- else if (140 <= ch && ch <= 144)
- gp = 9;
- else if (149 <= ch && ch <= 155)
- gp = 10;
- else if (157 <= ch && ch <= 161)
- gp = 11;
- else if (165 <= ch && ch <= 171)
- gp = 12;
- else if (173 <= ch && ch <= 177)
- gp = 13;
- else
- band = BAND_MAX;
- }
- if (band == BAND_MAX
- || (band == BAND_ON_2_4G && cck_gp == -1)
- || gp == -1
- ) {
- RTW_WARN("%s invalid channel:%u", __func__, ch);
- rtw_warn_on(1);
- goto exit;
- }
- if (group)
- *group = gp;
- if (cck_group && band == BAND_ON_2_4G)
- *cck_group = cck_gp;
- exit:
- return band;
- }
- int rtw_ch2freq(int chan)
- {
- /* see 802.11 17.3.8.3.2 and Annex J
- * there are overlapping channel numbers in 5GHz and 2GHz bands */
- /*
- * RTK: don't consider the overlapping channel numbers: 5G channel <= 14,
- * because we don't support it. simply judge from channel number
- */
- if (chan >= 1 && chan <= 14) {
- if (chan == 14)
- return 2484;
- else if (chan < 14)
- return 2407 + chan * 5;
- } else if (chan >= 36 && chan <= 177)
- return 5000 + chan * 5;
- return 0; /* not supported */
- }
- int rtw_freq2ch(int freq)
- {
- /* see 802.11 17.3.8.3.2 and Annex J */
- if (freq == 2484)
- return 14;
- else if (freq < 2484)
- return (freq - 2407) / 5;
- else if (freq >= 4910 && freq <= 4980)
- return (freq - 4000) / 5;
- else if (freq <= 45000) /* DMG band lower limit */
- return (freq - 5000) / 5;
- else if (freq >= 58320 && freq <= 64800)
- return (freq - 56160) / 2160;
- else
- return 0;
- }
- bool rtw_chbw_to_freq_range(u8 ch, u8 bw, u8 offset, u32 *hi, u32 *lo)
- {
- u8 c_ch;
- u32 freq;
- u32 hi_ret = 0, lo_ret = 0;
- bool valid = _FALSE;
- if (hi)
- *hi = 0;
- if (lo)
- *lo = 0;
- c_ch = rtw_get_center_ch(ch, bw, offset);
- freq = rtw_ch2freq(c_ch);
- if (!freq) {
- rtw_warn_on(1);
- goto exit;
- }
- if (bw == CHANNEL_WIDTH_80) {
- hi_ret = freq + 40;
- lo_ret = freq - 40;
- } else if (bw == CHANNEL_WIDTH_40) {
- hi_ret = freq + 20;
- lo_ret = freq - 20;
- } else if (bw == CHANNEL_WIDTH_20) {
- hi_ret = freq + 10;
- lo_ret = freq - 10;
- } else
- rtw_warn_on(1);
- if (hi)
- *hi = hi_ret;
- if (lo)
- *lo = lo_ret;
- valid = _TRUE;
- exit:
- return valid;
- }
- const char *const _ch_width_str[CHANNEL_WIDTH_MAX] = {
- "20MHz",
- "40MHz",
- "80MHz",
- "160MHz",
- "80_80MHz",
- "5MHz",
- "10MHz",
- };
- const u8 _ch_width_to_bw_cap[CHANNEL_WIDTH_MAX] = {
- BW_CAP_20M,
- BW_CAP_40M,
- BW_CAP_80M,
- BW_CAP_160M,
- BW_CAP_80_80M,
- BW_CAP_5M,
- BW_CAP_10M,
- };
- const char *const _band_str[] = {
- "2.4G",
- "5G",
- "BOTH",
- "BAND_MAX",
- };
- const u8 _band_to_band_cap[] = {
- BAND_CAP_2G,
- BAND_CAP_5G,
- 0,
- 0,
- };
- const u8 _rf_type_to_rf_tx_cnt[] = {
- 1, /*RF_1T1R*/
- 1, /*RF_1T2R*/
- 2, /*RF_2T2R*/
- 2, /*RF_2T3R*/
- 2, /*RF_2T4R*/
- 3, /*RF_3T3R*/
- 3, /*RF_3T4R*/
- 4, /*RF_4T4R*/
- 1, /*RF_TYPE_MAX*/
- };
- const u8 _rf_type_to_rf_rx_cnt[] = {
- 1, /*RF_1T1R*/
- 2, /*RF_1T2R*/
- 2, /*RF_2T2R*/
- 3, /*RF_2T3R*/
- 4, /*RF_2T4R*/
- 3, /*RF_3T3R*/
- 4, /*RF_3T4R*/
- 4, /*RF_4T4R*/
- 1, /*RF_TYPE_MAX*/
- };
- const char *const _regd_str[] = {
- "NONE",
- "FCC",
- "MKK",
- "ETSI",
- "IC",
- "KCC",
- "ACMA",
- "CHILE",
- "WW",
- };
- #ifdef CONFIG_TXPWR_LIMIT
- void _dump_regd_exc_list(void *sel, struct rf_ctl_t *rfctl)
- {
- struct regd_exc_ent *ent;
- _list *cur, *head;
- RTW_PRINT_SEL(sel, "regd_exc_num:%u\n", rfctl->regd_exc_num);
- if (!rfctl->regd_exc_num)
- goto exit;
- RTW_PRINT_SEL(sel, "%-7s %-6s %-9s\n", "country", "domain", "regd_name");
- head = &rfctl->reg_exc_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- u8 has_country;
- ent = LIST_CONTAINOR(cur, struct regd_exc_ent, list);
- cur = get_next(cur);
- has_country = (ent->country[0] == '\0' && ent->country[1] == '\0') ? 0 : 1;
- RTW_PRINT_SEL(sel, " %c%c 0x%02x %s\n"
- , has_country ? ent->country[0] : '0'
- , has_country ? ent->country[1] : '0'
- , ent->domain
- , ent->regd_name
- );
- }
- exit:
- return;
- }
- inline void dump_regd_exc_list(void *sel, struct rf_ctl_t *rfctl)
- {
- _irqL irqL;
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- _dump_regd_exc_list(sel, rfctl);
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- }
- void rtw_regd_exc_add_with_nlen(struct rf_ctl_t *rfctl, const char *country, u8 domain, const char *regd_name, u32 nlen)
- {
- struct regd_exc_ent *ent;
- _irqL irqL;
- if (!regd_name || !nlen) {
- rtw_warn_on(1);
- goto exit;
- }
- ent = (struct regd_exc_ent *)rtw_zmalloc(sizeof(struct regd_exc_ent) + nlen + 1);
- if (!ent)
- goto exit;
- _rtw_init_listhead(&ent->list);
- if (country)
- _rtw_memcpy(ent->country, country, 2);
- ent->domain = domain;
- _rtw_memcpy(ent->regd_name, regd_name, nlen);
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- rtw_list_insert_tail(&ent->list, &rfctl->reg_exc_list);
- rfctl->regd_exc_num++;
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- exit:
- return;
- }
- inline void rtw_regd_exc_add(struct rf_ctl_t *rfctl, const char *country, u8 domain, const char *regd_name)
- {
- rtw_regd_exc_add_with_nlen(rfctl, country, domain, regd_name, strlen(regd_name));
- }
- struct regd_exc_ent *_rtw_regd_exc_search(struct rf_ctl_t *rfctl, const char *country, u8 domain)
- {
- struct regd_exc_ent *ent;
- _list *cur, *head;
- u8 match = 0;
- head = &rfctl->reg_exc_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- u8 has_country;
- ent = LIST_CONTAINOR(cur, struct regd_exc_ent, list);
- cur = get_next(cur);
- has_country = (ent->country[0] == '\0' && ent->country[1] == '\0') ? 0 : 1;
- /* entry has country condition to match */
- if (has_country) {
- if (!country)
- continue;
- if (ent->country[0] != country[0]
- || ent->country[1] != country[1])
- continue;
- }
- /* entry has domain condition to match */
- if (ent->domain != 0xFF) {
- if (domain == 0xFF)
- continue;
- if (ent->domain != domain)
- continue;
- }
- match = 1;
- break;
- }
- if (match)
- return ent;
- else
- return NULL;
- }
- inline struct regd_exc_ent *rtw_regd_exc_search(struct rf_ctl_t *rfctl, const char *country, u8 domain)
- {
- struct regd_exc_ent *ent;
- _irqL irqL;
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- ent = _rtw_regd_exc_search(rfctl, country, domain);
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- return ent;
- }
- void rtw_regd_exc_list_free(struct rf_ctl_t *rfctl)
- {
- struct regd_exc_ent *ent;
- _irqL irqL;
- _list *cur, *head;
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- head = &rfctl->reg_exc_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct regd_exc_ent, list);
- cur = get_next(cur);
- rtw_list_delete(&ent->list);
- rtw_mfree((u8 *)ent, sizeof(struct regd_exc_ent) + strlen(ent->regd_name) + 1);
- }
- rfctl->regd_exc_num = 0;
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- }
- void dump_txpwr_lmt(void *sel, _adapter *adapter)
- {
- #define TMP_STR_LEN 16
- struct rf_ctl_t *rfctl = adapter_to_rfctl(adapter);
- struct hal_spec_t *hal_spec = GET_HAL_SPEC(adapter);
- _irqL irqL;
- char fmt[16];
- char tmp_str[TMP_STR_LEN];
- s8 *lmt_idx = NULL;
- int bw, band, ch_num, tlrs, ntx_idx, rs, i, path;
- u8 ch, n, rfpath_num;
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- _dump_regd_exc_list(sel, rfctl);
- RTW_PRINT_SEL(sel, "\n");
- if (!rfctl->txpwr_regd_num)
- goto release_lock;
- lmt_idx = rtw_malloc(sizeof(s8) * RF_PATH_MAX * rfctl->txpwr_regd_num);
- if (!lmt_idx) {
- RTW_ERR("%s alloc fail\n", __func__);
- goto release_lock;
- }
- RTW_PRINT_SEL(sel, "txpwr_lmt_2g_cck_ofdm_state:0x%02x\n", rfctl->txpwr_lmt_2g_cck_ofdm_state);
- #ifdef CONFIG_IEEE80211_BAND_5GHZ
- if (IS_HARDWARE_TYPE_JAGUAR_AND_JAGUAR2(adapter))
- RTW_PRINT_SEL(sel, "txpwr_lmt_5g_cck_ofdm_state:0x%02x\n", rfctl->txpwr_lmt_5g_cck_ofdm_state);
- RTW_PRINT_SEL(sel, "txpwr_lmt_5g_20_40_ref:0x%02x\n", rfctl->txpwr_lmt_5g_20_40_ref);
- #endif
- RTW_PRINT_SEL(sel, "\n");
- for (band = BAND_ON_2_4G; band <= BAND_ON_5G; band++) {
- if (!hal_is_band_support(adapter, band))
- continue;
- rfpath_num = (band == BAND_ON_2_4G ? hal_spec->rfpath_num_2g : hal_spec->rfpath_num_5g);
- for (bw = 0; bw < MAX_5G_BANDWIDTH_NUM; bw++) {
- if (bw >= CHANNEL_WIDTH_160)
- break;
- if (band == BAND_ON_2_4G && bw >= CHANNEL_WIDTH_80)
- break;
- if (band == BAND_ON_2_4G)
- ch_num = CENTER_CH_2G_NUM;
- else
- ch_num = center_chs_5g_num(bw);
- if (ch_num == 0) {
- rtw_warn_on(1);
- break;
- }
- for (tlrs = TXPWR_LMT_RS_CCK; tlrs < TXPWR_LMT_RS_NUM; tlrs++) {
- if (band == BAND_ON_2_4G && tlrs == TXPWR_LMT_RS_VHT)
- continue;
- if (band == BAND_ON_5G && tlrs == TXPWR_LMT_RS_CCK)
- continue;
- if (bw > CHANNEL_WIDTH_20 && (tlrs == TXPWR_LMT_RS_CCK || tlrs == TXPWR_LMT_RS_OFDM))
- continue;
- if (bw > CHANNEL_WIDTH_40 && tlrs == TXPWR_LMT_RS_HT)
- continue;
- if (tlrs == TXPWR_LMT_RS_VHT && !IS_HARDWARE_TYPE_JAGUAR_AND_JAGUAR2(adapter))
- continue;
- for (ntx_idx = RF_1TX; ntx_idx < MAX_TX_COUNT; ntx_idx++) {
- struct txpwr_lmt_ent *ent;
- _list *cur, *head;
- if (ntx_idx >= hal_spec->tx_nss_num)
- continue;
- /* bypass CCK multi-TX is not defined */
- if (tlrs == TXPWR_LMT_RS_CCK && ntx_idx > RF_1TX) {
- if (band == BAND_ON_2_4G
- && !(rfctl->txpwr_lmt_2g_cck_ofdm_state & (TXPWR_LMT_HAS_CCK_1T << ntx_idx)))
- continue;
- }
- /* bypass OFDM multi-TX is not defined */
- if (tlrs == TXPWR_LMT_RS_OFDM && ntx_idx > RF_1TX) {
- if (band == BAND_ON_2_4G
- && !(rfctl->txpwr_lmt_2g_cck_ofdm_state & (TXPWR_LMT_HAS_OFDM_1T << ntx_idx)))
- continue;
- #ifdef CONFIG_IEEE80211_BAND_5GHZ
- if (band == BAND_ON_5G
- && !(rfctl->txpwr_lmt_5g_cck_ofdm_state & (TXPWR_LMT_HAS_OFDM_1T << ntx_idx)))
- continue;
- #endif
- }
- /* bypass 5G 20M, 40M pure reference */
- #ifdef CONFIG_IEEE80211_BAND_5GHZ
- if (band == BAND_ON_5G && (bw == CHANNEL_WIDTH_20 || bw == CHANNEL_WIDTH_40)) {
- if (rfctl->txpwr_lmt_5g_20_40_ref == TXPWR_LMT_REF_HT_FROM_VHT) {
- if (tlrs == TXPWR_LMT_RS_HT)
- continue;
- } else if (rfctl->txpwr_lmt_5g_20_40_ref == TXPWR_LMT_REF_VHT_FROM_HT) {
- if (tlrs == TXPWR_LMT_RS_VHT && bw <= CHANNEL_WIDTH_40)
- continue;
- }
- }
- #endif
- /* choose n-SS mapping rate section to get lmt diff value */
- if (tlrs == TXPWR_LMT_RS_CCK)
- rs = CCK;
- else if (tlrs == TXPWR_LMT_RS_OFDM)
- rs = OFDM;
- else if (tlrs == TXPWR_LMT_RS_HT)
- rs = HT_1SS + ntx_idx;
- else if (tlrs == TXPWR_LMT_RS_VHT)
- rs = VHT_1SS + ntx_idx;
- else {
- RTW_ERR("%s invalid tlrs %u\n", __func__, tlrs);
- continue;
- }
- RTW_PRINT_SEL(sel, "[%s][%s][%s][%uT]\n"
- , band_str(band)
- , ch_width_str(bw)
- , txpwr_lmt_rs_str(tlrs)
- , ntx_idx + 1
- );
- /* header for limit in db */
- RTW_PRINT_SEL(sel, "%3s ", "ch");
- head = &rfctl->txpwr_lmt_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct txpwr_lmt_ent, list);
- cur = get_next(cur);
- sprintf(fmt, "%%%zus%%s ", strlen(ent->regd_name) >= 6 ? 1 : 6 - strlen(ent->regd_name));
- snprintf(tmp_str, TMP_STR_LEN, fmt
- , strcmp(ent->regd_name, rfctl->regd_name) == 0 ? "*" : ""
- , ent->regd_name);
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- }
- sprintf(fmt, "%%%zus%%s ", strlen(regd_str(TXPWR_LMT_WW)) >= 6 ? 1 : 6 - strlen(regd_str(TXPWR_LMT_WW)));
- snprintf(tmp_str, TMP_STR_LEN, fmt
- , strcmp(rfctl->regd_name, regd_str(TXPWR_LMT_WW)) == 0 ? "*" : ""
- , regd_str(TXPWR_LMT_WW));
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- /* header for limit offset */
- for (path = 0; path < RF_PATH_MAX; path++) {
- if (path >= rfpath_num)
- break;
- _RTW_PRINT_SEL(sel, "|");
- head = &rfctl->txpwr_lmt_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct txpwr_lmt_ent, list);
- cur = get_next(cur);
- _RTW_PRINT_SEL(sel, "%3c "
- , strcmp(ent->regd_name, rfctl->regd_name) == 0 ? rf_path_char(path) : ' ');
- }
- _RTW_PRINT_SEL(sel, "%3c "
- , strcmp(rfctl->regd_name, regd_str(TXPWR_LMT_WW)) == 0 ? rf_path_char(path) : ' ');
- }
- _RTW_PRINT_SEL(sel, "\n");
- for (n = 0; n < ch_num; n++) {
- s8 lmt;
- s8 lmt_offset;
- u8 base;
- if (band == BAND_ON_2_4G)
- ch = n + 1;
- else
- ch = center_chs_5g(bw, n);
- if (ch == 0) {
- rtw_warn_on(1);
- break;
- }
- /* dump limit in db */
- RTW_PRINT_SEL(sel, "%3u ", ch);
- head = &rfctl->txpwr_lmt_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct txpwr_lmt_ent, list);
- cur = get_next(cur);
- lmt = phy_get_txpwr_lmt_abs(adapter, ent->regd_name, band, bw, tlrs, ntx_idx, ch, 0);
- if (lmt == hal_spec->txgi_max) {
- sprintf(fmt, "%%%zus ", strlen(ent->regd_name) >= 6 ? strlen(ent->regd_name) + 1 : 6);
- snprintf(tmp_str, TMP_STR_LEN, fmt, "NA");
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- } else if (lmt > -hal_spec->txgi_pdbm && lmt < 0) { /* -0.xx */
- sprintf(fmt, "%%%zus-0.%%d ", strlen(ent->regd_name) >= 6 ? strlen(ent->regd_name) - 4 : 1);
- snprintf(tmp_str, TMP_STR_LEN, fmt, "", (rtw_abs(lmt) % hal_spec->txgi_pdbm) * 100 / hal_spec->txgi_pdbm);
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- } else if (lmt % hal_spec->txgi_pdbm) { /* d.xx */
- sprintf(fmt, "%%%zud.%%d ", strlen(ent->regd_name) >= 6 ? strlen(ent->regd_name) - 2 : 3);
- snprintf(tmp_str, TMP_STR_LEN, fmt, lmt / hal_spec->txgi_pdbm, (rtw_abs(lmt) % hal_spec->txgi_pdbm) * 100 / hal_spec->txgi_pdbm);
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- } else { /* d */
- sprintf(fmt, "%%%zud ", strlen(ent->regd_name) >= 6 ? strlen(ent->regd_name) + 1 : 6);
- snprintf(tmp_str, TMP_STR_LEN, fmt, lmt / hal_spec->txgi_pdbm);
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- }
- }
- lmt = phy_get_txpwr_lmt_abs(adapter, regd_str(TXPWR_LMT_WW), band, bw, tlrs, ntx_idx, ch, 0);
- if (lmt == hal_spec->txgi_max) {
- sprintf(fmt, "%%%zus ", strlen(regd_str(TXPWR_LMT_WW)) >= 6 ? strlen(regd_str(TXPWR_LMT_WW)) + 1 : 6);
- snprintf(tmp_str, TMP_STR_LEN, fmt, "NA");
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- } else if (lmt > -hal_spec->txgi_pdbm && lmt < 0) { /* -0.xx */
- sprintf(fmt, "%%%zus-0.%%d ", strlen(regd_str(TXPWR_LMT_WW)) >= 6 ? strlen(regd_str(TXPWR_LMT_WW)) - 4 : 1);
- snprintf(tmp_str, TMP_STR_LEN, fmt, "", (rtw_abs(lmt) % hal_spec->txgi_pdbm) * 100 / hal_spec->txgi_pdbm);
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- } else if (lmt % hal_spec->txgi_pdbm) { /* d.xx */
- sprintf(fmt, "%%%zud.%%d ", strlen(regd_str(TXPWR_LMT_WW)) >= 6 ? strlen(regd_str(TXPWR_LMT_WW)) - 2 : 3);
- snprintf(tmp_str, TMP_STR_LEN, fmt, lmt / hal_spec->txgi_pdbm, (rtw_abs(lmt) % hal_spec->txgi_pdbm) * 100 / hal_spec->txgi_pdbm);
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- } else { /* d */
- sprintf(fmt, "%%%zud ", strlen(regd_str(TXPWR_LMT_WW)) >= 6 ? strlen(regd_str(TXPWR_LMT_WW)) + 1 : 6);
- snprintf(tmp_str, TMP_STR_LEN, fmt, lmt / hal_spec->txgi_pdbm);
- _RTW_PRINT_SEL(sel, "%s", tmp_str);
- }
- /* dump limit offset of each path */
- for (path = RF_PATH_A; path < RF_PATH_MAX; path++) {
- if (path >= rfpath_num)
- break;
- base = PHY_GetTxPowerByRateBase(adapter, band, path, rs);
- _RTW_PRINT_SEL(sel, "|");
- head = &rfctl->txpwr_lmt_list;
- cur = get_next(head);
- i = 0;
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct txpwr_lmt_ent, list);
- cur = get_next(cur);
- lmt_offset = phy_get_txpwr_lmt(adapter, ent->regd_name, band, bw, path, rs, ntx_idx, ch, 0);
- if (lmt_offset == hal_spec->txgi_max) {
- *(lmt_idx + i * RF_PATH_MAX + path) = hal_spec->txgi_max;
- _RTW_PRINT_SEL(sel, "%3s ", "NA");
- } else {
- *(lmt_idx + i * RF_PATH_MAX + path) = lmt_offset + base;
- _RTW_PRINT_SEL(sel, "%3d ", lmt_offset);
- }
- i++;
- }
- lmt_offset = phy_get_txpwr_lmt(adapter, regd_str(TXPWR_LMT_WW), band, bw, path, rs, ntx_idx, ch, 0);
- if (lmt_offset == hal_spec->txgi_max)
- _RTW_PRINT_SEL(sel, "%3s ", "NA");
- else
- _RTW_PRINT_SEL(sel, "%3d ", lmt_offset);
- }
- /* compare limit_idx of each path, print 'x' when mismatch */
- if (rfpath_num > 1) {
- for (i = 0; i < rfctl->txpwr_regd_num; i++) {
- for (path = 0; path < RF_PATH_MAX; path++) {
- if (path >= rfpath_num)
- break;
- if (*(lmt_idx + i * RF_PATH_MAX + path) != *(lmt_idx + i * RF_PATH_MAX + ((path + 1) % rfpath_num)))
- break;
- }
- if (path >= rfpath_num)
- _RTW_PRINT_SEL(sel, " ");
- else
- _RTW_PRINT_SEL(sel, "x");
- }
- }
- _RTW_PRINT_SEL(sel, "\n");
- }
- RTW_PRINT_SEL(sel, "\n");
- }
- } /* loop for rate sections */
- } /* loop for bandwidths */
- } /* loop for bands */
- if (lmt_idx)
- rtw_mfree(lmt_idx, sizeof(s8) * RF_PATH_MAX * rfctl->txpwr_regd_num);
- release_lock:
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- }
- /* search matcing first, if not found, alloc one */
- void rtw_txpwr_lmt_add_with_nlen(struct rf_ctl_t *rfctl, const char *regd_name, u32 nlen
- , u8 band, u8 bw, u8 tlrs, u8 ntx_idx, u8 ch_idx, s8 lmt)
- {
- struct hal_spec_t *hal_spec = GET_HAL_SPEC(dvobj_get_primary_adapter(rfctl_to_dvobj(rfctl)));
- struct txpwr_lmt_ent *ent;
- _irqL irqL;
- _list *cur, *head;
- s8 pre_lmt;
- if (!regd_name || !nlen) {
- rtw_warn_on(1);
- goto exit;
- }
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- /* search for existed entry */
- head = &rfctl->txpwr_lmt_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct txpwr_lmt_ent, list);
- cur = get_next(cur);
- if (strlen(ent->regd_name) == nlen
- && _rtw_memcmp(ent->regd_name, regd_name, nlen) == _TRUE)
- goto chk_lmt_val;
- }
- /* alloc new one */
- ent = (struct txpwr_lmt_ent *)rtw_zvmalloc(sizeof(struct txpwr_lmt_ent) + nlen + 1);
- if (!ent)
- goto release_lock;
- _rtw_init_listhead(&ent->list);
- _rtw_memcpy(ent->regd_name, regd_name, nlen);
- {
- u8 j, k, l, m;
- for (j = 0; j < MAX_2_4G_BANDWIDTH_NUM; ++j)
- for (k = 0; k < TXPWR_LMT_RS_NUM_2G; ++k)
- for (m = 0; m < CENTER_CH_2G_NUM; ++m)
- for (l = 0; l < MAX_TX_COUNT; ++l)
- ent->lmt_2g[j][k][m][l] = hal_spec->txgi_max;
- #ifdef CONFIG_IEEE80211_BAND_5GHZ
- for (j = 0; j < MAX_5G_BANDWIDTH_NUM; ++j)
- for (k = 0; k < TXPWR_LMT_RS_NUM_5G; ++k)
- for (m = 0; m < CENTER_CH_5G_ALL_NUM; ++m)
- for (l = 0; l < MAX_TX_COUNT; ++l)
- ent->lmt_5g[j][k][m][l] = hal_spec->txgi_max;
- #endif
- }
- rtw_list_insert_tail(&ent->list, &rfctl->txpwr_lmt_list);
- rfctl->txpwr_regd_num++;
- chk_lmt_val:
- if (band == BAND_ON_2_4G)
- pre_lmt = ent->lmt_2g[bw][tlrs][ch_idx][ntx_idx];
- #ifdef CONFIG_IEEE80211_BAND_5GHZ
- else if (band == BAND_ON_5G)
- pre_lmt = ent->lmt_5g[bw][tlrs - 1][ch_idx][ntx_idx];
- #endif
- else
- goto release_lock;
- if (pre_lmt != hal_spec->txgi_max)
- RTW_PRINT("duplicate txpwr_lmt for [%s][%s][%s][%s][%uT][%d]\n"
- , regd_name, band_str(band), ch_width_str(bw), txpwr_lmt_rs_str(tlrs), ntx_idx + 1
- , band == BAND_ON_2_4G ? ch_idx + 1 : center_ch_5g_all[ch_idx]);
- lmt = rtw_min(pre_lmt, lmt);
- if (band == BAND_ON_2_4G)
- ent->lmt_2g[bw][tlrs][ch_idx][ntx_idx] = lmt;
- #ifdef CONFIG_IEEE80211_BAND_5GHZ
- else if (band == BAND_ON_5G)
- ent->lmt_5g[bw][tlrs - 1][ch_idx][ntx_idx] = lmt;
- #endif
- if (0)
- RTW_PRINT("%s, %4s, %6s, %7s, %uT, ch%3d = %d\n"
- , regd_name, band_str(band), ch_width_str(bw), txpwr_lmt_rs_str(tlrs), ntx_idx + 1
- , band == BAND_ON_2_4G ? ch_idx + 1 : center_ch_5g_all[ch_idx]
- , lmt);
- release_lock:
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- exit:
- return;
- }
- inline void rtw_txpwr_lmt_add(struct rf_ctl_t *rfctl, const char *regd_name
- , u8 band, u8 bw, u8 tlrs, u8 ntx_idx, u8 ch_idx, s8 lmt)
- {
- rtw_txpwr_lmt_add_with_nlen(rfctl, regd_name, strlen(regd_name)
- , band, bw, tlrs, ntx_idx, ch_idx, lmt);
- }
- struct txpwr_lmt_ent *_rtw_txpwr_lmt_get_by_name(struct rf_ctl_t *rfctl, const char *regd_name)
- {
- struct txpwr_lmt_ent *ent;
- _list *cur, *head;
- u8 found = 0;
- head = &rfctl->txpwr_lmt_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct txpwr_lmt_ent, list);
- cur = get_next(cur);
- if (strcmp(ent->regd_name, regd_name) == 0) {
- found = 1;
- break;
- }
- }
- if (found)
- return ent;
- return NULL;
- }
- inline struct txpwr_lmt_ent *rtw_txpwr_lmt_get_by_name(struct rf_ctl_t *rfctl, const char *regd_name)
- {
- struct txpwr_lmt_ent *ent;
- _irqL irqL;
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- ent = _rtw_txpwr_lmt_get_by_name(rfctl, regd_name);
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- return ent;
- }
- void rtw_txpwr_lmt_list_free(struct rf_ctl_t *rfctl)
- {
- struct txpwr_lmt_ent *ent;
- _irqL irqL;
- _list *cur, *head;
- _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- head = &rfctl->txpwr_lmt_list;
- cur = get_next(head);
- while ((rtw_end_of_queue_search(head, cur)) == _FALSE) {
- ent = LIST_CONTAINOR(cur, struct txpwr_lmt_ent, list);
- cur = get_next(cur);
- if (ent->regd_name == rfctl->regd_name)
- rfctl->regd_name = regd_str(TXPWR_LMT_NONE);
- rtw_list_delete(&ent->list);
- rtw_vmfree((u8 *)ent, sizeof(struct txpwr_lmt_ent) + strlen(ent->regd_name) + 1);
- }
- rfctl->txpwr_regd_num = 0;
- _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
- }
- #endif /* CONFIG_TXPWR_LIMIT */
- int rtw_ch_to_bb_gain_sel(int ch)
- {
- int sel = -1;
- if (ch >= 1 && ch <= 14)
- sel = BB_GAIN_2G;
- #ifdef CONFIG_IEEE80211_BAND_5GHZ
- else if (ch >= 36 && ch < 48)
- sel = BB_GAIN_5GLB1;
- else if (ch >= 52 && ch <= 64)
- sel = BB_GAIN_5GLB2;
- else if (ch >= 100 && ch <= 120)
- sel = BB_GAIN_5GMB1;
- else if (ch >= 124 && ch <= 144)
- sel = BB_GAIN_5GMB2;
- else if (ch >= 149 && ch <= 177)
- sel = BB_GAIN_5GHB;
- #endif
- return sel;
- }
- s8 rtw_rf_get_kfree_tx_gain_offset(_adapter *padapter, u8 path, u8 ch)
- {
- s8 kfree_offset = 0;
- #ifdef CONFIG_RF_POWER_TRIM
- struct kfree_data_t *kfree_data = GET_KFREE_DATA(padapter);
- s8 bb_gain_sel = rtw_ch_to_bb_gain_sel(ch);
- if (bb_gain_sel < BB_GAIN_2G || bb_gain_sel >= BB_GAIN_NUM) {
- rtw_warn_on(1);
- goto exit;
- }
- if (kfree_data->flag & KFREE_FLAG_ON) {
- kfree_offset = kfree_data->bb_gain[bb_gain_sel][path];
- if (IS_HARDWARE_TYPE_8723D(padapter))
- RTW_INFO("%s path:%s, ch:%u, bb_gain_sel:%d, kfree_offset:%d\n"
- , __func__, (path == 0)?"S1":"S0",
- ch, bb_gain_sel, kfree_offset);
- else
- RTW_INFO("%s path:%u, ch:%u, bb_gain_sel:%d, kfree_offset:%d\n"
- , __func__, path, ch, bb_gain_sel, kfree_offset);
- }
- exit:
- #endif /* CONFIG_RF_POWER_TRIM */
- return kfree_offset;
- }
- void rtw_rf_set_tx_gain_offset(_adapter *adapter, u8 path, s8 offset)
- {
- #if !defined(CONFIG_RTL8814A) && !defined(CONFIG_RTL8822B) && !defined(CONFIG_RTL8821C)
- u8 write_value;
- #endif
- u8 target_path = 0;
- u32 val32 = 0;
- if (IS_HARDWARE_TYPE_8723D(adapter)) {
- target_path = RF_PATH_A; /*in 8723D case path means S0/S1*/
- if (path == PPG_8723D_S1)
- RTW_INFO("kfree gain_offset 0x55:0x%x ",
- rtw_hal_read_rfreg(adapter, target_path, 0x55, 0xffffffff));
- else if (path == PPG_8723D_S0)
- RTW_INFO("kfree gain_offset 0x65:0x%x ",
- rtw_hal_read_rfreg(adapter, target_path, 0x65, 0xffffffff));
- } else {
- target_path = path;
- RTW_INFO("kfree gain_offset 0x55:0x%x ", rtw_hal_read_rfreg(adapter, target_path, 0x55, 0xffffffff));
- }
- switch (rtw_get_chip_type(adapter)) {
- #ifdef CONFIG_RTL8723D
- case RTL8723D:
- write_value = RF_TX_GAIN_OFFSET_8723D(offset);
- if (path == PPG_8723D_S1)
- rtw_hal_write_rfreg(adapter, target_path, 0x55, 0x0f8000, write_value);
- else if (path == PPG_8723D_S0)
- rtw_hal_write_rfreg(adapter, target_path, 0x65, 0x0f8000, write_value);
- break;
- #endif /* CONFIG_RTL8723D */
- #ifdef CONFIG_RTL8703B
- case RTL8703B:
- write_value = RF_TX_GAIN_OFFSET_8703B(offset);
- rtw_hal_write_rfreg(adapter, target_path, 0x55, 0x0fc000, write_value);
- break;
- #endif /* CONFIG_RTL8703B */
- #ifdef CONFIG_RTL8188F
- case RTL8188F:
- write_value = RF_TX_GAIN_OFFSET_8188F(offset);
- rtw_hal_write_rfreg(adapter, target_path, 0x55, 0x0fc000, write_value);
- break;
- #endif /* CONFIG_RTL8188F */
- #ifdef CONFIG_RTL8188GTV
- case RTL8188GTV:
- write_value = RF_TX_GAIN_OFFSET_8188GTV(offset);
- rtw_hal_write_rfreg(adapter, target_path, 0x55, 0x0fc000, write_value);
- break;
- #endif /* CONFIG_RTL8188GTV */
- #ifdef CONFIG_RTL8192E
- case RTL8192E:
- write_value = RF_TX_GAIN_OFFSET_8192E(offset);
- rtw_hal_write_rfreg(adapter, target_path, 0x55, 0x0f8000, write_value);
- break;
- #endif /* CONFIG_RTL8188F */
- #ifdef CONFIG_RTL8821A
- case RTL8821:
- write_value = RF_TX_GAIN_OFFSET_8821A(offset);
- rtw_hal_write_rfreg(adapter, target_path, 0x55, 0x0f8000, write_value);
- break;
- #endif /* CONFIG_RTL8821A */
- #if defined(CONFIG_RTL8814A) || defined(CONFIG_RTL8822B) || defined(CONFIG_RTL8821C) || defined(CONFIG_RTL8192F)
- case RTL8814A:
- case RTL8822B:
- case RTL8821C:
- case RTL8192F:
- RTW_INFO("\nkfree by PhyDM on the sw CH. path %d\n", path);
- break;
- #endif /* CONFIG_RTL8814A || CONFIG_RTL8822B || CONFIG_RTL8821C */
- default:
- rtw_warn_on(1);
- break;
- }
- if (IS_HARDWARE_TYPE_8723D(adapter)) {
- if (path == PPG_8723D_S1)
- val32 = rtw_hal_read_rfreg(adapter, target_path, 0x55, 0xffffffff);
- else if (path == PPG_8723D_S0)
- val32 = rtw_hal_read_rfreg(adapter, target_path, 0x65, 0xffffffff);
- } else {
- val32 = rtw_hal_read_rfreg(adapter, target_path, 0x55, 0xffffffff);
- }
- RTW_INFO(" after :0x%x\n", val32);
- }
- void rtw_rf_apply_tx_gain_offset(_adapter *adapter, u8 ch)
- {
- HAL_DATA_TYPE *hal_data = GET_HAL_DATA(adapter);
- s8 kfree_offset = 0;
- s8 tx_pwr_track_offset = 0; /* TODO: 8814A should consider tx pwr track when setting tx gain offset */
- s8 total_offset;
- int i, total = 0;
- if (IS_HARDWARE_TYPE_8723D(adapter))
- total = 2; /* S1 and S0 */
- else
- total = hal_data->NumTotalRFPath;
- for (i = 0; i < total; i++) {
- kfree_offset = rtw_rf_get_kfree_tx_gain_offset(adapter, i, ch);
- total_offset = kfree_offset + tx_pwr_track_offset;
- rtw_rf_set_tx_gain_offset(adapter, i, total_offset);
- }
- }
- inline u8 rtw_is_dfs_range(u32 hi, u32 lo)
- {
- return rtw_is_range_overlap(hi, lo, 5720 + 10, 5260 - 10);
- }
- u8 rtw_is_dfs_ch(u8 ch)
- {
- u32 hi, lo;
- if (!rtw_chbw_to_freq_range(ch, CHANNEL_WIDTH_20, HAL_PRIME_CHNL_OFFSET_DONT_CARE, &hi, &lo))
- return 0;
- return rtw_is_dfs_range(hi, lo);
- }
- u8 rtw_is_dfs_chbw(u8 ch, u8 bw, u8 offset)
- {
- u32 hi, lo;
- if (!rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo))
- return 0;
- return rtw_is_dfs_range(hi, lo);
- }
- bool rtw_is_long_cac_range(u32 hi, u32 lo, u8 dfs_region)
- {
- return (dfs_region == PHYDM_DFS_DOMAIN_ETSI && rtw_is_range_overlap(hi, lo, 5650, 5600)) ? _TRUE : _FALSE;
- }
- bool rtw_is_long_cac_ch(u8 ch, u8 bw, u8 offset, u8 dfs_region)
- {
- u32 hi, lo;
- if (rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo) == _FALSE)
- return _FALSE;
- return rtw_is_long_cac_range(hi, lo, dfs_region) ? _TRUE : _FALSE;
- }
|