/* * Copyright (C) 2013 Texas Instruments * Author: Tomi Valkeinen <tomi.valkeinen@ti.com> * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 as published by * the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. */ #include <linux/device.h> #include <linux/err.h> #include <linux/module.h> #include <linux/of.h> #include <linux/seq_file.h> #include <video/omapdss.h> struct device_node * omapdss_of_get_next_port(const struct device_node *parent, struct device_node *prev) { struct device_node *port = NULL; if (!parent) return NULL; if (!prev) { struct device_node *ports; /* * It's the first call, we have to find a port subnode * within this node or within an optional 'ports' node. */ ports = of_get_child_by_name(parent, "ports"); if (ports) parent = ports; port = of_get_child_by_name(parent, "port"); /* release the 'ports' node */ of_node_put(ports); } else { struct device_node *ports; ports = of_get_parent(prev); if (!ports) return NULL; do { port = of_get_next_child(ports, prev); if (!port) { of_node_put(ports); return NULL; } prev = port; } while (of_node_cmp(port->name, "port") != 0); } return port; } EXPORT_SYMBOL_GPL(omapdss_of_get_next_port); struct device_node * omapdss_of_get_next_endpoint(const struct device_node *parent, struct device_node *prev) { struct device_node *ep = NULL; if (!parent) return NULL; do { ep = of_get_next_child(parent, prev); if (!ep) return NULL; prev = ep; } while (of_node_cmp(ep->name, "endpoint") != 0); return ep; } EXPORT_SYMBOL_GPL(omapdss_of_get_next_endpoint); static struct device_node * omapdss_of_get_remote_device_node(const struct device_node *node) { struct device_node *np; int i; np = of_parse_phandle(node, "remote-endpoint", 0); if (!np) return NULL; np = of_get_next_parent(np); for (i = 0; i < 3 && np; ++i) { struct property *prop; prop = of_find_property(np, "compatible", NULL); if (prop) return np; np = of_get_next_parent(np); } return NULL; } struct device_node * omapdss_of_get_first_endpoint(const struct device_node *parent) { struct device_node *port, *ep; port = omapdss_of_get_next_port(parent, NULL); if (!port) return NULL; ep = omapdss_of_get_next_endpoint(port, NULL); of_node_put(port); return ep; } EXPORT_SYMBOL_GPL(omapdss_of_get_first_endpoint); struct omap_dss_device * omapdss_of_find_source_for_first_ep(struct device_node *node) { struct device_node *ep; struct device_node *src_node; struct omap_dss_device *src; ep = omapdss_of_get_first_endpoint(node); if (!ep) return ERR_PTR(-EINVAL); src_node = omapdss_of_get_remote_device_node(ep); of_node_put(ep); if (!src_node) return ERR_PTR(-EINVAL); src = omap_dss_find_output_by_node(src_node); of_node_put(src_node); if (!src) return ERR_PTR(-EPROBE_DEFER); return src; } EXPORT_SYMBOL_GPL(omapdss_of_find_source_for_first_ep);