Browse Source

icnss2: Enable Support for WFC call TWT config

Update icnss2 driver to support WFC call TWT config
params exchange between IMS and WLAN firmware.

Change-Id: Ib565019c0b51db3cc3def8a994094eeeed7a701a
CRs-Fixed: 3710947
Naman Padhiar 1 year ago
parent
commit
87ab436e3d

+ 1 - 1
cnss2/Makefile

@@ -23,4 +23,4 @@ cnss2-y += pci.o
 cnss2-y += power.o
 cnss2-y += genl.o
 cnss2-$(CONFIG_PCI_MSM) += pci_qcom.o
-cnss2-$(CONFIG_CNSS2_QMI) += qmi.o coexistence_service_v01.o ip_multimedia_subsystem_private_service_v01.o
+cnss2-$(CONFIG_CNSS2_QMI) += qmi.o coexistence_service_v01.o

+ 3 - 3
cnss2/qmi.c

@@ -1,7 +1,7 @@
 // SPDX-License-Identifier: GPL-2.0-only
 /*
  * Copyright (c) 2015-2021, The Linux Foundation. All rights reserved.
- * Copyright (c) 2021-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2021-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  */
 
 #include <linux/module.h>
@@ -3942,7 +3942,7 @@ int ims_subscribe_for_indication_send_async(struct cnss_plat_data *plat_priv)
 	(&plat_priv->ims_qmi, NULL, txn,
 	QMI_IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_REQ_V01,
 	IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_REQ_MSG_V01_MAX_MSG_LEN,
-	ims_private_service_subscribe_for_indications_req_msg_v01_ei, req);
+	ims_private_service_subscribe_ind_req_msg_v01_ei, req);
 	if (ret < 0) {
 		qmi_txn_cancel(txn);
 		cnss_pr_err("Fail to send ims subscribe for indication req %d\n",
@@ -4033,7 +4033,7 @@ static struct qmi_msg_handler qmi_ims_msg_handlers[] = {
 		.msg_id =
 		QMI_IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_REQ_V01,
 		.ei =
-		ims_private_service_subscribe_for_indications_rsp_msg_v01_ei,
+		ims_private_service_subscribe_ind_rsp_msg_v01_ei,
 		.decoded_size = sizeof(struct
 		ims_private_service_subscribe_for_indications_rsp_msg_v01),
 		.fn = ims_subscribe_for_indication_resp_cb

+ 1 - 1
cnss_utils/Makefile

@@ -7,7 +7,7 @@ endif
 obj-$(CONFIG_CNSS_UTILS) += cnss_utils.o
 
 obj-$(CONFIG_CNSS_QMI_SVC) += wlan_firmware_service.o
-wlan_firmware_service-y := wlan_firmware_service_v01.o device_management_service_v01.o
+wlan_firmware_service-y := wlan_firmware_service_v01.o device_management_service_v01.o ip_multimedia_subsystem_private_service_v01.o
 
 obj-$(CONFIG_CNSS_PLAT_IPC_QMI_SVC) += cnss_plat_ipc_qmi_svc.o
 cnss_plat_ipc_qmi_svc-y := cnss_plat_ipc_qmi.o cnss_plat_ipc_service_v01.o

+ 11 - 4
cnss2/ip_multimedia_subsystem_private_service_v01.c → cnss_utils/ip_multimedia_subsystem_private_service_v01.c

@@ -1,5 +1,7 @@
 // SPDX-License-Identifier: GPL-2.0-only
-/* Copyright (c) 2019-2020, The Linux Foundation. All rights reserved. */
+/* Copyright (c) 2019-2020, The Linux Foundation. All rights reserved.
+ * Copyright (c) 2024 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
 
 #include "ip_multimedia_subsystem_private_service_v01.h"
 
@@ -30,7 +32,7 @@ static struct qmi_elem_info ims_private_service_header_value_v01_ei[] = {
 };
 
 struct qmi_elem_info
-ims_private_service_subscribe_for_indications_req_msg_v01_ei[] = {
+ims_private_service_subscribe_ind_req_msg_v01_ei[] = {
 	{
 		.data_type      = QMI_OPT_FLAG,
 		.elem_len       = 1,
@@ -77,9 +79,10 @@ ims_private_service_subscribe_for_indications_req_msg_v01_ei[] = {
 		.tlv_type       = QMI_COMMON_TLV_TYPE,
 	},
 };
+EXPORT_SYMBOL(ims_private_service_subscribe_ind_req_msg_v01_ei);
 
 struct qmi_elem_info
-ims_private_service_subscribe_for_indications_rsp_msg_v01_ei[] = {
+ims_private_service_subscribe_ind_rsp_msg_v01_ei[] = {
 	{
 		.data_type      = QMI_STRUCT,
 		.elem_len       = 1,
@@ -97,6 +100,7 @@ ims_private_service_subscribe_for_indications_rsp_msg_v01_ei[] = {
 		.tlv_type       = QMI_COMMON_TLV_TYPE,
 	},
 };
+EXPORT_SYMBOL(ims_private_service_subscribe_ind_rsp_msg_v01_ei);
 
 struct qmi_elem_info ims_private_service_mt_invite_ind_msg_v01_ei[] = {
 	{
@@ -167,6 +171,7 @@ struct qmi_elem_info ims_private_service_mt_invite_ind_msg_v01_ei[] = {
 		.tlv_type       = QMI_COMMON_TLV_TYPE,
 	},
 };
+EXPORT_SYMBOL(ims_private_service_mt_invite_ind_msg_v01_ei);
 
 struct qmi_elem_info ims_private_service_wfc_call_status_ind_msg_v01_ei[] = {
 	{
@@ -285,6 +290,7 @@ struct qmi_elem_info ims_private_service_wfc_call_status_ind_msg_v01_ei[] = {
 		.tlv_type       = QMI_COMMON_TLV_TYPE,
 	},
 };
+EXPORT_SYMBOL(ims_private_service_wfc_call_status_ind_msg_v01_ei);
 
 struct qmi_elem_info
 ims_private_service_wfc_call_twt_config_req_msg_v01_ei[] = {
@@ -426,6 +432,7 @@ ims_private_service_wfc_call_twt_config_req_msg_v01_ei[] = {
 		.tlv_type       = QMI_COMMON_TLV_TYPE,
 	},
 };
+EXPORT_SYMBOL(ims_private_service_wfc_call_twt_config_req_msg_v01_ei);
 
 struct qmi_elem_info
 ims_private_service_wfc_call_twt_config_rsp_msg_v01_ei[] = {
@@ -447,4 +454,4 @@ ims_private_service_wfc_call_twt_config_rsp_msg_v01_ei[] = {
 		.tlv_type       = QMI_COMMON_TLV_TYPE,
 	},
 };
-
+EXPORT_SYMBOL(ims_private_service_wfc_call_twt_config_rsp_msg_v01_ei);

+ 5 - 3
cnss2/ip_multimedia_subsystem_private_service_v01.h → cnss_utils/ip_multimedia_subsystem_private_service_v01.h

@@ -1,5 +1,7 @@
 /* SPDX-License-Identifier: GPL-2.0-only */
-/* Copyright (c) 2020, The Linux Foundation. All rights reserved. */
+/* Copyright (c) 2020, The Linux Foundation. All rights reserved.
+ * Copyright (c) 2024 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
 
 #ifndef IP_MULTIMEDIA_SUBSYSTEM_PRIVATE_SERVICE_V01_H
 #define IP_MULTIMEDIA_SUBSYSTEM_PRIVATE_SERVICE_V01_H
@@ -63,7 +65,7 @@ struct ims_private_service_subscribe_for_indications_req_msg_v01 {
 
 #define IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_REQ_MSG_V01_MAX_MSG_LEN 8
 extern struct qmi_elem_info
-ims_private_service_subscribe_for_indications_req_msg_v01_ei[];
+ims_private_service_subscribe_ind_req_msg_v01_ei[];
 
 struct ims_private_service_subscribe_for_indications_rsp_msg_v01 {
 	struct qmi_response_type_v01 resp;
@@ -71,7 +73,7 @@ struct ims_private_service_subscribe_for_indications_rsp_msg_v01 {
 
 #define IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_RSP_MSG_V01_MAX_MSG_LEN 7
 extern struct qmi_elem_info
-ims_private_service_subscribe_for_indications_rsp_msg_v01_ei[];
+ims_private_service_subscribe_ind_rsp_msg_v01_ei[];
 
 struct ims_private_service_mt_invite_ind_msg_v01 {
 	enum ims_subscription_type_enum_v01 subscription_type;

+ 4 - 1
icnss2/debug.c

@@ -1,7 +1,7 @@
 // SPDX-License-Identifier: GPL-2.0-only
 /*
  * Copyright (c) 2015-2021, The Linux Foundation. All rights reserved.
- * Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2023-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  */
 #include <linux/err.h>
 #include <linux/seq_file.h>
@@ -414,6 +414,9 @@ static int icnss_stats_show_state(struct seq_file *s, struct icnss_priv *priv)
 		case ICNSS_PDR:
 			seq_puts(s, "PDR TRIGGERED");
 			continue;
+		case ICNSS_IMS_CONNECTED:
+			seq_puts(s, "IMS_CONNECTED");
+			continue;
 		case ICNSS_DEL_SERVER:
 			seq_puts(s, "DEL SERVER");
 			continue;

+ 16 - 1
icnss2/main.c

@@ -1,7 +1,7 @@
 // SPDX-License-Identifier: GPL-2.0-only
 /*
  * Copyright (c) 2015-2020, 2021, The Linux Foundation. All rights reserved.
- * Copyright (c) 2022-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2022-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  */
 
 #define pr_fmt(fmt) "icnss2: " fmt
@@ -234,6 +234,10 @@ char *icnss_driver_event_to_str(enum icnss_driver_event_type type)
 		return "QDSS_TRACE_FREE";
 	case ICNSS_DRIVER_EVENT_M3_DUMP_UPLOAD_REQ:
 		return "M3_DUMP_UPLOAD";
+	case ICNSS_DRIVER_EVENT_IMS_WFC_CALL_IND:
+		return "IMS_WFC_CALL_IND";
+	case ICNSS_DRIVER_EVENT_WLFW_TWT_CFG_IND:
+		return "WLFW_TWC_CFG_IND";
 	case ICNSS_DRIVER_EVENT_QDSS_TRACE_REQ_DATA:
 		return "QDSS_TRACE_REQ_DATA";
 	case ICNSS_DRIVER_EVENT_SUBSYS_RESTART_LEVEL:
@@ -1965,6 +1969,14 @@ static void icnss_driver_event_work(struct work_struct *work)
 		case ICNSS_DRIVER_EVENT_SUBSYS_RESTART_LEVEL:
 			ret = icnss_subsys_restart_level(priv, event->data);
 			break;
+		case ICNSS_DRIVER_EVENT_IMS_WFC_CALL_IND:
+			ret = icnss_process_wfc_call_ind_event(priv,
+							      event->data);
+			break;
+		case ICNSS_DRIVER_EVENT_WLFW_TWT_CFG_IND:
+			ret = icnss_process_twt_cfg_ind_event(priv,
+							     event->data);
+			break;
 		default:
 			icnss_pr_err("Invalid Event type: %d", event->type);
 			kfree(event);
@@ -4878,6 +4890,7 @@ static int icnss_probe(struct platform_device *pdev)
 			    icnss_wpss_ssr_timeout_hdlr, 0);
 	}
 
+	icnss_register_ims_service(priv);
 	INIT_LIST_HEAD(&priv->icnss_tcdev_list);
 
 	icnss_pr_info("Platform driver probed successfully\n");
@@ -4936,6 +4949,8 @@ static int icnss_remove(struct platform_device *pdev)
 
 	device_init_wakeup(&priv->pdev->dev, false);
 
+	icnss_unregister_ims_service(priv);
+
 	icnss_debugfs_destroy(priv);
 
 	icnss_unregister_power_supply_notifier(penv);

+ 6 - 1
icnss2/main.h

@@ -1,7 +1,7 @@
 /* SPDX-License-Identifier: GPL-2.0-only */
 /*
  * Copyright (c) 2017-2020, 2021, The Linux Foundation. All rights reserved.
- * Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2023-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  */
 
 #ifndef __MAIN_H__
@@ -68,6 +68,8 @@ enum icnss_driver_event_type {
 	ICNSS_DRIVER_EVENT_QDSS_TRACE_SAVE,
 	ICNSS_DRIVER_EVENT_QDSS_TRACE_FREE,
 	ICNSS_DRIVER_EVENT_M3_DUMP_UPLOAD_REQ,
+	ICNSS_DRIVER_EVENT_IMS_WFC_CALL_IND,
+	ICNSS_DRIVER_EVENT_WLFW_TWT_CFG_IND,
 	ICNSS_DRIVER_EVENT_QDSS_TRACE_REQ_DATA,
 	ICNSS_DRIVER_EVENT_SUBSYS_RESTART_LEVEL,
 	ICNSS_DRIVER_EVENT_MAX,
@@ -127,6 +129,7 @@ enum icnss_driver_state {
 	ICNSS_MODE_ON,
 	ICNSS_BLOCK_SHUTDOWN,
 	ICNSS_PDR,
+	ICNSS_IMS_CONNECTED,
 	ICNSS_DEL_SERVER,
 	ICNSS_COLD_BOOT_CAL,
 	ICNSS_QMI_DMS_CONNECTED,
@@ -402,6 +405,8 @@ struct icnss_priv {
 	size_t smmu_iova_ipa_len;
 	struct qmi_handle qmi;
 	struct qmi_handle qmi_dms;
+	struct qmi_handle ims_qmi;
+	struct qmi_txn ims_async_txn;
 	struct list_head event_list;
 	struct list_head soc_wake_msg_list;
 	spinlock_t event_lock;

+ 429 - 1
icnss2/qmi.c

@@ -1,7 +1,7 @@
 // SPDX-License-Identifier: GPL-2.0-only
 /*
  * Copyright (c) 2017-2021, The Linux Foundation. All rights reserved.
- * Copyright (c) 2022-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2022-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  */
 
 #define pr_fmt(fmt) "icnss2_qmi: " fmt
@@ -58,6 +58,7 @@
 #define MAX_SHADOW_REG_RESERVED		2
 #define MAX_NUM_SHADOW_REG_V3		(QMI_WLFW_MAX_NUM_SHADOW_REG_V3_USAGE_V01 - \
 					MAX_SHADOW_REG_RESERVED)
+#define IMSPRIVATE_SERVICE_MAX_MSG_LEN  SZ_8K
 
 #ifdef CONFIG_ICNSS2_DEBUG
 bool ignore_fw_timeout;
@@ -2777,6 +2778,223 @@ out:
 	kfree(event_data);
 }
 
+static int icnss_wlfw_wfc_call_status_send_sync
+	(struct icnss_priv *priv,
+	 const struct ims_private_service_wfc_call_status_ind_msg_v01 *ind_msg)
+{
+	struct wlfw_wfc_call_status_req_msg_v01 *req;
+	struct wlfw_wfc_call_status_resp_msg_v01 *resp;
+	struct qmi_txn txn;
+	int ret = 0;
+
+	if (!test_bit(ICNSS_FW_READY, &priv->state)) {
+		icnss_pr_err("Drop IMS WFC indication as FW not initialized\n");
+		return -EINVAL;
+	}
+	req = kzalloc(sizeof(*req), GFP_KERNEL);
+	if (!req)
+		return -ENOMEM;
+
+	resp = kzalloc(sizeof(*resp), GFP_KERNEL);
+	if (!resp) {
+		kfree(req);
+		return -ENOMEM;
+	}
+
+	/**
+	 * WFC Call r1 design has CNSS as pass thru using opaque hex buffer.
+	 * But in r2 update QMI structure is expanded and as an effect qmi
+	 * decoded structures have padding. Thus we cannot use buffer design.
+	 * For backward compatibility for r1 design copy only wfc_call_active
+	 * value in hex buffer.
+	 */
+	req->wfc_call_status_len = sizeof(ind_msg->wfc_call_active);
+	req->wfc_call_status[0] = ind_msg->wfc_call_active;
+
+	/* wfc_call_active is mandatory in IMS indication */
+	req->wfc_call_active_valid = 1;
+	req->wfc_call_active = ind_msg->wfc_call_active;
+	req->all_wfc_calls_held_valid = ind_msg->all_wfc_calls_held_valid;
+	req->all_wfc_calls_held = ind_msg->all_wfc_calls_held;
+	req->is_wfc_emergency_valid = ind_msg->is_wfc_emergency_valid;
+	req->is_wfc_emergency = ind_msg->is_wfc_emergency;
+	req->twt_ims_start_valid = ind_msg->twt_ims_start_valid;
+	req->twt_ims_start = ind_msg->twt_ims_start;
+	req->twt_ims_int_valid = ind_msg->twt_ims_int_valid;
+	req->twt_ims_int = ind_msg->twt_ims_int;
+	req->media_quality_valid = ind_msg->media_quality_valid;
+	req->media_quality =
+		(enum wlfw_wfc_media_quality_v01)ind_msg->media_quality;
+
+	icnss_pr_dbg("CNSS->FW: WFC_CALL_REQ: state: 0x%lx\n",
+		     priv->state);
+
+	ret = qmi_txn_init(&priv->qmi, &txn,
+			   wlfw_wfc_call_status_resp_msg_v01_ei, resp);
+	if (ret < 0) {
+		icnss_pr_err("CNSS->FW: WFC_CALL_REQ: QMI Txn Init: Err %d\n",
+			     ret);
+		goto out;
+	}
+
+	ret = qmi_send_request(&priv->qmi, NULL, &txn,
+			       QMI_WLFW_WFC_CALL_STATUS_REQ_V01,
+			       WLFW_WFC_CALL_STATUS_REQ_MSG_V01_MAX_MSG_LEN,
+			       wlfw_wfc_call_status_req_msg_v01_ei, req);
+	if (ret < 0) {
+		qmi_txn_cancel(&txn);
+		icnss_pr_err("CNSS->FW: WFC_CALL_REQ: QMI Send Err: %d\n",
+			     ret);
+		goto out;
+	}
+
+	ret = qmi_txn_wait(&txn, priv->ctrl_params.qmi_timeout);
+	if (ret < 0) {
+		icnss_pr_err("FW->CNSS: WFC_CALL_RSP: QMI Wait Err: %d\n",
+			     ret);
+		goto out;
+	}
+
+	if (resp->resp.result != QMI_RESULT_SUCCESS_V01) {
+		icnss_pr_err("FW->CNSS: WFC_CALL_RSP: Result: %d Err: %d\n",
+			     resp->resp.result, resp->resp.error);
+		ret = -resp->resp.result;
+		goto out;
+	}
+	ret = 0;
+out:
+	kfree(req);
+	kfree(resp);
+	return ret;
+}
+
+static int icnss_ims_wfc_call_twt_cfg_send_sync
+	(struct icnss_priv *priv,
+	const struct wlfw_wfc_call_twt_config_ind_msg_v01 *ind_msg)
+{
+	struct ims_private_service_wfc_call_twt_config_req_msg_v01 *req;
+	struct ims_private_service_wfc_call_twt_config_rsp_msg_v01 *resp;
+	struct qmi_txn txn;
+	int ret = 0;
+
+	if (!test_bit(ICNSS_IMS_CONNECTED, &priv->state)) {
+		icnss_pr_err("Drop FW WFC indication as IMS QMI not connected\n");
+		return -EINVAL;
+	}
+
+	req = kzalloc(sizeof(*req), GFP_KERNEL);
+	if (!req)
+		return -ENOMEM;
+
+	resp = kzalloc(sizeof(*resp), GFP_KERNEL);
+	if (!resp) {
+		kfree(req);
+		return -ENOMEM;
+	}
+
+	req->twt_sta_start_valid = ind_msg->twt_sta_start_valid;
+	req->twt_sta_start = ind_msg->twt_sta_start;
+	req->twt_sta_int_valid = ind_msg->twt_sta_int_valid;
+	req->twt_sta_int = ind_msg->twt_sta_int;
+	req->twt_sta_upo_valid = ind_msg->twt_sta_upo_valid;
+	req->twt_sta_upo = ind_msg->twt_sta_upo;
+	req->twt_sta_sp_valid = ind_msg->twt_sta_sp_valid;
+	req->twt_sta_sp = ind_msg->twt_sta_sp;
+	req->twt_sta_dl_valid = req->twt_sta_dl_valid;
+	req->twt_sta_dl = req->twt_sta_dl;
+	req->twt_sta_config_changed_valid =
+				ind_msg->twt_sta_config_changed_valid;
+	req->twt_sta_config_changed = ind_msg->twt_sta_config_changed;
+
+	icnss_pr_dbg("CNSS->IMS: TWT_CFG_REQ: state: 0x%lx\n",
+		     priv->state);
+
+	ret =
+	qmi_txn_init(&priv->ims_qmi, &txn,
+		     ims_private_service_wfc_call_twt_config_rsp_msg_v01_ei,
+		     resp);
+	if (ret < 0) {
+		icnss_pr_err("CNSS->IMS: TWT_CFG_REQ: QMI Txn Init Err: %d\n",
+			    ret);
+		goto out;
+	}
+
+	ret =
+	qmi_send_request(&priv->ims_qmi, NULL, &txn,
+			 QMI_IMS_PRIVATE_SERVICE_WFC_CALL_TWT_CONFIG_REQ_V01,
+		IMS_PRIVATE_SERVICE_WFC_CALL_TWT_CONFIG_REQ_MSG_V01_MAX_MSG_LEN,
+		ims_private_service_wfc_call_twt_config_req_msg_v01_ei, req);
+	if (ret < 0) {
+		qmi_txn_cancel(&txn);
+		icnss_pr_err("CNSS->IMS: TWT_CFG_REQ: QMI Send Err: %d\n", ret);
+		goto out;
+	}
+
+	ret = qmi_txn_wait(&txn, priv->ctrl_params.qmi_timeout);
+	if (ret < 0) {
+		icnss_pr_err("IMS->CNSS: TWT_CFG_RSP: QMI Wait Err: %d\n", ret);
+		goto out;
+	}
+
+	if (resp->resp.result != QMI_RESULT_SUCCESS_V01) {
+		icnss_pr_err("IMS->CNSS: TWT_CFG_RSP: Result: %d Err: %d\n",
+			    resp->resp.result, resp->resp.error);
+		ret = -resp->resp.result;
+		goto out;
+	}
+	ret = 0;
+out:
+	kfree(req);
+	kfree(resp);
+	return ret;
+}
+
+int icnss_process_twt_cfg_ind_event(struct icnss_priv *priv,
+				    void *data)
+{
+	int ret;
+	struct wlfw_wfc_call_twt_config_ind_msg_v01 *ind_msg = data;
+
+	ret = icnss_ims_wfc_call_twt_cfg_send_sync(priv, ind_msg);
+	kfree(data);
+	return ret;
+}
+
+static void icnss_wlfw_process_twt_cfg_ind(struct qmi_handle *qmi,
+					   struct sockaddr_qrtr *sq,
+					   struct qmi_txn *txn,
+					   const void *data)
+{
+	struct icnss_priv *priv =
+		container_of(qmi, struct icnss_priv, qmi);
+	const struct wlfw_wfc_call_twt_config_ind_msg_v01 *ind_msg = data;
+	struct wlfw_wfc_call_twt_config_ind_msg_v01 *event_data;
+
+	if (!txn) {
+		icnss_pr_err("FW->CNSS: TWT_CFG_IND: Spurious indication\n");
+		return;
+	}
+
+	if (!ind_msg) {
+		icnss_pr_err("FW->CNSS: TWT_CFG_IND: Invalid indication\n");
+		return;
+	}
+	icnss_pr_dbg("FW->CNSS: TWT_CFG_IND: %x %llx, %x %x, %x %x, %x %x, %x %x, %x %x\n",
+		     ind_msg->twt_sta_start_valid, ind_msg->twt_sta_start,
+		     ind_msg->twt_sta_int_valid, ind_msg->twt_sta_int,
+		     ind_msg->twt_sta_upo_valid, ind_msg->twt_sta_upo,
+		     ind_msg->twt_sta_sp_valid, ind_msg->twt_sta_sp,
+		     ind_msg->twt_sta_dl_valid, ind_msg->twt_sta_dl,
+		     ind_msg->twt_sta_config_changed_valid,
+		     ind_msg->twt_sta_config_changed);
+
+	event_data = kmemdup(ind_msg, sizeof(*event_data), GFP_KERNEL);
+	if (!event_data)
+		return;
+	icnss_driver_event_post(priv, ICNSS_DRIVER_EVENT_WLFW_TWT_CFG_IND, 0,
+			       event_data);
+}
+
 static struct qmi_msg_handler wlfw_msg_handlers[] = {
 	{
 		.type = QMI_INDICATION,
@@ -2861,6 +3079,14 @@ static struct qmi_msg_handler wlfw_msg_handlers[] = {
 		sizeof(struct wlfw_m3_dump_upload_segments_req_ind_msg_v01),
 		.fn = icnss_wlfw_m3_dump_upload_segs_req_ind_cb
 	},
+	{
+		.type = QMI_INDICATION,
+		.msg_id = QMI_WLFW_WFC_CALL_TWT_CONFIG_IND_V01,
+		.ei = wlfw_wfc_call_twt_config_ind_msg_v01_ei,
+		.decoded_size =
+		sizeof(struct wlfw_wfc_call_twt_config_ind_msg_v01),
+		.fn = icnss_wlfw_process_twt_cfg_ind
+	},
 	{}
 };
 
@@ -3553,3 +3779,205 @@ out:
 	kfree(resp);
 	return ret;
 }
+
+/* IMS Service */
+int ims_subscribe_for_indication_send_async(struct icnss_priv *priv)
+{
+	int ret;
+	struct ims_private_service_subscribe_for_indications_req_msg_v01 *req;
+	struct qmi_txn *txn;
+
+	if (!priv)
+		return -ENODEV;
+
+	icnss_pr_dbg("Sending ASYNC ims subscribe for indication\n");
+
+	req = kzalloc(sizeof(*req), GFP_KERNEL);
+	if (!req)
+		return -ENOMEM;
+
+	req->wfc_call_status_valid = 1;
+	req->wfc_call_status = 1;
+
+	txn = &priv->ims_async_txn;
+	ret = qmi_txn_init(&priv->ims_qmi, txn, NULL, NULL);
+	if (ret < 0) {
+		icnss_pr_err("Fail to init txn for ims subscribe for indication resp %d\n",
+			     ret);
+		goto out;
+	}
+
+	ret = qmi_send_request
+	(&priv->ims_qmi, NULL, txn,
+	QMI_IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_REQ_V01,
+	IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_REQ_MSG_V01_MAX_MSG_LEN,
+	ims_private_service_subscribe_ind_req_msg_v01_ei, req);
+	if (ret < 0) {
+		qmi_txn_cancel(txn);
+		icnss_pr_err("Fail to send ims subscribe for indication req %d\n",
+			     ret);
+		goto out;
+	}
+
+	kfree(req);
+	return 0;
+
+out:
+	kfree(req);
+	return ret;
+}
+
+static void ims_subscribe_for_indication_resp_cb(struct qmi_handle *qmi,
+						 struct sockaddr_qrtr *sq,
+						 struct qmi_txn *txn,
+						 const void *data)
+{
+	const
+	struct ims_private_service_subscribe_for_indications_rsp_msg_v01 *resp =
+		data;
+
+	icnss_pr_dbg("Received IMS subscribe indication response\n");
+
+	if (!txn) {
+		icnss_pr_err("spurious response\n");
+		return;
+	}
+
+	if (resp->resp.result != QMI_RESULT_SUCCESS_V01) {
+		icnss_pr_err("IMS subscribe for indication request rejected, result:%d error:%d\n",
+			     resp->resp.result, resp->resp.error);
+		txn->result = -resp->resp.result;
+	}
+}
+
+int icnss_process_wfc_call_ind_event(struct icnss_priv *priv,
+				     void *data)
+{
+	int ret;
+	struct ims_private_service_wfc_call_status_ind_msg_v01 *ind_msg = data;
+
+	ret = icnss_wlfw_wfc_call_status_send_sync(priv, ind_msg);
+	kfree(data);
+	return ret;
+}
+
+static void
+icnss_ims_process_wfc_call_ind_cb(struct qmi_handle *ims_qmi,
+				  struct sockaddr_qrtr *sq,
+				  struct qmi_txn *txn, const void *data)
+{
+	struct icnss_priv *priv =
+		container_of(ims_qmi, struct icnss_priv, ims_qmi);
+	const
+	struct ims_private_service_wfc_call_status_ind_msg_v01 *ind_msg = data;
+	struct ims_private_service_wfc_call_status_ind_msg_v01 *event_data;
+
+	if (!txn) {
+		icnss_pr_err("IMS->CNSS: WFC_CALL_IND: Spurious indication\n");
+		return;
+	}
+
+	if (!ind_msg) {
+		icnss_pr_err("IMS->CNSS: WFC_CALL_IND: Invalid indication\n");
+		return;
+	}
+	icnss_pr_dbg("IMS->CNSS: WFC_CALL_IND: %x, %x %x, %x %x, %x %llx, %x %x, %x %x\n",
+		     ind_msg->wfc_call_active, ind_msg->all_wfc_calls_held_valid,
+		     ind_msg->all_wfc_calls_held,
+		     ind_msg->is_wfc_emergency_valid, ind_msg->is_wfc_emergency,
+		     ind_msg->twt_ims_start_valid, ind_msg->twt_ims_start,
+		     ind_msg->twt_ims_int_valid, ind_msg->twt_ims_int,
+		     ind_msg->media_quality_valid, ind_msg->media_quality);
+
+	event_data = kmemdup(ind_msg, sizeof(*event_data), GFP_KERNEL);
+	if (!event_data)
+		return;
+	icnss_driver_event_post(priv, ICNSS_DRIVER_EVENT_IMS_WFC_CALL_IND,
+			       0, event_data);
+}
+
+static struct qmi_msg_handler qmi_ims_msg_handlers[] = {
+	{
+		.type = QMI_RESPONSE,
+		.msg_id =
+		QMI_IMS_PRIVATE_SERVICE_SUBSCRIBE_FOR_INDICATIONS_REQ_V01,
+		.ei =
+		ims_private_service_subscribe_ind_rsp_msg_v01_ei,
+		.decoded_size = sizeof(struct
+		ims_private_service_subscribe_for_indications_rsp_msg_v01),
+		.fn = ims_subscribe_for_indication_resp_cb
+	},
+	{
+		.type = QMI_INDICATION,
+		.msg_id = QMI_IMS_PRIVATE_SERVICE_WFC_CALL_STATUS_IND_V01,
+		.ei = ims_private_service_wfc_call_status_ind_msg_v01_ei,
+		.decoded_size =
+		sizeof(struct ims_private_service_wfc_call_status_ind_msg_v01),
+		.fn = icnss_ims_process_wfc_call_ind_cb
+	},
+	{}
+};
+
+static int ims_new_server(struct qmi_handle *qmi,
+			  struct qmi_service *service)
+{
+	struct icnss_priv *priv =
+		container_of(qmi, struct icnss_priv, ims_qmi);
+	struct sockaddr_qrtr sq = { 0 };
+	int ret = 0;
+
+	icnss_pr_dbg("IMS server arrive: node %u port %u\n",
+		    service->node, service->port);
+
+	sq.sq_family = AF_QIPCRTR;
+	sq.sq_node = service->node;
+	sq.sq_port = service->port;
+	ret = kernel_connect(qmi->sock, (struct sockaddr *)&sq, sizeof(sq), 0);
+	if (ret < 0) {
+		icnss_pr_err("Fail to connect to remote service port\n");
+		return ret;
+	}
+
+	set_bit(ICNSS_IMS_CONNECTED, &priv->state);
+	icnss_pr_dbg("IMS Server Connected: 0x%lx\n",
+		    priv->state);
+
+	ret = ims_subscribe_for_indication_send_async(priv);
+	return ret;
+}
+
+static void ims_del_server(struct qmi_handle *qmi,
+			   struct qmi_service *service)
+{
+	struct icnss_priv *priv =
+		container_of(qmi, struct icnss_priv, ims_qmi);
+
+	icnss_pr_dbg("IMS server exit\n");
+
+	clear_bit(ICNSS_IMS_CONNECTED, &priv->state);
+}
+
+static struct qmi_ops ims_qmi_ops = {
+	.new_server = ims_new_server,
+	.del_server = ims_del_server,
+};
+
+int icnss_register_ims_service(struct icnss_priv *priv)
+{
+	int ret;
+
+	ret = qmi_handle_init(&priv->ims_qmi,
+			      IMSPRIVATE_SERVICE_MAX_MSG_LEN,
+			      &ims_qmi_ops, qmi_ims_msg_handlers);
+	if (ret < 0)
+		return ret;
+
+	ret = qmi_add_lookup(&priv->ims_qmi, IMSPRIVATE_SERVICE_ID_V01,
+			     IMSPRIVATE_SERVICE_VERS_V01, 0);
+	return ret;
+}
+
+void icnss_unregister_ims_service(struct icnss_priv *priv)
+{
+	qmi_handle_release(&priv->ims_qmi);
+}

+ 31 - 0
icnss2/qmi.h

@@ -1,12 +1,14 @@
 /* SPDX-License-Identifier: GPL-2.0-only */
 /*
  * Copyright (c) 2017-2021, The Linux Foundation. All rights reserved.
+ * Copyright (c) 2024 Qualcomm Innovation Center, Inc. All rights reserved.
  */
 
 #ifndef __ICNSS_QMI_H__
 #define __ICNSS_QMI_H__
 
 #include "device_management_service_v01.h"
+#include "ip_multimedia_subsystem_private_service_v01.h"
 
 #define QDSS_TRACE_SEG_LEN_MAX 32
 #define QDSS_TRACE_FILE_NAME_MAX 16
@@ -189,6 +191,29 @@ void icnss_dms_deinit(struct icnss_priv *priv)
 {
 }
 
+static inline
+int icnss_process_wfc_call_ind_event(struct icnss_priv *priv,
+				     void *data)
+{
+	return 0;
+}
+
+static inline
+int icnss_process_twt_cfg_ind_event(struct icnss_priv *priv,
+				    void *data)
+{
+	return 0;
+}
+
+static inline
+int icnss_register_ims_service(struct icnss_priv *priv)
+{
+	return 0;
+}
+
+static inline
+void icnss_unregister_ims_service(struct icnss_priv *priv) {}
+
 int wlfw_subsys_restart_level_msg(struct icnss_priv *penv, uint8_t restart_level)
 {
 	return 0;
@@ -259,6 +284,12 @@ int icnss_wlfw_m3_dump_upload_done_send_sync(struct icnss_priv *priv,
 int icnss_qmi_get_dms_mac(struct icnss_priv *priv);
 int icnss_wlfw_wlan_mac_req_send_sync(struct icnss_priv *priv,
 				      u8 *mac, u32 mac_len);
+int icnss_process_wfc_call_ind_event(struct icnss_priv *priv,
+				     void *data);
+int icnss_process_twt_cfg_ind_event(struct icnss_priv *priv,
+				    void *data);
+int icnss_register_ims_service(struct icnss_priv *priv);
+void icnss_unregister_ims_service(struct icnss_priv *priv);
 int icnss_dms_init(struct icnss_priv *priv);
 void icnss_dms_deinit(struct icnss_priv *priv);
 int wlfw_subsys_restart_level_msg(struct icnss_priv *penv, uint8_t restart_level);

+ 1 - 1
wlan_platform_modules.bzl

@@ -98,7 +98,6 @@ def _define_modules_for_target_variant(target, variant):
                     True: [
                         "cnss2/qmi.c",
                         "cnss2/coexistence_service_v01.c",
-                        "cnss2/ip_multimedia_subsystem_private_service_v01.c",
                     ]
                 },
                 "CONFIG_PCI_MSM": {
@@ -219,6 +218,7 @@ def _define_modules_for_target_variant(target, variant):
         srcs = native.glob([
             "cnss_utils/wlan_firmware_service_v01.c",
             "cnss_utils/device_management_service_v01.c",
+            "cnss_utils/ip_multimedia_subsystem_private_service_v01.c",
             "cnss_utils/*.h"
         ]),
         kconfig = "cnss_utils/Kconfig",