Skip to content

Commit

Permalink
x86/IOMMU: mark IOMMU / intremap not in use when ACPI tables are missing
Browse files Browse the repository at this point in the history
x2apic_bsp_setup() gets called ahead of iommu_setup(), and since x2APIC
mode (physical vs clustered) depends on iommu_intremap, that variable
needs to be set to off as soon as we know we can't / won't enable
interrupt remapping, i.e. in particular when parsing of the respective
ACPI tables failed. Move the turning off of iommu_intremap from AMD
specific code into acpi_iommu_init(), accompanying it by clearing of
iommu_enable.

Take the opportunity and also fully skip ACPI table parsing logic on
VT-d when both "iommu=off" and "iommu=no-intremap" are in effect anyway,
like was already the case for AMD.

The tag below only references the commit uncovering a pre-existing
anomaly.

Fixes: d8bd823 ("AMD/IOMMU: obtain IVHD type to use earlier")
Reported-by: Andrew Cooper <andrew.cooper3@citrix.com>
Signed-off-by: Jan Beulich <jbeulich@suse.com>
Reviewed-by: Roger Pau Monné <roger.pau@citrix.com>
Reviewed-by: Kevin Tian <kevin.tian@intel.com>
Release-Acked-by: Ian Jackson <iwj@xenproject.org>
  • Loading branch information
jbeulich committed Nov 4, 2021
1 parent f7f4a52 commit 46c4061
Show file tree
Hide file tree
Showing 4 changed files with 20 additions and 18 deletions.
6 changes: 0 additions & 6 deletions xen/drivers/passthrough/amd/pci_amd_iommu.c
Original file line number Diff line number Diff line change
Expand Up @@ -183,20 +183,14 @@ int __init acpi_ivrs_init(void)
{
int rc;

if ( !iommu_enable && !iommu_intremap )
return 0;

rc = amd_iommu_get_supported_ivhd_type();
if ( rc < 0 )
return rc;
BUG_ON(!rc);
ivhd_type = rc;

if ( (amd_iommu_detect_acpi() !=0) || (iommu_found() == 0) )
{
iommu_intremap = iommu_intremap_off;
return -ENODEV;
}

iommu_init_ops = &_iommu_init_ops;

Expand Down
6 changes: 1 addition & 5 deletions xen/drivers/passthrough/vtd/dmar.c
Original file line number Diff line number Diff line change
Expand Up @@ -777,11 +777,7 @@ static int __init acpi_parse_dmar(struct acpi_table_header *table)
dmar = (struct acpi_table_dmar *)table;
dmar_flags = dmar->flags;

if ( !iommu_enable && !iommu_intremap )
{
ret = -EINVAL;
goto out;
}
ASSERT(iommu_enable || iommu_intremap);

if ( !dmar->width )
{
Expand Down
18 changes: 18 additions & 0 deletions xen/drivers/passthrough/x86/iommu.c
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,24 @@ enum iommu_intremap __read_mostly iommu_intremap = iommu_intremap_full;
bool __read_mostly iommu_intpost;
#endif

void __init acpi_iommu_init(void)
{
int ret;

if ( !iommu_enable && !iommu_intremap )
return;

ret = acpi_dmar_init();
if ( ret == -ENODEV )
ret = acpi_ivrs_init();

if ( ret )
{
iommu_enable = false;
iommu_intremap = iommu_intremap_off;
}
}

int __init iommu_hardware_setup(void)
{
struct IO_APIC_route_entry **ioapic_entries = NULL;
Expand Down
8 changes: 1 addition & 7 deletions xen/include/asm-x86/acpi.h
Original file line number Diff line number Diff line change
Expand Up @@ -141,16 +141,10 @@ extern u32 x86_acpiid_to_apicid[];
extern u32 pmtmr_ioport;
extern unsigned int pmtmr_width;

void acpi_iommu_init(void);
int acpi_dmar_init(void);
int acpi_ivrs_init(void);

static inline int acpi_iommu_init(void)
{
int ret = acpi_dmar_init();

return ret == -ENODEV ? acpi_ivrs_init() : ret;
}

void acpi_mmcfg_init(void);

/* Incremented whenever we transition through S3. Value is 1 during boot. */
Expand Down

0 comments on commit 46c4061

Please sign in to comment.