| 
    
 [Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] Re: [PATCH V2] xen/virtio: Handle PCI devices which Host controller is described in DT
 On 10/19/22 22:41, Oleksandr Tyshchenko wrote: Hi Oleksandr On 19.10.22 11:47, Xenia Ragiadakou wrote: Hello XeniaOn 10/19/22 03:58, Stefano Stabellini wrote:On Sat, 15 Oct 2022, Oleksandr Tyshchenko wrote:From: Oleksandr Tyshchenko <oleksandr_tyshchenko@xxxxxxxx> Use the same "xen-grant-dma" device concept for the PCI devices behind device-tree based PCI Host controller, but with one modification. Unlike for platform devices, we cannot use generic IOMMU bindings (iommus property), as we need to support more flexible configuration. The problem is that PCI devices under the single PCI Host controller may have the backends running in different Xen domains and thus have different endpoints ID (backend domains ID). So use generic PCI-IOMMU bindings instead (iommu-map/iommu-map-mask properties) which allows us to describe relationship between PCI devices and backend domains ID properly. Signed-off-by: Oleksandr Tyshchenko <oleksandr_tyshchenko@xxxxxxxx>Now that I understood the approach and the reasons for it, I can review the patch :-) Please add an example of the bindings in the commit message.--- Slightly RFC. This is needed to support Xen grant mappings for virtio-pci devices on Arm at some point in the future. The Xen toolstack side is not completely ready yet. Here, for PCI devices we use more flexible way to pass backend domid to the guest than for platform devices. Changes V1 -> V2: - update commit description - rebase - rework to use generic PCI-IOMMU bindings instead of generic IOMMU bindings Previous discussion is at: https://urldefense.com/v3/__https://lore.kernel.org/xen-devel/20221006174804.2003029-1-olekstysh@xxxxxxxxx/__;!!GF_29dbcQIUBPA!3-vq7Edm3XfKtD5cnNjnOzDQvuo_XrhJ73yH-nPfqOkGGU0IjLG7R7MR_nAJCAPeOutHRLT44wKYwQwz3SauACie_ZAy$ [lore[.]kernel[.]org] Based on: https://urldefense.com/v3/__https://git.kernel.org/pub/scm/linux/kernel/git/xen/tip.git/log/?h=for-linus-6.1__;!!GF_29dbcQIUBPA!3-vq7Edm3XfKtD5cnNjnOzDQvuo_XrhJ73yH-nPfqOkGGU0IjLG7R7MR_nAJCAPeOutHRLT44wKYwQwz3SauAEnMDHAq$ [git[.]kernel[.]org] --- drivers/xen/grant-dma-ops.c | 87 ++++++++++++++++++++++++++++++++----- 1 file changed, 76 insertions(+), 11 deletions(-) diff --git a/drivers/xen/grant-dma-ops.c b/drivers/xen/grant-dma-ops.c index daa525df7bdc..b79d9d6ce154 100644 --- a/drivers/xen/grant-dma-ops.c +++ b/drivers/xen/grant-dma-ops.c @@ -10,6 +10,7 @@ #include <linux/module.h> #include <linux/dma-map-ops.h> #include <linux/of.h> +#include <linux/pci.h> #include <linux/pfn.h> #include <linux/xarray.h> #include <linux/virtio_anchor.h> @@ -292,12 +293,55 @@ static const struct dma_map_ops xen_grant_dma_ops = { .dma_supported = xen_grant_dma_supported, }; +static struct device_node *xen_dt_get_pci_host_node(struct device *dev) +{ + struct pci_dev *pdev = to_pci_dev(dev); + struct pci_bus *bus = pdev->bus; + + /* Walk up to the root bus to look for PCI Host controller */ + while (!pci_is_root_bus(bus)) + bus = bus->parent; + + return of_node_get(bus->bridge->parent->of_node); +}It seems silly that we need to walk the hierachy that way, but I couldn't find another way to do it You are right. I prefer your version instead of the above. With Stefano's suggestion, we won't walk the PCI hierarchy twice when executing xen_is_grant_dma_device() for PCI device: xen_is_grant_dma_device() -> xen_is_dt_grant_dma_device() -> xen_dt_map_id() -> xen_dt_get_pci_host_node() What do you think? I was thinking passing the device_node along with the device in the function arguments. More specifically, of doing this (not tested, just an idea): 
bool xen_is_grant_dma_device(struct device *dev)
{
    struct device_node *np;
    bool has_iommu = false;
    /* XXX Handle only DT devices for now */
    np = xen_dt_get_node(dev);
    if (np)
        has_iommu = xen_is_dt_grant_dma_device(dev, np);
    of_node_put(np);
    return has_iommu;
}
static bool xen_is_dt_grant_dma_device(struct device *dev,
                                       struct device_node *np)
{
    struct device_node *iommu_np = NULL;
    bool has_iommu;
    if (dev_is_pci(dev)) {
        struct pci_dev *pdev = to_pci_dev(dev);
        u32 id = PCI_DEVID(pdev->bus->number, pdev->devfn);
        of_map_id(np, id, "iommu-map", "iommu-map-mask", &iommu_np, NULL);
    } else {
        iommu_np = of_parse_phandle(np, "iommus", 0);
    }
        
    has_iommu = iommu_np && of_device_is_compatible(iommu_np, 
"xen,grant-dma");
    of_node_put(iommu_np);
    return has_iommu;
}
I 'm wondering ... is it possible for the host bridge device node to 
have the iommus property set? meaning that all of its pci devs will have 
the same backend?
 -- Xenia 
 
 
  | 
  
![]()  | 
            
         Lists.xenproject.org is hosted with RackSpace, monitoring our  |