diff options
Diffstat (limited to 'drivers/pci/intel-iommu.c')
-rw-r--r-- | drivers/pci/intel-iommu.c | 32 |
1 files changed, 19 insertions, 13 deletions
diff --git a/drivers/pci/intel-iommu.c b/drivers/pci/intel-iommu.c index d4217f737159..ff8b7ce4a013 100644 --- a/drivers/pci/intel-iommu.c +++ b/drivers/pci/intel-iommu.c | |||
@@ -56,6 +56,7 @@ | |||
56 | #define MAX_AGAW_WIDTH 64 | 56 | #define MAX_AGAW_WIDTH 64 |
57 | 57 | ||
58 | #define DOMAIN_MAX_ADDR(gaw) ((((u64)1) << gaw) - 1) | 58 | #define DOMAIN_MAX_ADDR(gaw) ((((u64)1) << gaw) - 1) |
59 | #define DOMAIN_MAX_PFN(gaw) ((((u64)1) << (gaw-VTD_PAGE_SHIFT)) - 1) | ||
59 | 60 | ||
60 | #define IOVA_PFN(addr) ((addr) >> PAGE_SHIFT) | 61 | #define IOVA_PFN(addr) ((addr) >> PAGE_SHIFT) |
61 | #define DMA_32BIT_PFN IOVA_PFN(DMA_BIT_MASK(32)) | 62 | #define DMA_32BIT_PFN IOVA_PFN(DMA_BIT_MASK(32)) |
@@ -777,17 +778,17 @@ static void dma_pte_clear_one(struct dmar_domain *domain, unsigned long pfn) | |||
777 | } | 778 | } |
778 | 779 | ||
779 | /* clear last level pte, a tlb flush should be followed */ | 780 | /* clear last level pte, a tlb flush should be followed */ |
780 | static void dma_pte_clear_range(struct dmar_domain *domain, u64 start, u64 end) | 781 | static void dma_pte_clear_range(struct dmar_domain *domain, |
782 | unsigned long start_pfn, | ||
783 | unsigned long last_pfn) | ||
781 | { | 784 | { |
782 | unsigned long start_pfn = IOVA_PFN(start); | ||
783 | unsigned long end_pfn = IOVA_PFN(end-1); | ||
784 | int addr_width = agaw_to_width(domain->agaw) - VTD_PAGE_SHIFT; | 785 | int addr_width = agaw_to_width(domain->agaw) - VTD_PAGE_SHIFT; |
785 | 786 | ||
786 | BUG_ON(addr_width < BITS_PER_LONG && start_pfn >> addr_width); | 787 | BUG_ON(addr_width < BITS_PER_LONG && start_pfn >> addr_width); |
787 | BUG_ON(addr_width < BITS_PER_LONG && end_pfn >> addr_width); | 788 | BUG_ON(addr_width < BITS_PER_LONG && last_pfn >> addr_width); |
788 | 789 | ||
789 | /* we don't need lock here; nobody else touches the iova range */ | 790 | /* we don't need lock here; nobody else touches the iova range */ |
790 | while (start_pfn <= end_pfn) { | 791 | while (start_pfn <= last_pfn) { |
791 | dma_pte_clear_one(domain, start_pfn); | 792 | dma_pte_clear_one(domain, start_pfn); |
792 | start_pfn++; | 793 | start_pfn++; |
793 | } | 794 | } |
@@ -1424,7 +1425,7 @@ static void domain_exit(struct dmar_domain *domain) | |||
1424 | end = end & (~PAGE_MASK); | 1425 | end = end & (~PAGE_MASK); |
1425 | 1426 | ||
1426 | /* clear ptes */ | 1427 | /* clear ptes */ |
1427 | dma_pte_clear_range(domain, 0, end); | 1428 | dma_pte_clear_range(domain, 0, DOMAIN_MAX_PFN(domain->gaw)); |
1428 | 1429 | ||
1429 | /* free page tables */ | 1430 | /* free page tables */ |
1430 | dma_pte_free_pagetable(domain, 0, end); | 1431 | dma_pte_free_pagetable(domain, 0, end); |
@@ -1890,7 +1891,8 @@ static int iommu_domain_identity_map(struct dmar_domain *domain, | |||
1890 | * RMRR range might have overlap with physical memory range, | 1891 | * RMRR range might have overlap with physical memory range, |
1891 | * clear it first | 1892 | * clear it first |
1892 | */ | 1893 | */ |
1893 | dma_pte_clear_range(domain, base, base + size); | 1894 | dma_pte_clear_range(domain, base >> VTD_PAGE_SHIFT, |
1895 | (base + size - 1) >> VTD_PAGE_SHIFT); | ||
1894 | 1896 | ||
1895 | return domain_page_mapping(domain, base, base, size, | 1897 | return domain_page_mapping(domain, base, base, size, |
1896 | DMA_PTE_READ|DMA_PTE_WRITE); | 1898 | DMA_PTE_READ|DMA_PTE_WRITE); |
@@ -2618,7 +2620,8 @@ static void intel_unmap_page(struct device *dev, dma_addr_t dev_addr, | |||
2618 | pci_name(pdev), size, (unsigned long long)start_addr); | 2620 | pci_name(pdev), size, (unsigned long long)start_addr); |
2619 | 2621 | ||
2620 | /* clear the whole page */ | 2622 | /* clear the whole page */ |
2621 | dma_pte_clear_range(domain, start_addr, start_addr + size); | 2623 | dma_pte_clear_range(domain, start_addr >> VTD_PAGE_SHIFT, |
2624 | (start_addr + size - 1) >> VTD_PAGE_SHIFT); | ||
2622 | /* free page tables */ | 2625 | /* free page tables */ |
2623 | dma_pte_free_pagetable(domain, start_addr, start_addr + size); | 2626 | dma_pte_free_pagetable(domain, start_addr, start_addr + size); |
2624 | if (intel_iommu_strict) { | 2627 | if (intel_iommu_strict) { |
@@ -2710,7 +2713,8 @@ static void intel_unmap_sg(struct device *hwdev, struct scatterlist *sglist, | |||
2710 | start_addr = iova->pfn_lo << PAGE_SHIFT; | 2713 | start_addr = iova->pfn_lo << PAGE_SHIFT; |
2711 | 2714 | ||
2712 | /* clear the whole page */ | 2715 | /* clear the whole page */ |
2713 | dma_pte_clear_range(domain, start_addr, start_addr + size); | 2716 | dma_pte_clear_range(domain, start_addr >> VTD_PAGE_SHIFT, |
2717 | (start_addr + size - 1) >> VTD_PAGE_SHIFT); | ||
2714 | /* free page tables */ | 2718 | /* free page tables */ |
2715 | dma_pte_free_pagetable(domain, start_addr, start_addr + size); | 2719 | dma_pte_free_pagetable(domain, start_addr, start_addr + size); |
2716 | 2720 | ||
@@ -2792,8 +2796,9 @@ static int intel_map_sg(struct device *hwdev, struct scatterlist *sglist, int ne | |||
2792 | size, prot); | 2796 | size, prot); |
2793 | if (ret) { | 2797 | if (ret) { |
2794 | /* clear the page */ | 2798 | /* clear the page */ |
2795 | dma_pte_clear_range(domain, start_addr, | 2799 | dma_pte_clear_range(domain, |
2796 | start_addr + offset); | 2800 | start_addr >> VTD_PAGE_SHIFT, |
2801 | (start_addr + offset - 1) >> VTD_PAGE_SHIFT); | ||
2797 | /* free page tables */ | 2802 | /* free page tables */ |
2798 | dma_pte_free_pagetable(domain, start_addr, | 2803 | dma_pte_free_pagetable(domain, start_addr, |
2799 | start_addr + offset); | 2804 | start_addr + offset); |
@@ -3382,7 +3387,7 @@ static void vm_domain_exit(struct dmar_domain *domain) | |||
3382 | end = end & (~VTD_PAGE_MASK); | 3387 | end = end & (~VTD_PAGE_MASK); |
3383 | 3388 | ||
3384 | /* clear ptes */ | 3389 | /* clear ptes */ |
3385 | dma_pte_clear_range(domain, 0, end); | 3390 | dma_pte_clear_range(domain, 0, DOMAIN_MAX_PFN(domain->gaw)); |
3386 | 3391 | ||
3387 | /* free page tables */ | 3392 | /* free page tables */ |
3388 | dma_pte_free_pagetable(domain, 0, end); | 3393 | dma_pte_free_pagetable(domain, 0, end); |
@@ -3526,7 +3531,8 @@ static void intel_iommu_unmap_range(struct iommu_domain *domain, | |||
3526 | /* The address might not be aligned */ | 3531 | /* The address might not be aligned */ |
3527 | base = iova & VTD_PAGE_MASK; | 3532 | base = iova & VTD_PAGE_MASK; |
3528 | size = VTD_PAGE_ALIGN(size); | 3533 | size = VTD_PAGE_ALIGN(size); |
3529 | dma_pte_clear_range(dmar_domain, base, base + size); | 3534 | dma_pte_clear_range(dmar_domain, base >> VTD_PAGE_SHIFT, |
3535 | (base + size - 1) >> VTD_PAGE_SHIFT); | ||
3530 | 3536 | ||
3531 | if (dmar_domain->max_addr == base + size) | 3537 | if (dmar_domain->max_addr == base + size) |
3532 | dmar_domain->max_addr = base; | 3538 | dmar_domain->max_addr = base; |