diff --git a/src/plugins/eva/api/values_request.ml b/src/plugins/eva/api/values_request.ml index 1e109a39629a2e95115db3b403be1112542972ba..d810aa0d99919467335cedb4939e4c56408bd45a 100644 --- a/src/plugins/eva/api/values_request.ml +++ b/src/plugins/eva/api/values_request.ml @@ -360,6 +360,8 @@ end module Proxy(A : Analysis.S) : EvaProxy = struct + include Cvalue_domain.Getters (A.Dom) + open Eval type dstate = A.Dom.state or_top_bottom @@ -448,7 +450,7 @@ module Proxy(A : Analysis.S) : EvaProxy = struct (* --- Evaluates an expression or lvalue into an evaluation [result]. ----- *) let lval_to_offsetmap lval state = - let cvalue_state = A.Dom.get_cvalue_or_top state in + let cvalue_state = get_cvalue_or_top state in match lval with | Var vi, NoOffset -> let r = extract_single_var vi cvalue_state in diff --git a/src/plugins/eva/domains/apron/apron_domain.ml b/src/plugins/eva/domains/apron/apron_domain.ml index ff4d21571c61738d0be479e498f7f7f60ca32cfa..19a6077e455b005d09948033cac317b0808b6a88 100644 --- a/src/plugins/eva/domains/apron/apron_domain.ml +++ b/src/plugins/eva/domains/apron/apron_domain.ml @@ -714,16 +714,16 @@ let () = Floating_point.set_round_nearest_even () let make name (module Man: Input) = let module Domain = Make (Man) in + let name = "apron-" ^ name and experimental = true and priority = 1 in let descr = "Binding to the " ^ name ^ " domain of the Apron library. " ^ "See http://apron.cri.ensmp.fr/library for more details." in - let name = "apron-" ^ name in - let abstraction = - Abstractions.{ values = Single (module Main_values.Interval); - domain = Domain (module Domain); } - in - Abstractions.register ~name ~descr ~experimental:true ~priority:1 abstraction + Abstractions.Domain.register ~name ~descr ~experimental ~priority @@ Domain + { key = Domain.key ; domain = (module Domain) + ; values = Last Main_values.Interval.registered + ; locations = Last Main_locations.PLoc.registered + } let octagon = make "octagon" (module Apron_Octagon) let box = make "box" (module Apron_Box) @@ -731,6 +731,32 @@ let polka_loose = make "polka-loose" (module Apron_Polka_Loose) let polka_strict = make "polka-strict" (module Apron_Polka_Strict) let polka_equality = make "polka-equality" (module Apron_Polka_Equalities) +(* When the value abstraction contains both a cvalue and an interval + component (coming currently from an Apron domain), reduce them from each + other. If the Cvalue is not a scalar do nothing, because we do not + currently use Apron for pointer offsets. *) +let reduce_apron_itv cvalue ival = + match ival with + | None -> begin + try cvalue, Some (Cvalue.V.project_ival cvalue) + with Cvalue.V.Not_based_on_null -> cvalue, ival + end + | Some ival -> + try + let ival' = Cvalue.V.project_ival cvalue in + if Ival.is_int ival' + then + let reduced_ival = Ival.narrow ival ival' in + let cvalue = Cvalue.V.inject_ival reduced_ival in + cvalue, Some reduced_ival + else cvalue, Some ival + with Cvalue.V.Not_based_on_null -> cvalue, Some ival + +let () = + Abstractions.Reducer.register + Main_values.CVal.key Main_values.Interval.key reduce_apron_itv + + (* Local Variables: compile-command: "make -C ../../../../.. -j" diff --git a/src/plugins/eva/domains/apron/apron_domain.mli b/src/plugins/eva/domains/apron/apron_domain.mli index ede77986e150365603b0a878d6a84296bc42c17e..fb06185c0fa58124d767e27515907ea33a980230 100644 --- a/src/plugins/eva/domains/apron/apron_domain.mli +++ b/src/plugins/eva/domains/apron/apron_domain.mli @@ -24,11 +24,11 @@ the APRON library: http://apron.cri.ensmp.fr/library For now, this binding only processes scalar integer variables. *) -val octagon: Abstractions.flag -val box: Abstractions.flag -val polka_loose: Abstractions.flag -val polka_strict: Abstractions.flag -val polka_equality: Abstractions.flag +val octagon: Abstractions.Domain.registered +val box: Abstractions.Domain.registered +val polka_loose: Abstractions.Domain.registered +val polka_strict: Abstractions.Domain.registered +val polka_equality: Abstractions.Domain.registered (* Local Variables: diff --git a/src/plugins/eva/domains/cvalue/cvalue_domain.ml b/src/plugins/eva/domains/cvalue/cvalue_domain.ml index 6dca30d2ccbd4b4f031acc222c6c5018ed26f9de..53d65ab877e8d114d209cfa9c88ba52b20a301f7 100644 --- a/src/plugins/eva/domains/cvalue/cvalue_domain.ml +++ b/src/plugins/eva/domains/cvalue/cvalue_domain.ml @@ -585,6 +585,18 @@ end let () = Db.Value.display := (fun fmt kf -> State.display ~fmt kf) +let registered = + let name = "cvalue" and priority = 9 in + let descr = + "Main analysis domain, enabled by default. Should not be disabled." + in + Abstractions.Domain.register ~name ~priority ~descr @@ Domain + { key = State.key ; domain = (module State) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } + + type prefix = Hptmap.prefix module Subpart = struct type t = Model.subtree @@ -598,6 +610,23 @@ let distinct_subpart (a, _) (b, _) = with Model.Found_prefix (p, s1, s2) -> Some (p, s1, s2) let find_subpart (s, _) prefix = Model.find_prefix s prefix + +module Getters (Dom : Abstract.Domain.External) = struct + let get_cvalue = + match Dom.get State.key with + | None -> None + | Some get -> Some (fun s -> fst (get s)) + + let get_cvalue_or_top = + match Dom.get 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 + (* Local Variables: compile-command: "make -C ../../../../.." diff --git a/src/plugins/eva/domains/cvalue/cvalue_domain.mli b/src/plugins/eva/domains/cvalue/cvalue_domain.mli index 9c603e71b66e274c0befb703affd35767f17c8ac..c1b6c225d789b44de7cee952f2fa1f3dd9f923d8 100644 --- a/src/plugins/eva/domains/cvalue/cvalue_domain.mli +++ b/src/plugins/eva/domains/cvalue/cvalue_domain.mli @@ -27,6 +27,8 @@ module State : Abstract_domain.Leaf and type location = Main_locations.PLoc.location and type state = Cvalue.Model.t * Locals_scoping.clobbered_set +val registered : Abstractions.Domain.registered + (** Specific functions for partitioning optimizations. *) type prefix @@ -35,6 +37,13 @@ val distinct_subpart : State.t -> State.t -> (prefix * Subpart.t * Subpart.t) option val find_subpart : State.t -> prefix -> Subpart.t option +(** Special getters. *) + +module Getters (Dom : Abstract.Domain.External) : sig + val get_cvalue : (Dom.t -> Cvalue.Model.t) option + val get_cvalue_or_top : Dom.t -> Cvalue.Model.t + val get_cvalue_or_bottom : Dom.t Lattice_bounds.or_bottom -> Cvalue.Model.t +end (* diff --git a/src/plugins/eva/domains/domain_lift.ml b/src/plugins/eva/domains/domain_lift.ml index d527ada3927f690d19a2051d03b999c474e899ca..892db6b19f7783c5c9f6eb39d8d1325283c0ac4c 100644 --- a/src/plugins/eva/domains/domain_lift.ml +++ b/src/plugins/eva/domains/domain_lift.ml @@ -23,23 +23,16 @@ open Eval module type Conversion = sig - type extended_value - type extended_location - type internal_value - type internal_location - - val extend_val : internal_value -> extended_value - val restrict_val : extended_value -> internal_value - - val extend_loc : internal_location -> extended_location - val restrict_loc : extended_location -> internal_location + type extended + type internal + val extend: internal -> extended + val restrict: extended -> internal end - module Make (Domain: Abstract_domain.Leaf) - (Convert : Conversion with type internal_value := Domain.value - and type internal_location := Domain.location) + (Val: Conversion with type internal := Domain.value) + (Loc: Conversion with type internal := Domain.location) = struct include (Domain : Datatype.S_with_collections with type t = Domain.t) @@ -49,38 +42,38 @@ module Make let log_category = Domain.log_category - type value = Convert.extended_value - type location = Convert.extended_location + type value = Val.extended + type location = Loc.extended type origin = Domain.origin let extract_expr ~oracle context state exp = - let oracle exp = oracle exp >>=: Convert.restrict_val in + let oracle exp = oracle exp >>=: Val.restrict in Domain.extract_expr ~oracle context state exp >>=: fun (value, origin) -> - Convert.extend_val value, origin + Val.extend value, origin let extract_lval ~oracle context state lval typ loc = - let oracle exp = oracle exp >>=: Convert.restrict_val in - let loc = Convert.restrict_loc loc in + let oracle exp = oracle exp >>=: Val.restrict in + let loc = Loc.restrict loc in Domain.extract_lval ~oracle context state lval typ loc >>=: fun (value, origin) -> - Convert.extend_val value, origin + Val.extend value, origin let backward_location state lval typ loc value = Domain.backward_location - state lval typ (Convert.restrict_loc loc) (Convert.restrict_val value) + state lval typ (Loc.restrict loc) (Val.restrict value) >>-: fun (loc, value) -> - Convert.extend_loc loc, Convert.extend_val value + Loc.extend loc, Val.extend value let reduce_further state expr value = - let list = Domain.reduce_further state expr (Convert.restrict_val value) in - List.map (fun (e, v) -> e, Convert.extend_val v) list + let list = Domain.reduce_further state expr (Val.restrict value) in + List.map (fun (e, v) -> e, Val.extend v) list - let lift_left left = { left with lloc = Convert.restrict_loc left.lloc } + let lift_left left = { left with lloc = Loc.restrict left.lloc } let lift_flagged_value value = - { value with v = value.v >>-: Convert.restrict_val } + { value with v = value.v >>-: Val.restrict } let lift_assigned = function - | Assign value -> Assign (Convert.restrict_val value) + | Assign value -> Assign (Val.restrict value) | Copy (lval, value) -> Copy (lift_left lval, lift_flagged_value value) let lift_argument arg = { arg with avalue = lift_assigned arg.avalue } @@ -98,7 +91,7 @@ module Make | `Top -> `Top in let lift_record r = { r with value = lift_flagged_value r.value } in - let lift_loc_record r = { r with loc = Convert.restrict_loc r.loc } in + let lift_loc_record r = { r with loc = Loc.restrict r.loc } in let open Abstract_domain in let find expr = valuation.find expr >>> lift_record in let find_loc lval = valuation.find_loc lval >>> lift_loc_record in @@ -126,7 +119,7 @@ module Make let show_expr valuation = Domain.show_expr (lift_valuation valuation) let lift_logic_dep dep = - let location = Option.map Convert.restrict_loc dep.location in + let location = Option.map Loc.restrict dep.location in { dep with location } let lift_logic_assigns = function @@ -135,7 +128,7 @@ module Make let logic_assign assigns location state = let assigns = Option.map (fun (a, s) -> lift_logic_assigns a, s) assigns in - Domain.logic_assign assigns (Convert.restrict_loc location) state + Domain.logic_assign assigns (Loc.restrict location) state let evaluate_predicate = Domain.evaluate_predicate let reduce_by_predicate = Domain.reduce_by_predicate @@ -150,7 +143,7 @@ module Make let empty = Domain.empty let initialize_variable lval loc ~initialized init_value state = - let loc = Convert.restrict_loc loc in + let loc = Loc.restrict loc in Domain.initialize_variable lval loc ~initialized init_value state let initialize_variable_using_type = Domain.initialize_variable_using_type diff --git a/src/plugins/eva/domains/domain_lift.mli b/src/plugins/eva/domains/domain_lift.mli index a1f2fba0a689502ef007bf109e27bdda7e67d7a0..1630af2e95f852715ecb83e93548c442d29bc10e 100644 --- a/src/plugins/eva/domains/domain_lift.mli +++ b/src/plugins/eva/domains/domain_lift.mli @@ -20,28 +20,20 @@ (* *) (**************************************************************************) - module type Conversion = sig - type extended_value - type extended_location - type internal_value - type internal_location - - val extend_val : internal_value -> extended_value - val restrict_val : extended_value -> internal_value - - val extend_loc : internal_location -> extended_location - val restrict_loc : extended_location -> internal_location + type extended + type internal + val extend: internal -> extended + val restrict: extended -> internal end - module Make (Domain: Abstract_domain.Leaf) - (Convert : Conversion with type internal_value := Domain.value - and type internal_location := Domain.location) + (Val: Conversion with type internal := Domain.value) + (Loc: Conversion with type internal := Domain.location) : Abstract.Domain.Internal with type state = Domain.state - and type value = Convert.extended_value - and type location = Convert.extended_location + and type value = Val.extended + and type location = Loc.extended and type origin = Domain.origin diff --git a/src/plugins/eva/domains/equality/equality_domain.ml b/src/plugins/eva/domains/equality/equality_domain.ml index 9c3d309cafddc1127bb56bd9fe631cdc0823da08..514bcc6815da9a3d4360beee7b4fc5c2e33f367e 100644 --- a/src/plugins/eva/domains/equality/equality_domain.ml +++ b/src/plugins/eva/domains/equality/equality_domain.ml @@ -528,3 +528,22 @@ module Make | ISEmpty | ISFormals -> Base.SetLattice.empty | ISCaller -> Base.SetLattice.top end + + + +module Register = struct + type location = Precise_locs.precise_location + module Make (V : Abstract.Value.External) = Make (V) +end + +let registered = + let name = "equality" and priority = 8 in + let descr = + "Infers equalities between syntactic C expressions. \ + Makes the analysis less dependent on temporary variables and \ + intermediate computations." + in + Abstractions.Domain.register ~name ~priority ~descr @@ Functor + { domain = (module Register) + ; locations = Last Main_locations.PLoc.registered + } diff --git a/src/plugins/eva/domains/equality/equality_domain.mli b/src/plugins/eva/domains/equality/equality_domain.mli index 56a182e4a562bc10ba07cb09b6d82a8dc5b2827b..f01057153ea6a01cd5f76c60ad819a72688bf687 100644 --- a/src/plugins/eva/domains/equality/equality_domain.mli +++ b/src/plugins/eva/domains/equality/equality_domain.mli @@ -42,3 +42,5 @@ module Make (Value : Abstract.Value.External) : sig val pretty_debug : Format.formatter -> t -> unit end + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/gauges/gauges_domain.ml b/src/plugins/eva/domains/gauges/gauges_domain.ml index 4906186a931315ff8a6ad691b4846240d0630b35..4214218302687d1d33e43d2c81f84bfcaf1dd968 100644 --- a/src/plugins/eva/domains/gauges/gauges_domain.ml +++ b/src/plugins/eva/domains/gauges/gauges_domain.ml @@ -1288,3 +1288,15 @@ module D : Abstract_domain.Leaf let top = G.empty (* must not be used, not neutral w.r.t. join (because join crashes...)!! *) end + +let registered = + let name = "gauges" and priority = 6 in + let descr = + "Infers linear inequalities between the variables modified within a loop \ + and a special loop counter." + in + Abstractions.Domain.register ~name ~priority ~descr @@ Domain + { key = D.key ; domain = (module D) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } diff --git a/src/plugins/eva/domains/gauges/gauges_domain.mli b/src/plugins/eva/domains/gauges/gauges_domain.mli index d268a29ab40632c4c2a0108e27baba86158db861..8a9636ef90e32912238fb6807137068f5f903565 100644 --- a/src/plugins/eva/domains/gauges/gauges_domain.mli +++ b/src/plugins/eva/domains/gauges/gauges_domain.mli @@ -26,3 +26,5 @@ module D: Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/inout_domain.ml b/src/plugins/eva/domains/inout_domain.ml index fce469fb49423b4973d549844d795b42ded00e54..04e59101a87a7347786c15ef0cdf1668c11a9c9c 100644 --- a/src/plugins/eva/domains/inout_domain.ml +++ b/src/plugins/eva/domains/inout_domain.ml @@ -267,6 +267,16 @@ module D let extract_lval ~oracle:_ _context _state _lv _typ _locs = top_query end +let registered = + let name = "inout" and priority = 5 and experimental = true in + let descr = "Infers the inputs and outputs of each function." in + Abstractions.Domain.register ~name ~priority ~experimental ~descr @@ Domain + { key = D.key ; domain = (module D) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } + + (* Local Variables: compile-command: "make -C ../../.." diff --git a/src/plugins/eva/domains/inout_domain.mli b/src/plugins/eva/domains/inout_domain.mli index a224b7ec61a763fcaabbb16c53f27b55bb8e1d12..282d599008c19fdbc76034752474275216ea206a 100644 --- a/src/plugins/eva/domains/inout_domain.mli +++ b/src/plugins/eva/domains/inout_domain.mli @@ -25,3 +25,5 @@ module D: Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/multidim/multidim_domain.ml b/src/plugins/eva/domains/multidim/multidim_domain.ml index 4d2cf0c8a9bad96390659645499ac1f0575017fe..96d084d7ff1514b9c6cb4561930a47175e304ecd 100644 --- a/src/plugins/eva/domains/multidim/multidim_domain.ml +++ b/src/plugins/eva/domains/multidim/multidim_domain.ml @@ -941,15 +941,16 @@ let multidim_hook (module Abstract: Abstractions.S) : (module Abstractions.S) = end) (* Registers the domain. *) -let flag = +let registered = let name = "multidim" and descr = "Improve the precision over arrays of structures \ or multidimensional arrays." and experimental = true - and abstraction = - Abstractions.{ values = Single (module Main_values.CVal); - domain = Domain (module Domain); } in - Abstractions.register ~name ~descr ~experimental abstraction + Abstractions.Domain.register ~name ~descr ~experimental @@ Domain + { key = Domain.key ; domain = (module Domain) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } -let () = Abstractions.register_hook multidim_hook +let () = Abstractions.Hooks.register multidim_hook diff --git a/src/plugins/eva/domains/multidim/multidim_domain.mli b/src/plugins/eva/domains/multidim/multidim_domain.mli index 5df3635d248f49ca8d75027a40dd28a28303ed1f..ce8f376e509af1586a9c62aca8961812da09b2c6 100644 --- a/src/plugins/eva/domains/multidim/multidim_domain.mli +++ b/src/plugins/eva/domains/multidim/multidim_domain.mli @@ -24,4 +24,4 @@ include Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location -val flag: Abstractions.flag +val registered: Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/numerors/numerors_domain.ml b/src/plugins/eva/domains/numerors/numerors_domain.ml index ec40528a61bead026670aee4fa5c71375fd94195..de24c17397dd96f4e478461ca0490e0d14974d91 100644 --- a/src/plugins/eva/domains/numerors/numerors_domain.ml +++ b/src/plugins/eva/domains/numerors/numerors_domain.ml @@ -151,17 +151,19 @@ let reduce_cast (module Abstract: Abstractions.S) = end: Abstractions.S) (* Register the domain as an Eva abstractions. *) +let registered = + let name = "numerors" and experimental = true in + let descr = + "Infers ranges for the absolute and relative errors \ + in floating-point computations. No support of loops." + in + Abstractions.Domain.register ~name ~experimental ~descr @@ Domain + { key = Domain.key ; domain = (module Domain) + ; values = Last Numerors_Value.registered + ; locations = Last Main_locations.PLoc.registered + } + let () = let open Abstractions in - let name = "numerors" - and descr = "Infers ranges for the absolute and relative errors \ - in floating-point computations. No support of loops." - and experimental = true - and abstraction = - { values = Single (module Numerors_value); - domain = Domain (module Domain); } - in - let reduced_product = Main_values.CVal.key, Numerors_value.key, reduce_error in - ignore (register ~name ~descr ~experimental abstraction); - register_value_reduction reduced_product; - register_hook reduce_cast + Reducer.register Main_values.CVal.key Numerors_Value.key reduce_error ; + Hooks.register reduce_cast diff --git a/src/plugins/eva/domains/numerors/numerors_domain.mli b/src/plugins/eva/domains/numerors/numerors_domain.mli index 707b9094300b0f0f16d9f5c235924e2fea512815..184f827d6225b2022c452269ed95f3785c6122aa 100644 --- a/src/plugins/eva/domains/numerors/numerors_domain.mli +++ b/src/plugins/eva/domains/numerors/numerors_domain.mli @@ -24,3 +24,5 @@ of floating-point computations. Nothing is exported: the domain is registered as an analysis abstraction in the Eva engine, enabled by the -eva-domains numerors option. *) + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/numerors/numerors_value.ml b/src/plugins/eva/domains/numerors/numerors_value.ml index 574087982663173f898299f30c7dc913dcb07f8d..6fcf0148880d9328523238309e2b6a4ae5bee494 100644 --- a/src/plugins/eva/domains/numerors/numerors_value.ml +++ b/src/plugins/eva/domains/numerors/numerors_value.ml @@ -29,369 +29,375 @@ module P = Precisions module Arith = Numerors_arithmetics +module Value = struct + + (*----------------------------------------------------------------------------- + * Abstract value for numerical errors estimation + *----------------------------------------------------------------------------- + * The abstract value is a record with four fields : + * - exact : interval abstraction of the real value + * - approx : interval abstraction of the float value + * - abs_err : interval abstraction of the absolute error value + * - rel_err : interval abstraction of the relative error value + * A zonotope abstraction for each of those fields may be added + *---------------------------------------------------------------------------*) + type err = Top | Zero | Elt of Arith.t + + + (*----------------------------------------------------------------------------- + * Pretty printer + *---------------------------------------------------------------------------*) + let pp_print fmt = function + | Elt t -> Arith.pretty fmt t + | Zero -> Format.fprintf fmt "{ZERO}" + | Top -> Format.fprintf fmt "{TOP}" + + + (*----------------------------------------------------------------------------- + * Set errors to top + *---------------------------------------------------------------------------*) + let set_absolute_to_top = function + | Elt e -> Elt { e with Arith.abs_err = I.top ~prec:P.Real } + | err -> err + + let set_relative_to_top = function + | Elt e -> Elt { e with Arith.rel_err = I.top ~prec:P.Real } + | err -> err + + + (*----------------------------------------------------------------------------- + * Lattice structure + *---------------------------------------------------------------------------*) + let top = Top + + let is_included x y = + match x, y with + | Zero, Zero | _, Top -> true + | Zero, Elt t -> Arith.is_included (Arith.zero t) t + | Elt a, Elt b -> Arith.is_included a b + | Elt _, Zero | Top, _ -> false -(*----------------------------------------------------------------------------- - * Abstract value for numerical errors estimation - *----------------------------------------------------------------------------- - * The abstract value is a record with four fields : - * - exact : interval abstraction of the real value - * - approx : interval abstraction of the float value - * - abs_err : interval abstraction of the absolute error value - * - rel_err : interval abstraction of the relative error value - * A zonotope abstraction for each of those fields may be added - *---------------------------------------------------------------------------*) -type err = Top | Zero | Elt of Arith.t - - -(*----------------------------------------------------------------------------- - * Pretty printer - *---------------------------------------------------------------------------*) -let pp_print fmt = function - | Elt t -> Arith.pretty fmt t - | Zero -> Format.fprintf fmt "{ZERO}" - | Top -> Format.fprintf fmt "{TOP}" - - -(*----------------------------------------------------------------------------- - * Set errors to top - *---------------------------------------------------------------------------*) -let set_absolute_to_top = function - | Elt e -> Elt { e with Arith.abs_err = I.top ~prec:P.Real } - | err -> err - -let set_relative_to_top = function - | Elt e -> Elt { e with Arith.rel_err = I.top ~prec:P.Real } - | err -> err - - -(*----------------------------------------------------------------------------- - * Lattice structure - *---------------------------------------------------------------------------*) -let top = Top - -let is_included x y = - match x, y with - | Zero, Zero | _, Top -> true - | Zero, Elt t -> Arith.is_included (Arith.zero t) t - | Elt a, Elt b -> Arith.is_included a b - | Elt _, Zero | Top, _ -> false - -let join x y = - match x, y with - | Zero, Zero -> Zero - | Top, _ | _, Top -> Top - | Elt a, Zero | Zero, Elt a -> Elt (Arith.join (Arith.zero a) a) - | Elt a, Elt b -> Elt (Arith.join a b) - -let narrow x y = - match x, y with - | Zero, Zero -> `Value Zero - | Top, t | t, Top -> `Value t - | Elt a, Zero | Zero, Elt a -> - Arith.narrow (Arith.zero a) a >>- fun t -> `Value (Elt t) - | Elt a, Elt b -> Arith.narrow a b >>- fun t -> `Value (Elt t) - -let reduce _ t = `Value t - - -(*----------------------------------------------------------------------------- - * Elements needed for Eva core - *---------------------------------------------------------------------------*) -module T = struct - type t = err - include Datatype.Undefined - let structural_descr = Structural_descr.t_unknown - let compare x y = + let join x y = match x, y with - | Elt a, Elt b -> Arith.compare a b - | Top, Top | Zero, Zero -> 0 - | Top, _ | _, Zero -> 1 - | _, Top | Zero, _ -> -1 - let equal = Datatype.from_compare - let hash = Hashtbl.hash - let reprs = [top] - let name = "Value.Numerors_values.Numerors" - let pretty = pp_print -end -include Datatype.Make(T) -let pretty_debug = pretty -let pretty_typ _ = pretty -let key = Structure.Key_Value.create_key "numerors_values" + | Zero, Zero -> Zero + | Top, _ | _, Top -> Top + | Elt a, Zero | Zero, Elt a -> Elt (Arith.join (Arith.zero a) a) + | Elt a, Elt b -> Elt (Arith.join a b) + let narrow x y = + match x, y with + | Zero, Zero -> `Value Zero + | Top, t | t, Top -> `Value t + | Elt a, Zero | Zero, Elt a -> + Arith.narrow (Arith.zero a) a >>- fun t -> `Value (Elt t) + | Elt a, Elt b -> Arith.narrow a b >>- fun t -> `Value (Elt t) + + let reduce _ t = `Value t + + + (*----------------------------------------------------------------------------- + * Elements needed for Eva core + *---------------------------------------------------------------------------*) + module T = struct + type t = err + include Datatype.Undefined + let structural_descr = Structural_descr.t_unknown + let compare x y = + match x, y with + | Elt a, Elt b -> Arith.compare a b + | Top, Top | Zero, Zero -> 0 + | Top, _ | _, Zero -> 1 + | _, Top | Zero, _ -> -1 + let equal = Datatype.from_compare + let hash = Hashtbl.hash + let reprs = [top] + let name = "Value.Numerors_values.Numerors" + let pretty = pp_print + end + include Datatype.Make(T) + let pretty_debug = pretty + let pretty_typ _ = pretty + + + (*----------------------------------------------------------------------------- + * Constructors + *---------------------------------------------------------------------------*) + let zero = Zero + let one = top + let top_int = top + let inject_int _ _ = top + + let create exact approx abs_err rel_err = + Elt (Arith.create exact approx abs_err rel_err) + + let of_ints ~prec min max = + let exact = I.of_ints ~prec:P.Real (min, max) in + let approx = I.of_ints ~prec:prec (min, max) in + let abs_err = I.zero ~prec:P.Real in + let rel_err = I.zero ~prec:P.Real in + create exact approx abs_err rel_err + + + (*----------------------------------------------------------------------------- + * Miscellaneous + *---------------------------------------------------------------------------*) + + (* Handle the computation mode for the forward operations *) + let mode_on_errors exact approx abs_err rel_err = + Elt (Arith.forward_interaction (Arith.create exact approx abs_err rel_err)) + + (*----------------------------------------------------------------------------- + * Arithmetic import + *---------------------------------------------------------------------------*) + module Exact = Arith.Exact + module Approx = Arith.Approx + module Abs_Err = Arith.Abs_Err + module Rel_Err = Arith.Rel_Err + + + (*----------------------------------------------------------------------------- + * Numerors value of a constant + *---------------------------------------------------------------------------*) + let constant _ = function + | CReal (r, fkind, opt) -> + let prec = Precisions.of_fkind fkind in + let exact = + match opt with + | Some s -> I.of_strings ~prec:Precisions.Real (s, s) + | None -> I.of_floats ~prec:Precisions.Real (r, r) + in + let approx = I.of_floats_without_rounding ~prec (r, r) in + let abs_err = I.sub approx exact in + let rel_err = + if I.is_zero exact then I.of_floats ~prec:P.Real (0.0, 0.0) + else I.div abs_err exact + in + mode_on_errors exact approx abs_err rel_err + | _ -> top + + + (*----------------------------------------------------------------------------- + * Forward unary operations on Numerors value + *---------------------------------------------------------------------------*) + let forward_unop _typ op v = + match v, op with + | Elt v, Neg -> + let exact , approx = Exact.Forward.neg v, Approx.Forward.neg v in + let abs_err = Abs_Err.Forward.neg v ~exact ~approx in + let rel_err = Rel_Err.Forward.neg v ~exact ~abs_err in + `Value (mode_on_errors exact approx abs_err rel_err) + | Zero, Neg -> `Value zero + | _, LNot | _, BNot | Top, _ -> `Value top + + + (*----------------------------------------------------------------------------- + * Forward cast on Numerors value + *----------------------------------------------------------------------------- + * The cast of integers into floats is actually handled by the Numerors + * domain in the module <MakeNumerorsCValuesProduct>. + *---------------------------------------------------------------------------*) + let forward_cast ~src_type ~dst_type = function + | Top -> `Value Top + | Zero -> `Value Zero + | Elt t -> + match src_type, dst_type with + | Eval_typ.TSFloat _, Eval_typ.TSFloat fk -> + `Value (Elt (Arith.change_prec (Precisions.of_fkind fk) t)) + | _, _ -> `Value top + + + (*----------------------------------------------------------------------------- + * Forward binary operations on Numerors values + *---------------------------------------------------------------------------*) + let forward_binop _typ op x y = + match x, y, op with + | Elt x, Elt y, PlusA -> + let exact , approx = Exact.Forward.add x y, Approx.Forward.add x y in + let abs_err = Abs_Err.Forward.add x y ~exact ~approx in + let rel_err = Rel_Err.Forward.add x y ~exact ~abs_err in + `Value (mode_on_errors exact approx abs_err rel_err) + | Elt x, Elt y, MinusA -> + let exact , approx = Exact.Forward.sub x y, Approx.Forward.sub x y in + let abs_err = Abs_Err.Forward.sub x y ~exact ~approx in + let rel_err = Rel_Err.Forward.sub x y ~exact ~abs_err in + `Value (mode_on_errors exact approx abs_err rel_err) + | Elt x, Elt y, Mult -> + let exact , approx = Exact.Forward.mul x y, Approx.Forward.mul x y in + let abs_err = Abs_Err.Forward.mul x y ~exact ~approx in + let rel_err = Rel_Err.Forward.mul x y ~exact ~abs_err in + `Value (mode_on_errors exact approx abs_err rel_err) + | Elt x, Elt y, Div -> + let exact , approx = Exact.Forward.div x y, Approx.Forward.div x y in + let abs_err = Abs_Err.Forward.div x y ~exact ~approx in + let rel_err = Rel_Err.Forward.div x y ~exact ~abs_err in + `Value (mode_on_errors exact approx abs_err rel_err) + | _, _, _ -> `Value top + + (*----------------------------------------------------------------------------- + * Backward unary operations on Numerors values + *---------------------------------------------------------------------------*) + let backward_unop ~typ_arg:_ op ~arg ~res = + match arg, res, op with + | Elt x, Elt r, Neg -> + Exact.Backward.neg x r >>- fun exact -> + Approx.Backward.neg x r >>- fun approx -> + Abs_Err.Backward.neg x r >>- fun abs_err -> + Rel_Err.Backward.neg x r >>- fun rel_err -> + `Value (Some (create exact approx abs_err rel_err)) + | _, _, _ -> `Value None + + + (*----------------------------------------------------------------------------- + * Backward binary operations on Numerors values + *---------------------------------------------------------------------------*) + let backward_binop ~input_type:_ ~resulting_type:_ op ~left ~right ~result = + match left, right, result, op with + | Elt x, Elt y, Elt r, PlusA -> + Exact.Backward.add x y r >>- fun (x_exact , y_exact ) -> + Approx.Backward.add x y r >>- fun (x_approx , y_approx ) -> + Abs_Err.Backward.add x y r >>- fun (x_abs_err , y_abs_err ) -> + Rel_Err.Backward.add x y r >>- fun (x_rel_err , y_rel_err ) -> + let x = create x_exact x_approx x_abs_err x_rel_err in + let y = create y_exact y_approx y_abs_err y_rel_err in + `Value (Some x, Some y) + | Elt x, Elt y, Elt r, MinusA -> + Exact.Backward.sub x y r >>- fun (x_exact , y_exact ) -> + Approx.Backward.sub x y r >>- fun (x_approx , y_approx ) -> + Abs_Err.Backward.sub x y r >>- fun (x_abs_err , y_abs_err ) -> + Rel_Err.Backward.sub x y r >>- fun (x_rel_err , y_rel_err ) -> + let x = create x_exact x_approx x_abs_err x_rel_err in + let y = create y_exact y_approx y_abs_err y_rel_err in + `Value (Some x, Some y) + | Elt x, Elt y, Elt r, Mult -> + Exact.Backward.mul x y r >>- fun (x_exact , y_exact ) -> + Approx.Backward.mul x y r >>- fun (x_approx , y_approx ) -> + Abs_Err.Backward.mul x y r >>- fun (x_abs_err , y_abs_err ) -> + Rel_Err.Backward.mul x y r >>- fun (x_rel_err , y_rel_err ) -> + let x = create x_exact x_approx x_abs_err x_rel_err in + let y = create y_exact y_approx y_abs_err y_rel_err in + `Value (Some x, Some y) + | Elt x, Elt y, Elt r, Div -> + Exact.Backward.div x y r >>- fun (x_exact , y_exact ) -> + Approx.Backward.div x y r >>- fun (x_approx , y_approx ) -> + Abs_Err.Backward.div x y r >>- fun (x_abs_err , y_abs_err ) -> + Rel_Err.Backward.div x y r >>- fun (x_rel_err , y_rel_err ) -> + let x = create x_exact x_approx x_abs_err x_rel_err in + let y = create y_exact y_approx y_abs_err y_rel_err in + `Value (Some x, Some y) + (* x == y *) + | _, _, Zero, Ne -> narrow left right >>-: fun t -> Some t, Some t + (* x < y *) + | Elt x, Elt y, Zero, Ge -> + Arith.Backward_Comparisons.lt x y >>-: fun (x, y) -> + Some (Elt x), Some (Elt y) + (* x <= y *) + | Elt x, Elt y, Zero, Gt -> (* x >= y *) + Arith.Backward_Comparisons.le x y >>-: fun (x, y) -> + Some (Elt x), Some (Elt y) + (* x >= y *) + | Elt x, Elt y, Zero, Lt -> + Arith.Backward_Comparisons.ge x y >>-: fun (x, y) -> + Some (Elt x), Some (Elt y) + (* x > y *) + | Elt x, Elt y, Zero, Le -> + Arith.Backward_Comparisons.gt x y >>-: fun (x, y) -> + Some (Elt x), Some (Elt y) + | _ -> `Value (None, None) + + + (*----------------------------------------------------------------------------- + * Operations not handled on Numerors values + *---------------------------------------------------------------------------*) + + let assume_non_zero v = `Unknown v + let assume_bounded _kind _bound v = `Unknown v + let assume_not_nan ~assume_finite:_ _fkind v = `Unknown v + let assume_pointer v = `Unknown v + let assume_comparable _cmp v1 v2 = `Unknown (v1, v2) + + let rewrap_integer _ _ = top + let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = `Value None + let resolve_functions _ = `Top, true + let replace_base _substitution t = t + + + (*----------------------------------------------------------------------------- + * Built-in to create values in a interval + *---------------------------------------------------------------------------*) + let dbetween min max = + match min, max with + | Elt min, Elt max -> + let z = I.zero ~prec:P.Real in + let f = I.join min.Arith.approx max.Arith.approx in + let r = I.mul f (I.of_ints ~prec:P.Real (1, 1)) in + let r = { Arith.exact = r + ; Arith.approx = f + ; Arith.abs_err = z + ; Arith.rel_err = z + } + in `Value (Elt r) + | _, _ -> `Value top -(*----------------------------------------------------------------------------- - * Constructors - *---------------------------------------------------------------------------*) -let zero = Zero -let one = top -let top_int = top -let inject_int _ _ = top - -let create exact approx abs_err rel_err = - Elt (Arith.create exact approx abs_err rel_err) - -let of_ints ~prec min max = - let exact = I.of_ints ~prec:P.Real (min, max) in - let approx = I.of_ints ~prec:prec (min, max) in - let abs_err = I.zero ~prec:P.Real in - let rel_err = I.zero ~prec:P.Real in - create exact approx abs_err rel_err - - -(*----------------------------------------------------------------------------- - * Miscellaneous - *---------------------------------------------------------------------------*) - -(* Handle the computation mode for the forward operations *) -let mode_on_errors exact approx abs_err rel_err = - Elt (Arith.forward_interaction (Arith.create exact approx abs_err rel_err)) - -(*----------------------------------------------------------------------------- - * Arithmetic import - *---------------------------------------------------------------------------*) -module Exact = Arith.Exact -module Approx = Arith.Approx -module Abs_Err = Arith.Abs_Err -module Rel_Err = Arith.Rel_Err - - -(*----------------------------------------------------------------------------- - * Numerors value of a constant - *---------------------------------------------------------------------------*) -let constant _ = function - | CReal (r, fkind, opt) -> - let prec = Precisions.of_fkind fkind in - let exact = - match opt with - | Some s -> I.of_strings ~prec:Precisions.Real (s, s) - | None -> I.of_floats ~prec:Precisions.Real (r, r) - in - let approx = I.of_floats_without_rounding ~prec (r, r) in - let abs_err = I.sub approx exact in - let rel_err = - if I.is_zero exact then I.of_floats ~prec:P.Real (0.0, 0.0) - else I.div abs_err exact - in - mode_on_errors exact approx abs_err rel_err - | _ -> top - - -(*----------------------------------------------------------------------------- - * Forward unary operations on Numerors value - *---------------------------------------------------------------------------*) -let forward_unop _typ op v = - match v, op with - | Elt v, Neg -> - let exact , approx = Exact.Forward.neg v, Approx.Forward.neg v in - let abs_err = Abs_Err.Forward.neg v ~exact ~approx in - let rel_err = Rel_Err.Forward.neg v ~exact ~abs_err in - `Value (mode_on_errors exact approx abs_err rel_err) - | Zero, Neg -> `Value zero - | _, LNot | _, BNot | Top, _ -> `Value top - - -(*----------------------------------------------------------------------------- - * Forward cast on Numerors value - *----------------------------------------------------------------------------- - * The cast of integers into floats is actually handled by the Numerors - * domain in the module <MakeNumerorsCValuesProduct>. - *---------------------------------------------------------------------------*) -let forward_cast ~src_type ~dst_type = function - | Top -> `Value Top - | Zero -> `Value Zero - | Elt t -> - match src_type, dst_type with - | Eval_typ.TSFloat _, Eval_typ.TSFloat fk -> - `Value (Elt (Arith.change_prec (Precisions.of_fkind fk) t)) + let rbetween min max = + match min, max with + | Elt min, Elt max -> + let approx = I.join min.Arith.approx max.Arith.approx in + let exact = I.mul approx (I.of_ints ~prec:P.Real (1, 1)) in + let rel_err = I.epsilon (I.prec min.Arith.approx) in + let abs_err = I.mul rel_err exact in + let res = { Arith.exact = exact + ; Arith.approx = approx + ; Arith.abs_err = abs_err + ; Arith.rel_err = rel_err + } + in `Value (Elt res) | _, _ -> `Value top -(*----------------------------------------------------------------------------- - * Forward binary operations on Numerors values - *---------------------------------------------------------------------------*) -let forward_binop _typ op x y = - match x, y, op with - | Elt x, Elt y, PlusA -> - let exact , approx = Exact.Forward.add x y, Approx.Forward.add x y in - let abs_err = Abs_Err.Forward.add x y ~exact ~approx in - let rel_err = Rel_Err.Forward.add x y ~exact ~abs_err in - `Value (mode_on_errors exact approx abs_err rel_err) - | Elt x, Elt y, MinusA -> - let exact , approx = Exact.Forward.sub x y, Approx.Forward.sub x y in - let abs_err = Abs_Err.Forward.sub x y ~exact ~approx in - let rel_err = Rel_Err.Forward.sub x y ~exact ~abs_err in - `Value (mode_on_errors exact approx abs_err rel_err) - | Elt x, Elt y, Mult -> - let exact , approx = Exact.Forward.mul x y, Approx.Forward.mul x y in - let abs_err = Abs_Err.Forward.mul x y ~exact ~approx in - let rel_err = Rel_Err.Forward.mul x y ~exact ~abs_err in - `Value (mode_on_errors exact approx abs_err rel_err) - | Elt x, Elt y, Div -> - let exact , approx = Exact.Forward.div x y, Approx.Forward.div x y in - let abs_err = Abs_Err.Forward.div x y ~exact ~approx in - let rel_err = Rel_Err.Forward.div x y ~exact ~abs_err in - `Value (mode_on_errors exact approx abs_err rel_err) - | _, _, _ -> `Value top - -(*----------------------------------------------------------------------------- - * Backward unary operations on Numerors values - *---------------------------------------------------------------------------*) -let backward_unop ~typ_arg:_ op ~arg ~res = - match arg, res, op with - | Elt x, Elt r, Neg -> - Exact.Backward.neg x r >>- fun exact -> - Approx.Backward.neg x r >>- fun approx -> - Abs_Err.Backward.neg x r >>- fun abs_err -> - Rel_Err.Backward.neg x r >>- fun rel_err -> - `Value (Some (create exact approx abs_err rel_err)) - | _, _, _ -> `Value None - - -(*----------------------------------------------------------------------------- - * Backward binary operations on Numerors values - *---------------------------------------------------------------------------*) -let backward_binop ~input_type:_ ~resulting_type:_ op ~left ~right ~result = - match left, right, result, op with - | Elt x, Elt y, Elt r, PlusA -> - Exact.Backward.add x y r >>- fun (x_exact , y_exact ) -> - Approx.Backward.add x y r >>- fun (x_approx , y_approx ) -> - Abs_Err.Backward.add x y r >>- fun (x_abs_err , y_abs_err ) -> - Rel_Err.Backward.add x y r >>- fun (x_rel_err , y_rel_err ) -> - let x = create x_exact x_approx x_abs_err x_rel_err in - let y = create y_exact y_approx y_abs_err y_rel_err in - `Value (Some x, Some y) - | Elt x, Elt y, Elt r, MinusA -> - Exact.Backward.sub x y r >>- fun (x_exact , y_exact ) -> - Approx.Backward.sub x y r >>- fun (x_approx , y_approx ) -> - Abs_Err.Backward.sub x y r >>- fun (x_abs_err , y_abs_err ) -> - Rel_Err.Backward.sub x y r >>- fun (x_rel_err , y_rel_err ) -> - let x = create x_exact x_approx x_abs_err x_rel_err in - let y = create y_exact y_approx y_abs_err y_rel_err in - `Value (Some x, Some y) - | Elt x, Elt y, Elt r, Mult -> - Exact.Backward.mul x y r >>- fun (x_exact , y_exact ) -> - Approx.Backward.mul x y r >>- fun (x_approx , y_approx ) -> - Abs_Err.Backward.mul x y r >>- fun (x_abs_err , y_abs_err ) -> - Rel_Err.Backward.mul x y r >>- fun (x_rel_err , y_rel_err ) -> - let x = create x_exact x_approx x_abs_err x_rel_err in - let y = create y_exact y_approx y_abs_err y_rel_err in - `Value (Some x, Some y) - | Elt x, Elt y, Elt r, Div -> - Exact.Backward.div x y r >>- fun (x_exact , y_exact ) -> - Approx.Backward.div x y r >>- fun (x_approx , y_approx ) -> - Abs_Err.Backward.div x y r >>- fun (x_abs_err , y_abs_err ) -> - Rel_Err.Backward.div x y r >>- fun (x_rel_err , y_rel_err ) -> - let x = create x_exact x_approx x_abs_err x_rel_err in - let y = create y_exact y_approx y_abs_err y_rel_err in - `Value (Some x, Some y) - (* x == y *) - | _, _, Zero, Ne -> narrow left right >>-: fun t -> Some t, Some t - (* x < y *) - | Elt x, Elt y, Zero, Ge -> - Arith.Backward_Comparisons.lt x y >>-: fun (x, y) -> - Some (Elt x), Some (Elt y) - (* x <= y *) - | Elt x, Elt y, Zero, Gt -> (* x >= y *) - Arith.Backward_Comparisons.le x y >>-: fun (x, y) -> - Some (Elt x), Some (Elt y) - (* x >= y *) - | Elt x, Elt y, Zero, Lt -> - Arith.Backward_Comparisons.ge x y >>-: fun (x, y) -> - Some (Elt x), Some (Elt y) - (* x > y *) - | Elt x, Elt y, Zero, Le -> - Arith.Backward_Comparisons.gt x y >>-: fun (x, y) -> - Some (Elt x), Some (Elt y) - | _ -> `Value (None, None) - - -(*----------------------------------------------------------------------------- - * Operations not handled on Numerors values - *---------------------------------------------------------------------------*) - -let assume_non_zero v = `Unknown v -let assume_bounded _kind _bound v = `Unknown v -let assume_not_nan ~assume_finite:_ _fkind v = `Unknown v -let assume_pointer v = `Unknown v -let assume_comparable _cmp v1 v2 = `Unknown (v1, v2) - -let rewrap_integer _ _ = top -let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = `Value None -let resolve_functions _ = `Top, true -let replace_base _substitution t = t - - -(*----------------------------------------------------------------------------- - * Built-in to create values in a interval - *---------------------------------------------------------------------------*) -let dbetween min max = - match min, max with - | Elt min, Elt max -> - let z = I.zero ~prec:P.Real in - let f = I.join min.Arith.approx max.Arith.approx in - let r = I.mul f (I.of_ints ~prec:P.Real (1, 1)) in - let r = { Arith.exact = r - ; Arith.approx = f - ; Arith.abs_err = z - ; Arith.rel_err = z - } - in `Value (Elt r) - | _, _ -> `Value top - -let rbetween min max = - match min, max with - | Elt min, Elt max -> - let approx = I.join min.Arith.approx max.Arith.approx in - let exact = I.mul approx (I.of_ints ~prec:P.Real (1, 1)) in - let rel_err = I.epsilon (I.prec min.Arith.approx) in - let abs_err = I.mul rel_err exact in - let res = { Arith.exact = exact - ; Arith.approx = approx - ; Arith.abs_err = abs_err - ; Arith.rel_err = rel_err - } - in `Value (Elt res) - | _, _ -> `Value top - - -(*----------------------------------------------------------------------------- - * Built-in for square root - *---------------------------------------------------------------------------*) -let sqrt = function - | Elt x -> - let exact = Exact.Forward.sqrt x in - let approx = Approx.Forward.sqrt x in - let abs_err = Abs_Err.Forward.sqrt x ~exact ~approx in - let rel_err = Rel_Err.Forward.sqrt x ~exact ~abs_err in - `Value (create exact approx abs_err rel_err) - | _ -> `Value top - - -(*----------------------------------------------------------------------------- - * Built-in for transcendental functions - *---------------------------------------------------------------------------*) -let log = function - | Elt x -> - let exact = Exact.Forward.log x in - let approx = Approx.Forward.log x in - let abs_err = Abs_Err.Forward.log x ~exact ~approx in - let rel_err = Rel_Err.Forward.log x ~exact ~abs_err in - `Value (create exact approx abs_err rel_err) - | _ -> `Value top - -let exp = function - | Elt x -> - let exact = Exact.Forward.exp x in - let approx = Approx.Forward.exp x in - let abs_err = Abs_Err.Forward.exp x ~exact ~approx in - let rel_err = Rel_Err.Forward.exp x ~exact ~abs_err in - `Value (create exact approx abs_err rel_err) - | _ -> `Value top - -let get_max_absolute_error = function - | Elt x -> Some (snd (I.get_bounds (I.abs (x.Arith.abs_err)))) - | _ -> None - -let get_max_relative_error = function - | Elt x -> Some (snd (I.get_bounds (I.abs (x.Arith.rel_err)))) - | _ -> None + (*----------------------------------------------------------------------------- + * Built-in for square root + *---------------------------------------------------------------------------*) + let sqrt = function + | Elt x -> + let exact = Exact.Forward.sqrt x in + let approx = Approx.Forward.sqrt x in + let abs_err = Abs_Err.Forward.sqrt x ~exact ~approx in + let rel_err = Rel_Err.Forward.sqrt x ~exact ~abs_err in + `Value (create exact approx abs_err rel_err) + | _ -> `Value top + + + (*----------------------------------------------------------------------------- + * Built-in for transcendental functions + *---------------------------------------------------------------------------*) + let log = function + | Elt x -> + let exact = Exact.Forward.log x in + let approx = Approx.Forward.log x in + let abs_err = Abs_Err.Forward.log x ~exact ~approx in + let rel_err = Rel_Err.Forward.log x ~exact ~abs_err in + `Value (create exact approx abs_err rel_err) + | _ -> `Value top + + let exp = function + | Elt x -> + let exact = Exact.Forward.exp x in + let approx = Approx.Forward.exp x in + let abs_err = Abs_Err.Forward.exp x ~exact ~approx in + let rel_err = Rel_Err.Forward.exp x ~exact ~abs_err in + `Value (create exact approx abs_err rel_err) + | _ -> `Value top + + let get_max_absolute_error = function + | Elt x -> Some (snd (I.get_bounds (I.abs (x.Arith.abs_err)))) + | _ -> None + + let get_max_relative_error = function + | Elt x -> Some (snd (I.get_bounds (I.abs (x.Arith.rel_err)))) + | _ -> None +end + + +let key = Structure.Key_Value.create_key "numerors_values" +let registered = Abstractions.Value.register { key ; value = (module Value) } +include Value diff --git a/src/plugins/eva/domains/numerors/numerors_value.mli b/src/plugins/eva/domains/numerors/numerors_value.mli index 8d860a502b5a92ef31b5d45afd60d74e036d6b24..bb489ae9797b42c3a5d0811e9a5aeab44ab9e83c 100644 --- a/src/plugins/eva/domains/numerors/numerors_value.mli +++ b/src/plugins/eva/domains/numerors/numerors_value.mli @@ -21,6 +21,7 @@ (**************************************************************************) include Abstract_value.Leaf +val registered : t Abstractions.Value.registered val pretty_debug : t Pretty_utils.formatter diff --git a/src/plugins/eva/domains/octagons.ml b/src/plugins/eva/domains/octagons.ml index 00adb0b62fdff9fa42f670690688e674271b7a16..c1533364c88daa1d6795fcbae654b169177207e0 100644 --- a/src/plugins/eva/domains/octagons.ml +++ b/src/plugins/eva/domains/octagons.ml @@ -1953,3 +1953,15 @@ module Domain = struct end include Domain + +let registered = + let name = "octagon" and priority = 6 in + let descr = + "Infers relations between scalar variables of the form b ≤ ±X ± Y ≤ e, \ + where X, Y are program variables and b, e are constants." + in + Abstractions.Domain.register ~name ~priority ~descr @@ Domain + { key ; domain = (module Domain) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } diff --git a/src/plugins/eva/domains/octagons.mli b/src/plugins/eva/domains/octagons.mli index 2625b8b6b75f0296d68e8cdb9082babcf40cc8d0..9968be3051ef9e9e5adaa30003203acfa8788553 100644 --- a/src/plugins/eva/domains/octagons.mli +++ b/src/plugins/eva/domains/octagons.mli @@ -23,3 +23,5 @@ include Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/offsm_domain.ml b/src/plugins/eva/domains/offsm_domain.ml index 26aad6c665b306b8e84a9c2d55e5eba8f38fa08d..49f49c3548fd46687442dc2ef19b663eb49924a5 100644 --- a/src/plugins/eva/domains/offsm_domain.ml +++ b/src/plugins/eva/domains/offsm_domain.ml @@ -224,3 +224,15 @@ module D : Abstract_domain.Leaf let loc = Precise_locs.imprecise_location location in kill loc state end + + +let registered = + let name = "bitwise" and priority = 3 in + let descr = + "Infers bitwise information to interpret more precisely bitwise operators." + in + Abstractions.Domain.register ~name ~priority ~descr @@ Domain + { key = D.key ; domain = (module D) + ; values = Last Offsm_value.Offsm.registered + ; locations = Last Main_locations.PLoc.registered + } diff --git a/src/plugins/eva/domains/offsm_domain.mli b/src/plugins/eva/domains/offsm_domain.mli index 9b69966616999b75f6208ea619504e1f0a25246b..ed4bee0dda7010e6221b57052a540b7f436f44cb 100644 --- a/src/plugins/eva/domains/offsm_domain.mli +++ b/src/plugins/eva/domains/offsm_domain.mli @@ -23,3 +23,5 @@ module D : Abstract_domain.Leaf with type value = Offsm_value.offsm_or_top and type location = Precise_locs.precise_location + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/printer_domain.ml b/src/plugins/eva/domains/printer_domain.ml index 270925f3ad699eca48fa9607b75ac758a012320f..0e04039aa556fb0a8b6216d4b19d34f15110bbd4 100644 --- a/src/plugins/eva/domains/printer_domain.ml +++ b/src/plugins/eva/domains/printer_domain.ml @@ -125,4 +125,19 @@ module Simple : Simpler_domains.Simple_Cvalue = struct state end -include Domain_builder.Complete_Simple_Cvalue (Simple) +module Domain = Domain_builder.Complete_Simple_Cvalue (Simple) +include Domain + +let registered = + let name = "printer" in + let priority = 2 in + let descr = + "Debug domain, only useful for developers. Prints the transfer functions \ + used during the analysis." + in + Abstractions.Domain.register ~name ~descr ~priority @@ Domain + { key ; domain = (module Domain) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } + diff --git a/src/plugins/eva/domains/printer_domain.mli b/src/plugins/eva/domains/printer_domain.mli index c23cfc215f0c44367b4d2d17f0b68f16352ebf7a..2aa295d606ca971fde616646e8ab90559ef68c78 100644 --- a/src/plugins/eva/domains/printer_domain.mli +++ b/src/plugins/eva/domains/printer_domain.mli @@ -25,3 +25,5 @@ during an analysis. *) include Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/sign_domain.ml b/src/plugins/eva/domains/sign_domain.ml index ab0ce64335e51ca7b7f010faa459f30a07a2cc94..e2f073cb88412dc93fd51a58b0fae5d7edf62cb9 100644 --- a/src/plugins/eva/domains/sign_domain.ml +++ b/src/plugins/eva/domains/sign_domain.ml @@ -34,4 +34,15 @@ module Sign_Value = struct let builtins = [] end -include Simple_memory.Make_Domain (struct let name = "sign" end) (Sign_Value) +module Name = struct let name = "sign" end +module D = Simple_memory.Make_Domain (Name) (Sign_Value) +include D + +let registered = + let name = Name.name and priority = 4 in + let descr = "Infers the sign of program variables." in + Abstractions.Domain.register ~name ~priority ~descr @@ Domain + { key ; domain = (module D) + ; values = Last Sign_value.registered + ; locations = Last Main_locations.PLoc.registered + } diff --git a/src/plugins/eva/domains/sign_domain.mli b/src/plugins/eva/domains/sign_domain.mli index e8d8622c30a537200663fd9190a61f77582cf456..b5d5dfcd333eb039b8bb5dc9bd558c424b766d92 100644 --- a/src/plugins/eva/domains/sign_domain.mli +++ b/src/plugins/eva/domains/sign_domain.mli @@ -24,3 +24,5 @@ include Abstract_domain.Leaf with type value = Sign_value.t and type location = Precise_locs.precise_location + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/symbolic_locs.ml b/src/plugins/eva/domains/symbolic_locs.ml index 8dd06c85fa33a88e13d6daed4daddeee7173870f..319966cb259b3fe0488773d8df4ba298e356fc12 100644 --- a/src/plugins/eva/domains/symbolic_locs.ml +++ b/src/plugins/eva/domains/symbolic_locs.ml @@ -637,3 +637,17 @@ module D : Abstract_domain.Leaf let loc = Precise_locs.imprecise_location location in Memory.kill loc state end + + + +let registered = + let name = "symbolic-locations" and priority = 7 in + let descr = + "Infers values of symbolic locations represented by imprecise lvalues, \ + such as t[i] or *p when the possible values of [i] or [p] are imprecise." + in + Abstractions.Domain.register ~name ~priority ~descr @@ Domain + { key = D.key ; domain = (module D) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } diff --git a/src/plugins/eva/domains/symbolic_locs.mli b/src/plugins/eva/domains/symbolic_locs.mli index dba5b5878f7b7b7aaf2bc76f1a7b0e28b120fd0d..0c7e8d5f52f3dc6210d777456e327fe5c84c6170 100644 --- a/src/plugins/eva/domains/symbolic_locs.mli +++ b/src/plugins/eva/domains/symbolic_locs.mli @@ -26,3 +26,5 @@ module D: Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/domains/taint_domain.ml b/src/plugins/eva/domains/taint_domain.ml index 6abf186ae3fd6e7b52aa9c261fb0cabfa20b769f..a5ed670f5cd7464c64dcb0546a79a3849a40149d 100644 --- a/src/plugins/eva/domains/taint_domain.ml +++ b/src/plugins/eva/domains/taint_domain.ml @@ -582,17 +582,18 @@ let interpret_taint_logic end) (* Registers the domain. *) -let flag = +let registered = let name = "taint" and descr = "Taint analysis" and experimental = true - and abstraction = - Abstractions.{ values = Single (module Main_values.CVal); - domain = Domain (module TaintDomain); } in - Abstractions.register ~name ~descr ~experimental abstraction + Abstractions.Domain.register ~name ~descr ~experimental @@ Domain + { key = TaintDomain.key ; domain = (module TaintDomain) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } -let () = Abstractions.register_hook interpret_taint_logic +let () = Abstractions.Hooks.register interpret_taint_logic type taint = Direct | Indirect | Untainted diff --git a/src/plugins/eva/domains/taint_domain.mli b/src/plugins/eva/domains/taint_domain.mli index fa31e885b038f02c351fa5b87c073a8b4e5b8c44..2b26fa23af604b9eeb6ce7ff3274a226beee2957 100644 --- a/src/plugins/eva/domains/taint_domain.mli +++ b/src/plugins/eva/domains/taint_domain.mli @@ -26,7 +26,7 @@ include Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location -val flag: Abstractions.flag +val registered: Abstractions.Domain.registered type taint = | Direct | Indirect | Untainted diff --git a/src/plugins/eva/domains/traces_domain.ml b/src/plugins/eva/domains/traces_domain.ml index 396d0cfca6bc06b4a58a60255cb599cc4b1f58eb..e759bbfd3259161b7d79f3a27b0739df65c241b4 100644 --- a/src/plugins/eva/domains/traces_domain.ml +++ b/src/plugins/eva/domains/traces_domain.ml @@ -1260,6 +1260,16 @@ module D = struct then project_of_cfg return_exp state end +let registered = + let name = "traces" and priority = 2 and experimental = true in + let descr = + "Builds an over-approximation of all the traces that lead to a statement." + in + Abstractions.Domain.register ~name ~priority ~descr ~experimental @@ Domain + { key = D.key ; domain = (module D) + ; values = Last Main_values.CVal.registered + ; locations = Last Main_locations.PLoc.registered + } (* Local Variables: diff --git a/src/plugins/eva/domains/traces_domain.mli b/src/plugins/eva/domains/traces_domain.mli index 7e13fe9d4212dcdf761de59c8329580c06090f56..6c05df34c29aa6ed9c385bba763238016d4fb7d4 100644 --- a/src/plugins/eva/domains/traces_domain.mli +++ b/src/plugins/eva/domains/traces_domain.mli @@ -76,3 +76,5 @@ module D: Abstract_domain.Leaf with type value = Cvalue.V.t and type location = Precise_locs.precise_location and type state = state + +val registered : Abstractions.Domain.registered diff --git a/src/plugins/eva/engine/abstractions.ml b/src/plugins/eva/engine/abstractions.ml index 573457ba34d0d5877b5bc14e9c3250c314e523c7..f1e303a997477f45319baa16127df886550e2702 100644 --- a/src/plugins/eva/engine/abstractions.ml +++ b/src/plugins/eva/engine/abstractions.ml @@ -20,527 +20,786 @@ (* *) (**************************************************************************) -(* --- Registration types --------------------------------------------------- *) -type 'v value = - | Single of (module Abstract_value.Leaf with type t = 'v) - | Struct of 'v Abstract.Value.structure -type precise_loc = Precise_locs.precise_location - -module type leaf_domain = Abstract_domain.Leaf with type location = precise_loc - -module type domain_functor = - functor (Value: Abstract.Value.External) -> - (leaf_domain with type value = Value.t) - -type 'v domain = - | Domain: (module leaf_domain with type value = 'v) -> 'v domain - | Functor: (module domain_functor) -> _ domain +(* --- Values abstraction --------------------------------------------------- *) + +module Value = struct + type 'v structure = 'v Abstract.Value.structure + type 'v key = 'v Abstract.Value.key + let dec_eq = Abstract.Value.eq_structure + + type 'v value = (module Abstract_value.S with type t = 'v) + + + (* Value registration *) + type 'v register = { key : 'v key ; value : 'v value } + type 'v registered = 'v key * 'v value + let register : type v. v register -> v registered = + fun { key ; value } -> (key, value) + + + (* When registering a location or a domain, the user has to declare its + values dependencies, i.e the values it uses during its computations. To + simplify the abstraction build and the dependencies declaration, they + are handled as a simple list. Arguably not so simple as it is + heterogenous, but it is still a list. *) + type 'v dependencies = + | Last : 'v registered -> 'v dependencies + | (::) : 'a registered * 'b dependencies -> ('a * 'b) dependencies + + (* When building the abstraction, we will need to compare the dependencies + structure with the structured values. *) + let rec outline : type v. v dependencies -> v structure = function + | Last (key, value) -> Abstract.Value.Leaf (key, value) + | (k, v) :: tl -> Abstract.Value.(Node (Leaf (k, v), outline tl)) + + (* Folding over values dependencies *) + type 'a folder = { folder : 'v. 'v registered -> 'a -> 'a } + let rec fold : type v. 'a folder -> v dependencies -> 'a -> 'a = + fun { folder } dependencies acc -> + match dependencies with + | Last registered -> folder registered acc + | hd :: tl -> fold { folder } tl (folder hd acc) + + + (* The value abstraction build consists of accumulating registered values + into a structured value and then adding the needed operators, thus making + it interactive. A [Unit] structured abstraction is used for the initial + state and is discarded as soon as a real value abstraction is added. *) + module type Structured = Abstract.Value.Internal + module type Interactive = Abstract.Value.External + type 'a or_unit = Unit | Value of 'a + type structured = (module Structured) or_unit + type interactive = (module Interactive) or_unit + + (* Making a structured value interactive simply consists of adding the + needed operations using the Structure.Open functor.*) + let make_interactive : structured -> interactive = function + | Unit -> Unit + | Value (module Structured) -> + Value (module struct + include Structured + include Structure.Open (Abstract.Value) (Structured) + end) -type 'v abstraction = - { values: 'v value; - domain: 'v domain; } + (* Adding a registered value into a structured one consists of deciding if + a product is needed (which comes down to checking if the registered + value key we want to add is not in the structure), computing it, and + updating the structure. *) + let add : type v. v registered -> structured -> structured = + fun (key, input) structured -> + let open Abstract.Value in + match make_interactive structured with + | Unit -> + Value (module struct + include (val input) + let structure = Leaf (key, input) + end) + | Value (module Interactive) when not (Interactive.mem key) -> + Value (module struct + include Value_product.Make (Interactive) (val input) + let structure = Node (Interactive.structure, Leaf (key, input)) + end) + | _ -> structured + + (* The minimal value abstraction to use. *) + let init : structured = Unit + + (* During the complete abstraction build, we need to verify that there is at + least one value in the computed abstraction. + TODO: better error handling. *) + let assert_not_unit = function + | Unit -> Self.fatal "The built value cannot be unit." + | Value interactive -> interactive + + + (* When building the complete abstraction, we need to trick locations and + domains into thinking that their value dependencies are there, even if + the structured value type is not the good one. This is done through a + lift that requires conversion operations to interact with the subpart + of the structured value that matters for the location or the domain. + This functor is responsible of building such conversion operations. *) + module type From = sig type value val structure : value structure end + module Converter (From : From) (To : Interactive) = struct + type extended = To.t + let structure = From.structure + + let void_value () = + Self.fatal "Cannot register a value module from a Void structure." + + let rec set : type v. v structure -> v -> extended -> extended = function + | Leaf (key, _) -> To.set key + | Node (s1, s2) -> fun (v1, v2) ext -> set s2 v2 ext |> set s1 v1 + | Option (s, default) -> fun v -> set s (Option.value ~default v) + | Unit -> fun () value -> value + | Void -> void_value () + + let rec get : type v. v structure -> extended -> v = function + | Leaf (key, _) -> Option.get (To.get key) + | Node (s1, s2) -> fun v -> get s1 v, get s2 v + | Option (s, _) -> fun v -> Some (get s v) + | Unit -> fun _ -> () + | Void -> void_value () + + let replace = set structure + let extend v = replace v To.top + let restrict = get structure + end +end -type 't with_info = - { name: string; - experimental: bool; - priority: int; - abstraction: 't; } -type flag = Flag: 'v abstraction with_info -> flag -(* --- Config and registration ---------------------------------------------- *) +(* --- Locations abstraction ------------------------------------------------ *) -module Config = struct - module OptMode = Datatype.Option (Domain_mode) - module Element = struct - type t = flag * Domain_mode.t option - - (* Flags are sorted by increasing priority order, and then by name. *) - let compare (Flag f1, mode1) (Flag f2, mode2) = - let c = Datatype.Int.compare f1.priority f2.priority in - if c <> 0 then c else - let c = Datatype.String.compare f1.name f2.name in - if c <> 0 then c else - OptMode.compare mode1 mode2 - end +module Location = struct + type 'l structure = 'l Abstract.Location.structure + type 'l key = 'l Abstract.Location.key + let dec_eq = Abstract.Location.eq_structure - include Set.Make (Element) + type ('v, 'l) location = + (module Abstract_location.S with type value = 'v and type location = 'l) - let mem (Flag domain) = - exists (fun (Flag flag, _mode) -> flag.name = domain.name) - let abstractions = ref [] - let dynamic_abstractions = ref [] + (* When registering a location, we also need its key and its values + dependencies. *) + type ('v, 'l) register = + { key: 'l key + ; location : ('v, 'l) location + ; dependencies : 'v Value.dependencies + } - let register_domain_option ~name ~experimental ~descr = - let descr = if experimental then "Experimental. " ^ descr else descr in - Parameters.register_domain ~name ~descr + (* A registered location is simply a location with its key and its values + dependencies. *) + module type Registered = sig + include Abstract_location.S + val key : location key + val structure : location structure + val dependencies : value Value.dependencies + end + type 'l registered = (module Registered with type location = 'l) - let register ~name ~descr ?(experimental=false) ?(priority=0) abstraction = - register_domain_option ~name ~experimental ~descr; - let flag = Flag { name; experimental; priority; abstraction } in - abstractions := flag :: !abstractions; - flag + (* Location registration. *) + let register : type v l. (v, l) register -> l registered = + fun { key ; location ; dependencies } -> + (module struct + module Location = (val location) + include Location + let key = key + let structure = Abstract.Location.Leaf (key, (module Location)) + let dependencies = dependencies + end) + + + (* When registering a domain, the user has to declare its locations + dependencies, i.e the locations it uses during its computations. As + for the values, they are handled as a simple heterogenous list. *) + type 'l dependencies = + | Last : 'l registered -> 'l dependencies + | (::) : 'a registered * 'b dependencies -> ('a * 'b) dependencies + + (* When building the abstraction, we will need to compare dependencies + with the structured locations. *) + let rec outline : type l. l dependencies -> l structure = function + | Last (module R) -> R.structure + | (module R) :: tl -> Abstract.Location.Node (R.structure, outline tl) + + (* Folding over locations dependencies *) + type 'a folder = { folder : 'v. 'v registered -> 'a -> 'a } + let rec fold : type v. 'a folder -> v dependencies -> 'a -> 'a = + fun { folder } dependencies acc -> + match dependencies with + | Last registered -> folder registered acc + | hd :: tl -> fold { folder } tl (folder hd acc) + + (* Folding over the values dependencies of some locations dependencies. *) + let rec fold_values : type v. 'a Value.folder -> v dependencies -> 'a -> 'a = + fun folder dependencies acc -> + match dependencies with + | Last (module R) -> Value.fold folder R.dependencies acc + | (module R) :: tl -> + fold_values folder tl (Value.fold folder R.dependencies acc) + + + (* As for the value abstraction, building the location abstraction consists + of structuring the needed registered locations and then adding the needed + operators to make it interactive. However, a structured location is not + as simple as a structured value, as it needs to keep track of the value + abstraction it is based on. This value is supposed to be the complete + aggregation of all the values that are needed by the requested domains. *) + module type Structured = sig + type value + module Value : Value.Interactive with type t = value + module Location : Abstract.Location.Internal with type value = value + end - let dynamic_register ~name ~descr ?(experimental=false) ?(priority=0) make = - register_domain_option ~name ~experimental ~descr; - let make' () : flag = - Flag { name; experimental; priority; abstraction = make () } - in - dynamic_abstractions := (name,make') :: !dynamic_abstractions + module type Interactive = Abstract.Location.External + + (* We expose the type of the structured value we are based on to statically + ensure that we do not temper with it. As for the value abstractions, a + [Unit] structured abstraction is used for the initial state and is + discarded as soon as a location is added. *) + type ('u, 'l) or_unit = Unit of 'u | Location of 'l + type 'v value = (module Value.Interactive with type t = 'v) + type 'v structured_module = (module Structured with type value = 'v) + type 'v structured = ('v value, 'v structured_module) or_unit + type 'v interactive_module = (module Interactive with type value = 'v) + type 'v interactive = ('v value, 'v interactive_module) or_unit + + (* Initial location builder *) + let init (value : 'v value) : 'v structured = Unit value + + (* During the complete abstraction build, we need to verify that there is at + least one location in the computed abstraction. + TODO: better error handling. *) + let assert_not_unit = function + | Unit _ -> Self.fatal "The built location cannot be unit." + | Location interactive -> interactive + + + (* Making a structured value interactive simply consists of adding the + needed operations using the Structure.Open functor.*) + let make_interactive : type v. v structured -> v interactive = function + | Unit value -> Unit value + | Location (module Structured) -> + Location (module struct + include Structured.Location + include Structure.Open (Abstract.Location) (struct + include Structured.Location + type t = location + end) + end) - let configure () = - let add_main_mode mode = - let main, _ = Globals.entry_point () in - (main, Domain_mode.Mode.all) :: mode - in - let add config (name, make) = - let enabled = Parameters.Domains.mem name in - try - let mode = Parameters.DomainsFunction.find name in - let mode = if enabled then add_main_mode mode else mode in - add (make (), Some mode) config - with Not_found -> - if enabled then add (make (), None) config else config - in - let aux config (Flag domain as flag) = - add config (domain.name, (fun () -> flag)) + (* Retrieves the value contained in a structured location. *) + let get_value : type v. v structured -> v value = function + | Unit value -> value + | Location (module S) -> (module S.Value) + + + (* Adding a registered location into a structured one is done in three steps: + 1. Lifting the location abstraction we want to add to match the value + abstraction contained in the structured abstraction. + 2. Combine the given location abstraction with the one contained in the + structured abstraction. It comes down to decide if a reduced product is + needed. + 3. Rebuild a structured abstraction with the new location abstraction. *) + let add : type v l. l registered -> v structured -> v structured = + fun (module Registered) structured -> + let deps = Value.outline Registered.dependencies in + let module To = (val get_value structured) in + let module From = struct include Registered let structure = deps end in + let module Converter = Value.Converter (From) (To) in + let lifted : (module Abstract.Location.Internal with type value = v) = + match Value.dec_eq deps To.structure with + | Some Eq -> (module Registered) + | None -> (module Location_lift.Make (Registered) (Converter)) in - let config = List.fold_left aux empty !abstractions in - List.fold_left add config !dynamic_abstractions - - (* --- Register default abstractions -------------------------------------- *) - - let create_domain ?experimental priority name descr values domain = - let abstraction = { values = Single values; domain = Domain domain } in - register ~name ~descr ~priority ?experimental abstraction - - (* Register standard domains over cvalues. *) - let make ?experimental rank name descr = - create_domain ?experimental rank name descr (module Main_values.CVal) - - let cvalue = - make 9 "cvalue" - "Main analysis domain, enabled by default. Should not be disabled." - (module Cvalue_domain.State) - - let symbolic_locations = - make 7 "symbolic-locations" - "Infers values of symbolic locations represented by imprecise lvalues, \ - such as t[i] or *p when the possible values of [i] or [p] are imprecise." - (module Symbolic_locs.D) - - let equality = - let descr = "Infers equalities between syntactic C expressions. \ - Makes the analysis less dependent on temporary variables and \ - intermediate computations." - and abstraction = - { values = Struct Abstract.Value.Unit; - domain = Functor (module Equality_domain.Make); } + let combined : (module Abstract.Location.Internal with type value = v) = + match make_interactive structured with + | Unit _ -> lifted + | Location (module Loc) when Loc.mem Registered.key -> (module Loc) + | Location (module Loc) -> + (module Locations_product.Make (To) (val lifted) (Loc)) in - register ~name:"equality" ~descr ~priority:8 abstraction - - let gauges = - make 6 "gauges" - "Infers linear inequalities between the variables modified within a loop \ - and a special loop counter." - (module Gauges_domain.D) - - let octagon = - make 6 "octagon" - "Infers relations between scalar variables of the form b ≤ ±X ± Y ≤ e, \ - where X, Y are program variables and b, e are constants." - (module Octagons) - - let bitwise = - create_domain 3 "bitwise" - "Infers bitwise information to interpret more precisely bitwise operators." - (module Offsm_value.Offsm) (module Offsm_domain.D) - - let sign = - create_domain 4 "sign" - "Infers the sign of program variables." - (module Sign_value) (module Sign_domain) - - let inout = make 5 "inout" ~experimental:true - "Infers the inputs and outputs of each function." - (module Inout_domain.D) - - let traces = - make 2 "traces" ~experimental:true - "Builds an over-approximation of all the traces that lead \ - to a statement." - (module Traces_domain.D) - - let printer = - make 2 "printer" - "Debug domain, only useful for developers. Prints the transfer functions \ - used during the analysis." - (module Printer_domain) - - (* --- Default and legacy configurations ---------------------------------- *) - - let default = configure () - let legacy = singleton (cvalue, None) + Location (module struct + type value = v + module Value = To + module Location = (val combined) + end) + + + (* When building the complete abstraction, we need to trick domains into + thinking that their locations dependencies are there, even if the + structured location type is not the good one. This is done through a + lift that requires conversion operations to interact with the subpart + of the structured location that matters for the domains. This functor is + responsible of building such conversion operations. *) + module type From = sig type location val structure : location structure end + module Converter (From : From) (To : Interactive) = struct + type extended = To.location + let structure = From.structure + + let void_location () = + Self.fatal "Cannot register a location module from a Void structure." + + let rec set : type l. l structure -> l -> extended -> extended = function + | Leaf (key, _) -> To.set key + | Node (s1, s2) -> fun (l1, l2) ext -> set s2 l2 ext |> set s1 l1 + | Option (s, default) -> fun l -> set s (Option.value ~default l) + | Unit -> fun () loc -> loc + | Void -> void_location () + + let rec get : type l. l structure -> extended -> l = function + | Leaf (key, _) -> Option.get (To.get key) + | Node (s1, s2) -> fun l -> get s1 l, get s2 l + | Option (s, _) -> fun l -> Some (get s l) + | Unit -> fun _ -> () + | Void -> void_location () + + let replace = set structure + let extend l = replace l To.top + let restrict = get structure + end end -let register = Config.register -let dynamic_register = Config.dynamic_register -(* --- Building value abstractions ------------------------------------------ *) -module Leaf_Value (V: Abstract_value.Leaf) = struct - include V - let structure = Abstract.Value.Leaf (V.key, (module V)) -end +(* --- Domains abstraction -------------------------------------------------- *) -module Leaf_Location (Loc: Abstract_location.Leaf) = struct - include Loc - let structure = Abstract.Location.Leaf (Loc.key, (module Loc)) -end +module Domain = struct + type 's structure = 's Abstract.Domain.structure + type 's key = 's Abstract.Domain.key + module type S = Abstract_domain.S -module Leaf_Domain (D: Abstract_domain.Leaf) = struct - include D - let structure = Abstract.Domain.Leaf (D.key, (module D)) -end -module type Acc = sig - module Val : Abstract.Value.External - module Loc : Abstract.Location.Internal with type value = Val.t - and type location = precise_loc - module Dom : Abstract.Domain.Internal with type value = Val.t - and type location = Loc.location -end + (* Registration of leaf domains, i.e simple domain with a fixed value. *) + module Leaf = struct + type ('v, 'l, 's) domain = + (module S with type value = 'v and type location = 'l and type state = 's) + + (* We need the values and locations dependendies to register a domain. *) + type ('v, 'l, 's) register = + { key : 's key + ; domain : ('v, 'l, 's) domain + ; values : 'v Value.dependencies + ; locations : 'l Location.dependencies + } -module Internal_Value = struct - open Abstract.Value + (* As for the locations, a registered domain is a simple module that + contains all the information given during the register along with + the domain structure (which is a leaf). *) + module type Registered = sig + include Abstract_domain.S + val key : state key + val structure : state structure + val values : value Value.dependencies + val locations : location Location.dependencies + end - type value_key_module = V : 'v key * 'v data -> value_key_module + (* Registration procedure. *) + type 's registered = (module Registered with type state = 's) + let register : type v l s. (v, l, s) register -> s registered = + fun { key ; domain ; values ; locations } -> + (module struct + module Domain = (val domain) + include Domain + let key = key + let structure = Abstract.Domain.Leaf (key, (module Domain)) + let values = values + let locations = locations + end) + end - let open_value_abstraction (module Value : Internal) = - (module struct - include Value - include Structure.Open (Abstract.Value) (Value) - end : Abstract.Value.External) - let add_value_leaf value (V (key, v)) = - let module Value = (val open_value_abstraction value) in - if Value.mem key then value else + (* Registration of a functor domain, i.e a domain that can be built on top of + any value. *) + module Functor = struct + + (* A functor domain is declared as an existantiel. This module type has to + be read as: there exists a type <location> such as, given any Value, we + can build a domain using this Value and relying on a Location of type + <location>. *) + module type Domain = sig + type location + module Make (V : Value.Interactive) : sig + include Abstract_domain.S + with type value = V.t + and type location = location + val key : state key + end + end + type 'l domain = (module Domain with type location = 'l) + + (* As a functor domain can be instanciated for any value, we only need + locations dependencies to register it. *) + type 'l register = + { domain : 'l domain + ; locations : 'l Location.dependencies + } + + (* Once registered, a functor domain builds a domain along with its + structure and its key. *) + module type RegisteredLeaf = sig + include Abstract.Domain.Internal + val key : state key + end + + (* A registered functor domain contains its locations dependencies and a + functor that builds a domain along its structure and its key. *) + module type Registered = sig + type location + val locations : location Location.dependencies + module Make (V : Value.Interactive) : + RegisteredLeaf with type value = V.t and type location = location + end + + (* Registration procedure. *) + type registered = (module Registered) + let register : type l. l register -> registered = + fun { domain ; locations } -> + let module Domain = (val domain) in (module struct - include Value_product.Make (Value) (val v) - let structure = Node (Value.structure, Leaf (key, v)) + type location = Domain.location + let locations = locations + module Make (V : Value.Interactive) = struct + module Result = Domain.Make (V) + include Result + let structure = Abstract.Domain.Leaf (key, (module Result)) + end end) + end - let void_value () = - Self.fatal - "Cannot register a value module from a Void structure." - - let add_value_structure value internal = - let rec aux: type v. (module Internal) -> v structure -> (module Internal) = - fun value -> function - | Option (s, _) -> aux value s - | Leaf (key, v) -> add_value_leaf value (V (key, v)) - | Node (s1, s2) -> aux (aux value s1) s2 - | Unit -> value - | Void -> void_value () - in - aux value internal - let build_values config initial_value = - let build (Flag flag, _) acc = - match flag.abstraction.values with - | Struct structure -> add_value_structure acc structure - | Single (module V) -> add_value_leaf acc (V (V.key, (module V))) - in - let value = Config.fold build config initial_value in - open_value_abstraction value - - - module Convert - (Value: Abstract.Value.External) - (Struct: sig type v val s : v value end) - = struct - - let structure = match Struct.s with - | Single (module V) -> Abstract.Value.Leaf (V.key, (module V)) - | Struct s -> s - - type extended_value = Value.t - - let replace_val = - let rec set: type v. v structure -> v -> Value.t -> Value.t = - function - | Leaf (key, _) -> Value.set key - | Node (s1, s2) -> - let set1 = set s1 and set2 = set s2 in - fun (v1, v2) value -> set1 v1 (set2 v2 value) - | Option (s, default) -> fun v -> set s (Option.value ~default:default v) - | Unit -> fun () value -> value - | Void -> void_value () - in - set structure - - let extend_val v = replace_val v Value.top - - let restrict_val = - let rec get: type v. v structure -> Value.t -> v = function - | Leaf (key, _) -> Option.get (Value.get key) - | Node (s1, s2) -> - let get1 = get s1 and get2 = get s2 in - fun v -> get1 v, get2 v - | Option (s, _) -> fun v -> Some (get s v) - | Unit -> fun _ -> () - | Void -> void_value () - in - get structure + (* To simplify the domain registration procedure, we provide common types. + However, the code above is still useful to prove some properties, mainly + that we do not temper with the dependencies. *) + type register = + | Domain : ('v, 'l, 's) Leaf.register -> register + | Functor : 'l Functor.register -> register + + type registered = + | RegisteredDomain : 's Leaf.registered -> registered + | RegisteredFunctor : Functor.registered -> registered + + (* Registered domain are saved in mutable lists along with there information, + i.e. there name, there experimental and there priority flags. *) + type 't with_info = + { name : string + ; experimental : bool + ; priority : int + ; abstraction : 't + } - type extended_location = Main_locations.PLoc.location + (* The configuration of an analysis contains a set of domains along with their + mode. *) + type 't with_mode = 't * Domain_mode.t option - let restrict_loc = fun x -> x - let extend_loc = fun x -> x + + (* Mutable lists containing statically and dynamically registered domains. *) + let static_domains = ref [] + let dynamic_domains = ref [] + + (* Helper function used to register the parameters of a domain. *) + let register_domain_option ~name ~experimental ~descr = + let descr = if experimental then "Experimental. " ^ descr else descr in + Parameters.register_domain ~name ~descr + + (* Helper function used to dispatch the registration process to the good + registration process depending of the domain kind. *) + let register_abstraction = function + | Domain register -> RegisteredDomain (Leaf.register register) + | Functor register -> RegisteredFunctor (Functor.register register) + + (* Registration of a static domain. *) + let register ~name ~descr ?(experimental=false) ?(priority=0) register = + register_domain_option ~name ~descr ~experimental ; + let abstraction = register_abstraction register in + let registered = { name ; experimental ; priority ; abstraction } in + static_domains := registered :: !static_domains ; + abstraction + + (* Registration of a dynamic domain. *) + let dynamic_register ~name ~descr ?(experimental=false) ?(priority=0) make = + register_domain_option ~name ~descr ~experimental ; + let make () = make () |> register_abstraction in + let make () = { name ; experimental ; priority ; abstraction = make () } in + dynamic_domains := (name, make) :: !dynamic_domains + + + (* Building the domain abstraction consists of structuring the requested + registered domains. To do so, we need to keep track of the values and + locations abstraction on which the structured domain will rely. Those + abstractions are supposed to be the complete aggregations of all the + values (resp locations) that are needed by the requested domains. *) + module type Structured = sig + type value + type location + module Value : Value.Interactive with type t = value + module Location : Location.Interactive + with type value = value and type location = location + module Domain : Abstract.Domain.Internal + with type value = value and type location = location end -end -(* --- Building domain abstractions ----------------------------------------- *) - -module type internal_loc = - Abstract.Location.Internal with type location = precise_loc -module type internal_domain = - Abstract.Domain.Internal with type location = precise_loc - -let eq_value: - type a b. a Abstract.Value.structure -> b value -> (a,b) Structure.eq option - = fun structure -> function - | Struct s -> Abstract.Value.eq_structure structure s - | Single (module V) -> - match structure with - | Abstract.Value.Leaf (key, _) -> Abstract.Value.eq_type key V.key - | _ -> None - -let add_domain (type v) dname mode (abstraction: v abstraction) (module Acc: Acc) = - let domain : (module internal_domain with type value = Acc.Val.t) = - match abstraction.domain with - | Functor make -> - let module Make = (val make: domain_functor) in - (module Leaf_Domain (Make (Acc.Val))) - | Domain domain -> - match eq_value Acc.Val.structure abstraction.values with - | Some Structure.Eq -> - let module Domain = (val domain) in - (module Leaf_Domain (Domain)) - | None -> - let module Domain = (val domain : leaf_domain with type value = v) in - let module Struct = struct - type v = Domain.value - let s = abstraction.values - end in - let module Convert = Internal_Value.Convert (Acc.Val) (Struct) in - (module Domain_lift.Make (Domain) (Convert)) - in - (* Set the name of the domain. *) - let module Domain = struct - include (val domain) - module Store = struct - include Store - let register_global_state storage state = - let no_results = Parameters.NoResultsDomains.mem dname in - register_global_state (storage && not no_results) state - end - end in - (* Restricts the domain according to [mode]. *) - let domain : (module internal_domain with type value = Acc.Val.t) = - match mode with - | None -> (module Domain) - | Some kf_modes -> - let module Scope = struct let functions = kf_modes end in - let module Domain = - Domain_builder.Restrict - (Acc.Val) - (Domain) - (Scope) + (* As for the value and location abstractions, a [Unit] structured domain is + used for the initial state. *) + type ('v, 'l, 's) or_unit = Unit of 'v * 'l | State of 's + type 'v value = (module Value.Interactive with type t = 'v) + type ('v, 'l) location = + (module Location.Interactive with type value = 'v and type location = 'l) + type ('v, 'l) structured_module = + (module Structured with type value = 'v and type location = 'l) + type ('v, 'l) structured = + ('v value, ('v, 'l) location, ('v, 'l) structured_module) or_unit + + (* Recovers the value and location abstractions of a structured domain. *) + let get : type v l. (v, l) structured -> v value * (v, l) location = function + | Unit (value, location) -> (value, location) + | State (module S) -> ((module S.Value), (module S.Location)) + + (* During the complete abstraction build, we need to verify that there is at + least one domain in the computed abstraction. + TODO: better error handling. *) + let assert_not_unit = function + | Unit _ -> Self.fatal "The built domain cannot be unit." + | State structured -> structured + + + (* Internal type used for intermediate results of the add procedure. *) + type ('v, 'l) structured_domain = + (module Abstract.Domain.Internal with type value = 'v and type location = 'l) + + (* Utility function used to create an identity converter. *) + module type Typ = sig type t end + let conversion_id (type t) (module T: Typ with type t = t) = + (module struct + type extended = T.t + type internal = T.t + let extend x = x + let restrict x = x + end: Domain_lift.Conversion with type extended = t and type internal = t) + + (* Adding a registered domain into a structured one consists of performing a + lifting of the registered one if needed before performing the product, + configuring the name and restricting the domain depending of the mode. *) + type add_input = registered with_info with_mode + let add : type v l. add_input -> (v, l) structured -> (v, l) structured = + fun (registered, mode) structured -> + let wkey = Self.wkey_experimental in + let { experimental = exp ; name } = registered in + if exp then Self.warning ~wkey "The %s domain is experimental." name ; + let value, location = get structured in + let module Val = (val value) in + let module Loc = (val location) in + let lifted : (v, l) structured_domain = + match registered.abstraction with + | RegisteredFunctor (module Functor) -> + let locs = Location.outline Functor.locations in + let eq_loc = Location.dec_eq locs Loc.structure in + let module D = Functor.Make (Val) in + begin match eq_loc with + | Some Eq -> (module D) + | None -> + let module Val = (val conversion_id (module Val)) in + let module From = struct include D let structure = locs end in + let module Loc = Location.Converter (From) (Loc) in + (module Domain_lift.Make (D) (Val) (Loc)) + end + | RegisteredDomain (module D) -> + let loc_deps = Location.outline D.locations in + let val_deps = Value.outline D.values in + let eq_loc = Location.dec_eq loc_deps Loc.structure in + let eq_val = Value.dec_eq val_deps Val.structure in + begin match eq_val, eq_loc with + | Some Eq, Some Eq -> (module D) + | Some Eq, None -> + let module Val = (val conversion_id (module Val)) in + let module From = struct include D let structure = loc_deps end in + let module Loc = Location.Converter (From) (Loc) in + (module Domain_lift.Make (D) (Val) (Loc)) + | None, Some Eq -> + let module From = struct include D let structure = val_deps end in + let module Val = Value.Converter (From) (Val) in + let module LocTyp = struct type t = Loc.location end in + let module Loc = (val conversion_id (module LocTyp)) in + (module Domain_lift.Make (D) (Val) (Loc)) + | _, _ -> + let module From = struct include D let structure = val_deps end in + let module Val = Value.Converter (From) (Val) in + let module From = struct include D let structure = loc_deps end in + let module Loc = Location.Converter (From) (Loc) in + (module Domain_lift.Make (D) (Val) (Loc)) + end + in + (* Set the name of the domain. *) + let module Named = struct + include (val lifted) + module Store = struct + include Store + let register_global_state storage state = + let no_results = Parameters.NoResultsDomains.mem registered.name in + register_global_state (storage && not no_results) state + end + end in + (* Restricts the domain according to [mode]. *) + let restricted : (v, l) structured_domain = + match mode with + | None -> (module Named) + | Some kf_modes -> + let module Scope = struct let functions = kf_modes end in + (module Domain_builder.Restrict (Val) (Named) (Scope)) + in + let combined : (v, l) structured_domain = + match structured with + | Unit _ -> restricted + | State (module Structured) -> + (* The new [domain] becomes the left leaf of the domain product, and + will be processed before the domains from [Acc.Dom] during the + analysis. *) + let module Dom = Structured.Domain in + (module Domain_product.Make (Val) (val restricted) (Dom)) + in + State (module struct + type value = v + type location = l + module Value = Val + module Location = Loc + module Domain = (val combined) + end) + + + (* Build a complete abstraction based on a list of registered domains and a + value initial configuration. *) + let build domains : (module Structured) = + let values = + let add_value = Value.{ folder = add } in + let add_values values (registered, _) = + match registered.abstraction with + | RegisteredDomain (module Domain) -> + Value.fold add_value Domain.values values |> + Location.fold_values add_value Domain.locations + | RegisteredFunctor (module F) -> + Location.fold_values add_value F.locations values + in + List.fold_left add_values Value.init domains |> + Value.make_interactive + in + let module V = (val Value.assert_not_unit values) in + let locations = + let init : V.t Location.structured = Location.init (module V) in + let add_loc = Location.{ folder = add } in + let add_locations locs (registered, _) = + match registered.abstraction with + | RegisteredDomain (module D) -> Location.fold add_loc D.locations locs + | RegisteredFunctor (module D) -> Location.fold add_loc D.locations locs in - (module Domain) - in - let domain : (module internal_domain with type value = Acc.Val.t) = - match Abstract.Domain.(eq_structure Acc.Dom.structure Unit) with - | Some _ -> domain - | None -> - (* The new [domain] becomes the left leaf of the domain product, and will - be processed before the domains from [Acc.Dom] during the analysis. *) - (module Domain_product.Make (Acc.Val) ((val domain)) (Acc.Dom)) - in - (module struct - module Val = Acc.Val - module Loc = Acc.Loc - module Dom = (val domain) - end : Acc) - -let warn_experimental flag = - if flag.experimental then - Self.(warning ~wkey:wkey_experimental - "The %s domain is experimental." flag.name) - -let build_domain config abstract = - let build (Flag flag, mode) acc = - warn_experimental flag; - add_domain flag.name mode flag.abstraction acc - in - (* Domains in the [config] are sorted by increasing priority: domains with - higher priority are added last: they will be at the top of the domains - tree, and thus will be processed first during the analysis. *) - Config.fold build config abstract + List.fold_left add_locations init domains |> + Location.make_interactive + in + let module L = (val Location.assert_not_unit locations) in + let structured : (V.t, L.location) structured = + Unit ((module V), (module L)) + in + let structured = List.fold_left (fun s d -> add d s) structured domains in + let module Structured : Structured = (val assert_not_unit structured) in + (module Structured) +end -(* --- Value reduced product ----------------------------------------------- *) -module type Value = sig - include Abstract.Value.External - val reduce : t -> t -end +(* --- Configuration -------------------------------------------------------- *) -module type S = sig - module Val : Value - module Loc : Abstract.Location.External with type value = Val.t - module Dom : Abstract.Domain.External with type value = Val.t - and type location = Loc.location -end +module Config = struct + module Mode = Datatype.Option (Domain_mode) + + include Set.Make (struct + open Domain + type t = registered with_info with_mode + let compare (d1, m1) (d2, m2) = + let c = Datatype.Int.compare d1.priority d2.priority in + if c = 0 then + let c = Datatype.String.compare d1.name d2.name in + if c = 0 then Mode.compare m1 m2 else c + else c + end) + + let singleton name abstraction = + let experimental = false and priority = 9 in + let abstraction = Domain.{ name ; priority ; experimental ; abstraction } in + singleton (abstraction, None) -module type Eva = sig - include S - module Eval: Evaluation.S with type state = Dom.t - and type value = Val.t - and type loc = Loc.location - and type origin = Dom.origin + let configure () = + let find = Parameters.DomainsFunction.find in + let find name = try Some (find name) with Not_found -> None in + let main () = Globals.entry_point () |> fst in + let add_main_mode modes = (main (), Domain_mode.Mode.all) :: modes in + let dynamic (name, make) config = + let enabled = Parameters.Domains.mem name in + let enable modes = if enabled then add_main_mode modes else modes in + match find name with + | None -> if enabled then add (make (), None) config else config + | Some modes -> add (make (), Some (enable modes)) config + in + let static d = dynamic (d.Domain.name, fun () -> d) in + let fold f xs acc = List.fold_left (fun acc x -> f x acc) acc xs in + fold static !Domain.static_domains empty |> + fold dynamic !Domain.dynamic_domains end -type ('a, 'b) value_reduced_product = - 'a Abstract.Value.key * 'b Abstract.Value.key * ('a -> 'b -> 'a * 'b) -type v_reduced_product = R: ('a, 'b) value_reduced_product -> v_reduced_product +(* --- Value reduced product ----------------------------------------------- *) -let value_reduced_product = ref [] +module Reducer = struct + type 'a key = 'a Value.key + type ('a, 'b) reducer = 'a -> 'b -> 'a * 'b + type action = Action : 'a key * 'b key * ('a, 'b) reducer -> action -let register_value_reduction reduced_product = - value_reduced_product := (R reduced_product) :: !value_reduced_product + let actions = ref [] -(* When the value abstraction contains both a cvalue and an interval - component (coming currently from an Apron domain), reduce them from each - other. If the Cvalue is not a scalar do nothing, because we do not - currently use Apron for pointer offsets. *) -let reduce_apron_itv cvalue ival = - match ival with - | None -> begin - try cvalue, Some (Cvalue.V.project_ival cvalue) - with Cvalue.V.Not_based_on_null -> cvalue, ival - end - | Some ival -> - try - let ival' = Cvalue.V.project_ival cvalue in - if Ival.is_int ival' - then - let reduced_ival = Ival.narrow ival ival' in - let cvalue = Cvalue.V.inject_ival reduced_ival in - cvalue, Some reduced_ival - else cvalue, Some ival - with Cvalue.V.Not_based_on_null -> cvalue, Some ival - -let () = - register_value_reduction - (Main_values.CVal.key, Main_values.Interval.key, reduce_apron_itv) - -module Reduce (Value : Abstract.Value.External) = struct - include Value - - let make_reduction acc (R (key1, key2, f)) = - match Value.get key1, Value.get key2 with - | Some get1, Some get2 -> - let set1 = Value.set key1 - and set2 = Value.set key2 in - let reduce v = let v1, v2 = f (get1 v) (get2 v) in set1 v1 (set2 v2 v) in - reduce :: acc - | _, _ -> acc - - let reduce = - let list = List.fold_left make_reduction [] !value_reduced_product in - fun v -> List.fold_left (fun v reduce -> reduce v) v list -end + let register left right reducer = + actions := (Action (left, right, reducer)) :: !actions -(* --- Final hook ----------------------------------------------------------- *) + module type Value = sig + include Abstract.Value.External + val reduce : t -> t + end -let final_hooks = ref [] + module Make (Value : Abstract.Value.External) = struct + include Value + + let make_reduction acc (Action (key1, key2, f)) = + match Value.get key1, Value.get key2 with + | Some get1, Some get2 -> + let set1 = Value.set key1 and set2 = Value.set key2 in + let reduce v = f (get1 v) (get2 v) in + let reducer v = let v1, v2 = reduce v in set1 v1 (set2 v2 v) in + reducer :: acc + | _, _ -> acc + + let reduce = + let list = List.fold_left make_reduction [] !actions in + fun v -> List.fold_left (fun v reduce -> reduce v) v list + end +end -let register_hook f = - final_hooks := f :: !final_hooks -let apply_final_hooks abstractions = - List.fold_left (fun acc f -> f acc) abstractions !final_hooks -(* --- Building abstractions ------------------------------------------------ *) +(* --- Finalizing abstractions build ---------------------------------------- *) -module Open (Acc: Acc) : S = struct - module Val = Reduce (Acc.Val) - module Loc = struct - include Acc.Loc - include Structure.Open (Abstract.Location) - (struct include Acc.Loc type t = location end) - end - module Dom = struct - include Acc.Dom - include Structure.Open (Abstract.Domain) (Acc.Dom) +module type S = sig + module Val : Reducer.Value + module Loc : Abstract.Location.External with type value = Val.t + module Dom : Abstract.Domain.External + with type value = Val.t and type location = Loc.location +end - let get_cvalue = match get Cvalue_domain.State.key with - | None -> None - | Some get -> Some (fun s -> fst (get s)) +module type S_with_evaluation = sig + include S + module Eval : Evaluation_sig.S + with type state = Dom.t + and type value = Val.t + and type loc = Loc.location + and type origin = Dom.origin +end - let get_cvalue_or_top = match get Cvalue_domain.State.key with - | None -> fun _ -> Cvalue.Model.top - | Some get -> fun s -> fst (get s) +module Hooks = struct + let hooks = ref [] + type hook = (module S) -> (module S) + let register (f : hook) = hooks := f :: !hooks + let apply abst = List.fold_left (fun acc f -> f acc) abst !hooks +end - let get_cvalue_or_bottom = function - | `Bottom -> Cvalue.Model.bottom - | `Value state -> get_cvalue_or_top state +module Open (Structured : Domain.Structured) : S = struct + module Val = Reducer.Make (Structured.Value) + module Loc = Structured.Location + module Dom = struct + include Structured.Domain + include Structure.Open (Abstract.Domain) (Structured.Domain) end end -module CVal = Leaf_Value (Main_values.CVal) - -let unit_acc (module Value: Abstract.Value.External) = - let loc : (module internal_loc with type value = Value.t) = - match Abstract.Value.eq_structure Value.structure CVal.structure with - | Some Structure.Eq -> (module Leaf_Location (Main_locations.PLoc)) - | _ -> - let module Struct = struct - type v = Cvalue.V.t - let s = Single (module Main_values.CVal) - end in - let module Conv = Internal_Value.Convert (Value) (Struct) in - (module Location_lift.Make (Main_locations.PLoc) (Conv)) - in - (module struct - module Val = Value - module Loc = (val loc) - module Dom = Unit_domain.Make (Val) (Loc) - end : Acc) - -let build_abstractions config = - let initial_value : (module Abstract.Value.Internal) = - if Config.mem Config.bitwise config - then (module Offsm_value.CvalueOffsm) - else (module CVal) - in - let value = Internal_Value.build_values config initial_value in - let acc = unit_acc value in - build_domain config acc - -let configure = Config.configure - let make config = - let abstractions = build_abstractions config in - let abstractions = (module Open (val abstractions): S) in - apply_final_hooks abstractions - -module Default = (val make Config.default) -module Legacy = (val make Config.legacy) + let abstractions = Config.elements config |> Domain.build in + let abstractions = (module Open (val abstractions) : S) in + Hooks.apply abstractions diff --git a/src/plugins/eva/engine/abstractions.mli b/src/plugins/eva/engine/abstractions.mli index be3c32e05155142ce18afb9f54512883ffc30a6e..21a16ba10c30c3748980237edaa2ac7dc21fba63 100644 --- a/src/plugins/eva/engine/abstractions.mli +++ b/src/plugins/eva/engine/abstractions.mli @@ -20,6 +20,8 @@ (* *) (**************************************************************************) + + (** Registration and building of the analysis abstractions. *) (** {2 Registration of abstractions.} *) @@ -30,152 +32,198 @@ - state abstractions, or abstract domains, detailed in {!Abstract_domain}. *) -(** Module types of value abstractions: either a single leaf module, or - a compound of several modules described by a structure. In this last case, - the structure must not contain the Void constructor. *) -type 'v value = - | Single of (module Abstract_value.Leaf with type t = 'v) - | Struct of 'v Abstract.Value.structure - -(** For the moment, all domains must use [precise_loc] as their location - abstraction, and no new location abstraction can be registered for an - analysis. - If you need to build a new location abstraction, please contact us. *) -type precise_loc = Precise_locs.precise_location - -(** Module type of a leaf domain over precise_loc abstraction. *) -module type leaf_domain = Abstract_domain.Leaf with type location = precise_loc - -(** Module type of a functor building a leaf domain from a value abstraction. - The resulting domain must use the input value as value abstraction. *) -module type domain_functor = functor - (Value: Abstract.Value.External) -> (leaf_domain with type value = Value.t) - -(** Type of domain to be registered: either a leaf module with ['v] as value - abstraction, or a functor building a domain from any value abstraction. *) -type 'v domain = - | Domain: (module leaf_domain with type value = 'v) -> 'v domain - | Functor: (module domain_functor) -> _ domain - -(** Abstraction to be registered. *) -type 'v abstraction = - { values: 'v value; (** The value abstraction. *) - domain: 'v domain ; (** The domain over the value abstraction. *) - } - -(** Information about a registered abstraction. *) -type 't with_info = - { name: string; (** Name of the domain. Must be unique. *) - experimental: bool; (** Is the domain experimental? *) - priority: int; (** Domains with higher priority are processed first. *) - abstraction: 't; (** The abstract value and the domain. *) - } - -(** Flag for an abstract domain. A domain can be programmatically enabled via - its flag. See module {!Config} for more details. *) -type flag = Flag: 'v abstraction with_info -> flag - -(** Registers an abstract domain. Returns a flag for the given domain. - - [name] must be unique. The domain is used if the -eva-domains option - has been set to [name]. - - [descr] is a description printed in the help message of -eva-domains. - - [experimental] is false by default. If set to true, a warning is emitted - when the domain is enabled. - - [priority] can be any integer; domains with higher priority are always - processed first. The domains currently provided by Eva have priority - ranging between 1 and 19, so a priority of 0 (respectively 20) ensures - that a new domain is processed after (respectively before) the classic - Eva domains. The default priority is 0. *) -val register: - name:string -> descr:string -> ?experimental:bool -> ?priority:int -> - 'v abstraction -> flag - -(** Register a dynamic abstraction: the abstraction is built by applying - the last argument when starting an analysis, if the -eva-domains option - has been set to [name]. See function {!register} for more details. *) -val dynamic_register: - name:string -> descr:string -> ?experimental:bool -> ?priority:int -> - (unit -> 'v abstraction) -> unit - -(** Reduced product between two value abstractions, identified by their keys. *) -type ('a, 'b) value_reduced_product = - 'a Abstract.Value.key * 'b Abstract.Value.key * ('a -> 'b -> 'a * 'b) - -(** Register a reduction function for a value reduced product. *) -val register_value_reduction: ('a, 'b) value_reduced_product -> unit - - -(** {2 Types used in the engine.} *) - -(** The external signature of value abstractions, plus the reduction function - of the reduced product. *) -module type Value = sig - include Abstract.Value.External - val reduce : t -> t +(** Value abstractions registration. *) +module Value : sig + type 'v key = 'v Abstract.Value.key + type 'v value = (module Abstract_value.S with type t = 'v) + + (** Registering a value abstraction requires a single module and a key. The + returned [`v registered] is a witness of the registration process and is + used by the location and domain abstractions registration to declare their + value dependencies. *) + type 'v register = { key : 'v key ; value : 'v value } + type 'v registered + val register : 'v register -> 'v registered + + (** Other abstractions need to declare their value dependencies, i.e. the + value abstractions they rely on to perform their computations. Those + dependencies are declared as a heterogenous list containing at least one + element. *) + type 'v dependencies = + | Last : 'v registered -> 'v dependencies + | (::) : 'a registered * 'b dependencies -> ('a * 'b) dependencies end -(** The three abstractions used in an Eva analysis. *) -module type S = sig - module Val : Value - module Loc : Abstract.Location.External with type value = Val.t - module Dom : Abstract.Domain.External with type value = Val.t - and type location = Loc.location + +(** Location abstractions registration. *) +module Location : sig + type 'l key = 'l Abstract.Location.key + type ('v, 'l) location = + (module Abstract_location.S with type value = 'v and type location = 'l) + + (** Registering a location abstraction requires a single module, a key and the + registered value abstractions needed to perform the computations. The + returned [`l registered] is a witness of the registration process and is + used by the domain abstractions registration to declare their location + dependencies. *) + type ('v, 'l) register = + { key : 'l key + ; location : ('v, 'l) location + ; dependencies : 'v Value.dependencies + } + type 'l registered + val register : ('v, 'l) register -> 'l registered + + (** Domain abstractions need to declare their location dependencies. As for + the value dependencies, they are declared as a heterogenous list. *) + type 'l dependencies = + | Last : 'l registered -> 'l dependencies + | (::) : 'a registered * 'b dependencies -> ('a * 'b) dependencies end -(** The three abstractions plus an evaluation engine for these abstractions. *) -module type Eva = sig - include S - module Eval: Evaluation.S with type state = Dom.t - and type value = Val.t - and type loc = Loc.location - and type origin = Dom.origin + +(** Domain abstractions registration. *) +module Domain : sig + type 's key = 's Abstract.Domain.key + + (** Leaf domain abstraction, i.e. a simple domain with fixed values and + locations dependencies. The registration of such domains requires a + key, a single module and values and locations dependencies. *) + module Leaf : sig + type ('v, 'l, 's) domain = + (module Abstract_domain.S + with type value = 'v and type location = 'l and type state = 's) + + type ('v, 'l, 's) register = + { key : 's key + ; domain : ('v, 'l, 's) domain + ; values : 'v Value.dependencies + ; locations : 'l Location.dependencies + } + end + + (** Functor domain abstraction, i.e. a domain that can be built over any value + abstractions, but with fixed locations dependencies. The registraction of + such domains requires only the location dependencies, as the abstraction + produced by the functor cannot rely on any particular value and the key + can depend on the value abstractions used. *) + module Functor : sig + module type Domain = sig + type location + module Make (V : Abstract.Value.External) : sig + include Abstract_domain.S + with type value = V.t and type location = location + val key : state key + end + end + type 'l domain = (module Domain with type location = 'l) + + type 'l register = + { domain : 'l domain + ; locations : 'l Location.dependencies + } + end + + (** Registration of both kind of domain abstractions is done using the same + functions. The two kind of registration information are thus regrouped + under the same type. The returned [registered] is a witness of the + registration process and can be used to programmatically enable the + domain. *) + type register = + | Domain : ('v, 'l, 's) Leaf.register -> register + | Functor : 'l Functor.register -> register + type registered + + (** Registers an abstract domain. Returns a flag for the given domain. + - [name] must be unique. The domain is used if the -eva-domains option + has been set to [name]. + - [descr] is a description printed in the help message of -eva-domains. + - [experimental] is false by default. If set to true, a warning is emitted + when the domain is enabled. + - [priority] can be any integer; domains with higher priority are always + processed first. The domains currently provided by Eva have priority + ranging between 1 and 19, so a priority of 0 (respectively 20) ensures + that a new domain is processed after (respectively before) the classic + Eva domains. The default priority is 0. *) + val register : + name:string -> descr:string -> ?experimental:bool -> ?priority:int -> + register -> registered + + (** Register a dynamic abstraction: the abstraction is built by applying + the last argument when starting an analysis, if the -eva-domains option + has been set to [name]. See function {!register} for more details. *) + val dynamic_register : + name:string -> descr:string -> ?experimental:bool -> ?priority:int -> + (unit -> register) -> unit +end + + +(** Value reduced product registration. Registering a value reduced product + requires the keys of each value abstractions involved along with a reducer, + i.e. a function that perform the reduction. *) +module Reducer : sig + type ('a, 'b) reducer = 'a -> 'b -> 'a * 'b + val register : 'a Value.key -> 'b Value.key -> ('a, 'b) reducer -> unit + + (** The value abstractions signature used in the engine. It is composed of the + external signature of value abstractions, plus the reduction function of + the reduced product. *) + module type Value = sig + include Abstract.Value.External + val reduce : t -> t + end end -(** Register a hook modifying the three abstractions after their building by - the engine, before the start of each analysis. *) -val register_hook: ((module S) -> (module S)) -> unit + (** {2 Configuration of an analysis.} *) -(** Configuration defining the abstractions to be used in an analysis. - A configuration is a set of flags, i.e. a set of abstract domains. Each flag - comes with an optional mode. None is the default mode: the given domain is - enabled for the whole analysis. See {!Domain_mode} for more details. *) +(** Configuration defining the abstractions to be used in an analysis. A + configuration can either be built from a given domain along with its + name or can be built based on the command line parameters. The first + approach relies on the [singleton] function and is mainly used to + build a default abstraction during the engine initialization. The + second one relies on the [configure] function. *) module Config : sig - include Set.S with type elt = flag * Domain_mode.t option - - (** Returns true if the given flag is in the configuration. *) - val mem: flag -> t -> bool - - (** Flags for the standard domains currently provided in Eva. *) - - val cvalue: flag - val equality: flag - val symbolic_locations: flag - val gauges: flag - val octagon: flag - val bitwise: flag - val inout: flag - val sign: flag - val traces: flag - val printer: flag - - val default: t - (** The default configuration of Eva. *) - - val legacy: t - (** The configuration corresponding to the old "Value" analysis, - with only the cvalue domain enabled. *) + type t + val equal : t -> t -> bool + val singleton : string -> Domain.registered -> t + val configure : unit -> t end -(** Creates the configuration according to the analysis parameters. *) -val configure: unit -> Config.t + + +(** {2 Types and functions used in the engine.} *) + +(** The three abstractions used in an Eva analysis. *) +module type S = sig + module Val : Reducer.Value + module Loc : Abstract.Location.External with type value = Val.t + module Dom : Abstract.Domain.External + with type value = Val.t and type location = Loc.location +end + +(* The three abstractions plus an evaluation engine for these abstractions. *) +module type S_with_evaluation = sig + include S + module Eval : Evaluation_sig.S + with type state = Dom.t + and type value = Val.t + and type loc = Loc.location + and type origin = Dom.origin +end (** Builds the abstractions according to a configuration. *) -val make: Config.t -> (module S) +val make : Config.t -> (module S) -(** Two abstractions are instantiated at compile time for the default and legacy - configurations (which may be the same). *) -module Legacy : S -module Default : S + +(** {2 Analysis low level modifications.} *) + +(** Registration of a hook, i.e. a function that modifies directly the three + abstractions after their building by the engine and before the start of + each analysis. *) +module Hooks : sig + type hook = (module S) -> (module S) + val register : hook -> unit +end diff --git a/src/plugins/eva/engine/analysis.ml b/src/plugins/eva/engine/analysis.ml index a390c9398cfd06d88ea2a8f813466d6d78b4b143..c6fe1ee2bc3103382d5a897c99b89f4434bd8c89 100644 --- a/src/plugins/eva/engine/analysis.ml +++ b/src/plugins/eva/engine/analysis.ml @@ -72,7 +72,7 @@ module type Results = sig end module type S = sig - include Abstractions.Eva + include Abstractions.S_with_evaluation include Results with type state := Dom.state and type value := Val.t and type location := Loc.location @@ -81,7 +81,7 @@ end module type Analyzer = sig include S val compute_from_entry_point : kernel_function -> lib_entry:bool -> unit - val compute_from_init_state: kernel_function -> Dom.t -> unit + (* val compute_from_init_state: kernel_function -> Dom.t -> unit *) val initial_state: lib_entry:bool -> Dom.t or_bottom end @@ -90,7 +90,7 @@ module Make (Abstract: Abstractions.S) = struct module Abstract = struct include Abstract - module Eval = Evaluation.Make (Abstract.Val) (Abstract.Loc) (Abstract.Dom) + module Eval = Evaluation.Make (Val) (Loc) (Dom) end include Abstract @@ -149,20 +149,15 @@ module Make (Abstract: Abstractions.S) = struct end -module Legacy = Make (Abstractions.Legacy) -module Default = - (val - (if Abstractions.Config.(equal default legacy) - then (module Legacy) - else (module Make (Abstractions.Default))) - : Analyzer) +let default = Abstractions.Config.singleton "cvalue" Cvalue_domain.registered +module Default : Analyzer = Make (val Abstractions.make default) + (* Reference to the current configuration (built by Abstractions.configure from the parameters of Eva regarding the abstractions used in the analysis) and the current Analyzer module. *) -let ref_analyzer = - ref (Abstractions.Config.default, (module Default : Analyzer)) +let ref_analyzer = ref (default, (module Default : Analyzer)) (* Returns the current Analyzer module. *) let current_analyzer () = (module (val (snd !ref_analyzer)): S) @@ -182,15 +177,15 @@ let set_current_analyzer config (analyzer: (module Analyzer)) = let cvalue_initial_state () = let module A = (val snd !ref_analyzer) in + let module G = (Cvalue_domain.Getters (A.Dom)) in let _, lib_entry = Globals.entry_point () in - A.Dom.get_cvalue_or_bottom (A.initial_state ~lib_entry) + G.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. *) let make_analyzer config = let analyzer = - if Abstractions.Config.(equal config legacy) then (module Legacy: Analyzer) - else if Abstractions.Config.(equal config default) then (module Default) + if Abstractions.Config.(equal config default) then (module Default : Analyzer) else let module Abstract = (val Abstractions.make config) in let module Analyzer = Make (Abstract) in @@ -200,7 +195,7 @@ let make_analyzer config = (* Builds the analyzer according to the parameters of Eva. *) let reset_analyzer () = - let config = Abstractions.configure () in + let config = Abstractions.Config.configure () in (* If the configuration has not changed, do not reset the Analyzer but uses the reference instead. *) if not (Abstractions.Config.equal config (fst !ref_analyzer)) diff --git a/src/plugins/eva/engine/analysis.mli b/src/plugins/eva/engine/analysis.mli index 3cb30baf2214b8afd75d27e8ef5fed42b9d88798..dc017343ef221e88d4369946e52ffc6463218251 100644 --- a/src/plugins/eva/engine/analysis.mli +++ b/src/plugins/eva/engine/analysis.mli @@ -60,7 +60,7 @@ end module type S = sig - include Abstractions.Eva + include Abstractions.S_with_evaluation include Results with type state := Dom.state and type value := Val.t and type location := Loc.location diff --git a/src/plugins/eva/engine/compute_functions.ml b/src/plugins/eva/engine/compute_functions.ml index 5e913a57d2c6738e82bcb20c861fa11a0cbc8177..11a90011b3aec48f415ac7bc8bfc4059ac70d5f4 100644 --- a/src/plugins/eva/engine/compute_functions.ml +++ b/src/plugins/eva/engine/compute_functions.ml @@ -141,7 +141,7 @@ let register_signal_handler () = let restore_sigint = register_handler Sys.sigint interrupt in fun () -> restore_sigusr1 (); restore_sigint () -module Make (Abstract: Abstractions.Eva) = struct +module Make (Abstract: Abstractions.S_with_evaluation) = struct module PowersetDomain = Powerset.Make (Abstract.Dom) @@ -154,6 +154,8 @@ module Make (Abstract: Abstractions.Eva) = struct Iterator.Computer (Abstract) (PowersetDomain) (Transfer) (Init) (Logic) (Spec) + include Cvalue_domain.Getters (Abstract.Dom) + let initial_state = Init.initial_state let get_cval = @@ -185,7 +187,7 @@ module Make (Abstract: Abstractions.Eva) = struct in call_result | Some (states, i) -> - let cvalue = Abstract.Dom.get_cvalue_or_top init_state in + let cvalue = get_cvalue_or_top init_state in Cvalue_callbacks.apply_call_hooks call.callstack call.kf `Memexec cvalue; (* Evaluate the preconditions of kf, to update the statuses at this call. *) @@ -240,7 +242,7 @@ module Make (Abstract: Abstractions.Eva) = struct "@[computing for function %a.@\nCalled from %a.@]" Value_types.Callstack.pretty_short call.callstack Cil_datatype.Location.pretty (Cil_datatype.Kinstr.loc kinstr); - let cvalue_state = Abstract.Dom.get_cvalue_or_top state in + let cvalue_state = get_cvalue_or_top state in let compute, kind = match target with | `Def (fundec, save_results) -> @@ -290,7 +292,7 @@ module Make (Abstract: Abstractions.Eva) = struct in Locations.Location_Bytes.do_track_garbled_mix true; let final_state = join_states states in - let cvalue_state = Abstract.Dom.get_cvalue_or_top state in + let cvalue_state = get_cvalue_or_top state in match final_state with | `Bottom -> let kind = `Spec spec in @@ -299,7 +301,7 @@ module Make (Abstract: Abstractions.Eva) = struct Transfer.{states; cacheable; builtin=true} | `Value final_state -> let cvalue_call = get_cvalue_call call in - let post = Abstract.Dom.get_cvalue_or_top final_state in + let post = get_cvalue_or_top final_state in let cvalue_states = Builtins.apply_builtin builtin cvalue_call ~pre:cvalue_state ~post in @@ -344,7 +346,7 @@ module Make (Abstract: Abstractions.Eva) = struct let store_initial_state kf init_state = Abstract.Dom.Store.register_initial_state (Eva_utils.call_stack ()) init_state; - let cvalue_state = Abstract.Dom.get_cvalue_or_top init_state in + let cvalue_state = 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/eva/engine/compute_functions.mli b/src/plugins/eva/engine/compute_functions.mli index 2eea8219cdbbf5e9a8a7c664e4640155da7d6919..28f78965dbb2d54008aabe51674bbf084a49f3d8 100644 --- a/src/plugins/eva/engine/compute_functions.mli +++ b/src/plugins/eva/engine/compute_functions.mli @@ -25,7 +25,7 @@ open Cil_types open Eval -module Make (Abstract: Abstractions.Eva) +module Make (Abstract: Abstractions.S_with_evaluation) : sig (** Compute a call to the main function. *) diff --git a/src/plugins/eva/engine/evaluation.ml b/src/plugins/eva/engine/evaluation.ml index fa55a38798a0e020a7e632e9aa39de4bbd4712c3..7a58005d92be8f394687268b4a7f44a76084b3e7 100644 --- a/src/plugins/eva/engine/evaluation.ml +++ b/src/plugins/eva/engine/evaluation.ml @@ -183,36 +183,6 @@ let exp_alarm_signed_converted_downcast = let signed_exp = Cil.new_exp ~loc:exp.eloc (CastE (signed_typ, exp)) in signed_exp) -module type S = sig - type state - type value - type origin - type loc - module Valuation : Valuation with type value = value - and type origin = origin - and type loc = loc - val to_domain_valuation: - Valuation.t -> (value, loc, origin) Abstract_domain.valuation - val evaluate : - ?valuation:Valuation.t -> ?reduction:bool -> ?subdivnb:int -> - state -> exp -> (Valuation.t * value) evaluated - val copy_lvalue : - ?valuation:Valuation.t -> ?subdivnb:int -> - state -> lval -> (Valuation.t * value flagged_value) evaluated - val lvaluate : - ?valuation:Valuation.t -> ?subdivnb:int -> for_writing:bool -> - state -> lval -> (Valuation.t * loc * typ) evaluated - val reduce: - ?valuation:Valuation.t -> state -> exp -> bool -> Valuation.t evaluated - val assume: - ?valuation:Valuation.t -> state -> exp -> value -> Valuation.t or_bottom - val eval_function_exp: - ?subdivnb:int -> exp -> ?args:exp list -> state -> - (Kernel_function.t * Valuation.t) list evaluated - val interpret_truth: - alarm:(unit -> Alarms.t) -> 'a -> 'a Abstract_value.truth -> 'a evaluated -end - let return t = `Value t, Alarmset.none (* Intersects [alarms] with the only possible alarms from the dereference of @@ -895,7 +865,7 @@ module Make | AddrOf v | StartOf v -> lval_to_loc context ~for_writing:false ~reduction:false v >>= fun (loc, _, _) -> - let value = Loc.to_value loc in + (Loc.to_value loc, Alarmset.none) >>= fun value -> let v = assume_pointer expr value in compute_reduction v false @@ -1394,7 +1364,7 @@ module Make | Mem expr, offset -> match offset with | NoOffset -> - let loc_value = Loc.to_value location in + Loc.to_value location >>- fun loc_value -> backward_eval fuel state expr (Some loc_value) >>-: fun _ -> () | _ -> let reduce_valid_index = true in @@ -1691,7 +1661,6 @@ module Make | _ -> assert false end - (* Local Variables: compile-command: "make -C ../../../.." diff --git a/src/plugins/eva/engine/evaluation.mli b/src/plugins/eva/engine/evaluation.mli index 6197f1d07fb23dae081e2cb38e226c17dbb3148e..ff7487ecf1bfdbf17609864a1cd00161b3329066 100644 --- a/src/plugins/eva/engine/evaluation.mli +++ b/src/plugins/eva/engine/evaluation.mli @@ -20,111 +20,6 @@ (* *) (**************************************************************************) -open Cil_types -open Eval - -(** Generic evaluation and reduction of expressions and left values. *) - -module type S = sig - - (** State of abstract domain. *) - type state - - (** Numeric values to which the expressions are evaluated. *) - type value - - (** Origin of values. *) - type origin - - (** Location of an lvalue. *) - type loc - - (** Results of an evaluation: the results of all intermediate calculation (the - value of each expression and the location of each lvalue) are cached here. - See {!Eval} for more details. *) - module Valuation : Valuation with type value = value - and type origin = origin - and type loc = loc - - (** Evaluation functions store the results of an evaluation into [Valuation.t] - maps. Abstract domains read these results from [Abstract_domain.valuation] - records. This function converts the former into the latter. *) - val to_domain_valuation: - Valuation.t -> (value, loc, origin) Abstract_domain.valuation - - (** [evaluate ~valuation state expr] evaluates the expression [expr] in the - state [state], and returns the pair [result, alarms], where: - - [alarms] are the set of alarms ensuring the soundness of the evaluation; - - [result] is either `Bottom if the evaluation leads to an error, - or `Value (valuation, value), where [value] is the numeric value computed - for the expression [expr], and [valuation] contains all the intermediate - results of the evaluation. - - Optional arguments are: - - [valuation] is a cache of already computed expressions; empty by default. - - [reduction] allows the deactivation of the backward reduction performed - after the forward evaluation; true by default. - - [subdivnb] is the maximum number of subdivisions performed on non-linear - sub-expressions of [expr]. If a lvalue occurs several times in [expr], - its value can be split up to [subdivnb] times to gain more precision. - Set to the value of the option -eva-subdivide-non-linear by default. *) - val evaluate : - ?valuation:Valuation.t -> ?reduction:bool -> ?subdivnb:int -> - state -> exp -> (Valuation.t * value) evaluated - - (** Computes the value of a lvalue, with possible indeterminateness: the - returned value may be uninitialized, or contain escaping addresses. - Also returns the alarms resulting of the evaluation of the lvalue location, - and a valuation containing all the intermediate results of the evaluation. - The [valuation] argument is a cache of already computed expressions. - It is empty by default. - [subdivnb] is the maximum number of subdivisions performed on non-linear - expressions. *) - val copy_lvalue : - ?valuation:Valuation.t -> ?subdivnb:int -> - state -> lval -> (Valuation.t * value flagged_value) evaluated - - (** [lvaluate ~valuation ~for_writing state lval] evaluates the left value - [lval] in the state [state]. Same general behavior as [evaluate] above - but evaluates the lvalue into a location and its type. - The boolean [for_writing] indicates whether the lvalue is evaluated to be - read or written. It is useful for the emission of the alarms, and for the - reduction of the location. - [subdivnb] is the maximum number of subdivisions performed on non-linear - expressions (including the possible pointer and offset of the lvalue). *) - val lvaluate : - ?valuation:Valuation.t -> ?subdivnb:int -> for_writing:bool -> - state -> lval -> (Valuation.t * loc * typ) evaluated - - (** [reduce ~valuation state expr positive] evaluates the expression [expr] - in the state [state], and then reduces the [valuation] such that - the expression [expr] evaluates to a zero or a non-zero value, according - to [positive]. By default, the empty valuation is used. *) - val reduce: - ?valuation:Valuation.t -> - state -> exp -> bool -> Valuation.t evaluated - - (** [assume ~valuation state expr value] assumes in the [valuation] that - the expression [expr] has the value [value] in the state [state], - and backward propagates this information to the subterm of [expr]. - If [expr] has not been already evaluated in the [valuation], its forward - evaluation takes place first, but the alarms are dismissed. By default, - the empty valuation is used. - The function returns the updated valuation, or bottom if it discovers - a contradiction. *) - val assume: - ?valuation:Valuation.t -> - state -> exp -> value -> Valuation.t or_bottom - - val eval_function_exp: - ?subdivnb:int -> exp -> ?args:exp list -> state -> - (Kernel_function.t * Valuation.t) list evaluated - (** Evaluation of the function argument of a [Call] constructor *) - - val interpret_truth: - alarm:(unit -> Alarms.t) -> 'a -> 'a Abstract_value.truth -> 'a evaluated -end - module type Value = sig include Abstract.Value.External @@ -146,11 +41,10 @@ module Make (Loc : Abstract_location.S with type value = Value.t) (Domain : Queries with type value = Value.t and type location = Loc.location) - : S with type state = Domain.state - and type value = Value.t - and type origin = Domain.origin - and type loc = Loc.location - + : Evaluation_sig.S with type state = Domain.state + and type value = Value.t + and type origin = Domain.origin + and type loc = Loc.location (* Local Variables: diff --git a/src/plugins/eva/engine/evaluation_sig.ml b/src/plugins/eva/engine/evaluation_sig.ml new file mode 100644 index 0000000000000000000000000000000000000000..75673541afa231aa24ae910d904560b2e69cafd4 --- /dev/null +++ b/src/plugins/eva/engine/evaluation_sig.ml @@ -0,0 +1,126 @@ +(**************************************************************************) +(* *) +(* This file is part of Frama-C. *) +(* *) +(* Copyright (C) 2007-2023 *) +(* 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). *) +(* *) +(**************************************************************************) + +open Cil_types +open Eval + +(** Generic evaluation and reduction of expressions and left values. *) + +module type S = sig + + (** State of abstract domain. *) + type state + + (** Numeric values to which the expressions are evaluated. *) + type value + + (** Origin of values. *) + type origin + + (** Location of an lvalue. *) + type loc + + (** Results of an evaluation: the results of all intermediate calculation (the + value of each expression and the location of each lvalue) are cached here. + See {!Eval} for more details. *) + module Valuation : Valuation with type value = value + and type origin = origin + and type loc = loc + + (** Evaluation functions store the results of an evaluation into [Valuation.t] + maps. Abstract domains read these results from [Abstract_domain.valuation] + records. This function converts the former into the latter. *) + val to_domain_valuation: + Valuation.t -> (value, loc, origin) Abstract_domain.valuation + + (** [evaluate ~valuation state expr] evaluates the expression [expr] in the + state [state], and returns the pair [result, alarms], where: + - [alarms] are the set of alarms ensuring the soundness of the evaluation; + - [result] is either `Bottom if the evaluation leads to an error, + or `Value (valuation, value), where [value] is the numeric value computed + for the expression [expr], and [valuation] contains all the intermediate + results of the evaluation. + + Optional arguments are: + - [valuation] is a cache of already computed expressions; empty by default. + - [reduction] allows the deactivation of the backward reduction performed + after the forward evaluation; true by default. + - [subdivnb] is the maximum number of subdivisions performed on non-linear + sub-expressions of [expr]. If a lvalue occurs several times in [expr], + its value can be split up to [subdivnb] times to gain more precision. + Set to the value of the option -eva-subdivide-non-linear by default. *) + val evaluate : + ?valuation:Valuation.t -> ?reduction:bool -> ?subdivnb:int -> + state -> exp -> (Valuation.t * value) evaluated + + (** Computes the value of a lvalue, with possible indeterminateness: the + returned value may be uninitialized, or contain escaping addresses. + Also returns the alarms resulting of the evaluation of the lvalue location, + and a valuation containing all the intermediate results of the evaluation. + The [valuation] argument is a cache of already computed expressions. + It is empty by default. + [subdivnb] is the maximum number of subdivisions performed on non-linear + expressions. *) + val copy_lvalue : + ?valuation:Valuation.t -> ?subdivnb:int -> + state -> lval -> (Valuation.t * value flagged_value) evaluated + + (** [lvaluate ~valuation ~for_writing state lval] evaluates the left value + [lval] in the state [state]. Same general behavior as [evaluate] above + but evaluates the lvalue into a location and its type. + The boolean [for_writing] indicates whether the lvalue is evaluated to be + read or written. It is useful for the emission of the alarms, and for the + reduction of the location. + [subdivnb] is the maximum number of subdivisions performed on non-linear + expressions (including the possible pointer and offset of the lvalue). *) + val lvaluate : + ?valuation:Valuation.t -> ?subdivnb:int -> for_writing:bool -> + state -> lval -> (Valuation.t * loc * typ) evaluated + + (** [reduce ~valuation state expr positive] evaluates the expression [expr] + in the state [state], and then reduces the [valuation] such that + the expression [expr] evaluates to a zero or a non-zero value, according + to [positive]. By default, the empty valuation is used. *) + val reduce: + ?valuation:Valuation.t -> + state -> exp -> bool -> Valuation.t evaluated + + (** [assume ~valuation state expr value] assumes in the [valuation] that + the expression [expr] has the value [value] in the state [state], + and backward propagates this information to the subterm of [expr]. + If [expr] has not been already evaluated in the [valuation], its forward + evaluation takes place first, but the alarms are dismissed. By default, + the empty valuation is used. + The function returns the updated valuation, or bottom if it discovers + a contradiction. *) + val assume: + ?valuation:Valuation.t -> + state -> exp -> value -> Valuation.t or_bottom + + val eval_function_exp: + ?subdivnb:int -> exp -> ?args:exp list -> state -> + (Kernel_function.t * Valuation.t) list evaluated + (** Evaluation of the function argument of a [Call] constructor *) + + val interpret_truth: + alarm:(unit -> Alarms.t) -> 'a -> 'a Abstract_value.truth -> 'a evaluated +end diff --git a/src/plugins/eva/engine/initialization.ml b/src/plugins/eva/engine/initialization.ml index 7cccb65020e25018c7191d12951acd08ca77ffc8..ba5c5ea1c1539b2d72543981a0f827cab92722e1 100644 --- a/src/plugins/eva/engine/initialization.ml +++ b/src/plugins/eva/engine/initialization.ml @@ -82,8 +82,8 @@ let counter = ref 0 module Make (Domain: Abstract.Domain.External) - (Eva: Evaluation.S with type state = Domain.state - and type loc = Domain.location) + (Eva: Evaluation_sig.S with type state = Domain.state + and type loc = Domain.location) (Transfer: Transfer_stmt.S with type state = Domain.t) = struct @@ -95,6 +95,7 @@ module Make fst (Eva.lvaluate ~for_writing:false Domain.top lval) >>> fun (_valuation, loc, _typ) -> loc + include Cvalue_domain.Getters (Domain) (* ------------------------- Apply initializer ---------------------------- *) @@ -272,7 +273,7 @@ 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 with + match get_cvalue with | None -> Self.abort "Function Db.Value.fun_set_args cannot be \ used without the Cvalue domain" | Some get_cvalue -> @@ -366,7 +367,7 @@ module Make else global_state ~lib_entry let print_initial_cvalue_state state = - let cvalue_state = Domain.get_cvalue_or_bottom state in + let cvalue_state = 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/eva/engine/initialization.mli b/src/plugins/eva/engine/initialization.mli index 9012946abe082baf204f551152efae99468bb9da..872206abe61796ac482a107efdd3d344668ca87b 100644 --- a/src/plugins/eva/engine/initialization.mli +++ b/src/plugins/eva/engine/initialization.mli @@ -45,8 +45,8 @@ end module Make (Domain: Abstract.Domain.External) - (Eva: Evaluation.S with type state = Domain.state - and type loc = Domain.location) + (Eva: Evaluation_sig.S with type state = Domain.state + and type loc = Domain.location) (Transfer: Transfer_stmt.S with type state = Domain.t) : S with type state := Domain.t diff --git a/src/plugins/eva/engine/iterator.ml b/src/plugins/eva/engine/iterator.ml index 2a0c7af021ac216b6eae8cea7f8315e3604f3ece..7979fd721696059972af7faf81b24327e98c4c0f 100644 --- a/src/plugins/eva/engine/iterator.ml +++ b/src/plugins/eva/engine/iterator.ml @@ -45,7 +45,7 @@ let blocks_share_locals b1 b2 = | _, _ -> false module Make_Dataflow - (Abstract : Abstractions.Eva) + (Abstract : Abstractions.S_with_evaluation) (States : Powerset.S with type state = Abstract.Dom.t) (Transfer : Transfer_stmt.S with type state = Abstract.Dom.t) (Init: Initialization.S with type state := Abstract.Dom.t) @@ -63,6 +63,7 @@ module Make_Dataflow = struct module Domain = Abstract.Dom + include Cvalue_domain.Getters (Domain) (* --- Analysis parameters --- *) @@ -441,7 +442,7 @@ module Make_Dataflow edge_info.fireable <- true; flow - let gather_cvalues states = match Domain.get_cvalue with + let gather_cvalues states = match get_cvalue with | Some get -> List.map get states | None -> [] @@ -681,7 +682,7 @@ module Make_Dataflow then VertexTable.memo merged_states v get_smashed_store else `Bottom and lift_to_cvalues table = - StmtTable.map (fun _ s -> Domain.get_cvalue_or_top s) (Lazy.force table) + StmtTable.map (fun _ s -> 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) @@ -696,7 +697,7 @@ module Make_Dataflow (StmtTable.map (fun _stmt (v,_) -> let store = get_vertex_store v in let states = Partitioning.expanded store in - List.map (fun (_k,x) -> Domain.get_cvalue_or_top x) states) + List.map (fun (_k,x) -> get_cvalue_or_top x) states) automaton.stmt_table) in let merged_pre_cvalues = lazy (lift_to_cvalues merged_pre_states) @@ -744,7 +745,7 @@ end module Computer - (Abstract : Abstractions.Eva) + (Abstract : Abstractions.S_with_evaluation) (States : Powerset.S with type state = Abstract.Dom.t) (Transfer : Transfer_stmt.S with type state = Abstract.Dom.t) (Init: Initialization.S with type state := Abstract.Dom.t) diff --git a/src/plugins/eva/engine/iterator.mli b/src/plugins/eva/engine/iterator.mli index 1c70fd4b0e17a75e7454e721b731edd3b606a829..0d1f9944063057b116f95a53af45b2766169cd64 100644 --- a/src/plugins/eva/engine/iterator.mli +++ b/src/plugins/eva/engine/iterator.mli @@ -27,7 +27,7 @@ val signal_abort: unit -> unit module Computer (* Abstractions with the evaluator. *) - (Abstract: Abstractions.Eva) + (Abstract: Abstractions.S_with_evaluation) (* Set of states of abstract domain. *) (States : Powerset.S with type state = Abstract.Dom.t) (* Transfer functions for statement on the abstract domain. *) diff --git a/src/plugins/eva/engine/subdivided_evaluation.ml b/src/plugins/eva/engine/subdivided_evaluation.ml index 5e0ea2abac87900a9cc23b77e0e8cf90000a6bcd..60949a0ef0a7ade04b1e9f198791293a5ef1029c 100644 --- a/src/plugins/eva/engine/subdivided_evaluation.ml +++ b/src/plugins/eva/engine/subdivided_evaluation.ml @@ -856,8 +856,8 @@ module Make | Lval (host, off as lval) -> if Cil.typeHasQualifier "volatile" (Cil.typeOfLval lval) then `Value acc else - let loc = find_loc valuation lval in - if Cvalue.V.cardinal_zero_or_one (get_cval (Loc.to_value loc)) + Loc.to_value (find_loc valuation lval) >>- fun value -> + if Cvalue.V.cardinal_zero_or_one (get_cval value) then (* no variable in the host or in the offset can be influential. Check the contents of the location, on which we might want to enumerate*) diff --git a/src/plugins/eva/engine/transfer_specification.ml b/src/plugins/eva/engine/transfer_specification.ml index 0821c12b0356dc91faf181ece6239c65c245c2d8..ebde0b8888044622355b7b71954dc3915149f5b7 100644 --- a/src/plugins/eva/engine/transfer_specification.ml +++ b/src/plugins/eva/engine/transfer_specification.ml @@ -179,6 +179,7 @@ module Make module Domain = Abstract.Dom module Location = Abstract.Loc + include Cvalue_domain.Getters (Domain) (* Most transfer functions about logic return a set of states instead of a single state, and States.empty instead of bottom. We thus use this monad @@ -239,7 +240,7 @@ module Make let set_location loc = set_ploc (Main_locations.PLoc.make loc) let make_env state = - Eval_terms.env_assigns ~pre:(Domain.get_cvalue_or_top state) + Eval_terms.env_assigns ~pre:(get_cvalue_or_top state) (* Evaluates the location affected by an assigns, allocates, frees or \from clause. Returns None if the clause cannot be interpreted. *) @@ -322,7 +323,7 @@ module Make end in let check_one_state state = - let cvalue_state = Domain.get_cvalue_or_top state in + let cvalue_state = 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/eva/engine/transfer_stmt.ml b/src/plugins/eva/engine/transfer_stmt.ml index 71042b2b2e41baab8d8f7d0c49e811f686231910..3d17e1177e9756eeeafe100a238db064756dc675 100644 --- a/src/plugins/eva/engine/transfer_stmt.ml +++ b/src/plugins/eva/engine/transfer_stmt.ml @@ -114,12 +114,13 @@ let substitution_visitor table = object | Some vi -> Cil.ChangeTo vi end -module Make (Abstract: Abstractions.Eva) = struct +module Make (Abstract: Abstractions.S_with_evaluation) = struct module Value = Abstract.Val module Location = Abstract.Loc module Domain = Abstract.Dom module Eval = Abstract.Eval + include Cvalue_domain.Getters (Domain) type state = Domain.t type value = Value.t @@ -633,7 +634,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, Location.get Main_locations.PLoc.key with + match 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 -> @@ -729,7 +730,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) :: Eva_utils.call_stack () in - let cvalue_state = Domain.get_cvalue_or_top state in + let cvalue_state = get_cvalue_or_top state in Db.Value.Call_Value_Callbacks.apply (cvalue_state, stack_with_call); let kind = `Builtin None in Cvalue_callbacks.apply_call_hooks stack_with_call kf kind cvalue_state diff --git a/src/plugins/eva/engine/transfer_stmt.mli b/src/plugins/eva/engine/transfer_stmt.mli index a54bfe15b9cc5b7b6849a5cabfaeba4c9674ede0..7da3e2be794787a059471923e14b7d7bdacf7260 100644 --- a/src/plugins/eva/engine/transfer_stmt.mli +++ b/src/plugins/eva/engine/transfer_stmt.mli @@ -58,7 +58,7 @@ module type S = sig (stmt -> (loc, value) call -> recursion option -> state -> call_result) ref end -module Make (Abstract: Abstractions.Eva) +module Make (Abstract: Abstractions.S_with_evaluation) : S with type state = Abstract.Dom.t and type value = Abstract.Val.t and type loc = Abstract.Loc.location diff --git a/src/plugins/eva/gui/gui_eval.ml b/src/plugins/eva/gui/gui_eval.ml index 8dd3dbd6d1682d60541f7e119ebd035cfda7470d..25d985d9e4d77693633421bc8e3567105bd11f26 100644 --- a/src/plugins/eva/gui/gui_eval.ml +++ b/src/plugins/eva/gui/gui_eval.ml @@ -139,6 +139,7 @@ end module Make (X: Analysis.S) = struct module Analysis = X + include Cvalue_domain.Getters (X.Dom) let get_precise_loc = match X.Loc.get Main_locations.PLoc.key with @@ -202,7 +203,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 = X.Dom.get_cvalue_or_top state in + let state = get_cvalue_or_top state in let aux loc (acc_res, acc_ok) = let res, ok = match lv with (* catch simplest pattern *) @@ -265,7 +266,7 @@ module Make (X: Analysis.S) = struct } let null_to_offsetmap state (_:unit) = - let state = X.Dom.get_cvalue_or_top state in + let state = 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 @@ -327,17 +328,17 @@ module Make (X: Analysis.S) = struct let env_here kf here callstack = let pre = pre_kf kf callstack in - let here = X.Dom.get_cvalue_or_top here in + let here = 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 = X.Dom.get_cvalue_or_top here in + let here = 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 = X.Dom.get_cvalue_or_top post in + let post = get_cvalue_or_top post in let result = if Function_calls.use_spec_instead_of_definition kf then None diff --git a/src/plugins/eva/gui/gui_types.ml b/src/plugins/eva/gui/gui_types.ml index e52339afa5e8918bdbd00c42cb15234e8671064b..94f675343aa8ecb17515ad6137ae9596a5a033d3 100644 --- a/src/plugins/eva/gui/gui_types.ml +++ b/src/plugins/eva/gui/gui_types.ml @@ -135,7 +135,7 @@ module type S = sig val equal_gui_after : value gui_after -> value gui_after -> bool end -module Make (V: Abstractions.Value) = struct +module Make (V: Abstract.Value.External) = struct let pretty_gui_res fmt = function | GR_Empty -> () diff --git a/src/plugins/eva/gui/gui_types.mli b/src/plugins/eva/gui/gui_types.mli index fdc05aa6401b9103a1521c7ff6974bf902535ace..e4d5018d9c10a678c9a4c719d470f9280ff5d719 100644 --- a/src/plugins/eva/gui/gui_types.mli +++ b/src/plugins/eva/gui/gui_types.mli @@ -99,7 +99,7 @@ module type S = sig end (** The types below depend on the abstract values currently available. *) -module Make (V : Abstractions.Value) : sig +module Make (V : Abstract.Value.External) : sig include S with type value := V.t val get_cvalue : (V.t -> Main_values.CVal.t) option diff --git a/src/plugins/eva/locations/locations_product.ml b/src/plugins/eva/locations/locations_product.ml new file mode 100644 index 0000000000000000000000000000000000000000..e6fad814205d20093d00dd1ced03c3f8c47ece82 --- /dev/null +++ b/src/plugins/eva/locations/locations_product.ml @@ -0,0 +1,125 @@ +(**************************************************************************) +(* *) +(* This file is part of Frama-C. *) +(* *) +(* Copyright (C) 2007-2023 *) +(* 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). *) +(* *) +(**************************************************************************) + +open Eval + + + +module Make + (Value: Abstract_value.S) + (Left: Abstract.Location.Internal with type value = Value.t) + (Right: Abstract.Location.Internal with type value = Value.t) += struct + type value = Value.t + let structure = Abstract.Location.Node (Left.structure, Right.structure) + + type offset = Left.offset * Right.offset + type location = Left.location * Right.location + + let top = Left.top, Right.top + + let equal_loc (l, r) (l', r') = + Left.equal_loc l l' && Right.equal_loc r r' + let pretty_loc = + Pretty_utils.pp_pair Left.pretty_loc Right.pretty_loc + + let equal_offset (l, r) (l', r') = + Left.equal_offset l l' && Right.equal_offset r r' + let pretty_offset = + Pretty_utils.pp_pair Left.pretty_offset Right.pretty_offset + + (* TODO: don't know what to do, used max by default *) + let size (l, r) = + match Left.size l, Right.size r with + | Int_Base.Top, size + | size, Int_Base.Top -> size + | Int_Base.Value lsize as size, Int_Base.Value rsize -> + if Integer.equal lsize rsize then size else + Self.fatal + "Location product: inconsistent size of the same location \ + represented by %a (size %a) and %a (size %a)." + Left.pretty_loc l Integer.pretty lsize + Right.pretty_loc r Integer.pretty rsize + + let replace_base subst (l, r) = + Left.replace_base subst l, Right.replace_base subst r + + let assume_no_overlap ~partial (l1, r1) (l2, r2) = + let l_truth = Left.assume_no_overlap ~partial l1 l2 in + let r_truth = Right.assume_no_overlap ~partial r1 r2 in + Value_product.narrow_truth_pair ((l1, l2), l_truth) ((r1, r2), r_truth) + + let assume_valid_location ~for_writing ~bitfield (l, r) = + let l_truth = Left.assume_valid_location ~for_writing ~bitfield l in + let r_truth = Right.assume_valid_location ~for_writing ~bitfield r in + Value_product.narrow_truth (l, l_truth) (r, r_truth) + + let no_offset = Left.no_offset, Right.no_offset + + let forward_field typ varinfo (l, r) = + Left.forward_field typ varinfo l, Right.forward_field typ varinfo r + + let forward_variable typ varinfo (l, r) = + let* l = Left.forward_variable typ varinfo l in + let* r = Right.forward_variable typ varinfo r in + `Value (l, r) + + let eval_varinfo varinfo = + Left.eval_varinfo varinfo, Right.eval_varinfo varinfo + + let backward_variable varinfo (l, r) = + let* l = Left.backward_variable varinfo l in + let* r = Right.backward_variable varinfo r in + `Value (l, r) + + let backward_field typ varinfo (lo, ro) = + let* lo = Left.backward_field typ varinfo lo in + let* ro = Right.backward_field typ varinfo ro in + `Value (lo, ro) + + (** Both value abstractions produce a sound value abstraction for the same + location, so we can narrow their results. *) + let to_value (l, r) = + let* vleft = Left.to_value l + and* vright = Right.to_value r in + Value.narrow vleft vright + + let forward_index typ v (l, r) = + Left.forward_index typ v l, Right.forward_index typ v r + + let forward_pointer typ v (lo, ro) = + let* l = Left.forward_pointer typ v lo in + let* r = Right.forward_pointer typ v ro in + `Value (l, r) + + let backward_pointer v (lo, ro) (l, r) = + let* (lv, lo) = Left.backward_pointer v lo l in + let* (rv, ro) = Right.backward_pointer v ro r in + let* v = Value.narrow lv rv in + `Value (v, (lo, ro)) + + let backward_index typ ~index ~remaining:(lrem, rrem) (lo, ro) = + let* (lv, lo) = Left.backward_index typ ~index ~remaining:lrem lo in + let* (rv, ro) = Right.backward_index typ ~index ~remaining:rrem ro in + let* v = Value.narrow lv rv in + `Value (v, (lo, ro)) +end diff --git a/src/plugins/eva/locations/locations_product.mli b/src/plugins/eva/locations/locations_product.mli new file mode 100644 index 0000000000000000000000000000000000000000..150c21c89eba0fca1c895b2b9fc553e7e4a02392 --- /dev/null +++ b/src/plugins/eva/locations/locations_product.mli @@ -0,0 +1,30 @@ +(**************************************************************************) +(* *) +(* This file is part of Frama-C. *) +(* *) +(* Copyright (C) 2007-2023 *) +(* 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). *) +(* *) +(**************************************************************************) + +module Make + (Value: Abstract_value.S) + (Left: Abstract.Location.Internal with type value = Value.t) + (Right: Abstract.Location.Internal with type value = Value.t) + : Abstract.Location.Internal + with type value = Value.t + and type location = Left.location * Right.location + and type offset = Left.offset * Right.offset diff --git a/src/plugins/eva/partitioning/auto_loop_unroll.ml b/src/plugins/eva/partitioning/auto_loop_unroll.ml index 3df1e5d6b2efb702367092fb9854ffc779d43ad5..58b52807752271e0a9c58bb69d832deeb6f78844 100644 --- a/src/plugins/eva/partitioning/auto_loop_unroll.ml +++ b/src/plugins/eva/partitioning/auto_loop_unroll.ml @@ -323,7 +323,7 @@ let cross_equality loop lval = | Some lval -> lval | None | exception No_equality -> lval -module Make (Abstract: Abstractions.Eva) = struct +module Make (Abstract: Abstractions.S_with_evaluation) = struct open Eval open Abstract diff --git a/src/plugins/eva/partitioning/auto_loop_unroll.mli b/src/plugins/eva/partitioning/auto_loop_unroll.mli index 4bbc478a68584f5c3bcf6543c41883006692d528..1922eb815dfc850ff97e624f4730ff9c58444028 100644 --- a/src/plugins/eva/partitioning/auto_loop_unroll.mli +++ b/src/plugins/eva/partitioning/auto_loop_unroll.mli @@ -22,7 +22,7 @@ (** Heuristic for automatic loop unrolling. *) -module Make (Abstract: Abstractions.Eva) : sig +module Make (Abstract: Abstractions.S_with_evaluation) : sig val compute: max_unroll:int -> Abstract.Dom.t -> Cil_types.stmt -> int option diff --git a/src/plugins/eva/partitioning/partition.ml b/src/plugins/eva/partitioning/partition.ml index 14723800087101c1b1049d3fd7a0b910ff631278..a994030d11119fdf8254e9cde51a4d6962fca531 100644 --- a/src/plugins/eva/partitioning/partition.ml +++ b/src/plugins/eva/partitioning/partition.ml @@ -322,7 +322,7 @@ exception InvalidAction (* --- Flows --- *) -module MakeFlow (Abstract: Abstractions.Eva) = +module MakeFlow (Abstract: Abstractions.S_with_evaluation) = struct type state = Abstract.Dom.t type t = (key * state) list diff --git a/src/plugins/eva/partitioning/partition.mli b/src/plugins/eva/partitioning/partition.mli index caf4e08c2487c27dba54be9916b43945669c14d4..18256b222e2249af59dc67eef3c5128956a6b4f4 100644 --- a/src/plugins/eva/partitioning/partition.mli +++ b/src/plugins/eva/partitioning/partition.mli @@ -175,7 +175,7 @@ exception InvalidAction (** Flows are used to transfer states from one partition to another, by applying transfer functions and partitioning actions. They do not enforce the unicity of keys. *) -module MakeFlow (Abstract: Abstractions.Eva) : +module MakeFlow (Abstract: Abstractions.S_with_evaluation) : sig type state = Abstract.Dom.t type t diff --git a/src/plugins/eva/partitioning/trace_partitioning.ml b/src/plugins/eva/partitioning/trace_partitioning.ml index 2e38c548ff027ad0587432d265d6e0beaff02f6a..ae449dc72983922a90d7215ed43e9f298d7935db 100644 --- a/src/plugins/eva/partitioning/trace_partitioning.ml +++ b/src/plugins/eva/partitioning/trace_partitioning.ml @@ -26,7 +26,7 @@ open Partition let stat_max_widenings = Statistics.register_statement_stat "max-widenings" module Make - (Abstract: Abstractions.Eva) + (Abstract: Abstractions.S_with_evaluation) (Kf : sig val kf: kernel_function end) = struct module Partition_parameters = Partitioning_parameters.Make (Kf) diff --git a/src/plugins/eva/partitioning/trace_partitioning.mli b/src/plugins/eva/partitioning/trace_partitioning.mli index dd8cade8865e1ec7ae490547566e411efb46b588..d6e22fb5345a33f170950df629cd0b38f6b495d1 100644 --- a/src/plugins/eva/partitioning/trace_partitioning.mli +++ b/src/plugins/eva/partitioning/trace_partitioning.mli @@ -21,7 +21,7 @@ (**************************************************************************) module Make - (Abstract : Abstractions.Eva) + (Abstract : Abstractions.S_with_evaluation) (Kf : sig val kf: Cil_types.kernel_function end) : sig (** The states being partitioned *) diff --git a/src/plugins/eva/utils/abstract.ml b/src/plugins/eva/utils/abstract.ml index 11c54bedde65a23402bffb172c8168234f2f42cb..1ef84e43b66b2516ada82693c6488252cb76a5d8 100644 --- a/src/plugins/eva/utils/abstract.ml +++ b/src/plugins/eva/utils/abstract.ml @@ -80,9 +80,5 @@ module Domain = struct include Structure.External with type t := t and type 'a key := 'a key and type 'a data := 'a data - - val get_cvalue: (t -> Cvalue.Model.t) option - val get_cvalue_or_top: t -> Cvalue.Model.t - val get_cvalue_or_bottom: t Lattice_bounds.or_bottom -> Cvalue.Model.t end end diff --git a/src/plugins/eva/utils/abstract.mli b/src/plugins/eva/utils/abstract.mli index d3d5e3a3a62a07f60af87054df0a7ac32f00e57f..399aac89323e8e0d894e720a5c4d5ee4e9cc6329 100644 --- a/src/plugins/eva/utils/abstract.mli +++ b/src/plugins/eva/utils/abstract.mli @@ -90,10 +90,5 @@ module Domain : sig include Structure.External with type t := t and type 'a key := 'a key and type 'a data := 'a data - - (** 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 Lattice_bounds.or_bottom -> Cvalue.Model.t end end diff --git a/src/plugins/eva/utils/private.ml b/src/plugins/eva/utils/private.ml index 3c28cfb69da993bfc8f99d62ffb18c3c4647359f..2994cdfb8d21cf96fd800f045be6b6982b9bd780 100644 --- a/src/plugins/eva/utils/private.ml +++ b/src/plugins/eva/utils/private.ml @@ -22,10 +22,12 @@ module Abstract_domain = Abstract_domain module Abstract_value = Abstract_value +module Abstract = Abstract module Abstractions = Abstractions module Active_behaviors = Active_behaviors module Alarmset = Alarmset module Analysis = Analysis +module Cvalue_domain = Cvalue_domain module Domain_builder = Domain_builder module Eva_dynamic = Eva_dynamic module Eva_results = Eva_results diff --git a/src/plugins/eva/utils/private.mli b/src/plugins/eva/utils/private.mli index 2012d89d2920890ab2e46f43185c330c2c050d7f..41ab31a858140a4854d048dd62d190a8bf6b02b0 100644 --- a/src/plugins/eva/utils/private.mli +++ b/src/plugins/eva/utils/private.mli @@ -26,10 +26,12 @@ module Abstract_domain = Abstract_domain module Abstract_value = Abstract_value +module Abstract = Abstract module Abstractions = Abstractions module Active_behaviors = Active_behaviors module Alarmset = Alarmset module Analysis = Analysis +module Cvalue_domain = Cvalue_domain module Domain_builder = Domain_builder module Eva_dynamic = Eva_dynamic module Eva_results = Eva_results diff --git a/src/plugins/eva/utils/results.ml b/src/plugins/eva/utils/results.ml index 6133518219ca79430760b556261fb28f1329b98c..2d048a8995e6b381a6c223c74ea543cf108c16be 100644 --- a/src/plugins/eva/utils/results.ml +++ b/src/plugins/eva/utils/results.ml @@ -314,10 +314,11 @@ struct convert r let get_cvalue_model req = - match A.Dom.get_cvalue with + match A.Dom.get Cvalue_domain.State.key with | None -> Result.error DisabledDomain | Some extract -> + let extract s = extract s |> fst in convert (Response.map_join extract Cvalue.Model.join (get req)) let get_state req key join = diff --git a/src/plugins/eva/values/abstract_location.ml b/src/plugins/eva/values/abstract_location.ml index 1cbc439a1f8dbd7e5dc4b23904dc88ff6ded3d36..e4b18564d97f0099492b1a3890670da1a8ffcab5 100644 --- a/src/plugins/eva/values/abstract_location.ml +++ b/src/plugins/eva/values/abstract_location.ml @@ -44,7 +44,7 @@ module type S = sig val pretty_loc: Format.formatter -> location -> unit val pretty_offset : Format.formatter -> offset -> unit - val to_value : location -> value + val to_value : location -> value or_bottom val size : location -> Int_Base.t (** [replace_base substitution location] replaces the variables represented diff --git a/src/plugins/eva/values/location_lift.ml b/src/plugins/eva/values/location_lift.ml index 5aaee286c0d3d1178a417fcd8e8d7663883e6e7b..769faab7a8c797b552699a8a3ea996bee1c9812c 100644 --- a/src/plugins/eva/values/location_lift.ml +++ b/src/plugins/eva/values/location_lift.ml @@ -23,17 +23,16 @@ open Eval module type Conversion = sig - type extended_value - type internal_value - - val extend_val : internal_value -> extended_value - val replace_val : internal_value -> extended_value -> extended_value - val restrict_val : extended_value -> internal_value + type extended + type internal + val extend : internal -> extended + val replace : internal -> extended -> extended + val restrict : extended -> internal end module Make (Loc: Abstract_location.Leaf) - (Convert : Conversion with type internal_value := Loc.value) + (Convert : Conversion with type internal := Loc.value) = struct (* Import most of [Loc] *) @@ -41,29 +40,29 @@ module Make with type value := Loc.value (* we are converting this type *) and type location = Loc.location and type offset = Loc.offset) - type value = Convert.extended_value + type value = Convert.extended let structure = Abstract.Location.Leaf (Loc.key, (module Loc)) (* Now lift the functions that contain {!value} in their type. *) - let to_value loc = Convert.extend_val (Loc.to_value loc) + let to_value loc = Loc.to_value loc >>-: Convert.extend let forward_index typ value offset = - Loc.forward_index typ (Convert.restrict_val value) offset + Loc.forward_index typ (Convert.restrict value) offset let forward_pointer typ value offset = - Loc.forward_pointer typ (Convert.restrict_val value) offset + Loc.forward_pointer typ (Convert.restrict value) offset let backward_pointer value offset loc = - let v = Convert.restrict_val value in + let v = Convert.restrict value in Loc.backward_pointer v offset loc >>-: fun (v, off) -> - Convert.replace_val v value, off + Convert.replace v value, off let backward_index typ ~index:value ~remaining offset = - let index = Convert.restrict_val value in + let index = Convert.restrict value in Loc.backward_index typ ~index ~remaining offset >>-: fun (v, off) -> - Convert.replace_val v value, off + Convert.replace v value, off end diff --git a/src/plugins/eva/values/location_lift.mli b/src/plugins/eva/values/location_lift.mli index 34b80aaa11daac14e7796e10184b66b6bd645e1d..d66d68fb6ca44873bbf1c31eb0ce6943bad92227 100644 --- a/src/plugins/eva/values/location_lift.mli +++ b/src/plugins/eva/values/location_lift.mli @@ -21,20 +21,19 @@ (**************************************************************************) module type Conversion = sig - type extended_value - type internal_value - - val extend_val : internal_value -> extended_value - val replace_val : internal_value -> extended_value -> extended_value - val restrict_val : extended_value -> internal_value + type extended + type internal + val extend : internal -> extended + val replace : internal -> extended -> extended + val restrict : extended -> internal end module Make (Loc: Abstract_location.Leaf) - (Convert : Conversion with type internal_value := Loc.value) + (Convert : Conversion with type internal := Loc.value) : Abstract.Location.Internal with type location = Loc.location and type offset = Loc.offset - and type value = Convert.extended_value + and type value = Convert.extended (* diff --git a/src/plugins/eva/values/main_locations.ml b/src/plugins/eva/values/main_locations.ml index 92d31c7f54b1baeaaf2723e373815aef3cd9bfc3..f74e3799a852ab747d9494b98fb2f10edbd6cedd 100644 --- a/src/plugins/eva/values/main_locations.ml +++ b/src/plugins/eva/values/main_locations.ml @@ -20,7 +20,7 @@ (* *) (**************************************************************************) -module PLoc = struct +module Location = struct type value = Cvalue.V.t type location = Precise_locs.precise_location @@ -28,8 +28,6 @@ module PLoc = struct | Precise of Precise_locs.precise_offset | Imprecise of Cvalue.V.t (* when the offset contains addresses *) - let key = Structure.Key_Location.create_key "precise_locs" - let equal_loc = Precise_locs.equal_loc let equal_offset o1 o2 = match o1, o2 with | Precise o1, Precise o2 -> Precise_locs.equal_offset o1 o2 @@ -43,7 +41,7 @@ module PLoc = struct let to_value t = let loc = Precise_locs.imprecise_location t in - Locations.loc_to_loc_without_size loc + `Value (Locations.loc_to_loc_without_size loc) let size loc = Precise_locs.loc_size loc @@ -235,10 +233,18 @@ module PLoc = struct `Value (index, rem) (* No reduction if the offsets are not arithmetics. *) with Cvalue.V.Not_based_on_null -> `Value (index, remaining) - end +module PLoc = struct + include Location + let key = Structure.Key_Location.create_key "precise_locs" + let registered = Abstractions.Location.register + { key ; location = (module Location) + ; dependencies = Last Main_values.CVal.registered + } +end + (* Local Variables: compile-command: "make -C ../../../.." diff --git a/src/plugins/eva/values/main_locations.mli b/src/plugins/eva/values/main_locations.mli index 4d95a9933e380e44166fae2d006cc0ce957b77a8..7349704d2bb1910d67e730d711ad688d349b70fe 100644 --- a/src/plugins/eva/values/main_locations.mli +++ b/src/plugins/eva/values/main_locations.mli @@ -24,13 +24,14 @@ (** Abstract locations built over Precise_locs. *) module PLoc : sig - - include Abstract_location.Leaf + include Abstract_location.S with type value = Cvalue.V.t and type location = Precise_locs.precise_location val make: Locations.location -> location + val key : location Abstract_location.key + val registered : location Abstractions.Location.registered end (* diff --git a/src/plugins/eva/values/main_values.ml b/src/plugins/eva/values/main_values.ml index fdd47e5052ce7028de7f6f1cb341b4fc0f9a0012..909106265a10f2c9c85f583b4a67be4301decdf6 100644 --- a/src/plugins/eva/values/main_values.ml +++ b/src/plugins/eva/values/main_values.ml @@ -23,175 +23,184 @@ open Cil_types module CVal = struct - include Cvalue.V - - let key = Structure.Key_Value.create_key "cvalue" - - let zero = Cvalue.V.singleton_zero - let one = Cvalue.V.singleton_one - - let top = Cvalue.V.top - let top_int = Cvalue.V.top_int - let inject_int _typ = Cvalue.V.inject_int - - let equal = Cvalue.V.equal - let is_included = Cvalue.V.is_included - let join = Cvalue.V.join - let narrow a b = - let n = Cvalue.V.narrow a b in - if Cvalue.V.is_bottom n - then `Bottom - else `Value n - - let assume_non_zero = Cvalue_forward.assume_non_zero - let assume_bounded = Cvalue_forward.assume_bounded - let assume_not_nan = Cvalue_forward.assume_not_nan - let assume_pointer = Cvalue_forward.assume_pointer - let assume_comparable = Cvalue_forward.assume_comparable - - let constant exp = function - | CInt64 (i,_k,_s) -> Cvalue.V.inject_int i - | CChr c -> Cvalue.V.inject_int (Cil.charConstToInt c) - | CWStr _ | CStr _ -> Cvalue.V.inject (Base.of_string_exp exp) Ival.zero - | CReal (f, fkind, fstring) -> - Cvalue_forward.eval_float_constant f fkind fstring - | CEnum _ -> assert false - - let forward_unop typ unop value = - let value = Cvalue_forward.forward_unop typ unop value in - (* TODO: `Bottom must be in CValue and Cvalue_forward. *) - if Cvalue.V.is_bottom value then `Bottom else `Value value - - let forward_binop typ binop v1 v2 = - let value = - match typ with - | TFloat (fkind, _) -> - Cvalue_forward.forward_binop_float (Fval.kind fkind) v1 binop v2 - | TInt _ | TPtr _ | _ as typ -> - Cvalue_forward.forward_binop_int ~typ v1 binop v2 - in - if Cvalue.V.is_bottom value - then `Bottom - else `Value value - - let rewrap_integer = Cvalue_forward.rewrap_integer - - let forward_cast ~src_type ~dst_type v = - let v = Cvalue_forward.forward_cast ~src_type ~dst_type v in - if Cvalue.V.is_bottom v then `Bottom else `Value v - - let backward_binop ~input_type ~resulting_type binop ~left ~right ~result = - let reduction = - Cvalue_backward.backward_binop - ~typ_res:resulting_type ~res_value:result ~typ_e1:input_type left binop right - in - match reduction with - | None -> `Value (None, None) - | Some (v1, v2) -> - if Cvalue.V.is_bottom v1 || Cvalue.V.is_bottom v2 - then `Bottom - else `Value (Some v1, Some v2) - - let backward_unop ~typ_arg op ~arg ~res = - let reduction = Cvalue_backward.backward_unop ~typ_arg op ~arg ~res in - match reduction with - | None -> `Value None - | Some v as r -> - if Cvalue.V.is_bottom v + module Value = struct + include Cvalue.V + + let zero = Cvalue.V.singleton_zero + let one = Cvalue.V.singleton_one + + let top = Cvalue.V.top + let top_int = Cvalue.V.top_int + let inject_int _typ = Cvalue.V.inject_int + + let equal = Cvalue.V.equal + let is_included = Cvalue.V.is_included + let join = Cvalue.V.join + let narrow a b = + let n = Cvalue.V.narrow a b in + if Cvalue.V.is_bottom n then `Bottom - else `Value r - - let backward_cast ~src_typ ~dst_typ ~src_val ~dst_val = - let reduction = - Cvalue_backward.backward_cast ~src_typ ~dst_typ ~src_val ~dst_val - in - match reduction with - | None -> `Value None - | Some v -> - if Cvalue.V.is_bottom v + else `Value n + + let assume_non_zero = Cvalue_forward.assume_non_zero + let assume_bounded = Cvalue_forward.assume_bounded + let assume_not_nan = Cvalue_forward.assume_not_nan + let assume_pointer = Cvalue_forward.assume_pointer + let assume_comparable = Cvalue_forward.assume_comparable + + let constant exp = function + | CInt64 (i,_k,_s) -> Cvalue.V.inject_int i + | CChr c -> Cvalue.V.inject_int (Cil.charConstToInt c) + | CWStr _ | CStr _ -> Cvalue.V.inject (Base.of_string_exp exp) Ival.zero + | CReal (f, fkind, fstring) -> + Cvalue_forward.eval_float_constant f fkind fstring + | CEnum _ -> assert false + + let forward_unop typ unop value = + let value = Cvalue_forward.forward_unop typ unop value in + (* TODO: `Bottom must be in CValue and Cvalue_forward. *) + if Cvalue.V.is_bottom value then `Bottom else `Value value + + let forward_binop typ binop v1 v2 = + let value = + match typ with + | TFloat (fkind, _) -> + Cvalue_forward.forward_binop_float (Fval.kind fkind) v1 binop v2 + | TInt _ | TPtr _ | _ as typ -> + Cvalue_forward.forward_binop_int ~typ v1 binop v2 + in + if Cvalue.V.is_bottom value then `Bottom - else if Cvalue.V.is_included src_val v - then `Value None - else `Value (Some v) - - let resolve_functions v = - let aux base offs (acc, alarm) = - match base with - | Base.String (_,_) | Base.Null | Base.CLogic_Var _ | Base.Allocated _ -> - acc, true - | Base.Var (v,_) -> - if Cil.isFunctionType v.vtype then - let alarm = alarm || Ival.contains_non_zero offs in - let kf = Globals.Functions.get v in - let list = if Ival.contains_zero offs then kf :: acc else acc in - list, alarm - else acc, true - in - try - let init = [], false in - let kfs, alarm = Locations.Location_Bytes.fold_topset_ok aux v init in - `Value kfs, alarm - with Abstract_interp.Error_Top -> `Top, true - - let replace_base substitution t = snd (Cvalue.V.replace_base substitution t) + else `Value value + + let rewrap_integer = Cvalue_forward.rewrap_integer + + let forward_cast ~src_type ~dst_type v = + let v = Cvalue_forward.forward_cast ~src_type ~dst_type v in + if Cvalue.V.is_bottom v then `Bottom else `Value v + + let backward_binop ~input_type ~resulting_type binop ~left ~right ~result = + let reduction = + Cvalue_backward.backward_binop + ~typ_res:resulting_type ~res_value:result ~typ_e1:input_type left binop right + in + match reduction with + | None -> `Value (None, None) + | Some (v1, v2) -> + if Cvalue.V.is_bottom v1 || Cvalue.V.is_bottom v2 + then `Bottom + else `Value (Some v1, Some v2) + + let backward_unop ~typ_arg op ~arg ~res = + let reduction = Cvalue_backward.backward_unop ~typ_arg op ~arg ~res in + match reduction with + | None -> `Value None + | Some v as r -> + if Cvalue.V.is_bottom v + then `Bottom + else `Value r + + let backward_cast ~src_typ ~dst_typ ~src_val ~dst_val = + let reduction = + Cvalue_backward.backward_cast ~src_typ ~dst_typ ~src_val ~dst_val + in + match reduction with + | None -> `Value None + | Some v -> + if Cvalue.V.is_bottom v + then `Bottom + else if Cvalue.V.is_included src_val v + then `Value None + else `Value (Some v) + + let resolve_functions v = + let aux base offs (acc, alarm) = + match base with + | Base.String (_,_) | Base.Null | Base.CLogic_Var _ | Base.Allocated _ -> + acc, true + | Base.Var (v,_) -> + if Cil.isFunctionType v.vtype then + let alarm = alarm || Ival.contains_non_zero offs in + let kf = Globals.Functions.get v in + let list = if Ival.contains_zero offs then kf :: acc else acc in + list, alarm + else acc, true + in + try + let init = [], false in + let kfs, alarm = Locations.Location_Bytes.fold_topset_ok aux v init in + `Value kfs, alarm + with Abstract_interp.Error_Top -> `Top, true + + let replace_base substitution t = snd (Cvalue.V.replace_base substitution t) + end + + let key = Structure.Key_Value.create_key "cvalue" + let registered = Abstractions.Value.register { key ; value = (module Value) } + include Value end + module Interval = struct + module Value = struct + include Datatype.Option (Ival) + + let pretty_typ _ = pretty + + let top = None + + let is_included a b = match a, b with + | _, None -> true + | None, _ -> false + | Some a, Some b -> Ival.is_included a b + + let join a b = match a, b with + | None, _ | _, None -> None + | Some a, Some b -> Some (Ival.join a b) + + let narrow a b = match a, b with + | None, x | x, None -> `Value x + | Some a, Some b -> + let res = Ival.narrow a b in + if Ival.is_bottom res then `Bottom else `Value (Some res) + + let zero = None + let one = None + let top_int = None + let inject_int _typ i = Some (Ival.inject_singleton i) + + let assume_non_zero v = `Unknown v + let assume_bounded _ _ v = `Unknown v + let assume_not_nan ~assume_finite:_ _ v = `Unknown v + let assume_pointer v = `Unknown v + let assume_comparable _ v1 v2 = `Unknown (v1, v2) + + let constant _ _ = top + let forward_unop _ _ _ = `Value top + let forward_binop _ _ _ _ = `Value top + let forward_cast ~src_type:_ ~dst_type:_ _ = `Value top + + let resolve_functions _ = `Top, true + let replace_base _substitution t = t + + let rewrap_integer range value = + match value with + | None -> value + | Some value -> + let size = Integer.of_int range.Eval_typ.i_bits in + let signed = range.Eval_typ.i_signed in + Some (Ival.cast_int_to_int ~signed ~size value) + + let backward_unop ~typ_arg:_ _unop ~arg:_ ~res:_ = `Value None + let backward_binop ~input_type:_ ~resulting_type:_ _binop ~left:_ ~right:_ ~result:_ = + `Value (None, None) + let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = + `Value None + end - include Datatype.Option (Ival) let key = Structure.Key_Value.create_key "interval" - - let pretty_typ _ = pretty - - let top = None - - let is_included a b = match a, b with - | _, None -> true - | None, _ -> false - | Some a, Some b -> Ival.is_included a b - - let join a b = match a, b with - | None, _ | _, None -> None - | Some a, Some b -> Some (Ival.join a b) - - let narrow a b = match a, b with - | None, x | x, None -> `Value x - | Some a, Some b -> - let res = Ival.narrow a b in - if Ival.is_bottom res then `Bottom else `Value (Some res) - - let zero = None - let one = None - let top_int = None - let inject_int _typ i = Some (Ival.inject_singleton i) - - let assume_non_zero v = `Unknown v - let assume_bounded _ _ v = `Unknown v - let assume_not_nan ~assume_finite:_ _ v = `Unknown v - let assume_pointer v = `Unknown v - let assume_comparable _ v1 v2 = `Unknown (v1, v2) - - let constant _ _ = top - let forward_unop _ _ _ = `Value top - let forward_binop _ _ _ _ = `Value top - let forward_cast ~src_type:_ ~dst_type:_ _ = `Value top - - let resolve_functions _ = `Top, true - let replace_base _substitution t = t - - let rewrap_integer range value = - match value with - | None -> value - | Some value -> - let size = Integer.of_int range.Eval_typ.i_bits in - let signed = range.Eval_typ.i_signed in - Some (Ival.cast_int_to_int ~signed ~size value) - - let backward_unop ~typ_arg:_ _unop ~arg:_ ~res:_ = `Value None - let backward_binop ~input_type:_ ~resulting_type:_ _binop ~left:_ ~right:_ ~result:_ = - `Value (None, None) - let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = - `Value None + let registered = Abstractions.Value.register { key ; value = (module Value) } + include Value end (* diff --git a/src/plugins/eva/values/main_values.mli b/src/plugins/eva/values/main_values.mli index dc71abad115a51b916cab31f719006fe27921f29..e203d3b8f217a0c9fd1a154271777c44cf6298e4 100644 --- a/src/plugins/eva/values/main_values.mli +++ b/src/plugins/eva/values/main_values.mli @@ -23,11 +23,19 @@ (** Main numeric values of Eva. *) (** Abstract values built over Cvalue.V *) -module CVal : Abstract_value.Leaf with type t = Cvalue.V.t +module CVal : sig + include Abstract_value.S with type t = Cvalue.V.t + val key : t Abstract_value.key + val registered : t Abstractions.Value.registered +end (** Dummy interval: no forward nor backward propagations. [None] is top. *) -module Interval : Abstract_value.Leaf with type t = Ival.t option +module Interval : sig + include Abstract_value.S with type t = Ival.t option + val key : t Abstract_value.key + val registered : t Abstractions.Value.registered +end (* Local Variables: diff --git a/src/plugins/eva/values/offsm_value.ml b/src/plugins/eva/values/offsm_value.ml index 35ddb8f38afc8051fb2a606e551258c3ab7753b8..c4b990bf6d8ca67d25b339c0fbb72853c911a55d 100644 --- a/src/plugins/eva/values/offsm_value.ml +++ b/src/plugins/eva/values/offsm_value.ml @@ -369,175 +369,186 @@ module Datatype_Offsm_or_top = Datatype.Make_with_collections(struct end) -module Offsm : Abstract_value.Leaf with type t = offsm_or_top = struct - include Datatype_Offsm_or_top +module Offsm = struct + module Value : Abstract_value.S with type t = offsm_or_top = struct + include Datatype_Offsm_or_top + + let pretty_typ typ fmt = function + | Top as o -> pretty fmt o + | O o -> + Format.fprintf fmt "O @[%a@]" + (V_Offsetmap.pretty_generic ?typ ()) o + + let top = Top + + let is_included o1 o2 = match o1, o2 with + | _, Top -> true + | O o1, O o2 -> V_Offsetmap.is_included o1 o2 + | Top, O _ -> false + + let join o1 o2 = match o1, o2 with + | Top, _ | _, Top -> Top + | O o1, O o2 -> O (V_Offsetmap.join o1 o2) + + let narrow o1 o2 = match o1, o2 with + | Top, o | o, Top -> `Value o + | O o1, O o2 -> + V_Offsetmap.narrow_reinterpret o1 o2 >>-: (fun o -> O o) + + (* Simple values cannot be injected because we do not known their type + (hence size in bits *) + let zero = Top + let one = Top + let top_int = Top + + let inject_int typ i = + try + let size = Integer.of_int (Cil.bitsSizeOf typ) in + O (inject ~size (V.inject_int i)) + with Cil.SizeOfError _ -> Top + + let assume_non_zero v = `Unknown v + let assume_bounded _ _ v = `Unknown v + let assume_not_nan ~assume_finite:_ _ v = `Unknown v + let assume_pointer v = `Unknown v + let assume_comparable _ v1 v2 = `Unknown (v1, v2) + + let constant e _c = + if store_redundant then + match Cil.constFoldToInt e with + | Some i -> inject_int (Cil.typeOf e) i + | None -> Top + else Top + + let resolve_functions _ = `Top, true (* TODO: extract value *) + let replace_base substitution = function + | Top -> Top + | O offsm -> + let f v = snd (Cvalue.V_Or_Uninitialized.replace_base substitution v) in + O (Cvalue.V_Offsetmap.map_on_values f offsm) + + let forward_unop _typ op o = + let o' = match o, op with + | Top, _ | _, (Neg | LNot) -> Top + | O o, BNot -> O (bnot o) + in + `Value o' + + let forward_binop _typ op o1 o2 = + let o' = + match o1, o2, op with + | O _o1, O _o2, (Shiftlt | Shiftrt) -> + (* It is inconvenient to handle shift here, because we need a + constant for o2 *) + Top + | O o1, O o2, BAnd -> O (bitwise_and o1 o2) + | O o1, O o2, BOr -> O (bitwise_or o1 o2) + | O o1, O o2, BXor -> O (bitwise_xor o1 o2) + | _ -> Top + in + `Value o' + + let backward_binop ~input_type:_ ~resulting_type:_ _op ~left:_ ~right:_ ~result:_ = + `Value (None, None) + + let backward_unop ~typ_arg:_ _unop ~arg:_ ~res:_ = `Value None + + let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = + `Value None + + let rewrap_integer _range o = o + + let forward_cast ~src_type ~dst_type o = + let open Eval_typ in + match o, src_type, dst_type with + | O o, (TSInt src | TSPtr src), (TSInt dst | TSPtr dst) -> + let old_size = Int.of_int src.i_bits in + let new_size = Int.of_int dst.i_bits in + let signed = src.i_signed in + `Value (O (cast ~old_size ~new_size ~signed o)) + | _ -> `Value Top + end let key = Structure.Key_Value.create_key "offsetmap_value" - - let pretty_typ typ fmt = function - | Top as o -> pretty fmt o - | O o -> - Format.fprintf fmt "O @[%a@]" - (V_Offsetmap.pretty_generic ?typ ()) o - - let top = Top - - let is_included o1 o2 = match o1, o2 with - | _, Top -> true - | O o1, O o2 -> V_Offsetmap.is_included o1 o2 - | Top, O _ -> false - - let join o1 o2 = match o1, o2 with - | Top, _ | _, Top -> Top - | O o1, O o2 -> O (V_Offsetmap.join o1 o2) - - let narrow o1 o2 = match o1, o2 with - | Top, o | o, Top -> `Value o - | O o1, O o2 -> - V_Offsetmap.narrow_reinterpret o1 o2 >>-: (fun o -> O o) - - (* Simple values cannot be injected because we do not known their type - (hence size in bits *) - let zero = Top - let one = Top - let top_int = Top - - let inject_int typ i = - try - let size = Integer.of_int (Cil.bitsSizeOf typ) in - O (inject ~size (V.inject_int i)) - with Cil.SizeOfError _ -> Top - - let assume_non_zero v = `Unknown v - let assume_bounded _ _ v = `Unknown v - let assume_not_nan ~assume_finite:_ _ v = `Unknown v - let assume_pointer v = `Unknown v - let assume_comparable _ v1 v2 = `Unknown (v1, v2) - - let constant e _c = - if store_redundant then - match Cil.constFoldToInt e with - | Some i -> inject_int (Cil.typeOf e) i - | None -> Top - else Top - - let resolve_functions _ = `Top, true (* TODO: extract value *) - let replace_base substitution = function - | Top -> Top - | O offsm -> - let f v = snd (Cvalue.V_Or_Uninitialized.replace_base substitution v) in - O (Cvalue.V_Offsetmap.map_on_values f offsm) - - let forward_unop _typ op o = - let o' = match o, op with - | Top, _ | _, (Neg | LNot) -> Top - | O o, BNot -> O (bnot o) - in - `Value o' - - let forward_binop _typ op o1 o2 = - let o' = - match o1, o2, op with - | O _o1, O _o2, (Shiftlt | Shiftrt) -> - (* It is inconvenient to handle shift here, because we need a - constant for o2 *) - Top - | O o1, O o2, BAnd -> O (bitwise_and o1 o2) - | O o1, O o2, BOr -> O (bitwise_or o1 o2) - | O o1, O o2, BXor -> O (bitwise_xor o1 o2) - | _ -> Top - in - `Value o' - - let backward_binop ~input_type:_ ~resulting_type:_ _op ~left:_ ~right:_ ~result:_ = - `Value (None, None) - - let backward_unop ~typ_arg:_ _unop ~arg:_ ~res:_ = `Value None - - let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = - `Value None - - let rewrap_integer _range o = o - - let forward_cast ~src_type ~dst_type o = - let open Eval_typ in - match o, src_type, dst_type with - | O o, (TSInt src | TSPtr src), (TSInt dst | TSPtr dst) -> - let old_size = Int.of_int src.i_bits in - let new_size = Int.of_int dst.i_bits in - let signed = src.i_signed in - `Value (O (cast ~old_size ~new_size ~signed o)) - | _ -> `Value Top - + let registered = Abstractions.Value.register { key ; value = (module Value) } + include Value end - -module CvalueOffsm : Abstract.Value.Internal with type t = V.t * offsm_or_top -= struct - include Value_product.Make (Main_values.CVal) (Offsm) - - let structure = - Abstract.Value.(Node (Leaf (Main_values.CVal.key, (module Main_values.CVal)), - Leaf (Offsm.key, (module Offsm)))) - - let size typ = Integer.of_int (Cil.bitsSizeOf typ) - - (* Extract an offsetmap from a pair, by converting the value when needed. *) - let to_offsm typ (v, o : t) = - match o with - | Top -> inject ~size:(size typ) v - | O o -> o - - (* Ensure that the offsetmap component is not empty *) - let strengthen_offsm typ (v, o as p : t) : t = - if o = Top then - (v, O (to_offsm typ p)) - else p - - (* Refine the value component according to the contents of the offsetmap *) - let strengthen_v typ (v, o as p : t) : t or_bottom = - match o with - | Top -> `Value p - | O o' -> - let size = size typ in - (* TODO: this should be done by the transfer function itself... *) - let v = Cvalue_forward.reinterpret typ v in - let v_o = V_Or_Uninitialized.get_v (basic_find ~size o') in - let v_o = Cvalue_forward.reinterpret typ v_o in - let v = V.narrow v v_o in - if V.is_bottom v then `Bottom else `Value (v, o) - - let forward_unop typ op p = - match op with - | BNot -> - let p' = strengthen_offsm typ p in - forward_unop typ op p' >>- fun p'' -> - strengthen_v typ p'' - | _ -> forward_unop typ op p - - let forward_binop typ op l r = - match op with - | BAnd | BOr | BXor -> - let l = strengthen_offsm typ l in - let r = strengthen_offsm typ r in - forward_binop typ op l r >>- fun p -> - strengthen_v typ p - | Shiftlt | Shiftrt -> - let (v_r, _) = r in - let (v_l, _) = l in - begin - try - let i = V.project_ival v_r in - let i = Ival.project_int i in - let size = size typ in - let signed = Bit_utils.is_signed_int_enum_pointer typ in - let dir = if op = Shiftlt then Left else Right in - let o = shift ~size ~signed (to_offsm typ l) dir i in - Main_values.CVal.forward_binop typ op v_l v_r >>-: fun v -> - v, O o - with V.Not_based_on_null | Ival.Not_Singleton_Int -> - forward_binop typ op l r - end - | _ -> forward_binop typ op l r - -end +(* -------------------------------------------------------------------------- *) +(* Reduced product between Cvalues and Offsetmaps values *) +(* -------------------------------------------------------------------------- *) + +let size typ = Integer.of_int (Cil.bitsSizeOf typ) + +(* Extract an offsetmap from a pair, by converting the value when needed. *) +let to_offsm typ v = function + | Top -> inject ~size:(size typ) v + | O o -> o + +(* Refine the cvalue according to the contents of the offsetmap. *) +let strengthen_v typ v offsm : Cvalue.V.t or_bottom = + let size = size typ in + (* TODO: this should be done by the transfer function itself... *) + let v = Cvalue_forward.reinterpret typ v in + let v_o = V_Or_Uninitialized.get_v (basic_find ~size offsm) in + let v_o = Cvalue_forward.reinterpret typ v_o in + let v = V.narrow v v_o in + if V.is_bottom v then `Bottom else `Value v + +let () = Abstractions.Hooks.register @@ fun (module Abstraction) -> + let module Val = Abstraction.Val in + match Val.get Main_values.CVal.key, Val.get Offsm.key with + | None, _ | _, None -> (module Abstraction) + | Some get_cvalue, Some get_offsm -> + let module Value = struct + include Abstraction.Val + + let set_cvalue = set Main_values.CVal.key + let set_offsm = set Offsm.key + + let to_offsm typ t = to_offsm typ (get_cvalue t) (get_offsm t) + + (* Ensure that the offsetmap component is not empty. *) + let strengthen_offsm typ t = set_offsm (O (to_offsm typ t)) t + + (* Refine the cvalue component according to the offsetmap component. *) + let strengthen_v typ t = + match get_offsm t with + | Top -> `Value t + | O o -> + let* v = strengthen_v typ (get_cvalue t) o in + `Value (set_cvalue v t) + + let forward_unop typ op t = + match op with + | BNot -> + let t = strengthen_offsm typ t in + let* t = forward_unop typ op t in + strengthen_v typ t + | _ -> forward_unop typ op t + + let forward_binop typ op l r = + match op with + | BAnd | BOr | BXor -> + let l = strengthen_offsm typ l + and r = strengthen_offsm typ r in + let* t = forward_binop typ op l r in + strengthen_v typ t + | Shiftlt | Shiftrt -> + let* p = forward_binop typ op l r in + begin + try + let i = get_cvalue r |> V.project_ival |> Ival.project_int in + let size = size typ in + let signed = Bit_utils.is_signed_int_enum_pointer typ in + let dir = if op = Shiftlt then Left else Right in + let offsm = shift ~size ~signed (to_offsm typ l) dir i in + `Value (set_offsm (O offsm) p) + with V.Not_based_on_null | Ival.Not_Singleton_Int -> `Value p + end + | _ -> forward_binop typ op l r + end in + (module struct + include Abstraction + module Val = Value + end) diff --git a/src/plugins/eva/values/offsm_value.mli b/src/plugins/eva/values/offsm_value.mli index cae5bd0efeec1484b5605279916b1ab56403e7c5..2cf4b088ae641e81addd04b0c3311f4a23ae4704 100644 --- a/src/plugins/eva/values/offsm_value.mli +++ b/src/plugins/eva/values/offsm_value.mli @@ -26,6 +26,8 @@ val cast : old_size: Integer.t -> new_size: Integer.t -> signed: bool -> Cvalue.V_Offsetmap.t -> Cvalue.V_Offsetmap.t -module Offsm : Abstract_value.Leaf with type t = offsm_or_top - -module CvalueOffsm : Abstract.Value.Internal with type t = Cvalue.V.t * offsm_or_top +module Offsm : sig + include Abstract_value.S with type t = offsm_or_top + val key : t Abstract_value.key + val registered : t Abstractions.Value.registered +end diff --git a/src/plugins/eva/values/sign_value.ml b/src/plugins/eva/values/sign_value.ml index d60458ec1e0ed6eeb36d4bb26b0b54476b6ad2d8..9ac0956e0dd4343e5b369f0bb60f928b848f40ea 100644 --- a/src/plugins/eva/values/sign_value.ml +++ b/src/plugins/eva/values/sign_value.ml @@ -25,321 +25,326 @@ open Eval open Abstract_interp (** Sign domain: abstraction of integer numerical values by their signs. *) - -type signs = { - pos: bool; (** true: maybe positive, false: never positive *) - zero: bool; (** true: maybe zero, false: never zero *) - neg: bool; (** true: maybe negative, false: never negative *) -} - -let top = { pos = true; zero = true; neg = true } -let pos_or_zero = { pos = true; zero = true; neg = false } -let pos = { pos = true; zero = false; neg = false } -let neg_or_zero = { pos = false; zero = true; neg = true } -let neg = { pos = false; zero = false; neg = true } -let zero = { pos = false; zero = true; neg = false } -let one = { pos = true; zero = false; neg = false } -let non_zero = { pos = true; zero = false; neg = true } -let ge_zero v = not v.neg -let le_zero v = not v.pos - -(* Bottom is a special value (`Bottom) in Eva, and need not be part of - the lattice. Here, we have a value which is equivalent to it, defined - there only for commodity. *) -let empty = { pos = false; zero = false; neg = false } - -(* Datatypes are Frama-C specific modules used among other things for - serialization. There is no need to understand them in detail. - They are created mostly via copy/paste of templates. *) -include Datatype.Make(struct - type t = signs - include Datatype.Serializable_undefined - let compare = Stdlib.compare - let equal = Datatype.from_compare - let hash = Hashtbl.hash - let reprs = [top] - let name = "Value.Sign_values.signs" - let pretty fmt v = - Format.fprintf fmt "%s%s%s" - (if v.neg then "-" else "") - (if v.zero then "0" else "") - (if v.pos then "+" else "") - end) -let pretty_debug = pretty -let pretty_typ _ = pretty - -(* Inclusion: test inclusion of each field. *) -let is_included v1 v2 = - let bincl b1 b2 = (not b1) || b2 in - bincl v1.pos v2.pos && bincl v1.zero v2.zero && bincl v1.neg v2.neg - -(* Join of the lattice: pointwise logical or. *) -let join v1 v2 = { - pos = v1.pos || v2.pos; - zero = v1.zero || v2.zero; - neg = v1.neg || v2.neg; -} - -(* Meet of the lattice (called 'narrow' in Eva for historical reasons). - We detect the case where the values have incompatible concretization, - and report this as `Bottom. *) -let narrow v1 v2 = - let r = { - pos = v1.pos && v2.pos; - zero = v1.zero && v2.zero; - neg = v1.neg && v2.neg; - } in - if r = empty then `Bottom else `Value r - -let top_int = top - -(* [inject_int] creates an abstract value corresponding to the singleton [i]. *) -let inject_int _ i = - if Integer.lt i Integer.zero then neg - else if Integer.gt i Integer.zero then pos - else zero - -let constant _ = function - | CInt64 (i, _, _) -> inject_int () i - | _ -> top - -(* Extracting function pointers from an abstraction. Not implemented - precisely *) -let resolve_functions _ = `Top, true - -let replace_base _substitution t = t - -(** {2 Alarms} *) - -let assume_non_zero v = - if equal v zero - then `False - else if v.zero - then `Unknown {v with zero = false} - else `True - -(* TODO: use the bound to reduce the value when possible. *) -let assume_bounded _ _ v = `Unknown v - -let assume_not_nan ~assume_finite:_ _ v = `Unknown v -let assume_pointer v = `Unknown v -let assume_comparable _ v1 v2 = `Unknown (v1, v2) - -(** {2 Forward transfer functions} *) - -(* Functions [neg_unop], [plus], [mul] and [div] below are forward transformers - for the mathematical operations -, +, *, /. The potential overflows and - wrappings for the operations on machine integers are taken into account by - the functions [truncate_integer] and [rewrap_integer]. *) - -let neg_unop v = { v with neg = v.pos; pos = v.neg } - -let bitwise_not typ v = - match Cil.unrollType typ with - | TInt (ikind, _) | TEnum ({ekind=ikind}, _) -> - if Cil.isSigned ikind - then { pos = v.neg; neg = v.pos || v.zero; zero = v.neg } - else { pos = v.pos || v.zero; neg = false; zero = v.pos } - | _ -> top - -let logical_not v = { pos = v.zero; neg = false; zero = v.pos || v.neg } - -let forward_unop typ op v = - match op with - | Neg -> `Value (neg_unop v) - | BNot -> `Value (bitwise_not typ v) - | LNot -> `Value (logical_not v) - -let plus v1 v2 = - let neg = v1.neg || v2.neg in - let pos = v1.pos || v2.pos in - let same_sign v1 v2 = - (le_zero v1 && le_zero v2) || (ge_zero v1 && ge_zero v2) - in - let zero = not (same_sign v1 v2) || (v1.zero && v2.zero) in - { neg; pos; zero } - -let mul v1 v2 = - let pos = (v1.pos && v2.pos) || (v1.neg && v2.neg) in - let neg = (v1.pos && v2.neg) || (v1.neg && v2.pos) in - let zero = v1.zero || v2.zero in - { neg; pos; zero } - -let div v1 v2 = - let pos = (v1.pos && v2.pos) || (v1.neg && v2.neg) in - let neg = (v1.pos && v2.neg) || (v1.neg && v2.pos) in - let zero = true in (* zero can appear with large enough v2 *) - { neg; pos; zero } - -(* The implementation of the bitwise operators below relies on this table - giving the sign of the result according to the sign of both operands. - - v1 v2 v1&v2 v1|v2 v1^v2 - ----------------------------------- - | + + +0 + +0 - | + 0 0 + + - | + - +0 - - - | 0 + 0 + + - | 0 0 0 0 0 - | 0 - 0 - - - | - + +0 - - - | - 0 0 - - - | - - - - +0 -*) - -let bitwise_and v1 v2 = - let pos = (v1.pos && (v2.pos || v2.neg)) || (v2.pos && v1.neg) in - let neg = v1.neg && v2.neg in - let zero = v1.zero || v1.pos || v2.zero || v2.pos in - { neg; pos; zero } - -let bitwise_or v1 v2 = - let pos = (v1.pos && (v2.pos || v2.zero)) || (v1.zero && v2.pos) in - let neg = v1.neg || v2.neg in - let zero = v1.zero && v2.zero in - { neg; pos; zero } - -let bitwise_xor v1 v2 = - let pos = - (v1.pos && v2.pos) || (v1.pos && v2.zero) || (v1.zero && v2.pos) - || (v1.neg && v2.neg) - in - let neg = - (v1.neg && (v2.pos || v2.zero)) || - (v2.neg && (v1.pos || v1.zero)) - in - let zero = (v1.zero && v2.zero) || (v1.pos && v2.pos) || (v1.neg && v2.neg) in - { neg; pos; zero } - -let logical_and v1 v2 = - let pos = (v1.pos || v1.neg) && (v2.pos || v2.neg) in - let neg = false in - let zero = v1.zero || v2.zero in - { pos; neg; zero } - -let logical_or v1 v2 = - let pos = v1.pos || v1.neg || v2.pos || v2.neg in - let neg = false in - let zero = v1.zero && v2.zero in - { pos; neg; zero } - -let forward_binop _ op v1 v2 = - match op with - | PlusA -> `Value (plus v1 v2) - | MinusA -> `Value (plus v1 (neg_unop v2)) - | Mult -> `Value (mul v1 v2) - | Div -> if equal zero v2 then `Bottom else `Value (div v1 v2) - | BAnd -> `Value (bitwise_and v1 v2) - | BOr -> `Value (bitwise_or v1 v2) - | BXor -> `Value (bitwise_xor v1 v2) - | LAnd -> `Value (logical_and v1 v2) - | LOr -> `Value (logical_or v1 v2) - | _ -> `Value top - -let rewrap_integer range v = - if equal v zero then v - else if range.Eval_typ.i_signed then top else pos_or_zero - -(* Casts from type [src_typ] to type [dst_typ]. As downcasting can wrap, - we only handle upcasts precisely *) -let forward_cast ~src_type ~dst_type v = - let open Eval_typ in - match src_type, dst_type with - | TSInt range_src, TSInt range_dst -> - if equal v zero then `Value v else - if range_inclusion range_src range_dst then - `Value v (* upcast *) - else if range_dst.i_signed then - `Value top (*dst_typ is signed, return all possible values*) - else - `Value pos_or_zero (* dst_typ is unsigned *) - | _ -> - (* at least one non-integer type. not handled precisely. *) - `Value top - - -(** {2 Backward transfer functions} *) - -(* Backward transfer functions are used to reduce the abstraction of a value, - knowing other information. For example '[0+] > [0]' means that the - first value can only be [+]. - - In the OCaml signatures, 'None' means 'I cannot reduce'. *) - -(* Value to return when no reduction is possible *) -let unreduced = `Value None -(* Function to use when a reduction is possible *) -let reduced v = `Value (Some v) - -(* This function must reduce the value [right] assuming that the - comparison [left op right] holds. *) -let backward_comp_right op ~left ~right = - let open Abstract_interp.Comp in - match op with - | Eq -> - narrow left right >>- reduced - | Ne -> - if equal left zero then - narrow right non_zero >>- reduced - else unreduced - | Le -> - if ge_zero left then - (* [left] is positive or zero. Hence, [right] is at least also positive - or zero. *) - if left.zero then - (* [left] may be zero, [right] is positive or zero *) - narrow right pos_or_zero >>- reduced +module Value = struct + + type signs = { + pos: bool; (** true: maybe positive, false: never positive *) + zero: bool; (** true: maybe zero, false: never zero *) + neg: bool; (** true: maybe negative, false: never negative *) + } + + let top = { pos = true; zero = true; neg = true } + let pos_or_zero = { pos = true; zero = true; neg = false } + let pos = { pos = true; zero = false; neg = false } + let neg_or_zero = { pos = false; zero = true; neg = true } + let neg = { pos = false; zero = false; neg = true } + let zero = { pos = false; zero = true; neg = false } + let one = { pos = true; zero = false; neg = false } + let non_zero = { pos = true; zero = false; neg = true } + let ge_zero v = not v.neg + let le_zero v = not v.pos + + (* Bottom is a special value (`Bottom) in Eva, and need not be part of + the lattice. Here, we have a value which is equivalent to it, defined + there only for commodity. *) + let empty = { pos = false; zero = false; neg = false } + + (* Datatypes are Frama-C specific modules used among other things for + serialization. There is no need to understand them in detail. + They are created mostly via copy/paste of templates. *) + include Datatype.Make(struct + type t = signs + include Datatype.Serializable_undefined + let compare = Stdlib.compare + let equal = Datatype.from_compare + let hash = Hashtbl.hash + let reprs = [top] + let name = "Value.Sign_values.signs" + let pretty fmt v = + Format.fprintf fmt "%s%s%s" + (if v.neg then "-" else "") + (if v.zero then "0" else "") + (if v.pos then "+" else "") + end) + let pretty_debug = pretty + let pretty_typ _ = pretty + + (* Inclusion: test inclusion of each field. *) + let is_included v1 v2 = + let bincl b1 b2 = (not b1) || b2 in + bincl v1.pos v2.pos && bincl v1.zero v2.zero && bincl v1.neg v2.neg + + (* Join of the lattice: pointwise logical or. *) + let join v1 v2 = { + pos = v1.pos || v2.pos; + zero = v1.zero || v2.zero; + neg = v1.neg || v2.neg; + } + + (* Meet of the lattice (called 'narrow' in Eva for historical reasons). + We detect the case where the values have incompatible concretization, + and report this as `Bottom. *) + let narrow v1 v2 = + let r = { + pos = v1.pos && v2.pos; + zero = v1.zero && v2.zero; + neg = v1.neg && v2.neg; + } in + if r = empty then `Bottom else `Value r + + let top_int = top + + (* [inject_int] creates an abstract value corresponding to the singleton [i]. *) + let inject_int _ i = + if Integer.lt i Integer.zero then neg + else if Integer.gt i Integer.zero then pos + else zero + + let constant _ = function + | CInt64 (i, _, _) -> inject_int () i + | _ -> top + + (* Extracting function pointers from an abstraction. Not implemented + precisely *) + let resolve_functions _ = `Top, true + + let replace_base _substitution t = t + + (** {2 Alarms} *) + + let assume_non_zero v = + if equal v zero + then `False + else if v.zero + then `Unknown {v with zero = false} + else `True + + (* TODO: use the bound to reduce the value when possible. *) + let assume_bounded _ _ v = `Unknown v + + let assume_not_nan ~assume_finite:_ _ v = `Unknown v + let assume_pointer v = `Unknown v + let assume_comparable _ v1 v2 = `Unknown (v1, v2) + + (** {2 Forward transfer functions} *) + + (* Functions [neg_unop], [plus], [mul] and [div] below are forward transformers + for the mathematical operations -, +, *, /. The potential overflows and + wrappings for the operations on machine integers are taken into account by + the functions [truncate_integer] and [rewrap_integer]. *) + + let neg_unop v = { v with neg = v.pos; pos = v.neg } + + let bitwise_not typ v = + match Cil.unrollType typ with + | TInt (ikind, _) | TEnum ({ekind=ikind}, _) -> + if Cil.isSigned ikind + then { pos = v.neg; neg = v.pos || v.zero; zero = v.neg } + else { pos = v.pos || v.zero; neg = false; zero = v.pos } + | _ -> top + + let logical_not v = { pos = v.zero; neg = false; zero = v.pos || v.neg } + + let forward_unop typ op v = + match op with + | Neg -> `Value (neg_unop v) + | BNot -> `Value (bitwise_not typ v) + | LNot -> `Value (logical_not v) + + let plus v1 v2 = + let neg = v1.neg || v2.neg in + let pos = v1.pos || v2.pos in + let same_sign v1 v2 = + (le_zero v1 && le_zero v2) || (ge_zero v1 && ge_zero v2) + in + let zero = not (same_sign v1 v2) || (v1.zero && v2.zero) in + { neg; pos; zero } + + let mul v1 v2 = + let pos = (v1.pos && v2.pos) || (v1.neg && v2.neg) in + let neg = (v1.pos && v2.neg) || (v1.neg && v2.pos) in + let zero = v1.zero || v2.zero in + { neg; pos; zero } + + let div v1 v2 = + let pos = (v1.pos && v2.pos) || (v1.neg && v2.neg) in + let neg = (v1.pos && v2.neg) || (v1.neg && v2.pos) in + let zero = true in (* zero can appear with large enough v2 *) + { neg; pos; zero } + + (* The implementation of the bitwise operators below relies on this table + giving the sign of the result according to the sign of both operands. + + v1 v2 v1&v2 v1|v2 v1^v2 + ----------------------------------- + | + + +0 + +0 + | + 0 0 + + + | + - +0 - - + | 0 + 0 + + + | 0 0 0 0 0 + | 0 - 0 - - + | - + +0 - - + | - 0 0 - - + | - - - - +0 + *) + + let bitwise_and v1 v2 = + let pos = (v1.pos && (v2.pos || v2.neg)) || (v2.pos && v1.neg) in + let neg = v1.neg && v2.neg in + let zero = v1.zero || v1.pos || v2.zero || v2.pos in + { neg; pos; zero } + + let bitwise_or v1 v2 = + let pos = (v1.pos && (v2.pos || v2.zero)) || (v1.zero && v2.pos) in + let neg = v1.neg || v2.neg in + let zero = v1.zero && v2.zero in + { neg; pos; zero } + + let bitwise_xor v1 v2 = + let pos = + (v1.pos && v2.pos) || (v1.pos && v2.zero) || (v1.zero && v2.pos) + || (v1.neg && v2.neg) + in + let neg = + (v1.neg && (v2.pos || v2.zero)) || + (v2.neg && (v1.pos || v1.zero)) + in + let zero = (v1.zero && v2.zero) || (v1.pos && v2.pos) || (v1.neg && v2.neg) in + { neg; pos; zero } + + let logical_and v1 v2 = + let pos = (v1.pos || v1.neg) && (v2.pos || v2.neg) in + let neg = false in + let zero = v1.zero || v2.zero in + { pos; neg; zero } + + let logical_or v1 v2 = + let pos = v1.pos || v1.neg || v2.pos || v2.neg in + let neg = false in + let zero = v1.zero && v2.zero in + { pos; neg; zero } + + let forward_binop _ op v1 v2 = + match op with + | PlusA -> `Value (plus v1 v2) + | MinusA -> `Value (plus v1 (neg_unop v2)) + | Mult -> `Value (mul v1 v2) + | Div -> if equal zero v2 then `Bottom else `Value (div v1 v2) + | BAnd -> `Value (bitwise_and v1 v2) + | BOr -> `Value (bitwise_or v1 v2) + | BXor -> `Value (bitwise_xor v1 v2) + | LAnd -> `Value (logical_and v1 v2) + | LOr -> `Value (logical_or v1 v2) + | _ -> `Value top + + let rewrap_integer range v = + if equal v zero then v + else if range.Eval_typ.i_signed then top else pos_or_zero + + (* Casts from type [src_typ] to type [dst_typ]. As downcasting can wrap, + we only handle upcasts precisely *) + let forward_cast ~src_type ~dst_type v = + let open Eval_typ in + match src_type, dst_type with + | TSInt range_src, TSInt range_dst -> + if equal v zero then `Value v else + if range_inclusion range_src range_dst then + `Value v (* upcast *) + else if range_dst.i_signed then + `Value top (*dst_typ is signed, return all possible values*) else - (* [left] is strictly positive, hence so is [right] *) + `Value pos_or_zero (* dst_typ is unsigned *) + | _ -> + (* at least one non-integer type. not handled precisely. *) + `Value top + + + (** {2 Backward transfer functions} *) + + (* Backward transfer functions are used to reduce the abstraction of a value, + knowing other information. For example '[0+] > [0]' means that the + first value can only be [+]. + + In the OCaml signatures, 'None' means 'I cannot reduce'. *) + + (* Value to return when no reduction is possible *) + let unreduced = `Value None + (* Function to use when a reduction is possible *) + let reduced v = `Value (Some v) + + (* This function must reduce the value [right] assuming that the + comparison [left op right] holds. *) + let backward_comp_right op ~left ~right = + let open Abstract_interp.Comp in + match op with + | Eq -> + narrow left right >>- reduced + | Ne -> + if equal left zero then + narrow right non_zero >>- reduced + else unreduced + | Le -> + if ge_zero left then + (* [left] is positive or zero. Hence, [right] is at least also positive + or zero. *) + if left.zero then + (* [left] may be zero, [right] is positive or zero *) + narrow right pos_or_zero >>- reduced + else + (* [left] is strictly positive, hence so is [right] *) + narrow right pos >>- reduced + else unreduced + | Lt -> + if ge_zero left then narrow right pos >>- reduced - else unreduced - | Lt -> - if ge_zero left then - narrow right pos >>- reduced - else unreduced - | Ge -> - if le_zero left then - if left.zero then - narrow right neg_or_zero >>- reduced - else + else unreduced + | Ge -> + if le_zero left then + if left.zero then + narrow right neg_or_zero >>- reduced + else + narrow right neg >>- reduced + else unreduced + | Gt -> + if le_zero left then narrow right neg >>- reduced - else unreduced - | Gt -> - if le_zero left then - narrow right neg >>- reduced - else unreduced - -(* This functions must reduce the values [left] and [right], assuming - that [left op right == result] holds. Currently, it is only implemented - for comparison operators. *) -let backward_binop ~input_type:_ ~resulting_type:_ op ~left ~right ~result = - match op with - | Ne | Eq | Le | Lt | Ge | Gt -> - let op = Eva_utils.conv_comp op in - if equal zero result then - (* The comparison is false, as it always evaluate to false. Reduce by the - fact that the inverse comparison is true. *) - let op = Comp.inv op in - backward_comp_right op ~left ~right >>- fun right' -> - backward_comp_right (Comp.sym op) ~left:right ~right:left >>- fun left' -> - `Value (left', right') - else if not result.zero then - (* The comparison always hold, as it never evaluates to false. *) - backward_comp_right op ~left ~right >>- fun right' -> - backward_comp_right (Comp.sym op) ~left:right ~right:left >>- fun left' -> - `Value (left', right') - else - (* The comparison may or may not hold, it is not possible to reduce *) - `Value (None, None) - | _ -> `Value (None, None) - -(* Not implemented precisely *) -let backward_unop ~typ_arg:_ _op ~arg:_ ~res:_ = `Value None -(* Not implemented precisely *) -let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = `Value None + else unreduced + + (* This functions must reduce the values [left] and [right], assuming + that [left op right == result] holds. Currently, it is only implemented + for comparison operators. *) + let backward_binop ~input_type:_ ~resulting_type:_ op ~left ~right ~result = + match op with + | Ne | Eq | Le | Lt | Ge | Gt -> + let op = Eva_utils.conv_comp op in + if equal zero result then + (* The comparison is false, as it always evaluate to false. Reduce by the + fact that the inverse comparison is true. *) + let op = Comp.inv op in + backward_comp_right op ~left ~right >>- fun right' -> + backward_comp_right (Comp.sym op) ~left:right ~right:left >>- fun left' -> + `Value (left', right') + else if not result.zero then + (* The comparison always hold, as it never evaluates to false. *) + backward_comp_right op ~left ~right >>- fun right' -> + backward_comp_right (Comp.sym op) ~left:right ~right:left >>- fun left' -> + `Value (left', right') + else + (* The comparison may or may not hold, it is not possible to reduce *) + `Value (None, None) + | _ -> `Value (None, None) + + (* Not implemented precisely *) + let backward_unop ~typ_arg:_ _op ~arg:_ ~res:_ = `Value None + (* Not implemented precisely *) + let backward_cast ~src_typ:_ ~dst_typ:_ ~src_val:_ ~dst_val:_ = `Value None + +end (** {2 Misc} *) (* Eva boilerplate, used to retrieve the domain. *) let key = Structure.Key_Value.create_key "sign_values" +let registered = Abstractions.Value.register { key ; value = (module Value) } +include Value diff --git a/src/plugins/eva/values/sign_value.mli b/src/plugins/eva/values/sign_value.mli index aa4030e1e386fdf7d2ee8e05e1f65fdbd6d63d85..7c8853fbd1c253af2702d31751aa0746714b5c0e 100644 --- a/src/plugins/eva/values/sign_value.mli +++ b/src/plugins/eva/values/sign_value.mli @@ -28,6 +28,7 @@ type signs = { neg: bool; (** true: maybe negative, false: never negative *) } -include Abstract_value.Leaf with type t = signs - +include Abstract_value.S with type t = signs +val key : t Abstract_value.key val pretty_debug: t Pretty_utils.formatter +val registered : t Abstractions.Value.registered diff --git a/src/plugins/eva/values/value_product.ml b/src/plugins/eva/values/value_product.ml index d0f99098c43092082b9c585ceebec1623ca6ba34..8f2101a68cd756802b98f379afae7fbefb4d47f9 100644 --- a/src/plugins/eva/values/value_product.ml +++ b/src/plugins/eva/values/value_product.ml @@ -22,6 +22,28 @@ open Eval +(* Intersects the truth values [t1] and [t2] coming from [assume_] functions + from both abstract values. [v1] and [v2] are the initial values leading to + these truth values, that may be reduced by the assumption. [combine] + combines values from both abstract values into values of the product. *) +let narrow_any_truth combine (v1, t1) (v2, t2) = match t1, t2 with + | `Unreachable, _ | _, `Unreachable + | (`True | `TrueReduced _), `False + | `False, (`True | `TrueReduced _) -> `Unreachable + | `False, _ | _, `False -> `False + | `Unknown v1, `Unknown v2 -> `Unknown (combine v1 v2) + | (`Unknown v1 | `TrueReduced v1), `True -> `TrueReduced (combine v1 v2) + | `True, (`Unknown v2 | `TrueReduced v2) -> `TrueReduced (combine v1 v2) + | (`Unknown v1 | `TrueReduced v1), + (`Unknown v2 | `TrueReduced v2) -> `TrueReduced (combine v1 v2) + | `True, `True -> `True + +let narrow_truth x y = narrow_any_truth (fun left right -> left, right) x y + +let narrow_truth_pair x y = + let combine (l1, l2) (r1, r2) = (l1, r1), (l2, r2) in + narrow_any_truth combine x y + module Make (Left: Abstract_value.S) (Right: Abstract_value.S) @@ -48,24 +70,6 @@ module Make let top_int = Left.top_int, Right.top_int let inject_int typ i = Left.inject_int typ i, Right.inject_int typ i - (* Intersects the truth values [t1] and [t2] coming from [assume_] functions - from both abstract values. [v1] and [v2] are the initial values leading to - these truth values, that may be reduced by the assumption. [combine] - combines values from both abstract values into values of the product. *) - let narrow_any_truth combine (v1, t1) (v2, t2) = match t1, t2 with - | `Unreachable, _ | _, `Unreachable - | (`True | `TrueReduced _), `False - | `False, (`True | `TrueReduced _) -> `Unreachable - | `False, _ | _, `False -> `False - | `Unknown v1, `Unknown v2 -> `Unknown (combine v1 v2) - | (`Unknown v1 | `TrueReduced v1), `True -> `TrueReduced (combine v1 v2) - | `True, (`Unknown v2 | `TrueReduced v2) -> `TrueReduced (combine v1 v2) - | (`Unknown v1 | `TrueReduced v1), - (`Unknown v2 | `TrueReduced v2) -> `TrueReduced (combine v1 v2) - | `True, `True -> `True - - let narrow_truth = narrow_any_truth (fun left right -> left, right) - let assume_non_zero (left, right) = let left_truth = Left.assume_non_zero left and right_truth = Right.assume_non_zero right in @@ -89,8 +93,7 @@ module Make let assume_comparable op (l1, r1) (l2, r2) = let left_truth = Left.assume_comparable op l1 l2 and right_truth = Right.assume_comparable op r1 r2 in - let combine (l1, l2) (r1, r2) = (l1, r1), (l2, r2) in - narrow_any_truth combine ((l1, l2), left_truth) ((r1, r2), right_truth) + narrow_truth_pair ((l1, l2), left_truth) ((r1, r2), right_truth) let constant expr constant = let left = Left.constant expr constant diff --git a/src/plugins/eva/values/value_product.mli b/src/plugins/eva/values/value_product.mli index 43aca79e8140ec25f9c5ea50b33a68ba157a42f8..f53f265b25b1670f777045be22892c8f096c665b 100644 --- a/src/plugins/eva/values/value_product.mli +++ b/src/plugins/eva/values/value_product.mli @@ -22,6 +22,18 @@ (** Cartesian product of two value abstractions. *) +type 'v truth := 'v Abstract_value.truth + +(** [narrow_truth (v1, t1) (v2, t2)] intersects the truth values [t1] and [t2] + resulting from [assume_] functions for abstract values [v1] and [v2] + (that may be reduced by the assumption). *) +val narrow_truth: 'a * 'a truth -> 'b * 'b truth -> ('a * 'b) truth + +(** Same as narrow_truth for truth values involving pairs of abstract values. *) +val narrow_truth_pair: + ('a * 'a) * ('a * 'a) truth -> ('b * 'b) * ('b * 'b) truth -> + (('a * 'b) * ('a * 'b)) truth + module Make (Left: Abstract_value.S) (Right: Abstract_value.S) diff --git a/tests/value/oracle_bitwise/addition.res.oracle b/tests/value/oracle_bitwise/addition.res.oracle index 5bf7aa4f39c61e0f2a36094f97327d3f11610ef7..729310a2affcab008aba9cbf34a61e8629f09d58 100644 --- a/tests/value/oracle_bitwise/addition.res.oracle +++ b/tests/value/oracle_bitwise/addition.res.oracle @@ -4,9 +4,9 @@ < The imprecision originates from Arithmetic {addition.i:52} --- > [eva] addition.i:52: Assigning imprecise value to p10. -163a162 +162a161 > {{ garbled mix of &{p1} (origin: Misaligned {addition.i:52}) }} -165a165 +163a163 > {{ garbled mix of &{p2} (origin: Misaligned {addition.i:56}) }} 186c186 < p10 ∈ {{ garbled mix of &{p1} (origin: Arithmetic {addition.i:52}) }}