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.
181 lines
7.1 KiB
181 lines
7.1 KiB
(*
|
|
* 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! IStd
|
|
|
|
(** Utility methods to support the translation of clang ast constructs into sil instructions. *)
|
|
|
|
type continuation =
|
|
{ break: Procdesc.Node.t list
|
|
; continue: Procdesc.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 Clang_ast_t.pointer
|
|
|
|
(** A translation state. It provides the translation function with the info it needs to carry on the
|
|
translation. *)
|
|
type trans_state =
|
|
{ context: CContext.t (** current context of the translation *)
|
|
; succ_nodes: Procdesc.Node.t list (** successor nodes in the cfg *)
|
|
; continuation: continuation option (** current continuation *)
|
|
; priority: priority_node
|
|
; var_exp_typ: (Exp.t * Typ.t) option
|
|
; opaque_exp: (Exp.t * Typ.t) option
|
|
; is_fst_arg_objc_method_call: bool }
|
|
|
|
(** A translation result. It is returned by the translation function. *)
|
|
type trans_result =
|
|
{ root_nodes: Procdesc.Node.t list (** Top cfg nodes (root) created by the translation *)
|
|
; leaf_nodes: Procdesc.Node.t list (** Bottom cfg nodes (leaf) created by the translate *)
|
|
; instrs: Sil.instr list
|
|
(** list of SIL instruction that need to be placed in cfg nodes of the parent*)
|
|
; exps: (Exp.t * Typ.t) list (** SIL expressions resulting from translation of clang stmt *)
|
|
; initd_exps: Exp.t list
|
|
; is_cpp_call_virtual: bool }
|
|
|
|
val empty_res_trans : trans_result
|
|
|
|
val default_trans_state : CContext.t -> trans_state
|
|
|
|
val undefined_expression : unit -> Exp.t
|
|
|
|
val collect_res_trans : Procdesc.t -> trans_result list -> trans_result
|
|
(** Collect the results of translating a list of instructions, and link up the nodes created. *)
|
|
|
|
val is_return_temp : continuation option -> bool
|
|
|
|
val mk_cond_continuation : continuation option -> continuation option
|
|
|
|
val extract_exp_from_list : (Exp.t * Typ.t) list -> string -> Exp.t * Typ.t
|
|
|
|
val define_condition_side_effects :
|
|
(Exp.t * Typ.t) list -> Sil.instr list -> Location.t -> (Exp.t * Typ.t) 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 dereference_value_from_result :
|
|
Location.t -> trans_result -> strip_pointer:bool -> trans_result
|
|
(** Given trans_result with ONE expression, create temporary variable with dereferenced value of an
|
|
expression assigned to it *)
|
|
|
|
val cast_operation :
|
|
Clang_ast_t.cast_kind -> (Exp.t * Typ.t) list -> Typ.t -> Location.t
|
|
-> Sil.instr list * (Exp.t * Typ.t)
|
|
|
|
val trans_assertion : trans_state -> Location.t -> trans_result
|
|
|
|
val contains_opaque_value_expr : Clang_ast_t.stmt -> bool
|
|
|
|
val builtin_trans :
|
|
trans_state -> Location.t -> trans_result list -> Typ.Procname.t -> trans_result option
|
|
|
|
val cxx_method_builtin_trans :
|
|
trans_state -> Location.t -> trans_result list -> Typ.Procname.t -> trans_result option
|
|
|
|
val new_or_alloc_trans :
|
|
trans_state -> Location.t -> Clang_ast_t.stmt_info -> Clang_ast_t.qual_type -> Typ.Name.t option
|
|
-> string -> trans_result
|
|
|
|
val cpp_new_trans : Location.t -> Typ.t -> Exp.t option -> (Exp.t * Typ.typ) list -> trans_result
|
|
|
|
(** 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 create_node :
|
|
Procdesc.Node.nodekind -> Sil.instr list -> Location.t -> CContext.t -> Procdesc.Node.t
|
|
|
|
val create_prune_node :
|
|
branch:bool -> negate_cond:bool -> (Exp.t * Typ.t) list -> Sil.instr list -> Location.t
|
|
-> Sil.if_kind -> CContext.t -> Procdesc.Node.t
|
|
|
|
val is_true_prune_node : Procdesc.Node.t -> bool
|
|
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 force_claim_priority_node : trans_state -> Clang_ast_t.stmt_info -> trans_state
|
|
|
|
val own_priority_node : t -> Clang_ast_t.stmt_info -> bool
|
|
|
|
(* Used by translation functions to handle potenatial cfg nodes. *)
|
|
(* It connects nodes returned by translation of stmt children and *)
|
|
(* deals with creating or not a cfg node depending of owning the *)
|
|
(* priority_node. It returns nodes, ids, instrs that should be passed to parent *)
|
|
|
|
val compute_results_to_parent :
|
|
trans_state -> Location.t -> string -> Clang_ast_t.stmt_info -> trans_result list
|
|
-> 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 -> Procdesc.Node.t
|
|
end
|
|
|
|
(** Module that provides utility functions for translating different types of loops. *)
|
|
module Loops : sig
|
|
type loop_kind =
|
|
| For of
|
|
{ init: Clang_ast_t.stmt
|
|
; decl_stmt: Clang_ast_t.stmt
|
|
; condition: Clang_ast_t.stmt
|
|
; increment: Clang_ast_t.stmt
|
|
; body: Clang_ast_t.stmt }
|
|
| While of {decl_stmt: Clang_ast_t.stmt; condition: Clang_ast_t.stmt; body: Clang_ast_t.stmt}
|
|
| DoWhile of {condition: Clang_ast_t.stmt; body: Clang_ast_t.stmt}
|
|
|
|
val get_cond : loop_kind -> Clang_ast_t.stmt
|
|
|
|
val get_body : loop_kind -> Clang_ast_t.stmt
|
|
end
|
|
|
|
(** Module that provides utilities for scopes *)
|
|
module Scope : sig
|
|
module StmtMap = ClangPointers.Map
|
|
|
|
val compute_vars_to_destroy : Clang_ast_t.stmt -> Clang_ast_t.decl list StmtMap.t
|
|
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
|
|
{ class_name: Typ.Name.t
|
|
; position: Logging.ocaml_pos
|
|
; source_range: Clang_ast_t.source_range }
|
|
|
|
val add_self_parameter_for_super_instance :
|
|
Clang_ast_t.stmt_info -> CContext.t -> Typ.Procname.t -> Location.t
|
|
-> Clang_ast_t.obj_c_message_expr_info -> trans_result
|
|
|
|
val is_var_self : Pvar.t -> bool -> bool
|
|
end
|
|
|
|
val is_logical_negation_of_int :
|
|
Tenv.t -> Clang_ast_t.expr_info -> Clang_ast_t.unary_operator_info -> bool
|