diff options
author | Vincent Ambo <mail@tazj.in> | 2022-08-28T14·28+0300 |
---|---|---|
committer | tazjin <tazjin@tvl.su> | 2022-09-06T14·58+0000 |
commit | 025a9a4a0a66c8593cd6b7e4b0f0fa7aea84c353 (patch) | |
tree | e7a6921ebafcdd65c87394f58e9c1614f65eb79c /tvix | |
parent | 1bfe32f4129a1c8627a254fe0fefe18d15339d83 (diff) |
feat(tvix/eval): implement OpFinalise instruction r/4667
This instruction finalises the initialisation of deferred upvalues in closures (and soon, thunks). The compiler does not yet emit this instruction, some more accounting is needed for that. Change-Id: Ic4181b26e19779e206f51e17388559400da5f93a Reviewed-on: https://cl.tvl.fyi/c/depot/+/6337 Tested-by: BuildkiteCI Reviewed-by: sterni <sternenseemann@systemli.org>
Diffstat (limited to 'tvix')
-rw-r--r-- | tvix/eval/src/opcode.rs | 5 | ||||
-rw-r--r-- | tvix/eval/src/value/function.rs | 11 | ||||
-rw-r--r-- | tvix/eval/src/vm.rs | 13 |
3 files changed, 28 insertions, 1 deletions
diff --git a/tvix/eval/src/opcode.rs b/tvix/eval/src/opcode.rs index 12a76d368e4f..45a6337229d2 100644 --- a/tvix/eval/src/opcode.rs +++ b/tvix/eval/src/opcode.rs @@ -32,7 +32,6 @@ pub struct JumpOffset(pub usize); #[derive(Clone, Copy, Debug)] pub struct Count(pub usize); -#[allow(clippy::enum_variant_names)] #[warn(variant_size_differences)] #[derive(Clone, Copy, Debug)] pub enum OpCode { @@ -108,6 +107,10 @@ pub enum OpCode { OpGetUpvalue(UpvalueIdx), OpClosure(ConstantIdx), + /// Finalise initialisation of the upvalues of the value in the + /// given stack index after the scope is fully bound. + OpFinalise(StackIdx), + // The closure and thunk creation instructions have a variable // number of arguments to the instruction, which is represented // here by making their data part of the opcodes. diff --git a/tvix/eval/src/value/function.rs b/tvix/eval/src/value/function.rs index e5db43d58ace..e8301d82da11 100644 --- a/tvix/eval/src/value/function.rs +++ b/tvix/eval/src/value/function.rs @@ -60,4 +60,15 @@ impl Closure { pub fn push_upvalue(&self, value: Value) { self.0.borrow_mut().upvalues.push(value) } + + /// Resolve the deferred upvalues in the closure from a slice of + /// the current stack, using the indices stored in the deferred + /// values. + pub fn resolve_deferred_upvalues(&self, stack: &[Value]) { + for upvalue in self.0.borrow_mut().upvalues.iter_mut() { + if let Value::DeferredUpvalue(idx) = upvalue { + *upvalue = stack[idx.0].clone(); + } + } + } } diff --git a/tvix/eval/src/vm.rs b/tvix/eval/src/vm.rs index d4eb0657251a..1aaba9b0573e 100644 --- a/tvix/eval/src/vm.rs +++ b/tvix/eval/src/vm.rs @@ -435,6 +435,19 @@ impl VM { } } + OpCode::OpFinalise(StackIdx(idx)) => { + match &self.stack[self.frame().stack_offset + idx] { + Value::Closure(closure) => closure + .resolve_deferred_upvalues(&self.stack[self.frame().stack_offset..]), + + v => { + #[cfg(feature = "disassembler")] + drop(tracer); + panic!("compiler error: invalid finaliser value: {}", v); + } + } + } + // Data-carrying operands should never be executed, // that is a critical error in the VM. OpCode::DataLocalIdx(_) |