Message ID | 20220429192329.3034378-17-willy@infradead.org (mailing list archive) |
---|---|
State | New |
Headers | show |
Series | Folio patches for 5.19 | expand |
On 2022-04-29 20:23:24 [+0100], Matthew Wilcox (Oracle) wrote: > diff --git a/mm/shmem.c b/mm/shmem.c > index 3461bdec6b38..4331a4daac01 100644 > --- a/mm/shmem.c > +++ b/mm/shmem.c > @@ -736,16 +735,16 @@ static int shmem_add_to_page_cache(struct page *page, > xas_set_err(&xas, -EEXIST); > goto unlock; > } > - xas_store(&xas, page); > + xas_store(&xas, folio); > if (xas_error(&xas)) > goto unlock; > - if (PageTransHuge(page)) { > + if (folio_test_large(folio)) { > count_vm_event(THP_FILE_ALLOC); > - __mod_lruvec_page_state(page, NR_SHMEM_THPS, nr); > + __lruvec_stat_mod_folio(folio, NR_SHMEM_THPS, nr); > } | CC mm/shmem.o |In file included from <command-line>: |mm/shmem.c: In function ‘shmem_add_to_page_cache’: |include/linux/compiler_types.h:352:45: error: call to ‘__compiletime_assert_262’ declared with attribute error: BUILD_BUG failed | 352 | _compiletime_assert(condition, msg, __compiletime_assert_, __COUNTER__) | | ^ … |mm/shmem.c:743:40: note: in expansion of macro ‘THP_FILE_ALLOC’ | 743 | count_vm_event(THP_FILE_ALLOC); | | ^~~~~~~~~~~~~~ | and | $ git grep THP_FILE_ALLOC | include/linux/vm_event_item.h: THP_FILE_ALLOC, | include/linux/vm_event_item.h:#define THP_FILE_ALLOC ({ BUILD_BUG(); 0; }) | mm/shmem.c: count_vm_event(THP_FILE_ALLOC); | $ grep CONFIG_TRANSPARENT_HUGEPAGE .config | # CONFIG_TRANSPARENT_HUGEPAGE is not set Sebastian
On Tue, May 03, 2022 at 01:10:09PM +0200, Sebastian Andrzej Siewior wrote: > On 2022-04-29 20:23:24 [+0100], Matthew Wilcox (Oracle) wrote: > > diff --git a/mm/shmem.c b/mm/shmem.c > > index 3461bdec6b38..4331a4daac01 100644 > > --- a/mm/shmem.c > > +++ b/mm/shmem.c > > @@ -736,16 +735,16 @@ static int shmem_add_to_page_cache(struct page *page, > > xas_set_err(&xas, -EEXIST); > > goto unlock; > > } > > - xas_store(&xas, page); > > + xas_store(&xas, folio); > > if (xas_error(&xas)) > > goto unlock; > > - if (PageTransHuge(page)) { > > + if (folio_test_large(folio)) { > > count_vm_event(THP_FILE_ALLOC); > > - __mod_lruvec_page_state(page, NR_SHMEM_THPS, nr); > > + __lruvec_stat_mod_folio(folio, NR_SHMEM_THPS, nr); > > } > > | CC mm/shmem.o > |In file included from <command-line>: > |mm/shmem.c: In function ‘shmem_add_to_page_cache’: > |include/linux/compiler_types.h:352:45: error: call to ‘__compiletime_assert_262’ declared with attribute error: BUILD_BUG failed > | 352 | _compiletime_assert(condition, msg, __compiletime_assert_, __COUNTER__) > | | ^ > … > |mm/shmem.c:743:40: note: in expansion of macro ‘THP_FILE_ALLOC’ > | 743 | count_vm_event(THP_FILE_ALLOC); Thanks. Stephen already reported that; fix here: https://lore.kernel.org/all/Ym++SI1ftbRg+9zK@casper.infradead.org/
On 2022-05-03 13:48:50 [+0100], Matthew Wilcox wrote: > Thanks. Stephen already reported that; fix here: > > https://lore.kernel.org/all/Ym++SI1ftbRg+9zK@casper.infradead.org/ Stephen says "I applied the above patch to the mm tree merge today" and I have here next-20220503. I don't have the BUILD_BUG() in can_split_folio() anymore so I have this change. My guess is that since THP_FILE_ALLOC is defined as BUILD_BUG() for !CONFIG_TRANSPARENT_HUGEPAGE and there is nothing that removes that part of the code, I end up in BUILD_BUG with CGROUP and no THP. PageTransHuge() used to "return false" for !CONFIG_TRANSPARENT_HUGEPAGE which isn't the case for folio_test_large(). Sebastian
On Tue, May 03, 2022 at 03:00:05PM +0200, Sebastian Andrzej Siewior wrote: > On 2022-05-03 13:48:50 [+0100], Matthew Wilcox wrote: > > Thanks. Stephen already reported that; fix here: > > > > https://lore.kernel.org/all/Ym++SI1ftbRg+9zK@casper.infradead.org/ > > Stephen says "I applied the above patch to the mm tree merge today" and > I have here next-20220503. I don't have the BUILD_BUG() in > can_split_folio() anymore so I have this change. Ah! I didn't realise you were testing next; I thought you'd picked up these patches some other way. > My guess is that since THP_FILE_ALLOC is defined as BUILD_BUG() for > !CONFIG_TRANSPARENT_HUGEPAGE and there is nothing that removes that part > of the code, I end up in BUILD_BUG with CGROUP and no THP. > > PageTransHuge() used to "return false" for !CONFIG_TRANSPARENT_HUGEPAGE > which isn't the case for folio_test_large(). Indeed, indeed. I missed another case. This fixes it: diff --git a/mm/shmem.c b/mm/shmem.c index 5b161a92e6f1..019ad8bf0d21 100644 --- a/mm/shmem.c +++ b/mm/shmem.c @@ -716,7 +716,7 @@ static int shmem_add_to_page_cache(struct folio *folio, if (!folio_test_swapcache(folio)) { error = mem_cgroup_charge(folio, charge_mm, gfp); if (error) { - if (folio_test_large(folio)) { + if (folio_test_pmd_mappable(folio)) { count_vm_event(THP_FILE_FALLBACK); count_vm_event(THP_FILE_FALLBACK_CHARGE); }
On 2022-05-03 14:05:25 [+0100], Matthew Wilcox wrote:
> Indeed, indeed. I missed another case. This fixes it:
Yup, thank you. This works.
Sebastian
diff --git a/mm/shmem.c b/mm/shmem.c index 3461bdec6b38..4331a4daac01 100644 --- a/mm/shmem.c +++ b/mm/shmem.c @@ -695,36 +695,35 @@ static unsigned long shmem_unused_huge_shrink(struct shmem_sb_info *sbinfo, /* * Like add_to_page_cache_locked, but error if expected item has gone. */ -static int shmem_add_to_page_cache(struct page *page, +static int shmem_add_to_page_cache(struct folio *folio, struct address_space *mapping, pgoff_t index, void *expected, gfp_t gfp, struct mm_struct *charge_mm) { - XA_STATE_ORDER(xas, &mapping->i_pages, index, compound_order(page)); - unsigned long nr = compound_nr(page); + XA_STATE_ORDER(xas, &mapping->i_pages, index, folio_order(folio)); + long nr = folio_nr_pages(folio); int error; - VM_BUG_ON_PAGE(PageTail(page), page); - VM_BUG_ON_PAGE(index != round_down(index, nr), page); - VM_BUG_ON_PAGE(!PageLocked(page), page); - VM_BUG_ON_PAGE(!PageSwapBacked(page), page); - VM_BUG_ON(expected && PageTransHuge(page)); + VM_BUG_ON_FOLIO(index != round_down(index, nr), folio); + VM_BUG_ON_FOLIO(!folio_test_locked(folio), folio); + VM_BUG_ON_FOLIO(!folio_test_swapbacked(folio), folio); + VM_BUG_ON(expected && folio_test_large(folio)); - page_ref_add(page, nr); - page->mapping = mapping; - page->index = index; + folio_ref_add(folio, nr); + folio->mapping = mapping; + folio->index = index; - if (!PageSwapCache(page)) { - error = mem_cgroup_charge(page_folio(page), charge_mm, gfp); + if (!folio_test_swapcache(folio)) { + error = mem_cgroup_charge(folio, charge_mm, gfp); if (error) { - if (PageTransHuge(page)) { + if (folio_test_large(folio)) { count_vm_event(THP_FILE_FALLBACK); count_vm_event(THP_FILE_FALLBACK_CHARGE); } goto error; } } - cgroup_throttle_swaprate(page, gfp); + folio_throttle_swaprate(folio, gfp); do { xas_lock_irq(&xas); @@ -736,16 +735,16 @@ static int shmem_add_to_page_cache(struct page *page, xas_set_err(&xas, -EEXIST); goto unlock; } - xas_store(&xas, page); + xas_store(&xas, folio); if (xas_error(&xas)) goto unlock; - if (PageTransHuge(page)) { + if (folio_test_large(folio)) { count_vm_event(THP_FILE_ALLOC); - __mod_lruvec_page_state(page, NR_SHMEM_THPS, nr); + __lruvec_stat_mod_folio(folio, NR_SHMEM_THPS, nr); } mapping->nrpages += nr; - __mod_lruvec_page_state(page, NR_FILE_PAGES, nr); - __mod_lruvec_page_state(page, NR_SHMEM, nr); + __lruvec_stat_mod_folio(folio, NR_FILE_PAGES, nr); + __lruvec_stat_mod_folio(folio, NR_SHMEM, nr); unlock: xas_unlock_irq(&xas); } while (xas_nomem(&xas, gfp)); @@ -757,8 +756,8 @@ static int shmem_add_to_page_cache(struct page *page, return 0; error: - page->mapping = NULL; - page_ref_sub(page, nr); + folio->mapping = NULL; + folio_ref_sub(folio, nr); return error; } @@ -1690,7 +1689,8 @@ static int shmem_swapin_page(struct inode *inode, pgoff_t index, struct address_space *mapping = inode->i_mapping; struct shmem_inode_info *info = SHMEM_I(inode); struct mm_struct *charge_mm = vma ? vma->vm_mm : NULL; - struct page *page; + struct page *page = NULL; + struct folio *folio; swp_entry_t swap; int error; @@ -1740,7 +1740,8 @@ static int shmem_swapin_page(struct inode *inode, pgoff_t index, goto failed; } - error = shmem_add_to_page_cache(page, mapping, index, + folio = page_folio(page); + error = shmem_add_to_page_cache(folio, mapping, index, swp_to_radix_entry(swap), gfp, charge_mm); if (error) @@ -1791,6 +1792,7 @@ static int shmem_getpage_gfp(struct inode *inode, pgoff_t index, struct shmem_inode_info *info = SHMEM_I(inode); struct shmem_sb_info *sbinfo; struct mm_struct *charge_mm; + struct folio *folio; struct page *page; pgoff_t hindex = index; gfp_t huge_gfp; @@ -1905,7 +1907,8 @@ static int shmem_getpage_gfp(struct inode *inode, pgoff_t index, if (sgp == SGP_WRITE) __SetPageReferenced(page); - error = shmem_add_to_page_cache(page, mapping, hindex, + folio = page_folio(page); + error = shmem_add_to_page_cache(folio, mapping, hindex, NULL, gfp & GFP_RECLAIM_MASK, charge_mm); if (error) @@ -2327,6 +2330,7 @@ int shmem_mfill_atomic_pte(struct mm_struct *dst_mm, gfp_t gfp = mapping_gfp_mask(mapping); pgoff_t pgoff = linear_page_index(dst_vma, dst_addr); void *page_kaddr; + struct folio *folio; struct page *page; int ret; pgoff_t max_off; @@ -2385,7 +2389,8 @@ int shmem_mfill_atomic_pte(struct mm_struct *dst_mm, if (unlikely(pgoff >= max_off)) goto out_release; - ret = shmem_add_to_page_cache(page, mapping, pgoff, NULL, + folio = page_folio(page); + ret = shmem_add_to_page_cache(folio, mapping, pgoff, NULL, gfp & GFP_RECLAIM_MASK, dst_mm); if (ret) goto out_release;
Shrinks shmem_add_to_page_cache() by 16 bytes. All the callers grow, but this is temporary as they will all be converted to folios soon. Signed-off-by: Matthew Wilcox (Oracle) <willy@infradead.org> --- mm/shmem.c | 57 +++++++++++++++++++++++++++++------------------------- 1 file changed, 31 insertions(+), 26 deletions(-)