|
|
|
(*
|
|
|
|
* 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.
|
|
|
|
*)
|
|
|
|
|
|
|
|
(** Utility module to retrieve fields of structs of classes *)
|
|
|
|
|
|
|
|
open Utils
|
|
|
|
open CFrontend_utils
|
|
|
|
|
|
|
|
module L = Logging
|
|
|
|
|
|
|
|
type field_type = Ident.fieldname * Sil.typ * (Sil.annotation * bool) list
|
|
|
|
|
|
|
|
let rec get_fields_super_classes tenv super_class =
|
|
|
|
Printing.log_out " ... Getting fields of superclass '%s'\n" (Sil.typename_to_string super_class);
|
|
|
|
match Sil.tenv_lookup tenv super_class with
|
|
|
|
| None -> []
|
|
|
|
| Some Sil.Tstruct (fields, _, _, _, (Sil.Class, sc):: _, _, _) ->
|
|
|
|
let sc_fields = get_fields_super_classes tenv (Sil.TN_csu (Sil.Class, sc)) in
|
|
|
|
General_utils.append_no_duplicates_fields fields sc_fields
|
|
|
|
| Some Sil.Tstruct (fields, _, _, _, _, _, _) -> fields
|
|
|
|
| Some _ -> []
|
|
|
|
|
|
|
|
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 = CTypes.mk_classname (Ast_utils.get_qualified_name sc) in
|
|
|
|
get_fields_super_classes tenv classname
|
|
|
|
| _ -> [])
|
|
|
|
| _ -> []
|
|
|
|
|
|
|
|
let build_sil_field type_ptr_to_sil_type tenv field_name type_ptr prop_attributes =
|
|
|
|
let prop_atts = IList.map Clang_ast_j.string_of_property_attribute prop_attributes in
|
|
|
|
let annotation_from_type t =
|
|
|
|
match t with
|
|
|
|
| Sil.Tptr (_, Sil.Pk_objc_weak) -> [Config.weak]
|
|
|
|
| Sil.Tptr (_, Sil.Pk_objc_unsafe_unretained) -> [Config.unsafe_unret]
|
|
|
|
| _ -> [] in
|
|
|
|
let fname = General_utils.mk_class_field_name field_name in
|
|
|
|
let typ = type_ptr_to_sil_type tenv type_ptr in
|
|
|
|
let item_annotations = match prop_atts with
|
|
|
|
| [] ->
|
|
|
|
[({ Sil.class_name = Config.ivar_attributes; Sil.parameters = annotation_from_type typ }, true)]
|
|
|
|
| _ ->
|
|
|
|
[({ Sil.class_name = Config.property_attributes; Sil.parameters = prop_atts }, true)] in
|
|
|
|
fname, typ, item_annotations
|
|
|
|
|
|
|
|
(* From an ivar look for its property and if it finds it returns its attributes *)
|
|
|
|
let ivar_property curr_class ivar =
|
|
|
|
Printing.log_out "Checking if a property is defined for the ivar: '%s'@."
|
|
|
|
ivar.Clang_ast_t.ni_name;
|
|
|
|
match ObjcProperty_decl.Property.find_property_name_from_ivar curr_class ivar with
|
|
|
|
| Some pname' ->
|
|
|
|
(Printing.log_out "Found property name from ivar: '%s'" pname'.Clang_ast_t.ni_name;
|
|
|
|
try
|
|
|
|
let _, atts, _, _, _, _ = ObjcProperty_decl.Property.find_property curr_class pname' in
|
|
|
|
atts
|
|
|
|
with Not_found ->
|
|
|
|
Printing.log_out "Didn't find property for pname '%s'" pname'.Clang_ast_t.ni_name;
|
|
|
|
[])
|
|
|
|
| None -> Printing.log_out "No property found for ivar '%s'@." ivar.Clang_ast_t.ni_name;
|
|
|
|
[]
|
|
|
|
|
|
|
|
let build_sil_field_property type_ptr_to_sil_type curr_class tenv field_name type_ptr att_opt =
|
|
|
|
let prop_attributes =
|
|
|
|
match att_opt with
|
|
|
|
| Some prop_attributes -> prop_attributes
|
|
|
|
| None -> ivar_property curr_class field_name in
|
|
|
|
build_sil_field type_ptr_to_sil_type tenv field_name type_ptr prop_attributes
|
|
|
|
|
|
|
|
(* Given a list of declarations in an interface returns a list of fields *)
|
|
|
|
let rec get_fields type_ptr_to_sil_type tenv curr_class decl_list =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match decl_list with
|
|
|
|
| [] -> []
|
|
|
|
| ObjCIvarDecl (decl_info, name_info, type_ptr, field_decl_info, obj_c_ivar_decl_info) :: decl_list' ->
|
|
|
|
let fields = get_fields type_ptr_to_sil_type tenv curr_class decl_list' in
|
|
|
|
(* Doing a post visit here. Adding Ivar after all the declaration have been visited so that *)
|
|
|
|
(* ivar names will be added in the property list. *)
|
|
|
|
Printing.log_out " ...Adding Instance Variable '%s' @." name_info.Clang_ast_t.ni_name;
|
|
|
|
let (fname, typ, ia) =
|
|
|
|
build_sil_field_property type_ptr_to_sil_type curr_class tenv name_info type_ptr None in
|
|
|
|
Printing.log_out " ...Resulting sil field: (%s) with attributes:@." ((Ident.fieldname_to_string fname) ^":"^(Sil.typ_to_string typ));
|
|
|
|
IList.iter (fun (ia', _) ->
|
|
|
|
IList.iter (fun a -> Printing.log_out " '%s'@." a) ia'.Sil.parameters) ia;
|
|
|
|
(fname, typ, ia):: fields
|
|
|
|
| ObjCPropertyImplDecl (decl_info, property_impl_decl_info):: decl_list' ->
|
|
|
|
let property_fields_decl =
|
|
|
|
ObjcProperty_decl.prepare_dynamic_property curr_class decl_info property_impl_decl_info in
|
|
|
|
get_fields type_ptr_to_sil_type tenv curr_class (property_fields_decl @ decl_list')
|
|
|
|
| _ :: decl_list' -> get_fields type_ptr_to_sil_type tenv curr_class 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 fields =
|
|
|
|
let mang_name = Mangled.from_string class_name in
|
|
|
|
let class_tn_name = Sil.TN_csu (Sil.Class, mang_name) in
|
|
|
|
match Sil.tenv_lookup tenv class_tn_name with
|
|
|
|
| Some Sil.Tstruct(intf_fields, _, _, _, superclass, methods, annotation) ->
|
|
|
|
let new_fields = General_utils.append_no_duplicates_fields fields intf_fields in
|
|
|
|
let new_fields = CFrontend_utils.General_utils.sort_fields new_fields in
|
|
|
|
let class_type_info =
|
|
|
|
Sil.Tstruct (
|
|
|
|
new_fields, [], Sil.Class, Some mang_name, superclass, methods, annotation
|
|
|
|
) in
|
|
|
|
Printing.log_out " Updating info for class '%s' in tenv\n" class_name;
|
|
|
|
Sil.tenv_add tenv class_tn_name class_type_info
|
|
|
|
| _ -> ()
|
|
|
|
|
|
|
|
(* checks if ivar is defined among a set of fields and if it is atomic *)
|
|
|
|
let is_ivar_atomic ivar fields =
|
|
|
|
let do_one_annot a =
|
|
|
|
(a.Sil.class_name = Config.property_attributes) &&
|
|
|
|
IList.exists (fun p -> p = CFrontend_config.atomic_att) a.Sil.parameters in
|
|
|
|
let has_atomic_annot ans =
|
|
|
|
IList.exists (fun (a, _) -> do_one_annot a) ans in
|
|
|
|
try
|
|
|
|
let _, _, annot = IList.find (fun (fn, _, _) -> Ident.fieldname_equal ivar fn) fields in
|
|
|
|
has_atomic_annot annot
|
|
|
|
with Not_found -> (
|
|
|
|
Printing.log_out "NOT Found field ivar = '%s' " (Ident.fieldname_to_string ivar);
|
|
|
|
false)
|
|
|
|
|
|
|
|
let get_property_corresponding_ivar tenv type_ptr_to_sil_type class_name property_decl =
|
|
|
|
let open Clang_ast_t in
|
|
|
|
match property_decl with
|
|
|
|
| ObjCPropertyDecl (decl_info, named_decl_info, obj_c_property_decl_info) ->
|
|
|
|
(let ivar_decl_ref = obj_c_property_decl_info.Clang_ast_t.opdi_ivar_decl in
|
|
|
|
match Ast_utils.get_decl_opt_with_decl_ref ivar_decl_ref with
|
|
|
|
| Some ObjCIvarDecl (decl_info, named_decl_info, type_ptr, _, _) ->
|
|
|
|
General_utils.mk_class_field_name named_decl_info
|
|
|
|
| _ -> (* Ivar is not known, so add a default one to the tenv *)
|
|
|
|
let type_ptr = obj_c_property_decl_info.Clang_ast_t.opdi_type_ptr in
|
|
|
|
let field_name_str = Ast_utils.generated_ivar_name named_decl_info in
|
|
|
|
let prop_attributes = obj_c_property_decl_info.Clang_ast_t.opdi_property_attributes in
|
|
|
|
let field_name, typ, attr = build_sil_field type_ptr_to_sil_type tenv
|
|
|
|
field_name_str type_ptr prop_attributes in
|
|
|
|
ignore (add_missing_fields tenv class_name [(field_name, typ, attr)]);
|
|
|
|
field_name)
|
|
|
|
| _ -> assert false
|