From patchwork Fri Dec 14 13:57:34 2012 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Daniel Lezcano X-Patchwork-Id: 1878691 Return-Path: X-Original-To: patchwork-linux-pm@patchwork.kernel.org Delivered-To: patchwork-process-083081@patchwork1.kernel.org Received: from vger.kernel.org (vger.kernel.org [209.132.180.67]) by patchwork1.kernel.org (Postfix) with ESMTP id 435E6402E1 for ; Fri, 14 Dec 2012 13:58:44 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1756106Ab2LNN5r (ORCPT ); Fri, 14 Dec 2012 08:57:47 -0500 Received: from mail-ea0-f174.google.com ([209.85.215.174]:34057 "EHLO mail-ea0-f174.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1756015Ab2LNN5m (ORCPT ); Fri, 14 Dec 2012 08:57:42 -0500 Received: by mail-ea0-f174.google.com with SMTP id e13so1299620eaa.19 for ; Fri, 14 Dec 2012 05:57:41 -0800 (PST) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20120113; h=from:to:cc:subject:date:message-id:x-mailer:in-reply-to:references :x-gm-message-state; bh=TvFJZdpnea3qLn3FteAA8RC1mNrcds+Lsx97Ai7egBk=; b=Y52LZn4fblCQ58WfiorQJf8TUcdCsk68smmCMUyYwBaMFm5CdmIMyofvXRq29pR/fM v0z3HLayl6zMU6tFXXHD/j9z2kThrAY+DClyDJhev9+5yYOpYbnLF51AlIpF+Y8/pj+A nr53RBivCZIoNJF7aIhUyfWj3D6QkVE8zjv++UHt2fc3CSdKhjL3bmgg81IAwr+Ouck+ dkrA7CdShK0ItYm4JYbuK8upMueCJK8VRtHj6LyCH+xVuulTqvNHG/giX1BL5b3+HR5x p69dISOJoDNCAGe5XWORV51E0qyVyGumrarWjHUmlvmabH1q8Oe72Y4oc0j1LhSQK/PG UDjA== Received: by 10.14.215.194 with SMTP id e42mr15395155eep.32.1355493460917; Fri, 14 Dec 2012 05:57:40 -0800 (PST) Received: from localhost.localdomain (AToulouse-654-1-446-186.w83-205.abo.wanadoo.fr. [83.205.197.186]) by mx.google.com with ESMTPS id d3sm9599217eeo.13.2012.12.14.05.57.38 (version=TLSv1/SSLv3 cipher=OTHER); Fri, 14 Dec 2012 05:57:39 -0800 (PST) From: Daniel Lezcano To: rjw@sisk.pl Cc: jwerner@chromium.org, francescolavra.fl@gmail.com, linux-pm@vger.kernel.org, deepthi@linux.vnet.ibm.com, g.trinabh@gmail.com, linaro-dev@lists.linaro.org, len.brown@intel.com, linux-kernel@vger.kernel.org, akpm@linux-foundation.org, snanda@chromium.org Subject: [PATCH 1/2][V2] cpuidle - remove the power_specified field in the driver Date: Fri, 14 Dec 2012 14:57:34 +0100 Message-Id: <1355493455-30665-2-git-send-email-daniel.lezcano@linaro.org> X-Mailer: git-send-email 1.7.5.4 In-Reply-To: <1355493455-30665-1-git-send-email-daniel.lezcano@linaro.org> References: <1355493455-30665-1-git-send-email-daniel.lezcano@linaro.org> X-Gm-Message-State: ALoCoQkeH9kkHuXTlGZbtk7NGJl1yXESAtZYENvsoz/EMdQSoBPQs2R1/KL07Dr3OjVYHxQEe1RU Sender: linux-pm-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-pm@vger.kernel.org This patch follows the discussion about reinitializing the power usage when a C-state is added/removed. https://lkml.org/lkml/2012/10/16/518 We realized the power usage field is never filled and when it is filled for tegra, the power_specified flag is not set making all these values to be resetted when the driver is initialized with the set_power_state function. Julius and I feel this is over-engineered and the power_specified flag could be simply removed and continue assuming the states are backward sorted. The menu governor select function is simplified as the power is ordered. Actually the condition is always true with the current code. The cpuidle_play_dead function is also simplified by doing a reverse lookup on the array. The set_power_states function is removed as it does no make sense anymore. As a consequence, this patch fix also the bug where on the dynamic C-state system, the power fields is not initialized. Signed-off-by: Daniel Lezcano Cc: Julius Werner --- drivers/cpuidle/cpuidle.c | 17 ++++------------- drivers/cpuidle/driver.c | 25 ------------------------- drivers/cpuidle/governors/menu.c | 8 ++------ include/linux/cpuidle.h | 2 +- 4 files changed, 7 insertions(+), 45 deletions(-) diff --git a/drivers/cpuidle/cpuidle.c b/drivers/cpuidle/cpuidle.c index 8df53dd..e1f6860 100644 --- a/drivers/cpuidle/cpuidle.c +++ b/drivers/cpuidle/cpuidle.c @@ -69,24 +69,15 @@ int cpuidle_play_dead(void) { struct cpuidle_device *dev = __this_cpu_read(cpuidle_devices); struct cpuidle_driver *drv = cpuidle_get_cpu_driver(dev); - int i, dead_state = -1; - int power_usage = -1; + int i; if (!drv) return -ENODEV; /* Find lowest-power state that supports long-term idle */ - for (i = CPUIDLE_DRIVER_STATE_START; i < drv->state_count; i++) { - struct cpuidle_state *s = &drv->states[i]; - - if (s->power_usage < power_usage && s->enter_dead) { - power_usage = s->power_usage; - dead_state = i; - } - } - - if (dead_state != -1) - return drv->states[dead_state].enter_dead(dev, dead_state); + for (i = drv->state_count - 1; i >= CPUIDLE_DRIVER_STATE_START; i--) + if (drv->states[i].enter_dead) + return drv->states[i].enter_dead(dev, i); return -ENODEV; } diff --git a/drivers/cpuidle/driver.c b/drivers/cpuidle/driver.c index 3af841f..bb045f1 100644 --- a/drivers/cpuidle/driver.c +++ b/drivers/cpuidle/driver.c @@ -19,34 +19,9 @@ DEFINE_SPINLOCK(cpuidle_driver_lock); static void __cpuidle_set_cpu_driver(struct cpuidle_driver *drv, int cpu); static struct cpuidle_driver * __cpuidle_get_cpu_driver(int cpu); -static void set_power_states(struct cpuidle_driver *drv) -{ - int i; - - /* - * cpuidle driver should set the drv->power_specified bit - * before registering if the driver provides - * power_usage numbers. - * - * If power_specified is not set, - * we fill in power_usage with decreasing values as the - * cpuidle code has an implicit assumption that state Cn - * uses less power than C(n-1). - * - * With CONFIG_ARCH_HAS_CPU_RELAX, C0 is already assigned - * an power value of -1. So we use -2, -3, etc, for other - * c-states. - */ - for (i = CPUIDLE_DRIVER_STATE_START; i < drv->state_count; i++) - drv->states[i].power_usage = -1 - i; -} - static void __cpuidle_driver_init(struct cpuidle_driver *drv) { drv->refcnt = 0; - - if (!drv->power_specified) - set_power_states(drv); } static int __cpuidle_register_driver(struct cpuidle_driver *drv, int cpu) diff --git a/drivers/cpuidle/governors/menu.c b/drivers/cpuidle/governors/menu.c index bd40b94..fe343a0 100644 --- a/drivers/cpuidle/governors/menu.c +++ b/drivers/cpuidle/governors/menu.c @@ -312,7 +312,6 @@ static int menu_select(struct cpuidle_driver *drv, struct cpuidle_device *dev) { struct menu_device *data = &__get_cpu_var(menu_devices); int latency_req = pm_qos_request(PM_QOS_CPU_DMA_LATENCY); - int power_usage = -1; int i; int multiplier; struct timespec t; @@ -383,11 +382,8 @@ static int menu_select(struct cpuidle_driver *drv, struct cpuidle_device *dev) if (s->exit_latency * multiplier > data->predicted_us) continue; - if (s->power_usage < power_usage) { - power_usage = s->power_usage; - data->last_state_idx = i; - data->exit_us = s->exit_latency; - } + data->last_state_idx = i; + data->exit_us = s->exit_latency; } /* not deepest C-state chosen for low predicted residency */ diff --git a/include/linux/cpuidle.h b/include/linux/cpuidle.h index 3711b34..24cd103 100644 --- a/include/linux/cpuidle.h +++ b/include/linux/cpuidle.h @@ -126,9 +126,9 @@ struct cpuidle_driver { struct module *owner; int refcnt; - unsigned int power_specified:1; /* set to 1 to use the core cpuidle time keeping (for all states). */ unsigned int en_core_tk_irqen:1; + /* states array must be ordered in decreasing power consumption */ struct cpuidle_state states[CPUIDLE_STATE_MAX]; int state_count; int safe_state_index;