(* * 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 (** General utility functions such as functions on lists *) module L = Logging module F = Format type var_info = Clang_ast_t.decl_info * Clang_ast_t.qual_type * Clang_ast_t.var_decl_info * bool let rec swap_elements_list l = match l with | el1:: el2:: rest -> el2:: el1:: (swap_elements_list rest) | [] -> [] | _ -> assert false let rec string_from_list l = match l with | [] -> "" | [item] -> item | item:: l' -> item^" "^(string_from_list l') let rec append_no_duplicates eq list1 list2 = match list2 with | el:: rest2 -> if (List.mem ~equal:eq list1 el) then (append_no_duplicates eq list1 rest2) else (append_no_duplicates eq list1 rest2)@[el] | [] -> list1 let append_no_duplicates_csu list1 list2 = append_no_duplicates Typename.equal list1 list2 let append_no_duplicates_methods list1 list2 = append_no_duplicates Procname.equal list1 list2 let append_no_duplicates_annotations list1 list2 = let eq (annot1, _) (annot2, _) = String.equal annot1.Annot.class_name annot2.Annot.class_name in append_no_duplicates eq list1 list2 let add_no_duplicates_fields field_tuple l = let rec replace_field field_tuple l found = match field_tuple, l with | (field, typ, annot), ((old_field, old_typ, old_annot) as old_field_tuple :: rest) -> let ret_list, ret_found = replace_field field_tuple rest found in if Ident.equal_fieldname field old_field && Typ.equal typ old_typ then let annotations = append_no_duplicates_annotations annot old_annot in (field, typ, annotations) :: ret_list, true else old_field_tuple :: ret_list, ret_found | _, [] -> [], found in let new_list, found = replace_field field_tuple l false in if found then new_list else field_tuple :: l let rec append_no_duplicates_fields list1 list2 = match list1 with | field_tuple :: rest -> let updated_list2 = append_no_duplicates_fields rest list2 in add_no_duplicates_fields field_tuple updated_list2 | [] -> list2 let sort_fields fields = let compare (name1, _, _) (name2, _, _) = Ident.compare_fieldname name1 name2 in List.sort ~cmp:compare fields let sort_fields_tenv tenv = let sort_fields_struct name ({StructTyp.fields} as st) = ignore (Tenv.mk_struct tenv ~default:st ~fields:(sort_fields fields) name) in Tenv.iter sort_fields_struct tenv let rec collect_list_tuples l (a, a1, b, c, d) = match l with | [] -> (a, a1, b, c, d) | (a', a1', b', c', d'):: l' -> collect_list_tuples l' (a@a', a1@a1', b@b', c@c', d@d') let is_static_var var_decl_info = match var_decl_info.Clang_ast_t.vdi_storage_class with | Some sc -> String.equal sc CFrontend_config.static | _ -> false let block_procname_with_index defining_proc i = Config.anonymous_block_prefix ^ (Procname.to_string defining_proc) ^ Config.anonymous_block_num_sep ^ (string_of_int i) (* Global counter for anonymous block*) let block_counter = ref 0 (* Returns a fresh index for a new anonymous block *) let get_fresh_block_index () = block_counter := !block_counter +1; !block_counter (* Makes a fresh name for a block defined inside the defining procedure.*) (* It updates the global block_counter *) let mk_fresh_block_procname defining_proc = let name = block_procname_with_index defining_proc (get_fresh_block_index ()) in Procname.mangled_objc_block name (* Returns the next fresh name for a block defined inside the defining procedure *) (* It does not update the global block_counter *) 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 (* Reset block counter *) let reset_block_counter () = block_counter := 0 let rec zip xs ys = match xs, ys with | [], _ | _, [] -> [] | x :: xs, y :: ys -> (x, y) :: zip xs ys let list_range i j = let rec aux n acc = if n < i then acc else aux (n -1) (n :: acc) in aux j [] ;; let replicate n el = List.map ~f:(fun _ -> el) (list_range 0 (n -1)) let mk_class_field_name field_qual_name = let field_name = field_qual_name.Clang_ast_t.ni_name in let class_name = CAst_utils.get_class_name_from_member field_qual_name in Ident.create_fieldname (Mangled.mangled field_name class_name) 0 let is_cpp_translation translation_unit_context = let lang = translation_unit_context.CFrontend_config.lang in CFrontend_config.equal_clang_lang lang CFrontend_config.CPP || CFrontend_config.equal_clang_lang lang CFrontend_config.ObjCPP let is_objc_extension translation_unit_context = let lang = translation_unit_context.CFrontend_config.lang in CFrontend_config.equal_clang_lang lang CFrontend_config.ObjC || CFrontend_config.equal_clang_lang lang CFrontend_config.ObjCPP 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 mk_procname_from_function translation_unit_context 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 in Option.value_map ~f:SourceFile.to_string ~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 is_cpp_translation translation_unit_context -> m | _ -> "" in let mangled = (Utils.string_crc_hex32 file) ^ mangled_name in if String.is_empty file && String.is_empty mangled_name then Procname.from_string_c_fun name else Procname.C (Procname.c name mangled) let mk_procname_from_objc_method class_name method_name method_kind = Procname.ObjC_Cpp (Procname.objc_cpp class_name method_name method_kind) let mk_procname_from_cpp_method class_name method_name ?meth_decl mangled = let method_kind = match meth_decl with | Some (Clang_ast_t.CXXConstructorDecl (_, _, _, _, {xmdi_is_constexpr})) -> Procname.CPPConstructor (mangled, xmdi_is_constexpr) | _ -> Procname.CPPMethod mangled in Procname.ObjC_Cpp (Procname.objc_cpp class_name method_name method_kind) let get_objc_method_name name_info mdi class_name = 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 = Procname.objc_method_kind_of_bool is_instance in mk_procname_from_objc_method class_name method_name method_kind let procname_of_decl translation_unit_context 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_procname_from_function translation_unit_context name function_info | CXXMethodDecl (_, name_info, _, fdi, mdi) | CXXConstructorDecl (_, name_info, _, fdi, mdi) | CXXConversionDecl (_, name_info, _, fdi, mdi) | CXXDestructorDecl (_, 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_name = CAst_utils.get_class_name_from_member name_info in mk_procname_from_cpp_method class_name method_name ~meth_decl mangled | ObjCMethodDecl (_, name_info, mdi) -> let class_name = CAst_utils.get_class_name_from_member name_info in get_objc_method_name name_info mdi class_name | BlockDecl _ -> let name = Config.anonymous_block_prefix ^ Config.anonymous_block_num_sep ^ (string_of_int (get_fresh_block_index ())) in Procname.mangled_objc_block name | _ -> assert false let get_var_name_mangled name_info var_decl_info = let clang_name = CAst_utils.get_qualified_name name_info in let param_idx_opt = var_decl_info.Clang_ast_t.vdi_parm_index_in_function in let name_string = match clang_name, param_idx_opt with | "", Some index -> "__param_" ^ string_of_int index | "", None -> assert false | _ -> clang_name in let mangled = match param_idx_opt with | Some index -> Mangled.mangled name_string (string_of_int index) | None -> Mangled.from_string name_string in name_string, mangled let mk_sil_global_var {CFrontend_config.source_file} ?(mk_name=fun _ x -> x) named_decl_info var_decl_info qt = let name_string, simple_name = get_var_name_mangled named_decl_info var_decl_info in let translation_unit = match (var_decl_info.Clang_ast_t.vdi_storage_class, var_decl_info.Clang_ast_t.vdi_init_expr) with | Some "extern", None -> (* some compilers simply disregard "extern" when the global is given some initialisation code, which is why we make sure that [vdi_init_expr] is None here... *) SourceFile.empty | _ -> source_file in let is_constexpr = var_decl_info.Clang_ast_t.vdi_is_const_expr in let is_pod = CAst_utils.get_desugared_type qt.Clang_ast_t.qt_type_ptr |> Fn.flip Option.bind (function | Clang_ast_t.RecordType(_, decl_ptr) -> CAst_utils.get_decl decl_ptr | _ -> None) |> Option.value_map ~default:true ~f:(function | Clang_ast_t.CXXRecordDecl(_, _, _, _, _, _, _, {xrdi_is_pod}) | Clang_ast_t.ClassTemplateSpecializationDecl(_, _, _, _, _, _, _, {xrdi_is_pod}, _) -> xrdi_is_pod | _ -> true) in Pvar.mk_global ~is_constexpr ~is_pod ~is_static_local:(var_decl_info.Clang_ast_t.vdi_is_static_local) (mk_name name_string simple_name) translation_unit let mk_sil_var trans_unit_ctx named_decl_info decl_info_type_ptr_opt procname outer_procname = match decl_info_type_ptr_opt with | Some (decl_info, qt, var_decl_info, should_be_mangled) -> let name_string, simple_name = get_var_name_mangled named_decl_info var_decl_info in if var_decl_info.Clang_ast_t.vdi_is_global then let mk_name = if var_decl_info.Clang_ast_t.vdi_is_static_local then Some (fun name_string _ -> Mangled.from_string ((Procname.to_string outer_procname) ^ "_" ^ name_string)) else None in mk_sil_global_var trans_unit_ctx ?mk_name named_decl_info var_decl_info qt else if not should_be_mangled then Pvar.mk simple_name procname else let start_location = fst decl_info.Clang_ast_t.di_source_range in let line_opt = start_location.Clang_ast_t.sl_line in let line_str = match line_opt with | Some line -> string_of_int line | None -> "" in let mangled = Utils.string_crc_hex32 line_str in let mangled_name = Mangled.mangled name_string mangled in Pvar.mk mangled_name procname | None -> let name_string = CAst_utils.get_qualified_name named_decl_info in Pvar.mk (Mangled.from_string name_string) procname