diff mbox series

[v3,13/13] coresight: Fix CTI module refcount leak by making it a helper device

Message ID 20230329115329.2747724-14-james.clark@arm.com (mailing list archive)
State New, archived
Headers show
Series coresight: Fix CTI module refcount leak by making it a helper device | expand

Commit Message

James Clark March 29, 2023, 11:53 a.m. UTC
The CTI module has some hard coded refcounting code that has a leak.
For example running perf and then trying to unload it fails:

  perf record -e cs_etm// -a -- ls
  rmmod coresight_cti

  rmmod: ERROR: Module coresight_cti is in use

The coresight core already handles references of devices in use, so by
making CTI a normal helper device, we get working refcounting for free.

Signed-off-by: James Clark <james.clark@arm.com>
---
 drivers/hwtracing/coresight/coresight-core.c  | 99 ++++++-------------
 .../hwtracing/coresight/coresight-cti-core.c  | 52 +++++-----
 .../hwtracing/coresight/coresight-cti-sysfs.c |  4 +-
 drivers/hwtracing/coresight/coresight-cti.h   |  4 +-
 drivers/hwtracing/coresight/coresight-priv.h  |  4 +-
 drivers/hwtracing/coresight/coresight-sysfs.c |  4 +
 include/linux/coresight.h                     | 30 +-----
 7 files changed, 70 insertions(+), 127 deletions(-)

Comments

Suzuki K Poulose April 4, 2023, 9:21 a.m. UTC | #1
On 29/03/2023 12:53, James Clark wrote:
> The CTI module has some hard coded refcounting code that has a leak.
> For example running perf and then trying to unload it fails:
> 
>    perf record -e cs_etm// -a -- ls
>    rmmod coresight_cti
> 
>    rmmod: ERROR: Module coresight_cti is in use
> 
> The coresight core already handles references of devices in use, so by
> making CTI a normal helper device, we get working refcounting for free.
> 
> Signed-off-by: James Clark <james.clark@arm.com>
> ---
>   drivers/hwtracing/coresight/coresight-core.c  | 99 ++++++-------------
>   .../hwtracing/coresight/coresight-cti-core.c  | 52 +++++-----
>   .../hwtracing/coresight/coresight-cti-sysfs.c |  4 +-
>   drivers/hwtracing/coresight/coresight-cti.h   |  4 +-
>   drivers/hwtracing/coresight/coresight-priv.h  |  4 +-
>   drivers/hwtracing/coresight/coresight-sysfs.c |  4 +
>   include/linux/coresight.h                     | 30 +-----
>   7 files changed, 70 insertions(+), 127 deletions(-)
> 
> diff --git a/drivers/hwtracing/coresight/coresight-core.c b/drivers/hwtracing/coresight/coresight-core.c
> index 65f5bd8516d8..458d91b4e23f 100644
> --- a/drivers/hwtracing/coresight/coresight-core.c
> +++ b/drivers/hwtracing/coresight/coresight-core.c
> @@ -254,60 +254,39 @@ void coresight_disclaim_device(struct coresight_device *csdev)
>   }
>   EXPORT_SYMBOL_GPL(coresight_disclaim_device);
>   
> -/* enable or disable an associated CTI device of the supplied CS device */
> -static int
> -coresight_control_assoc_ectdev(struct coresight_device *csdev, bool enable)
> -{
> -	int ect_ret = 0;
> -	struct coresight_device *ect_csdev = csdev->ect_dev;
> -	struct module *mod;
> -
> -	if (!ect_csdev)
> -		return 0;
> -	if ((!ect_ops(ect_csdev)->enable) || (!ect_ops(ect_csdev)->disable))
> -		return 0;
> -
> -	mod = ect_csdev->dev.parent->driver->owner;
> -	if (enable) {
> -		if (try_module_get(mod)) {
> -			ect_ret = ect_ops(ect_csdev)->enable(ect_csdev);
> -			if (ect_ret) {
> -				module_put(mod);
> -			} else {
> -				get_device(ect_csdev->dev.parent);
> -				csdev->ect_enabled = true;
> -			}
> -		} else
> -			ect_ret = -ENODEV;
> -	} else {
> -		if (csdev->ect_enabled) {
> -			ect_ret = ect_ops(ect_csdev)->disable(ect_csdev);
> -			put_device(ect_csdev->dev.parent);
> -			module_put(mod);
> -			csdev->ect_enabled = false;
> -		}
> -	}
> -
> -	/* output warning if ECT enable is preventing trace operation */
> -	if (ect_ret)
> -		dev_info(&csdev->dev, "Associated ECT device (%s) %s failed\n",
> -			 dev_name(&ect_csdev->dev),
> -			 enable ? "enable" : "disable");
> -	return ect_ret;
> -}
> -
>   /*
> - * Set the associated ect / cti device while holding the coresight_mutex
> + * Add a helper as an output device while holding the coresight_mutex
>    * to avoid a race with coresight_enable that may try to use this value.
>    */
> -void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
> -				      struct coresight_device *ect_csdev)
> +void coresight_add_helper_mutex(struct coresight_device *csdev,
> +				struct coresight_device *helper)

minor nit: It may be a good idea to rename this, in line with the
kernel naming convention :

	coresight_add_helper_unlocked()

Or if this is the only variant, it is OK to leave it as :
	coresight_add_helper()
with a big fat comment in the function description to indicate
that it takes the mutex and may be even add a :

might_sleep() and lockdep_assert_not_held(&coresight_mutex);

in the function.

>   {
> +	int i;
> +	struct coresight_connection conn = {};
> +
>   	mutex_lock(&coresight_mutex);
> -	csdev->ect_dev = ect_csdev;
> +	conn.dest_fwnode = fwnode_handle_get(dev_fwnode(&helper->dev));
> +	conn.dest_dev = helper;
> +	conn.dest_port = conn.src_port = -1;
> +	conn.src_dev = csdev;
> +
> +	/*
> +	 * Check for duplicates because this is called every time a helper
> +	 * device is re-loaded. Existing connections will get re-linked
> +	 * automatically.
> +	 */

Thanks for adding this comment here. It does look like the already added
output connection to the "origin" device would automatically resolve the
connection and add in the "in-connection" to the CTI device.

> +	for (i = 0; i < csdev->pdata->nr_outconns; ++i)
> +		if (csdev->pdata->out_conns[i]->dest_fwnode == conn.dest_fwnode)
> +			goto unlock;
> +
> +	coresight_add_out_conn(csdev->dev.parent, csdev->pdata, &conn);

This makes me wonder if we should return the new connection in 
coresight_add_out_conn() in case of success, rather than assuming the 
last one (which is always the case though.).

i.e.,
	new_conn = coresight_add_out_conn(...)
	if (new_conn)
		coresight_add_in_conn(new_conn);

> +	coresight_add_in_conn(
> +		csdev->pdata->out_conns[csdev->pdata->nr_outconns - 1]);
> +
> +unlock:
>   	mutex_unlock(&coresight_mutex);
>   }
> -EXPORT_SYMBOL_GPL(coresight_set_assoc_ectdev_mutex);
> +EXPORT_SYMBOL_GPL(coresight_add_helper_mutex);
>   
>   static int coresight_enable_sink(struct coresight_device *csdev,
>   				 enum cs_mode mode, void *data)
> @@ -321,12 +300,8 @@ static int coresight_enable_sink(struct coresight_device *csdev,
>   	if (!sink_ops(csdev)->enable)
>   		return -EINVAL;
>   
> -	ret = coresight_control_assoc_ectdev(csdev, true);
> -	if (ret)
> -		return ret;
>   	ret = sink_ops(csdev)->enable(csdev, mode, data);
>   	if (ret) {
> -		coresight_control_assoc_ectdev(csdev, false);
>   		return ret;
>   	}
>   	csdev->enable = true;
> @@ -344,7 +319,6 @@ static void coresight_disable_sink(struct coresight_device *csdev)
>   	ret = sink_ops(csdev)->disable(csdev);
>   	if (ret)
>   		return;
> -	coresight_control_assoc_ectdev(csdev, false);
>   	csdev->enable = false;
>   }
>   
> @@ -369,17 +343,11 @@ static int coresight_enable_link(struct coresight_device *csdev,
>   		return PTR_ERR(outconn);
>   
>   	if (link_ops(csdev)->enable) {
> -		ret = coresight_control_assoc_ectdev(csdev, true);
> -		if (!ret) {
> -			ret = link_ops(csdev)->enable(csdev, inconn, outconn);
> -			if (ret)
> -				coresight_control_assoc_ectdev(csdev, false);
> -		}
> +		ret = link_ops(csdev)->enable(csdev, inconn, outconn);
> +		if (!ret)
> +			csdev->enable = true;
>   	}
>   
> -	if (!ret)
> -		csdev->enable = true;
> -
>   	return ret;
>   }
>   
> @@ -400,7 +368,6 @@ static void coresight_disable_link(struct coresight_device *csdev,
>   
>   	if (link_ops(csdev)->disable) {
>   		link_ops(csdev)->disable(csdev, inconn, outconn);
> -		coresight_control_assoc_ectdev(csdev, false);
>   	}
>   
>   	if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) {
> @@ -428,14 +395,9 @@ int coresight_enable_source(struct coresight_device *csdev, void *data,
>   
>   	if (!csdev->enable) {
>   		if (source_ops(csdev)->enable) {
> -			ret = coresight_control_assoc_ectdev(csdev, true);
> -			if (ret)
> -				return ret;
>   			ret = source_ops(csdev)->enable(csdev, data, mode);
> -			if (ret) {
> -				coresight_control_assoc_ectdev(csdev, false);
> +			if (ret)
>   				return ret;
> -			}
>   		}
>   		csdev->enable = true;
>   	}
> @@ -499,7 +461,6 @@ bool coresight_disable_source(struct coresight_device *csdev, void *data)
>   	if (atomic_dec_return(&csdev->refcnt) == 0) {
>   		if (source_ops(csdev)->disable)
>   			source_ops(csdev)->disable(csdev, data);
> -		coresight_control_assoc_ectdev(csdev, false);
>   		coresight_disable_helpers(csdev);
>   		csdev->enable = false;
>   	}
> diff --git a/drivers/hwtracing/coresight/coresight-cti-core.c b/drivers/hwtracing/coresight/coresight-cti-core.c
> index 277c890a1f1f..db7a2212ec18 100644
> --- a/drivers/hwtracing/coresight/coresight-cti-core.c
> +++ b/drivers/hwtracing/coresight/coresight-cti-core.c
> @@ -555,7 +555,10 @@ static void cti_add_assoc_to_csdev(struct coresight_device *csdev)
>   	mutex_lock(&ect_mutex);
>   
>   	/* exit if current is an ECT device.*/
> -	if ((csdev->type == CORESIGHT_DEV_TYPE_ECT) || list_empty(&ect_net))
> +	if ((csdev->type == CORESIGHT_DEV_TYPE_HELPER &&
> +	     csdev->subtype.helper_subtype ==
> +		     CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI) ||
> +	    list_empty(&ect_net))
>   		goto cti_add_done;
>   
>   	/* if we didn't find the csdev previously we used the fwnode name */
> @@ -571,8 +574,7 @@ static void cti_add_assoc_to_csdev(struct coresight_device *csdev)
>   			 * if we found a matching csdev then update the ECT
>   			 * association pointer for the device with this CTI.
>   			 */
> -			coresight_set_assoc_ectdev_mutex(csdev,
> -							 ect_item->csdev);
> +			coresight_add_helper_mutex(csdev, ect_item->csdev);
>   			break;
>   		}
>   	}
> @@ -582,26 +584,30 @@ static void cti_add_assoc_to_csdev(struct coresight_device *csdev)
>   
>   /*
>    * Removing the associated devices is easier.
> - * A CTI will not have a value for csdev->ect_dev.
>    */
>   static void cti_remove_assoc_from_csdev(struct coresight_device *csdev)
>   {
>   	struct cti_drvdata *ctidrv;
>   	struct cti_trig_con *tc;
> +	union coresight_dev_subtype cti_subtype = {
> +		.helper_subtype = CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
> +	};
> +	struct coresight_device *cti_csdev = coresight_find_output_type(
> +		csdev->pdata, CORESIGHT_DEV_TYPE_HELPER, cti_subtype);

minor nit: Please could we split the initialisation ? Or at least move
this after the next variable declaration below ?


Rest looks fine to me.

Suzuki

>   	struct cti_device *ctidev;
>   
> +	if (!cti_csdev)
> +		return;
> +
>   	mutex_lock(&ect_mutex);
> -	if (csdev->ect_dev) {
> -		ctidrv = csdev_to_cti_drvdata(csdev->ect_dev);
> -		ctidev = &ctidrv->ctidev;
> -		list_for_each_entry(tc, &ctidev->trig_cons, node) {
> -			if (tc->con_dev == csdev) {
> -				cti_remove_sysfs_link(ctidrv, tc);
> -				tc->con_dev = NULL;
> -				break;
> -			}
> +	ctidrv = csdev_to_cti_drvdata(cti_csdev);
> +	ctidev = &ctidrv->ctidev;
> +	list_for_each_entry(tc, &ctidev->trig_cons, node) {
> +		if (tc->con_dev == csdev) {
> +			cti_remove_sysfs_link(ctidrv, tc);
> +			tc->con_dev = NULL;
> +			break;
>   		}
> -		csdev->ect_dev = NULL;
>   	}
>   	mutex_unlock(&ect_mutex);
>   }
> @@ -630,8 +636,8 @@ static void cti_update_conn_xrefs(struct cti_drvdata *drvdata)
>   			/* if we can set the sysfs link */
>   			if (cti_add_sysfs_link(drvdata, tc))
>   				/* set the CTI/csdev association */
> -				coresight_set_assoc_ectdev_mutex(tc->con_dev,
> -							 drvdata->csdev);
> +				coresight_add_helper_mutex(tc->con_dev,
> +							   drvdata->csdev);
>   			else
>   				/* otherwise remove reference from CTI */
>   				tc->con_dev = NULL;
> @@ -646,8 +652,6 @@ static void cti_remove_conn_xrefs(struct cti_drvdata *drvdata)
>   
>   	list_for_each_entry(tc, &ctidev->trig_cons, node) {
>   		if (tc->con_dev) {
> -			coresight_set_assoc_ectdev_mutex(tc->con_dev,
> -							 NULL);
>   			cti_remove_sysfs_link(drvdata, tc);
>   			tc->con_dev = NULL;
>   		}
> @@ -795,27 +799,27 @@ static void cti_pm_release(struct cti_drvdata *drvdata)
>   }
>   
>   /** cti ect operations **/
> -int cti_enable(struct coresight_device *csdev)
> +int cti_enable(struct coresight_device *csdev, enum cs_mode mode, void *data)
>   {
>   	struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
>   
>   	return cti_enable_hw(drvdata);
>   }
>   
> -int cti_disable(struct coresight_device *csdev)
> +int cti_disable(struct coresight_device *csdev, void *data)
>   {
>   	struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
>   
>   	return cti_disable_hw(drvdata);
>   }
>   
> -static const struct coresight_ops_ect cti_ops_ect = {
> +static const struct coresight_ops_helper cti_ops_ect = {
>   	.enable = cti_enable,
>   	.disable = cti_disable,
>   };
>   
>   static const struct coresight_ops cti_ops = {
> -	.ect_ops = &cti_ops_ect,
> +	.helper_ops = &cti_ops_ect,
>   };
>   
>   /*
> @@ -922,8 +926,8 @@ static int cti_probe(struct amba_device *adev, const struct amba_id *id)
>   
>   	/* set up coresight component description */
>   	cti_desc.pdata = pdata;
> -	cti_desc.type = CORESIGHT_DEV_TYPE_ECT;
> -	cti_desc.subtype.ect_subtype = CORESIGHT_DEV_SUBTYPE_ECT_CTI;
> +	cti_desc.type = CORESIGHT_DEV_TYPE_HELPER;
> +	cti_desc.subtype.helper_subtype = CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI;
>   	cti_desc.ops = &cti_ops;
>   	cti_desc.groups = drvdata->ctidev.con_groups;
>   	cti_desc.dev = dev;
> diff --git a/drivers/hwtracing/coresight/coresight-cti-sysfs.c b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
> index e528cff9d4e2..d25dd2737b49 100644
> --- a/drivers/hwtracing/coresight/coresight-cti-sysfs.c
> +++ b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
> @@ -112,11 +112,11 @@ static ssize_t enable_store(struct device *dev,
>   		ret = pm_runtime_resume_and_get(dev->parent);
>   		if (ret)
>   			return ret;
> -		ret = cti_enable(drvdata->csdev);
> +		ret = cti_enable(drvdata->csdev, CS_MODE_SYSFS, NULL);
>   		if (ret)
>   			pm_runtime_put(dev->parent);
>   	} else {
> -		ret = cti_disable(drvdata->csdev);
> +		ret = cti_disable(drvdata->csdev, NULL);
>   		if (!ret)
>   			pm_runtime_put(dev->parent);
>   	}
> diff --git a/drivers/hwtracing/coresight/coresight-cti.h b/drivers/hwtracing/coresight/coresight-cti.h
> index 8b106b13a244..cb9ee616d01f 100644
> --- a/drivers/hwtracing/coresight/coresight-cti.h
> +++ b/drivers/hwtracing/coresight/coresight-cti.h
> @@ -215,8 +215,8 @@ int cti_add_connection_entry(struct device *dev, struct cti_drvdata *drvdata,
>   			     const char *assoc_dev_name);
>   struct cti_trig_con *cti_allocate_trig_con(struct device *dev, int in_sigs,
>   					   int out_sigs);
> -int cti_enable(struct coresight_device *csdev);
> -int cti_disable(struct coresight_device *csdev);
> +int cti_enable(struct coresight_device *csdev, enum cs_mode mode, void *data);
> +int cti_disable(struct coresight_device *csdev, void *data);
>   void cti_write_all_hw_regs(struct cti_drvdata *drvdata);
>   void cti_write_intack(struct device *dev, u32 ackval);
>   void cti_write_single_reg(struct cti_drvdata *drvdata, int offset, u32 value);
> diff --git a/drivers/hwtracing/coresight/coresight-priv.h b/drivers/hwtracing/coresight/coresight-priv.h
> index a843f9d5c737..fff565d1cb42 100644
> --- a/drivers/hwtracing/coresight/coresight-priv.h
> +++ b/drivers/hwtracing/coresight/coresight-priv.h
> @@ -211,8 +211,8 @@ void coresight_release_platform_data(struct coresight_device *csdev,
>   				     struct coresight_platform_data *pdata);
>   struct coresight_device *
>   coresight_find_csdev_by_fwnode(struct fwnode_handle *r_fwnode);
> -void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
> -				      struct coresight_device *ect_csdev);
> +void coresight_add_helper_mutex(struct coresight_device *csdev,
> +				struct coresight_device *helper);
>   
>   void coresight_set_percpu_sink(int cpu, struct coresight_device *csdev);
>   struct coresight_device *coresight_get_percpu_sink(int cpu);
> diff --git a/drivers/hwtracing/coresight/coresight-sysfs.c b/drivers/hwtracing/coresight/coresight-sysfs.c
> index 464ba5e1343b..dd78e9fcfc4d 100644
> --- a/drivers/hwtracing/coresight/coresight-sysfs.c
> +++ b/drivers/hwtracing/coresight/coresight-sysfs.c
> @@ -148,6 +148,10 @@ int coresight_make_links(struct coresight_device *orig,
>   	char *outs = NULL, *ins = NULL;
>   	struct coresight_sysfs_link *link = NULL;
>   
> +	/* Helper devices aren't shown in sysfs */
> +	if (conn->dest_port == -1 && conn->src_port == -1)
> +		return 0;
> +
>   	do {
>   		outs = devm_kasprintf(&orig->dev, GFP_KERNEL,
>   				      "out:%d", conn->src_port);
> diff --git a/include/linux/coresight.h b/include/linux/coresight.h
> index d2739a0286f1..ed37552761e4 100644
> --- a/include/linux/coresight.h
> +++ b/include/linux/coresight.h
> @@ -40,8 +40,7 @@ enum coresight_dev_type {
>   	CORESIGHT_DEV_TYPE_LINK,
>   	CORESIGHT_DEV_TYPE_LINKSINK,
>   	CORESIGHT_DEV_TYPE_SOURCE,
> -	CORESIGHT_DEV_TYPE_HELPER,
> -	CORESIGHT_DEV_TYPE_ECT,
> +	CORESIGHT_DEV_TYPE_HELPER
>   };
>   
>   enum coresight_dev_subtype_sink {
> @@ -66,12 +65,7 @@ enum coresight_dev_subtype_source {
>   
>   enum coresight_dev_subtype_helper {
>   	CORESIGHT_DEV_SUBTYPE_HELPER_CATU,
> -};
> -
> -/* Embedded Cross Trigger (ECT) sub-types */
> -enum coresight_dev_subtype_ect {
> -	CORESIGHT_DEV_SUBTYPE_ECT_NONE,
> -	CORESIGHT_DEV_SUBTYPE_ECT_CTI,
> +	CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
>   };
>   
>   /**
> @@ -84,8 +78,6 @@ enum coresight_dev_subtype_ect {
>    *			by @coresight_dev_subtype_source.
>    * @helper_subtype:	type of helper this component is, as defined
>    *			by @coresight_dev_subtype_helper.
> - * @ect_subtype:        type of cross trigger this component is, as
> - *			defined by @coresight_dev_subtype_ect
>    */
>   union coresight_dev_subtype {
>   	/* We have some devices which acts as LINK and SINK */
> @@ -95,7 +87,6 @@ union coresight_dev_subtype {
>   	};
>   	enum coresight_dev_subtype_source source_subtype;
>   	enum coresight_dev_subtype_helper helper_subtype;
> -	enum coresight_dev_subtype_ect ect_subtype;
>   };
>   
>   /**
> @@ -237,8 +228,6 @@ struct coresight_sysfs_link {
>    *		from source to that sink.
>    * @ea:		Device attribute for sink representation under PMU directory.
>    * @def_sink:	cached reference to default sink found for this device.
> - * @ect_dev:	Associated cross trigger device. Not part of the trace data
> - *		path or connections.
>    * @nr_links:   number of sysfs links created to other components from this
>    *		device. These will appear in the "connections" group.
>    * @has_conns_grp: Have added a "connections" group for sysfs links.
> @@ -261,12 +250,9 @@ struct coresight_device {
>   	bool activated;	/* true only if a sink is part of a path */
>   	struct dev_ext_attribute *ea;
>   	struct coresight_device *def_sink;
> -	/* cross trigger handling */
> -	struct coresight_device *ect_dev;
>   	/* sysfs links between components */
>   	int nr_links;
>   	bool has_conns_grp;
> -	bool ect_enabled; /* true only if associated ect device is enabled */
>   	/* system configuration and feature lists */
>   	struct list_head feature_csdev_list;
>   	struct list_head config_csdev_list;
> @@ -378,23 +364,11 @@ struct coresight_ops_helper {
>   	int (*disable)(struct coresight_device *csdev, void *data);
>   };
>   
> -/**
> - * struct coresight_ops_ect - Ops for an embedded cross trigger device
> - *
> - * @enable	: Enable the device
> - * @disable	: Disable the device
> - */
> -struct coresight_ops_ect {
> -	int (*enable)(struct coresight_device *csdev);
> -	int (*disable)(struct coresight_device *csdev);
> -};
> -
>   struct coresight_ops {
>   	const struct coresight_ops_sink *sink_ops;
>   	const struct coresight_ops_link *link_ops;
>   	const struct coresight_ops_source *source_ops;
>   	const struct coresight_ops_helper *helper_ops;
> -	const struct coresight_ops_ect *ect_ops;
>   };
>   
>   #if IS_ENABLED(CONFIG_CORESIGHT)
James Clark April 4, 2023, 12:55 p.m. UTC | #2
On 04/04/2023 10:21, Suzuki K Poulose wrote:
> On 29/03/2023 12:53, James Clark wrote:
>> The CTI module has some hard coded refcounting code that has a leak.
>> For example running perf and then trying to unload it fails:
>>
>>    perf record -e cs_etm// -a -- ls
>>    rmmod coresight_cti
>>
>>    rmmod: ERROR: Module coresight_cti is in use
>>
>> The coresight core already handles references of devices in use, so by
>> making CTI a normal helper device, we get working refcounting for free.
>>
>> Signed-off-by: James Clark <james.clark@arm.com>
>> ---
>>   drivers/hwtracing/coresight/coresight-core.c  | 99 ++++++-------------
>>   .../hwtracing/coresight/coresight-cti-core.c  | 52 +++++-----
>>   .../hwtracing/coresight/coresight-cti-sysfs.c |  4 +-
>>   drivers/hwtracing/coresight/coresight-cti.h   |  4 +-
>>   drivers/hwtracing/coresight/coresight-priv.h  |  4 +-
>>   drivers/hwtracing/coresight/coresight-sysfs.c |  4 +
>>   include/linux/coresight.h                     | 30 +-----
>>   7 files changed, 70 insertions(+), 127 deletions(-)
>>
>> diff --git a/drivers/hwtracing/coresight/coresight-core.c
>> b/drivers/hwtracing/coresight/coresight-core.c
>> index 65f5bd8516d8..458d91b4e23f 100644
>> --- a/drivers/hwtracing/coresight/coresight-core.c
>> +++ b/drivers/hwtracing/coresight/coresight-core.c
>> @@ -254,60 +254,39 @@ void coresight_disclaim_device(struct
>> coresight_device *csdev)
>>   }
>>   EXPORT_SYMBOL_GPL(coresight_disclaim_device);
>>   -/* enable or disable an associated CTI device of the supplied CS
>> device */
>> -static int
>> -coresight_control_assoc_ectdev(struct coresight_device *csdev, bool
>> enable)
>> -{
>> -    int ect_ret = 0;
>> -    struct coresight_device *ect_csdev = csdev->ect_dev;
>> -    struct module *mod;
>> -
>> -    if (!ect_csdev)
>> -        return 0;
>> -    if ((!ect_ops(ect_csdev)->enable) || (!ect_ops(ect_csdev)->disable))
>> -        return 0;
>> -
>> -    mod = ect_csdev->dev.parent->driver->owner;
>> -    if (enable) {
>> -        if (try_module_get(mod)) {
>> -            ect_ret = ect_ops(ect_csdev)->enable(ect_csdev);
>> -            if (ect_ret) {
>> -                module_put(mod);
>> -            } else {
>> -                get_device(ect_csdev->dev.parent);
>> -                csdev->ect_enabled = true;
>> -            }
>> -        } else
>> -            ect_ret = -ENODEV;
>> -    } else {
>> -        if (csdev->ect_enabled) {
>> -            ect_ret = ect_ops(ect_csdev)->disable(ect_csdev);
>> -            put_device(ect_csdev->dev.parent);
>> -            module_put(mod);
>> -            csdev->ect_enabled = false;
>> -        }
>> -    }
>> -
>> -    /* output warning if ECT enable is preventing trace operation */
>> -    if (ect_ret)
>> -        dev_info(&csdev->dev, "Associated ECT device (%s) %s failed\n",
>> -             dev_name(&ect_csdev->dev),
>> -             enable ? "enable" : "disable");
>> -    return ect_ret;
>> -}
>> -
>>   /*
>> - * Set the associated ect / cti device while holding the coresight_mutex
>> + * Add a helper as an output device while holding the coresight_mutex
>>    * to avoid a race with coresight_enable that may try to use this
>> value.
>>    */
>> -void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
>> -                      struct coresight_device *ect_csdev)
>> +void coresight_add_helper_mutex(struct coresight_device *csdev,
>> +                struct coresight_device *helper)
> 
> minor nit: It may be a good idea to rename this, in line with the
> kernel naming convention :
> 
>     coresight_add_helper_unlocked()
> 
> Or if this is the only variant, it is OK to leave it as :
>     coresight_add_helper()
> with a big fat comment in the function description to indicate
> that it takes the mutex and may be even add a :
> 
There is already a bit of a comment in the description but I can expand
on it more.

> might_sleep() and lockdep_assert_not_held(&coresight_mutex);
> 
> in the function.
> 

I'm not sure if lockdep_assert_not_held() would be right because
sometimes it could be held if another device is being created at the
same time? Or something like a session is started at the same time a CTI
device is added.

>>   {
>> +    int i;
>> +    struct coresight_connection conn = {};
>> +
>>       mutex_lock(&coresight_mutex);
>> -    csdev->ect_dev = ect_csdev;
>> +    conn.dest_fwnode = fwnode_handle_get(dev_fwnode(&helper->dev));
>> +    conn.dest_dev = helper;
>> +    conn.dest_port = conn.src_port = -1;
>> +    conn.src_dev = csdev;
>> +
>> +    /*
>> +     * Check for duplicates because this is called every time a helper
>> +     * device is re-loaded. Existing connections will get re-linked
>> +     * automatically.
>> +     */
> 
> Thanks for adding this comment here. It does look like the already added
> output connection to the "origin" device would automatically resolve the
> connection and add in the "in-connection" to the CTI device.
> 
>> +    for (i = 0; i < csdev->pdata->nr_outconns; ++i)
>> +        if (csdev->pdata->out_conns[i]->dest_fwnode == conn.dest_fwnode)
>> +            goto unlock;
>> +
>> +    coresight_add_out_conn(csdev->dev.parent, csdev->pdata, &conn);
> 
> This makes me wonder if we should return the new connection in
> coresight_add_out_conn() in case of success, rather than assuming the
> last one (which is always the case though.).
> 
> i.e.,
>     new_conn = coresight_add_out_conn(...)
>     if (new_conn)
>         coresight_add_in_conn(new_conn);
> 

Yeah I thought about doing that too, will change it.

>> +    coresight_add_in_conn(
>> +        csdev->pdata->out_conns[csdev->pdata->nr_outconns - 1]);
>> +
>> +unlock:
>>       mutex_unlock(&coresight_mutex);
>>   }
>> -EXPORT_SYMBOL_GPL(coresight_set_assoc_ectdev_mutex);
>> +EXPORT_SYMBOL_GPL(coresight_add_helper_mutex);
>>     static int coresight_enable_sink(struct coresight_device *csdev,
>>                    enum cs_mode mode, void *data)
>> @@ -321,12 +300,8 @@ static int coresight_enable_sink(struct
>> coresight_device *csdev,
>>       if (!sink_ops(csdev)->enable)
>>           return -EINVAL;
>>   -    ret = coresight_control_assoc_ectdev(csdev, true);
>> -    if (ret)
>> -        return ret;
>>       ret = sink_ops(csdev)->enable(csdev, mode, data);
>>       if (ret) {
>> -        coresight_control_assoc_ectdev(csdev, false);
>>           return ret;
>>       }
>>       csdev->enable = true;
>> @@ -344,7 +319,6 @@ static void coresight_disable_sink(struct
>> coresight_device *csdev)
>>       ret = sink_ops(csdev)->disable(csdev);
>>       if (ret)
>>           return;
>> -    coresight_control_assoc_ectdev(csdev, false);
>>       csdev->enable = false;
>>   }
>>   @@ -369,17 +343,11 @@ static int coresight_enable_link(struct
>> coresight_device *csdev,
>>           return PTR_ERR(outconn);
>>         if (link_ops(csdev)->enable) {
>> -        ret = coresight_control_assoc_ectdev(csdev, true);
>> -        if (!ret) {
>> -            ret = link_ops(csdev)->enable(csdev, inconn, outconn);
>> -            if (ret)
>> -                coresight_control_assoc_ectdev(csdev, false);
>> -        }
>> +        ret = link_ops(csdev)->enable(csdev, inconn, outconn);
>> +        if (!ret)
>> +            csdev->enable = true;
>>       }
>>   -    if (!ret)
>> -        csdev->enable = true;
>> -
>>       return ret;
>>   }
>>   @@ -400,7 +368,6 @@ static void coresight_disable_link(struct
>> coresight_device *csdev,
>>         if (link_ops(csdev)->disable) {
>>           link_ops(csdev)->disable(csdev, inconn, outconn);
>> -        coresight_control_assoc_ectdev(csdev, false);
>>       }
>>         if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) {
>> @@ -428,14 +395,9 @@ int coresight_enable_source(struct
>> coresight_device *csdev, void *data,
>>         if (!csdev->enable) {
>>           if (source_ops(csdev)->enable) {
>> -            ret = coresight_control_assoc_ectdev(csdev, true);
>> -            if (ret)
>> -                return ret;
>>               ret = source_ops(csdev)->enable(csdev, data, mode);
>> -            if (ret) {
>> -                coresight_control_assoc_ectdev(csdev, false);
>> +            if (ret)
>>                   return ret;
>> -            }
>>           }
>>           csdev->enable = true;
>>       }
>> @@ -499,7 +461,6 @@ bool coresight_disable_source(struct
>> coresight_device *csdev, void *data)
>>       if (atomic_dec_return(&csdev->refcnt) == 0) {
>>           if (source_ops(csdev)->disable)
>>               source_ops(csdev)->disable(csdev, data);
>> -        coresight_control_assoc_ectdev(csdev, false);
>>           coresight_disable_helpers(csdev);
>>           csdev->enable = false;
>>       }
>> diff --git a/drivers/hwtracing/coresight/coresight-cti-core.c
>> b/drivers/hwtracing/coresight/coresight-cti-core.c
>> index 277c890a1f1f..db7a2212ec18 100644
>> --- a/drivers/hwtracing/coresight/coresight-cti-core.c
>> +++ b/drivers/hwtracing/coresight/coresight-cti-core.c
>> @@ -555,7 +555,10 @@ static void cti_add_assoc_to_csdev(struct
>> coresight_device *csdev)
>>       mutex_lock(&ect_mutex);
>>         /* exit if current is an ECT device.*/
>> -    if ((csdev->type == CORESIGHT_DEV_TYPE_ECT) || list_empty(&ect_net))
>> +    if ((csdev->type == CORESIGHT_DEV_TYPE_HELPER &&
>> +         csdev->subtype.helper_subtype ==
>> +             CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI) ||
>> +        list_empty(&ect_net))
>>           goto cti_add_done;
>>         /* if we didn't find the csdev previously we used the fwnode
>> name */
>> @@ -571,8 +574,7 @@ static void cti_add_assoc_to_csdev(struct
>> coresight_device *csdev)
>>                * if we found a matching csdev then update the ECT
>>                * association pointer for the device with this CTI.
>>                */
>> -            coresight_set_assoc_ectdev_mutex(csdev,
>> -                             ect_item->csdev);
>> +            coresight_add_helper_mutex(csdev, ect_item->csdev);
>>               break;
>>           }
>>       }
>> @@ -582,26 +584,30 @@ static void cti_add_assoc_to_csdev(struct
>> coresight_device *csdev)
>>     /*
>>    * Removing the associated devices is easier.
>> - * A CTI will not have a value for csdev->ect_dev.
>>    */
>>   static void cti_remove_assoc_from_csdev(struct coresight_device *csdev)
>>   {
>>       struct cti_drvdata *ctidrv;
>>       struct cti_trig_con *tc;
>> +    union coresight_dev_subtype cti_subtype = {
>> +        .helper_subtype = CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
>> +    };
>> +    struct coresight_device *cti_csdev = coresight_find_output_type(
>> +        csdev->pdata, CORESIGHT_DEV_TYPE_HELPER, cti_subtype);
> 
> minor nit: Please could we split the initialisation ? Or at least move
> this after the next variable declaration below ?
> 
> 
> Rest looks fine to me.
> 
> Suzuki
> 
>>       struct cti_device *ctidev;
>>   +    if (!cti_csdev)
>> +        return;
>> +
>>       mutex_lock(&ect_mutex);
>> -    if (csdev->ect_dev) {
>> -        ctidrv = csdev_to_cti_drvdata(csdev->ect_dev);
>> -        ctidev = &ctidrv->ctidev;
>> -        list_for_each_entry(tc, &ctidev->trig_cons, node) {
>> -            if (tc->con_dev == csdev) {
>> -                cti_remove_sysfs_link(ctidrv, tc);
>> -                tc->con_dev = NULL;
>> -                break;
>> -            }
>> +    ctidrv = csdev_to_cti_drvdata(cti_csdev);
>> +    ctidev = &ctidrv->ctidev;
>> +    list_for_each_entry(tc, &ctidev->trig_cons, node) {
>> +        if (tc->con_dev == csdev) {
>> +            cti_remove_sysfs_link(ctidrv, tc);
>> +            tc->con_dev = NULL;
>> +            break;
>>           }
>> -        csdev->ect_dev = NULL;
>>       }
>>       mutex_unlock(&ect_mutex);
>>   }
>> @@ -630,8 +636,8 @@ static void cti_update_conn_xrefs(struct
>> cti_drvdata *drvdata)
>>               /* if we can set the sysfs link */
>>               if (cti_add_sysfs_link(drvdata, tc))
>>                   /* set the CTI/csdev association */
>> -                coresight_set_assoc_ectdev_mutex(tc->con_dev,
>> -                             drvdata->csdev);
>> +                coresight_add_helper_mutex(tc->con_dev,
>> +                               drvdata->csdev);
>>               else
>>                   /* otherwise remove reference from CTI */
>>                   tc->con_dev = NULL;
>> @@ -646,8 +652,6 @@ static void cti_remove_conn_xrefs(struct
>> cti_drvdata *drvdata)
>>         list_for_each_entry(tc, &ctidev->trig_cons, node) {
>>           if (tc->con_dev) {
>> -            coresight_set_assoc_ectdev_mutex(tc->con_dev,
>> -                             NULL);
>>               cti_remove_sysfs_link(drvdata, tc);
>>               tc->con_dev = NULL;
>>           }
>> @@ -795,27 +799,27 @@ static void cti_pm_release(struct cti_drvdata
>> *drvdata)
>>   }
>>     /** cti ect operations **/
>> -int cti_enable(struct coresight_device *csdev)
>> +int cti_enable(struct coresight_device *csdev, enum cs_mode mode,
>> void *data)
>>   {
>>       struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
>>         return cti_enable_hw(drvdata);
>>   }
>>   -int cti_disable(struct coresight_device *csdev)
>> +int cti_disable(struct coresight_device *csdev, void *data)
>>   {
>>       struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
>>         return cti_disable_hw(drvdata);
>>   }
>>   -static const struct coresight_ops_ect cti_ops_ect = {
>> +static const struct coresight_ops_helper cti_ops_ect = {
>>       .enable = cti_enable,
>>       .disable = cti_disable,
>>   };
>>     static const struct coresight_ops cti_ops = {
>> -    .ect_ops = &cti_ops_ect,
>> +    .helper_ops = &cti_ops_ect,
>>   };
>>     /*
>> @@ -922,8 +926,8 @@ static int cti_probe(struct amba_device *adev,
>> const struct amba_id *id)
>>         /* set up coresight component description */
>>       cti_desc.pdata = pdata;
>> -    cti_desc.type = CORESIGHT_DEV_TYPE_ECT;
>> -    cti_desc.subtype.ect_subtype = CORESIGHT_DEV_SUBTYPE_ECT_CTI;
>> +    cti_desc.type = CORESIGHT_DEV_TYPE_HELPER;
>> +    cti_desc.subtype.helper_subtype =
>> CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI;
>>       cti_desc.ops = &cti_ops;
>>       cti_desc.groups = drvdata->ctidev.con_groups;
>>       cti_desc.dev = dev;
>> diff --git a/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>> b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>> index e528cff9d4e2..d25dd2737b49 100644
>> --- a/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>> +++ b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>> @@ -112,11 +112,11 @@ static ssize_t enable_store(struct device *dev,
>>           ret = pm_runtime_resume_and_get(dev->parent);
>>           if (ret)
>>               return ret;
>> -        ret = cti_enable(drvdata->csdev);
>> +        ret = cti_enable(drvdata->csdev, CS_MODE_SYSFS, NULL);
>>           if (ret)
>>               pm_runtime_put(dev->parent);
>>       } else {
>> -        ret = cti_disable(drvdata->csdev);
>> +        ret = cti_disable(drvdata->csdev, NULL);
>>           if (!ret)
>>               pm_runtime_put(dev->parent);
>>       }
>> diff --git a/drivers/hwtracing/coresight/coresight-cti.h
>> b/drivers/hwtracing/coresight/coresight-cti.h
>> index 8b106b13a244..cb9ee616d01f 100644
>> --- a/drivers/hwtracing/coresight/coresight-cti.h
>> +++ b/drivers/hwtracing/coresight/coresight-cti.h
>> @@ -215,8 +215,8 @@ int cti_add_connection_entry(struct device *dev,
>> struct cti_drvdata *drvdata,
>>                    const char *assoc_dev_name);
>>   struct cti_trig_con *cti_allocate_trig_con(struct device *dev, int
>> in_sigs,
>>                          int out_sigs);
>> -int cti_enable(struct coresight_device *csdev);
>> -int cti_disable(struct coresight_device *csdev);
>> +int cti_enable(struct coresight_device *csdev, enum cs_mode mode,
>> void *data);
>> +int cti_disable(struct coresight_device *csdev, void *data);
>>   void cti_write_all_hw_regs(struct cti_drvdata *drvdata);
>>   void cti_write_intack(struct device *dev, u32 ackval);
>>   void cti_write_single_reg(struct cti_drvdata *drvdata, int offset,
>> u32 value);
>> diff --git a/drivers/hwtracing/coresight/coresight-priv.h
>> b/drivers/hwtracing/coresight/coresight-priv.h
>> index a843f9d5c737..fff565d1cb42 100644
>> --- a/drivers/hwtracing/coresight/coresight-priv.h
>> +++ b/drivers/hwtracing/coresight/coresight-priv.h
>> @@ -211,8 +211,8 @@ void coresight_release_platform_data(struct
>> coresight_device *csdev,
>>                        struct coresight_platform_data *pdata);
>>   struct coresight_device *
>>   coresight_find_csdev_by_fwnode(struct fwnode_handle *r_fwnode);
>> -void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
>> -                      struct coresight_device *ect_csdev);
>> +void coresight_add_helper_mutex(struct coresight_device *csdev,
>> +                struct coresight_device *helper);
>>     void coresight_set_percpu_sink(int cpu, struct coresight_device
>> *csdev);
>>   struct coresight_device *coresight_get_percpu_sink(int cpu);
>> diff --git a/drivers/hwtracing/coresight/coresight-sysfs.c
>> b/drivers/hwtracing/coresight/coresight-sysfs.c
>> index 464ba5e1343b..dd78e9fcfc4d 100644
>> --- a/drivers/hwtracing/coresight/coresight-sysfs.c
>> +++ b/drivers/hwtracing/coresight/coresight-sysfs.c
>> @@ -148,6 +148,10 @@ int coresight_make_links(struct coresight_device
>> *orig,
>>       char *outs = NULL, *ins = NULL;
>>       struct coresight_sysfs_link *link = NULL;
>>   +    /* Helper devices aren't shown in sysfs */
>> +    if (conn->dest_port == -1 && conn->src_port == -1)
>> +        return 0;
>> +
>>       do {
>>           outs = devm_kasprintf(&orig->dev, GFP_KERNEL,
>>                         "out:%d", conn->src_port);
>> diff --git a/include/linux/coresight.h b/include/linux/coresight.h
>> index d2739a0286f1..ed37552761e4 100644
>> --- a/include/linux/coresight.h
>> +++ b/include/linux/coresight.h
>> @@ -40,8 +40,7 @@ enum coresight_dev_type {
>>       CORESIGHT_DEV_TYPE_LINK,
>>       CORESIGHT_DEV_TYPE_LINKSINK,
>>       CORESIGHT_DEV_TYPE_SOURCE,
>> -    CORESIGHT_DEV_TYPE_HELPER,
>> -    CORESIGHT_DEV_TYPE_ECT,
>> +    CORESIGHT_DEV_TYPE_HELPER
>>   };
>>     enum coresight_dev_subtype_sink {
>> @@ -66,12 +65,7 @@ enum coresight_dev_subtype_source {
>>     enum coresight_dev_subtype_helper {
>>       CORESIGHT_DEV_SUBTYPE_HELPER_CATU,
>> -};
>> -
>> -/* Embedded Cross Trigger (ECT) sub-types */
>> -enum coresight_dev_subtype_ect {
>> -    CORESIGHT_DEV_SUBTYPE_ECT_NONE,
>> -    CORESIGHT_DEV_SUBTYPE_ECT_CTI,
>> +    CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
>>   };
>>     /**
>> @@ -84,8 +78,6 @@ enum coresight_dev_subtype_ect {
>>    *            by @coresight_dev_subtype_source.
>>    * @helper_subtype:    type of helper this component is, as defined
>>    *            by @coresight_dev_subtype_helper.
>> - * @ect_subtype:        type of cross trigger this component is, as
>> - *            defined by @coresight_dev_subtype_ect
>>    */
>>   union coresight_dev_subtype {
>>       /* We have some devices which acts as LINK and SINK */
>> @@ -95,7 +87,6 @@ union coresight_dev_subtype {
>>       };
>>       enum coresight_dev_subtype_source source_subtype;
>>       enum coresight_dev_subtype_helper helper_subtype;
>> -    enum coresight_dev_subtype_ect ect_subtype;
>>   };
>>     /**
>> @@ -237,8 +228,6 @@ struct coresight_sysfs_link {
>>    *        from source to that sink.
>>    * @ea:        Device attribute for sink representation under PMU
>> directory.
>>    * @def_sink:    cached reference to default sink found for this
>> device.
>> - * @ect_dev:    Associated cross trigger device. Not part of the
>> trace data
>> - *        path or connections.
>>    * @nr_links:   number of sysfs links created to other components
>> from this
>>    *        device. These will appear in the "connections" group.
>>    * @has_conns_grp: Have added a "connections" group for sysfs links.
>> @@ -261,12 +250,9 @@ struct coresight_device {
>>       bool activated;    /* true only if a sink is part of a path */
>>       struct dev_ext_attribute *ea;
>>       struct coresight_device *def_sink;
>> -    /* cross trigger handling */
>> -    struct coresight_device *ect_dev;
>>       /* sysfs links between components */
>>       int nr_links;
>>       bool has_conns_grp;
>> -    bool ect_enabled; /* true only if associated ect device is
>> enabled */
>>       /* system configuration and feature lists */
>>       struct list_head feature_csdev_list;
>>       struct list_head config_csdev_list;
>> @@ -378,23 +364,11 @@ struct coresight_ops_helper {
>>       int (*disable)(struct coresight_device *csdev, void *data);
>>   };
>>   -/**
>> - * struct coresight_ops_ect - Ops for an embedded cross trigger device
>> - *
>> - * @enable    : Enable the device
>> - * @disable    : Disable the device
>> - */
>> -struct coresight_ops_ect {
>> -    int (*enable)(struct coresight_device *csdev);
>> -    int (*disable)(struct coresight_device *csdev);
>> -};
>> -
>>   struct coresight_ops {
>>       const struct coresight_ops_sink *sink_ops;
>>       const struct coresight_ops_link *link_ops;
>>       const struct coresight_ops_source *source_ops;
>>       const struct coresight_ops_helper *helper_ops;
>> -    const struct coresight_ops_ect *ect_ops;
>>   };
>>     #if IS_ENABLED(CONFIG_CORESIGHT)
>
James Clark April 4, 2023, 1:04 p.m. UTC | #3
On 04/04/2023 13:55, James Clark wrote:
> 
> 
> On 04/04/2023 10:21, Suzuki K Poulose wrote:
>> On 29/03/2023 12:53, James Clark wrote:
>>> The CTI module has some hard coded refcounting code that has a leak.
>>> For example running perf and then trying to unload it fails:
>>>
>>>    perf record -e cs_etm// -a -- ls
>>>    rmmod coresight_cti
>>>
>>>    rmmod: ERROR: Module coresight_cti is in use
>>>
>>> The coresight core already handles references of devices in use, so by
>>> making CTI a normal helper device, we get working refcounting for free.
>>>
>>> Signed-off-by: James Clark <james.clark@arm.com>
>>> ---
>>>   drivers/hwtracing/coresight/coresight-core.c  | 99 ++++++-------------
>>>   .../hwtracing/coresight/coresight-cti-core.c  | 52 +++++-----
>>>   .../hwtracing/coresight/coresight-cti-sysfs.c |  4 +-
>>>   drivers/hwtracing/coresight/coresight-cti.h   |  4 +-
>>>   drivers/hwtracing/coresight/coresight-priv.h  |  4 +-
>>>   drivers/hwtracing/coresight/coresight-sysfs.c |  4 +
>>>   include/linux/coresight.h                     | 30 +-----
>>>   7 files changed, 70 insertions(+), 127 deletions(-)
>>>
>>> diff --git a/drivers/hwtracing/coresight/coresight-core.c
>>> b/drivers/hwtracing/coresight/coresight-core.c
>>> index 65f5bd8516d8..458d91b4e23f 100644
>>> --- a/drivers/hwtracing/coresight/coresight-core.c
>>> +++ b/drivers/hwtracing/coresight/coresight-core.c
>>> @@ -254,60 +254,39 @@ void coresight_disclaim_device(struct
>>> coresight_device *csdev)
>>>   }
>>>   EXPORT_SYMBOL_GPL(coresight_disclaim_device);
>>>   -/* enable or disable an associated CTI device of the supplied CS
>>> device */
>>> -static int
>>> -coresight_control_assoc_ectdev(struct coresight_device *csdev, bool
>>> enable)
>>> -{
>>> -    int ect_ret = 0;
>>> -    struct coresight_device *ect_csdev = csdev->ect_dev;
>>> -    struct module *mod;
>>> -
>>> -    if (!ect_csdev)
>>> -        return 0;
>>> -    if ((!ect_ops(ect_csdev)->enable) || (!ect_ops(ect_csdev)->disable))
>>> -        return 0;
>>> -
>>> -    mod = ect_csdev->dev.parent->driver->owner;
>>> -    if (enable) {
>>> -        if (try_module_get(mod)) {
>>> -            ect_ret = ect_ops(ect_csdev)->enable(ect_csdev);
>>> -            if (ect_ret) {
>>> -                module_put(mod);
>>> -            } else {
>>> -                get_device(ect_csdev->dev.parent);
>>> -                csdev->ect_enabled = true;
>>> -            }
>>> -        } else
>>> -            ect_ret = -ENODEV;
>>> -    } else {
>>> -        if (csdev->ect_enabled) {
>>> -            ect_ret = ect_ops(ect_csdev)->disable(ect_csdev);
>>> -            put_device(ect_csdev->dev.parent);
>>> -            module_put(mod);
>>> -            csdev->ect_enabled = false;
>>> -        }
>>> -    }
>>> -
>>> -    /* output warning if ECT enable is preventing trace operation */
>>> -    if (ect_ret)
>>> -        dev_info(&csdev->dev, "Associated ECT device (%s) %s failed\n",
>>> -             dev_name(&ect_csdev->dev),
>>> -             enable ? "enable" : "disable");
>>> -    return ect_ret;
>>> -}
>>> -
>>>   /*
>>> - * Set the associated ect / cti device while holding the coresight_mutex
>>> + * Add a helper as an output device while holding the coresight_mutex
>>>    * to avoid a race with coresight_enable that may try to use this
>>> value.
>>>    */
>>> -void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
>>> -                      struct coresight_device *ect_csdev)
>>> +void coresight_add_helper_mutex(struct coresight_device *csdev,
>>> +                struct coresight_device *helper)
>>
>> minor nit: It may be a good idea to rename this, in line with the
>> kernel naming convention :
>>
>>     coresight_add_helper_unlocked()
>>
>> Or if this is the only variant, it is OK to leave it as :
>>     coresight_add_helper()
>> with a big fat comment in the function description to indicate
>> that it takes the mutex and may be even add a :
>>
> There is already a bit of a comment in the description but I can expand
> on it more.
> 
>> might_sleep() and lockdep_assert_not_held(&coresight_mutex);
>>
>> in the function.
>>
> 
> I'm not sure if lockdep_assert_not_held() would be right because
> sometimes it could be held if another device is being created at the
> same time? Or something like a session is started at the same time a CTI
> device is added.
> 

Oh I see it's not for any task, it's just for the current one. That
makes sense then I can add it.

Although it looks like it only warns when lockdep is enabled, but don't
you get a warning anyway if you try to take the lock twice with lockdep
enabled? So I'm not sure why we would add lockdep_assert_not_held() here
and not on all the mutex_lock() calls?

>>>   {
>>> +    int i;
>>> +    struct coresight_connection conn = {};
>>> +
>>>       mutex_lock(&coresight_mutex);
>>> -    csdev->ect_dev = ect_csdev;
>>> +    conn.dest_fwnode = fwnode_handle_get(dev_fwnode(&helper->dev));
>>> +    conn.dest_dev = helper;
>>> +    conn.dest_port = conn.src_port = -1;
>>> +    conn.src_dev = csdev;
>>> +
>>> +    /*
>>> +     * Check for duplicates because this is called every time a helper
>>> +     * device is re-loaded. Existing connections will get re-linked
>>> +     * automatically.
>>> +     */
>>
>> Thanks for adding this comment here. It does look like the already added
>> output connection to the "origin" device would automatically resolve the
>> connection and add in the "in-connection" to the CTI device.
>>
>>> +    for (i = 0; i < csdev->pdata->nr_outconns; ++i)
>>> +        if (csdev->pdata->out_conns[i]->dest_fwnode == conn.dest_fwnode)
>>> +            goto unlock;
>>> +
>>> +    coresight_add_out_conn(csdev->dev.parent, csdev->pdata, &conn);
>>
>> This makes me wonder if we should return the new connection in
>> coresight_add_out_conn() in case of success, rather than assuming the
>> last one (which is always the case though.).
>>
>> i.e.,
>>     new_conn = coresight_add_out_conn(...)
>>     if (new_conn)
>>         coresight_add_in_conn(new_conn);
>>
> 
> Yeah I thought about doing that too, will change it.
> 
>>> +    coresight_add_in_conn(
>>> +        csdev->pdata->out_conns[csdev->pdata->nr_outconns - 1]);
>>> +
>>> +unlock:
>>>       mutex_unlock(&coresight_mutex);
>>>   }
>>> -EXPORT_SYMBOL_GPL(coresight_set_assoc_ectdev_mutex);
>>> +EXPORT_SYMBOL_GPL(coresight_add_helper_mutex);
>>>     static int coresight_enable_sink(struct coresight_device *csdev,
>>>                    enum cs_mode mode, void *data)
>>> @@ -321,12 +300,8 @@ static int coresight_enable_sink(struct
>>> coresight_device *csdev,
>>>       if (!sink_ops(csdev)->enable)
>>>           return -EINVAL;
>>>   -    ret = coresight_control_assoc_ectdev(csdev, true);
>>> -    if (ret)
>>> -        return ret;
>>>       ret = sink_ops(csdev)->enable(csdev, mode, data);
>>>       if (ret) {
>>> -        coresight_control_assoc_ectdev(csdev, false);
>>>           return ret;
>>>       }
>>>       csdev->enable = true;
>>> @@ -344,7 +319,6 @@ static void coresight_disable_sink(struct
>>> coresight_device *csdev)
>>>       ret = sink_ops(csdev)->disable(csdev);
>>>       if (ret)
>>>           return;
>>> -    coresight_control_assoc_ectdev(csdev, false);
>>>       csdev->enable = false;
>>>   }
>>>   @@ -369,17 +343,11 @@ static int coresight_enable_link(struct
>>> coresight_device *csdev,
>>>           return PTR_ERR(outconn);
>>>         if (link_ops(csdev)->enable) {
>>> -        ret = coresight_control_assoc_ectdev(csdev, true);
>>> -        if (!ret) {
>>> -            ret = link_ops(csdev)->enable(csdev, inconn, outconn);
>>> -            if (ret)
>>> -                coresight_control_assoc_ectdev(csdev, false);
>>> -        }
>>> +        ret = link_ops(csdev)->enable(csdev, inconn, outconn);
>>> +        if (!ret)
>>> +            csdev->enable = true;
>>>       }
>>>   -    if (!ret)
>>> -        csdev->enable = true;
>>> -
>>>       return ret;
>>>   }
>>>   @@ -400,7 +368,6 @@ static void coresight_disable_link(struct
>>> coresight_device *csdev,
>>>         if (link_ops(csdev)->disable) {
>>>           link_ops(csdev)->disable(csdev, inconn, outconn);
>>> -        coresight_control_assoc_ectdev(csdev, false);
>>>       }
>>>         if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) {
>>> @@ -428,14 +395,9 @@ int coresight_enable_source(struct
>>> coresight_device *csdev, void *data,
>>>         if (!csdev->enable) {
>>>           if (source_ops(csdev)->enable) {
>>> -            ret = coresight_control_assoc_ectdev(csdev, true);
>>> -            if (ret)
>>> -                return ret;
>>>               ret = source_ops(csdev)->enable(csdev, data, mode);
>>> -            if (ret) {
>>> -                coresight_control_assoc_ectdev(csdev, false);
>>> +            if (ret)
>>>                   return ret;
>>> -            }
>>>           }
>>>           csdev->enable = true;
>>>       }
>>> @@ -499,7 +461,6 @@ bool coresight_disable_source(struct
>>> coresight_device *csdev, void *data)
>>>       if (atomic_dec_return(&csdev->refcnt) == 0) {
>>>           if (source_ops(csdev)->disable)
>>>               source_ops(csdev)->disable(csdev, data);
>>> -        coresight_control_assoc_ectdev(csdev, false);
>>>           coresight_disable_helpers(csdev);
>>>           csdev->enable = false;
>>>       }
>>> diff --git a/drivers/hwtracing/coresight/coresight-cti-core.c
>>> b/drivers/hwtracing/coresight/coresight-cti-core.c
>>> index 277c890a1f1f..db7a2212ec18 100644
>>> --- a/drivers/hwtracing/coresight/coresight-cti-core.c
>>> +++ b/drivers/hwtracing/coresight/coresight-cti-core.c
>>> @@ -555,7 +555,10 @@ static void cti_add_assoc_to_csdev(struct
>>> coresight_device *csdev)
>>>       mutex_lock(&ect_mutex);
>>>         /* exit if current is an ECT device.*/
>>> -    if ((csdev->type == CORESIGHT_DEV_TYPE_ECT) || list_empty(&ect_net))
>>> +    if ((csdev->type == CORESIGHT_DEV_TYPE_HELPER &&
>>> +         csdev->subtype.helper_subtype ==
>>> +             CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI) ||
>>> +        list_empty(&ect_net))
>>>           goto cti_add_done;
>>>         /* if we didn't find the csdev previously we used the fwnode
>>> name */
>>> @@ -571,8 +574,7 @@ static void cti_add_assoc_to_csdev(struct
>>> coresight_device *csdev)
>>>                * if we found a matching csdev then update the ECT
>>>                * association pointer for the device with this CTI.
>>>                */
>>> -            coresight_set_assoc_ectdev_mutex(csdev,
>>> -                             ect_item->csdev);
>>> +            coresight_add_helper_mutex(csdev, ect_item->csdev);
>>>               break;
>>>           }
>>>       }
>>> @@ -582,26 +584,30 @@ static void cti_add_assoc_to_csdev(struct
>>> coresight_device *csdev)
>>>     /*
>>>    * Removing the associated devices is easier.
>>> - * A CTI will not have a value for csdev->ect_dev.
>>>    */
>>>   static void cti_remove_assoc_from_csdev(struct coresight_device *csdev)
>>>   {
>>>       struct cti_drvdata *ctidrv;
>>>       struct cti_trig_con *tc;
>>> +    union coresight_dev_subtype cti_subtype = {
>>> +        .helper_subtype = CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
>>> +    };
>>> +    struct coresight_device *cti_csdev = coresight_find_output_type(
>>> +        csdev->pdata, CORESIGHT_DEV_TYPE_HELPER, cti_subtype);
>>
>> minor nit: Please could we split the initialisation ? Or at least move
>> this after the next variable declaration below ?
>>
>>
>> Rest looks fine to me.
>>
>> Suzuki
>>
>>>       struct cti_device *ctidev;
>>>   +    if (!cti_csdev)
>>> +        return;
>>> +
>>>       mutex_lock(&ect_mutex);
>>> -    if (csdev->ect_dev) {
>>> -        ctidrv = csdev_to_cti_drvdata(csdev->ect_dev);
>>> -        ctidev = &ctidrv->ctidev;
>>> -        list_for_each_entry(tc, &ctidev->trig_cons, node) {
>>> -            if (tc->con_dev == csdev) {
>>> -                cti_remove_sysfs_link(ctidrv, tc);
>>> -                tc->con_dev = NULL;
>>> -                break;
>>> -            }
>>> +    ctidrv = csdev_to_cti_drvdata(cti_csdev);
>>> +    ctidev = &ctidrv->ctidev;
>>> +    list_for_each_entry(tc, &ctidev->trig_cons, node) {
>>> +        if (tc->con_dev == csdev) {
>>> +            cti_remove_sysfs_link(ctidrv, tc);
>>> +            tc->con_dev = NULL;
>>> +            break;
>>>           }
>>> -        csdev->ect_dev = NULL;
>>>       }
>>>       mutex_unlock(&ect_mutex);
>>>   }
>>> @@ -630,8 +636,8 @@ static void cti_update_conn_xrefs(struct
>>> cti_drvdata *drvdata)
>>>               /* if we can set the sysfs link */
>>>               if (cti_add_sysfs_link(drvdata, tc))
>>>                   /* set the CTI/csdev association */
>>> -                coresight_set_assoc_ectdev_mutex(tc->con_dev,
>>> -                             drvdata->csdev);
>>> +                coresight_add_helper_mutex(tc->con_dev,
>>> +                               drvdata->csdev);
>>>               else
>>>                   /* otherwise remove reference from CTI */
>>>                   tc->con_dev = NULL;
>>> @@ -646,8 +652,6 @@ static void cti_remove_conn_xrefs(struct
>>> cti_drvdata *drvdata)
>>>         list_for_each_entry(tc, &ctidev->trig_cons, node) {
>>>           if (tc->con_dev) {
>>> -            coresight_set_assoc_ectdev_mutex(tc->con_dev,
>>> -                             NULL);
>>>               cti_remove_sysfs_link(drvdata, tc);
>>>               tc->con_dev = NULL;
>>>           }
>>> @@ -795,27 +799,27 @@ static void cti_pm_release(struct cti_drvdata
>>> *drvdata)
>>>   }
>>>     /** cti ect operations **/
>>> -int cti_enable(struct coresight_device *csdev)
>>> +int cti_enable(struct coresight_device *csdev, enum cs_mode mode,
>>> void *data)
>>>   {
>>>       struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
>>>         return cti_enable_hw(drvdata);
>>>   }
>>>   -int cti_disable(struct coresight_device *csdev)
>>> +int cti_disable(struct coresight_device *csdev, void *data)
>>>   {
>>>       struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
>>>         return cti_disable_hw(drvdata);
>>>   }
>>>   -static const struct coresight_ops_ect cti_ops_ect = {
>>> +static const struct coresight_ops_helper cti_ops_ect = {
>>>       .enable = cti_enable,
>>>       .disable = cti_disable,
>>>   };
>>>     static const struct coresight_ops cti_ops = {
>>> -    .ect_ops = &cti_ops_ect,
>>> +    .helper_ops = &cti_ops_ect,
>>>   };
>>>     /*
>>> @@ -922,8 +926,8 @@ static int cti_probe(struct amba_device *adev,
>>> const struct amba_id *id)
>>>         /* set up coresight component description */
>>>       cti_desc.pdata = pdata;
>>> -    cti_desc.type = CORESIGHT_DEV_TYPE_ECT;
>>> -    cti_desc.subtype.ect_subtype = CORESIGHT_DEV_SUBTYPE_ECT_CTI;
>>> +    cti_desc.type = CORESIGHT_DEV_TYPE_HELPER;
>>> +    cti_desc.subtype.helper_subtype =
>>> CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI;
>>>       cti_desc.ops = &cti_ops;
>>>       cti_desc.groups = drvdata->ctidev.con_groups;
>>>       cti_desc.dev = dev;
>>> diff --git a/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>>> b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>>> index e528cff9d4e2..d25dd2737b49 100644
>>> --- a/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>>> +++ b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
>>> @@ -112,11 +112,11 @@ static ssize_t enable_store(struct device *dev,
>>>           ret = pm_runtime_resume_and_get(dev->parent);
>>>           if (ret)
>>>               return ret;
>>> -        ret = cti_enable(drvdata->csdev);
>>> +        ret = cti_enable(drvdata->csdev, CS_MODE_SYSFS, NULL);
>>>           if (ret)
>>>               pm_runtime_put(dev->parent);
>>>       } else {
>>> -        ret = cti_disable(drvdata->csdev);
>>> +        ret = cti_disable(drvdata->csdev, NULL);
>>>           if (!ret)
>>>               pm_runtime_put(dev->parent);
>>>       }
>>> diff --git a/drivers/hwtracing/coresight/coresight-cti.h
>>> b/drivers/hwtracing/coresight/coresight-cti.h
>>> index 8b106b13a244..cb9ee616d01f 100644
>>> --- a/drivers/hwtracing/coresight/coresight-cti.h
>>> +++ b/drivers/hwtracing/coresight/coresight-cti.h
>>> @@ -215,8 +215,8 @@ int cti_add_connection_entry(struct device *dev,
>>> struct cti_drvdata *drvdata,
>>>                    const char *assoc_dev_name);
>>>   struct cti_trig_con *cti_allocate_trig_con(struct device *dev, int
>>> in_sigs,
>>>                          int out_sigs);
>>> -int cti_enable(struct coresight_device *csdev);
>>> -int cti_disable(struct coresight_device *csdev);
>>> +int cti_enable(struct coresight_device *csdev, enum cs_mode mode,
>>> void *data);
>>> +int cti_disable(struct coresight_device *csdev, void *data);
>>>   void cti_write_all_hw_regs(struct cti_drvdata *drvdata);
>>>   void cti_write_intack(struct device *dev, u32 ackval);
>>>   void cti_write_single_reg(struct cti_drvdata *drvdata, int offset,
>>> u32 value);
>>> diff --git a/drivers/hwtracing/coresight/coresight-priv.h
>>> b/drivers/hwtracing/coresight/coresight-priv.h
>>> index a843f9d5c737..fff565d1cb42 100644
>>> --- a/drivers/hwtracing/coresight/coresight-priv.h
>>> +++ b/drivers/hwtracing/coresight/coresight-priv.h
>>> @@ -211,8 +211,8 @@ void coresight_release_platform_data(struct
>>> coresight_device *csdev,
>>>                        struct coresight_platform_data *pdata);
>>>   struct coresight_device *
>>>   coresight_find_csdev_by_fwnode(struct fwnode_handle *r_fwnode);
>>> -void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
>>> -                      struct coresight_device *ect_csdev);
>>> +void coresight_add_helper_mutex(struct coresight_device *csdev,
>>> +                struct coresight_device *helper);
>>>     void coresight_set_percpu_sink(int cpu, struct coresight_device
>>> *csdev);
>>>   struct coresight_device *coresight_get_percpu_sink(int cpu);
>>> diff --git a/drivers/hwtracing/coresight/coresight-sysfs.c
>>> b/drivers/hwtracing/coresight/coresight-sysfs.c
>>> index 464ba5e1343b..dd78e9fcfc4d 100644
>>> --- a/drivers/hwtracing/coresight/coresight-sysfs.c
>>> +++ b/drivers/hwtracing/coresight/coresight-sysfs.c
>>> @@ -148,6 +148,10 @@ int coresight_make_links(struct coresight_device
>>> *orig,
>>>       char *outs = NULL, *ins = NULL;
>>>       struct coresight_sysfs_link *link = NULL;
>>>   +    /* Helper devices aren't shown in sysfs */
>>> +    if (conn->dest_port == -1 && conn->src_port == -1)
>>> +        return 0;
>>> +
>>>       do {
>>>           outs = devm_kasprintf(&orig->dev, GFP_KERNEL,
>>>                         "out:%d", conn->src_port);
>>> diff --git a/include/linux/coresight.h b/include/linux/coresight.h
>>> index d2739a0286f1..ed37552761e4 100644
>>> --- a/include/linux/coresight.h
>>> +++ b/include/linux/coresight.h
>>> @@ -40,8 +40,7 @@ enum coresight_dev_type {
>>>       CORESIGHT_DEV_TYPE_LINK,
>>>       CORESIGHT_DEV_TYPE_LINKSINK,
>>>       CORESIGHT_DEV_TYPE_SOURCE,
>>> -    CORESIGHT_DEV_TYPE_HELPER,
>>> -    CORESIGHT_DEV_TYPE_ECT,
>>> +    CORESIGHT_DEV_TYPE_HELPER
>>>   };
>>>     enum coresight_dev_subtype_sink {
>>> @@ -66,12 +65,7 @@ enum coresight_dev_subtype_source {
>>>     enum coresight_dev_subtype_helper {
>>>       CORESIGHT_DEV_SUBTYPE_HELPER_CATU,
>>> -};
>>> -
>>> -/* Embedded Cross Trigger (ECT) sub-types */
>>> -enum coresight_dev_subtype_ect {
>>> -    CORESIGHT_DEV_SUBTYPE_ECT_NONE,
>>> -    CORESIGHT_DEV_SUBTYPE_ECT_CTI,
>>> +    CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
>>>   };
>>>     /**
>>> @@ -84,8 +78,6 @@ enum coresight_dev_subtype_ect {
>>>    *            by @coresight_dev_subtype_source.
>>>    * @helper_subtype:    type of helper this component is, as defined
>>>    *            by @coresight_dev_subtype_helper.
>>> - * @ect_subtype:        type of cross trigger this component is, as
>>> - *            defined by @coresight_dev_subtype_ect
>>>    */
>>>   union coresight_dev_subtype {
>>>       /* We have some devices which acts as LINK and SINK */
>>> @@ -95,7 +87,6 @@ union coresight_dev_subtype {
>>>       };
>>>       enum coresight_dev_subtype_source source_subtype;
>>>       enum coresight_dev_subtype_helper helper_subtype;
>>> -    enum coresight_dev_subtype_ect ect_subtype;
>>>   };
>>>     /**
>>> @@ -237,8 +228,6 @@ struct coresight_sysfs_link {
>>>    *        from source to that sink.
>>>    * @ea:        Device attribute for sink representation under PMU
>>> directory.
>>>    * @def_sink:    cached reference to default sink found for this
>>> device.
>>> - * @ect_dev:    Associated cross trigger device. Not part of the
>>> trace data
>>> - *        path or connections.
>>>    * @nr_links:   number of sysfs links created to other components
>>> from this
>>>    *        device. These will appear in the "connections" group.
>>>    * @has_conns_grp: Have added a "connections" group for sysfs links.
>>> @@ -261,12 +250,9 @@ struct coresight_device {
>>>       bool activated;    /* true only if a sink is part of a path */
>>>       struct dev_ext_attribute *ea;
>>>       struct coresight_device *def_sink;
>>> -    /* cross trigger handling */
>>> -    struct coresight_device *ect_dev;
>>>       /* sysfs links between components */
>>>       int nr_links;
>>>       bool has_conns_grp;
>>> -    bool ect_enabled; /* true only if associated ect device is
>>> enabled */
>>>       /* system configuration and feature lists */
>>>       struct list_head feature_csdev_list;
>>>       struct list_head config_csdev_list;
>>> @@ -378,23 +364,11 @@ struct coresight_ops_helper {
>>>       int (*disable)(struct coresight_device *csdev, void *data);
>>>   };
>>>   -/**
>>> - * struct coresight_ops_ect - Ops for an embedded cross trigger device
>>> - *
>>> - * @enable    : Enable the device
>>> - * @disable    : Disable the device
>>> - */
>>> -struct coresight_ops_ect {
>>> -    int (*enable)(struct coresight_device *csdev);
>>> -    int (*disable)(struct coresight_device *csdev);
>>> -};
>>> -
>>>   struct coresight_ops {
>>>       const struct coresight_ops_sink *sink_ops;
>>>       const struct coresight_ops_link *link_ops;
>>>       const struct coresight_ops_source *source_ops;
>>>       const struct coresight_ops_helper *helper_ops;
>>> -    const struct coresight_ops_ect *ect_ops;
>>>   };
>>>     #if IS_ENABLED(CONFIG_CORESIGHT)
>>
Suzuki K Poulose April 4, 2023, 1:59 p.m. UTC | #4
On 04/04/2023 14:04, James Clark wrote:
> 
> 
> On 04/04/2023 13:55, James Clark wrote:
>>
>>
>> On 04/04/2023 10:21, Suzuki K Poulose wrote:
>>> On 29/03/2023 12:53, James Clark wrote:
>>>> The CTI module has some hard coded refcounting code that has a leak.
>>>> For example running perf and then trying to unload it fails:
>>>>
>>>>     perf record -e cs_etm// -a -- ls
>>>>     rmmod coresight_cti
>>>>
>>>>     rmmod: ERROR: Module coresight_cti is in use
>>>>
>>>> The coresight core already handles references of devices in use, so by
>>>> making CTI a normal helper device, we get working refcounting for free.
>>>>
>>>> Signed-off-by: James Clark <james.clark@arm.com>
>>>> ---
>>>>    drivers/hwtracing/coresight/coresight-core.c  | 99 ++++++-------------
>>>>    .../hwtracing/coresight/coresight-cti-core.c  | 52 +++++-----
>>>>    .../hwtracing/coresight/coresight-cti-sysfs.c |  4 +-
>>>>    drivers/hwtracing/coresight/coresight-cti.h   |  4 +-
>>>>    drivers/hwtracing/coresight/coresight-priv.h  |  4 +-
>>>>    drivers/hwtracing/coresight/coresight-sysfs.c |  4 +
>>>>    include/linux/coresight.h                     | 30 +-----
>>>>    7 files changed, 70 insertions(+), 127 deletions(-)
>>>>
>>>> diff --git a/drivers/hwtracing/coresight/coresight-core.c
>>>> b/drivers/hwtracing/coresight/coresight-core.c
>>>> index 65f5bd8516d8..458d91b4e23f 100644
>>>> --- a/drivers/hwtracing/coresight/coresight-core.c
>>>> +++ b/drivers/hwtracing/coresight/coresight-core.c
>>>> @@ -254,60 +254,39 @@ void coresight_disclaim_device(struct
>>>> coresight_device *csdev)
>>>>    }
>>>>    EXPORT_SYMBOL_GPL(coresight_disclaim_device);
>>>>    -/* enable or disable an associated CTI device of the supplied CS
>>>> device */
>>>> -static int
>>>> -coresight_control_assoc_ectdev(struct coresight_device *csdev, bool
>>>> enable)
>>>> -{
>>>> -    int ect_ret = 0;
>>>> -    struct coresight_device *ect_csdev = csdev->ect_dev;
>>>> -    struct module *mod;
>>>> -
>>>> -    if (!ect_csdev)
>>>> -        return 0;
>>>> -    if ((!ect_ops(ect_csdev)->enable) || (!ect_ops(ect_csdev)->disable))
>>>> -        return 0;
>>>> -
>>>> -    mod = ect_csdev->dev.parent->driver->owner;
>>>> -    if (enable) {
>>>> -        if (try_module_get(mod)) {
>>>> -            ect_ret = ect_ops(ect_csdev)->enable(ect_csdev);
>>>> -            if (ect_ret) {
>>>> -                module_put(mod);
>>>> -            } else {
>>>> -                get_device(ect_csdev->dev.parent);
>>>> -                csdev->ect_enabled = true;
>>>> -            }
>>>> -        } else
>>>> -            ect_ret = -ENODEV;
>>>> -    } else {
>>>> -        if (csdev->ect_enabled) {
>>>> -            ect_ret = ect_ops(ect_csdev)->disable(ect_csdev);
>>>> -            put_device(ect_csdev->dev.parent);
>>>> -            module_put(mod);
>>>> -            csdev->ect_enabled = false;
>>>> -        }
>>>> -    }
>>>> -
>>>> -    /* output warning if ECT enable is preventing trace operation */
>>>> -    if (ect_ret)
>>>> -        dev_info(&csdev->dev, "Associated ECT device (%s) %s failed\n",
>>>> -             dev_name(&ect_csdev->dev),
>>>> -             enable ? "enable" : "disable");
>>>> -    return ect_ret;
>>>> -}
>>>> -
>>>>    /*
>>>> - * Set the associated ect / cti device while holding the coresight_mutex
>>>> + * Add a helper as an output device while holding the coresight_mutex
>>>>     * to avoid a race with coresight_enable that may try to use this
>>>> value.
>>>>     */
>>>> -void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
>>>> -                      struct coresight_device *ect_csdev)
>>>> +void coresight_add_helper_mutex(struct coresight_device *csdev,
>>>> +                struct coresight_device *helper)
>>>
>>> minor nit: It may be a good idea to rename this, in line with the
>>> kernel naming convention :
>>>
>>>      coresight_add_helper_unlocked()
>>>
>>> Or if this is the only variant, it is OK to leave it as :
>>>      coresight_add_helper()
>>> with a big fat comment in the function description to indicate
>>> that it takes the mutex and may be even add a :
>>>
>> There is already a bit of a comment in the description but I can expand
>> on it more.
>>
>>> might_sleep() and lockdep_assert_not_held(&coresight_mutex);
>>>
>>> in the function.
>>>
>>
>> I'm not sure if lockdep_assert_not_held() would be right because
>> sometimes it could be held if another device is being created at the
>> same time? Or something like a session is started at the same time a CTI
>> device is added.
>>
> 
> Oh I see it's not for any task, it's just for the current one. That
> makes sense then I can add it.
> 
> Although it looks like it only warns when lockdep is enabled, but don't
> you get a warning anyway if you try to take the lock twice with lockdep
> enabled?

Thats true, you could ignore the lockdep check.

  So I'm not sure why we would add lockdep_assert_not_held() here
> and not on all the mutex_lock() calls?\

Ah. I double checked this and the coresight_mutex is static and local to
  coresight-core.c. So there is no point in talking about locking for
external users. So I would just leave out any suffixes and simply use
the lockdep check implicit from mutex_lock().

Suzuki
diff mbox series

Patch

diff --git a/drivers/hwtracing/coresight/coresight-core.c b/drivers/hwtracing/coresight/coresight-core.c
index 65f5bd8516d8..458d91b4e23f 100644
--- a/drivers/hwtracing/coresight/coresight-core.c
+++ b/drivers/hwtracing/coresight/coresight-core.c
@@ -254,60 +254,39 @@  void coresight_disclaim_device(struct coresight_device *csdev)
 }
 EXPORT_SYMBOL_GPL(coresight_disclaim_device);
 
-/* enable or disable an associated CTI device of the supplied CS device */
-static int
-coresight_control_assoc_ectdev(struct coresight_device *csdev, bool enable)
-{
-	int ect_ret = 0;
-	struct coresight_device *ect_csdev = csdev->ect_dev;
-	struct module *mod;
-
-	if (!ect_csdev)
-		return 0;
-	if ((!ect_ops(ect_csdev)->enable) || (!ect_ops(ect_csdev)->disable))
-		return 0;
-
-	mod = ect_csdev->dev.parent->driver->owner;
-	if (enable) {
-		if (try_module_get(mod)) {
-			ect_ret = ect_ops(ect_csdev)->enable(ect_csdev);
-			if (ect_ret) {
-				module_put(mod);
-			} else {
-				get_device(ect_csdev->dev.parent);
-				csdev->ect_enabled = true;
-			}
-		} else
-			ect_ret = -ENODEV;
-	} else {
-		if (csdev->ect_enabled) {
-			ect_ret = ect_ops(ect_csdev)->disable(ect_csdev);
-			put_device(ect_csdev->dev.parent);
-			module_put(mod);
-			csdev->ect_enabled = false;
-		}
-	}
-
-	/* output warning if ECT enable is preventing trace operation */
-	if (ect_ret)
-		dev_info(&csdev->dev, "Associated ECT device (%s) %s failed\n",
-			 dev_name(&ect_csdev->dev),
-			 enable ? "enable" : "disable");
-	return ect_ret;
-}
-
 /*
- * Set the associated ect / cti device while holding the coresight_mutex
+ * Add a helper as an output device while holding the coresight_mutex
  * to avoid a race with coresight_enable that may try to use this value.
  */
-void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
-				      struct coresight_device *ect_csdev)
+void coresight_add_helper_mutex(struct coresight_device *csdev,
+				struct coresight_device *helper)
 {
+	int i;
+	struct coresight_connection conn = {};
+
 	mutex_lock(&coresight_mutex);
-	csdev->ect_dev = ect_csdev;
+	conn.dest_fwnode = fwnode_handle_get(dev_fwnode(&helper->dev));
+	conn.dest_dev = helper;
+	conn.dest_port = conn.src_port = -1;
+	conn.src_dev = csdev;
+
+	/*
+	 * Check for duplicates because this is called every time a helper
+	 * device is re-loaded. Existing connections will get re-linked
+	 * automatically.
+	 */
+	for (i = 0; i < csdev->pdata->nr_outconns; ++i)
+		if (csdev->pdata->out_conns[i]->dest_fwnode == conn.dest_fwnode)
+			goto unlock;
+
+	coresight_add_out_conn(csdev->dev.parent, csdev->pdata, &conn);
+	coresight_add_in_conn(
+		csdev->pdata->out_conns[csdev->pdata->nr_outconns - 1]);
+
+unlock:
 	mutex_unlock(&coresight_mutex);
 }
-EXPORT_SYMBOL_GPL(coresight_set_assoc_ectdev_mutex);
+EXPORT_SYMBOL_GPL(coresight_add_helper_mutex);
 
 static int coresight_enable_sink(struct coresight_device *csdev,
 				 enum cs_mode mode, void *data)
@@ -321,12 +300,8 @@  static int coresight_enable_sink(struct coresight_device *csdev,
 	if (!sink_ops(csdev)->enable)
 		return -EINVAL;
 
-	ret = coresight_control_assoc_ectdev(csdev, true);
-	if (ret)
-		return ret;
 	ret = sink_ops(csdev)->enable(csdev, mode, data);
 	if (ret) {
-		coresight_control_assoc_ectdev(csdev, false);
 		return ret;
 	}
 	csdev->enable = true;
@@ -344,7 +319,6 @@  static void coresight_disable_sink(struct coresight_device *csdev)
 	ret = sink_ops(csdev)->disable(csdev);
 	if (ret)
 		return;
-	coresight_control_assoc_ectdev(csdev, false);
 	csdev->enable = false;
 }
 
@@ -369,17 +343,11 @@  static int coresight_enable_link(struct coresight_device *csdev,
 		return PTR_ERR(outconn);
 
 	if (link_ops(csdev)->enable) {
-		ret = coresight_control_assoc_ectdev(csdev, true);
-		if (!ret) {
-			ret = link_ops(csdev)->enable(csdev, inconn, outconn);
-			if (ret)
-				coresight_control_assoc_ectdev(csdev, false);
-		}
+		ret = link_ops(csdev)->enable(csdev, inconn, outconn);
+		if (!ret)
+			csdev->enable = true;
 	}
 
-	if (!ret)
-		csdev->enable = true;
-
 	return ret;
 }
 
@@ -400,7 +368,6 @@  static void coresight_disable_link(struct coresight_device *csdev,
 
 	if (link_ops(csdev)->disable) {
 		link_ops(csdev)->disable(csdev, inconn, outconn);
-		coresight_control_assoc_ectdev(csdev, false);
 	}
 
 	if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) {
@@ -428,14 +395,9 @@  int coresight_enable_source(struct coresight_device *csdev, void *data,
 
 	if (!csdev->enable) {
 		if (source_ops(csdev)->enable) {
-			ret = coresight_control_assoc_ectdev(csdev, true);
-			if (ret)
-				return ret;
 			ret = source_ops(csdev)->enable(csdev, data, mode);
-			if (ret) {
-				coresight_control_assoc_ectdev(csdev, false);
+			if (ret)
 				return ret;
-			}
 		}
 		csdev->enable = true;
 	}
@@ -499,7 +461,6 @@  bool coresight_disable_source(struct coresight_device *csdev, void *data)
 	if (atomic_dec_return(&csdev->refcnt) == 0) {
 		if (source_ops(csdev)->disable)
 			source_ops(csdev)->disable(csdev, data);
-		coresight_control_assoc_ectdev(csdev, false);
 		coresight_disable_helpers(csdev);
 		csdev->enable = false;
 	}
diff --git a/drivers/hwtracing/coresight/coresight-cti-core.c b/drivers/hwtracing/coresight/coresight-cti-core.c
index 277c890a1f1f..db7a2212ec18 100644
--- a/drivers/hwtracing/coresight/coresight-cti-core.c
+++ b/drivers/hwtracing/coresight/coresight-cti-core.c
@@ -555,7 +555,10 @@  static void cti_add_assoc_to_csdev(struct coresight_device *csdev)
 	mutex_lock(&ect_mutex);
 
 	/* exit if current is an ECT device.*/
-	if ((csdev->type == CORESIGHT_DEV_TYPE_ECT) || list_empty(&ect_net))
+	if ((csdev->type == CORESIGHT_DEV_TYPE_HELPER &&
+	     csdev->subtype.helper_subtype ==
+		     CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI) ||
+	    list_empty(&ect_net))
 		goto cti_add_done;
 
 	/* if we didn't find the csdev previously we used the fwnode name */
@@ -571,8 +574,7 @@  static void cti_add_assoc_to_csdev(struct coresight_device *csdev)
 			 * if we found a matching csdev then update the ECT
 			 * association pointer for the device with this CTI.
 			 */
-			coresight_set_assoc_ectdev_mutex(csdev,
-							 ect_item->csdev);
+			coresight_add_helper_mutex(csdev, ect_item->csdev);
 			break;
 		}
 	}
@@ -582,26 +584,30 @@  static void cti_add_assoc_to_csdev(struct coresight_device *csdev)
 
 /*
  * Removing the associated devices is easier.
- * A CTI will not have a value for csdev->ect_dev.
  */
 static void cti_remove_assoc_from_csdev(struct coresight_device *csdev)
 {
 	struct cti_drvdata *ctidrv;
 	struct cti_trig_con *tc;
+	union coresight_dev_subtype cti_subtype = {
+		.helper_subtype = CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
+	};
+	struct coresight_device *cti_csdev = coresight_find_output_type(
+		csdev->pdata, CORESIGHT_DEV_TYPE_HELPER, cti_subtype);
 	struct cti_device *ctidev;
 
+	if (!cti_csdev)
+		return;
+
 	mutex_lock(&ect_mutex);
-	if (csdev->ect_dev) {
-		ctidrv = csdev_to_cti_drvdata(csdev->ect_dev);
-		ctidev = &ctidrv->ctidev;
-		list_for_each_entry(tc, &ctidev->trig_cons, node) {
-			if (tc->con_dev == csdev) {
-				cti_remove_sysfs_link(ctidrv, tc);
-				tc->con_dev = NULL;
-				break;
-			}
+	ctidrv = csdev_to_cti_drvdata(cti_csdev);
+	ctidev = &ctidrv->ctidev;
+	list_for_each_entry(tc, &ctidev->trig_cons, node) {
+		if (tc->con_dev == csdev) {
+			cti_remove_sysfs_link(ctidrv, tc);
+			tc->con_dev = NULL;
+			break;
 		}
-		csdev->ect_dev = NULL;
 	}
 	mutex_unlock(&ect_mutex);
 }
@@ -630,8 +636,8 @@  static void cti_update_conn_xrefs(struct cti_drvdata *drvdata)
 			/* if we can set the sysfs link */
 			if (cti_add_sysfs_link(drvdata, tc))
 				/* set the CTI/csdev association */
-				coresight_set_assoc_ectdev_mutex(tc->con_dev,
-							 drvdata->csdev);
+				coresight_add_helper_mutex(tc->con_dev,
+							   drvdata->csdev);
 			else
 				/* otherwise remove reference from CTI */
 				tc->con_dev = NULL;
@@ -646,8 +652,6 @@  static void cti_remove_conn_xrefs(struct cti_drvdata *drvdata)
 
 	list_for_each_entry(tc, &ctidev->trig_cons, node) {
 		if (tc->con_dev) {
-			coresight_set_assoc_ectdev_mutex(tc->con_dev,
-							 NULL);
 			cti_remove_sysfs_link(drvdata, tc);
 			tc->con_dev = NULL;
 		}
@@ -795,27 +799,27 @@  static void cti_pm_release(struct cti_drvdata *drvdata)
 }
 
 /** cti ect operations **/
-int cti_enable(struct coresight_device *csdev)
+int cti_enable(struct coresight_device *csdev, enum cs_mode mode, void *data)
 {
 	struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
 
 	return cti_enable_hw(drvdata);
 }
 
-int cti_disable(struct coresight_device *csdev)
+int cti_disable(struct coresight_device *csdev, void *data)
 {
 	struct cti_drvdata *drvdata = csdev_to_cti_drvdata(csdev);
 
 	return cti_disable_hw(drvdata);
 }
 
-static const struct coresight_ops_ect cti_ops_ect = {
+static const struct coresight_ops_helper cti_ops_ect = {
 	.enable = cti_enable,
 	.disable = cti_disable,
 };
 
 static const struct coresight_ops cti_ops = {
-	.ect_ops = &cti_ops_ect,
+	.helper_ops = &cti_ops_ect,
 };
 
 /*
@@ -922,8 +926,8 @@  static int cti_probe(struct amba_device *adev, const struct amba_id *id)
 
 	/* set up coresight component description */
 	cti_desc.pdata = pdata;
-	cti_desc.type = CORESIGHT_DEV_TYPE_ECT;
-	cti_desc.subtype.ect_subtype = CORESIGHT_DEV_SUBTYPE_ECT_CTI;
+	cti_desc.type = CORESIGHT_DEV_TYPE_HELPER;
+	cti_desc.subtype.helper_subtype = CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI;
 	cti_desc.ops = &cti_ops;
 	cti_desc.groups = drvdata->ctidev.con_groups;
 	cti_desc.dev = dev;
diff --git a/drivers/hwtracing/coresight/coresight-cti-sysfs.c b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
index e528cff9d4e2..d25dd2737b49 100644
--- a/drivers/hwtracing/coresight/coresight-cti-sysfs.c
+++ b/drivers/hwtracing/coresight/coresight-cti-sysfs.c
@@ -112,11 +112,11 @@  static ssize_t enable_store(struct device *dev,
 		ret = pm_runtime_resume_and_get(dev->parent);
 		if (ret)
 			return ret;
-		ret = cti_enable(drvdata->csdev);
+		ret = cti_enable(drvdata->csdev, CS_MODE_SYSFS, NULL);
 		if (ret)
 			pm_runtime_put(dev->parent);
 	} else {
-		ret = cti_disable(drvdata->csdev);
+		ret = cti_disable(drvdata->csdev, NULL);
 		if (!ret)
 			pm_runtime_put(dev->parent);
 	}
diff --git a/drivers/hwtracing/coresight/coresight-cti.h b/drivers/hwtracing/coresight/coresight-cti.h
index 8b106b13a244..cb9ee616d01f 100644
--- a/drivers/hwtracing/coresight/coresight-cti.h
+++ b/drivers/hwtracing/coresight/coresight-cti.h
@@ -215,8 +215,8 @@  int cti_add_connection_entry(struct device *dev, struct cti_drvdata *drvdata,
 			     const char *assoc_dev_name);
 struct cti_trig_con *cti_allocate_trig_con(struct device *dev, int in_sigs,
 					   int out_sigs);
-int cti_enable(struct coresight_device *csdev);
-int cti_disable(struct coresight_device *csdev);
+int cti_enable(struct coresight_device *csdev, enum cs_mode mode, void *data);
+int cti_disable(struct coresight_device *csdev, void *data);
 void cti_write_all_hw_regs(struct cti_drvdata *drvdata);
 void cti_write_intack(struct device *dev, u32 ackval);
 void cti_write_single_reg(struct cti_drvdata *drvdata, int offset, u32 value);
diff --git a/drivers/hwtracing/coresight/coresight-priv.h b/drivers/hwtracing/coresight/coresight-priv.h
index a843f9d5c737..fff565d1cb42 100644
--- a/drivers/hwtracing/coresight/coresight-priv.h
+++ b/drivers/hwtracing/coresight/coresight-priv.h
@@ -211,8 +211,8 @@  void coresight_release_platform_data(struct coresight_device *csdev,
 				     struct coresight_platform_data *pdata);
 struct coresight_device *
 coresight_find_csdev_by_fwnode(struct fwnode_handle *r_fwnode);
-void coresight_set_assoc_ectdev_mutex(struct coresight_device *csdev,
-				      struct coresight_device *ect_csdev);
+void coresight_add_helper_mutex(struct coresight_device *csdev,
+				struct coresight_device *helper);
 
 void coresight_set_percpu_sink(int cpu, struct coresight_device *csdev);
 struct coresight_device *coresight_get_percpu_sink(int cpu);
diff --git a/drivers/hwtracing/coresight/coresight-sysfs.c b/drivers/hwtracing/coresight/coresight-sysfs.c
index 464ba5e1343b..dd78e9fcfc4d 100644
--- a/drivers/hwtracing/coresight/coresight-sysfs.c
+++ b/drivers/hwtracing/coresight/coresight-sysfs.c
@@ -148,6 +148,10 @@  int coresight_make_links(struct coresight_device *orig,
 	char *outs = NULL, *ins = NULL;
 	struct coresight_sysfs_link *link = NULL;
 
+	/* Helper devices aren't shown in sysfs */
+	if (conn->dest_port == -1 && conn->src_port == -1)
+		return 0;
+
 	do {
 		outs = devm_kasprintf(&orig->dev, GFP_KERNEL,
 				      "out:%d", conn->src_port);
diff --git a/include/linux/coresight.h b/include/linux/coresight.h
index d2739a0286f1..ed37552761e4 100644
--- a/include/linux/coresight.h
+++ b/include/linux/coresight.h
@@ -40,8 +40,7 @@  enum coresight_dev_type {
 	CORESIGHT_DEV_TYPE_LINK,
 	CORESIGHT_DEV_TYPE_LINKSINK,
 	CORESIGHT_DEV_TYPE_SOURCE,
-	CORESIGHT_DEV_TYPE_HELPER,
-	CORESIGHT_DEV_TYPE_ECT,
+	CORESIGHT_DEV_TYPE_HELPER
 };
 
 enum coresight_dev_subtype_sink {
@@ -66,12 +65,7 @@  enum coresight_dev_subtype_source {
 
 enum coresight_dev_subtype_helper {
 	CORESIGHT_DEV_SUBTYPE_HELPER_CATU,
-};
-
-/* Embedded Cross Trigger (ECT) sub-types */
-enum coresight_dev_subtype_ect {
-	CORESIGHT_DEV_SUBTYPE_ECT_NONE,
-	CORESIGHT_DEV_SUBTYPE_ECT_CTI,
+	CORESIGHT_DEV_SUBTYPE_HELPER_ECT_CTI
 };
 
 /**
@@ -84,8 +78,6 @@  enum coresight_dev_subtype_ect {
  *			by @coresight_dev_subtype_source.
  * @helper_subtype:	type of helper this component is, as defined
  *			by @coresight_dev_subtype_helper.
- * @ect_subtype:        type of cross trigger this component is, as
- *			defined by @coresight_dev_subtype_ect
  */
 union coresight_dev_subtype {
 	/* We have some devices which acts as LINK and SINK */
@@ -95,7 +87,6 @@  union coresight_dev_subtype {
 	};
 	enum coresight_dev_subtype_source source_subtype;
 	enum coresight_dev_subtype_helper helper_subtype;
-	enum coresight_dev_subtype_ect ect_subtype;
 };
 
 /**
@@ -237,8 +228,6 @@  struct coresight_sysfs_link {
  *		from source to that sink.
  * @ea:		Device attribute for sink representation under PMU directory.
  * @def_sink:	cached reference to default sink found for this device.
- * @ect_dev:	Associated cross trigger device. Not part of the trace data
- *		path or connections.
  * @nr_links:   number of sysfs links created to other components from this
  *		device. These will appear in the "connections" group.
  * @has_conns_grp: Have added a "connections" group for sysfs links.
@@ -261,12 +250,9 @@  struct coresight_device {
 	bool activated;	/* true only if a sink is part of a path */
 	struct dev_ext_attribute *ea;
 	struct coresight_device *def_sink;
-	/* cross trigger handling */
-	struct coresight_device *ect_dev;
 	/* sysfs links between components */
 	int nr_links;
 	bool has_conns_grp;
-	bool ect_enabled; /* true only if associated ect device is enabled */
 	/* system configuration and feature lists */
 	struct list_head feature_csdev_list;
 	struct list_head config_csdev_list;
@@ -378,23 +364,11 @@  struct coresight_ops_helper {
 	int (*disable)(struct coresight_device *csdev, void *data);
 };
 
-/**
- * struct coresight_ops_ect - Ops for an embedded cross trigger device
- *
- * @enable	: Enable the device
- * @disable	: Disable the device
- */
-struct coresight_ops_ect {
-	int (*enable)(struct coresight_device *csdev);
-	int (*disable)(struct coresight_device *csdev);
-};
-
 struct coresight_ops {
 	const struct coresight_ops_sink *sink_ops;
 	const struct coresight_ops_link *link_ops;
 	const struct coresight_ops_source *source_ops;
 	const struct coresight_ops_helper *helper_ops;
-	const struct coresight_ops_ect *ect_ops;
 };
 
 #if IS_ENABLED(CONFIG_CORESIGHT)