diff options
author | Mike Larkin <mlarkin@cvs.openbsd.org> | 2012-03-26 16:15:43 +0000 |
---|---|---|
committer | Mike Larkin <mlarkin@cvs.openbsd.org> | 2012-03-26 16:15:43 +0000 |
commit | 4ac206d01035fca15c436fdc6ace3f9ba322ddc0 (patch) | |
tree | ef3186fb4214eea54d3a3786af50b3560d1c8aac /sys/kern | |
parent | e5b18d68d720b5e8167a56973fde63224f035621 (diff) |
Fix an integer math error when using the result of uvm_page_rle, and
at the same time increase said function's max RLE page count return value.
Add hooks in the right places to call the hibernate suspend and resume
routines, so that we can enable hibernation with a HIBERNATE option
line in GENERIC and appropriate acpi.c goo.
discussed on and off with deraadt@ over the past few months
Diffstat (limited to 'sys/kern')
-rw-r--r-- | sys/kern/subr_hibernate.c | 25 |
1 files changed, 12 insertions, 13 deletions
diff --git a/sys/kern/subr_hibernate.c b/sys/kern/subr_hibernate.c index 4c3c9c7d06a..45ed48f1867 100644 --- a/sys/kern/subr_hibernate.c +++ b/sys/kern/subr_hibernate.c @@ -1,4 +1,4 @@ -/* $OpenBSD: subr_hibernate.c,v 1.32 2011/11/29 05:21:08 deraadt Exp $ */ +/* $OpenBSD: subr_hibernate.c,v 1.33 2012/03/26 16:15:42 mlarkin Exp $ */ /* * Copyright (c) 2011 Ariane van der Steldt <ariane@stack.nl> @@ -565,10 +565,10 @@ uvm_pmr_free_piglet(vaddr_t va, vsize_t sz) * Physmem RLE compression support. * * Given a physical page address, it will return the number of pages - * starting at the address, that are free. Clamps to a max of 255 pages. - * Returns 0 if the page at addr is not free. + * starting at the address, that are free. Clamps to the number of pages in + * HIBERNATE_CHUNK_SIZE. Returns 0 if the page at addr is not free. */ -u_char +int uvm_page_rle(paddr_t addr) { struct vm_page *pg, *pg_end; @@ -592,7 +592,7 @@ uvm_page_rle(paddr_t addr) for (pg_end = pg; pg_end <= vmp->lastpg && (pg_end->pg_flags & PQ_FREE) == PQ_FREE; pg_end++) ; - return max(pg_end - pg, 255); + return min((pg_end - pg), HIBERNATE_CHUNK_SIZE/PAGE_SIZE); } /* @@ -722,8 +722,7 @@ void hibernate_inflate(union hibernate_info *hiber_info, paddr_t dest, paddr_t src, size_t size) { - int i; - u_char rle; + int i, rle; hibernate_state->hib_stream.next_in = (char *)src; hibernate_state->hib_stream.avail_in = size; @@ -1190,11 +1189,11 @@ int hibernate_write_chunks(union hibernate_info *hiber_info) { paddr_t range_base, range_end, inaddr, temp_inaddr; - size_t nblocks, out_remaining, used, offset = 0; + size_t nblocks, out_remaining, used; struct hibernate_disk_chunk *chunks; vaddr_t hibernate_io_page = hiber_info->piglet_va + PAGE_SIZE; - daddr_t blkctr = hiber_info->image_offset; - int i; + daddr_t blkctr = hiber_info->image_offset, offset = 0; + int i, rle; hiber_info->chunk_ctr = 0; @@ -1263,7 +1262,6 @@ hibernate_write_chunks(union hibernate_info *hiber_info) while (inaddr < range_end) { out_remaining = PAGE_SIZE; while (out_remaining > 0 && inaddr < range_end) { - u_char rle; /* * Adjust for regions that are not evenly @@ -1615,8 +1613,8 @@ hibernate_read_chunks(union hibernate_info *hib_info, paddr_t pig_start, */ for (i = 0; i < nchunks; i++) { if (chunks[i].end <= pig_start || chunks[i].base >= pig_end) { - ochunks[nochunks] = (u_int8_t)i; - fchunks[nfchunks] = (u_int8_t)i; + ochunks[nochunks] = i; + fchunks[nfchunks] = i; nochunks++; nfchunks++; chunks[i].flags |= HIBERNATE_CHUNK_USED; @@ -1681,6 +1679,7 @@ hibernate_read_chunks(union hibernate_info *hib_info, paddr_t pig_start, piglet_cur = piglet_base; npchunks = 0; j = i; + while (copy_start < copy_end && j < nochunks) { piglet_cur += chunks[ochunks[j]].compressed_size; pchunks[npchunks] = ochunks[j]; |