2011-01-24 12:28:55 +08:00
|
|
|
#include <linux/kernel.h>
|
2011-05-28 05:06:52 +08:00
|
|
|
#include <linux/export.h>
|
2011-07-24 13:52:48 +08:00
|
|
|
#include <linux/of.h>
|
2011-01-24 12:28:55 +08:00
|
|
|
#include <linux/of_pci.h>
|
|
|
|
#include <asm/prom.h>
|
|
|
|
|
2011-04-11 09:37:07 +08:00
|
|
|
static inline int __of_pci_pci_compare(struct device_node *node,
|
|
|
|
unsigned int devfn)
|
2011-01-24 12:28:55 +08:00
|
|
|
{
|
2011-04-11 09:37:07 +08:00
|
|
|
unsigned int size;
|
|
|
|
const __be32 *reg = of_get_property(node, "reg", &size);
|
2011-01-24 12:28:55 +08:00
|
|
|
|
2011-04-11 09:37:07 +08:00
|
|
|
if (!reg || size < 5 * sizeof(__be32))
|
|
|
|
return 0;
|
|
|
|
return ((be32_to_cpup(®[0]) >> 8) & 0xff) == devfn;
|
|
|
|
}
|
2011-01-24 12:28:55 +08:00
|
|
|
|
2011-04-11 09:37:07 +08:00
|
|
|
struct device_node *of_pci_find_child_device(struct device_node *parent,
|
|
|
|
unsigned int devfn)
|
|
|
|
{
|
|
|
|
struct device_node *node, *node2;
|
|
|
|
|
|
|
|
for_each_child_of_node(parent, node) {
|
|
|
|
if (__of_pci_pci_compare(node, devfn))
|
|
|
|
return node;
|
|
|
|
/*
|
|
|
|
* Some OFs create a parent node "multifunc-device" as
|
|
|
|
* a fake root for all functions of a multi-function
|
|
|
|
* device we go down them as well.
|
2011-01-24 12:28:55 +08:00
|
|
|
*/
|
2011-04-11 09:37:07 +08:00
|
|
|
if (!strcmp(node->name, "multifunc-device")) {
|
|
|
|
for_each_child_of_node(node, node2) {
|
|
|
|
if (__of_pci_pci_compare(node2, devfn)) {
|
|
|
|
of_node_put(node);
|
|
|
|
return node2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-01-24 12:28:55 +08:00
|
|
|
}
|
2011-04-11 09:37:07 +08:00
|
|
|
return NULL;
|
2011-01-24 12:28:55 +08:00
|
|
|
}
|
2011-04-11 09:37:07 +08:00
|
|
|
EXPORT_SYMBOL_GPL(of_pci_find_child_device);
|