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"
226f9e6901SZachary Turner #include "lldb/Utility/Log.h"
2338d0632eSPavel Labath #include "lldb/Utility/Timer.h"
24b9c1b51eSKate Stone #include "lldb/lldb-private.h"
2530fdc8d8SChris Lattner 
2630fdc8d8SChris Lattner using namespace lldb;
2730fdc8d8SChris Lattner using namespace lldb_private;
2830fdc8d8SChris Lattner 
29e84f7841SPavel Labath char ObjectFile::ID;
30e84f7841SPavel Labath 
31d97e9f1aSJonas Devlieghere static ObjectFileSP
32d97e9f1aSJonas Devlieghere CreateObjectFromContainer(const lldb::ModuleSP &module_sp, const FileSpec *file,
33d97e9f1aSJonas Devlieghere                           lldb::offset_t file_offset, lldb::offset_t file_size,
34d97e9f1aSJonas Devlieghere                           DataBufferSP &data_sp, lldb::offset_t &data_offset) {
35d97e9f1aSJonas Devlieghere   ObjectContainerCreateInstance callback;
36d97e9f1aSJonas Devlieghere   for (uint32_t idx = 0;
37d97e9f1aSJonas Devlieghere        (callback = PluginManager::GetObjectContainerCreateCallbackAtIndex(
38d97e9f1aSJonas Devlieghere             idx)) != nullptr;
39d97e9f1aSJonas Devlieghere        ++idx) {
40d97e9f1aSJonas Devlieghere     std::unique_ptr<ObjectContainer> object_container_up(callback(
41d97e9f1aSJonas Devlieghere         module_sp, data_sp, data_offset, file, file_offset, file_size));
42d97e9f1aSJonas Devlieghere     if (object_container_up)
43d97e9f1aSJonas Devlieghere       return object_container_up->GetObjectFile(file);
44d97e9f1aSJonas Devlieghere   }
45d97e9f1aSJonas Devlieghere   return {};
46d97e9f1aSJonas Devlieghere }
47d97e9f1aSJonas Devlieghere 
48762f7135SGreg Clayton ObjectFileSP
49b9c1b51eSKate Stone ObjectFile::FindPlugin(const lldb::ModuleSP &module_sp, const FileSpec *file,
50b9c1b51eSKate Stone                        lldb::offset_t file_offset, lldb::offset_t file_size,
51b9c1b51eSKate Stone                        DataBufferSP &data_sp, lldb::offset_t &data_offset) {
525c1c8443SJonas Devlieghere   LLDB_SCOPED_TIMERF(
53b9c1b51eSKate Stone       "ObjectFile::FindPlugin (module = %s, file = %p, file_offset = "
54b9c1b51eSKate Stone       "0x%8.8" PRIx64 ", file_size = 0x%8.8" PRIx64 ")",
55b5ad4ec7SGreg Clayton       module_sp->GetFileSpec().GetPath().c_str(),
56b9c1b51eSKate Stone       static_cast<const void *>(file), static_cast<uint64_t>(file_offset),
57324a1036SSaleem Abdulrasool       static_cast<uint64_t>(file_size));
58d97e9f1aSJonas Devlieghere 
59d97e9f1aSJonas Devlieghere   if (!module_sp)
60d97e9f1aSJonas Devlieghere     return {};
61d97e9f1aSJonas Devlieghere 
62d97e9f1aSJonas Devlieghere   if (!file)
63d97e9f1aSJonas Devlieghere     return {};
645ce9c565SGreg Clayton 
65b9c1b51eSKate Stone   if (!data_sp) {
66a4a00cedSFred Riss     const bool file_exists = FileSystem::Instance().Exists(*file);
6705097246SAdrian Prantl     // We have an object name which most likely means we have a .o file in
6805097246SAdrian Prantl     // a static archive (.a file). Try and see if we have a cached archive
6905097246SAdrian Prantl     // first without reading any data first
70b9c1b51eSKate Stone     if (file_exists && module_sp->GetObjectName()) {
71d97e9f1aSJonas Devlieghere       ObjectFileSP object_file_sp = CreateObjectFromContainer(
72d97e9f1aSJonas Devlieghere           module_sp, file, file_offset, file_size, data_sp, data_offset);
73d97e9f1aSJonas Devlieghere       if (object_file_sp)
745ce9c565SGreg Clayton         return object_file_sp;
755ce9c565SGreg Clayton     }
7605097246SAdrian Prantl     // Ok, we didn't find any containers that have a named object, now lets
7705097246SAdrian Prantl     // read the first 512 bytes from the file so the object file and object
7805097246SAdrian Prantl     // container plug-ins can use these bytes to see if they can parse this
7905097246SAdrian Prantl     // file.
80b9c1b51eSKate Stone     if (file_size > 0) {
81d97e9f1aSJonas Devlieghere       data_sp = FileSystem::Instance().CreateDataBuffer(file->GetPath(), 512,
82d97e9f1aSJonas Devlieghere                                                         file_offset);
835ce9c565SGreg Clayton       data_offset = 0;
845ce9c565SGreg Clayton     }
8544435ed0SGreg Clayton   }
8630fdc8d8SChris Lattner 
87b9c1b51eSKate Stone   if (!data_sp || data_sp->GetByteSize() == 0) {
8830fdc8d8SChris Lattner     // Check for archive file with format "/path/to/archive.a(object.o)"
89ea637750SAdrian Prantl     llvm::SmallString<256> path_with_object;
90ea637750SAdrian Prantl     module_sp->GetFileSpec().GetPath(path_with_object);
9130fdc8d8SChris Lattner 
92d97e9f1aSJonas Devlieghere     FileSpec archive_file;
931f746071SGreg Clayton     ConstString archive_object;
94906ba471SGreg Clayton     const bool must_exist = true;
95d97e9f1aSJonas Devlieghere     if (ObjectFile::SplitArchivePathWithObject(path_with_object, archive_file,
96d97e9f1aSJonas Devlieghere                                                archive_object, must_exist)) {
9759b78bcbSJonas Devlieghere       file_size = FileSystem::Instance().GetByteSize(archive_file);
98b9c1b51eSKate Stone       if (file_size > 0) {
995ce9c565SGreg Clayton         file = &archive_file;
1001f746071SGreg Clayton         module_sp->SetFileSpecAndObjectName(archive_file, archive_object);
101b9c1b51eSKate Stone         // Check if this is a object container by iterating through all
10205097246SAdrian Prantl         // object container plugin instances and then trying to get an
10305097246SAdrian Prantl         // object file from the container plugins since we had a name.
10405097246SAdrian Prantl         // Also, don't read
1055ce9c565SGreg Clayton         // ANY data in case there is data cached in the container plug-ins
10605097246SAdrian Prantl         // (like BSD archives caching the contained objects within an
10705097246SAdrian Prantl         // file).
108d97e9f1aSJonas Devlieghere         ObjectFileSP object_file_sp = CreateObjectFromContainer(
109d97e9f1aSJonas Devlieghere             module_sp, file, file_offset, file_size, data_sp, data_offset);
110d97e9f1aSJonas Devlieghere         if (object_file_sp)
1115ce9c565SGreg Clayton           return object_file_sp;
11205097246SAdrian Prantl         // We failed to find any cached object files in the container plug-
11305097246SAdrian Prantl         // ins, so lets read the first 512 bytes and try again below...
11487e403aaSJonas Devlieghere         data_sp = FileSystem::Instance().CreateDataBuffer(
11587e403aaSJonas Devlieghere             archive_file.GetPath(), 512, file_offset);
1165ce9c565SGreg Clayton       }
1175ce9c565SGreg Clayton     }
1185ce9c565SGreg Clayton   }
1195ce9c565SGreg Clayton 
120b9c1b51eSKate Stone   if (data_sp && data_sp->GetByteSize() > 0) {
12105097246SAdrian Prantl     // Check if this is a normal object file by iterating through all
12205097246SAdrian Prantl     // object file plugin instances.
123d97e9f1aSJonas Devlieghere     ObjectFileCreateInstance callback;
124b9c1b51eSKate Stone     for (uint32_t idx = 0;
125d97e9f1aSJonas Devlieghere          (callback = PluginManager::GetObjectFileCreateCallbackAtIndex(idx)) !=
126b9c1b51eSKate Stone          nullptr;
127b9c1b51eSKate Stone          ++idx) {
128d97e9f1aSJonas Devlieghere       ObjectFileSP object_file_sp(callback(module_sp, data_sp, data_offset,
129d97e9f1aSJonas Devlieghere                                            file, file_offset, file_size));
130762f7135SGreg Clayton       if (object_file_sp.get())
131762f7135SGreg Clayton         return object_file_sp;
13230fdc8d8SChris Lattner     }
13330fdc8d8SChris Lattner 
13405097246SAdrian Prantl     // Check if this is a object container by iterating through all object
13505097246SAdrian Prantl     // container plugin instances and then trying to get an object file
13605097246SAdrian Prantl     // from the container.
137d97e9f1aSJonas Devlieghere     ObjectFileSP object_file_sp = CreateObjectFromContainer(
138d97e9f1aSJonas Devlieghere         module_sp, file, file_offset, file_size, data_sp, data_offset);
139d97e9f1aSJonas Devlieghere     if (object_file_sp)
140762f7135SGreg Clayton       return object_file_sp;
14130fdc8d8SChris Lattner   }
142d97e9f1aSJonas Devlieghere 
14305097246SAdrian Prantl   // We didn't find it, so clear our shared pointer in case it contains
14405097246SAdrian Prantl   // anything and return an empty shared pointer
145d97e9f1aSJonas Devlieghere   return {};
146762f7135SGreg Clayton }
147762f7135SGreg Clayton 
148b9c1b51eSKate Stone ObjectFileSP ObjectFile::FindPlugin(const lldb::ModuleSP &module_sp,
149c9660546SGreg Clayton                                     const ProcessSP &process_sp,
150c9660546SGreg Clayton                                     lldb::addr_t header_addr,
151b9c1b51eSKate Stone                                     DataBufferSP &data_sp) {
152c9660546SGreg Clayton   ObjectFileSP object_file_sp;
153c9660546SGreg Clayton 
154b9c1b51eSKate Stone   if (module_sp) {
1555c1c8443SJonas Devlieghere     LLDB_SCOPED_TIMERF("ObjectFile::FindPlugin (module = "
156b9c1b51eSKate Stone                        "%s, process = %p, header_addr = "
157b9c1b51eSKate Stone                        "0x%" PRIx64 ")",
158b5ad4ec7SGreg Clayton                        module_sp->GetFileSpec().GetPath().c_str(),
159324a1036SSaleem Abdulrasool                        static_cast<void *>(process_sp.get()), header_addr);
160c9660546SGreg Clayton     uint32_t idx;
161c9660546SGreg Clayton 
16205097246SAdrian Prantl     // Check if this is a normal object file by iterating through all object
16305097246SAdrian Prantl     // file plugin instances.
164c9660546SGreg Clayton     ObjectFileCreateMemoryInstance create_callback;
165b9c1b51eSKate Stone     for (idx = 0;
166b9c1b51eSKate Stone          (create_callback =
167b9c1b51eSKate Stone               PluginManager::GetObjectFileCreateMemoryCallbackAtIndex(idx)) !=
168b9c1b51eSKate Stone          nullptr;
169b9c1b51eSKate Stone          ++idx) {
170b9c1b51eSKate Stone       object_file_sp.reset(
171b9c1b51eSKate Stone           create_callback(module_sp, data_sp, process_sp, header_addr));
172c9660546SGreg Clayton       if (object_file_sp.get())
173c9660546SGreg Clayton         return object_file_sp;
174c9660546SGreg Clayton     }
175c9660546SGreg Clayton   }
176324a1036SSaleem Abdulrasool 
17705097246SAdrian Prantl   // We didn't find it, so clear our shared pointer in case it contains
17805097246SAdrian Prantl   // anything and return an empty shared pointer
179c9660546SGreg Clayton   object_file_sp.reset();
180c9660546SGreg Clayton   return object_file_sp;
181c9660546SGreg Clayton }
182c9660546SGreg Clayton 
183b9c1b51eSKate Stone size_t ObjectFile::GetModuleSpecifications(const FileSpec &file,
184f4d6de6aSGreg Clayton                                            lldb::offset_t file_offset,
1852540a8a7SGreg Clayton                                            lldb::offset_t file_size,
186a4a00cedSFred Riss                                            ModuleSpecList &specs,
187a4a00cedSFred Riss                                            DataBufferSP data_sp) {
188a4a00cedSFred Riss   if (!data_sp)
189a4a00cedSFred Riss     data_sp = FileSystem::Instance().CreateDataBuffer(file.GetPath(), 512,
190a4a00cedSFred Riss                                                       file_offset);
191b9c1b51eSKate Stone   if (data_sp) {
192b9c1b51eSKate Stone     if (file_size == 0) {
19359b78bcbSJonas Devlieghere       const lldb::offset_t actual_file_size =
19459b78bcbSJonas Devlieghere           FileSystem::Instance().GetByteSize(file);
1952540a8a7SGreg Clayton       if (actual_file_size > file_offset)
1962540a8a7SGreg Clayton         file_size = actual_file_size - file_offset;
1972540a8a7SGreg Clayton     }
198f4d6de6aSGreg Clayton     return ObjectFile::GetModuleSpecifications(file,        // file spec
199f4d6de6aSGreg Clayton                                                data_sp,     // data bytes
200f4d6de6aSGreg Clayton                                                0,           // data offset
201f4d6de6aSGreg Clayton                                                file_offset, // file offset
2022540a8a7SGreg Clayton                                                file_size,   // file length
203f4d6de6aSGreg Clayton                                                specs);
2042540a8a7SGreg Clayton   }
205f4d6de6aSGreg Clayton   return 0;
206f4d6de6aSGreg Clayton }
207f4d6de6aSGreg Clayton 
208b9c1b51eSKate Stone size_t ObjectFile::GetModuleSpecifications(
209b9c1b51eSKate Stone     const lldb_private::FileSpec &file, lldb::DataBufferSP &data_sp,
210b9c1b51eSKate Stone     lldb::offset_t data_offset, lldb::offset_t file_offset,
211b9c1b51eSKate Stone     lldb::offset_t file_size, lldb_private::ModuleSpecList &specs) {
212f4d6de6aSGreg Clayton   const size_t initial_count = specs.GetSize();
213f4d6de6aSGreg Clayton   ObjectFileGetModuleSpecifications callback;
214f4d6de6aSGreg Clayton   uint32_t i;
215f4d6de6aSGreg Clayton   // Try the ObjectFile plug-ins
216b9c1b51eSKate Stone   for (i = 0;
217b9c1b51eSKate Stone        (callback =
218b9c1b51eSKate Stone             PluginManager::GetObjectFileGetModuleSpecificationsCallbackAtIndex(
219b9c1b51eSKate Stone                 i)) != nullptr;
220b9c1b51eSKate Stone        ++i) {
2212540a8a7SGreg Clayton     if (callback(file, data_sp, data_offset, file_offset, file_size, specs) > 0)
222f4d6de6aSGreg Clayton       return specs.GetSize() - initial_count;
223f4d6de6aSGreg Clayton   }
224f4d6de6aSGreg Clayton 
225f4d6de6aSGreg Clayton   // Try the ObjectContainer plug-ins
226b9c1b51eSKate Stone   for (i = 0;
227b9c1b51eSKate Stone        (callback = PluginManager::
228b9c1b51eSKate Stone             GetObjectContainerGetModuleSpecificationsCallbackAtIndex(i)) !=
229b9c1b51eSKate Stone        nullptr;
230b9c1b51eSKate Stone        ++i) {
2312540a8a7SGreg Clayton     if (callback(file, data_sp, data_offset, file_offset, file_size, specs) > 0)
232f4d6de6aSGreg Clayton       return specs.GetSize() - initial_count;
233f4d6de6aSGreg Clayton   }
234f4d6de6aSGreg Clayton   return 0;
235f4d6de6aSGreg Clayton }
236f4d6de6aSGreg Clayton 
237e72dfb32SGreg Clayton ObjectFile::ObjectFile(const lldb::ModuleSP &module_sp,
238762f7135SGreg Clayton                        const FileSpec *file_spec_ptr,
239b9c1b51eSKate Stone                        lldb::offset_t file_offset, lldb::offset_t length,
24023f8c95aSGreg Clayton                        const lldb::DataBufferSP &data_sp,
241b9c1b51eSKate Stone                        lldb::offset_t data_offset)
242b9c1b51eSKate Stone     : ModuleChild(module_sp),
243762f7135SGreg Clayton       m_file(), // This file could be different from the original module's file
244b9c1b51eSKate Stone       m_type(eTypeInvalid), m_strata(eStrataInvalid),
24566d88326SPavel Labath       m_file_offset(file_offset), m_length(length), m_data(), m_process_wp(),
246d5b44036SJonas Devlieghere       m_memory_addr(LLDB_INVALID_ADDRESS), m_sections_up(), m_symtab_up(),
247*7e6df41fSGreg Clayton       m_symtab_once_up(new llvm::once_flag()) {
248762f7135SGreg Clayton   if (file_spec_ptr)
249762f7135SGreg Clayton     m_file = *file_spec_ptr;
2505ce9c565SGreg Clayton   if (data_sp)
2515ce9c565SGreg Clayton     m_data.SetData(data_sp, data_offset, length);
2525160ce5cSGreg Clayton   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_OBJECT));
25363e5fb76SJonas Devlieghere   LLDB_LOGF(log,
25463e5fb76SJonas Devlieghere             "%p ObjectFile::ObjectFile() module = %p (%s), file = %s, "
255b9c1b51eSKate Stone             "file_offset = 0x%8.8" PRIx64 ", size = %" PRIu64,
256b9c1b51eSKate Stone             static_cast<void *>(this), static_cast<void *>(module_sp.get()),
257b5ad4ec7SGreg Clayton             module_sp->GetSpecificationDescription().c_str(),
258b9c1b51eSKate Stone             m_file ? m_file.GetPath().c_str() : "<NULL>", m_file_offset,
259b9c1b51eSKate Stone             m_length);
260762f7135SGreg Clayton }
261762f7135SGreg Clayton 
262e72dfb32SGreg Clayton ObjectFile::ObjectFile(const lldb::ModuleSP &module_sp,
263b9c1b51eSKate Stone                        const ProcessSP &process_sp, lldb::addr_t header_addr,
264b9c1b51eSKate Stone                        DataBufferSP &header_data_sp)
265b9c1b51eSKate Stone     : ModuleChild(module_sp), m_file(), m_type(eTypeInvalid),
266b9c1b51eSKate Stone       m_strata(eStrataInvalid), m_file_offset(0), m_length(0), m_data(),
26766d88326SPavel Labath       m_process_wp(process_sp), m_memory_addr(header_addr), m_sections_up(),
268*7e6df41fSGreg Clayton       m_symtab_up(), m_symtab_once_up(new llvm::once_flag()) {
269c9660546SGreg Clayton   if (header_data_sp)
270c9660546SGreg Clayton     m_data.SetData(header_data_sp, 0, header_data_sp->GetByteSize());
2715160ce5cSGreg Clayton   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_OBJECT));
27263e5fb76SJonas Devlieghere   LLDB_LOGF(log,
27363e5fb76SJonas Devlieghere             "%p ObjectFile::ObjectFile() module = %p (%s), process = %p, "
274b9c1b51eSKate Stone             "header_addr = 0x%" PRIx64,
275b9c1b51eSKate Stone             static_cast<void *>(this), static_cast<void *>(module_sp.get()),
276b5ad4ec7SGreg Clayton             module_sp->GetSpecificationDescription().c_str(),
277324a1036SSaleem Abdulrasool             static_cast<void *>(process_sp.get()), m_memory_addr);
278c9660546SGreg Clayton }
279c9660546SGreg Clayton 
280b9c1b51eSKate Stone ObjectFile::~ObjectFile() {
2815160ce5cSGreg Clayton   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_OBJECT));
28263e5fb76SJonas Devlieghere   LLDB_LOGF(log, "%p ObjectFile::~ObjectFile ()\n", static_cast<void *>(this));
28330fdc8d8SChris Lattner }
2845aee162fSJim Ingham 
285b9c1b51eSKate Stone bool ObjectFile::SetModulesArchitecture(const ArchSpec &new_arch) {
286e72dfb32SGreg Clayton   ModuleSP module_sp(GetModule());
287e72dfb32SGreg Clayton   if (module_sp)
288e72dfb32SGreg Clayton     return module_sp->SetArchitecture(new_arch);
289e72dfb32SGreg Clayton   return false;
2905aee162fSJim Ingham }
2915aee162fSJim Ingham 
292b9c1b51eSKate Stone AddressClass ObjectFile::GetAddressClass(addr_t file_addr) {
2933046e668SGreg Clayton   Symtab *symtab = GetSymtab();
294b9c1b51eSKate Stone   if (symtab) {
295ded470d3SGreg Clayton     Symbol *symbol = symtab->FindSymbolContainingFileAddress(file_addr);
296b9c1b51eSKate Stone     if (symbol) {
297b9c1b51eSKate Stone       if (symbol->ValueIsAddress()) {
298358cf1eaSGreg Clayton         const SectionSP section_sp(symbol->GetAddressRef().GetSection());
299b9c1b51eSKate Stone         if (section_sp) {
300e72dfb32SGreg Clayton           const SectionType section_type = section_sp->GetType();
301b9c1b51eSKate Stone           switch (section_type) {
302324a1036SSaleem Abdulrasool           case eSectionTypeInvalid:
30304803b3eSTatyana Krasnukha             return AddressClass::eUnknown;
304324a1036SSaleem Abdulrasool           case eSectionTypeCode:
30504803b3eSTatyana Krasnukha             return AddressClass::eCode;
306324a1036SSaleem Abdulrasool           case eSectionTypeContainer:
30704803b3eSTatyana Krasnukha             return AddressClass::eUnknown;
3085009f9d5SGreg Clayton           case eSectionTypeData:
3095009f9d5SGreg Clayton           case eSectionTypeDataCString:
3105009f9d5SGreg Clayton           case eSectionTypeDataCStringPointers:
3115009f9d5SGreg Clayton           case eSectionTypeDataSymbolAddress:
3125009f9d5SGreg Clayton           case eSectionTypeData4:
3135009f9d5SGreg Clayton           case eSectionTypeData8:
3145009f9d5SGreg Clayton           case eSectionTypeData16:
3155009f9d5SGreg Clayton           case eSectionTypeDataPointers:
3165009f9d5SGreg Clayton           case eSectionTypeZeroFill:
3175009f9d5SGreg Clayton           case eSectionTypeDataObjCMessageRefs:
3185009f9d5SGreg Clayton           case eSectionTypeDataObjCCFStrings:
31965d4d5c3SRyan Brown           case eSectionTypeGoSymtab:
32004803b3eSTatyana Krasnukha             return AddressClass::eData;
3215009f9d5SGreg Clayton           case eSectionTypeDebug:
3225009f9d5SGreg Clayton           case eSectionTypeDWARFDebugAbbrev:
323004bcb78SGeorge Rimar           case eSectionTypeDWARFDebugAbbrevDwo:
324c178d4c0STamas Berghammer           case eSectionTypeDWARFDebugAddr:
3255009f9d5SGreg Clayton           case eSectionTypeDWARFDebugAranges:
326963ce483STamas Berghammer           case eSectionTypeDWARFDebugCuIndex:
3275009f9d5SGreg Clayton           case eSectionTypeDWARFDebugFrame:
3285009f9d5SGreg Clayton           case eSectionTypeDWARFDebugInfo:
329004bcb78SGeorge Rimar           case eSectionTypeDWARFDebugInfoDwo:
3305009f9d5SGreg Clayton           case eSectionTypeDWARFDebugLine:
331c6c7bfc4SGeorge Rimar           case eSectionTypeDWARFDebugLineStr:
3325009f9d5SGreg Clayton           case eSectionTypeDWARFDebugLoc:
3334b5bc388SPavel Labath           case eSectionTypeDWARFDebugLocDwo:
334e4dee269SGeorge Rimar           case eSectionTypeDWARFDebugLocLists:
3354b5bc388SPavel Labath           case eSectionTypeDWARFDebugLocListsDwo:
3365009f9d5SGreg Clayton           case eSectionTypeDWARFDebugMacInfo:
337d8335e9aSSiva Chandra           case eSectionTypeDWARFDebugMacro:
338a041d848SPavel Labath           case eSectionTypeDWARFDebugNames:
3395009f9d5SGreg Clayton           case eSectionTypeDWARFDebugPubNames:
3405009f9d5SGreg Clayton           case eSectionTypeDWARFDebugPubTypes:
3415009f9d5SGreg Clayton           case eSectionTypeDWARFDebugRanges:
3426e357123SGeorge Rimar           case eSectionTypeDWARFDebugRngLists:
3434023bd05SPavel Labath           case eSectionTypeDWARFDebugRngListsDwo:
3445009f9d5SGreg Clayton           case eSectionTypeDWARFDebugStr:
345004bcb78SGeorge Rimar           case eSectionTypeDWARFDebugStrDwo:
346c178d4c0STamas Berghammer           case eSectionTypeDWARFDebugStrOffsets:
347004bcb78SGeorge Rimar           case eSectionTypeDWARFDebugStrOffsetsDwo:
3487b59ff2fSPavel Labath           case eSectionTypeDWARFDebugTuIndex:
3492550ca1eSGreg Clayton           case eSectionTypeDWARFDebugTypes:
350ad805ef9SPavel Labath           case eSectionTypeDWARFDebugTypesDwo:
3515009f9d5SGreg Clayton           case eSectionTypeDWARFAppleNames:
3525009f9d5SGreg Clayton           case eSectionTypeDWARFAppleTypes:
3535009f9d5SGreg Clayton           case eSectionTypeDWARFAppleNamespaces:
3545009f9d5SGreg Clayton           case eSectionTypeDWARFAppleObjC:
355e4777a9dSJan Kratochvil           case eSectionTypeDWARFGNUDebugAltLink:
35604803b3eSTatyana Krasnukha             return AddressClass::eDebug;
357324a1036SSaleem Abdulrasool           case eSectionTypeEHFrame:
358648f3c7eSTamas Berghammer           case eSectionTypeARMexidx:
359648f3c7eSTamas Berghammer           case eSectionTypeARMextab:
360e589e7e3SJason Molenda           case eSectionTypeCompactUnwind:
36104803b3eSTatyana Krasnukha             return AddressClass::eRuntime;
362a7499c98SMichael Sartain           case eSectionTypeELFSymbolTable:
363a7499c98SMichael Sartain           case eSectionTypeELFDynamicSymbols:
364a7499c98SMichael Sartain           case eSectionTypeELFRelocationEntries:
365a7499c98SMichael Sartain           case eSectionTypeELFDynamicLinkInfo:
366324a1036SSaleem Abdulrasool           case eSectionTypeOther:
36704803b3eSTatyana Krasnukha             return AddressClass::eUnknown;
36831d315b3STamas Berghammer           case eSectionTypeAbsoluteAddress:
369b9c1b51eSKate Stone             // In case of absolute sections decide the address class based on
37005097246SAdrian Prantl             // the symbol type because the section type isn't specify if it is
37105097246SAdrian Prantl             // a code or a data section.
37231d315b3STamas Berghammer             break;
373ded470d3SGreg Clayton           }
374ded470d3SGreg Clayton         }
375ded470d3SGreg Clayton       }
376ded470d3SGreg Clayton 
377e0d378b3SGreg Clayton       const SymbolType symbol_type = symbol->GetType();
378b9c1b51eSKate Stone       switch (symbol_type) {
379b9c1b51eSKate Stone       case eSymbolTypeAny:
38004803b3eSTatyana Krasnukha         return AddressClass::eUnknown;
381b9c1b51eSKate Stone       case eSymbolTypeAbsolute:
38204803b3eSTatyana Krasnukha         return AddressClass::eUnknown;
383b9c1b51eSKate Stone       case eSymbolTypeCode:
38404803b3eSTatyana Krasnukha         return AddressClass::eCode;
385b9c1b51eSKate Stone       case eSymbolTypeTrampoline:
38604803b3eSTatyana Krasnukha         return AddressClass::eCode;
387b9c1b51eSKate Stone       case eSymbolTypeResolver:
38804803b3eSTatyana Krasnukha         return AddressClass::eCode;
389b9c1b51eSKate Stone       case eSymbolTypeData:
39004803b3eSTatyana Krasnukha         return AddressClass::eData;
391b9c1b51eSKate Stone       case eSymbolTypeRuntime:
39204803b3eSTatyana Krasnukha         return AddressClass::eRuntime;
393b9c1b51eSKate Stone       case eSymbolTypeException:
39404803b3eSTatyana Krasnukha         return AddressClass::eRuntime;
395b9c1b51eSKate Stone       case eSymbolTypeSourceFile:
39604803b3eSTatyana Krasnukha         return AddressClass::eDebug;
397b9c1b51eSKate Stone       case eSymbolTypeHeaderFile:
39804803b3eSTatyana Krasnukha         return AddressClass::eDebug;
399b9c1b51eSKate Stone       case eSymbolTypeObjectFile:
40004803b3eSTatyana Krasnukha         return AddressClass::eDebug;
401b9c1b51eSKate Stone       case eSymbolTypeCommonBlock:
40204803b3eSTatyana Krasnukha         return AddressClass::eDebug;
403b9c1b51eSKate Stone       case eSymbolTypeBlock:
40404803b3eSTatyana Krasnukha         return AddressClass::eDebug;
405b9c1b51eSKate Stone       case eSymbolTypeLocal:
40604803b3eSTatyana Krasnukha         return AddressClass::eData;
407b9c1b51eSKate Stone       case eSymbolTypeParam:
40804803b3eSTatyana Krasnukha         return AddressClass::eData;
409b9c1b51eSKate Stone       case eSymbolTypeVariable:
41004803b3eSTatyana Krasnukha         return AddressClass::eData;
411b9c1b51eSKate Stone       case eSymbolTypeVariableType:
41204803b3eSTatyana Krasnukha         return AddressClass::eDebug;
413b9c1b51eSKate Stone       case eSymbolTypeLineEntry:
41404803b3eSTatyana Krasnukha         return AddressClass::eDebug;
415b9c1b51eSKate Stone       case eSymbolTypeLineHeader:
41604803b3eSTatyana Krasnukha         return AddressClass::eDebug;
417b9c1b51eSKate Stone       case eSymbolTypeScopeBegin:
41804803b3eSTatyana Krasnukha         return AddressClass::eDebug;
419b9c1b51eSKate Stone       case eSymbolTypeScopeEnd:
42004803b3eSTatyana Krasnukha         return AddressClass::eDebug;
421b9c1b51eSKate Stone       case eSymbolTypeAdditional:
42204803b3eSTatyana Krasnukha         return AddressClass::eUnknown;
423b9c1b51eSKate Stone       case eSymbolTypeCompiler:
42404803b3eSTatyana Krasnukha         return AddressClass::eDebug;
425b9c1b51eSKate Stone       case eSymbolTypeInstrumentation:
42604803b3eSTatyana Krasnukha         return AddressClass::eDebug;
427b9c1b51eSKate Stone       case eSymbolTypeUndefined:
42804803b3eSTatyana Krasnukha         return AddressClass::eUnknown;
429b9c1b51eSKate Stone       case eSymbolTypeObjCClass:
43004803b3eSTatyana Krasnukha         return AddressClass::eRuntime;
431b9c1b51eSKate Stone       case eSymbolTypeObjCMetaClass:
43204803b3eSTatyana Krasnukha         return AddressClass::eRuntime;
433b9c1b51eSKate Stone       case eSymbolTypeObjCIVar:
43404803b3eSTatyana Krasnukha         return AddressClass::eRuntime;
435b9c1b51eSKate Stone       case eSymbolTypeReExported:
43604803b3eSTatyana Krasnukha         return AddressClass::eRuntime;
437ded470d3SGreg Clayton       }
438ded470d3SGreg Clayton     }
439ded470d3SGreg Clayton   }
44004803b3eSTatyana Krasnukha   return AddressClass::eUnknown;
441ded470d3SGreg Clayton }
442ded470d3SGreg Clayton 
443b9c1b51eSKate Stone DataBufferSP ObjectFile::ReadMemory(const ProcessSP &process_sp,
444b9c1b51eSKate Stone                                     lldb::addr_t addr, size_t byte_size) {
445c9660546SGreg Clayton   DataBufferSP data_sp;
446b9c1b51eSKate Stone   if (process_sp) {
447d5b44036SJonas Devlieghere     std::unique_ptr<DataBufferHeap> data_up(new DataBufferHeap(byte_size, 0));
44897206d57SZachary Turner     Status error;
449b9c1b51eSKate Stone     const size_t bytes_read = process_sp->ReadMemory(
450d5b44036SJonas Devlieghere         addr, data_up->GetBytes(), data_up->GetByteSize(), error);
451c9660546SGreg Clayton     if (bytes_read == byte_size)
452d5b44036SJonas Devlieghere       data_sp.reset(data_up.release());
453c9660546SGreg Clayton   }
454c9660546SGreg Clayton   return data_sp;
455c9660546SGreg Clayton }
456c9660546SGreg Clayton 
457b9c1b51eSKate Stone size_t ObjectFile::GetData(lldb::offset_t offset, size_t length,
458b9c1b51eSKate Stone                            DataExtractor &data) const {
459b9c1b51eSKate Stone   // The entire file has already been mmap'ed into m_data, so just copy from
46005097246SAdrian Prantl   // there as the back mmap buffer will be shared with shared pointers.
461a0f72441SMartin Storsjö   return data.SetData(m_data, offset, length);
46244435ed0SGreg Clayton }
46344435ed0SGreg Clayton 
464b9c1b51eSKate Stone size_t ObjectFile::CopyData(lldb::offset_t offset, size_t length,
465b9c1b51eSKate Stone                             void *dst) const {
466b9c1b51eSKate Stone   // The entire file has already been mmap'ed into m_data, so just copy from
46705097246SAdrian Prantl   // there Note that the data remains in target byte order.
468b0e33d41SEd Maste   return m_data.CopyData(offset, length, dst);
46944435ed0SGreg Clayton }
470ded470d3SGreg Clayton 
471d13f691fSEd Maste size_t ObjectFile::ReadSectionData(Section *section,
472b9c1b51eSKate Stone                                    lldb::offset_t section_offset, void *dst,
473d13f691fSEd Maste                                    size_t dst_len) {
474f03e6d84SMatthew Gardiner   assert(section);
475f03e6d84SMatthew Gardiner   section_offset *= section->GetTargetByteSize();
476f03e6d84SMatthew Gardiner 
477a7499c98SMichael Sartain   // If some other objectfile owns this data, pass this to them.
478a7499c98SMichael Sartain   if (section->GetObjectFile() != this)
479b9c1b51eSKate Stone     return section->GetObjectFile()->ReadSectionData(section, section_offset,
480b9c1b51eSKate Stone                                                      dst, dst_len);
481a7499c98SMichael Sartain 
482203b4774SStefan Gränitz   if (!section->IsRelocated())
483203b4774SStefan Gränitz     RelocateSection(section);
484203b4774SStefan Gränitz 
485b9c1b51eSKate Stone   if (IsInMemory()) {
486c9660546SGreg Clayton     ProcessSP process_sp(m_process_wp.lock());
487b9c1b51eSKate Stone     if (process_sp) {
48897206d57SZachary Turner       Status error;
489b9c1b51eSKate Stone       const addr_t base_load_addr =
490b9c1b51eSKate Stone           section->GetLoadBaseAddress(&process_sp->GetTarget());
49139f7ee86SGreg Clayton       if (base_load_addr != LLDB_INVALID_ADDRESS)
492b9c1b51eSKate Stone         return process_sp->ReadMemory(base_load_addr + section_offset, dst,
493b9c1b51eSKate Stone                                       dst_len, error);
494c9660546SGreg Clayton     }
495b9c1b51eSKate Stone   } else {
496a746e8e5SZachary Turner     const lldb::offset_t section_file_size = section->GetFileSize();
497b9c1b51eSKate Stone     if (section_offset < section_file_size) {
498a746e8e5SZachary Turner       const size_t section_bytes_left = section_file_size - section_offset;
499a746e8e5SZachary Turner       size_t section_dst_len = dst_len;
500ee212e2cSGreg Clayton       if (section_dst_len > section_bytes_left)
501ee212e2cSGreg Clayton         section_dst_len = section_bytes_left;
502b9c1b51eSKate Stone       return CopyData(section->GetFileOffset() + section_offset,
503b9c1b51eSKate Stone                       section_dst_len, dst);
504b9c1b51eSKate Stone     } else {
505b9c1b51eSKate Stone       if (section->GetType() == eSectionTypeZeroFill) {
506ecda2b2dSSean Callanan         const uint64_t section_size = section->GetByteSize();
507ecda2b2dSSean Callanan         const uint64_t section_bytes_left = section_size - section_offset;
508ecda2b2dSSean Callanan         uint64_t section_dst_len = dst_len;
509ecda2b2dSSean Callanan         if (section_dst_len > section_bytes_left)
510ecda2b2dSSean Callanan           section_dst_len = section_bytes_left;
511b2f1fb29SVirgile Bello         memset(dst, 0, section_dst_len);
512ecda2b2dSSean Callanan         return section_dst_len;
513ecda2b2dSSean Callanan       }
514ecda2b2dSSean Callanan     }
515c9660546SGreg Clayton   }
516c9660546SGreg Clayton   return 0;
517c9660546SGreg Clayton }
518c9660546SGreg Clayton 
519c9660546SGreg Clayton // Get the section data the file on disk
520d13f691fSEd Maste size_t ObjectFile::ReadSectionData(Section *section,
521d13f691fSEd Maste                                    DataExtractor &section_data) {
522a7499c98SMichael Sartain   // If some other objectfile owns this data, pass this to them.
523a7499c98SMichael Sartain   if (section->GetObjectFile() != this)
524a7499c98SMichael Sartain     return section->GetObjectFile()->ReadSectionData(section, section_data);
525a7499c98SMichael Sartain 
526203b4774SStefan Gränitz   if (!section->IsRelocated())
527203b4774SStefan Gränitz     RelocateSection(section);
528203b4774SStefan Gränitz 
529b9c1b51eSKate Stone   if (IsInMemory()) {
530c9660546SGreg Clayton     ProcessSP process_sp(m_process_wp.lock());
531b9c1b51eSKate Stone     if (process_sp) {
532b9c1b51eSKate Stone       const addr_t base_load_addr =
533b9c1b51eSKate Stone           section->GetLoadBaseAddress(&process_sp->GetTarget());
534b9c1b51eSKate Stone       if (base_load_addr != LLDB_INVALID_ADDRESS) {
535b9c1b51eSKate Stone         DataBufferSP data_sp(
536b9c1b51eSKate Stone             ReadMemory(process_sp, base_load_addr, section->GetByteSize()));
537b9c1b51eSKate Stone         if (data_sp) {
538c9660546SGreg Clayton           section_data.SetData(data_sp, 0, data_sp->GetByteSize());
539c9660546SGreg Clayton           section_data.SetByteOrder(process_sp->GetByteOrder());
540c9660546SGreg Clayton           section_data.SetAddressByteSize(process_sp->GetAddressByteSize());
541c9660546SGreg Clayton           return section_data.GetByteSize();
542c9660546SGreg Clayton         }
543c9660546SGreg Clayton       }
544c9660546SGreg Clayton     }
545203b4774SStefan Gränitz   }
546203b4774SStefan Gränitz 
54705097246SAdrian Prantl   // The object file now contains a full mmap'ed copy of the object file
54805097246SAdrian Prantl   // data, so just use this
549b9c1b51eSKate Stone   return GetData(section->GetFileOffset(), section->GetFileSize(),
550b9c1b51eSKate Stone                   section_data);
551c9660546SGreg Clayton }
552c9660546SGreg Clayton 
553ea637750SAdrian Prantl bool ObjectFile::SplitArchivePathWithObject(llvm::StringRef path_with_object,
554b9c1b51eSKate Stone                                             FileSpec &archive_file,
555b9c1b51eSKate Stone                                             ConstString &archive_object,
556b9c1b51eSKate Stone                                             bool must_exist) {
557ea637750SAdrian Prantl   size_t len = path_with_object.size();
558ea637750SAdrian Prantl   if (len < 2 || path_with_object.back() != ')')
5591f746071SGreg Clayton     return false;
560ea637750SAdrian Prantl   llvm::StringRef archive = path_with_object.substr(0, path_with_object.rfind('('));
561ea637750SAdrian Prantl   if (archive.empty())
562ea637750SAdrian Prantl     return false;
563ea637750SAdrian Prantl   llvm::StringRef object = path_with_object.substr(archive.size() + 1).drop_back();
564ea637750SAdrian Prantl   archive_file.SetFile(archive, FileSpec::Style::native);
565ea637750SAdrian Prantl   if (must_exist && !FileSystem::Instance().Exists(archive_file))
566ea637750SAdrian Prantl     return false;
567ea637750SAdrian Prantl   archive_object.SetString(object);
568ea637750SAdrian Prantl   return true;
5691f746071SGreg Clayton }
5701f746071SGreg Clayton 
571b9c1b51eSKate Stone void ObjectFile::ClearSymtab() {
5729422dd64SGreg Clayton   ModuleSP module_sp(GetModule());
573b9c1b51eSKate Stone   if (module_sp) {
5745160ce5cSGreg Clayton     Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_OBJECT));
57563e5fb76SJonas Devlieghere     LLDB_LOGF(log, "%p ObjectFile::ClearSymtab () symtab = %p",
576324a1036SSaleem Abdulrasool               static_cast<void *>(this),
577d5b44036SJonas Devlieghere               static_cast<void *>(m_symtab_up.get()));
578*7e6df41fSGreg Clayton     // Since we need to clear the symbol table, we need a new llvm::once_flag
579*7e6df41fSGreg Clayton     // instance so we can safely create another symbol table
580*7e6df41fSGreg Clayton     m_symtab_once_up.reset(new llvm::once_flag());
581d5b44036SJonas Devlieghere     m_symtab_up.reset();
5829422dd64SGreg Clayton   }
5839422dd64SGreg Clayton }
5843046e668SGreg Clayton 
585b9c1b51eSKate Stone SectionList *ObjectFile::GetSectionList(bool update_module_section_list) {
586d5b44036SJonas Devlieghere   if (m_sections_up == nullptr) {
587b9c1b51eSKate Stone     if (update_module_section_list) {
5883046e668SGreg Clayton       ModuleSP module_sp(GetModule());
589b9c1b51eSKate Stone       if (module_sp) {
59016ff8604SSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(module_sp->GetMutex());
5913046e668SGreg Clayton         CreateSections(*module_sp->GetUnifiedSectionList());
5923046e668SGreg Clayton       }
593b9c1b51eSKate Stone     } else {
594eb882fc1STamas Berghammer       SectionList unified_section_list;
595eb882fc1STamas Berghammer       CreateSections(unified_section_list);
596eb882fc1STamas Berghammer     }
597eb882fc1STamas Berghammer   }
598d5b44036SJonas Devlieghere   return m_sections_up.get();
599a7499c98SMichael Sartain }
600649a607eSJason Molenda 
601649a607eSJason Molenda lldb::SymbolType
602649a607eSJason Molenda ObjectFile::GetSymbolTypeFromName(llvm::StringRef name,
603b9c1b51eSKate Stone                                   lldb::SymbolType symbol_type_hint) {
604b9c1b51eSKate Stone   if (!name.empty()) {
605b9c1b51eSKate Stone     if (name.startswith("_OBJC_")) {
606649a607eSJason Molenda       // ObjC
607649a607eSJason Molenda       if (name.startswith("_OBJC_CLASS_$_"))
608649a607eSJason Molenda         return lldb::eSymbolTypeObjCClass;
609649a607eSJason Molenda       if (name.startswith("_OBJC_METACLASS_$_"))
610649a607eSJason Molenda         return lldb::eSymbolTypeObjCMetaClass;
611649a607eSJason Molenda       if (name.startswith("_OBJC_IVAR_$_"))
612649a607eSJason Molenda         return lldb::eSymbolTypeObjCIVar;
613b9c1b51eSKate Stone     } else if (name.startswith(".objc_class_name_")) {
614649a607eSJason Molenda       // ObjC v1
615649a607eSJason Molenda       return lldb::eSymbolTypeObjCClass;
616649a607eSJason Molenda     }
617649a607eSJason Molenda   }
618649a607eSJason Molenda   return symbol_type_hint;
619649a607eSJason Molenda }
6206b63b148STamas Berghammer 
62116064d35SPavel Labath std::vector<ObjectFile::LoadableData>
62216064d35SPavel Labath ObjectFile::GetLoadableData(Target &target) {
62316064d35SPavel Labath   std::vector<LoadableData> loadables;
6244687db0eSHafiz Abid Qadeer   SectionList *section_list = GetSectionList();
6254687db0eSHafiz Abid Qadeer   if (!section_list)
62616064d35SPavel Labath     return loadables;
62716064d35SPavel Labath   // Create a list of loadable data from loadable sections
6284687db0eSHafiz Abid Qadeer   size_t section_count = section_list->GetNumSections(0);
6294687db0eSHafiz Abid Qadeer   for (size_t i = 0; i < section_count; ++i) {
63016064d35SPavel Labath     LoadableData loadable;
6314687db0eSHafiz Abid Qadeer     SectionSP section_sp = section_list->GetSectionAtIndex(i);
63216064d35SPavel Labath     loadable.Dest =
63316064d35SPavel Labath         target.GetSectionLoadList().GetSectionLoadAddress(section_sp);
63416064d35SPavel Labath     if (loadable.Dest == LLDB_INVALID_ADDRESS)
63516064d35SPavel Labath       continue;
6364687db0eSHafiz Abid Qadeer     // We can skip sections like bss
6374687db0eSHafiz Abid Qadeer     if (section_sp->GetFileSize() == 0)
6384687db0eSHafiz Abid Qadeer       continue;
63916064d35SPavel Labath     DataExtractor section_data;
6404687db0eSHafiz Abid Qadeer     section_sp->GetSectionData(section_data);
64116064d35SPavel Labath     loadable.Contents = llvm::ArrayRef<uint8_t>(section_data.GetDataStart(),
64216064d35SPavel Labath                                                 section_data.GetByteSize());
64316064d35SPavel Labath     loadables.push_back(loadable);
644ec03d7e3SPavel Labath   }
64516064d35SPavel Labath   return loadables;
6464687db0eSHafiz Abid Qadeer }
647d13f691fSEd Maste 
64830c2441aSAleksandr Urakov std::unique_ptr<CallFrameInfo> ObjectFile::CreateCallFrameInfo() {
64930c2441aSAleksandr Urakov   return {};
65030c2441aSAleksandr Urakov }
65130c2441aSAleksandr Urakov 
652d13f691fSEd Maste void ObjectFile::RelocateSection(lldb_private::Section *section)
653d13f691fSEd Maste {
654d13f691fSEd Maste }
65550251fc7SPavel Labath 
65650251fc7SPavel Labath DataBufferSP ObjectFile::MapFileData(const FileSpec &file, uint64_t Size,
65750251fc7SPavel Labath                                      uint64_t Offset) {
65887e403aaSJonas Devlieghere   return FileSystem::Instance().CreateDataBuffer(file.GetPath(), Size, Offset);
65950251fc7SPavel Labath }
6601f6b2477SPavel Labath 
6611f6b2477SPavel Labath void llvm::format_provider<ObjectFile::Type>::format(
6621f6b2477SPavel Labath     const ObjectFile::Type &type, raw_ostream &OS, StringRef Style) {
6631f6b2477SPavel Labath   switch (type) {
6641f6b2477SPavel Labath   case ObjectFile::eTypeInvalid:
6651f6b2477SPavel Labath     OS << "invalid";
6661f6b2477SPavel Labath     break;
6671f6b2477SPavel Labath   case ObjectFile::eTypeCoreFile:
6681f6b2477SPavel Labath     OS << "core file";
6691f6b2477SPavel Labath     break;
6701f6b2477SPavel Labath   case ObjectFile::eTypeExecutable:
6711f6b2477SPavel Labath     OS << "executable";
6721f6b2477SPavel Labath     break;
6731f6b2477SPavel Labath   case ObjectFile::eTypeDebugInfo:
6741f6b2477SPavel Labath     OS << "debug info";
6751f6b2477SPavel Labath     break;
6761f6b2477SPavel Labath   case ObjectFile::eTypeDynamicLinker:
6771f6b2477SPavel Labath     OS << "dynamic linker";
6781f6b2477SPavel Labath     break;
6791f6b2477SPavel Labath   case ObjectFile::eTypeObjectFile:
6801f6b2477SPavel Labath     OS << "object file";
6811f6b2477SPavel Labath     break;
6821f6b2477SPavel Labath   case ObjectFile::eTypeSharedLibrary:
6831f6b2477SPavel Labath     OS << "shared library";
6841f6b2477SPavel Labath     break;
6851f6b2477SPavel Labath   case ObjectFile::eTypeStubLibrary:
6861f6b2477SPavel Labath     OS << "stub library";
6871f6b2477SPavel Labath     break;
6881f6b2477SPavel Labath   case ObjectFile::eTypeJIT:
6891f6b2477SPavel Labath     OS << "jit";
6901f6b2477SPavel Labath     break;
6911f6b2477SPavel Labath   case ObjectFile::eTypeUnknown:
6921f6b2477SPavel Labath     OS << "unknown";
6931f6b2477SPavel Labath     break;
6941f6b2477SPavel Labath   }
6951f6b2477SPavel Labath }
6961f6b2477SPavel Labath 
6971f6b2477SPavel Labath void llvm::format_provider<ObjectFile::Strata>::format(
6981f6b2477SPavel Labath     const ObjectFile::Strata &strata, raw_ostream &OS, StringRef Style) {
6991f6b2477SPavel Labath   switch (strata) {
7001f6b2477SPavel Labath   case ObjectFile::eStrataInvalid:
7011f6b2477SPavel Labath     OS << "invalid";
7021f6b2477SPavel Labath     break;
7031f6b2477SPavel Labath   case ObjectFile::eStrataUnknown:
7041f6b2477SPavel Labath     OS << "unknown";
7051f6b2477SPavel Labath     break;
7061f6b2477SPavel Labath   case ObjectFile::eStrataUser:
7071f6b2477SPavel Labath     OS << "user";
7081f6b2477SPavel Labath     break;
7091f6b2477SPavel Labath   case ObjectFile::eStrataKernel:
7101f6b2477SPavel Labath     OS << "kernel";
7111f6b2477SPavel Labath     break;
7121f6b2477SPavel Labath   case ObjectFile::eStrataRawImage:
7131f6b2477SPavel Labath     OS << "raw image";
7141f6b2477SPavel Labath     break;
7151f6b2477SPavel Labath   case ObjectFile::eStrataJIT:
7161f6b2477SPavel Labath     OS << "jit";
7171f6b2477SPavel Labath     break;
7181f6b2477SPavel Labath   }
7191f6b2477SPavel Labath }
720*7e6df41fSGreg Clayton 
721*7e6df41fSGreg Clayton 
722*7e6df41fSGreg Clayton Symtab *ObjectFile::GetSymtab() {
723*7e6df41fSGreg Clayton   ModuleSP module_sp(GetModule());
724*7e6df41fSGreg Clayton   if (module_sp) {
725*7e6df41fSGreg Clayton     // We can't take the module lock in ObjectFile::GetSymtab() or we can
726*7e6df41fSGreg Clayton     // deadlock in DWARF indexing when any file asks for the symbol table from
727*7e6df41fSGreg Clayton     // an object file. This currently happens in the preloading of symbols in
728*7e6df41fSGreg Clayton     // SymbolFileDWARF::PreloadSymbols() because the main thread will take the
729*7e6df41fSGreg Clayton     // module lock, and then threads will be spun up to index the DWARF and
730*7e6df41fSGreg Clayton     // any of those threads might end up trying to relocate items in the DWARF
731*7e6df41fSGreg Clayton     // sections which causes ObjectFile::GetSectionData(...) to relocate section
732*7e6df41fSGreg Clayton     // data which requires the symbol table.
733*7e6df41fSGreg Clayton     //
734*7e6df41fSGreg Clayton     // So to work around this, we create the symbol table one time using
735*7e6df41fSGreg Clayton     // llvm::once_flag, lock it, and then set the unique pointer. Any other
736*7e6df41fSGreg Clayton     // thread that gets ahold of the symbol table before parsing is done, will
737*7e6df41fSGreg Clayton     // not be able to access the symbol table contents since all APIs in Symtab
738*7e6df41fSGreg Clayton     // are protected by a mutex in the Symtab object itself.
739*7e6df41fSGreg Clayton     llvm::call_once(*m_symtab_once_up, [&]() {
740*7e6df41fSGreg Clayton        ElapsedTime elapsed(module_sp->GetSymtabParseTime());
741*7e6df41fSGreg Clayton        Symtab *symtab = new Symtab(this);
742*7e6df41fSGreg Clayton        std::lock_guard<std::recursive_mutex> symtab_guard(symtab->GetMutex());
743*7e6df41fSGreg Clayton        m_symtab_up.reset(symtab);
744*7e6df41fSGreg Clayton        ParseSymtab(*m_symtab_up);
745*7e6df41fSGreg Clayton        m_symtab_up->Finalize();
746*7e6df41fSGreg Clayton     });
747*7e6df41fSGreg Clayton   }
748*7e6df41fSGreg Clayton   return m_symtab_up.get();
749*7e6df41fSGreg Clayton }
750