diff options
author | mark gross <mgross@linux.intel.com> | 2008-02-08 07:18:38 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2008-02-08 12:22:24 -0500 |
commit | f8bab73515ca5b392680bb033dceeb37b8463e95 (patch) | |
tree | c450e793839868c23209e6fcbfb189975d9d4db6 /drivers/pci/intel-iommu.c | |
parent | 2d3a4e3666325a9709cc8ea2e88151394e8f20fc (diff) |
intel-iommu: PMEN support
Add support for protected memory enable bits by clearing them if they are
set at startup time. Some future boot loaders or firmware could have this
bit set after it loads the kernel, and it needs to be cleared if DMA's are
going to happen effectively.
Signed-off-by: mark gross <mgross@intel.com>
Acked-by: Muli Ben-Yehuda <muli@il.ibm.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Andi Kleen <ak@suse.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'drivers/pci/intel-iommu.c')
-rw-r--r-- | drivers/pci/intel-iommu.c | 21 |
1 files changed, 20 insertions, 1 deletions
diff --git a/drivers/pci/intel-iommu.c b/drivers/pci/intel-iommu.c index 31fa6c92aa5e..585e188c1746 100644 --- a/drivers/pci/intel-iommu.c +++ b/drivers/pci/intel-iommu.c | |||
@@ -692,6 +692,23 @@ static int iommu_flush_iotlb_psi(struct intel_iommu *iommu, u16 did, | |||
692 | DMA_TLB_PSI_FLUSH, non_present_entry_flush); | 692 | DMA_TLB_PSI_FLUSH, non_present_entry_flush); |
693 | } | 693 | } |
694 | 694 | ||
695 | static void iommu_disable_protect_mem_regions(struct intel_iommu *iommu) | ||
696 | { | ||
697 | u32 pmen; | ||
698 | unsigned long flags; | ||
699 | |||
700 | spin_lock_irqsave(&iommu->register_lock, flags); | ||
701 | pmen = readl(iommu->reg + DMAR_PMEN_REG); | ||
702 | pmen &= ~DMA_PMEN_EPM; | ||
703 | writel(pmen, iommu->reg + DMAR_PMEN_REG); | ||
704 | |||
705 | /* wait for the protected region status bit to clear */ | ||
706 | IOMMU_WAIT_OP(iommu, DMAR_PMEN_REG, | ||
707 | readl, !(pmen & DMA_PMEN_PRS), pmen); | ||
708 | |||
709 | spin_unlock_irqrestore(&iommu->register_lock, flags); | ||
710 | } | ||
711 | |||
695 | static int iommu_enable_translation(struct intel_iommu *iommu) | 712 | static int iommu_enable_translation(struct intel_iommu *iommu) |
696 | { | 713 | { |
697 | u32 sts; | 714 | u32 sts; |
@@ -745,7 +762,7 @@ static char *fault_reason_strings[] = | |||
745 | "non-zero reserved fields in PTE", | 762 | "non-zero reserved fields in PTE", |
746 | "Unknown" | 763 | "Unknown" |
747 | }; | 764 | }; |
748 | #define MAX_FAULT_REASON_IDX ARRAY_SIZE(fault_reason_strings) - 1 | 765 | #define MAX_FAULT_REASON_IDX (ARRAY_SIZE(fault_reason_strings) - 1) |
749 | 766 | ||
750 | char *dmar_get_fault_reason(u8 fault_reason) | 767 | char *dmar_get_fault_reason(u8 fault_reason) |
751 | { | 768 | { |
@@ -1730,6 +1747,8 @@ int __init init_dmars(void) | |||
1730 | iommu_flush_context_global(iommu, 0); | 1747 | iommu_flush_context_global(iommu, 0); |
1731 | iommu_flush_iotlb_global(iommu, 0); | 1748 | iommu_flush_iotlb_global(iommu, 0); |
1732 | 1749 | ||
1750 | iommu_disable_protect_mem_regions(iommu); | ||
1751 | |||
1733 | ret = iommu_enable_translation(iommu); | 1752 | ret = iommu_enable_translation(iommu); |
1734 | if (ret) | 1753 | if (ret) |
1735 | goto error; | 1754 | goto error; |