|
|
|
(*
|
|
|
|
* Copyright (c) 2017 - 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
|
|
|
|
|
|
|
|
let rec get_mangled_method_name function_decl_info method_decl_info =
|
|
|
|
(* For virtual methods return mangled name of the method from most base class
|
|
|
|
Go recursively until there is no method in any parent class. All names
|
|
|
|
of the same method need to be the same, otherwise dynamic dispatch won't
|
|
|
|
work. *)
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match method_decl_info.xmdi_overriden_methods with
|
|
|
|
| [] -> function_decl_info.fdi_mangled_name
|
|
|
|
| base1_dr :: _ ->
|
|
|
|
(let base1 = match CAst_utils.get_decl base1_dr.dr_decl_pointer with
|
|
|
|
| Some b -> b
|
|
|
|
| _ -> assert false in
|
|
|
|
match base1 with
|
|
|
|
| CXXMethodDecl (_, _, _, fdi, mdi)
|
|
|
|
| CXXConstructorDecl (_, _, _, fdi, mdi)
|
|
|
|
| CXXConversionDecl (_, _, _, fdi, mdi)
|
|
|
|
| CXXDestructorDecl (_, _, _, fdi, mdi) ->
|
|
|
|
get_mangled_method_name fdi mdi
|
|
|
|
| _ -> assert false)
|
|
|
|
|
|
|
|
let get_template_info tenv (fdi : Clang_ast_t.function_decl_info) : Typ.template_spec_info =
|
|
|
|
match fdi.fdi_template_specialization with
|
|
|
|
| Some spec_info -> Typ.Template (
|
|
|
|
List.map spec_info.tsi_specialization_args ~f:(function
|
|
|
|
| `Type qual_type -> Some (CType_decl.qual_type_to_sil_type tenv qual_type)
|
|
|
|
| _ -> None))
|
|
|
|
| None -> Typ.NoTemplate
|
|
|
|
|
|
|
|
let is_decl_info_generic_model {Clang_ast_t.di_attributes} =
|
|
|
|
let f = function
|
|
|
|
| Clang_ast_t.AnnotateAttr {ai_parameters=[_; name; _]}
|
|
|
|
when String.equal name "__infer_generic_model" -> true
|
|
|
|
| _ -> false in
|
|
|
|
List.exists ~f di_attributes
|
|
|
|
|
|
|
|
let mk_c_function translation_unit_context ?tenv name function_decl_info_opt =
|
|
|
|
let file =
|
|
|
|
match function_decl_info_opt with
|
|
|
|
| Some (decl_info, function_decl_info) ->
|
|
|
|
(match function_decl_info.Clang_ast_t.fdi_storage_class with
|
|
|
|
| Some "static" ->
|
|
|
|
let file_opt = (fst decl_info.Clang_ast_t.di_source_range).Clang_ast_t.sl_file |> Option.map ~f:SourceFile.from_abs_path in
|
|
|
|
let file_to_hex src = SourceFile.to_string src |> Utils.string_crc_hex32 in
|
|
|
|
Option.value_map ~f:file_to_hex ~default:"" file_opt
|
|
|
|
| _ -> "")
|
|
|
|
| None -> "" in
|
|
|
|
let mangled_opt = match function_decl_info_opt with
|
|
|
|
| Some (_, function_decl_info) -> function_decl_info.Clang_ast_t.fdi_mangled_name
|
|
|
|
| _ -> None in
|
|
|
|
let mangled_name = match mangled_opt with
|
|
|
|
| Some m when CGeneral_utils.is_cpp_translation translation_unit_context -> m
|
|
|
|
| _ -> "" in
|
|
|
|
let template_info, is_generic_model = match function_decl_info_opt, tenv with
|
|
|
|
| Some (decl_info, function_decl_info), Some t ->
|
|
|
|
get_template_info t function_decl_info, is_decl_info_generic_model decl_info
|
|
|
|
| _ -> Typ.NoTemplate, false in
|
|
|
|
let mangled = file ^ mangled_name in
|
|
|
|
if String.is_empty mangled then
|
|
|
|
Typ.Procname.from_string_c_fun (QualifiedCppName.to_qual_string name)
|
|
|
|
else
|
|
|
|
Typ.Procname.C (Typ.Procname.c name mangled template_info ~is_generic_model)
|
|
|
|
|
|
|
|
let mk_cpp_method ?tenv class_name method_name ?meth_decl mangled =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
let method_kind = match meth_decl with
|
|
|
|
| Some (Clang_ast_t.CXXConstructorDecl (_, _, _, _, {xmdi_is_constexpr})) ->
|
|
|
|
Typ.Procname.CPPConstructor (mangled, xmdi_is_constexpr)
|
|
|
|
| _ ->
|
|
|
|
Typ.Procname.CPPMethod mangled in
|
|
|
|
let template_info, is_generic_model = match meth_decl with
|
|
|
|
| Some (CXXMethodDecl (di, _, _, fdi, _))
|
|
|
|
| Some (CXXConstructorDecl (di, _, _, fdi, _))
|
|
|
|
| Some (CXXConversionDecl (di, _, _, fdi, _))
|
|
|
|
| Some (CXXDestructorDecl (di, _, _, fdi, _)) -> (
|
|
|
|
let templ_info = match tenv with
|
|
|
|
| Some t -> get_template_info t fdi
|
|
|
|
| None -> Typ.NoTemplate in
|
|
|
|
let is_gen_model = is_decl_info_generic_model di ||
|
|
|
|
(* read whether parent class is annoatated as generic model *)
|
|
|
|
di.di_parent_pointer
|
|
|
|
|> Option.value_map ~f:CAst_utils.get_decl ~default:None
|
|
|
|
|> Option.map ~f:Clang_ast_proj.get_decl_tuple
|
|
|
|
|> Option.value_map ~f:is_decl_info_generic_model ~default:false in
|
|
|
|
templ_info, is_gen_model
|
|
|
|
)
|
|
|
|
| _ -> Typ.NoTemplate, false in
|
|
|
|
Typ.Procname.ObjC_Cpp
|
|
|
|
(Typ.Procname.objc_cpp class_name method_name method_kind template_info ~is_generic_model)
|
|
|
|
|
|
|
|
let mk_objc_method class_typename method_name method_kind =
|
|
|
|
Typ.Procname.ObjC_Cpp
|
|
|
|
(Typ.Procname.objc_cpp class_typename method_name method_kind Typ.NoTemplate
|
|
|
|
~is_generic_model:false)
|
|
|
|
|
|
|
|
let block_procname_with_index defining_proc i =
|
|
|
|
Config.anonymous_block_prefix ^
|
|
|
|
(Typ.Procname.to_string defining_proc) ^
|
|
|
|
Config.anonymous_block_num_sep ^
|
|
|
|
(string_of_int i)
|
|
|
|
|
|
|
|
(* Global counter for anonymous block*)
|
|
|
|
let block_counter = ref 0
|
|
|
|
|
|
|
|
let get_next_block_pvar defining_proc =
|
|
|
|
let name = block_procname_with_index defining_proc (!block_counter +1) in
|
|
|
|
Pvar.mk_tmp name defining_proc
|
|
|
|
|
|
|
|
let reset_block_counter () =
|
|
|
|
block_counter := 0
|
|
|
|
|
|
|
|
let get_fresh_block_index () =
|
|
|
|
block_counter := !block_counter +1;
|
|
|
|
!block_counter
|
|
|
|
|
|
|
|
let mk_fresh_block_procname defining_proc =
|
|
|
|
let name = block_procname_with_index defining_proc (get_fresh_block_index ()) in
|
|
|
|
Typ.Procname.mangled_objc_block name
|
|
|
|
|
|
|
|
|
|
|
|
let get_class_typename ?tenv method_decl_info =
|
|
|
|
let class_ptr = Option.value_exn method_decl_info.Clang_ast_t.di_parent_pointer in
|
|
|
|
match CAst_utils.get_decl class_ptr with
|
|
|
|
| Some class_decl -> CType_decl.get_record_typename ?tenv class_decl
|
|
|
|
| None -> assert false
|
|
|
|
|
|
|
|
module NoAstDecl = struct
|
|
|
|
let c_function_of_string translation_unit_context tenv name =
|
|
|
|
let qual_name = QualifiedCppName.of_qual_string name in
|
|
|
|
mk_c_function translation_unit_context ~tenv qual_name None
|
|
|
|
|
|
|
|
let cpp_method_of_string tenv class_name method_name =
|
|
|
|
mk_cpp_method ~tenv class_name method_name None
|
|
|
|
|
|
|
|
let objc_method_of_string_kind class_name method_name method_kind =
|
|
|
|
mk_objc_method class_name method_name method_kind
|
|
|
|
end
|
|
|
|
|
|
|
|
let from_decl translation_unit_context ?tenv meth_decl =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match meth_decl with
|
|
|
|
| FunctionDecl (decl_info, name_info, _, fdi) ->
|
|
|
|
let name = CAst_utils.get_qualified_name name_info in
|
|
|
|
let function_info = Some (decl_info, fdi) in
|
|
|
|
mk_c_function translation_unit_context ?tenv name function_info
|
|
|
|
| CXXMethodDecl (decl_info, name_info, _, fdi, mdi)
|
|
|
|
| CXXConstructorDecl (decl_info, name_info, _, fdi, mdi)
|
|
|
|
| CXXConversionDecl (decl_info, name_info, _, fdi, mdi)
|
|
|
|
| CXXDestructorDecl (decl_info, name_info, _, fdi, mdi) ->
|
|
|
|
let mangled = get_mangled_method_name fdi mdi in
|
|
|
|
let method_name = CAst_utils.get_unqualified_name name_info in
|
|
|
|
let class_typename = get_class_typename ?tenv decl_info in
|
|
|
|
mk_cpp_method ?tenv class_typename method_name ~meth_decl mangled
|
|
|
|
| ObjCMethodDecl (decl_info, name_info, mdi) ->
|
|
|
|
let class_typename = get_class_typename ?tenv decl_info in
|
|
|
|
let method_name = name_info.Clang_ast_t.ni_name in
|
|
|
|
let is_instance = mdi.Clang_ast_t.omdi_is_instance_method in
|
|
|
|
let method_kind = Typ.Procname.objc_method_kind_of_bool is_instance in
|
|
|
|
mk_objc_method class_typename method_name method_kind
|
|
|
|
| BlockDecl _ ->
|
|
|
|
let name = Config.anonymous_block_prefix ^ Config.anonymous_block_num_sep ^
|
|
|
|
(string_of_int (get_fresh_block_index ())) in
|
|
|
|
Typ.Procname.mangled_objc_block name
|
|
|
|
| _ -> assert false
|