diff --git a/src/kernel_internals/typing/mergecil.ml b/src/kernel_internals/typing/mergecil.ml index cbbebbd4f3cb1d61062a3dab45df1ba60f389090..1ef46b685b7ea640fac086d71229c8f65a397055 100644 --- a/src/kernel_internals/typing/mergecil.ml +++ b/src/kernel_internals/typing/mergecil.ml @@ -2201,9 +2201,22 @@ class renameInlineVisitorClass = object(self) end let renameInlinesVisitor = new renameInlineVisitorClass -type context = Toplevel | InAxiomatic | InModule +type context = Toplevel of global | InAxiomatic | InModule -let rec logic_annot_pass2 context g a = +let errorContext context toplevel = + Kernel.abort ~current:true "%s not allowed inside %s" toplevel context + +let checkContext ~toplevel = function + | Toplevel _ -> () + | InAxiomatic -> errorContext "axiomatic" toplevel + | InModule -> errorContext "module" toplevel + +let pushGlobal ?toplevel = function + | Toplevel g -> mergePushGlobals (visitCilGlobal renameVisitor g) + | InAxiomatic -> Option.iter (errorContext "axiomatic") toplevel + | InModule -> Option.iter (errorContext "module") toplevel + +let rec logic_annot_pass2 context a = let open Current_loc.Operators in let<> UpdatedCurrentLoc = Cil_datatype.Global_annotation.loc a in match a with @@ -2211,8 +2224,7 @@ let rec logic_annot_pass2 context g a = begin match LogicMerging.findReplacement true lfEq !currentFidx li with | None -> - if context = Toplevel then - mergePushGlobals (visitCilGlobal renameVisitor g); + pushGlobal context ; Logic_utils.add_logic_function li; | Some _ -> () (* FIXME: should we perform same actions @@ -2222,27 +2234,23 @@ let rec logic_annot_pass2 context g a = begin match PlainMerging.findReplacement true ltEq !currentFidx t.lt_name with | None -> - if context = Toplevel then - mergePushGlobals (visitCilGlobal renameVisitor g); - let def = - (PlainMerging.find_eq_table ltEq (!currentFidx,t.lt_name)).ndata - in + pushGlobal context ; + let node = PlainMerging.find_eq_table ltEq (!currentFidx,t.lt_name) in + let def = node.ndata in Logic_env.add_logic_type t.lt_name def; - (match def.lt_def with - | Some (LTsum l) -> - List.iter (fun c -> Logic_env.add_logic_ctor c.ctor_name c) l - | Some (LTsyn _) - | None -> () - ) + begin + match def.lt_def with + | Some (LTsum l) -> + List.iter (fun c -> Logic_env.add_logic_ctor c.ctor_name c) l + | Some (LTsyn _) | None -> () + end | Some _ -> () end | Dinvariant (li, _) -> begin match LogicMerging.findReplacement true lfEq !currentFidx li with | None -> - if context <> Toplevel then Kernel.abort ~current:true - "nested axiomatics are not allowed in ACSL"; - mergePushGlobals (visitCilGlobal renameVisitor g); + pushGlobal ~toplevel:"invariant" context ; Logic_utils.add_logic_function (LogicMerging.find_eq_table lfEq (!currentFidx,li)).ndata | Some _ -> () @@ -2251,9 +2259,7 @@ let rec logic_annot_pass2 context g a = begin match LogicMerging.findReplacement true lfEq !currentFidx n with | None -> - let g = visitCilGlobal renameVisitor g in - if context = Toplevel then - mergePushGlobals g; + pushGlobal context ; Logic_utils.add_logic_function (LogicMerging.find_eq_table lfEq (!currentFidx,n)).ndata | Some _ -> () @@ -2284,9 +2290,8 @@ let rec logic_annot_pass2 context g a = (!currentFidx,(mf'.mi_name,mf'.mi_base_type)) my_node'; end; - if context = Toplevel then begin - mergePushGlobals [GAnnot (Dmodel_annot(mf',l),l)]; - end; + checkContext ~toplevel:"model annotation" context ; + mergePushGlobal (GAnnot (Dmodel_annot(mf',l),l)) ; Logic_env.add_model_field (ModelMerging.find_eq_table mfEq (!currentFidx,(mf'.mi_name,mf'.mi_base_type))).ndata; @@ -2295,9 +2300,7 @@ let rec logic_annot_pass2 context g a = | Dlemma (n, _, _, _, _, _) -> begin match PlainMerging.findReplacement true llEq !currentFidx n with - None -> - if context = Toplevel then - mergePushGlobals (visitCilGlobal renameVisitor g) + None -> pushGlobal context | Some _ -> () end | Dvolatile(vi, rd, wr, attr, loc) -> @@ -2344,8 +2347,9 @@ let rec logic_annot_pass2 context g a = let no_drop, full_representative, only_reads, only_writes = List.fold_left check_one_location (true,[],[],[]) vi in - if no_drop then mergePushGlobals (visitCilGlobal renameVisitor g) + if no_drop then pushGlobal ~toplevel:"volatile" context else begin + checkContext ~toplevel:"volatile" context; push_volatile full_representative rd wr; if Option.is_some rd then push_volatile only_reads rd None; if Option.is_some wr then push_volatile only_writes None wr @@ -2353,40 +2357,27 @@ let rec logic_annot_pass2 context g a = | Daxiomatic(n, l, _, _) -> begin match PlainMerging.findReplacement true laEq !currentFidx n with - None -> - begin match context with - | Toplevel -> () - | InAxiomatic -> - Kernel.abort ~current:true - "nested axiomatics are not allowed in ACSL"; - | InModule -> - Kernel.abort ~current:true - "mixed axiomatics and modules are not allowed in ACSL"; - end ; - mergePushGlobals (visitCilGlobal renameVisitor g); - List.iter (logic_annot_pass2 InAxiomatic g) l + | None -> + pushGlobal context ; + List.iter (logic_annot_pass2 InAxiomatic) l | Some _ -> () end | Dmodule(n, l, _, _, _) -> begin match PlainMerging.findReplacement true laEq !currentFidx n with None -> - begin match context with - | Toplevel | InModule -> () - | InAxiomatic -> - Kernel.abort ~current:true - "mixed axiomatics and modules are not allowed in ACSL"; - end ; - mergePushGlobals (visitCilGlobal renameVisitor g); - List.iter (logic_annot_pass2 InModule g) l + pushGlobal context ; + List.iter (logic_annot_pass2 InModule) l | Some _ -> () end | Dextended(ext, _, _) -> - (match ExtMerging.findReplacement true extEq !currentFidx ext with - | None -> mergePushGlobals (visitCilGlobal renameVisitor g); - | Some _ -> ()) + begin + match ExtMerging.findReplacement true extEq !currentFidx ext with + | None -> pushGlobal context + | Some _ -> () + end -let global_annot_pass2 g a = logic_annot_pass2 Toplevel g a +let global_annot_pass2 g a = logic_annot_pass2 (Toplevel g) a (* sm: First attempt at a semantic checksum for function bodies. * Ideally, two function's checksums would be equal only when their diff --git a/src/kernel_services/ast_data/property_status.ml b/src/kernel_services/ast_data/property_status.ml index be7acef29cd93082cd8a823bfef35dfda1d1b1ab..68a3b6b494fe0f93968fef7272277da1b6564bf9 100644 --- a/src/kernel_services/ast_data/property_status.ml +++ b/src/kernel_services/ast_data/property_status.ml @@ -405,7 +405,7 @@ let rec register ppt = Property.pretty ppt Project.pretty (Project.current ()); if Status.mem ppt then - Kernel.fatal "trying to register twice property `%a'.\n\ + Kernel.fatal "trying to register twice property @[<hov 2>'%a'@].@\n\ That is forbidden (kernel invariant broken)." Property.pretty ppt; let h = Emitter_with_properties.Hashtbl.create 7 in diff --git a/src/kernel_services/ast_queries/file.ml b/src/kernel_services/ast_queries/file.ml index 4c20f878df88aa89970c1cd9271f35ac730a9940..bcc24015088980afdcdf610feec5133f3ca88931 100644 --- a/src/kernel_services/ast_queries/file.ml +++ b/src/kernel_services/ast_queries/file.ml @@ -699,6 +699,7 @@ let files_to_cabs_cil files cpp_commands = if Kernel.UnspecifiedAccess.get () then Undefined_sequence.check_sequences merged_file; merged_file, cabs_files + (* "Implicit" annotations are those added by the kernel with ACSL name 'Frama_C_implicit_init'. Currently, this concerns statements that are generated to initialize local variables. *) diff --git a/tests/spec/oracle/submodule.res.oracle b/tests/spec/oracle/submodule.res.oracle new file mode 100644 index 0000000000000000000000000000000000000000..b58a753739a9b01d602245bdd2f39080dfbbcf31 --- /dev/null +++ b/tests/spec/oracle/submodule.res.oracle @@ -0,0 +1,16 @@ +[kernel] Parsing submodule.i (no preprocessing) +/* Generated by Frama-C */ +/*@ +module Foo { + type t; + + logic t elt; + + module Bar { + predicate test(Foo::t x) = x ≡ Foo::elt; + + } + + } + */ + diff --git a/tests/spec/submodule.i b/tests/spec/submodule.i new file mode 100644 index 0000000000000000000000000000000000000000..d57036e0c5803d2d65923664b6a00e46b368993d --- /dev/null +++ b/tests/spec/submodule.i @@ -0,0 +1,9 @@ +/*@ + module Foo { + type t; + logic t elt; + module Bar { + predicate test(t x) = x == Foo::elt; + } + } +*/