From mboxrd@z Thu Jan 1 00:00:00 1970 From: Steven Rostedt Subject: [PATCH RT 19/24] sched/core: Avoid __schedule() being called twice in a row Date: Fri, 07 Sep 2018 16:59:16 -0400 Message-ID: <20180907205932.908152135@goodmis.org> References: <20180907205857.262840394@goodmis.org> Mime-Version: 1.0 Content-Type: text/plain; charset=ISO-8859-15 Cc: Thomas Gleixner , Carsten Emde , Sebastian Andrzej Siewior , John Kacur , Paul Gortmaker , Julia Cartwright , Daniel Wagner , tom.zanussi@linux.intel.com, Daniel Bristot de Oliveira , Clark Williams , Tommaso Cucinotta , Romulo da Silva de Oliveira , Ingo Molnar , Peter Zijlstra To: linux-kernel@vger.kernel.org, linux-rt-users Return-path: Sender: linux-kernel-owner@vger.kernel.org List-Id: linux-rt-users.vger.kernel.org 4.14.63-rt41-rc2 stable review patch. If anyone has any objections, please let me know. ------------------ From: Daniel Bristot de Oliveira [ Upstream commit 2bb94c48b2ffaabf8c15a51e5cc1b4c541988cab ] If a worker invokes schedule() then we may have the call chain: schedule() -> sched_submit_work() -> wq_worker_sleeping() -> wake_up_worker() -> wake_up_process(). The last wakeup may cause a schedule which is unnecessary because we are already in schedule() and do it anyway. Add a preempt_disable() + preempt_enable_no_resched() around wq_worker_sleeping() so the context switch could be delayed until __schedule(). Signed-off-by: Daniel Bristot de Oliveira Cc: Clark Williams Cc: Tommaso Cucinotta Cc: Romulo da Silva de Oliveira Cc: Sebastian Andrzej Siewior Cc: Steven Rostedt Cc: Thomas Gleixner Cc: Ingo Molnar Cc: Peter Zijlstra Signed-off-by: Steven Rostedt (VMware) [bigeasy: rewrite changelog] Signed-off-by: Sebastian Andrzej Siewior --- kernel/sched/core.c | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/kernel/sched/core.c b/kernel/sched/core.c index fa5b76255f8c..a5ce37b90fca 100644 --- a/kernel/sched/core.c +++ b/kernel/sched/core.c @@ -3482,10 +3482,15 @@ static inline void sched_submit_work(struct task_struct *tsk) /* * If a worker went to sleep, notify and ask workqueue whether * it wants to wake up a task to maintain concurrency. + * As this function is called inside the schedule() context, + * we disable preemption to avoid it calling schedule() again + * in the possible wakeup of a kworker. */ - if (tsk->flags & PF_WQ_WORKER) + if (tsk->flags & PF_WQ_WORKER) { + preempt_disable(); wq_worker_sleeping(tsk); - + preempt_enable_no_resched(); + } if (tsk_is_pi_blocked(tsk)) return; -- 2.18.0