diff options
author | Rafael J. Wysocki <rjw@sisk.pl> | 2009-03-16 17:34:06 -0400 |
---|---|---|
committer | Rafael J. Wysocki <rjw@sisk.pl> | 2009-03-30 15:46:54 -0400 |
commit | 2ed8d2b3a81bdbb0418301628ccdb008ac9f40b7 (patch) | |
tree | 54ab0cd7aa7db73151533b463bd490b62a29c462 /arch/x86/kernel/apm_32.c | |
parent | 0a0c5168df270a50e3518e4f12bddb31f8f5f38f (diff) |
PM: Rework handling of interrupts during suspend-resume
Use the functions introduced in by the previous patch,
suspend_device_irqs(), resume_device_irqs() and check_wakeup_irqs(),
to rework the handling of interrupts during suspend (hibernation) and
resume. Namely, interrupts will only be disabled on the CPU right
before suspending sysdevs, while device drivers will be prevented
from receiving interrupts, with the help of the new helper function,
before their "late" suspend callbacks run (and analogously during
resume).
In addition, since the device interrups are now disabled before the
CPU has turned all interrupts off and the CPU will ACK the interrupts
setting the IRQ_PENDING bit for them, check in sysdev_suspend() if
any wake-up interrupts are pending and abort suspend if that's the
case.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Acked-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch/x86/kernel/apm_32.c')
-rw-r--r-- | arch/x86/kernel/apm_32.c | 15 |
1 files changed, 11 insertions, 4 deletions
diff --git a/arch/x86/kernel/apm_32.c b/arch/x86/kernel/apm_32.c index 10033fe718e0..ac7783a67432 100644 --- a/arch/x86/kernel/apm_32.c +++ b/arch/x86/kernel/apm_32.c | |||
@@ -1190,8 +1190,10 @@ static int suspend(int vetoable) | |||
1190 | struct apm_user *as; | 1190 | struct apm_user *as; |
1191 | 1191 | ||
1192 | device_suspend(PMSG_SUSPEND); | 1192 | device_suspend(PMSG_SUSPEND); |
1193 | local_irq_disable(); | 1193 | |
1194 | device_power_down(PMSG_SUSPEND); | 1194 | device_power_down(PMSG_SUSPEND); |
1195 | |||
1196 | local_irq_disable(); | ||
1195 | sysdev_suspend(PMSG_SUSPEND); | 1197 | sysdev_suspend(PMSG_SUSPEND); |
1196 | 1198 | ||
1197 | local_irq_enable(); | 1199 | local_irq_enable(); |
@@ -1209,9 +1211,12 @@ static int suspend(int vetoable) | |||
1209 | if (err != APM_SUCCESS) | 1211 | if (err != APM_SUCCESS) |
1210 | apm_error("suspend", err); | 1212 | apm_error("suspend", err); |
1211 | err = (err == APM_SUCCESS) ? 0 : -EIO; | 1213 | err = (err == APM_SUCCESS) ? 0 : -EIO; |
1214 | |||
1212 | sysdev_resume(); | 1215 | sysdev_resume(); |
1213 | device_power_up(PMSG_RESUME); | ||
1214 | local_irq_enable(); | 1216 | local_irq_enable(); |
1217 | |||
1218 | device_power_up(PMSG_RESUME); | ||
1219 | |||
1215 | device_resume(PMSG_RESUME); | 1220 | device_resume(PMSG_RESUME); |
1216 | queue_event(APM_NORMAL_RESUME, NULL); | 1221 | queue_event(APM_NORMAL_RESUME, NULL); |
1217 | spin_lock(&user_list_lock); | 1222 | spin_lock(&user_list_lock); |
@@ -1228,8 +1233,9 @@ static void standby(void) | |||
1228 | { | 1233 | { |
1229 | int err; | 1234 | int err; |
1230 | 1235 | ||
1231 | local_irq_disable(); | ||
1232 | device_power_down(PMSG_SUSPEND); | 1236 | device_power_down(PMSG_SUSPEND); |
1237 | |||
1238 | local_irq_disable(); | ||
1233 | sysdev_suspend(PMSG_SUSPEND); | 1239 | sysdev_suspend(PMSG_SUSPEND); |
1234 | local_irq_enable(); | 1240 | local_irq_enable(); |
1235 | 1241 | ||
@@ -1239,8 +1245,9 @@ static void standby(void) | |||
1239 | 1245 | ||
1240 | local_irq_disable(); | 1246 | local_irq_disable(); |
1241 | sysdev_resume(); | 1247 | sysdev_resume(); |
1242 | device_power_up(PMSG_RESUME); | ||
1243 | local_irq_enable(); | 1248 | local_irq_enable(); |
1249 | |||
1250 | device_power_up(PMSG_RESUME); | ||
1244 | } | 1251 | } |
1245 | 1252 | ||
1246 | static apm_event_t get_event(void) | 1253 | static apm_event_t get_event(void) |