disk.c 20.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 * kernel/power/disk.c - Suspend-to-disk support.
 *
 * Copyright (c) 2003 Patrick Mochel
 * Copyright (c) 2003 Open Source Development Lab
 * Copyright (c) 2004 Pavel Machek <pavel@suse.cz>
 *
 * This file is released under the GPLv2.
 *
 */

#include <linux/suspend.h>
#include <linux/syscalls.h>
#include <linux/reboot.h>
#include <linux/string.h>
#include <linux/device.h>
#include <linux/delay.h>
#include <linux/fs.h>
19
#include <linux/mount.h>
20
#include <linux/pm.h>
21
#include <linux/console.h>
22
#include <linux/cpu.h>
23
#include <linux/freezer.h>
24

L
Linus Torvalds 已提交
25 26 27 28
#include "power.h"


static int noresume = 0;
29
static char resume_file[256] = CONFIG_PM_STD_PARTITION;
L
Linus Torvalds 已提交
30
dev_t swsusp_resume_device;
31
sector_t swsusp_resume_block;
L
Linus Torvalds 已提交
32

33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
enum {
	HIBERNATION_INVALID,
	HIBERNATION_PLATFORM,
	HIBERNATION_TEST,
	HIBERNATION_TESTPROC,
	HIBERNATION_SHUTDOWN,
	HIBERNATION_REBOOT,
	/* keep last */
	__HIBERNATION_AFTER_LAST
};
#define HIBERNATION_MAX (__HIBERNATION_AFTER_LAST-1)
#define HIBERNATION_FIRST (HIBERNATION_INVALID + 1)

static int hibernation_mode = HIBERNATION_SHUTDOWN;

48
static struct platform_hibernation_ops *hibernation_ops;
49 50 51 52 53 54

/**
 * hibernation_set_ops - set the global hibernate operations
 * @ops: the hibernation operations to use in subsequent hibernation transitions
 */

55
void hibernation_set_ops(struct platform_hibernation_ops *ops)
56
{
57 58
	if (ops && !(ops->begin && ops->end &&  ops->pre_snapshot
	    && ops->prepare && ops->finish && ops->enter && ops->pre_restore
59
	    && ops->restore_cleanup)) {
60 61 62 63 64 65 66 67 68 69 70 71 72
		WARN_ON(1);
		return;
	}
	mutex_lock(&pm_mutex);
	hibernation_ops = ops;
	if (ops)
		hibernation_mode = HIBERNATION_PLATFORM;
	else if (hibernation_mode == HIBERNATION_PLATFORM)
		hibernation_mode = HIBERNATION_SHUTDOWN;

	mutex_unlock(&pm_mutex);
}

73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
#ifdef CONFIG_PM_DEBUG
static void hibernation_debug_sleep(void)
{
	printk(KERN_INFO "hibernation debug: Waiting for 5 seconds.\n");
	mdelay(5000);
}

static int hibernation_testmode(int mode)
{
	if (hibernation_mode == mode) {
		hibernation_debug_sleep();
		return 1;
	}
	return 0;
}

static int hibernation_test(int level)
{
	if (pm_test_level == level) {
		hibernation_debug_sleep();
		return 1;
	}
	return 0;
}
#else /* !CONFIG_PM_DEBUG */
static int hibernation_testmode(int mode) { return 0; }
static int hibernation_test(int level) { return 0; }
#endif /* !CONFIG_PM_DEBUG */

102
/**
103
 *	platform_begin - tell the platform driver that we're starting
104 105 106
 *	hibernation
 */

107
static int platform_begin(int platform_mode)
108 109
{
	return (platform_mode && hibernation_ops) ?
110 111 112 113 114 115 116 117 118 119 120 121
		hibernation_ops->begin() : 0;
}

/**
 *	platform_end - tell the platform driver that we've entered the
 *	working state
 */

static void platform_end(int platform_mode)
{
	if (platform_mode && hibernation_ops)
		hibernation_ops->end();
122
}
123

124
/**
125
 *	platform_pre_snapshot - prepare the machine for hibernation using the
126 127 128
 *	platform driver if so configured and return an error code if it fails
 */

129
static int platform_pre_snapshot(int platform_mode)
130
{
131
	return (platform_mode && hibernation_ops) ?
132
		hibernation_ops->pre_snapshot() : 0;
133
}
134

135 136 137 138 139 140 141 142 143 144 145
/**
 *	platform_leave - prepare the machine for switching to the normal mode
 *	of operation using the platform driver (called with interrupts disabled)
 */

static void platform_leave(int platform_mode)
{
	if (platform_mode && hibernation_ops)
		hibernation_ops->leave();
}

146 147 148 149 150
/**
 *	platform_finish - switch the machine to the normal mode of operation
 *	using the platform driver (must be called after platform_prepare())
 */

151
static void platform_finish(int platform_mode)
152
{
153
	if (platform_mode && hibernation_ops)
154
		hibernation_ops->finish();
155 156
}

157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181
/**
 *	platform_pre_restore - prepare the platform for the restoration from a
 *	hibernation image.  If the restore fails after this function has been
 *	called, platform_restore_cleanup() must be called.
 */

static int platform_pre_restore(int platform_mode)
{
	return (platform_mode && hibernation_ops) ?
		hibernation_ops->pre_restore() : 0;
}

/**
 *	platform_restore_cleanup - switch the platform to the normal mode of
 *	operation after a failing restore.  If platform_pre_restore() has been
 *	called before the failing restore, this function must be called too,
 *	regardless of the result of platform_pre_restore().
 */

static void platform_restore_cleanup(int platform_mode)
{
	if (platform_mode && hibernation_ops)
		hibernation_ops->restore_cleanup();
}

182 183 184 185 186 187 188 189 190 191 192
/**
 *	platform_recover - recover the platform from a failure to suspend
 *	devices.
 */

static void platform_recover(int platform_mode)
{
	if (platform_mode && hibernation_ops && hibernation_ops->recover)
		hibernation_ops->recover();
}

193 194 195 196 197 198
/**
 *	create_image - freeze devices that need to be frozen with interrupts
 *	off, create the hibernation image and thaw those devices.  Control
 *	reappears in this routine after a restore.
 */

199
static int create_image(int platform_mode)
200 201 202 203 204 205 206
{
	int error;

	error = arch_prepare_suspend();
	if (error)
		return error;

207
	device_pm_lock();
208 209 210 211 212 213 214 215 216
	local_irq_disable();
	/* At this point, device_suspend() has been called, but *not*
	 * device_power_down(). We *must* call device_power_down() now.
	 * Otherwise, drivers for some devices (e.g. interrupt controllers)
	 * become desynchronized with the actual state of the hardware
	 * at resume time, and evil weirdness ensues.
	 */
	error = device_power_down(PMSG_FREEZE);
	if (error) {
R
Rafael J. Wysocki 已提交
217 218
		printk(KERN_ERR "PM: Some devices failed to power down, "
			"aborting hibernation\n");
219 220 221
		goto Enable_irqs;
	}

222 223 224 225
	if (hibernation_test(TEST_CORE))
		goto Power_up;

	in_suspend = 1;
226 227 228
	save_processor_state();
	error = swsusp_arch_suspend();
	if (error)
R
Rafael J. Wysocki 已提交
229 230
		printk(KERN_ERR "PM: Error %d creating hibernation image\n",
			error);
231 232 233 234
	/* Restore control flow magically appears here */
	restore_processor_state();
	if (!in_suspend)
		platform_leave(platform_mode);
235
 Power_up:
236 237 238
	/* NOTE:  device_power_up() is just a resume() for devices
	 * that suspended with irqs off ... no overall powerup.
	 */
239 240
	device_power_up(in_suspend ?
		(error ? PMSG_RECOVER : PMSG_THAW) : PMSG_RESTORE);
241 242
 Enable_irqs:
	local_irq_enable();
243
	device_pm_unlock();
244 245 246
	return error;
}

247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262
/**
 *	hibernation_snapshot - quiesce devices and create the hibernation
 *	snapshot image.
 *	@platform_mode - if set, use the platform driver, if available, to
 *			 prepare the platform frimware for the power transition.
 *
 *	Must be called with pm_mutex held
 */

int hibernation_snapshot(int platform_mode)
{
	int error;

	/* Free memory before shutting down devices. */
	error = swsusp_shrink_memory();
	if (error)
263
		return error;
264

265
	error = platform_begin(platform_mode);
266
	if (error)
267
		goto Close;
268

269 270
	suspend_console();
	error = device_suspend(PMSG_FREEZE);
271
	if (error)
272
		goto Recover_platform;
273

274
	if (hibernation_test(TEST_DEVICES))
275
		goto Recover_platform;
276

277 278 279 280
	error = platform_pre_snapshot(platform_mode);
	if (error || hibernation_test(TEST_PLATFORM))
		goto Finish;

281 282
	error = disable_nonboot_cpus();
	if (!error) {
283 284 285 286 287 288 289 290
		if (hibernation_test(TEST_CPUS))
			goto Enable_cpus;

		if (hibernation_testmode(HIBERNATION_TEST))
			goto Enable_cpus;

		error = create_image(platform_mode);
		/* Control returns here after successful restore */
291
	}
292
 Enable_cpus:
293
	enable_nonboot_cpus();
294
 Finish:
295
	platform_finish(platform_mode);
296
 Resume_devices:
297 298
	device_resume(in_suspend ?
		(error ? PMSG_RECOVER : PMSG_THAW) : PMSG_RESTORE);
299
	resume_console();
300 301
 Close:
	platform_end(platform_mode);
302
	return error;
303 304 305 306

 Recover_platform:
	platform_recover(platform_mode);
	goto Resume_devices;
307 308
}

309 310 311 312 313 314 315 316 317 318 319 320
/**
 *	resume_target_kernel - prepare devices that need to be suspended with
 *	interrupts off, restore the contents of highmem that have not been
 *	restored yet from the image and run the low level code that will restore
 *	the remaining contents of memory and switch to the just restored target
 *	kernel.
 */

static int resume_target_kernel(void)
{
	int error;

321
	device_pm_lock();
322
	local_irq_disable();
323
	error = device_power_down(PMSG_QUIESCE);
324
	if (error) {
R
Rafael J. Wysocki 已提交
325
		printk(KERN_ERR "PM: Some devices failed to power down, "
326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350
			"aborting resume\n");
		goto Enable_irqs;
	}
	/* We'll ignore saved state, but this gets preempt count (etc) right */
	save_processor_state();
	error = restore_highmem();
	if (!error) {
		error = swsusp_arch_resume();
		/*
		 * The code below is only ever reached in case of a failure.
		 * Otherwise execution continues at place where
		 * swsusp_arch_suspend() was called
		 */
		BUG_ON(!error);
		/* This call to restore_highmem() undos the previous one */
		restore_highmem();
	}
	/*
	 * The only reason why swsusp_arch_resume() can fail is memory being
	 * very tight, so we have to free it as soon as we can to avoid
	 * subsequent failures
	 */
	swsusp_free();
	restore_processor_state();
	touch_softlockup_watchdog();
351
	device_power_up(PMSG_RECOVER);
352 353
 Enable_irqs:
	local_irq_enable();
354
	device_pm_unlock();
355 356 357
	return error;
}

358 359 360
/**
 *	hibernation_restore - quiesce devices and restore the hibernation
 *	snapshot image.  If successful, control returns in hibernation_snaphot()
361 362
 *	@platform_mode - if set, use the platform driver, if available, to
 *			 prepare the platform frimware for the transition.
363 364 365 366
 *
 *	Must be called with pm_mutex held
 */

367
int hibernation_restore(int platform_mode)
368 369 370 371 372
{
	int error;

	pm_prepare_console();
	suspend_console();
373
	error = device_suspend(PMSG_QUIESCE);
374 375 376
	if (error)
		goto Finish;

377 378 379 380
	error = platform_pre_restore(platform_mode);
	if (!error) {
		error = disable_nonboot_cpus();
		if (!error)
381
			error = resume_target_kernel();
382 383 384
		enable_nonboot_cpus();
	}
	platform_restore_cleanup(platform_mode);
385
	device_resume(PMSG_RECOVER);
386
 Finish:
387 388 389 390 391 392 393 394 395 396 397 398
	resume_console();
	pm_restore_console();
	return error;
}

/**
 *	hibernation_platform_enter - enter the hibernation state using the
 *	platform driver (if available)
 */

int hibernation_platform_enter(void)
{
399 400
	int error;

401 402 403 404 405 406 407 408
	if (!hibernation_ops)
		return -ENOSYS;

	/*
	 * We have cancelled the power transition by running
	 * hibernation_ops->finish() before saving the image, so we should let
	 * the firmware know that we're going to enter the sleep state after all
	 */
409
	error = hibernation_ops->begin();
410
	if (error)
411
		goto Close;
412 413

	suspend_console();
414
	error = device_suspend(PMSG_HIBERNATE);
415 416 417 418 419
	if (error) {
		if (hibernation_ops->recover)
			hibernation_ops->recover();
		goto Resume_devices;
	}
420 421 422 423 424 425 426 427 428

	error = hibernation_ops->prepare();
	if (error)
		goto Resume_devices;

	error = disable_nonboot_cpus();
	if (error)
		goto Finish;

429
	device_pm_lock();
430
	local_irq_disable();
431
	error = device_power_down(PMSG_HIBERNATE);
432 433 434 435
	if (!error) {
		hibernation_ops->enter();
		/* We should never get here */
		while (1);
436
	}
437
	local_irq_enable();
438
	device_pm_unlock();
439 440 441 442 443 444 445 446

	/*
	 * We don't need to reenable the nonboot CPUs or resume consoles, since
	 * the system is going to be halted anyway.
	 */
 Finish:
	hibernation_ops->finish();
 Resume_devices:
447
	device_resume(PMSG_RESTORE);
448
	resume_console();
449 450
 Close:
	hibernation_ops->end();
451
	return error;
452 453
}

L
Linus Torvalds 已提交
454
/**
455
 *	power_down - Shut the machine down for hibernation.
L
Linus Torvalds 已提交
456
 *
457 458
 *	Use the platform driver, if configured so; otherwise try
 *	to power off or reboot.
L
Linus Torvalds 已提交
459 460
 */

461
static void power_down(void)
L
Linus Torvalds 已提交
462
{
463 464 465
	switch (hibernation_mode) {
	case HIBERNATION_TEST:
	case HIBERNATION_TESTPROC:
466
		break;
467
	case HIBERNATION_REBOOT:
468
		kernel_restart(NULL);
L
Linus Torvalds 已提交
469
		break;
470
	case HIBERNATION_PLATFORM:
471
		hibernation_platform_enter();
472 473 474
	case HIBERNATION_SHUTDOWN:
		kernel_power_off();
		break;
L
Linus Torvalds 已提交
475
	}
476
	kernel_halt();
477 478 479 480
	/*
	 * Valid image is on the disk, if we continue we risk serious data
	 * corruption after resume.
	 */
R
Rafael J. Wysocki 已提交
481
	printk(KERN_CRIT "PM: Please power down manually\n");
L
Linus Torvalds 已提交
482 483 484 485 486
	while(1);
}

static int prepare_processes(void)
{
R
Rafael J. Wysocki 已提交
487
	int error = 0;
L
Linus Torvalds 已提交
488 489 490

	if (freeze_processes()) {
		error = -EBUSY;
491
		thaw_processes();
L
Linus Torvalds 已提交
492
	}
L
Li Shaohua 已提交
493
	return error;
L
Linus Torvalds 已提交
494 495 496
}

/**
497
 *	hibernate - The granpappy of the built-in hibernation management
L
Linus Torvalds 已提交
498 499
 */

500
int hibernate(void)
L
Linus Torvalds 已提交
501 502 503
{
	int error;

504
	mutex_lock(&pm_mutex);
505
	/* The snapshot device should not be opened while we're running */
506 507 508 509 510
	if (!atomic_add_unless(&snapshot_device_available, -1, 0)) {
		error = -EBUSY;
		goto Unlock;
	}

511
	pm_prepare_console();
512 513 514
	error = pm_notifier_call_chain(PM_HIBERNATION_PREPARE);
	if (error)
		goto Exit;
515 516 517 518 519 520

	/* Allocate memory management structures */
	error = create_basic_memory_bitmaps();
	if (error)
		goto Exit;

R
Rafael J. Wysocki 已提交
521
	printk(KERN_INFO "PM: Syncing filesystems ... ");
522 523 524
	sys_sync();
	printk("done.\n");

L
Linus Torvalds 已提交
525
	error = prepare_processes();
L
Li Shaohua 已提交
526
	if (error)
527
		goto Finish;
L
Linus Torvalds 已提交
528

529
	if (hibernation_test(TEST_FREEZER))
530
		goto Thaw;
531 532 533 534

	if (hibernation_testmode(HIBERNATION_TESTPROC))
		goto Thaw;

535 536
	error = hibernation_snapshot(hibernation_mode == HIBERNATION_PLATFORM);
	if (in_suspend && !error) {
537 538 539 540
		unsigned int flags = 0;

		if (hibernation_mode == HIBERNATION_PLATFORM)
			flags |= SF_PLATFORM_MODE;
L
Linus Torvalds 已提交
541
		pr_debug("PM: writing image.\n");
542
		error = swsusp_write(flags);
543
		swsusp_free();
L
Linus Torvalds 已提交
544
		if (!error)
545
			power_down();
R
Rafael J. Wysocki 已提交
546
	} else {
L
Linus Torvalds 已提交
547
		pr_debug("PM: Image restored successfully.\n");
548
		swsusp_free();
R
Rafael J. Wysocki 已提交
549 550
	}
 Thaw:
551
	thaw_processes();
552 553 554
 Finish:
	free_basic_memory_bitmaps();
 Exit:
555
	pm_notifier_call_chain(PM_POST_HIBERNATION);
556
	pm_restore_console();
557
	atomic_inc(&snapshot_device_available);
558 559
 Unlock:
	mutex_unlock(&pm_mutex);
L
Linus Torvalds 已提交
560 561 562 563 564 565 566 567 568 569
	return error;
}


/**
 *	software_resume - Resume from a saved image.
 *
 *	Called as a late_initcall (so all devices are discovered and
 *	initialized), we call swsusp to see if we have a saved image or not.
 *	If so, we quiesce devices, the restore the saved image. We will
570
 *	return above (in hibernate() ) if everything goes well.
L
Linus Torvalds 已提交
571 572 573 574 575 576 577 578
 *	Otherwise, we fail gracefully and return to the normally
 *	scheduled program.
 *
 */

static int software_resume(void)
{
	int error;
579
	unsigned int flags;
L
Linus Torvalds 已提交
580

J
Johannes Berg 已提交
581 582 583 584 585 586 587 588 589 590 591
	/*
	 * name_to_dev_t() below takes a sysfs buffer mutex when sysfs
	 * is configured into the kernel. Since the regular hibernate
	 * trigger path is via sysfs which takes a buffer mutex before
	 * calling hibernate functions (which take pm_mutex) this can
	 * cause lockdep to complain about a possible ABBA deadlock
	 * which cannot happen since we're in the boot code here and
	 * sysfs can't be invoked yet. Therefore, we use a subclass
	 * here to avoid lockdep complaining.
	 */
	mutex_lock_nested(&pm_mutex, SINGLE_DEPTH_NESTING);
592
	if (!swsusp_resume_device) {
593
		if (!strlen(resume_file)) {
594
			mutex_unlock(&pm_mutex);
595
			return -ENOENT;
596
		}
597
		swsusp_resume_device = name_to_dev_t(resume_file);
R
Rafael J. Wysocki 已提交
598
		pr_debug("PM: Resume from partition %s\n", resume_file);
599
	} else {
R
Rafael J. Wysocki 已提交
600 601 602
		pr_debug("PM: Resume from partition %d:%d\n",
				MAJOR(swsusp_resume_device),
				MINOR(swsusp_resume_device));
603 604
	}

L
Linus Torvalds 已提交
605 606
	if (noresume) {
		/**
607 608
		 * FIXME: If noresume is specified, we need to find the
		 * partition and reset it back to normal swap space.
L
Linus Torvalds 已提交
609
		 */
610
		mutex_unlock(&pm_mutex);
L
Linus Torvalds 已提交
611 612 613
		return 0;
	}

R
Rafael J. Wysocki 已提交
614
	pr_debug("PM: Checking hibernation image.\n");
615 616
	error = swsusp_check();
	if (error)
617
		goto Unlock;
L
Linus Torvalds 已提交
618

619 620 621 622 623 624
	/* The snapshot device should not be opened while we're running */
	if (!atomic_add_unless(&snapshot_device_available, -1, 0)) {
		error = -EBUSY;
		goto Unlock;
	}

625
	pm_prepare_console();
626 627 628 629
	error = pm_notifier_call_chain(PM_RESTORE_PREPARE);
	if (error)
		goto Finish;

630 631
	error = create_basic_memory_bitmaps();
	if (error)
632
		goto Finish;
L
Linus Torvalds 已提交
633

634
	pr_debug("PM: Preparing processes for restore.\n");
635 636
	error = prepare_processes();
	if (error) {
L
Linus Torvalds 已提交
637
		swsusp_close();
L
Li Shaohua 已提交
638
		goto Done;
L
Linus Torvalds 已提交
639 640
	}

R
Rafael J. Wysocki 已提交
641
	pr_debug("PM: Reading hibernation image.\n");
L
Linus Torvalds 已提交
642

643
	error = swsusp_read(&flags);
644
	if (!error)
645
		hibernation_restore(flags & SF_PLATFORM_MODE);
L
Linus Torvalds 已提交
646

647
	printk(KERN_ERR "PM: Restore failed, recovering.\n");
648
	swsusp_free();
649
	thaw_processes();
L
Linus Torvalds 已提交
650
 Done:
651
	free_basic_memory_bitmaps();
652
 Finish:
653
	pm_notifier_call_chain(PM_POST_RESTORE);
654
	pm_restore_console();
655
	atomic_inc(&snapshot_device_available);
656
	/* For success case, the suspend path will release the lock */
657
 Unlock:
658
	mutex_unlock(&pm_mutex);
L
Linus Torvalds 已提交
659
	pr_debug("PM: Resume from disk failed.\n");
660
	return error;
L
Linus Torvalds 已提交
661 662 663 664 665
}

late_initcall(software_resume);


666 667 668 669 670 671
static const char * const hibernation_modes[] = {
	[HIBERNATION_PLATFORM]	= "platform",
	[HIBERNATION_SHUTDOWN]	= "shutdown",
	[HIBERNATION_REBOOT]	= "reboot",
	[HIBERNATION_TEST]	= "test",
	[HIBERNATION_TESTPROC]	= "testproc",
L
Linus Torvalds 已提交
672 673 674
};

/**
675
 *	disk - Control hibernation mode
L
Linus Torvalds 已提交
676
 *
677 678
 *	Suspend-to-disk can be handled in several ways. We have a few options
 *	for putting the system to sleep - using the platform driver (e.g. ACPI
679 680
 *	or other hibernation_ops), powering off the system or rebooting the
 *	system (for testing) as well as the two test modes.
L
Linus Torvalds 已提交
681
 *
682
 *	The system can support 'platform', and that is known a priori (and
683 684 685
 *	encoded by the presence of hibernation_ops). However, the user may
 *	choose 'shutdown' or 'reboot' as alternatives, as well as one fo the
 *	test modes, 'test' or 'testproc'.
L
Linus Torvalds 已提交
686 687 688 689 690 691 692
 *
 *	show() will display what the mode is currently set to.
 *	store() will accept one of
 *
 *	'platform'
 *	'shutdown'
 *	'reboot'
693 694
 *	'test'
 *	'testproc'
L
Linus Torvalds 已提交
695
 *
696
 *	It will only change to 'platform' if the system
697
 *	supports it (as determined by having hibernation_ops).
L
Linus Torvalds 已提交
698 699
 */

700 701
static ssize_t disk_show(struct kobject *kobj, struct kobj_attribute *attr,
			 char *buf)
L
Linus Torvalds 已提交
702
{
703 704 705
	int i;
	char *start = buf;

706 707
	for (i = HIBERNATION_FIRST; i <= HIBERNATION_MAX; i++) {
		if (!hibernation_modes[i])
708 709
			continue;
		switch (i) {
710 711 712 713
		case HIBERNATION_SHUTDOWN:
		case HIBERNATION_REBOOT:
		case HIBERNATION_TEST:
		case HIBERNATION_TESTPROC:
714
			break;
715 716
		case HIBERNATION_PLATFORM:
			if (hibernation_ops)
717 718 719 720
				break;
			/* not a valid mode, continue with loop */
			continue;
		}
721 722
		if (i == hibernation_mode)
			buf += sprintf(buf, "[%s] ", hibernation_modes[i]);
723
		else
724
			buf += sprintf(buf, "%s ", hibernation_modes[i]);
725 726 727
	}
	buf += sprintf(buf, "\n");
	return buf-start;
L
Linus Torvalds 已提交
728 729 730
}


731 732
static ssize_t disk_store(struct kobject *kobj, struct kobj_attribute *attr,
			  const char *buf, size_t n)
L
Linus Torvalds 已提交
733 734 735 736 737
{
	int error = 0;
	int i;
	int len;
	char *p;
738
	int mode = HIBERNATION_INVALID;
L
Linus Torvalds 已提交
739 740 741 742

	p = memchr(buf, '\n', n);
	len = p ? p - buf : n;

743
	mutex_lock(&pm_mutex);
744
	for (i = HIBERNATION_FIRST; i <= HIBERNATION_MAX; i++) {
R
Rafael J. Wysocki 已提交
745 746
		if (len == strlen(hibernation_modes[i])
		    && !strncmp(buf, hibernation_modes[i], len)) {
L
Linus Torvalds 已提交
747 748 749 750
			mode = i;
			break;
		}
	}
751
	if (mode != HIBERNATION_INVALID) {
752
		switch (mode) {
753 754 755 756 757
		case HIBERNATION_SHUTDOWN:
		case HIBERNATION_REBOOT:
		case HIBERNATION_TEST:
		case HIBERNATION_TESTPROC:
			hibernation_mode = mode;
758
			break;
759 760 761
		case HIBERNATION_PLATFORM:
			if (hibernation_ops)
				hibernation_mode = mode;
L
Linus Torvalds 已提交
762 763 764
			else
				error = -EINVAL;
		}
765
	} else
L
Linus Torvalds 已提交
766 767
		error = -EINVAL;

768
	if (!error)
R
Rafael J. Wysocki 已提交
769
		pr_debug("PM: Hibernation mode set to '%s'\n",
770
			 hibernation_modes[mode]);
771
	mutex_unlock(&pm_mutex);
L
Linus Torvalds 已提交
772 773 774 775 776
	return error ? error : n;
}

power_attr(disk);

777 778
static ssize_t resume_show(struct kobject *kobj, struct kobj_attribute *attr,
			   char *buf)
L
Linus Torvalds 已提交
779 780 781 782 783
{
	return sprintf(buf,"%d:%d\n", MAJOR(swsusp_resume_device),
		       MINOR(swsusp_resume_device));
}

784 785
static ssize_t resume_store(struct kobject *kobj, struct kobj_attribute *attr,
			    const char *buf, size_t n)
L
Linus Torvalds 已提交
786 787 788
{
	unsigned int maj, min;
	dev_t res;
789
	int ret = -EINVAL;
L
Linus Torvalds 已提交
790

791 792
	if (sscanf(buf, "%u:%u", &maj, &min) != 2)
		goto out;
L
Linus Torvalds 已提交
793

794 795 796
	res = MKDEV(maj,min);
	if (maj != MAJOR(res) || min != MINOR(res))
		goto out;
L
Linus Torvalds 已提交
797

798
	mutex_lock(&pm_mutex);
799
	swsusp_resume_device = res;
800
	mutex_unlock(&pm_mutex);
R
Rafael J. Wysocki 已提交
801
	printk(KERN_INFO "PM: Starting manual resume from disk\n");
802 803 804
	noresume = 0;
	software_resume();
	ret = n;
R
Rafael J. Wysocki 已提交
805
 out:
806
	return ret;
L
Linus Torvalds 已提交
807 808 809 810
}

power_attr(resume);

811 812
static ssize_t image_size_show(struct kobject *kobj, struct kobj_attribute *attr,
			       char *buf)
813
{
814
	return sprintf(buf, "%lu\n", image_size);
815 816
}

817 818
static ssize_t image_size_store(struct kobject *kobj, struct kobj_attribute *attr,
				const char *buf, size_t n)
819
{
820
	unsigned long size;
821

822
	if (sscanf(buf, "%lu", &size) == 1) {
823 824 825 826 827 828 829 830 831
		image_size = size;
		return n;
	}

	return -EINVAL;
}

power_attr(image_size);

L
Linus Torvalds 已提交
832 833 834
static struct attribute * g[] = {
	&disk_attr.attr,
	&resume_attr.attr,
835
	&image_size_attr.attr,
L
Linus Torvalds 已提交
836 837 838 839 840 841 842 843 844 845 846
	NULL,
};


static struct attribute_group attr_group = {
	.attrs = g,
};


static int __init pm_disk_init(void)
{
847
	return sysfs_create_group(power_kobj, &attr_group);
L
Linus Torvalds 已提交
848 849 850 851 852 853 854 855 856 857 858 859 860 861
}

core_initcall(pm_disk_init);


static int __init resume_setup(char *str)
{
	if (noresume)
		return 1;

	strncpy( resume_file, str, 255 );
	return 1;
}

862 863 864 865 866 867 868 869 870 871 872 873 874
static int __init resume_offset_setup(char *str)
{
	unsigned long long offset;

	if (noresume)
		return 1;

	if (sscanf(str, "%llu", &offset) == 1)
		swsusp_resume_block = offset;

	return 1;
}

L
Linus Torvalds 已提交
875 876 877 878 879 880 881
static int __init noresume_setup(char *str)
{
	noresume = 1;
	return 1;
}

__setup("noresume", noresume_setup);
882
__setup("resume_offset=", resume_offset_setup);
L
Linus Torvalds 已提交
883
__setup("resume=", resume_setup);