Files
wifi-rtl8852bs/core/rtw_rf.c
2024-03-01 19:54:35 +08:00

2273 lines
54 KiB
C

/******************************************************************************
*
* Copyright(c) 2007 - 2019 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>
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
};
#define ch_to_cch_2g_idx(ch) ((ch) - 1)
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 */
};
#if CONFIG_IEEE80211_BAND_5GHZ
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
};
#define ch_to_cch_5g_20m_idx(ch) \
( \
((ch) >= 36 && (ch) <= 64) ? (((ch) - 36) >> 2) : \
((ch) >= 100 && (ch) <= 144) ? 8 + (((ch) - 100) >> 2) : \
((ch) >= 149 && (ch) <= 177) ? 20 + (((ch) - 149) >> 2) : 255 \
)
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 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 */
};
#endif /* CONFIG_IEEE80211_BAND_5GHZ */
#if CONFIG_IEEE80211_BAND_6GHZ
u8 center_ch_6g_20m[CENTER_CH_6G_20M_NUM] = {
1, 5, 9, 13, 17, 21, 25, 29,
33, 37, 41, 45, 49, 53, 57, 61,
65, 69, 73, 77, 81, 85, 89, 93,
97, 101, 105, 109, 113, 117, 121, 125,
129, 133, 137, 141, 145, 149, 153, 157,
161, 165, 169, 173, 177, 181, 185, 189,
193, 197, 201, 205, 209, 213, 217, 221,
225, 229, 233, 237, 241, 245, 249, 253,
};
u8 center_ch_6g_40m[CENTER_CH_6G_40M_NUM] = {
3, 11, 19, 27,
35, 43, 51, 59,
67, 75, 83, 91,
99, 107, 115, 123,
131, 139, 147, 155,
163, 171, 179, 187,
195, 203, 211, 219,
227, 235, 243, 251,
};
u8 op_chs_of_cch_6g_40m[CENTER_CH_6G_40M_NUM][2] = {
{1, 5}, /* 3 */
{9, 13}, /* 11 */
{17, 21}, /* 19 */
{25, 29}, /* 27 */
{33, 37}, /* 35 */
{41, 45}, /* 43 */
{49, 53}, /* 51 */
{57, 61}, /* 59 */
{65, 69}, /* 67 */
{73, 77}, /* 75 */
{81, 85}, /* 83 */
{89, 93}, /* 91 */
{97, 101}, /* 99 */
{105, 109}, /* 107 */
{113, 117}, /* 115 */
{121, 125}, /* 123 */
{129, 133}, /* 131 */
{137, 141}, /* 139 */
{145, 149}, /* 147 */
{153, 157}, /* 155 */
{161, 165}, /* 163 */
{169, 173}, /* 171 */
{177, 181}, /* 179 */
{185, 189}, /* 187 */
{193, 197}, /* 195 */
{201, 205}, /* 203 */
{209, 213}, /* 211 */
{217, 221}, /* 219 */
{225, 229}, /* 227 */
{233, 237}, /* 235 */
{241, 245}, /* 243 */
{249, 253}, /* 251 */
};
u8 center_ch_6g_80m[CENTER_CH_6G_80M_NUM] = {
7, 23,
39, 55,
71, 87,
103, 119,
135, 151,
167, 183,
199, 215,
231, 247,
};
u8 op_chs_of_cch_6g_80m[CENTER_CH_6G_80M_NUM][4] = {
{1, 5, 9, 13}, /* 7 */
{17, 21, 25, 29}, /* 23 */
{33, 37, 41, 45}, /* 39 */
{49, 53, 57, 61}, /* 55 */
{65, 69, 73, 77}, /* 71 */
{81, 85, 89, 93}, /* 87 */
{97, 101, 105, 109}, /* 103 */
{113, 117, 121, 125}, /* 119 */
{129, 133, 137, 141}, /* 135 */
{145, 149, 153, 157}, /* 151 */
{161, 165, 169, 173}, /* 167 */
{177, 181, 185, 189}, /* 183 */
{193, 197, 201, 205}, /* 199 */
{209, 213, 217, 221}, /* 215 */
{225, 229, 233, 237}, /* 231 */
{241, 245, 249, 253}, /* 247 */
};
u8 center_ch_6g_160m[CENTER_CH_6G_160M_NUM] = {
15,
47,
79,
111,
143,
175,
207,
239,
};
u8 op_chs_of_cch_6g_160m[CENTER_CH_6G_160M_NUM][8] = {
{1, 5, 9, 13, 17, 21, 25, 29}, /* 15 */
{33, 37, 41, 45, 49, 53, 57, 61}, /* 47 */
{65, 69, 73, 77, 81, 85, 89, 93}, /* 79 */
{97, 101, 105, 109, 113, 117, 121, 125}, /* 111 */
{129, 133, 137, 141, 145, 149, 153, 157}, /* 143 */
{161, 165, 169, 173, 177, 181, 185, 189}, /* 175 */
{193, 197, 201, 205, 209, 213, 217, 221}, /* 207 */
{225, 229, 233, 237, 241, 245, 249, 253}, /* 239 */
};
#endif /* CONFIG_IEEE80211_BAND_6GHZ */
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},
};
#if CONFIG_IEEE80211_BAND_5GHZ
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},
};
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
struct center_chs_ent_t center_chs_6g_by_bw[] = {
{CENTER_CH_6G_20M_NUM, center_ch_6g_20m},
{CENTER_CH_6G_40M_NUM, center_ch_6g_40m},
{CENTER_CH_6G_80M_NUM, center_ch_6g_80m},
{CENTER_CH_6G_160M_NUM, center_ch_6g_160m},
};
#endif
/*
* Get center channel of smaller bandwidth by @param band, @param cch, @param bw, @param offset
* @band: the given band
* @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
*/
static u8 _rtw_get_scch_by_bcch_offset(enum band_type band, u8 cch, u8 bw, u8 offset)
{
u8 t_cch = 0;
if (bw == CHANNEL_WIDTH_20) {
t_cch = cch;
goto exit;
}
if (offset == CHAN_OFFSET_NO_EXT) {
rtw_warn_on(1);
goto exit;
}
if (band == BAND_ON_24G) {
/* 2.4G, 40MHz */
if (cch >= 3 && cch <= 11 && bw == CHANNEL_WIDTH_40) {
t_cch = (offset == CHAN_OFFSET_LOWER) ? cch + 2 : cch - 2;
goto exit;
}
}
#if CONFIG_IEEE80211_BAND_5GHZ
else if (band == BAND_ON_5G) {
/* 5G, 160MHz */
if (cch >= 50 && cch <= 163 && bw == CHANNEL_WIDTH_160) {
t_cch = (offset == CHAN_OFFSET_LOWER) ? cch + 8 : cch - 8;
goto exit;
/* 5G, 80MHz */
} else if (cch >= 42 && cch <= 171 && bw == CHANNEL_WIDTH_80) {
t_cch = (offset == CHAN_OFFSET_LOWER) ? cch + 4 : cch - 4;
goto exit;
/* 5G, 40MHz */
} else if (cch >= 38 && cch <= 175 && bw == CHANNEL_WIDTH_40) {
t_cch = (offset == CHAN_OFFSET_LOWER) ? cch + 2 : cch - 2;
goto exit;
}
}
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
else if (band == BAND_ON_6G) {
/* 6G, 160MHz */
if (cch >= 15 && cch <= 239 && bw == CHANNEL_WIDTH_160) {
t_cch = (offset == CHAN_OFFSET_LOWER) ? cch + 8 : cch - 8;
goto exit;
/* 6G, 80MHz */
} else if (cch >= 7 && cch <= 247 && bw == CHANNEL_WIDTH_80) {
t_cch = (offset == CHAN_OFFSET_LOWER) ? cch + 4 : cch - 4;
goto exit;
/* 6G, 40MHz */
} else if (cch >= 3 && cch <= 251 && bw == CHANNEL_WIDTH_40) {
t_cch = (offset == CHAN_OFFSET_LOWER) ? cch + 2 : cch - 2;
goto exit;
}
}
#endif
rtw_warn_on(1);
exit:
return t_cch;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY u8 rtw_get_scch_by_cch_offset(u8 cch, u8 bw, u8 offset)
{
return _rtw_get_scch_by_bcch_offset(rtw_is_2g_ch(cch) ? BAND_ON_24G : BAND_ON_5G, cch, bw, offset);
}
RTW_FUNC_2G_5G_ONLY u8 rtw_get_scch_by_cch_opch(u8 cch, u8 bw, u8 opch)
{
u8 offset = CHAN_OFFSET_NO_EXT;
if (opch > cch)
offset = CHAN_OFFSET_LOWER;
else if (opch < cch)
offset = CHAN_OFFSET_UPPER;
return rtw_get_scch_by_cch_offset(cch, bw, offset);
}
#endif
u8 rtw_get_scch_by_bcch_offset(enum band_type band, u8 cch, u8 bw, u8 offset)
{
return _rtw_get_scch_by_bcch_offset(band, cch, bw, offset);
}
/*
* Get center channel of smaller bandwidth by @param band, @param cch, @param bw, @param opch
* @band: the given band
* @cch: the given center channel
* @bw: the given bandwidth
* @opch: the given operating channel
*
* return center channel of smaller bandiwdth if valid, or 0
*/
u8 rtw_get_scch_by_bcch_opch(enum band_type band, u8 cch, u8 bw, u8 opch)
{
u8 offset = CHAN_OFFSET_NO_EXT;
if (opch > cch)
offset = CHAN_OFFSET_LOWER;
else if (opch < cch)
offset = CHAN_OFFSET_UPPER;
return rtw_get_scch_by_bcch_offset(band, cch, bw, offset);
}
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},
};
#if CONFIG_IEEE80211_BAND_5GHZ
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},
};
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
struct op_chs_ent_t op_chs_of_cch_6g_by_bw[] = {
{1, center_ch_6g_20m},
{2, (u8 *)op_chs_of_cch_6g_40m},
{4, (u8 *)op_chs_of_cch_6g_80m},
{8, (u8 *)op_chs_of_cch_6g_160m},
};
#endif
u8 center_chs_2g_num(u8 bw)
{
if (bw > CHANNEL_WIDTH_40)
return 0;
return center_chs_2g_by_bw[bw].ch_num;
}
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];
}
#if CONFIG_IEEE80211_BAND_5GHZ
u8 center_chs_5g_num(u8 bw)
{
if (bw > CHANNEL_WIDTH_160)
return 0;
return center_chs_5g_by_bw[bw].ch_num;
}
u8 center_chs_5g(u8 bw, u8 id)
{
if (bw > CHANNEL_WIDTH_160)
return 0;
if (id >= center_chs_5g_num(bw))
return 0;
return center_chs_5g_by_bw[bw].chs[id];
}
#endif /* CONFIG_IEEE80211_BAND_5GHZ */
#if CONFIG_IEEE80211_BAND_6GHZ
u8 center_chs_6g_num(u8 bw)
{
if (bw > CHANNEL_WIDTH_160)
return 0;
return center_chs_6g_by_bw[bw].ch_num;
}
u8 center_chs_6g(u8 bw, u8 id)
{
if (bw > CHANNEL_WIDTH_160)
return 0;
if (id >= center_chs_6g_num(bw))
return 0;
return center_chs_6g_by_bw[bw].chs[id];
}
#endif /* CONFIG_IEEE80211_BAND_6GHZ */
u8 (*center_chs_num_of_band[BAND_MAX])(u8 bw) = {
[BAND_ON_24G] = center_chs_2g_num,
#if CONFIG_IEEE80211_BAND_5GHZ
[BAND_ON_5G] = center_chs_5g_num,
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
[BAND_ON_6G] = center_chs_6g_num,
#endif
};
u8 (*center_chs_of_band[BAND_MAX])(u8 bw, u8 id) = {
[BAND_ON_24G] = center_chs_2g,
#if CONFIG_IEEE80211_BAND_5GHZ
[BAND_ON_5G] = center_chs_5g,
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
[BAND_ON_6G] = center_chs_6g,
#endif
};
/*
* 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)
*/
static u8 _rtw_get_op_chs_by_bcch_bw(enum band_type band, 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 (band == BAND_ON_24G
&& bw <= CHANNEL_WIDTH_40
) {
c_chs_ent = &center_chs_2g_by_bw[bw];
op_chs_ent = &op_chs_of_cch_2g_by_bw[bw];
#if CONFIG_IEEE80211_BAND_5GHZ
} else if (band == BAND_ON_5G
&& bw <= CHANNEL_WIDTH_160
) {
c_chs_ent = &center_chs_5g_by_bw[bw];
op_chs_ent = &op_chs_of_cch_5g_by_bw[bw];
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
} else if (band == BAND_ON_6G
&& bw <= CHANNEL_WIDTH_160
) {
c_chs_ent = &center_chs_6g_by_bw[bw];
op_chs_ent = &op_chs_of_cch_6g_by_bw[bw];
#endif
} 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;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY u8 rtw_get_op_chs_by_cch_bw(u8 cch, u8 bw, u8 **op_chs, u8 *op_ch_num)
{
return _rtw_get_op_chs_by_bcch_bw(rtw_is_2g_ch(cch) ? BAND_ON_24G : BAND_ON_5G, cch, bw, op_chs, op_ch_num);
}
#endif
u8 rtw_get_op_chs_by_bcch_bw(enum band_type band, u8 cch, u8 bw, u8 **op_chs, u8 *op_ch_num)
{
return _rtw_get_op_chs_by_bcch_bw(band, cch, bw, op_chs, op_ch_num);
}
static u8 rtw_get_offsets_by_chbw_2g(u8 ch, u8 bw, u8 *r_offset, u8 *r_offset_num)
{
u8 valid = 1;
u8 offset[2] = {CHAN_OFFSET_NO_EXT, CHAN_OFFSET_NO_EXT};
u8 offset_num = 1;
if (bw >= CHANNEL_WIDTH_80
|| ch < 1 || ch > 14
) {
valid = 0;
goto exit;
}
if (bw == CHANNEL_WIDTH_20)
goto exit;
if (ch >= 1 && ch <= 4)
offset[0] = CHAN_OFFSET_UPPER;
else if (ch >= 5 && ch <= 9) {
if (!r_offset_num) {
/* return single offset */
if (r_offset && (*r_offset == CHAN_OFFSET_UPPER || *r_offset == CHAN_OFFSET_LOWER))
offset[0] = *r_offset; /* both lower and upper is valid, obey input value */
else
offset[0] = CHAN_OFFSET_LOWER; /* default use primary upper */
} else {
offset_num = 2;
offset[0] = CHAN_OFFSET_LOWER;
offset[1] = CHAN_OFFSET_UPPER;
}
} else if (ch >= 10 && ch <= 13)
offset[0] = CHAN_OFFSET_LOWER;
else {
valid = 0; /* ch14 doesn't support 40MHz bandwidth */
goto exit;
}
exit:
if (valid) {
if (r_offset) {
*r_offset = offset[0];
if (offset_num == 2)
*(r_offset + 1) = offset[1];
}
if (r_offset_num)
*r_offset_num = offset_num;
}
return valid;
}
#if CONFIG_IEEE80211_BAND_5GHZ
static u8 rtw_get_offset_by_chbw_5g(u8 ch, u8 bw, u8 *r_offset)
{
u8 valid = 1;
u8 offset = CHAN_OFFSET_NO_EXT;
if (ch < 36 || ch > 177) {
valid = 0;
goto exit;
}
switch (ch) {
case 36:
case 44:
case 52:
case 60:
case 100:
case 108:
case 116:
case 124:
case 132:
case 140:
case 149:
case 157:
case 165:
case 173:
if (bw >= CHANNEL_WIDTH_40 && bw <= CHANNEL_WIDTH_160)
offset = CHAN_OFFSET_UPPER;
break;
case 40:
case 48:
case 56:
case 64:
case 104:
case 112:
case 120:
case 128:
case 136:
case 144:
case 153:
case 161:
case 169:
case 177:
if (bw >= CHANNEL_WIDTH_40 && bw <= CHANNEL_WIDTH_160)
offset = CHAN_OFFSET_LOWER;
break;
default:
valid = 0;
break;
}
exit:
if (valid && r_offset)
*r_offset = offset;
return valid;
}
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
static u8 rtw_get_offset_by_chbw_6g(u8 ch, u8 bw, u8 *r_offset)
{
if (ch >= 1 && ch <= 253) {
u8 mod8 = ch % 8;
if (mod8 == 1) {
if (bw == CHANNEL_WIDTH_20)
*r_offset = CHAN_OFFSET_NO_EXT;
else
*r_offset = CHAN_OFFSET_UPPER;
return 1;
}
if (mod8 == 5) {
if (bw == CHANNEL_WIDTH_20)
*r_offset = CHAN_OFFSET_NO_EXT;
else
*r_offset = CHAN_OFFSET_LOWER;
return 1;
}
}
return 0;
}
#endif
u8 rtw_get_offset_by_bchbw(enum band_type band, u8 ch, u8 bw, u8 *r_offset)
{
if (band == BAND_ON_24G) {
u8 offset[1] = {r_offset ? *r_offset : CHAN_OFFSET_NO_EXT};
u8 ret = rtw_get_offsets_by_chbw_2g(ch, bw, offset, NULL);
if (ret && r_offset)
*r_offset = offset[0];
return ret;
}
#if CONFIG_IEEE80211_BAND_5GHZ
else if (band == BAND_ON_5G)
return rtw_get_offset_by_chbw_5g(ch, bw, r_offset);
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
else if (band == BAND_ON_6G)
return rtw_get_offset_by_chbw_6g(ch, bw, r_offset);
#endif
return 0;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY u8 rtw_get_offset_by_chbw(u8 ch, u8 bw, u8 *r_offset)
{
return rtw_get_offset_by_bchbw(rtw_is_2g_ch(ch) ? BAND_ON_24G : BAND_ON_5G, ch, bw, r_offset);
}
#endif
u8 rtw_get_offsets_by_bchbw(enum band_type band, u8 ch, u8 bw, u8 *r_offset, u8 *r_offset_num)
{
u8 ret = 0;
if (band == BAND_ON_24G)
ret = rtw_get_offsets_by_chbw_2g(ch, bw, r_offset, r_offset_num);
#if CONFIG_IEEE80211_BAND_5GHZ
else if (band == BAND_ON_5G) {
ret = rtw_get_offset_by_chbw_5g(ch, bw, r_offset);
if (ret && r_offset_num)
*r_offset_num = 1;
}
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
else if (band == BAND_ON_6G) {
ret = rtw_get_offset_by_chbw_6g(ch, bw, r_offset);
if (ret && r_offset_num)
*r_offset_num = 1;
}
#endif
return ret;
}
static u8 rtw_get_center_ch_2g(u8 ch, u8 bw, u8 offset)
{
u8 cch = ch;
if (bw == CHANNEL_WIDTH_80) {
/* special case for 2.4G */
cch = 7;
} else if (bw == CHANNEL_WIDTH_40) {
if (offset == CHAN_OFFSET_UPPER)
cch = ch + 2;
else if (offset == CHAN_OFFSET_LOWER)
cch = ch - 2;
} else if (bw == CHANNEL_WIDTH_20
|| bw == CHANNEL_WIDTH_10
|| bw == CHANNEL_WIDTH_5
)
; /* same as ch */
else
rtw_warn_on(1);
return cch;
}
#if CONFIG_IEEE80211_BAND_5GHZ
static u8 rtw_get_center_ch_5g(u8 ch, u8 bw, u8 offset)
{
u8 cch = ch;
if (bw == CHANNEL_WIDTH_160) {
if (ch % 4 == 0) {
if (ch >= 36 && ch <= 64)
cch = 50;
else if (ch >= 100 && ch <= 128)
cch = 114;
} else if (ch % 4 == 1) {
if (ch >= 149 && ch <= 177)
cch = 163;
}
} else if (bw == CHANNEL_WIDTH_80) {
if (ch % 4 == 0) {
if (ch >= 36 && ch <= 48)
cch = 42;
else if (ch >= 52 && ch <= 64)
cch = 58;
else if (ch >= 100 && ch <= 112)
cch = 106;
else if (ch >= 116 && ch <= 128)
cch = 122;
else if (ch >= 132 && ch <= 144)
cch = 138;
} else if (ch % 4 == 1) {
if (ch >= 149 && ch <= 161)
cch = 155;
else if (ch >= 165 && ch <= 177)
cch = 171;
}
} else if (bw == CHANNEL_WIDTH_40) {
if (offset == CHAN_OFFSET_UPPER)
cch = ch + 2;
else if (offset == CHAN_OFFSET_LOWER)
cch = ch - 2;
} else if (bw == CHANNEL_WIDTH_20
|| bw == CHANNEL_WIDTH_10
|| bw == CHANNEL_WIDTH_5
)
; /* same as ch */
else
rtw_warn_on(1);
return cch;
}
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
static u8 rtw_get_center_ch_6g(u8 ch, u8 bw, u8 offset)
{
static const u8 start[CHANNEL_WIDTH_MAX] = {
[CHANNEL_WIDTH_40] = 3,
[CHANNEL_WIDTH_80] = 7,
[CHANNEL_WIDTH_160] = 15,
};
static const u8 shift[CHANNEL_WIDTH_MAX] = {
[CHANNEL_WIDTH_40] = 3,
[CHANNEL_WIDTH_80] = 4,
[CHANNEL_WIDTH_160] = 5,
};
u8 cch = ch;
if (bw < CHANNEL_WIDTH_20 || bw > CHANNEL_WIDTH_160) {
rtw_warn_on(1);
goto exit;
}
if (bw == CHANNEL_WIDTH_20)
goto exit;
cch = (((ch - 1) >> shift[bw]) << shift[bw]) + start[bw];
exit:
return cch;
}
#endif
u8 rtw_get_center_ch_by_band(enum band_type band, u8 ch, u8 bw, u8 offset)
{
if (band == BAND_ON_24G)
return rtw_get_center_ch_2g(ch, bw, offset);
#if CONFIG_IEEE80211_BAND_5GHZ
else if (band == BAND_ON_5G)
return rtw_get_center_ch_5g(ch, bw, offset);
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
else if (band == BAND_ON_6G)
return rtw_get_center_ch_6g(ch, bw, offset);
#endif
return 0;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY u8 rtw_get_center_ch(u8 ch, u8 bw, u8 offset)
{
return rtw_get_center_ch_by_band(rtw_is_2g_ch(ch) ? BAND_ON_24G : BAND_ON_5G, ch, bw, offset);
}
#endif
/**
* rtw_is_bchbw_grouped - test if the two ch settings can be grouped together
* @band_a: band of set a
* @ch_a: ch of set a
* @bw_a: bw of set a
* @offset_a: offset of set a
* @band_b: band of set b
* @ch_b: ch of set b
* @bw_b: bw of set b
* @offset_b: offset of set b
*/
bool _rtw_is_bchbw_grouped(enum band_type band_a, u8 ch_a, u8 bw_a, u8 offset_a
, enum band_type band_b, u8 ch_b, u8 bw_b, u8 offset_b)
{
bool is_grouped = _FALSE;
if (band_a != band_b || ch_a != ch_b) {
/* band/ch is different */
goto exit;
}
if ((bw_a == CHANNEL_WIDTH_40 || bw_a == CHANNEL_WIDTH_80 || bw_a == CHANNEL_WIDTH_160)
&& (bw_b == CHANNEL_WIDTH_40 || bw_b == CHANNEL_WIDTH_80 || bw_b == CHANNEL_WIDTH_160)
) {
if (offset_a != offset_b)
goto exit;
}
is_grouped = _TRUE;
exit:
return is_grouped;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY bool rtw_is_chbw_grouped(u8 ch_a, u8 bw_a, u8 offset_a, u8 ch_b, u8 bw_b, u8 offset_b)
{
enum band_type band = BAND_ON_24G; /* unknown band, use same band for a & b */
return _rtw_is_bchbw_grouped(band, ch_a, bw_a, offset_a, band, ch_b, bw_b, offset_b);
}
#endif
bool rtw_is_bchbw_grouped(enum band_type band_a, u8 ch_a, u8 bw_a, u8 offset_a
, enum band_type band_b, u8 ch_b, u8 bw_b, u8 offset_b)
{
return _rtw_is_bchbw_grouped(band_a, ch_a, bw_a, offset_a, band_b, ch_b, bw_b, offset_b);
}
/**
* rtw_sync_bchbw - obey g_band, g_ch, adjust g_bw, g_offset, bw, offset
* @req_band: pointer of the request band, may be modified further
* @req_ch: pointer of the request ch, may be modified further
* @req_bw: pointer of the request bw, may be modified further
* @req_offset: pointer of the request offset, may be modified further
* @g_band: pointer of the ongoing group band
* @g_ch: pointer of the ongoing group ch
* @g_bw: pointer of the ongoing group bw, may be modified further
* @g_offset: pointer of the ongoing group offset, may be modified further
*/
void _rtw_sync_bchbw(enum band_type *req_band, u8 *req_ch, u8 *req_bw, u8 *req_offset
, enum band_type *g_band, u8 *g_ch, u8 *g_bw, u8 *g_offset)
{
*req_band = *g_band;
*req_ch = *g_ch;
if (*req_bw == CHANNEL_WIDTH_80 && *g_band == BAND_ON_24G) {
/*2.4G ch, downgrade to 40Mhz */
*req_bw = CHANNEL_WIDTH_40;
}
switch (*req_bw) {
case CHANNEL_WIDTH_160:
case CHANNEL_WIDTH_80:
case CHANNEL_WIDTH_40:
if (*g_bw == CHANNEL_WIDTH_40 || *g_bw == CHANNEL_WIDTH_80 || *g_bw == CHANNEL_WIDTH_160)
*req_offset = *g_offset;
else if (*g_bw == CHANNEL_WIDTH_20)
rtw_get_offset_by_bchbw(*req_band, *req_ch, *req_bw, req_offset);
if (*req_offset == CHAN_OFFSET_NO_EXT) {
RTW_ERR("%s req %s BW without offset, down to 20MHz\n", __func__, ch_width_str(*req_bw));
rtw_warn_on(1);
*req_bw = CHANNEL_WIDTH_20;
}
break;
case CHANNEL_WIDTH_20:
*req_offset = CHAN_OFFSET_NO_EXT;
break;
default:
RTW_ERR("%s req unsupported BW:%u\n", __func__, *req_bw);
rtw_warn_on(1);
}
if (*req_bw > *g_bw) {
*g_bw = *req_bw;
*g_offset = *req_offset;
}
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY void rtw_sync_chbw(u8 *req_ch, u8 *req_bw, u8 *req_offset, u8 *g_ch, u8 *g_bw, u8 *g_offset)
{
enum band_type band = rtw_is_2g_ch(*g_ch) ? BAND_ON_24G : BAND_ON_5G; /* follow g_ch's band */
_rtw_sync_bchbw(&band, req_ch, req_bw, req_offset, &band, g_ch, g_bw, g_offset);
}
#endif
void rtw_sync_bchbw(enum band_type *req_band, u8 *req_ch, u8 *req_bw, u8 *req_offset
, enum band_type *g_band, u8 *g_ch, u8 *g_bw, u8 *g_offset)
{
_rtw_sync_bchbw(req_band, req_ch, req_bw, req_offset, g_band, g_ch, g_bw, g_offset);
}
static int rtw_ch2freq_2g(int ch)
{
if (ch >= 1 && ch <= 14) {
if (ch == 14)
return 2484;
else if (ch < 14)
return 2407 + ch * 5;
}
return 0; /* not supported */
}
#if CONFIG_IEEE80211_BAND_5GHZ
static int rtw_ch2freq_5g(int ch)
{
if (ch >= 36 && ch <= 177)
return 5000 + ch * 5;
return 0; /* not supported */
}
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
static int rtw_ch2freq_6g(int ch)
{
if (ch >= 1 && ch <= 253)
return 5950 + ch * 5;
return 0; /* not supported */
}
#endif
int rtw_bch2freq(enum band_type band, int ch)
{
if (band == BAND_ON_24G)
return rtw_ch2freq_2g(ch);
#if CONFIG_IEEE80211_BAND_5GHZ
else if (band == BAND_ON_5G)
return rtw_ch2freq_5g(ch);
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
else if (band == BAND_ON_6G)
return rtw_ch2freq_6g(ch);
#endif
return 0;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY int rtw_ch2freq(int ch)
{
return rtw_bch2freq(rtw_is_2g_ch(ch) ? BAND_ON_24G : BAND_ON_5G, ch);
}
#endif
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 >= 5000 && freq < 5950)
return (freq - 5000) / 5;
else if (freq >= 5950 && freq <= 7215)
return (freq - 5950) / 5;
else if (freq >= 58320 && freq <= 64800)
return (freq - 56160) / 2160;
else
return 0;
}
enum band_type rtw_freq2band(int freq)
{
if (freq <= 2484)
return BAND_ON_24G;
#if CONFIG_IEEE80211_BAND_5GHZ
else if (freq >= 5000 && freq < 5950)
return BAND_ON_5G;
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
else if (freq >= 5950 && freq <= 7215)
return BAND_ON_6G;
#endif
else
return BAND_MAX;
}
enum channel_width rtw_frange_to_bw(u32 hi, u32 lo)
{
u32 width = hi - lo;
switch (width) {
case 160:
return CHANNEL_WIDTH_160;
case 80:
return CHANNEL_WIDTH_80;
case 40:
return CHANNEL_WIDTH_40;
case 20:
return CHANNEL_WIDTH_20;
case 10:
return CHANNEL_WIDTH_10;
case 5:
return CHANNEL_WIDTH_5;
default:
return CHANNEL_WIDTH_MAX;
}
}
bool rtw_freq_consecutive(int a, int b)
{
enum band_type band_a, band_b;
band_a = rtw_freq2band(a);
if (band_a == BAND_MAX)
return 0;
band_b = rtw_freq2band(b);
if (band_b == BAND_MAX || band_a != band_b)
return 0;
switch (band_a) {
case BAND_ON_24G:
return rtw_abs(a - b) == 5;
case BAND_ON_5G:
#if CONFIG_IEEE80211_BAND_6GHZ
case BAND_ON_6G:
#endif
return rtw_abs(a - b) == 20;
default:
return 0;
}
}
bool rtw_bcchbw_to_freq_range(enum band_type band, u8 c_ch, u8 bw, u32 *hi, u32 *lo)
{
u32 freq;
u32 hi_ret = 0, lo_ret = 0;
bool valid = _FALSE;
if (hi)
*hi = 0;
if (lo)
*lo = 0;
freq = rtw_bch2freq(band, c_ch);
if (!freq) {
rtw_warn_on(1);
goto exit;
}
if (bw == CHANNEL_WIDTH_160) {
hi_ret = freq + 80;
lo_ret = freq - 80;
} else 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;
}
bool rtw_bchbw_to_freq_range(enum band_type band, u8 ch, u8 bw, u8 offset, u32 *hi, u32 *lo)
{
return rtw_bcchbw_to_freq_range(band
, rtw_get_center_ch_by_band(band, ch, bw, offset)
, bw, hi, lo);
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY bool rtw_chbw_to_freq_range(u8 ch, u8 bw, u8 offset, u32 *hi, u32 *lo)
{
return rtw_bchbw_to_freq_range(rtw_is_2g_ch(ch) ? BAND_ON_24G : BAND_ON_5G, ch, bw, offset, hi, lo);
}
#endif
const char *const _ch_width_str[CHANNEL_WIDTH_MAX] = {
[CHANNEL_WIDTH_20] = "20MHz",
[CHANNEL_WIDTH_40] = "40MHz",
[CHANNEL_WIDTH_80] = "80MHz",
[CHANNEL_WIDTH_160] = "160MHz",
[CHANNEL_WIDTH_80_80] = "80_80MHz",
[CHANNEL_WIDTH_5] = "5MHz",
[CHANNEL_WIDTH_10] = "10MHz",
};
const u8 _ch_width_to_bw_cap[CHANNEL_WIDTH_MAX] = {
[CHANNEL_WIDTH_20] = BW_CAP_20M,
[CHANNEL_WIDTH_40] = BW_CAP_40M,
[CHANNEL_WIDTH_80] = BW_CAP_80M,
[CHANNEL_WIDTH_160] = BW_CAP_160M,
[CHANNEL_WIDTH_80_80] = BW_CAP_80_80M,
[CHANNEL_WIDTH_5] = BW_CAP_5M,
[CHANNEL_WIDTH_10] = BW_CAP_10M,
};
const char *const _rtw_band_str[] = {
[BAND_ON_24G] = "2.4G",
[BAND_ON_5G] = "5G",
[BAND_ON_6G] = "6G",
[BAND_MAX] = "BAND_MAX",
};
const u8 _band_to_band_cap[] = {
[BAND_ON_24G] = BAND_CAP_2G,
[BAND_ON_5G] = BAND_CAP_5G,
[BAND_ON_6G] = BAND_CAP_6G,
[BAND_MAX] = 0,
};
const char *const _opc_bw_str[OPC_BW_NUM] = {
[OPC_BW20] = "20M ",
[OPC_BW40PLUS] = "40M+",
[OPC_BW40MINUS] = "40M-",
[OPC_BW40] = "40M ",
[OPC_BW80] = "80M ",
[OPC_BW160] = "160M ",
[OPC_BW80P80] = "80+80M ",
};
const u8 _opc_bw_to_ch_width[OPC_BW_NUM] = {
[OPC_BW20] = CHANNEL_WIDTH_20,
[OPC_BW40PLUS] = CHANNEL_WIDTH_40,
[OPC_BW40MINUS] = CHANNEL_WIDTH_40,
[OPC_BW40] = CHANNEL_WIDTH_40,
[OPC_BW80] = CHANNEL_WIDTH_80,
[OPC_BW160] = CHANNEL_WIDTH_160,
[OPC_BW80P80] = CHANNEL_WIDTH_80_80,
};
/* global operating class database */
#define OP_CLASS_ENT(_class, _band, _bw, _len, arg...) \
{.class_id = _class, .band = _band, .bw = _bw, .len_ch_attr = (uint8_t[_len + 1]) {_len, ##arg},}
/* 802.11-2020, 802.11ax-2021 Table E-4, partial */
const struct op_class_t global_op_class[] = {
/* 2G ch1~13, 20M */
OP_CLASS_ENT(81, BAND_ON_24G, OPC_BW20, 13, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13),
/* 2G ch14, 20M */
OP_CLASS_ENT(82, BAND_ON_24G, OPC_BW20, 1, 14),
/* 2G, 40M */
OP_CLASS_ENT(83, BAND_ON_24G, OPC_BW40PLUS, 9, 1, 2, 3, 4, 5, 6, 7, 8, 9),
OP_CLASS_ENT(84, BAND_ON_24G, OPC_BW40MINUS, 9, 5, 6, 7, 8, 9, 10, 11, 12, 13),
#if CONFIG_IEEE80211_BAND_5GHZ
/* 5G band 1, 20M & 40M */
OP_CLASS_ENT(115, BAND_ON_5G, OPC_BW20, 4, 36, 40, 44, 48),
OP_CLASS_ENT(116, BAND_ON_5G, OPC_BW40PLUS, 2, 36, 44),
OP_CLASS_ENT(117, BAND_ON_5G, OPC_BW40MINUS, 2, 40, 48),
/* 5G band 2, 20M & 40M */
OP_CLASS_ENT(118, BAND_ON_5G, OPC_BW20, 4, 52, 56, 60, 64),
OP_CLASS_ENT(119, BAND_ON_5G, OPC_BW40PLUS, 2, 52, 60),
OP_CLASS_ENT(120, BAND_ON_5G, OPC_BW40MINUS, 2, 56, 64),
/* 5G band 3, 20M & 40M */
OP_CLASS_ENT(121, BAND_ON_5G, OPC_BW20, 12, 100, 104, 108, 112, 116, 120, 124, 128, 132, 136, 140, 144),
OP_CLASS_ENT(122, BAND_ON_5G, OPC_BW40PLUS, 6, 100, 108, 116, 124, 132, 140),
OP_CLASS_ENT(123, BAND_ON_5G, OPC_BW40MINUS, 6, 104, 112, 120, 128, 136, 144),
/* 5G band 4, 20M & 40M */
OP_CLASS_ENT(124, BAND_ON_5G, OPC_BW20, 4, 149, 153, 157, 161),
OP_CLASS_ENT(125, BAND_ON_5G, OPC_BW20, 8, 149, 153, 157, 161, 165, 169, 173, 177),
OP_CLASS_ENT(126, BAND_ON_5G, OPC_BW40PLUS, 4, 149, 157, 165, 173),
OP_CLASS_ENT(127, BAND_ON_5G, OPC_BW40MINUS, 4, 153, 161, 169, 177),
/* 5G, 80M & 160M */
OP_CLASS_ENT(128, BAND_ON_5G, OPC_BW80, 28, 36, 40, 44, 48, 52, 56, 60, 64, 100, 104, 108, 112, 116, 120, 124, 128, 132, 136, 140, 144, 149, 153, 157, 161, 165, 169, 173, 177),
OP_CLASS_ENT(129, BAND_ON_5G, OPC_BW160, 24, 36, 40, 44, 48, 52, 56, 60, 64, 100, 104, 108, 112, 116, 120, 124, 128, 149, 153, 157, 161, 165, 169, 173, 177),
#if 0 /* TODO */
/* 5G, 80+80M */
OP_CLASS_ENT(130, BAND_ON_5G, OPC_BW80P80, 28, 36, 40, 44, 48, 52, 56, 60, 64, 100, 104, 108, 112, 116, 120, 124, 128, 132, 136, 140, 144, 149, 153, 157, 161, 165, 169, 173, 177),
#endif
#endif
#if CONFIG_IEEE80211_BAND_6GHZ
/* 6G, 20M */
OP_CLASS_ENT(131, BAND_ON_6G, OPC_BW20, 59, 1, 5, 9, 13, 17, 21, 25, 29, 33, 37, 41, 45, 49, 53, 57, 61, 65, 69, 73, 77, 81, 85, 89, 93
, 97, 101, 105, 109, 113, 117
, 121, 125, 129, 133, 137, 141, 145, 149, 153, 157, 161, 165, 169, 173, 177, 181, 185, 189
, 193, 197, 201, 205, 209, 213, 217, 221, 225, 229, 233),
/* 6G, 40M */
OP_CLASS_ENT(132, BAND_ON_6G, OPC_BW40, 58, 1, 5, 9, 13, 17, 21, 25, 29, 33, 37, 41, 45, 49, 53, 57, 61, 65, 69, 73, 77, 81, 85, 89, 93
, 97, 101, 105, 109, 113, 117
, 121, 125, 129, 133, 137, 141, 145, 149, 153, 157, 161, 165, 169, 173, 177, 181, 185, 189
, 193, 197, 201, 205, 209, 213, 217, 221, 225, 229),
/* 6G, 80M */
OP_CLASS_ENT(133, BAND_ON_6G, OPC_BW80, 56, 1, 5, 9, 13, 17, 21, 25, 29, 33, 37, 41, 45, 49, 53, 57, 61, 65, 69, 73, 77, 81, 85, 89, 93
, 97, 101, 105, 109, 113, 117
, 121, 125, 129, 133, 137, 141, 145, 149, 153, 157, 161, 165, 169, 173, 177, 181, 185, 189
, 193, 197, 201, 205, 209, 213, 217, 221),
/* 6G, 160M */
OP_CLASS_ENT(134, BAND_ON_6G, OPC_BW160, 56, 1, 5, 9, 13, 17, 21, 25, 29, 33, 37, 41, 45, 49, 53, 57, 61, 65, 69, 73, 77, 81, 85, 89, 93
, 97, 101, 105, 109, 113, 117
, 121, 125, 129, 133, 137, 141, 145, 149, 153, 157, 161, 165, 169, 173, 177, 181, 185, 189
, 193, 197, 201, 205, 209, 213, 217, 221),
#if 0 /* TODO */
/* 6G, 80+80M */
OP_CLASS_ENT(135, BAND_ON_6G, OPC_BW80P80, 56, 1, 5, 9, 13, 17, 21, 25, 29, 33, 37, 41, 45, 49, 53, 57, 61, 65, 69, 73, 77, 81, 85, 89, 93
, 97, 101, 105, 109, 113, 117
, 121, 125, 129, 133, 137, 141, 145, 149, 153, 157, 161, 165, 169, 173, 177, 181, 185, 189
, 193, 197, 201, 205, 209, 213, 217, 221),
/* 6G, 20M, ch2 */
OP_CLASS_ENT(136, BAND_ON_6G, OPC_BW20, 1, 2),
#endif
#endif
};
const int global_op_class_num = sizeof(global_op_class) / sizeof(struct op_class_t);
const struct op_class_t *get_global_op_class_by_id(u8 gid)
{
int i;
for (i = 0; i < global_op_class_num; i++)
if (global_op_class[i].class_id == gid)
break;
return i < global_op_class_num ? &global_op_class[i] : NULL;
}
bool is_valid_global_op_class_id(u8 gid)
{
return get_global_op_class_by_id(gid) ? 1 : 0;
}
bool is_valid_global_op_class_ch(const struct op_class_t *opc, u8 ch)
{
int array_idx;
int i;
if (opc < global_op_class
|| (((u8 *)opc) - ((u8 *)global_op_class)) % sizeof(struct op_class_t)
) {
RTW_ERR("Invalid opc pointer:%p (global_op_class:%p, sizeof(struct op_class_t):%zu, %zu)\n"
, opc, global_op_class, sizeof(struct op_class_t), (((u8 *)opc) - ((u8 *)global_op_class)) % sizeof(struct op_class_t));
return 0;
}
array_idx = (((u8 *)opc) - ((u8 *)global_op_class)) / sizeof(struct op_class_t);
for (i = 0; i < OPC_CH_LIST_LEN(global_op_class[array_idx]); i++)
if (OPC_CH_LIST_CH(global_op_class[array_idx], i) == ch)
break;
return i < OPC_CH_LIST_LEN(global_op_class[array_idx]);
}
/* -2: logic error, -1: error, 0: is already BW20 */
s16 get_sub_op_class(u8 gid, u8 ch)
{
const struct op_class_t *opc = get_global_op_class_by_id(gid);
int i;
enum channel_width bw;
if (!opc)
return -1;
if (!is_valid_global_op_class_ch(opc, ch)) {
return -1;
}
if (opc->bw == OPC_BW20)
return 0;
bw = opc_bw_to_ch_width(opc->bw);
for (i = 0; i < global_op_class_num; i++) {
if (bw != opc_bw_to_ch_width(global_op_class[i].bw) + 1)
continue;
if (is_valid_global_op_class_ch(&global_op_class[i], ch))
break;
}
return i < global_op_class_num ? global_op_class[i].class_id : -2;
}
static void dump_op_class_ch_title(void *sel)
{
RTW_PRINT_SEL(sel, "%-5s %-4s %-7s ch_list\n"
, "class", "band", "bw");
}
static void dump_global_op_class_ch_single(void *sel, u8 gid)
{
u8 i;
RTW_PRINT_SEL(sel, "%5u %4s %7s"
, global_op_class[gid].class_id
, band_str(global_op_class[gid].band)
, opc_bw_str(global_op_class[gid].bw));
for (i = 0; i < OPC_CH_LIST_LEN(global_op_class[gid]); i++)
_RTW_PRINT_SEL(sel, " %u", OPC_CH_LIST_CH(global_op_class[gid], i));
_RTW_PRINT_SEL(sel, "\n");
}
#ifdef CONFIG_RTW_DEBUG
static bool dbg_global_op_class_validate(u8 gid)
{
u8 i;
u8 ch, bw, offset, cch;
bool ret = 1;
switch (global_op_class[gid].bw) {
case OPC_BW20:
bw = CHANNEL_WIDTH_20;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW40PLUS:
bw = CHANNEL_WIDTH_40;
offset = CHAN_OFFSET_UPPER;
break;
case OPC_BW40MINUS:
bw = CHANNEL_WIDTH_40;
offset = CHAN_OFFSET_LOWER;
break;
case OPC_BW40:
bw = CHANNEL_WIDTH_40;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW80:
bw = CHANNEL_WIDTH_80;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW160:
bw = CHANNEL_WIDTH_160;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW80P80: /* TODO */
default:
RTW_ERR("%s class:%u unsupported opc_bw:%u\n"
, __func__, global_op_class[gid].class_id, global_op_class[gid].bw);
ret = 0;
goto exit;
}
for (i = 0; i < OPC_CH_LIST_LEN(global_op_class[gid]); i++) {
u8 *op_chs;
u8 op_ch_num;
u8 k;
ch = OPC_CH_LIST_CH(global_op_class[gid], i);
cch = rtw_get_center_ch_by_band(global_op_class[gid].band, ch ,bw, offset);
if (!cch) {
RTW_ERR("%s can't get cch from class:%u ch:%u\n"
, __func__, global_op_class[gid].class_id, ch);
ret = 0;
continue;
}
if (!rtw_get_op_chs_by_bcch_bw(global_op_class[gid].band, cch, bw, &op_chs, &op_ch_num)) {
RTW_ERR("%s can't get op chs from class:%u cch:%u\n"
, __func__, global_op_class[gid].class_id, cch);
ret = 0;
continue;
}
for (k = 0; k < op_ch_num; k++) {
if (*(op_chs + k) == ch)
break;
}
if (k >= op_ch_num) {
RTW_ERR("%s can't get ch:%u from op_chs class:%u cch:%u\n"
, __func__, ch, global_op_class[i].class_id, cch);
ret = 0;
}
}
exit:
return ret;
}
#endif /* CONFIG_RTW_DEBUG */
void dump_global_op_class(void *sel)
{
u8 i;
dump_op_class_ch_title(sel);
for (i = 0; i < global_op_class_num; i++)
dump_global_op_class_ch_single(sel, i);
}
u8 _rtw_get_op_class_by_bchbw(enum band_type band, u8 ch, u8 bw, u8 offset)
{
int i;
u8 gid = 0; /* invalid */
switch (bw) {
case CHANNEL_WIDTH_20:
case CHANNEL_WIDTH_40:
case CHANNEL_WIDTH_80:
case CHANNEL_WIDTH_160:
#if 0 /* TODO */
case CHANNEL_WIDTH_80_80:
#endif
break;
default:
goto exit;
}
for (i = 0; i < global_op_class_num; i++) {
if (band != global_op_class[i].band)
continue;
if (opc_bw_to_ch_width(global_op_class[i].bw) != bw)
continue;
if ((global_op_class[i].bw == OPC_BW40PLUS
&& offset != CHAN_OFFSET_UPPER)
|| (global_op_class[i].bw == OPC_BW40MINUS
&& offset != CHAN_OFFSET_LOWER)
)
continue;
if (is_valid_global_op_class_ch(&global_op_class[i], ch))
goto get;
}
get:
if (i < global_op_class_num) {
#if 0 /* TODO */
if (bw == CHANNEL_WIDTH_80_80) {
/* search another ch */
if (!is_valid_global_op_class_ch(&global_op_class[i], ch2))
goto exit;
}
#endif
gid = global_op_class[i].class_id;
}
exit:
return gid;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY u8 rtw_get_op_class_by_chbw(u8 ch, u8 bw, u8 offset)
{
enum band_type band = BAND_MAX;
if (rtw_is_2g_ch(ch))
band = BAND_ON_24G;
else if (rtw_is_5g_ch(ch))
band = BAND_ON_5G;
else
return 0; /* invalid */
return _rtw_get_op_class_by_bchbw(band, ch, bw, offset);
}
#endif
u8 rtw_get_op_class_by_bchbw(enum band_type band, u8 ch, u8 bw, u8 offset)
{
return _rtw_get_op_class_by_bchbw(band, ch, bw, offset);
}
u8 rtw_get_bw_offset_by_op_class_ch(u8 gid, u8 ch, u8 *bw, u8 *offset)
{
const struct op_class_t *opc;
u8 valid = 0;
int i;
opc = get_global_op_class_by_id(gid);
if (!opc)
goto exit;
*bw = opc_bw_to_ch_width(opc->bw);
if (opc->bw == OPC_BW40PLUS)
*offset = CHAN_OFFSET_UPPER;
else if (opc->bw == OPC_BW40MINUS)
*offset = CHAN_OFFSET_LOWER;
if (rtw_get_offset_by_bchbw(opc->band, ch, *bw, offset))
valid = 1;
exit:
return valid;
}
#ifdef CONFIG_ECSA_PHL
int alink_get_supported_op_class(struct _ADAPTER_LINK *padapter_link, u8 *op_set, int len)
{
_adapter *padapter = padapter_link->adapter;
struct link_mlme_ext_priv *pmlmeext = &(padapter_link->mlmeextpriv);
struct rtw_chset *chset = adapter_to_chset(padapter);
int match = 0, i = 0, j, k = 0;
const struct op_class_t *cl;
u8 cur_op_class;
u8 *ch;
cur_op_class = rtw_get_op_class_by_bchbw(pmlmeext->chandef.band,
pmlmeext->chandef.chan,
pmlmeext->chandef.bw,
pmlmeext->chandef.offset);
if (cur_op_class && k < len) {
/* current op class SHALL be the 1st supported op class */
*op_set = cur_op_class;
k++;
}
for (i = 0; i < global_op_class_num; i++) {
cl = &global_op_class[i];
ch = cl->len_ch_attr;
for (j = 0; j < ch[0]; j++) {
if ((match = rtw_chset_search_ch(chset, ch[j+1])) == -1)
break;
}
if (match != -1 && cl->class_id != cur_op_class && k < len)
op_set[k++] = cl->class_id;
}
return (k > len ? len : k);
}
int get_supported_op_class(_adapter *padapter, u8 *op_set, int len)
{
return alink_get_supported_op_class(GET_PRIMARY_LINK(padapter), op_set, len);
}
#endif /* CONFIG_ECSA_PHL */
static struct op_class_pref_t *opc_pref_alloc(u8 class_id)
{
int i, j;
struct op_class_pref_t *opc_pref = NULL;
u8 ch_num;
for (i = 0; i < global_op_class_num; i++)
if (global_op_class[i].class_id == class_id)
break;
if (i >= global_op_class_num)
goto exit;
ch_num = OPC_CH_LIST_LEN(global_op_class[i]);
opc_pref = rtw_zmalloc(sizeof(*opc_pref) + (sizeof(struct op_ch_t) * ch_num));
if (!opc_pref)
goto exit;
opc_pref->class_id = global_op_class[i].class_id;
opc_pref->band = global_op_class[i].band;
opc_pref->bw = global_op_class[i].bw;
for (j = 0; j < OPC_CH_LIST_LEN(global_op_class[i]); j++) {
opc_pref->chs[j].ch = OPC_CH_LIST_CH(global_op_class[i], j);
opc_pref->chs[j].static_non_op = 1;
opc_pref->chs[j].no_ir = 1;
opc_pref->chs[j].max_txpwr = UNSPECIFIED_MBM;
}
opc_pref->ch_num = ch_num;
exit:
return opc_pref;
}
static void opc_pref_free(struct op_class_pref_t *opc_pref)
{
rtw_mfree(opc_pref, sizeof(*opc_pref) + (sizeof(struct op_ch_t) * opc_pref->ch_num));
}
enum band_type rtw_get_band_by_op_class(u8 op_class)
{
int array_idx;
const struct op_class_t *opc = get_global_op_class_by_id(op_class);
if (opc < global_op_class
|| (((u8 *)opc) - ((u8 *)global_op_class)) % sizeof(struct op_class_t)
) {
RTW_ERR("Invalid opc pointer:%p (global_op_class:%p, sizeof(struct op_class_t):%zu, %zu)\n"
, opc, global_op_class, sizeof(struct op_class_t),
(((u8 *)opc) - ((u8 *)global_op_class)) % sizeof(struct op_class_t));
return BAND_ON_24G;
}
array_idx = (((u8 *)opc) - ((u8 *)global_op_class)) / sizeof(struct op_class_t);
return global_op_class[array_idx].band;
}
int rtw_rfctl_op_class_pref_init(struct rf_ctl_t *rfctl, u8 band_bmp, u8 bw_bmp[])
{
u8 bw;
struct op_class_pref_t *opc_pref;
int i;
u8 op_class_num = 0;
int ret = _FAIL;
rfctl->spt_op_class_ch = rtw_zmalloc(sizeof(struct op_class_pref_t *) * global_op_class_num);
if (!rfctl->spt_op_class_ch) {
RTW_ERR("%s alloc rfctl->spt_op_class_ch fail\n", __func__);
goto exit;
}
for (i = 0; i < global_op_class_num; i++) {
#ifdef CONFIG_RTW_DEBUG
rtw_warn_on(!dbg_global_op_class_validate(i));
#endif
if (!(band_bmp & band_to_band_cap(global_op_class[i].band)))
continue;
bw = opc_bw_to_ch_width(global_op_class[i].bw);
if (bw == CHANNEL_WIDTH_MAX
|| bw == CHANNEL_WIDTH_80_80 /* TODO */
)
continue;
if (!(bw_bmp[global_op_class[i].band] & ch_width_to_bw_cap(bw)))
continue;
opc_pref = opc_pref_alloc(global_op_class[i].class_id);
if (!opc_pref) {
RTW_ERR("%s opc_pref_alloc(%u) fail\n", __func__, global_op_class[i].class_id);
goto exit;
}
if (opc_pref->ch_num) {
rfctl->spt_op_class_ch[i] = opc_pref;
op_class_num++;
} else
opc_pref_free(opc_pref);
}
rfctl->cap_spt_op_class_num = op_class_num;
ret = _SUCCESS;
exit:
if (ret != _SUCCESS)
rtw_rfctl_op_class_pref_deinit(rfctl);
return ret;
}
void rtw_rfctl_op_class_pref_deinit(struct rf_ctl_t *rfctl)
{
int i;
if (!rfctl->spt_op_class_ch)
return;
for (i = 0; i < global_op_class_num; i++) {
if (rfctl->spt_op_class_ch[i]) {
opc_pref_free(rfctl->spt_op_class_ch[i]);
rfctl->spt_op_class_ch[i] = NULL;
}
}
rtw_mfree(rfctl->spt_op_class_ch, sizeof(struct op_class_pref_t *) * global_op_class_num);
rfctl->spt_op_class_ch = NULL;
}
void op_class_pref_apply_regulatory(struct rf_ctl_t *rfctl, u8 reason)
{
struct rtw_chset *chset = &rfctl->chset;
u8 ch, bw, offset, cch;
struct op_class_pref_t *opc_pref;
int i, j;
u8 reg_op_class_num = 0;
u8 op_class_num = 0;
for (i = 0; i < global_op_class_num; i++) {
if (!rfctl->spt_op_class_ch[i])
continue;
opc_pref = rfctl->spt_op_class_ch[i];
/* reset all channel */
for (j = 0; j < opc_pref->ch_num; j++) {
if (reason >= REG_CHANGE)
opc_pref->chs[j].static_non_op = 1;
if (reason != REG_TXPWR_CHANGE)
opc_pref->chs[j].no_ir = 1;
if (reason >= REG_TXPWR_CHANGE)
opc_pref->chs[j].max_txpwr = UNSPECIFIED_MBM;
}
if (reason >= REG_CHANGE)
opc_pref->op_ch_num = 0;
if (reason != REG_TXPWR_CHANGE)
opc_pref->ir_ch_num = 0;
switch (opc_pref->bw) {
case OPC_BW20:
bw = CHANNEL_WIDTH_20;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW40PLUS:
bw = CHANNEL_WIDTH_40;
offset = CHAN_OFFSET_UPPER;
break;
case OPC_BW40MINUS:
bw = CHANNEL_WIDTH_40;
offset = CHAN_OFFSET_LOWER;
break;
case OPC_BW40:
bw = CHANNEL_WIDTH_40;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW80:
bw = CHANNEL_WIDTH_80;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW160:
bw = CHANNEL_WIDTH_160;
offset = CHAN_OFFSET_NO_EXT;
break;
case OPC_BW80P80: /* TODO */
default:
continue;
}
if (!RFCTL_REG_EN_11AC(rfctl)
&& (bw == CHANNEL_WIDTH_80 || bw == CHANNEL_WIDTH_160))
continue;
for (j = 0; j < opc_pref->ch_num; j++) {
u8 *op_chs;
u8 op_ch_num;
u8 k, l;
int chset_idx;
ch = opc_pref->chs[j].ch;
if (reason >= REG_TXPWR_CHANGE)
opc_pref->chs[j].max_txpwr = rtw_rfctl_get_reg_max_txpwr_mbm(rfctl, opc_pref->band, ch, bw, offset, 1);
if (reason == REG_TXPWR_CHANGE)
continue;
cch = rtw_get_center_ch_by_band(opc_pref->band, ch ,bw, offset);
if (!cch)
continue;
if (!rtw_get_op_chs_by_bcch_bw(opc_pref->band, cch, bw, &op_chs, &op_ch_num))
continue;
for (k = 0, l = 0; k < op_ch_num; k++) {
chset_idx = rtw_chset_search_bch(chset, opc_pref->band, *(op_chs + k));
if (chset_idx == -1)
break;
if (bw >= CHANNEL_WIDTH_40) {
if ((chset->chs[chset_idx].flags & RTW_CHF_NO_HT40U) && k % 2 == 0)
break;
if ((chset->chs[chset_idx].flags & RTW_CHF_NO_HT40L) && k % 2 == 1)
break;
}
if (bw >= CHANNEL_WIDTH_80 && (chset->chs[chset_idx].flags & RTW_CHF_NO_80MHZ))
break;
if (bw >= CHANNEL_WIDTH_160 && (chset->chs[chset_idx].flags & RTW_CHF_NO_160MHZ))
break;
if ((chset->chs[chset_idx].flags & RTW_CHF_DFS) && !rtw_rfctl_radar_detect_supported(rfctl))
continue;
if (chset->chs[chset_idx].flags & RTW_CHF_NO_IR)
continue;
l++;
}
if (k < op_ch_num)
continue;
if (reason >= REG_CHANGE) {
opc_pref->chs[j].static_non_op = 0;
opc_pref->op_ch_num++;
}
if (l >= op_ch_num) {
opc_pref->chs[j].no_ir = 0;
opc_pref->ir_ch_num++;
}
}
if (opc_pref->op_ch_num)
reg_op_class_num++;
if (opc_pref->ir_ch_num)
op_class_num++;
}
rfctl->reg_spt_op_class_num = reg_op_class_num;
rfctl->cur_spt_op_class_num = op_class_num;
}
static void dump_opc_pref_single(void *sel, struct op_class_pref_t *opc_pref, bool show_snon_ocp, bool show_no_ir, bool detail)
{
u8 i;
u8 ch_num = 0;
if (!show_snon_ocp && !opc_pref->op_ch_num)
return;
if (!show_no_ir && !opc_pref->ir_ch_num)
return;
RTW_PRINT_SEL(sel, "%5u %4s %7s"
, opc_pref->class_id
, band_str(opc_pref->band)
, opc_bw_str(opc_pref->bw));
for (i = 0; i < opc_pref->ch_num; i++) {
if ((show_snon_ocp || !opc_pref->chs[i].static_non_op)
&& (show_no_ir || !opc_pref->chs[i].no_ir)
) {
if (detail)
_RTW_PRINT_SEL(sel, " %4u", opc_pref->chs[i].ch);
else
_RTW_PRINT_SEL(sel, " %u", opc_pref->chs[i].ch);
}
}
_RTW_PRINT_SEL(sel, "\n");
if (!detail)
return;
RTW_PRINT_SEL(sel, " ");
for (i = 0; i < opc_pref->ch_num; i++) {
if ((show_snon_ocp || !opc_pref->chs[i].static_non_op)
&& (show_no_ir || !opc_pref->chs[i].no_ir)
) {
_RTW_PRINT_SEL(sel, " %c%c"
, opc_pref->chs[i].no_ir ? ' ' : 'I'
, opc_pref->chs[i].static_non_op ? ' ' : 'E'
);
}
}
_RTW_PRINT_SEL(sel, "\n");
RTW_PRINT_SEL(sel, " ");
for (i = 0; i < opc_pref->ch_num; i++) {
if ((show_snon_ocp || !opc_pref->chs[i].static_non_op)
&& (show_no_ir || !opc_pref->chs[i].no_ir)
) {
if (opc_pref->chs[i].max_txpwr == UNSPECIFIED_MBM)
_RTW_PRINT_SEL(sel, " ");
else
_RTW_PRINT_SEL(sel, " %4d", opc_pref->chs[i].max_txpwr);
}
}
_RTW_PRINT_SEL(sel, "\n");
}
void dump_cap_spt_op_class_ch(void *sel, struct rf_ctl_t *rfctl, bool detail)
{
u8 i;
dump_op_class_ch_title(sel);
for (i = 0; i < global_op_class_num; i++) {
if (!rfctl->spt_op_class_ch[i])
continue;
dump_opc_pref_single(sel, rfctl->spt_op_class_ch[i], 1, 1, detail);
}
RTW_PRINT_SEL(sel, "op_class number:%d\n", rfctl->cap_spt_op_class_num);
}
void dump_reg_spt_op_class_ch(void *sel, struct rf_ctl_t *rfctl, bool detail)
{
u8 i;
dump_op_class_ch_title(sel);
for (i = 0; i < global_op_class_num; i++) {
if (!rfctl->spt_op_class_ch[i])
continue;
dump_opc_pref_single(sel, rfctl->spt_op_class_ch[i], 0, 1, detail);
}
RTW_PRINT_SEL(sel, "op_class number:%d\n", rfctl->reg_spt_op_class_num);
}
void dump_cur_spt_op_class_ch(void *sel, struct rf_ctl_t *rfctl, bool detail)
{
u8 i;
dump_op_class_ch_title(sel);
for (i = 0; i < global_op_class_num; i++) {
if (!rfctl->spt_op_class_ch[i])
continue;
dump_opc_pref_single(sel, rfctl->spt_op_class_ch[i], 0, 0, detail);
}
RTW_PRINT_SEL(sel, "op_class number:%d\n", rfctl->cur_spt_op_class_num);
}
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 _rf_type_to_rfpath_str[] = {
"RF_1T1R",
"RF_1T2R",
"RF_2T2R",
"RF_2T3R",
"RF_2T4R",
"RF_3T3R",
"RF_3T4R",
"RF_4T4R",
"RF_TYPE_MAX"
};
static const u8 _trx_num_to_rf_type[RF_PATH_MAX][RF_PATH_MAX] = {
{RF_1T1R, RF_1T2R, RF_TYPE_MAX, RF_TYPE_MAX},
{RF_TYPE_MAX, RF_2T2R, RF_2T3R, RF_2T4R},
{RF_TYPE_MAX, RF_TYPE_MAX, RF_3T3R, RF_3T4R},
{RF_TYPE_MAX, RF_TYPE_MAX, RF_TYPE_MAX, RF_4T4R},
};
enum rf_type trx_num_to_rf_type(u8 tx_num, u8 rx_num)
{
if (tx_num > 0 && tx_num <= RF_PATH_MAX && rx_num > 0 && rx_num <= RF_PATH_MAX)
return _trx_num_to_rf_type[tx_num - 1][rx_num - 1];
return RF_TYPE_MAX;
}
enum rf_type trx_bmp_to_rf_type(u8 tx_bmp, u8 rx_bmp)
{
u8 tx_num = 0;
u8 rx_num = 0;
int i;
for (i = 0; i < RF_PATH_MAX; i++) {
if (tx_bmp >> i & BIT0)
tx_num++;
if (rx_bmp >> i & BIT0)
rx_num++;
}
return trx_num_to_rf_type(tx_num, rx_num);
}
bool rf_type_is_a_in_b(enum rf_type a, enum rf_type b)
{
return rf_type_to_rf_tx_cnt(a) <= rf_type_to_rf_tx_cnt(b)
&& rf_type_to_rf_rx_cnt(a) <= rf_type_to_rf_rx_cnt(b);
}
static void rtw_path_bmp_limit_from_higher(u8 *bmp, u8 *bmp_bit_cnt, u8 bit_cnt_lmt)
{
int i;
for (i = RF_PATH_MAX - 1; *bmp_bit_cnt > bit_cnt_lmt && i >= 0; i--) {
if (*bmp & BIT(i)) {
*bmp &= ~BIT(i);
(*bmp_bit_cnt)--;
}
}
}
u8 rtw_restrict_trx_path_bmp_by_rftype(u8 trx_path_bmp, enum rf_type type, u8 *tx_num, u8 *rx_num)
{
u8 bmp_tx = (trx_path_bmp & 0xF0) >> 4;
u8 bmp_rx = trx_path_bmp & 0x0F;
u8 bmp_tx_num = 0, bmp_rx_num = 0;
u8 tx_num_lmt, rx_num_lmt;
enum rf_type ret_type = RF_TYPE_MAX;
int i, j;
for (i = 0; i < RF_PATH_MAX; i++) {
if (bmp_tx & BIT(i))
bmp_tx_num++;
if (bmp_rx & BIT(i))
bmp_rx_num++;
}
/* limit higher bit first according to input type */
tx_num_lmt = rf_type_to_rf_tx_cnt(type);
rx_num_lmt = rf_type_to_rf_rx_cnt(type);
rtw_path_bmp_limit_from_higher(&bmp_tx, &bmp_tx_num, tx_num_lmt);
rtw_path_bmp_limit_from_higher(&bmp_rx, &bmp_rx_num, rx_num_lmt);
/* search for valid rf_type (larger RX prefer) */
for (j = bmp_rx_num; j > 0; j--) {
for (i = bmp_tx_num; i > 0; i--) {
ret_type = trx_num_to_rf_type(i, j);
if (RF_TYPE_VALID(ret_type)) {
rtw_path_bmp_limit_from_higher(&bmp_tx, &bmp_tx_num, i);
rtw_path_bmp_limit_from_higher(&bmp_rx, &bmp_rx_num, j);
if (tx_num)
*tx_num = bmp_tx_num;
if (rx_num)
*rx_num = bmp_rx_num;
goto exit;
}
}
}
exit:
return RF_TYPE_VALID(ret_type) ? ((bmp_tx << 4) | bmp_rx) : 0x00;
}
/*
* input with txpwr value in unit of txpwr index
* return string in length 6 at least (for -xx.xx)
*/
void txpwr_idx_get_dbm_str(s8 idx, u8 txgi_max, s8 txgi_ww, u8 txgi_pdbm, SIZE_T cwidth, char dbm_str[], u8 dbm_str_len)
{
char fmt[16];
if (idx == txgi_max) {
snprintf(fmt, 16, "%%%zus", cwidth >= 6 ? cwidth + 1 : 6);
snprintf(dbm_str, dbm_str_len, fmt, "NA");
} else if (idx == txgi_ww) {
snprintf(fmt, 16, "%%%zus", cwidth >= 6 ? cwidth + 1 : 6);
snprintf(dbm_str, dbm_str_len, fmt, "WW");
} else if (idx > -txgi_pdbm && idx < 0) { /* -0.xx */
snprintf(fmt, 16, "%%%zus-0.%%02d", cwidth >= 6 ? cwidth - 4 : 1);
snprintf(dbm_str, dbm_str_len, fmt, "", (rtw_abs(idx) % txgi_pdbm) * 100 / txgi_pdbm);
} else if (idx % txgi_pdbm) { /* d.xx */
snprintf(fmt, 16, "%%%zud.%%02d", cwidth >= 6 ? cwidth - 2 : 3);
snprintf(dbm_str, dbm_str_len, fmt, idx / txgi_pdbm, (rtw_abs(idx) % txgi_pdbm) * 100 / txgi_pdbm);
} else { /* d */
snprintf(fmt, 16, "%%%zud", cwidth >= 6 ? cwidth + 1 : 6);
snprintf(dbm_str, dbm_str_len, fmt, idx / txgi_pdbm);
}
}
/*
* input with txpwr value in unit of mbm
* return string in length 6 at least (for -xx.xx)
*/
void txpwr_mbm_get_dbm_str(s16 mbm, SIZE_T cwidth, char dbm_str[], u8 dbm_str_len)
{
char fmt[16];
if (mbm == UNSPECIFIED_MBM) {
snprintf(fmt, 16, "%%%zus", cwidth >= 6 ? cwidth + 1 : 6);
snprintf(dbm_str, dbm_str_len, fmt, "NA");
} else if (mbm > -MBM_PDBM && mbm < 0) { /* -0.xx */
snprintf(fmt, 16, "%%%zus-0.%%02d", cwidth >= 6 ? cwidth - 4 : 1);
snprintf(dbm_str, dbm_str_len, fmt, "", (rtw_abs(mbm) % MBM_PDBM) * 100 / MBM_PDBM);
} else if (mbm % MBM_PDBM) { /* d.xx */
snprintf(fmt, 16, "%%%zud.%%02d", cwidth >= 6 ? cwidth - 2 : 3);
snprintf(dbm_str, dbm_str_len, fmt, mbm / MBM_PDBM, (rtw_abs(mbm) % MBM_PDBM) * 100 / MBM_PDBM);
} else { /* d */
snprintf(fmt, 16, "%%%zud", cwidth >= 6 ? cwidth + 1 : 6);
snprintf(dbm_str, dbm_str_len, fmt, mbm / MBM_PDBM);
}
}
static const s16 _mb_of_ntx[] = {
0, /* 1TX */
301, /* 2TX */
477, /* 3TX */
602, /* 4TX */
699, /* 5TX */
778, /* 6TX */
845, /* 7TX */
903, /* 8TX */
};
/* get mB(100 *dB) for specifc TX count relative to 1TX */
s16 mb_of_ntx(u8 ntx)
{
if (ntx == 0 || ntx > 8) {
RTW_ERR("ntx=%u, out of range\n", ntx);
rtw_warn_on(1);
}
return _mb_of_ntx[ntx - 1];
}
#if CONFIG_TXPWR_LIMIT
void dump_regd_exc_list(void *sel, struct rf_ctl_t *rfctl)
{
/* TODO: get from phl */
}
#endif /* CONFIG_TXPWR_LIMIT */
bool rtw_is_long_cac_range(u32 hi, u32 lo, u8 dfs_region)
{
return (dfs_region == RTW_DFS_REGD_ETSI && rtw_is_range_overlap(hi, lo, 5650, 5600)) ? _TRUE : _FALSE;
}
#if CONFIG_ALLOW_FUNC_2G_5G_ONLY
RTW_FUNC_2G_5G_ONLY 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;
}
#endif
bool rtw_is_long_cac_bch(enum band_type band, u8 ch, u8 bw, u8 offset, u8 dfs_region)
{
u32 hi, lo;
if (rtw_bchbw_to_freq_range(band, ch, bw, offset, &hi, &lo) == _FALSE)
return _FALSE;
return rtw_is_long_cac_range(hi, lo, dfs_region) ? _TRUE : _FALSE;
}