ligo/src/passes/2-simplify/pascaligo.ml

1196 lines
43 KiB
OCaml
Raw Normal View History

2019-05-13 00:56:22 +04:00
open Trace
open Ast_simplified
module Raw = Parser.Pascaligo.AST
module SMap = Map.String
open Combinators
let nseq_to_list (hd, tl) = hd :: tl
let npseq_to_list (hd, tl) = hd :: (List.map snd tl)
let pseq_to_list = function
| None -> []
| Some lst -> npseq_to_list lst
let get_value : 'a Raw.reg -> 'a = fun x -> x.value
2019-11-15 21:10:24 +04:00
let is_compiler_generated = fun name -> String.contains name '#'
2019-05-13 00:56:22 +04:00
module Errors = struct
let unsupported_cst_constr p =
let title () = "constant constructor" in
let message () =
Format.asprintf "constant constructors are not supported yet" in
let pattern_loc = Raw.pattern_to_region p in
let data = [
("pattern_loc",
fun () -> Format.asprintf "%a" Location.pp_lift @@ pattern_loc)
] in
error ~data title message
2019-09-07 20:42:59 +04:00
let bad_bytes loc str =
let title () = "bad bytes string" in
let message () =
Format.asprintf "bytes string contained non-hexadecimal chars" in
let data = [
("location", fun () -> Format.asprintf "%a" Location.pp loc) ;
("bytes", fun () -> str) ;
] in
error ~data title message
let corner_case ~loc message =
let title () = "corner case" in
let content () = "We don't have a good error message for this case. \
We are striving find ways to better report them and \
find the use-cases that generate them. \
Please report this to the developers." in
let data = [
("location" , fun () -> loc) ;
("message" , fun () -> message) ;
] in
error ~data title content
let unknown_predefined_type name =
let title () = "type constants" in
let message () =
Format.asprintf "unknown predefined type \"%s\"" name.Region.value in
let data = [
("typename_loc",
fun () -> Format.asprintf "%a" Location.pp_lift @@ name.Region.region)
] in
error ~data title message
let unsupported_non_var_pattern p =
let title () = "pattern is not a variable" in
let message () =
Format.asprintf "non-variable patterns in constructors \
are not supported yet" in
let pattern_loc = Raw.pattern_to_region p in
let data = [
("pattern_loc",
fun () -> Format.asprintf "%a" Location.pp_lift @@ pattern_loc)
] in
error ~data title message
let only_constructors p =
let title () = "constructors in patterns" in
let message () =
Format.asprintf "currently, only constructors are supported in patterns" in
let pattern_loc = Raw.pattern_to_region p in
let data = [
("pattern_loc",
fun () -> Format.asprintf "%a" Location.pp_lift @@ pattern_loc)
] in
error ~data title message
let unsupported_tuple_pattern p =
let title () = "tuple pattern" in
let message () =
Format.asprintf "tuple patterns are not supported yet" in
let pattern_loc = Raw.pattern_to_region p in
let data = [
("pattern_loc",
2019-09-21 13:30:41 +04:00
fun () -> Format.asprintf "%a" Location.pp_lift @@ pattern_loc) ;
("pattern",
fun () -> Parser.Pascaligo.ParserLog.pattern_to_string p)
] in
error ~data title message
let unsupported_deep_Some_patterns pattern =
let title () = "option patterns" in
let message () =
Format.asprintf "currently, only variables in Some constructors \
in patterns are supported" in
let pattern_loc = Raw.pattern_to_region pattern in
let data = [
("pattern_loc",
fun () -> Format.asprintf "%a" Location.pp_lift @@ pattern_loc)
] in
error ~data title message
let unsupported_deep_list_patterns cons =
let title () = "lists in patterns" in
let message () =
Format.asprintf "currently, only empty lists and x::y \
are supported in patterns" in
let data = [
("pattern_loc",
fun () -> Format.asprintf "%a" Location.pp_lift @@ cons.Region.region)
] in
error ~data title message
let unexpected_anonymous_function loc =
let title () = "unexpected anonymous function" in
let message () = "you provided a function declaration without name" in
let data = [
("loc" , fun () -> Format.asprintf "%a" Location.pp @@ loc)
] in
error ~data title message
let unexpected_named_function loc =
let title () = "unexpected named function" in
let message () = "you provided a function expression with a name (remove it)" in
let data = [
("loc" , fun () -> Format.asprintf "%a" Location.pp @@ loc)
] in
error ~data title message
(* Logging *)
let simplifying_instruction t =
let title () = "simplifiying instruction" in
let message () = "" in
let data = [
("instruction",
fun () -> Parser.Pascaligo.ParserLog.instruction_to_string t)
] in
error ~data title message
end
open Errors
2019-05-23 16:16:12 +04:00
open Operators.Simplify.Pascaligo
2019-05-13 00:56:22 +04:00
2019-05-28 19:36:14 +04:00
let r_split = Location.r_split
2019-09-10 14:42:49 +04:00
(*
Statements can't be simplified in isolation. `a ; b ; c` can get simplified either
as `let x = expr in (b ; c)` if `a` is a ` const x = expr` declaration or as
`sequence(a , sequence(b , c))` for everything else.
Because of this, simplifying sequences depend on their contents. To avoid peeking in
their contents, we instead simplify sequences elements as functions from their next
elements to the actual result.
For `return_let_in`, if there is no follow-up element, an error is triggered, as
you can't have `let x = expr in ...` with no `...`. A cleaner option might be to add
a `unit` instead of erroring.
`return_statement` is used for non-let_in statements.
*)
2019-05-28 19:36:14 +04:00
let return_let_in ?loc binder rhs = ok @@ fun expr'_opt ->
match expr'_opt with
| None -> fail @@ corner_case ~loc:__LOC__ "missing return"
2019-05-28 19:36:14 +04:00
| Some expr' -> ok @@ e_let_in ?loc binder rhs expr'
2019-09-10 14:42:49 +04:00
let return_statement expr = ok @@ fun expr'_opt ->
match expr'_opt with
| None -> ok @@ expr
| Some expr' -> ok @@ e_sequence expr expr'
2019-05-13 00:56:22 +04:00
let rec simpl_type_expression (t:Raw.type_expr) : type_expression result =
match t with
2019-11-06 20:23:49 +04:00
TPar x -> simpl_type_expression x.value.inside
| TVar v -> (
2019-05-13 00:56:22 +04:00
match List.assoc_opt v.value type_constants with
2019-05-23 16:16:12 +04:00
| Some s -> ok @@ T_constant (s , [])
| None -> ok @@ T_variable v.value
2019-05-13 00:56:22 +04:00
)
| TFun x -> (
let%bind (a , b) =
let (a , _ , b) = x.value in
bind_map_pair simpl_type_expression (a , b) in
ok @@ T_function (a , b)
)
| TApp x ->
let (name, tuple) = x.value in
let lst = npseq_to_list tuple.value.inside in
let%bind lst' = bind_list @@ List.map simpl_type_expression lst in
2019-05-23 16:16:12 +04:00
let%bind cst =
trace_option (unknown_predefined_type name) @@
2019-05-23 16:16:12 +04:00
List.assoc_opt name.value type_constants in
ok @@ T_constant (cst , lst')
2019-05-13 00:56:22 +04:00
| TProd p ->
let%bind tpl = simpl_list_type_expression
2019-09-10 14:42:49 +04:00
@@ npseq_to_list p.value in
2019-05-13 00:56:22 +04:00
ok tpl
| TRecord r ->
2019-09-10 14:42:49 +04:00
let aux = fun (x, y) ->
let%bind y = simpl_type_expression y in
ok (x, y)
in
let apply =
fun (x:Raw.field_decl Raw.reg) -> (x.value.field_name.value, x.value.field_type) in
2019-05-13 00:56:22 +04:00
let%bind lst = bind_list
@@ List.map aux
@@ List.map apply
@@ npseq_to_list r.value.ne_elements in
2019-05-13 00:56:22 +04:00
let m = List.fold_left (fun m (x, y) -> SMap.add x y m) SMap.empty lst in
ok @@ T_record m
| TSum s ->
let aux (v:Raw.variant Raw.reg) =
let args =
2019-11-06 20:23:49 +04:00
match v.value.arg with
None -> []
2019-11-06 20:23:49 +04:00
| Some (_, TProd product) -> npseq_to_list product.value
| Some (_, t_expr) -> [t_expr] in
let%bind te = simpl_list_type_expression @@ args in
2019-05-13 00:56:22 +04:00
ok (v.value.constr.value, te)
in
let%bind lst = bind_list
@@ List.map aux
@@ npseq_to_list s.value in
let m = List.fold_left (fun m (x, y) -> SMap.add x y m) SMap.empty lst in
ok @@ T_sum m
and simpl_list_type_expression (lst:Raw.type_expr list) : type_expression result =
match lst with
2019-06-07 00:49:36 +04:00
| [] -> ok @@ t_unit
2019-05-13 00:56:22 +04:00
| [hd] -> simpl_type_expression hd
| lst ->
let%bind lst = bind_list @@ List.map simpl_type_expression lst in
ok @@ T_tuple lst
2019-09-10 14:42:49 +04:00
let simpl_projection : Raw.projection Region.reg -> _ = fun p ->
let (p' , loc) = r_split p in
let var =
let name = p'.struct_name.value in
e_variable name in
let path = p'.field_path in
let path' =
let aux (s:Raw.selection) =
match s with
| FieldName property -> Access_record property.value
| Component index -> Access_tuple (Z.to_int (snd index.value))
in
List.map aux @@ npseq_to_list path in
ok @@ e_accessor ~loc var path'
let rec simpl_expression (t:Raw.expr) : expr result =
let return x = ok x in
2019-05-13 00:56:22 +04:00
match t with
| EAnnot a -> (
2019-05-28 19:36:14 +04:00
let ((expr , type_expr) , loc) = r_split a in
let%bind expr' = simpl_expression expr in
let%bind type_expr' = simpl_type_expression type_expr in
2019-05-28 19:36:14 +04:00
return @@ e_annotation ~loc expr' type_expr'
2019-05-13 00:56:22 +04:00
)
| EVar c -> (
2019-05-28 19:36:14 +04:00
let (c' , loc) = r_split c in
2019-05-13 00:56:22 +04:00
match List.assoc_opt c' constants with
2019-05-28 19:36:14 +04:00
| None -> return @@ e_variable ~loc c.value
| Some s -> return @@ e_constant ~loc s []
2019-05-13 00:56:22 +04:00
)
| ECall x -> (
let ((f, args) , loc) = r_split x in
2019-05-28 19:36:14 +04:00
let (args , args_loc) = r_split args in
let args' = npseq_to_list args.inside in
match f with
| EVar name -> (
let (f_name , f_loc) = r_split name in
match List.assoc_opt f_name constants with
| None ->
let%bind arg = simpl_tuple_expression ~loc:args_loc args' in
return @@ e_application ~loc (e_variable ~loc:f_loc f_name) arg
| Some s ->
let%bind lst = bind_map_list simpl_expression args' in
return @@ e_constant ~loc s lst
)
| f -> (
let%bind f' = simpl_expression f in
let%bind arg = simpl_tuple_expression ~loc:args_loc args' in
return @@ e_application ~loc f' arg
)
2019-05-13 00:56:22 +04:00
)
| EPar x -> simpl_expression x.value.inside
2019-05-28 19:36:14 +04:00
| EUnit reg ->
let loc = Location.lift reg in
return @@ e_literal ~loc Literal_unit
| EBytes x ->
let (x' , loc) = r_split x in
return @@ e_literal ~loc (Literal_bytes (Bytes.of_string @@ fst x'))
2019-05-13 00:56:22 +04:00
| ETuple tpl ->
let (tpl' , loc) = r_split tpl in
2019-05-28 19:36:14 +04:00
simpl_tuple_expression ~loc @@ npseq_to_list tpl'.inside
2019-05-13 00:56:22 +04:00
| ERecord r ->
let%bind fields = bind_list
@@ List.map (fun ((k : _ Raw.reg), v) -> let%bind v = simpl_expression v in ok (k.value, v))
@@ List.map (fun (x:Raw.field_assign Raw.reg) -> (x.value.field_name, x.value.field_expr))
2019-11-06 20:23:49 +04:00
@@ npseq_to_list r.value.ne_elements in
2019-05-13 00:56:22 +04:00
let aux prev (k, v) = SMap.add k v prev in
2019-05-28 19:36:14 +04:00
return @@ e_record (List.fold_left aux SMap.empty fields)
| EProj p -> simpl_projection p
| EConstr (ConstrApp c) -> (
let ((c, args) , loc) = r_split c in
match args with
None -> simpl_tuple_expression []
| Some args ->
let args, args_loc = r_split args in
let%bind arg =
simpl_tuple_expression ~loc:args_loc
@@ npseq_to_list args.inside in
return @@ e_constructor ~loc c.value arg
2019-05-28 19:36:14 +04:00
)
2019-05-13 00:56:22 +04:00
| EConstr (SomeApp a) ->
2019-05-28 19:36:14 +04:00
let ((_, args) , loc) = r_split a in
let (args , args_loc) = r_split args in
2019-05-13 00:56:22 +04:00
let%bind arg =
2019-05-28 19:36:14 +04:00
simpl_tuple_expression ~loc:args_loc
@@ npseq_to_list args.inside in
return @@ e_constant ~loc "SOME" [arg]
| EConstr (NoneExpr reg) -> (
let loc = Location.lift reg in
return @@ e_none ~loc ()
)
2019-05-13 00:56:22 +04:00
| EArith (Add c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "ADD" c
2019-05-13 00:56:22 +04:00
| EArith (Sub c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "SUB" c
2019-05-13 00:56:22 +04:00
| EArith (Mult c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "TIMES" c
2019-05-13 00:56:22 +04:00
| EArith (Div c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "DIV" c
2019-05-13 00:56:22 +04:00
| EArith (Mod c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "MOD" c
| EArith (Int n) -> (
let (n , loc) = r_split n in
let n = Z.to_int @@ snd n in
return @@ e_literal ~loc (Literal_int n)
)
| EArith (Nat n) -> (
let (n , loc) = r_split n in
let n = Z.to_int @@ snd @@ n in
return @@ e_literal ~loc (Literal_nat n)
)
2019-10-27 20:50:24 +04:00
| EArith (Mutez n) -> (
2019-07-19 14:13:09 +04:00
let (n , loc) = r_split n in
let n = Z.to_int @@ snd @@ n in
2019-09-24 16:29:18 +04:00
return @@ e_literal ~loc (Literal_mutez n)
2019-07-19 14:13:09 +04:00
)
| EArith (Neg e) -> simpl_unop "NEG" e
2019-05-13 00:56:22 +04:00
| EString (String s) ->
2019-05-28 19:36:14 +04:00
let (s , loc) = r_split s in
2019-05-13 00:56:22 +04:00
let s' =
2019-05-28 19:36:14 +04:00
(* S contains quotes *)
String.(sub s 1 (length s - 2))
2019-05-13 00:56:22 +04:00
in
2019-05-28 19:36:14 +04:00
return @@ e_literal ~loc (Literal_string s')
2019-10-07 19:16:03 +04:00
| EString (Cat bo) ->
let (bo , loc) = r_split bo in
let%bind sl = simpl_expression bo.arg1 in
let%bind sr = simpl_expression bo.arg2 in
return @@ e_string_cat ~loc sl sr
| ELogic l -> simpl_logic_expression l
| EList l -> simpl_list_expression l
2019-07-19 16:35:47 +04:00
| ESet s -> simpl_set_expression s
2019-10-18 16:47:04 +04:00
| ECond c ->
let (c , loc) = r_split c in
let%bind expr = simpl_expression c.test in
let%bind match_true = simpl_expression c.ifso in
let%bind match_false = simpl_expression c.ifnot in
2019-10-18 16:47:04 +04:00
return @@ e_matching expr ~loc (Match_bool {match_true; match_false})
2019-05-28 19:36:14 +04:00
| ECase c -> (
let (c , loc) = r_split c in
let%bind e = simpl_expression c.expr in
2019-05-13 00:56:22 +04:00
let%bind lst =
let aux (x : Raw.expr Raw.case_clause) =
let%bind expr = simpl_expression x.rhs in
ok (x.pattern, expr) in
bind_list
@@ List.map aux
@@ List.map get_value
2019-05-28 19:36:14 +04:00
@@ npseq_to_list c.cases.value in
2019-05-13 00:56:22 +04:00
let%bind cases = simpl_cases lst in
2019-05-28 19:36:14 +04:00
return @@ e_matching ~loc e cases
)
2019-10-21 15:04:28 +04:00
| EMap (MapInj mi) -> (
2019-05-28 19:36:14 +04:00
let (mi , loc) = r_split mi in
2019-05-13 00:56:22 +04:00
let%bind lst =
2019-05-28 19:36:14 +04:00
let lst = List.map get_value @@ pseq_to_list mi.elements in
let aux : Raw.binding -> (expression * expression) result =
fun b ->
let%bind src = simpl_expression b.source in
let%bind dst = simpl_expression b.image in
ok (src, dst) in
2019-05-13 00:56:22 +04:00
bind_map_list aux lst in
2019-05-28 19:36:14 +04:00
return @@ e_map ~loc lst
)
2019-10-21 15:04:28 +04:00
| EMap (BigMapInj mi) -> (
let (mi , loc) = r_split mi in
let%bind lst =
let lst = List.map get_value @@ pseq_to_list mi.elements in
let aux : Raw.binding -> (expression * expression) result =
fun b ->
let%bind src = simpl_expression b.source in
let%bind dst = simpl_expression b.image in
ok (src, dst) in
bind_map_list aux lst in
return @@ e_big_map ~loc lst
)
2019-05-28 19:36:14 +04:00
| EMap (MapLookUp lu) -> (
let (lu , loc) = r_split lu in
let%bind path = match lu.path with
| Name v -> (
let (v , loc) = r_split v in
return @@ e_variable ~loc v
)
| Path p -> simpl_projection p
2019-05-13 00:56:22 +04:00
in
2019-05-28 19:36:14 +04:00
let%bind index = simpl_expression lu.index.value.inside in
return @@ e_look_up ~loc path index
)
| EFun f -> (
let (f , loc) = r_split f in
let%bind ((name_opt , _ty_opt) , f') = simpl_fun_expression ~loc f in
match name_opt with
| None -> return @@ f'
| Some _ -> fail @@ unexpected_named_function loc
)
2019-05-13 00:56:22 +04:00
and simpl_logic_expression (t:Raw.logic_expr) : expression result =
let return x = ok x in
2019-05-13 00:56:22 +04:00
match t with
2019-05-28 19:36:14 +04:00
| BoolExpr (False reg) -> (
let loc = Location.lift reg in
return @@ e_literal ~loc (Literal_bool false)
)
| BoolExpr (True reg) -> (
let loc = Location.lift reg in
return @@ e_literal ~loc (Literal_bool true)
)
2019-05-13 00:56:22 +04:00
| BoolExpr (Or b) ->
2019-05-28 19:36:14 +04:00
simpl_binop "OR" b
2019-05-13 00:56:22 +04:00
| BoolExpr (And b) ->
2019-05-28 19:36:14 +04:00
simpl_binop "AND" b
2019-05-13 00:56:22 +04:00
| BoolExpr (Not b) ->
2019-05-28 19:36:14 +04:00
simpl_unop "NOT" b
2019-05-13 00:56:22 +04:00
| CompExpr (Lt c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "LT" c
2019-05-13 00:56:22 +04:00
| CompExpr (Gt c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "GT" c
2019-05-13 00:56:22 +04:00
| CompExpr (Leq c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "LE" c
2019-05-13 00:56:22 +04:00
| CompExpr (Geq c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "GE" c
2019-05-13 00:56:22 +04:00
| CompExpr (Equal c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "EQ" c
2019-05-13 00:56:22 +04:00
| CompExpr (Neq c) ->
2019-05-28 19:36:14 +04:00
simpl_binop "NEQ" c
2019-05-13 00:56:22 +04:00
and simpl_list_expression (t:Raw.list_expr) : expression result =
let return x = ok x in
2019-05-13 00:56:22 +04:00
match t with
2019-11-06 20:23:49 +04:00
ECons c ->
2019-05-28 19:36:14 +04:00
simpl_binop "CONS" c
2019-11-06 20:23:49 +04:00
| EListComp lst ->
2019-05-28 19:36:14 +04:00
let (lst , loc) = r_split lst in
2019-05-13 00:56:22 +04:00
let%bind lst' =
bind_map_list simpl_expression @@
2019-05-28 19:36:14 +04:00
pseq_to_list lst.elements in
return @@ e_list ~loc lst'
2019-11-06 20:23:49 +04:00
| ENil reg ->
2019-05-28 19:36:14 +04:00
let loc = Location.lift reg in
return @@ e_list ~loc []
2019-05-13 00:56:22 +04:00
2019-07-19 16:35:47 +04:00
and simpl_set_expression (t:Raw.set_expr) : expression result =
match t with
| SetMem x -> (
let (x' , loc) = r_split x in
let%bind set' = simpl_expression x'.set in
let%bind element' = simpl_expression x'.element in
ok @@ e_constant ~loc "SET_MEM" [ element' ; set' ]
)
| SetInj x -> (
let (x' , loc) = r_split x in
let elements = pseq_to_list x'.elements in
let%bind elements' = bind_map_list simpl_expression elements in
ok @@ e_set ~loc elements'
)
2019-05-28 19:36:14 +04:00
and simpl_binop (name:string) (t:_ Raw.bin_op Region.reg) : expression result =
let return x = ok x in
2019-05-28 19:36:14 +04:00
let (t , loc) = r_split t in
2019-05-13 00:56:22 +04:00
let%bind a = simpl_expression t.arg1 in
let%bind b = simpl_expression t.arg2 in
2019-05-28 19:36:14 +04:00
return @@ e_constant ~loc name [ a ; b ]
2019-05-13 00:56:22 +04:00
2019-05-28 19:36:14 +04:00
and simpl_unop (name:string) (t:_ Raw.un_op Region.reg) : expression result =
let return x = ok x in
2019-05-28 19:36:14 +04:00
let (t , loc) = r_split t in
2019-05-13 00:56:22 +04:00
let%bind a = simpl_expression t.arg in
2019-05-28 19:36:14 +04:00
return @@ e_constant ~loc name [ a ]
2019-05-13 00:56:22 +04:00
2019-05-28 19:36:14 +04:00
and simpl_tuple_expression ?loc (lst:Raw.expr list) : expression result =
let return x = ok x in
2019-05-13 00:56:22 +04:00
match lst with
2019-05-28 19:36:14 +04:00
| [] -> return @@ e_literal Literal_unit
| [hd] -> simpl_expression hd
| lst ->
2019-05-13 00:56:22 +04:00
let%bind lst = bind_list @@ List.map simpl_expression lst in
2019-05-28 19:36:14 +04:00
return @@ e_tuple ?loc lst
2019-05-13 00:56:22 +04:00
and simpl_local_declaration : Raw.local_decl -> _ result = fun t ->
2019-05-13 00:56:22 +04:00
match t with
| LocalData d ->
simpl_data_declaration d
and simpl_data_declaration : Raw.data_decl -> _ result = fun t ->
2019-05-13 00:56:22 +04:00
match t with
| LocalVar x ->
2019-05-28 19:36:14 +04:00
let (x , loc) = r_split x in
2019-05-13 00:56:22 +04:00
let name = x.name.value in
let%bind t = simpl_type_expression x.var_type in
let%bind expression = simpl_expression x.init in
2019-05-28 19:36:14 +04:00
return_let_in ~loc (name , Some t) expression
2019-05-13 00:56:22 +04:00
| LocalConst x ->
2019-05-28 19:36:14 +04:00
let (x , loc) = r_split x in
2019-05-13 00:56:22 +04:00
let name = x.name.value in
let%bind t = simpl_type_expression x.const_type in
let%bind expression = simpl_expression x.init in
2019-05-28 19:36:14 +04:00
return_let_in ~loc (name , Some t) expression
| LocalFun f ->
let (f , loc) = r_split f in
let%bind ((name_opt , ty_opt) , e) = simpl_fun_expression ~loc f.fun_expr.value in
let%bind name = trace_option (unexpected_anonymous_function loc) name_opt in
return_let_in ~loc (name , ty_opt) e
2019-05-13 00:56:22 +04:00
and simpl_param : Raw.param_decl -> (type_name * type_expression) result =
fun t ->
2019-05-13 00:56:22 +04:00
match t with
| ParamConst c ->
let c = c.value in
let type_name = c.var.value in
let%bind type_expression = simpl_type_expression c.param_type in
ok (type_name , type_expression)
2019-05-13 00:56:22 +04:00
| ParamVar v ->
let c = v.value in
let type_name = c.var.value in
let%bind type_expression = simpl_type_expression c.param_type in
ok (type_name , type_expression)
2019-05-13 00:56:22 +04:00
and simpl_fun_expression :
loc:_ -> Raw.fun_expr -> ((name option * type_expression option) * expression) result =
fun ~loc x ->
2019-05-13 00:56:22 +04:00
let open! Raw in
let {name;param;ret_type;local_decls;block;return} : fun_expr = x in
let statements =
match block with
| Some block -> npseq_to_list block.value.statements
| None -> []
2019-10-19 20:11:18 +04:00
in
(match param.value.inside with
a, [] -> (
2019-05-13 00:56:22 +04:00
let%bind input = simpl_param a in
let name = Option.map (fun (x : _ reg) -> x.value) name in
let (binder , input_type) = input in
2019-05-13 00:56:22 +04:00
let%bind local_declarations =
bind_map_list simpl_local_declaration local_decls in
2019-05-13 00:56:22 +04:00
let%bind instructions = bind_list
@@ List.map simpl_statement
@@ statements in
2019-05-13 00:56:22 +04:00
let%bind result = simpl_expression return in
let%bind output_type = simpl_type_expression ret_type in
let body = local_declarations @ instructions in
let%bind result =
let aux prec cur = cur (Some prec) in
bind_fold_right_list aux result body in
2019-05-28 19:36:14 +04:00
let expression : expression = e_lambda ~loc binder (Some input_type)
(Some output_type) result in
2019-05-13 00:56:22 +04:00
let type_annotation = Some (T_function (input_type, output_type)) in
ok ((name , type_annotation) , expression)
2019-05-13 00:56:22 +04:00
)
| lst -> (
let lst = npseq_to_list lst in
2019-05-13 00:56:22 +04:00
let arguments_name = "arguments" in
let%bind params = bind_map_list simpl_param lst in
let (binder , input_type) =
let type_expression = T_tuple (List.map snd params) in
(arguments_name , type_expression) in
let%bind tpl_declarations =
let aux = fun i x ->
let expr = e_accessor (e_variable arguments_name) [Access_tuple i] in
let type_ = Some (snd x) in
let ass = return_let_in (fst x , type_) expr in
2019-05-13 00:56:22 +04:00
ass
in
bind_list @@ List.mapi aux params in
2019-05-13 00:56:22 +04:00
let%bind local_declarations =
bind_map_list simpl_local_declaration local_decls in
2019-05-13 00:56:22 +04:00
let%bind instructions = bind_list
@@ List.map simpl_statement
@@ statements in
2019-05-13 00:56:22 +04:00
let%bind result = simpl_expression return in
let%bind output_type = simpl_type_expression ret_type in
let body = tpl_declarations @ local_declarations @ instructions in
let%bind result =
let aux prec cur = cur (Some prec) in
bind_fold_right_list aux result body in
let expression =
e_lambda ~loc binder (Some input_type) (Some output_type) result in
2019-05-13 00:56:22 +04:00
let type_annotation = Some (T_function (input_type, output_type)) in
let name = Option.map (fun (x : _ reg) -> x.value) name in
ok ((name , type_annotation) , expression)
2019-05-13 00:56:22 +04:00
)
)
and simpl_declaration : Raw.declaration -> declaration Location.wrap result =
fun t ->
2019-05-13 00:56:22 +04:00
let open! Raw in
match t with
2019-05-28 19:36:14 +04:00
| TypeDecl x -> (
let (x , loc) = r_split x in
let {name;type_expr} : Raw.type_decl = x in
2019-05-13 00:56:22 +04:00
let%bind type_expression = simpl_type_expression type_expr in
2019-05-28 19:36:14 +04:00
ok @@ Location.wrap ~loc (Declaration_type (name.value , type_expression))
)
2019-05-13 00:56:22 +04:00
| ConstDecl x ->
let simpl_const_decl = fun {name;const_type;init} ->
let%bind expression = simpl_expression init in
let%bind t = simpl_type_expression const_type in
let type_annotation = Some t in
ok @@ Declaration_constant (name.value , type_annotation , expression)
2019-05-13 00:56:22 +04:00
in
bind_map_location simpl_const_decl (Location.lift_region x)
| FunDecl x -> (
2019-05-28 19:36:14 +04:00
let (x , loc) = r_split x in
let%bind ((name_opt , ty_opt) , expr) = simpl_fun_expression ~loc x.fun_expr.value in
let%bind name = trace_option (unexpected_anonymous_function loc) name_opt in
2019-05-28 19:36:14 +04:00
ok @@ Location.wrap ~loc (Declaration_constant (name , ty_opt , expr))
)
2019-05-13 00:56:22 +04:00
and simpl_statement : Raw.statement -> (_ -> expression result) result =
fun s ->
2019-05-13 00:56:22 +04:00
match s with
| Instr i -> simpl_instruction i
| Data d -> simpl_data_declaration d
2019-05-13 00:56:22 +04:00
2019-10-17 20:33:58 +04:00
and simpl_single_instruction : Raw.instruction -> (_ -> expression result) result =
fun t ->
2019-05-13 00:56:22 +04:00
match t with
2019-07-20 15:46:42 +04:00
| ProcCall x -> (
let ((f, args) , loc) = r_split x in
let (args , args_loc) = r_split args in
let args' = npseq_to_list args.inside in
match f with
| EVar name -> (
let (f_name , f_loc) = r_split name in
match List.assoc_opt f_name constants with
2019-07-20 15:46:42 +04:00
| None ->
let%bind arg = simpl_tuple_expression ~loc:args_loc args' in
return_statement @@ e_application ~loc (e_variable ~loc:f_loc f_name) arg
2019-07-20 15:46:42 +04:00
| Some s ->
let%bind lst = bind_map_list simpl_expression args' in
return_statement @@ e_constant ~loc s lst
)
| f -> (
let%bind f' = simpl_expression f in
let%bind arg = simpl_tuple_expression ~loc:args_loc args' in
return_statement @@ e_application ~loc f' arg
2019-07-20 15:46:42 +04:00
)
)
2019-05-28 19:36:14 +04:00
| Skip reg -> (
let loc = Location.lift reg in
2019-09-10 14:42:49 +04:00
return_statement @@ e_skip ~loc ()
2019-05-28 19:36:14 +04:00
)
2019-05-13 00:56:22 +04:00
| Loop (While l) ->
let l = l.value in
let%bind cond = simpl_expression l.cond in
let%bind body = simpl_block l.block.value in
let%bind body = body None in
2019-09-10 14:42:49 +04:00
return_statement @@ e_loop cond body
2019-11-06 20:23:49 +04:00
| Loop (For (ForInt fi)) ->
2019-10-11 20:31:04 +04:00
let%bind loop = simpl_for_int fi.value in
2019-11-06 20:23:49 +04:00
let%bind loop = loop None in
2019-10-11 20:31:04 +04:00
return_statement @@ loop
2019-10-15 15:14:00 +04:00
| Loop (For (ForCollect fc)) ->
let%bind loop = simpl_for_collect fc.value in
2019-11-06 20:23:49 +04:00
let%bind loop = loop None in
2019-10-15 15:14:00 +04:00
return_statement @@ loop
2019-05-28 19:36:14 +04:00
| Cond c -> (
let (c , loc) = r_split c in
2019-05-13 00:56:22 +04:00
let%bind expr = simpl_expression c.test in
let%bind match_true = match c.ifso with
2019-10-17 20:33:58 +04:00
ClauseInstr i ->
simpl_single_instruction i
| ClauseBlock b ->
match b with
LongBlock {value; _} ->
simpl_block value
| ShortBlock {value; _} ->
simpl_statements @@ fst value.inside in
2019-05-13 00:56:22 +04:00
let%bind match_false = match c.ifnot with
2019-10-17 20:33:58 +04:00
ClauseInstr i ->
simpl_single_instruction i
| ClauseBlock b ->
match b with
LongBlock {value; _} ->
simpl_block value
| ShortBlock {value; _} ->
simpl_statements @@ fst value.inside in
let%bind match_true = match_true None in
let%bind match_false = match_false None in
2019-09-10 14:42:49 +04:00
return_statement @@ e_matching expr ~loc (Match_bool {match_true; match_false})
2019-05-28 19:36:14 +04:00
)
2019-05-13 00:56:22 +04:00
| Assign a -> (
2019-05-28 19:36:14 +04:00
let (a , loc) = r_split a in
2019-10-07 18:24:56 +04:00
let%bind value_expr = simpl_expression a.rhs in
2019-05-13 00:56:22 +04:00
match a.lhs with
| Path path -> (
let (name , path') = simpl_path path in
2019-09-10 14:42:49 +04:00
return_statement @@ e_assign ~loc name path' value_expr
2019-05-13 00:56:22 +04:00
)
| MapPath v -> (
let v' = v.value in
let%bind (varname,map,path) = match v'.path with
| Name name -> ok (name.value , e_variable name.value, [])
| Path p ->
let (name,p') = simpl_path v'.path in
let%bind accessor = simpl_projection p in
ok @@ (name , accessor , p')
in
2019-05-13 00:56:22 +04:00
let%bind key_expr = simpl_expression v'.index.value.inside in
let expr' = e_map_add key_expr value_expr map in
return_statement @@ e_assign ~loc varname path expr'
2019-05-13 00:56:22 +04:00
)
)
| CaseInstr c -> (
2019-05-28 19:36:14 +04:00
let (c , loc) = r_split c in
2019-05-13 00:56:22 +04:00
let%bind expr = simpl_expression c.expr in
let%bind cases =
let aux (x : Raw.if_clause Raw.case_clause Raw.reg) =
let%bind case_clause =
match x.value.rhs with
ClauseInstr i ->
simpl_single_instruction i
| ClauseBlock b ->
match b with
LongBlock {value; _} ->
simpl_block value
| ShortBlock {value; _} ->
simpl_statements @@ fst value.inside in
let%bind case_clause = case_clause None in
ok (x.value.pattern, case_clause) in
2019-05-13 00:56:22 +04:00
bind_list
@@ List.map aux
@@ npseq_to_list c.cases.value in
let%bind m = simpl_cases cases in
return_statement @@ e_matching ~loc expr m
)
2019-05-13 00:56:22 +04:00
| RecordPatch r -> (
let r = r.value in
let (name , access_path) = simpl_path r.path in
let head, tail = r.record_inj.value.ne_elements in
let%bind tail' = bind_list
@@ List.map (fun (x: Raw.field_assign Region.reg) ->
2019-05-28 19:36:14 +04:00
let (x , loc) = r_split x in
let%bind e = simpl_expression x.field_expr
in ok (x.field_name.value, e , loc)
2019-05-28 19:36:14 +04:00
)
@@ List.map snd tail in
let%bind head' =
let (x , loc) = r_split head in
let%bind e = simpl_expression x.field_expr
in ok (x.field_name.value, e , loc) in
let%bind expr =
2019-05-28 19:36:14 +04:00
let aux = fun (access , v , loc) ->
e_assign ~loc name (access_path @ [Access_record access]) v in
let hd, tl = aux head', List.map aux tail' in
let aux acc cur = e_sequence acc cur in
ok @@ List.fold_left aux hd tl
in
2019-09-10 14:42:49 +04:00
return_statement @@ expr
)
| MapPatch patch -> (
let (map_p, loc) = r_split patch in
let (name, access_path) = simpl_path map_p.path in
let%bind inj = bind_list
@@ List.map (fun (x:Raw.binding Region.reg) ->
let x = x.value in
let (key, value) = x.source, x.image in
let%bind key' = simpl_expression key in
let%bind value' = simpl_expression value
in ok @@ (key', value')
)
@@ npseq_to_list map_p.map_inj.value.ne_elements in
let expr =
match inj with
| [] -> e_skip ~loc ()
| _ :: _ ->
2019-10-14 23:05:35 +04:00
let assigns = List.fold_right
(fun (key, value) map -> (e_map_add key value map))
inj
2019-10-14 23:05:35 +04:00
(e_accessor ~loc (e_variable name) access_path)
in e_assign ~loc name access_path assigns
in return_statement @@ expr
)
| SetPatch patch -> (
let (setp, loc) = r_split patch in
let (name , access_path) = simpl_path setp.path in
2019-10-12 00:10:08 +04:00
let%bind inj =
bind_list @@
List.map simpl_expression @@
npseq_to_list setp.set_inj.value.ne_elements in
2019-10-12 00:10:08 +04:00
let expr =
match inj with
| [] -> e_skip ~loc ()
| _ :: _ ->
2019-10-14 23:05:35 +04:00
let assigns = List.fold_right
(fun hd s -> e_constant "SET_ADD" [hd ; s])
inj (e_accessor ~loc (e_variable name) access_path) in
2019-10-12 00:10:08 +04:00
e_assign ~loc name access_path assigns in
return_statement @@ expr
)
2019-05-28 19:36:14 +04:00
| MapRemove r -> (
let (v , loc) = r_split r in
2019-05-13 00:56:22 +04:00
let key = v.key in
let%bind (varname,map,path) = match v.map with
| Name v -> ok (v.value , e_variable v.value , [])
| Path p ->
let (name,p') = simpl_path v.map in
let%bind accessor = simpl_projection p in
ok @@ (name , accessor , p')
in
2019-05-13 00:56:22 +04:00
let%bind key' = simpl_expression key in
let expr = e_constant ~loc "MAP_REMOVE" [key' ; map] in
return_statement @@ e_assign ~loc varname path expr
2019-05-28 19:36:14 +04:00
)
| SetRemove r -> (
let (set_rm, loc) = r_split r in
let%bind (varname, set, path) = match set_rm.set with
| Name v -> ok (v.value, e_variable v.value, [])
| Path path ->
let(name, p') = simpl_path set_rm.set in
let%bind accessor = simpl_projection path in
ok @@ (name, accessor, p')
in
let%bind removed' = simpl_expression set_rm.element in
let expr = e_constant ~loc "SET_REMOVE" [removed' ; set] in
return_statement @@ e_assign ~loc varname path expr
)
2019-05-13 00:56:22 +04:00
and simpl_path : Raw.path -> string * Ast_simplified.access_path = fun p ->
match p with
| Raw.Name v -> (v.value , [])
| Raw.Path p -> (
let p' = p.value in
let var = p'.struct_name.value in
let path = p'.field_path in
let path' =
let aux (s:Raw.selection) =
match s with
| FieldName property -> Access_record property.value
| Component index -> Access_tuple (Z.to_int (snd index.value))
in
List.map aux @@ npseq_to_list path in
(var , path')
)
and simpl_cases : type a . (Raw.pattern * a) list -> a matching result = fun t ->
let open Raw in
let get_var (t:Raw.pattern) =
match t with
2019-05-13 00:56:22 +04:00
| PVar v -> ok v.value
| p -> fail @@ unsupported_non_var_pattern p in
let get_tuple (t: Raw.pattern) =
match t with
2019-05-13 00:56:22 +04:00
| PTuple v -> npseq_to_list v.value.inside
| x -> [ x ] in
let get_single (t: Raw.pattern) =
2019-05-13 00:56:22 +04:00
let t' = get_tuple t in
let%bind () =
trace_strong (unsupported_tuple_pattern t) @@
2019-05-13 00:56:22 +04:00
Assert.assert_list_size t' 1 in
ok (List.hd t') in
2019-09-21 13:30:41 +04:00
let get_toplevel (t : Raw.pattern) =
match t with
2019-11-06 20:23:49 +04:00
| PList PCons x -> (
2019-09-21 13:30:41 +04:00
let (x' , lst) = x.value in
match lst with
| [] -> ok x'
| _ -> ok t
)
| pattern -> ok pattern in
let get_constr (t: Raw.pattern) =
match t with
2019-11-06 20:23:49 +04:00
| PConstr (PConstrApp v) -> (
let const, pat_opt = v.value in
let%bind pat =
trace_option (unsupported_cst_constr t) @@
pat_opt in
let%bind single_pat = get_single (PTuple pat) in
let%bind var = get_var single_pat in
ok (const.value , var)
)
| _ -> fail @@ only_constructors t in
2019-05-13 00:56:22 +04:00
let%bind patterns =
let aux (x , y) =
2019-09-21 13:30:41 +04:00
let%bind x' = get_toplevel x in
ok (x' , y)
in bind_map_list aux t in
2019-05-13 00:56:22 +04:00
match patterns with
2019-11-06 20:23:49 +04:00
| [(PConstr PFalse _ , f) ; (PConstr PTrue _ , t)]
| [(PConstr PTrue _ , t) ; (PConstr PFalse _ , f)] ->
ok @@ Match_bool {match_true = t ; match_false = f}
2019-11-06 20:23:49 +04:00
| [(PConstr PSomeApp v , some) ; (PConstr PNone _ , none)]
| [(PConstr PNone _ , none) ; (PConstr PSomeApp v , some)] -> (
2019-05-13 00:56:22 +04:00
let (_, v) = v.value in
let%bind v = match v.value.inside with
| PVar v -> ok v.value
| p -> fail @@ unsupported_deep_Some_patterns p in
2019-05-13 00:56:22 +04:00
ok @@ Match_option {match_none = none ; match_some = (v, some) }
)
2019-11-06 20:23:49 +04:00
| [(PList PCons c, cons) ; (PList (PNil _), nil)]
| [(PList (PNil _), nil) ; (PList PCons c, cons)] ->
2019-05-13 00:56:22 +04:00
let%bind (a, b) =
match c.value with
| a, [(_, b)] ->
let%bind a = get_var a in
let%bind b = get_var b in
ok (a, b)
| _ -> fail @@ unsupported_deep_list_patterns c
2019-05-13 00:56:22 +04:00
in
ok @@ Match_list {match_cons = (a, b, cons) ; match_nil = nil}
| lst ->
trace (simple_info "currently, only booleans, options, lists and \
user-defined constructors are supported in patterns") @@
2019-05-13 00:56:22 +04:00
let%bind constrs =
let aux (x , y) =
let error =
let title () = "Pattern" in
let content () =
Printf.sprintf "Pattern : %s"
(Parser.Pascaligo.ParserLog.pattern_to_string x) in
2019-05-13 00:56:22 +04:00
error title content in
let%bind x' =
trace error @@
get_constr x in
ok (x' , y) in
bind_map_list aux lst in
ok @@ Match_variant constrs
and simpl_instruction : Raw.instruction -> (_ -> expression result) result =
fun t ->
2019-10-17 20:33:58 +04:00
trace (simplifying_instruction t) @@ simpl_single_instruction t
2019-05-13 00:56:22 +04:00
and simpl_statements : Raw.statements -> (_ -> expression result) result =
fun ss ->
2019-05-13 00:56:22 +04:00
let lst = npseq_to_list ss in
let%bind fs = bind_map_list simpl_statement lst in
let aux : _ -> (expression option -> expression result) -> _ =
fun prec cur ->
let%bind res = cur prec in
ok @@ Some res in
ok @@ fun (expr' : _ option) ->
let%bind ret = bind_fold_right_list aux expr' fs in
ok @@ Option.unopt_exn ret
2019-05-13 00:56:22 +04:00
and simpl_block : Raw.block -> (_ -> expression result) result = fun t ->
2019-05-13 00:56:22 +04:00
simpl_statements t.statements
2019-10-11 20:31:04 +04:00
and simpl_for_int : Raw.for_int -> (_ -> expression result) result = fun fi ->
(* cond part *)
2019-10-15 15:56:21 +04:00
let var = e_variable fi.assign.value.name.value in
2019-10-11 20:31:04 +04:00
let%bind value = simpl_expression fi.assign.value.expr in
let%bind bound = simpl_expression fi.bound in
2019-10-22 14:12:19 +04:00
let comp = e_annotation (e_constant "LE" [var ; bound]) t_bool
2019-10-11 20:31:04 +04:00
in
(* body part *)
let%bind body = simpl_block fi.block.value in
let%bind body = body None in
2019-10-22 14:12:19 +04:00
let step = e_int 1 in
let ctrl = e_assign
fi.assign.value.name.value [] (e_constant "ADD" [ var ; step ]) in
2019-10-11 20:31:04 +04:00
let rec add_to_seq expr = match expr.expression with
| E_sequence (_,a) -> add_to_seq a
| _ -> e_sequence body ctrl in
2019-10-15 15:56:21 +04:00
let body' = add_to_seq body in
let loop = e_loop comp body' in
2019-10-11 20:31:04 +04:00
return_statement @@ e_let_in (fi.assign.value.name.value, Some t_int) value loop
2019-10-28 21:40:53 +04:00
(** simpl_for_collect
For loops over collections, like
``` concrete syntax :
for x : int in set myset
begin
myint := myint + x ;
myst := myst ^ "to" ;
end
```
are implemented using a MAP_FOLD, LIST_FOLD or SET_FOLD:
``` pseudo Ast_simplified
2019-11-06 20:23:49 +04:00
let #COMPILER#folded_record = list_fold( mylist ,
2019-10-28 21:40:53 +04:00
record st = st; acc = acc; end;
2019-11-06 20:23:49 +04:00
lamby = fun arguments -> (
2019-10-28 21:40:53 +04:00
let #COMPILER#acc = arguments.0 in
let #COMPILER#elt_x = arguments.1 in
#COMPILER#acc.myint := #COMPILER#acc.myint + #COMPILER#elt_x ;
2019-10-28 21:40:53 +04:00
#COMPILER#acc.myst := #COMPILER#acc.myst ^ "to" ;
#COMPILER#acc
)
) in
{
myst := #COMPILER#folded_record.myst ;
myint := #COMPILER#folded_record.myint ;
}
```
2019-11-06 20:23:49 +04:00
2019-10-28 21:40:53 +04:00
We are performing the following steps:
2019-10-29 14:41:59 +04:00
1) Simplifying the for body using ̀simpl_block`
2019-10-28 21:40:53 +04:00
2) Detect the free variables and build a list of their names
(myint and myst in the previous example)
2019-11-15 21:10:24 +04:00
Free variables are simply variables being assigned.
Note: In the case of a nested loops, assignements to a compiler
generated value (#COMPILER#acc) correspond to variables
that were already renamed in the inner loop.
e.g :
```
#COMPILER#acc.myint := #COMPILER#acc.myint + #COMPILER#elt_x ;
2019-11-15 21:10:24 +04:00
#COMPILER#acc.myst := #COMPILER#acc.myst ^ "to" ;
```
They must not be considered as free variables
2019-10-28 21:40:53 +04:00
3) Build the initial record (later passed as 2nd argument of
`MAP/SET/LIST_FOLD`) capturing the environment using the
free variables list of (2)
4) In the filtered body of (1), replace occurences:
- free variable of name X as rhs ==> accessor `#COMPILER#acc.X`
- free variable of name X as lhs ==> accessor `#COMPILER#acc.X`
2019-11-15 21:10:24 +04:00
And, in the case of a map:
- references to the iterated key ==> variable `#COMPILER#elt_K`
- references to the iterated value ==> variable `#COMPILER#elt_V`
2019-11-15 21:10:24 +04:00
in the case of a set/list:
- references to the iterated value ==> variable `#COMPILER#elt_X`
2019-11-15 21:10:24 +04:00
Note: In the case of an inner loop capturing variable from an outer loop
the free variable name can be `#COMPILER#acc.Y` and because we do not
capture the accumulator record in the inner loop, we don't want to
generate `#COMPILER#acc.#COMPILER#acc.Y` but `#COMPILER#acc.Y`
2019-10-28 21:40:53 +04:00
5) Append the return value to the body
6) Prepend the declaration of the lambda arguments to the body which
is a serie of `let .. in`'s
Note that the parameter of the lambda ̀arguments` is a tree of
tuple holding:
* In the case of `list` or ̀set`:
( folding record , current list/set element ) as
( #COMPILER#acc , #COMPILER#elt_X )
2019-10-28 21:40:53 +04:00
* In the case of `map`:
( folding record , current map key , current map value ) as
( #COMPILER#acc , #COMPILER#elt_K , #COMPILER#elt_V )
Note: X , K and V above have to be replaced with their given name
2019-10-28 21:40:53 +04:00
7) Build the lambda using the final body of (6)
2019-11-06 20:23:49 +04:00
8) Build a sequence of assignments for all the captured variables
2019-10-28 21:40:53 +04:00
to their new value, namely an access to the folded record
(#COMPILER#folded_record)
2019-11-06 20:23:49 +04:00
9) Attach the sequence of 8 to the ̀let .. in` declaration
2019-10-28 21:40:53 +04:00
of #COMPILER#folded_record
**)
2019-10-15 15:14:00 +04:00
and simpl_for_collect : Raw.for_collect -> (_ -> expression result) result = fun fc ->
let elt_name = "#COMPILER#elt_"^fc.var.value in
let elt_v_name = match fc.bind_to with
| Some v -> "#COMPILER#elt"^(snd v).value
| None -> "#COMPILER#elt_unused" in
2019-10-28 21:40:53 +04:00
(* STEP 1 *)
2019-10-29 14:41:59 +04:00
let%bind for_body = simpl_block fc.block.value in
let%bind for_body = for_body None in
2019-10-28 21:40:53 +04:00
(* STEP 2 *)
2019-10-29 14:41:59 +04:00
let%bind captured_name_list = Self_ast_simplified.fold_expression
(fun (prev : type_name list) (ass_exp : expression) ->
2019-10-28 21:40:53 +04:00
match ass_exp.expression with
| E_assign ( name , _ , _ ) ->
2019-11-15 21:10:24 +04:00
if is_compiler_generated name then ok prev
else ok (name::prev)
2019-10-29 14:41:59 +04:00
| _ -> ok prev )
[]
for_body in
2019-10-28 21:40:53 +04:00
(* STEP 3 *)
2019-10-27 16:03:08 +04:00
let add_to_record (prev: expression type_name_map) (captured_name: string) =
SMap.add captured_name (e_variable captured_name) prev in
let init_record = e_record (List.fold_left add_to_record SMap.empty captured_name_list) in
2019-10-28 21:40:53 +04:00
(* STEP 4 *)
2019-10-27 14:32:03 +04:00
let replace exp =
match exp.expression with
2019-11-15 21:10:24 +04:00
(* replace references to fold accumulator as lhs *)
2019-10-28 21:40:53 +04:00
| E_assign ( name , path , expr ) -> (
let path' = List.filter
( fun el ->
match el with
2019-11-15 21:10:24 +04:00
| Access_record name -> not @@ is_compiler_generated name
| _ -> true )
((Access_record name)::path) in
ok @@ e_assign "#COMPILER#acc" path' expr)
2019-10-28 21:40:53 +04:00
| E_variable name -> (
2019-10-29 14:41:59 +04:00
if (List.mem name captured_name_list) then
2019-11-15 21:10:24 +04:00
(* replace references to fold accumulator as rhs *)
2019-10-29 14:41:59 +04:00
ok @@ e_accessor (e_variable "#COMPILER#acc") [Access_record name]
2019-11-06 20:23:49 +04:00
else match fc.collection with
(* loop on map *)
| Map _ ->
let k' = e_variable elt_name in
2019-10-29 14:41:59 +04:00
if ( name = fc.var.value ) then
ok @@ k' (* replace references to the the key *)
else (
match fc.bind_to with
| Some (_,v) ->
let v' = e_variable elt_v_name in
2019-10-29 14:41:59 +04:00
if ( name = v.value ) then
ok @@ v' (* replace references to the the value *)
else ok @@ exp
| None -> ok @@ exp
)
(* loop on set or list *)
| (Set _ | List _) ->
if (name = fc.var.value ) then
(* replace references to the collection element *)
ok @@ (e_variable elt_name)
else ok @@ exp
)
| _ -> ok @@ exp in
2019-10-27 16:03:08 +04:00
let%bind for_body = Self_ast_simplified.map_expression replace for_body in
2019-10-28 21:40:53 +04:00
(* STEP 5 *)
2019-10-27 16:03:08 +04:00
let rec add_return (expr : expression) = match expr.expression with
| E_sequence (a,b) -> e_sequence a (add_return b)
2019-10-28 21:40:53 +04:00
| _ -> e_sequence expr (e_variable "#COMPILER#acc") in
2019-10-27 16:03:08 +04:00
let for_body = add_return for_body in
2019-10-28 21:40:53 +04:00
(* STEP 6 *)
let for_body =
let ( arg_access: Types.access_path -> expression ) = e_accessor (e_variable "arguments") in
2019-11-06 20:23:49 +04:00
( match fc.collection with
| Map _ ->
let acc = arg_access [Access_tuple 0 ] in
let collec_elt_v = arg_access [Access_tuple 1 ; Access_tuple 0] in
let collec_elt_k = arg_access [Access_tuple 1 ; Access_tuple 1] in
2019-10-28 21:40:53 +04:00
e_let_in ("#COMPILER#acc", None) acc @@
e_let_in (elt_name, None) collec_elt_v @@
e_let_in (elt_v_name, None) collec_elt_k (for_body)
| _ ->
let acc = arg_access [Access_tuple 0] in
let collec_elt = arg_access [Access_tuple 1] in
2019-10-28 21:40:53 +04:00
e_let_in ("#COMPILER#acc", None) acc @@
e_let_in (elt_name, None) collec_elt (for_body)
) in
2019-10-28 21:40:53 +04:00
(* STEP 7 *)
let%bind collect = simpl_expression fc.expr in
2019-10-27 16:03:08 +04:00
let lambda = e_lambda "arguments" None None for_body in
let op_name = match fc.collection with
2019-10-27 16:03:08 +04:00
| Map _ -> "MAP_FOLD" | Set _ -> "SET_FOLD" | List _ -> "LIST_FOLD" in
let fold = e_constant op_name [collect ; init_record ; lambda] in
2019-10-28 21:40:53 +04:00
(* STEP 8 *)
2019-10-27 16:03:08 +04:00
let assign_back (prev : expression option) (captured_varname : string) : expression option =
2019-10-28 21:40:53 +04:00
let access = e_accessor (e_variable "#COMPILER#folded_record")
[Access_record captured_varname] in
let assign = e_assign captured_varname [] access in
2019-11-06 20:23:49 +04:00
match prev with
| None -> Some assign
2019-10-27 14:32:03 +04:00
| Some p -> Some (e_sequence p assign) in
2019-10-27 16:03:08 +04:00
let reassign_sequence = List.fold_left assign_back None captured_name_list in
2019-10-28 21:40:53 +04:00
(* STEP 9 *)
2019-10-27 14:32:03 +04:00
let final_sequence = match reassign_sequence with
(* None case means that no variables were captured *)
2019-10-28 21:53:40 +04:00
| None -> e_skip ()
2019-10-28 21:40:53 +04:00
| Some seq -> e_let_in ("#COMPILER#folded_record", None) fold seq in
return_statement @@ final_sequence
2019-05-13 00:56:22 +04:00
let simpl_program : Raw.ast -> program result = fun t ->
2019-11-06 20:23:49 +04:00
bind_list @@ List.map simpl_declaration @@ nseq_to_list t.decl