diff --git a/src/kernel_services/ast_queries/ast_diff.ml b/src/kernel_services/ast_queries/ast_diff.ml index 24757a156e653bb93fe74b912f6883e267902512..18a27559e31038fbfaf367c95f8abc073aeda672 100644 --- a/src/kernel_services/ast_queries/ast_diff.ml +++ b/src/kernel_services/ast_queries/ast_diff.ml @@ -80,6 +80,7 @@ module Correspondance = type kf_correspondance = [ kernel_function correspondance | `Same_spec of kernel_function (** body has changed, but spec is identical *) + | `Different_calls of kernel_function ] module Kf_correspondance = @@ -97,15 +98,25 @@ module Kf_correspondance = | `Same_spec f1, `Same_spec f2 -> Kernel_function.compare f1 f2 | `Same_spec _, _ -> 1 | _, `Same_spec _ -> -1 + | `Different_calls f1, `Different_calls f2 -> Kernel_function.compare f1 f2 + | `Different_calls _, _ -> 1 + | _, `Different_calls _ -> -1 | (#correspondance as x), (#correspondance as y) -> C.compare x y let equal = Datatype.from_compare let hash = function | `Same_spec f -> 57 * (Kernel_function.hash f) + | `Different_calls f -> 79 * (Kernel_function.hash f) | #correspondance as x -> C.hash x let internal_pretty_code p fmt = function | `Same_spec f -> let pp fmt = - Format.fprintf fmt "`Same %a" + Format.fprintf fmt "`Same_spec %a" + (Kernel_function.internal_pretty_code Type.Call) f + in + Type.par p Call fmt pp + | `Different_calls f -> + let pp fmt = + Format.fprintf fmt "`Different_calls %a" (Kernel_function.internal_pretty_code Type.Call) f in Type.par p Call fmt pp @@ -113,6 +124,8 @@ module Kf_correspondance = let pretty fmt = function | `Same_spec f -> Format.fprintf fmt "-> (contract) %a" Kernel_function.pretty f + | `Different_calls f -> + Format.fprintf fmt " -> (callees differ) %a" Kernel_function.pretty f | #correspondance as x -> C.pretty fmt x end) @@ -154,3 +167,121 @@ module Kernel_function = (Info(struct let name = "Ast_diff.Kernel_function" end)) module Fundec = Build(Cil_datatype.Fundec) + +(** TODO: use location info to detect potential renaming. + Requires some information about syntactic diff. *) +let find_candidate_type ?loc:_loc ti = + if Globals.Types.mem_type Logic_typing.Typedef ti.tname then begin + match Globals.Types.global Logic_typing.Typedef ti.tname with + | GType(ti,_) -> Some ti + | g -> + Kernel.fatal + "Expected typeinfo instead of %a" Cil_datatype.Global.pretty g + end else None + +let find_candidate_compinfo ?loc:_loc ci = + let su = if ci.cstruct then Logic_typing.Struct else Logic_typing.Union in + if Globals.Types.mem_type su ci.cname then begin + match Globals.Types.global su ci.cname with + | GCompTag (ci,_) | GCompTagDecl(ci,_) -> Some ci + | g -> + Kernel.fatal + "Expected aggregate definition instead of %a" + Cil_datatype.Global.pretty g + end else None + +let find_candidate_enuminfo ?loc:_loc ei = + if Globals.Types.mem_type Logic_typing.Enum ei.ename then begin + match Globals.Types.global Logic_typing.Enum ei.ename with + | GEnumTag(ei,_) | GEnumTagDecl(ei,_) -> Some ei + | g -> + Kernel.fatal + "Expected enumeration definition instead of %a" + Cil_datatype.Global.pretty g + end else None + +let is_same_opt f o o' = + match o, o' with + | None, None -> true + | Some v, Some v' -> f v v' + | _ -> false + +let rec is_same_type t t' = + match t, t' with + | (TVoid _ | TInt _ | TFloat _ | TBuiltin_va_list _), _ -> + Cil_datatype.TypByName.equal t t' + | TPtr(t,a), TPtr(t',a') -> + is_same_type t t' && Cil_datatype.Attributes.equal a a' + | TArray(t,s,a), TArray(t',s',a') -> + is_same_type t t' && is_same_opt is_same_exp s s' + && Cil_datatype.Attributes.equal a a' + | TFun(rt,l,var,a), TFun(rt', l', var', a') -> + is_same_type rt rt' && + is_same_opt (Logic_utils.is_same_list is_same_formal) l l' && + var = var' && + Cil_datatype.Attributes.equal a a' + | TNamed(t,a), TNamed(t',a') -> + let correspondance = typeinfo_correspondance t in + (match correspondance with + | `Not_present -> false + | `Same t'' -> Cil_datatype.Typeinfo.equal t' t'') && + Cil_datatype.Attributes.equal a a' + | TComp(c,a), TComp(c', a') -> + let correspondance = compinfo_correspondance c in + (match correspondance with + | `Not_present -> false + | `Same c'' -> Cil_datatype.Compinfo.equal c' c'') && + Cil_datatype.Attributes.equal a a' + | TEnum(e,a), TEnum(e',a') -> + let correspondance = enuminfo_correspondance e in + (match correspondance with + | `Not_present -> false + | `Same e'' -> Cil_datatype.Enuminfo.equal e' e'') && + Cil_datatype.Attributes.equal a a' + | (TPtr _ | TArray _ | TFun _ | TNamed _ | TComp _ | TEnum _), _ -> false + +and is_same_compinfo _ci _ci' = false + +and is_same_enuminfo _ei _ei' = false + +and is_same_formal (_,t,a) (_,t',a') = + is_same_type t t' && Cil_datatype.Attributes.equal a a' + +and is_same_exp _e _e' = false + +and typeinfo_correspondance ?loc ti = + let add ti = + match find_candidate_type ?loc ti with + | None -> `Not_present + | Some ti' -> + if is_same_type ti.ttype ti'.ttype then `Same ti' else `Not_present + in + Typeinfo.memo add ti + +and compinfo_correspondance ?loc ci = + let add ci = + match find_candidate_compinfo ?loc ci with + | None -> `Not_present + | Some ci' -> + if is_same_compinfo ci ci' then `Same ci' else `Not_present + in + Compinfo.memo add ci + +and enuminfo_correspondance ?loc ei = + let add ei = + match find_candidate_enuminfo ?loc ei with + | None -> `Not_present + | Some ei' -> + if is_same_enuminfo ei ei' then `Same ei' else `Not_present + in + Enuminfo.memo add ei + +let global_correspondance g = + match g with + | GType (ti,loc) -> ignore (typeinfo_correspondance ~loc ti) + | _ -> () + +let compare_ast prj = + Orig_project.set prj; + let ast = Project.on prj Ast.get () in + Cil.iterGlobals ast global_correspondance diff --git a/src/kernel_services/ast_queries/ast_diff.mli b/src/kernel_services/ast_queries/ast_diff.mli index fe83242b179ab102af7a404f8418c0e4ee57a0c2..8aa926a5d3ab036f021c8d9faaf902a30003b79e 100644 --- a/src/kernel_services/ast_queries/ast_diff.mli +++ b/src/kernel_services/ast_queries/ast_diff.mli @@ -40,6 +40,8 @@ type 'a correspondance = type kf_correspondance = [ kernel_function correspondance | `Same_spec of kernel_function (** body has changed, but spec is identical *) + | `Different_calls of kernel_function + (** body is identical, but there are calls to functions that have changed. *) ] (** varinfos correspondances *) @@ -94,3 +96,6 @@ module Kernel_function: module Fundec: State_builder.Hashtbl with type key = fundec and type data = fundec correspondance + +(** [compare_ast prj] sets [prj] as the original project and fill the tables. *) +val compare_ast: Project.t -> unit