|
|
|
(*
|
|
|
|
* 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.
|
|
|
|
*)
|
|
|
|
|
|
|
|
(** Utility methods to support the translation of clang ast constructs into sil instructions. *)
|
|
|
|
|
|
|
|
type continuation = {
|
|
|
|
break: Cfg.Node.t list;
|
|
|
|
continue: Cfg.Node.t list;
|
|
|
|
return_temp : bool; (* true if temps should not be removed in the node but returned to ancestors *)
|
|
|
|
}
|
|
|
|
|
|
|
|
type priority_node =
|
|
|
|
| Free
|
|
|
|
| Busy of string
|
|
|
|
|
|
|
|
type trans_state = {
|
|
|
|
context: CContext.t;
|
|
|
|
succ_nodes: Cfg.Node.t list;
|
|
|
|
continuation: continuation option;
|
|
|
|
parent_line_number: int;
|
|
|
|
priority: priority_node;
|
|
|
|
}
|
|
|
|
|
|
|
|
type trans_result = {
|
|
|
|
root_nodes: Cfg.Node.t list;
|
|
|
|
leaf_nodes: Cfg.Node.t list;
|
|
|
|
ids: Ident.t list;
|
|
|
|
instrs: Sil.instr list;
|
|
|
|
exps: (Sil.exp * Sil.typ) list;
|
|
|
|
}
|
|
|
|
|
|
|
|
val empty_res_trans: trans_result
|
|
|
|
|
|
|
|
(** Collect the results of translating a list of instructions, and link up the nodes created. *)
|
|
|
|
val collect_res_trans: trans_result list -> trans_result
|
|
|
|
|
|
|
|
val is_return_temp: continuation option -> bool
|
|
|
|
|
|
|
|
val ids_to_parent: continuation option -> Ident.t list -> Ident.t list
|
|
|
|
|
|
|
|
val ids_to_node: continuation option -> Ident.t list -> Ident.t list
|
|
|
|
|
|
|
|
val mk_cond_continuation : continuation option -> continuation option
|
|
|
|
|
|
|
|
val extract_item_from_singleton : 'a list -> string -> 'a -> 'a
|
|
|
|
|
|
|
|
val extract_exp_from_list : (Sil.exp * Sil.typ) list -> string -> (Sil.exp * Sil.typ)
|
|
|
|
|
|
|
|
val fix_param_exps_mismatch : 'a list -> (Sil.exp * Sil.typ) list -> (Sil.exp * Sil.typ)list
|
|
|
|
|
|
|
|
val get_selector_receiver : Clang_ast_t.obj_c_message_expr_info -> string * Clang_ast_t.receiver_kind
|
|
|
|
|
|
|
|
val define_condition_side_effects :
|
|
|
|
CContext.t -> (Sil.exp * Sil.typ) list -> Sil.instr list -> Location.t ->
|
|
|
|
(Sil.exp * Sil.typ) list * Sil.instr list
|
|
|
|
|
|
|
|
val extract_stmt_from_singleton : Clang_ast_t.stmt list -> string -> Clang_ast_t.stmt
|
|
|
|
|
|
|
|
val is_null_stmt : Clang_ast_t.stmt -> bool
|
|
|
|
|
|
|
|
val compute_instr_ids_exp_to_parent :
|
|
|
|
Clang_ast_t.stmt_info -> Sil.instr list -> Ident.t list -> (Sil.exp * Sil.typ) list ->
|
|
|
|
Sil.exp -> Sil.typ -> Location.t -> priority_node ->
|
|
|
|
Sil.instr list * Ident.t list * (Sil.exp * Sil.typ) list
|
|
|
|
|
|
|
|
val get_name_decl_ref_exp_info : Clang_ast_t.decl_ref_expr_info -> Clang_ast_t.stmt_info -> string
|
|
|
|
|
|
|
|
val get_type_decl_ref_exp_info : Clang_ast_t.decl_ref_expr_info -> Clang_ast_t.type_ptr
|
|
|
|
|
|
|
|
val get_decl_kind : Clang_ast_t.decl_ref_expr_info -> Clang_ast_t.decl_kind
|
|
|
|
|
|
|
|
val get_decl_pointer : Clang_ast_t.decl_ref_expr_info -> Clang_ast_t.pointer
|
|
|
|
|
|
|
|
val is_enumeration_constant : Clang_ast_t.stmt -> bool
|
|
|
|
|
|
|
|
val is_member_exp : Clang_ast_t.stmt -> bool
|
|
|
|
|
|
|
|
val get_type_from_exp_stmt : Clang_ast_t.stmt -> Clang_ast_t.type_ptr
|
|
|
|
|
|
|
|
(** Given trans_result with ONE expression, create temporary variable with *)
|
|
|
|
(** dereferenced value of an expression assigned to it *)
|
|
|
|
val dereference_value_from_result : Location.t -> trans_result -> trans_result
|
|
|
|
|
|
|
|
val cast_operation :
|
|
|
|
CContext.t -> Clang_ast_t.cast_kind -> (Sil.exp * Sil.typ) list -> Sil.typ -> Location.t ->
|
|
|
|
bool -> Ident.t list * Sil.instr list * Sil.exp
|
|
|
|
|
|
|
|
val trans_assertion_failure : Location.t -> CContext.t -> trans_result
|
|
|
|
|
|
|
|
val trans_assume_false : Location.t -> CContext.t -> Cfg.Node.t list -> trans_result
|
|
|
|
|
|
|
|
val is_owning_method : Clang_ast_t.stmt -> bool
|
|
|
|
|
|
|
|
val is_owning_name : string -> bool
|
|
|
|
|
|
|
|
val is_method_call : Clang_ast_t.stmt -> bool
|
|
|
|
|
|
|
|
val contains_opaque_value_expr : Clang_ast_t.stmt -> bool
|
|
|
|
|
|
|
|
val get_decl_ref_info : Clang_ast_t.stmt -> int -> Clang_ast_t.decl_ref * int
|
|
|
|
|
|
|
|
val builtin_trans : trans_state -> Location.t -> Clang_ast_t.stmt_info ->
|
|
|
|
Sil.typ -> Procname.t option -> trans_result option
|
|
|
|
|
|
|
|
val alloc_trans :
|
|
|
|
trans_state -> Location.t -> Clang_ast_t.stmt_info -> Sil.typ -> bool -> trans_result
|
|
|
|
|
|
|
|
val new_or_alloc_trans :
|
|
|
|
trans_state -> Location.t -> Clang_ast_t.stmt_info -> string -> string -> trans_result
|
|
|
|
|
|
|
|
val cpp_new_trans : trans_state -> Location.t -> Clang_ast_t.stmt_info -> Sil.typ -> trans_result
|
|
|
|
|
|
|
|
val cast_trans :
|
|
|
|
CContext.t -> (Sil.exp * Sil.typ) list -> Location.t -> Procname.t option -> Sil.typ ->
|
|
|
|
(Ident.t * Sil.instr * Sil.exp) option
|
|
|
|
|
|
|
|
(** Module for creating cfg nodes and other utility functions related to them. *)
|
|
|
|
module Nodes :
|
|
|
|
sig
|
|
|
|
val is_binary_assign_op : Clang_ast_t.binary_operator_info -> bool
|
|
|
|
|
|
|
|
val need_unary_op_node : Clang_ast_t.unary_operator_info -> bool
|
|
|
|
|
|
|
|
val create_node : Cfg.Node.nodekind -> Ident.t list -> Sil.instr list ->
|
|
|
|
Location.t -> CContext.t -> Cfg.Node.t
|
|
|
|
|
|
|
|
val is_join_node : Cfg.Node.t -> bool
|
|
|
|
|
|
|
|
val create_prune_node :
|
|
|
|
bool -> (Sil.exp * Sil.typ) list -> Ident.t list -> Sil.instr list -> Location.t ->
|
|
|
|
Sil.if_kind -> CContext.t -> Cfg.Node.t
|
|
|
|
|
|
|
|
val is_prune_node : Cfg.Node.t -> bool
|
|
|
|
|
|
|
|
val is_true_prune_node : Cfg.Node.t -> bool
|
|
|
|
|
|
|
|
val prune_kind : bool -> Cfg.Node.nodekind
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
(** priority_node is used to enforce some kind of policy for creating nodes *)
|
|
|
|
(** in the cfg. Certain elements of the AST _must_ create nodes therefore *)
|
|
|
|
(** there is no need for them to use priority_node. Certain elements *)
|
|
|
|
(** instead need or need not to create a node depending of certain factors. *)
|
|
|
|
(** When an element of the latter kind wants to create a node it must claim *)
|
|
|
|
(** priority first (like taking a lock). priority can be claimes only when *)
|
|
|
|
(** it is free. If an element of AST succedes in claiming priority its id *)
|
|
|
|
(** (pointer) is recorded in priority. After an element has finished it *)
|
|
|
|
(** frees the priority. In general an AST element E checks if an ancestor *)
|
|
|
|
(** has claimed priority. If priority is already claimed E does not have to *)
|
|
|
|
(** create a node. If priority is free then it means E has to create the *)
|
|
|
|
(** node. Then E claims priority and release it afterward. *)
|
|
|
|
module PriorityNode :
|
|
|
|
sig
|
|
|
|
|
|
|
|
type t = priority_node
|
|
|
|
|
|
|
|
val is_priority_free : trans_state -> bool
|
|
|
|
|
|
|
|
val try_claim_priority_node : trans_state -> Clang_ast_t.stmt_info -> trans_state
|
|
|
|
|
|
|
|
val own_priority_node : t -> Clang_ast_t.stmt_info -> bool
|
|
|
|
|
|
|
|
(* Used for function call and method call. It deals with creating or not *)
|
|
|
|
(* a cfg node depending of owning the priority_node and the nodes returned *)
|
|
|
|
(* by the parameters of the call *)
|
|
|
|
val compute_results_to_parent :
|
|
|
|
trans_state -> Location.t -> string -> Clang_ast_t.stmt_info -> trans_result -> trans_result
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
(** Module for translating goto instructions by keeping a map of labels. *)
|
|
|
|
module GotoLabel :
|
|
|
|
sig
|
|
|
|
val find_goto_label : CContext.t -> string -> Location.t -> Cfg.Node.t
|
|
|
|
|
|
|
|
val reset_all_labels : unit -> unit
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
(** Module that provides utility functions for translating different types of loops. *)
|
|
|
|
module Loops :
|
|
|
|
sig
|
|
|
|
type loop_kind =
|
|
|
|
| For of Clang_ast_t.stmt * Clang_ast_t.stmt * Clang_ast_t.stmt * Clang_ast_t.stmt
|
|
|
|
(* init, condition, increment and body *)
|
|
|
|
| While of Clang_ast_t.stmt * Clang_ast_t.stmt (* condition and body *)
|
|
|
|
| DoWhile of Clang_ast_t.stmt * Clang_ast_t.stmt (* condition and body *)
|
|
|
|
|
|
|
|
val loop_kind_to_if_kind : loop_kind -> Sil.if_kind
|
|
|
|
|
|
|
|
val get_cond : loop_kind -> Clang_ast_t.stmt
|
|
|
|
|
|
|
|
val get_body : loop_kind -> Clang_ast_t.stmt
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
(** This module handles the translation of the variable self which is challenging because self *)
|
|
|
|
(** is used both as a variable in instance method calls and also as a type in class method calls. *)
|
|
|
|
module Self :
|
|
|
|
sig
|
|
|
|
|
|
|
|
exception SelfClassException of string
|
|
|
|
|
|
|
|
val add_self_parameter_for_super_instance :
|
|
|
|
CContext.t -> Procname.t -> Location.t -> Clang_ast_t.obj_c_message_expr_info ->
|
|
|
|
trans_result -> trans_result
|
|
|
|
|
|
|
|
val is_var_self : Sil.pvar -> bool -> bool
|
|
|
|
end
|
|
|
|
|
|
|
|
val is_logical_negation_of_int : Sil.tenv -> Clang_ast_t.expr_info -> Clang_ast_t.unary_operator_info -> bool
|
|
|
|
|
|
|
|
val is_dispatch_function : Clang_ast_t.stmt list -> int option
|
|
|
|
|
|
|
|
val assign_default_params : Clang_ast_t.stmt list -> Clang_ast_t.stmt -> Clang_ast_t.stmt list
|
|
|
|
|
|
|
|
val is_block_enumerate_function : Clang_ast_t.obj_c_message_expr_info -> bool
|
|
|
|
|
|
|
|
val pointer_of_call_expr : Clang_ast_t.stmt -> Clang_ast_t.pointer option
|