mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 d735d925f9
			
		
	
	
		d735d925f9
		
	
	
	
	
		
			
			Export symbols used among coresight modules. Tested-by: Mike Leach <mike.leach@linaro.org> Reviewed-by: Suzuki K Poulose <suzuki.poulose@arm.com> Signed-off-by: Mian Yousaf Kaukab <ykaukab@suse.de> Signed-off-by: Tingwei Zhang <tingwei@codeaurora.org> Signed-off-by: Mathieu Poirier <mathieu.poirier@linaro.org> Link: https://lore.kernel.org/r/20200928163513.70169-6-mathieu.poirier@linaro.org Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
		
			
				
	
	
		
			207 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			207 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Copyright (c) 2019, Linaro Limited, All rights reserved.
 | |
|  * Author: Mike Leach <mike.leach@linaro.org>
 | |
|  */
 | |
| 
 | |
| #include <linux/device.h>
 | |
| #include <linux/kernel.h>
 | |
| 
 | |
| #include "coresight-priv.h"
 | |
| 
 | |
| /*
 | |
|  * Connections group - links attribute.
 | |
|  * Count of created links between coresight components in the group.
 | |
|  */
 | |
| static ssize_t nr_links_show(struct device *dev,
 | |
| 			     struct device_attribute *attr,
 | |
| 			     char *buf)
 | |
| {
 | |
| 	struct coresight_device *csdev = to_coresight_device(dev);
 | |
| 
 | |
| 	return sprintf(buf, "%d\n", csdev->nr_links);
 | |
| }
 | |
| static DEVICE_ATTR_RO(nr_links);
 | |
| 
 | |
| static struct attribute *coresight_conns_attrs[] = {
 | |
| 	&dev_attr_nr_links.attr,
 | |
| 	NULL,
 | |
| };
 | |
| 
 | |
| static struct attribute_group coresight_conns_group = {
 | |
| 	.attrs = coresight_conns_attrs,
 | |
| 	.name = "connections",
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * Create connections group for CoreSight devices.
 | |
|  * This group will then be used to collate the sysfs links between
 | |
|  * devices.
 | |
|  */
 | |
| int coresight_create_conns_sysfs_group(struct coresight_device *csdev)
 | |
| {
 | |
| 	int ret = 0;
 | |
| 
 | |
| 	if (!csdev)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	ret = sysfs_create_group(&csdev->dev.kobj, &coresight_conns_group);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	csdev->has_conns_grp = true;
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| void coresight_remove_conns_sysfs_group(struct coresight_device *csdev)
 | |
| {
 | |
| 	if (!csdev)
 | |
| 		return;
 | |
| 
 | |
| 	if (csdev->has_conns_grp) {
 | |
| 		sysfs_remove_group(&csdev->dev.kobj, &coresight_conns_group);
 | |
| 		csdev->has_conns_grp = false;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int coresight_add_sysfs_link(struct coresight_sysfs_link *info)
 | |
| {
 | |
| 	int ret = 0;
 | |
| 
 | |
| 	if (!info)
 | |
| 		return -EINVAL;
 | |
| 	if (!info->orig || !info->target ||
 | |
| 	    !info->orig_name || !info->target_name)
 | |
| 		return -EINVAL;
 | |
| 	if (!info->orig->has_conns_grp || !info->target->has_conns_grp)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	/* first link orig->target */
 | |
| 	ret = sysfs_add_link_to_group(&info->orig->dev.kobj,
 | |
| 				      coresight_conns_group.name,
 | |
| 				      &info->target->dev.kobj,
 | |
| 				      info->orig_name);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	/* second link target->orig */
 | |
| 	ret = sysfs_add_link_to_group(&info->target->dev.kobj,
 | |
| 				      coresight_conns_group.name,
 | |
| 				      &info->orig->dev.kobj,
 | |
| 				      info->target_name);
 | |
| 
 | |
| 	/* error in second link - remove first - otherwise inc counts */
 | |
| 	if (ret) {
 | |
| 		sysfs_remove_link_from_group(&info->orig->dev.kobj,
 | |
| 					     coresight_conns_group.name,
 | |
| 					     info->orig_name);
 | |
| 	} else {
 | |
| 		info->orig->nr_links++;
 | |
| 		info->target->nr_links++;
 | |
| 	}
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(coresight_add_sysfs_link);
 | |
| 
 | |
| void coresight_remove_sysfs_link(struct coresight_sysfs_link *info)
 | |
| {
 | |
| 	if (!info)
 | |
| 		return;
 | |
| 	if (!info->orig || !info->target ||
 | |
| 	    !info->orig_name || !info->target_name)
 | |
| 		return;
 | |
| 
 | |
| 	sysfs_remove_link_from_group(&info->orig->dev.kobj,
 | |
| 				     coresight_conns_group.name,
 | |
| 				     info->orig_name);
 | |
| 
 | |
| 	sysfs_remove_link_from_group(&info->target->dev.kobj,
 | |
| 				     coresight_conns_group.name,
 | |
| 				     info->target_name);
 | |
| 
 | |
| 	info->orig->nr_links--;
 | |
| 	info->target->nr_links--;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(coresight_remove_sysfs_link);
 | |
| 
 | |
| /*
 | |
|  * coresight_make_links: Make a link for a connection from a @orig
 | |
|  * device to @target, represented by @conn.
 | |
|  *
 | |
|  *   e.g, for devOrig[output_X] -> devTarget[input_Y] is represented
 | |
|  *   as two symbolic links :
 | |
|  *
 | |
|  *	/sys/.../devOrig/out:X	-> /sys/.../devTarget/
 | |
|  *	/sys/.../devTarget/in:Y	-> /sys/.../devOrig/
 | |
|  *
 | |
|  * The link names are allocated for a device where it appears. i.e, the
 | |
|  * "out" link on the master and "in" link on the slave device.
 | |
|  * The link info is stored in the connection record for avoiding
 | |
|  * the reconstruction of names for removal.
 | |
|  */
 | |
| int coresight_make_links(struct coresight_device *orig,
 | |
| 			 struct coresight_connection *conn,
 | |
| 			 struct coresight_device *target)
 | |
| {
 | |
| 	int ret = -ENOMEM;
 | |
| 	char *outs = NULL, *ins = NULL;
 | |
| 	struct coresight_sysfs_link *link = NULL;
 | |
| 
 | |
| 	do {
 | |
| 		outs = devm_kasprintf(&orig->dev, GFP_KERNEL,
 | |
| 				      "out:%d", conn->outport);
 | |
| 		if (!outs)
 | |
| 			break;
 | |
| 		ins = devm_kasprintf(&target->dev, GFP_KERNEL,
 | |
| 				     "in:%d", conn->child_port);
 | |
| 		if (!ins)
 | |
| 			break;
 | |
| 		link = devm_kzalloc(&orig->dev,
 | |
| 				    sizeof(struct coresight_sysfs_link),
 | |
| 				    GFP_KERNEL);
 | |
| 		if (!link)
 | |
| 			break;
 | |
| 
 | |
| 		link->orig = orig;
 | |
| 		link->target = target;
 | |
| 		link->orig_name = outs;
 | |
| 		link->target_name = ins;
 | |
| 
 | |
| 		ret = coresight_add_sysfs_link(link);
 | |
| 		if (ret)
 | |
| 			break;
 | |
| 
 | |
| 		conn->link = link;
 | |
| 
 | |
| 		/*
 | |
| 		 * Install the device connection. This also indicates that
 | |
| 		 * the links are operational on both ends.
 | |
| 		 */
 | |
| 		conn->child_dev = target;
 | |
| 		return 0;
 | |
| 	} while (0);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * coresight_remove_links: Remove the sysfs links for a given connection @conn,
 | |
|  * from @orig device to @target device. See coresight_make_links() for more
 | |
|  * details.
 | |
|  */
 | |
| void coresight_remove_links(struct coresight_device *orig,
 | |
| 			    struct coresight_connection *conn)
 | |
| {
 | |
| 	if (!orig || !conn->link)
 | |
| 		return;
 | |
| 
 | |
| 	coresight_remove_sysfs_link(conn->link);
 | |
| 
 | |
| 	devm_kfree(&conn->child_dev->dev, conn->link->target_name);
 | |
| 	devm_kfree(&orig->dev, conn->link->orig_name);
 | |
| 	devm_kfree(&orig->dev, conn->link);
 | |
| 	conn->link = NULL;
 | |
| 	conn->child_dev = NULL;
 | |
| }
 |