Commit 637e9e52 authored by Heikki Krogerus's avatar Heikki Krogerus Committed by Greg Kroah-Hartman

device connection: Find device connections also from device graphs

If connections between devices are described in OF graph or
ACPI device graph, we can find them by using the
fwnode_graph_*() functions.
Acked-by: default avatarHans de Goede <hdegoede@redhat.com>
Reviewed-by: default avatarAndy Shevchenko <andy.shevchenko@gmail.com>
Reviewed-by: default avatarJun Li <jun.li@nxp.com>
Tested-by: default avatarJun Li <jun.li@nxp.com>
Signed-off-by: default avatarHeikki Krogerus <heikki.krogerus@linux.intel.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 6a0bbcf9
...@@ -7,10 +7,37 @@ ...@@ -7,10 +7,37 @@
*/ */
#include <linux/device.h> #include <linux/device.h>
#include <linux/property.h>
static DEFINE_MUTEX(devcon_lock); static DEFINE_MUTEX(devcon_lock);
static LIST_HEAD(devcon_list); static LIST_HEAD(devcon_list);
typedef void *(*devcon_match_fn_t)(struct device_connection *con, int ep,
void *data);
static void *
fwnode_graph_devcon_match(struct fwnode_handle *fwnode, const char *con_id,
void *data, devcon_match_fn_t match)
{
struct device_connection con = { .id = con_id };
struct fwnode_handle *ep;
void *ret;
fwnode_graph_for_each_endpoint(fwnode, ep) {
con.fwnode = fwnode_graph_get_remote_port_parent(ep);
if (!fwnode_device_is_available(con.fwnode))
continue;
ret = match(&con, -1, data);
fwnode_handle_put(con.fwnode);
if (ret) {
fwnode_handle_put(ep);
return ret;
}
}
return NULL;
}
/** /**
* device_connection_find_match - Find physical connection to a device * device_connection_find_match - Find physical connection to a device
* @dev: Device with the connection * @dev: Device with the connection
...@@ -23,10 +50,9 @@ static LIST_HEAD(devcon_list); ...@@ -23,10 +50,9 @@ static LIST_HEAD(devcon_list);
* caller is expecting to be returned. * caller is expecting to be returned.
*/ */
void *device_connection_find_match(struct device *dev, const char *con_id, void *device_connection_find_match(struct device *dev, const char *con_id,
void *data, void *data, devcon_match_fn_t match)
void *(*match)(struct device_connection *con,
int ep, void *data))
{ {
struct fwnode_handle *fwnode = dev_fwnode(dev);
const char *devname = dev_name(dev); const char *devname = dev_name(dev);
struct device_connection *con; struct device_connection *con;
void *ret = NULL; void *ret = NULL;
...@@ -35,6 +61,12 @@ void *device_connection_find_match(struct device *dev, const char *con_id, ...@@ -35,6 +61,12 @@ void *device_connection_find_match(struct device *dev, const char *con_id,
if (!match) if (!match)
return NULL; return NULL;
if (fwnode) {
ret = fwnode_graph_devcon_match(fwnode, con_id, data, match);
if (ret)
return ret;
}
mutex_lock(&devcon_lock); mutex_lock(&devcon_lock);
list_for_each_entry(con, &devcon_list, list) { list_for_each_entry(con, &devcon_list, list) {
......
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