From patchwork Thu May 29 08:15:31 2014 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Amit Kachhap X-Patchwork-Id: 4262981 Return-Path: X-Original-To: patchwork-linux-arm@patchwork.kernel.org Delivered-To: patchwork-parsemail@patchwork1.web.kernel.org Received: from mail.kernel.org (mail.kernel.org [198.145.19.201]) by patchwork1.web.kernel.org (Postfix) with ESMTP id 9FD2F9F333 for ; Thu, 29 May 2014 08:19:03 +0000 (UTC) Received: from mail.kernel.org (localhost [127.0.0.1]) by mail.kernel.org (Postfix) with ESMTP id 9BFF82012E for ; Thu, 29 May 2014 08:19:02 +0000 (UTC) Received: from bombadil.infradead.org (bombadil.infradead.org [198.137.202.9]) (using TLSv1.2 with cipher DHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPS id 98507200C1 for ; Thu, 29 May 2014 08:19:01 +0000 (UTC) Received: from localhost ([127.0.0.1] helo=bombadil.infradead.org) by bombadil.infradead.org with esmtp (Exim 4.80.1 #2 (Red Hat Linux)) id 1WpvWD-0005wB-TR; Thu, 29 May 2014 08:16:57 +0000 Received: from mail-pa0-x230.google.com ([2607:f8b0:400e:c03::230]) by bombadil.infradead.org with esmtps (Exim 4.80.1 #2 (Red Hat Linux)) id 1WpvVj-0005ci-Fu for linux-arm-kernel@lists.infradead.org; Thu, 29 May 2014 08:16:28 +0000 Received: by mail-pa0-f48.google.com with SMTP id rd3so12440182pab.21 for ; Thu, 29 May 2014 01:16:05 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20120113; h=sender:from:to:cc:subject:date:message-id:in-reply-to:references; bh=HdOyqr1V0LTRQk9GE91f7/WKYOlDfUKaf6bf5EFmPpM=; b=f9MaQlyPgN/F7PYvlCrm9XMj01QvwI3tFhI6qouceXAknlnUJxvZwtlN2fntgOy/3t EFYq6BlunU5kX7PZtkDaxtGyVMJlJW2OUoDqM3aqI7ExSYmNOItlX6G2U0VrZBSJFWhx yM3ly/S5a9xZhXTvYj/TW4uK7qs0t/gCCXpf52pYSxJAxnF4cESXO5jJDupl/ICmk1r2 ipFJ4iO9JtEsgWfWU7mMpj0pzrA4q4v03Bg2uVpxhEXPt4Urz+wyTsKw9g4Yks7piO/c FyaPEVPHKce6P10VUHWT8E8KtgWxjeLpWCH99uxWJwqrViTckpR29grnENzXeuwJhXB5 4PQQ== X-Received: by 10.68.226.197 with SMTP id ru5mr6628096pbc.77.1401351365567; Thu, 29 May 2014 01:16:05 -0700 (PDT) Received: from localhost.localdomain ([14.140.216.146]) by mx.google.com with ESMTPSA id ee5sm5006897pbc.47.2014.05.29.01.16.02 for (version=TLSv1 cipher=RC4-SHA bits=128/128); Thu, 29 May 2014 01:16:05 -0700 (PDT) From: Amit Daniel Kachhap To: linux-pm@vger.kernel.org, linux-acpi@vger.kernel.org Subject: [PATCH v1 3/6] thermal: thermal-core: Add notifications support for the cooling states Date: Thu, 29 May 2014 13:45:31 +0530 Message-Id: <1401351334-11210-4-git-send-email-amit.daniel@samsung.com> X-Mailer: git-send-email 1.7.1 In-Reply-To: <1401351334-11210-1-git-send-email-amit.daniel@samsung.com> References: <1401351334-11210-1-git-send-email-amit.daniel@samsung.com> X-CRM114-Version: 20100106-BlameMichelson ( TRE 0.8.0 (BSD) ) MR-646709E3 X-CRM114-CacheID: sfid-20140529_011627_608657_C6533DE9 X-CRM114-Status: GOOD ( 19.91 ) X-Spam-Score: 0.0 (/) Cc: amit.kachhap@gmail.com, rjw@rjwysocki.net, linux-kernel@vger.kernel.org, edubezval@gmail.com, Zhang Rui , linux-arm-kernel@lists.infradead.org, lenb@kernel.org X-BeenThere: linux-arm-kernel@lists.infradead.org X-Mailman-Version: 2.1.15 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , MIME-Version: 1.0 Sender: "linux-arm-kernel" Errors-To: linux-arm-kernel-bounces+patchwork-linux-arm=patchwork.kernel.org@lists.infradead.org X-Spam-Status: No, score=-2.4 required=5.0 tests=BAYES_00,DKIM_SIGNED, RP_MATCHES_RCVD, T_DKIM_INVALID, UNPARSEABLE_RELAY autolearn=unavailable version=3.3.1 X-Spam-Checker-Version: SpamAssassin 3.3.1 (2010-03-16) on mail.kernel.org X-Virus-Scanned: ClamAV using ClamSMTP This patch adds notification infrastructure for any requests related to cooling states. The notifier structure passed is of both Get/Set type. So the receiver of these can sense the new/cur/max cooling state as decided by thermal governor. In addition to that it can also override the cooling state and may do something interesting after receiving these CPU cooling events such as masking some states, enabling some extra conditional states or perform any extra operation for aggressive thermal cooling. The notfications events can be of type, 1. COOLING_SET_STATE_PRE 2. COOLING_SET_STATE_POST 3. COOLING_GET_CUR_STATE 4. COOLING_GET_MAX_STATE Signed-off-by: Amit Daniel Kachhap --- Documentation/thermal/sysfs-api.txt | 21 +++++++++++ drivers/thermal/thermal_core.c | 69 ++++++++++++++++++++++++++++++++++- include/linux/thermal.h | 21 +++++++++++ 3 files changed, 109 insertions(+), 2 deletions(-) diff --git a/Documentation/thermal/sysfs-api.txt b/Documentation/thermal/sysfs-api.txt index 87519cb..5f45e03 100644 --- a/Documentation/thermal/sysfs-api.txt +++ b/Documentation/thermal/sysfs-api.txt @@ -92,6 +92,27 @@ temperature) and throttle appropriate devices. It deletes the corresponding entry form /sys/class/thermal folder and unbind itself from all the thermal zone devices using it. +1.2.3 int thermal_cooling_register_notifier(struct notifier_block *nb) + + This interface function registers the client notifier handler. The notifier + handler can use this to monitor or update any cooling state requests. + nb: notifier structure containing client notifier handler. + +1.2.4 int thermal_cooling_unregister_notifier(struct notifier_block *nb) + + This interface function unregisters the client notifier handler. + nb: notifier structure containing client notifier handler. + +1.2.5 int thermal_cooling_notify_states(struct thermal_cooling_status *request,enum cooling_state_ops op) + + This interface function invokes the earlier registered cooling states handler. + request: holds the relevant cooling state value. + .cur_state: current cooling state. + .new_state: new cooling state to be set. + .max_state: max cooling state. + .devdata: driver private data pointer. + op: describes various operation supported. + 1.3 interface for binding a thermal zone device with a thermal cooling device 1.3.1 int thermal_zone_bind_cooling_device(struct thermal_zone_device *tz, int trip, struct thermal_cooling_device *cdev, diff --git a/drivers/thermal/thermal_core.c b/drivers/thermal/thermal_core.c index 71b0ec0..1a60f83 100644 --- a/drivers/thermal/thermal_core.c +++ b/drivers/thermal/thermal_core.c @@ -52,6 +52,8 @@ static DEFINE_MUTEX(thermal_idr_lock); static LIST_HEAD(thermal_tz_list); static LIST_HEAD(thermal_cdev_list); static LIST_HEAD(thermal_governor_list); +/* Notfier list to validates/updates the cpufreq cooling states */ +static BLOCKING_NOTIFIER_HEAD(cooling_state_notifier_list); static DEFINE_MUTEX(thermal_list_lock); static DEFINE_MUTEX(thermal_governor_lock); @@ -1073,8 +1075,71 @@ static struct class thermal_class = { }; /** - * __thermal_cooling_device_register() - register a new thermal cooling device - * @np: a pointer to a device tree node. + * thermal_cooling_notify_states - Invoke the necessary cooling states handler. + * @request: holds the relevant cooling state value. say if the cooling state + * operation is of type COOLING_GET_MAX_STATE, then request holds + * the current max cooling state value. + * @op: different operations supported + * + * This API allows the registered user to recieve the different cooling + * notifications like current state, max state and set state. + * + * Return: 0 (success) + */ +int thermal_cooling_notify_states(struct thermal_cooling_status *request, + enum cooling_state_ops op) +{ + /* Invoke the notifiers which have registered for this state change */ + if (op == COOLING_SET_STATE_PRE || + op == COOLING_SET_STATE_POST || + op == COOLING_GET_MAX_STATE || + op == COOLING_GET_CUR_STATE) { + blocking_notifier_call_chain( + &cooling_state_notifier_list, op, request); + } + return 0; +} +EXPORT_SYMBOL_GPL(thermal_cooling_notify_states); + +/** + * thermal_cooling_register_notifier - registers a notifier with thermal cooling. + * @nb: notifier function to register. + * + * Add a driver to receive all cooling notifications like current state, + * max state and set state. The drivers after reading the events can perform + * some mapping like grouping some P states into 1 cooling state. + * + * Return: 0 (success) + */ +int thermal_cooling_register_notifier(struct notifier_block *nb) +{ + int ret = 0; + ret = blocking_notifier_chain_register( + &cooling_state_notifier_list, nb); + return ret; +} +EXPORT_SYMBOL_GPL(thermal_cooling_register_notifier); + +/** + * thermal_cooling_unregister_notifier - unregisters a notifier with thermal + * cooling. + * @nb: notifier function to unregister. + * + * Removes a driver to receive further cooling notifications. + * + * Return: 0 (success) + */ +int thermal_cooling_unregister_notifier(struct notifier_block *nb) +{ + int ret = 0; + ret = blocking_notifier_chain_unregister( + &cooling_state_notifier_list, nb); + return ret; +} +EXPORT_SYMBOL_GPL(thermal_cooling_unregister_notifier); + +/** + * thermal_cooling_device_register() - register a new thermal cooling device * @type: the thermal cooling device type. * @devdata: device private data. * @ops: standard thermal cooling devices callbacks. diff --git a/include/linux/thermal.h b/include/linux/thermal.h index f7e11c7..3fb1b92 100644 --- a/include/linux/thermal.h +++ b/include/linux/thermal.h @@ -106,6 +106,21 @@ enum { }; #define THERMAL_GENL_CMD_MAX (__THERMAL_GENL_CMD_MAX - 1) +/* Cooling states supported operations */ +enum cooling_state_ops { + COOLING_SET_STATE_PRE, + COOLING_SET_STATE_POST, + COOLING_GET_CUR_STATE, + COOLING_GET_MAX_STATE, +}; + +struct thermal_cooling_status { + unsigned long cur_state; + unsigned long new_state; + unsigned long max_state; + void *devdata; +}; + struct thermal_zone_device_ops { int (*bind) (struct thermal_zone_device *, struct thermal_cooling_device *); @@ -285,6 +300,12 @@ struct thermal_cooling_device * thermal_of_cooling_device_register(struct device_node *np, char *, void *, const struct thermal_cooling_device_ops *); void thermal_cooling_device_unregister(struct thermal_cooling_device *); + +int thermal_cooling_notify_states(struct thermal_cooling_status *request, + enum cooling_state_ops op); +int thermal_cooling_register_notifier(struct notifier_block *nb); +int thermal_cooling_unregister_notifier(struct notifier_block *nb); + struct thermal_zone_device *thermal_zone_get_zone_by_name(const char *name); int thermal_zone_get_temp(struct thermal_zone_device *tz, unsigned long *temp);