diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/Interpreter.ml | 124 |
1 files changed, 88 insertions, 36 deletions
diff --git a/src/Interpreter.ml b/src/Interpreter.ml index 2acabd4f..bc2209c3 100644 --- a/src/Interpreter.ml +++ b/src/Interpreter.ml @@ -1057,6 +1057,47 @@ let write_place_unwrap (config : C.config) (access : access_kind) (p : E.place) | Error _ -> failwith "Unreachable" | Ok env -> env +(** Compute an expanded ADT bottom value *) +let compute_expanded_bottom_adt_value (tyctx : T.type_def T.TypeDefId.vector) + (def_id : T.TypeDefId.id) (opt_variant_id : T.VariantId.id option) + (regions : T.erased_region list) (types : T.ety list) : V.typed_value = + (* Lookup the definition and check if it is an enumeration - it + should be an enumeration if and only if the projection element + is a field projection with *some* variant id. Retrieve the list + of fields at the same time. *) + let def = T.TypeDefId.nth tyctx def_id in + assert (List.length regions = T.RegionVarId.length def.T.region_params); + (* Compute the field types *) + let field_types = + Subst.type_def_get_instantiated_field_type def opt_variant_id types + in + (* Initialize the expanded value *) + let field_types = T.FieldId.vector_to_list field_types in + let fields = List.map (fun ty -> { V.value = Bottom; ty }) field_types in + let fields = T.FieldId.vector_of_list fields in + let av = + V.Adt + { + def_id; + variant_id = opt_variant_id; + regions; + types; + field_values = fields; + } + in + let ty = T.Adt (def_id, regions, types) in + { V.value = av; V.ty } + +(** Compute an expanded tuple bottom value *) +let compute_expanded_bottom_tuple_value (field_types : T.ety list) : + V.typed_value = + (* Generate the field values *) + let fields = List.map (fun ty -> { V.value = Bottom; ty }) field_types in + let fields = T.FieldId.vector_of_list fields in + let v = V.Tuple fields in + let ty = T.Tuple field_types in + { V.value = v; V.ty } + (** Auxiliary helper to expand [Bottom] values. During compilation, rustc desaggregates the ADT initializations. The @@ -1077,6 +1118,8 @@ let write_place_unwrap (config : C.config) (access : access_kind) (p : E.place) to, say, [Cons Bottom Bottom] (note that field projection contains information about which variant we should project to, which is why we *can* set the variant index when writing one of its fields). + + TODO: rename to express the fact that this function uses a projection *) let expand_bottom_value (config : C.config) (access : access_kind) (tyctx : T.type_def T.TypeDefId.vector) (p : E.place) (remaining_pes : int) @@ -1101,46 +1144,15 @@ let expand_bottom_value (config : C.config) (access : access_kind) match (pe, ty) with | ( Field (ProjAdt (def_id, opt_variant_id), _), T.Adt (def_id', regions, types) ) -> - (* Lookup the definition and check if it is an enumeration - it - should be an enumeration if and only if the projection element - is a field projection with *some* variant id. Retrieve the list - of fields at the same time. *) assert (def_id = def_id'); - let def = T.TypeDefId.nth tyctx def_id in - let fields = - match (def.kind, opt_variant_id) with - | Struct fields, None -> fields - | Enum variants, Some variant_id -> - (* Retrieve the proper variant *) - let variant = T.VariantId.nth variants variant_id in - variant.fields - | _ -> failwith "Unreachable" - in - (* Initialize the expanded value *) - let fields = T.FieldId.vector_to_list fields in - let fields = - List.map - (fun f -> { V.value = Bottom; ty = T.erase_regions f.T.field_ty }) - fields - in - let fields = T.FieldId.vector_of_list fields in - V.Adt - { - def_id; - variant_id = opt_variant_id; - regions; - types; - field_values = fields; - } + compute_expanded_bottom_adt_value tyctx def_id opt_variant_id regions + types | Field (ProjTuple arity, _), T.Tuple tys -> assert (arity = List.length tys); (* Generate the field values *) - let fields = List.map (fun ty -> { V.value = Bottom; ty }) tys in - let fields = T.FieldId.vector_of_list fields in - V.Tuple fields + compute_expanded_bottom_tuple_value tys | _ -> failwith "Unreachable" in - let nv = { V.value = nv; ty } in (* Update the environment by inserting the expanded value at the proper place *) match write_place config access p' nv env with | Ok env -> env @@ -1816,6 +1828,41 @@ let prepare_lplace (config : C.config) (p : E.place) (ctx : C.eval_ctx) : let ctx3 = { ctx with env = env3 } in (ctx3, v) +(** Read the value at a given place. + As long as it is a loan, end the loan *) +let rec end_loan_exactly_at_place (config : C.config) (access : access_kind) + (p : E.place) (ctx : C.eval_ctx) : C.eval_ctx = + let v = read_place_unwrap config access p ctx.env in + match v.V.value with + | V.Loan (V.SharedLoan (bids, _)) -> + let env1 = end_borrows config Outer bids ctx.env in + let ctx1 = { ctx with env = env1 } in + end_loan_exactly_at_place config access p ctx1 + | V.Loan (V.MutLoan bid) -> + let env1 = end_borrow config Outer bid ctx.env in + let ctx1 = { ctx with env = env1 } in + end_loan_exactly_at_place config access p ctx1 + | _ -> ctx + +let set_discriminant (config : C.config) (ctx : C.eval_ctx) (p : E.place) + (variant_id : T.VariantId.id) : + (C.eval_ctx * statement_eval_res) eval_result = + (* Access the value *) + let access = Write in + let env1 = update_env_along_read_place config access p ctx.env in + let ctx1 = { ctx with env = env1 } in + let ctx2 = end_loan_exactly_at_place config access p ctx1 in + let v = read_place_unwrap config access p ctx2.env in + (* Update the value *) + match v.V.value with + | Adt av -> raise Unimplemented + | Bottom -> raise Unimplemented + | Symbolic _ -> + assert (config.mode = SymbolicMode); + (* TODO *) raise Unimplemented + | Concrete _ | Tuple _ | Borrow _ | Loan _ | Assumed _ -> + failwith "Unexpected value" + let rec eval_statement (config : C.config) (ctx : C.eval_ctx) (st : A.statement) : (C.eval_ctx * statement_eval_res) eval_result = match st with @@ -1834,7 +1881,8 @@ let rec eval_statement (config : C.config) (ctx : C.eval_ctx) (st : A.statement) | A.FakeRead p -> let ctx1, _ = prepare_rplace config Read p ctx in Ok (ctx1, Unit) - | A.SetDiscriminant (p, variant_id) -> raise Unimplemented + | A.SetDiscriminant (p, variant_id) -> + set_discriminant config ctx p variant_id | A.Drop p -> let ctx1, v = prepare_lplace config p ctx in let nv = { v with value = V.Bottom } in @@ -1848,9 +1896,13 @@ let rec eval_statement (config : C.config) (ctx : C.eval_ctx) (st : A.statement) | Concrete (Bool b) -> if b = assertion.expected then Ok (ctx1, Unit) else Error Panic | _ -> failwith "Expected a boolean") - | A.Call call -> raise Unimplemented + | A.Call call -> eval_function_call config ctx call | A.Panic -> Error Panic | A.Return -> Ok (ctx, Return) | A.Break i -> Ok (ctx, Break i) | A.Continue i -> Ok (ctx, Continue i) | A.Nop -> Ok (ctx, Unit) + +and eval_function_call (config : C.config) (ctx : C.eval_ctx) (call : A.call) : + (C.eval_ctx * statement_eval_res) eval_result = + raise Unimplemented |