Lines 37-42
static DEFINE_MUTEX(dpm_list_mtx);
Link Here
|
37 |
|
37 |
|
38 |
int (*platform_enable_wakeup)(struct device *dev, int is_on); |
38 |
int (*platform_enable_wakeup)(struct device *dev, int is_on); |
39 |
|
39 |
|
|
|
40 |
#ifdef CONFIG_PM_VERBOSE |
41 |
unsigned int dpm_active_cnt; |
42 |
unsigned int dpm_fail; |
43 |
|
44 |
static unsigned int dpm_off_cnt; |
45 |
|
46 |
static void debug_inc_active_count(void) |
47 |
{ |
48 |
dpm_active_cnt++; |
49 |
} |
50 |
|
51 |
static void debug_dec_active_count(void) |
52 |
{ |
53 |
dpm_active_cnt--; |
54 |
} |
55 |
|
56 |
static void debug_inc_inactive_count(void) |
57 |
{ |
58 |
dpm_off_cnt++; |
59 |
} |
60 |
|
61 |
static void debug_dec_inactive_count(void) |
62 |
{ |
63 |
dpm_off_cnt--; |
64 |
} |
65 |
|
66 |
static int debug_fake_suspend_failure(void) |
67 |
{ |
68 |
return dpm_active_cnt <= dpm_fail; |
69 |
} |
70 |
|
71 |
static void debug_print_counts(void) |
72 |
{ |
73 |
printk(KERN_DEBUG "PM: Active devices: %d\n", dpm_active_cnt); |
74 |
printk(KERN_DEBUG "PM: Inactive devices: %d\n", dpm_off_cnt); |
75 |
} |
76 |
#else /* !CONFIG_PM_VERBOSE */ |
77 |
static inline void debug_inc_active_count(void) {} |
78 |
static inline void debug_dec_active_count(void) {} |
79 |
static inline void debug_inc_inactive_count(void) {} |
80 |
static inline void debug_dec_inactive_count(void) {} |
81 |
static inline int debug_fake_suspend_failure(void) { return false; } |
82 |
static inline void debug_print_counts(void) {} |
83 |
#endif /* !CONFIG_PM_VERBOSE */ |
40 |
|
84 |
|
41 |
void device_pm_add(struct device *dev) |
85 |
void device_pm_add(struct device *dev) |
42 |
{ |
86 |
{ |
Lines 45-50
void device_pm_add(struct device *dev)
Link Here
|
45 |
kobject_name(&dev->kobj)); |
89 |
kobject_name(&dev->kobj)); |
46 |
mutex_lock(&dpm_list_mtx); |
90 |
mutex_lock(&dpm_list_mtx); |
47 |
list_add_tail(&dev->power.entry, &dpm_active); |
91 |
list_add_tail(&dev->power.entry, &dpm_active); |
|
|
92 |
debug_inc_active_count(); |
48 |
mutex_unlock(&dpm_list_mtx); |
93 |
mutex_unlock(&dpm_list_mtx); |
49 |
} |
94 |
} |
50 |
|
95 |
|
Lines 56-61
void device_pm_remove(struct device *dev
Link Here
|
56 |
mutex_lock(&dpm_list_mtx); |
101 |
mutex_lock(&dpm_list_mtx); |
57 |
dpm_sysfs_remove(dev); |
102 |
dpm_sysfs_remove(dev); |
58 |
list_del_init(&dev->power.entry); |
103 |
list_del_init(&dev->power.entry); |
|
|
104 |
debug_dec_active_count(); |
59 |
mutex_unlock(&dpm_list_mtx); |
105 |
mutex_unlock(&dpm_list_mtx); |
60 |
} |
106 |
} |
61 |
|
107 |
|
Lines 121-138
static int resume_device_early(struct de
Link Here
|
121 |
static void dpm_resume(void) |
167 |
static void dpm_resume(void) |
122 |
{ |
168 |
{ |
123 |
mutex_lock(&dpm_list_mtx); |
169 |
mutex_lock(&dpm_list_mtx); |
|
|
170 |
debug_print_counts(); |
124 |
while(!list_empty(&dpm_off)) { |
171 |
while(!list_empty(&dpm_off)) { |
125 |
struct list_head * entry = dpm_off.next; |
172 |
struct list_head * entry = dpm_off.next; |
126 |
struct device * dev = to_device(entry); |
173 |
struct device * dev = to_device(entry); |
127 |
|
174 |
|
128 |
get_device(dev); |
175 |
get_device(dev); |
129 |
list_move_tail(entry, &dpm_active); |
176 |
list_move_tail(entry, &dpm_active); |
|
|
177 |
debug_dec_inactive_count(); |
178 |
debug_inc_active_count(); |
130 |
|
179 |
|
131 |
mutex_unlock(&dpm_list_mtx); |
180 |
mutex_unlock(&dpm_list_mtx); |
132 |
resume_device(dev); |
181 |
resume_device(dev); |
133 |
mutex_lock(&dpm_list_mtx); |
182 |
mutex_lock(&dpm_list_mtx); |
134 |
put_device(dev); |
183 |
put_device(dev); |
135 |
} |
184 |
} |
|
|
185 |
debug_print_counts(); |
136 |
mutex_unlock(&dpm_list_mtx); |
186 |
mutex_unlock(&dpm_list_mtx); |
137 |
} |
187 |
} |
138 |
|
188 |
|
Lines 173-178
static void dpm_power_up(void)
Link Here
|
173 |
struct device * dev = to_device(entry); |
223 |
struct device * dev = to_device(entry); |
174 |
|
224 |
|
175 |
list_move_tail(entry, &dpm_off); |
225 |
list_move_tail(entry, &dpm_off); |
|
|
226 |
debug_inc_inactive_count(); |
176 |
resume_device_early(dev); |
227 |
resume_device_early(dev); |
177 |
} |
228 |
} |
178 |
} |
229 |
} |
Lines 241-247
static int suspend_device(struct device
Link Here
|
241 |
{ |
292 |
{ |
242 |
int error = 0; |
293 |
int error = 0; |
243 |
|
294 |
|
|
|
295 |
if (debug_fake_suspend_failure()) |
296 |
return -ECANCELED; |
297 |
|
244 |
down(&dev->sem); |
298 |
down(&dev->sem); |
|
|
299 |
|
300 |
pr_debug("PM: Suspending device %s:%s\n", |
301 |
dev->bus ? dev->bus->name : "No Bus", |
302 |
kobject_name(&dev->kobj)); |
303 |
|
245 |
if (dev->power.power_state.event) { |
304 |
if (dev->power.power_state.event) { |
246 |
dev_dbg(dev, "PM: suspend %d-->%d\n", |
305 |
dev_dbg(dev, "PM: suspend %d-->%d\n", |
247 |
dev->power.power_state.event, state.event); |
306 |
dev->power.power_state.event, state.event); |
Lines 312-317
int device_suspend(pm_message_t state)
Link Here
|
312 |
might_sleep(); |
371 |
might_sleep(); |
313 |
mutex_lock(&dpm_mtx); |
372 |
mutex_lock(&dpm_mtx); |
314 |
mutex_lock(&dpm_list_mtx); |
373 |
mutex_lock(&dpm_list_mtx); |
|
|
374 |
debug_print_counts(); |
315 |
while (!list_empty(&dpm_active) && error == 0) { |
375 |
while (!list_empty(&dpm_active) && error == 0) { |
316 |
struct list_head * entry = dpm_active.prev; |
376 |
struct list_head * entry = dpm_active.prev; |
317 |
struct device * dev = to_device(entry); |
377 |
struct device * dev = to_device(entry); |
Lines 326-333
int device_suspend(pm_message_t state)
Link Here
|
326 |
/* Check if the device got removed */ |
386 |
/* Check if the device got removed */ |
327 |
if (!list_empty(&dev->power.entry)) { |
387 |
if (!list_empty(&dev->power.entry)) { |
328 |
/* Move it to the dpm_off list */ |
388 |
/* Move it to the dpm_off list */ |
329 |
if (!error) |
389 |
if (!error) { |
330 |
list_move(&dev->power.entry, &dpm_off); |
390 |
list_move(&dev->power.entry, &dpm_off); |
|
|
391 |
debug_dec_active_count(); |
392 |
debug_inc_inactive_count(); |
393 |
} |
331 |
} |
394 |
} |
332 |
if (error) |
395 |
if (error) |
333 |
printk(KERN_ERR "Could not suspend device %s: " |
396 |
printk(KERN_ERR "Could not suspend device %s: " |
Lines 336-341
int device_suspend(pm_message_t state)
Link Here
|
336 |
error == -EAGAIN ? " (please convert to suspend_late)" : ""); |
399 |
error == -EAGAIN ? " (please convert to suspend_late)" : ""); |
337 |
put_device(dev); |
400 |
put_device(dev); |
338 |
} |
401 |
} |
|
|
402 |
debug_print_counts(); |
339 |
mutex_unlock(&dpm_list_mtx); |
403 |
mutex_unlock(&dpm_list_mtx); |
340 |
if (error) |
404 |
if (error) |
341 |
dpm_resume(); |
405 |
dpm_resume(); |
Lines 368-373
int device_power_down(pm_message_t state
Link Here
|
368 |
if (error) |
432 |
if (error) |
369 |
goto Error; |
433 |
goto Error; |
370 |
list_move(&dev->power.entry, &dpm_off_irq); |
434 |
list_move(&dev->power.entry, &dpm_off_irq); |
|
|
435 |
debug_dec_inactive_count(); |
371 |
} |
436 |
} |
372 |
|
437 |
|
373 |
error = sysdev_suspend(state); |
438 |
error = sysdev_suspend(state); |