From: Rik van Riel <riel@surriel.com>
To: linux-kernel@vger.kernel.org
Cc: kernel-team@meta.com, linux-mm@kvack.org, david@kernel.org,
willy@infradead.org, surenb@google.com, hannes@cmpxchg.org,
ljs@kernel.org, ziy@nvidia.com, usama.arif@linux.dev,
fvdl@google.com, Rik van Riel <riel@surriel.com>
Subject: [RFC PATCH 10/40] mm: page_alloc: add superpageblock fullness lists for allocation steering
Date: Wed, 20 May 2026 10:59:16 -0400 [thread overview]
Message-ID: <20260520150018.2491267-11-riel@surriel.com> (raw)
In-Reply-To: <20260520150018.2491267-1-riel@surriel.com>
Organize superpageblocks into bucketed lists by fullness level and taint
status to enable efficient allocation steering without sorting.
Five fullness buckets (FULL, 75%, 50%, 25%, ALMOST_EMPTY) track what
fraction of a superpageblock's pageblocks are in use. Two categories (CLEAN
vs TAINTED) distinguish superpageblocks that contain only free and movable
pageblocks from those contaminated with unmovable, reclaimable, or reserved
pageblocks. A separate sb_empty list tracks completely free
superpageblocks.
Track fully-free pageblocks with a PB_all_free pageblock flag. When buddy
coalescing reconstructs a full pageblock, increment nr_free. Type counters
are driven by PB_has_* bit transitions, not by migratetype label changes.
For tainted superpageblocks, fullness is based on unmovable + reclaimable
pageblock counts rather than total usage, correctly reflecting how full
they are with the content types we're trying to concentrate.
Add a debugfs interface at /sys/kernel/debug/superpageblocks.
Signed-off-by: Rik van Riel <riel@surriel.com>
Assisted-by: Claude:claude-opus-4.7 syzkaller
---
include/linux/mmzone.h | 22 +++
include/linux/pageblock-flags.h | 1 +
mm/mm_init.c | 26 ++-
mm/page_alloc.c | 295 +++++++++++++++++++++++++++++++-
4 files changed, 339 insertions(+), 5 deletions(-)
diff --git a/include/linux/mmzone.h b/include/linux/mmzone.h
index 19190328e0c7..b8ada3d13a34 100644
--- a/include/linux/mmzone.h
+++ b/include/linux/mmzone.h
@@ -995,6 +995,23 @@ enum zone_type {
*/
#define SUPERPAGEBLOCK_NR_PAGEBLOCKS (1UL << (SUPERPAGEBLOCK_ORDER - pageblock_order))
+/* Superpageblock fullness buckets (by % of pageblocks in use) */
+enum sb_fullness {
+ SB_FULL, /* 100% full, 0 free pageblocks */
+ SB_FULL_75, /* 75-99% full */
+ SB_FULL_50, /* 50-74% full */
+ SB_FULL_25, /* 25-49% full */
+ SB_ALMOST_EMPTY, /* 1-24% full */
+ __NR_SB_FULLNESS,
+};
+
+/* Superpageblock taint categories */
+enum sb_category {
+ SB_CLEAN, /* only free + movable pageblocks */
+ SB_TAINTED, /* has unmovable/reclaimable/reserved */
+ __NR_SB_CATEGORIES,
+};
+
struct superpageblock {
/* Pageblock counts by current migratetype */
u16 nr_free;
@@ -1002,6 +1019,7 @@ struct superpageblock {
u16 nr_reclaimable;
u16 nr_movable;
u16 nr_reserved; /* holes, firmware, etc. */
+ u16 total_pageblocks; /* zone-clipped total */
/* For organizing superpageblocks by fullness category */
struct list_head list;
@@ -1059,6 +1077,10 @@ struct zone {
unsigned long superpageblock_base_pfn; /* 1GB-aligned base */
bool spb_kvmalloced; /* true if from kvmalloc (hotplug) */
+ /* Superpageblock fullness lists for allocation steering */
+ struct list_head spb_empty; /* completely free superpageblocks */
+ struct list_head spb_lists[__NR_SB_CATEGORIES][__NR_SB_FULLNESS];
+
/* zone_start_pfn == zone_start_paddr >> PAGE_SHIFT */
unsigned long zone_start_pfn;
diff --git a/include/linux/pageblock-flags.h b/include/linux/pageblock-flags.h
index 21bfcdf80b2e..4dce39d054a9 100644
--- a/include/linux/pageblock-flags.h
+++ b/include/linux/pageblock-flags.h
@@ -28,6 +28,7 @@ enum pageblock_bits {
PB_has_unmovable,
PB_has_reclaimable,
PB_has_movable,
+ PB_all_free, /* All pages in pageblock are free in buddy */
#ifdef CONFIG_MEMORY_ISOLATION
/*
diff --git a/mm/mm_init.c b/mm/mm_init.c
index ad1cbc2b4498..2dc73d8a8d6c 100644
--- a/mm/mm_init.c
+++ b/mm/mm_init.c
@@ -1548,7 +1548,17 @@ static void __meminit init_one_superpageblock(struct superpageblock *sb,
actual_pbs = (pb_end > pb_start) ?
((pb_end - pb_start + pageblock_nr_pages - 1) >>
pageblock_order) : 0;
+ sb->total_pageblocks = actual_pbs;
sb->nr_reserved = actual_pbs;
+ if (actual_pbs) {
+ /*
+ * All superpageblocks start as reserved (tainted+full).
+ * They move to the correct category when the pages
+ * inside are freed during boot.
+ */
+ list_add_tail(&sb->list,
+ &zone->spb_lists[SB_TAINTED][SB_FULL]);
+ }
}
static void __init setup_superpageblocks(struct zone *zone)
@@ -1558,11 +1568,18 @@ static void __init setup_superpageblocks(struct zone *zone)
unsigned long sb_base, nr_superpageblocks;
size_t alloc_size;
unsigned long i;
+ int cat, full;
zone->superpageblocks = NULL;
zone->nr_superpageblocks = 0;
zone->superpageblock_base_pfn = 0;
+ /* Fullness lists steer allocations to preferred superpageblocks */
+ INIT_LIST_HEAD(&zone->spb_empty);
+ for (cat = 0; cat < __NR_SB_CATEGORIES; cat++)
+ for (full = 0; full < __NR_SB_FULLNESS; full++)
+ INIT_LIST_HEAD(&zone->spb_lists[cat][full]);
+
if (!zone->spanned_pages)
return;
@@ -1688,8 +1705,9 @@ void __meminit resize_zone_superpageblocks(struct zone *zone)
}
/*
- * Update existing superpageblocks whose nr_reserved may have
- * increased due to the zone span growing into them.
+ * Update existing superpageblocks whose nr_reserved and
+ * total_pageblocks may have increased due to the zone
+ * span growing into them.
*/
if (zone->superpageblocks) {
old_offset = (zone->superpageblock_base_pfn - new_sb_base) >>
@@ -1707,8 +1725,10 @@ void __meminit resize_zone_superpageblocks(struct zone *zone)
sb->nr_reclaimable + sb->nr_movable +
sb->nr_reserved;
- if (new_pbs > old_pbs)
+ if (new_pbs > old_pbs) {
sb->nr_reserved += new_pbs - old_pbs;
+ sb->total_pageblocks = new_pbs;
+ }
}
}
diff --git a/mm/page_alloc.c b/mm/page_alloc.c
index b9b7d54a869c..c0f86a30b5c7 100644
--- a/mm/page_alloc.c
+++ b/mm/page_alloc.c
@@ -56,6 +56,8 @@
#include <linux/delayacct.h>
#include <linux/cacheinfo.h>
#include <linux/pgalloc_tag.h>
+#include <linux/debugfs.h>
+#include <linux/seq_file.h>
#include <asm/div64.h>
#include "internal.h"
#include "shuffle.h"
@@ -514,7 +516,157 @@ static void __spb_set_has_type(struct page *page, int migratetype)
}
/**
- * set_pageblock_migratetype - Set the migratetype of a pageblock
+ * spb_get_category - Determine if a superpageblock is clean or tainted
+ * @sb: superpageblock to classify
+ *
+ * A superpageblock is clean if it contains only free and movable pageblocks.
+ * Any unmovable, reclaimable, or reserved pageblocks make it tainted.
+ * Reserved pageblocks (memory holes) taint the superpageblock because it
+ * can never be used for 1GB hugepages, making it a better home for
+ * unmovable/reclaimable allocations.
+ */
+static inline enum sb_category spb_get_category(struct superpageblock *sb)
+{
+ if (sb->nr_unmovable || sb->nr_reclaimable || sb->nr_reserved)
+ return SB_TAINTED;
+ return SB_CLEAN;
+}
+
+/**
+ * sb_get_fullness - Determine the fullness bucket for a superpageblock
+ * @sb: superpageblock to classify
+ * @cat: the category (CLEAN or TAINTED) of this superpageblock
+ *
+ * For clean SPBs, fullness is based on total usage (total - nr_free).
+ * For tainted SPBs, fullness is based only on unmovable + reclaimable
+ * pageblocks, since those are what we're trying to concentrate.
+ * Uses SUPERPAGEBLOCK_NR_PAGEBLOCKS as divisor so that partial
+ * superpageblocks at zone boundaries are preferred over whole ones.
+ */
+static inline enum sb_fullness sb_get_fullness(struct superpageblock *sb,
+ enum sb_category cat)
+{
+ unsigned int used, total = sb->total_pageblocks;
+ unsigned int quarter = SUPERPAGEBLOCK_NR_PAGEBLOCKS / 4;
+
+ if (!total)
+ return SB_FULL;
+
+ if (cat == SB_TAINTED)
+ used = sb->nr_unmovable + sb->nr_reclaimable;
+ else
+ used = total - sb->nr_free;
+
+ if (used >= total)
+ return SB_FULL;
+
+ if (used >= 3 * quarter)
+ return SB_FULL_75;
+ if (used >= 2 * quarter)
+ return SB_FULL_50;
+ if (used >= quarter)
+ return SB_FULL_25;
+ return SB_ALMOST_EMPTY;
+}
+
+/**
+ * spb_update_list - Move a superpageblock to the correct fullness list
+ * @sb: superpageblock to reclassify
+ *
+ * Called after counters change. Removes from current list (if any)
+ * and adds to the appropriate list based on current fullness and
+ * taint status.
+ */
+static void spb_update_list(struct superpageblock *sb)
+{
+ struct zone *zone = sb->zone;
+ enum sb_category cat;
+ enum sb_fullness full;
+
+ list_del_init(&sb->list);
+
+ if (sb->nr_free == SUPERPAGEBLOCK_NR_PAGEBLOCKS) {
+ list_add_tail(&sb->list, &zone->spb_empty);
+ return;
+ }
+
+ cat = spb_get_category(sb);
+ full = sb_get_fullness(sb, cat);
+ list_add_tail(&sb->list, &zone->spb_lists[cat][full]);
+}
+
+/**
+ * superpageblock_pb_now_free - A pageblock just became fully free in buddy
+ * @page: page in the pageblock
+ *
+ * When buddy coalescing reconstructs a complete pageblock-order page,
+ * increment nr_free. Type counters are handled separately in
+ * mark_pageblock_free().
+ */
+static void superpageblock_pb_now_free(struct page *page)
+{
+ unsigned long pfn = page_to_pfn(page);
+ struct superpageblock *sb = pfn_to_superpageblock(page_zone(page), pfn);
+
+ if (!sb)
+ return;
+
+ sb->nr_free++;
+
+ spb_update_list(sb);
+}
+
+/**
+ * superpageblock_pb_now_used - A fully-free pageblock just got its first allocation
+ * @page: page in the pageblock
+ *
+ * When allocating from an order >= pageblock_order free page, decrement
+ * nr_free. Type counters are handled separately by __spb_set_has_type()
+ * at allocation time.
+ */
+static void superpageblock_pb_now_used(struct page *page)
+{
+ unsigned long pfn = page_to_pfn(page);
+ struct superpageblock *sb = pfn_to_superpageblock(page_zone(page), pfn);
+
+ if (!sb)
+ return;
+
+ if (sb->nr_free)
+ sb->nr_free--;
+
+ spb_update_list(sb);
+}
+
+/**
+ * superpageblock_range_now_used - Mark a multi-pageblock free range as no longer free
+ * @page: first page of the range (must be pageblock-aligned)
+ * @order: order of the range (must be >= pageblock_order)
+ *
+ * When a free page of order >= pageblock_order is removed from buddy outside
+ * the normal allocation path (e.g. __isolate_free_page, memory hotplug,
+ * HW poison takeoff), every constituent pageblock leaves its PB_all_free
+ * state. Walk the range, clear PB_all_free, and decrement nr_free for each
+ * affected pageblock. PB_has_* bits are not touched: the pages are not being
+ * allocated to a specific migratetype here. They will be re-established by
+ * mark_pageblock_free() if the pages later return to buddy and coalesce.
+ */
+static void superpageblock_range_now_used(struct page *page, unsigned int order)
+{
+ unsigned long pfn = page_to_pfn(page);
+ unsigned long end_pfn = pfn + (1UL << order);
+
+ for (; pfn < end_pfn; pfn += pageblock_nr_pages) {
+ struct page *pb_page = pfn_to_page(pfn);
+
+ if (get_pfnblock_bit(pb_page, pfn, PB_all_free)) {
+ clear_pfnblock_bit(pb_page, pfn, PB_all_free);
+ superpageblock_pb_now_used(pb_page);
+ }
+ }
+}
+
+/** set_pageblock_migratetype - Set the migratetype of a pageblock
* @page: The page within the block of interest
* @migratetype: migratetype to set
*/
@@ -577,6 +729,7 @@ void __meminit init_pageblock_migratetype(struct page *page,
if (sb->nr_reserved)
sb->nr_reserved--;
__spb_set_has_type(page, migratetype);
+ spb_update_list(sb);
}
}
@@ -1015,6 +1168,11 @@ static void mark_pageblock_free(struct page *page, unsigned long pfn)
clear_pfnblock_bit(page, pfn, PB_has_unmovable);
clear_pfnblock_bit(page, pfn, PB_has_reclaimable);
clear_pfnblock_bit(page, pfn, PB_has_movable);
+
+ if (!get_pfnblock_bit(page, pfn, PB_all_free)) {
+ set_pfnblock_bit(page, pfn, PB_all_free);
+ superpageblock_pb_now_free(page);
+ }
}
/*
@@ -1063,7 +1221,8 @@ static inline void __free_one_page(struct page *page,
/*
* When freeing a whole pageblock, clear stale PCP ownership
- * and actual-contents tracking flags up front. The in-loop
+ * and actual-contents tracking flags up front, and mark it
+ * as fully free for superpageblock accounting. The in-loop
* check only fires when sub-pageblock pages merge *up to*
* pageblock_order, not when entering at pageblock_order
* directly.
@@ -2006,6 +2165,20 @@ static __always_inline void page_del_and_expand(struct zone *zone,
{
int nr_pages = 1 << high;
+ /*
+ * If we're splitting a page that spans at least a full pageblock,
+ * the allocated pageblock transitions from fully-free to in-use.
+ * Clear PB_all_free and update superpageblock accounting.
+ */
+ if (high >= pageblock_order) {
+ unsigned long pfn = page_to_pfn(page);
+
+ if (get_pfnblock_bit(page, pfn, PB_all_free)) {
+ clear_pfnblock_bit(page, pfn, PB_all_free);
+ superpageblock_pb_now_used(page);
+ }
+ }
+
__del_page_from_free_list(page, zone, high, migratetype);
nr_pages -= expand(zone, page, low, high, migratetype);
account_freepages(zone, -nr_pages, migratetype);
@@ -2535,6 +2708,25 @@ try_to_claim_block(struct zone *zone, struct page *page,
/* Take ownership for orders >= pageblock_order */
if (current_order >= pageblock_order) {
unsigned int nr_added;
+ unsigned long pb_pfn;
+
+ /*
+ * Clear PB_all_free for pageblocks being claimed.
+ * This path bypasses page_del_and_expand(), so we
+ * must handle the free→used transition here.
+ * Use block_type (the original migratetype) because
+ * that's what was decremented when PB_all_free was set.
+ */
+ for (pb_pfn = page_to_pfn(page);
+ pb_pfn < page_to_pfn(page) + (1 << current_order);
+ pb_pfn += pageblock_nr_pages) {
+ struct page *pb_page = pfn_to_page(pb_pfn);
+
+ if (get_pfnblock_bit(pb_page, pb_pfn, PB_all_free)) {
+ clear_pfnblock_bit(pb_page, pb_pfn, PB_all_free);
+ superpageblock_pb_now_used(pb_page);
+ }
+ }
del_page_from_free_list(page, zone, current_order, block_type);
change_pageblock_range(page, current_order, start_type);
@@ -3651,6 +3843,14 @@ int __isolate_free_page(struct page *page, unsigned int order)
del_page_from_free_list(page, zone, order, mt);
+ /*
+ * The free page is leaving buddy. For order >= pageblock_order, every
+ * constituent pageblock had PB_all_free set; clear those bits and
+ * decrement nr_free so the SPB pageblock-level counter stays in sync.
+ */
+ if (order >= pageblock_order)
+ superpageblock_range_now_used(page, order);
+
/*
* Set the pageblock if the isolated page is at least half of a
* pageblock
@@ -8163,6 +8363,8 @@ unsigned long __offline_isolated_pages(unsigned long start_pfn,
BUG_ON(!PageBuddy(page));
VM_WARN_ON(get_pageblock_migratetype(page) != MIGRATE_ISOLATE);
order = buddy_order(page);
+ if (order >= pageblock_order)
+ superpageblock_range_now_used(page, order);
del_page_from_free_list(page, zone, order, MIGRATE_ISOLATE);
pfn += (1 << order);
}
@@ -8254,6 +8456,25 @@ bool take_page_off_buddy(struct page *page)
del_page_from_free_list(page_head, zone, page_order,
migratetype);
+ /*
+ * break_down_buddy_pages() re-adds every non-target
+ * pageblock to buddy at order >= pageblock_order, so
+ * those keep their PB_all_free state. Only the target's
+ * pageblock loses its fully-free status -- clear that
+ * one bit and decrement the SPB nr_free counter.
+ */
+ if (page_order >= pageblock_order) {
+ unsigned long pfn_pb = ALIGN_DOWN(pfn,
+ pageblock_nr_pages);
+ struct page *pb_page = pfn_to_page(pfn_pb);
+
+ if (get_pfnblock_bit(pb_page, pfn_pb,
+ PB_all_free)) {
+ clear_pfnblock_bit(pb_page, pfn_pb,
+ PB_all_free);
+ superpageblock_pb_now_used(pb_page);
+ }
+ }
break_down_buddy_pages(zone, page_head, page, 0,
page_order, migratetype);
SetPageHWPoisonTakenOff(page);
@@ -8558,3 +8779,73 @@ struct page *alloc_pages_nolock_noprof(gfp_t gfp_flags, int nid, unsigned int or
return page;
}
EXPORT_SYMBOL_GPL(alloc_pages_nolock_noprof);
+
+#ifdef CONFIG_DEBUG_FS
+static const char * const sb_fullness_names[] = {
+ "full", "75pct", "50pct", "25pct", "almost_empty"
+};
+
+static const char * const sb_category_names[] = {
+ "clean", "tainted"
+};
+
+static int superpageblock_debugfs_show(struct seq_file *m, void *v)
+{
+ struct zone *zone;
+ int cat, full;
+
+ for_each_populated_zone(zone) {
+ unsigned long i;
+ int empty_count = 0;
+ struct superpageblock *sb;
+
+ if (!zone->superpageblocks)
+ continue;
+
+ seq_printf(m, "Node %d, zone %8s: %lu superpageblocks, base_pfn=0x%lx\n",
+ zone->zone_pgdat->node_id, zone->name,
+ zone->nr_superpageblocks, zone->superpageblock_base_pfn);
+
+ list_for_each_entry(sb, &zone->spb_empty, list)
+ empty_count++;
+ if (empty_count)
+ seq_printf(m, " empty: %d\n", empty_count);
+
+ for (cat = 0; cat < __NR_SB_CATEGORIES; cat++) {
+ for (full = 0; full < __NR_SB_FULLNESS; full++) {
+ int count = 0;
+
+ list_for_each_entry(sb,
+ &zone->spb_lists[cat][full], list)
+ count++;
+ if (count)
+ seq_printf(m, " %s/%s: %d\n",
+ sb_category_names[cat],
+ sb_fullness_names[full],
+ count);
+ }
+ }
+
+ /* Per-superpageblock detail */
+ for (i = 0; i < zone->nr_superpageblocks; i++) {
+ sb = &zone->superpageblocks[i];
+ seq_printf(m, " sb[%lu] pfn=0x%lx: unmov=%u recl=%u mov=%u rsv=%u free=%u total=%u\n",
+ i, sb->start_pfn,
+ sb->nr_unmovable, sb->nr_reclaimable,
+ sb->nr_movable, sb->nr_reserved,
+ sb->nr_free, sb->total_pageblocks);
+ }
+ }
+ return 0;
+}
+
+DEFINE_SHOW_ATTRIBUTE(superpageblock_debugfs);
+
+static int __init superpageblock_debugfs_init(void)
+{
+ debugfs_create_file("superpageblocks", 0444, NULL, NULL,
+ &superpageblock_debugfs_fops);
+ return 0;
+}
+late_initcall(superpageblock_debugfs_init);
+#endif /* CONFIG_DEBUG_FS */
--
2.54.0
next prev parent reply other threads:[~2026-05-20 15:01 UTC|newest]
Thread overview: 51+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-05-20 14:59 [RFC PATCH 00/40] mm: reliable 1GB page allocation Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 01/40] mm: page_alloc: replace pageblock_flags bitmap with struct pageblock_data Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 02/40] mm: page_alloc: per-cpu pageblock buddy allocator Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 03/40] mm: page_alloc: split-path PCP free with local-trylock + remote-llist Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 04/40] mm: mm_init: fix zone assignment for pages in unavailable ranges Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 05/40] mm: page_alloc: remove watermark boost mechanism Rik van Riel
2026-05-26 14:02 ` Usama Arif
2026-05-20 14:59 ` [RFC PATCH 06/40] mm: page_alloc: async evacuation of stolen movable pageblocks Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 07/40] mm: page_alloc: track actual page contents in pageblock flags Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 08/40] mm: page_alloc: superpageblock metadata for 1GB anti-fragmentation Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 09/40] mm: page_alloc: support superpageblock resize for memory hotplug Rik van Riel
2026-05-20 14:59 ` Rik van Riel [this message]
2026-05-20 14:59 ` [RFC PATCH 11/40] mm: page_alloc: steer pageblock stealing to tainted superpageblocks Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 12/40] mm: page_alloc: steer movable allocations to fullest clean superpageblocks Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 13/40] mm: page_alloc: extract claim_whole_block from try_to_claim_block Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 14/40] mm: page_alloc: add per-superpageblock free lists Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 15/40] mm: page_alloc: add background superpageblock defragmentation worker Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 16/40] mm: compaction: walk per-superpageblock free lists for migration targets Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 17/40] mm: page_alloc: superpageblock-aware contiguous and higher order allocation Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 18/40] mm: page_alloc: prevent atomic allocations from tainting clean SPBs Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 19/40] mm: page_alloc: aggressively pack non-movable allocs in tainted SPBs on large systems Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 20/40] mm: page_alloc: prefer reclaim over tainting clean superpageblocks Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 21/40] mm: page_alloc: adopt partial pageblocks from tainted superpageblocks Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 22/40] mm: page_alloc: add CONFIG_DEBUG_VM sanity checks for SPB counters Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 23/40] mm: page_alloc: targeted evacuation and dynamic reserves for tainted SPBs Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 24/40] mm: page_alloc: prevent UNMOVABLE/RECLAIMABLE mixing in pageblocks Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 25/40] mm: trigger deferred SPB evac when atomic allocs would taint a clean SPB Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 26/40] mm: page_alloc: refuse fragmenting fallback for callers with cheap fallback Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 27/40] mm: page_alloc: cross-migratetype buddy borrow within tainted SPBs Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 28/40] mm: page_alloc: drive slab shrink from SPB anti-fragmentation pressure Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 29/40] mm: page_reporting: walk per-superpageblock free lists Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 30/40] mm: show_mem: collect migratetype letters from per-superpageblock lists Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 31/40] mm: page_alloc: per-(zone, order, mt) PASS_1 hint cache Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 32/40] mm: debug: prevent infinite recursion in dump_page() with CMA Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 33/40] PM: hibernate: walk per-superpageblock free lists in mark_free_pages Rik van Riel
2026-05-20 18:19 ` Rafael J. Wysocki
2026-05-20 14:59 ` [RFC PATCH 34/40] btrfs: allocate eb-attached btree pages as movable Rik van Riel
2026-05-20 17:47 ` Boris Burkov
2026-05-23 15:58 ` David Sterba
2026-05-24 1:43 ` Rik van Riel
2026-05-24 19:59 ` Matthew Wilcox
2026-05-25 6:57 ` Christoph Hellwig
2026-05-20 14:59 ` [RFC PATCH 35/40] mm: page_alloc: refuse best-effort high-order allocs servable at lower orders Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 36/40] mm: page_alloc: set ALLOC_NOFRAGMENT on alloc_frozen_pages_nolock_noprof Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 37/40] mm: page_alloc: move spb_get_category and spb_tainted_reserve to mmzone.h Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 38/40] mm: compaction: skip empty tainted superpageblocks as migration source Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 39/40] mm: compaction: respect tainted SPB reserve in destination selection Rik van Riel
2026-05-20 14:59 ` [RFC PATCH 40/40] mm: page_alloc: SPB tracepoint instrumentation [DO-NOT-MERGE] Rik van Riel
2026-05-21 7:39 ` [syzbot ci] Re: mm: reliable 1GB page allocation syzbot ci
2026-05-22 11:02 ` [RFC PATCH 00/40] " Usama Arif
2026-05-22 13:55 ` Rik van Riel
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20260520150018.2491267-11-riel@surriel.com \
--to=riel@surriel.com \
--cc=david@kernel.org \
--cc=fvdl@google.com \
--cc=hannes@cmpxchg.org \
--cc=kernel-team@meta.com \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-mm@kvack.org \
--cc=ljs@kernel.org \
--cc=surenb@google.com \
--cc=usama.arif@linux.dev \
--cc=willy@infradead.org \
--cc=ziy@nvidia.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox