You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

417 lines
14 KiB

(*
* Copyright (c) 2009 - 2013 Monoidics ltd.
* Copyright (c) 2013 - present Facebook, Inc.
* All rights reserved.
*
* This source code is licensed under the BSD style license found in the
* LICENSE file in the root directory of this source tree. An additional grant
* of patent rights can be found in the PATENTS file in the same directory.
*)
open! Utils
(** Functions for Propositions (i.e., Symbolic Heaps) *)
open Sil
type normal (** kind for normal props, i.e. normalized *)
type exposed (** kind for exposed props *)
(** Proposition. *)
type pi = Sil.atom list
type sigma = Sil.hpred list
(** the kind 'a should range over [normal] and [exposed] *)
type 'a t = private
{
sigma: sigma; (** spatial part *)
sub: Sil.subst; (** substitution *)
pi: pi; (** pure part *)
sigma_fp : sigma; (** abduced spatial part *)
pi_fp: pi; (** abduced pure part *)
} [@@deriving compare]
(** type to describe different strategies for initializing fields of a structure. [No_init] does not
initialize any fields of the struct. [Fld_init] initializes the fields of the struct with fresh
variables (C) or default values (Java). *)
type struct_init_mode =
| No_init
| Fld_init
(** {2 Basic Functions for propositions} *)
(** Compare propositions *)
val compare_prop : 'a t -> 'a t -> int
(** Check the equality of two sigma's *)
val equal_sigma : sigma -> sigma -> bool
(** Check the equality of two propositions *)
val equal_prop : 'a t -> 'a t -> bool
(** Pretty print a substitution. *)
val pp_sub : printenv -> Format.formatter -> subst -> unit
(** Dump a substitution. *)
val d_sub : subst -> unit
(** Pretty print a pi. *)
val pp_pi : printenv -> Format.formatter -> pi -> unit
(** Dump a pi. *)
val d_pi : pi -> unit
(** Pretty print a sigma. *)
val pp_sigma : printenv -> Format.formatter -> sigma -> unit
(** Dump a sigma. *)
val d_sigma : sigma -> unit
(** Dump a pi and a sigma *)
val d_pi_sigma: pi -> sigma -> unit
(** Split sigma into stack and nonstack parts.
The boolean indicates whether the stack should only include local variales. *)
val sigma_get_stack_nonstack : bool -> sigma -> sigma * sigma
(** Update the object substitution given the stack variables in the prop *)
val prop_update_obj_sub : printenv -> 'a t -> printenv
(** Pretty print a proposition. *)
val pp_prop : printenv -> Format.formatter -> 'a t -> unit
(** Pretty print a proposition with type information *)
val pp_prop_with_typ : printenv -> Format.formatter -> normal t -> unit
(** Create a predicate environment for a prop *)
val prop_pred_env : 'a t -> Sil.Predicates.env
(** Dump a proposition. *)
val d_prop : 'a t -> unit
(** Dump a proposition with type information *)
val d_prop_with_typ : 'a t -> unit
(** Pretty print a list propositions with type information *)
val pp_proplist_with_typ : printenv -> Format.formatter -> normal t list -> unit
val d_proplist_with_typ : 'a t list -> unit
(** Compute free non-program variables of pi *)
val pi_fav : atom list -> fav
val pi_fav_add : fav -> atom list -> unit
(** Compute free non-program variables of sigma *)
val sigma_fav_add : fav -> hpred list -> unit
val sigma_fav : hpred list -> fav
(** returns free non-program variables that are used to express
the contents of stack variables *)
val sigma_fav_in_pvars_add : fav -> hpred list -> unit
(** Compute free non-program variables of prop *)
val prop_fav_add : fav -> 'a t -> unit
(** Compute free non-program variables of prop, visited in depth first order *)
val prop_fav_add_dfs : Tenv.t -> fav -> 'a t -> unit
val prop_fav: normal t -> fav
(** free vars, except pi and sub, of current and footprint parts *)
val prop_fav_nonpure : normal t -> fav
(** Find fav of the footprint part of the prop *)
val prop_footprint_fav : 'a t -> fav
(** Compute all the free program variables in the prop *)
val prop_fpv: 'a t -> Pvar.t list
(** Apply substitution for pi *)
val pi_sub : subst -> atom list -> atom list
(** Apply subsitution for sigma *)
val sigma_sub : subst -> hpred list -> hpred list
(** Apply subsitution to prop. Result is not normalized. *)
val prop_sub : subst -> 'a t -> exposed t
(** Apply the substitution to all the expressions in the prop. *)
val prop_expmap : (Exp.t -> Exp.t) -> 'a t -> exposed t
(** Relaces all expressions in the [hpred list] using the first argument.
Assume that the first parameter defines a partial function.
No expressions inside hpara are replaced. *)
val sigma_replace_exp : Tenv.t -> (Exp.t * Exp.t) list -> hpred list -> hpred list
(** {2 Normalization} *)
(** Turn an inequality expression into an atom *)
val mk_inequality : Tenv.t -> Exp.t -> Sil.atom
(** Return [true] if the atom is an inequality *)
val atom_is_inequality : Sil.atom -> bool
(** If the atom is [e<=n] return [e,n] *)
val atom_exp_le_const : Sil.atom -> (Exp.t * IntLit.t) option
(** If the atom is [n<e] return [n,e] *)
val atom_const_lt_exp : Sil.atom -> (IntLit.t * Exp.t) option
(** Normalize [exp] using the pure part of [prop]. Later, we should
change this such that the normalization exposes offsets of [exp]
as much as possible. *)
val exp_normalize_prop : Tenv.t -> 'a t -> Exp.t -> Exp.t
(** Normalize the expression without abstracting complex subexpressions *)
val exp_normalize_noabs : Tenv.t -> Sil.subst -> Exp.t -> Exp.t
(** Collapse consecutive indices that should be added. For instance,
this function reduces x[1][1] to x[2]. The [typ] argument is used
to ensure the soundness of this collapsing. *)
val exp_collapse_consecutive_indices_prop : Typ.t -> Exp.t -> Exp.t
(** Normalize [exp] used for the address of a heap cell.
This normalization does not combine two offsets inside [exp]. *)
val lexp_normalize_prop : Tenv.t -> 'a t -> Exp.t -> Exp.t
val atom_normalize_prop : Tenv.t -> 'a t -> atom -> atom
val strexp_normalize_prop : Tenv.t -> 'a t -> strexp -> strexp
val hpred_normalize_prop : Tenv.t -> 'a t -> hpred -> hpred
val sigma_normalize_prop : Tenv.t -> 'a t -> hpred list -> hpred list
val pi_normalize_prop : Tenv.t -> 'a t -> atom list -> atom list
(** normalize a prop *)
val normalize : Tenv.t -> exposed t -> normal t
(** expose a prop, no-op used to instantiate the sub-type relation *)
val expose : normal t -> exposed t
(** {2 Compaction} *)
(** Return a compact representation of the prop *)
val prop_compact : sharing_env -> normal t -> normal t
(** {2 Queries about propositions} *)
(** Check if the sigma part of the proposition is emp *)
val prop_is_emp : 'a t -> bool
(** {2 Functions for changing and generating propositions} *)
(** Construct a disequality. *)
val mk_neq : Tenv.t -> Exp.t -> Exp.t -> atom
(** Construct an equality. *)
val mk_eq : Tenv.t -> Exp.t -> Exp.t -> atom
(** Construct a positive pred. *)
val mk_pred : Tenv.t -> PredSymb.t -> Exp.t list -> atom
(** Construct a negative pred. *)
val mk_npred : Tenv.t -> PredSymb.t -> Exp.t list -> atom
(** create a strexp of the given type, populating the structures if [expand_structs] is true *)
val create_strexp_of_type :
Tenv.t -> struct_init_mode -> Typ.t -> Exp.t option -> Sil.inst -> Sil.strexp
(** Construct a pointsto. *)
val mk_ptsto : Tenv.t -> Exp.t -> strexp -> Exp.t -> hpred
(** Construct a points-to predicate for an expression using either the provided expression [name] as
base for fresh identifiers. *)
val mk_ptsto_exp : Tenv.t -> struct_init_mode -> Exp.t * Exp.t * Exp.t option -> Sil.inst -> hpred
(** Construct a points-to predicate for a single program variable.
If [expand_structs] is true, initialize the fields of structs with fresh variables. *)
val mk_ptsto_lvar : Tenv.t -> struct_init_mode -> Sil.inst -> Pvar.t * Exp.t * Exp.t option -> hpred
(** Construct a lseg predicate *)
val mk_lseg : Tenv.t -> lseg_kind -> hpara -> Exp.t -> Exp.t -> Exp.t list -> hpred
(** Construct a dllseg predicate *)
val mk_dllseg : Tenv.t -> lseg_kind -> hpara_dll -> Exp.t -> Exp.t -> Exp.t -> Exp.t -> Exp.t list -> hpred
(** Construct a hpara *)
val mk_hpara : Tenv.t -> Ident.t -> Ident.t -> Ident.t list -> Ident.t list -> hpred list -> hpara
(** Construct a dll_hpara *)
val mk_dll_hpara :
Tenv.t -> Ident.t -> Ident.t -> Ident.t -> Ident.t list -> Ident.t list -> hpred list -> hpara_dll
(** Proposition [true /\ emp]. *)
val prop_emp : normal t
(** Reset every inst in the prop using the given map *)
val prop_reset_inst : (Sil.inst -> Sil.inst) -> 'a t -> exposed t
(** Conjoin a heap predicate by separating conjunction. *)
val prop_hpred_star : 'a t -> hpred -> exposed t
(** Conjoin a list of heap predicates by separating conjunction *)
val prop_sigma_star : 'a t -> hpred list -> exposed t
(** Conjoin a pure atomic predicate by normal conjunction. *)
val prop_atom_and : Tenv.t -> ?footprint: bool -> normal t -> atom -> normal t
(** Conjoin [exp1]=[exp2] with a symbolic heap [prop]. *)
val conjoin_eq : Tenv.t -> ?footprint: bool -> Exp.t -> Exp.t -> normal t -> normal t
(** Conjoin [exp1]!=[exp2] with a symbolic heap [prop]. *)
val conjoin_neq : Tenv.t -> ?footprint: bool -> Exp.t -> Exp.t -> normal t -> normal t
(** Return the pure part of [prop]. *)
val get_pure : 'a t -> atom list
(** Canonicalize the names of primed variables. *)
val prop_rename_primed_footprint_vars : Tenv.t -> normal t -> normal t
(** Extract the footprint and return it as a prop *)
val extract_footprint : 'a t -> exposed t
(** Extract the (footprint,current) pair *)
val extract_spec : normal t -> (normal t * normal t)
(** [prop_set_fooprint p p_foot] sets proposition [p_foot] as footprint of [p]. *)
val prop_set_footprint : 'a t -> 'b t -> exposed t
(** Expand PE listsegs if the flag is on. *)
val prop_expand : Tenv.t -> normal t -> normal t list
(** {2 Functions for existentially quantifying and unquantifying variables} *)
(** Existentially quantify the [ids] in [prop]. *)
val exist_quantify : Tenv.t -> fav -> normal t -> normal t
(** convert the footprint vars to primed vars. *)
val prop_normal_vars_to_primed_vars : Tenv.t -> normal t -> normal t
(** convert the primed vars to normal vars. *)
val prop_primed_vars_to_normal_vars : Tenv.t -> normal t -> normal t
(** Build an exposed prop from pi *)
val from_pi : pi -> exposed t
(** Build an exposed prop from sigma *)
val from_sigma : sigma -> exposed t
(** Set individual fields of the prop. *)
val set : ?sub:Sil.subst -> ?pi:pi -> ?sigma:sigma -> ?pi_fp:pi -> ?sigma_fp:sigma ->
'a t -> exposed t
(** Rename free variables in a prop replacing them with existentially quantified vars *)
val prop_rename_fav_with_existentials : Tenv.t -> normal t -> normal t
(** Removes seeds variables from a prop corresponding to captured variables in an objc block *)
val remove_seed_captured_vars_block: Tenv.t -> Mangled.t list -> normal t -> normal t
(** {2 Prop iterators} *)
(** Iterator over the sigma part. Each iterator has a current [hpred]. *)
type 'a prop_iter
(** Create an iterator, return None if sigma part is empty. *)
val prop_iter_create : normal t -> unit prop_iter option
(** Return the prop associated to the iterator. *)
val prop_iter_to_prop : Tenv.t -> 'a prop_iter -> normal t
(** Add an atom to the pi part of prop iter. The
first parameter records whether it is done
during footprint or during re - execution. *)
val prop_iter_add_atom : bool -> 'a prop_iter -> atom -> 'a prop_iter
(** Remove the current element from the iterator, and return the prop
associated to the resulting iterator. *)
val prop_iter_remove_curr_then_to_prop : Tenv.t -> 'a prop_iter -> normal t
(** Return the current hpred and state. *)
val prop_iter_current : Tenv.t -> 'a prop_iter -> (hpred * 'a)
(** Return the next iterator. *)
val prop_iter_next : 'a prop_iter -> unit prop_iter option
(** Remove the current hpred and return the next iterator. *)
val prop_iter_remove_curr_then_next : 'a prop_iter -> unit prop_iter option
(** Update the current element of the iterator. *)
val prop_iter_update_current : 'a prop_iter -> hpred -> 'a prop_iter
(** Insert before the current element of the iterator. *)
val prop_iter_prev_then_insert : 'a prop_iter -> hpred -> 'a prop_iter
(** Find fav of the footprint part of the iterator *)
val prop_iter_footprint_fav : 'a prop_iter -> fav
(** Find fav of the iterator *)
val prop_iter_fav : 'a prop_iter -> fav
(** Extract the sigma part of the footprint *)
val prop_iter_get_footprint_sigma : 'a prop_iter -> hpred list
(** Replace the sigma part of the footprint *)
val prop_iter_replace_footprint_sigma : 'a prop_iter -> hpred list -> 'a prop_iter
(** Scan sigma to find an [hpred] satisfying the filter function. *)
val prop_iter_find : unit prop_iter -> (hpred -> 'a option) -> 'a prop_iter option
(** Update the current element of the iterator by a nonempty list of elements. *)
val prop_iter_update_current_by_list : 'a prop_iter -> hpred list -> unit prop_iter
(** Set the state of an iterator *)
val prop_iter_set_state : 'a prop_iter -> 'b -> 'b prop_iter
(** Rename [ident] in [iter] by a fresh primed identifier *)
val prop_iter_make_id_primed : Tenv.t -> Ident.t -> 'a prop_iter -> 'a prop_iter
(** Collect garbage fields. *)
val prop_iter_gc_fields : unit prop_iter -> unit prop_iter
(** return the set of subexpressions of [strexp] *)
val strexp_get_exps : Sil.strexp -> Exp.Set.t
(** get the set of expressions on the righthand side of [hpred] *)
val hpred_get_targets : Sil.hpred -> Exp.Set.t
(** return the set of hpred's and exp's in [sigma] that are reachable from an expression in
[exps] *)
val compute_reachable_hpreds : hpred list -> Exp.Set.t -> Sil.HpredSet.t * Exp.Set.t
(** {2 Internal modules} *)
module Metrics : sig
(** Compute a size value for the prop, which indicates its complexity *)
val prop_size : 'a t -> int
(** Approximate the size of the longest chain by counting the max
number of |-> with the same type and whose lhs is primed or
footprint *)
val prop_chain_size : 'a t -> int
end
module CategorizePreconditions : sig
type pre_category =
(* no preconditions *)
| NoPres
(* the preconditions impose no restrictions *)
| Empty
(* the preconditions only demand that some pointers are allocated *)
| OnlyAllocation
(* the preconditions impose constraints on the values of variables and/or memory *)
| DataConstraints
(** categorize a list of preconditions *)
val categorize : 'a t list -> pre_category
end