@@ -384,3 +384,28 @@ int pci_dev_present(const struct pci_device_id *ids)
return 0;
}
EXPORT_SYMBOL(pci_dev_present);
+
+/**
+ * pci_find_common_upstream_dev - Returns the first common upstream device
+ * @dev: the PCI device from which the bus hierarchy walk will start
+ * @other: the PCI device whose bus number will be used for the search
+ *
+ * Walks up the bus hierarchy from the @dev device, looking for the first bus
+ * which the @other device is downstream of. Returns %NULL if the devices do
+ * not share any upstream devices.
+ */
+struct pci_dev *pci_find_common_upstream_dev(struct pci_dev *dev,
+ struct pci_dev *other)
+{
+ struct pci_dev *pdev = dev;
+
+ while (pdev != NULL) {
+ if ((other->bus->number >= pdev->bus->busn_res.start) &&
+ (other->bus->number <= pdev->bus->busn_res.end))
+ return pdev;
+ pdev = pci_upstream_bridge(pdev);
+ }
+
+ return NULL;
+}
+EXPORT_SYMBOL(pci_find_common_upstream_dev);
@@ -860,6 +860,8 @@ static inline struct pci_dev *pci_get_bus_and_slot(unsigned int bus,
}
struct pci_dev *pci_get_class(unsigned int class, struct pci_dev *from);
int pci_dev_present(const struct pci_device_id *ids);
+struct pci_dev *pci_find_common_upstream_dev(struct pci_dev *from,
+ struct pci_dev *to);
int pci_bus_read_config_byte(struct pci_bus *bus, unsigned int devfn,
int where, u8 *val);
Add an interface to find the first device which is upstream of both devices. Signed-off-by: Will Davis <wdavis@nvidia.com> --- drivers/pci/search.c | 25 +++++++++++++++++++++++++ include/linux/pci.h | 2 ++ 2 files changed, 27 insertions(+)