[<prev] [next>] [thread-next>] [day] [month] [year] [list]
Message-ID: <20251002204636.4016712-1-joshua.hahnjy@gmail.com>
Date: Thu, 2 Oct 2025 13:46:30 -0700
From: Joshua Hahn <joshua.hahnjy@...il.com>
To: Andrew Morton <akpm@...ux-foundation.org>,
Johannes Weiner <hannes@...xchg.org>
Cc: Chris Mason <clm@...com>,
Kiryl Shutsemau <kirill@...temov.name>,
"Liam R. Howlett" <Liam.Howlett@...cle.com>,
Brendan Jackman <jackmanb@...gle.com>,
David Hildenbrand <david@...hat.com>,
Lorenzo Stoakes <lorenzo.stoakes@...cle.com>,
Michal Hocko <mhocko@...e.com>,
Mike Rapoport <rppt@...nel.org>,
Suren Baghdasaryan <surenb@...gle.com>,
Vlastimil Babka <vbabka@...e.cz>,
Zi Yan <ziy@...dia.com>,
Hillf Danton <hdanton@...a.com>,
linux-kernel@...r.kernel.org,
linux-mm@...ck.org,
kernel-team@...a.com
Subject: [PATCH v3 0/3] mm/page_alloc: Batch callers of free_pcppages_bulk
Motivation & Approach
=====================
While testing workloads with high sustained memory pressure on large machines
in the Meta fleet (1Tb memory, 316 CPUs), we saw an unexpectedly high number
of softlockups. Further investigation showed that the zone lock in
free_pcppages_bulk was being held for a long time, and was called to free
2k+ pages over 100 times just during boot.
This causes starvation in other processes for the zone lock, which can lead
to the system stalling as multiple threads cannot make progress without the
locks. We can see these issues manifesting as warnings:
[ 4512.591979] rcu: INFO: rcu_sched self-detected stall on CPU
[ 4512.604370] rcu: 20-....: (9312 ticks this GP) idle=a654/1/0x4000000000000000 softirq=309340/309344 fqs=5426
[ 4512.626401] rcu: hardirqs softirqs csw/system
[ 4512.638793] rcu: number: 0 145 0
[ 4512.651177] rcu: cputime: 30 10410 174 ==> 10558(ms)
[ 4512.666657] rcu: (t=21077 jiffies g=783665 q=1242213 ncpus=316)
While these warnings are benign, they do point to the underlying issue of
lock contention. To prevent starvation in both locks, batch the freeing of
pages using pcp->batch.
Because free_pcppages_bulk is called with both the pcp and zone lock,
relinquishing and reacquiring the locks are only effective when both of them
are broken together (unless the system was built with queued spinlocks).
Thus, instead of modifying free_pcppages_bulk to break both locks, batch the
freeing from its callers instead.
A similar fix has been implemented in the Meta fleet, and we have seen
significantly less softlockups.
Testing
=======
The following are a few synthetic benchmarks, made on two machines. The
first is a large, single-node machine with 754GiB memory and 316 processors.
The second is a relatively smaller single-node machine with 251GiB memory
and 176 processors.
On both machines, I kick off a kernel build with -j$(nproc).
Lower delta is better (faster compilation).
Large machine (754GiB memory, 316 processors)
make -j$(nproc)
+------------+---------------+----------+
| Metric (s) | Variation (%) | Delta(%) |
+------------+---------------+----------+
| real | 0.4627 | -1.2627 |
| user | 0.2281 | +0.2680 |
| sys | 4.6345 | -7.5425 |
+------------+---------------+----------+
Medium machine (251GiB memory, 176 processors)
make -j$(nproc)
+------------+---------------+----------+
| Metric (s) | Variation (%) | Delta(%) |
+------------+---------------+----------+
| real | 0.2321 | +0.0888 |
| user | 0.1730 | -0.1182 |
| sys | 0.7680 | +1.2067 |
+------------+---------------+----------+
Small machine (62GiB memory, 36 processors)
make -j$(nproc)
+------------+---------------+----------+
| Metric (s) | Variation (%) | Delta(%) |
+------------+---------------+----------+
| real | 0.1920 | -0.1270 |
| user | 0.1730 | -0.0358 |
| sys | 0.7680 | +0.9143 |
+------------+---------------+----------+
Here, variation is the coefficient of variation, i.e. standard deviation / mean.
Based on these results, there is definitely some gain to be had when
lock contention is observed in a larger machine, especially if it is running
on one node. It leads to both a measurable decrease in compilation time for
both the real and system times (i.e. relieving lock contention).
For the medium machine, there is negligible regression in real time
(<< coefficient of variation), although it leads to a measurable increase
in the system time.
For the small machine, there is a negligible performance gain in real time,
but has a similar regression to the medium machine.
Despite the regressions (~1%) of system time in the smaller machines, it
seems to be (1) not observable in realtime and (2) is much smaller than the
gain made in the large machine.
Changelog
=========
v2 --> v3:
- Refactored on top of mm-new
- Wordsmithing the cover letter & commit messages to clarify which lock
is contended, as suggested by Hillf Danton.
- Ran new tests for the cover letter, instead of running stress-ng, I decided
to compile the kernel which I think will be more reflective of the "default"
workload that might be run. Also ran on a smaller machines to show the
expected behavior of this patchset when there is lock contention vs.
lower lock contention.
- Removed patch 2/4, which would have batched page freeing for
drain_pages_zone. It is not a good candidate for this series since it is
called on each CPU in __drain_all_pages.
- Small change in 1/4 to initialize todo, as suggested by Christoph Lameter
- Small change in 1/4 to avoid bit manipulation, as suggested by SeongJae Park.
- Change in 4/4 to handle the case when the thread gets migrated to a different
CPU during the window between unlocking & reacquiring the pcp lock, as
suggested by Vlastimil Babka.
- Small change in 4/4 to handle the case when pcp lock could not be acquired
within the loop in free_unref_folios.
v1 --> v2:
- Reworded cover letter to be more explicit about what kinds of issues
running processes might face as a result of the existing lock starvation
- Reworded cover letter to be in sections to make it easier to read
- Fixed patch 4/4 to properly store & restore UP flags.
- Re-ran tests, updated the testing results and interpretation
Joshua Hahn (3):
mm/page_alloc/vmstat: Simplify refresh_cpu_vm_stats change detection
mm/page_alloc: Batch page freeing in decay_pcp_high
mm/page_alloc: Batch page freeing in free_frozen_page_commit
include/linux/gfp.h | 2 +-
mm/page_alloc.c | 83 ++++++++++++++++++++++++++++++++++++---------
mm/vmstat.c | 28 ++++++++-------
3 files changed, 83 insertions(+), 30 deletions(-)
base-commit: 71fffcaf9c5c5eb17f90a8db478586091cd300c5
--
2.47.3
Powered by blists - more mailing lists