summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSon Ho2022-11-30 23:28:37 +0100
committerSon HO2023-02-03 11:21:46 +0100
commit4412989abff237c566cad323a8a56cf08bb8e294 (patch)
treee2f5ae45b02996ab13ee5a536bfb9d5659917225
parent39ae214a7ebc2b833da2b759df13717e8a0dfeae (diff)
Improve merge_abstractions
-rw-r--r--compiler/InterpreterBorrows.ml72
-rw-r--r--compiler/InterpreterBorrows.mli13
-rw-r--r--compiler/Substitute.ml12
3 files changed, 91 insertions, 6 deletions
diff --git a/compiler/InterpreterBorrows.ml b/compiler/InterpreterBorrows.ml
index a8d50720..4e05dda7 100644
--- a/compiler/InterpreterBorrows.ml
+++ b/compiler/InterpreterBorrows.ml
@@ -2115,7 +2115,11 @@ type merge_duplicates_funcs = {
*)
}
-let merge_abstractions (abs_kind : V.abs_kind) (can_end : bool)
+(** Auxiliary function.
+
+ Merge two abstractions into one, without updating the context.
+ *)
+let merge_abstractions_aux (abs_kind : V.abs_kind) (can_end : bool)
(merge_funs : merge_duplicates_funcs option) (ctx : C.eval_ctx)
(abs0 : V.abs) (abs1 : V.abs) : V.abs =
(* Check that the abstractions are destructured *)
@@ -2359,6 +2363,8 @@ let merge_abstractions (abs_kind : V.abs_kind) (can_end : bool)
(* Reverse the values *)
let avalues = List.rev !avalues in
+ (* Filter the regions *)
+
(* Create the new abstraction *)
let abs_id = C.fresh_abstraction_id () in
(* Note that one of the two abstractions might a parent of the other *)
@@ -2384,7 +2390,71 @@ let merge_abstractions (abs_kind : V.abs_kind) (can_end : bool)
avalues;
}
in
+
(* Sanity check *)
if !Config.check_invariants then assert (abs_is_destructured true ctx abs);
(* Return *)
abs
+
+(** Merge the regions in a context to a single region *)
+let ctx_merge_regions (ctx : C.eval_ctx) (rid : T.RegionId.id)
+ (rids : T.RegionId.Set.t) : C.eval_ctx =
+ let rsubst x = if T.RegionId.Set.mem x rids then rid else x in
+ let merge_in_abs (abs : V.abs) : V.abs =
+ let avalues =
+ List.map (Substitute.typed_avalue_subst_rids rsubst) abs.V.avalues
+ in
+ let regions = T.RegionId.Set.diff abs.V.regions rids in
+ let ancestors_regions = T.RegionId.Set.diff abs.V.ancestors_regions rids in
+ { abs with V.avalues; regions; ancestors_regions }
+ in
+
+ let env =
+ List.map
+ (fun ee ->
+ match ee with
+ | C.Abs abs -> C.Abs (merge_in_abs abs)
+ | Var _ | Frame -> ee)
+ ctx.env
+ in
+ { ctx with C.env }
+
+let merge_abstractions (abs_kind : V.abs_kind) (can_end : bool)
+ (merge_funs : merge_duplicates_funcs option) (ctx : C.eval_ctx)
+ (abs_id0 : V.AbstractionId.id) (abs_id1 : V.AbstractionId.id) :
+ C.eval_ctx * V.AbstractionId.id =
+ (* Lookup the abstractions *)
+ let abs0 = C.ctx_lookup_abs ctx abs_id0 in
+ let abs1 = C.ctx_lookup_abs ctx abs_id1 in
+
+ (* Merge them *)
+ let nabs = merge_abstractions_aux abs_kind can_end merge_funs ctx abs0 abs1 in
+
+ (* Update the environment: replace the first abstraction with the result of the merge,
+ remove the second abstraction *)
+ let ctx = fst (C.ctx_subst_abs ctx abs_id1 nabs) in
+ let ctx = fst (C.ctx_remove_abs ctx abs_id0) in
+
+ (* Merge all the regions from the abstraction into one (the first - i.e., the
+ one with the smallest id). Note that we need to do this in the whole
+ environment, as those regions may be referenced as ancestor regions by
+ the other abstractions, and may be present in symbolic values, etc. (this
+ is not the case if there are no nested borrows, but we anticipate).
+ *)
+ let ctx =
+ let regions = nabs.regions in
+ (* Pick the first region id (this is the smallest) *)
+ let rid = T.RegionId.Set.min_elt regions in
+ (* If there is only one region, do nothing *)
+ if T.RegionId.Set.cardinal regions = 1 then ctx
+ else
+ let rids = T.RegionId.Set.remove rid regions in
+ ctx_merge_regions ctx rid rids
+ in
+
+ (* Sanity check *)
+ (* Sanity check *)
+ if !Config.check_invariants then Invariants.check_invariants ctx;
+
+ (* Return *)
+ (ctx, nabs.abs_id)
diff --git a/compiler/InterpreterBorrows.mli b/compiler/InterpreterBorrows.mli
index 01ce206a..53297bc7 100644
--- a/compiler/InterpreterBorrows.mli
+++ b/compiler/InterpreterBorrows.mli
@@ -231,14 +231,17 @@ type merge_duplicates_funcs = {
actually simply performs some sort of a union.
- [ctx]
- - [abs0]
- - [abs1]
+ - [abs_id0]
+ - [abs_id1]
+
+ We return the updated context as well as the id of the new abstraction which
+ results from the merge.
*)
val merge_abstractions :
V.abs_kind ->
bool ->
merge_duplicates_funcs option ->
C.eval_ctx ->
- V.abs ->
- V.abs ->
- V.abs
+ V.AbstractionId.id ->
+ V.AbstractionId.id ->
+ C.eval_ctx * V.AbstractionId.id
diff --git a/compiler/Substitute.ml b/compiler/Substitute.ml
index 415f4d59..8348424c 100644
--- a/compiler/Substitute.ml
+++ b/compiler/Substitute.ml
@@ -456,3 +456,15 @@ let abs_subst_ids (rsubst : T.RegionId.id -> T.RegionId.id)
(bsubst : V.BorrowId.id -> V.BorrowId.id)
(asubst : V.AbstractionId.id -> V.AbstractionId.id) (x : V.abs) : V.abs =
(subst_ids_visitor rsubst rvsubst tsubst ssubst bsubst asubst)#visit_abs x
+
+let typed_avalue_subst_rids (rsubst : T.RegionId.id -> T.RegionId.id)
+ (x : V.typed_avalue) : V.typed_avalue =
+ let asubst _ = raise (Failure "Unreachable") in
+ (subst_ids_visitor rsubst
+ (fun x -> x)
+ (fun x -> x)
+ (fun x -> x)
+ (fun x -> x)
+ asubst)
+ #visit_typed_avalue
+ x