diff options
author | Antti P Miettinen <amiettinen@nvidia.com> | 2012-08-20 19:36:38 +0300 |
---|---|---|
committer | Dan Willemsen <dwillemsen@nvidia.com> | 2013-09-14 01:33:01 -0700 |
commit | 5ec1068d47f8c22eacaee450424a1158dc4f440d (patch) | |
tree | d2f2299c16a8515031a832f549506e3f60413783 /kernel/power | |
parent | bddadeee2389b9db21229997e74c5684bd6d999a (diff) |
PM QoS: Add disable parameter
For testing purposes it is useful to be able to disable
PM Qos.
Bug 1020898
Bug 917572
Reviewed-on: http://git-master/r/124667
Change-Id: I266f5b5730cfe4705197d8b09db7f9eda6766c7c
Signed-off-by: Antti P Miettinen <amiettinen@nvidia.com>
Signed-off-by: Varun Wadekar <vwadekar@nvidia.com>
Rebase-Id: Re2088674f90436e0b9dd74310d5cda1f9e2868e4
Diffstat (limited to 'kernel/power')
-rw-r--r-- | kernel/power/qos.c | 65 |
1 files changed, 62 insertions, 3 deletions
diff --git a/kernel/power/qos.c b/kernel/power/qos.c index b343ea2fbdb2..8e662c3dc2ef 100644 --- a/kernel/power/qos.c +++ b/kernel/power/qos.c @@ -41,7 +41,7 @@ #include <linux/platform_device.h> #include <linux/init.h> #include <linux/kernel.h> - +#include <linux/moduleparam.h> #include <linux/uaccess.h> #include <linux/export.h> @@ -184,6 +184,8 @@ static const struct file_operations pm_qos_power_fops = { .llseek = noop_llseek, }; +static bool pm_qos_enabled __read_mostly = true; + /* unlocked internal variant */ static inline int pm_qos_get_value(struct pm_qos_constraints *c) { @@ -258,8 +260,12 @@ int pm_qos_update_target(struct pm_qos_constraints *c, struct plist_node *node, ; } - curr_value = pm_qos_get_value(c); - pm_qos_set_value(c, curr_value); + if (pm_qos_enabled) { + curr_value = pm_qos_get_value(c); + pm_qos_set_value(c, curr_value); + } else { + curr_value = c->default_value; + } spin_unlock_irqrestore(&pm_qos_lock, flags); @@ -495,6 +501,59 @@ void pm_qos_remove_request(struct pm_qos_request *req) } EXPORT_SYMBOL_GPL(pm_qos_remove_request); +static int pm_qos_enabled_set(const char *arg, const struct kernel_param *kp) +{ + unsigned long flags; + bool old; + s32 prev[PM_QOS_NUM_CLASSES], curr[PM_QOS_NUM_CLASSES]; + int ret, i; + + old = pm_qos_enabled; + ret = param_set_bool(arg, kp); + if (ret != 0) { + pr_warn("%s: cannot set PM QoS enable to %s\n", + __FUNCTION__, arg); + return ret; + } + spin_lock_irqsave(&pm_qos_lock, flags); + for (i = 1; i < PM_QOS_NUM_CLASSES; i++) + prev[i] = pm_qos_read_value(pm_qos_array[i]->constraints); + if (old && !pm_qos_enabled) { + /* got disabled */ + for (i = 1; i < PM_QOS_NUM_CLASSES; i++) { + curr[i] = pm_qos_array[i]->constraints->default_value; + pm_qos_set_value(pm_qos_array[i]->constraints, curr[i]); + } + } else if (!old && pm_qos_enabled) { + /* got enabled */ + for (i = 1; i < PM_QOS_NUM_CLASSES; i++) { + curr[i] = pm_qos_get_value(pm_qos_array[i]->constraints); + pm_qos_set_value(pm_qos_array[i]->constraints, curr[i]); + } + } + spin_unlock_irqrestore(&pm_qos_lock, flags); + for (i = 1; i < PM_QOS_NUM_CLASSES; i++) + if (prev[i] != curr[i]) + blocking_notifier_call_chain( + pm_qos_array[i]->constraints->notifiers, + (unsigned long)curr[i], + NULL); + + return ret; +} + +static int pm_qos_enabled_get(char *buffer, const struct kernel_param *kp) +{ + return param_get_bool(buffer, kp); +} + +static struct kernel_param_ops pm_qos_enabled_ops = { + .set = pm_qos_enabled_set, + .get = pm_qos_enabled_get, +}; + +module_param_cb(enable, &pm_qos_enabled_ops, &pm_qos_enabled, 0644); + /** * pm_qos_add_notifier - sets notification entry for changes to target value * @pm_qos_class: identifies which qos target changes should be notified. |