mirror of
https://github.com/Gericom/teak-llvm.git
synced 2025-06-23 21:45:46 -04:00

DWARF accelerator table sections to the DWARF parser. These sections are similar to the .debug_pubnames and .debug_pubtypes, but they are designed to be hash tables that are saved to disc in a way that the sections can just be loaded into memory and used without any work on the debugger side. The .debug_pubnames and .debug_pubtypes sections are not ordered, contain a copy of the name in the section itself which makes these sections quite large, they only include publicly exported names (so no static functions, no types defined inside functions), many compilers put different information in them making them very unreliable so most debugger ignore these sections and parse the DWARF on their own. The tables must also be parsed and sorted in order to be used effectively. The new sections can be quickly loaded and very efficiently be used to do name to DIE lookups with very little up front work. The format of these new sections will be changing while we work out the bugs, but we hope to have really fast name to DIE lookups soon. llvm-svn: 138979
192 lines
9.4 KiB
C++
192 lines
9.4 KiB
C++
//===-- ObjectFile.cpp ------------------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "lldb/lldb-private.h"
|
|
#include "lldb/Core/Module.h"
|
|
#include "lldb/Core/PluginManager.h"
|
|
#include "lldb/Core/RegularExpression.h"
|
|
#include "lldb/Core/Timer.h"
|
|
#include "lldb/Symbol/ObjectFile.h"
|
|
#include "lldb/Symbol/ObjectContainer.h"
|
|
#include "lldb/Symbol/SymbolFile.h"
|
|
|
|
using namespace lldb;
|
|
using namespace lldb_private;
|
|
|
|
ObjectFile*
|
|
ObjectFile::FindPlugin (Module* module, const FileSpec* file, lldb::addr_t file_offset, lldb::addr_t file_size)
|
|
{
|
|
Timer scoped_timer (__PRETTY_FUNCTION__,
|
|
"ObjectFile::FindPlugin (module = %s/%s, file = %p, file_offset = 0x%z8.8x, file_size = 0x%z8.8x)",
|
|
module->GetFileSpec().GetDirectory().AsCString(),
|
|
module->GetFileSpec().GetFilename().AsCString(),
|
|
file, file_offset, file_size);
|
|
std::auto_ptr<ObjectFile> object_file_ap;
|
|
|
|
if (module != NULL)
|
|
{
|
|
if (file)
|
|
{
|
|
if (file_size == 0)
|
|
file_size = file->GetByteSize();
|
|
|
|
if (file_size == 0)
|
|
{
|
|
// Check for archive file with format "/path/to/archive.a(object.o)"
|
|
char path_with_object[PATH_MAX*2];
|
|
module->GetFileSpec().GetPath(path_with_object, sizeof(path_with_object));
|
|
|
|
RegularExpression g_object_regex("(.*)\\(([^\\)]+)\\)$");
|
|
if (g_object_regex.Execute (path_with_object, 2))
|
|
{
|
|
FileSpec archive_file;
|
|
std::string path;
|
|
std::string object;
|
|
if (g_object_regex.GetMatchAtIndex (path_with_object, 1, path) &&
|
|
g_object_regex.GetMatchAtIndex (path_with_object, 2, object))
|
|
{
|
|
archive_file.SetFile (path.c_str(), false);
|
|
file_size = archive_file.GetByteSize();
|
|
if (file_size > 0)
|
|
module->SetFileSpecAndObjectName (archive_file, ConstString(object.c_str()));
|
|
}
|
|
}
|
|
}
|
|
|
|
// No need to delegate further if (file_offset, file_size) exceeds the total file size.
|
|
// This is the base case.
|
|
// if (file_offset + file_size > file->GetByteSize())
|
|
// return NULL;
|
|
|
|
DataBufferSP file_header_data_sp(file->ReadFileContents(file_offset, 512));
|
|
uint32_t idx;
|
|
|
|
// Check if this is a normal object file by iterating through
|
|
// all object file plugin instances.
|
|
ObjectFileCreateInstance create_object_file_callback;
|
|
for (idx = 0; (create_object_file_callback = PluginManager::GetObjectFileCreateCallbackAtIndex(idx)) != NULL; ++idx)
|
|
{
|
|
object_file_ap.reset (create_object_file_callback(module, file_header_data_sp, file, file_offset, file_size));
|
|
if (object_file_ap.get())
|
|
return object_file_ap.release();
|
|
}
|
|
|
|
// Check if this is a object container by iterating through
|
|
// all object container plugin instances and then trying to get
|
|
// an object file from the container.
|
|
ObjectContainerCreateInstance create_object_container_callback;
|
|
for (idx = 0; (create_object_container_callback = PluginManager::GetObjectContainerCreateCallbackAtIndex(idx)) != NULL; ++idx)
|
|
{
|
|
std::auto_ptr<ObjectContainer> object_container_ap(create_object_container_callback(module, file_header_data_sp, file, file_offset, file_size));
|
|
|
|
if (object_container_ap.get())
|
|
object_file_ap.reset (object_container_ap->GetObjectFile(file));
|
|
|
|
if (object_file_ap.get())
|
|
return object_file_ap.release();
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
bool
|
|
ObjectFile::SetModulesArchitecture (const ArchSpec &new_arch)
|
|
{
|
|
return m_module->SetArchitecture (new_arch);
|
|
}
|
|
|
|
AddressClass
|
|
ObjectFile::GetAddressClass (lldb::addr_t file_addr)
|
|
{
|
|
Symtab *symtab = GetSymtab();
|
|
if (symtab)
|
|
{
|
|
Symbol *symbol = symtab->FindSymbolContainingFileAddress(file_addr);
|
|
if (symbol)
|
|
{
|
|
const AddressRange *range_ptr = symbol->GetAddressRangePtr();
|
|
if (range_ptr)
|
|
{
|
|
const Section *section = range_ptr->GetBaseAddress().GetSection();
|
|
if (section)
|
|
{
|
|
const SectionType section_type = section->GetType();
|
|
switch (section_type)
|
|
{
|
|
case eSectionTypeInvalid: return eAddressClassUnknown;
|
|
case eSectionTypeCode: return eAddressClassCode;
|
|
case eSectionTypeContainer: return eAddressClassUnknown;
|
|
case eSectionTypeData: return eAddressClassData;
|
|
case eSectionTypeDataCString: return eAddressClassData;
|
|
case eSectionTypeDataCStringPointers: return eAddressClassData;
|
|
case eSectionTypeDataSymbolAddress: return eAddressClassData;
|
|
case eSectionTypeData4: return eAddressClassData;
|
|
case eSectionTypeData8: return eAddressClassData;
|
|
case eSectionTypeData16: return eAddressClassData;
|
|
case eSectionTypeDataPointers: return eAddressClassData;
|
|
case eSectionTypeZeroFill: return eAddressClassData;
|
|
case eSectionTypeDataObjCMessageRefs: return eAddressClassData;
|
|
case eSectionTypeDataObjCCFStrings: return eAddressClassData;
|
|
case eSectionTypeDebug: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugAbbrev: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugAranges: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugFrame: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugInfo: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugLine: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugLoc: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugMacInfo: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugPubNames: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugPubTypes: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugRanges: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugStr: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugNames: return eAddressClassDebug;
|
|
case eSectionTypeDWARFDebugTypes: return eAddressClassDebug;
|
|
case eSectionTypeEHFrame: return eAddressClassRuntime;
|
|
case eSectionTypeOther: return eAddressClassUnknown;
|
|
}
|
|
}
|
|
}
|
|
|
|
const SymbolType symbol_type = symbol->GetType();
|
|
switch (symbol_type)
|
|
{
|
|
case eSymbolTypeAny: return eAddressClassUnknown;
|
|
case eSymbolTypeAbsolute: return eAddressClassUnknown;
|
|
case eSymbolTypeExtern: return eAddressClassUnknown;
|
|
case eSymbolTypeCode: return eAddressClassCode;
|
|
case eSymbolTypeTrampoline: return eAddressClassCode;
|
|
case eSymbolTypeData: return eAddressClassData;
|
|
case eSymbolTypeRuntime: return eAddressClassRuntime;
|
|
case eSymbolTypeException: return eAddressClassRuntime;
|
|
case eSymbolTypeSourceFile: return eAddressClassDebug;
|
|
case eSymbolTypeHeaderFile: return eAddressClassDebug;
|
|
case eSymbolTypeObjectFile: return eAddressClassDebug;
|
|
case eSymbolTypeCommonBlock: return eAddressClassDebug;
|
|
case eSymbolTypeBlock: return eAddressClassDebug;
|
|
case eSymbolTypeLocal: return eAddressClassData;
|
|
case eSymbolTypeParam: return eAddressClassData;
|
|
case eSymbolTypeVariable: return eAddressClassData;
|
|
case eSymbolTypeVariableType: return eAddressClassDebug;
|
|
case eSymbolTypeLineEntry: return eAddressClassDebug;
|
|
case eSymbolTypeLineHeader: return eAddressClassDebug;
|
|
case eSymbolTypeScopeBegin: return eAddressClassDebug;
|
|
case eSymbolTypeScopeEnd: return eAddressClassDebug;
|
|
case eSymbolTypeAdditional: return eAddressClassUnknown;
|
|
case eSymbolTypeCompiler: return eAddressClassDebug;
|
|
case eSymbolTypeInstrumentation:return eAddressClassDebug;
|
|
case eSymbolTypeUndefined: return eAddressClassUnknown;
|
|
}
|
|
}
|
|
}
|
|
return eAddressClassUnknown;
|
|
}
|
|
|
|
|