// 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; }