diff options
-rw-r--r-- | sys/kern/subr_hibernate.c | 44 |
1 files changed, 30 insertions, 14 deletions
diff --git a/sys/kern/subr_hibernate.c b/sys/kern/subr_hibernate.c index 1916ee88b85..9d928096b12 100644 --- a/sys/kern/subr_hibernate.c +++ b/sys/kern/subr_hibernate.c @@ -1,4 +1,4 @@ -/* $OpenBSD: subr_hibernate.c,v 1.90 2014/05/21 02:26:49 mlarkin Exp $ */ +/* $OpenBSD: subr_hibernate.c,v 1.91 2014/05/29 08:00:24 mlarkin Exp $ */ /* * Copyright (c) 2011 Ariane van der Steldt <ariane@stack.nl> @@ -1653,6 +1653,7 @@ hibernate_read_chunks(union hibernate_info *hib, paddr_t pig_start, daddr_t blkctr; size_t processed, compressed_size, read_size; int overlap, found, nchunks, nochunks = 0, nfchunks = 0, npchunks = 0; + int num_io_pages; short *ochunks, *pchunks, *fchunks, i, j; vaddr_t tempva = (vaddr_t)NULL, hibernate_fchunk_area = (vaddr_t)NULL; @@ -1665,7 +1666,8 @@ hibernate_read_chunks(union hibernate_info *hib, paddr_t pig_start, * used only during image read. They dissappear from existence * when the suspended kernel is unpacked on top of us. */ - tempva = (vaddr_t)km_alloc(2*PAGE_SIZE, &kv_any, &kp_none, &kd_nowait); + tempva = (vaddr_t)km_alloc(MAXPHYS + PAGE_SIZE, &kv_any, &kp_none, + &kd_nowait); if (!tempva) return (1); hibernate_fchunk_area = (vaddr_t)km_alloc(24*PAGE_SIZE, &kv_any, @@ -1683,11 +1685,10 @@ hibernate_read_chunks(union hibernate_info *hib, paddr_t pig_start, ochunks = (short *)(hibernate_fchunk_area + (16*PAGE_SIZE)); /* Map the chunk ordering region */ - for(i=0; i<24 ; i++) { + for(i=0; i<24 ; i++) pmap_kenter_pa(hibernate_fchunk_area + (i*PAGE_SIZE), piglet_base + ((4+i)*PAGE_SIZE), VM_PROT_ALL); - pmap_update(pmap_kernel()); - } + pmap_update(pmap_kernel()); nchunks = hib->chunk_ctr; @@ -1803,23 +1804,38 @@ hibernate_read_chunks(union hibernate_info *hib, paddr_t pig_start, compressed_size = chunks[fchunks[i]].compressed_size; while (processed < compressed_size) { - pmap_kenter_pa(tempva, img_cur, VM_PROT_ALL); - pmap_kenter_pa(tempva + PAGE_SIZE, img_cur+PAGE_SIZE, - VM_PROT_ALL); - pmap_update(pmap_kernel()); - - if (compressed_size - processed >= PAGE_SIZE) - read_size = PAGE_SIZE; + if (compressed_size - processed >= MAXPHYS) + read_size = MAXPHYS; else read_size = compressed_size - processed; + /* + * We're reading read_size bytes, offset from the + * start of a page by img_cur % PAGE_SIZE, so the + * end will be read_size + (img_cur % PAGE_SIZE) + * from the start of the first page. Round that + * up to the next page size. + */ + num_io_pages = (read_size + (img_cur % PAGE_SIZE) + + PAGE_SIZE - 1) / PAGE_SIZE; + + KASSERT(num_io_pages <= MAXPHYS/PAGE_SIZE + 1); + + /* Map pages for this read */ + for (j = 0; j < num_io_pages; j ++) + pmap_kenter_pa(tempva + j * PAGE_SIZE, + img_cur + j * PAGE_SIZE, VM_PROT_ALL); + + pmap_update(pmap_kernel()); + hibernate_block_io(hib, blkctr, read_size, tempva + (img_cur & PAGE_MASK), 0); blkctr += (read_size / DEV_BSIZE); - pmap_kremove(tempva, PAGE_SIZE); - pmap_kremove(tempva + PAGE_SIZE, PAGE_SIZE); + pmap_kremove(tempva, num_io_pages * PAGE_SIZE); + pmap_update(pmap_kernel()); + processed += read_size; img_cur += read_size; } |