diff --git a/src/plugins/value/domains/cvalue/cvalue_domain.ml b/src/plugins/value/domains/cvalue/cvalue_domain.ml
index 3c0579fa6462ea7496f28af5b5ebc819b84f0cec..019f87cc0ee61fa5f2fb0ab04a5b43ce16e56256 100644
--- a/src/plugins/value/domains/cvalue/cvalue_domain.ml
+++ b/src/plugins/value/domains/cvalue/cvalue_domain.ml
@@ -22,15 +22,8 @@
 
 open Eval
 
-let key = Structure.Key_Domain.create_key "cvalue_domain"
 let dkey_card = Value_parameters.register_category "cardinal"
 
-let extract get = match get key with
-  | None -> fun _ -> Cvalue.Model.top
-  | Some get -> function
-    | `Bottom -> Cvalue.Model.bottom
-    | `Value state -> fst (get state)
-
 module Model = struct
 
   include Cvalue.Model
@@ -189,7 +182,7 @@ module State = struct
     end )
 
   let name = "Cvalue domain"
-  let key = key
+  let key = Structure.Key_Domain.create_key "cvalue_domain"
 
   type value = Model.value
   type location = Model.location
@@ -549,9 +542,6 @@ end
 
 let () = Db.Value.display := (fun fmt kf -> State.display ~fmt kf)
 
-let inject cvalue_model = cvalue_model, Locals_scoping.bottom ()
-let project (state, _) = state
-
 
 type prefix = Hptmap.prefix
 module Subpart = struct
diff --git a/src/plugins/value/domains/cvalue/cvalue_domain.mli b/src/plugins/value/domains/cvalue/cvalue_domain.mli
index bd1284006ff3f389db475654aee11f3e21604cf4..cf750b4e60f4caad41803517a7d468d98c6b948b 100644
--- a/src/plugins/value/domains/cvalue/cvalue_domain.mli
+++ b/src/plugins/value/domains/cvalue/cvalue_domain.mli
@@ -27,15 +27,6 @@ module State : Abstract_domain.Leaf
    and type location = Main_locations.PLoc.location
    and type state = Cvalue.Model.t * Locals_scoping.clobbered_set
 
-val extract :
-  (State.t Structure.Key_Domain.key ->
-   ('state -> State.t) option) ->
-  'state Eval.or_bottom -> Cvalue.Model.t
-
-val inject : Cvalue.Model.t -> State.t
-val project : State.t -> Cvalue.Model.t
-
-
 (** Specific functions for partitioning optimizations.  *)
 
 type prefix
diff --git a/src/plugins/value/engine/abstractions.ml b/src/plugins/value/engine/abstractions.ml
index b0aca9a3150d88bd68121a435572fb694852222b..3a52d334d1f27ed177e53edd7c76f29099355316 100644
--- a/src/plugins/value/engine/abstractions.ml
+++ b/src/plugins/value/engine/abstractions.ml
@@ -384,6 +384,18 @@ module Open (Acc: Acc) : S = struct
   module Dom = struct
     include Acc.Dom
     include Structure.Open (Abstract.Domain) (Acc.Dom)
+
+    let get_cvalue = match get Cvalue_domain.State.key with
+      | None -> None
+      | Some get -> Some (fun s -> fst (get s))
+
+    let get_cvalue_or_top = match get Cvalue_domain.State.key with
+      | None -> fun _ -> Cvalue.Model.top
+      | Some get -> fun s -> fst (get s)
+
+    let get_cvalue_or_bottom = function
+      | `Bottom -> Cvalue.Model.bottom
+      | `Value state -> get_cvalue_or_top state
   end
 end
 
diff --git a/src/plugins/value/engine/analysis.ml b/src/plugins/value/engine/analysis.ml
index b045bee75c807d965db72fe2bfeef2dd7bda8f65..ee1b5680eeac44e2ad88489209edf6dbbb7fceb6 100644
--- a/src/plugins/value/engine/analysis.ml
+++ b/src/plugins/value/engine/analysis.ml
@@ -133,7 +133,7 @@ let set_current_analyzer config (analyzer: (module Analyzer)) =
 let cvalue_initial_state () =
   let module A = (val snd !ref_analyzer) in
   let _, lib_entry = Globals.entry_point () in
-  Cvalue_domain.extract A.Dom.get (A.initial_state ~lib_entry)
+  A.Dom.get_cvalue_or_bottom (A.initial_state ~lib_entry)
 
 (* Builds the Analyzer module corresponding to a given configuration,
    and sets it as the current analyzer. *)
diff --git a/src/plugins/value/engine/compute_functions.ml b/src/plugins/value/engine/compute_functions.ml
index 4440415631b5dd90793189134a401c3e655e9dfa..21a9876322a1e7dfe27fdc0a8c9d0dc8803ce778 100644
--- a/src/plugins/value/engine/compute_functions.ml
+++ b/src/plugins/value/engine/compute_functions.ml
@@ -148,11 +148,6 @@ module Make (Abstract: Abstractions.Eva) = struct
 
   let initial_state = Init.initial_state
 
-  let get_cvalue =
-    match Abstract.Dom.get Cvalue_domain.State.key with
-    | None -> fun _ -> Cvalue.Model.top
-    | Some get -> fun state -> Cvalue_domain.project (get state)
-
   let get_cval =
     match Abstract.Val.get Main_values.CVal.key with
     | None -> fun _ -> assert false
@@ -191,7 +186,7 @@ module Make (Abstract: Abstractions.Eva) = struct
           then `Spec (Annotations.funspec kf)
           else `Def def
     in
-    let cvalue_state = get_cvalue state in
+    let cvalue_state = Abstract.Dom.get_cvalue_or_top state in
     let resulting_states, cacheable = match use_spec with
       | `Spec spec ->
         Db.Value.Call_Type_Value_Callbacks.apply
@@ -237,9 +232,9 @@ module Make (Abstract: Abstractions.Eva) = struct
         in
         call_result
       | Some (states, i) ->
-        let stack_with_call = Value_util.call_stack () in
-        Db.Value.Call_Type_Value_Callbacks.apply
-          (`Memexec, get_cvalue init_state, stack_with_call);
+        let stack = Value_util.call_stack () in
+        let cvalue = Abstract.Dom.get_cvalue_or_top init_state in
+        Db.Value.Call_Type_Value_Callbacks.apply (`Memexec, cvalue, stack);
         (* Evaluate the preconditions of kf, to update the statuses
            at this call. *)
         let spec = Annotations.funspec call.kf in
@@ -300,7 +295,7 @@ module Make (Abstract: Abstractions.Eva) = struct
       in
       Locations.Location_Bytes.do_track_garbled_mix true;
       let final_state = states >>- join_states in
-      let cvalue_state = get_cvalue state in
+      let cvalue_state = Abstract.Dom.get_cvalue_or_top state in
       match final_state with
       | `Bottom ->
         let cs = Value_util.call_stack () in
@@ -329,7 +324,7 @@ module Make (Abstract: Abstractions.Eva) = struct
 
   let store_initial_state kf init_state =
     Abstract.Dom.Store.register_initial_state (Value_util.call_stack ()) init_state;
-    let cvalue_state = get_cvalue init_state in
+    let cvalue_state = Abstract.Dom.get_cvalue_or_top init_state in
     Db.Value.Call_Value_Callbacks.apply (cvalue_state, [kf, Kglobal])
 
   let compute kf init_state =
diff --git a/src/plugins/value/engine/initialization.ml b/src/plugins/value/engine/initialization.ml
index 6f15c07d5d656a644e39895b095726493790376a..fd124a687b6a27f5c823130697fcad9ca2042a7a 100644
--- a/src/plugins/value/engine/initialization.ml
+++ b/src/plugins/value/engine/initialization.ml
@@ -270,15 +270,14 @@ module Make
   (* Use the values supplied in [actuals] for the formals of [kf], and
      bind them in [state] *)
   let add_supplied_main_formals kf actuals state =
-    match Domain.get Cvalue_domain.State.key with
-    | None ->
-      Value_parameters.abort "Function Db.Value.fun_set_args cannot be used \
-                              without the Cvalue domain"
+    match Domain.get_cvalue with
+    | None -> Value_parameters.abort "Function Db.Value.fun_set_args cannot be \
+                                      used without the Cvalue domain"
     | Some get_cvalue ->
       let formals = Kernel_function.get_formals kf in
       if (List.length formals) <> List.length actuals then
         raise Db.Value.Incorrect_number_of_arguments;
-      let cvalue_state = Cvalue_domain.project (get_cvalue state) in
+      let cvalue_state = get_cvalue state in
       let add_actual state actual formal =
         let actual = Eval_op.offsetmap_of_v ~typ:formal.vtype actual in
         Cvalue.Model.add_base (Base.of_varinfo formal) actual state
@@ -287,7 +286,7 @@ module Make
         List.fold_left2 add_actual cvalue_state actuals formals
       in
       let set_domain = Domain.set Cvalue_domain.State.key in
-      set_domain (Cvalue_domain.inject cvalue_state) state
+      set_domain (cvalue_state, Locals_scoping.bottom ()) state
 
   let add_main_formals kf state =
     match Db.Value.fun_get_args () with
@@ -355,7 +354,7 @@ module Make
     let cvalue_state = Db.Value.globals_state () in
     if Cvalue.Model.is_reachable cvalue_state
     then
-      let cvalue_state = Cvalue_domain.inject cvalue_state in
+      let cvalue_state = cvalue_state, Locals_scoping.bottom () in
       `Value (Domain.set Cvalue_domain.State.key cvalue_state Domain.top)
     else `Bottom
 
@@ -365,7 +364,7 @@ module Make
     else global_state ~lib_entry
 
   let print_initial_cvalue_state state =
-    let cvalue_state = Cvalue_domain.extract Domain.get state in
+    let cvalue_state = Domain.get_cvalue_or_bottom state in
     (* Do not show variables from the frama-c libc specifications. *)
     let print_base base =
       try
diff --git a/src/plugins/value/engine/iterator.ml b/src/plugins/value/engine/iterator.ml
index 0e83fb39280cd62436229a5d74a6b8cd5ff5def8..bae4fbcc7b35ec5ec246816cd8ab55f9f6bcfe47 100644
--- a/src/plugins/value/engine/iterator.ml
+++ b/src/plugins/value/engine/iterator.ml
@@ -433,10 +433,8 @@ module Make_Dataflow
       edge_info.fireable <- true;
     flow
 
-  let get_cvalue = Domain.get Cvalue_domain.State.key
-  let gather_cvalues states =
-    match get_cvalue with
-    | Some get -> List.map (fun s -> Cvalue_domain.project (get s)) states
+  let gather_cvalues states = match Domain.get_cvalue with
+    | Some get -> List.map get states
     | None -> []
 
   let call_statement_callbacks (stmt : stmt) (f : flow) : unit =
@@ -633,9 +631,6 @@ module Make_Dataflow
     G.iter_edges_e fill graph;
     Db.Value.merge_conditions table
 
-  let extract_cvalue (state : state) : 'a =
-    Cvalue_domain.extract Domain.get (`Value state)
-
   let is_instr s = match s.skind with Instr _ -> true | _ -> false
 
   let states_after_stmt states_before states_after =
@@ -676,7 +671,7 @@ module Make_Dataflow
         then VertexTable.memo merged_states v get_smashed_store
         else `Bottom
     and lift_to_cvalues table =
-      StmtTable.map (fun _ s -> extract_cvalue s) (Lazy.force table)
+      StmtTable.map (fun _ s -> Domain.get_cvalue_or_top s) (Lazy.force table)
     in
     let merged_pre_states = lazy
       (StmtTable.map' (fun s (v,_) -> get_merged_states ~all:true s v) automaton.stmt_table)
@@ -691,7 +686,7 @@ module Make_Dataflow
       (StmtTable.map (fun _stmt (v,_) ->
            let store = get_vertex_store v in
            let states = Partition.expanded store in
-           List.map (fun x -> extract_cvalue x) states)
+           List.map (fun x -> Domain.get_cvalue_or_top x) states)
           automaton.stmt_table)
     in
     let merged_pre_cvalues = lazy (lift_to_cvalues merged_pre_states)
diff --git a/src/plugins/value/engine/transfer_specification.ml b/src/plugins/value/engine/transfer_specification.ml
index 69171aaeec78721bd2c156951fea6e5d791df366..e15091974e5c8cdc7de19f10e67d858ec9a6babc 100644
--- a/src/plugins/value/engine/transfer_specification.ml
+++ b/src/plugins/value/engine/transfer_specification.ml
@@ -232,11 +232,8 @@ module Make
     | Some get -> get
   let set_ploc = Location.set Main_locations.PLoc.key
   let set_location loc = set_ploc (Main_locations.PLoc.make loc)
-  let get_cvalue_state = match Domain.get Cvalue_domain.State.key with
-    | None -> fun _ -> Cvalue.Model.top
-    | Some get -> fun s -> Cvalue_domain.project (get s)
 
-  let make_env state = Eval_terms.env_assigns (get_cvalue_state state)
+  let make_env state = Eval_terms.env_assigns (Domain.get_cvalue_or_top state)
 
   let is_result = function
     | Assigns (term, _)
@@ -299,7 +296,7 @@ module Make
         end
     in
     let check_one_state state =
-      let cvalue_state = get_cvalue_state state in
+      let cvalue_state = Domain.get_cvalue_or_top state in
       List.iter (check_one_assign cvalue_state) assigns
     in
     States.iter check_one_state states
diff --git a/src/plugins/value/engine/transfer_stmt.ml b/src/plugins/value/engine/transfer_stmt.ml
index c0f99adcbda13fd6af6563f097ae04be9534acee..514e4ea789c0db62b5c02857c3417d66e9128ee5 100644
--- a/src/plugins/value/engine/transfer_stmt.ml
+++ b/src/plugins/value/engine/transfer_stmt.ml
@@ -559,10 +559,6 @@ module Make (Abstract: Abstractions.Eva) = struct
 
   (* ----------------- show_each and dump_each directives ------------------- *)
 
-  let extract_cvalue = match Domain.get Cvalue_domain.State.key with
-    | None -> fun _ -> Cvalue.Model.top
-    | Some get -> fun s -> Cvalue_domain.project (get s)
-
   (* The product of domains formats the printing of each leaf domains, by
      checking their log_category and adding their name before the dump. If the
      domain is not a product, this needs to be done here. *)
@@ -609,7 +605,7 @@ module Make (Abstract: Abstractions.Eva) = struct
 
   (* For non scalar expressions, prints the offsetmap of the cvalue domain. *)
   let show_offsm =
-    match Domain.get Cvalue_domain.State.key, Location.get Main_locations.PLoc.key with
+    match Domain.get_cvalue, Location.get Main_locations.PLoc.key with
     | None, _ | _, None ->
       fun fmt _ _ -> Format.fprintf fmt "%s" (Unicode.top_string ())
     | Some get_cvalue, Some get_ploc ->
@@ -621,9 +617,7 @@ module Make (Abstract: Abstractions.Eva) = struct
               let offsm =
                 fst (Eval.lvaluate ~for_writing:false state lval)
                 >>- fun (_, loc, _) ->
-                let ploc = get_ploc loc
-                and cvalue_state = Cvalue_domain.project (get_cvalue state) in
-                Eval_op.offsetmap_of_loc ploc cvalue_state
+                Eval_op.offsetmap_of_loc (get_ploc loc) (get_cvalue state)
               in
               let typ = Cil.typeOf expr in
               (Bottom.pretty (Eval_op.pretty_offsetmap typ)) fmt offsm
@@ -707,7 +701,7 @@ module Make (Abstract: Abstractions.Eva) = struct
      {Cvalue_transfer.start_call}. *)
   let apply_cvalue_callback kf ki_call state =
     let stack_with_call = (kf, ki_call) :: Value_util.call_stack () in
-    let cvalue_state = extract_cvalue state in
+    let cvalue_state = Domain.get_cvalue_or_top state in
     Db.Value.Call_Value_Callbacks.apply (cvalue_state, stack_with_call);
     Db.Value.merge_initial_state (Value_util.call_stack ()) cvalue_state;
     let result =
diff --git a/src/plugins/value/gui_files/gui_eval.ml b/src/plugins/value/gui_files/gui_eval.ml
index 2c64ede8741331662b1b3085078c4ba5e77364af..8239f3974fe81a150ebdeaa53a488ec29935eb31 100644
--- a/src/plugins/value/gui_files/gui_eval.ml
+++ b/src/plugins/value/gui_files/gui_eval.ml
@@ -143,11 +143,6 @@ module Make (X: Analysis.S) = struct
 
   module Analysis = X
 
-  let get_cvalue_state =
-    match X.Dom.get Cvalue_domain.State.key with
-    | None -> fun _ -> Cvalue.Model.top
-    | Some get -> fun state -> Cvalue_domain.project (get state)
-
   let get_precise_loc =
     match X.Loc.get Main_locations.PLoc.key with
     | None -> fun _ -> Precise_locs.loc_top
@@ -210,7 +205,7 @@ module Make (X: Analysis.S) = struct
   let lval_to_offsetmap state lv =
     let loc, alarms = X.eval_lval_to_loc state lv in
     let ok = Alarmset.is_empty alarms in
-    let state = get_cvalue_state state in
+    let state = X.Dom.get_cvalue_or_top state in
     let aux loc (acc_res, acc_ok) =
       let res, ok =
         match lv with (* catch simplest pattern *)
@@ -273,7 +268,7 @@ module Make (X: Analysis.S) = struct
     }
 
   let null_to_offsetmap state (_:unit) =
-    let state = get_cvalue_state state in
+    let state = X.Dom.get_cvalue_or_top state in
     match Cvalue.Model.find_base_or_default Base.null state with
     | `Bottom -> GO_InvalidLoc, false, false
     | `Top -> GO_Top, false, false
@@ -335,17 +330,17 @@ module Make (X: Analysis.S) = struct
 
   let env_here kf here callstack =
     let pre = pre_kf kf callstack in
-    let here = get_cvalue_state here in
+    let here = X.Dom.get_cvalue_or_top here in
     let c_labels = Eval_annots.c_labels kf callstack in
     Eval_terms.env_annot ~c_labels ~pre ~here ()
 
   let env_pre _kf here _callstack =
-    let here = get_cvalue_state here in
+    let here = X.Dom.get_cvalue_or_top here in
     Eval_terms.env_pre_f ~pre:here ()
 
   let env_post kf post callstack =
     let pre = pre_kf kf callstack in
-    let post = get_cvalue_state post in
+    let post = X.Dom.get_cvalue_or_top post in
     let result =
       if !Db.Value.use_spec_instead_of_definition kf then
         None
diff --git a/src/plugins/value/register.ml b/src/plugins/value/register.ml
index 480323cf90b98819280de507bd88acf0b527c9e6..f8617e5ad4eb5c0b021ed543a5102af398231f72 100644
--- a/src/plugins/value/register.ml
+++ b/src/plugins/value/register.ml
@@ -188,6 +188,8 @@ module Eva =
 
 module Transfer = Cvalue_domain.State.Transfer (Eva.Valuation)
 
+let inject_cvalue state = state, Locals_scoping.bottom ()
+
 let bot_value = function
   | `Bottom -> Cvalue.V.bottom
   | `Value v -> v
@@ -197,7 +199,7 @@ let bot_state = function
   | `Value s -> s
 
 let update valuation state =
-  bot_state (Transfer.update valuation state >>-: Cvalue_domain.project)
+  bot_state (Transfer.update valuation state >>-: fst)
 
 let rec eval_deps state e =
   match e.enode with
@@ -236,7 +238,7 @@ let notify_opt with_alarms alarms =
   Extlib.may (fun mode -> Alarmset.notify mode alarms) with_alarms
 
 let eval_expr_with_valuation ?with_alarms deps state expr=
-  let state = Cvalue_domain.inject state in
+  let state = inject_cvalue state in
   let deps = match deps with
     | None -> None
     | Some deps ->
@@ -256,7 +258,7 @@ let eval_expr_with_valuation ?with_alarms deps state expr=
 module Eval = struct
 
   let eval_expr ?with_alarms state expr =
-    let state = Cvalue_domain.inject state in
+    let state = inject_cvalue state in
     let eval, alarms = Eva.evaluate ~reduction:false state expr in
     notify_opt with_alarms alarms;
     bot_value (eval >>-: snd)
@@ -278,7 +280,7 @@ module Eval = struct
 
 
   let reduce_by_cond state expr positive =
-    let state = Cvalue_domain.inject state in
+    let state = inject_cvalue state in
     let eval, _alarms =
       Eva.reduce state expr positive
     in
@@ -289,7 +291,7 @@ module Eval = struct
     if not (Cvalue.Model.is_reachable state)
     then state, deps, Precise_locs.loc_bottom, (Cil.typeOfLval lval)
     else
-      let state = Cvalue_domain.inject state in
+      let state = inject_cvalue state in
       let deps = match deps with
         | None -> None
         | Some deps ->
@@ -338,7 +340,7 @@ module Eval = struct
 
   let resolv_func_vinfo ?with_alarms deps state funcexp =
     let open Cil_types in
-    let state = Cvalue_domain.inject state in
+    let state = inject_cvalue state in
     let deps = match funcexp.enode with
       | Lval (Var _, NoOffset) -> deps
       | Lval (Mem v, _) ->
diff --git a/src/plugins/value/utils/abstract.ml b/src/plugins/value/utils/abstract.ml
index 1ccaa235ba42bb7c456b769b36efdba281386497..e5153a5241bcbd0682795eb13ee79fbe48976499 100644
--- a/src/plugins/value/utils/abstract.ml
+++ b/src/plugins/value/utils/abstract.ml
@@ -86,5 +86,9 @@ module Domain = struct
     include Internal
     include Structure.External with type t := t
                                 and type 'a key := 'a key
+
+    val get_cvalue: (t -> Cvalue.Model.t) option
+    val get_cvalue_or_top: t -> Cvalue.Model.t
+    val get_cvalue_or_bottom: t Bottom.or_bottom -> Cvalue.Model.t
   end
 end
diff --git a/src/plugins/value/utils/abstract.mli b/src/plugins/value/utils/abstract.mli
index 258ddf831f82f00bd9af1bf2df6c85b5db0c1c89..3e797f8eddba95cddfa69765f3097ac7a3081ffa 100644
--- a/src/plugins/value/utils/abstract.mli
+++ b/src/plugins/value/utils/abstract.mli
@@ -109,5 +109,10 @@ module Domain : sig
     include Internal
     include Interface with type t := t
                        and type 'a key := 'a key
+
+    (** Special accessors for the main cvalue domain. *)
+    val get_cvalue: (t -> Cvalue.Model.t) option
+    val get_cvalue_or_top: t -> Cvalue.Model.t
+    val get_cvalue_or_bottom: t Bottom.or_bottom -> Cvalue.Model.t
   end
 end