Commit 807174a9 authored by Kirill A. Shutemov's avatar Kirill A. Shutemov Committed by Andrew Morton

mm: fix endless reclaim on machines with unaccepted memory

Unaccepted memory is considered unusable free memory, which is not counted
as free on the zone watermark check.  This causes get_page_from_freelist()
to accept more memory to hit the high watermark, but it creates problems
in the reclaim path.

The reclaim path encounters a failed zone watermark check and attempts to
reclaim memory.  This is usually successful, but if there is little or no
reclaimable memory, it can result in endless reclaim with little to no
progress.  This can occur early in the boot process, just after start of
the init process when the only reclaimable memory is the page cache of the
init executable and its libraries.

Make unaccepted memory free from watermark check point of view.  This way
unaccepted memory will never be the trigger of memory reclaim.  Accept
more memory in the get_page_from_freelist() if needed.

Link: https://lkml.kernel.org/r/20240809114854.3745464-2-kirill.shutemov@linux.intel.com
Fixes: dcdfdd40 ("mm: Add support for unaccepted memory")
Signed-off-by: default avatarKirill A. Shutemov <kirill.shutemov@linux.intel.com>
Reported-by: default avatarJianxiong Gao <jxgao@google.com>
Acked-by: default avatarDavid Hildenbrand <david@redhat.com>
Tested-by: default avatarJianxiong Gao <jxgao@google.com>
Cc: Borislav Petkov <bp@alien8.de>
Cc: Johannes Weiner <hannes@cmpxchg.org>
Cc: Kirill A. Shutemov <kirill.shutemov@linux.intel.com>
Cc: Matthew Wilcox <willy@infradead.org>
Cc: Mel Gorman <mgorman@suse.de>
Cc: Mike Rapoport (Microsoft) <rppt@kernel.org>
Cc: Tom Lendacky <thomas.lendacky@amd.com>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: <stable@vger.kernel.org>	[6.5+]
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
parent af3b7d09
...@@ -287,7 +287,7 @@ EXPORT_SYMBOL(nr_online_nodes); ...@@ -287,7 +287,7 @@ EXPORT_SYMBOL(nr_online_nodes);
static bool page_contains_unaccepted(struct page *page, unsigned int order); static bool page_contains_unaccepted(struct page *page, unsigned int order);
static void accept_page(struct page *page, unsigned int order); static void accept_page(struct page *page, unsigned int order);
static bool try_to_accept_memory(struct zone *zone, unsigned int order); static bool cond_accept_memory(struct zone *zone, unsigned int order);
static inline bool has_unaccepted_memory(void); static inline bool has_unaccepted_memory(void);
static bool __free_unaccepted(struct page *page); static bool __free_unaccepted(struct page *page);
...@@ -3072,9 +3072,6 @@ static inline long __zone_watermark_unusable_free(struct zone *z, ...@@ -3072,9 +3072,6 @@ static inline long __zone_watermark_unusable_free(struct zone *z,
if (!(alloc_flags & ALLOC_CMA)) if (!(alloc_flags & ALLOC_CMA))
unusable_free += zone_page_state(z, NR_FREE_CMA_PAGES); unusable_free += zone_page_state(z, NR_FREE_CMA_PAGES);
#endif #endif
#ifdef CONFIG_UNACCEPTED_MEMORY
unusable_free += zone_page_state(z, NR_UNACCEPTED);
#endif
return unusable_free; return unusable_free;
} }
...@@ -3368,6 +3365,8 @@ get_page_from_freelist(gfp_t gfp_mask, unsigned int order, int alloc_flags, ...@@ -3368,6 +3365,8 @@ get_page_from_freelist(gfp_t gfp_mask, unsigned int order, int alloc_flags,
} }
} }
cond_accept_memory(zone, order);
/* /*
* Detect whether the number of free pages is below high * Detect whether the number of free pages is below high
* watermark. If so, we will decrease pcp->high and free * watermark. If so, we will decrease pcp->high and free
...@@ -3393,10 +3392,8 @@ get_page_from_freelist(gfp_t gfp_mask, unsigned int order, int alloc_flags, ...@@ -3393,10 +3392,8 @@ get_page_from_freelist(gfp_t gfp_mask, unsigned int order, int alloc_flags,
gfp_mask)) { gfp_mask)) {
int ret; int ret;
if (has_unaccepted_memory()) { if (cond_accept_memory(zone, order))
if (try_to_accept_memory(zone, order)) goto try_this_zone;
goto try_this_zone;
}
#ifdef CONFIG_DEFERRED_STRUCT_PAGE_INIT #ifdef CONFIG_DEFERRED_STRUCT_PAGE_INIT
/* /*
...@@ -3450,10 +3447,8 @@ get_page_from_freelist(gfp_t gfp_mask, unsigned int order, int alloc_flags, ...@@ -3450,10 +3447,8 @@ get_page_from_freelist(gfp_t gfp_mask, unsigned int order, int alloc_flags,
return page; return page;
} else { } else {
if (has_unaccepted_memory()) { if (cond_accept_memory(zone, order))
if (try_to_accept_memory(zone, order)) goto try_this_zone;
goto try_this_zone;
}
#ifdef CONFIG_DEFERRED_STRUCT_PAGE_INIT #ifdef CONFIG_DEFERRED_STRUCT_PAGE_INIT
/* Try again if zone has deferred pages */ /* Try again if zone has deferred pages */
...@@ -6950,9 +6945,6 @@ static bool try_to_accept_memory_one(struct zone *zone) ...@@ -6950,9 +6945,6 @@ static bool try_to_accept_memory_one(struct zone *zone)
struct page *page; struct page *page;
bool last; bool last;
if (list_empty(&zone->unaccepted_pages))
return false;
spin_lock_irqsave(&zone->lock, flags); spin_lock_irqsave(&zone->lock, flags);
page = list_first_entry_or_null(&zone->unaccepted_pages, page = list_first_entry_or_null(&zone->unaccepted_pages,
struct page, lru); struct page, lru);
...@@ -6978,23 +6970,29 @@ static bool try_to_accept_memory_one(struct zone *zone) ...@@ -6978,23 +6970,29 @@ static bool try_to_accept_memory_one(struct zone *zone)
return true; return true;
} }
static bool try_to_accept_memory(struct zone *zone, unsigned int order) static bool cond_accept_memory(struct zone *zone, unsigned int order)
{ {
long to_accept; long to_accept;
int ret = false; bool ret = false;
if (!has_unaccepted_memory())
return false;
if (list_empty(&zone->unaccepted_pages))
return false;
/* How much to accept to get to high watermark? */ /* How much to accept to get to high watermark? */
to_accept = high_wmark_pages(zone) - to_accept = high_wmark_pages(zone) -
(zone_page_state(zone, NR_FREE_PAGES) - (zone_page_state(zone, NR_FREE_PAGES) -
__zone_watermark_unusable_free(zone, order, 0)); __zone_watermark_unusable_free(zone, order, 0) -
zone_page_state(zone, NR_UNACCEPTED));
/* Accept at least one page */ while (to_accept > 0) {
do {
if (!try_to_accept_memory_one(zone)) if (!try_to_accept_memory_one(zone))
break; break;
ret = true; ret = true;
to_accept -= MAX_ORDER_NR_PAGES; to_accept -= MAX_ORDER_NR_PAGES;
} while (to_accept > 0); }
return ret; return ret;
} }
...@@ -7037,7 +7035,7 @@ static void accept_page(struct page *page, unsigned int order) ...@@ -7037,7 +7035,7 @@ static void accept_page(struct page *page, unsigned int order)
{ {
} }
static bool try_to_accept_memory(struct zone *zone, unsigned int order) static bool cond_accept_memory(struct zone *zone, unsigned int order)
{ {
return false; return false;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment