blob: 66e3e72999e66d6e2f95f02a1b6069e3633b43b8 [file] [log] [blame]
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001//===-- SymbolFileDWARF.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 "SymbolFileDWARF.h"
11
12// Other libraries and framework includes
Sean Callanancc427fa2011-07-30 02:42:06 +000013#include "llvm/Support/Casting.h"
14
Robert Flackeb83fab2015-05-15 18:59:59 +000015#include "lldb/Core/ArchSpec.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000016#include "lldb/Core/Module.h"
Sean Callananf0c5aeb2015-04-20 16:31:29 +000017#include "lldb/Core/ModuleList.h"
18#include "lldb/Core/ModuleSpec.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000019#include "lldb/Core/PluginManager.h"
20#include "lldb/Core/RegularExpression.h"
21#include "lldb/Core/Scalar.h"
22#include "lldb/Core/Section.h"
Greg Claytonc685f8e2010-09-15 04:15:46 +000023#include "lldb/Core/StreamFile.h"
Jim Ingham318c9f22011-08-26 19:44:13 +000024#include "lldb/Core/StreamString.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000025#include "lldb/Core/Timer.h"
26#include "lldb/Core/Value.h"
27
Sean Callanan4dbb2712015-09-25 20:35:58 +000028#include "Plugins/ExpressionParser/Clang/ClangModulesDeclVendor.h"
Sean Callananf0c5aeb2015-04-20 16:31:29 +000029
Oleksiy Vyalov5d9c50b2015-07-21 02:09:42 +000030#include "lldb/Host/FileSystem.h"
Greg Clayton20568dd2011-10-13 23:13:20 +000031#include "lldb/Host/Host.h"
32
Oleksiy Vyalovabb5a352015-07-29 22:18:16 +000033#include "lldb/Interpreter/OptionValueFileSpecList.h"
34#include "lldb/Interpreter/OptionValueProperties.h"
35
Chris Lattner30fdc8d2010-06-08 16:52:24 +000036#include "lldb/Symbol/Block.h"
Bruce Mitchener937e3962015-09-21 16:56:08 +000037#include "lldb/Symbol/ClangASTContext.h"
Zachary Turnerd133f6a2016-03-28 22:53:41 +000038#include "lldb/Symbol/ClangUtil.h"
39#include "lldb/Symbol/CompileUnit.h"
Paul Hermand628cbb2015-09-15 23:44:17 +000040#include "lldb/Symbol/CompilerDecl.h"
41#include "lldb/Symbol/CompilerDeclContext.h"
Siva Chandrad8335e92015-12-16 00:22:08 +000042#include "lldb/Symbol/DebugMacros.h"
Zachary Turnerd133f6a2016-03-28 22:53:41 +000043#include "lldb/Symbol/LineTable.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000044#include "lldb/Symbol/ObjectFile.h"
45#include "lldb/Symbol/SymbolVendor.h"
Zachary Turnerd133f6a2016-03-28 22:53:41 +000046#include "lldb/Symbol/TypeMap.h"
Bruce Mitchener937e3962015-09-21 16:56:08 +000047#include "lldb/Symbol/TypeSystem.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000048#include "lldb/Symbol/VariableList.h"
49
Jim Inghamaa816b82015-09-02 01:59:14 +000050#include "Plugins/Language/CPlusPlus/CPlusPlusLanguage.h"
51#include "Plugins/Language/ObjC/ObjCLanguage.h"
Jim Inghamb7f6b2f2011-09-08 22:13:49 +000052
Jim Ingham0e0984e2015-09-02 01:06:46 +000053#include "lldb/Target/Language.h"
54
Tamas Berghammer2ff88702015-10-23 10:34:49 +000055#include "lldb/Utility/TaskPool.h"
56
Greg Clayton261ac3f2015-08-28 01:01:03 +000057#include "DWARFASTParser.h"
Zachary Turnerd133f6a2016-03-28 22:53:41 +000058#include "DWARFASTParserClang.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000059#include "DWARFCompileUnit.h"
Zachary Turnerd133f6a2016-03-28 22:53:41 +000060#include "DWARFDIECollection.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000061#include "DWARFDebugAbbrev.h"
62#include "DWARFDebugAranges.h"
63#include "DWARFDebugInfo.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000064#include "DWARFDebugLine.h"
Siva Chandrad8335e92015-12-16 00:22:08 +000065#include "DWARFDebugMacro.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000066#include "DWARFDebugPubnames.h"
67#include "DWARFDebugRanges.h"
Greg Claytona8022fa2012-04-24 21:22:41 +000068#include "DWARFDeclContext.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000069#include "DWARFFormValue.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000070#include "LogChannelDWARF.h"
Greg Clayton450e3f32010-10-12 02:24:53 +000071#include "SymbolFileDWARFDebugMap.h"
Zachary Turnerd133f6a2016-03-28 22:53:41 +000072#include "SymbolFileDWARFDwo.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000073
74#include <map>
75
Matthew Gardinere81df3b2014-08-26 06:57:23 +000076#include <ctype.h>
77#include <string.h>
78
Greg Clayton62742b12010-11-11 01:09:45 +000079//#define ENABLE_DEBUG_PRINTF // COMMENT OUT THIS LINE PRIOR TO CHECKIN
Greg Claytonc93237c2010-10-01 20:48:32 +000080
81#ifdef ENABLE_DEBUG_PRINTF
82#include <stdio.h>
Ed Mastea0191d12013-10-17 20:42:56 +000083#define DEBUG_PRINTF(fmt, ...) printf(fmt, __VA_ARGS__)
Greg Claytonc93237c2010-10-01 20:48:32 +000084#else
85#define DEBUG_PRINTF(fmt, ...)
86#endif
87
Chris Lattner30fdc8d2010-06-08 16:52:24 +000088using namespace lldb;
89using namespace lldb_private;
90
Greg Clayton219cf312012-03-30 00:51:13 +000091//static inline bool
92//child_requires_parent_class_union_or_struct_to_be_completed (dw_tag_t tag)
93//{
94// switch (tag)
95// {
96// default:
97// break;
98// case DW_TAG_subprogram:
99// case DW_TAG_inlined_subroutine:
100// case DW_TAG_class_type:
101// case DW_TAG_structure_type:
102// case DW_TAG_union_type:
103// return true;
104// }
105// return false;
106//}
107//
Oleksiy Vyalovabb5a352015-07-29 22:18:16 +0000108
109namespace {
110
111 PropertyDefinition
112 g_properties[] =
113 {
114 { "comp-dir-symlink-paths" , OptionValue::eTypeFileSpecList, true, 0 , nullptr, nullptr, "If the DW_AT_comp_dir matches any of these paths the symbolic links will be resolved at DWARF parse time." },
115 { nullptr , OptionValue::eTypeInvalid , false, 0, nullptr, nullptr, nullptr }
116 };
117
118 enum
119 {
120 ePropertySymLinkPaths
121 };
122
123
124 class PluginProperties : public Properties
125 {
126 public:
127 static ConstString
128 GetSettingName()
129 {
130 return SymbolFileDWARF::GetPluginNameStatic();
131 }
132
133 PluginProperties()
134 {
135 m_collection_sp.reset (new OptionValueProperties(GetSettingName()));
136 m_collection_sp->Initialize(g_properties);
137 }
138
139 FileSpecList&
140 GetSymLinkPaths()
141 {
142 OptionValueFileSpecList *option_value = m_collection_sp->GetPropertyAtIndexAsOptionValueFileSpecList(nullptr, true, ePropertySymLinkPaths);
143 assert(option_value);
144 return option_value->GetCurrentValue();
145 }
146
147 };
148
149 typedef std::shared_ptr<PluginProperties> SymbolFileDWARFPropertiesSP;
150
151 static const SymbolFileDWARFPropertiesSP&
152 GetGlobalPluginProperties()
153 {
154 static const auto g_settings_sp(std::make_shared<PluginProperties>());
155 return g_settings_sp;
156 }
157
158} // anonymous namespace end
159
160
Matthew Gardinere81df3b2014-08-26 06:57:23 +0000161static const char*
162removeHostnameFromPathname(const char* path_from_dwarf)
163{
164 if (!path_from_dwarf || !path_from_dwarf[0])
165 {
166 return path_from_dwarf;
167 }
Enrico Granata99e5e222015-07-27 21:27:02 +0000168
Matthew Gardinere81df3b2014-08-26 06:57:23 +0000169 const char *colon_pos = strchr(path_from_dwarf, ':');
Enrico Granata99e5e222015-07-27 21:27:02 +0000170 if (nullptr == colon_pos)
171 {
172 return path_from_dwarf;
173 }
174
175 const char *slash_pos = strchr(path_from_dwarf, '/');
176 if (slash_pos && (slash_pos < colon_pos))
Matthew Gardinere81df3b2014-08-26 06:57:23 +0000177 {
178 return path_from_dwarf;
179 }
180
181 // check whether we have a windows path, and so the first character
182 // is a drive-letter not a hostname.
183 if (
184 colon_pos == path_from_dwarf + 1 &&
185 isalpha(*path_from_dwarf) &&
186 strlen(path_from_dwarf) > 2 &&
187 '\\' == path_from_dwarf[2])
188 {
189 return path_from_dwarf;
190 }
Enrico Granata99e5e222015-07-27 21:27:02 +0000191
Matthew Gardinere81df3b2014-08-26 06:57:23 +0000192 return colon_pos + 1;
193}
194
Oleksiy Vyalov5d9c50b2015-07-21 02:09:42 +0000195static const char*
196resolveCompDir(const char* path_from_dwarf)
197{
198 if (!path_from_dwarf)
199 return nullptr;
200
201 // DWARF2/3 suggests the form hostname:pathname for compilation directory.
202 // Remove the host part if present.
203 const char* local_path = removeHostnameFromPathname(path_from_dwarf);
204 if (!local_path)
205 return nullptr;
206
207 bool is_symlink = false;
Oleksiy Vyalovabb5a352015-07-29 22:18:16 +0000208 FileSpec local_path_spec(local_path, false);
209 const auto& file_specs = GetGlobalPluginProperties()->GetSymLinkPaths();
210 for (size_t i = 0; i < file_specs.GetSize() && !is_symlink; ++i)
211 is_symlink = FileSpec::Equal(file_specs.GetFileSpecAtIndex(i), local_path_spec, true);
Oleksiy Vyalov5d9c50b2015-07-21 02:09:42 +0000212
213 if (!is_symlink)
214 return local_path;
215
Oleksiy Vyalov5d9c50b2015-07-21 02:09:42 +0000216 if (!local_path_spec.IsSymbolicLink())
217 return local_path;
218
219 FileSpec resolved_local_path_spec;
220 const auto error = FileSystem::Readlink(local_path_spec, resolved_local_path_spec);
221 if (error.Success())
222 return resolved_local_path_spec.GetCString();
223
224 return nullptr;
225}
226
227
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000228void
229SymbolFileDWARF::Initialize()
230{
231 LogChannelDWARF::Initialize();
232 PluginManager::RegisterPlugin (GetPluginNameStatic(),
233 GetPluginDescriptionStatic(),
Oleksiy Vyalovabb5a352015-07-29 22:18:16 +0000234 CreateInstance,
235 DebuggerInitialize);
236}
237
238void
239SymbolFileDWARF::DebuggerInitialize(Debugger &debugger)
240{
241 if (!PluginManager::GetSettingForSymbolFilePlugin(debugger, PluginProperties::GetSettingName()))
242 {
243 const bool is_global_setting = true;
244 PluginManager::CreateSettingForSymbolFilePlugin(debugger,
245 GetGlobalPluginProperties()->GetValueProperties(),
246 ConstString ("Properties for the dwarf symbol-file plug-in."),
247 is_global_setting);
248 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000249}
250
251void
252SymbolFileDWARF::Terminate()
253{
254 PluginManager::UnregisterPlugin (CreateInstance);
255 LogChannelDWARF::Initialize();
256}
257
258
Greg Clayton57abc5d2013-05-10 21:47:16 +0000259lldb_private::ConstString
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000260SymbolFileDWARF::GetPluginNameStatic()
261{
Greg Clayton57abc5d2013-05-10 21:47:16 +0000262 static ConstString g_name("dwarf");
263 return g_name;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000264}
265
266const char *
267SymbolFileDWARF::GetPluginDescriptionStatic()
268{
269 return "DWARF and DWARF3 debug symbol file reader.";
270}
271
272
273SymbolFile*
274SymbolFileDWARF::CreateInstance (ObjectFile* obj_file)
275{
276 return new SymbolFileDWARF(obj_file);
277}
278
Greg Clayton2d95dc9b2010-11-10 04:57:04 +0000279TypeList *
280SymbolFileDWARF::GetTypeList ()
281{
Greg Clayton1f746072012-08-29 21:13:06 +0000282 if (GetDebugMapSymfile ())
Greg Clayton2d95dc9b2010-11-10 04:57:04 +0000283 return m_debug_map_symfile->GetTypeList();
284 return m_obj_file->GetModule()->GetTypeList();
285
286}
Greg Claytonf02500c2013-06-18 22:51:05 +0000287void
Greg Clayton6071e6f2015-08-26 22:57:51 +0000288SymbolFileDWARF::GetTypes (const DWARFDIE &die,
Greg Claytonf02500c2013-06-18 22:51:05 +0000289 dw_offset_t min_die_offset,
290 dw_offset_t max_die_offset,
291 uint32_t type_mask,
292 TypeSet &type_set)
293{
Greg Clayton6071e6f2015-08-26 22:57:51 +0000294 if (die)
Greg Claytonf02500c2013-06-18 22:51:05 +0000295 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000296 const dw_offset_t die_offset = die.GetOffset();
297
298 if (die_offset >= max_die_offset)
299 return;
300
301 if (die_offset >= min_die_offset)
Greg Claytonf02500c2013-06-18 22:51:05 +0000302 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000303 const dw_tag_t tag = die.Tag();
Greg Claytonf02500c2013-06-18 22:51:05 +0000304
Greg Clayton6071e6f2015-08-26 22:57:51 +0000305 bool add_type = false;
306
307 switch (tag)
Greg Claytonf02500c2013-06-18 22:51:05 +0000308 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000309 case DW_TAG_array_type: add_type = (type_mask & eTypeClassArray ) != 0; break;
310 case DW_TAG_unspecified_type:
311 case DW_TAG_base_type: add_type = (type_mask & eTypeClassBuiltin ) != 0; break;
312 case DW_TAG_class_type: add_type = (type_mask & eTypeClassClass ) != 0; break;
313 case DW_TAG_structure_type: add_type = (type_mask & eTypeClassStruct ) != 0; break;
314 case DW_TAG_union_type: add_type = (type_mask & eTypeClassUnion ) != 0; break;
315 case DW_TAG_enumeration_type: add_type = (type_mask & eTypeClassEnumeration ) != 0; break;
316 case DW_TAG_subroutine_type:
317 case DW_TAG_subprogram:
318 case DW_TAG_inlined_subroutine: add_type = (type_mask & eTypeClassFunction ) != 0; break;
319 case DW_TAG_pointer_type: add_type = (type_mask & eTypeClassPointer ) != 0; break;
320 case DW_TAG_rvalue_reference_type:
321 case DW_TAG_reference_type: add_type = (type_mask & eTypeClassReference ) != 0; break;
322 case DW_TAG_typedef: add_type = (type_mask & eTypeClassTypedef ) != 0; break;
323 case DW_TAG_ptr_to_member_type: add_type = (type_mask & eTypeClassMemberPointer ) != 0; break;
324 }
Greg Claytonf02500c2013-06-18 22:51:05 +0000325
Greg Clayton6071e6f2015-08-26 22:57:51 +0000326 if (add_type)
327 {
328 const bool assert_not_being_parsed = true;
329 Type *type = ResolveTypeUID (die, assert_not_being_parsed);
330 if (type)
Greg Claytonf02500c2013-06-18 22:51:05 +0000331 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000332 if (type_set.find(type) == type_set.end())
333 type_set.insert(type);
Greg Claytonf02500c2013-06-18 22:51:05 +0000334 }
335 }
Greg Clayton6071e6f2015-08-26 22:57:51 +0000336 }
337
338 for (DWARFDIE child_die = die.GetFirstChild();
339 child_die.IsValid();
340 child_die = child_die.GetSibling())
341 {
342 GetTypes (child_die, min_die_offset, max_die_offset, type_mask, type_set);
Greg Claytonf02500c2013-06-18 22:51:05 +0000343 }
344 }
345}
346
347size_t
348SymbolFileDWARF::GetTypes (SymbolContextScope *sc_scope,
349 uint32_t type_mask,
350 TypeList &type_list)
351
352{
353 TypeSet type_set;
354
355 CompileUnit *comp_unit = NULL;
356 DWARFCompileUnit* dwarf_cu = NULL;
357 if (sc_scope)
358 comp_unit = sc_scope->CalculateSymbolContextCompileUnit();
359
360 if (comp_unit)
361 {
362 dwarf_cu = GetDWARFCompileUnit(comp_unit);
363 if (dwarf_cu == 0)
364 return 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +0000365 GetTypes (dwarf_cu->DIE(),
Greg Claytonf02500c2013-06-18 22:51:05 +0000366 dwarf_cu->GetOffset(),
367 dwarf_cu->GetNextCompileUnitOffset(),
368 type_mask,
369 type_set);
370 }
371 else
372 {
373 DWARFDebugInfo* info = DebugInfo();
374 if (info)
375 {
376 const size_t num_cus = info->GetNumCompileUnits();
377 for (size_t cu_idx=0; cu_idx<num_cus; ++cu_idx)
378 {
379 dwarf_cu = info->GetCompileUnitAtIndex(cu_idx);
380 if (dwarf_cu)
381 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000382 GetTypes (dwarf_cu->DIE(),
Greg Claytonf02500c2013-06-18 22:51:05 +0000383 0,
384 UINT32_MAX,
385 type_mask,
386 type_set);
387 }
388 }
389 }
390 }
Greg Clayton6071e6f2015-08-26 22:57:51 +0000391
Bruce Mitchener3ad353f2015-09-24 03:54:50 +0000392 std::set<CompilerType> compiler_type_set;
Greg Claytonf02500c2013-06-18 22:51:05 +0000393 size_t num_types_added = 0;
394 for (Type *type : type_set)
395 {
Bruce Mitchener3ad353f2015-09-24 03:54:50 +0000396 CompilerType compiler_type = type->GetForwardCompilerType ();
397 if (compiler_type_set.find(compiler_type) == compiler_type_set.end())
Greg Clayton0fc4f312013-06-20 01:23:18 +0000398 {
Bruce Mitchener3ad353f2015-09-24 03:54:50 +0000399 compiler_type_set.insert(compiler_type);
Greg Clayton0fc4f312013-06-20 01:23:18 +0000400 type_list.Insert (type->shared_from_this());
401 ++num_types_added;
402 }
Greg Claytonf02500c2013-06-18 22:51:05 +0000403 }
404 return num_types_added;
405}
406
Greg Clayton2d95dc9b2010-11-10 04:57:04 +0000407
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000408//----------------------------------------------------------------------
409// Gets the first parent that is a lexical block, function or inlined
410// subroutine, or compile unit.
411//----------------------------------------------------------------------
Greg Clayton6071e6f2015-08-26 22:57:51 +0000412DWARFDIE
413SymbolFileDWARF::GetParentSymbolContextDIE(const DWARFDIE &child_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000414{
Greg Clayton6071e6f2015-08-26 22:57:51 +0000415 DWARFDIE die;
416 for (die = child_die.GetParent(); die; die = die.GetParent())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000417 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000418 dw_tag_t tag = die.Tag();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000419
420 switch (tag)
421 {
422 case DW_TAG_compile_unit:
423 case DW_TAG_subprogram:
424 case DW_TAG_inlined_subroutine:
425 case DW_TAG_lexical_block:
426 return die;
427 }
428 }
Greg Clayton6071e6f2015-08-26 22:57:51 +0000429 return DWARFDIE();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000430}
431
432
Greg Clayton450e3f32010-10-12 02:24:53 +0000433SymbolFileDWARF::SymbolFileDWARF(ObjectFile* objfile) :
434 SymbolFile (objfile),
Greg Clayton81c22f62011-10-19 18:09:39 +0000435 UserID (0), // Used by SymbolFileDWARFDebugMap to when this class parses .o files to contain the .o file index/ID
Greg Clayton1f746072012-08-29 21:13:06 +0000436 m_debug_map_module_wp (),
Greg Clayton450e3f32010-10-12 02:24:53 +0000437 m_debug_map_symfile (NULL),
Greg Claytond4a2b372011-09-12 23:21:58 +0000438 m_data_debug_abbrev (),
439 m_data_debug_aranges (),
440 m_data_debug_frame (),
441 m_data_debug_info (),
442 m_data_debug_line (),
Siva Chandrad8335e92015-12-16 00:22:08 +0000443 m_data_debug_macro (),
Greg Claytond4a2b372011-09-12 23:21:58 +0000444 m_data_debug_loc (),
445 m_data_debug_ranges (),
446 m_data_debug_str (),
Greg Clayton17674402011-09-28 17:06:40 +0000447 m_data_apple_names (),
448 m_data_apple_types (),
Greg Clayton7f995132011-10-04 22:41:51 +0000449 m_data_apple_namespaces (),
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000450 m_abbr(),
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000451 m_info(),
452 m_line(),
Greg Clayton7f995132011-10-04 22:41:51 +0000453 m_apple_names_ap (),
454 m_apple_types_ap (),
455 m_apple_namespaces_ap (),
Greg Clayton5009f9d2011-10-27 17:55:14 +0000456 m_apple_objc_ap (),
Greg Claytonc685f8e2010-09-15 04:15:46 +0000457 m_function_basename_index(),
458 m_function_fullname_index(),
459 m_function_method_index(),
460 m_function_selector_index(),
Greg Clayton450e3f32010-10-12 02:24:53 +0000461 m_objc_class_selectors_index(),
Greg Claytonc685f8e2010-09-15 04:15:46 +0000462 m_global_index(),
Greg Clayton69b04882010-10-15 02:03:22 +0000463 m_type_index(),
464 m_namespace_index(),
Greg Clayton6beaaa62011-01-17 03:46:26 +0000465 m_indexed (false),
Greg Clayton97fbc342011-10-20 22:30:33 +0000466 m_using_apple_tables (false),
Sean Callananf0c5aeb2015-04-20 16:31:29 +0000467 m_fetched_external_modules (false),
Greg Claytonc7f03b62012-01-12 04:33:28 +0000468 m_supports_DW_AT_APPLE_objc_complete_type (eLazyBoolCalculate),
Greg Clayton1c9e5ac2011-02-09 19:06:17 +0000469 m_ranges(),
470 m_unique_ast_type_map ()
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000471{
472}
473
474SymbolFileDWARF::~SymbolFileDWARF()
475{
Greg Clayton6beaaa62011-01-17 03:46:26 +0000476}
477
478static const ConstString &
479GetDWARFMachOSegmentName ()
480{
481 static ConstString g_dwarf_section_name ("__DWARF");
482 return g_dwarf_section_name;
483}
484
Greg Claytone576ab22011-02-15 00:19:15 +0000485UniqueDWARFASTTypeMap &
486SymbolFileDWARF::GetUniqueDWARFASTTypeMap ()
487{
Greg Clayton1f746072012-08-29 21:13:06 +0000488 if (GetDebugMapSymfile ())
Greg Claytone576ab22011-02-15 00:19:15 +0000489 return m_debug_map_symfile->GetUniqueDWARFASTTypeMap ();
490 return m_unique_ast_type_map;
491}
492
Greg Clayton8b4edba2015-08-14 20:02:05 +0000493TypeSystem *
494SymbolFileDWARF::GetTypeSystemForLanguage (LanguageType language)
495{
496 SymbolFileDWARFDebugMap * debug_map_symfile = GetDebugMapSymfile ();
Ryan Brown57bee1e2015-09-14 22:45:11 +0000497 TypeSystem *type_system;
Greg Clayton8b4edba2015-08-14 20:02:05 +0000498 if (debug_map_symfile)
Ryan Brown57bee1e2015-09-14 22:45:11 +0000499 {
500 type_system = debug_map_symfile->GetTypeSystemForLanguage(language);
501 }
Greg Clayton8b4edba2015-08-14 20:02:05 +0000502 else
Ryan Brown57bee1e2015-09-14 22:45:11 +0000503 {
504 type_system = m_obj_file->GetModule()->GetTypeSystemForLanguage(language);
505 if (type_system)
506 type_system->SetSymbolFile(this);
507 }
508 return type_system;
Greg Clayton8b4edba2015-08-14 20:02:05 +0000509}
510
Greg Clayton6beaaa62011-01-17 03:46:26 +0000511void
512SymbolFileDWARF::InitializeObject()
513{
Greg Claytone72dfb32012-02-24 01:59:29 +0000514 ModuleSP module_sp (m_obj_file->GetModule());
515 if (module_sp)
Greg Clayton6beaaa62011-01-17 03:46:26 +0000516 {
Greg Clayton3046e662013-07-10 01:23:25 +0000517 const SectionList *section_list = module_sp->GetSectionList();
Greg Clayton6beaaa62011-01-17 03:46:26 +0000518 const Section* section = section_list->FindSectionByName(GetDWARFMachOSegmentName ()).get();
519
520 // Memory map the DWARF mach-o segment so we have everything mmap'ed
521 // to keep our heap memory usage down.
522 if (section)
Greg Claytonc9660542012-02-05 02:38:54 +0000523 m_obj_file->MemoryMapSectionData(section, m_dwarf_data);
Greg Clayton6beaaa62011-01-17 03:46:26 +0000524 }
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000525
Greg Clayton4d01ace2011-09-29 16:58:15 +0000526 get_apple_names_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000527 if (m_data_apple_names.m_data.GetByteSize() > 0)
Greg Clayton7f995132011-10-04 22:41:51 +0000528 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000529 m_apple_names_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_names.m_data,
530 get_debug_str_data(),
531 ".apple_names"));
Greg Clayton97fbc342011-10-20 22:30:33 +0000532 if (m_apple_names_ap->IsValid())
533 m_using_apple_tables = true;
534 else
Greg Clayton7f995132011-10-04 22:41:51 +0000535 m_apple_names_ap.reset();
536 }
Greg Clayton4d01ace2011-09-29 16:58:15 +0000537 get_apple_types_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000538 if (m_data_apple_types.m_data.GetByteSize() > 0)
Greg Clayton7f995132011-10-04 22:41:51 +0000539 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000540 m_apple_types_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_types.m_data,
541 get_debug_str_data(),
542 ".apple_types"));
Greg Clayton97fbc342011-10-20 22:30:33 +0000543 if (m_apple_types_ap->IsValid())
544 m_using_apple_tables = true;
545 else
Greg Clayton7f995132011-10-04 22:41:51 +0000546 m_apple_types_ap.reset();
547 }
548
549 get_apple_namespaces_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000550 if (m_data_apple_namespaces.m_data.GetByteSize() > 0)
Greg Clayton7f995132011-10-04 22:41:51 +0000551 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000552 m_apple_namespaces_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_namespaces.m_data,
553 get_debug_str_data(),
554 ".apple_namespaces"));
Greg Clayton97fbc342011-10-20 22:30:33 +0000555 if (m_apple_namespaces_ap->IsValid())
556 m_using_apple_tables = true;
557 else
Greg Clayton7f995132011-10-04 22:41:51 +0000558 m_apple_namespaces_ap.reset();
559 }
Greg Clayton4d01ace2011-09-29 16:58:15 +0000560
Greg Clayton5009f9d2011-10-27 17:55:14 +0000561 get_apple_objc_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000562 if (m_data_apple_objc.m_data.GetByteSize() > 0)
Greg Clayton5009f9d2011-10-27 17:55:14 +0000563 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000564 m_apple_objc_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_objc.m_data,
565 get_debug_str_data(),
566 ".apple_objc"));
Greg Clayton5009f9d2011-10-27 17:55:14 +0000567 if (m_apple_objc_ap->IsValid())
568 m_using_apple_tables = true;
569 else
570 m_apple_objc_ap.reset();
571 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000572}
573
574bool
575SymbolFileDWARF::SupportedVersion(uint16_t version)
576{
Greg Claytonabcbfe52013-04-04 00:00:36 +0000577 return version == 2 || version == 3 || version == 4;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000578}
579
580uint32_t
Sean Callananbfaf54d2011-12-03 04:38:43 +0000581SymbolFileDWARF::CalculateAbilities ()
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000582{
583 uint32_t abilities = 0;
584 if (m_obj_file != NULL)
585 {
586 const Section* section = NULL;
587 const SectionList *section_list = m_obj_file->GetSectionList();
588 if (section_list == NULL)
589 return 0;
590
591 uint64_t debug_abbrev_file_size = 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000592 uint64_t debug_info_file_size = 0;
593 uint64_t debug_line_file_size = 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000594
Greg Clayton6beaaa62011-01-17 03:46:26 +0000595 section = section_list->FindSectionByName(GetDWARFMachOSegmentName ()).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000596
597 if (section)
Greg Clayton4ceb9982010-07-21 22:54:26 +0000598 section_list = &section->GetChildren ();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000599
Greg Clayton4ceb9982010-07-21 22:54:26 +0000600 section = section_list->FindSectionByType (eSectionTypeDWARFDebugInfo, true).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000601 if (section != NULL)
602 {
Greg Clayton47037bc2012-03-27 02:40:46 +0000603 debug_info_file_size = section->GetFileSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000604
Greg Clayton4ceb9982010-07-21 22:54:26 +0000605 section = section_list->FindSectionByType (eSectionTypeDWARFDebugAbbrev, true).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000606 if (section)
Greg Clayton47037bc2012-03-27 02:40:46 +0000607 debug_abbrev_file_size = section->GetFileSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000608
Greg Clayton4ceb9982010-07-21 22:54:26 +0000609 section = section_list->FindSectionByType (eSectionTypeDWARFDebugLine, true).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000610 if (section)
Greg Clayton47037bc2012-03-27 02:40:46 +0000611 debug_line_file_size = section->GetFileSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000612 }
Greg Clayton6c596612012-05-18 21:47:20 +0000613 else
614 {
615 const char *symfile_dir_cstr = m_obj_file->GetFileSpec().GetDirectory().GetCString();
616 if (symfile_dir_cstr)
617 {
618 if (strcasestr(symfile_dir_cstr, ".dsym"))
619 {
620 if (m_obj_file->GetType() == ObjectFile::eTypeDebugInfo)
621 {
622 // We have a dSYM file that didn't have a any debug info.
623 // If the string table has a size of 1, then it was made from
624 // an executable with no debug info, or from an executable that
625 // was stripped.
626 section = section_list->FindSectionByType (eSectionTypeDWARFDebugStr, true).get();
627 if (section && section->GetFileSize() == 1)
628 {
629 m_obj_file->GetModule()->ReportWarning ("empty dSYM file detected, dSYM was created with an executable with no debug info.");
630 }
631 }
632 }
633 }
634 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000635
636 if (debug_abbrev_file_size > 0 && debug_info_file_size > 0)
637 abilities |= CompileUnits | Functions | Blocks | GlobalVariables | LocalVariables | VariableTypes;
638
639 if (debug_line_file_size > 0)
640 abilities |= LineTables;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000641 }
642 return abilities;
643}
644
Ed Masteeeae7212013-10-24 20:43:47 +0000645const DWARFDataExtractor&
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000646SymbolFileDWARF::GetCachedSectionData (lldb::SectionType sect_type, DWARFDataSegment& data_segment)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000647{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000648 std::call_once(data_segment.m_flag,
649 &SymbolFileDWARF::LoadSectionData,
650 this,
651 sect_type,
652 std::ref(data_segment.m_data));
653 return data_segment.m_data;
654}
655
656void
657SymbolFileDWARF::LoadSectionData (lldb::SectionType sect_type, DWARFDataExtractor& data)
658{
659 ModuleSP module_sp (m_obj_file->GetModule());
660 const SectionList *section_list = module_sp->GetSectionList();
661 if (section_list)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000662 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000663 SectionSP section_sp (section_list->FindSectionByType(sect_type, true));
664 if (section_sp)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000665 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000666 // See if we memory mapped the DWARF segment?
667 if (m_dwarf_data.GetByteSize())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000668 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000669 data.SetData(m_dwarf_data, section_sp->GetOffset(), section_sp->GetFileSize());
670 }
671 else
672 {
673 if (m_obj_file->ReadSectionData(section_sp.get(), data) == 0)
674 data.Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000675 }
676 }
677 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000678}
679
Ed Masteeeae7212013-10-24 20:43:47 +0000680const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000681SymbolFileDWARF::get_debug_abbrev_data()
682{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000683 return GetCachedSectionData (eSectionTypeDWARFDebugAbbrev, m_data_debug_abbrev);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000684}
685
Ed Masteeeae7212013-10-24 20:43:47 +0000686const DWARFDataExtractor&
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000687SymbolFileDWARF::get_debug_addr_data()
688{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000689 return GetCachedSectionData (eSectionTypeDWARFDebugAddr, m_data_debug_addr);
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000690}
691
692const DWARFDataExtractor&
Greg Claytond4a2b372011-09-12 23:21:58 +0000693SymbolFileDWARF::get_debug_aranges_data()
694{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000695 return GetCachedSectionData (eSectionTypeDWARFDebugAranges, m_data_debug_aranges);
Greg Claytond4a2b372011-09-12 23:21:58 +0000696}
697
Ed Masteeeae7212013-10-24 20:43:47 +0000698const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000699SymbolFileDWARF::get_debug_frame_data()
700{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000701 return GetCachedSectionData (eSectionTypeDWARFDebugFrame, m_data_debug_frame);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000702}
703
Ed Masteeeae7212013-10-24 20:43:47 +0000704const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000705SymbolFileDWARF::get_debug_info_data()
706{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000707 return GetCachedSectionData (eSectionTypeDWARFDebugInfo, m_data_debug_info);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000708}
709
Ed Masteeeae7212013-10-24 20:43:47 +0000710const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000711SymbolFileDWARF::get_debug_line_data()
712{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000713 return GetCachedSectionData (eSectionTypeDWARFDebugLine, m_data_debug_line);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000714}
715
Ed Masteeeae7212013-10-24 20:43:47 +0000716const DWARFDataExtractor&
Siva Chandrad8335e92015-12-16 00:22:08 +0000717SymbolFileDWARF::get_debug_macro_data()
718{
719 return GetCachedSectionData (eSectionTypeDWARFDebugMacro, m_data_debug_macro);
720}
721
722const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000723SymbolFileDWARF::get_debug_loc_data()
724{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000725 return GetCachedSectionData (eSectionTypeDWARFDebugLoc, m_data_debug_loc);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000726}
727
Ed Masteeeae7212013-10-24 20:43:47 +0000728const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000729SymbolFileDWARF::get_debug_ranges_data()
730{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000731 return GetCachedSectionData (eSectionTypeDWARFDebugRanges, m_data_debug_ranges);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000732}
733
Ed Masteeeae7212013-10-24 20:43:47 +0000734const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000735SymbolFileDWARF::get_debug_str_data()
736{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000737 return GetCachedSectionData (eSectionTypeDWARFDebugStr, m_data_debug_str);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000738}
739
Ed Masteeeae7212013-10-24 20:43:47 +0000740const DWARFDataExtractor&
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000741SymbolFileDWARF::get_debug_str_offsets_data()
742{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000743 return GetCachedSectionData (eSectionTypeDWARFDebugStrOffsets, m_data_debug_str_offsets);
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000744}
745
746const DWARFDataExtractor&
Greg Clayton17674402011-09-28 17:06:40 +0000747SymbolFileDWARF::get_apple_names_data()
Greg Claytonf9eec202011-09-01 23:16:13 +0000748{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000749 return GetCachedSectionData (eSectionTypeDWARFAppleNames, m_data_apple_names);
Greg Claytonf9eec202011-09-01 23:16:13 +0000750}
751
Ed Masteeeae7212013-10-24 20:43:47 +0000752const DWARFDataExtractor&
Greg Clayton17674402011-09-28 17:06:40 +0000753SymbolFileDWARF::get_apple_types_data()
Greg Claytonf9eec202011-09-01 23:16:13 +0000754{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000755 return GetCachedSectionData (eSectionTypeDWARFAppleTypes, m_data_apple_types);
Greg Claytonf9eec202011-09-01 23:16:13 +0000756}
757
Ed Masteeeae7212013-10-24 20:43:47 +0000758const DWARFDataExtractor&
Greg Clayton7f995132011-10-04 22:41:51 +0000759SymbolFileDWARF::get_apple_namespaces_data()
760{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000761 return GetCachedSectionData (eSectionTypeDWARFAppleNamespaces, m_data_apple_namespaces);
Greg Clayton5009f9d2011-10-27 17:55:14 +0000762}
763
Ed Masteeeae7212013-10-24 20:43:47 +0000764const DWARFDataExtractor&
Greg Clayton5009f9d2011-10-27 17:55:14 +0000765SymbolFileDWARF::get_apple_objc_data()
766{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000767 return GetCachedSectionData (eSectionTypeDWARFAppleObjC, m_data_apple_objc);
Greg Clayton7f995132011-10-04 22:41:51 +0000768}
769
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000770
771DWARFDebugAbbrev*
772SymbolFileDWARF::DebugAbbrev()
773{
774 if (m_abbr.get() == NULL)
775 {
Ed Masteeeae7212013-10-24 20:43:47 +0000776 const DWARFDataExtractor &debug_abbrev_data = get_debug_abbrev_data();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000777 if (debug_abbrev_data.GetByteSize() > 0)
778 {
779 m_abbr.reset(new DWARFDebugAbbrev());
780 if (m_abbr.get())
781 m_abbr->Parse(debug_abbrev_data);
782 }
783 }
784 return m_abbr.get();
785}
786
787const DWARFDebugAbbrev*
788SymbolFileDWARF::DebugAbbrev() const
789{
790 return m_abbr.get();
791}
792
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000793
794DWARFDebugInfo*
795SymbolFileDWARF::DebugInfo()
796{
797 if (m_info.get() == NULL)
798 {
Saleem Abdulrasool324a1032014-04-04 04:06:10 +0000799 Timer scoped_timer(__PRETTY_FUNCTION__, "%s this = %p",
800 __PRETTY_FUNCTION__, static_cast<void*>(this));
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000801 if (get_debug_info_data().GetByteSize() > 0)
802 {
803 m_info.reset(new DWARFDebugInfo());
804 if (m_info.get())
805 {
806 m_info->SetDwarfData(this);
807 }
808 }
809 }
810 return m_info.get();
811}
812
813const DWARFDebugInfo*
814SymbolFileDWARF::DebugInfo() const
815{
816 return m_info.get();
817}
818
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000819DWARFCompileUnit*
Greg Clayton1f746072012-08-29 21:13:06 +0000820SymbolFileDWARF::GetDWARFCompileUnit(lldb_private::CompileUnit *comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000821{
Greg Claytonea4a5bb2015-09-04 22:29:46 +0000822 if (!comp_unit)
823 return nullptr;
824
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000825 DWARFDebugInfo* info = DebugInfo();
Greg Clayton1f746072012-08-29 21:13:06 +0000826 if (info)
827 {
828 if (GetDebugMapSymfile ())
829 {
830 // The debug map symbol file made the compile units for this DWARF
831 // file which is .o file with DWARF in it, and we should have
832 // only 1 compile unit which is at offset zero in the DWARF.
833 // TODO: modify to support LTO .o files where each .o file might
834 // have multiple DW_TAG_compile_unit tags.
Greg Clayton68c00bd2015-02-05 02:10:29 +0000835
Greg Clayton6071e6f2015-08-26 22:57:51 +0000836 DWARFCompileUnit *dwarf_cu = info->GetCompileUnit(0);
Greg Clayton68c00bd2015-02-05 02:10:29 +0000837 if (dwarf_cu && dwarf_cu->GetUserData() == NULL)
838 dwarf_cu->SetUserData(comp_unit);
839 return dwarf_cu;
Greg Clayton1f746072012-08-29 21:13:06 +0000840 }
841 else
842 {
843 // Just a normal DWARF file whose user ID for the compile unit is
844 // the DWARF offset itself
Greg Clayton68c00bd2015-02-05 02:10:29 +0000845
Greg Clayton6071e6f2015-08-26 22:57:51 +0000846 DWARFCompileUnit *dwarf_cu = info->GetCompileUnit((dw_offset_t)comp_unit->GetID());
Greg Clayton68c00bd2015-02-05 02:10:29 +0000847 if (dwarf_cu && dwarf_cu->GetUserData() == NULL)
848 dwarf_cu->SetUserData(comp_unit);
849 return dwarf_cu;
850
Greg Clayton1f746072012-08-29 21:13:06 +0000851 }
852 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000853 return NULL;
854}
855
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000856
857DWARFDebugRanges*
858SymbolFileDWARF::DebugRanges()
859{
860 if (m_ranges.get() == NULL)
861 {
Saleem Abdulrasool324a1032014-04-04 04:06:10 +0000862 Timer scoped_timer(__PRETTY_FUNCTION__, "%s this = %p",
863 __PRETTY_FUNCTION__, static_cast<void*>(this));
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000864 if (get_debug_ranges_data().GetByteSize() > 0)
865 {
866 m_ranges.reset(new DWARFDebugRanges());
867 if (m_ranges.get())
868 m_ranges->Extract(this);
869 }
870 }
871 return m_ranges.get();
872}
873
874const DWARFDebugRanges*
875SymbolFileDWARF::DebugRanges() const
876{
877 return m_ranges.get();
878}
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000879
Greg Clayton53eb1c22012-04-02 22:59:12 +0000880lldb::CompUnitSP
881SymbolFileDWARF::ParseCompileUnit (DWARFCompileUnit* dwarf_cu, uint32_t cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000882{
Greg Clayton53eb1c22012-04-02 22:59:12 +0000883 CompUnitSP cu_sp;
884 if (dwarf_cu)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000885 {
Greg Clayton53eb1c22012-04-02 22:59:12 +0000886 CompileUnit *comp_unit = (CompileUnit*)dwarf_cu->GetUserData();
887 if (comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000888 {
Greg Clayton53eb1c22012-04-02 22:59:12 +0000889 // We already parsed this compile unit, had out a shared pointer to it
890 cu_sp = comp_unit->shared_from_this();
891 }
892 else
893 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +0000894 if (dwarf_cu->GetSymbolFileDWARF() != this)
895 {
896 return dwarf_cu->GetSymbolFileDWARF()->ParseCompileUnit(dwarf_cu, cu_idx);
897 }
898 else if (GetDebugMapSymfile ())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000899 {
Greg Clayton1f746072012-08-29 21:13:06 +0000900 // Let the debug map create the compile unit
901 cu_sp = m_debug_map_symfile->GetCompileUnit(this);
902 dwarf_cu->SetUserData(cu_sp.get());
903 }
904 else
905 {
906 ModuleSP module_sp (m_obj_file->GetModule());
907 if (module_sp)
Greg Clayton53eb1c22012-04-02 22:59:12 +0000908 {
Greg Clayton5ce1a842015-08-27 18:09:44 +0000909 const DWARFDIE cu_die = dwarf_cu->GetCompileUnitDIEOnly ();
Greg Clayton1f746072012-08-29 21:13:06 +0000910 if (cu_die)
Greg Clayton53eb1c22012-04-02 22:59:12 +0000911 {
Greg Clayton5ce1a842015-08-27 18:09:44 +0000912 FileSpec cu_file_spec{cu_die.GetName(), false};
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000913 if (cu_file_spec)
Greg Clayton1f746072012-08-29 21:13:06 +0000914 {
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000915 // If we have a full path to the compile unit, we don't need to resolve
916 // the file. This can be expensive e.g. when the source files are NFS mounted.
Chaoren Lin372e9062015-06-09 17:54:27 +0000917 if (cu_file_spec.IsRelative())
Greg Clayton53eb1c22012-04-02 22:59:12 +0000918 {
Greg Clayton5ce1a842015-08-27 18:09:44 +0000919 const char *cu_comp_dir{cu_die.GetAttributeValueAsString(DW_AT_comp_dir, nullptr)};
Oleksiy Vyalov5d9c50b2015-07-21 02:09:42 +0000920 cu_file_spec.PrependPathComponent(resolveCompDir(cu_comp_dir));
Greg Clayton1f746072012-08-29 21:13:06 +0000921 }
922
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000923 std::string remapped_file;
924 if (module_sp->RemapSourceFile(cu_file_spec.GetCString(), remapped_file))
925 cu_file_spec.SetFile(remapped_file, false);
David Srbeckyd515e942015-07-08 14:00:04 +0000926 }
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000927
Greg Clayton5ce1a842015-08-27 18:09:44 +0000928 LanguageType cu_language = DWARFCompileUnit::LanguageTypeFromDWARF(cu_die.GetAttributeValueAsUnsigned(DW_AT_language, 0));
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000929
Jason Molendac7099582015-07-31 05:47:00 +0000930 bool is_optimized = dwarf_cu->GetIsOptimized ();
David Srbeckyd515e942015-07-08 14:00:04 +0000931 cu_sp.reset(new CompileUnit (module_sp,
932 dwarf_cu,
933 cu_file_spec,
Greg Clayton6071e6f2015-08-26 22:57:51 +0000934 dwarf_cu->GetID(),
Jason Molenda6ab659a2015-07-29 00:42:47 +0000935 cu_language,
936 is_optimized));
David Srbeckyd515e942015-07-08 14:00:04 +0000937 if (cu_sp)
938 {
939 // If we just created a compile unit with an invalid file spec, try and get the
940 // first entry in the supports files from the line table as that should be the
941 // compile unit.
942 if (!cu_file_spec)
Greg Clayton1f746072012-08-29 21:13:06 +0000943 {
David Srbeckyd515e942015-07-08 14:00:04 +0000944 cu_file_spec = cu_sp->GetSupportFiles().GetFileSpecAtIndex(1);
945 if (cu_file_spec)
946 {
947 (FileSpec &)(*cu_sp) = cu_file_spec;
948 // Also fix the invalid file spec which was copied from the compile unit.
949 cu_sp->GetSupportFiles().Replace(0, cu_file_spec);
950 }
Greg Clayton53eb1c22012-04-02 22:59:12 +0000951 }
David Srbeckyd515e942015-07-08 14:00:04 +0000952
953 dwarf_cu->SetUserData(cu_sp.get());
954
955 // Figure out the compile unit index if we weren't given one
956 if (cu_idx == UINT32_MAX)
957 DebugInfo()->GetCompileUnit(dwarf_cu->GetOffset(), &cu_idx);
958
959 m_obj_file->GetModule()->GetSymbolVendor()->SetCompileUnitAtIndex(cu_idx, cu_sp);
Greg Clayton53eb1c22012-04-02 22:59:12 +0000960 }
961 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000962 }
963 }
964 }
965 }
Greg Clayton53eb1c22012-04-02 22:59:12 +0000966 return cu_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000967}
968
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000969uint32_t
970SymbolFileDWARF::GetNumCompileUnits()
971{
972 DWARFDebugInfo* info = DebugInfo();
973 if (info)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000974 return info->GetNumCompileUnits();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000975 return 0;
976}
977
978CompUnitSP
979SymbolFileDWARF::ParseCompileUnitAtIndex(uint32_t cu_idx)
980{
Greg Clayton53eb1c22012-04-02 22:59:12 +0000981 CompUnitSP cu_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000982 DWARFDebugInfo* info = DebugInfo();
983 if (info)
984 {
Greg Clayton53eb1c22012-04-02 22:59:12 +0000985 DWARFCompileUnit* dwarf_cu = info->GetCompileUnitAtIndex(cu_idx);
986 if (dwarf_cu)
987 cu_sp = ParseCompileUnit(dwarf_cu, cu_idx);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000988 }
Greg Clayton53eb1c22012-04-02 22:59:12 +0000989 return cu_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000990}
991
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000992Function *
Greg Clayton6071e6f2015-08-26 22:57:51 +0000993SymbolFileDWARF::ParseCompileUnitFunction (const SymbolContext& sc, const DWARFDIE &die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000994{
Greg Clayton6071e6f2015-08-26 22:57:51 +0000995 if (die.IsValid())
996 {
997 TypeSystem *type_system = GetTypeSystemForLanguage(die.GetCU()->GetLanguageType());
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000998
Greg Clayton6071e6f2015-08-26 22:57:51 +0000999 if (type_system)
1000 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00001001 DWARFASTParser *dwarf_ast = type_system->GetDWARFParser();
1002 if (dwarf_ast)
1003 return dwarf_ast->ParseFunctionFromDWARF(sc, die);
Greg Clayton6071e6f2015-08-26 22:57:51 +00001004 }
1005 }
1006 return nullptr;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001007}
1008
Greg Clayton9422dd62013-03-04 21:46:16 +00001009bool
1010SymbolFileDWARF::FixupAddress (Address &addr)
1011{
1012 SymbolFileDWARFDebugMap * debug_map_symfile = GetDebugMapSymfile ();
1013 if (debug_map_symfile)
1014 {
1015 return debug_map_symfile->LinkOSOAddress(addr);
1016 }
1017 // This is a normal DWARF file, no address fixups need to happen
1018 return true;
1019}
Greg Clayton1f746072012-08-29 21:13:06 +00001020lldb::LanguageType
1021SymbolFileDWARF::ParseCompileUnitLanguage (const SymbolContext& sc)
1022{
1023 assert (sc.comp_unit);
1024 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
1025 if (dwarf_cu)
Greg Clayton5ce1a842015-08-27 18:09:44 +00001026 return dwarf_cu->GetLanguageType();
1027 else
1028 return eLanguageTypeUnknown;
Greg Clayton1f746072012-08-29 21:13:06 +00001029}
1030
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001031size_t
1032SymbolFileDWARF::ParseCompileUnitFunctions(const SymbolContext &sc)
1033{
1034 assert (sc.comp_unit);
1035 size_t functions_added = 0;
Greg Clayton1f746072012-08-29 21:13:06 +00001036 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001037 if (dwarf_cu)
1038 {
1039 DWARFDIECollection function_dies;
Greg Clayton1f746072012-08-29 21:13:06 +00001040 const size_t num_functions = dwarf_cu->AppendDIEsWithTag (DW_TAG_subprogram, function_dies);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001041 size_t func_idx;
Greg Clayton1f746072012-08-29 21:13:06 +00001042 for (func_idx = 0; func_idx < num_functions; ++func_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001043 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001044 DWARFDIE die = function_dies.GetDIEAtIndex(func_idx);
1045 if (sc.comp_unit->FindFunctionByUID (die.GetID()).get() == NULL)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001046 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001047 if (ParseCompileUnitFunction(sc, die))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001048 ++functions_added;
1049 }
1050 }
1051 //FixupTypes();
1052 }
1053 return functions_added;
1054}
1055
1056bool
1057SymbolFileDWARF::ParseCompileUnitSupportFiles (const SymbolContext& sc, FileSpecList& support_files)
1058{
1059 assert (sc.comp_unit);
Greg Clayton1f746072012-08-29 21:13:06 +00001060 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Greg Claytonda2455b2012-11-01 17:28:37 +00001061 if (dwarf_cu)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001062 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001063 const DWARFDIE cu_die = dwarf_cu->GetCompileUnitDIEOnly();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001064
Greg Claytonda2455b2012-11-01 17:28:37 +00001065 if (cu_die)
1066 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001067 const char * cu_comp_dir = resolveCompDir(cu_die.GetAttributeValueAsString(DW_AT_comp_dir, nullptr));
Greg Clayton5ce1a842015-08-27 18:09:44 +00001068 const dw_offset_t stmt_list = cu_die.GetAttributeValueAsUnsigned(DW_AT_stmt_list, DW_INVALID_OFFSET);
Tamas Berghammer1966ac32016-01-11 14:56:05 +00001069 if (stmt_list != DW_INVALID_OFFSET)
1070 {
1071 // All file indexes in DWARF are one based and a file of index zero is
1072 // supposed to be the compile unit itself.
1073 support_files.Append (*sc.comp_unit);
1074 return DWARFDebugLine::ParseSupportFiles(sc.comp_unit->GetModule(),
1075 get_debug_line_data(),
1076 cu_comp_dir,
1077 stmt_list,
1078 support_files);
1079 }
Greg Claytonda2455b2012-11-01 17:28:37 +00001080 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001081 }
1082 return false;
1083}
1084
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001085bool
1086SymbolFileDWARF::ParseImportedModules (const lldb_private::SymbolContext &sc, std::vector<lldb_private::ConstString> &imported_modules)
1087{
1088 assert (sc.comp_unit);
1089 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
1090 if (dwarf_cu)
1091 {
1092 if (ClangModulesDeclVendor::LanguageSupportsClangModules(sc.comp_unit->GetLanguage()))
1093 {
1094 UpdateExternalModuleListIfNeeded();
Sean Callananb0300a42016-01-14 21:46:09 +00001095
1096 if (sc.comp_unit)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001097 {
Sean Callananb0300a42016-01-14 21:46:09 +00001098 const DWARFDIE die = dwarf_cu->GetCompileUnitDIEOnly();
1099
1100 if (die)
1101 {
1102 for (DWARFDIE child_die = die.GetFirstChild();
1103 child_die;
1104 child_die = child_die.GetSibling())
1105 {
1106 if (child_die.Tag() == DW_TAG_imported_declaration)
1107 {
1108 if (DWARFDIE module_die = child_die.GetReferencedDIE(DW_AT_import))
1109 {
1110 if (module_die.Tag() == DW_TAG_module)
1111 {
1112 if (const char *name = module_die.GetAttributeValueAsString(DW_AT_name, nullptr))
1113 {
1114 ConstString const_name(name);
1115 imported_modules.push_back(const_name);
1116 }
1117 }
1118 }
1119 }
1120 }
1121 }
1122 }
1123 else
1124 {
1125 for (const auto &pair : m_external_type_modules)
1126 {
1127 imported_modules.push_back(pair.first);
1128 }
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001129 }
1130 }
1131 }
1132 return false;
1133}
1134
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001135struct ParseDWARFLineTableCallbackInfo
1136{
1137 LineTable* line_table;
Greg Clayton7b0992d2013-04-18 22:45:39 +00001138 std::unique_ptr<LineSequence> sequence_ap;
Jaydeep Patil44d07fc2015-09-22 06:36:56 +00001139 lldb::addr_t addr_mask;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001140};
1141
1142//----------------------------------------------------------------------
1143// ParseStatementTableCallback
1144//----------------------------------------------------------------------
1145static void
1146ParseDWARFLineTableCallback(dw_offset_t offset, const DWARFDebugLine::State& state, void* userData)
1147{
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001148 if (state.row == DWARFDebugLine::State::StartParsingLineTable)
1149 {
1150 // Just started parsing the line table
1151 }
1152 else if (state.row == DWARFDebugLine::State::DoneParsingLineTable)
1153 {
1154 // Done parsing line table, nothing to do for the cleanup
1155 }
1156 else
1157 {
1158 ParseDWARFLineTableCallbackInfo* info = (ParseDWARFLineTableCallbackInfo*)userData;
Greg Clayton9422dd62013-03-04 21:46:16 +00001159 LineTable* line_table = info->line_table;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001160
Greg Clayton9422dd62013-03-04 21:46:16 +00001161 // If this is our first time here, we need to create a
1162 // sequence container.
1163 if (!info->sequence_ap.get())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001164 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001165 info->sequence_ap.reset(line_table->CreateLineSequenceContainer());
1166 assert(info->sequence_ap.get());
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001167 }
Greg Clayton9422dd62013-03-04 21:46:16 +00001168 line_table->AppendLineEntryToSequence (info->sequence_ap.get(),
Jaydeep Patil44d07fc2015-09-22 06:36:56 +00001169 state.address & info->addr_mask,
Greg Clayton9422dd62013-03-04 21:46:16 +00001170 state.line,
1171 state.column,
1172 state.file,
1173 state.is_stmt,
1174 state.basic_block,
1175 state.prologue_end,
1176 state.epilogue_begin,
1177 state.end_sequence);
1178 if (state.end_sequence)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001179 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001180 // First, put the current sequence into the line table.
1181 line_table->InsertSequence(info->sequence_ap.get());
1182 // Then, empty it to prepare for the next sequence.
1183 info->sequence_ap->Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001184 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001185 }
1186}
1187
1188bool
1189SymbolFileDWARF::ParseCompileUnitLineTable (const SymbolContext &sc)
1190{
1191 assert (sc.comp_unit);
1192 if (sc.comp_unit->GetLineTable() != NULL)
1193 return true;
1194
Greg Clayton1f746072012-08-29 21:13:06 +00001195 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001196 if (dwarf_cu)
1197 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001198 const DWARFDIE dwarf_cu_die = dwarf_cu->GetCompileUnitDIEOnly();
Greg Clayton129d12c2011-11-28 03:29:03 +00001199 if (dwarf_cu_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001200 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001201 const dw_offset_t cu_line_offset = dwarf_cu_die.GetAttributeValueAsUnsigned(DW_AT_stmt_list, DW_INVALID_OFFSET);
Greg Clayton129d12c2011-11-28 03:29:03 +00001202 if (cu_line_offset != DW_INVALID_OFFSET)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001203 {
Greg Clayton7b0992d2013-04-18 22:45:39 +00001204 std::unique_ptr<LineTable> line_table_ap(new LineTable(sc.comp_unit));
Greg Clayton129d12c2011-11-28 03:29:03 +00001205 if (line_table_ap.get())
1206 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001207 ParseDWARFLineTableCallbackInfo info;
1208 info.line_table = line_table_ap.get();
Jaydeep Patil44d07fc2015-09-22 06:36:56 +00001209
1210 /*
1211 * MIPS:
1212 * The SymbolContext may not have a valid target, thus we may not be able
1213 * to call Address::GetOpcodeLoadAddress() which would clear the bit #0
1214 * for MIPS. Use ArchSpec to clear the bit #0.
1215 */
1216 ArchSpec arch;
1217 GetObjectFile()->GetArchitecture(arch);
1218 switch (arch.GetMachine())
1219 {
1220 case llvm::Triple::mips:
1221 case llvm::Triple::mipsel:
1222 case llvm::Triple::mips64:
1223 case llvm::Triple::mips64el:
1224 info.addr_mask = ~((lldb::addr_t)1);
1225 break;
1226 default:
1227 info.addr_mask = ~((lldb::addr_t)0);
1228 break;
1229 }
1230
Greg Claytonc7bece562013-01-25 18:06:21 +00001231 lldb::offset_t offset = cu_line_offset;
Greg Clayton129d12c2011-11-28 03:29:03 +00001232 DWARFDebugLine::ParseStatementTable(get_debug_line_data(), &offset, ParseDWARFLineTableCallback, &info);
Greg Clayton9422dd62013-03-04 21:46:16 +00001233 if (m_debug_map_symfile)
1234 {
1235 // We have an object file that has a line table with addresses
1236 // that are not linked. We need to link the line table and convert
1237 // the addresses that are relative to the .o file into addresses
1238 // for the main executable.
1239 sc.comp_unit->SetLineTable (m_debug_map_symfile->LinkOSOLineTable (this, line_table_ap.get()));
1240 }
1241 else
1242 {
1243 sc.comp_unit->SetLineTable(line_table_ap.release());
1244 return true;
1245 }
Greg Clayton129d12c2011-11-28 03:29:03 +00001246 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001247 }
1248 }
1249 }
1250 return false;
1251}
1252
Siva Chandrad8335e92015-12-16 00:22:08 +00001253lldb_private::DebugMacrosSP
1254SymbolFileDWARF::ParseDebugMacros(lldb::offset_t *offset)
1255{
1256 auto iter = m_debug_macros_map.find(*offset);
1257 if (iter != m_debug_macros_map.end())
1258 return iter->second;
1259
1260 const DWARFDataExtractor &debug_macro_data = get_debug_macro_data();
1261 if (debug_macro_data.GetByteSize() == 0)
1262 return DebugMacrosSP();
1263
1264 lldb_private::DebugMacrosSP debug_macros_sp(new lldb_private::DebugMacros());
1265 m_debug_macros_map[*offset] = debug_macros_sp;
1266
1267 const DWARFDebugMacroHeader &header = DWARFDebugMacroHeader::ParseHeader(debug_macro_data, offset);
1268 DWARFDebugMacroEntry::ReadMacroEntries(
1269 debug_macro_data, get_debug_str_data(), header.OffsetIs64Bit(), offset, this, debug_macros_sp);
1270
1271 return debug_macros_sp;
1272}
1273
1274bool
1275SymbolFileDWARF::ParseCompileUnitDebugMacros(const SymbolContext& sc)
1276{
1277 assert (sc.comp_unit);
1278
1279 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
1280 if (dwarf_cu == nullptr)
1281 return false;
1282
1283 const DWARFDIE dwarf_cu_die = dwarf_cu->GetCompileUnitDIEOnly();
1284 if (!dwarf_cu_die)
1285 return false;
1286
1287 lldb::offset_t sect_offset = dwarf_cu_die.GetAttributeValueAsUnsigned(DW_AT_macros, DW_INVALID_OFFSET);
1288 if (sect_offset == DW_INVALID_OFFSET)
1289 sect_offset = dwarf_cu_die.GetAttributeValueAsUnsigned(DW_AT_GNU_macros, DW_INVALID_OFFSET);
1290 if (sect_offset == DW_INVALID_OFFSET)
1291 return false;
1292
1293 sc.comp_unit->SetDebugMacros(ParseDebugMacros(&sect_offset));
1294
1295 return true;
1296}
1297
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001298size_t
Greg Clayton6071e6f2015-08-26 22:57:51 +00001299SymbolFileDWARF::ParseFunctionBlocks (const SymbolContext& sc,
1300 Block *parent_block,
1301 const DWARFDIE &orig_die,
1302 addr_t subprogram_low_pc,
1303 uint32_t depth)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001304{
1305 size_t blocks_added = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00001306 DWARFDIE die = orig_die;
1307 while (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001308 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001309 dw_tag_t tag = die.Tag();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001310
1311 switch (tag)
1312 {
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001313 case DW_TAG_inlined_subroutine:
Greg Claytonb4d37332011-08-12 16:22:48 +00001314 case DW_TAG_subprogram:
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001315 case DW_TAG_lexical_block:
1316 {
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001317 Block *block = NULL;
Greg Claytondd7feaf2011-08-12 17:54:33 +00001318 if (tag == DW_TAG_subprogram)
1319 {
1320 // Skip any DW_TAG_subprogram DIEs that are inside
1321 // of a normal or inlined functions. These will be
1322 // parsed on their own as separate entities.
1323
1324 if (depth > 0)
1325 break;
1326
1327 block = parent_block;
1328 }
1329 else
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001330 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001331 BlockSP block_sp(new Block (die.GetID()));
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001332 parent_block->AddChild(block_sp);
1333 block = block_sp.get();
1334 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00001335 DWARFRangeList ranges;
Greg Claytondd7feaf2011-08-12 17:54:33 +00001336 const char *name = NULL;
1337 const char *mangled_name = NULL;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001338
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001339 int decl_file = 0;
1340 int decl_line = 0;
1341 int decl_column = 0;
1342 int call_file = 0;
1343 int call_line = 0;
1344 int call_column = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00001345 if (die.GetDIENamesAndRanges (name,
1346 mangled_name,
1347 ranges,
1348 decl_file, decl_line, decl_column,
1349 call_file, call_line, call_column, nullptr))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001350 {
1351 if (tag == DW_TAG_subprogram)
1352 {
1353 assert (subprogram_low_pc == LLDB_INVALID_ADDRESS);
Greg Claytonea3e7d52011-10-08 00:49:15 +00001354 subprogram_low_pc = ranges.GetMinRangeBase(0);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001355 }
Jim Inghamb0be4422010-08-12 01:20:14 +00001356 else if (tag == DW_TAG_inlined_subroutine)
1357 {
1358 // We get called here for inlined subroutines in two ways.
1359 // The first time is when we are making the Function object
1360 // for this inlined concrete instance. Since we're creating a top level block at
1361 // here, the subprogram_low_pc will be LLDB_INVALID_ADDRESS. So we need to
1362 // adjust the containing address.
1363 // The second time is when we are parsing the blocks inside the function that contains
1364 // the inlined concrete instance. Since these will be blocks inside the containing "real"
1365 // function the offset will be for that function.
1366 if (subprogram_low_pc == LLDB_INVALID_ADDRESS)
1367 {
Greg Claytonea3e7d52011-10-08 00:49:15 +00001368 subprogram_low_pc = ranges.GetMinRangeBase(0);
Jim Inghamb0be4422010-08-12 01:20:14 +00001369 }
1370 }
Greg Clayton103f3092015-01-15 03:04:37 +00001371
1372 const size_t num_ranges = ranges.GetSize();
1373 for (size_t i = 0; i<num_ranges; ++i)
1374 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001375 const DWARFRangeList::Entry &range = ranges.GetEntryRef (i);
Greg Clayton103f3092015-01-15 03:04:37 +00001376 const addr_t range_base = range.GetRangeBase();
1377 if (range_base >= subprogram_low_pc)
1378 block->AddRange(Block::Range (range_base - subprogram_low_pc, range.GetByteSize()));
1379 else
1380 {
1381 GetObjectFile()->GetModule()->ReportError ("0x%8.8" PRIx64 ": adding range [0x%" PRIx64 "-0x%" PRIx64 ") which has a base that is less than the function's low PC 0x%" PRIx64 ". Please file a bug and attach the file at the start of this error message",
1382 block->GetID(),
1383 range_base,
1384 range.GetRangeEnd(),
1385 subprogram_low_pc);
1386 }
1387 }
1388 block->FinalizeRanges ();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001389
1390 if (tag != DW_TAG_subprogram && (name != NULL || mangled_name != NULL))
1391 {
Greg Clayton7b0992d2013-04-18 22:45:39 +00001392 std::unique_ptr<Declaration> decl_ap;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001393 if (decl_file != 0 || decl_line != 0 || decl_column != 0)
Jim Inghamb0be4422010-08-12 01:20:14 +00001394 decl_ap.reset(new Declaration(sc.comp_unit->GetSupportFiles().GetFileSpecAtIndex(decl_file),
1395 decl_line, decl_column));
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001396
Greg Clayton7b0992d2013-04-18 22:45:39 +00001397 std::unique_ptr<Declaration> call_ap;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001398 if (call_file != 0 || call_line != 0 || call_column != 0)
Jim Inghamb0be4422010-08-12 01:20:14 +00001399 call_ap.reset(new Declaration(sc.comp_unit->GetSupportFiles().GetFileSpecAtIndex(call_file),
1400 call_line, call_column));
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001401
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001402 block->SetInlinedFunctionInfo (name, mangled_name, decl_ap.get(), call_ap.get());
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001403 }
1404
1405 ++blocks_added;
1406
Greg Clayton6071e6f2015-08-26 22:57:51 +00001407 if (die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001408 {
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001409 blocks_added += ParseFunctionBlocks (sc,
1410 block,
Greg Clayton6071e6f2015-08-26 22:57:51 +00001411 die.GetFirstChild(),
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001412 subprogram_low_pc,
Greg Claytondd7feaf2011-08-12 17:54:33 +00001413 depth + 1);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001414 }
1415 }
1416 }
1417 break;
1418 default:
1419 break;
1420 }
1421
Greg Claytondd7feaf2011-08-12 17:54:33 +00001422 // Only parse siblings of the block if we are not at depth zero. A depth
1423 // of zero indicates we are currently parsing the top level
1424 // DW_TAG_subprogram DIE
1425
1426 if (depth == 0)
Greg Clayton6071e6f2015-08-26 22:57:51 +00001427 die.Clear();
Greg Claytondd7feaf2011-08-12 17:54:33 +00001428 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00001429 die = die.GetSibling();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001430 }
1431 return blocks_added;
1432}
1433
Greg Claytonf0705c82011-10-22 03:33:13 +00001434bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00001435SymbolFileDWARF::ClassOrStructIsVirtual (const DWARFDIE &parent_die)
Greg Claytonc4ffd662013-03-08 01:37:30 +00001436{
1437 if (parent_die)
1438 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001439 for (DWARFDIE die = parent_die.GetFirstChild(); die; die = die.GetSibling())
Greg Claytonc4ffd662013-03-08 01:37:30 +00001440 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001441 dw_tag_t tag = die.Tag();
Greg Claytonc4ffd662013-03-08 01:37:30 +00001442 bool check_virtuality = false;
1443 switch (tag)
1444 {
1445 case DW_TAG_inheritance:
1446 case DW_TAG_subprogram:
1447 check_virtuality = true;
1448 break;
1449 default:
1450 break;
1451 }
1452 if (check_virtuality)
1453 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001454 if (die.GetAttributeValueAsUnsigned(DW_AT_virtuality, 0) != 0)
Greg Claytonc4ffd662013-03-08 01:37:30 +00001455 return true;
1456 }
1457 }
1458 }
1459 return false;
1460}
1461
Paul Hermanea188fc2015-09-16 18:48:30 +00001462void
1463SymbolFileDWARF::ParseDeclsForContext (CompilerDeclContext decl_ctx)
1464{
1465 TypeSystem *type_system = decl_ctx.GetTypeSystem();
1466 DWARFASTParser *ast_parser = type_system->GetDWARFParser();
1467 std::vector<DWARFDIE> decl_ctx_die_list = ast_parser->GetDIEForDeclContext(decl_ctx);
1468
1469 for (DWARFDIE decl_ctx_die : decl_ctx_die_list)
1470 for (DWARFDIE decl = decl_ctx_die.GetFirstChild(); decl; decl = decl.GetSibling())
1471 ast_parser->GetDeclForUIDFromDWARF(decl);
1472}
1473
Paul Hermand628cbb2015-09-15 23:44:17 +00001474CompilerDecl
1475SymbolFileDWARF::GetDeclForUID (lldb::user_id_t type_uid)
1476{
1477 if (UserIDMatches(type_uid))
1478 {
1479 DWARFDebugInfo* debug_info = DebugInfo();
1480 if (debug_info)
1481 {
1482 DWARFDIE die = debug_info->GetDIE(DIERef(type_uid));
1483 if (die)
1484 {
1485 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
1486 if (dwarf_ast)
1487 return dwarf_ast->GetDeclForUIDFromDWARF(die);
1488 }
1489 }
1490 }
1491 return CompilerDecl();
1492}
1493
Greg Clayton99558cc42015-08-24 23:46:31 +00001494CompilerDeclContext
1495SymbolFileDWARF::GetDeclContextForUID (lldb::user_id_t type_uid)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001496{
Greg Clayton8b4edba2015-08-14 20:02:05 +00001497 if (UserIDMatches(type_uid))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001498 {
Greg Clayton8b4edba2015-08-14 20:02:05 +00001499 DWARFDebugInfo* debug_info = DebugInfo();
1500 if (debug_info)
1501 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001502 DWARFDIE die = debug_info->GetDIE(DIERef(type_uid));
Greg Clayton8b4edba2015-08-14 20:02:05 +00001503 if (die)
1504 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00001505 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
1506 if (dwarf_ast)
1507 return dwarf_ast->GetDeclContextForUIDFromDWARF(die);
Greg Clayton8b4edba2015-08-14 20:02:05 +00001508 }
1509 }
Sean Callanan72e49402011-08-05 23:43:37 +00001510 }
Greg Clayton99558cc42015-08-24 23:46:31 +00001511 return CompilerDeclContext();
Sean Callanan72e49402011-08-05 23:43:37 +00001512}
1513
Greg Clayton99558cc42015-08-24 23:46:31 +00001514CompilerDeclContext
1515SymbolFileDWARF::GetDeclContextContainingUID (lldb::user_id_t type_uid)
Sean Callanan72e49402011-08-05 23:43:37 +00001516{
Greg Clayton81c22f62011-10-19 18:09:39 +00001517 if (UserIDMatches(type_uid))
Greg Clayton8b4edba2015-08-14 20:02:05 +00001518 {
1519 DWARFDebugInfo* debug_info = DebugInfo();
1520 if (debug_info)
1521 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001522 DWARFDIE die = debug_info->GetDIE(DIERef(type_uid));
Greg Clayton8b4edba2015-08-14 20:02:05 +00001523 if (die)
1524 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00001525 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
1526 if (dwarf_ast)
1527 return dwarf_ast->GetDeclContextContainingUIDFromDWARF(die);
Greg Clayton8b4edba2015-08-14 20:02:05 +00001528 }
1529 }
1530 }
Greg Clayton99558cc42015-08-24 23:46:31 +00001531 return CompilerDeclContext();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001532}
1533
Greg Clayton99558cc42015-08-24 23:46:31 +00001534
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001535Type*
Greg Claytonc685f8e2010-09-15 04:15:46 +00001536SymbolFileDWARF::ResolveTypeUID (lldb::user_id_t type_uid)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001537{
Greg Clayton81c22f62011-10-19 18:09:39 +00001538 if (UserIDMatches(type_uid))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001539 {
Greg Clayton81c22f62011-10-19 18:09:39 +00001540 DWARFDebugInfo* debug_info = DebugInfo();
1541 if (debug_info)
Greg Claytonca512b32011-01-14 04:54:56 +00001542 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001543 DWARFDIE type_die = debug_info->GetDIE (DIERef(type_uid));
Greg Clayton6071e6f2015-08-26 22:57:51 +00001544 if (type_die)
1545 {
1546 const bool assert_not_being_parsed = true;
1547 return ResolveTypeUID (type_die, assert_not_being_parsed);
1548 }
Greg Claytonca512b32011-01-14 04:54:56 +00001549 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001550 }
1551 return NULL;
1552}
1553
Greg Claytoncab36a32011-12-08 05:16:30 +00001554Type*
Greg Clayton6071e6f2015-08-26 22:57:51 +00001555SymbolFileDWARF::ResolveTypeUID (const DWARFDIE &die, bool assert_not_being_parsed)
Sean Callanan5b26f272012-02-04 08:49:35 +00001556{
Greg Clayton6071e6f2015-08-26 22:57:51 +00001557 if (die)
Greg Claytoncab36a32011-12-08 05:16:30 +00001558 {
Greg Clayton5160ce52013-03-27 23:08:40 +00001559 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_DEBUG_INFO));
Greg Clayton3bffb082011-12-10 02:15:28 +00001560 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00001561 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00001562 "SymbolFileDWARF::ResolveTypeUID (die = 0x%8.8x) %s '%s'",
Greg Clayton6071e6f2015-08-26 22:57:51 +00001563 die.GetOffset(),
1564 die.GetTagAsCString(),
1565 die.GetName());
Greg Clayton3bffb082011-12-10 02:15:28 +00001566
Greg Claytoncab36a32011-12-08 05:16:30 +00001567 // We might be coming in in the middle of a type tree (a class
1568 // withing a class, an enum within a class), so parse any needed
1569 // parent DIEs before we get to this one...
Greg Clayton6071e6f2015-08-26 22:57:51 +00001570 DWARFDIE decl_ctx_die = GetDeclContextDIEContainingDIE (die);
1571 if (decl_ctx_die)
Greg Claytoncab36a32011-12-08 05:16:30 +00001572 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001573 if (log)
Greg Claytoncab36a32011-12-08 05:16:30 +00001574 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001575 switch (decl_ctx_die.Tag())
1576 {
1577 case DW_TAG_structure_type:
1578 case DW_TAG_union_type:
1579 case DW_TAG_class_type:
1580 {
1581 // Get the type, which could be a forward declaration
1582 if (log)
1583 GetObjectFile()->GetModule()->LogMessage (log,
1584 "SymbolFileDWARF::ResolveTypeUID (die = 0x%8.8x) %s '%s' resolve parent forward type for 0x%8.8x",
1585 die.GetOffset(),
1586 die.GetTagAsCString(),
1587 die.GetName(),
1588 decl_ctx_die.GetOffset());
1589 }
1590 break;
Greg Claytoncab36a32011-12-08 05:16:30 +00001591
Greg Clayton6071e6f2015-08-26 22:57:51 +00001592 default:
1593 break;
1594 }
1595 }
Greg Claytoncab36a32011-12-08 05:16:30 +00001596 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00001597 return ResolveType (die);
Greg Claytoncab36a32011-12-08 05:16:30 +00001598 }
1599 return NULL;
1600}
1601
Greg Clayton6beaaa62011-01-17 03:46:26 +00001602// This function is used when SymbolFileDWARFDebugMap owns a bunch of
1603// SymbolFileDWARF objects to detect if this DWARF file is the one that
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001604// can resolve a compiler_type.
Greg Clayton6beaaa62011-01-17 03:46:26 +00001605bool
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001606SymbolFileDWARF::HasForwardDeclForClangType (const CompilerType &compiler_type)
Greg Clayton6beaaa62011-01-17 03:46:26 +00001607{
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001608 CompilerType compiler_type_no_qualifiers = ClangUtil::RemoveFastQualifiers(compiler_type);
Greg Claytone6b36cd2015-12-08 01:02:08 +00001609 if (GetForwardDeclClangTypeToDie().count (compiler_type_no_qualifiers.GetOpaqueQualType()))
1610 {
1611 return true;
1612 }
1613 TypeSystem *type_system = compiler_type.GetTypeSystem();
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001614
1615 ClangASTContext *clang_type_system = llvm::dyn_cast_or_null<ClangASTContext>(type_system);
1616 if (!clang_type_system)
1617 return false;
1618 DWARFASTParserClang *ast_parser = static_cast<DWARFASTParserClang *>(clang_type_system->GetDWARFParser());
1619 return ast_parser->GetClangASTImporter().CanImport(compiler_type);
Greg Clayton6beaaa62011-01-17 03:46:26 +00001620}
1621
1622
Greg Clayton57ee3062013-07-11 22:46:58 +00001623bool
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001624SymbolFileDWARF::CompleteType (CompilerType &compiler_type)
Greg Clayton1be10fc2010-09-29 01:12:09 +00001625{
Greg Claytonfb85e622016-02-09 22:36:24 +00001626 lldb_private::Mutex::Locker locker(GetObjectFile()->GetModule()->GetMutex());
1627
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001628 ClangASTContext *clang_type_system = llvm::dyn_cast_or_null<ClangASTContext>(compiler_type.GetTypeSystem());
1629 if (clang_type_system)
Greg Claytone6b36cd2015-12-08 01:02:08 +00001630 {
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001631 DWARFASTParserClang *ast_parser = static_cast<DWARFASTParserClang *>(clang_type_system->GetDWARFParser());
1632 if (ast_parser && ast_parser->GetClangASTImporter().CanImport(compiler_type))
1633 return ast_parser->GetClangASTImporter().CompleteType(compiler_type);
Greg Claytone6b36cd2015-12-08 01:02:08 +00001634 }
1635
Greg Clayton1be10fc2010-09-29 01:12:09 +00001636 // We have a struct/union/class/enum that needs to be fully resolved.
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001637 CompilerType compiler_type_no_qualifiers = ClangUtil::RemoveFastQualifiers(compiler_type);
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001638 auto die_it = GetForwardDeclClangTypeToDie().find (compiler_type_no_qualifiers.GetOpaqueQualType());
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001639 if (die_it == GetForwardDeclClangTypeToDie().end())
Greg Clayton73b472d2010-10-27 03:32:59 +00001640 {
1641 // We have already resolved this type...
Greg Clayton57ee3062013-07-11 22:46:58 +00001642 return true;
Greg Clayton73b472d2010-10-27 03:32:59 +00001643 }
Tamas Berghammerf8fd9b52015-09-14 15:44:29 +00001644
1645 DWARFDebugInfo* debug_info = DebugInfo();
1646 DWARFDIE dwarf_die = debug_info->GetDIE(die_it->getSecond());
1647
Tamas Berghammer69d0b332015-10-09 12:43:08 +00001648 assert(UserIDMatches(die_it->getSecond().GetUID()) && "CompleteType called on the wrong SymbolFile");
1649
Greg Clayton73b472d2010-10-27 03:32:59 +00001650 // Once we start resolving this type, remove it from the forward declaration
1651 // map in case anyone child members or other types require this type to get resolved.
1652 // The type will get resolved when all of the calls to SymbolFileDWARF::ResolveClangOpaqueTypeDefinition
1653 // are done.
Tamas Berghammerf8fd9b52015-09-14 15:44:29 +00001654 GetForwardDeclClangTypeToDie().erase (die_it);
Greg Clayton1be10fc2010-09-29 01:12:09 +00001655
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001656 Type *type = GetDIEToType().lookup (dwarf_die.GetDIE());
Greg Clayton1be10fc2010-09-29 01:12:09 +00001657
Greg Clayton5160ce52013-03-27 23:08:40 +00001658 Log *log (LogChannelDWARF::GetLogIfAny(DWARF_LOG_DEBUG_INFO|DWARF_LOG_TYPE_COMPLETION));
Greg Clayton3bffb082011-12-10 02:15:28 +00001659 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00001660 GetObjectFile()->GetModule()->LogMessageVerboseBacktrace (log,
Daniel Malead01b2952012-11-29 21:49:15 +00001661 "0x%8.8" PRIx64 ": %s '%s' resolving forward declaration...",
Greg Clayton6071e6f2015-08-26 22:57:51 +00001662 dwarf_die.GetID(),
1663 dwarf_die.GetTagAsCString(),
Greg Claytond61c0fc2012-04-23 22:55:20 +00001664 type->GetName().AsCString());
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001665 assert (compiler_type);
Greg Clayton261ac3f2015-08-28 01:01:03 +00001666 DWARFASTParser *dwarf_ast = dwarf_die.GetDWARFParser();
1667 if (dwarf_ast)
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001668 return dwarf_ast->CompleteTypeFromDWARF (dwarf_die, type, compiler_type);
Ashok Thirumurthia4658a52013-07-30 14:58:39 +00001669 return false;
Greg Clayton1be10fc2010-09-29 01:12:09 +00001670}
1671
Greg Clayton8b4edba2015-08-14 20:02:05 +00001672Type*
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00001673SymbolFileDWARF::ResolveType (const DWARFDIE &die, bool assert_not_being_parsed, bool resolve_function_context)
Greg Clayton8b4edba2015-08-14 20:02:05 +00001674{
Greg Clayton6071e6f2015-08-26 22:57:51 +00001675 if (die)
Greg Clayton8b4edba2015-08-14 20:02:05 +00001676 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001677 Type *type = GetDIEToType().lookup (die.GetDIE());
Greg Claytoncab36a32011-12-08 05:16:30 +00001678
Greg Claytonc685f8e2010-09-15 04:15:46 +00001679 if (type == NULL)
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00001680 type = GetTypeForDIE (die, resolve_function_context).get();
Greg Claytoncab36a32011-12-08 05:16:30 +00001681
Greg Clayton24739922010-10-13 03:15:28 +00001682 if (assert_not_being_parsed)
Jim Inghamc3549282012-01-11 02:21:12 +00001683 {
1684 if (type != DIE_IS_BEING_PARSED)
1685 return type;
1686
1687 GetObjectFile()->GetModule()->ReportError ("Parsing a die that is being parsed die: 0x%8.8x: %s %s",
Greg Clayton6071e6f2015-08-26 22:57:51 +00001688 die.GetOffset(),
1689 die.GetTagAsCString(),
1690 die.GetName());
Jim Inghamc3549282012-01-11 02:21:12 +00001691
1692 }
1693 else
1694 return type;
Greg Claytonc685f8e2010-09-15 04:15:46 +00001695 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00001696 return nullptr;
Greg Claytonc685f8e2010-09-15 04:15:46 +00001697}
1698
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001699CompileUnit*
Greg Clayton53eb1c22012-04-02 22:59:12 +00001700SymbolFileDWARF::GetCompUnitForDWARFCompUnit (DWARFCompileUnit* dwarf_cu, uint32_t cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001701{
1702 // Check if the symbol vendor already knows about this compile unit?
Greg Clayton53eb1c22012-04-02 22:59:12 +00001703 if (dwarf_cu->GetUserData() == NULL)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001704 {
1705 // The symbol vendor doesn't know about this compile unit, we
1706 // need to parse and add it to the symbol vendor object.
Greg Clayton53eb1c22012-04-02 22:59:12 +00001707 return ParseCompileUnit(dwarf_cu, cu_idx).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001708 }
Greg Clayton53eb1c22012-04-02 22:59:12 +00001709 return (CompileUnit*)dwarf_cu->GetUserData();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001710}
1711
Greg Clayton8b4edba2015-08-14 20:02:05 +00001712size_t
1713SymbolFileDWARF::GetObjCMethodDIEOffsets (ConstString class_name, DIEArray &method_die_offsets)
1714{
1715 method_die_offsets.clear();
1716 if (m_using_apple_tables)
1717 {
1718 if (m_apple_objc_ap.get())
1719 m_apple_objc_ap->FindByName(class_name.GetCString(), method_die_offsets);
1720 }
1721 else
1722 {
1723 if (!m_indexed)
1724 Index ();
1725
1726 m_objc_class_selectors_index.Find (class_name, method_die_offsets);
1727 }
1728 return method_die_offsets.size();
1729}
1730
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001731bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00001732SymbolFileDWARF::GetFunction (const DWARFDIE &die, SymbolContext& sc)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001733{
Greg Clayton72310352013-02-23 04:12:47 +00001734 sc.Clear(false);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001735
Greg Clayton6071e6f2015-08-26 22:57:51 +00001736 if (die)
1737 {
1738 // Check if the symbol vendor already knows about this compile unit?
1739 sc.comp_unit = GetCompUnitForDWARFCompUnit(die.GetCU(), UINT32_MAX);
1740
1741 sc.function = sc.comp_unit->FindFunctionByUID (die.GetID()).get();
1742 if (sc.function == NULL)
1743 sc.function = ParseCompileUnitFunction(sc, die);
1744
1745 if (sc.function)
1746 {
1747 sc.module_sp = sc.function->CalculateSymbolContextModule();
1748 return true;
1749 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00001750 }
1751
1752 return false;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001753}
1754
Greg Claytone6b36cd2015-12-08 01:02:08 +00001755lldb::ModuleSP
1756SymbolFileDWARF::GetDWOModule (ConstString name)
1757{
1758 UpdateExternalModuleListIfNeeded();
1759 const auto &pos = m_external_type_modules.find(name);
1760 if (pos != m_external_type_modules.end())
1761 return pos->second;
1762 else
1763 return lldb::ModuleSP();
1764}
1765
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001766void
1767SymbolFileDWARF::UpdateExternalModuleListIfNeeded()
1768{
1769 if (m_fetched_external_modules)
1770 return;
1771 m_fetched_external_modules = true;
1772
1773 DWARFDebugInfo * debug_info = DebugInfo();
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001774
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001775 const uint32_t num_compile_units = GetNumCompileUnits();
1776 for (uint32_t cu_idx = 0; cu_idx < num_compile_units; ++cu_idx)
1777 {
1778 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx);
1779
Greg Clayton5ce1a842015-08-27 18:09:44 +00001780 const DWARFDIE die = dwarf_cu->GetCompileUnitDIEOnly();
1781 if (die && die.HasChildren() == false)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001782 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001783 const char *name = die.GetAttributeValueAsString(DW_AT_name, nullptr);
1784
1785 if (name)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001786 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001787 ConstString const_name(name);
1788 if (m_external_type_modules.find(const_name) == m_external_type_modules.end())
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001789 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001790 ModuleSP module_sp;
1791 const char *dwo_path = die.GetAttributeValueAsString(DW_AT_GNU_dwo_name, nullptr);
1792 if (dwo_path)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001793 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001794 ModuleSpec dwo_module_spec;
1795 dwo_module_spec.GetFileSpec().SetFile(dwo_path, false);
1796 dwo_module_spec.GetArchitecture() = m_obj_file->GetModule()->GetArchitecture();
1797 //printf ("Loading dwo = '%s'\n", dwo_path);
1798 Error error = ModuleList::GetSharedModule (dwo_module_spec, module_sp, NULL, NULL, NULL);
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001799 }
Greg Claytone6b36cd2015-12-08 01:02:08 +00001800 m_external_type_modules[const_name] = module_sp;
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001801 }
1802 }
1803 }
1804 }
1805}
Greg Clayton2501e5e2015-01-15 02:59:20 +00001806
1807SymbolFileDWARF::GlobalVariableMap &
1808SymbolFileDWARF::GetGlobalAranges()
1809{
1810 if (!m_global_aranges_ap)
1811 {
1812 m_global_aranges_ap.reset (new GlobalVariableMap());
1813
1814 ModuleSP module_sp = GetObjectFile()->GetModule();
1815 if (module_sp)
1816 {
1817 const size_t num_cus = module_sp->GetNumCompileUnits();
1818 for (size_t i = 0; i < num_cus; ++i)
1819 {
1820 CompUnitSP cu_sp = module_sp->GetCompileUnitAtIndex(i);
1821 if (cu_sp)
1822 {
1823 VariableListSP globals_sp = cu_sp->GetVariableList(true);
1824 if (globals_sp)
1825 {
1826 const size_t num_globals = globals_sp->GetSize();
1827 for (size_t g = 0; g < num_globals; ++g)
1828 {
1829 VariableSP var_sp = globals_sp->GetVariableAtIndex(g);
1830 if (var_sp && !var_sp->GetLocationIsConstantValueData())
1831 {
1832 const DWARFExpression &location = var_sp->LocationExpression();
1833 Value location_result;
1834 Error error;
Tamas Berghammer5b42c7a2016-02-26 14:21:10 +00001835 if (location.Evaluate(nullptr, nullptr, nullptr, LLDB_INVALID_ADDRESS, nullptr, nullptr, location_result, &error))
Greg Clayton2501e5e2015-01-15 02:59:20 +00001836 {
1837 if (location_result.GetValueType() == Value::eValueTypeFileAddress)
1838 {
1839 lldb::addr_t file_addr = location_result.GetScalar().ULongLong();
1840 lldb::addr_t byte_size = 1;
1841 if (var_sp->GetType())
1842 byte_size = var_sp->GetType()->GetByteSize();
1843 m_global_aranges_ap->Append(GlobalVariableMap::Entry(file_addr, byte_size, var_sp.get()));
1844 }
1845 }
1846 }
1847 }
1848 }
1849 }
1850 }
1851 }
1852 m_global_aranges_ap->Sort();
1853 }
1854 return *m_global_aranges_ap;
1855}
1856
1857
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001858uint32_t
1859SymbolFileDWARF::ResolveSymbolContext (const Address& so_addr, uint32_t resolve_scope, SymbolContext& sc)
1860{
1861 Timer scoped_timer(__PRETTY_FUNCTION__,
Daniel Malead01b2952012-11-29 21:49:15 +00001862 "SymbolFileDWARF::ResolveSymbolContext (so_addr = { section = %p, offset = 0x%" PRIx64 " }, resolve_scope = 0x%8.8x)",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00001863 static_cast<void*>(so_addr.GetSection().get()),
1864 so_addr.GetOffset(), resolve_scope);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001865 uint32_t resolved = 0;
Greg Clayton2501e5e2015-01-15 02:59:20 +00001866 if (resolve_scope & ( eSymbolContextCompUnit |
1867 eSymbolContextFunction |
1868 eSymbolContextBlock |
1869 eSymbolContextLineEntry |
1870 eSymbolContextVariable ))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001871 {
1872 lldb::addr_t file_vm_addr = so_addr.GetFileAddress();
1873
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001874 DWARFDebugInfo* debug_info = DebugInfo();
Greg Claytond4a2b372011-09-12 23:21:58 +00001875 if (debug_info)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001876 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00001877 const dw_offset_t cu_offset = debug_info->GetCompileUnitAranges().FindAddress(file_vm_addr);
Greg Clayton2501e5e2015-01-15 02:59:20 +00001878 if (cu_offset == DW_INVALID_OFFSET)
1879 {
1880 // Global variables are not in the compile unit address ranges. The only way to
1881 // currently find global variables is to iterate over the .debug_pubnames or the
1882 // __apple_names table and find all items in there that point to DW_TAG_variable
1883 // DIEs and then find the address that matches.
1884 if (resolve_scope & eSymbolContextVariable)
1885 {
1886 GlobalVariableMap &map = GetGlobalAranges();
1887 const GlobalVariableMap::Entry *entry = map.FindEntryThatContains(file_vm_addr);
1888 if (entry && entry->data)
1889 {
1890 Variable *variable = entry->data;
1891 SymbolContextScope *scc = variable->GetSymbolContextScope();
1892 if (scc)
1893 {
1894 scc->CalculateSymbolContext(&sc);
1895 sc.variable = variable;
1896 }
1897 return sc.GetResolvedMask();
1898 }
1899 }
1900 }
1901 else
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001902 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00001903 uint32_t cu_idx = DW_INVALID_INDEX;
Greg Clayton6071e6f2015-08-26 22:57:51 +00001904 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnit(cu_offset, &cu_idx);
Greg Clayton53eb1c22012-04-02 22:59:12 +00001905 if (dwarf_cu)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001906 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00001907 sc.comp_unit = GetCompUnitForDWARFCompUnit(dwarf_cu, cu_idx);
Greg Clayton526a4ae2012-05-16 22:09:01 +00001908 if (sc.comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001909 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00001910 resolved |= eSymbolContextCompUnit;
1911
Greg Clayton6ab80132012-12-12 17:30:52 +00001912 bool force_check_line_table = false;
Greg Clayton526a4ae2012-05-16 22:09:01 +00001913 if (resolve_scope & (eSymbolContextFunction | eSymbolContextBlock))
1914 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001915 DWARFDIE function_die = dwarf_cu->LookupAddress(file_vm_addr);
1916 DWARFDIE block_die;
1917 if (function_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001918 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001919 sc.function = sc.comp_unit->FindFunctionByUID (function_die.GetID()).get();
Greg Clayton526a4ae2012-05-16 22:09:01 +00001920 if (sc.function == NULL)
Greg Clayton6071e6f2015-08-26 22:57:51 +00001921 sc.function = ParseCompileUnitFunction(sc, function_die);
1922
1923 if (sc.function && (resolve_scope & eSymbolContextBlock))
1924 block_die = function_die.LookupDeepestBlock(file_vm_addr);
Greg Clayton526a4ae2012-05-16 22:09:01 +00001925 }
1926 else
1927 {
1928 // We might have had a compile unit that had discontiguous
1929 // address ranges where the gaps are symbols that don't have
1930 // any debug info. Discontiguous compile unit address ranges
1931 // should only happen when there aren't other functions from
1932 // other compile units in these gaps. This helps keep the size
1933 // of the aranges down.
Greg Clayton6ab80132012-12-12 17:30:52 +00001934 force_check_line_table = true;
Greg Clayton526a4ae2012-05-16 22:09:01 +00001935 }
1936
1937 if (sc.function != NULL)
1938 {
1939 resolved |= eSymbolContextFunction;
1940
1941 if (resolve_scope & eSymbolContextBlock)
1942 {
1943 Block& block = sc.function->GetBlock (true);
1944
Greg Clayton6071e6f2015-08-26 22:57:51 +00001945 if (block_die)
1946 sc.block = block.FindBlockByID (block_die.GetID());
Greg Clayton526a4ae2012-05-16 22:09:01 +00001947 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00001948 sc.block = block.FindBlockByID (function_die.GetID());
Greg Clayton526a4ae2012-05-16 22:09:01 +00001949 if (sc.block)
1950 resolved |= eSymbolContextBlock;
1951 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001952 }
1953 }
Greg Clayton6ab80132012-12-12 17:30:52 +00001954
1955 if ((resolve_scope & eSymbolContextLineEntry) || force_check_line_table)
1956 {
1957 LineTable *line_table = sc.comp_unit->GetLineTable();
1958 if (line_table != NULL)
1959 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001960 // And address that makes it into this function should be in terms
1961 // of this debug file if there is no debug map, or it will be an
1962 // address in the .o file which needs to be fixed up to be in terms
1963 // of the debug map executable. Either way, calling FixupAddress()
1964 // will work for us.
1965 Address exe_so_addr (so_addr);
1966 if (FixupAddress(exe_so_addr))
Greg Clayton6ab80132012-12-12 17:30:52 +00001967 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001968 if (line_table->FindLineEntryByAddress (exe_so_addr, sc.line_entry))
Greg Clayton6ab80132012-12-12 17:30:52 +00001969 {
1970 resolved |= eSymbolContextLineEntry;
1971 }
1972 }
Greg Clayton6ab80132012-12-12 17:30:52 +00001973 }
1974 }
1975
1976 if (force_check_line_table && !(resolved & eSymbolContextLineEntry))
1977 {
1978 // We might have had a compile unit that had discontiguous
1979 // address ranges where the gaps are symbols that don't have
1980 // any debug info. Discontiguous compile unit address ranges
1981 // should only happen when there aren't other functions from
1982 // other compile units in these gaps. This helps keep the size
1983 // of the aranges down.
1984 sc.comp_unit = NULL;
1985 resolved &= ~eSymbolContextCompUnit;
1986 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001987 }
Greg Clayton526a4ae2012-05-16 22:09:01 +00001988 else
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001989 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00001990 GetObjectFile()->GetModule()->ReportWarning ("0x%8.8x: compile unit %u failed to create a valid lldb_private::CompileUnit class.",
1991 cu_offset,
1992 cu_idx);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001993 }
1994 }
1995 }
1996 }
1997 }
1998 return resolved;
1999}
2000
2001
2002
2003uint32_t
2004SymbolFileDWARF::ResolveSymbolContext(const FileSpec& file_spec, uint32_t line, bool check_inlines, uint32_t resolve_scope, SymbolContextList& sc_list)
2005{
2006 const uint32_t prev_size = sc_list.GetSize();
2007 if (resolve_scope & eSymbolContextCompUnit)
2008 {
2009 DWARFDebugInfo* debug_info = DebugInfo();
2010 if (debug_info)
2011 {
2012 uint32_t cu_idx;
Greg Clayton53eb1c22012-04-02 22:59:12 +00002013 DWARFCompileUnit* dwarf_cu = NULL;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002014
Greg Clayton53eb1c22012-04-02 22:59:12 +00002015 for (cu_idx = 0; (dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx)) != NULL; ++cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002016 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00002017 CompileUnit *dc_cu = GetCompUnitForDWARFCompUnit(dwarf_cu, cu_idx);
Sean Callananddd7a2a2013-10-03 22:27:29 +00002018 const bool full_match = (bool)file_spec.GetDirectory();
Greg Clayton1f746072012-08-29 21:13:06 +00002019 bool file_spec_matches_cu_file_spec = dc_cu != NULL && FileSpec::Equal(file_spec, *dc_cu, full_match);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002020 if (check_inlines || file_spec_matches_cu_file_spec)
2021 {
2022 SymbolContext sc (m_obj_file->GetModule());
Greg Clayton53eb1c22012-04-02 22:59:12 +00002023 sc.comp_unit = GetCompUnitForDWARFCompUnit(dwarf_cu, cu_idx);
Greg Clayton526a4ae2012-05-16 22:09:01 +00002024 if (sc.comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002025 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002026 uint32_t file_idx = UINT32_MAX;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002027
Greg Clayton526a4ae2012-05-16 22:09:01 +00002028 // If we are looking for inline functions only and we don't
2029 // find it in the support files, we are done.
2030 if (check_inlines)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002031 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002032 file_idx = sc.comp_unit->GetSupportFiles().FindFileIndex (1, file_spec, true);
2033 if (file_idx == UINT32_MAX)
2034 continue;
2035 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002036
Greg Clayton526a4ae2012-05-16 22:09:01 +00002037 if (line != 0)
2038 {
2039 LineTable *line_table = sc.comp_unit->GetLineTable();
2040
2041 if (line_table != NULL && line != 0)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002042 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002043 // We will have already looked up the file index if
2044 // we are searching for inline entries.
2045 if (!check_inlines)
2046 file_idx = sc.comp_unit->GetSupportFiles().FindFileIndex (1, file_spec, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002047
Greg Clayton526a4ae2012-05-16 22:09:01 +00002048 if (file_idx != UINT32_MAX)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002049 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002050 uint32_t found_line;
2051 uint32_t line_idx = line_table->FindLineEntryIndexByFileIndex (0, file_idx, line, false, &sc.line_entry);
2052 found_line = sc.line_entry.line;
2053
2054 while (line_idx != UINT32_MAX)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002055 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002056 sc.function = NULL;
2057 sc.block = NULL;
2058 if (resolve_scope & (eSymbolContextFunction | eSymbolContextBlock))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002059 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002060 const lldb::addr_t file_vm_addr = sc.line_entry.range.GetBaseAddress().GetFileAddress();
2061 if (file_vm_addr != LLDB_INVALID_ADDRESS)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002062 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002063 DWARFDIE function_die = dwarf_cu->LookupAddress(file_vm_addr);
2064 DWARFDIE block_die;
2065 if (function_die)
Greg Clayton526a4ae2012-05-16 22:09:01 +00002066 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002067 sc.function = sc.comp_unit->FindFunctionByUID (function_die.GetID()).get();
Greg Clayton526a4ae2012-05-16 22:09:01 +00002068 if (sc.function == NULL)
Greg Clayton6071e6f2015-08-26 22:57:51 +00002069 sc.function = ParseCompileUnitFunction(sc, function_die);
2070
2071 if (sc.function && (resolve_scope & eSymbolContextBlock))
2072 block_die = function_die.LookupDeepestBlock(file_vm_addr);
Greg Clayton526a4ae2012-05-16 22:09:01 +00002073 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002074
Greg Clayton526a4ae2012-05-16 22:09:01 +00002075 if (sc.function != NULL)
2076 {
2077 Block& block = sc.function->GetBlock (true);
2078
Greg Clayton6071e6f2015-08-26 22:57:51 +00002079 if (block_die)
2080 sc.block = block.FindBlockByID (block_die.GetID());
2081 else if (function_die)
2082 sc.block = block.FindBlockByID (function_die.GetID());
Greg Clayton526a4ae2012-05-16 22:09:01 +00002083 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002084 }
2085 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002086
Greg Clayton526a4ae2012-05-16 22:09:01 +00002087 sc_list.Append(sc);
2088 line_idx = line_table->FindLineEntryIndexByFileIndex (line_idx + 1, file_idx, found_line, true, &sc.line_entry);
2089 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002090 }
2091 }
Greg Clayton526a4ae2012-05-16 22:09:01 +00002092 else if (file_spec_matches_cu_file_spec && !check_inlines)
2093 {
2094 // only append the context if we aren't looking for inline call sites
2095 // by file and line and if the file spec matches that of the compile unit
2096 sc_list.Append(sc);
2097 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002098 }
2099 else if (file_spec_matches_cu_file_spec && !check_inlines)
2100 {
2101 // only append the context if we aren't looking for inline call sites
2102 // by file and line and if the file spec matches that of the compile unit
2103 sc_list.Append(sc);
2104 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002105
Greg Clayton526a4ae2012-05-16 22:09:01 +00002106 if (!check_inlines)
2107 break;
2108 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002109 }
2110 }
2111 }
2112 }
2113 return sc_list.GetSize() - prev_size;
2114}
2115
2116void
2117SymbolFileDWARF::Index ()
2118{
2119 if (m_indexed)
2120 return;
2121 m_indexed = true;
2122 Timer scoped_timer (__PRETTY_FUNCTION__,
2123 "SymbolFileDWARF::Index (%s)",
Jim Ingham4af59612014-12-19 19:20:44 +00002124 GetObjectFile()->GetFileSpec().GetFilename().AsCString("<Unknown>"));
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002125
2126 DWARFDebugInfo* debug_info = DebugInfo();
2127 if (debug_info)
2128 {
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002129 const uint32_t num_compile_units = GetNumCompileUnits();
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002130 std::vector<NameToDIE> function_basename_index(num_compile_units);
2131 std::vector<NameToDIE> function_fullname_index(num_compile_units);
2132 std::vector<NameToDIE> function_method_index(num_compile_units);
2133 std::vector<NameToDIE> function_selector_index(num_compile_units);
2134 std::vector<NameToDIE> objc_class_selectors_index(num_compile_units);
2135 std::vector<NameToDIE> global_index(num_compile_units);
2136 std::vector<NameToDIE> type_index(num_compile_units);
2137 std::vector<NameToDIE> namespace_index(num_compile_units);
2138
2139 auto parser_fn = [this,
2140 debug_info,
2141 &function_basename_index,
2142 &function_fullname_index,
2143 &function_method_index,
2144 &function_selector_index,
2145 &objc_class_selectors_index,
2146 &global_index,
2147 &type_index,
2148 &namespace_index](uint32_t cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002149 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00002150 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx);
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002151 bool clear_dies = dwarf_cu->ExtractDIEsIfNeeded(false) > 1;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002152
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002153 dwarf_cu->Index(function_basename_index[cu_idx],
2154 function_fullname_index[cu_idx],
2155 function_method_index[cu_idx],
2156 function_selector_index[cu_idx],
2157 objc_class_selectors_index[cu_idx],
2158 global_index[cu_idx],
2159 type_index[cu_idx],
2160 namespace_index[cu_idx]);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002161
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002162 // Keep memory down by clearing DIEs if this generate function
2163 // caused them to be parsed
2164 if (clear_dies)
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002165 dwarf_cu->ClearDIEs(true);
2166
2167 return cu_idx;
2168 };
2169
2170 TaskRunner<uint32_t> task_runner;
2171 for (uint32_t cu_idx = 0; cu_idx < num_compile_units; ++cu_idx)
2172 task_runner.AddTask(parser_fn, cu_idx);
2173
2174 while (true)
2175 {
2176 std::future<uint32_t> f = task_runner.WaitForNextCompletedTask();
2177 if (!f.valid())
2178 break;
2179 uint32_t cu_idx = f.get();
2180
2181 m_function_basename_index.Append(function_basename_index[cu_idx]);
2182 m_function_fullname_index.Append(function_fullname_index[cu_idx]);
2183 m_function_method_index.Append(function_method_index[cu_idx]);
2184 m_function_selector_index.Append(function_selector_index[cu_idx]);
2185 m_objc_class_selectors_index.Append(objc_class_selectors_index[cu_idx]);
2186 m_global_index.Append(global_index[cu_idx]);
2187 m_type_index.Append(type_index[cu_idx]);
2188 m_namespace_index.Append(namespace_index[cu_idx]);
Tamas Berghammerda4e8ed2015-10-20 15:43:40 +00002189 }
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002190
2191 TaskPool::RunTasks(
2192 [&]() { m_function_basename_index.Finalize(); },
2193 [&]() { m_function_fullname_index.Finalize(); },
2194 [&]() { m_function_method_index.Finalize(); },
2195 [&]() { m_function_selector_index.Finalize(); },
2196 [&]() { m_objc_class_selectors_index.Finalize(); },
2197 [&]() { m_global_index.Finalize(); },
2198 [&]() { m_type_index.Finalize(); },
2199 [&]() { m_namespace_index.Finalize(); });
Greg Claytonc685f8e2010-09-15 04:15:46 +00002200
Greg Clayton24739922010-10-13 03:15:28 +00002201#if defined (ENABLE_DEBUG_PRINTF)
Greg Clayton7bd65b92011-02-09 23:39:34 +00002202 StreamFile s(stdout, false);
Greg Claytonb5ad4ec2013-04-29 17:25:54 +00002203 s.Printf ("DWARF index for '%s':",
2204 GetObjectFile()->GetFileSpec().GetPath().c_str());
Greg Claytonba2d22d2010-11-13 22:57:37 +00002205 s.Printf("\nFunction basenames:\n"); m_function_basename_index.Dump (&s);
2206 s.Printf("\nFunction fullnames:\n"); m_function_fullname_index.Dump (&s);
2207 s.Printf("\nFunction methods:\n"); m_function_method_index.Dump (&s);
2208 s.Printf("\nFunction selectors:\n"); m_function_selector_index.Dump (&s);
2209 s.Printf("\nObjective C class selectors:\n"); m_objc_class_selectors_index.Dump (&s);
2210 s.Printf("\nGlobals and statics:\n"); m_global_index.Dump (&s);
Greg Clayton69b04882010-10-15 02:03:22 +00002211 s.Printf("\nTypes:\n"); m_type_index.Dump (&s);
Bruce Mitchenere171da52015-07-22 00:16:02 +00002212 s.Printf("\nNamespaces:\n") m_namespace_index.Dump (&s);
Greg Claytonc685f8e2010-09-15 04:15:46 +00002213#endif
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002214 }
2215}
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002216
2217bool
Greg Clayton99558cc42015-08-24 23:46:31 +00002218SymbolFileDWARF::DeclContextMatchesThisSymbolFile (const lldb_private::CompilerDeclContext *decl_ctx)
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002219{
Greg Clayton99558cc42015-08-24 23:46:31 +00002220 if (decl_ctx == nullptr || !decl_ctx->IsValid())
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002221 {
2222 // Invalid namespace decl which means we aren't matching only things
2223 // in this symbol file, so return true to indicate it matches this
2224 // symbol file.
2225 return true;
2226 }
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002227
Greg Clayton56939cb2015-09-17 22:23:34 +00002228 TypeSystem *decl_ctx_type_system = decl_ctx->GetTypeSystem();
2229 TypeSystem *type_system = GetTypeSystemForLanguage(decl_ctx_type_system->GetMinimumLanguage(nullptr));
2230 if (decl_ctx_type_system == type_system)
Greg Clayton99558cc42015-08-24 23:46:31 +00002231 return true; // The type systems match, return true
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002232
2233 // The namespace AST was valid, and it does not match...
Greg Clayton5160ce52013-03-27 23:08:40 +00002234 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Sean Callananc41e68b2011-10-13 21:08:11 +00002235
2236 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00002237 GetObjectFile()->GetModule()->LogMessage(log, "Valid namespace does not match symbol file");
Sean Callananc41e68b2011-10-13 21:08:11 +00002238
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002239 return false;
2240}
2241
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002242uint32_t
Greg Clayton99558cc42015-08-24 23:46:31 +00002243SymbolFileDWARF::FindGlobalVariables (const ConstString &name, const CompilerDeclContext *parent_decl_ctx, bool append, uint32_t max_matches, VariableList& variables)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002244{
Greg Clayton5160ce52013-03-27 23:08:40 +00002245 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00002246
2247 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00002248 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00002249 "SymbolFileDWARF::FindGlobalVariables (name=\"%s\", parent_decl_ctx=%p, append=%u, max_matches=%u, variables)",
Greg Claytone38a5ed2012-01-05 03:57:59 +00002250 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00002251 static_cast<const void*>(parent_decl_ctx),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002252 append, max_matches);
2253
Greg Clayton99558cc42015-08-24 23:46:31 +00002254 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
Ed Maste4c24b122013-10-17 20:13:14 +00002255 return 0;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002256
Greg Claytonc685f8e2010-09-15 04:15:46 +00002257 DWARFDebugInfo* info = DebugInfo();
2258 if (info == NULL)
2259 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002260
2261 // If we aren't appending the results to this list, then clear the list
2262 if (!append)
2263 variables.Clear();
2264
2265 // Remember how many variables are in the list before we search in case
2266 // we are appending the results to a variable list.
2267 const uint32_t original_size = variables.GetSize();
2268
Greg Claytond4a2b372011-09-12 23:21:58 +00002269 DIEArray die_offsets;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002270
Greg Clayton97fbc342011-10-20 22:30:33 +00002271 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00002272 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002273 if (m_apple_names_ap.get())
2274 {
2275 const char *name_cstr = name.GetCString();
Jim Inghamfa39bb42014-10-25 00:33:55 +00002276 llvm::StringRef basename;
2277 llvm::StringRef context;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002278
Jim Inghamaa816b82015-09-02 01:59:14 +00002279 if (!CPlusPlusLanguage::ExtractContextAndIdentifier(name_cstr, context, basename))
Jim Inghamfa39bb42014-10-25 00:33:55 +00002280 basename = name_cstr;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002281
Jim Inghamfa39bb42014-10-25 00:33:55 +00002282 m_apple_names_ap->FindByName (basename.data(), die_offsets);
Greg Clayton97fbc342011-10-20 22:30:33 +00002283 }
Greg Clayton7f995132011-10-04 22:41:51 +00002284 }
2285 else
2286 {
2287 // Index the DWARF if we haven't already
2288 if (!m_indexed)
2289 Index ();
2290
2291 m_global_index.Find (name, die_offsets);
2292 }
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002293
Greg Clayton437a1352012-04-09 22:43:43 +00002294 const size_t num_die_matches = die_offsets.size();
2295 if (num_die_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002296 {
Greg Clayton7f995132011-10-04 22:41:51 +00002297 SymbolContext sc;
Greg Claytone72dfb32012-02-24 01:59:29 +00002298 sc.module_sp = m_obj_file->GetModule();
Greg Clayton7f995132011-10-04 22:41:51 +00002299 assert (sc.module_sp);
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002300
Greg Claytond4a2b372011-09-12 23:21:58 +00002301 DWARFDebugInfo* debug_info = DebugInfo();
Greg Clayton437a1352012-04-09 22:43:43 +00002302 bool done = false;
2303 for (size_t i=0; i<num_die_matches && !done; ++i)
Greg Claytond4a2b372011-09-12 23:21:58 +00002304 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002305 const DIERef& die_ref = die_offsets[i];
2306 DWARFDIE die = debug_info->GetDIE (die_ref);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002307
Greg Clayton95d87902011-11-11 03:16:25 +00002308 if (die)
2309 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002310 switch (die.Tag())
Greg Clayton437a1352012-04-09 22:43:43 +00002311 {
2312 default:
2313 case DW_TAG_subprogram:
2314 case DW_TAG_inlined_subroutine:
2315 case DW_TAG_try_block:
2316 case DW_TAG_catch_block:
2317 break;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002318
Greg Clayton437a1352012-04-09 22:43:43 +00002319 case DW_TAG_variable:
2320 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002321 sc.comp_unit = GetCompUnitForDWARFCompUnit(die.GetCU(), UINT32_MAX);
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002322
Greg Clayton99558cc42015-08-24 23:46:31 +00002323 if (parent_decl_ctx)
Greg Clayton8b4edba2015-08-14 20:02:05 +00002324 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002325 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
2326 if (dwarf_ast)
Greg Clayton99558cc42015-08-24 23:46:31 +00002327 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002328 CompilerDeclContext actual_parent_decl_ctx = dwarf_ast->GetDeclContextContainingUIDFromDWARF (die);
Greg Clayton99558cc42015-08-24 23:46:31 +00002329 if (!actual_parent_decl_ctx || actual_parent_decl_ctx != *parent_decl_ctx)
2330 continue;
2331 }
Greg Clayton8b4edba2015-08-14 20:02:05 +00002332 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002333
Greg Clayton6071e6f2015-08-26 22:57:51 +00002334 ParseVariables(sc, die, LLDB_INVALID_ADDRESS, false, false, &variables);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002335
Greg Clayton437a1352012-04-09 22:43:43 +00002336 if (variables.GetSize() - original_size >= max_matches)
2337 done = true;
2338 }
2339 break;
2340 }
Greg Clayton95d87902011-11-11 03:16:25 +00002341 }
2342 else
2343 {
2344 if (m_using_apple_tables)
2345 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00002346 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_names accelerator table had bad die 0x%8.8x for '%s')\n",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002347 die_ref.die_offset, name.GetCString());
Greg Clayton95d87902011-11-11 03:16:25 +00002348 }
2349 }
Greg Claytond4a2b372011-09-12 23:21:58 +00002350 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002351 }
2352
2353 // Return the number of variable that were appended to the list
Greg Clayton437a1352012-04-09 22:43:43 +00002354 const uint32_t num_matches = variables.GetSize() - original_size;
2355 if (log && num_matches > 0)
2356 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002357 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00002358 "SymbolFileDWARF::FindGlobalVariables (name=\"%s\", parent_decl_ctx=%p, append=%u, max_matches=%u, variables) => %u",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002359 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00002360 static_cast<const void*>(parent_decl_ctx),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002361 append, max_matches,
Greg Clayton437a1352012-04-09 22:43:43 +00002362 num_matches);
2363 }
2364 return num_matches;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002365}
2366
2367uint32_t
2368SymbolFileDWARF::FindGlobalVariables(const RegularExpression& regex, bool append, uint32_t max_matches, VariableList& variables)
2369{
Greg Clayton5160ce52013-03-27 23:08:40 +00002370 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002371
Greg Clayton21f2a492011-10-06 00:09:08 +00002372 if (log)
2373 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002374 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002375 "SymbolFileDWARF::FindGlobalVariables (regex=\"%s\", append=%u, max_matches=%u, variables)",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002376 regex.GetText(), append,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002377 max_matches);
Greg Clayton21f2a492011-10-06 00:09:08 +00002378 }
2379
Greg Claytonc685f8e2010-09-15 04:15:46 +00002380 DWARFDebugInfo* info = DebugInfo();
2381 if (info == NULL)
2382 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002383
2384 // If we aren't appending the results to this list, then clear the list
2385 if (!append)
2386 variables.Clear();
2387
2388 // Remember how many variables are in the list before we search in case
2389 // we are appending the results to a variable list.
2390 const uint32_t original_size = variables.GetSize();
2391
Greg Clayton7f995132011-10-04 22:41:51 +00002392 DIEArray die_offsets;
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002393
Greg Clayton97fbc342011-10-20 22:30:33 +00002394 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00002395 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002396 if (m_apple_names_ap.get())
Greg Claytond1767f02011-12-08 02:13:16 +00002397 {
2398 DWARFMappedHash::DIEInfoArray hash_data_array;
2399 if (m_apple_names_ap->AppendAllDIEsThatMatchingRegex (regex, hash_data_array))
2400 DWARFMappedHash::ExtractDIEArray (hash_data_array, die_offsets);
2401 }
Greg Clayton7f995132011-10-04 22:41:51 +00002402 }
2403 else
2404 {
2405 // Index the DWARF if we haven't already
2406 if (!m_indexed)
2407 Index ();
2408
2409 m_global_index.Find (regex, die_offsets);
2410 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002411
Greg Claytonc685f8e2010-09-15 04:15:46 +00002412 SymbolContext sc;
Greg Claytone72dfb32012-02-24 01:59:29 +00002413 sc.module_sp = m_obj_file->GetModule();
Greg Claytonc685f8e2010-09-15 04:15:46 +00002414 assert (sc.module_sp);
2415
Greg Clayton7f995132011-10-04 22:41:51 +00002416 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00002417 if (num_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002418 {
Greg Claytond4a2b372011-09-12 23:21:58 +00002419 DWARFDebugInfo* debug_info = DebugInfo();
2420 for (size_t i=0; i<num_matches; ++i)
2421 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002422 const DIERef& die_ref = die_offsets[i];
2423 DWARFDIE die = debug_info->GetDIE (die_ref);
Greg Clayton95d87902011-11-11 03:16:25 +00002424
2425 if (die)
2426 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002427 sc.comp_unit = GetCompUnitForDWARFCompUnit(die.GetCU(), UINT32_MAX);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002428
Greg Clayton6071e6f2015-08-26 22:57:51 +00002429 ParseVariables(sc, die, LLDB_INVALID_ADDRESS, false, false, &variables);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002430
Greg Clayton95d87902011-11-11 03:16:25 +00002431 if (variables.GetSize() - original_size >= max_matches)
2432 break;
2433 }
2434 else
2435 {
2436 if (m_using_apple_tables)
2437 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00002438 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_names accelerator table had bad die 0x%8.8x for regex '%s')\n",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002439 die_ref.die_offset, regex.GetText());
Greg Clayton95d87902011-11-11 03:16:25 +00002440 }
2441 }
Greg Claytond4a2b372011-09-12 23:21:58 +00002442 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002443 }
2444
2445 // Return the number of variable that were appended to the list
2446 return variables.GetSize() - original_size;
2447}
2448
Greg Claytonaa044962011-10-13 00:59:38 +00002449
Jim Ingham4cda6e02011-10-07 22:23:45 +00002450bool
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002451SymbolFileDWARF::ResolveFunction (const DIERef& die_ref,
Pavel Labatha73d6572015-03-13 10:22:00 +00002452 bool include_inlines,
Jim Ingham4cda6e02011-10-07 22:23:45 +00002453 SymbolContextList& sc_list)
Greg Clayton9e315582011-09-02 04:03:59 +00002454{
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002455 DWARFDIE die = DebugInfo()->GetDIE (die_ref);
Greg Clayton6071e6f2015-08-26 22:57:51 +00002456 return ResolveFunction (die, include_inlines, sc_list);
Greg Claytonaa044962011-10-13 00:59:38 +00002457}
2458
2459
2460bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00002461SymbolFileDWARF::ResolveFunction (const DWARFDIE &orig_die,
Pavel Labatha73d6572015-03-13 10:22:00 +00002462 bool include_inlines,
Greg Claytonaa044962011-10-13 00:59:38 +00002463 SymbolContextList& sc_list)
2464{
Greg Clayton9e315582011-09-02 04:03:59 +00002465 SymbolContext sc;
Greg Claytonaa044962011-10-13 00:59:38 +00002466
Greg Clayton6071e6f2015-08-26 22:57:51 +00002467 if (!orig_die)
Greg Claytonaa044962011-10-13 00:59:38 +00002468 return false;
2469
Jim Ingham4cda6e02011-10-07 22:23:45 +00002470 // If we were passed a die that is not a function, just return false...
Greg Clayton6071e6f2015-08-26 22:57:51 +00002471 if (!(orig_die.Tag() == DW_TAG_subprogram || (include_inlines && orig_die.Tag() == DW_TAG_inlined_subroutine)))
Jim Ingham4cda6e02011-10-07 22:23:45 +00002472 return false;
Greg Clayton6071e6f2015-08-26 22:57:51 +00002473
2474 DWARFDIE die = orig_die;
2475 DWARFDIE inlined_die;
2476 if (die.Tag() == DW_TAG_inlined_subroutine)
Greg Clayton9e315582011-09-02 04:03:59 +00002477 {
Jim Ingham4cda6e02011-10-07 22:23:45 +00002478 inlined_die = die;
Greg Clayton9e315582011-09-02 04:03:59 +00002479
Greg Clayton6071e6f2015-08-26 22:57:51 +00002480 while (1)
Greg Clayton2bc22f82011-09-30 03:20:47 +00002481 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002482 die = die.GetParent();
2483
2484 if (die)
2485 {
2486 if (die.Tag() == DW_TAG_subprogram)
2487 break;
2488 }
2489 else
Jim Ingham4cda6e02011-10-07 22:23:45 +00002490 break;
Greg Clayton9e315582011-09-02 04:03:59 +00002491 }
2492 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00002493 assert (die && die.Tag() == DW_TAG_subprogram);
2494 if (GetFunction (die, sc))
Jim Ingham4cda6e02011-10-07 22:23:45 +00002495 {
2496 Address addr;
2497 // Parse all blocks if needed
2498 if (inlined_die)
2499 {
Greg Claytonf7bb1fb2015-01-15 03:13:44 +00002500 Block &function_block = sc.function->GetBlock (true);
Greg Clayton6071e6f2015-08-26 22:57:51 +00002501 sc.block = function_block.FindBlockByID (inlined_die.GetID());
Greg Claytonf7bb1fb2015-01-15 03:13:44 +00002502 if (sc.block == NULL)
Greg Clayton6071e6f2015-08-26 22:57:51 +00002503 sc.block = function_block.FindBlockByID (inlined_die.GetOffset());
Greg Claytonf7bb1fb2015-01-15 03:13:44 +00002504 if (sc.block == NULL || sc.block->GetStartAddress (addr) == false)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002505 addr.Clear();
2506 }
2507 else
2508 {
2509 sc.block = NULL;
2510 addr = sc.function->GetAddressRange().GetBaseAddress();
2511 }
2512
2513 if (addr.IsValid())
2514 {
Jim Ingham4cda6e02011-10-07 22:23:45 +00002515 sc_list.Append(sc);
Greg Claytonaa044962011-10-13 00:59:38 +00002516 return true;
Jim Ingham4cda6e02011-10-07 22:23:45 +00002517 }
2518 }
2519
Greg Claytonaa044962011-10-13 00:59:38 +00002520 return false;
Greg Clayton9e315582011-09-02 04:03:59 +00002521}
2522
Greg Clayton7f995132011-10-04 22:41:51 +00002523void
2524SymbolFileDWARF::FindFunctions (const ConstString &name,
2525 const NameToDIE &name_to_die,
Pavel Labatha73d6572015-03-13 10:22:00 +00002526 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002527 SymbolContextList& sc_list)
2528{
Greg Claytond4a2b372011-09-12 23:21:58 +00002529 DIEArray die_offsets;
Greg Clayton7f995132011-10-04 22:41:51 +00002530 if (name_to_die.Find (name, die_offsets))
2531 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002532 ParseFunctions (die_offsets, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002533 }
2534}
2535
2536
2537void
2538SymbolFileDWARF::FindFunctions (const RegularExpression &regex,
2539 const NameToDIE &name_to_die,
Pavel Labatha73d6572015-03-13 10:22:00 +00002540 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002541 SymbolContextList& sc_list)
2542{
2543 DIEArray die_offsets;
2544 if (name_to_die.Find (regex, die_offsets))
2545 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002546 ParseFunctions (die_offsets, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002547 }
2548}
2549
2550
2551void
2552SymbolFileDWARF::FindFunctions (const RegularExpression &regex,
2553 const DWARFMappedHash::MemoryTable &memory_table,
Pavel Labatha73d6572015-03-13 10:22:00 +00002554 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002555 SymbolContextList& sc_list)
2556{
2557 DIEArray die_offsets;
Greg Claytond1767f02011-12-08 02:13:16 +00002558 DWARFMappedHash::DIEInfoArray hash_data_array;
2559 if (memory_table.AppendAllDIEsThatMatchingRegex (regex, hash_data_array))
Greg Clayton7f995132011-10-04 22:41:51 +00002560 {
Greg Claytond1767f02011-12-08 02:13:16 +00002561 DWARFMappedHash::ExtractDIEArray (hash_data_array, die_offsets);
Pavel Labatha73d6572015-03-13 10:22:00 +00002562 ParseFunctions (die_offsets, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002563 }
2564}
2565
2566void
2567SymbolFileDWARF::ParseFunctions (const DIEArray &die_offsets,
Pavel Labatha73d6572015-03-13 10:22:00 +00002568 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002569 SymbolContextList& sc_list)
2570{
2571 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00002572 if (num_matches)
Greg Claytonc685f8e2010-09-15 04:15:46 +00002573 {
Greg Claytond4a2b372011-09-12 23:21:58 +00002574 for (size_t i=0; i<num_matches; ++i)
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002575 ResolveFunction (die_offsets[i], include_inlines, sc_list);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002576 }
Greg Claytonc685f8e2010-09-15 04:15:46 +00002577}
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002578
Jim Ingham4cda6e02011-10-07 22:23:45 +00002579bool
Greg Clayton99558cc42015-08-24 23:46:31 +00002580SymbolFileDWARF::DIEInDeclContext (const CompilerDeclContext *decl_ctx,
Greg Clayton6071e6f2015-08-26 22:57:51 +00002581 const DWARFDIE &die)
Greg Clayton99558cc42015-08-24 23:46:31 +00002582{
2583 // If we have no parent decl context to match this DIE matches, and if the parent
2584 // decl context isn't valid, we aren't trying to look for any particular decl
2585 // context so any die matches.
2586 if (decl_ctx == nullptr || !decl_ctx->IsValid())
2587 return true;
2588
Greg Clayton6071e6f2015-08-26 22:57:51 +00002589 if (die)
Greg Clayton99558cc42015-08-24 23:46:31 +00002590 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002591 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
2592 if (dwarf_ast)
Greg Clayton99558cc42015-08-24 23:46:31 +00002593 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002594 CompilerDeclContext actual_decl_ctx = dwarf_ast->GetDeclContextContainingUIDFromDWARF (die);
Greg Clayton99558cc42015-08-24 23:46:31 +00002595 if (actual_decl_ctx)
2596 return actual_decl_ctx == *decl_ctx;
2597 }
2598 }
2599 return false;
2600}
2601
Greg Clayton0c5cd902010-06-28 21:30:43 +00002602uint32_t
Greg Clayton99558cc42015-08-24 23:46:31 +00002603SymbolFileDWARF::FindFunctions (const ConstString &name,
2604 const CompilerDeclContext *parent_decl_ctx,
Sean Callanan9df05fb2012-02-10 22:52:19 +00002605 uint32_t name_type_mask,
2606 bool include_inlines,
Greg Clayton2bc22f82011-09-30 03:20:47 +00002607 bool append,
2608 SymbolContextList& sc_list)
Greg Clayton0c5cd902010-06-28 21:30:43 +00002609{
2610 Timer scoped_timer (__PRETTY_FUNCTION__,
2611 "SymbolFileDWARF::FindFunctions (name = '%s')",
2612 name.AsCString());
2613
Greg Clayton43fe2172013-04-03 02:00:15 +00002614 // eFunctionNameTypeAuto should be pre-resolved by a call to Module::PrepareForFunctionNameLookup()
2615 assert ((name_type_mask & eFunctionNameTypeAuto) == 0);
2616
Greg Clayton5160ce52013-03-27 23:08:40 +00002617 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00002618
2619 if (log)
2620 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002621 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002622 "SymbolFileDWARF::FindFunctions (name=\"%s\", name_type_mask=0x%x, append=%u, sc_list)",
2623 name.GetCString(),
2624 name_type_mask,
2625 append);
Greg Clayton21f2a492011-10-06 00:09:08 +00002626 }
2627
Greg Clayton0c5cd902010-06-28 21:30:43 +00002628 // If we aren't appending the results to this list, then clear the list
2629 if (!append)
2630 sc_list.Clear();
Sean Callanan213fdb82011-10-13 01:49:10 +00002631
Greg Clayton99558cc42015-08-24 23:46:31 +00002632 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
Ed Maste4c24b122013-10-17 20:13:14 +00002633 return 0;
Jim Ingham4cda6e02011-10-07 22:23:45 +00002634
2635 // If name is empty then we won't find anything.
2636 if (name.IsEmpty())
2637 return 0;
Greg Clayton0c5cd902010-06-28 21:30:43 +00002638
2639 // Remember how many sc_list are in the list before we search in case
2640 // we are appending the results to a variable list.
Greg Clayton9e315582011-09-02 04:03:59 +00002641
Jim Ingham4cda6e02011-10-07 22:23:45 +00002642 const char *name_cstr = name.GetCString();
Greg Clayton43fe2172013-04-03 02:00:15 +00002643
2644 const uint32_t original_size = sc_list.GetSize();
2645
Jim Ingham4cda6e02011-10-07 22:23:45 +00002646 DWARFDebugInfo* info = DebugInfo();
2647 if (info == NULL)
2648 return 0;
2649
Greg Clayton43fe2172013-04-03 02:00:15 +00002650 std::set<const DWARFDebugInfoEntry *> resolved_dies;
Greg Clayton97fbc342011-10-20 22:30:33 +00002651 if (m_using_apple_tables)
Greg Clayton4d01ace2011-09-29 16:58:15 +00002652 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002653 if (m_apple_names_ap.get())
Jim Ingham4cda6e02011-10-07 22:23:45 +00002654 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002655
2656 DIEArray die_offsets;
2657
2658 uint32_t num_matches = 0;
2659
Greg Clayton43fe2172013-04-03 02:00:15 +00002660 if (name_type_mask & eFunctionNameTypeFull)
Greg Claytonaa044962011-10-13 00:59:38 +00002661 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002662 // If they asked for the full name, match what they typed. At some point we may
2663 // want to canonicalize this (strip double spaces, etc. For now, we just add all the
2664 // dies that we find by exact match.
Jim Ingham4cda6e02011-10-07 22:23:45 +00002665 num_matches = m_apple_names_ap->FindByName (name_cstr, die_offsets);
Jim Ingham4cda6e02011-10-07 22:23:45 +00002666 for (uint32_t i = 0; i < num_matches; i++)
2667 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002668 const DIERef& die_ref = die_offsets[i];
2669 DWARFDIE die = info->GetDIE (die_ref);
Greg Claytonaa044962011-10-13 00:59:38 +00002670 if (die)
2671 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002672 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00002673 continue; // The containing decl contexts don't match
Greg Clayton8b4edba2015-08-14 20:02:05 +00002674
Greg Clayton6071e6f2015-08-26 22:57:51 +00002675 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002676 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002677 if (ResolveFunction (die, include_inlines, sc_list))
2678 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002679 }
Greg Claytonaa044962011-10-13 00:59:38 +00002680 }
Greg Clayton95d87902011-11-11 03:16:25 +00002681 else
2682 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00002683 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_names accelerator table had bad die 0x%8.8x for '%s')",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002684 die_ref.die_offset, name_cstr);
Greg Clayton95d87902011-11-11 03:16:25 +00002685 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002686 }
Greg Clayton97fbc342011-10-20 22:30:33 +00002687 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002688
2689 if (name_type_mask & eFunctionNameTypeSelector)
2690 {
Greg Clayton99558cc42015-08-24 23:46:31 +00002691 if (parent_decl_ctx && parent_decl_ctx->IsValid())
Greg Clayton43fe2172013-04-03 02:00:15 +00002692 return 0; // no selectors in namespaces
Greg Clayton97fbc342011-10-20 22:30:33 +00002693
Greg Clayton43fe2172013-04-03 02:00:15 +00002694 num_matches = m_apple_names_ap->FindByName (name_cstr, die_offsets);
2695 // Now make sure these are actually ObjC methods. In this case we can simply look up the name,
2696 // and if it is an ObjC method name, we're good.
Greg Clayton97fbc342011-10-20 22:30:33 +00002697
Greg Clayton43fe2172013-04-03 02:00:15 +00002698 for (uint32_t i = 0; i < num_matches; i++)
Greg Clayton97fbc342011-10-20 22:30:33 +00002699 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002700 const DIERef& die_ref = die_offsets[i];
2701 DWARFDIE die = info->GetDIE (die_ref);
Greg Clayton43fe2172013-04-03 02:00:15 +00002702 if (die)
Greg Clayton97fbc342011-10-20 22:30:33 +00002703 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002704 const char *die_name = die.GetName();
Jim Inghamaa816b82015-09-02 01:59:14 +00002705 if (ObjCLanguage::IsPossibleObjCMethodName(die_name))
Greg Clayton97fbc342011-10-20 22:30:33 +00002706 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002707 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002708 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002709 if (ResolveFunction (die, include_inlines, sc_list))
2710 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002711 }
Greg Clayton97fbc342011-10-20 22:30:33 +00002712 }
2713 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002714 else
2715 {
2716 GetObjectFile()->GetModule()->ReportError ("the DWARF debug information has been modified (.apple_names accelerator table had bad die 0x%8.8x for '%s')",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002717 die_ref.die_offset, name_cstr);
Greg Clayton43fe2172013-04-03 02:00:15 +00002718 }
Greg Clayton97fbc342011-10-20 22:30:33 +00002719 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002720 die_offsets.clear();
2721 }
2722
Greg Clayton99558cc42015-08-24 23:46:31 +00002723 if (((name_type_mask & eFunctionNameTypeMethod) && !parent_decl_ctx) || name_type_mask & eFunctionNameTypeBase)
Greg Clayton43fe2172013-04-03 02:00:15 +00002724 {
2725 // The apple_names table stores just the "base name" of C++ methods in the table. So we have to
2726 // extract the base name, look that up, and if there is any other information in the name we were
2727 // passed in we have to post-filter based on that.
2728
2729 // FIXME: Arrange the logic above so that we don't calculate the base name twice:
2730 num_matches = m_apple_names_ap->FindByName (name_cstr, die_offsets);
2731
2732 for (uint32_t i = 0; i < num_matches; i++)
2733 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002734 const DIERef& die_ref = die_offsets[i];
2735 DWARFDIE die = info->GetDIE (die_ref);
Greg Clayton43fe2172013-04-03 02:00:15 +00002736 if (die)
2737 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002738 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00002739 continue; // The containing decl contexts don't match
Greg Clayton8b4edba2015-08-14 20:02:05 +00002740
Greg Clayton43fe2172013-04-03 02:00:15 +00002741
2742 // If we get to here, the die is good, and we should add it:
Greg Clayton6071e6f2015-08-26 22:57:51 +00002743 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end() && ResolveFunction (die, include_inlines, sc_list))
Greg Clayton43fe2172013-04-03 02:00:15 +00002744 {
2745 bool keep_die = true;
2746 if ((name_type_mask & (eFunctionNameTypeBase|eFunctionNameTypeMethod)) != (eFunctionNameTypeBase|eFunctionNameTypeMethod))
2747 {
2748 // We are looking for either basenames or methods, so we need to
2749 // trim out the ones we won't want by looking at the type
2750 SymbolContext sc;
2751 if (sc_list.GetLastContext(sc))
2752 {
2753 if (sc.block)
2754 {
2755 // We have an inlined function
2756 }
2757 else if (sc.function)
2758 {
2759 Type *type = sc.function->GetType();
2760
Sean Callananc370a8a2013-09-18 22:59:55 +00002761 if (type)
Greg Clayton43fe2172013-04-03 02:00:15 +00002762 {
Greg Clayton99558cc42015-08-24 23:46:31 +00002763 CompilerDeclContext decl_ctx = GetDeclContextContainingUID (type->GetID());
2764 if (decl_ctx.IsStructUnionOrClass())
Greg Clayton43fe2172013-04-03 02:00:15 +00002765 {
Sean Callananc370a8a2013-09-18 22:59:55 +00002766 if (name_type_mask & eFunctionNameTypeBase)
2767 {
2768 sc_list.RemoveContextAtIndex(sc_list.GetSize()-1);
2769 keep_die = false;
2770 }
2771 }
2772 else
2773 {
2774 if (name_type_mask & eFunctionNameTypeMethod)
2775 {
2776 sc_list.RemoveContextAtIndex(sc_list.GetSize()-1);
2777 keep_die = false;
2778 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002779 }
2780 }
2781 else
2782 {
Sean Callananc370a8a2013-09-18 22:59:55 +00002783 GetObjectFile()->GetModule()->ReportWarning ("function at die offset 0x%8.8x had no function type",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002784 die_ref.die_offset);
Greg Clayton43fe2172013-04-03 02:00:15 +00002785 }
2786 }
2787 }
2788 }
2789 if (keep_die)
Greg Clayton6071e6f2015-08-26 22:57:51 +00002790 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002791 }
2792 }
2793 else
2794 {
2795 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_names accelerator table had bad die 0x%8.8x for '%s')",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002796 die_ref.die_offset, name_cstr);
Greg Clayton43fe2172013-04-03 02:00:15 +00002797 }
2798 }
2799 die_offsets.clear();
Jim Ingham4cda6e02011-10-07 22:23:45 +00002800 }
2801 }
Greg Clayton7f995132011-10-04 22:41:51 +00002802 }
2803 else
2804 {
2805
2806 // Index the DWARF if we haven't already
2807 if (!m_indexed)
2808 Index ();
2809
Greg Clayton7f995132011-10-04 22:41:51 +00002810 if (name_type_mask & eFunctionNameTypeFull)
Matt Kopecd6089962013-05-10 17:53:48 +00002811 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002812 FindFunctions (name, m_function_fullname_index, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002813
Ed Mastefc7baa02013-09-09 18:00:45 +00002814 // FIXME Temporary workaround for global/anonymous namespace
Robert Flack5cbd3bf2015-05-13 18:20:02 +00002815 // functions debugging FreeBSD and Linux binaries.
Matt Kopecd6089962013-05-10 17:53:48 +00002816 // If we didn't find any functions in the global namespace try
2817 // looking in the basename index but ignore any returned
Robert Flackeb83fab2015-05-15 18:59:59 +00002818 // functions that have a namespace but keep functions which
2819 // have an anonymous namespace
2820 // TODO: The arch in the object file isn't correct for MSVC
2821 // binaries on windows, we should find a way to make it
2822 // correct and handle those symbols as well.
Aidan Doddsc78e8992015-11-10 14:10:57 +00002823 if (sc_list.GetSize() == original_size)
Matt Kopecd6089962013-05-10 17:53:48 +00002824 {
Robert Flackeb83fab2015-05-15 18:59:59 +00002825 ArchSpec arch;
Greg Clayton99558cc42015-08-24 23:46:31 +00002826 if (!parent_decl_ctx &&
Robert Flackeb83fab2015-05-15 18:59:59 +00002827 GetObjectFile()->GetArchitecture(arch) &&
2828 (arch.GetTriple().isOSFreeBSD() || arch.GetTriple().isOSLinux() ||
2829 arch.GetMachine() == llvm::Triple::hexagon))
Matt Kopecd6089962013-05-10 17:53:48 +00002830 {
Robert Flackeb83fab2015-05-15 18:59:59 +00002831 SymbolContextList temp_sc_list;
2832 FindFunctions (name, m_function_basename_index, include_inlines, temp_sc_list);
Matt Kopecd6089962013-05-10 17:53:48 +00002833 SymbolContext sc;
2834 for (uint32_t i = 0; i < temp_sc_list.GetSize(); i++)
2835 {
2836 if (temp_sc_list.GetContextAtIndex(i, sc))
2837 {
Matt Kopeca189d492013-05-10 22:55:24 +00002838 ConstString mangled_name = sc.GetFunctionName(Mangled::ePreferMangled);
2839 ConstString demangled_name = sc.GetFunctionName(Mangled::ePreferDemangled);
Robert Flackeb83fab2015-05-15 18:59:59 +00002840 // Mangled names on Linux and FreeBSD are of the form:
2841 // _ZN18function_namespace13function_nameEv.
Matt Kopec04e5d582013-05-14 19:00:41 +00002842 if (strncmp(mangled_name.GetCString(), "_ZN", 3) ||
2843 !strncmp(demangled_name.GetCString(), "(anonymous namespace)", 21))
Matt Kopecd6089962013-05-10 17:53:48 +00002844 {
2845 sc_list.Append(sc);
2846 }
2847 }
2848 }
2849 }
2850 }
Matt Kopecd6089962013-05-10 17:53:48 +00002851 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002852 DIEArray die_offsets;
Greg Clayton43fe2172013-04-03 02:00:15 +00002853 if (name_type_mask & eFunctionNameTypeBase)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002854 {
Greg Clayton43fe2172013-04-03 02:00:15 +00002855 uint32_t num_base = m_function_basename_index.Find(name, die_offsets);
Greg Claytonaa044962011-10-13 00:59:38 +00002856 for (uint32_t i = 0; i < num_base; i++)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002857 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002858 DWARFDIE die = info->GetDIE (die_offsets[i]);
Greg Claytonaa044962011-10-13 00:59:38 +00002859 if (die)
2860 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002861 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00002862 continue; // The containing decl contexts don't match
Greg Clayton8b4edba2015-08-14 20:02:05 +00002863
Greg Claytonaa044962011-10-13 00:59:38 +00002864 // If we get to here, the die is good, and we should add it:
Greg Clayton6071e6f2015-08-26 22:57:51 +00002865 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002866 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002867 if (ResolveFunction (die, include_inlines, sc_list))
2868 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002869 }
Greg Claytonaa044962011-10-13 00:59:38 +00002870 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002871 }
2872 die_offsets.clear();
2873 }
2874
Greg Clayton43fe2172013-04-03 02:00:15 +00002875 if (name_type_mask & eFunctionNameTypeMethod)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002876 {
Greg Clayton99558cc42015-08-24 23:46:31 +00002877 if (parent_decl_ctx && parent_decl_ctx->IsValid())
Sean Callanan213fdb82011-10-13 01:49:10 +00002878 return 0; // no methods in namespaces
2879
Greg Clayton43fe2172013-04-03 02:00:15 +00002880 uint32_t num_base = m_function_method_index.Find(name, die_offsets);
Jim Ingham4cda6e02011-10-07 22:23:45 +00002881 {
Greg Claytonaa044962011-10-13 00:59:38 +00002882 for (uint32_t i = 0; i < num_base; i++)
2883 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002884 DWARFDIE die = info->GetDIE (die_offsets[i]);
Greg Claytonaa044962011-10-13 00:59:38 +00002885 if (die)
2886 {
Greg Claytonaa044962011-10-13 00:59:38 +00002887 // If we get to here, the die is good, and we should add it:
Greg Clayton6071e6f2015-08-26 22:57:51 +00002888 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002889 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002890 if (ResolveFunction (die, include_inlines, sc_list))
2891 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002892 }
Greg Claytonaa044962011-10-13 00:59:38 +00002893 }
2894 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002895 }
2896 die_offsets.clear();
2897 }
Greg Clayton7f995132011-10-04 22:41:51 +00002898
Greg Clayton99558cc42015-08-24 23:46:31 +00002899 if ((name_type_mask & eFunctionNameTypeSelector) && (!parent_decl_ctx || !parent_decl_ctx->IsValid()))
Jim Ingham4cda6e02011-10-07 22:23:45 +00002900 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002901 FindFunctions (name, m_function_selector_index, include_inlines, sc_list);
Jim Ingham4cda6e02011-10-07 22:23:45 +00002902 }
2903
Greg Clayton4d01ace2011-09-29 16:58:15 +00002904 }
2905
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002906 // Return the number of variable that were appended to the list
Greg Clayton437a1352012-04-09 22:43:43 +00002907 const uint32_t num_matches = sc_list.GetSize() - original_size;
2908
2909 if (log && num_matches > 0)
2910 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002911 GetObjectFile()->GetModule()->LogMessage (log,
Pavel Labatha73d6572015-03-13 10:22:00 +00002912 "SymbolFileDWARF::FindFunctions (name=\"%s\", name_type_mask=0x%x, include_inlines=%d, append=%u, sc_list) => %u",
Greg Clayton437a1352012-04-09 22:43:43 +00002913 name.GetCString(),
2914 name_type_mask,
Pavel Labatha73d6572015-03-13 10:22:00 +00002915 include_inlines,
Greg Clayton437a1352012-04-09 22:43:43 +00002916 append,
2917 num_matches);
2918 }
2919 return num_matches;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002920}
2921
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002922uint32_t
Sean Callanan9df05fb2012-02-10 22:52:19 +00002923SymbolFileDWARF::FindFunctions(const RegularExpression& regex, bool include_inlines, bool append, SymbolContextList& sc_list)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002924{
2925 Timer scoped_timer (__PRETTY_FUNCTION__,
2926 "SymbolFileDWARF::FindFunctions (regex = '%s')",
2927 regex.GetText());
2928
Greg Clayton5160ce52013-03-27 23:08:40 +00002929 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00002930
2931 if (log)
2932 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002933 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002934 "SymbolFileDWARF::FindFunctions (regex=\"%s\", append=%u, sc_list)",
2935 regex.GetText(),
2936 append);
Greg Clayton21f2a492011-10-06 00:09:08 +00002937 }
2938
2939
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002940 // If we aren't appending the results to this list, then clear the list
2941 if (!append)
2942 sc_list.Clear();
2943
2944 // Remember how many sc_list are in the list before we search in case
2945 // we are appending the results to a variable list.
2946 uint32_t original_size = sc_list.GetSize();
2947
Greg Clayton97fbc342011-10-20 22:30:33 +00002948 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00002949 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002950 if (m_apple_names_ap.get())
Pavel Labatha73d6572015-03-13 10:22:00 +00002951 FindFunctions (regex, *m_apple_names_ap, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002952 }
2953 else
2954 {
Jim Ingham4cda6e02011-10-07 22:23:45 +00002955 // Index the DWARF if we haven't already
Greg Clayton7f995132011-10-04 22:41:51 +00002956 if (!m_indexed)
2957 Index ();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002958
Pavel Labatha73d6572015-03-13 10:22:00 +00002959 FindFunctions (regex, m_function_basename_index, include_inlines, sc_list);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002960
Pavel Labatha73d6572015-03-13 10:22:00 +00002961 FindFunctions (regex, m_function_fullname_index, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002962 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002963
2964 // Return the number of variable that were appended to the list
2965 return sc_list.GetSize() - original_size;
2966}
Jim Ingham318c9f22011-08-26 19:44:13 +00002967
Siva Chandra9293fc42016-01-07 23:32:34 +00002968void
2969SymbolFileDWARF::GetMangledNamesForFunction (const std::string &scope_qualified_name,
2970 std::vector<ConstString> &mangled_names)
2971{
2972 DWARFDebugInfo* info = DebugInfo();
2973 uint32_t num_comp_units = 0;
2974 if (info)
2975 num_comp_units = info->GetNumCompileUnits();
2976
2977 for (uint32_t i = 0; i < num_comp_units; i++)
2978 {
2979 DWARFCompileUnit *cu = info->GetCompileUnitAtIndex(i);
2980 if (cu == nullptr)
2981 continue;
2982
2983 SymbolFileDWARFDwo *dwo = cu->GetDwoSymbolFile();
2984 if (dwo)
2985 dwo->GetMangledNamesForFunction(scope_qualified_name, mangled_names);
2986 }
2987
2988 NameToOffsetMap::iterator iter = m_function_scope_qualified_name_map.find(scope_qualified_name);
2989 if (iter == m_function_scope_qualified_name_map.end())
2990 return;
2991
2992 DIERefSetSP set_sp = (*iter).second;
2993 std::set<DIERef>::iterator set_iter;
2994 for (set_iter = set_sp->begin(); set_iter != set_sp->end(); set_iter++)
2995 {
2996 DWARFDIE die = DebugInfo()->GetDIE (*set_iter);
2997 mangled_names.push_back(ConstString(die.GetMangledName()));
2998 }
2999}
3000
3001
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003002uint32_t
Greg Claytond1767f02011-12-08 02:13:16 +00003003SymbolFileDWARF::FindTypes (const SymbolContext& sc,
3004 const ConstString &name,
Greg Clayton99558cc42015-08-24 23:46:31 +00003005 const CompilerDeclContext *parent_decl_ctx,
Greg Claytond1767f02011-12-08 02:13:16 +00003006 bool append,
Greg Claytonae088e52016-02-10 21:28:13 +00003007 uint32_t max_matches,
3008 llvm::DenseSet<lldb_private::SymbolFile *> &searched_symbol_files,
Ravitheja Addepally40697302015-10-08 09:45:41 +00003009 TypeMap& types)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003010{
Greg Claytonae088e52016-02-10 21:28:13 +00003011 // If we aren't appending the results to this list, then clear the list
3012 if (!append)
3013 types.Clear();
3014
3015 // Make sure we haven't already searched this SymbolFile before...
3016 if (searched_symbol_files.count(this))
3017 return 0;
3018 else
3019 searched_symbol_files.insert(this);
3020
Greg Claytonc685f8e2010-09-15 04:15:46 +00003021 DWARFDebugInfo* info = DebugInfo();
3022 if (info == NULL)
3023 return 0;
3024
Greg Clayton5160ce52013-03-27 23:08:40 +00003025 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003026
Greg Clayton21f2a492011-10-06 00:09:08 +00003027 if (log)
3028 {
Greg Clayton99558cc42015-08-24 23:46:31 +00003029 if (parent_decl_ctx)
Greg Clayton5160ce52013-03-27 23:08:40 +00003030 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003031 "SymbolFileDWARF::FindTypes (sc, name=\"%s\", parent_decl_ctx = %p (\"%s\"), append=%u, max_matches=%u, type_list)",
Greg Clayton437a1352012-04-09 22:43:43 +00003032 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00003033 static_cast<const void*>(parent_decl_ctx),
3034 parent_decl_ctx->GetName().AsCString("<NULL>"),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003035 append, max_matches);
Greg Clayton437a1352012-04-09 22:43:43 +00003036 else
Greg Clayton5160ce52013-03-27 23:08:40 +00003037 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003038 "SymbolFileDWARF::FindTypes (sc, name=\"%s\", parent_decl_ctx = NULL, append=%u, max_matches=%u, type_list)",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003039 name.GetCString(), append,
Greg Clayton437a1352012-04-09 22:43:43 +00003040 max_matches);
Greg Clayton21f2a492011-10-06 00:09:08 +00003041 }
3042
Greg Clayton99558cc42015-08-24 23:46:31 +00003043 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
Ed Maste4c24b122013-10-17 20:13:14 +00003044 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003045
Greg Claytond4a2b372011-09-12 23:21:58 +00003046 DIEArray die_offsets;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003047
Greg Clayton97fbc342011-10-20 22:30:33 +00003048 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00003049 {
Greg Clayton97fbc342011-10-20 22:30:33 +00003050 if (m_apple_types_ap.get())
3051 {
3052 const char *name_cstr = name.GetCString();
3053 m_apple_types_ap->FindByName (name_cstr, die_offsets);
3054 }
Greg Clayton7f995132011-10-04 22:41:51 +00003055 }
3056 else
3057 {
3058 if (!m_indexed)
3059 Index ();
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003060
Greg Clayton7f995132011-10-04 22:41:51 +00003061 m_type_index.Find (name, die_offsets);
3062 }
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003063
Greg Clayton437a1352012-04-09 22:43:43 +00003064 const size_t num_die_matches = die_offsets.size();
Greg Clayton7f995132011-10-04 22:41:51 +00003065
Greg Clayton437a1352012-04-09 22:43:43 +00003066 if (num_die_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003067 {
Greg Clayton7f995132011-10-04 22:41:51 +00003068 const uint32_t initial_types_size = types.GetSize();
Greg Claytond4a2b372011-09-12 23:21:58 +00003069 DWARFDebugInfo* debug_info = DebugInfo();
Greg Clayton437a1352012-04-09 22:43:43 +00003070 for (size_t i=0; i<num_die_matches; ++i)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003071 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003072 const DIERef& die_ref = die_offsets[i];
3073 DWARFDIE die = debug_info->GetDIE (die_ref);
Greg Claytond4a2b372011-09-12 23:21:58 +00003074
Greg Clayton95d87902011-11-11 03:16:25 +00003075 if (die)
Greg Clayton73bf5db2011-06-17 01:22:15 +00003076 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003077 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00003078 continue; // The containing decl contexts don't match
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003079
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003080 Type *matching_type = ResolveType (die, true, true);
Greg Clayton95d87902011-11-11 03:16:25 +00003081 if (matching_type)
3082 {
3083 // We found a type pointer, now find the shared pointer form our type list
Greg Claytone1cd1be2012-01-29 20:56:30 +00003084 types.InsertUnique (matching_type->shared_from_this());
Greg Clayton95d87902011-11-11 03:16:25 +00003085 if (types.GetSize() >= max_matches)
3086 break;
3087 }
Greg Clayton73bf5db2011-06-17 01:22:15 +00003088 }
Greg Clayton95d87902011-11-11 03:16:25 +00003089 else
3090 {
3091 if (m_using_apple_tables)
3092 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00003093 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_types accelerator table had bad die 0x%8.8x for '%s')\n",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003094 die_ref.die_offset, name.GetCString());
Greg Clayton95d87902011-11-11 03:16:25 +00003095 }
3096 }
3097
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003098 }
Greg Clayton437a1352012-04-09 22:43:43 +00003099 const uint32_t num_matches = types.GetSize() - initial_types_size;
3100 if (log && num_matches)
3101 {
Greg Clayton99558cc42015-08-24 23:46:31 +00003102 if (parent_decl_ctx)
Greg Clayton437a1352012-04-09 22:43:43 +00003103 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003104 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003105 "SymbolFileDWARF::FindTypes (sc, name=\"%s\", parent_decl_ctx = %p (\"%s\"), append=%u, max_matches=%u, type_list) => %u",
Greg Clayton437a1352012-04-09 22:43:43 +00003106 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00003107 static_cast<const void*>(parent_decl_ctx),
3108 parent_decl_ctx->GetName().AsCString("<NULL>"),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003109 append, max_matches,
Greg Clayton437a1352012-04-09 22:43:43 +00003110 num_matches);
3111 }
3112 else
3113 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003114 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003115 "SymbolFileDWARF::FindTypes (sc, name=\"%s\", parent_decl_ctx = NULL, append=%u, max_matches=%u, type_list) => %u",
Greg Clayton437a1352012-04-09 22:43:43 +00003116 name.GetCString(),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003117 append, max_matches,
Greg Clayton437a1352012-04-09 22:43:43 +00003118 num_matches);
3119 }
3120 }
3121 return num_matches;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003122 }
Greg Claytone6b36cd2015-12-08 01:02:08 +00003123 else
3124 {
3125 UpdateExternalModuleListIfNeeded();
3126
3127 for (const auto &pair : m_external_type_modules)
3128 {
3129 ModuleSP external_module_sp = pair.second;
3130 if (external_module_sp)
3131 {
3132 SymbolVendor *sym_vendor = external_module_sp->GetSymbolVendor();
3133 if (sym_vendor)
3134 {
3135 const uint32_t num_external_matches = sym_vendor->FindTypes (sc,
3136 name,
3137 parent_decl_ctx,
3138 append,
3139 max_matches,
Greg Claytonae088e52016-02-10 21:28:13 +00003140 searched_symbol_files,
Greg Claytone6b36cd2015-12-08 01:02:08 +00003141 types);
3142 if (num_external_matches)
3143 return num_external_matches;
3144 }
3145 }
3146 }
3147 }
3148
3149 return 0;
3150}
3151
3152
3153size_t
3154SymbolFileDWARF::FindTypes (const std::vector<CompilerContext> &context,
3155 bool append,
3156 TypeMap& types)
3157{
3158 if (!append)
3159 types.Clear();
3160
3161 if (context.empty())
3162 return 0;
3163
3164 DIEArray die_offsets;
3165
3166 ConstString name = context.back().name;
3167
Sean Callanan9b0cfe22016-01-14 18:59:49 +00003168 if (!name)
3169 return 0;
3170
Greg Claytone6b36cd2015-12-08 01:02:08 +00003171 if (m_using_apple_tables)
3172 {
3173 if (m_apple_types_ap.get())
3174 {
3175 const char *name_cstr = name.GetCString();
3176 m_apple_types_ap->FindByName (name_cstr, die_offsets);
3177 }
3178 }
3179 else
3180 {
3181 if (!m_indexed)
3182 Index ();
3183
3184 m_type_index.Find (name, die_offsets);
3185 }
3186
3187 const size_t num_die_matches = die_offsets.size();
3188
3189 if (num_die_matches)
3190 {
3191 size_t num_matches = 0;
3192 DWARFDebugInfo* debug_info = DebugInfo();
3193 for (size_t i=0; i<num_die_matches; ++i)
3194 {
3195 const DIERef& die_ref = die_offsets[i];
3196 DWARFDIE die = debug_info->GetDIE (die_ref);
3197
3198 if (die)
3199 {
3200 std::vector<CompilerContext> die_context;
3201 die.GetDWOContext(die_context);
3202 if (die_context != context)
3203 continue;
3204
3205 Type *matching_type = ResolveType (die, true, true);
3206 if (matching_type)
3207 {
3208 // We found a type pointer, now find the shared pointer form our type list
3209 types.InsertUnique (matching_type->shared_from_this());
3210 ++num_matches;
3211 }
3212 }
3213 else
3214 {
3215 if (m_using_apple_tables)
3216 {
3217 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_types accelerator table had bad die 0x%8.8x for '%s')\n",
3218 die_ref.die_offset, name.GetCString());
3219 }
3220 }
3221
3222 }
3223 return num_matches;
3224 }
Greg Clayton7f995132011-10-04 22:41:51 +00003225 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003226}
3227
3228
Greg Clayton99558cc42015-08-24 23:46:31 +00003229CompilerDeclContext
Greg Clayton96d7d742010-11-10 23:42:09 +00003230SymbolFileDWARF::FindNamespace (const SymbolContext& sc,
Sean Callanan213fdb82011-10-13 01:49:10 +00003231 const ConstString &name,
Greg Clayton99558cc42015-08-24 23:46:31 +00003232 const CompilerDeclContext *parent_decl_ctx)
Greg Clayton96d7d742010-11-10 23:42:09 +00003233{
Greg Clayton5160ce52013-03-27 23:08:40 +00003234 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00003235
3236 if (log)
3237 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003238 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00003239 "SymbolFileDWARF::FindNamespace (sc, name=\"%s\")",
3240 name.GetCString());
Greg Clayton21f2a492011-10-06 00:09:08 +00003241 }
3242
Greg Clayton99558cc42015-08-24 23:46:31 +00003243 CompilerDeclContext namespace_decl_ctx;
3244
3245 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
3246 return namespace_decl_ctx;
3247
3248
Greg Clayton96d7d742010-11-10 23:42:09 +00003249 DWARFDebugInfo* info = DebugInfo();
Greg Clayton526e5af2010-11-13 03:52:47 +00003250 if (info)
Greg Clayton96d7d742010-11-10 23:42:09 +00003251 {
Greg Clayton7f995132011-10-04 22:41:51 +00003252 DIEArray die_offsets;
3253
Greg Clayton526e5af2010-11-13 03:52:47 +00003254 // Index if we already haven't to make sure the compile units
3255 // get indexed and make their global DIE index list
Greg Clayton97fbc342011-10-20 22:30:33 +00003256 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00003257 {
Greg Clayton97fbc342011-10-20 22:30:33 +00003258 if (m_apple_namespaces_ap.get())
3259 {
3260 const char *name_cstr = name.GetCString();
3261 m_apple_namespaces_ap->FindByName (name_cstr, die_offsets);
3262 }
Greg Clayton7f995132011-10-04 22:41:51 +00003263 }
3264 else
3265 {
3266 if (!m_indexed)
3267 Index ();
Greg Clayton96d7d742010-11-10 23:42:09 +00003268
Greg Clayton7f995132011-10-04 22:41:51 +00003269 m_namespace_index.Find (name, die_offsets);
3270 }
Greg Claytond4a2b372011-09-12 23:21:58 +00003271
Greg Clayton7f995132011-10-04 22:41:51 +00003272 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00003273 if (num_matches)
Greg Clayton526e5af2010-11-13 03:52:47 +00003274 {
Greg Claytond4a2b372011-09-12 23:21:58 +00003275 DWARFDebugInfo* debug_info = DebugInfo();
3276 for (size_t i=0; i<num_matches; ++i)
Greg Clayton526e5af2010-11-13 03:52:47 +00003277 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003278 const DIERef& die_ref = die_offsets[i];
3279 DWARFDIE die = debug_info->GetDIE (die_ref);
Greg Clayton6071e6f2015-08-26 22:57:51 +00003280
Greg Clayton95d87902011-11-11 03:16:25 +00003281 if (die)
Greg Claytond4a2b372011-09-12 23:21:58 +00003282 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003283 if (!DIEInDeclContext (parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00003284 continue; // The containing decl contexts don't match
Greg Clayton95d87902011-11-11 03:16:25 +00003285
Greg Clayton261ac3f2015-08-28 01:01:03 +00003286 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
3287 if (dwarf_ast)
Greg Clayton8b4edba2015-08-14 20:02:05 +00003288 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00003289 namespace_decl_ctx = dwarf_ast->GetDeclContextForUIDFromDWARF (die);
Greg Clayton99558cc42015-08-24 23:46:31 +00003290 if (namespace_decl_ctx)
Greg Clayton8b4edba2015-08-14 20:02:05 +00003291 break;
Greg Clayton95d87902011-11-11 03:16:25 +00003292 }
Greg Claytond4a2b372011-09-12 23:21:58 +00003293 }
Greg Clayton95d87902011-11-11 03:16:25 +00003294 else
3295 {
3296 if (m_using_apple_tables)
3297 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00003298 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_namespaces accelerator table had bad die 0x%8.8x for '%s')\n",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003299 die_ref.die_offset, name.GetCString());
Greg Clayton95d87902011-11-11 03:16:25 +00003300 }
3301 }
3302
Greg Clayton526e5af2010-11-13 03:52:47 +00003303 }
3304 }
Greg Clayton96d7d742010-11-10 23:42:09 +00003305 }
Greg Clayton99558cc42015-08-24 23:46:31 +00003306 if (log && namespace_decl_ctx)
Greg Clayton437a1352012-04-09 22:43:43 +00003307 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003308 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003309 "SymbolFileDWARF::FindNamespace (sc, name=\"%s\") => CompilerDeclContext(%p/%p) \"%s\"",
Greg Clayton437a1352012-04-09 22:43:43 +00003310 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00003311 static_cast<const void*>(namespace_decl_ctx.GetTypeSystem()),
3312 static_cast<const void*>(namespace_decl_ctx.GetOpaqueDeclContext()),
3313 namespace_decl_ctx.GetName().AsCString("<NULL>"));
Greg Clayton437a1352012-04-09 22:43:43 +00003314 }
3315
Greg Clayton99558cc42015-08-24 23:46:31 +00003316 return namespace_decl_ctx;
Greg Clayton96d7d742010-11-10 23:42:09 +00003317}
3318
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003319TypeSP
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003320SymbolFileDWARF::GetTypeForDIE (const DWARFDIE &die, bool resolve_function_context)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003321{
3322 TypeSP type_sp;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003323 if (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003324 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003325 Type *type_ptr = GetDIEToType().lookup (die.GetDIE());
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003326 if (type_ptr == NULL)
3327 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003328 CompileUnit* lldb_cu = GetCompUnitForDWARFCompUnit(die.GetCU());
Greg Claytonca512b32011-01-14 04:54:56 +00003329 assert (lldb_cu);
3330 SymbolContext sc(lldb_cu);
Ravitheja Addepally40697302015-10-08 09:45:41 +00003331 const DWARFDebugInfoEntry* parent_die = die.GetParent().GetDIE();
3332 while (parent_die != nullptr)
3333 {
3334 if (parent_die->Tag() == DW_TAG_subprogram)
3335 break;
3336 parent_die = parent_die->GetParent();
3337 }
3338 SymbolContext sc_backup = sc;
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003339 if (resolve_function_context && parent_die != nullptr && !GetFunction(DWARFDIE(die.GetCU(),parent_die), sc))
Ravitheja Addepally40697302015-10-08 09:45:41 +00003340 sc = sc_backup;
3341
Greg Clayton6071e6f2015-08-26 22:57:51 +00003342 type_sp = ParseType(sc, die, NULL);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003343 }
3344 else if (type_ptr != DIE_IS_BEING_PARSED)
3345 {
3346 // Grab the existing type from the master types lists
Greg Claytone1cd1be2012-01-29 20:56:30 +00003347 type_sp = type_ptr->shared_from_this();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003348 }
3349
3350 }
3351 return type_sp;
3352}
3353
Greg Clayton2bc22f82011-09-30 03:20:47 +00003354
Greg Clayton6071e6f2015-08-26 22:57:51 +00003355DWARFDIE
3356SymbolFileDWARF::GetDeclContextDIEContainingDIE (const DWARFDIE &orig_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003357{
Greg Clayton6071e6f2015-08-26 22:57:51 +00003358 if (orig_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003359 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003360 DWARFDIE die = orig_die;
Greg Clayton2bc22f82011-09-30 03:20:47 +00003361
Greg Clayton6071e6f2015-08-26 22:57:51 +00003362 while (die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003363 {
3364 // If this is the original DIE that we are searching for a declaration
3365 // for, then don't look in the cache as we don't want our own decl
3366 // context to be our decl context...
Greg Clayton6071e6f2015-08-26 22:57:51 +00003367 if (orig_die != die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003368 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003369 switch (die.Tag())
Greg Clayton2bc22f82011-09-30 03:20:47 +00003370 {
3371 case DW_TAG_compile_unit:
3372 case DW_TAG_namespace:
3373 case DW_TAG_structure_type:
3374 case DW_TAG_union_type:
3375 case DW_TAG_class_type:
Paul Hermand628cbb2015-09-15 23:44:17 +00003376 case DW_TAG_lexical_block:
3377 case DW_TAG_subprogram:
Greg Clayton2bc22f82011-09-30 03:20:47 +00003378 return die;
3379
3380 default:
3381 break;
3382 }
3383 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003384
3385 DWARFDIE spec_die = die.GetReferencedDIE(DW_AT_specification);
3386 if (spec_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003387 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003388 DWARFDIE decl_ctx_die = GetDeclContextDIEContainingDIE(spec_die);
3389 if (decl_ctx_die)
3390 return decl_ctx_die;
Greg Clayton2bc22f82011-09-30 03:20:47 +00003391 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003392
3393 DWARFDIE abs_die = die.GetReferencedDIE(DW_AT_abstract_origin);
3394 if (abs_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003395 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003396 DWARFDIE decl_ctx_die = GetDeclContextDIEContainingDIE(abs_die);
3397 if (decl_ctx_die)
3398 return decl_ctx_die;
Greg Clayton2bc22f82011-09-30 03:20:47 +00003399 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003400
3401 die = die.GetParent();
Greg Clayton2bc22f82011-09-30 03:20:47 +00003402 }
3403 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003404 return DWARFDIE();
Greg Clayton2bc22f82011-09-30 03:20:47 +00003405}
3406
3407
Greg Clayton901c5ca2011-12-03 04:40:03 +00003408Symbol *
3409SymbolFileDWARF::GetObjCClassSymbol (const ConstString &objc_class_name)
3410{
3411 Symbol *objc_class_symbol = NULL;
3412 if (m_obj_file)
3413 {
Greg Clayton3046e662013-07-10 01:23:25 +00003414 Symtab *symtab = m_obj_file->GetSymtab ();
Greg Clayton901c5ca2011-12-03 04:40:03 +00003415 if (symtab)
3416 {
3417 objc_class_symbol = symtab->FindFirstSymbolWithNameAndType (objc_class_name,
3418 eSymbolTypeObjCClass,
3419 Symtab::eDebugNo,
3420 Symtab::eVisibilityAny);
3421 }
3422 }
3423 return objc_class_symbol;
3424}
3425
Greg Claytonc7f03b62012-01-12 04:33:28 +00003426// Some compilers don't emit the DW_AT_APPLE_objc_complete_type attribute. If they don't
3427// then we can end up looking through all class types for a complete type and never find
3428// the full definition. We need to know if this attribute is supported, so we determine
3429// this here and cache th result. We also need to worry about the debug map DWARF file
3430// if we are doing darwin DWARF in .o file debugging.
3431bool
3432SymbolFileDWARF::Supports_DW_AT_APPLE_objc_complete_type (DWARFCompileUnit *cu)
3433{
3434 if (m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolCalculate)
3435 {
3436 m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolNo;
3437 if (cu && cu->Supports_DW_AT_APPLE_objc_complete_type())
3438 m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolYes;
3439 else
3440 {
3441 DWARFDebugInfo* debug_info = DebugInfo();
3442 const uint32_t num_compile_units = GetNumCompileUnits();
3443 for (uint32_t cu_idx = 0; cu_idx < num_compile_units; ++cu_idx)
3444 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00003445 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx);
3446 if (dwarf_cu != cu && dwarf_cu->Supports_DW_AT_APPLE_objc_complete_type())
Greg Claytonc7f03b62012-01-12 04:33:28 +00003447 {
3448 m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolYes;
3449 break;
3450 }
3451 }
3452 }
Greg Clayton1f746072012-08-29 21:13:06 +00003453 if (m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolNo && GetDebugMapSymfile ())
Greg Claytonc7f03b62012-01-12 04:33:28 +00003454 return m_debug_map_symfile->Supports_DW_AT_APPLE_objc_complete_type (this);
3455 }
3456 return m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolYes;
3457}
Greg Clayton901c5ca2011-12-03 04:40:03 +00003458
3459// This function can be used when a DIE is found that is a forward declaration
3460// DIE and we want to try and find a type that has the complete definition.
3461TypeSP
Greg Clayton6071e6f2015-08-26 22:57:51 +00003462SymbolFileDWARF::FindCompleteObjCDefinitionTypeForDIE (const DWARFDIE &die,
Greg Claytonc7f03b62012-01-12 04:33:28 +00003463 const ConstString &type_name,
3464 bool must_be_implementation)
Greg Clayton901c5ca2011-12-03 04:40:03 +00003465{
3466
3467 TypeSP type_sp;
3468
Greg Claytonc7f03b62012-01-12 04:33:28 +00003469 if (!type_name || (must_be_implementation && !GetObjCClassSymbol (type_name)))
Greg Clayton901c5ca2011-12-03 04:40:03 +00003470 return type_sp;
3471
3472 DIEArray die_offsets;
3473
3474 if (m_using_apple_tables)
3475 {
3476 if (m_apple_types_ap.get())
3477 {
3478 const char *name_cstr = type_name.GetCString();
Greg Clayton68221ec2012-01-18 20:58:12 +00003479 m_apple_types_ap->FindCompleteObjCClassByName (name_cstr, die_offsets, must_be_implementation);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003480 }
3481 }
3482 else
3483 {
3484 if (!m_indexed)
3485 Index ();
3486
3487 m_type_index.Find (type_name, die_offsets);
3488 }
3489
Greg Clayton901c5ca2011-12-03 04:40:03 +00003490 const size_t num_matches = die_offsets.size();
3491
Greg Clayton901c5ca2011-12-03 04:40:03 +00003492 if (num_matches)
3493 {
3494 DWARFDebugInfo* debug_info = DebugInfo();
3495 for (size_t i=0; i<num_matches; ++i)
3496 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003497 const DIERef& die_ref = die_offsets[i];
3498 DWARFDIE type_die = debug_info->GetDIE (die_ref);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003499
3500 if (type_die)
3501 {
3502 bool try_resolving_type = false;
3503
3504 // Don't try and resolve the DIE we are looking for with the DIE itself!
3505 if (type_die != die)
3506 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003507 switch (type_die.Tag())
Greg Clayton901c5ca2011-12-03 04:40:03 +00003508 {
Greg Claytonc7f03b62012-01-12 04:33:28 +00003509 case DW_TAG_class_type:
3510 case DW_TAG_structure_type:
3511 try_resolving_type = true;
3512 break;
3513 default:
3514 break;
Greg Clayton901c5ca2011-12-03 04:40:03 +00003515 }
3516 }
3517
3518 if (try_resolving_type)
3519 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003520 if (must_be_implementation && type_die.Supports_DW_AT_APPLE_objc_complete_type())
3521 try_resolving_type = type_die.GetAttributeValueAsUnsigned (DW_AT_APPLE_objc_complete_type, 0);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003522
3523 if (try_resolving_type)
3524 {
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003525 Type *resolved_type = ResolveType (type_die, false, true);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003526 if (resolved_type && resolved_type != DIE_IS_BEING_PARSED)
3527 {
Ed Mastea0191d12013-10-17 20:42:56 +00003528 DEBUG_PRINTF ("resolved 0x%8.8" PRIx64 " from %s to 0x%8.8" PRIx64 " (cu 0x%8.8" PRIx64 ")\n",
Greg Clayton6071e6f2015-08-26 22:57:51 +00003529 die.GetID(),
Jim Ingham4af59612014-12-19 19:20:44 +00003530 m_obj_file->GetFileSpec().GetFilename().AsCString("<Unknown>"),
Greg Clayton6071e6f2015-08-26 22:57:51 +00003531 type_die.GetID(),
3532 type_cu->GetID());
Greg Clayton901c5ca2011-12-03 04:40:03 +00003533
Greg Claytonc7f03b62012-01-12 04:33:28 +00003534 if (die)
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003535 GetDIEToType()[die.GetDIE()] = resolved_type;
Greg Claytone1cd1be2012-01-29 20:56:30 +00003536 type_sp = resolved_type->shared_from_this();
Greg Clayton901c5ca2011-12-03 04:40:03 +00003537 break;
3538 }
3539 }
3540 }
3541 }
3542 else
3543 {
3544 if (m_using_apple_tables)
3545 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00003546 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_types accelerator table had bad die 0x%8.8x for '%s')\n",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003547 die_ref.die_offset, type_name.GetCString());
Greg Clayton901c5ca2011-12-03 04:40:03 +00003548 }
3549 }
3550
3551 }
3552 }
3553 return type_sp;
3554}
3555
Greg Claytona8022fa2012-04-24 21:22:41 +00003556
Greg Clayton80c26302012-02-05 06:12:47 +00003557//----------------------------------------------------------------------
3558// This function helps to ensure that the declaration contexts match for
3559// two different DIEs. Often times debug information will refer to a
3560// forward declaration of a type (the equivalent of "struct my_struct;".
3561// There will often be a declaration of that type elsewhere that has the
3562// full definition. When we go looking for the full type "my_struct", we
3563// will find one or more matches in the accelerator tables and we will
3564// then need to make sure the type was in the same declaration context
3565// as the original DIE. This function can efficiently compare two DIEs
3566// and will return true when the declaration context matches, and false
3567// when they don't.
3568//----------------------------------------------------------------------
Greg Clayton890ff562012-02-02 05:48:16 +00003569bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00003570SymbolFileDWARF::DIEDeclContextsMatch (const DWARFDIE &die1,
3571 const DWARFDIE &die2)
Greg Clayton890ff562012-02-02 05:48:16 +00003572{
Greg Claytona8022fa2012-04-24 21:22:41 +00003573 if (die1 == die2)
3574 return true;
3575
Greg Clayton890ff562012-02-02 05:48:16 +00003576 DWARFDIECollection decl_ctx_1;
3577 DWARFDIECollection decl_ctx_2;
Greg Clayton80c26302012-02-05 06:12:47 +00003578 //The declaration DIE stack is a stack of the declaration context
3579 // DIEs all the way back to the compile unit. If a type "T" is
3580 // declared inside a class "B", and class "B" is declared inside
3581 // a class "A" and class "A" is in a namespace "lldb", and the
3582 // namespace is in a compile unit, there will be a stack of DIEs:
3583 //
3584 // [0] DW_TAG_class_type for "B"
3585 // [1] DW_TAG_class_type for "A"
3586 // [2] DW_TAG_namespace for "lldb"
3587 // [3] DW_TAG_compile_unit for the source file.
3588 //
3589 // We grab both contexts and make sure that everything matches
3590 // all the way back to the compiler unit.
3591
3592 // First lets grab the decl contexts for both DIEs
Greg Clayton6071e6f2015-08-26 22:57:51 +00003593 die1.GetDeclContextDIEs (decl_ctx_1);
3594 die2.GetDeclContextDIEs (decl_ctx_2);
Greg Clayton80c26302012-02-05 06:12:47 +00003595 // Make sure the context arrays have the same size, otherwise
3596 // we are done
Greg Clayton890ff562012-02-02 05:48:16 +00003597 const size_t count1 = decl_ctx_1.Size();
3598 const size_t count2 = decl_ctx_2.Size();
3599 if (count1 != count2)
3600 return false;
Greg Clayton80c26302012-02-05 06:12:47 +00003601
Bruce Mitchenerd93c4a32014-07-01 21:22:11 +00003602 // Make sure the DW_TAG values match all the way back up the
Greg Clayton80c26302012-02-05 06:12:47 +00003603 // compile unit. If they don't, then we are done.
Greg Clayton6071e6f2015-08-26 22:57:51 +00003604 DWARFDIE decl_ctx_die1;
3605 DWARFDIE decl_ctx_die2;
Greg Clayton890ff562012-02-02 05:48:16 +00003606 size_t i;
3607 for (i=0; i<count1; i++)
3608 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003609 decl_ctx_die1 = decl_ctx_1.GetDIEAtIndex (i);
3610 decl_ctx_die2 = decl_ctx_2.GetDIEAtIndex (i);
3611 if (decl_ctx_die1.Tag() != decl_ctx_die2.Tag())
Greg Clayton890ff562012-02-02 05:48:16 +00003612 return false;
3613 }
Greg Clayton890ff562012-02-02 05:48:16 +00003614#if defined LLDB_CONFIGURATION_DEBUG
Greg Clayton80c26302012-02-05 06:12:47 +00003615
3616 // Make sure the top item in the decl context die array is always
3617 // DW_TAG_compile_unit. If it isn't then something went wrong in
Greg Clayton5ce1a842015-08-27 18:09:44 +00003618 // the DWARFDIE::GetDeclContextDIEs() function...
Greg Clayton6071e6f2015-08-26 22:57:51 +00003619 assert (decl_ctx_1.GetDIEAtIndex (count1 - 1).Tag() == DW_TAG_compile_unit);
Greg Clayton80c26302012-02-05 06:12:47 +00003620
Greg Clayton890ff562012-02-02 05:48:16 +00003621#endif
3622 // Always skip the compile unit when comparing by only iterating up to
Greg Clayton80c26302012-02-05 06:12:47 +00003623 // "count - 1". Here we compare the names as we go.
Greg Clayton890ff562012-02-02 05:48:16 +00003624 for (i=0; i<count1 - 1; i++)
3625 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003626 decl_ctx_die1 = decl_ctx_1.GetDIEAtIndex (i);
3627 decl_ctx_die2 = decl_ctx_2.GetDIEAtIndex (i);
3628 const char *name1 = decl_ctx_die1.GetName();
3629 const char *name2 = decl_ctx_die2.GetName();
Greg Clayton890ff562012-02-02 05:48:16 +00003630 // If the string was from a DW_FORM_strp, then the pointer will often
3631 // be the same!
Greg Clayton5569e642012-02-06 01:44:54 +00003632 if (name1 == name2)
3633 continue;
3634
3635 // Name pointers are not equal, so only compare the strings
3636 // if both are not NULL.
3637 if (name1 && name2)
Greg Clayton890ff562012-02-02 05:48:16 +00003638 {
Greg Clayton5569e642012-02-06 01:44:54 +00003639 // If the strings don't compare, we are done...
3640 if (strcmp(name1, name2) != 0)
Greg Clayton890ff562012-02-02 05:48:16 +00003641 return false;
Greg Clayton5569e642012-02-06 01:44:54 +00003642 }
3643 else
3644 {
3645 // One name was NULL while the other wasn't
3646 return false;
Greg Clayton890ff562012-02-02 05:48:16 +00003647 }
3648 }
Greg Clayton80c26302012-02-05 06:12:47 +00003649 // We made it through all of the checks and the declaration contexts
3650 // are equal.
Greg Clayton890ff562012-02-02 05:48:16 +00003651 return true;
3652}
Greg Clayton220a0072011-12-09 08:48:30 +00003653
Greg Clayton2ccf8cf2010-11-07 21:02:03 +00003654
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003655TypeSP
Greg Claytona8022fa2012-04-24 21:22:41 +00003656SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext (const DWARFDeclContext &dwarf_decl_ctx)
3657{
3658 TypeSP type_sp;
3659
3660 const uint32_t dwarf_decl_ctx_count = dwarf_decl_ctx.GetSize();
3661 if (dwarf_decl_ctx_count > 0)
3662 {
3663 const ConstString type_name(dwarf_decl_ctx[0].name);
3664 const dw_tag_t tag = dwarf_decl_ctx[0].tag;
3665
3666 if (type_name)
3667 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003668 Log *log (LogChannelDWARF::GetLogIfAny(DWARF_LOG_TYPE_COMPLETION|DWARF_LOG_LOOKUPS));
Greg Claytona8022fa2012-04-24 21:22:41 +00003669 if (log)
3670 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003671 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytona8022fa2012-04-24 21:22:41 +00003672 "SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext(tag=%s, qualified-name='%s')",
3673 DW_TAG_value_to_name(dwarf_decl_ctx[0].tag),
3674 dwarf_decl_ctx.GetQualifiedName());
3675 }
3676
3677 DIEArray die_offsets;
3678
3679 if (m_using_apple_tables)
3680 {
3681 if (m_apple_types_ap.get())
3682 {
Greg Claytoncb9c8cf2013-02-06 23:56:13 +00003683 const bool has_tag = m_apple_types_ap->GetHeader().header_data.ContainsAtom (DWARFMappedHash::eAtomTypeTag);
3684 const bool has_qualified_name_hash = m_apple_types_ap->GetHeader().header_data.ContainsAtom (DWARFMappedHash::eAtomTypeQualNameHash);
3685 if (has_tag && has_qualified_name_hash)
Greg Claytona8022fa2012-04-24 21:22:41 +00003686 {
Greg Claytoncb9c8cf2013-02-06 23:56:13 +00003687 const char *qualified_name = dwarf_decl_ctx.GetQualifiedName();
3688 const uint32_t qualified_name_hash = MappedHash::HashStringUsingDJB (qualified_name);
3689 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00003690 GetObjectFile()->GetModule()->LogMessage (log,"FindByNameAndTagAndQualifiedNameHash()");
Greg Claytoncb9c8cf2013-02-06 23:56:13 +00003691 m_apple_types_ap->FindByNameAndTagAndQualifiedNameHash (type_name.GetCString(), tag, qualified_name_hash, die_offsets);
3692 }
3693 else if (has_tag)
3694 {
3695 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00003696 GetObjectFile()->GetModule()->LogMessage (log,"FindByNameAndTag()");
Greg Claytona8022fa2012-04-24 21:22:41 +00003697 m_apple_types_ap->FindByNameAndTag (type_name.GetCString(), tag, die_offsets);
3698 }
3699 else
3700 {
3701 m_apple_types_ap->FindByName (type_name.GetCString(), die_offsets);
3702 }
3703 }
3704 }
3705 else
3706 {
3707 if (!m_indexed)
3708 Index ();
3709
3710 m_type_index.Find (type_name, die_offsets);
3711 }
3712
3713 const size_t num_matches = die_offsets.size();
3714
3715
Greg Claytona8022fa2012-04-24 21:22:41 +00003716 if (num_matches)
3717 {
3718 DWARFDebugInfo* debug_info = DebugInfo();
3719 for (size_t i=0; i<num_matches; ++i)
3720 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003721 const DIERef& die_ref = die_offsets[i];
3722 DWARFDIE type_die = debug_info->GetDIE (die_ref);
Greg Claytona8022fa2012-04-24 21:22:41 +00003723
3724 if (type_die)
3725 {
3726 bool try_resolving_type = false;
3727
3728 // Don't try and resolve the DIE we are looking for with the DIE itself!
Greg Clayton6071e6f2015-08-26 22:57:51 +00003729 const dw_tag_t type_tag = type_die.Tag();
Greg Claytona8022fa2012-04-24 21:22:41 +00003730 // Make sure the tags match
3731 if (type_tag == tag)
3732 {
3733 // The tags match, lets try resolving this type
3734 try_resolving_type = true;
3735 }
3736 else
3737 {
3738 // The tags don't match, but we need to watch our for a
3739 // forward declaration for a struct and ("struct foo")
3740 // ends up being a class ("class foo { ... };") or
3741 // vice versa.
3742 switch (type_tag)
3743 {
3744 case DW_TAG_class_type:
3745 // We had a "class foo", see if we ended up with a "struct foo { ... };"
3746 try_resolving_type = (tag == DW_TAG_structure_type);
3747 break;
3748 case DW_TAG_structure_type:
3749 // We had a "struct foo", see if we ended up with a "class foo { ... };"
3750 try_resolving_type = (tag == DW_TAG_class_type);
3751 break;
3752 default:
3753 // Tags don't match, don't event try to resolve
3754 // using this type whose name matches....
3755 break;
3756 }
3757 }
3758
3759 if (try_resolving_type)
3760 {
3761 DWARFDeclContext type_dwarf_decl_ctx;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003762 type_die.GetDWARFDeclContext (type_dwarf_decl_ctx);
Greg Claytona8022fa2012-04-24 21:22:41 +00003763
3764 if (log)
3765 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003766 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytona8022fa2012-04-24 21:22:41 +00003767 "SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext(tag=%s, qualified-name='%s') trying die=0x%8.8x (%s)",
3768 DW_TAG_value_to_name(dwarf_decl_ctx[0].tag),
3769 dwarf_decl_ctx.GetQualifiedName(),
Greg Clayton6071e6f2015-08-26 22:57:51 +00003770 type_die.GetOffset(),
Greg Claytona8022fa2012-04-24 21:22:41 +00003771 type_dwarf_decl_ctx.GetQualifiedName());
3772 }
3773
3774 // Make sure the decl contexts match all the way up
3775 if (dwarf_decl_ctx == type_dwarf_decl_ctx)
3776 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003777 Type *resolved_type = ResolveType (type_die, false);
Greg Claytona8022fa2012-04-24 21:22:41 +00003778 if (resolved_type && resolved_type != DIE_IS_BEING_PARSED)
3779 {
3780 type_sp = resolved_type->shared_from_this();
3781 break;
3782 }
3783 }
3784 }
3785 else
3786 {
3787 if (log)
3788 {
3789 std::string qualified_name;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003790 type_die.GetQualifiedName(qualified_name);
Greg Clayton5160ce52013-03-27 23:08:40 +00003791 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytona8022fa2012-04-24 21:22:41 +00003792 "SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext(tag=%s, qualified-name='%s') ignoring die=0x%8.8x (%s)",
3793 DW_TAG_value_to_name(dwarf_decl_ctx[0].tag),
3794 dwarf_decl_ctx.GetQualifiedName(),
Greg Clayton6071e6f2015-08-26 22:57:51 +00003795 type_die.GetOffset(),
Greg Claytona8022fa2012-04-24 21:22:41 +00003796 qualified_name.c_str());
3797 }
3798 }
3799 }
3800 else
3801 {
3802 if (m_using_apple_tables)
3803 {
3804 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_types accelerator table had bad die 0x%8.8x for '%s')\n",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003805 die_ref.die_offset, type_name.GetCString());
Greg Claytona8022fa2012-04-24 21:22:41 +00003806 }
3807 }
3808
3809 }
3810 }
3811 }
3812 }
3813 return type_sp;
3814}
3815
Greg Claytona8022fa2012-04-24 21:22:41 +00003816TypeSP
Greg Clayton6071e6f2015-08-26 22:57:51 +00003817SymbolFileDWARF::ParseType (const SymbolContext& sc, const DWARFDIE &die, bool *type_is_new_ptr)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003818{
Greg Clayton196e8cd2015-08-17 20:31:46 +00003819 TypeSP type_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003820
Greg Clayton6071e6f2015-08-26 22:57:51 +00003821 if (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003822 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003823 TypeSystem *type_system = GetTypeSystemForLanguage(die.GetCU()->GetLanguageType());
3824
3825 if (type_system)
Greg Clayton196e8cd2015-08-17 20:31:46 +00003826 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00003827 DWARFASTParser *dwarf_ast = type_system->GetDWARFParser();
3828 if (dwarf_ast)
Greg Clayton6071e6f2015-08-26 22:57:51 +00003829 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00003830 Log *log = LogChannelDWARF::GetLogIfAll(DWARF_LOG_DEBUG_INFO);
3831 type_sp = dwarf_ast->ParseTypeFromDWARF (sc, die, log, type_is_new_ptr);
3832 if (type_sp)
3833 {
3834 TypeList* type_list = GetTypeList();
3835 if (type_list)
3836 type_list->Insert(type_sp);
Siva Chandra9293fc42016-01-07 23:32:34 +00003837
3838 if (die.Tag() == DW_TAG_subprogram)
3839 {
3840 DIERef die_ref = die.GetDIERef();
3841 std::string scope_qualified_name(GetDeclContextForUID(die.GetID()).GetScopeQualifiedName().AsCString(""));
3842 if (scope_qualified_name.size())
3843 {
3844 NameToOffsetMap::iterator iter = m_function_scope_qualified_name_map.find(scope_qualified_name);
3845 if (iter != m_function_scope_qualified_name_map.end())
3846 (*iter).second->insert(die_ref);
3847 else
3848 {
3849 DIERefSetSP new_set(new std::set<DIERef>);
3850 new_set->insert(die_ref);
3851 m_function_scope_qualified_name_map.emplace(std::make_pair(scope_qualified_name, new_set));
3852 }
3853 }
3854 }
Greg Clayton261ac3f2015-08-28 01:01:03 +00003855 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003856 }
Greg Clayton196e8cd2015-08-17 20:31:46 +00003857 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003858 }
Greg Clayton196e8cd2015-08-17 20:31:46 +00003859
3860 return type_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003861}
3862
3863size_t
Greg Clayton1be10fc2010-09-29 01:12:09 +00003864SymbolFileDWARF::ParseTypes
3865(
3866 const SymbolContext& sc,
Greg Clayton6071e6f2015-08-26 22:57:51 +00003867 const DWARFDIE &orig_die,
Greg Clayton1be10fc2010-09-29 01:12:09 +00003868 bool parse_siblings,
3869 bool parse_children
3870)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003871{
3872 size_t types_added = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003873 DWARFDIE die = orig_die;
3874 while (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003875 {
3876 bool type_is_new = false;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003877 if (ParseType(sc, die, &type_is_new).get())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003878 {
3879 if (type_is_new)
3880 ++types_added;
3881 }
3882
Greg Clayton6071e6f2015-08-26 22:57:51 +00003883 if (parse_children && die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003884 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003885 if (die.Tag() == DW_TAG_subprogram)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003886 {
3887 SymbolContext child_sc(sc);
Greg Clayton6071e6f2015-08-26 22:57:51 +00003888 child_sc.function = sc.comp_unit->FindFunctionByUID(die.GetID()).get();
3889 types_added += ParseTypes(child_sc, die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003890 }
3891 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00003892 types_added += ParseTypes(sc, die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003893 }
3894
3895 if (parse_siblings)
Greg Clayton6071e6f2015-08-26 22:57:51 +00003896 die = die.GetSibling();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003897 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00003898 die.Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003899 }
3900 return types_added;
3901}
3902
3903
3904size_t
3905SymbolFileDWARF::ParseFunctionBlocks (const SymbolContext &sc)
3906{
3907 assert(sc.comp_unit && sc.function);
3908 size_t functions_added = 0;
Greg Clayton1f746072012-08-29 21:13:06 +00003909 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003910 if (dwarf_cu)
3911 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003912 const dw_offset_t function_die_offset = sc.function->GetID();
3913 DWARFDIE function_die = dwarf_cu->GetDIE (function_die_offset);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003914 if (function_die)
3915 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003916 ParseFunctionBlocks(sc, &sc.function->GetBlock (false), function_die, LLDB_INVALID_ADDRESS, 0);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003917 }
3918 }
3919
3920 return functions_added;
3921}
3922
3923
3924size_t
3925SymbolFileDWARF::ParseTypes (const SymbolContext &sc)
3926{
3927 // At least a compile unit must be valid
3928 assert(sc.comp_unit);
3929 size_t types_added = 0;
Greg Clayton1f746072012-08-29 21:13:06 +00003930 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003931 if (dwarf_cu)
3932 {
3933 if (sc.function)
3934 {
3935 dw_offset_t function_die_offset = sc.function->GetID();
Greg Clayton6071e6f2015-08-26 22:57:51 +00003936 DWARFDIE func_die = dwarf_cu->GetDIE(function_die_offset);
3937 if (func_die && func_die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003938 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003939 types_added = ParseTypes(sc, func_die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003940 }
3941 }
3942 else
3943 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003944 DWARFDIE dwarf_cu_die = dwarf_cu->DIE();
3945 if (dwarf_cu_die && dwarf_cu_die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003946 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003947 types_added = ParseTypes(sc, dwarf_cu_die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003948 }
3949 }
3950 }
3951
3952 return types_added;
3953}
3954
3955size_t
3956SymbolFileDWARF::ParseVariablesForContext (const SymbolContext& sc)
3957{
3958 if (sc.comp_unit != NULL)
3959 {
Greg Clayton4b3dc102010-11-01 20:32:12 +00003960 DWARFDebugInfo* info = DebugInfo();
3961 if (info == NULL)
3962 return 0;
3963
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003964 if (sc.function)
3965 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003966 DWARFDIE function_die = info->GetDIE(DIERef(sc.function->GetID()));
Greg Clayton9422dd62013-03-04 21:46:16 +00003967
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003968 const dw_addr_t func_lo_pc = function_die.GetAttributeValueAsAddress (DW_AT_low_pc, LLDB_INVALID_ADDRESS);
Greg Claytonc7bece562013-01-25 18:06:21 +00003969 if (func_lo_pc != LLDB_INVALID_ADDRESS)
Greg Claytone38a5ed2012-01-05 03:57:59 +00003970 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003971 const size_t num_variables = ParseVariables(sc, function_die.GetFirstChild(), func_lo_pc, true, true);
Greg Claytonc662ec82011-06-17 22:10:16 +00003972
Greg Claytone38a5ed2012-01-05 03:57:59 +00003973 // Let all blocks know they have parse all their variables
3974 sc.function->GetBlock (false).SetDidParseVariables (true, true);
3975 return num_variables;
3976 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003977 }
3978 else if (sc.comp_unit)
3979 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003980 DWARFCompileUnit* dwarf_cu = info->GetCompileUnit(sc.comp_unit->GetID());
Greg Clayton9422dd62013-03-04 21:46:16 +00003981
3982 if (dwarf_cu == NULL)
3983 return 0;
3984
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003985 uint32_t vars_added = 0;
3986 VariableListSP variables (sc.comp_unit->GetVariableList(false));
3987
3988 if (variables.get() == NULL)
3989 {
3990 variables.reset(new VariableList());
3991 sc.comp_unit->SetVariableList(variables);
3992
Greg Claytond4a2b372011-09-12 23:21:58 +00003993 DIEArray die_offsets;
Greg Clayton97fbc342011-10-20 22:30:33 +00003994 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00003995 {
Greg Clayton97fbc342011-10-20 22:30:33 +00003996 if (m_apple_names_ap.get())
Greg Claytond1767f02011-12-08 02:13:16 +00003997 {
3998 DWARFMappedHash::DIEInfoArray hash_data_array;
3999 if (m_apple_names_ap->AppendAllDIEsInRange (dwarf_cu->GetOffset(),
4000 dwarf_cu->GetNextCompileUnitOffset(),
4001 hash_data_array))
4002 {
4003 DWARFMappedHash::ExtractDIEArray (hash_data_array, die_offsets);
4004 }
4005 }
Greg Clayton7f995132011-10-04 22:41:51 +00004006 }
4007 else
4008 {
4009 // Index if we already haven't to make sure the compile units
4010 // get indexed and make their global DIE index list
4011 if (!m_indexed)
4012 Index ();
4013
4014 m_global_index.FindAllEntriesForCompileUnit (dwarf_cu->GetOffset(),
Greg Clayton7f995132011-10-04 22:41:51 +00004015 die_offsets);
4016 }
4017
4018 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00004019 if (num_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004020 {
Greg Claytond4a2b372011-09-12 23:21:58 +00004021 DWARFDebugInfo* debug_info = DebugInfo();
4022 for (size_t i=0; i<num_matches; ++i)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004023 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004024 const DIERef& die_ref = die_offsets[i];
4025 DWARFDIE die = debug_info->GetDIE (die_ref);
Greg Clayton95d87902011-11-11 03:16:25 +00004026 if (die)
Greg Claytond4a2b372011-09-12 23:21:58 +00004027 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004028 VariableSP var_sp (ParseVariableDIE(sc, die, LLDB_INVALID_ADDRESS));
Greg Clayton95d87902011-11-11 03:16:25 +00004029 if (var_sp)
4030 {
4031 variables->AddVariableIfUnique (var_sp);
4032 ++vars_added;
4033 }
Greg Claytond4a2b372011-09-12 23:21:58 +00004034 }
Greg Clayton95d87902011-11-11 03:16:25 +00004035 else
4036 {
4037 if (m_using_apple_tables)
4038 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004039 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_names accelerator table had bad die 0x%8.8x)\n", die_ref.die_offset);
Greg Clayton95d87902011-11-11 03:16:25 +00004040 }
4041 }
4042
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004043 }
4044 }
4045 }
4046 return vars_added;
4047 }
4048 }
4049 return 0;
4050}
4051
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004052VariableSP
4053SymbolFileDWARF::ParseVariableDIE
4054(
4055 const SymbolContext& sc,
Greg Clayton6071e6f2015-08-26 22:57:51 +00004056 const DWARFDIE &die,
Greg Clayton016a95e2010-09-14 02:20:48 +00004057 const lldb::addr_t func_low_pc
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004058)
4059{
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004060 if (die.GetDWARF() != this)
4061 return die.GetDWARF()->ParseVariableDIE(sc, die, func_low_pc);
4062
Greg Clayton6071e6f2015-08-26 22:57:51 +00004063 VariableSP var_sp;
4064 if (!die)
4065 return var_sp;
4066
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004067 var_sp = GetDIEToVariable()[die.GetDIE()];
Greg Clayton83c5cd92010-11-14 22:13:40 +00004068 if (var_sp)
4069 return var_sp; // Already been parsed!
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004070
Greg Clayton6071e6f2015-08-26 22:57:51 +00004071 const dw_tag_t tag = die.Tag();
Richard Mitton0a558352013-10-17 21:14:00 +00004072 ModuleSP module = GetObjectFile()->GetModule();
Greg Clayton7f995132011-10-04 22:41:51 +00004073
4074 if ((tag == DW_TAG_variable) ||
4075 (tag == DW_TAG_constant) ||
4076 (tag == DW_TAG_formal_parameter && sc.function))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004077 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004078 DWARFAttributes attributes;
4079 const size_t num_attributes = die.GetAttributes(attributes);
Paul Hermand628cbb2015-09-15 23:44:17 +00004080 DWARFDIE spec_die;
Greg Clayton7f995132011-10-04 22:41:51 +00004081 if (num_attributes > 0)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004082 {
Greg Clayton7f995132011-10-04 22:41:51 +00004083 const char *name = NULL;
4084 const char *mangled = NULL;
4085 Declaration decl;
4086 uint32_t i;
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004087 DWARFFormValue type_die_form;
Greg Clayton6071e6f2015-08-26 22:57:51 +00004088 DWARFExpression location(die.GetCU());
Greg Clayton7f995132011-10-04 22:41:51 +00004089 bool is_external = false;
4090 bool is_artificial = false;
4091 bool location_is_const_value_data = false;
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004092 bool has_explicit_location = false;
Enrico Granata4ec130d2014-08-11 19:16:35 +00004093 DWARFFormValue const_value;
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004094 Variable::RangeList scope_ranges;
Greg Clayton23f59502012-07-17 03:23:13 +00004095 //AccessType accessibility = eAccessNone;
Greg Clayton7f995132011-10-04 22:41:51 +00004096
4097 for (i=0; i<num_attributes; ++i)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004098 {
Greg Clayton7f995132011-10-04 22:41:51 +00004099 dw_attr_t attr = attributes.AttributeAtIndex(i);
4100 DWARFFormValue form_value;
Greg Clayton54166af2014-11-22 01:58:59 +00004101
Greg Clayton6071e6f2015-08-26 22:57:51 +00004102 if (attributes.ExtractFormValueAtIndex(i, form_value))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004103 {
Greg Clayton7f995132011-10-04 22:41:51 +00004104 switch (attr)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004105 {
Greg Clayton7f995132011-10-04 22:41:51 +00004106 case DW_AT_decl_file: decl.SetFile(sc.comp_unit->GetSupportFiles().GetFileSpecAtIndex(form_value.Unsigned())); break;
4107 case DW_AT_decl_line: decl.SetLine(form_value.Unsigned()); break;
4108 case DW_AT_decl_column: decl.SetColumn(form_value.Unsigned()); break;
Greg Clayton6071e6f2015-08-26 22:57:51 +00004109 case DW_AT_name: name = form_value.AsCString(); break;
Greg Clayton71415542012-12-08 00:24:40 +00004110 case DW_AT_linkage_name:
Greg Clayton6071e6f2015-08-26 22:57:51 +00004111 case DW_AT_MIPS_linkage_name: mangled = form_value.AsCString(); break;
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004112 case DW_AT_type: type_die_form = form_value; break;
Greg Clayton1c8ef472013-04-05 23:27:21 +00004113 case DW_AT_external: is_external = form_value.Boolean(); break;
Greg Clayton7f995132011-10-04 22:41:51 +00004114 case DW_AT_const_value:
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004115 // If we have already found a DW_AT_location attribute, ignore this attribute.
4116 if (!has_explicit_location)
4117 {
4118 location_is_const_value_data = true;
4119 // The constant value will be either a block, a data value or a string.
Ed Masteeeae7212013-10-24 20:43:47 +00004120 const DWARFDataExtractor& debug_info_data = get_debug_info_data();
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004121 if (DWARFFormValue::IsBlockForm(form_value.Form()))
4122 {
4123 // Retrieve the value as a block expression.
4124 uint32_t block_offset = form_value.BlockData() - debug_info_data.GetDataStart();
4125 uint32_t block_length = form_value.Unsigned();
Richard Mitton0a558352013-10-17 21:14:00 +00004126 location.CopyOpcodeData(module, debug_info_data, block_offset, block_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004127 }
4128 else if (DWARFFormValue::IsDataForm(form_value.Form()))
4129 {
4130 // Retrieve the value as a data expression.
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004131 DWARFFormValue::FixedFormSizes fixed_form_sizes =
4132 DWARFFormValue::GetFixedFormSizesForAddressSize (
4133 attributes.CompileUnitAtIndex(i)->GetAddressByteSize(),
4134 attributes.CompileUnitAtIndex(i)->IsDWARF64());
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004135 uint32_t data_offset = attributes.DIEOffsetAtIndex(i);
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004136 uint32_t data_length = fixed_form_sizes.GetSize(form_value.Form());
Enrico Granata4ec130d2014-08-11 19:16:35 +00004137 if (data_length == 0)
4138 {
4139 const uint8_t *data_pointer = form_value.BlockData();
4140 if (data_pointer)
4141 {
Jason Molenda18f5fd32014-10-16 07:52:17 +00004142 form_value.Unsigned();
Enrico Granata4ec130d2014-08-11 19:16:35 +00004143 }
4144 else if (DWARFFormValue::IsDataForm(form_value.Form()))
4145 {
4146 // we need to get the byte size of the type later after we create the variable
4147 const_value = form_value;
4148 }
4149 }
4150 else
4151 location.CopyOpcodeData(module, debug_info_data, data_offset, data_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004152 }
4153 else
4154 {
4155 // Retrieve the value as a string expression.
4156 if (form_value.Form() == DW_FORM_strp)
4157 {
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004158 DWARFFormValue::FixedFormSizes fixed_form_sizes =
4159 DWARFFormValue::GetFixedFormSizesForAddressSize (
4160 attributes.CompileUnitAtIndex(i)->GetAddressByteSize(),
4161 attributes.CompileUnitAtIndex(i)->IsDWARF64());
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004162 uint32_t data_offset = attributes.DIEOffsetAtIndex(i);
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004163 uint32_t data_length = fixed_form_sizes.GetSize(form_value.Form());
Richard Mitton0a558352013-10-17 21:14:00 +00004164 location.CopyOpcodeData(module, debug_info_data, data_offset, data_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004165 }
4166 else
4167 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004168 const char *str = form_value.AsCString();
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004169 uint32_t string_offset = str - (const char *)debug_info_data.GetDataStart();
4170 uint32_t string_length = strlen(str) + 1;
Richard Mitton0a558352013-10-17 21:14:00 +00004171 location.CopyOpcodeData(module, debug_info_data, string_offset, string_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004172 }
4173 }
4174 }
4175 break;
Greg Clayton7f995132011-10-04 22:41:51 +00004176 case DW_AT_location:
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004177 {
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004178 location_is_const_value_data = false;
4179 has_explicit_location = true;
Greg Clayton7f995132011-10-04 22:41:51 +00004180 if (form_value.BlockData())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004181 {
Ed Masteeeae7212013-10-24 20:43:47 +00004182 const DWARFDataExtractor& debug_info_data = get_debug_info_data();
Greg Clayton7f995132011-10-04 22:41:51 +00004183
4184 uint32_t block_offset = form_value.BlockData() - debug_info_data.GetDataStart();
4185 uint32_t block_length = form_value.Unsigned();
Richard Mitton0a558352013-10-17 21:14:00 +00004186 location.CopyOpcodeData(module, get_debug_info_data(), block_offset, block_length);
Greg Clayton7f995132011-10-04 22:41:51 +00004187 }
4188 else
4189 {
Tamas Berghammer1f5e4482015-09-16 12:37:06 +00004190 const DWARFDataExtractor& debug_loc_data = get_debug_loc_data();
Greg Clayton7f995132011-10-04 22:41:51 +00004191 const dw_offset_t debug_loc_offset = form_value.Unsigned();
4192
Tamas Berghammer1f5e4482015-09-16 12:37:06 +00004193 size_t loc_list_length = DWARFExpression::LocationListSize(die.GetCU(), debug_loc_data, debug_loc_offset);
Greg Clayton7f995132011-10-04 22:41:51 +00004194 if (loc_list_length > 0)
4195 {
Richard Mitton0a558352013-10-17 21:14:00 +00004196 location.CopyOpcodeData(module, debug_loc_data, debug_loc_offset, loc_list_length);
Greg Clayton7f995132011-10-04 22:41:51 +00004197 assert (func_low_pc != LLDB_INVALID_ADDRESS);
Greg Clayton54166af2014-11-22 01:58:59 +00004198 location.SetLocationListSlide (func_low_pc - attributes.CompileUnitAtIndex(i)->GetBaseAddress());
Greg Clayton7f995132011-10-04 22:41:51 +00004199 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004200 }
4201 }
Greg Clayton7f995132011-10-04 22:41:51 +00004202 break;
Paul Hermand628cbb2015-09-15 23:44:17 +00004203 case DW_AT_specification:
Greg Clayton31460392016-03-18 20:33:49 +00004204 {
4205 DWARFDebugInfo* debug_info = DebugInfo();
4206 if (debug_info)
4207 spec_die = debug_info->GetDIE(DIERef(form_value));
4208 }
Paul Hermand628cbb2015-09-15 23:44:17 +00004209 break;
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004210 case DW_AT_start_scope:
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004211 {
Greg Clayton31460392016-03-18 20:33:49 +00004212 if (form_value.Form() == DW_FORM_sec_offset)
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004213 {
Greg Clayton31460392016-03-18 20:33:49 +00004214 DWARFRangeList dwarf_scope_ranges;
4215 const DWARFDebugRanges* debug_ranges = DebugRanges();
4216 debug_ranges->FindRanges(form_value.Unsigned(), dwarf_scope_ranges);
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004217
Greg Clayton31460392016-03-18 20:33:49 +00004218 // All DW_AT_start_scope are relative to the base address of the
4219 // compile unit. We add the compile unit base address to make
4220 // sure all the addresses are properly fixed up.
4221 for (size_t i = 0, count = dwarf_scope_ranges.GetSize(); i < count; ++i)
4222 {
4223 const DWARFRangeList::Entry& range = dwarf_scope_ranges.GetEntryRef(i);
4224 scope_ranges.Append(range.GetRangeBase() + die.GetCU()->GetBaseAddress(),
4225 range.GetByteSize());
4226 }
4227 }
4228 else
4229 {
4230 // TODO: Handle the case when DW_AT_start_scope have form constant. The
4231 // dwarf spec is a bit ambiguous about what is the expected behavior in
4232 // case the enclosing block have a non coninious address range and the
4233 // DW_AT_start_scope entry have a form constant.
4234 GetObjectFile()->GetModule()->ReportWarning ("0x%8.8" PRIx64 ": DW_AT_start_scope has unsupported form type (0x%x)\n",
4235 die.GetID(),
4236 form_value.Form());
4237 }
4238
4239 scope_ranges.Sort();
4240 scope_ranges.CombineConsecutiveRanges();
4241 }
4242 break;
Greg Clayton1c8ef472013-04-05 23:27:21 +00004243 case DW_AT_artificial: is_artificial = form_value.Boolean(); break;
Greg Clayton23f59502012-07-17 03:23:13 +00004244 case DW_AT_accessibility: break; //accessibility = DW_ACCESS_to_AccessType(form_value.Unsigned()); break;
Greg Clayton7f995132011-10-04 22:41:51 +00004245 case DW_AT_declaration:
4246 case DW_AT_description:
4247 case DW_AT_endianity:
4248 case DW_AT_segment:
Greg Clayton7f995132011-10-04 22:41:51 +00004249 case DW_AT_visibility:
4250 default:
4251 case DW_AT_abstract_origin:
4252 case DW_AT_sibling:
Greg Clayton7f995132011-10-04 22:41:51 +00004253 break;
4254 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004255 }
4256 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004257
Greg Clayton6071e6f2015-08-26 22:57:51 +00004258 const DWARFDIE parent_context_die = GetDeclContextDIEContainingDIE(die);
4259 const dw_tag_t parent_tag = die.GetParent().Tag();
4260 bool is_static_member = parent_tag == DW_TAG_compile_unit && (parent_context_die.Tag() == DW_TAG_class_type || parent_context_die.Tag() == DW_TAG_structure_type);
Paul Herman10bc1a42015-08-18 22:46:57 +00004261
Greg Clayton9e9f2192013-05-17 00:55:28 +00004262 ValueType scope = eValueTypeInvalid;
4263
Greg Clayton6071e6f2015-08-26 22:57:51 +00004264 const DWARFDIE sc_parent_die = GetParentSymbolContextDIE(die);
Greg Clayton9e9f2192013-05-17 00:55:28 +00004265 SymbolContextScope * symbol_context_scope = NULL;
4266
Siva Chandra0783ab92015-03-24 18:32:27 +00004267 if (!mangled)
4268 {
4269 // LLDB relies on the mangled name (DW_TAG_linkage_name or DW_AT_MIPS_linkage_name) to
4270 // generate fully qualified names of global variables with commands like "frame var j".
4271 // For example, if j were an int variable holding a value 4 and declared in a namespace
4272 // B which in turn is contained in a namespace A, the command "frame var j" returns
4273 // "(int) A::B::j = 4". If the compiler does not emit a linkage name, we should be able
4274 // to generate a fully qualified name from the declaration context.
Greg Clayton6071e6f2015-08-26 22:57:51 +00004275 if (parent_tag == DW_TAG_compile_unit &&
Jim Ingham0e0984e2015-09-02 01:06:46 +00004276 Language::LanguageIsCPlusPlus(die.GetLanguage()))
Siva Chandra0783ab92015-03-24 18:32:27 +00004277 {
4278 DWARFDeclContext decl_ctx;
4279
Greg Clayton6071e6f2015-08-26 22:57:51 +00004280 die.GetDWARFDeclContext(decl_ctx);
Siva Chandra0783ab92015-03-24 18:32:27 +00004281 mangled = decl_ctx.GetQualifiedNameAsConstString().GetCString();
4282 }
4283 }
4284
Greg Clayton9e9f2192013-05-17 00:55:28 +00004285 // DWARF doesn't specify if a DW_TAG_variable is a local, global
4286 // or static variable, so we have to do a little digging by
Bruce Mitchenerd93c4a32014-07-01 21:22:11 +00004287 // looking at the location of a variable to see if it contains
Greg Clayton9e9f2192013-05-17 00:55:28 +00004288 // a DW_OP_addr opcode _somewhere_ in the definition. I say
4289 // somewhere because clang likes to combine small global variables
4290 // into the same symbol and have locations like:
4291 // DW_OP_addr(0x1000), DW_OP_constu(2), DW_OP_plus
4292 // So if we don't have a DW_TAG_formal_parameter, we can look at
4293 // the location to see if it contains a DW_OP_addr opcode, and
4294 // then we can correctly classify our variables.
4295 if (tag == DW_TAG_formal_parameter)
4296 scope = eValueTypeVariableArgument;
4297 else
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004298 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004299 bool op_error = false;
4300 // Check if the location has a DW_OP_addr with any address value...
4301 lldb::addr_t location_DW_OP_addr = LLDB_INVALID_ADDRESS;
4302 if (!location_is_const_value_data)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004303 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004304 location_DW_OP_addr = location.GetLocation_DW_OP_addr (0, op_error);
4305 if (op_error)
Greg Clayton96c09682012-01-04 22:56:43 +00004306 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004307 StreamString strm;
4308 location.DumpLocationForAddress (&strm, eDescriptionLevelFull, 0, 0, NULL);
Greg Clayton6071e6f2015-08-26 22:57:51 +00004309 GetObjectFile()->GetModule()->ReportError ("0x%8.8x: %s has an invalid location: %s", die.GetOffset(), die.GetTagAsCString(), strm.GetString().c_str());
Greg Clayton96c09682012-01-04 22:56:43 +00004310 }
Greg Clayton9e9f2192013-05-17 00:55:28 +00004311 }
Greg Claytond1767f02011-12-08 02:13:16 +00004312
Greg Clayton9e9f2192013-05-17 00:55:28 +00004313 if (location_DW_OP_addr != LLDB_INVALID_ADDRESS)
4314 {
4315 if (is_external)
4316 scope = eValueTypeVariableGlobal;
4317 else
4318 scope = eValueTypeVariableStatic;
4319
4320
4321 SymbolFileDWARFDebugMap *debug_map_symfile = GetDebugMapSymfile ();
4322
4323 if (debug_map_symfile)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004324 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004325 // When leaving the DWARF in the .o files on darwin,
4326 // when we have a global variable that wasn't initialized,
4327 // the .o file might not have allocated a virtual
4328 // address for the global variable. In this case it will
4329 // have created a symbol for the global variable
4330 // that is undefined/data and external and the value will
4331 // be the byte size of the variable. When we do the
4332 // address map in SymbolFileDWARFDebugMap we rely on
4333 // having an address, we need to do some magic here
4334 // so we can get the correct address for our global
4335 // variable. The address for all of these entries
4336 // will be zero, and there will be an undefined symbol
4337 // in this object file, and the executable will have
4338 // a matching symbol with a good address. So here we
4339 // dig up the correct address and replace it in the
4340 // location for the variable, and set the variable's
4341 // symbol context scope to be that of the main executable
4342 // so the file address will resolve correctly.
4343 bool linked_oso_file_addr = false;
4344 if (is_external && location_DW_OP_addr == 0)
Greg Clayton9422dd62013-03-04 21:46:16 +00004345 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004346 // we have a possible uninitialized extern global
4347 ConstString const_name(mangled ? mangled : name);
4348 ObjectFile *debug_map_objfile = debug_map_symfile->GetObjectFile();
4349 if (debug_map_objfile)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004350 {
Greg Clayton3046e662013-07-10 01:23:25 +00004351 Symtab *debug_map_symtab = debug_map_objfile->GetSymtab();
Greg Clayton9e9f2192013-05-17 00:55:28 +00004352 if (debug_map_symtab)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004353 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004354 Symbol *exe_symbol = debug_map_symtab->FindFirstSymbolWithNameAndType (const_name,
4355 eSymbolTypeData,
4356 Symtab::eDebugYes,
4357 Symtab::eVisibilityExtern);
4358 if (exe_symbol)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004359 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004360 if (exe_symbol->ValueIsAddress())
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004361 {
Greg Clayton358cf1e2015-06-25 21:46:34 +00004362 const addr_t exe_file_addr = exe_symbol->GetAddressRef().GetFileAddress();
Greg Clayton9e9f2192013-05-17 00:55:28 +00004363 if (exe_file_addr != LLDB_INVALID_ADDRESS)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004364 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004365 if (location.Update_DW_OP_addr (exe_file_addr))
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004366 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004367 linked_oso_file_addr = true;
4368 symbol_context_scope = exe_symbol;
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004369 }
4370 }
4371 }
4372 }
4373 }
4374 }
Greg Clayton9e9f2192013-05-17 00:55:28 +00004375 }
Greg Clayton9422dd62013-03-04 21:46:16 +00004376
Greg Clayton9e9f2192013-05-17 00:55:28 +00004377 if (!linked_oso_file_addr)
4378 {
4379 // The DW_OP_addr is not zero, but it contains a .o file address which
4380 // needs to be linked up correctly.
4381 const lldb::addr_t exe_file_addr = debug_map_symfile->LinkOSOFileAddress(this, location_DW_OP_addr);
4382 if (exe_file_addr != LLDB_INVALID_ADDRESS)
Greg Clayton9422dd62013-03-04 21:46:16 +00004383 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004384 // Update the file address for this variable
4385 location.Update_DW_OP_addr (exe_file_addr);
4386 }
4387 else
4388 {
4389 // Variable didn't make it into the final executable
4390 return var_sp;
Greg Clayton9422dd62013-03-04 21:46:16 +00004391 }
Greg Claytond1767f02011-12-08 02:13:16 +00004392 }
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004393 }
Greg Clayton5cf58b92011-10-05 22:22:08 +00004394 }
4395 else
4396 {
Ewan Crawford37395ad2015-12-17 11:59:47 +00004397 if (location_is_const_value_data)
4398 scope = eValueTypeVariableStatic;
4399 else
4400 scope = eValueTypeVariableLocal;
Greg Clayton5cf58b92011-10-05 22:22:08 +00004401 }
Greg Clayton7f995132011-10-04 22:41:51 +00004402 }
Greg Clayton9e9f2192013-05-17 00:55:28 +00004403
4404 if (symbol_context_scope == NULL)
4405 {
4406 switch (parent_tag)
4407 {
4408 case DW_TAG_subprogram:
4409 case DW_TAG_inlined_subroutine:
4410 case DW_TAG_lexical_block:
4411 if (sc.function)
4412 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004413 symbol_context_scope = sc.function->GetBlock(true).FindBlockByID(sc_parent_die.GetID());
Greg Clayton9e9f2192013-05-17 00:55:28 +00004414 if (symbol_context_scope == NULL)
4415 symbol_context_scope = sc.function;
4416 }
4417 break;
4418
4419 default:
4420 symbol_context_scope = sc.comp_unit;
4421 break;
4422 }
4423 }
4424
4425 if (symbol_context_scope)
4426 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004427 SymbolFileTypeSP type_sp(new SymbolFileType(*this, DIERef(type_die_form).GetUID()));
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004428
Enrico Granata4ec130d2014-08-11 19:16:35 +00004429 if (const_value.Form() && type_sp && type_sp->GetType())
Greg Clayton6071e6f2015-08-26 22:57:51 +00004430 location.CopyOpcodeData(const_value.Unsigned(), type_sp->GetType()->GetByteSize(), die.GetCU()->GetAddressByteSize());
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004431
Greg Clayton6071e6f2015-08-26 22:57:51 +00004432 var_sp.reset (new Variable (die.GetID(),
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004433 name,
Greg Clayton9e9f2192013-05-17 00:55:28 +00004434 mangled,
Enrico Granata4ec130d2014-08-11 19:16:35 +00004435 type_sp,
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004436 scope,
4437 symbol_context_scope,
4438 scope_ranges,
4439 &decl,
4440 location,
4441 is_external,
Paul Herman10bc1a42015-08-18 22:46:57 +00004442 is_artificial,
4443 is_static_member));
Greg Clayton9e9f2192013-05-17 00:55:28 +00004444
4445 var_sp->SetLocationIsConstantValueData (location_is_const_value_data);
4446 }
4447 else
4448 {
4449 // Not ready to parse this variable yet. It might be a global
4450 // or static variable that is in a function scope and the function
4451 // in the symbol context wasn't filled in yet
4452 return var_sp;
4453 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004454 }
Greg Clayton7f995132011-10-04 22:41:51 +00004455 // Cache var_sp even if NULL (the variable was just a specification or
4456 // was missing vital information to be able to be displayed in the debugger
4457 // (missing location due to optimization, etc)) so we don't re-parse
4458 // this DIE over and over later...
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004459 GetDIEToVariable()[die.GetDIE()] = var_sp;
Paul Hermand628cbb2015-09-15 23:44:17 +00004460 if (spec_die)
4461 GetDIEToVariable()[spec_die.GetDIE()] = var_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004462 }
4463 return var_sp;
4464}
4465
Greg Claytonc662ec82011-06-17 22:10:16 +00004466
Greg Clayton6071e6f2015-08-26 22:57:51 +00004467DWARFDIE
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004468SymbolFileDWARF::FindBlockContainingSpecification (const DIERef& func_die_ref,
Greg Clayton6071e6f2015-08-26 22:57:51 +00004469 dw_offset_t spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004470{
4471 // Give the concrete function die specified by "func_die_offset", find the
4472 // concrete block whose DW_AT_specification or DW_AT_abstract_origin points
4473 // to "spec_block_die_offset"
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004474 return FindBlockContainingSpecification (DebugInfo()->GetDIE (func_die_ref), spec_block_die_offset);
Greg Claytonc662ec82011-06-17 22:10:16 +00004475}
4476
4477
Greg Clayton6071e6f2015-08-26 22:57:51 +00004478DWARFDIE
4479SymbolFileDWARF::FindBlockContainingSpecification(const DWARFDIE &die,
4480 dw_offset_t spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004481{
4482 if (die)
4483 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004484 switch (die.Tag())
Greg Claytonc662ec82011-06-17 22:10:16 +00004485 {
4486 case DW_TAG_subprogram:
4487 case DW_TAG_inlined_subroutine:
4488 case DW_TAG_lexical_block:
4489 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004490 if (die.GetAttributeValueAsReference (DW_AT_specification, DW_INVALID_OFFSET) == spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004491 return die;
Greg Claytonc662ec82011-06-17 22:10:16 +00004492
Greg Clayton6071e6f2015-08-26 22:57:51 +00004493 if (die.GetAttributeValueAsReference (DW_AT_abstract_origin, DW_INVALID_OFFSET) == spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004494 return die;
Greg Claytonc662ec82011-06-17 22:10:16 +00004495 }
4496 break;
4497 }
4498
4499 // Give the concrete function die specified by "func_die_offset", find the
4500 // concrete block whose DW_AT_specification or DW_AT_abstract_origin points
4501 // to "spec_block_die_offset"
Greg Clayton6071e6f2015-08-26 22:57:51 +00004502 for (DWARFDIE child_die = die.GetFirstChild(); child_die; child_die = child_die.GetSibling())
Greg Claytonc662ec82011-06-17 22:10:16 +00004503 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004504 DWARFDIE result_die = FindBlockContainingSpecification (child_die, spec_block_die_offset);
Greg Claytonc662ec82011-06-17 22:10:16 +00004505 if (result_die)
4506 return result_die;
4507 }
4508 }
4509
Greg Clayton6071e6f2015-08-26 22:57:51 +00004510 return DWARFDIE();
Greg Claytonc662ec82011-06-17 22:10:16 +00004511}
4512
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004513size_t
Greg Clayton6071e6f2015-08-26 22:57:51 +00004514SymbolFileDWARF::ParseVariables (const SymbolContext& sc,
4515 const DWARFDIE &orig_die,
4516 const lldb::addr_t func_low_pc,
4517 bool parse_siblings,
4518 bool parse_children,
4519 VariableList* cc_variable_list)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004520{
Greg Clayton6071e6f2015-08-26 22:57:51 +00004521 if (!orig_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004522 return 0;
4523
Greg Claytonc662ec82011-06-17 22:10:16 +00004524 VariableListSP variable_list_sp;
4525
4526 size_t vars_added = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00004527 DWARFDIE die = orig_die;
4528 while (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004529 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004530 dw_tag_t tag = die.Tag();
Greg Claytonc662ec82011-06-17 22:10:16 +00004531
4532 // Check to see if we have already parsed this variable or constant?
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004533 VariableSP var_sp = GetDIEToVariable()[die.GetDIE()];
Greg Clayton6071e6f2015-08-26 22:57:51 +00004534 if (var_sp)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004535 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004536 if (cc_variable_list)
Greg Clayton6071e6f2015-08-26 22:57:51 +00004537 cc_variable_list->AddVariableIfUnique (var_sp);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004538 }
4539 else
4540 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004541 // We haven't already parsed it, lets do that now.
4542 if ((tag == DW_TAG_variable) ||
4543 (tag == DW_TAG_constant) ||
4544 (tag == DW_TAG_formal_parameter && sc.function))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004545 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004546 if (variable_list_sp.get() == NULL)
Greg Clayton73bf5db2011-06-17 01:22:15 +00004547 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004548 DWARFDIE sc_parent_die = GetParentSymbolContextDIE(orig_die);
4549 dw_tag_t parent_tag = sc_parent_die.Tag();
Greg Claytonc662ec82011-06-17 22:10:16 +00004550 switch (parent_tag)
4551 {
4552 case DW_TAG_compile_unit:
4553 if (sc.comp_unit != NULL)
4554 {
4555 variable_list_sp = sc.comp_unit->GetVariableList(false);
4556 if (variable_list_sp.get() == NULL)
4557 {
4558 variable_list_sp.reset(new VariableList());
4559 sc.comp_unit->SetVariableList(variable_list_sp);
4560 }
4561 }
4562 else
4563 {
Daniel Malead01b2952012-11-29 21:49:15 +00004564 GetObjectFile()->GetModule()->ReportError ("parent 0x%8.8" PRIx64 " %s with no valid compile unit in symbol context for 0x%8.8" PRIx64 " %s.\n",
Greg Clayton6071e6f2015-08-26 22:57:51 +00004565 sc_parent_die.GetID(),
4566 sc_parent_die.GetTagAsCString(),
4567 orig_die.GetID(),
4568 orig_die.GetTagAsCString());
Greg Claytonc662ec82011-06-17 22:10:16 +00004569 }
4570 break;
4571
4572 case DW_TAG_subprogram:
4573 case DW_TAG_inlined_subroutine:
4574 case DW_TAG_lexical_block:
4575 if (sc.function != NULL)
4576 {
4577 // Check to see if we already have parsed the variables for the given scope
4578
Greg Clayton6071e6f2015-08-26 22:57:51 +00004579 Block *block = sc.function->GetBlock(true).FindBlockByID(sc_parent_die.GetID());
Greg Claytonc662ec82011-06-17 22:10:16 +00004580 if (block == NULL)
4581 {
4582 // This must be a specification or abstract origin with
Bruce Mitchenerd93c4a32014-07-01 21:22:11 +00004583 // a concrete block counterpart in the current function. We need
Greg Claytonc662ec82011-06-17 22:10:16 +00004584 // to find the concrete block so we can correctly add the
4585 // variable to it
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004586 const DWARFDIE concrete_block_die = FindBlockContainingSpecification (DIERef(sc.function->GetID()),
Greg Clayton6071e6f2015-08-26 22:57:51 +00004587 sc_parent_die.GetOffset());
Greg Claytonc662ec82011-06-17 22:10:16 +00004588 if (concrete_block_die)
Greg Clayton6071e6f2015-08-26 22:57:51 +00004589 block = sc.function->GetBlock(true).FindBlockByID(concrete_block_die.GetID());
Greg Claytonc662ec82011-06-17 22:10:16 +00004590 }
4591
4592 if (block != NULL)
4593 {
4594 const bool can_create = false;
4595 variable_list_sp = block->GetBlockVariableList (can_create);
4596 if (variable_list_sp.get() == NULL)
4597 {
4598 variable_list_sp.reset(new VariableList());
4599 block->SetVariableList(variable_list_sp);
4600 }
4601 }
4602 }
4603 break;
4604
4605 default:
Daniel Malead01b2952012-11-29 21:49:15 +00004606 GetObjectFile()->GetModule()->ReportError ("didn't find appropriate parent DIE for variable list for 0x%8.8" PRIx64 " %s.\n",
Greg Clayton6071e6f2015-08-26 22:57:51 +00004607 orig_die.GetID(),
4608 orig_die.GetTagAsCString());
Greg Claytonc662ec82011-06-17 22:10:16 +00004609 break;
4610 }
Greg Clayton73bf5db2011-06-17 01:22:15 +00004611 }
Greg Claytonc662ec82011-06-17 22:10:16 +00004612
4613 if (variable_list_sp)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004614 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004615 VariableSP var_sp (ParseVariableDIE(sc, die, func_low_pc));
Greg Clayton73bf5db2011-06-17 01:22:15 +00004616 if (var_sp)
4617 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004618 variable_list_sp->AddVariableIfUnique (var_sp);
Greg Clayton73bf5db2011-06-17 01:22:15 +00004619 if (cc_variable_list)
4620 cc_variable_list->AddVariableIfUnique (var_sp);
4621 ++vars_added;
4622 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004623 }
4624 }
4625 }
Greg Claytonc662ec82011-06-17 22:10:16 +00004626
4627 bool skip_children = (sc.function == NULL && tag == DW_TAG_subprogram);
4628
Greg Clayton6071e6f2015-08-26 22:57:51 +00004629 if (!skip_children && parse_children && die.HasChildren())
Greg Claytonc662ec82011-06-17 22:10:16 +00004630 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004631 vars_added += ParseVariables(sc, die.GetFirstChild(), func_low_pc, true, true, cc_variable_list);
Greg Claytonc662ec82011-06-17 22:10:16 +00004632 }
4633
4634 if (parse_siblings)
Greg Clayton6071e6f2015-08-26 22:57:51 +00004635 die = die.GetSibling();
Greg Claytonc662ec82011-06-17 22:10:16 +00004636 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00004637 die.Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004638 }
Greg Claytonc662ec82011-06-17 22:10:16 +00004639 return vars_added;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004640}
4641
4642//------------------------------------------------------------------
4643// PluginInterface protocol
4644//------------------------------------------------------------------
Greg Clayton57abc5d2013-05-10 21:47:16 +00004645ConstString
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004646SymbolFileDWARF::GetPluginName()
4647{
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004648 return GetPluginNameStatic();
4649}
4650
4651uint32_t
4652SymbolFileDWARF::GetPluginVersion()
4653{
4654 return 1;
4655}
4656
4657void
Sean Callanancc427fa2011-07-30 02:42:06 +00004658SymbolFileDWARF::DumpIndexes ()
4659{
4660 StreamFile s(stdout, false);
4661
Greg Claytonb5ad4ec2013-04-29 17:25:54 +00004662 s.Printf ("DWARF index for (%s) '%s':",
Sean Callanancc427fa2011-07-30 02:42:06 +00004663 GetObjectFile()->GetModule()->GetArchitecture().GetArchitectureName(),
Greg Claytonb5ad4ec2013-04-29 17:25:54 +00004664 GetObjectFile()->GetFileSpec().GetPath().c_str());
Sean Callanancc427fa2011-07-30 02:42:06 +00004665 s.Printf("\nFunction basenames:\n"); m_function_basename_index.Dump (&s);
4666 s.Printf("\nFunction fullnames:\n"); m_function_fullname_index.Dump (&s);
4667 s.Printf("\nFunction methods:\n"); m_function_method_index.Dump (&s);
4668 s.Printf("\nFunction selectors:\n"); m_function_selector_index.Dump (&s);
4669 s.Printf("\nObjective C class selectors:\n"); m_objc_class_selectors_index.Dump (&s);
4670 s.Printf("\nGlobals and statics:\n"); m_global_index.Dump (&s);
4671 s.Printf("\nTypes:\n"); m_type_index.Dump (&s);
Bruce Mitchenere171da52015-07-22 00:16:02 +00004672 s.Printf("\nNamespaces:\n"); m_namespace_index.Dump (&s);
Sean Callanancc427fa2011-07-30 02:42:06 +00004673}
4674
Greg Claytoncaab74e2012-01-28 00:48:57 +00004675
Greg Clayton1f746072012-08-29 21:13:06 +00004676SymbolFileDWARFDebugMap *
4677SymbolFileDWARF::GetDebugMapSymfile ()
4678{
4679 if (m_debug_map_symfile == NULL && !m_debug_map_module_wp.expired())
4680 {
4681 lldb::ModuleSP module_sp (m_debug_map_module_wp.lock());
4682 if (module_sp)
4683 {
4684 SymbolVendor *sym_vendor = module_sp->GetSymbolVendor();
4685 if (sym_vendor)
4686 m_debug_map_symfile = (SymbolFileDWARFDebugMap *)sym_vendor->GetSymbolFile();
4687 }
4688 }
4689 return m_debug_map_symfile;
4690}
Tamas Berghammer1f5e4482015-09-16 12:37:06 +00004691
4692DWARFExpression::LocationListFormat
4693SymbolFileDWARF::GetLocationListFormat() const
4694{
4695 return DWARFExpression::RegularLocationList;
4696}