diff --git a/ivette/api/generated/kernel/ast/index.ts b/ivette/api/generated/kernel/ast/index.ts
index 83fe1a4fe381959d597e6ca6ac36a19cf56d2a29..dc1398e2ab95606276eb5ab92a4475f1ffbe3299 100644
--- a/ivette/api/generated/kernel/ast/index.ts
+++ b/ivette/api/generated/kernel/ast/index.ts
@@ -280,6 +280,16 @@ export interface functionsData {
   name: string;
   /** Signature */
   signature: string;
+  /** Is the function the main entry point */
+  main?: boolean;
+  /** Is the function defined? */
+  defined?: boolean;
+  /** Is the function from the Frama-C stdlib? */
+  stdlib?: boolean;
+  /** Is the function a Frama-C builtin? */
+  builtin?: boolean;
+  /** Has the function been analyzed by Eva */
+  eva_analyzed?: boolean;
 }
 
 /** Loose decoder for `functionsData` */
@@ -289,6 +299,11 @@ export const jFunctionsData: Json.Loose<functionsData> =
            '#functions expected'),
     name: Json.jFail(Json.jString,'String expected'),
     signature: Json.jFail(Json.jString,'String expected'),
+    main: Json.jBoolean,
+    defined: Json.jBoolean,
+    stdlib: Json.jBoolean,
+    builtin: Json.jBoolean,
+    eva_analyzed: Json.jBoolean,
   });
 
 /** Safe decoder for `functionsData` */
@@ -298,10 +313,17 @@ export const jFunctionsDataSafe: Json.Safe<functionsData> =
 /** Natural order for `functionsData` */
 export const byFunctionsData: Compare.Order<functionsData> =
   Compare.byFields
-    <{ key: Json.key<'#functions'>, name: string, signature: string }>({
+    <{ key: Json.key<'#functions'>, name: string, signature: string,
+       main?: boolean, defined?: boolean, stdlib?: boolean,
+       builtin?: boolean, eva_analyzed?: boolean }>({
     key: Compare.string,
     name: Compare.alpha,
     signature: Compare.string,
+    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),
   });
 
 /** Signal for array [`functions`](#functions)  */
diff --git a/ivette/api/generated/plugins/eva/general/index.ts b/ivette/api/generated/plugins/eva/general/index.ts
index 008df7649b0abefc37880b90d6e771a8e6d49cc8..d16a45ad200fe7939ed2b1f81461ec0d78dce8e8 100644
--- a/ivette/api/generated/plugins/eva/general/index.ts
+++ b/ivette/api/generated/plugins/eva/general/index.ts
@@ -24,6 +24,15 @@ import { jMarkerSafe } from 'frama-c/api/kernel/ast';
 //@ts-ignore
 import { marker } from 'frama-c/api/kernel/ast';
 
+const isComputed_internal: Server.GetRequest<null,boolean> = {
+  kind: Server.RqKind.GET,
+  name:   'plugins.eva.general.isComputed',
+  input:  Json.jNull,
+  output: Json.jBoolean,
+};
+/** True if the Eva analysis has been done */
+export const isComputed: Server.GetRequest<null,boolean>= isComputed_internal;
+
 const getCallers_internal: Server.GetRequest<
   Json.key<'#fct'>,
   [ Json.key<'#fct'>, Json.key<'#stmt'> ][]
diff --git a/ivette/src/dome/src/renderer/frame/sidebars.tsx b/ivette/src/dome/src/renderer/frame/sidebars.tsx
index 27fea8e0cc749bcda882d8e1176c8ea7358cded9..52dc2b2cd91bbb4d579259ed9318f8e31840ec25 100644
--- a/ivette/src/dome/src/renderer/frame/sidebars.tsx
+++ b/ivette/src/dome/src/renderer/frame/sidebars.tsx
@@ -99,6 +99,8 @@ export interface SectionProps {
   disabled?: boolean;
   /** Badge summary (only visible when folded). */
   summary?: Badge;
+  /** Right-click callback. */
+  onContextMenu?: () => void;
   /** Section contents. */
   children?: React.ReactNode;
 }
@@ -132,7 +134,8 @@ export function Section(props: SectionProps) {
     <div className="dome-xSideBarSection">
       <div
         className="dome-xSideBarSection-title dome-color-frame"
-        title={props.label}
+        title={props.title}
+        onContextMenu={props.onContextMenu}
       >
         <Label label={props.label} />
         {!visible && makeBadge(props.summary)}
@@ -168,6 +171,10 @@ export interface ItemProps {
   onSelection?: () => void;
   /** Right-click callback. */
   onContextMenu?: () => void;
+  /** Additional class. */
+  className?: string;
+  /** Additional styles. */
+  style?: React.CSSProperties;
   /** Other item elements. */
   children?: React.ReactNode;
 }
@@ -182,10 +189,12 @@ export function Item(props: ItemProps) {
     'dome-xSideBarItem',
     selected ? 'dome-active' : 'dome-inactive',
     isDisabled && 'dome-disabled',
+    props.className,
   );
   return (
     <div
       className={className}
+      style={props.style}
       title={props.title}
       onContextMenu={onContextMenu}
       onClick={onClick}
diff --git a/ivette/src/renderer/Globals.tsx b/ivette/src/renderer/Globals.tsx
index 4b153c03dde17807d64943188e5a4667c6a0e063..ffd3959366776d3a1c9ed4ff9fba1a4932985645 100644
--- a/ivette/src/renderer/Globals.tsx
+++ b/ivette/src/renderer/Globals.tsx
@@ -5,9 +5,13 @@
 import React from 'react';
 import { Section, Item } from 'dome/frame/sidebars';
 import type { Hint } from 'dome/frame/toolbars';
+import { classes } from 'dome/misc/utils';
 import * as States from 'frama-c/states';
+import { useFlipSettings } from 'dome';
 import { alpha } from 'dome/data/compare';
 import { functions, functionsData } from 'frama-c/api/kernel/ast';
+import { isComputed } from 'frama-c/api/plugins/eva/general';
+import * as Dome from 'dome';
 
 // --------------------------------------------------------------------------
 // --- Global Search Hints
@@ -37,6 +41,39 @@ export function useHints(): [GlobalHint[], (pattern: string) => void] {
   return [hints, onSearch];
 }
 
+// --------------------------------------------------------------------------
+// --- Function Item
+// --------------------------------------------------------------------------
+
+interface FctItemProps {
+  fct: functionsData;
+  current: string | undefined;
+  onSelection: (name: string) => void;
+}
+
+function FctItem(props: FctItemProps) {
+  const { name, signature, main, stdlib, builtin, defined } = props.fct;
+  const className = classes(
+    main && 'globals-main',
+    (stdlib || builtin) && 'globals-stdlib',
+  );
+  const attributes = classes(
+    main && '(main)',
+    !stdlib && !builtin && !defined && '(ext)',
+  );
+  return (
+    <Item
+      className={className}
+      label={name}
+      title={signature}
+      selected={name === props.current}
+      onSelection={() => props.onSelection(name)}
+    >
+      {attributes && <span className="globals-attr">{attributes}</span>}
+    </Item>
+  );
+}
+
 // --------------------------------------------------------------------------
 // --- Globals Section(s)
 // --------------------------------------------------------------------------
@@ -48,25 +85,98 @@ export default () => {
   const fcts = States.useSyncArray(functions).getArray().sort(
     (f, g) => alpha(f.name, g.name),
   );
+  const [stdlib, flipStdlib] =
+    useFlipSettings('ivette.globals.stdlib', false);
+  const [builtin, flipBuiltin] =
+    useFlipSettings('ivette.globals.builtin', false);
+  const [undef, flipUndef] =
+    useFlipSettings('ivette.globals.undef', true);
+  const [selected, flipSelected] =
+    useFlipSettings('ivette.globals.selected', false);
+  const [evaOnly, flipEvaOnly] =
+    useFlipSettings('ivette.globals.evaonly', false);
+  const multipleSelection = selection?.multiple;
+  const multipleSelectionActive = multipleSelection?.allSelections.length > 0;
+  const evaComputed = States.useRequest(isComputed, null);
+
+  function isSelected(fct: functionsData) {
+    return multipleSelection?.allSelections.some(
+      (l) => fct.name === l?.function
+    );
+  }
+
+  function showFunction(fct: functionsData) {
+    const visible =
+      (stdlib || !fct.stdlib)
+      && (builtin || !fct.builtin)
+      && (undef || fct.defined)
+      && (!evaOnly || !evaComputed || (fct.eva_analyzed === true))
+      && (!selected || !multipleSelectionActive || isSelected(fct));
+    return visible;
+  }
+
+  function onSelection(name: string) {
+    updateSelection({ location: { function: name } });
+  }
+
+  async function onContextMenu() {
+    const items: Dome.PopupMenuItem[] = [
+      {
+        label: 'Show Frama-C builtins',
+        checked: builtin,
+        onClick: flipBuiltin,
+      },
+      {
+        label: 'Show stdlib functions',
+        checked: stdlib,
+        onClick: flipStdlib,
+      },
+      {
+        label: 'Show undefined functions',
+        checked: undef,
+        onClick: flipUndef,
+      },
+      'separator',
+      {
+        label: 'Selected only',
+        enabled: multipleSelectionActive,
+        checked: selected,
+        onClick: flipSelected,
+      },
+      {
+        label: 'Analyzed by Eva only',
+        enabled: evaComputed,
+        checked: evaOnly,
+        onClick: flipEvaOnly,
+      },
+    ];
+    Dome.popupMenu(items);
+  }
 
   // Items
   const current: undefined | string = selection?.current?.function;
-  const makeFctItem = (fct: functionsData) => {
-    const kf = fct.name;
-    return (
-      <Item
-        key={kf}
-        label={kf}
-        title={fct.signature}
-        selected={kf === current}
-        onSelection={() => updateSelection({ location: { function: kf } })}
-      />
-    );
-  };
 
+  // Filtered
+
+  const filtered = fcts.filter(showFunction);
+  const nTotal = fcts.length;
+  const nFilter = filtered.length;
+  const title = `Functions ${nFilter} / ${nTotal}`;
   return (
-    <Section label="Functions" defaultUnfold>
-      {fcts.map(makeFctItem)}
+    <Section
+      label="Functions"
+      title={title}
+      onContextMenu={onContextMenu}
+      defaultUnfold
+    >
+      {filtered.map((fct) => (
+        <FctItem
+          key={fct.name}
+          fct={fct}
+          current={current}
+          onSelection={onSelection}
+        />
+      ))}
     </Section>
   );
 
diff --git a/ivette/src/renderer/style.css b/ivette/src/renderer/style.css
index 09df391d8df39183598a6fa9d2636fc49ce2372b..f559cb1dfb3b9b7be75f2f70d3d1d33fef544430 100644
--- a/ivette/src/renderer/style.css
+++ b/ivette/src/renderer/style.css
@@ -36,4 +36,18 @@
     border-bottom: solid 0.2em #BBB;
 }
 
+.globals-main label {
+    text-decoration: underline;
+}
+
+.globals-stdlib label {
+    font-style: italic;
+}
+
+.globals-attr {
+    font-size: smaller;
+    font-weight: lighter;
+    color: #aaa;
+}
+
 /* -------------------------------------------------------------------------- */
diff --git a/src/kernel_services/ast_data/kernel_function.ml b/src/kernel_services/ast_data/kernel_function.ml
index e428ec472cf0d420869f8a080b9d7e5dfa3a87e1..9d77d6ecb2d8a66cf7f851d2e658901dd79c14a6 100644
--- a/src/kernel_services/ast_data/kernel_function.ml
+++ b/src/kernel_services/ast_data/kernel_function.ml
@@ -578,12 +578,11 @@ let is_return_stmt kf stmt =
     false
 
 let is_entry_point kf =
-  let main, _ = Globals.entry_point () in
-  equal kf main
+  try equal kf (fst (Globals.entry_point ()))
+  with Globals.No_such_entry_point _ -> false
 
 let is_main kf =
-  let name = get_name kf in
-  Datatype.String.equal name "main"
+  String.equal (get_name kf) "main"
 
 let returns_void kf =
   let result_type,_,_,_ = Cil.splitFunctionType (get_type kf) in
diff --git a/src/plugins/server/kernel_ast.ml b/src/plugins/server/kernel_ast.ml
index 832c1adce2b70ba182a4549234391d18512c4f28..6dd76c134789afb91c5d3db8ef6eb3213da42721 100644
--- a/src/plugins/server/kernel_ast.ml
+++ b/src/plugins/server/kernel_ast.ml
@@ -338,11 +338,13 @@ let () = Request.register ~package
     ~input:(module Kf) ~output:(module Jtext)
     begin fun kf ->
       let libc = Kernel.PrintLibc.get () in
-      if not libc then Kernel.PrintLibc.set true ;
-      let global = Kernel_function.get_global kf in
-      let ast = Jbuffer.to_json Printer.pp_global global in
-      if not libc then Kernel.PrintLibc.set false ;
-      ast
+      try
+        if not libc then Kernel.PrintLibc.set true ;
+        let global = Kernel_function.get_global kf in
+        let ast = Jbuffer.to_json Printer.pp_global global in
+        if not libc then Kernel.PrintLibc.set false ; ast
+      with err ->
+        if not libc then Kernel.PrintLibc.set false ; raise err
     end
 
 module Functions =
@@ -352,7 +354,31 @@ struct
 
   let signature kf =
     let global = Kernel_function.get_global kf in
-    Rich_text.to_string Printer_tag.pretty (PGlobal global)
+    let libc = Kernel.PrintLibc.get () in
+    try
+      if not libc then Kernel.PrintLibc.set true ;
+      let txt = Rich_text.to_string Printer_tag.pretty (PGlobal global) in
+      if not libc then Kernel.PrintLibc.set false ;
+      if Kernel_function.is_entry_point kf then (txt ^ " /* main */") else txt
+    with err ->
+      if not libc then Kernel.PrintLibc.set false ; raise err
+
+  let is_builtin kf =
+    Cil_builtins.is_builtin (Kernel_function.get_vi kf)
+
+  let is_stdlib kf =
+    let vi = Kernel_function.get_vi kf in
+    Cil.hasAttribute "fc_stdlib" vi.vattr ||
+    Cil.hasAttribute "fc_stdlib_generated" vi.vattr
+
+  let is_eva_analyzed kf =
+    if Db.Value.is_computed () then !Db.Value.is_called kf else false
+
+  let iter f =
+    Globals.Functions.iter
+      (fun kf ->
+         let name = Kernel_function.get_name kf in
+         if not (Ast_info.is_frama_c_builtin name) then f kf)
 
   let array : kernel_function States.array =
     begin
@@ -367,11 +393,41 @@ struct
         ~descr:(Md.plain "Signature")
         ~data:(module Data.Jstring)
         ~get:signature ;
+      States.column model
+        ~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.Jbool)
+        ~default:false
+        ~get:is_eva_analyzed;
       States.register_array model
         ~package ~key
         ~name:"functions"
         ~descr:(Md.plain "AST Functions")
-        ~iter:Globals.Functions.iter
+        ~iter
         ~add_reload_hook:Ast.add_hook_on_update ;
     end
 
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 865d7d9f0c0c0821b47b4dbfefd7ef66440c82c8..3dc3730f8f4b2707777bde129a6916dcb1f94d84 100644
--- a/src/plugins/server/tests/batch/oracle/ast_services.out.json
+++ b/src/plugins/server/tests/batch/oracle/ast_services.out.json
@@ -3,8 +3,18 @@
     "id": "GET-1",
     "data": {
       "updated": [
-        { "key": "kf#24", "name": "g", "signature": "int g(int y);" },
-        { "key": "kf#20", "name": "f", "signature": "int f(int x);" }
+        {
+          "key": "kf#24",
+          "name": "g",
+          "signature": "int g(int y);",
+          "defined": true
+        },
+        {
+          "key": "kf#20",
+          "name": "f",
+          "signature": "int f(int x);",
+          "defined": true
+        }
       ],
       "removed": [],
       "reload": true,
diff --git a/src/plugins/value/api/general_requests.ml b/src/plugins/value/api/general_requests.ml
index abf821d1d69063f79af1cc14354030b3920ae856..bbb5c95724375e6306e6e47c70497fcd2c28e9f5 100644
--- a/src/plugins/value/api/general_requests.ml
+++ b/src/plugins/value/api/general_requests.ml
@@ -31,6 +31,12 @@ let package =
     ~readme:"eva.md"
     ()
 
+let () = Request.register ~package
+    ~kind:`GET ~name:"isComputed"
+    ~descr:(Markdown.plain "True if the Eva analysis has been done")
+    ~input:(module Data.Junit) ~output:(module Data.Jbool)
+    Db.Value.is_computed
+
 let is_computed kf =
   Db.Value.is_computed () &&
   match kf with