mirror of
https://github.com/torvalds/linux.git
synced 2024-11-26 06:02:05 +00:00
mm: userfaultfd: fix unexpected change to src_folio when UFFDIO_MOVE fails
After ptep_clear_flush(), if we find that src_folio is pinned we will fail
UFFDIO_MOVE and put src_folio back to src_pte entry, but the change to
src_folio->{mapping,index} is not restored in this process. This is not
what we expected, so fix it.
This can cause the rmap for that page to be invalid, possibly resulting
in memory corruption. At least swapout+migration would no longer work,
because we might fail to locate the mappings of that folio.
Link: https://lkml.kernel.org/r/20240222080815.46291-1-zhengqi.arch@bytedance.com
Fixes: adef440691
("userfaultfd: UFFDIO_MOVE uABI")
Signed-off-by: Qi Zheng <zhengqi.arch@bytedance.com>
Reviewed-by: David Hildenbrand <david@redhat.com>
Reviewed-by: Suren Baghdasaryan <surenb@google.com>
Cc: Andrea Arcangeli <aarcange@redhat.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
This commit is contained in:
parent
803de9000f
commit
d7a08838ab
@ -914,9 +914,6 @@ static int move_present_pte(struct mm_struct *mm,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
folio_move_anon_rmap(src_folio, dst_vma);
|
|
||||||
WRITE_ONCE(src_folio->index, linear_page_index(dst_vma, dst_addr));
|
|
||||||
|
|
||||||
orig_src_pte = ptep_clear_flush(src_vma, src_addr, src_pte);
|
orig_src_pte = ptep_clear_flush(src_vma, src_addr, src_pte);
|
||||||
/* Folio got pinned from under us. Put it back and fail the move. */
|
/* Folio got pinned from under us. Put it back and fail the move. */
|
||||||
if (folio_maybe_dma_pinned(src_folio)) {
|
if (folio_maybe_dma_pinned(src_folio)) {
|
||||||
@ -925,6 +922,9 @@ static int move_present_pte(struct mm_struct *mm,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
folio_move_anon_rmap(src_folio, dst_vma);
|
||||||
|
WRITE_ONCE(src_folio->index, linear_page_index(dst_vma, dst_addr));
|
||||||
|
|
||||||
orig_dst_pte = mk_pte(&src_folio->page, dst_vma->vm_page_prot);
|
orig_dst_pte = mk_pte(&src_folio->page, dst_vma->vm_page_prot);
|
||||||
/* Follow mremap() behavior and treat the entry dirty after the move */
|
/* Follow mremap() behavior and treat the entry dirty after the move */
|
||||||
orig_dst_pte = pte_mkwrite(pte_mkdirty(orig_dst_pte), dst_vma);
|
orig_dst_pte = pte_mkwrite(pte_mkdirty(orig_dst_pte), dst_vma);
|
||||||
|
Loading…
Reference in New Issue
Block a user