LLDB mainline
SymbolVendorELF.cpp
Go to the documentation of this file.
1//===-- SymbolVendorELF.cpp -----------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#include "SymbolVendorELF.h"
10
11#include <cstring>
12
14#include "lldb/Core/Module.h"
17#include "lldb/Core/Section.h"
18#include "lldb/Host/Host.h"
20#include "lldb/Target/Target.h"
22#include "lldb/Utility/Timer.h"
23
24using namespace lldb;
25using namespace lldb_private;
26
28
29// SymbolVendorELF constructor
31 : SymbolVendor(module_sp) {}
32
36}
37
40}
41
43 return "Symbol vendor for ELF that looks for dSYM files that match "
44 "executables.";
45}
46
47// If this is needed elsewhere, it can be exported/moved.
48static bool IsDwpSymbolFile(const lldb::ModuleSP &module_sp,
49 const FileSpec &file_spec) {
50 DataBufferSP dwp_file_data_sp;
51 lldb::offset_t dwp_file_data_offset = 0;
52 // Try to create an ObjectFile from the file_spec.
54 module_sp, &file_spec, 0, FileSystem::Instance().GetByteSize(file_spec),
55 dwp_file_data_sp, dwp_file_data_offset);
56 // The presence of a debug_cu_index section is the key identifying feature of
57 // a DWP file. Make sure we don't fill in the section list on dwp_obj_file
58 // (by calling GetSectionList(false)) as this function could be called before
59 // we may have all the symbol files collected and available.
60 return dwp_obj_file && ObjectFileELF::classof(dwp_obj_file.get()) &&
61 dwp_obj_file->GetSectionList(false)->FindSectionByType(
63}
64
65// CreateInstance
66//
67// Platforms can register a callback to use when creating symbol vendors to
68// allow for complex debug information file setups, and to also allow for
69// finding separate debug information files.
72 lldb_private::Stream *feedback_strm) {
73 if (!module_sp)
74 return nullptr;
75
76 ObjectFileELF *obj_file =
77 llvm::dyn_cast_or_null<ObjectFileELF>(module_sp->GetObjectFile());
78 if (!obj_file)
79 return nullptr;
80
81 lldb_private::UUID uuid = obj_file->GetUUID();
82 if (!uuid)
83 return nullptr;
84
85 // If the main object file already contains debug info, then we are done.
86 if (obj_file->GetSectionList()->FindSectionByType(
88 return nullptr;
89
90 // If the module specified a filespec, use that.
91 FileSpec fspec = module_sp->GetSymbolFileFileSpec();
92 // Otherwise, try gnu_debuglink, if one exists.
93 if (!fspec)
94 fspec = obj_file->GetDebugLink().value_or(FileSpec());
95
96 LLDB_SCOPED_TIMERF("SymbolVendorELF::CreateInstance (module = %s)",
97 module_sp->GetFileSpec().GetPath().c_str());
98
99 ModuleSpec module_spec;
100
101 module_spec.GetFileSpec() = obj_file->GetFileSpec();
103 module_spec.GetSymbolFileSpec() = fspec;
104 module_spec.GetUUID() = uuid;
106 FileSpec dsym_fspec =
107 PluginManager::LocateExecutableSymbolFile(module_spec, search_paths);
108 if (!dsym_fspec || IsDwpSymbolFile(module_sp, dsym_fspec)) {
109 // If we have a stripped binary or if we have a DWP file, SymbolLocator
110 // plugins may be able to give us an unstripped binary or an
111 // 'only-keep-debug' stripped file.
112 ModuleSpec unstripped_spec =
114 if (!unstripped_spec)
115 return nullptr;
116 // The default SymbolLocator plugin returns the original binary if no other
117 // plugin finds something better.
118 if (unstripped_spec.GetFileSpec() == module_spec.GetFileSpec())
119 return nullptr;
120 dsym_fspec = unstripped_spec.GetFileSpec();
121 }
122
123 DataBufferSP dsym_file_data_sp;
124 lldb::offset_t dsym_file_data_offset = 0;
125 ObjectFileSP dsym_objfile_sp = ObjectFile::FindPlugin(
126 module_sp, &dsym_fspec, 0, FileSystem::Instance().GetByteSize(dsym_fspec),
127 dsym_file_data_sp, dsym_file_data_offset);
128 if (!dsym_objfile_sp)
129 return nullptr;
130
131 // This objfile is for debugging purposes. Sadly, ObjectFileELF won't
132 // be able to figure this out consistently as the symbol file may not
133 // have stripped the code sections, etc.
134 dsym_objfile_sp->SetType(ObjectFile::eTypeDebugInfo);
135
136 SymbolVendorELF *symbol_vendor = new SymbolVendorELF(module_sp);
137
138 // Get the module unified section list and add our debug sections to
139 // that.
140 SectionList *module_section_list = module_sp->GetSectionList();
141 SectionList *objfile_section_list = dsym_objfile_sp->GetSectionList();
142
143 if (!module_section_list || !objfile_section_list)
144 return nullptr;
145
146 static const SectionType g_sections[] = {
158 };
159 for (SectionType section_type : g_sections) {
160 if (SectionSP section_sp =
161 objfile_section_list->FindSectionByType(section_type, true)) {
162 if (SectionSP module_section_sp =
163 module_section_list->FindSectionByType(section_type, true))
164 module_section_list->ReplaceSection(module_section_sp->GetID(),
165 section_sp);
166 else
167 module_section_list->AddSection(section_sp);
168 }
169 }
170
171 symbol_vendor->AddSymbolFileRepresentation(dsym_objfile_sp);
172 return symbol_vendor;
173}
#define LLDB_PLUGIN_DEFINE(PluginName)
Definition: PluginManager.h:32
static bool IsDwpSymbolFile(const lldb::ModuleSP &module_sp, const FileSpec &file_spec)
#define LLDB_SCOPED_TIMERF(...)
Definition: Timer.h:86
Generic COFF object file reader.
Definition: ObjectFileELF.h:58
static bool classof(const ObjectFile *obj)
Definition: ObjectFileELF.h:98
std::optional< lldb_private::FileSpec > GetDebugLink()
Return the contents of the .gnu_debuglink section, if the object file contains it.
lldb_private::UUID GetUUID() override
Gets the UUID for this object file.
static llvm::StringRef GetPluginDescriptionStatic()
static void Initialize()
static void Terminate()
static lldb_private::SymbolVendor * CreateInstance(const lldb::ModuleSP &module_sp, lldb_private::Stream *feedback_strm)
static llvm::StringRef GetPluginNameStatic()
A file collection class.
Definition: FileSpecList.h:85
A file utility class.
Definition: FileSpec.h:56
void Resolve(llvm::SmallVectorImpl< char > &path)
Resolve path to make it canonical.
static FileSystem & Instance()
FileSpec & GetFileSpec()
Definition: ModuleSpec.h:53
FileSpec & GetSymbolFileSpec()
Definition: ModuleSpec.h:77
static lldb::ObjectFileSP FindPlugin(const lldb::ModuleSP &module_sp, const FileSpec *file_spec, lldb::offset_t file_offset, lldb::offset_t file_size, lldb::DataBufferSP &data_sp, lldb::offset_t &data_offset)
Find a ObjectFile plug-in that can parse file_spec.
Definition: ObjectFile.cpp:53
@ eTypeDebugInfo
An object file that contains only debug information.
Definition: ObjectFile.h:55
virtual FileSpec & GetFileSpec()
Get accessor to the object file specification.
Definition: ObjectFile.h:275
virtual SectionList * GetSectionList(bool update_module_section_list=true)
Gets the section list for the currently selected architecture (and object for archives).
Definition: ObjectFile.cpp:600
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static ModuleSpec LocateExecutableObjectFile(const ModuleSpec &module_spec)
static FileSpec LocateExecutableSymbolFile(const ModuleSpec &module_spec, const FileSpecList &default_search_paths)
static bool UnregisterPlugin(ABICreateInstance create_callback)
size_t AddSection(const lldb::SectionSP &section_sp)
Definition: Section.cpp:475
bool ReplaceSection(lldb::user_id_t sect_id, const lldb::SectionSP &section_sp, uint32_t depth=UINT32_MAX)
Definition: Section.cpp:515
lldb::SectionSP FindSectionByType(lldb::SectionType sect_type, bool check_children, size_t start_idx=0) const
Definition: Section.cpp:592
A stream class that can stream formatted output to a file.
Definition: Stream.h:28
void AddSymbolFileRepresentation(const lldb::ObjectFileSP &objfile_sp)
static FileSpecList GetDefaultDebugFileSearchPaths()
Definition: Target.cpp:2610
A class that represents a running process on the host machine.
Definition: SBAddress.h:15
uint64_t offset_t
Definition: lldb-types.h:85
std::shared_ptr< lldb_private::ObjectFile > ObjectFileSP
Definition: lldb-forward.h:372
std::shared_ptr< lldb_private::DataBuffer > DataBufferSP
Definition: lldb-forward.h:333
std::shared_ptr< lldb_private::Section > SectionSP
Definition: lldb-forward.h:413
@ eSectionTypeDWARFDebugStrOffsets
@ eSectionTypeDWARFDebugPubNames
@ eSectionTypeDWARFDebugFrame
@ eSectionTypeDWARFDebugLocLists
DWARF v5 .debug_loclists.
@ eSectionTypeDWARFDebugTypes
DWARF .debug_types section.
@ eSectionTypeDWARFDebugMacInfo
@ eSectionTypeDWARFDebugNames
DWARF v5 .debug_names.
@ eSectionTypeDWARFDebugRngLists
DWARF v5 .debug_rnglists.
@ eSectionTypeDWARFDebugMacro
@ eSectionTypeDWARFDebugInfo
@ eSectionTypeDWARFDebugRanges
@ eSectionTypeDWARFDebugLine
@ eSectionTypeDWARFDebugPubTypes
@ eSectionTypeDWARFDebugStr
@ eSectionTypeDWARFDebugLineStr
DWARF v5 .debug_line_str.
@ eSectionTypeDWARFDebugLoc
@ eSectionTypeDWARFDebugCuIndex
@ eSectionTypeDWARFDebugAranges
@ eSectionTypeDWARFGNUDebugAltLink
@ eSectionTypeDWARFDebugAbbrev
@ eSectionTypeDWARFDebugAddr
@ eSectionTypeELFSymbolTable
Elf SHT_SYMTAB section.
std::shared_ptr< lldb_private::Module > ModuleSP
Definition: lldb-forward.h:370