forked from mmtk/mmtk-ruby
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcollection.rs
88 lines (78 loc) · 2.89 KB
/
collection.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
use crate::abi::GCThreadTLS;
use crate::api::RubyMutator;
use crate::{mmtk, upcalls, Ruby};
use mmtk::memory_manager;
use mmtk::scheduler::*;
use mmtk::util::{VMMutatorThread, VMThread, VMWorkerThread};
use mmtk::vm::{Collection, GCThreadContext};
use std::sync::atomic::Ordering;
use std::thread;
pub struct VMCollection {}
impl Collection<Ruby> for VMCollection {
fn is_collection_enabled() -> bool {
crate::BINDING_FAST.gc_enabled.load(Ordering::Relaxed)
}
fn stop_all_mutators<F>(tls: VMWorkerThread, mut mutator_visitor: F)
where
F: FnMut(&'static mut mmtk::Mutator<Ruby>),
{
(upcalls().stop_the_world)(tls);
crate::binding().ppp_registry.pin_ppp_children(tls);
(upcalls().get_mutators)(
Self::notify_mutator_ready::<F>,
&mut mutator_visitor as *mut F as *mut _,
);
}
fn resume_mutators(tls: VMWorkerThread) {
(upcalls().resume_mutators)(tls);
}
fn block_for_gc(tls: VMMutatorThread) {
(upcalls().block_for_gc)(tls);
}
fn spawn_gc_thread(_tls: VMThread, ctx: GCThreadContext<Ruby>) {
let join_handle = match ctx {
GCThreadContext::Worker(mut worker) => thread::Builder::new()
.name("MMTk Worker Thread".to_string())
.spawn(move || {
let ordinal = worker.ordinal;
debug!(
"Hello! This is MMTk Worker Thread running! ordinal: {}",
ordinal
);
crate::register_gc_thread(thread::current().id());
let ptr_worker = &mut *worker as *mut GCWorker<Ruby>;
let gc_thread_tls =
Box::into_raw(Box::new(GCThreadTLS::for_worker(ptr_worker)));
(upcalls().init_gc_worker_thread)(gc_thread_tls);
memory_manager::start_worker(
mmtk(),
GCThreadTLS::to_vwt(gc_thread_tls),
worker,
);
debug!(
"An MMTk Worker Thread is quitting. Good bye! ordinal: {}",
ordinal
);
crate::unregister_gc_thread(thread::current().id());
})
.unwrap(),
};
{
let mut handles = crate::binding().gc_thread_join_handles.lock().unwrap();
handles.push(join_handle);
}
}
fn vm_live_bytes() -> usize {
(upcalls().vm_live_bytes)()
}
}
impl VMCollection {
extern "C" fn notify_mutator_ready<F>(mutator_ptr: *mut RubyMutator, data: *mut libc::c_void)
where
F: FnMut(&'static mut mmtk::Mutator<Ruby>),
{
let mutator = unsafe { &mut *mutator_ptr };
let mutator_visitor = unsafe { &mut *(data as *mut F) };
mutator_visitor(mutator);
}
}