diff --git a/src/kernel_internals/typing/cabs2cil.ml b/src/kernel_internals/typing/cabs2cil.ml index cccda3a9169be72d7c7d2202aec9827b2b477ada..8725e512b03f735eefc3107947a1c8d22ea29b51 100644 --- a/src/kernel_internals/typing/cabs2cil.ml +++ b/src/kernel_internals/typing/cabs2cil.ml @@ -841,6 +841,21 @@ let env : (string, envdata * location) H.t = H.create 307 (* We also keep a global environment. This is always a subset of the env *) let genv : (string, envdata * location) H.t = H.create 307 +let label_env = Datatype.String.Hashtbl.create 307 + +let add_label_env lab = + let add_if_absent v (d,_) map = + match d with + | EnvVar vi when not (Datatype.String.Map.mem v map) -> + Datatype.String.Map.add v vi map + | _ -> map + in + let lab_env = H.fold add_if_absent env Datatype.String.Map.empty in + Datatype.String.Hashtbl.add label_env lab lab_env + +let remove_label_env lab = + Datatype.String.Hashtbl.remove label_env lab + (* In the scope we keep the original name, so we can remove them from the * hash table easily *) type undoScope = @@ -1021,6 +1036,7 @@ let constrExprId = ref 0 let startFile () = + Datatype.String.Hashtbl.clear label_env; H.clear env; H.clear genv; H.clear alphaTable; @@ -3719,9 +3735,27 @@ struct let anonCompFieldName = anonCompFieldName let conditionalConversion = logicConditionalConversion let find_macro _ = raise Not_found - let find_var x = match H.find env x with - | EnvVar vi, _ -> cvar_to_lvar vi - | _ -> raise Not_found + let find_var ?label ~var = + let find_from_curr_env test = + match H.find env var with + | EnvVar vi, _ when test vi -> cvar_to_lvar vi + | _ -> raise Not_found + in + match label with + | None -> find_from_curr_env (fun _ -> true) + | Some "Here" | Some "Old" | Some "Post" -> + (* the last two labels can only be found in contracts and refer + to the pre/post state of the contracts: all local variables + in scope at current point are also in scope in the labels. *) + find_from_curr_env (fun _ -> true) + | Some "Pre" -> + find_from_curr_env (fun vi -> vi.vformal || vi.vglob) + | Some "Init" -> find_from_curr_env (fun vi -> vi.vglob) + | Some lab -> + cvar_to_lvar + (Datatype.String.Map.find var + (Datatype.String.Hashtbl.find label_env lab)) + let find_enum_tag x = match H.find env x with | EnvEnum item,_ -> dummy_exp (Const (CEnum item)), typeOf item.eival @@ -3764,6 +3798,8 @@ module Ltyping = Logic_typing.Make (C_logic_env) let startLoop iswhile = incr C_logic_env.nb_loop; + add_label_env "LoopEntry"; + add_label_env "LoopCurrent"; continues := (if iswhile then While (ref "") else NotWhile (ref "")) :: !continues; enter_break_env () @@ -3771,6 +3807,8 @@ let startLoop iswhile = let exitLoop () = decr C_logic_env.nb_loop; exit_break_env (); + remove_label_env "LoopEntry"; + remove_label_env "LoopCurrent"; match !continues with | [] -> Kernel.error ~once:true ~current:true "exit Loop not in a loop" | _ :: rest -> continues := rest @@ -9663,6 +9701,7 @@ and doStatement local_env (s : A.statement) : chunk = | A.LABEL (l, s, loc) -> let loc' = convLoc loc in CurrentLoc.set loc'; + add_label_env l; C_logic_env.add_current_label l; (* Lookup the label because it might have been locally defined *) let chunk = diff --git a/src/kernel_services/ast_queries/logic_typing.ml b/src/kernel_services/ast_queries/logic_typing.ml index 918ca0447b734454367bde0b04082b03bec3ff96..b0e2522c5bc025fbef5ad04970bb1257b2d2d8dc 100644 --- a/src/kernel_services/ast_queries/logic_typing.ml +++ b/src/kernel_services/ast_queries/logic_typing.ml @@ -326,6 +326,26 @@ module Lenv = struct *) } + let string_of_current_label env = + Extlib.opt_bind ( + function + | FormalLabel _ -> None + | BuiltinLabel Init -> Some "Init" + | BuiltinLabel Pre -> Some "Pre" + | BuiltinLabel Old -> Some "Old" + | BuiltinLabel Post -> Some "Post" + | BuiltinLabel Here -> Some "Here" + | BuiltinLabel LoopCurrent -> Some "LoopCurrent" + | BuiltinLabel LoopEntry -> Some "LoopEntry" + | StmtLabel s -> + (match + List.find_opt (function Label (_,_,b) -> b | _ -> false) !s.labels + with + | None -> None + | Some (Label (lab,_,_)) -> Some lab + | Some _ -> None)) + env.current_logic_label + let fresh_var env name kind typ = let name = let exists name = @@ -468,7 +488,7 @@ type typing_context = { anonCompFieldName : string; conditionalConversion : typ -> typ -> typ; find_macro : string -> lexpr; - find_var : string -> logic_var; + find_var : ?label:string -> var:string -> logic_var; find_enum_tag : string -> exp * typ; find_comp_field: compinfo -> string -> offset; find_type : type_namespace -> string -> typ; @@ -633,7 +653,7 @@ module Make val anonCompFieldName : string val conditionalConversion : typ -> typ -> typ val find_macro : string -> lexpr - val find_var : string -> logic_var + val find_var : ?label:string -> var:string -> logic_var val find_enum_tag : string -> exp * typ val find_comp_field: compinfo -> string -> offset val find_type : type_namespace -> string -> typ @@ -2512,7 +2532,8 @@ struct | _ -> old_val lv) with Not_found -> try - let info = ctxt.find_var x in + let label = Lenv.string_of_current_label env in + let info = ctxt.find_var ?label ~var:x in (match info.lv_origin with | Some lv -> check_current_label loc env; diff --git a/src/kernel_services/ast_queries/logic_typing.mli b/src/kernel_services/ast_queries/logic_typing.mli index a05bd2a0ebba3432098286cae6c3596702b3463c..11a48ce5cd5009fda29cba4d40cfbf4a8aa824c3 100644 --- a/src/kernel_services/ast_queries/logic_typing.mli +++ b/src/kernel_services/ast_queries/logic_typing.mli @@ -97,7 +97,11 @@ type typing_context = { anonCompFieldName : string; conditionalConversion : typ -> typ -> typ; find_macro : string -> Logic_ptree.lexpr; - find_var : string -> logic_var; + find_var : ?label:string -> var:string -> logic_var; + (** the label argument is a C label (obeying the restrictions + of which label can be present in a \at). If present, the scope for + searching local C variables is the one of the statement with + the corresponding label. *) find_enum_tag : string -> exp * typ; find_comp_field: compinfo -> string -> offset; find_type : type_namespace -> string -> typ; @@ -245,7 +249,8 @@ module Make val anonCompFieldName : string val conditionalConversion : typ -> typ -> typ val find_macro : string -> Logic_ptree.lexpr - val find_var : string -> logic_var + val find_var : ?label:string -> var:string -> logic_var + (** see corresponding field in {!Logic_typing.typing_context}. *) val find_enum_tag : string -> exp * typ val find_type : type_namespace -> string -> typ val find_comp_field: compinfo -> string -> offset diff --git a/src/plugins/aorai/data_for_aorai.ml b/src/plugins/aorai/data_for_aorai.ml index b2933c2f6e1ffc0be435368090607402e25e9095..f5faca352f87dd495392aca4076b3f42c1f17b8d 100644 --- a/src/plugins/aorai/data_for_aorai.ml +++ b/src/plugins/aorai/data_for_aorai.ml @@ -652,7 +652,7 @@ struct let conditionalConversion = Cabs2cil.logicConditionalConversion let is_loop () = false let find_macro _ = raise Not_found - let find_var _ = raise Not_found + let find_var ?label:_ ~var:_ = raise Not_found let find_enum_tag _ = raise Not_found (*let find_comp_type ~kind:_ _ = raise Not_found*) let find_comp_field info s =