mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-07 14:32:23 +00:00
wifi: ath12k: Add MLO WMI setup and teardown functions
In case of multi device group abstraction, host has to exchange the MLO commands such as setup, teardown and ready to firmware. Once multi device group is ready, host has to exchange MLO setup command with partner devices link information and followed by MLO ready command to firmware. During deinit, MLO teardown command should be sent to firmware. Firmware would send MLO setup complete and MLO teardown complete to host for MLO setup command and MLO teardown command respectively. Added WMI helper functions for the MLO setup, ready and teardown command and the handling for corresponding event from firmware. Add appropriate WMI tag, command id and event id to parse the event and send request. Tested-on: QCN9274 hw2.0 PCI WLAN.WBE.1.3.1-00173-QCAHKSWPL_SILICONZ-1 Tested-on: WCN7850 hw2.0 PCI WLAN.HMT.1.0.c5-00481-QCAHMTSWPL_V1.0_V2.0_SILICONZ-3 Co-developed-by: Karthikeyan Periyasamy <quic_periyasa@quicinc.com> Signed-off-by: Karthikeyan Periyasamy <quic_periyasa@quicinc.com> Signed-off-by: Bhagavathi Perumal S <quic_bperumal@quicinc.com> Signed-off-by: Harshitha Prem <quic_hprem@quicinc.com> Signed-off-by: Kalle Valo <quic_kvalo@quicinc.com> Link: https://patch.msgid.link/20241211153432.775335-7-kvalo@kernel.org Signed-off-by: Jeff Johnson <jeff.johnson@oss.qualcomm.com>
This commit is contained in:
parent
48090fae67
commit
cc64deef0b
@ -7327,6 +7327,76 @@ static void ath12k_wmi_gtk_offload_status_event(struct ath12k_base *ab,
|
|||||||
kfree(tb);
|
kfree(tb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void ath12k_wmi_event_mlo_setup_complete(struct ath12k_base *ab,
|
||||||
|
struct sk_buff *skb)
|
||||||
|
{
|
||||||
|
const struct wmi_mlo_setup_complete_event *ev;
|
||||||
|
struct ath12k *ar = NULL;
|
||||||
|
struct ath12k_pdev *pdev;
|
||||||
|
const void **tb;
|
||||||
|
int ret, i;
|
||||||
|
|
||||||
|
tb = ath12k_wmi_tlv_parse_alloc(ab, skb, GFP_ATOMIC);
|
||||||
|
if (IS_ERR(tb)) {
|
||||||
|
ret = PTR_ERR(tb);
|
||||||
|
ath12k_warn(ab, "failed to parse mlo setup complete event tlv: %d\n",
|
||||||
|
ret);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ev = tb[WMI_TAG_MLO_SETUP_COMPLETE_EVENT];
|
||||||
|
if (!ev) {
|
||||||
|
ath12k_warn(ab, "failed to fetch mlo setup complete event\n");
|
||||||
|
kfree(tb);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (le32_to_cpu(ev->pdev_id) > ab->num_radios)
|
||||||
|
goto skip_lookup;
|
||||||
|
|
||||||
|
for (i = 0; i < ab->num_radios; i++) {
|
||||||
|
pdev = &ab->pdevs[i];
|
||||||
|
if (pdev && pdev->pdev_id == le32_to_cpu(ev->pdev_id)) {
|
||||||
|
ar = pdev->ar;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
skip_lookup:
|
||||||
|
if (!ar) {
|
||||||
|
ath12k_warn(ab, "invalid pdev_id %d status %u in setup complete event\n",
|
||||||
|
ev->pdev_id, ev->status);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
kfree(tb);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ath12k_wmi_event_teardown_complete(struct ath12k_base *ab,
|
||||||
|
struct sk_buff *skb)
|
||||||
|
{
|
||||||
|
const struct wmi_mlo_teardown_complete_event *ev;
|
||||||
|
const void **tb;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
tb = ath12k_wmi_tlv_parse_alloc(ab, skb, GFP_ATOMIC);
|
||||||
|
if (IS_ERR(tb)) {
|
||||||
|
ret = PTR_ERR(tb);
|
||||||
|
ath12k_warn(ab, "failed to parse teardown complete event tlv: %d\n", ret);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ev = tb[WMI_TAG_MLO_TEARDOWN_COMPLETE];
|
||||||
|
if (!ev) {
|
||||||
|
ath12k_warn(ab, "failed to fetch teardown complete event\n");
|
||||||
|
kfree(tb);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
kfree(tb);
|
||||||
|
}
|
||||||
|
|
||||||
static void ath12k_wmi_op_rx(struct ath12k_base *ab, struct sk_buff *skb)
|
static void ath12k_wmi_op_rx(struct ath12k_base *ab, struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
struct wmi_cmd_hdr *cmd_hdr;
|
struct wmi_cmd_hdr *cmd_hdr;
|
||||||
@ -7453,6 +7523,12 @@ static void ath12k_wmi_op_rx(struct ath12k_base *ab, struct sk_buff *skb)
|
|||||||
case WMI_GTK_OFFLOAD_STATUS_EVENTID:
|
case WMI_GTK_OFFLOAD_STATUS_EVENTID:
|
||||||
ath12k_wmi_gtk_offload_status_event(ab, skb);
|
ath12k_wmi_gtk_offload_status_event(ab, skb);
|
||||||
break;
|
break;
|
||||||
|
case WMI_MLO_SETUP_COMPLETE_EVENTID:
|
||||||
|
ath12k_wmi_event_mlo_setup_complete(ab, skb);
|
||||||
|
break;
|
||||||
|
case WMI_MLO_TEARDOWN_COMPLETE_EVENTID:
|
||||||
|
ath12k_wmi_event_teardown_complete(ab, skb);
|
||||||
|
break;
|
||||||
/* TODO: Add remaining events */
|
/* TODO: Add remaining events */
|
||||||
default:
|
default:
|
||||||
ath12k_dbg(ab, ATH12K_DBG_WMI, "Unknown eventid: 0x%x\n", id);
|
ath12k_dbg(ab, ATH12K_DBG_WMI, "Unknown eventid: 0x%x\n", id);
|
||||||
@ -8269,3 +8345,104 @@ int ath12k_wmi_sta_keepalive(struct ath12k *ar,
|
|||||||
|
|
||||||
return ath12k_wmi_cmd_send(wmi, skb, WMI_STA_KEEPALIVE_CMDID);
|
return ath12k_wmi_cmd_send(wmi, skb, WMI_STA_KEEPALIVE_CMDID);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int ath12k_wmi_mlo_setup(struct ath12k *ar, struct wmi_mlo_setup_arg *mlo_params)
|
||||||
|
{
|
||||||
|
struct wmi_mlo_setup_cmd *cmd;
|
||||||
|
struct ath12k_wmi_pdev *wmi = ar->wmi;
|
||||||
|
u32 *partner_links, num_links;
|
||||||
|
int i, ret, buf_len, arg_len;
|
||||||
|
struct sk_buff *skb;
|
||||||
|
struct wmi_tlv *tlv;
|
||||||
|
void *ptr;
|
||||||
|
|
||||||
|
num_links = mlo_params->num_partner_links;
|
||||||
|
arg_len = num_links * sizeof(u32);
|
||||||
|
buf_len = sizeof(*cmd) + TLV_HDR_SIZE + arg_len;
|
||||||
|
|
||||||
|
skb = ath12k_wmi_alloc_skb(wmi->wmi_ab, buf_len);
|
||||||
|
if (!skb)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
cmd = (struct wmi_mlo_setup_cmd *)skb->data;
|
||||||
|
cmd->tlv_header = ath12k_wmi_tlv_cmd_hdr(WMI_TAG_MLO_SETUP_CMD,
|
||||||
|
sizeof(*cmd));
|
||||||
|
cmd->mld_group_id = mlo_params->group_id;
|
||||||
|
cmd->pdev_id = cpu_to_le32(ar->pdev->pdev_id);
|
||||||
|
ptr = skb->data + sizeof(*cmd);
|
||||||
|
|
||||||
|
tlv = ptr;
|
||||||
|
tlv->header = ath12k_wmi_tlv_hdr(WMI_TAG_ARRAY_UINT32, arg_len);
|
||||||
|
ptr += TLV_HDR_SIZE;
|
||||||
|
|
||||||
|
partner_links = ptr;
|
||||||
|
for (i = 0; i < num_links; i++)
|
||||||
|
partner_links[i] = mlo_params->partner_link_id[i];
|
||||||
|
|
||||||
|
ret = ath12k_wmi_cmd_send(wmi, skb, WMI_MLO_SETUP_CMDID);
|
||||||
|
if (ret) {
|
||||||
|
ath12k_warn(ar->ab, "failed to submit WMI_MLO_SETUP_CMDID command: %d\n",
|
||||||
|
ret);
|
||||||
|
dev_kfree_skb(skb);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ath12k_wmi_mlo_ready(struct ath12k *ar)
|
||||||
|
{
|
||||||
|
struct wmi_mlo_ready_cmd *cmd;
|
||||||
|
struct ath12k_wmi_pdev *wmi = ar->wmi;
|
||||||
|
struct sk_buff *skb;
|
||||||
|
int ret, len;
|
||||||
|
|
||||||
|
len = sizeof(*cmd);
|
||||||
|
skb = ath12k_wmi_alloc_skb(wmi->wmi_ab, len);
|
||||||
|
if (!skb)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
cmd = (struct wmi_mlo_ready_cmd *)skb->data;
|
||||||
|
cmd->tlv_header = ath12k_wmi_tlv_cmd_hdr(WMI_TAG_MLO_READY_CMD,
|
||||||
|
sizeof(*cmd));
|
||||||
|
cmd->pdev_id = cpu_to_le32(ar->pdev->pdev_id);
|
||||||
|
|
||||||
|
ret = ath12k_wmi_cmd_send(wmi, skb, WMI_MLO_READY_CMDID);
|
||||||
|
if (ret) {
|
||||||
|
ath12k_warn(ar->ab, "failed to submit WMI_MLO_READY_CMDID command: %d\n",
|
||||||
|
ret);
|
||||||
|
dev_kfree_skb(skb);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ath12k_wmi_mlo_teardown(struct ath12k *ar)
|
||||||
|
{
|
||||||
|
struct wmi_mlo_teardown_cmd *cmd;
|
||||||
|
struct ath12k_wmi_pdev *wmi = ar->wmi;
|
||||||
|
struct sk_buff *skb;
|
||||||
|
int ret, len;
|
||||||
|
|
||||||
|
len = sizeof(*cmd);
|
||||||
|
skb = ath12k_wmi_alloc_skb(wmi->wmi_ab, len);
|
||||||
|
if (!skb)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
cmd = (struct wmi_mlo_teardown_cmd *)skb->data;
|
||||||
|
cmd->tlv_header = ath12k_wmi_tlv_cmd_hdr(WMI_TAG_MLO_TEARDOWN_CMD,
|
||||||
|
sizeof(*cmd));
|
||||||
|
cmd->pdev_id = cpu_to_le32(ar->pdev->pdev_id);
|
||||||
|
cmd->reason_code = WMI_MLO_TEARDOWN_SSR_REASON;
|
||||||
|
|
||||||
|
ret = ath12k_wmi_cmd_send(wmi, skb, WMI_MLO_TEARDOWN_CMDID);
|
||||||
|
if (ret) {
|
||||||
|
ath12k_warn(ar->ab, "failed to submit WMI MLO teardown command: %d\n",
|
||||||
|
ret);
|
||||||
|
dev_kfree_skb(skb);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
@ -285,6 +285,7 @@ enum wmi_cmd_group {
|
|||||||
WMI_GRP_TWT = 0x3e,
|
WMI_GRP_TWT = 0x3e,
|
||||||
WMI_GRP_MOTION_DET = 0x3f,
|
WMI_GRP_MOTION_DET = 0x3f,
|
||||||
WMI_GRP_SPATIAL_REUSE = 0x40,
|
WMI_GRP_SPATIAL_REUSE = 0x40,
|
||||||
|
WMI_GRP_MLO = 0x48,
|
||||||
};
|
};
|
||||||
|
|
||||||
#define WMI_CMD_GRP(grp_id) (((grp_id) << 12) | 0x1)
|
#define WMI_CMD_GRP(grp_id) (((grp_id) << 12) | 0x1)
|
||||||
@ -665,6 +666,10 @@ enum wmi_tlv_cmd_id {
|
|||||||
WMI_PDEV_OBSS_PD_SPATIAL_REUSE_CMDID =
|
WMI_PDEV_OBSS_PD_SPATIAL_REUSE_CMDID =
|
||||||
WMI_TLV_CMD(WMI_GRP_SPATIAL_REUSE),
|
WMI_TLV_CMD(WMI_GRP_SPATIAL_REUSE),
|
||||||
WMI_PDEV_OBSS_PD_SPATIAL_REUSE_SET_DEF_OBSS_THRESH_CMDID,
|
WMI_PDEV_OBSS_PD_SPATIAL_REUSE_SET_DEF_OBSS_THRESH_CMDID,
|
||||||
|
WMI_MLO_LINK_SET_ACTIVE_CMDID = WMI_TLV_CMD(WMI_GRP_MLO),
|
||||||
|
WMI_MLO_SETUP_CMDID,
|
||||||
|
WMI_MLO_READY_CMDID,
|
||||||
|
WMI_MLO_TEARDOWN_CMDID,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum wmi_tlv_event_id {
|
enum wmi_tlv_event_id {
|
||||||
@ -874,6 +879,9 @@ enum wmi_tlv_event_id {
|
|||||||
WMI_TWT_DEL_DIALOG_EVENTID,
|
WMI_TWT_DEL_DIALOG_EVENTID,
|
||||||
WMI_TWT_PAUSE_DIALOG_EVENTID,
|
WMI_TWT_PAUSE_DIALOG_EVENTID,
|
||||||
WMI_TWT_RESUME_DIALOG_EVENTID,
|
WMI_TWT_RESUME_DIALOG_EVENTID,
|
||||||
|
WMI_MLO_LINK_SET_ACTIVE_RESP_EVENTID = WMI_EVT_GRP_START_ID(WMI_GRP_MLO),
|
||||||
|
WMI_MLO_SETUP_COMPLETE_EVENTID,
|
||||||
|
WMI_MLO_TEARDOWN_COMPLETE_EVENTID,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum wmi_tlv_pdev_param {
|
enum wmi_tlv_pdev_param {
|
||||||
@ -5026,6 +5034,43 @@ struct wmi_twt_disable_event {
|
|||||||
__le32 status;
|
__le32 status;
|
||||||
} __packed;
|
} __packed;
|
||||||
|
|
||||||
|
struct wmi_mlo_setup_cmd {
|
||||||
|
__le32 tlv_header;
|
||||||
|
__le32 mld_group_id;
|
||||||
|
__le32 pdev_id;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
struct wmi_mlo_setup_arg {
|
||||||
|
__le32 group_id;
|
||||||
|
u8 num_partner_links;
|
||||||
|
u8 *partner_link_id;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct wmi_mlo_ready_cmd {
|
||||||
|
__le32 tlv_header;
|
||||||
|
__le32 pdev_id;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
enum wmi_mlo_tear_down_reason_code_type {
|
||||||
|
WMI_MLO_TEARDOWN_SSR_REASON,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct wmi_mlo_teardown_cmd {
|
||||||
|
__le32 tlv_header;
|
||||||
|
__le32 pdev_id;
|
||||||
|
__le32 reason_code;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
struct wmi_mlo_setup_complete_event {
|
||||||
|
__le32 pdev_id;
|
||||||
|
__le32 status;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
struct wmi_mlo_teardown_complete_event {
|
||||||
|
__le32 pdev_id;
|
||||||
|
__le32 status;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
/* WOW structures */
|
/* WOW structures */
|
||||||
enum wmi_wow_wakeup_event {
|
enum wmi_wow_wakeup_event {
|
||||||
WOW_BMISS_EVENT = 0,
|
WOW_BMISS_EVENT = 0,
|
||||||
@ -5751,5 +5796,8 @@ int ath12k_wmi_gtk_rekey_getinfo(struct ath12k *ar,
|
|||||||
struct ath12k_link_vif *arvif);
|
struct ath12k_link_vif *arvif);
|
||||||
int ath12k_wmi_sta_keepalive(struct ath12k *ar,
|
int ath12k_wmi_sta_keepalive(struct ath12k *ar,
|
||||||
const struct wmi_sta_keepalive_arg *arg);
|
const struct wmi_sta_keepalive_arg *arg);
|
||||||
|
int ath12k_wmi_mlo_setup(struct ath12k *ar, struct wmi_mlo_setup_arg *mlo_params);
|
||||||
|
int ath12k_wmi_mlo_ready(struct ath12k *ar);
|
||||||
|
int ath12k_wmi_mlo_teardown(struct ath12k *ar);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
Reference in New Issue
Block a user