mirror of
https://github.com/nxp-imx/linux-imx.git
synced 2025-07-10 11:25:18 +02:00
BACKPORT: mm: extend rmap flags arguments for folio_add_new_anon_rmap
Patch series "mm: clarify folio_add_new_anon_rmap() and
__folio_add_anon_rmap()", v2.
This patchset is preparatory work for mTHP swapin.
folio_add_new_anon_rmap() assumes that new anon rmaps are always
exclusive. However, this assumption doesn’t hold true for cases like
do_swap_page(), where a new anon might be added to the swapcache and is
not necessarily exclusive.
The patchset extends the rmap flags to allow folio_add_new_anon_rmap() to
handle both exclusive and non-exclusive new anon folios. The
do_swap_page() function is updated to use this extended API with rmap
flags. Consequently, all new anon folios now consistently use
folio_add_new_anon_rmap(). The special case for !folio_test_anon() in
__folio_add_anon_rmap() can be safely removed.
In conclusion, new anon folios always use folio_add_new_anon_rmap(),
regardless of exclusivity. Old anon folios continue to use
__folio_add_anon_rmap() via folio_add_anon_rmap_pmd() and
folio_add_anon_rmap_ptes().
This patch (of 3):
In the case of a swap-in, a new anonymous folio is not necessarily
exclusive. This patch updates the rmap flags to allow a new anonymous
folio to be treated as either exclusive or non-exclusive. To maintain the
existing behavior, we always use EXCLUSIVE as the default setting.
[akpm@linux-foundation.org: cleanup and constifications per David and akpm]
[v-songbaohua@oppo.com: fix missing doc for flags of folio_add_new_anon_rmap()]
Link: https://lkml.kernel.org/r/20240619210641.62542-1-21cnbao@gmail.com
[v-songbaohua@oppo.com: enhance doc for extend rmap flags arguments for folio_add_new_anon_rmap]
Link: https://lkml.kernel.org/r/20240622030256.43775-1-21cnbao@gmail.com
Link: https://lkml.kernel.org/r/20240617231137.80726-1-21cnbao@gmail.com
Link: https://lkml.kernel.org/r/20240617231137.80726-2-21cnbao@gmail.com
Signed-off-by: Barry Song <v-songbaohua@oppo.com>
Suggested-by: David Hildenbrand <david@redhat.com>
Tested-by: Shuai Yuan <yuanshuai@oppo.com>
Acked-by: David Hildenbrand <david@redhat.com>
Cc: Baolin Wang <baolin.wang@linux.alibaba.com>
Cc: Chris Li <chrisl@kernel.org>
Cc: "Huang, Ying" <ying.huang@intel.com>
Cc: Matthew Wilcox (Oracle) <willy@infradead.org>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Ryan Roberts <ryan.roberts@arm.com>
Cc: Suren Baghdasaryan <surenb@google.com>
Cc: Yang Shi <shy828301@gmail.com>
Cc: Yosry Ahmed <yosryahmed@google.com>
Cc: Yu Zhao <yuzhao@google.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
(cherry picked from commit 15bde4abab
)
[ Fix trivial conflicts in kernel/events/uprobesc.,
mm/memory.c and mm/rmap.c - Kalesh Singh ]
Bug: 313807618
Signed-off-by: Kalesh Singh <kaleshsingh@google.com>
Change-Id: Ia17c3cdb2410270a95eed9a0a23c6aa24e3965bb
This commit is contained in:
parent
4c68f621cd
commit
870700e450
|
@ -244,7 +244,7 @@ void folio_add_anon_rmap_ptes(struct folio *, struct page *, int nr_pages,
|
|||
void folio_add_anon_rmap_pmd(struct folio *, struct page *,
|
||||
struct vm_area_struct *, unsigned long address, rmap_t flags);
|
||||
void folio_add_new_anon_rmap(struct folio *, struct vm_area_struct *,
|
||||
unsigned long address);
|
||||
unsigned long address, rmap_t flags);
|
||||
void folio_add_file_rmap_ptes(struct folio *, struct page *, int nr_pages,
|
||||
struct vm_area_struct *);
|
||||
#define folio_add_file_rmap_pte(folio, page, vma) \
|
||||
|
|
|
@ -184,7 +184,7 @@ static int __replace_page(struct vm_area_struct *vma, unsigned long addr,
|
|||
|
||||
if (new_page) {
|
||||
folio_get(new_folio);
|
||||
folio_add_new_anon_rmap(new_folio, vma, addr);
|
||||
folio_add_new_anon_rmap(new_folio, vma, addr, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(new_folio, vma);
|
||||
trace_android_vh_uprobes_replace_page(new_folio, old_folio);
|
||||
} else
|
||||
|
|
|
@ -935,7 +935,7 @@ static vm_fault_t __do_huge_pmd_anonymous_page(struct vm_fault *vmf,
|
|||
|
||||
entry = mk_huge_pmd(page, vma->vm_page_prot);
|
||||
entry = maybe_pmd_mkwrite(pmd_mkdirty(entry), vma);
|
||||
folio_add_new_anon_rmap(folio, vma, haddr);
|
||||
folio_add_new_anon_rmap(folio, vma, haddr, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(folio, vma);
|
||||
pgtable_trans_huge_deposit(vma->vm_mm, vmf->pmd, pgtable);
|
||||
set_pmd_at(vma->vm_mm, haddr, vmf->pmd, entry);
|
||||
|
|
|
@ -1224,7 +1224,7 @@ static int collapse_huge_page(struct mm_struct *mm, unsigned long address,
|
|||
|
||||
spin_lock(pmd_ptl);
|
||||
BUG_ON(!pmd_none(*pmd));
|
||||
folio_add_new_anon_rmap(folio, vma, address);
|
||||
folio_add_new_anon_rmap(folio, vma, address, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(folio, vma);
|
||||
pgtable_trans_huge_deposit(mm, pmd, pgtable);
|
||||
set_pmd_at(mm, address, pmd, _pmd);
|
||||
|
|
10
mm/memory.c
10
mm/memory.c
|
@ -926,7 +926,7 @@ copy_present_page(struct vm_area_struct *dst_vma, struct vm_area_struct *src_vma
|
|||
*prealloc = NULL;
|
||||
copy_user_highpage(&new_folio->page, page, addr, src_vma);
|
||||
__folio_mark_uptodate(new_folio);
|
||||
folio_add_new_anon_rmap(new_folio, dst_vma, addr);
|
||||
folio_add_new_anon_rmap(new_folio, dst_vma, addr, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(new_folio, dst_vma);
|
||||
rss[MM_ANONPAGES]++;
|
||||
|
||||
|
@ -3358,7 +3358,7 @@ static vm_fault_t wp_page_copy(struct vm_fault *vmf)
|
|||
* some TLBs while the old PTE remains in others.
|
||||
*/
|
||||
ptep_clear_flush(vma, vmf->address, vmf->pte);
|
||||
folio_add_new_anon_rmap(new_folio, vma, vmf->address);
|
||||
folio_add_new_anon_rmap(new_folio, vma, vmf->address, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(new_folio, vma);
|
||||
/*
|
||||
* We call the notify macro here because, when using secondary
|
||||
|
@ -4286,7 +4286,7 @@ check_folio:
|
|||
|
||||
/* ksm created a completely new copy */
|
||||
if (unlikely(folio != swapcache && swapcache)) {
|
||||
folio_add_new_anon_rmap(folio, vma, address);
|
||||
folio_add_new_anon_rmap(folio, vma, address, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(folio, vma);
|
||||
} else {
|
||||
folio_add_anon_rmap_ptes(folio, page, nr_pages, vma, address,
|
||||
|
@ -4545,7 +4545,7 @@ static vm_fault_t do_anonymous_page(struct vm_fault *vmf)
|
|||
#ifdef CONFIG_TRANSPARENT_HUGEPAGE
|
||||
count_mthp_stat(folio_order(folio), MTHP_STAT_ANON_FAULT_ALLOC);
|
||||
#endif
|
||||
folio_add_new_anon_rmap(folio, vma, addr);
|
||||
folio_add_new_anon_rmap(folio, vma, addr, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(folio, vma);
|
||||
setpte:
|
||||
if (uffd_wp)
|
||||
|
@ -4745,7 +4745,7 @@ void set_pte_range(struct vm_fault *vmf, struct folio *folio,
|
|||
if (write && !(vma->vm_flags & VM_SHARED)) {
|
||||
add_mm_counter(vma->vm_mm, MM_ANONPAGES, nr);
|
||||
VM_BUG_ON_FOLIO(nr != 1, folio);
|
||||
folio_add_new_anon_rmap(folio, vma, addr);
|
||||
folio_add_new_anon_rmap(folio, vma, addr, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(folio, vma);
|
||||
} else {
|
||||
add_mm_counter(vma->vm_mm, mm_counter_file(folio), nr);
|
||||
|
|
|
@ -656,7 +656,7 @@ static void migrate_vma_insert_page(struct migrate_vma *migrate,
|
|||
goto unlock_abort;
|
||||
|
||||
inc_mm_counter(mm, MM_ANONPAGES);
|
||||
folio_add_new_anon_rmap(folio, vma, addr);
|
||||
folio_add_new_anon_rmap(folio, vma, addr, RMAP_EXCLUSIVE);
|
||||
if (!folio_is_zone_device(folio))
|
||||
folio_add_lru_vma(folio, vma);
|
||||
folio_get(folio);
|
||||
|
|
24
mm/rmap.c
24
mm/rmap.c
|
@ -1404,29 +1404,35 @@ void folio_add_anon_rmap_pmd(struct folio *folio, struct page *page,
|
|||
* @folio: The folio to add the mapping to.
|
||||
* @vma: the vm area in which the mapping is added
|
||||
* @address: the user virtual address mapped
|
||||
* @flags: The rmap flags
|
||||
*
|
||||
* Like folio_add_anon_rmap_*() but must only be called on *new* folios.
|
||||
* This means the inc-and-test can be bypassed.
|
||||
* The folio does not have to be locked.
|
||||
* The folio doesn't necessarily need to be locked while it's exclusive
|
||||
* unless two threads map it concurrently. However, the folio must be
|
||||
* locked if it's shared.
|
||||
*
|
||||
* If the folio is pmd-mappable, it is accounted as a THP. As the folio
|
||||
* is new, it's assumed to be mapped exclusively by a single process.
|
||||
* If the folio is pmd-mappable, it is accounted as a THP.
|
||||
*/
|
||||
void folio_add_new_anon_rmap(struct folio *folio, struct vm_area_struct *vma,
|
||||
unsigned long address)
|
||||
unsigned long address, rmap_t flags)
|
||||
{
|
||||
int nr = folio_nr_pages(folio);
|
||||
const int nr = folio_nr_pages(folio);
|
||||
const bool exclusive = flags & RMAP_EXCLUSIVE;
|
||||
|
||||
VM_WARN_ON_FOLIO(folio_test_hugetlb(folio), folio);
|
||||
VM_WARN_ON_FOLIO(!exclusive && !folio_test_locked(folio), folio);
|
||||
VM_BUG_ON_VMA(address < vma->vm_start ||
|
||||
address + (nr << PAGE_SHIFT) > vma->vm_end, vma);
|
||||
__folio_set_swapbacked(folio);
|
||||
__folio_set_anon(folio, vma, address, true);
|
||||
__folio_set_anon(folio, vma, address, exclusive);
|
||||
|
||||
if (likely(!folio_test_large(folio))) {
|
||||
/* increment count (starts at -1) */
|
||||
atomic_set(&folio->_mapcount, 0);
|
||||
SetPageAnonExclusive(&folio->page);
|
||||
if (exclusive)
|
||||
SetPageAnonExclusive(&folio->page);
|
||||
} else if (!folio_test_pmd_mappable(folio)) {
|
||||
int i;
|
||||
|
||||
|
@ -1435,7 +1441,8 @@ void folio_add_new_anon_rmap(struct folio *folio, struct vm_area_struct *vma,
|
|||
|
||||
/* increment count (starts at -1) */
|
||||
atomic_set(&page->_mapcount, 0);
|
||||
SetPageAnonExclusive(page);
|
||||
if (exclusive)
|
||||
SetPageAnonExclusive(page);
|
||||
}
|
||||
|
||||
atomic_set(&folio->_nr_pages_mapped, nr);
|
||||
|
@ -1443,7 +1450,8 @@ void folio_add_new_anon_rmap(struct folio *folio, struct vm_area_struct *vma,
|
|||
/* increment count (starts at -1) */
|
||||
atomic_set(&folio->_entire_mapcount, 0);
|
||||
atomic_set(&folio->_nr_pages_mapped, ENTIRELY_MAPPED);
|
||||
SetPageAnonExclusive(&folio->page);
|
||||
if (exclusive)
|
||||
SetPageAnonExclusive(&folio->page);
|
||||
__lruvec_stat_mod_folio(folio, NR_ANON_THPS, nr);
|
||||
}
|
||||
|
||||
|
|
|
@ -1829,7 +1829,7 @@ static int unuse_pte(struct vm_area_struct *vma, pmd_t *pmd,
|
|||
|
||||
folio_add_anon_rmap_pte(folio, page, vma, addr, rmap_flags);
|
||||
} else { /* ksm created a completely new copy */
|
||||
folio_add_new_anon_rmap(folio, vma, addr);
|
||||
folio_add_new_anon_rmap(folio, vma, addr, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(folio, vma);
|
||||
}
|
||||
new_pte = pte_mkold(mk_pte(page, vma->vm_page_prot));
|
||||
|
|
|
@ -217,7 +217,7 @@ int mfill_atomic_install_pte(pmd_t *dst_pmd,
|
|||
folio_add_lru(folio);
|
||||
folio_add_file_rmap_pte(folio, page, dst_vma);
|
||||
} else {
|
||||
folio_add_new_anon_rmap(folio, dst_vma, dst_addr);
|
||||
folio_add_new_anon_rmap(folio, dst_vma, dst_addr, RMAP_EXCLUSIVE);
|
||||
folio_add_lru_vma(folio, dst_vma);
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue
Block a user