swap.c 16.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/*
 * linux/kernel/power/swap.c
 *
 * This file provides functions for reading the suspend image from
 * and writing it to a swap partition.
 *
 * Copyright (C) 1998,2001-2005 Pavel Machek <pavel@suse.cz>
 * Copyright (C) 2006 Rafael J. Wysocki <rjw@sisk.pl>
 *
 * This file is released under the GPLv2.
 *
 */

#include <linux/module.h>
#include <linux/file.h>
#include <linux/delay.h>
#include <linux/bitops.h>
#include <linux/genhd.h>
#include <linux/device.h>
#include <linux/buffer_head.h>
#include <linux/bio.h>
A
Andrew Morton 已提交
22
#include <linux/blkdev.h>
23 24 25
#include <linux/swap.h>
#include <linux/swapops.h>
#include <linux/pm.h>
26
#include <linux/slab.h>
27 28 29 30 31

#include "power.h"

#define SWSUSP_SIG	"S1SUSPEND"

32
struct swsusp_header {
33
	char reserved[PAGE_SIZE - 20 - sizeof(sector_t) - sizeof(int)];
34
	sector_t image;
35
	unsigned int flags;	/* Flags to pass to the "boot" kernel */
36 37
	char	orig_sig[10];
	char	sig[10];
38 39 40
} __attribute__((packed));

static struct swsusp_header *swsusp_header;
41

42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 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 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
/**
 *	The following functions are used for tracing the allocated
 *	swap pages, so that they can be freed in case of an error.
 */

struct swsusp_extent {
	struct rb_node node;
	unsigned long start;
	unsigned long end;
};

static struct rb_root swsusp_extents = RB_ROOT;

static int swsusp_extents_insert(unsigned long swap_offset)
{
	struct rb_node **new = &(swsusp_extents.rb_node);
	struct rb_node *parent = NULL;
	struct swsusp_extent *ext;

	/* Figure out where to put the new node */
	while (*new) {
		ext = container_of(*new, struct swsusp_extent, node);
		parent = *new;
		if (swap_offset < ext->start) {
			/* Try to merge */
			if (swap_offset == ext->start - 1) {
				ext->start--;
				return 0;
			}
			new = &((*new)->rb_left);
		} else if (swap_offset > ext->end) {
			/* Try to merge */
			if (swap_offset == ext->end + 1) {
				ext->end++;
				return 0;
			}
			new = &((*new)->rb_right);
		} else {
			/* It already is in the tree */
			return -EINVAL;
		}
	}
	/* Add the new node and rebalance the tree. */
	ext = kzalloc(sizeof(struct swsusp_extent), GFP_KERNEL);
	if (!ext)
		return -ENOMEM;

	ext->start = swap_offset;
	ext->end = swap_offset;
	rb_link_node(&ext->node, parent, new);
	rb_insert_color(&ext->node, &swsusp_extents);
	return 0;
}

/**
 *	alloc_swapdev_block - allocate a swap page and register that it has
 *	been allocated, so that it can be freed in case of an error.
 */

sector_t alloc_swapdev_block(int swap)
{
	unsigned long offset;

	offset = swp_offset(get_swap_page_of_type(swap));
	if (offset) {
		if (swsusp_extents_insert(offset))
			swap_free(swp_entry(swap, offset));
		else
			return swapdev_block(swap, offset);
	}
	return 0;
}

/**
 *	free_all_swap_pages - free swap pages allocated for saving image data.
 *	It also frees the extents used to register which swap entres had been
 *	allocated.
 */

void free_all_swap_pages(int swap)
{
	struct rb_node *node;

	while ((node = swsusp_extents.rb_node)) {
		struct swsusp_extent *ext;
		unsigned long offset;

		ext = container_of(node, struct swsusp_extent, node);
		rb_erase(node, &swsusp_extents);
		for (offset = ext->start; offset <= ext->end; offset++)
			swap_free(swp_entry(swap, offset));

		kfree(ext);
	}
}

int swsusp_swap_in_use(void)
{
	return (swsusp_extents.rb_node != NULL);
}

143
/*
144
 * General things
145 146 147
 */

static unsigned short root_swap = 0xffff;
148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
static struct block_device *resume_bdev;

/**
 *	submit - submit BIO request.
 *	@rw:	READ or WRITE.
 *	@off	physical offset of page.
 *	@page:	page we're reading or writing.
 *	@bio_chain: list of pending biod (for async reading)
 *
 *	Straight from the textbook - allocate and initialize the bio.
 *	If we're reading, make sure the page is marked as dirty.
 *	Then submit it and, if @bio_chain == NULL, wait.
 */
static int submit(int rw, pgoff_t page_off, struct page *page,
			struct bio **bio_chain)
{
164
	const int bio_rw = rw | (1 << BIO_RW_SYNCIO) | (1 << BIO_RW_UNPLUG);
165 166
	struct bio *bio;

R
Rafael J. Wysocki 已提交
167
	bio = bio_alloc(__GFP_WAIT | __GFP_HIGH, 1);
168 169 170 171 172
	bio->bi_sector = page_off * (PAGE_SIZE >> 9);
	bio->bi_bdev = resume_bdev;
	bio->bi_end_io = end_swap_bio_read;

	if (bio_add_page(bio, page, PAGE_SIZE, 0) < PAGE_SIZE) {
R
Rafael J. Wysocki 已提交
173 174
		printk(KERN_ERR "PM: Adding page to bio failed at %ld\n",
			page_off);
175 176 177 178 179 180 181 182
		bio_put(bio);
		return -EFAULT;
	}

	lock_page(page);
	bio_get(bio);

	if (bio_chain == NULL) {
183
		submit_bio(bio_rw, bio);
184 185 186 187 188 189 190 191 192
		wait_on_page_locked(page);
		if (rw == READ)
			bio_set_pages_dirty(bio);
		bio_put(bio);
	} else {
		if (rw == READ)
			get_page(page);	/* These pages are freed later */
		bio->bi_private = *bio_chain;
		*bio_chain = bio;
193
		submit_bio(bio_rw, bio);
194 195 196 197 198 199 200 201 202
	}
	return 0;
}

static int bio_read_page(pgoff_t page_off, void *addr, struct bio **bio_chain)
{
	return submit(READ, page_off, virt_to_page(addr), bio_chain);
}

203
static int bio_write_page(pgoff_t page_off, void *addr, struct bio **bio_chain)
204
{
205
	return submit(WRITE, page_off, virt_to_page(addr), bio_chain);
206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238
}

static int wait_on_bio_chain(struct bio **bio_chain)
{
	struct bio *bio;
	struct bio *next_bio;
	int ret = 0;

	if (bio_chain == NULL)
		return 0;

	bio = *bio_chain;
	if (bio == NULL)
		return 0;
	while (bio) {
		struct page *page;

		next_bio = bio->bi_private;
		page = bio->bi_io_vec[0].bv_page;
		wait_on_page_locked(page);
		if (!PageUptodate(page) || PageError(page))
			ret = -EIO;
		put_page(page);
		bio_put(bio);
		bio = next_bio;
	}
	*bio_chain = NULL;
	return ret;
}

/*
 * Saving part
 */
239

240
static int mark_swapfiles(sector_t start, unsigned int flags)
241 242 243
{
	int error;

244 245 246 247 248 249
	bio_read_page(swsusp_resume_block, swsusp_header, NULL);
	if (!memcmp("SWAP-SPACE",swsusp_header->sig, 10) ||
	    !memcmp("SWAPSPACE2",swsusp_header->sig, 10)) {
		memcpy(swsusp_header->orig_sig,swsusp_header->sig, 10);
		memcpy(swsusp_header->sig,SWSUSP_SIG, 10);
		swsusp_header->image = start;
250
		swsusp_header->flags = flags;
251
		error = bio_write_page(swsusp_resume_block,
252
					swsusp_header, NULL);
253
	} else {
R
Rafael J. Wysocki 已提交
254
		printk(KERN_ERR "PM: Swap header not found!\n");
255 256 257 258 259 260 261 262 263 264 265 266
		error = -ENODEV;
	}
	return error;
}

/**
 *	swsusp_swap_check - check if the resume device is a swap device
 *	and get its index (if so)
 */

static int swsusp_swap_check(void) /* This is called before saving image */
{
267 268
	int res;

269 270
	res = swap_type_of(swsusp_resume_device, swsusp_resume_block,
			&resume_bdev);
271 272 273 274
	if (res < 0)
		return res;

	root_swap = res;
275
	res = blkdev_get(resume_bdev, FMODE_WRITE);
276 277
	if (res)
		return res;
278 279 280

	res = set_blocksize(resume_bdev, PAGE_SIZE);
	if (res < 0)
A
Al Viro 已提交
281
		blkdev_put(resume_bdev, FMODE_WRITE);
282 283 284 285 286 287 288 289

	return res;
}

/**
 *	write_page - Write one page to given swap location.
 *	@buf:		Address we're writing.
 *	@offset:	Offset of the swap page we're writing to.
A
Andrew Morton 已提交
290
 *	@bio_chain:	Link the next write BIO here
291 292
 */

293
static int write_page(void *buf, sector_t offset, struct bio **bio_chain)
294
{
295 296 297 298 299 300
	void *src;

	if (!offset)
		return -ENOSPC;

	if (bio_chain) {
R
Rafael J. Wysocki 已提交
301
		src = (void *)__get_free_page(__GFP_WAIT | __GFP_HIGH);
302 303 304 305 306 307
		if (src) {
			memcpy(src, buf, PAGE_SIZE);
		} else {
			WARN_ON_ONCE(1);
			bio_chain = NULL;	/* Go synchronous */
			src = buf;
A
Andrew Morton 已提交
308
		}
309 310
	} else {
		src = buf;
311
	}
312
	return bio_write_page(offset, src, bio_chain);
313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
}

/*
 *	The swap map is a data structure used for keeping track of each page
 *	written to a swap partition.  It consists of many swap_map_page
 *	structures that contain each an array of MAP_PAGE_SIZE swap entries.
 *	These structures are stored on the swap and linked together with the
 *	help of the .next_swap member.
 *
 *	The swap map is created during suspend.  The swap map pages are
 *	allocated and populated one at a time, so we only need one memory
 *	page to set up the entire structure.
 *
 *	During resume we also only need to use one swap_map_page structure
 *	at a time.
 */

330
#define MAP_PAGE_ENTRIES	(PAGE_SIZE / sizeof(sector_t) - 1)
331 332

struct swap_map_page {
333 334
	sector_t entries[MAP_PAGE_ENTRIES];
	sector_t next_swap;
335 336 337 338 339 340 341 342 343
};

/**
 *	The swap_map_handle structure is used for handling swap in
 *	a file-alike way
 */

struct swap_map_handle {
	struct swap_map_page *cur;
344
	sector_t cur_swap;
345 346 347 348 349 350 351 352 353 354 355 356 357 358 359
	unsigned int k;
};

static void release_swap_writer(struct swap_map_handle *handle)
{
	if (handle->cur)
		free_page((unsigned long)handle->cur);
	handle->cur = NULL;
}

static int get_swap_writer(struct swap_map_handle *handle)
{
	handle->cur = (struct swap_map_page *)get_zeroed_page(GFP_KERNEL);
	if (!handle->cur)
		return -ENOMEM;
360
	handle->cur_swap = alloc_swapdev_block(root_swap);
361 362 363 364 365 366 367 368
	if (!handle->cur_swap) {
		release_swap_writer(handle);
		return -ENOSPC;
	}
	handle->k = 0;
	return 0;
}

A
Andrew Morton 已提交
369 370 371 372
static int swap_write_page(struct swap_map_handle *handle, void *buf,
				struct bio **bio_chain)
{
	int error = 0;
373
	sector_t offset;
374 375 376

	if (!handle->cur)
		return -EINVAL;
377
	offset = alloc_swapdev_block(root_swap);
A
Andrew Morton 已提交
378
	error = write_page(buf, offset, bio_chain);
379 380 381 382
	if (error)
		return error;
	handle->cur->entries[handle->k++] = offset;
	if (handle->k >= MAP_PAGE_ENTRIES) {
A
Andrew Morton 已提交
383 384 385
		error = wait_on_bio_chain(bio_chain);
		if (error)
			goto out;
386
		offset = alloc_swapdev_block(root_swap);
387 388 389
		if (!offset)
			return -ENOSPC;
		handle->cur->next_swap = offset;
A
Andrew Morton 已提交
390
		error = write_page(handle->cur, handle->cur_swap, NULL);
391
		if (error)
A
Andrew Morton 已提交
392
			goto out;
393 394 395 396
		memset(handle->cur, 0, PAGE_SIZE);
		handle->cur_swap = offset;
		handle->k = 0;
	}
R
Rafael J. Wysocki 已提交
397
 out:
A
Andrew Morton 已提交
398
	return error;
399 400 401 402 403
}

static int flush_swap_writer(struct swap_map_handle *handle)
{
	if (handle->cur && handle->cur_swap)
A
Andrew Morton 已提交
404
		return write_page(handle->cur, handle->cur_swap, NULL);
405 406 407 408 409 410 411 412 413 414
	else
		return -EINVAL;
}

/**
 *	save_image - save the suspend image data
 */

static int save_image(struct swap_map_handle *handle,
                      struct snapshot_handle *snapshot,
415
                      unsigned int nr_to_write)
416 417 418
{
	unsigned int m;
	int ret;
419
	int nr_pages;
A
Andrew Morton 已提交
420 421
	int err2;
	struct bio *bio;
422 423
	struct timeval start;
	struct timeval stop;
424

R
Rafael J. Wysocki 已提交
425 426
	printk(KERN_INFO "PM: Saving image data pages (%u pages) ...     ",
		nr_to_write);
427
	m = nr_to_write / 100;
428 429 430
	if (!m)
		m = 1;
	nr_pages = 0;
A
Andrew Morton 已提交
431
	bio = NULL;
432
	do_gettimeofday(&start);
433
	while (1) {
J
Jiri Slaby 已提交
434
		ret = snapshot_read_next(snapshot);
435 436 437 438 439 440
		if (ret <= 0)
			break;
		ret = swap_write_page(handle, data_of(*snapshot), &bio);
		if (ret)
			break;
		if (!(nr_pages % m))
J
Jiri Slaby 已提交
441
			printk(KERN_CONT "\b\b\b\b%3d%%", nr_pages / m);
442 443
		nr_pages++;
	}
A
Andrew Morton 已提交
444
	err2 = wait_on_bio_chain(&bio);
445
	do_gettimeofday(&stop);
446 447 448
	if (!ret)
		ret = err2;
	if (!ret)
J
Jiri Slaby 已提交
449
		printk(KERN_CONT "\b\b\b\bdone\n");
450
	else
J
Jiri Slaby 已提交
451
		printk(KERN_CONT "\n");
452
	swsusp_show_speed(&start, &stop, nr_to_write, "Wrote");
453
	return ret;
454 455 456 457 458 459 460 461 462 463 464 465 466
}

/**
 *	enough_swap - Make sure we have enough swap to save the image.
 *
 *	Returns TRUE or FALSE after checking the total amount of swap
 *	space avaiable from the resume partition.
 */

static int enough_swap(unsigned int nr_pages)
{
	unsigned int free_swap = count_swap_pages(root_swap, 1);

R
Rafael J. Wysocki 已提交
467
	pr_debug("PM: Free swap pages: %u\n", free_swap);
468
	return free_swap > nr_pages + PAGES_FOR_IO;
469 470 471 472
}

/**
 *	swsusp_write - Write entire image and metadata.
473
 *	@flags: flags to pass to the "boot" kernel in the image header
474 475 476 477 478 479 480
 *
 *	It is important _NOT_ to umount filesystems at this point. We want
 *	them synced (in case something goes wrong) but we DO not want to mark
 *	filesystem clean: it is not. (And it does not matter, if we resume
 *	correctly, we'll mark system clean, anyway.)
 */

481
int swsusp_write(unsigned int flags)
482 483 484 485 486 487
{
	struct swap_map_handle handle;
	struct snapshot_handle snapshot;
	struct swsusp_info *header;
	int error;

488 489
	error = swsusp_swap_check();
	if (error) {
R
Rafael J. Wysocki 已提交
490
		printk(KERN_ERR "PM: Cannot find swap device, try "
A
Andrew Morton 已提交
491
				"swapon -a.\n");
492 493 494
		return error;
	}
	memset(&snapshot, 0, sizeof(struct snapshot_handle));
J
Jiri Slaby 已提交
495
	error = snapshot_read_next(&snapshot);
496 497 498 499 500 501
	if (error < PAGE_SIZE) {
		if (error >= 0)
			error = -EFAULT;

		goto out;
	}
502 503
	header = (struct swsusp_info *)data_of(snapshot);
	if (!enough_swap(header->pages)) {
R
Rafael J. Wysocki 已提交
504
		printk(KERN_ERR "PM: Not enough free swap\n");
505 506
		error = -ENOSPC;
		goto out;
507 508 509
	}
	error = get_swap_writer(&handle);
	if (!error) {
510 511
		sector_t start = handle.cur_swap;

A
Andrew Morton 已提交
512
		error = swap_write_page(&handle, header, NULL);
A
Andrew Morton 已提交
513 514 515
		if (!error)
			error = save_image(&handle, &snapshot,
					header->pages - 1);
516

A
Andrew Morton 已提交
517 518
		if (!error) {
			flush_swap_writer(&handle);
R
Rafael J. Wysocki 已提交
519
			printk(KERN_INFO "PM: S");
520
			error = mark_swapfiles(start, flags);
A
Andrew Morton 已提交
521 522
			printk("|\n");
		}
523 524
	}
	if (error)
525 526
		free_all_swap_pages(root_swap);

527
	release_swap_writer(&handle);
R
Rafael J. Wysocki 已提交
528
 out:
529
	swsusp_close(FMODE_WRITE);
530 531 532 533 534 535 536 537 538 539 540 541 542 543 544
	return error;
}

/**
 *	The following functions allow us to read data using a swap map
 *	in a file-alike way
 */

static void release_swap_reader(struct swap_map_handle *handle)
{
	if (handle->cur)
		free_page((unsigned long)handle->cur);
	handle->cur = NULL;
}

545
static int get_swap_reader(struct swap_map_handle *handle, sector_t start)
546 547 548
{
	int error;

549
	if (!start)
550
		return -EINVAL;
551

R
Rafael J. Wysocki 已提交
552
	handle->cur = (struct swap_map_page *)get_zeroed_page(__GFP_WAIT | __GFP_HIGH);
553 554
	if (!handle->cur)
		return -ENOMEM;
555 556

	error = bio_read_page(start, handle->cur, NULL);
557 558 559 560 561 562 563 564
	if (error) {
		release_swap_reader(handle);
		return error;
	}
	handle->k = 0;
	return 0;
}

A
Andrew Morton 已提交
565 566
static int swap_read_page(struct swap_map_handle *handle, void *buf,
				struct bio **bio_chain)
567
{
568
	sector_t offset;
569 570 571 572 573 574 575
	int error;

	if (!handle->cur)
		return -EINVAL;
	offset = handle->cur->entries[handle->k];
	if (!offset)
		return -EFAULT;
A
Andrew Morton 已提交
576
	error = bio_read_page(offset, buf, bio_chain);
577 578 579
	if (error)
		return error;
	if (++handle->k >= MAP_PAGE_ENTRIES) {
A
Andrew Morton 已提交
580
		error = wait_on_bio_chain(bio_chain);
581 582 583 584
		handle->k = 0;
		offset = handle->cur->next_swap;
		if (!offset)
			release_swap_reader(handle);
A
Andrew Morton 已提交
585 586
		else if (!error)
			error = bio_read_page(offset, handle->cur, NULL);
587 588 589 590 591 592 593 594 595 596 597 598
	}
	return error;
}

/**
 *	load_image - load the image using the swap map handle
 *	@handle and the snapshot handle @snapshot
 *	(assume there are @nr_pages pages to load)
 */

static int load_image(struct swap_map_handle *handle,
                      struct snapshot_handle *snapshot,
A
Andrew Morton 已提交
599
                      unsigned int nr_to_read)
600 601 602
{
	unsigned int m;
	int error = 0;
603 604
	struct timeval start;
	struct timeval stop;
A
Andrew Morton 已提交
605 606 607
	struct bio *bio;
	int err2;
	unsigned nr_pages;
608

R
Rafael J. Wysocki 已提交
609 610
	printk(KERN_INFO "PM: Loading image data pages (%u pages) ...     ",
		nr_to_read);
A
Andrew Morton 已提交
611
	m = nr_to_read / 100;
612 613 614
	if (!m)
		m = 1;
	nr_pages = 0;
A
Andrew Morton 已提交
615
	bio = NULL;
616
	do_gettimeofday(&start);
A
Andrew Morton 已提交
617
	for ( ; ; ) {
J
Jiri Slaby 已提交
618
		error = snapshot_write_next(snapshot);
A
Andrew Morton 已提交
619 620 621 622 623 624 625 626 627 628 629 630 631 632
		if (error <= 0)
			break;
		error = swap_read_page(handle, data_of(*snapshot), &bio);
		if (error)
			break;
		if (snapshot->sync_read)
			error = wait_on_bio_chain(&bio);
		if (error)
			break;
		if (!(nr_pages % m))
			printk("\b\b\b\b%3d%%", nr_pages / m);
		nr_pages++;
	}
	err2 = wait_on_bio_chain(&bio);
633
	do_gettimeofday(&stop);
A
Andrew Morton 已提交
634 635
	if (!error)
		error = err2;
636
	if (!error) {
637
		printk("\b\b\b\bdone\n");
638
		snapshot_write_finalize(snapshot);
639 640
		if (!snapshot_image_loaded(snapshot))
			error = -ENODATA;
641 642
	} else
		printk("\n");
643
	swsusp_show_speed(&start, &stop, nr_to_read, "Read");
644 645 646
	return error;
}

647 648 649 650 651 652 653
/**
 *	swsusp_read - read the hibernation image.
 *	@flags_p: flags passed by the "frozen" kernel in the image header should
 *		  be written into this memeory location
 */

int swsusp_read(unsigned int *flags_p)
654 655 656 657 658 659
{
	int error;
	struct swap_map_handle handle;
	struct snapshot_handle snapshot;
	struct swsusp_info *header;

660
	*flags_p = swsusp_header->flags;
661 662

	memset(&snapshot, 0, sizeof(struct snapshot_handle));
J
Jiri Slaby 已提交
663
	error = snapshot_write_next(&snapshot);
664 665 666
	if (error < PAGE_SIZE)
		return error < 0 ? error : -EFAULT;
	header = (struct swsusp_info *)data_of(snapshot);
667
	error = get_swap_reader(&handle, swsusp_header->image);
668
	if (!error)
A
Andrew Morton 已提交
669
		error = swap_read_page(&handle, header, NULL);
670 671 672 673 674
	if (!error)
		error = load_image(&handle, &snapshot, header->pages - 1);
	release_swap_reader(&handle);

	if (!error)
R
Rafael J. Wysocki 已提交
675
		pr_debug("PM: Image successfully loaded\n");
676
	else
R
Rafael J. Wysocki 已提交
677
		pr_debug("PM: Error %d resuming\n", error);
678 679 680 681 682 683 684 685 686 687 688 689 690 691
	return error;
}

/**
 *      swsusp_check - Check for swsusp signature in the resume device
 */

int swsusp_check(void)
{
	int error;

	resume_bdev = open_by_devnum(swsusp_resume_device, FMODE_READ);
	if (!IS_ERR(resume_bdev)) {
		set_blocksize(resume_bdev, PAGE_SIZE);
692
		memset(swsusp_header, 0, PAGE_SIZE);
693
		error = bio_read_page(swsusp_resume_block,
694
					swsusp_header, NULL);
695
		if (error)
J
Jiri Slaby 已提交
696
			goto put;
697

698 699
		if (!memcmp(SWSUSP_SIG, swsusp_header->sig, 10)) {
			memcpy(swsusp_header->sig, swsusp_header->orig_sig, 10);
700
			/* Reset swap signature now */
701
			error = bio_write_page(swsusp_resume_block,
702
						swsusp_header, NULL);
703
		} else {
J
Jiri Slaby 已提交
704
			error = -EINVAL;
705
		}
J
Jiri Slaby 已提交
706 707

put:
708
		if (error)
A
Al Viro 已提交
709
			blkdev_put(resume_bdev, FMODE_READ);
710
		else
R
Rafael J. Wysocki 已提交
711
			pr_debug("PM: Signature found, resuming\n");
712 713 714 715 716
	} else {
		error = PTR_ERR(resume_bdev);
	}

	if (error)
R
Rafael J. Wysocki 已提交
717
		pr_debug("PM: Error %d checking image file\n", error);
718 719 720 721 722 723 724 725

	return error;
}

/**
 *	swsusp_close - close swap device.
 */

726
void swsusp_close(fmode_t mode)
727 728
{
	if (IS_ERR(resume_bdev)) {
R
Rafael J. Wysocki 已提交
729
		pr_debug("PM: Image device not initialised\n");
730 731 732
		return;
	}

733
	blkdev_put(resume_bdev, mode);
734
}
735 736 737 738 739 740 741 742 743 744

static int swsusp_header_init(void)
{
	swsusp_header = (struct swsusp_header*) __get_free_page(GFP_KERNEL);
	if (!swsusp_header)
		panic("Could not allocate memory for swsusp_header\n");
	return 0;
}

core_initcall(swsusp_header_init);