mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-18 03:06:43 +00:00
mm: ksm: use more folio api in ksm_might_need_to_copy()
Patch series "mm: cleanup and use more folio in page fault", v3. Rename page_copy_prealloc() to folio_prealloc(), which is used by more functions, also do more folio conversion in page fault. This patch (of 5): Since ksm only support normal page, no swapout/in for ksm large folio too, add large folio check in ksm_might_need_to_copy(), also convert page->index to folio->index as page->index is going away. Then convert ksm_might_need_to_copy() to use more folio api to save nine compound_head() calls, short 'address' to reduce max-line-length. Link: https://lkml.kernel.org/r/20231118023232.1409103-1-wangkefeng.wang@huawei.com Link: https://lkml.kernel.org/r/20231118023232.1409103-2-wangkefeng.wang@huawei.com Signed-off-by: Kefeng Wang <wangkefeng.wang@huawei.com> Cc: David Hildenbrand <david@redhat.com> Cc: Matthew Wilcox (Oracle) <willy@infradead.org> Cc: Sidhartha Kumar <sidhartha.kumar@oracle.com> Cc: Vishal Moola (Oracle) <vishal.moola@gmail.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
This commit is contained in:
parent
6140edeea8
commit
1486fb5013
@ -77,7 +77,7 @@ static inline void ksm_exit(struct mm_struct *mm)
|
|||||||
* but what if the vma was unmerged while the page was swapped out?
|
* but what if the vma was unmerged while the page was swapped out?
|
||||||
*/
|
*/
|
||||||
struct page *ksm_might_need_to_copy(struct page *page,
|
struct page *ksm_might_need_to_copy(struct page *page,
|
||||||
struct vm_area_struct *vma, unsigned long address);
|
struct vm_area_struct *vma, unsigned long addr);
|
||||||
|
|
||||||
void rmap_walk_ksm(struct folio *folio, struct rmap_walk_control *rwc);
|
void rmap_walk_ksm(struct folio *folio, struct rmap_walk_control *rwc);
|
||||||
void folio_migrate_ksm(struct folio *newfolio, struct folio *folio);
|
void folio_migrate_ksm(struct folio *newfolio, struct folio *folio);
|
||||||
@ -130,7 +130,7 @@ static inline int ksm_madvise(struct vm_area_struct *vma, unsigned long start,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline struct page *ksm_might_need_to_copy(struct page *page,
|
static inline struct page *ksm_might_need_to_copy(struct page *page,
|
||||||
struct vm_area_struct *vma, unsigned long address)
|
struct vm_area_struct *vma, unsigned long addr)
|
||||||
{
|
{
|
||||||
return page;
|
return page;
|
||||||
}
|
}
|
||||||
|
39
mm/ksm.c
39
mm/ksm.c
@ -2876,48 +2876,51 @@ void __ksm_exit(struct mm_struct *mm)
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct page *ksm_might_need_to_copy(struct page *page,
|
struct page *ksm_might_need_to_copy(struct page *page,
|
||||||
struct vm_area_struct *vma, unsigned long address)
|
struct vm_area_struct *vma, unsigned long addr)
|
||||||
{
|
{
|
||||||
struct folio *folio = page_folio(page);
|
struct folio *folio = page_folio(page);
|
||||||
struct anon_vma *anon_vma = folio_anon_vma(folio);
|
struct anon_vma *anon_vma = folio_anon_vma(folio);
|
||||||
struct page *new_page;
|
struct folio *new_folio;
|
||||||
|
|
||||||
if (PageKsm(page)) {
|
if (folio_test_large(folio))
|
||||||
if (page_stable_node(page) &&
|
return page;
|
||||||
|
|
||||||
|
if (folio_test_ksm(folio)) {
|
||||||
|
if (folio_stable_node(folio) &&
|
||||||
!(ksm_run & KSM_RUN_UNMERGE))
|
!(ksm_run & KSM_RUN_UNMERGE))
|
||||||
return page; /* no need to copy it */
|
return page; /* no need to copy it */
|
||||||
} else if (!anon_vma) {
|
} else if (!anon_vma) {
|
||||||
return page; /* no need to copy it */
|
return page; /* no need to copy it */
|
||||||
} else if (page->index == linear_page_index(vma, address) &&
|
} else if (folio->index == linear_page_index(vma, addr) &&
|
||||||
anon_vma->root == vma->anon_vma->root) {
|
anon_vma->root == vma->anon_vma->root) {
|
||||||
return page; /* still no need to copy it */
|
return page; /* still no need to copy it */
|
||||||
}
|
}
|
||||||
if (PageHWPoison(page))
|
if (PageHWPoison(page))
|
||||||
return ERR_PTR(-EHWPOISON);
|
return ERR_PTR(-EHWPOISON);
|
||||||
if (!PageUptodate(page))
|
if (!folio_test_uptodate(folio))
|
||||||
return page; /* let do_swap_page report the error */
|
return page; /* let do_swap_page report the error */
|
||||||
|
|
||||||
new_page = alloc_page_vma(GFP_HIGHUSER_MOVABLE, vma, address);
|
new_folio = vma_alloc_folio(GFP_HIGHUSER_MOVABLE, 0, vma, addr, false);
|
||||||
if (new_page &&
|
if (new_folio &&
|
||||||
mem_cgroup_charge(page_folio(new_page), vma->vm_mm, GFP_KERNEL)) {
|
mem_cgroup_charge(new_folio, vma->vm_mm, GFP_KERNEL)) {
|
||||||
put_page(new_page);
|
folio_put(new_folio);
|
||||||
new_page = NULL;
|
new_folio = NULL;
|
||||||
}
|
}
|
||||||
if (new_page) {
|
if (new_folio) {
|
||||||
if (copy_mc_user_highpage(new_page, page, address, vma)) {
|
if (copy_mc_user_highpage(&new_folio->page, page, addr, vma)) {
|
||||||
put_page(new_page);
|
folio_put(new_folio);
|
||||||
memory_failure_queue(page_to_pfn(page), 0);
|
memory_failure_queue(page_to_pfn(page), 0);
|
||||||
return ERR_PTR(-EHWPOISON);
|
return ERR_PTR(-EHWPOISON);
|
||||||
}
|
}
|
||||||
SetPageDirty(new_page);
|
folio_set_dirty(new_folio);
|
||||||
__SetPageUptodate(new_page);
|
__folio_mark_uptodate(new_folio);
|
||||||
__SetPageLocked(new_page);
|
__folio_set_locked(new_folio);
|
||||||
#ifdef CONFIG_SWAP
|
#ifdef CONFIG_SWAP
|
||||||
count_vm_event(KSM_SWPIN_COPY);
|
count_vm_event(KSM_SWPIN_COPY);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
return new_page;
|
return new_folio ? &new_folio->page : NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void rmap_walk_ksm(struct folio *folio, struct rmap_walk_control *rwc)
|
void rmap_walk_ksm(struct folio *folio, struct rmap_walk_control *rwc)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user