refactor(tvix/eval): flatten call stack of VM using generators
Warning: This is probably the biggest refactor in tvix-eval history, so far. This replaces all instances of trampolines and recursion during evaluation of the VM loop with generators. A generator is an asynchronous function that can be suspended to yield a message (in our case, vm::generators::GeneratorRequest) and receive a response (vm::generators::GeneratorResponsee). The `genawaiter` crate provides an interpreter for generators that can drive their execution and lets us move control flow between the VM and suspended generators. To do this, massive changes have occured basically everywhere in the code. On a high-level: 1. The VM is now organised around a frame stack. A frame is either a call frame (execution of Tvix bytecode) or a generator frame (a running or suspended generator). The VM has an outer loop that pops a frame off the frame stack, and then enters an inner loop either driving the execution of the bytecode or the execution of a generator. Both types of frames have several branches that can result in the frame re-enqueuing itself, and enqueuing some other work (in the form of a different frame) on top of itself. The VM will eventually resume the frame when everything "above" it has been suspended. In this way, the VM's new frame stack takes over much of the work that was previously achieved by recursion. 2. All methods previously taking a VM have been refactored into async functions that instead emit/receive generator messages for communication with the VM. Notably, this includes *all* builtins. This has had some other effects: - Some test have been removed or commented out, either because they tested code that was mostly already dead (nix_eq) or because they now require generator scaffolding which we do not have in place for tests (yet). - Because generator functions are technically async (though no async IO is involved), we lose the ability to use much of the Rust standard library e.g. in builtins. This has led to many algorithms being unrolled into iterative versions instead of iterator combinations, and things like sorting had to be implemented from scratch. - Many call sites that previously saw a `Result<..., ErrorKind>` bubble up now only see the result value, as the error handling is encapsulated within the generator loop. This reduces number of places inside of builtin implementations where error context can be attached to calls that can fail. Currently what we gain in this tradeoff is significantly more detailed span information (which we still need to bubble up, this commit does not change the error display). We'll need to do some analysis later of how useful the errors turn out to be and potentially introduce some methods for attaching context to a generator frame again. This change is very difficult to do in stages, as it is very much an "all or nothing" change that affects huge parts of the codebase. I've tried to isolate changes that can be isolated into the parent CLs of this one, but this change is still quite difficult to wrap one's mind and I'm available to discuss it and explain things to any reviewer. Fixes: b/238, b/237, b/251 and potentially others. Change-Id: I39244163ff5bbecd169fe7b274df19262b515699 Reviewed-on: https://cl.tvl.fyi/c/depot/+/8104 Reviewed-by: raitobezarius <tvl@lahfa.xyz> Reviewed-by: Adam Joseph <adam@westernsemico.com> Tested-by: BuildkiteCI
This commit is contained in:
parent
cbb4137dc0
commit
025c67bf4d
24 changed files with 2449 additions and 2626 deletions
|
@ -5,7 +5,8 @@ use std::cell::RefCell;
|
|||
use std::collections::{btree_map, BTreeSet};
|
||||
use std::rc::Rc;
|
||||
use tvix_eval::builtin_macros::builtins;
|
||||
use tvix_eval::{AddContext, CoercionKind, ErrorKind, NixAttrs, NixList, Value, VM};
|
||||
use tvix_eval::generators::{self, GenCo};
|
||||
use tvix_eval::{AddContext, CoercionKind, ErrorKind, NixAttrs, NixList, Value};
|
||||
|
||||
use crate::errors::Error;
|
||||
use crate::known_paths::{KnownPaths, PathKind, PathName};
|
||||
|
@ -17,13 +18,17 @@ const IGNORE_NULLS: &str = "__ignoreNulls";
|
|||
/// Helper function for populating the `drv.outputs` field from a
|
||||
/// manually specified set of outputs, instead of the default
|
||||
/// `outputs`.
|
||||
fn populate_outputs(vm: &mut VM, drv: &mut Derivation, outputs: NixList) -> Result<(), ErrorKind> {
|
||||
async fn populate_outputs(
|
||||
co: &GenCo,
|
||||
drv: &mut Derivation,
|
||||
outputs: NixList,
|
||||
) -> Result<(), ErrorKind> {
|
||||
// Remove the original default `out` output.
|
||||
drv.outputs.clear();
|
||||
|
||||
for output in outputs {
|
||||
let output_name = output
|
||||
.force(vm)?
|
||||
let output_name = generators::request_force(co, output)
|
||||
.await
|
||||
.to_str()
|
||||
.context("determining output name")?;
|
||||
|
||||
|
@ -144,9 +149,9 @@ fn populate_output_configuration(
|
|||
/// Handles derivation parameters which are not just forwarded to
|
||||
/// the environment. The return value indicates whether the
|
||||
/// parameter should be included in the environment.
|
||||
fn handle_derivation_parameters(
|
||||
async fn handle_derivation_parameters(
|
||||
drv: &mut Derivation,
|
||||
vm: &mut VM,
|
||||
co: &GenCo,
|
||||
name: &str,
|
||||
value: &Value,
|
||||
val_str: &str,
|
||||
|
@ -158,11 +163,7 @@ fn handle_derivation_parameters(
|
|||
"args" => {
|
||||
let args = value.to_list()?;
|
||||
for arg in args {
|
||||
drv.arguments.push(strong_coerce_to_string(
|
||||
vm,
|
||||
&arg,
|
||||
"handling command-line builder arguments",
|
||||
)?);
|
||||
drv.arguments.push(strong_coerce_to_string(co, arg).await?);
|
||||
}
|
||||
|
||||
// The arguments do not appear in the environment.
|
||||
|
@ -176,7 +177,7 @@ fn handle_derivation_parameters(
|
|||
.context("looking at the `outputs` parameter of the derivation")?;
|
||||
|
||||
drv.outputs.clear();
|
||||
populate_outputs(vm, drv, outputs)?;
|
||||
populate_outputs(co, drv, outputs).await?;
|
||||
}
|
||||
|
||||
"builder" => {
|
||||
|
@ -193,22 +194,20 @@ fn handle_derivation_parameters(
|
|||
Ok(true)
|
||||
}
|
||||
|
||||
fn strong_coerce_to_string(vm: &mut VM, val: &Value, ctx: &str) -> Result<String, ErrorKind> {
|
||||
Ok(val
|
||||
.force(vm)
|
||||
.context(ctx)?
|
||||
.coerce_to_string(CoercionKind::Strong, vm)
|
||||
.context(ctx)?
|
||||
.as_str()
|
||||
.to_string())
|
||||
async fn strong_coerce_to_string(co: &GenCo, val: Value) -> Result<String, ErrorKind> {
|
||||
let val = generators::request_force(co, val).await;
|
||||
let val_str = generators::request_string_coerce(co, val, CoercionKind::Strong).await;
|
||||
|
||||
Ok(val_str.as_str().to_string())
|
||||
}
|
||||
|
||||
#[builtins(state = "Rc<RefCell<KnownPaths>>")]
|
||||
mod derivation_builtins {
|
||||
use super::*;
|
||||
use tvix_eval::generators::Gen;
|
||||
|
||||
#[builtin("placeholder")]
|
||||
fn builtin_placeholder(_: &mut VM, input: Value) -> Result<Value, ErrorKind> {
|
||||
async fn builtin_placeholder(co: GenCo, input: Value) -> Result<Value, ErrorKind> {
|
||||
let placeholder = hash_placeholder(
|
||||
input
|
||||
.to_str()
|
||||
|
@ -224,29 +223,28 @@ mod derivation_builtins {
|
|||
/// This is considered an internal function, users usually want to
|
||||
/// use the higher-level `builtins.derivation` instead.
|
||||
#[builtin("derivationStrict")]
|
||||
fn builtin_derivation_strict(
|
||||
async fn builtin_derivation_strict(
|
||||
state: Rc<RefCell<KnownPaths>>,
|
||||
vm: &mut VM,
|
||||
co: GenCo,
|
||||
input: Value,
|
||||
) -> Result<Value, ErrorKind> {
|
||||
let input = input.to_attrs()?;
|
||||
let name = input
|
||||
.select_required("name")?
|
||||
.force(vm)?
|
||||
let name = generators::request_force(&co, input.select_required("name")?.clone())
|
||||
.await
|
||||
.to_str()
|
||||
.context("determining derivation name")?;
|
||||
|
||||
// Check whether attributes should be passed as a JSON file.
|
||||
// TODO: the JSON serialisation has to happen here.
|
||||
if let Some(sa) = input.select(STRUCTURED_ATTRS) {
|
||||
if sa.force(vm)?.as_bool()? {
|
||||
if generators::request_force(&co, sa.clone()).await.as_bool()? {
|
||||
return Err(ErrorKind::NotImplemented(STRUCTURED_ATTRS));
|
||||
}
|
||||
}
|
||||
|
||||
// Check whether null attributes should be ignored or passed through.
|
||||
let ignore_nulls = match input.select(IGNORE_NULLS) {
|
||||
Some(b) => b.force(vm)?.as_bool()?,
|
||||
Some(b) => generators::request_force(&co, b.clone()).await.as_bool()?,
|
||||
None => false,
|
||||
};
|
||||
|
||||
|
@ -254,37 +252,45 @@ mod derivation_builtins {
|
|||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
// Configure fixed-output derivations if required.
|
||||
|
||||
async fn select_string(
|
||||
co: &GenCo,
|
||||
attrs: &NixAttrs,
|
||||
key: &str,
|
||||
) -> Result<Option<String>, ErrorKind> {
|
||||
if let Some(attr) = attrs.select(key) {
|
||||
return Ok(Some(strong_coerce_to_string(co, attr.clone()).await?));
|
||||
}
|
||||
|
||||
Ok(None)
|
||||
}
|
||||
|
||||
populate_output_configuration(
|
||||
&mut drv,
|
||||
input
|
||||
.select("outputHash")
|
||||
.map(|v| strong_coerce_to_string(vm, v, "evaluating the `outputHash` parameter"))
|
||||
.transpose()?,
|
||||
input
|
||||
.select("outputHashAlgo")
|
||||
.map(|v| {
|
||||
strong_coerce_to_string(vm, v, "evaluating the `outputHashAlgo` parameter")
|
||||
})
|
||||
.transpose()?,
|
||||
input
|
||||
.select("outputHashMode")
|
||||
.map(|v| {
|
||||
strong_coerce_to_string(vm, v, "evaluating the `outputHashMode` parameter")
|
||||
})
|
||||
.transpose()?,
|
||||
select_string(&co, &input, "outputHash")
|
||||
.await
|
||||
.context("evaluating the `outputHash` parameter")?,
|
||||
select_string(&co, &input, "outputHashAlgo")
|
||||
.await
|
||||
.context("evaluating the `outputHashAlgo` parameter")?,
|
||||
select_string(&co, &input, "outputHashMode")
|
||||
.await
|
||||
.context("evaluating the `outputHashMode` parameter")?,
|
||||
)?;
|
||||
|
||||
for (name, value) in input.into_iter_sorted() {
|
||||
if ignore_nulls && matches!(*value.force(vm)?, Value::Null) {
|
||||
let value = generators::request_force(&co, value).await;
|
||||
if ignore_nulls && matches!(value, Value::Null) {
|
||||
continue;
|
||||
}
|
||||
|
||||
let val_str = strong_coerce_to_string(vm, &value, "evaluating derivation attributes")?;
|
||||
let val_str = strong_coerce_to_string(&co, value.clone()).await?;
|
||||
|
||||
// handle_derivation_parameters tells us whether the
|
||||
// argument should be added to the environment; continue
|
||||
// to the next one otherwise
|
||||
if !handle_derivation_parameters(&mut drv, vm, name.as_str(), &value, &val_str)? {
|
||||
if !handle_derivation_parameters(&mut drv, &co, name.as_str(), &value, &val_str).await?
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
|
@ -375,9 +381,9 @@ mod derivation_builtins {
|
|||
}
|
||||
|
||||
#[builtin("toFile")]
|
||||
fn builtin_to_file(
|
||||
async fn builtin_to_file(
|
||||
state: Rc<RefCell<KnownPaths>>,
|
||||
_: &mut VM,
|
||||
co: GenCo,
|
||||
name: Value,
|
||||
content: Value,
|
||||
) -> Result<Value, ErrorKind> {
|
||||
|
@ -421,247 +427,251 @@ mod tests {
|
|||
use super::*;
|
||||
use tvix_eval::observer::NoOpObserver;
|
||||
|
||||
static mut OBSERVER: NoOpObserver = NoOpObserver {};
|
||||
// TODO: These tests are commented out because we do not have
|
||||
// scaffolding to drive generators during testing at the moment.
|
||||
|
||||
// Creates a fake VM for tests, which can *not* actually be
|
||||
// used to force (most) values but can satisfy the type
|
||||
// parameter.
|
||||
fn fake_vm() -> VM<'static> {
|
||||
// safe because accessing the observer doesn't actually do anything
|
||||
unsafe {
|
||||
VM::new(
|
||||
Default::default(),
|
||||
Box::new(tvix_eval::DummyIO),
|
||||
&mut OBSERVER,
|
||||
Default::default(),
|
||||
)
|
||||
}
|
||||
}
|
||||
// static mut OBSERVER: NoOpObserver = NoOpObserver {};
|
||||
|
||||
#[test]
|
||||
fn populate_outputs_ok() {
|
||||
let mut vm = fake_vm();
|
||||
let mut drv = Derivation::default();
|
||||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
// // Creates a fake VM for tests, which can *not* actually be
|
||||
// // used to force (most) values but can satisfy the type
|
||||
// // parameter.
|
||||
// fn fake_vm() -> VM<'static> {
|
||||
// // safe because accessing the observer doesn't actually do anything
|
||||
// unsafe {
|
||||
// VM::new(
|
||||
// Default::default(),
|
||||
// Box::new(tvix_eval::DummyIO),
|
||||
// &mut OBSERVER,
|
||||
// Default::default(),
|
||||
// todo!(),
|
||||
// )
|
||||
// }
|
||||
// }
|
||||
|
||||
let outputs = NixList::construct(
|
||||
2,
|
||||
vec![Value::String("foo".into()), Value::String("bar".into())],
|
||||
);
|
||||
// #[test]
|
||||
// fn populate_outputs_ok() {
|
||||
// let mut vm = fake_vm();
|
||||
// let mut drv = Derivation::default();
|
||||
// drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
populate_outputs(&mut vm, &mut drv, outputs).expect("populate_outputs should succeed");
|
||||
// let outputs = NixList::construct(
|
||||
// 2,
|
||||
// vec![Value::String("foo".into()), Value::String("bar".into())],
|
||||
// );
|
||||
|
||||
assert_eq!(drv.outputs.len(), 2);
|
||||
assert!(drv.outputs.contains_key("bar"));
|
||||
assert!(drv.outputs.contains_key("foo"));
|
||||
}
|
||||
// populate_outputs(&mut vm, &mut drv, outputs).expect("populate_outputs should succeed");
|
||||
|
||||
#[test]
|
||||
fn populate_outputs_duplicate() {
|
||||
let mut vm = fake_vm();
|
||||
let mut drv = Derivation::default();
|
||||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
// assert_eq!(drv.outputs.len(), 2);
|
||||
// assert!(drv.outputs.contains_key("bar"));
|
||||
// assert!(drv.outputs.contains_key("foo"));
|
||||
// }
|
||||
|
||||
let outputs = NixList::construct(
|
||||
2,
|
||||
vec![Value::String("foo".into()), Value::String("foo".into())],
|
||||
);
|
||||
// #[test]
|
||||
// fn populate_outputs_duplicate() {
|
||||
// let mut vm = fake_vm();
|
||||
// let mut drv = Derivation::default();
|
||||
// drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
populate_outputs(&mut vm, &mut drv, outputs)
|
||||
.expect_err("supplying duplicate outputs should fail");
|
||||
}
|
||||
// let outputs = NixList::construct(
|
||||
// 2,
|
||||
// vec![Value::String("foo".into()), Value::String("foo".into())],
|
||||
// );
|
||||
|
||||
#[test]
|
||||
fn populate_inputs_empty() {
|
||||
let mut drv = Derivation::default();
|
||||
let paths = KnownPaths::default();
|
||||
let inputs = vec![];
|
||||
// populate_outputs(&mut vm, &mut drv, outputs)
|
||||
// .expect_err("supplying duplicate outputs should fail");
|
||||
// }
|
||||
|
||||
populate_inputs(&mut drv, &paths, inputs);
|
||||
// #[test]
|
||||
// fn populate_inputs_empty() {
|
||||
// let mut drv = Derivation::default();
|
||||
// let paths = KnownPaths::default();
|
||||
// let inputs = vec![];
|
||||
|
||||
assert!(drv.input_sources.is_empty());
|
||||
assert!(drv.input_derivations.is_empty());
|
||||
}
|
||||
// populate_inputs(&mut drv, &paths, inputs);
|
||||
|
||||
#[test]
|
||||
fn populate_inputs_all() {
|
||||
let mut drv = Derivation::default();
|
||||
// assert!(drv.input_sources.is_empty());
|
||||
// assert!(drv.input_derivations.is_empty());
|
||||
// }
|
||||
|
||||
let mut paths = KnownPaths::default();
|
||||
paths.plain("/nix/store/fn7zvafq26f0c8b17brs7s95s10ibfzs-foo");
|
||||
paths.drv(
|
||||
"/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv",
|
||||
&["out"],
|
||||
);
|
||||
paths.output(
|
||||
"/nix/store/zvpskvjwi72fjxg0vzq822sfvq20mq4l-bar",
|
||||
"out",
|
||||
"/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv",
|
||||
);
|
||||
// #[test]
|
||||
// fn populate_inputs_all() {
|
||||
// let mut drv = Derivation::default();
|
||||
|
||||
let inputs = vec![
|
||||
"/nix/store/fn7zvafq26f0c8b17brs7s95s10ibfzs-foo".into(),
|
||||
"/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv".into(),
|
||||
"/nix/store/zvpskvjwi72fjxg0vzq822sfvq20mq4l-bar".into(),
|
||||
];
|
||||
// let mut paths = KnownPaths::default();
|
||||
// paths.plain("/nix/store/fn7zvafq26f0c8b17brs7s95s10ibfzs-foo");
|
||||
// paths.drv(
|
||||
// "/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv",
|
||||
// &["out"],
|
||||
// );
|
||||
// paths.output(
|
||||
// "/nix/store/zvpskvjwi72fjxg0vzq822sfvq20mq4l-bar",
|
||||
// "out",
|
||||
// "/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv",
|
||||
// );
|
||||
|
||||
populate_inputs(&mut drv, &paths, inputs);
|
||||
// let inputs = vec![
|
||||
// "/nix/store/fn7zvafq26f0c8b17brs7s95s10ibfzs-foo".into(),
|
||||
// "/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv".into(),
|
||||
// "/nix/store/zvpskvjwi72fjxg0vzq822sfvq20mq4l-bar".into(),
|
||||
// ];
|
||||
|
||||
assert_eq!(drv.input_sources.len(), 1);
|
||||
assert!(drv
|
||||
.input_sources
|
||||
.contains("/nix/store/fn7zvafq26f0c8b17brs7s95s10ibfzs-foo"));
|
||||
// populate_inputs(&mut drv, &paths, inputs);
|
||||
|
||||
assert_eq!(drv.input_derivations.len(), 1);
|
||||
assert!(drv
|
||||
.input_derivations
|
||||
.contains_key("/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv"));
|
||||
}
|
||||
// assert_eq!(drv.input_sources.len(), 1);
|
||||
// assert!(drv
|
||||
// .input_sources
|
||||
// .contains("/nix/store/fn7zvafq26f0c8b17brs7s95s10ibfzs-foo"));
|
||||
|
||||
#[test]
|
||||
fn populate_output_config_std() {
|
||||
let mut drv = Derivation::default();
|
||||
// assert_eq!(drv.input_derivations.len(), 1);
|
||||
// assert!(drv
|
||||
// .input_derivations
|
||||
// .contains_key("/nix/store/aqffiyqx602lbam7n1zsaz3yrh6v08pc-bar.drv"));
|
||||
// }
|
||||
|
||||
populate_output_configuration(&mut drv, None, None, None)
|
||||
.expect("populate_output_configuration() should succeed");
|
||||
// #[test]
|
||||
// fn populate_output_config_std() {
|
||||
// let mut drv = Derivation::default();
|
||||
|
||||
assert_eq!(drv, Derivation::default(), "derivation should be unchanged");
|
||||
}
|
||||
// populate_output_configuration(&mut drv, None, None, None)
|
||||
// .expect("populate_output_configuration() should succeed");
|
||||
|
||||
#[test]
|
||||
fn populate_output_config_fod() {
|
||||
let mut drv = Derivation::default();
|
||||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
// assert_eq!(drv, Derivation::default(), "derivation should be unchanged");
|
||||
// }
|
||||
|
||||
populate_output_configuration(
|
||||
&mut drv,
|
||||
Some("0000000000000000000000000000000000000000000000000000000000000000".into()),
|
||||
Some("sha256".into()),
|
||||
None,
|
||||
)
|
||||
.expect("populate_output_configuration() should succeed");
|
||||
// #[test]
|
||||
// fn populate_output_config_fod() {
|
||||
// let mut drv = Derivation::default();
|
||||
// drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
let expected = Hash {
|
||||
algo: "sha256".into(),
|
||||
digest: "0000000000000000000000000000000000000000000000000000000000000000".into(),
|
||||
};
|
||||
// populate_output_configuration(
|
||||
// &mut drv,
|
||||
// Some("0000000000000000000000000000000000000000000000000000000000000000".into()),
|
||||
// Some("sha256".into()),
|
||||
// None,
|
||||
// )
|
||||
// .expect("populate_output_configuration() should succeed");
|
||||
|
||||
assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
}
|
||||
// let expected = Hash {
|
||||
// algo: "sha256".into(),
|
||||
// digest: "0000000000000000000000000000000000000000000000000000000000000000".into(),
|
||||
// };
|
||||
|
||||
#[test]
|
||||
fn populate_output_config_fod_recursive() {
|
||||
let mut drv = Derivation::default();
|
||||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
// assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
// }
|
||||
|
||||
populate_output_configuration(
|
||||
&mut drv,
|
||||
Some("0000000000000000000000000000000000000000000000000000000000000000".into()),
|
||||
Some("sha256".into()),
|
||||
Some("recursive".into()),
|
||||
)
|
||||
.expect("populate_output_configuration() should succeed");
|
||||
// #[test]
|
||||
// fn populate_output_config_fod_recursive() {
|
||||
// let mut drv = Derivation::default();
|
||||
// drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
let expected = Hash {
|
||||
algo: "r:sha256".into(),
|
||||
digest: "0000000000000000000000000000000000000000000000000000000000000000".into(),
|
||||
};
|
||||
// populate_output_configuration(
|
||||
// &mut drv,
|
||||
// Some("0000000000000000000000000000000000000000000000000000000000000000".into()),
|
||||
// Some("sha256".into()),
|
||||
// Some("recursive".into()),
|
||||
// )
|
||||
// .expect("populate_output_configuration() should succeed");
|
||||
|
||||
assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
}
|
||||
// let expected = Hash {
|
||||
// algo: "r:sha256".into(),
|
||||
// digest: "0000000000000000000000000000000000000000000000000000000000000000".into(),
|
||||
// };
|
||||
|
||||
#[test]
|
||||
/// hash_algo set to sha256, but SRI hash passed
|
||||
fn populate_output_config_flat_sri_sha256() {
|
||||
let mut drv = Derivation::default();
|
||||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
// assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
// }
|
||||
|
||||
populate_output_configuration(
|
||||
&mut drv,
|
||||
Some("sha256-swapHA/ZO8QoDPwumMt6s5gf91oYe+oyk4EfRSyJqMg=".into()),
|
||||
Some("sha256".into()),
|
||||
Some("flat".into()),
|
||||
)
|
||||
.expect("populate_output_configuration() should succeed");
|
||||
// #[test]
|
||||
// /// hash_algo set to sha256, but SRI hash passed
|
||||
// fn populate_output_config_flat_sri_sha256() {
|
||||
// let mut drv = Derivation::default();
|
||||
// drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
let expected = Hash {
|
||||
algo: "sha256".into(),
|
||||
digest: "b306a91c0fd93bc4280cfc2e98cb7ab3981ff75a187bea3293811f452c89a8c8".into(), // lower hex
|
||||
};
|
||||
// populate_output_configuration(
|
||||
// &mut drv,
|
||||
// Some("sha256-swapHA/ZO8QoDPwumMt6s5gf91oYe+oyk4EfRSyJqMg=".into()),
|
||||
// Some("sha256".into()),
|
||||
// Some("flat".into()),
|
||||
// )
|
||||
// .expect("populate_output_configuration() should succeed");
|
||||
|
||||
assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
}
|
||||
// let expected = Hash {
|
||||
// algo: "sha256".into(),
|
||||
// digest: "b306a91c0fd93bc4280cfc2e98cb7ab3981ff75a187bea3293811f452c89a8c8".into(), // lower hex
|
||||
// };
|
||||
|
||||
#[test]
|
||||
/// hash_algo set to empty string, SRI hash passed
|
||||
fn populate_output_config_flat_sri() {
|
||||
let mut drv = Derivation::default();
|
||||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
// assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
// }
|
||||
|
||||
populate_output_configuration(
|
||||
&mut drv,
|
||||
Some("sha256-s6JN6XqP28g1uYMxaVAQMLiXcDG8tUs7OsE3QPhGqzA=".into()),
|
||||
Some("".into()),
|
||||
Some("flat".into()),
|
||||
)
|
||||
.expect("populate_output_configuration() should succeed");
|
||||
// #[test]
|
||||
// /// hash_algo set to empty string, SRI hash passed
|
||||
// fn populate_output_config_flat_sri() {
|
||||
// let mut drv = Derivation::default();
|
||||
// drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
let expected = Hash {
|
||||
algo: "sha256".into(),
|
||||
digest: "b3a24de97a8fdbc835b9833169501030b8977031bcb54b3b3ac13740f846ab30".into(), // lower hex
|
||||
};
|
||||
// populate_output_configuration(
|
||||
// &mut drv,
|
||||
// Some("sha256-s6JN6XqP28g1uYMxaVAQMLiXcDG8tUs7OsE3QPhGqzA=".into()),
|
||||
// Some("".into()),
|
||||
// Some("flat".into()),
|
||||
// )
|
||||
// .expect("populate_output_configuration() should succeed");
|
||||
|
||||
assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
}
|
||||
// let expected = Hash {
|
||||
// algo: "sha256".into(),
|
||||
// digest: "b3a24de97a8fdbc835b9833169501030b8977031bcb54b3b3ac13740f846ab30".into(), // lower hex
|
||||
// };
|
||||
|
||||
#[test]
|
||||
fn handle_outputs_parameter() {
|
||||
let mut vm = fake_vm();
|
||||
let mut drv = Derivation::default();
|
||||
drv.outputs.insert("out".to_string(), Default::default());
|
||||
// assert_eq!(drv.outputs["out"].hash, Some(expected));
|
||||
// }
|
||||
|
||||
let outputs = Value::List(NixList::construct(
|
||||
2,
|
||||
vec![Value::String("foo".into()), Value::String("bar".into())],
|
||||
));
|
||||
let outputs_str = outputs
|
||||
.coerce_to_string(CoercionKind::Strong, &mut vm)
|
||||
.unwrap();
|
||||
// #[test]
|
||||
// fn handle_outputs_parameter() {
|
||||
// let mut vm = fake_vm();
|
||||
// let mut drv = Derivation::default();
|
||||
// drv.outputs.insert("out".to_string(), Default::default());
|
||||
|
||||
handle_derivation_parameters(&mut drv, &mut vm, "outputs", &outputs, outputs_str.as_str())
|
||||
.expect("handling 'outputs' parameter should succeed");
|
||||
// let outputs = Value::List(NixList::construct(
|
||||
// 2,
|
||||
// vec![Value::String("foo".into()), Value::String("bar".into())],
|
||||
// ));
|
||||
// let outputs_str = outputs
|
||||
// .coerce_to_string(CoercionKind::Strong, &mut vm)
|
||||
// .unwrap();
|
||||
|
||||
assert_eq!(drv.outputs.len(), 2);
|
||||
assert!(drv.outputs.contains_key("bar"));
|
||||
assert!(drv.outputs.contains_key("foo"));
|
||||
}
|
||||
// handle_derivation_parameters(&mut drv, &mut vm, "outputs", &outputs, outputs_str.as_str())
|
||||
// .expect("handling 'outputs' parameter should succeed");
|
||||
|
||||
#[test]
|
||||
fn handle_args_parameter() {
|
||||
let mut vm = fake_vm();
|
||||
let mut drv = Derivation::default();
|
||||
// assert_eq!(drv.outputs.len(), 2);
|
||||
// assert!(drv.outputs.contains_key("bar"));
|
||||
// assert!(drv.outputs.contains_key("foo"));
|
||||
// }
|
||||
|
||||
let args = Value::List(NixList::construct(
|
||||
3,
|
||||
vec![
|
||||
Value::String("--foo".into()),
|
||||
Value::String("42".into()),
|
||||
Value::String("--bar".into()),
|
||||
],
|
||||
));
|
||||
// #[test]
|
||||
// fn handle_args_parameter() {
|
||||
// let mut vm = fake_vm();
|
||||
// let mut drv = Derivation::default();
|
||||
|
||||
let args_str = args
|
||||
.coerce_to_string(CoercionKind::Strong, &mut vm)
|
||||
.unwrap();
|
||||
// let args = Value::List(NixList::construct(
|
||||
// 3,
|
||||
// vec![
|
||||
// Value::String("--foo".into()),
|
||||
// Value::String("42".into()),
|
||||
// Value::String("--bar".into()),
|
||||
// ],
|
||||
// ));
|
||||
|
||||
handle_derivation_parameters(&mut drv, &mut vm, "args", &args, args_str.as_str())
|
||||
.expect("handling 'args' parameter should succeed");
|
||||
// let args_str = args
|
||||
// .coerce_to_string(CoercionKind::Strong, &mut vm)
|
||||
// .unwrap();
|
||||
|
||||
assert_eq!(
|
||||
drv.arguments,
|
||||
vec!["--foo".to_string(), "42".to_string(), "--bar".to_string()]
|
||||
);
|
||||
}
|
||||
// handle_derivation_parameters(&mut drv, &mut vm, "args", &args, args_str.as_str())
|
||||
// .expect("handling 'args' parameter should succeed");
|
||||
|
||||
// assert_eq!(
|
||||
// drv.arguments,
|
||||
// vec!["--foo".to_string(), "42".to_string(), "--bar".to_string()]
|
||||
// );
|
||||
// }
|
||||
|
||||
#[test]
|
||||
fn builtins_placeholder_hashes() {
|
||||
|
|
|
@ -12,7 +12,7 @@ use clap::Parser;
|
|||
use known_paths::KnownPaths;
|
||||
use rustyline::{error::ReadlineError, Editor};
|
||||
use tvix_eval::observer::{DisassemblingObserver, TracingObserver};
|
||||
use tvix_eval::{Builtin, BuiltinArgument, Value, VM};
|
||||
use tvix_eval::{Builtin, Value};
|
||||
|
||||
#[derive(Parser)]
|
||||
struct Args {
|
||||
|
|
|
@ -6,20 +6,24 @@ use quote::{quote, quote_spanned, ToTokens};
|
|||
use syn::parse::Parse;
|
||||
use syn::spanned::Spanned;
|
||||
use syn::{
|
||||
parse2, parse_macro_input, parse_quote, Attribute, FnArg, Ident, Item, ItemMod, LitStr, Meta,
|
||||
Pat, PatIdent, PatType, Token, Type,
|
||||
parse2, parse_macro_input, parse_quote, parse_quote_spanned, Attribute, FnArg, Ident, Item,
|
||||
ItemMod, LitStr, Meta, Pat, PatIdent, PatType, Token, Type,
|
||||
};
|
||||
|
||||
struct BuiltinArgs {
|
||||
name: LitStr,
|
||||
}
|
||||
/// Description of a single argument passed to a builtin
|
||||
struct BuiltinArgument {
|
||||
/// The name of the argument, to be used in docstrings and error messages
|
||||
name: Ident,
|
||||
|
||||
impl Parse for BuiltinArgs {
|
||||
fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
|
||||
Ok(BuiltinArgs {
|
||||
name: input.parse()?,
|
||||
})
|
||||
}
|
||||
/// Type of the argument.
|
||||
ty: Box<Type>,
|
||||
|
||||
/// Whether the argument should be forced before the underlying builtin
|
||||
/// function is called.
|
||||
strict: bool,
|
||||
|
||||
/// Span at which the argument was defined.
|
||||
span: Span,
|
||||
}
|
||||
|
||||
fn extract_docstring(attrs: &[Attribute]) -> Option<String> {
|
||||
|
@ -97,6 +101,11 @@ fn parse_module_args(args: TokenStream) -> Option<Type> {
|
|||
/// builtin upon instantiation. Using this, builtins that close over some external state can be
|
||||
/// written.
|
||||
///
|
||||
/// The type of each function is rewritten to receive a `Vec<Value>`, containing each `Value`
|
||||
/// argument that the function receives. The body of functions is accordingly rewritten to "unwrap"
|
||||
/// values from this vector and bind them to the correct names, so unless a static error occurs this
|
||||
/// transformation is mostly invisible to users of the macro.
|
||||
///
|
||||
/// A function `fn builtins() -> Vec<Builtin>` will be defined within the annotated module,
|
||||
/// returning a list of [`tvix_eval::Builtin`] for each function annotated with the `#[builtin]`
|
||||
/// attribute within the module. If a `state` type is specified, the `builtins` function will take a
|
||||
|
@ -114,10 +123,10 @@ fn parse_module_args(args: TokenStream) -> Option<Type> {
|
|||
///
|
||||
/// #[builtins]
|
||||
/// mod builtins {
|
||||
/// use tvix_eval::{ErrorKind, Value, VM};
|
||||
/// use tvix_eval::{GenCo, ErrorKind, Value};
|
||||
///
|
||||
/// #[builtin("identity")]
|
||||
/// pub fn builtin_identity(_vm: &mut VM, x: Value) -> Result<Value, ErrorKind> {
|
||||
/// pub async fn builtin_identity(co: GenCo, x: Value) -> Result<Value, ErrorKind> {
|
||||
/// Ok(x)
|
||||
/// }
|
||||
///
|
||||
|
@ -125,7 +134,7 @@ fn parse_module_args(args: TokenStream) -> Option<Type> {
|
|||
/// // argument with the `#[lazy]` attribute
|
||||
///
|
||||
/// #[builtin("tryEval")]
|
||||
/// pub fn builtin_try_eval(vm: &mut VM, #[lazy] x: Value) -> Result<Value, ErrorKind> {
|
||||
/// pub async fn builtin_try_eval(co: GenCo, #[lazy] x: Value) -> Result<Value, ErrorKind> {
|
||||
/// todo!()
|
||||
/// }
|
||||
/// }
|
||||
|
@ -156,7 +165,7 @@ pub fn builtins(args: TokenStream, item: TokenStream) -> TokenStream {
|
|||
.position(|attr| attr.path.get_ident().iter().any(|id| *id == "builtin"))
|
||||
{
|
||||
let builtin_attr = f.attrs.remove(builtin_attr_pos);
|
||||
let BuiltinArgs { name } = match builtin_attr.parse_args() {
|
||||
let name: LitStr = match builtin_attr.parse_args() {
|
||||
Ok(args) => args,
|
||||
Err(err) => return err.into_compile_error().into(),
|
||||
};
|
||||
|
@ -169,10 +178,11 @@ pub fn builtins(args: TokenStream, item: TokenStream) -> TokenStream {
|
|||
.into();
|
||||
}
|
||||
|
||||
// Determine if this function is taking the state parameter.
|
||||
let mut args_iter = f.sig.inputs.iter_mut().peekable();
|
||||
// Inspect the first argument to determine if this function is
|
||||
// taking the state parameter.
|
||||
// TODO(tazjin): add a test in //tvix/eval that covers this
|
||||
let mut captures_state = false;
|
||||
if let Some(FnArg::Typed(PatType { pat, .. })) = args_iter.peek() {
|
||||
if let FnArg::Typed(PatType { pat, .. }) = &f.sig.inputs[0] {
|
||||
if let Pat::Ident(PatIdent { ident, .. }) = pat.as_ref() {
|
||||
if ident.to_string() == "state" {
|
||||
if state_type.is_none() {
|
||||
|
@ -184,20 +194,28 @@ pub fn builtins(args: TokenStream, item: TokenStream) -> TokenStream {
|
|||
}
|
||||
}
|
||||
|
||||
// skip state and/or VM args ..
|
||||
let skip_num = if captures_state { 2 } else { 1 };
|
||||
let mut rewritten_args = std::mem::take(&mut f.sig.inputs)
|
||||
.into_iter()
|
||||
.collect::<Vec<_>>();
|
||||
|
||||
let builtin_arguments = args_iter
|
||||
.skip(skip_num)
|
||||
// Split out the value arguments from the static arguments.
|
||||
let split_idx = if captures_state { 2 } else { 1 };
|
||||
let value_args = rewritten_args.split_off(split_idx);
|
||||
|
||||
let builtin_arguments = value_args
|
||||
.into_iter()
|
||||
.map(|arg| {
|
||||
let span = arg.span();
|
||||
let mut strict = true;
|
||||
let name = match arg {
|
||||
let (name, ty) = match arg {
|
||||
FnArg::Receiver(_) => {
|
||||
return Err(quote_spanned!(arg.span() => {
|
||||
compile_error!("Unexpected receiver argument in builtin")
|
||||
return Err(quote_spanned!(span => {
|
||||
compile_error!("unexpected receiver argument in builtin")
|
||||
}))
|
||||
}
|
||||
FnArg::Typed(PatType { attrs, pat, .. }) => {
|
||||
FnArg::Typed(PatType {
|
||||
mut attrs, pat, ty, ..
|
||||
}) => {
|
||||
attrs.retain(|attr| {
|
||||
attr.path.get_ident().into_iter().any(|id| {
|
||||
if id == "lazy" {
|
||||
|
@ -209,34 +227,66 @@ pub fn builtins(args: TokenStream, item: TokenStream) -> TokenStream {
|
|||
})
|
||||
});
|
||||
match pat.as_ref() {
|
||||
Pat::Ident(PatIdent { ident, .. }) => ident.to_string(),
|
||||
_ => "unknown".to_string(),
|
||||
Pat::Ident(PatIdent { ident, .. }) => {
|
||||
(ident.clone(), ty.clone())
|
||||
}
|
||||
_ => panic!("ignored value parameters must be named, e.g. `_x` and not just `_`"),
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
Ok(quote_spanned!(arg.span() => {
|
||||
crate::BuiltinArgument {
|
||||
strict: #strict,
|
||||
name: #name,
|
||||
}
|
||||
}))
|
||||
Ok(BuiltinArgument {
|
||||
strict,
|
||||
span,
|
||||
name,
|
||||
ty,
|
||||
})
|
||||
})
|
||||
.collect::<Result<Vec<_>, _>>();
|
||||
.collect::<Result<Vec<BuiltinArgument>, _>>();
|
||||
|
||||
let builtin_arguments = match builtin_arguments {
|
||||
Ok(args) => args,
|
||||
Err(err) => return err.into(),
|
||||
|
||||
// reverse argument order, as they are popped from the stack
|
||||
// slice in opposite order
|
||||
Ok(args) => args,
|
||||
};
|
||||
|
||||
let fn_name = f.sig.ident.clone();
|
||||
let num_args = f.sig.inputs.len() - skip_num;
|
||||
let args = (0..num_args)
|
||||
.map(|n| Ident::new(&format!("arg_{n}"), Span::call_site()))
|
||||
.collect::<Vec<_>>();
|
||||
let mut reversed_args = args.clone();
|
||||
reversed_args.reverse();
|
||||
// Rewrite the argument to the actual function to take a
|
||||
// `Vec<Value>`, which is then destructured into the
|
||||
// user-defined values in the function header.
|
||||
let sig_span = f.sig.span();
|
||||
rewritten_args.push(parse_quote_spanned!(sig_span=> mut values: Vec<Value>));
|
||||
f.sig.inputs = rewritten_args.into_iter().collect();
|
||||
|
||||
// Rewrite the body of the function to do said argument forcing.
|
||||
//
|
||||
// This is done by creating a new block for each of the
|
||||
// arguments that evaluates it, and wraps the inner block.
|
||||
for arg in &builtin_arguments {
|
||||
let block = &f.block;
|
||||
let ty = &arg.ty;
|
||||
let ident = &arg.name;
|
||||
|
||||
if arg.strict {
|
||||
f.block = Box::new(parse_quote_spanned! {arg.span=> {
|
||||
let #ident: #ty = generators::request_force(&co, values.pop()
|
||||
.expect("Tvix bug: builtin called with incorrect number of arguments")).await;
|
||||
|
||||
#block
|
||||
}});
|
||||
} else {
|
||||
f.block = Box::new(parse_quote_spanned! {arg.span=> {
|
||||
let #ident: #ty = values.pop()
|
||||
.expect("Tvix bug: builtin called with incorrect number of arguments");
|
||||
|
||||
#block
|
||||
}})
|
||||
}
|
||||
}
|
||||
|
||||
let fn_name = f.sig.ident.clone();
|
||||
let arg_count = builtin_arguments.len();
|
||||
let docstring = match extract_docstring(&f.attrs) {
|
||||
Some(docs) => quote!(Some(#docs)),
|
||||
None => quote!(None),
|
||||
|
@ -247,24 +297,18 @@ pub fn builtins(args: TokenStream, item: TokenStream) -> TokenStream {
|
|||
let inner_state = state.clone();
|
||||
crate::Builtin::new(
|
||||
#name,
|
||||
&[#(#builtin_arguments),*],
|
||||
#docstring,
|
||||
move |mut args: Vec<crate::Value>, vm: &mut crate::VM| {
|
||||
#(let #reversed_args = args.pop().unwrap();)*
|
||||
#fn_name(inner_state.clone(), vm, #(#args),*)
|
||||
}
|
||||
#arg_count,
|
||||
move |values| Gen::new(|co| generators::pin_generator(#fn_name(inner_state.clone(), co, values))),
|
||||
)
|
||||
}});
|
||||
} else {
|
||||
builtins.push(quote_spanned! { builtin_attr.span() => {
|
||||
crate::Builtin::new(
|
||||
#name,
|
||||
&[#(#builtin_arguments),*],
|
||||
#docstring,
|
||||
|mut args: Vec<crate::Value>, vm: &mut crate::VM| {
|
||||
#(let #reversed_args = args.pop().unwrap();)*
|
||||
#fn_name(vm, #(#args),*)
|
||||
}
|
||||
#arg_count,
|
||||
|values| Gen::new(|co| generators::pin_generator(#fn_name(co, values))),
|
||||
)
|
||||
}});
|
||||
}
|
||||
|
|
|
@ -1,20 +1,21 @@
|
|||
pub use tvix_eval::{Builtin, BuiltinArgument, Value, VM};
|
||||
pub use tvix_eval::{Builtin, Value};
|
||||
use tvix_eval_builtin_macros::builtins;
|
||||
|
||||
#[builtins]
|
||||
mod builtins {
|
||||
use tvix_eval::{ErrorKind, Value, VM};
|
||||
use tvix_eval::generators::{self, Gen, GenCo};
|
||||
use tvix_eval::{ErrorKind, Value};
|
||||
|
||||
/// Test docstring.
|
||||
///
|
||||
/// It has multiple lines!
|
||||
#[builtin("identity")]
|
||||
pub fn builtin_identity(_vm: &mut VM, x: Value) -> Result<Value, ErrorKind> {
|
||||
Ok(x)
|
||||
pub async fn builtin_identity(co: GenCo, x: Value) -> Result<Value, ErrorKind> {
|
||||
Ok(todo!())
|
||||
}
|
||||
|
||||
#[builtin("tryEval")]
|
||||
pub fn builtin_try_eval(_: &mut VM, #[lazy] _x: Value) -> Result<Value, ErrorKind> {
|
||||
pub async fn builtin_try_eval(co: GenCo, #[lazy] _x: Value) -> Result<Value, ErrorKind> {
|
||||
todo!()
|
||||
}
|
||||
}
|
||||
|
|
File diff suppressed because one or more lines are too long
|
@ -1,4 +1,5 @@
|
|||
use builtin_macros::builtins;
|
||||
use genawaiter::rc::Gen;
|
||||
use smol_str::SmolStr;
|
||||
|
||||
use std::{
|
||||
|
@ -6,7 +7,13 @@ use std::{
|
|||
time::{SystemTime, UNIX_EPOCH},
|
||||
};
|
||||
|
||||
use crate::{errors::ErrorKind, io::FileType, value::NixAttrs, vm::VM, Value};
|
||||
use crate::{
|
||||
errors::ErrorKind,
|
||||
io::FileType,
|
||||
value::NixAttrs,
|
||||
vm::generators::{self, GenCo},
|
||||
Value,
|
||||
};
|
||||
|
||||
#[builtins]
|
||||
mod impure_builtins {
|
||||
|
@ -14,21 +21,22 @@ mod impure_builtins {
|
|||
use crate::builtins::coerce_value_to_path;
|
||||
|
||||
#[builtin("getEnv")]
|
||||
fn builtin_get_env(_: &mut VM, var: Value) -> Result<Value, ErrorKind> {
|
||||
async fn builtin_get_env(co: GenCo, var: Value) -> Result<Value, ErrorKind> {
|
||||
Ok(env::var(var.to_str()?).unwrap_or_else(|_| "".into()).into())
|
||||
}
|
||||
|
||||
#[builtin("pathExists")]
|
||||
fn builtin_path_exists(vm: &mut VM, s: Value) -> Result<Value, ErrorKind> {
|
||||
let path = coerce_value_to_path(&s, vm)?;
|
||||
vm.io().path_exists(path).map(Value::Bool)
|
||||
async fn builtin_path_exists(co: GenCo, path: Value) -> Result<Value, ErrorKind> {
|
||||
let path = coerce_value_to_path(&co, path).await?;
|
||||
Ok(generators::request_path_exists(&co, path).await)
|
||||
}
|
||||
|
||||
#[builtin("readDir")]
|
||||
fn builtin_read_dir(vm: &mut VM, path: Value) -> Result<Value, ErrorKind> {
|
||||
let path = coerce_value_to_path(&path, vm)?;
|
||||
async fn builtin_read_dir(co: GenCo, path: Value) -> Result<Value, ErrorKind> {
|
||||
let path = coerce_value_to_path(&co, path).await?;
|
||||
|
||||
let res = vm.io().read_dir(path)?.into_iter().map(|(name, ftype)| {
|
||||
let dir = generators::request_read_dir(&co, path).await;
|
||||
let res = dir.into_iter().map(|(name, ftype)| {
|
||||
(
|
||||
name,
|
||||
Value::String(
|
||||
|
@ -47,11 +55,9 @@ mod impure_builtins {
|
|||
}
|
||||
|
||||
#[builtin("readFile")]
|
||||
fn builtin_read_file(vm: &mut VM, path: Value) -> Result<Value, ErrorKind> {
|
||||
let path = coerce_value_to_path(&path, vm)?;
|
||||
vm.io()
|
||||
.read_to_string(path)
|
||||
.map(|s| Value::String(s.into()))
|
||||
async fn builtin_read_file(co: GenCo, path: Value) -> Result<Value, ErrorKind> {
|
||||
let path = coerce_value_to_path(&co, path).await?;
|
||||
Ok(generators::request_read_to_string(&co, path).await)
|
||||
}
|
||||
}
|
||||
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -5,21 +5,98 @@
|
|||
//! compiler and VM state (such as the [`crate::SourceCode`]
|
||||
//! instance, or observers).
|
||||
|
||||
use super::GlobalsMap;
|
||||
use genawaiter::rc::Gen;
|
||||
use std::rc::Weak;
|
||||
|
||||
use crate::{
|
||||
builtins::coerce_value_to_path,
|
||||
generators::pin_generator,
|
||||
observer::NoOpObserver,
|
||||
value::{Builtin, BuiltinArgument, Thunk},
|
||||
vm::VM,
|
||||
value::{Builtin, Thunk},
|
||||
vm::generators::{self, GenCo},
|
||||
ErrorKind, SourceCode, Value,
|
||||
};
|
||||
|
||||
use super::GlobalsMap;
|
||||
use crate::builtins::coerce_value_to_path;
|
||||
async fn import_impl(
|
||||
co: GenCo,
|
||||
globals: Weak<GlobalsMap>,
|
||||
source: SourceCode,
|
||||
mut args: Vec<Value>,
|
||||
) -> Result<Value, ErrorKind> {
|
||||
let mut path = coerce_value_to_path(&co, args.pop().unwrap()).await?;
|
||||
|
||||
/// Constructs and inserts the `import` builtin. This builtin is special in that
|
||||
/// it needs to capture the [crate::SourceCode] structure to correctly track
|
||||
/// source code locations while invoking a compiler.
|
||||
if path.is_dir() {
|
||||
path.push("default.nix");
|
||||
}
|
||||
|
||||
if let Some(cached) = generators::request_import_cache_lookup(&co, path.clone()).await {
|
||||
return Ok(cached);
|
||||
}
|
||||
|
||||
// TODO(tazjin): make this return a string directly instead
|
||||
let contents = generators::request_read_to_string(&co, path.clone())
|
||||
.await
|
||||
.to_str()?
|
||||
.as_str()
|
||||
.to_string();
|
||||
|
||||
let parsed = rnix::ast::Root::parse(&contents);
|
||||
let errors = parsed.errors();
|
||||
let file = source.add_file(path.to_string_lossy().to_string(), contents);
|
||||
|
||||
if !errors.is_empty() {
|
||||
return Err(ErrorKind::ImportParseError {
|
||||
path,
|
||||
file,
|
||||
errors: errors.to_vec(),
|
||||
});
|
||||
}
|
||||
|
||||
let result = crate::compiler::compile(
|
||||
&parsed.tree().expr().unwrap(),
|
||||
Some(path.clone()),
|
||||
file,
|
||||
// The VM must ensure that a strong reference to the globals outlives
|
||||
// any self-references (which are weak) embedded within the globals. If
|
||||
// the expect() below panics, it means that did not happen.
|
||||
globals
|
||||
.upgrade()
|
||||
.expect("globals dropped while still in use"),
|
||||
&mut NoOpObserver::default(),
|
||||
)
|
||||
.map_err(|err| ErrorKind::ImportCompilerError {
|
||||
path: path.clone(),
|
||||
errors: vec![err],
|
||||
})?;
|
||||
|
||||
if !result.errors.is_empty() {
|
||||
return Err(ErrorKind::ImportCompilerError {
|
||||
path,
|
||||
errors: result.errors,
|
||||
});
|
||||
}
|
||||
|
||||
// TODO: emit not just the warning kind, hmm
|
||||
// for warning in result.warnings {
|
||||
// vm.push_warning(warning);
|
||||
// }
|
||||
|
||||
// Compilation succeeded, we can construct a thunk from whatever it spat
|
||||
// out and return that.
|
||||
let res = Value::Thunk(Thunk::new_suspended(
|
||||
result.lambda,
|
||||
generators::request_span(&co).await,
|
||||
));
|
||||
|
||||
generators::request_import_cache_put(&co, path, res.clone()).await;
|
||||
|
||||
Ok(res)
|
||||
}
|
||||
|
||||
/// Constructs the `import` builtin. This builtin is special in that
|
||||
/// it needs to capture the [crate::SourceCode] structure to correctly
|
||||
/// track source code locations while invoking a compiler.
|
||||
// TODO: need to be able to pass through a CompilationObserver, too.
|
||||
// TODO: can the `SourceCode` come from the compiler?
|
||||
pub(super) fn builtins_import(globals: &Weak<GlobalsMap>, source: SourceCode) -> Builtin {
|
||||
|
@ -31,75 +108,10 @@ pub(super) fn builtins_import(globals: &Weak<GlobalsMap>, source: SourceCode) ->
|
|||
|
||||
Builtin::new(
|
||||
"import",
|
||||
&[BuiltinArgument {
|
||||
strict: true,
|
||||
name: "path",
|
||||
}],
|
||||
None,
|
||||
move |mut args: Vec<Value>, vm: &mut VM| {
|
||||
let mut path = coerce_value_to_path(&args.pop().unwrap(), vm)?;
|
||||
if path.is_dir() {
|
||||
path.push("default.nix");
|
||||
}
|
||||
|
||||
let current_span = vm.current_light_span();
|
||||
|
||||
if let Some(cached) = vm.import_cache.get(&path) {
|
||||
return Ok(cached.clone());
|
||||
}
|
||||
|
||||
let contents = vm.io().read_to_string(path.clone())?;
|
||||
|
||||
let parsed = rnix::ast::Root::parse(&contents);
|
||||
let errors = parsed.errors();
|
||||
|
||||
let file = source.add_file(path.to_string_lossy().to_string(), contents);
|
||||
|
||||
if !errors.is_empty() {
|
||||
return Err(ErrorKind::ImportParseError {
|
||||
path,
|
||||
file,
|
||||
errors: errors.to_vec(),
|
||||
});
|
||||
}
|
||||
|
||||
let result = crate::compiler::compile(
|
||||
&parsed.tree().expr().unwrap(),
|
||||
Some(path.clone()),
|
||||
file,
|
||||
// The VM must ensure that a strong reference to the
|
||||
// globals outlives any self-references (which are
|
||||
// weak) embedded within the globals. If the
|
||||
// expect() below panics, it means that did not
|
||||
// happen.
|
||||
globals
|
||||
.upgrade()
|
||||
.expect("globals dropped while still in use"),
|
||||
&mut NoOpObserver::default(),
|
||||
)
|
||||
.map_err(|err| ErrorKind::ImportCompilerError {
|
||||
path: path.clone(),
|
||||
errors: vec![err],
|
||||
})?;
|
||||
|
||||
if !result.errors.is_empty() {
|
||||
return Err(ErrorKind::ImportCompilerError {
|
||||
path,
|
||||
errors: result.errors,
|
||||
});
|
||||
}
|
||||
|
||||
// Compilation succeeded, we can construct a thunk from whatever it spat
|
||||
// out and return that.
|
||||
let res = Value::Thunk(Thunk::new_suspended(result.lambda, current_span));
|
||||
|
||||
vm.import_cache.insert(path, res.clone());
|
||||
|
||||
for warning in result.warnings {
|
||||
vm.push_warning(warning);
|
||||
}
|
||||
|
||||
Ok(res)
|
||||
Some("Import the given file and return the Nix value it evaluates to"),
|
||||
1,
|
||||
move |args| {
|
||||
Gen::new(|co| pin_generator(import_impl(co, globals.clone(), source.clone(), args)))
|
||||
},
|
||||
)
|
||||
}
|
||||
|
|
|
@ -1021,6 +1021,12 @@ impl Compiler<'_> {
|
|||
// lambda as a constant.
|
||||
let mut compiled = self.contexts.pop().unwrap();
|
||||
|
||||
// Emit an instruction to inform the VM that the chunk has ended.
|
||||
compiled
|
||||
.lambda
|
||||
.chunk
|
||||
.push_op(OpCode::OpReturn, self.span_for(node));
|
||||
|
||||
// Capturing the with stack counts as an upvalue, as it is
|
||||
// emitted as an upvalue data instruction.
|
||||
if compiled.captures_with_stack {
|
||||
|
@ -1433,6 +1439,7 @@ pub fn compile(
|
|||
// unevaluated state (though in practice, a value *containing* a
|
||||
// thunk might be returned).
|
||||
c.emit_force(expr);
|
||||
c.push_op(OpCode::OpReturn, &root_span);
|
||||
|
||||
let lambda = Rc::new(c.contexts.pop().unwrap().lambda);
|
||||
c.observer.observe_compiled_toplevel(&lambda);
|
||||
|
|
|
@ -365,7 +365,6 @@ to a missing value in the attribute set(s) included via `with`."#,
|
|||
|
||||
ErrorKind::NotCoercibleToString { kind, from } => {
|
||||
let kindly = match kind {
|
||||
CoercionKind::ThunksOnly => "thunksonly",
|
||||
CoercionKind::Strong => "strongly",
|
||||
CoercionKind::Weak => "weakly",
|
||||
};
|
||||
|
|
|
@ -52,13 +52,11 @@ pub use crate::errors::{AddContext, Error, ErrorKind, EvalResult};
|
|||
pub use crate::io::{DummyIO, EvalIO, FileType};
|
||||
pub use crate::pretty_ast::pretty_print_expr;
|
||||
pub use crate::source::SourceCode;
|
||||
pub use crate::vm::VM;
|
||||
pub use crate::vm::{generators, VM};
|
||||
pub use crate::warnings::{EvalWarning, WarningKind};
|
||||
pub use builtin_macros;
|
||||
|
||||
pub use crate::value::{
|
||||
Builtin, BuiltinArgument, CoercionKind, NixAttrs, NixList, NixString, Value,
|
||||
};
|
||||
pub use crate::value::{Builtin, CoercionKind, NixAttrs, NixList, NixString, Value};
|
||||
|
||||
#[cfg(feature = "impure")]
|
||||
pub use crate::io::StdIO;
|
||||
|
|
|
@ -11,9 +11,9 @@ use std::rc::Rc;
|
|||
use tabwriter::TabWriter;
|
||||
|
||||
use crate::chunk::Chunk;
|
||||
use crate::generators::GeneratorRequest;
|
||||
use crate::opcode::{CodeIdx, OpCode};
|
||||
use crate::value::Lambda;
|
||||
use crate::vm::generators::GeneratorRequest;
|
||||
use crate::SourceCode;
|
||||
use crate::Value;
|
||||
|
||||
|
|
|
@ -154,6 +154,14 @@ pub enum OpCode {
|
|||
/// index (which must be a Value::Thunk) after the scope is fully bound.
|
||||
OpFinalise(StackIdx),
|
||||
|
||||
/// Final instruction emitted in a chunk. Does not have an
|
||||
/// inherent effect, but can simplify VM logic as a marker in some
|
||||
/// cases.
|
||||
///
|
||||
/// Can be thought of as "returning" the value to the parent
|
||||
/// frame, hence the name.
|
||||
OpReturn,
|
||||
|
||||
// [`OpClosure`], [`OpThunkSuspended`], and [`OpThunkClosure`] have a
|
||||
// variable number of arguments to the instruction, which is
|
||||
// represented here by making their data part of the opcodes.
|
||||
|
|
|
@ -10,12 +10,12 @@ mod one_offs;
|
|||
mod mock_builtins {
|
||||
//! Builtins which are required by language tests, but should not
|
||||
//! actually exist in //tvix/eval.
|
||||
use crate::generators::GenCo;
|
||||
use crate::*;
|
||||
use genawaiter::rc::Gen;
|
||||
|
||||
#[builtin("derivation")]
|
||||
fn builtin_derivation(vm: &mut VM, input: Value) -> Result<Value, ErrorKind> {
|
||||
vm.emit_warning(WarningKind::NotImplemented("builtins.derivation"));
|
||||
|
||||
async fn builtin_derivation(co: GenCo, input: Value) -> Result<Value, ErrorKind> {
|
||||
let input = input.to_attrs()?;
|
||||
let attrs = input.update(NixAttrs::from_iter(
|
||||
[
|
||||
|
|
|
@ -13,7 +13,6 @@ use serde::ser::SerializeMap;
|
|||
use serde::{Deserialize, Serialize};
|
||||
|
||||
use crate::errors::ErrorKind;
|
||||
use crate::vm::VM;
|
||||
|
||||
use super::string::NixString;
|
||||
use super::thunk::ThunkSet;
|
||||
|
@ -394,72 +393,6 @@ impl NixAttrs {
|
|||
pub(crate) fn from_kv(name: Value, value: Value) -> Self {
|
||||
NixAttrs(AttrsRep::KV { name, value })
|
||||
}
|
||||
|
||||
/// Compare `self` against `other` for equality using Nix equality semantics
|
||||
pub fn nix_eq(&self, other: &Self, vm: &mut VM) -> Result<bool, ErrorKind> {
|
||||
match (&self.0, &other.0) {
|
||||
(AttrsRep::Empty, AttrsRep::Empty) => Ok(true),
|
||||
|
||||
// It is possible to create an empty attribute set that
|
||||
// has Map representation like so: ` { ${null} = 1; }`.
|
||||
//
|
||||
// Preventing this would incur a cost on all attribute set
|
||||
// construction (we'd have to check the actual number of
|
||||
// elements after key construction). In practice this
|
||||
// probably does not happen, so it's better to just bite
|
||||
// the bullet and implement this branch.
|
||||
(AttrsRep::Empty, AttrsRep::Im(map)) | (AttrsRep::Im(map), AttrsRep::Empty) => {
|
||||
Ok(map.is_empty())
|
||||
}
|
||||
|
||||
// Other specialised representations (KV ...) definitely
|
||||
// do not match `Empty`.
|
||||
(AttrsRep::Empty, _) | (_, AttrsRep::Empty) => Ok(false),
|
||||
|
||||
(
|
||||
AttrsRep::KV {
|
||||
name: n1,
|
||||
value: v1,
|
||||
},
|
||||
AttrsRep::KV {
|
||||
name: n2,
|
||||
value: v2,
|
||||
},
|
||||
) => Ok(n1.nix_eq(n2, vm)? && v1.nix_eq(v2, vm)?),
|
||||
|
||||
(AttrsRep::Im(map), AttrsRep::KV { name, value })
|
||||
| (AttrsRep::KV { name, value }, AttrsRep::Im(map)) => {
|
||||
if map.len() != 2 {
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
if let (Some(m_name), Some(m_value)) =
|
||||
(map.get(&NixString::NAME), map.get(&NixString::VALUE))
|
||||
{
|
||||
return Ok(name.nix_eq(m_name, vm)? && value.nix_eq(m_value, vm)?);
|
||||
}
|
||||
|
||||
Ok(false)
|
||||
}
|
||||
|
||||
(AttrsRep::Im(m1), AttrsRep::Im(m2)) => {
|
||||
if m1.len() != m2.len() {
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
for (k, v1) in m1 {
|
||||
if let Some(v2) = m2.get(k) {
|
||||
if !v1.nix_eq(v2, vm)? {
|
||||
return Ok(false);
|
||||
}
|
||||
} else {
|
||||
return Ok(false);
|
||||
}
|
||||
}
|
||||
Ok(true)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// In Nix, name/value attribute pairs are frequently constructed from
|
||||
|
|
|
@ -1,57 +1,5 @@
|
|||
use super::*;
|
||||
|
||||
mod nix_eq {
|
||||
use crate::observer::NoOpObserver;
|
||||
|
||||
use super::*;
|
||||
use proptest::prelude::ProptestConfig;
|
||||
use test_strategy::proptest;
|
||||
|
||||
#[proptest(ProptestConfig { cases: 2, ..Default::default() })]
|
||||
fn reflexive(x: NixAttrs) {
|
||||
let mut observer = NoOpObserver {};
|
||||
let mut vm = VM::new(
|
||||
Default::default(),
|
||||
Box::new(crate::DummyIO),
|
||||
&mut observer,
|
||||
Default::default(),
|
||||
);
|
||||
|
||||
assert!(x.nix_eq(&x, &mut vm).unwrap())
|
||||
}
|
||||
|
||||
#[proptest(ProptestConfig { cases: 2, ..Default::default() })]
|
||||
fn symmetric(x: NixAttrs, y: NixAttrs) {
|
||||
let mut observer = NoOpObserver {};
|
||||
let mut vm = VM::new(
|
||||
Default::default(),
|
||||
Box::new(crate::DummyIO),
|
||||
&mut observer,
|
||||
Default::default(),
|
||||
);
|
||||
|
||||
assert_eq!(
|
||||
x.nix_eq(&y, &mut vm).unwrap(),
|
||||
y.nix_eq(&x, &mut vm).unwrap()
|
||||
)
|
||||
}
|
||||
|
||||
#[proptest(ProptestConfig { cases: 2, ..Default::default() })]
|
||||
fn transitive(x: NixAttrs, y: NixAttrs, z: NixAttrs) {
|
||||
let mut observer = NoOpObserver {};
|
||||
let mut vm = VM::new(
|
||||
Default::default(),
|
||||
Box::new(crate::DummyIO),
|
||||
&mut observer,
|
||||
Default::default(),
|
||||
);
|
||||
|
||||
if x.nix_eq(&y, &mut vm).unwrap() && y.nix_eq(&z, &mut vm).unwrap() {
|
||||
assert!(x.nix_eq(&z, &mut vm).unwrap())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_empty_attrs() {
|
||||
let attrs = NixAttrs::construct(0, vec![]).expect("empty attr construction should succeed");
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
//!
|
||||
//! Builtins are directly backed by Rust code operating on Nix values.
|
||||
|
||||
use crate::{errors::ErrorKind, vm::VM};
|
||||
use crate::vm::generators::Generator;
|
||||
|
||||
use super::Value;
|
||||
|
||||
|
@ -12,40 +12,38 @@ use std::{
|
|||
rc::Rc,
|
||||
};
|
||||
|
||||
/// Trait for closure types of builtins implemented directly by
|
||||
/// backing Rust code.
|
||||
/// Trait for closure types of builtins.
|
||||
///
|
||||
/// Builtins declare their arity and are passed a vector with the
|
||||
/// right number of arguments. Additionally, as they might have to
|
||||
/// force the evaluation of thunks, they are passed a reference to the
|
||||
/// current VM which they can use for forcing a value.
|
||||
/// Builtins are expected to yield a generator which can be run by the VM to
|
||||
/// produce the final value.
|
||||
///
|
||||
/// Errors returned from a builtin will be annotated with the location
|
||||
/// of the call to the builtin.
|
||||
pub trait BuiltinFn: Fn(Vec<Value>, &mut VM) -> Result<Value, ErrorKind> {}
|
||||
impl<F: Fn(Vec<Value>, &mut VM) -> Result<Value, ErrorKind>> BuiltinFn for F {}
|
||||
|
||||
/// Description of a single argument passed to a builtin
|
||||
pub struct BuiltinArgument {
|
||||
/// Whether the argument should be forced before the underlying builtin function is called
|
||||
pub strict: bool,
|
||||
/// The name of the argument, to be used in docstrings and error messages
|
||||
pub name: &'static str,
|
||||
}
|
||||
/// Implementors should use the builtins-macros to create these functions
|
||||
/// instead of handling the argument-passing logic manually.
|
||||
pub trait BuiltinGen: Fn(Vec<Value>) -> Generator {}
|
||||
impl<F: Fn(Vec<Value>) -> Generator> BuiltinGen for F {}
|
||||
|
||||
#[derive(Clone)]
|
||||
pub struct BuiltinRepr {
|
||||
name: &'static str,
|
||||
/// Array of arguments to the builtin.
|
||||
arguments: &'static [BuiltinArgument],
|
||||
/// Optional documentation for the builtin.
|
||||
documentation: Option<&'static str>,
|
||||
func: Rc<dyn BuiltinFn>,
|
||||
arg_count: usize,
|
||||
|
||||
func: Rc<dyn BuiltinGen>,
|
||||
|
||||
/// Partially applied function arguments.
|
||||
partials: Vec<Value>,
|
||||
}
|
||||
|
||||
pub enum BuiltinResult {
|
||||
/// Builtin was not ready to be called (arguments missing) and remains
|
||||
/// partially applied.
|
||||
Partial(Builtin),
|
||||
|
||||
/// Builtin was called and constructed a generator that the VM must run.
|
||||
Called(Generator),
|
||||
}
|
||||
|
||||
/// Represents a single built-in function which directly executes Rust
|
||||
/// code that operates on a Nix value.
|
||||
///
|
||||
|
@ -68,16 +66,16 @@ impl From<BuiltinRepr> for Builtin {
|
|||
}
|
||||
|
||||
impl Builtin {
|
||||
pub fn new<F: BuiltinFn + 'static>(
|
||||
pub fn new<F: BuiltinGen + 'static>(
|
||||
name: &'static str,
|
||||
arguments: &'static [BuiltinArgument],
|
||||
documentation: Option<&'static str>,
|
||||
arg_count: usize,
|
||||
func: F,
|
||||
) -> Self {
|
||||
BuiltinRepr {
|
||||
name,
|
||||
arguments,
|
||||
documentation,
|
||||
arg_count,
|
||||
func: Rc::new(func),
|
||||
partials: vec![],
|
||||
}
|
||||
|
@ -92,23 +90,25 @@ impl Builtin {
|
|||
self.0.documentation
|
||||
}
|
||||
|
||||
/// Apply an additional argument to the builtin, which will either
|
||||
/// lead to execution of the function or to returning a partial
|
||||
/// builtin.
|
||||
pub fn apply(mut self, vm: &mut VM, arg: Value) -> Result<Value, ErrorKind> {
|
||||
/// Apply an additional argument to the builtin. After this, [`call`] *must*
|
||||
/// be called, otherwise it may leave the builtin in an incorrect state.
|
||||
pub fn apply_arg(&mut self, arg: Value) {
|
||||
self.0.partials.push(arg);
|
||||
|
||||
if self.0.partials.len() == self.0.arguments.len() {
|
||||
for (idx, BuiltinArgument { strict, .. }) in self.0.arguments.iter().enumerate() {
|
||||
if *strict {
|
||||
self.0.partials[idx].force(vm)?;
|
||||
}
|
||||
}
|
||||
return (self.0.func)(self.0.partials, vm);
|
||||
}
|
||||
debug_assert!(
|
||||
self.0.partials.len() <= self.0.arg_count,
|
||||
"Tvix bug: pushed too many arguments to builtin"
|
||||
);
|
||||
}
|
||||
|
||||
// Function is not yet ready to be called.
|
||||
Ok(Value::Builtin(self))
|
||||
/// Attempt to call a builtin, which will produce a generator if it is fully
|
||||
/// applied or return the builtin if it is partially applied.
|
||||
pub fn call(self) -> BuiltinResult {
|
||||
if self.0.partials.len() == self.0.arg_count {
|
||||
BuiltinResult::Called((self.0.func)(self.0.partials))
|
||||
} else {
|
||||
BuiltinResult::Partial(self)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -5,11 +5,10 @@ use imbl::{vector, Vector};
|
|||
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
use crate::errors::AddContext;
|
||||
use crate::errors::ErrorKind;
|
||||
use crate::vm::generators;
|
||||
use crate::vm::generators::GenCo;
|
||||
use crate::vm::VM;
|
||||
use crate::generators;
|
||||
use crate::generators::GenCo;
|
||||
use crate::AddContext;
|
||||
use crate::ErrorKind;
|
||||
|
||||
use super::thunk::ThunkSet;
|
||||
use super::TotalDisplay;
|
||||
|
@ -70,29 +69,6 @@ impl NixList {
|
|||
self.0.ptr_eq(&other.0)
|
||||
}
|
||||
|
||||
/// Compare `self` against `other` for equality using Nix equality semantics
|
||||
pub fn nix_eq(&self, other: &Self, vm: &mut VM) -> Result<bool, ErrorKind> {
|
||||
if self.ptr_eq(other) {
|
||||
return Ok(true);
|
||||
}
|
||||
if self.len() != other.len() {
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
for (v1, v2) in self.iter().zip(other.iter()) {
|
||||
if !v1.nix_eq(v2, vm)? {
|
||||
return Ok(false);
|
||||
}
|
||||
}
|
||||
|
||||
Ok(true)
|
||||
}
|
||||
|
||||
/// force each element of the list (shallowly), making it safe to call .get().value()
|
||||
pub fn force_elements(&self, vm: &mut VM) -> Result<(), ErrorKind> {
|
||||
self.iter().try_for_each(|v| v.force(vm).map(|_| ()))
|
||||
}
|
||||
|
||||
pub fn into_inner(self) -> Vector<Value> {
|
||||
self.0
|
||||
}
|
||||
|
|
|
@ -1,11 +1,12 @@
|
|||
//! This module implements the backing representation of runtime
|
||||
//! values in the Nix language.
|
||||
use std::cmp::Ordering;
|
||||
use std::fmt::Display;
|
||||
use std::future::Future;
|
||||
use std::num::{NonZeroI32, NonZeroUsize};
|
||||
use std::ops::Deref;
|
||||
use std::path::PathBuf;
|
||||
use std::pin::Pin;
|
||||
use std::rc::Rc;
|
||||
use std::{cell::Ref, fmt::Display};
|
||||
|
||||
use lexical_core::format::CXX_LITERAL;
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
@ -20,12 +21,12 @@ mod path;
|
|||
mod string;
|
||||
mod thunk;
|
||||
|
||||
use crate::errors::{AddContext, ErrorKind};
|
||||
use crate::errors::ErrorKind;
|
||||
use crate::opcode::StackIdx;
|
||||
use crate::vm::generators::{self, GenCo};
|
||||
use crate::vm::VM;
|
||||
use crate::AddContext;
|
||||
pub use attrs::NixAttrs;
|
||||
pub use builtin::{Builtin, BuiltinArgument};
|
||||
pub use builtin::{Builtin, BuiltinResult};
|
||||
pub(crate) use function::Formals;
|
||||
pub use function::{Closure, Lambda};
|
||||
pub use list::NixList;
|
||||
|
@ -139,8 +140,6 @@ macro_rules! gen_is {
|
|||
/// Describes what input types are allowed when coercing a `Value` to a string
|
||||
#[derive(Clone, Copy, PartialEq, Debug)]
|
||||
pub enum CoercionKind {
|
||||
/// Force thunks, but perform no other coercions.
|
||||
ThunksOnly,
|
||||
/// Only coerce already "stringly" types like strings and paths, but also
|
||||
/// coerce sets that have a `__toString` attribute. Equivalent to
|
||||
/// `!coerceMore` in C++ Nix.
|
||||
|
@ -151,26 +150,6 @@ pub enum CoercionKind {
|
|||
Strong,
|
||||
}
|
||||
|
||||
/// A reference to a [`Value`] returned by a call to [`Value::force`], whether the value was
|
||||
/// originally a thunk or not.
|
||||
///
|
||||
/// Implements [`Deref`] to [`Value`], so can generally be used as a [`Value`]
|
||||
pub enum ForceResult<'a> {
|
||||
ForcedThunk(Ref<'a, Value>),
|
||||
Immediate(&'a Value),
|
||||
}
|
||||
|
||||
impl<'a> Deref for ForceResult<'a> {
|
||||
type Target = Value;
|
||||
|
||||
fn deref(&self) -> &Self::Target {
|
||||
match self {
|
||||
ForceResult::ForcedThunk(r) => r,
|
||||
ForceResult::Immediate(v) => v,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<T> From<T> for Value
|
||||
where
|
||||
T: Into<NixString>,
|
||||
|
@ -204,33 +183,80 @@ pub enum PointerEquality {
|
|||
}
|
||||
|
||||
impl Value {
|
||||
/// Deeply forces a value, traversing e.g. lists and attribute sets and forcing
|
||||
/// their contents, too.
|
||||
///
|
||||
/// This is a generator function.
|
||||
pub(super) async fn deep_force(
|
||||
self,
|
||||
co: GenCo,
|
||||
thunk_set: SharedThunkSet,
|
||||
) -> Result<Value, ErrorKind> {
|
||||
// Get rid of any top-level thunks, and bail out of self-recursive
|
||||
// thunks.
|
||||
let value = if let Value::Thunk(ref t) = &self {
|
||||
if !thunk_set.insert(t) {
|
||||
return Ok(self);
|
||||
}
|
||||
generators::request_force(&co, self).await
|
||||
} else {
|
||||
self
|
||||
};
|
||||
|
||||
match &value {
|
||||
// Short-circuit on already evaluated values, or fail on internal values.
|
||||
Value::Null
|
||||
| Value::Bool(_)
|
||||
| Value::Integer(_)
|
||||
| Value::Float(_)
|
||||
| Value::String(_)
|
||||
| Value::Path(_)
|
||||
| Value::Closure(_)
|
||||
| Value::Builtin(_) => return Ok(value),
|
||||
|
||||
Value::List(list) => {
|
||||
for val in list {
|
||||
generators::request_deep_force(&co, val.clone(), thunk_set.clone()).await;
|
||||
}
|
||||
}
|
||||
|
||||
Value::Attrs(attrs) => {
|
||||
for (_, val) in attrs.iter() {
|
||||
generators::request_deep_force(&co, val.clone(), thunk_set.clone()).await;
|
||||
}
|
||||
}
|
||||
|
||||
Value::Thunk(_) => panic!("Tvix bug: force_value() returned a thunk"),
|
||||
|
||||
Value::AttrNotFound
|
||||
| Value::Blueprint(_)
|
||||
| Value::DeferredUpvalue(_)
|
||||
| Value::UnresolvedPath(_) => panic!(
|
||||
"Tvix bug: internal value left on stack: {}",
|
||||
value.type_of()
|
||||
),
|
||||
};
|
||||
|
||||
Ok(value)
|
||||
}
|
||||
|
||||
/// Coerce a `Value` to a string. See `CoercionKind` for a rundown of what
|
||||
/// input types are accepted under what circumstances.
|
||||
pub fn coerce_to_string(
|
||||
&self,
|
||||
kind: CoercionKind,
|
||||
vm: &mut VM,
|
||||
) -> Result<NixString, ErrorKind> {
|
||||
// TODO: eventually, this will need to handle string context and importing
|
||||
// files into the Nix store depending on what context the coercion happens in
|
||||
if let Value::Thunk(t) = self {
|
||||
t.force(vm)?;
|
||||
}
|
||||
|
||||
match (self, kind) {
|
||||
// deal with thunks
|
||||
(Value::Thunk(t), _) => t.value().coerce_to_string(kind, vm),
|
||||
pub async fn coerce_to_string(self, co: GenCo, kind: CoercionKind) -> Result<Value, ErrorKind> {
|
||||
let value = generators::request_force(&co, self).await;
|
||||
|
||||
match (value, kind) {
|
||||
// coercions that are always done
|
||||
(Value::String(s), _) => Ok(s.clone()),
|
||||
tuple @ (Value::String(_), _) => Ok(tuple.0),
|
||||
|
||||
// TODO(sterni): Think about proper encoding handling here. This needs
|
||||
// general consideration anyways, since one current discrepancy between
|
||||
// C++ Nix and Tvix is that the former's strings are arbitrary byte
|
||||
// sequences without NUL bytes, whereas Tvix only allows valid
|
||||
// Unicode. See also b/189.
|
||||
(Value::Path(p), kind) if kind != CoercionKind::ThunksOnly => {
|
||||
let imported = vm.io().import_path(p)?;
|
||||
(Value::Path(p), _) => {
|
||||
// TODO(tazjin): there are cases where coerce_to_string does not import
|
||||
let imported = generators::request_path_import(&co, p).await;
|
||||
Ok(imported.to_string_lossy().into_owned().into())
|
||||
}
|
||||
|
||||
|
@ -238,38 +264,32 @@ impl Value {
|
|||
// `__toString` attribute which holds a function that receives the
|
||||
// set itself or an `outPath` attribute which should be a string.
|
||||
// `__toString` is preferred.
|
||||
(Value::Attrs(attrs), kind) if kind != CoercionKind::ThunksOnly => {
|
||||
(Value::Attrs(attrs), kind) => {
|
||||
match (attrs.select("__toString"), attrs.select("outPath")) {
|
||||
(None, None) => Err(ErrorKind::NotCoercibleToString { from: "set", kind }),
|
||||
|
||||
(Some(f), _) => {
|
||||
// use a closure here to deal with the thunk borrow we need to do below
|
||||
let call_to_string = |value: &Value, vm: &mut VM| {
|
||||
// Leave self on the stack as an argument to the function call.
|
||||
vm.push(self.clone());
|
||||
vm.call_value(value)?;
|
||||
let result = vm.pop();
|
||||
let callable = generators::request_force(&co, f.clone()).await;
|
||||
|
||||
match result {
|
||||
Value::String(s) => Ok(s),
|
||||
// Attribute set coercion actually works
|
||||
// recursively, e.g. you can even return
|
||||
// /another/ set with a __toString attr.
|
||||
_ => result.coerce_to_string(kind, vm),
|
||||
}
|
||||
};
|
||||
// Leave the attribute set on the stack as an argument
|
||||
// to the function call.
|
||||
generators::request_stack_push(&co, Value::Attrs(attrs)).await;
|
||||
|
||||
if let Value::Thunk(t) = f {
|
||||
t.force(vm)?;
|
||||
let guard = t.value();
|
||||
call_to_string(&guard, vm)
|
||||
} else {
|
||||
call_to_string(f, vm)
|
||||
}
|
||||
// Call the callable ...
|
||||
let result = generators::request_call(&co, callable).await;
|
||||
|
||||
// Recurse on the result, as attribute set coercion
|
||||
// actually works recursively, e.g. you can even return
|
||||
// /another/ set with a __toString attr.
|
||||
let s = generators::request_string_coerce(&co, result, kind).await;
|
||||
Ok(Value::String(s))
|
||||
}
|
||||
|
||||
// Similarly to `__toString` we also coerce recursively for `outPath`
|
||||
(None, Some(s)) => s.coerce_to_string(kind, vm),
|
||||
(None, Some(s)) => {
|
||||
let s = generators::request_string_coerce(&co, s.clone(), kind).await;
|
||||
Ok(Value::String(s))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -287,30 +307,31 @@ impl Value {
|
|||
}
|
||||
|
||||
// Lists are coerced by coercing their elements and interspersing spaces
|
||||
(Value::List(l), CoercionKind::Strong) => {
|
||||
// TODO(sterni): use intersperse when it becomes available?
|
||||
// https://github.com/rust-lang/rust/issues/79524
|
||||
l.iter()
|
||||
.map(|v| v.coerce_to_string(kind, vm))
|
||||
.reduce(|acc, string| {
|
||||
let a = acc?;
|
||||
let s = &string?;
|
||||
Ok(a.concat(&" ".into()).concat(s))
|
||||
})
|
||||
// None from reduce indicates empty iterator
|
||||
.unwrap_or_else(|| Ok("".into()))
|
||||
(Value::List(list), CoercionKind::Strong) => {
|
||||
let mut out = String::new();
|
||||
|
||||
for (idx, elem) in list.into_iter().enumerate() {
|
||||
if idx > 0 {
|
||||
out.push(' ');
|
||||
}
|
||||
|
||||
let s = generators::request_string_coerce(&co, elem, kind).await;
|
||||
out.push_str(s.as_str());
|
||||
}
|
||||
|
||||
Ok(Value::String(out.into()))
|
||||
}
|
||||
|
||||
(Value::Path(_), _)
|
||||
| (Value::Attrs(_), _)
|
||||
| (Value::Closure(_), _)
|
||||
| (Value::Builtin(_), _)
|
||||
| (Value::Null, _)
|
||||
| (Value::Bool(_), _)
|
||||
| (Value::Integer(_), _)
|
||||
| (Value::Float(_), _)
|
||||
| (Value::List(_), _) => Err(ErrorKind::NotCoercibleToString {
|
||||
from: self.type_of(),
|
||||
(Value::Thunk(_), _) => panic!("Tvix bug: force returned unforced thunk"),
|
||||
|
||||
val @ (Value::Closure(_), _)
|
||||
| val @ (Value::Builtin(_), _)
|
||||
| val @ (Value::Null, _)
|
||||
| val @ (Value::Bool(_), _)
|
||||
| val @ (Value::Integer(_), _)
|
||||
| val @ (Value::Float(_), _)
|
||||
| val @ (Value::List(_), _) => Err(ErrorKind::NotCoercibleToString {
|
||||
from: val.0.type_of(),
|
||||
kind,
|
||||
}),
|
||||
|
||||
|
@ -333,7 +354,7 @@ impl Value {
|
|||
/// The `top_level` parameter controls whether this invocation is the top-level
|
||||
/// comparison, or a nested value comparison. See
|
||||
/// `//tvix/docs/value-pointer-equality.md`
|
||||
pub(crate) async fn neo_nix_eq(
|
||||
pub(crate) async fn nix_eq(
|
||||
self,
|
||||
other: Value,
|
||||
co: GenCo,
|
||||
|
@ -530,49 +551,52 @@ impl Value {
|
|||
gen_is!(is_number, Value::Integer(_) | Value::Float(_));
|
||||
gen_is!(is_bool, Value::Bool(_));
|
||||
|
||||
/// Compare `self` against `other` for equality using Nix equality semantics.
|
||||
///
|
||||
/// Takes a reference to the `VM` to allow forcing thunks during comparison
|
||||
pub fn nix_eq(&self, other: &Self, vm: &mut VM) -> Result<bool, ErrorKind> {
|
||||
match (self, other) {
|
||||
// Trivial comparisons
|
||||
(Value::Null, Value::Null) => Ok(true),
|
||||
(Value::Bool(b1), Value::Bool(b2)) => Ok(b1 == b2),
|
||||
(Value::String(s1), Value::String(s2)) => Ok(s1 == s2),
|
||||
(Value::Path(p1), Value::Path(p2)) => Ok(p1 == p2),
|
||||
|
||||
// Numerical comparisons (they work between float & int)
|
||||
(Value::Integer(i1), Value::Integer(i2)) => Ok(i1 == i2),
|
||||
(Value::Integer(i), Value::Float(f)) => Ok(*i as f64 == *f),
|
||||
(Value::Float(f1), Value::Float(f2)) => Ok(f1 == f2),
|
||||
(Value::Float(f), Value::Integer(i)) => Ok(*i as f64 == *f),
|
||||
|
||||
(Value::Attrs(_), Value::Attrs(_))
|
||||
| (Value::List(_), Value::List(_))
|
||||
| (Value::Thunk(_), _)
|
||||
| (_, Value::Thunk(_)) => Ok(vm.nix_eq(self.clone(), other.clone(), false)?),
|
||||
|
||||
// Everything else is either incomparable (e.g. internal
|
||||
// types) or false.
|
||||
_ => Ok(false),
|
||||
}
|
||||
/// Internal helper to allow `nix_cmp_ordering` to recurse.
|
||||
fn nix_cmp_boxed(
|
||||
self,
|
||||
other: Self,
|
||||
co: GenCo,
|
||||
) -> Pin<Box<dyn Future<Output = Result<Option<Ordering>, ErrorKind>>>> {
|
||||
Box::pin(self.nix_cmp_ordering(other, co))
|
||||
}
|
||||
|
||||
/// Compare `self` against other using (fallible) Nix ordering semantics.
|
||||
pub fn nix_cmp(&self, other: &Self, vm: &mut VM) -> Result<Option<Ordering>, ErrorKind> {
|
||||
///
|
||||
/// Note that as this returns an `Option<Ordering>` it can not directly be
|
||||
/// used as a generator function in the VM. The exact use depends on the
|
||||
/// callsite, as the meaning is interpreted in different ways e.g. based on
|
||||
/// the comparison operator used.
|
||||
///
|
||||
/// The function is intended to be used from within other generator
|
||||
/// functions or `gen!` blocks.
|
||||
pub async fn nix_cmp_ordering(
|
||||
self,
|
||||
other: Self,
|
||||
co: GenCo,
|
||||
) -> Result<Option<Ordering>, ErrorKind> {
|
||||
match (self, other) {
|
||||
// same types
|
||||
(Value::Integer(i1), Value::Integer(i2)) => Ok(i1.partial_cmp(i2)),
|
||||
(Value::Float(f1), Value::Float(f2)) => Ok(f1.partial_cmp(f2)),
|
||||
(Value::String(s1), Value::String(s2)) => Ok(s1.partial_cmp(s2)),
|
||||
(Value::Integer(i1), Value::Integer(i2)) => Ok(i1.partial_cmp(&i2)),
|
||||
(Value::Float(f1), Value::Float(f2)) => Ok(f1.partial_cmp(&f2)),
|
||||
(Value::String(s1), Value::String(s2)) => Ok(s1.partial_cmp(&s2)),
|
||||
(Value::List(l1), Value::List(l2)) => {
|
||||
for i in 0.. {
|
||||
if i == l2.len() {
|
||||
return Ok(Some(Ordering::Greater));
|
||||
} else if i == l1.len() {
|
||||
return Ok(Some(Ordering::Less));
|
||||
} else if !vm.nix_eq(l1[i].clone(), l2[i].clone(), true)? {
|
||||
return l1[i].force(vm)?.nix_cmp(&*l2[i].force(vm)?, vm);
|
||||
} else if !generators::check_equality(
|
||||
&co,
|
||||
l1[i].clone(),
|
||||
l2[i].clone(),
|
||||
PointerEquality::AllowAll,
|
||||
)
|
||||
.await?
|
||||
{
|
||||
// TODO: do we need to control `top_level` here?
|
||||
let v1 = generators::request_force(&co, l1[i].clone()).await;
|
||||
let v2 = generators::request_force(&co, l2[i].clone()).await;
|
||||
return v1.nix_cmp_boxed(v2, co).await;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -580,8 +604,8 @@ impl Value {
|
|||
}
|
||||
|
||||
// different types
|
||||
(Value::Integer(i1), Value::Float(f2)) => Ok((*i1 as f64).partial_cmp(f2)),
|
||||
(Value::Float(f1), Value::Integer(i2)) => Ok(f1.partial_cmp(&(*i2 as f64))),
|
||||
(Value::Integer(i1), Value::Float(f2)) => Ok((i1 as f64).partial_cmp(&f2)),
|
||||
(Value::Float(f1), Value::Integer(i2)) => Ok(f1.partial_cmp(&(i2 as f64))),
|
||||
|
||||
// unsupported types
|
||||
(lhs, rhs) => Err(ErrorKind::Incomparable {
|
||||
|
@ -591,58 +615,12 @@ impl Value {
|
|||
}
|
||||
}
|
||||
|
||||
/// Ensure `self` is forced if it is a thunk, and return a reference to the resulting value.
|
||||
pub fn force(&self, vm: &mut VM) -> Result<ForceResult, ErrorKind> {
|
||||
match self {
|
||||
Self::Thunk(thunk) => {
|
||||
thunk.force(vm)?;
|
||||
Ok(ForceResult::ForcedThunk(thunk.value()))
|
||||
}
|
||||
_ => Ok(ForceResult::Immediate(self)),
|
||||
pub async fn force(self, co: GenCo) -> Result<Value, ErrorKind> {
|
||||
if let Value::Thunk(thunk) = self {
|
||||
return thunk.force(co).await;
|
||||
}
|
||||
}
|
||||
|
||||
/// Ensure `self` is *deeply* forced, including all recursive sub-values
|
||||
pub(crate) fn deep_force(
|
||||
&self,
|
||||
vm: &mut VM,
|
||||
thunk_set: &mut ThunkSet,
|
||||
) -> Result<(), ErrorKind> {
|
||||
match self {
|
||||
Value::Null
|
||||
| Value::Bool(_)
|
||||
| Value::Integer(_)
|
||||
| Value::Float(_)
|
||||
| Value::String(_)
|
||||
| Value::Path(_)
|
||||
| Value::Closure(_)
|
||||
| Value::Builtin(_)
|
||||
| Value::AttrNotFound
|
||||
| Value::Blueprint(_)
|
||||
| Value::DeferredUpvalue(_)
|
||||
| Value::UnresolvedPath(_) => Ok(()),
|
||||
Value::Attrs(a) => {
|
||||
for (_, v) in a.iter() {
|
||||
v.deep_force(vm, thunk_set)?;
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
Value::List(l) => {
|
||||
for val in l {
|
||||
val.deep_force(vm, thunk_set)?;
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
Value::Thunk(thunk) => {
|
||||
if !thunk_set.insert(thunk) {
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
thunk.force(vm)?;
|
||||
let value = thunk.value().clone();
|
||||
value.deep_force(vm, thunk_set)
|
||||
}
|
||||
}
|
||||
Ok(self)
|
||||
}
|
||||
|
||||
/// Explain a value in a human-readable way, e.g. by presenting
|
||||
|
@ -835,9 +813,6 @@ fn type_error(expected: &'static str, actual: &Value) -> ErrorKind {
|
|||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use imbl::vector;
|
||||
|
||||
mod floats {
|
||||
use crate::value::total_fmt_float;
|
||||
|
||||
|
@ -865,72 +840,4 @@ mod tests {
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
mod nix_eq {
|
||||
use crate::observer::NoOpObserver;
|
||||
|
||||
use super::*;
|
||||
use proptest::prelude::ProptestConfig;
|
||||
use test_strategy::proptest;
|
||||
|
||||
#[proptest(ProptestConfig { cases: 5, ..Default::default() })]
|
||||
fn reflexive(x: Value) {
|
||||
let mut observer = NoOpObserver {};
|
||||
let mut vm = VM::new(
|
||||
Default::default(),
|
||||
Box::new(crate::DummyIO),
|
||||
&mut observer,
|
||||
Default::default(),
|
||||
);
|
||||
|
||||
assert!(x.nix_eq(&x, &mut vm).unwrap())
|
||||
}
|
||||
|
||||
#[proptest(ProptestConfig { cases: 5, ..Default::default() })]
|
||||
fn symmetric(x: Value, y: Value) {
|
||||
let mut observer = NoOpObserver {};
|
||||
let mut vm = VM::new(
|
||||
Default::default(),
|
||||
Box::new(crate::DummyIO),
|
||||
&mut observer,
|
||||
Default::default(),
|
||||
);
|
||||
|
||||
assert_eq!(
|
||||
x.nix_eq(&y, &mut vm).unwrap(),
|
||||
y.nix_eq(&x, &mut vm).unwrap()
|
||||
)
|
||||
}
|
||||
|
||||
#[proptest(ProptestConfig { cases: 5, ..Default::default() })]
|
||||
fn transitive(x: Value, y: Value, z: Value) {
|
||||
let mut observer = NoOpObserver {};
|
||||
let mut vm = VM::new(
|
||||
Default::default(),
|
||||
Box::new(crate::DummyIO),
|
||||
&mut observer,
|
||||
Default::default(),
|
||||
);
|
||||
|
||||
if x.nix_eq(&y, &mut vm).unwrap() && y.nix_eq(&z, &mut vm).unwrap() {
|
||||
assert!(x.nix_eq(&z, &mut vm).unwrap())
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn list_int_float_fungibility() {
|
||||
let mut observer = NoOpObserver {};
|
||||
let mut vm = VM::new(
|
||||
Default::default(),
|
||||
Box::new(crate::DummyIO),
|
||||
&mut observer,
|
||||
Default::default(),
|
||||
);
|
||||
|
||||
let v1 = Value::List(NixList::from(vector![Value::Integer(1)]));
|
||||
let v2 = Value::List(NixList::from(vector![Value::Float(1.0)]));
|
||||
|
||||
assert!(v1.nix_eq(&v2, &mut vm).unwrap())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -28,11 +28,11 @@ use std::{
|
|||
use serde::Serialize;
|
||||
|
||||
use crate::{
|
||||
errors::{Error, ErrorKind},
|
||||
errors::ErrorKind,
|
||||
spans::LightSpan,
|
||||
upvalues::Upvalues,
|
||||
value::Closure,
|
||||
vm::{Trampoline, TrampolineAction, VM},
|
||||
vm::generators::{self, GenCo},
|
||||
Value,
|
||||
};
|
||||
|
||||
|
@ -115,78 +115,16 @@ impl Thunk {
|
|||
)))))
|
||||
}
|
||||
|
||||
/// Force a thunk from a context that can't handle trampoline
|
||||
/// continuations, eg outside the VM's normal execution loop. Calling
|
||||
/// `force_trampoline()` instead should be preferred whenever possible.
|
||||
pub fn force(&self, vm: &mut VM) -> Result<(), ErrorKind> {
|
||||
pub async fn force(self, co: GenCo) -> Result<Value, ErrorKind> {
|
||||
// If the current thunk is already fully evaluated, return its evaluated
|
||||
// value. The VM will continue running the code that landed us here.
|
||||
if self.is_forced() {
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
let mut trampoline = Self::force_trampoline(vm, Value::Thunk(self.clone()))?;
|
||||
loop {
|
||||
match trampoline.action {
|
||||
None => (),
|
||||
Some(TrampolineAction::EnterFrame {
|
||||
lambda,
|
||||
upvalues,
|
||||
arg_count,
|
||||
light_span: _,
|
||||
}) => vm.enter_frame(lambda, upvalues, arg_count)?,
|
||||
}
|
||||
match trampoline.continuation {
|
||||
None => break,
|
||||
Some(cont) => {
|
||||
trampoline = cont(vm)?;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
}
|
||||
vm.pop();
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Evaluate the content of a thunk, potentially repeatedly, until a
|
||||
/// non-thunk value is returned.
|
||||
///
|
||||
/// When this function returns, the result of one "round" of forcing is left
|
||||
/// at the top of the stack. This may still be a partially evaluated thunk
|
||||
/// which must be further run through the trampoline.
|
||||
pub fn force_trampoline(vm: &mut VM, outer: Value) -> Result<Trampoline, ErrorKind> {
|
||||
match outer {
|
||||
Value::Thunk(thunk) => thunk.force_trampoline_self(vm),
|
||||
v => {
|
||||
vm.push(v);
|
||||
Ok(Trampoline::default())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Analyses `self` and, upon finding a suspended thunk, requests evaluation
|
||||
/// of the contained code from the VM. Control flow may pass back and forth
|
||||
/// between this function and the VM multiple times through continuations
|
||||
/// that call `force_trampoline` again if nested thunks are encountered.
|
||||
///
|
||||
/// This function is entered again by returning a continuation that calls
|
||||
/// [force_trampoline].
|
||||
// When working on this function, care should be taken to ensure that each
|
||||
// evaluated thunk's *own copy* of its inner representation is replaced by
|
||||
// evaluated results and blackholes, as appropriate. It is a critical error
|
||||
// to move the representation of one thunk into another and can lead to
|
||||
// hard-to-debug performance issues.
|
||||
// TODO: check Rc count when replacing inner repr, to skip it optionally
|
||||
fn force_trampoline_self(&self, vm: &mut VM) -> Result<Trampoline, ErrorKind> {
|
||||
// If the current thunk is already fully evaluated, leave its evaluated
|
||||
// value on the stack and return an empty trampoline. The VM will
|
||||
// continue running the code that landed us here.
|
||||
if self.is_forced() {
|
||||
vm.push(self.value().clone());
|
||||
return Ok(Trampoline::default());
|
||||
return Ok(self.value().clone());
|
||||
}
|
||||
|
||||
// Begin evaluation of this thunk by marking it as a blackhole, meaning
|
||||
// that any other trampoline loop round encountering this thunk before
|
||||
// its evaluation is completed detected an evaluation cycle.
|
||||
// that any other forcing frame encountering this thunk before its
|
||||
// evaluation is completed detected an evaluation cycle.
|
||||
let inner = self.0.replace(ThunkRepr::Blackhole);
|
||||
|
||||
match inner {
|
||||
|
@ -195,171 +133,44 @@ impl Thunk {
|
|||
ThunkRepr::Blackhole => Err(ErrorKind::InfiniteRecursion),
|
||||
|
||||
// If there is a native function stored in the thunk, evaluate it
|
||||
// and replace this thunk's representation with it. Then bounces off
|
||||
// the trampoline, to handle the case of the native function
|
||||
// returning another thunk.
|
||||
// and replace this thunk's representation with the result.
|
||||
ThunkRepr::Native(native) => {
|
||||
let value = native.0()?;
|
||||
self.0.replace(ThunkRepr::Evaluated(value));
|
||||
let self_clone = self.clone();
|
||||
|
||||
Ok(Trampoline {
|
||||
action: None,
|
||||
continuation: Some(Box::new(move |vm| {
|
||||
Thunk::force_trampoline(vm, Value::Thunk(self_clone))
|
||||
.map_err(|kind| Error::new(kind, todo!("BUG: b/238")))
|
||||
})),
|
||||
})
|
||||
// Force the returned value again, in case the native call
|
||||
// returned a thunk.
|
||||
let value = generators::request_force(&co, value).await;
|
||||
|
||||
self.0.replace(ThunkRepr::Evaluated(value.clone()));
|
||||
Ok(value)
|
||||
}
|
||||
|
||||
// When encountering a suspended thunk, construct a trampoline that
|
||||
// enters the thunk's code in the VM and replaces the thunks
|
||||
// representation with the evaluated one upon return.
|
||||
// When encountering a suspended thunk, request that the VM enters
|
||||
// it and produces the result.
|
||||
//
|
||||
// Thunks may be nested, so this case initiates another round of
|
||||
// trampolining to ensure that the returned value is forced.
|
||||
ThunkRepr::Suspended {
|
||||
lambda,
|
||||
upvalues,
|
||||
light_span,
|
||||
} => {
|
||||
// Clone self to move an Rc pointing to *this* thunk instance
|
||||
// into the continuation closure.
|
||||
let self_clone = self.clone();
|
||||
let value =
|
||||
generators::request_enter_lambda(&co, lambda, upvalues, light_span).await;
|
||||
|
||||
Ok(Trampoline {
|
||||
// Ask VM to enter frame of this thunk ...
|
||||
action: Some(TrampolineAction::EnterFrame {
|
||||
lambda,
|
||||
upvalues,
|
||||
arg_count: 0,
|
||||
light_span: light_span.clone(),
|
||||
}),
|
||||
// This may have returned another thunk, so we need to request
|
||||
// that the VM forces this value, too.
|
||||
let value = generators::request_force(&co, value).await;
|
||||
|
||||
// ... and replace the inner representation once that is done,
|
||||
// looping back around to here.
|
||||
continuation: Some(Box::new(move |vm: &mut VM| {
|
||||
let should_be_blackhole =
|
||||
self_clone.0.replace(ThunkRepr::Evaluated(vm.pop()));
|
||||
debug_assert!(matches!(should_be_blackhole, ThunkRepr::Blackhole));
|
||||
|
||||
Thunk::force_trampoline(vm, Value::Thunk(self_clone))
|
||||
.map_err(|kind| Error::new(kind, light_span.span()))
|
||||
})),
|
||||
})
|
||||
self.0.replace(ThunkRepr::Evaluated(value.clone()));
|
||||
Ok(value)
|
||||
}
|
||||
|
||||
// Note by tazjin: I have decided at this point to fully unroll the inner thunk handling
|
||||
// here, leaving no room for confusion about how inner thunks are handled. This *could*
|
||||
// be written in a shorter way (for example by using a helper function that handles all
|
||||
// cases in which inner thunks can trivially be turned into a value), but given that we
|
||||
// have been bitten by this logic repeatedly, I think it is better to let it be slightly
|
||||
// verbose for now.
|
||||
|
||||
// If an inner thunk is found and already fully-forced, we can
|
||||
// short-circuit and replace the representation of self with it.
|
||||
ThunkRepr::Evaluated(Value::Thunk(ref inner)) if inner.is_forced() => {
|
||||
self.0.replace(ThunkRepr::Evaluated(inner.value().clone()));
|
||||
vm.push(inner.value().clone());
|
||||
Ok(Trampoline::default())
|
||||
// If an inner value is found, force it and then update. This is
|
||||
// most likely an inner thunk, as `Thunk:is_forced` returned false.
|
||||
ThunkRepr::Evaluated(val) => {
|
||||
let value = generators::request_force(&co, val).await;
|
||||
self.0.replace(ThunkRepr::Evaluated(value.clone()));
|
||||
Ok(value)
|
||||
}
|
||||
|
||||
// Otherwise we handle inner thunks mostly as above, with the
|
||||
// primary difference that we set the representations of *both*
|
||||
// thunks in this case.
|
||||
ThunkRepr::Evaluated(Value::Thunk(ref inner)) => {
|
||||
// The inner thunk is now under evaluation, mark it as such.
|
||||
let inner_repr = inner.0.replace(ThunkRepr::Blackhole);
|
||||
|
||||
match inner_repr {
|
||||
ThunkRepr::Blackhole => Err(ErrorKind::InfiniteRecursion),
|
||||
|
||||
// Same as for the native case above, but results are placed
|
||||
// in *both* thunks.
|
||||
ThunkRepr::Native(native) => {
|
||||
let value = native.0()?;
|
||||
self.0.replace(ThunkRepr::Evaluated(value.clone()));
|
||||
inner.0.replace(ThunkRepr::Evaluated(value));
|
||||
let self_clone = self.clone();
|
||||
|
||||
Ok(Trampoline {
|
||||
action: None,
|
||||
continuation: Some(Box::new(move |vm| {
|
||||
Thunk::force_trampoline(vm, Value::Thunk(self_clone))
|
||||
.map_err(|kind| Error::new(kind, todo!("BUG: b/238")))
|
||||
})),
|
||||
})
|
||||
}
|
||||
|
||||
// Inner suspended thunks are trampolined to the VM, and
|
||||
// their results written to both thunks in the continuation.
|
||||
ThunkRepr::Suspended {
|
||||
lambda,
|
||||
upvalues,
|
||||
light_span,
|
||||
} => {
|
||||
let self_clone = self.clone();
|
||||
let inner_clone = inner.clone();
|
||||
|
||||
Ok(Trampoline {
|
||||
// Ask VM to enter frame of this thunk ...
|
||||
action: Some(TrampolineAction::EnterFrame {
|
||||
lambda,
|
||||
upvalues,
|
||||
arg_count: 0,
|
||||
light_span: light_span.clone(),
|
||||
}),
|
||||
|
||||
// ... and replace the inner representations.
|
||||
continuation: Some(Box::new(move |vm: &mut VM| {
|
||||
let result = vm.pop();
|
||||
|
||||
let self_blackhole =
|
||||
self_clone.0.replace(ThunkRepr::Evaluated(result.clone()));
|
||||
debug_assert!(matches!(self_blackhole, ThunkRepr::Blackhole));
|
||||
|
||||
let inner_blackhole =
|
||||
inner_clone.0.replace(ThunkRepr::Evaluated(result));
|
||||
debug_assert!(matches!(inner_blackhole, ThunkRepr::Blackhole));
|
||||
|
||||
Thunk::force_trampoline(vm, Value::Thunk(self_clone))
|
||||
.map_err(|kind| Error::new(kind, light_span.span()))
|
||||
})),
|
||||
})
|
||||
}
|
||||
|
||||
// If the inner thunk is some arbitrary other value (this is
|
||||
// almost guaranteed to be another thunk), change our
|
||||
// representation to the same inner thunk and bounce off the
|
||||
// trampoline. The inner thunk is changed *back* to the same
|
||||
// state.
|
||||
//
|
||||
// This is safe because we are not cloning the innermost
|
||||
// thunk's representation, so while the inner thunk will not
|
||||
// eventually have its representation replaced by _this_
|
||||
// trampoline run, we will return the correct representation
|
||||
// out of here and memoize the innermost thunk.
|
||||
ThunkRepr::Evaluated(v) => {
|
||||
self.0.replace(ThunkRepr::Evaluated(v.clone()));
|
||||
inner.0.replace(ThunkRepr::Evaluated(v));
|
||||
let self_clone = self.clone();
|
||||
|
||||
Ok(Trampoline {
|
||||
action: None,
|
||||
continuation: Some(Box::new(move |vm: &mut VM| {
|
||||
// TODO(tazjin): not sure about this span ...
|
||||
// let span = vm.current_span();
|
||||
Thunk::force_trampoline(vm, Value::Thunk(self_clone))
|
||||
.map_err(|kind| Error::new(kind, todo!("BUG: b/238")))
|
||||
})),
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// This branch can not occur here, it would have been caught by our
|
||||
// `self.is_forced()` check above.
|
||||
ThunkRepr::Evaluated(_) => unreachable!("BUG: definition of Thunk::is_forced changed"),
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -381,7 +192,6 @@ impl Thunk {
|
|||
/// Returns true if forcing this thunk will not change it.
|
||||
pub fn is_forced(&self) -> bool {
|
||||
match *self.0.borrow() {
|
||||
ThunkRepr::Blackhole => panic!("is_forced() called on a blackholed thunk"),
|
||||
ThunkRepr::Evaluated(Value::Thunk(_)) => false,
|
||||
ThunkRepr::Evaluated(_) => true,
|
||||
_ => false,
|
||||
|
@ -452,13 +262,9 @@ impl TotalDisplay for Thunk {
|
|||
return f.write_str("<CYCLE>");
|
||||
}
|
||||
|
||||
match self.0.try_borrow() {
|
||||
Ok(repr) => match &*repr {
|
||||
ThunkRepr::Evaluated(v) => v.total_fmt(f, set),
|
||||
_ => f.write_str("internal[thunk]"),
|
||||
},
|
||||
|
||||
_ => f.write_str("internal[thunk]"),
|
||||
match &*self.0.borrow() {
|
||||
ThunkRepr::Evaluated(v) => v.total_fmt(f, set),
|
||||
other => write!(f, "internal[{}]", other.debug_repr()),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
1218
tvix/eval/src/vm.rs
1218
tvix/eval/src/vm.rs
File diff suppressed because it is too large
Load diff
|
@ -136,7 +136,6 @@ impl Display for GeneratorRequest {
|
|||
GeneratorRequest::StringCoerce(v, kind) => match kind {
|
||||
CoercionKind::Weak => write!(f, "weak_string_coerce({})", v),
|
||||
CoercionKind::Strong => write!(f, "strong_string_coerce({})", v),
|
||||
CoercionKind::ThunksOnly => todo!("remove this branch (not live)"),
|
||||
},
|
||||
GeneratorRequest::Call(v) => write!(f, "call({})", v),
|
||||
GeneratorRequest::EnterLambda { lambda, .. } => {
|
||||
|
@ -211,6 +210,240 @@ pub fn pin_generator(
|
|||
Box::pin(f)
|
||||
}
|
||||
|
||||
impl<'o> VM<'o> {
|
||||
/// Helper function to re-enqueue the current generator while it
|
||||
/// is awaiting a value.
|
||||
fn reenqueue_generator(&mut self, span: LightSpan, generator: Generator) {
|
||||
self.frames.push(Frame::Generator {
|
||||
generator,
|
||||
span,
|
||||
state: GeneratorState::AwaitingValue,
|
||||
});
|
||||
}
|
||||
|
||||
/// Helper function to enqueue a new generator.
|
||||
pub(super) fn enqueue_generator<F, G>(&mut self, span: LightSpan, gen: G)
|
||||
where
|
||||
F: Future<Output = Result<Value, ErrorKind>> + 'static,
|
||||
G: FnOnce(GenCo) -> F,
|
||||
{
|
||||
self.frames.push(Frame::Generator {
|
||||
span,
|
||||
state: GeneratorState::Running,
|
||||
generator: Gen::new(|co| pin_generator(gen(co))),
|
||||
});
|
||||
}
|
||||
|
||||
/// Run a generator frame until it yields to the outer control loop, or runs
|
||||
/// to completion.
|
||||
///
|
||||
/// The return value indicates whether the generator has completed (true),
|
||||
/// or was suspended (false).
|
||||
pub(crate) fn run_generator(
|
||||
&mut self,
|
||||
span: LightSpan,
|
||||
frame_id: usize,
|
||||
state: GeneratorState,
|
||||
mut generator: Generator,
|
||||
initial_message: Option<GeneratorResponse>,
|
||||
) -> EvalResult<bool> {
|
||||
// Determine what to send to the generator based on its state.
|
||||
let mut message = match (initial_message, state) {
|
||||
(Some(msg), _) => msg,
|
||||
(_, GeneratorState::Running) => GeneratorResponse::Empty,
|
||||
|
||||
// If control returned here, and the generator is
|
||||
// awaiting a value, send it the top of the stack.
|
||||
(_, GeneratorState::AwaitingValue) => GeneratorResponse::Value(self.stack_pop()),
|
||||
};
|
||||
|
||||
loop {
|
||||
match generator.resume_with(message) {
|
||||
// If the generator yields, it contains an instruction
|
||||
// for what the VM should do.
|
||||
genawaiter::GeneratorState::Yielded(request) => {
|
||||
self.observer.observe_generator_request(&request);
|
||||
|
||||
match request {
|
||||
GeneratorRequest::StackPush(value) => {
|
||||
self.stack.push(value);
|
||||
message = GeneratorResponse::Empty;
|
||||
}
|
||||
|
||||
GeneratorRequest::StackPop => {
|
||||
message = GeneratorResponse::Value(self.stack_pop());
|
||||
}
|
||||
|
||||
// Generator has requested a force, which means that
|
||||
// this function prepares the frame stack and yields
|
||||
// back to the outer VM loop.
|
||||
GeneratorRequest::ForceValue(value) => {
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
self.enqueue_generator(span, |co| value.force(co));
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
// Generator has requested a deep-force.
|
||||
GeneratorRequest::DeepForceValue(value, thunk_set) => {
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
self.enqueue_generator(span, |co| value.deep_force(co, thunk_set));
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
// Generator has requested a value from the with-stack.
|
||||
// Logic is similar to `ForceValue`, except with the
|
||||
// value being taken from that stack.
|
||||
GeneratorRequest::WithValue(idx) => {
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
|
||||
let value = self.stack[self.with_stack[idx]].clone();
|
||||
self.enqueue_generator(span, |co| value.force(co));
|
||||
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
// Generator has requested a value from the *captured*
|
||||
// with-stack. Logic is same as above, except for the
|
||||
// value being from that stack.
|
||||
GeneratorRequest::CapturedWithValue(idx) => {
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
|
||||
let call_frame = self.last_call_frame()
|
||||
.expect("Tvix bug: generator requested captured with-value, but there is no call frame");
|
||||
|
||||
let value = call_frame.upvalues.with_stack().unwrap()[idx].clone();
|
||||
self.enqueue_generator(span, |co| value.force(co));
|
||||
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
GeneratorRequest::NixEquality(values, ptr_eq) => {
|
||||
let values = *values;
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
self.enqueue_generator(span, |co| {
|
||||
values.0.nix_eq(values.1, co, ptr_eq)
|
||||
});
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
GeneratorRequest::StringCoerce(val, kind) => {
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
self.enqueue_generator(span, |co| val.coerce_to_string(co, kind));
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
GeneratorRequest::Call(callable) => {
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
self.tail_call_value(span, None, callable)?;
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
GeneratorRequest::EnterLambda {
|
||||
lambda,
|
||||
upvalues,
|
||||
light_span,
|
||||
} => {
|
||||
self.reenqueue_generator(span, generator);
|
||||
|
||||
self.frames.push(Frame::CallFrame {
|
||||
span: light_span,
|
||||
call_frame: CallFrame {
|
||||
lambda,
|
||||
upvalues,
|
||||
ip: CodeIdx(0),
|
||||
stack_offset: self.stack.len(),
|
||||
},
|
||||
});
|
||||
|
||||
return Ok(false);
|
||||
}
|
||||
|
||||
GeneratorRequest::EmitWarning(kind) => {
|
||||
self.emit_warning(kind);
|
||||
message = GeneratorResponse::Empty;
|
||||
}
|
||||
|
||||
GeneratorRequest::ImportCacheLookup(path) => {
|
||||
if let Some(cached) = self.import_cache.get(&path) {
|
||||
message = GeneratorResponse::Value(cached.clone());
|
||||
} else {
|
||||
message = GeneratorResponse::Empty;
|
||||
}
|
||||
}
|
||||
|
||||
GeneratorRequest::ImportCachePut(path, value) => {
|
||||
self.import_cache.insert(path, value);
|
||||
message = GeneratorResponse::Empty;
|
||||
}
|
||||
|
||||
GeneratorRequest::PathImport(path) => {
|
||||
let imported = self
|
||||
.io_handle
|
||||
.import_path(&path)
|
||||
.map_err(|kind| Error::new(kind, span.span()))?;
|
||||
|
||||
message = GeneratorResponse::Path(imported);
|
||||
}
|
||||
|
||||
GeneratorRequest::ReadToString(path) => {
|
||||
let content = self
|
||||
.io_handle
|
||||
.read_to_string(path)
|
||||
.map_err(|kind| Error::new(kind, span.span()))?;
|
||||
|
||||
message = GeneratorResponse::Value(Value::String(content.into()))
|
||||
}
|
||||
|
||||
GeneratorRequest::PathExists(path) => {
|
||||
let exists = self
|
||||
.io_handle
|
||||
.path_exists(path)
|
||||
.map(Value::Bool)
|
||||
.map_err(|kind| Error::new(kind, span.span()))?;
|
||||
|
||||
message = GeneratorResponse::Value(exists);
|
||||
}
|
||||
|
||||
GeneratorRequest::ReadDir(path) => {
|
||||
let dir = self
|
||||
.io_handle
|
||||
.read_dir(path)
|
||||
.map_err(|kind| Error::new(kind, span.span()))?;
|
||||
|
||||
message = GeneratorResponse::Directory(dir);
|
||||
}
|
||||
|
||||
GeneratorRequest::Span => {
|
||||
message = GeneratorResponse::Span(self.reasonable_light_span());
|
||||
}
|
||||
|
||||
GeneratorRequest::TryForce(value) => {
|
||||
self.try_eval_frames.push(frame_id);
|
||||
self.reenqueue_generator(span.clone(), generator);
|
||||
|
||||
debug_assert!(
|
||||
self.frames.len() == frame_id + 1,
|
||||
"generator should be reenqueued with the same frame ID"
|
||||
);
|
||||
|
||||
self.enqueue_generator(span, |co| value.force(co));
|
||||
return Ok(false);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Generator has completed, and its result value should
|
||||
// be left on the stack.
|
||||
genawaiter::GeneratorState::Complete(result) => {
|
||||
let value = result.map_err(|kind| Error::new(kind, span.span()))?;
|
||||
self.stack.push(value);
|
||||
return Ok(true);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub type GenCo = Co<GeneratorRequest, GeneratorResponse>;
|
||||
|
||||
// -- Implementation of concrete generator use-cases.
|
||||
|
@ -335,28 +568,6 @@ pub async fn request_deep_force(co: &GenCo, val: Value, thunk_set: SharedThunkSe
|
|||
}
|
||||
}
|
||||
|
||||
/// Fetch and force a value on the with-stack from the VM.
|
||||
async fn fetch_forced_with(co: &GenCo, idx: usize) -> Value {
|
||||
match co.yield_(GeneratorRequest::WithValue(idx)).await {
|
||||
GeneratorResponse::Value(value) => value,
|
||||
msg => panic!(
|
||||
"Tvix bug: VM responded with incorrect generator message: {}",
|
||||
msg
|
||||
),
|
||||
}
|
||||
}
|
||||
|
||||
/// Fetch and force a value on the *captured* with-stack from the VM.
|
||||
async fn fetch_captured_with(co: &GenCo, idx: usize) -> Value {
|
||||
match co.yield_(GeneratorRequest::CapturedWithValue(idx)).await {
|
||||
GeneratorResponse::Value(value) => value,
|
||||
msg => panic!(
|
||||
"Tvix bug: VM responded with incorrect generator message: {}",
|
||||
msg
|
||||
),
|
||||
}
|
||||
}
|
||||
|
||||
/// Ask the VM to compare two values for equality.
|
||||
pub(crate) async fn check_equality(
|
||||
co: &GenCo,
|
||||
|
@ -486,34 +697,6 @@ pub(crate) async fn request_span(co: &GenCo) -> LightSpan {
|
|||
}
|
||||
}
|
||||
|
||||
pub(crate) async fn neo_resolve_with(
|
||||
co: GenCo,
|
||||
ident: String,
|
||||
vm_with_len: usize,
|
||||
upvalue_with_len: usize,
|
||||
) -> Result<Value, ErrorKind> {
|
||||
for with_stack_idx in (0..vm_with_len).rev() {
|
||||
// TODO(tazjin): is this branch still live with the current with-thunking?
|
||||
let with = fetch_forced_with(&co, with_stack_idx).await;
|
||||
|
||||
match with.to_attrs()?.select(&ident) {
|
||||
None => continue,
|
||||
Some(val) => return Ok(val.clone()),
|
||||
}
|
||||
}
|
||||
|
||||
for upvalue_with_idx in (0..upvalue_with_len).rev() {
|
||||
let with = fetch_captured_with(&co, upvalue_with_idx).await;
|
||||
|
||||
match with.to_attrs()?.select(&ident) {
|
||||
None => continue,
|
||||
Some(val) => return Ok(val.clone()),
|
||||
}
|
||||
}
|
||||
|
||||
Err(ErrorKind::UnknownDynamicVariable(ident))
|
||||
}
|
||||
|
||||
/// Call the given value as if it was an attribute set containing a functor. The
|
||||
/// arguments must already be prepared on the stack when a generator frame from
|
||||
/// this function is invoked.
|
||||
|
|
70
tvix/eval/src/vm/macros.rs
Normal file
70
tvix/eval/src/vm/macros.rs
Normal file
|
@ -0,0 +1,70 @@
|
|||
/// This module provides macros which are used in the implementation
|
||||
/// of the VM for the implementation of repetitive operations.
|
||||
|
||||
/// This macro simplifies the implementation of arithmetic operations,
|
||||
/// correctly handling the behaviour on different pairings of number
|
||||
/// types.
|
||||
#[macro_export]
|
||||
macro_rules! arithmetic_op {
|
||||
( $self:ident, $op:tt ) => {{ // TODO: remove
|
||||
let b = $self.pop();
|
||||
let a = $self.pop();
|
||||
let result = fallible!($self, arithmetic_op!(&a, &b, $op));
|
||||
$self.push(result);
|
||||
}};
|
||||
|
||||
( $a:expr, $b:expr, $op:tt ) => {{
|
||||
match ($a, $b) {
|
||||
(Value::Integer(i1), Value::Integer(i2)) => Ok(Value::Integer(i1 $op i2)),
|
||||
(Value::Float(f1), Value::Float(f2)) => Ok(Value::Float(f1 $op f2)),
|
||||
(Value::Integer(i1), Value::Float(f2)) => Ok(Value::Float(*i1 as f64 $op f2)),
|
||||
(Value::Float(f1), Value::Integer(i2)) => Ok(Value::Float(f1 $op *i2 as f64)),
|
||||
|
||||
(v1, v2) => Err(ErrorKind::TypeError {
|
||||
expected: "number (either int or float)",
|
||||
actual: if v1.is_number() {
|
||||
v2.type_of()
|
||||
} else {
|
||||
v1.type_of()
|
||||
},
|
||||
}),
|
||||
}
|
||||
}};
|
||||
}
|
||||
|
||||
/// This macro simplifies the implementation of comparison operations.
|
||||
#[macro_export]
|
||||
macro_rules! cmp_op {
|
||||
( $vm:ident, $frame:ident, $span:ident, $op:tt ) => {{
|
||||
let b = $vm.stack_pop();
|
||||
let a = $vm.stack_pop();
|
||||
|
||||
async fn compare(a: Value, b: Value, co: GenCo) -> Result<Value, ErrorKind> {
|
||||
let a = generators::request_force(&co, a).await;
|
||||
let b = generators::request_force(&co, b).await;
|
||||
let ordering = a.nix_cmp_ordering(b, co).await?;
|
||||
Ok(Value::Bool(cmp_op!(@order $op ordering)))
|
||||
}
|
||||
|
||||
let gen_span = $frame.current_light_span();
|
||||
$vm.push_call_frame($span, $frame);
|
||||
$vm.enqueue_generator(gen_span, |co| compare(a, b, co));
|
||||
return Ok(false);
|
||||
}};
|
||||
|
||||
(@order < $ordering:expr) => {
|
||||
$ordering == Some(Ordering::Less)
|
||||
};
|
||||
|
||||
(@order > $ordering:expr) => {
|
||||
$ordering == Some(Ordering::Greater)
|
||||
};
|
||||
|
||||
(@order <= $ordering:expr) => {
|
||||
!matches!($ordering, None | Some(Ordering::Greater))
|
||||
};
|
||||
|
||||
(@order >= $ordering:expr) => {
|
||||
!matches!($ordering, None | Some(Ordering::Less))
|
||||
};
|
||||
}
|
1120
tvix/eval/src/vm/mod.rs
Normal file
1120
tvix/eval/src/vm/mod.rs
Normal file
File diff suppressed because it is too large
Load diff
Loading…
Reference in a new issue