On Tue, 19 Apr 2022, Bjorn Helgaas wrote:
linux-pci-setup-res-fw-address-nobridge.diff Index: linux-macro/drivers/pci/setup-res.c =================================================================== --- linux-macro.orig/drivers/pci/setup-res.c +++ linux-macro/drivers/pci/setup-res.c @@ -212,9 +212,19 @@ static int pci_revert_fw_address(struct res->end = res->start + size - 1; res->flags &= ~IORESOURCE_UNSET;
- /*
* If we're behind a P2P or CardBus bridge, make sure we're
* inside the relevant forwarding window, or otherwise the
* assignment must have been bogus and accesses intended for
* the range assigned would not reach the device anyway.
* On the root bus accept anything under the assumption the
* host bridge will let it through.
root = pci_find_parent_resource(dev, res); if (!root) {*/
if (res->flags & IORESOURCE_IO)
if (dev->bus->parent)
return -ENXIO;
else root = &iomem_resource;else if (res->flags & IORESOURCE_IO) root = &ioport_resource;
Is there value in the ioport_resource and iomem_resource lines here? They've been there since 58c84eda0756, when we didn't look for a parent resource.
My understanding is this is the whole point of this code and said commit. If you look at PR16263 and e.g. comment #12, then you'll see that the case was that we have a PCI-to-PCI bridge device on the root bus (00:04.0) that decodes outside the root bus resource for the purpose of forwarding to its secondary bus and it works. It is fine presumably because the host bridge does additional decoding outside the root bus resource and doesn't tell us about it. However when we try to squeeze device 00:04.0 into space within the root bus resource as we know it, we overrun it and fail the assignment.
So the fallback is to revert to what the firmware has left in the BARs, which is outside any PCI resource, causing `pci_find_parent_resource' to fail. But to do such an assignment within our infrastructure, we need to allow assigning outside the root bus resource, which the reference to either `ioport_resource' or `iomem_resource' serves as required.
Now that only works for devices on the root bus and not ones behind PCI-to-PCI bridges, where we do have to respect the parent's resource, because a PCI-to-PCI bridge is not allowed to randomly decode ranges beyond ones programmed in standard bridge BARs.
351fc6d1a517 ("PCI: Fix starting basis for resource requests") added the pci_find_parent_resource() call, and I think that *should* find the root bus resources if the device is on the root bus. And the root bus resources should already include ioport_resource and iomem_resource if we don't have anything better.
Now the description of this commit doesn't really explain (to me anyway) why it is needed or what scenario it addresses that commit 58c84eda0756 didn't by itself handle correctly.
My understanding is `pci_find_parent_resource' is expected to always fail in this path as otherwise there shouldn't have been the need to revert to firmware settings and this function wouldn't have been called in the first place. This is because if there indeed is a parent resource the original firmware settings are supposed to fit, then our resource allocator would have found them and used them (or similar ones) rather than failing.
So I really do not understand what case this commit is supposed to address. So much for terse change descriptions without actual details of the case addressed! If you are able to infer more or track down an actual discussion with extra information around this commit, then I'm all ears.
So I wonder if we should just always return failure when we don't find a parent resource?
IIUC it would defeat the purpose of commit 58c84eda0756 and is therefore a no-no AFAICT.
If you agree with my interpretation, then I'll trim down the change description as you requested (but please bear in mind then it's better to have a little too much information rather than not enough or someone else may end up scratching their head over this change ten years from now just as I do over commit 351fc6d1a517 here) and repost the patch with the code update proper unchanged. Let me know if you have any further questions or comments.
Thank you for your review.
Maciej