From 8431b7af4677241a41dacf6aae45994b6fd9e0e8 Mon Sep 17 00:00:00 2001 From: Yi Lin Date: Mon, 19 Dec 2022 14:57:37 +1100 Subject: [PATCH 1/2] Add Plan.end_of_gc. Generational plans set next_heap_full_size in end_of_gc --- src/plan/generational/copying/global.rs | 7 ++----- src/plan/generational/immix/global.rs | 7 ++----- src/plan/global.rs | 7 ++++++- src/scheduler/gc_work.rs | 5 +++++ 4 files changed, 15 insertions(+), 11 deletions(-) diff --git a/src/plan/generational/copying/global.rs b/src/plan/generational/copying/global.rs index b529567751..65f6098467 100644 --- a/src/plan/generational/copying/global.rs +++ b/src/plan/generational/copying/global.rs @@ -123,12 +123,9 @@ impl Plan for GenCopy { if full_heap { self.fromspace().release(); } + } - // TODO: Refactor so that we set the next_gc_full_heap in gen.release(). Currently have to fight with Rust borrow checker - // NOTE: We have to take care that the `Gen::should_next_gc_be_full_heap()` function is - // called _after_ all spaces have been released (including ones in `gen`) as otherwise we - // may get incorrect results since the function uses values such as available pages that - // will change dependant on which spaces have been released + fn end_of_gc(&mut self, _tls: VMWorkerThread) { self.gen .set_next_gc_full_heap(Gen::should_next_gc_be_full_heap(self)); } diff --git a/src/plan/generational/immix/global.rs b/src/plan/generational/immix/global.rs index 1abb4f0a06..2b9b81a467 100644 --- a/src/plan/generational/immix/global.rs +++ b/src/plan/generational/immix/global.rs @@ -165,12 +165,9 @@ impl Plan for GenImmix { } self.last_gc_was_full_heap .store(full_heap, Ordering::Relaxed); + } - // TODO: Refactor so that we set the next_gc_full_heap in gen.release(). Currently have to fight with Rust borrow checker - // NOTE: We have to take care that the `Gen::should_next_gc_be_full_heap()` function is - // called _after_ all spaces have been released (including ones in `gen`) as otherwise we - // may get incorrect results since the function uses values such as available pages that - // will change dependant on which spaces have been released + fn end_of_gc(&mut self, _tls: VMWorkerThread) { self.gen .set_next_gc_full_heap(Gen::should_next_gc_be_full_heap(self)); } diff --git a/src/plan/global.rs b/src/plan/global.rs index 4688148599..d7ec5303d3 100644 --- a/src/plan/global.rs +++ b/src/plan/global.rs @@ -228,10 +228,15 @@ pub trait Plan: 'static + Sync + Downcast { /// per-worker preparation. This method is invoked once per worker by the worker thread passed as the argument. fn prepare_worker(&self, _worker: &mut GCWorker) {} - /// Release the plan after a GC. This is invoked at the end of a GC when most GC work is finished. + /// Release the plan after transitive closure. A plan can implement this method to call each policy's release, + /// or create any work packet that should be done in release. /// This is invoked once per GC by one worker thread. 'tls' is the worker thread that executes this method. fn release(&mut self, tls: VMWorkerThread); + /// Inform the plan about the end of a GC. It is guaranteed that there is no further work for this GC. + /// This is invoked once per GC by one worker thread. 'tls' is the worker thread that exectues this method. + fn end_of_gc(&mut self, _tls: VMWorkerThread) {} + /// Ask the plan if they would trigger a GC. If MMTk is in charge of triggering GCs, this method is called /// periodically during allocation. However, MMTk may delegate the GC triggering decision to the runtime, /// in which case, this method may not be called. This method returns true to trigger a collection. diff --git a/src/scheduler/gc_work.rs b/src/scheduler/gc_work.rs index 4e1ad6afd4..e9be0e05b5 100644 --- a/src/scheduler/gc_work.rs +++ b/src/scheduler/gc_work.rs @@ -223,6 +223,11 @@ impl GCWork for EndOfGC { fn do_work(&mut self, worker: &mut GCWorker, mmtk: &'static MMTK) { info!("End of GC"); + // We assume this is the only running work packet that accesses plan at the point of execution + #[allow(clippy::cast_ref_to_mut)] + let plan_mut: &mut dyn Plan = unsafe { &mut *(&*mmtk.plan as *const _ as *mut _) }; + plan_mut.end_of_gc(worker.tls); + #[cfg(feature = "extreme_assertions")] if crate::util::edge_logger::should_check_duplicate_edges(&*mmtk.plan) { // reset the logging info at the end of each GC From c34c6b4b804493460d50e0935d17bd2232060ca9 Mon Sep 17 00:00:00 2001 From: Yi Lin Date: Thu, 22 Dec 2022 08:05:20 +1100 Subject: [PATCH 2/2] Fix typo --- src/plan/global.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/plan/global.rs b/src/plan/global.rs index d7ec5303d3..6ad5a5cf98 100644 --- a/src/plan/global.rs +++ b/src/plan/global.rs @@ -221,7 +221,7 @@ pub trait Plan: 'static + Sync + Downcast { } /// Prepare the plan before a GC. This is invoked in an initial step in the GC. - /// This is invoked once per GC by one worker thread. 'tls' is the worker thread that executes this method. + /// This is invoked once per GC by one worker thread. `tls` is the worker thread that executes this method. fn prepare(&mut self, tls: VMWorkerThread); /// Prepare a worker for a GC. Each worker has its own prepare method. This hook is for plan-specific @@ -230,11 +230,11 @@ pub trait Plan: 'static + Sync + Downcast { /// Release the plan after transitive closure. A plan can implement this method to call each policy's release, /// or create any work packet that should be done in release. - /// This is invoked once per GC by one worker thread. 'tls' is the worker thread that executes this method. + /// This is invoked once per GC by one worker thread. `tls` is the worker thread that executes this method. fn release(&mut self, tls: VMWorkerThread); /// Inform the plan about the end of a GC. It is guaranteed that there is no further work for this GC. - /// This is invoked once per GC by one worker thread. 'tls' is the worker thread that exectues this method. + /// This is invoked once per GC by one worker thread. `tls` is the worker thread that executes this method. fn end_of_gc(&mut self, _tls: VMWorkerThread) {} /// Ask the plan if they would trigger a GC. If MMTk is in charge of triggering GCs, this method is called