Commit 3c124435 authored by Logan Gunthorpe's avatar Logan Gunthorpe Committed by Joerg Roedel

iommu/amd: Support multiple PCI DMA aliases in IRQ Remapping

Non-Transparent Bridge (NTB) devices (among others) may have many DMA
aliases seeing the hardware will send requests with different device ids
depending on their origin across the bridged hardware.

See commit ad281ecf ("PCI: Add DMA alias quirk for Microsemi Switchtec
NTB") for more information on this.

The AMD IOMMU IRQ remapping functionality ignores all PCI aliases for
IRQs so if devices send an interrupt from one of their aliases they
will be blocked on AMD hardware with the IOMMU enabled.

To fix this, ensure IRQ remapping is enabled for all aliases with
MSI interrupts.

This is analogous to the functionality added to the Intel IRQ remapping
code in commit 3f0c625c ("iommu/vt-d: Allow interrupts from the entire
bus for aliased devices")
Signed-off-by: default avatarLogan Gunthorpe <logang@deltatee.com>
Signed-off-by: default avatarJoerg Roedel <jroedel@suse.de>
parent 3332364e
...@@ -3174,7 +3174,20 @@ static void set_remap_table_entry(struct amd_iommu *iommu, u16 devid, ...@@ -3174,7 +3174,20 @@ static void set_remap_table_entry(struct amd_iommu *iommu, u16 devid,
iommu_flush_dte(iommu, devid); iommu_flush_dte(iommu, devid);
} }
static struct irq_remap_table *alloc_irq_table(u16 devid) static int set_remap_table_entry_alias(struct pci_dev *pdev, u16 alias,
void *data)
{
struct irq_remap_table *table = data;
irq_lookup_table[alias] = table;
set_dte_irq_entry(alias, table);
iommu_flush_dte(amd_iommu_rlookup_table[alias], alias);
return 0;
}
static struct irq_remap_table *alloc_irq_table(u16 devid, struct pci_dev *pdev)
{ {
struct irq_remap_table *table = NULL; struct irq_remap_table *table = NULL;
struct irq_remap_table *new_table = NULL; struct irq_remap_table *new_table = NULL;
...@@ -3220,7 +3233,12 @@ static struct irq_remap_table *alloc_irq_table(u16 devid) ...@@ -3220,7 +3233,12 @@ static struct irq_remap_table *alloc_irq_table(u16 devid)
table = new_table; table = new_table;
new_table = NULL; new_table = NULL;
if (pdev)
pci_for_each_dma_alias(pdev, set_remap_table_entry_alias,
table);
else
set_remap_table_entry(iommu, devid, table); set_remap_table_entry(iommu, devid, table);
if (devid != alias) if (devid != alias)
set_remap_table_entry(iommu, alias, table); set_remap_table_entry(iommu, alias, table);
...@@ -3237,7 +3255,8 @@ static struct irq_remap_table *alloc_irq_table(u16 devid) ...@@ -3237,7 +3255,8 @@ static struct irq_remap_table *alloc_irq_table(u16 devid)
return table; return table;
} }
static int alloc_irq_index(u16 devid, int count, bool align) static int alloc_irq_index(u16 devid, int count, bool align,
struct pci_dev *pdev)
{ {
struct irq_remap_table *table; struct irq_remap_table *table;
int index, c, alignment = 1; int index, c, alignment = 1;
...@@ -3247,7 +3266,7 @@ static int alloc_irq_index(u16 devid, int count, bool align) ...@@ -3247,7 +3266,7 @@ static int alloc_irq_index(u16 devid, int count, bool align)
if (!iommu) if (!iommu)
return -ENODEV; return -ENODEV;
table = alloc_irq_table(devid); table = alloc_irq_table(devid, pdev);
if (!table) if (!table)
return -ENODEV; return -ENODEV;
...@@ -3680,7 +3699,7 @@ static int irq_remapping_alloc(struct irq_domain *domain, unsigned int virq, ...@@ -3680,7 +3699,7 @@ static int irq_remapping_alloc(struct irq_domain *domain, unsigned int virq,
struct irq_remap_table *table; struct irq_remap_table *table;
struct amd_iommu *iommu; struct amd_iommu *iommu;
table = alloc_irq_table(devid); table = alloc_irq_table(devid, NULL);
if (table) { if (table) {
if (!table->min_index) { if (!table->min_index) {
/* /*
...@@ -3697,11 +3716,15 @@ static int irq_remapping_alloc(struct irq_domain *domain, unsigned int virq, ...@@ -3697,11 +3716,15 @@ static int irq_remapping_alloc(struct irq_domain *domain, unsigned int virq,
} else { } else {
index = -ENOMEM; index = -ENOMEM;
} }
} else { } else if (info->type == X86_IRQ_ALLOC_TYPE_MSI ||
info->type == X86_IRQ_ALLOC_TYPE_MSIX) {
bool align = (info->type == X86_IRQ_ALLOC_TYPE_MSI); bool align = (info->type == X86_IRQ_ALLOC_TYPE_MSI);
index = alloc_irq_index(devid, nr_irqs, align); index = alloc_irq_index(devid, nr_irqs, align, info->msi_dev);
} else {
index = alloc_irq_index(devid, nr_irqs, false, NULL);
} }
if (index < 0) { if (index < 0) {
pr_warn("Failed to allocate IRTE\n"); pr_warn("Failed to allocate IRTE\n");
ret = index; ret = index;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment