On Fri, May 22, 2020 at 11:57:36AM +0200, Mauro Carvalho Chehab wrote: > Em Thu, 21 May 2020 11:00:19 +0300 > Andy Shevchenko <andy.shevchenko@xxxxxxxxx> escreveu: > > > +Cc: Heikki (swnode expert) > > > > On Wed, May 20, 2020 at 2:19 PM Mauro Carvalho Chehab > > <mchehab+huawei@xxxxxxxxxx> wrote: > > > Em Wed, 20 May 2020 11:26:08 +0300 > > > Sakari Ailus <sakari.ailus@xxxxxxxxxxxxxxx> escreveu: > > > > ... > > > > > As I said, the problem is not probing the sensor via ACPI, but, instead, > > > to be able receive platform-specific data. > > > > There is no problem with swnodes, except missing parts (*). > > I have Skylake laptop with IPU3 and with half-baked ACPI tables, but > > since we have drivers in place with fwnode support, we only need to > > recreate fwnode graph in some board file to compensate the gap in > > ACPI. > > > > *) Missing part is graph support for swnodes. With that done it will > > be feasible to achieve the rest. > > I forgot if we have anything for this already done. Heikki? > > Hmm... I guess I should try this approach. I never heard about swnodes > before. Do you have already some patch with the needed swnodes setup, > and the missing parts to recreate the fwnode graph? Here you go. I tested it with this code: static const struct software_node nodes[]; static const struct property_entry ep0_props[] = { PROPERTY_ENTRY_REF("remote-endpoint", &nodes[5]), { } }; static const struct property_entry ep1_props[] = { PROPERTY_ENTRY_REF("remote-endpoint", &nodes[2]), { } }; static const struct software_node nodes[] = { { "dev0" }, { "port0", &nodes[0] }, { "endpoint", &nodes[1], ep0_props }, { "dev1" }, { "port0", &nodes[3] }, { "endpoint", &nodes[4], ep1_props }, { } }; void test(void) { const struct software_node *swnode; struct fwnode_handle *fwnode; software_node_register_nodes(nodes); fwnode = fwnode_graph_get_remote_port_parent(software_node_fwnode(&nodes[5])); swnode = to_software_node(fwnode); printk("first parent: %s\n", swnode->name); fwnode = fwnode_graph_get_remote_port_parent(software_node_fwnode(&nodes[2])); swnode = to_software_node(fwnode); printk("second parent: %s\n", swnode->name); software_node_unregister_nodes(nodes); } thanks, -- heikki
>From c6f8f2253b09e68bfb74a9110165f04fc2f50c51 Mon Sep 17 00:00:00 2001 From: Heikki Krogerus <heikki.krogerus@xxxxxxxxxxxxxxx> Date: Mon, 19 Aug 2019 11:09:41 +0300 Subject: [PATCH] software node: Add support for fwnode_graph* family of functions This implements the remaining .graph_* callbacks in the fwnode operations vector for the software nodes. That makes the fwnode_graph*() functions available in the drivers also when software nodes are used. The implementation tries to mimic the "OF graph" as much as possible, but there is no support for the "reg" device property. The ports will need to have the index in their name which starts with "port" (for example "port0", "port1", ...) and endpoints will use the index of the software node that is given to them during creation. The port nodes can also be grouped under a specially named "ports" subnode, just like in DT, if necessary. The remote-endpoints are reference properties under the endpoint nodes that are named "remote-endpoint". Signed-off-by: Heikki Krogerus <heikki.krogerus@xxxxxxxxxxxxxxx> --- drivers/base/swnode.c | 109 +++++++++++++++++++++++++++++++++++++++++- 1 file changed, 108 insertions(+), 1 deletion(-) diff --git a/drivers/base/swnode.c b/drivers/base/swnode.c index de8d3543e8fe3..7359b3f4e5daa 100644 --- a/drivers/base/swnode.c +++ b/drivers/base/swnode.c @@ -536,6 +536,108 @@ software_node_get_reference_args(const struct fwnode_handle *fwnode, return 0; } +static struct fwnode_handle * +swnode_graph_find_next_port(const struct fwnode_handle *parent, + struct fwnode_handle *port) +{ + struct fwnode_handle *old = port; + + while ((port = software_node_get_next_child(parent, old))) { + if (!strncmp(to_swnode(port)->node->name, "port", 4)) + return port; + fwnode_handle_put(old); + old = port; + } + + return NULL; +} + +static struct fwnode_handle * +software_node_graph_get_next_endpoint(const struct fwnode_handle *fwnode, + struct fwnode_handle *endpoint) +{ + struct swnode *swnode = to_swnode(fwnode); + struct fwnode_handle *old = endpoint; + struct fwnode_handle *parent; + struct fwnode_handle *port; + + if (!swnode) + return NULL; + + if (endpoint) { + port = software_node_get_parent(endpoint); + parent = software_node_get_parent(port); + } else { + parent = software_node_get_named_child_node(fwnode, "ports"); + if (!parent) + parent = software_node_get(&swnode->fwnode); + + port = swnode_graph_find_next_port(parent, NULL); + } + + for (; port; port = swnode_graph_find_next_port(parent, port)) { + endpoint = software_node_get_next_child(port, old); + fwnode_handle_put(old); + if (endpoint) + break; + } + + fwnode_handle_put(port); + software_node_put(parent); + + return endpoint; +} + +static struct fwnode_handle * +software_node_graph_get_remote_endpoint(const struct fwnode_handle *fwnode) +{ + struct swnode *swnode = to_swnode(fwnode); + const struct software_node_ref_args *ref; + const struct property_entry *prop; + + if (!swnode) + return NULL; + + prop = property_entry_get(swnode->node->properties, "remote-endpoint"); + if (!prop || prop->type != DEV_PROP_REF || prop->is_inline) + return NULL; + + ref = prop->pointer; + + return software_node_get(software_node_fwnode(ref[0].node)); +} + +static struct fwnode_handle * +software_node_graph_get_port_parent(struct fwnode_handle *fwnode) +{ + struct swnode *swnode = to_swnode(fwnode); + struct fwnode_handle *parent; + + if (!strcmp(swnode->parent->node->name, "ports")) + parent = &swnode->parent->parent->fwnode; + else + parent = &swnode->parent->fwnode; + + return software_node_get(parent); +} + +static int +software_node_graph_parse_endpoint(const struct fwnode_handle *fwnode, + struct fwnode_endpoint *endpoint) +{ + struct swnode *swnode = to_swnode(fwnode); + int ret; + + ret = kstrtou32(swnode->parent->node->name + 4, 10, &endpoint->port); + if (ret) + return ret; + + endpoint->id = swnode->id; + endpoint->local_fwnode = fwnode; + + return 0; +} + static const struct fwnode_operations software_node_ops = { .get = software_node_get, .put = software_node_put, @@ -547,7 +649,12 @@ static const struct fwnode_operations software_node_ops = { .get_parent = software_node_get_parent, .get_next_child_node = software_node_get_next_child, .get_named_child_node = software_node_get_named_child_node, - .get_reference_args = software_node_get_reference_args + .get_reference_args = software_node_get_reference_args, + + .graph_get_next_endpoint = software_node_graph_get_next_endpoint, + .graph_get_remote_endpoint = software_node_graph_get_remote_endpoint, + .graph_get_port_parent = software_node_graph_get_port_parent, + .graph_parse_endpoint = software_node_graph_parse_endpoint, }; /* -------------------------------------------------------------------------- */ -- 2.26.2