xref: /openbsd-src/gnu/llvm/lldb/source/Symbol/Symbol.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1dda28197Spatrick //===-- Symbol.cpp --------------------------------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick 
9061da546Spatrick #include "lldb/Symbol/Symbol.h"
10061da546Spatrick 
11061da546Spatrick #include "lldb/Core/Module.h"
12061da546Spatrick #include "lldb/Core/ModuleSpec.h"
13061da546Spatrick #include "lldb/Core/Section.h"
14061da546Spatrick #include "lldb/Symbol/Function.h"
15061da546Spatrick #include "lldb/Symbol/ObjectFile.h"
16061da546Spatrick #include "lldb/Symbol/SymbolVendor.h"
17061da546Spatrick #include "lldb/Symbol/Symtab.h"
18061da546Spatrick #include "lldb/Target/Process.h"
19061da546Spatrick #include "lldb/Target/Target.h"
20*f6aab3d8Srobert #include "lldb/Utility/DataEncoder.h"
21061da546Spatrick #include "lldb/Utility/Stream.h"
22061da546Spatrick 
23061da546Spatrick using namespace lldb;
24061da546Spatrick using namespace lldb_private;
25061da546Spatrick 
Symbol()26061da546Spatrick Symbol::Symbol()
27be691f3bSpatrick     : SymbolContextScope(), m_type_data_resolved(false), m_is_synthetic(false),
28be691f3bSpatrick       m_is_debug(false), m_is_external(false), m_size_is_sibling(false),
29061da546Spatrick       m_size_is_synthesized(false), m_size_is_valid(false),
30061da546Spatrick       m_demangled_is_synthesized(false), m_contains_linker_annotations(false),
31be691f3bSpatrick       m_is_weak(false), m_type(eSymbolTypeInvalid), m_mangled(),
32be691f3bSpatrick       m_addr_range() {}
33061da546Spatrick 
Symbol(uint32_t symID,llvm::StringRef name,SymbolType type,bool external,bool is_debug,bool is_trampoline,bool is_artificial,const lldb::SectionSP & section_sp,addr_t offset,addr_t size,bool size_is_valid,bool contains_linker_annotations,uint32_t flags)34*f6aab3d8Srobert Symbol::Symbol(uint32_t symID, llvm::StringRef name, SymbolType type,
35*f6aab3d8Srobert                bool external, bool is_debug, bool is_trampoline,
36*f6aab3d8Srobert                bool is_artificial, const lldb::SectionSP &section_sp,
37*f6aab3d8Srobert                addr_t offset, addr_t size, bool size_is_valid,
38*f6aab3d8Srobert                bool contains_linker_annotations, uint32_t flags)
39*f6aab3d8Srobert     : SymbolContextScope(), m_uid(symID), m_type_data_resolved(false),
40*f6aab3d8Srobert       m_is_synthetic(is_artificial), m_is_debug(is_debug),
41*f6aab3d8Srobert       m_is_external(external), m_size_is_sibling(false),
42061da546Spatrick       m_size_is_synthesized(false), m_size_is_valid(size_is_valid || size > 0),
43061da546Spatrick       m_demangled_is_synthesized(false),
44061da546Spatrick       m_contains_linker_annotations(contains_linker_annotations),
45*f6aab3d8Srobert       m_is_weak(false), m_type(type), m_mangled(name),
46061da546Spatrick       m_addr_range(section_sp, offset, size), m_flags(flags) {}
47061da546Spatrick 
Symbol(uint32_t symID,const Mangled & mangled,SymbolType type,bool external,bool is_debug,bool is_trampoline,bool is_artificial,const AddressRange & range,bool size_is_valid,bool contains_linker_annotations,uint32_t flags)48061da546Spatrick Symbol::Symbol(uint32_t symID, const Mangled &mangled, SymbolType type,
49061da546Spatrick                bool external, bool is_debug, bool is_trampoline,
50061da546Spatrick                bool is_artificial, const AddressRange &range,
51061da546Spatrick                bool size_is_valid, bool contains_linker_annotations,
52061da546Spatrick                uint32_t flags)
53*f6aab3d8Srobert     : SymbolContextScope(), m_uid(symID), m_type_data_resolved(false),
54*f6aab3d8Srobert       m_is_synthetic(is_artificial), m_is_debug(is_debug),
55*f6aab3d8Srobert       m_is_external(external), m_size_is_sibling(false),
56061da546Spatrick       m_size_is_synthesized(false),
57061da546Spatrick       m_size_is_valid(size_is_valid || range.GetByteSize() > 0),
58061da546Spatrick       m_demangled_is_synthesized(false),
59061da546Spatrick       m_contains_linker_annotations(contains_linker_annotations),
60061da546Spatrick       m_is_weak(false), m_type(type), m_mangled(mangled), m_addr_range(range),
61061da546Spatrick       m_flags(flags) {}
62061da546Spatrick 
Symbol(const Symbol & rhs)63061da546Spatrick Symbol::Symbol(const Symbol &rhs)
64061da546Spatrick     : SymbolContextScope(rhs), m_uid(rhs.m_uid), m_type_data(rhs.m_type_data),
65061da546Spatrick       m_type_data_resolved(rhs.m_type_data_resolved),
66061da546Spatrick       m_is_synthetic(rhs.m_is_synthetic), m_is_debug(rhs.m_is_debug),
67061da546Spatrick       m_is_external(rhs.m_is_external),
68061da546Spatrick       m_size_is_sibling(rhs.m_size_is_sibling), m_size_is_synthesized(false),
69061da546Spatrick       m_size_is_valid(rhs.m_size_is_valid),
70061da546Spatrick       m_demangled_is_synthesized(rhs.m_demangled_is_synthesized),
71061da546Spatrick       m_contains_linker_annotations(rhs.m_contains_linker_annotations),
72061da546Spatrick       m_is_weak(rhs.m_is_weak), m_type(rhs.m_type), m_mangled(rhs.m_mangled),
73061da546Spatrick       m_addr_range(rhs.m_addr_range), m_flags(rhs.m_flags) {}
74061da546Spatrick 
operator =(const Symbol & rhs)75061da546Spatrick const Symbol &Symbol::operator=(const Symbol &rhs) {
76061da546Spatrick   if (this != &rhs) {
77061da546Spatrick     SymbolContextScope::operator=(rhs);
78061da546Spatrick     m_uid = rhs.m_uid;
79061da546Spatrick     m_type_data = rhs.m_type_data;
80061da546Spatrick     m_type_data_resolved = rhs.m_type_data_resolved;
81061da546Spatrick     m_is_synthetic = rhs.m_is_synthetic;
82061da546Spatrick     m_is_debug = rhs.m_is_debug;
83061da546Spatrick     m_is_external = rhs.m_is_external;
84061da546Spatrick     m_size_is_sibling = rhs.m_size_is_sibling;
85061da546Spatrick     m_size_is_synthesized = rhs.m_size_is_sibling;
86061da546Spatrick     m_size_is_valid = rhs.m_size_is_valid;
87061da546Spatrick     m_demangled_is_synthesized = rhs.m_demangled_is_synthesized;
88061da546Spatrick     m_contains_linker_annotations = rhs.m_contains_linker_annotations;
89061da546Spatrick     m_is_weak = rhs.m_is_weak;
90061da546Spatrick     m_type = rhs.m_type;
91061da546Spatrick     m_mangled = rhs.m_mangled;
92061da546Spatrick     m_addr_range = rhs.m_addr_range;
93061da546Spatrick     m_flags = rhs.m_flags;
94061da546Spatrick   }
95061da546Spatrick   return *this;
96061da546Spatrick }
97061da546Spatrick 
Clear()98061da546Spatrick void Symbol::Clear() {
99061da546Spatrick   m_uid = UINT32_MAX;
100061da546Spatrick   m_mangled.Clear();
101061da546Spatrick   m_type_data = 0;
102061da546Spatrick   m_type_data_resolved = false;
103061da546Spatrick   m_is_synthetic = false;
104061da546Spatrick   m_is_debug = false;
105061da546Spatrick   m_is_external = false;
106061da546Spatrick   m_size_is_sibling = false;
107061da546Spatrick   m_size_is_synthesized = false;
108061da546Spatrick   m_size_is_valid = false;
109061da546Spatrick   m_demangled_is_synthesized = false;
110061da546Spatrick   m_contains_linker_annotations = false;
111061da546Spatrick   m_is_weak = false;
112061da546Spatrick   m_type = eSymbolTypeInvalid;
113061da546Spatrick   m_flags = 0;
114061da546Spatrick   m_addr_range.Clear();
115061da546Spatrick }
116061da546Spatrick 
ValueIsAddress() const117061da546Spatrick bool Symbol::ValueIsAddress() const {
118*f6aab3d8Srobert   return (bool)m_addr_range.GetBaseAddress().GetSection();
119061da546Spatrick }
120061da546Spatrick 
GetDisplayName() const121061da546Spatrick ConstString Symbol::GetDisplayName() const {
122be691f3bSpatrick   return GetMangled().GetDisplayDemangledName();
123061da546Spatrick }
124061da546Spatrick 
GetReExportedSymbolName() const125061da546Spatrick ConstString Symbol::GetReExportedSymbolName() const {
126061da546Spatrick   if (m_type == eSymbolTypeReExported) {
127061da546Spatrick     // For eSymbolTypeReExported, the "const char *" from a ConstString is used
128061da546Spatrick     // as the offset in the address range base address. We can then make this
129061da546Spatrick     // back into a string that is the re-exported name.
130061da546Spatrick     intptr_t str_ptr = m_addr_range.GetBaseAddress().GetOffset();
131061da546Spatrick     if (str_ptr != 0)
132061da546Spatrick       return ConstString((const char *)str_ptr);
133061da546Spatrick     else
134061da546Spatrick       return GetName();
135061da546Spatrick   }
136061da546Spatrick   return ConstString();
137061da546Spatrick }
138061da546Spatrick 
GetReExportedSymbolSharedLibrary() const139061da546Spatrick FileSpec Symbol::GetReExportedSymbolSharedLibrary() const {
140061da546Spatrick   if (m_type == eSymbolTypeReExported) {
141061da546Spatrick     // For eSymbolTypeReExported, the "const char *" from a ConstString is used
142061da546Spatrick     // as the offset in the address range base address. We can then make this
143061da546Spatrick     // back into a string that is the re-exported name.
144061da546Spatrick     intptr_t str_ptr = m_addr_range.GetByteSize();
145061da546Spatrick     if (str_ptr != 0)
146061da546Spatrick       return FileSpec((const char *)str_ptr);
147061da546Spatrick   }
148061da546Spatrick   return FileSpec();
149061da546Spatrick }
150061da546Spatrick 
SetReExportedSymbolName(ConstString name)151061da546Spatrick void Symbol::SetReExportedSymbolName(ConstString name) {
152061da546Spatrick   SetType(eSymbolTypeReExported);
153061da546Spatrick   // For eSymbolTypeReExported, the "const char *" from a ConstString is used
154061da546Spatrick   // as the offset in the address range base address.
155061da546Spatrick   m_addr_range.GetBaseAddress().SetOffset((uintptr_t)name.GetCString());
156061da546Spatrick }
157061da546Spatrick 
SetReExportedSymbolSharedLibrary(const FileSpec & fspec)158061da546Spatrick bool Symbol::SetReExportedSymbolSharedLibrary(const FileSpec &fspec) {
159061da546Spatrick   if (m_type == eSymbolTypeReExported) {
160061da546Spatrick     // For eSymbolTypeReExported, the "const char *" from a ConstString is used
161061da546Spatrick     // as the offset in the address range base address.
162061da546Spatrick     m_addr_range.SetByteSize(
163061da546Spatrick         (uintptr_t)ConstString(fspec.GetPath().c_str()).GetCString());
164061da546Spatrick     return true;
165061da546Spatrick   }
166061da546Spatrick   return false;
167061da546Spatrick }
168061da546Spatrick 
GetSiblingIndex() const169061da546Spatrick uint32_t Symbol::GetSiblingIndex() const {
170061da546Spatrick   return m_size_is_sibling ? m_addr_range.GetByteSize() : UINT32_MAX;
171061da546Spatrick }
172061da546Spatrick 
IsTrampoline() const173061da546Spatrick bool Symbol::IsTrampoline() const { return m_type == eSymbolTypeTrampoline; }
174061da546Spatrick 
IsIndirect() const175061da546Spatrick bool Symbol::IsIndirect() const { return m_type == eSymbolTypeResolver; }
176061da546Spatrick 
GetDescription(Stream * s,lldb::DescriptionLevel level,Target * target) const177061da546Spatrick void Symbol::GetDescription(Stream *s, lldb::DescriptionLevel level,
178061da546Spatrick                             Target *target) const {
179061da546Spatrick   s->Printf("id = {0x%8.8x}", m_uid);
180061da546Spatrick 
181061da546Spatrick   if (m_addr_range.GetBaseAddress().GetSection()) {
182061da546Spatrick     if (ValueIsAddress()) {
183061da546Spatrick       const lldb::addr_t byte_size = GetByteSize();
184061da546Spatrick       if (byte_size > 0) {
185061da546Spatrick         s->PutCString(", range = ");
186061da546Spatrick         m_addr_range.Dump(s, target, Address::DumpStyleLoadAddress,
187061da546Spatrick                           Address::DumpStyleFileAddress);
188061da546Spatrick       } else {
189061da546Spatrick         s->PutCString(", address = ");
190061da546Spatrick         m_addr_range.GetBaseAddress().Dump(s, target,
191061da546Spatrick                                            Address::DumpStyleLoadAddress,
192061da546Spatrick                                            Address::DumpStyleFileAddress);
193061da546Spatrick       }
194061da546Spatrick     } else
195061da546Spatrick       s->Printf(", value = 0x%16.16" PRIx64,
196061da546Spatrick                 m_addr_range.GetBaseAddress().GetOffset());
197061da546Spatrick   } else {
198061da546Spatrick     if (m_size_is_sibling)
199061da546Spatrick       s->Printf(", sibling = %5" PRIu64,
200061da546Spatrick                 m_addr_range.GetBaseAddress().GetOffset());
201061da546Spatrick     else
202061da546Spatrick       s->Printf(", value = 0x%16.16" PRIx64,
203061da546Spatrick                 m_addr_range.GetBaseAddress().GetOffset());
204061da546Spatrick   }
205be691f3bSpatrick   ConstString demangled = GetMangled().GetDemangledName();
206061da546Spatrick   if (demangled)
207061da546Spatrick     s->Printf(", name=\"%s\"", demangled.AsCString());
208061da546Spatrick   if (m_mangled.GetMangledName())
209061da546Spatrick     s->Printf(", mangled=\"%s\"", m_mangled.GetMangledName().AsCString());
210061da546Spatrick }
211061da546Spatrick 
Dump(Stream * s,Target * target,uint32_t index,Mangled::NamePreference name_preference) const212061da546Spatrick void Symbol::Dump(Stream *s, Target *target, uint32_t index,
213061da546Spatrick                   Mangled::NamePreference name_preference) const {
214061da546Spatrick   s->Printf("[%5u] %6u %c%c%c %-15s ", index, GetID(), m_is_debug ? 'D' : ' ',
215061da546Spatrick             m_is_synthetic ? 'S' : ' ', m_is_external ? 'X' : ' ',
216061da546Spatrick             GetTypeAsString());
217061da546Spatrick 
218061da546Spatrick   // Make sure the size of the symbol is up to date before dumping
219061da546Spatrick   GetByteSize();
220061da546Spatrick 
221be691f3bSpatrick   ConstString name = GetMangled().GetName(name_preference);
222061da546Spatrick   if (ValueIsAddress()) {
223061da546Spatrick     if (!m_addr_range.GetBaseAddress().Dump(s, nullptr,
224061da546Spatrick                                             Address::DumpStyleFileAddress))
225061da546Spatrick       s->Printf("%*s", 18, "");
226061da546Spatrick 
227061da546Spatrick     s->PutChar(' ');
228061da546Spatrick 
229061da546Spatrick     if (!m_addr_range.GetBaseAddress().Dump(s, target,
230061da546Spatrick                                             Address::DumpStyleLoadAddress))
231061da546Spatrick       s->Printf("%*s", 18, "");
232061da546Spatrick 
233061da546Spatrick     const char *format = m_size_is_sibling ? " Sibling -> [%5llu] 0x%8.8x %s\n"
234061da546Spatrick                                            : " 0x%16.16" PRIx64 " 0x%8.8x %s\n";
235061da546Spatrick     s->Printf(format, GetByteSize(), m_flags, name.AsCString(""));
236061da546Spatrick   } else if (m_type == eSymbolTypeReExported) {
237061da546Spatrick     s->Printf(
238061da546Spatrick         "                                                         0x%8.8x %s",
239061da546Spatrick         m_flags, name.AsCString(""));
240061da546Spatrick 
241061da546Spatrick     ConstString reexport_name = GetReExportedSymbolName();
242061da546Spatrick     intptr_t shlib = m_addr_range.GetByteSize();
243061da546Spatrick     if (shlib)
244061da546Spatrick       s->Printf(" -> %s`%s\n", (const char *)shlib, reexport_name.GetCString());
245061da546Spatrick     else
246061da546Spatrick       s->Printf(" -> %s\n", reexport_name.GetCString());
247061da546Spatrick   } else {
248061da546Spatrick     const char *format =
249061da546Spatrick         m_size_is_sibling
250061da546Spatrick             ? "0x%16.16" PRIx64
251061da546Spatrick               "                    Sibling -> [%5llu] 0x%8.8x %s\n"
252061da546Spatrick             : "0x%16.16" PRIx64 "                    0x%16.16" PRIx64
253061da546Spatrick               " 0x%8.8x %s\n";
254061da546Spatrick     s->Printf(format, m_addr_range.GetBaseAddress().GetOffset(), GetByteSize(),
255061da546Spatrick               m_flags, name.AsCString(""));
256061da546Spatrick   }
257061da546Spatrick }
258061da546Spatrick 
GetPrologueByteSize()259061da546Spatrick uint32_t Symbol::GetPrologueByteSize() {
260061da546Spatrick   if (m_type == eSymbolTypeCode || m_type == eSymbolTypeResolver) {
261061da546Spatrick     if (!m_type_data_resolved) {
262061da546Spatrick       m_type_data_resolved = true;
263061da546Spatrick 
264061da546Spatrick       const Address &base_address = m_addr_range.GetBaseAddress();
265061da546Spatrick       Function *function = base_address.CalculateSymbolContextFunction();
266061da546Spatrick       if (function) {
267061da546Spatrick         // Functions have line entries which can also potentially have end of
268061da546Spatrick         // prologue information. So if this symbol points to a function, use
269061da546Spatrick         // the prologue information from there.
270061da546Spatrick         m_type_data = function->GetPrologueByteSize();
271061da546Spatrick       } else {
272061da546Spatrick         ModuleSP module_sp(base_address.GetModule());
273061da546Spatrick         SymbolContext sc;
274061da546Spatrick         if (module_sp) {
275061da546Spatrick           uint32_t resolved_flags = module_sp->ResolveSymbolContextForAddress(
276061da546Spatrick               base_address, eSymbolContextLineEntry, sc);
277061da546Spatrick           if (resolved_flags & eSymbolContextLineEntry) {
278061da546Spatrick             // Default to the end of the first line entry.
279061da546Spatrick             m_type_data = sc.line_entry.range.GetByteSize();
280061da546Spatrick 
281061da546Spatrick             // Set address for next line.
282061da546Spatrick             Address addr(base_address);
283061da546Spatrick             addr.Slide(m_type_data);
284061da546Spatrick 
285061da546Spatrick             // Check the first few instructions and look for one that has a
286061da546Spatrick             // line number that is different than the first entry. This is also
287061da546Spatrick             // done in Function::GetPrologueByteSize().
288061da546Spatrick             uint16_t total_offset = m_type_data;
289061da546Spatrick             for (int idx = 0; idx < 6; ++idx) {
290061da546Spatrick               SymbolContext sc_temp;
291061da546Spatrick               resolved_flags = module_sp->ResolveSymbolContextForAddress(
292061da546Spatrick                   addr, eSymbolContextLineEntry, sc_temp);
293061da546Spatrick               // Make sure we got line number information...
294061da546Spatrick               if (!(resolved_flags & eSymbolContextLineEntry))
295061da546Spatrick                 break;
296061da546Spatrick 
297061da546Spatrick               // If this line number is different than our first one, use it
298061da546Spatrick               // and we're done.
299061da546Spatrick               if (sc_temp.line_entry.line != sc.line_entry.line) {
300061da546Spatrick                 m_type_data = total_offset;
301061da546Spatrick                 break;
302061da546Spatrick               }
303061da546Spatrick 
304061da546Spatrick               // Slide addr up to the next line address.
305061da546Spatrick               addr.Slide(sc_temp.line_entry.range.GetByteSize());
306061da546Spatrick               total_offset += sc_temp.line_entry.range.GetByteSize();
307061da546Spatrick               // If we've gone too far, bail out.
308061da546Spatrick               if (total_offset >= m_addr_range.GetByteSize())
309061da546Spatrick                 break;
310061da546Spatrick             }
311061da546Spatrick 
312061da546Spatrick             // Sanity check - this may be a function in the middle of code that
313061da546Spatrick             // has debug information, but not for this symbol.  So the line
314061da546Spatrick             // entries surrounding us won't lie inside our function. In that
315061da546Spatrick             // case, the line entry will be bigger than we are, so we do that
316061da546Spatrick             // quick check and if that is true, we just return 0.
317061da546Spatrick             if (m_type_data >= m_addr_range.GetByteSize())
318061da546Spatrick               m_type_data = 0;
319061da546Spatrick           } else {
320061da546Spatrick             // TODO: expose something in Process to figure out the
321061da546Spatrick             // size of a function prologue.
322061da546Spatrick             m_type_data = 0;
323061da546Spatrick           }
324061da546Spatrick         }
325061da546Spatrick       }
326061da546Spatrick     }
327061da546Spatrick     return m_type_data;
328061da546Spatrick   }
329061da546Spatrick   return 0;
330061da546Spatrick }
331061da546Spatrick 
Compare(ConstString name,SymbolType type) const332061da546Spatrick bool Symbol::Compare(ConstString name, SymbolType type) const {
333be691f3bSpatrick   if (type == eSymbolTypeAny || m_type == type) {
334be691f3bSpatrick     const Mangled &mangled = GetMangled();
335be691f3bSpatrick     return mangled.GetMangledName() == name ||
336be691f3bSpatrick            mangled.GetDemangledName() == name;
337be691f3bSpatrick   }
338061da546Spatrick   return false;
339061da546Spatrick }
340061da546Spatrick 
341061da546Spatrick #define ENUM_TO_CSTRING(x)                                                     \
342061da546Spatrick   case eSymbolType##x:                                                         \
343061da546Spatrick     return #x;
344061da546Spatrick 
GetTypeAsString() const345061da546Spatrick const char *Symbol::GetTypeAsString() const {
346061da546Spatrick   switch (m_type) {
347061da546Spatrick     ENUM_TO_CSTRING(Invalid);
348061da546Spatrick     ENUM_TO_CSTRING(Absolute);
349061da546Spatrick     ENUM_TO_CSTRING(Code);
350061da546Spatrick     ENUM_TO_CSTRING(Resolver);
351061da546Spatrick     ENUM_TO_CSTRING(Data);
352061da546Spatrick     ENUM_TO_CSTRING(Trampoline);
353061da546Spatrick     ENUM_TO_CSTRING(Runtime);
354061da546Spatrick     ENUM_TO_CSTRING(Exception);
355061da546Spatrick     ENUM_TO_CSTRING(SourceFile);
356061da546Spatrick     ENUM_TO_CSTRING(HeaderFile);
357061da546Spatrick     ENUM_TO_CSTRING(ObjectFile);
358061da546Spatrick     ENUM_TO_CSTRING(CommonBlock);
359061da546Spatrick     ENUM_TO_CSTRING(Block);
360061da546Spatrick     ENUM_TO_CSTRING(Local);
361061da546Spatrick     ENUM_TO_CSTRING(Param);
362061da546Spatrick     ENUM_TO_CSTRING(Variable);
363061da546Spatrick     ENUM_TO_CSTRING(VariableType);
364061da546Spatrick     ENUM_TO_CSTRING(LineEntry);
365061da546Spatrick     ENUM_TO_CSTRING(LineHeader);
366061da546Spatrick     ENUM_TO_CSTRING(ScopeBegin);
367061da546Spatrick     ENUM_TO_CSTRING(ScopeEnd);
368061da546Spatrick     ENUM_TO_CSTRING(Additional);
369061da546Spatrick     ENUM_TO_CSTRING(Compiler);
370061da546Spatrick     ENUM_TO_CSTRING(Instrumentation);
371061da546Spatrick     ENUM_TO_CSTRING(Undefined);
372061da546Spatrick     ENUM_TO_CSTRING(ObjCClass);
373061da546Spatrick     ENUM_TO_CSTRING(ObjCMetaClass);
374061da546Spatrick     ENUM_TO_CSTRING(ObjCIVar);
375061da546Spatrick     ENUM_TO_CSTRING(ReExported);
376061da546Spatrick   default:
377061da546Spatrick     break;
378061da546Spatrick   }
379061da546Spatrick   return "<unknown SymbolType>";
380061da546Spatrick }
381061da546Spatrick 
CalculateSymbolContext(SymbolContext * sc)382061da546Spatrick void Symbol::CalculateSymbolContext(SymbolContext *sc) {
383061da546Spatrick   // Symbols can reconstruct the symbol and the module in the symbol context
384061da546Spatrick   sc->symbol = this;
385061da546Spatrick   if (ValueIsAddress())
386061da546Spatrick     sc->module_sp = GetAddressRef().GetModule();
387061da546Spatrick   else
388061da546Spatrick     sc->module_sp.reset();
389061da546Spatrick }
390061da546Spatrick 
CalculateSymbolContextModule()391061da546Spatrick ModuleSP Symbol::CalculateSymbolContextModule() {
392061da546Spatrick   if (ValueIsAddress())
393061da546Spatrick     return GetAddressRef().GetModule();
394061da546Spatrick   return ModuleSP();
395061da546Spatrick }
396061da546Spatrick 
CalculateSymbolContextSymbol()397061da546Spatrick Symbol *Symbol::CalculateSymbolContextSymbol() { return this; }
398061da546Spatrick 
DumpSymbolContext(Stream * s)399061da546Spatrick void Symbol::DumpSymbolContext(Stream *s) {
400061da546Spatrick   bool dumped_module = false;
401061da546Spatrick   if (ValueIsAddress()) {
402061da546Spatrick     ModuleSP module_sp(GetAddressRef().GetModule());
403061da546Spatrick     if (module_sp) {
404061da546Spatrick       dumped_module = true;
405061da546Spatrick       module_sp->DumpSymbolContext(s);
406061da546Spatrick     }
407061da546Spatrick   }
408061da546Spatrick   if (dumped_module)
409061da546Spatrick     s->PutCString(", ");
410061da546Spatrick 
411061da546Spatrick   s->Printf("Symbol{0x%8.8x}", GetID());
412061da546Spatrick }
413061da546Spatrick 
GetByteSize() const414061da546Spatrick lldb::addr_t Symbol::GetByteSize() const { return m_addr_range.GetByteSize(); }
415061da546Spatrick 
ResolveReExportedSymbolInModuleSpec(Target & target,ConstString & reexport_name,ModuleSpec & module_spec,ModuleList & seen_modules) const416061da546Spatrick Symbol *Symbol::ResolveReExportedSymbolInModuleSpec(
417061da546Spatrick     Target &target, ConstString &reexport_name, ModuleSpec &module_spec,
418061da546Spatrick     ModuleList &seen_modules) const {
419061da546Spatrick   ModuleSP module_sp;
420061da546Spatrick   if (module_spec.GetFileSpec()) {
421061da546Spatrick     // Try searching for the module file spec first using the full path
422061da546Spatrick     module_sp = target.GetImages().FindFirstModule(module_spec);
423061da546Spatrick     if (!module_sp) {
424061da546Spatrick       // Next try and find the module by basename in case environment variables
425061da546Spatrick       // or other runtime trickery causes shared libraries to be loaded from
426061da546Spatrick       // alternate paths
427*f6aab3d8Srobert       module_spec.GetFileSpec().ClearDirectory();
428061da546Spatrick       module_sp = target.GetImages().FindFirstModule(module_spec);
429061da546Spatrick     }
430061da546Spatrick   }
431061da546Spatrick 
432061da546Spatrick   if (module_sp) {
433061da546Spatrick     // There should not be cycles in the reexport list, but we don't want to
434061da546Spatrick     // crash if there are so make sure we haven't seen this before:
435061da546Spatrick     if (!seen_modules.AppendIfNeeded(module_sp))
436061da546Spatrick       return nullptr;
437061da546Spatrick 
438061da546Spatrick     lldb_private::SymbolContextList sc_list;
439061da546Spatrick     module_sp->FindSymbolsWithNameAndType(reexport_name, eSymbolTypeAny,
440061da546Spatrick                                           sc_list);
441061da546Spatrick     const size_t num_scs = sc_list.GetSize();
442061da546Spatrick     if (num_scs > 0) {
443061da546Spatrick       for (size_t i = 0; i < num_scs; ++i) {
444061da546Spatrick         lldb_private::SymbolContext sc;
445061da546Spatrick         if (sc_list.GetContextAtIndex(i, sc)) {
446061da546Spatrick           if (sc.symbol->IsExternal())
447061da546Spatrick             return sc.symbol;
448061da546Spatrick         }
449061da546Spatrick       }
450061da546Spatrick     }
451061da546Spatrick     // If we didn't find the symbol in this module, it may be because this
452061da546Spatrick     // module re-exports some whole other library.  We have to search those as
453061da546Spatrick     // well:
454061da546Spatrick     seen_modules.Append(module_sp);
455061da546Spatrick 
456061da546Spatrick     FileSpecList reexported_libraries =
457061da546Spatrick         module_sp->GetObjectFile()->GetReExportedLibraries();
458061da546Spatrick     size_t num_reexported_libraries = reexported_libraries.GetSize();
459061da546Spatrick     for (size_t idx = 0; idx < num_reexported_libraries; idx++) {
460061da546Spatrick       ModuleSpec reexported_module_spec;
461061da546Spatrick       reexported_module_spec.GetFileSpec() =
462061da546Spatrick           reexported_libraries.GetFileSpecAtIndex(idx);
463061da546Spatrick       Symbol *result_symbol = ResolveReExportedSymbolInModuleSpec(
464061da546Spatrick           target, reexport_name, reexported_module_spec, seen_modules);
465061da546Spatrick       if (result_symbol)
466061da546Spatrick         return result_symbol;
467061da546Spatrick     }
468061da546Spatrick   }
469061da546Spatrick   return nullptr;
470061da546Spatrick }
471061da546Spatrick 
ResolveReExportedSymbol(Target & target) const472061da546Spatrick Symbol *Symbol::ResolveReExportedSymbol(Target &target) const {
473061da546Spatrick   ConstString reexport_name(GetReExportedSymbolName());
474061da546Spatrick   if (reexport_name) {
475061da546Spatrick     ModuleSpec module_spec;
476061da546Spatrick     ModuleList seen_modules;
477061da546Spatrick     module_spec.GetFileSpec() = GetReExportedSymbolSharedLibrary();
478061da546Spatrick     if (module_spec.GetFileSpec()) {
479061da546Spatrick       return ResolveReExportedSymbolInModuleSpec(target, reexport_name,
480061da546Spatrick                                                  module_spec, seen_modules);
481061da546Spatrick     }
482061da546Spatrick   }
483061da546Spatrick   return nullptr;
484061da546Spatrick }
485061da546Spatrick 
GetFileAddress() const486061da546Spatrick lldb::addr_t Symbol::GetFileAddress() const {
487061da546Spatrick   if (ValueIsAddress())
488061da546Spatrick     return GetAddressRef().GetFileAddress();
489061da546Spatrick   else
490061da546Spatrick     return LLDB_INVALID_ADDRESS;
491061da546Spatrick }
492061da546Spatrick 
GetLoadAddress(Target * target) const493061da546Spatrick lldb::addr_t Symbol::GetLoadAddress(Target *target) const {
494061da546Spatrick   if (ValueIsAddress())
495061da546Spatrick     return GetAddressRef().GetLoadAddress(target);
496061da546Spatrick   else
497061da546Spatrick     return LLDB_INVALID_ADDRESS;
498061da546Spatrick }
499061da546Spatrick 
GetName() const500be691f3bSpatrick ConstString Symbol::GetName() const { return GetMangled().GetName(); }
501061da546Spatrick 
GetNameNoArguments() const502061da546Spatrick ConstString Symbol::GetNameNoArguments() const {
503be691f3bSpatrick   return GetMangled().GetName(Mangled::ePreferDemangledWithoutArguments);
504061da546Spatrick }
505061da546Spatrick 
ResolveCallableAddress(Target & target) const506061da546Spatrick lldb::addr_t Symbol::ResolveCallableAddress(Target &target) const {
507061da546Spatrick   if (GetType() == lldb::eSymbolTypeUndefined)
508061da546Spatrick     return LLDB_INVALID_ADDRESS;
509061da546Spatrick 
510061da546Spatrick   Address func_so_addr;
511061da546Spatrick 
512061da546Spatrick   bool is_indirect = IsIndirect();
513061da546Spatrick   if (GetType() == eSymbolTypeReExported) {
514061da546Spatrick     Symbol *reexported_symbol = ResolveReExportedSymbol(target);
515061da546Spatrick     if (reexported_symbol) {
516061da546Spatrick       func_so_addr = reexported_symbol->GetAddress();
517061da546Spatrick       is_indirect = reexported_symbol->IsIndirect();
518061da546Spatrick     }
519061da546Spatrick   } else {
520061da546Spatrick     func_so_addr = GetAddress();
521061da546Spatrick     is_indirect = IsIndirect();
522061da546Spatrick   }
523061da546Spatrick 
524061da546Spatrick   if (func_so_addr.IsValid()) {
525061da546Spatrick     if (!target.GetProcessSP() && is_indirect) {
526061da546Spatrick       // can't resolve indirect symbols without calling a function...
527061da546Spatrick       return LLDB_INVALID_ADDRESS;
528061da546Spatrick     }
529061da546Spatrick 
530061da546Spatrick     lldb::addr_t load_addr =
531061da546Spatrick         func_so_addr.GetCallableLoadAddress(&target, is_indirect);
532061da546Spatrick 
533061da546Spatrick     if (load_addr != LLDB_INVALID_ADDRESS) {
534061da546Spatrick       return load_addr;
535061da546Spatrick     }
536061da546Spatrick   }
537061da546Spatrick 
538061da546Spatrick   return LLDB_INVALID_ADDRESS;
539061da546Spatrick }
540061da546Spatrick 
GetInstructions(const ExecutionContext & exe_ctx,const char * flavor,bool prefer_file_cache)541061da546Spatrick lldb::DisassemblerSP Symbol::GetInstructions(const ExecutionContext &exe_ctx,
542061da546Spatrick                                              const char *flavor,
543061da546Spatrick                                              bool prefer_file_cache) {
544061da546Spatrick   ModuleSP module_sp(m_addr_range.GetBaseAddress().GetModule());
545dda28197Spatrick   if (module_sp && exe_ctx.HasTargetScope()) {
546061da546Spatrick     return Disassembler::DisassembleRange(module_sp->GetArchitecture(), nullptr,
547dda28197Spatrick                                           flavor, exe_ctx.GetTargetRef(),
548be691f3bSpatrick                                           m_addr_range, !prefer_file_cache);
549061da546Spatrick   }
550061da546Spatrick   return lldb::DisassemblerSP();
551061da546Spatrick }
552061da546Spatrick 
GetDisassembly(const ExecutionContext & exe_ctx,const char * flavor,bool prefer_file_cache,Stream & strm)553061da546Spatrick bool Symbol::GetDisassembly(const ExecutionContext &exe_ctx, const char *flavor,
554061da546Spatrick                             bool prefer_file_cache, Stream &strm) {
555061da546Spatrick   lldb::DisassemblerSP disassembler_sp =
556061da546Spatrick       GetInstructions(exe_ctx, flavor, prefer_file_cache);
557061da546Spatrick   if (disassembler_sp) {
558061da546Spatrick     const bool show_address = true;
559061da546Spatrick     const bool show_bytes = false;
560*f6aab3d8Srobert     const bool show_control_flow_kind = false;
561*f6aab3d8Srobert     disassembler_sp->GetInstructionList().Dump(
562*f6aab3d8Srobert         &strm, show_address, show_bytes, show_control_flow_kind, &exe_ctx);
563061da546Spatrick     return true;
564061da546Spatrick   }
565061da546Spatrick   return false;
566061da546Spatrick }
567061da546Spatrick 
ContainsFileAddress(lldb::addr_t file_addr) const568061da546Spatrick bool Symbol::ContainsFileAddress(lldb::addr_t file_addr) const {
569061da546Spatrick   return m_addr_range.ContainsFileAddress(file_addr);
570061da546Spatrick }
571be691f3bSpatrick 
IsSyntheticWithAutoGeneratedName() const572be691f3bSpatrick bool Symbol::IsSyntheticWithAutoGeneratedName() const {
573be691f3bSpatrick   if (!IsSynthetic())
574be691f3bSpatrick     return false;
575be691f3bSpatrick   if (!m_mangled)
576be691f3bSpatrick     return true;
577be691f3bSpatrick   ConstString demangled = m_mangled.GetDemangledName();
578be691f3bSpatrick   return demangled.GetStringRef().startswith(GetSyntheticSymbolPrefix());
579be691f3bSpatrick }
580be691f3bSpatrick 
SynthesizeNameIfNeeded() const581be691f3bSpatrick void Symbol::SynthesizeNameIfNeeded() const {
582be691f3bSpatrick   if (m_is_synthetic && !m_mangled) {
583be691f3bSpatrick     // Synthetic symbol names don't mean anything, but they do uniquely
584be691f3bSpatrick     // identify individual symbols so we give them a unique name. The name
585be691f3bSpatrick     // starts with the synthetic symbol prefix, followed by a unique number.
586be691f3bSpatrick     // Typically the UserID of a real symbol is the symbol table index of the
587be691f3bSpatrick     // symbol in the object file's symbol table(s), so it will be the same
588be691f3bSpatrick     // every time you read in the object file. We want the same persistence for
589be691f3bSpatrick     // synthetic symbols so that users can identify them across multiple debug
590be691f3bSpatrick     // sessions, to understand crashes in those symbols and to reliably set
591be691f3bSpatrick     // breakpoints on them.
592be691f3bSpatrick     llvm::SmallString<256> name;
593be691f3bSpatrick     llvm::raw_svector_ostream os(name);
594be691f3bSpatrick     os << GetSyntheticSymbolPrefix() << GetID();
595be691f3bSpatrick     m_mangled.SetDemangledName(ConstString(os.str()));
596be691f3bSpatrick   }
597be691f3bSpatrick }
598*f6aab3d8Srobert 
Decode(const DataExtractor & data,lldb::offset_t * offset_ptr,const SectionList * section_list,const StringTableReader & strtab)599*f6aab3d8Srobert bool Symbol::Decode(const DataExtractor &data, lldb::offset_t *offset_ptr,
600*f6aab3d8Srobert                     const SectionList *section_list,
601*f6aab3d8Srobert                     const StringTableReader &strtab) {
602*f6aab3d8Srobert   if (!data.ValidOffsetForDataOfSize(*offset_ptr, 8))
603*f6aab3d8Srobert     return false;
604*f6aab3d8Srobert   m_uid = data.GetU32(offset_ptr);
605*f6aab3d8Srobert   m_type_data = data.GetU16(offset_ptr);
606*f6aab3d8Srobert   const uint16_t bitfields = data.GetU16(offset_ptr);
607*f6aab3d8Srobert   m_type_data_resolved = (1u << 15 & bitfields) != 0;
608*f6aab3d8Srobert   m_is_synthetic = (1u << 14 & bitfields) != 0;
609*f6aab3d8Srobert   m_is_debug = (1u << 13 & bitfields) != 0;
610*f6aab3d8Srobert   m_is_external = (1u << 12 & bitfields) != 0;
611*f6aab3d8Srobert   m_size_is_sibling = (1u << 11 & bitfields) != 0;
612*f6aab3d8Srobert   m_size_is_synthesized = (1u << 10 & bitfields) != 0;
613*f6aab3d8Srobert   m_size_is_valid = (1u << 9 & bitfields) != 0;
614*f6aab3d8Srobert   m_demangled_is_synthesized = (1u << 8 & bitfields) != 0;
615*f6aab3d8Srobert   m_contains_linker_annotations = (1u << 7 & bitfields) != 0;
616*f6aab3d8Srobert   m_is_weak = (1u << 6 & bitfields) != 0;
617*f6aab3d8Srobert   m_type = bitfields & 0x003f;
618*f6aab3d8Srobert   if (!m_mangled.Decode(data, offset_ptr, strtab))
619*f6aab3d8Srobert     return false;
620*f6aab3d8Srobert   if (!data.ValidOffsetForDataOfSize(*offset_ptr, 20))
621*f6aab3d8Srobert     return false;
622*f6aab3d8Srobert   const bool is_addr = data.GetU8(offset_ptr) != 0;
623*f6aab3d8Srobert   const uint64_t value = data.GetU64(offset_ptr);
624*f6aab3d8Srobert   if (is_addr) {
625*f6aab3d8Srobert     m_addr_range.GetBaseAddress().ResolveAddressUsingFileSections(
626*f6aab3d8Srobert         value, section_list);
627*f6aab3d8Srobert   } else {
628*f6aab3d8Srobert     m_addr_range.GetBaseAddress().Clear();
629*f6aab3d8Srobert     m_addr_range.GetBaseAddress().SetOffset(value);
630*f6aab3d8Srobert   }
631*f6aab3d8Srobert   m_addr_range.SetByteSize(data.GetU64(offset_ptr));
632*f6aab3d8Srobert   m_flags =  data.GetU32(offset_ptr);
633*f6aab3d8Srobert   return true;
634*f6aab3d8Srobert }
635*f6aab3d8Srobert 
636*f6aab3d8Srobert /// The encoding format for the symbol is as follows:
637*f6aab3d8Srobert ///
638*f6aab3d8Srobert /// uint32_t m_uid;
639*f6aab3d8Srobert /// uint16_t m_type_data;
640*f6aab3d8Srobert /// uint16_t bitfield_data;
641*f6aab3d8Srobert /// Mangled mangled;
642*f6aab3d8Srobert /// uint8_t is_addr;
643*f6aab3d8Srobert /// uint64_t file_addr_or_value;
644*f6aab3d8Srobert /// uint64_t size;
645*f6aab3d8Srobert /// uint32_t flags;
646*f6aab3d8Srobert ///
647*f6aab3d8Srobert /// The only tricky thing in this encoding is encoding all of the bits in the
648*f6aab3d8Srobert /// bitfields. We use a trick to store all bitfields as a 16 bit value and we
649*f6aab3d8Srobert /// do the same thing when decoding the symbol. There are test that ensure this
650*f6aab3d8Srobert /// encoding works for each individual bit. Everything else is very easy to
651*f6aab3d8Srobert /// store.
Encode(DataEncoder & file,ConstStringTable & strtab) const652*f6aab3d8Srobert void Symbol::Encode(DataEncoder &file, ConstStringTable &strtab) const {
653*f6aab3d8Srobert   file.AppendU32(m_uid);
654*f6aab3d8Srobert   file.AppendU16(m_type_data);
655*f6aab3d8Srobert   uint16_t bitfields = m_type;
656*f6aab3d8Srobert   if (m_type_data_resolved)
657*f6aab3d8Srobert     bitfields |= 1u << 15;
658*f6aab3d8Srobert   if (m_is_synthetic)
659*f6aab3d8Srobert     bitfields |= 1u << 14;
660*f6aab3d8Srobert   if (m_is_debug)
661*f6aab3d8Srobert     bitfields |= 1u << 13;
662*f6aab3d8Srobert   if (m_is_external)
663*f6aab3d8Srobert     bitfields |= 1u << 12;
664*f6aab3d8Srobert   if (m_size_is_sibling)
665*f6aab3d8Srobert     bitfields |= 1u << 11;
666*f6aab3d8Srobert   if (m_size_is_synthesized)
667*f6aab3d8Srobert     bitfields |= 1u << 10;
668*f6aab3d8Srobert   if (m_size_is_valid)
669*f6aab3d8Srobert     bitfields |= 1u << 9;
670*f6aab3d8Srobert   if (m_demangled_is_synthesized)
671*f6aab3d8Srobert     bitfields |= 1u << 8;
672*f6aab3d8Srobert   if (m_contains_linker_annotations)
673*f6aab3d8Srobert     bitfields |= 1u << 7;
674*f6aab3d8Srobert   if (m_is_weak)
675*f6aab3d8Srobert     bitfields |= 1u << 6;
676*f6aab3d8Srobert   file.AppendU16(bitfields);
677*f6aab3d8Srobert   m_mangled.Encode(file, strtab);
678*f6aab3d8Srobert   // A symbol's value might be an address, or it might be a constant. If the
679*f6aab3d8Srobert   // symbol's base address doesn't have a section, then it is a constant value.
680*f6aab3d8Srobert   // If it does have a section, we will encode the file address and re-resolve
681*f6aab3d8Srobert   // the address when we decode it.
682*f6aab3d8Srobert   bool is_addr = m_addr_range.GetBaseAddress().GetSection().get() != nullptr;
683*f6aab3d8Srobert   file.AppendU8(is_addr);
684*f6aab3d8Srobert   file.AppendU64(m_addr_range.GetBaseAddress().GetFileAddress());
685*f6aab3d8Srobert   file.AppendU64(m_addr_range.GetByteSize());
686*f6aab3d8Srobert   file.AppendU32(m_flags);
687*f6aab3d8Srobert }
688*f6aab3d8Srobert 
operator ==(const Symbol & rhs) const689*f6aab3d8Srobert bool Symbol::operator==(const Symbol &rhs) const {
690*f6aab3d8Srobert   if (m_uid != rhs.m_uid)
691*f6aab3d8Srobert     return false;
692*f6aab3d8Srobert   if (m_type_data != rhs.m_type_data)
693*f6aab3d8Srobert     return false;
694*f6aab3d8Srobert   if (m_type_data_resolved != rhs.m_type_data_resolved)
695*f6aab3d8Srobert     return false;
696*f6aab3d8Srobert   if (m_is_synthetic != rhs.m_is_synthetic)
697*f6aab3d8Srobert     return false;
698*f6aab3d8Srobert   if (m_is_debug != rhs.m_is_debug)
699*f6aab3d8Srobert     return false;
700*f6aab3d8Srobert   if (m_is_external != rhs.m_is_external)
701*f6aab3d8Srobert     return false;
702*f6aab3d8Srobert   if (m_size_is_sibling != rhs.m_size_is_sibling)
703*f6aab3d8Srobert     return false;
704*f6aab3d8Srobert   if (m_size_is_synthesized != rhs.m_size_is_synthesized)
705*f6aab3d8Srobert     return false;
706*f6aab3d8Srobert   if (m_size_is_valid != rhs.m_size_is_valid)
707*f6aab3d8Srobert     return false;
708*f6aab3d8Srobert   if (m_demangled_is_synthesized != rhs.m_demangled_is_synthesized)
709*f6aab3d8Srobert     return false;
710*f6aab3d8Srobert   if (m_contains_linker_annotations != rhs.m_contains_linker_annotations)
711*f6aab3d8Srobert     return false;
712*f6aab3d8Srobert   if (m_is_weak != rhs.m_is_weak)
713*f6aab3d8Srobert     return false;
714*f6aab3d8Srobert   if (m_type != rhs.m_type)
715*f6aab3d8Srobert     return false;
716*f6aab3d8Srobert   if (m_mangled != rhs.m_mangled)
717*f6aab3d8Srobert     return false;
718*f6aab3d8Srobert   if (m_addr_range.GetBaseAddress() != rhs.m_addr_range.GetBaseAddress())
719*f6aab3d8Srobert     return false;
720*f6aab3d8Srobert   if (m_addr_range.GetByteSize() != rhs.m_addr_range.GetByteSize())
721*f6aab3d8Srobert     return false;
722*f6aab3d8Srobert   if (m_flags != rhs.m_flags)
723*f6aab3d8Srobert     return false;
724*f6aab3d8Srobert   return true;
725*f6aab3d8Srobert }
726