[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <2b4bb9ce-fe92-47fe-9185-e29571502ae9@arm.com>
Date: Thu, 21 Dec 2023 08:35:40 +0000
From: Metin Kaya <metin.kaya@....com>
To: John Stultz <jstultz@...gle.com>, LKML <linux-kernel@...r.kernel.org>
Cc: Joel Fernandes <joelaf@...gle.com>, Qais Yousef <qyousef@...gle.com>,
Ingo Molnar <mingo@...hat.com>, Peter Zijlstra <peterz@...radead.org>,
Juri Lelli <juri.lelli@...hat.com>,
Vincent Guittot <vincent.guittot@...aro.org>,
Dietmar Eggemann <dietmar.eggemann@....com>,
Valentin Schneider <vschneid@...hat.com>,
Steven Rostedt <rostedt@...dmis.org>, Ben Segall <bsegall@...gle.com>,
Zimuzo Ezeozue <zezeozue@...gle.com>, Youssef Esmat
<youssefesmat@...gle.com>, Mel Gorman <mgorman@...e.de>,
Daniel Bristot de Oliveira <bristot@...hat.com>,
Will Deacon <will@...nel.org>, Waiman Long <longman@...hat.com>,
Boqun Feng <boqun.feng@...il.com>, "Paul E. McKenney" <paulmck@...nel.org>,
Xuewen Yan <xuewen.yan94@...il.com>, K Prateek Nayak
<kprateek.nayak@....com>, Thomas Gleixner <tglx@...utronix.de>,
kernel-team@...roid.com
Subject: Re: [PATCH v7 00/23] Proxy Execution: A generalized form of Priority
Inheritance v7
On 20/12/2023 12:18 am, John Stultz wrote:
> Looking for patches to review before closing out the year?
> Have I got something for you! :)
>
> Since sending v6 out and after my presentation at Linux
> Plumbers[1], I got some nice feedback from a number of
> developers. I particularly want to thank Metin Kaya, who provided
> a ton of detailed off-list review comments, questions and cleanup
> suggestions. Additionally Xuewen Yan, who pointed out an issue
> with placement in my return migration logic that motivated me to
> rework the return migration back into the ttwu path. This helped
> to resolve the performance regression I had been seeing from the
> v4->v5 rework!
>
> The other focus of this revision has been to properly
> conditionalize and re-integrate the chain-migration feature that
> Connor had implemented to address concerns about the preserving
> the RT load balancing invariant (always running top N priority
> tasks across N available cpus) when we’re migrating tasks around
> for proxy-execution. I’ve done a fair amount of reworking the
> patches, but they aren’t quite yet to where I’d like them, but
> I’ve included them here which you can consider as a preview.
>
> Validating this invariant isn’t trivial. Long ago I wrote a
> userland test case that’s now in LTP to check this called
> sched_football:
> https://github.com/linux-test-project/ltp/blob/master/testcases/realtime/func/sched_football/sched_football.c
>
> So I’ve reimplemented this idea as an in-kernel test case,
> extended to have lock chains across priority levels. The good
> news is that the test case does not show any issues with RT load
> balancing invariant when we add the chain-migration feature, but
> I’m not actually seeing any issues with the test when
> chain-migration isn’t added either, so I need to further extend
> the test to try to manufacture the specific type of invariant
> breakage I can imagine we don’t handle properly:
> ie:
> CPU0: P99, P98(boP2), P2
> CPU1: P50
>
> Which chain-migration should adjust to become:
> CPU0: P99
> CPU1: P98(boP2), P50, P2
>
> On the stability front, the series has continued to fare much
> better than pre-v6 patches. The only stability issue I had seen
> with v6 (workqueue lockups when stressing under 64 core qemu
> environment with KVM disabled) has so far not reproduced against
> 6.7-rc. At plumbers Thomas mentioned there had been a workqueue
> issue in 6.6 that was since fixed, so I’m optimistic that was
> what I was tripping on. If you run into any stability issues in
> testing, please do let me know.
>
> This patch series is actually coarser than what I’ve been
> developing with, as there are a number of small “test” steps to
> help validate behavior I changed, which would then be replaced by
> the real logic afterwards. Including those here would just cause
> more work for reviewers, so I’ve folded them together, but if
> you’re interested you can find the fine-grained tree here:
> https://github.com/johnstultz-work/linux-dev/commits/proxy-exec-v7-6.7-rc6-fine-grained
> https://github.com/johnstultz-work/linux-dev.git proxy-exec-v7-6.7-rc6-fine-grained
>
> Credit/Disclaimer:
> —--------------------
> As mentioned previously, this Proxy Execution series has a long
> history:
>
> First described in a paper[2] by Watkins, Straub, Niehaus, then
> from patches from Peter Zijlstra, extended with lots of work by
> Juri Lelli, Valentin Schneider, and Connor O'Brien. (and thank
> you to Steven Rostedt for providing additional details here!)
>
> So again, many thanks to those above, as all the credit for this
> series really is due to them - while the mistakes are likely
> mine.
>
> Overview:
> —----------
> Proxy Execution is a generalized form of priority inheritance.
> Classic priority inheritance works well for real-time tasks where
> there is a straight forward priority order to how things are run.
> But it breaks down when used between CFS or DEADLINE tasks, as
> there are lots of parameters involved outside of just the task’s
> nice value when selecting the next task to run (via
> pick_next_task()). So ideally we want to imbue the mutex holder
> with all the scheduler attributes of the blocked waiting task.
>
> Proxy Execution does this via a few changes:
> * Keeping tasks that are blocked on a mutex *on* the runqueue
> * Keeping additional tracking of which mutex a task is blocked
> on, and which task holds a specific mutex.
> * Special handling for when we select a blocked task to run, so
> that we instead run the mutex holder.
>
> By leaving blocked tasks on the runqueue, we allow
> pick_next_task() to choose the task that should run next (even if
> it’s blocked waiting on a mutex). If we do select a blocked task,
> we look at the task’s blocked_on mutex and from there look at the
> mutex’s owner task. And in the simple case, the task which owns
> the mutex is what we then choose to run, allowing it to release
> the mutex.
>
> This means that instead of just tracking “curr”, the scheduler
> needs to track both the scheduler context (what was picked and
> all the state used for scheduling decisions), and the execution
> context (what we’re actually running).
>
> In this way, the mutex owner is run “on behalf” of the blocked
> task that was picked to run, essentially inheriting the scheduler
> context of the waiting blocked task.
>
> As Connor outlined in a previous submission of this patch series,
Nit: Better to have a reference to Connor's patch series (i.e.,
https://lore.kernel.org/lkml/20221003214501.2050087-1-connoro@google.com/)
here?
> this raises a number of complicated situations: The mutex owner
> might itself be blocked on another mutex, or it could be
> sleeping, running on a different CPU, in the process of migrating
> between CPUs, etc.
>
> The complexity from these edge cases are imposing, but currently
> in Android we have a large number of cases where we are seeing
> priority inversion (not unbounded, but much longer than we’d
> like) between “foreground” and “background” SCHED_NORMAL
> applications. As a result, currently we cannot usefully limit
> background activity without introducing inconsistent behavior. So
> Proxy Execution is a very attractive approach to resolving this
> critical issue.
>
> New in v7:
> ---------
> * Extended blocked_on state tracking to use a tri-state so we
> can avoid running tasks before they are return-migrated.
>
> * Switched return migration back to the ttwu path to avoid extra
> lock handling and resolve performance regression seen since v5
>
> * _Tons_ of typos, cleanups, runtime conditionalization
> improvements, build fixes for different config options, and
> clarifications suggested by Metin Kaya. *Many* *many* thanks
> for all the review and help here!
>
> * Split up and reworked Connor’s chain-migration patches to be
> properly conditionalized on the config value, and hopefully a
> little easier to review.
>
> * Added first stab at RT load-balancing invariant test
> (sched_football)
>
> * Fixed wake_cpu not being preserved during
> activate_blocked_entities
>
> * Fix build warnings Reported-by: kernel test robot <lkp@...el.com>
>
> * Other minor cleanups
>
> Performance:
> —----------
> v7 of the series improves over v6 and v5 by moving proxy return
> migration back into the ttwu path, which avoids a lot of extra
> locking. This gets performance back to where we were in v4.
>
> K Prateek Nayak did some benchmarking and performance analysis
> with the v6 series and while he noted with a number of benchmarks
> (schbench, tbench, netperf, ycsb-mongodb, DeathStarBench) he
> found “little to no difference” when running with
> proxy-execution[3], he did find a large regression with the
> “perf bench sched messaging” test.
>
> I’ve reproduced this issue and the majority of the regression
> seems to come from the fact this patch series switches mutexes to
> use handoff mode rather than optimistic spinning. This has been a
> concern for cases where locks are under high contention, so I
> need to spend some time on finding a solution to restore some
> ability for optimistic spinning. Many thanks to Preteek for
> raising this issue!
>
> Previously Chenyu also report a regression[4], which seems
> similar, but I need to look further into it.
>
> Issues still to address:
> —----------
> * The chain migration patches still need further iterations and
> better validation to ensure they resolve the RT/DL load
> balancing invariants.
>
> * Xuewen Yan earlier pointed out that we may see task
> mis-placement on EAS systems if we do return migration based
> strictly on cpu allowability. I tried an optimization to
> always try to return migrate to the wake_cpu (which was saved
> on proxy-migration), but this seemed to undo a chunk of the
> benefit I saw in moving return migration back to ttwu, at
> least with my prio-inversion-demo microbenchmark. Need to do
> some broader performance analysis with the variations there.
>
> * Optimization to avoid migrating blocked tasks (to preserve
> optimistic mutex spinning) if the runnable lock-owner at the
> end of the blocked_on chain is already running (though this is
> difficult due to the limitations from locking rules needed to
> traverse the blocked on chain across run queues).
>
> * Similarly as we’re often dealing with lists of tasks or chains
> of tasks and mutexes, iterating across these chains of objects
> can be done safely while holding the rq lock, but as these
> chains can cross runqueues our ability to traverse the chains
> safely is somewhat limited.
>
> * CFS load balancing. Blocked tasks may carry forward load
> (PELT) to the lock owner's CPU, so CPU may look like it is
> overloaded.
>
> * The sleeping owner handling (where we deactivate waiting tasks
> and enqueue them onto a list, then reactivate them when the
> owner wakes up) doesn’t feel great. This is in part because
> when we want to activate tasks, we’re already holding a
> task.pi_lock and a rq_lock, just not the locks for the task
> we’re activating, nor the rq we’re enqueuing it onto. So there
> has to be a bit of lock juggling to drop and acquire the right
> locks (in the right order). It feels like there’s got to be a
> better way.
>
> * “rq_selected()” naming. Peter doesn’t like it, but I’ve not
> thought of a better name. Open to suggestions.
>
> * As discussed at OSPM[5], I like to split pick_next_task() up
> into two phases selecting and setting the next tasks, as
> currently pick_next_task() assumes the returned task will be
> run which results in various side-effects in sched class logic
> when it’s run. I tried to take a pass at this earlier, but
> it’s hairy and lower on the priority list for now.
Do you think we should mention virtual runqueue idea and adding trace
points to measure task migration times? They are not "open issues", but
kind of to-do items in the agenda.
>
>
> If folks find it easier to test/tinker with, this patch series
> can also be found here:
> https://github.com/johnstultz-work/linux-dev/commits/proxy-exec-v7-6.7-rc6
> https://github.com/johnstultz-work/linux-dev.git proxy-exec-v7-6.7-rc6
>
> Feedback would be very welcome!
>
> Thanks so much!
> -john
>
> [1] https://lwn.net/Articles/953438/
> [2] https://static.lwn.net/images/conf/rtlws11/papers/proc/p38.pdf
> [3] https://lore.kernel.org/lkml/c6787831-f659-12cb-4954-fd13a05ed590@amd.com/
> [4] https://lore.kernel.org/lkml/Y7vVqE0M%2FAoDoVbj@chenyu5-mobl1/
> [5] https://youtu.be/QEWqRhVS3lI (video of my OSPM talk)
>
>
> Cc: Joel Fernandes <joelaf@...gle.com>
> Cc: Qais Yousef <qyousef@...gle.com>
> Cc: Ingo Molnar <mingo@...hat.com>
> Cc: Peter Zijlstra <peterz@...radead.org>
> Cc: Juri Lelli <juri.lelli@...hat.com>
> Cc: Vincent Guittot <vincent.guittot@...aro.org>
> Cc: Dietmar Eggemann <dietmar.eggemann@....com>
> Cc: Valentin Schneider <vschneid@...hat.com>
> Cc: Steven Rostedt <rostedt@...dmis.org>
> Cc: Ben Segall <bsegall@...gle.com>
> Cc: Zimuzo Ezeozue <zezeozue@...gle.com>
> Cc: Youssef Esmat <youssefesmat@...gle.com>
> Cc: Mel Gorman <mgorman@...e.de>
> Cc: Daniel Bristot de Oliveira <bristot@...hat.com>
> Cc: Will Deacon <will@...nel.org>
> Cc: Waiman Long <longman@...hat.com>
> Cc: Boqun Feng <boqun.feng@...il.com>
> Cc: "Paul E. McKenney" <paulmck@...nel.org>
> Cc: Metin Kaya <Metin.Kaya@....com>
> Cc: Xuewen Yan <xuewen.yan94@...il.com>
> Cc: K Prateek Nayak <kprateek.nayak@....com>
> Cc: Thomas Gleixner <tglx@...utronix.de>
> Cc: kernel-team@...roid.com
>
> Connor O'Brien (5):
> sched: Add push_task_chain helper
> sched: Consolidate pick_*_task to task_is_pushable helper
> sched: Push execution and scheduler context split into deadline and rt
> paths
> sched: Add find_exec_ctx helper
> sched: Fix rt/dl load balancing via chain level balance
>
> John Stultz (8):
> locking/mutex: Remove wakeups from under mutex::wait_lock
> sched: Add CONFIG_SCHED_PROXY_EXEC & boot argument to enable/disable
> sched: Fix runtime accounting w/ split exec & sched contexts
> sched: Split out __sched() deactivate task logic into a helper
> sched: Add a initial sketch of the find_proxy_task() function
> sched: Handle blocked-waiter migration (and return migration)
> sched: Initial sched_football test implementation
> sched: Refactor dl/rt find_lowest/latest_rq logic
>
> Juri Lelli (2):
> locking/mutex: Make mutex::wait_lock irq safe
> locking/mutex: Expose __mutex_owner()
>
> Peter Zijlstra (7):
> sched: Unify runtime accounting across classes
> locking/mutex: Rework task_struct::blocked_on
> locking/mutex: Switch to mutex handoffs for CONFIG_SCHED_PROXY_EXEC
> sched: Split scheduler and execution contexts
> sched: Start blocked_on chain processing in find_proxy_task()
> sched: Add blocked_donor link to task for smarter mutex handoffs
> sched: Add deactivated (sleeping) owner handling to find_proxy_task()
>
> Valentin Schneider (1):
> sched: Fix proxy/current (push,pull)ability
>
> .../admin-guide/kernel-parameters.txt | 5 +
> Documentation/locking/mutex-design.rst | 3 +
> include/linux/sched.h | 59 +-
> init/Kconfig | 7 +
> init/init_task.c | 1 +
> kernel/fork.c | 9 +-
> kernel/locking/mutex-debug.c | 9 +-
> kernel/locking/mutex.c | 166 ++--
> kernel/locking/mutex.h | 25 +
> kernel/locking/rtmutex.c | 26 +-
> kernel/locking/rwbase_rt.c | 4 +-
> kernel/locking/rwsem.c | 4 +-
> kernel/locking/spinlock_rt.c | 3 +-
> kernel/locking/ww_mutex.h | 73 +-
> kernel/sched/Makefile | 1 +
> kernel/sched/core.c | 822 ++++++++++++++++--
> kernel/sched/cpudeadline.c | 12 +-
> kernel/sched/cpudeadline.h | 3 +-
> kernel/sched/cpupri.c | 31 +-
> kernel/sched/cpupri.h | 6 +-
> kernel/sched/deadline.c | 218 +++--
> kernel/sched/fair.c | 102 ++-
> kernel/sched/rt.c | 298 +++++--
> kernel/sched/sched.h | 105 ++-
> kernel/sched/stop_task.c | 13 +-
> kernel/sched/test_sched_football.c | 242 ++++++
> lib/Kconfig.debug | 14 +
> 27 files changed, 1861 insertions(+), 400 deletions(-)
> create mode 100644 kernel/sched/test_sched_football.c
>
Powered by blists - more mailing lists