(* * 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. *) open Utils open CFrontend_utils open CContext open Clang_ast_t module L = Logging (* Extract the element of a singleton list. If the list is not a singleton *) (* It stops the computation giving a warning. We use this because we *) (* assume in many places that a list is just a singleton. We use the *) (* warning if to see which assumption was not correct *) let extract_item_from_singleton l warning_string failure_val = match l with | [item] -> item | _ -> Printing.log_err "%s" warning_string; failure_val let dummy_exp = (Sil.exp_minus_one, Sil.Tint Sil.IInt) (* Extract the element of a singleton list. If the list is not a singleton *) (* Gives a warning and return -1 as standard value indicating something *) (* went wrong. *) let extract_exp_from_list el warning_string = extract_item_from_singleton el warning_string dummy_exp module Nodes = struct let prune_kind b = Cfg.Node.Prune_node(b, Sil.Ik_bexp , ((string_of_bool b)^" Branch")) let is_join_node n = match Cfg.Node.get_kind n with | Cfg.Node.Join_node -> true | _ -> false let is_prune_node n = match Cfg.Node.get_kind n with | Cfg.Node.Prune_node _ -> true | _ -> false let is_true_prune_node n = match Cfg.Node.get_kind n with | Cfg.Node.Prune_node(true, _, _) -> true | _ -> false let create_node node_kind temps instrs loc context = let procdesc = CContext.get_procdesc context in Cfg.Node.create (CContext.get_cfg context) loc node_kind instrs procdesc temps let create_prune_node branch e_cond ids_cond instrs_cond loc ik context = let (e_cond', _) = extract_exp_from_list e_cond "\nWARNING: Missing expression for Conditional operator. Need to be fixed" in let e_cond'' = if branch then Sil.BinOp(Sil.Ne, e_cond', Sil.exp_zero) else Sil.BinOp(Sil.Eq, e_cond', Sil.exp_zero) in let instrs_cond'= instrs_cond @ [Sil.Prune(e_cond'', loc, branch, ik)] in create_node (prune_kind branch) ids_cond instrs_cond' loc context (** Check if this binary opertor requires the creation of a node in the cfg. *) let is_binary_assign_op boi = match boi.Clang_ast_t.boi_kind with | `Assign | `MulAssign | `DivAssign | `RemAssign | `AddAssign | `SubAssign | `ShlAssign | `ShrAssign | `AndAssign | `XorAssign | `OrAssign -> true | `PtrMemD | `PtrMemI | `Mul | `Div | `Rem | `Add | `Sub | `Shl | `Shr | `LT | `GT | `LE | `GE | `EQ | `NE | `And | `Xor | `Or | `LAnd | `LOr | `Comma -> false (** Check if this unary opertor requires the creation of a node in the cfg. *) let need_unary_op_node uoi = match uoi.Clang_ast_t.uoi_kind with | `PostInc | `PostDec | `PreInc | `PreDec | `AddrOf | `Deref | `Plus -> true | `Minus | `Not | `LNot | `Real | `Imag | `Extension -> false end type str_node_map = (string, Cfg.Node.t) Hashtbl.t module GotoLabel = struct (* stores goto labels local to a function, with the relative node in the cfg *) let goto_label_node_map : str_node_map = Hashtbl.create 17 let reset_all_labels () = Hashtbl.reset goto_label_node_map let find_goto_label context label sil_loc = try Hashtbl.find goto_label_node_map label with Not_found -> let node_name = Format.sprintf "GotoLabel_%s" label in let new_node = Nodes.create_node (Cfg.Node.Skip_node node_name) [] [] sil_loc context in Hashtbl.add goto_label_node_map label new_node; new_node end 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 *) } let is_return_temp continuation = match continuation with | Some cont -> cont.return_temp | _ -> false let ids_to_parent cont ids = if is_return_temp cont then ids else [] let ids_to_node cont ids = if is_return_temp cont then [] else ids let mk_cond_continuation cont = match cont with | Some cont' -> Some { cont' with return_temp = true; } | None -> Some { break =[]; continue =[]; return_temp = true;} type priority_node = | Free | Busy of string (* A translation state. It provides the translation function with the info*) (* it need to carry on the tranlsation. *) type trans_state = { context: CContext.t; (* current context of the translation *) succ_nodes: Cfg.Node.t list; (* successor nodes in the cfg *) continuation: continuation option; (* current continuation *) parent_line_number: int; (* line numbeer of the parent element in the AST *) priority: priority_node; } (* A translation result. It is returned by the translation function. *) type trans_result = { root_nodes: Cfg.Node.t list; (* Top cfg nodes (root) created by the translation *) leaf_nodes: Cfg.Node.t list; (* Bottom cfg nodes (leaf) created by the translate *) ids: Ident.t list; (* list of temp identifiers created that need to be removed by the caller *) instrs: Sil.instr list; (* list of SIL instruction that need to be placed in cfg nodes of the parent*) exps: (Sil.exp * Sil.typ) list; (* SIL expressions resulting from the translation of the clang stmt *) } (* Empty result translation *) let empty_res_trans = { root_nodes =[]; leaf_nodes =[]; ids =[]; instrs =[]; exps =[]} (** Collect the results of translating a list of instructions, and link up the nodes created. *) let collect_res_trans l = let rec collect l rt = match l with | [] -> rt | rt':: l' -> let root_nodes = if rt.root_nodes <> [] then rt.root_nodes else rt'.root_nodes in let leaf_nodes = if rt'.leaf_nodes <> [] then rt'.leaf_nodes else rt.leaf_nodes in if rt'.root_nodes <> [] then list_iter (fun n -> Cfg.Node.set_succs_exn n rt'.root_nodes []) rt.leaf_nodes; collect l' { root_nodes = root_nodes; leaf_nodes = leaf_nodes; ids = rt.ids@rt'.ids; instrs = rt.instrs@rt'.instrs; exps = rt.exps@rt'.exps } in collect l empty_res_trans (* 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 = struct type t = priority_node let try_claim_priority_node trans_state stmt_info = match trans_state.priority with | Free -> { trans_state with priority = Busy stmt_info.Clang_ast_t.si_pointer } | _ -> trans_state let is_priority_free trans_state = match trans_state.priority with | Free -> true | _ -> false let own_priority_node pri stmt_info = match pri with | Busy p when p = stmt_info.Clang_ast_t.si_pointer -> true | _ -> false (* 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, ids, instrs returned *) (* by the parameters of the call *) let compute_results_to_parent trans_state loc nd_name stmt_info res_state_param = let mk_node () = let ids_node = ids_to_node trans_state.continuation res_state_param.ids in let node_kind = Cfg.Node.Stmt_node (nd_name) in Nodes.create_node node_kind ids_node res_state_param.instrs loc trans_state.context in (* Invariant: if leaf_nodes is empty then the params have not created a node.*) match res_state_param.leaf_nodes, own_priority_node trans_state.priority stmt_info with | _, false -> (* The node is created by the parent. We just pass back nodes/leafs params *) { res_state_param with exps = []} | [], true -> (* We need to create a node and params did not create a node.*) let node' = mk_node () in let ids_parent = ids_to_parent trans_state.continuation res_state_param.ids in Cfg.Node.set_succs_exn node' trans_state.succ_nodes []; { root_nodes =[node']; leaf_nodes =[node']; ids = ids_parent; instrs =[]; exps = []} | _, true -> (* We need to create a node but params also created some,*) (* so we need to pass back the nodes/leafs params*) let node' = mk_node () in Cfg.Node.set_succs_exn node' trans_state.succ_nodes []; let ids_parent = ids_to_parent trans_state.continuation res_state_param.ids in list_iter (fun n' -> Cfg.Node.set_succs_exn n' [node'] []) res_state_param.leaf_nodes; { root_nodes = res_state_param.root_nodes; leaf_nodes = [node']; ids = ids_parent; instrs =[]; exps =[]} end module Loops = struct 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 *) let loop_kind_to_if_kind loop_kind = match loop_kind with | For _ -> Sil.Ik_for | While _ -> Sil.Ik_while | DoWhile _ -> Sil.Ik_dowhile let get_body loop_kind = match loop_kind with | For (_, _, _, body) | While (_, body) | DoWhile (_, body) -> body let get_cond loop_kind = match loop_kind with | For (_, cond, _, _) | While (cond, _) | DoWhile (cond, _) -> cond end let create_alloc_instrs context sil_loc function_type is_cf_non_null_alloc = let fname = if is_cf_non_null_alloc then SymExec.ModelBuiltins.__objc_alloc_no_fail else SymExec.ModelBuiltins.__objc_alloc in let function_type, function_type_np = match function_type with | Sil.Tptr (styp, Sil.Pk_pointer) | Sil.Tptr (styp, Sil.Pk_objc_weak) | Sil.Tptr (styp, Sil.Pk_objc_unsafe_unretained) | Sil.Tptr (styp, Sil.Pk_objc_autoreleasing) -> function_type, CTypes_decl.expand_structured_type context.tenv styp | _ -> Sil.Tptr (function_type, Sil.Pk_pointer), function_type in let sizeof_exp = Sil.Sizeof (function_type_np, Sil.Subtype.exact) in let exp = (sizeof_exp, function_type) in let ret_id = Ident.create_fresh Ident.knormal in let stmt_call = Sil.Call([ret_id], (Sil.Const (Sil.Cfun fname)), [exp], sil_loc, Sil.cf_default) in (function_type, ret_id, stmt_call, Sil.Var ret_id) let alloc_trans trans_state loc stmt_info function_type is_cf_non_null_alloc = let (function_type, ret_id, stmt_call, exp) = create_alloc_instrs trans_state.context loc function_type is_cf_non_null_alloc in let res_trans_tmp = { empty_res_trans with ids =[ret_id]; instrs =[stmt_call]} in let res_trans = PriorityNode.compute_results_to_parent trans_state loc "Call alloc" stmt_info res_trans_tmp in { res_trans with exps =[(exp, function_type)]} let new_trans trans_state loc stmt_info cls_name function_type = let (alloc_ret_type, alloc_ret_id, alloc_stmt_call, alloc_exp) = create_alloc_instrs trans_state.context loc function_type true in let init_ret_id = Ident.create_fresh Ident.knormal in let is_instance = true in let call_flags = { Sil.cf_virtual = is_instance; Sil.cf_noreturn = false; Sil.cf_is_objc_block = false; } in let pname = CMethod_trans.mk_procname_from_method cls_name CFrontend_config.init in CMethod_trans.create_external_procdesc trans_state.context.cfg pname is_instance None; let args = [(Sil.Var alloc_ret_id, alloc_ret_type)] in let init_stmt_call = Sil.Call([init_ret_id], (Sil.Const (Sil.Cfun pname)), args, loc, call_flags) in let instrs = [alloc_stmt_call; init_stmt_call] in let ids = [alloc_ret_id; init_ret_id] in let res_trans_tmp = { empty_res_trans with ids = ids; instrs = instrs } in let res_trans = PriorityNode.compute_results_to_parent trans_state loc "Call new" stmt_info res_trans_tmp in { res_trans with exps = [(Sil.Var init_ret_id, alloc_ret_type)]} let new_or_alloc_trans trans_state loc stmt_info class_name selector = let function_type = CTypes_decl.type_name_to_sil_type trans_state.context.tenv class_name in if selector = CFrontend_config.alloc then alloc_trans trans_state loc stmt_info function_type true else if selector = CFrontend_config.new_str then new_trans trans_state loc stmt_info class_name function_type else assert false let create_cast_instrs context exp cast_from_typ cast_to_typ sil_loc = let ret_id = Ident.create_fresh Ident.knormal in let cast_typ_no_pointer = CTypes_decl.expand_structured_type context.tenv (CTypes.remove_pointer_to_typ cast_to_typ) in let sizeof_exp = Sil.Sizeof (cast_typ_no_pointer, Sil.Subtype.exact) in let pname = SymExec.ModelBuiltins.__objc_cast in let args = [(exp, cast_from_typ); (sizeof_exp, Sil.Tvoid)] in let stmt_call = Sil.Call([ret_id], (Sil.Const (Sil.Cfun pname)), args, sil_loc, Sil.cf_default) in (ret_id, stmt_call, Sil.Var ret_id) let cast_trans context exps sil_loc callee_pname_opt function_type = if CTrans_models.is_toll_free_bridging callee_pname_opt then match exps with | [exp, typ] -> Some (create_cast_instrs context exp typ function_type sil_loc) | _ -> assert false else None let builtin_trans trans_state loc stmt_info function_type callee_pname_opt = if CTrans_models.is_cf_non_null_alloc callee_pname_opt || CTrans_models.is_alloc_model function_type callee_pname_opt then Some (alloc_trans trans_state loc stmt_info function_type true) else if CTrans_models.is_alloc callee_pname_opt then Some (alloc_trans trans_state loc stmt_info function_type false) else None let cast_operation context cast_kind exps cast_typ sil_loc is_objc_bridged = let (exp, typ) = extract_exp_from_list exps "" in if is_objc_bridged then let id, instr, exp = create_cast_instrs context exp typ cast_typ sil_loc in [id], [instr], exp else match cast_kind with | `NoOp | `BitCast | `IntegralCast | `IntegralToBoolean -> (* This is treated as a nop by returning the same expressions exps*) ([],[], exp) | `LValueToRValue -> (* Takes an LValue and allow it to use it as RValue. *) (* So we assign the LValue to a temp and we pass it to the parent.*) let id = Ident.create_fresh Ident.knormal in let sil_instr = [Sil.Letderef (id, exp, typ, sil_loc)] in ([id], sil_instr, Sil.Var id) | `CPointerToObjCPointerCast -> ([], [], Sil.Cast(typ, exp)) | _ -> Printing.log_err "\nWARNING: Missing translation for Cast Kind %s. The construct has been ignored...\n" (Clang_ast_j.string_of_cast_kind cast_kind); ([],[], exp) let trans_assertion_failure sil_loc context = let assert_fail_builtin = Sil.Const (Sil.Cfun SymExec.ModelBuiltins.__infer_fail) in let args = [Sil.Const (Sil.Cstr Config.default_failure_name), Sil.Tvoid] in let call_instr = Sil.Call ([], assert_fail_builtin, args, sil_loc, Sil.cf_default) in let exit_node = Cfg.Procdesc.get_exit_node (CContext.get_procdesc context) and failure_node = Nodes.create_node (Cfg.Node.Stmt_node "Assertion failure") [] [call_instr] sil_loc context in Cfg.Node.set_succs_exn failure_node [exit_node] []; { root_nodes = [failure_node]; leaf_nodes = [failure_node]; ids = []; instrs =[]; exps = [] } let trans_assume_false sil_loc context succ_nodes = let instrs_cond = [Sil.Prune (Sil.exp_zero, sil_loc, true, Sil.Ik_land_lor)] in let prune_node = Nodes.create_node (Nodes.prune_kind true) [] instrs_cond sil_loc context in Cfg.Node.set_succs_exn prune_node succ_nodes []; { root_nodes = [prune_node]; leaf_nodes = [prune_node]; ids = []; instrs = []; exps = [] } let define_condition_side_effects context e_cond instrs_cond sil_loc = let (e', typ) = extract_exp_from_list e_cond "\nWARNING: Missing expression in IfStmt. Need to be fixed\n" in match e' with | Sil.Lvar pvar -> let id = Ident.create_fresh Ident.knormal in [(Sil.Var id, typ)], [Sil.Letderef (id, Sil.Lvar pvar, typ, sil_loc)] | _ -> [(e', typ)], instrs_cond (* Given a list of instuctions, ids, an expression, lhs of an compound *) (* assignment its type and loc it computes which instructions, ids, and *) (* expression need to be returned to the AST's parent node. This function *) (* is used by a compount assignment. The expression e is the result of *) (* translating the rhs of the assignment *) let compute_instr_ids_exp_to_parent stmt_info instr ids e lhs typ loc pri = if PriorityNode.own_priority_node pri stmt_info then( (* The current AST element has created a node then instr and ids have *) (* been already included in the node. *) [], [], e ) else ( (* The node will be created by the parent. We pass the instr and ids. *) (* For the expression we need to save the constend of the lhs in a new *) (* temp so that can be used by the parent node (for example: x=(y=10)) *) let id = Ident.create_fresh Ident.knormal in let res_instr = [Sil.Letderef (id, lhs, typ, loc)] in instr@res_instr, ids @ [id], [(Sil.Var id, typ)]) let fix_param_exps_mismatch params_stmt exps_param = let diff = list_length params_stmt - list_length exps_param in let args = if diff >0 then Array.make diff dummy_exp else assert false in let exps'= exps_param @ (Array.to_list args) in exps' let get_name_decl_ref_exp_info decl_ref_expr_info si = match decl_ref_expr_info.Clang_ast_t.drti_decl_ref with | Some d -> (match d.Clang_ast_t.dr_name with | Some n -> n.Clang_ast_t.ni_name | _ -> assert false) | _ -> L.err "FAILING WITH %s pointer=%s@.@." (Clang_ast_j.string_of_decl_ref_expr_info decl_ref_expr_info ) (Clang_ast_j.string_of_stmt_info si); assert false let is_superinstance mei = match mei.Clang_ast_t.omei_receiver_kind with | `SuperInstance -> true | _ -> false let get_name_decl_ref_exp stmt = match stmt with | `DeclRefExpr(si, _, _, drei) -> get_name_decl_ref_exp_info drei si | _ -> assert false (* given the type of the enumeration and an enumeration constant (defined *) (* by stmt), returns the associated value *) let get_value_enum_constant tenv enum_type stmt = let constant = get_name_decl_ref_exp stmt in let typename = Sil.TN_enum(Mangled.from_string enum_type) in match Sil.tenv_lookup tenv typename with | Some (Sil.Tenum enum_constants) -> Printing.log_out ">>>Found enum with typename TN_typename('%s')\n" (Sil.typename_to_string typename); let _, v = try list_find (fun (c, _) -> Mangled.equal c (Mangled.from_string constant)) enum_constants with _ -> (Printing.log_err "Enumeration constant '%s' not found. Cannot continue...\n" constant; assert false) in v | _ -> Printing.log_err "Enum type '%s' not found in tenv. Cannot continue...\n" (Sil.typename_to_string typename); assert false let get_selector_receiver obj_c_message_expr_info = obj_c_message_expr_info.Clang_ast_t.omei_selector, obj_c_message_expr_info.Clang_ast_t.omei_receiver_kind (* Similar to extract_item_from_singleton but for option type *) let extract_item_from_option op warning_string = match op with | Some item -> item | _ -> Printing.log_err warning_string; assert false let is_member_exp stmt = match stmt with | MemberExpr _ -> true | _ -> false let is_enumeration_constant stmt = match stmt with | DeclRefExpr(_, _, _, drei) -> (match drei.Clang_ast_t.drti_decl_ref with | Some d -> (match d.Clang_ast_t.dr_kind with | `EnumConstant -> true | _ -> false) | _ -> false) | _ -> false let is_null_stmt s = match s with | NullStmt _ -> true | _ -> false let dummy_id () = Ident.create_normal (Ident.string_to_name "DUMMY_ID_INFER") 0 let extract_stmt_from_singleton stmt_list warning_string = extract_item_from_singleton stmt_list warning_string (Ast_expressions.dummy_stmt ()) let extract_id_from_singleton id_list warning_string = extract_item_from_singleton id_list warning_string (dummy_id ()) let rec get_type_from_exp_stmt stmt = let do_decl_ref_exp i = match i.Clang_ast_t.drti_decl_ref with | Some d -> (match d.Clang_ast_t.dr_qual_type with | Some n -> n | _ -> assert false ) | _ -> assert false in match stmt with | CXXOperatorCallExpr(_, _, ei) | CallExpr(_, _, ei) -> ei.Clang_ast_t.ei_qual_type | MemberExpr (_, _, ei, _) -> ei.Clang_ast_t.ei_qual_type | ParenExpr (_, _, ei) -> ei.Clang_ast_t.ei_qual_type | ArraySubscriptExpr(_, _, ei) -> ei.Clang_ast_t.ei_qual_type | ObjCIvarRefExpr (_, _, ei, _) -> ei.Clang_ast_t.ei_qual_type | ObjCMessageExpr (_, _, ei, _ ) -> ei.Clang_ast_t.ei_qual_type | PseudoObjectExpr(_, _, ei) -> ei.Clang_ast_t.ei_qual_type | CStyleCastExpr(_, stmt_list, _, _, _) | UnaryOperator(_, stmt_list, _, _) | ImplicitCastExpr(_, stmt_list, _, _) -> get_type_from_exp_stmt (extract_stmt_from_singleton stmt_list "WARNING: We expect only one stmt.") | DeclRefExpr(_, _, _, info) -> do_decl_ref_exp info | _ -> Printing.log_err "Failing with: %s \n%!" (Clang_ast_j.string_of_stmt stmt); Printing.print_failure_info ""; assert false module Self = struct exception SelfClassException of string let add_self_parameter_for_super_instance context procname loc mei trans_result = if is_superinstance mei then let typ, self_expr, id, ins = let t' = CTypes.add_pointer_to_typ (CTypes_decl.get_type_curr_class context.tenv context.curr_class) in let e = Sil.Lvar (Sil.mk_pvar (Mangled.from_string CFrontend_config.self) procname) in let id = Ident.create_fresh Ident.knormal in t', Sil.Var id, [id], [Sil.Letderef (id, e, t', loc)] in { trans_result with exps = (self_expr, typ):: trans_result.exps; ids = id@trans_result.ids; instrs = ins@trans_result.instrs } else trans_result let is_var_self pvar is_objc_method = let is_self = Mangled.to_string (Sil.pvar_get_name pvar) = CFrontend_config.self in is_self && is_objc_method end let get_decl_kind decl_ref_expr_info = match decl_ref_expr_info.Clang_ast_t.drti_decl_ref with | Some decl_ref -> decl_ref.Clang_ast_t.dr_kind | None -> assert false (* From the manual: A selector is in a certain selector family if, ignoring any leading underscores, *) (* the first component of the selector either consists entirely of the name of *) (* the method family or it begins with that followed by character other than lower case letter.*) (* For example: '__perform:with' and 'performWith:' would fall into the 'perform' family (if we had one),*) (* but 'performing:with' would not. *) let is_owning_name n = let is_family fam s'= if String.length s' < String.length fam then false else ( let prefix = Str.string_before s' (String.length fam) in let suffix = Str.string_after s' (String.length fam) in prefix = fam && not (Str.string_match (Str.regexp "[a-z]") suffix 0) ) in match Str.split (Str.regexp_string ":") n with | fst:: _ -> (match Str.split (Str.regexp "['_']+") fst with | [no_und] | _:: no_und:: _ -> is_family CFrontend_config.alloc no_und || is_family CFrontend_config.copy no_und || is_family CFrontend_config.new_str no_und || is_family CFrontend_config.mutableCopy no_und || is_family CFrontend_config.init no_und | _ -> assert false) | _ -> assert false let rec is_owning_method s = match s with | ObjCMessageExpr(_, _ , _, mei) -> is_owning_name mei.Clang_ast_t.omei_selector | _ -> (match snd (Clang_ast_proj.get_stmt_tuple s) with | [] -> false | s'':: _ -> is_owning_method s'') let rec is_method_call s = match s with | ObjCMessageExpr(_, _ , _, mei) -> true | _ -> (match snd (Clang_ast_proj.get_stmt_tuple s) with | [] -> false | s'':: _ -> is_method_call s'') let rec get_decl_ref_info s parent_line_number = match s with | DeclRefExpr (stmt_info, stmt_list, expr_info, decl_ref_expr_info) -> let line_number = CLocation.get_line stmt_info parent_line_number in stmt_info.Clang_ast_t.si_pointer, line_number | _ -> (match Clang_ast_proj.get_stmt_tuple s with | stmt_info, [] -> assert false | stmt_info, s'':: _ -> let line_number = CLocation.get_line stmt_info parent_line_number in get_decl_ref_info s'' line_number) let rec contains_opaque_value_expr s = match s with | OpaqueValueExpr (_, _, _, _) -> true | _ -> (match snd (Clang_ast_proj.get_stmt_tuple s) with | [] -> false | s'':: _ -> contains_opaque_value_expr s'') let rec compute_autorelease_pool_vars context stmts = match stmts with | [] -> [] | DeclRefExpr(si, sl, ei, drei):: stmts' -> let name = get_name_decl_ref_exp_info drei si in let procname = Cfg.Procdesc.get_proc_name context.procdesc in let local_vars = Cfg.Procdesc.get_locals context.procdesc in let mname = try list_filter (fun (m, t) -> Mangled.to_string m = name) local_vars with _ -> [] in (match mname with | [(m, t)] -> CFrontend_utils.General_utils.append_no_duplicated_pvars [(Sil.Lvar (Sil.mk_pvar m procname), t)] (compute_autorelease_pool_vars context stmts') | _ -> compute_autorelease_pool_vars context stmts') | s:: stmts' -> let sl = snd(Clang_ast_proj.get_stmt_tuple s) in compute_autorelease_pool_vars context (sl@stmts') (* checks if a unary operator is a logic negation applied to integers*) let is_logical_negation_of_int tenv ei uoi = match CTypes_decl.qual_type_to_sil_type tenv ei.Clang_ast_t.ei_qual_type, uoi.Clang_ast_t.uoi_kind with | Sil.Tint Sil.IInt,`LNot -> true | _, _ -> false (* Checks if stmt_list is a call to a special dispatch function *) let is_dispatch_function stmt_list = match stmt_list with | ImplicitCastExpr(_,[DeclRefExpr(_, _, _, di)], _, _):: stmts -> (match di.Clang_ast_t.drti_decl_ref with | None -> None | Some d -> (match d.Clang_ast_t.dr_kind, d.Clang_ast_t.dr_name with | `Function, Some name_info -> let s = name_info.Clang_ast_t.ni_name in (match (CTrans_models.is_dispatch_function_name s) with | None -> None | Some (dispatch_function, block_arg_pos) -> try (match list_nth stmts block_arg_pos with | BlockExpr _ -> Some block_arg_pos | _ -> None) with Not_found -> None ) | _ -> None)) | _ -> None let assign_default_params params_stmt callee_pname_opt = match callee_pname_opt with | None -> params_stmt | Some callee_pname -> try let callee_ms = CMethod_signature.find callee_pname in let args = CMethod_signature.ms_get_args callee_ms in let params_args = list_combine params_stmt args in let replace_default_arg param = match param with | CXXDefaultArgExpr(_, _, _), (_, _, Some default_instr) -> default_instr | instr, _ -> instr in list_map replace_default_arg params_args with | Invalid_argument _ -> (* list_combine failed because of different list lengths *) Printing.log_err "Param count doesn't match %s\n" (Procname.to_string callee_pname); params_stmt | Not_found -> params_stmt