180814287SRaphael Isemann //===-- ObjectFile.cpp ----------------------------------------------------===// 230fdc8d8SChris Lattner // 32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 630fdc8d8SChris Lattner // 730fdc8d8SChris Lattner //===----------------------------------------------------------------------===// 830fdc8d8SChris Lattner 9b9c1b51eSKate Stone #include "lldb/Symbol/ObjectFile.h" 1030fdc8d8SChris Lattner #include "lldb/Core/Module.h" 11f4d6de6aSGreg Clayton #include "lldb/Core/ModuleSpec.h" 1230fdc8d8SChris Lattner #include "lldb/Core/PluginManager.h" 131f746071SGreg Clayton #include "lldb/Core/Section.h" 1430c2441aSAleksandr Urakov #include "lldb/Symbol/CallFrameInfo.h" 1530fdc8d8SChris Lattner #include "lldb/Symbol/ObjectContainer.h" 1630fdc8d8SChris Lattner #include "lldb/Symbol/SymbolFile.h" 17bf9a7730SZachary Turner #include "lldb/Target/Process.h" 184687db0eSHafiz Abid Qadeer #include "lldb/Target/SectionLoadList.h" 19bf9a7730SZachary Turner #include "lldb/Target/Target.h" 20666cc0b2SZachary Turner #include "lldb/Utility/DataBuffer.h" 21666cc0b2SZachary Turner #include "lldb/Utility/DataBufferHeap.h" 22c34698a8SPavel Labath #include "lldb/Utility/LLDBLog.h" 236f9e6901SZachary Turner #include "lldb/Utility/Log.h" 2438d0632eSPavel Labath #include "lldb/Utility/Timer.h" 25b9c1b51eSKate Stone #include "lldb/lldb-private.h" 2630fdc8d8SChris Lattner 27da816ca0SGreg Clayton #include "llvm/Support/DJB.h" 28da816ca0SGreg Clayton 2930fdc8d8SChris Lattner using namespace lldb; 3030fdc8d8SChris Lattner using namespace lldb_private; 3130fdc8d8SChris Lattner 32e84f7841SPavel Labath char ObjectFile::ID; 33*fc54427eSJonas Devlieghere size_t ObjectFile::g_initial_bytes_to_read = 512; 34e84f7841SPavel Labath 35d97e9f1aSJonas Devlieghere static ObjectFileSP 36d97e9f1aSJonas Devlieghere CreateObjectFromContainer(const lldb::ModuleSP &module_sp, const FileSpec *file, 37d97e9f1aSJonas Devlieghere lldb::offset_t file_offset, lldb::offset_t file_size, 38d97e9f1aSJonas Devlieghere DataBufferSP &data_sp, lldb::offset_t &data_offset) { 39d97e9f1aSJonas Devlieghere ObjectContainerCreateInstance callback; 40d97e9f1aSJonas Devlieghere for (uint32_t idx = 0; 41d97e9f1aSJonas Devlieghere (callback = PluginManager::GetObjectContainerCreateCallbackAtIndex( 42d97e9f1aSJonas Devlieghere idx)) != nullptr; 43d97e9f1aSJonas Devlieghere ++idx) { 44d97e9f1aSJonas Devlieghere std::unique_ptr<ObjectContainer> object_container_up(callback( 45d97e9f1aSJonas Devlieghere module_sp, data_sp, data_offset, file, file_offset, file_size)); 46d97e9f1aSJonas Devlieghere if (object_container_up) 47d97e9f1aSJonas Devlieghere return object_container_up->GetObjectFile(file); 48d97e9f1aSJonas Devlieghere } 49d97e9f1aSJonas Devlieghere return {}; 50d97e9f1aSJonas Devlieghere } 51d97e9f1aSJonas Devlieghere 52762f7135SGreg Clayton ObjectFileSP 53b9c1b51eSKate Stone ObjectFile::FindPlugin(const lldb::ModuleSP &module_sp, const FileSpec *file, 54b9c1b51eSKate Stone lldb::offset_t file_offset, lldb::offset_t file_size, 55b9c1b51eSKate Stone DataBufferSP &data_sp, lldb::offset_t &data_offset) { 565c1c8443SJonas Devlieghere LLDB_SCOPED_TIMERF( 57b9c1b51eSKate Stone "ObjectFile::FindPlugin (module = %s, file = %p, file_offset = " 58b9c1b51eSKate Stone "0x%8.8" PRIx64 ", file_size = 0x%8.8" PRIx64 ")", 59b5ad4ec7SGreg Clayton module_sp->GetFileSpec().GetPath().c_str(), 60b9c1b51eSKate Stone static_cast<const void *>(file), static_cast<uint64_t>(file_offset), 61324a1036SSaleem Abdulrasool static_cast<uint64_t>(file_size)); 62d97e9f1aSJonas Devlieghere 63d97e9f1aSJonas Devlieghere if (!module_sp) 64d97e9f1aSJonas Devlieghere return {}; 65d97e9f1aSJonas Devlieghere 66d97e9f1aSJonas Devlieghere if (!file) 67d97e9f1aSJonas Devlieghere return {}; 685ce9c565SGreg Clayton 69b9c1b51eSKate Stone if (!data_sp) { 70a4a00cedSFred Riss const bool file_exists = FileSystem::Instance().Exists(*file); 7105097246SAdrian Prantl // We have an object name which most likely means we have a .o file in 7205097246SAdrian Prantl // a static archive (.a file). Try and see if we have a cached archive 7305097246SAdrian Prantl // first without reading any data first 74b9c1b51eSKate Stone if (file_exists && module_sp->GetObjectName()) { 75d97e9f1aSJonas Devlieghere ObjectFileSP object_file_sp = CreateObjectFromContainer( 76d97e9f1aSJonas Devlieghere module_sp, file, file_offset, file_size, data_sp, data_offset); 77d97e9f1aSJonas Devlieghere if (object_file_sp) 785ce9c565SGreg Clayton return object_file_sp; 795ce9c565SGreg Clayton } 8005097246SAdrian Prantl // Ok, we didn't find any containers that have a named object, now lets 8105097246SAdrian Prantl // read the first 512 bytes from the file so the object file and object 8205097246SAdrian Prantl // container plug-ins can use these bytes to see if they can parse this 8305097246SAdrian Prantl // file. 84b9c1b51eSKate Stone if (file_size > 0) { 85*fc54427eSJonas Devlieghere data_sp = FileSystem::Instance().CreateDataBuffer( 86*fc54427eSJonas Devlieghere file->GetPath(), g_initial_bytes_to_read, file_offset); 875ce9c565SGreg Clayton data_offset = 0; 885ce9c565SGreg Clayton } 8944435ed0SGreg Clayton } 9030fdc8d8SChris Lattner 91b9c1b51eSKate Stone if (!data_sp || data_sp->GetByteSize() == 0) { 9230fdc8d8SChris Lattner // Check for archive file with format "/path/to/archive.a(object.o)" 93ea637750SAdrian Prantl llvm::SmallString<256> path_with_object; 94ea637750SAdrian Prantl module_sp->GetFileSpec().GetPath(path_with_object); 9530fdc8d8SChris Lattner 96d97e9f1aSJonas Devlieghere FileSpec archive_file; 971f746071SGreg Clayton ConstString archive_object; 98906ba471SGreg Clayton const bool must_exist = true; 99d97e9f1aSJonas Devlieghere if (ObjectFile::SplitArchivePathWithObject(path_with_object, archive_file, 100d97e9f1aSJonas Devlieghere archive_object, must_exist)) { 10159b78bcbSJonas Devlieghere file_size = FileSystem::Instance().GetByteSize(archive_file); 102b9c1b51eSKate Stone if (file_size > 0) { 1035ce9c565SGreg Clayton file = &archive_file; 1041f746071SGreg Clayton module_sp->SetFileSpecAndObjectName(archive_file, archive_object); 105b9c1b51eSKate Stone // Check if this is a object container by iterating through all 10605097246SAdrian Prantl // object container plugin instances and then trying to get an 10705097246SAdrian Prantl // object file from the container plugins since we had a name. 10805097246SAdrian Prantl // Also, don't read 1095ce9c565SGreg Clayton // ANY data in case there is data cached in the container plug-ins 11005097246SAdrian Prantl // (like BSD archives caching the contained objects within an 11105097246SAdrian Prantl // file). 112d97e9f1aSJonas Devlieghere ObjectFileSP object_file_sp = CreateObjectFromContainer( 113d97e9f1aSJonas Devlieghere module_sp, file, file_offset, file_size, data_sp, data_offset); 114d97e9f1aSJonas Devlieghere if (object_file_sp) 1155ce9c565SGreg Clayton return object_file_sp; 11605097246SAdrian Prantl // We failed to find any cached object files in the container plug- 11705097246SAdrian Prantl // ins, so lets read the first 512 bytes and try again below... 11887e403aaSJonas Devlieghere data_sp = FileSystem::Instance().CreateDataBuffer( 119*fc54427eSJonas Devlieghere archive_file.GetPath(), g_initial_bytes_to_read, file_offset); 1205ce9c565SGreg Clayton } 1215ce9c565SGreg Clayton } 1225ce9c565SGreg Clayton } 1235ce9c565SGreg Clayton 124b9c1b51eSKate Stone if (data_sp && data_sp->GetByteSize() > 0) { 12505097246SAdrian Prantl // Check if this is a normal object file by iterating through all 12605097246SAdrian Prantl // object file plugin instances. 127d97e9f1aSJonas Devlieghere ObjectFileCreateInstance callback; 128b9c1b51eSKate Stone for (uint32_t idx = 0; 129d97e9f1aSJonas Devlieghere (callback = PluginManager::GetObjectFileCreateCallbackAtIndex(idx)) != 130b9c1b51eSKate Stone nullptr; 131b9c1b51eSKate Stone ++idx) { 132d97e9f1aSJonas Devlieghere ObjectFileSP object_file_sp(callback(module_sp, data_sp, data_offset, 133d97e9f1aSJonas Devlieghere file, file_offset, file_size)); 134762f7135SGreg Clayton if (object_file_sp.get()) 135762f7135SGreg Clayton return object_file_sp; 13630fdc8d8SChris Lattner } 13730fdc8d8SChris Lattner 13805097246SAdrian Prantl // Check if this is a object container by iterating through all object 13905097246SAdrian Prantl // container plugin instances and then trying to get an object file 14005097246SAdrian Prantl // from the container. 141d97e9f1aSJonas Devlieghere ObjectFileSP object_file_sp = CreateObjectFromContainer( 142d97e9f1aSJonas Devlieghere module_sp, file, file_offset, file_size, data_sp, data_offset); 143d97e9f1aSJonas Devlieghere if (object_file_sp) 144762f7135SGreg Clayton return object_file_sp; 14530fdc8d8SChris Lattner } 146d97e9f1aSJonas Devlieghere 14705097246SAdrian Prantl // We didn't find it, so clear our shared pointer in case it contains 14805097246SAdrian Prantl // anything and return an empty shared pointer 149d97e9f1aSJonas Devlieghere return {}; 150762f7135SGreg Clayton } 151762f7135SGreg Clayton 152b9c1b51eSKate Stone ObjectFileSP ObjectFile::FindPlugin(const lldb::ModuleSP &module_sp, 153c9660546SGreg Clayton const ProcessSP &process_sp, 154c9660546SGreg Clayton lldb::addr_t header_addr, 155b9c1b51eSKate Stone DataBufferSP &data_sp) { 156c9660546SGreg Clayton ObjectFileSP object_file_sp; 157c9660546SGreg Clayton 158b9c1b51eSKate Stone if (module_sp) { 1595c1c8443SJonas Devlieghere LLDB_SCOPED_TIMERF("ObjectFile::FindPlugin (module = " 160b9c1b51eSKate Stone "%s, process = %p, header_addr = " 161b9c1b51eSKate Stone "0x%" PRIx64 ")", 162b5ad4ec7SGreg Clayton module_sp->GetFileSpec().GetPath().c_str(), 163324a1036SSaleem Abdulrasool static_cast<void *>(process_sp.get()), header_addr); 164c9660546SGreg Clayton uint32_t idx; 165c9660546SGreg Clayton 16605097246SAdrian Prantl // Check if this is a normal object file by iterating through all object 16705097246SAdrian Prantl // file plugin instances. 168c9660546SGreg Clayton ObjectFileCreateMemoryInstance create_callback; 169b9c1b51eSKate Stone for (idx = 0; 170b9c1b51eSKate Stone (create_callback = 171b9c1b51eSKate Stone PluginManager::GetObjectFileCreateMemoryCallbackAtIndex(idx)) != 172b9c1b51eSKate Stone nullptr; 173b9c1b51eSKate Stone ++idx) { 174b9c1b51eSKate Stone object_file_sp.reset( 175b9c1b51eSKate Stone create_callback(module_sp, data_sp, process_sp, header_addr)); 176c9660546SGreg Clayton if (object_file_sp.get()) 177c9660546SGreg Clayton return object_file_sp; 178c9660546SGreg Clayton } 179c9660546SGreg Clayton } 180324a1036SSaleem Abdulrasool 18105097246SAdrian Prantl // We didn't find it, so clear our shared pointer in case it contains 18205097246SAdrian Prantl // anything and return an empty shared pointer 183c9660546SGreg Clayton object_file_sp.reset(); 184c9660546SGreg Clayton return object_file_sp; 185c9660546SGreg Clayton } 186c9660546SGreg Clayton 187b9c1b51eSKate Stone size_t ObjectFile::GetModuleSpecifications(const FileSpec &file, 188f4d6de6aSGreg Clayton lldb::offset_t file_offset, 1892540a8a7SGreg Clayton lldb::offset_t file_size, 190a4a00cedSFred Riss ModuleSpecList &specs, 191a4a00cedSFred Riss DataBufferSP data_sp) { 192a4a00cedSFred Riss if (!data_sp) 193*fc54427eSJonas Devlieghere data_sp = FileSystem::Instance().CreateDataBuffer( 194*fc54427eSJonas Devlieghere file.GetPath(), g_initial_bytes_to_read, file_offset); 195b9c1b51eSKate Stone if (data_sp) { 196b9c1b51eSKate Stone if (file_size == 0) { 19759b78bcbSJonas Devlieghere const lldb::offset_t actual_file_size = 19859b78bcbSJonas Devlieghere FileSystem::Instance().GetByteSize(file); 1992540a8a7SGreg Clayton if (actual_file_size > file_offset) 2002540a8a7SGreg Clayton file_size = actual_file_size - file_offset; 2012540a8a7SGreg Clayton } 202f4d6de6aSGreg Clayton return ObjectFile::GetModuleSpecifications(file, // file spec 203f4d6de6aSGreg Clayton data_sp, // data bytes 204f4d6de6aSGreg Clayton 0, // data offset 205f4d6de6aSGreg Clayton file_offset, // file offset 2062540a8a7SGreg Clayton file_size, // file length 207f4d6de6aSGreg Clayton specs); 2082540a8a7SGreg Clayton } 209f4d6de6aSGreg Clayton return 0; 210f4d6de6aSGreg Clayton } 211f4d6de6aSGreg Clayton 212b9c1b51eSKate Stone size_t ObjectFile::GetModuleSpecifications( 213b9c1b51eSKate Stone const lldb_private::FileSpec &file, lldb::DataBufferSP &data_sp, 214b9c1b51eSKate Stone lldb::offset_t data_offset, lldb::offset_t file_offset, 215b9c1b51eSKate Stone lldb::offset_t file_size, lldb_private::ModuleSpecList &specs) { 216f4d6de6aSGreg Clayton const size_t initial_count = specs.GetSize(); 217f4d6de6aSGreg Clayton ObjectFileGetModuleSpecifications callback; 218f4d6de6aSGreg Clayton uint32_t i; 219f4d6de6aSGreg Clayton // Try the ObjectFile plug-ins 220b9c1b51eSKate Stone for (i = 0; 221b9c1b51eSKate Stone (callback = 222b9c1b51eSKate Stone PluginManager::GetObjectFileGetModuleSpecificationsCallbackAtIndex( 223b9c1b51eSKate Stone i)) != nullptr; 224b9c1b51eSKate Stone ++i) { 2252540a8a7SGreg Clayton if (callback(file, data_sp, data_offset, file_offset, file_size, specs) > 0) 226f4d6de6aSGreg Clayton return specs.GetSize() - initial_count; 227f4d6de6aSGreg Clayton } 228f4d6de6aSGreg Clayton 229f4d6de6aSGreg Clayton // Try the ObjectContainer plug-ins 230b9c1b51eSKate Stone for (i = 0; 231b9c1b51eSKate Stone (callback = PluginManager:: 232b9c1b51eSKate Stone GetObjectContainerGetModuleSpecificationsCallbackAtIndex(i)) != 233b9c1b51eSKate Stone nullptr; 234b9c1b51eSKate Stone ++i) { 2352540a8a7SGreg Clayton if (callback(file, data_sp, data_offset, file_offset, file_size, specs) > 0) 236f4d6de6aSGreg Clayton return specs.GetSize() - initial_count; 237f4d6de6aSGreg Clayton } 238f4d6de6aSGreg Clayton return 0; 239f4d6de6aSGreg Clayton } 240f4d6de6aSGreg Clayton 241e72dfb32SGreg Clayton ObjectFile::ObjectFile(const lldb::ModuleSP &module_sp, 242762f7135SGreg Clayton const FileSpec *file_spec_ptr, 243b9c1b51eSKate Stone lldb::offset_t file_offset, lldb::offset_t length, 24423f8c95aSGreg Clayton const lldb::DataBufferSP &data_sp, 245b9c1b51eSKate Stone lldb::offset_t data_offset) 246b9c1b51eSKate Stone : ModuleChild(module_sp), 247762f7135SGreg Clayton m_file(), // This file could be different from the original module's file 248b9c1b51eSKate Stone m_type(eTypeInvalid), m_strata(eStrataInvalid), 24966d88326SPavel Labath m_file_offset(file_offset), m_length(length), m_data(), m_process_wp(), 250d5b44036SJonas Devlieghere m_memory_addr(LLDB_INVALID_ADDRESS), m_sections_up(), m_symtab_up(), 2517e6df41fSGreg Clayton m_symtab_once_up(new llvm::once_flag()) { 252762f7135SGreg Clayton if (file_spec_ptr) 253762f7135SGreg Clayton m_file = *file_spec_ptr; 2545ce9c565SGreg Clayton if (data_sp) 2555ce9c565SGreg Clayton m_data.SetData(data_sp, data_offset, length); 256a007a6d8SPavel Labath Log *log = GetLog(LLDBLog::Object); 25763e5fb76SJonas Devlieghere LLDB_LOGF(log, 25863e5fb76SJonas Devlieghere "%p ObjectFile::ObjectFile() module = %p (%s), file = %s, " 259b9c1b51eSKate Stone "file_offset = 0x%8.8" PRIx64 ", size = %" PRIu64, 260b9c1b51eSKate Stone static_cast<void *>(this), static_cast<void *>(module_sp.get()), 261b5ad4ec7SGreg Clayton module_sp->GetSpecificationDescription().c_str(), 262b9c1b51eSKate Stone m_file ? m_file.GetPath().c_str() : "<NULL>", m_file_offset, 263b9c1b51eSKate Stone m_length); 264762f7135SGreg Clayton } 265762f7135SGreg Clayton 266e72dfb32SGreg Clayton ObjectFile::ObjectFile(const lldb::ModuleSP &module_sp, 267b9c1b51eSKate Stone const ProcessSP &process_sp, lldb::addr_t header_addr, 268b9c1b51eSKate Stone DataBufferSP &header_data_sp) 269b9c1b51eSKate Stone : ModuleChild(module_sp), m_file(), m_type(eTypeInvalid), 270b9c1b51eSKate Stone m_strata(eStrataInvalid), m_file_offset(0), m_length(0), m_data(), 27166d88326SPavel Labath m_process_wp(process_sp), m_memory_addr(header_addr), m_sections_up(), 2727e6df41fSGreg Clayton m_symtab_up(), m_symtab_once_up(new llvm::once_flag()) { 273c9660546SGreg Clayton if (header_data_sp) 274c9660546SGreg Clayton m_data.SetData(header_data_sp, 0, header_data_sp->GetByteSize()); 275a007a6d8SPavel Labath Log *log = GetLog(LLDBLog::Object); 27663e5fb76SJonas Devlieghere LLDB_LOGF(log, 27763e5fb76SJonas Devlieghere "%p ObjectFile::ObjectFile() module = %p (%s), process = %p, " 278b9c1b51eSKate Stone "header_addr = 0x%" PRIx64, 279b9c1b51eSKate Stone static_cast<void *>(this), static_cast<void *>(module_sp.get()), 280b5ad4ec7SGreg Clayton module_sp->GetSpecificationDescription().c_str(), 281324a1036SSaleem Abdulrasool static_cast<void *>(process_sp.get()), m_memory_addr); 282c9660546SGreg Clayton } 283c9660546SGreg Clayton 284b9c1b51eSKate Stone ObjectFile::~ObjectFile() { 285a007a6d8SPavel Labath Log *log = GetLog(LLDBLog::Object); 28663e5fb76SJonas Devlieghere LLDB_LOGF(log, "%p ObjectFile::~ObjectFile ()\n", static_cast<void *>(this)); 28730fdc8d8SChris Lattner } 2885aee162fSJim Ingham 289b9c1b51eSKate Stone bool ObjectFile::SetModulesArchitecture(const ArchSpec &new_arch) { 290e72dfb32SGreg Clayton ModuleSP module_sp(GetModule()); 291e72dfb32SGreg Clayton if (module_sp) 292e72dfb32SGreg Clayton return module_sp->SetArchitecture(new_arch); 293e72dfb32SGreg Clayton return false; 2945aee162fSJim Ingham } 2955aee162fSJim Ingham 296b9c1b51eSKate Stone AddressClass ObjectFile::GetAddressClass(addr_t file_addr) { 2973046e668SGreg Clayton Symtab *symtab = GetSymtab(); 298b9c1b51eSKate Stone if (symtab) { 299ded470d3SGreg Clayton Symbol *symbol = symtab->FindSymbolContainingFileAddress(file_addr); 300b9c1b51eSKate Stone if (symbol) { 301b9c1b51eSKate Stone if (symbol->ValueIsAddress()) { 302358cf1eaSGreg Clayton const SectionSP section_sp(symbol->GetAddressRef().GetSection()); 303b9c1b51eSKate Stone if (section_sp) { 304e72dfb32SGreg Clayton const SectionType section_type = section_sp->GetType(); 305b9c1b51eSKate Stone switch (section_type) { 306324a1036SSaleem Abdulrasool case eSectionTypeInvalid: 30704803b3eSTatyana Krasnukha return AddressClass::eUnknown; 308324a1036SSaleem Abdulrasool case eSectionTypeCode: 30904803b3eSTatyana Krasnukha return AddressClass::eCode; 310324a1036SSaleem Abdulrasool case eSectionTypeContainer: 31104803b3eSTatyana Krasnukha return AddressClass::eUnknown; 3125009f9d5SGreg Clayton case eSectionTypeData: 3135009f9d5SGreg Clayton case eSectionTypeDataCString: 3145009f9d5SGreg Clayton case eSectionTypeDataCStringPointers: 3155009f9d5SGreg Clayton case eSectionTypeDataSymbolAddress: 3165009f9d5SGreg Clayton case eSectionTypeData4: 3175009f9d5SGreg Clayton case eSectionTypeData8: 3185009f9d5SGreg Clayton case eSectionTypeData16: 3195009f9d5SGreg Clayton case eSectionTypeDataPointers: 3205009f9d5SGreg Clayton case eSectionTypeZeroFill: 3215009f9d5SGreg Clayton case eSectionTypeDataObjCMessageRefs: 3225009f9d5SGreg Clayton case eSectionTypeDataObjCCFStrings: 32365d4d5c3SRyan Brown case eSectionTypeGoSymtab: 32404803b3eSTatyana Krasnukha return AddressClass::eData; 3255009f9d5SGreg Clayton case eSectionTypeDebug: 3265009f9d5SGreg Clayton case eSectionTypeDWARFDebugAbbrev: 327004bcb78SGeorge Rimar case eSectionTypeDWARFDebugAbbrevDwo: 328c178d4c0STamas Berghammer case eSectionTypeDWARFDebugAddr: 3295009f9d5SGreg Clayton case eSectionTypeDWARFDebugAranges: 330963ce483STamas Berghammer case eSectionTypeDWARFDebugCuIndex: 3315009f9d5SGreg Clayton case eSectionTypeDWARFDebugFrame: 3325009f9d5SGreg Clayton case eSectionTypeDWARFDebugInfo: 333004bcb78SGeorge Rimar case eSectionTypeDWARFDebugInfoDwo: 3345009f9d5SGreg Clayton case eSectionTypeDWARFDebugLine: 335c6c7bfc4SGeorge Rimar case eSectionTypeDWARFDebugLineStr: 3365009f9d5SGreg Clayton case eSectionTypeDWARFDebugLoc: 3374b5bc388SPavel Labath case eSectionTypeDWARFDebugLocDwo: 338e4dee269SGeorge Rimar case eSectionTypeDWARFDebugLocLists: 3394b5bc388SPavel Labath case eSectionTypeDWARFDebugLocListsDwo: 3405009f9d5SGreg Clayton case eSectionTypeDWARFDebugMacInfo: 341d8335e9aSSiva Chandra case eSectionTypeDWARFDebugMacro: 342a041d848SPavel Labath case eSectionTypeDWARFDebugNames: 3435009f9d5SGreg Clayton case eSectionTypeDWARFDebugPubNames: 3445009f9d5SGreg Clayton case eSectionTypeDWARFDebugPubTypes: 3455009f9d5SGreg Clayton case eSectionTypeDWARFDebugRanges: 3466e357123SGeorge Rimar case eSectionTypeDWARFDebugRngLists: 3474023bd05SPavel Labath case eSectionTypeDWARFDebugRngListsDwo: 3485009f9d5SGreg Clayton case eSectionTypeDWARFDebugStr: 349004bcb78SGeorge Rimar case eSectionTypeDWARFDebugStrDwo: 350c178d4c0STamas Berghammer case eSectionTypeDWARFDebugStrOffsets: 351004bcb78SGeorge Rimar case eSectionTypeDWARFDebugStrOffsetsDwo: 3527b59ff2fSPavel Labath case eSectionTypeDWARFDebugTuIndex: 3532550ca1eSGreg Clayton case eSectionTypeDWARFDebugTypes: 354ad805ef9SPavel Labath case eSectionTypeDWARFDebugTypesDwo: 3555009f9d5SGreg Clayton case eSectionTypeDWARFAppleNames: 3565009f9d5SGreg Clayton case eSectionTypeDWARFAppleTypes: 3575009f9d5SGreg Clayton case eSectionTypeDWARFAppleNamespaces: 3585009f9d5SGreg Clayton case eSectionTypeDWARFAppleObjC: 359e4777a9dSJan Kratochvil case eSectionTypeDWARFGNUDebugAltLink: 36004803b3eSTatyana Krasnukha return AddressClass::eDebug; 361324a1036SSaleem Abdulrasool case eSectionTypeEHFrame: 362648f3c7eSTamas Berghammer case eSectionTypeARMexidx: 363648f3c7eSTamas Berghammer case eSectionTypeARMextab: 364e589e7e3SJason Molenda case eSectionTypeCompactUnwind: 36504803b3eSTatyana Krasnukha return AddressClass::eRuntime; 366a7499c98SMichael Sartain case eSectionTypeELFSymbolTable: 367a7499c98SMichael Sartain case eSectionTypeELFDynamicSymbols: 368a7499c98SMichael Sartain case eSectionTypeELFRelocationEntries: 369a7499c98SMichael Sartain case eSectionTypeELFDynamicLinkInfo: 370324a1036SSaleem Abdulrasool case eSectionTypeOther: 37104803b3eSTatyana Krasnukha return AddressClass::eUnknown; 37231d315b3STamas Berghammer case eSectionTypeAbsoluteAddress: 373b9c1b51eSKate Stone // In case of absolute sections decide the address class based on 37405097246SAdrian Prantl // the symbol type because the section type isn't specify if it is 37505097246SAdrian Prantl // a code or a data section. 37631d315b3STamas Berghammer break; 377ded470d3SGreg Clayton } 378ded470d3SGreg Clayton } 379ded470d3SGreg Clayton } 380ded470d3SGreg Clayton 381e0d378b3SGreg Clayton const SymbolType symbol_type = symbol->GetType(); 382b9c1b51eSKate Stone switch (symbol_type) { 383b9c1b51eSKate Stone case eSymbolTypeAny: 38404803b3eSTatyana Krasnukha return AddressClass::eUnknown; 385b9c1b51eSKate Stone case eSymbolTypeAbsolute: 38604803b3eSTatyana Krasnukha return AddressClass::eUnknown; 387b9c1b51eSKate Stone case eSymbolTypeCode: 38804803b3eSTatyana Krasnukha return AddressClass::eCode; 389b9c1b51eSKate Stone case eSymbolTypeTrampoline: 39004803b3eSTatyana Krasnukha return AddressClass::eCode; 391b9c1b51eSKate Stone case eSymbolTypeResolver: 39204803b3eSTatyana Krasnukha return AddressClass::eCode; 393b9c1b51eSKate Stone case eSymbolTypeData: 39404803b3eSTatyana Krasnukha return AddressClass::eData; 395b9c1b51eSKate Stone case eSymbolTypeRuntime: 39604803b3eSTatyana Krasnukha return AddressClass::eRuntime; 397b9c1b51eSKate Stone case eSymbolTypeException: 39804803b3eSTatyana Krasnukha return AddressClass::eRuntime; 399b9c1b51eSKate Stone case eSymbolTypeSourceFile: 40004803b3eSTatyana Krasnukha return AddressClass::eDebug; 401b9c1b51eSKate Stone case eSymbolTypeHeaderFile: 40204803b3eSTatyana Krasnukha return AddressClass::eDebug; 403b9c1b51eSKate Stone case eSymbolTypeObjectFile: 40404803b3eSTatyana Krasnukha return AddressClass::eDebug; 405b9c1b51eSKate Stone case eSymbolTypeCommonBlock: 40604803b3eSTatyana Krasnukha return AddressClass::eDebug; 407b9c1b51eSKate Stone case eSymbolTypeBlock: 40804803b3eSTatyana Krasnukha return AddressClass::eDebug; 409b9c1b51eSKate Stone case eSymbolTypeLocal: 41004803b3eSTatyana Krasnukha return AddressClass::eData; 411b9c1b51eSKate Stone case eSymbolTypeParam: 41204803b3eSTatyana Krasnukha return AddressClass::eData; 413b9c1b51eSKate Stone case eSymbolTypeVariable: 41404803b3eSTatyana Krasnukha return AddressClass::eData; 415b9c1b51eSKate Stone case eSymbolTypeVariableType: 41604803b3eSTatyana Krasnukha return AddressClass::eDebug; 417b9c1b51eSKate Stone case eSymbolTypeLineEntry: 41804803b3eSTatyana Krasnukha return AddressClass::eDebug; 419b9c1b51eSKate Stone case eSymbolTypeLineHeader: 42004803b3eSTatyana Krasnukha return AddressClass::eDebug; 421b9c1b51eSKate Stone case eSymbolTypeScopeBegin: 42204803b3eSTatyana Krasnukha return AddressClass::eDebug; 423b9c1b51eSKate Stone case eSymbolTypeScopeEnd: 42404803b3eSTatyana Krasnukha return AddressClass::eDebug; 425b9c1b51eSKate Stone case eSymbolTypeAdditional: 42604803b3eSTatyana Krasnukha return AddressClass::eUnknown; 427b9c1b51eSKate Stone case eSymbolTypeCompiler: 42804803b3eSTatyana Krasnukha return AddressClass::eDebug; 429b9c1b51eSKate Stone case eSymbolTypeInstrumentation: 43004803b3eSTatyana Krasnukha return AddressClass::eDebug; 431b9c1b51eSKate Stone case eSymbolTypeUndefined: 43204803b3eSTatyana Krasnukha return AddressClass::eUnknown; 433b9c1b51eSKate Stone case eSymbolTypeObjCClass: 43404803b3eSTatyana Krasnukha return AddressClass::eRuntime; 435b9c1b51eSKate Stone case eSymbolTypeObjCMetaClass: 43604803b3eSTatyana Krasnukha return AddressClass::eRuntime; 437b9c1b51eSKate Stone case eSymbolTypeObjCIVar: 43804803b3eSTatyana Krasnukha return AddressClass::eRuntime; 439b9c1b51eSKate Stone case eSymbolTypeReExported: 44004803b3eSTatyana Krasnukha return AddressClass::eRuntime; 441ded470d3SGreg Clayton } 442ded470d3SGreg Clayton } 443ded470d3SGreg Clayton } 44404803b3eSTatyana Krasnukha return AddressClass::eUnknown; 445ded470d3SGreg Clayton } 446ded470d3SGreg Clayton 447b9c1b51eSKate Stone DataBufferSP ObjectFile::ReadMemory(const ProcessSP &process_sp, 448b9c1b51eSKate Stone lldb::addr_t addr, size_t byte_size) { 449c9660546SGreg Clayton DataBufferSP data_sp; 450b9c1b51eSKate Stone if (process_sp) { 451d5b44036SJonas Devlieghere std::unique_ptr<DataBufferHeap> data_up(new DataBufferHeap(byte_size, 0)); 45297206d57SZachary Turner Status error; 453b9c1b51eSKate Stone const size_t bytes_read = process_sp->ReadMemory( 454d5b44036SJonas Devlieghere addr, data_up->GetBytes(), data_up->GetByteSize(), error); 455c9660546SGreg Clayton if (bytes_read == byte_size) 456d5b44036SJonas Devlieghere data_sp.reset(data_up.release()); 457c9660546SGreg Clayton } 458c9660546SGreg Clayton return data_sp; 459c9660546SGreg Clayton } 460c9660546SGreg Clayton 461b9c1b51eSKate Stone size_t ObjectFile::GetData(lldb::offset_t offset, size_t length, 462b9c1b51eSKate Stone DataExtractor &data) const { 463b9c1b51eSKate Stone // The entire file has already been mmap'ed into m_data, so just copy from 46405097246SAdrian Prantl // there as the back mmap buffer will be shared with shared pointers. 465a0f72441SMartin Storsjö return data.SetData(m_data, offset, length); 46644435ed0SGreg Clayton } 46744435ed0SGreg Clayton 468b9c1b51eSKate Stone size_t ObjectFile::CopyData(lldb::offset_t offset, size_t length, 469b9c1b51eSKate Stone void *dst) const { 470b9c1b51eSKate Stone // The entire file has already been mmap'ed into m_data, so just copy from 47105097246SAdrian Prantl // there Note that the data remains in target byte order. 472b0e33d41SEd Maste return m_data.CopyData(offset, length, dst); 47344435ed0SGreg Clayton } 474ded470d3SGreg Clayton 475d13f691fSEd Maste size_t ObjectFile::ReadSectionData(Section *section, 476b9c1b51eSKate Stone lldb::offset_t section_offset, void *dst, 477d13f691fSEd Maste size_t dst_len) { 478f03e6d84SMatthew Gardiner assert(section); 479f03e6d84SMatthew Gardiner section_offset *= section->GetTargetByteSize(); 480f03e6d84SMatthew Gardiner 481a7499c98SMichael Sartain // If some other objectfile owns this data, pass this to them. 482a7499c98SMichael Sartain if (section->GetObjectFile() != this) 483b9c1b51eSKate Stone return section->GetObjectFile()->ReadSectionData(section, section_offset, 484b9c1b51eSKate Stone dst, dst_len); 485a7499c98SMichael Sartain 486203b4774SStefan Gränitz if (!section->IsRelocated()) 487203b4774SStefan Gränitz RelocateSection(section); 488203b4774SStefan Gränitz 489b9c1b51eSKate Stone if (IsInMemory()) { 490c9660546SGreg Clayton ProcessSP process_sp(m_process_wp.lock()); 491b9c1b51eSKate Stone if (process_sp) { 49297206d57SZachary Turner Status error; 493b9c1b51eSKate Stone const addr_t base_load_addr = 494b9c1b51eSKate Stone section->GetLoadBaseAddress(&process_sp->GetTarget()); 49539f7ee86SGreg Clayton if (base_load_addr != LLDB_INVALID_ADDRESS) 496b9c1b51eSKate Stone return process_sp->ReadMemory(base_load_addr + section_offset, dst, 497b9c1b51eSKate Stone dst_len, error); 498c9660546SGreg Clayton } 499b9c1b51eSKate Stone } else { 500a746e8e5SZachary Turner const lldb::offset_t section_file_size = section->GetFileSize(); 501b9c1b51eSKate Stone if (section_offset < section_file_size) { 502a746e8e5SZachary Turner const size_t section_bytes_left = section_file_size - section_offset; 503a746e8e5SZachary Turner size_t section_dst_len = dst_len; 504ee212e2cSGreg Clayton if (section_dst_len > section_bytes_left) 505ee212e2cSGreg Clayton section_dst_len = section_bytes_left; 506b9c1b51eSKate Stone return CopyData(section->GetFileOffset() + section_offset, 507b9c1b51eSKate Stone section_dst_len, dst); 508b9c1b51eSKate Stone } else { 509b9c1b51eSKate Stone if (section->GetType() == eSectionTypeZeroFill) { 510ecda2b2dSSean Callanan const uint64_t section_size = section->GetByteSize(); 511ecda2b2dSSean Callanan const uint64_t section_bytes_left = section_size - section_offset; 512ecda2b2dSSean Callanan uint64_t section_dst_len = dst_len; 513ecda2b2dSSean Callanan if (section_dst_len > section_bytes_left) 514ecda2b2dSSean Callanan section_dst_len = section_bytes_left; 515b2f1fb29SVirgile Bello memset(dst, 0, section_dst_len); 516ecda2b2dSSean Callanan return section_dst_len; 517ecda2b2dSSean Callanan } 518ecda2b2dSSean Callanan } 519c9660546SGreg Clayton } 520c9660546SGreg Clayton return 0; 521c9660546SGreg Clayton } 522c9660546SGreg Clayton 523c9660546SGreg Clayton // Get the section data the file on disk 524d13f691fSEd Maste size_t ObjectFile::ReadSectionData(Section *section, 525d13f691fSEd Maste DataExtractor §ion_data) { 526a7499c98SMichael Sartain // If some other objectfile owns this data, pass this to them. 527a7499c98SMichael Sartain if (section->GetObjectFile() != this) 528a7499c98SMichael Sartain return section->GetObjectFile()->ReadSectionData(section, section_data); 529a7499c98SMichael Sartain 530203b4774SStefan Gränitz if (!section->IsRelocated()) 531203b4774SStefan Gränitz RelocateSection(section); 532203b4774SStefan Gränitz 533b9c1b51eSKate Stone if (IsInMemory()) { 534c9660546SGreg Clayton ProcessSP process_sp(m_process_wp.lock()); 535b9c1b51eSKate Stone if (process_sp) { 536b9c1b51eSKate Stone const addr_t base_load_addr = 537b9c1b51eSKate Stone section->GetLoadBaseAddress(&process_sp->GetTarget()); 538b9c1b51eSKate Stone if (base_load_addr != LLDB_INVALID_ADDRESS) { 539b9c1b51eSKate Stone DataBufferSP data_sp( 540b9c1b51eSKate Stone ReadMemory(process_sp, base_load_addr, section->GetByteSize())); 541b9c1b51eSKate Stone if (data_sp) { 542c9660546SGreg Clayton section_data.SetData(data_sp, 0, data_sp->GetByteSize()); 543c9660546SGreg Clayton section_data.SetByteOrder(process_sp->GetByteOrder()); 544c9660546SGreg Clayton section_data.SetAddressByteSize(process_sp->GetAddressByteSize()); 545c9660546SGreg Clayton return section_data.GetByteSize(); 546c9660546SGreg Clayton } 547c9660546SGreg Clayton } 548c9660546SGreg Clayton } 549203b4774SStefan Gränitz } 550203b4774SStefan Gränitz 55105097246SAdrian Prantl // The object file now contains a full mmap'ed copy of the object file 55205097246SAdrian Prantl // data, so just use this 553b9c1b51eSKate Stone return GetData(section->GetFileOffset(), section->GetFileSize(), 554b9c1b51eSKate Stone section_data); 555c9660546SGreg Clayton } 556c9660546SGreg Clayton 557ea637750SAdrian Prantl bool ObjectFile::SplitArchivePathWithObject(llvm::StringRef path_with_object, 558b9c1b51eSKate Stone FileSpec &archive_file, 559b9c1b51eSKate Stone ConstString &archive_object, 560b9c1b51eSKate Stone bool must_exist) { 561ea637750SAdrian Prantl size_t len = path_with_object.size(); 562ea637750SAdrian Prantl if (len < 2 || path_with_object.back() != ')') 5631f746071SGreg Clayton return false; 564ea637750SAdrian Prantl llvm::StringRef archive = path_with_object.substr(0, path_with_object.rfind('(')); 565ea637750SAdrian Prantl if (archive.empty()) 566ea637750SAdrian Prantl return false; 567ea637750SAdrian Prantl llvm::StringRef object = path_with_object.substr(archive.size() + 1).drop_back(); 568ea637750SAdrian Prantl archive_file.SetFile(archive, FileSpec::Style::native); 569ea637750SAdrian Prantl if (must_exist && !FileSystem::Instance().Exists(archive_file)) 570ea637750SAdrian Prantl return false; 571ea637750SAdrian Prantl archive_object.SetString(object); 572ea637750SAdrian Prantl return true; 5731f746071SGreg Clayton } 5741f746071SGreg Clayton 575b9c1b51eSKate Stone void ObjectFile::ClearSymtab() { 5769422dd64SGreg Clayton ModuleSP module_sp(GetModule()); 577b9c1b51eSKate Stone if (module_sp) { 578a007a6d8SPavel Labath Log *log = GetLog(LLDBLog::Object); 57963e5fb76SJonas Devlieghere LLDB_LOGF(log, "%p ObjectFile::ClearSymtab () symtab = %p", 580324a1036SSaleem Abdulrasool static_cast<void *>(this), 581d5b44036SJonas Devlieghere static_cast<void *>(m_symtab_up.get())); 5827e6df41fSGreg Clayton // Since we need to clear the symbol table, we need a new llvm::once_flag 5837e6df41fSGreg Clayton // instance so we can safely create another symbol table 5847e6df41fSGreg Clayton m_symtab_once_up.reset(new llvm::once_flag()); 585d5b44036SJonas Devlieghere m_symtab_up.reset(); 5869422dd64SGreg Clayton } 5879422dd64SGreg Clayton } 5883046e668SGreg Clayton 589b9c1b51eSKate Stone SectionList *ObjectFile::GetSectionList(bool update_module_section_list) { 590d5b44036SJonas Devlieghere if (m_sections_up == nullptr) { 591b9c1b51eSKate Stone if (update_module_section_list) { 5923046e668SGreg Clayton ModuleSP module_sp(GetModule()); 593b9c1b51eSKate Stone if (module_sp) { 59416ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(module_sp->GetMutex()); 5953046e668SGreg Clayton CreateSections(*module_sp->GetUnifiedSectionList()); 5963046e668SGreg Clayton } 597b9c1b51eSKate Stone } else { 598eb882fc1STamas Berghammer SectionList unified_section_list; 599eb882fc1STamas Berghammer CreateSections(unified_section_list); 600eb882fc1STamas Berghammer } 601eb882fc1STamas Berghammer } 602d5b44036SJonas Devlieghere return m_sections_up.get(); 603a7499c98SMichael Sartain } 604649a607eSJason Molenda 605649a607eSJason Molenda lldb::SymbolType 606649a607eSJason Molenda ObjectFile::GetSymbolTypeFromName(llvm::StringRef name, 607b9c1b51eSKate Stone lldb::SymbolType symbol_type_hint) { 608b9c1b51eSKate Stone if (!name.empty()) { 609b9c1b51eSKate Stone if (name.startswith("_OBJC_")) { 610649a607eSJason Molenda // ObjC 611649a607eSJason Molenda if (name.startswith("_OBJC_CLASS_$_")) 612649a607eSJason Molenda return lldb::eSymbolTypeObjCClass; 613649a607eSJason Molenda if (name.startswith("_OBJC_METACLASS_$_")) 614649a607eSJason Molenda return lldb::eSymbolTypeObjCMetaClass; 615649a607eSJason Molenda if (name.startswith("_OBJC_IVAR_$_")) 616649a607eSJason Molenda return lldb::eSymbolTypeObjCIVar; 617b9c1b51eSKate Stone } else if (name.startswith(".objc_class_name_")) { 618649a607eSJason Molenda // ObjC v1 619649a607eSJason Molenda return lldb::eSymbolTypeObjCClass; 620649a607eSJason Molenda } 621649a607eSJason Molenda } 622649a607eSJason Molenda return symbol_type_hint; 623649a607eSJason Molenda } 6246b63b148STamas Berghammer 62516064d35SPavel Labath std::vector<ObjectFile::LoadableData> 62616064d35SPavel Labath ObjectFile::GetLoadableData(Target &target) { 62716064d35SPavel Labath std::vector<LoadableData> loadables; 6284687db0eSHafiz Abid Qadeer SectionList *section_list = GetSectionList(); 6294687db0eSHafiz Abid Qadeer if (!section_list) 63016064d35SPavel Labath return loadables; 63116064d35SPavel Labath // Create a list of loadable data from loadable sections 6324687db0eSHafiz Abid Qadeer size_t section_count = section_list->GetNumSections(0); 6334687db0eSHafiz Abid Qadeer for (size_t i = 0; i < section_count; ++i) { 63416064d35SPavel Labath LoadableData loadable; 6354687db0eSHafiz Abid Qadeer SectionSP section_sp = section_list->GetSectionAtIndex(i); 63616064d35SPavel Labath loadable.Dest = 63716064d35SPavel Labath target.GetSectionLoadList().GetSectionLoadAddress(section_sp); 63816064d35SPavel Labath if (loadable.Dest == LLDB_INVALID_ADDRESS) 63916064d35SPavel Labath continue; 6404687db0eSHafiz Abid Qadeer // We can skip sections like bss 6414687db0eSHafiz Abid Qadeer if (section_sp->GetFileSize() == 0) 6424687db0eSHafiz Abid Qadeer continue; 64316064d35SPavel Labath DataExtractor section_data; 6444687db0eSHafiz Abid Qadeer section_sp->GetSectionData(section_data); 64516064d35SPavel Labath loadable.Contents = llvm::ArrayRef<uint8_t>(section_data.GetDataStart(), 64616064d35SPavel Labath section_data.GetByteSize()); 64716064d35SPavel Labath loadables.push_back(loadable); 648ec03d7e3SPavel Labath } 64916064d35SPavel Labath return loadables; 6504687db0eSHafiz Abid Qadeer } 651d13f691fSEd Maste 65230c2441aSAleksandr Urakov std::unique_ptr<CallFrameInfo> ObjectFile::CreateCallFrameInfo() { 65330c2441aSAleksandr Urakov return {}; 65430c2441aSAleksandr Urakov } 65530c2441aSAleksandr Urakov 656d13f691fSEd Maste void ObjectFile::RelocateSection(lldb_private::Section *section) 657d13f691fSEd Maste { 658d13f691fSEd Maste } 65950251fc7SPavel Labath 66050251fc7SPavel Labath DataBufferSP ObjectFile::MapFileData(const FileSpec &file, uint64_t Size, 66150251fc7SPavel Labath uint64_t Offset) { 66287e403aaSJonas Devlieghere return FileSystem::Instance().CreateDataBuffer(file.GetPath(), Size, Offset); 66350251fc7SPavel Labath } 6641f6b2477SPavel Labath 6651f6b2477SPavel Labath void llvm::format_provider<ObjectFile::Type>::format( 6661f6b2477SPavel Labath const ObjectFile::Type &type, raw_ostream &OS, StringRef Style) { 6671f6b2477SPavel Labath switch (type) { 6681f6b2477SPavel Labath case ObjectFile::eTypeInvalid: 6691f6b2477SPavel Labath OS << "invalid"; 6701f6b2477SPavel Labath break; 6711f6b2477SPavel Labath case ObjectFile::eTypeCoreFile: 6721f6b2477SPavel Labath OS << "core file"; 6731f6b2477SPavel Labath break; 6741f6b2477SPavel Labath case ObjectFile::eTypeExecutable: 6751f6b2477SPavel Labath OS << "executable"; 6761f6b2477SPavel Labath break; 6771f6b2477SPavel Labath case ObjectFile::eTypeDebugInfo: 6781f6b2477SPavel Labath OS << "debug info"; 6791f6b2477SPavel Labath break; 6801f6b2477SPavel Labath case ObjectFile::eTypeDynamicLinker: 6811f6b2477SPavel Labath OS << "dynamic linker"; 6821f6b2477SPavel Labath break; 6831f6b2477SPavel Labath case ObjectFile::eTypeObjectFile: 6841f6b2477SPavel Labath OS << "object file"; 6851f6b2477SPavel Labath break; 6861f6b2477SPavel Labath case ObjectFile::eTypeSharedLibrary: 6871f6b2477SPavel Labath OS << "shared library"; 6881f6b2477SPavel Labath break; 6891f6b2477SPavel Labath case ObjectFile::eTypeStubLibrary: 6901f6b2477SPavel Labath OS << "stub library"; 6911f6b2477SPavel Labath break; 6921f6b2477SPavel Labath case ObjectFile::eTypeJIT: 6931f6b2477SPavel Labath OS << "jit"; 6941f6b2477SPavel Labath break; 6951f6b2477SPavel Labath case ObjectFile::eTypeUnknown: 6961f6b2477SPavel Labath OS << "unknown"; 6971f6b2477SPavel Labath break; 6981f6b2477SPavel Labath } 6991f6b2477SPavel Labath } 7001f6b2477SPavel Labath 7011f6b2477SPavel Labath void llvm::format_provider<ObjectFile::Strata>::format( 7021f6b2477SPavel Labath const ObjectFile::Strata &strata, raw_ostream &OS, StringRef Style) { 7031f6b2477SPavel Labath switch (strata) { 7041f6b2477SPavel Labath case ObjectFile::eStrataInvalid: 7051f6b2477SPavel Labath OS << "invalid"; 7061f6b2477SPavel Labath break; 7071f6b2477SPavel Labath case ObjectFile::eStrataUnknown: 7081f6b2477SPavel Labath OS << "unknown"; 7091f6b2477SPavel Labath break; 7101f6b2477SPavel Labath case ObjectFile::eStrataUser: 7111f6b2477SPavel Labath OS << "user"; 7121f6b2477SPavel Labath break; 7131f6b2477SPavel Labath case ObjectFile::eStrataKernel: 7141f6b2477SPavel Labath OS << "kernel"; 7151f6b2477SPavel Labath break; 7161f6b2477SPavel Labath case ObjectFile::eStrataRawImage: 7171f6b2477SPavel Labath OS << "raw image"; 7181f6b2477SPavel Labath break; 7191f6b2477SPavel Labath case ObjectFile::eStrataJIT: 7201f6b2477SPavel Labath OS << "jit"; 7211f6b2477SPavel Labath break; 7221f6b2477SPavel Labath } 7231f6b2477SPavel Labath } 7247e6df41fSGreg Clayton 7257e6df41fSGreg Clayton 7267e6df41fSGreg Clayton Symtab *ObjectFile::GetSymtab() { 7277e6df41fSGreg Clayton ModuleSP module_sp(GetModule()); 7287e6df41fSGreg Clayton if (module_sp) { 7297e6df41fSGreg Clayton // We can't take the module lock in ObjectFile::GetSymtab() or we can 7307e6df41fSGreg Clayton // deadlock in DWARF indexing when any file asks for the symbol table from 7317e6df41fSGreg Clayton // an object file. This currently happens in the preloading of symbols in 7327e6df41fSGreg Clayton // SymbolFileDWARF::PreloadSymbols() because the main thread will take the 7337e6df41fSGreg Clayton // module lock, and then threads will be spun up to index the DWARF and 7347e6df41fSGreg Clayton // any of those threads might end up trying to relocate items in the DWARF 7357e6df41fSGreg Clayton // sections which causes ObjectFile::GetSectionData(...) to relocate section 7367e6df41fSGreg Clayton // data which requires the symbol table. 7377e6df41fSGreg Clayton // 7387e6df41fSGreg Clayton // So to work around this, we create the symbol table one time using 7397e6df41fSGreg Clayton // llvm::once_flag, lock it, and then set the unique pointer. Any other 7407e6df41fSGreg Clayton // thread that gets ahold of the symbol table before parsing is done, will 7417e6df41fSGreg Clayton // not be able to access the symbol table contents since all APIs in Symtab 7427e6df41fSGreg Clayton // are protected by a mutex in the Symtab object itself. 7437e6df41fSGreg Clayton llvm::call_once(*m_symtab_once_up, [&]() { 7447e6df41fSGreg Clayton Symtab *symtab = new Symtab(this); 7457e6df41fSGreg Clayton std::lock_guard<std::recursive_mutex> symtab_guard(symtab->GetMutex()); 7467e6df41fSGreg Clayton m_symtab_up.reset(symtab); 747da816ca0SGreg Clayton if (!m_symtab_up->LoadFromCache()) { 748da816ca0SGreg Clayton ElapsedTime elapsed(module_sp->GetSymtabParseTime()); 7497e6df41fSGreg Clayton ParseSymtab(*m_symtab_up); 7507e6df41fSGreg Clayton m_symtab_up->Finalize(); 751da816ca0SGreg Clayton } 7527e6df41fSGreg Clayton }); 7537e6df41fSGreg Clayton } 7547e6df41fSGreg Clayton return m_symtab_up.get(); 7557e6df41fSGreg Clayton } 756da816ca0SGreg Clayton 757da816ca0SGreg Clayton uint32_t ObjectFile::GetCacheHash() { 758da816ca0SGreg Clayton if (m_cache_hash) 759da816ca0SGreg Clayton return *m_cache_hash; 760da816ca0SGreg Clayton StreamString strm; 761da816ca0SGreg Clayton strm.Format("{0}-{1}-{2}", m_file, GetType(), GetStrata()); 762da816ca0SGreg Clayton m_cache_hash = llvm::djbHash(strm.GetString()); 763da816ca0SGreg Clayton return *m_cache_hash; 764da816ca0SGreg Clayton } 765