|
|
|
(*
|
|
|
|
* 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
|
|
|
|
open! PVariant
|
|
|
|
|
|
|
|
(** Translates instructions: (statements and expressions) from the ast into sil *)
|
|
|
|
|
|
|
|
open CTrans_utils
|
|
|
|
open CTrans_utils.Nodes
|
|
|
|
module L = Logging
|
|
|
|
|
|
|
|
module CTrans_funct (F : CModule_type.CFrontend) : CModule_type.CTranslation = struct
|
|
|
|
(* Returns the procname and whether is instance, according to the selector information and
|
|
|
|
according to the method signature with the following priority:
|
|
|
|
1. method is a predefined model
|
|
|
|
2. method is found by clang's resolution
|
|
|
|
3. Method is found by our resolution *)
|
|
|
|
let get_callee_objc_method context obj_c_message_expr_info act_params =
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let open CContext in
|
|
|
|
let selector, method_pointer_opt, mc_type =
|
|
|
|
CMethod_trans.get_objc_method_data obj_c_message_expr_info
|
|
|
|
in
|
|
|
|
let is_instance = mc_type <> CMethod_trans.MCStatic in
|
|
|
|
let method_kind = Typ.Procname.objc_method_kind_of_bool is_instance in
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let ms_opt =
|
|
|
|
match method_pointer_opt with
|
|
|
|
| Some pointer ->
|
|
|
|
CMethod_trans.method_signature_of_pointer context.translation_unit_context context.tenv
|
|
|
|
pointer
|
|
|
|
| None ->
|
|
|
|
None
|
|
|
|
in
|
|
|
|
let proc_name =
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
match CMethod_trans.get_method_name_from_clang context.tenv ms_opt with
|
|
|
|
| Some ms ->
|
|
|
|
CMethod_signature.ms_get_name ms
|
|
|
|
| None ->
|
|
|
|
(* fall back to our method resolution if clang's fails *)
|
|
|
|
let class_name =
|
|
|
|
CMethod_trans.get_class_name_method_call_from_receiver_kind context
|
|
|
|
obj_c_message_expr_info act_params
|
|
|
|
in
|
|
|
|
CProcname.NoAstDecl.objc_method_of_string_kind class_name selector method_kind
|
|
|
|
in
|
|
|
|
let predefined_ms_opt =
|
|
|
|
match proc_name with
|
|
|
|
| Typ.Procname.ObjC_Cpp objc_cpp ->
|
|
|
|
let class_name = Typ.Procname.objc_cpp_get_class_type_name objc_cpp in
|
|
|
|
CTrans_models.get_predefined_model_method_signature class_name selector
|
|
|
|
CProcname.NoAstDecl.objc_method_of_string_kind CFrontend_config.ObjC
|
|
|
|
| _ ->
|
|
|
|
None
|
|
|
|
in
|
|
|
|
match (predefined_ms_opt, ms_opt) with
|
|
|
|
| Some ms, _ ->
|
|
|
|
ignore
|
|
|
|
(CMethod_trans.create_local_procdesc context.translation_unit_context context.cfg
|
|
|
|
context.tenv ms [] [] is_instance) ;
|
|
|
|
(CMethod_signature.ms_get_name ms, CMethod_trans.MCNoVirtual)
|
|
|
|
| None, Some ms ->
|
|
|
|
ignore
|
|
|
|
(CMethod_trans.create_local_procdesc context.translation_unit_context context.cfg
|
|
|
|
context.tenv ms [] [] is_instance) ;
|
|
|
|
if CMethod_signature.ms_is_getter ms || CMethod_signature.ms_is_setter ms then
|
|
|
|
(proc_name, CMethod_trans.MCNoVirtual)
|
|
|
|
else (proc_name, mc_type)
|
|
|
|
| _ ->
|
|
|
|
CMethod_trans.create_external_procdesc context.cfg proc_name is_instance None ;
|
|
|
|
(proc_name, mc_type)
|
|
|
|
|
|
|
|
|
|
|
|
let add_autorelease_call context exp typ sil_loc =
|
|
|
|
let method_name = Typ.Procname.get_method (Procdesc.get_proc_name context.CContext.procdesc) in
|
|
|
|
if !Config.arc_mode && not (CTrans_utils.is_owning_name method_name)
|
|
|
|
&& ObjcInterface_decl.is_pointer_to_objc_class typ
|
|
|
|
then
|
|
|
|
let fname = BuiltinDecl.__set_autorelease_attribute in
|
|
|
|
let ret_id = Some (Ident.create_fresh Ident.knormal, Typ.mk Typ.Tvoid) in
|
|
|
|
(* TODO(jjb): change ret_id to None? *)
|
|
|
|
let stmt_call =
|
|
|
|
Sil.Call (ret_id, Exp.Const (Const.Cfun fname), [(exp, typ)], sil_loc, CallFlags.default)
|
|
|
|
in
|
|
|
|
[stmt_call]
|
|
|
|
else []
|
|
|
|
|
|
|
|
|
|
|
|
let rec is_block_expr s =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match s with
|
|
|
|
| BlockExpr _ ->
|
|
|
|
true
|
|
|
|
(* the block can be wrapped in ExprWithCleanups or ImplicitCastExpr*)
|
|
|
|
| ImplicitCastExpr (_, [s'], _, _)
|
|
|
|
| ExprWithCleanups (_, [s'], _, _) ->
|
|
|
|
is_block_expr s'
|
|
|
|
| _ ->
|
|
|
|
false
|
|
|
|
|
|
|
|
|
|
|
|
let objc_exp_of_type_block fun_exp_stmt =
|
|
|
|
match fun_exp_stmt with
|
|
|
|
| Clang_ast_t.ImplicitCastExpr (_, _, ei, _)
|
|
|
|
when CType.is_block_type ei.Clang_ast_t.ei_qual_type ->
|
|
|
|
true
|
|
|
|
| _ ->
|
|
|
|
false
|
|
|
|
|
|
|
|
|
|
|
|
(* This function add in tenv a class representing an objc block. *)
|
|
|
|
(* An object of this class has type:*)
|
|
|
|
(* name_of_block |-> {capture_var1:typ_of_capture_var1,... capture_varn:typ_of_capture_varn} *)
|
|
|
|
(* It allocates one element and sets its fields with the current values of the *)
|
|
|
|
(* captured variables. This allocated instance
|
|
|
|
is used to detect retain cycles involving the block.*)
|
|
|
|
let allocate_block trans_state block_name captured_vars loc =
|
|
|
|
let tenv = trans_state.context.CContext.tenv in
|
|
|
|
let procdesc = trans_state.context.CContext.procdesc in
|
|
|
|
let procname = Procdesc.get_proc_name procdesc in
|
|
|
|
let mk_field_from_captured_var (var, typ) =
|
|
|
|
let vname = Pvar.get_name var in
|
|
|
|
let tname = Typ.Name.C.from_string block_name in
|
|
|
|
let fname = CGeneral_utils.mk_class_field_name tname (Mangled.to_string vname) in
|
|
|
|
let item_annot = Annot.Item.empty in
|
|
|
|
(fname, typ, item_annot)
|
|
|
|
in
|
|
|
|
let fields = List.map ~f:mk_field_from_captured_var captured_vars in
|
|
|
|
L.(debug Capture Verbose) "Block %s field:@\n" block_name ;
|
|
|
|
List.iter
|
|
|
|
~f:(fun (fn, _, _) ->
|
|
|
|
L.(debug Capture Verbose) "-----> field: '%s'@\n" (Typ.Fieldname.to_string fn))
|
|
|
|
fields ;
|
|
|
|
let block_typename = Typ.Name.Objc.from_string block_name in
|
|
|
|
ignore (Tenv.mk_struct tenv ~fields block_typename) ;
|
|
|
|
let block_type = Typ.mk (Typ.Tstruct block_typename) in
|
|
|
|
let trans_res =
|
|
|
|
CTrans_utils.alloc_trans trans_state ~alloc_builtin:BuiltinDecl.__objc_alloc_no_fail loc
|
|
|
|
(Ast_expressions.dummy_stmt_info ())
|
|
|
|
block_type
|
|
|
|
in
|
|
|
|
let id_block = match trans_res.exps with [(Exp.Var id, _)] -> id | _ -> assert false in
|
|
|
|
let mblock = Mangled.from_string block_name in
|
|
|
|
let block_var = Pvar.mk mblock procname in
|
|
|
|
let declare_block_local =
|
|
|
|
Sil.Declare_locals ([(block_var, CType.add_pointer_to_typ block_type)], loc)
|
|
|
|
in
|
|
|
|
let set_instr = Sil.Store (Exp.Lvar block_var, block_type, Exp.Var id_block, loc) in
|
|
|
|
let create_field_exp (var, typ) =
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
(id, Sil.Load (id, Exp.Lvar var, typ, loc))
|
|
|
|
in
|
|
|
|
let ids, captured_instrs = List.unzip (List.map ~f:create_field_exp captured_vars) in
|
|
|
|
let fields_ids = List.zip_exn fields ids in
|
|
|
|
let set_fields =
|
|
|
|
List.map
|
|
|
|
~f:(fun ((f, t, _), id) ->
|
|
|
|
Sil.Store (Exp.Lfield (Exp.Var id_block, f, block_type), t, Exp.Var id, loc))
|
|
|
|
fields_ids
|
|
|
|
in
|
|
|
|
declare_block_local :: trans_res.instrs @ [set_instr] @ captured_instrs @ set_fields
|
|
|
|
|
|
|
|
|
|
|
|
(* From a list of expression extract blocks from tuples and *)
|
|
|
|
(* returns block names and assignment to temp vars *)
|
|
|
|
let extract_block_from_tuple procname exps loc =
|
|
|
|
let insts = ref [] in
|
|
|
|
let make_function_name typ bn =
|
|
|
|
let bn' = Typ.Procname.to_string bn in
|
|
|
|
let bn'' = Mangled.from_string bn' in
|
|
|
|
let block = Exp.Lvar (Pvar.mk bn'' procname) in
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
insts := Sil.Load (id, block, typ, loc) :: !insts ;
|
|
|
|
(Exp.Var id, typ)
|
|
|
|
in
|
|
|
|
let make_arg typ (id, _, _) = (id, typ) in
|
|
|
|
let f = function
|
|
|
|
| Exp.Closure {name; captured_vars}, ({Typ.desc= Tptr ({Typ.desc= Tfun _}, _)} as t) ->
|
|
|
|
let function_name = make_function_name t name in
|
|
|
|
let args = List.map ~f:(make_arg t) captured_vars in
|
|
|
|
function_name :: args
|
|
|
|
| e ->
|
|
|
|
[e]
|
|
|
|
in
|
|
|
|
(* evaluation order matters here *)
|
|
|
|
let exps' = List.concat_map ~f exps in
|
|
|
|
let insts' = !insts in
|
|
|
|
(exps', insts')
|
|
|
|
|
|
|
|
|
|
|
|
let collect_exprs res_trans_list =
|
|
|
|
List.concat_map ~f:(fun res_trans -> res_trans.exps) res_trans_list
|
|
|
|
|
|
|
|
|
|
|
|
(* If e is a block and the calling node has the priority then *)
|
|
|
|
(* we need to release the priority to allow*)
|
|
|
|
(* creation of nodes inside the block.*)
|
|
|
|
(* At the end of block translation, we need to get the proirity back.*)
|
|
|
|
(* the parameter f will be called with function instruction *)
|
|
|
|
let exec_with_block_priority_exception f trans_state e stmt_info =
|
|
|
|
if is_block_expr e && PriorityNode.own_priority_node trans_state.priority stmt_info then (
|
|
|
|
L.(debug Capture Verbose) "Translating block expression by freeing the priority" ;
|
|
|
|
f {trans_state with priority= Free} e )
|
|
|
|
else f trans_state e
|
|
|
|
|
|
|
|
|
|
|
|
let exec_with_node_creation f trans_state stmt =
|
|
|
|
let res_trans = f trans_state stmt in
|
|
|
|
if res_trans.instrs <> [] then
|
|
|
|
let stmt_info, _ = Clang_ast_proj.get_stmt_tuple stmt in
|
|
|
|
let stmt_info' = {stmt_info with Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()} in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info' in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info' trans_state.context in
|
|
|
|
let to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "Fallback node" stmt_info'
|
|
|
|
[res_trans]
|
|
|
|
in
|
|
|
|
{to_parent with exps= res_trans.exps}
|
|
|
|
else res_trans
|
|
|
|
|
|
|
|
|
|
|
|
(* This is the standard way of dealing with self:Class or a call [a class]. We translate it as
|
|
|
|
sizeof(<type pf a>) The only time when we want to translate those expressions differently is
|
|
|
|
when they are the first argument of method calls. In that case they are not translated as
|
|
|
|
expressions, but we take the type and create a static method call from it. This is done in
|
|
|
|
objcMessageExpr_trans. *)
|
|
|
|
let exec_with_self_exception f trans_state stmt =
|
|
|
|
try f trans_state stmt
|
|
|
|
with Self.SelfClassException class_name ->
|
|
|
|
let typ = Typ.mk (Tstruct class_name) in
|
|
|
|
{ empty_res_trans with
|
|
|
|
exps=
|
|
|
|
[ ( Exp.Sizeof {typ; nbytes= None; dynamic_length= None; subtype= Subtype.exact}
|
|
|
|
, Typ.mk (Tint IULong) ) ] }
|
|
|
|
|
|
|
|
|
|
|
|
let add_reference_if_glvalue (typ: Typ.t) expr_info =
|
|
|
|
(* glvalue definition per C++11:*)
|
|
|
|
(* http://en.cppreference.com/w/cpp/language/value_category *)
|
|
|
|
let is_glvalue =
|
|
|
|
match expr_info.Clang_ast_t.ei_value_kind with `LValue | `XValue -> true | `RValue -> false
|
|
|
|
in
|
|
|
|
match (is_glvalue, typ.desc) with
|
|
|
|
| true, Tptr (_, Pk_reference) ->
|
|
|
|
(* reference of reference is not allowed in C++ - it's most likely frontend *)
|
|
|
|
(* trying to add same reference to same type twice*)
|
|
|
|
(* this is hacky and should be fixed (t9838691) *)
|
|
|
|
typ
|
|
|
|
| true, _ ->
|
|
|
|
Typ.mk (Tptr (typ, Pk_reference))
|
|
|
|
| _ ->
|
|
|
|
typ
|
|
|
|
|
|
|
|
|
|
|
|
(** Execute translation and then possibly adjust the type of the result of translation:
|
|
|
|
In C++, when expression returns reference to type T, it will be lvalue to T, not T&, but
|
|
|
|
infer needs it to be T& *)
|
|
|
|
let exec_with_glvalue_as_reference f trans_state stmt =
|
|
|
|
let expr_info =
|
|
|
|
match Clang_ast_proj.get_expr_tuple stmt with Some (_, _, ei) -> ei | None -> assert false
|
|
|
|
in
|
|
|
|
let res_trans = f trans_state stmt in
|
|
|
|
let exp, typ =
|
|
|
|
extract_exp_from_list res_trans.exps
|
|
|
|
"[Warning] Need exactly one expression to add reference type@\n"
|
|
|
|
in
|
|
|
|
{res_trans with exps= [(exp, add_reference_if_glvalue typ expr_info)]}
|
|
|
|
|
|
|
|
|
|
|
|
(* Execute translation of e forcing to release priority
|
|
|
|
(if it's not free) and then setting it back.*)
|
|
|
|
(* This is used in conditional operators where we need to force
|
|
|
|
the priority to be free for the *)
|
|
|
|
(* computation of the expressions*)
|
|
|
|
let exec_with_priority_exception trans_state e f =
|
|
|
|
if PriorityNode.is_priority_free trans_state then f trans_state e
|
|
|
|
else f {trans_state with priority= Free} e
|
|
|
|
|
|
|
|
|
|
|
|
let call_translation context decl =
|
|
|
|
let open CContext in
|
|
|
|
(* translation will reset Ident counter, save it's state and restore it afterwards *)
|
|
|
|
let ident_state = Ident.NameGenerator.get_current () in
|
|
|
|
F.translate_one_declaration context.translation_unit_context context.tenv context.cg
|
|
|
|
context.cfg `Translation decl ;
|
|
|
|
Ident.NameGenerator.set_current ident_state
|
|
|
|
|
|
|
|
|
|
|
|
let mk_temp_sil_var procdesc var_name_suffix =
|
|
|
|
let procname = Procdesc.get_proc_name procdesc in
|
|
|
|
Pvar.mk_tmp var_name_suffix procname
|
|
|
|
|
|
|
|
|
|
|
|
let mk_temp_sil_var_for_expr tenv procdesc var_name_prefix expr_info =
|
|
|
|
let qual_type = expr_info.Clang_ast_t.ei_qual_type in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type tenv qual_type in
|
|
|
|
(mk_temp_sil_var procdesc var_name_prefix, typ)
|
|
|
|
|
|
|
|
|
|
|
|
let create_var_exp_tmp_var trans_state expr_info var_name =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let procdesc = context.CContext.procdesc in
|
|
|
|
let pvar, typ = mk_temp_sil_var_for_expr context.CContext.tenv procdesc var_name expr_info in
|
|
|
|
Procdesc.append_locals procdesc [(Pvar.get_name pvar, typ)] ;
|
|
|
|
(Exp.Lvar pvar, typ)
|
|
|
|
|
|
|
|
|
|
|
|
let create_call_instr trans_state (return_type: Typ.t) function_sil params_sil sil_loc call_flags
|
|
|
|
~is_objc_method =
|
|
|
|
let ret_id =
|
|
|
|
if Typ.equal_desc return_type.desc Typ.Tvoid then None
|
|
|
|
else Some (Ident.create_fresh Ident.knormal, return_type)
|
|
|
|
in
|
|
|
|
let ret_id', params, initd_exps, ret_exps =
|
|
|
|
(* Assumption: should_add_return_param will return true only for struct types *)
|
|
|
|
if CMethod_trans.should_add_return_param return_type ~is_objc_method then
|
|
|
|
let param_type = Typ.mk (Typ.Tptr (return_type, Typ.Pk_pointer)) in
|
|
|
|
let var_exp =
|
|
|
|
match trans_state.var_exp_typ with
|
|
|
|
| Some (exp, _) ->
|
|
|
|
exp
|
|
|
|
| _ ->
|
|
|
|
let procdesc = trans_state.context.CContext.procdesc in
|
|
|
|
let pvar = mk_temp_sil_var procdesc "__temp_return_" in
|
|
|
|
Procdesc.append_locals procdesc [(Pvar.get_name pvar, return_type)] ;
|
|
|
|
Exp.Lvar pvar
|
|
|
|
in
|
|
|
|
(* It is very confusing - same expression has two different types in two contexts:*)
|
|
|
|
(* 1. if passed as parameter it's RETURN_TYPE* since we are passing it as rvalue *)
|
|
|
|
(* 2. for return expression it's RETURN_TYPE since backend allows to treat it as lvalue*)
|
|
|
|
(* of RETURN_TYPE *)
|
|
|
|
(* Implications: *)
|
|
|
|
(* Fields: field_deref_trans relies on it - if exp has RETURN_TYPE then *)
|
|
|
|
(* it means that it's not lvalue in clang's AST (it'd be reference otherwise) *)
|
|
|
|
(* Methods: method_deref_trans actually wants a pointer to the object, which is*)
|
|
|
|
(* equivalent of value of ret_param. Since ret_exp has type RETURN_TYPE,*)
|
|
|
|
(* we optionally add pointer there to avoid backend confusion. *)
|
|
|
|
(* It works either way *)
|
|
|
|
(* Passing by value: may cause problems - there needs to be extra Sil.Load, but*)
|
|
|
|
(* doing so would create problems with methods. Passing structs by*)
|
|
|
|
(* value doesn't work good anyway. This may need to be revisited later*)
|
|
|
|
let ret_param = (var_exp, param_type) in
|
|
|
|
let ret_exp = (var_exp, return_type) in
|
|
|
|
(None, params_sil @ [ret_param], [var_exp], [ret_exp])
|
|
|
|
else
|
|
|
|
(ret_id, params_sil, [], match ret_id with Some (i, t) -> [(Exp.Var i, t)] | None -> [])
|
|
|
|
in
|
|
|
|
let call_instr = Sil.Call (ret_id', function_sil, params, sil_loc, call_flags) in
|
|
|
|
{empty_res_trans with instrs= [call_instr]; exps= ret_exps; initd_exps}
|
|
|
|
|
|
|
|
|
|
|
|
let stringLiteral_trans trans_state expr_info str =
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info trans_state.context.CContext.tenv in
|
|
|
|
let exp = Exp.Const (Const.Cstr str) in
|
|
|
|
{empty_res_trans with exps= [(exp, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
(* FROM CLANG DOCS: "Implements the GNU __null extension,
|
|
|
|
which is a name for a null pointer constant *)
|
|
|
|
(* that has integral type (e.g., int or long) and is the same
|
|
|
|
size and alignment as a pointer. The __null *)
|
|
|
|
(* extension is typically only used by system headers,
|
|
|
|
which define NULL as __null in C++ rather than using 0 *)
|
|
|
|
(* (which is an integer that may not match the size of a pointer)".
|
|
|
|
So we implement it as the constant zero *)
|
|
|
|
let gNUNullExpr_trans trans_state expr_info =
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info trans_state.context.CContext.tenv in
|
|
|
|
let exp = Exp.Const (Const.Cint IntLit.zero) in
|
|
|
|
{empty_res_trans with exps= [(exp, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
let nullPtrExpr_trans trans_state expr_info =
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info trans_state.context.CContext.tenv in
|
|
|
|
{empty_res_trans with exps= [(Exp.null, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
let objCSelectorExpr_trans trans_state expr_info selector =
|
|
|
|
stringLiteral_trans trans_state expr_info selector
|
|
|
|
|
|
|
|
|
|
|
|
let objCEncodeExpr_trans trans_state expr_info objc_encode_expr_info =
|
|
|
|
let type_raw = objc_encode_expr_info.Clang_ast_t.oeei_raw in
|
|
|
|
stringLiteral_trans trans_state expr_info type_raw
|
|
|
|
|
|
|
|
|
|
|
|
let objCProtocolExpr_trans trans_state expr_info decl_ref =
|
|
|
|
let name =
|
|
|
|
match decl_ref.Clang_ast_t.dr_name with Some s -> s.Clang_ast_t.ni_name | _ -> ""
|
|
|
|
in
|
|
|
|
stringLiteral_trans trans_state expr_info name
|
|
|
|
|
|
|
|
|
|
|
|
let characterLiteral_trans trans_state expr_info n =
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info trans_state.context.CContext.tenv in
|
|
|
|
let exp = Exp.Const (Const.Cint (IntLit.of_int n)) in
|
|
|
|
{empty_res_trans with exps= [(exp, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
let booleanValue_trans trans_state expr_info b =
|
|
|
|
characterLiteral_trans trans_state expr_info (if b then 1 else 0)
|
|
|
|
|
|
|
|
|
|
|
|
let floatingLiteral_trans trans_state expr_info float_string =
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info trans_state.context.CContext.tenv in
|
|
|
|
let exp = Exp.Const (Const.Cfloat (float_of_string float_string)) in
|
|
|
|
{empty_res_trans with exps= [(exp, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
(* Note currently we don't have support for different qual *)
|
|
|
|
(* type like long, unsigned long, etc *)
|
|
|
|
and integerLiteral_trans trans_state expr_info integer_literal_info =
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info trans_state.context.CContext.tenv in
|
|
|
|
let exp =
|
|
|
|
try
|
|
|
|
let i = Int64.of_string integer_literal_info.Clang_ast_t.ili_value in
|
|
|
|
let exp = Exp.int (IntLit.of_int64 i) in
|
|
|
|
exp
|
|
|
|
with Failure _ ->
|
|
|
|
(* Parse error: return a nondeterministic value *)
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
Exp.Var id
|
|
|
|
in
|
|
|
|
{empty_res_trans with exps= [(exp, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
let cxxScalarValueInitExpr_trans trans_state expr_info =
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info trans_state.context.CContext.tenv in
|
|
|
|
(* constant will be different depending on type *)
|
|
|
|
let zero_opt =
|
|
|
|
match typ.desc with
|
|
|
|
| Typ.Tfloat _ | Typ.Tptr _ | Typ.Tint _ ->
|
|
|
|
Some (Sil.zero_value_of_numerical_type typ)
|
|
|
|
| Typ.Tvoid ->
|
|
|
|
None
|
|
|
|
| _ ->
|
|
|
|
Some (Exp.Const (Const.Cint IntLit.zero))
|
|
|
|
in
|
|
|
|
match zero_opt with
|
|
|
|
| Some zero ->
|
|
|
|
{empty_res_trans with exps= [(zero, typ)]}
|
|
|
|
| _ ->
|
|
|
|
empty_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
(** Create instructions to initialize record with zeroes. It needs to traverse
|
|
|
|
whole type structure, to assign 0 values to all transitive fields because of
|
|
|
|
AST construction in C translation *)
|
|
|
|
let implicitValueInitExpr_trans trans_state stmt_info =
|
|
|
|
match trans_state.var_exp_typ with
|
|
|
|
| Some var_exp_typ ->
|
|
|
|
(* This node will always be child of InitListExpr, claiming priority will always fail *)
|
|
|
|
let tenv = trans_state.context.CContext.tenv in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
let flatten_res_trans = collect_res_trans trans_state.context.procdesc in
|
|
|
|
(* Traverse structure of a type and initialize int/float/ptr fields with zero *)
|
|
|
|
let rec fill_typ_with_zero (exp, typ) : trans_result =
|
|
|
|
match typ.Typ.desc with
|
|
|
|
| Tstruct tn ->
|
|
|
|
let field_exps =
|
|
|
|
match Tenv.lookup tenv tn with
|
|
|
|
| Some {fields} ->
|
|
|
|
List.filter_map fields ~f:(fun (fieldname, fieldtype, _) ->
|
|
|
|
if Typ.Fieldname.is_hidden fieldname then None
|
|
|
|
else Some (Exp.Lfield (exp, fieldname, typ), fieldtype) )
|
|
|
|
| None ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
List.map ~f:fill_typ_with_zero field_exps |> flatten_res_trans
|
|
|
|
| Tarray (field_typ, Some n, _) ->
|
|
|
|
let size = IntLit.to_int n in
|
|
|
|
let indices = CGeneral_utils.list_range 0 (size - 1) in
|
|
|
|
List.map indices ~f:(fun i ->
|
|
|
|
let idx_exp = Exp.Const (Const.Cint (IntLit.of_int i)) in
|
|
|
|
let field_exp = Exp.Lindex (exp, idx_exp) in
|
|
|
|
fill_typ_with_zero (field_exp, field_typ) )
|
|
|
|
|> flatten_res_trans
|
|
|
|
| Tint _ | Tfloat _ | Tptr _ ->
|
|
|
|
let zero_exp = Sil.zero_value_of_numerical_type typ in
|
|
|
|
let instrs = [Sil.Store (exp, typ, zero_exp, sil_loc)] in
|
|
|
|
let exps = [(exp, typ)] in
|
|
|
|
{empty_res_trans with exps; instrs}
|
|
|
|
| Tfun _ | Tvoid | Tarray _ | TVar _ ->
|
|
|
|
CFrontend_config.unimplemented "fill_typ_with_zero on type %a" (Typ.pp Pp.text) typ
|
|
|
|
in
|
|
|
|
let res_trans = fill_typ_with_zero var_exp_typ in
|
|
|
|
{res_trans with initd_exps= [fst var_exp_typ]}
|
|
|
|
| None ->
|
|
|
|
CFrontend_config.unimplemented "Retrieving var from non-InitListExpr parent"
|
|
|
|
|
|
|
|
|
|
|
|
let no_op_trans succ_nodes = {empty_res_trans with root_nodes= succ_nodes}
|
|
|
|
|
|
|
|
(* The stmt seems to be always empty *)
|
|
|
|
let unaryExprOrTypeTraitExpr_trans trans_state expr_info unary_expr_or_type_trait_expr_info =
|
|
|
|
let tenv = trans_state.context.CContext.tenv in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type tenv expr_info.Clang_ast_t.ei_qual_type in
|
|
|
|
match unary_expr_or_type_trait_expr_info.Clang_ast_t.uttei_kind with
|
|
|
|
| `SizeOf | `SizeOfWithSize _ as size ->
|
|
|
|
let qt_opt =
|
|
|
|
CAst_utils.type_from_unary_expr_or_type_trait_expr_info
|
|
|
|
unary_expr_or_type_trait_expr_info
|
|
|
|
in
|
|
|
|
let sizeof_typ =
|
|
|
|
match qt_opt with Some qt -> CType_decl.qual_type_to_sil_type tenv qt | None -> typ
|
|
|
|
(* Some default type since the type is missing *)
|
|
|
|
in
|
|
|
|
let nbytes = match size with `SizeOfWithSize nbytes -> Some nbytes | _ -> None in
|
|
|
|
let sizeof_data =
|
|
|
|
{Exp.typ= sizeof_typ; nbytes; dynamic_length= None; subtype= Subtype.exact}
|
|
|
|
in
|
|
|
|
{empty_res_trans with exps= [(Exp.Sizeof sizeof_data, sizeof_typ)]}
|
|
|
|
| k ->
|
|
|
|
L.(debug Capture Medium)
|
|
|
|
"@\nWARNING: Missing translation of Uniry_Expression_Or_Trait of kind: %s . Expression ignored, returned -1... @\n"
|
|
|
|
(Clang_ast_j.string_of_unary_expr_or_type_trait_kind k) ;
|
|
|
|
{empty_res_trans with exps= [(Exp.minus_one, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
(* search the label into the hashtbl - create a fake node eventually *)
|
|
|
|
(* connect that node with this stmt *)
|
|
|
|
let gotoStmt_trans trans_state stmt_info label_name =
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
let root_node' = GotoLabel.find_goto_label trans_state.context label_name sil_loc in
|
|
|
|
{empty_res_trans with root_nodes= [root_node']; leaf_nodes= trans_state.succ_nodes}
|
|
|
|
|
|
|
|
|
|
|
|
let get_builtin_pname_opt trans_unit_ctx qual_name decl_opt (qual_type: Clang_ast_t.qual_type) =
|
|
|
|
let get_annotate_attr_arg decl =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
let decl_info = Clang_ast_proj.get_decl_tuple decl in
|
|
|
|
let get_attr_opt = function AnnotateAttr a -> Some a | _ -> None in
|
|
|
|
match List.find_map ~f:get_attr_opt decl_info.di_attributes with
|
|
|
|
| Some attribute_info -> (
|
|
|
|
match attribute_info.ai_parameters with
|
|
|
|
| [_; arg; _] ->
|
|
|
|
Some arg
|
|
|
|
| _ ->
|
|
|
|
(* it's not supposed to happen due to hardcoded exporting logic
|
|
|
|
coming from ASTExporter.h in facebook-clang-plugins *)
|
|
|
|
assert false )
|
|
|
|
| None ->
|
|
|
|
None
|
|
|
|
in
|
|
|
|
let name = QualifiedCppName.to_qual_string qual_name in
|
|
|
|
let function_attr_opt = Option.bind decl_opt ~f:get_annotate_attr_arg in
|
|
|
|
match function_attr_opt with
|
|
|
|
| Some attr when CTrans_models.is_modeled_attribute attr ->
|
|
|
|
Some (Typ.Procname.from_string_c_fun attr)
|
|
|
|
| _ when CTrans_models.is_modeled_builtin name ->
|
|
|
|
Some (Typ.Procname.from_string_c_fun (CFrontend_config.infer ^ name))
|
|
|
|
| _ when CTrans_models.is_release_builtin name qual_type.qt_type_ptr ->
|
|
|
|
Some BuiltinDecl.__objc_release_cf
|
|
|
|
| _ when CTrans_models.is_retain_builtin name qual_type.qt_type_ptr ->
|
|
|
|
Some BuiltinDecl.__objc_retain_cf
|
|
|
|
| _
|
|
|
|
when String.equal name CFrontend_config.malloc
|
|
|
|
&& CGeneral_utils.is_objc_extension trans_unit_ctx ->
|
|
|
|
Some BuiltinDecl.malloc_no_fail
|
|
|
|
| _ ->
|
|
|
|
None
|
|
|
|
|
|
|
|
|
|
|
|
let function_deref_trans trans_state decl_ref =
|
|
|
|
let open CContext in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let name_info, decl_ptr, qual_type = CAst_utils.get_info_from_decl_ref decl_ref in
|
|
|
|
let decl_opt = CAst_utils.get_function_decl_with_body decl_ptr in
|
|
|
|
Option.iter ~f:(call_translation context) decl_opt ;
|
|
|
|
let qual_name = CAst_utils.get_qualified_name name_info in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type context.tenv qual_type in
|
|
|
|
let pname =
|
|
|
|
match
|
|
|
|
get_builtin_pname_opt context.translation_unit_context qual_name decl_opt qual_type
|
|
|
|
with
|
|
|
|
| Some builtin_pname ->
|
|
|
|
builtin_pname
|
|
|
|
| None ->
|
|
|
|
let name = QualifiedCppName.to_qual_string qual_name in
|
|
|
|
CMethod_trans.create_procdesc_with_pointer context decl_ptr None name
|
|
|
|
in
|
|
|
|
{empty_res_trans with exps= [(Exp.Const (Const.Cfun pname), typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
let field_deref_trans trans_state stmt_info pre_trans_result decl_ref ~is_constructor_init =
|
|
|
|
let open CContext in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let name_info, decl_ptr, qual_type = CAst_utils.get_info_from_decl_ref decl_ref in
|
|
|
|
let field_string = name_info.Clang_ast_t.ni_name in
|
|
|
|
L.(debug Capture Verbose) "!!!!! Dealing with field '%s' @." field_string ;
|
|
|
|
let field_typ = CType_decl.qual_type_to_sil_type context.tenv qual_type in
|
|
|
|
let obj_sil, class_typ =
|
|
|
|
extract_exp_from_list pre_trans_result.exps
|
|
|
|
"WARNING: in Field dereference we expect to know the object@\n"
|
|
|
|
in
|
|
|
|
let is_pointer_typ = match class_typ.desc with Typ.Tptr _ -> true | _ -> false in
|
|
|
|
let class_typ = match class_typ.desc with Typ.Tptr (t, _) -> t | _ -> class_typ in
|
|
|
|
L.(debug Capture Verbose) "Type is '%s' @." (Typ.to_string class_typ) ;
|
|
|
|
let class_tname =
|
|
|
|
match CAst_utils.get_decl decl_ptr with
|
|
|
|
| Some FieldDecl ({di_parent_pointer}, _, _, _)
|
|
|
|
| Some ObjCIvarDecl ({di_parent_pointer}, _, _, _, _) -> (
|
|
|
|
match CAst_utils.get_decl_opt di_parent_pointer with
|
|
|
|
| Some decl ->
|
|
|
|
CType_decl.get_record_typename ~tenv:context.tenv decl
|
|
|
|
| _ ->
|
|
|
|
assert false )
|
|
|
|
| _ as decl ->
|
|
|
|
(* FIXME(t21762295): we do not expect this to happen but it does *)
|
|
|
|
CFrontend_config.incorrect_assumption
|
|
|
|
"di_parent_pointer should be always set for fields/ivars, but got %a"
|
|
|
|
(Pp.option (Pp.to_string ~f:Clang_ast_j.string_of_decl))
|
|
|
|
decl
|
|
|
|
in
|
|
|
|
let field_name = CGeneral_utils.mk_class_field_name class_tname field_string in
|
|
|
|
let field_exp = Exp.Lfield (obj_sil, field_name, class_typ) in
|
|
|
|
(* In certain cases, there is be no LValueToRValue cast, but backend needs dereference*)
|
|
|
|
(* there either way:*)
|
|
|
|
(* 1. Class is not a pointer type - it means that it's rvalue struct most likely coming from*)
|
|
|
|
(* create_call_instr - more info there*)
|
|
|
|
(* 2. Field has reference type - we need to add extra dereference in same fashion*)
|
|
|
|
(* it's done in var_deref_trans. The only exception is during field initialization in*)
|
|
|
|
(* constructor's initializer list (when reference itself is initialized) *)
|
|
|
|
let should_add_deref =
|
|
|
|
not is_pointer_typ || not is_constructor_init && CType.is_reference_type qual_type
|
|
|
|
in
|
|
|
|
let exp, deref_instrs =
|
|
|
|
if should_add_deref then
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
let deref_instr = Sil.Load (id, field_exp, field_typ, sil_loc) in
|
|
|
|
(Exp.Var id, [deref_instr])
|
|
|
|
else (field_exp, [])
|
|
|
|
in
|
|
|
|
let instrs = pre_trans_result.instrs @ deref_instrs in
|
|
|
|
{pre_trans_result with instrs; exps= [(exp, field_typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
let method_deref_trans ?(is_inner_destructor= false) trans_state pre_trans_result decl_ref
|
|
|
|
stmt_info decl_kind =
|
|
|
|
let open CContext in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let name_info, decl_ptr, qual_type = CAst_utils.get_info_from_decl_ref decl_ref in
|
|
|
|
let decl_opt = CAst_utils.get_function_decl_with_body decl_ptr in
|
|
|
|
Option.iter ~f:(call_translation context) decl_opt ;
|
|
|
|
let method_name = CAst_utils.get_unqualified_name name_info in
|
|
|
|
L.(debug Capture Verbose) "!!!!! Dealing with method '%s' @." method_name ;
|
|
|
|
let method_typ = CType_decl.qual_type_to_sil_type context.tenv qual_type in
|
|
|
|
let ms_opt =
|
|
|
|
CMethod_trans.method_signature_of_pointer context.translation_unit_context context.tenv
|
|
|
|
decl_ptr
|
|
|
|
in
|
|
|
|
let is_instance_method =
|
|
|
|
match ms_opt with Some ms -> CMethod_signature.ms_is_instance ms | _ -> true
|
|
|
|
(* might happen for methods that are not exported yet (some templates). *)
|
|
|
|
in
|
|
|
|
let is_cpp_virtual =
|
|
|
|
match ms_opt with Some ms -> CMethod_signature.ms_is_cpp_virtual ms | _ -> false
|
|
|
|
in
|
|
|
|
let extra_exps, extra_instrs =
|
|
|
|
if is_instance_method then
|
|
|
|
match
|
|
|
|
(* pre_trans_result.exps may contain expr for 'this' parameter:*)
|
|
|
|
(* if it comes from CXXMemberCallExpr it will be there *)
|
|
|
|
(* if it comes from CXXOperatorCallExpr it won't be there and will be added later *)
|
|
|
|
(* In case of CXXMemberCallExpr it's possible that type of 'this' parameter *)
|
|
|
|
(* won't have a pointer - if that happens add a pointer to type of the object *)
|
|
|
|
pre_trans_result.exps
|
|
|
|
with
|
|
|
|
| [] ->
|
|
|
|
([], [])
|
|
|
|
(* We need to add a dereference before a method call to find null dereferences when *)
|
|
|
|
(* calling a method with null *)
|
|
|
|
| [(exp, {Typ.desc= Tptr (typ, _)})]
|
|
|
|
when decl_kind <> `CXXConstructor ->
|
|
|
|
let no_id = Ident.create_none () in
|
|
|
|
let extra_instrs = [Sil.Load (no_id, exp, typ, sil_loc)] in
|
|
|
|
(pre_trans_result.exps, extra_instrs)
|
|
|
|
| [(_, {Typ.desc= Tptr _})] ->
|
|
|
|
(pre_trans_result.exps, [])
|
|
|
|
| [(sil, typ)] ->
|
|
|
|
([(sil, Typ.mk (Tptr (typ, Typ.Pk_reference)))], [])
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
else (* don't add 'this' expression for static methods *)
|
|
|
|
([], [])
|
|
|
|
in
|
|
|
|
(* unlike field access, for method calls there is no need to expand class type *)
|
|
|
|
(* use qualified method name for builtin matching, but use unqualified name elsewhere *)
|
|
|
|
let qual_method_name = CAst_utils.get_qualified_name name_info in
|
|
|
|
let pname =
|
|
|
|
match
|
|
|
|
get_builtin_pname_opt context.translation_unit_context qual_method_name decl_opt qual_type
|
|
|
|
with
|
|
|
|
| Some builtin_pname ->
|
|
|
|
builtin_pname
|
|
|
|
| None ->
|
|
|
|
let class_typename =
|
|
|
|
Typ.Name.Cpp.from_qual_name Typ.NoTemplate
|
|
|
|
(CAst_utils.get_class_name_from_member name_info)
|
|
|
|
in
|
|
|
|
if is_inner_destructor then
|
|
|
|
match ms_opt with
|
|
|
|
| Some ms ->
|
|
|
|
let procname = CMethod_signature.ms_get_name ms in
|
|
|
|
let new_method_name =
|
|
|
|
Config.clang_inner_destructor_prefix ^ Typ.Procname.get_method procname
|
|
|
|
in
|
|
|
|
let ms' =
|
|
|
|
CMethod_signature.replace_name_ms ms
|
|
|
|
(Typ.Procname.objc_cpp_replace_method_name procname new_method_name)
|
|
|
|
in
|
|
|
|
ignore
|
|
|
|
(CMethod_trans.create_local_procdesc context.translation_unit_context context.cfg
|
|
|
|
context.tenv ms' [] [] false) ;
|
|
|
|
CMethod_signature.ms_get_name ms'
|
|
|
|
| None ->
|
|
|
|
CMethod_trans.create_procdesc_with_pointer context decl_ptr (Some class_typename)
|
|
|
|
method_name
|
|
|
|
else
|
|
|
|
CMethod_trans.create_procdesc_with_pointer context decl_ptr (Some class_typename)
|
|
|
|
method_name
|
|
|
|
in
|
|
|
|
let method_exp = (Exp.Const (Const.Cfun pname), method_typ) in
|
|
|
|
{ pre_trans_result with
|
|
|
|
is_cpp_call_virtual= is_cpp_virtual
|
|
|
|
; exps= [method_exp] @ extra_exps
|
|
|
|
; instrs= pre_trans_result.instrs @ extra_instrs }
|
|
|
|
|
|
|
|
|
|
|
|
let destructor_deref_trans trans_state pvar_trans_result class_type_ptr si ~is_inner_destructor =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
let destruct_decl_ref_opt =
|
|
|
|
match CAst_utils.get_decl_from_typ_ptr class_type_ptr with
|
|
|
|
| Some CXXRecordDecl (_, _, _, _, _, _, _, cxx_record_info)
|
|
|
|
| Some ClassTemplateSpecializationDecl (_, _, _, _, _, _, _, cxx_record_info, _, _) ->
|
|
|
|
cxx_record_info.xrdi_destructor
|
|
|
|
| _ ->
|
|
|
|
None
|
|
|
|
in
|
|
|
|
match destruct_decl_ref_opt with
|
|
|
|
| Some decl_ref -> (
|
|
|
|
match CAst_utils.get_decl decl_ref.Clang_ast_t.dr_decl_pointer with
|
|
|
|
| Some CXXDestructorDecl (_, named_decl_info, _, {fdi_body= None}, _) ->
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
"@\n Trying to translate destructor call, but found empty destructor body for %s@\n@."
|
|
|
|
(CAst_utils.get_unqualified_name named_decl_info) ;
|
|
|
|
empty_res_trans
|
|
|
|
| Some CXXDestructorDecl (_, _, _, {fdi_body= Some _}, _)
|
|
|
|
(* Translate only those destructors that have bodies *) ->
|
|
|
|
method_deref_trans ~is_inner_destructor trans_state pvar_trans_result decl_ref si
|
|
|
|
`CXXDestructor
|
|
|
|
| _ ->
|
|
|
|
empty_res_trans )
|
|
|
|
| None ->
|
|
|
|
empty_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
let this_expr_trans trans_state sil_loc class_qual_type =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let procname = Procdesc.get_proc_name context.CContext.procdesc in
|
|
|
|
let name = CFrontend_config.this in
|
|
|
|
let pvar = Pvar.mk (Mangled.from_string name) procname in
|
|
|
|
let exp = Exp.Lvar pvar in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type context.CContext.tenv class_qual_type in
|
|
|
|
let exps = [(exp, typ)] in
|
|
|
|
(* there is no cast operation in AST, but backend needs it *)
|
|
|
|
dereference_value_from_result sil_loc {empty_res_trans with exps} ~strip_pointer:false
|
|
|
|
|
|
|
|
|
|
|
|
let cxxThisExpr_trans trans_state stmt_info expr_info =
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
this_expr_trans trans_state sil_loc expr_info.Clang_ast_t.ei_qual_type
|
|
|
|
|
|
|
|
|
|
|
|
let rec labelStmt_trans trans_state stmt_info stmt_list label_name =
|
|
|
|
let context = trans_state.context in
|
|
|
|
(* go ahead with the translation *)
|
|
|
|
let res_trans =
|
|
|
|
match stmt_list with [stmt] -> instruction trans_state stmt | _ -> assert false
|
|
|
|
(* expected a stmt or at most a compoundstmt *)
|
|
|
|
in
|
|
|
|
(* create the label root node into the hashtbl *)
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let root_node' = GotoLabel.find_goto_label trans_state.context label_name sil_loc in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc root_node' res_trans.root_nodes [] ;
|
|
|
|
{empty_res_trans with root_nodes= [root_node']; leaf_nodes= trans_state.succ_nodes}
|
|
|
|
|
|
|
|
|
|
|
|
and var_deref_trans trans_state stmt_info (decl_ref: Clang_ast_t.decl_ref) =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let _, _, qual_type = CAst_utils.get_info_from_decl_ref decl_ref in
|
|
|
|
let ast_typ = CType_decl.qual_type_to_sil_type context.tenv qual_type in
|
|
|
|
let typ =
|
|
|
|
match ast_typ.Typ.desc with
|
|
|
|
| Tstruct _ when decl_ref.dr_kind = `ParmVar ->
|
|
|
|
if CGeneral_utils.is_cpp_translation context.translation_unit_context then
|
|
|
|
Typ.mk (Tptr (ast_typ, Pk_reference))
|
|
|
|
else ast_typ
|
|
|
|
| _ ->
|
|
|
|
ast_typ
|
|
|
|
in
|
|
|
|
let procname = Procdesc.get_proc_name context.procdesc in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let pvar = CVar_decl.sil_var_of_decl_ref context decl_ref procname in
|
|
|
|
CContext.add_block_static_var context procname (pvar, typ) ;
|
|
|
|
let var_exp = Exp.Lvar pvar in
|
|
|
|
let exps =
|
|
|
|
if Self.is_var_self pvar (CContext.is_objc_method context) then
|
|
|
|
let class_typename = CContext.get_curr_class_typename context in
|
|
|
|
if CType.is_class typ then raise (Self.SelfClassException class_typename)
|
|
|
|
else
|
|
|
|
let typ = CType.add_pointer_to_typ (Typ.mk (Tstruct class_typename)) in
|
|
|
|
[(var_exp, typ)]
|
|
|
|
else [(var_exp, typ)]
|
|
|
|
in
|
|
|
|
L.(debug Capture Verbose) "@\n@\n PVAR ='%s'@\n@\n" (Pvar.to_string pvar) ;
|
|
|
|
let res_trans = {empty_res_trans with exps} in
|
|
|
|
match typ.desc with
|
|
|
|
| Tptr (_, Pk_reference) ->
|
|
|
|
(* dereference pvar due to the behavior of reference types in clang's AST *)
|
|
|
|
dereference_value_from_result sil_loc res_trans ~strip_pointer:false
|
|
|
|
| _ ->
|
|
|
|
res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and decl_ref_trans trans_state pre_trans_result stmt_info decl_ref ~is_constructor_init =
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
" priority node free = '%s'@\n@."
|
|
|
|
(string_of_bool (PriorityNode.is_priority_free trans_state)) ;
|
|
|
|
let decl_kind = decl_ref.Clang_ast_t.dr_kind in
|
|
|
|
match decl_kind with
|
|
|
|
| `EnumConstant ->
|
|
|
|
enum_constant_trans trans_state decl_ref
|
|
|
|
| `Function ->
|
|
|
|
function_deref_trans trans_state decl_ref
|
|
|
|
| `Var | `ImplicitParam | `ParmVar ->
|
|
|
|
var_deref_trans trans_state stmt_info decl_ref
|
|
|
|
| `Field | `ObjCIvar ->
|
|
|
|
field_deref_trans trans_state stmt_info pre_trans_result decl_ref ~is_constructor_init
|
|
|
|
| `CXXMethod | `CXXConversion | `CXXConstructor | `CXXDestructor ->
|
|
|
|
method_deref_trans trans_state pre_trans_result decl_ref stmt_info decl_kind
|
|
|
|
| _ ->
|
|
|
|
CFrontend_config.unimplemented
|
|
|
|
"Decl ref expression %a with pointer %d still needs to be translated"
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_decl_kind)
|
|
|
|
decl_kind decl_ref.Clang_ast_t.dr_decl_pointer
|
|
|
|
|
|
|
|
|
|
|
|
and declRefExpr_trans trans_state stmt_info decl_ref_expr_info _ =
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
" priority node free = '%s'@\n@."
|
|
|
|
(string_of_bool (PriorityNode.is_priority_free trans_state)) ;
|
|
|
|
let decl_ref =
|
|
|
|
match decl_ref_expr_info.Clang_ast_t.drti_decl_ref with
|
|
|
|
| Some dr ->
|
|
|
|
dr
|
|
|
|
| None ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
decl_ref_trans trans_state empty_res_trans stmt_info decl_ref ~is_constructor_init:false
|
|
|
|
|
|
|
|
|
|
|
|
(* evaluates an enum constant *)
|
|
|
|
and enum_const_eval context enum_constant_pointer prev_enum_constant_opt zero =
|
|
|
|
match CAst_utils.get_decl enum_constant_pointer with
|
|
|
|
| Some Clang_ast_t.EnumConstantDecl (_, _, _, enum_constant_decl_info) -> (
|
|
|
|
match enum_constant_decl_info.Clang_ast_t.ecdi_init_expr with
|
|
|
|
| Some stmt ->
|
|
|
|
expression_trans context stmt "WARNING: Expression in Enumeration constant not found@\n"
|
|
|
|
| None ->
|
|
|
|
match prev_enum_constant_opt with
|
|
|
|
| Some prev_constant_pointer ->
|
|
|
|
let previous_exp = get_enum_constant_expr context prev_constant_pointer in
|
|
|
|
CArithmetic_trans.sil_const_plus_one previous_exp
|
|
|
|
| None ->
|
|
|
|
zero )
|
|
|
|
| _ ->
|
|
|
|
zero
|
|
|
|
|
|
|
|
|
|
|
|
(* get the sil value of the enum constant from the map or by evaluating it *)
|
|
|
|
and get_enum_constant_expr context enum_constant_pointer =
|
|
|
|
let zero = Exp.Const (Const.Cint IntLit.zero) in
|
|
|
|
try
|
|
|
|
let prev_enum_constant_opt, sil_exp_opt =
|
|
|
|
CAst_utils.get_enum_constant_exp enum_constant_pointer
|
|
|
|
in
|
|
|
|
match sil_exp_opt with
|
|
|
|
| Some exp ->
|
|
|
|
exp
|
|
|
|
| None ->
|
|
|
|
let exp = enum_const_eval context enum_constant_pointer prev_enum_constant_opt zero in
|
|
|
|
CAst_utils.update_enum_map enum_constant_pointer exp ;
|
|
|
|
exp
|
|
|
|
with Not_found -> zero
|
|
|
|
|
|
|
|
|
|
|
|
and enum_constant_trans trans_state decl_ref =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let _, _, qual_type = CAst_utils.get_info_from_decl_ref decl_ref in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type context.CContext.tenv qual_type in
|
|
|
|
let const_exp = get_enum_constant_expr context decl_ref.Clang_ast_t.dr_decl_pointer in
|
|
|
|
{empty_res_trans with exps= [(const_exp, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
and arraySubscriptExpr_trans trans_state expr_info stmt_list =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info context.tenv in
|
|
|
|
let array_stmt, idx_stmt =
|
|
|
|
match stmt_list with
|
|
|
|
| [a; i] ->
|
|
|
|
(a, i)
|
|
|
|
(* Assumption: the statement list contains 2 elements, the first is the array expr and the
|
|
|
|
second the index *)
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
(* Let's get notified if the assumption is wrong...*)
|
|
|
|
in
|
|
|
|
let res_trans_a = instruction trans_state array_stmt in
|
|
|
|
let res_trans_idx = instruction trans_state idx_stmt in
|
|
|
|
let a_exp, _ =
|
|
|
|
extract_exp_from_list res_trans_a.exps
|
|
|
|
"WARNING: In ArraySubscriptExpr there was a problem in translating array exp.@\n"
|
|
|
|
in
|
|
|
|
let i_exp, _ =
|
|
|
|
extract_exp_from_list res_trans_idx.exps
|
|
|
|
"WARNING: In ArraySubscriptExpr there was a problem in translating index exp.@\n"
|
|
|
|
in
|
|
|
|
let array_exp = Exp.Lindex (a_exp, i_exp) in
|
|
|
|
let root_nodes =
|
|
|
|
if res_trans_a.root_nodes <> [] then res_trans_a.root_nodes else res_trans_idx.root_nodes
|
|
|
|
in
|
|
|
|
let leaf_nodes =
|
|
|
|
if res_trans_idx.leaf_nodes <> [] then res_trans_idx.leaf_nodes else res_trans_a.leaf_nodes
|
|
|
|
in
|
|
|
|
if res_trans_idx.root_nodes <> [] then
|
|
|
|
List.iter
|
|
|
|
~f:(fun n -> Procdesc.node_set_succs_exn context.procdesc n res_trans_idx.root_nodes [])
|
|
|
|
res_trans_a.leaf_nodes ;
|
|
|
|
(* Note the order of res_trans_idx.ids @ res_trans_a.ids is important. *)
|
|
|
|
(* We expect to use only res_trans_idx.ids in construction of other operation. *)
|
|
|
|
(* res_trans_a.ids is passed to be Removed.*)
|
|
|
|
{ empty_res_trans with
|
|
|
|
root_nodes
|
|
|
|
; leaf_nodes
|
|
|
|
; instrs= res_trans_a.instrs @ res_trans_idx.instrs
|
|
|
|
; exps= [(array_exp, typ)]
|
|
|
|
; initd_exps= res_trans_idx.initd_exps @ res_trans_a.initd_exps }
|
|
|
|
|
|
|
|
|
|
|
|
and binaryOperator_trans trans_state binary_operator_info stmt_info expr_info stmt_list =
|
|
|
|
let bok =
|
|
|
|
Clang_ast_j.string_of_binary_operator_kind binary_operator_info.Clang_ast_t.boi_kind
|
|
|
|
in
|
|
|
|
L.(debug Capture Verbose) " BinaryOperator '%s' " bok ;
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
" priority node free = '%s'@\n@."
|
|
|
|
(string_of_bool (PriorityNode.is_priority_free trans_state)) ;
|
|
|
|
let context = trans_state.context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let nname = "BinaryOperatorStmt: " ^ CArithmetic_trans.bin_op_to_string binary_operator_info in
|
|
|
|
let trans_state' = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let typ =
|
|
|
|
CType_decl.qual_type_to_sil_type context.CContext.tenv expr_info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
|
|
|
match stmt_list with
|
|
|
|
| [s1; s2] ->
|
|
|
|
(* Assumption: We expect precisely 2 stmt corresponding to the 2 operands*)
|
|
|
|
let rhs_owning_method = CTrans_utils.is_owning_method s2 in
|
|
|
|
(* NOTE: we create a node only if required. In that case this node *)
|
|
|
|
(* becomes the successor of the nodes that may be created when *)
|
|
|
|
(* translating the operands. *)
|
|
|
|
let res_trans_e1 = exec_with_self_exception instruction trans_state' s1 in
|
|
|
|
let var_exp, var_exp_typ =
|
|
|
|
extract_exp_from_list res_trans_e1.exps
|
|
|
|
"@\nWARNING: Missing LHS operand in BinOp. Returning -1. Fix needed...@\n"
|
|
|
|
in
|
|
|
|
let trans_state'' = {trans_state' with var_exp_typ= Some (var_exp, var_exp_typ)} in
|
|
|
|
let res_trans_e2 =
|
|
|
|
(* translation of s2 is done taking care of block special case *)
|
|
|
|
exec_with_block_priority_exception
|
|
|
|
(exec_with_self_exception instruction)
|
|
|
|
trans_state'' s2 stmt_info
|
|
|
|
in
|
|
|
|
let sil_e2, _ =
|
|
|
|
extract_exp_from_list res_trans_e2.exps
|
|
|
|
"@\nWARNING: Missing RHS operand in BinOp. Returning -1. Fix needed...@\n"
|
|
|
|
in
|
|
|
|
let binop_res_trans, exp_to_parent =
|
|
|
|
if List.exists ~f:(Exp.equal var_exp) res_trans_e2.initd_exps then ([], [])
|
|
|
|
else
|
|
|
|
let exp_op, instr_bin =
|
|
|
|
CArithmetic_trans.binary_operation_instruction binary_operator_info var_exp typ
|
|
|
|
sil_e2 sil_loc rhs_owning_method
|
|
|
|
in
|
|
|
|
(* Create a node if the priority if free and there are instructions *)
|
|
|
|
let creating_node =
|
|
|
|
PriorityNode.own_priority_node trans_state_pri.priority stmt_info
|
|
|
|
&& List.length instr_bin > 0
|
|
|
|
in
|
|
|
|
let extra_instrs, exp_to_parent =
|
|
|
|
if is_binary_assign_op binary_operator_info
|
|
|
|
(* assignment operator result is lvalue in CPP, rvalue in C, *)
|
|
|
|
(* hence the difference *)
|
|
|
|
&& not (CGeneral_utils.is_cpp_translation context.translation_unit_context)
|
|
|
|
&& (not creating_node || is_return_temp trans_state.continuation)
|
|
|
|
then
|
|
|
|
(* We are in this case when an assignment is inside *)
|
|
|
|
(* another operator that creates a node. Eg. another *)
|
|
|
|
(* assignment. *)
|
|
|
|
(* As no node is created here ids are passed to the parent *)
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
let res_instr = Sil.Load (id, var_exp, var_exp_typ, sil_loc) in
|
|
|
|
([res_instr], Exp.Var id)
|
|
|
|
else ([], exp_op)
|
|
|
|
in
|
|
|
|
let binop_res_trans = {empty_res_trans with instrs= instr_bin @ extra_instrs} in
|
|
|
|
([binop_res_trans], [(exp_to_parent, var_exp_typ)])
|
|
|
|
in
|
|
|
|
let all_res_trans = [res_trans_e1; res_trans_e2] @ binop_res_trans in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname stmt_info
|
|
|
|
all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans_to_parent with exps= exp_to_parent}
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
(* Binary operator should have two operands *)
|
|
|
|
and callExpr_trans trans_state si stmt_list expr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let fn_type_no_ref = CType_decl.get_type_from_expr_info expr_info context.CContext.tenv in
|
|
|
|
let function_type = add_reference_if_glvalue fn_type_no_ref expr_info in
|
|
|
|
let procname = Procdesc.get_proc_name context.CContext.procdesc in
|
|
|
|
let sil_loc = CLocation.get_sil_location si context in
|
|
|
|
(* First stmt is the function expr and the rest are params *)
|
|
|
|
let fun_exp_stmt, params_stmt =
|
|
|
|
match stmt_list with fe :: params -> (fe, params) | _ -> assert false
|
|
|
|
in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state si in
|
|
|
|
(* claim priority if no ancestors has claimed priority before *)
|
|
|
|
let trans_state_callee = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let res_trans_callee = instruction trans_state_callee fun_exp_stmt in
|
|
|
|
let sil_fe, _ =
|
|
|
|
extract_exp_from_list res_trans_callee.exps
|
|
|
|
"WARNING: The translation of fun_exp did not return an expression.Returning -1. NEED TO BE FIXED"
|
|
|
|
in
|
|
|
|
let callee_pname_opt =
|
|
|
|
match sil_fe with Exp.Const Const.Cfun pn -> Some pn | _ -> None
|
|
|
|
(* function pointer *)
|
|
|
|
in
|
|
|
|
(* we cannot translate the arguments of __builtin_object_size because preprocessing copies
|
|
|
|
them verbatim from a call to a different function, and they might be side-effecting *)
|
|
|
|
let should_translate_args =
|
|
|
|
not
|
|
|
|
(Option.value_map ~f:CTrans_models.is_builtin_object_size ~default:false callee_pname_opt)
|
|
|
|
in
|
|
|
|
let params_stmt = if should_translate_args then params_stmt else [] in
|
|
|
|
(* As we may have nodes coming from different parameters we need to *)
|
|
|
|
(* call instruction for each parameter and collect the results *)
|
|
|
|
(* afterwards. The 'instructions' function does not do that *)
|
|
|
|
let trans_state_param = {trans_state_pri with succ_nodes= []; var_exp_typ= None} in
|
|
|
|
let result_trans_subexprs =
|
|
|
|
let instruction' = exec_with_self_exception (exec_with_glvalue_as_reference instruction) in
|
|
|
|
let res_trans_p = List.map ~f:(instruction' trans_state_param) params_stmt in
|
|
|
|
res_trans_callee :: res_trans_p
|
|
|
|
in
|
|
|
|
match
|
|
|
|
Option.bind callee_pname_opt
|
|
|
|
~f:
|
|
|
|
(CTrans_utils.builtin_trans trans_state_pri sil_loc si function_type
|
|
|
|
result_trans_subexprs)
|
|
|
|
with
|
|
|
|
| Some builtin ->
|
|
|
|
builtin
|
|
|
|
| None ->
|
|
|
|
let is_cf_retain_release =
|
|
|
|
Option.value_map ~f:CTrans_models.is_cf_retain_release ~default:false callee_pname_opt
|
|
|
|
in
|
|
|
|
let act_params =
|
|
|
|
let params = List.tl_exn (collect_exprs result_trans_subexprs) in
|
|
|
|
if Int.equal (List.length params) (List.length params_stmt) then params
|
|
|
|
else
|
|
|
|
(* FIXME(t21762295) this is reachable *)
|
|
|
|
CFrontend_config.incorrect_assumption
|
|
|
|
"In call to %a: stmt_list and res_trans_par.exps must have same size but they don't:@\nstmt_list(%d)=[%a]@\nres_trans_par.exps(%d)=[%a]@\n"
|
|
|
|
Typ.Procname.pp procname (List.length params) (Pp.seq Exp.pp)
|
|
|
|
(List.map ~f:fst params) (List.length params_stmt)
|
|
|
|
(Pp.seq (Pp.to_string ~f:Clang_ast_j.string_of_stmt))
|
|
|
|
params_stmt
|
|
|
|
in
|
|
|
|
let act_params =
|
|
|
|
if is_cf_retain_release then
|
|
|
|
(Exp.Const (Const.Cint IntLit.one), Typ.mk (Tint Typ.IBool)) :: act_params
|
|
|
|
else act_params
|
|
|
|
in
|
|
|
|
let res_trans_call =
|
|
|
|
let cast_trans_fun = cast_trans act_params sil_loc function_type in
|
|
|
|
match Option.bind callee_pname_opt ~f:cast_trans_fun with
|
|
|
|
| Some (instr, cast_exp) ->
|
|
|
|
{empty_res_trans with instrs= [instr]; exps= [(cast_exp, function_type)]}
|
|
|
|
| _ ->
|
|
|
|
let is_call_to_block = objc_exp_of_type_block fun_exp_stmt in
|
|
|
|
let call_flags =
|
|
|
|
{CallFlags.default with CallFlags.cf_is_objc_block= is_call_to_block}
|
|
|
|
in
|
|
|
|
create_call_instr trans_state function_type sil_fe act_params sil_loc call_flags
|
|
|
|
~is_objc_method:false
|
|
|
|
in
|
|
|
|
let nname = "Call " ^ Exp.to_string sil_fe in
|
|
|
|
let all_res_trans = result_trans_subexprs @ [res_trans_call] in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname si all_res_trans
|
|
|
|
in
|
|
|
|
let add_cg_edge callee_pname = Cg.add_edge context.CContext.cg procname callee_pname in
|
|
|
|
Option.iter ~f:add_cg_edge callee_pname_opt ;
|
|
|
|
{res_trans_to_parent with exps= res_trans_call.exps}
|
|
|
|
|
|
|
|
|
|
|
|
and cxx_method_construct_call_trans trans_state_pri result_trans_callee params_stmt si
|
|
|
|
function_type is_cpp_call_virtual extra_res_trans =
|
|
|
|
let open CContext in
|
|
|
|
let context = trans_state_pri.context in
|
|
|
|
let procname = Procdesc.get_proc_name context.procdesc in
|
|
|
|
let sil_loc = CLocation.get_sil_location si context in
|
|
|
|
(* first for method address, second for 'this' expression and other parameters *)
|
|
|
|
assert (List.length result_trans_callee.exps >= 1) ;
|
|
|
|
let sil_method, _ = List.hd_exn result_trans_callee.exps in
|
|
|
|
let callee_pname =
|
|
|
|
match sil_method with
|
|
|
|
| Exp.Const Const.Cfun pn ->
|
|
|
|
pn
|
|
|
|
| _ ->
|
|
|
|
(* method pointer not implemented, this shouldn't happen but it does (t21762295) *)
|
|
|
|
CFrontend_config.incorrect_assumption "Could not resolve CXX method call %a" Exp.pp
|
|
|
|
sil_method
|
|
|
|
in
|
|
|
|
(* As we may have nodes coming from different parameters we need to call instruction for each
|
|
|
|
parameter and collect the results afterwards. The 'instructions' function does not do that *)
|
|
|
|
let result_trans_subexprs =
|
|
|
|
let trans_state_param = {trans_state_pri with succ_nodes= []; var_exp_typ= None} in
|
|
|
|
let instruction' = exec_with_self_exception (exec_with_glvalue_as_reference instruction) in
|
|
|
|
let res_trans_p = List.map ~f:(instruction' trans_state_param) params_stmt in
|
|
|
|
result_trans_callee :: res_trans_p
|
|
|
|
in
|
|
|
|
(* first expr is method address, rest are params including 'this' parameter *)
|
|
|
|
let actual_params = List.tl_exn (collect_exprs result_trans_subexprs) in
|
|
|
|
match cxx_method_builtin_trans trans_state_pri sil_loc result_trans_subexprs callee_pname with
|
|
|
|
| Some builtin ->
|
|
|
|
builtin
|
|
|
|
| _ ->
|
|
|
|
let call_flags = {CallFlags.default with CallFlags.cf_virtual= is_cpp_call_virtual} in
|
|
|
|
let res_trans_call =
|
|
|
|
create_call_instr trans_state_pri function_type sil_method actual_params sil_loc
|
|
|
|
call_flags ~is_objc_method:false
|
|
|
|
in
|
|
|
|
let nname = "Call " ^ Exp.to_string sil_method in
|
|
|
|
let all_res_trans = result_trans_subexprs @ [res_trans_call; extra_res_trans] in
|
|
|
|
let result_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname si all_res_trans
|
|
|
|
in
|
|
|
|
Cg.add_edge context.CContext.cg procname callee_pname ;
|
|
|
|
{result_trans_to_parent with exps= res_trans_call.exps}
|
|
|
|
|
|
|
|
|
|
|
|
and cxxMemberCallExpr_trans trans_state si stmt_list expr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
(* Structure is the following: *)
|
|
|
|
(* CXXMemberCallExpr: first stmt is method+this expr and the rest are normal params *)
|
|
|
|
(* CXXOperatorCallExpr: First stmt is method/function deref without this expr and the *)
|
|
|
|
(* rest are params, possibly including 'this' *)
|
|
|
|
let fun_exp_stmt, params_stmt =
|
|
|
|
match stmt_list with fe :: params -> (fe, params) | _ -> assert false
|
|
|
|
in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state si in
|
|
|
|
(* claim priority if no ancestors has claimed priority before *)
|
|
|
|
let trans_state_callee = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let result_trans_callee = instruction trans_state_callee fun_exp_stmt in
|
|
|
|
let is_cpp_call_virtual = result_trans_callee.is_cpp_call_virtual in
|
|
|
|
let fn_type_no_ref = CType_decl.get_type_from_expr_info expr_info context.CContext.tenv in
|
|
|
|
let function_type = add_reference_if_glvalue fn_type_no_ref expr_info in
|
|
|
|
cxx_method_construct_call_trans trans_state_pri result_trans_callee params_stmt si
|
|
|
|
function_type is_cpp_call_virtual empty_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and cxxConstructExpr_trans trans_state si params_stmt ei cxx_constr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state si in
|
|
|
|
let sil_loc = CLocation.get_sil_location si context in
|
|
|
|
let decl_ref = cxx_constr_info.Clang_ast_t.xcei_decl_ref in
|
|
|
|
let var_exp, class_type =
|
|
|
|
match trans_state.var_exp_typ with
|
|
|
|
| Some exp_typ ->
|
|
|
|
exp_typ
|
|
|
|
| None ->
|
|
|
|
let procdesc = trans_state.context.CContext.procdesc in
|
|
|
|
let pvar = Pvar.mk_tmp "__temp_construct_" (Procdesc.get_proc_name procdesc) in
|
|
|
|
let class_type = CType_decl.get_type_from_expr_info ei context.CContext.tenv in
|
|
|
|
Procdesc.append_locals procdesc [(Pvar.get_name pvar, class_type)] ;
|
|
|
|
(Exp.Lvar pvar, class_type)
|
|
|
|
in
|
|
|
|
let this_type = CType.add_pointer_to_typ class_type in
|
|
|
|
let this_res_trans =
|
|
|
|
{empty_res_trans with exps= [(var_exp, this_type)]; initd_exps= [var_exp]}
|
|
|
|
in
|
|
|
|
let tmp_res_trans = {empty_res_trans with exps= [(var_exp, class_type)]} in
|
|
|
|
(* When class type is translated as pointer (std::shared_ptr for example), there needs
|
|
|
|
to be extra Load instruction before returning the trans_result of constructorExpr.
|
|
|
|
There is no LValueToRvalue cast in the AST afterwards since clang doesn't know
|
|
|
|
that class type is translated as pointer type. It gets added here instead. *)
|
|
|
|
let extra_res_trans =
|
|
|
|
match class_type.desc with
|
|
|
|
| Typ.Tptr _ ->
|
|
|
|
dereference_value_from_result sil_loc tmp_res_trans ~strip_pointer:false
|
|
|
|
| _ ->
|
|
|
|
tmp_res_trans
|
|
|
|
in
|
|
|
|
let res_trans_callee =
|
|
|
|
decl_ref_trans trans_state this_res_trans si decl_ref ~is_constructor_init:false
|
|
|
|
in
|
|
|
|
let res_trans =
|
|
|
|
cxx_method_construct_call_trans trans_state_pri res_trans_callee params_stmt si
|
|
|
|
(Typ.mk Tvoid) false extra_res_trans
|
|
|
|
in
|
|
|
|
{res_trans with exps= extra_res_trans.exps}
|
|
|
|
|
|
|
|
|
|
|
|
and cxx_destructor_call_trans trans_state si this_res_trans class_type_ptr ~is_inner_destructor =
|
|
|
|
(* cxx_method_construct_call_trans claims a priority with the same `si`.
|
|
|
|
New pointer is generated to avoid premature node creation *)
|
|
|
|
let si' = {si with Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()} in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state si' in
|
|
|
|
let this_exp, this_typ =
|
|
|
|
extract_exp_from_list this_res_trans.exps
|
|
|
|
"WARNING: There should be one expression for 'this' in constructor. @\n"
|
|
|
|
in
|
|
|
|
let this_res_trans' =
|
|
|
|
{this_res_trans with exps= [(this_exp, CType.add_pointer_to_typ this_typ)]}
|
|
|
|
in
|
|
|
|
let res_trans_callee =
|
|
|
|
destructor_deref_trans trans_state this_res_trans' class_type_ptr si' ~is_inner_destructor
|
|
|
|
in
|
|
|
|
let is_cpp_call_virtual = res_trans_callee.is_cpp_call_virtual in
|
|
|
|
if res_trans_callee.exps <> [] then
|
|
|
|
cxx_method_construct_call_trans trans_state_pri res_trans_callee [] si' (Typ.mk Tvoid)
|
|
|
|
is_cpp_call_virtual empty_res_trans
|
|
|
|
else empty_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and objCMessageExpr_trans_special_cases trans_state si obj_c_message_expr_info method_type
|
|
|
|
trans_state_pri sil_loc act_params =
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let context = trans_state.context in
|
|
|
|
let receiver_kind = obj_c_message_expr_info.Clang_ast_t.omei_receiver_kind in
|
|
|
|
let selector = obj_c_message_expr_info.Clang_ast_t.omei_selector in
|
|
|
|
(* class method *)
|
|
|
|
if String.equal selector CFrontend_config.class_method && CType.is_class method_type then
|
|
|
|
let class_name =
|
|
|
|
CMethod_trans.get_class_name_method_call_from_receiver_kind context obj_c_message_expr_info
|
|
|
|
act_params
|
|
|
|
in
|
|
|
|
(* alloc or new *)
|
|
|
|
(* FIXME(t21762295): we do not expect this to propagate to the top but it does *)
|
|
|
|
raise (Self.SelfClassException class_name)
|
|
|
|
else if String.equal selector CFrontend_config.alloc
|
|
|
|
|| String.equal selector CFrontend_config.new_str
|
|
|
|
then
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
match receiver_kind with
|
|
|
|
| `Class qual_type ->
|
|
|
|
let class_opt =
|
|
|
|
CMethod_trans.get_class_name_method_call_from_clang context.translation_unit_context
|
|
|
|
context.CContext.tenv obj_c_message_expr_info
|
|
|
|
in
|
|
|
|
Some (new_or_alloc_trans trans_state_pri sil_loc si qual_type class_opt selector)
|
|
|
|
| _ ->
|
|
|
|
None (* assertions *)
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
else if CTrans_models.is_handleFailureInMethod selector then
|
|
|
|
Some (CTrans_utils.trans_assertion trans_state sil_loc)
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
else None
|
|
|
|
|
|
|
|
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
(* If the first argument of the call is self in a static context, remove it as an argument *)
|
|
|
|
(* and change the call from instance to static *)
|
|
|
|
and objCMessageExpr_deal_with_static_self trans_state_param stmt_list obj_c_message_expr_info =
|
|
|
|
match stmt_list with
|
|
|
|
| stmt :: rest ->
|
|
|
|
let obj_c_message_expr_info, fst_res_trans =
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
try
|
|
|
|
let fst_res_trans = instruction trans_state_param stmt in
|
|
|
|
(obj_c_message_expr_info, fst_res_trans)
|
|
|
|
with Self.SelfClassException class_typename ->
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let pointer = obj_c_message_expr_info.Clang_ast_t.omei_decl_pointer in
|
|
|
|
let selector = obj_c_message_expr_info.Clang_ast_t.omei_selector in
|
|
|
|
let obj_c_message_expr_info =
|
|
|
|
Ast_expressions.make_obj_c_message_expr_info_class selector class_typename pointer
|
|
|
|
in
|
|
|
|
(obj_c_message_expr_info, empty_res_trans)
|
|
|
|
in
|
|
|
|
let instruction' = exec_with_self_exception (exec_with_glvalue_as_reference instruction) in
|
|
|
|
let l = List.map ~f:(instruction' trans_state_param) rest in
|
|
|
|
(obj_c_message_expr_info, fst_res_trans :: l)
|
|
|
|
| [] ->
|
|
|
|
(obj_c_message_expr_info, [empty_res_trans])
|
|
|
|
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
|
|
|
|
and objCMessageExpr_trans trans_state si obj_c_message_expr_info stmt_list expr_info =
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
" priority node free = '%s'@\n@."
|
|
|
|
(string_of_bool (PriorityNode.is_priority_free trans_state)) ;
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location si context in
|
|
|
|
let method_type_no_ref = CType_decl.get_type_from_expr_info expr_info context.CContext.tenv in
|
|
|
|
let method_type = add_reference_if_glvalue method_type_no_ref expr_info in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state si in
|
|
|
|
let trans_state_param = {trans_state_pri with succ_nodes= []; var_exp_typ= None} in
|
|
|
|
let obj_c_message_expr_info, res_trans_subexpr_list =
|
|
|
|
objCMessageExpr_deal_with_static_self trans_state_param stmt_list obj_c_message_expr_info
|
|
|
|
in
|
|
|
|
let subexpr_exprs = collect_exprs res_trans_subexpr_list in
|
|
|
|
match
|
|
|
|
objCMessageExpr_trans_special_cases trans_state si obj_c_message_expr_info method_type
|
|
|
|
trans_state_pri sil_loc subexpr_exprs
|
|
|
|
with
|
|
|
|
| Some res ->
|
|
|
|
res
|
|
|
|
| None ->
|
|
|
|
let procname = Procdesc.get_proc_name context.CContext.procdesc in
|
|
|
|
let callee_name, method_call_type =
|
|
|
|
get_callee_objc_method context obj_c_message_expr_info subexpr_exprs
|
|
|
|
in
|
|
|
|
let res_trans_add_self =
|
|
|
|
Self.add_self_parameter_for_super_instance context procname sil_loc
|
|
|
|
obj_c_message_expr_info
|
|
|
|
in
|
|
|
|
let res_trans_subexpr_list = res_trans_add_self :: res_trans_subexpr_list in
|
|
|
|
let subexpr_exprs = collect_exprs res_trans_subexpr_list in
|
|
|
|
let is_virtual =
|
|
|
|
CMethod_trans.equal_method_call_type method_call_type CMethod_trans.MCVirtual
|
|
|
|
in
|
|
|
|
Cg.add_edge context.CContext.cg procname callee_name ;
|
|
|
|
let param_exps, instr_block_param =
|
|
|
|
extract_block_from_tuple procname subexpr_exprs sil_loc
|
|
|
|
in
|
|
|
|
let res_trans_block = {empty_res_trans with instrs= instr_block_param} in
|
|
|
|
let call_flags = {CallFlags.default with CallFlags.cf_virtual= is_virtual} in
|
|
|
|
let method_sil = Exp.Const (Const.Cfun callee_name) in
|
|
|
|
let res_trans_call =
|
|
|
|
create_call_instr trans_state method_type method_sil param_exps sil_loc call_flags
|
|
|
|
~is_objc_method:true
|
|
|
|
in
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let selector = obj_c_message_expr_info.Clang_ast_t.omei_selector in
|
|
|
|
let nname = "Message Call: " ^ selector in
|
|
|
|
let all_res_trans = res_trans_subexpr_list @ [res_trans_block; res_trans_call] in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname si all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans_to_parent with exps= res_trans_call.exps}
|
|
|
|
|
|
|
|
|
|
|
|
and dispatch_function_trans trans_state stmt_info stmt_list n =
|
|
|
|
L.(debug Capture Verbose) "@\n Call to a dispatch function treated as special case...@\n" ;
|
|
|
|
let transformed_stmt = Ast_expressions.translate_dispatch_function stmt_info stmt_list n in
|
|
|
|
instruction trans_state transformed_stmt
|
|
|
|
|
|
|
|
|
|
|
|
and block_enumeration_trans trans_state stmt_info stmt_list ei =
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
"@\n Call to a block enumeration function treated as special case...@\n@." ;
|
|
|
|
let procname = Procdesc.get_proc_name trans_state.context.CContext.procdesc in
|
|
|
|
let pvar = CProcname.get_next_block_pvar procname in
|
|
|
|
let transformed_stmt, _ =
|
|
|
|
Ast_expressions.translate_block_enumerate (Pvar.to_string pvar) stmt_info stmt_list ei
|
|
|
|
in
|
|
|
|
instruction trans_state transformed_stmt
|
|
|
|
|
|
|
|
|
|
|
|
and compute_this_for_destructor_calls trans_state stmt_info class_ptr =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let class_qual_type = CAst_utils.qual_type_of_decl_ptr class_ptr in
|
|
|
|
let this_res_trans =
|
|
|
|
this_expr_trans trans_state sil_loc
|
|
|
|
(Ast_expressions.create_pointer_qual_type class_qual_type)
|
|
|
|
in
|
|
|
|
let obj_sil, class_typ =
|
|
|
|
extract_exp_from_list this_res_trans.exps
|
|
|
|
"WARNING: There should be one expression for 'this' in constructor. @\n"
|
|
|
|
in
|
|
|
|
let this_qual_type = match class_typ.desc with Typ.Tptr (t, _) -> t | _ -> class_typ in
|
|
|
|
(obj_sil, this_qual_type, this_res_trans)
|
|
|
|
|
|
|
|
|
|
|
|
and inject_base_class_destructor_calls trans_state stmt_info bases obj_sil this_qual_type =
|
|
|
|
List.rev_map bases ~f:(fun base ->
|
|
|
|
let this_res_trans_destruct = {empty_res_trans with exps= [(obj_sil, this_qual_type)]} in
|
|
|
|
cxx_destructor_call_trans trans_state stmt_info this_res_trans_destruct base
|
|
|
|
~is_inner_destructor:true )
|
|
|
|
|
|
|
|
|
|
|
|
and add_this_instrs_if_result_empty res_trans this_res_trans =
|
|
|
|
let all_res_trans = List.filter ~f:(fun res -> res <> empty_res_trans) res_trans in
|
|
|
|
let all_res_trans =
|
|
|
|
if all_res_trans <> [] then
|
|
|
|
{empty_res_trans with instrs= this_res_trans.instrs} :: all_res_trans
|
|
|
|
else all_res_trans
|
|
|
|
in
|
|
|
|
all_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and cxx_inject_virtual_base_class_destructors trans_state stmt_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
if not (CGeneral_utils.is_cpp_translation context.translation_unit_context) then
|
|
|
|
empty_res_trans
|
|
|
|
else
|
|
|
|
(* get virtual base classes of the current class *)
|
|
|
|
let class_ptr = CContext.get_curr_class_decl_ptr context.CContext.curr_class in
|
|
|
|
let decl = Option.value_exn (CAst_utils.get_decl class_ptr) in
|
|
|
|
let typ_pointer_opt = CAst_utils.type_of_decl decl in
|
|
|
|
let bases = CAst_utils.get_cxx_virtual_base_classes decl in
|
|
|
|
let bases = match typ_pointer_opt with Some p -> bases @ [p] | None -> bases in
|
|
|
|
let _, sloc2 = stmt_info.Clang_ast_t.si_source_range in
|
|
|
|
let stmt_info_loc = {stmt_info with Clang_ast_t.si_source_range= (sloc2, sloc2)} in
|
|
|
|
(* compute `this` once that is used for all destructor calls of virtual base class *)
|
|
|
|
let obj_sil, this_qual_type, this_res_trans =
|
|
|
|
compute_this_for_destructor_calls trans_state stmt_info_loc class_ptr
|
|
|
|
in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info_loc in
|
|
|
|
let bases_res_trans =
|
|
|
|
inject_base_class_destructor_calls trans_state_pri stmt_info_loc bases obj_sil
|
|
|
|
this_qual_type
|
|
|
|
in
|
|
|
|
let all_res_trans = add_this_instrs_if_result_empty bases_res_trans this_res_trans in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info_loc context in
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "Destruction" stmt_info_loc
|
|
|
|
all_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and cxx_inject_field_destructors_in_destructor_body trans_state stmt_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
if not (CGeneral_utils.is_cpp_translation context.translation_unit_context) then
|
|
|
|
empty_res_trans
|
|
|
|
else
|
|
|
|
(* get fields and base classes of the current class *)
|
|
|
|
let class_ptr = CContext.get_curr_class_decl_ptr context.CContext.curr_class in
|
|
|
|
let decl = Option.value_exn (CAst_utils.get_decl class_ptr) in
|
|
|
|
let fields = CAst_utils.get_record_fields decl in
|
|
|
|
let bases = CAst_utils.get_cxx_base_classes decl in
|
|
|
|
let _, sloc2 = stmt_info.Clang_ast_t.si_source_range in
|
|
|
|
let stmt_info_loc = {stmt_info with Clang_ast_t.si_source_range= (sloc2, sloc2)} in
|
|
|
|
(* compute `this` once that is used for all destructors of fields and base classes *)
|
|
|
|
let obj_sil, this_qual_type, this_res_trans =
|
|
|
|
compute_this_for_destructor_calls trans_state stmt_info_loc class_ptr
|
|
|
|
in
|
|
|
|
(* ReturnStmt claims a priority with the same `stmt_info`.
|
|
|
|
New pointer is generated to avoid premature node creation *)
|
|
|
|
let stmt_info' =
|
|
|
|
{stmt_info_loc with Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()}
|
|
|
|
in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info' in
|
|
|
|
let all_res_trans =
|
|
|
|
List.rev_map fields ~f:(function
|
|
|
|
| Clang_ast_t.FieldDecl ({di_parent_pointer}, {ni_name}, qual_type, _) ->
|
|
|
|
let class_tname =
|
|
|
|
match CAst_utils.get_decl_opt di_parent_pointer with
|
|
|
|
| Some decl ->
|
|
|
|
CType_decl.get_record_typename ~tenv:context.tenv decl
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
let field_name = CGeneral_utils.mk_class_field_name class_tname ni_name in
|
|
|
|
let field_exp = Exp.Lfield (obj_sil, field_name, this_qual_type) in
|
|
|
|
let field_typ = CType_decl.qual_type_to_sil_type context.tenv qual_type in
|
|
|
|
let this_res_trans_destruct =
|
|
|
|
{empty_res_trans with exps= [(field_exp, field_typ)]}
|
|
|
|
in
|
|
|
|
cxx_destructor_call_trans trans_state_pri stmt_info_loc this_res_trans_destruct
|
|
|
|
qual_type.Clang_ast_t.qt_type_ptr ~is_inner_destructor:false
|
|
|
|
| _ ->
|
|
|
|
assert false )
|
|
|
|
in
|
|
|
|
let bases_res_trans =
|
|
|
|
inject_base_class_destructor_calls trans_state_pri stmt_info_loc bases obj_sil
|
|
|
|
this_qual_type
|
|
|
|
in
|
|
|
|
let all_res_trans =
|
|
|
|
add_this_instrs_if_result_empty (all_res_trans @ bases_res_trans) this_res_trans
|
|
|
|
in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "Destruction" stmt_info'
|
|
|
|
all_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and inject_destructors trans_state stmt_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
if not (CGeneral_utils.is_cpp_translation context.translation_unit_context) then
|
|
|
|
empty_res_trans
|
|
|
|
else
|
|
|
|
let procname = Procdesc.get_proc_name context.CContext.procdesc in
|
|
|
|
(* The source location of destructor should reflect the end of the statement *)
|
|
|
|
let _, sloc2 = stmt_info.Clang_ast_t.si_source_range in
|
|
|
|
let stmt_info_loc = {stmt_info with Clang_ast_t.si_source_range= (sloc2, sloc2)} in
|
|
|
|
(* ReturnStmt claims a priority with the same `stmt_info`.
|
|
|
|
New pointer is generated to avoid premature node creation *)
|
|
|
|
let stmt_info' =
|
|
|
|
{stmt_info_loc with Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()}
|
|
|
|
in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info' in
|
|
|
|
let all_res_trans =
|
|
|
|
try
|
|
|
|
let map = context.CContext.vars_to_destroy in
|
|
|
|
let vars_to_destroy = CContext.StmtMap.find_exn map stmt_info.Clang_ast_t.si_pointer in
|
|
|
|
List.map
|
|
|
|
~f:(function
|
|
|
|
| Clang_ast_t.VarDecl (_, _, qual_type, _) as decl ->
|
|
|
|
let pvar = CVar_decl.sil_var_of_decl context decl procname in
|
|
|
|
let exp = Exp.Lvar pvar in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type context.CContext.tenv qual_type in
|
|
|
|
let this_res_trans_destruct = {empty_res_trans with exps= [(exp, typ)]} in
|
|
|
|
cxx_destructor_call_trans trans_state_pri stmt_info_loc this_res_trans_destruct
|
|
|
|
qual_type.Clang_ast_t.qt_type_ptr ~is_inner_destructor:false
|
|
|
|
| _ ->
|
|
|
|
assert false)
|
|
|
|
vars_to_destroy
|
|
|
|
with Not_found ->
|
|
|
|
L.(debug Capture Verbose) "@\n Variables that go out of scope are not found...@\n@." ;
|
|
|
|
[]
|
|
|
|
in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "Destruction" stmt_info'
|
|
|
|
all_res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and compoundStmt_trans trans_state stmt_info stmt_list =
|
|
|
|
(* Computing destructor call nodes to inject at the end of the compound statement,
|
|
|
|
except if the statement ends with Return statemenent *)
|
|
|
|
let destr_trans_result =
|
|
|
|
match List.last stmt_list with
|
|
|
|
| Some Clang_ast_t.ReturnStmt _ ->
|
|
|
|
empty_res_trans
|
|
|
|
| _ ->
|
|
|
|
inject_destructors trans_state stmt_info
|
|
|
|
in
|
|
|
|
(* Injecting destructor call nodes at the end of the compound statement *)
|
|
|
|
let succ_nodes =
|
|
|
|
if destr_trans_result.root_nodes <> [] then destr_trans_result.root_nodes
|
|
|
|
else trans_state.succ_nodes
|
|
|
|
in
|
|
|
|
let trans_state' = {trans_state with succ_nodes} in
|
|
|
|
instructions trans_state' stmt_list
|
|
|
|
|
|
|
|
|
|
|
|
and conditionalOperator_trans trans_state stmt_info stmt_list expr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let succ_nodes = trans_state.succ_nodes in
|
|
|
|
let procdesc = context.CContext.procdesc in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let do_branch branch stmt var_typ prune_nodes join_node pvar =
|
|
|
|
let trans_state_pri = PriorityNode.force_claim_priority_node trans_state stmt_info in
|
|
|
|
let trans_state' = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let res_trans_b = instruction trans_state' stmt in
|
|
|
|
let e', _ =
|
|
|
|
extract_exp_from_list res_trans_b.exps
|
|
|
|
"@\nWARNING: Missing branch expression for Conditional operator. Need to be fixed@\n"
|
|
|
|
in
|
|
|
|
let set_temp_var = [Sil.Store (Exp.Lvar pvar, var_typ, e', sil_loc)] in
|
|
|
|
let tmp_var_res_trans = {empty_res_trans with instrs= set_temp_var} in
|
|
|
|
let trans_state'' = {trans_state' with succ_nodes= [join_node]} in
|
|
|
|
let all_res_trans = [res_trans_b; tmp_var_res_trans] in
|
|
|
|
let res_trans =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state'' sil_loc "ConditinalStmt Branch"
|
|
|
|
stmt_info all_res_trans
|
|
|
|
in
|
|
|
|
let prune_nodes_t, prune_nodes_f = List.partition_tf ~f:is_true_prune_node prune_nodes in
|
|
|
|
let prune_nodes' = if branch then prune_nodes_t else prune_nodes_f in
|
|
|
|
List.iter
|
|
|
|
~f:(fun n -> Procdesc.node_set_succs_exn context.procdesc n res_trans.root_nodes [])
|
|
|
|
prune_nodes'
|
|
|
|
in
|
|
|
|
match stmt_list with
|
|
|
|
| [cond; exp1; exp2] ->
|
|
|
|
let typ =
|
|
|
|
CType_decl.qual_type_to_sil_type context.CContext.tenv expr_info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
|
|
|
let var_typ = add_reference_if_glvalue typ expr_info in
|
|
|
|
let join_node = create_node Procdesc.Node.Join_node [] sil_loc context in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc join_node succ_nodes [] ;
|
|
|
|
let pvar = mk_temp_sil_var procdesc "SIL_temp_conditional___" in
|
|
|
|
Procdesc.append_locals procdesc [(Pvar.get_name pvar, var_typ)] ;
|
|
|
|
let continuation' = mk_cond_continuation trans_state.continuation in
|
|
|
|
let trans_state' = {trans_state with continuation= continuation'; succ_nodes= []} in
|
|
|
|
let res_trans_cond =
|
|
|
|
exec_with_priority_exception trans_state' cond (cond_trans ~negate_cond:false)
|
|
|
|
in
|
|
|
|
(* Note: by contruction prune nodes are leafs_nodes_cond *)
|
|
|
|
do_branch true exp1 var_typ res_trans_cond.leaf_nodes join_node pvar ;
|
|
|
|
do_branch false exp2 var_typ res_trans_cond.leaf_nodes join_node pvar ;
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
let instrs = [Sil.Load (id, Exp.Lvar pvar, var_typ, sil_loc)] in
|
|
|
|
{ empty_res_trans with
|
|
|
|
root_nodes= res_trans_cond.root_nodes
|
|
|
|
; leaf_nodes= [join_node]
|
|
|
|
; instrs
|
|
|
|
; exps= [(Exp.Var id, typ)]
|
|
|
|
; initd_exps= [] (* TODO we should get exps from branches+cond *) }
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
(* The GNU extension to the conditional operator which allows the middle operand to be omitted. *)
|
|
|
|
and binaryConditionalOperator_trans trans_state stmt_info stmt_list expr_info =
|
|
|
|
match stmt_list with
|
|
|
|
| [stmt1; ostmt1; ostmt2; stmt2]
|
|
|
|
when contains_opaque_value_expr ostmt1 && contains_opaque_value_expr ostmt2 ->
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
let trans_state_pri = PriorityNode.force_claim_priority_node trans_state stmt_info in
|
|
|
|
let trans_state_cond =
|
|
|
|
{trans_state_pri with continuation= mk_cond_continuation trans_state_pri.continuation}
|
|
|
|
in
|
|
|
|
(* evaluate stmt1 once. Then, use it as replacement for OpaqueValueExpr*)
|
|
|
|
(* when translating ostmt1 and ostmt2 *)
|
|
|
|
let init_res_trans = instruction trans_state_cond stmt1 in
|
|
|
|
let opaque_exp = extract_exp_from_list init_res_trans.exps "" in
|
|
|
|
let trans_state' = {trans_state_pri with opaque_exp= Some opaque_exp} in
|
|
|
|
let op_res_trans =
|
|
|
|
conditionalOperator_trans trans_state' stmt_info [ostmt1; ostmt2; stmt2] expr_info
|
|
|
|
in
|
|
|
|
let trans_state'' = {trans_state_cond with succ_nodes= op_res_trans.root_nodes} in
|
|
|
|
let init_res_trans' =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state'' sil_loc "BinaryConditinalStmt Init"
|
|
|
|
stmt_info [init_res_trans]
|
|
|
|
in
|
|
|
|
let root_nodes = init_res_trans'.root_nodes in
|
|
|
|
let root_nodes' = if root_nodes <> [] then root_nodes else op_res_trans.root_nodes in
|
|
|
|
{op_res_trans with root_nodes= root_nodes'}
|
|
|
|
| _ ->
|
|
|
|
CFrontend_config.unimplemented "BinaryConditionalOperator not translated"
|
|
|
|
|
|
|
|
|
|
|
|
(* Translate a condition for if/loops statement. It shorts-circuit and/or. *)
|
|
|
|
(* The invariant is that the translation of a condition always contains (at least) *)
|
|
|
|
(* the prune nodes. Moreover these are always the leaf nodes of the translation. *)
|
|
|
|
and cond_trans ~negate_cond trans_state cond =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let si, _ = Clang_ast_proj.get_stmt_tuple cond in
|
|
|
|
let sil_loc = CLocation.get_sil_location si context in
|
|
|
|
let mk_prune_node ~branch ~negate_cond e ins =
|
|
|
|
create_prune_node ~branch ~negate_cond e ins sil_loc Sil.Ik_if context
|
|
|
|
in
|
|
|
|
let extract_exp el =
|
|
|
|
extract_exp_from_list el
|
|
|
|
"@\nWARNING: Missing expression for Conditional operator. Need to be fixed"
|
|
|
|
in
|
|
|
|
(* this function translate cond without doing shortcircuit *)
|
|
|
|
let no_short_circuit_cond ~is_cmp =
|
|
|
|
L.(debug Capture Verbose) " No short-circuit condition@\n" ;
|
|
|
|
let res_trans_cond =
|
|
|
|
if is_null_stmt cond then
|
|
|
|
{ empty_res_trans with
|
|
|
|
exps= [(Exp.Const (Const.Cint IntLit.one), Typ.mk (Tint Typ.IBool))] }
|
|
|
|
(* Assumption: If it's a null_stmt, it is a loop with no bound, so we set condition to 1 *)
|
|
|
|
else if is_cmp then
|
|
|
|
let open Clang_ast_t in
|
|
|
|
(* If we have a comparision here, do not dispatch it to `instruction` function, which
|
|
|
|
* invokes binaryOperator_trans_with_cond -> conditionalOperator_trans -> cond_trans.
|
|
|
|
* This will throw the translation process into an infinite loop immediately.
|
|
|
|
* Instead, dispatch to binaryOperator_trans directly. *)
|
|
|
|
(* If one wants to add a new kind of `BinaryOperator` that will have the same behavior,
|
|
|
|
* she need to change both the codes here and the `match` in
|
|
|
|
* binaryOperator_trans_with_cond *)
|
|
|
|
match cond with
|
|
|
|
| BinaryOperator (si, ss, ei, boi) ->
|
|
|
|
binaryOperator_trans trans_state boi si ei ss
|
|
|
|
| _ ->
|
|
|
|
instruction trans_state cond
|
|
|
|
else instruction trans_state cond
|
|
|
|
in
|
|
|
|
let e', instrs' =
|
|
|
|
define_condition_side_effects res_trans_cond.exps res_trans_cond.instrs sil_loc
|
|
|
|
in
|
|
|
|
let prune_t = mk_prune_node ~branch:true ~negate_cond e' instrs' in
|
|
|
|
let prune_f = mk_prune_node ~branch:false ~negate_cond:(not negate_cond) e' instrs' in
|
|
|
|
List.iter
|
|
|
|
~f:(fun n' -> Procdesc.node_set_succs_exn context.procdesc n' [prune_t; prune_f] [])
|
|
|
|
res_trans_cond.leaf_nodes ;
|
|
|
|
let rnodes =
|
|
|
|
if List.is_empty res_trans_cond.root_nodes then [prune_t; prune_f]
|
|
|
|
else res_trans_cond.root_nodes
|
|
|
|
in
|
|
|
|
{ empty_res_trans with
|
|
|
|
root_nodes= rnodes; leaf_nodes= [prune_t; prune_f]; instrs= instrs'; exps= e' }
|
|
|
|
in
|
|
|
|
(* This function translate (s1 binop s2) doing shortcircuit for '&&' and '||' *)
|
|
|
|
(* At the high level it does cond_trans s1; cond_trans s2; glue_nodes *)
|
|
|
|
(* The glue_nodes partitions the prune nodes of s1's translation.*)
|
|
|
|
(* Some of them need to go to the statement to be executed after the *)
|
|
|
|
(* condition (prune_to_short_c) and others to the root nodes of the *)
|
|
|
|
(* translation of s2 (i.e., the case when we need to fully evaluate*)
|
|
|
|
(* the condition to decide its truth value). *)
|
|
|
|
let short_circuit binop s1 s2 =
|
|
|
|
let res_trans_s1 = cond_trans ~negate_cond trans_state s1 in
|
|
|
|
let prune_nodes_t, prune_nodes_f =
|
|
|
|
List.partition_tf ~f:is_true_prune_node res_trans_s1.leaf_nodes
|
|
|
|
in
|
|
|
|
let res_trans_s2 = cond_trans ~negate_cond trans_state s2 in
|
|
|
|
(* prune_to_s2 is the prune node that is connected with the root node of the *)
|
|
|
|
(* translation of s2.*)
|
|
|
|
(* prune_to_short_c is the prune node that is connected directly with the branch *)
|
|
|
|
(* where the control flow goes in case of short circuit *)
|
|
|
|
let prune_to_s2, prune_to_short_c =
|
|
|
|
match binop with
|
|
|
|
| Binop.LAnd ->
|
|
|
|
(prune_nodes_t, prune_nodes_f)
|
|
|
|
| Binop.LOr ->
|
|
|
|
(prune_nodes_f, prune_nodes_t)
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
List.iter
|
|
|
|
~f:(fun n -> Procdesc.node_set_succs_exn context.procdesc n res_trans_s2.root_nodes [])
|
|
|
|
prune_to_s2 ;
|
|
|
|
let root_nodes_to_parent =
|
|
|
|
if List.is_empty res_trans_s1.root_nodes then res_trans_s1.leaf_nodes
|
|
|
|
else res_trans_s1.root_nodes
|
|
|
|
in
|
|
|
|
let exp1, typ1 = extract_exp res_trans_s1.exps in
|
|
|
|
let exp2, _ = extract_exp res_trans_s2.exps in
|
|
|
|
let e_cond = Exp.BinOp (binop, exp1, exp2) in
|
|
|
|
{ empty_res_trans with
|
|
|
|
root_nodes= root_nodes_to_parent
|
|
|
|
; leaf_nodes= prune_to_short_c @ res_trans_s2.leaf_nodes
|
|
|
|
; instrs= res_trans_s1.instrs @ res_trans_s2.instrs
|
|
|
|
; exps= [(e_cond, typ1)] }
|
|
|
|
in
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
"Translating Condition for If-then-else/Loop/Conditional Operator @\n" ;
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match cond with
|
|
|
|
| BinaryOperator (_, [s1; s2], _, boi) -> (
|
|
|
|
match boi.boi_kind with
|
|
|
|
| `LAnd ->
|
|
|
|
short_circuit (if negate_cond then Binop.LOr else Binop.LAnd) s1 s2
|
|
|
|
| `LOr ->
|
|
|
|
short_circuit (if negate_cond then Binop.LAnd else Binop.LOr) s1 s2
|
|
|
|
| `LT | `GT | `LE | `GE | `EQ | `NE ->
|
|
|
|
no_short_circuit_cond ~is_cmp:true
|
|
|
|
| _ ->
|
|
|
|
no_short_circuit_cond ~is_cmp:false )
|
|
|
|
| ParenExpr (_, [s], _) ->
|
|
|
|
(* condition can be wrapped in parenthesys *)
|
|
|
|
cond_trans ~negate_cond trans_state s
|
|
|
|
| UnaryOperator (_, [s], _, {uoi_kind= `LNot}) ->
|
|
|
|
cond_trans ~negate_cond:(not negate_cond) trans_state s
|
|
|
|
| _ ->
|
|
|
|
no_short_circuit_cond ~is_cmp:false
|
|
|
|
|
|
|
|
|
|
|
|
and declStmt_in_condition_trans trans_state decl_stmt res_trans_cond =
|
|
|
|
match decl_stmt with
|
|
|
|
| Clang_ast_t.DeclStmt (stmt_info, _, decl_list) ->
|
|
|
|
let trans_state_decl = {trans_state with succ_nodes= res_trans_cond.root_nodes} in
|
|
|
|
declStmt_trans trans_state_decl decl_list stmt_info
|
|
|
|
| _ ->
|
|
|
|
res_trans_cond
|
|
|
|
|
|
|
|
|
|
|
|
and ifStmt_trans trans_state stmt_info stmt_list =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let succ_nodes = trans_state.succ_nodes in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let join_node = create_node Procdesc.Node.Join_node [] sil_loc context in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc join_node succ_nodes [] ;
|
|
|
|
let trans_state' = {trans_state with succ_nodes= [join_node]} in
|
|
|
|
let do_branch branch stmt_branch prune_nodes =
|
|
|
|
(* leaf nodes are ignored here as they will be already attached to join_node *)
|
|
|
|
let res_trans_b = instruction trans_state' stmt_branch in
|
|
|
|
let nodes_branch =
|
|
|
|
match res_trans_b.root_nodes with
|
|
|
|
| [] ->
|
|
|
|
[ create_node (Procdesc.Node.Stmt_node "IfStmt Branch") res_trans_b.instrs sil_loc
|
|
|
|
context ]
|
|
|
|
| _ ->
|
|
|
|
res_trans_b.root_nodes
|
|
|
|
in
|
|
|
|
let prune_nodes_t, prune_nodes_f = List.partition_tf ~f:is_true_prune_node prune_nodes in
|
|
|
|
let prune_nodes' = if branch then prune_nodes_t else prune_nodes_f in
|
|
|
|
List.iter
|
|
|
|
~f:(fun n -> Procdesc.node_set_succs_exn context.procdesc n nodes_branch [])
|
|
|
|
prune_nodes'
|
|
|
|
in
|
|
|
|
match stmt_list with
|
|
|
|
| [_; decl_stmt; cond; stmt1; stmt2] ->
|
|
|
|
(* set the flat to inform that we are translating a condition of a if *)
|
|
|
|
let continuation' = mk_cond_continuation trans_state.continuation in
|
|
|
|
let trans_state'' = {trans_state with continuation= continuation'; succ_nodes= []} in
|
|
|
|
let res_trans_cond = cond_trans ~negate_cond:false trans_state'' cond in
|
|
|
|
let res_trans_decl = declStmt_in_condition_trans trans_state decl_stmt res_trans_cond in
|
|
|
|
(* Note: by contruction prune nodes are leafs_nodes_cond *)
|
|
|
|
do_branch true stmt1 res_trans_cond.leaf_nodes ;
|
|
|
|
do_branch false stmt2 res_trans_cond.leaf_nodes ;
|
|
|
|
{empty_res_trans with root_nodes= res_trans_decl.root_nodes; leaf_nodes= [join_node]}
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
(* Assumption: the CompoundStmt can be made of different stmts, not just CaseStmts *)
|
|
|
|
and switchStmt_trans trans_state stmt_info switch_stmt_list =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let succ_nodes = trans_state.succ_nodes in
|
|
|
|
let continuation = trans_state.continuation in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match switch_stmt_list with
|
|
|
|
| [_; decl_stmt; cond; (CompoundStmt (stmt_info, stmt_list))] ->
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let trans_state' = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let res_trans_cond_tmp = instruction trans_state' cond in
|
|
|
|
let switch_special_cond_node =
|
|
|
|
let node_kind = Procdesc.Node.Stmt_node "Switch_stmt" in
|
|
|
|
create_node node_kind res_trans_cond_tmp.instrs sil_loc context
|
|
|
|
in
|
|
|
|
List.iter
|
|
|
|
~f:(fun n' ->
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc n' [switch_special_cond_node] [])
|
|
|
|
res_trans_cond_tmp.leaf_nodes ;
|
|
|
|
let root_nodes =
|
|
|
|
if res_trans_cond_tmp.root_nodes <> [] then res_trans_cond_tmp.root_nodes
|
|
|
|
else [switch_special_cond_node]
|
|
|
|
in
|
|
|
|
let switch_e_cond', switch_e_cond'_typ =
|
|
|
|
extract_exp_from_list res_trans_cond_tmp.exps
|
|
|
|
"@\nWARNING: The condition of the SwitchStmt is not singleton. Need to be fixed@\n"
|
|
|
|
in
|
|
|
|
let res_trans_cond =
|
|
|
|
{res_trans_cond_tmp with root_nodes; leaf_nodes= [switch_special_cond_node]}
|
|
|
|
in
|
|
|
|
let res_trans_decl = declStmt_in_condition_trans trans_state decl_stmt res_trans_cond in
|
|
|
|
let trans_state_no_pri =
|
|
|
|
if PriorityNode.own_priority_node trans_state_pri.priority stmt_info then
|
|
|
|
{trans_state_pri with priority= Free}
|
|
|
|
else trans_state_pri
|
|
|
|
in
|
|
|
|
let switch_exit_point = succ_nodes in
|
|
|
|
let continuation' =
|
|
|
|
match continuation with
|
|
|
|
| Some cont ->
|
|
|
|
Some {cont with break= switch_exit_point}
|
|
|
|
| None ->
|
|
|
|
Some {break= switch_exit_point; continue= []; return_temp= false}
|
|
|
|
in
|
|
|
|
let trans_state'' = {trans_state_no_pri with continuation= continuation'} in
|
|
|
|
let merge_into_cases stmt_list =
|
|
|
|
(* returns list_of_cases * before_any_case_instrs *)
|
|
|
|
let rec aux rev_stmt_list acc cases =
|
|
|
|
match rev_stmt_list with
|
|
|
|
| (CaseStmt (info, a :: b :: (CaseStmt x) :: c)) :: rest ->
|
|
|
|
(* case x: case y: ... *)
|
|
|
|
if c <> [] (* empty case with nested case, then followed by some instructions *)
|
|
|
|
then assert false ;
|
|
|
|
let rest' = CaseStmt (info, [a; b]) :: rest in
|
|
|
|
let rev_stmt_list' = CaseStmt x :: rest' in
|
|
|
|
aux rev_stmt_list' acc cases
|
|
|
|
| (CaseStmt (info, a :: b :: (DefaultStmt x) :: c)) :: rest ->
|
|
|
|
(* case x: default: ... *)
|
|
|
|
if c <> [] (* empty case with nested case, then followed by some instructions *)
|
|
|
|
then assert false ;
|
|
|
|
let rest' = CaseStmt (info, [a; b]) :: rest in
|
|
|
|
let rev_stmt_list' = DefaultStmt x :: rest' in
|
|
|
|
aux rev_stmt_list' acc cases
|
|
|
|
| (DefaultStmt (info, (CaseStmt x) :: c)) :: rest ->
|
|
|
|
(* default: case x: ... *)
|
|
|
|
if c <> [] (* empty case with nested case, then followed by some instructions *)
|
|
|
|
then assert false ;
|
|
|
|
let rest' = DefaultStmt (info, []) :: rest in
|
|
|
|
let rev_stmt_list' = CaseStmt x :: rest' in
|
|
|
|
aux rev_stmt_list' acc cases
|
|
|
|
| (CaseStmt (info, a :: b :: c)) :: rest ->
|
|
|
|
aux rest [] (CaseStmt (info, a :: b :: c @ acc) :: cases)
|
|
|
|
| (DefaultStmt (info, c)) :: rest ->
|
|
|
|
(* default is always the last in the list *)
|
|
|
|
aux rest [] (DefaultStmt (info, c @ acc) :: cases)
|
|
|
|
| x :: rest ->
|
|
|
|
aux rest (x :: acc) cases
|
|
|
|
| [] ->
|
|
|
|
(cases, acc)
|
|
|
|
in
|
|
|
|
aux (List.rev stmt_list) [] []
|
|
|
|
in
|
|
|
|
let list_of_cases, pre_case_stmts = merge_into_cases stmt_list in
|
|
|
|
let rec connected_instruction rev_instr_list successor_nodes =
|
|
|
|
(* returns the entry point of the translated set of instr *)
|
|
|
|
match rev_instr_list with
|
|
|
|
| [] ->
|
|
|
|
successor_nodes
|
|
|
|
| instr :: rest ->
|
|
|
|
let trans_state''' = {trans_state'' with succ_nodes= successor_nodes} in
|
|
|
|
let res_trans_instr = instruction trans_state''' instr in
|
|
|
|
let instr_entry_points = res_trans_instr.root_nodes in
|
|
|
|
connected_instruction rest instr_entry_points
|
|
|
|
in
|
|
|
|
let rec translate_and_connect_cases cases next_nodes next_prune_nodes =
|
|
|
|
let create_prune_nodes_for_case case =
|
|
|
|
match case with
|
|
|
|
| CaseStmt (stmt_info, case_const :: _ :: _) ->
|
|
|
|
let trans_state_pri =
|
|
|
|
PriorityNode.try_claim_priority_node trans_state'' stmt_info
|
|
|
|
in
|
|
|
|
let res_trans_case_const = instruction trans_state_pri case_const in
|
|
|
|
let e_const = res_trans_case_const.exps in
|
|
|
|
let e_const' = match e_const with [(head, _)] -> head | _ -> assert false in
|
|
|
|
let sil_eq_cond = Exp.BinOp (Binop.Eq, switch_e_cond', e_const') in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let true_prune_node =
|
|
|
|
create_prune_node ~branch:true ~negate_cond:false
|
|
|
|
[(sil_eq_cond, switch_e_cond'_typ)] res_trans_case_const.instrs sil_loc
|
|
|
|
Sil.Ik_switch context
|
|
|
|
in
|
|
|
|
let false_prune_node =
|
|
|
|
create_prune_node ~branch:false ~negate_cond:true
|
|
|
|
[(sil_eq_cond, switch_e_cond'_typ)] res_trans_case_const.instrs sil_loc
|
|
|
|
Sil.Ik_switch context
|
|
|
|
in
|
|
|
|
(true_prune_node, false_prune_node)
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
match cases with
|
|
|
|
(* top-down to handle default cases *)
|
|
|
|
| [] ->
|
|
|
|
(next_nodes, next_prune_nodes)
|
|
|
|
| (CaseStmt (_, _ :: _ :: case_content) as case) :: rest ->
|
|
|
|
let last_nodes, last_prune_nodes =
|
|
|
|
translate_and_connect_cases rest next_nodes next_prune_nodes
|
|
|
|
in
|
|
|
|
let case_entry_point = connected_instruction (List.rev case_content) last_nodes in
|
|
|
|
(* connects between cases, then continuation has priority about breaks *)
|
|
|
|
let prune_node_t, prune_node_f = create_prune_nodes_for_case case in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc prune_node_t case_entry_point [] ;
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc prune_node_f last_prune_nodes [] ;
|
|
|
|
(case_entry_point, [prune_node_t; prune_node_f])
|
|
|
|
| (DefaultStmt (stmt_info, default_content)) :: rest ->
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let placeholder_entry_point =
|
|
|
|
create_node (Procdesc.Node.Stmt_node "DefaultStmt_placeholder") [] sil_loc context
|
|
|
|
in
|
|
|
|
let last_nodes, last_prune_nodes =
|
|
|
|
translate_and_connect_cases rest next_nodes [placeholder_entry_point]
|
|
|
|
in
|
|
|
|
let default_entry_point =
|
|
|
|
connected_instruction (List.rev default_content) last_nodes
|
|
|
|
in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc placeholder_entry_point
|
|
|
|
default_entry_point [] ;
|
|
|
|
(default_entry_point, last_prune_nodes)
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
let top_entry_point, top_prune_nodes =
|
|
|
|
translate_and_connect_cases list_of_cases succ_nodes succ_nodes
|
|
|
|
in
|
|
|
|
let _ = connected_instruction (List.rev pre_case_stmts) top_entry_point in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc switch_special_cond_node top_prune_nodes [] ;
|
|
|
|
let top_nodes = res_trans_decl.root_nodes in
|
|
|
|
(* succ_nodes will remove the temps *)
|
|
|
|
{empty_res_trans with root_nodes= top_nodes; leaf_nodes= succ_nodes}
|
|
|
|
| _ ->
|
|
|
|
(* TODO(t21762295) this raises sometimes *)
|
|
|
|
CFrontend_config.incorrect_assumption
|
|
|
|
"Unexpected Switch Statement sub-expression list: [%a]"
|
|
|
|
(Pp.semicolon_seq (Pp.to_string ~f:Clang_ast_j.string_of_stmt))
|
|
|
|
switch_stmt_list
|
|
|
|
|
|
|
|
|
|
|
|
and stmtExpr_trans trans_state stmt_list =
|
|
|
|
let stmt =
|
|
|
|
extract_stmt_from_singleton stmt_list "ERROR: StmtExpr should have only one statement.@\n"
|
|
|
|
in
|
|
|
|
let trans_state' = {trans_state with priority= Free} in
|
|
|
|
let res_trans_stmt = instruction trans_state' stmt in
|
|
|
|
let exps' = List.rev res_trans_stmt.exps in
|
|
|
|
match exps' with
|
|
|
|
| last_exp :: _ ->
|
|
|
|
{res_trans_stmt with exps= [last_exp]}
|
|
|
|
| [] ->
|
|
|
|
res_trans_stmt
|
|
|
|
|
|
|
|
|
|
|
|
and loop_instruction trans_state loop_kind stmt_info =
|
|
|
|
let outer_continuation = trans_state.continuation in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let succ_nodes = trans_state.succ_nodes in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let join_node = create_node Procdesc.Node.Join_node [] sil_loc context in
|
|
|
|
let continuation = Some {break= succ_nodes; continue= [join_node]; return_temp= false} in
|
|
|
|
(* set the flag to inform that we are translating a condition of a if *)
|
|
|
|
let continuation_cond = mk_cond_continuation outer_continuation in
|
|
|
|
let init_incr_nodes =
|
|
|
|
match loop_kind with
|
|
|
|
| Loops.For (init, _, _, incr, _) ->
|
|
|
|
let trans_state' = {trans_state with succ_nodes= [join_node]; continuation} in
|
|
|
|
let res_trans_init = instruction trans_state' init in
|
|
|
|
let res_trans_incr = instruction trans_state' incr in
|
|
|
|
Some (res_trans_init.root_nodes, res_trans_incr.root_nodes)
|
|
|
|
| _ ->
|
|
|
|
None
|
|
|
|
in
|
|
|
|
let cond_stmt = Loops.get_cond loop_kind in
|
|
|
|
let trans_state_cond = {trans_state with continuation= continuation_cond; succ_nodes= []} in
|
|
|
|
let res_trans_cond = cond_trans ~negate_cond:false trans_state_cond cond_stmt in
|
|
|
|
let decl_stmt_opt =
|
|
|
|
match loop_kind with
|
|
|
|
| Loops.For (_, decl_stmt, _, _, _) ->
|
|
|
|
Some decl_stmt
|
|
|
|
| Loops.While (decl_stmt_opt, _, _) ->
|
|
|
|
decl_stmt_opt
|
|
|
|
| _ ->
|
|
|
|
None
|
|
|
|
in
|
|
|
|
let res_trans_decl =
|
|
|
|
match decl_stmt_opt with
|
|
|
|
| Some decl_stmt ->
|
|
|
|
declStmt_in_condition_trans trans_state decl_stmt res_trans_cond
|
|
|
|
| _ ->
|
|
|
|
res_trans_cond
|
|
|
|
in
|
|
|
|
let body_succ_nodes =
|
|
|
|
match loop_kind with
|
|
|
|
| Loops.For _ -> (
|
|
|
|
match init_incr_nodes with Some (_, nodes_incr) -> nodes_incr | None -> assert false )
|
|
|
|
| Loops.While _ ->
|
|
|
|
[join_node]
|
|
|
|
| Loops.DoWhile _ ->
|
|
|
|
res_trans_cond.root_nodes
|
|
|
|
in
|
|
|
|
let body_continuation =
|
|
|
|
match (loop_kind, continuation, init_incr_nodes) with
|
|
|
|
| Loops.DoWhile _, Some c, _ ->
|
|
|
|
Some {c with continue= res_trans_cond.root_nodes}
|
|
|
|
| _, Some c, Some (_, nodes_incr) ->
|
|
|
|
Some {c with continue= nodes_incr}
|
|
|
|
| _ ->
|
|
|
|
continuation
|
|
|
|
in
|
|
|
|
let res_trans_body =
|
|
|
|
let trans_state_body =
|
|
|
|
{trans_state with succ_nodes= body_succ_nodes; continuation= body_continuation}
|
|
|
|
in
|
|
|
|
instruction trans_state_body (Loops.get_body loop_kind)
|
|
|
|
in
|
|
|
|
let join_succ_nodes =
|
|
|
|
match loop_kind with
|
|
|
|
| Loops.For _ | Loops.While _ ->
|
|
|
|
res_trans_decl.root_nodes
|
|
|
|
| Loops.DoWhile _ ->
|
|
|
|
res_trans_body.root_nodes
|
|
|
|
in
|
|
|
|
(* Note: prune nodes are by contruction the res_trans_cond.leaf_nodes *)
|
|
|
|
let prune_nodes_t, prune_nodes_f =
|
|
|
|
List.partition_tf ~f:is_true_prune_node res_trans_cond.leaf_nodes
|
|
|
|
in
|
|
|
|
let prune_t_succ_nodes =
|
|
|
|
match loop_kind with
|
|
|
|
| Loops.For _ | Loops.While _ ->
|
|
|
|
res_trans_body.root_nodes
|
|
|
|
| Loops.DoWhile _ ->
|
|
|
|
[join_node]
|
|
|
|
in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc join_node join_succ_nodes [] ;
|
|
|
|
List.iter
|
|
|
|
~f:(fun n -> Procdesc.node_set_succs_exn context.procdesc n prune_t_succ_nodes [])
|
|
|
|
prune_nodes_t ;
|
|
|
|
List.iter
|
|
|
|
~f:(fun n -> Procdesc.node_set_succs_exn context.procdesc n succ_nodes [])
|
|
|
|
prune_nodes_f ;
|
|
|
|
let root_nodes =
|
|
|
|
match loop_kind with
|
|
|
|
| Loops.For _ -> (
|
|
|
|
match init_incr_nodes with Some (nodes_init, _) -> nodes_init | None -> assert false )
|
|
|
|
| Loops.While _ | Loops.DoWhile _ ->
|
|
|
|
[join_node]
|
|
|
|
in
|
|
|
|
{empty_res_trans with root_nodes; leaf_nodes= prune_nodes_f}
|
|
|
|
|
|
|
|
|
|
|
|
and forStmt_trans trans_state init decl_stmt cond incr body stmt_info =
|
|
|
|
let for_kind = Loops.For (init, decl_stmt, cond, incr, body) in
|
|
|
|
loop_instruction trans_state for_kind stmt_info
|
|
|
|
|
|
|
|
|
|
|
|
and whileStmt_trans trans_state decl_stmt cond body stmt_info =
|
|
|
|
let while_kind = Loops.While (Some decl_stmt, cond, body) in
|
|
|
|
loop_instruction trans_state while_kind stmt_info
|
|
|
|
|
|
|
|
|
|
|
|
and doStmt_trans trans_state stmt_info cond body =
|
|
|
|
let dowhile_kind = Loops.DoWhile (cond, body) in
|
|
|
|
loop_instruction trans_state dowhile_kind stmt_info
|
|
|
|
|
|
|
|
|
|
|
|
(* Iteration over colection
|
|
|
|
|
|
|
|
for (v : C) { body; }
|
|
|
|
|
|
|
|
is translated as follows:
|
|
|
|
|
|
|
|
TypeC __range = C;
|
|
|
|
for (__begin = __range.begin(), __end = __range.end();
|
|
|
|
__begin != __end;
|
|
|
|
++__begin)
|
|
|
|
{
|
|
|
|
v = *__begin;
|
|
|
|
loop_body;
|
|
|
|
}
|
|
|
|
*)
|
|
|
|
and cxxForRangeStmt_trans trans_state stmt_info stmt_list =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match stmt_list with
|
|
|
|
| [iterator_decl; begin_stmt; end_stmt; exit_cond; increment; assign_current_index; loop_body] ->
|
|
|
|
let loop_body' = CompoundStmt (stmt_info, [assign_current_index; loop_body]) in
|
|
|
|
let null_stmt = NullStmt (stmt_info, []) in
|
|
|
|
let beginend_stmt = CompoundStmt (stmt_info, [begin_stmt; end_stmt]) in
|
|
|
|
let for_loop =
|
|
|
|
ForStmt (stmt_info, [beginend_stmt; null_stmt; exit_cond; increment; loop_body'])
|
|
|
|
in
|
|
|
|
instruction trans_state (CompoundStmt (stmt_info, [iterator_decl; for_loop]))
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
(* Fast iteration for colection
|
|
|
|
for (type_it i in collection) { body }
|
|
|
|
is translate as
|
|
|
|
{
|
|
|
|
i = type_next_object();
|
|
|
|
while(i != nil) { body; i = type_next_object();}
|
|
|
|
}
|
|
|
|
*)
|
|
|
|
and objCForCollectionStmt_trans trans_state item items body stmt_info =
|
|
|
|
let _ = instruction trans_state item in
|
|
|
|
(* Here we do ast transformation, so we don't need the value of the translation of the *)
|
|
|
|
(* variable item but we still need to add the variable to the locals *)
|
|
|
|
let assign_next_object, cond = Ast_expressions.make_next_object_exp stmt_info item items in
|
|
|
|
let body' = Clang_ast_t.CompoundStmt (stmt_info, [body; assign_next_object]) in
|
|
|
|
let null_stmt = Clang_ast_t.NullStmt (stmt_info, []) in
|
|
|
|
let loop = Clang_ast_t.WhileStmt (stmt_info, [null_stmt; cond; body']) in
|
|
|
|
instruction trans_state (Clang_ast_t.CompoundStmt (stmt_info, [assign_next_object; loop]))
|
|
|
|
|
|
|
|
|
|
|
|
and initListExpr_array_trans trans_state stmt_info stmts var_exp field_typ =
|
|
|
|
let lh_exp idx =
|
|
|
|
let idx_exp = Exp.Const (Const.Cint (IntLit.of_int idx)) in
|
|
|
|
Exp.Lindex (var_exp, idx_exp)
|
|
|
|
in
|
|
|
|
let init_field idx stmt =
|
|
|
|
init_expr_trans trans_state (lh_exp idx, field_typ) stmt_info (Some stmt)
|
|
|
|
in
|
|
|
|
(* rest of fields when length(stmts) < size is ignored *)
|
|
|
|
List.mapi ~f:init_field stmts
|
|
|
|
|
|
|
|
|
|
|
|
and initListExpr_struct_trans trans_state stmt_info stmts var_exp var_typ =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let tenv = context.tenv in
|
|
|
|
let tname = match var_typ.Typ.desc with Tstruct tname -> tname | _ -> assert false in
|
|
|
|
let field_exps =
|
|
|
|
match Tenv.lookup tenv tname with
|
|
|
|
| Some {fields} ->
|
|
|
|
List.filter_map fields ~f:(fun (fieldname, fieldtype, _) ->
|
|
|
|
if Typ.Fieldname.is_hidden fieldname then None
|
|
|
|
else Some (Exp.Lfield (var_exp, fieldname, var_typ), fieldtype) )
|
|
|
|
| None ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
let init_field field_exp_typ stmt =
|
|
|
|
init_expr_trans trans_state field_exp_typ stmt_info (Some stmt)
|
|
|
|
in
|
|
|
|
match List.map2 field_exps stmts ~f:init_field with
|
|
|
|
| Ok result ->
|
|
|
|
result
|
|
|
|
| Unequal_lengths ->
|
|
|
|
(* This can happen with union initializers. Skip them for now *) []
|
|
|
|
|
|
|
|
|
|
|
|
and initListExpr_builtin_trans trans_state stmt_info stmts var_exp var_typ =
|
|
|
|
match stmts with
|
|
|
|
| [stmt] ->
|
|
|
|
[init_expr_trans trans_state (var_exp, var_typ) stmt_info (Some stmt)]
|
|
|
|
| _ ->
|
|
|
|
CFrontend_config.unimplemented
|
|
|
|
"InitListExpression for var %a type %a with multiple init statements" Exp.pp var_exp
|
|
|
|
(Typ.pp_full Pp.text) var_typ
|
|
|
|
|
|
|
|
|
|
|
|
(** InitListExpr can have following meanings:
|
|
|
|
- initialize all record fields
|
|
|
|
- initialize array
|
|
|
|
- initialize primitive type (int/flaot/pointer/...)
|
|
|
|
- perform zero initalization - http://en.cppreference.com/w/cpp/language/zero_initialization
|
|
|
|
Decision which case happens is based on the type of the InitListExpr
|
|
|
|
*)
|
|
|
|
and initListExpr_trans trans_state stmt_info expr_info stmts =
|
|
|
|
let var_exp, var_typ =
|
|
|
|
match trans_state.var_exp_typ with
|
|
|
|
| Some var_exp_typ ->
|
|
|
|
var_exp_typ
|
|
|
|
| None ->
|
|
|
|
create_var_exp_tmp_var trans_state expr_info "SIL_init_list__"
|
|
|
|
in
|
|
|
|
if Int.equal (List.length stmts) 0 then
|
|
|
|
(* perform zero initialization of a primitive type, record types will have
|
|
|
|
ImplicitValueInitExpr nodes *)
|
|
|
|
let exps =
|
|
|
|
match Sil.zero_value_of_numerical_type_option var_typ with
|
|
|
|
| Some zero_exp ->
|
|
|
|
[(zero_exp, var_typ)]
|
|
|
|
| None ->
|
|
|
|
[]
|
|
|
|
in
|
|
|
|
{empty_res_trans with root_nodes= trans_state.succ_nodes; exps}
|
|
|
|
else
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let init_stmt_info =
|
|
|
|
{stmt_info with Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()}
|
|
|
|
in
|
|
|
|
let all_res_trans =
|
|
|
|
match var_typ.Typ.desc with
|
|
|
|
| Typ.Tarray (typ_inside, _, _) ->
|
|
|
|
initListExpr_array_trans trans_state_pri init_stmt_info stmts var_exp typ_inside
|
|
|
|
| Tstruct _ ->
|
|
|
|
initListExpr_struct_trans trans_state_pri init_stmt_info stmts var_exp var_typ
|
|
|
|
| Tint _ | Tfloat _ | Tptr _ ->
|
|
|
|
initListExpr_builtin_trans trans_state_pri init_stmt_info stmts var_exp var_typ
|
|
|
|
| _ ->
|
|
|
|
CFrontend_config.unimplemented "InitListExp for var %a of type %a" Exp.pp var_exp
|
|
|
|
(Typ.pp Pp.text) var_typ
|
|
|
|
in
|
|
|
|
let nname = "InitListExp" in
|
|
|
|
let res_trans =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname stmt_info
|
|
|
|
all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans with exps= [(var_exp, var_typ)]; initd_exps= [var_exp]}
|
|
|
|
|
|
|
|
|
|
|
|
and init_dynamic_array trans_state array_exp_typ array_stmt_info dynlength_stmt_pointer =
|
|
|
|
let dynlength_stmt =
|
|
|
|
Int.Table.find_exn ClangPointers.pointer_stmt_table dynlength_stmt_pointer
|
|
|
|
in
|
|
|
|
let dynlength_stmt_info, _ = Clang_ast_proj.get_stmt_tuple dynlength_stmt in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state array_stmt_info in
|
|
|
|
let dynlength_trans_result = instruction trans_state_pri dynlength_stmt in
|
|
|
|
let dynlength_exp_typ =
|
|
|
|
extract_exp_from_list dynlength_trans_result.exps
|
|
|
|
"WARNING: There should be one expression.@\n"
|
|
|
|
in
|
|
|
|
let sil_loc = CLocation.get_sil_location dynlength_stmt_info trans_state_pri.context in
|
|
|
|
let call_instr =
|
|
|
|
let call_exp = Exp.Const (Const.Cfun BuiltinDecl.__set_array_length) in
|
|
|
|
let actuals = [array_exp_typ; dynlength_exp_typ] in
|
|
|
|
Sil.Call (None, call_exp, actuals, sil_loc, CallFlags.default)
|
|
|
|
in
|
|
|
|
let call_trans_result = {empty_res_trans with instrs= [call_instr]} in
|
|
|
|
let res_trans =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc
|
|
|
|
"Initialize dynamic array length" dynlength_stmt_info
|
|
|
|
[dynlength_trans_result; call_trans_result]
|
|
|
|
in
|
|
|
|
{res_trans with exps= []}
|
|
|
|
|
|
|
|
|
|
|
|
and init_expr_trans trans_state var_exp_typ ?qual_type var_stmt_info init_expr_opt =
|
|
|
|
match init_expr_opt with
|
|
|
|
| None -> (
|
|
|
|
match
|
|
|
|
Option.map ~f:(fun qt -> qt.Clang_ast_t.qt_type_ptr) qual_type
|
|
|
|
|> Option.find_map ~f:CAst_utils.get_type
|
|
|
|
with
|
|
|
|
| Some Clang_ast_t.VariableArrayType (_, _, stmt_pointer) ->
|
|
|
|
(* Set the dynamic length of the variable length array. Variable length array cannot
|
|
|
|
have an initialization expression. *)
|
|
|
|
init_dynamic_array trans_state var_exp_typ var_stmt_info stmt_pointer
|
|
|
|
| _ ->
|
|
|
|
(* Nothing to do if no init expression and not a variable length array *)
|
|
|
|
{empty_res_trans with root_nodes= trans_state.succ_nodes} )
|
|
|
|
| Some ie ->
|
|
|
|
(*For init expr, translate how to compute it and assign to the var*)
|
|
|
|
let var_exp, _ = var_exp_typ in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location var_stmt_info context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state var_stmt_info in
|
|
|
|
(* if ie is a block the translation need to be done
|
|
|
|
with the block special cases by exec_with_block_priority *)
|
|
|
|
let res_trans_ie =
|
|
|
|
let trans_state' =
|
|
|
|
{trans_state_pri with succ_nodes= []; var_exp_typ= Some var_exp_typ}
|
|
|
|
in
|
|
|
|
let instruction' =
|
|
|
|
exec_with_self_exception (exec_with_glvalue_as_reference instruction)
|
|
|
|
in
|
|
|
|
exec_with_block_priority_exception instruction' trans_state' ie var_stmt_info
|
|
|
|
in
|
|
|
|
let sil_e1', ie_typ =
|
|
|
|
extract_exp_from_list res_trans_ie.exps
|
|
|
|
"WARNING: In DeclStmt we expect only one expression returned in recursive call@\n"
|
|
|
|
in
|
|
|
|
let rhs_owning_method = CTrans_utils.is_owning_method ie in
|
|
|
|
let _, instrs_assign =
|
|
|
|
(* variable might be initialized already - do nothing in that case*)
|
|
|
|
if List.exists ~f:(Exp.equal var_exp) res_trans_ie.initd_exps then ([], [])
|
|
|
|
else if !Config.arc_mode
|
|
|
|
&& ( CTrans_utils.is_method_call ie
|
|
|
|
|| ObjcInterface_decl.is_pointer_to_objc_class ie_typ )
|
|
|
|
then
|
|
|
|
(* In arc mode, if it's a method call or we are initializing
|
|
|
|
with a pointer to objc class *)
|
|
|
|
(* we need to add retain/release *)
|
|
|
|
let e, instrs =
|
|
|
|
CArithmetic_trans.assignment_arc_mode var_exp ie_typ sil_e1' sil_loc
|
|
|
|
rhs_owning_method true
|
|
|
|
in
|
|
|
|
([(e, ie_typ)], instrs)
|
|
|
|
else ([], [Sil.Store (var_exp, ie_typ, sil_e1', sil_loc)])
|
|
|
|
in
|
|
|
|
let res_trans_assign = {empty_res_trans with instrs= instrs_assign} in
|
|
|
|
let all_res_trans = [res_trans_ie; res_trans_assign] in
|
|
|
|
let res_trans =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "DeclStmt" var_stmt_info
|
|
|
|
all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans with exps= [(var_exp, ie_typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
and collect_all_decl trans_state var_decls next_nodes stmt_info =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let procdesc = context.CContext.procdesc in
|
|
|
|
let procname = Procdesc.get_proc_name procdesc in
|
|
|
|
let do_var_dec (di, var_name, qual_type, vdi) next_node =
|
|
|
|
let var_decl = VarDecl (di, var_name, qual_type, vdi) in
|
|
|
|
let pvar = CVar_decl.sil_var_of_decl context var_decl procname in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type context.CContext.tenv qual_type in
|
|
|
|
CVar_decl.add_var_to_locals procdesc var_decl typ pvar ;
|
|
|
|
let trans_state' = {trans_state with succ_nodes= next_node} in
|
|
|
|
init_expr_trans trans_state' (Exp.Lvar pvar, typ) ~qual_type stmt_info
|
|
|
|
vdi.Clang_ast_t.vdi_init_expr
|
|
|
|
in
|
|
|
|
match var_decls with
|
|
|
|
| [] ->
|
|
|
|
{empty_res_trans with root_nodes= next_nodes}
|
|
|
|
| (VarDecl (di, n, qt, vdi)) :: var_decls' ->
|
|
|
|
(* Var are defined when procdesc is created, here we only take care of initialization*)
|
|
|
|
let res_trans_vd = collect_all_decl trans_state var_decls' next_nodes stmt_info in
|
|
|
|
let res_trans_tmp = do_var_dec (di, n, qt, vdi) res_trans_vd.root_nodes in
|
|
|
|
{ empty_res_trans with
|
|
|
|
root_nodes= res_trans_tmp.root_nodes
|
|
|
|
; leaf_nodes= []
|
|
|
|
; instrs= res_trans_tmp.instrs @ res_trans_vd.instrs
|
|
|
|
; exps= []
|
|
|
|
; initd_exps= res_trans_tmp.initd_exps @ res_trans_vd.initd_exps }
|
|
|
|
| (CXXRecordDecl _) :: var_decls'
|
|
|
|
(*C++/C record decl treated in the same way *)
|
|
|
|
| (RecordDecl _) :: var_decls' ->
|
|
|
|
(* Record declaration is done in the beginning when procdesc is defined.*)
|
|
|
|
collect_all_decl trans_state var_decls' next_nodes stmt_info
|
|
|
|
| decl :: _ ->
|
|
|
|
CFrontend_config.incorrect_assumption "unexpected decl type %s in collect_all_decl: %a"
|
|
|
|
(Clang_ast_proj.get_decl_kind_string decl)
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_decl)
|
|
|
|
decl
|
|
|
|
|
|
|
|
|
|
|
|
(* stmt_list is ignored because it contains the same instructions as *)
|
|
|
|
(* the init expression. We use the latter info. *)
|
|
|
|
and declStmt_trans trans_state decl_list stmt_info =
|
|
|
|
let succ_nodes = trans_state.succ_nodes in
|
|
|
|
let res_trans =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match decl_list with
|
|
|
|
| (VarDecl _) :: _
|
|
|
|
(* Case for simple variable declarations*)
|
|
|
|
| (CXXRecordDecl _) :: _
|
|
|
|
(*C++/C record decl treated in the same way *)
|
|
|
|
| (RecordDecl _) :: _ ->
|
|
|
|
(* Case for struct *)
|
|
|
|
collect_all_decl trans_state decl_list succ_nodes stmt_info
|
|
|
|
| (TypedefDecl _) :: _ | (UsingDirectiveDecl _) :: _ ->
|
|
|
|
empty_res_trans
|
|
|
|
| decl :: _ ->
|
|
|
|
CFrontend_config.unimplemented "In DeclStmt found an unknown declaration type %s"
|
|
|
|
(Clang_ast_j.string_of_decl decl)
|
|
|
|
| [] ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
{res_trans with leaf_nodes= []}
|
|
|
|
|
|
|
|
|
|
|
|
and objCPropertyRefExpr_trans trans_state stmt_list =
|
|
|
|
match stmt_list with [stmt] -> instruction trans_state stmt | _ -> assert false
|
|
|
|
|
|
|
|
|
|
|
|
(* For OpaqueValueExpr we return the translation generated from its source expression*)
|
|
|
|
and opaqueValueExpr_trans trans_state opaque_value_expr_info =
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
" priority node free = '%s'@\n@."
|
|
|
|
(string_of_bool (PriorityNode.is_priority_free trans_state)) ;
|
|
|
|
match trans_state.opaque_exp with
|
|
|
|
| Some exp ->
|
|
|
|
{empty_res_trans with exps= [exp]}
|
|
|
|
| _ ->
|
|
|
|
match opaque_value_expr_info.Clang_ast_t.ovei_source_expr with
|
|
|
|
| Some stmt ->
|
|
|
|
instruction trans_state stmt
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
(* NOTE: This translation has several hypothesis. Need to be verified when we have more*)
|
|
|
|
(* experience with this construct. Assert false will help to see if we encounter programs*)
|
|
|
|
(* that do not conform with this hypothesis.*)
|
|
|
|
(* Hypotheses:*)
|
|
|
|
(* 1. stmt_list is composed by 2 parts: the first element is a syntactic description of the*)
|
|
|
|
(* expression. The rest of the list has a semantic caracterization of the expression and*)
|
|
|
|
(* defines how that expression is going to be implemented at runtime. *)
|
|
|
|
(* 2. the semantic description is composed by a list of OpaqueValueExpr that define the *)
|
|
|
|
(* various expressions involved and one finale expression that define how the final value of*)
|
|
|
|
(* the PseudoObjectExpr is obtained.
|
|
|
|
All the OpaqueValueExpr will be part of the last expression.*)
|
|
|
|
(* So they can be skipped. *)
|
|
|
|
(* For example: 'x.f = a' when 'f' is a property will be
|
|
|
|
translated with a call to f's setter [x f:a]*)
|
|
|
|
(* the stmt_list will be [x.f = a; x; a; CallToSetter]
|
|
|
|
Among all element of the list we only need*)
|
|
|
|
(* to translate the CallToSetter which is
|
|
|
|
how x.f = a is actually implemented by the runtime.*)
|
|
|
|
and pseudoObjectExpr_trans trans_state stmt_list =
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
" priority node free = '%s'@\n@\n"
|
|
|
|
(string_of_bool (PriorityNode.is_priority_free trans_state)) ;
|
|
|
|
let rec do_semantic_elements el =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match el with
|
|
|
|
| (OpaqueValueExpr _) :: el' ->
|
|
|
|
do_semantic_elements el'
|
|
|
|
| stmt :: _ ->
|
|
|
|
instruction trans_state stmt
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
match stmt_list with
|
|
|
|
| _ :: semantic_form ->
|
|
|
|
do_semantic_elements semantic_form
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
(* Cast expression are treated the same apart from the cast operation kind*)
|
|
|
|
and cast_exprs_trans trans_state stmt_info stmt_list expr_info cast_expr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
" priority node free = '%s'@\n@."
|
|
|
|
(string_of_bool (PriorityNode.is_priority_free trans_state)) ;
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let stmt =
|
|
|
|
extract_stmt_from_singleton stmt_list
|
|
|
|
"WARNING: In CastExpr There must be only one stmt defining the expression to be cast.@\n"
|
|
|
|
in
|
|
|
|
let res_trans_stmt = instruction trans_state stmt in
|
|
|
|
let typ =
|
|
|
|
CType_decl.qual_type_to_sil_type context.CContext.tenv expr_info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
|
|
|
let cast_kind = cast_expr_info.Clang_ast_t.cei_cast_kind in
|
|
|
|
(* This gives the differnece among cast operations kind*)
|
|
|
|
let is_objc_bridged_cast_expr _ stmt =
|
|
|
|
match stmt with Clang_ast_t.ObjCBridgedCastExpr _ -> true | _ -> false
|
|
|
|
in
|
|
|
|
let is_objc_bridged = CAst_utils.exists_eventually_st is_objc_bridged_cast_expr () stmt in
|
|
|
|
let cast_inst, cast_exp =
|
|
|
|
cast_operation trans_state cast_kind res_trans_stmt.exps typ sil_loc is_objc_bridged
|
|
|
|
in
|
|
|
|
{res_trans_stmt with instrs= res_trans_stmt.instrs @ cast_inst; exps= [cast_exp]}
|
|
|
|
|
|
|
|
|
|
|
|
(* function used in the computation for both Member_Expr and ObjCIVarRefExpr *)
|
|
|
|
and do_memb_ivar_ref_exp trans_state stmt_info stmt_list decl_ref =
|
|
|
|
let exp_stmt =
|
|
|
|
extract_stmt_from_singleton stmt_list
|
|
|
|
"WARNING: in MemberExpr there must be only one stmt defining its expression.@\n"
|
|
|
|
in
|
|
|
|
(* Don't pass var_exp_typ to child of MemberExpr - this may lead to initializing variable *)
|
|
|
|
(* with wrong value. For example, we don't want p to be initialized with X(1) for:*)
|
|
|
|
(* int p = X(1).field; *)
|
|
|
|
let trans_state' = {trans_state with var_exp_typ= None} in
|
|
|
|
let result_trans_exp_stmt = exec_with_glvalue_as_reference instruction trans_state' exp_stmt in
|
|
|
|
decl_ref_trans trans_state result_trans_exp_stmt stmt_info decl_ref ~is_constructor_init:false
|
|
|
|
|
|
|
|
|
|
|
|
and objCIvarRefExpr_trans trans_state stmt_info stmt_list obj_c_ivar_ref_expr_info =
|
|
|
|
let decl_ref = obj_c_ivar_ref_expr_info.Clang_ast_t.ovrei_decl_ref in
|
|
|
|
do_memb_ivar_ref_exp trans_state stmt_info stmt_list decl_ref
|
|
|
|
|
|
|
|
|
|
|
|
and memberExpr_trans trans_state stmt_info stmt_list member_expr_info =
|
|
|
|
let decl_ref = member_expr_info.Clang_ast_t.mei_decl_ref in
|
|
|
|
let res_trans = do_memb_ivar_ref_exp trans_state stmt_info stmt_list decl_ref in
|
|
|
|
let is_virtual_dispatch = member_expr_info.Clang_ast_t.mei_performs_virtual_dispatch in
|
|
|
|
{res_trans with is_cpp_call_virtual= res_trans.is_cpp_call_virtual && is_virtual_dispatch}
|
|
|
|
|
|
|
|
|
|
|
|
and unaryOperator_trans trans_state stmt_info expr_info stmt_list unary_operator_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let stmt =
|
|
|
|
extract_stmt_from_singleton stmt_list
|
|
|
|
"WARNING: We expect only one element in stmt list defining the operand in UnaryOperator. NEED FIXING@\n"
|
|
|
|
in
|
|
|
|
let trans_state' = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let res_trans_stmt = instruction trans_state' stmt in
|
|
|
|
(* Assumption: the operand does not create a cfg node*)
|
|
|
|
let sil_e', _ =
|
|
|
|
extract_exp_from_list res_trans_stmt.exps
|
|
|
|
"@\nWARNING: Missing operand in unary operator. NEED FIXING.@\n"
|
|
|
|
in
|
|
|
|
let ret_typ =
|
|
|
|
CType_decl.qual_type_to_sil_type context.CContext.tenv expr_info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
|
|
|
let exp_op, instr_op =
|
|
|
|
CArithmetic_trans.unary_operation_instruction context.translation_unit_context
|
|
|
|
unary_operator_info sil_e' ret_typ sil_loc
|
|
|
|
in
|
|
|
|
let unary_op_res_trans = {empty_res_trans with instrs= instr_op} in
|
|
|
|
let all_res_trans = [res_trans_stmt; unary_op_res_trans] in
|
|
|
|
let nname = "UnaryOperator" in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname stmt_info all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans_to_parent with exps= [(exp_op, ret_typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
and returnStmt_trans trans_state stmt_info stmt_list =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let succ_nodes = trans_state.succ_nodes in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let procdesc = context.CContext.procdesc in
|
|
|
|
let procname = Procdesc.get_proc_name procdesc in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let mk_ret_node instrs =
|
|
|
|
let destr_trans_result = inject_destructors trans_state_pri stmt_info in
|
|
|
|
(* `inject_destructors` should not create new nodes for return statement,
|
|
|
|
this is ensured by creating a fresh pointer in `inject_destructors`
|
|
|
|
*)
|
|
|
|
if destr_trans_result.root_nodes <> [] then assert false ;
|
|
|
|
let is_destructor = Typ.Procname.is_destructor procname in
|
|
|
|
let destructor_res =
|
|
|
|
if is_destructor then
|
|
|
|
cxx_inject_field_destructors_in_destructor_body trans_state_pri stmt_info
|
|
|
|
else empty_res_trans
|
|
|
|
in
|
|
|
|
(* `cxx_inject_field_destructors_in_destructor_body` should not create new nodes for return statement,
|
|
|
|
this is ensured by creating a fresh pointer in `cxx_inject_field_destructors_in_destructor_body`
|
|
|
|
*)
|
|
|
|
if destructor_res.root_nodes <> [] then assert false ;
|
|
|
|
let ret_node =
|
|
|
|
create_node (Procdesc.Node.Stmt_node "Return Stmt")
|
|
|
|
(instrs @ destr_trans_result.instrs @ destructor_res.instrs)
|
|
|
|
sil_loc context
|
|
|
|
in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc ret_node
|
|
|
|
[Procdesc.get_exit_node context.CContext.procdesc]
|
|
|
|
[] ;
|
|
|
|
ret_node
|
|
|
|
in
|
|
|
|
let trans_result =
|
|
|
|
match stmt_list with
|
|
|
|
| [stmt] ->
|
|
|
|
(* return exp; *)
|
|
|
|
let ret_type = Procdesc.get_ret_type procdesc in
|
|
|
|
let ret_exp, ret_typ, var_instrs =
|
|
|
|
match context.CContext.return_param_typ with
|
|
|
|
| Some ret_param_typ ->
|
|
|
|
let name = CFrontend_config.return_param in
|
|
|
|
let pvar = Pvar.mk (Mangled.from_string name) procname in
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
let instr = Sil.Load (id, Exp.Lvar pvar, ret_param_typ, sil_loc) in
|
|
|
|
let ret_typ =
|
|
|
|
match ret_param_typ.desc with Typ.Tptr (t, _) -> t | _ -> assert false
|
|
|
|
in
|
|
|
|
(Exp.Var id, ret_typ, [instr])
|
|
|
|
| None ->
|
|
|
|
(Exp.Lvar (Procdesc.get_ret_var procdesc), ret_type, [])
|
|
|
|
in
|
|
|
|
let trans_state' =
|
|
|
|
{trans_state_pri with succ_nodes= []; var_exp_typ= Some (ret_exp, ret_typ)}
|
|
|
|
in
|
|
|
|
let res_trans_stmt = exec_with_self_exception instruction trans_state' stmt in
|
|
|
|
let sil_expr, _ =
|
|
|
|
extract_exp_from_list res_trans_stmt.exps
|
|
|
|
"WARNING: There should be only one return expression.@\n"
|
|
|
|
in
|
|
|
|
let ret_instrs =
|
|
|
|
if List.exists ~f:(Exp.equal ret_exp) res_trans_stmt.initd_exps then []
|
|
|
|
else [Sil.Store (ret_exp, ret_type, sil_expr, sil_loc)]
|
|
|
|
in
|
|
|
|
let autorelease_instrs = add_autorelease_call context sil_expr ret_type sil_loc in
|
|
|
|
let instrs = var_instrs @ res_trans_stmt.instrs @ ret_instrs @ autorelease_instrs in
|
|
|
|
let ret_node = mk_ret_node instrs in
|
|
|
|
List.iter
|
|
|
|
~f:(fun n -> Procdesc.node_set_succs_exn procdesc n [ret_node] [])
|
|
|
|
res_trans_stmt.leaf_nodes ;
|
|
|
|
let root_nodes_to_parent =
|
|
|
|
if List.length res_trans_stmt.root_nodes > 0 then res_trans_stmt.root_nodes
|
|
|
|
else [ret_node]
|
|
|
|
in
|
|
|
|
{empty_res_trans with root_nodes= root_nodes_to_parent; leaf_nodes= []}
|
|
|
|
| [] ->
|
|
|
|
(* return; *)
|
|
|
|
let ret_node = mk_ret_node [] in
|
|
|
|
{empty_res_trans with root_nodes= [ret_node]; leaf_nodes= []}
|
|
|
|
| _ ->
|
|
|
|
L.(debug Capture Verbose)
|
|
|
|
"@\nWARNING: Missing translation of Return Expression. Return Statement ignored. Need fixing!@\n" ;
|
|
|
|
{empty_res_trans with root_nodes= succ_nodes}
|
|
|
|
in
|
|
|
|
(* We expect a return with only one expression *)
|
|
|
|
trans_result
|
|
|
|
|
|
|
|
|
|
|
|
(* We analyze the content of the expr. We treat ExprWithCleanups as a wrapper. *)
|
|
|
|
(* It may be that later on (when we treat ARC) some info can be taken from it. *)
|
|
|
|
(* For ParenExpression we translate its body composed by the stmt_list. *)
|
|
|
|
(* In paren expression there should be only one stmt that defines the expression *)
|
|
|
|
and parenExpr_trans trans_state stmt_list =
|
|
|
|
let stmt =
|
|
|
|
extract_stmt_from_singleton stmt_list
|
|
|
|
"WARNING: In ParenExpression there should be only one stmt.@\n"
|
|
|
|
in
|
|
|
|
instruction trans_state stmt
|
|
|
|
|
|
|
|
|
|
|
|
and objCBoxedExpr_trans trans_state info sel stmt_info stmts =
|
|
|
|
let typ =
|
|
|
|
CType_decl.class_from_pointer_type trans_state.context.CContext.tenv
|
|
|
|
info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
|
|
|
let obj_c_message_expr_info =
|
|
|
|
Ast_expressions.make_obj_c_message_expr_info_class sel typ None
|
|
|
|
in
|
|
|
|
let message_stmt =
|
|
|
|
Clang_ast_t.ObjCMessageExpr (stmt_info, stmts, info, obj_c_message_expr_info)
|
|
|
|
in
|
|
|
|
instruction trans_state message_stmt
|
|
|
|
|
|
|
|
|
|
|
|
and objCArrayLiteral_trans trans_state info stmt_info stmts =
|
|
|
|
let typ =
|
|
|
|
CType_decl.class_from_pointer_type trans_state.context.CContext.tenv
|
|
|
|
info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let meth = CFrontend_config.array_with_objects_count_m in
|
|
|
|
let obj_c_mes_expr_info = Ast_expressions.make_obj_c_message_expr_info_class meth typ None in
|
|
|
|
let stmts = stmts @ [Ast_expressions.create_nil stmt_info] in
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let message_stmt = Clang_ast_t.ObjCMessageExpr (stmt_info, stmts, info, obj_c_mes_expr_info) in
|
|
|
|
instruction trans_state message_stmt
|
|
|
|
|
|
|
|
|
|
|
|
and objCDictionaryLiteral_trans trans_state info stmt_info stmts =
|
|
|
|
let typ =
|
|
|
|
CType_decl.class_from_pointer_type trans_state.context.CContext.tenv
|
|
|
|
info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
|
|
|
let dictionary_literal_pname = BuiltinDecl.__objc_dictionary_literal in
|
|
|
|
let dictionary_literal_s = Typ.Procname.get_method dictionary_literal_pname in
|
|
|
|
let obj_c_message_expr_info =
|
|
|
|
Ast_expressions.make_obj_c_message_expr_info_class dictionary_literal_s typ None
|
|
|
|
in
|
|
|
|
let stmts = CGeneral_utils.swap_elements_list stmts in
|
|
|
|
let stmts = stmts @ [Ast_expressions.create_nil stmt_info] in
|
|
|
|
let message_stmt =
|
|
|
|
Clang_ast_t.ObjCMessageExpr (stmt_info, stmts, info, obj_c_message_expr_info)
|
|
|
|
in
|
|
|
|
instruction trans_state message_stmt
|
|
|
|
|
|
|
|
|
|
|
|
and objCStringLiteral_trans trans_state stmt_info stmts info =
|
|
|
|
let char_star_typ =
|
|
|
|
Ast_expressions.create_char_star_type ~quals:(Typ.mk_type_quals ~is_const:true ()) ()
|
|
|
|
in
|
|
|
|
let stmts =
|
|
|
|
[Ast_expressions.create_implicit_cast_expr stmt_info stmts char_star_typ `ArrayToPointerDecay]
|
|
|
|
in
|
|
|
|
let typ =
|
|
|
|
CType_decl.class_from_pointer_type trans_state.context.CContext.tenv
|
|
|
|
info.Clang_ast_t.ei_qual_type
|
|
|
|
in
|
Using clang's method resolution if possible
Summary: public
Using clang's method resolution. This means that, in method calls, clang gives you a pointer to the declaration of the method.
In some cases though, clang doesn't find the right method. For example, when it finds a method in a category, we
need to make it into a method in the corresponding class, because that's how we treat categories in Infer. Moreover,
when it finds a method in a protocol, that is not useful for us, since the implementation will be in some class. Finally,
sometimes the call is on an object of type id, in which case clang doesn't know what is the correct declaration. In
those cases, we fall back to what we were doing before of approximating the method resolution. We also refactor
some of the code.
Reviewed By: akotulski
Differential Revision: D2679766
fb-gh-sync-id: b79bb85
9 years ago
|
|
|
let meth = CFrontend_config.string_with_utf8_m in
|
|
|
|
let obj_c_mess_expr_info = Ast_expressions.make_obj_c_message_expr_info_class meth typ None in
|
|
|
|
let message_stmt =
|
|
|
|
Clang_ast_t.ObjCMessageExpr (stmt_info, stmts, info, obj_c_mess_expr_info)
|
|
|
|
in
|
|
|
|
instruction trans_state message_stmt
|
|
|
|
|
|
|
|
|
|
|
|
(** When objects are autoreleased, they get added a flag AUTORELEASE. All these objects will be
|
|
|
|
ignored when checking for memory leaks. When the end of the block autoreleasepool is reached,
|
|
|
|
then those objects are released and the autorelease flag is removed. *)
|
|
|
|
and objcAutoreleasePool_trans trans_state stmt_info stmts =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let fname = BuiltinDecl.__objc_release_autorelease_pool in
|
|
|
|
let ret_id = Some (Ident.create_fresh Ident.knormal, Typ.mk Tvoid) in
|
|
|
|
(* TODO(jjb): change ret_id to None? *)
|
|
|
|
let autorelease_pool_vars = CVar_decl.compute_autorelease_pool_vars context stmts in
|
|
|
|
let stmt_call =
|
|
|
|
Sil.Call
|
|
|
|
(ret_id, Exp.Const (Const.Cfun fname), autorelease_pool_vars, sil_loc, CallFlags.default)
|
|
|
|
in
|
|
|
|
let node_kind = Procdesc.Node.Stmt_node "Release the autorelease pool" in
|
|
|
|
let call_node = create_node node_kind [stmt_call] sil_loc context in
|
|
|
|
Procdesc.node_set_succs_exn context.procdesc call_node trans_state.succ_nodes [] ;
|
|
|
|
let trans_state' = {trans_state with continuation= None; succ_nodes= [call_node]} in
|
|
|
|
instructions trans_state' stmts
|
|
|
|
|
|
|
|
|
|
|
|
(* Assumption: stmt_list contains 2 items, the first can be ObjCMessageExpr or ParenExpr *)
|
|
|
|
(* We ignore this item since we don't deal with the concurrency problem yet *)
|
|
|
|
(* For the same reason we also ignore the stmt_info that
|
|
|
|
is related with the ObjCAtSynchronizedStmt construct *)
|
|
|
|
(* Finally we recursively work on the CompoundStmt, the second item of stmt_list *)
|
|
|
|
and objCAtSynchronizedStmt_trans trans_state stmt_list =
|
|
|
|
match stmt_list with
|
|
|
|
| [_; compound_stmt] ->
|
|
|
|
instruction trans_state compound_stmt
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
and blockExpr_trans trans_state stmt_info expr_info decl =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let procname = Procdesc.get_proc_name context.CContext.procdesc in
|
|
|
|
let loc =
|
|
|
|
match stmt_info.Clang_ast_t.si_source_range with
|
|
|
|
| l1, _ ->
|
|
|
|
CLocation.clang_to_sil_location context.CContext.translation_unit_context l1
|
|
|
|
in
|
|
|
|
(* Given a captured var, return the instruction to assign it to a temp *)
|
|
|
|
let assign_captured_var (cvar, typ) =
|
|
|
|
let id = Ident.create_fresh Ident.knormal in
|
|
|
|
let instr = Sil.Load (id, Exp.Lvar cvar, typ, loc) in
|
|
|
|
(id, instr)
|
|
|
|
in
|
|
|
|
match decl with
|
|
|
|
| Clang_ast_t.BlockDecl (_, block_decl_info) ->
|
|
|
|
let open CContext in
|
|
|
|
let qual_type = expr_info.Clang_ast_t.ei_qual_type in
|
|
|
|
let block_pname = CProcname.mk_fresh_block_procname procname in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type context.tenv qual_type in
|
|
|
|
(* We need to set the explicit dependency between the newly created block and the *)
|
|
|
|
(* defining procedure. We add an edge in the call graph.*)
|
|
|
|
Cg.add_edge context.cg procname block_pname ;
|
|
|
|
let captured_block_vars = block_decl_info.Clang_ast_t.bdi_captured_variables in
|
|
|
|
let captureds = CVar_decl.captured_vars_from_block_info context captured_block_vars in
|
|
|
|
let ids_instrs = List.map ~f:assign_captured_var captureds in
|
|
|
|
let ids, instrs = List.unzip ids_instrs in
|
|
|
|
let block_data = (context, qual_type, block_pname, captureds) in
|
|
|
|
F.function_decl context.translation_unit_context context.tenv context.cfg context.cg decl
|
|
|
|
(Some block_data) ;
|
|
|
|
let captured_vars =
|
|
|
|
List.map2_exn ~f:(fun id (pvar, typ) -> (Exp.Var id, pvar, typ)) ids captureds
|
|
|
|
in
|
|
|
|
let closure = Exp.Closure {name= block_pname; captured_vars} in
|
|
|
|
let block_name = Typ.Procname.to_string block_pname in
|
|
|
|
let static_vars = CContext.static_vars_for_block context block_pname in
|
|
|
|
let captured_static_vars = captureds @ static_vars in
|
|
|
|
let alloc_block_instr = allocate_block trans_state block_name captured_static_vars loc in
|
|
|
|
{empty_res_trans with instrs= alloc_block_instr @ instrs; exps= [(closure, typ)]}
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
and lambdaExpr_trans trans_state stmt_info expr_info {Clang_ast_t.lei_lambda_decl} =
|
|
|
|
let open CContext in
|
|
|
|
let qual_type = expr_info.Clang_ast_t.ei_qual_type in
|
|
|
|
let context = trans_state.context in
|
|
|
|
call_translation context lei_lambda_decl ;
|
|
|
|
let procname = Procdesc.get_proc_name context.procdesc in
|
|
|
|
let lambda_pname = CMethod_trans.get_procname_from_cpp_lambda context lei_lambda_decl in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type context.tenv qual_type in
|
|
|
|
(* We need to set the explicit dependency between the newly created lambda and the *)
|
|
|
|
(* defining procedure. We add an edge in the call graph.*)
|
|
|
|
Cg.add_edge context.cg procname lambda_pname ;
|
|
|
|
let make_captured_tuple (pvar, typ) = (Exp.Lvar pvar, pvar, typ) in
|
|
|
|
let get_captured_pvar_typ decl_ref =
|
|
|
|
CVar_decl.sil_var_of_captured_var decl_ref context procname
|
|
|
|
in
|
|
|
|
let translate_capture_init (pvar, typ) init_decl =
|
|
|
|
match init_decl with
|
|
|
|
| Clang_ast_t.VarDecl (_, _, _, {vdi_init_expr}) ->
|
|
|
|
init_expr_trans trans_state (Exp.Lvar pvar, typ) stmt_info vdi_init_expr
|
|
|
|
| _ ->
|
|
|
|
L.die ExternalError "Unexpected: capture-init statement without var decl"
|
|
|
|
in
|
|
|
|
let translate_captured {Clang_ast_t.lci_captured_var; lci_init_captured_vardecl}
|
|
|
|
((trans_results_acc, captured_vars_acc) as acc) =
|
|
|
|
match (lci_captured_var, lci_init_captured_vardecl) with
|
|
|
|
| Some captured_var_decl_ref, Some init_decl ->
|
|
|
|
(* capture and init *)
|
|
|
|
let pvar_typ = get_captured_pvar_typ captured_var_decl_ref in
|
|
|
|
( translate_capture_init pvar_typ init_decl :: trans_results_acc
|
|
|
|
, make_captured_tuple pvar_typ :: captured_vars_acc )
|
|
|
|
| Some captured_var_decl_ref, None ->
|
|
|
|
(* just capture *)
|
|
|
|
let pvar_typ = get_captured_pvar_typ captured_var_decl_ref in
|
|
|
|
(trans_results_acc, make_captured_tuple pvar_typ :: captured_vars_acc)
|
|
|
|
| None, None ->
|
|
|
|
acc
|
|
|
|
| None, Some _ ->
|
|
|
|
L.die InternalError "Capture-init with init, but no capture"
|
|
|
|
in
|
|
|
|
let lei_captures = CMethod_trans.get_captures_from_cpp_lambda lei_lambda_decl in
|
|
|
|
let trans_results, captured_vars =
|
|
|
|
List.fold_right ~f:translate_captured ~init:([], []) lei_captures
|
|
|
|
in
|
|
|
|
let final_trans_result = collect_res_trans context.procdesc trans_results in
|
|
|
|
let closure = Exp.Closure {name= lambda_pname; captured_vars} in
|
|
|
|
{final_trans_result with exps= [(closure, typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
and cxxNewExpr_trans trans_state stmt_info stmt_list expr_info cxx_new_expr_info =
|
|
|
|
let instructions_trans_result = instructions trans_state stmt_list in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info context.tenv in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let is_dyn_array = cxx_new_expr_info.Clang_ast_t.xnei_is_array in
|
|
|
|
let size_exp_opt, res_trans_size =
|
|
|
|
if is_dyn_array then
|
|
|
|
match CAst_utils.get_stmt_opt cxx_new_expr_info.Clang_ast_t.xnei_array_size_expr with
|
|
|
|
| Some stmt
|
|
|
|
-> (
|
|
|
|
let trans_state_size = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let res_trans_size = instruction trans_state_size stmt in
|
|
|
|
match res_trans_size.exps with
|
|
|
|
| [(exp, _)] ->
|
|
|
|
(Some exp, res_trans_size)
|
|
|
|
| _ ->
|
|
|
|
(None, empty_res_trans) )
|
|
|
|
| None ->
|
|
|
|
(Some (Exp.Const (Const.Cint IntLit.minus_one)), empty_res_trans)
|
|
|
|
else (None, empty_res_trans)
|
|
|
|
in
|
|
|
|
let res_trans_new = cpp_new_trans sil_loc typ size_exp_opt in
|
|
|
|
let stmt_opt = CAst_utils.get_stmt_opt cxx_new_expr_info.Clang_ast_t.xnei_initializer_expr in
|
|
|
|
let trans_state_init = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let var_exp_typ =
|
|
|
|
match res_trans_new.exps with
|
|
|
|
| [(var_exp, ({desc= Tptr (t, _)} as var_typ))] when is_dyn_array ->
|
|
|
|
(* represent dynamic array as Tarray *)
|
|
|
|
(var_exp, Typ.mk ~default:var_typ (Typ.Tarray (t, None, None)))
|
|
|
|
| [(var_exp, {desc= Tptr (t, _)})] when not is_dyn_array ->
|
|
|
|
(var_exp, t)
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
(* Need a new stmt_info for the translation of the initializer, so that it can create nodes *)
|
|
|
|
(* if it needs to, with the same stmt_info it doesn't work. *)
|
|
|
|
let init_stmt_info =
|
|
|
|
{stmt_info with Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()}
|
|
|
|
in
|
|
|
|
let res_trans_init =
|
|
|
|
match stmt_opt with
|
|
|
|
| Some InitListExpr _ ->
|
|
|
|
[init_expr_trans trans_state_init var_exp_typ init_stmt_info stmt_opt]
|
|
|
|
| _ when is_dyn_array && Typ.is_pointer_to_cpp_class typ ->
|
|
|
|
(* NOTE: this is heuristic to initialize C++ objects when the size of dynamic
|
|
|
|
array is constant, it doesn't do anything for non-const lengths, it should be translated as a loop *)
|
|
|
|
let rec create_stmts stmt_opt size_exp_opt =
|
|
|
|
match (stmt_opt, size_exp_opt) with
|
|
|
|
| Some stmt, Some Exp.Const Const.Cint n when not (IntLit.iszero n) ->
|
|
|
|
let n_minus_1 = Some (Exp.Const (Const.Cint (IntLit.sub n IntLit.one))) in
|
|
|
|
stmt :: create_stmts stmt_opt n_minus_1
|
|
|
|
| _ ->
|
|
|
|
[]
|
|
|
|
in
|
|
|
|
let stmts = create_stmts stmt_opt size_exp_opt in
|
|
|
|
let var_exp, var_typ = var_exp_typ in
|
|
|
|
initListExpr_array_trans trans_state_init init_stmt_info stmts var_exp var_typ
|
|
|
|
| _ ->
|
|
|
|
[init_expr_trans trans_state_init var_exp_typ init_stmt_info stmt_opt]
|
|
|
|
in
|
|
|
|
let all_res_trans = [res_trans_size; res_trans_new] @ res_trans_init in
|
|
|
|
let nname = "CXXNewExpr" in
|
|
|
|
let result_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname stmt_info all_res_trans
|
|
|
|
in
|
|
|
|
collect_res_trans context.CContext.procdesc
|
|
|
|
[instructions_trans_result; {result_trans_to_parent with exps= res_trans_new.exps}]
|
|
|
|
|
|
|
|
|
|
|
|
and cxxDeleteExpr_trans trans_state stmt_info stmt_list delete_expr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let is_array = delete_expr_info.Clang_ast_t.xdei_is_array in
|
|
|
|
let fname = if is_array then BuiltinDecl.__delete_array else BuiltinDecl.__delete in
|
|
|
|
let param = match stmt_list with [p] -> p | _ -> assert false in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let trans_state_param = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let result_trans_param = exec_with_self_exception instruction trans_state_param param in
|
|
|
|
let exp =
|
|
|
|
extract_exp_from_list result_trans_param.exps
|
|
|
|
"WARNING: There should be one expression to delete. @\n"
|
|
|
|
in
|
|
|
|
let call_instr =
|
|
|
|
Sil.Call (None, Exp.Const (Const.Cfun fname), [exp], sil_loc, CallFlags.default)
|
|
|
|
in
|
|
|
|
let call_res_trans = {empty_res_trans with instrs= [call_instr]} in
|
|
|
|
let all_res_trans =
|
|
|
|
if false then
|
|
|
|
(* FIXME (t10135167): call destructor on deleted pointer if it's not null *)
|
|
|
|
(* Right now it's dead code hidden by the 'false' flag *)
|
|
|
|
let deleted_type = delete_expr_info.Clang_ast_t.xdei_destroyed_type in
|
|
|
|
(* create stmt_info with new pointer so that destructor call doesn't create a node *)
|
|
|
|
let destruct_stmt_info =
|
|
|
|
{stmt_info with Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()}
|
|
|
|
in
|
|
|
|
(* use empty_res_trans to avoid ending up with same instruction twice *)
|
|
|
|
(* otherwise it would happen due to structutre of all_res_trans *)
|
|
|
|
let this_res_trans_destruct = {empty_res_trans with exps= result_trans_param.exps} in
|
|
|
|
let destruct_res_trans =
|
|
|
|
cxx_destructor_call_trans trans_state_pri destruct_stmt_info this_res_trans_destruct
|
|
|
|
deleted_type.Clang_ast_t.qt_type_ptr ~is_inner_destructor:false
|
|
|
|
in
|
|
|
|
[result_trans_param; destruct_res_trans; call_res_trans] (* --- END OF DEAD CODE --- *)
|
|
|
|
else [result_trans_param; call_res_trans]
|
|
|
|
in
|
|
|
|
let res_trans =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "Call delete" stmt_info
|
|
|
|
all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans with exps= []}
|
|
|
|
|
|
|
|
|
|
|
|
and materializeTemporaryExpr_trans trans_state stmt_info stmt_list expr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let procdesc = context.CContext.procdesc in
|
|
|
|
(* typ_tmp is 'best guess' type of variable - translation may decide to use different type
|
|
|
|
later *)
|
|
|
|
let pvar, typ_tmp =
|
|
|
|
mk_temp_sil_var_for_expr context.CContext.tenv procdesc "SIL_materialize_temp__" expr_info
|
|
|
|
in
|
|
|
|
let temp_exp = match stmt_list with [p] -> p | _ -> assert false in
|
|
|
|
let var_exp_typ = (Exp.Lvar pvar, typ_tmp) in
|
|
|
|
let res_trans = init_expr_trans trans_state var_exp_typ stmt_info (Some temp_exp) in
|
|
|
|
let _, typ = extract_exp_from_list res_trans.exps "MaterializeExpr initializer missing@\n" in
|
|
|
|
Procdesc.append_locals procdesc [(Pvar.get_name pvar, typ)] ;
|
|
|
|
res_trans
|
|
|
|
|
|
|
|
|
|
|
|
and compoundLiteralExpr_trans trans_state stmt_list expr_info =
|
|
|
|
let stmt = match stmt_list with [stmt] -> stmt | _ -> assert false in
|
|
|
|
let var_exp_typ =
|
|
|
|
if Option.is_some trans_state.var_exp_typ then trans_state.var_exp_typ
|
|
|
|
else Some (create_var_exp_tmp_var trans_state expr_info "SIL_compound_literal__")
|
|
|
|
in
|
|
|
|
let trans_state' = {trans_state with var_exp_typ} in
|
|
|
|
instruction trans_state' stmt
|
|
|
|
|
|
|
|
|
|
|
|
and cxxDynamicCastExpr_trans trans_state stmt_info stmts cast_qual_type =
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let trans_state' = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let context = trans_state.context in
|
|
|
|
let subtype = Subtype.subtypes_cast in
|
|
|
|
let tenv = context.CContext.tenv in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info context in
|
|
|
|
let cast_type = CType_decl.qual_type_to_sil_type tenv cast_qual_type in
|
|
|
|
let sizeof_expr =
|
|
|
|
match cast_type.desc with
|
|
|
|
| Typ.Tptr (typ, _) ->
|
|
|
|
Exp.Sizeof {typ; nbytes= None; dynamic_length= None; subtype}
|
|
|
|
| _ ->
|
|
|
|
assert false
|
|
|
|
in
|
|
|
|
let builtin = Exp.Const (Const.Cfun BuiltinDecl.__cast) in
|
|
|
|
let stmt = match stmts with [stmt] -> stmt | _ -> assert false in
|
|
|
|
let res_trans_stmt = exec_with_glvalue_as_reference instruction trans_state' stmt in
|
|
|
|
let exp = match res_trans_stmt.exps with [e] -> e | _ -> assert false in
|
|
|
|
let args = [exp; (sizeof_expr, Typ.mk Tvoid)] in
|
|
|
|
let ret_id = Ident.create_fresh Ident.knormal in
|
|
|
|
let call = Sil.Call (Some (ret_id, cast_type), builtin, args, sil_loc, CallFlags.default) in
|
|
|
|
let res_ex = Exp.Var ret_id in
|
|
|
|
let res_trans_dynamic_cast = {empty_res_trans with instrs= [call]} in
|
|
|
|
let all_res_trans = [res_trans_stmt; res_trans_dynamic_cast] in
|
|
|
|
let nname = "CxxDynamicCast" in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc nname stmt_info all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans_to_parent with exps= [(res_ex, cast_type)]}
|
|
|
|
|
|
|
|
|
|
|
|
and cxxDefaultExpr_trans trans_state default_expr_info =
|
|
|
|
match default_expr_info.Clang_ast_t.xdaei_init_expr with
|
|
|
|
| Some exp ->
|
|
|
|
instruction trans_state exp
|
|
|
|
| None ->
|
|
|
|
assert false
|
|
|
|
|
|
|
|
|
|
|
|
and call_function_with_args instr_name pname trans_state stmt_info stmts =
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let trans_state_param = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let res_trans_subexpr_list =
|
|
|
|
List.map ~f:(exec_with_glvalue_as_reference instruction trans_state_param) stmts
|
|
|
|
in
|
|
|
|
let params = collect_exprs res_trans_subexpr_list in
|
|
|
|
let sil_fun = Exp.Const (Const.Cfun pname) in
|
|
|
|
let call_instr = Sil.Call (None, sil_fun, params, sil_loc, CallFlags.default) in
|
|
|
|
let res_trans_call = {empty_res_trans with instrs= [call_instr]; exps= []} in
|
|
|
|
let all_res_trans = res_trans_subexpr_list @ [res_trans_call] in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc instr_name stmt_info
|
|
|
|
all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans_to_parent with exps= res_trans_call.exps}
|
|
|
|
|
|
|
|
|
|
|
|
and gccAsmStmt_trans trans_state =
|
|
|
|
let pname = Typ.Procname.from_string_c_fun CFrontend_config.infer_skip_gcc_asm_stmt in
|
|
|
|
call_function_with_args "GCCAsmStmt" pname trans_state
|
|
|
|
|
|
|
|
|
|
|
|
and objc_cxx_throw_trans trans_state =
|
|
|
|
call_function_with_args "ObjCCPPThrow" BuiltinDecl.objc_cpp_throw trans_state
|
|
|
|
|
|
|
|
|
|
|
|
and cxxPseudoDestructorExpr_trans () =
|
|
|
|
let fun_name = Typ.Procname.from_string_c_fun CFrontend_config.infer_skip_fun in
|
|
|
|
{empty_res_trans with exps= [(Exp.Const (Const.Cfun fun_name), Typ.mk Tvoid)]}
|
|
|
|
|
|
|
|
|
|
|
|
and cxxTypeidExpr_trans trans_state stmt_info stmts expr_info =
|
|
|
|
let tenv = trans_state.context.CContext.tenv in
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info tenv in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let res_trans_subexpr =
|
|
|
|
match stmts with
|
|
|
|
| [stmt] ->
|
|
|
|
let trans_state_param = {trans_state_pri with succ_nodes= []} in
|
|
|
|
instruction trans_state_param stmt
|
|
|
|
| _ ->
|
|
|
|
empty_res_trans
|
|
|
|
in
|
|
|
|
let fun_name = BuiltinDecl.__cxx_typeid in
|
|
|
|
let sil_fun = Exp.Const (Const.Cfun fun_name) in
|
|
|
|
let ret_id = Ident.create_fresh Ident.knormal in
|
|
|
|
let void_typ = Typ.mk Tvoid in
|
|
|
|
let type_info_objc =
|
|
|
|
(Exp.Sizeof {typ; nbytes= None; dynamic_length= None; subtype= Subtype.exact}, void_typ)
|
|
|
|
in
|
|
|
|
let class_tname =
|
|
|
|
Typ.Name.Cpp.from_qual_name Typ.NoTemplate (QualifiedCppName.of_list ["std"; "type_info"])
|
|
|
|
in
|
|
|
|
let field_name = CGeneral_utils.mk_class_field_name class_tname "__type_name" in
|
|
|
|
let ret_exp = Exp.Var ret_id in
|
|
|
|
let field_exp = Exp.Lfield (ret_exp, field_name, typ) in
|
|
|
|
let args = type_info_objc :: (field_exp, void_typ) :: res_trans_subexpr.exps in
|
|
|
|
let call_instr = Sil.Call (Some (ret_id, typ), sil_fun, args, sil_loc, CallFlags.default) in
|
|
|
|
let res_trans_call = {empty_res_trans with instrs= [call_instr]; exps= [(ret_exp, typ)]} in
|
|
|
|
let all_res_trans = [res_trans_subexpr; res_trans_call] in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "CXXTypeidExpr" stmt_info
|
|
|
|
all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans_to_parent with exps= res_trans_call.exps}
|
|
|
|
|
|
|
|
|
|
|
|
and cxxStdInitializerListExpr_trans trans_state stmt_info stmts expr_info =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let tenv = context.CContext.tenv in
|
|
|
|
let sil_loc = CLocation.get_sil_location stmt_info trans_state.context in
|
|
|
|
let typ = CType_decl.qual_type_to_sil_type tenv expr_info.Clang_ast_t.ei_qual_type in
|
|
|
|
let fun_name = Typ.Procname.from_string_c_fun CFrontend_config.infer_skip_fun in
|
|
|
|
let trans_state_pri = PriorityNode.try_claim_priority_node trans_state stmt_info in
|
|
|
|
let trans_state_param = {trans_state_pri with succ_nodes= []} in
|
|
|
|
let res_trans_subexpr_list = List.map ~f:(instruction trans_state_param) stmts in
|
|
|
|
let params = collect_exprs res_trans_subexpr_list in
|
|
|
|
let sil_fun = Exp.Const (Const.Cfun fun_name) in
|
|
|
|
let ret_id = Ident.create_fresh Ident.knormal in
|
|
|
|
let ret_exp = Exp.Var ret_id in
|
|
|
|
let call_instr = Sil.Call (Some (ret_id, typ), sil_fun, params, sil_loc, CallFlags.default) in
|
|
|
|
let res_trans_call = {empty_res_trans with instrs= [call_instr]; exps= [(ret_exp, typ)]} in
|
|
|
|
let all_res_trans = res_trans_subexpr_list @ [res_trans_call] in
|
|
|
|
let res_trans_to_parent =
|
|
|
|
PriorityNode.compute_results_to_parent trans_state_pri sil_loc "CXXStdInitializerListExpr"
|
|
|
|
stmt_info all_res_trans
|
|
|
|
in
|
|
|
|
{res_trans_to_parent with exps= res_trans_call.exps}
|
|
|
|
|
|
|
|
|
|
|
|
and objCBridgedCastExpr_trans trans_state stmts expr_info =
|
|
|
|
let stmt = extract_stmt_from_singleton stmts "" in
|
|
|
|
let tenv = trans_state.context.CContext.tenv in
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info tenv in
|
|
|
|
let trans_state' = {trans_state with obj_bridged_cast_typ= Some typ} in
|
|
|
|
instruction trans_state' stmt
|
|
|
|
|
|
|
|
|
|
|
|
and binaryOperator_trans_with_cond trans_state stmt_info stmt_list expr_info binop_info =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match binop_info.boi_kind with
|
|
|
|
| `LAnd | `LOr | `LT | `GT | `LE | `GE | `EQ | `NE ->
|
|
|
|
(* For LAnd/LOr/comparison operators we compiles a binary expression bo into an semantic
|
|
|
|
equivalent conditional operator 'bo ? 1:0'.
|
|
|
|
The conditional operator takes care of shortcircuit when/where needed *)
|
|
|
|
let bo = BinaryOperator (stmt_info, stmt_list, expr_info, binop_info) in
|
|
|
|
let cond = Ast_expressions.trans_with_conditional stmt_info expr_info [bo] in
|
|
|
|
instruction trans_state cond
|
|
|
|
| _ ->
|
|
|
|
binaryOperator_trans trans_state binop_info stmt_info expr_info stmt_list
|
|
|
|
|
|
|
|
|
|
|
|
and attributedStmt_trans trans_state stmts attrs =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match (stmts, attrs) with
|
|
|
|
| [stmt], [attr] -> (
|
|
|
|
match (stmt, attr) with
|
|
|
|
| NullStmt _, FallThroughAttr _ ->
|
|
|
|
no_op_trans trans_state.succ_nodes
|
|
|
|
| _ ->
|
|
|
|
CFrontend_config.unimplemented
|
|
|
|
"attributedStmt [stmt] [attr] with:@\nstmt=%s@\nattr=%s@\n"
|
|
|
|
(Clang_ast_j.string_of_stmt stmt)
|
|
|
|
(Clang_ast_j.string_of_attribute attr) )
|
|
|
|
| _ ->
|
|
|
|
CFrontend_config.unimplemented "attributedStmt with:@\nstmts=[%a]@\nattrs=[%a]@\n"
|
|
|
|
(Pp.semicolon_seq (Pp.to_string ~f:Clang_ast_j.string_of_stmt))
|
|
|
|
stmts
|
|
|
|
(Pp.semicolon_seq (Pp.to_string ~f:Clang_ast_j.string_of_attribute))
|
|
|
|
attrs
|
|
|
|
|
|
|
|
|
|
|
|
and breakStmt_trans trans_state stmt_info =
|
|
|
|
match trans_state.continuation with
|
|
|
|
| Some bn ->
|
|
|
|
let trans_state' = {trans_state with succ_nodes= bn.break} in
|
|
|
|
let destr_trans_result = inject_destructors trans_state' stmt_info in
|
|
|
|
if destr_trans_result.root_nodes <> [] then destr_trans_result
|
|
|
|
else {empty_res_trans with root_nodes= bn.break}
|
|
|
|
| _ (* t21762295 *) ->
|
|
|
|
CFrontend_config.incorrect_assumption "Break stmt without continuation: %a"
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_stmt_info)
|
|
|
|
stmt_info
|
|
|
|
|
|
|
|
|
|
|
|
and continueStmt_trans trans_state stmt_info =
|
|
|
|
match trans_state.continuation with
|
|
|
|
| Some bn ->
|
|
|
|
let trans_state' = {trans_state with succ_nodes= bn.continue} in
|
|
|
|
let destr_trans_result = inject_destructors trans_state' stmt_info in
|
|
|
|
if destr_trans_result.root_nodes <> [] then destr_trans_result
|
|
|
|
else {empty_res_trans with root_nodes= bn.continue}
|
|
|
|
| _ (* t21762295 *) ->
|
|
|
|
CFrontend_config.incorrect_assumption "Continue stmt without continuation: %a"
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_stmt_info)
|
|
|
|
stmt_info
|
|
|
|
|
|
|
|
|
|
|
|
(* Expect that this doesn't happen *)
|
|
|
|
and trans_into_undefined_expr trans_state expr_info =
|
|
|
|
let tenv = trans_state.context.CContext.tenv in
|
|
|
|
let typ = CType_decl.get_type_from_expr_info expr_info tenv in
|
|
|
|
{empty_res_trans with exps= [(CTrans_utils.undefined_expression (), typ)]}
|
|
|
|
|
|
|
|
|
|
|
|
(* no-op translated for unsupported instructions that will at least translate subexpressions *)
|
|
|
|
and skip_unimplemented trans_state stmts = instructions trans_state stmts
|
|
|
|
|
|
|
|
(* Translates a clang instruction into SIL instructions. It takes a *)
|
|
|
|
(* a trans_state containing current info on the translation and it returns *)
|
|
|
|
(* a result_state.*)
|
|
|
|
and instruction trans_state instr =
|
|
|
|
let stmt_kind = Clang_ast_proj.get_stmt_kind_string instr in
|
|
|
|
let stmt_info, _ = Clang_ast_proj.get_stmt_tuple instr in
|
|
|
|
let stmt_pointer = stmt_info.Clang_ast_t.si_pointer in
|
|
|
|
L.(debug Capture Verbose) "@\nPassing from %s '%d' @\n" stmt_kind stmt_pointer ;
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match instr with
|
|
|
|
| GotoStmt (stmt_info, _, {Clang_ast_t.gsi_label= label_name; _}) ->
|
|
|
|
gotoStmt_trans trans_state stmt_info label_name
|
|
|
|
| LabelStmt (stmt_info, stmt_list, label_name) ->
|
|
|
|
labelStmt_trans trans_state stmt_info stmt_list label_name
|
|
|
|
| ArraySubscriptExpr (_, stmt_list, expr_info) ->
|
|
|
|
arraySubscriptExpr_trans trans_state expr_info stmt_list
|
|
|
|
| BinaryOperator (stmt_info, stmt_list, expr_info, binop_info) ->
|
|
|
|
binaryOperator_trans_with_cond trans_state stmt_info stmt_list expr_info binop_info
|
|
|
|
| CallExpr (stmt_info, stmt_list, ei) -> (
|
|
|
|
match is_dispatch_function stmt_list with
|
|
|
|
| Some block_arg_pos ->
|
|
|
|
dispatch_function_trans trans_state stmt_info stmt_list block_arg_pos
|
|
|
|
| None ->
|
|
|
|
callExpr_trans trans_state stmt_info stmt_list ei )
|
|
|
|
| CXXMemberCallExpr (stmt_info, stmt_list, ei) ->
|
|
|
|
cxxMemberCallExpr_trans trans_state stmt_info stmt_list ei
|
|
|
|
| CXXOperatorCallExpr (stmt_info, stmt_list, ei) ->
|
|
|
|
callExpr_trans trans_state stmt_info stmt_list ei
|
|
|
|
| CXXConstructExpr (stmt_info, stmt_list, expr_info, cxx_constr_info)
|
|
|
|
| CXXTemporaryObjectExpr (stmt_info, stmt_list, expr_info, cxx_constr_info) ->
|
|
|
|
cxxConstructExpr_trans trans_state stmt_info stmt_list expr_info cxx_constr_info
|
|
|
|
| ObjCMessageExpr (stmt_info, stmt_list, expr_info, obj_c_message_expr_info) ->
|
|
|
|
if is_block_enumerate_function obj_c_message_expr_info then
|
|
|
|
block_enumeration_trans trans_state stmt_info stmt_list expr_info
|
|
|
|
else
|
|
|
|
objCMessageExpr_trans trans_state stmt_info obj_c_message_expr_info stmt_list expr_info
|
|
|
|
| CompoundStmt (stmt_info, stmt_list) ->
|
|
|
|
(* No node for this statement. We just collect its statement list*)
|
|
|
|
compoundStmt_trans trans_state stmt_info stmt_list
|
|
|
|
| ConditionalOperator (stmt_info, stmt_list, expr_info) ->
|
|
|
|
(* Ternary operator "cond ? exp1 : exp2" *)
|
|
|
|
conditionalOperator_trans trans_state stmt_info stmt_list expr_info
|
|
|
|
| IfStmt (stmt_info, stmt_list) ->
|
|
|
|
ifStmt_trans trans_state stmt_info stmt_list
|
|
|
|
| SwitchStmt (stmt_info, switch_stmt_list) ->
|
|
|
|
switchStmt_trans trans_state stmt_info switch_stmt_list
|
|
|
|
| CaseStmt _ ->
|
|
|
|
(* where do we even get case stmts outside of the switch stmt? (t21762295) *)
|
|
|
|
CFrontend_config.incorrect_assumption "Case statement outside of switch statement: %a"
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_stmt)
|
|
|
|
instr
|
|
|
|
| StmtExpr (_, stmt_list, _) ->
|
|
|
|
stmtExpr_trans trans_state stmt_list
|
|
|
|
| ForStmt (stmt_info, [init; decl_stmt; cond; incr; body]) ->
|
|
|
|
forStmt_trans trans_state init decl_stmt cond incr body stmt_info
|
|
|
|
| WhileStmt (stmt_info, [decl_stmt; cond; body]) ->
|
|
|
|
whileStmt_trans trans_state decl_stmt cond body stmt_info
|
|
|
|
| DoStmt (stmt_info, [body; cond]) ->
|
|
|
|
doStmt_trans trans_state stmt_info cond body
|
|
|
|
| CXXForRangeStmt (stmt_info, stmt_list) ->
|
|
|
|
cxxForRangeStmt_trans trans_state stmt_info stmt_list
|
|
|
|
| ObjCForCollectionStmt (stmt_info, [item; items; body]) ->
|
|
|
|
objCForCollectionStmt_trans trans_state item items body stmt_info
|
|
|
|
| NullStmt _ ->
|
|
|
|
no_op_trans trans_state.succ_nodes
|
|
|
|
| CompoundAssignOperator (stmt_info, stmt_list, expr_info, binary_operator_info, _) ->
|
|
|
|
binaryOperator_trans trans_state binary_operator_info stmt_info expr_info stmt_list
|
|
|
|
| DeclStmt (stmt_info, _, decl_list) ->
|
|
|
|
declStmt_trans trans_state decl_list stmt_info
|
|
|
|
| DeclRefExpr (stmt_info, _, _, decl_ref_expr_info) as d ->
|
|
|
|
declRefExpr_trans trans_state stmt_info decl_ref_expr_info d
|
|
|
|
| ObjCPropertyRefExpr (_, stmt_list, _, _) ->
|
|
|
|
objCPropertyRefExpr_trans trans_state stmt_list
|
|
|
|
| CXXThisExpr (stmt_info, _, expr_info) ->
|
|
|
|
cxxThisExpr_trans trans_state stmt_info expr_info
|
|
|
|
| OpaqueValueExpr (_, _, _, opaque_value_expr_info) ->
|
|
|
|
opaqueValueExpr_trans trans_state opaque_value_expr_info
|
|
|
|
| PseudoObjectExpr (_, stmt_list, _) ->
|
|
|
|
pseudoObjectExpr_trans trans_state stmt_list
|
|
|
|
| UnaryExprOrTypeTraitExpr (_, _, expr_info, ei) ->
|
|
|
|
unaryExprOrTypeTraitExpr_trans trans_state expr_info ei
|
|
|
|
| ObjCBridgedCastExpr (_, stmt_list, expr_info, _, _) ->
|
|
|
|
objCBridgedCastExpr_trans trans_state stmt_list expr_info
|
|
|
|
| ImplicitCastExpr (stmt_info, stmt_list, expr_info, cast_kind)
|
|
|
|
| CStyleCastExpr (stmt_info, stmt_list, expr_info, cast_kind, _)
|
|
|
|
| CXXReinterpretCastExpr (stmt_info, stmt_list, expr_info, cast_kind, _, _)
|
|
|
|
| CXXConstCastExpr (stmt_info, stmt_list, expr_info, cast_kind, _, _)
|
|
|
|
| CXXStaticCastExpr (stmt_info, stmt_list, expr_info, cast_kind, _, _)
|
|
|
|
| CXXFunctionalCastExpr (stmt_info, stmt_list, expr_info, cast_kind, _) ->
|
|
|
|
cast_exprs_trans trans_state stmt_info stmt_list expr_info cast_kind
|
|
|
|
| IntegerLiteral (_, _, expr_info, integer_literal_info) ->
|
|
|
|
integerLiteral_trans trans_state expr_info integer_literal_info
|
|
|
|
| StringLiteral (_, _, expr_info, str_list) ->
|
|
|
|
stringLiteral_trans trans_state expr_info (String.concat ~sep:"" str_list)
|
|
|
|
| GNUNullExpr (_, _, expr_info) ->
|
|
|
|
gNUNullExpr_trans trans_state expr_info
|
|
|
|
| CXXNullPtrLiteralExpr (_, _, expr_info) ->
|
|
|
|
nullPtrExpr_trans trans_state expr_info
|
|
|
|
| ObjCSelectorExpr (_, _, expr_info, selector) ->
|
|
|
|
objCSelectorExpr_trans trans_state expr_info selector
|
|
|
|
| ObjCEncodeExpr (_, _, expr_info, objc_encode_expr_info) ->
|
|
|
|
objCEncodeExpr_trans trans_state expr_info objc_encode_expr_info
|
|
|
|
| ObjCProtocolExpr (_, _, expr_info, decl_ref) ->
|
|
|
|
objCProtocolExpr_trans trans_state expr_info decl_ref
|
|
|
|
| ObjCIvarRefExpr (stmt_info, stmt_list, _, obj_c_ivar_ref_expr_info) ->
|
|
|
|
objCIvarRefExpr_trans trans_state stmt_info stmt_list obj_c_ivar_ref_expr_info
|
|
|
|
| MemberExpr (stmt_info, stmt_list, _, member_expr_info) ->
|
|
|
|
memberExpr_trans trans_state stmt_info stmt_list member_expr_info
|
|
|
|
| UnaryOperator (stmt_info, stmt_list, expr_info, unary_operator_info) ->
|
|
|
|
if is_logical_negation_of_int trans_state.context.CContext.tenv expr_info
|
|
|
|
unary_operator_info
|
|
|
|
then
|
|
|
|
let conditional =
|
|
|
|
Ast_expressions.trans_negation_with_conditional stmt_info expr_info stmt_list
|
|
|
|
in
|
|
|
|
instruction trans_state conditional
|
|
|
|
else unaryOperator_trans trans_state stmt_info expr_info stmt_list unary_operator_info
|
|
|
|
| ReturnStmt (stmt_info, stmt_list) ->
|
|
|
|
returnStmt_trans trans_state stmt_info stmt_list
|
|
|
|
(* We analyze the content of the expr. We treat ExprWithCleanups as a wrapper. *)
|
|
|
|
(* It may be that later on (when we treat ARC) some info can be taken from it. *)
|
|
|
|
| ExprWithCleanups (_, stmt_list, _, _)
|
|
|
|
| ParenExpr (_, stmt_list, _) ->
|
|
|
|
parenExpr_trans trans_state stmt_list
|
|
|
|
| ObjCBoolLiteralExpr (_, _, expr_info, n)
|
|
|
|
| CharacterLiteral (_, _, expr_info, n)
|
|
|
|
| CXXBoolLiteralExpr (_, _, expr_info, n) ->
|
|
|
|
characterLiteral_trans trans_state expr_info n
|
|
|
|
| FloatingLiteral (_, _, expr_info, float_string) ->
|
|
|
|
floatingLiteral_trans trans_state expr_info float_string
|
|
|
|
| CXXScalarValueInitExpr (_, _, expr_info) ->
|
|
|
|
cxxScalarValueInitExpr_trans trans_state expr_info
|
|
|
|
| ObjCBoxedExpr (stmt_info, stmts, info, boxed_expr_info) -> (
|
|
|
|
match boxed_expr_info.Clang_ast_t.obei_boxing_method with
|
|
|
|
| Some sel ->
|
|
|
|
objCBoxedExpr_trans trans_state info sel stmt_info stmts
|
|
|
|
| None ->
|
|
|
|
assert false )
|
|
|
|
| ObjCArrayLiteral (stmt_info, stmts, info) ->
|
|
|
|
objCArrayLiteral_trans trans_state info stmt_info stmts
|
|
|
|
| ObjCDictionaryLiteral (stmt_info, stmts, info) ->
|
|
|
|
objCDictionaryLiteral_trans trans_state info stmt_info stmts
|
|
|
|
| ObjCStringLiteral (stmt_info, stmts, info) ->
|
|
|
|
objCStringLiteral_trans trans_state stmt_info stmts info
|
|
|
|
| BreakStmt (stmt_info, _) ->
|
|
|
|
breakStmt_trans trans_state stmt_info
|
|
|
|
| ContinueStmt (stmt_info, _) ->
|
|
|
|
continueStmt_trans trans_state stmt_info
|
|
|
|
| ObjCAtSynchronizedStmt (_, stmt_list) ->
|
|
|
|
objCAtSynchronizedStmt_trans trans_state stmt_list
|
|
|
|
| ObjCIndirectCopyRestoreExpr (_, stmt_list, _) ->
|
|
|
|
instructions trans_state stmt_list
|
|
|
|
| BlockExpr (stmt_info, _, expr_info, decl) ->
|
|
|
|
blockExpr_trans trans_state stmt_info expr_info decl
|
|
|
|
| ObjCAutoreleasePoolStmt (stmt_info, stmts) ->
|
|
|
|
objcAutoreleasePool_trans trans_state stmt_info stmts
|
|
|
|
| ObjCAtTryStmt (stmt_info, stmts) ->
|
|
|
|
compoundStmt_trans trans_state stmt_info stmts
|
|
|
|
| CXXTryStmt (stmt_info, stmts) ->
|
|
|
|
L.(debug Capture Medium)
|
|
|
|
"@\n!!!!WARNING: found statement %s. @\nTranslation need to be improved.... @\n"
|
|
|
|
(Clang_ast_proj.get_stmt_kind_string instr) ;
|
|
|
|
compoundStmt_trans trans_state stmt_info stmts
|
|
|
|
| ObjCAtThrowStmt (stmt_info, stmts) | CXXThrowExpr (stmt_info, stmts, _) ->
|
|
|
|
objc_cxx_throw_trans trans_state stmt_info stmts
|
|
|
|
| ObjCAtFinallyStmt (stmt_info, stmts) ->
|
|
|
|
compoundStmt_trans trans_state stmt_info stmts
|
|
|
|
| ObjCAtCatchStmt (stmt_info, _, _) | CXXCatchStmt (stmt_info, _, _) ->
|
|
|
|
compoundStmt_trans trans_state stmt_info []
|
|
|
|
| PredefinedExpr (_, _, expr_info, _) ->
|
|
|
|
stringLiteral_trans trans_state expr_info ""
|
|
|
|
| BinaryConditionalOperator (stmt_info, stmts, expr_info) ->
|
|
|
|
binaryConditionalOperator_trans trans_state stmt_info stmts expr_info
|
|
|
|
| CXXNewExpr (stmt_info, stmt_list, expr_info, cxx_new_expr_info) ->
|
|
|
|
cxxNewExpr_trans trans_state stmt_info stmt_list expr_info cxx_new_expr_info
|
|
|
|
| CXXDeleteExpr (stmt_info, stmt_list, _, delete_expr_info) ->
|
|
|
|
cxxDeleteExpr_trans trans_state stmt_info stmt_list delete_expr_info
|
|
|
|
| MaterializeTemporaryExpr (stmt_info, stmt_list, expr_info, _) ->
|
|
|
|
materializeTemporaryExpr_trans trans_state stmt_info stmt_list expr_info
|
|
|
|
| CompoundLiteralExpr (_, stmt_list, expr_info) ->
|
|
|
|
compoundLiteralExpr_trans trans_state stmt_list expr_info
|
|
|
|
| InitListExpr (stmt_info, stmts, expr_info) ->
|
|
|
|
initListExpr_trans trans_state stmt_info expr_info stmts
|
|
|
|
| CXXBindTemporaryExpr (_, stmt_list, _, _) ->
|
|
|
|
(* right now we ignore this expression and try to translate the child node *)
|
|
|
|
parenExpr_trans trans_state stmt_list
|
|
|
|
| CXXDynamicCastExpr (stmt_info, stmts, _, _, qual_type, _) ->
|
|
|
|
cxxDynamicCastExpr_trans trans_state stmt_info stmts qual_type
|
|
|
|
| CXXDefaultArgExpr (_, _, _, default_expr_info)
|
|
|
|
| CXXDefaultInitExpr (_, _, _, default_expr_info) ->
|
|
|
|
cxxDefaultExpr_trans trans_state default_expr_info
|
|
|
|
| ImplicitValueInitExpr (stmt_info, _, _) ->
|
|
|
|
implicitValueInitExpr_trans trans_state stmt_info
|
|
|
|
| GenericSelectionExpr _
|
|
|
|
(* to be fixed when we dump the right info in the ast *)
|
|
|
|
| SizeOfPackExpr _ ->
|
|
|
|
{empty_res_trans with exps= [(Exp.get_undefined false, Typ.mk Tvoid)]}
|
|
|
|
| GCCAsmStmt (stmt_info, stmts) ->
|
|
|
|
gccAsmStmt_trans trans_state stmt_info stmts
|
|
|
|
| CXXPseudoDestructorExpr _ ->
|
|
|
|
cxxPseudoDestructorExpr_trans ()
|
|
|
|
| CXXTypeidExpr (stmt_info, stmts, expr_info) ->
|
|
|
|
cxxTypeidExpr_trans trans_state stmt_info stmts expr_info
|
|
|
|
| CXXStdInitializerListExpr (stmt_info, stmts, expr_info) ->
|
|
|
|
cxxStdInitializerListExpr_trans trans_state stmt_info stmts expr_info
|
|
|
|
| LambdaExpr (stmt_info, _, expr_info, lambda_expr_info) ->
|
|
|
|
let trans_state' = {trans_state with priority= Free} in
|
|
|
|
lambdaExpr_trans trans_state' stmt_info expr_info lambda_expr_info
|
|
|
|
| AttributedStmt (_, stmts, attrs) ->
|
|
|
|
attributedStmt_trans trans_state stmts attrs
|
|
|
|
| TypeTraitExpr (_, _, expr_info, type_trait_info) ->
|
|
|
|
booleanValue_trans trans_state expr_info type_trait_info.Clang_ast_t.xtti_value
|
|
|
|
| CXXNoexceptExpr (_, _, expr_info, cxx_noexcept_expr_info) ->
|
|
|
|
booleanValue_trans trans_state expr_info cxx_noexcept_expr_info.Clang_ast_t.xnee_value
|
|
|
|
| OffsetOfExpr (_, _, expr_info) | VAArgExpr (_, _, expr_info) ->
|
|
|
|
trans_into_undefined_expr trans_state expr_info
|
|
|
|
| ArrayInitIndexExpr _ | ArrayInitLoopExpr _ ->
|
|
|
|
no_op_trans trans_state.succ_nodes
|
|
|
|
(* vector instructions for OpenCL etc. we basically ignore these for now; just translate the
|
|
|
|
sub-expressions *)
|
|
|
|
| ObjCAvailabilityCheckExpr (_, _, expr_info, _) ->
|
|
|
|
trans_into_undefined_expr trans_state expr_info
|
|
|
|
| ExtVectorElementExpr (_, stmts, _)
|
|
|
|
| ShuffleVectorExpr (_, stmts, _)
|
|
|
|
| UserDefinedLiteral (_, stmts, _) ->
|
|
|
|
skip_unimplemented trans_state stmts
|
|
|
|
(* Infer somehow ended up in templated non instantiated code - right now
|
|
|
|
it's not supported and failure in those cases is expected. *)
|
|
|
|
| SubstNonTypeTemplateParmExpr _
|
|
|
|
| SubstNonTypeTemplateParmPackExpr _
|
|
|
|
| CXXDependentScopeMemberExpr _ ->
|
|
|
|
CFrontend_config.unimplemented "Translation of templated code is unsupported: %a"
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_stmt)
|
|
|
|
instr
|
|
|
|
| ForStmt (_, _) | WhileStmt (_, _) | DoStmt (_, _) | ObjCForCollectionStmt (_, _) ->
|
|
|
|
CFrontend_config.incorrect_assumption "Unexpected shape for %a: %a"
|
|
|
|
(Pp.to_string ~f:Clang_ast_proj.get_stmt_kind_string)
|
|
|
|
instr
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_stmt)
|
|
|
|
instr
|
|
|
|
| MSAsmStmt _
|
|
|
|
| CapturedStmt _
|
|
|
|
| CoreturnStmt _
|
|
|
|
| CoroutineBodyStmt _
|
|
|
|
| AddrLabelExpr _
|
|
|
|
| ArrayTypeTraitExpr _
|
|
|
|
| AsTypeExpr _
|
|
|
|
| AtomicExpr _
|
|
|
|
| CXXFoldExpr _
|
|
|
|
| CXXInheritedCtorInitExpr _
|
|
|
|
| CXXUnresolvedConstructExpr _
|
|
|
|
| CXXUuidofExpr _
|
|
|
|
| CUDAKernelCallExpr _
|
|
|
|
| ChooseExpr _
|
|
|
|
| ConvertVectorExpr _
|
|
|
|
| CoawaitExpr _
|
|
|
|
| CoyieldExpr _
|
|
|
|
| DependentCoawaitExpr _
|
|
|
|
| DependentScopeDeclRefExpr _
|
|
|
|
| DesignatedInitExpr _
|
|
|
|
| DesignatedInitUpdateExpr _
|
|
|
|
| ExpressionTraitExpr _
|
|
|
|
| FunctionParmPackExpr _
|
|
|
|
| ImaginaryLiteral _
|
|
|
|
| MSPropertyRefExpr _
|
|
|
|
| MSPropertySubscriptExpr _
|
|
|
|
| NoInitExpr _
|
|
|
|
| OMPArraySectionExpr _
|
|
|
|
| ObjCIsaExpr _
|
|
|
|
| ObjCSubscriptRefExpr _
|
|
|
|
| UnresolvedLookupExpr _
|
|
|
|
| UnresolvedMemberExpr _
|
|
|
|
| PackExpansionExpr _
|
|
|
|
| ParenListExpr _
|
|
|
|
| TypoExpr _
|
|
|
|
| IndirectGotoStmt _
|
|
|
|
| MSDependentExistsStmt _
|
|
|
|
| OMPAtomicDirective _
|
|
|
|
| OMPBarrierDirective _
|
|
|
|
| OMPCancelDirective _
|
|
|
|
| OMPCancellationPointDirective _
|
|
|
|
| OMPCriticalDirective _
|
|
|
|
| OMPFlushDirective _
|
|
|
|
| OMPDistributeDirective _
|
|
|
|
| OMPDistributeParallelForDirective _
|
|
|
|
| OMPDistributeParallelForSimdDirective _
|
|
|
|
| OMPDistributeSimdDirective _
|
|
|
|
| OMPForDirective _
|
|
|
|
| OMPForSimdDirective _
|
|
|
|
| OMPParallelForDirective _
|
|
|
|
| OMPParallelForSimdDirective _
|
|
|
|
| OMPSimdDirective _
|
|
|
|
| OMPTargetParallelForSimdDirective _
|
|
|
|
| OMPTargetSimdDirective _
|
|
|
|
| OMPTargetTeamsDistributeDirective _
|
|
|
|
| OMPTargetTeamsDistributeParallelForDirective _
|
|
|
|
| OMPTargetTeamsDistributeParallelForSimdDirective _
|
|
|
|
| OMPTargetTeamsDistributeSimdDirective _
|
|
|
|
| OMPTaskLoopDirective _
|
|
|
|
| OMPTaskLoopSimdDirective _
|
|
|
|
| OMPTeamsDistributeDirective _
|
|
|
|
| OMPTeamsDistributeParallelForDirective _
|
|
|
|
| OMPTeamsDistributeParallelForSimdDirective _
|
|
|
|
| OMPTeamsDistributeSimdDirective _
|
|
|
|
| OMPMasterDirective _
|
|
|
|
| OMPOrderedDirective _
|
|
|
|
| OMPParallelDirective _
|
|
|
|
| OMPParallelSectionsDirective _
|
|
|
|
| OMPSectionDirective _
|
|
|
|
| OMPSectionsDirective _
|
|
|
|
| OMPSingleDirective _
|
|
|
|
| OMPTargetDataDirective _
|
|
|
|
| OMPTargetDirective _
|
|
|
|
| OMPTargetEnterDataDirective _
|
|
|
|
| OMPTargetExitDataDirective _
|
|
|
|
| OMPTargetParallelDirective _
|
|
|
|
| OMPTargetParallelForDirective _
|
|
|
|
| OMPTargetTeamsDirective _
|
|
|
|
| OMPTargetUpdateDirective _
|
|
|
|
| OMPTaskDirective _
|
|
|
|
| OMPTaskgroupDirective _
|
|
|
|
| OMPTaskwaitDirective _
|
|
|
|
| OMPTaskyieldDirective _
|
|
|
|
| OMPTeamsDirective _
|
|
|
|
| SEHExceptStmt _
|
|
|
|
| SEHFinallyStmt _
|
|
|
|
| SEHLeaveStmt _
|
|
|
|
| SEHTryStmt _
|
|
|
|
| DefaultStmt _ ->
|
|
|
|
CFrontend_config.unimplemented "Statement translation for kind %s: %a"
|
|
|
|
(Clang_ast_proj.get_stmt_kind_string instr)
|
|
|
|
(Pp.to_string ~f:Clang_ast_j.string_of_stmt)
|
|
|
|
instr
|
|
|
|
|
|
|
|
|
|
|
|
(* Function similar to instruction function, but it takes C++ constructor initializer as
|
|
|
|
an input parameter. *)
|
|
|
|
and cxx_constructor_init_trans ctor_init trans_state =
|
|
|
|
let context = trans_state.context in
|
|
|
|
let class_ptr = CContext.get_curr_class_decl_ptr context.CContext.curr_class in
|
|
|
|
let source_range = ctor_init.Clang_ast_t.xci_source_range in
|
|
|
|
let sil_loc =
|
|
|
|
CLocation.get_sil_location_from_range context.CContext.translation_unit_context source_range
|
|
|
|
true
|
|
|
|
in
|
|
|
|
(* its pointer will be used in PriorityNode *)
|
|
|
|
let this_stmt_info = Ast_expressions.dummy_stmt_info () in
|
|
|
|
(* this will be used to avoid creating node in init_expr_trans *)
|
|
|
|
let child_stmt_info =
|
|
|
|
{(Ast_expressions.dummy_stmt_info ()) with Clang_ast_t.si_source_range= source_range}
|
|
|
|
in
|
|
|
|
let trans_state' = PriorityNode.try_claim_priority_node trans_state this_stmt_info in
|
|
|
|
let class_qual_type =
|
|
|
|
Ast_expressions.create_pointer_qual_type (CAst_utils.qual_type_of_decl_ptr class_ptr)
|
|
|
|
in
|
|
|
|
let this_res_trans = this_expr_trans trans_state' sil_loc class_qual_type in
|
|
|
|
let var_res_trans =
|
|
|
|
match ctor_init.Clang_ast_t.xci_subject with
|
|
|
|
| `Delegating _ | `BaseClass _ ->
|
|
|
|
let this_exp, this_typ =
|
|
|
|
extract_exp_from_list this_res_trans.exps
|
|
|
|
"WARNING: There should be one expression for 'this' in constructor. @\n"
|
|
|
|
in
|
|
|
|
(* Hack: Strip pointer from type here since cxxConstructExpr_trans expects it this way *)
|
|
|
|
(* it will add pointer back before making it a parameter to a call *)
|
|
|
|
let class_typ = match this_typ.Typ.desc with Tptr (t, _) -> t | _ -> assert false in
|
|
|
|
{this_res_trans with exps= [(this_exp, class_typ)]}
|
|
|
|
| `Member decl_ref ->
|
|
|
|
decl_ref_trans trans_state' this_res_trans child_stmt_info decl_ref
|
|
|
|
~is_constructor_init:true
|
|
|
|
in
|
|
|
|
let var_exp_typ =
|
|
|
|
extract_exp_from_list var_res_trans.exps
|
|
|
|
"WARNING: There should be one expression to initialize in constructor initializer. @\n"
|
|
|
|
in
|
|
|
|
let init_expr = ctor_init.Clang_ast_t.xci_init_expr in
|
|
|
|
let init_res_trans = init_expr_trans trans_state' var_exp_typ child_stmt_info init_expr in
|
|
|
|
PriorityNode.compute_results_to_parent trans_state' sil_loc "Constructor Init" this_stmt_info
|
|
|
|
[var_res_trans; init_res_trans]
|
|
|
|
|
|
|
|
|
|
|
|
(** Given a translation state and list of translation functions it executes translation *)
|
|
|
|
and exec_trans_instrs trans_state trans_stmt_fun_list =
|
|
|
|
let rec exec_trans_instrs_no_rev trans_state rev_trans_fun_list =
|
|
|
|
match rev_trans_fun_list with
|
|
|
|
| [] ->
|
|
|
|
{empty_res_trans with root_nodes= trans_state.succ_nodes}
|
|
|
|
| trans_stmt_fun :: trans_stmt_fun_list' ->
|
|
|
|
let res_trans_s = trans_stmt_fun trans_state in
|
|
|
|
let trans_state' =
|
|
|
|
if res_trans_s.root_nodes <> [] then
|
|
|
|
{trans_state with succ_nodes= res_trans_s.root_nodes}
|
|
|
|
else trans_state
|
|
|
|
in
|
|
|
|
let res_trans_tail = exec_trans_instrs_no_rev trans_state' trans_stmt_fun_list' in
|
|
|
|
{ empty_res_trans with
|
|
|
|
root_nodes= res_trans_tail.root_nodes
|
|
|
|
; leaf_nodes= res_trans_s.leaf_nodes
|
|
|
|
; instrs= res_trans_tail.instrs @ res_trans_s.instrs
|
|
|
|
; exps= res_trans_tail.exps @ res_trans_s.exps
|
|
|
|
; initd_exps= res_trans_tail.initd_exps @ res_trans_s.initd_exps }
|
|
|
|
in
|
|
|
|
exec_trans_instrs_no_rev trans_state (List.rev trans_stmt_fun_list)
|
|
|
|
|
|
|
|
|
|
|
|
and get_clang_stmt_trans stmt trans_state = exec_with_node_creation instruction trans_state stmt
|
|
|
|
|
|
|
|
(* TODO write translate function for cxx constructor exprs *)
|
|
|
|
and get_custom_stmt_trans stmt =
|
|
|
|
match stmt with
|
|
|
|
| `ClangStmt stmt ->
|
|
|
|
get_clang_stmt_trans stmt
|
|
|
|
| `CXXConstructorInit instr ->
|
|
|
|
cxx_constructor_init_trans instr
|
|
|
|
|
|
|
|
|
|
|
|
(** Given a translation state, this function translates a list of clang statements. *)
|
|
|
|
and instructions trans_state stmt_list =
|
|
|
|
let stmt_trans_fun = List.map ~f:get_clang_stmt_trans stmt_list in
|
|
|
|
exec_trans_instrs trans_state stmt_trans_fun
|
|
|
|
|
|
|
|
|
|
|
|
and expression_trans context stmt warning =
|
|
|
|
let trans_state =
|
|
|
|
{ context
|
|
|
|
; succ_nodes= []
|
|
|
|
; continuation= None
|
|
|
|
; priority= Free
|
|
|
|
; var_exp_typ= None
|
|
|
|
; opaque_exp= None
|
|
|
|
; obj_bridged_cast_typ= None }
|
|
|
|
in
|
|
|
|
let res_trans_stmt = instruction trans_state stmt in
|
|
|
|
fst (CTrans_utils.extract_exp_from_list res_trans_stmt.exps warning)
|
|
|
|
|
|
|
|
|
|
|
|
let instructions_trans context body extra_instrs exit_node ~is_destructor_wrapper =
|
|
|
|
let trans_state =
|
|
|
|
{ context
|
|
|
|
; succ_nodes= [exit_node]
|
|
|
|
; continuation= None
|
|
|
|
; priority= Free
|
|
|
|
; var_exp_typ= None
|
|
|
|
; opaque_exp= None
|
|
|
|
; obj_bridged_cast_typ= None }
|
|
|
|
in
|
|
|
|
let procname = Procdesc.get_proc_name context.CContext.procdesc in
|
|
|
|
let is_destructor = Typ.Procname.is_destructor procname in
|
|
|
|
let stmt_info, _ = Clang_ast_proj.get_stmt_tuple body in
|
|
|
|
let destructor_res, body =
|
|
|
|
if is_destructor_wrapper then
|
|
|
|
let stmt_info' = {stmt_info with si_pointer= CAst_utils.get_fresh_pointer ()} in
|
|
|
|
( cxx_inject_virtual_base_class_destructors trans_state stmt_info'
|
|
|
|
(* destructor wrapper only have calls to virtual base class destructors in its body *)
|
|
|
|
, Clang_ast_t.CompoundStmt (stmt_info', []) )
|
|
|
|
else if is_destructor then
|
|
|
|
(cxx_inject_field_destructors_in_destructor_body trans_state stmt_info, body)
|
|
|
|
else (empty_res_trans, body)
|
|
|
|
in
|
|
|
|
(* Injecting destructor call nodes of fields at the end of the body *)
|
|
|
|
let succ_nodes =
|
|
|
|
if destructor_res.root_nodes <> [] then destructor_res.root_nodes else trans_state.succ_nodes
|
|
|
|
in
|
|
|
|
let trans_state' = {trans_state with succ_nodes} in
|
|
|
|
let instrs = extra_instrs @ [`ClangStmt body] in
|
|
|
|
let instrs_trans = List.map ~f:get_custom_stmt_trans instrs in
|
|
|
|
let res_trans = exec_trans_instrs trans_state' instrs_trans in
|
|
|
|
res_trans.root_nodes
|
|
|
|
|
|
|
|
end
|