[PATCH 9/9] iommu/vt-d: Support the new DMA_REMAP_OPT_OUT flag bit
From: Kevin Tian
Date: Thu Jun 04 2026 - 01:12:30 EST
Some BIOS already provides config options to expose/hide VT-d units
as a whole to/from system software. A new demand is to allow exposing
VT-d units but requesting system software to disable DMA remapping
while sustaining interrupt remapping. This can be communicated now by
setting the new DMA_REMAP_OPT_OUT flag bit in the DMAR table, as
introduced in VT-d spec v5.2 (section 8.1, DMA Remapping Reporting
Structure).
Introduce a new disabled state (DMAR_DISABLED_FW) for DMA_REMAP_OPT_OUT.
As the strongest disabled state, it cannot be overridden by user opts
or any force_on types. If tboot is enabled in the meantime, kernel will
panic. It is user responsibility to configure BIOS properly.
One cleanup is left for future - the DMAR flag is parsed multiple
times, in detect_intel_iommu(), dmar_platform_optin() (which can be
called at run-time), etc. Caching it is a cleaner way.
Signed-off-by: Kevin Tian <kevin.tian@xxxxxxxxx>
---
drivers/iommu/intel/dmar.c | 32 ++++++++++++++++++++++----------
drivers/iommu/intel/iommu.h | 4 ++++
include/linux/dmar.h | 1 +
3 files changed, 27 insertions(+), 10 deletions(-)
diff --git a/drivers/iommu/intel/dmar.c b/drivers/iommu/intel/dmar.c
index 791b91a7be29..24b173d33afd 100644
--- a/drivers/iommu/intel/dmar.c
+++ b/drivers/iommu/intel/dmar.c
@@ -931,7 +931,9 @@ dmar_validate_one_drhd(struct acpi_dmar_header *entry, void *arg)
*
* - DMAR_FORCEON_TBOOT: tboot strictly requires DMA remapping for secure
* boot hence supersedes any user opts ("iommu=off" or "intel_iommu=off")
- * and weaker disables.
+ * and weaker disables. But if firmware forces DMA remapping off (by
+ * setting DMAR_REMAP_OPT_OUT in the DMAR table), no force_on is allowed.
+ * Firmware settings must be changed to unblock tboot.
*
* - DMAR_FORCEON_PLATFORM: external-facing devices requires DMA
* remapping to prevent malicious downstream external devices from
@@ -975,31 +977,41 @@ static bool dmar_required(void)
void __init detect_intel_iommu(void)
{
- int ret;
struct dmar_res_callback validate_drhd_cb = {
.cb[ACPI_DMAR_TYPE_HARDWARE_UNIT] = &dmar_validate_one_drhd,
.ignore_unhandled = true,
};
+ struct acpi_table_dmar *dmar;
+ int ret;
down_write(&dmar_global_lock);
if (no_iommu)
dmar_state = DMAR_DISABLED_USER;
ret = dmar_table_detect();
- if (!ret)
- ret = dmar_walk_dmar_table((struct acpi_table_dmar *)dmar_tbl,
- &validate_drhd_cb);
- if (!ret && !iommu_detected && dmar_required()) {
+ if (!ret) {
+ dmar = (struct acpi_table_dmar *)dmar_tbl;
+ ret = dmar_walk_dmar_table(dmar, &validate_drhd_cb);
+ }
+
+ if (ret)
+ goto out;
+
+ if (dmar->flags & DMAR_REMAP_OPT_OUT) {
+ dmar_state = DMAR_DISABLED_FW;
+ pr_info("Firmware forces DMA remapping off\n");
+ }
+
+ if (!iommu_detected && dmar_required()) {
iommu_detected = 1;
/* Make sure ACS will be enabled */
pci_request_acs();
}
- if (!ret) {
- x86_init.iommu.iommu_init = intel_iommu_init;
- x86_platform.iommu_shutdown = intel_iommu_shutdown;
- }
+ x86_init.iommu.iommu_init = intel_iommu_init;
+ x86_platform.iommu_shutdown = intel_iommu_shutdown;
+out:
if (dmar_tbl) {
acpi_put_table(dmar_tbl);
dmar_tbl = NULL;
diff --git a/drivers/iommu/intel/iommu.h b/drivers/iommu/intel/iommu.h
index 1832debf4982..67ecf3521fd4 100644
--- a/drivers/iommu/intel/iommu.h
+++ b/drivers/iommu/intel/iommu.h
@@ -1364,6 +1364,9 @@ enum dmar_force_on {
* - DMAR_DISABLED_USER
* disabled by user opts ("intel_iommu=off" or "iommu=off").
*
+ * - DMAR_DISABLED_FW
+ * disabled due to firmware opt-out (DMAR_REMAP_OPT_OUT)
+ *
* - '0' is invalid, compared to decide dmar enabled vs. disabled
*
*/
@@ -1371,6 +1374,7 @@ enum dmar_force_on {
#define DMAR_ENABLED 1
#define DMAR_DISABLED_AUTO -1
#define DMAR_DISABLED_USER -2
+#define DMAR_DISABLED_FW -3
extern int dmar_state;
static inline bool dmar_is_enabled(void)
diff --git a/include/linux/dmar.h b/include/linux/dmar.h
index 692b2b445761..63e35df2cef4 100644
--- a/include/linux/dmar.h
+++ b/include/linux/dmar.h
@@ -24,6 +24,7 @@ struct acpi_dmar_header;
#define DMAR_INTR_REMAP 0x1
#define DMAR_X2APIC_OPT_OUT 0x2
#define DMAR_PLATFORM_OPT_IN 0x4
+#define DMAR_REMAP_OPT_OUT 0x8
struct intel_iommu;
--
2.43.0