summaryrefslogtreecommitdiff
path: root/compiler
diff options
context:
space:
mode:
Diffstat (limited to 'compiler')
-rw-r--r--compiler/PrePasses.ml111
1 files changed, 107 insertions, 4 deletions
diff --git a/compiler/PrePasses.ml b/compiler/PrePasses.ml
index 8f1354b7..1bdaf174 100644
--- a/compiler/PrePasses.ml
+++ b/compiler/PrePasses.ml
@@ -8,10 +8,12 @@ module E = Expressions
module C = Contexts
module A = LlbcAst
module L = Logging
+open LlbcAstUtils
let log = L.pre_passes_log
(** Rustc inserts a lot of drops before the assignments.
+
We consider those drops are part of the assignment, and splitting the
drop and the assignment is problematic for us because it can introduce
[⊥] under borrows. For instance, we encountered situations like the
@@ -22,7 +24,7 @@ let log = L.pre_passes_log
*x = move ...;
]}
- TODO: this is not necessary anymore
+ Rem.: we don't use this anymore
*)
let filter_drop_assigns (f : A.fun_decl) : A.fun_decl =
(* The visitor *)
@@ -49,6 +51,107 @@ let filter_drop_assigns (f : A.fun_decl) : A.fun_decl =
in
{ f with body }
-let apply_passes (m : A.crate) : A.crate =
- let functions = List.map filter_drop_assigns m.functions in
- { m with functions }
+(** This pass slightly restructures the control-flow to remove the need to
+ merge branches during the symbolic execution in some quite common cases
+ where doing a merge is actually not necessary and leads to an ugly translation.
+
+ For instance, it performs the following transformation:
+ {[
+ if b {
+ var@0 := &mut *x;
+ }
+ else {
+ var@0 := move y;
+ }
+ return;
+
+ ~~>
+
+ if b {
+ var@0 := &mut *x;
+ return;
+ }
+ else {
+ var@0 := move y;
+ return;
+ }
+ ]}
+
+ This way, the translated body doesn't have an intermediate assignment,
+ for the `if ... then ... else ...` expression (together with a backward
+ function).
+
+ More precisly, we move (and duplicate) a statement happening after a branching
+ inside the branches if:
+ - this statement ends with [return] or [panic]
+ - this statement is only made of a sequence of nops, assignments (with some
+ restrictions on the rvalue), fake reads, drops (usually, returns will be
+ followed by such statements)
+ *)
+let remove_useless_cf_merges (crate : A.crate) (f : A.fun_decl) : A.fun_decl =
+ let f0 = f in
+ (* Return [true] if the statement can be moved inside the branches of a switch.
+ *
+ * [must_end_with_exit]: we need this boolean because the inner statements
+ * (inside the encountered sequences) don't need to end with [return] or [panic],
+ * but all the paths inside the whole statement have to.
+ * *)
+ let rec can_be_moved_aux (must_end_with_exit : bool) (st : A.statement) : bool
+ =
+ match st.content with
+ | SetDiscriminant _ | Assert _ | Call _ | Break _ | Continue _ | Switch _
+ | Loop _ ->
+ false
+ | Assign (_, rv) -> (
+ match rv with
+ | Use _ | Ref _ -> not must_end_with_exit
+ | Aggregate (AggregatedTuple, []) -> not must_end_with_exit
+ | _ -> false)
+ | FakeRead _ | Drop _ | Nop -> not must_end_with_exit
+ | Panic | Return -> true
+ | Sequence (st1, st2) ->
+ can_be_moved_aux false st1 && can_be_moved_aux must_end_with_exit st2
+ in
+ let can_be_moved = can_be_moved_aux true in
+
+ (* The visitor *)
+ let obj =
+ object
+ inherit [_] A.map_statement as super
+
+ method! visit_Sequence env st1 st2 =
+ match st1.content with
+ | Switch (op, tgts) ->
+ if can_be_moved st2 then
+ super#visit_Switch env op
+ (chain_statements_in_switch_targets tgts st2)
+ else super#visit_Sequence env st1 st2
+ | _ -> super#visit_Sequence env st1 st2
+ end
+ in
+
+ (* Map *)
+ let body =
+ match f.body with
+ | Some body -> Some { body with body = obj#visit_statement () body.body }
+ | None -> None
+ in
+ let f = { f with body } in
+ log#ldebug
+ (lazy
+ ("Before/after [remove_useless_cf_merges]:\n"
+ ^ Print.Crate.crate_fun_decl_to_string crate f0
+ ^ "\n\n"
+ ^ Print.Crate.crate_fun_decl_to_string crate f
+ ^ "\n"));
+ f
+
+let apply_passes (crate : A.crate) : A.crate =
+ let passes = [ remove_useless_cf_merges crate ] in
+ let functions =
+ List.fold_left (fun fl pass -> List.map pass fl) crate.functions passes
+ in
+ let crate = { crate with functions } in
+ log#ldebug
+ (lazy ("After pre-passes:\n" ^ Print.Crate.crate_to_string crate ^ "\n"));
+ crate