Switch the global list to Weak based pointers, allowing us to actually use rust lifetimes for heaps
This commit is contained in:
46
src/heap.rs
46
src/heap.rs
@@ -1,12 +1,14 @@
|
|||||||
use value::Value;
|
use value::Value;
|
||||||
use std::rc::Rc;
|
use std::rc::{Rc, Weak};
|
||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
|
|
||||||
pub type Heap = Rc<RefCell<Vec<Value>>>;
|
pub type HeapInner = RefCell<Vec<Value>>;
|
||||||
|
pub type Heap = Rc<HeapInner>;
|
||||||
|
pub type WeakHeap = Weak<HeapInner>;
|
||||||
|
|
||||||
#[derive(Debug,Default)]
|
#[derive(Debug,Default)]
|
||||||
pub struct Heaps {
|
pub struct Heaps {
|
||||||
pub heaps: Vec<Heap>
|
pub heaps: Vec<WeakHeap>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Heaps {
|
impl Heaps {
|
||||||
@@ -20,13 +22,24 @@ impl Heaps {
|
|||||||
self.heaps.len()
|
self.heaps.len()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get(&mut self, i: usize) -> Heap {
|
pub fn get(&mut self, i: usize) -> Option<Heap> {
|
||||||
self.heaps[i].clone()
|
if i >= self.len() {
|
||||||
|
return None
|
||||||
|
}
|
||||||
|
|
||||||
|
let weak_ref = self.heaps[i].clone();
|
||||||
|
let possibly_strong = weak_ref.upgrade();
|
||||||
|
|
||||||
|
if possibly_strong.is_none() {
|
||||||
|
self.heaps.remove(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
possibly_strong
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn alloc(&mut self, size: usize) -> Heap {
|
pub fn alloc(&mut self, size: usize) -> Heap {
|
||||||
let heap = Rc::new(RefCell::new(vec![Value::Int(0); size]));
|
let heap = Rc::new(RefCell::new(vec![Value::Int(0); size]));
|
||||||
self.heaps.push(heap.clone());
|
self.heaps.push(Rc::downgrade(&heap));
|
||||||
heap
|
heap
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -35,23 +48,10 @@ impl Heaps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn gc(&mut self) {
|
pub fn gc(&mut self) {
|
||||||
let pre_size = self.len();
|
println!("GC PRE : {} weaks", self.len());
|
||||||
|
for i in 0..self.len() {
|
||||||
if cfg!(feature = "debug:gc") {
|
println!("\theap: {} = {:?}", i, self.get(i));
|
||||||
for (i, heap) in self.heaps.iter().enumerate() {
|
|
||||||
eprintln!("GC heap {} ref = {}", i, Rc::strong_count(heap));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
self.heaps = self.heaps.iter()
|
|
||||||
.filter(|e| Rc::strong_count(e) > 1)
|
|
||||||
.map(|e| Rc::clone(e))
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let post_size = self.len();
|
|
||||||
|
|
||||||
if cfg!(feature = "debug:gc") {
|
|
||||||
eprintln!("GC reaped {} heaps", pre_size - post_size);
|
|
||||||
}
|
}
|
||||||
|
println!("GC POST: {} weaks", self.len());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
22
src/ops.rs
22
src/ops.rs
@@ -267,20 +267,14 @@ fn invokevirtual(machine: &mut Machine) -> Result<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn ireturn(machine: &mut Machine) -> Result<()> {
|
fn ireturn(machine: &mut Machine) -> Result<()> {
|
||||||
// Lifetime for prev frame, allows gc to reap into the frame
|
let mut prev_frame: Frame = match machine.frame.pop() {
|
||||||
{
|
Some(a) => a,
|
||||||
let mut prev_frame: Frame = match machine.frame.pop() {
|
None => return Err("Got no frame... somehow..."),
|
||||||
Some(a) => a,
|
};
|
||||||
None => return Err("Got no frame... somehow..."),
|
let result = prev_frame.stack.pop()?;
|
||||||
};
|
let return_addr: i32 = prev_frame.get(0)?.try_into()?;
|
||||||
let result = prev_frame.stack.pop()?;
|
machine.cur_stack().push(result);
|
||||||
let return_addr: i32 = prev_frame.get(0)?.try_into()?;
|
machine.block.seek(return_addr as usize)
|
||||||
machine.cur_stack().push(result);
|
|
||||||
machine.block.seek(return_addr as usize)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
machine.heap.gc();
|
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "extra:sleep")]
|
#[cfg(feature = "extra:sleep")]
|
||||||
|
|||||||
Reference in New Issue
Block a user