diff options
-rw-r--r-- | compiler/Interpreter.ml | 4 | ||||
-rw-r--r-- | compiler/InterpreterLoops.ml | 686 | ||||
-rw-r--r-- | compiler/InterpreterUtils.ml | 25 |
3 files changed, 635 insertions, 80 deletions
diff --git a/compiler/Interpreter.ml b/compiler/Interpreter.ml index d8a99d80..4c1bc047 100644 --- a/compiler/Interpreter.ml +++ b/compiler/Interpreter.ml @@ -270,11 +270,13 @@ let evaluate_function_symbolic (synthesize : bool) (* Put everything together *) S.synthesize_forward_end fwd_e back_el else None + | EndEnterLoop -> failwith "Unimplemented" + | EndContinue -> failwith "Unimplemented" | Panic -> (* Note that as we explore all the execution branches, one of * the executions can lead to a panic *) if synthesize then Some SA.Panic else None - | _ -> + | Unit | Break _ | Continue _ -> raise (Failure ("evaluate_function_symbolic failed on: " ^ name_to_string ())) in diff --git a/compiler/InterpreterLoops.ml b/compiler/InterpreterLoops.ml index f8ac0fbd..18a8f1f0 100644 --- a/compiler/InterpreterLoops.ml +++ b/compiler/InterpreterLoops.ml @@ -1274,11 +1274,13 @@ let join_ctxs (loop_id : V.LoopId.id) (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) must be the same and we must join their values. Otherwise, it means we are not in the prefix anymore *) if C.DummyVarId.Set.mem b0 fixed_ids.dids then ( - (* Still in the prefix: the values must match *) + (* Still in the prefix: match the values *) assert (b0 = b1); - assert (v0 = v1); + let b = b0 in + let v = M.match_typed_values ctx v0 v1 in + let var = C.Var (C.DummyBinder b, v) in (* Continue *) - var0 :: join_prefixes env0' env1') + var :: join_prefixes env0' env1') else (* Not in the prefix anymore *) join_suffixes env0 env1 | ( (C.Var (C.VarBinder b0, v0) as var0) :: env0', @@ -1372,15 +1374,39 @@ let join_ctxs (loop_id : V.LoopId.id) (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) } with ValueMatchFailure e -> Error e -(* Very annoying: functors only take modules as inputs... *) +(** See {!MakeCheckEquivMatcher}. + + Very annoying: functors only take modules as inputs... + *) module type MatchCheckEquivState = sig + (** [true] if we check equivalence between contexts, [false] if we match + a source context with a target context. *) + val check_equiv : bool + val ctx : C.eval_ctx val rid_map : T.RegionId.InjSubst.t ref + + (** Substitution for the loan and borrow ids - used only if [check_equiv] is true *) val bid_map : V.BorrowId.InjSubst.t ref + + (** Substitution for the borrow ids - used only if [check_equiv] is false *) + val borrow_id_map : V.BorrowId.InjSubst.t ref + + (** Substitution for the loans ids - used only if [check_equiv] is false *) + val loan_id_map : V.BorrowId.InjSubst.t ref + val sid_map : V.SymbolicValueId.InjSubst.t ref + val sid_to_value_map : V.typed_value V.SymbolicValueId.Map.t ref val aid_map : V.AbstractionId.InjSubst.t ref end +(** An auxiliary matcher that we use for two purposes: + - to check if two contexts are equivalent modulo id substitution (i.e., + alpha equivalent) (see {!ctxs_are_equivalent}). + - to compute a mapping between the borrows and the symbolic values in a + target context to the values and borrows in a source context (see + {!match_ctx_with_target}). + *) module MakeCheckEquivMatcher (S : MatchCheckEquivState) = struct module MkGetSetM (Id : Identifiers.Id) = struct module Inj = Id.InjSubst @@ -1415,7 +1441,8 @@ module MakeCheckEquivMatcher (S : MatchCheckEquivState) = struct module GetSetRid = MkGetSetM (T.RegionId) let match_rid = GetSetRid.match_e S.rid_map - let match_ridl = GetSetRid.match_el S.rid_map + + (* let match_ridl = GetSetRid.match_el S.rid_map *) let match_rids = GetSetRid.match_es S.rid_map module GetSetBid = MkGetSetM (V.BorrowId) @@ -1425,11 +1452,52 @@ module MakeCheckEquivMatcher (S : MatchCheckEquivState) = struct let match_bidl = GetSetBid.match_el S.bid_map let match_bids = GetSetBid.match_es S.bid_map + let get_borrow_id = + if S.check_equiv then get_bid else GetSetBid.get S.borrow_id_map + + let match_borrow_id = + if S.check_equiv then match_bid else GetSetBid.match_e S.borrow_id_map + + let match_borrow_idl = + if S.check_equiv then match_bidl else GetSetBid.match_el S.borrow_id_map + + let match_borrow_ids = + if S.check_equiv then match_bids else GetSetBid.match_es S.borrow_id_map + + let get_loan_id = + if S.check_equiv then get_bid else GetSetBid.get S.loan_id_map + + let match_loan_id = + if S.check_equiv then match_bid else GetSetBid.match_e S.loan_id_map + + let match_loan_idl = + if S.check_equiv then match_bidl else GetSetBid.match_el S.loan_id_map + + let match_loan_ids = + if S.check_equiv then match_bids else GetSetBid.match_es S.loan_id_map + module GetSetSid = MkGetSetM (V.SymbolicValueId) - let match_sid = GetSetSid.match_e S.sid_map - let match_sidl = GetSetSid.match_el S.sid_map - let match_sids = GetSetSid.match_es S.sid_map + (* If we check for equivalence, we map sids to sids, otherwise we map sids + to values. *) + let match_sid = + if S.check_equiv then GetSetSid.match_e S.sid_map + else fun _ -> raise (Failure "Unexpected") + + let match_value_with_sid (v : V.typed_value) (id : V.SymbolicValueId.id) : + unit = + (* Even when we don't use it, the sids map contains the fixed ids: check + that we are not trying to map a fixed id. *) + assert (not (V.SymbolicValueId.InjSubst.mem id !S.sid_map)); + + (* Check that we are not overriding a binding *) + assert (not (V.SymbolicValueId.Map.mem id !S.sid_to_value_map)); + + (* Add the mapping *) + S.sid_to_value_map := V.SymbolicValueId.Map.add id v !S.sid_to_value_map + + (* let match_sidl = GetSetSid.match_el S.sid_map *) + (* let match_sids = GetSetSid.match_es S.sid_map *) module GetSetAid = MkGetSetM (V.AbstractionId) @@ -1462,39 +1530,51 @@ module MakeCheckEquivMatcher (S : MatchCheckEquivState) = struct let match_shared_borrows (_ty : T.ety) (bid0 : V.borrow_id) (bid1 : V.borrow_id) : V.borrow_id = - let bid = match_bid bid0 bid1 in + let bid = match_borrow_id bid0 bid1 in bid let match_mut_borrows (_ty : T.ety) (bid0 : V.borrow_id) (_bv0 : V.typed_value) (bid1 : V.borrow_id) (_bv1 : V.typed_value) (bv : V.typed_value) : V.borrow_id * V.typed_value = - let bid = match_bid bid0 bid1 in + let bid = match_borrow_id bid0 bid1 in (bid, bv) let match_shared_loans (_ : T.ety) (ids0 : V.loan_id_set) (ids1 : V.loan_id_set) (sv : V.typed_value) : V.loan_id_set * V.typed_value = - let ids = match_bids ids0 ids1 in + let ids = match_loan_ids ids0 ids1 in (ids, sv) let match_mut_loans (_ : T.ety) (bid0 : V.loan_id) (bid1 : V.loan_id) : V.loan_id = - match_bid bid0 bid1 + match_loan_id bid0 bid1 let match_symbolic_values (sv0 : V.symbolic_value) (sv1 : V.symbolic_value) : V.symbolic_value = - let id0 = sv0.sv_id in let id1 = sv1.sv_id in - let sv_id = match_sid id0 id1 in - let sv_ty = match_rtys sv0.V.sv_ty sv1.V.sv_ty in - let sv_kind = - if sv0.V.sv_kind = sv1.V.sv_kind then sv0.V.sv_kind else raise Distinct - in - { V.sv_id; sv_ty; sv_kind } + if S.check_equiv then + let id0 = sv0.sv_id in + let sv_id = match_sid id0 id1 in + let sv_ty = match_rtys sv0.V.sv_ty sv1.V.sv_ty in + let sv_kind = + if sv0.V.sv_kind = sv1.V.sv_kind then sv0.V.sv_kind else raise Distinct + in + { V.sv_id; sv_ty; sv_kind } + else ( + (* Update the binding for the target symbolic value *) + match_value_with_sid (mk_typed_value_from_symbolic_value sv0) id1; + (* Return - the returned value is not used, so we can return whatever *) + sv1) - let match_symbolic_with_other (_left : bool) (_sv : V.symbolic_value) - (_v : V.typed_value) : V.typed_value = - raise Distinct + let match_symbolic_with_other (left : bool) (sv : V.symbolic_value) + (v : V.typed_value) : V.typed_value = + if S.check_equiv then raise Distinct + else ( + assert (not left); + (* Update the binding for the target symbolic value *) + match_value_with_sid v sv.sv_id; + (* Return - the returned value is not used, so we can return whatever *) + v) let match_bottom_with_other (_left : bool) (_v : V.typed_value) : V.typed_value = @@ -1503,22 +1583,22 @@ module MakeCheckEquivMatcher (S : MatchCheckEquivState) = struct let match_distinct_aadts _ _ _ _ _ = raise Distinct let match_ashared_borrows _ty0 bid0 _ty1 bid1 ty = - let bid = match_bid bid0 bid1 in + let bid = match_borrow_id bid0 bid1 in let value = V.ABorrow (V.ASharedBorrow bid) in { V.value; ty } let match_amut_borrows _ty0 bid0 _av0 _ty1 bid1 _av1 ty av = - let bid = match_bid bid0 bid1 in + let bid = match_borrow_id bid0 bid1 in let value = V.ABorrow (V.AMutBorrow (bid, av)) in { V.value; ty } let match_ashared_loans _ty0 ids0 _v0 _av0 _ty1 ids1 _v1 _av1 ty v av = - let bids = match_bids ids0 ids1 in + let bids = match_loan_ids ids0 ids1 in let value = V.ALoan (V.ASharedLoan (bids, v, av)) in { V.value; ty } let match_amut_loans _ty0 id0 _av0 _ty1 id1 _av1 ty av = - let id = match_bid id0 id1 in + let id = match_loan_id id0 id1 in let value = V.ALoan (V.AMutLoan (id, av)) in { V.value; ty } @@ -1532,6 +1612,19 @@ module MakeCheckEquivMatcher (S : MatchCheckEquivState) = struct raise Distinct end +(** See {!match_ctxs} *) +type ids_maps = { + aid_map : V.AbstractionId.InjSubst.t; + bid_map : V.BorrowId.InjSubst.t; + (** Substitution for the loan and borrow ids *) + borrow_id_map : V.BorrowId.InjSubst.t; (** Substitution for the borrow ids *) + loan_id_map : V.BorrowId.InjSubst.t; (** Substitution for the loan ids *) + rid_map : T.RegionId.InjSubst.t; + sid_map : V.SymbolicValueId.InjSubst.t; + sid_to_value_map : V.typed_value V.SymbolicValueId.Map.t; +} +[@@deriving show] + (** Compute whether two contexts are equivalent modulo an identifier substitution. [fixed_ids]: ids for which we force the mapping to be the identity. @@ -1540,9 +1633,15 @@ end have the same prefixes (we check it). See the explanations for {!join_ctxs}. TODO: explanations. + + [check_equiv]: if [true], check if the two contexts are equivalent. + If [false], compute a mapping from the first context to the second context, + in the sense of [match_ctx_with_target]. + + We return an optional ids map: [Some] if the match succeeded, [None] otherwise. *) -let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) - (ctx1 : C.eval_ctx) : bool = +let match_ctxs (check_equiv : bool) (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) + (ctx1 : C.eval_ctx) : ids_maps option = log#ldebug (lazy ("ctxs_are_equivalent:\n\n- fixed_ids:\n" ^ show_ids_sets fixed_ids @@ -1553,36 +1652,52 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) ^ "\n\n")); (* Initialize the maps and instantiate the matcher *) + let module IdMap (Id : Identifiers.Id) = struct + let mk_map_ref (ids : Id.Set.t) : Id.InjSubst.t ref = + ref + (Id.InjSubst.of_list (List.map (fun x -> (x, x)) (Id.Set.elements ids))) + end in let rid_map = - ref - (T.RegionId.InjSubst.of_list - (List.map (fun x -> (x, x)) (T.RegionId.Set.elements fixed_ids.rids))) + let module IdMap = IdMap (T.RegionId) in + IdMap.mk_map_ref fixed_ids.rids in let bid_map = - ref - (V.BorrowId.InjSubst.of_list - (List.map (fun x -> (x, x)) (V.BorrowId.Set.elements fixed_ids.bids))) + let module IdMap = IdMap (V.BorrowId) in + IdMap.mk_map_ref fixed_ids.bids + in + let borrow_id_map = + let module IdMap = IdMap (V.BorrowId) in + IdMap.mk_map_ref fixed_ids.borrow_ids + in + let loan_id_map = + let module IdMap = IdMap (V.BorrowId) in + IdMap.mk_map_ref fixed_ids.loan_ids in let aid_map = - ref - (V.AbstractionId.InjSubst.of_list - (List.map - (fun x -> (x, x)) - (V.AbstractionId.Set.elements fixed_ids.aids))) + let module IdMap = IdMap (V.AbstractionId) in + IdMap.mk_map_ref fixed_ids.aids in let sid_map = - ref - (V.SymbolicValueId.InjSubst.of_list - (List.map - (fun x -> (x, x)) - (V.SymbolicValueId.Set.elements fixed_ids.sids))) + let module IdMap = IdMap (V.SymbolicValueId) in + IdMap.mk_map_ref fixed_ids.sids + in + (* In case we don't try to check equivalence but want to compute a mapping + from a source context to a target context, we use a map from symbolic + value ids to values (rather than to ids). + *) + let sid_to_value_map : V.typed_value V.SymbolicValueId.Map.t ref = + ref V.SymbolicValueId.Map.empty in let module S : MatchCheckEquivState = struct + let check_equiv = check_equiv let ctx = ctx0 let rid_map = rid_map let bid_map = bid_map + let borrow_id_map = borrow_id_map + let loan_id_map = loan_id_map let sid_map = sid_map + let sid_to_value_map = sid_to_value_map let aid_map = aid_map end in let module CEM = MakeCheckEquivMatcher (S) in @@ -1592,9 +1707,10 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) (* Small utility: check that ids are fixed/mapped to themselves *) let ids_are_fixed (ids : ids_sets) : bool = - let { aids; bids; dids; rids; sids } = ids in + let { aids; bids = _; borrow_ids; loan_ids; dids; rids; sids } = ids in V.AbstractionId.Set.subset aids fixed_ids.aids - && V.BorrowId.Set.subset bids fixed_ids.bids + && V.BorrowId.Set.subset borrow_ids fixed_ids.borrow_ids + && V.BorrowId.Set.subset loan_ids fixed_ids.loan_ids && C.DummyVarId.Set.subset dids fixed_ids.dids && T.RegionId.Set.subset rids fixed_ids.rids && V.SymbolicValueId.Set.subset sids fixed_ids.sids @@ -1603,7 +1719,8 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) (* We need to pick a context for some functions like [match_typed_values]: the context is only used to lookup module data, so we can pick whichever we want. - TODO: this is not very clean. Maybe we should just carry this data around. + TODO: this is not very clean. Maybe we should just carry the relevant data + (i.e.e, not the whole context) around. *) let ctx = ctx0 in @@ -1682,7 +1799,7 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) assert (v0 = v1); (* The ids present in the left value must be fixed *) let ids = compute_typed_value_ids v0 in - assert (ids_are_fixed ids); + assert ((not S.check_equiv) || ids_are_fixed ids); (* Continue *) match_envs env0' env1') else @@ -1702,7 +1819,7 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) assert (abs0 = abs1); (* Their ids must be fixed *) let ids = compute_abs_ids abs0 in - assert (ids_are_fixed ids); + assert ((not S.check_equiv) || ids_are_fixed ids); (* Continue *) match_envs env0' env1') else ( @@ -1732,8 +1849,33 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) in match_envs env0 env1; - true - with Distinct -> false + let maps = + { + aid_map = !aid_map; + bid_map = !bid_map; + borrow_id_map = !borrow_id_map; + loan_id_map = !loan_id_map; + rid_map = !rid_map; + sid_map = !sid_map; + sid_to_value_map = !sid_to_value_map; + } + in + Some maps + with Distinct -> None + +(** Compute whether two contexts are equivalent modulo an identifier substitution. + + [fixed_ids]: ids for which we force the mapping to be the identity. + + We also take advantage of the structure of the environments, which should + have the same prefixes (we check it). See the explanations for {!join_ctxs}. + + TODO: explanations. + *) +let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx) + (ctx1 : C.eval_ctx) : bool = + let check_equivalent = true in + Option.is_some (match_ctxs check_equivalent fixed_ids ctx0 ctx1) (** Join the context at the entry of the loop with the contexts upon reentry (upon reaching the [Continue] statement - the goal is to compute a fixed @@ -1818,8 +1960,11 @@ let loop_join_origin_with_continue_ctxs (config : C.config) (* # Return *) ((old_ctx, ctxl), !joined_ctx) +(** Compute a fixed-point for the context at the entry of the loop. + We also return the sets of fixed ids. + *) let compute_loop_entry_fixed_point (config : C.config) (loop_id : V.LoopId.id) - (eval_loop_body : st_cm_fun) (ctx0 : C.eval_ctx) : C.eval_ctx = + (eval_loop_body : st_cm_fun) (ctx0 : C.eval_ctx) : C.eval_ctx * ids_sets = (* The continuation for when we exit the loop - we register the environments upon loop *reentry*, and synthesize nothing by returning [None] @@ -1893,20 +2038,25 @@ let compute_loop_entry_fixed_point (config : C.config) (loop_id : V.LoopId.id) ctx1 in (* Check if two contexts are equivalent - modulo alpha conversion on the - existentially quantified borrows/abstractions/symbolic values *) - let equiv_ctxs (ctx1 : C.eval_ctx) (ctx2 : C.eval_ctx) : bool = + existentially quantified borrows/abstractions/symbolic values. + *) + let compute_fixed_ids (ctx1 : C.eval_ctx) (ctx2 : C.eval_ctx) : ids_sets = (* Compute the set of fixed ids - for the symbolic ids, we compute the intersection of ids between the original environment and [ctx1] and [ctx2] *) let fixed_ids = compute_context_ids (Option.get !ctx_upon_entry) in - let { aids; bids; dids; rids; sids } = fixed_ids in + let { aids; bids; borrow_ids; loan_ids; dids; rids; sids } = fixed_ids in let fixed_ids1 = compute_context_ids ctx1 in let fixed_ids2 = compute_context_ids ctx2 in let sids = V.SymbolicValueId.Set.inter sids (V.SymbolicValueId.Set.inter fixed_ids1.sids fixed_ids2.sids) in - let fixed_ids = { aids; bids; dids; rids; sids } in + let fixed_ids = { aids; bids; borrow_ids; loan_ids; dids; rids; sids } in + fixed_ids + in + let equiv_ctxs (ctx1 : C.eval_ctx) (ctx2 : C.eval_ctx) : bool = + let fixed_ids = compute_fixed_ids ctx1 ctx2 in ctxs_are_equivalent fixed_ids ctx1 ctx2 in let max_num_iter = Config.loop_fixed_point_max_num_iters in @@ -1935,22 +2085,400 @@ let compute_loop_entry_fixed_point (config : C.config) (loop_id : V.LoopId.id) (* Check if we reached a fixed point: if not, iterate *) if equiv_ctxs ctx ctx1 then ctx1 else compute_fixed_point ctx1 (i - 1) in - compute_fixed_point ctx0 max_num_iter + let fp = compute_fixed_point ctx0 max_num_iter in + let fixed_ids = compute_fixed_ids (Option.get !ctx_upon_entry) fp in + (fp, fixed_ids) + +(** Split an environment between the fixed abstractions, values, etc. and + the new abstractions, values, etc. + + Returns: (fixed, new abs, new dummies) + *) +let ctx_split_fixed_new (fixed_ids : ids_sets) (ctx : C.eval_ctx) : + C.env * V.abs list * V.typed_value list = + let is_fresh_did (id : C.DummyVarId.id) : bool = + not (C.DummyVarId.Set.mem id fixed_ids.dids) + in + let is_fresh_abs_id (id : V.AbstractionId.id) : bool = + not (V.AbstractionId.Set.mem id fixed_ids.aids) + in + (* Filter the new abstractions and dummy variables (there shouldn't be any new dummy variable + though) in the target context *) + let is_fresh (ee : C.env_elem) : bool = + match ee with + | C.Var (VarBinder _, _) | C.Frame -> false + | C.Var (DummyBinder bv, _) -> not (is_fresh_did bv) + | C.Abs abs -> is_fresh_abs_id abs.abs_id + in + let new_eel, filt_env = List.partition is_fresh ctx.env in + let is_abs ee = match ee with C.Abs _ -> true | _ -> false in + let new_absl, new_dummyl = List.partition is_abs new_eel in + let new_absl = + List.map + (fun ee -> + match ee with C.Abs abs -> abs | _ -> raise (Failure "Unreachable")) + new_absl + in + let new_dummyl = + List.map + (fun ee -> + match ee with + | C.Var (DummyBinder _, v) -> v + | _ -> raise (Failure "Unreachable")) + new_dummyl + in + (filt_env, new_absl, new_dummyl) + +type borrow_loan_corresp = { + borrow_to_loan_id_map : V.BorrowId.InjSubst.t; + loan_to_borrow_id_map : V.BorrowId.InjSubst.t; +} + +let ids_sets_empty_borrows_loans (ids : ids_sets) : ids_sets = + let { aids; bids = _; borrow_ids = _; loan_ids = _; dids; rids; sids } = + ids + in + let empty = V.BorrowId.Set.empty in + let ids = + { + aids; + bids = empty; + borrow_ids = empty; + loan_ids = empty; + dids; + rids; + sids; + } + in + ids -let match_ctx_with_target (config : C.config) (tgt_ctx : C.eval_ctx) : cm_fun = +(** For the abstractions in the fixed point, compute the correspondance between + the borrows ids and the loans ids, if we want to introduce equivalent + identify abstractions (i.e., abstractions which do nothing - the input + borrows are exactly the output loans). + + TODO: detailed explanations + *) +let compute_fixed_point_id_correspondance (fixed_ids : ids_sets) + (src_ctx : C.eval_ctx) (tgt_ctx : C.eval_ctx) : borrow_loan_corresp = + log#ldebug + (lazy + ("compute_fixed_point_id_correspondance:\n\n- fixed_ids:\n" + ^ show_ids_sets fixed_ids ^ "\n\n- src_ctx:\n" ^ eval_ctx_to_string src_ctx + ^ "\n\n- tgt_ctx:\n" ^ eval_ctx_to_string tgt_ctx ^ "\n\n")); + + let filt_src_env, _, _ = ctx_split_fixed_new fixed_ids src_ctx in + let filt_src_ctx = { src_ctx with env = filt_src_env } in + let filt_tgt_env, new_absl, _ = ctx_split_fixed_new fixed_ids tgt_ctx in + let filt_tgt_ctx = { tgt_ctx with env = filt_tgt_env } in + + log#ldebug + (lazy + ("compute_fixed_point_id_correspondance:\n\n- fixed_ids:\n" + ^ show_ids_sets fixed_ids ^ "\n\n- filt_src_ctx:\n" + ^ eval_ctx_to_string filt_src_ctx + ^ "\n\n- filt_tgt_ctx:\n" + ^ eval_ctx_to_string filt_tgt_ctx + ^ "\n\n")); + + (* Match the source context and the filtered target context *) + let check_equiv = false in + let maps = + let fixed_ids = ids_sets_empty_borrows_loans fixed_ids in + Option.get (match_ctxs check_equiv fixed_ids filt_tgt_ctx filt_src_ctx) + in + + log#ldebug + (lazy + ("compute_fixed_point_id_correspondance:\n\n- tgt_to_src_maps:\n" + ^ show_ids_maps maps ^ "\n\n")); + + let src_to_tgt_borrow_map = + V.BorrowId.Map.of_list + (List.map + (fun (x, y) -> (y, x)) + (V.BorrowId.InjSubst.bindings maps.borrow_id_map)) + in + + (* Sanity check: for every abstraction, the target loans and borrows are mapped + to the same set of source loans and borrows *) + List.iter + (fun abs -> + let ids = compute_abs_ids abs in + (* Map the *loan* ids (we just match the corresponding *loans* ) *) + let loan_ids = + V.BorrowId.Set.map + (fun x -> V.BorrowId.InjSubst.find x maps.borrow_id_map) + ids.loan_ids + in + (* Check that the loan and borrows are related *) + assert (ids.borrow_ids = loan_ids)) + new_absl; + + (* For every target abstraction: + - go through the tgt borrows + - for every tgt borrow, find the corresponding src borrow + - from there, find the corresponding tgt loan + *) + let tgt_borrow_to_loan = ref V.BorrowId.InjSubst.empty in + let visit_tgt = + object + inherit [_] V.iter_abs + + method! visit_borrow_id _ id = + (* Find the target borrow *) + let tgt_borrow_id = V.BorrowId.Map.find id src_to_tgt_borrow_map in + (* Update the map *) + tgt_borrow_to_loan := + V.BorrowId.InjSubst.add id tgt_borrow_id !tgt_borrow_to_loan + end + in + List.iter (visit_tgt#visit_abs ()) new_absl; + + (* Compute the map from loan to borrows *) + let tgt_loan_to_borrow = + V.BorrowId.InjSubst.of_list + (List.map + (fun (x, y) -> (y, x)) + (V.BorrowId.InjSubst.bindings !tgt_borrow_to_loan)) + in + + (* Return *) + { + borrow_to_loan_id_map = !tgt_borrow_to_loan; + loan_to_borrow_id_map = tgt_loan_to_borrow; + } + +(** Match a context with a target context. + + This is used to compute application of loop translations: we use this match + to compute the mapping from symbolic values from the target environment + to values in the original environment (this allows us to compute the inputs + of the call to the loop translation). + + We only match the variables (while checking that the fixed dummy variables + and the fixed abstractions are equal - we ignore the newly introduced + abstractions and dummy variables). + + [is_loop_entry]: [true] if first entry into the loop, [false] if re-entry + (i.e., continue). + *) +let match_ctx_with_target (config : C.config) (loop_id : V.LoopId.id) + (is_loop_entry : bool) (fp_bl_maps : borrow_loan_corresp) + (fixed_ids : ids_sets) (tgt_ctx : C.eval_ctx) : st_cm_fun = fun cf src_ctx -> + (* Debug *) + log#ldebug + (lazy + ("match_ctx_with_target:\n" ^ "\n- src_ctx: " ^ eval_ctx_to_string src_ctx + ^ "\n- tgt_ctx: " ^ eval_ctx_to_string tgt_ctx)); + (* We first reorganize [src_ctx] so that we can match it with [tgt_ctx] *) - (* First, collect all the borrows and abstractions which are in [ctx] - and not in [tgt_ctx]: we need to end them *) - let src_ids = compute_context_ids src_ctx in - let tgt_ids = compute_context_ids tgt_ctx in - let bids = V.BorrowId.Set.diff src_ids.bids tgt_ids.bids in - let aids = V.AbstractionId.Set.diff src_ids.aids tgt_ids.aids in - (* End those borrows and abstractions *) - let cc = InterpreterBorrows.end_borrows config bids in - let cc = comp cc (InterpreterBorrows.end_abstractions config aids) in - (* TODO *) - raise (Failure "Unimplemented") + let rec cf_reorganize_src : cm_fun = + fun cf src_ctx -> + (* Collect fixed values in the source and target contexts: end the loans in the + source context which don't appear in the target context *) + let filt_src_env, _, _ = ctx_split_fixed_new fixed_ids src_ctx in + let filt_tgt_env, _, _ = ctx_split_fixed_new fixed_ids tgt_ctx in + + (* Remove the abstractions *) + let filter (ee : C.env_elem) : bool = + match ee with Var _ -> true | Abs _ | Frame -> false + in + let filt_src_env = List.filter filter filt_src_env in + let filt_tgt_env = List.filter filter filt_tgt_env in + + (* Match the values to check if there are loans to eliminate *) + + (* We need to pick a context for some functions like [match_typed_values]: + the context is only used to lookup module data, so we can pick whichever + we want. + TODO: this is not very clean. Maybe we should just carry this data around. + *) + let ctx = src_ctx in + + let nabs = ref [] in + + let module S : MatchJoinState = struct + (* The context is only used to lookup module data: we can pick whichever we want *) + let ctx = ctx + let loop_id = loop_id + let nabs = nabs + end in + let module JM = MakeJoinMatcher (S) in + let module M = Match (JM) in + try + let _ = + List.iter + (fun (var0, var1) -> + match (var0, var1) with + | C.Var (C.DummyBinder b0, v0), C.Var (C.DummyBinder b1, v1) -> + assert (b0 = b1); + let _ = M.match_typed_values ctx v0 v1 in + () + | C.Var (C.VarBinder b0, v0), C.Var (C.VarBinder b1, v1) -> + assert (b0 = b1); + let _ = M.match_typed_values ctx v0 v1 in + () + | _ -> raise (Failure "Unexpected")) + (List.combine filt_src_env filt_tgt_env) + in + (* No exception was thrown: continue *) + cf src_ctx + with ValueMatchFailure e -> + (* Exception: end the corresponding borrows, and continue *) + let cc = + match e with + | LoanInLeft bid -> InterpreterBorrows.end_borrow config bid + | LoansInLeft bids -> InterpreterBorrows.end_borrows config bids + | AbsInLeft _ | AbsInRight _ | LoanInRight _ | LoansInRight _ -> + raise (Failure "Unexpected") + in + comp cc cf_reorganize_src cf src_ctx + in + + (* Introduce the "identity" abstractions for the loop reentry. + + Match the src and target contexts so as to compute how to map the borrows + from the target context to the borrows in the source context. + + Substitute the *loans* in the new abstractions of the target context + (the abstractions of the fixed point) to properly link those abstraction: + we introduce *identity* abstractions (the loans are equal to the borrows): + we substitute the loans and introduce fresh ids for the borrows, symbolic + values, etc. + *) + let cf_introduce_loop_fp_abs : m_fun = + fun src_ctx -> + (* Match the src and target contexts *) + let filt_src_env, _, _ = ctx_split_fixed_new fixed_ids src_ctx in + let filt_tgt_env, new_absl, new_dummyl = + ctx_split_fixed_new fixed_ids tgt_ctx + in + assert (new_dummyl = []); + let filt_src_ctx = { src_ctx with env = filt_src_env } in + let filt_tgt_ctx = { tgt_ctx with env = filt_tgt_env } in + + let check_equiv = false in + let tgt_to_src_maps = + let fixed_ids = ids_sets_empty_borrows_loans fixed_ids in + Option.get (match_ctxs check_equiv fixed_ids filt_tgt_ctx filt_src_ctx) + in + let src_to_tgt_borrow_map = + V.BorrowId.Map.of_list + (List.map + (fun (x, y) -> (y, x)) + (V.BorrowId.InjSubst.bindings tgt_to_src_maps.borrow_id_map)) + in + + (* Debug *) + log#ldebug + (lazy + ("match_ctx_with_target:cf_introduce_loop_fp_abs:" + ^ "\n\n- filt_src_ctx: " + ^ eval_ctx_to_string filt_src_ctx + ^ "\n\n- filt_tgt_ctx: " + ^ eval_ctx_to_string filt_tgt_ctx + ^ "\n\n- tgt_to_src_maps: " + ^ show_ids_maps tgt_to_src_maps)); + + (* Update the borrows and symbolic ids in the source context *) + let tgt_fresh_borrows_map = ref V.BorrowId.Map.empty in + let visit_src = + object + inherit [_] C.map_eval_ctx + + method! visit_borrow_id _ id = + (* Map the borrow, if it needs to be mapped *) + if + V.BorrowId.InjSubst.Set.mem id + (V.BorrowId.InjSubst.elements tgt_to_src_maps.borrow_id_map) + then ( + let tgt_id = V.BorrowId.Map.find id src_to_tgt_borrow_map in + let nid = C.fresh_borrow_id () in + tgt_fresh_borrows_map := + V.BorrowId.Map.add tgt_id nid !tgt_fresh_borrows_map; + nid) + else id + end + in + let src_ctx = visit_src#visit_eval_ctx () src_ctx in + + (* Rem.: we don't update the symbolic values. It is not necessary + because there shouldn't be any symbolic value containing borrows. + + Rem.: we will need to do something about the symbolic values in the + abstractions and in the *variable bindings* once we allow symbolic + values containing borrows to not be eagerly expanded. + *) + assert Config.greedy_expand_symbolics_with_borrows; + + (* Update the borrows and loans in the abstractions of the target context *) + let region_id_map = ref T.RegionId.Map.empty in + let get_rid rid = + match T.RegionId.Map.find_opt rid !region_id_map with + | Some rid -> rid + | None -> + let nid = C.fresh_region_id () in + region_id_map := T.RegionId.Map.add rid nid !region_id_map; + nid + in + let visit_tgt = + object + inherit [_] C.map_eval_ctx + + method! visit_borrow_id _ bid = + (* Lookup the id of the loan corresponding to this borrow *) + let tgt_lid = + V.BorrowId.InjSubst.find bid fp_bl_maps.borrow_to_loan_id_map + in + (* Lookup the src borrow id to which this borrow was mapped *) + let src_bid = + V.BorrowId.InjSubst.find tgt_lid tgt_to_src_maps.borrow_id_map + in + src_bid + + method! visit_loan_id _ id = + (* Map the borrow - rem.: we mapped the borrows *in the values*, + meaning we know how to map the *corresponding loans in the + abstractions* *) + V.BorrowId.Map.find id !tgt_fresh_borrows_map + + method! visit_symbolic_value_id _ _ = C.fresh_symbolic_value_id () + method! visit_abstraction_id _ _ = C.fresh_abstraction_id () + method! visit_region_id _ id = get_rid id + end + in + let new_absl = List.map (visit_tgt#visit_abs ()) new_absl in + + (* Add the abstractions from the target context to the source context *) + let nenv = + List.append (List.map (fun abs -> C.Abs abs) new_absl) src_ctx.env + in + let src_ctx = { src_ctx with env = nenv } in + + log#ldebug + (lazy + ("match_ctx_with_target:cf_introduce_loop_fp_abs:\n- result ctx:" + ^ eval_ctx_to_string src_ctx)); + + (* Sanity check *) + if !Config.check_invariants then + Invariants.check_borrowed_values_invariant src_ctx; + + (* End all the borrows which appear in the *new* abstractions *) + let new_borrows = + V.BorrowId.Set.of_list + (List.map snd (V.BorrowId.Map.bindings !tgt_fresh_borrows_map)) + in + let cc = InterpreterBorrows.end_borrows config new_borrows in + + (* Continue *) + cc (cf (if is_loop_entry then EndEnterLoop else EndContinue)) src_ctx + in + + (* Compose and continue *) + cf_reorganize_src cf_introduce_loop_fp_abs src_ctx (** Evaluate a loop in concrete mode *) let eval_loop_concrete (eval_loop_body : st_cm_fun) : st_cm_fun = @@ -2006,7 +2534,7 @@ let eval_loop_symbolic (config : C.config) (eval_loop_body : st_cm_fun) : (* Compute a fresh loop id *) let loop_id = C.fresh_loop_id () in (* Compute the fixed point at the loop entrance *) - let fp_ctx = + let fp_ctx, fixed_ids = compute_loop_entry_fixed_point config loop_id eval_loop_body ctx in @@ -2016,13 +2544,18 @@ let eval_loop_symbolic (config : C.config) (eval_loop_body : st_cm_fun) : ("eval_loop_symbolic:\nInitial context:\n" ^ eval_ctx_to_string ctx ^ "\n\nFixed point:\n" ^ eval_ctx_to_string fp_ctx)); - failwith "Unimplemented"; - (* Synthesize the end of the function - we simply match the context at the loop entry with the fixed point: in the synthesized code, the function will end with a call to the loop translation *) - let end_expr = match_ctx_with_target config fp_ctx (cf EndEnterLoop) ctx in + let fp_bl_corresp = + compute_fixed_point_id_correspondance fixed_ids ctx fp_ctx + in + let end_expr = + match_ctx_with_target config loop_id true fp_bl_corresp fixed_ids fp_ctx cf + ctx + in + (* Synthesize the loop body by evaluating it, with the continuation for after the loop starting at the *fixed point*, but with a special treatment for the [Break] and [Continue] cases *) @@ -2037,7 +2570,11 @@ let eval_loop_symbolic (config : C.config) (eval_loop_body : st_cm_fun) : | Continue i -> (* We don't support nested loops for now *) assert (i = 0); - match_ctx_with_target config fp_ctx (cf EndContinue) ctx + let fp_bl_corresp = + compute_fixed_point_id_correspondance fixed_ids ctx fp_ctx + in + match_ctx_with_target config loop_id false fp_bl_corresp fixed_ids + fp_ctx cf ctx | Unit | EndEnterLoop | EndContinue -> (* For why we can't get [Unit], see the comments inside {!eval_loop_concrete}. For [EndEnterLoop] and [EndContinue]: we don't support nested loops for now. @@ -2045,6 +2582,7 @@ let eval_loop_symbolic (config : C.config) (eval_loop_body : st_cm_fun) : raise (Failure "Unreachable") in let loop_expr = eval_loop_body cf_loop fp_ctx in + (* Put together *) S.synthesize_loop end_expr loop_expr diff --git a/compiler/InterpreterUtils.ml b/compiler/InterpreterUtils.ml index 4a628118..f5932d47 100644 --- a/compiler/InterpreterUtils.ml +++ b/compiler/InterpreterUtils.ml @@ -270,6 +270,9 @@ let value_has_loans_or_borrows (ctx : C.eval_ctx) (v : V.value) : bool = type ids_sets = { aids : V.AbstractionId.Set.t; bids : V.BorrowId.Set.t; + (** All the borrow/loan ids. TODO: rename to [blids] *) + borrow_ids : V.BorrowId.Set.t; (** Only the borrow ids *) + loan_ids : V.BorrowId.Set.t; (** Only the loan ids *) dids : C.DummyVarId.Set.t; rids : T.RegionId.Set.t; sids : V.SymbolicValueId.Set.t; @@ -278,24 +281,36 @@ type ids_sets = { let compute_ids () = let bids = ref V.BorrowId.Set.empty in + let borrow_ids = ref V.BorrowId.Set.empty in + let loan_ids = ref V.BorrowId.Set.empty in let aids = ref V.AbstractionId.Set.empty in let dids = ref C.DummyVarId.Set.empty in let rids = ref T.RegionId.Set.empty in let sids = ref V.SymbolicValueId.Set.empty in let get_ids () = - { aids = !aids; bids = !bids; dids = !dids; rids = !rids; sids = !sids } + { + aids = !aids; + bids = !bids; + borrow_ids = !borrow_ids; + loan_ids = !loan_ids; + dids = !dids; + rids = !rids; + sids = !sids; + } in let obj = object inherit [_] C.iter_eval_ctx method! visit_dummy_var_id _ did = dids := C.DummyVarId.Set.add did !dids - method! visit_borrow_id _ id = bids := V.BorrowId.Set.add id !bids + + method! visit_borrow_id _ id = + bids := V.BorrowId.Set.add id !bids; + borrow_ids := V.BorrowId.Set.add id !borrow_ids method! visit_loan_id _ id = - (* Actually, this is not necessary because all loans have a - corresponding borrow *) - bids := V.BorrowId.Set.add id !bids + bids := V.BorrowId.Set.add id !bids; + loan_ids := V.BorrowId.Set.add id !loan_ids method! visit_abstraction_id _ id = aids := V.AbstractionId.Set.add id !aids |