diff options
| author | Rafael J. Wysocki <rjw@sisk.pl> | 2007-05-06 17:50:45 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-05-07 15:12:59 -0400 |
| commit | 0709db6072c2e799eba1aa61bd19e0d7f38aa2cd (patch) | |
| tree | e348c73adbfdcb46b05b23f36dc420ecdd62ca8a /kernel/power | |
| parent | 1525a2ad76f991eba9755f75c9b6d4d97abad25e (diff) | |
swsusp: use GFP_KERNEL for creating basic data structures
Make swsusp call create_basic_memory_bitmaps() before processes are frozen, so
that GFP_KERNEL allocations can be made in it. Additionally, ensure that the
swsusp's userland interface won't be used while either pm_suspend_disk() or
software_resume() is being executed.
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>
Diffstat (limited to 'kernel/power')
| -rw-r--r-- | kernel/power/disk.c | 37 | ||||
| -rw-r--r-- | kernel/power/power.h | 3 | ||||
| -rw-r--r-- | kernel/power/snapshot.c | 8 | ||||
| -rw-r--r-- | kernel/power/user.c | 10 |
4 files changed, 38 insertions, 20 deletions
diff --git a/kernel/power/disk.c b/kernel/power/disk.c index 403bc3722fee..e518379b667a 100644 --- a/kernel/power/disk.c +++ b/kernel/power/disk.c | |||
| @@ -130,28 +130,33 @@ int pm_suspend_disk(void) | |||
| 130 | { | 130 | { |
| 131 | int error; | 131 | int error; |
| 132 | 132 | ||
| 133 | /* The snapshot device should not be opened while we're running */ | ||
| 134 | if (!atomic_add_unless(&snapshot_device_available, -1, 0)) | ||
| 135 | return -EBUSY; | ||
| 136 | |||
| 137 | /* Allocate memory management structures */ | ||
| 138 | error = create_basic_memory_bitmaps(); | ||
| 139 | if (error) | ||
| 140 | goto Exit; | ||
| 141 | |||
| 133 | error = prepare_processes(); | 142 | error = prepare_processes(); |
| 134 | if (error) | 143 | if (error) |
| 135 | return error; | 144 | goto Finish; |
| 136 | 145 | ||
| 137 | if (pm_disk_mode == PM_DISK_TESTPROC) { | 146 | if (pm_disk_mode == PM_DISK_TESTPROC) { |
| 138 | printk("swsusp debug: Waiting for 5 seconds.\n"); | 147 | printk("swsusp debug: Waiting for 5 seconds.\n"); |
| 139 | mdelay(5000); | 148 | mdelay(5000); |
| 140 | goto Thaw; | 149 | goto Thaw; |
| 141 | } | 150 | } |
| 142 | /* Allocate memory management structures */ | ||
| 143 | error = create_basic_memory_bitmaps(); | ||
| 144 | if (error) | ||
| 145 | goto Thaw; | ||
| 146 | 151 | ||
| 147 | /* Free memory before shutting down devices. */ | 152 | /* Free memory before shutting down devices. */ |
| 148 | error = swsusp_shrink_memory(); | 153 | error = swsusp_shrink_memory(); |
| 149 | if (error) | 154 | if (error) |
| 150 | goto Finish; | 155 | goto Thaw; |
| 151 | 156 | ||
| 152 | error = platform_prepare(); | 157 | error = platform_prepare(); |
| 153 | if (error) | 158 | if (error) |
| 154 | goto Finish; | 159 | goto Thaw; |
| 155 | 160 | ||
| 156 | suspend_console(); | 161 | suspend_console(); |
| 157 | error = device_suspend(PMSG_FREEZE); | 162 | error = device_suspend(PMSG_FREEZE); |
| @@ -186,7 +191,7 @@ int pm_suspend_disk(void) | |||
| 186 | power_down(); | 191 | power_down(); |
| 187 | else { | 192 | else { |
| 188 | swsusp_free(); | 193 | swsusp_free(); |
| 189 | goto Finish; | 194 | goto Thaw; |
| 190 | } | 195 | } |
| 191 | } else { | 196 | } else { |
| 192 | pr_debug("PM: Image restored successfully.\n"); | 197 | pr_debug("PM: Image restored successfully.\n"); |
| @@ -199,10 +204,12 @@ int pm_suspend_disk(void) | |||
| 199 | platform_finish(); | 204 | platform_finish(); |
| 200 | device_resume(); | 205 | device_resume(); |
| 201 | resume_console(); | 206 | resume_console(); |
| 202 | Finish: | ||
| 203 | free_basic_memory_bitmaps(); | ||
| 204 | Thaw: | 207 | Thaw: |
| 205 | unprepare_processes(); | 208 | unprepare_processes(); |
| 209 | Finish: | ||
| 210 | free_basic_memory_bitmaps(); | ||
| 211 | Exit: | ||
| 212 | atomic_inc(&snapshot_device_available); | ||
| 206 | return error; | 213 | return error; |
| 207 | } | 214 | } |
| 208 | 215 | ||
| @@ -250,9 +257,15 @@ static int software_resume(void) | |||
| 250 | if (error) | 257 | if (error) |
| 251 | goto Unlock; | 258 | goto Unlock; |
| 252 | 259 | ||
| 260 | /* The snapshot device should not be opened while we're running */ | ||
| 261 | if (!atomic_add_unless(&snapshot_device_available, -1, 0)) { | ||
| 262 | error = -EBUSY; | ||
| 263 | goto Unlock; | ||
| 264 | } | ||
| 265 | |||
| 253 | error = create_basic_memory_bitmaps(); | 266 | error = create_basic_memory_bitmaps(); |
| 254 | if (error) | 267 | if (error) |
| 255 | goto Unlock; | 268 | goto Finish; |
| 256 | 269 | ||
| 257 | pr_debug("PM: Preparing processes for restore.\n"); | 270 | pr_debug("PM: Preparing processes for restore.\n"); |
| 258 | error = prepare_processes(); | 271 | error = prepare_processes(); |
| @@ -290,6 +303,8 @@ static int software_resume(void) | |||
| 290 | unprepare_processes(); | 303 | unprepare_processes(); |
| 291 | Done: | 304 | Done: |
| 292 | free_basic_memory_bitmaps(); | 305 | free_basic_memory_bitmaps(); |
| 306 | Finish: | ||
| 307 | atomic_inc(&snapshot_device_available); | ||
| 293 | /* For success case, the suspend path will release the lock */ | 308 | /* For success case, the suspend path will release the lock */ |
| 294 | Unlock: | 309 | Unlock: |
| 295 | mutex_unlock(&pm_mutex); | 310 | mutex_unlock(&pm_mutex); |
diff --git a/kernel/power/power.h b/kernel/power/power.h index 1f8052bda0f7..a64d3f22de97 100644 --- a/kernel/power/power.h +++ b/kernel/power/power.h | |||
| @@ -140,6 +140,9 @@ struct resume_swap_area { | |||
| 140 | #define PMOPS_ENTER 2 | 140 | #define PMOPS_ENTER 2 |
| 141 | #define PMOPS_FINISH 3 | 141 | #define PMOPS_FINISH 3 |
| 142 | 142 | ||
| 143 | /* If unset, the snapshot device cannot be open. */ | ||
| 144 | extern atomic_t snapshot_device_available; | ||
| 145 | |||
| 143 | /** | 146 | /** |
| 144 | * The bitmap is used for tracing allocated swap pages | 147 | * The bitmap is used for tracing allocated swap pages |
| 145 | * | 148 | * |
diff --git a/kernel/power/snapshot.c b/kernel/power/snapshot.c index f66e4411795b..128da11f01c2 100644 --- a/kernel/power/snapshot.c +++ b/kernel/power/snapshot.c | |||
| @@ -723,19 +723,19 @@ int create_basic_memory_bitmaps(void) | |||
| 723 | 723 | ||
| 724 | BUG_ON(forbidden_pages_map || free_pages_map); | 724 | BUG_ON(forbidden_pages_map || free_pages_map); |
| 725 | 725 | ||
| 726 | bm1 = kzalloc(sizeof(struct memory_bitmap), GFP_ATOMIC); | 726 | bm1 = kzalloc(sizeof(struct memory_bitmap), GFP_KERNEL); |
| 727 | if (!bm1) | 727 | if (!bm1) |
| 728 | return -ENOMEM; | 728 | return -ENOMEM; |
| 729 | 729 | ||
| 730 | error = memory_bm_create(bm1, GFP_ATOMIC | __GFP_COLD, PG_ANY); | 730 | error = memory_bm_create(bm1, GFP_KERNEL, PG_ANY); |
| 731 | if (error) | 731 | if (error) |
| 732 | goto Free_first_object; | 732 | goto Free_first_object; |
| 733 | 733 | ||
| 734 | bm2 = kzalloc(sizeof(struct memory_bitmap), GFP_ATOMIC); | 734 | bm2 = kzalloc(sizeof(struct memory_bitmap), GFP_KERNEL); |
| 735 | if (!bm2) | 735 | if (!bm2) |
| 736 | goto Free_first_bitmap; | 736 | goto Free_first_bitmap; |
| 737 | 737 | ||
| 738 | error = memory_bm_create(bm2, GFP_ATOMIC | __GFP_COLD, PG_ANY); | 738 | error = memory_bm_create(bm2, GFP_KERNEL, PG_ANY); |
| 739 | if (error) | 739 | if (error) |
| 740 | goto Free_second_object; | 740 | goto Free_second_object; |
| 741 | 741 | ||
diff --git a/kernel/power/user.c b/kernel/power/user.c index bd1771f7a64e..72dbfd01408e 100644 --- a/kernel/power/user.c +++ b/kernel/power/user.c | |||
| @@ -40,21 +40,21 @@ static struct snapshot_data { | |||
| 40 | char platform_suspend; | 40 | char platform_suspend; |
| 41 | } snapshot_state; | 41 | } snapshot_state; |
| 42 | 42 | ||
| 43 | static atomic_t device_available = ATOMIC_INIT(1); | 43 | atomic_t snapshot_device_available = ATOMIC_INIT(1); |
| 44 | 44 | ||
| 45 | static int snapshot_open(struct inode *inode, struct file *filp) | 45 | static int snapshot_open(struct inode *inode, struct file *filp) |
| 46 | { | 46 | { |
| 47 | struct snapshot_data *data; | 47 | struct snapshot_data *data; |
| 48 | 48 | ||
| 49 | if (!atomic_add_unless(&device_available, -1, 0)) | 49 | if (!atomic_add_unless(&snapshot_device_available, -1, 0)) |
| 50 | return -EBUSY; | 50 | return -EBUSY; |
| 51 | 51 | ||
| 52 | if ((filp->f_flags & O_ACCMODE) == O_RDWR) { | 52 | if ((filp->f_flags & O_ACCMODE) == O_RDWR) { |
| 53 | atomic_inc(&device_available); | 53 | atomic_inc(&snapshot_device_available); |
| 54 | return -ENOSYS; | 54 | return -ENOSYS; |
| 55 | } | 55 | } |
| 56 | if(create_basic_memory_bitmaps()) { | 56 | if(create_basic_memory_bitmaps()) { |
| 57 | atomic_inc(&device_available); | 57 | atomic_inc(&snapshot_device_available); |
| 58 | return -ENOMEM; | 58 | return -ENOMEM; |
| 59 | } | 59 | } |
| 60 | nonseekable_open(inode, filp); | 60 | nonseekable_open(inode, filp); |
| @@ -92,7 +92,7 @@ static int snapshot_release(struct inode *inode, struct file *filp) | |||
| 92 | enable_nonboot_cpus(); | 92 | enable_nonboot_cpus(); |
| 93 | mutex_unlock(&pm_mutex); | 93 | mutex_unlock(&pm_mutex); |
| 94 | } | 94 | } |
| 95 | atomic_inc(&device_available); | 95 | atomic_inc(&snapshot_device_available); |
| 96 | return 0; | 96 | return 0; |
| 97 | } | 97 | } |
| 98 | 98 | ||
