mirror of
https://github.com/roc-lang/roc.git
synced 2025-09-30 23:31:12 +00:00
opt tests
This commit is contained in:
parent
5c0dc268f8
commit
dab6621127
1 changed files with 22 additions and 201 deletions
|
@ -1,200 +1,12 @@
|
||||||
use roc_gen::llvm::build::Scope;
|
|
||||||
|
|
||||||
// #[macro_export]
|
|
||||||
macro_rules! assert_llvm_evals_to {
|
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr, $leak:expr) => {
|
|
||||||
let target = target_lexicon::Triple::host();
|
|
||||||
let ptr_bytes = target.pointer_width().unwrap().bytes() as u32;
|
|
||||||
let arena = Bump::new();
|
|
||||||
let CanExprOut { loc_expr, var_store, var, constraint, home, interns, problems, .. } = can_expr($src);
|
|
||||||
let errors = problems.into_iter().filter(|problem| {
|
|
||||||
use roc_problem::can::Problem::*;
|
|
||||||
|
|
||||||
// Ignore "unused" problems
|
|
||||||
match problem {
|
|
||||||
UnusedDef(_, _) | UnusedArgument(_, _, _) | UnusedImport(_, _) => false,
|
|
||||||
_ => true,
|
|
||||||
}
|
|
||||||
}).collect::<Vec<roc_problem::can::Problem>>();
|
|
||||||
|
|
||||||
assert_eq!(errors, Vec::new(), "Encountered errors: {:?}", errors);
|
|
||||||
|
|
||||||
let subs = Subs::new(var_store.into());
|
|
||||||
let mut unify_problems = Vec::new();
|
|
||||||
let (content, mut subs) = infer_expr(subs, &mut unify_problems, &constraint, var);
|
|
||||||
|
|
||||||
assert_eq!(unify_problems, Vec::new(), "Encountered type mismatches: {:?}", unify_problems);
|
|
||||||
|
|
||||||
let context = Context::create();
|
|
||||||
let module = roc_gen::llvm::build::module_from_builtins(&context, "app");
|
|
||||||
let builder = context.create_builder();
|
|
||||||
let opt_level = if cfg!(debug_assertions) {
|
|
||||||
roc_gen::llvm::build::OptLevel::Normal
|
|
||||||
} else {
|
|
||||||
roc_gen::llvm::build::OptLevel::Optimize
|
|
||||||
};
|
|
||||||
let fpm = PassManager::create(&module);
|
|
||||||
|
|
||||||
roc_gen::llvm::build::add_passes(&fpm, opt_level);
|
|
||||||
|
|
||||||
fpm.initialize();
|
|
||||||
|
|
||||||
// Compute main_fn_type before moving subs to Env
|
|
||||||
let layout = Layout::new(&arena, content, &subs, ptr_bytes)
|
|
||||||
.unwrap_or_else(|err| panic!("Code gen error in NON-OPTIMIZED test: could not convert to layout. Err was {:?}", err));
|
|
||||||
let execution_engine =
|
|
||||||
module
|
|
||||||
.create_jit_execution_engine(OptimizationLevel::None)
|
|
||||||
.expect("Error creating JIT execution engine for test");
|
|
||||||
|
|
||||||
let main_fn_type = basic_type_from_layout(&arena, &context, &layout, ptr_bytes)
|
|
||||||
.fn_type(&[], false);
|
|
||||||
let main_fn_name = "$Test.main";
|
|
||||||
|
|
||||||
// Compile and add all the Procs before adding main
|
|
||||||
let mut env = roc_gen::llvm::build::Env {
|
|
||||||
arena: &arena,
|
|
||||||
builder: &builder,
|
|
||||||
context: &context,
|
|
||||||
interns,
|
|
||||||
module: arena.alloc(module),
|
|
||||||
ptr_bytes,
|
|
||||||
leak: $leak
|
|
||||||
|
|
||||||
};
|
|
||||||
let mut procs = Procs::default();
|
|
||||||
let mut ident_ids = env.interns.all_ident_ids.remove(&home).unwrap();
|
|
||||||
let mut layout_ids = roc_gen::layout_id::LayoutIds::default();
|
|
||||||
|
|
||||||
// Populate Procs and get the low-level Expr from the canonical Expr
|
|
||||||
let mut mono_problems = Vec::new();
|
|
||||||
let mut mono_env = roc_mono::expr::Env {
|
|
||||||
arena: &arena,
|
|
||||||
subs: &mut subs,
|
|
||||||
problems: &mut mono_problems,
|
|
||||||
home,
|
|
||||||
ident_ids: &mut ident_ids,
|
|
||||||
pointer_size: ptr_bytes,
|
|
||||||
jump_counter: arena.alloc(0),
|
|
||||||
};
|
|
||||||
|
|
||||||
let main_body = Expr::new(&mut mono_env, loc_expr.value, &mut procs);
|
|
||||||
let mut headers = {
|
|
||||||
let num_headers = match &procs.pending_specializations {
|
|
||||||
Some(map) => map.len(),
|
|
||||||
None => 0
|
|
||||||
};
|
|
||||||
|
|
||||||
Vec::with_capacity(num_headers)
|
|
||||||
};
|
|
||||||
let mut layout_cache = roc_mono::layout::LayoutCache::default();
|
|
||||||
let mut procs = roc_mono::expr::specialize_all(&mut mono_env, procs, &mut layout_cache);
|
|
||||||
|
|
||||||
assert_eq!(procs.runtime_errors, roc_collections::all::MutMap::default());
|
|
||||||
|
|
||||||
// Put this module's ident_ids back in the interns, so we can use them in env.
|
|
||||||
// This must happen *after* building the headers, because otherwise there's
|
|
||||||
// a conflicting mutable borrow on ident_ids.
|
|
||||||
env.interns.all_ident_ids.insert(home, ident_ids);
|
|
||||||
|
|
||||||
// Add all the Proc headers to the module.
|
|
||||||
// We have to do this in a separate pass first,
|
|
||||||
// because their bodies may reference each other.
|
|
||||||
for ((symbol, layout), proc) in procs.specialized.drain() {
|
|
||||||
use roc_mono::expr::InProgressProc::*;
|
|
||||||
|
|
||||||
match proc {
|
|
||||||
InProgress => {
|
|
||||||
panic!("A specialization was still marked InProgress after monomorphization had completed: {:?} with layout {:?}", symbol, layout);
|
|
||||||
}
|
|
||||||
Done(proc) => {
|
|
||||||
let (fn_val, arg_basic_types) =
|
|
||||||
build_proc_header(&env, &mut layout_ids, symbol, &layout, &proc);
|
|
||||||
|
|
||||||
headers.push((proc, fn_val, arg_basic_types));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build each proc using its header info.
|
|
||||||
for (proc, fn_val, arg_basic_types) in headers {
|
|
||||||
build_proc(&env, &mut layout_ids, proc, fn_val, arg_basic_types);
|
|
||||||
|
|
||||||
if fn_val.verify(true) {
|
|
||||||
fpm.run_on(&fn_val);
|
|
||||||
} else {
|
|
||||||
eprintln!(
|
|
||||||
"\n\nFunction {:?} failed LLVM verification in NON-OPTIMIZED build. Its content was:\n", fn_val.get_name().to_str().unwrap()
|
|
||||||
);
|
|
||||||
|
|
||||||
fn_val.print_to_stderr();
|
|
||||||
|
|
||||||
panic!(
|
|
||||||
"The preceding code was from {:?}, which failed LLVM verification in NON-OPTIMIZED build.", fn_val.get_name().to_str().unwrap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add main to the module.
|
|
||||||
let main_fn = env.module.add_function(main_fn_name, main_fn_type, None);
|
|
||||||
let cc = roc_gen::llvm::build::get_call_conventions(target.default_calling_convention().unwrap());
|
|
||||||
|
|
||||||
main_fn.set_call_conventions(cc);
|
|
||||||
|
|
||||||
// Add main's body
|
|
||||||
let basic_block = context.append_basic_block(main_fn, "entry");
|
|
||||||
|
|
||||||
builder.position_at_end(basic_block);
|
|
||||||
|
|
||||||
let ret = roc_gen::llvm::build::build_expr(
|
|
||||||
&env,
|
|
||||||
&mut layout_ids,
|
|
||||||
&mut Scope::default(),
|
|
||||||
main_fn,
|
|
||||||
&main_body,
|
|
||||||
);
|
|
||||||
|
|
||||||
builder.build_return(Some(&ret));
|
|
||||||
|
|
||||||
// Uncomment this to see the module's un-optimized LLVM instruction output:
|
|
||||||
env.module.print_to_stderr();
|
|
||||||
|
|
||||||
if main_fn.verify(true) {
|
|
||||||
fpm.run_on(&main_fn);
|
|
||||||
} else {
|
|
||||||
panic!("Main function {} failed LLVM verification in NON-OPTIMIZED build. Uncomment things nearby to see more details.", main_fn_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify the module
|
|
||||||
if let Err(errors) = env.module.verify() {
|
|
||||||
panic!("Errors defining module: {:?}", errors);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Uncomment this to see the module's optimized LLVM instruction output:
|
|
||||||
// env.module.print_to_stderr();
|
|
||||||
|
|
||||||
unsafe {
|
|
||||||
let main: JitFunction<unsafe extern "C" fn() -> $ty> = execution_engine
|
|
||||||
.get_function(main_fn_name)
|
|
||||||
.ok()
|
|
||||||
.ok_or(format!("Unable to JIT compile `{}`", main_fn_name))
|
|
||||||
.expect("errored");
|
|
||||||
|
|
||||||
assert_eq!($transform(main.call()), $expected);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
|
||||||
assert_llvm_evals_to!($src, $expected, $ty, $transform, false);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO this is almost all code duplication with assert_llvm_evals_to
|
// TODO this is almost all code duplication with assert_llvm_evals_to
|
||||||
// the only difference is that this calls uniq_expr instead of can_expr.
|
// the only difference is that this calls uniq_expr instead of can_expr.
|
||||||
// Should extract the common logic into test helpers.
|
// Should extract the common logic into test helpers.
|
||||||
// #[macro_export]
|
#[macro_export]
|
||||||
macro_rules! assert_opt_evals_to {
|
macro_rules! assert_opt_evals_to {
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr, $leak:expr) => {
|
($src:expr, $expected:expr, $ty:ty, $transform:expr, $leak:expr) => {
|
||||||
|
use roc_gen::llvm::build::Scope;
|
||||||
|
use roc_gen::llvm::build::{build_proc_header_ir, build_proc_ir};
|
||||||
|
|
||||||
let arena = Bump::new();
|
let arena = Bump::new();
|
||||||
let target = target_lexicon::Triple::host();
|
let target = target_lexicon::Triple::host();
|
||||||
let ptr_bytes = target.pointer_width().unwrap().bytes() as u32;
|
let ptr_bytes = target.pointer_width().unwrap().bytes() as u32;
|
||||||
|
@ -253,13 +65,13 @@ macro_rules! assert_opt_evals_to {
|
||||||
ptr_bytes,
|
ptr_bytes,
|
||||||
leak: $leak
|
leak: $leak
|
||||||
};
|
};
|
||||||
let mut procs = Procs::default();
|
let mut procs = roc_mono::experiment::Procs::default();
|
||||||
let mut ident_ids = env.interns.all_ident_ids.remove(&home).unwrap();
|
let mut ident_ids = env.interns.all_ident_ids.remove(&home).unwrap();
|
||||||
let mut layout_ids = roc_gen::layout_id::LayoutIds::default();
|
let mut layout_ids = roc_gen::layout_id::LayoutIds::default();
|
||||||
|
|
||||||
// Populate Procs and get the low-level Expr from the canonical Expr
|
// Populate Procs and get the low-level Expr from the canonical Expr
|
||||||
let mut mono_problems = Vec::new();
|
let mut mono_problems = Vec::new();
|
||||||
let mut mono_env = roc_mono::expr::Env {
|
let mut mono_env = roc_mono::experiment::Env {
|
||||||
arena: &arena,
|
arena: &arena,
|
||||||
subs: &mut subs,
|
subs: &mut subs,
|
||||||
problems: &mut mono_problems,
|
problems: &mut mono_problems,
|
||||||
|
@ -268,8 +80,8 @@ macro_rules! assert_opt_evals_to {
|
||||||
pointer_size: ptr_bytes,
|
pointer_size: ptr_bytes,
|
||||||
jump_counter: arena.alloc(0),
|
jump_counter: arena.alloc(0),
|
||||||
};
|
};
|
||||||
let main_body = Expr::new(&mut mono_env, loc_expr.value, &mut procs);
|
|
||||||
|
|
||||||
|
let main_body = roc_mono::experiment::Stmt::new(&mut mono_env, loc_expr.value, &mut procs);
|
||||||
let mut headers = {
|
let mut headers = {
|
||||||
let num_headers = match &procs.pending_specializations {
|
let num_headers = match &procs.pending_specializations {
|
||||||
Some(map) => map.len(),
|
Some(map) => map.len(),
|
||||||
|
@ -279,7 +91,7 @@ macro_rules! assert_opt_evals_to {
|
||||||
Vec::with_capacity(num_headers)
|
Vec::with_capacity(num_headers)
|
||||||
};
|
};
|
||||||
let mut layout_cache = roc_mono::layout::LayoutCache::default();
|
let mut layout_cache = roc_mono::layout::LayoutCache::default();
|
||||||
let mut procs = roc_mono::expr::specialize_all(&mut mono_env, procs, &mut layout_cache);
|
let mut procs = roc_mono::experiment::specialize_all(&mut mono_env, procs, &mut layout_cache);
|
||||||
|
|
||||||
assert_eq!(procs.runtime_errors, roc_collections::all::MutMap::default());
|
assert_eq!(procs.runtime_errors, roc_collections::all::MutMap::default());
|
||||||
|
|
||||||
|
@ -292,7 +104,7 @@ macro_rules! assert_opt_evals_to {
|
||||||
// We have to do this in a separate pass first,
|
// We have to do this in a separate pass first,
|
||||||
// because their bodies may reference each other.
|
// because their bodies may reference each other.
|
||||||
for ((symbol, layout), proc) in procs.specialized.drain() {
|
for ((symbol, layout), proc) in procs.specialized.drain() {
|
||||||
use roc_mono::expr::InProgressProc::*;
|
use roc_mono::experiment::InProgressProc::*;
|
||||||
|
|
||||||
match proc {
|
match proc {
|
||||||
InProgress => {
|
InProgress => {
|
||||||
|
@ -300,7 +112,7 @@ macro_rules! assert_opt_evals_to {
|
||||||
}
|
}
|
||||||
Done(proc) => {
|
Done(proc) => {
|
||||||
let (fn_val, arg_basic_types) =
|
let (fn_val, arg_basic_types) =
|
||||||
build_proc_header(&env, &mut layout_ids, symbol, &layout, &proc);
|
build_proc_header_ir(&env, &mut layout_ids, symbol, &layout, &proc);
|
||||||
|
|
||||||
headers.push((proc, fn_val, arg_basic_types));
|
headers.push((proc, fn_val, arg_basic_types));
|
||||||
}
|
}
|
||||||
|
@ -309,7 +121,7 @@ macro_rules! assert_opt_evals_to {
|
||||||
|
|
||||||
// Build each proc using its header info.
|
// Build each proc using its header info.
|
||||||
for (proc, fn_val, arg_basic_types) in headers {
|
for (proc, fn_val, arg_basic_types) in headers {
|
||||||
build_proc(&env, &mut layout_ids, proc, fn_val, arg_basic_types);
|
build_proc_ir(&env, &mut layout_ids, proc, fn_val, arg_basic_types);
|
||||||
|
|
||||||
if fn_val.verify(true) {
|
if fn_val.verify(true) {
|
||||||
fpm.run_on(&fn_val);
|
fpm.run_on(&fn_val);
|
||||||
|
@ -337,7 +149,7 @@ macro_rules! assert_opt_evals_to {
|
||||||
|
|
||||||
builder.position_at_end(basic_block);
|
builder.position_at_end(basic_block);
|
||||||
|
|
||||||
let ret = roc_gen::llvm::build::build_expr(
|
let ret = roc_gen::llvm::build::build_exp_stmt(
|
||||||
&env,
|
&env,
|
||||||
&mut layout_ids,
|
&mut layout_ids,
|
||||||
&mut Scope::default(),
|
&mut Scope::default(),
|
||||||
|
@ -376,7 +188,7 @@ macro_rules! assert_opt_evals_to {
|
||||||
};
|
};
|
||||||
|
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
||||||
assert_llvm_evals_to!($src, $expected, $ty, $transform, false)
|
assert_opt_evals_to!($src, $expected, $ty, $transform, false)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -615,17 +427,26 @@ macro_rules! assert_evals_to_ir {
|
||||||
{
|
{
|
||||||
assert_llvm_ir_evals_to!($src, $expected, $ty, (|val| val));
|
assert_llvm_ir_evals_to!($src, $expected, $ty, (|val| val));
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
assert_opt_evals_to!($src, $expected, $ty, (|val| val));
|
||||||
|
}
|
||||||
};
|
};
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
||||||
// Same as above, except with an additional transformation argument.
|
// Same as above, except with an additional transformation argument.
|
||||||
{
|
{
|
||||||
assert_llvm_ir_evals_to!($src, $expected, $ty, $transform);
|
assert_llvm_ir_evals_to!($src, $expected, $ty, $transform);
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
assert_opt_evals_to!($src, $expected, $ty, $transform);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr, $leak:expr) => {
|
($src:expr, $expected:expr, $ty:ty, $transform:expr, $leak:expr) => {
|
||||||
// Same as above, except with an additional transformation argument.
|
// Same as above, except with an additional transformation argument.
|
||||||
{
|
{
|
||||||
assert_llvm_ir_evals_to!($src, $expected, $ty, $transform, $leak);
|
assert_llvm_ir_evals_to!($src, $expected, $ty, $transform, $leak);
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
assert_opt_evals_to!($src, $expected, $ty, $transform, $leak);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue