diff --git a/Changelog b/Changelog
index 356c0d9f00df51165f5a03714c12c180256bd2db..89126925807b1f70a5c6a1ac8265ecad0f82796d 100644
--- a/Changelog
+++ b/Changelog
@@ -17,6 +17,15 @@
 Open Source Release <next-release>
 ##################################
 
+-*  Kernel    [2019/08/20] Fixes a rare but critical bug which occured when
+              Frama-C internally switched the current project in presence of >2
+              projects and destroyed the old current project at about the same
+              time: the Frama-C internal state became inconsistent and lead to
+              unsound computations and crashes. It may be revealed to the
+              end-user when using a long sequence of -then-replace (at least 3
+              of them).
+-*  Kernel    [2019/08/20] Fixed sequence of -removed-projects and -then
+              options.
 o   Ptests    [2019/08/05] Add new MODULE directive for compiling and loading
               an auxiliary OCaml module for a test
 -   Kernel    [2019/08/05] Add -keep-unused-types normalization option
diff --git a/Makefile b/Makefile
index f1bd75c77b82ed090b286126906ca855c90d89b8..43c11ae3d2657d51b590cd0944d0d2d696344025 100644
--- a/Makefile
+++ b/Makefile
@@ -453,7 +453,8 @@ LIB_CMO =\
 	src/libraries/stdlib/integer \
 	src/libraries/utils/json \
 	src/libraries/utils/markdown \
-	src/libraries/utils/rich_text
+	src/libraries/utils/rich_text \
+	src/libraries/utils/dotgraph
 
 NON_OPAQUE_DEPS+=\
   src/libraries/datatype/unmarshal_z \
diff --git a/Makefile.generating b/Makefile.generating
index 12dd18b3f0afb4cb03bfaf8a9e8a9e682730a238..31a445a75ca111f32afc84a2b6a6748409d9436f 100644
--- a/Makefile.generating
+++ b/Makefile.generating
@@ -133,11 +133,23 @@ ifeq ($(HAS_OCAML408),yes)
         Format.String_tag str -> str \
       | _ -> raise (Invalid_argument "unsupported tag extension")
   FORMAT_STAG_OF_STRING=Format.String_tag s
+  HAS_OCAML407_OR_408=yes
 else
   DYNLINK_INIT=Dynlink.init
   FORMAT_STAG=tag
   FORMAT_STRING_OF_STAG=s
   FORMAT_STAG_OF_STRING=s
+  ifeq ($(HAS_OCAML407),yes)
+    HAS_OCAML407_OR_408=yes
+  else
+    HAS_OCAML407_OR_408=no
+  endif
+endif
+
+ifeq ($(HAS_OCAML407_OR_408),yes)
+  FLOAT_MAX_FLOAT=Float.max_float
+else
+  FLOAT_MAX_FLOAT=Pervasives.max_float
 endif
 
 src/libraries/stdlib/transitioning.ml: \
@@ -153,6 +165,7 @@ src/libraries/stdlib/transitioning.ml: \
          -e 's/@ASSOC_OPT@/$(ASSOC_OPT)/g' \
          -e 's/@ASSQ_OPT@/$(ASSQ_OPT)/g' \
          -e 's/@DYNLINK_INIT@/$(DYNLINK_INIT)/g' \
+         -e 's/@FLOAT_MAX_FLOAT@/$(FLOAT_MAX_FLOAT)/g' \
          -e 's/@FORMAT_STAG@/$(FORMAT_STAG)/g' \
          -e 's/@FORMAT_STRING_OF_STAG@/$(FORMAT_STRING_OF_STAG)/g' \
          -e 's/@FORMAT_STAG_OF_STRING@/$(FORMAT_STAG_OF_STRING)/g' \
diff --git a/headers/header_spec.txt b/headers/header_spec.txt
index 2fb69fa83c4e5a138cd2c1289ebecb6d2be67232..ec689685fd2bca4b2d4bbccc5c6a0a52487960c1 100644
--- a/headers/header_spec.txt
+++ b/headers/header_spec.txt
@@ -648,6 +648,8 @@ src/libraries/utils/cilconfig.ml: CIL
 src/libraries/utils/cilconfig.mli: CIL
 src/libraries/utils/command.ml: CEA_LGPL
 src/libraries/utils/command.mli: CEA_LGPL
+src/libraries/utils/dotgraph.ml: CEA_LGPL
+src/libraries/utils/dotgraph.mli: CEA_LGPL
 src/libraries/utils/escape.ml: CIL
 src/libraries/utils/escape.mli: CIL
 src/libraries/utils/filepath.ml: CEA_LGPL
diff --git a/opam/opam b/opam/opam
index 0c0a81eaa9a61b5d9886d6386a24d3c28c80cbde..3b6424cf61d919a5f75ebeb120dd05eb2d6a6d04 100644
--- a/opam/opam
+++ b/opam/opam
@@ -94,7 +94,7 @@ depends: [
   ( ( "lablgtk" { >= "2.18.2" } & "conf-gnomecanvas" )
   | ( "lablgtk3" { >= "3.0.beta4" & os!="macos" } & "lablgtk3-sourceview3" ))
   "conf-gtksourceview"
-  ( "alt-ergo-free" | "alt-ergo" { <= "2.2.0" } )
+  ( "alt-ergo-free" | "alt-ergo" )
   "conf-graphviz" { post }
   "yojson"
 ]
diff --git a/share/libc/__fc_select.h b/share/libc/__fc_select.h
index 82a079e98bd4fab78b03020d910170aaa3b64b1e..226cab728c71b687a44893152ae307cd1af3e012 100644
--- a/share/libc/__fc_select.h
+++ b/share/libc/__fc_select.h
@@ -51,11 +51,7 @@ extern int pselect(int nfds, fd_set * readfds,
   requires errorfds: errorfds == \null || \valid(errorfds);
   requires timeout: timeout == \null || \valid(timeout);
   assigns __fc_fds_state \from __fc_fds_state;
-  assigns readfds  == \null ? \empty : *readfds,
-          writefds == \null ? \empty : *writefds,
-          errorfds == \null ? \empty : *errorfds,
-          timeout == \null ? \empty : *timeout,
-          \result
+  assigns *readfds, *writefds, *errorfds, *timeout, \result
     \from indirect:nfds,
           indirect:readfds, indirect:*readfds,
           indirect:writefds, indirect:*writefds,
diff --git a/share/libc/signal.h b/share/libc/signal.h
index f42d35711ab12190c0f7e2900f36ce4a2b922333..27ae78965980e8df8e4473bf131b5c445da1faa8 100644
--- a/share/libc/signal.h
+++ b/share/libc/signal.h
@@ -205,8 +205,8 @@ struct sigaction *__fc_p_sigaction = __fc_sigaction;
   requires valid_oldact_or_null: oldact == \null || \valid(oldact);
   requires valid_read_act_or_null: act == \null || \valid_read(act);
   requires separation:separated_acts: \separated(act, oldact);
-  assigns oldact == \null ? \empty : *oldact \from __fc_p_sigaction;
-  assigns act == \null ? \empty : __fc_p_sigaction[signum] \from *act;
+  assigns *oldact \from __fc_p_sigaction;
+  assigns __fc_p_sigaction[signum] \from *act;
   assigns \result \from indirect:signum, indirect:act, indirect:*act,
                         indirect:oldact, indirect:*oldact;
   ensures act_changed: act == \null || \subset(__fc_p_sigaction[signum], *act);
@@ -225,8 +225,7 @@ extern int sigaction(int signum, const struct sigaction *restrict act,
   requires separation: (set == oldset == \null) ||
                        \separated(set, oldset);
   assigns \result \from indirect:how, indirect:set, indirect:oldset;
-  assigns oldset == \null ? \empty : *oldset
-          \from indirect:how, indirect:oldset;
+  assigns *oldset \from indirect:how, indirect:oldset;
   ensures result_ok_or_error: \result == 0 || \result == -1;
   ensures initialization:oldset_initialized:
     oldset != \null && \result == 0 ==> \initialized(oldset);
diff --git a/share/libc/stropts.h b/share/libc/stropts.h
index 48351585e7e3537806592d6ac9d2d6cf81d9f2af..3b0ee82968b303fbf98723c0d45d05198c11e4ea 100644
--- a/share/libc/stropts.h
+++ b/share/libc/stropts.h
@@ -172,7 +172,7 @@ extern int    __va_ioctl_int(int fd, int request, int arg);
 
 /*@ assigns \result \from indirect:fd, indirect:request,
       indirect:((char*)argp)[0..];
-    assigns argp != \null ? ((char*)argp)[0..] : \empty \from
+    assigns ((char*)argp)[0..] \from
       indirect:fd, indirect:request, ((char*)argp)[0..]; */
 extern int    __va_ioctl_ptr(int fd, int request, void* argp);
 
diff --git a/share/libc/sys/time.h b/share/libc/sys/time.h
index 73fed482445f4c6a663ae88e8535b45fc5c704a8..b860763f4d3909a5c198a66af5701417801f78c9 100644
--- a/share/libc/sys/time.h
+++ b/share/libc/sys/time.h
@@ -151,16 +151,16 @@ extern int getitimer(int which, struct itimerval *curr_value);
 /*@
   requires valid_new_value: \valid_read(new_value);
   requires old_value_null_or_valid: old_value == \null || \valid(old_value);
-  assigns old_value != \null ? *old_value : \empty \from
-    indirect:which, indirect:old_value, indirect:new_value,
-    __fc_itimer_real, __fc_itimer_virtual, __fc_itimer_prof;
+  assigns *old_value \from indirect:which, indirect:old_value,
+                           indirect:new_value, __fc_itimer_real,
+                           __fc_itimer_virtual, __fc_itimer_prof;
   assigns \result \from indirect:which, indirect:new_value, indirect:*new_value;
   ensures result_ok_or_error: \result == 0 || \result == -1;
   behavior real:
     assumes itimer_real_and_valid:
       which == ITIMER_REAL && __VALID_ITIMERVAL(new_value);
     assigns \result \from \nothing;
-    assigns old_value != \null ? *old_value : \empty \from __fc_itimer_real;
+    assigns *old_value \from __fc_itimer_real;
     assigns __fc_itimer_real \from *new_value;
     ensures result_ok: \result == 0;
     ensures initialization:old_value: \initialized(old_value);
@@ -168,14 +168,14 @@ extern int getitimer(int which, struct itimerval *curr_value);
     assumes itimer_virtual_and_valid:
       which == ITIMER_VIRTUAL && __VALID_ITIMERVAL(new_value);
     assigns \result \from \nothing;
-    assigns old_value != \null ? *old_value : \empty  \from __fc_itimer_virtual;
+    assigns *old_value \from __fc_itimer_virtual;
     ensures result_ok: \result == 0;
     ensures initialization:old_value: \initialized(old_value);
   behavior prof:
     assumes itimer_prof_and_valid:
       which == ITIMER_PROF && __VALID_ITIMERVAL(new_value);
     assigns \result \from \nothing;
-    assigns old_value != \null ? *old_value : \empty  \from __fc_itimer_prof;
+    assigns *old_value \from __fc_itimer_prof;
     ensures result_ok: \result == 0;
     ensures initialization:old_value: \initialized(old_value);
   behavior invalid:
diff --git a/share/libc/time.h b/share/libc/time.h
index 60e5031571f9ef5e2d527e9b5e8a81907ae66a36..52c66700cc80c3df73b708b8f6a302a30b2d6877 100644
--- a/share/libc/time.h
+++ b/share/libc/time.h
@@ -216,9 +216,8 @@ extern int clock_gettime(clockid_t clk_id, struct timespec *tp);
     assumes no_einval: valid_clock_id(clock_id);
     assigns \result \from indirect:__fc_time, indirect:clock_id, indirect:rqtp,
                           indirect:*rqtp;
-    assigns rmtp == \null ? \empty : *rmtp \from __fc_time, indirect:clock_id,
-                                                 indirect:rqtp, indirect:*rqtp,
-                                                 indirect:rmtp;
+    assigns *rmtp \from __fc_time, indirect:clock_id, indirect:rqtp,
+                        indirect:*rqtp, indirect:rmtp;
     ensures result_interrupted: \result == EINTR;
     ensures initialization:interrupted_remaining:
       rmtp != \null ==> \initialized(&rmtp->tv_sec) && \initialized(&rmtp->tv_nsec);
@@ -263,9 +262,8 @@ extern struct tm *localtime_r(const time_t *restrict timep,
   requires valid_nanosecs: 0 <= rqtp->tv_nsec < 1000000000;
   requires valid_remaining_or_null: rmtp == \null || \valid(rmtp);
   assigns \result \from indirect:__fc_time, indirect:rqtp, indirect:*rqtp;
-  assigns rmtp == \null ? \empty : *rmtp \from indirect:__fc_time,
-                                               indirect:rqtp, indirect:*rqtp,
-                                               indirect:rmtp;
+  assigns *rmtp \from indirect:__fc_time, indirect:rqtp, indirect:*rqtp,
+                      indirect:rmtp;
   ensures result_elapsed_or_interrupted: \result == 0 || \result == -1;
   ensures initialization:interrupted_remaining:
     rmtp != \null && \result == -1 ==>
diff --git a/src/kernel_internals/runtime/special_hooks.ml b/src/kernel_internals/runtime/special_hooks.ml
index 9aecb292da713a7214e26574f10a62229de8f029..7448601a76488d7a93c75987b7b33f8a9e6d434e 100644
--- a/src/kernel_internals/runtime/special_hooks.ml
+++ b/src/kernel_internals/runtime/special_hooks.ml
@@ -138,8 +138,9 @@ let () =
 
 (* Load Frama-c from disk if required *)
 let load_binary () =
-  let filename = Kernel.LoadState.get () in
-  if filename <> "" then begin
+  let filepath = Kernel.LoadState.get () in
+  if filepath <> Filepath.Normalized.unknown then begin
+    let filename = Filepath.Normalized.to_pretty_string filepath in
     try
       Project.load_all filename
     with Project.IOError s ->
diff --git a/src/kernel_internals/typing/cabs2cil.ml b/src/kernel_internals/typing/cabs2cil.ml
index 1b595764bf03ef02f94a6901f87658e97ab5b21d..a8012c9c465ba4586ebad9e4b558678c24d625ee 100644
--- a/src/kernel_internals/typing/cabs2cil.ml
+++ b/src/kernel_internals/typing/cabs2cil.ml
@@ -5230,7 +5230,7 @@ and makeCompType ghost (isstruct: bool)
       if Cil.isFunctionType ftype then
         Kernel.error ~current:true
           "field `%s' declared as a function" n
-      else if Cil.has_flexible_array_member ftype then
+      else if Cil.has_flexible_array_member ftype && isstruct then
         Kernel.error ~current:true
           "field `%s' declared with a type containing a flexible array member."
           n
diff --git a/src/kernel_services/cmdline_parameters/parameter_builder.ml b/src/kernel_services/cmdline_parameters/parameter_builder.ml
index d74de525bfa58a548f5990215ab66750e7423ef6..d20cc20285af3175eabfb4202058616734e426ae 100644
--- a/src/kernel_services/cmdline_parameters/parameter_builder.ml
+++ b/src/kernel_services/cmdline_parameters/parameter_builder.ml
@@ -442,6 +442,91 @@ struct
   module Empty_string(X: Parameter_sig.Input_with_arg) =
     String(struct include X let default = empty_string end)
 
+  (* ************************************************************************ *)
+  (** {3 Filepath} *)
+  (* ************************************************************************ *)
+
+  module Fc_Filepath = Filepath
+
+  module Filepath
+      (X: sig
+         include Parameter_sig.Input_with_arg
+         val existence : Parameter_sig.existence
+       end) =
+  struct
+
+    exception No_file
+    exception File_exists
+
+    include Build
+        (struct
+          include Datatype.Filepath
+          include X
+          let default () = Filepath.Normalized.unknown
+          let functor_name = "Filepath"
+        end)
+
+    let check_existence existence fp =
+      match existence with
+      | Parameter_sig.Indifferent -> ()
+      | Parameter_sig.Must_exist ->
+        if not (Sys.file_exists (Filepath.Normalized.to_pretty_string fp)) then
+          raise No_file
+      | Parameter_sig.Must_not_exist ->
+        if Sys.file_exists (Filepath.Normalized.to_pretty_string fp) then
+          raise File_exists
+
+    let existence = X.existence
+
+    let convert f oldstr newstr =
+      let oldfp = Filepath.Normalized.to_pretty_string oldstr in
+      let newfp = Filepath.Normalized.to_pretty_string newstr in
+      f oldfp newfp
+
+    let set fp = check_existence existence fp ; set fp
+
+    let set_str s = set (Filepath.Normalized.of_string s)
+
+    let add_option name =
+      Cmdline.add_option
+        name
+        ~argname:X.arg_name
+        ~help:X.help
+        ~visible:is_visible
+        ~ext_help:!Parameter_customize.optional_help_ref
+        ~plugin:P.shortname
+        ~group
+        stage
+        (Cmdline.String set_str)
+
+    let parameter_get fp = Filepath.Normalized.to_pretty_string (get fp)
+    let parameter_add_set_hook f = add_set_hook (convert f)
+    let parameter_add_update_hook f = add_update_hook (convert f)
+
+    let parameter =
+      let accessor =
+        Typed_parameter.String
+          ({ Typed_parameter.get = parameter_get;
+             set = set_str;
+             add_set_hook = parameter_add_set_hook;
+             add_update_hook = parameter_add_update_hook },
+           fun () -> [])
+      in
+      let p =
+	Typed_parameter.create ~name ~help:X.help ~accessor ~is_set
+      in
+      add_parameter !Parameter_customize.group_ref stage p;
+      add_option X.option_name;
+      Parameter_customize.reset ();
+      if is_dynamic then
+	let plugin = empty_string in
+	Dynamic.register
+          ~plugin X.option_name Typed_parameter.ty ~journalize:false p
+      else
+	p
+
+  end
+
   (* ************************************************************************ *)
   (** {3 Collections} *)
   (* ************************************************************************ *)
diff --git a/src/kernel_services/cmdline_parameters/parameter_sig.mli b/src/kernel_services/cmdline_parameters/parameter_sig.mli
index bb827691793567e0bbe804f712a2fb7293d5a96b..a36076d68471974e040e4d0cebef792d3d185d74 100644
--- a/src/kernel_services/cmdline_parameters/parameter_sig.mli
+++ b/src/kernel_services/cmdline_parameters/parameter_sig.mli
@@ -321,6 +321,24 @@ module type Specific_dir = sig
 
 end
 
+type existence = Must_exist | Must_not_exist | Indifferent
+
+(** signature for normalized pathnames. *)
+module type Filepath = sig
+
+  exception No_file
+  (** raised by {!set} if no file exists and [existence] is [Must_exist]. *)
+
+  exception File_exists
+  (** raised by {!set} if some file exists and [existence] is
+      [Must_nos_exist]. *)
+
+  val existence: existence
+
+  include S with type t = Filepath.Normalized.t
+
+end
+
 (* ************************************************************************** *)
 (** {3 Collections} *)
 (* ************************************************************************** *)
@@ -515,6 +533,13 @@ module type Builder = sig
   (** @plugin development guide *)
   module Empty_string(X: Input_with_arg): String
 
+  module Fc_Filepath = Filepath
+
+  module Filepath(X: sig
+      include Input_with_arg
+      val existence: existence
+    end): Filepath
+
   exception Cannot_build of string
   module Make_set
     (E:
diff --git a/src/kernel_services/plugin_entry_points/dynamic.ml b/src/kernel_services/plugin_entry_points/dynamic.ml
index 10241320736c4495275430db1a8a6ac2bbecc563..f686cd7684e27896afc96eae7409215a3f6d11be 100644
--- a/src/kernel_services/plugin_entry_points/dynamic.ml
+++ b/src/kernel_services/plugin_entry_points/dynamic.ml
@@ -445,6 +445,14 @@ module Parameter = struct
         let modname = "String"
       end)
 
+  module Filepath =
+    Common
+      (struct
+        type t = Datatype.Filepath.t
+        let ty = Datatype.Filepath.ty
+        let modname = "Filepath"
+      end)
+
   module StringSet = struct
     include Common
         (struct include Datatype.String.Set let modname = "StringSet" end)
diff --git a/src/kernel_services/plugin_entry_points/dynamic.mli b/src/kernel_services/plugin_entry_points/dynamic.mli
index c837a15601e045eb08dfe2a936ba8c4960687689..3650174dc2a424590dc3ba3473e6db44c1abc3be 100644
--- a/src/kernel_services/plugin_entry_points/dynamic.mli
+++ b/src/kernel_services/plugin_entry_points/dynamic.mli
@@ -118,6 +118,9 @@ module Parameter : sig
   (** String parameters. *)
   module String : Common with type t = string
 
+  (** Filepath parameters. *)
+  module Filepath : Common with type t = Datatype.Filepath.t
+
   (** Set of string parameters. *)
   module StringSet : sig
     include Common with type t = Datatype.String.Set.t
diff --git a/src/kernel_services/plugin_entry_points/kernel.ml b/src/kernel_services/plugin_entry_points/kernel.ml
index 02f357e8ebbc52cf9fb7883e3551c04d322a9a6b..3bcff739e6a5064f21a7fd18923c3992c43c6ce3 100644
--- a/src/kernel_services/plugin_entry_points/kernel.ml
+++ b/src/kernel_services/plugin_entry_points/kernel.ml
@@ -716,10 +716,11 @@ let () = Parameter_customize.set_cmdline_stage Cmdline.Loading
    reset *) 
 (*let () = Parameter_customize.do_not_projectify ()*)
 module LoadState =
-  P.Empty_string
+  P.Filepath
     (struct
        let option_name = "-load"
        let arg_name = "filename"
+       let existence = Parameter_sig.Must_exist
        let help = "load a previously-saved session from file <filename>"
      end)
 
@@ -1522,7 +1523,14 @@ let _ =
 
 let () =
   Cmdline.run_after_configuring_stage
-    (fun () -> Remove_projects.iter (fun project -> Project.remove ~project ()))
+    (fun () ->
+       (* clear "-remove-projects" before itering over (a copy of) its contents
+          in order to prevent warnings about dangling pointer deletion (since it
+          is itself projectified and so contains a pointer to the project being
+          removed). *)
+       let s = Remove_projects.get () in
+       Remove_projects.clear ();
+       Project.Datatype.Set.iter (fun project -> Project.remove ~project ()) s)
 
 (* ************************************************************************* *)
 (** {2 Others options} *)
diff --git a/src/kernel_services/plugin_entry_points/kernel.mli b/src/kernel_services/plugin_entry_points/kernel.mli
index 7976e6c86673f65f73788c64e6a77b384b0a0fed..0b43666bb0891fda4c20fae462daa956571e9dec 100644
--- a/src/kernel_services/plugin_entry_points/kernel.mli
+++ b/src/kernel_services/plugin_entry_points/kernel.mli
@@ -299,7 +299,7 @@ module BigIntsHex: Parameter_sig.Int
 module SaveState: Parameter_sig.String
   (** Behavior of option "-save" *)
 
-module LoadState: Parameter_sig.String
+module LoadState: Parameter_sig.Filepath
   (** Behavior of option "-load" *)
 
 module LoadModule: Parameter_sig.String_list
diff --git a/src/kernel_services/plugin_entry_points/plugin.ml b/src/kernel_services/plugin_entry_points/plugin.ml
index f24bdbcfb2680fd819c456b99b0c6d7ca87c8dd9..01d5831cbe50b7cc0048d60f594a888d4a58ad32 100644
--- a/src/kernel_services/plugin_entry_points/plugin.ml
+++ b/src/kernel_services/plugin_entry_points/plugin.ml
@@ -534,7 +534,7 @@ struct
 
     let pp_source fmt = function
       | None -> ()
-      | Some src -> Format.fprintf fmt "%a:" Filepath.pp_pos src
+      | Some src -> Format.fprintf fmt "%a:" Fc_Filepath.pp_pos src
   end
 
   (* Output must be synchronized with functions [prefix_*] in module Log. *)
diff --git a/src/libraries/project/project.ml b/src/libraries/project/project.ml
index 55280721f01c32fbc314ed7bc74e1571363aae86..27ff62f432dcb24838c06800b3c6a07c4da9e4f8 100644
--- a/src/libraries/project/project.ml
+++ b/src/libraries/project/project.ml
@@ -361,6 +361,9 @@ let journalized_set_current =
 let set_current ?(on=false) ?(selection=State_selection.full) p =
   if not (equal p (current ())) then journalized_set_current on selection p
 
+let set_current_as_last_created () =
+  Extlib.may (fun p -> set_current p) !last_created_by_copy_ref
+
 (** Indicates if we should keep [p] as the current project when calling {!on p}. *)
 let keep_current: bool ref = ref false
 
@@ -376,12 +379,12 @@ let on ?selection p f x =
   if old_current == p then f x
   else
     let set p = set_current ~on:true ?selection p in
-    let set_to_old () =
+    let rec set_to_old old =
       if not !keep_current then
         try
           (* if someone asks for keeping [p] as current during the execution of
              [f], do not restore [old_current] at the end. *)
-          set old_current
+          set old
         with Invalid_argument _ ->
           (* the old current project has been remove: replace it by the previous
              one, if any *)
@@ -390,16 +393,16 @@ let on ?selection p f x =
               old_current.unique_name
               (current ()).unique_name
           else
-            Q.move_at_top (Q.nth 1 projects) projects
+            set_to_old (Q.nth 1 projects)
     in
     let go () =
       set p;
       let r = f x in
-      set_to_old ();
+      set_to_old old_current;
       r
     in
     if debug_atleast 1 then go ()
-    else begin try go () with e -> set_to_old (); raise e end
+    else begin try go () with e -> set_to_old old_current; raise e end
 
 (* [set_current] must never be called internally. *)
 module Hide_set_current = struct let set_current () = assert false end
diff --git a/src/libraries/project/project.mli b/src/libraries/project/project.mli
index d24b9b72f021aa85a77861a275c364da006b82cf..ec2f4d49c9be6e619e6cd6dc672ffe4a53f31c8e 100644
--- a/src/libraries/project/project.mli
+++ b/src/libraries/project/project.mli
@@ -142,6 +142,10 @@ val set_keep_current: bool -> unit
       of the current {!on}) iff [b] is [true].
       @since Aluminium-20160501 *)
 
+(**/**)
+val set_current_as_last_created: unit -> unit
+(**/**)
+
 val copy: ?selection:State_selection.t -> ?src:t -> t -> unit
   (** Copy a project into another one. Default project for [src] is [current
       ()]. Replace the destination by [src].
diff --git a/src/libraries/stdlib/transitioning.ml.in b/src/libraries/stdlib/transitioning.ml.in
index de6215991de78ead186f5deb8cca34095989fb99..cf09ddd667ed096ced79582ad7905f8ddc82ed19 100644
--- a/src/libraries/stdlib/transitioning.ml.in
+++ b/src/libraries/stdlib/transitioning.ml.in
@@ -37,6 +37,10 @@ module Dynlink = struct
   let init = @DYNLINK_INIT@
 end
 
+module Float = struct
+  let max_float = @FLOAT_MAX_FLOAT@
+end
+
 module Format = struct
   type stag = Format.@FORMAT_STAG@
   let string_of_stag s = @FORMAT_STRING_OF_STAG@
diff --git a/src/libraries/stdlib/transitioning.mli b/src/libraries/stdlib/transitioning.mli
index 521cf30c013bf939d5bbecc4444a682620267319..94ae54311a9ca97d31b0eab45927b7d4454c96c8 100644
--- a/src/libraries/stdlib/transitioning.mli
+++ b/src/libraries/stdlib/transitioning.mli
@@ -49,6 +49,11 @@ module Dynlink: sig
   val init: unit -> unit
 end
 
+(** 4.07 *)
+module Float: sig
+  val max_float: float
+end
+
 (** 4.08 *)
 module Format: sig
   type stag
diff --git a/src/libraries/utils/dotgraph.ml b/src/libraries/utils/dotgraph.ml
new file mode 100644
index 0000000000000000000000000000000000000000..f9939a2eb3677e29cafc5c34cbdbcbcefc2e07ef
--- /dev/null
+++ b/src/libraries/utils/dotgraph.ml
@@ -0,0 +1,405 @@
+(**************************************************************************)
+(*                                                                        *)
+(*  This file is part of Frama-C.                                         *)
+(*                                                                        *)
+(*  Copyright (C) 2007-2019                                               *)
+(*    CEA (Commissariat à l'énergie atomique et aux énergies              *)
+(*         alternatives)                                                  *)
+(*                                                                        *)
+(*  you can redistribute it and/or modify it under the terms of the GNU   *)
+(*  Lesser General Public License as published by the Free Software       *)
+(*  Foundation, version 2.1.                                              *)
+(*                                                                        *)
+(*  It is distributed in the hope that it will be useful,                 *)
+(*  but WITHOUT ANY WARRANTY; without even the implied warranty of        *)
+(*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *)
+(*  GNU Lesser General Public License for more details.                   *)
+(*                                                                        *)
+(*  See the GNU Lesser General Public License version 2.1                 *)
+(*  for more details (enclosed in the file licenses/LGPLv2.1).            *)
+(*                                                                        *)
+(**************************************************************************)
+
+(* -------------------------------------------------------------------------- *)
+(* --- Attributes and Such                                                --- *)
+(* -------------------------------------------------------------------------- *)
+
+type attr = [
+  | `LR
+  | `TB
+  | `Label of string
+  | `Color of string
+  | `Fillcolor of string
+  | `Shape of string
+  | `Style of string
+  | `Circle
+  | `Box
+  | `Oval
+  | `Point
+  | `Dotted
+  | `Filled
+  | `ArrowBoth
+  | `ArrowBack
+  | `ArrowForward
+  | `ArrowHead of string
+  | `ArrowTail of string
+  | `Attr of string * string
+]
+
+let attr = function
+  | `LR -> "rankdir" , "LR"
+  | `TB -> "rankdir" , "TB"
+  | `Label txt -> "label" , txt
+  | `Color c -> "color" , c
+  | `Fillcolor c -> "fillcolor" , c
+  | `Shape sh -> "shape" , sh
+  | `Style sty -> "style" , sty
+  | `Box -> "shape" , "box"
+  | `Oval -> "shape" , "oval"
+  | `Point -> "shape" , "point"
+  | `Circle -> "shape" , "circle"
+  | `Filled -> "style" , "filled"
+  | `Dotted -> "style" , "dotted"
+  | `ArrowBoth -> "dir" , "both"
+  | `ArrowBack -> "dir" , "back"
+  | `ArrowForward -> "dir" , "forward"
+  | `ArrowHead sh -> "arrowhead" , sh
+  | `ArrowTail sh -> "arrowtail" , sh
+  | `Attr(name,value) -> name , value
+
+let pp_attr fmt (a : attr) =
+  let name,value = attr a in Format.fprintf fmt "%s=%S" name value
+
+let pp_attributes fmt = function
+  | [] -> ()
+  | p::ps ->
+    begin
+      Format.fprintf fmt "@ %a" pp_attr p ;
+      List.iter (fun p -> Format.fprintf fmt ",@ %a" pp_attr p) ps ;
+    end
+
+(* -------------------------------------------------------------------------- *)
+(* --- Dot Output                                                         --- *)
+(* -------------------------------------------------------------------------- *)
+
+type dot = {
+  file : string ;
+  queue : (unit -> unit) Queue.t ;
+  mutable indent : string ;
+  mutable id : int ;
+  mutable fmt : Format.formatter ;
+  mutable out : out_channel option ;
+}
+
+let open_dot ?(name="G") ?(attr=[]) ?(file) () =
+  let file = match file with None -> name ^ ".dot" | Some f -> f in
+  let out = open_out file in
+  let fmt = Format.formatter_of_out_channel out in
+  begin
+    Format.fprintf fmt "digraph %S {@\n" name ;
+    List.iter
+      (fun p ->
+         Format.fprintf fmt "  %a ;@\n" pp_attr p
+      ) attr ;
+    Format.pp_print_flush fmt () ;
+    { file ; fmt ;
+      queue = Queue.create () ;
+      indent = "  " ; out = Some out ; id = 0 }
+  end
+
+let flush dot =
+  match dot.out with
+  | Some out -> Format.pp_print_flush dot.fmt () ; flush out
+  | None -> ()
+
+let basename f =
+  if Filename.check_suffix f ".dot" then Filename.chop_suffix f ".dot" else f
+
+let installed = ref None
+let is_dot_installed () =
+  match !installed with
+  | Some s -> s
+  | None -> let s = (Sys.command "dot -V" = 0) in installed := Some s ; s
+
+let close dot =
+  match dot.out with
+  | None -> ()
+  | Some out ->
+    begin
+      Format.fprintf dot.fmt "}@." ;
+      dot.fmt <- Format.err_formatter ;
+      Pervasives.flush out ; close_out out ;
+      dot.out <- None ;
+    end
+
+let layout ?(force=false) ?(target="pdf") ?(engine="dot") ?output dot =
+  begin
+    if dot.out <> None then raise (Invalid_argument "DotGraph: not closed") ;
+    let input = dot.file in
+    let output =
+      match output with Some f -> f | None ->
+        Printf.sprintf "%s.%s" (basename dot.file) target in
+    let cmd = Printf.sprintf "dot -K%s -T%s %s -o %s"
+        engine target input output in
+    let status = Sys.command cmd in
+    if status=0 then output else
+    if force then
+      let msg = Printf.sprintf "dot failed with status %d" status in
+      raise (Invalid_argument msg)
+    else dot.file
+  end
+
+let printf dot msg = Format.fprintf dot.fmt msg
+let println dot msg =
+  Format.kfprintf (fun fmt -> Format.pp_print_newline fmt ()) dot.fmt msg
+
+let push dot f = Queue.push f dot.queue
+let pop_all dot =
+  while not (Queue.is_empty dot.queue) do
+    (Queue.pop dot.queue) ()
+  done
+
+(* -------------------------------------------------------------------------- *)
+(* --- Nodes and Edges                                                    --- *)
+(* -------------------------------------------------------------------------- *)
+
+type node = string
+
+let fresh ?(prefix="_") dot =
+  let k = dot.id in dot.id <- succ k ; Printf.sprintf "%s%03d" prefix k
+
+let pp_node = Format.pp_print_string
+let pp_edge fmt (a,b) = Format.fprintf fmt "%s -> %s" a b
+
+let pp_stmt dot pp stmt attr =
+  Format.fprintf dot.fmt "%s@[<hv 0>@[<hov 2>%a [%a@]@ ]@];@."
+    dot.indent pp stmt pp_attributes attr
+
+let inode dot ?prefix ?id attr =
+  let a = match id with Some a -> a | None -> fresh ?prefix dot in
+  pp_stmt dot pp_node a attr ; a
+
+let node_default dot attr =
+  pp_stmt dot pp_node "node" attr
+
+let edge_default dot attr =
+  pp_stmt dot pp_node "edge" attr
+
+let node dot id attr = ignore (inode dot ~id attr)
+
+let edge dot a b attr =
+  if attr = []
+  then Format.fprintf dot.fmt "%s%a ;@." dot.indent pp_edge (a,b)
+  else pp_stmt dot pp_edge (a,b) attr
+
+let link dot ps attr =
+  let rec walk dot attr p = function
+    | q :: ps -> edge dot p q attr ; walk dot attr q ps
+    | [] -> ()
+  in match ps with
+  | p :: ps -> walk dot attr p ps
+  | [] -> ()
+
+(* -------------------------------------------------------------------------- *)
+(* --- Clustering                                                         --- *)
+(* -------------------------------------------------------------------------- *)
+
+let rank dot nodes =
+  begin
+    Format.fprintf dot.fmt "%s@[<hov 2>{ rank=same;" dot.indent ;
+    List.iter (Format.fprintf dot.fmt "@ %s;") nodes ;
+    Format.fprintf dot.fmt "@ }@]@\n" ;
+  end
+
+let subgraph dot ?(cluster=true) attr content =
+  begin
+    let k = dot.id in dot.id <- succ k ;
+    let indent = dot.indent in
+    Format.fprintf dot.fmt "%ssubgraph %s%d {@\n" indent
+      (if cluster then "cluster" else "g") k ;
+    List.iter
+      (fun a -> Format.fprintf dot.fmt "%s  %a;@\n" indent pp_attr a) attr ;
+    dot.indent <- indent ^ "  " ;
+    content () ;
+    dot.indent <- indent ;
+    Format.fprintf dot.fmt "%s}@\n" indent ;
+  end
+
+type record = [
+  | `Empty
+  | `Hbox of record list
+  | `Vbox of record list
+  | `Label of string
+  | `Port of string * link list * string
+] and link = string * attr list * node
+
+let port a b = if b <> "" then Printf.sprintf "%s:%s" a b else a
+
+module Record =
+struct
+  let hbox = function `Empty -> [] | `Hbox hs -> hs | h -> [h]
+  let vbox = function `Empty -> [] | `Vbox vs -> vs | v -> [v]
+  let (<->) a b = `Vbox (vbox a @ vbox b)
+  let (<|>) a b = `Hbox (hbox a @ hbox b)
+  let link ?(anchor="") ?(attr=[]) node : link = anchor,attr,node
+  let label ?(port="") ?(link=[]) txt : record =
+    if port="" && link=[] then `Label txt else `Port(port,link,txt)
+end
+
+type env = {
+  buffer : Buffer.t ;
+  mutable links : (string * (string * attr list * node) list) list ;
+  mutable port : int ;
+}
+
+let rec mk_record env hv = function
+  | `Empty -> ()
+  | `Label txt -> Buffer.add_string env.buffer txt
+  | `Port(port,links,txt) ->
+    if links <> [] then
+      let port =
+        if port = "" then
+          let p = env.port in env.port <- succ p ; Printf.sprintf "_p%d" p
+        else port in
+      env.links <- (port,links) :: env.links ;
+      Printf.bprintf env.buffer "<%s> %s" port txt
+  | `Hbox rs -> mk_box env hv true rs
+  | `Vbox rs -> mk_box env hv false rs
+
+and mk_box env hv hb = function
+  | [] -> ()
+  | r::rs ->
+    begin
+      let buffer = env.buffer in
+      let boxed = hv <> hb in
+      let hv = not hv in
+      if boxed then Buffer.add_char buffer '{' ;
+      mk_record env hv r ;
+      List.iter (fun r ->
+          Buffer.add_char buffer '|' ;
+          mk_record env hv r
+        ) rs ;
+      if boxed then Buffer.add_char buffer '}' ;
+    end
+
+let irecord dot ?prefix ?id ?(rounded=false) ?(attr=[]) (box : record) =
+  let shape = if rounded then "Mrecord" else "record" in
+  let env = { buffer = Buffer.create 80 ; links = [] ; port = 1 } in
+  mk_record env true box ;
+  let label = Buffer.contents env.buffer in
+  let node = inode dot ?prefix ?id (`Shape shape :: `Label label :: attr) in
+  List.iter
+    (fun (name,links) ->
+       List.iter
+         (fun (anchor,attr,target) ->
+            edge dot (port (port node name) anchor) target attr
+         ) links
+    ) env.links ;
+  node
+
+let record dot node ?rounded ?attr box =
+  ignore (irecord dot ~id:node ?rounded ?attr box)
+
+(* -------------------------------------------------------------------------- *)
+(* --- Indexing                                                           --- *)
+(* -------------------------------------------------------------------------- *)
+
+module type Map =
+sig
+  type key
+  type 'a t
+  val empty : 'a t
+  val find : key -> 'a t -> 'a
+  val add : key -> 'a -> 'a t -> 'a t
+end
+
+let kp = ref 0
+
+module Node(M : Map) =
+struct
+  type t = M.key
+
+  let kid = ref 0
+  let prefix = ref None
+  let skip _ _ = ()
+  let once = ref skip
+  let index : node M.t ref = ref M.empty
+
+  let get_prefix () =
+    match !prefix with Some p -> p | None ->
+      let k = !kp in
+      incr kp ;
+      let p =
+        if k < 26 then String.make 1 (char_of_int (int_of_char 'A' + k))
+        else Printf.sprintf "A%d_" k in
+      prefix := Some p ; p
+
+  let get a =
+    try M.find a !index
+    with Not_found ->
+      let k = !kid in incr kid ;
+      let prefix = get_prefix () in
+      let node = Printf.sprintf "%s%03d" prefix k in
+      index := M.add a node !index ; !once a node ; node
+
+  let node dot a attr = node dot (get a) attr
+  let inode dot a attr = inode dot ~id:(get a) attr
+
+  let record dot a ?rounded ?attr box =
+    record dot (get a) ?rounded ?attr box
+
+  let irecord dot a ?rounded ?attr box =
+    irecord dot ~id:(get a) ?rounded ?attr box
+
+  let prefix p = prefix := Some p
+
+  let once f = once := f
+  let push dot f = once (fun a n -> push dot (fun () -> f a n))
+  let clear () = index := M.empty ; kid := 0 ; once skip
+end
+
+(* -------------------------------------------------------------------------- *)
+(* --- Decorator                                                          --- *)
+(* -------------------------------------------------------------------------- *)
+
+type buffer = {
+  label : Buffer.t ;
+  mutable attributes : attr list ;
+}
+
+let apply_label buffer = function
+  | `Label txt ->
+    let buf = buffer.label in
+    Buffer.add_string buf txt ; true
+  | _ -> false
+
+let apply buffer a =
+  if not (apply_label buffer a) then
+    let name = fst (attr a) in
+    let filter name a = fst (attr a) <> name in
+    buffer.attributes <- a :: List.filter (filter name) buffer.attributes
+
+let add_attr buffer a = List.iter (apply buffer) a
+
+let buffer attr =
+  let buffer = { label = Buffer.create 20 ; attributes = [] } in
+  add_attr buffer attr ; buffer
+
+let add_char buffer = Buffer.add_char buffer.label
+let add_label buffer = Buffer.add_string buffer.label
+let bprintf buffer msg =
+  let fmt = Format.formatter_of_buffer buffer.label in
+  Format.kfprintf (fun fmt -> Format.pp_print_flush fmt ()) fmt msg
+
+let attributes buffer =
+  let label = Buffer.contents buffer.label in
+  `Label label :: List.rev buffer.attributes
+
+let add_options buffer options =
+  List.iter (fun (b,a) -> if b then add_attr buffer a) options
+
+let decorate attr options =
+  let buffer = buffer attr in
+  add_options buffer options ; attributes buffer
+
+(* -------------------------------------------------------------------------- *)
diff --git a/src/libraries/utils/dotgraph.mli b/src/libraries/utils/dotgraph.mli
new file mode 100644
index 0000000000000000000000000000000000000000..778b90740259b2e4109a5ea98cc3e7a04c495f99
--- /dev/null
+++ b/src/libraries/utils/dotgraph.mli
@@ -0,0 +1,280 @@
+(**************************************************************************)
+(*                                                                        *)
+(*  This file is part of Frama-C.                                         *)
+(*                                                                        *)
+(*  Copyright (C) 2007-2019                                               *)
+(*    CEA (Commissariat à l'énergie atomique et aux énergies              *)
+(*         alternatives)                                                  *)
+(*                                                                        *)
+(*  you can redistribute it and/or modify it under the terms of the GNU   *)
+(*  Lesser General Public License as published by the Free Software       *)
+(*  Foundation, version 2.1.                                              *)
+(*                                                                        *)
+(*  It is distributed in the hope that it will be useful,                 *)
+(*  but WITHOUT ANY WARRANTY; without even the implied warranty of        *)
+(*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *)
+(*  GNU Lesser General Public License for more details.                   *)
+(*                                                                        *)
+(*  See the GNU Lesser General Public License version 2.1                 *)
+(*  for more details (enclosed in the file licenses/LGPLv2.1).            *)
+(*                                                                        *)
+(**************************************************************************)
+
+(** Helper for Printing Dot-graphs.
+
+    This module provides smart-constructors for constructing Dot input
+    files. Basically, a [dot] object is a buffer to a [<file.dot>] on
+    disk where smart constructors write Dot statements.
+
+    Once the [<file.dot>] has been created, it is possible to layout it
+    by running the [dot] command with various engines.
+
+    Typically, let say you have a graph with nodes of type
+    [M.elt] with [M : Map.S] and assumes the graph is stored
+    as a map [graph : M.elt list M.t] with [roots : M.elt list]
+    then you can use:
+
+    {[
+      let module G = Dotgraph in
+      let module N = G.Node(M) in
+      begin
+        let dot = G.open_dot ~name:"mygraph" () in
+        (* For each generated node, declare it and link to its children. *)
+        N.push dot
+          (fun a ->
+             let na = N.inode dot a in
+             try
+               List.iter
+                 (fun b -> G.link dot na (N.get b))
+                 (M.find a graph)
+             with Not_found -> ()) ;
+        (* Starts by emitting roots *)
+        List.iter
+          (fun r -> ignore (N.get r))
+          roots ;
+        (* Proceeds to the traversal *)
+        G.pop_all dot ;
+        (* You may then complete your graph
+           with other decorations after the traversal... *)
+        G.close dot ;
+        (* Now call the layout engine, if installed. *)
+        G.layout dot ~format:"pdf" () ;
+      end
+    ]}
+
+*)
+
+open Pretty_utils
+
+(** {1 Attributes} *)
+
+type attr = [
+  | `LR
+  | `TB
+  | `Label of string
+  | `Color of string
+  | `Fillcolor of string
+  | `Shape of string
+  | `Style of string
+  | `Circle
+  | `Box
+  | `Oval
+  | `Point
+  | `Dotted
+  | `Filled
+  | `ArrowBoth
+  | `ArrowBack
+  | `ArrowForward
+  | `ArrowHead of string
+  | `ArrowTail of string
+  | `Attr of string * string
+]
+
+val pp_attr : Format.formatter -> attr -> unit
+
+(** {1 Dot Ouput} *)
+
+type dot
+(** Buffer to a [dot] file with a graph environment (nodes, edges, etc.) *)
+
+val open_dot : ?name:string -> ?attr:attr list -> ?file:string -> unit -> dot
+val close : dot -> unit
+
+val is_dot_installed : unit -> bool
+(** Memoized *)
+
+val layout :
+  ?force:bool ->
+  ?target:string ->
+  ?engine:string ->
+  ?output:string -> dot -> string
+(** Invoke [dot] command (if installed) with specified target and engine.
+    Defaults are [~force:false], [~target:"pdf"], [~engine:"dot"].
+
+    The [dot] buffer must be {i closed} before being laid out, although you can
+    invoke several layouts.
+
+    Output is stored in [~output] or in a filename derived from the one of [dot].
+    The function returns the generated output file.
+
+    @raise Invalid_argument if [dot] buffer is not closed, or when [dot]
+    command fails and [~force:true] (not by default). *)
+
+val printf : dot -> ('a,Format.formatter,unit,unit) format4 -> 'a
+(** Low-level routine to directly write material in the [dot] file *)
+
+val println : dot -> ('a,Format.formatter,unit,unit) format4 -> 'a
+(** Low-level routine to directly write material with an end-of-line (["\n"])
+    in the [dot] file *)
+
+val flush : dot -> unit
+(** Flushes the [dot] file buffer to disk. *)
+
+(** {1 Nodes and Edges} *)
+
+type node
+
+(** Set default node attributes *)
+val node_default : dot -> attr list -> unit
+
+(** Set default edge attributes *)
+val edge_default : dot -> attr list -> unit
+
+(** Create a fresh node identifier *)
+val fresh : ?prefix:string -> dot -> node
+
+val pp_node : node formatter
+val pp_edge : (node * node) formatter (** [a -> b] *)
+
+(** Set attributes to node *)
+val node : dot -> node -> attr list -> unit
+
+(** Create an edge with attributes *)
+val edge : dot -> node -> node -> attr list -> unit
+
+(** Link the node sequence with attributed edges *)
+val link : dot -> node list -> attr list -> unit
+
+(** Combinaison of [fresh] and [node] *)
+val inode : dot -> ?prefix:string -> ?id:node -> attr list -> node
+
+(** {1 Clustering} *)
+
+val rank : dot -> node list -> unit
+(** Layout nodes at the same rank *)
+
+val subgraph : dot -> ?cluster:bool -> attr list -> (unit -> unit) -> unit
+(** The continuation shall add the graph content in the [dot] file.
+    Clustering is true by default *)
+
+type record = [
+  | `Empty
+  | `Hbox of record list
+  | `Vbox of record list
+  | `Label of string
+  | `Port of string * link list * string
+  (** Port with output edges to other nodes.
+      Use [Record.link] and [Record.label] smart-constructors to create
+      complex ports. *)
+] and link = string * attr list * node
+
+(** Complex node layout. Smart constructors to create records. *)
+module Record :
+sig
+  val (<->) : record -> record -> record
+  val (<|>) : record -> record -> record
+  val link : ?anchor:string -> ?attr:attr list -> node -> link
+  val label : ?port:string -> ?link:link list -> string -> record
+end
+
+(** Create a port to a node, and returns the associated pseudo-node so you
+    can link an edge to it. *)
+val port : node -> string -> node
+
+(** Define the node to be a record *)
+val record : dot -> node ->
+  ?rounded:bool -> ?attr:attr list -> record -> unit
+
+(** Create a new node from a record (combines [fresh] and [record]) *)
+val irecord : dot -> ?prefix:string -> ?id:node ->
+  ?rounded:bool -> ?attr:attr list -> record -> node
+
+(** {1 Node Indexing} *)
+
+module type Map =
+sig
+  type key
+  type 'a t
+  val empty : 'a t
+  val find : key -> 'a t -> 'a
+  val add : key -> 'a -> 'a t -> 'a t
+end
+
+(** Lazily associates a node to any element. *)
+module Node(M : Map) :
+sig
+  type t = M.key
+  val get : t -> node
+  val node : dot -> t -> attr list -> unit
+  val inode : dot -> t -> attr list -> node
+  val record : dot -> t -> ?rounded:bool -> ?attr:attr list -> record -> unit
+  val irecord : dot -> t -> ?rounded:bool -> ?attr:attr list -> record -> node
+  val clear : unit -> unit
+
+  (** Executes the callback {i once} for all created nodes.
+      Any previously registered callback by [once] or [push] is replaced
+      by the new one.
+
+      {b Warning:} the callback is executed as soon as [get] is called
+      for the first time, possibly interfering with your current output
+      on a [dot] buffer. To insert additional Dot material with a callback,
+      use [push] instead.
+  *)
+  val once : (t -> node -> unit) -> unit
+
+  (** Pushes the callback {i once} for all created nodes.
+      You must call [pop_call] at some point to flush them.
+      Any previsously registred callback by [once] or [push] is replaced
+      by the new one. *)
+  val push : dot -> (t -> node -> unit) -> unit
+
+  (** Set node prefix.
+      Otherwize, some default one is created for each functor application. *)
+  val prefix : string -> unit
+end
+
+(** Register a continuation to be executed later. *)
+val push : dot -> (unit -> unit) -> unit
+
+(** Flushes all pending continuations. *)
+val pop_all : dot -> unit
+
+(** {1 Decorator} *)
+
+(** A text buffer to compose labels and attributes.
+    You can add text and attributes to the buffer, and finally
+    flush it by calling [attributes].
+    A single [`Label] attribute is finally emitted with
+    all the added text (if non-empty). *)
+type buffer
+
+(** Create a buffer initialized with the given attributes. *)
+val buffer : attr list -> buffer
+
+(** Add text material to buffer label. *)
+val bprintf : buffer -> ('a,Format.formatter,unit,unit) format4 -> 'a
+
+val add_char : buffer -> char -> unit
+val add_label : buffer -> string -> unit
+
+(** Add attributes to the buffer. *)
+val add_attr : buffer -> attr list -> unit
+
+(** Only add attributes with a [true] boolean flag *)
+val add_options : buffer -> (bool * attr list) list -> unit
+
+(** Flushes the buffer into a list of attributes *)
+val attributes : buffer -> attr list
+
+(** Concat the attributes with flagged ones *)
+val decorate : attr list -> ( bool * attr list ) list -> attr list
diff --git a/src/libraries/utils/filepath.ml b/src/libraries/utils/filepath.ml
index 0e774657a818b0b5f91cafba14f4b21197480427..eb86c85f6945e397d7e3b1af0e3959c71faa5500 100644
--- a/src/libraries/utils/filepath.ml
+++ b/src/libraries/utils/filepath.ml
@@ -224,6 +224,7 @@ module Normalized = struct
   let pretty fmt p = Format.fprintf fmt "%s" (pretty p)
   let pp_abs fmt p = Format.fprintf fmt "%s" p
   let unknown = normalize ""
+  let is_unknown fp = equal fp unknown
 end
 
 type position =
diff --git a/src/libraries/utils/filepath.mli b/src/libraries/utils/filepath.mli
index 6ea46d821541a1de524e1baad38bbadcf6339b54..74eef31bbdbb8d6f2ff340258beffeef3539cd65 100644
--- a/src/libraries/utils/filepath.mli
+++ b/src/libraries/utils/filepath.mli
@@ -120,6 +120,9 @@ module Normalized: sig
 
   (** Unknown filepath, used as 'dummy' for [Datatype.Filepath]. *)
   val unknown: t
+
+  (** @since Frama-C+dev *)
+  val is_unknown: t -> bool
 end
 
 (** Describes a position in a source file.
diff --git a/src/plugins/gui/design.ml b/src/plugins/gui/design.ml
index bb11d8d56877746f7cea5b12005396ff6ddbeaad..287a5d8e0b01a50e36a5dd37f0f4ea3ba4d3209f 100644
--- a/src/plugins/gui/design.ml
+++ b/src/plugins/gui/design.ml
@@ -1863,7 +1863,9 @@ let toplevel play =
             Task.on_idle :=
               (fun f -> ignore (Glib.Timeout.add ~ms:50 ~callback:f));
             let project_name = Gui_parameters.Project_name.get () in
-            if project_name <> "" then
+            if project_name = "" then
+              Project.set_current_as_last_created ()
+            else
               Project.set_current (Project.from_unique_name project_name);
             Ast.compute ()
           with e -> (* An error occurred: we need to enforce the splash screen
diff --git a/src/plugins/value/engine/transfer_specification.ml b/src/plugins/value/engine/transfer_specification.ml
index 5515e44a3b884a7281f00342677c689c8333baa1..d5438b227de994181748fcdadfd7c43f3a49b05b 100644
--- a/src/plugins/value/engine/transfer_specification.ml
+++ b/src/plugins/value/engine/transfer_specification.ml
@@ -138,6 +138,7 @@ let reduce_to_valid_location out loc =
       begin
         if is_assigns out && not (Locations.is_bottom_loc loc) then
           Value_parameters.warning ~current:true ~once:true
+            ~wkey:Value_parameters.wkey_invalid_assigns
             "@[Completely invalid destination@ for %a.@ \
              Ignoring.@]" pp_assign_free_alloc out;
         None
diff --git a/src/plugins/value/legacy/eval_terms.ml b/src/plugins/value/legacy/eval_terms.ml
index e5c0bb41a9370ff6c99362bf8a1a5ca7992ea761..ce9a20a55a312fa437321e4c9a52affec6a9aa2e 100644
--- a/src/plugins/value/legacy/eval_terms.ml
+++ b/src/plugins/value/legacy/eval_terms.ml
@@ -1919,20 +1919,27 @@ let rec reduce_by_predicate ~alarm_mode env positive p =
           (* See comments in the code for the evaluation of Pinitialized *)
           let star_tsets = deref_tsets tsets in
           let rlocb = eval_tlval ~alarm_mode env star_tsets in
-          let size = Eval_typ.sizeof_lval_typ rlocb.etype in
-          let state = env_state env lbl_initialized in
-          let fred = match p_content with
-            | Pinitialized _ -> V_Or_Uninitialized.reduce_by_initializedness
-            | Pdangling _ -> V_Or_Uninitialized.reduce_by_danglingness
-            | _ -> assert false
-          in
-          let fred = Eval_op.reduce_by_initialized_defined (fred positive) in
-          let state_reduced =
-            let loc = make_loc rlocb.eunder size in
-            let loc = Eval_op.make_loc_contiguous loc in
-            Eval_op.apply_on_all_locs fred loc state
-          in
-          overwrite_state env state_reduced lbl_initialized
+          (* No reduction on negations of \initialized or \dangling on multiple
+             locations: at least one of them is non initialized/dangling, but
+             which one? Reduction would only be possible in the rare case where
+             only one of the locations might be non initialized/dangling. *)
+          if not (positive || Location_Bits.cardinal_zero_or_one rlocb.eover)
+          then env
+          else
+            let size = Eval_typ.sizeof_lval_typ rlocb.etype in
+            let state = env_state env lbl_initialized in
+            let fred = match p_content with
+              | Pinitialized _ -> V_Or_Uninitialized.reduce_by_initializedness
+              | Pdangling _ -> V_Or_Uninitialized.reduce_by_danglingness
+              | _ -> assert false
+            in
+            let fred = Eval_op.reduce_by_initialized_defined (fred positive) in
+            let state_reduced =
+              let loc = make_loc rlocb.eunder size in
+              let loc = Eval_op.make_loc_contiguous loc in
+              Eval_op.apply_on_all_locs fred loc state
+            in
+            overwrite_state env state_reduced lbl_initialized
         with LogicEvalError _ -> env
       end
 
diff --git a/src/plugins/value/value_parameters.ml b/src/plugins/value/value_parameters.ml
index 5471f4b343ffdf7d312a50bdf3420e27ebf77731..5eb874d2566e44e9e243f6d50027377060d47373 100644
--- a/src/plugins/value/value_parameters.ml
+++ b/src/plugins/value/value_parameters.ml
@@ -98,6 +98,8 @@ let () = set_warn_status wkey_missing_loop_unroll Log.Winactive
 let wkey_missing_loop_unroll_for = register_warn_category "missing-loop-unroll:for"
 let () = set_warn_status wkey_missing_loop_unroll_for Log.Winactive
 let wkey_signed_overflow = register_warn_category "signed-overflow"
+let wkey_invalid_assigns = register_warn_category "invalid-assigns"
+let () = set_warn_status wkey_invalid_assigns Log.Wfeedback
 
 module ForceValues =
   WithOutput
diff --git a/src/plugins/value/value_parameters.mli b/src/plugins/value/value_parameters.mli
index fe24f33b22e81e956807ffba17fae01f421217a2..a5f62e9cff6d10c4d4ebaef1a4dae9da8b8294f2 100644
--- a/src/plugins/value/value_parameters.mli
+++ b/src/plugins/value/value_parameters.mli
@@ -204,6 +204,9 @@ val wkey_missing_loop_unroll_for : warn_category
 (** Warning category for signed overflows *)
 val wkey_signed_overflow : warn_category
 
+(** Warning category for 'completely invalid' assigns clause *)
+val wkey_invalid_assigns : warn_category
+
 (** Debug category used to print information about invalid pointer comparisons*)
 val dkey_pointer_comparison: category
 
diff --git a/src/plugins/variadic/tests/known/oracle/ioctl.res.oracle b/src/plugins/variadic/tests/known/oracle/ioctl.res.oracle
index 834a0dde5f6db9f527a604df92cba0f2fa2a7226..195847c4a798c48e04d7613f0a9ff77e106c4001 100644
--- a/src/plugins/variadic/tests/known/oracle/ioctl.res.oracle
+++ b/src/plugins/variadic/tests/known/oracle/ioctl.res.oracle
@@ -13,6 +13,9 @@
 [eva] Initial state computed
 [eva] using specification for function __va_ioctl_void
 [eva] using specification for function __va_ioctl_ptr
+[eva:invalid-assigns] tests/known/ioctl.c:17: 
+  Completely invalid destination for assigns clause *((char *)argp + (0 ..)).
+  Ignoring.
 [eva] using specification for function __va_ioctl_int
 [eva] done for function main
 [eva] ====== VALUES COMPUTED ======
diff --git a/src/plugins/wp/wp_parameters.ml b/src/plugins/wp/wp_parameters.ml
index 877191a8d974c87bdc90c050a25ebcf2e32609da..1c7b3d303cc981c5a84a939d39327c9ae7f12276 100644
--- a/src/plugins/wp/wp_parameters.ml
+++ b/src/plugins/wp/wp_parameters.ml
@@ -984,7 +984,7 @@ let base_output () =
               | dir ->
                   make_output_dir dir ; dir in
       base_output := Some output;
-      Filepath.add_symbolic_dir "WPOUT" output ;
+      Fc_Filepath.add_symbolic_dir "WPOUT" output ;
       output
   | Some output -> output
 
@@ -1026,7 +1026,7 @@ let has_print_generated () = has_dkey cat_print_generated
 
 let print_generated ?header file =
   let header = match header with
-    | None -> Filepath.Normalized.to_pretty_string (Datatype.Filepath.of_string file)
+    | None -> Fc_Filepath.Normalized.to_pretty_string (Datatype.Filepath.of_string file)
     | Some head -> head in
   debug ~dkey:cat_print_generated "%S@\n%t@." header
     begin fun fmt ->
diff --git a/tests/builtins/oracle/imprecise.res.oracle b/tests/builtins/oracle/imprecise.res.oracle
index e27e83c8af3557509f71e9dee55793438f58b39d..35235461d13ed2ecd8659d12fcde6992bb47ce76 100644
--- a/tests/builtins/oracle/imprecise.res.oracle
+++ b/tests/builtins/oracle/imprecise.res.oracle
@@ -39,7 +39,7 @@
 [eva] computing for function f <- invalid_assigns_imprecise <- main.
   Called from tests/builtins/imprecise.c:11.
 [eva] using specification for function f
-[eva] tests/builtins/imprecise.c:11: Warning: 
+[eva:invalid-assigns] tests/builtins/imprecise.c:11: 
   Completely invalid destination for assigns clause *p. Ignoring.
 [eva] Done for function f
 [eva] Recording results for invalid_assigns_imprecise
diff --git a/tests/builtins/oracle/memchr.res.oracle b/tests/builtins/oracle/memchr.res.oracle
index 94cae8ab088a9ed33b7c3ab50eb097a755f62178..d77c490f0c7d086f6a2150c57f9c6585a7e3c1f2 100644
--- a/tests/builtins/oracle/memchr.res.oracle
+++ b/tests/builtins/oracle/memchr.res.oracle
@@ -100,8 +100,8 @@
 [eva] tests/builtins/memchr.c:122: Call to builtin memchr
 [eva:alarm] tests/builtins/memchr.c:122: Warning: 
   function memchr: precondition 'valid' got status unknown.
-[eva] tests/builtins/memchr.c:122: 
-  function memchr: precondition 'initialization' got status valid.
+[eva:alarm] tests/builtins/memchr.c:122: Warning: 
+  function memchr: precondition 'initialization' got status unknown.
 [eva:alarm] tests/builtins/memchr.c:122: Warning: 
   function memchr: precondition 'danglingness' got status unknown.
 [eva] tests/builtins/memchr.c:128: Call to builtin memchr
@@ -134,8 +134,8 @@
 [eva] tests/builtins/memchr.c:145: Call to builtin memchr
 [eva] tests/builtins/memchr.c:145: 
   function memchr: precondition 'valid' got status valid.
-[eva] tests/builtins/memchr.c:145: 
-  function memchr: precondition 'initialization' got status valid.
+[eva:alarm] tests/builtins/memchr.c:145: Warning: 
+  function memchr: precondition 'initialization' got status unknown.
 [eva] tests/builtins/memchr.c:145: 
   function memchr: precondition 'danglingness' got status valid.
 [eva] tests/builtins/memchr.c:145: Frama_C_show_each_mymemchr: {1}
@@ -143,8 +143,8 @@
 [eva] tests/builtins/memchr.c:152: Call to builtin memchr
 [eva] tests/builtins/memchr.c:152: 
   function memchr: precondition 'valid' got status valid.
-[eva] tests/builtins/memchr.c:152: 
-  function memchr: precondition 'initialization' got status valid.
+[eva:alarm] tests/builtins/memchr.c:152: Warning: 
+  function memchr: precondition 'initialization' got status unknown.
 [eva] tests/builtins/memchr.c:152: 
   function memchr: precondition 'danglingness' got status valid.
 [eva] tests/builtins/memchr.c:152: Frama_C_show_each_mymemchr: {3}
@@ -443,8 +443,8 @@
 [eva] tests/builtins/memchr.c:348: Call to builtin memchr
 [eva] tests/builtins/memchr.c:348: 
   function memchr: precondition 'valid' got status valid.
-[eva] tests/builtins/memchr.c:348: 
-  function memchr: precondition 'initialization' got status valid.
+[eva:alarm] tests/builtins/memchr.c:348: Warning: 
+  function memchr: precondition 'initialization' got status unknown.
 [eva] tests/builtins/memchr.c:348: 
   function memchr: precondition 'danglingness' got status valid.
 [eva] tests/builtins/memchr.c:348: Frama_C_show_each_mymemchr: {1}
diff --git a/tests/libc/oracle/fc_libc.1.res.oracle b/tests/libc/oracle/fc_libc.1.res.oracle
index 107f92c67f3f62604ce53ca25bf5a5672e0c65be..417ea214e24ff7f7103182badf324309869ad81f 100644
--- a/tests/libc/oracle/fc_libc.1.res.oracle
+++ b/tests/libc/oracle/fc_libc.1.res.oracle
@@ -3145,10 +3145,9 @@ struct sigaction *__fc_p_sigaction = __fc_sigaction;
         \old(oldact) ≡ \null ∨
         *\old(oldact) ∈ *(__fc_p_sigaction + \old(signum));
     ensures result_ok_or_error: \result ≡ 0 ∨ \result ≡ -1;
-    assigns oldact ≡ \null? \empty: *oldact,
-            act ≡ \null? \empty: *(__fc_p_sigaction + signum), \result;
-    assigns oldact ≡ \null? \empty: *oldact \from __fc_p_sigaction;
-    assigns act ≡ \null? \empty: *(__fc_p_sigaction + signum) \from *act;
+    assigns *oldact, *(__fc_p_sigaction + signum), \result;
+    assigns *oldact \from __fc_p_sigaction;
+    assigns *(__fc_p_sigaction + signum) \from *act;
     assigns \result
       \from (indirect: signum), (indirect: act), (indirect: *act),
             (indirect: oldact), (indirect: *oldact);
@@ -3166,11 +3165,10 @@ extern int sigaction(int signum, struct sigaction const * __restrict act,
       initialization: oldset_initialized:
         \old(oldset) ≢ \null ∧ \result ≡ 0 ⇒
         \initialized(\old(oldset));
-    assigns \result, oldset ≡ \null? \empty: *oldset;
+    assigns \result, *oldset;
     assigns \result
       \from (indirect: how), (indirect: set), (indirect: oldset);
-    assigns oldset ≡ \null? \empty: *oldset
-      \from (indirect: how), (indirect: oldset);
+    assigns *oldset \from (indirect: how), (indirect: oldset);
  */
 extern int sigprocmask(int how, sigset_t const * __restrict set,
                        sigset_t * __restrict oldset);
@@ -5840,11 +5838,11 @@ axiomatic nanosleep_predicates {
       ensures
         remaining_valid:
           \old(rmtp) ≢ \null ⇒ 0 ≤ \old(rmtp)->tv_nsec < 1000000000;
-      assigns \result, rmtp ≡ \null? \empty: *rmtp;
+      assigns \result, *rmtp;
       assigns \result
         \from (indirect: __fc_time), (indirect: clock_id), (indirect: rqtp),
               (indirect: *rqtp);
-      assigns rmtp ≡ \null? \empty: *rmtp
+      assigns *rmtp
         \from __fc_time, (indirect: clock_id), (indirect: rqtp),
               (indirect: *rqtp), (indirect: rmtp);
     
@@ -5901,10 +5899,10 @@ extern int clock_nanosleep(clockid_t clock_id, int flags,
       interrupted_remaining_valid:
         \old(rmtp) ≢ \null ∧ \result ≡ -1 ⇒
         0 ≤ \old(rmtp)->tv_nsec < 1000000000;
-    assigns \result, rmtp ≡ \null? \empty: *rmtp;
+    assigns \result, *rmtp;
     assigns \result
       \from (indirect: __fc_time), (indirect: rqtp), (indirect: *rqtp);
-    assigns rmtp ≡ \null? \empty: *rmtp
+    assigns *rmtp
       \from (indirect: __fc_time), (indirect: rqtp), (indirect: *rqtp),
             (indirect: rmtp);
  */
@@ -6753,8 +6751,8 @@ extern int getitimer(int which, struct itimerval *curr_value);
     requires
       old_value_null_or_valid: old_value ≡ \null ∨ \valid(old_value);
     ensures result_ok_or_error: \result ≡ 0 ∨ \result ≡ -1;
-    assigns old_value ≢ \null? *old_value: \empty, \result;
-    assigns old_value ≢ \null? *old_value: \empty
+    assigns *old_value, \result;
+    assigns *old_value
       \from (indirect: which), (indirect: old_value), (indirect: new_value),
             __fc_itimer_real, __fc_itimer_virtual, __fc_itimer_prof;
     assigns \result
@@ -6767,10 +6765,9 @@ extern int getitimer(int which, struct itimerval *curr_value);
           0 ≤ new_value->it_interval.tv_usec ≤ 999999;
       ensures result_ok: \result ≡ 0;
       ensures initialization: old_value: \initialized(\old(old_value));
-      assigns \result, old_value ≢ \null? *old_value: \empty,
-              __fc_itimer_real;
+      assigns \result, *old_value, __fc_itimer_real;
       assigns \result \from \nothing;
-      assigns old_value ≢ \null? *old_value: \empty \from __fc_itimer_real;
+      assigns *old_value \from __fc_itimer_real;
       assigns __fc_itimer_real \from *new_value;
     
     behavior virtual:
@@ -6780,10 +6777,9 @@ extern int getitimer(int which, struct itimerval *curr_value);
           0 ≤ new_value->it_interval.tv_usec ≤ 999999;
       ensures result_ok: \result ≡ 0;
       ensures initialization: old_value: \initialized(\old(old_value));
-      assigns \result, old_value ≢ \null? *old_value: \empty;
+      assigns \result, *old_value;
       assigns \result \from \nothing;
-      assigns old_value ≢ \null? *old_value: \empty
-        \from __fc_itimer_virtual;
+      assigns *old_value \from __fc_itimer_virtual;
     
     behavior prof:
       assumes
@@ -6792,9 +6788,9 @@ extern int getitimer(int which, struct itimerval *curr_value);
           0 ≤ new_value->it_interval.tv_usec ≤ 999999;
       ensures result_ok: \result ≡ 0;
       ensures initialization: old_value: \initialized(\old(old_value));
-      assigns \result, old_value ≢ \null? *old_value: \empty;
+      assigns \result, *old_value;
       assigns \result \from \nothing;
-      assigns old_value ≢ \null? *old_value: \empty \from __fc_itimer_prof;
+      assigns *old_value \from __fc_itimer_prof;
     
     behavior invalid:
       assumes
@@ -6818,27 +6814,25 @@ extern int setitimer(int which,
     requires writefds: writefds ≡ \null ∨ \valid(writefds);
     requires errorfds: errorfds ≡ \null ∨ \valid(errorfds);
     requires timeout: timeout ≡ \null ∨ \valid(timeout);
-    assigns __fc_fds_state, readfds ≡ \null? \empty: *readfds,
-            writefds ≡ \null? \empty: *writefds,
-            errorfds ≡ \null? \empty: *errorfds,
-            timeout ≡ \null? \empty: *timeout, \result;
+    assigns __fc_fds_state, *readfds, *writefds, *errorfds, *timeout,
+            \result;
     assigns __fc_fds_state \from __fc_fds_state;
-    assigns readfds ≡ \null? \empty: *readfds
+    assigns *readfds
       \from (indirect: nfds), (indirect: readfds), (indirect: *readfds),
             (indirect: writefds), (indirect: *writefds),
             (indirect: errorfds), (indirect: *errorfds), (indirect: timeout),
             (indirect: *timeout), __fc_fds_state;
-    assigns writefds ≡ \null? \empty: *writefds
+    assigns *writefds
       \from (indirect: nfds), (indirect: readfds), (indirect: *readfds),
             (indirect: writefds), (indirect: *writefds),
             (indirect: errorfds), (indirect: *errorfds), (indirect: timeout),
             (indirect: *timeout), __fc_fds_state;
-    assigns errorfds ≡ \null? \empty: *errorfds
+    assigns *errorfds
       \from (indirect: nfds), (indirect: readfds), (indirect: *readfds),
             (indirect: writefds), (indirect: *writefds),
             (indirect: errorfds), (indirect: *errorfds), (indirect: timeout),
             (indirect: *timeout), __fc_fds_state;
-    assigns timeout ≡ \null? \empty: *timeout
+    assigns *timeout
       \from (indirect: nfds), (indirect: readfds), (indirect: *readfds),
             (indirect: writefds), (indirect: *writefds),
             (indirect: errorfds), (indirect: *errorfds), (indirect: timeout),
@@ -7129,11 +7123,11 @@ extern int __va_ioctl_void(int fd, int request);
  */
 extern int __va_ioctl_int(int fd, int request, int arg);
 
-/*@ assigns \result, argp ≢ \null? *((char *)argp + (0 ..)): \empty;
+/*@ assigns \result, *((char *)argp + (0 ..));
     assigns \result
       \from (indirect: fd), (indirect: request),
             (indirect: *((char *)argp + (0 ..)));
-    assigns argp ≢ \null? *((char *)argp + (0 ..)): \empty
+    assigns *((char *)argp + (0 ..))
       \from (indirect: fd), (indirect: request), *((char *)argp + (0 ..));
  */
 extern int __va_ioctl_ptr(int fd, int request, void *argp);
diff --git a/tests/libc/oracle/libgen_h.res.oracle b/tests/libc/oracle/libgen_h.res.oracle
index ca158666adf08acb2890a53b2ae8fecfddb3e132..a5ac900a0e19e23678ca830eeef10d3ef012d55b 100644
--- a/tests/libc/oracle/libgen_h.res.oracle
+++ b/tests/libc/oracle/libgen_h.res.oracle
@@ -15,7 +15,7 @@
   Called from tests/libc/libgen_h.c:11.
 [eva] tests/libc/libgen_h.c:11: 
   function basename: precondition 'null_or_valid_string_path' got status valid.
-[eva] tests/libc/libgen_h.c:11: Warning: 
+[eva:invalid-assigns] tests/libc/libgen_h.c:11: 
   Completely invalid destination for assigns clause *(path + (0 ..)). Ignoring.
 [eva] Done for function basename
 [eva:alarm] tests/libc/libgen_h.c:12: Warning: assertion got status unknown.
@@ -30,7 +30,7 @@
   Called from tests/libc/libgen_h.c:16.
 [eva] tests/libc/libgen_h.c:16: 
   function dirname: precondition 'null_or_valid_string_path' got status valid.
-[eva] tests/libc/libgen_h.c:16: Warning: 
+[eva:invalid-assigns] tests/libc/libgen_h.c:16: 
   Completely invalid destination for assigns clause *(path + (0 ..)). Ignoring.
 [eva] Done for function dirname
 [eva:alarm] tests/libc/libgen_h.c:17: Warning: assertion got status unknown.
diff --git a/tests/libc/oracle/signal_h.res.oracle b/tests/libc/oracle/signal_h.res.oracle
index 42c80f2642d6fdde9b66d2ab9487b73f4ff7b8a8..9d071e5884c3e373da23a02893036f0d8bbbaee3 100644
--- a/tests/libc/oracle/signal_h.res.oracle
+++ b/tests/libc/oracle/signal_h.res.oracle
@@ -83,6 +83,8 @@
   function sigprocmask: precondition 'valid_oldset_or_null' got status valid.
 [eva] tests/libc/signal_h.c:35: 
   function sigprocmask: precondition 'separation' got status valid.
+[eva:invalid-assigns] tests/libc/signal_h.c:35: 
+  Completely invalid destination for assigns clause *oldset. Ignoring.
 [eva] Done for function sigprocmask
 [eva] computing for function sigprocmask <- main.
   Called from tests/libc/signal_h.c:38.
@@ -137,6 +139,8 @@
   function sigaction: precondition 'valid_read_act_or_null' got status valid.
 [eva] tests/libc/signal_h.c:51: 
   function sigaction: precondition 'separation,separated_acts' got status valid.
+[eva:invalid-assigns] tests/libc/signal_h.c:51: 
+  Completely invalid destination for assigns clause *oldact. Ignoring.
 [eva] Done for function sigaction
 [eva] Recording results for main
 [eva] done for function main
@@ -176,9 +180,7 @@
                 [15]{.sa_handler; .sa_sigaction} ∈ {0}
                 [15]{.sa_mask; .sa_flags} ∈ [--..--]
                 [16]{.sa_handler; .sa_sigaction} ∈ {0}
-                [16]{.sa_mask; .sa_flags} ∈ [--..--]
-                [17]{.sa_handler; .sa_sigaction} ∈ {0}
-                [17]{.sa_mask; .sa_flags} ∈ [--..--]
+                {[16]{.sa_mask; .sa_flags}; [17]} ∈ [--..--]
                 [18] ∈
                 {{ garbled mix of &{__fc_sigaction}
                  (origin: Library function) }}
diff --git a/tests/libc/oracle/string_h.res.oracle b/tests/libc/oracle/string_h.res.oracle
index 75780c1ba4a10561b67f6deb39440fddbcfbf8cb..55c5afe6dad117fc0beb325fa0f45052db727460 100644
--- a/tests/libc/oracle/string_h.res.oracle
+++ b/tests/libc/oracle/string_h.res.oracle
@@ -134,7 +134,7 @@
   function strtok: precondition 'valid_string_delim' got status valid.
 [eva] tests/libc/string_h.c:72: 
   function strtok, behavior new_str: precondition 'valid_string_s_or_delim_not_found' got status valid.
-[eva] tests/libc/string_h.c:72: Warning: 
+[eva:invalid-assigns] tests/libc/string_h.c:72: 
   Completely invalid destination for assigns clause *(s + (0 ..)). Ignoring.
 [eva] Done for function strtok
 [eva:alarm] tests/libc/string_h.c:73: Warning: assertion got status unknown.
@@ -144,7 +144,7 @@
   function strtok: precondition 'valid_string_delim' got status valid.
 [eva:alarm] tests/libc/string_h.c:75: Warning: 
   function strtok, behavior new_str: precondition 'valid_string_s_or_delim_not_found' got status unknown.
-[eva] tests/libc/string_h.c:75: Warning: 
+[eva:invalid-assigns] tests/libc/string_h.c:75: 
   Completely invalid destination for assigns clause *(s + (0 ..)). Ignoring.
 [eva] Done for function strtok
 [eva:alarm] tests/libc/string_h.c:76: Warning: 
@@ -221,7 +221,7 @@
   function strtok_r: precondition 'valid_saveptr' got status valid.
 [eva] tests/libc/string_h.c:102: 
   function strtok_r, behavior new_str: precondition 'valid_string_s_or_delim_not_found' got status valid.
-[eva] tests/libc/string_h.c:102: Warning: 
+[eva:invalid-assigns] tests/libc/string_h.c:102: 
   Completely invalid destination for assigns clause *(s + (0 ..)). Ignoring.
 [eva] Done for function strtok_r
 [eva:alarm] tests/libc/string_h.c:103: Warning: assertion got status unknown.
@@ -233,7 +233,7 @@
   function strtok_r: precondition 'valid_saveptr' got status valid.
 [eva:alarm] tests/libc/string_h.c:105: Warning: 
   function strtok_r, behavior new_str: precondition 'valid_string_s_or_delim_not_found' got status unknown.
-[eva] tests/libc/string_h.c:105: Warning: 
+[eva:invalid-assigns] tests/libc/string_h.c:105: 
   Completely invalid destination for assigns clause *(s + (0 ..)). Ignoring.
 [eva] Done for function strtok_r
 [eva:alarm] tests/libc/string_h.c:106: Warning: 
diff --git a/tests/libc/oracle/sys_select.res.oracle b/tests/libc/oracle/sys_select.res.oracle
index 41fa7f4bc4969060e4bf3300451702b172018a50..453a3465161df7694e3616278d3db5120e82a673 100644
--- a/tests/libc/oracle/sys_select.res.oracle
+++ b/tests/libc/oracle/sys_select.res.oracle
@@ -72,6 +72,10 @@
   function select: precondition 'errorfds' got status valid.
 [eva] tests/libc/sys_select.c:31: 
   function select: precondition 'timeout' got status valid.
+[eva:invalid-assigns] tests/libc/sys_select.c:31: 
+  Completely invalid destination for assigns clause *writefds. Ignoring.
+[eva:invalid-assigns] tests/libc/sys_select.c:31: 
+  Completely invalid destination for assigns clause *errorfds. Ignoring.
 [eva] Done for function select
 [eva] computing for function FD_ISSET <- main.
   Called from tests/libc/sys_select.c:32.
diff --git a/tests/libc/oracle/sys_time.res.oracle b/tests/libc/oracle/sys_time.res.oracle
index 3c7c79cab30db005eb3c7f1d6087a189d033aa56..18c18fa9f8cf4ac0e2cd2252f3a3baf3d7dcd62c 100644
--- a/tests/libc/oracle/sys_time.res.oracle
+++ b/tests/libc/oracle/sys_time.res.oracle
@@ -11,6 +11,8 @@
   function setitimer: precondition 'valid_new_value' got status valid.
 [eva] tests/libc/sys_time.c:6: 
   function setitimer: precondition 'old_value_null_or_valid' got status valid.
+[eva:invalid-assigns] tests/libc/sys_time.c:6: 
+  Completely invalid destination for assigns clause *old_value. Ignoring.
 [eva] Done for function setitimer
 [eva] tests/libc/sys_time.c:7: assertion got status valid.
 [eva] computing for function setitimer <- main.
diff --git a/tests/libc/oracle/time_h.res.oracle b/tests/libc/oracle/time_h.res.oracle
index 5fcd6c6d027bc01e634adfc623934e6e724e7607..7e8f729d1beae0a1300355d51b3f1c84f7826b6c 100644
--- a/tests/libc/oracle/time_h.res.oracle
+++ b/tests/libc/oracle/time_h.res.oracle
@@ -37,9 +37,13 @@
   function nanosleep: precondition 'valid_nanosecs' got status valid.
 [eva] tests/libc/time_h.c:22: 
   function nanosleep: precondition 'valid_remaining_or_null' got status valid.
+[eva:invalid-assigns] tests/libc/time_h.c:22: 
+  Completely invalid destination for assigns clause *rmtp. Ignoring.
 [eva] Done for function nanosleep
 [eva] computing for function nanosleep <- main.
   Called from tests/libc/time_h.c:22.
+[eva:invalid-assigns] tests/libc/time_h.c:22: 
+  Completely invalid destination for assigns clause *rmtp. Ignoring.
 [eva] Done for function nanosleep
 [eva] computing for function clock_nanosleep <- main.
   Called from tests/libc/time_h.c:28.
diff --git a/tests/syntax/flexible_array_member.i b/tests/syntax/flexible_array_member.i
index fbbd55dd676762831d5780ec467bd41a29c8c356..9fda08120828a0b2c8bde3058a345a862bb81ed3 100644
--- a/tests/syntax/flexible_array_member.i
+++ b/tests/syntax/flexible_array_member.i
@@ -1,9 +1,16 @@
 // valid flexible array member declarations
-struct {
+struct s1 {
   int size;
   char data[];
-} s1;
+} ss1;
 
-struct {
+struct s2 {
   char len, data[];
-} s2;
+} ss2;
+
+union u {
+  struct s {
+    char len;
+    char data[];
+  } fam;
+} u1;
diff --git a/tests/syntax/flexible_array_member_invalid1.i b/tests/syntax/flexible_array_member_invalid1.i
index bd4b3f7fc0beb135e67fc780b6b9bae433b41ce2..62ebdab3e24b225226ed31ca6ca17e0cdd50b33f 100644
--- a/tests/syntax/flexible_array_member_invalid1.i
+++ b/tests/syntax/flexible_array_member_invalid1.i
@@ -1,4 +1,4 @@
 // invalid flexible array member (empty struct otherwise)
-struct {
+struct s1 {
   char data[];
-} s1;
+} ss1;
diff --git a/tests/syntax/flexible_array_member_invalid2.i b/tests/syntax/flexible_array_member_invalid2.i
index e5a1463ec5432beb168dfc1b405f6213d1ca08a8..eab5dccd0c3e11ce97ad2511ab92beaeb96bc405 100644
--- a/tests/syntax/flexible_array_member_invalid2.i
+++ b/tests/syntax/flexible_array_member_invalid2.i
@@ -1,6 +1,6 @@
 // invalid flexible array member (two incomplete fields)
-struct {
+struct s {
   int len;
   char data[];
   char more_data[];
-} s;
+} ss;
diff --git a/tests/syntax/flexible_array_member_invalid3.i b/tests/syntax/flexible_array_member_invalid3.i
index 30f2a733c7ee7377ae332c86bba63b0622184dd0..4baf088c14f4a8baeba250ad8f2abe33a2b79766 100644
--- a/tests/syntax/flexible_array_member_invalid3.i
+++ b/tests/syntax/flexible_array_member_invalid3.i
@@ -1,5 +1,5 @@
 // invalid flexible array member (two incomplete fields in same field group)
-struct {
+struct s {
   int len;
   char data[], more_data[];
-} s;
+} ss;
diff --git a/tests/syntax/flexible_array_member_invalid4.i b/tests/syntax/flexible_array_member_invalid4.i
index bd7666227597316d55ba00b5b54ce9fa0b792ddd..e84cce94b9891a77d85135a75691d9f9b7ad1b87 100644
--- a/tests/syntax/flexible_array_member_invalid4.i
+++ b/tests/syntax/flexible_array_member_invalid4.i
@@ -1,6 +1,6 @@
 // invalid flexible array member (incomplete field is not last)
-struct {
+struct s {
   int len;
   char data[];
   char b;
-} s;
+} ss;
diff --git a/tests/syntax/flexible_array_member_invalid5.i b/tests/syntax/flexible_array_member_invalid5.i
index dd3eac2f1c52361a2dbadf2303ba550baf481886..a0339194afe9def80ec7aeba80647ec0e8084981 100644
--- a/tests/syntax/flexible_array_member_invalid5.i
+++ b/tests/syntax/flexible_array_member_invalid5.i
@@ -4,7 +4,7 @@ typedef struct {
   char data[];
 } fam;
 
-struct {
+struct st {
   int len;
   fam f;
-} st;
+} sst;
diff --git a/tests/syntax/oracle/flexible_array_member.res.oracle b/tests/syntax/oracle/flexible_array_member.res.oracle
index a7317194e390ea0960fa4a3cbb72e062ae312798..90af690c44153978d928abf3544bb5802d620eea 100644
--- a/tests/syntax/oracle/flexible_array_member.res.oracle
+++ b/tests/syntax/oracle/flexible_array_member.res.oracle
@@ -1,13 +1,21 @@
 [kernel] Parsing tests/syntax/flexible_array_member.i (no preprocessing)
 /* Generated by Frama-C */
-struct __anonstruct_s1_1 {
+struct s1 {
    int size ;
    char data[] ;
 };
-struct __anonstruct_s2_2 {
+struct s2 {
    char len ;
    char data[] ;
 };
-struct __anonstruct_s1_1 s1;
-struct __anonstruct_s2_2 s2;
+struct s {
+   char len ;
+   char data[] ;
+};
+union u {
+   struct s fam ;
+};
+struct s1 ss1;
+struct s2 ss2;
+union u u1;
 
diff --git a/tests/value/initialized.c b/tests/value/initialized.c
index 2292df35211489154198dc5402681dd5ebd0f99e..3e04b3cc71674441815e291dae3d6f580cf4586b 100644
--- a/tests/value/initialized.c
+++ b/tests/value/initialized.c
@@ -154,6 +154,41 @@ void g7() {
   //@ assert !\initialized(&key[0..127]);
 }
 
+/* Tests the reduction by the negation of the \initialized predicate. */
+void reduce_by_negation () {
+  int x, y;
+  int *p = rand ? &x : &y;
+  if (rand) x = 0;
+  if (rand) y = 0;
+  if (rand) {
+    //@ assert !\initialized(&x);
+    //@ check invalid: \initialized(&x);
+  }
+  if (rand) {
+    //@ assert !\initialized(p);
+    //@ check unknown: \initialized(&x) && \initialized(&y);
+  }
+  if (rand) {
+    //@ assert !\initialized({&x, &y});
+    //@ check unknown: \initialized(&x) && \initialized(&y);
+  }
+  if (rand) {
+    y = 0;
+    //@ assert !\initialized(p);
+    //@ check unknown: \initialized(&x);
+  }
+  if (rand) {
+    y = 0;
+    //@ assert !\initialized({&x, &y});
+    //@ check invalid: \initialized(&x);
+  }
+  char t[10];
+  for (int i = 0; i < 10; i++)
+    t[i] = i;
+  //@ assert !\initialized(&t[0..9]);
+  //@ check unknown: \initialized(&t[0..9]);
+}
+
 int main () {
   g1();
   g2();
@@ -162,5 +197,6 @@ int main () {
   g5();
   g6();
   g7();
+  reduce_by_negation();
   return 0;
 }
diff --git a/tests/value/oracle/initialized.res.oracle b/tests/value/oracle/initialized.res.oracle
index 79cf5c4a3129d812402778bbe4f0ab7ba79dc9d2..f3d1c20aa17d60f4f052fe7a489807dd3537ff5f 100644
--- a/tests/value/oracle/initialized.res.oracle
+++ b/tests/value/oracle/initialized.res.oracle
@@ -13,7 +13,7 @@
   v1 ∈ {0}
   i6 ∈ [--..--]
 [eva] computing for function g1 <- main.
-  Called from tests/value/initialized.c:158.
+  Called from tests/value/initialized.c:193.
 [eva] tests/value/initialized.c:19: starting to merge loop iterations
 [eva:alarm] tests/value/initialized.c:21: Warning: assertion got status unknown.
 [eva:alarm] tests/value/initialized.c:22: Warning: assertion got status unknown.
@@ -65,7 +65,7 @@
 [eva] Recording results for g1
 [eva] Done for function g1
 [eva] computing for function g2 <- main.
-  Called from tests/value/initialized.c:159.
+  Called from tests/value/initialized.c:194.
 [eva:alarm] tests/value/initialized.c:50: Warning: 
   signed overflow. assert -2147483648 ≤ (int)(&b4) + (int)(&b4);
 [eva:alarm] tests/value/initialized.c:50: Warning: 
@@ -149,7 +149,7 @@
 [eva] Recording results for g2
 [eva] Done for function g2
 [eva] computing for function g3 <- main.
-  Called from tests/value/initialized.c:160.
+  Called from tests/value/initialized.c:195.
 [eva:alarm] tests/value/initialized.c:89: Warning: assertion got status unknown.
 [eva:alarm] tests/value/initialized.c:93: Warning: 
   accessing uninitialized left-value. assert \initialized(&r2);
@@ -171,13 +171,13 @@
 [eva] Recording results for g3
 [eva] Done for function g3
 [eva] computing for function g4 <- main.
-  Called from tests/value/initialized.c:161.
+  Called from tests/value/initialized.c:196.
 [eva:alarm] tests/value/initialized.c:104: Warning: 
   accessing uninitialized left-value. assert \initialized(&y);
 [eva] Recording results for g4
 [eva] Done for function g4
 [eva] computing for function g5 <- main.
-  Called from tests/value/initialized.c:162.
+  Called from tests/value/initialized.c:197.
 [eva] computing for function wrong_assigns <- g5 <- main.
   Called from tests/value/initialized.c:127.
 [eva] using specification for function wrong_assigns
@@ -193,7 +193,7 @@
 [eva] Recording results for g5
 [eva] Done for function g5
 [eva] computing for function g6 <- main.
-  Called from tests/value/initialized.c:163.
+  Called from tests/value/initialized.c:198.
 [eva:alarm] tests/value/initialized.c:143: Warning: 
   assertion got status unknown.
 [eva:alarm] tests/value/initialized.c:144: Warning: 
@@ -207,7 +207,7 @@
 [eva] Recording results for g6
 [eva] Done for function g6
 [eva] computing for function g7 <- main.
-  Called from tests/value/initialized.c:164.
+  Called from tests/value/initialized.c:199.
 [eva] computing for function Frama_C_make_unknown <- g7 <- main.
   Called from tests/value/initialized.c:153.
 [eva] using specification for function Frama_C_make_unknown
@@ -217,6 +217,35 @@
 [eva] tests/value/initialized.c:154: assertion got status valid.
 [eva] Recording results for g7
 [eva] Done for function g7
+[eva] computing for function reduce_by_negation <- main.
+  Called from tests/value/initialized.c:200.
+[eva:alarm] tests/value/initialized.c:164: Warning: 
+  assertion got status unknown.
+[eva:alarm] tests/value/initialized.c:165: Warning: 
+  check 'invalid' got status invalid.
+[eva:alarm] tests/value/initialized.c:168: Warning: 
+  assertion got status unknown.
+[eva:alarm] tests/value/initialized.c:169: Warning: 
+  check 'unknown' got status unknown.
+[eva:alarm] tests/value/initialized.c:172: Warning: 
+  assertion got status unknown.
+[eva:alarm] tests/value/initialized.c:173: Warning: 
+  check 'unknown' got status unknown.
+[eva:alarm] tests/value/initialized.c:177: Warning: 
+  assertion got status unknown.
+[eva:alarm] tests/value/initialized.c:178: Warning: 
+  check 'unknown' got status unknown.
+[eva:alarm] tests/value/initialized.c:182: Warning: 
+  assertion got status unknown.
+[eva:alarm] tests/value/initialized.c:183: Warning: 
+  check 'invalid' got status unknown.
+[eva] tests/value/initialized.c:186: starting to merge loop iterations
+[eva:alarm] tests/value/initialized.c:188: Warning: 
+  assertion got status unknown.
+[eva:alarm] tests/value/initialized.c:189: Warning: 
+  check 'unknown' got status unknown.
+[eva] Recording results for reduce_by_negation
+[eva] Done for function reduce_by_negation
 [eva] Recording results for main
 [eva] done for function main
 [eva] tests/value/initialized.c:93: 
@@ -293,6 +322,11 @@
   Frama_C_entropy_source ∈ [--..--]
   key[0..63] ∈ [--..--]
      [64..127] ∈ UNINITIALIZED
+[eva:final-states] Values at end of function reduce_by_negation:
+  x ∈ {0} or UNINITIALIZED
+  y ∈ {0} or UNINITIALIZED
+  p ∈ {{ &x ; &y }}
+  t[0..9] ∈ [0..9] or UNINITIALIZED
 [eva:final-states] Values at end of function g5:
   v ∈ UNINITIALIZED
   p ∈ {{ &v1 ; &v2 }}
@@ -319,6 +353,8 @@
 [from] Computing for function Frama_C_make_unknown <-g7
 [from] Done for function Frama_C_make_unknown
 [from] Done for function g7
+[from] Computing for function reduce_by_negation
+[from] Done for function reduce_by_negation
 [from] Computing for function g5
 [from] Computing for function wrong_assigns <-g5
 [from] Done for function wrong_assigns
@@ -348,6 +384,8 @@
   i6 FROM rand (and SELF)
 [from] Function g7:
   Frama_C_entropy_source FROM Frama_C_entropy_source (and SELF)
+[from] Function reduce_by_negation:
+  NO EFFECTS
 [from] Function wrong_assigns:
   v{.a; .b} FROM \nothing
 [from] Function g5:
@@ -386,6 +424,10 @@
     Frama_C_entropy_source; key[0..63]
 [inout] Inputs for function g7:
     Frama_C_entropy_source
+[inout] Out (internal) for function reduce_by_negation:
+    x; y; p; tmp; t[0..9]; i
+[inout] Inputs for function reduce_by_negation:
+    rand
 [inout] Out (internal) for function g5:
     v{.a; .b}; p; tmp
 [inout] Inputs for function g5:
diff --git a/tests/value/oracle/leaf_spec.0.res.oracle b/tests/value/oracle/leaf_spec.0.res.oracle
index 761d38bcf9e226c2597073376f838576ca755dd9..4dda7f7cf8b0609d64b17c78ec40d983449de12b 100644
--- a/tests/value/oracle/leaf_spec.0.res.oracle
+++ b/tests/value/oracle/leaf_spec.0.res.oracle
@@ -27,7 +27,7 @@
 [kernel:annot:missing-spec] tests/value/leaf_spec.i:22: Warning: 
   Neither code nor specification for function k, generating default assigns from the prototype
 [eva] using specification for function k
-[eva] tests/value/leaf_spec.i:22: Warning: 
+[eva:invalid-assigns] tests/value/leaf_spec.i:22: 
   Completely invalid destination for assigns clause *l. Ignoring.
 [eva] Done for function k
 [eva] computing for function k0 <- main.
diff --git a/tests/value/oracle/leaf_spec.1.res.oracle b/tests/value/oracle/leaf_spec.1.res.oracle
index 666a0ee653ea406fd618ea0f479ba41d7741adf0..aa1978e94dfdb9044fb6881d77a2a219b2701da0 100644
--- a/tests/value/oracle/leaf_spec.1.res.oracle
+++ b/tests/value/oracle/leaf_spec.1.res.oracle
@@ -9,9 +9,9 @@
 [kernel:annot:missing-spec] tests/value/leaf_spec.i:27: Warning: 
   Neither code nor specification for function f, generating default assigns from the prototype
 [eva] using specification for function f
-[eva] tests/value/leaf_spec.i:27: Warning: 
+[eva:invalid-assigns] tests/value/leaf_spec.i:27: 
   Completely invalid destination for assigns clause *x. Ignoring.
-[eva] tests/value/leaf_spec.i:27: Warning: 
+[eva:invalid-assigns] tests/value/leaf_spec.i:27: 
   Completely invalid destination for assigns clause *y. Ignoring.
 [eva] Done for function f
 [eva] Recording results for main1