Gates the observes behind the envvars `TVIX_DUMP_BYTECODE` and `TVIX_TRACE_RUNTIME`. (hi grfn, yes, we should probably introduce CLI flags soon) Change-Id: I4fa194a84b04593d609b96b44471c3644fb30296 Reviewed-on: https://cl.tvl.fyi/c/depot/+/6459 Tested-by: BuildkiteCI Reviewed-by: sterni <sternenseemann@systemli.org>
89 lines
2.4 KiB
Rust
89 lines
2.4 KiB
Rust
use std::{path::PathBuf, rc::Rc};
|
|
|
|
use crate::{
|
|
builtins::global_builtins,
|
|
errors::{Error, ErrorKind, EvalResult},
|
|
observer::{DisassemblingObserver, NoOpObserver, TracingObserver},
|
|
value::Value,
|
|
};
|
|
|
|
pub fn interpret(code: &str, location: Option<PathBuf>) -> EvalResult<Value> {
|
|
let mut codemap = codemap::CodeMap::new();
|
|
let file = codemap.add_file(
|
|
location
|
|
.as_ref()
|
|
.map(|p| p.to_string_lossy().to_string())
|
|
.unwrap_or_else(|| "<repl>".into()),
|
|
code.into(),
|
|
);
|
|
let codemap = Rc::new(codemap);
|
|
|
|
let parsed = rnix::ast::Root::parse(code);
|
|
let errors = parsed.errors();
|
|
|
|
if !errors.is_empty() {
|
|
for err in errors {
|
|
eprintln!("parse error: {}", err);
|
|
}
|
|
return Err(Error {
|
|
kind: ErrorKind::ParseErrors(errors.to_vec()),
|
|
span: file.span,
|
|
});
|
|
}
|
|
|
|
// If we've reached this point, there are no errors.
|
|
let root_expr = parsed
|
|
.tree()
|
|
.expr()
|
|
.expect("expression should exist if no errors occured");
|
|
|
|
if std::env::var("TVIX_DISPLAY_AST").is_ok() {
|
|
println!("{:?}", root_expr);
|
|
}
|
|
|
|
let result = if std::env::var("TVIX_DUMP_BYTECODE").is_ok() {
|
|
crate::compiler::compile(
|
|
root_expr,
|
|
location,
|
|
&file,
|
|
global_builtins(),
|
|
&mut DisassemblingObserver::new(codemap, std::io::stderr()),
|
|
)
|
|
} else {
|
|
crate::compiler::compile(
|
|
root_expr,
|
|
location,
|
|
&file,
|
|
global_builtins(),
|
|
&mut NoOpObserver::default(),
|
|
)
|
|
}?;
|
|
|
|
for warning in result.warnings {
|
|
eprintln!(
|
|
"warning: {:?} at `{}`[line {}]",
|
|
warning.kind,
|
|
file.source_slice(warning.span),
|
|
file.find_line(warning.span.low()) + 1
|
|
)
|
|
}
|
|
|
|
for error in &result.errors {
|
|
eprintln!(
|
|
"compiler error: {:?} at `{}`[line {}]",
|
|
error.kind,
|
|
file.source_slice(error.span),
|
|
file.find_line(error.span.low()) + 1
|
|
);
|
|
}
|
|
|
|
if let Some(err) = result.errors.last() {
|
|
return Err(err.clone());
|
|
}
|
|
|
|
if std::env::var("TVIX_TRACE_RUNTIME").is_ok() {
|
|
crate::vm::run_lambda(&mut TracingObserver::new(std::io::stderr()), result.lambda)
|
|
} else {
|
|
crate::vm::run_lambda(&mut NoOpObserver::default(), result.lambda)
|
|
}
|
|
}
|