fix(tvix/eval): Pop frames even if running op fails
Previously, the VM assumed that if an error was returned from `run()`, the evaluation was "finished" and the state of the VM didn't matter. This used to be a reasonable assumption, but now that we've got `tryEval` around we need to actually make sure that we clean up after ourselves if we're about to return an error. Specifically, if the *last* instruction in an evaluation frame returns an error, we previously wouldn't pop that evaluation frame, which could cause all sorts of bizarre errors based on what happened to be in the stack at the time. This commit splits out a `run_op` method from `VM::run`, and uses that to check the evaluation frame return condition even if the op we're running is about to return an error, and pop the evaluation frame if we're at the last instruction. Change-Id: Ib40649d8915ee1571153cb71e3d76492542fc3d7 Reviewed-on: https://cl.tvl.fyi/c/depot/+/6940 Tested-by: BuildkiteCI Reviewed-by: tazjin <tazjin@tvl.su>
This commit is contained in:
parent
76e0c37b9e
commit
4a058a9c00
3 changed files with 350 additions and 334 deletions
|
@ -1 +1 @@
|
|||
{ w = { success = false; value = false; }; x = { success = true; value = "x"; }; y = { success = false; value = false; }; z = { success = false; value = false; }; }
|
||||
{ v = false; w = { success = false; value = false; }; x = { success = true; value = "x"; }; y = { success = false; value = false; }; z = { success = false; value = false; }; }
|
||||
|
|
|
@ -1,4 +1,5 @@
|
|||
{
|
||||
v = (builtins.tryEval (toString <oink>)).value;
|
||||
w = builtins.tryEval <nope>;
|
||||
x = builtins.tryEval "x";
|
||||
y = builtins.tryEval (assert false; "y");
|
||||
|
|
|
@ -362,6 +362,18 @@ impl<'o> VM<'o> {
|
|||
self.observer
|
||||
.observe_execute_op(self.frame().ip, &op, &self.stack);
|
||||
|
||||
let res = self.run_op(op);
|
||||
|
||||
if self.frame().ip.0 == self.chunk().code.len() {
|
||||
self.frames.pop();
|
||||
return res;
|
||||
} else {
|
||||
res?;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn run_op(&mut self, op: OpCode) -> EvalResult<()> {
|
||||
match op {
|
||||
OpCode::OpConstant(idx) => {
|
||||
let c = self.chunk()[idx].clone();
|
||||
|
@ -673,11 +685,13 @@ impl<'o> VM<'o> {
|
|||
|
||||
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..]),
|
||||
Value::Closure(closure) => {
|
||||
closure.resolve_deferred_upvalues(&self.stack[self.frame().stack_offset..])
|
||||
}
|
||||
|
||||
Value::Thunk(thunk) => thunk
|
||||
.resolve_deferred_upvalues(&self.stack[self.frame().stack_offset..]),
|
||||
Value::Thunk(thunk) => {
|
||||
thunk.resolve_deferred_upvalues(&self.stack[self.frame().stack_offset..])
|
||||
}
|
||||
|
||||
// In functions with "formals" attributes, it is
|
||||
// possible for `OpFinalise` to be called on a
|
||||
|
@ -697,7 +711,8 @@ impl<'o> VM<'o> {
|
|||
panic!("VM bug: attempted to execute data-carrying operand")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn run_attrset(&mut self, count: usize) -> EvalResult<()> {
|
||||
|
|
Loading…
Reference in a new issue