2025-08-27 17:06:46 +09:00
|
|
|
//! GC hook abstractions for switchable runtime (Phase 10.4 preparation)
|
|
|
|
|
//!
|
|
|
|
|
//! Minimal, no-alloc, no-type-coupling interfaces that VM can call.
|
|
|
|
|
//! Default implementation is a no-op. Real collectors can plug later.
|
|
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy)]
|
2025-09-17 07:43:07 +09:00
|
|
|
pub enum BarrierKind {
|
|
|
|
|
Read,
|
|
|
|
|
Write,
|
|
|
|
|
}
|
2025-08-27 17:06:46 +09:00
|
|
|
|
|
|
|
|
/// GC hooks that execution engines may call at key points.
|
|
|
|
|
/// Implementations must be Send + Sync for multi-thread preparation.
|
2025-09-17 20:33:19 +09:00
|
|
|
pub trait GcHooks: Send + Sync + std::any::Any {
|
2025-08-27 17:06:46 +09:00
|
|
|
/// Safe point for cooperative GC (e.g., poll or yield).
|
|
|
|
|
fn safepoint(&self) {}
|
|
|
|
|
/// Memory barrier hint for loads/stores.
|
|
|
|
|
fn barrier(&self, _kind: BarrierKind) {}
|
2025-09-17 20:33:19 +09:00
|
|
|
/// Allocation accounting (bytes are best-effort; may be 0 when unknown)
|
|
|
|
|
fn alloc(&self, _bytes: u64) {}
|
2025-08-27 17:06:46 +09:00
|
|
|
/// Optional counters snapshot for diagnostics. Default: None.
|
2025-09-17 07:43:07 +09:00
|
|
|
fn snapshot_counters(&self) -> Option<(u64, u64, u64)> {
|
|
|
|
|
None
|
|
|
|
|
}
|
2025-08-27 17:06:46 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Default no-op hooks.
|
|
|
|
|
pub struct NullGc;
|
|
|
|
|
|
|
|
|
|
impl GcHooks for NullGc {}
|
|
|
|
|
|
2025-09-17 20:33:19 +09:00
|
|
|
/// CountingGc is now a thin wrapper around the unified GcController.
|
2025-08-27 17:06:46 +09:00
|
|
|
pub struct CountingGc {
|
2025-09-17 20:33:19 +09:00
|
|
|
inner: crate::runtime::gc_controller::GcController,
|
2025-08-27 17:06:46 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl CountingGc {
|
2025-09-24 23:27:59 +09:00
|
|
|
pub fn new() -> Self { Self::new_with_mode(crate::runtime::gc_mode::GcMode::RcCycle) }
|
|
|
|
|
pub fn new_with_mode(mode: crate::runtime::gc_mode::GcMode) -> Self {
|
|
|
|
|
Self { inner: crate::runtime::gc_controller::GcController::new(mode) }
|
2025-08-27 17:06:46 +09:00
|
|
|
}
|
|
|
|
|
pub fn snapshot(&self) -> (u64, u64, u64) {
|
2025-09-17 20:33:19 +09:00
|
|
|
self.inner.snapshot()
|
2025-08-27 17:06:46 +09:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl GcHooks for CountingGc {
|
|
|
|
|
fn safepoint(&self) {
|
2025-09-17 20:33:19 +09:00
|
|
|
self.inner.safepoint();
|
2025-08-27 17:06:46 +09:00
|
|
|
}
|
|
|
|
|
fn barrier(&self, kind: BarrierKind) {
|
2025-09-17 20:33:19 +09:00
|
|
|
self.inner.barrier(kind);
|
|
|
|
|
}
|
|
|
|
|
fn alloc(&self, bytes: u64) {
|
|
|
|
|
self.inner.alloc(bytes);
|
2025-08-27 17:06:46 +09:00
|
|
|
}
|
|
|
|
|
fn snapshot_counters(&self) -> Option<(u64, u64, u64)> {
|
2025-09-17 20:33:19 +09:00
|
|
|
Some(self.inner.snapshot())
|
2025-08-27 17:06:46 +09:00
|
|
|
}
|
|
|
|
|
}
|