aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/pci/dmar.c
diff options
context:
space:
mode:
authorDavid Woodhouse <David.Woodhouse@intel.com>2009-12-02 04:20:27 -0500
committerDavid Woodhouse <David.Woodhouse@intel.com>2009-12-08 05:02:39 -0500
commit6ecbf01c7ce4c0f4c3bdfa0e64ac6258328fda6c (patch)
tree91b5c5cdf8aa7a3f4dea1c8ae46a98765fdd9a6f /drivers/pci/dmar.c
parent2c99220810c1c79322034628b993573b088ff2da (diff)
intel-iommu: Apply BIOS sanity checks for interrupt remapping too.
The BIOS errors where an IOMMU is reported either at zero or a bogus address are causing problems even when the IOMMU is disabled -- because interrupt remapping uses the same hardware. Ensure that the checks get applied for the interrupt remapping initialisation too. Cc: stable@kernel.org Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
Diffstat (limited to 'drivers/pci/dmar.c')
-rw-r--r--drivers/pci/dmar.c33
1 files changed, 26 insertions, 7 deletions
diff --git a/drivers/pci/dmar.c b/drivers/pci/dmar.c
index 56883fc1c7be..beeaef84e151 100644
--- a/drivers/pci/dmar.c
+++ b/drivers/pci/dmar.c
@@ -613,6 +613,8 @@ int __init dmar_table_init(void)
613 return 0; 613 return 0;
614} 614}
615 615
616static int bios_warned;
617
616int __init check_zero_address(void) 618int __init check_zero_address(void)
617{ 619{
618 struct acpi_table_dmar *dmar; 620 struct acpi_table_dmar *dmar;
@@ -643,6 +645,7 @@ int __init check_zero_address(void)
643 dmi_get_system_info(DMI_BIOS_VENDOR), 645 dmi_get_system_info(DMI_BIOS_VENDOR),
644 dmi_get_system_info(DMI_BIOS_VERSION), 646 dmi_get_system_info(DMI_BIOS_VERSION),
645 dmi_get_system_info(DMI_PRODUCT_VERSION)); 647 dmi_get_system_info(DMI_PRODUCT_VERSION));
648 bios_warned = 1;
646 goto failed; 649 goto failed;
647 } 650 }
648 651
@@ -662,6 +665,7 @@ int __init check_zero_address(void)
662 dmi_get_system_info(DMI_BIOS_VENDOR), 665 dmi_get_system_info(DMI_BIOS_VENDOR),
663 dmi_get_system_info(DMI_BIOS_VERSION), 666 dmi_get_system_info(DMI_BIOS_VERSION),
664 dmi_get_system_info(DMI_PRODUCT_VERSION)); 667 dmi_get_system_info(DMI_PRODUCT_VERSION));
668 bios_warned = 1;
665 goto failed; 669 goto failed;
666 } 670 }
667 } 671 }
@@ -722,6 +726,18 @@ int alloc_iommu(struct dmar_drhd_unit *drhd)
722 int agaw = 0; 726 int agaw = 0;
723 int msagaw = 0; 727 int msagaw = 0;
724 728
729 if (!drhd->reg_base_addr) {
730 if (!bios_warned) {
731 WARN(1, "Your BIOS is broken; DMAR reported at address zero!\n"
732 "BIOS vendor: %s; Ver: %s; Product Version: %s\n",
733 dmi_get_system_info(DMI_BIOS_VENDOR),
734 dmi_get_system_info(DMI_BIOS_VERSION),
735 dmi_get_system_info(DMI_PRODUCT_VERSION));
736 bios_warned = 1;
737 }
738 return -EINVAL;
739 }
740
725 iommu = kzalloc(sizeof(*iommu), GFP_KERNEL); 741 iommu = kzalloc(sizeof(*iommu), GFP_KERNEL);
726 if (!iommu) 742 if (!iommu)
727 return -ENOMEM; 743 return -ENOMEM;
@@ -738,13 +754,16 @@ int alloc_iommu(struct dmar_drhd_unit *drhd)
738 iommu->ecap = dmar_readq(iommu->reg + DMAR_ECAP_REG); 754 iommu->ecap = dmar_readq(iommu->reg + DMAR_ECAP_REG);
739 755
740 if (iommu->cap == (uint64_t)-1 && iommu->ecap == (uint64_t)-1) { 756 if (iommu->cap == (uint64_t)-1 && iommu->ecap == (uint64_t)-1) {
741 /* Promote an attitude of violence to a BIOS engineer today */ 757 if (!bios_warned) {
742 WARN(1, "Your BIOS is broken; DMAR reported at address %llx returns all ones!\n" 758 /* Promote an attitude of violence to a BIOS engineer today */
743 "BIOS vendor: %s; Ver: %s; Product Version: %s\n", 759 WARN(1, "Your BIOS is broken; DMAR reported at address %llx returns all ones!\n"
744 drhd->reg_base_addr, 760 "BIOS vendor: %s; Ver: %s; Product Version: %s\n",
745 dmi_get_system_info(DMI_BIOS_VENDOR), 761 drhd->reg_base_addr,
746 dmi_get_system_info(DMI_BIOS_VERSION), 762 dmi_get_system_info(DMI_BIOS_VENDOR),
747 dmi_get_system_info(DMI_PRODUCT_VERSION)); 763 dmi_get_system_info(DMI_BIOS_VERSION),
764 dmi_get_system_info(DMI_PRODUCT_VERSION));
765 bios_warned = 1;
766 }
748 goto err_unmap; 767 goto err_unmap;
749 } 768 }
750 769