Received: by 2002:a05:6a10:9848:0:0:0:0 with SMTP id x8csp29538pxf; Wed, 24 Mar 2021 19:55:03 -0700 (PDT) X-Google-Smtp-Source: ABdhPJxdBM0yZAiOFZkfK0kbmKte4BbEfyMsp7TG+53BaGgLnzXfkcPIsqN/dNJTH4/rG0A9cSkp X-Received: by 2002:aa7:df84:: with SMTP id b4mr6625989edy.240.1616640903335; Wed, 24 Mar 2021 19:55:03 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1616640903; cv=none; d=google.com; s=arc-20160816; b=0xSL4wc/KKqmRvBfqf+dc3iZvCWTQl2xZ1jOJux23BpZwL1lUGGMkpCU2t4Whvr3se vy/ptOhm30zyeIn3Js2HaftGM2WBs9hJrdsAT408c5hxLtVqMmDGTVGY9AFncSLU4FYe 6utri2Ab6+JwFPZshlf0+GXVQmx8/+HgLcVKUspIGTgLD0t/CXihhI0yaL1cb0axWSPp uGv/gmfrcBnJpWfmigDh4SpKAWTbrgzWjN0yfDqEkNK0M0NCDLwIsMYEqL28JRxpoGgi LKxcXJPBLzSuRT8xFDOXCre+ZQNnVtFT43j4s6BVbb+6JNTucmExEAe1PjN5k0+3ZEzX Eppw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:in-reply-to:content-disposition:mime-version :references:message-id:subject:cc:to:from:date:dkim-signature; bh=POSrgar70SlhtC0MB+c6+4sBxWfdvz20BQJ8i3GXyAY=; b=DZrXAnt81Y0cxnV9nagv4XPV8C+ZeMtVBmoMlZDkMa9/g6GqBNETQZT32ttTMAkn5Z /G4HSnxBIqfcbU3jBzZhERveETYkbvL57hQVh5oNZEa9x8xPckm86E0bAKJoDNz27UX5 bowASVgqV+6If/UfZMmxgAQwRv5lvHw+z1z/vegDg3mC1bNPT487/xqgd+kTjSPqXlsc YUWOhgUuMdgeJgxdnmppClFwzHiVF06uPl03YPpNIY+/IUcWSfIBFSV5p/Sz44ypOgbN GySQqtgRung6UzLIqNjM5XZ163pTYDdupvnpfzCaoWYoWNlD1YqOqNYxXF+CgkUue1HQ ZcsA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@infradead.org header.s=casper.20170209 header.b=b6TSHXQx; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id l12si3281709edi.290.2021.03.24.19.54.41; Wed, 24 Mar 2021 19:55:03 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@infradead.org header.s=casper.20170209 header.b=b6TSHXQx; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S234468AbhCXK53 (ORCPT + 99 others); Wed, 24 Mar 2021 06:57:29 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60156 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S234431AbhCXK5Q (ORCPT ); Wed, 24 Mar 2021 06:57:16 -0400 Received: from casper.infradead.org (casper.infradead.org [IPv6:2001:8b0:10b:1236::1]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 43735C061763; Wed, 24 Mar 2021 03:57:16 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=infradead.org; s=casper.20170209; h=In-Reply-To:Content-Type:MIME-Version: References:Message-ID:Subject:Cc:To:From:Date:Sender:Reply-To: Content-Transfer-Encoding:Content-ID:Content-Description; bh=POSrgar70SlhtC0MB+c6+4sBxWfdvz20BQJ8i3GXyAY=; b=b6TSHXQxLyZI/ZDi6HWgC65FJF gYh45yZTsYpgnvceVdNWG8j4+ISg4iSdGwiRVJcQFlXsAk6WLA1i3F61P+FUr0CgmzeUguz5FVv3m XRcYwp+QEI4xB8qquTXuJL1LTRpgqw0OaOHKy/8SxJYSYDYpNc64Wj9I46TqYI1TrL/kiPAKcQUQG 7C6v1H8urwfEHSCwjdPeNvFKO2nqtDlAWLPZ/NXdsMmpgrpN2pTZ04lSGQKzXAOU0iI3Q4FtoXN/i wJuGzDlGB+l8LMw+QjtCgbmctMxCQy+FuWfHgw5eUlWOF7JFDqKjiyyURzL4T0ihHdWXZN3pzAGJn e+fjgfBg==; Received: from j217100.upc-j.chello.nl ([24.132.217.100] helo=noisy.programming.kicks-ass.net) by casper.infradead.org with esmtpsa (Exim 4.94 #2 (Red Hat Linux)) id 1lP1Aj-00BGM1-MV; Wed, 24 Mar 2021 10:55:45 +0000 Received: from hirez.programming.kicks-ass.net (hirez.programming.kicks-ass.net [192.168.1.225]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (Client did not present a certificate) by noisy.programming.kicks-ass.net (Postfix) with ESMTPS id 0D2463010C8; Wed, 24 Mar 2021 11:55:33 +0100 (CET) Received: by hirez.programming.kicks-ass.net (Postfix, from userid 1000) id F1B0428B290D0; Wed, 24 Mar 2021 11:55:32 +0100 (CET) Date: Wed, 24 Mar 2021 11:55:32 +0100 From: Peter Zijlstra To: Josh Don Cc: Ingo Molnar , Juri Lelli , Vincent Guittot , Dietmar Eggemann , Steven Rostedt , Ben Segall , Mel Gorman , Daniel Bristot de Oliveira , Luis Chamberlain , Kees Cook , Iurii Zaikin , linux-kernel@vger.kernel.org, linux-fsdevel@vger.kernel.org, David Rientjes , Oleg Rombakh , linux-doc@vger.kernel.org, Paul Turner Subject: Re: [PATCH v2] sched: Warn on long periods of pending need_resched Message-ID: References: <20210323035706.572953-1-joshdon@google.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Wed, Mar 24, 2021 at 11:54:24AM +0100, Peter Zijlstra wrote: > On Wed, Mar 24, 2021 at 10:37:43AM +0100, Peter Zijlstra wrote: > > Should we perhaps take out all SCHED_DEBUG sysctls and move them to > > /debug/sched/ ? (along with the existing /debug/sched_{debug,features,preemp} > > files) > > > > Having all that in sysctl and documented gives them far too much sheen > > of ABI. > > ... a little something like this ... > And then the parent post becomes something like this.. --- Subject: sched: Warn on long periods of pending need_resched From: Paul Turner Date: Mon, 22 Mar 2021 20:57:06 -0700 From: Paul Turner CPU scheduler marks need_resched flag to signal a schedule() on a particular CPU. But, schedule() may not happen immediately in cases where the current task is executing in the kernel mode (no preemption state) for extended periods of time. This patch adds a warn_on if need_resched is pending for more than the time specified in sysctl resched_latency_warn_ms. If it goes off, it is likely that there is a missing cond_resched() somewhere. Monitoring is done via the tick and the accuracy is hence limited to jiffy scale. This also means that we won't trigger the warning if the tick is disabled. This feature is default disabled. Signed-off-by: Paul Turner Signed-off-by: Josh Don Signed-off-by: Peter Zijlstra (Intel) Link: https://lkml.kernel.org/r/20210323035706.572953-1-joshdon@google.com --- include/linux/sched/sysctl.h | 3 + kernel/sched/core.c | 75 ++++++++++++++++++++++++++++++++++++++++++- kernel/sched/debug.c | 13 +++++++ kernel/sched/features.h | 2 + kernel/sched/sched.h | 10 +++++ 5 files changed, 102 insertions(+), 1 deletion(-) --- a/include/linux/sched/sysctl.h +++ b/include/linux/sched/sysctl.h @@ -48,6 +48,9 @@ extern unsigned int sysctl_numa_balancin #ifdef CONFIG_SCHED_DEBUG extern __read_mostly unsigned int sysctl_sched_migration_cost; extern __read_mostly unsigned int sysctl_sched_nr_migrate; + +extern int sysctl_resched_latency_warn_ms; +extern int sysctl_resched_latency_warn_once; #endif /* --- a/kernel/sched/core.c +++ b/kernel/sched/core.c @@ -58,7 +58,21 @@ const_debug unsigned int sysctl_sched_fe #include "features.h" 0; #undef SCHED_FEAT -#endif + +/* + * Print a warning if need_resched is set for the given duration (if + * resched_latency_warn_enabled is set). + * + * If sysctl_resched_latency_warn_once is set, only one warning will be shown + * per boot. + * + * Resched latency will be ignored for the first resched_boot_quiet_sec, to + * reduce false alarms. + */ +int sysctl_resched_latency_warn_ms = 100; +int sysctl_resched_latency_warn_once = 1; +const long resched_boot_quiet_sec = 600; +#endif /* CONFIG_SCHED_DEBUG */ /* * Number of tasks to iterate in a single balance run. @@ -4485,6 +4499,56 @@ unsigned long long task_sched_runtime(st return ns; } +#ifdef CONFIG_SCHED_DEBUG +static u64 resched_latency_check(struct rq *rq) +{ + int latency_warn_ms = READ_ONCE(sysctl_resched_latency_warn_ms); + u64 need_resched_latency, now = rq_clock(rq); + static bool warned_once; + + if (sysctl_resched_latency_warn_once && warned_once) + return 0; + + if (!need_resched() || WARN_ON_ONCE(latency_warn_ms < 2)) + return 0; + + /* Disable this warning for the first few mins after boot */ + if (now < resched_boot_quiet_sec * NSEC_PER_SEC) + return 0; + + if (!rq->last_seen_need_resched_ns) { + rq->last_seen_need_resched_ns = now; + rq->ticks_without_resched = 0; + return 0; + } + + rq->ticks_without_resched++; + need_resched_latency = now - rq->last_seen_need_resched_ns; + if (need_resched_latency <= latency_warn_ms * NSEC_PER_MSEC) + return 0; + + warned_once = true; + + return need_resched_latency; +} + +static int __init setup_resched_latency_warn_ms(char *str) +{ + long val; + + if ((kstrtol(str, 0, &val))) { + pr_warn("Unable to set resched_latency_warn_ms\n"); + return 1; + } + + sysctl_resched_latency_warn_ms = val; + return 1; +} +__setup("resched_latency_warn_ms=", setup_resched_latency_warn_ms); +#else +static inline u64 resched_latency_check(struct rq *rq) { return 0; } +#endif /* CONFIG_SCHED_DEBUG */ + /* * This function gets called by the timer code, with HZ frequency. * We call it with interrupts disabled. @@ -4496,6 +4560,7 @@ void scheduler_tick(void) struct task_struct *curr = rq->curr; struct rq_flags rf; unsigned long thermal_pressure; + u64 resched_latency; arch_scale_freq_tick(); sched_clock_tick(); @@ -4506,10 +4571,15 @@ void scheduler_tick(void) thermal_pressure = arch_scale_thermal_pressure(cpu_of(rq)); update_thermal_load_avg(rq_clock_thermal(rq), rq, thermal_pressure); curr->sched_class->task_tick(rq, curr, 0); + if (sched_feat(LATENCY_WARN)) + resched_latency = resched_latency_check(rq); calc_global_load_tick(rq); rq_unlock(rq, &rf); + if (sched_feat(LATENCY_WARN) && resched_latency) + resched_latency_warn(cpu, resched_latency); + perf_event_task_tick(); #ifdef CONFIG_SMP @@ -5004,6 +5074,9 @@ static void __sched notrace __schedule(b next = pick_next_task(rq, prev, &rf); clear_tsk_need_resched(prev); clear_preempt_need_resched(); +#ifdef CONFIG_SCHED_DEBUG + rq->last_seen_need_resched_ns = 0; +#endif if (likely(prev != next)) { rq->nr_switches++; --- a/kernel/sched/debug.c +++ b/kernel/sched/debug.c @@ -321,6 +321,9 @@ static __init int sched_init_debug(void) debugfs_create_u32("min_granularity_ns", 0644, debugfs_sched, &sysctl_sched_min_granularity); debugfs_create_u32("wakeup_granularity_ns", 0644, debugfs_sched, &sysctl_sched_wakeup_granularity); + debugfs_create_u32("latency_warn_ms", 0644, debugfs_sched, &sysctl_resched_latency_warn_ms); + debugfs_create_u32("latency_warn_once", 0644, debugfs_sched, &sysctl_resched_latency_warn_once); + #ifdef CONFIG_SMP debugfs_create_file("tunable_scaling", 0644, debugfs_sched, NULL, &sched_scaling_fops); debugfs_create_u32("migration_cost_ns", 0644, debugfs_sched, &sysctl_sched_migration_cost); @@ -1194,3 +1197,13 @@ void proc_sched_set_task(struct task_str memset(&p->se.statistics, 0, sizeof(p->se.statistics)); #endif } + +void resched_latency_warn(int cpu, u64 latency) +{ + static DEFINE_RATELIMIT_STATE(latency_check_ratelimit, 60 * 60 * HZ, 1); + + WARN(__ratelimit(&latency_check_ratelimit), + "sched: CPU %d need_resched set for > %llu ns (%d ticks) " + "without schedule\n", + cpu, latency, cpu_rq(cpu)->ticks_without_resched); +} --- a/kernel/sched/features.h +++ b/kernel/sched/features.h @@ -90,3 +90,5 @@ SCHED_FEAT(WA_BIAS, true) */ SCHED_FEAT(UTIL_EST, true) SCHED_FEAT(UTIL_EST_FASTUP, true) + +SCHED_FEAT(LATENCY_WARN, false) --- a/kernel/sched/sched.h +++ b/kernel/sched/sched.h @@ -58,6 +58,7 @@ #include #include #include +#include #include #include #include @@ -971,6 +972,11 @@ struct rq { atomic_t nr_iowait; +#ifdef CONFIG_SCHED_DEBUG + u64 last_seen_need_resched_ns; + int ticks_without_resched; +#endif + #ifdef CONFIG_MEMBARRIER int membarrier_state; #endif @@ -2374,6 +2380,8 @@ extern void print_dl_stats(struct seq_fi extern void print_cfs_rq(struct seq_file *m, int cpu, struct cfs_rq *cfs_rq); extern void print_rt_rq(struct seq_file *m, int cpu, struct rt_rq *rt_rq); extern void print_dl_rq(struct seq_file *m, int cpu, struct dl_rq *dl_rq); + +extern void resched_latency_warn(int cpu, u64 latency); #ifdef CONFIG_NUMA_BALANCING extern void show_numa_stats(struct task_struct *p, struct seq_file *m); @@ -2381,6 +2389,8 @@ extern void print_numa_stats(struct seq_file *m, int node, unsigned long tsf, unsigned long tpf, unsigned long gsf, unsigned long gpf); #endif /* CONFIG_NUMA_BALANCING */ +#else +static inline void resched_latency_warn(int cpu, u64 latency) {} #endif /* CONFIG_SCHED_DEBUG */ extern void init_cfs_rq(struct cfs_rq *cfs_rq);