Hibernation: Handle DEBUG_PAGEALLOC on x86

Make hibernation work with CONFIG_DEBUG_PAGEALLOC set on x86, by
checking if the pages to be copied are marked as present in the
kernel mapping and temporarily marking them as present if that's not
the case.  No functional modifications are introduced if
CONFIG_DEBUG_PAGEALLOC is unset.

Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Signed-off-by: Len Brown <len.brown@intel.com>
This commit is contained in:
Rafael J. Wysocki 2008-02-20 01:47:44 +01:00 committed by Len Brown
parent e80af3a8db
commit 8a235efad5
3 changed files with 53 additions and 14 deletions

View File

@ -899,7 +899,24 @@ void kernel_map_pages(struct page *page, int numpages, int enable)
*/ */
cpa_fill_pool(); cpa_fill_pool();
} }
#endif
#ifdef CONFIG_HIBERNATION
bool kernel_page_present(struct page *page)
{
unsigned int level;
pte_t *pte;
if (PageHighMem(page))
return false;
pte = lookup_address((unsigned long)page_address(page), &level);
return (pte_val(*pte) & _PAGE_PRESENT);
}
#endif /* CONFIG_HIBERNATION */
#endif /* CONFIG_DEBUG_PAGEALLOC */
/* /*
* The testcases use internal knowledge of the implementation that shouldn't * The testcases use internal knowledge of the implementation that shouldn't

View File

@ -1171,12 +1171,18 @@ static inline void enable_debug_pagealloc(void)
{ {
debug_pagealloc_enabled = 1; debug_pagealloc_enabled = 1;
} }
#ifdef CONFIG_HIBERNATION
extern bool kernel_page_present(struct page *page);
#endif /* CONFIG_HIBERNATION */
#else #else
static inline void static inline void
kernel_map_pages(struct page *page, int numpages, int enable) {} kernel_map_pages(struct page *page, int numpages, int enable) {}
static inline void enable_debug_pagealloc(void) static inline void enable_debug_pagealloc(void)
{ {
} }
#ifdef CONFIG_HIBERNATION
static inline bool kernel_page_present(struct page *page) { return true; }
#endif /* CONFIG_HIBERNATION */
#endif #endif
extern struct vm_area_struct *get_gate_vma(struct task_struct *tsk); extern struct vm_area_struct *get_gate_vma(struct task_struct *tsk);

View File

@ -875,8 +875,8 @@ static inline void *saveable_highmem_page(unsigned long pfn) { return NULL; }
#endif /* CONFIG_HIGHMEM */ #endif /* CONFIG_HIGHMEM */
/** /**
* saveable - Determine whether a non-highmem page should be included in * saveable_page - Determine whether a non-highmem page should be included
* the suspend image. * in the suspend image.
* *
* We should save the page if it isn't Nosave, and is not in the range * We should save the page if it isn't Nosave, and is not in the range
* of pages statically defined as 'unsaveable', and it isn't a part of * of pages statically defined as 'unsaveable', and it isn't a part of
@ -897,7 +897,8 @@ static struct page *saveable_page(unsigned long pfn)
if (swsusp_page_is_forbidden(page) || swsusp_page_is_free(page)) if (swsusp_page_is_forbidden(page) || swsusp_page_is_free(page))
return NULL; return NULL;
if (PageReserved(page) && pfn_is_nosave(pfn)) if (PageReserved(page)
&& (!kernel_page_present(page) || pfn_is_nosave(pfn)))
return NULL; return NULL;
return page; return page;
@ -938,6 +939,25 @@ static inline void do_copy_page(long *dst, long *src)
*dst++ = *src++; *dst++ = *src++;
} }
/**
* safe_copy_page - check if the page we are going to copy is marked as
* present in the kernel page tables (this always is the case if
* CONFIG_DEBUG_PAGEALLOC is not set and in that case
* kernel_page_present() always returns 'true').
*/
static void safe_copy_page(void *dst, struct page *s_page)
{
if (kernel_page_present(s_page)) {
do_copy_page(dst, page_address(s_page));
} else {
kernel_map_pages(s_page, 1, 1);
do_copy_page(dst, page_address(s_page));
kernel_map_pages(s_page, 1, 0);
}
}
#ifdef CONFIG_HIGHMEM #ifdef CONFIG_HIGHMEM
static inline struct page * static inline struct page *
page_is_saveable(struct zone *zone, unsigned long pfn) page_is_saveable(struct zone *zone, unsigned long pfn)
@ -946,8 +966,7 @@ page_is_saveable(struct zone *zone, unsigned long pfn)
saveable_highmem_page(pfn) : saveable_page(pfn); saveable_highmem_page(pfn) : saveable_page(pfn);
} }
static inline void static void copy_data_page(unsigned long dst_pfn, unsigned long src_pfn)
copy_data_page(unsigned long dst_pfn, unsigned long src_pfn)
{ {
struct page *s_page, *d_page; struct page *s_page, *d_page;
void *src, *dst; void *src, *dst;
@ -961,29 +980,26 @@ copy_data_page(unsigned long dst_pfn, unsigned long src_pfn)
kunmap_atomic(src, KM_USER0); kunmap_atomic(src, KM_USER0);
kunmap_atomic(dst, KM_USER1); kunmap_atomic(dst, KM_USER1);
} else { } else {
src = page_address(s_page);
if (PageHighMem(d_page)) { if (PageHighMem(d_page)) {
/* Page pointed to by src may contain some kernel /* Page pointed to by src may contain some kernel
* data modified by kmap_atomic() * data modified by kmap_atomic()
*/ */
do_copy_page(buffer, src); safe_copy_page(buffer, s_page);
dst = kmap_atomic(pfn_to_page(dst_pfn), KM_USER0); dst = kmap_atomic(pfn_to_page(dst_pfn), KM_USER0);
memcpy(dst, buffer, PAGE_SIZE); memcpy(dst, buffer, PAGE_SIZE);
kunmap_atomic(dst, KM_USER0); kunmap_atomic(dst, KM_USER0);
} else { } else {
dst = page_address(d_page); safe_copy_page(page_address(d_page), s_page);
do_copy_page(dst, src);
} }
} }
} }
#else #else
#define page_is_saveable(zone, pfn) saveable_page(pfn) #define page_is_saveable(zone, pfn) saveable_page(pfn)
static inline void static inline void copy_data_page(unsigned long dst_pfn, unsigned long src_pfn)
copy_data_page(unsigned long dst_pfn, unsigned long src_pfn)
{ {
do_copy_page(page_address(pfn_to_page(dst_pfn)), safe_copy_page(page_address(pfn_to_page(dst_pfn)),
page_address(pfn_to_page(src_pfn))); pfn_to_page(src_pfn));
} }
#endif /* CONFIG_HIGHMEM */ #endif /* CONFIG_HIGHMEM */