summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorSon Ho2022-01-19 01:57:05 +0100
committerSon Ho2022-01-19 01:57:05 +0100
commit62a6cc50a80a499f39ba83a7810e2c94f595b421 (patch)
treecbf5f2f59ac59e11d75f11348d92d61b41e5e29b /src
parent2c93d514d6d62f8eccba47b1efefa2db1e56954e (diff)
Implement ty_has_borrow_under_mut
Diffstat (limited to 'src')
-rw-r--r--src/InterpreterBorrows.ml2
-rw-r--r--src/InterpreterUtils.ml2
-rw-r--r--src/TypesAnalysis.ml159
-rw-r--r--src/TypesUtils.ml7
4 files changed, 106 insertions, 64 deletions
diff --git a/src/InterpreterBorrows.ml b/src/InterpreterBorrows.ml
index 034d566a..6f469044 100644
--- a/src/InterpreterBorrows.ml
+++ b/src/InterpreterBorrows.ml
@@ -458,7 +458,7 @@ let give_back_symbolic_value (_config : C.config)
* meaning the given back value can't have borrows below mutable
* borrows. *)
let has_borrow_below_mut =
- ty_has_borrow_below_mut ctx.type_context.type_infos nsv.sv_ty
+ ty_has_borrow_under_mut ctx.type_context.type_infos nsv.sv_ty
in
assert (not has_borrow_below_mut);
(* We don't allow overwrites: we must thus not project over the
diff --git a/src/InterpreterUtils.ml b/src/InterpreterUtils.ml
index be899c08..c3fe9781 100644
--- a/src/InterpreterUtils.ml
+++ b/src/InterpreterUtils.ml
@@ -234,7 +234,7 @@ let value_has_ret_symbolic_value_with_borrow_under_mut (ctx : C.eval_ctx)
method! visit_symbolic_value _ s =
match s.sv_kind with
| V.FunCallRet ->
- if ty_has_borrow_below_mut ctx.type_context.type_infos s.sv_ty then
+ if ty_has_borrow_under_mut ctx.type_context.type_infos s.sv_ty then
raise Found
else ()
| V.SynthInput -> ()
diff --git a/src/TypesAnalysis.ml b/src/TypesAnalysis.ml
index acb0f963..5fa8fc15 100644
--- a/src/TypesAnalysis.ml
+++ b/src/TypesAnalysis.ml
@@ -3,7 +3,7 @@ open Modules
type subtype_info = {
under_borrow : bool; (** Are we inside a borrow? *)
- under_nested_borrows : bool; (** Are we inside nested borrows? *)
+ under_mut_borrow : bool; (** Are we inside a mut borrow? *)
}
[@@deriving show]
@@ -12,13 +12,20 @@ type type_param_info = subtype_info [@@deriving show]
type expl_info = subtype_info [@@deriving show]
-type 'p g_type_info = {
+type type_borrows_info = {
contains_static : bool;
(** Does the type (transitively) contains a static borrow? *)
contains_borrow : bool;
(** Does the type (transitively) contains a borrow? *)
contains_nested_borrows : bool;
(** Does the type (transitively) contains nested borrows? *)
+ contains_borrow_under_mut : bool;
+}
+[@@deriving show]
+
+type 'p g_type_info = {
+ borrows_info : type_borrows_info;
+ (** Various informations about the borrows *)
param_infos : 'p; (** Gives information about the type parameters *)
}
[@@deriving show]
@@ -27,7 +34,7 @@ type 'p g_type_info = {
type type_def_info = type_param_info list g_type_info [@@deriving show]
(** Information about a type definition. *)
-type ty_info = unit g_type_info [@@deriving show]
+type ty_info = type_borrows_info [@@deriving show]
(** Information about a type. *)
type partial_type_info = type_param_info list option g_type_info
@@ -39,46 +46,37 @@ type partial_type_info = type_param_info list option g_type_info
type type_infos = type_def_info TypeDefId.Map.t [@@deriving show]
-let expl_info_init = { under_borrow = false; under_nested_borrows = false }
+let expl_info_init = { under_borrow = false; under_mut_borrow = false }
-let initialize_g_type_info (param_infos : 'p) : 'p g_type_info =
+let type_borrows_info_init : type_borrows_info =
{
contains_static = false;
contains_borrow = false;
contains_nested_borrows = false;
- param_infos;
+ contains_borrow_under_mut = false;
}
+let initialize_g_type_info (param_infos : 'p) : 'p g_type_info =
+ { borrows_info = type_borrows_info_init; param_infos }
+
let initialize_type_def_info (def : type_def) : type_def_info =
- let param_info = { under_borrow = false; under_nested_borrows = false } in
+ let param_info = { under_borrow = false; under_mut_borrow = false } in
let param_infos = List.map (fun _ -> param_info) def.type_params in
initialize_g_type_info param_infos
let type_def_info_to_partial_type_info (info : type_def_info) :
partial_type_info =
- {
- contains_static = info.contains_static;
- contains_borrow = info.contains_borrow;
- contains_nested_borrows = info.contains_nested_borrows;
- param_infos = Some info.param_infos;
- }
+ { borrows_info = info.borrows_info; param_infos = Some info.param_infos }
let partial_type_info_to_type_def_info (info : partial_type_info) :
type_def_info =
{
- contains_static = info.contains_static;
- contains_borrow = info.contains_borrow;
- contains_nested_borrows = info.contains_nested_borrows;
+ borrows_info = info.borrows_info;
param_infos = Option.get info.param_infos;
}
let partial_type_info_to_ty_info (info : partial_type_info) : ty_info =
- {
- contains_static = info.contains_static;
- contains_borrow = info.contains_borrow;
- contains_nested_borrows = info.contains_nested_borrows;
- param_infos = ();
- }
+ info.borrows_info
let analyze_full_ty (r_is_static : 'r -> bool) (updated : bool ref)
(infos : type_infos) (ty_info : partial_type_info) (ty : 'r ty) :
@@ -91,19 +89,33 @@ let analyze_full_ty (r_is_static : 'r -> bool) (updated : bool ref)
else original
in
- let update_ty_info_from_expl_info (ty_info : partial_type_info)
- (expl_info : expl_info) : partial_type_info =
- (* Set `contains_borrow` *)
+ (* Update a partial_type_info, while registering if we actually performed an update *)
+ let update_ty_info (ty_info : partial_type_info)
+ (ty_b_info : type_borrows_info) : partial_type_info =
+ let original = ty_info.borrows_info in
+ let contains_static =
+ check_update_bool original.contains_static ty_b_info.contains_static
+ in
let contains_borrow =
- check_update_bool ty_info.contains_borrow expl_info.under_borrow
+ check_update_bool original.contains_borrow ty_b_info.contains_borrow
in
- (* Set `contains_nested_borrows` *)
let contains_nested_borrows =
- check_update_bool ty_info.contains_nested_borrows
- expl_info.under_nested_borrows
+ check_update_bool original.contains_nested_borrows
+ ty_b_info.contains_nested_borrows
in
- (* Update ty_info *)
- { ty_info with contains_borrow; contains_nested_borrows }
+ let contains_borrow_under_mut =
+ check_update_bool original.contains_borrow_under_mut
+ ty_b_info.contains_borrow_under_mut
+ in
+ let updated_borrows_info =
+ {
+ contains_static;
+ contains_borrow;
+ contains_nested_borrows;
+ contains_borrow_under_mut;
+ }
+ in
+ { ty_info with borrows_info = updated_borrows_info }
in
(* The recursive function which explores the type *)
@@ -122,12 +134,12 @@ let analyze_full_ty (r_is_static : 'r -> bool) (updated : bool ref)
check_update_bool param_info.under_borrow expl_info.under_borrow
in
(* Set `under_nested_borrows` *)
- let under_nested_borrows =
- check_update_bool param_info.under_nested_borrows
- expl_info.under_nested_borrows
+ let under_mut_borrow =
+ check_update_bool param_info.under_mut_borrow
+ expl_info.under_mut_borrow
in
(* Update param_info *)
- let param_info = { under_borrow; under_nested_borrows } in
+ let param_info = { under_borrow; under_mut_borrow } in
let param_infos =
TypeVarId.update_nth param_infos var_id param_info
in
@@ -136,18 +148,28 @@ let analyze_full_ty (r_is_static : 'r -> bool) (updated : bool ref)
| Array ty | Slice ty ->
(* Just dive in *)
analyze expl_info ty_info ty
- | Ref (r, rty, _) ->
+ | Ref (r, rty, rkind) ->
+ (* Update the type info *)
+ let contains_static = r_is_static r in
+ let contains_borrow = true in
+ let contains_nested_borrows = expl_info.under_borrow in
+ let contains_borrow_under_mut = expl_info.under_mut_borrow in
+ let ty_b_info =
+ {
+ contains_static;
+ contains_borrow;
+ contains_nested_borrows;
+ contains_borrow_under_mut;
+ }
+ in
+ let ty_info = update_ty_info ty_info ty_b_info in
(* Update the exploration info *)
let expl_info =
- { under_borrow = true; under_nested_borrows = expl_info.under_borrow }
+ {
+ under_borrow = true;
+ under_mut_borrow = expl_info.under_mut_borrow || rkind = Mut;
+ }
in
- (* Set `contains_static` *)
- let contains_static =
- check_update_bool ty_info.contains_static (r_is_static r)
- in
- let ty_info = { ty_info with contains_static } in
- (* Update the type info *)
- let ty_info = update_ty_info_from_expl_info ty_info expl_info in
(* Continue exploring *)
analyze expl_info ty_info rty
| Adt ((Tuple | Assumed Box), _, tys) ->
@@ -158,35 +180,54 @@ let analyze_full_ty (r_is_static : 'r -> bool) (updated : bool ref)
| Adt (AdtId adt_id, regions, tys) ->
(* Lookup the information for this type definition *)
let adt_info = TypeDefId.Map.find adt_id infos in
- (* Check if 'static appears *)
- let ty_info =
- List.fold_left
- (fun ty_info r ->
- {
- ty_info with
- contains_static =
- check_update_bool ty_info.contains_static (r_is_static r);
- })
- ty_info regions
+ (* Update the type info with the information from the adt *)
+ let ty_info = update_ty_info ty_info adt_info.borrows_info in
+ (* Check if 'static appears in the region parameters *)
+ let found_static = List.exists r_is_static regions in
+ let borrows_info = ty_info.borrows_info in
+ let borrows_info =
+ {
+ borrows_info with
+ contains_static =
+ check_update_bool borrows_info.contains_static found_static;
+ }
in
+ let ty_info = { ty_info with borrows_info } in
(* For every instantiated type parameter: update the exploration info
* then explore the type *)
let params_tys = List.combine adt_info.param_infos tys in
let ty_info =
List.fold_left
(fun ty_info (param_info, ty) ->
+ (* Update the type info *)
+ (* Below: we use only the information which we learn only
+ * by taking the type parameter into account. *)
+ let contains_static = false in
+ let contains_borrow = param_info.under_borrow in
+ let contains_nested_borrows =
+ expl_info.under_borrow && param_info.under_borrow
+ in
+ let contains_borrow_under_mut =
+ expl_info.under_mut_borrow && param_info.under_borrow
+ in
+ let ty_b_info =
+ {
+ contains_static;
+ contains_borrow;
+ contains_nested_borrows;
+ contains_borrow_under_mut;
+ }
+ in
+ let ty_info = update_ty_info ty_info ty_b_info in
(* Update the exploration info *)
let expl_info =
{
under_borrow =
expl_info.under_borrow || param_info.under_borrow;
- under_nested_borrows =
- expl_info.under_nested_borrows
- || param_info.under_nested_borrows;
+ under_mut_borrow =
+ expl_info.under_mut_borrow || param_info.under_mut_borrow;
}
in
- (* Propagate the updates to the ty info *)
- let ty_info = update_ty_info_from_expl_info ty_info expl_info in
(* Continue exploring *)
analyze expl_info ty_info ty)
ty_info params_tys
diff --git a/src/TypesUtils.ml b/src/TypesUtils.ml
index e34add63..86076469 100644
--- a/src/TypesUtils.ml
+++ b/src/TypesUtils.ml
@@ -112,9 +112,10 @@ let ty_has_nested_borrows (infos : TA.type_infos) (ty : 'r ty) : bool =
let info = TA.analyze_ty infos ty in
info.TA.contains_nested_borrows
-(** Retuns true if the type contains a borrow below a mutable borrow *)
-let ty_has_borrow_below_mut (infos : TA.type_infos) (ty : 'r ty) : bool =
- raise Errors.Unimplemented
+(** Retuns true if the type contains a borrow under a mutable borrow *)
+let ty_has_borrow_under_mut (infos : TA.type_infos) (ty : 'r ty) : bool =
+ let info = TA.analyze_ty infos ty in
+ info.TA.contains_borrow_under_mut
(** Check if a [ty] contains regions from a given set *)
let ty_has_regions_in_set (rset : RegionId.Set.t) (ty : rty) : bool =