You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

166 lines
6.5 KiB

(*
* Copyright (c) 2013-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*)
open! IStd
(** In this module an ObjC interface declaration or implementation is processed. The class *)
(** is saved in the tenv as a struct with the corresponding fields, potential superclass and *)
(** list of defined methods *)
(* ObjectiveC doesn't have a notion of static or class fields. *)
(* So, in this module we translate a class into a sil srtuct with an empty list of static fields.*)
module L = Logging
let get_super_interface_decl otdi_super =
match otdi_super with
| Some dr ->
Option.map ~f:CAst_utils.get_qualified_name dr.Clang_ast_t.dr_name
| _ ->
None
let get_protocols protocols =
let protocol_names =
List.map
~f:(fun decl ->
match decl.Clang_ast_t.dr_name with
| Some name_info ->
CAst_utils.get_qualified_name name_info
| None ->
assert false )
protocols
in
protocol_names
let add_class_decl qual_type_to_sil_type tenv idi =
let decl_ref_opt = idi.Clang_ast_t.oidi_class_interface in
CAst_utils.add_type_from_decl_ref_opt qual_type_to_sil_type tenv decl_ref_opt true
let add_super_class_decl qual_type_to_sil_type tenv ocdi =
let decl_ref_opt = ocdi.Clang_ast_t.otdi_super in
CAst_utils.add_type_from_decl_ref_opt qual_type_to_sil_type tenv decl_ref_opt false
let add_protocols_decl qual_type_to_sil_type tenv protocols =
CAst_utils.add_type_from_decl_ref_list qual_type_to_sil_type tenv protocols
let add_categories_decl qual_type_to_sil_type tenv categories =
CAst_utils.add_type_from_decl_ref_list qual_type_to_sil_type tenv categories
let add_class_implementation qual_type_to_sil_type tenv idi =
let decl_ref_opt = idi.Clang_ast_t.otdi_implementation in
CAst_utils.add_type_from_decl_ref_opt qual_type_to_sil_type tenv decl_ref_opt false
(*The superclass is the first element in the list of super classes of structs in the tenv, *)
(* then come the protocols and categories. *)
let get_interface_supers super_opt protocols =
let super_class =
match super_opt with None -> [] | Some super -> [Typ.Name.Objc.from_qual_name super]
in
let protocol_names = List.map ~f:Typ.Name.Objc.protocol_from_qual_name protocols in
let super_classes = super_class @ protocol_names in
super_classes
let create_supers_fields qual_type_to_sil_type tenv class_tname decl_list otdi_super otdi_protocols =
let super = get_super_interface_decl otdi_super in
let protocols = get_protocols otdi_protocols in
let supers = get_interface_supers super protocols in
let fields = CField_decl.get_fields qual_type_to_sil_type tenv class_tname decl_list in
(supers, fields)
let append_no_duplicates_typ_name =
Staged.unstage (IList.append_no_duplicates ~cmp:Typ.Name.compare)
(* Adds pairs (interface name, interface_type_info) to the global environment. *)
let add_class_to_tenv qual_type_to_sil_type procname_from_decl tenv decl_info name_info decl_list
ocidi =
let class_name = CAst_utils.get_qualified_name name_info in
let interface_name = Typ.Name.Objc.from_qual_name class_name in
let interface_desc = Typ.Tstruct interface_name in
let decl_key = Clang_ast_extend.DeclPtr decl_info.Clang_ast_t.di_pointer in
CAst_utils.update_sil_types_map decl_key interface_desc ;
(* We don't need to add the methods of the superclass *)
let decl_supers, decl_fields =
create_supers_fields qual_type_to_sil_type tenv interface_name decl_list
ocidi.Clang_ast_t.otdi_super ocidi.Clang_ast_t.otdi_protocols
in
let fields_sc = CField_decl.fields_superclass tenv ocidi in
(*In case we found categories, or partial definition of this class earlier and they are already in the tenv *)
let fields, (supers: Typ.Name.t list), methods =
match Tenv.lookup tenv interface_name with
| Some {fields; supers; methods} ->
( CGeneral_utils.append_no_duplicates_fields decl_fields fields
, append_no_duplicates_typ_name decl_supers supers
, methods )
| _ ->
(decl_fields, decl_supers, [])
in
let fields = CGeneral_utils.append_no_duplicates_fields fields fields_sc in
let modelled_fields = CField_decl.modelled_field name_info in
let all_fields = CGeneral_utils.append_no_duplicates_fields modelled_fields fields in
let methods =
CGeneral_utils.append_no_duplicates_methods
(ObjcMethod_decl.get_methods procname_from_decl tenv decl_list)
methods
in
ignore
(Tenv.mk_struct tenv ~fields:all_fields ~supers ~methods ~annots:Annot.Class.objc
interface_name) ;
interface_desc
(* Interface_type_info has the name of instance variables and the name of methods. *)
let interface_declaration qual_type_to_sil_type procname_from_decl tenv decl =
let open Clang_ast_t in
match decl with
| ObjCInterfaceDecl (decl_info, name_info, decl_list, _, ocidi) ->
let typ =
add_class_to_tenv qual_type_to_sil_type procname_from_decl tenv decl_info name_info
decl_list ocidi
in
let _ = add_class_implementation qual_type_to_sil_type tenv ocidi in
let _ = add_super_class_decl qual_type_to_sil_type tenv ocidi in
let _ = add_protocols_decl qual_type_to_sil_type tenv ocidi.Clang_ast_t.otdi_protocols in
let known_categories = ocidi.Clang_ast_t.otdi_known_categories in
let _ = add_categories_decl qual_type_to_sil_type tenv known_categories in
typ
| _ ->
assert false
(* Translate the methods defined in the implementation.*)
let interface_impl_declaration qual_type_to_sil_type procname_from_decl tenv decl =
let open Clang_ast_t in
match decl with
| ObjCImplementationDecl (decl_info, name_info, decl_list, _, idi) ->
let class_name = CAst_utils.get_qualified_name name_info in
L.(debug Capture Verbose)
"ADDING: ObjCImplementationDecl for class '%a'@\n" QualifiedCppName.pp class_name ;
let _ = add_class_decl qual_type_to_sil_type tenv idi in
let class_tn_name = Typ.Name.Objc.from_qual_name class_name in
let fields = CField_decl.get_fields qual_type_to_sil_type tenv class_tn_name decl_list in
CField_decl.add_missing_fields tenv class_name fields ;
let methods = ObjcMethod_decl.get_methods procname_from_decl tenv decl_list in
ObjcMethod_decl.add_missing_methods tenv class_tn_name methods ;
let decl_key = Clang_ast_extend.DeclPtr decl_info.Clang_ast_t.di_pointer in
let class_desc = Typ.Tstruct class_tn_name in
CAst_utils.update_sil_types_map decl_key class_desc ;
class_desc
| _ ->
assert false