Skip to content

Commit 7304f47

Browse files
authored
wifi: ath12k: Add pdev-level OBSS PD configuration (#363)
wifi: ath12k: Add pdev-level OBSS PD configuration
2 parents e4cae56 + 8ae32da commit 7304f47

4 files changed

Lines changed: 359 additions & 3 deletions

File tree

drivers/net/wireless/ath/ath12k/mac.c

Lines changed: 167 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -4507,6 +4507,166 @@ static void ath12k_wmi_vdev_params_up(struct ath12k *ar,
45074507
arvif->vdev_id, ret);
45084508
}
45094509

4510+
static int ath12k_mac_config_obss_pd(struct ath12k_link_vif *arvif,
4511+
const struct ieee80211_he_obss_pd *he_obss_pd)
4512+
{
4513+
struct ath12k_wmi_obss_pd_arg obss_pd_arg = {};
4514+
u32 srg_bitmap[2], non_srg_bitmap[2];
4515+
struct ath12k *ar = arvif->ar;
4516+
u32 param_id, pdev_id;
4517+
u32 param_val;
4518+
int ret;
4519+
4520+
if (ar->ab->hw_params->single_pdev_only)
4521+
pdev_id = ath12k_mac_get_target_pdev_id_from_vif(arvif);
4522+
else
4523+
pdev_id = ar->pdev->pdev_id;
4524+
4525+
/* Set and enable SRG/non-SRG OBSS PD threshold */
4526+
param_id = WMI_PDEV_PARAM_SET_CMD_OBSS_PD_THRESHOLD;
4527+
if (ar->monitor_started || !he_obss_pd->enable) {
4528+
ret = ath12k_wmi_pdev_set_param(ar, param_id, 0, pdev_id);
4529+
if (ret)
4530+
ath12k_warn(ar->ab,
4531+
"failed to set OBSS PD threshold for pdev %u: %d\n",
4532+
pdev_id, ret);
4533+
return ret;
4534+
}
4535+
4536+
/*
4537+
* This service flag indicates firmware support for SRG/SRP-based
4538+
* spatial reuse. It also specifies whether OBSS PD threshold values
4539+
* should be interpreted as dB (offset) or dBm (absolute) units.
4540+
*/
4541+
obss_pd_arg.srp_support = test_bit(WMI_TLV_SERVICE_SRG_SRP_SPATIAL_REUSE_SUPPORT,
4542+
ar->ab->wmi_ab.svc_map);
4543+
4544+
if (!(he_obss_pd->sr_ctrl &
4545+
IEEE80211_HE_SPR_NON_SRG_OBSS_PD_SR_DISALLOWED)) {
4546+
if (he_obss_pd->sr_ctrl & IEEE80211_HE_SPR_NON_SRG_OFFSET_PRESENT)
4547+
obss_pd_arg.non_srg_th = ATH12K_OBSS_PD_MAX_THRESHOLD +
4548+
he_obss_pd->non_srg_max_offset;
4549+
else
4550+
obss_pd_arg.non_srg_th = ATH12K_OBSS_PD_NON_SRG_MAX_THRESHOLD;
4551+
4552+
if (!obss_pd_arg.srp_support)
4553+
obss_pd_arg.non_srg_th -= ATH12K_DEFAULT_NOISE_FLOOR;
4554+
4555+
obss_pd_arg.non_srg_enabled = true;
4556+
}
4557+
4558+
if (he_obss_pd->sr_ctrl & IEEE80211_HE_SPR_SRG_INFORMATION_PRESENT) {
4559+
obss_pd_arg.srg_th = ATH12K_OBSS_PD_MAX_THRESHOLD +
4560+
he_obss_pd->max_offset;
4561+
obss_pd_arg.srg_enabled = true;
4562+
}
4563+
4564+
ath12k_dbg(ar->ab, ATH12K_DBG_MAC,
4565+
"pdev %u OBSS PD sr_ctrl 0x%x srg_th %d dBm non_srg_th %d dBm\n",
4566+
pdev_id, he_obss_pd->sr_ctrl,
4567+
obss_pd_arg.srg_th, obss_pd_arg.non_srg_th);
4568+
4569+
param_val = ath12k_wmi_build_obss_pd(&obss_pd_arg);
4570+
ret = ath12k_wmi_pdev_set_param(ar, param_id, param_val, pdev_id);
4571+
if (ret) {
4572+
ath12k_warn(ar->ab,
4573+
"failed to set OBSS PD threshold for pdev %u: %d\n",
4574+
pdev_id, ret);
4575+
return ret;
4576+
}
4577+
4578+
/* Enable OBSS PD for all access category */
4579+
param_id = WMI_PDEV_PARAM_SET_CMD_OBSS_PD_PER_AC;
4580+
param_val = 0xf;
4581+
ret = ath12k_wmi_pdev_set_param(ar, param_id, param_val, pdev_id);
4582+
if (ret) {
4583+
ath12k_warn(ar->ab,
4584+
"failed to set OBSS PD per ac for pdev %u: %d\n",
4585+
pdev_id, ret);
4586+
return ret;
4587+
}
4588+
4589+
/* Set SR prohibit */
4590+
param_id = WMI_PDEV_PARAM_ENABLE_SR_PROHIBIT;
4591+
param_val = !!(he_obss_pd->sr_ctrl &
4592+
IEEE80211_HE_SPR_HESIGA_SR_VAL15_ALLOWED);
4593+
ret = ath12k_wmi_pdev_set_param(ar, param_id, param_val, pdev_id);
4594+
if (ret) {
4595+
ath12k_warn(ar->ab, "failed to set SR prohibit for pdev %u: %d\n",
4596+
pdev_id, ret);
4597+
return ret;
4598+
}
4599+
4600+
if (!obss_pd_arg.srp_support)
4601+
return 0;
4602+
4603+
memcpy(srg_bitmap, he_obss_pd->bss_color_bitmap, sizeof(srg_bitmap));
4604+
/* Set SRG BSS color bitmap */
4605+
ret = ath12k_wmi_pdev_set_srg_bss_color_bitmap(ar, pdev_id, srg_bitmap);
4606+
if (ret) {
4607+
ath12k_warn(ar->ab,
4608+
"failed to set SRG bss color bitmap for pdev %u: %d\n",
4609+
pdev_id, ret);
4610+
return ret;
4611+
}
4612+
4613+
/* Enable BSS colors for SRG */
4614+
ret = ath12k_wmi_pdev_srg_obss_color_enable_bitmap(ar, pdev_id, srg_bitmap);
4615+
if (ret) {
4616+
ath12k_warn(ar->ab,
4617+
"failed to enable SRG bss color bitmap pdev %u: %d\n",
4618+
pdev_id, ret);
4619+
return ret;
4620+
}
4621+
4622+
memcpy(srg_bitmap, he_obss_pd->partial_bssid_bitmap, sizeof(srg_bitmap));
4623+
/* Set SRG partial bssid bitmap */
4624+
ret = ath12k_wmi_pdev_set_srg_partial_bssid_bitmap(ar, pdev_id, srg_bitmap);
4625+
if (ret) {
4626+
ath12k_warn(ar->ab,
4627+
"failed to set SRG partial bssid bitmap for pdev %u: %d\n",
4628+
pdev_id, ret);
4629+
return ret;
4630+
}
4631+
4632+
/* Enable partial bssid mask for SRG */
4633+
ret = ath12k_wmi_pdev_srg_obss_bssid_enable_bitmap(ar, pdev_id, srg_bitmap);
4634+
if (ret) {
4635+
ath12k_warn(ar->ab,
4636+
"failed to enable SRG bssid bitmap pdev %u: %d\n",
4637+
pdev_id, ret);
4638+
return ret;
4639+
}
4640+
4641+
/*
4642+
* No explicit non-SRG bitmap from mac80211; enable all colors/bssids
4643+
* as non-SRG candidates. Actual SRG members are filtered by SRG bitmaps.
4644+
*/
4645+
memset(non_srg_bitmap, 0xff, sizeof(non_srg_bitmap));
4646+
4647+
/* Enable BSS colors for non-SRG */
4648+
ret = ath12k_wmi_pdev_non_srg_obss_color_enable_bitmap(ar, pdev_id,
4649+
non_srg_bitmap);
4650+
if (ret) {
4651+
ath12k_warn(ar->ab,
4652+
"failed to enable non SRG color bitmap pdev %u: %d\n",
4653+
pdev_id, ret);
4654+
return ret;
4655+
}
4656+
4657+
/* Enable partial bssid mask for non-SRG */
4658+
ret = ath12k_wmi_pdev_non_srg_obss_bssid_enable_bitmap(ar, pdev_id,
4659+
non_srg_bitmap);
4660+
if (ret) {
4661+
ath12k_warn(ar->ab,
4662+
"failed to enable non SRG bssid bitmap pdev %u: %d\n",
4663+
pdev_id, ret);
4664+
return ret;
4665+
}
4666+
4667+
return 0;
4668+
}
4669+
45104670
static void ath12k_mac_bss_info_changed(struct ath12k *ar,
45114671
struct ath12k_link_vif *arvif,
45124672
struct ieee80211_bss_conf *info,
@@ -4798,9 +4958,13 @@ static void ath12k_mac_bss_info_changed(struct ath12k *ar,
47984958
ath12k_wmi_send_twt_disable_cmd(ar, ar->pdev->pdev_id);
47994959
}
48004960

4801-
if (changed & BSS_CHANGED_HE_OBSS_PD)
4802-
ath12k_wmi_send_obss_spr_cmd(ar, arvif->vdev_id,
4803-
&info->he_obss_pd);
4961+
if (changed & BSS_CHANGED_HE_OBSS_PD) {
4962+
if (vif->type == NL80211_IFTYPE_AP)
4963+
ath12k_mac_config_obss_pd(arvif, &info->he_obss_pd);
4964+
else
4965+
ath12k_wmi_send_obss_spr_cmd(ar, arvif->vdev_id,
4966+
&info->he_obss_pd);
4967+
}
48044968

48054969
if (changed & BSS_CHANGED_HE_BSS_COLOR) {
48064970
if (vif->type == NL80211_IFTYPE_AP) {

drivers/net/wireless/ath/ath12k/mac.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -138,6 +138,9 @@ struct ath12k_reg_tpc_power_info {
138138
struct ath12k_chan_power_info chan_power_info[ATH12K_NUM_PWR_LEVELS];
139139
};
140140

141+
#define ATH12K_OBSS_PD_MAX_THRESHOLD -82
142+
#define ATH12K_OBSS_PD_NON_SRG_MAX_THRESHOLD -62
143+
141144
extern const struct htt_rx_ring_tlv_filter ath12k_mac_mon_status_filter_default;
142145

143146
#define ATH12K_SCAN_11D_INTERVAL 600000

drivers/net/wireless/ath/ath12k/wmi.c

Lines changed: 142 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -126,6 +126,14 @@ struct wmi_tlv_mgmt_rx_parse {
126126
bool frame_buf_done;
127127
};
128128

129+
struct wmi_pdev_set_obss_bitmap_arg {
130+
u32 tlv_tag;
131+
u32 pdev_id;
132+
u32 cmd_id;
133+
const u32 *bitmap;
134+
const char *label;
135+
};
136+
129137
static const struct ath12k_wmi_tlv_policy ath12k_wmi_tlv_policies[] = {
130138
[WMI_TAG_ARRAY_BYTE] = { .min_len = 0 },
131139
[WMI_TAG_ARRAY_UINT32] = { .min_len = 0 },
@@ -3560,6 +3568,140 @@ ath12k_wmi_send_obss_spr_cmd(struct ath12k *ar, u32 vdev_id,
35603568
return ret;
35613569
}
35623570

3571+
u32 ath12k_wmi_build_obss_pd(const struct ath12k_wmi_obss_pd_arg *arg)
3572+
{
3573+
u32 param_val = 0;
3574+
3575+
param_val |= u32_encode_bits((u8)arg->srg_th, GENMASK(15, 8));
3576+
param_val |= u32_encode_bits((u8)arg->non_srg_th, GENMASK(7, 0));
3577+
3578+
if (arg->srp_support)
3579+
param_val |= ATH12K_OBSS_PD_THRESHOLD_IN_DBM;
3580+
3581+
if (arg->srg_enabled && arg->srp_support)
3582+
param_val |= ATH12K_OBSS_PD_SRG_EN;
3583+
3584+
if (arg->non_srg_enabled)
3585+
param_val |= ATH12K_OBSS_PD_NON_SRG_EN;
3586+
3587+
return param_val;
3588+
}
3589+
3590+
static int ath12k_wmi_pdev_set_obss_bitmap(struct ath12k *ar,
3591+
const struct wmi_pdev_set_obss_bitmap_arg *arg)
3592+
{
3593+
struct wmi_pdev_obss_pd_bitmap_cmd *cmd;
3594+
struct ath12k_wmi_pdev *wmi = ar->wmi;
3595+
const int len = sizeof(*cmd);
3596+
struct sk_buff *skb;
3597+
int ret;
3598+
3599+
skb = ath12k_wmi_alloc_skb(wmi->wmi_ab, len);
3600+
if (!skb)
3601+
return -ENOMEM;
3602+
3603+
cmd = (struct wmi_pdev_obss_pd_bitmap_cmd *)skb->data;
3604+
cmd->tlv_header = ath12k_wmi_tlv_cmd_hdr(arg->tlv_tag, len);
3605+
cmd->pdev_id = cpu_to_le32(arg->pdev_id);
3606+
memcpy(cmd->bitmap, arg->bitmap, sizeof(cmd->bitmap));
3607+
3608+
ath12k_dbg(ar->ab, ATH12K_DBG_WMI,
3609+
"wmi set pdev %u %s %08x %08x\n",
3610+
arg->pdev_id, arg->label, arg->bitmap[0], arg->bitmap[1]);
3611+
3612+
ret = ath12k_wmi_cmd_send(wmi, skb, arg->cmd_id);
3613+
if (ret) {
3614+
ath12k_warn(ar->ab, "failed to send %s: %d\n", arg->label, ret);
3615+
dev_kfree_skb(skb);
3616+
}
3617+
3618+
return ret;
3619+
}
3620+
3621+
int ath12k_wmi_pdev_set_srg_bss_color_bitmap(struct ath12k *ar,
3622+
u32 pdev_id, const u32 *bitmap)
3623+
{
3624+
struct wmi_pdev_set_obss_bitmap_arg arg = {
3625+
.tlv_tag = WMI_TAG_PDEV_SRG_BSS_COLOR_BITMAP_CMD,
3626+
.pdev_id = pdev_id,
3627+
.cmd_id = WMI_PDEV_SET_SRG_BSS_COLOR_BITMAP_CMDID,
3628+
.bitmap = bitmap,
3629+
.label = "SRG bss color bitmap",
3630+
};
3631+
3632+
return ath12k_wmi_pdev_set_obss_bitmap(ar, &arg);
3633+
}
3634+
3635+
int ath12k_wmi_pdev_set_srg_partial_bssid_bitmap(struct ath12k *ar,
3636+
u32 pdev_id, const u32 *bitmap)
3637+
{
3638+
struct wmi_pdev_set_obss_bitmap_arg arg = {
3639+
.tlv_tag = WMI_TAG_PDEV_SRG_PARTIAL_BSSID_BITMAP_CMD,
3640+
.pdev_id = pdev_id,
3641+
.cmd_id = WMI_PDEV_SET_SRG_PARTIAL_BSSID_BITMAP_CMDID,
3642+
.bitmap = bitmap,
3643+
.label = "SRG partial bssid bitmap",
3644+
};
3645+
3646+
return ath12k_wmi_pdev_set_obss_bitmap(ar, &arg);
3647+
}
3648+
3649+
int ath12k_wmi_pdev_srg_obss_color_enable_bitmap(struct ath12k *ar,
3650+
u32 pdev_id, const u32 *bitmap)
3651+
{
3652+
struct wmi_pdev_set_obss_bitmap_arg arg = {
3653+
.tlv_tag = WMI_TAG_PDEV_SRG_OBSS_COLOR_ENABLE_BITMAP_CMD,
3654+
.pdev_id = pdev_id,
3655+
.cmd_id = WMI_PDEV_SET_SRG_OBSS_COLOR_ENABLE_BITMAP_CMDID,
3656+
.bitmap = bitmap,
3657+
.label = "SRG obss color enable bitmap",
3658+
};
3659+
3660+
return ath12k_wmi_pdev_set_obss_bitmap(ar, &arg);
3661+
}
3662+
3663+
int ath12k_wmi_pdev_srg_obss_bssid_enable_bitmap(struct ath12k *ar,
3664+
u32 pdev_id, const u32 *bitmap)
3665+
{
3666+
struct wmi_pdev_set_obss_bitmap_arg arg = {
3667+
.tlv_tag = WMI_TAG_PDEV_SRG_OBSS_BSSID_ENABLE_BITMAP_CMD,
3668+
.pdev_id = pdev_id,
3669+
.cmd_id = WMI_PDEV_SET_SRG_OBSS_BSSID_ENABLE_BITMAP_CMDID,
3670+
.bitmap = bitmap,
3671+
.label = "SRG obss bssid enable bitmap",
3672+
};
3673+
3674+
return ath12k_wmi_pdev_set_obss_bitmap(ar, &arg);
3675+
}
3676+
3677+
int ath12k_wmi_pdev_non_srg_obss_color_enable_bitmap(struct ath12k *ar,
3678+
u32 pdev_id, const u32 *bitmap)
3679+
{
3680+
struct wmi_pdev_set_obss_bitmap_arg arg = {
3681+
.tlv_tag = WMI_TAG_PDEV_NON_SRG_OBSS_COLOR_ENABLE_BITMAP_CMD,
3682+
.pdev_id = pdev_id,
3683+
.cmd_id = WMI_PDEV_SET_NON_SRG_OBSS_COLOR_ENABLE_BITMAP_CMDID,
3684+
.bitmap = bitmap,
3685+
.label = "non SRG obss color enable bitmap",
3686+
};
3687+
3688+
return ath12k_wmi_pdev_set_obss_bitmap(ar, &arg);
3689+
}
3690+
3691+
int ath12k_wmi_pdev_non_srg_obss_bssid_enable_bitmap(struct ath12k *ar,
3692+
u32 pdev_id, const u32 *bitmap)
3693+
{
3694+
struct wmi_pdev_set_obss_bitmap_arg arg = {
3695+
.tlv_tag = WMI_TAG_PDEV_NON_SRG_OBSS_BSSID_ENABLE_BITMAP_CMD,
3696+
.pdev_id = pdev_id,
3697+
.cmd_id = WMI_PDEV_SET_NON_SRG_OBSS_BSSID_ENABLE_BITMAP_CMDID,
3698+
.bitmap = bitmap,
3699+
.label = "non SRG obss bssid enable bitmap",
3700+
};
3701+
3702+
return ath12k_wmi_pdev_set_obss_bitmap(ar, &arg);
3703+
}
3704+
35633705
int ath12k_wmi_obss_color_cfg_cmd(struct ath12k *ar, u32 vdev_id,
35643706
u8 bss_color, u32 period,
35653707
bool enable)

0 commit comments

Comments
 (0)