Skip to content

Commit 7af52b6

Browse files
anadavshiloong
authored andcommitted
hugetlbfs: flush TLBs correctly after huge_pmd_unshare
ANBZ: torvalds#636 commit a4a118f upstream. When __unmap_hugepage_range() calls to huge_pmd_unshare() succeed, a TLB flush is missing. This TLB flush must be performed before releasing the i_mmap_rwsem, in order to prevent an unshared PMDs page from being released and reused before the TLB flush took place. Arguably, a comprehensive solution would use mmu_gather interface to batch the TLB flushes and the PMDs page release, however it is not an easy solution: (1) try_to_unmap_one() and try_to_migrate_one() also call huge_pmd_unshare() and they cannot use the mmu_gather interface; and (2) deferring the release of the page reference for the PMDs page until after i_mmap_rwsem is dropeed can confuse huge_pmd_unshare() into thinking PMDs are shared when they are not. Fix __unmap_hugepage_range() by adding the missing TLB flush, and forcing a flush when unshare is successful. Fixes: 24669e5 ("hugetlb: use mmu_gather instead of a temporary linked list for accumulating pages)" # 3.6 Signed-off-by: Nadav Amit <[email protected]> Reviewed-by: Mike Kravetz <[email protected]> Cc: Aneesh Kumar K.V <[email protected]> Cc: KAMEZAWA Hiroyuki <[email protected]> Cc: Andrew Morton <[email protected]> Signed-off-by: Linus Torvalds <[email protected]> Signed-off-by: Greg Kroah-Hartman <[email protected]> Fixes: CVE-2021-4002 Signed-off-by: Shile Zhang <[email protected]> Reviewed-by: Gang Deng <[email protected]> Reviewed-by: Tianjia Zhang <[email protected]>
1 parent c53484a commit 7af52b6

File tree

8 files changed

+87
-4
lines changed

8 files changed

+87
-4
lines changed

arch/arm/include/asm/tlb.h

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -280,6 +280,14 @@ tlb_remove_pmd_tlb_entry(struct mmu_gather *tlb, pmd_t *pmdp, unsigned long addr
280280
tlb_add_flush(tlb, addr);
281281
}
282282

283+
static inline void
284+
tlb_flush_pmd_range(struct mmu_gather *tlb, unsigned long address,
285+
unsigned long size)
286+
{
287+
tlb_add_flush(tlb, address);
288+
tlb_add_flush(tlb, address + size - PMD_SIZE);
289+
}
290+
283291
#define pte_free_tlb(tlb, ptep, addr) __pte_free_tlb(tlb, ptep, addr)
284292
#define pmd_free_tlb(tlb, pmdp, addr) __pmd_free_tlb(tlb, pmdp, addr)
285293
#define pud_free_tlb(tlb, pudp, addr) pud_free((tlb)->mm, pudp)

arch/ia64/include/asm/tlb.h

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -268,6 +268,16 @@ __tlb_remove_tlb_entry (struct mmu_gather *tlb, pte_t *ptep, unsigned long addre
268268
tlb->end_addr = address + PAGE_SIZE;
269269
}
270270

271+
static inline void
272+
tlb_flush_pmd_range(struct mmu_gather *tlb, unsigned long address,
273+
unsigned long size)
274+
{
275+
if (tlb->start_addr > address)
276+
tlb->start_addr = address;
277+
if (tlb->end_addr < address + size)
278+
tlb->end_addr = address + size;
279+
}
280+
271281
#define tlb_migrate_finish(mm) platform_tlb_migrate_finish(mm)
272282

273283
#define tlb_start_vma(tlb, vma) do { } while (0)

arch/s390/include/asm/tlb.h

Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,20 @@ static inline void tlb_remove_page_size(struct mmu_gather *tlb,
116116
return tlb_remove_page(tlb, page);
117117
}
118118

119+
static inline void tlb_flush_pmd_range(struct mmu_gather *tlb,
120+
unsigned long address, unsigned long size)
121+
{
122+
/*
123+
* the range might exceed the original range that was provided to
124+
* tlb_gather_mmu(), so we need to update it despite the fact it is
125+
* usually not updated.
126+
*/
127+
if (tlb->start > address)
128+
tlb->start = address;
129+
if (tlb->end < address + size)
130+
tlb->end = address + size;
131+
}
132+
119133
/*
120134
* pte_free_tlb frees a pte table and clears the CRSTE for the
121135
* page table from the tlb.
@@ -177,6 +191,8 @@ static inline void pud_free_tlb(struct mmu_gather *tlb, pud_t *pud,
177191
#define tlb_remove_tlb_entry(tlb, ptep, addr) do { } while (0)
178192
#define tlb_remove_pmd_tlb_entry(tlb, pmdp, addr) do { } while (0)
179193
#define tlb_migrate_finish(mm) do { } while (0)
194+
#define tlb_flush_pmd_range(tlb, addr, sz) do { } while (0)
195+
180196
#define tlb_remove_huge_tlb_entry(h, tlb, ptep, address) \
181197
tlb_remove_tlb_entry(tlb, ptep, address)
182198

arch/sh/include/asm/tlb.h

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -127,6 +127,16 @@ static inline void tlb_remove_page_size(struct mmu_gather *tlb,
127127
return tlb_remove_page(tlb, page);
128128
}
129129

130+
static inline void
131+
tlb_flush_pmd_range(struct mmu_gather *tlb, unsigned long address,
132+
unsigned long size)
133+
{
134+
if (tlb->start > address)
135+
tlb->start = address;
136+
if (tlb->end < address + size)
137+
tlb->end = address + size;
138+
}
139+
130140
#define tlb_remove_check_page_size_change tlb_remove_check_page_size_change
131141
static inline void tlb_remove_check_page_size_change(struct mmu_gather *tlb,
132142
unsigned int page_size)

arch/um/include/asm/tlb.h

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -130,6 +130,18 @@ static inline void tlb_remove_page_size(struct mmu_gather *tlb,
130130
return tlb_remove_page(tlb, page);
131131
}
132132

133+
static inline void
134+
tlb_flush_pmd_range(struct mmu_gather *tlb, unsigned long address,
135+
unsigned long size)
136+
{
137+
tlb->need_flush = 1;
138+
139+
if (tlb->start > address)
140+
tlb->start = address;
141+
if (tlb->end < address + size)
142+
tlb->end = address + size;
143+
}
144+
133145
/**
134146
* tlb_remove_tlb_entry - remember a pte unmapping for later tlb invalidation.
135147
*

include/asm-generic/tlb.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,8 @@ void arch_tlb_gather_mmu(struct mmu_gather *tlb,
118118
void tlb_flush_mmu(struct mmu_gather *tlb);
119119
void arch_tlb_finish_mmu(struct mmu_gather *tlb,
120120
unsigned long start, unsigned long end, bool force);
121+
void tlb_flush_pmd_range(struct mmu_gather *tlb, unsigned long address,
122+
unsigned long size);
121123
extern bool __tlb_remove_page_size(struct mmu_gather *tlb, struct page *page,
122124
int page_size);
123125

mm/hugetlb.c

Lines changed: 19 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -3481,6 +3481,7 @@ void __unmap_hugepage_range(struct mmu_gather *tlb, struct vm_area_struct *vma,
34813481
unsigned long sz = huge_page_size(h);
34823482
unsigned long mmun_start = start; /* For mmu_notifiers */
34833483
unsigned long mmun_end = end; /* For mmu_notifiers */
3484+
bool force_flush = false;
34843485

34853486
WARN_ON(!is_vm_hugetlb_page(vma));
34863487
BUG_ON(start & ~huge_page_mask(h));
@@ -3507,10 +3508,8 @@ void __unmap_hugepage_range(struct mmu_gather *tlb, struct vm_area_struct *vma,
35073508
ptl = huge_pte_lock(h, mm, ptep);
35083509
if (huge_pmd_unshare(mm, &address, ptep)) {
35093510
spin_unlock(ptl);
3510-
/*
3511-
* We just unmapped a page of PMDs by clearing a PUD.
3512-
* The caller's TLB flush range should cover this area.
3513-
*/
3511+
tlb_flush_pmd_range(tlb, address & PUD_MASK, PUD_SIZE);
3512+
force_flush = true;
35143513
continue;
35153514
}
35163515

@@ -3567,6 +3566,22 @@ void __unmap_hugepage_range(struct mmu_gather *tlb, struct vm_area_struct *vma,
35673566
}
35683567
mmu_notifier_invalidate_range_end(mm, mmun_start, mmun_end);
35693568
tlb_end_vma(tlb, vma);
3569+
3570+
/*
3571+
* If we unshared PMDs, the TLB flush was not recorded in mmu_gather. We
3572+
* could defer the flush until now, since by holding i_mmap_rwsem we
3573+
* guaranteed that the last refernece would not be dropped. But we must
3574+
* do the flushing before we return, as otherwise i_mmap_rwsem will be
3575+
* dropped and the last reference to the shared PMDs page might be
3576+
* dropped as well.
3577+
*
3578+
* In theory we could defer the freeing of the PMD pages as well, but
3579+
* huge_pmd_unshare() relies on the exact page_count for the PMD page to
3580+
* detect sharing, so we cannot defer the release of the page either.
3581+
* Instead, do flush now.
3582+
*/
3583+
if (force_flush)
3584+
tlb_flush_mmu_tlbonly(tlb);
35703585
}
35713586

35723587
void __unmap_hugepage_range_final(struct mmu_gather *tlb,

mm/memory.c

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -324,6 +324,16 @@ bool __tlb_remove_page_size(struct mmu_gather *tlb, struct page *page, int page_
324324
return false;
325325
}
326326

327+
void tlb_flush_pmd_range(struct mmu_gather *tlb, unsigned long address,
328+
unsigned long size)
329+
{
330+
if (tlb->page_size != 0 && tlb->page_size != PMD_SIZE)
331+
tlb_flush_mmu(tlb);
332+
333+
tlb->page_size = PMD_SIZE;
334+
tlb->start = min(tlb->start, address);
335+
tlb->end = max(tlb->end, address + size);
336+
}
327337
#endif /* HAVE_GENERIC_MMU_GATHER */
328338

329339
#ifdef CONFIG_HAVE_RCU_TABLE_FREE

0 commit comments

Comments
 (0)