diff --git a/ivette/api/generated/kernel/ast/index.ts b/ivette/api/generated/kernel/ast/index.ts index 319a69f78ed8167bb05221dfa84694d8be113315..dc1398e2ab95606276eb5ab92a4475f1ffbe3299 100644 --- a/ivette/api/generated/kernel/ast/index.ts +++ b/ivette/api/generated/kernel/ast/index.ts @@ -281,13 +281,13 @@ export interface functionsData { /** Signature */ signature: string; /** Is the function the main entry point */ - main: boolean; + main?: boolean; /** Is the function defined? */ - defined: boolean; + defined?: boolean; /** Is the function from the Frama-C stdlib? */ - stdlib: boolean; + stdlib?: boolean; /** Is the function a Frama-C builtin? */ - builtin: boolean; + builtin?: boolean; /** Has the function been analyzed by Eva */ eva_analyzed?: boolean; } @@ -299,10 +299,10 @@ export const jFunctionsData: Json.Loose<functionsData> = '#functions expected'), name: Json.jFail(Json.jString,'String expected'), signature: Json.jFail(Json.jString,'String expected'), - main: Json.jFail(Json.jBoolean,'Boolean expected'), - defined: Json.jFail(Json.jBoolean,'Boolean expected'), - stdlib: Json.jFail(Json.jBoolean,'Boolean expected'), - builtin: Json.jFail(Json.jBoolean,'Boolean expected'), + main: Json.jBoolean, + defined: Json.jBoolean, + stdlib: Json.jBoolean, + builtin: Json.jBoolean, eva_analyzed: Json.jBoolean, }); @@ -314,15 +314,15 @@ export const jFunctionsDataSafe: Json.Safe<functionsData> = export const byFunctionsData: Compare.Order<functionsData> = Compare.byFields <{ key: Json.key<'#functions'>, name: string, signature: string, - main: boolean, defined: boolean, stdlib: boolean, builtin: boolean, - eva_analyzed?: boolean }>({ + main?: boolean, defined?: boolean, stdlib?: boolean, + builtin?: boolean, eva_analyzed?: boolean }>({ key: Compare.string, name: Compare.alpha, signature: Compare.string, - main: Compare.boolean, - defined: Compare.boolean, - stdlib: Compare.boolean, - builtin: Compare.boolean, + main: Compare.defined(Compare.boolean), + defined: Compare.defined(Compare.boolean), + stdlib: Compare.defined(Compare.boolean), + builtin: Compare.defined(Compare.boolean), eva_analyzed: Compare.defined(Compare.boolean), }); diff --git a/src/plugins/server/kernel_ast.ml b/src/plugins/server/kernel_ast.ml index cb1c70324132e281a16cb32f20cf01fa9100fcb5..6dd76c134789afb91c5d3db8ef6eb3213da42721 100644 --- a/src/plugins/server/kernel_ast.ml +++ b/src/plugins/server/kernel_ast.ml @@ -371,8 +371,8 @@ struct Cil.hasAttribute "fc_stdlib" vi.vattr || Cil.hasAttribute "fc_stdlib_generated" vi.vattr - let is_analyzed kf = - if Db.Value.is_computed () then Some (!Db.Value.is_called kf) else None + let is_eva_analyzed kf = + if Db.Value.is_computed () then !Db.Value.is_called kf else false let iter f = Globals.Functions.iter @@ -397,27 +397,32 @@ struct ~name:"main" ~descr:(Md.plain "Is the function the main entry point") ~data:(module Data.Jbool) + ~default:false ~get:Kernel_function.is_entry_point; States.column model ~name:"defined" ~descr:(Md.plain "Is the function defined?") ~data:(module Data.Jbool) + ~default:false ~get:Kernel_function.is_definition; States.column model ~name:"stdlib" ~descr:(Md.plain "Is the function from the Frama-C stdlib?") ~data:(module Data.Jbool) + ~default:false ~get:is_stdlib; States.column model ~name:"builtin" ~descr:(Md.plain "Is the function a Frama-C builtin?") ~data:(module Data.Jbool) + ~default:false ~get:is_builtin; States.column model ~name:"eva_analyzed" ~descr:(Md.plain "Has the function been analyzed by Eva") - ~data:(module Data.Joption (Data.Jbool)) - ~get:is_analyzed; + ~data:(module Data.Jbool) + ~default:false + ~get:is_eva_analyzed; States.register_array model ~package ~key ~name:"functions" diff --git a/src/plugins/server/states.ml b/src/plugins/server/states.ml index 5b172c2d4efdfc5e9a25b7839fa276761fb0766b..f9ae51a5b8acb166d074e7b2dc88b1624e90c1b1 100644 --- a/src/plugins/server/states.ml +++ b/src/plugins/server/states.ml @@ -92,23 +92,55 @@ let register_state (type a) ~package ~name ~descr (* --- Model Signature --- *) (* -------------------------------------------------------------------------- *) -type 'a column = Package.fieldInfo * ('a -> json) +type 'a column = Package.fieldInfo * ('a -> json option) type 'a model = 'a column list ref let model () = ref [] +let mkfield (model : 'a model) fd (js : 'a -> json option) = + let open Package in + let name = fd.fd_name in + if List.exists (fun (fd,_) -> fd.fd_name = name) !model then + raise (Invalid_argument "Server.States.column: duplicate name") ; + model := (fd , js) :: !model + let column (type a b) ~name ~descr - ~(data: b Request.output) ~(get : a -> b) (model : a model) = + ~(data: b Request.output) + ~(get : a -> b) + ?(default: b option) + (model : a model) = + let module D = (val data) in + match default with + | None -> + let fd = Package.{ + fd_name = name ; + fd_type = D.jtype ; + fd_descr = descr ; + } in + mkfield model fd (fun a -> Some (D.to_json (get a))) + | Some d -> + let fd = Package.{ + fd_name = name ; + fd_type = Joption D.jtype ; + fd_descr = descr ; + } in + mkfield model fd (fun a -> + let v = get a in + if v = d then None else Some (D.to_json v) + ) + +let option (type a b) ~name ~descr + ~(data: b Request.output) ~(get : a -> b option) (model : a model) = let module D = (val data) in - if List.exists (fun (fd,_) -> fd.Package.fd_name = name) !model then - raise (Invalid_argument "Server.States.column: duplicate name") ; let fd = Package.{ fd_name = name ; - fd_type = D.jtype ; + fd_type = Joption D.jtype ; fd_descr = descr ; } in - model := (fd , fun a -> D.to_json (get a)) :: !model + mkfield model fd (fun a -> match get a with + | None -> None + | Some b -> Some (D.to_json b)) module Kmap = Map.Make(String) @@ -127,7 +159,7 @@ type 'a array = { fkey : string ; key : 'a -> string ; iter : ('a -> unit) -> unit ; - getter : (string * ('a -> json)) list ; + getter : (string * ('a -> json option)) list ; (* [LC+JS] The two following fields allow to keep an array in sync with the current project and still have a polymorphic data type. *) @@ -206,8 +238,9 @@ type buffer = { let add_entry buffer cols fkey key v = let fjs = List.fold_left (fun fjs (fd,to_json) -> - try (fd , to_json v) :: fjs - with Not_found -> fjs + match to_json v with + | Some js -> (fd , js) :: fjs + | None | exception Not_found -> fjs ) [] cols in let row = (fkey, `String key) :: fjs in buffer.updated <- `Assoc row :: buffer.updated ; @@ -267,7 +300,7 @@ let register_array ~package ~name ~descr ~key ?(add_update_hook : 'a callback option) ?(add_remove_hook : 'a callback option) ?(add_reload_hook : unit callback option) - model = + (model : 'a model) = let open Markdown in let href = link ~name () in let columns = List.rev !model in diff --git a/src/plugins/server/states.mli b/src/plugins/server/states.mli index 3e2c681cb406e8b08cf75a0e218871525f3a9967..a75c6c76c8776b067e39b4544c66198d3424fe89 100644 --- a/src/plugins/server/states.mli +++ b/src/plugins/server/states.mli @@ -76,12 +76,23 @@ type 'a model (** Columns array model *) val model : unit -> 'a model (** Populate an array model with a new field. - Columns with name `"id"` and `"_index"` are reserved for internal use. *) + If a [~default] value is given, the field becomes optional and + the field is omitted when equal to the default value (compared with [=]). +*) val column : name:string -> descr:Markdown.text -> data:('b Request.output) -> get:('a -> 'b) -> + ?default:'b -> + 'a model -> unit + +(** Populate an array model with a new optional field. *) +val option : + name:string -> + descr:Markdown.text -> + data:('b Request.output) -> + get:('a -> 'b option) -> 'a model -> unit (** Synchronized array state. *) diff --git a/src/plugins/server/tests/batch/oracle/ast_services.out.json b/src/plugins/server/tests/batch/oracle/ast_services.out.json index 2738c8e311a532572d29fa115ddb9228aa04998f..3dc3730f8f4b2707777bde129a6916dcb1f94d84 100644 --- a/src/plugins/server/tests/batch/oracle/ast_services.out.json +++ b/src/plugins/server/tests/batch/oracle/ast_services.out.json @@ -7,19 +7,13 @@ "key": "kf#24", "name": "g", "signature": "int g(int y);", - "defined": true, - "stdlib": false, - "builtin": false, - "eva_analyzed": null + "defined": true }, { "key": "kf#20", "name": "f", "signature": "int f(int x);", - "defined": true, - "stdlib": false, - "builtin": false, - "eva_analyzed": null + "defined": true } ], "removed": [],