Commit 98e1a74e authored by Todd Poynor's avatar Todd Poynor Committed by Greg Kroah-Hartman

[PATCH] Device runtime suspend/resume fixes

(1) Set device power state at runtime resume (as is done for runtime
suspend) so that a later suspend does not think the device is still
suspended (refusing to suspend it again).

(2) Move devices from the active list to the off list only when
suspending all devices as part of a system suspend, not for runtime
suspend.  This matches the resume code, which only moves devices from
off to active during system resume, such that runtime resume currently
doesn't move the suspended device back to the active list.  (This also
avoids reordering the device list for runtime suspends; the list is in
order of registration and suspend/resume works best that way -- granted,
more sweeping improvements in how device dependencies are accounted for
in the suspend/resume order are also needed someday.)

Runtime device suspend/resume is in some cases used frequently on
battery-powered embedded devices, to save additional power and to handle
device power state interactions with overall system power state on
certain platforms.
Signed-off-by: default avatarGreg Kroah-Hartman <greg@kroah.com>
parent 8714c381
...@@ -14,7 +14,10 @@ static void runtime_resume(struct device * dev) ...@@ -14,7 +14,10 @@ static void runtime_resume(struct device * dev)
{ {
if (!dev->power.power_state) if (!dev->power.power_state)
return; return;
resume_device(dev); if (! resume_device(dev))
dev->power.power_state = 0;
return;
} }
......
...@@ -42,13 +42,6 @@ int suspend_device(struct device * dev, u32 state) ...@@ -42,13 +42,6 @@ int suspend_device(struct device * dev, u32 state)
if (dev->bus && dev->bus->suspend) if (dev->bus && dev->bus->suspend)
error = dev->bus->suspend(dev,state); error = dev->bus->suspend(dev,state);
if (!error) {
list_del(&dev->power.entry);
list_add(&dev->power.entry,&dpm_off);
} else if (error == -EAGAIN) {
list_del(&dev->power.entry);
list_add(&dev->power.entry,&dpm_off_irq);
}
return error; return error;
} }
...@@ -81,12 +74,16 @@ int device_suspend(u32 state) ...@@ -81,12 +74,16 @@ int device_suspend(u32 state)
while(!list_empty(&dpm_active)) { while(!list_empty(&dpm_active)) {
struct list_head * entry = dpm_active.prev; struct list_head * entry = dpm_active.prev;
struct device * dev = to_device(entry); struct device * dev = to_device(entry);
if ((error = suspend_device(dev,state))) { error = suspend_device(dev,state);
if (error != -EAGAIN)
if (!error) {
list_del(&dev->power.entry);
list_add(&dev->power.entry,&dpm_off);
} else if (error == -EAGAIN) {
list_del(&dev->power.entry);
list_add(&dev->power.entry,&dpm_off_irq);
} else
goto Error; goto Error;
else
error = 0;
}
} }
Done: Done:
up(&dpm_sem); up(&dpm_sem);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment