Commit b3e94405 authored by Mathieu Poirier's avatar Mathieu Poirier Committed by Greg Kroah-Hartman

coresight: associating path with session rather than tracer

When using the Coresight framework from the sysFS interface a
tracer is always handling a single session and as such, a path
can be associated with a tracer.  But when supporting multiple
session per tracer there is no guarantee that sessions will always
have the same path from source to sink.

This patch is removing the automatic association between path and
tracers.  The building of a path and enablement of the components
in the path are decoupled, allowing for the association of a path
with a session rather than a tracer.

To keep backward functionality with the current sysFS access methods
a per-cpu place holder is used to keep a handle on the path built when
tracers are enabled.  Lastly APIs to build paths and enable tracers are
made public so that other subsystem can interact with the Coresight
framework.
Signed-off-by: default avatarMathieu Poirier <mathieu.poirier@linaro.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 61390593
...@@ -52,6 +52,11 @@ static inline void CS_UNLOCK(void __iomem *addr) ...@@ -52,6 +52,11 @@ static inline void CS_UNLOCK(void __iomem *addr)
} while (0); } while (0);
} }
void coresight_disable_path(struct list_head *path);
int coresight_enable_path(struct list_head *path);
struct list_head *coresight_build_path(struct coresight_device *csdev);
void coresight_release_path(struct list_head *path);
#ifdef CONFIG_CORESIGHT_SOURCE_ETM3X #ifdef CONFIG_CORESIGHT_SOURCE_ETM3X
extern int etm_readl_cp14(u32 off, unsigned int *val); extern int etm_readl_cp14(u32 off, unsigned int *val);
extern int etm_writel_cp14(u32 off, u32 val); extern int etm_writel_cp14(u32 off, u32 val);
......
...@@ -29,6 +29,22 @@ ...@@ -29,6 +29,22 @@
static DEFINE_MUTEX(coresight_mutex); static DEFINE_MUTEX(coresight_mutex);
/**
* struct coresight_node - elements of a path, from source to sink
* @csdev: Address of an element.
* @link: hook to the list.
*/
struct coresight_node {
struct coresight_device *csdev;
struct list_head link;
};
/*
* When operating Coresight drivers from the sysFS interface, only a single
* path can exist from a tracer (associated to a CPU) to a sink.
*/
static DEFINE_PER_CPU(struct list_head *, sysfs_path);
static int coresight_id_match(struct device *dev, void *data) static int coresight_id_match(struct device *dev, void *data)
{ {
int trace_id, i_trace_id; int trace_id, i_trace_id;
...@@ -68,15 +84,12 @@ static int coresight_source_is_unique(struct coresight_device *csdev) ...@@ -68,15 +84,12 @@ static int coresight_source_is_unique(struct coresight_device *csdev)
csdev, coresight_id_match); csdev, coresight_id_match);
} }
static int coresight_find_link_inport(struct coresight_device *csdev) static int coresight_find_link_inport(struct coresight_device *csdev,
struct coresight_device *parent)
{ {
int i; int i;
struct coresight_device *parent;
struct coresight_connection *conn; struct coresight_connection *conn;
parent = container_of(csdev->path_link.next,
struct coresight_device, path_link);
for (i = 0; i < parent->nr_outport; i++) { for (i = 0; i < parent->nr_outport; i++) {
conn = &parent->conns[i]; conn = &parent->conns[i];
if (conn->child_dev == csdev) if (conn->child_dev == csdev)
...@@ -89,15 +102,12 @@ static int coresight_find_link_inport(struct coresight_device *csdev) ...@@ -89,15 +102,12 @@ static int coresight_find_link_inport(struct coresight_device *csdev)
return 0; return 0;
} }
static int coresight_find_link_outport(struct coresight_device *csdev) static int coresight_find_link_outport(struct coresight_device *csdev,
struct coresight_device *child)
{ {
int i; int i;
struct coresight_device *child;
struct coresight_connection *conn; struct coresight_connection *conn;
child = container_of(csdev->path_link.prev,
struct coresight_device, path_link);
for (i = 0; i < csdev->nr_outport; i++) { for (i = 0; i < csdev->nr_outport; i++) {
conn = &csdev->conns[i]; conn = &csdev->conns[i];
if (conn->child_dev == child) if (conn->child_dev == child)
...@@ -138,14 +148,19 @@ static void coresight_disable_sink(struct coresight_device *csdev) ...@@ -138,14 +148,19 @@ static void coresight_disable_sink(struct coresight_device *csdev)
} }
} }
static int coresight_enable_link(struct coresight_device *csdev) static int coresight_enable_link(struct coresight_device *csdev,
struct coresight_device *parent,
struct coresight_device *child)
{ {
int ret; int ret;
int link_subtype; int link_subtype;
int refport, inport, outport; int refport, inport, outport;
inport = coresight_find_link_inport(csdev); if (!parent || !child)
outport = coresight_find_link_outport(csdev); return -EINVAL;
inport = coresight_find_link_inport(csdev, parent);
outport = coresight_find_link_outport(csdev, child);
link_subtype = csdev->subtype.link_subtype; link_subtype = csdev->subtype.link_subtype;
if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG)
...@@ -168,14 +183,19 @@ static int coresight_enable_link(struct coresight_device *csdev) ...@@ -168,14 +183,19 @@ static int coresight_enable_link(struct coresight_device *csdev)
return 0; return 0;
} }
static void coresight_disable_link(struct coresight_device *csdev) static void coresight_disable_link(struct coresight_device *csdev,
struct coresight_device *parent,
struct coresight_device *child)
{ {
int i, nr_conns; int i, nr_conns;
int link_subtype; int link_subtype;
int refport, inport, outport; int refport, inport, outport;
inport = coresight_find_link_inport(csdev); if (!parent || !child)
outport = coresight_find_link_outport(csdev); return;
inport = coresight_find_link_inport(csdev, parent);
outport = coresight_find_link_outport(csdev, child);
link_subtype = csdev->subtype.link_subtype; link_subtype = csdev->subtype.link_subtype;
if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) { if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) {
...@@ -235,109 +255,167 @@ static void coresight_disable_source(struct coresight_device *csdev) ...@@ -235,109 +255,167 @@ static void coresight_disable_source(struct coresight_device *csdev)
} }
} }
static int coresight_enable_path(struct list_head *path) void coresight_disable_path(struct list_head *path)
{ {
int ret = 0; struct coresight_node *nd;
struct coresight_device *cd; struct coresight_device *csdev, *parent, *child;
/* list_for_each_entry(nd, path, link) {
* At this point we have a full @path, from source to sink. The csdev = nd->csdev;
* sink is the first entry and the source the last one. Go through
* all the components and enable them one by one. switch (csdev->type) {
*/ case CORESIGHT_DEV_TYPE_SINK:
list_for_each_entry(cd, path, path_link) { case CORESIGHT_DEV_TYPE_LINKSINK:
if (cd == list_first_entry(path, struct coresight_device, coresight_disable_sink(csdev);
path_link)) { break;
ret = coresight_enable_sink(cd); case CORESIGHT_DEV_TYPE_SOURCE:
} else if (list_is_last(&cd->path_link, path)) { /* sources are disabled from either sysFS or Perf */
/* break;
* Don't enable the source just yet - this needs to case CORESIGHT_DEV_TYPE_LINK:
* happen at the very end when all links and sink parent = list_prev_entry(nd, link)->csdev;
* along the path have been configured properly. child = list_next_entry(nd, link)->csdev;
*/ coresight_disable_link(csdev, parent, child);
; break;
} else { default:
ret = coresight_enable_link(cd); break;
} }
if (ret)
goto err;
} }
}
return 0; int coresight_enable_path(struct list_head *path)
err: {
list_for_each_entry_continue_reverse(cd, path, path_link) {
if (cd == list_first_entry(path, struct coresight_device, int ret = 0;
path_link)) { struct coresight_node *nd;
coresight_disable_sink(cd); struct coresight_device *csdev, *parent, *child;
} else if (list_is_last(&cd->path_link, path)) {
; list_for_each_entry_reverse(nd, path, link) {
} else { csdev = nd->csdev;
coresight_disable_link(cd);
switch (csdev->type) {
case CORESIGHT_DEV_TYPE_SINK:
case CORESIGHT_DEV_TYPE_LINKSINK:
ret = coresight_enable_sink(csdev);
if (ret)
goto err;
break;
case CORESIGHT_DEV_TYPE_SOURCE:
/* sources are enabled from either sysFS or Perf */
break;
case CORESIGHT_DEV_TYPE_LINK:
parent = list_prev_entry(nd, link)->csdev;
child = list_next_entry(nd, link)->csdev;
ret = coresight_enable_link(csdev, parent, child);
if (ret)
goto err;
break;
default:
goto err;
} }
} }
out:
return ret; return ret;
err:
coresight_disable_path(path);
goto out;
} }
static int coresight_disable_path(struct list_head *path) /**
* _coresight_build_path - recursively build a path from a @csdev to a sink.
* @csdev: The device to start from.
* @path: The list to add devices to.
*
* The tree of Coresight device is traversed until an activated sink is
* found. From there the sink is added to the list along with all the
* devices that led to that point - the end result is a list from source
* to sink. In that list the source is the first device and the sink the
* last one.
*/
static int _coresight_build_path(struct coresight_device *csdev,
struct list_head *path)
{ {
struct coresight_device *cd; int i;
bool found = false;
struct coresight_node *node;
struct coresight_connection *conn;
list_for_each_entry_reverse(cd, path, path_link) { /* An activated sink has been found. Enqueue the element */
if (cd == list_first_entry(path, struct coresight_device, if ((csdev->type == CORESIGHT_DEV_TYPE_SINK ||
path_link)) { csdev->type == CORESIGHT_DEV_TYPE_LINKSINK) && csdev->activated)
coresight_disable_sink(cd); goto out;
} else if (list_is_last(&cd->path_link, path)) {
/* /* Not a sink - recursively explore each port found on this element */
* The source has already been stopped, no need for (i = 0; i < csdev->nr_outport; i++) {
* to do it again here. conn = &csdev->conns[i];
*/ if (_coresight_build_path(conn->child_dev, path) == 0) {
; found = true;
} else { break;
coresight_disable_link(cd);
} }
} }
if (!found)
return -ENODEV;
out:
/*
* A path from this element to a sink has been found. The elements
* leading to the sink are already enqueued, all that is left to do
* is add a node for this element.
*/
node = kzalloc(sizeof(struct coresight_node), GFP_KERNEL);
if (!node)
return -ENOMEM;
node->csdev = csdev;
list_add(&node->link, path);
return 0; return 0;
} }
static int coresight_build_paths(struct coresight_device *csdev, struct list_head *coresight_build_path(struct coresight_device *csdev)
struct list_head *path,
bool enable)
{ {
int i, ret = -EINVAL; struct list_head *path;
struct coresight_connection *conn;
list_add(&csdev->path_link, path); path = kzalloc(sizeof(struct list_head), GFP_KERNEL);
if (!path)
return NULL;
if ((csdev->type == CORESIGHT_DEV_TYPE_SINK || INIT_LIST_HEAD(path);
csdev->type == CORESIGHT_DEV_TYPE_LINKSINK) &&
csdev->activated) { if (_coresight_build_path(csdev, path)) {
if (enable) kfree(path);
ret = coresight_enable_path(path); path = NULL;
else
ret = coresight_disable_path(path);
} else {
for (i = 0; i < csdev->nr_outport; i++) {
conn = &csdev->conns[i];
if (coresight_build_paths(conn->child_dev,
path, enable) == 0)
ret = 0;
}
} }
if (list_first_entry(path, struct coresight_device, path_link) != csdev) return path;
dev_err(&csdev->dev, "wrong device in %s\n", __func__); }
/**
* coresight_release_path - release a previously built path.
* @path: the path to release.
*
* Go through all the elements of a path and 1) removed it from the list and
* 2) free the memory allocated for each node.
*/
void coresight_release_path(struct list_head *path)
{
struct coresight_node *nd, *next;
list_del(&csdev->path_link); list_for_each_entry_safe(nd, next, path, link) {
list_del(&nd->link);
kfree(nd);
}
return ret; kfree(path);
path = NULL;
} }
int coresight_enable(struct coresight_device *csdev) int coresight_enable(struct coresight_device *csdev)
{ {
int ret = 0; int ret = 0;
LIST_HEAD(path); int cpu;
struct list_head *path;
mutex_lock(&coresight_mutex); mutex_lock(&coresight_mutex);
if (csdev->type != CORESIGHT_DEV_TYPE_SOURCE) { if (csdev->type != CORESIGHT_DEV_TYPE_SOURCE) {
...@@ -348,22 +426,47 @@ int coresight_enable(struct coresight_device *csdev) ...@@ -348,22 +426,47 @@ int coresight_enable(struct coresight_device *csdev)
if (csdev->enable) if (csdev->enable)
goto out; goto out;
if (coresight_build_paths(csdev, &path, true)) { path = coresight_build_path(csdev);
dev_err(&csdev->dev, "building path(s) failed\n"); if (!path) {
pr_err("building path(s) failed\n");
goto out; goto out;
} }
if (coresight_enable_source(csdev)) ret = coresight_enable_path(path);
dev_err(&csdev->dev, "source enable failed\n"); if (ret)
goto err_path;
ret = coresight_enable_source(csdev);
if (ret)
goto err_source;
/*
* When working from sysFS it is important to keep track
* of the paths that were created so that they can be
* undone in 'coresight_disable()'. Since there can only
* be a single session per tracer (when working from sysFS)
* a per-cpu variable will do just fine.
*/
cpu = source_ops(csdev)->cpu_id(csdev);
per_cpu(sysfs_path, cpu) = path;
out: out:
mutex_unlock(&coresight_mutex); mutex_unlock(&coresight_mutex);
return ret; return ret;
err_source:
coresight_disable_path(path);
err_path:
coresight_release_path(path);
goto out;
} }
EXPORT_SYMBOL_GPL(coresight_enable); EXPORT_SYMBOL_GPL(coresight_enable);
void coresight_disable(struct coresight_device *csdev) void coresight_disable(struct coresight_device *csdev)
{ {
LIST_HEAD(path); int cpu;
struct list_head *path;
mutex_lock(&coresight_mutex); mutex_lock(&coresight_mutex);
if (csdev->type != CORESIGHT_DEV_TYPE_SOURCE) { if (csdev->type != CORESIGHT_DEV_TYPE_SOURCE) {
...@@ -373,9 +476,12 @@ void coresight_disable(struct coresight_device *csdev) ...@@ -373,9 +476,12 @@ void coresight_disable(struct coresight_device *csdev)
if (!csdev->enable) if (!csdev->enable)
goto out; goto out;
cpu = source_ops(csdev)->cpu_id(csdev);
path = per_cpu(sysfs_path, cpu);
coresight_disable_source(csdev); coresight_disable_source(csdev);
if (coresight_build_paths(csdev, &path, false)) coresight_disable_path(path);
dev_err(&csdev->dev, "releasing path(s) failed\n"); coresight_release_path(path);
per_cpu(sysfs_path, cpu) = NULL;
out: out:
mutex_unlock(&coresight_mutex); mutex_unlock(&coresight_mutex);
......
...@@ -152,7 +152,6 @@ struct coresight_connection { ...@@ -152,7 +152,6 @@ struct coresight_connection {
by @coresight_ops. by @coresight_ops.
* @dev: The device entity associated to this component. * @dev: The device entity associated to this component.
* @refcnt: keep track of what is in use. * @refcnt: keep track of what is in use.
* @path_link: link of current component into the path being enabled.
* @orphan: true if the component has connections that haven't been linked. * @orphan: true if the component has connections that haven't been linked.
* @enable: 'true' if component is currently part of an active path. * @enable: 'true' if component is currently part of an active path.
* @activated: 'true' only if a _sink_ has been activated. A sink can be * @activated: 'true' only if a _sink_ has been activated. A sink can be
...@@ -168,7 +167,6 @@ struct coresight_device { ...@@ -168,7 +167,6 @@ struct coresight_device {
const struct coresight_ops *ops; const struct coresight_ops *ops;
struct device dev; struct device dev;
atomic_t *refcnt; atomic_t *refcnt;
struct list_head path_link;
bool orphan; bool orphan;
bool enable; /* true only if configured as part of a path */ bool enable; /* true only if configured as part of a path */
bool activated; /* true only if a sink is part of a path */ bool activated; /* true only if a sink is part of a path */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment