From: Aaron Lu <ziqianlu@bytedance.com>
To: Florian Bezdeka <florian.bezdeka@siemens.com>
Cc: Valentin Schneider <vschneid@redhat.com>,
Ben Segall <bsegall@google.com>,
K Prateek Nayak <kprateek.nayak@amd.com>,
Peter Zijlstra <peterz@infradead.org>,
Josh Don <joshdon@google.com>, Ingo Molnar <mingo@redhat.com>,
Vincent Guittot <vincent.guittot@linaro.org>,
Xi Wang <xii@google.com>,
linux-kernel@vger.kernel.org, Juri Lelli <juri.lelli@redhat.com>,
Dietmar Eggemann <dietmar.eggemann@arm.com>,
Steven Rostedt <rostedt@goodmis.org>,
Mel Gorman <mgorman@suse.de>,
Chengming Zhou <chengming.zhou@linux.dev>,
Chuyi Zhou <zhouchuyi@bytedance.com>,
Jan Kiszka <jan.kiszka@siemens.com>
Subject: Re: [RFC PATCH v2 2/7] sched/fair: Handle throttle path for task based throttle
Date: Mon, 14 Apr 2025 20:10:38 +0800 [thread overview]
Message-ID: <20250414121038.GD3558904@bytedance> (raw)
In-Reply-To: <d584dfb39a391143e4b0006f5a22903899d726b3.camel@siemens.com>
On Mon, Apr 14, 2025 at 10:54:59AM +0200, Florian Bezdeka wrote:
> On Wed, 2025-04-09 at 20:07 +0800, Aaron Lu wrote:
> > From: Valentin Schneider <vschneid@redhat.com>
> >
> > In current throttle model, when a cfs_rq is throttled, its entity will
> > be dequeued from cpu's rq, making tasks attached to it not able to run,
> > thus achiveing the throttle target.
> >
> > This has a drawback though: assume a task is a reader of percpu_rwsem
> > and is waiting. When it gets wakeup, it can not run till its task group's
> > next period comes, which can be a relatively long time. Waiting writer
> > will have to wait longer due to this and it also makes further reader
> > build up and eventually trigger task hung.
> >
> > To improve this situation, change the throttle model to task based, i.e.
> > when a cfs_rq is throttled, record its throttled status but do not
> > remove it from cpu's rq. Instead, for tasks that belong to this cfs_rq,
> > when they get picked, add a task work to them so that when they return
> > to user, they can be dequeued. In this way, tasks throttled will not
> > hold any kernel resources.
> >
> > Signed-off-by: Valentin Schneider <vschneid@redhat.com>
> > Signed-off-by: Aaron Lu <ziqianlu@bytedance.com>
> > ---
> > kernel/sched/fair.c | 185 +++++++++++++++++++++----------------------
> > kernel/sched/sched.h | 1 +
> > 2 files changed, 93 insertions(+), 93 deletions(-)
> >
> > diff --git a/kernel/sched/fair.c b/kernel/sched/fair.c
> > index 894202d232efd..c566a5a90d065 100644
> > --- a/kernel/sched/fair.c
> > +++ b/kernel/sched/fair.c
> > @@ -5516,8 +5516,11 @@ dequeue_entity(struct cfs_rq *cfs_rq, struct sched_entity *se, int flags)
> > if (flags & DEQUEUE_DELAYED)
> > finish_delayed_dequeue_entity(se);
> >
> > - if (cfs_rq->nr_queued == 0)
> > + if (cfs_rq->nr_queued == 0) {
> > update_idle_cfs_rq_clock_pelt(cfs_rq);
> > + if (throttled_hierarchy(cfs_rq))
> > + list_del_leaf_cfs_rq(cfs_rq);
> > + }
> >
> > return true;
> > }
> > @@ -5598,7 +5601,7 @@ pick_next_entity(struct rq *rq, struct cfs_rq *cfs_rq)
> > return se;
> > }
> >
> > -static bool check_cfs_rq_runtime(struct cfs_rq *cfs_rq);
> > +static void check_cfs_rq_runtime(struct cfs_rq *cfs_rq);
> >
> > static void put_prev_entity(struct cfs_rq *cfs_rq, struct sched_entity *prev)
> > {
> > @@ -5823,8 +5826,48 @@ static inline int throttled_lb_pair(struct task_group *tg,
> > throttled_hierarchy(dest_cfs_rq);
> > }
> >
> > +static bool dequeue_task_fair(struct rq *rq, struct task_struct *p, int flags);
> > static void throttle_cfs_rq_work(struct callback_head *work)
> > {
> > + struct task_struct *p = container_of(work, struct task_struct, sched_throttle_work);
> > + struct sched_entity *se;
> > + struct cfs_rq *cfs_rq;
> > + struct rq *rq;
> > +
> > + WARN_ON_ONCE(p != current);
> > + p->sched_throttle_work.next = &p->sched_throttle_work;
> > +
> > + /*
> > + * If task is exiting, then there won't be a return to userspace, so we
> > + * don't have to bother with any of this.
> > + */
> > + if ((p->flags & PF_EXITING))
> > + return;
> > +
> > + scoped_guard(task_rq_lock, p) {
> > + se = &p->se;
> > + cfs_rq = cfs_rq_of(se);
> > +
> > + /* Raced, forget */
> > + if (p->sched_class != &fair_sched_class)
> > + return;
> > +
> > + /*
> > + * If not in limbo, then either replenish has happened or this
> > + * task got migrated out of the throttled cfs_rq, move along.
> > + */
> > + if (!cfs_rq->throttle_count)
> > + return;
> > +
> > + rq = scope.rq;
> > + update_rq_clock(rq);
> > + WARN_ON_ONCE(!list_empty(&p->throttle_node));
> > + dequeue_task_fair(rq, p, DEQUEUE_SLEEP | DEQUEUE_SPECIAL);
> > + list_add(&p->throttle_node, &cfs_rq->throttled_limbo_list);
> > + resched_curr(rq);
> > + }
> > +
> > + cond_resched_tasks_rcu_qs();
> > }
> >
> > void init_cfs_throttle_work(struct task_struct *p)
> > @@ -5864,32 +5907,53 @@ static int tg_unthrottle_up(struct task_group *tg, void *data)
> > return 0;
> > }
> >
> > +static inline bool task_has_throttle_work(struct task_struct *p)
> > +{
> > + return p->sched_throttle_work.next != &p->sched_throttle_work;
> > +}
> > +
> > +static inline void task_throttle_setup_work(struct task_struct *p)
> > +{
> > + if (task_has_throttle_work(p))
> > + return;
> > +
> > + /*
> > + * Kthreads and exiting tasks don't return to userspace, so adding the
> > + * work is pointless
> > + */
> > + if ((p->flags & (PF_EXITING | PF_KTHREAD)))
> > + return;
> > +
> > + task_work_add(p, &p->sched_throttle_work, TWA_RESUME);
> > +}
> > +
> > static int tg_throttle_down(struct task_group *tg, void *data)
> > {
> > struct rq *rq = data;
> > struct cfs_rq *cfs_rq = tg->cfs_rq[cpu_of(rq)];
> >
> > + cfs_rq->throttle_count++;
> > + if (cfs_rq->throttle_count > 1)
> > + return 0;
> > +
> > /* group is entering throttled state, stop time */
> > - if (!cfs_rq->throttle_count) {
> > - cfs_rq->throttled_clock_pelt = rq_clock_pelt(rq);
> > - list_del_leaf_cfs_rq(cfs_rq);
> > + cfs_rq->throttled_clock_pelt = rq_clock_pelt(rq);
> >
> > - WARN_ON_ONCE(cfs_rq->throttled_clock_self);
> > - if (cfs_rq->nr_queued)
> > - cfs_rq->throttled_clock_self = rq_clock(rq);
> > - }
> > - cfs_rq->throttle_count++;
> > + WARN_ON_ONCE(cfs_rq->throttled_clock_self);
> > + if (cfs_rq->nr_queued)
> > + cfs_rq->throttled_clock_self = rq_clock(rq);
> > + else
> > + list_del_leaf_cfs_rq(cfs_rq);
> >
> > + WARN_ON_ONCE(!list_empty(&cfs_rq->throttled_limbo_list));
> > return 0;
> > }
>
> tg_throttle_down() is touched twice in this series. Some code added
> here (as part of patch 2) is later removed again in patch 7.
>
> Maybe there is some room for improvement...
Yes.
So the purpose of patch7 is to show an alternative accounting of this
new per-task throttle model and since we haven't decided the proper way
to do accounting yet so I chose to separate it out. Another rationale
is, I want to keep the core of the patchset(patch2 and patch3) as simple
as possible to ease reviewing. Does this make sense? If folding them is
better, I can do that too for next version.
> >
> > -static bool throttle_cfs_rq(struct cfs_rq *cfs_rq)
> > +static void throttle_cfs_rq(struct cfs_rq *cfs_rq)
> > {
> > struct rq *rq = rq_of(cfs_rq);
> > struct cfs_bandwidth *cfs_b = tg_cfs_bandwidth(cfs_rq->tg);
> > - struct sched_entity *se;
> > - long queued_delta, runnable_delta, idle_delta, dequeue = 1;
> > - long rq_h_nr_queued = rq->cfs.h_nr_queued;
> > + int dequeue = 1;
> >
> > raw_spin_lock(&cfs_b->lock);
> > /* This will start the period timer if necessary */
> > @@ -5910,74 +5974,13 @@ static bool throttle_cfs_rq(struct cfs_rq *cfs_rq)
> > raw_spin_unlock(&cfs_b->lock);
> >
> > if (!dequeue)
> > - return false; /* Throttle no longer required. */
> > -
> > - se = cfs_rq->tg->se[cpu_of(rq_of(cfs_rq))];
> > + return; /* Throttle no longer required. */
> >
> > /* freeze hierarchy runnable averages while throttled */
> > rcu_read_lock();
> > walk_tg_tree_from(cfs_rq->tg, tg_throttle_down, tg_nop, (void *)rq);
> > rcu_read_unlock();
> >
> > - queued_delta = cfs_rq->h_nr_queued;
> > - runnable_delta = cfs_rq->h_nr_runnable;
> > - idle_delta = cfs_rq->h_nr_idle;
> > - for_each_sched_entity(se) {
> > - struct cfs_rq *qcfs_rq = cfs_rq_of(se);
> > - int flags;
> > -
> > - /* throttled entity or throttle-on-deactivate */
> > - if (!se->on_rq)
> > - goto done;
> > -
> > - /*
> > - * Abuse SPECIAL to avoid delayed dequeue in this instance.
> > - * This avoids teaching dequeue_entities() about throttled
> > - * entities and keeps things relatively simple.
> > - */
> > - flags = DEQUEUE_SLEEP | DEQUEUE_SPECIAL;
> > - if (se->sched_delayed)
> > - flags |= DEQUEUE_DELAYED;
> > - dequeue_entity(qcfs_rq, se, flags);
> > -
> > - if (cfs_rq_is_idle(group_cfs_rq(se)))
> > - idle_delta = cfs_rq->h_nr_queued;
> > -
> > - qcfs_rq->h_nr_queued -= queued_delta;
> > - qcfs_rq->h_nr_runnable -= runnable_delta;
> > - qcfs_rq->h_nr_idle -= idle_delta;
> > -
> > - if (qcfs_rq->load.weight) {
> > - /* Avoid re-evaluating load for this entity: */
> > - se = parent_entity(se);
> > - break;
> > - }
> > - }
> > -
> > - for_each_sched_entity(se) {
> > - struct cfs_rq *qcfs_rq = cfs_rq_of(se);
> > - /* throttled entity or throttle-on-deactivate */
> > - if (!se->on_rq)
> > - goto done;
> > -
> > - update_load_avg(qcfs_rq, se, 0);
> > - se_update_runnable(se);
> > -
> > - if (cfs_rq_is_idle(group_cfs_rq(se)))
> > - idle_delta = cfs_rq->h_nr_queued;
> > -
> > - qcfs_rq->h_nr_queued -= queued_delta;
> > - qcfs_rq->h_nr_runnable -= runnable_delta;
> > - qcfs_rq->h_nr_idle -= idle_delta;
> > - }
> > -
> > - /* At this point se is NULL and we are at root level*/
> > - sub_nr_running(rq, queued_delta);
> > -
> > - /* Stop the fair server if throttling resulted in no runnable tasks */
> > - if (rq_h_nr_queued && !rq->cfs.h_nr_queued)
> > - dl_server_stop(&rq->fair_server);
> > -done:
> > /*
> > * Note: distribution will already see us throttled via the
> > * throttled-list. rq->lock protects completion.
> > @@ -5986,7 +5989,7 @@ static bool throttle_cfs_rq(struct cfs_rq *cfs_rq)
> > WARN_ON_ONCE(cfs_rq->throttled_clock);
> > if (cfs_rq->nr_queued)
> > cfs_rq->throttled_clock = rq_clock(rq);
> > - return true;
> > + return;
>
> Obsolete now, could be removed.
Indeed and one less line of code :-)
Thanks,
Aaron
> > }
> >
> > void unthrottle_cfs_rq(struct cfs_rq *cfs_rq)
> > @@ -6462,22 +6465,22 @@ static void sync_throttle(struct task_group *tg, int cpu)
> > }
> >
> > /* conditionally throttle active cfs_rq's from put_prev_entity() */
> > -static bool check_cfs_rq_runtime(struct cfs_rq *cfs_rq)
> > +static void check_cfs_rq_runtime(struct cfs_rq *cfs_rq)
> > {
> > if (!cfs_bandwidth_used())
> > - return false;
> > + return;
> >
> > if (likely(!cfs_rq->runtime_enabled || cfs_rq->runtime_remaining > 0))
> > - return false;
> > + return;
> >
> > /*
> > * it's possible for a throttled entity to be forced into a running
> > * state (e.g. set_curr_task), in this case we're finished.
> > */
> > if (cfs_rq_throttled(cfs_rq))
> > - return true;
> > + return;
> >
> > - return throttle_cfs_rq(cfs_rq);
> > + throttle_cfs_rq(cfs_rq);
> > }
> >
> > static enum hrtimer_restart sched_cfs_slack_timer(struct hrtimer *timer)
> > @@ -6573,6 +6576,7 @@ static void init_cfs_rq_runtime(struct cfs_rq *cfs_rq)
> > cfs_rq->runtime_enabled = 0;
> > INIT_LIST_HEAD(&cfs_rq->throttled_list);
> > INIT_LIST_HEAD(&cfs_rq->throttled_csd_list);
> > + INIT_LIST_HEAD(&cfs_rq->throttled_limbo_list);
> > }
> >
> > void start_cfs_bandwidth(struct cfs_bandwidth *cfs_b)
> > @@ -6738,10 +6742,11 @@ static void sched_fair_update_stop_tick(struct rq *rq, struct task_struct *p)
> > #else /* CONFIG_CFS_BANDWIDTH */
> >
> > static void account_cfs_rq_runtime(struct cfs_rq *cfs_rq, u64 delta_exec) {}
> > -static bool check_cfs_rq_runtime(struct cfs_rq *cfs_rq) { return false; }
> > +static void check_cfs_rq_runtime(struct cfs_rq *cfs_rq) {}
> > static void check_enqueue_throttle(struct cfs_rq *cfs_rq) {}
> > static inline void sync_throttle(struct task_group *tg, int cpu) {}
> > static __always_inline void return_cfs_rq_runtime(struct cfs_rq *cfs_rq) {}
> > +static void task_throttle_setup_work(struct task_struct *p) {}
> >
> > static inline int cfs_rq_throttled(struct cfs_rq *cfs_rq)
> > {
> > @@ -7108,10 +7113,6 @@ static int dequeue_entities(struct rq *rq, struct sched_entity *se, int flags)
> > if (cfs_rq_is_idle(cfs_rq))
> > h_nr_idle = h_nr_queued;
> >
> > - /* end evaluation on encountering a throttled cfs_rq */
> > - if (cfs_rq_throttled(cfs_rq))
> > - return 0;
> > -
> > /* Don't dequeue parent if it has other entities besides us */
> > if (cfs_rq->load.weight) {
> > slice = cfs_rq_min_slice(cfs_rq);
> > @@ -7148,10 +7149,6 @@ static int dequeue_entities(struct rq *rq, struct sched_entity *se, int flags)
> >
> > if (cfs_rq_is_idle(cfs_rq))
> > h_nr_idle = h_nr_queued;
> > -
> > - /* end evaluation on encountering a throttled cfs_rq */
> > - if (cfs_rq_throttled(cfs_rq))
> > - return 0;
> > }
> >
> > sub_nr_running(rq, h_nr_queued);
> > @@ -8860,8 +8857,7 @@ static struct task_struct *pick_task_fair(struct rq *rq)
> > if (cfs_rq->curr && cfs_rq->curr->on_rq)
> > update_curr(cfs_rq);
> >
> > - if (unlikely(check_cfs_rq_runtime(cfs_rq)))
> > - goto again;
> > + check_cfs_rq_runtime(cfs_rq);
> >
> > se = pick_next_entity(rq, cfs_rq);
> > if (!se)
> > @@ -8888,6 +8884,9 @@ pick_next_task_fair(struct rq *rq, struct task_struct *prev, struct rq_flags *rf
> > goto idle;
> > se = &p->se;
> >
> > + if (throttled_hierarchy(cfs_rq_of(se)))
> > + task_throttle_setup_work(p);
> > +
> > #ifdef CONFIG_FAIR_GROUP_SCHED
> > if (prev->sched_class != &fair_sched_class)
> > goto simple;
> > diff --git a/kernel/sched/sched.h b/kernel/sched/sched.h
> > index 921527327f107..97be6a6f53b9c 100644
> > --- a/kernel/sched/sched.h
> > +++ b/kernel/sched/sched.h
> > @@ -736,6 +736,7 @@ struct cfs_rq {
> > int throttle_count;
> > struct list_head throttled_list;
> > struct list_head throttled_csd_list;
> > + struct list_head throttled_limbo_list;
> > #endif /* CONFIG_CFS_BANDWIDTH */
> > #endif /* CONFIG_FAIR_GROUP_SCHED */
> > };
>
next prev parent reply other threads:[~2025-04-14 12:10 UTC|newest]
Thread overview: 45+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-04-09 12:07 [RFC PATCH v2 0/7] Defer throttle when task exits to user Aaron Lu
2025-04-09 12:07 ` [RFC PATCH v2 1/7] sched/fair: Add related data structure for task based throttle Aaron Lu
2025-04-14 3:58 ` K Prateek Nayak
2025-04-14 11:55 ` Aaron Lu
2025-04-14 13:37 ` K Prateek Nayak
2025-04-09 12:07 ` [RFC PATCH v2 2/7] sched/fair: Handle throttle path " Aaron Lu
2025-04-14 8:54 ` Florian Bezdeka
2025-04-14 12:10 ` Aaron Lu [this message]
2025-04-14 14:39 ` Florian Bezdeka
2025-04-14 15:02 ` K Prateek Nayak
2025-04-30 10:01 ` Aaron Lu
2025-04-09 12:07 ` [RFC PATCH v2 3/7] sched/fair: Handle unthrottle " Aaron Lu
2025-04-09 12:07 ` [RFC PATCH v2 4/7] sched/fair: Take care of group/affinity/sched_class change for throttled task Aaron Lu
2025-04-09 12:07 ` [RFC PATCH v2 5/7] sched/fair: get rid of throttled_lb_pair() Aaron Lu
2025-04-09 12:07 ` [RFC PATCH v2 6/7] sched/fair: fix h_nr_runnable accounting with per-task throttle Aaron Lu
2025-04-09 12:07 ` [RFC PATCH v2 7/7] sched/fair: alternative way of accounting throttle time Aaron Lu
2025-04-09 14:24 ` Aaron Lu
2025-04-17 14:06 ` Florian Bezdeka
2025-04-18 3:15 ` Aaron Lu
2025-04-22 15:03 ` Florian Bezdeka
2025-04-23 11:26 ` Aaron Lu
2025-04-23 12:15 ` Florian Bezdeka
2025-04-24 2:26 ` Aaron Lu
2025-05-07 9:09 ` Aaron Lu
2025-05-07 9:33 ` Florian Bezdeka
2025-05-08 2:45 ` Aaron Lu
2025-05-08 6:13 ` Jan Kiszka
2025-05-08 13:43 ` Steven Rostedt
2025-04-14 3:05 ` [RFC PATCH v2 0/7] Defer throttle when task exits to user Chengming Zhou
2025-04-14 11:47 ` Aaron Lu
2025-04-14 8:54 ` Florian Bezdeka
2025-04-14 12:04 ` Aaron Lu
2025-04-15 5:29 ` Jan Kiszka
2025-04-15 6:05 ` K Prateek Nayak
2025-04-15 6:09 ` Jan Kiszka
2025-04-15 8:45 ` K Prateek Nayak
2025-04-15 10:21 ` Jan Kiszka
2025-04-15 11:14 ` K Prateek Nayak
[not found] ` <ec2cea83-07fe-472f-8320-911d215473fd@amd.com>
2025-04-15 15:49 ` K Prateek Nayak
2025-04-22 2:10 ` Aaron Lu
2025-04-22 2:54 ` K Prateek Nayak
2025-04-22 14:54 ` Florian Bezdeka
2025-04-15 10:34 ` K Prateek Nayak
2025-04-14 16:34 ` K Prateek Nayak
2025-04-15 11:25 ` Aaron Lu
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=20250414121038.GD3558904@bytedance \
--to=ziqianlu@bytedance.com \
--cc=bsegall@google.com \
--cc=chengming.zhou@linux.dev \
--cc=dietmar.eggemann@arm.com \
--cc=florian.bezdeka@siemens.com \
--cc=jan.kiszka@siemens.com \
--cc=joshdon@google.com \
--cc=juri.lelli@redhat.com \
--cc=kprateek.nayak@amd.com \
--cc=linux-kernel@vger.kernel.org \
--cc=mgorman@suse.de \
--cc=mingo@redhat.com \
--cc=peterz@infradead.org \
--cc=rostedt@goodmis.org \
--cc=vincent.guittot@linaro.org \
--cc=vschneid@redhat.com \
--cc=xii@google.com \
--cc=zhouchuyi@bytedance.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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.