Received: by 10.223.185.116 with SMTP id b49csp3622533wrg; Tue, 6 Mar 2018 02:07:07 -0800 (PST) X-Google-Smtp-Source: AG47ELuH7j2kfLqVp9alqZAbEojYgUnf1xcD2VuBMUzn7y0ofbXsMdcPc4HtwT6NqxfxGhDtvgf0 X-Received: by 2002:a17:902:8d92:: with SMTP id v18-v6mr8819436plo.21.1520330826915; Tue, 06 Mar 2018 02:07:06 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1520330826; cv=none; d=google.com; s=arc-20160816; b=ZynHvf8syFKADPmL8aZx5DQ8dOVk5CKWgYCjqucBJZRuBS6FLsLfqPS5Tq7sn+vX/N OIQSuJ8Odqy95kMz4l24J0p89hMyW7a4jD9nE+8v9Fw2PyCqrLukm7VlLc9cfACUfqn1 SYKJhWqq41wceSGF58s7WDMPX21MLIFzFNFvOSGU3BK5AKoJo7sod/67Ui9oAE2el4AF UhG3FNbggrY552I9Jixx0ZNiPHHNdN8caeTtyoxLRLsZlpdDEkTCH1wqYFEOqNH8gmYH pV2xp1XL80Pt+3ENsXgzfgnEF55T0fI8KscJ7Lrp6HLg7DN427CN1ymPE6un7c4NeC85 ypjg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from :arc-authentication-results; bh=PnK/07lRbN62vWakTTv1yAs/CjlJpK51UbXpkBNFNjs=; b=oRyEN6K6ALAwd87yBugWEU6RXaqfy4xGXOj4VZNygIK6+2Te0RzbM2EWKIU2c8AUQG 12xvlHnrzDrj/ZzVmJmqk8W66sxKq+RGZYo4t/0q03NQlnq3i4MoprytPOGBB6iDD5V9 CH6/5jbAwUoKPFEigKYINRxD87vr9/O7fa8qTw9O4ZmkNJ+YOKRV1He4wZw9dQa+6f9H WSEOMVQoGBbFkjVCjvKILGEEgdjCpTMfdtUrwYiBd6CamQAD7GsJlA3QF1avLb2d214R wQ0KPMSCi0esua4w9coY8WFHYS0zEBtPeQGaX2lzBJfWueM915oZ2CUH8xhpX9s9irjJ zlzg== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id 92-v6si4832954plw.562.2018.03.06.02.06.52; Tue, 06 Mar 2018 02:07:06 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753335AbeCFKF3 (ORCPT + 99 others); Tue, 6 Mar 2018 05:05:29 -0500 Received: from cloudserver094114.home.pl ([79.96.170.134]:56080 "EHLO cloudserver094114.home.pl" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1750947AbeCFKF1 (ORCPT ); Tue, 6 Mar 2018 05:05:27 -0500 Received: from 79.184.254.228.ipv4.supernova.orange.pl (79.184.254.228) (HELO aspire.rjw.lan) by serwer1319399.home.pl (79.96.170.134) with SMTP (IdeaSmtpServer 0.83) id c26ae3e54d5a72c0; Tue, 6 Mar 2018 11:05:25 +0100 From: "Rafael J. Wysocki" To: Peter Zijlstra , Linux PM Cc: Thomas Gleixner , Frederic Weisbecker , Paul McKenney , Thomas Ilsche , Doug Smythies , Rik van Riel , Aubrey Li , Mike Galbraith , LKML Subject: [Update][RFC/RFT][PATCH v2 4/6] cpuidle: Return nohz hint from cpuidle_select() Date: Tue, 06 Mar 2018 11:06:10 +0100 Message-ID: <6393309.frkCvRUP95@aspire.rjw.lan> In-Reply-To: <1756432.KnKEJ20PhI@aspire.rjw.lan> References: <2067762.1uWBf5RSRc@aspire.rjw.lan> <1756432.KnKEJ20PhI@aspire.rjw.lan> MIME-Version: 1.0 Content-Transfer-Encoding: 7Bit Content-Type: text/plain; charset="us-ascii" Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Rafael J. Wysocki Add a new pointer argument to cpuidle_select() and to the ->select cpuidle governor callback to allow a boolean value indicating whether or not the tick should be stopped before entering the selected state to be returned from there. Make the ladder governor ignore that pointer (to preserve its current behavior) and make the menu governor return 'false" through it if: (1) the idle exit latency is constrained at 0, (2) the selected state is a polling one, or (3) the target residency of the selected state is less than the length of the tick period and there is at least one deeper idle state available within the tick period range. Since the value returned through the new argument pointer is not used yet, this change is not expected to alter the functionality of the code. Signed-off-by: Rafael J. Wysocki --- Update: Fix a reversed value of *nohz_ret in menu_select(). --- drivers/cpuidle/cpuidle.c | 10 ++++++++-- drivers/cpuidle/governors/ladder.c | 3 ++- drivers/cpuidle/governors/menu.c | 26 ++++++++++++++++++++++---- include/linux/cpuidle.h | 6 ++++-- kernel/sched/idle.c | 4 +++- 5 files changed, 39 insertions(+), 10 deletions(-) Index: linux-pm/include/linux/cpuidle.h =================================================================== --- linux-pm.orig/include/linux/cpuidle.h +++ linux-pm/include/linux/cpuidle.h @@ -131,7 +131,8 @@ extern bool cpuidle_not_available(struct struct cpuidle_device *dev); extern int cpuidle_select(struct cpuidle_driver *drv, - struct cpuidle_device *dev); + struct cpuidle_device *dev, + bool *nohz_ret); extern int cpuidle_enter(struct cpuidle_driver *drv, struct cpuidle_device *dev, int index); extern void cpuidle_reflect(struct cpuidle_device *dev, int index); @@ -246,7 +247,8 @@ struct cpuidle_governor { struct cpuidle_device *dev); int (*select) (struct cpuidle_driver *drv, - struct cpuidle_device *dev); + struct cpuidle_device *dev, + bool *nohz_ret); void (*reflect) (struct cpuidle_device *dev, int index); }; Index: linux-pm/kernel/sched/idle.c =================================================================== --- linux-pm.orig/kernel/sched/idle.c +++ linux-pm/kernel/sched/idle.c @@ -186,13 +186,15 @@ static void cpuidle_idle_call(void) next_state = cpuidle_find_deepest_state(drv, dev); call_cpuidle(drv, dev, next_state); } else { + bool nohz = true; + tick_nohz_idle_go_idle(true); rcu_idle_enter(); /* * Ask the cpuidle framework to choose a convenient idle state. */ - next_state = cpuidle_select(drv, dev); + next_state = cpuidle_select(drv, dev, &nohz); entered_state = call_cpuidle(drv, dev, next_state); /* * Give the governor an opportunity to reflect on the outcome Index: linux-pm/drivers/cpuidle/cpuidle.c =================================================================== --- linux-pm.orig/drivers/cpuidle/cpuidle.c +++ linux-pm/drivers/cpuidle/cpuidle.c @@ -263,12 +263,18 @@ int cpuidle_enter_state(struct cpuidle_d * * @drv: the cpuidle driver * @dev: the cpuidle device + * @nohz_ret: indication on whether or not to stop the tick * * Returns the index of the idle state. The return value must not be negative. + * + * The memory location pointed to by @nohz_ret is expected to be written the + * 'false' boolean value if the scheduler tick should not be stopped before + * entering the returned state. */ -int cpuidle_select(struct cpuidle_driver *drv, struct cpuidle_device *dev) +int cpuidle_select(struct cpuidle_driver *drv, struct cpuidle_device *dev, + bool *nohz_ret) { - return cpuidle_curr_governor->select(drv, dev); + return cpuidle_curr_governor->select(drv, dev, nohz_ret); } /** Index: linux-pm/drivers/cpuidle/governors/ladder.c =================================================================== --- linux-pm.orig/drivers/cpuidle/governors/ladder.c +++ linux-pm/drivers/cpuidle/governors/ladder.c @@ -63,9 +63,10 @@ static inline void ladder_do_selection(s * ladder_select_state - selects the next state to enter * @drv: cpuidle driver * @dev: the CPU + * @dummy: not used */ static int ladder_select_state(struct cpuidle_driver *drv, - struct cpuidle_device *dev) + struct cpuidle_device *dev, bool *dummy) { struct ladder_device *ldev = this_cpu_ptr(&ladder_devices); struct device *device = get_cpu_device(dev->cpu); Index: linux-pm/drivers/cpuidle/governors/menu.c =================================================================== --- linux-pm.orig/drivers/cpuidle/governors/menu.c +++ linux-pm/drivers/cpuidle/governors/menu.c @@ -279,8 +279,10 @@ again: * menu_select - selects the next idle state to enter * @drv: cpuidle driver containing state data * @dev: the CPU + * @nohz_ret: indication on whether or not to stop the tick */ -static int menu_select(struct cpuidle_driver *drv, struct cpuidle_device *dev) +static int menu_select(struct cpuidle_driver *drv, struct cpuidle_device *dev, + bool *nohz_ret) { struct menu_device *data = this_cpu_ptr(&menu_devices); struct device *device = get_cpu_device(dev->cpu); @@ -303,8 +305,10 @@ static int menu_select(struct cpuidle_dr latency_req = resume_latency; /* Special case when user has set very strict latency requirement */ - if (unlikely(latency_req == 0)) + if (unlikely(latency_req == 0)) { + *nohz_ret = false; return 0; + } /* determine the expected residency time, round up */ data->next_timer_us = ktime_to_us(tick_nohz_get_sleep_length()); @@ -367,10 +371,12 @@ static int menu_select(struct cpuidle_dr continue; if (idx == -1) idx = i; /* first enabled state */ - if (s->target_residency > data->predicted_us) - break; if (s->exit_latency > latency_req) break; + if (s->target_residency > data->predicted_us) { + first_idx = i; + break; + } idx = i; } @@ -378,6 +384,18 @@ static int menu_select(struct cpuidle_dr if (idx == -1) idx = 0; /* No states enabled. Must use 0. */ + if (drv->states[idx].flags & CPUIDLE_FLAG_POLLING) { + *nohz_ret = false; + } else if (drv->states[idx].target_residency < TICK_USEC) { + /* + * Do not stop the tick if there is at least one more state + * within the tick period range that could be used if longer + * idle duration was predicted. + */ + *nohz_ret = !(first_idx > idx && + drv->states[first_idx].target_residency < TICK_USEC); + } + data->last_state_idx = idx; return data->last_state_idx;