Skip to content

Commit 541298f

Browse files
authored
Fix a subtraction overflow in get_free_pages. (#1261)
The used pages can also be greater than the total pages for the same reason as those in computing `get_available_pages`, and it can also happen if the VM binding disabled GC, in which case we may over-allocate without triggering GC. When it overflows, `get_free_pages` will cause subtraction overflow, and will panic in debug build. We switch to `saturating_sub` so that it will return 0 if overflow happens. It still makes sense. 0 means there is no free pages because we are over-allocating beyond the current heap size set by the GC trigger.
1 parent 68bf1b6 commit 541298f

File tree

1 file changed

+7
-1
lines changed

1 file changed

+7
-1
lines changed

src/plan/global.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -266,6 +266,7 @@ pub trait Plan: 'static + HasSpaces + Sync + Downcast {
266266
// the reserved pages is larger than total pages after the copying GC (the reserved pages after a GC
267267
// may be larger than the reserved pages before a GC, as we may end up using more memory for thread local
268268
// buffers for copy allocators).
269+
// 3. the binding disabled GC, and we end up over-allocating beyond the total pages determined by the GC trigger.
269270
let available_pages = total_pages.saturating_sub(reserved_pages);
270271
trace!(
271272
"Total pages = {}, reserved pages = {}, available pages = {}",
@@ -289,7 +290,12 @@ pub trait Plan: 'static + HasSpaces + Sync + Downcast {
289290
/// Get the number of pages that are NOT used. This is clearly different from available pages.
290291
/// Free pages are unused, but some of them may have been reserved for some reason.
291292
fn get_free_pages(&self) -> usize {
292-
self.get_total_pages() - self.get_used_pages()
293+
let total_pages = self.get_total_pages();
294+
let used_pages = self.get_used_pages();
295+
296+
// It is possible that the used pages is larger than the total pages, so we use saturating
297+
// subtraction. See the comments in `get_available_pages`.
298+
total_pages.saturating_sub(used_pages)
293299
}
294300

295301
/// Return whether last GC was an exhaustive attempt to collect the heap.

0 commit comments

Comments
 (0)