aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/pci/intel-iommu.c40
1 files changed, 22 insertions, 18 deletions
diff --git a/drivers/pci/intel-iommu.c b/drivers/pci/intel-iommu.c
index ff8b7ce4a013..1526864a9d6f 100644
--- a/drivers/pci/intel-iommu.c
+++ b/drivers/pci/intel-iommu.c
@@ -669,27 +669,27 @@ static inline int width_to_agaw(int width)
669 669
670static inline unsigned int level_to_offset_bits(int level) 670static inline unsigned int level_to_offset_bits(int level)
671{ 671{
672 return (12 + (level - 1) * LEVEL_STRIDE); 672 return (level - 1) * LEVEL_STRIDE;
673} 673}
674 674
675static inline int pfn_level_offset(unsigned long pfn, int level) 675static inline int pfn_level_offset(unsigned long pfn, int level)
676{ 676{
677 return (pfn >> (level_to_offset_bits(level) - 12)) & LEVEL_MASK; 677 return (pfn >> level_to_offset_bits(level)) & LEVEL_MASK;
678} 678}
679 679
680static inline u64 level_mask(int level) 680static inline unsigned long level_mask(int level)
681{ 681{
682 return ((u64)-1 << level_to_offset_bits(level)); 682 return -1UL << level_to_offset_bits(level);
683} 683}
684 684
685static inline u64 level_size(int level) 685static inline unsigned long level_size(int level)
686{ 686{
687 return ((u64)1 << level_to_offset_bits(level)); 687 return 1UL << level_to_offset_bits(level);
688} 688}
689 689
690static inline u64 align_to_level(u64 addr, int level) 690static inline unsigned long align_to_level(unsigned long pfn, int level)
691{ 691{
692 return ((addr + level_size(level) - 1) & level_mask(level)); 692 return (pfn + level_size(level) - 1) & level_mask(level);
693} 693}
694 694
695static struct dma_pte * addr_to_dma_pte(struct dmar_domain *domain, u64 addr) 695static struct dma_pte * addr_to_dma_pte(struct dmar_domain *domain, u64 addr)
@@ -798,25 +798,29 @@ static void dma_pte_clear_range(struct dmar_domain *domain,
798static void dma_pte_free_pagetable(struct dmar_domain *domain, 798static void dma_pte_free_pagetable(struct dmar_domain *domain,
799 u64 start, u64 end) 799 u64 start, u64 end)
800{ 800{
801 int addr_width = agaw_to_width(domain->agaw); 801 int addr_width = agaw_to_width(domain->agaw) - VTD_PAGE_SHIFT;
802 unsigned long start_pfn = start >> VTD_PAGE_SHIFT;
803 unsigned long last_pfn = (end-1) >> VTD_PAGE_SHIFT;
802 struct dma_pte *pte; 804 struct dma_pte *pte;
803 int total = agaw_to_level(domain->agaw); 805 int total = agaw_to_level(domain->agaw);
804 int level; 806 int level;
805 u64 tmp; 807 unsigned long tmp;
806 808
807 BUG_ON(start >> addr_width); 809 BUG_ON(addr_width < BITS_PER_LONG && start_pfn >> addr_width);
808 BUG_ON(end >> addr_width); 810 BUG_ON(addr_width < BITS_PER_LONG && last_pfn >> addr_width);
809 811
810 /* we don't need lock here, nobody else touches the iova range */ 812 /* we don't need lock here, nobody else touches the iova range */
811 level = 2; 813 level = 2;
812 while (level <= total) { 814 while (level <= total) {
813 tmp = align_to_level(start, level); 815 tmp = align_to_level(start_pfn, level);
814 if (tmp >= end || (tmp + level_size(level) > end)) 816
817 /* Only clear this pte/pmd if we're asked to clear its
818 _whole_ range */
819 if (tmp + level_size(level) - 1 > last_pfn)
815 return; 820 return;
816 821
817 while (tmp < end) { 822 while (tmp <= last_pfn) {
818 pte = dma_pfn_level_pte(domain, tmp >> VTD_PAGE_SHIFT, 823 pte = dma_pfn_level_pte(domain, tmp, level);
819 level);
820 if (pte) { 824 if (pte) {
821 free_pgtable_page( 825 free_pgtable_page(
822 phys_to_virt(dma_pte_addr(pte))); 826 phys_to_virt(dma_pte_addr(pte)));
@@ -828,7 +832,7 @@ static void dma_pte_free_pagetable(struct dmar_domain *domain,
828 level++; 832 level++;
829 } 833 }
830 /* free pgd */ 834 /* free pgd */
831 if (start == 0 && end >= ((((u64)1) << addr_width) - 1)) { 835 if (start == 0 && last_pfn == DOMAIN_MAX_PFN(domain->gaw)) {
832 free_pgtable_page(domain->pgd); 836 free_pgtable_page(domain->pgd);
833 domain->pgd = NULL; 837 domain->pgd = NULL;
834 } 838 }