PCI/MSI: Setup MSI domain on a per-device basis using IORT ACPI table
authorTomasz Nowicki <tn@semihalf.com>
Mon, 12 Sep 2016 18:32:22 +0000 (20:32 +0200)
committerMarc Zyngier <marc.zyngier@arm.com>
Mon, 12 Sep 2016 19:32:41 +0000 (20:32 +0100)
It is possible to provide information about which MSI controller to
use on a per-device basis for DT. This patch supply this with ACPI support.

Currently, IORT is the only one ACPI table which can provide such mapping.
In order to plug IORT into MSI infrastructure we are adding ACPI
equivalents for finding PCI device domain and its RID translation
(pci_msi_domain_get_msi_rid and pci_msi_domain_get_msi_rid calls).

Signed-off-by: Tomasz Nowicki <tn@semihalf.com>
Reviewed-by: Hanjun Guo <hanjun.guo@linaro.org>
Acked-by: Marc Zyngier <marc.zyngier@arm.com>
Acked-by: Bjorn Helgaas <bhelgaas@google.com>
Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
drivers/pci/msi.c

index 98f12223c734f15835edf438a94d7eccf15c991d..137b4c5fb638a29bff7b280e9f60f5d3c415ec35 100644 (file)
@@ -19,6 +19,7 @@
 #include <linux/smp.h>
 #include <linux/errno.h>
 #include <linux/io.h>
+#include <linux/acpi_iort.h>
 #include <linux/slab.h>
 #include <linux/irqdomain.h>
 #include <linux/of_irq.h>
@@ -1502,8 +1503,8 @@ u32 pci_msi_domain_get_msi_rid(struct irq_domain *domain, struct pci_dev *pdev)
        pci_for_each_dma_alias(pdev, get_msi_id_cb, &rid);
 
        of_node = irq_domain_get_of_node(domain);
-       if (of_node)
-               rid = of_msi_map_rid(&pdev->dev, of_node, rid);
+       rid = of_node ? of_msi_map_rid(&pdev->dev, of_node, rid) :
+                       iort_msi_map_rid(&pdev->dev, rid);
 
        return rid;
 }
@@ -1519,9 +1520,13 @@ u32 pci_msi_domain_get_msi_rid(struct irq_domain *domain, struct pci_dev *pdev)
  */
 struct irq_domain *pci_msi_get_device_domain(struct pci_dev *pdev)
 {
+       struct irq_domain *dom;
        u32 rid = 0;
 
        pci_for_each_dma_alias(pdev, get_msi_id_cb, &rid);
-       return of_msi_map_get_device_domain(&pdev->dev, rid);
+       dom = of_msi_map_get_device_domain(&pdev->dev, rid);
+       if (!dom)
+               dom = iort_get_device_domain(&pdev->dev, rid);
+       return dom;
 }
 #endif /* CONFIG_PCI_MSI_IRQ_DOMAIN */