diff options
-rw-r--r-- | Documentation/ABI/testing/sysfs-class-devfreq | 44 | ||||
-rw-r--r-- | drivers/devfreq/devfreq.c | 69 |
2 files changed, 113 insertions, 0 deletions
diff --git a/Documentation/ABI/testing/sysfs-class-devfreq b/Documentation/ABI/testing/sysfs-class-devfreq new file mode 100644 index 000000000000..0ec855f479ce --- /dev/null +++ b/Documentation/ABI/testing/sysfs-class-devfreq | |||
@@ -0,0 +1,44 @@ | |||
1 | What: /sys/class/devfreq/.../ | ||
2 | Date: September 2011 | ||
3 | Contact: MyungJoo Ham <myungjoo.ham@samsung.com> | ||
4 | Description: | ||
5 | Provide a place in sysfs for the devfreq objects. | ||
6 | This allows accessing various devfreq specific variables. | ||
7 | The name of devfreq object denoted as ... is same as the | ||
8 | name of device using devfreq. | ||
9 | |||
10 | What: /sys/class/devfreq/.../governor | ||
11 | Date: September 2011 | ||
12 | Contact: MyungJoo Ham <myungjoo.ham@samsung.com> | ||
13 | Description: | ||
14 | The /sys/class/devfreq/.../governor shows the name of the | ||
15 | governor used by the corresponding devfreq object. | ||
16 | |||
17 | What: /sys/class/devfreq/.../cur_freq | ||
18 | Date: September 2011 | ||
19 | Contact: MyungJoo Ham <myungjoo.ham@samsung.com> | ||
20 | Description: | ||
21 | The /sys/class/devfreq/.../cur_freq shows the current | ||
22 | frequency of the corresponding devfreq object. | ||
23 | |||
24 | What: /sys/class/devfreq/.../central_polling | ||
25 | Date: September 2011 | ||
26 | Contact: MyungJoo Ham <myungjoo.ham@samsung.com> | ||
27 | Description: | ||
28 | The /sys/class/devfreq/.../central_polling shows whether | ||
29 | the devfreq ojbect is using devfreq-provided central | ||
30 | polling mechanism or not. | ||
31 | |||
32 | What: /sys/class/devfreq/.../polling_interval | ||
33 | Date: September 2011 | ||
34 | Contact: MyungJoo Ham <myungjoo.ham@samsung.com> | ||
35 | Description: | ||
36 | The /sys/class/devfreq/.../polling_interval shows and sets | ||
37 | the requested polling interval of the corresponding devfreq | ||
38 | object. The values are represented in ms. If the value is | ||
39 | less than 1 jiffy, it is considered to be 0, which means | ||
40 | no polling. This value is meaningless if the governor is | ||
41 | not polling; thus. If the governor is not using | ||
42 | devfreq-provided central polling | ||
43 | (/sys/class/devfreq/.../central_polling is 0), this value | ||
44 | may be useless. | ||
diff --git a/drivers/devfreq/devfreq.c b/drivers/devfreq/devfreq.c index f3100b19f798..5d15b812377b 100644 --- a/drivers/devfreq/devfreq.c +++ b/drivers/devfreq/devfreq.c | |||
@@ -437,6 +437,74 @@ int devfreq_remove_device(struct devfreq *devfreq) | |||
437 | return 0; | 437 | return 0; |
438 | } | 438 | } |
439 | 439 | ||
440 | static ssize_t show_governor(struct device *dev, | ||
441 | struct device_attribute *attr, char *buf) | ||
442 | { | ||
443 | return sprintf(buf, "%s\n", to_devfreq(dev)->governor->name); | ||
444 | } | ||
445 | |||
446 | static ssize_t show_freq(struct device *dev, | ||
447 | struct device_attribute *attr, char *buf) | ||
448 | { | ||
449 | return sprintf(buf, "%lu\n", to_devfreq(dev)->previous_freq); | ||
450 | } | ||
451 | |||
452 | static ssize_t show_polling_interval(struct device *dev, | ||
453 | struct device_attribute *attr, char *buf) | ||
454 | { | ||
455 | return sprintf(buf, "%d\n", to_devfreq(dev)->profile->polling_ms); | ||
456 | } | ||
457 | |||
458 | static ssize_t store_polling_interval(struct device *dev, | ||
459 | struct device_attribute *attr, | ||
460 | const char *buf, size_t count) | ||
461 | { | ||
462 | struct devfreq *df = to_devfreq(dev); | ||
463 | unsigned int value; | ||
464 | int ret; | ||
465 | |||
466 | ret = sscanf(buf, "%u", &value); | ||
467 | if (ret != 1) | ||
468 | goto out; | ||
469 | |||
470 | mutex_lock(&df->lock); | ||
471 | df->profile->polling_ms = value; | ||
472 | df->next_polling = df->polling_jiffies | ||
473 | = msecs_to_jiffies(value); | ||
474 | mutex_unlock(&df->lock); | ||
475 | |||
476 | ret = count; | ||
477 | |||
478 | if (df->governor->no_central_polling) | ||
479 | goto out; | ||
480 | |||
481 | mutex_lock(&devfreq_list_lock); | ||
482 | if (df->next_polling > 0 && !polling) { | ||
483 | polling = true; | ||
484 | queue_delayed_work(devfreq_wq, &devfreq_work, | ||
485 | df->next_polling); | ||
486 | } | ||
487 | mutex_unlock(&devfreq_list_lock); | ||
488 | out: | ||
489 | return ret; | ||
490 | } | ||
491 | |||
492 | static ssize_t show_central_polling(struct device *dev, | ||
493 | struct device_attribute *attr, char *buf) | ||
494 | { | ||
495 | return sprintf(buf, "%d\n", | ||
496 | !to_devfreq(dev)->governor->no_central_polling); | ||
497 | } | ||
498 | |||
499 | static struct device_attribute devfreq_attrs[] = { | ||
500 | __ATTR(governor, S_IRUGO, show_governor, NULL), | ||
501 | __ATTR(cur_freq, S_IRUGO, show_freq, NULL), | ||
502 | __ATTR(central_polling, S_IRUGO, show_central_polling, NULL), | ||
503 | __ATTR(polling_interval, S_IRUGO | S_IWUSR, show_polling_interval, | ||
504 | store_polling_interval), | ||
505 | { }, | ||
506 | }; | ||
507 | |||
440 | /** | 508 | /** |
441 | * devfreq_start_polling() - Initialize data structure for devfreq framework and | 509 | * devfreq_start_polling() - Initialize data structure for devfreq framework and |
442 | * start polling registered devfreq devices. | 510 | * start polling registered devfreq devices. |
@@ -461,6 +529,7 @@ static int __init devfreq_init(void) | |||
461 | pr_err("%s: couldn't create class\n", __FILE__); | 529 | pr_err("%s: couldn't create class\n", __FILE__); |
462 | return PTR_ERR(devfreq_class); | 530 | return PTR_ERR(devfreq_class); |
463 | } | 531 | } |
532 | devfreq_class->dev_attrs = devfreq_attrs; | ||
464 | return 0; | 533 | return 0; |
465 | } | 534 | } |
466 | subsys_initcall(devfreq_init); | 535 | subsys_initcall(devfreq_init); |