Am Montag, den 30.03.2015, 10:52 -0600 schrieb Alex Williamson: > On Mon, 2015-03-30 at 18:06 +0200, Lucas Stach wrote: > > Am Montag, den 30.03.2015, 08:33 -0600 schrieb Alex Williamson: > > > On Mon, 2015-03-30 at 12:55 +0200, Lucas Stach wrote: > > > > This adds a simple way to get the root port a given device > > > > is connected to. > > > > > > > > Signed-off-by: Lucas Stach <l.stach@xxxxxxxxxxxxxx> > > > > --- > > > > v2: new patch in v2 > > > > v3: rename to pci_find_rootport to fit better with other API > > > > v4: - rename to make it obvious that this function is PCIe specific > > > > - fixes wrong assumption about what is a root bus in the presence > > > > virtual buses > > > > --- > > > > drivers/pci/search.c | 20 ++++++++++++++++++++ > > > > include/linux/pci.h | 1 + > > > > 2 files changed, 21 insertions(+) > > > > > > > > diff --git a/drivers/pci/search.c b/drivers/pci/search.c > > > > index a20ce7d5e2a7..d7c599103ae1 100644 > > > > --- a/drivers/pci/search.c > > > > +++ b/drivers/pci/search.c > > > > @@ -384,3 +384,23 @@ int pci_dev_present(const struct pci_device_id *ids) > > > > return 0; > > > > } > > > > EXPORT_SYMBOL(pci_dev_present); > > > > + > > > > +/** > > > > + * pcie_find_root_port - Returns the root port the given device is connected to. > > > > + * @dev: PCI device for which the root port should be found. > > > > + */ > > > > +struct pci_dev *pcie_find_root_port(struct pci_dev *dev) > > > > +{ > > > > + struct pci_bus *bus = dev->bus; > > > > + > > > > + /* if this device is located on the root bus, it is a root port */ > > > > + if (pci_is_root_bus(bus)) > > > > + return dev; > > > > > > It could also be a root complex endpoint or a conventional PCI > > > device/bridge sitting on the host bridge bus. > > > > > > + > > > > + /* walk up the PCI hierarchy to the first level below the root */ > > > > + while (bus->parent && bus->parent->parent) > > > > + bus = bus->parent; > > > > + > > > > + return bus->self; > > > > +} > > > > +EXPORT_SYMBOL(pcie_find_root_port); > > > > > > IMHO, this makes too many assumptions about the topology that it's > > > working with for a generic interface. Your usage may be fairly fixed, > > > but there are too many cases where it could return something that's not > > > a root port as a general interface. Thanks, > > > > > I'm open to suggestions on how to improve the detection. I really need > > something which works reliable in the majority of cases, as the Tegra > > quirk should not be executed on other platforms. > > > > Do you think filtering out EP devices and conventional PCI bridges on > > the root bus is enough? > > I'm actually pretty confused by the implementation of the quirk as well, > why do you even need this pcie_find_root_port() function? Your fixup is > called for every single PCI device in the system, so why do you need to > go to the trouble of scanning the topology for the root port? You'll be > passed the root port eventually and it will match your ID table w/o any > extra effort. As coded, you're calling the set capability function > multiple times per root port, once for itself and once for each device > below it. > > Personally, I'd probably do away with the table, declare a fixup for > each entry for the specific vendor/device ID, and make a simple quirk > callback that sets the capability bit. Just my preference though. > Thanks, > No, you missed the point of the fixup here. We need to apply this fixup on every device in the system if the root port is a Tegra. So the fixup needs to get called for every device, filtering on a specific vendor/device ID is not possible. But on the other hand this fixup may be compiled into a multiplatform kernel. If this kernel is strated on a device which isn't a Tegra (and so has no Tegra root port) we don't want to apply the fixup at all. Regards, Lucas -- Pengutronix e.K. | Lucas Stach | Industrial Linux Solutions | http://www.pengutronix.de/ | -- To unsubscribe from this list: send the line "unsubscribe linux-pci" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html