Prechádzať zdrojové kódy

ipa: Add TSP feature

The change adds Traffic Shaping and Policing (TSP) feature
to the IPA driver:
* HAL addition for TSP tables translation to and from HW format
* Core TSP for table management
* IOCTL handlers for table management commands from userspace
* Debugfs addition that dumps the TSP configuration

The whole feature is added under CONFIG_IPA_TSP kbuild switch.

Change-Id: I11e444aeb93e10d6809873d02a8b3be7cc366755
Signed-off-by: Ilia Lin <[email protected]>
Ilia Lin 2 rokov pred
rodič
commit
ce9c934ced

+ 3 - 0
drivers/platform/msm/ipa/Kbuild

@@ -35,6 +35,9 @@ ipam-y += \
 	ipa_v3/ipa_eth_i.o \
 	ipa_v3/ipa_stats.o \
 
+ipam-$(CONFIG_IPA_TSP) += ipa_v3/ipa_tsp.o \
+	ipa_v3/ipahal/ipahal_tsp.o \
+
 ipam-$(CONFIG_RMNET_IPA3) += ipa_v3/rmnet_ipa.o ipa_v3/ipa_qmi_service_v01.o \
 	ipa_v3/ipa_qmi_service.o ipa_v3/rmnet_ctl_ipa.o \
 	ipa_v3/rmnet_ipa_fd_ioctl.o ipa_v3/rmnet_ll_ipa.o

+ 190 - 0
drivers/platform/msm/ipa/ipa_test_module/ipa_test_module_tsp.h

@@ -0,0 +1,190 @@
+//SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+/* This file should be removed once TSP feature POR */
+#ifndef _IPA_TEST_MODULE_TSP_H_
+#define _IPA_TEST_MODULE_TSP_H_
+
+#include <linux/msm_ipa.h>
+
+#define IPA_IOCTL_TSP_GET_INGR_TC_NUM           91
+#define IPA_IOCTL_TSP_GET_EGR_EP_NUM            92
+#define IPA_IOCTL_TSP_GET_EGR_TC_NUM            93
+#define IPA_IOCTL_TSP_GET_INGR_TC               94
+#define IPA_IOCTL_TSP_GET_EGR_EP                95
+#define IPA_IOCTL_TSP_GET_EGR_TC                96
+#define IPA_IOCTL_TSP_SET_INGR_TC               97
+#define IPA_IOCTL_TSP_SET_EGR_EP                98
+#define IPA_IOCTL_TSP_SET_EGR_TC                99
+#define IPA_IOCTL_TSP_COMMIT                    100
+#define IPA_IOCTL_TSP_RESET                     101
+
+/**
+ * struct ipa_ioc_tsp_ingress_class_params - IPA Ingress traffic-class params
+ *
+ * @max_burst: Maximal-burst allowed in bytes
+ * @max_rate: Maximal bandwidth rate for matching ingress traffic-class in 1.2MB/second units
+ * @include_l2_len: Bool - Include L2 size for bandwidth calculation?
+ */
+struct ipa_ioc_tsp_ingress_class_params {
+	__u16 max_burst;
+	__u16 max_rate;
+	__u32 include_l2_len;
+};
+
+/**
+ * struct ipa_ioc_tsp_ingress_class_get - IPA Ingress traffic-class get ioctl struct
+ *
+ * @index: Ingress traffic-class index
+ * @reserved: Reserved for alignment
+ * @params: Output parameter - Ingress traffic-class Params, valid only when
+ *          ioctl return val is non-negative
+ */
+struct ipa_ioc_tsp_ingress_class_get {
+	__u32 index;
+	__u32 reserved;
+	struct ipa_ioc_tsp_ingress_class_params params;
+};
+
+/**
+ * struct ipa_ioc_tsp_ingress_class_set - IPA Ingress traffic-class set ioctl struct
+ *
+ * @index: Ingress traffic-class index
+ * @commit: Bool - Commit the setting to the HW
+ * @params: Params to set
+ */
+struct ipa_ioc_tsp_ingress_class_set {
+	__u32 index;
+	__u32 commit;
+	struct ipa_ioc_tsp_ingress_class_params params;
+};
+
+/**
+ * struct ipa_ioc_tsp_egress_prod_params - IPA TSP-enabled producer params
+ *
+ * @client: For output - which "clients" pipe does this entry apply to?
+ * @max_rate: Maximal bandwidth rate for producer in 1.2MB/second units
+ * @max_burst: Maximal-burst allowed in bytes
+ * @max_out_bytes: max output size in bytes allowed per producer
+ * @tc_lo: Lowest egress traffic-class index assignes to this producer
+ * @tc_hi: Highest egress traffic-class index assignes to this producer
+ * @policing_by_max_out: Bool - enable policing by max output size
+ *                       in case of valid egress_tc, max output size policing will be valid
+ *                       regardless to this flag
+ * @reserved: Reserved for alignment
+ */
+struct ipa_ioc_tsp_egress_prod_params {
+	enum ipa_client_type client;
+	__u16 max_rate;
+	__u16 max_burst;
+	__u32 max_out_bytes;
+	__u8 tc_lo;
+	__u8 tc_hi;
+	__u8 policing_by_max_out;
+	__u8 reserved;
+};
+
+/**
+ * struct ipa_ioc_tsp_egress_prod_get - IPA TSP-enabled producer get ioctl struct
+ *
+ * @index: TSP-enabled producer index
+ * @reserved: Reserved for alignment
+ * @params: Output parameter - TSP-enabled producer Params, valid only when
+ *          ioctl return val is non-negative
+ */
+struct ipa_ioc_tsp_egress_prod_get {
+	__u32 index;
+	__u32 reserved;
+	struct ipa_ioc_tsp_egress_prod_params params;
+};
+
+/**
+ * struct ipa_ioc_tsp_egress_prod_set - IPA TSP-enabled producer set ioctl struct
+ *
+ * @index: Producer index
+ * @commit: Bool - Commit the setting to the HW
+ * @params: Params to set
+ */
+struct ipa_ioc_tsp_egress_prod_set {
+	__u32 index;
+	__u32 commit;
+	struct ipa_ioc_tsp_egress_prod_params params;
+};
+
+/**
+ * struct ipa_ioc_tsp_egress_class_params - IPA egress traffic-class params
+ *
+ * @guaranteed_rate: Guaranteed bandwidth rate for traffic-class in 1.2MB/second units
+ *      If guaranteed_rate, guaranteed_freq and guaranteed_burst are all set to 0,
+ *      the guaranteed bandwidth rate will be disabled,
+ *      and only maximal bandwidth rate will be considered.
+ * @max_rate: Maximal bandwidth rate for traffic-class in 1.2MB/second units
+ * @guaranteed_burst: Maximal-burst allowed for guaranteed bandwidth rate (in bytes)
+ * @max_burst: Maximal-burst allowed for maximal bandwidth rate (in bytes)
+ */
+struct ipa_ioc_tsp_egress_class_params {
+	__u16 guaranteed_rate;
+	__u16 max_rate;
+	__u16 guaranteed_burst;
+	__u16 max_burst;
+};
+
+/**
+ * struct ipa_ioc_tsp_egress_class_get - IPA Egress traffic-class get ioctl struct
+ *
+ * @index: Egress traffic-class index
+ * @reserved: Reserved for alignment
+ * @params: Output parameter - Egress traffic-class Params, valid only when
+ *          ioctl return val is non-negative
+ */
+struct ipa_ioc_tsp_egress_class_get {
+	__u32 index;
+	__u32 reserved;
+	struct ipa_ioc_tsp_egress_class_params params;
+};
+
+/**
+ * struct ipa_ioc_tsp_egress_class_set - IPA Engress traffic-class set ioctl struct
+ *
+ * @index: Egress traffic-class index
+ * @commit: Bool - Commit the setting to the HW
+ * @params: Params to set
+ */
+struct ipa_ioc_tsp_egress_class_set {
+	__u32 index;
+	__u32 commit;
+	struct ipa_ioc_tsp_egress_class_params params;
+};
+
+#define IPA_IOC_TSP_GET_INGR_TC_NUM _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_GET_INGR_TC_NUM, \
+				uint32_t)
+#define IPA_IOC_TSP_GET_EGR_EP_NUM _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_GET_EGR_EP_NUM, \
+				uint32_t)
+#define IPA_IOC_TSP_GET_EGR_TC_NUM _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_GET_EGR_TC_NUM, \
+				uint32_t)
+#define IPA_IOC_TSP_GET_INGR_TC _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_GET_INGR_TC, \
+				struct ipa_ioc_tsp_ingress_class_get)
+#define IPA_IOC_TSP_GET_EGR_EP _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_GET_EGR_EP, \
+				struct ipa_ioc_tsp_egress_prod_get)
+#define IPA_IOC_TSP_GET_EGR_TC _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_GET_EGR_TC, \
+				struct ipa_ioc_tsp_egress_class_get)
+#define IPA_IOC_TSP_SET_INGR_TC _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_SET_INGR_TC, \
+				struct ipa_ioc_tsp_ingress_class_set)
+#define IPA_IOC_TSP_SET_EGR_EP _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_SET_EGR_EP, \
+				struct ipa_ioc_tsp_egress_prod_set)
+#define IPA_IOC_TSP_SET_EGR_TC _IOWR(IPA_IOC_MAGIC, \
+				IPA_IOCTL_TSP_SET_EGR_TC, \
+				struct ipa_ioc_tsp_egress_class_set)
+#define IPA_IOC_TSP_COMMIT _IO(IPA_IOC_MAGIC, IPA_IOCTL_TSP_COMMIT)
+#define IPA_IOC_TSP_RESET _IO(IPA_IOC_MAGIC, IPA_IOCTL_TSP_RESET)
+
+#endif /* _IPA_TEST_MODULE_TSP_H_ */

+ 174 - 0
drivers/platform/msm/ipa/ipa_v3/ipa.c

@@ -63,6 +63,9 @@
 
 #include "ipa_i.h"
 #include "ipa_rm_i.h"
+#if defined(CONFIG_IPA_TSP)
+#include "ipa_tsp.h"
+#endif
 #include "ipahal.h"
 #include "ipahal_fltrt.h"
 
@@ -2829,6 +2832,15 @@ static long ipa3_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
 	struct ipa_ioc_eogre_info eogre_info;
 	struct ipa_ioc_macsec_info macsec_info;
 	struct ipa_macsec_map *macsec_map;
+#if defined(CONFIG_IPA_TSP)
+	struct ipa_ioc_tsp_ingress_class_get ingr_tc_get;
+	struct ipa_ioc_tsp_egress_class_get egr_tc_get;
+	struct ipa_ioc_tsp_egress_prod_get egr_ep_get;
+	struct ipa_ioc_tsp_ingress_class_set ingr_tc_set;
+	struct ipa_ioc_tsp_egress_class_set egr_tc_set;
+	struct ipa_ioc_tsp_egress_prod_set egr_ep_set;
+	u32 u32temp;
+#endif
 	bool send2uC, send2ipacm;
 	size_t sz;
 	int pre_entry;
@@ -4270,6 +4282,161 @@ static long ipa3_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
 	case IPA_IOC_SET_CONN_TRACK_EXC_RT_TBL_IDX:
 		retval = ipa3_set_nat_conn_track_exc_rt_tbl(arg, IPA_IP_v6);
 		break;
+#if defined(CONFIG_IPA_TSP)
+	case IPA_IOC_TSP_GET_INGR_TC_NUM:
+		u32temp = (u32)ipa3_ctx->tsp.ingr_tc_max;
+		goto send;
+	case IPA_IOC_TSP_GET_EGR_EP_NUM:
+		u32temp = (u32)ipa3_ctx->tsp.egr_ep_max;
+		goto send;
+	case IPA_IOC_TSP_GET_EGR_TC_NUM:
+		u32temp = (u32)ipa3_ctx->tsp.egr_tc_max;
+send:
+		if (copy_to_user((void __user *)arg, &u32temp, sizeof(u32temp))) {
+			retval = -EFAULT;
+			break;
+		}
+		break;
+
+	case IPA_IOC_TSP_GET_INGR_TC:
+		if (copy_from_user(&ingr_tc_get, (const void __user *)arg,
+			sizeof(struct ipa_ioc_tsp_ingress_class_get))) {
+			retval = -EFAULT;
+			break;
+		}
+
+		if (ingr_tc_get.index == 0 || ingr_tc_get.index > (u32)ipa3_ctx->tsp.ingr_tc_max) {
+			retval = -EINVAL;
+			break;
+		}
+
+		retval = ipa_tsp_get_ingr_tc(ingr_tc_get.index, &(ingr_tc_get.params));
+		if (retval != 0)
+			break;
+
+		if (copy_to_user((void __user *)arg, &ingr_tc_get,
+			sizeof(struct ipa_ioc_tsp_ingress_class_get))) {
+			retval = -EFAULT;
+			break;
+		}
+		break;
+
+	case IPA_IOC_TSP_GET_EGR_EP:
+		if (copy_from_user(&egr_ep_get, (const void __user *)arg,
+			sizeof(struct ipa_ioc_tsp_egress_prod_get))) {
+			retval = -EFAULT;
+			break;
+		}
+
+		if (egr_ep_get.index >= (u32)ipa3_ctx->tsp.egr_ep_max) {
+			retval = -EINVAL;
+			break;
+		}
+
+		retval = ipa_tsp_get_egr_ep(egr_ep_get.index, &(egr_ep_get.params));
+		if (retval != 0)
+			break;
+
+		if (copy_to_user((void __user *)arg, &egr_ep_get,
+			sizeof(struct ipa_ioc_tsp_egress_prod_get))) {
+			retval = -EFAULT;
+			break;
+		}
+		break;
+
+	case IPA_IOC_TSP_GET_EGR_TC:
+		if (copy_from_user(&egr_tc_get, (const void __user *)arg,
+			sizeof(struct ipa_ioc_tsp_egress_class_get))) {
+			retval = -EFAULT;
+			break;
+		}
+
+		if (egr_tc_get.index == 0 || egr_tc_get.index > (u32)ipa3_ctx->tsp.egr_tc_max) {
+			retval = -EINVAL;
+			break;
+		}
+
+		retval = ipa_tsp_get_egr_tc(egr_tc_get.index, &(egr_tc_get.params));
+		if (retval != 0)
+			break;
+
+		if (copy_to_user((void __user *)arg, &egr_tc_get,
+			sizeof(struct ipa_ioc_tsp_egress_class_get))) {
+			retval = -EFAULT;
+			break;
+		}
+		break;
+
+	case IPA_IOC_TSP_SET_INGR_TC:
+		if (copy_from_user(&ingr_tc_set, (const void __user *)arg,
+			sizeof(struct ipa_ioc_tsp_ingress_class_set))) {
+			retval = -EFAULT;
+			break;
+		}
+
+		if (ingr_tc_set.index == 0 || ingr_tc_set.index > (u32)ipa3_ctx->tsp.ingr_tc_max) {
+			retval = -EINVAL;
+			break;
+		}
+
+		retval = ipa_tsp_set_ingr_tc(ingr_tc_set.index, &(ingr_tc_set.params));
+		if (retval != 0)
+			break;
+
+		if (ingr_tc_set.commit)
+			retval = ipa_tsp_commit();
+
+		break;
+
+	case IPA_IOC_TSP_SET_EGR_EP:
+		if (copy_from_user(&egr_ep_set, (const void __user *)arg,
+			sizeof(struct ipa_ioc_tsp_egress_prod_set))) {
+			retval = -EFAULT;
+			break;
+		}
+
+		if (egr_ep_set.index >= (u32)ipa3_ctx->tsp.egr_ep_max) {
+			retval = -EINVAL;
+			break;
+		}
+
+		retval = ipa_tsp_set_egr_ep(egr_ep_set.index, &(egr_ep_set.params));
+		if (retval != 0)
+			break;
+
+		if (egr_ep_set.commit)
+			retval = ipa_tsp_commit();
+		break;
+
+	case IPA_IOC_TSP_SET_EGR_TC:
+		if (copy_from_user(&egr_tc_set, (const void __user *)arg,
+			sizeof(struct ipa_ioc_tsp_egress_class_set))) {
+			retval = -EFAULT;
+			break;
+		}
+
+		if (egr_tc_set.index == 0 || egr_tc_set.index > (u32)ipa3_ctx->tsp.egr_tc_max) {
+			retval = -EINVAL;
+			break;
+		}
+
+		retval = ipa_tsp_set_egr_tc(egr_tc_set.index, &(egr_tc_set.params));
+		if (retval != 0)
+			break;
+
+		if (egr_tc_set.commit)
+			retval = ipa_tsp_commit();
+
+		break;
+
+	case IPA_IOC_TSP_COMMIT:
+		retval = ipa_tsp_commit();
+		break;
+
+	case IPA_IOC_TSP_RESET:
+		retval = ipa_tsp_reset();
+		break;
+#endif
 
 	default:
 		IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
@@ -8003,6 +8170,13 @@ static int ipa3_post_init(const struct ipa3_plat_drv_res *resource_p,
 		IPAERR(":ntn init failed (%d)\n", -result);
 	else
 		IPADBG(":ntn init ok\n");
+#if defined(CONFIG_IPA_TSP)
+	result = ipa_tsp_init();
+	if (result)
+		IPAERR(":TSP init failed (%d)\n", -result);
+	else
+		IPADBG(":TSP init ok\n");
+#endif
 
 	result = ipa_hw_stats_init();
 	if (result)

+ 112 - 3
drivers/platform/msm/ipa/ipa_v3/ipa_debugfs.c

@@ -12,10 +12,15 @@
 #include <linux/stringify.h>
 #include "ipa_i.h"
 #include "ipa_rm_i.h"
+#include "ipahal_reg.h"
 #include "ipahal_nat.h"
 #include "ipa_odl.h"
 #include "ipa_qmi_service.h"
-
+#if defined(CONFIG_IPA_TSP)
+/* The following line should be removed once TSP feature is POR */
+#include "ipa_test_module_tsp.h"
+#include "ipahal_tsp.h"
+#endif
 #define IPA_MAX_ENTRY_STRING_LEN 500
 #define IPA_MAX_MSG_LEN 4096
 #define IPA_DBG_MAX_RULE_IN_TBL 128
@@ -368,7 +373,8 @@ int _ipa_read_ep_reg_v3_0(char *buf, int max_len, int pipe)
 		"IPA_ENDP_INIT_HOL_EN_%u=0x%x\n"
 		"IPA_ENDP_INIT_HOL_TIMER_%u=0x%x\n"
 		"IPA_ENDP_INIT_DEAGGR_%u=0x%x\n"
-		"IPA_ENDP_INIT_CFG_%u=0x%x\n",
+		"IPA_ENDP_INIT_CFG_%u=0x%x\n"
+		"IPA_ENDP_INIT_PROD_CFG_%u=0x%x\n",
 		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_NAT_n, pipe),
 		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_HDR_n, pipe),
 		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_HDR_EXT_n, pipe),
@@ -379,7 +385,8 @@ int _ipa_read_ep_reg_v3_0(char *buf, int max_len, int pipe)
 		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_HOL_BLOCK_EN_n, pipe),
 		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_HOL_BLOCK_TIMER_n, pipe),
 		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_DEAGGR_n, pipe),
-		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_CFG_n, pipe));
+		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_CFG_n, pipe),
+		pipe, ipahal_read_reg_n(IPA_ENDP_INIT_PROD_CFG_n, pipe));
 }
 
 /**
@@ -3306,8 +3313,103 @@ static ssize_t ipa3_write_nat_table_move(struct file *file,
 
 	return count;
 }
+#if defined(CONFIG_IPA_TSP)
+static ssize_t ipa3_read_tsp(struct file *file, char __user *buf, size_t count, loff_t *ppos)
+{
+	int i, nbytes = 0;
+	struct ipahal_ipa_state_tsp state_tsp;
+	u32 qm_non_empty;
+	struct ipa_ioc_tsp_ingress_class_params ingr_tc;
+	struct ipa_ioc_tsp_egress_prod_params egr_ep;
+	struct ipa_ioc_tsp_egress_class_params egr_tc;
+
+	/* Print the global TSP state flags */
+	IPA_ACTIVE_CLIENTS_INC_SIMPLE();
+	ipahal_read_reg_fields(IPA_STATE_TSP, &state_tsp);
+	ipahal_read_reg_fields(IPA_STATE_QMNGR_QUEUE_NONEMPTY, &qm_non_empty);
+	IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
+
+	if (state_tsp.traffic_shaper_idle)
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"Traffic-Sahper module IDLE\n");
+	if (state_tsp.traffic_shaper_fifo_empty)
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"Traffic-Sahper FIFO empty\n");
+	if (state_tsp.queue_mngr_idle)
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"QMNGR overall IDLE\n");
+	if (state_tsp.queue_mngr_head_idle)
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"QMNGR head module IDLE\n");
+	if (state_tsp.queue_mngr_shared_idle)
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"QMNGR shared module IDLE\n");
+	if (state_tsp.queue_mngr_tail_idle)
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"QMNGR tail module IDLE\n");
+	if (state_tsp.queue_mngr_block_ctrl_idle)
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"Block control module IDLE\n");
+
+	nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"QM non-empty bitmask: 0x%08X\n", qm_non_empty);
+
+	/* Dump Ingress Class, Egress Producer and Egress Class tables */
+	nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"Ingress Trafic Class Table:\n");
+	nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"TC Index\tMax Rate\tMax Burst\tInclude L2\n");
+	for (i = 1; i <= ipa3_ctx->tsp.ingr_tc_max; i++) {
+		ipahal_tsp_parse_hw_ingr_tc(ipa3_ctx->tsp.ingr_tc_tbl.base, i, &ingr_tc);
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"%02d:\t\t%u\t\t%u\t\t%u\n",
+			i, ingr_tc.max_rate, ingr_tc.max_burst, ingr_tc.include_l2_len);
+	}
+
+	nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"Egress Producer Table:\n");
+	nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"EP Index\tClient\tMax Rate\tMax Burst\n");
+	for (i = 0; i < ipa3_ctx->tsp.egr_ep_max; i++) {
+		ipahal_tsp_parse_hw_egr_ep(ipa3_ctx->tsp.egr_ep_tbl.base, i, &egr_ep);
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"%d:\t\t%d\t%u\t\t%u\n",
+			i, ipa3_ctx->tsp.egr_ep_config[i], egr_ep.max_rate, egr_ep.max_burst);
+	}
+
+	nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"Egress Trafic Class Table:\n");
+	nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"TC Index\tMax Rate\tMax Burst\tG. Rate\tG. Burst\n");
+	for (i = 1; i <= ipa3_ctx->tsp.egr_tc_max; i++) {
+		ipahal_tsp_parse_hw_egr_tc(ipa3_ctx->tsp.egr_tc_tbl.base, i, &egr_tc);
+		nbytes += scnprintf(dbg_buff + nbytes, IPA_MAX_MSG_LEN - nbytes,
+			"%02d:\t\t%u\t\t%u\t\t%u\t%u\n",
+			i, egr_tc.max_rate, egr_tc.max_burst,
+			egr_tc.guaranteed_rate, egr_tc.guaranteed_burst);
+	}
+
+	return simple_read_from_buffer(buf, count, ppos, dbg_buff, nbytes);
+}
 
+static ssize_t ipa3_write_tsp(struct file *file, const char __user *buf,
+			      size_t count, loff_t *ppos) {
 
+	int ret;
+	u8 option = 0;
+
+	if (count >= sizeof(dbg_buff))
+		return -EFAULT;
+
+	ret = kstrtou8_from_user(buf, count, 0, &option);
+	if(ret)
+		return ret;
+
+	pr_err("TSP write is not implemented.\n");
+
+	return count;
+}
+#endif
 static const struct ipa3_debugfs_file debugfs_files[] = {
 	{
 		"gen_reg", IPA_READ_ONLY_MODE, NULL, {
@@ -3532,6 +3634,13 @@ static const struct ipa3_debugfs_file debugfs_files[] = {
 			.read = ipa3_read_ipa_max_napi_sort_page_thrshld,
 			.write = ipa3_write_ipa_max_napi_sort_page_thrshld,
 		}
+#if defined(CONFIG_IPA_TSP)
+	}, {
+		"tsp", IPA_READ_WRITE_MODE, NULL, {
+			.read = ipa3_read_tsp,
+			.write = ipa3_write_tsp,
+		}
+#endif
 	},
 };
 

+ 16 - 1
drivers/platform/msm/ipa/ipa_v3/ipa_i.h

@@ -2078,7 +2078,19 @@ struct ipa_ntn3_client_stats {
 	struct ipa_ntn3_stats_rx rx_stats;
 	struct ipa_ntn3_stats_tx tx_stats;
 };
-
+#if defined(CONFIG_IPA_TSP)
+struct ipa3_tsp_ctx {
+	u8 ingr_tc_max;
+	u8 egr_ep_max;
+	u8 egr_tc_max;
+	enum ipa_client_type *egr_ep_config;
+	u32 egr_tc_range_mask;
+	struct ipa_mem_buffer ingr_tc_tbl;
+	struct ipa_mem_buffer egr_ep_tbl;
+	struct ipa_mem_buffer egr_tc_tbl;
+	struct ipa_mem_buffer qm_tlv_mem;
+};
+#endif
 
 /**
  * struct ipa3_context - IPA context
@@ -2368,6 +2380,9 @@ struct ipa3_context {
 	struct ipa3_aqc_ctx aqc_ctx;
 	struct ipa3_rtk_ctx rtk_ctx;
 	struct ipa3_ntn_ctx ntn_ctx;
+#if defined(CONFIG_IPA_TSP)
+	struct ipa3_tsp_ctx tsp;
+#endif
 	atomic_t ipa_clk_vote;
 
 	int (*client_lock_unlock[IPA_MAX_CLNT])(bool is_lock);

+ 365 - 0
drivers/platform/msm/ipa/ipa_v3/ipa_tsp.c

@@ -0,0 +1,365 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+
+#include <linux/bits.h>
+#include "ipa_i.h"
+#include "ipahal.h"
+#include "ipahal_reg.h"
+#include "ipahal_tsp.h"
+
+/*
+ * Every Producer can hold up to 16 K Queue elements and 8 Queues.
+ * Thus, software allocates up to 512 Kbytes (32 bytes *16 K QEs) contiguously in memory.
+ */
+#define IPA_TSP_QM_DRAM_BYTESIZE (16384 * 32)
+
+static void __ipa_tsp_get_supported_constrains(void)
+{
+	if (ipa3_ctx->ipa_hw_type >= IPA_HW_v5_5) {
+		struct ipahal_ipa_flavor_9 ipa_flavor_9;
+
+		ipahal_read_reg_fields(IPA_FLAVOR_9, &ipa_flavor_9);
+		ipa3_ctx->tsp.ingr_tc_max = ipa_flavor_9.ipa_tsp_max_ingr_tc;
+		ipa3_ctx->tsp.egr_ep_max = ipa_flavor_9.ipa_tsp_max_prod;
+		ipa3_ctx->tsp.egr_tc_max = ipa_flavor_9.ipa_tsp_max_egr_tc;
+	} else {
+		ipa3_ctx->tsp.ingr_tc_max = 0;
+		ipa3_ctx->tsp.egr_ep_max = 0;
+		ipa3_ctx->tsp.egr_tc_max = 0;
+
+		IPAERR("TSP not supported. ingr_tc_max=%d egr_ep_max=%d egr_tc_max=%d\n",
+		       ipa3_ctx->tsp.ingr_tc_max, ipa3_ctx->tsp.egr_ep_max, ipa3_ctx->tsp.egr_tc_max);
+	}
+}
+
+int ipa_tsp_init(void)
+{
+	int i, ret = 0;
+	gfp_t flag = GFP_KERNEL;
+	dma_addr_t qm_tlv_base;
+	u32 qm_tlv_size;
+
+	/*
+	 * Cache the TSP table size constrains
+	 */
+	__ipa_tsp_get_supported_constrains();
+	if (!ipa3_ctx->tsp.ingr_tc_max || !ipa3_ctx->tsp.egr_ep_max || !ipa3_ctx->tsp.egr_tc_max) {
+		IPAERR("TSP not supported. ingr_tc_max=%d egr_ep_max=%d egr_tc_max=%d\n",
+		       ipa3_ctx->tsp.ingr_tc_max, ipa3_ctx->tsp.egr_ep_max, ipa3_ctx->tsp.egr_tc_max);
+		return -EFAULT;
+	}
+
+	/*
+	 * Allocate DMA accessible memory for HW commitable ipa3_ctx->tsp.ingr_tc_tbl,
+	 * ipa3_ctx->tsp.egr_ep_tbl and ipa3_ctx->tsp.egr_tc_tbl
+	 */
+	ipa3_ctx->tsp.ingr_tc_tbl.size = ipa3_ctx->tsp.ingr_tc_max * IPA_TSP_INGR_TC_SIZE;
+	ipa3_ctx->tsp.egr_ep_tbl.size = ipa3_ctx->tsp.egr_ep_max * IPA_TSP_EGR_EP_SIZE;
+	ipa3_ctx->tsp.egr_tc_tbl.size = ipa3_ctx->tsp.egr_tc_max * IPA_TSP_EGR_TC_SIZE;
+
+	if (!ipa3_ctx->tsp.ingr_tc_tbl.size || !ipa3_ctx->tsp.egr_ep_tbl.size || !ipa3_ctx->tsp.egr_tc_tbl.size) {
+		IPAERR("ingr_tc_tbl.size=%d egr_ep_tbl.size=%d egr_tc_tbl.size=%d\n",
+		       ipa3_ctx->tsp.ingr_tc_tbl.size, ipa3_ctx->tsp.egr_ep_tbl.size, ipa3_ctx->tsp.egr_tc_tbl.size);
+		return -EFAULT;
+	}
+
+	if (!ipa3_ctx->pdev || !&(ipa3_ctx->tsp.ingr_tc_tbl.phys_base)) {
+		IPAERR("ipa3_ctx->pdev=%16X &(ipa3_ctx->tsp.ingr_tc_tbl.phys_base)=%16X\n",
+		       ipa3_ctx->pdev, &(ipa3_ctx->tsp.ingr_tc_tbl.phys_base));
+		return -EFAULT;
+	}
+	IPAERR("ipa3_ctx->pdev=%16X &(ipa3_ctx->tsp.ingr_tc_tbl.phys_base)=%16X\n",
+	       ipa3_ctx->pdev, &(ipa3_ctx->tsp.ingr_tc_tbl.phys_base));
+
+	ipa3_ctx->tsp.ingr_tc_tbl.base = dma_alloc_coherent(ipa3_ctx->pdev,
+		ipa3_ctx->tsp.ingr_tc_tbl.size, &(ipa3_ctx->tsp.ingr_tc_tbl.phys_base), GFP_KERNEL);
+	if (!ipa3_ctx->tsp.ingr_tc_tbl.base) {
+		IPAERR("Failed to allocate cache memory for ingress TC TSP table.\n");
+		return -ENOMEM;
+	}
+
+	ipa3_ctx->tsp.egr_ep_tbl.base = dma_alloc_coherent(ipa3_ctx->pdev,
+		ipa3_ctx->tsp.egr_ep_tbl.size, &(ipa3_ctx->tsp.egr_ep_tbl.phys_base), GFP_KERNEL);
+	if (!ipa3_ctx->tsp.egr_ep_tbl.base) {
+		IPAERR("Failed to allocate cache memory for egress producer TSP table.\n");
+		ret = -ENOMEM;
+		goto free_ingr;
+	}
+
+	ipa3_ctx->tsp.egr_tc_tbl.base = dma_alloc_coherent(ipa3_ctx->pdev,
+		ipa3_ctx->tsp.egr_tc_tbl.size, &(ipa3_ctx->tsp.egr_tc_tbl.phys_base), GFP_KERNEL);
+	if (!ipa3_ctx->tsp.egr_tc_tbl.base) {
+		IPAERR("Failed to allocate cache memory for egress TC TSP table.\n");
+		ret = -ENOMEM;
+		goto free_ep;
+	}
+
+	/*
+	 * Allocate configured producer array
+	 */
+	ipa3_ctx->tsp.egr_ep_config =
+		(enum ipa_client_type *)
+		kzalloc(ipa3_ctx->tsp.egr_ep_max * sizeof(enum ipa_client_type), GFP_KERNEL);
+
+	if (ipa3_ctx->tsp.egr_ep_config == NULL) {
+		IPAERR("Failed to allocate cache memory for egress producer config.\n");
+		ret = -ENOMEM;
+		goto free_egr;
+	}
+
+	/*
+	 * Init configured producer array
+	 */
+	for (i = 0; i < ipa3_ctx->tsp.egr_ep_max; i++)
+		ipa3_ctx->tsp.egr_ep_config[i] = IPA_CLIENT_MAX;
+	ipa3_ctx->tsp.egr_tc_range_mask = 0;
+
+	/*
+	 * Allocate memory for TLV-IN queues:
+	 * Every Producer can hold up to 16 K Queue elements and 8 Queues.
+	 * Thus, software allocates up to 512 Kbytes (32 bytes *16 K QEs) contiguously in memory.
+	 * Note: All queues of all PRODs have to be together in contiguous memory!
+	 * Must be 128B aligned.
+	 */
+	qm_tlv_size = IPA_TSP_QM_DRAM_BYTESIZE;
+	ipa3_ctx->tsp.qm_tlv_mem.size = qm_tlv_size + 128; // + 128 bytes to fit alignment
+alloc:
+	ipa3_ctx->tsp.qm_tlv_mem.base = dma_alloc_coherent(ipa3_ctx->pdev,
+		ipa3_ctx->tsp.qm_tlv_mem.size, &(ipa3_ctx->tsp.qm_tlv_mem.phys_base), flag);
+	if (!ipa3_ctx->tsp.qm_tlv_mem.base) {
+		if (flag == GFP_KERNEL) {
+			flag = GFP_ATOMIC;
+			goto alloc;
+		}
+		IPAERR("fail to alloc DMA buff of size %d\n", ipa3_ctx->tsp.qm_tlv_mem.size);
+		ret = -ENOMEM;
+		goto free_ep_conf;
+	}
+
+	/* Get 128B aligned address inside the buffer */
+	qm_tlv_base = ipa3_ctx->tsp.qm_tlv_mem.phys_base;
+	if (qm_tlv_base & 0x7F) {
+		qm_tlv_base += 0x80;
+		qm_tlv_base &= !0x7F;
+	}
+
+	ipahal_write_reg(IPA_TSP_QM_EXTERNAL_BADDR_LSB, qm_tlv_base & 0xFFFFFFFF);
+	ipahal_write_reg(IPA_TSP_QM_EXTERNAL_BADDR_MSB, qm_tlv_base >> 32);
+	ipahal_write_reg(IPA_TSP_QM_EXTERNAL_SIZE, qm_tlv_size >> 12); //Size in 4kB resolution
+
+	goto done;
+
+free_ep_conf:
+	kfree(ipa3_ctx->tsp.egr_ep_config);
+	ipa3_ctx->tsp.egr_ep_config = NULL;
+free_egr:
+	dma_free_coherent(ipa3_ctx->pdev, ipa3_ctx->tsp.egr_tc_tbl.size,
+		ipa3_ctx->tsp.egr_tc_tbl.base, ipa3_ctx->tsp.egr_tc_tbl.phys_base);
+	ipa3_ctx->tsp.egr_tc_tbl.base = NULL;
+free_ep:
+	dma_free_coherent(ipa3_ctx->pdev, ipa3_ctx->tsp.egr_ep_tbl.size,
+		ipa3_ctx->tsp.egr_ep_tbl.base, ipa3_ctx->tsp.egr_ep_tbl.phys_base);
+	ipa3_ctx->tsp.egr_ep_tbl.base = NULL;
+free_ingr:
+	dma_free_coherent(ipa3_ctx->pdev, ipa3_ctx->tsp.ingr_tc_tbl.size,
+		ipa3_ctx->tsp.ingr_tc_tbl.base, ipa3_ctx->tsp.ingr_tc_tbl.phys_base);
+	ipa3_ctx->tsp.ingr_tc_tbl.base = NULL;
+done:
+	return ret;
+}
+
+int ipa_tsp_get_ingr_tc(u8 index, struct ipa_ioc_tsp_ingress_class_params *output)
+{
+	/* The function is internal only, assuming valid params */
+
+	ipahal_tsp_parse_hw_ingr_tc(ipa3_ctx->tsp.ingr_tc_tbl.base, index, output);
+	output->include_l2_len = !!(ipahal_read_reg(IPA_TSP_INGRESS_POLICING_CFG) & (0x1 << index));
+
+	return 0;
+}
+
+int ipa_tsp_get_egr_ep(u8 index, struct ipa_ioc_tsp_egress_prod_params *output)
+{
+	u32 regval;
+	struct ipa_ep_cfg_prod_cfg prod_cfg;
+
+	/* The function is internal only, assuming valid params */
+
+	ipahal_tsp_parse_hw_egr_ep(ipa3_ctx->tsp.egr_ep_tbl.base, index, output);
+
+	output->client = ipa3_ctx->tsp.egr_ep_config[index];
+
+	regval = ipahal_read_reg_n_fields(IPA_ENDP_INIT_PROD_CFG_n,
+		ipa3_get_ep_mapping(output->client), (void *)&prod_cfg);
+
+	output->max_out_bytes = prod_cfg.max_output_size << 6; // max_output_size*64
+	output->policing_by_max_out = prod_cfg.max_output_size_drop_enable;
+	output->tc_lo = prod_cfg.egress_tc_lowest;
+	output->tc_hi = prod_cfg.egress_tc_highest;
+
+	return 0;
+}
+
+int ipa_tsp_get_egr_tc(u8 index, struct ipa_ioc_tsp_egress_class_params *output)
+{
+	/* The function is internal only, assuming valid params */
+
+	ipahal_tsp_parse_hw_egr_tc(ipa3_ctx->tsp.egr_tc_tbl.base, index, output);
+
+	return 0;
+}
+
+int ipa_tsp_set_ingr_tc(u8 index, const struct ipa_ioc_tsp_ingress_class_params *input)
+{
+	/* The function is internal only, assuming valid params */
+
+	ipahal_tsp_fill_hw_ingr_tc(input, ipa3_ctx->tsp.ingr_tc_tbl.base, index);
+	ipahal_write_reg_mask(IPA_TSP_INGRESS_POLICING_CFG,
+		0x1 << input->include_l2_len, 0x1 << index);
+
+	return 0;
+}
+
+int ipa_tsp_set_egr_ep(u8 index, const struct ipa_ioc_tsp_egress_prod_params *input)
+{
+	u32 regval, ep_index, ep_tc_mask, new_tc_range_mask;
+	struct ipa_ep_cfg_prod_cfg prod_cfg;
+	bool cleanup = false;
+
+	ep_tc_mask = GENMASK(input->tc_hi, input->tc_lo);
+	new_tc_range_mask = ipa3_ctx->tsp.egr_tc_range_mask;
+
+	ep_index = ipa3_get_ep_mapping(ipa3_ctx->tsp.egr_ep_config[index]);
+	regval = ipahal_read_reg_n_fields(
+		IPA_ENDP_INIT_PROD_CFG_n, ep_index, (void *)&prod_cfg);
+
+	if (ipa3_ctx->tsp.egr_ep_config[index] != IPA_CLIENT_MAX &&
+	    ipa3_ctx->tsp.egr_ep_config[index] != input->client) {
+		cleanup = true;
+		new_tc_range_mask &= !GENMASK(prod_cfg.egress_tc_highest,prod_cfg.egress_tc_lowest);
+	}
+
+	if (ep_tc_mask & new_tc_range_mask) {
+		IPAERR("New egress TC range overlaps existing.\n");
+		return -EINVAL;
+	}
+
+	if (cleanup) {
+		/* Cleanup old producer config */
+		prod_cfg.tsp_enable = false;
+		prod_cfg.egress_tc_lowest = 0;
+		prod_cfg.egress_tc_highest = 0;
+		if (ipa3_cfg_ep_prod_cfg(ep_index, &prod_cfg) != 0) {
+			IPAERR("Failed configuring the producer EP.\n");
+			return -EFAULT;
+		}
+	}
+
+	prod_cfg.tsp_enable = true;
+	prod_cfg.tsp_idx = index;
+	prod_cfg.max_output_size = input->max_out_bytes >> 6; // max_out_bytes/64
+	prod_cfg.max_output_size_drop_enable = input->policing_by_max_out;
+	prod_cfg.egress_tc_lowest = input->tc_lo;
+	prod_cfg.egress_tc_highest = input->tc_hi;
+	if (ipa3_cfg_ep_prod_cfg(ipa3_get_ep_mapping(input->client), &prod_cfg) != 0) {
+		IPAERR("Failed configuring the producer EP.\n");
+		return -EFAULT;
+	}
+
+	ipa3_ctx->tsp.egr_ep_config[index] = input->client;
+	ipa3_ctx->tsp.egr_tc_range_mask = new_tc_range_mask | ep_tc_mask;
+	ipahal_tsp_fill_hw_egr_ep(input, ipa3_ctx->tsp.egr_ep_tbl.base, index);
+
+	return 0;
+}
+
+int ipa_tsp_set_egr_tc(u8 index, const struct ipa_ioc_tsp_egress_class_params *input)
+{
+	/* The function is internal only, assuming valid params */
+
+	ipahal_tsp_fill_hw_egr_tc(input, ipa3_ctx->tsp.egr_tc_tbl.base, index);
+
+	/*
+	 * If guaranteed_rate and guaranteed_burst are set to 0,
+	 * the guaranteed bandwidth rate will be disabled,
+	 * and only maximal bandwidth rate will be considered.
+	 */
+	if (input->guaranteed_rate || input->guaranteed_burst)
+		ipahal_write_reg_mask(IPA_TSP_EGRESS_POLICING_CFG, 0x1 << index, 0x1 << index);
+	else
+		ipahal_write_reg_mask(IPA_TSP_EGRESS_POLICING_CFG, 0x0, 0x1 << index);
+
+	return 0;
+}
+
+int ipa_tsp_commit(void)
+{
+	int ret = 0;
+	u32 ingr_tc_base, egr_tc_base, prod_base;
+	void *ingr_tc_mmio, *egr_tc_mmio, *prod_mmio;
+
+	ingr_tc_base = ipahal_read_reg(IPA_RAM_INGRESS_POLICER_DB_BASE_ADDR);
+	egr_tc_base = ipahal_read_reg(IPA_RAM_EGRESS_SHAPING_TC_DB_BASE_ADDR);
+	prod_base = ipahal_read_reg(IPA_RAM_EGRESS_SHAPING_PROD_DB_BASE_ADDR);
+
+	/* map IPA SRAM */
+	ingr_tc_mmio = ioremap(ipa3_ctx->ipa_wrapper_base + ingr_tc_base,
+		ipa3_ctx->tsp.ingr_tc_tbl.size);
+	egr_tc_mmio = ioremap(ipa3_ctx->ipa_wrapper_base + egr_tc_base,
+		ipa3_ctx->tsp.egr_tc_tbl.size);
+	prod_mmio = ioremap(ipa3_ctx->ipa_wrapper_base + prod_base,
+		ipa3_ctx->tsp.egr_ep_tbl.size);
+	if (!ingr_tc_mmio || !egr_tc_mmio || !prod_mmio) {
+		IPAERR("Failed to ioremap TSP SRAM\n");
+		ret = -ENOMEM;
+		goto end;
+	}
+
+	/*
+	 * The tables are located in the HW SRAM area, and we can't issue the DMA IMM, because
+	 * the offset field is restricted to 16 addresses in it.
+	 * Therefore, we do memcopy instead, but the infrastructure will be ready for DMA
+	 * in future IPA versions.
+	 */
+	memcpy_toio(ingr_tc_mmio, ipa3_ctx->tsp.ingr_tc_tbl.base, ipa3_ctx->tsp.ingr_tc_tbl.size);
+	memcpy_toio(egr_tc_mmio, ipa3_ctx->tsp.egr_tc_tbl.base, ipa3_ctx->tsp.egr_tc_tbl.size);
+	memcpy_toio(prod_mmio, ipa3_ctx->tsp.egr_ep_tbl.base, ipa3_ctx->tsp.egr_ep_tbl.size);
+
+end:
+	if (ingr_tc_mmio)
+		iounmap(ingr_tc_mmio);
+	if (egr_tc_mmio)
+		iounmap(egr_tc_mmio);
+	if (prod_mmio)
+		iounmap(prod_mmio);
+	return ret;
+}
+
+int ipa_tsp_reset(void)
+{
+	int i;
+	struct ipa_ep_cfg_prod_cfg prod_cfg = {0};
+
+	for (i = 0;
+	      i < ipa3_ctx->tsp.egr_ep_max && ipa3_ctx->tsp.egr_ep_config[i] < IPA_CLIENT_MAX; i++)
+		ipa3_cfg_ep_prod_cfg(ipa3_get_ep_mapping(ipa3_ctx->tsp.egr_ep_config[i]),
+		      &prod_cfg);
+
+	if (ipa3_ctx->tsp.ingr_tc_tbl.base)
+		memset(ipa3_ctx->tsp.ingr_tc_tbl.base, 0, ipa3_ctx->tsp.ingr_tc_tbl.size);
+	if (ipa3_ctx->tsp.egr_tc_tbl.base)
+		memset(ipa3_ctx->tsp.egr_tc_tbl.base, 0, ipa3_ctx->tsp.egr_tc_tbl.size);
+	if (ipa3_ctx->tsp.egr_ep_tbl.base)
+		memset(ipa3_ctx->tsp.egr_ep_tbl.base, 0, ipa3_ctx->tsp.egr_ep_tbl.size);
+
+	/* Reinit configured producer array */
+	for (i = 0; i < ipa3_ctx->tsp.egr_ep_max; i++)
+		ipa3_ctx->tsp.egr_ep_config[i] = IPA_CLIENT_MAX;
+
+	ipa3_ctx->tsp.egr_tc_range_mask = 0;
+
+	return ipa_tsp_commit();
+}
+

+ 23 - 0
drivers/platform/msm/ipa/ipa_v3/ipa_tsp.h

@@ -0,0 +1,23 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+/*
+ * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+
+#ifndef _IPA_TSP_H_
+#define _IPA_TSP_H_
+
+#include <linux/msm_ipa.h>
+/* The following line should be removed once TSP feature is POR */
+#include "ipa_test_module_tsp.h"
+
+int ipa_tsp_init(void);
+int ipa_tsp_commit(void);
+int ipa_tsp_reset(void);
+int ipa_tsp_get_ingr_tc(u8 index, struct ipa_ioc_tsp_ingress_class_params *output);
+int ipa_tsp_get_egr_ep(u8 index, struct ipa_ioc_tsp_egress_prod_params *output);
+int ipa_tsp_get_egr_tc(u8 index, struct ipa_ioc_tsp_egress_class_params *output);
+int ipa_tsp_set_ingr_tc(u8 index, const struct ipa_ioc_tsp_ingress_class_params *input);
+int ipa_tsp_set_egr_ep(u8 index, const struct ipa_ioc_tsp_egress_prod_params *input);
+int ipa_tsp_set_egr_tc(u8 index, const struct ipa_ioc_tsp_egress_class_params *input);
+
+#endif /* _IPA_TSP_H_ */

+ 4 - 0
drivers/platform/msm/ipa/ipa_v3/ipahal/ipahal_reg.c

@@ -177,6 +177,7 @@ static const char *ipareg_name_to_str[IPA_REG_MAX] = {
 	__stringify(IPA_TSP_INGRESS_POLICING_CFG),
 	__stringify(IPA_TSP_EGRESS_POLICING_CFG),
 	__stringify(IPA_STAT_TSP_DROP_BASE),
+	__stringify(IPA_STATE_QMNGR_QUEUE_NONEMPTY),
 	__stringify(IPA_RAM_INGRESS_POLICER_DB_BASE_ADDR),
 	__stringify(IPA_RAM_EGRESS_SHAPING_PROD_DB_BASE_ADDR),
 	__stringify(IPA_RAM_EGRESS_SHAPING_TC_DB_BASE_ADDR),
@@ -5208,6 +5209,9 @@ static struct ipahal_reg_obj ipahal_reg_objs[IPA_HW_MAX][IPA_REG_MAX] = {
 	[IPA_HW_v5_5][IPA_STAT_TSP_DROP_BASE] = {
 		ipareg_construct_dummy, ipareg_parse_dummy,
 		0x00000A14, 0, 0, 0, 0, 0},
+	[IPA_HW_v5_5][IPA_STATE_QMNGR_QUEUE_NONEMPTY] = {
+		ipareg_construct_dummy, ipareg_parse_dummy,
+		0x00000A18, 0, 0, 0, 0, 0},
 	[IPA_HW_v5_5][IPA_ENDP_INIT_ULSO_CFG_n] = {
 		ipareg_construct_dummy, ipareg_parse_dummy,
 		0x00001070, 0x80, 0, 0, 0, 0},

+ 1 - 0
drivers/platform/msm/ipa/ipa_v3/ipahal/ipahal_reg.h

@@ -175,6 +175,7 @@ enum ipahal_reg_name {
 	IPA_TSP_INGRESS_POLICING_CFG,
 	IPA_TSP_EGRESS_POLICING_CFG,
 	IPA_STAT_TSP_DROP_BASE,
+	IPA_STATE_QMNGR_QUEUE_NONEMPTY,
 	IPA_RAM_INGRESS_POLICER_DB_BASE_ADDR,
 	IPA_RAM_EGRESS_SHAPING_PROD_DB_BASE_ADDR,
 	IPA_RAM_EGRESS_SHAPING_TC_DB_BASE_ADDR,

+ 128 - 0
drivers/platform/msm/ipa/ipa_v3/ipahal/ipahal_tsp.c

@@ -0,0 +1,128 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+
+
+#include "ipa_tsp.h"
+#include "ipahal_i.h"
+#include "ipahal_tsp_i.h"
+#include "ipahal_reg.h"
+#include "ipahal_tsp.h"
+
+
+void ipahal_tsp_fill_hw_ingr_tc(const struct ipa_ioc_tsp_ingress_class_params *input,
+	void *table, u8 index)
+{
+	/* The first index is 1 */
+	struct ipahal_tsp_ingress_class *hal_ingr_tc =
+		(struct ipahal_tsp_ingress_class *)table + index - 1;
+
+	hal_ingr_tc->max_rate = input->max_rate;
+	hal_ingr_tc->max_burst = input->max_burst;
+	hal_ingr_tc->max_bucket = input->max_burst;
+	hal_ingr_tc->last_rtc = 0x0;
+}
+
+void ipahal_tsp_fill_hw_egr_ep(const struct ipa_ioc_tsp_egress_prod_params *input,
+	void *table, u8 index)
+{
+	union ipahal_tsp_egress_prod *hal_egr_ep =
+		(union ipahal_tsp_egress_prod *)table + index;
+
+	if ((index & 0x1)) {
+		/* even index */
+		hal_egr_ep->even.max_rate = input->max_rate;
+		hal_egr_ep->even.max_freq = 65536 / input->max_rate;
+		hal_egr_ep->even.max_burst = input->max_burst;
+		hal_egr_ep->even.max_bucket = input->max_burst;
+		hal_egr_ep->even.last_rtc = 0x0;
+	} else {
+		/* odd index */
+		hal_egr_ep->odd.max_rate = input->max_rate;
+		hal_egr_ep->odd.max_freq = 65536 / input->max_rate;
+		hal_egr_ep->odd.max_burst = input->max_burst;
+		hal_egr_ep->odd.max_bucket = input->max_burst;
+		hal_egr_ep->odd.last_rtc = 0x0;
+	}
+}
+
+void ipahal_tsp_fill_hw_egr_tc(const struct ipa_ioc_tsp_egress_class_params *input,
+	void *table, u8 index)
+{
+	/* The first index is 1 */
+	union ipahal_tsp_egress_class *hal_egr_tc =
+		(union ipahal_tsp_egress_class *)table + index - 1;
+
+	if (!(index & 0x1)) {
+		/* even index */
+		hal_egr_tc->even.guaranteed_rate = input->guaranteed_rate;
+		hal_egr_tc->even.max_rate = input->max_rate;
+		hal_egr_tc->even.guaranteed_freq = 65536 / input->guaranteed_rate;
+		hal_egr_tc->even.max_freq = 65536 / input->max_rate;
+		hal_egr_tc->even.guaranteed_burst = input->guaranteed_burst;
+		hal_egr_tc->even.max_burst = input->max_burst;
+		hal_egr_tc->even.max_bucket = input->max_burst;
+		hal_egr_tc->even.last_rtc = 0x0;
+	} else {
+		/* odd index */
+		hal_egr_tc->odd.guaranteed_rate = input->guaranteed_rate;
+		hal_egr_tc->odd.max_rate = input->max_rate;
+		hal_egr_tc->odd.guaranteed_freq = 65536 / input->guaranteed_rate;
+		hal_egr_tc->odd.max_freq = 65536 / input->max_rate;
+		hal_egr_tc->odd.guaranteed_burst = input->guaranteed_burst;
+		hal_egr_tc->odd.max_burst = input->max_burst;
+		hal_egr_tc->odd.max_bucket = input->max_burst;
+		hal_egr_tc->odd.last_rtc = 0x0;
+	}
+}
+
+void ipahal_tsp_parse_hw_ingr_tc(const void *table, u8 index,
+	struct ipa_ioc_tsp_ingress_class_params *output)
+{
+	/* The first index is 1 */
+	struct ipahal_tsp_ingress_class *hal_ingr_tc =
+		(struct ipahal_tsp_ingress_class *)table + index - 1;
+
+	output->max_rate = hal_ingr_tc->max_rate;
+	output->max_burst = hal_ingr_tc->max_burst;
+}
+
+void ipahal_tsp_parse_hw_egr_tc(const void *table, u8 index,
+	struct ipa_ioc_tsp_egress_class_params *output)
+{
+	/* The first index is 1 */
+	union ipahal_tsp_egress_class *hal_egr_tc =
+		(union ipahal_tsp_egress_class *)table + index - 1;
+
+	if (!(index & 0x1)) {
+		/* even index */
+		output->guaranteed_rate = hal_egr_tc->even.guaranteed_rate;
+		output->max_rate = hal_egr_tc->even.max_rate;
+		output->guaranteed_burst = hal_egr_tc->even.guaranteed_burst;
+		output->max_burst = hal_egr_tc->even.max_burst;
+	} else {
+		/* odd index */
+		output->guaranteed_rate = hal_egr_tc->odd.guaranteed_rate;
+		output->max_rate = hal_egr_tc->odd.max_rate;
+		output->guaranteed_burst = hal_egr_tc->odd.guaranteed_burst;
+		output->max_burst = hal_egr_tc->odd.max_burst;
+	}
+}
+
+void ipahal_tsp_parse_hw_egr_ep(const void *table, u8 index,
+	struct ipa_ioc_tsp_egress_prod_params *output)
+{
+	union ipahal_tsp_egress_prod *hal_egr_ep =
+		(union ipahal_tsp_egress_prod *)table + index;
+
+	if ((index & 0x1)) {
+		/* even index */
+		output->max_rate = hal_egr_ep->even.max_rate;
+		output->max_burst = hal_egr_ep->even.max_burst;
+	} else {
+		/* odd index */
+		output->max_rate = hal_egr_ep->odd.max_rate;
+		output->max_burst = hal_egr_ep->odd.max_burst;
+	}
+}

+ 32 - 0
drivers/platform/msm/ipa/ipa_v3/ipahal/ipahal_tsp.h

@@ -0,0 +1,32 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+/*
+ * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+
+#ifndef _IPAHAL_TSP_H_
+#define _IPAHAL_TSP_H_
+
+#include <linux/msm_ipa.h>
+#include "ipa_tsp.h"
+
+#define IPA_TSP_IN_TC_NUM 16 // TBD: Get rid of it
+
+#define IPA_TSP_INGR_TC_SIZE 8
+#define IPA_TSP_EGR_EP_SIZE 12
+#define IPA_TSP_EGR_TC_SIZE 20
+
+
+void ipahal_tsp_fill_hw_ingr_tc(const struct ipa_ioc_tsp_ingress_class_params *input,
+	void *table, u8 index);
+void ipahal_tsp_fill_hw_egr_ep(const struct ipa_ioc_tsp_egress_prod_params *input,
+	void *table, u8 index);
+void ipahal_tsp_fill_hw_egr_tc(const struct ipa_ioc_tsp_egress_class_params *input,
+	void *table, u8 index);
+void ipahal_tsp_parse_hw_ingr_tc(const void *table, u8 index,
+	struct ipa_ioc_tsp_ingress_class_params *output);
+void ipahal_tsp_parse_hw_egr_ep(const void *table, u8 index,
+	struct ipa_ioc_tsp_egress_prod_params *output);
+void ipahal_tsp_parse_hw_egr_tc(const void *table, u8 index,
+	struct ipa_ioc_tsp_egress_class_params *output);
+
+#endif /* _IPAHAL_TSP_H_ */

+ 167 - 0
drivers/platform/msm/ipa/ipa_v3/ipahal/ipahal_tsp_i.h

@@ -0,0 +1,167 @@
+
+/* SPDX-License-Identifier: GPL-2.0-only */
+/*
+ * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+
+#ifndef _IPAHAL_TSP_I_H_
+#define _IPAHAL_TSP_I_H_
+
+/**
+* struct ipahal_tsp_ingress_class - IPA Ingress traffic-class
+*
+* @last_rtc: For HW use, initialize to 0x0000
+* @max_bucket: For HW use, initialize to same value as max_burst
+* @max_burst: Maximal-burst allowed in bytes
+* @max_rate: Maximal bandwidth rate for matching ingress traffic-class in 1.2MB/second units
+*/
+struct ipahal_tsp_ingress_class {
+	u16 last_rtc;
+	u16 max_bucket;
+	u16 max_burst;
+	u16 max_rate;
+};
+
+/**
+ * struct ipahal_tsp_egress_prod_even - IPA TSP-enabled producer (even index)
+ *
+ * @last_rtc: For HW use, initialize to 0x0000
+ * @max_bucket: For HW use, initialize to same value as max_burst
+ * @max_rate: Maximal bandwidth rate for producer in 1.2MB/second units
+ * @max_freq: In units of 0.833*usec/64KB. Claculated as:
+ * 		MAX-Freq = 65536/max_rate (always be rounded up)
+ * @max_burst: Maximal-burst allowed in bytes
+ * @reserved: Reserved
+ */
+struct ipahal_tsp_egress_prod_even {
+	u16 last_rtc;
+	u16 max_bucket;
+	u16 max_rate;
+	u16 max_freq;
+	u16 max_burst;
+	u16 reserved;
+};
+
+/**
+ * struct ipahal_tsp_egress_prod_odd - IPA TSP-enabled producer (odd index)
+ *
+ * @reserved: Reserved
+ * @max_burst: Maximal-burst allowed in bytes
+ * @last_rtc: For HW use, initialize to 0x0000
+ * @max_bucket: For HW use, initialize to same value as max_burst
+ * @max_rate: Maximal bandwidth rate for producer in 1.2MB/second units
+ * @max_freq: In units of 0.833*usec/64KB. Claculated as:
+ * 		MAX-Freq = 65536/max_rate (always be rounded up)
+ */
+struct ipahal_tsp_egress_prod_odd {
+	u16 reserved;
+	u16 max_burst;
+	u16 last_rtc;
+	u16 max_bucket;
+	u16 max_rate;
+	u16 max_freq;
+};
+
+/**
+ * union ipahal_tsp_egress_prod - IPA TSP-enabled producer (even or odd)
+ *
+ * @even: TSP-enabled producer (even index)
+ * @odd: TSP-enabled producer (odd index)
+ */
+union ipahal_tsp_egress_prod {
+	struct ipahal_tsp_egress_prod_even even;
+	struct ipahal_tsp_egress_prod_odd odd;
+};
+
+/**
+ * struct ipahal_tsp_egress_prod_pair - IPA TSP-enabled producer pair (even and odd)
+ *
+ * @even: TSP-enabled producer (even index)
+ * @odd: TSP-enabled producer (odd index)
+ */
+struct ipahal_tsp_egress_prod_pair {
+	struct ipahal_tsp_egress_prod_even even;
+	struct ipahal_tsp_egress_prod_odd odd;
+};
+
+/**
+ * struct ipahal_tsp_egress_class_even - IPA egress traffic-class (even index)
+ *
+ * @last_rtc: For HW use, initialize to 0x0000
+ * @reserved: Reserved
+ * @guaranteed_bucket: For HW use, initialize to same value as guaranteed_burst
+ * @max_bucket: For HW use, initialize to same value as max_burst
+ * @guaranteed_rate: Guaranteed bandwidth rate for traffic-class in 1.2MB/second units
+ * @max_rate: Maximal bandwidth rate for traffic-class in 1.2MB/second units
+ * @guaranteed_freq: In units of 0.833*usec/64KB, Calculated as:
+ * 			 guaranteed_freq = 65536/guaranteed_rate (always be rounded up)
+ * @max_freq: In units of 0.833*usec/64KB, Calculated as:
+ * 			 max_freq = 65536/max_rate (always be rounded up)
+ * @guaranteed_burst: Maximal-burst allowed for guaranteed bandwidth rate (in bytes)
+ * @max_burst: Maximal-burst allowed for maximal bandwidth rate (in bytes)
+ */
+struct ipahal_tsp_egress_class_even {
+	u16 last_rtc;
+	u16 reserved;
+	u16 guaranteed_bucket;
+	u16 max_bucket;
+	u16 guaranteed_rate;
+	u16 max_rate;
+	u16 guaranteed_freq;
+	u16 max_freq;
+	u16 guaranteed_burst;
+	u16 max_burst;
+};
+
+/**
+ * struct ipahal_tsp_egress_class_odd - IPA egress traffic-class (odd index)
+ *
+ * @guaranteed_burst: Maximal-burst allowed for guaranteed bandwidth rate (in bytes)
+ * @max_burst: Maximal-burst allowed for maximal bandwidth rate (in bytes)
+ * @last_rtc: For HW use, initialize to 0x0000
+ * @reserved: Reserved
+ * @guaranteed_bucket: For HW use, initialize to same value as guaranteed_burst
+ * @max_bucket: For HW use, initialize to same value as max_burst
+ * @guaranteed_rate: Guaranteed bandwidth rate for traffic-class in 1.2MB/second units
+ * @max_rate: Maximal bandwidth rate for traffic-class in 1.2MB/second units
+ * @guaranteed_freq: In units of 0.833*usec/64KB, Calculated as:
+ * 			 guaranteed_freq = 65536/guaranteed_rate (always be rounded up)
+ * @max_freq: In units of 0.833*usec/64KB, Calculated as:
+ * 			 max_freq = 65536/max_rate (always be rounded up)
+ */
+struct ipahal_tsp_egress_class_odd {
+	u16 guaranteed_burst;
+	u16 max_burst;
+	u16 last_rtc;
+	u16 reserved;
+	u16 guaranteed_bucket;
+	u16 max_bucket;
+	u16 guaranteed_rate;
+	u16 max_rate;
+	u16 guaranteed_freq;
+	u16 max_freq;
+};
+
+/**
+ * union ipahal_tsp_egress_prod - IPA egress traffic-class (even or odd)
+ *
+ * @even: egress traffic-class (even index)
+ * @odd: egress traffic-class (odd index)
+ */
+union ipahal_tsp_egress_class {
+	struct ipahal_tsp_egress_class_even even;
+	struct ipahal_tsp_egress_class_odd odd;
+};
+
+/**
+ * struct ipahal_tsp_egress_prod_pair - IPA egress traffic-class pair (even and odd)
+ *
+ * @even: egress traffic-class (even index)
+ * @odd: egress traffic-class (odd index)
+ */
+struct ipahal_tsp_egress_class_pair {
+	struct ipahal_tsp_egress_class_even even;
+	struct ipahal_tsp_egress_class_odd odd;
+};
+
+#endif /* _IPAHAL_TSP_I_H_ */