aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--arch/x86_64/kernel/e820.c26
-rw-r--r--include/linux/suspend.h58
-rw-r--r--kernel/power/disk.c23
-rw-r--r--kernel/power/main.c2
-rw-r--r--kernel/power/power.h2
-rw-r--r--kernel/power/snapshot.c250
-rw-r--r--kernel/power/user.c4
7 files changed, 283 insertions, 82 deletions
diff --git a/arch/x86_64/kernel/e820.c b/arch/x86_64/kernel/e820.c
index be8965427a93..13c6c37610e0 100644
--- a/arch/x86_64/kernel/e820.c
+++ b/arch/x86_64/kernel/e820.c
@@ -17,6 +17,8 @@
17#include <linux/kexec.h> 17#include <linux/kexec.h>
18#include <linux/module.h> 18#include <linux/module.h>
19#include <linux/mm.h> 19#include <linux/mm.h>
20#include <linux/suspend.h>
21#include <linux/pfn.h>
20 22
21#include <asm/pgtable.h> 23#include <asm/pgtable.h>
22#include <asm/page.h> 24#include <asm/page.h>
@@ -256,22 +258,6 @@ void __init e820_reserve_resources(void)
256 } 258 }
257} 259}
258 260
259/* Mark pages corresponding to given address range as nosave */
260static void __init
261e820_mark_nosave_range(unsigned long start, unsigned long end)
262{
263 unsigned long pfn, max_pfn;
264
265 if (start >= end)
266 return;
267
268 printk("Nosave address range: %016lx - %016lx\n", start, end);
269 max_pfn = end >> PAGE_SHIFT;
270 for (pfn = start >> PAGE_SHIFT; pfn < max_pfn; pfn++)
271 if (pfn_valid(pfn))
272 SetPageNosave(pfn_to_page(pfn));
273}
274
275/* 261/*
276 * Find the ranges of physical addresses that do not correspond to 262 * Find the ranges of physical addresses that do not correspond to
277 * e820 RAM areas and mark the corresponding pages as nosave for software 263 * e820 RAM areas and mark the corresponding pages as nosave for software
@@ -290,13 +276,13 @@ void __init e820_mark_nosave_regions(void)
290 struct e820entry *ei = &e820.map[i]; 276 struct e820entry *ei = &e820.map[i];
291 277
292 if (paddr < ei->addr) 278 if (paddr < ei->addr)
293 e820_mark_nosave_range(paddr, 279 register_nosave_region(PFN_DOWN(paddr),
294 round_up(ei->addr, PAGE_SIZE)); 280 PFN_UP(ei->addr));
295 281
296 paddr = round_down(ei->addr + ei->size, PAGE_SIZE); 282 paddr = round_down(ei->addr + ei->size, PAGE_SIZE);
297 if (ei->type != E820_RAM) 283 if (ei->type != E820_RAM)
298 e820_mark_nosave_range(round_up(ei->addr, PAGE_SIZE), 284 register_nosave_region(PFN_UP(ei->addr),
299 paddr); 285 PFN_DOWN(paddr));
300 286
301 if (paddr >= (end_pfn << PAGE_SHIFT)) 287 if (paddr >= (end_pfn << PAGE_SHIFT))
302 break; 288 break;
diff --git a/include/linux/suspend.h b/include/linux/suspend.h
index a45b9f514492..3cc4d6394c07 100644
--- a/include/linux/suspend.h
+++ b/include/linux/suspend.h
@@ -24,63 +24,41 @@ struct pbe {
24extern void drain_local_pages(void); 24extern void drain_local_pages(void);
25extern void mark_free_pages(struct zone *zone); 25extern void mark_free_pages(struct zone *zone);
26 26
27#ifdef CONFIG_PM 27#if defined(CONFIG_PM) && defined(CONFIG_VT) && defined(CONFIG_VT_CONSOLE)
28/* kernel/power/swsusp.c */
29extern int software_suspend(void);
30
31#if defined(CONFIG_VT) && defined(CONFIG_VT_CONSOLE)
32extern int pm_prepare_console(void); 28extern int pm_prepare_console(void);
33extern void pm_restore_console(void); 29extern void pm_restore_console(void);
34#else 30#else
35static inline int pm_prepare_console(void) { return 0; } 31static inline int pm_prepare_console(void) { return 0; }
36static inline void pm_restore_console(void) {} 32static inline void pm_restore_console(void) {}
37#endif /* defined(CONFIG_VT) && defined(CONFIG_VT_CONSOLE) */ 33#endif
34
35#if defined(CONFIG_PM) && defined(CONFIG_SOFTWARE_SUSPEND)
36/* kernel/power/swsusp.c */
37extern int software_suspend(void);
38/* kernel/power/snapshot.c */
39extern void __init register_nosave_region(unsigned long, unsigned long);
40extern int swsusp_page_is_forbidden(struct page *);
41extern void swsusp_set_page_free(struct page *);
42extern void swsusp_unset_page_free(struct page *);
43extern unsigned long get_safe_page(gfp_t gfp_mask);
38#else 44#else
39static inline int software_suspend(void) 45static inline int software_suspend(void)
40{ 46{
41 printk("Warning: fake suspend called\n"); 47 printk("Warning: fake suspend called\n");
42 return -ENOSYS; 48 return -ENOSYS;
43} 49}
44#endif /* CONFIG_PM */ 50
51static inline void register_nosave_region(unsigned long b, unsigned long e) {}
52static inline int swsusp_page_is_forbidden(struct page *p) { return 0; }
53static inline void swsusp_set_page_free(struct page *p) {}
54static inline void swsusp_unset_page_free(struct page *p) {}
55#endif /* defined(CONFIG_PM) && defined(CONFIG_SOFTWARE_SUSPEND) */
45 56
46void save_processor_state(void); 57void save_processor_state(void);
47void restore_processor_state(void); 58void restore_processor_state(void);
48struct saved_context; 59struct saved_context;
49void __save_processor_state(struct saved_context *ctxt); 60void __save_processor_state(struct saved_context *ctxt);
50void __restore_processor_state(struct saved_context *ctxt); 61void __restore_processor_state(struct saved_context *ctxt);
51unsigned long get_safe_page(gfp_t gfp_mask);
52
53/* Page management functions for the software suspend (swsusp) */
54
55static inline void swsusp_set_page_forbidden(struct page *page)
56{
57 SetPageNosave(page);
58}
59
60static inline int swsusp_page_is_forbidden(struct page *page)
61{
62 return PageNosave(page);
63}
64
65static inline void swsusp_unset_page_forbidden(struct page *page)
66{
67 ClearPageNosave(page);
68}
69
70static inline void swsusp_set_page_free(struct page *page)
71{
72 SetPageNosaveFree(page);
73}
74
75static inline int swsusp_page_is_free(struct page *page)
76{
77 return PageNosaveFree(page);
78}
79
80static inline void swsusp_unset_page_free(struct page *page)
81{
82 ClearPageNosaveFree(page);
83}
84 62
85/* 63/*
86 * XXX: We try to keep some more pages free so that I/O operations succeed 64 * XXX: We try to keep some more pages free so that I/O operations succeed
diff --git a/kernel/power/disk.c b/kernel/power/disk.c
index 8df51c23bba4..403bc3722fee 100644
--- a/kernel/power/disk.c
+++ b/kernel/power/disk.c
@@ -139,14 +139,19 @@ int pm_suspend_disk(void)
139 mdelay(5000); 139 mdelay(5000);
140 goto Thaw; 140 goto Thaw;
141 } 141 }
142 /* Allocate memory management structures */
143 error = create_basic_memory_bitmaps();
144 if (error)
145 goto Thaw;
146
142 /* Free memory before shutting down devices. */ 147 /* Free memory before shutting down devices. */
143 error = swsusp_shrink_memory(); 148 error = swsusp_shrink_memory();
144 if (error) 149 if (error)
145 goto Thaw; 150 goto Finish;
146 151
147 error = platform_prepare(); 152 error = platform_prepare();
148 if (error) 153 if (error)
149 goto Thaw; 154 goto Finish;
150 155
151 suspend_console(); 156 suspend_console();
152 error = device_suspend(PMSG_FREEZE); 157 error = device_suspend(PMSG_FREEZE);
@@ -181,7 +186,7 @@ int pm_suspend_disk(void)
181 power_down(); 186 power_down();
182 else { 187 else {
183 swsusp_free(); 188 swsusp_free();
184 goto Thaw; 189 goto Finish;
185 } 190 }
186 } else { 191 } else {
187 pr_debug("PM: Image restored successfully.\n"); 192 pr_debug("PM: Image restored successfully.\n");
@@ -194,6 +199,8 @@ int pm_suspend_disk(void)
194 platform_finish(); 199 platform_finish();
195 device_resume(); 200 device_resume();
196 resume_console(); 201 resume_console();
202 Finish:
203 free_basic_memory_bitmaps();
197 Thaw: 204 Thaw:
198 unprepare_processes(); 205 unprepare_processes();
199 return error; 206 return error;
@@ -239,13 +246,15 @@ static int software_resume(void)
239 } 246 }
240 247
241 pr_debug("PM: Checking swsusp image.\n"); 248 pr_debug("PM: Checking swsusp image.\n");
242
243 error = swsusp_check(); 249 error = swsusp_check();
244 if (error) 250 if (error)
245 goto Done; 251 goto Unlock;
246 252
247 pr_debug("PM: Preparing processes for restore.\n"); 253 error = create_basic_memory_bitmaps();
254 if (error)
255 goto Unlock;
248 256
257 pr_debug("PM: Preparing processes for restore.\n");
249 error = prepare_processes(); 258 error = prepare_processes();
250 if (error) { 259 if (error) {
251 swsusp_close(); 260 swsusp_close();
@@ -280,7 +289,9 @@ static int software_resume(void)
280 printk(KERN_ERR "PM: Restore failed, recovering.\n"); 289 printk(KERN_ERR "PM: Restore failed, recovering.\n");
281 unprepare_processes(); 290 unprepare_processes();
282 Done: 291 Done:
292 free_basic_memory_bitmaps();
283 /* For success case, the suspend path will release the lock */ 293 /* For success case, the suspend path will release the lock */
294 Unlock:
284 mutex_unlock(&pm_mutex); 295 mutex_unlock(&pm_mutex);
285 pr_debug("PM: Resume from disk failed.\n"); 296 pr_debug("PM: Resume from disk failed.\n");
286 return 0; 297 return 0;
diff --git a/kernel/power/main.c b/kernel/power/main.c
index b21c2a56f960..5a779270cdbc 100644
--- a/kernel/power/main.c
+++ b/kernel/power/main.c
@@ -244,6 +244,7 @@ static int enter_state(suspend_state_t state)
244 return error; 244 return error;
245} 245}
246 246
247#ifdef CONFIG_SOFTWARE_SUSPEND
247/* 248/*
248 * This is main interface to the outside world. It needs to be 249 * This is main interface to the outside world. It needs to be
249 * called from process context. 250 * called from process context.
@@ -252,6 +253,7 @@ int software_suspend(void)
252{ 253{
253 return enter_state(PM_SUSPEND_DISK); 254 return enter_state(PM_SUSPEND_DISK);
254} 255}
256#endif
255 257
256 258
257/** 259/**
diff --git a/kernel/power/power.h b/kernel/power/power.h
index 33bd94ceba32..1f8052bda0f7 100644
--- a/kernel/power/power.h
+++ b/kernel/power/power.h
@@ -48,6 +48,8 @@ extern sector_t swsusp_resume_block;
48extern asmlinkage int swsusp_arch_suspend(void); 48extern asmlinkage int swsusp_arch_suspend(void);
49extern asmlinkage int swsusp_arch_resume(void); 49extern asmlinkage int swsusp_arch_resume(void);
50 50
51extern int create_basic_memory_bitmaps(void);
52extern void free_basic_memory_bitmaps(void);
51extern unsigned int count_data_pages(void); 53extern unsigned int count_data_pages(void);
52 54
53/** 55/**
diff --git a/kernel/power/snapshot.c b/kernel/power/snapshot.c
index 48fc7a35571b..f66e4411795b 100644
--- a/kernel/power/snapshot.c
+++ b/kernel/power/snapshot.c
@@ -21,6 +21,7 @@
21#include <linux/kernel.h> 21#include <linux/kernel.h>
22#include <linux/pm.h> 22#include <linux/pm.h>
23#include <linux/device.h> 23#include <linux/device.h>
24#include <linux/init.h>
24#include <linux/bootmem.h> 25#include <linux/bootmem.h>
25#include <linux/syscalls.h> 26#include <linux/syscalls.h>
26#include <linux/console.h> 27#include <linux/console.h>
@@ -34,6 +35,10 @@
34 35
35#include "power.h" 36#include "power.h"
36 37
38static int swsusp_page_is_free(struct page *);
39static void swsusp_set_page_forbidden(struct page *);
40static void swsusp_unset_page_forbidden(struct page *);
41
37/* List of PBEs needed for restoring the pages that were allocated before 42/* List of PBEs needed for restoring the pages that were allocated before
38 * the suspend and included in the suspend image, but have also been 43 * the suspend and included in the suspend image, but have also been
39 * allocated by the "resume" kernel, so their contents cannot be written 44 * allocated by the "resume" kernel, so their contents cannot be written
@@ -224,11 +229,6 @@ static void chain_free(struct chain_allocator *ca, int clear_page_nosave)
224 * of type unsigned long each). It also contains the pfns that 229 * of type unsigned long each). It also contains the pfns that
225 * correspond to the start and end of the represented memory area and 230 * correspond to the start and end of the represented memory area and
226 * the number of bit chunks in the block. 231 * the number of bit chunks in the block.
227 *
228 * NOTE: Memory bitmaps are used for two types of operations only:
229 * "set a bit" and "find the next bit set". Moreover, the searching
230 * is always carried out after all of the "set a bit" operations
231 * on given bitmap.
232 */ 232 */
233 233
234#define BM_END_OF_MAP (~0UL) 234#define BM_END_OF_MAP (~0UL)
@@ -443,15 +443,13 @@ static void memory_bm_free(struct memory_bitmap *bm, int clear_nosave_free)
443} 443}
444 444
445/** 445/**
446 * memory_bm_set_bit - set the bit in the bitmap @bm that corresponds 446 * memory_bm_find_bit - find the bit in the bitmap @bm that corresponds
447 * to given pfn. The cur_zone_bm member of @bm and the cur_block member 447 * to given pfn. The cur_zone_bm member of @bm and the cur_block member
448 * of @bm->cur_zone_bm are updated. 448 * of @bm->cur_zone_bm are updated.
449 *
450 * If the bit cannot be set, the function returns -EINVAL .
451 */ 449 */
452 450
453static int 451static void memory_bm_find_bit(struct memory_bitmap *bm, unsigned long pfn,
454memory_bm_set_bit(struct memory_bitmap *bm, unsigned long pfn) 452 void **addr, unsigned int *bit_nr)
455{ 453{
456 struct zone_bitmap *zone_bm; 454 struct zone_bitmap *zone_bm;
457 struct bm_block *bb; 455 struct bm_block *bb;
@@ -463,8 +461,8 @@ memory_bm_set_bit(struct memory_bitmap *bm, unsigned long pfn)
463 /* We don't assume that the zones are sorted by pfns */ 461 /* We don't assume that the zones are sorted by pfns */
464 while (pfn < zone_bm->start_pfn || pfn >= zone_bm->end_pfn) { 462 while (pfn < zone_bm->start_pfn || pfn >= zone_bm->end_pfn) {
465 zone_bm = zone_bm->next; 463 zone_bm = zone_bm->next;
466 if (unlikely(!zone_bm)) 464
467 return -EINVAL; 465 BUG_ON(!zone_bm);
468 } 466 }
469 bm->cur.zone_bm = zone_bm; 467 bm->cur.zone_bm = zone_bm;
470 } 468 }
@@ -475,13 +473,40 @@ memory_bm_set_bit(struct memory_bitmap *bm, unsigned long pfn)
475 473
476 while (pfn >= bb->end_pfn) { 474 while (pfn >= bb->end_pfn) {
477 bb = bb->next; 475 bb = bb->next;
478 if (unlikely(!bb)) 476
479 return -EINVAL; 477 BUG_ON(!bb);
480 } 478 }
481 zone_bm->cur_block = bb; 479 zone_bm->cur_block = bb;
482 pfn -= bb->start_pfn; 480 pfn -= bb->start_pfn;
483 set_bit(pfn % BM_BITS_PER_CHUNK, bb->data + pfn / BM_BITS_PER_CHUNK); 481 *bit_nr = pfn % BM_BITS_PER_CHUNK;
484 return 0; 482 *addr = bb->data + pfn / BM_BITS_PER_CHUNK;
483}
484
485static void memory_bm_set_bit(struct memory_bitmap *bm, unsigned long pfn)
486{
487 void *addr;
488 unsigned int bit;
489
490 memory_bm_find_bit(bm, pfn, &addr, &bit);
491 set_bit(bit, addr);
492}
493
494static void memory_bm_clear_bit(struct memory_bitmap *bm, unsigned long pfn)
495{
496 void *addr;
497 unsigned int bit;
498
499 memory_bm_find_bit(bm, pfn, &addr, &bit);
500 clear_bit(bit, addr);
501}
502
503static int memory_bm_test_bit(struct memory_bitmap *bm, unsigned long pfn)
504{
505 void *addr;
506 unsigned int bit;
507
508 memory_bm_find_bit(bm, pfn, &addr, &bit);
509 return test_bit(bit, addr);
485} 510}
486 511
487/* Two auxiliary functions for memory_bm_next_pfn */ 512/* Two auxiliary functions for memory_bm_next_pfn */
@@ -564,6 +589,199 @@ static unsigned long memory_bm_next_pfn(struct memory_bitmap *bm)
564} 589}
565 590
566/** 591/**
592 * This structure represents a range of page frames the contents of which
593 * should not be saved during the suspend.
594 */
595
596struct nosave_region {
597 struct list_head list;
598 unsigned long start_pfn;
599 unsigned long end_pfn;
600};
601
602static LIST_HEAD(nosave_regions);
603
604/**
605 * register_nosave_region - register a range of page frames the contents
606 * of which should not be saved during the suspend (to be used in the early
607 * initialization code)
608 */
609
610void __init
611register_nosave_region(unsigned long start_pfn, unsigned long end_pfn)
612{
613 struct nosave_region *region;
614
615 if (start_pfn >= end_pfn)
616 return;
617
618 if (!list_empty(&nosave_regions)) {
619 /* Try to extend the previous region (they should be sorted) */
620 region = list_entry(nosave_regions.prev,
621 struct nosave_region, list);
622 if (region->end_pfn == start_pfn) {
623 region->end_pfn = end_pfn;
624 goto Report;
625 }
626 }
627 /* This allocation cannot fail */
628 region = alloc_bootmem_low(sizeof(struct nosave_region));
629 region->start_pfn = start_pfn;
630 region->end_pfn = end_pfn;
631 list_add_tail(&region->list, &nosave_regions);
632 Report:
633 printk("swsusp: Registered nosave memory region: %016lx - %016lx\n",
634 start_pfn << PAGE_SHIFT, end_pfn << PAGE_SHIFT);
635}
636
637/*
638 * Set bits in this map correspond to the page frames the contents of which
639 * should not be saved during the suspend.
640 */
641static struct memory_bitmap *forbidden_pages_map;
642
643/* Set bits in this map correspond to free page frames. */
644static struct memory_bitmap *free_pages_map;
645
646/*
647 * Each page frame allocated for creating the image is marked by setting the
648 * corresponding bits in forbidden_pages_map and free_pages_map simultaneously
649 */
650
651void swsusp_set_page_free(struct page *page)
652{
653 if (free_pages_map)
654 memory_bm_set_bit(free_pages_map, page_to_pfn(page));
655}
656
657static int swsusp_page_is_free(struct page *page)
658{
659 return free_pages_map ?
660 memory_bm_test_bit(free_pages_map, page_to_pfn(page)) : 0;
661}
662
663void swsusp_unset_page_free(struct page *page)
664{
665 if (free_pages_map)
666 memory_bm_clear_bit(free_pages_map, page_to_pfn(page));
667}
668
669static void swsusp_set_page_forbidden(struct page *page)
670{
671 if (forbidden_pages_map)
672 memory_bm_set_bit(forbidden_pages_map, page_to_pfn(page));
673}
674
675int swsusp_page_is_forbidden(struct page *page)
676{
677 return forbidden_pages_map ?
678 memory_bm_test_bit(forbidden_pages_map, page_to_pfn(page)) : 0;
679}
680
681static void swsusp_unset_page_forbidden(struct page *page)
682{
683 if (forbidden_pages_map)
684 memory_bm_clear_bit(forbidden_pages_map, page_to_pfn(page));
685}
686
687/**
688 * mark_nosave_pages - set bits corresponding to the page frames the
689 * contents of which should not be saved in a given bitmap.
690 */
691
692static void mark_nosave_pages(struct memory_bitmap *bm)
693{
694 struct nosave_region *region;
695
696 if (list_empty(&nosave_regions))
697 return;
698
699 list_for_each_entry(region, &nosave_regions, list) {
700 unsigned long pfn;
701
702 printk("swsusp: Marking nosave pages: %016lx - %016lx\n",
703 region->start_pfn << PAGE_SHIFT,
704 region->end_pfn << PAGE_SHIFT);
705
706 for (pfn = region->start_pfn; pfn < region->end_pfn; pfn++)
707 memory_bm_set_bit(bm, pfn);
708 }
709}
710
711/**
712 * create_basic_memory_bitmaps - create bitmaps needed for marking page
713 * frames that should not be saved and free page frames. The pointers
714 * forbidden_pages_map and free_pages_map are only modified if everything
715 * goes well, because we don't want the bits to be used before both bitmaps
716 * are set up.
717 */
718
719int create_basic_memory_bitmaps(void)
720{
721 struct memory_bitmap *bm1, *bm2;
722 int error = 0;
723
724 BUG_ON(forbidden_pages_map || free_pages_map);
725
726 bm1 = kzalloc(sizeof(struct memory_bitmap), GFP_ATOMIC);
727 if (!bm1)
728 return -ENOMEM;
729
730 error = memory_bm_create(bm1, GFP_ATOMIC | __GFP_COLD, PG_ANY);
731 if (error)
732 goto Free_first_object;
733
734 bm2 = kzalloc(sizeof(struct memory_bitmap), GFP_ATOMIC);
735 if (!bm2)
736 goto Free_first_bitmap;
737
738 error = memory_bm_create(bm2, GFP_ATOMIC | __GFP_COLD, PG_ANY);
739 if (error)
740 goto Free_second_object;
741
742 forbidden_pages_map = bm1;
743 free_pages_map = bm2;
744 mark_nosave_pages(forbidden_pages_map);
745
746 printk("swsusp: Basic memory bitmaps created\n");
747
748 return 0;
749
750 Free_second_object:
751 kfree(bm2);
752 Free_first_bitmap:
753 memory_bm_free(bm1, PG_UNSAFE_CLEAR);
754 Free_first_object:
755 kfree(bm1);
756 return -ENOMEM;
757}
758
759/**
760 * free_basic_memory_bitmaps - free memory bitmaps allocated by
761 * create_basic_memory_bitmaps(). The auxiliary pointers are necessary
762 * so that the bitmaps themselves are not referred to while they are being
763 * freed.
764 */
765
766void free_basic_memory_bitmaps(void)
767{
768 struct memory_bitmap *bm1, *bm2;
769
770 BUG_ON(!(forbidden_pages_map && free_pages_map));
771
772 bm1 = forbidden_pages_map;
773 bm2 = free_pages_map;
774 forbidden_pages_map = NULL;
775 free_pages_map = NULL;
776 memory_bm_free(bm1, PG_UNSAFE_CLEAR);
777 kfree(bm1);
778 memory_bm_free(bm2, PG_UNSAFE_CLEAR);
779 kfree(bm2);
780
781 printk("swsusp: Basic memory bitmaps freed\n");
782}
783
784/**
567 * snapshot_additional_pages - estimate the number of additional pages 785 * snapshot_additional_pages - estimate the number of additional pages
568 * be needed for setting up the suspend image data structures for given 786 * be needed for setting up the suspend image data structures for given
569 * zone (usually the returned value is greater than the exact number) 787 * zone (usually the returned value is greater than the exact number)
diff --git a/kernel/power/user.c b/kernel/power/user.c
index 7cf6713b2325..845acd84cb23 100644
--- a/kernel/power/user.c
+++ b/kernel/power/user.c
@@ -52,6 +52,9 @@ static int snapshot_open(struct inode *inode, struct file *filp)
52 if ((filp->f_flags & O_ACCMODE) == O_RDWR) 52 if ((filp->f_flags & O_ACCMODE) == O_RDWR)
53 return -ENOSYS; 53 return -ENOSYS;
54 54
55 if(create_basic_memory_bitmaps())
56 return -ENOMEM;
57
55 nonseekable_open(inode, filp); 58 nonseekable_open(inode, filp);
56 data = &snapshot_state; 59 data = &snapshot_state;
57 filp->private_data = data; 60 filp->private_data = data;
@@ -77,6 +80,7 @@ static int snapshot_release(struct inode *inode, struct file *filp)
77 struct snapshot_data *data; 80 struct snapshot_data *data;
78 81
79 swsusp_free(); 82 swsusp_free();
83 free_basic_memory_bitmaps();
80 data = filp->private_data; 84 data = filp->private_data;
81 free_all_swap_pages(data->swap, data->bitmap); 85 free_all_swap_pages(data->swap, data->bitmap);
82 free_bitmap(data->bitmap); 86 free_bitmap(data->bitmap);