ACPI/IORT: take _DMA methods into account for named components
authorArd Biesheuvel <ardb@kernel.org>
Mon, 20 Apr 2020 09:27:53 +0000 (11:27 +0200)
committerWill Deacon <will@kernel.org>
Tue, 28 Apr 2020 17:06:07 +0000 (18:06 +0100)
Where IORT nodes for named components can describe simple DMA limits
expressed as the number of address bits a device can drive, _DMA methods
in AML can express more complex topologies, involving DMA translation in
particular.

Currently, we only take this _DMA method into account if it appears on a
ACPI device node describing a PCIe root complex, but it is perfectly
acceptable to use them for named components as well, so let's ensure
we take them into account in those cases too.

Note that such named components are expected to reside under a
pseudo-bus node such as the ACPI0004 container device, which should be
providing the _DMA method as well as a _CRS (as mandated by the ACPI
spec). This is not enforced by the code however.

Reported-by: Andrei Warkentin <awarkentin@vmware.com>
Signed-off-by: Ard Biesheuvel <ardb@kernel.org>
Acked-by: Lorenzo Pieralisi <lorenzo.pieralisi@arm.com>
Link: https://lore.kernel.org/r/20200420092753.9819-1-ardb@kernel.org
Signed-off-by: Will Deacon <will@kernel.org>
drivers/acpi/arm64/iort.c

index 7d04424189dfbbf7f837cdb0cb6427f8d947261f..051b2ce030701e68280150afe468eb3fe71859c4 100644 (file)
@@ -1148,13 +1148,10 @@ void iort_dma_setup(struct device *dev, u64 *dma_addr, u64 *dma_size)
        else
                size = 1ULL << 32;
 
-       if (dev_is_pci(dev)) {
-               ret = acpi_dma_get_range(dev, &dmaaddr, &offset, &size);
-               if (ret == -ENODEV)
-                       ret = rc_dma_get_range(dev, &size);
-       } else {
-               ret = nc_dma_get_range(dev, &size);
-       }
+       ret = acpi_dma_get_range(dev, &dmaaddr, &offset, &size);
+       if (ret == -ENODEV)
+               ret = dev_is_pci(dev) ? rc_dma_get_range(dev, &size)
+                                     : nc_dma_get_range(dev, &size);
 
        if (!ret) {
                /*