aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPali Rohár <pali@kernel.org>2021-12-28 20:11:27 +0100
committerMartin Mares <mj@ucw.cz>2022-02-10 13:49:35 +0100
commite9e7fab13d16067658f02d47094a949f56aef596 (patch)
tree5bf0843294b4fe5286836513fd9f8e9ee543524e
parent5e9714acb554052f45185fb1cc8246c864ef76dd (diff)
downloadpciutils-e9e7fab13d16067658f02d47094a949f56aef596.tar.gz
libpci: sysfs: Implement support for PCI_FILL_PARENT
-rw-r--r--lib/sysfs.c33
1 files changed, 33 insertions, 0 deletions
diff --git a/lib/sysfs.c b/lib/sysfs.c
index db54110..735c144 100644
--- a/lib/sysfs.c
+++ b/lib/sysfs.c
@@ -381,6 +381,39 @@ sysfs_fill_info(struct pci_dev *d, unsigned int flags)
d->irq = sysfs_get_value(d, "irq", 1);
if (want_fill(d, flags, PCI_FILL_BASES | PCI_FILL_ROM_BASE | PCI_FILL_SIZES | PCI_FILL_IO_FLAGS | PCI_FILL_BRIDGE_BASES))
sysfs_get_resources(d);
+ if (want_fill(d, flags, PCI_FILL_PARENT))
+ {
+ unsigned int domain, bus, dev, func;
+ char *path_abs, *path_canon, *name;
+ char path_rel[OBJNAMELEN];
+ struct pci_dev *parent;
+
+ /* Construct sysfs path for parent device */
+ sysfs_obj_name(d, "..", path_rel);
+ path_abs = realpath(path_rel, NULL);
+ name = path_abs ? strrchr(path_abs, '/') : NULL;
+ name = name ? name+1 : name;
+ parent = NULL;
+
+ if (name && sscanf(name, "%x:%x:%x.%d", &domain, &bus, &dev, &func) == 4 && domain <= 0x7fffffff)
+ for (parent = d->access->devices; parent; parent = parent->next)
+ if (parent->domain == (int)domain && parent->bus == bus && parent->dev == dev && parent->func == func)
+ break;
+
+ if (parent)
+ {
+ /* Check if parsed BDF address from parent sysfs device is really expected PCI device */
+ sysfs_obj_name(parent, ".", path_rel);
+ path_canon = realpath(path_rel, NULL);
+ if (!path_canon || strcmp(path_canon, path_abs) != 0)
+ parent = NULL;
+ }
+
+ if (parent)
+ d->parent = parent;
+ else
+ clear_fill(d, PCI_FILL_PARENT);
+ }
}
if (want_fill(d, flags, PCI_FILL_PHYS_SLOT))