PM / Sleep: Change wakeup source statistics to follow Android

Wakeup statistics used by Android are slightly different from what we
have in wakeup sources at the moment and there aren't any known
users of those statistics other than Android, so modify them to make
it easier for Android to switch to wakeup sources.

This removes the struct wakeup_source's hit_cout field, which is very
rough and therefore not very useful, and adds two new fields,
wakeup_count and expire_count.  The first one tracks how many times
the wakeup source is activated with events_check_enabled set (which
roughly corresponds to the situations when a system power transition
to a sleep state is in progress and would be aborted by this wakeup
source if it were the only active one at that time) and the second
one is the number of times the wakeup source has been activated with
a timeout that expired.

Additionally, the last_time field is now updated when the wakeup
source is deactivated too (previously it was only updated during
the wakeup source's activation), which seems to be what Android does
with the analogous counter for wakelocks.

Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Acked-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
Rafael J. Wysocki 2012-04-29 22:52:52 +02:00
parent 60af106691
commit 30e3ce6dcb
4 changed files with 77 additions and 52 deletions

View File

@ -96,16 +96,26 @@ Description:
is read-only. If the device is not enabled to wake up the is read-only. If the device is not enabled to wake up the
system from sleep states, this attribute is not present. system from sleep states, this attribute is not present.
What: /sys/devices/.../power/wakeup_hit_count What: /sys/devices/.../power/wakeup_abort_count
Date: September 2010 Date: February 2012
Contact: Rafael J. Wysocki <rjw@sisk.pl> Contact: Rafael J. Wysocki <rjw@sisk.pl>
Description: Description:
The /sys/devices/.../wakeup_hit_count attribute contains the The /sys/devices/.../wakeup_abort_count attribute contains the
number of times the processing of a wakeup event associated with number of times the processing of a wakeup event associated with
the device might prevent the system from entering a sleep state. the device might have aborted system transition into a sleep
This attribute is read-only. If the device is not enabled to state in progress. This attribute is read-only. If the device
wake up the system from sleep states, this attribute is not is not enabled to wake up the system from sleep states, this
present. attribute is not present.
What: /sys/devices/.../power/wakeup_expire_count
Date: February 2012
Contact: Rafael J. Wysocki <rjw@sisk.pl>
Description:
The /sys/devices/.../wakeup_expire_count attribute contains the
number of times a wakeup event associated with the device has
been reported with a timeout that expired. This attribute is
read-only. If the device is not enabled to wake up the system
from sleep states, this attribute is not present.
What: /sys/devices/.../power/wakeup_active What: /sys/devices/.../power/wakeup_active
Date: September 2010 Date: September 2010

View File

@ -314,22 +314,41 @@ static ssize_t wakeup_active_count_show(struct device *dev,
static DEVICE_ATTR(wakeup_active_count, 0444, wakeup_active_count_show, NULL); static DEVICE_ATTR(wakeup_active_count, 0444, wakeup_active_count_show, NULL);
static ssize_t wakeup_hit_count_show(struct device *dev, static ssize_t wakeup_abort_count_show(struct device *dev,
struct device_attribute *attr, char *buf) struct device_attribute *attr,
char *buf)
{ {
unsigned long count = 0; unsigned long count = 0;
bool enabled = false; bool enabled = false;
spin_lock_irq(&dev->power.lock); spin_lock_irq(&dev->power.lock);
if (dev->power.wakeup) { if (dev->power.wakeup) {
count = dev->power.wakeup->hit_count; count = dev->power.wakeup->wakeup_count;
enabled = true; enabled = true;
} }
spin_unlock_irq(&dev->power.lock); spin_unlock_irq(&dev->power.lock);
return enabled ? sprintf(buf, "%lu\n", count) : sprintf(buf, "\n"); return enabled ? sprintf(buf, "%lu\n", count) : sprintf(buf, "\n");
} }
static DEVICE_ATTR(wakeup_hit_count, 0444, wakeup_hit_count_show, NULL); static DEVICE_ATTR(wakeup_abort_count, 0444, wakeup_abort_count_show, NULL);
static ssize_t wakeup_expire_count_show(struct device *dev,
struct device_attribute *attr,
char *buf)
{
unsigned long count = 0;
bool enabled = false;
spin_lock_irq(&dev->power.lock);
if (dev->power.wakeup) {
count = dev->power.wakeup->expire_count;
enabled = true;
}
spin_unlock_irq(&dev->power.lock);
return enabled ? sprintf(buf, "%lu\n", count) : sprintf(buf, "\n");
}
static DEVICE_ATTR(wakeup_expire_count, 0444, wakeup_expire_count_show, NULL);
static ssize_t wakeup_active_show(struct device *dev, static ssize_t wakeup_active_show(struct device *dev,
struct device_attribute *attr, char *buf) struct device_attribute *attr, char *buf)
@ -486,7 +505,8 @@ static struct attribute *wakeup_attrs[] = {
&dev_attr_wakeup.attr, &dev_attr_wakeup.attr,
&dev_attr_wakeup_count.attr, &dev_attr_wakeup_count.attr,
&dev_attr_wakeup_active_count.attr, &dev_attr_wakeup_active_count.attr,
&dev_attr_wakeup_hit_count.attr, &dev_attr_wakeup_abort_count.attr,
&dev_attr_wakeup_expire_count.attr,
&dev_attr_wakeup_active.attr, &dev_attr_wakeup_active.attr,
&dev_attr_wakeup_total_time_ms.attr, &dev_attr_wakeup_total_time_ms.attr,
&dev_attr_wakeup_max_time_ms.attr, &dev_attr_wakeup_max_time_ms.attr,

View File

@ -21,7 +21,7 @@
* If set, the suspend/hibernate code will abort transitions to a sleep state * If set, the suspend/hibernate code will abort transitions to a sleep state
* if wakeup events are registered during or immediately before the transition. * if wakeup events are registered during or immediately before the transition.
*/ */
bool events_check_enabled; bool events_check_enabled __read_mostly;
/* /*
* Combined counters of registered wakeup events and wakeup events in progress. * Combined counters of registered wakeup events and wakeup events in progress.
@ -382,6 +382,21 @@ static void wakeup_source_activate(struct wakeup_source *ws)
atomic_inc(&combined_event_count); atomic_inc(&combined_event_count);
} }
/**
* wakeup_source_report_event - Report wakeup event using the given source.
* @ws: Wakeup source to report the event for.
*/
static void wakeup_source_report_event(struct wakeup_source *ws)
{
ws->event_count++;
/* This is racy, but the counter is approximate anyway. */
if (events_check_enabled)
ws->wakeup_count++;
if (!ws->active)
wakeup_source_activate(ws);
}
/** /**
* __pm_stay_awake - Notify the PM core of a wakeup event. * __pm_stay_awake - Notify the PM core of a wakeup event.
* @ws: Wakeup source object associated with the source of the event. * @ws: Wakeup source object associated with the source of the event.
@ -397,10 +412,7 @@ void __pm_stay_awake(struct wakeup_source *ws)
spin_lock_irqsave(&ws->lock, flags); spin_lock_irqsave(&ws->lock, flags);
ws->event_count++; wakeup_source_report_event(ws);
if (!ws->active)
wakeup_source_activate(ws);
del_timer(&ws->timer); del_timer(&ws->timer);
ws->timer_expires = 0; ws->timer_expires = 0;
@ -469,6 +481,7 @@ static void wakeup_source_deactivate(struct wakeup_source *ws)
if (ktime_to_ns(duration) > ktime_to_ns(ws->max_time)) if (ktime_to_ns(duration) > ktime_to_ns(ws->max_time))
ws->max_time = duration; ws->max_time = duration;
ws->last_time = now;
del_timer(&ws->timer); del_timer(&ws->timer);
ws->timer_expires = 0; ws->timer_expires = 0;
@ -541,8 +554,10 @@ static void pm_wakeup_timer_fn(unsigned long data)
spin_lock_irqsave(&ws->lock, flags); spin_lock_irqsave(&ws->lock, flags);
if (ws->active && ws->timer_expires if (ws->active && ws->timer_expires
&& time_after_eq(jiffies, ws->timer_expires)) && time_after_eq(jiffies, ws->timer_expires)) {
wakeup_source_deactivate(ws); wakeup_source_deactivate(ws);
ws->expire_count++;
}
spin_unlock_irqrestore(&ws->lock, flags); spin_unlock_irqrestore(&ws->lock, flags);
} }
@ -569,9 +584,7 @@ void __pm_wakeup_event(struct wakeup_source *ws, unsigned int msec)
spin_lock_irqsave(&ws->lock, flags); spin_lock_irqsave(&ws->lock, flags);
ws->event_count++; wakeup_source_report_event(ws);
if (!ws->active)
wakeup_source_activate(ws);
if (!msec) { if (!msec) {
wakeup_source_deactivate(ws); wakeup_source_deactivate(ws);
@ -613,24 +626,6 @@ void pm_wakeup_event(struct device *dev, unsigned int msec)
} }
EXPORT_SYMBOL_GPL(pm_wakeup_event); EXPORT_SYMBOL_GPL(pm_wakeup_event);
/**
* pm_wakeup_update_hit_counts - Update hit counts of all active wakeup sources.
*/
static void pm_wakeup_update_hit_counts(void)
{
unsigned long flags;
struct wakeup_source *ws;
rcu_read_lock();
list_for_each_entry_rcu(ws, &wakeup_sources, entry) {
spin_lock_irqsave(&ws->lock, flags);
if (ws->active)
ws->hit_count++;
spin_unlock_irqrestore(&ws->lock, flags);
}
rcu_read_unlock();
}
/** /**
* pm_wakeup_pending - Check if power transition in progress should be aborted. * pm_wakeup_pending - Check if power transition in progress should be aborted.
* *
@ -653,8 +648,6 @@ bool pm_wakeup_pending(void)
events_check_enabled = !ret; events_check_enabled = !ret;
} }
spin_unlock_irqrestore(&events_lock, flags); spin_unlock_irqrestore(&events_lock, flags);
if (ret)
pm_wakeup_update_hit_counts();
return ret; return ret;
} }
@ -680,7 +673,6 @@ bool pm_get_wakeup_count(unsigned int *count)
split_counters(&cnt, &inpr); split_counters(&cnt, &inpr);
if (inpr == 0 || signal_pending(current)) if (inpr == 0 || signal_pending(current))
break; break;
pm_wakeup_update_hit_counts();
schedule(); schedule();
} }
@ -713,8 +705,6 @@ bool pm_save_wakeup_count(unsigned int count)
events_check_enabled = true; events_check_enabled = true;
} }
spin_unlock_irq(&events_lock); spin_unlock_irq(&events_lock);
if (!events_check_enabled)
pm_wakeup_update_hit_counts();
return events_check_enabled; return events_check_enabled;
} }
@ -749,9 +739,10 @@ static int print_wakeup_source_stats(struct seq_file *m,
active_time = ktime_set(0, 0); active_time = ktime_set(0, 0);
} }
ret = seq_printf(m, "%-12s\t%lu\t\t%lu\t\t%lu\t\t" ret = seq_printf(m, "%-12s\t%lu\t\t%lu\t\t%lu\t\t%lu\t\t"
"%lld\t\t%lld\t\t%lld\t\t%lld\n", "%lld\t\t%lld\t\t%lld\t\t%lld\n",
ws->name, active_count, ws->event_count, ws->hit_count, ws->name, active_count, ws->event_count,
ws->wakeup_count, ws->expire_count,
ktime_to_ms(active_time), ktime_to_ms(total_time), ktime_to_ms(active_time), ktime_to_ms(total_time),
ktime_to_ms(max_time), ktime_to_ms(ws->last_time)); ktime_to_ms(max_time), ktime_to_ms(ws->last_time));
@ -768,8 +759,9 @@ static int wakeup_sources_stats_show(struct seq_file *m, void *unused)
{ {
struct wakeup_source *ws; struct wakeup_source *ws;
seq_puts(m, "name\t\tactive_count\tevent_count\thit_count\t" seq_puts(m, "name\t\tactive_count\tevent_count\twakeup_count\t"
"active_since\ttotal_time\tmax_time\tlast_change\n"); "expire_count\tactive_since\ttotal_time\tmax_time\t"
"last_change\n");
rcu_read_lock(); rcu_read_lock();
list_for_each_entry_rcu(ws, &wakeup_sources, entry) list_for_each_entry_rcu(ws, &wakeup_sources, entry)

View File

@ -33,12 +33,14 @@
* *
* @total_time: Total time this wakeup source has been active. * @total_time: Total time this wakeup source has been active.
* @max_time: Maximum time this wakeup source has been continuously active. * @max_time: Maximum time this wakeup source has been continuously active.
* @last_time: Monotonic clock when the wakeup source's was activated last time. * @last_time: Monotonic clock when the wakeup source's was touched last time.
* @event_count: Number of signaled wakeup events. * @event_count: Number of signaled wakeup events.
* @active_count: Number of times the wakeup sorce was activated. * @active_count: Number of times the wakeup sorce was activated.
* @relax_count: Number of times the wakeup sorce was deactivated. * @relax_count: Number of times the wakeup sorce was deactivated.
* @hit_count: Number of times the wakeup sorce might abort system suspend. * @expire_count: Number of times the wakeup source's timeout has expired.
* @wakeup_count: Number of times the wakeup source might abort suspend.
* @active: Status of the wakeup source. * @active: Status of the wakeup source.
* @has_timeout: The wakeup source has been activated with a timeout.
*/ */
struct wakeup_source { struct wakeup_source {
const char *name; const char *name;
@ -52,8 +54,9 @@ struct wakeup_source {
unsigned long event_count; unsigned long event_count;
unsigned long active_count; unsigned long active_count;
unsigned long relax_count; unsigned long relax_count;
unsigned long hit_count; unsigned long expire_count;
unsigned int active:1; unsigned long wakeup_count;
bool active:1;
}; };
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP