summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSon Ho2022-02-03 19:44:26 +0100
committerSon Ho2022-02-03 19:44:26 +0100
commit5eacfc7cdbe99f401d6cf925cbb50d63c3a780c3 (patch)
tree940463f3d64b43534bb09bafafd8beccee366d43
parent1dd71ca1d37d5ca7fae5a7e9766e03194dfb764f (diff)
Implement extraction of `if ... then ... else ...`
-rw-r--r--src/ExtractToFStar.ml37
-rw-r--r--src/PureUtils.ml11
2 files changed, 47 insertions, 1 deletions
diff --git a/src/ExtractToFStar.ml b/src/ExtractToFStar.ml
index 100b5ecb..689bd797 100644
--- a/src/ExtractToFStar.ml
+++ b/src/ExtractToFStar.ml
@@ -759,7 +759,42 @@ let rec extract_texpression (ctx : extraction_ctx) (fmt : F.formatter)
F.pp_close_box fmt ();
(* Print the next expression *)
extract_texpression ctx fmt inside next_e
- | Switch (e, body) -> raise Unimplemented
+ | Switch (scrut, body) -> (
+ match body with
+ | If (e_then, e_else) ->
+ (* Open a box for the whole `if ... then ... else ...` *)
+ F.pp_open_hvbox fmt 0;
+ (* Open a box for the `if` *)
+ F.pp_open_hovbox fmt ctx.indent_incr;
+ F.pp_print_string fmt "if";
+ F.pp_print_space fmt ();
+ let ctx = extract_texpression ctx fmt false scrut in
+ (* Close the box for the `if` *)
+ F.pp_close_box fmt ();
+ let extract_branch (is_then : bool) (e_branch : texpression) :
+ extraction_ctx =
+ F.pp_print_space fmt ();
+ (* Open a box for the branch *)
+ F.pp_open_hovbox fmt ctx.indent_incr;
+ let then_or_else = if is_then then "then" else "else" in
+ F.pp_print_string fmt then_or_else;
+ F.pp_print_space fmt ();
+ let parenth = PureUtils.expression_requires_parentheses e_branch in
+ if parenth then F.pp_print_string fmt "(";
+ let ctx = extract_texpression ctx fmt false e_branch in
+ if parenth then F.pp_print_string fmt ")";
+ (* Close the box for the branch *)
+ F.pp_close_box fmt ();
+ ctx
+ in
+ let ctx = extract_branch false e_then in
+ let ctx = extract_branch false e_else in
+ (* Close the box for the whole `if ... then ... else ...` *)
+ F.pp_close_box fmt ();
+ (* Return *)
+ ctx
+ | SwitchInt (_, branches, otherwise) -> raise Unimplemented
+ | Match branches -> raise Unimplemented)
| Meta (_, e) -> extract_texpression ctx fmt inside e
(** Extract a function definition.
diff --git a/src/PureUtils.ml b/src/PureUtils.ml
index dfa8c1a3..aa383d8c 100644
--- a/src/PureUtils.ml
+++ b/src/PureUtils.ml
@@ -183,3 +183,14 @@ let functions_not_mutually_recursive (funs : fun_def list) : bool =
with Utils.Found -> false
in
List.for_all body_only_calls_itself funs
+
+(** We use this to check whether we need to add parentheses around expressions.
+ We only look for outer monadic let-bindings.
+ *)
+let rec expression_requires_parentheses (e : texpression) : bool =
+ match e.e with
+ | Value _ | Call _ -> false
+ | Let (monadic, _, _, next_e) ->
+ if monadic then true else expression_requires_parentheses next_e
+ | Switch (_, _) -> false
+ | Meta (_, next_e) -> expression_requires_parentheses next_e