From 3b8a9bef5379cd458b2a083baabbdbf11e4a7dfc Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?David=20B=C3=BChler?= <david.buhler@cea.fr>
Date: Mon, 22 Jul 2024 11:16:24 +0200
Subject: [PATCH] [Hptmap] Simplifies functor use: no compositional bool by
 default.

Complete functor Make_with_compositional_bool takes an additional module
for the compositional boolean.
Functor Make simply uses Comp_unused by default.
---
 src/kernel_services/abstract_interp/base.ml   |  2 +-
 src/kernel_services/abstract_interp/lmap.ml   |  9 +--
 .../abstract_interp/lmap_bitwise.ml           |  2 +-
 .../abstract_interp/locations.ml              |  9 +--
 src/kernel_services/analysis/stmts_graph.ml   |  1 -
 src/libraries/utils/hptmap.ml                 | 28 +++++----
 src/libraries/utils/hptmap.mli                | 60 ++++++++++---------
 src/libraries/utils/hptset.ml                 |  2 +-
 .../eva/domains/cvalue/builtins_malloc.ml     |  3 +-
 src/plugins/eva/domains/equality/equality.ml  |  2 +-
 .../eva/domains/gauges/gauges_domain.ml       |  4 +-
 src/plugins/eva/domains/hcexprs.ml            |  4 +-
 .../domains/multidim/abstract_structure.ml    |  3 +-
 .../eva/domains/multidim/multidim_domain.ml   |  2 +-
 src/plugins/eva/domains/octagons.ml           | 10 ++--
 src/plugins/eva/domains/simple_memory.ml      |  2 +-
 src/plugins/eva/domains/symbolic_locs.ml      |  2 +-
 src/plugins/eva/domains/traces_domain.ml      |  2 +-
 src/plugins/from/from_compute.ml              |  2 +-
 src/plugins/impact/pdg_aux.ml                 |  6 +-
 src/plugins/pdg/pdg_types/pdgTypes.ml         |  2 +-
 21 files changed, 79 insertions(+), 78 deletions(-)

diff --git a/src/kernel_services/abstract_interp/base.ml b/src/kernel_services/abstract_interp/base.ml
index b5f98f4b825..0a87aa8f1fd 100644
--- a/src/kernel_services/abstract_interp/base.ml
+++ b/src/kernel_services/abstract_interp/base.ml
@@ -555,7 +555,7 @@ let of_string_exp e =
 module SetLattice = Make_Hashconsed_Lattice_Set(Base)(Hptset)
 
 module BMap =
-  Hptmap.Make (Base) (Base) (Hptmap.Comp_unused)
+  Hptmap.Make (Base) (Base)
     (struct
       let initial_values = []
       let dependencies = [ Ast.self ]
diff --git a/src/kernel_services/abstract_interp/lmap.ml b/src/kernel_services/abstract_interp/lmap.ml
index 8c8b3f3c20e..54240aad4fa 100644
--- a/src/kernel_services/abstract_interp/lmap.ml
+++ b/src/kernel_services/abstract_interp/lmap.ml
@@ -88,17 +88,18 @@ struct
     end
 
     module Comp = struct
-      let f _base offsetmap = Offsetmap.cardinal_zero_or_one offsetmap
+      let empty = true
+      let leaf _base offsetmap = Offsetmap.cardinal_zero_or_one offsetmap
       let compose a b = a && b
-      let e = true
     end
 
-    module Hptmap_Info = struct
+    module Info = struct
       let initial_values = []
       let dependencies = [ Ast.self ]
     end
 
-    module M = Hptmap.Make (Base.Base) (Offsetmap) (Comp) (Hptmap_Info)
+    module M =
+      Hptmap.Make_with_compositional_bool (Base.Base) (Offsetmap) (Comp) (Info)
     let () = Ast.add_monotonic_state M.self
 
     include M
diff --git a/src/kernel_services/abstract_interp/lmap_bitwise.ml b/src/kernel_services/abstract_interp/lmap_bitwise.ml
index ffc2bafe92e..04ca90b91a1 100644
--- a/src/kernel_services/abstract_interp/lmap_bitwise.ml
+++ b/src/kernel_services/abstract_interp/lmap_bitwise.ml
@@ -145,7 +145,7 @@ struct
       let initial_values = []
       let dependencies = [ Ast.self ]
     end
-    include Hptmap.Make (Base.Base) (LOffset) (Hptmap.Comp_unused) (Hptmap_Info)
+    include Hptmap.Make (Base.Base) (LOffset) (Hptmap_Info)
     let () = Ast.add_monotonic_state self
 
     (* We override [add] so that the map is canonical: no key should be
diff --git a/src/kernel_services/abstract_interp/locations.ml b/src/kernel_services/abstract_interp/locations.ml
index af174d7cd4d..a47a5d2cbee 100644
--- a/src/kernel_services/abstract_interp/locations.ml
+++ b/src/kernel_services/abstract_interp/locations.ml
@@ -46,9 +46,9 @@ end
    are included in it. Said otherwise, we do not consider the cardinality
    of the concretization, but instead the one of the Ocaml datastructure. *)
 module Comp_exact = struct
-  let e = true (* corresponds to bottom *)
+  let empty = true (* corresponds to bottom *)
 
-  let f _b v = Ival.cardinal_zero_or_one v
+  let leaf _b v = Ival.cardinal_zero_or_one v
   (* on Ival, both forms of cardinal coincide *)
 
   let compose _ _ = false
@@ -60,7 +60,8 @@ end
 module Location_Bytes = struct
 
   module M = struct
-    include Hptmap.Make (Base.Base) (Ival) (Comp_exact) (Hptmap_Info)
+    include Hptmap.Make_with_compositional_bool
+        (Base.Base) (Ival) (Comp_exact) (Hptmap_Info)
     let shape x = x
   end
   let () = Ast.add_monotonic_state M.self
@@ -376,7 +377,7 @@ module Zone = struct
     let dependencies = [ Ast.self ]
   end
 
-  module M = Hptmap.Make (Base.Base) (Int_Intervals) (Hptmap.Comp_unused) (Info)
+  module M = Hptmap.Make (Base.Base) (Int_Intervals) (Info)
   let () = Ast.add_monotonic_state M.self
   let clear_caches = M.clear_caches
 
diff --git a/src/kernel_services/analysis/stmts_graph.ml b/src/kernel_services/analysis/stmts_graph.ml
index bd7e0df01be..f93858ebf92 100644
--- a/src/kernel_services/analysis/stmts_graph.ml
+++ b/src/kernel_services/analysis/stmts_graph.ml
@@ -35,7 +35,6 @@ struct
   module HptmapStmtBool = Hptmap.Make
       (Cil_datatype.Stmt_Id)
       (struct include Datatype.Bool let pretty_debug = pretty end)
-      (Hptmap.Comp_unused)
       (struct
         let initial_values = []
         let dependencies = [ Ast.self ]
diff --git a/src/libraries/utils/hptmap.ml b/src/libraries/utils/hptmap.ml
index 9bc3422c355..a4678edd33b 100644
--- a/src/libraries/utils/hptmap.ml
+++ b/src/libraries/utils/hptmap.ml
@@ -93,13 +93,6 @@ struct
 end
 type tag = Tag_comp.t
 
-module Comp_unused =
-struct
-  let e = false
-  let f _ _ = false
-  let compose _ _ = false
-end
-
 (* A tree is either empty, or a leaf node, containing both
    the integer key and a piece of data, or a binary node.
    Each binary node carries two integers. The first one is
@@ -592,8 +585,8 @@ module type Compositional_bool = sig
   type key
   type v
 
-  val e : bool
-  val f : key -> v -> bool
+  val empty : bool
+  val leaf : key -> v -> bool
   val compose : bool -> bool -> bool
 end
 
@@ -604,7 +597,7 @@ module type Info = sig
   val dependencies : State.t list
 end
 
-module Make
+module Make_with_compositional_bool
     (Key: Id_Datatype)
     (V : V)
     (Compositional_bool : Compositional_bool with type key := Key.t
@@ -649,7 +642,7 @@ struct
 
   let compositional_bool t =
     match t with
-    | Empty -> Compositional_bool.e
+    | Empty -> Compositional_bool.empty
     | Leaf (_,_,tc)
     | Branch (_,_,_,_,tc) -> Tag_comp.get_comp tc
 
@@ -659,7 +652,7 @@ struct
 
   let initial_values =
     let tc k v =
-      let b = Compositional_bool.f k v in
+      let b = Compositional_bool.leaf k v in
       let tag = !current_tag in
       incr current_tag;
       Tag_comp.encode tag b
@@ -754,7 +747,7 @@ struct
     (* The test k < p+m and the implementation of [highest_bit] do not work
        with negative keys. *)
     assert (Key.id k >= 0);
-    let b = Compositional_bool.f k v in
+    let b = Compositional_bool.leaf k v in
     let tag = !current_tag in
     let new_tr = Leaf (k, v, Tag_comp.encode tag b) in
     let result = PatriciaHashconsTbl.merge new_tr in
@@ -1438,6 +1431,15 @@ struct
   let equal_subtree = equal
 end
 
+module Comp_unused = struct
+  let empty = false
+  let leaf _ _ = false
+  let compose _ _ = false
+end
+
+module Make (Key: Id_Datatype) (V : V) (Info: Info with type key := Key.t
+                                                    and type v := V.t)
+  = Make_with_compositional_bool (Key) (V) (Comp_unused) (Info)
 
 (*
 Local Variables:
diff --git a/src/libraries/utils/hptmap.mli b/src/libraries/utils/hptmap.mli
index 2c80736a6b6..725c205712d 100644
--- a/src/libraries/utils/hptmap.mli
+++ b/src/libraries/utils/hptmap.mli
@@ -32,12 +32,6 @@ module type Id_Datatype = sig
                        [equal k1 k2 ==> id k1 = id k2] *)
 end
 
-(** Values stored in the map *)
-module type V = sig
-  include Datatype.S
-  val pretty_debug: t Pretty_utils.formatter
-end
-
 (** This functor builds {!Hptmap_sig.Shape} for maps indexed by keys [Key],
     which contains all functions on hptmap that do not create or modify maps. *)
 module Shape (Key : Id_Datatype): sig
@@ -45,21 +39,10 @@ module Shape (Key : Id_Datatype): sig
   type 'a t = 'a map
 end
 
-(** A optional boolean information is maintained for each tree, by composing the
-    boolean on the subtrees and the value information present on each leaf.
-    Use {!Comp_unused} for a default implementation. *)
-module type Compositional_bool = sig
-  type key
-  type v
-
-  val e : bool
-  (** Value for the empty tree *)
-
-  val f : key -> v -> bool
-  (** Value for a leaf *)
-
-  val compose : bool -> bool -> bool
-  (** Composition of the values of two subtrees *)
+(** Values stored in the map *)
+module type V = sig
+  include Datatype.S
+  val pretty_debug: t Pretty_utils.formatter
 end
 
 (** Required information for the correctness of the hptmaps. *)
@@ -83,8 +66,6 @@ end
 module Make
     (Key : Id_Datatype)
     (V : V)
-    (_ : Compositional_bool with type key := Key.t
-                             and type v := V.t)
     (_ : Info with type key := Key.t
                and type v := V.t)
   : Hptmap_sig.S with type key = Key.t
@@ -92,14 +73,37 @@ module Make
                   and type 'v map = 'v Shape(Key).map
                   and type prefix = prefix
 
-(** Default implementation for the [Compositional_bool] argument of the functor
-    {!Make}. To be used when no interesting compositional bit can be computed. *)
-module Comp_unused : sig
-  val e : bool
-  val f : 'a -> 'b -> bool
+(** An additional boolean information is computed for each tree, by composing
+    the boolean on the subtrees and the value information on each leaf. *)
+module type Compositional_bool = sig
+  type key
+  type v
+
+  val empty : bool
+  (** Value for the empty tree *)
+
+  val leaf : key -> v -> bool
+  (** Value for a leaf *)
+
   val compose : bool -> bool -> bool
+  (** Composition of the values of two subtrees *)
 end
 
+(** This functor builds the complete module of maps indexed by keys [Key]
+    to values [V], with an additional boolean information maintained for
+    each tree. *)
+module Make_with_compositional_bool
+    (Key : Id_Datatype)
+    (V : V)
+    (_ : Compositional_bool with type key := Key.t
+                             and type v := V.t)
+    (_ : Info with type key := Key.t
+               and type v := V.t)
+  : Hptmap_sig.S with type key = Key.t
+                  and type v = V.t
+                  and type 'v map = 'v Shape(Key).map
+                  and type prefix = prefix
+
 (*
 Local Variables:
 compile-command: "make -C .."
diff --git a/src/libraries/utils/hptset.ml b/src/libraries/utils/hptset.ml
index 59a1dd0955c..28c3164935d 100644
--- a/src/libraries/utils/hptset.ml
+++ b/src/libraries/utils/hptset.ml
@@ -120,7 +120,7 @@ module Make
     let dependencies = Info.dependencies
   end
 
-  module M = Hptmap.Make (X) (Unit) (Hptmap.Comp_unused) (Hptmap_Info)
+  module M = Hptmap.Make (X) (Unit) (Hptmap_Info)
   include M
 
   let add k s = add k () s
diff --git a/src/plugins/eva/domains/cvalue/builtins_malloc.ml b/src/plugins/eva/domains/cvalue/builtins_malloc.ml
index 8c6155ad86b..b3be407a16d 100644
--- a/src/plugins/eva/domains/cvalue/builtins_malloc.ml
+++ b/src/plugins/eva/domains/cvalue/builtins_malloc.ml
@@ -42,8 +42,7 @@ module Hptmap_Info = struct
   let dependencies = [ Ast.self ]
 end
 
-module Base_hptmap =
-  Hptmap.Make (Base.Base) (Callstack) (Hptmap.Comp_unused) (Hptmap_Info)
+module Base_hptmap = Hptmap.Make (Base.Base) (Callstack) (Hptmap_Info)
 
 module Dynamic_Alloc_Bases =
   State_builder.Ref
diff --git a/src/plugins/eva/domains/equality/equality.ml b/src/plugins/eva/domains/equality/equality.ml
index 6f1de9bfc41..7d85984fcb2 100644
--- a/src/plugins/eva/domains/equality/equality.ml
+++ b/src/plugins/eva/domains/equality/equality.ml
@@ -98,7 +98,7 @@ module Set = struct
     let initial_values = []
     let dependencies = [ Ast.self ]
   end
-  include Hptmap.Make (HCE) (Data) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (HCE) (Data) (Hptmap_Info)
 
   let find_option elt map =
     try
diff --git a/src/plugins/eva/domains/gauges/gauges_domain.ml b/src/plugins/eva/domains/gauges/gauges_domain.ml
index fc759a3f6e6..a082f405044 100644
--- a/src/plugins/eva/domains/gauges/gauges_domain.ml
+++ b/src/plugins/eva/domains/gauges/gauges_domain.ml
@@ -285,7 +285,7 @@ module G = struct
   *)
   module MV = struct
 
-    include Hptmap.Make (Base) (Cvalue.V) (Hptmap.Comp_unused) (Hptmap_Info)
+    include Hptmap.Make (Base) (Cvalue.V) (Hptmap_Info)
 
     (* This function computes a pointwise union on two MVs assumed to have
        disjoint set of keys. *)
@@ -322,7 +322,7 @@ module G = struct
      are not incremented in one inner, but only in outemost one. *)
   module MC = struct
 
-    include Hptmap.Make (Base) (Bounds) (Hptmap.Comp_unused) (Hptmap_Info)
+    include Hptmap.Make (Base) (Bounds) (Hptmap_Info)
 
     (* This function computes a pointwise union on two MCs assumed to have
        disjoint set of keys. *)
diff --git a/src/plugins/eva/domains/hcexprs.ml b/src/plugins/eva/domains/hcexprs.ml
index d34f7bdfa6d..951d897535f 100644
--- a/src/plugins/eva/domains/hcexprs.ml
+++ b/src/plugins/eva/domains/hcexprs.ml
@@ -162,7 +162,7 @@ module HCEToZone = struct
 
   let cache_prefix = "Value.Symbolic_exprs.K2Z"
 
-  include Hptmap.Make (HCE) (Locations.Zone) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (HCE) (Locations.Zone) (Hptmap_Info)
 
   let is_included =
     let cache_name = cache_prefix ^ ".is_included" in
@@ -202,7 +202,7 @@ end
 
 module BaseToHCESet = struct
 
-  include Hptmap.Make (Base.Base) (HCESet) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Base.Base) (HCESet) (Hptmap_Info)
 
   let cache_prefix = "Value.Symbolic_exprs.B2K"
 
diff --git a/src/plugins/eva/domains/multidim/abstract_structure.ml b/src/plugins/eva/domains/multidim/abstract_structure.ml
index 3b00b1ddb72..eac548f2d77 100644
--- a/src/plugins/eva/domains/multidim/abstract_structure.ml
+++ b/src/plugins/eva/domains/multidim/abstract_structure.ml
@@ -83,8 +83,7 @@ struct
     let initial_values = []
     let dependencies = [ Ast.self ]
   end
-  module FieldMap =
-    Hptmap.Make (Field) (Values) (Hptmap.Comp_unused) (Hptmap_Info)
+  module FieldMap = Hptmap.Make (Field) (Values) (Hptmap_Info)
 
   type t = {
     padding: Bit.t;
diff --git a/src/plugins/eva/domains/multidim/multidim_domain.ml b/src/plugins/eva/domains/multidim/multidim_domain.ml
index 5a27426cb27..63db3c5dc72 100644
--- a/src/plugins/eva/domains/multidim/multidim_domain.ml
+++ b/src/plugins/eva/domains/multidim/multidim_domain.ml
@@ -346,7 +346,7 @@ struct
 
   module BaseMap =
   struct
-    include Hptmap.Make (Base.Base) (V) (Hptmap.Comp_unused) (Hptmap_Info)
+    include Hptmap.Make (Base.Base) (V) (Hptmap_Info)
 
     let find_or_top (state : t) (b : Base.t) =
       try find b state with Not_found -> V.top
diff --git a/src/plugins/eva/domains/octagons.ml b/src/plugins/eva/domains/octagons.ml
index 30367800d64..bd82bcbad8a 100644
--- a/src/plugins/eva/domains/octagons.ml
+++ b/src/plugins/eva/domains/octagons.ml
@@ -617,7 +617,7 @@ end
 (* Maps linking pairs of variables (X, Y) to intervals for X+Y and X-Y. *)
 module Octagons = struct
 
-  include Hptmap.Make (Pair) (Diamond) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Pair) (Diamond) (Hptmap_Info)
 
   let internal_join = join
 
@@ -738,7 +738,7 @@ module Relations = struct
 
   include Hptmap.Make (Variable)
       (struct include VarSet let pretty_debug = pretty end)
-      (Hptmap.Comp_unused) (Hptmap_Info)
+      (Hptmap_Info)
 
   let inter =
     let cache = Hptmap_sig.PersistentCache "Octagons.Relations.inter" in
@@ -779,7 +779,7 @@ end
 
 module Intervals = struct
 
-  include Hptmap.Make (Variable) (Ival) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Variable) (Ival) (Hptmap_Info)
 
   let internal_join = join
 
@@ -832,7 +832,7 @@ module VariableToDeps =
 struct
   let cache_prefix = "Eva.Octagons.VariableToDeps"
 
-  include Hptmap.Make (Variable) (Deps) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Variable) (Deps) (Hptmap_Info)
 
   let is_included: t -> t -> bool =
     let cache_name = cache_prefix ^ ".is_included" in
@@ -901,7 +901,7 @@ module BaseToVariables = struct
     let is_empty (s, t) = VSet.is_empty s && VSet.is_empty t
   end
 
-  include Hptmap.Make (Base.Base) (VSetPair) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Base.Base) (VSetPair) (Hptmap_Info)
 
   let cache_prefix = "Eva.Octagons.BaseToVariables"
 
diff --git a/src/plugins/eva/domains/simple_memory.ml b/src/plugins/eva/domains/simple_memory.ml
index 3dcd4d525a0..e8a9140134e 100644
--- a/src/plugins/eva/domains/simple_memory.ml
+++ b/src/plugins/eva/domains/simple_memory.ml
@@ -49,7 +49,7 @@ module Make_Memory (Info: sig val name: string end) (Value: Value) = struct
     let dependencies = [ Ast.self ]
   end
 
-  include Hptmap.Make (Base.Base) (Value) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Base.Base) (Value) (Hptmap_Info)
 
   let name = Info.name
 
diff --git a/src/plugins/eva/domains/symbolic_locs.ml b/src/plugins/eva/domains/symbolic_locs.ml
index cfb81b31b08..b7e3cf5ecc7 100644
--- a/src/plugins/eva/domains/symbolic_locs.ml
+++ b/src/plugins/eva/domains/symbolic_locs.ml
@@ -37,7 +37,7 @@ module K2V = struct
     let initial_values = []
     let dependencies = [ Ast.self ]
   end
-  module M = Hptmap.Make (K.HCE) (V) (Hptmap.Comp_unused) (Hptmap_Info)
+  module M = Hptmap.Make (K.HCE) (V) (Hptmap_Info)
 
   include M
 
diff --git a/src/plugins/eva/domains/traces_domain.ml b/src/plugins/eva/domains/traces_domain.ml
index 8d71f8881d0..e45a1cbba8a 100644
--- a/src/plugins/eva/domains/traces_domain.ml
+++ b/src/plugins/eva/domains/traces_domain.ml
@@ -232,7 +232,7 @@ module Graph = struct
     let initial_values = []
     let dependencies = [ Ast.self ]
   end
-  include Hptmap.Make (Node) (EdgeList) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Node) (EdgeList) (Hptmap_Info)
 
   let is_included =
     let cache = Hptmap_sig.NoCache in
diff --git a/src/plugins/from/from_compute.ml b/src/plugins/from/from_compute.ml
index dd4fec14857..14c0cc2acda 100644
--- a/src/plugins/from/from_compute.ml
+++ b/src/plugins/from/from_compute.ml
@@ -148,7 +148,7 @@ module ZoneStmtMap = struct
     let dependencies = [ Ast.self ]
   end
 
-  include Hptmap.Make (Stmt_Id) (Zone) (Hptmap.Comp_unused) (Hptmap_Info)
+  include Hptmap.Make (Stmt_Id) (Zone) (Hptmap_Info)
 
   let join =
     let decide _k z1 z2 = Zone.join z1 z2 in
diff --git a/src/plugins/impact/pdg_aux.ml b/src/plugins/impact/pdg_aux.ml
index d9ea3006d88..0163c1bf7d0 100644
--- a/src/plugins/impact/pdg_aux.ml
+++ b/src/plugins/impact/pdg_aux.ml
@@ -32,11 +32,7 @@ module Hptmap_Info = struct
 end
 
 module NS = struct
-  include Hptmap.Make
-      (PdgTypes.Node)
-      (Locations.Zone)
-      (Hptmap.Comp_unused)
-      (Hptmap_Info)
+  include Hptmap.Make (PdgTypes.Node) (Locations.Zone) (Hptmap_Info)
 
   let intersects =
     let name = "Impact.Pdg_aux.NS.intersects" in
diff --git a/src/plugins/pdg/pdg_types/pdgTypes.ml b/src/plugins/pdg/pdg_types/pdgTypes.ml
index e3e19dfd24c..e4af154aef0 100644
--- a/src/plugins/pdg/pdg_types/pdgTypes.ml
+++ b/src/plugins/pdg/pdg_types/pdgTypes.ml
@@ -281,7 +281,7 @@ module G = struct
     let dependencies = [ Ast.self ]
   end
 
-  module To = Hptmap.Make (Node) (DpdZone) (Hptmap.Comp_unused) (Hptmap_Info)
+  module To = Hptmap.Make (Node) (DpdZone) (Hptmap_Info)
   let () = Ast.add_hook_on_update (fun _ -> To.clear_caches ())
   (* See comment on previous call to Ast.add_hook_on_update *)
 
-- 
GitLab