Commit e39473d0b9448e770f49b0b15e514be884264438

Authored by Rafael J. Wysocki
1 parent ae0fb4b72c

PM / QoS: Make it possible to expose PM QoS device flags to user space

Define two device PM QoS flags, PM_QOS_FLAG_NO_POWER_OFF
and PM_QOS_FLAG_REMOTE_WAKEUP, and introduce routines
dev_pm_qos_expose_flags() and dev_pm_qos_hide_flags() allowing the
caller to expose those two flags to user space or to hide them
from it, respectively.

After the flags have been exposed, user space will see two
additional sysfs attributes, pm_qos_no_power_off and
pm_qos_remote_wakeup, under the device's /sys/devices/.../power/
directory.  Then, writing 1 to one of them will update the
PM QoS flags request owned by user space so that the corresponding
flag is requested to be set.  In turn, writing 0 to one of them
will cause the corresponding flag in the user space's request to
be cleared (however, the owners of the other PM QoS flags requests
for the same device may still request the flag to be set and it
may be effectively set even if user space doesn't request that).

Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Reviewed-by: Jean Pihet <j-pihet@ti.com>
Acked-by: mark gross <markgross@thegnar.org>

Showing 6 changed files with 278 additions and 48 deletions Side-by-side Diff

Documentation/ABI/testing/sysfs-devices-power
... ... @@ -204,4 +204,35 @@
204 204  
205 205 This attribute has no effect on system-wide suspend/resume and
206 206 hibernation.
  207 +
  208 +What: /sys/devices/.../power/pm_qos_no_power_off
  209 +Date: September 2012
  210 +Contact: Rafael J. Wysocki <rjw@sisk.pl>
  211 +Description:
  212 + The /sys/devices/.../power/pm_qos_no_power_off attribute
  213 + is used for manipulating the PM QoS "no power off" flag. If
  214 + set, this flag indicates to the kernel that power should not
  215 + be removed entirely from the device.
  216 +
  217 + Not all drivers support this attribute. If it isn't supported,
  218 + it is not present.
  219 +
  220 + This attribute has no effect on system-wide suspend/resume and
  221 + hibernation.
  222 +
  223 +What: /sys/devices/.../power/pm_qos_remote_wakeup
  224 +Date: September 2012
  225 +Contact: Rafael J. Wysocki <rjw@sisk.pl>
  226 +Description:
  227 + The /sys/devices/.../power/pm_qos_remote_wakeup attribute
  228 + is used for manipulating the PM QoS "remote wakeup required"
  229 + flag. If set, this flag indicates to the kernel that the
  230 + device is a source of user events that have to be signaled from
  231 + its low-power states.
  232 +
  233 + Not all drivers support this attribute. If it isn't supported,
  234 + it is not present.
  235 +
  236 + This attribute has no effect on system-wide suspend/resume and
  237 + hibernation.
drivers/base/power/power.h
... ... @@ -93,8 +93,10 @@
93 93 extern void rpm_sysfs_remove(struct device *dev);
94 94 extern int wakeup_sysfs_add(struct device *dev);
95 95 extern void wakeup_sysfs_remove(struct device *dev);
96   -extern int pm_qos_sysfs_add(struct device *dev);
97   -extern void pm_qos_sysfs_remove(struct device *dev);
  96 +extern int pm_qos_sysfs_add_latency(struct device *dev);
  97 +extern void pm_qos_sysfs_remove_latency(struct device *dev);
  98 +extern int pm_qos_sysfs_add_flags(struct device *dev);
  99 +extern void pm_qos_sysfs_remove_flags(struct device *dev);
98 100  
99 101 #else /* CONFIG_PM */
100 102  
drivers/base/power/qos.c
... ... @@ -40,6 +40,7 @@
40 40 #include <linux/device.h>
41 41 #include <linux/mutex.h>
42 42 #include <linux/export.h>
  43 +#include <linux/pm_runtime.h>
43 44  
44 45 #include "power.h"
45 46  
... ... @@ -322,6 +323,37 @@
322 323 EXPORT_SYMBOL_GPL(dev_pm_qos_add_request);
323 324  
324 325 /**
  326 + * __dev_pm_qos_update_request - Modify an existing device PM QoS request.
  327 + * @req : PM QoS request to modify.
  328 + * @new_value: New value to request.
  329 + */
  330 +static int __dev_pm_qos_update_request(struct dev_pm_qos_request *req,
  331 + s32 new_value)
  332 +{
  333 + s32 curr_value;
  334 + int ret = 0;
  335 +
  336 + if (!req->dev->power.qos)
  337 + return -ENODEV;
  338 +
  339 + switch(req->type) {
  340 + case DEV_PM_QOS_LATENCY:
  341 + curr_value = req->data.pnode.prio;
  342 + break;
  343 + case DEV_PM_QOS_FLAGS:
  344 + curr_value = req->data.flr.flags;
  345 + break;
  346 + default:
  347 + return -EINVAL;
  348 + }
  349 +
  350 + if (curr_value != new_value)
  351 + ret = apply_constraint(req, PM_QOS_UPDATE_REQ, new_value);
  352 +
  353 + return ret;
  354 +}
  355 +
  356 +/**
325 357 * dev_pm_qos_update_request - modifies an existing qos request
326 358 * @req : handle to list element holding a dev_pm_qos request to use
327 359 * @new_value: defines the qos request
328 360  
... ... @@ -336,11 +368,9 @@
336 368 * -EINVAL in case of wrong parameters, -ENODEV if the device has been
337 369 * removed from the system
338 370 */
339   -int dev_pm_qos_update_request(struct dev_pm_qos_request *req,
340   - s32 new_value)
  371 +int dev_pm_qos_update_request(struct dev_pm_qos_request *req, s32 new_value)
341 372 {
342   - s32 curr_value;
343   - int ret = 0;
  373 + int ret;
344 374  
345 375 if (!req) /*guard against callers passing in null */
346 376 return -EINVAL;
347 377  
... ... @@ -350,29 +380,9 @@
350 380 return -EINVAL;
351 381  
352 382 mutex_lock(&dev_pm_qos_mtx);
353   -
354   - if (!req->dev->power.qos) {
355   - ret = -ENODEV;
356   - goto out;
357   - }
358   -
359   - switch(req->type) {
360   - case DEV_PM_QOS_LATENCY:
361   - curr_value = req->data.pnode.prio;
362   - break;
363   - case DEV_PM_QOS_FLAGS:
364   - curr_value = req->data.flr.flags;
365   - break;
366   - default:
367   - ret = -EINVAL;
368   - goto out;
369   - }
370   -
371   - if (curr_value != new_value)
372   - ret = apply_constraint(req, PM_QOS_UPDATE_REQ, new_value);
373   -
374   - out:
  383 + __dev_pm_qos_update_request(req, new_value);
375 384 mutex_unlock(&dev_pm_qos_mtx);
  385 +
376 386 return ret;
377 387 }
378 388 EXPORT_SYMBOL_GPL(dev_pm_qos_update_request);
379 389  
... ... @@ -533,10 +543,19 @@
533 543 EXPORT_SYMBOL_GPL(dev_pm_qos_add_ancestor_request);
534 544  
535 545 #ifdef CONFIG_PM_RUNTIME
536   -static void __dev_pm_qos_drop_user_request(struct device *dev)
  546 +static void __dev_pm_qos_drop_user_request(struct device *dev,
  547 + enum dev_pm_qos_req_type type)
537 548 {
538   - dev_pm_qos_remove_request(dev->power.pq_req);
539   - dev->power.pq_req = NULL;
  549 + switch(type) {
  550 + case DEV_PM_QOS_LATENCY:
  551 + dev_pm_qos_remove_request(dev->power.qos->latency_req);
  552 + dev->power.qos->latency_req = NULL;
  553 + break;
  554 + case DEV_PM_QOS_FLAGS:
  555 + dev_pm_qos_remove_request(dev->power.qos->flags_req);
  556 + dev->power.qos->flags_req = NULL;
  557 + break;
  558 + }
540 559 }
541 560  
542 561 /**
... ... @@ -552,7 +571,7 @@
552 571 if (!device_is_registered(dev) || value < 0)
553 572 return -EINVAL;
554 573  
555   - if (dev->power.pq_req)
  574 + if (dev->power.qos && dev->power.qos->latency_req)
556 575 return -EEXIST;
557 576  
558 577 req = kzalloc(sizeof(*req), GFP_KERNEL);
559 578  
... ... @@ -563,10 +582,10 @@
563 582 if (ret < 0)
564 583 return ret;
565 584  
566   - dev->power.pq_req = req;
567   - ret = pm_qos_sysfs_add(dev);
  585 + dev->power.qos->latency_req = req;
  586 + ret = pm_qos_sysfs_add_latency(dev);
568 587 if (ret)
569   - __dev_pm_qos_drop_user_request(dev);
  588 + __dev_pm_qos_drop_user_request(dev, DEV_PM_QOS_LATENCY);
570 589  
571 590 return ret;
572 591 }
573 592  
... ... @@ -578,11 +597,88 @@
578 597 */
579 598 void dev_pm_qos_hide_latency_limit(struct device *dev)
580 599 {
581   - if (dev->power.pq_req) {
582   - pm_qos_sysfs_remove(dev);
583   - __dev_pm_qos_drop_user_request(dev);
  600 + if (dev->power.qos && dev->power.qos->latency_req) {
  601 + pm_qos_sysfs_remove_latency(dev);
  602 + __dev_pm_qos_drop_user_request(dev, DEV_PM_QOS_LATENCY);
584 603 }
585 604 }
586 605 EXPORT_SYMBOL_GPL(dev_pm_qos_hide_latency_limit);
  606 +
  607 +/**
  608 + * dev_pm_qos_expose_flags - Expose PM QoS flags of a device to user space.
  609 + * @dev: Device whose PM QoS flags are to be exposed to user space.
  610 + * @val: Initial values of the flags.
  611 + */
  612 +int dev_pm_qos_expose_flags(struct device *dev, s32 val)
  613 +{
  614 + struct dev_pm_qos_request *req;
  615 + int ret;
  616 +
  617 + if (!device_is_registered(dev))
  618 + return -EINVAL;
  619 +
  620 + if (dev->power.qos && dev->power.qos->flags_req)
  621 + return -EEXIST;
  622 +
  623 + req = kzalloc(sizeof(*req), GFP_KERNEL);
  624 + if (!req)
  625 + return -ENOMEM;
  626 +
  627 + ret = dev_pm_qos_add_request(dev, req, DEV_PM_QOS_FLAGS, val);
  628 + if (ret < 0)
  629 + return ret;
  630 +
  631 + dev->power.qos->flags_req = req;
  632 + ret = pm_qos_sysfs_add_flags(dev);
  633 + if (ret)
  634 + __dev_pm_qos_drop_user_request(dev, DEV_PM_QOS_FLAGS);
  635 +
  636 + return ret;
  637 +}
  638 +EXPORT_SYMBOL_GPL(dev_pm_qos_expose_flags);
  639 +
  640 +/**
  641 + * dev_pm_qos_hide_flags - Hide PM QoS flags of a device from user space.
  642 + * @dev: Device whose PM QoS flags are to be hidden from user space.
  643 + */
  644 +void dev_pm_qos_hide_flags(struct device *dev)
  645 +{
  646 + if (dev->power.qos && dev->power.qos->flags_req) {
  647 + pm_qos_sysfs_remove_flags(dev);
  648 + __dev_pm_qos_drop_user_request(dev, DEV_PM_QOS_FLAGS);
  649 + }
  650 +}
  651 +EXPORT_SYMBOL_GPL(dev_pm_qos_hide_flags);
  652 +
  653 +/**
  654 + * dev_pm_qos_update_flags - Update PM QoS flags request owned by user space.
  655 + * @dev: Device to update the PM QoS flags request for.
  656 + * @mask: Flags to set/clear.
  657 + * @set: Whether to set or clear the flags (true means set).
  658 + */
  659 +int dev_pm_qos_update_flags(struct device *dev, s32 mask, bool set)
  660 +{
  661 + s32 value;
  662 + int ret;
  663 +
  664 + if (!dev->power.qos || !dev->power.qos->flags_req)
  665 + return -EINVAL;
  666 +
  667 + pm_runtime_get_sync(dev);
  668 + mutex_lock(&dev_pm_qos_mtx);
  669 +
  670 + value = dev_pm_qos_requested_flags(dev);
  671 + if (set)
  672 + value |= mask;
  673 + else
  674 + value &= ~mask;
  675 +
  676 + ret = __dev_pm_qos_update_request(dev->power.qos->flags_req, value);
  677 +
  678 + mutex_unlock(&dev_pm_qos_mtx);
  679 + pm_runtime_put(dev);
  680 +
  681 + return ret;
  682 +}
587 683 #endif /* CONFIG_PM_RUNTIME */
drivers/base/power/sysfs.c
... ... @@ -221,7 +221,7 @@
221 221 static ssize_t pm_qos_latency_show(struct device *dev,
222 222 struct device_attribute *attr, char *buf)
223 223 {
224   - return sprintf(buf, "%d\n", dev->power.pq_req->data.pnode.prio);
  224 + return sprintf(buf, "%d\n", dev_pm_qos_requested_latency(dev));
225 225 }
226 226  
227 227 static ssize_t pm_qos_latency_store(struct device *dev,
228 228  
... ... @@ -237,12 +237,66 @@
237 237 if (value < 0)
238 238 return -EINVAL;
239 239  
240   - ret = dev_pm_qos_update_request(dev->power.pq_req, value);
  240 + ret = dev_pm_qos_update_request(dev->power.qos->latency_req, value);
241 241 return ret < 0 ? ret : n;
242 242 }
243 243  
244 244 static DEVICE_ATTR(pm_qos_resume_latency_us, 0644,
245 245 pm_qos_latency_show, pm_qos_latency_store);
  246 +
  247 +static ssize_t pm_qos_no_power_off_show(struct device *dev,
  248 + struct device_attribute *attr,
  249 + char *buf)
  250 +{
  251 + return sprintf(buf, "%d\n", !!(dev_pm_qos_requested_flags(dev)
  252 + & PM_QOS_FLAG_NO_POWER_OFF));
  253 +}
  254 +
  255 +static ssize_t pm_qos_no_power_off_store(struct device *dev,
  256 + struct device_attribute *attr,
  257 + const char *buf, size_t n)
  258 +{
  259 + int ret;
  260 +
  261 + if (kstrtoint(buf, 0, &ret))
  262 + return -EINVAL;
  263 +
  264 + if (ret != 0 && ret != 1)
  265 + return -EINVAL;
  266 +
  267 + ret = dev_pm_qos_update_flags(dev, PM_QOS_FLAG_NO_POWER_OFF, ret);
  268 + return ret < 0 ? ret : n;
  269 +}
  270 +
  271 +static DEVICE_ATTR(pm_qos_no_power_off, 0644,
  272 + pm_qos_no_power_off_show, pm_qos_no_power_off_store);
  273 +
  274 +static ssize_t pm_qos_remote_wakeup_show(struct device *dev,
  275 + struct device_attribute *attr,
  276 + char *buf)
  277 +{
  278 + return sprintf(buf, "%d\n", !!(dev_pm_qos_requested_flags(dev)
  279 + & PM_QOS_FLAG_REMOTE_WAKEUP));
  280 +}
  281 +
  282 +static ssize_t pm_qos_remote_wakeup_store(struct device *dev,
  283 + struct device_attribute *attr,
  284 + const char *buf, size_t n)
  285 +{
  286 + int ret;
  287 +
  288 + if (kstrtoint(buf, 0, &ret))
  289 + return -EINVAL;
  290 +
  291 + if (ret != 0 && ret != 1)
  292 + return -EINVAL;
  293 +
  294 + ret = dev_pm_qos_update_flags(dev, PM_QOS_FLAG_REMOTE_WAKEUP, ret);
  295 + return ret < 0 ? ret : n;
  296 +}
  297 +
  298 +static DEVICE_ATTR(pm_qos_remote_wakeup, 0644,
  299 + pm_qos_remote_wakeup_show, pm_qos_remote_wakeup_store);
246 300 #endif /* CONFIG_PM_RUNTIME */
247 301  
248 302 #ifdef CONFIG_PM_SLEEP
249 303  
250 304  
251 305  
... ... @@ -564,17 +618,29 @@
564 618 .attrs = runtime_attrs,
565 619 };
566 620  
567   -static struct attribute *pm_qos_attrs[] = {
  621 +static struct attribute *pm_qos_latency_attrs[] = {
568 622 #ifdef CONFIG_PM_RUNTIME
569 623 &dev_attr_pm_qos_resume_latency_us.attr,
570 624 #endif /* CONFIG_PM_RUNTIME */
571 625 NULL,
572 626 };
573   -static struct attribute_group pm_qos_attr_group = {
  627 +static struct attribute_group pm_qos_latency_attr_group = {
574 628 .name = power_group_name,
575   - .attrs = pm_qos_attrs,
  629 + .attrs = pm_qos_latency_attrs,
576 630 };
577 631  
  632 +static struct attribute *pm_qos_flags_attrs[] = {
  633 +#ifdef CONFIG_PM_RUNTIME
  634 + &dev_attr_pm_qos_no_power_off.attr,
  635 + &dev_attr_pm_qos_remote_wakeup.attr,
  636 +#endif /* CONFIG_PM_RUNTIME */
  637 + NULL,
  638 +};
  639 +static struct attribute_group pm_qos_flags_attr_group = {
  640 + .name = power_group_name,
  641 + .attrs = pm_qos_flags_attrs,
  642 +};
  643 +
578 644 int dpm_sysfs_add(struct device *dev)
579 645 {
580 646 int rc;
581 647  
582 648  
583 649  
... ... @@ -615,14 +681,24 @@
615 681 sysfs_unmerge_group(&dev->kobj, &pm_wakeup_attr_group);
616 682 }
617 683  
618   -int pm_qos_sysfs_add(struct device *dev)
  684 +int pm_qos_sysfs_add_latency(struct device *dev)
619 685 {
620   - return sysfs_merge_group(&dev->kobj, &pm_qos_attr_group);
  686 + return sysfs_merge_group(&dev->kobj, &pm_qos_latency_attr_group);
621 687 }
622 688  
623   -void pm_qos_sysfs_remove(struct device *dev)
  689 +void pm_qos_sysfs_remove_latency(struct device *dev)
624 690 {
625   - sysfs_unmerge_group(&dev->kobj, &pm_qos_attr_group);
  691 + sysfs_unmerge_group(&dev->kobj, &pm_qos_latency_attr_group);
  692 +}
  693 +
  694 +int pm_qos_sysfs_add_flags(struct device *dev)
  695 +{
  696 + return sysfs_merge_group(&dev->kobj, &pm_qos_flags_attr_group);
  697 +}
  698 +
  699 +void pm_qos_sysfs_remove_flags(struct device *dev)
  700 +{
  701 + sysfs_unmerge_group(&dev->kobj, &pm_qos_flags_attr_group);
626 702 }
627 703  
628 704 void rpm_sysfs_remove(struct device *dev)
... ... @@ -546,7 +546,6 @@
546 546 unsigned long active_jiffies;
547 547 unsigned long suspended_jiffies;
548 548 unsigned long accounting_timestamp;
549   - struct dev_pm_qos_request *pq_req;
550 549 #endif
551 550 struct pm_subsys_data *subsys_data; /* Owned by the subsystem. */
552 551 struct dev_pm_qos *qos;
include/linux/pm_qos.h
... ... @@ -34,6 +34,9 @@
34 34 #define PM_QOS_NETWORK_THROUGHPUT_DEFAULT_VALUE 0
35 35 #define PM_QOS_DEV_LAT_DEFAULT_VALUE 0
36 36  
  37 +#define PM_QOS_FLAG_NO_POWER_OFF (1 << 0)
  38 +#define PM_QOS_FLAG_REMOTE_WAKEUP (1 << 1)
  39 +
37 40 struct pm_qos_request {
38 41 struct plist_node node;
39 42 int pm_qos_class;
... ... @@ -86,6 +89,8 @@
86 89 struct dev_pm_qos {
87 90 struct pm_qos_constraints latency;
88 91 struct pm_qos_flags flags;
  92 + struct dev_pm_qos_request *latency_req;
  93 + struct dev_pm_qos_request *flags_req;
89 94 };
90 95  
91 96 /* Action requested to pm_qos_update_target */
92 97  
... ... @@ -187,10 +192,31 @@
187 192 #ifdef CONFIG_PM_RUNTIME
188 193 int dev_pm_qos_expose_latency_limit(struct device *dev, s32 value);
189 194 void dev_pm_qos_hide_latency_limit(struct device *dev);
  195 +int dev_pm_qos_expose_flags(struct device *dev, s32 value);
  196 +void dev_pm_qos_hide_flags(struct device *dev);
  197 +int dev_pm_qos_update_flags(struct device *dev, s32 mask, bool set);
  198 +
  199 +static inline s32 dev_pm_qos_requested_latency(struct device *dev)
  200 +{
  201 + return dev->power.qos->latency_req->data.pnode.prio;
  202 +}
  203 +
  204 +static inline s32 dev_pm_qos_requested_flags(struct device *dev)
  205 +{
  206 + return dev->power.qos->flags_req->data.flr.flags;
  207 +}
190 208 #else
191 209 static inline int dev_pm_qos_expose_latency_limit(struct device *dev, s32 value)
192 210 { return 0; }
193 211 static inline void dev_pm_qos_hide_latency_limit(struct device *dev) {}
  212 +static inline int dev_pm_qos_expose_flags(struct device *dev, s32 value)
  213 + { return 0; }
  214 +static inline void dev_pm_qos_hide_flags(struct device *dev) {}
  215 +static inline int dev_pm_qos_update_flags(struct device *dev, s32 m, bool set)
  216 + { return 0; }
  217 +
  218 +static inline s32 dev_pm_qos_requested_latency(struct device *dev) { return 0; }
  219 +static inline s32 dev_pm_qos_requested_flags(struct device *dev) { return 0; }
194 220 #endif
195 221  
196 222 #endif