mirror of
https://github.com/roc-lang/roc.git
synced 2025-09-29 06:44:46 +00:00
run gen tests as modules
This commit is contained in:
parent
2bceaf0503
commit
89a1146c19
11 changed files with 485 additions and 771 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -2433,6 +2433,7 @@ dependencies = [
|
||||||
"roc_can",
|
"roc_can",
|
||||||
"roc_collections",
|
"roc_collections",
|
||||||
"roc_constrain",
|
"roc_constrain",
|
||||||
|
"roc_load",
|
||||||
"roc_module",
|
"roc_module",
|
||||||
"roc_mono",
|
"roc_mono",
|
||||||
"roc_parse",
|
"roc_parse",
|
||||||
|
|
|
@ -44,6 +44,7 @@ target-lexicon = "0.10"
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
roc_can = { path = "../can" }
|
roc_can = { path = "../can" }
|
||||||
roc_parse = { path = "../parse" }
|
roc_parse = { path = "../parse" }
|
||||||
|
roc_load = { path = "../load" }
|
||||||
pretty_assertions = "0.5.1"
|
pretty_assertions = "0.5.1"
|
||||||
maplit = "1.0.1"
|
maplit = "1.0.1"
|
||||||
indoc = "0.3.3"
|
indoc = "0.3.3"
|
||||||
|
|
|
@ -416,26 +416,47 @@ pub fn build_roc_main<'a, 'ctx, 'env>(
|
||||||
env.arena.alloc(roc_main_fn)
|
env.arena.alloc(roc_main_fn)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn promote_to_main_function<'a, 'ctx, 'env>(
|
||||||
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
symbol: Symbol,
|
||||||
|
layout: &Layout<'a>,
|
||||||
|
) -> (&'static str, &'a FunctionValue<'ctx>) {
|
||||||
|
let fn_name = layout_ids
|
||||||
|
.get(symbol, layout)
|
||||||
|
.to_symbol_string(symbol, &env.interns);
|
||||||
|
|
||||||
|
let wrapped = env.module.get_function(&fn_name).unwrap();
|
||||||
|
|
||||||
|
make_main_function_help(env, layout, wrapped)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn make_main_function<'a, 'ctx, 'env>(
|
pub fn make_main_function<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
layout: &Layout<'a>,
|
layout: &Layout<'a>,
|
||||||
main_body: &roc_mono::ir::Stmt<'a>,
|
main_body: &roc_mono::ir::Stmt<'a>,
|
||||||
) -> (&'static str, &'a FunctionValue<'ctx>) {
|
) -> (&'static str, &'a FunctionValue<'ctx>) {
|
||||||
|
// internal main function
|
||||||
|
let roc_main_fn = *build_roc_main(env, layout_ids, layout, main_body);
|
||||||
|
|
||||||
|
make_main_function_help(env, layout, roc_main_fn)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn make_main_function_help<'a, 'ctx, 'env>(
|
||||||
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
|
layout: &Layout<'a>,
|
||||||
|
roc_main_fn: FunctionValue<'ctx>,
|
||||||
|
) -> (&'static str, &'a FunctionValue<'ctx>) {
|
||||||
|
// build the C calling convention wrapper
|
||||||
use inkwell::types::BasicType;
|
use inkwell::types::BasicType;
|
||||||
use PassVia::*;
|
use PassVia::*;
|
||||||
|
|
||||||
let context = env.context;
|
let context = env.context;
|
||||||
let builder = env.builder;
|
let builder = env.builder;
|
||||||
|
|
||||||
let u8_ptr = context.i8_type().ptr_type(AddressSpace::Generic);
|
|
||||||
|
|
||||||
// internal main function
|
|
||||||
let roc_main_fn = *build_roc_main(env, layout_ids, layout, main_body);
|
|
||||||
|
|
||||||
// build the C calling convention wrapper
|
|
||||||
|
|
||||||
let main_fn_name = "$Test.main";
|
let main_fn_name = "$Test.main";
|
||||||
|
let u8_ptr = env.context.i8_type().ptr_type(AddressSpace::Generic);
|
||||||
|
|
||||||
let fields = [Layout::Builtin(Builtin::Int64), layout.clone()];
|
let fields = [Layout::Builtin(Builtin::Int64), layout.clone()];
|
||||||
let main_return_layout = Layout::Struct(&fields);
|
let main_return_layout = Layout::Struct(&fields);
|
||||||
|
|
|
@ -1019,8 +1019,7 @@ mod gen_list {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \shared ->
|
wrapper = \shared ->
|
||||||
|
|
||||||
# This should not mutate the original
|
# This should not mutate the original
|
||||||
x =
|
x =
|
||||||
when List.get (List.set shared 1 7.7) 1 is
|
when List.get (List.set shared 1 7.7) 1 is
|
||||||
|
@ -1034,7 +1033,7 @@ mod gen_list {
|
||||||
|
|
||||||
{ x, y }
|
{ x, y }
|
||||||
|
|
||||||
main [ 2.1, 4.3 ]
|
wrapper [ 2.1, 4.3 ]
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
(7.7, 4.3),
|
(7.7, 4.3),
|
||||||
|
@ -1047,23 +1046,20 @@ mod gen_list {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
shared = [ 2, 4 ]
|
||||||
shared = [ 2, 4 ]
|
|
||||||
|
|
||||||
# This List.set is out of bounds, and should have no effect
|
# This List.set is out of bounds, and should have no effect
|
||||||
x =
|
x =
|
||||||
when List.get (List.set shared 422 0) 1 is
|
when List.get (List.set shared 422 0) 1 is
|
||||||
Ok num -> num
|
Ok num -> num
|
||||||
Err _ -> 0
|
Err _ -> 0
|
||||||
|
|
||||||
y =
|
y =
|
||||||
when List.get shared 1 is
|
when List.get shared 1 is
|
||||||
Ok num -> num
|
Ok num -> num
|
||||||
Err _ -> 0
|
Err _ -> 0
|
||||||
|
|
||||||
{ x, y }
|
{ x, y }
|
||||||
|
|
||||||
main {}
|
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
(4, 4),
|
(4, 4),
|
||||||
|
@ -1149,16 +1145,21 @@ mod gen_list {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
swap : Int, Int, List a -> List a
|
app Quicksort provides [ main ] imports []
|
||||||
swap = \i, j, list ->
|
|
||||||
when Pair (List.get list i) (List.get list j) is
|
|
||||||
Pair (Ok atI) (Ok atJ) ->
|
|
||||||
list
|
|
||||||
|> List.set i atJ
|
|
||||||
|> List.set j atI
|
|
||||||
|
|
||||||
_ ->
|
|
||||||
[]
|
swap : Int, Int, List a -> List a
|
||||||
|
swap = \i, j, list ->
|
||||||
|
when Pair (List.get list i) (List.get list j) is
|
||||||
|
Pair (Ok atI) (Ok atJ) ->
|
||||||
|
list
|
||||||
|
|> List.set i atJ
|
||||||
|
|> List.set j atI
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
[]
|
||||||
|
|
||||||
|
main =
|
||||||
swap 0 1 [ 1, 2 ]
|
swap 0 1 [ 1, 2 ]
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
|
|
|
@ -482,12 +482,12 @@ mod gen_num {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
when 10 is
|
when 10 is
|
||||||
x if x == 5 -> 0
|
x if x == 5 -> 0
|
||||||
_ -> 42
|
_ -> 42
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
42,
|
42,
|
||||||
|
@ -500,12 +500,12 @@ mod gen_num {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
when 10 is
|
when 10 is
|
||||||
x if x == 10 -> 42
|
x if x == 10 -> 42
|
||||||
_ -> 0
|
_ -> 0
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
42,
|
42,
|
||||||
|
|
|
@ -276,10 +276,10 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
(\a -> a) 5
|
(\a -> a) 5
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
5,
|
5,
|
||||||
|
@ -292,14 +292,14 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
alwaysFloatIdentity : Int -> (Float -> Float)
|
alwaysFloatIdentity : Int -> (Float -> Float)
|
||||||
alwaysFloatIdentity = \num ->
|
alwaysFloatIdentity = \num ->
|
||||||
(\a -> a)
|
(\a -> a)
|
||||||
|
|
||||||
(alwaysFloatIdentity 2) 3.14
|
(alwaysFloatIdentity 2) 3.14
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3.14,
|
3.14,
|
||||||
|
@ -403,6 +403,7 @@ mod gen_primitives {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn gen_nested_defs() {
|
fn gen_nested_defs() {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
|
@ -469,15 +470,15 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Peano : [ S Peano, Z ]
|
Peano : [ S Peano, Z ]
|
||||||
|
|
||||||
three : Peano
|
three : Peano
|
||||||
three = S (S (S Z))
|
three = S (S (S Z))
|
||||||
|
|
||||||
when three is
|
when three is
|
||||||
Z -> 2
|
Z -> 2
|
||||||
S _ -> 1
|
S _ -> 1
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
1,
|
1,
|
||||||
i64
|
i64
|
||||||
|
@ -489,16 +490,16 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Peano : [ S Peano, Z ]
|
Peano : [ S Peano, Z ]
|
||||||
|
|
||||||
three : Peano
|
three : Peano
|
||||||
three = S (S (S Z))
|
three = S (S (S Z))
|
||||||
|
|
||||||
when three is
|
when three is
|
||||||
S (S _) -> 1
|
S (S _) -> 1
|
||||||
S (_) -> 0
|
S (_) -> 0
|
||||||
Z -> 0
|
Z -> 0
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
1,
|
1,
|
||||||
i64
|
i64
|
||||||
|
@ -506,24 +507,25 @@ mod gen_primitives {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn linked_list_len_0() {
|
fn linked_list_len_0() {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
||||||
|
|
||||||
nil : LinkedList Int
|
nil : LinkedList Int
|
||||||
nil = Nil
|
nil = Nil
|
||||||
|
|
||||||
length : LinkedList a -> Int
|
length : LinkedList a -> Int
|
||||||
length = \list ->
|
length = \list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> 0
|
Nil -> 0
|
||||||
Cons _ rest -> 1 + length rest
|
Cons _ rest -> 1 + length rest
|
||||||
|
|
||||||
|
|
||||||
length nil
|
length nil
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
0,
|
0,
|
||||||
i64,
|
i64,
|
||||||
|
@ -533,23 +535,24 @@ mod gen_primitives {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn linked_list_len_twice_0() {
|
fn linked_list_len_twice_0() {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
||||||
|
|
||||||
nil : LinkedList Int
|
nil : LinkedList Int
|
||||||
nil = Nil
|
nil = Nil
|
||||||
|
|
||||||
length : LinkedList a -> Int
|
length : LinkedList a -> Int
|
||||||
length = \list ->
|
length = \list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> 0
|
Nil -> 0
|
||||||
Cons _ rest -> 1 + length rest
|
Cons _ rest -> 1 + length rest
|
||||||
|
|
||||||
length nil + length nil
|
length nil + length nil
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
0,
|
0,
|
||||||
i64,
|
i64,
|
||||||
|
@ -559,24 +562,25 @@ mod gen_primitives {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn linked_list_len_1() {
|
fn linked_list_len_1() {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
||||||
|
|
||||||
one : LinkedList Int
|
one : LinkedList Int
|
||||||
one = Cons 1 Nil
|
one = Cons 1 Nil
|
||||||
|
|
||||||
length : LinkedList a -> Int
|
length : LinkedList a -> Int
|
||||||
length = \list ->
|
length = \list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> 0
|
Nil -> 0
|
||||||
Cons _ rest -> 1 + length rest
|
Cons _ rest -> 1 + length rest
|
||||||
|
|
||||||
|
|
||||||
length one
|
length one
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
1,
|
1,
|
||||||
i64,
|
i64,
|
||||||
|
@ -586,24 +590,25 @@ mod gen_primitives {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn linked_list_len_twice_1() {
|
fn linked_list_len_twice_1() {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
||||||
|
|
||||||
one : LinkedList Int
|
one : LinkedList Int
|
||||||
one = Cons 1 Nil
|
one = Cons 1 Nil
|
||||||
|
|
||||||
length : LinkedList a -> Int
|
length : LinkedList a -> Int
|
||||||
length = \list ->
|
length = \list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> 0
|
Nil -> 0
|
||||||
Cons _ rest -> 1 + length rest
|
Cons _ rest -> 1 + length rest
|
||||||
|
|
||||||
|
|
||||||
length one + length one
|
length one + length one
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
2,
|
2,
|
||||||
i64,
|
i64,
|
||||||
|
@ -613,24 +618,25 @@ mod gen_primitives {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn linked_list_len_3() {
|
fn linked_list_len_3() {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
||||||
|
|
||||||
three : LinkedList Int
|
three : LinkedList Int
|
||||||
three = Cons 3 (Cons 2 (Cons 1 Nil))
|
three = Cons 3 (Cons 2 (Cons 1 Nil))
|
||||||
|
|
||||||
length : LinkedList a -> Int
|
length : LinkedList a -> Int
|
||||||
length = \list ->
|
length = \list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> 0
|
Nil -> 0
|
||||||
Cons _ rest -> 1 + length rest
|
Cons _ rest -> 1 + length rest
|
||||||
|
|
||||||
|
|
||||||
length three
|
length three
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3,
|
3,
|
||||||
i64,
|
i64,
|
||||||
|
@ -640,23 +646,24 @@ mod gen_primitives {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn linked_list_sum() {
|
fn linked_list_sum() {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
||||||
|
|
||||||
three : LinkedList Int
|
three : LinkedList Int
|
||||||
three = Cons 3 (Cons 2 (Cons 1 Nil))
|
three = Cons 3 (Cons 2 (Cons 1 Nil))
|
||||||
|
|
||||||
sum : LinkedList a -> Int
|
sum : LinkedList a -> Int
|
||||||
sum = \list ->
|
sum = \list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> 0
|
Nil -> 0
|
||||||
Cons x rest -> x + sum rest
|
Cons x rest -> x + sum rest
|
||||||
|
|
||||||
sum three
|
sum three
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3 + 2 + 1,
|
3 + 2 + 1,
|
||||||
i64
|
i64
|
||||||
|
@ -664,30 +671,30 @@ mod gen_primitives {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
#[ignore]
|
||||||
fn linked_list_map() {
|
fn linked_list_map() {
|
||||||
// `f` is not actually a function, so the call to it fails currently
|
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
LinkedList a : [ Nil, Cons a (LinkedList a) ]
|
||||||
|
|
||||||
three : LinkedList Int
|
three : LinkedList Int
|
||||||
three = Cons 3 (Cons 2 (Cons 1 Nil))
|
three = Cons 3 (Cons 2 (Cons 1 Nil))
|
||||||
|
|
||||||
sum : LinkedList a -> Int
|
sum : LinkedList a -> Int
|
||||||
sum = \list ->
|
sum = \list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> 0
|
Nil -> 0
|
||||||
Cons x rest -> x + sum rest
|
Cons x rest -> x + sum rest
|
||||||
|
|
||||||
map : (a -> b), LinkedList a -> LinkedList b
|
map : (a -> b), LinkedList a -> LinkedList b
|
||||||
map = \f, list ->
|
map = \f, list ->
|
||||||
when list is
|
when list is
|
||||||
Nil -> Nil
|
Nil -> Nil
|
||||||
Cons x rest -> Cons (f x) (map f rest)
|
Cons x rest -> Cons (f x) (map f rest)
|
||||||
|
|
||||||
sum (map (\_ -> 1) three)
|
sum (map (\_ -> 1) three)
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3,
|
3,
|
||||||
i64
|
i64
|
||||||
|
@ -699,15 +706,15 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Maybe a : [ Nothing, Just a ]
|
Maybe a : [ Nothing, Just a ]
|
||||||
|
|
||||||
x : Maybe (Maybe Int)
|
x : Maybe (Maybe Int)
|
||||||
x = Just (Just 41)
|
x = Just (Just 41)
|
||||||
|
|
||||||
when x is
|
when x is
|
||||||
Just (Just v) -> v + 0x1
|
Just (Just v) -> v + 0x1
|
||||||
_ -> 0x1
|
_ -> 0x1
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
42,
|
42,
|
||||||
i64
|
i64
|
||||||
|
@ -716,16 +723,16 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Maybe a : [ Nothing, Just a ]
|
Maybe a : [ Nothing, Just a ]
|
||||||
|
|
||||||
x : Maybe (Maybe Int)
|
x : Maybe (Maybe Int)
|
||||||
x = Just Nothing
|
x = Just Nothing
|
||||||
|
|
||||||
when x is
|
when x is
|
||||||
Just (Just v) -> v + 0x1
|
Just (Just v) -> v + 0x1
|
||||||
Just Nothing -> 0x2
|
Just Nothing -> 0x2
|
||||||
Nothing -> 0x1
|
Nothing -> 0x1
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
2,
|
2,
|
||||||
i64
|
i64
|
||||||
|
@ -734,16 +741,16 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Maybe a : [ Nothing, Just a ]
|
Maybe a : [ Nothing, Just a ]
|
||||||
|
|
||||||
x : Maybe (Maybe Int)
|
x : Maybe (Maybe Int)
|
||||||
x = Nothing
|
x = Nothing
|
||||||
|
|
||||||
when x is
|
when x is
|
||||||
Just (Just v) -> v + 0x1
|
Just (Just v) -> v + 0x1
|
||||||
Just Nothing -> 0x2
|
Just Nothing -> 0x2
|
||||||
Nothing -> 0x1
|
Nothing -> 0x1
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
1,
|
1,
|
||||||
i64
|
i64
|
||||||
|
@ -755,16 +762,16 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Peano : [ S Peano, Z ]
|
Peano : [ S Peano, Z ]
|
||||||
|
|
||||||
three : Peano
|
three : Peano
|
||||||
three = S (S (S Z))
|
three = S (S (S Z))
|
||||||
|
|
||||||
when three is
|
when three is
|
||||||
S (S _) -> 1
|
S (S _) -> 1
|
||||||
S (_) -> 2
|
S (_) -> 2
|
||||||
Z -> 3
|
Z -> 3
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
1,
|
1,
|
||||||
i64
|
i64
|
||||||
|
@ -773,16 +780,16 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Peano : [ S Peano, Z ]
|
Peano : [ S Peano, Z ]
|
||||||
|
|
||||||
three : Peano
|
three : Peano
|
||||||
three = S Z
|
three = S Z
|
||||||
|
|
||||||
when three is
|
when three is
|
||||||
S (S _) -> 1
|
S (S _) -> 1
|
||||||
S (_) -> 2
|
S (_) -> 2
|
||||||
Z -> 3
|
Z -> 3
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
2,
|
2,
|
||||||
i64
|
i64
|
||||||
|
@ -791,16 +798,16 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
Peano : [ S Peano, Z ]
|
Peano : [ S Peano, Z ]
|
||||||
|
|
||||||
three : Peano
|
three : Peano
|
||||||
three = Z
|
three = Z
|
||||||
|
|
||||||
when three is
|
when three is
|
||||||
S (S _) -> 1
|
S (S _) -> 1
|
||||||
S (_) -> 2
|
S (_) -> 2
|
||||||
Z -> 3
|
Z -> 3
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3,
|
3,
|
||||||
i64
|
i64
|
||||||
|
@ -813,11 +820,11 @@ mod gen_primitives {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
if True then
|
if True then
|
||||||
x + z
|
x + z
|
||||||
else
|
else
|
||||||
y + z
|
y + z
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3,
|
3,
|
||||||
i64
|
i64
|
||||||
|
|
|
@ -455,12 +455,12 @@ mod gen_tags {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
when 2 is
|
when 2 is
|
||||||
2 if False -> 0
|
2 if False -> 0
|
||||||
_ -> 42
|
_ -> 42
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
42,
|
42,
|
||||||
|
@ -473,12 +473,12 @@ mod gen_tags {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
when 2 is
|
when 2 is
|
||||||
2 if True -> 42
|
2 if True -> 42
|
||||||
_ -> 0
|
_ -> 0
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
42,
|
42,
|
||||||
|
@ -491,12 +491,12 @@ mod gen_tags {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
when 2 is
|
when 2 is
|
||||||
_ if False -> 0
|
_ if False -> 0
|
||||||
_ -> 42
|
_ -> 42
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
42,
|
42,
|
||||||
|
@ -674,7 +674,7 @@ mod gen_tags {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
x : [ Red, White, Blue ]
|
x : [ Red, White, Blue ]
|
||||||
x = Blue
|
x = Blue
|
||||||
|
|
||||||
|
@ -686,7 +686,7 @@ mod gen_tags {
|
||||||
|
|
||||||
y
|
y
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3.1,
|
3.1,
|
||||||
|
@ -699,7 +699,7 @@ mod gen_tags {
|
||||||
assert_evals_to!(
|
assert_evals_to!(
|
||||||
indoc!(
|
indoc!(
|
||||||
r#"
|
r#"
|
||||||
main = \{} ->
|
wrapper = \{} ->
|
||||||
y =
|
y =
|
||||||
when 1 + 2 is
|
when 1 + 2 is
|
||||||
3 -> 3
|
3 -> 3
|
||||||
|
@ -708,7 +708,7 @@ mod gen_tags {
|
||||||
|
|
||||||
y
|
y
|
||||||
|
|
||||||
main {}
|
wrapper {}
|
||||||
"#
|
"#
|
||||||
),
|
),
|
||||||
3,
|
3,
|
||||||
|
|
|
@ -1,9 +1,22 @@
|
||||||
use roc_collections::all::MutSet;
|
use roc_collections::all::{MutMap, MutSet};
|
||||||
use roc_types::subs::Subs;
|
|
||||||
|
|
||||||
pub fn helper_without_uniqueness<'a>(
|
fn promote_expr_to_module(src: &str) -> String {
|
||||||
|
let mut buffer = String::from("app Quicksort provides [ main ] imports []\n\nmain =\n");
|
||||||
|
|
||||||
|
for line in src.lines() {
|
||||||
|
// indent the body!
|
||||||
|
buffer.push_str(" ");
|
||||||
|
buffer.push_str(line);
|
||||||
|
buffer.push('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn helper<'a>(
|
||||||
arena: &'a bumpalo::Bump,
|
arena: &'a bumpalo::Bump,
|
||||||
src: &str,
|
src: &str,
|
||||||
|
stdlib: roc_builtins::std::StdLib,
|
||||||
leak: bool,
|
leak: bool,
|
||||||
context: &'a inkwell::context::Context,
|
context: &'a inkwell::context::Context,
|
||||||
) -> (
|
) -> (
|
||||||
|
@ -11,26 +24,62 @@ pub fn helper_without_uniqueness<'a>(
|
||||||
Vec<roc_problem::can::Problem>,
|
Vec<roc_problem::can::Problem>,
|
||||||
inkwell::execution_engine::ExecutionEngine<'a>,
|
inkwell::execution_engine::ExecutionEngine<'a>,
|
||||||
) {
|
) {
|
||||||
use crate::helpers::{can_expr, infer_expr, CanExprOut};
|
|
||||||
use inkwell::OptimizationLevel;
|
use inkwell::OptimizationLevel;
|
||||||
use roc_gen::llvm::build::{build_proc, build_proc_header};
|
use roc_gen::llvm::build::{build_proc, build_proc_header};
|
||||||
use roc_mono::layout::{Layout, LayoutCache};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
|
let stdlib_mode = stdlib.mode;
|
||||||
|
let filename = PathBuf::from("Test.roc");
|
||||||
|
let src_dir = Path::new("fake/test/path");
|
||||||
|
|
||||||
|
let module_src;
|
||||||
|
let temp;
|
||||||
|
if src.starts_with("app") {
|
||||||
|
// this is already a module
|
||||||
|
module_src = src;
|
||||||
|
} else {
|
||||||
|
// this is an expression, promote it to a module
|
||||||
|
temp = promote_expr_to_module(src);
|
||||||
|
module_src = &temp;
|
||||||
|
}
|
||||||
|
|
||||||
|
let exposed_types = MutMap::default();
|
||||||
|
let loaded = roc_load::file::load_and_monomorphize_from_str(
|
||||||
|
arena,
|
||||||
|
filename,
|
||||||
|
&module_src,
|
||||||
|
stdlib,
|
||||||
|
src_dir,
|
||||||
|
exposed_types,
|
||||||
|
);
|
||||||
|
|
||||||
|
let loaded = loaded.expect("failed to load module");
|
||||||
|
|
||||||
|
use roc_load::file::MonomorphizedModule;
|
||||||
|
let MonomorphizedModule {
|
||||||
|
can_problems,
|
||||||
|
type_problems,
|
||||||
|
mono_problems,
|
||||||
|
mut procedures,
|
||||||
|
interns,
|
||||||
|
exposed_to_host,
|
||||||
|
..
|
||||||
|
} = loaded;
|
||||||
|
|
||||||
|
debug_assert_eq!(exposed_to_host.len(), 1);
|
||||||
|
let main_fn_symbol = exposed_to_host.iter().copied().nth(0).unwrap();
|
||||||
|
|
||||||
|
let (_, main_fn_layout) = procedures
|
||||||
|
.keys()
|
||||||
|
.find(|(s, _)| *s == main_fn_symbol)
|
||||||
|
.unwrap()
|
||||||
|
.clone();
|
||||||
|
|
||||||
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;
|
||||||
let CanExprOut {
|
|
||||||
loc_expr,
|
|
||||||
var_store,
|
|
||||||
var,
|
|
||||||
constraint,
|
|
||||||
home,
|
|
||||||
interns,
|
|
||||||
problems,
|
|
||||||
..
|
|
||||||
} = can_expr(src);
|
|
||||||
|
|
||||||
// don't panic based on the errors here, so we can test that RuntimeError generates the correct code
|
// don't panic based on the errors here, so we can test that RuntimeError generates the correct code
|
||||||
let errors = problems
|
let errors = can_problems
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.filter(|problem| {
|
.filter(|problem| {
|
||||||
use roc_problem::can::Problem::*;
|
use roc_problem::can::Problem::*;
|
||||||
|
@ -43,15 +92,18 @@ pub fn helper_without_uniqueness<'a>(
|
||||||
})
|
})
|
||||||
.collect::<Vec<roc_problem::can::Problem>>();
|
.collect::<Vec<roc_problem::can::Problem>>();
|
||||||
|
|
||||||
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!(
|
assert_eq!(
|
||||||
unify_problems,
|
type_problems,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
"Encountered type mismatches: {:?}",
|
"Encountered type mismatches: {:?}",
|
||||||
unify_problems
|
type_problems,
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
mono_problems,
|
||||||
|
Vec::new(),
|
||||||
|
"Encountered monomorphization errors: {:?}",
|
||||||
|
mono_problems,
|
||||||
);
|
);
|
||||||
|
|
||||||
let module = roc_gen::llvm::build::module_from_builtins(context, "app");
|
let module = roc_gen::llvm::build::module_from_builtins(context, "app");
|
||||||
|
@ -66,77 +118,30 @@ pub fn helper_without_uniqueness<'a>(
|
||||||
let (module_pass, function_pass) =
|
let (module_pass, function_pass) =
|
||||||
roc_gen::llvm::build::construct_optimization_passes(module, opt_level);
|
roc_gen::llvm::build::construct_optimization_passes(module, opt_level);
|
||||||
|
|
||||||
// Compute main_fn_type before moving subs to Env
|
|
||||||
let return_layout = Layout::new(&arena, content, &subs).unwrap_or_else(|err| {
|
|
||||||
panic!(
|
|
||||||
"Code gen error in NON-OPTIMIZED test: could not convert to layout. Err was {:?}",
|
|
||||||
err
|
|
||||||
)
|
|
||||||
});
|
|
||||||
let execution_engine = module
|
let execution_engine = module
|
||||||
.create_jit_execution_engine(OptimizationLevel::None)
|
.create_jit_execution_engine(OptimizationLevel::None)
|
||||||
.expect("Error creating JIT execution engine for test");
|
.expect("Error creating JIT execution engine for test");
|
||||||
|
|
||||||
// Compile and add all the Procs before adding main
|
// Compile and add all the Procs before adding main
|
||||||
let mut env = roc_gen::llvm::build::Env {
|
let env = roc_gen::llvm::build::Env {
|
||||||
arena: &arena,
|
arena: &arena,
|
||||||
builder: &builder,
|
builder: &builder,
|
||||||
context: context,
|
context,
|
||||||
interns,
|
interns,
|
||||||
module,
|
module,
|
||||||
ptr_bytes,
|
ptr_bytes,
|
||||||
leak: leak,
|
leak,
|
||||||
|
// important! we don't want any procedures to get the C calling convention
|
||||||
exposed_to_host: MutSet::default(),
|
exposed_to_host: MutSet::default(),
|
||||||
};
|
};
|
||||||
let mut procs = roc_mono::ir::Procs::default();
|
|
||||||
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
|
let mut headers = Vec::with_capacity(procedures.len());
|
||||||
let mut mono_problems = Vec::new();
|
|
||||||
let mut mono_env = roc_mono::ir::Env {
|
|
||||||
arena: &arena,
|
|
||||||
subs: &mut subs,
|
|
||||||
problems: &mut mono_problems,
|
|
||||||
home,
|
|
||||||
ident_ids: &mut ident_ids,
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut layout_cache = LayoutCache::default();
|
|
||||||
let main_body =
|
|
||||||
roc_mono::ir::Stmt::new(&mut mono_env, loc_expr.value, &mut procs, &mut layout_cache);
|
|
||||||
|
|
||||||
let mut headers = {
|
|
||||||
let num_headers = match &procs.pending_specializations {
|
|
||||||
Some(map) => map.len(),
|
|
||||||
None => 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
Vec::with_capacity(num_headers)
|
|
||||||
};
|
|
||||||
let procs = roc_mono::ir::specialize_all(&mut mono_env, procs, &mut layout_cache);
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
procs.runtime_errors,
|
|
||||||
roc_collections::all::MutMap::default()
|
|
||||||
);
|
|
||||||
|
|
||||||
let (mut procs, param_map) = procs.get_specialized_procs_help(mono_env.arena);
|
|
||||||
let main_body = roc_mono::inc_dec::visit_declaration(
|
|
||||||
mono_env.arena,
|
|
||||||
param_map,
|
|
||||||
mono_env.arena.alloc(main_body),
|
|
||||||
);
|
|
||||||
|
|
||||||
// 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.
|
// Add all the Proc headers to the module.
|
||||||
// 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.drain() {
|
for ((symbol, layout), proc) in procedures.drain() {
|
||||||
let fn_val = build_proc_header(&env, &mut layout_ids, symbol, &layout, &proc);
|
let fn_val = build_proc_header(&env, &mut layout_ids, symbol, &layout, &proc);
|
||||||
|
|
||||||
headers.push((proc, fn_val));
|
headers.push((proc, fn_val));
|
||||||
|
@ -149,20 +154,34 @@ pub fn helper_without_uniqueness<'a>(
|
||||||
if fn_val.verify(true) {
|
if fn_val.verify(true) {
|
||||||
function_pass.run_on(&fn_val);
|
function_pass.run_on(&fn_val);
|
||||||
} else {
|
} else {
|
||||||
|
use roc_builtins::std::Mode;
|
||||||
|
|
||||||
|
let mode = match stdlib_mode {
|
||||||
|
Mode::Uniqueness => "OPTIMIZED",
|
||||||
|
Mode::Standard => "NON-OPTIMIZED",
|
||||||
|
};
|
||||||
|
|
||||||
eprintln!(
|
eprintln!(
|
||||||
"\n\nFunction {:?} failed LLVM verification in NON-OPTIMIZED build. Its content was:\n", fn_val.get_name().to_str().unwrap()
|
"\n\nFunction {:?} failed LLVM verification in {} build. Its content was:\n",
|
||||||
);
|
fn_val.get_name().to_str().unwrap(),
|
||||||
|
mode,
|
||||||
|
);
|
||||||
|
|
||||||
fn_val.print_to_stderr();
|
fn_val.print_to_stderr();
|
||||||
|
|
||||||
panic!(
|
panic!(
|
||||||
"The preceding code was from {:?}, which failed LLVM verification in NON-OPTIMIZED build.", fn_val.get_name().to_str().unwrap()
|
"The preceding code was from {:?}, which failed LLVM verification in {} build.",
|
||||||
);
|
fn_val.get_name().to_str().unwrap(),
|
||||||
|
mode,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
let (main_fn_name, main_fn) = roc_gen::llvm::build::promote_to_main_function(
|
||||||
let (main_fn_name, main_fn) =
|
&env,
|
||||||
roc_gen::llvm::build::make_main_function(&env, &mut layout_ids, &return_layout, &main_body);
|
&mut layout_ids,
|
||||||
|
main_fn_symbol,
|
||||||
|
&main_fn_layout,
|
||||||
|
);
|
||||||
|
|
||||||
// Uncomment this to see the module's un-optimized LLVM instruction output:
|
// Uncomment this to see the module's un-optimized LLVM instruction output:
|
||||||
// env.module.print_to_stderr();
|
// env.module.print_to_stderr();
|
||||||
|
@ -186,178 +205,6 @@ pub fn helper_without_uniqueness<'a>(
|
||||||
(main_fn_name, errors, execution_engine.clone())
|
(main_fn_name, errors, execution_engine.clone())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn helper_with_uniqueness<'a>(
|
|
||||||
arena: &'a bumpalo::Bump,
|
|
||||||
src: &str,
|
|
||||||
leak: bool,
|
|
||||||
context: &'a inkwell::context::Context,
|
|
||||||
) -> (&'static str, inkwell::execution_engine::ExecutionEngine<'a>) {
|
|
||||||
use crate::helpers::{infer_expr, uniq_expr};
|
|
||||||
use inkwell::OptimizationLevel;
|
|
||||||
use roc_gen::llvm::build::{build_proc, build_proc_header};
|
|
||||||
use roc_mono::layout::{Layout, LayoutCache};
|
|
||||||
|
|
||||||
let target = target_lexicon::Triple::host();
|
|
||||||
let ptr_bytes = target.pointer_width().unwrap().bytes() as u32;
|
|
||||||
let (loc_expr, _output, problems, subs, var, constraint, home, interns) = uniq_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 mut unify_problems = Vec::new();
|
|
||||||
let (content, mut subs) = infer_expr(subs, &mut unify_problems, &constraint, var);
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
unify_problems,
|
|
||||||
Vec::new(),
|
|
||||||
"Encountered one or more type mismatches: {:?}",
|
|
||||||
unify_problems
|
|
||||||
);
|
|
||||||
|
|
||||||
let module = arena.alloc(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 (mpm, fpm) = roc_gen::llvm::build::construct_optimization_passes(module, opt_level);
|
|
||||||
|
|
||||||
// Compute main_fn_type before moving subs to Env
|
|
||||||
let return_layout = Layout::new(&arena, content, &subs).unwrap_or_else(|err| {
|
|
||||||
panic!(
|
|
||||||
"Code gen error in 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");
|
|
||||||
|
|
||||||
// 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,
|
|
||||||
ptr_bytes,
|
|
||||||
leak: leak,
|
|
||||||
exposed_to_host: MutSet::default(),
|
|
||||||
};
|
|
||||||
let mut procs = roc_mono::ir::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::ir::Env {
|
|
||||||
arena: &arena,
|
|
||||||
subs: &mut subs,
|
|
||||||
problems: &mut mono_problems,
|
|
||||||
home,
|
|
||||||
ident_ids: &mut ident_ids,
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut layout_cache = LayoutCache::default();
|
|
||||||
let main_body =
|
|
||||||
roc_mono::ir::Stmt::new(&mut mono_env, loc_expr.value, &mut procs, &mut layout_cache);
|
|
||||||
let mut headers = {
|
|
||||||
let num_headers = match &procs.pending_specializations {
|
|
||||||
Some(map) => map.len(),
|
|
||||||
None => 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
Vec::with_capacity(num_headers)
|
|
||||||
};
|
|
||||||
let procs = roc_mono::ir::specialize_all(&mut mono_env, procs, &mut layout_cache);
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
procs.runtime_errors,
|
|
||||||
roc_collections::all::MutMap::default()
|
|
||||||
);
|
|
||||||
|
|
||||||
let (mut procs, param_map) = procs.get_specialized_procs_help(mono_env.arena);
|
|
||||||
let main_body = roc_mono::inc_dec::visit_declaration(
|
|
||||||
mono_env.arena,
|
|
||||||
param_map,
|
|
||||||
mono_env.arena.alloc(main_body),
|
|
||||||
);
|
|
||||||
|
|
||||||
// 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.drain() {
|
|
||||||
let fn_val = build_proc_header(&env, &mut layout_ids, symbol, &layout, &proc);
|
|
||||||
|
|
||||||
headers.push((proc, fn_val));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build each proc using its header info.
|
|
||||||
for (proc, fn_val) in headers {
|
|
||||||
build_proc(&env, &mut layout_ids, proc, fn_val);
|
|
||||||
|
|
||||||
if fn_val.verify(true) {
|
|
||||||
fpm.run_on(&fn_val);
|
|
||||||
} else {
|
|
||||||
eprintln!(
|
|
||||||
"\n\nFunction {:?} failed LLVM verification in 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 OPTIMIZED build.", fn_val.get_name().to_str().unwrap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let (main_fn_name, main_fn) =
|
|
||||||
roc_gen::llvm::build::make_main_function(&env, &mut layout_ids, &return_layout, &main_body);
|
|
||||||
|
|
||||||
// you're in the version with uniqueness!
|
|
||||||
|
|
||||||
// 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 OPTIMIZED build. Uncomment nearby statements to see more details.", main_fn_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
mpm.run_on(module);
|
|
||||||
|
|
||||||
// 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();
|
|
||||||
|
|
||||||
(main_fn_name, execution_engine)
|
|
||||||
}
|
|
||||||
|
|
||||||
// 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.
|
||||||
|
@ -372,11 +219,17 @@ macro_rules! assert_opt_evals_to {
|
||||||
|
|
||||||
let context = Context::create();
|
let context = Context::create();
|
||||||
|
|
||||||
let (main_fn_name, execution_engine) =
|
let stdlib = roc_builtins::unique::uniq_stdlib();
|
||||||
$crate::helpers::eval::helper_with_uniqueness(&arena, $src, $leak, &context);
|
|
||||||
|
|
||||||
let transform = |success| assert_eq!($transform(success), $expected);
|
let (main_fn_name, errors, execution_engine) =
|
||||||
run_jit_function!(execution_engine, main_fn_name, $ty, transform)
|
$crate::helpers::eval::helper(&arena, $src, stdlib, $leak, &context);
|
||||||
|
|
||||||
|
let transform = |success| {
|
||||||
|
let expected = $expected;
|
||||||
|
let given = $transform(success);
|
||||||
|
assert_eq!(&given, &expected);
|
||||||
|
};
|
||||||
|
run_jit_function!(execution_engine, main_fn_name, $ty, transform, errors)
|
||||||
};
|
};
|
||||||
|
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
||||||
|
@ -394,9 +247,10 @@ macro_rules! assert_llvm_evals_to {
|
||||||
let arena = Bump::new();
|
let arena = Bump::new();
|
||||||
|
|
||||||
let context = Context::create();
|
let context = Context::create();
|
||||||
|
let stdlib = roc_builtins::std::standard_stdlib();
|
||||||
|
|
||||||
let (main_fn_name, errors, execution_engine) =
|
let (main_fn_name, errors, execution_engine) =
|
||||||
$crate::helpers::eval::helper_without_uniqueness(&arena, $src, $leak, &context);
|
$crate::helpers::eval::helper(&arena, $src, stdlib, $leak, &context);
|
||||||
|
|
||||||
let transform = |success| {
|
let transform = |success| {
|
||||||
let expected = $expected;
|
let expected = $expected;
|
||||||
|
@ -413,29 +267,20 @@ macro_rules! assert_llvm_evals_to {
|
||||||
|
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! assert_evals_to {
|
macro_rules! assert_evals_to {
|
||||||
($src:expr, $expected:expr, $ty:ty) => {
|
($src:expr, $expected:expr, $ty:ty) => {{
|
||||||
|
assert_evals_to!($src, $expected, $ty, (|val| val));
|
||||||
|
}};
|
||||||
|
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
||||||
|
// Same as above, except with an additional transformation argument.
|
||||||
|
{
|
||||||
|
assert_evals_to!($src, $expected, $ty, $transform, true);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
($src:expr, $expected:expr, $ty:ty, $transform:expr, $leak:expr) => {
|
||||||
// Run un-optimized tests, and then optimized tests, in separate scopes.
|
// Run un-optimized tests, and then optimized tests, in separate scopes.
|
||||||
// These each rebuild everything from scratch, starting with
|
// These each rebuild everything from scratch, starting with
|
||||||
// parsing the source, so that there's no chance their passing
|
// parsing the source, so that there's no chance their passing
|
||||||
// or failing depends on leftover state from the previous one.
|
// or failing depends on leftover state from the previous one.
|
||||||
{
|
|
||||||
assert_llvm_evals_to!($src, $expected, $ty, (|val| val));
|
|
||||||
}
|
|
||||||
{
|
|
||||||
assert_opt_evals_to!($src, $expected, $ty, (|val| val));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr) => {
|
|
||||||
// Same as above, except with an additional transformation argument.
|
|
||||||
{
|
|
||||||
assert_llvm_evals_to!($src, $expected, $ty, $transform);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
assert_opt_evals_to!($src, $expected, $ty, $transform);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
($src:expr, $expected:expr, $ty:ty, $transform:expr, $leak:expr) => {
|
|
||||||
// Same as above, except with an additional transformation argument.
|
|
||||||
{
|
{
|
||||||
assert_llvm_evals_to!($src, $expected, $ty, $transform, $leak);
|
assert_llvm_evals_to!($src, $expected, $ty, $transform, $leak);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,32 +3,6 @@ extern crate bumpalo;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
pub mod eval;
|
pub mod eval;
|
||||||
|
|
||||||
use self::bumpalo::Bump;
|
|
||||||
use roc_builtins::unique::uniq_stdlib;
|
|
||||||
use roc_can::constraint::Constraint;
|
|
||||||
use roc_can::env::Env;
|
|
||||||
use roc_can::expected::Expected;
|
|
||||||
use roc_can::expr::{canonicalize_expr, Expr, Output};
|
|
||||||
use roc_can::operator;
|
|
||||||
use roc_can::scope::Scope;
|
|
||||||
use roc_collections::all::{ImMap, MutMap, SendMap};
|
|
||||||
use roc_constrain::expr::constrain_expr;
|
|
||||||
use roc_constrain::module::{constrain_imported_values, load_builtin_aliases, Import};
|
|
||||||
use roc_module::ident::Ident;
|
|
||||||
use roc_module::symbol::{IdentIds, Interns, ModuleId, ModuleIds, Symbol};
|
|
||||||
use roc_parse::ast::{self, Attempting};
|
|
||||||
use roc_parse::blankspace::space0_before;
|
|
||||||
use roc_parse::parser::{loc, Fail, Parser, State};
|
|
||||||
use roc_problem::can::Problem;
|
|
||||||
use roc_region::all::{Located, Region};
|
|
||||||
use roc_solve::solve;
|
|
||||||
use roc_types::subs::{Content, Subs, VarStore, Variable};
|
|
||||||
use roc_types::types::Type;
|
|
||||||
|
|
||||||
pub fn test_home() -> ModuleId {
|
|
||||||
ModuleIds::default().get_or_insert(&"Test".into())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Used in the with_larger_debug_stack() function, for tests that otherwise
|
/// Used in the with_larger_debug_stack() function, for tests that otherwise
|
||||||
/// run out of stack space in debug builds (but don't in --release builds)
|
/// run out of stack space in debug builds (but don't in --release builds)
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
|
@ -68,249 +42,3 @@ where
|
||||||
{
|
{
|
||||||
run_test()
|
run_test()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn infer_expr(
|
|
||||||
subs: Subs,
|
|
||||||
problems: &mut Vec<roc_solve::solve::TypeError>,
|
|
||||||
constraint: &Constraint,
|
|
||||||
expr_var: Variable,
|
|
||||||
) -> (Content, Subs) {
|
|
||||||
let env = solve::Env {
|
|
||||||
aliases: MutMap::default(),
|
|
||||||
vars_by_symbol: SendMap::default(),
|
|
||||||
};
|
|
||||||
let (solved, _) = solve::run(&env, problems, subs, constraint);
|
|
||||||
|
|
||||||
let content = solved.inner().get_without_compacting(expr_var).content;
|
|
||||||
|
|
||||||
(content, solved.into_inner())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_loc_with<'a>(arena: &'a Bump, input: &'a str) -> Result<Located<ast::Expr<'a>>, Fail> {
|
|
||||||
let state = State::new(input.trim().as_bytes(), Attempting::Module);
|
|
||||||
let parser = space0_before(loc(roc_parse::expr::expr(0)), 0);
|
|
||||||
let answer = parser.parse(&arena, state);
|
|
||||||
|
|
||||||
answer
|
|
||||||
.map(|(loc_expr, _)| loc_expr)
|
|
||||||
.map_err(|(fail, _)| fail)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn can_expr(expr_str: &str) -> CanExprOut {
|
|
||||||
can_expr_with(&Bump::new(), test_home(), expr_str)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn uniq_expr(
|
|
||||||
expr_str: &str,
|
|
||||||
) -> (
|
|
||||||
Located<Expr>,
|
|
||||||
Output,
|
|
||||||
Vec<Problem>,
|
|
||||||
Subs,
|
|
||||||
Variable,
|
|
||||||
Constraint,
|
|
||||||
ModuleId,
|
|
||||||
Interns,
|
|
||||||
) {
|
|
||||||
let declared_idents: &ImMap<Ident, (Symbol, Region)> = &ImMap::default();
|
|
||||||
|
|
||||||
uniq_expr_with(&Bump::new(), expr_str, declared_idents)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn uniq_expr_with(
|
|
||||||
arena: &Bump,
|
|
||||||
expr_str: &str,
|
|
||||||
declared_idents: &ImMap<Ident, (Symbol, Region)>,
|
|
||||||
) -> (
|
|
||||||
Located<Expr>,
|
|
||||||
Output,
|
|
||||||
Vec<Problem>,
|
|
||||||
Subs,
|
|
||||||
Variable,
|
|
||||||
Constraint,
|
|
||||||
ModuleId,
|
|
||||||
Interns,
|
|
||||||
) {
|
|
||||||
let home = test_home();
|
|
||||||
let CanExprOut {
|
|
||||||
loc_expr,
|
|
||||||
output,
|
|
||||||
problems,
|
|
||||||
var_store: mut old_var_store,
|
|
||||||
var,
|
|
||||||
interns,
|
|
||||||
..
|
|
||||||
} = can_expr_with(arena, home, expr_str);
|
|
||||||
|
|
||||||
// double check
|
|
||||||
let mut var_store = VarStore::new(old_var_store.fresh());
|
|
||||||
|
|
||||||
let expected2 = Expected::NoExpectation(Type::Variable(var));
|
|
||||||
let constraint = roc_constrain::uniq::constrain_declaration(
|
|
||||||
home,
|
|
||||||
&mut var_store,
|
|
||||||
Region::zero(),
|
|
||||||
&loc_expr,
|
|
||||||
declared_idents,
|
|
||||||
expected2,
|
|
||||||
);
|
|
||||||
|
|
||||||
let stdlib = uniq_stdlib();
|
|
||||||
|
|
||||||
let types = stdlib.types;
|
|
||||||
let imports: Vec<_> = types
|
|
||||||
.into_iter()
|
|
||||||
.map(|(symbol, (solved_type, region))| Import {
|
|
||||||
loc_symbol: Located::at(region, symbol),
|
|
||||||
solved_type,
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
// load builtin values
|
|
||||||
|
|
||||||
// TODO what to do with those rigids?
|
|
||||||
let (_introduced_rigids, constraint) =
|
|
||||||
constrain_imported_values(imports, constraint, &mut var_store);
|
|
||||||
|
|
||||||
// load builtin types
|
|
||||||
let mut constraint = load_builtin_aliases(stdlib.aliases, constraint, &mut var_store);
|
|
||||||
|
|
||||||
constraint.instantiate_aliases(&mut var_store);
|
|
||||||
|
|
||||||
let subs2 = Subs::new(var_store.into());
|
|
||||||
|
|
||||||
(
|
|
||||||
loc_expr, output, problems, subs2, var, constraint, home, interns,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct CanExprOut {
|
|
||||||
pub loc_expr: Located<Expr>,
|
|
||||||
pub output: Output,
|
|
||||||
pub problems: Vec<Problem>,
|
|
||||||
pub home: ModuleId,
|
|
||||||
pub interns: Interns,
|
|
||||||
pub var_store: VarStore,
|
|
||||||
pub var: Variable,
|
|
||||||
pub constraint: Constraint,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn can_expr_with(arena: &Bump, home: ModuleId, expr_str: &str) -> CanExprOut {
|
|
||||||
let loc_expr = parse_loc_with(&arena, expr_str).unwrap_or_else(|e| {
|
|
||||||
panic!(
|
|
||||||
"can_expr_with() got a parse error when attempting to canonicalize:\n\n{:?} {:?}",
|
|
||||||
expr_str, e
|
|
||||||
)
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut var_store = VarStore::default();
|
|
||||||
let var = var_store.fresh();
|
|
||||||
let expected = Expected::NoExpectation(Type::Variable(var));
|
|
||||||
let module_ids = ModuleIds::default();
|
|
||||||
|
|
||||||
// Desugar operators (convert them to Apply calls, taking into account
|
|
||||||
// operator precedence and associativity rules), before doing other canonicalization.
|
|
||||||
//
|
|
||||||
// If we did this *during* canonicalization, then each time we
|
|
||||||
// visited a BinOp node we'd recursively try to apply this to each of its nested
|
|
||||||
// operators, and then again on *their* nested operators, ultimately applying the
|
|
||||||
// rules multiple times unnecessarily.
|
|
||||||
let loc_expr = operator::desugar_expr(arena, &loc_expr);
|
|
||||||
|
|
||||||
let mut scope = Scope::new(home);
|
|
||||||
let dep_idents = IdentIds::exposed_builtins(0);
|
|
||||||
let mut env = Env::new(home, dep_idents, &module_ids, IdentIds::default());
|
|
||||||
let (loc_expr, output) = canonicalize_expr(
|
|
||||||
&mut env,
|
|
||||||
&mut var_store,
|
|
||||||
&mut scope,
|
|
||||||
Region::zero(),
|
|
||||||
&loc_expr.value,
|
|
||||||
);
|
|
||||||
|
|
||||||
// Add builtin defs (e.g. List.get) directly to the canonical Expr,
|
|
||||||
// since we aren't using modules here.
|
|
||||||
let mut with_builtins = loc_expr.value;
|
|
||||||
let builtin_defs = roc_can::builtins::builtin_defs(&mut var_store);
|
|
||||||
|
|
||||||
for (symbol, def) in builtin_defs {
|
|
||||||
if output.references.lookups.contains(&symbol) || output.references.calls.contains(&symbol)
|
|
||||||
{
|
|
||||||
with_builtins = Expr::LetNonRec(
|
|
||||||
Box::new(def),
|
|
||||||
Box::new(Located {
|
|
||||||
region: Region::zero(),
|
|
||||||
value: with_builtins,
|
|
||||||
}),
|
|
||||||
var_store.fresh(),
|
|
||||||
SendMap::default(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let loc_expr = Located {
|
|
||||||
region: loc_expr.region,
|
|
||||||
value: with_builtins,
|
|
||||||
};
|
|
||||||
|
|
||||||
let constraint = constrain_expr(
|
|
||||||
&roc_constrain::expr::Env {
|
|
||||||
rigids: ImMap::default(),
|
|
||||||
home,
|
|
||||||
},
|
|
||||||
loc_expr.region,
|
|
||||||
&loc_expr.value,
|
|
||||||
expected,
|
|
||||||
);
|
|
||||||
|
|
||||||
let types = roc_builtins::std::types();
|
|
||||||
|
|
||||||
let imports: Vec<_> = types
|
|
||||||
.into_iter()
|
|
||||||
.map(|(symbol, (solved_type, region))| Import {
|
|
||||||
loc_symbol: Located::at(region, symbol),
|
|
||||||
solved_type,
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
// load builtin values
|
|
||||||
let (_introduced_rigids, constraint) =
|
|
||||||
constrain_imported_values(imports, constraint, &mut var_store);
|
|
||||||
|
|
||||||
// TODO determine what to do with those rigids
|
|
||||||
// for var in introduced_rigids {
|
|
||||||
// output.ftv.insert(var, format!("internal_{:?}", var).into());
|
|
||||||
// }
|
|
||||||
|
|
||||||
//load builtin types
|
|
||||||
let mut constraint =
|
|
||||||
load_builtin_aliases(roc_builtins::std::aliases(), constraint, &mut var_store);
|
|
||||||
|
|
||||||
constraint.instantiate_aliases(&mut var_store);
|
|
||||||
|
|
||||||
let mut all_ident_ids = MutMap::default();
|
|
||||||
|
|
||||||
// When pretty printing types, we may need the exposed builtins,
|
|
||||||
// so include them in the Interns we'll ultimately return.
|
|
||||||
for (module_id, ident_ids) in IdentIds::exposed_builtins(0) {
|
|
||||||
all_ident_ids.insert(module_id, ident_ids);
|
|
||||||
}
|
|
||||||
|
|
||||||
all_ident_ids.insert(home, env.ident_ids);
|
|
||||||
|
|
||||||
let interns = Interns {
|
|
||||||
module_ids: env.module_ids.clone(),
|
|
||||||
all_ident_ids,
|
|
||||||
};
|
|
||||||
|
|
||||||
CanExprOut {
|
|
||||||
loc_expr,
|
|
||||||
output,
|
|
||||||
problems: env.problems,
|
|
||||||
home: env.home,
|
|
||||||
var_store,
|
|
||||||
interns,
|
|
||||||
var,
|
|
||||||
constraint,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -43,7 +43,7 @@ const ROC_FILE_EXTENSION: &str = "roc";
|
||||||
/// The . in between module names like Foo.Bar.Baz
|
/// The . in between module names like Foo.Bar.Baz
|
||||||
const MODULE_SEPARATOR: char = '.';
|
const MODULE_SEPARATOR: char = '.';
|
||||||
|
|
||||||
const SHOW_MESSAGE_LOG: bool = true;
|
const SHOW_MESSAGE_LOG: bool = false;
|
||||||
|
|
||||||
macro_rules! log {
|
macro_rules! log {
|
||||||
() => (if SHOW_MESSAGE_LOG { println!()} else {});
|
() => (if SHOW_MESSAGE_LOG { println!()} else {});
|
||||||
|
@ -736,9 +736,11 @@ pub fn load_and_typecheck(
|
||||||
) -> Result<LoadedModule, LoadingProblem> {
|
) -> Result<LoadedModule, LoadingProblem> {
|
||||||
use LoadResult::*;
|
use LoadResult::*;
|
||||||
|
|
||||||
|
let load_start = LoadStart::from_path(arena, filename)?;
|
||||||
|
|
||||||
match load(
|
match load(
|
||||||
arena,
|
arena,
|
||||||
filename,
|
load_start,
|
||||||
stdlib,
|
stdlib,
|
||||||
src_dir,
|
src_dir,
|
||||||
exposed_types,
|
exposed_types,
|
||||||
|
@ -758,9 +760,11 @@ pub fn load_and_monomorphize<'a>(
|
||||||
) -> Result<MonomorphizedModule<'a>, LoadingProblem> {
|
) -> Result<MonomorphizedModule<'a>, LoadingProblem> {
|
||||||
use LoadResult::*;
|
use LoadResult::*;
|
||||||
|
|
||||||
|
let load_start = LoadStart::from_path(arena, filename)?;
|
||||||
|
|
||||||
match load(
|
match load(
|
||||||
arena,
|
arena,
|
||||||
filename,
|
load_start,
|
||||||
stdlib,
|
stdlib,
|
||||||
src_dir,
|
src_dir,
|
||||||
exposed_types,
|
exposed_types,
|
||||||
|
@ -771,6 +775,97 @@ pub fn load_and_monomorphize<'a>(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn load_and_monomorphize_from_str<'a>(
|
||||||
|
arena: &'a Bump,
|
||||||
|
filename: PathBuf,
|
||||||
|
src: &'a str,
|
||||||
|
stdlib: StdLib,
|
||||||
|
src_dir: &Path,
|
||||||
|
exposed_types: SubsByModule,
|
||||||
|
) -> Result<MonomorphizedModule<'a>, LoadingProblem> {
|
||||||
|
use LoadResult::*;
|
||||||
|
|
||||||
|
let load_start = LoadStart::from_str(arena, filename, src)?;
|
||||||
|
|
||||||
|
match load(
|
||||||
|
arena,
|
||||||
|
load_start,
|
||||||
|
stdlib,
|
||||||
|
src_dir,
|
||||||
|
exposed_types,
|
||||||
|
Phase::MakeSpecializations,
|
||||||
|
)? {
|
||||||
|
Monomorphized(module) => Ok(module),
|
||||||
|
TypeChecked(_) => unreachable!(""),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct LoadStart<'a> {
|
||||||
|
pub arc_modules: Arc<Mutex<ModuleIds>>,
|
||||||
|
pub ident_ids_by_module: Arc<Mutex<IdentIdsByModule>>,
|
||||||
|
pub root_id: ModuleId,
|
||||||
|
pub root_msg: Msg<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> LoadStart<'a> {
|
||||||
|
pub fn from_path(arena: &'a Bump, filename: PathBuf) -> Result<Self, LoadingProblem> {
|
||||||
|
let arc_modules = Arc::new(Mutex::new(ModuleIds::default()));
|
||||||
|
let root_exposed_ident_ids = IdentIds::exposed_builtins(0);
|
||||||
|
let ident_ids_by_module = Arc::new(Mutex::new(root_exposed_ident_ids));
|
||||||
|
|
||||||
|
// Load the root module synchronously; we can't proceed until we have its id.
|
||||||
|
let (root_id, root_msg) = {
|
||||||
|
let root_start_time = SystemTime::now();
|
||||||
|
|
||||||
|
load_filename(
|
||||||
|
arena,
|
||||||
|
filename,
|
||||||
|
Arc::clone(&arc_modules),
|
||||||
|
Arc::clone(&ident_ids_by_module),
|
||||||
|
root_start_time,
|
||||||
|
)?
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(LoadStart {
|
||||||
|
arc_modules,
|
||||||
|
ident_ids_by_module,
|
||||||
|
root_id,
|
||||||
|
root_msg,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_str(
|
||||||
|
arena: &'a Bump,
|
||||||
|
filename: PathBuf,
|
||||||
|
src: &'a str,
|
||||||
|
) -> Result<Self, LoadingProblem> {
|
||||||
|
let arc_modules = Arc::new(Mutex::new(ModuleIds::default()));
|
||||||
|
let root_exposed_ident_ids = IdentIds::exposed_builtins(0);
|
||||||
|
let ident_ids_by_module = Arc::new(Mutex::new(root_exposed_ident_ids));
|
||||||
|
|
||||||
|
// Load the root module synchronously; we can't proceed until we have its id.
|
||||||
|
let (root_id, root_msg) = {
|
||||||
|
let root_start_time = SystemTime::now();
|
||||||
|
|
||||||
|
load_from_str(
|
||||||
|
arena,
|
||||||
|
filename,
|
||||||
|
src,
|
||||||
|
Arc::clone(&arc_modules),
|
||||||
|
Arc::clone(&ident_ids_by_module),
|
||||||
|
root_start_time,
|
||||||
|
)?
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(LoadStart {
|
||||||
|
arc_modules,
|
||||||
|
ident_ids_by_module,
|
||||||
|
root_id,
|
||||||
|
root_msg,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
enum LoadResult<'a> {
|
enum LoadResult<'a> {
|
||||||
TypeChecked(LoadedModule),
|
TypeChecked(LoadedModule),
|
||||||
Monomorphized(MonomorphizedModule<'a>),
|
Monomorphized(MonomorphizedModule<'a>),
|
||||||
|
@ -821,7 +916,8 @@ enum LoadResult<'a> {
|
||||||
/// to rebuild the module and can link in the cached one directly.)
|
/// to rebuild the module and can link in the cached one directly.)
|
||||||
fn load<'a>(
|
fn load<'a>(
|
||||||
arena: &'a Bump,
|
arena: &'a Bump,
|
||||||
filename: PathBuf,
|
//filename: PathBuf,
|
||||||
|
load_start: LoadStart<'a>,
|
||||||
stdlib: StdLib,
|
stdlib: StdLib,
|
||||||
src_dir: &Path,
|
src_dir: &Path,
|
||||||
exposed_types: SubsByModule,
|
exposed_types: SubsByModule,
|
||||||
|
@ -829,6 +925,18 @@ fn load<'a>(
|
||||||
) -> Result<LoadResult<'a>, LoadingProblem>
|
) -> Result<LoadResult<'a>, LoadingProblem>
|
||||||
where
|
where
|
||||||
{
|
{
|
||||||
|
let LoadStart {
|
||||||
|
arc_modules,
|
||||||
|
ident_ids_by_module,
|
||||||
|
root_id,
|
||||||
|
root_msg,
|
||||||
|
} = load_start;
|
||||||
|
|
||||||
|
let (msg_tx, msg_rx) = bounded(1024);
|
||||||
|
msg_tx
|
||||||
|
.send(root_msg)
|
||||||
|
.map_err(|_| LoadingProblem::MsgChannelDied)?;
|
||||||
|
|
||||||
// Reserve one CPU for the main thread, and let all the others be eligible
|
// Reserve one CPU for the main thread, and let all the others be eligible
|
||||||
// to spawn workers. We use .max(2) to enforce that we always
|
// to spawn workers. We use .max(2) to enforce that we always
|
||||||
// end up with at least 1 worker - since (.max(2) - 1) will
|
// end up with at least 1 worker - since (.max(2) - 1) will
|
||||||
|
@ -848,28 +956,6 @@ where
|
||||||
worker_arenas.push(Bump::new());
|
worker_arenas.push(Bump::new());
|
||||||
}
|
}
|
||||||
|
|
||||||
let (msg_tx, msg_rx) = bounded(1024);
|
|
||||||
let arc_modules = Arc::new(Mutex::new(ModuleIds::default()));
|
|
||||||
let root_exposed_ident_ids = IdentIds::exposed_builtins(0);
|
|
||||||
let ident_ids_by_module = Arc::new(Mutex::new(root_exposed_ident_ids));
|
|
||||||
|
|
||||||
// Load the root module synchronously; we can't proceed until we have its id.
|
|
||||||
let (root_id, root_msg) = {
|
|
||||||
let root_start_time = SystemTime::now();
|
|
||||||
|
|
||||||
load_filename(
|
|
||||||
arena,
|
|
||||||
filename,
|
|
||||||
Arc::clone(&arc_modules),
|
|
||||||
Arc::clone(&ident_ids_by_module),
|
|
||||||
root_start_time,
|
|
||||||
)?
|
|
||||||
};
|
|
||||||
|
|
||||||
msg_tx
|
|
||||||
.send(root_msg)
|
|
||||||
.map_err(|_| LoadingProblem::MsgChannelDied)?;
|
|
||||||
|
|
||||||
// We'll add tasks to this, and then worker threads will take tasks from it.
|
// We'll add tasks to this, and then worker threads will take tasks from it.
|
||||||
let injector = Injector::new();
|
let injector = Injector::new();
|
||||||
|
|
||||||
|
@ -1598,6 +1684,30 @@ fn load_filename<'a>(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Load a module from a str
|
||||||
|
/// the `filename` is never read, but used for the module name
|
||||||
|
fn load_from_str<'a>(
|
||||||
|
arena: &'a Bump,
|
||||||
|
filename: PathBuf,
|
||||||
|
src: &'a str,
|
||||||
|
module_ids: Arc<Mutex<ModuleIds>>,
|
||||||
|
ident_ids_by_module: Arc<Mutex<IdentIdsByModule>>,
|
||||||
|
module_start_time: SystemTime,
|
||||||
|
) -> Result<(ModuleId, Msg<'a>), LoadingProblem> {
|
||||||
|
let file_io_start = SystemTime::now();
|
||||||
|
let file_io_duration = file_io_start.elapsed().unwrap();
|
||||||
|
|
||||||
|
parse_header(
|
||||||
|
arena,
|
||||||
|
file_io_duration,
|
||||||
|
filename,
|
||||||
|
module_ids,
|
||||||
|
ident_ids_by_module,
|
||||||
|
src.as_bytes(),
|
||||||
|
module_start_time,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn send_header<'a>(
|
fn send_header<'a>(
|
||||||
name: Located<roc_parse::header::ModuleName<'a>>,
|
name: Located<roc_parse::header::ModuleName<'a>>,
|
||||||
|
|
|
@ -14,7 +14,7 @@ use roc_types::subs::{Content, FlatType, Subs, Variable};
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use ven_pretty::{BoxAllocator, DocAllocator, DocBuilder};
|
use ven_pretty::{BoxAllocator, DocAllocator, DocBuilder};
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
pub enum MonoProblem {
|
pub enum MonoProblem {
|
||||||
PatternProblem(crate::exhaustive::Error),
|
PatternProblem(crate::exhaustive::Error),
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue