fix(tvix/eval): fix branching on catchable defaults (b/343)
This commit adds Opcode::OpJumpIfCatchable, which can be inserted ahead of most VM operations which expect a boolean on the stack, in order to handle catchables in branching position properly. Other than remembering to patch the jump, no other changes should be required. This commit also fixes b/343 by emitting this new opcode when compiling if-then-else. There are probably other places where we need to do the same thing. Change-Id: I48de3010014c0bbeba15d34fc0d4800e0bb5a1ef Reviewed-on: https://cl.tvl.fyi/c/depot/+/10288 Tested-by: BuildkiteCI Reviewed-by: tazjin <tazjin@tvl.su> Autosubmit: Adam Joseph <adam@westernsemico.com>
This commit is contained in:
parent
e54eeda0ff
commit
9792920f8c
6 changed files with 20 additions and 0 deletions
|
@ -864,6 +864,10 @@ impl Compiler<'_> {
|
||||||
self.compile(slot, node.condition().unwrap());
|
self.compile(slot, node.condition().unwrap());
|
||||||
self.emit_force(&node.condition().unwrap());
|
self.emit_force(&node.condition().unwrap());
|
||||||
|
|
||||||
|
let throw_idx = self.push_op(
|
||||||
|
OpCode::OpJumpIfCatchable(JumpOffset(0)),
|
||||||
|
&node.condition().unwrap(),
|
||||||
|
);
|
||||||
let then_idx = self.push_op(
|
let then_idx = self.push_op(
|
||||||
OpCode::OpJumpIfFalse(JumpOffset(0)),
|
OpCode::OpJumpIfFalse(JumpOffset(0)),
|
||||||
&node.condition().unwrap(),
|
&node.condition().unwrap(),
|
||||||
|
@ -879,6 +883,7 @@ impl Compiler<'_> {
|
||||||
self.compile(slot, node.else_body().unwrap());
|
self.compile(slot, node.else_body().unwrap());
|
||||||
|
|
||||||
self.patch_jump(else_idx); // patch jump *over* else body
|
self.patch_jump(else_idx); // patch jump *over* else body
|
||||||
|
self.patch_jump(throw_idx); // patch jump *over* else body
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Compile `with` expressions by emitting instructions that
|
/// Compile `with` expressions by emitting instructions that
|
||||||
|
@ -1328,6 +1333,7 @@ impl Compiler<'_> {
|
||||||
OpCode::OpJump(n)
|
OpCode::OpJump(n)
|
||||||
| OpCode::OpJumpIfFalse(n)
|
| OpCode::OpJumpIfFalse(n)
|
||||||
| OpCode::OpJumpIfTrue(n)
|
| OpCode::OpJumpIfTrue(n)
|
||||||
|
| OpCode::OpJumpIfCatchable(n)
|
||||||
| OpCode::OpJumpIfNotFound(n)
|
| OpCode::OpJumpIfNotFound(n)
|
||||||
| OpCode::OpJumpIfNoFinaliseRequest(n) => {
|
| OpCode::OpJumpIfNoFinaliseRequest(n) => {
|
||||||
*n = offset;
|
*n = offset;
|
||||||
|
|
|
@ -130,6 +130,12 @@ pub enum OpCode {
|
||||||
/// of the stack is `false`.
|
/// of the stack is `false`.
|
||||||
OpJumpIfFalse(JumpOffset),
|
OpJumpIfFalse(JumpOffset),
|
||||||
|
|
||||||
|
/// Pop one stack item and jump forward in the bytecode
|
||||||
|
/// specified by the number of instructions in its usize
|
||||||
|
/// operand, *if* the value at the top of the stack is a
|
||||||
|
/// Value::Catchable.
|
||||||
|
OpJumpIfCatchable(JumpOffset),
|
||||||
|
|
||||||
/// Jump forward in the bytecode specified by the number of
|
/// Jump forward in the bytecode specified by the number of
|
||||||
/// instructions in its usize operand, *if* the value at the top
|
/// instructions in its usize operand, *if* the value at the top
|
||||||
/// of the stack is the internal value representing a missing
|
/// of the stack is the internal value representing a missing
|
||||||
|
|
|
@ -639,6 +639,7 @@ impl Value {
|
||||||
gen_is!(is_number, Value::Integer(_) | Value::Float(_));
|
gen_is!(is_number, Value::Integer(_) | Value::Float(_));
|
||||||
gen_is!(is_bool, Value::Bool(_));
|
gen_is!(is_bool, Value::Bool(_));
|
||||||
gen_is!(is_attrs, Value::Attrs(_));
|
gen_is!(is_attrs, Value::Attrs(_));
|
||||||
|
gen_is!(is_catchable, Value::Catchable(_));
|
||||||
|
|
||||||
/// Compare `self` against other using (fallible) Nix ordering semantics.
|
/// Compare `self` against other using (fallible) Nix ordering semantics.
|
||||||
///
|
///
|
||||||
|
|
|
@ -562,6 +562,13 @@ impl<'o> VM<'o> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
OpCode::OpJumpIfCatchable(JumpOffset(offset)) => {
|
||||||
|
debug_assert!(offset != 0);
|
||||||
|
if self.stack_peek(0).is_catchable() {
|
||||||
|
frame.ip += offset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
OpCode::OpJumpIfNoFinaliseRequest(JumpOffset(offset)) => {
|
OpCode::OpJumpIfNoFinaliseRequest(JumpOffset(offset)) => {
|
||||||
debug_assert!(offset != 0);
|
debug_assert!(offset != 0);
|
||||||
match self.stack_peek(0) {
|
match self.stack_peek(0) {
|
||||||
|
|
Loading…
Reference in a new issue