wifi: ath11k: add parse of transmit power envelope element
The transmit power envelope element has some fields for power, ath11k should parse it according to IEEE Std 802.11ax™‐2021. Tested-on: WCN6855 hw2.0 PCI WLAN.HSP.1.1-03125-QCAHSPSWPL_V1_V2_SILICONZ_LITE-3.6510.23 Signed-off-by: Wen Gong <quic_wgong@quicinc.com> Acked-by: Jeff Johnson <quic_jjohnson@quicinc.com> Signed-off-by: Baochen Qiang <quic_bqiang@quicinc.com> Signed-off-by: Kalle Valo <quic_kvalo@quicinc.com> Link: https://msgid.link/20231218085844.2658-8-quic_bqiang@quicinc.com
This commit is contained in:
parent
28f64d368b
commit
6f4e235be6
2 changed files with 227 additions and 0 deletions
|
@ -314,6 +314,43 @@ struct ath11k_rekey_data {
|
||||||
bool enable_offload;
|
bool enable_offload;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct ath11k_chan_power_info - TPE containing power info per channel chunk
|
||||||
|
* @chan_cfreq: channel center freq (MHz)
|
||||||
|
* e.g.
|
||||||
|
* channel 37/20 MHz, it is 6135
|
||||||
|
* channel 37/40 MHz, it is 6125
|
||||||
|
* channel 37/80 MHz, it is 6145
|
||||||
|
* channel 37/160 MHz, it is 6185
|
||||||
|
* @tx_power: transmit power (dBm)
|
||||||
|
*/
|
||||||
|
struct ath11k_chan_power_info {
|
||||||
|
u16 chan_cfreq;
|
||||||
|
s8 tx_power;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct ath11k_reg_tpc_power_info - regulatory TPC power info
|
||||||
|
* @is_psd_power: is PSD power or not
|
||||||
|
* @eirp_power: Maximum EIRP power (dBm), valid only if power is PSD
|
||||||
|
* @ap_power_type: type of power (SP/LPI/VLP)
|
||||||
|
* @num_pwr_levels: number of power levels
|
||||||
|
* @reg_max: Array of maximum TX power (dBm) per PSD value
|
||||||
|
* @ap_constraint_power: AP constraint power (dBm)
|
||||||
|
* @tpe: TPE values processed from TPE IE
|
||||||
|
* @chan_power_info: power info to send to firmware
|
||||||
|
*/
|
||||||
|
struct ath11k_reg_tpc_power_info {
|
||||||
|
bool is_psd_power;
|
||||||
|
u8 eirp_power;
|
||||||
|
enum wmi_reg_6ghz_ap_type ap_power_type;
|
||||||
|
u8 num_pwr_levels;
|
||||||
|
u8 reg_max[IEEE80211_MAX_NUM_PWR_LEVEL];
|
||||||
|
u8 ap_constraint_power;
|
||||||
|
s8 tpe[IEEE80211_MAX_NUM_PWR_LEVEL];
|
||||||
|
struct ath11k_chan_power_info chan_power_info[IEEE80211_MAX_NUM_PWR_LEVEL];
|
||||||
|
};
|
||||||
|
|
||||||
struct ath11k_vif {
|
struct ath11k_vif {
|
||||||
u32 vdev_id;
|
u32 vdev_id;
|
||||||
enum wmi_vdev_type vdev_type;
|
enum wmi_vdev_type vdev_type;
|
||||||
|
@ -369,6 +406,8 @@ struct ath11k_vif {
|
||||||
struct ath11k_arp_ns_offload arp_ns_offload;
|
struct ath11k_arp_ns_offload arp_ns_offload;
|
||||||
struct ath11k_rekey_data rekey_data;
|
struct ath11k_rekey_data rekey_data;
|
||||||
|
|
||||||
|
struct ath11k_reg_tpc_power_info reg_tpc_info;
|
||||||
|
|
||||||
#ifdef CONFIG_ATH11K_DEBUGFS
|
#ifdef CONFIG_ATH11K_DEBUGFS
|
||||||
struct dentry *debugfs_twt;
|
struct dentry *debugfs_twt;
|
||||||
#endif /* CONFIG_ATH11K_DEBUGFS */
|
#endif /* CONFIG_ATH11K_DEBUGFS */
|
||||||
|
|
|
@ -7608,6 +7608,192 @@ static int ath11k_start_vdev_delay(struct ieee80211_hw *hw,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u8 ath11k_mac_get_tpe_count(u8 txpwr_intrprt, u8 txpwr_cnt)
|
||||||
|
{
|
||||||
|
switch (txpwr_intrprt) {
|
||||||
|
/* Refer "Table 9-276-Meaning of Maximum Transmit Power Count subfield
|
||||||
|
* if the Maximum Transmit Power Interpretation subfield is 0 or 2" of
|
||||||
|
* "IEEE Std 802.11ax 2021".
|
||||||
|
*/
|
||||||
|
case IEEE80211_TPE_LOCAL_EIRP:
|
||||||
|
case IEEE80211_TPE_REG_CLIENT_EIRP:
|
||||||
|
txpwr_cnt = txpwr_cnt <= 3 ? txpwr_cnt : 3;
|
||||||
|
txpwr_cnt = txpwr_cnt + 1;
|
||||||
|
break;
|
||||||
|
/* Refer "Table 9-277-Meaning of Maximum Transmit Power Count subfield
|
||||||
|
* if Maximum Transmit Power Interpretation subfield is 1 or 3" of
|
||||||
|
* "IEEE Std 802.11ax 2021".
|
||||||
|
*/
|
||||||
|
case IEEE80211_TPE_LOCAL_EIRP_PSD:
|
||||||
|
case IEEE80211_TPE_REG_CLIENT_EIRP_PSD:
|
||||||
|
txpwr_cnt = txpwr_cnt <= 4 ? txpwr_cnt : 4;
|
||||||
|
txpwr_cnt = txpwr_cnt ? (BIT(txpwr_cnt - 1)) : 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return txpwr_cnt;
|
||||||
|
}
|
||||||
|
|
||||||
|
static u8 ath11k_mac_get_num_pwr_levels(struct cfg80211_chan_def *chan_def)
|
||||||
|
{
|
||||||
|
if (chan_def->chan->flags & IEEE80211_CHAN_PSD) {
|
||||||
|
switch (chan_def->width) {
|
||||||
|
case NL80211_CHAN_WIDTH_20:
|
||||||
|
return 1;
|
||||||
|
case NL80211_CHAN_WIDTH_40:
|
||||||
|
return 2;
|
||||||
|
case NL80211_CHAN_WIDTH_80:
|
||||||
|
return 4;
|
||||||
|
case NL80211_CHAN_WIDTH_80P80:
|
||||||
|
case NL80211_CHAN_WIDTH_160:
|
||||||
|
return 8;
|
||||||
|
default:
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
switch (chan_def->width) {
|
||||||
|
case NL80211_CHAN_WIDTH_20:
|
||||||
|
return 1;
|
||||||
|
case NL80211_CHAN_WIDTH_40:
|
||||||
|
return 2;
|
||||||
|
case NL80211_CHAN_WIDTH_80:
|
||||||
|
return 3;
|
||||||
|
case NL80211_CHAN_WIDTH_80P80:
|
||||||
|
case NL80211_CHAN_WIDTH_160:
|
||||||
|
return 4;
|
||||||
|
default:
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ath11k_mac_parse_tx_pwr_env(struct ath11k *ar,
|
||||||
|
struct ieee80211_vif *vif,
|
||||||
|
struct ieee80211_chanctx_conf *ctx)
|
||||||
|
{
|
||||||
|
struct ath11k_base *ab = ar->ab;
|
||||||
|
struct ath11k_vif *arvif = (void *)vif->drv_priv;
|
||||||
|
struct ieee80211_bss_conf *bss_conf = &vif->bss_conf;
|
||||||
|
struct ieee80211_tx_pwr_env *single_tpe;
|
||||||
|
enum wmi_reg_6ghz_client_type client_type;
|
||||||
|
struct cur_regulatory_info *reg_info;
|
||||||
|
int i;
|
||||||
|
u8 pwr_count, pwr_interpret, pwr_category;
|
||||||
|
u8 psd_index = 0, non_psd_index = 0, local_tpe_count = 0, reg_tpe_count = 0;
|
||||||
|
bool use_local_tpe, non_psd_set = false, psd_set = false;
|
||||||
|
|
||||||
|
reg_info = &ab->reg_info_store[ar->pdev_idx];
|
||||||
|
client_type = reg_info->client_type;
|
||||||
|
|
||||||
|
for (i = 0; i < bss_conf->tx_pwr_env_num; i++) {
|
||||||
|
single_tpe = &bss_conf->tx_pwr_env[i];
|
||||||
|
pwr_category = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_CATEGORY);
|
||||||
|
pwr_interpret = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_INTERPRET);
|
||||||
|
|
||||||
|
if (pwr_category == client_type) {
|
||||||
|
if (pwr_interpret == IEEE80211_TPE_LOCAL_EIRP ||
|
||||||
|
pwr_interpret == IEEE80211_TPE_LOCAL_EIRP_PSD)
|
||||||
|
local_tpe_count++;
|
||||||
|
else if (pwr_interpret == IEEE80211_TPE_REG_CLIENT_EIRP ||
|
||||||
|
pwr_interpret == IEEE80211_TPE_REG_CLIENT_EIRP_PSD)
|
||||||
|
reg_tpe_count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!reg_tpe_count && !local_tpe_count) {
|
||||||
|
ath11k_warn(ab,
|
||||||
|
"no transmit power envelope match client power type %d\n",
|
||||||
|
client_type);
|
||||||
|
return;
|
||||||
|
} else if (!reg_tpe_count) {
|
||||||
|
use_local_tpe = true;
|
||||||
|
} else {
|
||||||
|
use_local_tpe = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < bss_conf->tx_pwr_env_num; i++) {
|
||||||
|
single_tpe = &bss_conf->tx_pwr_env[i];
|
||||||
|
pwr_category = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_CATEGORY);
|
||||||
|
pwr_interpret = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_INTERPRET);
|
||||||
|
|
||||||
|
if (pwr_category != client_type)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* get local transmit power envelope */
|
||||||
|
if (use_local_tpe) {
|
||||||
|
if (pwr_interpret == IEEE80211_TPE_LOCAL_EIRP) {
|
||||||
|
non_psd_index = i;
|
||||||
|
non_psd_set = true;
|
||||||
|
} else if (pwr_interpret == IEEE80211_TPE_LOCAL_EIRP_PSD) {
|
||||||
|
psd_index = i;
|
||||||
|
psd_set = true;
|
||||||
|
}
|
||||||
|
/* get regulatory transmit power envelope */
|
||||||
|
} else {
|
||||||
|
if (pwr_interpret == IEEE80211_TPE_REG_CLIENT_EIRP) {
|
||||||
|
non_psd_index = i;
|
||||||
|
non_psd_set = true;
|
||||||
|
} else if (pwr_interpret == IEEE80211_TPE_REG_CLIENT_EIRP_PSD) {
|
||||||
|
psd_index = i;
|
||||||
|
psd_set = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (non_psd_set && !psd_set) {
|
||||||
|
single_tpe = &bss_conf->tx_pwr_env[non_psd_index];
|
||||||
|
pwr_count = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_COUNT);
|
||||||
|
pwr_interpret = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_INTERPRET);
|
||||||
|
arvif->reg_tpc_info.is_psd_power = false;
|
||||||
|
arvif->reg_tpc_info.eirp_power = 0;
|
||||||
|
|
||||||
|
arvif->reg_tpc_info.num_pwr_levels =
|
||||||
|
ath11k_mac_get_tpe_count(pwr_interpret, pwr_count);
|
||||||
|
|
||||||
|
for (i = 0; i < arvif->reg_tpc_info.num_pwr_levels; i++) {
|
||||||
|
ath11k_dbg(ab, ATH11K_DBG_MAC,
|
||||||
|
"non PSD power[%d] : %d\n",
|
||||||
|
i, single_tpe->tx_power[i]);
|
||||||
|
arvif->reg_tpc_info.tpe[i] = single_tpe->tx_power[i] / 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (psd_set) {
|
||||||
|
single_tpe = &bss_conf->tx_pwr_env[psd_index];
|
||||||
|
pwr_count = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_COUNT);
|
||||||
|
pwr_interpret = u8_get_bits(single_tpe->tx_power_info,
|
||||||
|
IEEE80211_TX_PWR_ENV_INFO_INTERPRET);
|
||||||
|
arvif->reg_tpc_info.is_psd_power = true;
|
||||||
|
|
||||||
|
if (pwr_count == 0) {
|
||||||
|
ath11k_dbg(ab, ATH11K_DBG_MAC,
|
||||||
|
"TPE PSD power : %d\n", single_tpe->tx_power[0]);
|
||||||
|
arvif->reg_tpc_info.num_pwr_levels =
|
||||||
|
ath11k_mac_get_num_pwr_levels(&ctx->def);
|
||||||
|
|
||||||
|
for (i = 0; i < arvif->reg_tpc_info.num_pwr_levels; i++)
|
||||||
|
arvif->reg_tpc_info.tpe[i] = single_tpe->tx_power[0] / 2;
|
||||||
|
} else {
|
||||||
|
arvif->reg_tpc_info.num_pwr_levels =
|
||||||
|
ath11k_mac_get_tpe_count(pwr_interpret, pwr_count);
|
||||||
|
|
||||||
|
for (i = 0; i < arvif->reg_tpc_info.num_pwr_levels; i++) {
|
||||||
|
ath11k_dbg(ab, ATH11K_DBG_MAC,
|
||||||
|
"TPE PSD power[%d] : %d\n",
|
||||||
|
i, single_tpe->tx_power[i]);
|
||||||
|
arvif->reg_tpc_info.tpe[i] = single_tpe->tx_power[i] / 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
ath11k_mac_op_assign_vif_chanctx(struct ieee80211_hw *hw,
|
ath11k_mac_op_assign_vif_chanctx(struct ieee80211_hw *hw,
|
||||||
struct ieee80211_vif *vif,
|
struct ieee80211_vif *vif,
|
||||||
|
@ -7642,6 +7828,8 @@ ath11k_mac_op_assign_vif_chanctx(struct ieee80211_hw *hw,
|
||||||
}
|
}
|
||||||
|
|
||||||
ath11k_reg_handle_chan_list(ab, reg_info, power_type);
|
ath11k_reg_handle_chan_list(ab, reg_info, power_type);
|
||||||
|
|
||||||
|
ath11k_mac_parse_tx_pwr_env(ar, vif, ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* for QCA6390 bss peer must be created before vdev_start */
|
/* for QCA6390 bss peer must be created before vdev_start */
|
||||||
|
|
Loading…
Add table
Reference in a new issue