1 //===-- SymbolVendorELF.cpp ----------------------------------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "SymbolVendorELF.h"
11 
12 //#include <libxml/parser.h>
13 // #include <libxml/tree.h>
14 #include <string.h>
15 
16 // #include <AvailabilityMacros.h>
17 
18 #include "lldb/Core/Module.h"
19 #include "lldb/Core/ModuleSpec.h"
20 #include "lldb/Core/PluginManager.h"
21 #include "lldb/Core/Section.h"
22 #include "lldb/Core/StreamString.h"
23 #include "lldb/Core/Timer.h"
24 #include "lldb/Host/Host.h"
25 #include "lldb/Host/Symbols.h"
26 #include "lldb/Symbol/ObjectFile.h"
27 
28 using namespace lldb;
29 using namespace lldb_private;
30 
31 //----------------------------------------------------------------------
32 // SymbolVendorELF constructor
33 //----------------------------------------------------------------------
SymbolVendorELF(const lldb::ModuleSP & module_sp)34 SymbolVendorELF::SymbolVendorELF(const lldb::ModuleSP &module_sp) :
35     SymbolVendor (module_sp)
36 {
37 }
38 
39 //----------------------------------------------------------------------
40 // Destructor
41 //----------------------------------------------------------------------
~SymbolVendorELF()42 SymbolVendorELF::~SymbolVendorELF()
43 {
44 }
45 
46 void
Initialize()47 SymbolVendorELF::Initialize()
48 {
49     PluginManager::RegisterPlugin (GetPluginNameStatic(),
50                                    GetPluginDescriptionStatic(),
51                                    CreateInstance);
52 }
53 
54 void
Terminate()55 SymbolVendorELF::Terminate()
56 {
57     PluginManager::UnregisterPlugin (CreateInstance);
58 }
59 
60 
61 lldb_private::ConstString
GetPluginNameStatic()62 SymbolVendorELF::GetPluginNameStatic()
63 {
64     static ConstString g_name("ELF");
65     return g_name;
66 }
67 
68 const char *
GetPluginDescriptionStatic()69 SymbolVendorELF::GetPluginDescriptionStatic()
70 {
71     return "Symbol vendor for ELF that looks for dSYM files that match executables.";
72 }
73 
74 
75 
76 //----------------------------------------------------------------------
77 // CreateInstance
78 //
79 // Platforms can register a callback to use when creating symbol
80 // vendors to allow for complex debug information file setups, and to
81 // also allow for finding separate debug information files.
82 //----------------------------------------------------------------------
83 SymbolVendor*
CreateInstance(const lldb::ModuleSP & module_sp,lldb_private::Stream * feedback_strm)84 SymbolVendorELF::CreateInstance (const lldb::ModuleSP &module_sp, lldb_private::Stream *feedback_strm)
85 {
86     if (!module_sp)
87         return NULL;
88 
89     ObjectFile *obj_file = module_sp->GetObjectFile();
90     if (!obj_file)
91         return NULL;
92 
93     static ConstString obj_file_elf("elf");
94     ConstString obj_name = obj_file->GetPluginName();
95     if (obj_name != obj_file_elf)
96         return NULL;
97 
98     lldb_private::UUID uuid;
99     if (!obj_file->GetUUID (&uuid))
100         return NULL;
101 
102     // Get the .gnu_debuglink file (if specified).
103     FileSpecList file_spec_list = obj_file->GetDebugSymbolFilePaths();
104 
105     // If the module specified a filespec, use it first.
106     FileSpec debug_symbol_fspec (module_sp->GetSymbolFileFileSpec());
107     if (debug_symbol_fspec)
108         file_spec_list.Insert (0, debug_symbol_fspec);
109 
110     // If we have no debug symbol files, then nothing to do.
111     if (file_spec_list.IsEmpty())
112         return NULL;
113 
114     Timer scoped_timer (__PRETTY_FUNCTION__,
115                         "SymbolVendorELF::CreateInstance (module = %s)",
116                         module_sp->GetFileSpec().GetPath().c_str());
117 
118     for (size_t idx = 0; idx < file_spec_list.GetSize(); ++idx)
119     {
120         ModuleSpec module_spec;
121         const FileSpec fspec = file_spec_list.GetFileSpecAtIndex (idx);
122 
123         module_spec.GetFileSpec() = obj_file->GetFileSpec();
124         module_spec.GetFileSpec().ResolvePath();
125         module_spec.GetSymbolFileSpec() = fspec;
126         module_spec.GetUUID() = uuid;
127         FileSpec dsym_fspec = Symbols::LocateExecutableSymbolFile (module_spec);
128         if (dsym_fspec)
129         {
130             DataBufferSP dsym_file_data_sp;
131             lldb::offset_t dsym_file_data_offset = 0;
132             ObjectFileSP dsym_objfile_sp = ObjectFile::FindPlugin(module_sp, &dsym_fspec, 0, dsym_fspec.GetByteSize(), dsym_file_data_sp, dsym_file_data_offset);
133             if (dsym_objfile_sp)
134             {
135                 // This objfile is for debugging purposes. Sadly, ObjectFileELF won't be able
136                 // to figure this out consistently as the symbol file may not have stripped the
137                 // code sections, etc.
138                 dsym_objfile_sp->SetType (ObjectFile::eTypeDebugInfo);
139 
140                 SymbolVendorELF* symbol_vendor = new SymbolVendorELF(module_sp);
141                 if (symbol_vendor)
142                 {
143                     // Get the module unified section list and add our debug sections to that.
144                     SectionList *module_section_list = module_sp->GetSectionList();
145                     SectionList *objfile_section_list = dsym_objfile_sp->GetSectionList();
146 
147                     static const SectionType g_sections[] =
148                     {
149                         eSectionTypeDWARFDebugAranges,
150                         eSectionTypeDWARFDebugInfo,
151                         eSectionTypeDWARFDebugAbbrev,
152                         eSectionTypeDWARFDebugFrame,
153                         eSectionTypeDWARFDebugLine,
154                         eSectionTypeDWARFDebugStr,
155                         eSectionTypeDWARFDebugLoc,
156                         eSectionTypeDWARFDebugMacInfo,
157                         eSectionTypeDWARFDebugPubNames,
158                         eSectionTypeDWARFDebugPubTypes,
159                         eSectionTypeDWARFDebugRanges,
160                         eSectionTypeELFSymbolTable,
161                     };
162                     for (size_t idx = 0; idx < sizeof(g_sections) / sizeof(g_sections[0]); ++idx)
163                     {
164                         SectionType section_type = g_sections[idx];
165                         SectionSP section_sp (objfile_section_list->FindSectionByType (section_type, true));
166                         if (section_sp)
167                         {
168                             SectionSP module_section_sp (module_section_list->FindSectionByType (section_type, true));
169                             if (module_section_sp)
170                                 module_section_list->ReplaceSection (module_section_sp->GetID(), section_sp);
171                             else
172                                 module_section_list->AddSection (section_sp);
173                         }
174                     }
175 
176                     symbol_vendor->AddSymbolFileRepresentation (dsym_objfile_sp);
177                     return symbol_vendor;
178                 }
179             }
180         }
181     }
182     return NULL;
183 }
184 
185 //------------------------------------------------------------------
186 // PluginInterface protocol
187 //------------------------------------------------------------------
188 ConstString
GetPluginName()189 SymbolVendorELF::GetPluginName()
190 {
191     return GetPluginNameStatic();
192 }
193 
194 uint32_t
GetPluginVersion()195 SymbolVendorELF::GetPluginVersion()
196 {
197     return 1;
198 }
199 
200