You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			510 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			510 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===-- DWARFASTParserJava.cpp ----------------------------------*- C++ -*-===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "DWARFASTParserJava.h"
 | ||
|  | #include "DWARFAttribute.h"
 | ||
|  | #include "DWARFCompileUnit.h"
 | ||
|  | #include "DWARFDebugInfoEntry.h"
 | ||
|  | #include "DWARFDebugInfoEntry.h"
 | ||
|  | #include "DWARFDeclContext.h"
 | ||
|  | #include "SymbolFileDWARF.h"
 | ||
|  | 
 | ||
|  | #include "lldb/Core/Module.h"
 | ||
|  | #include "lldb/Symbol/CompileUnit.h"
 | ||
|  | #include "lldb/Symbol/SymbolContextScope.h"
 | ||
|  | #include "lldb/Symbol/TypeList.h"
 | ||
|  | 
 | ||
|  | using namespace lldb; | ||
|  | using namespace lldb_private; | ||
|  | 
 | ||
|  | DWARFASTParserJava::DWARFASTParserJava(JavaASTContext &ast) : m_ast(ast) {} | ||
|  | 
 | ||
|  | DWARFASTParserJava::~DWARFASTParserJava() {} | ||
|  | 
 | ||
|  | TypeSP DWARFASTParserJava::ParseBaseTypeFromDIE(const DWARFDIE &die) { | ||
|  |   SymbolFileDWARF *dwarf = die.GetDWARF(); | ||
|  |   dwarf->m_die_to_type[die.GetDIE()] = DIE_IS_BEING_PARSED; | ||
|  | 
 | ||
|  |   ConstString type_name; | ||
|  |   uint64_t byte_size = 0; | ||
|  | 
 | ||
|  |   DWARFAttributes attributes; | ||
|  |   const size_t num_attributes = die.GetAttributes(attributes); | ||
|  |   for (uint32_t i = 0; i < num_attributes; ++i) { | ||
|  |     DWARFFormValue form_value; | ||
|  |     dw_attr_t attr = attributes.AttributeAtIndex(i); | ||
|  |     if (attributes.ExtractFormValueAtIndex(i, form_value)) { | ||
|  |       switch (attr) { | ||
|  |       case DW_AT_name: | ||
|  |         type_name.SetCString(form_value.AsCString()); | ||
|  |         break; | ||
|  |       case DW_AT_byte_size: | ||
|  |         byte_size = form_value.Unsigned(); | ||
|  |         break; | ||
|  |       case DW_AT_encoding: | ||
|  |         break; | ||
|  |       default: | ||
|  |         assert(false && "Unsupported attribute for DW_TAG_base_type"); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   Declaration decl; | ||
|  |   CompilerType compiler_type = m_ast.CreateBaseType(type_name); | ||
|  |   return std::make_shared<Type>(die.GetID(), dwarf, type_name, byte_size, | ||
|  |                                 nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, | ||
|  |                                 decl, compiler_type, Type::eResolveStateFull); | ||
|  | } | ||
|  | 
 | ||
|  | TypeSP DWARFASTParserJava::ParseArrayTypeFromDIE(const DWARFDIE &die) { | ||
|  |   SymbolFileDWARF *dwarf = die.GetDWARF(); | ||
|  |   dwarf->m_die_to_type[die.GetDIE()] = DIE_IS_BEING_PARSED; | ||
|  | 
 | ||
|  |   ConstString linkage_name; | ||
|  |   DWARFFormValue type_attr_value; | ||
|  |   lldb::addr_t data_offset = LLDB_INVALID_ADDRESS; | ||
|  |   DWARFExpression length_expression(die.GetCU()); | ||
|  | 
 | ||
|  |   DWARFAttributes attributes; | ||
|  |   const size_t num_attributes = die.GetAttributes(attributes); | ||
|  |   for (uint32_t i = 0; i < num_attributes; ++i) { | ||
|  |     DWARFFormValue form_value; | ||
|  |     dw_attr_t attr = attributes.AttributeAtIndex(i); | ||
|  |     if (attributes.ExtractFormValueAtIndex(i, form_value)) { | ||
|  |       switch (attr) { | ||
|  |       case DW_AT_linkage_name: | ||
|  |         linkage_name.SetCString(form_value.AsCString()); | ||
|  |         break; | ||
|  |       case DW_AT_type: | ||
|  |         type_attr_value = form_value; | ||
|  |         break; | ||
|  |       case DW_AT_data_member_location: | ||
|  |         data_offset = form_value.Unsigned(); | ||
|  |         break; | ||
|  |       case DW_AT_declaration: | ||
|  |         break; | ||
|  |       default: | ||
|  |         assert(false && "Unsupported attribute for DW_TAG_array_type"); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   for (DWARFDIE child_die = die.GetFirstChild(); child_die.IsValid(); | ||
|  |        child_die = child_die.GetSibling()) { | ||
|  |     if (child_die.Tag() == DW_TAG_subrange_type) { | ||
|  |       DWARFAttributes attributes; | ||
|  |       const size_t num_attributes = child_die.GetAttributes(attributes); | ||
|  |       for (uint32_t i = 0; i < num_attributes; ++i) { | ||
|  |         DWARFFormValue form_value; | ||
|  |         dw_attr_t attr = attributes.AttributeAtIndex(i); | ||
|  |         if (attributes.ExtractFormValueAtIndex(i, form_value)) { | ||
|  |           switch (attr) { | ||
|  |           case DW_AT_count: | ||
|  |             if (form_value.BlockData()) | ||
|  |               length_expression.CopyOpcodeData( | ||
|  |                   form_value.BlockData(), form_value.Unsigned(), | ||
|  |                   child_die.GetCU()->GetByteOrder(), | ||
|  |                   child_die.GetCU()->GetAddressByteSize()); | ||
|  |             break; | ||
|  |           default: | ||
|  |             assert(false && "Unsupported attribute for DW_TAG_subrange_type"); | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  |     } else { | ||
|  |       assert(false && "Unsupported child for DW_TAG_array_type"); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   DIERef type_die_ref(type_attr_value); | ||
|  |   Type *element_type = dwarf->ResolveTypeUID(type_die_ref); | ||
|  |   if (!element_type) | ||
|  |     return nullptr; | ||
|  | 
 | ||
|  |   CompilerType element_compiler_type = element_type->GetForwardCompilerType(); | ||
|  |   CompilerType array_compiler_type = m_ast.CreateArrayType( | ||
|  |       linkage_name, element_compiler_type, length_expression, data_offset); | ||
|  | 
 | ||
|  |   Declaration decl; | ||
|  |   TypeSP type_sp(new Type(die.GetID(), dwarf, array_compiler_type.GetTypeName(), | ||
|  |                           -1, nullptr, type_die_ref.GetUID(dwarf), | ||
|  |                           Type::eEncodingIsUID, &decl, array_compiler_type, | ||
|  |                           Type::eResolveStateFull)); | ||
|  |   type_sp->SetEncodingType(element_type); | ||
|  |   return type_sp; | ||
|  | } | ||
|  | 
 | ||
|  | TypeSP DWARFASTParserJava::ParseReferenceTypeFromDIE(const DWARFDIE &die) { | ||
|  |   SymbolFileDWARF *dwarf = die.GetDWARF(); | ||
|  |   dwarf->m_die_to_type[die.GetDIE()] = DIE_IS_BEING_PARSED; | ||
|  | 
 | ||
|  |   Declaration decl; | ||
|  |   DWARFFormValue type_attr_value; | ||
|  | 
 | ||
|  |   DWARFAttributes attributes; | ||
|  |   const size_t num_attributes = die.GetAttributes(attributes); | ||
|  |   for (uint32_t i = 0; i < num_attributes; ++i) { | ||
|  |     DWARFFormValue form_value; | ||
|  |     dw_attr_t attr = attributes.AttributeAtIndex(i); | ||
|  |     if (attributes.ExtractFormValueAtIndex(i, form_value)) { | ||
|  |       switch (attr) { | ||
|  |       case DW_AT_type: | ||
|  |         type_attr_value = form_value; | ||
|  |         break; | ||
|  |       default: | ||
|  |         assert(false && "Unsupported attribute for DW_TAG_array_type"); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   DIERef type_die_ref(type_attr_value); | ||
|  |   Type *pointee_type = dwarf->ResolveTypeUID(type_die_ref); | ||
|  |   if (!pointee_type) | ||
|  |     return nullptr; | ||
|  | 
 | ||
|  |   CompilerType pointee_compiler_type = pointee_type->GetForwardCompilerType(); | ||
|  |   CompilerType reference_compiler_type = | ||
|  |       m_ast.CreateReferenceType(pointee_compiler_type); | ||
|  |   TypeSP type_sp( | ||
|  |       new Type(die.GetID(), dwarf, reference_compiler_type.GetTypeName(), -1, | ||
|  |                nullptr, type_die_ref.GetUID(dwarf), Type::eEncodingIsUID, &decl, | ||
|  |                reference_compiler_type, Type::eResolveStateFull)); | ||
|  |   type_sp->SetEncodingType(pointee_type); | ||
|  |   return type_sp; | ||
|  | } | ||
|  | 
 | ||
|  | lldb::TypeSP DWARFASTParserJava::ParseClassTypeFromDIE(const DWARFDIE &die, | ||
|  |                                                        bool &is_new_type) { | ||
|  |   SymbolFileDWARF *dwarf = die.GetDWARF(); | ||
|  |   dwarf->m_die_to_type[die.GetDIE()] = DIE_IS_BEING_PARSED; | ||
|  | 
 | ||
|  |   Declaration decl; | ||
|  |   ConstString name; | ||
|  |   ConstString linkage_name; | ||
|  |   bool is_forward_declaration = false; | ||
|  |   uint32_t byte_size = 0; | ||
|  | 
 | ||
|  |   DWARFAttributes attributes; | ||
|  |   const size_t num_attributes = die.GetAttributes(attributes); | ||
|  |   for (uint32_t i = 0; i < num_attributes; ++i) { | ||
|  |     DWARFFormValue form_value; | ||
|  |     dw_attr_t attr = attributes.AttributeAtIndex(i); | ||
|  |     if (attributes.ExtractFormValueAtIndex(i, form_value)) { | ||
|  |       switch (attr) { | ||
|  |       case DW_AT_name: | ||
|  |         name.SetCString(form_value.AsCString()); | ||
|  |         break; | ||
|  |       case DW_AT_declaration: | ||
|  |         is_forward_declaration = form_value.Boolean(); | ||
|  |         break; | ||
|  |       case DW_AT_byte_size: | ||
|  |         byte_size = form_value.Unsigned(); | ||
|  |         break; | ||
|  |       case DW_AT_linkage_name: | ||
|  |         linkage_name.SetCString(form_value.AsCString()); | ||
|  |         break; | ||
|  |       default: | ||
|  |         assert(false && "Unsupported attribute for DW_TAG_class_type"); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   UniqueDWARFASTType unique_ast_entry; | ||
|  |   if (name) { | ||
|  |     std::string qualified_name; | ||
|  |     if (die.GetQualifiedName(qualified_name)) { | ||
|  |       name.SetCString(qualified_name.c_str()); | ||
|  |       if (dwarf->GetUniqueDWARFASTTypeMap().Find(name, die, Declaration(), -1, | ||
|  |                                                  unique_ast_entry)) { | ||
|  |         if (unique_ast_entry.m_type_sp) { | ||
|  |           dwarf->GetDIEToType()[die.GetDIE()] = | ||
|  |               unique_ast_entry.m_type_sp.get(); | ||
|  |           is_new_type = false; | ||
|  |           return unique_ast_entry.m_type_sp; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (is_forward_declaration) { | ||
|  |     DWARFDeclContext die_decl_ctx; | ||
|  |     die.GetDWARFDeclContext(die_decl_ctx); | ||
|  | 
 | ||
|  |     TypeSP type_sp = dwarf->FindDefinitionTypeForDWARFDeclContext(die_decl_ctx); | ||
|  |     if (type_sp) { | ||
|  |       // We found a real definition for this type elsewhere so lets use it
 | ||
|  |       dwarf->GetDIEToType()[die.GetDIE()] = type_sp.get(); | ||
|  |       is_new_type = false; | ||
|  |       return type_sp; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   CompilerType compiler_type( | ||
|  |       &m_ast, dwarf->GetForwardDeclDieToClangType().lookup(die.GetDIE())); | ||
|  |   if (!compiler_type) | ||
|  |     compiler_type = m_ast.CreateObjectType(name, linkage_name, byte_size); | ||
|  | 
 | ||
|  |   is_new_type = true; | ||
|  |   TypeSP type_sp(new Type(die.GetID(), dwarf, name, | ||
|  |                           -1, // byte size isn't specified
 | ||
|  |                           nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, | ||
|  |                           &decl, compiler_type, Type::eResolveStateForward)); | ||
|  | 
 | ||
|  |   // Add our type to the unique type map
 | ||
|  |   unique_ast_entry.m_type_sp = type_sp; | ||
|  |   unique_ast_entry.m_die = die; | ||
|  |   unique_ast_entry.m_declaration = decl; | ||
|  |   unique_ast_entry.m_byte_size = -1; | ||
|  |   dwarf->GetUniqueDWARFASTTypeMap().Insert(name, unique_ast_entry); | ||
|  | 
 | ||
|  |   if (!is_forward_declaration) { | ||
|  |     // Leave this as a forward declaration until we need to know the details of
 | ||
|  |     // the type
 | ||
|  |     dwarf->GetForwardDeclDieToClangType()[die.GetDIE()] = | ||
|  |         compiler_type.GetOpaqueQualType(); | ||
|  |     dwarf->GetForwardDeclClangTypeToDie()[compiler_type.GetOpaqueQualType()] = | ||
|  |         die.GetDIERef(); | ||
|  |   } | ||
|  |   return type_sp; | ||
|  | } | ||
|  | 
 | ||
|  | lldb::TypeSP DWARFASTParserJava::ParseTypeFromDWARF( | ||
|  |     const lldb_private::SymbolContext &sc, const DWARFDIE &die, | ||
|  |     lldb_private::Log *log, bool *type_is_new_ptr) { | ||
|  |   if (type_is_new_ptr) | ||
|  |     *type_is_new_ptr = false; | ||
|  | 
 | ||
|  |   if (!die) | ||
|  |     return nullptr; | ||
|  | 
 | ||
|  |   SymbolFileDWARF *dwarf = die.GetDWARF(); | ||
|  | 
 | ||
|  |   Type *type_ptr = dwarf->m_die_to_type.lookup(die.GetDIE()); | ||
|  |   if (type_ptr == DIE_IS_BEING_PARSED) | ||
|  |     return nullptr; | ||
|  |   if (type_ptr != nullptr) | ||
|  |     return type_ptr->shared_from_this(); | ||
|  | 
 | ||
|  |   TypeSP type_sp; | ||
|  |   if (type_is_new_ptr) | ||
|  |     *type_is_new_ptr = true; | ||
|  | 
 | ||
|  |   switch (die.Tag()) { | ||
|  |   case DW_TAG_base_type: { | ||
|  |     type_sp = ParseBaseTypeFromDIE(die); | ||
|  |     break; | ||
|  |   } | ||
|  |   case DW_TAG_array_type: { | ||
|  |     type_sp = ParseArrayTypeFromDIE(die); | ||
|  |     break; | ||
|  |   } | ||
|  |   case DW_TAG_class_type: { | ||
|  |     bool is_new_type = false; | ||
|  |     type_sp = ParseClassTypeFromDIE(die, is_new_type); | ||
|  |     if (!is_new_type) | ||
|  |       return type_sp; | ||
|  |     break; | ||
|  |   } | ||
|  |   case DW_TAG_reference_type: { | ||
|  |     type_sp = ParseReferenceTypeFromDIE(die); | ||
|  |     break; | ||
|  |   } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!type_sp) | ||
|  |     return nullptr; | ||
|  | 
 | ||
|  |   DWARFDIE sc_parent_die = SymbolFileDWARF::GetParentSymbolContextDIE(die); | ||
|  |   dw_tag_t sc_parent_tag = sc_parent_die.Tag(); | ||
|  | 
 | ||
|  |   SymbolContextScope *symbol_context_scope = nullptr; | ||
|  |   if (sc_parent_tag == DW_TAG_compile_unit) { | ||
|  |     symbol_context_scope = sc.comp_unit; | ||
|  |   } else if (sc.function != nullptr && sc_parent_die) { | ||
|  |     symbol_context_scope = | ||
|  |         sc.function->GetBlock(true).FindBlockByID(sc_parent_die.GetID()); | ||
|  |     if (symbol_context_scope == nullptr) | ||
|  |       symbol_context_scope = sc.function; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (symbol_context_scope != nullptr) | ||
|  |     type_sp->SetSymbolContextScope(symbol_context_scope); | ||
|  | 
 | ||
|  |   dwarf->GetTypeList()->Insert(type_sp); | ||
|  |   dwarf->m_die_to_type[die.GetDIE()] = type_sp.get(); | ||
|  | 
 | ||
|  |   return type_sp; | ||
|  | } | ||
|  | 
 | ||
|  | lldb_private::Function *DWARFASTParserJava::ParseFunctionFromDWARF( | ||
|  |     const lldb_private::SymbolContext &sc, const DWARFDIE &die) { | ||
|  |   assert(die.Tag() == DW_TAG_subprogram); | ||
|  | 
 | ||
|  |   const char *name = nullptr; | ||
|  |   const char *mangled = nullptr; | ||
|  |   int decl_file = 0; | ||
|  |   int decl_line = 0; | ||
|  |   int decl_column = 0; | ||
|  |   int call_file = 0; | ||
|  |   int call_line = 0; | ||
|  |   int call_column = 0; | ||
|  |   DWARFRangeList func_ranges; | ||
|  |   DWARFExpression frame_base(die.GetCU()); | ||
|  | 
 | ||
|  |   if (die.GetDIENamesAndRanges(name, mangled, func_ranges, decl_file, decl_line, | ||
|  |                                decl_column, call_file, call_line, call_column, | ||
|  |                                &frame_base)) { | ||
|  |     // Union of all ranges in the function DIE (if the function is
 | ||
|  |     // discontiguous)
 | ||
|  |     AddressRange func_range; | ||
|  |     lldb::addr_t lowest_func_addr = func_ranges.GetMinRangeBase(0); | ||
|  |     lldb::addr_t highest_func_addr = func_ranges.GetMaxRangeEnd(0); | ||
|  |     if (lowest_func_addr != LLDB_INVALID_ADDRESS && | ||
|  |         lowest_func_addr <= highest_func_addr) { | ||
|  |       ModuleSP module_sp(die.GetModule()); | ||
|  |       func_range.GetBaseAddress().ResolveAddressUsingFileSections( | ||
|  |           lowest_func_addr, module_sp->GetSectionList()); | ||
|  |       if (func_range.GetBaseAddress().IsValid()) | ||
|  |         func_range.SetByteSize(highest_func_addr - lowest_func_addr); | ||
|  |     } | ||
|  | 
 | ||
|  |     if (func_range.GetBaseAddress().IsValid()) { | ||
|  |       std::unique_ptr<Declaration> decl_ap; | ||
|  |       if (decl_file != 0 || decl_line != 0 || decl_column != 0) | ||
|  |         decl_ap.reset(new Declaration( | ||
|  |             sc.comp_unit->GetSupportFiles().GetFileSpecAtIndex(decl_file), | ||
|  |             decl_line, decl_column)); | ||
|  | 
 | ||
|  |       if (die.GetDWARF()->FixupAddress(func_range.GetBaseAddress())) { | ||
|  |         FunctionSP func_sp(new Function(sc.comp_unit, die.GetID(), die.GetID(), | ||
|  |                                         Mangled(ConstString(name), false), | ||
|  |                                         nullptr, // No function types in java
 | ||
|  |                                         func_range)); | ||
|  |         if (frame_base.IsValid()) | ||
|  |           func_sp->GetFrameBaseExpression() = frame_base; | ||
|  |         sc.comp_unit->AddFunction(func_sp); | ||
|  | 
 | ||
|  |         return func_sp.get(); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  |   return nullptr; | ||
|  | } | ||
|  | 
 | ||
|  | bool DWARFASTParserJava::CompleteTypeFromDWARF( | ||
|  |     const DWARFDIE &die, lldb_private::Type *type, | ||
|  |     lldb_private::CompilerType &java_type) { | ||
|  |   switch (die.Tag()) { | ||
|  |   case DW_TAG_class_type: { | ||
|  |     if (die.GetAttributeValueAsUnsigned(DW_AT_declaration, 0) == 0) { | ||
|  |       if (die.HasChildren()) | ||
|  |         ParseChildMembers(die, java_type); | ||
|  |       m_ast.CompleteObjectType(java_type); | ||
|  |       return java_type.IsValid(); | ||
|  |     } | ||
|  |   } break; | ||
|  |   default: | ||
|  |     assert(false && "Not a forward java type declaration!"); | ||
|  |     break; | ||
|  |   } | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | void DWARFASTParserJava::ParseChildMembers(const DWARFDIE &parent_die, | ||
|  |                                            CompilerType &compiler_type) { | ||
|  |   DWARFCompileUnit *dwarf_cu = parent_die.GetCU(); | ||
|  |   for (DWARFDIE die = parent_die.GetFirstChild(); die.IsValid(); | ||
|  |        die = die.GetSibling()) { | ||
|  |     switch (die.Tag()) { | ||
|  |     case DW_TAG_member: { | ||
|  |       const char *name = nullptr; | ||
|  |       DWARFFormValue encoding_uid; | ||
|  |       uint32_t member_byte_offset = UINT32_MAX; | ||
|  |       DWARFExpression member_location_expression(dwarf_cu); | ||
|  | 
 | ||
|  |       DWARFAttributes attributes; | ||
|  |       size_t num_attributes = die.GetAttributes(attributes); | ||
|  |       for (size_t i = 0; i < num_attributes; ++i) { | ||
|  |         DWARFFormValue form_value; | ||
|  |         if (attributes.ExtractFormValueAtIndex(i, form_value)) { | ||
|  |           switch (attributes.AttributeAtIndex(i)) { | ||
|  |           case DW_AT_name: | ||
|  |             name = form_value.AsCString(); | ||
|  |             break; | ||
|  |           case DW_AT_type: | ||
|  |             encoding_uid = form_value; | ||
|  |             break; | ||
|  |           case DW_AT_data_member_location: | ||
|  |             if (form_value.BlockData()) | ||
|  |               member_location_expression.CopyOpcodeData( | ||
|  |                   form_value.BlockData(), form_value.Unsigned(), | ||
|  |                   dwarf_cu->GetByteOrder(), dwarf_cu->GetAddressByteSize()); | ||
|  |             else | ||
|  |               member_byte_offset = form_value.Unsigned(); | ||
|  |             break; | ||
|  |           case DW_AT_artificial: | ||
|  |             static_cast<void>(form_value.Boolean()); | ||
|  |             break; | ||
|  |           case DW_AT_accessibility: | ||
|  |             // TODO: Handle when needed
 | ||
|  |             break; | ||
|  |           default: | ||
|  |             assert(false && "Unhandled attribute for DW_TAG_member"); | ||
|  |             break; | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       if (strcmp(name, ".dynamic_type") == 0) | ||
|  |         m_ast.SetDynamicTypeId(compiler_type, member_location_expression); | ||
|  |       else { | ||
|  |         if (Type *member_type = die.ResolveTypeUID(DIERef(encoding_uid))) | ||
|  |           m_ast.AddMemberToObject(compiler_type, ConstString(name), | ||
|  |                                   member_type->GetFullCompilerType(), | ||
|  |                                   member_byte_offset); | ||
|  |       } | ||
|  |       break; | ||
|  |     } | ||
|  |     case DW_TAG_inheritance: { | ||
|  |       DWARFFormValue encoding_uid; | ||
|  |       uint32_t member_byte_offset = UINT32_MAX; | ||
|  | 
 | ||
|  |       DWARFAttributes attributes; | ||
|  |       size_t num_attributes = die.GetAttributes(attributes); | ||
|  |       for (size_t i = 0; i < num_attributes; ++i) { | ||
|  |         DWARFFormValue form_value; | ||
|  |         if (attributes.ExtractFormValueAtIndex(i, form_value)) { | ||
|  |           switch (attributes.AttributeAtIndex(i)) { | ||
|  |           case DW_AT_type: | ||
|  |             encoding_uid = form_value; | ||
|  |             break; | ||
|  |           case DW_AT_data_member_location: | ||
|  |             member_byte_offset = form_value.Unsigned(); | ||
|  |             break; | ||
|  |           case DW_AT_accessibility: | ||
|  |             // In java all base class is public so we can ignore this attribute
 | ||
|  |             break; | ||
|  |           default: | ||
|  |             assert(false && "Unhandled attribute for DW_TAG_member"); | ||
|  |             break; | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  |       if (Type *base_type = die.ResolveTypeUID(DIERef(encoding_uid))) | ||
|  |         m_ast.AddBaseClassToObject(compiler_type, | ||
|  |                                    base_type->GetFullCompilerType(), | ||
|  |                                    member_byte_offset); | ||
|  |       break; | ||
|  |     } | ||
|  |     default: | ||
|  |       break; | ||
|  |     } | ||
|  |   } | ||
|  | } |