Skip to content

Commit 6f339c3

Browse files
Ge Yanggregkh
authored andcommitted
mm/hugetlb: wait for hugetlb folios to be freed
[ Upstream commit 67bab13 ] Since the introduction of commit c77c0a8 ("mm/hugetlb: defer freeing of huge pages if in non-task context"), which supports deferring the freeing of hugetlb pages, the allocation of contiguous memory through cma_alloc() may fail probabilistically. In the CMA allocation process, if it is found that the CMA area is occupied by in-use hugetlb folios, these in-use hugetlb folios need to be migrated to another location. When there are no available hugetlb folios in the free hugetlb pool during the migration of in-use hugetlb folios, new folios are allocated from the buddy system. A temporary state is set on the newly allocated folio. Upon completion of the hugetlb folio migration, the temporary state is transferred from the new folios to the old folios. Normally, when the old folios with the temporary state are freed, it is directly released back to the buddy system. However, due to the deferred freeing of hugetlb pages, the PageBuddy() check fails, ultimately leading to the failure of cma_alloc(). Here is a simplified call trace illustrating the process: cma_alloc() ->__alloc_contig_migrate_range() // Migrate in-use hugetlb folios ->unmap_and_move_huge_page() ->folio_putback_hugetlb() // Free old folios ->test_pages_isolated() ->__test_page_isolated_in_pageblock() ->PageBuddy(page) // Check if the page is in buddy To resolve this issue, we have implemented a function named wait_for_freed_hugetlb_folios(). This function ensures that the hugetlb folios are properly released back to the buddy system after their migration is completed. By invoking wait_for_freed_hugetlb_folios() before calling PageBuddy(), we ensure that PageBuddy() will succeed. Link: https://lkml.kernel.org/r/1739936804-18199-1-git-send-email-yangge1116@126.com Fixes: c77c0a8 ("mm/hugetlb: defer freeing of huge pages if in non-task context") Signed-off-by: Ge Yang <yangge1116@126.com> Reviewed-by: Muchun Song <muchun.song@linux.dev> Acked-by: David Hildenbrand <david@redhat.com> Cc: Baolin Wang <baolin.wang@linux.alibaba.com> Cc: Barry Song <21cnbao@gmail.com> Cc: Oscar Salvador <osalvador@suse.de> Cc: <stable@vger.kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Sasha Levin <sashal@kernel.org>
1 parent d5e3ecb commit 6f339c3

File tree

3 files changed

+23
-0
lines changed

3 files changed

+23
-0
lines changed

include/linux/hugetlb.h

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -681,6 +681,7 @@ struct huge_bootmem_page {
681681
};
682682

683683
int isolate_or_dissolve_huge_page(struct page *page, struct list_head *list);
684+
void wait_for_freed_hugetlb_folios(void);
684685
struct folio *alloc_hugetlb_folio(struct vm_area_struct *vma,
685686
unsigned long addr, int avoid_reserve);
686687
struct folio *alloc_hugetlb_folio_nodemask(struct hstate *h, int preferred_nid,
@@ -1061,6 +1062,10 @@ static inline int isolate_or_dissolve_huge_page(struct page *page,
10611062
return -ENOMEM;
10621063
}
10631064

1065+
static inline void wait_for_freed_hugetlb_folios(void)
1066+
{
1067+
}
1068+
10641069
static inline struct folio *alloc_hugetlb_folio(struct vm_area_struct *vma,
10651070
unsigned long addr,
10661071
int avoid_reserve)

mm/hugetlb.c

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2956,6 +2956,14 @@ int isolate_or_dissolve_huge_page(struct page *page, struct list_head *list)
29562956
return ret;
29572957
}
29582958

2959+
void wait_for_freed_hugetlb_folios(void)
2960+
{
2961+
if (llist_empty(&hpage_freelist))
2962+
return;
2963+
2964+
flush_work(&free_hpage_work);
2965+
}
2966+
29592967
struct folio *alloc_hugetlb_folio(struct vm_area_struct *vma,
29602968
unsigned long addr, int avoid_reserve)
29612969
{

mm/page_isolation.c

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -611,6 +611,16 @@ int test_pages_isolated(unsigned long start_pfn, unsigned long end_pfn,
611611
struct zone *zone;
612612
int ret;
613613

614+
/*
615+
* Due to the deferred freeing of hugetlb folios, the hugepage folios may
616+
* not immediately release to the buddy system. This can cause PageBuddy()
617+
* to fail in __test_page_isolated_in_pageblock(). To ensure that the
618+
* hugetlb folios are properly released back to the buddy system, we
619+
* invoke the wait_for_freed_hugetlb_folios() function to wait for the
620+
* release to complete.
621+
*/
622+
wait_for_freed_hugetlb_folios();
623+
614624
/*
615625
* Note: pageblock_nr_pages != MAX_PAGE_ORDER. Then, chunks of free
616626
* pages are not aligned to pageblock_nr_pages.

0 commit comments

Comments
 (0)