(* * Copyright (c) 2009-2013, Monoidics ltd. * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. *) (** The Smallfoot Intermediate Language: Predicate Symbols *) open! IStd module L = Logging module F = Format (** Visibility modifiers. *) type access = Default | Public | Private | Protected [@@deriving compare] let equal_access = [%compare.equal: access] let string_of_access = function | Default -> "Default" | Public -> "Public" | Private -> "Private" | Protected -> "Protected" type mem_kind = | Mmalloc (** memory allocated with malloc *) | Mnew (** memory allocated with new *) | Mnew_array (** memory allocated with new[] *) | Mobjc (** memory allocated with objective-c alloc *) [@@deriving compare] (** resource that can be allocated *) type resource = Rmemory of mem_kind | Rfile | Rignore | Rlock [@@deriving compare] (** kind of resource action *) type res_act_kind = Racquire | Rrelease [@@deriving compare] let equal_res_act_kind = [%compare.equal: res_act_kind] (** kind of dangling pointers *) type dangling_kind = | DAuninit (** pointer is dangling because it is uninitialized *) | DAaddr_stack_var (** pointer is dangling because it is the address of a stack variable which went out of scope *) | DAminusone (** pointer is -1 *) [@@deriving compare] (** position in a path: proc name, node id *) type path_pos = Procname.t * int [@@deriving compare] let equal_path_pos = [%compare.equal: path_pos] (** acquire/release action on a resource *) type res_action = { ra_kind: res_act_kind (** kind of action *) ; ra_res: resource (** kind of resource *) ; ra_pname: Procname.t (** name of the procedure used to acquire/release the resource *) ; ra_loc: Location.t (** location of the acquire/release *) ; ra_vpath: DecompiledExp.vpath (** vpath of the resource value *) } (* ignore other values beside resources: arbitrary merging into one *) let compare_res_action {ra_kind= k1; ra_res= r1} {ra_kind= k2; ra_res= r2} = [%compare: res_act_kind * resource] (k1, r1) (k2, r2) (* type aliases for components of t values that compare should ignore *) type annot_item_ = Annot.Item.t let compare_annot_item_ _ _ = 0 type location_ = Location.t let compare_location_ _ _ = 0 type path_pos_ = path_pos let compare_path_pos_ _ _ = 0 (** Attributes are nary function symbols that are applied to expression arguments in Apred and Anpred atomic formulas. Many operations don't make much sense for nullary predicates, and are generally treated as no-ops. The first argument is treated specially, as the "anchor" of the predicate application. For example, adding or removing an attribute uses the anchor to identify the atom to operate on. Also, abstraction and normalization operations treat the anchor specially and maintain more information on it than other arguments. Therefore when attaching an attribute to an expression, that expression should be the first argument, optionally followed by additional related expressions. *) type t = | Aresource of res_action (** resource acquire/release *) | Aautorelease | Adangling of dangling_kind (** dangling pointer *) | Aundef of Procname.t * annot_item_ * location_ * path_pos_ | Alocked | Aunlocked | Adiv0 of path_pos (** value appeared in second argument of division at given path position *) | Aobjc_null (** attributed exp is null due to a call to a method with given path as null receiver *) | Aretval of Procname.t * Annot.Item.t (** value was returned from a call to the given procedure, plus the annots of the return value *) | Aobserver (** denotes an object registered as an observers to a notification center *) | Aunsubscribed_observer (** denotes an object unsubscribed from observers of a notification center *) | Awont_leak (** value do not participate in memory leak analysis *) [@@deriving compare] let equal = [%compare.equal: t] (** name of the allocation function for the given memory kind *) let mem_alloc_pname = function | Mmalloc -> Procname.from_string_c_fun "malloc" | Mnew -> Procname.from_string_c_fun "new" | Mnew_array -> Procname.from_string_c_fun "new[]" | Mobjc -> Procname.from_string_c_fun "alloc" (** name of the deallocation function for the given memory kind *) let mem_dealloc_pname = function | Mmalloc -> Procname.from_string_c_fun "free" | Mnew -> Procname.from_string_c_fun "delete" | Mnew_array -> Procname.from_string_c_fun "delete[]" | Mobjc -> Procname.from_string_c_fun "dealloc" (** Categories of attributes *) type category = | ACresource | ACautorelease | AClock | ACdiv0 | ACobjc_null | ACundef | ACretval | ACobserver | ACwontleak [@@deriving compare] let equal_category = [%compare.equal: category] let to_category att = match att with | Aresource _ | Adangling _ -> ACresource | Alocked | Aunlocked -> AClock | Aautorelease -> ACautorelease | Adiv0 _ -> ACdiv0 | Aobjc_null -> ACobjc_null | Aretval _ -> ACretval | Aundef _ -> ACundef | Aobserver | Aunsubscribed_observer -> ACobserver | Awont_leak -> ACwontleak let is_undef = function Aundef _ -> true | _ -> false (** convert the attribute to a string *) let to_string pe = function | Aresource ra -> let mk_name = function | Mmalloc -> "ma" | Mnew -> "ne" | Mnew_array -> "na" | Mobjc -> "oc" in let name = match (ra.ra_kind, ra.ra_res) with | Racquire, Rmemory mk -> "MEM" ^ mk_name mk | Racquire, Rfile -> "FILE" | Rrelease, Rmemory mk -> "FREED" ^ mk_name mk | Rrelease, Rfile -> "CLOSED" | _, Rignore -> "IGNORE" | Racquire, Rlock -> "LOCKED" | Rrelease, Rlock -> "UNLOCKED" in let str_vpath = if Config.trace_error then F.asprintf "%a" (DecompiledExp.pp_vpath pe) ra.ra_vpath else "" in name ^ Binop.str pe Lt ^ Procname.to_string ra.ra_pname ^ ":" ^ string_of_int ra.ra_loc.Location.line ^ Binop.str pe Gt ^ str_vpath | Aautorelease -> "AUTORELEASE" | Adangling dk -> let dks = match dk with | DAuninit -> "UNINIT" | DAaddr_stack_var -> "ADDR_STACK" | DAminusone -> "MINUS1" in "DANGL" ^ Binop.str pe Lt ^ dks ^ Binop.str pe Gt | Aundef (pn, _, loc, _) -> "UND" ^ Binop.str pe Lt ^ Procname.to_string pn ^ Binop.str pe Gt ^ ":" ^ string_of_int loc.Location.line | Alocked -> "LOCKED" | Aunlocked -> "UNLOCKED" | Adiv0 (_, _) -> "DIV0" | Aobjc_null -> "OBJC_NULL" | Aretval (pn, _) -> "RET" ^ Binop.str pe Lt ^ Procname.to_string pn ^ Binop.str pe Gt | Aobserver -> "OBSERVER" | Aunsubscribed_observer -> "UNSUBSCRIBED_OBSERVER" | Awont_leak -> "WONT_LEAK" let pp pe fmt a = F.pp_print_string fmt (to_string pe a) (** dump an attribute *) let d_attribute (a : t) = L.d_pp_with_pe pp a