blob: 49d38ad5c610cca654d7f4a7b84e212e69561453 [file] [log] [blame]
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001//===-- Address.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 "lldb/Core/Address.h"
11#include "lldb/Core/Module.h"
12#include "lldb/Core/Section.h"
13#include "lldb/Symbol/ObjectFile.h"
Greg Claytonf5e56de2010-09-14 23:36:40 +000014#include "lldb/Target/ExecutionContext.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000015#include "lldb/Target/Process.h"
16#include "lldb/Target/Target.h"
17
Greg Clayton3f5c08f2011-05-18 22:01:49 +000018#include "llvm/ADT/Triple.h"
19
Chris Lattner30fdc8d2010-06-08 16:52:24 +000020using namespace lldb;
21using namespace lldb_private;
22
23static size_t
24ReadBytes (ExecutionContextScope *exe_scope, const Address &address, void *dst, size_t dst_len)
25{
26 if (exe_scope == NULL)
27 return 0;
28
Chris Lattner30fdc8d2010-06-08 16:52:24 +000029 Target *target = exe_scope->CalculateTarget();
30 if (target)
31 {
32 Error error;
Greg Claytondb598232011-01-07 01:57:07 +000033 bool prefer_file_cache = false;
34 return target->ReadMemory (address, prefer_file_cache, dst, dst_len, error);
Chris Lattner30fdc8d2010-06-08 16:52:24 +000035 }
36 return 0;
37}
38
39static bool
40GetByteOrderAndAddressSize (ExecutionContextScope *exe_scope, const Address &address, ByteOrder& byte_order, uint32_t& addr_size)
41{
42 byte_order = eByteOrderInvalid;
43 addr_size = 0;
44 if (exe_scope == NULL)
45 return false;
46
Greg Clayton514487e2011-02-15 21:59:32 +000047 Target *target = exe_scope->CalculateTarget();
48 if (target)
Chris Lattner30fdc8d2010-06-08 16:52:24 +000049 {
Greg Clayton514487e2011-02-15 21:59:32 +000050 byte_order = target->GetArchitecture().GetByteOrder();
51 addr_size = target->GetArchitecture().GetAddressByteSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +000052 }
53
54 if (byte_order == eByteOrderInvalid || addr_size == 0)
55 {
56 Module *module = address.GetModule();
57 if (module)
58 {
Greg Clayton514487e2011-02-15 21:59:32 +000059 byte_order = module->GetArchitecture().GetByteOrder();
Chris Lattner30fdc8d2010-06-08 16:52:24 +000060 addr_size = module->GetArchitecture().GetAddressByteSize();
61 }
62 }
63 return byte_order != eByteOrderInvalid && addr_size != 0;
64}
65
66static uint64_t
67ReadUIntMax64 (ExecutionContextScope *exe_scope, const Address &address, uint32_t byte_size, bool &success)
68{
69 uint64_t uval64 = 0;
70 if (exe_scope == NULL || byte_size > sizeof(uint64_t))
71 {
72 success = false;
73 return 0;
74 }
75 uint64_t buf;
76
77 success = ReadBytes (exe_scope, address, &buf, byte_size) == byte_size;
78 if (success)
79 {
80 ByteOrder byte_order = eByteOrderInvalid;
81 uint32_t addr_size = 0;
82 if (GetByteOrderAndAddressSize (exe_scope, address, byte_order, addr_size))
83 {
84 DataExtractor data (&buf, sizeof(buf), byte_order, addr_size);
85 uint32_t offset = 0;
86 uval64 = data.GetU64(&offset);
87 }
88 else
89 success = false;
90 }
91 return uval64;
92}
93
94static bool
95ReadAddress (ExecutionContextScope *exe_scope, const Address &address, uint32_t pointer_size, Address &deref_so_addr)
96{
97 if (exe_scope == NULL)
98 return false;
99
100
101 bool success = false;
102 addr_t deref_addr = ReadUIntMax64 (exe_scope, address, pointer_size, success);
103 if (success)
104 {
Greg Claytonf5e56de2010-09-14 23:36:40 +0000105 ExecutionContext exe_ctx;
Greg Clayton0603aa92010-10-04 01:05:56 +0000106 exe_scope->CalculateExecutionContext(exe_ctx);
Greg Claytonf5e56de2010-09-14 23:36:40 +0000107 // If we have any sections that are loaded, try and resolve using the
108 // section load list
109 if (exe_ctx.target && !exe_ctx.target->GetSectionLoadList().IsEmpty())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000110 {
Greg Claytonf5e56de2010-09-14 23:36:40 +0000111 if (exe_ctx.target->GetSectionLoadList().ResolveLoadAddress (deref_addr, deref_so_addr))
112 return true;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000113 }
114 else
115 {
Greg Claytonf5e56de2010-09-14 23:36:40 +0000116 // If we were not running, yet able to read an integer, we must
117 // have a module
118 Module *module = address.GetModule();
119 assert (module);
120 if (module->ResolveFileAddress(deref_addr, deref_so_addr))
121 return true;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000122 }
Greg Claytonf5e56de2010-09-14 23:36:40 +0000123
124 // We couldn't make "deref_addr" into a section offset value, but we were
125 // able to read the address, so we return a section offset address with
126 // no section and "deref_addr" as the offset (address).
127 deref_so_addr.SetSection(NULL);
128 deref_so_addr.SetOffset(deref_addr);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000129 return true;
130 }
131 return false;
132}
133
134static bool
135DumpUInt (ExecutionContextScope *exe_scope, const Address &address, uint32_t byte_size, Stream* strm)
136{
Greg Clayton471b31c2010-07-20 22:52:08 +0000137 if (exe_scope == NULL || byte_size == 0)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000138 return 0;
139 std::vector<uint8_t> buf(byte_size, 0);
140
141 if (ReadBytes (exe_scope, address, &buf[0], buf.size()) == buf.size())
142 {
143 ByteOrder byte_order = eByteOrderInvalid;
144 uint32_t addr_size = 0;
145 if (GetByteOrderAndAddressSize (exe_scope, address, byte_order, addr_size))
146 {
Greg Clayton471b31c2010-07-20 22:52:08 +0000147 DataExtractor data (&buf.front(), buf.size(), byte_order, addr_size);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000148
149 data.Dump (strm,
150 0, // Start offset in "data"
151 eFormatHex, // Print as characters
152 buf.size(), // Size of item
153 1, // Items count
154 UINT32_MAX, // num per line
155 LLDB_INVALID_ADDRESS,// base address
156 0, // bitfield bit size
157 0); // bitfield bit offset
158
159 return true;
160 }
161 }
162 return false;
163}
164
165
166static size_t
167ReadCStringFromMemory (ExecutionContextScope *exe_scope, const Address &address, Stream *strm)
168{
169 if (exe_scope == NULL)
170 return 0;
171 const size_t k_buf_len = 256;
172 char buf[k_buf_len+1];
173 buf[k_buf_len] = '\0'; // NULL terminate
174
Greg Clayton710dd5a2011-01-08 20:28:42 +0000175 // Byte order and address size don't matter for C string dumping..
Greg Clayton7fb56d02011-02-01 01:31:41 +0000176 DataExtractor data (buf, sizeof(buf), lldb::endian::InlHostByteOrder(), 4);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000177 size_t total_len = 0;
178 size_t bytes_read;
179 Address curr_address(address);
180 strm->PutChar ('"');
181 while ((bytes_read = ReadBytes (exe_scope, curr_address, buf, k_buf_len)) > 0)
182 {
183 size_t len = strlen(buf);
184 if (len == 0)
185 break;
186 if (len > bytes_read)
187 len = bytes_read;
188
189 data.Dump (strm,
190 0, // Start offset in "data"
191 eFormatChar, // Print as characters
192 1, // Size of item (1 byte for a char!)
193 len, // How many bytes to print?
194 UINT32_MAX, // num per line
195 LLDB_INVALID_ADDRESS,// base address
196 0, // bitfield bit size
197
198 0); // bitfield bit offset
199
200 total_len += bytes_read;
201
202 if (len < k_buf_len)
203 break;
204 curr_address.SetOffset (curr_address.GetOffset() + bytes_read);
205 }
206 strm->PutChar ('"');
207 return total_len;
208}
209
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000210Address::Address (addr_t address, const SectionList * sections) :
211 m_section (NULL),
212 m_offset (LLDB_INVALID_ADDRESS)
213{
214 ResolveAddressUsingFileSections(address, sections);
215}
216
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000217const Address&
218Address::operator= (const Address& rhs)
219{
220 if (this != &rhs)
221 {
222 m_section = rhs.m_section;
223 m_offset = rhs.m_offset;
224 }
225 return *this;
226}
227
228bool
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000229Address::ResolveAddressUsingFileSections (addr_t addr, const SectionList *sections)
230{
231 if (sections)
232 m_section = sections->FindSectionContainingFileAddress(addr).get();
233 else
234 m_section = NULL;
235
236 if (m_section != NULL)
237 {
238 assert( m_section->ContainsFileAddress(addr) );
239 m_offset = addr - m_section->GetFileAddress();
240 return true; // Successfully transformed addr into a section offset address
241 }
242
243 m_offset = addr;
244 return false; // Failed to resolve this address to a section offset value
245}
246
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000247Module *
248Address::GetModule () const
249{
250 if (m_section)
251 return m_section->GetModule();
252 return NULL;
253}
254
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000255addr_t
256Address::GetFileAddress () const
257{
258 if (m_section != NULL)
259 {
260 addr_t sect_file_addr = m_section->GetFileAddress();
261 if (sect_file_addr == LLDB_INVALID_ADDRESS)
262 {
263 // Section isn't resolved, we can't return a valid file address
264 return LLDB_INVALID_ADDRESS;
265 }
266 // We have a valid file range, so we can return the file based
267 // address by adding the file base address to our offset
268 return sect_file_addr + m_offset;
269 }
270 // No section, we just return the offset since it is the value in this case
271 return m_offset;
272}
273
274addr_t
Greg Claytonf5e56de2010-09-14 23:36:40 +0000275Address::GetLoadAddress (Target *target) const
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000276{
Greg Claytonf5e56de2010-09-14 23:36:40 +0000277 if (m_section == NULL)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000278 {
Greg Claytonf5e56de2010-09-14 23:36:40 +0000279 // No section, we just return the offset since it is the value in this case
280 return m_offset;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000281 }
Greg Claytonf5e56de2010-09-14 23:36:40 +0000282
283 if (target)
284 {
285 addr_t sect_load_addr = m_section->GetLoadBaseAddress (target);
286
287 if (sect_load_addr != LLDB_INVALID_ADDRESS)
288 {
289 // We have a valid file range, so we can return the file based
290 // address by adding the file base address to our offset
291 return sect_load_addr + m_offset;
292 }
293 }
294 // The section isn't resolved or no process was supplied so we can't
295 // return a valid file address.
296 return LLDB_INVALID_ADDRESS;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000297}
298
Greg Clayton3f5c08f2011-05-18 22:01:49 +0000299addr_t
300Address::GetCallableLoadAddress (Target *target) const
301{
302 addr_t code_addr = GetLoadAddress (target);
Greg Claytonf3ef3d22011-05-22 22:46:53 +0000303
304 if (target)
305 return target->GetCallableLoadAddress (code_addr, GetAddressClass());
Greg Clayton3f5c08f2011-05-18 22:01:49 +0000306 return code_addr;
307}
308
Greg Claytoncff851a2011-05-22 04:32:55 +0000309bool
310Address::SetCallableLoadAddress (lldb::addr_t load_addr, Target *target)
311{
312 if (SetLoadAddress (load_addr, target))
313 {
Greg Claytonf3ef3d22011-05-22 22:46:53 +0000314 if (target)
315 m_offset = target->GetCallableLoadAddress(m_offset, GetAddressClass());
Greg Claytoncff851a2011-05-22 04:32:55 +0000316 return true;
317 }
318 return false;
319}
320
Greg Clayton92bb12c2011-05-19 18:17:41 +0000321addr_t
322Address::GetOpcodeLoadAddress (Target *target) const
323{
324 addr_t code_addr = GetLoadAddress (target);
Greg Clayton92bb12c2011-05-19 18:17:41 +0000325 if (code_addr != LLDB_INVALID_ADDRESS)
Greg Claytonf3ef3d22011-05-22 22:46:53 +0000326 code_addr = target->GetOpcodeLoadAddress (code_addr, GetAddressClass());
Greg Clayton92bb12c2011-05-19 18:17:41 +0000327 return code_addr;
328}
329
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000330bool
Greg Claytoncff851a2011-05-22 04:32:55 +0000331Address::SetOpcodeLoadAddress (lldb::addr_t load_addr, Target *target)
332{
333 if (SetLoadAddress (load_addr, target))
334 {
Greg Claytonf3ef3d22011-05-22 22:46:53 +0000335 if (target)
336 m_offset = target->GetOpcodeLoadAddress (m_offset, GetAddressClass());
Greg Claytoncff851a2011-05-22 04:32:55 +0000337 return true;
338 }
339 return false;
340}
341
342bool
Greg Claytondda4f7b2010-06-30 23:03:03 +0000343Address::Dump (Stream *s, ExecutionContextScope *exe_scope, DumpStyle style, DumpStyle fallback_style, uint32_t addr_size) const
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000344{
345 // If the section was NULL, only load address is going to work.
346 if (m_section == NULL)
347 style = DumpStyleLoadAddress;
348
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000349 Target *target = NULL;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000350 Process *process = NULL;
351 if (exe_scope)
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000352 {
353 target = exe_scope->CalculateTarget();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000354 process = exe_scope->CalculateProcess();
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000355 }
Greg Claytondda4f7b2010-06-30 23:03:03 +0000356 // If addr_byte_size is UINT32_MAX, then determine the correct address
357 // byte size for the process or default to the size of addr_t
358 if (addr_size == UINT32_MAX)
359 {
360 if (process)
Greg Clayton514487e2011-02-15 21:59:32 +0000361 addr_size = target->GetArchitecture().GetAddressByteSize ();
Greg Claytondda4f7b2010-06-30 23:03:03 +0000362 else
363 addr_size = sizeof(addr_t);
364 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000365
Greg Claytonc9800662010-09-10 01:30:46 +0000366 Address so_addr;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000367 switch (style)
368 {
Greg Claytonc982c762010-07-09 20:39:50 +0000369 case DumpStyleInvalid:
370 return false;
371
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000372 case DumpStyleSectionNameOffset:
373 if (m_section != NULL)
374 {
375 m_section->DumpName(s);
376 s->Printf (" + %llu", m_offset);
377 }
378 else
379 {
Greg Claytondda4f7b2010-06-30 23:03:03 +0000380 s->Address(m_offset, addr_size);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000381 }
382 break;
383
384 case DumpStyleSectionPointerOffset:
Greg Claytondda4f7b2010-06-30 23:03:03 +0000385 s->Printf("(Section *)%.*p + ", (int)sizeof(void*) * 2, m_section);
386 s->Address(m_offset, addr_size);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000387 break;
388
389 case DumpStyleModuleWithFileAddress:
Greg Claytoncfd1ace2010-10-31 03:01:06 +0000390 if (m_section)
391 s->Printf("%s[", m_section->GetModule()->GetFileSpec().GetFilename().AsCString());
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000392 // Fall through
393 case DumpStyleFileAddress:
394 {
395 addr_t file_addr = GetFileAddress();
396 if (file_addr == LLDB_INVALID_ADDRESS)
397 {
398 if (fallback_style != DumpStyleInvalid)
Greg Claytondda4f7b2010-06-30 23:03:03 +0000399 return Dump (s, exe_scope, fallback_style, DumpStyleInvalid, addr_size);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000400 return false;
401 }
402 s->Address (file_addr, addr_size);
Greg Claytoncfd1ace2010-10-31 03:01:06 +0000403 if (style == DumpStyleModuleWithFileAddress && m_section)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000404 s->PutChar(']');
405 }
406 break;
407
408 case DumpStyleLoadAddress:
409 {
Greg Claytonf5e56de2010-09-14 23:36:40 +0000410 addr_t load_addr = GetLoadAddress (target);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000411 if (load_addr == LLDB_INVALID_ADDRESS)
412 {
413 if (fallback_style != DumpStyleInvalid)
Greg Claytondda4f7b2010-06-30 23:03:03 +0000414 return Dump (s, exe_scope, fallback_style, DumpStyleInvalid, addr_size);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000415 return false;
416 }
417 s->Address (load_addr, addr_size);
418 }
419 break;
420
421 case DumpStyleResolvedDescription:
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000422 case DumpStyleResolvedDescriptionNoModule:
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000423 if (IsSectionOffset())
424 {
Greg Claytone0d378b2011-03-24 21:19:54 +0000425 AddressType addr_type = eAddressTypeLoad;
Greg Claytonf5e56de2010-09-14 23:36:40 +0000426 addr_t addr = GetLoadAddress (target);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000427 if (addr == LLDB_INVALID_ADDRESS)
428 {
429 addr = GetFileAddress();
430 addr_type = eAddressTypeFile;
431 }
432
433 uint32_t pointer_size = 4;
Greg Claytonc9800662010-09-10 01:30:46 +0000434 Module *module = GetModule();
Greg Clayton514487e2011-02-15 21:59:32 +0000435 if (target)
436 pointer_size = target->GetArchitecture().GetAddressByteSize();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000437 else if (module)
438 pointer_size = module->GetArchitecture().GetAddressByteSize();
439
440 bool showed_info = false;
441 const Section *section = GetSection();
442 if (section)
443 {
Greg Clayton70e33eb2010-07-21 21:49:46 +0000444 SectionType sect_type = section->GetType();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000445 switch (sect_type)
446 {
Greg Clayton89411422010-10-08 00:21:05 +0000447 case eSectionTypeData:
448 if (module)
449 {
450 ObjectFile *objfile = module->GetObjectFile();
451 if (objfile)
452 {
453 Symtab *symtab = objfile->GetSymtab();
454 if (symtab)
455 {
456 const addr_t file_Addr = GetFileAddress();
457 Symbol *symbol = symtab->FindSymbolContainingFileAddress (file_Addr);
458 if (symbol)
459 {
460 const char *symbol_name = symbol->GetName().AsCString();
461 if (symbol_name)
462 {
463 s->PutCString(symbol_name);
464 addr_t delta = file_Addr - symbol->GetAddressRangePtr()->GetBaseAddress().GetFileAddress();
465 if (delta)
466 s->Printf(" + %llu", delta);
467 showed_info = true;
468 }
469 }
470 }
471 }
472 }
473 break;
474
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000475 case eSectionTypeDataCString:
476 // Read the C string from memory and display it
477 showed_info = true;
478 ReadCStringFromMemory (exe_scope, *this, s);
479 break;
480
481 case eSectionTypeDataCStringPointers:
482 {
483 if (ReadAddress (exe_scope, *this, pointer_size, so_addr))
484 {
485#if VERBOSE_OUTPUT
486 s->PutCString("(char *)");
487 so_addr.Dump(s, exe_scope, DumpStyleLoadAddress, DumpStyleFileAddress);
488 s->PutCString(": ");
489#endif
490 showed_info = true;
491 ReadCStringFromMemory (exe_scope, so_addr, s);
492 }
493 }
494 break;
495
496 case eSectionTypeDataObjCMessageRefs:
497 {
498 if (ReadAddress (exe_scope, *this, pointer_size, so_addr))
499 {
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000500 if (target && so_addr.IsSectionOffset())
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000501 {
Greg Claytonc9800662010-09-10 01:30:46 +0000502 SymbolContext func_sc;
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000503 target->GetImages().ResolveSymbolContextForAddress (so_addr,
504 eSymbolContextEverything,
505 func_sc);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000506 if (func_sc.function || func_sc.symbol)
507 {
508 showed_info = true;
509#if VERBOSE_OUTPUT
510 s->PutCString ("(objc_msgref *) -> { (func*)");
511 so_addr.Dump(s, exe_scope, DumpStyleLoadAddress, DumpStyleFileAddress);
512#else
513 s->PutCString ("{ ");
514#endif
515 Address cstr_addr(*this);
516 cstr_addr.SetOffset(cstr_addr.GetOffset() + pointer_size);
Greg Clayton6dadd502010-09-02 21:44:10 +0000517 func_sc.DumpStopContext(s, exe_scope, so_addr, true, true, false);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000518 if (ReadAddress (exe_scope, cstr_addr, pointer_size, so_addr))
519 {
520#if VERBOSE_OUTPUT
521 s->PutCString("), (char *)");
522 so_addr.Dump(s, exe_scope, DumpStyleLoadAddress, DumpStyleFileAddress);
523 s->PutCString(" (");
524#else
525 s->PutCString(", ");
526#endif
527 ReadCStringFromMemory (exe_scope, so_addr, s);
528 }
529#if VERBOSE_OUTPUT
530 s->PutCString(") }");
531#else
532 s->PutCString(" }");
533#endif
534 }
535 }
536 }
537 }
538 break;
539
540 case eSectionTypeDataObjCCFStrings:
541 {
542 Address cfstring_data_addr(*this);
543 cfstring_data_addr.SetOffset(cfstring_data_addr.GetOffset() + (2 * pointer_size));
544 if (ReadAddress (exe_scope, cfstring_data_addr, pointer_size, so_addr))
545 {
546#if VERBOSE_OUTPUT
547 s->PutCString("(CFString *) ");
548 cfstring_data_addr.Dump(s, exe_scope, DumpStyleLoadAddress, DumpStyleFileAddress);
549 s->PutCString(" -> @");
550#else
551 s->PutChar('@');
552#endif
553 if (so_addr.Dump(s, exe_scope, DumpStyleResolvedDescription))
554 showed_info = true;
555 }
556 }
557 break;
558
559 case eSectionTypeData4:
560 // Read the 4 byte data and display it
561 showed_info = true;
562 s->PutCString("(uint32_t) ");
563 DumpUInt (exe_scope, *this, 4, s);
564 break;
565
566 case eSectionTypeData8:
567 // Read the 8 byte data and display it
568 showed_info = true;
569 s->PutCString("(uint64_t) ");
570 DumpUInt (exe_scope, *this, 8, s);
571 break;
572
573 case eSectionTypeData16:
574 // Read the 16 byte data and display it
575 showed_info = true;
576 s->PutCString("(uint128_t) ");
577 DumpUInt (exe_scope, *this, 16, s);
578 break;
579
580 case eSectionTypeDataPointers:
581 // Read the pointer data and display it
582 {
583 if (ReadAddress (exe_scope, *this, pointer_size, so_addr))
584 {
585 s->PutCString ("(void *)");
586 so_addr.Dump(s, exe_scope, DumpStyleLoadAddress, DumpStyleFileAddress);
587
588 showed_info = true;
589 if (so_addr.IsSectionOffset())
590 {
Greg Claytonc9800662010-09-10 01:30:46 +0000591 SymbolContext pointer_sc;
Greg Claytondda4f7b2010-06-30 23:03:03 +0000592 if (target)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000593 {
Greg Claytondda4f7b2010-06-30 23:03:03 +0000594 target->GetImages().ResolveSymbolContextForAddress (so_addr,
595 eSymbolContextEverything,
596 pointer_sc);
597 if (pointer_sc.function || pointer_sc.symbol)
598 {
599 s->PutCString(": ");
Greg Clayton6dadd502010-09-02 21:44:10 +0000600 pointer_sc.DumpStopContext(s, exe_scope, so_addr, true, false, false);
Greg Claytondda4f7b2010-06-30 23:03:03 +0000601 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000602 }
603 }
604 }
605 }
606 break;
Greg Claytonc982c762010-07-09 20:39:50 +0000607
608 default:
609 break;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000610 }
611 }
612
613 if (!showed_info)
614 {
615 if (module)
616 {
Greg Claytonc9800662010-09-10 01:30:46 +0000617 SymbolContext sc;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000618 module->ResolveSymbolContextForAddress(*this, eSymbolContextEverything, sc);
619 if (sc.function || sc.symbol)
620 {
621 bool show_stop_context = true;
Greg Clayton8dc0a982010-09-07 21:56:53 +0000622 const bool show_module = (style == DumpStyleResolvedDescription);
623 const bool show_fullpaths = false;
Greg Clayton513c26c2011-01-29 07:10:55 +0000624 const bool show_inlined_frames = true;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000625 if (sc.function == NULL && sc.symbol != NULL)
626 {
627 // If we have just a symbol make sure it is in the right section
628 if (sc.symbol->GetAddressRangePtr())
629 {
630 if (sc.symbol->GetAddressRangePtr()->GetBaseAddress().GetSection() != GetSection())
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000631 {
632 // don't show the module if the symbol is a trampoline symbol
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000633 show_stop_context = false;
Greg Clayton54b8b8c2010-07-01 01:26:43 +0000634 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000635 }
636 }
637 if (show_stop_context)
638 {
639 // We have a function or a symbol from the same
640 // sections as this address.
Greg Clayton8dc0a982010-09-07 21:56:53 +0000641 sc.DumpStopContext (s,
642 exe_scope,
643 *this,
644 show_fullpaths,
645 show_module,
646 show_inlined_frames);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000647 }
648 else
649 {
650 // We found a symbol but it was in a different
651 // section so it isn't the symbol we should be
652 // showing, just show the section name + offset
653 Dump (s, exe_scope, DumpStyleSectionNameOffset);
654 }
655 }
656 }
657 }
658 }
659 else
660 {
661 if (fallback_style != DumpStyleInvalid)
Greg Claytondda4f7b2010-06-30 23:03:03 +0000662 return Dump (s, exe_scope, fallback_style, DumpStyleInvalid, addr_size);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000663 return false;
664 }
665 break;
Greg Claytondda4f7b2010-06-30 23:03:03 +0000666
667 case DumpStyleDetailedSymbolContext:
668 if (IsSectionOffset())
669 {
Greg Claytonc9800662010-09-10 01:30:46 +0000670 Module *module = GetModule();
Greg Claytondda4f7b2010-06-30 23:03:03 +0000671 if (module)
672 {
Greg Claytonc9800662010-09-10 01:30:46 +0000673 SymbolContext sc;
Greg Claytondda4f7b2010-06-30 23:03:03 +0000674 module->ResolveSymbolContextForAddress(*this, eSymbolContextEverything, sc);
Greg Claytonb0b9fe62010-08-03 00:35:52 +0000675 if (sc.symbol)
Greg Claytondda4f7b2010-06-30 23:03:03 +0000676 {
Greg Claytonb0b9fe62010-08-03 00:35:52 +0000677 // If we have just a symbol make sure it is in the same section
678 // as our address. If it isn't, then we might have just found
679 // the last symbol that came before the address that we are
680 // looking up that has nothing to do with our address lookup.
681 if (sc.symbol->GetAddressRangePtr() && sc.symbol->GetAddressRangePtr()->GetBaseAddress().GetSection() != GetSection())
682 sc.symbol = NULL;
Greg Claytondda4f7b2010-06-30 23:03:03 +0000683 }
Greg Claytonf5e56de2010-09-14 23:36:40 +0000684 sc.GetDescription(s, eDescriptionLevelBrief, target);
Greg Claytondda4f7b2010-06-30 23:03:03 +0000685 }
686 }
687 if (fallback_style != DumpStyleInvalid)
688 return Dump (s, exe_scope, fallback_style, DumpStyleInvalid, addr_size);
689 return false;
690 break;
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000691 }
692
693 return true;
694}
695
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000696void
697Address::CalculateSymbolContext (SymbolContext *sc)
698{
699 sc->Clear();
700 // Absolute addresses don't have enough information to reconstruct even their target.
701 if (m_section == NULL)
702 return;
703
704 if (m_section->GetModule())
705 {
706 sc->module_sp = m_section->GetModule()->GetSP();
707 if (sc->module_sp)
708 sc->module_sp->ResolveSymbolContextForAddress (*this, eSymbolContextEverything, *sc);
709 }
710}
711
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000712int
713Address::CompareFileAddress (const Address& a, const Address& b)
714{
715 addr_t a_file_addr = a.GetFileAddress();
716 addr_t b_file_addr = b.GetFileAddress();
717 if (a_file_addr < b_file_addr)
718 return -1;
719 if (a_file_addr > b_file_addr)
720 return +1;
721 return 0;
722}
723
724
725int
Greg Claytonf5e56de2010-09-14 23:36:40 +0000726Address::CompareLoadAddress (const Address& a, const Address& b, Target *target)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000727{
Greg Claytonf5e56de2010-09-14 23:36:40 +0000728 assert (target != NULL);
729 addr_t a_load_addr = a.GetLoadAddress (target);
730 addr_t b_load_addr = b.GetLoadAddress (target);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000731 if (a_load_addr < b_load_addr)
732 return -1;
733 if (a_load_addr > b_load_addr)
734 return +1;
735 return 0;
736}
737
738int
739Address::CompareModulePointerAndOffset (const Address& a, const Address& b)
740{
741 Module *a_module = a.GetModule ();
742 Module *b_module = b.GetModule ();
743 if (a_module < b_module)
744 return -1;
745 if (a_module > b_module)
746 return +1;
747 // Modules are the same, just compare the file address since they should
748 // be unique
749 addr_t a_file_addr = a.GetFileAddress();
750 addr_t b_file_addr = b.GetFileAddress();
751 if (a_file_addr < b_file_addr)
752 return -1;
753 if (a_file_addr > b_file_addr)
754 return +1;
755 return 0;
756}
757
758
759size_t
760Address::MemorySize () const
761{
762 // Noting special for the memory size of a single Address object,
763 // it is just the size of itself.
764 return sizeof(Address);
765}
766
767
Greg Claytonb0848c52011-01-08 00:05:12 +0000768//----------------------------------------------------------------------
769// NOTE: Be careful using this operator. It can correctly compare two
770// addresses from the same Module correctly. It can't compare two
771// addresses from different modules in any meaningful way, but it will
772// compare the module pointers.
773//
774// To sum things up:
775// - works great for addresses within the same module
776// - it works for addresses across multiple modules, but don't expect the
777// address results to make much sense
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000778//
Greg Claytonb0848c52011-01-08 00:05:12 +0000779// This basically lets Address objects be used in ordered collection
780// classes.
781//----------------------------------------------------------------------
782
783bool
784lldb_private::operator< (const Address& lhs, const Address& rhs)
785{
786 Module *lhs_module = lhs.GetModule();
787 Module *rhs_module = rhs.GetModule();
788 if (lhs_module == rhs_module)
789 {
790 // Addresses are in the same module, just compare the file addresses
791 return lhs.GetFileAddress() < rhs.GetFileAddress();
792 }
793 else
794 {
795 // The addresses are from different modules, just use the module
796 // pointer value to get consistent ordering
797 return lhs_module < rhs_module;
798 }
799}
800
801bool
802lldb_private::operator> (const Address& lhs, const Address& rhs)
803{
804 Module *lhs_module = lhs.GetModule();
805 Module *rhs_module = rhs.GetModule();
806 if (lhs_module == rhs_module)
807 {
808 // Addresses are in the same module, just compare the file addresses
809 return lhs.GetFileAddress() > rhs.GetFileAddress();
810 }
811 else
812 {
813 // The addresses are from different modules, just use the module
814 // pointer value to get consistent ordering
815 return lhs_module > rhs_module;
816 }
817}
818
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000819
820// The operator == checks for exact equality only (same section, same offset)
821bool
822lldb_private::operator== (const Address& a, const Address& rhs)
823{
824 return a.GetSection() == rhs.GetSection() &&
825 a.GetOffset() == rhs.GetOffset();
826}
827// The operator != checks for exact inequality only (differing section, or
828// different offset)
829bool
830lldb_private::operator!= (const Address& a, const Address& rhs)
831{
832 return a.GetSection() != rhs.GetSection() ||
833 a.GetOffset() != rhs.GetOffset();
834}
835
836bool
837Address::IsLinkedAddress () const
838{
839 return m_section && m_section->GetLinkedSection();
840}
841
842
843void
844Address::ResolveLinkedAddress ()
845{
846 if (m_section)
847 {
848 const Section *linked_section = m_section->GetLinkedSection();
849 if (linked_section)
850 {
851 m_offset += m_section->GetLinkedOffset();
852 m_section = linked_section;
853 }
854 }
855}
Greg Claytonded470d2011-03-19 01:12:21 +0000856
Greg Claytone0d378b2011-03-24 21:19:54 +0000857AddressClass
Greg Claytonded470d2011-03-19 01:12:21 +0000858Address::GetAddressClass () const
859{
860 Module *module = GetModule();
861 if (module)
862 {
863 ObjectFile *obj_file = module->GetObjectFile();
864 if (obj_file)
865 return obj_file->GetAddressClass (GetFileAddress());
866 }
867 return eAddressClassUnknown;
868}
Greg Claytoncd482e32011-05-18 01:58:14 +0000869
870bool
871Address::SetLoadAddress (lldb::addr_t load_addr, Target *target)
872{
873 if (target && target->GetSectionLoadList().ResolveLoadAddress(load_addr, *this))
874 return true;
875 m_section = NULL;
876 m_offset = load_addr;
877 return false;
878}
879