diff options
Diffstat (limited to '')
| -rw-r--r-- | compiler/InterpreterBorrows.ml | 102 | ||||
| -rw-r--r-- | compiler/InterpreterLoops.ml | 206 | 
2 files changed, 222 insertions, 86 deletions
| diff --git a/compiler/InterpreterBorrows.ml b/compiler/InterpreterBorrows.ml index bb843714..eb3b9898 100644 --- a/compiler/InterpreterBorrows.ml +++ b/compiler/InterpreterBorrows.ml @@ -1748,9 +1748,8 @@ let convert_value_to_abstractions (abs_kind : V.abs_kind) (can_end : bool)    let push_abs (r_id : T.RegionId.id) (avalues : V.typed_avalue list) : unit =      if avalues = [] then ()      else -      (* Reverse the list of avalues *) -      let avalues = List.rev avalues in -      (* Create the abs *) +      (* Create the abs - note that we keep the order of the avalues as it is +         (unlike the environments) *)        let abs =          {            V.abs_id = C.fresh_abstraction_id (); @@ -2121,6 +2120,11 @@ type merge_duplicates_funcs = {  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 = +  log#ldebug +    (lazy +      ("merge_abstractions_aux:\n- abs0:\n" ^ abs_to_string ctx abs0 +     ^ "\n\n- abs1:\n" ^ abs_to_string ctx abs1)); +    (* Check that the abstractions are destructured *)    if !Config.check_invariants then (      let destructure_shared_values = true in @@ -2277,48 +2281,67 @@ let merge_abstractions_aux (abs_kind : V.abs_kind) (can_end : bool)          raise (Failure "Unreachable")    in -  let borrows_loans = List.append borrows_loans0 borrows_loans1 in -  (* Iterate over all the borrows/loans ids found in teh abstractions *) +  (* Note that we first explore the borrows/loans of [abs1], because we +     want to merge *into* this abstraction, and as a consequence we want to +     preserve its structure as much as we can *) +  let borrows_loans = List.append borrows_loans1 borrows_loans0 in +  (* Iterate over all the borrows/loans ids found in the abstractions *)    List.iter      (fun bl ->        match bl with -      | BorrowId bid -> ( -          (* Check if the borrow has already been merged *) -          assert (not (borrow_is_merged bid)); -          set_borrow_as_merged bid; -          (* Check if we need to filter it *) -          match filter_bid bid with -          | None -> () -          | Some bid -> -              (* Lookup the contents *) -              let bc0 = V.BorrowId.Map.find_opt bid borrow_to_content0 in -              let bc1 = V.BorrowId.Map.find_opt bid borrow_to_content1 in -              (* Merge *) -              let av : V.typed_avalue = -                match (bc0, bc1) with -                | None, Some bc | Some bc, None -> ( -                    match bc with -                    | Concrete (_, _) -> -                        (* This can happen only in case of nested borrows - -                           a concrete borrow can only happen inside a shared -                           loan -                        *) -                        raise (Failure "Unreachable") -                    | Abstract (ty, bc) -> { V.value = V.ABorrow bc; ty }) -                | Some bc0, Some bc1 -> -                    assert (merge_funs <> None); -                    merge_g_borrow_contents bc0 bc1 -                | None, None -> raise (Failure "Unreachable") -              in -              push_avalue av) -      | LoanId bid -> ( +      | BorrowId bid -> +          log#ldebug +            (lazy +              ("merge_abstractions_aux: merging borrow " +             ^ V.BorrowId.to_string bid)); + +          (* Check if the borrow has already been merged - this can happen +             because we go through all the borrows/loans in [abs0] *then* +             all the borrows/loans in [abs1], and there may be duplicates +             between the two *) +          if borrow_is_merged bid then () +          else ( +            set_borrow_as_merged bid; +            (* Check if we need to filter it *) +            match filter_bid bid with +            | None -> () +            | Some bid -> +                (* Lookup the contents *) +                let bc0 = V.BorrowId.Map.find_opt bid borrow_to_content0 in +                let bc1 = V.BorrowId.Map.find_opt bid borrow_to_content1 in +                (* Merge *) +                let av : V.typed_avalue = +                  match (bc0, bc1) with +                  | None, Some bc | Some bc, None -> ( +                      match bc with +                      | Concrete (_, _) -> +                          (* This can happen only in case of nested borrows - +                             a concrete borrow can only happen inside a shared +                             loan +                          *) +                          raise (Failure "Unreachable") +                      | Abstract (ty, bc) -> { V.value = V.ABorrow bc; ty }) +                  | Some bc0, Some bc1 -> +                      assert (merge_funs <> None); +                      merge_g_borrow_contents bc0 bc1 +                  | None, None -> raise (Failure "Unreachable") +                in +                push_avalue av) +      | LoanId bid ->            if              (* Check if the loan has already been treated - it can happen -               because shared loans contain sets of borrows. +               for the same reason as for borrows, and also because shared +               loans contain sets of borrows (meaning that when taking care +               of one loan, we can merge several other loans at once).              *)              loan_is_merged bid            then () -          else +          else ( +            log#ldebug +              (lazy +                ("merge_abstractions_aux: merging loan " +               ^ V.BorrowId.to_string bid)); +              (* Check if we need to filter it *)              match filter_bid bid with              | None -> () @@ -2359,8 +2382,9 @@ let merge_abstractions_aux (abs_kind : V.abs_kind) (can_end : bool)                  push_avalue av))      borrows_loans; -  (* Note that we *don't* reverse the avalues *) -  let avalues = !avalues in +  (* Reverse the avalues (we visited the loans/borrows in order, but pushed +     new values at the beggining of the stack of avalues) *) +  let avalues = List.rev !avalues in    (* Filter the regions *) diff --git a/compiler/InterpreterLoops.ml b/compiler/InterpreterLoops.ml index 2fc474fc..70a5f1bf 100644 --- a/compiler/InterpreterLoops.ml +++ b/compiler/InterpreterLoops.ml @@ -237,9 +237,7 @@ let collapse_ctx (loop_id : V.LoopId.id)    log#ldebug      (lazy        ("collapse_ctx:\n\n- fixed_ids:\n" ^ show_ids_sets old_ids -     ^ "\n\n- ctx0:\n" -      ^ eval_ctx_to_string_no_filter ctx0 -      ^ "\n\n")); +     ^ "\n\n- ctx0:\n" ^ eval_ctx_to_string ctx0 ^ "\n\n"));    let abs_kind = V.Loop loop_id in    let can_end = false in @@ -250,9 +248,8 @@ let collapse_ctx (loop_id : V.LoopId.id)    let is_fresh_did (id : C.DummyVarId.id) : bool =      not (C.DummyVarId.Set.mem id old_ids.dids)    in -  (* Convert the dummy values to abstractions, and destructure all the new -     abstractions at the same time (note that when we convert values to -     abstractions, the resulting abstraction should be destructured) *) +  (* Convert the dummy values to abstractions (note that when we convert +     values to abstractions, the resulting abstraction should be destructured) *)    (* Note that we preserve the order of the dummy values: we replace them with       abstractions in place - this makes matching easier *)    let env = @@ -271,6 +268,12 @@ let collapse_ctx (loop_id : V.LoopId.id)                 else [ ee ])           ctx0.env)    in +  log#ldebug +    (lazy +      ("collapse_ctx: after converting values to abstractions:\n" +     ^ show_ids_sets old_ids ^ "\n\n- ctx:\n" +      ^ eval_ctx_to_string { ctx0 with env } +      ^ "\n\n"));    (* Explore all the *new* abstractions, and compute various maps *)    let explore (abs : V.abs) = is_fresh_abs_id abs.abs_id in @@ -591,17 +594,20 @@ module Match (M : Matcher) = struct          let bc =            match (bc0, bc1) with            | SharedBorrow (mv0, bid0), SharedBorrow (mv1, bid1) -> -              (* Not completely sure what to do with the meta-value. If a shared -                 symbolic value gets expanded in a branch, it may be simplified -                 (by being folded back to a symbolic value) upon doing the join, -                 which as a result would lead to code where it is considered as -                 mutable (which is sound). On the other hand, if we access a -                 symbolic value in a loop, the translated loop should take it as -                 input anyway, so maybe this actually leads to equivalent -                 code. +              (* Not completely sure what to do with the meta-value, as we use +                 it for the translation. TODO: remove meta-values from shared borrows? + +                 If a shared symbolic value gets expanded in a branch, it may be +                 simplified (by being folded back to a symbolic value) upon +                 doing the join, which as a result would lead to code where it +                 is considered as mutable (which is sound). On the other hand, +                 if we access a symbolic value in a loop, the translated loop +                 should take it as input anyway, so maybe this actually leads to +                 equivalent code.                *) +              assert (not (value_has_borrows ctx mv0.V.value)); +              assert (not (value_has_borrows ctx mv1.V.value));                let mv = match_rec mv0 mv1 in -              assert (not (value_has_borrows ctx mv.V.value));                let mv, bid = M.match_shared_borrows ty mv bid0 bid1 in                V.SharedBorrow (mv, bid)            | MutBorrow (bid0, bv0), MutBorrow (bid1, bv1) -> @@ -672,6 +678,13 @@ module Match (M : Matcher) = struct     *)    and match_typed_avalues (ctx : C.eval_ctx) (v0 : V.typed_avalue)        (v1 : V.typed_avalue) : V.typed_avalue = +    log#ldebug +      (lazy +        ("match_typed_avalues:\n- value0: " +        ^ typed_avalue_to_string ctx v0 +        ^ "\n- value1: " +        ^ typed_avalue_to_string ctx v1)); +      let match_rec = match_typed_values ctx in      let match_arec = match_typed_avalues ctx in      let ty = M.match_rtys v0.V.ty v1.V.ty in @@ -691,14 +704,19 @@ module Match (M : Matcher) = struct            { V.value; ty }          else (* Merge *)            M.match_distinct_aadts v0.V.ty av0 v1.V.ty av1 ty -    | ABottom, ABottom -> v0 +    | ABottom, ABottom -> mk_abottom ty +    | AIgnored, AIgnored -> mk_aignored ty      | ABorrow bc0, ABorrow bc1 -> (          match (bc0, bc1) with          | ASharedBorrow bid0, ASharedBorrow bid1 ->              M.match_ashared_borrows v0.V.ty bid0 v1.V.ty bid1 ty          | AMutBorrow (mv0, bid0, av0), AMutBorrow (mv1, bid1, av1) -> -            (* Not sure what to do with the meta-value *) -            let mv = match_rec mv0 mv1 in +            (* The meta-value should be the value consumed by the abstracion. +               This only makes sense if the abstraction was introduced because +               of a function call (we use it for the translation). +               TODO: make it optional? +            *) +            let mv = mk_bottom mv0.V.ty in              let av = match_arec av0 av1 in              M.match_amut_borrows v0.V.ty mv0 bid0 av0 v1.V.ty mv1 bid1 av1 ty mv                av @@ -1051,8 +1069,34 @@ let collapse_ctx_with_merge (loop_id : V.LoopId.id) (old_ids : ids_sets)    end in    let module JM = MakeJoinMatcher (S) in    let module M = Match (JM) in -  let merge_amut_borrows _ _ _ _ _ _ _ = raise (Failure "Unexpected") in -  let merge_ashared_borrows _ _ _ = raise (Failure "Unexpected") in +  (* TODO: why not simply call: M.match_type_avalues? (or move this code to +     MakeJoinMatcher?) *) +  let merge_amut_borrows id ty0 (mv0 : V.typed_value) child0 _ty1 +      (mv1 : V.typed_value) child1 = +    (* Sanity checks *) +    assert (is_aignored child0.V.value); +    assert (is_aignored child1.V.value); +    assert (mv0.V.ty = mv1.V.ty); + +    (* Same remarks as for [merge_amut_loans] *) +    let ty = ty0 in +    let child = child0 in +    let mv = mk_bottom mv0.ty in +    let value = V.ABorrow (V.AMutBorrow (mv, id, child)) in +    { V.value; ty } +  in + +  let merge_ashared_borrows id ty0 ty1 = +    (* Sanity checks *) +    assert (not (ty_has_borrows ctx.type_context.type_infos ty0)); +    assert (not (ty_has_borrows ctx.type_context.type_infos ty1)); + +    (* Same remarks as for [merge_amut_loans] *) +    let ty = ty0 in +    let value = V.ABorrow (V.ASharedBorrow id) in +    { V.value; ty } +  in +    let merge_amut_loans id ty0 child0 _ty1 child1 =      (* Sanity checks *)      assert (is_aignored child0.V.value); @@ -1327,6 +1371,7 @@ let join_ctxs (loop_id : V.LoopId.id) (fixed_ids : ids_sets) (ctx0 : C.eval_ctx)  (* Very annoying: functors only take modules as inputs... *)  module type MatchCheckEquivState = sig +  val ctx : C.eval_ctx    val rid_map : T.RegionId.InjSubst.t ref    val bid_map : V.BorrowId.InjSubst.t ref    val sid_map : V.SymbolicValueId.InjSubst.t ref @@ -1474,7 +1519,14 @@ module MakeCheckEquivMatcher (S : MatchCheckEquivState) = struct      let value = V.ALoan (V.AMutLoan (id, av)) in      { V.value; ty } -  let match_avalues _ _ = raise Distinct +  let match_avalues v0 v1 = +    log#ldebug +      (lazy +        ("avalues don't match:\n- v0: " +        ^ typed_avalue_to_string S.ctx v0 +        ^ "\n- v1: " +        ^ typed_avalue_to_string S.ctx v1)); +    raise Distinct  end  (** Compute whether two contexts are equivalent modulo an identifier substitution. @@ -1524,6 +1576,7 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx)    in    let module S : MatchCheckEquivState = struct +    let ctx = ctx0      let rid_map = rid_map      let bid_map = bid_map      let sid_map = sid_map @@ -1585,6 +1638,8 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx)      let _ = CEM.match_aidl original_parents0 original_parents1 in      let _ = CEM.match_rids regions0 regions1 in      let _ = CEM.match_rids ancestors_regions0 ancestors_regions1 in + +    log#ldebug (lazy "match_abstractions: matching values");      let _ =        List.map          (fun (v0, v1) -> M.match_typed_avalues ctx v0 v1) @@ -1598,7 +1653,15 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx)      log#ldebug        (lazy          ("ctxs_are_equivalent: match_envs:\n\n- fixed_ids:\n" -       ^ show_ids_sets fixed_ids ^ "\n\n- ctx0:\n" +       ^ show_ids_sets fixed_ids ^ "\n\n- rid_map: " +        ^ T.RegionId.InjSubst.show_t !rid_map +        ^ "\n- bid_map: " +        ^ V.BorrowId.InjSubst.show_t !bid_map +        ^ "\n- sid_map: " +        ^ V.SymbolicValueId.InjSubst.show_t !sid_map +        ^ "\n- aid_map: " +        ^ V.AbstractionId.InjSubst.show_t !aid_map +        ^ "\n\n- ctx0:\n"          ^ eval_ctx_to_string_no_filter { ctx0 with env = List.rev env0 }          ^ "\n\n- ctx1:\n"          ^ eval_ctx_to_string_no_filter { ctx1 with env = List.rev env1 } @@ -1608,8 +1671,8 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx)      | ( C.Var (C.DummyBinder b0, v0) :: env0',          C.Var (C.DummyBinder b1, v1) :: env1' ) ->          (* Two cases: the dummy value is an old value, in which case the bindings -           must be the same and we must join their values. Otherwise, it means we -           are not in the prefix *) +           must be the same and their values equal (and the borrows/loans/symbolic +           values they reference must be fixed) *)          if C.DummyVarId.Set.mem b0 fixed_ids.dids then (            (* Fixed values: the values must be equal *)            assert (b0 = b1); @@ -1648,7 +1711,7 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx)          (* Done *)          ()      | _ -> -        (* The elements don't match: we are not in the prefix anymore *) +        (* The elements don't match *)          raise Distinct    in @@ -1677,21 +1740,8 @@ let ctxs_are_equivalent (fixed_ids : ids_sets) (ctx0 : C.eval_ctx)      we return those updated environments, and the joined environment.   *)  let loop_join_origin_with_continue_ctxs (config : C.config) -    (loop_id : V.LoopId.id) (old_ctx : C.eval_ctx) (ctxl : C.eval_ctx list) : -    (C.eval_ctx * C.eval_ctx list) * C.eval_ctx = -  (* # Look for borrows and abstractions that exist in [old_ctx] and not in [ctxl]: -   * we need to end those *) -  (* Compute the sets of borrows and abstractions *) -  let old_ids = compute_context_ids old_ctx in -  let new_ids = compute_contexts_ids ctxl in -  let bids = V.BorrowId.Set.diff old_ids.bids new_ids.bids in -  let aids = V.AbstractionId.Set.diff old_ids.aids new_ids.aids in -  (* End those borrows and abstractions *) -  let old_ctx = InterpreterBorrows.end_borrows_no_synth config bids old_ctx in -  let old_ctx = -    InterpreterBorrows.end_abstractions_no_synth config aids old_ctx -  in - +    (loop_id : V.LoopId.id) (fixed_ids : ids_sets) (old_ctx : C.eval_ctx) +    (ctxl : C.eval_ctx list) : (C.eval_ctx * C.eval_ctx list) * C.eval_ctx =    (* # Join with the new contexts, one by one       For every context, we repeteadly attempt to join it with the current @@ -1702,7 +1752,7 @@ let loop_join_origin_with_continue_ctxs (config : C.config)    *)    let joined_ctx = ref old_ctx in    let rec join_one_aux (ctx : C.eval_ctx) : C.eval_ctx = -    match join_ctxs loop_id old_ids !joined_ctx ctx with +    match join_ctxs loop_id fixed_ids !joined_ctx ctx with      | Ok nctx ->          joined_ctx := nctx;          ctx @@ -1719,16 +1769,41 @@ let loop_join_origin_with_continue_ctxs (config : C.config)          join_one_aux ctx    in    let join_one (ctx : C.eval_ctx) : C.eval_ctx = +    log#ldebug +      (lazy +        ("loop_join_origin_with_continue_ctxs:join_one: initial ctx:\n" +       ^ eval_ctx_to_string ctx)); +      (* Destructure the abstractions introduced in the new context *) -    let ctx = destructure_new_abs loop_id old_ids.aids ctx in +    let ctx = destructure_new_abs loop_id fixed_ids.aids ctx in +    log#ldebug +      (lazy +        ("loop_join_origin_with_continue_ctxs:join_one: after destructure:\n" +       ^ eval_ctx_to_string ctx)); +      (* Collapse the context we want to add to the join *) -    let ctx = collapse_ctx loop_id None old_ids ctx in +    let ctx = collapse_ctx loop_id None fixed_ids ctx in +    log#ldebug +      (lazy +        ("loop_join_origin_with_continue_ctxs:join_one: after collapse:\n" +       ^ eval_ctx_to_string ctx)); +      (* Join the two contexts  *)      let ctx1 = join_one_aux ctx in +    log#ldebug +      (lazy +        ("loop_join_origin_with_continue_ctxs:join_one: after join:\n" +       ^ eval_ctx_to_string ctx1)); +      (* Collapse again - the join might have introduce abstractions we want         to merge with the others (note that those abstractions may actually         lead to borrows/loans duplications) *) -    joined_ctx := collapse_ctx_with_merge loop_id old_ids !joined_ctx; +    joined_ctx := collapse_ctx_with_merge loop_id fixed_ids !joined_ctx; +    log#ldebug +      (lazy +        ("loop_join_origin_with_continue_ctxs:join_one: after join-collapse:\n" +        ^ eval_ctx_to_string !joined_ctx)); +      (* Sanity check *)      if !Config.check_invariants then Invariants.check_invariants !joined_ctx;      (* Return *) @@ -1765,10 +1840,39 @@ let compute_loop_entry_fixed_point (config : C.config) (loop_id : V.LoopId.id)          (* We don't support nested loops for now *)          raise (Failure "Nested loops are not supported for now")    in + +  (* The fixed ids. They are the ids of the original ctx, after we ended +     the borrows/loans which end during the first loop iteration (we do +     one loop iteration, then set it to [Some]. +  *) +  let fixed_ids = ref None in +    (* Join the contexts at the loop entry *)    let join_ctxs (ctx0 : C.eval_ctx) : C.eval_ctx = +    (* If this is the first iteration, end the borrows/loans/abs which +       appear in ctx0 and not in the other contexts, then compute the +       set of fixed ids. This means those borrows/loans have to end +       in the loop, and we rather end them *before* the loop. *) +    let ctx0 = +      match !fixed_ids with +      | Some _ -> ctx0 +      | None -> +          let old_ids = compute_context_ids ctx0 in +          let new_ids = compute_contexts_ids !ctxs in +          let bids = V.BorrowId.Set.diff old_ids.bids new_ids.bids in +          let aids = V.AbstractionId.Set.diff old_ids.aids new_ids.aids in +          (* End those borrows and abstractions *) +          let ctx0 = InterpreterBorrows.end_borrows_no_synth config bids ctx0 in +          let ctx0 = +            InterpreterBorrows.end_abstractions_no_synth config aids ctx0 +          in +          fixed_ids := Some (compute_context_ids ctx0); +          ctx0 +    in + +    let fixed_ids = Option.get !fixed_ids in      let (ctx0', _), ctx1 = -      loop_join_origin_with_continue_ctxs config loop_id ctx0 !ctxs +      loop_join_origin_with_continue_ctxs config loop_id fixed_ids ctx0 !ctxs      in      ctxs := [];      if !ctx_upon_entry = None then ctx_upon_entry := Some ctx0'; @@ -1777,9 +1881,17 @@ let compute_loop_entry_fixed_point (config : C.config) (loop_id : V.LoopId.id)    (* 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 = +    (* 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 sids = V.SymbolicValueId.Set.empty in +    let { aids; bids; 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      ctxs_are_equivalent fixed_ids ctx1 ctx2    in | 
