blob: e3e5b7cb0ed6f2124dad9a46027ccf53376b160c [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 Clayton2f869fe2016-03-30 20:14:35 +0000282 SymbolFileDWARFDebugMap *debug_map_symfile = GetDebugMapSymfile();
283 if (debug_map_symfile)
284 return debug_map_symfile->GetTypeList();
285 else
286 return m_obj_file->GetModule()->GetTypeList();
Greg Clayton2d95dc9b2010-11-10 04:57:04 +0000287
288}
Greg Claytonf02500c2013-06-18 22:51:05 +0000289void
Greg Clayton6071e6f2015-08-26 22:57:51 +0000290SymbolFileDWARF::GetTypes (const DWARFDIE &die,
Greg Claytonf02500c2013-06-18 22:51:05 +0000291 dw_offset_t min_die_offset,
292 dw_offset_t max_die_offset,
293 uint32_t type_mask,
294 TypeSet &type_set)
295{
Greg Clayton6071e6f2015-08-26 22:57:51 +0000296 if (die)
Greg Claytonf02500c2013-06-18 22:51:05 +0000297 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000298 const dw_offset_t die_offset = die.GetOffset();
299
300 if (die_offset >= max_die_offset)
301 return;
302
303 if (die_offset >= min_die_offset)
Greg Claytonf02500c2013-06-18 22:51:05 +0000304 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000305 const dw_tag_t tag = die.Tag();
Greg Claytonf02500c2013-06-18 22:51:05 +0000306
Greg Clayton6071e6f2015-08-26 22:57:51 +0000307 bool add_type = false;
308
309 switch (tag)
Greg Claytonf02500c2013-06-18 22:51:05 +0000310 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000311 case DW_TAG_array_type: add_type = (type_mask & eTypeClassArray ) != 0; break;
312 case DW_TAG_unspecified_type:
313 case DW_TAG_base_type: add_type = (type_mask & eTypeClassBuiltin ) != 0; break;
314 case DW_TAG_class_type: add_type = (type_mask & eTypeClassClass ) != 0; break;
315 case DW_TAG_structure_type: add_type = (type_mask & eTypeClassStruct ) != 0; break;
316 case DW_TAG_union_type: add_type = (type_mask & eTypeClassUnion ) != 0; break;
317 case DW_TAG_enumeration_type: add_type = (type_mask & eTypeClassEnumeration ) != 0; break;
318 case DW_TAG_subroutine_type:
319 case DW_TAG_subprogram:
320 case DW_TAG_inlined_subroutine: add_type = (type_mask & eTypeClassFunction ) != 0; break;
321 case DW_TAG_pointer_type: add_type = (type_mask & eTypeClassPointer ) != 0; break;
322 case DW_TAG_rvalue_reference_type:
323 case DW_TAG_reference_type: add_type = (type_mask & eTypeClassReference ) != 0; break;
324 case DW_TAG_typedef: add_type = (type_mask & eTypeClassTypedef ) != 0; break;
325 case DW_TAG_ptr_to_member_type: add_type = (type_mask & eTypeClassMemberPointer ) != 0; break;
326 }
Greg Claytonf02500c2013-06-18 22:51:05 +0000327
Greg Clayton6071e6f2015-08-26 22:57:51 +0000328 if (add_type)
329 {
330 const bool assert_not_being_parsed = true;
331 Type *type = ResolveTypeUID (die, assert_not_being_parsed);
332 if (type)
Greg Claytonf02500c2013-06-18 22:51:05 +0000333 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000334 if (type_set.find(type) == type_set.end())
335 type_set.insert(type);
Greg Claytonf02500c2013-06-18 22:51:05 +0000336 }
337 }
Greg Clayton6071e6f2015-08-26 22:57:51 +0000338 }
339
340 for (DWARFDIE child_die = die.GetFirstChild();
341 child_die.IsValid();
342 child_die = child_die.GetSibling())
343 {
344 GetTypes (child_die, min_die_offset, max_die_offset, type_mask, type_set);
Greg Claytonf02500c2013-06-18 22:51:05 +0000345 }
346 }
347}
348
349size_t
350SymbolFileDWARF::GetTypes (SymbolContextScope *sc_scope,
351 uint32_t type_mask,
352 TypeList &type_list)
353
354{
355 TypeSet type_set;
356
357 CompileUnit *comp_unit = NULL;
358 DWARFCompileUnit* dwarf_cu = NULL;
359 if (sc_scope)
360 comp_unit = sc_scope->CalculateSymbolContextCompileUnit();
361
362 if (comp_unit)
363 {
364 dwarf_cu = GetDWARFCompileUnit(comp_unit);
365 if (dwarf_cu == 0)
366 return 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +0000367 GetTypes (dwarf_cu->DIE(),
Greg Claytonf02500c2013-06-18 22:51:05 +0000368 dwarf_cu->GetOffset(),
369 dwarf_cu->GetNextCompileUnitOffset(),
370 type_mask,
371 type_set);
372 }
373 else
374 {
375 DWARFDebugInfo* info = DebugInfo();
376 if (info)
377 {
378 const size_t num_cus = info->GetNumCompileUnits();
379 for (size_t cu_idx=0; cu_idx<num_cus; ++cu_idx)
380 {
381 dwarf_cu = info->GetCompileUnitAtIndex(cu_idx);
382 if (dwarf_cu)
383 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000384 GetTypes (dwarf_cu->DIE(),
Greg Claytonf02500c2013-06-18 22:51:05 +0000385 0,
386 UINT32_MAX,
387 type_mask,
388 type_set);
389 }
390 }
391 }
392 }
Greg Clayton6071e6f2015-08-26 22:57:51 +0000393
Bruce Mitchener3ad353f2015-09-24 03:54:50 +0000394 std::set<CompilerType> compiler_type_set;
Greg Claytonf02500c2013-06-18 22:51:05 +0000395 size_t num_types_added = 0;
396 for (Type *type : type_set)
397 {
Bruce Mitchener3ad353f2015-09-24 03:54:50 +0000398 CompilerType compiler_type = type->GetForwardCompilerType ();
399 if (compiler_type_set.find(compiler_type) == compiler_type_set.end())
Greg Clayton0fc4f312013-06-20 01:23:18 +0000400 {
Bruce Mitchener3ad353f2015-09-24 03:54:50 +0000401 compiler_type_set.insert(compiler_type);
Greg Clayton0fc4f312013-06-20 01:23:18 +0000402 type_list.Insert (type->shared_from_this());
403 ++num_types_added;
404 }
Greg Claytonf02500c2013-06-18 22:51:05 +0000405 }
406 return num_types_added;
407}
408
Greg Clayton2d95dc9b2010-11-10 04:57:04 +0000409
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000410//----------------------------------------------------------------------
411// Gets the first parent that is a lexical block, function or inlined
412// subroutine, or compile unit.
413//----------------------------------------------------------------------
Greg Clayton6071e6f2015-08-26 22:57:51 +0000414DWARFDIE
415SymbolFileDWARF::GetParentSymbolContextDIE(const DWARFDIE &child_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000416{
Greg Clayton6071e6f2015-08-26 22:57:51 +0000417 DWARFDIE die;
418 for (die = child_die.GetParent(); die; die = die.GetParent())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000419 {
Greg Clayton6071e6f2015-08-26 22:57:51 +0000420 dw_tag_t tag = die.Tag();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000421
422 switch (tag)
423 {
424 case DW_TAG_compile_unit:
425 case DW_TAG_subprogram:
426 case DW_TAG_inlined_subroutine:
427 case DW_TAG_lexical_block:
428 return die;
429 }
430 }
Greg Clayton6071e6f2015-08-26 22:57:51 +0000431 return DWARFDIE();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000432}
433
434
Greg Clayton450e3f32010-10-12 02:24:53 +0000435SymbolFileDWARF::SymbolFileDWARF(ObjectFile* objfile) :
436 SymbolFile (objfile),
Greg Clayton81c22f62011-10-19 18:09:39 +0000437 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 +0000438 m_debug_map_module_wp (),
Greg Clayton450e3f32010-10-12 02:24:53 +0000439 m_debug_map_symfile (NULL),
Greg Claytond4a2b372011-09-12 23:21:58 +0000440 m_data_debug_abbrev (),
441 m_data_debug_aranges (),
442 m_data_debug_frame (),
443 m_data_debug_info (),
444 m_data_debug_line (),
Siva Chandrad8335e92015-12-16 00:22:08 +0000445 m_data_debug_macro (),
Greg Claytond4a2b372011-09-12 23:21:58 +0000446 m_data_debug_loc (),
447 m_data_debug_ranges (),
448 m_data_debug_str (),
Greg Clayton17674402011-09-28 17:06:40 +0000449 m_data_apple_names (),
450 m_data_apple_types (),
Greg Clayton7f995132011-10-04 22:41:51 +0000451 m_data_apple_namespaces (),
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000452 m_abbr(),
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000453 m_info(),
454 m_line(),
Greg Clayton7f995132011-10-04 22:41:51 +0000455 m_apple_names_ap (),
456 m_apple_types_ap (),
457 m_apple_namespaces_ap (),
Greg Clayton5009f9d2011-10-27 17:55:14 +0000458 m_apple_objc_ap (),
Greg Claytonc685f8e2010-09-15 04:15:46 +0000459 m_function_basename_index(),
460 m_function_fullname_index(),
461 m_function_method_index(),
462 m_function_selector_index(),
Greg Clayton450e3f32010-10-12 02:24:53 +0000463 m_objc_class_selectors_index(),
Greg Claytonc685f8e2010-09-15 04:15:46 +0000464 m_global_index(),
Greg Clayton69b04882010-10-15 02:03:22 +0000465 m_type_index(),
466 m_namespace_index(),
Greg Clayton6beaaa62011-01-17 03:46:26 +0000467 m_indexed (false),
Greg Clayton97fbc342011-10-20 22:30:33 +0000468 m_using_apple_tables (false),
Sean Callananf0c5aeb2015-04-20 16:31:29 +0000469 m_fetched_external_modules (false),
Greg Claytonc7f03b62012-01-12 04:33:28 +0000470 m_supports_DW_AT_APPLE_objc_complete_type (eLazyBoolCalculate),
Greg Clayton1c9e5ac2011-02-09 19:06:17 +0000471 m_ranges(),
472 m_unique_ast_type_map ()
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000473{
474}
475
476SymbolFileDWARF::~SymbolFileDWARF()
477{
Greg Clayton6beaaa62011-01-17 03:46:26 +0000478}
479
480static const ConstString &
481GetDWARFMachOSegmentName ()
482{
483 static ConstString g_dwarf_section_name ("__DWARF");
484 return g_dwarf_section_name;
485}
486
Greg Claytone576ab22011-02-15 00:19:15 +0000487UniqueDWARFASTTypeMap &
488SymbolFileDWARF::GetUniqueDWARFASTTypeMap ()
489{
Greg Clayton2f869fe2016-03-30 20:14:35 +0000490 SymbolFileDWARFDebugMap *debug_map_symfile = GetDebugMapSymfile();
491 if (debug_map_symfile)
492 return debug_map_symfile->GetUniqueDWARFASTTypeMap ();
493 else
494 return m_unique_ast_type_map;
Greg Claytone576ab22011-02-15 00:19:15 +0000495}
496
Greg Clayton8b4edba2015-08-14 20:02:05 +0000497TypeSystem *
498SymbolFileDWARF::GetTypeSystemForLanguage (LanguageType language)
499{
Greg Clayton2f869fe2016-03-30 20:14:35 +0000500 SymbolFileDWARFDebugMap *debug_map_symfile = GetDebugMapSymfile();
Ryan Brown57bee1e2015-09-14 22:45:11 +0000501 TypeSystem *type_system;
Greg Clayton8b4edba2015-08-14 20:02:05 +0000502 if (debug_map_symfile)
Ryan Brown57bee1e2015-09-14 22:45:11 +0000503 {
504 type_system = debug_map_symfile->GetTypeSystemForLanguage(language);
505 }
Greg Clayton8b4edba2015-08-14 20:02:05 +0000506 else
Ryan Brown57bee1e2015-09-14 22:45:11 +0000507 {
508 type_system = m_obj_file->GetModule()->GetTypeSystemForLanguage(language);
509 if (type_system)
510 type_system->SetSymbolFile(this);
511 }
512 return type_system;
Greg Clayton8b4edba2015-08-14 20:02:05 +0000513}
514
Greg Clayton6beaaa62011-01-17 03:46:26 +0000515void
516SymbolFileDWARF::InitializeObject()
517{
Greg Claytone72dfb32012-02-24 01:59:29 +0000518 ModuleSP module_sp (m_obj_file->GetModule());
519 if (module_sp)
Greg Clayton6beaaa62011-01-17 03:46:26 +0000520 {
Greg Clayton3046e662013-07-10 01:23:25 +0000521 const SectionList *section_list = module_sp->GetSectionList();
Greg Clayton6beaaa62011-01-17 03:46:26 +0000522 const Section* section = section_list->FindSectionByName(GetDWARFMachOSegmentName ()).get();
523
524 // Memory map the DWARF mach-o segment so we have everything mmap'ed
525 // to keep our heap memory usage down.
526 if (section)
Greg Claytonc9660542012-02-05 02:38:54 +0000527 m_obj_file->MemoryMapSectionData(section, m_dwarf_data);
Greg Clayton6beaaa62011-01-17 03:46:26 +0000528 }
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000529
Greg Clayton4d01ace2011-09-29 16:58:15 +0000530 get_apple_names_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000531 if (m_data_apple_names.m_data.GetByteSize() > 0)
Greg Clayton7f995132011-10-04 22:41:51 +0000532 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000533 m_apple_names_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_names.m_data,
534 get_debug_str_data(),
535 ".apple_names"));
Greg Clayton97fbc342011-10-20 22:30:33 +0000536 if (m_apple_names_ap->IsValid())
537 m_using_apple_tables = true;
538 else
Greg Clayton7f995132011-10-04 22:41:51 +0000539 m_apple_names_ap.reset();
540 }
Greg Clayton4d01ace2011-09-29 16:58:15 +0000541 get_apple_types_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000542 if (m_data_apple_types.m_data.GetByteSize() > 0)
Greg Clayton7f995132011-10-04 22:41:51 +0000543 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000544 m_apple_types_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_types.m_data,
545 get_debug_str_data(),
546 ".apple_types"));
Greg Clayton97fbc342011-10-20 22:30:33 +0000547 if (m_apple_types_ap->IsValid())
548 m_using_apple_tables = true;
549 else
Greg Clayton7f995132011-10-04 22:41:51 +0000550 m_apple_types_ap.reset();
551 }
552
553 get_apple_namespaces_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000554 if (m_data_apple_namespaces.m_data.GetByteSize() > 0)
Greg Clayton7f995132011-10-04 22:41:51 +0000555 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000556 m_apple_namespaces_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_namespaces.m_data,
557 get_debug_str_data(),
558 ".apple_namespaces"));
Greg Clayton97fbc342011-10-20 22:30:33 +0000559 if (m_apple_namespaces_ap->IsValid())
560 m_using_apple_tables = true;
561 else
Greg Clayton7f995132011-10-04 22:41:51 +0000562 m_apple_namespaces_ap.reset();
563 }
Greg Clayton4d01ace2011-09-29 16:58:15 +0000564
Greg Clayton5009f9d2011-10-27 17:55:14 +0000565 get_apple_objc_data();
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000566 if (m_data_apple_objc.m_data.GetByteSize() > 0)
Greg Clayton5009f9d2011-10-27 17:55:14 +0000567 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000568 m_apple_objc_ap.reset (new DWARFMappedHash::MemoryTable (m_data_apple_objc.m_data,
569 get_debug_str_data(),
570 ".apple_objc"));
Greg Clayton5009f9d2011-10-27 17:55:14 +0000571 if (m_apple_objc_ap->IsValid())
572 m_using_apple_tables = true;
573 else
574 m_apple_objc_ap.reset();
575 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000576}
577
578bool
579SymbolFileDWARF::SupportedVersion(uint16_t version)
580{
Greg Claytonabcbfe52013-04-04 00:00:36 +0000581 return version == 2 || version == 3 || version == 4;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000582}
583
584uint32_t
Sean Callananbfaf54d2011-12-03 04:38:43 +0000585SymbolFileDWARF::CalculateAbilities ()
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000586{
587 uint32_t abilities = 0;
588 if (m_obj_file != NULL)
589 {
590 const Section* section = NULL;
591 const SectionList *section_list = m_obj_file->GetSectionList();
592 if (section_list == NULL)
593 return 0;
594
595 uint64_t debug_abbrev_file_size = 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000596 uint64_t debug_info_file_size = 0;
597 uint64_t debug_line_file_size = 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000598
Greg Clayton6beaaa62011-01-17 03:46:26 +0000599 section = section_list->FindSectionByName(GetDWARFMachOSegmentName ()).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000600
601 if (section)
Greg Clayton4ceb9982010-07-21 22:54:26 +0000602 section_list = &section->GetChildren ();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000603
Greg Clayton4ceb9982010-07-21 22:54:26 +0000604 section = section_list->FindSectionByType (eSectionTypeDWARFDebugInfo, true).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000605 if (section != NULL)
606 {
Greg Clayton47037bc2012-03-27 02:40:46 +0000607 debug_info_file_size = section->GetFileSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000608
Greg Clayton4ceb9982010-07-21 22:54:26 +0000609 section = section_list->FindSectionByType (eSectionTypeDWARFDebugAbbrev, true).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000610 if (section)
Greg Clayton47037bc2012-03-27 02:40:46 +0000611 debug_abbrev_file_size = section->GetFileSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000612
Greg Clayton4ceb9982010-07-21 22:54:26 +0000613 section = section_list->FindSectionByType (eSectionTypeDWARFDebugLine, true).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000614 if (section)
Greg Clayton47037bc2012-03-27 02:40:46 +0000615 debug_line_file_size = section->GetFileSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000616 }
Greg Clayton6c596612012-05-18 21:47:20 +0000617 else
618 {
619 const char *symfile_dir_cstr = m_obj_file->GetFileSpec().GetDirectory().GetCString();
620 if (symfile_dir_cstr)
621 {
622 if (strcasestr(symfile_dir_cstr, ".dsym"))
623 {
624 if (m_obj_file->GetType() == ObjectFile::eTypeDebugInfo)
625 {
626 // We have a dSYM file that didn't have a any debug info.
627 // If the string table has a size of 1, then it was made from
628 // an executable with no debug info, or from an executable that
629 // was stripped.
630 section = section_list->FindSectionByType (eSectionTypeDWARFDebugStr, true).get();
631 if (section && section->GetFileSize() == 1)
632 {
633 m_obj_file->GetModule()->ReportWarning ("empty dSYM file detected, dSYM was created with an executable with no debug info.");
634 }
635 }
636 }
637 }
638 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000639
640 if (debug_abbrev_file_size > 0 && debug_info_file_size > 0)
641 abilities |= CompileUnits | Functions | Blocks | GlobalVariables | LocalVariables | VariableTypes;
642
643 if (debug_line_file_size > 0)
644 abilities |= LineTables;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000645 }
646 return abilities;
647}
648
Ed Masteeeae7212013-10-24 20:43:47 +0000649const DWARFDataExtractor&
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000650SymbolFileDWARF::GetCachedSectionData (lldb::SectionType sect_type, DWARFDataSegment& data_segment)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000651{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000652 std::call_once(data_segment.m_flag,
653 &SymbolFileDWARF::LoadSectionData,
654 this,
655 sect_type,
656 std::ref(data_segment.m_data));
657 return data_segment.m_data;
658}
659
660void
661SymbolFileDWARF::LoadSectionData (lldb::SectionType sect_type, DWARFDataExtractor& data)
662{
663 ModuleSP module_sp (m_obj_file->GetModule());
664 const SectionList *section_list = module_sp->GetSectionList();
665 if (section_list)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000666 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000667 SectionSP section_sp (section_list->FindSectionByType(sect_type, true));
668 if (section_sp)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000669 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000670 // See if we memory mapped the DWARF segment?
671 if (m_dwarf_data.GetByteSize())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000672 {
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000673 data.SetData(m_dwarf_data, section_sp->GetOffset(), section_sp->GetFileSize());
674 }
675 else
676 {
677 if (m_obj_file->ReadSectionData(section_sp.get(), data) == 0)
678 data.Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000679 }
680 }
681 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000682}
683
Ed Masteeeae7212013-10-24 20:43:47 +0000684const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000685SymbolFileDWARF::get_debug_abbrev_data()
686{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000687 return GetCachedSectionData (eSectionTypeDWARFDebugAbbrev, m_data_debug_abbrev);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000688}
689
Ed Masteeeae7212013-10-24 20:43:47 +0000690const DWARFDataExtractor&
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000691SymbolFileDWARF::get_debug_addr_data()
692{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000693 return GetCachedSectionData (eSectionTypeDWARFDebugAddr, m_data_debug_addr);
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000694}
695
696const DWARFDataExtractor&
Greg Claytond4a2b372011-09-12 23:21:58 +0000697SymbolFileDWARF::get_debug_aranges_data()
698{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000699 return GetCachedSectionData (eSectionTypeDWARFDebugAranges, m_data_debug_aranges);
Greg Claytond4a2b372011-09-12 23:21:58 +0000700}
701
Ed Masteeeae7212013-10-24 20:43:47 +0000702const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000703SymbolFileDWARF::get_debug_frame_data()
704{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000705 return GetCachedSectionData (eSectionTypeDWARFDebugFrame, m_data_debug_frame);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000706}
707
Ed Masteeeae7212013-10-24 20:43:47 +0000708const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000709SymbolFileDWARF::get_debug_info_data()
710{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000711 return GetCachedSectionData (eSectionTypeDWARFDebugInfo, m_data_debug_info);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000712}
713
Ed Masteeeae7212013-10-24 20:43:47 +0000714const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000715SymbolFileDWARF::get_debug_line_data()
716{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000717 return GetCachedSectionData (eSectionTypeDWARFDebugLine, m_data_debug_line);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000718}
719
Ed Masteeeae7212013-10-24 20:43:47 +0000720const DWARFDataExtractor&
Siva Chandrad8335e92015-12-16 00:22:08 +0000721SymbolFileDWARF::get_debug_macro_data()
722{
723 return GetCachedSectionData (eSectionTypeDWARFDebugMacro, m_data_debug_macro);
724}
725
726const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000727SymbolFileDWARF::get_debug_loc_data()
728{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000729 return GetCachedSectionData (eSectionTypeDWARFDebugLoc, m_data_debug_loc);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000730}
731
Ed Masteeeae7212013-10-24 20:43:47 +0000732const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000733SymbolFileDWARF::get_debug_ranges_data()
734{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000735 return GetCachedSectionData (eSectionTypeDWARFDebugRanges, m_data_debug_ranges);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000736}
737
Ed Masteeeae7212013-10-24 20:43:47 +0000738const DWARFDataExtractor&
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000739SymbolFileDWARF::get_debug_str_data()
740{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000741 return GetCachedSectionData (eSectionTypeDWARFDebugStr, m_data_debug_str);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000742}
743
Ed Masteeeae7212013-10-24 20:43:47 +0000744const DWARFDataExtractor&
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000745SymbolFileDWARF::get_debug_str_offsets_data()
746{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000747 return GetCachedSectionData (eSectionTypeDWARFDebugStrOffsets, m_data_debug_str_offsets);
Tamas Berghammerc178d4c2015-08-25 11:45:58 +0000748}
749
750const DWARFDataExtractor&
Greg Clayton17674402011-09-28 17:06:40 +0000751SymbolFileDWARF::get_apple_names_data()
Greg Claytonf9eec202011-09-01 23:16:13 +0000752{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000753 return GetCachedSectionData (eSectionTypeDWARFAppleNames, m_data_apple_names);
Greg Claytonf9eec202011-09-01 23:16:13 +0000754}
755
Ed Masteeeae7212013-10-24 20:43:47 +0000756const DWARFDataExtractor&
Greg Clayton17674402011-09-28 17:06:40 +0000757SymbolFileDWARF::get_apple_types_data()
Greg Claytonf9eec202011-09-01 23:16:13 +0000758{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000759 return GetCachedSectionData (eSectionTypeDWARFAppleTypes, m_data_apple_types);
Greg Claytonf9eec202011-09-01 23:16:13 +0000760}
761
Ed Masteeeae7212013-10-24 20:43:47 +0000762const DWARFDataExtractor&
Greg Clayton7f995132011-10-04 22:41:51 +0000763SymbolFileDWARF::get_apple_namespaces_data()
764{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000765 return GetCachedSectionData (eSectionTypeDWARFAppleNamespaces, m_data_apple_namespaces);
Greg Clayton5009f9d2011-10-27 17:55:14 +0000766}
767
Ed Masteeeae7212013-10-24 20:43:47 +0000768const DWARFDataExtractor&
Greg Clayton5009f9d2011-10-27 17:55:14 +0000769SymbolFileDWARF::get_apple_objc_data()
770{
Tamas Berghammer90b4dce2015-10-22 11:14:37 +0000771 return GetCachedSectionData (eSectionTypeDWARFAppleObjC, m_data_apple_objc);
Greg Clayton7f995132011-10-04 22:41:51 +0000772}
773
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000774
775DWARFDebugAbbrev*
776SymbolFileDWARF::DebugAbbrev()
777{
778 if (m_abbr.get() == NULL)
779 {
Ed Masteeeae7212013-10-24 20:43:47 +0000780 const DWARFDataExtractor &debug_abbrev_data = get_debug_abbrev_data();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000781 if (debug_abbrev_data.GetByteSize() > 0)
782 {
783 m_abbr.reset(new DWARFDebugAbbrev());
784 if (m_abbr.get())
785 m_abbr->Parse(debug_abbrev_data);
786 }
787 }
788 return m_abbr.get();
789}
790
791const DWARFDebugAbbrev*
792SymbolFileDWARF::DebugAbbrev() const
793{
794 return m_abbr.get();
795}
796
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000797
798DWARFDebugInfo*
799SymbolFileDWARF::DebugInfo()
800{
801 if (m_info.get() == NULL)
802 {
Saleem Abdulrasool324a1032014-04-04 04:06:10 +0000803 Timer scoped_timer(__PRETTY_FUNCTION__, "%s this = %p",
804 __PRETTY_FUNCTION__, static_cast<void*>(this));
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000805 if (get_debug_info_data().GetByteSize() > 0)
806 {
807 m_info.reset(new DWARFDebugInfo());
808 if (m_info.get())
809 {
810 m_info->SetDwarfData(this);
811 }
812 }
813 }
814 return m_info.get();
815}
816
817const DWARFDebugInfo*
818SymbolFileDWARF::DebugInfo() const
819{
820 return m_info.get();
821}
822
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000823DWARFCompileUnit*
Greg Clayton1f746072012-08-29 21:13:06 +0000824SymbolFileDWARF::GetDWARFCompileUnit(lldb_private::CompileUnit *comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000825{
Greg Claytonea4a5bb2015-09-04 22:29:46 +0000826 if (!comp_unit)
827 return nullptr;
828
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000829 DWARFDebugInfo* info = DebugInfo();
Greg Clayton1f746072012-08-29 21:13:06 +0000830 if (info)
831 {
Greg Clayton2f869fe2016-03-30 20:14:35 +0000832 // Just a normal DWARF file whose user ID for the compile unit is
833 // the DWARF offset itself
Greg Clayton68c00bd2015-02-05 02:10:29 +0000834
Greg Clayton2f869fe2016-03-30 20:14:35 +0000835 DWARFCompileUnit *dwarf_cu = info->GetCompileUnit((dw_offset_t)comp_unit->GetID());
836 if (dwarf_cu && dwarf_cu->GetUserData() == NULL)
837 dwarf_cu->SetUserData(comp_unit);
838 return dwarf_cu;
Greg Clayton1f746072012-08-29 21:13:06 +0000839 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000840 return NULL;
841}
842
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000843
844DWARFDebugRanges*
845SymbolFileDWARF::DebugRanges()
846{
847 if (m_ranges.get() == NULL)
848 {
Saleem Abdulrasool324a1032014-04-04 04:06:10 +0000849 Timer scoped_timer(__PRETTY_FUNCTION__, "%s this = %p",
850 __PRETTY_FUNCTION__, static_cast<void*>(this));
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000851 if (get_debug_ranges_data().GetByteSize() > 0)
852 {
853 m_ranges.reset(new DWARFDebugRanges());
854 if (m_ranges.get())
855 m_ranges->Extract(this);
856 }
857 }
858 return m_ranges.get();
859}
860
861const DWARFDebugRanges*
862SymbolFileDWARF::DebugRanges() const
863{
864 return m_ranges.get();
865}
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000866
Greg Clayton53eb1c22012-04-02 22:59:12 +0000867lldb::CompUnitSP
868SymbolFileDWARF::ParseCompileUnit (DWARFCompileUnit* dwarf_cu, uint32_t cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000869{
Greg Clayton53eb1c22012-04-02 22:59:12 +0000870 CompUnitSP cu_sp;
871 if (dwarf_cu)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000872 {
Greg Clayton53eb1c22012-04-02 22:59:12 +0000873 CompileUnit *comp_unit = (CompileUnit*)dwarf_cu->GetUserData();
874 if (comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000875 {
Greg Clayton53eb1c22012-04-02 22:59:12 +0000876 // We already parsed this compile unit, had out a shared pointer to it
877 cu_sp = comp_unit->shared_from_this();
878 }
879 else
880 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +0000881 if (dwarf_cu->GetSymbolFileDWARF() != this)
882 {
883 return dwarf_cu->GetSymbolFileDWARF()->ParseCompileUnit(dwarf_cu, cu_idx);
884 }
Greg Clayton2f869fe2016-03-30 20:14:35 +0000885 else if (dwarf_cu->GetOffset() == 0 && GetDebugMapSymfile ())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000886 {
Greg Clayton1f746072012-08-29 21:13:06 +0000887 // Let the debug map create the compile unit
888 cu_sp = m_debug_map_symfile->GetCompileUnit(this);
889 dwarf_cu->SetUserData(cu_sp.get());
890 }
891 else
892 {
893 ModuleSP module_sp (m_obj_file->GetModule());
894 if (module_sp)
Greg Clayton53eb1c22012-04-02 22:59:12 +0000895 {
Greg Clayton5ce1a842015-08-27 18:09:44 +0000896 const DWARFDIE cu_die = dwarf_cu->GetCompileUnitDIEOnly ();
Greg Clayton1f746072012-08-29 21:13:06 +0000897 if (cu_die)
Greg Clayton53eb1c22012-04-02 22:59:12 +0000898 {
Greg Clayton5ce1a842015-08-27 18:09:44 +0000899 FileSpec cu_file_spec{cu_die.GetName(), false};
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000900 if (cu_file_spec)
Greg Clayton1f746072012-08-29 21:13:06 +0000901 {
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000902 // If we have a full path to the compile unit, we don't need to resolve
903 // the file. This can be expensive e.g. when the source files are NFS mounted.
Chaoren Lin372e9062015-06-09 17:54:27 +0000904 if (cu_file_spec.IsRelative())
Greg Clayton53eb1c22012-04-02 22:59:12 +0000905 {
Greg Clayton5ce1a842015-08-27 18:09:44 +0000906 const char *cu_comp_dir{cu_die.GetAttributeValueAsString(DW_AT_comp_dir, nullptr)};
Oleksiy Vyalov5d9c50b2015-07-21 02:09:42 +0000907 cu_file_spec.PrependPathComponent(resolveCompDir(cu_comp_dir));
Greg Clayton1f746072012-08-29 21:13:06 +0000908 }
909
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000910 std::string remapped_file;
911 if (module_sp->RemapSourceFile(cu_file_spec.GetCString(), remapped_file))
912 cu_file_spec.SetFile(remapped_file, false);
David Srbeckyd515e942015-07-08 14:00:04 +0000913 }
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000914
Greg Clayton5ce1a842015-08-27 18:09:44 +0000915 LanguageType cu_language = DWARFCompileUnit::LanguageTypeFromDWARF(cu_die.GetAttributeValueAsUnsigned(DW_AT_language, 0));
Chaoren Lin0c5a9c12015-06-05 00:28:06 +0000916
Jason Molendac7099582015-07-31 05:47:00 +0000917 bool is_optimized = dwarf_cu->GetIsOptimized ();
David Srbeckyd515e942015-07-08 14:00:04 +0000918 cu_sp.reset(new CompileUnit (module_sp,
919 dwarf_cu,
920 cu_file_spec,
Greg Clayton6071e6f2015-08-26 22:57:51 +0000921 dwarf_cu->GetID(),
Jason Molenda6ab659a2015-07-29 00:42:47 +0000922 cu_language,
923 is_optimized));
David Srbeckyd515e942015-07-08 14:00:04 +0000924 if (cu_sp)
925 {
926 // If we just created a compile unit with an invalid file spec, try and get the
927 // first entry in the supports files from the line table as that should be the
928 // compile unit.
929 if (!cu_file_spec)
Greg Clayton1f746072012-08-29 21:13:06 +0000930 {
David Srbeckyd515e942015-07-08 14:00:04 +0000931 cu_file_spec = cu_sp->GetSupportFiles().GetFileSpecAtIndex(1);
932 if (cu_file_spec)
933 {
934 (FileSpec &)(*cu_sp) = cu_file_spec;
935 // Also fix the invalid file spec which was copied from the compile unit.
936 cu_sp->GetSupportFiles().Replace(0, cu_file_spec);
937 }
Greg Clayton53eb1c22012-04-02 22:59:12 +0000938 }
David Srbeckyd515e942015-07-08 14:00:04 +0000939
940 dwarf_cu->SetUserData(cu_sp.get());
941
942 // Figure out the compile unit index if we weren't given one
943 if (cu_idx == UINT32_MAX)
944 DebugInfo()->GetCompileUnit(dwarf_cu->GetOffset(), &cu_idx);
945
946 m_obj_file->GetModule()->GetSymbolVendor()->SetCompileUnitAtIndex(cu_idx, cu_sp);
Greg Clayton53eb1c22012-04-02 22:59:12 +0000947 }
948 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000949 }
950 }
951 }
952 }
Greg Clayton53eb1c22012-04-02 22:59:12 +0000953 return cu_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000954}
955
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000956uint32_t
957SymbolFileDWARF::GetNumCompileUnits()
958{
959 DWARFDebugInfo* info = DebugInfo();
960 if (info)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000961 return info->GetNumCompileUnits();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000962 return 0;
963}
964
965CompUnitSP
966SymbolFileDWARF::ParseCompileUnitAtIndex(uint32_t cu_idx)
967{
Greg Clayton53eb1c22012-04-02 22:59:12 +0000968 CompUnitSP cu_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000969 DWARFDebugInfo* info = DebugInfo();
970 if (info)
971 {
Greg Clayton53eb1c22012-04-02 22:59:12 +0000972 DWARFCompileUnit* dwarf_cu = info->GetCompileUnitAtIndex(cu_idx);
973 if (dwarf_cu)
974 cu_sp = ParseCompileUnit(dwarf_cu, cu_idx);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000975 }
Greg Clayton53eb1c22012-04-02 22:59:12 +0000976 return cu_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000977}
978
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000979Function *
Greg Clayton6071e6f2015-08-26 22:57:51 +0000980SymbolFileDWARF::ParseCompileUnitFunction (const SymbolContext& sc, const DWARFDIE &die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000981{
Greg Clayton6071e6f2015-08-26 22:57:51 +0000982 if (die.IsValid())
983 {
984 TypeSystem *type_system = GetTypeSystemForLanguage(die.GetCU()->GetLanguageType());
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000985
Greg Clayton6071e6f2015-08-26 22:57:51 +0000986 if (type_system)
987 {
Greg Clayton261ac3f2015-08-28 01:01:03 +0000988 DWARFASTParser *dwarf_ast = type_system->GetDWARFParser();
989 if (dwarf_ast)
990 return dwarf_ast->ParseFunctionFromDWARF(sc, die);
Greg Clayton6071e6f2015-08-26 22:57:51 +0000991 }
992 }
993 return nullptr;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000994}
995
Greg Clayton9422dd62013-03-04 21:46:16 +0000996bool
997SymbolFileDWARF::FixupAddress (Address &addr)
998{
Greg Clayton2f869fe2016-03-30 20:14:35 +0000999 SymbolFileDWARFDebugMap * debug_map_symfile = GetDebugMapSymfile();
Greg Clayton9422dd62013-03-04 21:46:16 +00001000 if (debug_map_symfile)
1001 {
1002 return debug_map_symfile->LinkOSOAddress(addr);
1003 }
1004 // This is a normal DWARF file, no address fixups need to happen
1005 return true;
1006}
Greg Clayton1f746072012-08-29 21:13:06 +00001007lldb::LanguageType
1008SymbolFileDWARF::ParseCompileUnitLanguage (const SymbolContext& sc)
1009{
1010 assert (sc.comp_unit);
1011 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
1012 if (dwarf_cu)
Greg Clayton5ce1a842015-08-27 18:09:44 +00001013 return dwarf_cu->GetLanguageType();
1014 else
1015 return eLanguageTypeUnknown;
Greg Clayton1f746072012-08-29 21:13:06 +00001016}
1017
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001018size_t
1019SymbolFileDWARF::ParseCompileUnitFunctions(const SymbolContext &sc)
1020{
1021 assert (sc.comp_unit);
1022 size_t functions_added = 0;
Greg Clayton1f746072012-08-29 21:13:06 +00001023 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001024 if (dwarf_cu)
1025 {
1026 DWARFDIECollection function_dies;
Greg Clayton1f746072012-08-29 21:13:06 +00001027 const size_t num_functions = dwarf_cu->AppendDIEsWithTag (DW_TAG_subprogram, function_dies);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001028 size_t func_idx;
Greg Clayton1f746072012-08-29 21:13:06 +00001029 for (func_idx = 0; func_idx < num_functions; ++func_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001030 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001031 DWARFDIE die = function_dies.GetDIEAtIndex(func_idx);
1032 if (sc.comp_unit->FindFunctionByUID (die.GetID()).get() == NULL)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001033 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001034 if (ParseCompileUnitFunction(sc, die))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001035 ++functions_added;
1036 }
1037 }
1038 //FixupTypes();
1039 }
1040 return functions_added;
1041}
1042
1043bool
1044SymbolFileDWARF::ParseCompileUnitSupportFiles (const SymbolContext& sc, FileSpecList& support_files)
1045{
1046 assert (sc.comp_unit);
Greg Clayton1f746072012-08-29 21:13:06 +00001047 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Greg Claytonda2455b2012-11-01 17:28:37 +00001048 if (dwarf_cu)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001049 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001050 const DWARFDIE cu_die = dwarf_cu->GetCompileUnitDIEOnly();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001051
Greg Claytonda2455b2012-11-01 17:28:37 +00001052 if (cu_die)
1053 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001054 const char * cu_comp_dir = resolveCompDir(cu_die.GetAttributeValueAsString(DW_AT_comp_dir, nullptr));
Greg Clayton5ce1a842015-08-27 18:09:44 +00001055 const dw_offset_t stmt_list = cu_die.GetAttributeValueAsUnsigned(DW_AT_stmt_list, DW_INVALID_OFFSET);
Tamas Berghammer1966ac32016-01-11 14:56:05 +00001056 if (stmt_list != DW_INVALID_OFFSET)
1057 {
1058 // All file indexes in DWARF are one based and a file of index zero is
1059 // supposed to be the compile unit itself.
1060 support_files.Append (*sc.comp_unit);
1061 return DWARFDebugLine::ParseSupportFiles(sc.comp_unit->GetModule(),
1062 get_debug_line_data(),
1063 cu_comp_dir,
1064 stmt_list,
1065 support_files);
1066 }
Greg Claytonda2455b2012-11-01 17:28:37 +00001067 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001068 }
1069 return false;
1070}
1071
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001072bool
1073SymbolFileDWARF::ParseImportedModules (const lldb_private::SymbolContext &sc, std::vector<lldb_private::ConstString> &imported_modules)
1074{
1075 assert (sc.comp_unit);
1076 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
1077 if (dwarf_cu)
1078 {
1079 if (ClangModulesDeclVendor::LanguageSupportsClangModules(sc.comp_unit->GetLanguage()))
1080 {
1081 UpdateExternalModuleListIfNeeded();
Sean Callananb0300a42016-01-14 21:46:09 +00001082
1083 if (sc.comp_unit)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001084 {
Sean Callananb0300a42016-01-14 21:46:09 +00001085 const DWARFDIE die = dwarf_cu->GetCompileUnitDIEOnly();
1086
1087 if (die)
1088 {
1089 for (DWARFDIE child_die = die.GetFirstChild();
1090 child_die;
1091 child_die = child_die.GetSibling())
1092 {
1093 if (child_die.Tag() == DW_TAG_imported_declaration)
1094 {
1095 if (DWARFDIE module_die = child_die.GetReferencedDIE(DW_AT_import))
1096 {
1097 if (module_die.Tag() == DW_TAG_module)
1098 {
1099 if (const char *name = module_die.GetAttributeValueAsString(DW_AT_name, nullptr))
1100 {
1101 ConstString const_name(name);
1102 imported_modules.push_back(const_name);
1103 }
1104 }
1105 }
1106 }
1107 }
1108 }
1109 }
1110 else
1111 {
1112 for (const auto &pair : m_external_type_modules)
1113 {
1114 imported_modules.push_back(pair.first);
1115 }
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001116 }
1117 }
1118 }
1119 return false;
1120}
1121
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001122struct ParseDWARFLineTableCallbackInfo
1123{
1124 LineTable* line_table;
Greg Clayton7b0992d2013-04-18 22:45:39 +00001125 std::unique_ptr<LineSequence> sequence_ap;
Jaydeep Patil44d07fc2015-09-22 06:36:56 +00001126 lldb::addr_t addr_mask;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001127};
1128
1129//----------------------------------------------------------------------
1130// ParseStatementTableCallback
1131//----------------------------------------------------------------------
1132static void
1133ParseDWARFLineTableCallback(dw_offset_t offset, const DWARFDebugLine::State& state, void* userData)
1134{
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001135 if (state.row == DWARFDebugLine::State::StartParsingLineTable)
1136 {
1137 // Just started parsing the line table
1138 }
1139 else if (state.row == DWARFDebugLine::State::DoneParsingLineTable)
1140 {
1141 // Done parsing line table, nothing to do for the cleanup
1142 }
1143 else
1144 {
1145 ParseDWARFLineTableCallbackInfo* info = (ParseDWARFLineTableCallbackInfo*)userData;
Greg Clayton9422dd62013-03-04 21:46:16 +00001146 LineTable* line_table = info->line_table;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001147
Greg Clayton9422dd62013-03-04 21:46:16 +00001148 // If this is our first time here, we need to create a
1149 // sequence container.
1150 if (!info->sequence_ap.get())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001151 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001152 info->sequence_ap.reset(line_table->CreateLineSequenceContainer());
1153 assert(info->sequence_ap.get());
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001154 }
Greg Clayton9422dd62013-03-04 21:46:16 +00001155 line_table->AppendLineEntryToSequence (info->sequence_ap.get(),
Jaydeep Patil44d07fc2015-09-22 06:36:56 +00001156 state.address & info->addr_mask,
Greg Clayton9422dd62013-03-04 21:46:16 +00001157 state.line,
1158 state.column,
1159 state.file,
1160 state.is_stmt,
1161 state.basic_block,
1162 state.prologue_end,
1163 state.epilogue_begin,
1164 state.end_sequence);
1165 if (state.end_sequence)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001166 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001167 // First, put the current sequence into the line table.
1168 line_table->InsertSequence(info->sequence_ap.get());
1169 // Then, empty it to prepare for the next sequence.
1170 info->sequence_ap->Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001171 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001172 }
1173}
1174
1175bool
1176SymbolFileDWARF::ParseCompileUnitLineTable (const SymbolContext &sc)
1177{
1178 assert (sc.comp_unit);
1179 if (sc.comp_unit->GetLineTable() != NULL)
1180 return true;
1181
Greg Clayton1f746072012-08-29 21:13:06 +00001182 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001183 if (dwarf_cu)
1184 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001185 const DWARFDIE dwarf_cu_die = dwarf_cu->GetCompileUnitDIEOnly();
Greg Clayton129d12c2011-11-28 03:29:03 +00001186 if (dwarf_cu_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001187 {
Greg Clayton5ce1a842015-08-27 18:09:44 +00001188 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 +00001189 if (cu_line_offset != DW_INVALID_OFFSET)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001190 {
Greg Clayton7b0992d2013-04-18 22:45:39 +00001191 std::unique_ptr<LineTable> line_table_ap(new LineTable(sc.comp_unit));
Greg Clayton129d12c2011-11-28 03:29:03 +00001192 if (line_table_ap.get())
1193 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001194 ParseDWARFLineTableCallbackInfo info;
1195 info.line_table = line_table_ap.get();
Jaydeep Patil44d07fc2015-09-22 06:36:56 +00001196
1197 /*
1198 * MIPS:
1199 * The SymbolContext may not have a valid target, thus we may not be able
1200 * to call Address::GetOpcodeLoadAddress() which would clear the bit #0
1201 * for MIPS. Use ArchSpec to clear the bit #0.
1202 */
1203 ArchSpec arch;
1204 GetObjectFile()->GetArchitecture(arch);
1205 switch (arch.GetMachine())
1206 {
1207 case llvm::Triple::mips:
1208 case llvm::Triple::mipsel:
1209 case llvm::Triple::mips64:
1210 case llvm::Triple::mips64el:
1211 info.addr_mask = ~((lldb::addr_t)1);
1212 break;
1213 default:
1214 info.addr_mask = ~((lldb::addr_t)0);
1215 break;
1216 }
1217
Greg Claytonc7bece562013-01-25 18:06:21 +00001218 lldb::offset_t offset = cu_line_offset;
Greg Clayton129d12c2011-11-28 03:29:03 +00001219 DWARFDebugLine::ParseStatementTable(get_debug_line_data(), &offset, ParseDWARFLineTableCallback, &info);
Greg Clayton2f869fe2016-03-30 20:14:35 +00001220 SymbolFileDWARFDebugMap *debug_map_symfile = GetDebugMapSymfile();
1221 if (debug_map_symfile)
Greg Clayton9422dd62013-03-04 21:46:16 +00001222 {
1223 // We have an object file that has a line table with addresses
1224 // that are not linked. We need to link the line table and convert
1225 // the addresses that are relative to the .o file into addresses
1226 // for the main executable.
Greg Clayton2f869fe2016-03-30 20:14:35 +00001227 sc.comp_unit->SetLineTable (debug_map_symfile->LinkOSOLineTable (this, line_table_ap.get()));
Greg Clayton9422dd62013-03-04 21:46:16 +00001228 }
1229 else
1230 {
1231 sc.comp_unit->SetLineTable(line_table_ap.release());
1232 return true;
1233 }
Greg Clayton129d12c2011-11-28 03:29:03 +00001234 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001235 }
1236 }
1237 }
1238 return false;
1239}
1240
Siva Chandrad8335e92015-12-16 00:22:08 +00001241lldb_private::DebugMacrosSP
1242SymbolFileDWARF::ParseDebugMacros(lldb::offset_t *offset)
1243{
1244 auto iter = m_debug_macros_map.find(*offset);
1245 if (iter != m_debug_macros_map.end())
1246 return iter->second;
1247
1248 const DWARFDataExtractor &debug_macro_data = get_debug_macro_data();
1249 if (debug_macro_data.GetByteSize() == 0)
1250 return DebugMacrosSP();
1251
1252 lldb_private::DebugMacrosSP debug_macros_sp(new lldb_private::DebugMacros());
1253 m_debug_macros_map[*offset] = debug_macros_sp;
1254
1255 const DWARFDebugMacroHeader &header = DWARFDebugMacroHeader::ParseHeader(debug_macro_data, offset);
1256 DWARFDebugMacroEntry::ReadMacroEntries(
1257 debug_macro_data, get_debug_str_data(), header.OffsetIs64Bit(), offset, this, debug_macros_sp);
1258
1259 return debug_macros_sp;
1260}
1261
1262bool
1263SymbolFileDWARF::ParseCompileUnitDebugMacros(const SymbolContext& sc)
1264{
1265 assert (sc.comp_unit);
1266
1267 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
1268 if (dwarf_cu == nullptr)
1269 return false;
1270
1271 const DWARFDIE dwarf_cu_die = dwarf_cu->GetCompileUnitDIEOnly();
1272 if (!dwarf_cu_die)
1273 return false;
1274
1275 lldb::offset_t sect_offset = dwarf_cu_die.GetAttributeValueAsUnsigned(DW_AT_macros, DW_INVALID_OFFSET);
1276 if (sect_offset == DW_INVALID_OFFSET)
1277 sect_offset = dwarf_cu_die.GetAttributeValueAsUnsigned(DW_AT_GNU_macros, DW_INVALID_OFFSET);
1278 if (sect_offset == DW_INVALID_OFFSET)
1279 return false;
1280
1281 sc.comp_unit->SetDebugMacros(ParseDebugMacros(&sect_offset));
1282
1283 return true;
1284}
1285
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001286size_t
Greg Clayton6071e6f2015-08-26 22:57:51 +00001287SymbolFileDWARF::ParseFunctionBlocks (const SymbolContext& sc,
1288 Block *parent_block,
1289 const DWARFDIE &orig_die,
1290 addr_t subprogram_low_pc,
1291 uint32_t depth)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001292{
1293 size_t blocks_added = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00001294 DWARFDIE die = orig_die;
1295 while (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001296 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001297 dw_tag_t tag = die.Tag();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001298
1299 switch (tag)
1300 {
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001301 case DW_TAG_inlined_subroutine:
Greg Claytonb4d37332011-08-12 16:22:48 +00001302 case DW_TAG_subprogram:
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001303 case DW_TAG_lexical_block:
1304 {
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001305 Block *block = NULL;
Greg Claytondd7feaf2011-08-12 17:54:33 +00001306 if (tag == DW_TAG_subprogram)
1307 {
1308 // Skip any DW_TAG_subprogram DIEs that are inside
1309 // of a normal or inlined functions. These will be
1310 // parsed on their own as separate entities.
1311
1312 if (depth > 0)
1313 break;
1314
1315 block = parent_block;
1316 }
1317 else
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001318 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001319 BlockSP block_sp(new Block (die.GetID()));
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001320 parent_block->AddChild(block_sp);
1321 block = block_sp.get();
1322 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00001323 DWARFRangeList ranges;
Greg Claytondd7feaf2011-08-12 17:54:33 +00001324 const char *name = NULL;
1325 const char *mangled_name = NULL;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001326
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001327 int decl_file = 0;
1328 int decl_line = 0;
1329 int decl_column = 0;
1330 int call_file = 0;
1331 int call_line = 0;
1332 int call_column = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00001333 if (die.GetDIENamesAndRanges (name,
1334 mangled_name,
1335 ranges,
1336 decl_file, decl_line, decl_column,
1337 call_file, call_line, call_column, nullptr))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001338 {
1339 if (tag == DW_TAG_subprogram)
1340 {
1341 assert (subprogram_low_pc == LLDB_INVALID_ADDRESS);
Greg Claytonea3e7d52011-10-08 00:49:15 +00001342 subprogram_low_pc = ranges.GetMinRangeBase(0);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001343 }
Jim Inghamb0be4422010-08-12 01:20:14 +00001344 else if (tag == DW_TAG_inlined_subroutine)
1345 {
1346 // We get called here for inlined subroutines in two ways.
1347 // The first time is when we are making the Function object
1348 // for this inlined concrete instance. Since we're creating a top level block at
1349 // here, the subprogram_low_pc will be LLDB_INVALID_ADDRESS. So we need to
1350 // adjust the containing address.
1351 // The second time is when we are parsing the blocks inside the function that contains
1352 // the inlined concrete instance. Since these will be blocks inside the containing "real"
1353 // function the offset will be for that function.
1354 if (subprogram_low_pc == LLDB_INVALID_ADDRESS)
1355 {
Greg Claytonea3e7d52011-10-08 00:49:15 +00001356 subprogram_low_pc = ranges.GetMinRangeBase(0);
Jim Inghamb0be4422010-08-12 01:20:14 +00001357 }
1358 }
Greg Clayton103f3092015-01-15 03:04:37 +00001359
1360 const size_t num_ranges = ranges.GetSize();
1361 for (size_t i = 0; i<num_ranges; ++i)
1362 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001363 const DWARFRangeList::Entry &range = ranges.GetEntryRef (i);
Greg Clayton103f3092015-01-15 03:04:37 +00001364 const addr_t range_base = range.GetRangeBase();
1365 if (range_base >= subprogram_low_pc)
1366 block->AddRange(Block::Range (range_base - subprogram_low_pc, range.GetByteSize()));
1367 else
1368 {
1369 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",
1370 block->GetID(),
1371 range_base,
1372 range.GetRangeEnd(),
1373 subprogram_low_pc);
1374 }
1375 }
1376 block->FinalizeRanges ();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001377
1378 if (tag != DW_TAG_subprogram && (name != NULL || mangled_name != NULL))
1379 {
Greg Clayton7b0992d2013-04-18 22:45:39 +00001380 std::unique_ptr<Declaration> decl_ap;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001381 if (decl_file != 0 || decl_line != 0 || decl_column != 0)
Jim Inghamb0be4422010-08-12 01:20:14 +00001382 decl_ap.reset(new Declaration(sc.comp_unit->GetSupportFiles().GetFileSpecAtIndex(decl_file),
1383 decl_line, decl_column));
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001384
Greg Clayton7b0992d2013-04-18 22:45:39 +00001385 std::unique_ptr<Declaration> call_ap;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001386 if (call_file != 0 || call_line != 0 || call_column != 0)
Jim Inghamb0be4422010-08-12 01:20:14 +00001387 call_ap.reset(new Declaration(sc.comp_unit->GetSupportFiles().GetFileSpecAtIndex(call_file),
1388 call_line, call_column));
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001389
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001390 block->SetInlinedFunctionInfo (name, mangled_name, decl_ap.get(), call_ap.get());
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001391 }
1392
1393 ++blocks_added;
1394
Greg Clayton6071e6f2015-08-26 22:57:51 +00001395 if (die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001396 {
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001397 blocks_added += ParseFunctionBlocks (sc,
1398 block,
Greg Clayton6071e6f2015-08-26 22:57:51 +00001399 die.GetFirstChild(),
Greg Clayton0b76a2c2010-08-21 02:22:51 +00001400 subprogram_low_pc,
Greg Claytondd7feaf2011-08-12 17:54:33 +00001401 depth + 1);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001402 }
1403 }
1404 }
1405 break;
1406 default:
1407 break;
1408 }
1409
Greg Claytondd7feaf2011-08-12 17:54:33 +00001410 // Only parse siblings of the block if we are not at depth zero. A depth
1411 // of zero indicates we are currently parsing the top level
1412 // DW_TAG_subprogram DIE
1413
1414 if (depth == 0)
Greg Clayton6071e6f2015-08-26 22:57:51 +00001415 die.Clear();
Greg Claytondd7feaf2011-08-12 17:54:33 +00001416 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00001417 die = die.GetSibling();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001418 }
1419 return blocks_added;
1420}
1421
Greg Claytonf0705c82011-10-22 03:33:13 +00001422bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00001423SymbolFileDWARF::ClassOrStructIsVirtual (const DWARFDIE &parent_die)
Greg Claytonc4ffd662013-03-08 01:37:30 +00001424{
1425 if (parent_die)
1426 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001427 for (DWARFDIE die = parent_die.GetFirstChild(); die; die = die.GetSibling())
Greg Claytonc4ffd662013-03-08 01:37:30 +00001428 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001429 dw_tag_t tag = die.Tag();
Greg Claytonc4ffd662013-03-08 01:37:30 +00001430 bool check_virtuality = false;
1431 switch (tag)
1432 {
1433 case DW_TAG_inheritance:
1434 case DW_TAG_subprogram:
1435 check_virtuality = true;
1436 break;
1437 default:
1438 break;
1439 }
1440 if (check_virtuality)
1441 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001442 if (die.GetAttributeValueAsUnsigned(DW_AT_virtuality, 0) != 0)
Greg Claytonc4ffd662013-03-08 01:37:30 +00001443 return true;
1444 }
1445 }
1446 }
1447 return false;
1448}
1449
Paul Hermanea188fc2015-09-16 18:48:30 +00001450void
1451SymbolFileDWARF::ParseDeclsForContext (CompilerDeclContext decl_ctx)
1452{
1453 TypeSystem *type_system = decl_ctx.GetTypeSystem();
1454 DWARFASTParser *ast_parser = type_system->GetDWARFParser();
1455 std::vector<DWARFDIE> decl_ctx_die_list = ast_parser->GetDIEForDeclContext(decl_ctx);
1456
1457 for (DWARFDIE decl_ctx_die : decl_ctx_die_list)
1458 for (DWARFDIE decl = decl_ctx_die.GetFirstChild(); decl; decl = decl.GetSibling())
1459 ast_parser->GetDeclForUIDFromDWARF(decl);
1460}
1461
Greg Clayton07c8c442016-04-25 23:39:19 +00001462SymbolFileDWARF *
1463SymbolFileDWARF::GetDWARFForUID (lldb::user_id_t uid)
1464{
1465 // Anytime we get a "lldb::user_id_t" from an lldb_private::SymbolFile API
1466 // we must make sure we use the correct DWARF file when resolving things.
1467 // On MacOSX, when using SymbolFileDWARFDebugMap, we will use multiple
1468 // SymbolFileDWARF classes, one for each .o file. We can often end up
1469 // with references to other DWARF objects and we must be ready to receive
1470 // a "lldb::user_id_t" that specifies a DIE from another SymbolFileDWARF
1471 // instance.
1472 SymbolFileDWARFDebugMap *debug_map = GetDebugMapSymfile();
1473 if (debug_map)
1474 return debug_map->GetSymbolFileByOSOIndex(debug_map->GetOSOIndexFromUserID(uid));
1475 return this;
1476}
1477
1478DWARFDIE
1479SymbolFileDWARF::GetDIEFromUID (lldb::user_id_t uid)
1480{
1481 // Anytime we get a "lldb::user_id_t" from an lldb_private::SymbolFile API
1482 // we must make sure we use the correct DWARF file when resolving things.
1483 // On MacOSX, when using SymbolFileDWARFDebugMap, we will use multiple
1484 // SymbolFileDWARF classes, one for each .o file. We can often end up
1485 // with references to other DWARF objects and we must be ready to receive
1486 // a "lldb::user_id_t" that specifies a DIE from another SymbolFileDWARF
1487 // instance.
1488 SymbolFileDWARF *dwarf = GetDWARFForUID(uid);
1489 if (dwarf)
1490 return dwarf->GetDIE(DIERef(uid, dwarf));
1491 return DWARFDIE();
1492}
1493
Paul Hermand628cbb2015-09-15 23:44:17 +00001494CompilerDecl
1495SymbolFileDWARF::GetDeclForUID (lldb::user_id_t type_uid)
1496{
Greg Clayton07c8c442016-04-25 23:39:19 +00001497 // Anytime we have a lldb::user_id_t, we must get the DIE by
1498 // calling SymbolFileDWARF::GetDIEFromUID(). See comments inside
1499 // the SymbolFileDWARF::GetDIEFromUID() for details.
1500 DWARFDIE die = GetDIEFromUID(type_uid);
Greg Clayton2f869fe2016-03-30 20:14:35 +00001501 if (die)
1502 return die.GetDecl();
Paul Hermand628cbb2015-09-15 23:44:17 +00001503 return CompilerDecl();
1504}
1505
Greg Clayton99558cc42015-08-24 23:46:31 +00001506CompilerDeclContext
1507SymbolFileDWARF::GetDeclContextForUID (lldb::user_id_t type_uid)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001508{
Greg Clayton07c8c442016-04-25 23:39:19 +00001509 // Anytime we have a lldb::user_id_t, we must get the DIE by
1510 // calling SymbolFileDWARF::GetDIEFromUID(). See comments inside
1511 // the SymbolFileDWARF::GetDIEFromUID() for details.
1512 DWARFDIE die = GetDIEFromUID(type_uid);
Greg Clayton2f869fe2016-03-30 20:14:35 +00001513 if (die)
1514 return die.GetDeclContext();
Greg Clayton99558cc42015-08-24 23:46:31 +00001515 return CompilerDeclContext();
Sean Callanan72e49402011-08-05 23:43:37 +00001516}
1517
Greg Clayton99558cc42015-08-24 23:46:31 +00001518CompilerDeclContext
1519SymbolFileDWARF::GetDeclContextContainingUID (lldb::user_id_t type_uid)
Sean Callanan72e49402011-08-05 23:43:37 +00001520{
Greg Clayton07c8c442016-04-25 23:39:19 +00001521 // Anytime we have a lldb::user_id_t, we must get the DIE by
1522 // calling SymbolFileDWARF::GetDIEFromUID(). See comments inside
1523 // the SymbolFileDWARF::GetDIEFromUID() for details.
1524 DWARFDIE die = GetDIEFromUID(type_uid);
Greg Clayton2f869fe2016-03-30 20:14:35 +00001525 if (die)
1526 return die.GetContainingDeclContext();
Greg Clayton99558cc42015-08-24 23:46:31 +00001527 return CompilerDeclContext();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001528}
1529
Greg Clayton99558cc42015-08-24 23:46:31 +00001530
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001531Type*
Greg Claytonc685f8e2010-09-15 04:15:46 +00001532SymbolFileDWARF::ResolveTypeUID (lldb::user_id_t type_uid)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001533{
Greg Clayton07c8c442016-04-25 23:39:19 +00001534 // Anytime we have a lldb::user_id_t, we must get the DIE by
1535 // calling SymbolFileDWARF::GetDIEFromUID(). See comments inside
1536 // the SymbolFileDWARF::GetDIEFromUID() for details.
1537 DWARFDIE type_die = GetDIEFromUID(type_uid);
Greg Clayton2f869fe2016-03-30 20:14:35 +00001538 if (type_die)
Greg Clayton07c8c442016-04-25 23:39:19 +00001539 return type_die.ResolveType();
1540 else
1541 return nullptr;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001542}
1543
Greg Claytoncab36a32011-12-08 05:16:30 +00001544Type*
Greg Clayton2f869fe2016-03-30 20:14:35 +00001545SymbolFileDWARF::ResolveTypeUID (const DIERef &die_ref)
1546{
1547 return ResolveType (GetDIE(die_ref), true);
1548}
1549
1550Type*
Greg Clayton6071e6f2015-08-26 22:57:51 +00001551SymbolFileDWARF::ResolveTypeUID (const DWARFDIE &die, bool assert_not_being_parsed)
Sean Callanan5b26f272012-02-04 08:49:35 +00001552{
Greg Clayton6071e6f2015-08-26 22:57:51 +00001553 if (die)
Greg Claytoncab36a32011-12-08 05:16:30 +00001554 {
Greg Clayton5160ce52013-03-27 23:08:40 +00001555 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_DEBUG_INFO));
Greg Clayton3bffb082011-12-10 02:15:28 +00001556 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00001557 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00001558 "SymbolFileDWARF::ResolveTypeUID (die = 0x%8.8x) %s '%s'",
Greg Clayton6071e6f2015-08-26 22:57:51 +00001559 die.GetOffset(),
1560 die.GetTagAsCString(),
1561 die.GetName());
Greg Clayton3bffb082011-12-10 02:15:28 +00001562
Greg Claytoncab36a32011-12-08 05:16:30 +00001563 // We might be coming in in the middle of a type tree (a class
1564 // withing a class, an enum within a class), so parse any needed
1565 // parent DIEs before we get to this one...
Greg Clayton6071e6f2015-08-26 22:57:51 +00001566 DWARFDIE decl_ctx_die = GetDeclContextDIEContainingDIE (die);
1567 if (decl_ctx_die)
Greg Claytoncab36a32011-12-08 05:16:30 +00001568 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001569 if (log)
Greg Claytoncab36a32011-12-08 05:16:30 +00001570 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001571 switch (decl_ctx_die.Tag())
1572 {
1573 case DW_TAG_structure_type:
1574 case DW_TAG_union_type:
1575 case DW_TAG_class_type:
1576 {
1577 // Get the type, which could be a forward declaration
1578 if (log)
1579 GetObjectFile()->GetModule()->LogMessage (log,
1580 "SymbolFileDWARF::ResolveTypeUID (die = 0x%8.8x) %s '%s' resolve parent forward type for 0x%8.8x",
1581 die.GetOffset(),
1582 die.GetTagAsCString(),
1583 die.GetName(),
1584 decl_ctx_die.GetOffset());
1585 }
1586 break;
Greg Claytoncab36a32011-12-08 05:16:30 +00001587
Greg Clayton6071e6f2015-08-26 22:57:51 +00001588 default:
1589 break;
1590 }
1591 }
Greg Claytoncab36a32011-12-08 05:16:30 +00001592 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00001593 return ResolveType (die);
Greg Claytoncab36a32011-12-08 05:16:30 +00001594 }
1595 return NULL;
1596}
1597
Greg Clayton6beaaa62011-01-17 03:46:26 +00001598// This function is used when SymbolFileDWARFDebugMap owns a bunch of
1599// SymbolFileDWARF objects to detect if this DWARF file is the one that
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001600// can resolve a compiler_type.
Greg Clayton6beaaa62011-01-17 03:46:26 +00001601bool
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001602SymbolFileDWARF::HasForwardDeclForClangType (const CompilerType &compiler_type)
Greg Clayton6beaaa62011-01-17 03:46:26 +00001603{
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001604 CompilerType compiler_type_no_qualifiers = ClangUtil::RemoveFastQualifiers(compiler_type);
Greg Claytone6b36cd2015-12-08 01:02:08 +00001605 if (GetForwardDeclClangTypeToDie().count (compiler_type_no_qualifiers.GetOpaqueQualType()))
1606 {
1607 return true;
1608 }
1609 TypeSystem *type_system = compiler_type.GetTypeSystem();
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001610
1611 ClangASTContext *clang_type_system = llvm::dyn_cast_or_null<ClangASTContext>(type_system);
1612 if (!clang_type_system)
1613 return false;
1614 DWARFASTParserClang *ast_parser = static_cast<DWARFASTParserClang *>(clang_type_system->GetDWARFParser());
1615 return ast_parser->GetClangASTImporter().CanImport(compiler_type);
Greg Clayton6beaaa62011-01-17 03:46:26 +00001616}
1617
1618
Greg Clayton57ee3062013-07-11 22:46:58 +00001619bool
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001620SymbolFileDWARF::CompleteType (CompilerType &compiler_type)
Greg Clayton1be10fc2010-09-29 01:12:09 +00001621{
Greg Claytonfb85e622016-02-09 22:36:24 +00001622 lldb_private::Mutex::Locker locker(GetObjectFile()->GetModule()->GetMutex());
1623
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001624 ClangASTContext *clang_type_system = llvm::dyn_cast_or_null<ClangASTContext>(compiler_type.GetTypeSystem());
1625 if (clang_type_system)
Greg Claytone6b36cd2015-12-08 01:02:08 +00001626 {
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001627 DWARFASTParserClang *ast_parser = static_cast<DWARFASTParserClang *>(clang_type_system->GetDWARFParser());
1628 if (ast_parser && ast_parser->GetClangASTImporter().CanImport(compiler_type))
1629 return ast_parser->GetClangASTImporter().CompleteType(compiler_type);
Greg Claytone6b36cd2015-12-08 01:02:08 +00001630 }
1631
Greg Clayton1be10fc2010-09-29 01:12:09 +00001632 // We have a struct/union/class/enum that needs to be fully resolved.
Zachary Turnerd133f6a2016-03-28 22:53:41 +00001633 CompilerType compiler_type_no_qualifiers = ClangUtil::RemoveFastQualifiers(compiler_type);
Bruce Mitchener3ad353f2015-09-24 03:54:50 +00001634 auto die_it = GetForwardDeclClangTypeToDie().find (compiler_type_no_qualifiers.GetOpaqueQualType());
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001635 if (die_it == GetForwardDeclClangTypeToDie().end())
Greg Clayton73b472d2010-10-27 03:32:59 +00001636 {
1637 // We have already resolved this type...
Greg Clayton57ee3062013-07-11 22:46:58 +00001638 return true;
Greg Clayton73b472d2010-10-27 03:32:59 +00001639 }
Tamas Berghammerf8fd9b52015-09-14 15:44:29 +00001640
Greg Clayton2f869fe2016-03-30 20:14:35 +00001641 DWARFDIE dwarf_die = GetDIE(die_it->getSecond());
1642 if (dwarf_die)
1643 {
1644 // Once we start resolving this type, remove it from the forward declaration
1645 // map in case anyone child members or other types require this type to get resolved.
1646 // The type will get resolved when all of the calls to SymbolFileDWARF::ResolveClangOpaqueTypeDefinition
1647 // are done.
1648 GetForwardDeclClangTypeToDie().erase (die_it);
Tamas Berghammerf8fd9b52015-09-14 15:44:29 +00001649
Greg Clayton2f869fe2016-03-30 20:14:35 +00001650 Type *type = GetDIEToType().lookup (dwarf_die.GetDIE());
Tamas Berghammer69d0b332015-10-09 12:43:08 +00001651
Greg Clayton2f869fe2016-03-30 20:14:35 +00001652 Log *log (LogChannelDWARF::GetLogIfAny(DWARF_LOG_DEBUG_INFO|DWARF_LOG_TYPE_COMPLETION));
1653 if (log)
1654 GetObjectFile()->GetModule()->LogMessageVerboseBacktrace (log,
1655 "0x%8.8" PRIx64 ": %s '%s' resolving forward declaration...",
1656 dwarf_die.GetID(),
1657 dwarf_die.GetTagAsCString(),
1658 type->GetName().AsCString());
1659 assert (compiler_type);
1660 DWARFASTParser *dwarf_ast = dwarf_die.GetDWARFParser();
1661 if (dwarf_ast)
1662 return dwarf_ast->CompleteTypeFromDWARF (dwarf_die, type, compiler_type);
1663 }
Ashok Thirumurthia4658a52013-07-30 14:58:39 +00001664 return false;
Greg Clayton1be10fc2010-09-29 01:12:09 +00001665}
1666
Greg Clayton8b4edba2015-08-14 20:02:05 +00001667Type*
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00001668SymbolFileDWARF::ResolveType (const DWARFDIE &die, bool assert_not_being_parsed, bool resolve_function_context)
Greg Clayton8b4edba2015-08-14 20:02:05 +00001669{
Greg Clayton6071e6f2015-08-26 22:57:51 +00001670 if (die)
Greg Clayton8b4edba2015-08-14 20:02:05 +00001671 {
Greg Clayton2f869fe2016-03-30 20:14:35 +00001672 Type *type = GetTypeForDIE (die, resolve_function_context).get();
Greg Claytoncab36a32011-12-08 05:16:30 +00001673
Greg Clayton24739922010-10-13 03:15:28 +00001674 if (assert_not_being_parsed)
Jim Inghamc3549282012-01-11 02:21:12 +00001675 {
1676 if (type != DIE_IS_BEING_PARSED)
1677 return type;
1678
1679 GetObjectFile()->GetModule()->ReportError ("Parsing a die that is being parsed die: 0x%8.8x: %s %s",
Greg Clayton6071e6f2015-08-26 22:57:51 +00001680 die.GetOffset(),
1681 die.GetTagAsCString(),
1682 die.GetName());
Jim Inghamc3549282012-01-11 02:21:12 +00001683
1684 }
1685 else
1686 return type;
Greg Claytonc685f8e2010-09-15 04:15:46 +00001687 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00001688 return nullptr;
Greg Claytonc685f8e2010-09-15 04:15:46 +00001689}
1690
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001691CompileUnit*
Greg Clayton53eb1c22012-04-02 22:59:12 +00001692SymbolFileDWARF::GetCompUnitForDWARFCompUnit (DWARFCompileUnit* dwarf_cu, uint32_t cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001693{
1694 // Check if the symbol vendor already knows about this compile unit?
Greg Clayton53eb1c22012-04-02 22:59:12 +00001695 if (dwarf_cu->GetUserData() == NULL)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001696 {
1697 // The symbol vendor doesn't know about this compile unit, we
1698 // need to parse and add it to the symbol vendor object.
Greg Clayton53eb1c22012-04-02 22:59:12 +00001699 return ParseCompileUnit(dwarf_cu, cu_idx).get();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001700 }
Greg Clayton53eb1c22012-04-02 22:59:12 +00001701 return (CompileUnit*)dwarf_cu->GetUserData();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001702}
1703
Greg Clayton8b4edba2015-08-14 20:02:05 +00001704size_t
1705SymbolFileDWARF::GetObjCMethodDIEOffsets (ConstString class_name, DIEArray &method_die_offsets)
1706{
1707 method_die_offsets.clear();
1708 if (m_using_apple_tables)
1709 {
1710 if (m_apple_objc_ap.get())
1711 m_apple_objc_ap->FindByName(class_name.GetCString(), method_die_offsets);
1712 }
1713 else
1714 {
1715 if (!m_indexed)
1716 Index ();
1717
1718 m_objc_class_selectors_index.Find (class_name, method_die_offsets);
1719 }
1720 return method_die_offsets.size();
1721}
1722
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001723bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00001724SymbolFileDWARF::GetFunction (const DWARFDIE &die, SymbolContext& sc)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001725{
Greg Clayton72310352013-02-23 04:12:47 +00001726 sc.Clear(false);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001727
Greg Clayton6071e6f2015-08-26 22:57:51 +00001728 if (die)
1729 {
1730 // Check if the symbol vendor already knows about this compile unit?
1731 sc.comp_unit = GetCompUnitForDWARFCompUnit(die.GetCU(), UINT32_MAX);
1732
1733 sc.function = sc.comp_unit->FindFunctionByUID (die.GetID()).get();
1734 if (sc.function == NULL)
1735 sc.function = ParseCompileUnitFunction(sc, die);
1736
1737 if (sc.function)
1738 {
1739 sc.module_sp = sc.function->CalculateSymbolContextModule();
1740 return true;
1741 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00001742 }
1743
1744 return false;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001745}
1746
Greg Claytone6b36cd2015-12-08 01:02:08 +00001747lldb::ModuleSP
1748SymbolFileDWARF::GetDWOModule (ConstString name)
1749{
1750 UpdateExternalModuleListIfNeeded();
1751 const auto &pos = m_external_type_modules.find(name);
1752 if (pos != m_external_type_modules.end())
1753 return pos->second;
1754 else
1755 return lldb::ModuleSP();
1756}
1757
Greg Clayton2f869fe2016-03-30 20:14:35 +00001758DWARFDIE
1759SymbolFileDWARF::GetDIE (const DIERef &die_ref)
1760{
1761 DWARFDebugInfo * debug_info = DebugInfo();
1762 if (debug_info)
1763 return debug_info->GetDIE(die_ref);
1764 else
1765 return DWARFDIE();
1766}
1767
1768
Pavel Labatheb0c5c872016-03-29 13:42:02 +00001769std::unique_ptr<SymbolFileDWARFDwo>
1770SymbolFileDWARF::GetDwoSymbolFileForCompileUnit(DWARFCompileUnit &dwarf_cu, const DWARFDebugInfoEntry &cu_die)
1771{
1772 const char *dwo_name = cu_die.GetAttributeValueAsString(this, &dwarf_cu, DW_AT_GNU_dwo_name, nullptr);
1773 if (!dwo_name)
1774 return nullptr;
1775
1776 FileSpec dwo_file(dwo_name, true);
1777 if (dwo_file.IsRelative())
1778 {
1779 const char *comp_dir = cu_die.GetAttributeValueAsString(this, &dwarf_cu, DW_AT_comp_dir, nullptr);
1780 if (!comp_dir)
1781 return nullptr;
1782
1783 dwo_file.SetFile(comp_dir, true);
1784 dwo_file.AppendPathComponent(dwo_name);
1785 }
1786
1787 if (!dwo_file.Exists())
1788 return nullptr;
1789
1790 const lldb::offset_t file_offset = 0;
1791 DataBufferSP dwo_file_data_sp;
1792 lldb::offset_t dwo_file_data_offset = 0;
1793 ObjectFileSP dwo_obj_file = ObjectFile::FindPlugin(GetObjectFile()->GetModule(), &dwo_file, file_offset,
1794 dwo_file.GetByteSize(), dwo_file_data_sp, dwo_file_data_offset);
1795 if (dwo_obj_file == nullptr)
1796 return nullptr;
1797
1798 return llvm::make_unique<SymbolFileDWARFDwo>(dwo_obj_file, &dwarf_cu);
1799}
1800
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001801void
1802SymbolFileDWARF::UpdateExternalModuleListIfNeeded()
1803{
1804 if (m_fetched_external_modules)
1805 return;
1806 m_fetched_external_modules = true;
1807
1808 DWARFDebugInfo * debug_info = DebugInfo();
Tamas Berghammereb882fc2015-09-09 10:20:48 +00001809
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001810 const uint32_t num_compile_units = GetNumCompileUnits();
1811 for (uint32_t cu_idx = 0; cu_idx < num_compile_units; ++cu_idx)
1812 {
1813 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx);
1814
Greg Clayton5ce1a842015-08-27 18:09:44 +00001815 const DWARFDIE die = dwarf_cu->GetCompileUnitDIEOnly();
1816 if (die && die.HasChildren() == false)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001817 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001818 const char *name = die.GetAttributeValueAsString(DW_AT_name, nullptr);
1819
1820 if (name)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001821 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001822 ConstString const_name(name);
1823 if (m_external_type_modules.find(const_name) == m_external_type_modules.end())
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001824 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001825 ModuleSP module_sp;
1826 const char *dwo_path = die.GetAttributeValueAsString(DW_AT_GNU_dwo_name, nullptr);
1827 if (dwo_path)
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001828 {
Greg Claytone6b36cd2015-12-08 01:02:08 +00001829 ModuleSpec dwo_module_spec;
1830 dwo_module_spec.GetFileSpec().SetFile(dwo_path, false);
1831 dwo_module_spec.GetArchitecture() = m_obj_file->GetModule()->GetArchitecture();
1832 //printf ("Loading dwo = '%s'\n", dwo_path);
1833 Error error = ModuleList::GetSharedModule (dwo_module_spec, module_sp, NULL, NULL, NULL);
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001834 }
Greg Claytone6b36cd2015-12-08 01:02:08 +00001835 m_external_type_modules[const_name] = module_sp;
Sean Callananf0c5aeb2015-04-20 16:31:29 +00001836 }
1837 }
1838 }
1839 }
1840}
Greg Clayton2501e5e2015-01-15 02:59:20 +00001841
1842SymbolFileDWARF::GlobalVariableMap &
1843SymbolFileDWARF::GetGlobalAranges()
1844{
1845 if (!m_global_aranges_ap)
1846 {
1847 m_global_aranges_ap.reset (new GlobalVariableMap());
1848
1849 ModuleSP module_sp = GetObjectFile()->GetModule();
1850 if (module_sp)
1851 {
1852 const size_t num_cus = module_sp->GetNumCompileUnits();
1853 for (size_t i = 0; i < num_cus; ++i)
1854 {
1855 CompUnitSP cu_sp = module_sp->GetCompileUnitAtIndex(i);
1856 if (cu_sp)
1857 {
1858 VariableListSP globals_sp = cu_sp->GetVariableList(true);
1859 if (globals_sp)
1860 {
1861 const size_t num_globals = globals_sp->GetSize();
1862 for (size_t g = 0; g < num_globals; ++g)
1863 {
1864 VariableSP var_sp = globals_sp->GetVariableAtIndex(g);
1865 if (var_sp && !var_sp->GetLocationIsConstantValueData())
1866 {
1867 const DWARFExpression &location = var_sp->LocationExpression();
1868 Value location_result;
1869 Error error;
Tamas Berghammer5b42c7a2016-02-26 14:21:10 +00001870 if (location.Evaluate(nullptr, nullptr, nullptr, LLDB_INVALID_ADDRESS, nullptr, nullptr, location_result, &error))
Greg Clayton2501e5e2015-01-15 02:59:20 +00001871 {
1872 if (location_result.GetValueType() == Value::eValueTypeFileAddress)
1873 {
1874 lldb::addr_t file_addr = location_result.GetScalar().ULongLong();
1875 lldb::addr_t byte_size = 1;
1876 if (var_sp->GetType())
1877 byte_size = var_sp->GetType()->GetByteSize();
1878 m_global_aranges_ap->Append(GlobalVariableMap::Entry(file_addr, byte_size, var_sp.get()));
1879 }
1880 }
1881 }
1882 }
1883 }
1884 }
1885 }
1886 }
1887 m_global_aranges_ap->Sort();
1888 }
1889 return *m_global_aranges_ap;
1890}
1891
1892
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001893uint32_t
1894SymbolFileDWARF::ResolveSymbolContext (const Address& so_addr, uint32_t resolve_scope, SymbolContext& sc)
1895{
1896 Timer scoped_timer(__PRETTY_FUNCTION__,
Daniel Malead01b2952012-11-29 21:49:15 +00001897 "SymbolFileDWARF::ResolveSymbolContext (so_addr = { section = %p, offset = 0x%" PRIx64 " }, resolve_scope = 0x%8.8x)",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00001898 static_cast<void*>(so_addr.GetSection().get()),
1899 so_addr.GetOffset(), resolve_scope);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001900 uint32_t resolved = 0;
Greg Clayton2501e5e2015-01-15 02:59:20 +00001901 if (resolve_scope & ( eSymbolContextCompUnit |
1902 eSymbolContextFunction |
1903 eSymbolContextBlock |
1904 eSymbolContextLineEntry |
1905 eSymbolContextVariable ))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001906 {
1907 lldb::addr_t file_vm_addr = so_addr.GetFileAddress();
1908
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001909 DWARFDebugInfo* debug_info = DebugInfo();
Greg Claytond4a2b372011-09-12 23:21:58 +00001910 if (debug_info)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001911 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00001912 const dw_offset_t cu_offset = debug_info->GetCompileUnitAranges().FindAddress(file_vm_addr);
Greg Clayton2501e5e2015-01-15 02:59:20 +00001913 if (cu_offset == DW_INVALID_OFFSET)
1914 {
1915 // Global variables are not in the compile unit address ranges. The only way to
1916 // currently find global variables is to iterate over the .debug_pubnames or the
1917 // __apple_names table and find all items in there that point to DW_TAG_variable
1918 // DIEs and then find the address that matches.
1919 if (resolve_scope & eSymbolContextVariable)
1920 {
1921 GlobalVariableMap &map = GetGlobalAranges();
1922 const GlobalVariableMap::Entry *entry = map.FindEntryThatContains(file_vm_addr);
1923 if (entry && entry->data)
1924 {
1925 Variable *variable = entry->data;
1926 SymbolContextScope *scc = variable->GetSymbolContextScope();
1927 if (scc)
1928 {
1929 scc->CalculateSymbolContext(&sc);
1930 sc.variable = variable;
1931 }
1932 return sc.GetResolvedMask();
1933 }
1934 }
1935 }
1936 else
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001937 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00001938 uint32_t cu_idx = DW_INVALID_INDEX;
Greg Clayton6071e6f2015-08-26 22:57:51 +00001939 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnit(cu_offset, &cu_idx);
Greg Clayton53eb1c22012-04-02 22:59:12 +00001940 if (dwarf_cu)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001941 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00001942 sc.comp_unit = GetCompUnitForDWARFCompUnit(dwarf_cu, cu_idx);
Greg Clayton526a4ae2012-05-16 22:09:01 +00001943 if (sc.comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001944 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00001945 resolved |= eSymbolContextCompUnit;
1946
Greg Clayton6ab80132012-12-12 17:30:52 +00001947 bool force_check_line_table = false;
Greg Clayton526a4ae2012-05-16 22:09:01 +00001948 if (resolve_scope & (eSymbolContextFunction | eSymbolContextBlock))
1949 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001950 DWARFDIE function_die = dwarf_cu->LookupAddress(file_vm_addr);
1951 DWARFDIE block_die;
1952 if (function_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001953 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00001954 sc.function = sc.comp_unit->FindFunctionByUID (function_die.GetID()).get();
Greg Clayton526a4ae2012-05-16 22:09:01 +00001955 if (sc.function == NULL)
Greg Clayton6071e6f2015-08-26 22:57:51 +00001956 sc.function = ParseCompileUnitFunction(sc, function_die);
1957
1958 if (sc.function && (resolve_scope & eSymbolContextBlock))
1959 block_die = function_die.LookupDeepestBlock(file_vm_addr);
Greg Clayton526a4ae2012-05-16 22:09:01 +00001960 }
1961 else
1962 {
1963 // We might have had a compile unit that had discontiguous
1964 // address ranges where the gaps are symbols that don't have
1965 // any debug info. Discontiguous compile unit address ranges
1966 // should only happen when there aren't other functions from
1967 // other compile units in these gaps. This helps keep the size
1968 // of the aranges down.
Greg Clayton6ab80132012-12-12 17:30:52 +00001969 force_check_line_table = true;
Greg Clayton526a4ae2012-05-16 22:09:01 +00001970 }
1971
1972 if (sc.function != NULL)
1973 {
1974 resolved |= eSymbolContextFunction;
1975
1976 if (resolve_scope & eSymbolContextBlock)
1977 {
1978 Block& block = sc.function->GetBlock (true);
1979
Greg Clayton6071e6f2015-08-26 22:57:51 +00001980 if (block_die)
1981 sc.block = block.FindBlockByID (block_die.GetID());
Greg Clayton526a4ae2012-05-16 22:09:01 +00001982 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00001983 sc.block = block.FindBlockByID (function_die.GetID());
Greg Clayton526a4ae2012-05-16 22:09:01 +00001984 if (sc.block)
1985 resolved |= eSymbolContextBlock;
1986 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001987 }
1988 }
Greg Clayton6ab80132012-12-12 17:30:52 +00001989
1990 if ((resolve_scope & eSymbolContextLineEntry) || force_check_line_table)
1991 {
1992 LineTable *line_table = sc.comp_unit->GetLineTable();
1993 if (line_table != NULL)
1994 {
Greg Clayton9422dd62013-03-04 21:46:16 +00001995 // And address that makes it into this function should be in terms
1996 // of this debug file if there is no debug map, or it will be an
1997 // address in the .o file which needs to be fixed up to be in terms
1998 // of the debug map executable. Either way, calling FixupAddress()
1999 // will work for us.
2000 Address exe_so_addr (so_addr);
2001 if (FixupAddress(exe_so_addr))
Greg Clayton6ab80132012-12-12 17:30:52 +00002002 {
Greg Clayton9422dd62013-03-04 21:46:16 +00002003 if (line_table->FindLineEntryByAddress (exe_so_addr, sc.line_entry))
Greg Clayton6ab80132012-12-12 17:30:52 +00002004 {
2005 resolved |= eSymbolContextLineEntry;
2006 }
2007 }
Greg Clayton6ab80132012-12-12 17:30:52 +00002008 }
2009 }
2010
2011 if (force_check_line_table && !(resolved & eSymbolContextLineEntry))
2012 {
2013 // We might have had a compile unit that had discontiguous
2014 // address ranges where the gaps are symbols that don't have
2015 // any debug info. Discontiguous compile unit address ranges
2016 // should only happen when there aren't other functions from
2017 // other compile units in these gaps. This helps keep the size
2018 // of the aranges down.
2019 sc.comp_unit = NULL;
2020 resolved &= ~eSymbolContextCompUnit;
2021 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002022 }
Greg Clayton526a4ae2012-05-16 22:09:01 +00002023 else
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002024 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002025 GetObjectFile()->GetModule()->ReportWarning ("0x%8.8x: compile unit %u failed to create a valid lldb_private::CompileUnit class.",
2026 cu_offset,
2027 cu_idx);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002028 }
2029 }
2030 }
2031 }
2032 }
2033 return resolved;
2034}
2035
2036
2037
2038uint32_t
2039SymbolFileDWARF::ResolveSymbolContext(const FileSpec& file_spec, uint32_t line, bool check_inlines, uint32_t resolve_scope, SymbolContextList& sc_list)
2040{
2041 const uint32_t prev_size = sc_list.GetSize();
2042 if (resolve_scope & eSymbolContextCompUnit)
2043 {
2044 DWARFDebugInfo* debug_info = DebugInfo();
2045 if (debug_info)
2046 {
2047 uint32_t cu_idx;
Greg Clayton53eb1c22012-04-02 22:59:12 +00002048 DWARFCompileUnit* dwarf_cu = NULL;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002049
Greg Clayton53eb1c22012-04-02 22:59:12 +00002050 for (cu_idx = 0; (dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx)) != NULL; ++cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002051 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00002052 CompileUnit *dc_cu = GetCompUnitForDWARFCompUnit(dwarf_cu, cu_idx);
Sean Callananddd7a2a2013-10-03 22:27:29 +00002053 const bool full_match = (bool)file_spec.GetDirectory();
Greg Clayton1f746072012-08-29 21:13:06 +00002054 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 +00002055 if (check_inlines || file_spec_matches_cu_file_spec)
2056 {
2057 SymbolContext sc (m_obj_file->GetModule());
Greg Clayton53eb1c22012-04-02 22:59:12 +00002058 sc.comp_unit = GetCompUnitForDWARFCompUnit(dwarf_cu, cu_idx);
Greg Clayton526a4ae2012-05-16 22:09:01 +00002059 if (sc.comp_unit)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002060 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002061 uint32_t file_idx = UINT32_MAX;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002062
Greg Clayton526a4ae2012-05-16 22:09:01 +00002063 // If we are looking for inline functions only and we don't
2064 // find it in the support files, we are done.
2065 if (check_inlines)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002066 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002067 file_idx = sc.comp_unit->GetSupportFiles().FindFileIndex (1, file_spec, true);
2068 if (file_idx == UINT32_MAX)
2069 continue;
2070 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002071
Greg Clayton526a4ae2012-05-16 22:09:01 +00002072 if (line != 0)
2073 {
2074 LineTable *line_table = sc.comp_unit->GetLineTable();
2075
2076 if (line_table != NULL && line != 0)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002077 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002078 // We will have already looked up the file index if
2079 // we are searching for inline entries.
2080 if (!check_inlines)
2081 file_idx = sc.comp_unit->GetSupportFiles().FindFileIndex (1, file_spec, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002082
Greg Clayton526a4ae2012-05-16 22:09:01 +00002083 if (file_idx != UINT32_MAX)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002084 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002085 uint32_t found_line;
2086 uint32_t line_idx = line_table->FindLineEntryIndexByFileIndex (0, file_idx, line, false, &sc.line_entry);
2087 found_line = sc.line_entry.line;
2088
2089 while (line_idx != UINT32_MAX)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002090 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002091 sc.function = NULL;
2092 sc.block = NULL;
2093 if (resolve_scope & (eSymbolContextFunction | eSymbolContextBlock))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002094 {
Greg Clayton526a4ae2012-05-16 22:09:01 +00002095 const lldb::addr_t file_vm_addr = sc.line_entry.range.GetBaseAddress().GetFileAddress();
2096 if (file_vm_addr != LLDB_INVALID_ADDRESS)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002097 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002098 DWARFDIE function_die = dwarf_cu->LookupAddress(file_vm_addr);
2099 DWARFDIE block_die;
2100 if (function_die)
Greg Clayton526a4ae2012-05-16 22:09:01 +00002101 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002102 sc.function = sc.comp_unit->FindFunctionByUID (function_die.GetID()).get();
Greg Clayton526a4ae2012-05-16 22:09:01 +00002103 if (sc.function == NULL)
Greg Clayton6071e6f2015-08-26 22:57:51 +00002104 sc.function = ParseCompileUnitFunction(sc, function_die);
2105
2106 if (sc.function && (resolve_scope & eSymbolContextBlock))
2107 block_die = function_die.LookupDeepestBlock(file_vm_addr);
Greg Clayton526a4ae2012-05-16 22:09:01 +00002108 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002109
Greg Clayton526a4ae2012-05-16 22:09:01 +00002110 if (sc.function != NULL)
2111 {
2112 Block& block = sc.function->GetBlock (true);
2113
Greg Clayton6071e6f2015-08-26 22:57:51 +00002114 if (block_die)
2115 sc.block = block.FindBlockByID (block_die.GetID());
2116 else if (function_die)
2117 sc.block = block.FindBlockByID (function_die.GetID());
Greg Clayton526a4ae2012-05-16 22:09:01 +00002118 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002119 }
2120 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002121
Greg Clayton526a4ae2012-05-16 22:09:01 +00002122 sc_list.Append(sc);
2123 line_idx = line_table->FindLineEntryIndexByFileIndex (line_idx + 1, file_idx, found_line, true, &sc.line_entry);
2124 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002125 }
2126 }
Greg Clayton526a4ae2012-05-16 22:09:01 +00002127 else if (file_spec_matches_cu_file_spec && !check_inlines)
2128 {
2129 // only append the context if we aren't looking for inline call sites
2130 // by file and line and if the file spec matches that of the compile unit
2131 sc_list.Append(sc);
2132 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002133 }
2134 else if (file_spec_matches_cu_file_spec && !check_inlines)
2135 {
2136 // only append the context if we aren't looking for inline call sites
2137 // by file and line and if the file spec matches that of the compile unit
2138 sc_list.Append(sc);
2139 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002140
Greg Clayton526a4ae2012-05-16 22:09:01 +00002141 if (!check_inlines)
2142 break;
2143 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002144 }
2145 }
2146 }
2147 }
2148 return sc_list.GetSize() - prev_size;
2149}
2150
2151void
2152SymbolFileDWARF::Index ()
2153{
2154 if (m_indexed)
2155 return;
2156 m_indexed = true;
2157 Timer scoped_timer (__PRETTY_FUNCTION__,
2158 "SymbolFileDWARF::Index (%s)",
Jim Ingham4af59612014-12-19 19:20:44 +00002159 GetObjectFile()->GetFileSpec().GetFilename().AsCString("<Unknown>"));
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002160
2161 DWARFDebugInfo* debug_info = DebugInfo();
2162 if (debug_info)
2163 {
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002164 const uint32_t num_compile_units = GetNumCompileUnits();
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002165 std::vector<NameToDIE> function_basename_index(num_compile_units);
2166 std::vector<NameToDIE> function_fullname_index(num_compile_units);
2167 std::vector<NameToDIE> function_method_index(num_compile_units);
2168 std::vector<NameToDIE> function_selector_index(num_compile_units);
2169 std::vector<NameToDIE> objc_class_selectors_index(num_compile_units);
2170 std::vector<NameToDIE> global_index(num_compile_units);
2171 std::vector<NameToDIE> type_index(num_compile_units);
2172 std::vector<NameToDIE> namespace_index(num_compile_units);
2173
2174 auto parser_fn = [this,
2175 debug_info,
2176 &function_basename_index,
2177 &function_fullname_index,
2178 &function_method_index,
2179 &function_selector_index,
2180 &objc_class_selectors_index,
2181 &global_index,
2182 &type_index,
2183 &namespace_index](uint32_t cu_idx)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002184 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00002185 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx);
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002186 bool clear_dies = dwarf_cu->ExtractDIEsIfNeeded(false) > 1;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002187
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002188 dwarf_cu->Index(function_basename_index[cu_idx],
2189 function_fullname_index[cu_idx],
2190 function_method_index[cu_idx],
2191 function_selector_index[cu_idx],
2192 objc_class_selectors_index[cu_idx],
2193 global_index[cu_idx],
2194 type_index[cu_idx],
2195 namespace_index[cu_idx]);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002196
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002197 // Keep memory down by clearing DIEs if this generate function
2198 // caused them to be parsed
2199 if (clear_dies)
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002200 dwarf_cu->ClearDIEs(true);
2201
2202 return cu_idx;
2203 };
2204
2205 TaskRunner<uint32_t> task_runner;
2206 for (uint32_t cu_idx = 0; cu_idx < num_compile_units; ++cu_idx)
2207 task_runner.AddTask(parser_fn, cu_idx);
2208
2209 while (true)
2210 {
2211 std::future<uint32_t> f = task_runner.WaitForNextCompletedTask();
2212 if (!f.valid())
2213 break;
2214 uint32_t cu_idx = f.get();
2215
2216 m_function_basename_index.Append(function_basename_index[cu_idx]);
2217 m_function_fullname_index.Append(function_fullname_index[cu_idx]);
2218 m_function_method_index.Append(function_method_index[cu_idx]);
2219 m_function_selector_index.Append(function_selector_index[cu_idx]);
2220 m_objc_class_selectors_index.Append(objc_class_selectors_index[cu_idx]);
2221 m_global_index.Append(global_index[cu_idx]);
2222 m_type_index.Append(type_index[cu_idx]);
2223 m_namespace_index.Append(namespace_index[cu_idx]);
Tamas Berghammerda4e8ed2015-10-20 15:43:40 +00002224 }
Tamas Berghammer2ff88702015-10-23 10:34:49 +00002225
2226 TaskPool::RunTasks(
2227 [&]() { m_function_basename_index.Finalize(); },
2228 [&]() { m_function_fullname_index.Finalize(); },
2229 [&]() { m_function_method_index.Finalize(); },
2230 [&]() { m_function_selector_index.Finalize(); },
2231 [&]() { m_objc_class_selectors_index.Finalize(); },
2232 [&]() { m_global_index.Finalize(); },
2233 [&]() { m_type_index.Finalize(); },
2234 [&]() { m_namespace_index.Finalize(); });
Greg Claytonc685f8e2010-09-15 04:15:46 +00002235
Greg Clayton24739922010-10-13 03:15:28 +00002236#if defined (ENABLE_DEBUG_PRINTF)
Greg Clayton7bd65b92011-02-09 23:39:34 +00002237 StreamFile s(stdout, false);
Greg Claytonb5ad4ec2013-04-29 17:25:54 +00002238 s.Printf ("DWARF index for '%s':",
2239 GetObjectFile()->GetFileSpec().GetPath().c_str());
Greg Claytonba2d22d2010-11-13 22:57:37 +00002240 s.Printf("\nFunction basenames:\n"); m_function_basename_index.Dump (&s);
2241 s.Printf("\nFunction fullnames:\n"); m_function_fullname_index.Dump (&s);
2242 s.Printf("\nFunction methods:\n"); m_function_method_index.Dump (&s);
2243 s.Printf("\nFunction selectors:\n"); m_function_selector_index.Dump (&s);
2244 s.Printf("\nObjective C class selectors:\n"); m_objc_class_selectors_index.Dump (&s);
2245 s.Printf("\nGlobals and statics:\n"); m_global_index.Dump (&s);
Greg Clayton69b04882010-10-15 02:03:22 +00002246 s.Printf("\nTypes:\n"); m_type_index.Dump (&s);
Bruce Mitchenere171da52015-07-22 00:16:02 +00002247 s.Printf("\nNamespaces:\n") m_namespace_index.Dump (&s);
Greg Claytonc685f8e2010-09-15 04:15:46 +00002248#endif
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002249 }
2250}
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002251
2252bool
Greg Clayton99558cc42015-08-24 23:46:31 +00002253SymbolFileDWARF::DeclContextMatchesThisSymbolFile (const lldb_private::CompilerDeclContext *decl_ctx)
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002254{
Greg Clayton99558cc42015-08-24 23:46:31 +00002255 if (decl_ctx == nullptr || !decl_ctx->IsValid())
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002256 {
2257 // Invalid namespace decl which means we aren't matching only things
2258 // in this symbol file, so return true to indicate it matches this
2259 // symbol file.
2260 return true;
2261 }
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002262
Greg Clayton56939cb2015-09-17 22:23:34 +00002263 TypeSystem *decl_ctx_type_system = decl_ctx->GetTypeSystem();
2264 TypeSystem *type_system = GetTypeSystemForLanguage(decl_ctx_type_system->GetMinimumLanguage(nullptr));
2265 if (decl_ctx_type_system == type_system)
Greg Clayton99558cc42015-08-24 23:46:31 +00002266 return true; // The type systems match, return true
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002267
2268 // The namespace AST was valid, and it does not match...
Greg Clayton5160ce52013-03-27 23:08:40 +00002269 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Sean Callananc41e68b2011-10-13 21:08:11 +00002270
2271 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00002272 GetObjectFile()->GetModule()->LogMessage(log, "Valid namespace does not match symbol file");
Sean Callananc41e68b2011-10-13 21:08:11 +00002273
Greg Claytonbfe3dd42011-10-13 00:00:53 +00002274 return false;
2275}
2276
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002277uint32_t
Greg Clayton99558cc42015-08-24 23:46:31 +00002278SymbolFileDWARF::FindGlobalVariables (const ConstString &name, const CompilerDeclContext *parent_decl_ctx, bool append, uint32_t max_matches, VariableList& variables)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002279{
Greg Clayton5160ce52013-03-27 23:08:40 +00002280 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00002281
2282 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00002283 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00002284 "SymbolFileDWARF::FindGlobalVariables (name=\"%s\", parent_decl_ctx=%p, append=%u, max_matches=%u, variables)",
Greg Claytone38a5ed2012-01-05 03:57:59 +00002285 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00002286 static_cast<const void*>(parent_decl_ctx),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002287 append, max_matches);
2288
Greg Clayton99558cc42015-08-24 23:46:31 +00002289 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
Ed Maste4c24b122013-10-17 20:13:14 +00002290 return 0;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002291
Greg Claytonc685f8e2010-09-15 04:15:46 +00002292 DWARFDebugInfo* info = DebugInfo();
2293 if (info == NULL)
2294 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002295
2296 // If we aren't appending the results to this list, then clear the list
2297 if (!append)
2298 variables.Clear();
2299
2300 // Remember how many variables are in the list before we search in case
2301 // we are appending the results to a variable list.
2302 const uint32_t original_size = variables.GetSize();
2303
Greg Claytond4a2b372011-09-12 23:21:58 +00002304 DIEArray die_offsets;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002305
Greg Clayton97fbc342011-10-20 22:30:33 +00002306 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00002307 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002308 if (m_apple_names_ap.get())
2309 {
2310 const char *name_cstr = name.GetCString();
Jim Inghamfa39bb42014-10-25 00:33:55 +00002311 llvm::StringRef basename;
2312 llvm::StringRef context;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002313
Jim Inghamaa816b82015-09-02 01:59:14 +00002314 if (!CPlusPlusLanguage::ExtractContextAndIdentifier(name_cstr, context, basename))
Jim Inghamfa39bb42014-10-25 00:33:55 +00002315 basename = name_cstr;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002316
Jim Inghamfa39bb42014-10-25 00:33:55 +00002317 m_apple_names_ap->FindByName (basename.data(), die_offsets);
Greg Clayton97fbc342011-10-20 22:30:33 +00002318 }
Greg Clayton7f995132011-10-04 22:41:51 +00002319 }
2320 else
2321 {
2322 // Index the DWARF if we haven't already
2323 if (!m_indexed)
2324 Index ();
2325
2326 m_global_index.Find (name, die_offsets);
2327 }
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002328
Greg Clayton437a1352012-04-09 22:43:43 +00002329 const size_t num_die_matches = die_offsets.size();
2330 if (num_die_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002331 {
Greg Clayton7f995132011-10-04 22:41:51 +00002332 SymbolContext sc;
Greg Claytone72dfb32012-02-24 01:59:29 +00002333 sc.module_sp = m_obj_file->GetModule();
Greg Clayton7f995132011-10-04 22:41:51 +00002334 assert (sc.module_sp);
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002335
Greg Clayton437a1352012-04-09 22:43:43 +00002336 bool done = false;
2337 for (size_t i=0; i<num_die_matches && !done; ++i)
Greg Claytond4a2b372011-09-12 23:21:58 +00002338 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002339 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00002340 DWARFDIE die = GetDIE (die_ref);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002341
Greg Clayton95d87902011-11-11 03:16:25 +00002342 if (die)
2343 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002344 switch (die.Tag())
Greg Clayton437a1352012-04-09 22:43:43 +00002345 {
2346 default:
2347 case DW_TAG_subprogram:
2348 case DW_TAG_inlined_subroutine:
2349 case DW_TAG_try_block:
2350 case DW_TAG_catch_block:
2351 break;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002352
Greg Clayton437a1352012-04-09 22:43:43 +00002353 case DW_TAG_variable:
2354 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002355 sc.comp_unit = GetCompUnitForDWARFCompUnit(die.GetCU(), UINT32_MAX);
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002356
Greg Clayton99558cc42015-08-24 23:46:31 +00002357 if (parent_decl_ctx)
Greg Clayton8b4edba2015-08-14 20:02:05 +00002358 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002359 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
2360 if (dwarf_ast)
Greg Clayton99558cc42015-08-24 23:46:31 +00002361 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002362 CompilerDeclContext actual_parent_decl_ctx = dwarf_ast->GetDeclContextContainingUIDFromDWARF (die);
Greg Clayton99558cc42015-08-24 23:46:31 +00002363 if (!actual_parent_decl_ctx || actual_parent_decl_ctx != *parent_decl_ctx)
2364 continue;
2365 }
Greg Clayton8b4edba2015-08-14 20:02:05 +00002366 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002367
Greg Clayton6071e6f2015-08-26 22:57:51 +00002368 ParseVariables(sc, die, LLDB_INVALID_ADDRESS, false, false, &variables);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002369
Greg Clayton437a1352012-04-09 22:43:43 +00002370 if (variables.GetSize() - original_size >= max_matches)
2371 done = true;
2372 }
2373 break;
2374 }
Greg Clayton95d87902011-11-11 03:16:25 +00002375 }
2376 else
2377 {
2378 if (m_using_apple_tables)
2379 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00002380 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 +00002381 die_ref.die_offset, name.GetCString());
Greg Clayton95d87902011-11-11 03:16:25 +00002382 }
2383 }
Greg Claytond4a2b372011-09-12 23:21:58 +00002384 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002385 }
2386
2387 // Return the number of variable that were appended to the list
Greg Clayton437a1352012-04-09 22:43:43 +00002388 const uint32_t num_matches = variables.GetSize() - original_size;
2389 if (log && num_matches > 0)
2390 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002391 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00002392 "SymbolFileDWARF::FindGlobalVariables (name=\"%s\", parent_decl_ctx=%p, append=%u, max_matches=%u, variables) => %u",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002393 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00002394 static_cast<const void*>(parent_decl_ctx),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002395 append, max_matches,
Greg Clayton437a1352012-04-09 22:43:43 +00002396 num_matches);
2397 }
2398 return num_matches;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002399}
2400
2401uint32_t
2402SymbolFileDWARF::FindGlobalVariables(const RegularExpression& regex, bool append, uint32_t max_matches, VariableList& variables)
2403{
Greg Clayton5160ce52013-03-27 23:08:40 +00002404 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002405
Greg Clayton21f2a492011-10-06 00:09:08 +00002406 if (log)
2407 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002408 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002409 "SymbolFileDWARF::FindGlobalVariables (regex=\"%s\", append=%u, max_matches=%u, variables)",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00002410 regex.GetText(), append,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002411 max_matches);
Greg Clayton21f2a492011-10-06 00:09:08 +00002412 }
2413
Greg Claytonc685f8e2010-09-15 04:15:46 +00002414 DWARFDebugInfo* info = DebugInfo();
2415 if (info == NULL)
2416 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002417
2418 // If we aren't appending the results to this list, then clear the list
2419 if (!append)
2420 variables.Clear();
2421
2422 // Remember how many variables are in the list before we search in case
2423 // we are appending the results to a variable list.
2424 const uint32_t original_size = variables.GetSize();
2425
Greg Clayton7f995132011-10-04 22:41:51 +00002426 DIEArray die_offsets;
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002427
Greg Clayton97fbc342011-10-20 22:30:33 +00002428 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00002429 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002430 if (m_apple_names_ap.get())
Greg Claytond1767f02011-12-08 02:13:16 +00002431 {
2432 DWARFMappedHash::DIEInfoArray hash_data_array;
2433 if (m_apple_names_ap->AppendAllDIEsThatMatchingRegex (regex, hash_data_array))
2434 DWARFMappedHash::ExtractDIEArray (hash_data_array, die_offsets);
2435 }
Greg Clayton7f995132011-10-04 22:41:51 +00002436 }
2437 else
2438 {
2439 // Index the DWARF if we haven't already
2440 if (!m_indexed)
2441 Index ();
2442
2443 m_global_index.Find (regex, die_offsets);
2444 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002445
Greg Claytonc685f8e2010-09-15 04:15:46 +00002446 SymbolContext sc;
Greg Claytone72dfb32012-02-24 01:59:29 +00002447 sc.module_sp = m_obj_file->GetModule();
Greg Claytonc685f8e2010-09-15 04:15:46 +00002448 assert (sc.module_sp);
2449
Greg Clayton7f995132011-10-04 22:41:51 +00002450 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00002451 if (num_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002452 {
Greg Claytond4a2b372011-09-12 23:21:58 +00002453 for (size_t i=0; i<num_matches; ++i)
2454 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002455 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00002456 DWARFDIE die = GetDIE (die_ref);
Greg Clayton95d87902011-11-11 03:16:25 +00002457
2458 if (die)
2459 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002460 sc.comp_unit = GetCompUnitForDWARFCompUnit(die.GetCU(), UINT32_MAX);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002461
Greg Clayton6071e6f2015-08-26 22:57:51 +00002462 ParseVariables(sc, die, LLDB_INVALID_ADDRESS, false, false, &variables);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002463
Greg Clayton95d87902011-11-11 03:16:25 +00002464 if (variables.GetSize() - original_size >= max_matches)
2465 break;
2466 }
2467 else
2468 {
2469 if (m_using_apple_tables)
2470 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00002471 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 +00002472 die_ref.die_offset, regex.GetText());
Greg Clayton95d87902011-11-11 03:16:25 +00002473 }
2474 }
Greg Claytond4a2b372011-09-12 23:21:58 +00002475 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002476 }
2477
2478 // Return the number of variable that were appended to the list
2479 return variables.GetSize() - original_size;
2480}
2481
Greg Claytonaa044962011-10-13 00:59:38 +00002482
Jim Ingham4cda6e02011-10-07 22:23:45 +00002483bool
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002484SymbolFileDWARF::ResolveFunction (const DIERef& die_ref,
Pavel Labatha73d6572015-03-13 10:22:00 +00002485 bool include_inlines,
Jim Ingham4cda6e02011-10-07 22:23:45 +00002486 SymbolContextList& sc_list)
Greg Clayton9e315582011-09-02 04:03:59 +00002487{
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002488 DWARFDIE die = DebugInfo()->GetDIE (die_ref);
Greg Clayton6071e6f2015-08-26 22:57:51 +00002489 return ResolveFunction (die, include_inlines, sc_list);
Greg Claytonaa044962011-10-13 00:59:38 +00002490}
2491
2492
2493bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00002494SymbolFileDWARF::ResolveFunction (const DWARFDIE &orig_die,
Pavel Labatha73d6572015-03-13 10:22:00 +00002495 bool include_inlines,
Greg Claytonaa044962011-10-13 00:59:38 +00002496 SymbolContextList& sc_list)
2497{
Greg Clayton9e315582011-09-02 04:03:59 +00002498 SymbolContext sc;
Greg Claytonaa044962011-10-13 00:59:38 +00002499
Greg Clayton6071e6f2015-08-26 22:57:51 +00002500 if (!orig_die)
Greg Claytonaa044962011-10-13 00:59:38 +00002501 return false;
2502
Jim Ingham4cda6e02011-10-07 22:23:45 +00002503 // If we were passed a die that is not a function, just return false...
Greg Clayton6071e6f2015-08-26 22:57:51 +00002504 if (!(orig_die.Tag() == DW_TAG_subprogram || (include_inlines && orig_die.Tag() == DW_TAG_inlined_subroutine)))
Jim Ingham4cda6e02011-10-07 22:23:45 +00002505 return false;
Greg Clayton6071e6f2015-08-26 22:57:51 +00002506
2507 DWARFDIE die = orig_die;
2508 DWARFDIE inlined_die;
2509 if (die.Tag() == DW_TAG_inlined_subroutine)
Greg Clayton9e315582011-09-02 04:03:59 +00002510 {
Jim Ingham4cda6e02011-10-07 22:23:45 +00002511 inlined_die = die;
Greg Clayton9e315582011-09-02 04:03:59 +00002512
Greg Clayton6071e6f2015-08-26 22:57:51 +00002513 while (1)
Greg Clayton2bc22f82011-09-30 03:20:47 +00002514 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002515 die = die.GetParent();
2516
2517 if (die)
2518 {
2519 if (die.Tag() == DW_TAG_subprogram)
2520 break;
2521 }
2522 else
Jim Ingham4cda6e02011-10-07 22:23:45 +00002523 break;
Greg Clayton9e315582011-09-02 04:03:59 +00002524 }
2525 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00002526 assert (die && die.Tag() == DW_TAG_subprogram);
2527 if (GetFunction (die, sc))
Jim Ingham4cda6e02011-10-07 22:23:45 +00002528 {
2529 Address addr;
2530 // Parse all blocks if needed
2531 if (inlined_die)
2532 {
Greg Claytonf7bb1fb2015-01-15 03:13:44 +00002533 Block &function_block = sc.function->GetBlock (true);
Greg Clayton6071e6f2015-08-26 22:57:51 +00002534 sc.block = function_block.FindBlockByID (inlined_die.GetID());
Greg Claytonf7bb1fb2015-01-15 03:13:44 +00002535 if (sc.block == NULL)
Greg Clayton6071e6f2015-08-26 22:57:51 +00002536 sc.block = function_block.FindBlockByID (inlined_die.GetOffset());
Greg Claytonf7bb1fb2015-01-15 03:13:44 +00002537 if (sc.block == NULL || sc.block->GetStartAddress (addr) == false)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002538 addr.Clear();
2539 }
2540 else
2541 {
2542 sc.block = NULL;
2543 addr = sc.function->GetAddressRange().GetBaseAddress();
2544 }
2545
2546 if (addr.IsValid())
2547 {
Jim Ingham4cda6e02011-10-07 22:23:45 +00002548 sc_list.Append(sc);
Greg Claytonaa044962011-10-13 00:59:38 +00002549 return true;
Jim Ingham4cda6e02011-10-07 22:23:45 +00002550 }
2551 }
2552
Greg Claytonaa044962011-10-13 00:59:38 +00002553 return false;
Greg Clayton9e315582011-09-02 04:03:59 +00002554}
2555
Greg Clayton7f995132011-10-04 22:41:51 +00002556void
2557SymbolFileDWARF::FindFunctions (const ConstString &name,
2558 const NameToDIE &name_to_die,
Pavel Labatha73d6572015-03-13 10:22:00 +00002559 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002560 SymbolContextList& sc_list)
2561{
Greg Claytond4a2b372011-09-12 23:21:58 +00002562 DIEArray die_offsets;
Greg Clayton7f995132011-10-04 22:41:51 +00002563 if (name_to_die.Find (name, die_offsets))
2564 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002565 ParseFunctions (die_offsets, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002566 }
2567}
2568
2569
2570void
2571SymbolFileDWARF::FindFunctions (const RegularExpression &regex,
2572 const NameToDIE &name_to_die,
Pavel Labatha73d6572015-03-13 10:22:00 +00002573 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002574 SymbolContextList& sc_list)
2575{
2576 DIEArray die_offsets;
2577 if (name_to_die.Find (regex, die_offsets))
2578 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002579 ParseFunctions (die_offsets, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002580 }
2581}
2582
2583
2584void
2585SymbolFileDWARF::FindFunctions (const RegularExpression &regex,
2586 const DWARFMappedHash::MemoryTable &memory_table,
Pavel Labatha73d6572015-03-13 10:22:00 +00002587 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002588 SymbolContextList& sc_list)
2589{
2590 DIEArray die_offsets;
Greg Claytond1767f02011-12-08 02:13:16 +00002591 DWARFMappedHash::DIEInfoArray hash_data_array;
2592 if (memory_table.AppendAllDIEsThatMatchingRegex (regex, hash_data_array))
Greg Clayton7f995132011-10-04 22:41:51 +00002593 {
Greg Claytond1767f02011-12-08 02:13:16 +00002594 DWARFMappedHash::ExtractDIEArray (hash_data_array, die_offsets);
Pavel Labatha73d6572015-03-13 10:22:00 +00002595 ParseFunctions (die_offsets, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002596 }
2597}
2598
2599void
2600SymbolFileDWARF::ParseFunctions (const DIEArray &die_offsets,
Pavel Labatha73d6572015-03-13 10:22:00 +00002601 bool include_inlines,
Greg Clayton7f995132011-10-04 22:41:51 +00002602 SymbolContextList& sc_list)
2603{
2604 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00002605 if (num_matches)
Greg Claytonc685f8e2010-09-15 04:15:46 +00002606 {
Greg Claytond4a2b372011-09-12 23:21:58 +00002607 for (size_t i=0; i<num_matches; ++i)
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002608 ResolveFunction (die_offsets[i], include_inlines, sc_list);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002609 }
Greg Claytonc685f8e2010-09-15 04:15:46 +00002610}
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002611
Jim Ingham4cda6e02011-10-07 22:23:45 +00002612bool
Greg Clayton99558cc42015-08-24 23:46:31 +00002613SymbolFileDWARF::DIEInDeclContext (const CompilerDeclContext *decl_ctx,
Greg Clayton6071e6f2015-08-26 22:57:51 +00002614 const DWARFDIE &die)
Greg Clayton99558cc42015-08-24 23:46:31 +00002615{
2616 // If we have no parent decl context to match this DIE matches, and if the parent
2617 // decl context isn't valid, we aren't trying to look for any particular decl
2618 // context so any die matches.
2619 if (decl_ctx == nullptr || !decl_ctx->IsValid())
2620 return true;
2621
Greg Clayton6071e6f2015-08-26 22:57:51 +00002622 if (die)
Greg Clayton99558cc42015-08-24 23:46:31 +00002623 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002624 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
2625 if (dwarf_ast)
Greg Clayton99558cc42015-08-24 23:46:31 +00002626 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00002627 CompilerDeclContext actual_decl_ctx = dwarf_ast->GetDeclContextContainingUIDFromDWARF (die);
Greg Clayton99558cc42015-08-24 23:46:31 +00002628 if (actual_decl_ctx)
2629 return actual_decl_ctx == *decl_ctx;
2630 }
2631 }
2632 return false;
2633}
2634
Greg Clayton0c5cd902010-06-28 21:30:43 +00002635uint32_t
Greg Clayton99558cc42015-08-24 23:46:31 +00002636SymbolFileDWARF::FindFunctions (const ConstString &name,
2637 const CompilerDeclContext *parent_decl_ctx,
Sean Callanan9df05fb2012-02-10 22:52:19 +00002638 uint32_t name_type_mask,
2639 bool include_inlines,
Greg Clayton2bc22f82011-09-30 03:20:47 +00002640 bool append,
2641 SymbolContextList& sc_list)
Greg Clayton0c5cd902010-06-28 21:30:43 +00002642{
2643 Timer scoped_timer (__PRETTY_FUNCTION__,
2644 "SymbolFileDWARF::FindFunctions (name = '%s')",
2645 name.AsCString());
2646
Greg Clayton43fe2172013-04-03 02:00:15 +00002647 // eFunctionNameTypeAuto should be pre-resolved by a call to Module::PrepareForFunctionNameLookup()
2648 assert ((name_type_mask & eFunctionNameTypeAuto) == 0);
2649
Greg Clayton5160ce52013-03-27 23:08:40 +00002650 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00002651
2652 if (log)
2653 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002654 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002655 "SymbolFileDWARF::FindFunctions (name=\"%s\", name_type_mask=0x%x, append=%u, sc_list)",
2656 name.GetCString(),
2657 name_type_mask,
2658 append);
Greg Clayton21f2a492011-10-06 00:09:08 +00002659 }
2660
Greg Clayton0c5cd902010-06-28 21:30:43 +00002661 // If we aren't appending the results to this list, then clear the list
2662 if (!append)
2663 sc_list.Clear();
Sean Callanan213fdb82011-10-13 01:49:10 +00002664
Greg Clayton99558cc42015-08-24 23:46:31 +00002665 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
Ed Maste4c24b122013-10-17 20:13:14 +00002666 return 0;
Jim Ingham4cda6e02011-10-07 22:23:45 +00002667
2668 // If name is empty then we won't find anything.
2669 if (name.IsEmpty())
2670 return 0;
Greg Clayton0c5cd902010-06-28 21:30:43 +00002671
2672 // Remember how many sc_list are in the list before we search in case
2673 // we are appending the results to a variable list.
Greg Clayton9e315582011-09-02 04:03:59 +00002674
Jim Ingham4cda6e02011-10-07 22:23:45 +00002675 const char *name_cstr = name.GetCString();
Greg Clayton43fe2172013-04-03 02:00:15 +00002676
2677 const uint32_t original_size = sc_list.GetSize();
2678
Jim Ingham4cda6e02011-10-07 22:23:45 +00002679 DWARFDebugInfo* info = DebugInfo();
2680 if (info == NULL)
2681 return 0;
2682
Greg Clayton43fe2172013-04-03 02:00:15 +00002683 std::set<const DWARFDebugInfoEntry *> resolved_dies;
Greg Clayton97fbc342011-10-20 22:30:33 +00002684 if (m_using_apple_tables)
Greg Clayton4d01ace2011-09-29 16:58:15 +00002685 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002686 if (m_apple_names_ap.get())
Jim Ingham4cda6e02011-10-07 22:23:45 +00002687 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002688
2689 DIEArray die_offsets;
2690
2691 uint32_t num_matches = 0;
2692
Greg Clayton43fe2172013-04-03 02:00:15 +00002693 if (name_type_mask & eFunctionNameTypeFull)
Greg Claytonaa044962011-10-13 00:59:38 +00002694 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002695 // If they asked for the full name, match what they typed. At some point we may
2696 // want to canonicalize this (strip double spaces, etc. For now, we just add all the
2697 // dies that we find by exact match.
Jim Ingham4cda6e02011-10-07 22:23:45 +00002698 num_matches = m_apple_names_ap->FindByName (name_cstr, die_offsets);
Jim Ingham4cda6e02011-10-07 22:23:45 +00002699 for (uint32_t i = 0; i < num_matches; i++)
2700 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002701 const DIERef& die_ref = die_offsets[i];
2702 DWARFDIE die = info->GetDIE (die_ref);
Greg Claytonaa044962011-10-13 00:59:38 +00002703 if (die)
2704 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002705 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00002706 continue; // The containing decl contexts don't match
Greg Clayton8b4edba2015-08-14 20:02:05 +00002707
Greg Clayton6071e6f2015-08-26 22:57:51 +00002708 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002709 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002710 if (ResolveFunction (die, include_inlines, sc_list))
2711 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002712 }
Greg Claytonaa044962011-10-13 00:59:38 +00002713 }
Greg Clayton95d87902011-11-11 03:16:25 +00002714 else
2715 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00002716 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 +00002717 die_ref.die_offset, name_cstr);
Greg Clayton95d87902011-11-11 03:16:25 +00002718 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002719 }
Greg Clayton97fbc342011-10-20 22:30:33 +00002720 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002721
2722 if (name_type_mask & eFunctionNameTypeSelector)
2723 {
Greg Clayton99558cc42015-08-24 23:46:31 +00002724 if (parent_decl_ctx && parent_decl_ctx->IsValid())
Greg Clayton43fe2172013-04-03 02:00:15 +00002725 return 0; // no selectors in namespaces
Greg Clayton97fbc342011-10-20 22:30:33 +00002726
Greg Clayton43fe2172013-04-03 02:00:15 +00002727 num_matches = m_apple_names_ap->FindByName (name_cstr, die_offsets);
2728 // Now make sure these are actually ObjC methods. In this case we can simply look up the name,
2729 // and if it is an ObjC method name, we're good.
Greg Clayton97fbc342011-10-20 22:30:33 +00002730
Greg Clayton43fe2172013-04-03 02:00:15 +00002731 for (uint32_t i = 0; i < num_matches; i++)
Greg Clayton97fbc342011-10-20 22:30:33 +00002732 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002733 const DIERef& die_ref = die_offsets[i];
2734 DWARFDIE die = info->GetDIE (die_ref);
Greg Clayton43fe2172013-04-03 02:00:15 +00002735 if (die)
Greg Clayton97fbc342011-10-20 22:30:33 +00002736 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002737 const char *die_name = die.GetName();
Jim Inghamaa816b82015-09-02 01:59:14 +00002738 if (ObjCLanguage::IsPossibleObjCMethodName(die_name))
Greg Clayton97fbc342011-10-20 22:30:33 +00002739 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002740 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002741 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002742 if (ResolveFunction (die, include_inlines, sc_list))
2743 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002744 }
Greg Clayton97fbc342011-10-20 22:30:33 +00002745 }
2746 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002747 else
2748 {
2749 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 +00002750 die_ref.die_offset, name_cstr);
Greg Clayton43fe2172013-04-03 02:00:15 +00002751 }
Greg Clayton97fbc342011-10-20 22:30:33 +00002752 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002753 die_offsets.clear();
2754 }
2755
Greg Clayton99558cc42015-08-24 23:46:31 +00002756 if (((name_type_mask & eFunctionNameTypeMethod) && !parent_decl_ctx) || name_type_mask & eFunctionNameTypeBase)
Greg Clayton43fe2172013-04-03 02:00:15 +00002757 {
2758 // The apple_names table stores just the "base name" of C++ methods in the table. So we have to
2759 // extract the base name, look that up, and if there is any other information in the name we were
2760 // passed in we have to post-filter based on that.
2761
2762 // FIXME: Arrange the logic above so that we don't calculate the base name twice:
2763 num_matches = m_apple_names_ap->FindByName (name_cstr, die_offsets);
2764
2765 for (uint32_t i = 0; i < num_matches; i++)
2766 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002767 const DIERef& die_ref = die_offsets[i];
2768 DWARFDIE die = info->GetDIE (die_ref);
Greg Clayton43fe2172013-04-03 02:00:15 +00002769 if (die)
2770 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002771 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00002772 continue; // The containing decl contexts don't match
Greg Clayton8b4edba2015-08-14 20:02:05 +00002773
Greg Clayton43fe2172013-04-03 02:00:15 +00002774
2775 // If we get to here, the die is good, and we should add it:
Greg Clayton6071e6f2015-08-26 22:57:51 +00002776 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end() && ResolveFunction (die, include_inlines, sc_list))
Greg Clayton43fe2172013-04-03 02:00:15 +00002777 {
2778 bool keep_die = true;
2779 if ((name_type_mask & (eFunctionNameTypeBase|eFunctionNameTypeMethod)) != (eFunctionNameTypeBase|eFunctionNameTypeMethod))
2780 {
2781 // We are looking for either basenames or methods, so we need to
2782 // trim out the ones we won't want by looking at the type
2783 SymbolContext sc;
2784 if (sc_list.GetLastContext(sc))
2785 {
2786 if (sc.block)
2787 {
2788 // We have an inlined function
2789 }
2790 else if (sc.function)
2791 {
2792 Type *type = sc.function->GetType();
2793
Sean Callananc370a8a2013-09-18 22:59:55 +00002794 if (type)
Greg Clayton43fe2172013-04-03 02:00:15 +00002795 {
Greg Clayton99558cc42015-08-24 23:46:31 +00002796 CompilerDeclContext decl_ctx = GetDeclContextContainingUID (type->GetID());
2797 if (decl_ctx.IsStructUnionOrClass())
Greg Clayton43fe2172013-04-03 02:00:15 +00002798 {
Sean Callananc370a8a2013-09-18 22:59:55 +00002799 if (name_type_mask & eFunctionNameTypeBase)
2800 {
2801 sc_list.RemoveContextAtIndex(sc_list.GetSize()-1);
2802 keep_die = false;
2803 }
2804 }
2805 else
2806 {
2807 if (name_type_mask & eFunctionNameTypeMethod)
2808 {
2809 sc_list.RemoveContextAtIndex(sc_list.GetSize()-1);
2810 keep_die = false;
2811 }
Greg Clayton43fe2172013-04-03 02:00:15 +00002812 }
2813 }
2814 else
2815 {
Sean Callananc370a8a2013-09-18 22:59:55 +00002816 GetObjectFile()->GetModule()->ReportWarning ("function at die offset 0x%8.8x had no function type",
Tamas Berghammereb882fc2015-09-09 10:20:48 +00002817 die_ref.die_offset);
Greg Clayton43fe2172013-04-03 02:00:15 +00002818 }
2819 }
2820 }
2821 }
2822 if (keep_die)
Greg Clayton6071e6f2015-08-26 22:57:51 +00002823 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002824 }
2825 }
2826 else
2827 {
2828 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 +00002829 die_ref.die_offset, name_cstr);
Greg Clayton43fe2172013-04-03 02:00:15 +00002830 }
2831 }
2832 die_offsets.clear();
Jim Ingham4cda6e02011-10-07 22:23:45 +00002833 }
2834 }
Greg Clayton7f995132011-10-04 22:41:51 +00002835 }
2836 else
2837 {
2838
2839 // Index the DWARF if we haven't already
2840 if (!m_indexed)
2841 Index ();
2842
Greg Clayton7f995132011-10-04 22:41:51 +00002843 if (name_type_mask & eFunctionNameTypeFull)
Matt Kopecd6089962013-05-10 17:53:48 +00002844 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002845 FindFunctions (name, m_function_fullname_index, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002846
Ed Mastefc7baa02013-09-09 18:00:45 +00002847 // FIXME Temporary workaround for global/anonymous namespace
Robert Flack5cbd3bf2015-05-13 18:20:02 +00002848 // functions debugging FreeBSD and Linux binaries.
Matt Kopecd6089962013-05-10 17:53:48 +00002849 // If we didn't find any functions in the global namespace try
2850 // looking in the basename index but ignore any returned
Robert Flackeb83fab2015-05-15 18:59:59 +00002851 // functions that have a namespace but keep functions which
2852 // have an anonymous namespace
2853 // TODO: The arch in the object file isn't correct for MSVC
2854 // binaries on windows, we should find a way to make it
2855 // correct and handle those symbols as well.
Aidan Doddsc78e8992015-11-10 14:10:57 +00002856 if (sc_list.GetSize() == original_size)
Matt Kopecd6089962013-05-10 17:53:48 +00002857 {
Robert Flackeb83fab2015-05-15 18:59:59 +00002858 ArchSpec arch;
Greg Clayton99558cc42015-08-24 23:46:31 +00002859 if (!parent_decl_ctx &&
Robert Flackeb83fab2015-05-15 18:59:59 +00002860 GetObjectFile()->GetArchitecture(arch) &&
2861 (arch.GetTriple().isOSFreeBSD() || arch.GetTriple().isOSLinux() ||
2862 arch.GetMachine() == llvm::Triple::hexagon))
Matt Kopecd6089962013-05-10 17:53:48 +00002863 {
Robert Flackeb83fab2015-05-15 18:59:59 +00002864 SymbolContextList temp_sc_list;
2865 FindFunctions (name, m_function_basename_index, include_inlines, temp_sc_list);
Matt Kopecd6089962013-05-10 17:53:48 +00002866 SymbolContext sc;
2867 for (uint32_t i = 0; i < temp_sc_list.GetSize(); i++)
2868 {
2869 if (temp_sc_list.GetContextAtIndex(i, sc))
2870 {
Matt Kopeca189d492013-05-10 22:55:24 +00002871 ConstString mangled_name = sc.GetFunctionName(Mangled::ePreferMangled);
2872 ConstString demangled_name = sc.GetFunctionName(Mangled::ePreferDemangled);
Robert Flackeb83fab2015-05-15 18:59:59 +00002873 // Mangled names on Linux and FreeBSD are of the form:
2874 // _ZN18function_namespace13function_nameEv.
Matt Kopec04e5d582013-05-14 19:00:41 +00002875 if (strncmp(mangled_name.GetCString(), "_ZN", 3) ||
2876 !strncmp(demangled_name.GetCString(), "(anonymous namespace)", 21))
Matt Kopecd6089962013-05-10 17:53:48 +00002877 {
2878 sc_list.Append(sc);
2879 }
2880 }
2881 }
2882 }
2883 }
Matt Kopecd6089962013-05-10 17:53:48 +00002884 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002885 DIEArray die_offsets;
Greg Clayton43fe2172013-04-03 02:00:15 +00002886 if (name_type_mask & eFunctionNameTypeBase)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002887 {
Greg Clayton43fe2172013-04-03 02:00:15 +00002888 uint32_t num_base = m_function_basename_index.Find(name, die_offsets);
Greg Claytonaa044962011-10-13 00:59:38 +00002889 for (uint32_t i = 0; i < num_base; i++)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002890 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002891 DWARFDIE die = info->GetDIE (die_offsets[i]);
Greg Claytonaa044962011-10-13 00:59:38 +00002892 if (die)
2893 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002894 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00002895 continue; // The containing decl contexts don't match
Greg Clayton8b4edba2015-08-14 20:02:05 +00002896
Greg Claytonaa044962011-10-13 00:59:38 +00002897 // If we get to here, the die is good, and we should add it:
Greg Clayton6071e6f2015-08-26 22:57:51 +00002898 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002899 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002900 if (ResolveFunction (die, include_inlines, sc_list))
2901 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002902 }
Greg Claytonaa044962011-10-13 00:59:38 +00002903 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002904 }
2905 die_offsets.clear();
2906 }
2907
Greg Clayton43fe2172013-04-03 02:00:15 +00002908 if (name_type_mask & eFunctionNameTypeMethod)
Jim Ingham4cda6e02011-10-07 22:23:45 +00002909 {
Greg Clayton99558cc42015-08-24 23:46:31 +00002910 if (parent_decl_ctx && parent_decl_ctx->IsValid())
Sean Callanan213fdb82011-10-13 01:49:10 +00002911 return 0; // no methods in namespaces
2912
Greg Clayton43fe2172013-04-03 02:00:15 +00002913 uint32_t num_base = m_function_method_index.Find(name, die_offsets);
Jim Ingham4cda6e02011-10-07 22:23:45 +00002914 {
Greg Claytonaa044962011-10-13 00:59:38 +00002915 for (uint32_t i = 0; i < num_base; i++)
2916 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002917 DWARFDIE die = info->GetDIE (die_offsets[i]);
Greg Claytonaa044962011-10-13 00:59:38 +00002918 if (die)
2919 {
Greg Claytonaa044962011-10-13 00:59:38 +00002920 // If we get to here, the die is good, and we should add it:
Greg Clayton6071e6f2015-08-26 22:57:51 +00002921 if (resolved_dies.find(die.GetDIE()) == resolved_dies.end())
Greg Clayton43fe2172013-04-03 02:00:15 +00002922 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00002923 if (ResolveFunction (die, include_inlines, sc_list))
2924 resolved_dies.insert(die.GetDIE());
Greg Clayton43fe2172013-04-03 02:00:15 +00002925 }
Greg Claytonaa044962011-10-13 00:59:38 +00002926 }
2927 }
Jim Ingham4cda6e02011-10-07 22:23:45 +00002928 }
2929 die_offsets.clear();
2930 }
Greg Clayton7f995132011-10-04 22:41:51 +00002931
Greg Clayton99558cc42015-08-24 23:46:31 +00002932 if ((name_type_mask & eFunctionNameTypeSelector) && (!parent_decl_ctx || !parent_decl_ctx->IsValid()))
Jim Ingham4cda6e02011-10-07 22:23:45 +00002933 {
Pavel Labatha73d6572015-03-13 10:22:00 +00002934 FindFunctions (name, m_function_selector_index, include_inlines, sc_list);
Jim Ingham4cda6e02011-10-07 22:23:45 +00002935 }
2936
Greg Clayton4d01ace2011-09-29 16:58:15 +00002937 }
2938
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002939 // Return the number of variable that were appended to the list
Greg Clayton437a1352012-04-09 22:43:43 +00002940 const uint32_t num_matches = sc_list.GetSize() - original_size;
2941
2942 if (log && num_matches > 0)
2943 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002944 GetObjectFile()->GetModule()->LogMessage (log,
Pavel Labatha73d6572015-03-13 10:22:00 +00002945 "SymbolFileDWARF::FindFunctions (name=\"%s\", name_type_mask=0x%x, include_inlines=%d, append=%u, sc_list) => %u",
Greg Clayton437a1352012-04-09 22:43:43 +00002946 name.GetCString(),
2947 name_type_mask,
Pavel Labatha73d6572015-03-13 10:22:00 +00002948 include_inlines,
Greg Clayton437a1352012-04-09 22:43:43 +00002949 append,
2950 num_matches);
2951 }
2952 return num_matches;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002953}
2954
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002955uint32_t
Sean Callanan9df05fb2012-02-10 22:52:19 +00002956SymbolFileDWARF::FindFunctions(const RegularExpression& regex, bool include_inlines, bool append, SymbolContextList& sc_list)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002957{
2958 Timer scoped_timer (__PRETTY_FUNCTION__,
2959 "SymbolFileDWARF::FindFunctions (regex = '%s')",
2960 regex.GetText());
2961
Greg Clayton5160ce52013-03-27 23:08:40 +00002962 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00002963
2964 if (log)
2965 {
Greg Clayton5160ce52013-03-27 23:08:40 +00002966 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00002967 "SymbolFileDWARF::FindFunctions (regex=\"%s\", append=%u, sc_list)",
2968 regex.GetText(),
2969 append);
Greg Clayton21f2a492011-10-06 00:09:08 +00002970 }
2971
2972
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002973 // If we aren't appending the results to this list, then clear the list
2974 if (!append)
2975 sc_list.Clear();
2976
2977 // Remember how many sc_list are in the list before we search in case
2978 // we are appending the results to a variable list.
2979 uint32_t original_size = sc_list.GetSize();
2980
Greg Clayton97fbc342011-10-20 22:30:33 +00002981 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00002982 {
Greg Clayton97fbc342011-10-20 22:30:33 +00002983 if (m_apple_names_ap.get())
Pavel Labatha73d6572015-03-13 10:22:00 +00002984 FindFunctions (regex, *m_apple_names_ap, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002985 }
2986 else
2987 {
Jim Ingham4cda6e02011-10-07 22:23:45 +00002988 // Index the DWARF if we haven't already
Greg Clayton7f995132011-10-04 22:41:51 +00002989 if (!m_indexed)
2990 Index ();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002991
Pavel Labatha73d6572015-03-13 10:22:00 +00002992 FindFunctions (regex, m_function_basename_index, include_inlines, sc_list);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002993
Pavel Labatha73d6572015-03-13 10:22:00 +00002994 FindFunctions (regex, m_function_fullname_index, include_inlines, sc_list);
Greg Clayton7f995132011-10-04 22:41:51 +00002995 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00002996
2997 // Return the number of variable that were appended to the list
2998 return sc_list.GetSize() - original_size;
2999}
Jim Ingham318c9f22011-08-26 19:44:13 +00003000
Siva Chandra9293fc42016-01-07 23:32:34 +00003001void
3002SymbolFileDWARF::GetMangledNamesForFunction (const std::string &scope_qualified_name,
3003 std::vector<ConstString> &mangled_names)
3004{
3005 DWARFDebugInfo* info = DebugInfo();
3006 uint32_t num_comp_units = 0;
3007 if (info)
3008 num_comp_units = info->GetNumCompileUnits();
3009
3010 for (uint32_t i = 0; i < num_comp_units; i++)
3011 {
3012 DWARFCompileUnit *cu = info->GetCompileUnitAtIndex(i);
3013 if (cu == nullptr)
3014 continue;
3015
3016 SymbolFileDWARFDwo *dwo = cu->GetDwoSymbolFile();
3017 if (dwo)
3018 dwo->GetMangledNamesForFunction(scope_qualified_name, mangled_names);
3019 }
3020
3021 NameToOffsetMap::iterator iter = m_function_scope_qualified_name_map.find(scope_qualified_name);
3022 if (iter == m_function_scope_qualified_name_map.end())
3023 return;
3024
3025 DIERefSetSP set_sp = (*iter).second;
3026 std::set<DIERef>::iterator set_iter;
3027 for (set_iter = set_sp->begin(); set_iter != set_sp->end(); set_iter++)
3028 {
3029 DWARFDIE die = DebugInfo()->GetDIE (*set_iter);
3030 mangled_names.push_back(ConstString(die.GetMangledName()));
3031 }
3032}
3033
3034
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003035uint32_t
Greg Claytond1767f02011-12-08 02:13:16 +00003036SymbolFileDWARF::FindTypes (const SymbolContext& sc,
3037 const ConstString &name,
Greg Clayton99558cc42015-08-24 23:46:31 +00003038 const CompilerDeclContext *parent_decl_ctx,
Greg Claytond1767f02011-12-08 02:13:16 +00003039 bool append,
Greg Claytonae088e52016-02-10 21:28:13 +00003040 uint32_t max_matches,
3041 llvm::DenseSet<lldb_private::SymbolFile *> &searched_symbol_files,
Ravitheja Addepally40697302015-10-08 09:45:41 +00003042 TypeMap& types)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003043{
Greg Claytonae088e52016-02-10 21:28:13 +00003044 // If we aren't appending the results to this list, then clear the list
3045 if (!append)
3046 types.Clear();
3047
3048 // Make sure we haven't already searched this SymbolFile before...
3049 if (searched_symbol_files.count(this))
3050 return 0;
3051 else
3052 searched_symbol_files.insert(this);
3053
Greg Claytonc685f8e2010-09-15 04:15:46 +00003054 DWARFDebugInfo* info = DebugInfo();
3055 if (info == NULL)
3056 return 0;
3057
Greg Clayton5160ce52013-03-27 23:08:40 +00003058 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003059
Greg Clayton21f2a492011-10-06 00:09:08 +00003060 if (log)
3061 {
Greg Clayton99558cc42015-08-24 23:46:31 +00003062 if (parent_decl_ctx)
Greg Clayton5160ce52013-03-27 23:08:40 +00003063 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003064 "SymbolFileDWARF::FindTypes (sc, name=\"%s\", parent_decl_ctx = %p (\"%s\"), append=%u, max_matches=%u, type_list)",
Greg Clayton437a1352012-04-09 22:43:43 +00003065 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00003066 static_cast<const void*>(parent_decl_ctx),
3067 parent_decl_ctx->GetName().AsCString("<NULL>"),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003068 append, max_matches);
Greg Clayton437a1352012-04-09 22:43:43 +00003069 else
Greg Clayton5160ce52013-03-27 23:08:40 +00003070 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003071 "SymbolFileDWARF::FindTypes (sc, name=\"%s\", parent_decl_ctx = NULL, append=%u, max_matches=%u, type_list)",
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003072 name.GetCString(), append,
Greg Clayton437a1352012-04-09 22:43:43 +00003073 max_matches);
Greg Clayton21f2a492011-10-06 00:09:08 +00003074 }
3075
Greg Clayton99558cc42015-08-24 23:46:31 +00003076 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
Ed Maste4c24b122013-10-17 20:13:14 +00003077 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003078
Greg Claytond4a2b372011-09-12 23:21:58 +00003079 DIEArray die_offsets;
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003080
Greg Clayton97fbc342011-10-20 22:30:33 +00003081 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00003082 {
Greg Clayton97fbc342011-10-20 22:30:33 +00003083 if (m_apple_types_ap.get())
3084 {
3085 const char *name_cstr = name.GetCString();
3086 m_apple_types_ap->FindByName (name_cstr, die_offsets);
3087 }
Greg Clayton7f995132011-10-04 22:41:51 +00003088 }
3089 else
3090 {
3091 if (!m_indexed)
3092 Index ();
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003093
Greg Clayton7f995132011-10-04 22:41:51 +00003094 m_type_index.Find (name, die_offsets);
3095 }
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003096
Greg Clayton437a1352012-04-09 22:43:43 +00003097 const size_t num_die_matches = die_offsets.size();
Greg Clayton7f995132011-10-04 22:41:51 +00003098
Greg Clayton437a1352012-04-09 22:43:43 +00003099 if (num_die_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003100 {
Greg Clayton7f995132011-10-04 22:41:51 +00003101 const uint32_t initial_types_size = types.GetSize();
Greg Clayton437a1352012-04-09 22:43:43 +00003102 for (size_t i=0; i<num_die_matches; ++i)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003103 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003104 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00003105 DWARFDIE die = GetDIE (die_ref);
Greg Claytond4a2b372011-09-12 23:21:58 +00003106
Greg Clayton95d87902011-11-11 03:16:25 +00003107 if (die)
Greg Clayton73bf5db2011-06-17 01:22:15 +00003108 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003109 if (!DIEInDeclContext(parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00003110 continue; // The containing decl contexts don't match
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003111
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003112 Type *matching_type = ResolveType (die, true, true);
Greg Clayton95d87902011-11-11 03:16:25 +00003113 if (matching_type)
3114 {
3115 // We found a type pointer, now find the shared pointer form our type list
Greg Claytone1cd1be2012-01-29 20:56:30 +00003116 types.InsertUnique (matching_type->shared_from_this());
Greg Clayton95d87902011-11-11 03:16:25 +00003117 if (types.GetSize() >= max_matches)
3118 break;
3119 }
Greg Clayton73bf5db2011-06-17 01:22:15 +00003120 }
Greg Clayton95d87902011-11-11 03:16:25 +00003121 else
3122 {
3123 if (m_using_apple_tables)
3124 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00003125 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 +00003126 die_ref.die_offset, name.GetCString());
Greg Clayton95d87902011-11-11 03:16:25 +00003127 }
3128 }
3129
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003130 }
Greg Clayton437a1352012-04-09 22:43:43 +00003131 const uint32_t num_matches = types.GetSize() - initial_types_size;
3132 if (log && num_matches)
3133 {
Greg Clayton99558cc42015-08-24 23:46:31 +00003134 if (parent_decl_ctx)
Greg Clayton437a1352012-04-09 22:43:43 +00003135 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003136 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003137 "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 +00003138 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00003139 static_cast<const void*>(parent_decl_ctx),
3140 parent_decl_ctx->GetName().AsCString("<NULL>"),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003141 append, max_matches,
Greg Clayton437a1352012-04-09 22:43:43 +00003142 num_matches);
3143 }
3144 else
3145 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003146 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003147 "SymbolFileDWARF::FindTypes (sc, name=\"%s\", parent_decl_ctx = NULL, append=%u, max_matches=%u, type_list) => %u",
Greg Clayton437a1352012-04-09 22:43:43 +00003148 name.GetCString(),
Saleem Abdulrasool324a1032014-04-04 04:06:10 +00003149 append, max_matches,
Greg Clayton437a1352012-04-09 22:43:43 +00003150 num_matches);
3151 }
3152 }
3153 return num_matches;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003154 }
Greg Claytone6b36cd2015-12-08 01:02:08 +00003155 else
3156 {
3157 UpdateExternalModuleListIfNeeded();
3158
3159 for (const auto &pair : m_external_type_modules)
3160 {
3161 ModuleSP external_module_sp = pair.second;
3162 if (external_module_sp)
3163 {
3164 SymbolVendor *sym_vendor = external_module_sp->GetSymbolVendor();
3165 if (sym_vendor)
3166 {
3167 const uint32_t num_external_matches = sym_vendor->FindTypes (sc,
3168 name,
3169 parent_decl_ctx,
3170 append,
3171 max_matches,
Greg Claytonae088e52016-02-10 21:28:13 +00003172 searched_symbol_files,
Greg Claytone6b36cd2015-12-08 01:02:08 +00003173 types);
3174 if (num_external_matches)
3175 return num_external_matches;
3176 }
3177 }
3178 }
3179 }
3180
3181 return 0;
3182}
3183
3184
3185size_t
3186SymbolFileDWARF::FindTypes (const std::vector<CompilerContext> &context,
3187 bool append,
3188 TypeMap& types)
3189{
3190 if (!append)
3191 types.Clear();
3192
3193 if (context.empty())
3194 return 0;
3195
3196 DIEArray die_offsets;
3197
3198 ConstString name = context.back().name;
3199
Sean Callanan9b0cfe22016-01-14 18:59:49 +00003200 if (!name)
3201 return 0;
3202
Greg Claytone6b36cd2015-12-08 01:02:08 +00003203 if (m_using_apple_tables)
3204 {
3205 if (m_apple_types_ap.get())
3206 {
3207 const char *name_cstr = name.GetCString();
3208 m_apple_types_ap->FindByName (name_cstr, die_offsets);
3209 }
3210 }
3211 else
3212 {
3213 if (!m_indexed)
3214 Index ();
3215
3216 m_type_index.Find (name, die_offsets);
3217 }
3218
3219 const size_t num_die_matches = die_offsets.size();
3220
3221 if (num_die_matches)
3222 {
3223 size_t num_matches = 0;
Greg Claytone6b36cd2015-12-08 01:02:08 +00003224 for (size_t i=0; i<num_die_matches; ++i)
3225 {
3226 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00003227 DWARFDIE die = GetDIE (die_ref);
Greg Claytone6b36cd2015-12-08 01:02:08 +00003228
3229 if (die)
3230 {
3231 std::vector<CompilerContext> die_context;
3232 die.GetDWOContext(die_context);
3233 if (die_context != context)
3234 continue;
3235
3236 Type *matching_type = ResolveType (die, true, true);
3237 if (matching_type)
3238 {
3239 // We found a type pointer, now find the shared pointer form our type list
3240 types.InsertUnique (matching_type->shared_from_this());
3241 ++num_matches;
3242 }
3243 }
3244 else
3245 {
3246 if (m_using_apple_tables)
3247 {
3248 GetObjectFile()->GetModule()->ReportErrorIfModifyDetected ("the DWARF debug information has been modified (.apple_types accelerator table had bad die 0x%8.8x for '%s')\n",
3249 die_ref.die_offset, name.GetCString());
3250 }
3251 }
3252
3253 }
3254 return num_matches;
3255 }
Greg Clayton7f995132011-10-04 22:41:51 +00003256 return 0;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003257}
3258
3259
Greg Clayton99558cc42015-08-24 23:46:31 +00003260CompilerDeclContext
Greg Clayton96d7d742010-11-10 23:42:09 +00003261SymbolFileDWARF::FindNamespace (const SymbolContext& sc,
Sean Callanan213fdb82011-10-13 01:49:10 +00003262 const ConstString &name,
Greg Clayton99558cc42015-08-24 23:46:31 +00003263 const CompilerDeclContext *parent_decl_ctx)
Greg Clayton96d7d742010-11-10 23:42:09 +00003264{
Greg Clayton5160ce52013-03-27 23:08:40 +00003265 Log *log (LogChannelDWARF::GetLogIfAll(DWARF_LOG_LOOKUPS));
Greg Clayton21f2a492011-10-06 00:09:08 +00003266
3267 if (log)
3268 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003269 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytone38a5ed2012-01-05 03:57:59 +00003270 "SymbolFileDWARF::FindNamespace (sc, name=\"%s\")",
3271 name.GetCString());
Greg Clayton21f2a492011-10-06 00:09:08 +00003272 }
3273
Greg Clayton99558cc42015-08-24 23:46:31 +00003274 CompilerDeclContext namespace_decl_ctx;
3275
3276 if (!DeclContextMatchesThisSymbolFile(parent_decl_ctx))
3277 return namespace_decl_ctx;
3278
3279
Greg Clayton96d7d742010-11-10 23:42:09 +00003280 DWARFDebugInfo* info = DebugInfo();
Greg Clayton526e5af2010-11-13 03:52:47 +00003281 if (info)
Greg Clayton96d7d742010-11-10 23:42:09 +00003282 {
Greg Clayton7f995132011-10-04 22:41:51 +00003283 DIEArray die_offsets;
3284
Greg Clayton526e5af2010-11-13 03:52:47 +00003285 // Index if we already haven't to make sure the compile units
3286 // get indexed and make their global DIE index list
Greg Clayton97fbc342011-10-20 22:30:33 +00003287 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00003288 {
Greg Clayton97fbc342011-10-20 22:30:33 +00003289 if (m_apple_namespaces_ap.get())
3290 {
3291 const char *name_cstr = name.GetCString();
3292 m_apple_namespaces_ap->FindByName (name_cstr, die_offsets);
3293 }
Greg Clayton7f995132011-10-04 22:41:51 +00003294 }
3295 else
3296 {
3297 if (!m_indexed)
3298 Index ();
Greg Clayton96d7d742010-11-10 23:42:09 +00003299
Greg Clayton7f995132011-10-04 22:41:51 +00003300 m_namespace_index.Find (name, die_offsets);
3301 }
Greg Claytond4a2b372011-09-12 23:21:58 +00003302
Greg Clayton7f995132011-10-04 22:41:51 +00003303 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00003304 if (num_matches)
Greg Clayton526e5af2010-11-13 03:52:47 +00003305 {
Greg Claytond4a2b372011-09-12 23:21:58 +00003306 for (size_t i=0; i<num_matches; ++i)
Greg Clayton526e5af2010-11-13 03:52:47 +00003307 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003308 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00003309 DWARFDIE die = GetDIE (die_ref);
Greg Clayton6071e6f2015-08-26 22:57:51 +00003310
Greg Clayton95d87902011-11-11 03:16:25 +00003311 if (die)
Greg Claytond4a2b372011-09-12 23:21:58 +00003312 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003313 if (!DIEInDeclContext (parent_decl_ctx, die))
Greg Clayton99558cc42015-08-24 23:46:31 +00003314 continue; // The containing decl contexts don't match
Greg Clayton95d87902011-11-11 03:16:25 +00003315
Greg Clayton261ac3f2015-08-28 01:01:03 +00003316 DWARFASTParser *dwarf_ast = die.GetDWARFParser();
3317 if (dwarf_ast)
Greg Clayton8b4edba2015-08-14 20:02:05 +00003318 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00003319 namespace_decl_ctx = dwarf_ast->GetDeclContextForUIDFromDWARF (die);
Greg Clayton99558cc42015-08-24 23:46:31 +00003320 if (namespace_decl_ctx)
Greg Clayton8b4edba2015-08-14 20:02:05 +00003321 break;
Greg Clayton95d87902011-11-11 03:16:25 +00003322 }
Greg Claytond4a2b372011-09-12 23:21:58 +00003323 }
Greg Clayton95d87902011-11-11 03:16:25 +00003324 else
3325 {
3326 if (m_using_apple_tables)
3327 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00003328 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 +00003329 die_ref.die_offset, name.GetCString());
Greg Clayton95d87902011-11-11 03:16:25 +00003330 }
3331 }
3332
Greg Clayton526e5af2010-11-13 03:52:47 +00003333 }
3334 }
Greg Clayton96d7d742010-11-10 23:42:09 +00003335 }
Greg Clayton99558cc42015-08-24 23:46:31 +00003336 if (log && namespace_decl_ctx)
Greg Clayton437a1352012-04-09 22:43:43 +00003337 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003338 GetObjectFile()->GetModule()->LogMessage (log,
Greg Clayton99558cc42015-08-24 23:46:31 +00003339 "SymbolFileDWARF::FindNamespace (sc, name=\"%s\") => CompilerDeclContext(%p/%p) \"%s\"",
Greg Clayton437a1352012-04-09 22:43:43 +00003340 name.GetCString(),
Greg Clayton99558cc42015-08-24 23:46:31 +00003341 static_cast<const void*>(namespace_decl_ctx.GetTypeSystem()),
3342 static_cast<const void*>(namespace_decl_ctx.GetOpaqueDeclContext()),
3343 namespace_decl_ctx.GetName().AsCString("<NULL>"));
Greg Clayton437a1352012-04-09 22:43:43 +00003344 }
3345
Greg Clayton99558cc42015-08-24 23:46:31 +00003346 return namespace_decl_ctx;
Greg Clayton96d7d742010-11-10 23:42:09 +00003347}
3348
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003349TypeSP
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003350SymbolFileDWARF::GetTypeForDIE (const DWARFDIE &die, bool resolve_function_context)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003351{
3352 TypeSP type_sp;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003353 if (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003354 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003355 Type *type_ptr = GetDIEToType().lookup (die.GetDIE());
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003356 if (type_ptr == NULL)
3357 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003358 CompileUnit* lldb_cu = GetCompUnitForDWARFCompUnit(die.GetCU());
Greg Claytonca512b32011-01-14 04:54:56 +00003359 assert (lldb_cu);
3360 SymbolContext sc(lldb_cu);
Ravitheja Addepally40697302015-10-08 09:45:41 +00003361 const DWARFDebugInfoEntry* parent_die = die.GetParent().GetDIE();
3362 while (parent_die != nullptr)
3363 {
3364 if (parent_die->Tag() == DW_TAG_subprogram)
3365 break;
3366 parent_die = parent_die->GetParent();
3367 }
3368 SymbolContext sc_backup = sc;
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003369 if (resolve_function_context && parent_die != nullptr && !GetFunction(DWARFDIE(die.GetCU(),parent_die), sc))
Ravitheja Addepally40697302015-10-08 09:45:41 +00003370 sc = sc_backup;
3371
Greg Clayton6071e6f2015-08-26 22:57:51 +00003372 type_sp = ParseType(sc, die, NULL);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003373 }
3374 else if (type_ptr != DIE_IS_BEING_PARSED)
3375 {
3376 // Grab the existing type from the master types lists
Greg Claytone1cd1be2012-01-29 20:56:30 +00003377 type_sp = type_ptr->shared_from_this();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003378 }
3379
3380 }
3381 return type_sp;
3382}
3383
Greg Clayton2bc22f82011-09-30 03:20:47 +00003384
Greg Clayton6071e6f2015-08-26 22:57:51 +00003385DWARFDIE
3386SymbolFileDWARF::GetDeclContextDIEContainingDIE (const DWARFDIE &orig_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003387{
Greg Clayton6071e6f2015-08-26 22:57:51 +00003388 if (orig_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003389 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003390 DWARFDIE die = orig_die;
Greg Clayton2bc22f82011-09-30 03:20:47 +00003391
Greg Clayton6071e6f2015-08-26 22:57:51 +00003392 while (die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003393 {
3394 // If this is the original DIE that we are searching for a declaration
3395 // for, then don't look in the cache as we don't want our own decl
3396 // context to be our decl context...
Greg Clayton6071e6f2015-08-26 22:57:51 +00003397 if (orig_die != die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003398 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003399 switch (die.Tag())
Greg Clayton2bc22f82011-09-30 03:20:47 +00003400 {
3401 case DW_TAG_compile_unit:
3402 case DW_TAG_namespace:
3403 case DW_TAG_structure_type:
3404 case DW_TAG_union_type:
3405 case DW_TAG_class_type:
Paul Hermand628cbb2015-09-15 23:44:17 +00003406 case DW_TAG_lexical_block:
3407 case DW_TAG_subprogram:
Greg Clayton2bc22f82011-09-30 03:20:47 +00003408 return die;
3409
3410 default:
3411 break;
3412 }
3413 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003414
3415 DWARFDIE spec_die = die.GetReferencedDIE(DW_AT_specification);
3416 if (spec_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003417 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003418 DWARFDIE decl_ctx_die = GetDeclContextDIEContainingDIE(spec_die);
3419 if (decl_ctx_die)
3420 return decl_ctx_die;
Greg Clayton2bc22f82011-09-30 03:20:47 +00003421 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003422
3423 DWARFDIE abs_die = die.GetReferencedDIE(DW_AT_abstract_origin);
3424 if (abs_die)
Greg Clayton2bc22f82011-09-30 03:20:47 +00003425 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003426 DWARFDIE decl_ctx_die = GetDeclContextDIEContainingDIE(abs_die);
3427 if (decl_ctx_die)
3428 return decl_ctx_die;
Greg Clayton2bc22f82011-09-30 03:20:47 +00003429 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003430
3431 die = die.GetParent();
Greg Clayton2bc22f82011-09-30 03:20:47 +00003432 }
3433 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003434 return DWARFDIE();
Greg Clayton2bc22f82011-09-30 03:20:47 +00003435}
3436
3437
Greg Clayton901c5ca2011-12-03 04:40:03 +00003438Symbol *
3439SymbolFileDWARF::GetObjCClassSymbol (const ConstString &objc_class_name)
3440{
3441 Symbol *objc_class_symbol = NULL;
3442 if (m_obj_file)
3443 {
Greg Clayton3046e662013-07-10 01:23:25 +00003444 Symtab *symtab = m_obj_file->GetSymtab ();
Greg Clayton901c5ca2011-12-03 04:40:03 +00003445 if (symtab)
3446 {
3447 objc_class_symbol = symtab->FindFirstSymbolWithNameAndType (objc_class_name,
3448 eSymbolTypeObjCClass,
3449 Symtab::eDebugNo,
3450 Symtab::eVisibilityAny);
3451 }
3452 }
3453 return objc_class_symbol;
3454}
3455
Greg Claytonc7f03b62012-01-12 04:33:28 +00003456// Some compilers don't emit the DW_AT_APPLE_objc_complete_type attribute. If they don't
3457// then we can end up looking through all class types for a complete type and never find
3458// the full definition. We need to know if this attribute is supported, so we determine
3459// this here and cache th result. We also need to worry about the debug map DWARF file
3460// if we are doing darwin DWARF in .o file debugging.
3461bool
3462SymbolFileDWARF::Supports_DW_AT_APPLE_objc_complete_type (DWARFCompileUnit *cu)
3463{
3464 if (m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolCalculate)
3465 {
3466 m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolNo;
3467 if (cu && cu->Supports_DW_AT_APPLE_objc_complete_type())
3468 m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolYes;
3469 else
3470 {
3471 DWARFDebugInfo* debug_info = DebugInfo();
3472 const uint32_t num_compile_units = GetNumCompileUnits();
3473 for (uint32_t cu_idx = 0; cu_idx < num_compile_units; ++cu_idx)
3474 {
Greg Clayton53eb1c22012-04-02 22:59:12 +00003475 DWARFCompileUnit* dwarf_cu = debug_info->GetCompileUnitAtIndex(cu_idx);
3476 if (dwarf_cu != cu && dwarf_cu->Supports_DW_AT_APPLE_objc_complete_type())
Greg Claytonc7f03b62012-01-12 04:33:28 +00003477 {
3478 m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolYes;
3479 break;
3480 }
3481 }
3482 }
Greg Clayton1f746072012-08-29 21:13:06 +00003483 if (m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolNo && GetDebugMapSymfile ())
Greg Claytonc7f03b62012-01-12 04:33:28 +00003484 return m_debug_map_symfile->Supports_DW_AT_APPLE_objc_complete_type (this);
3485 }
3486 return m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolYes;
3487}
Greg Clayton901c5ca2011-12-03 04:40:03 +00003488
3489// This function can be used when a DIE is found that is a forward declaration
3490// DIE and we want to try and find a type that has the complete definition.
3491TypeSP
Greg Clayton6071e6f2015-08-26 22:57:51 +00003492SymbolFileDWARF::FindCompleteObjCDefinitionTypeForDIE (const DWARFDIE &die,
Greg Claytonc7f03b62012-01-12 04:33:28 +00003493 const ConstString &type_name,
3494 bool must_be_implementation)
Greg Clayton901c5ca2011-12-03 04:40:03 +00003495{
3496
3497 TypeSP type_sp;
3498
Greg Claytonc7f03b62012-01-12 04:33:28 +00003499 if (!type_name || (must_be_implementation && !GetObjCClassSymbol (type_name)))
Greg Clayton901c5ca2011-12-03 04:40:03 +00003500 return type_sp;
3501
3502 DIEArray die_offsets;
3503
3504 if (m_using_apple_tables)
3505 {
3506 if (m_apple_types_ap.get())
3507 {
3508 const char *name_cstr = type_name.GetCString();
Greg Clayton68221ec2012-01-18 20:58:12 +00003509 m_apple_types_ap->FindCompleteObjCClassByName (name_cstr, die_offsets, must_be_implementation);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003510 }
3511 }
3512 else
3513 {
3514 if (!m_indexed)
3515 Index ();
3516
3517 m_type_index.Find (type_name, die_offsets);
3518 }
3519
Greg Clayton901c5ca2011-12-03 04:40:03 +00003520 const size_t num_matches = die_offsets.size();
3521
Greg Clayton901c5ca2011-12-03 04:40:03 +00003522 if (num_matches)
3523 {
Greg Clayton901c5ca2011-12-03 04:40:03 +00003524 for (size_t i=0; i<num_matches; ++i)
3525 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003526 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00003527 DWARFDIE type_die = GetDIE (die_ref);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003528
3529 if (type_die)
3530 {
3531 bool try_resolving_type = false;
3532
3533 // Don't try and resolve the DIE we are looking for with the DIE itself!
3534 if (type_die != die)
3535 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003536 switch (type_die.Tag())
Greg Clayton901c5ca2011-12-03 04:40:03 +00003537 {
Greg Claytonc7f03b62012-01-12 04:33:28 +00003538 case DW_TAG_class_type:
3539 case DW_TAG_structure_type:
3540 try_resolving_type = true;
3541 break;
3542 default:
3543 break;
Greg Clayton901c5ca2011-12-03 04:40:03 +00003544 }
3545 }
3546
3547 if (try_resolving_type)
3548 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003549 if (must_be_implementation && type_die.Supports_DW_AT_APPLE_objc_complete_type())
3550 try_resolving_type = type_die.GetAttributeValueAsUnsigned (DW_AT_APPLE_objc_complete_type, 0);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003551
3552 if (try_resolving_type)
3553 {
Ravitheja Addepally46bcbaa2015-11-03 14:24:24 +00003554 Type *resolved_type = ResolveType (type_die, false, true);
Greg Clayton901c5ca2011-12-03 04:40:03 +00003555 if (resolved_type && resolved_type != DIE_IS_BEING_PARSED)
3556 {
Ed Mastea0191d12013-10-17 20:42:56 +00003557 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 +00003558 die.GetID(),
Jim Ingham4af59612014-12-19 19:20:44 +00003559 m_obj_file->GetFileSpec().GetFilename().AsCString("<Unknown>"),
Greg Clayton6071e6f2015-08-26 22:57:51 +00003560 type_die.GetID(),
3561 type_cu->GetID());
Greg Clayton901c5ca2011-12-03 04:40:03 +00003562
Greg Claytonc7f03b62012-01-12 04:33:28 +00003563 if (die)
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003564 GetDIEToType()[die.GetDIE()] = resolved_type;
Greg Claytone1cd1be2012-01-29 20:56:30 +00003565 type_sp = resolved_type->shared_from_this();
Greg Clayton901c5ca2011-12-03 04:40:03 +00003566 break;
3567 }
3568 }
3569 }
3570 }
3571 else
3572 {
3573 if (m_using_apple_tables)
3574 {
Greg Claytone38a5ed2012-01-05 03:57:59 +00003575 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 +00003576 die_ref.die_offset, type_name.GetCString());
Greg Clayton901c5ca2011-12-03 04:40:03 +00003577 }
3578 }
3579
3580 }
3581 }
3582 return type_sp;
3583}
3584
Greg Claytona8022fa2012-04-24 21:22:41 +00003585
Greg Clayton80c26302012-02-05 06:12:47 +00003586//----------------------------------------------------------------------
3587// This function helps to ensure that the declaration contexts match for
3588// two different DIEs. Often times debug information will refer to a
3589// forward declaration of a type (the equivalent of "struct my_struct;".
3590// There will often be a declaration of that type elsewhere that has the
3591// full definition. When we go looking for the full type "my_struct", we
3592// will find one or more matches in the accelerator tables and we will
3593// then need to make sure the type was in the same declaration context
3594// as the original DIE. This function can efficiently compare two DIEs
3595// and will return true when the declaration context matches, and false
3596// when they don't.
3597//----------------------------------------------------------------------
Greg Clayton890ff562012-02-02 05:48:16 +00003598bool
Greg Clayton6071e6f2015-08-26 22:57:51 +00003599SymbolFileDWARF::DIEDeclContextsMatch (const DWARFDIE &die1,
3600 const DWARFDIE &die2)
Greg Clayton890ff562012-02-02 05:48:16 +00003601{
Greg Claytona8022fa2012-04-24 21:22:41 +00003602 if (die1 == die2)
3603 return true;
3604
Greg Clayton890ff562012-02-02 05:48:16 +00003605 DWARFDIECollection decl_ctx_1;
3606 DWARFDIECollection decl_ctx_2;
Greg Clayton80c26302012-02-05 06:12:47 +00003607 //The declaration DIE stack is a stack of the declaration context
3608 // DIEs all the way back to the compile unit. If a type "T" is
3609 // declared inside a class "B", and class "B" is declared inside
3610 // a class "A" and class "A" is in a namespace "lldb", and the
3611 // namespace is in a compile unit, there will be a stack of DIEs:
3612 //
3613 // [0] DW_TAG_class_type for "B"
3614 // [1] DW_TAG_class_type for "A"
3615 // [2] DW_TAG_namespace for "lldb"
3616 // [3] DW_TAG_compile_unit for the source file.
3617 //
3618 // We grab both contexts and make sure that everything matches
3619 // all the way back to the compiler unit.
3620
3621 // First lets grab the decl contexts for both DIEs
Greg Clayton6071e6f2015-08-26 22:57:51 +00003622 die1.GetDeclContextDIEs (decl_ctx_1);
3623 die2.GetDeclContextDIEs (decl_ctx_2);
Greg Clayton80c26302012-02-05 06:12:47 +00003624 // Make sure the context arrays have the same size, otherwise
3625 // we are done
Greg Clayton890ff562012-02-02 05:48:16 +00003626 const size_t count1 = decl_ctx_1.Size();
3627 const size_t count2 = decl_ctx_2.Size();
3628 if (count1 != count2)
3629 return false;
Greg Clayton80c26302012-02-05 06:12:47 +00003630
Bruce Mitchenerd93c4a32014-07-01 21:22:11 +00003631 // Make sure the DW_TAG values match all the way back up the
Greg Clayton80c26302012-02-05 06:12:47 +00003632 // compile unit. If they don't, then we are done.
Greg Clayton6071e6f2015-08-26 22:57:51 +00003633 DWARFDIE decl_ctx_die1;
3634 DWARFDIE decl_ctx_die2;
Greg Clayton890ff562012-02-02 05:48:16 +00003635 size_t i;
3636 for (i=0; i<count1; i++)
3637 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003638 decl_ctx_die1 = decl_ctx_1.GetDIEAtIndex (i);
3639 decl_ctx_die2 = decl_ctx_2.GetDIEAtIndex (i);
3640 if (decl_ctx_die1.Tag() != decl_ctx_die2.Tag())
Greg Clayton890ff562012-02-02 05:48:16 +00003641 return false;
3642 }
Greg Clayton890ff562012-02-02 05:48:16 +00003643#if defined LLDB_CONFIGURATION_DEBUG
Greg Clayton80c26302012-02-05 06:12:47 +00003644
3645 // Make sure the top item in the decl context die array is always
3646 // DW_TAG_compile_unit. If it isn't then something went wrong in
Greg Clayton5ce1a842015-08-27 18:09:44 +00003647 // the DWARFDIE::GetDeclContextDIEs() function...
Greg Clayton6071e6f2015-08-26 22:57:51 +00003648 assert (decl_ctx_1.GetDIEAtIndex (count1 - 1).Tag() == DW_TAG_compile_unit);
Greg Clayton80c26302012-02-05 06:12:47 +00003649
Greg Clayton890ff562012-02-02 05:48:16 +00003650#endif
3651 // Always skip the compile unit when comparing by only iterating up to
Greg Clayton80c26302012-02-05 06:12:47 +00003652 // "count - 1". Here we compare the names as we go.
Greg Clayton890ff562012-02-02 05:48:16 +00003653 for (i=0; i<count1 - 1; i++)
3654 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003655 decl_ctx_die1 = decl_ctx_1.GetDIEAtIndex (i);
3656 decl_ctx_die2 = decl_ctx_2.GetDIEAtIndex (i);
3657 const char *name1 = decl_ctx_die1.GetName();
3658 const char *name2 = decl_ctx_die2.GetName();
Greg Clayton890ff562012-02-02 05:48:16 +00003659 // If the string was from a DW_FORM_strp, then the pointer will often
3660 // be the same!
Greg Clayton5569e642012-02-06 01:44:54 +00003661 if (name1 == name2)
3662 continue;
3663
3664 // Name pointers are not equal, so only compare the strings
3665 // if both are not NULL.
3666 if (name1 && name2)
Greg Clayton890ff562012-02-02 05:48:16 +00003667 {
Greg Clayton5569e642012-02-06 01:44:54 +00003668 // If the strings don't compare, we are done...
3669 if (strcmp(name1, name2) != 0)
Greg Clayton890ff562012-02-02 05:48:16 +00003670 return false;
Greg Clayton5569e642012-02-06 01:44:54 +00003671 }
3672 else
3673 {
3674 // One name was NULL while the other wasn't
3675 return false;
Greg Clayton890ff562012-02-02 05:48:16 +00003676 }
3677 }
Greg Clayton80c26302012-02-05 06:12:47 +00003678 // We made it through all of the checks and the declaration contexts
3679 // are equal.
Greg Clayton890ff562012-02-02 05:48:16 +00003680 return true;
3681}
Greg Clayton220a0072011-12-09 08:48:30 +00003682
Greg Clayton2ccf8cf2010-11-07 21:02:03 +00003683
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003684TypeSP
Greg Claytona8022fa2012-04-24 21:22:41 +00003685SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext (const DWARFDeclContext &dwarf_decl_ctx)
3686{
3687 TypeSP type_sp;
3688
3689 const uint32_t dwarf_decl_ctx_count = dwarf_decl_ctx.GetSize();
3690 if (dwarf_decl_ctx_count > 0)
3691 {
3692 const ConstString type_name(dwarf_decl_ctx[0].name);
3693 const dw_tag_t tag = dwarf_decl_ctx[0].tag;
3694
3695 if (type_name)
3696 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003697 Log *log (LogChannelDWARF::GetLogIfAny(DWARF_LOG_TYPE_COMPLETION|DWARF_LOG_LOOKUPS));
Greg Claytona8022fa2012-04-24 21:22:41 +00003698 if (log)
3699 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003700 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytona8022fa2012-04-24 21:22:41 +00003701 "SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext(tag=%s, qualified-name='%s')",
3702 DW_TAG_value_to_name(dwarf_decl_ctx[0].tag),
3703 dwarf_decl_ctx.GetQualifiedName());
3704 }
3705
3706 DIEArray die_offsets;
3707
3708 if (m_using_apple_tables)
3709 {
3710 if (m_apple_types_ap.get())
3711 {
Greg Claytoncb9c8cf2013-02-06 23:56:13 +00003712 const bool has_tag = m_apple_types_ap->GetHeader().header_data.ContainsAtom (DWARFMappedHash::eAtomTypeTag);
3713 const bool has_qualified_name_hash = m_apple_types_ap->GetHeader().header_data.ContainsAtom (DWARFMappedHash::eAtomTypeQualNameHash);
3714 if (has_tag && has_qualified_name_hash)
Greg Claytona8022fa2012-04-24 21:22:41 +00003715 {
Greg Claytoncb9c8cf2013-02-06 23:56:13 +00003716 const char *qualified_name = dwarf_decl_ctx.GetQualifiedName();
3717 const uint32_t qualified_name_hash = MappedHash::HashStringUsingDJB (qualified_name);
3718 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00003719 GetObjectFile()->GetModule()->LogMessage (log,"FindByNameAndTagAndQualifiedNameHash()");
Greg Claytoncb9c8cf2013-02-06 23:56:13 +00003720 m_apple_types_ap->FindByNameAndTagAndQualifiedNameHash (type_name.GetCString(), tag, qualified_name_hash, die_offsets);
3721 }
3722 else if (has_tag)
3723 {
3724 if (log)
Greg Clayton5160ce52013-03-27 23:08:40 +00003725 GetObjectFile()->GetModule()->LogMessage (log,"FindByNameAndTag()");
Greg Claytona8022fa2012-04-24 21:22:41 +00003726 m_apple_types_ap->FindByNameAndTag (type_name.GetCString(), tag, die_offsets);
3727 }
3728 else
3729 {
3730 m_apple_types_ap->FindByName (type_name.GetCString(), die_offsets);
3731 }
3732 }
3733 }
3734 else
3735 {
3736 if (!m_indexed)
3737 Index ();
3738
3739 m_type_index.Find (type_name, die_offsets);
3740 }
3741
3742 const size_t num_matches = die_offsets.size();
Greg Clayton58a15d52016-04-01 22:57:22 +00003743
3744 // Get the type system that we are looking to find a type for. We will use this
3745 // to ensure any matches we find are in a language that this type system supports
3746 const LanguageType language = dwarf_decl_ctx.GetLanguage();
3747 TypeSystem *type_system = (language == eLanguageTypeUnknown) ? nullptr : GetTypeSystemForLanguage(language);
3748
Greg Claytona8022fa2012-04-24 21:22:41 +00003749 if (num_matches)
3750 {
Greg Claytona8022fa2012-04-24 21:22:41 +00003751 for (size_t i=0; i<num_matches; ++i)
3752 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00003753 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00003754 DWARFDIE type_die = GetDIE (die_ref);
Greg Claytona8022fa2012-04-24 21:22:41 +00003755
3756 if (type_die)
3757 {
Greg Clayton58a15d52016-04-01 22:57:22 +00003758 // Make sure type_die's langauge matches the type system we are looking for.
3759 // We don't want to find a "Foo" type from Java if we are looking for a "Foo"
3760 // type for C, C++, ObjC, or ObjC++.
3761 if (type_system && !type_system->SupportsLanguage(type_die.GetLanguage()))
3762 continue;
Greg Claytona8022fa2012-04-24 21:22:41 +00003763 bool try_resolving_type = false;
3764
3765 // Don't try and resolve the DIE we are looking for with the DIE itself!
Greg Clayton6071e6f2015-08-26 22:57:51 +00003766 const dw_tag_t type_tag = type_die.Tag();
Greg Claytona8022fa2012-04-24 21:22:41 +00003767 // Make sure the tags match
3768 if (type_tag == tag)
3769 {
3770 // The tags match, lets try resolving this type
3771 try_resolving_type = true;
3772 }
3773 else
3774 {
3775 // The tags don't match, but we need to watch our for a
3776 // forward declaration for a struct and ("struct foo")
3777 // ends up being a class ("class foo { ... };") or
3778 // vice versa.
3779 switch (type_tag)
3780 {
3781 case DW_TAG_class_type:
3782 // We had a "class foo", see if we ended up with a "struct foo { ... };"
3783 try_resolving_type = (tag == DW_TAG_structure_type);
3784 break;
3785 case DW_TAG_structure_type:
3786 // We had a "struct foo", see if we ended up with a "class foo { ... };"
3787 try_resolving_type = (tag == DW_TAG_class_type);
3788 break;
3789 default:
3790 // Tags don't match, don't event try to resolve
3791 // using this type whose name matches....
3792 break;
3793 }
3794 }
3795
3796 if (try_resolving_type)
3797 {
3798 DWARFDeclContext type_dwarf_decl_ctx;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003799 type_die.GetDWARFDeclContext (type_dwarf_decl_ctx);
Greg Claytona8022fa2012-04-24 21:22:41 +00003800
3801 if (log)
3802 {
Greg Clayton5160ce52013-03-27 23:08:40 +00003803 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytona8022fa2012-04-24 21:22:41 +00003804 "SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext(tag=%s, qualified-name='%s') trying die=0x%8.8x (%s)",
3805 DW_TAG_value_to_name(dwarf_decl_ctx[0].tag),
3806 dwarf_decl_ctx.GetQualifiedName(),
Greg Clayton6071e6f2015-08-26 22:57:51 +00003807 type_die.GetOffset(),
Greg Claytona8022fa2012-04-24 21:22:41 +00003808 type_dwarf_decl_ctx.GetQualifiedName());
3809 }
3810
3811 // Make sure the decl contexts match all the way up
3812 if (dwarf_decl_ctx == type_dwarf_decl_ctx)
3813 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003814 Type *resolved_type = ResolveType (type_die, false);
Greg Claytona8022fa2012-04-24 21:22:41 +00003815 if (resolved_type && resolved_type != DIE_IS_BEING_PARSED)
3816 {
3817 type_sp = resolved_type->shared_from_this();
3818 break;
3819 }
3820 }
3821 }
3822 else
3823 {
3824 if (log)
3825 {
3826 std::string qualified_name;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003827 type_die.GetQualifiedName(qualified_name);
Greg Clayton5160ce52013-03-27 23:08:40 +00003828 GetObjectFile()->GetModule()->LogMessage (log,
Greg Claytona8022fa2012-04-24 21:22:41 +00003829 "SymbolFileDWARF::FindDefinitionTypeForDWARFDeclContext(tag=%s, qualified-name='%s') ignoring die=0x%8.8x (%s)",
3830 DW_TAG_value_to_name(dwarf_decl_ctx[0].tag),
3831 dwarf_decl_ctx.GetQualifiedName(),
Greg Clayton6071e6f2015-08-26 22:57:51 +00003832 type_die.GetOffset(),
Greg Claytona8022fa2012-04-24 21:22:41 +00003833 qualified_name.c_str());
3834 }
3835 }
3836 }
3837 else
3838 {
3839 if (m_using_apple_tables)
3840 {
3841 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 +00003842 die_ref.die_offset, type_name.GetCString());
Greg Claytona8022fa2012-04-24 21:22:41 +00003843 }
3844 }
3845
3846 }
3847 }
3848 }
3849 }
3850 return type_sp;
3851}
3852
Greg Claytona8022fa2012-04-24 21:22:41 +00003853TypeSP
Greg Clayton6071e6f2015-08-26 22:57:51 +00003854SymbolFileDWARF::ParseType (const SymbolContext& sc, const DWARFDIE &die, bool *type_is_new_ptr)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003855{
Greg Clayton196e8cd2015-08-17 20:31:46 +00003856 TypeSP type_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003857
Greg Clayton6071e6f2015-08-26 22:57:51 +00003858 if (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003859 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003860 TypeSystem *type_system = GetTypeSystemForLanguage(die.GetCU()->GetLanguageType());
3861
3862 if (type_system)
Greg Clayton196e8cd2015-08-17 20:31:46 +00003863 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00003864 DWARFASTParser *dwarf_ast = type_system->GetDWARFParser();
3865 if (dwarf_ast)
Greg Clayton6071e6f2015-08-26 22:57:51 +00003866 {
Greg Clayton261ac3f2015-08-28 01:01:03 +00003867 Log *log = LogChannelDWARF::GetLogIfAll(DWARF_LOG_DEBUG_INFO);
3868 type_sp = dwarf_ast->ParseTypeFromDWARF (sc, die, log, type_is_new_ptr);
3869 if (type_sp)
3870 {
3871 TypeList* type_list = GetTypeList();
3872 if (type_list)
3873 type_list->Insert(type_sp);
Siva Chandra9293fc42016-01-07 23:32:34 +00003874
3875 if (die.Tag() == DW_TAG_subprogram)
3876 {
3877 DIERef die_ref = die.GetDIERef();
3878 std::string scope_qualified_name(GetDeclContextForUID(die.GetID()).GetScopeQualifiedName().AsCString(""));
3879 if (scope_qualified_name.size())
3880 {
3881 NameToOffsetMap::iterator iter = m_function_scope_qualified_name_map.find(scope_qualified_name);
3882 if (iter != m_function_scope_qualified_name_map.end())
3883 (*iter).second->insert(die_ref);
3884 else
3885 {
3886 DIERefSetSP new_set(new std::set<DIERef>);
3887 new_set->insert(die_ref);
3888 m_function_scope_qualified_name_map.emplace(std::make_pair(scope_qualified_name, new_set));
3889 }
3890 }
3891 }
Greg Clayton261ac3f2015-08-28 01:01:03 +00003892 }
Greg Clayton6071e6f2015-08-26 22:57:51 +00003893 }
Greg Clayton196e8cd2015-08-17 20:31:46 +00003894 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003895 }
Greg Clayton196e8cd2015-08-17 20:31:46 +00003896
3897 return type_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003898}
3899
3900size_t
Greg Clayton1be10fc2010-09-29 01:12:09 +00003901SymbolFileDWARF::ParseTypes
3902(
3903 const SymbolContext& sc,
Greg Clayton6071e6f2015-08-26 22:57:51 +00003904 const DWARFDIE &orig_die,
Greg Clayton1be10fc2010-09-29 01:12:09 +00003905 bool parse_siblings,
3906 bool parse_children
3907)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003908{
3909 size_t types_added = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003910 DWARFDIE die = orig_die;
3911 while (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003912 {
3913 bool type_is_new = false;
Greg Clayton6071e6f2015-08-26 22:57:51 +00003914 if (ParseType(sc, die, &type_is_new).get())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003915 {
3916 if (type_is_new)
3917 ++types_added;
3918 }
3919
Greg Clayton6071e6f2015-08-26 22:57:51 +00003920 if (parse_children && die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003921 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003922 if (die.Tag() == DW_TAG_subprogram)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003923 {
3924 SymbolContext child_sc(sc);
Greg Clayton6071e6f2015-08-26 22:57:51 +00003925 child_sc.function = sc.comp_unit->FindFunctionByUID(die.GetID()).get();
3926 types_added += ParseTypes(child_sc, die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003927 }
3928 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00003929 types_added += ParseTypes(sc, die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003930 }
3931
3932 if (parse_siblings)
Greg Clayton6071e6f2015-08-26 22:57:51 +00003933 die = die.GetSibling();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003934 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00003935 die.Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003936 }
3937 return types_added;
3938}
3939
3940
3941size_t
3942SymbolFileDWARF::ParseFunctionBlocks (const SymbolContext &sc)
3943{
3944 assert(sc.comp_unit && sc.function);
3945 size_t functions_added = 0;
Greg Clayton1f746072012-08-29 21:13:06 +00003946 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003947 if (dwarf_cu)
3948 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003949 const dw_offset_t function_die_offset = sc.function->GetID();
3950 DWARFDIE function_die = dwarf_cu->GetDIE (function_die_offset);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003951 if (function_die)
3952 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003953 ParseFunctionBlocks(sc, &sc.function->GetBlock (false), function_die, LLDB_INVALID_ADDRESS, 0);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003954 }
3955 }
3956
3957 return functions_added;
3958}
3959
3960
3961size_t
3962SymbolFileDWARF::ParseTypes (const SymbolContext &sc)
3963{
3964 // At least a compile unit must be valid
3965 assert(sc.comp_unit);
3966 size_t types_added = 0;
Greg Clayton1f746072012-08-29 21:13:06 +00003967 DWARFCompileUnit* dwarf_cu = GetDWARFCompileUnit(sc.comp_unit);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003968 if (dwarf_cu)
3969 {
3970 if (sc.function)
3971 {
3972 dw_offset_t function_die_offset = sc.function->GetID();
Greg Clayton6071e6f2015-08-26 22:57:51 +00003973 DWARFDIE func_die = dwarf_cu->GetDIE(function_die_offset);
3974 if (func_die && func_die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003975 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003976 types_added = ParseTypes(sc, func_die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003977 }
3978 }
3979 else
3980 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003981 DWARFDIE dwarf_cu_die = dwarf_cu->DIE();
3982 if (dwarf_cu_die && dwarf_cu_die.HasChildren())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003983 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00003984 types_added = ParseTypes(sc, dwarf_cu_die.GetFirstChild(), true, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00003985 }
3986 }
3987 }
3988
3989 return types_added;
3990}
3991
3992size_t
3993SymbolFileDWARF::ParseVariablesForContext (const SymbolContext& sc)
3994{
3995 if (sc.comp_unit != NULL)
3996 {
Greg Clayton4b3dc102010-11-01 20:32:12 +00003997 DWARFDebugInfo* info = DebugInfo();
3998 if (info == NULL)
3999 return 0;
4000
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004001 if (sc.function)
4002 {
Greg Clayton2f869fe2016-03-30 20:14:35 +00004003 DWARFDIE function_die = info->GetDIE(DIERef(sc.function->GetID(), this));
Greg Clayton9422dd62013-03-04 21:46:16 +00004004
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004005 const dw_addr_t func_lo_pc = function_die.GetAttributeValueAsAddress (DW_AT_low_pc, LLDB_INVALID_ADDRESS);
Greg Claytonc7bece562013-01-25 18:06:21 +00004006 if (func_lo_pc != LLDB_INVALID_ADDRESS)
Greg Claytone38a5ed2012-01-05 03:57:59 +00004007 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004008 const size_t num_variables = ParseVariables(sc, function_die.GetFirstChild(), func_lo_pc, true, true);
Greg Claytonc662ec82011-06-17 22:10:16 +00004009
Greg Claytone38a5ed2012-01-05 03:57:59 +00004010 // Let all blocks know they have parse all their variables
4011 sc.function->GetBlock (false).SetDidParseVariables (true, true);
4012 return num_variables;
4013 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004014 }
4015 else if (sc.comp_unit)
4016 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004017 DWARFCompileUnit* dwarf_cu = info->GetCompileUnit(sc.comp_unit->GetID());
Greg Clayton9422dd62013-03-04 21:46:16 +00004018
4019 if (dwarf_cu == NULL)
4020 return 0;
4021
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004022 uint32_t vars_added = 0;
4023 VariableListSP variables (sc.comp_unit->GetVariableList(false));
4024
4025 if (variables.get() == NULL)
4026 {
4027 variables.reset(new VariableList());
4028 sc.comp_unit->SetVariableList(variables);
4029
Greg Claytond4a2b372011-09-12 23:21:58 +00004030 DIEArray die_offsets;
Greg Clayton97fbc342011-10-20 22:30:33 +00004031 if (m_using_apple_tables)
Greg Clayton7f995132011-10-04 22:41:51 +00004032 {
Greg Clayton97fbc342011-10-20 22:30:33 +00004033 if (m_apple_names_ap.get())
Greg Claytond1767f02011-12-08 02:13:16 +00004034 {
4035 DWARFMappedHash::DIEInfoArray hash_data_array;
4036 if (m_apple_names_ap->AppendAllDIEsInRange (dwarf_cu->GetOffset(),
4037 dwarf_cu->GetNextCompileUnitOffset(),
4038 hash_data_array))
4039 {
4040 DWARFMappedHash::ExtractDIEArray (hash_data_array, die_offsets);
4041 }
4042 }
Greg Clayton7f995132011-10-04 22:41:51 +00004043 }
4044 else
4045 {
4046 // Index if we already haven't to make sure the compile units
4047 // get indexed and make their global DIE index list
4048 if (!m_indexed)
4049 Index ();
4050
4051 m_global_index.FindAllEntriesForCompileUnit (dwarf_cu->GetOffset(),
Greg Clayton7f995132011-10-04 22:41:51 +00004052 die_offsets);
4053 }
4054
4055 const size_t num_matches = die_offsets.size();
Greg Claytond4a2b372011-09-12 23:21:58 +00004056 if (num_matches)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004057 {
Greg Claytond4a2b372011-09-12 23:21:58 +00004058 for (size_t i=0; i<num_matches; ++i)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004059 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004060 const DIERef& die_ref = die_offsets[i];
Greg Clayton2f869fe2016-03-30 20:14:35 +00004061 DWARFDIE die = GetDIE (die_ref);
Greg Clayton95d87902011-11-11 03:16:25 +00004062 if (die)
Greg Claytond4a2b372011-09-12 23:21:58 +00004063 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004064 VariableSP var_sp (ParseVariableDIE(sc, die, LLDB_INVALID_ADDRESS));
Greg Clayton95d87902011-11-11 03:16:25 +00004065 if (var_sp)
4066 {
4067 variables->AddVariableIfUnique (var_sp);
4068 ++vars_added;
4069 }
Greg Claytond4a2b372011-09-12 23:21:58 +00004070 }
Greg Clayton95d87902011-11-11 03:16:25 +00004071 else
4072 {
4073 if (m_using_apple_tables)
4074 {
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004075 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 +00004076 }
4077 }
4078
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004079 }
4080 }
4081 }
4082 return vars_added;
4083 }
4084 }
4085 return 0;
4086}
4087
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004088VariableSP
4089SymbolFileDWARF::ParseVariableDIE
4090(
4091 const SymbolContext& sc,
Greg Clayton6071e6f2015-08-26 22:57:51 +00004092 const DWARFDIE &die,
Greg Clayton016a95e2010-09-14 02:20:48 +00004093 const lldb::addr_t func_low_pc
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004094)
4095{
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004096 if (die.GetDWARF() != this)
4097 return die.GetDWARF()->ParseVariableDIE(sc, die, func_low_pc);
4098
Greg Clayton6071e6f2015-08-26 22:57:51 +00004099 VariableSP var_sp;
4100 if (!die)
4101 return var_sp;
4102
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004103 var_sp = GetDIEToVariable()[die.GetDIE()];
Greg Clayton83c5cd92010-11-14 22:13:40 +00004104 if (var_sp)
4105 return var_sp; // Already been parsed!
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004106
Greg Clayton6071e6f2015-08-26 22:57:51 +00004107 const dw_tag_t tag = die.Tag();
Richard Mitton0a558352013-10-17 21:14:00 +00004108 ModuleSP module = GetObjectFile()->GetModule();
Greg Clayton7f995132011-10-04 22:41:51 +00004109
4110 if ((tag == DW_TAG_variable) ||
4111 (tag == DW_TAG_constant) ||
4112 (tag == DW_TAG_formal_parameter && sc.function))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004113 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004114 DWARFAttributes attributes;
4115 const size_t num_attributes = die.GetAttributes(attributes);
Paul Hermand628cbb2015-09-15 23:44:17 +00004116 DWARFDIE spec_die;
Greg Clayton7f995132011-10-04 22:41:51 +00004117 if (num_attributes > 0)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004118 {
Greg Clayton7f995132011-10-04 22:41:51 +00004119 const char *name = NULL;
4120 const char *mangled = NULL;
4121 Declaration decl;
4122 uint32_t i;
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004123 DWARFFormValue type_die_form;
Greg Clayton6071e6f2015-08-26 22:57:51 +00004124 DWARFExpression location(die.GetCU());
Greg Clayton7f995132011-10-04 22:41:51 +00004125 bool is_external = false;
4126 bool is_artificial = false;
4127 bool location_is_const_value_data = false;
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004128 bool has_explicit_location = false;
Enrico Granata4ec130d2014-08-11 19:16:35 +00004129 DWARFFormValue const_value;
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004130 Variable::RangeList scope_ranges;
Greg Clayton23f59502012-07-17 03:23:13 +00004131 //AccessType accessibility = eAccessNone;
Greg Clayton7f995132011-10-04 22:41:51 +00004132
4133 for (i=0; i<num_attributes; ++i)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004134 {
Greg Clayton7f995132011-10-04 22:41:51 +00004135 dw_attr_t attr = attributes.AttributeAtIndex(i);
4136 DWARFFormValue form_value;
Greg Clayton54166af2014-11-22 01:58:59 +00004137
Greg Clayton6071e6f2015-08-26 22:57:51 +00004138 if (attributes.ExtractFormValueAtIndex(i, form_value))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004139 {
Greg Clayton7f995132011-10-04 22:41:51 +00004140 switch (attr)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004141 {
Greg Clayton7f995132011-10-04 22:41:51 +00004142 case DW_AT_decl_file: decl.SetFile(sc.comp_unit->GetSupportFiles().GetFileSpecAtIndex(form_value.Unsigned())); break;
4143 case DW_AT_decl_line: decl.SetLine(form_value.Unsigned()); break;
4144 case DW_AT_decl_column: decl.SetColumn(form_value.Unsigned()); break;
Greg Clayton6071e6f2015-08-26 22:57:51 +00004145 case DW_AT_name: name = form_value.AsCString(); break;
Greg Clayton71415542012-12-08 00:24:40 +00004146 case DW_AT_linkage_name:
Greg Clayton6071e6f2015-08-26 22:57:51 +00004147 case DW_AT_MIPS_linkage_name: mangled = form_value.AsCString(); break;
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004148 case DW_AT_type: type_die_form = form_value; break;
Greg Clayton1c8ef472013-04-05 23:27:21 +00004149 case DW_AT_external: is_external = form_value.Boolean(); break;
Greg Clayton7f995132011-10-04 22:41:51 +00004150 case DW_AT_const_value:
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004151 // If we have already found a DW_AT_location attribute, ignore this attribute.
4152 if (!has_explicit_location)
4153 {
4154 location_is_const_value_data = true;
4155 // The constant value will be either a block, a data value or a string.
Ed Masteeeae7212013-10-24 20:43:47 +00004156 const DWARFDataExtractor& debug_info_data = get_debug_info_data();
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004157 if (DWARFFormValue::IsBlockForm(form_value.Form()))
4158 {
4159 // Retrieve the value as a block expression.
4160 uint32_t block_offset = form_value.BlockData() - debug_info_data.GetDataStart();
4161 uint32_t block_length = form_value.Unsigned();
Richard Mitton0a558352013-10-17 21:14:00 +00004162 location.CopyOpcodeData(module, debug_info_data, block_offset, block_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004163 }
4164 else if (DWARFFormValue::IsDataForm(form_value.Form()))
4165 {
4166 // Retrieve the value as a data expression.
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004167 DWARFFormValue::FixedFormSizes fixed_form_sizes =
4168 DWARFFormValue::GetFixedFormSizesForAddressSize (
4169 attributes.CompileUnitAtIndex(i)->GetAddressByteSize(),
4170 attributes.CompileUnitAtIndex(i)->IsDWARF64());
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004171 uint32_t data_offset = attributes.DIEOffsetAtIndex(i);
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004172 uint32_t data_length = fixed_form_sizes.GetSize(form_value.Form());
Enrico Granata4ec130d2014-08-11 19:16:35 +00004173 if (data_length == 0)
4174 {
4175 const uint8_t *data_pointer = form_value.BlockData();
4176 if (data_pointer)
4177 {
Jason Molenda18f5fd32014-10-16 07:52:17 +00004178 form_value.Unsigned();
Enrico Granata4ec130d2014-08-11 19:16:35 +00004179 }
4180 else if (DWARFFormValue::IsDataForm(form_value.Form()))
4181 {
4182 // we need to get the byte size of the type later after we create the variable
4183 const_value = form_value;
4184 }
4185 }
4186 else
4187 location.CopyOpcodeData(module, debug_info_data, data_offset, data_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004188 }
4189 else
4190 {
4191 // Retrieve the value as a string expression.
4192 if (form_value.Form() == DW_FORM_strp)
4193 {
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004194 DWARFFormValue::FixedFormSizes fixed_form_sizes =
4195 DWARFFormValue::GetFixedFormSizesForAddressSize (
4196 attributes.CompileUnitAtIndex(i)->GetAddressByteSize(),
4197 attributes.CompileUnitAtIndex(i)->IsDWARF64());
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004198 uint32_t data_offset = attributes.DIEOffsetAtIndex(i);
Tamas Berghammerb7c64652015-08-25 11:45:46 +00004199 uint32_t data_length = fixed_form_sizes.GetSize(form_value.Form());
Richard Mitton0a558352013-10-17 21:14:00 +00004200 location.CopyOpcodeData(module, debug_info_data, data_offset, data_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004201 }
4202 else
4203 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004204 const char *str = form_value.AsCString();
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004205 uint32_t string_offset = str - (const char *)debug_info_data.GetDataStart();
4206 uint32_t string_length = strlen(str) + 1;
Richard Mitton0a558352013-10-17 21:14:00 +00004207 location.CopyOpcodeData(module, debug_info_data, string_offset, string_length);
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004208 }
4209 }
4210 }
4211 break;
Greg Clayton7f995132011-10-04 22:41:51 +00004212 case DW_AT_location:
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004213 {
Andrew Kaylorb32581f2013-02-13 19:57:06 +00004214 location_is_const_value_data = false;
4215 has_explicit_location = true;
Greg Clayton7f995132011-10-04 22:41:51 +00004216 if (form_value.BlockData())
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004217 {
Ed Masteeeae7212013-10-24 20:43:47 +00004218 const DWARFDataExtractor& debug_info_data = get_debug_info_data();
Greg Clayton7f995132011-10-04 22:41:51 +00004219
4220 uint32_t block_offset = form_value.BlockData() - debug_info_data.GetDataStart();
4221 uint32_t block_length = form_value.Unsigned();
Richard Mitton0a558352013-10-17 21:14:00 +00004222 location.CopyOpcodeData(module, get_debug_info_data(), block_offset, block_length);
Greg Clayton7f995132011-10-04 22:41:51 +00004223 }
4224 else
4225 {
Tamas Berghammer1f5e4482015-09-16 12:37:06 +00004226 const DWARFDataExtractor& debug_loc_data = get_debug_loc_data();
Greg Clayton7f995132011-10-04 22:41:51 +00004227 const dw_offset_t debug_loc_offset = form_value.Unsigned();
4228
Tamas Berghammer1f5e4482015-09-16 12:37:06 +00004229 size_t loc_list_length = DWARFExpression::LocationListSize(die.GetCU(), debug_loc_data, debug_loc_offset);
Greg Clayton7f995132011-10-04 22:41:51 +00004230 if (loc_list_length > 0)
4231 {
Richard Mitton0a558352013-10-17 21:14:00 +00004232 location.CopyOpcodeData(module, debug_loc_data, debug_loc_offset, loc_list_length);
Greg Clayton7f995132011-10-04 22:41:51 +00004233 assert (func_low_pc != LLDB_INVALID_ADDRESS);
Greg Clayton54166af2014-11-22 01:58:59 +00004234 location.SetLocationListSlide (func_low_pc - attributes.CompileUnitAtIndex(i)->GetBaseAddress());
Greg Clayton7f995132011-10-04 22:41:51 +00004235 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004236 }
4237 }
Greg Clayton7f995132011-10-04 22:41:51 +00004238 break;
Paul Hermand628cbb2015-09-15 23:44:17 +00004239 case DW_AT_specification:
Greg Clayton2f869fe2016-03-30 20:14:35 +00004240 spec_die = GetDIE(DIERef(form_value));
Paul Hermand628cbb2015-09-15 23:44:17 +00004241 break;
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004242 case DW_AT_start_scope:
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004243 {
Greg Clayton31460392016-03-18 20:33:49 +00004244 if (form_value.Form() == DW_FORM_sec_offset)
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004245 {
Greg Clayton31460392016-03-18 20:33:49 +00004246 DWARFRangeList dwarf_scope_ranges;
4247 const DWARFDebugRanges* debug_ranges = DebugRanges();
4248 debug_ranges->FindRanges(form_value.Unsigned(), dwarf_scope_ranges);
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004249
Greg Clayton31460392016-03-18 20:33:49 +00004250 // All DW_AT_start_scope are relative to the base address of the
4251 // compile unit. We add the compile unit base address to make
4252 // sure all the addresses are properly fixed up.
4253 for (size_t i = 0, count = dwarf_scope_ranges.GetSize(); i < count; ++i)
4254 {
4255 const DWARFRangeList::Entry& range = dwarf_scope_ranges.GetEntryRef(i);
4256 scope_ranges.Append(range.GetRangeBase() + die.GetCU()->GetBaseAddress(),
4257 range.GetByteSize());
4258 }
4259 }
4260 else
4261 {
4262 // TODO: Handle the case when DW_AT_start_scope have form constant. The
4263 // dwarf spec is a bit ambiguous about what is the expected behavior in
4264 // case the enclosing block have a non coninious address range and the
4265 // DW_AT_start_scope entry have a form constant.
4266 GetObjectFile()->GetModule()->ReportWarning ("0x%8.8" PRIx64 ": DW_AT_start_scope has unsupported form type (0x%x)\n",
4267 die.GetID(),
4268 form_value.Form());
4269 }
4270
4271 scope_ranges.Sort();
4272 scope_ranges.CombineConsecutiveRanges();
4273 }
4274 break;
Greg Clayton1c8ef472013-04-05 23:27:21 +00004275 case DW_AT_artificial: is_artificial = form_value.Boolean(); break;
Greg Clayton23f59502012-07-17 03:23:13 +00004276 case DW_AT_accessibility: break; //accessibility = DW_ACCESS_to_AccessType(form_value.Unsigned()); break;
Greg Clayton7f995132011-10-04 22:41:51 +00004277 case DW_AT_declaration:
4278 case DW_AT_description:
4279 case DW_AT_endianity:
4280 case DW_AT_segment:
Greg Clayton7f995132011-10-04 22:41:51 +00004281 case DW_AT_visibility:
4282 default:
4283 case DW_AT_abstract_origin:
4284 case DW_AT_sibling:
Greg Clayton7f995132011-10-04 22:41:51 +00004285 break;
4286 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004287 }
4288 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004289
Greg Clayton6071e6f2015-08-26 22:57:51 +00004290 const DWARFDIE parent_context_die = GetDeclContextDIEContainingDIE(die);
4291 const dw_tag_t parent_tag = die.GetParent().Tag();
4292 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 +00004293
Greg Clayton9e9f2192013-05-17 00:55:28 +00004294 ValueType scope = eValueTypeInvalid;
4295
Greg Clayton6071e6f2015-08-26 22:57:51 +00004296 const DWARFDIE sc_parent_die = GetParentSymbolContextDIE(die);
Greg Clayton9e9f2192013-05-17 00:55:28 +00004297 SymbolContextScope * symbol_context_scope = NULL;
4298
Siva Chandra0783ab92015-03-24 18:32:27 +00004299 if (!mangled)
4300 {
4301 // LLDB relies on the mangled name (DW_TAG_linkage_name or DW_AT_MIPS_linkage_name) to
4302 // generate fully qualified names of global variables with commands like "frame var j".
4303 // For example, if j were an int variable holding a value 4 and declared in a namespace
4304 // B which in turn is contained in a namespace A, the command "frame var j" returns
4305 // "(int) A::B::j = 4". If the compiler does not emit a linkage name, we should be able
4306 // to generate a fully qualified name from the declaration context.
Greg Clayton6071e6f2015-08-26 22:57:51 +00004307 if (parent_tag == DW_TAG_compile_unit &&
Jim Ingham0e0984e2015-09-02 01:06:46 +00004308 Language::LanguageIsCPlusPlus(die.GetLanguage()))
Siva Chandra0783ab92015-03-24 18:32:27 +00004309 {
4310 DWARFDeclContext decl_ctx;
4311
Greg Clayton6071e6f2015-08-26 22:57:51 +00004312 die.GetDWARFDeclContext(decl_ctx);
Siva Chandra0783ab92015-03-24 18:32:27 +00004313 mangled = decl_ctx.GetQualifiedNameAsConstString().GetCString();
4314 }
4315 }
4316
Greg Clayton9e9f2192013-05-17 00:55:28 +00004317 // DWARF doesn't specify if a DW_TAG_variable is a local, global
4318 // or static variable, so we have to do a little digging by
Bruce Mitchenerd93c4a32014-07-01 21:22:11 +00004319 // looking at the location of a variable to see if it contains
Greg Clayton9e9f2192013-05-17 00:55:28 +00004320 // a DW_OP_addr opcode _somewhere_ in the definition. I say
4321 // somewhere because clang likes to combine small global variables
4322 // into the same symbol and have locations like:
4323 // DW_OP_addr(0x1000), DW_OP_constu(2), DW_OP_plus
4324 // So if we don't have a DW_TAG_formal_parameter, we can look at
4325 // the location to see if it contains a DW_OP_addr opcode, and
4326 // then we can correctly classify our variables.
4327 if (tag == DW_TAG_formal_parameter)
4328 scope = eValueTypeVariableArgument;
4329 else
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004330 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004331 bool op_error = false;
4332 // Check if the location has a DW_OP_addr with any address value...
4333 lldb::addr_t location_DW_OP_addr = LLDB_INVALID_ADDRESS;
4334 if (!location_is_const_value_data)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004335 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004336 location_DW_OP_addr = location.GetLocation_DW_OP_addr (0, op_error);
4337 if (op_error)
Greg Clayton96c09682012-01-04 22:56:43 +00004338 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004339 StreamString strm;
4340 location.DumpLocationForAddress (&strm, eDescriptionLevelFull, 0, 0, NULL);
Greg Clayton6071e6f2015-08-26 22:57:51 +00004341 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 +00004342 }
Greg Clayton9e9f2192013-05-17 00:55:28 +00004343 }
Greg Claytond1767f02011-12-08 02:13:16 +00004344
Greg Clayton9e9f2192013-05-17 00:55:28 +00004345 if (location_DW_OP_addr != LLDB_INVALID_ADDRESS)
4346 {
4347 if (is_external)
4348 scope = eValueTypeVariableGlobal;
4349 else
4350 scope = eValueTypeVariableStatic;
4351
4352
Greg Clayton2f869fe2016-03-30 20:14:35 +00004353 SymbolFileDWARFDebugMap *debug_map_symfile = GetDebugMapSymfile();
Greg Clayton9e9f2192013-05-17 00:55:28 +00004354
4355 if (debug_map_symfile)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004356 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004357 // When leaving the DWARF in the .o files on darwin,
4358 // when we have a global variable that wasn't initialized,
4359 // the .o file might not have allocated a virtual
4360 // address for the global variable. In this case it will
4361 // have created a symbol for the global variable
4362 // that is undefined/data and external and the value will
4363 // be the byte size of the variable. When we do the
4364 // address map in SymbolFileDWARFDebugMap we rely on
4365 // having an address, we need to do some magic here
4366 // so we can get the correct address for our global
4367 // variable. The address for all of these entries
4368 // will be zero, and there will be an undefined symbol
4369 // in this object file, and the executable will have
4370 // a matching symbol with a good address. So here we
4371 // dig up the correct address and replace it in the
4372 // location for the variable, and set the variable's
4373 // symbol context scope to be that of the main executable
4374 // so the file address will resolve correctly.
4375 bool linked_oso_file_addr = false;
4376 if (is_external && location_DW_OP_addr == 0)
Greg Clayton9422dd62013-03-04 21:46:16 +00004377 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004378 // we have a possible uninitialized extern global
4379 ConstString const_name(mangled ? mangled : name);
4380 ObjectFile *debug_map_objfile = debug_map_symfile->GetObjectFile();
4381 if (debug_map_objfile)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004382 {
Greg Clayton3046e662013-07-10 01:23:25 +00004383 Symtab *debug_map_symtab = debug_map_objfile->GetSymtab();
Greg Clayton9e9f2192013-05-17 00:55:28 +00004384 if (debug_map_symtab)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004385 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004386 Symbol *exe_symbol = debug_map_symtab->FindFirstSymbolWithNameAndType (const_name,
4387 eSymbolTypeData,
4388 Symtab::eDebugYes,
4389 Symtab::eVisibilityExtern);
4390 if (exe_symbol)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004391 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004392 if (exe_symbol->ValueIsAddress())
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004393 {
Greg Clayton358cf1e2015-06-25 21:46:34 +00004394 const addr_t exe_file_addr = exe_symbol->GetAddressRef().GetFileAddress();
Greg Clayton9e9f2192013-05-17 00:55:28 +00004395 if (exe_file_addr != LLDB_INVALID_ADDRESS)
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004396 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004397 if (location.Update_DW_OP_addr (exe_file_addr))
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004398 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004399 linked_oso_file_addr = true;
4400 symbol_context_scope = exe_symbol;
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004401 }
4402 }
4403 }
4404 }
4405 }
4406 }
Greg Clayton9e9f2192013-05-17 00:55:28 +00004407 }
Greg Clayton9422dd62013-03-04 21:46:16 +00004408
Greg Clayton9e9f2192013-05-17 00:55:28 +00004409 if (!linked_oso_file_addr)
4410 {
4411 // The DW_OP_addr is not zero, but it contains a .o file address which
4412 // needs to be linked up correctly.
4413 const lldb::addr_t exe_file_addr = debug_map_symfile->LinkOSOFileAddress(this, location_DW_OP_addr);
4414 if (exe_file_addr != LLDB_INVALID_ADDRESS)
Greg Clayton9422dd62013-03-04 21:46:16 +00004415 {
Greg Clayton9e9f2192013-05-17 00:55:28 +00004416 // Update the file address for this variable
4417 location.Update_DW_OP_addr (exe_file_addr);
4418 }
4419 else
4420 {
4421 // Variable didn't make it into the final executable
4422 return var_sp;
Greg Clayton9422dd62013-03-04 21:46:16 +00004423 }
Greg Claytond1767f02011-12-08 02:13:16 +00004424 }
Greg Clayton2fc93ea2011-11-13 04:15:56 +00004425 }
Greg Clayton5cf58b92011-10-05 22:22:08 +00004426 }
4427 else
4428 {
Ewan Crawford37395ad2015-12-17 11:59:47 +00004429 if (location_is_const_value_data)
4430 scope = eValueTypeVariableStatic;
4431 else
4432 scope = eValueTypeVariableLocal;
Greg Clayton5cf58b92011-10-05 22:22:08 +00004433 }
Greg Clayton7f995132011-10-04 22:41:51 +00004434 }
Greg Clayton9e9f2192013-05-17 00:55:28 +00004435
4436 if (symbol_context_scope == NULL)
4437 {
4438 switch (parent_tag)
4439 {
4440 case DW_TAG_subprogram:
4441 case DW_TAG_inlined_subroutine:
4442 case DW_TAG_lexical_block:
4443 if (sc.function)
4444 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004445 symbol_context_scope = sc.function->GetBlock(true).FindBlockByID(sc_parent_die.GetID());
Greg Clayton9e9f2192013-05-17 00:55:28 +00004446 if (symbol_context_scope == NULL)
4447 symbol_context_scope = sc.function;
4448 }
4449 break;
4450
4451 default:
4452 symbol_context_scope = sc.comp_unit;
4453 break;
4454 }
4455 }
4456
4457 if (symbol_context_scope)
4458 {
Greg Clayton2f869fe2016-03-30 20:14:35 +00004459 SymbolFileTypeSP type_sp(new SymbolFileType(*this, DIERef(type_die_form).GetUID(this)));
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004460
Enrico Granata4ec130d2014-08-11 19:16:35 +00004461 if (const_value.Form() && type_sp && type_sp->GetType())
Greg Clayton6071e6f2015-08-26 22:57:51 +00004462 location.CopyOpcodeData(const_value.Unsigned(), type_sp->GetType()->GetByteSize(), die.GetCU()->GetAddressByteSize());
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004463
Greg Clayton6071e6f2015-08-26 22:57:51 +00004464 var_sp.reset (new Variable (die.GetID(),
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004465 name,
Greg Clayton9e9f2192013-05-17 00:55:28 +00004466 mangled,
Enrico Granata4ec130d2014-08-11 19:16:35 +00004467 type_sp,
Tamas Berghammer72ac8a82016-02-25 12:23:37 +00004468 scope,
4469 symbol_context_scope,
4470 scope_ranges,
4471 &decl,
4472 location,
4473 is_external,
Paul Herman10bc1a42015-08-18 22:46:57 +00004474 is_artificial,
4475 is_static_member));
Greg Clayton9e9f2192013-05-17 00:55:28 +00004476
4477 var_sp->SetLocationIsConstantValueData (location_is_const_value_data);
4478 }
4479 else
4480 {
4481 // Not ready to parse this variable yet. It might be a global
4482 // or static variable that is in a function scope and the function
4483 // in the symbol context wasn't filled in yet
4484 return var_sp;
4485 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004486 }
Greg Clayton7f995132011-10-04 22:41:51 +00004487 // Cache var_sp even if NULL (the variable was just a specification or
4488 // was missing vital information to be able to be displayed in the debugger
4489 // (missing location due to optimization, etc)) so we don't re-parse
4490 // this DIE over and over later...
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004491 GetDIEToVariable()[die.GetDIE()] = var_sp;
Paul Hermand628cbb2015-09-15 23:44:17 +00004492 if (spec_die)
4493 GetDIEToVariable()[spec_die.GetDIE()] = var_sp;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004494 }
4495 return var_sp;
4496}
4497
Greg Claytonc662ec82011-06-17 22:10:16 +00004498
Greg Clayton6071e6f2015-08-26 22:57:51 +00004499DWARFDIE
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004500SymbolFileDWARF::FindBlockContainingSpecification (const DIERef& func_die_ref,
Greg Clayton6071e6f2015-08-26 22:57:51 +00004501 dw_offset_t spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004502{
4503 // Give the concrete function die specified by "func_die_offset", find the
4504 // concrete block whose DW_AT_specification or DW_AT_abstract_origin points
4505 // to "spec_block_die_offset"
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004506 return FindBlockContainingSpecification (DebugInfo()->GetDIE (func_die_ref), spec_block_die_offset);
Greg Claytonc662ec82011-06-17 22:10:16 +00004507}
4508
4509
Greg Clayton6071e6f2015-08-26 22:57:51 +00004510DWARFDIE
4511SymbolFileDWARF::FindBlockContainingSpecification(const DWARFDIE &die,
4512 dw_offset_t spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004513{
4514 if (die)
4515 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004516 switch (die.Tag())
Greg Claytonc662ec82011-06-17 22:10:16 +00004517 {
4518 case DW_TAG_subprogram:
4519 case DW_TAG_inlined_subroutine:
4520 case DW_TAG_lexical_block:
4521 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004522 if (die.GetAttributeValueAsReference (DW_AT_specification, DW_INVALID_OFFSET) == spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004523 return die;
Greg Claytonc662ec82011-06-17 22:10:16 +00004524
Greg Clayton6071e6f2015-08-26 22:57:51 +00004525 if (die.GetAttributeValueAsReference (DW_AT_abstract_origin, DW_INVALID_OFFSET) == spec_block_die_offset)
Greg Claytonc662ec82011-06-17 22:10:16 +00004526 return die;
Greg Claytonc662ec82011-06-17 22:10:16 +00004527 }
4528 break;
4529 }
4530
4531 // Give the concrete function die specified by "func_die_offset", find the
4532 // concrete block whose DW_AT_specification or DW_AT_abstract_origin points
4533 // to "spec_block_die_offset"
Greg Clayton6071e6f2015-08-26 22:57:51 +00004534 for (DWARFDIE child_die = die.GetFirstChild(); child_die; child_die = child_die.GetSibling())
Greg Claytonc662ec82011-06-17 22:10:16 +00004535 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004536 DWARFDIE result_die = FindBlockContainingSpecification (child_die, spec_block_die_offset);
Greg Claytonc662ec82011-06-17 22:10:16 +00004537 if (result_die)
4538 return result_die;
4539 }
4540 }
4541
Greg Clayton6071e6f2015-08-26 22:57:51 +00004542 return DWARFDIE();
Greg Claytonc662ec82011-06-17 22:10:16 +00004543}
4544
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004545size_t
Greg Clayton6071e6f2015-08-26 22:57:51 +00004546SymbolFileDWARF::ParseVariables (const SymbolContext& sc,
4547 const DWARFDIE &orig_die,
4548 const lldb::addr_t func_low_pc,
4549 bool parse_siblings,
4550 bool parse_children,
4551 VariableList* cc_variable_list)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004552{
Greg Clayton6071e6f2015-08-26 22:57:51 +00004553 if (!orig_die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004554 return 0;
4555
Greg Claytonc662ec82011-06-17 22:10:16 +00004556 VariableListSP variable_list_sp;
4557
4558 size_t vars_added = 0;
Greg Clayton6071e6f2015-08-26 22:57:51 +00004559 DWARFDIE die = orig_die;
4560 while (die)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004561 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004562 dw_tag_t tag = die.Tag();
Greg Claytonc662ec82011-06-17 22:10:16 +00004563
4564 // Check to see if we have already parsed this variable or constant?
Tamas Berghammereb882fc2015-09-09 10:20:48 +00004565 VariableSP var_sp = GetDIEToVariable()[die.GetDIE()];
Greg Clayton6071e6f2015-08-26 22:57:51 +00004566 if (var_sp)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004567 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004568 if (cc_variable_list)
Greg Clayton6071e6f2015-08-26 22:57:51 +00004569 cc_variable_list->AddVariableIfUnique (var_sp);
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004570 }
4571 else
4572 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004573 // We haven't already parsed it, lets do that now.
4574 if ((tag == DW_TAG_variable) ||
4575 (tag == DW_TAG_constant) ||
4576 (tag == DW_TAG_formal_parameter && sc.function))
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004577 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004578 if (variable_list_sp.get() == NULL)
Greg Clayton73bf5db2011-06-17 01:22:15 +00004579 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004580 DWARFDIE sc_parent_die = GetParentSymbolContextDIE(orig_die);
4581 dw_tag_t parent_tag = sc_parent_die.Tag();
Greg Claytonc662ec82011-06-17 22:10:16 +00004582 switch (parent_tag)
4583 {
4584 case DW_TAG_compile_unit:
4585 if (sc.comp_unit != NULL)
4586 {
4587 variable_list_sp = sc.comp_unit->GetVariableList(false);
4588 if (variable_list_sp.get() == NULL)
4589 {
4590 variable_list_sp.reset(new VariableList());
4591 sc.comp_unit->SetVariableList(variable_list_sp);
4592 }
4593 }
4594 else
4595 {
Daniel Malead01b2952012-11-29 21:49:15 +00004596 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 +00004597 sc_parent_die.GetID(),
4598 sc_parent_die.GetTagAsCString(),
4599 orig_die.GetID(),
4600 orig_die.GetTagAsCString());
Greg Claytonc662ec82011-06-17 22:10:16 +00004601 }
4602 break;
4603
4604 case DW_TAG_subprogram:
4605 case DW_TAG_inlined_subroutine:
4606 case DW_TAG_lexical_block:
4607 if (sc.function != NULL)
4608 {
4609 // Check to see if we already have parsed the variables for the given scope
4610
Greg Clayton6071e6f2015-08-26 22:57:51 +00004611 Block *block = sc.function->GetBlock(true).FindBlockByID(sc_parent_die.GetID());
Greg Claytonc662ec82011-06-17 22:10:16 +00004612 if (block == NULL)
4613 {
4614 // This must be a specification or abstract origin with
Bruce Mitchenerd93c4a32014-07-01 21:22:11 +00004615 // a concrete block counterpart in the current function. We need
Greg Claytonc662ec82011-06-17 22:10:16 +00004616 // to find the concrete block so we can correctly add the
4617 // variable to it
Greg Clayton2f869fe2016-03-30 20:14:35 +00004618 const DWARFDIE concrete_block_die = FindBlockContainingSpecification (DIERef(sc.function->GetID(), this),
Greg Clayton6071e6f2015-08-26 22:57:51 +00004619 sc_parent_die.GetOffset());
Greg Claytonc662ec82011-06-17 22:10:16 +00004620 if (concrete_block_die)
Greg Clayton6071e6f2015-08-26 22:57:51 +00004621 block = sc.function->GetBlock(true).FindBlockByID(concrete_block_die.GetID());
Greg Claytonc662ec82011-06-17 22:10:16 +00004622 }
4623
4624 if (block != NULL)
4625 {
4626 const bool can_create = false;
4627 variable_list_sp = block->GetBlockVariableList (can_create);
4628 if (variable_list_sp.get() == NULL)
4629 {
4630 variable_list_sp.reset(new VariableList());
4631 block->SetVariableList(variable_list_sp);
4632 }
4633 }
4634 }
4635 break;
4636
4637 default:
Daniel Malead01b2952012-11-29 21:49:15 +00004638 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 +00004639 orig_die.GetID(),
4640 orig_die.GetTagAsCString());
Greg Claytonc662ec82011-06-17 22:10:16 +00004641 break;
4642 }
Greg Clayton73bf5db2011-06-17 01:22:15 +00004643 }
Greg Claytonc662ec82011-06-17 22:10:16 +00004644
4645 if (variable_list_sp)
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004646 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004647 VariableSP var_sp (ParseVariableDIE(sc, die, func_low_pc));
Greg Clayton73bf5db2011-06-17 01:22:15 +00004648 if (var_sp)
4649 {
Greg Claytonc662ec82011-06-17 22:10:16 +00004650 variable_list_sp->AddVariableIfUnique (var_sp);
Greg Clayton73bf5db2011-06-17 01:22:15 +00004651 if (cc_variable_list)
4652 cc_variable_list->AddVariableIfUnique (var_sp);
4653 ++vars_added;
4654 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004655 }
4656 }
4657 }
Greg Claytonc662ec82011-06-17 22:10:16 +00004658
4659 bool skip_children = (sc.function == NULL && tag == DW_TAG_subprogram);
4660
Greg Clayton6071e6f2015-08-26 22:57:51 +00004661 if (!skip_children && parse_children && die.HasChildren())
Greg Claytonc662ec82011-06-17 22:10:16 +00004662 {
Greg Clayton6071e6f2015-08-26 22:57:51 +00004663 vars_added += ParseVariables(sc, die.GetFirstChild(), func_low_pc, true, true, cc_variable_list);
Greg Claytonc662ec82011-06-17 22:10:16 +00004664 }
4665
4666 if (parse_siblings)
Greg Clayton6071e6f2015-08-26 22:57:51 +00004667 die = die.GetSibling();
Greg Claytonc662ec82011-06-17 22:10:16 +00004668 else
Greg Clayton6071e6f2015-08-26 22:57:51 +00004669 die.Clear();
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004670 }
Greg Claytonc662ec82011-06-17 22:10:16 +00004671 return vars_added;
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004672}
4673
4674//------------------------------------------------------------------
4675// PluginInterface protocol
4676//------------------------------------------------------------------
Greg Clayton57abc5d2013-05-10 21:47:16 +00004677ConstString
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004678SymbolFileDWARF::GetPluginName()
4679{
Chris Lattner30fdc8d2010-06-08 16:52:24 +00004680 return GetPluginNameStatic();
4681}
4682
4683uint32_t
4684SymbolFileDWARF::GetPluginVersion()
4685{
4686 return 1;
4687}
4688
4689void
Sean Callanancc427fa2011-07-30 02:42:06 +00004690SymbolFileDWARF::DumpIndexes ()
4691{
4692 StreamFile s(stdout, false);
4693
Greg Claytonb5ad4ec2013-04-29 17:25:54 +00004694 s.Printf ("DWARF index for (%s) '%s':",
Sean Callanancc427fa2011-07-30 02:42:06 +00004695 GetObjectFile()->GetModule()->GetArchitecture().GetArchitectureName(),
Greg Claytonb5ad4ec2013-04-29 17:25:54 +00004696 GetObjectFile()->GetFileSpec().GetPath().c_str());
Sean Callanancc427fa2011-07-30 02:42:06 +00004697 s.Printf("\nFunction basenames:\n"); m_function_basename_index.Dump (&s);
4698 s.Printf("\nFunction fullnames:\n"); m_function_fullname_index.Dump (&s);
4699 s.Printf("\nFunction methods:\n"); m_function_method_index.Dump (&s);
4700 s.Printf("\nFunction selectors:\n"); m_function_selector_index.Dump (&s);
4701 s.Printf("\nObjective C class selectors:\n"); m_objc_class_selectors_index.Dump (&s);
4702 s.Printf("\nGlobals and statics:\n"); m_global_index.Dump (&s);
4703 s.Printf("\nTypes:\n"); m_type_index.Dump (&s);
Bruce Mitchenere171da52015-07-22 00:16:02 +00004704 s.Printf("\nNamespaces:\n"); m_namespace_index.Dump (&s);
Sean Callanancc427fa2011-07-30 02:42:06 +00004705}
4706
Greg Claytoncaab74e2012-01-28 00:48:57 +00004707
Greg Clayton1f746072012-08-29 21:13:06 +00004708SymbolFileDWARFDebugMap *
Greg Clayton2f869fe2016-03-30 20:14:35 +00004709SymbolFileDWARF::GetDebugMapSymfile()
Greg Clayton1f746072012-08-29 21:13:06 +00004710{
4711 if (m_debug_map_symfile == NULL && !m_debug_map_module_wp.expired())
4712 {
4713 lldb::ModuleSP module_sp (m_debug_map_module_wp.lock());
4714 if (module_sp)
4715 {
4716 SymbolVendor *sym_vendor = module_sp->GetSymbolVendor();
4717 if (sym_vendor)
4718 m_debug_map_symfile = (SymbolFileDWARFDebugMap *)sym_vendor->GetSymbolFile();
4719 }
4720 }
4721 return m_debug_map_symfile;
4722}
Tamas Berghammer1f5e4482015-09-16 12:37:06 +00004723
4724DWARFExpression::LocationListFormat
4725SymbolFileDWARF::GetLocationListFormat() const
4726{
4727 return DWARFExpression::RegularLocationList;
4728}