diff options
Diffstat (limited to 'compiler')
-rw-r--r-- | compiler/PrePasses.ml | 111 |
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 |