Switch the global list to Weak based pointers, allowing us to actually use rust lifetimes for heaps

This commit is contained in:
2019-06-03 10:12:49 +02:00
parent 84bdcf4b20
commit fa49500139
2 changed files with 31 additions and 37 deletions

View File

@@ -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)); println!("GC POST: {} weaks", self.len());
}
}
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);
}
} }
} }

View File

@@ -267,8 +267,6 @@ 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() { let mut prev_frame: Frame = match machine.frame.pop() {
Some(a) => a, Some(a) => a,
None => return Err("Got no frame... somehow..."), None => return Err("Got no frame... somehow..."),
@@ -276,11 +274,7 @@ fn ireturn(machine: &mut Machine) -> Result<()> {
let result = prev_frame.stack.pop()?; let result = prev_frame.stack.pop()?;
let return_addr: i32 = prev_frame.get(0)?.try_into()?; let return_addr: i32 = prev_frame.get(0)?.try_into()?;
machine.cur_stack().push(result); machine.cur_stack().push(result);
machine.block.seek(return_addr as usize)?; machine.block.seek(return_addr as usize)
}
machine.heap.gc();
Ok(())
} }
#[cfg(feature = "extra:sleep")] #[cfg(feature = "extra:sleep")]