2010-06-09 00:52:24 +08:00
|
|
|
//===-- SBAddress.cpp -------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2010-06-09 00:52:24 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "lldb/API/SBAddress.h"
|
|
|
|
#include "lldb/API/SBProcess.h"
|
2011-09-24 08:52:29 +08:00
|
|
|
#include "lldb/API/SBSection.h"
|
2010-09-20 13:20:02 +08:00
|
|
|
#include "lldb/API/SBStream.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Core/Address.h"
|
2011-03-31 09:08:07 +08:00
|
|
|
#include "lldb/Core/Module.h"
|
2015-03-04 03:23:09 +08:00
|
|
|
#include "lldb/Symbol/LineEntry.h"
|
2010-12-21 04:49:23 +08:00
|
|
|
#include "lldb/Target/Target.h"
|
2017-03-04 04:56:28 +08:00
|
|
|
#include "lldb/Utility/Log.h"
|
2017-02-03 05:39:50 +08:00
|
|
|
#include "lldb/Utility/StreamString.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
using namespace lldb;
|
2010-10-26 11:11:13 +08:00
|
|
|
using namespace lldb_private;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBAddress::SBAddress() : m_opaque_ap(new Address()) {}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBAddress::SBAddress(const Address *lldb_object_ptr)
|
|
|
|
: m_opaque_ap(new Address()) {
|
|
|
|
if (lldb_object_ptr)
|
|
|
|
ref() = *lldb_object_ptr;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBAddress::SBAddress(const SBAddress &rhs) : m_opaque_ap(new Address()) {
|
|
|
|
if (rhs.IsValid())
|
|
|
|
ref() = rhs.ref();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBAddress::SBAddress(lldb::SBSection section, lldb::addr_t offset)
|
|
|
|
: m_opaque_ap(new Address(section.GetSP(), offset)) {}
|
2012-02-04 10:58:17 +08:00
|
|
|
|
2011-07-23 00:46:35 +08:00
|
|
|
// Create an address by resolving a load address using the supplied target
|
2016-09-07 04:57:50 +08:00
|
|
|
SBAddress::SBAddress(lldb::addr_t load_addr, lldb::SBTarget &target)
|
|
|
|
: m_opaque_ap(new Address()) {
|
|
|
|
SetLoadAddress(load_addr, target);
|
2011-07-23 00:46:35 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBAddress::~SBAddress() {}
|
2011-07-23 00:46:35 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
const SBAddress &SBAddress::operator=(const SBAddress &rhs) {
|
|
|
|
if (this != &rhs) {
|
|
|
|
if (rhs.IsValid())
|
|
|
|
ref() = rhs.ref();
|
|
|
|
else
|
|
|
|
m_opaque_ap.reset(new Address());
|
|
|
|
}
|
|
|
|
return *this;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2017-05-04 19:34:42 +08:00
|
|
|
bool lldb::operator==(const SBAddress &lhs, const SBAddress &rhs) {
|
|
|
|
if (lhs.IsValid() && rhs.IsValid())
|
|
|
|
return lhs.ref() == rhs.ref();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
bool SBAddress::IsValid() const {
|
2018-12-21 05:02:55 +08:00
|
|
|
return m_opaque_ap != NULL && m_opaque_ap->IsValid();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
void SBAddress::Clear() { m_opaque_ap.reset(new Address()); }
|
2010-09-11 02:31:35 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
void SBAddress::SetAddress(lldb::SBSection section, lldb::addr_t offset) {
|
|
|
|
Address &addr = ref();
|
|
|
|
addr.SetSection(section.GetSP());
|
|
|
|
addr.SetOffset(offset);
|
2012-02-04 10:58:17 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
void SBAddress::SetAddress(const Address *lldb_object_ptr) {
|
|
|
|
if (lldb_object_ptr)
|
|
|
|
ref() = *lldb_object_ptr;
|
|
|
|
else
|
|
|
|
m_opaque_ap.reset(new Address());
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
lldb::addr_t SBAddress::GetFileAddress() const {
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
return m_opaque_ap->GetFileAddress();
|
|
|
|
else
|
|
|
|
return LLDB_INVALID_ADDRESS;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
lldb::addr_t SBAddress::GetLoadAddress(const SBTarget &target) const {
|
|
|
|
Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_API));
|
2014-04-04 12:06:10 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
lldb::addr_t addr = LLDB_INVALID_ADDRESS;
|
|
|
|
TargetSP target_sp(target.GetSP());
|
|
|
|
if (target_sp) {
|
|
|
|
if (m_opaque_ap->IsValid()) {
|
|
|
|
std::lock_guard<std::recursive_mutex> guard(target_sp->GetAPIMutex());
|
|
|
|
addr = m_opaque_ap->GetLoadAddress(target_sp.get());
|
2010-10-26 11:11:13 +08:00
|
|
|
}
|
2016-09-07 04:57:50 +08:00
|
|
|
}
|
2010-12-21 04:49:23 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
if (log) {
|
|
|
|
if (addr == LLDB_INVALID_ADDRESS)
|
|
|
|
log->Printf(
|
|
|
|
"SBAddress::GetLoadAddress (SBTarget(%p)) => LLDB_INVALID_ADDRESS",
|
|
|
|
static_cast<void *>(target_sp.get()));
|
2011-07-23 00:46:35 +08:00
|
|
|
else
|
2016-09-07 04:57:50 +08:00
|
|
|
log->Printf("SBAddress::GetLoadAddress (SBTarget(%p)) => 0x%" PRIx64,
|
|
|
|
static_cast<void *>(target_sp.get()), addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SBAddress::SetLoadAddress(lldb::addr_t load_addr, lldb::SBTarget &target) {
|
|
|
|
// Create the address object if we don't already have one
|
|
|
|
ref();
|
|
|
|
if (target.IsValid())
|
|
|
|
*this = target.ResolveLoadAddress(load_addr);
|
|
|
|
else
|
|
|
|
m_opaque_ap->Clear();
|
|
|
|
|
2018-05-01 00:49:04 +08:00
|
|
|
// Check if we weren't were able to resolve a section offset address. If we
|
|
|
|
// weren't it is ok, the load address might be a location on the stack or
|
|
|
|
// heap, so we should just have an address with no section and a valid offset
|
2016-09-07 04:57:50 +08:00
|
|
|
if (!m_opaque_ap->IsValid())
|
|
|
|
m_opaque_ap->SetOffset(load_addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SBAddress::OffsetAddress(addr_t offset) {
|
|
|
|
if (m_opaque_ap->IsValid()) {
|
|
|
|
addr_t addr_offset = m_opaque_ap->GetOffset();
|
|
|
|
if (addr_offset != LLDB_INVALID_ADDRESS) {
|
|
|
|
m_opaque_ap->SetOffset(addr_offset + offset);
|
|
|
|
return true;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2016-09-07 04:57:50 +08:00
|
|
|
}
|
|
|
|
return false;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
lldb::SBSection SBAddress::GetSection() {
|
|
|
|
lldb::SBSection sb_section;
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
sb_section.SetSP(m_opaque_ap->GetSection());
|
|
|
|
return sb_section;
|
2011-09-24 08:52:29 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
lldb::addr_t SBAddress::GetOffset() {
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
return m_opaque_ap->GetOffset();
|
|
|
|
return 0;
|
2012-01-29 14:07:39 +08:00
|
|
|
}
|
2011-09-24 08:52:29 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
Address *SBAddress::operator->() { return m_opaque_ap.get(); }
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
const Address *SBAddress::operator->() const { return m_opaque_ap.get(); }
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
Address &SBAddress::ref() {
|
2018-12-21 05:02:55 +08:00
|
|
|
if (m_opaque_ap == NULL)
|
2016-09-07 04:57:50 +08:00
|
|
|
m_opaque_ap.reset(new Address());
|
|
|
|
return *m_opaque_ap;
|
2010-09-11 02:31:35 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
const Address &SBAddress::ref() const {
|
2018-05-01 00:49:04 +08:00
|
|
|
// This object should already have checked with "IsValid()" prior to calling
|
|
|
|
// this function. In case you didn't we will assert and die to let you know.
|
2016-09-07 04:57:50 +08:00
|
|
|
assert(m_opaque_ap.get());
|
|
|
|
return *m_opaque_ap;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
Address *SBAddress::get() { return m_opaque_ap.get(); }
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
bool SBAddress::GetDescription(SBStream &description) {
|
|
|
|
// Call "ref()" on the stream to make sure it creates a backing stream in
|
|
|
|
// case there isn't one already...
|
|
|
|
Stream &strm = description.ref();
|
|
|
|
if (m_opaque_ap->IsValid()) {
|
|
|
|
m_opaque_ap->Dump(&strm, NULL, Address::DumpStyleResolvedDescription,
|
|
|
|
Address::DumpStyleModuleWithFileAddress, 4);
|
|
|
|
StreamString sstrm;
|
|
|
|
// m_opaque_ap->Dump (&sstrm, NULL,
|
|
|
|
// Address::DumpStyleResolvedDescription, Address::DumpStyleInvalid,
|
|
|
|
// 4);
|
|
|
|
// if (sstrm.GetData())
|
|
|
|
// strm.Printf (" (%s)", sstrm.GetData());
|
|
|
|
} else
|
|
|
|
strm.PutCString("No value");
|
2010-09-20 13:20:02 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
return true;
|
2010-09-20 13:20:02 +08:00
|
|
|
}
|
2011-03-31 09:08:07 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBModule SBAddress::GetModule() {
|
|
|
|
SBModule sb_module;
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
sb_module.SetSP(m_opaque_ap->GetModule());
|
|
|
|
return sb_module;
|
2011-03-31 09:08:07 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBSymbolContext SBAddress::GetSymbolContext(uint32_t resolve_scope) {
|
|
|
|
SBSymbolContext sb_sc;
|
2018-10-26 04:45:19 +08:00
|
|
|
SymbolContextItem scope = static_cast<SymbolContextItem>(resolve_scope);
|
2016-09-07 04:57:50 +08:00
|
|
|
if (m_opaque_ap->IsValid())
|
2018-10-26 04:45:19 +08:00
|
|
|
m_opaque_ap->CalculateSymbolContext(&sb_sc.ref(), scope);
|
2016-09-07 04:57:50 +08:00
|
|
|
return sb_sc;
|
2011-08-13 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBCompileUnit SBAddress::GetCompileUnit() {
|
|
|
|
SBCompileUnit sb_comp_unit;
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
sb_comp_unit.reset(m_opaque_ap->CalculateSymbolContextCompileUnit());
|
|
|
|
return sb_comp_unit;
|
2011-08-13 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBFunction SBAddress::GetFunction() {
|
|
|
|
SBFunction sb_function;
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
sb_function.reset(m_opaque_ap->CalculateSymbolContextFunction());
|
|
|
|
return sb_function;
|
2011-08-13 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBBlock SBAddress::GetBlock() {
|
|
|
|
SBBlock sb_block;
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
sb_block.SetPtr(m_opaque_ap->CalculateSymbolContextBlock());
|
|
|
|
return sb_block;
|
2011-08-13 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBSymbol SBAddress::GetSymbol() {
|
|
|
|
SBSymbol sb_symbol;
|
|
|
|
if (m_opaque_ap->IsValid())
|
|
|
|
sb_symbol.reset(m_opaque_ap->CalculateSymbolContextSymbol());
|
|
|
|
return sb_symbol;
|
2011-08-13 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
SBLineEntry SBAddress::GetLineEntry() {
|
|
|
|
SBLineEntry sb_line_entry;
|
|
|
|
if (m_opaque_ap->IsValid()) {
|
|
|
|
LineEntry line_entry;
|
|
|
|
if (m_opaque_ap->CalculateSymbolContextLineEntry(line_entry))
|
|
|
|
sb_line_entry.SetLineEntry(line_entry);
|
|
|
|
}
|
|
|
|
return sb_line_entry;
|
2011-08-13 05:40:01 +08:00
|
|
|
}
|