Commit 69643279 authored by Rafael J. Wysocki's avatar Rafael J. Wysocki Committed by Len Brown

Hibernate: Do not oops on resume if image data are incorrect

During resume from hibernation using the userland interface image
data are being passed from the used space process to the kernel.
These data need not be valid, but currently the kernel sometimes
oopses if it gets invalid image data, which is wrong.  Make the
kernel return error codes to the user space in such cases.
Signed-off-by: default avatarRafael J. Wysocki <rjw@sisk.pl>
Acked-by: default avatarPavel Machek <pavel@suse.cz>
Signed-off-by: default avatarLen Brown <len.brown@intel.com>
parent ba84ed95
...@@ -519,6 +519,14 @@ static int memory_bm_test_bit(struct memory_bitmap *bm, unsigned long pfn) ...@@ -519,6 +519,14 @@ static int memory_bm_test_bit(struct memory_bitmap *bm, unsigned long pfn)
return test_bit(bit, addr); return test_bit(bit, addr);
} }
static bool memory_bm_pfn_present(struct memory_bitmap *bm, unsigned long pfn)
{
void *addr;
unsigned int bit;
return !memory_bm_find_bit(bm, pfn, &addr, &bit);
}
/** /**
* memory_bm_next_pfn - find the pfn that corresponds to the next set bit * memory_bm_next_pfn - find the pfn that corresponds to the next set bit
* in the bitmap @bm. If the pfn cannot be found, BM_END_OF_MAP is * in the bitmap @bm. If the pfn cannot be found, BM_END_OF_MAP is
...@@ -1459,9 +1467,7 @@ load_header(struct swsusp_info *info) ...@@ -1459,9 +1467,7 @@ load_header(struct swsusp_info *info)
* unpack_orig_pfns - for each element of @buf[] (1 page at a time) set * unpack_orig_pfns - for each element of @buf[] (1 page at a time) set
* the corresponding bit in the memory bitmap @bm * the corresponding bit in the memory bitmap @bm
*/ */
static int unpack_orig_pfns(unsigned long *buf, struct memory_bitmap *bm)
static inline void
unpack_orig_pfns(unsigned long *buf, struct memory_bitmap *bm)
{ {
int j; int j;
...@@ -1469,8 +1475,13 @@ unpack_orig_pfns(unsigned long *buf, struct memory_bitmap *bm) ...@@ -1469,8 +1475,13 @@ unpack_orig_pfns(unsigned long *buf, struct memory_bitmap *bm)
if (unlikely(buf[j] == BM_END_OF_MAP)) if (unlikely(buf[j] == BM_END_OF_MAP))
break; break;
memory_bm_set_bit(bm, buf[j]); if (memory_bm_pfn_present(bm, buf[j]))
memory_bm_set_bit(bm, buf[j]);
else
return -EFAULT;
} }
return 0;
} }
/* List of "safe" pages that may be used to store data loaded from the suspend /* List of "safe" pages that may be used to store data loaded from the suspend
...@@ -1608,7 +1619,7 @@ get_highmem_page_buffer(struct page *page, struct chain_allocator *ca) ...@@ -1608,7 +1619,7 @@ get_highmem_page_buffer(struct page *page, struct chain_allocator *ca)
pbe = chain_alloc(ca, sizeof(struct highmem_pbe)); pbe = chain_alloc(ca, sizeof(struct highmem_pbe));
if (!pbe) { if (!pbe) {
swsusp_free(); swsusp_free();
return NULL; return ERR_PTR(-ENOMEM);
} }
pbe->orig_page = page; pbe->orig_page = page;
if (safe_highmem_pages > 0) { if (safe_highmem_pages > 0) {
...@@ -1677,7 +1688,7 @@ prepare_highmem_image(struct memory_bitmap *bm, unsigned int *nr_highmem_p) ...@@ -1677,7 +1688,7 @@ prepare_highmem_image(struct memory_bitmap *bm, unsigned int *nr_highmem_p)
static inline void * static inline void *
get_highmem_page_buffer(struct page *page, struct chain_allocator *ca) get_highmem_page_buffer(struct page *page, struct chain_allocator *ca)
{ {
return NULL; return ERR_PTR(-EINVAL);
} }
static inline void copy_last_highmem_page(void) {} static inline void copy_last_highmem_page(void) {}
...@@ -1788,8 +1799,13 @@ prepare_image(struct memory_bitmap *new_bm, struct memory_bitmap *bm) ...@@ -1788,8 +1799,13 @@ prepare_image(struct memory_bitmap *new_bm, struct memory_bitmap *bm)
static void *get_buffer(struct memory_bitmap *bm, struct chain_allocator *ca) static void *get_buffer(struct memory_bitmap *bm, struct chain_allocator *ca)
{ {
struct pbe *pbe; struct pbe *pbe;
struct page *page = pfn_to_page(memory_bm_next_pfn(bm)); struct page *page;
unsigned long pfn = memory_bm_next_pfn(bm);
if (pfn == BM_END_OF_MAP)
return ERR_PTR(-EFAULT);
page = pfn_to_page(pfn);
if (PageHighMem(page)) if (PageHighMem(page))
return get_highmem_page_buffer(page, ca); return get_highmem_page_buffer(page, ca);
...@@ -1805,7 +1821,7 @@ static void *get_buffer(struct memory_bitmap *bm, struct chain_allocator *ca) ...@@ -1805,7 +1821,7 @@ static void *get_buffer(struct memory_bitmap *bm, struct chain_allocator *ca)
pbe = chain_alloc(ca, sizeof(struct pbe)); pbe = chain_alloc(ca, sizeof(struct pbe));
if (!pbe) { if (!pbe) {
swsusp_free(); swsusp_free();
return NULL; return ERR_PTR(-ENOMEM);
} }
pbe->orig_address = page_address(page); pbe->orig_address = page_address(page);
pbe->address = safe_pages_list; pbe->address = safe_pages_list;
...@@ -1868,7 +1884,10 @@ int snapshot_write_next(struct snapshot_handle *handle, size_t count) ...@@ -1868,7 +1884,10 @@ int snapshot_write_next(struct snapshot_handle *handle, size_t count)
return error; return error;
} else if (handle->prev <= nr_meta_pages) { } else if (handle->prev <= nr_meta_pages) {
unpack_orig_pfns(buffer, &copy_bm); error = unpack_orig_pfns(buffer, &copy_bm);
if (error)
return error;
if (handle->prev == nr_meta_pages) { if (handle->prev == nr_meta_pages) {
error = prepare_image(&orig_bm, &copy_bm); error = prepare_image(&orig_bm, &copy_bm);
if (error) if (error)
...@@ -1879,12 +1898,14 @@ int snapshot_write_next(struct snapshot_handle *handle, size_t count) ...@@ -1879,12 +1898,14 @@ int snapshot_write_next(struct snapshot_handle *handle, size_t count)
restore_pblist = NULL; restore_pblist = NULL;
handle->buffer = get_buffer(&orig_bm, &ca); handle->buffer = get_buffer(&orig_bm, &ca);
handle->sync_read = 0; handle->sync_read = 0;
if (!handle->buffer) if (IS_ERR(handle->buffer))
return -ENOMEM; return PTR_ERR(handle->buffer);
} }
} else { } else {
copy_last_highmem_page(); copy_last_highmem_page();
handle->buffer = get_buffer(&orig_bm, &ca); handle->buffer = get_buffer(&orig_bm, &ca);
if (IS_ERR(handle->buffer))
return PTR_ERR(handle->buffer);
if (handle->buffer != buffer) if (handle->buffer != buffer)
handle->sync_read = 0; handle->sync_read = 0;
} }
......
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