refactor(tvix/eval): encapsulate internal mutability within Closure
This is required to efficiently construct the upvalue array at runtime, as there are situations where during Closure construction multiple things already have a reference to the closure (e.g. a self-reference). Change-Id: I35263b845fdc695dc873de489f5168d39b370f6a Reviewed-on: https://cl.tvl.fyi/c/depot/+/6312 Tested-by: BuildkiteCI Reviewed-by: sterni <sternenseemann@systemli.org>
This commit is contained in:
parent
e646d5170c
commit
3089e46eb1
3 changed files with 47 additions and 25 deletions
|
@ -87,8 +87,7 @@ impl Compiler {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn chunk(&mut self) -> &mut Chunk {
|
fn chunk(&mut self) -> &mut Chunk {
|
||||||
Rc::<Chunk>::get_mut(self.context_mut().lambda.chunk())
|
&mut self.context_mut().lambda.chunk
|
||||||
.expect("compiler flaw: long-lived chunk reference")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn scope(&self) -> &Scope {
|
fn scope(&self) -> &Scope {
|
||||||
|
|
|
@ -1,12 +1,15 @@
|
||||||
//! This module implements the runtime representation of functions.
|
//! This module implements the runtime representation of functions.
|
||||||
use std::rc::Rc;
|
use std::{
|
||||||
|
cell::{Ref, RefCell},
|
||||||
|
rc::Rc,
|
||||||
|
};
|
||||||
|
|
||||||
use crate::{chunk::Chunk, Value};
|
use crate::{chunk::Chunk, opcode::UpvalueIdx, Value};
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct Lambda {
|
pub struct Lambda {
|
||||||
// name: Option<NixString>,
|
// name: Option<NixString>,
|
||||||
pub(crate) chunk: Rc<Chunk>,
|
pub(crate) chunk: Chunk,
|
||||||
pub(crate) upvalue_count: usize,
|
pub(crate) upvalue_count: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,22 +22,42 @@ impl Lambda {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn chunk(&mut self) -> &mut Rc<Chunk> {
|
pub fn chunk(&mut self) -> &mut Chunk {
|
||||||
&mut self.chunk
|
&mut self.chunk
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct Closure {
|
pub struct InnerClosure {
|
||||||
pub lambda: Lambda,
|
pub lambda: Lambda,
|
||||||
pub upvalues: Vec<Value>,
|
pub upvalues: Vec<Value>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[repr(transparent)]
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct Closure(Rc<RefCell<InnerClosure>>);
|
||||||
|
|
||||||
impl Closure {
|
impl Closure {
|
||||||
pub fn new(lambda: Lambda) -> Self {
|
pub fn new(lambda: Lambda) -> Self {
|
||||||
Closure {
|
Closure(Rc::new(RefCell::new(InnerClosure {
|
||||||
upvalues: Vec::with_capacity(lambda.upvalue_count),
|
upvalues: Vec::with_capacity(lambda.upvalue_count),
|
||||||
lambda,
|
lambda,
|
||||||
}
|
})))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn chunk(&self) -> Ref<'_, Chunk> {
|
||||||
|
Ref::map(self.0.borrow(), |c| &c.lambda.chunk)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn upvalue(&self, idx: UpvalueIdx) -> Ref<'_, Value> {
|
||||||
|
Ref::map(self.0.borrow(), |c| &c.upvalues[idx.0])
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn upvalue_count(&self) -> usize {
|
||||||
|
self.0.borrow().lambda.upvalue_count
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn push_upvalue(&self, value: Value) {
|
||||||
|
self.0.borrow_mut().upvalues.push(value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
//! This module implements the virtual (or abstract) machine that runs
|
//! This module implements the virtual (or abstract) machine that runs
|
||||||
//! Tvix bytecode.
|
//! Tvix bytecode.
|
||||||
|
|
||||||
use std::rc::Rc;
|
use std::{cell::Ref, rc::Rc};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
chunk::Chunk,
|
chunk::Chunk,
|
||||||
errors::{ErrorKind, EvalResult},
|
errors::{ErrorKind, EvalResult},
|
||||||
opcode::{Count, JumpOffset, OpCode, StackIdx, UpvalueIdx},
|
opcode::{Count, JumpOffset, OpCode, StackIdx},
|
||||||
value::{Closure, Lambda, NixAttrs, NixList, Value},
|
value::{Closure, Lambda, NixAttrs, NixList, Value},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -85,8 +85,8 @@ impl VM {
|
||||||
&self.frames[self.frames.len() - 1]
|
&self.frames[self.frames.len() - 1]
|
||||||
}
|
}
|
||||||
|
|
||||||
fn chunk(&self) -> &Chunk {
|
fn chunk(&self) -> Ref<'_, Chunk> {
|
||||||
&self.frame().closure.lambda.chunk
|
self.frame().closure.chunk()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn frame_mut(&mut self) -> &mut CallFrame {
|
fn frame_mut(&mut self) -> &mut CallFrame {
|
||||||
|
@ -353,35 +353,35 @@ impl VM {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
OpCode::OpGetUpvalue(UpvalueIdx(upv_idx)) => {
|
OpCode::OpGetUpvalue(upv_idx) => {
|
||||||
let value = self.frame().closure.upvalues[upv_idx].clone();
|
let value = self.frame().closure.upvalue(upv_idx).clone();
|
||||||
self.push(value);
|
self.push(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
OpCode::OpClosure(idx) => {
|
OpCode::OpClosure(idx) => {
|
||||||
let mut closure = self.chunk().constant(idx).clone().to_closure()?;
|
let closure = self.chunk().constant(idx).clone().to_closure()?;
|
||||||
|
|
||||||
debug_assert!(
|
debug_assert!(
|
||||||
closure.lambda.upvalue_count > 0,
|
closure.upvalue_count() > 0,
|
||||||
"OpClosure should not be called for plain lambdas"
|
"OpClosure should not be called for plain lambdas"
|
||||||
);
|
);
|
||||||
|
|
||||||
for _ in 0..closure.lambda.upvalue_count {
|
for _ in 0..closure.upvalue_count() {
|
||||||
match self.inc_ip() {
|
match self.inc_ip() {
|
||||||
OpCode::DataLocalIdx(StackIdx(local_idx)) => {
|
OpCode::DataLocalIdx(StackIdx(local_idx)) => {
|
||||||
let idx = self.frame().stack_offset + local_idx;
|
let idx = self.frame().stack_offset + local_idx;
|
||||||
closure.upvalues.push(self.stack[idx].clone());
|
closure.push_upvalue(self.stack[idx].clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
OpCode::DataUpvalueIdx(UpvalueIdx(upv_idx)) => {
|
OpCode::DataUpvalueIdx(upv_idx) => {
|
||||||
closure
|
closure.push_upvalue(self.frame().closure.upvalue(upv_idx).clone());
|
||||||
.upvalues
|
|
||||||
.push(self.frame().closure.upvalues[upv_idx].clone());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
OpCode::DataDynamicIdx(ident_idx) => {
|
OpCode::DataDynamicIdx(ident_idx) => {
|
||||||
let ident = self.chunk().constant(ident_idx).as_str()?;
|
let chunk = self.chunk();
|
||||||
closure.upvalues.push(self.resolve_with(ident)?);
|
let ident = chunk.constant(ident_idx).as_str()?.to_string();
|
||||||
|
drop(chunk); // some lifetime trickery due to cell::Ref
|
||||||
|
closure.push_upvalue(self.resolve_with(&ident)?);
|
||||||
}
|
}
|
||||||
|
|
||||||
_ => panic!("compiler error: missing closure operand"),
|
_ => panic!("compiler error: missing closure operand"),
|
||||||
|
|
Loading…
Reference in a new issue