Commit a0f49651 authored by Rafael J. Wysocki's avatar Rafael J. Wysocki Committed by Linus Torvalds

[PATCH] swsusp: reduce the use of global variables

Signed-off-by: default avatarRafael J. Wysocki <rjw@sisk.pl>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 25761b6e
...@@ -69,4 +69,4 @@ extern int restore_highmem(void); ...@@ -69,4 +69,4 @@ extern int restore_highmem(void);
extern void free_pagedir(struct pbe *pblist); extern void free_pagedir(struct pbe *pblist);
extern struct pbe * alloc_pagedir(unsigned nr_pages); extern struct pbe * alloc_pagedir(unsigned nr_pages);
extern void create_pbe_list(struct pbe *pblist, unsigned nr_pages); extern void create_pbe_list(struct pbe *pblist, unsigned nr_pages);
extern int enough_swap(void); extern int enough_swap(unsigned nr_pages);
...@@ -187,37 +187,38 @@ static int saveable(struct zone * zone, unsigned long * zone_pfn) ...@@ -187,37 +187,38 @@ static int saveable(struct zone * zone, unsigned long * zone_pfn)
return 1; return 1;
} }
static void count_data_pages(void) static unsigned count_data_pages(void)
{ {
struct zone *zone; struct zone *zone;
unsigned long zone_pfn; unsigned long zone_pfn;
unsigned n;
nr_copy_pages = 0; n = 0;
for_each_zone (zone) { for_each_zone (zone) {
if (is_highmem(zone)) if (is_highmem(zone))
continue; continue;
mark_free_pages(zone); mark_free_pages(zone);
for (zone_pfn = 0; zone_pfn < zone->spanned_pages; ++zone_pfn) for (zone_pfn = 0; zone_pfn < zone->spanned_pages; ++zone_pfn)
nr_copy_pages += saveable(zone, &zone_pfn); n += saveable(zone, &zone_pfn);
} }
return n;
} }
static void copy_data_pages(void) static void copy_data_pages(struct pbe *pblist)
{ {
struct zone *zone; struct zone *zone;
unsigned long zone_pfn; unsigned long zone_pfn;
struct pbe *pbe = pagedir_nosave, *p; struct pbe *pbe, *p;
pr_debug("copy_data_pages(): pages to copy: %d\n", nr_copy_pages); pbe = pblist;
for_each_zone (zone) { for_each_zone (zone) {
if (is_highmem(zone)) if (is_highmem(zone))
continue; continue;
mark_free_pages(zone); mark_free_pages(zone);
/* This is necessary for swsusp_free() */ /* This is necessary for swsusp_free() */
for_each_pb_page (p, pagedir_nosave) for_each_pb_page (p, pblist)
SetPageNosaveFree(virt_to_page(p)); SetPageNosaveFree(virt_to_page(p));
for_each_pbe(p, pagedir_nosave) for_each_pbe (p, pblist)
SetPageNosaveFree(virt_to_page(p->address)); SetPageNosaveFree(virt_to_page(p->address));
for (zone_pfn = 0; zone_pfn < zone->spanned_pages; ++zone_pfn) { for (zone_pfn = 0; zone_pfn < zone->spanned_pages; ++zone_pfn) {
if (saveable(zone, &zone_pfn)) { if (saveable(zone, &zone_pfn)) {
...@@ -370,46 +371,39 @@ void swsusp_free(void) ...@@ -370,46 +371,39 @@ void swsusp_free(void)
* free pages. * free pages.
*/ */
static int enough_free_mem(void) static int enough_free_mem(unsigned nr_pages)
{ {
pr_debug("swsusp: available memory: %u pages\n", nr_free_pages()); pr_debug("swsusp: available memory: %u pages\n", nr_free_pages());
return nr_free_pages() > (nr_copy_pages + PAGES_FOR_IO + return nr_free_pages() > (nr_pages + PAGES_FOR_IO +
nr_copy_pages/PBES_PER_PAGE + !!(nr_copy_pages%PBES_PER_PAGE)); (nr_pages + PBES_PER_PAGE - 1) / PBES_PER_PAGE);
} }
static int swsusp_alloc(void) static struct pbe *swsusp_alloc(unsigned nr_pages)
{ {
struct pbe * p; struct pbe *pblist, *p;
pagedir_nosave = NULL;
if (MAX_PBES < nr_copy_pages / PBES_PER_PAGE +
!!(nr_copy_pages % PBES_PER_PAGE))
return -ENOSPC;
if (!(pagedir_save = alloc_pagedir(nr_copy_pages))) { if (!(pblist = alloc_pagedir(nr_pages))) {
printk(KERN_ERR "suspend: Allocating pagedir failed.\n"); printk(KERN_ERR "suspend: Allocating pagedir failed.\n");
return -ENOMEM; return NULL;
} }
create_pbe_list(pagedir_save, nr_copy_pages); create_pbe_list(pblist, nr_pages);
pagedir_nosave = pagedir_save;
for_each_pbe (p, pagedir_save) { for_each_pbe (p, pblist) {
p->address = (unsigned long)alloc_image_page(); p->address = (unsigned long)alloc_image_page();
if (!p->address) { if (!p->address) {
printk(KERN_ERR "suspend: Allocating image pages failed.\n"); printk(KERN_ERR "suspend: Allocating image pages failed.\n");
swsusp_free(); swsusp_free();
return -ENOMEM; return NULL;
} }
} }
return 0; return pblist;
} }
static int suspend_prepare_image(void) static int suspend_prepare_image(void)
{ {
int error; unsigned nr_pages;
pr_debug("swsusp: critical section: \n"); pr_debug("swsusp: critical section: \n");
if (save_highmem()) { if (save_highmem()) {
...@@ -419,33 +413,37 @@ static int suspend_prepare_image(void) ...@@ -419,33 +413,37 @@ static int suspend_prepare_image(void)
} }
drain_local_pages(); drain_local_pages();
count_data_pages(); nr_pages = count_data_pages();
printk("swsusp: Need to copy %u pages\n", nr_copy_pages); printk("swsusp: Need to copy %u pages\n", nr_pages);
pr_debug("swsusp: pages needed: %u + %lu + %u, free: %u\n", pr_debug("swsusp: pages needed: %u + %lu + %u, free: %u\n",
nr_copy_pages, nr_pages,
nr_copy_pages/PBES_PER_PAGE + !!(nr_copy_pages%PBES_PER_PAGE), (nr_pages + PBES_PER_PAGE - 1) / PBES_PER_PAGE,
PAGES_FOR_IO, nr_free_pages()); PAGES_FOR_IO, nr_free_pages());
if (!enough_free_mem()) { /* This is needed because of the fixed size of swsusp_info */
if (MAX_PBES < (nr_pages + PBES_PER_PAGE - 1) / PBES_PER_PAGE)
return -ENOSPC;
if (!enough_free_mem(nr_pages)) {
printk(KERN_ERR "swsusp: Not enough free memory\n"); printk(KERN_ERR "swsusp: Not enough free memory\n");
return -ENOMEM; return -ENOMEM;
} }
if (!enough_swap()) { if (!enough_swap(nr_pages)) {
printk(KERN_ERR "swsusp: Not enough free swap\n"); printk(KERN_ERR "swsusp: Not enough free swap\n");
return -ENOSPC; return -ENOSPC;
} }
error = swsusp_alloc(); pagedir_nosave = swsusp_alloc(nr_pages);
if (error) if (!pagedir_nosave)
return error; return -ENOMEM;
/* During allocating of suspend pagedir, new cold pages may appear. /* During allocating of suspend pagedir, new cold pages may appear.
* Kill them. * Kill them.
*/ */
drain_local_pages(); drain_local_pages();
copy_data_pages(); copy_data_pages(pagedir_nosave);
/* /*
* End of critical section. From now on, we can write to memory, * End of critical section. From now on, we can write to memory,
...@@ -453,7 +451,9 @@ static int suspend_prepare_image(void) ...@@ -453,7 +451,9 @@ static int suspend_prepare_image(void)
* touch swap space! Except we must write out our image of course. * touch swap space! Except we must write out our image of course.
*/ */
printk("swsusp: critical section/: done (%d pages copied)\n", nr_copy_pages ); nr_copy_pages = nr_pages;
printk("swsusp: critical section/: done (%d pages copied)\n", nr_pages);
return 0; return 0;
} }
......
...@@ -550,14 +550,14 @@ static int write_suspend_image(void) ...@@ -550,14 +550,14 @@ static int write_suspend_image(void)
* We should only consider resume_device. * We should only consider resume_device.
*/ */
int enough_swap(void) int enough_swap(unsigned nr_pages)
{ {
struct sysinfo i; struct sysinfo i;
si_swapinfo(&i); si_swapinfo(&i);
pr_debug("swsusp: available swap: %lu pages\n", i.freeswap); pr_debug("swsusp: available swap: %lu pages\n", i.freeswap);
return i.freeswap > (nr_copy_pages + PAGES_FOR_IO + return i.freeswap > (nr_pages + PAGES_FOR_IO +
nr_copy_pages/PBES_PER_PAGE + !!(nr_copy_pages%PBES_PER_PAGE)); (nr_pages + PBES_PER_PAGE - 1) / PBES_PER_PAGE);
} }
......
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