2019-05-13 00:56:22 +04:00
|
|
|
open Trace
|
|
|
|
open Ast_simplified
|
|
|
|
|
|
|
|
module Raw = Parser.Pascaligo.AST
|
|
|
|
module SMap = Map.String
|
2019-11-19 17:25:48 +04:00
|
|
|
module SSet = Set.Make (String)
|
2019-05-13 00:56:22 +04:00
|
|
|
|
|
|
|
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
|
|
|
|
2019-11-19 17:25:48 +04:00
|
|
|
let detect_local_declarations (for_body : expression) =
|
|
|
|
let%bind aux = Self_ast_simplified.fold_expression
|
|
|
|
(fun (nlist, cur_loop : type_name list * bool) (ass_exp : expression) ->
|
|
|
|
if cur_loop then
|
|
|
|
match ass_exp.expression with
|
|
|
|
| E_let_in {binder;rhs = _;result = _} ->
|
|
|
|
let (name,_) = binder in
|
|
|
|
ok (name::nlist, cur_loop)
|
|
|
|
| E_constant ("MAP_FOLD", _)
|
|
|
|
| E_constant ("SET_FOLD", _)
|
|
|
|
| E_constant ("LIST_FOLD", _) -> ok @@ (nlist, false)
|
|
|
|
| _ -> ok (nlist, cur_loop)
|
|
|
|
else
|
|
|
|
ok @@ (nlist, cur_loop)
|
|
|
|
)
|
|
|
|
([], true)
|
|
|
|
for_body in
|
|
|
|
ok @@ fst aux
|
|
|
|
|
|
|
|
let detect_free_variables (for_body : expression) (local_decl_names : string list) =
|
|
|
|
let%bind captured_names = Self_ast_simplified.fold_expression
|
|
|
|
(fun (prev : type_name list) (ass_exp : expression) ->
|
|
|
|
match ass_exp.expression with
|
|
|
|
| E_assign ( name , _ , _ ) ->
|
|
|
|
if is_compiler_generated name then ok prev
|
|
|
|
else ok (name::prev)
|
2019-11-21 04:32:43 +04:00
|
|
|
| E_constant (n, [a;b])
|
|
|
|
when n="OR" || n="AND" || n="LT" || n="GT" ||
|
|
|
|
n="LE" || n="GE" || n="EQ" || n="NEQ" -> (
|
|
|
|
match (a.expression,b.expression) with
|
|
|
|
| E_variable na , E_variable nb ->
|
|
|
|
let ret = [] in
|
|
|
|
let ret = if not (is_compiler_generated na) then
|
|
|
|
na::ret else ret in
|
|
|
|
let ret = if not (is_compiler_generated nb) then
|
|
|
|
nb::ret else ret in
|
|
|
|
ok (ret@prev)
|
|
|
|
| E_variable n , _
|
|
|
|
| _ , E_variable n ->
|
|
|
|
if not (is_compiler_generated n) then
|
|
|
|
ok (n::prev) else ok prev
|
|
|
|
| _ -> ok prev)
|
2019-11-19 17:25:48 +04:00
|
|
|
| _ -> ok prev )
|
|
|
|
[]
|
|
|
|
for_body in
|
|
|
|
ok @@ SSet.elements
|
|
|
|
@@ SSet.diff (SSet.of_list captured_names) (SSet.of_list local_decl_names)
|
|
|
|
|
2019-06-04 18:12:17 +04:00
|
|
|
module Errors = struct
|
2019-06-13 18:57:40 +04:00
|
|
|
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
|
|
|
|
|
2019-06-04 18:12:17 +04:00
|
|
|
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
|
|
|
|
|
2019-06-05 19:51:06 +04:00
|
|
|
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",
|
2019-10-09 18:07:13 +04:00
|
|
|
fun () -> Parser.Pascaligo.ParserLog.pattern_to_string p)
|
2019-06-05 19:51:06 +04:00
|
|
|
] 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
|
|
|
|
|
2019-11-18 19:10:48 +04:00
|
|
|
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
|
|
|
|
|
2019-06-05 19:51:06 +04:00
|
|
|
(* Logging *)
|
|
|
|
|
|
|
|
let simplifying_instruction t =
|
|
|
|
let title () = "simplifiying instruction" in
|
|
|
|
let message () = "" in
|
|
|
|
let data = [
|
|
|
|
("instruction",
|
2019-10-09 18:07:13 +04:00
|
|
|
fun () -> Parser.Pascaligo.ParserLog.instruction_to_string t)
|
2019-06-05 19:51:06 +04:00
|
|
|
] in
|
|
|
|
error ~data title message
|
2019-06-04 18:12:17 +04:00
|
|
|
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 ->
|
2019-05-22 04:46:54 +04:00
|
|
|
match expr'_opt with
|
2019-06-04 18:12:17 +04:00
|
|
|
| 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-05-22 04:46:54 +04:00
|
|
|
|
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 =
|
2019-06-04 18:12:17 +04:00
|
|
|
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
|
2019-06-04 18:12:17 +04:00
|
|
|
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
|
2019-06-04 18:12:17 +04:00
|
|
|
@@ List.map apply
|
2019-10-23 02:35:29 +04:00
|
|
|
@@ 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) =
|
2019-05-17 18:29:22 +04:00
|
|
|
let args =
|
2019-11-06 20:23:49 +04:00
|
|
|
match v.value.arg with
|
2019-05-17 18:29:22 +04:00
|
|
|
None -> []
|
2019-11-06 20:23:49 +04:00
|
|
|
| Some (_, TProd product) -> npseq_to_list product.value
|
|
|
|
| Some (_, t_expr) -> [t_expr] in
|
2019-10-15 23:03:46 +04:00
|
|
|
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'
|
|
|
|
|
|
|
|
|
2019-05-23 10:22:58 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
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 -> (
|
2019-11-14 23:12:41 +04:00
|
|
|
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
|
2019-11-14 23:12:41 +04:00
|
|
|
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
|
|
|
)
|
2019-05-23 10:22:58 +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 ->
|
2019-10-15 23:03:46 +04:00
|
|
|
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
|
2019-06-11 19:10:27 +04:00
|
|
|
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 *)
|
2019-06-04 18:12:17 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
| 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 ->
|
2019-10-18 16:32:58 +04:00
|
|
|
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
|
2019-06-04 18:12:17 +04:00
|
|
|
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
|
|
|
|
)
|
2019-11-18 19:10:48 +04:00
|
|
|
| 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
|
|
|
|
2019-05-23 10:22:58 +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
|
|
|
|
2019-05-23 10:22:58 +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 =
|
2019-05-23 10:22:58 +04:00
|
|
|
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 =
|
2019-05-23 10:22:58 +04:00
|
|
|
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 =
|
2019-05-23 10:22:58 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
| [hd] -> simpl_expression hd
|
2019-06-04 18:12:17 +04:00
|
|
|
| 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
|
|
|
|
2019-05-22 04:46:54 +04:00
|
|
|
and simpl_local_declaration : Raw.local_decl -> _ result = fun t ->
|
2019-05-13 00:56:22 +04:00
|
|
|
match t with
|
2019-06-04 18:12:17 +04:00
|
|
|
| LocalData d ->
|
|
|
|
simpl_data_declaration d
|
2019-10-16 17:39:08 +04:00
|
|
|
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
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-11-18 19:10:48 +04:00
|
|
|
| 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
|
|
|
|
2019-06-04 18:12:17 +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
|
2019-05-23 10:22:58 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
ok (type_name , type_expression)
|
2019-05-13 00:56:22 +04:00
|
|
|
|
2019-11-18 19:10:48 +04:00
|
|
|
and simpl_fun_expression :
|
|
|
|
loc:_ -> Raw.fun_expr -> ((name option * type_expression option) * expression) result =
|
2019-06-04 18:12:17 +04:00
|
|
|
fun ~loc x ->
|
2019-05-13 00:56:22 +04:00
|
|
|
let open! Raw in
|
2019-11-18 19:10:48 +04:00
|
|
|
let {name;param;ret_type;local_decls;block;return} : fun_expr = x in
|
2019-10-19 21:46:24 +04:00
|
|
|
let statements =
|
|
|
|
match block with
|
|
|
|
| Some block -> npseq_to_list block.value.statements
|
|
|
|
| None -> []
|
2019-10-19 20:11:18 +04:00
|
|
|
in
|
2019-10-17 20:46:40 +04:00
|
|
|
(match param.value.inside with
|
|
|
|
a, [] -> (
|
2019-05-13 00:56:22 +04:00
|
|
|
let%bind input = simpl_param a in
|
2019-11-18 19:10:48 +04:00
|
|
|
let name = Option.map (fun (x : _ reg) -> x.value) name in
|
2019-05-23 10:22:58 +04:00
|
|
|
let (binder , input_type) = input in
|
2019-05-13 00:56:22 +04:00
|
|
|
let%bind local_declarations =
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
2019-10-19 21:46:24 +04:00
|
|
|
@@ 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
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
ok ((name , type_annotation) , expression)
|
2019-05-13 00:56:22 +04:00
|
|
|
)
|
|
|
|
| lst -> (
|
2019-10-17 20:46:40 +04:00
|
|
|
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
|
2019-05-23 10:22:58 +04:00
|
|
|
let (binder , input_type) =
|
|
|
|
let type_expression = T_tuple (List.map snd params) in
|
|
|
|
(arguments_name , type_expression) in
|
2019-05-22 04:46:54 +04:00
|
|
|
let%bind tpl_declarations =
|
2019-05-23 10:22:58 +04:00
|
|
|
let aux = fun i x ->
|
2019-06-04 18:12:17 +04:00
|
|
|
let expr = e_accessor (e_variable arguments_name) [Access_tuple i] in
|
2019-05-23 10:22:58 +04:00
|
|
|
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
|
2019-05-22 04:46:54 +04:00
|
|
|
bind_list @@ List.mapi aux params in
|
2019-05-13 00:56:22 +04:00
|
|
|
let%bind local_declarations =
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
2019-10-19 21:46:24 +04:00
|
|
|
@@ statements in
|
2019-05-13 00:56:22 +04:00
|
|
|
let%bind result = simpl_expression return in
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
2019-06-04 18:12:17 +04:00
|
|
|
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
|
2019-11-18 19:10:48 +04:00
|
|
|
let name = Option.map (fun (x : _ reg) -> x.value) name in
|
|
|
|
ok ((name , type_annotation) , expression)
|
2019-05-13 00:56:22 +04:00
|
|
|
)
|
|
|
|
)
|
2019-06-04 18:12:17 +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
|
2019-05-23 10:22:58 +04:00
|
|
|
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)
|
2019-10-16 17:39:08 +04:00
|
|
|
| FunDecl x -> (
|
2019-05-28 19:36:14 +04:00
|
|
|
let (x , loc) = r_split x in
|
2019-11-18 19:10:48 +04:00
|
|
|
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
|
|
|
|
2019-06-04 18:12:17 +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
|
2019-05-22 04:46:54 +04:00
|
|
|
| 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 =
|
2019-06-04 18:12:17 +04:00
|
|
|
fun t ->
|
2019-05-13 00:56:22 +04:00
|
|
|
match t with
|
2019-07-20 15:46:42 +04:00
|
|
|
| ProcCall x -> (
|
2019-11-14 23:12:41 +04:00
|
|
|
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 ->
|
2019-11-14 23:12:41 +04:00
|
|
|
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 ->
|
2019-11-14 23:12:41 +04:00
|
|
|
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-11-14 23:12:41 +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
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
2019-10-08 18:41:47 +04:00
|
|
|
let%bind (varname,map,path) = match v'.path with
|
|
|
|
| Name name -> ok (name.value , e_variable name.value, [])
|
|
|
|
| Path p ->
|
2019-10-08 20:20:32 +04:00
|
|
|
let (name,p') = simpl_path v'.path in
|
2019-10-13 22:15:50 +04:00
|
|
|
let%bind accessor = simpl_projection p in
|
2019-10-08 20:20:32 +04:00
|
|
|
ok @@ (name , accessor , p')
|
2019-10-08 18:41:47 +04:00
|
|
|
in
|
2019-05-13 00:56:22 +04:00
|
|
|
let%bind key_expr = simpl_expression v'.index.value.inside in
|
2019-10-08 18:41:47 +04:00
|
|
|
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
|
|
|
)
|
|
|
|
)
|
2019-05-22 04:46:54 +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 =
|
2019-10-24 11:58:33 +04:00
|
|
|
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; _} ->
|
2019-10-24 12:29:41 +04:00
|
|
|
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
|
2019-10-24 12:29:41 +04:00
|
|
|
return_statement @@ e_matching ~loc expr m
|
2019-05-22 04:46:54 +04:00
|
|
|
)
|
2019-05-13 00:56:22 +04:00
|
|
|
| RecordPatch r -> (
|
|
|
|
let r = r.value in
|
|
|
|
let (name , access_path) = simpl_path r.path in
|
2019-10-23 02:35:29 +04:00
|
|
|
|
|
|
|
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
|
2019-06-05 19:51:06 +04:00
|
|
|
let%bind e = simpl_expression x.field_expr
|
|
|
|
in ok (x.field_name.value, e , loc)
|
2019-05-28 19:36:14 +04:00
|
|
|
)
|
2019-10-23 02:35:29 +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
|
|
|
|
|
2019-05-22 04:46:54 +04:00
|
|
|
let%bind expr =
|
2019-05-28 19:36:14 +04:00
|
|
|
let aux = fun (access , v , loc) ->
|
2019-10-23 02:35:29 +04:00
|
|
|
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
|
2019-05-22 04:46:54 +04:00
|
|
|
in
|
2019-09-10 14:42:49 +04:00
|
|
|
return_statement @@ expr
|
2019-10-23 02:35:29 +04:00
|
|
|
)
|
2019-10-11 05:26:28 +04:00
|
|
|
| MapPatch patch -> (
|
|
|
|
let (map_p, loc) = r_split patch in
|
|
|
|
let (name, access_path) = simpl_path map_p.path in
|
2019-10-12 02:44:16 +04:00
|
|
|
let%bind inj = bind_list
|
|
|
|
@@ List.map (fun (x:Raw.binding Region.reg) ->
|
|
|
|
let x = x.value in
|
2019-10-11 05:26:28 +04:00
|
|
|
let (key, value) = x.source, x.image in
|
|
|
|
let%bind key' = simpl_expression key in
|
|
|
|
let%bind value' = simpl_expression value
|
2019-10-12 02:44:16 +04:00
|
|
|
in ok @@ (key', value')
|
2019-10-11 05:26:28 +04:00
|
|
|
)
|
2019-10-23 02:35:29 +04:00
|
|
|
@@ npseq_to_list map_p.map_inj.value.ne_elements in
|
2019-10-12 02:44:16 +04:00
|
|
|
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))
|
2019-10-12 02:44:16 +04:00
|
|
|
inj
|
2019-10-14 23:05:35 +04:00
|
|
|
(e_accessor ~loc (e_variable name) access_path)
|
2019-10-12 02:44:16 +04:00
|
|
|
in e_assign ~loc name access_path assigns
|
|
|
|
in return_statement @@ expr
|
2019-10-11 05:26:28 +04:00
|
|
|
)
|
2019-10-10 04:08:58 +04:00
|
|
|
| SetPatch patch -> (
|
2019-10-11 00:35:38 +04:00
|
|
|
let (setp, loc) = r_split patch in
|
2019-10-10 04:08:58 +04:00
|
|
|
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 @@
|
2019-10-23 02:35:29 +04:00
|
|
|
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
|
2019-10-10 04:08:58 +04:00
|
|
|
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
|
2019-10-08 18:41:47 +04:00
|
|
|
let%bind (varname,map,path) = match v.map with
|
|
|
|
| Name v -> ok (v.value , e_variable v.value , [])
|
|
|
|
| Path p ->
|
2019-10-08 20:20:32 +04:00
|
|
|
let (name,p') = simpl_path v.map in
|
2019-10-13 22:15:50 +04:00
|
|
|
let%bind accessor = simpl_projection p in
|
2019-10-08 20:20:32 +04:00
|
|
|
ok @@ (name , accessor , p')
|
2019-10-08 18:41:47 +04:00
|
|
|
in
|
2019-05-13 00:56:22 +04:00
|
|
|
let%bind key' = simpl_expression key in
|
2019-10-08 18:41:47 +04:00
|
|
|
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
|
|
|
)
|
2019-10-08 08:41:36 +04:00
|
|
|
| SetRemove r -> (
|
|
|
|
let (set_rm, loc) = r_split r in
|
2019-10-15 03:04:48 +04:00
|
|
|
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
|
2019-10-08 08:41:36 +04:00
|
|
|
let%bind removed' = simpl_expression set_rm.element in
|
2019-10-15 03:04:48 +04:00
|
|
|
let expr = e_constant ~loc "SET_REMOVE" [removed' ; set] in
|
|
|
|
return_statement @@ e_assign ~loc varname path expr
|
2019-10-08 08:41:36 +04:00
|
|
|
)
|
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
|
2019-06-05 19:51:06 +04:00
|
|
|
let get_var (t:Raw.pattern) =
|
|
|
|
match t with
|
2019-05-13 00:56:22 +04:00
|
|
|
| PVar v -> ok v.value
|
2019-06-13 18:57:40 +04:00
|
|
|
| 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
|
2019-06-13 18:57:40 +04:00
|
|
|
| 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 () =
|
2019-06-05 19:51:06 +04:00
|
|
|
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
|
|
|
|
)
|
2019-10-13 22:15:50 +04:00
|
|
|
| pattern -> ok pattern in
|
2019-06-13 18:57:40 +04:00
|
|
|
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
|
2019-06-13 18:57:40 +04:00
|
|
|
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)
|
2019-06-13 18:57:40 +04:00
|
|
|
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)] ->
|
2019-06-05 19:51:06 +04:00
|
|
|
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
|
2019-06-05 19:51:06 +04:00
|
|
|
| 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)
|
2019-06-05 19:51:06 +04:00
|
|
|
| _ -> 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 ->
|
2019-06-05 19:51:06 +04:00
|
|
|
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 () =
|
2019-10-09 18:07:13 +04:00
|
|
|
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
|
|
|
|
|
2019-06-05 19:51:06 +04:00
|
|
|
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
|
|
|
|
2019-06-05 19:51:06 +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
|
2019-05-22 04:46:54 +04:00
|
|
|
let%bind fs = bind_map_list simpl_statement lst in
|
2019-06-05 19:51:06 +04:00
|
|
|
let aux : _ -> (expression option -> expression result) -> _ =
|
|
|
|
fun prec cur ->
|
|
|
|
let%bind res = cur prec in
|
|
|
|
ok @@ Some res in
|
2019-05-22 04:46:54 +04:00
|
|
|
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
|
|
|
|
2019-05-23 10:22:58 +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
|
2019-11-18 17:14:54 +04:00
|
|
|
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-19 17:25:48 +04:00
|
|
|
Free variables are simply variables being assigned but not defined
|
|
|
|
locally.
|
2019-11-15 21:10:24 +04:00
|
|
|
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 :
|
|
|
|
```
|
2019-11-18 17:14:54 +04:00
|
|
|
#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:
|
2019-11-18 17:14:54 +04:00
|
|
|
- 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:
|
2019-11-18 17:14:54 +04:00
|
|
|
- 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
|
2019-11-18 17:14:54 +04:00
|
|
|
( #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
|
2019-11-18 17:14:54 +04:00
|
|
|
( #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 ->
|
2019-11-18 17:14:54 +04:00
|
|
|
let elt_name = "#COMPILER#elt_"^fc.var.value in
|
|
|
|
let elt_v_name = match fc.bind_to with
|
2019-11-26 20:49:23 +04:00
|
|
|
| Some v -> "#COMPILER#elt_"^(snd v).value
|
2019-11-18 17:14:54 +04:00
|
|
|
| None -> "#COMPILER#elt_unused" in
|
2019-11-21 04:32:43 +04:00
|
|
|
let element_names = ok @@ match fc.bind_to with
|
|
|
|
| Some v -> [fc.var.value;(snd v).value]
|
|
|
|
| None -> [fc.var.value] 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-11-21 04:32:43 +04:00
|
|
|
let%bind local_decl_name_list = bind_concat (detect_local_declarations for_body) element_names in
|
2019-11-19 17:25:48 +04:00
|
|
|
let%bind captured_name_list = detect_free_variables for_body local_decl_name_list 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 =
|
2019-10-26 16:18:06 +04:00
|
|
|
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 ) -> (
|
2019-11-19 17:25:48 +04:00
|
|
|
if (List.mem name local_decl_name_list ) then
|
|
|
|
ok @@ exp
|
|
|
|
else
|
|
|
|
let path' = List.filter
|
|
|
|
( fun el ->
|
|
|
|
match el with
|
|
|
|
| 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
|
2019-10-27 19:42:11 +04:00
|
|
|
(* loop on map *)
|
|
|
|
| Map _ ->
|
2019-11-18 17:14:54 +04:00
|
|
|
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) ->
|
2019-11-18 17:14:54 +04:00
|
|
|
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
|
2019-10-27 19:42:11 +04:00
|
|
|
)
|
|
|
|
(* loop on set or list *)
|
|
|
|
| (Set _ | List _) ->
|
|
|
|
if (name = fc.var.value ) then
|
|
|
|
(* replace references to the collection element *)
|
2019-11-18 17:14:54 +04:00
|
|
|
ok @@ (e_variable elt_name)
|
2019-10-27 19:42:11 +04:00
|
|
|
else ok @@ exp
|
|
|
|
)
|
2019-10-26 16:18:06 +04:00
|
|
|
| _ -> 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
|
2019-10-27 00:07:42 +04:00
|
|
|
| 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 *)
|
2019-10-27 19:42:11 +04:00
|
|
|
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
|
2019-10-27 19:42:11 +04:00
|
|
|
| Map _ ->
|
2019-11-15 18:49:50 +04:00
|
|
|
let acc = arg_access [Access_tuple 0 ] in
|
2019-10-27 19:42:11 +04:00
|
|
|
let collec_elt_v = arg_access [Access_tuple 1 ; Access_tuple 0] in
|
2019-11-15 18:49:50 +04:00
|
|
|
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 @@
|
2019-11-18 17:14:54 +04:00
|
|
|
e_let_in (elt_name, None) collec_elt_v @@
|
|
|
|
e_let_in (elt_v_name, None) collec_elt_k (for_body)
|
2019-10-27 19:42:11 +04:00
|
|
|
| _ ->
|
|
|
|
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 @@
|
2019-11-18 17:14:54 +04:00
|
|
|
e_let_in (elt_name, None) collec_elt (for_body)
|
2019-10-27 19:42:11 +04:00
|
|
|
) in
|
2019-10-28 21:40:53 +04:00
|
|
|
(* STEP 7 *)
|
2019-10-26 16:18:06 +04:00
|
|
|
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
|
2019-10-26 16:27:29 +04:00
|
|
|
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
|
2019-11-20 16:16:31 +04:00
|
|
|
let fold = e_constant op_name [lambda; collect ; init_record] 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")
|
2019-10-26 16:18:06 +04:00
|
|
|
[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
|
2019-10-26 16:18:06 +04:00
|
|
|
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
|