[PATCH] swsusp: Reorder memory-allocating functions
Move some functions in kernel/power/snapshot.c to a better place (in the same file) and introduce free_image_page() (will be necessary in the future). Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl> Acked-by: Pavel Machek <pavel@suse.cz> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
Родитель
f623f0db8e
Коммит
f6143aa60e
|
@ -156,6 +156,58 @@ static inline int save_highmem(void) {return 0;}
|
||||||
static inline int restore_highmem(void) {return 0;}
|
static inline int restore_highmem(void) {return 0;}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @safe_needed - on resume, for storing the PBE list and the image,
|
||||||
|
* we can only use memory pages that do not conflict with the pages
|
||||||
|
* used before suspend.
|
||||||
|
*
|
||||||
|
* The unsafe pages are marked with the PG_nosave_free flag
|
||||||
|
* and we count them using unsafe_pages
|
||||||
|
*/
|
||||||
|
|
||||||
|
static unsigned int unsafe_pages;
|
||||||
|
|
||||||
|
static void *alloc_image_page(gfp_t gfp_mask, int safe_needed)
|
||||||
|
{
|
||||||
|
void *res;
|
||||||
|
|
||||||
|
res = (void *)get_zeroed_page(gfp_mask);
|
||||||
|
if (safe_needed)
|
||||||
|
while (res && PageNosaveFree(virt_to_page(res))) {
|
||||||
|
/* The page is unsafe, mark it for swsusp_free() */
|
||||||
|
SetPageNosave(virt_to_page(res));
|
||||||
|
unsafe_pages++;
|
||||||
|
res = (void *)get_zeroed_page(gfp_mask);
|
||||||
|
}
|
||||||
|
if (res) {
|
||||||
|
SetPageNosave(virt_to_page(res));
|
||||||
|
SetPageNosaveFree(virt_to_page(res));
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long get_safe_page(gfp_t gfp_mask)
|
||||||
|
{
|
||||||
|
return (unsigned long)alloc_image_page(gfp_mask, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* free_image_page - free page represented by @addr, allocated with
|
||||||
|
* alloc_image_page (page flags set by it must be cleared)
|
||||||
|
*/
|
||||||
|
|
||||||
|
static inline void free_image_page(void *addr, int clear_nosave_free)
|
||||||
|
{
|
||||||
|
ClearPageNosave(virt_to_page(addr));
|
||||||
|
if (clear_nosave_free)
|
||||||
|
ClearPageNosaveFree(virt_to_page(addr));
|
||||||
|
free_page((unsigned long)addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* pfn_is_nosave - check if given pfn is in the 'nosave' section
|
||||||
|
*/
|
||||||
|
|
||||||
static inline int pfn_is_nosave(unsigned long pfn)
|
static inline int pfn_is_nosave(unsigned long pfn)
|
||||||
{
|
{
|
||||||
unsigned long nosave_begin_pfn = __pa(&__nosave_begin) >> PAGE_SHIFT;
|
unsigned long nosave_begin_pfn = __pa(&__nosave_begin) >> PAGE_SHIFT;
|
||||||
|
@ -245,7 +297,6 @@ static void copy_data_pages(struct pbe *pblist)
|
||||||
BUG_ON(pbe);
|
BUG_ON(pbe);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* free_pagedir - free pages allocated with alloc_pagedir()
|
* free_pagedir - free pages allocated with alloc_pagedir()
|
||||||
*/
|
*/
|
||||||
|
@ -256,10 +307,7 @@ static void free_pagedir(struct pbe *pblist, int clear_nosave_free)
|
||||||
|
|
||||||
while (pblist) {
|
while (pblist) {
|
||||||
pbe = (pblist + PB_PAGE_SKIP)->next;
|
pbe = (pblist + PB_PAGE_SKIP)->next;
|
||||||
ClearPageNosave(virt_to_page(pblist));
|
free_image_page(pblist, clear_nosave_free);
|
||||||
if (clear_nosave_free)
|
|
||||||
ClearPageNosaveFree(virt_to_page(pblist));
|
|
||||||
free_page((unsigned long)pblist);
|
|
||||||
pblist = pbe;
|
pblist = pbe;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -303,41 +351,6 @@ static inline void create_pbe_list(struct pbe *pblist, unsigned int nr_pages)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned int unsafe_pages;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @safe_needed - on resume, for storing the PBE list and the image,
|
|
||||||
* we can only use memory pages that do not conflict with the pages
|
|
||||||
* used before suspend.
|
|
||||||
*
|
|
||||||
* The unsafe pages are marked with the PG_nosave_free flag
|
|
||||||
* and we count them using unsafe_pages
|
|
||||||
*/
|
|
||||||
|
|
||||||
static void *alloc_image_page(gfp_t gfp_mask, int safe_needed)
|
|
||||||
{
|
|
||||||
void *res;
|
|
||||||
|
|
||||||
res = (void *)get_zeroed_page(gfp_mask);
|
|
||||||
if (safe_needed)
|
|
||||||
while (res && PageNosaveFree(virt_to_page(res))) {
|
|
||||||
/* The page is unsafe, mark it for swsusp_free() */
|
|
||||||
SetPageNosave(virt_to_page(res));
|
|
||||||
unsafe_pages++;
|
|
||||||
res = (void *)get_zeroed_page(gfp_mask);
|
|
||||||
}
|
|
||||||
if (res) {
|
|
||||||
SetPageNosave(virt_to_page(res));
|
|
||||||
SetPageNosaveFree(virt_to_page(res));
|
|
||||||
}
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned long get_safe_page(gfp_t gfp_mask)
|
|
||||||
{
|
|
||||||
return (unsigned long)alloc_image_page(gfp_mask, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* alloc_pagedir - Allocate the page directory.
|
* alloc_pagedir - Allocate the page directory.
|
||||||
*
|
*
|
||||||
|
|
Загрузка…
Ссылка в новой задаче