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.

103 lines
4.6 KiB

(*
* 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
(** Utility module to retrieve fields of structs of classes *)
module L = Logging
type field_type = Typ.Fieldname.t * Typ.t * (Annot.t * bool) list
let rec get_fields_super_classes tenv super_class =
L.(debug Capture Verbose) " ... Getting fields of superclass '%s'@\n"
(Typ.Name.to_string super_class);
match Tenv.lookup tenv super_class with
| None -> []
| Some { fields; supers = super_class :: _ } ->
let sc_fields = get_fields_super_classes tenv super_class in
CGeneral_utils.append_no_duplicates_fields fields sc_fields
| Some { fields } -> fields
let fields_superclass tenv interface_decl_info =
match interface_decl_info.Clang_ast_t.otdi_super with
| Some dr ->
(match dr.Clang_ast_t.dr_name with
| Some sc ->
let classname = Typ.Name.Objc.from_qual_name (CAst_utils.get_qualified_name sc) in
get_fields_super_classes tenv classname
| _ -> [])
| _ -> []
let build_sil_field qual_type_to_sil_type tenv class_tname field_name qual_type prop_attributes =
let prop_atts = List.map ~f:Clang_ast_j.string_of_property_attribute prop_attributes in
let annotation_from_type t =
match t.Typ.desc with
| Typ.Tptr (_, Typ.Pk_objc_weak) -> [Config.weak]
| Typ.Tptr (_, Typ.Pk_objc_unsafe_unretained) -> [Config.unsafe_unret]
| _ -> [] in
let fname = CGeneral_utils.mk_class_field_name class_tname field_name.Clang_ast_t.ni_name in
let typ = qual_type_to_sil_type tenv qual_type in
let item_annotations = match prop_atts with
| [] ->
({ Annot.class_name = Config.ivar_attributes; parameters = annotation_from_type typ },
true)
| _ ->
({ Annot.class_name = Config.property_attributes; parameters = prop_atts },
true) in
let item_annotations = item_annotations :: (CAst_utils.sil_annot_of_type qual_type) in
fname, typ, item_annotations
(* Given a list of declarations in an interface returns a list of fields *)
let rec get_fields qual_type_to_sil_type tenv class_tname decl_list =
let open Clang_ast_t in
let add_field name_info (qt : qual_type) attributes decl_list' =
let fields = get_fields qual_type_to_sil_type tenv class_tname decl_list' in
let field_tuple = build_sil_field qual_type_to_sil_type tenv class_tname
name_info qt attributes in
CGeneral_utils.append_no_duplicates_fields [field_tuple] fields in
match decl_list with
| [] -> []
| ObjCPropertyDecl (_, _, obj_c_property_decl_info) :: decl_list' ->
(let ivar_decl_ref = obj_c_property_decl_info.Clang_ast_t.opdi_ivar_decl in
match CAst_utils.get_decl_opt_with_decl_ref ivar_decl_ref with
| Some (ObjCIvarDecl (_, name_info, qual_type, _, _)) ->
let attributes = obj_c_property_decl_info.Clang_ast_t.opdi_property_attributes in
add_field name_info qual_type attributes decl_list'
| _ -> get_fields qual_type_to_sil_type tenv class_tname decl_list')
| ObjCIvarDecl (_, name_info, qual_type, _, _) :: decl_list' ->
add_field name_info qual_type [] decl_list'
| _ :: decl_list' ->
get_fields qual_type_to_sil_type tenv class_tname decl_list'
(* Add potential extra fields defined only in the implementation of the class *)
(* to the info given in the interface. Update the tenv accordingly. *)
let add_missing_fields tenv class_name missing_fields =
let class_tn_name = Typ.Name.Objc.from_qual_name class_name in
match Tenv.lookup tenv class_tn_name with
| Some ({ fields } as struct_typ) ->
let new_fields = CGeneral_utils.append_no_duplicates_fields fields missing_fields in
ignore (Tenv.mk_struct tenv ~default:struct_typ ~fields:new_fields ~statics:[] class_tn_name);
L.(debug Capture Verbose) " Updating info for class '%a' in tenv@\n"
QualifiedCppName.pp class_name
| _ -> ()
let modelled_fields_in_classes =
[("NSData", "_bytes", Typ.mk (Tptr (Typ.mk Tvoid, Typ.Pk_pointer)));
("NSArray", "elementData", Typ.mk (Tint Typ.IInt))]
let modelled_field class_name_info =
let modelled_field_in_class res (class_name, field_name, typ) =
if String.equal class_name class_name_info.Clang_ast_t.ni_name then
let class_tname = Typ.Name.Objc.from_string class_name in
let name = Typ.Fieldname.Clang.from_class_name class_tname field_name in
(name, typ, Annot.Item.empty) :: res
else res in
List.fold ~f:modelled_field_in_class ~init:[] modelled_fields_in_classes