Skip to content

Commit

Permalink
PM: Fix error code paths executed after failing syscore_suspend()
Browse files Browse the repository at this point in the history
If syscore_suspend() fails in suspend_enter(), create_image() or
resume_target_kernel(), it is necessary to call sysdev_resume(),
because sysdev_suspend() has been called already and succeeded
and we are going to abort the transition.

Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Acked-by: Greg Kroah-Hartman <gregkh@suse.de>
  • Loading branch information
Rafael J. Wysocki committed Apr 18, 2011
1 parent a1b49cb commit 2ca6f62
Show file tree
Hide file tree
Showing 2 changed files with 12 additions and 3 deletions.
10 changes: 8 additions & 2 deletions kernel/power/hibernate.c
Original file line number Diff line number Diff line change
Expand Up @@ -273,8 +273,11 @@ static int create_image(int platform_mode)
local_irq_disable();

error = sysdev_suspend(PMSG_FREEZE);
if (!error)
if (!error) {
error = syscore_suspend();
if (error)
sysdev_resume();
}
if (error) {
printk(KERN_ERR "PM: Some system devices failed to power down, "
"aborting hibernation\n");
Expand Down Expand Up @@ -407,8 +410,11 @@ static int resume_target_kernel(bool platform_mode)
local_irq_disable();

error = sysdev_suspend(PMSG_QUIESCE);
if (!error)
if (!error) {
error = syscore_suspend();
if (error)
sysdev_resume();
}
if (error)
goto Enable_irqs;

Expand Down
5 changes: 4 additions & 1 deletion kernel/power/suspend.c
Original file line number Diff line number Diff line change
Expand Up @@ -164,8 +164,11 @@ static int suspend_enter(suspend_state_t state)
BUG_ON(!irqs_disabled());

error = sysdev_suspend(PMSG_SUSPEND);
if (!error)
if (!error) {
error = syscore_suspend();
if (error)
sysdev_resume();
}
if (!error) {
if (!(suspend_test(TEST_CORE) || pm_wakeup_pending())) {
error = suspend_ops->enter(state);
Expand Down

0 comments on commit 2ca6f62

Please sign in to comment.