diff options
author | Son Ho | 2022-02-08 20:07:37 +0100 |
---|---|---|
committer | Son Ho | 2022-02-08 20:07:37 +0100 |
commit | fb013997dda4c01fdc395ab52ba9dc3669f3d71a (patch) | |
tree | 9ba397bc7025c2219a50b7fe618751798d3f2c9f | |
parent | 311bbd7a5102a37b42414517310c5ca6913c4c65 (diff) |
Add PrePasses.ml which was forgotten in a previous commit
Diffstat (limited to '')
-rw-r--r-- | src/PrePasses.ml | 50 |
1 files changed, 50 insertions, 0 deletions
diff --git a/src/PrePasses.ml b/src/PrePasses.ml new file mode 100644 index 00000000..e9d2dad1 --- /dev/null +++ b/src/PrePasses.ml @@ -0,0 +1,50 @@ +(** This files contains passes we apply on the AST *before* calling the + (concrete/symbolic) interpreter on it + *) + +module T = Types +module V = Values +module E = Expressions +module C = Contexts +module A = CfimAst +module M = Modules +module L = Logging + +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 + following one: + + ``` + drop( *x ); // Illegal! Inserts a ⊥ under a borrow + *x = move ...; + ``` + + *) +let filter_drop_assigns (f : A.fun_def) : A.fun_def = + (* The visitor *) + let obj = + object (self) + inherit [_] A.map_statement as super + + method! visit_Sequence env st1 st2 = + match (st1, st2) with + | Drop p1, Assign (p2, _) -> + if p1 = p2 then self#visit_statement env st2 + else super#visit_Sequence env st1 st2 + | Drop p1, Sequence (Assign (p2, _), _) -> + if p1 = p2 then self#visit_statement env st2 + else super#visit_Sequence env st1 st2 + | _ -> super#visit_Sequence env st1 st2 + end + in + (* Map *) + let body = obj#visit_statement () f.body in + { f with body } + +let apply_passes (m : M.cfim_module) : M.cfim_module = + let functions = List.map filter_drop_assigns m.functions in + { m with functions } |