Commit c7e0831d385d620a58d95b25e4afa9b643f9a411

Authored by Rafael J. Wysocki
Committed by Linus Torvalds
1 parent efa4d2fb04

Hibernation: Check if ACPI is enabled during restore in the right place

The following scenario leads to total confusion of the platform firmware on
some boxes (eg. HPC nx6325):
* Hibernate with ACPI enabled
* Resume passing "acpi=off" to the boot kernel

To prevent this from happening it's necessary to check if ACPI is enabled (and
enable it if that's not the case) _right_ _after_ control has been transfered
from the boot kernel to the image kernel, before device_power_up() is called
(ie.  with interrupts disabled).   Enabling ACPI after calling
device_power_up() turns out to be insufficient.

For this reason, introduce new hibernation callback ->leave() that will be
executed before device_power_up() by the restored image kernel.   To make it
work, it also is necessary to move swsusp_suspend() from swsusp.c to disk.c
(it's name is changed to "create_image", which is more up to the point).

Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Acked-by: Pavel Machek <pavel@ucw.cz>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>

Showing 5 changed files with 74 additions and 35 deletions Side-by-side Diff

drivers/acpi/sleep/main.c
... ... @@ -257,6 +257,15 @@
257 257 return ACPI_SUCCESS(status) ? 0 : -EFAULT;
258 258 }
259 259  
  260 +static void acpi_hibernation_leave(void)
  261 +{
  262 + /*
  263 + * If ACPI is not enabled by the BIOS and the boot kernel, we need to
  264 + * enable it here.
  265 + */
  266 + acpi_enable();
  267 +}
  268 +
260 269 static void acpi_hibernation_finish(void)
261 270 {
262 271 acpi_leave_sleep_state(ACPI_STATE_S4);
... ... @@ -288,6 +297,7 @@
288 297 .finish = acpi_hibernation_finish,
289 298 .prepare = acpi_hibernation_prepare,
290 299 .enter = acpi_hibernation_enter,
  300 + .leave = acpi_hibernation_leave,
291 301 .pre_restore = acpi_hibernation_pre_restore,
292 302 .restore_cleanup = acpi_hibernation_restore_cleanup,
293 303 };
include/linux/suspend.h
... ... @@ -156,6 +156,12 @@
156 156 * Called after the nonboot CPUs have been disabled and all of the low
157 157 * level devices have been shut down (runs with IRQs off).
158 158 *
  159 + * @leave: Perform the first stage of the cleanup after the system sleep state
  160 + * indicated by @set_target() has been left.
  161 + * Called right after the control has been passed from the boot kernel to
  162 + * the image kernel, before the nonboot CPUs are enabled and before devices
  163 + * are resumed. Executed with interrupts disabled.
  164 + *
159 165 * @pre_restore: Prepare system for the restoration from a hibernation image.
160 166 * Called right after devices have been frozen and before the nonboot
161 167 * CPUs are disabled (runs with IRQs on).
... ... @@ -170,6 +176,7 @@
170 176 void (*finish)(void);
171 177 int (*prepare)(void);
172 178 int (*enter)(void);
  179 + void (*leave)(void);
173 180 int (*pre_restore)(void);
174 181 void (*restore_cleanup)(void);
175 182 };
... ... @@ -93,6 +93,17 @@
93 93 }
94 94  
95 95 /**
  96 + * platform_leave - prepare the machine for switching to the normal mode
  97 + * of operation using the platform driver (called with interrupts disabled)
  98 + */
  99 +
  100 +static void platform_leave(int platform_mode)
  101 +{
  102 + if (platform_mode && hibernation_ops)
  103 + hibernation_ops->leave();
  104 +}
  105 +
  106 +/**
96 107 * platform_finish - switch the machine to the normal mode of operation
97 108 * using the platform driver (must be called after platform_prepare())
98 109 */
... ... @@ -129,6 +140,51 @@
129 140 }
130 141  
131 142 /**
  143 + * create_image - freeze devices that need to be frozen with interrupts
  144 + * off, create the hibernation image and thaw those devices. Control
  145 + * reappears in this routine after a restore.
  146 + */
  147 +
  148 +int create_image(int platform_mode)
  149 +{
  150 + int error;
  151 +
  152 + error = arch_prepare_suspend();
  153 + if (error)
  154 + return error;
  155 +
  156 + local_irq_disable();
  157 + /* At this point, device_suspend() has been called, but *not*
  158 + * device_power_down(). We *must* call device_power_down() now.
  159 + * Otherwise, drivers for some devices (e.g. interrupt controllers)
  160 + * become desynchronized with the actual state of the hardware
  161 + * at resume time, and evil weirdness ensues.
  162 + */
  163 + error = device_power_down(PMSG_FREEZE);
  164 + if (error) {
  165 + printk(KERN_ERR "Some devices failed to power down, "
  166 + KERN_ERR "aborting suspend\n");
  167 + goto Enable_irqs;
  168 + }
  169 +
  170 + save_processor_state();
  171 + error = swsusp_arch_suspend();
  172 + if (error)
  173 + printk(KERN_ERR "Error %d while creating the image\n", error);
  174 + /* Restore control flow magically appears here */
  175 + restore_processor_state();
  176 + if (!in_suspend)
  177 + platform_leave(platform_mode);
  178 + /* NOTE: device_power_up() is just a resume() for devices
  179 + * that suspended with irqs off ... no overall powerup.
  180 + */
  181 + device_power_up();
  182 + Enable_irqs:
  183 + local_irq_enable();
  184 + return error;
  185 +}
  186 +
  187 +/**
132 188 * hibernation_snapshot - quiesce devices and create the hibernation
133 189 * snapshot image.
134 190 * @platform_mode - if set, use the platform driver, if available, to
... ... @@ -163,7 +219,7 @@
163 219 if (!error) {
164 220 if (hibernation_mode != HIBERNATION_TEST) {
165 221 in_suspend = 1;
166   - error = swsusp_suspend();
  222 + error = create_image(platform_mode);
167 223 /* Control returns here after successful restore */
168 224 } else {
169 225 printk("swsusp debug: Waiting for 5 seconds.\n");
kernel/power/power.h
... ... @@ -183,7 +183,6 @@
183 183 extern int swsusp_check(void);
184 184 extern int swsusp_shrink_memory(void);
185 185 extern void swsusp_free(void);
186   -extern int swsusp_suspend(void);
187 186 extern int swsusp_resume(void);
188 187 extern int swsusp_read(unsigned int *flags_p);
189 188 extern int swsusp_write(unsigned int flags);
kernel/power/swsusp.c
... ... @@ -270,39 +270,6 @@
270 270 return 0;
271 271 }
272 272  
273   -int swsusp_suspend(void)
274   -{
275   - int error;
276   -
277   - if ((error = arch_prepare_suspend()))
278   - return error;
279   -
280   - local_irq_disable();
281   - /* At this point, device_suspend() has been called, but *not*
282   - * device_power_down(). We *must* device_power_down() now.
283   - * Otherwise, drivers for some devices (e.g. interrupt controllers)
284   - * become desynchronized with the actual state of the hardware
285   - * at resume time, and evil weirdness ensues.
286   - */
287   - if ((error = device_power_down(PMSG_FREEZE))) {
288   - printk(KERN_ERR "Some devices failed to power down, aborting suspend\n");
289   - goto Enable_irqs;
290   - }
291   -
292   - save_processor_state();
293   - if ((error = swsusp_arch_suspend()))
294   - printk(KERN_ERR "Error %d suspending\n", error);
295   - /* Restore control flow magically appears here */
296   - restore_processor_state();
297   - /* NOTE: device_power_up() is just a resume() for devices
298   - * that suspended with irqs off ... no overall powerup.
299   - */
300   - device_power_up();
301   - Enable_irqs:
302   - local_irq_enable();
303   - return error;
304   -}
305   -
306 273 int swsusp_resume(void)
307 274 {
308 275 int error;