Commit 64a473cb authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

PM/Hibernate: Do not release preallocated memory unnecessarily (rev. 2)

Since the hibernation code is now going to use allocations of memory
to make enough room for the image, it can also use the page frames
allocated at this stage as image page frames.  The low-level
hibernation code needs to be rearranged for this purpose, but it
allows us to avoid freeing a great number of pages and allocating
these same pages once again later, so it generally is worth doing.

[rev. 2: Take highmem into account correctly.]
Signed-off-by: default avatarRafael J. Wysocki <rjw@sisk.pl>
parent 4bb33435
...@@ -298,8 +298,8 @@ int hibernation_snapshot(int platform_mode) ...@@ -298,8 +298,8 @@ int hibernation_snapshot(int platform_mode)
if (error) if (error)
return error; return error;
/* Free memory before shutting down devices. */ /* Preallocate image memory before shutting down devices. */
error = swsusp_shrink_memory(); error = hibernate_preallocate_memory();
if (error) if (error)
goto Close; goto Close;
...@@ -315,6 +315,10 @@ int hibernation_snapshot(int platform_mode) ...@@ -315,6 +315,10 @@ int hibernation_snapshot(int platform_mode)
/* Control returns here after successful restore */ /* Control returns here after successful restore */
Resume_devices: Resume_devices:
/* We may need to release the preallocated image pages here. */
if (error || !in_suspend)
swsusp_free();
dpm_resume_end(in_suspend ? dpm_resume_end(in_suspend ?
(error ? PMSG_RECOVER : PMSG_THAW) : PMSG_RESTORE); (error ? PMSG_RECOVER : PMSG_THAW) : PMSG_RESTORE);
resume_console(); resume_console();
...@@ -578,7 +582,10 @@ int hibernate(void) ...@@ -578,7 +582,10 @@ int hibernate(void)
goto Thaw; goto Thaw;
error = hibernation_snapshot(hibernation_mode == HIBERNATION_PLATFORM); error = hibernation_snapshot(hibernation_mode == HIBERNATION_PLATFORM);
if (in_suspend && !error) { if (error)
goto Thaw;
if (in_suspend) {
unsigned int flags = 0; unsigned int flags = 0;
if (hibernation_mode == HIBERNATION_PLATFORM) if (hibernation_mode == HIBERNATION_PLATFORM)
...@@ -590,8 +597,8 @@ int hibernate(void) ...@@ -590,8 +597,8 @@ int hibernate(void)
power_down(); power_down();
} else { } else {
pr_debug("PM: Image restored successfully.\n"); pr_debug("PM: Image restored successfully.\n");
swsusp_free();
} }
Thaw: Thaw:
thaw_processes(); thaw_processes();
Finish: Finish:
......
...@@ -74,7 +74,7 @@ extern asmlinkage int swsusp_arch_resume(void); ...@@ -74,7 +74,7 @@ extern asmlinkage int swsusp_arch_resume(void);
extern int create_basic_memory_bitmaps(void); extern int create_basic_memory_bitmaps(void);
extern void free_basic_memory_bitmaps(void); extern void free_basic_memory_bitmaps(void);
extern int swsusp_shrink_memory(void); extern int hibernate_preallocate_memory(void);
/** /**
* Auxiliary structure used for reading the snapshot image data and * Auxiliary structure used for reading the snapshot image data and
......
...@@ -1033,6 +1033,25 @@ copy_data_pages(struct memory_bitmap *copy_bm, struct memory_bitmap *orig_bm) ...@@ -1033,6 +1033,25 @@ copy_data_pages(struct memory_bitmap *copy_bm, struct memory_bitmap *orig_bm)
static unsigned int nr_copy_pages; static unsigned int nr_copy_pages;
/* Number of pages needed for saving the original pfns of the image pages */ /* Number of pages needed for saving the original pfns of the image pages */
static unsigned int nr_meta_pages; static unsigned int nr_meta_pages;
/*
* Numbers of normal and highmem page frames allocated for hibernation image
* before suspending devices.
*/
unsigned int alloc_normal, alloc_highmem;
/*
* Memory bitmap used for marking saveable pages (during hibernation) or
* hibernation image pages (during restore)
*/
static struct memory_bitmap orig_bm;
/*
* Memory bitmap used during hibernation for marking allocated page frames that
* will contain copies of saveable pages. During restore it is initially used
* for marking hibernation image pages, but then the set bits from it are
* duplicated in @orig_bm and it is released. On highmem systems it is next
* used for marking "safe" highmem pages, but it has to be reinitialized for
* this purpose.
*/
static struct memory_bitmap copy_bm;
/** /**
* swsusp_free - free pages allocated for the suspend. * swsusp_free - free pages allocated for the suspend.
...@@ -1064,6 +1083,8 @@ void swsusp_free(void) ...@@ -1064,6 +1083,8 @@ void swsusp_free(void)
nr_meta_pages = 0; nr_meta_pages = 0;
restore_pblist = NULL; restore_pblist = NULL;
buffer = NULL; buffer = NULL;
alloc_normal = 0;
alloc_highmem = 0;
} }
/* Helper functions used for the shrinking of memory. */ /* Helper functions used for the shrinking of memory. */
...@@ -1082,8 +1103,16 @@ static unsigned long preallocate_image_pages(unsigned long nr_pages, gfp_t mask) ...@@ -1082,8 +1103,16 @@ static unsigned long preallocate_image_pages(unsigned long nr_pages, gfp_t mask)
unsigned long nr_alloc = 0; unsigned long nr_alloc = 0;
while (nr_pages > 0) { while (nr_pages > 0) {
if (!alloc_image_page(mask)) struct page *page;
page = alloc_image_page(mask);
if (!page)
break; break;
memory_bm_set_bit(&copy_bm, page_to_pfn(page));
if (PageHighMem(page))
alloc_highmem++;
else
alloc_normal++;
nr_pages--; nr_pages--;
nr_alloc++; nr_alloc++;
} }
...@@ -1135,7 +1164,47 @@ static inline unsigned long preallocate_highmem_fraction(unsigned long nr_pages, ...@@ -1135,7 +1164,47 @@ static inline unsigned long preallocate_highmem_fraction(unsigned long nr_pages,
#endif /* CONFIG_HIGHMEM */ #endif /* CONFIG_HIGHMEM */
/** /**
* swsusp_shrink_memory - Make the kernel release as much memory as needed * free_unnecessary_pages - Release preallocated pages not needed for the image
*/
static void free_unnecessary_pages(void)
{
unsigned long save_highmem, to_free_normal, to_free_highmem;
to_free_normal = alloc_normal - count_data_pages();
save_highmem = count_highmem_pages();
if (alloc_highmem > save_highmem) {
to_free_highmem = alloc_highmem - save_highmem;
} else {
to_free_highmem = 0;
to_free_normal -= save_highmem - alloc_highmem;
}
memory_bm_position_reset(&copy_bm);
while (to_free_normal > 0 && to_free_highmem > 0) {
unsigned long pfn = memory_bm_next_pfn(&copy_bm);
struct page *page = pfn_to_page(pfn);
if (PageHighMem(page)) {
if (!to_free_highmem)
continue;
to_free_highmem--;
alloc_highmem--;
} else {
if (!to_free_normal)
continue;
to_free_normal--;
alloc_normal--;
}
memory_bm_clear_bit(&copy_bm, pfn);
swsusp_unset_page_forbidden(page);
swsusp_unset_page_free(page);
__free_page(page);
}
}
/**
* hibernate_preallocate_memory - Preallocate memory for hibernation image
* *
* To create a hibernation image it is necessary to make a copy of every page * To create a hibernation image it is necessary to make a copy of every page
* frame in use. We also need a number of page frames to be free during * frame in use. We also need a number of page frames to be free during
...@@ -1154,19 +1223,30 @@ static inline unsigned long preallocate_highmem_fraction(unsigned long nr_pages, ...@@ -1154,19 +1223,30 @@ static inline unsigned long preallocate_highmem_fraction(unsigned long nr_pages,
* pages in the system is below the requested image size or it is impossible to * pages in the system is below the requested image size or it is impossible to
* allocate more memory, whichever happens first. * allocate more memory, whichever happens first.
*/ */
int swsusp_shrink_memory(void) int hibernate_preallocate_memory(void)
{ {
struct zone *zone; struct zone *zone;
unsigned long saveable, size, max_size, count, highmem, pages = 0; unsigned long saveable, size, max_size, count, highmem, pages = 0;
unsigned long alloc, pages_highmem; unsigned long alloc, save_highmem, pages_highmem;
struct timeval start, stop; struct timeval start, stop;
int error = 0; int error;
printk(KERN_INFO "PM: Shrinking memory... "); printk(KERN_INFO "PM: Preallocating image memory... ");
do_gettimeofday(&start); do_gettimeofday(&start);
error = memory_bm_create(&orig_bm, GFP_IMAGE, PG_ANY);
if (error)
goto err_out;
error = memory_bm_create(&copy_bm, GFP_IMAGE, PG_ANY);
if (error)
goto err_out;
alloc_normal = 0;
alloc_highmem = 0;
/* Count the number of saveable data pages. */ /* Count the number of saveable data pages. */
highmem = count_highmem_pages(); save_highmem = count_highmem_pages();
saveable = count_data_pages(); saveable = count_data_pages();
/* /*
...@@ -1174,7 +1254,8 @@ int swsusp_shrink_memory(void) ...@@ -1174,7 +1254,8 @@ int swsusp_shrink_memory(void)
* number of pages needed for image metadata (size). * number of pages needed for image metadata (size).
*/ */
count = saveable; count = saveable;
saveable += highmem; saveable += save_highmem;
highmem = save_highmem;
size = 0; size = 0;
for_each_populated_zone(zone) { for_each_populated_zone(zone) {
size += snapshot_additional_pages(zone); size += snapshot_additional_pages(zone);
...@@ -1193,10 +1274,13 @@ int swsusp_shrink_memory(void) ...@@ -1193,10 +1274,13 @@ int swsusp_shrink_memory(void)
size = max_size; size = max_size;
/* /*
* If the maximum is not less than the current number of saveable pages * If the maximum is not less than the current number of saveable pages
* in memory, we don't need to do anything more. * in memory, allocate page frames for the image and we're done.
*/ */
if (size >= saveable) if (size >= saveable) {
pages = preallocate_image_highmem(save_highmem);
pages += preallocate_image_memory(saveable - pages);
goto out; goto out;
}
/* /*
* Let the memory management subsystem know that we're going to need a * Let the memory management subsystem know that we're going to need a
...@@ -1216,10 +1300,8 @@ int swsusp_shrink_memory(void) ...@@ -1216,10 +1300,8 @@ int swsusp_shrink_memory(void)
pages_highmem = preallocate_image_highmem(highmem / 2); pages_highmem = preallocate_image_highmem(highmem / 2);
alloc = (count - max_size) - pages_highmem; alloc = (count - max_size) - pages_highmem;
pages = preallocate_image_memory(alloc); pages = preallocate_image_memory(alloc);
if (pages < alloc) { if (pages < alloc)
error = -ENOMEM; goto err_out;
goto free_out;
}
size = max_size - size; size = max_size - size;
alloc = size; alloc = size;
size = preallocate_highmem_fraction(size, highmem, count); size = preallocate_highmem_fraction(size, highmem, count);
...@@ -1228,21 +1310,24 @@ int swsusp_shrink_memory(void) ...@@ -1228,21 +1310,24 @@ int swsusp_shrink_memory(void)
pages += preallocate_image_memory(alloc); pages += preallocate_image_memory(alloc);
pages += pages_highmem; pages += pages_highmem;
free_out: /*
/* Release all of the preallocated page frames. */ * We only need as many page frames for the image as there are saveable
swsusp_free(); * pages in memory, but we have allocated more. Release the excessive
* ones now.
if (error) { */
printk(KERN_CONT "\n"); free_unnecessary_pages();
return error;
}
out: out:
do_gettimeofday(&stop); do_gettimeofday(&stop);
printk(KERN_CONT "done (preallocated %lu free pages)\n", pages); printk(KERN_CONT "done (allocated %lu pages)\n", pages);
swsusp_show_speed(&start, &stop, pages, "Freed"); swsusp_show_speed(&start, &stop, pages, "Allocated");
return 0; return 0;
err_out:
printk(KERN_CONT "\n");
swsusp_free();
return -ENOMEM;
} }
#ifdef CONFIG_HIGHMEM #ifdef CONFIG_HIGHMEM
...@@ -1253,7 +1338,7 @@ int swsusp_shrink_memory(void) ...@@ -1253,7 +1338,7 @@ int swsusp_shrink_memory(void)
static unsigned int count_pages_for_highmem(unsigned int nr_highmem) static unsigned int count_pages_for_highmem(unsigned int nr_highmem)
{ {
unsigned int free_highmem = count_free_highmem_pages(); unsigned int free_highmem = count_free_highmem_pages() + alloc_highmem;
if (free_highmem >= nr_highmem) if (free_highmem >= nr_highmem)
nr_highmem = 0; nr_highmem = 0;
...@@ -1275,19 +1360,17 @@ count_pages_for_highmem(unsigned int nr_highmem) { return 0; } ...@@ -1275,19 +1360,17 @@ count_pages_for_highmem(unsigned int nr_highmem) { return 0; }
static int enough_free_mem(unsigned int nr_pages, unsigned int nr_highmem) static int enough_free_mem(unsigned int nr_pages, unsigned int nr_highmem)
{ {
struct zone *zone; struct zone *zone;
unsigned int free = 0, meta = 0; unsigned int free = alloc_normal;
for_each_zone(zone) { for_each_zone(zone)
meta += snapshot_additional_pages(zone);
if (!is_highmem(zone)) if (!is_highmem(zone))
free += zone_page_state(zone, NR_FREE_PAGES); free += zone_page_state(zone, NR_FREE_PAGES);
}
nr_pages += count_pages_for_highmem(nr_highmem); nr_pages += count_pages_for_highmem(nr_highmem);
pr_debug("PM: Normal pages needed: %u + %u + %u, available pages: %u\n", pr_debug("PM: Normal pages needed: %u + %u, available pages: %u\n",
nr_pages, PAGES_FOR_IO, meta, free); nr_pages, PAGES_FOR_IO, free);
return free > nr_pages + PAGES_FOR_IO + meta; return free > nr_pages + PAGES_FOR_IO;
} }
#ifdef CONFIG_HIGHMEM #ifdef CONFIG_HIGHMEM
...@@ -1309,7 +1392,7 @@ static inline int get_highmem_buffer(int safe_needed) ...@@ -1309,7 +1392,7 @@ static inline int get_highmem_buffer(int safe_needed)
*/ */
static inline unsigned int static inline unsigned int
alloc_highmem_image_pages(struct memory_bitmap *bm, unsigned int nr_highmem) alloc_highmem_pages(struct memory_bitmap *bm, unsigned int nr_highmem)
{ {
unsigned int to_alloc = count_free_highmem_pages(); unsigned int to_alloc = count_free_highmem_pages();
...@@ -1329,7 +1412,7 @@ alloc_highmem_image_pages(struct memory_bitmap *bm, unsigned int nr_highmem) ...@@ -1329,7 +1412,7 @@ alloc_highmem_image_pages(struct memory_bitmap *bm, unsigned int nr_highmem)
static inline int get_highmem_buffer(int safe_needed) { return 0; } static inline int get_highmem_buffer(int safe_needed) { return 0; }
static inline unsigned int static inline unsigned int
alloc_highmem_image_pages(struct memory_bitmap *bm, unsigned int n) { return 0; } alloc_highmem_pages(struct memory_bitmap *bm, unsigned int n) { return 0; }
#endif /* CONFIG_HIGHMEM */ #endif /* CONFIG_HIGHMEM */
/** /**
...@@ -1348,51 +1431,36 @@ static int ...@@ -1348,51 +1431,36 @@ static int
swsusp_alloc(struct memory_bitmap *orig_bm, struct memory_bitmap *copy_bm, swsusp_alloc(struct memory_bitmap *orig_bm, struct memory_bitmap *copy_bm,
unsigned int nr_pages, unsigned int nr_highmem) unsigned int nr_pages, unsigned int nr_highmem)
{ {
int error; int error = 0;
error = memory_bm_create(orig_bm, GFP_ATOMIC | __GFP_COLD, PG_ANY);
if (error)
goto Free;
error = memory_bm_create(copy_bm, GFP_ATOMIC | __GFP_COLD, PG_ANY);
if (error)
goto Free;
if (nr_highmem > 0) { if (nr_highmem > 0) {
error = get_highmem_buffer(PG_ANY); error = get_highmem_buffer(PG_ANY);
if (error) if (error)
goto Free; goto err_out;
if (nr_highmem > alloc_highmem) {
nr_pages += alloc_highmem_image_pages(copy_bm, nr_highmem); nr_highmem -= alloc_highmem;
nr_pages += alloc_highmem_pages(copy_bm, nr_highmem);
}
} }
if (nr_pages > alloc_normal) {
nr_pages -= alloc_normal;
while (nr_pages-- > 0) { while (nr_pages-- > 0) {
struct page *page = alloc_image_page(GFP_ATOMIC | __GFP_COLD); struct page *page;
page = alloc_image_page(GFP_ATOMIC | __GFP_COLD);
if (!page) if (!page)
goto Free; goto err_out;
memory_bm_set_bit(copy_bm, page_to_pfn(page)); memory_bm_set_bit(copy_bm, page_to_pfn(page));
} }
}
return 0; return 0;
Free: err_out:
swsusp_free(); swsusp_free();
return -ENOMEM; return error;
} }
/* Memory bitmap used for marking saveable pages (during suspend) or the
* suspend image pages (during resume)
*/
static struct memory_bitmap orig_bm;
/* Memory bitmap used on suspend for marking allocated pages that will contain
* the copies of saveable pages. During resume it is initially used for
* marking the suspend image pages, but then its set bits are duplicated in
* @orig_bm and it is released. Next, on systems with high memory, it may be
* used for marking "safe" highmem pages, but it has to be reinitialized for
* this purpose.
*/
static struct memory_bitmap copy_bm;
asmlinkage int swsusp_save(void) asmlinkage int swsusp_save(void)
{ {
unsigned int nr_pages, nr_highmem; unsigned int nr_pages, nr_highmem;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment