aboutsummaryrefslogtreecommitdiff
path: root/contrib/llvm-project/lldb/source/API/SBAddress.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'contrib/llvm-project/lldb/source/API/SBAddress.cpp')
-rw-r--r--contrib/llvm-project/lldb/source/API/SBAddress.cpp268
1 files changed, 268 insertions, 0 deletions
diff --git a/contrib/llvm-project/lldb/source/API/SBAddress.cpp b/contrib/llvm-project/lldb/source/API/SBAddress.cpp
new file mode 100644
index 000000000000..e519f0bcc83c
--- /dev/null
+++ b/contrib/llvm-project/lldb/source/API/SBAddress.cpp
@@ -0,0 +1,268 @@
+//===-- SBAddress.cpp -----------------------------------------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include "lldb/API/SBAddress.h"
+#include "Utils.h"
+#include "lldb/API/SBProcess.h"
+#include "lldb/API/SBSection.h"
+#include "lldb/API/SBStream.h"
+#include "lldb/Core/Address.h"
+#include "lldb/Core/Module.h"
+#include "lldb/Symbol/LineEntry.h"
+#include "lldb/Target/Target.h"
+#include "lldb/Utility/Instrumentation.h"
+#include "lldb/Utility/StreamString.h"
+
+using namespace lldb;
+using namespace lldb_private;
+
+SBAddress::SBAddress() : m_opaque_up(new Address()) {
+ LLDB_INSTRUMENT_VA(this);
+}
+
+SBAddress::SBAddress(const Address &address)
+ : m_opaque_up(std::make_unique<Address>(address)) {}
+
+SBAddress::SBAddress(const SBAddress &rhs) : m_opaque_up(new Address()) {
+ LLDB_INSTRUMENT_VA(this, rhs);
+
+ m_opaque_up = clone(rhs.m_opaque_up);
+}
+
+SBAddress::SBAddress(lldb::SBSection section, lldb::addr_t offset)
+ : m_opaque_up(new Address(section.GetSP(), offset)) {
+ LLDB_INSTRUMENT_VA(this, section, offset);
+}
+
+// Create an address by resolving a load address using the supplied target
+SBAddress::SBAddress(lldb::addr_t load_addr, lldb::SBTarget &target)
+ : m_opaque_up(new Address()) {
+ LLDB_INSTRUMENT_VA(this, load_addr, target);
+
+ SetLoadAddress(load_addr, target);
+}
+
+SBAddress::~SBAddress() = default;
+
+const SBAddress &SBAddress::operator=(const SBAddress &rhs) {
+ LLDB_INSTRUMENT_VA(this, rhs);
+
+ if (this != &rhs)
+ m_opaque_up = clone(rhs.m_opaque_up);
+ return *this;
+}
+
+bool lldb::operator==(const SBAddress &lhs, const SBAddress &rhs) {
+ if (lhs.IsValid() && rhs.IsValid())
+ return lhs.ref() == rhs.ref();
+ return false;
+}
+
+bool SBAddress::operator!=(const SBAddress &rhs) const {
+ LLDB_INSTRUMENT_VA(this, rhs);
+
+ return !(*this == rhs);
+}
+
+bool SBAddress::IsValid() const {
+ LLDB_INSTRUMENT_VA(this);
+ return this->operator bool();
+}
+SBAddress::operator bool() const {
+ LLDB_INSTRUMENT_VA(this);
+
+ return m_opaque_up != nullptr && m_opaque_up->IsValid();
+}
+
+void SBAddress::Clear() {
+ LLDB_INSTRUMENT_VA(this);
+
+ m_opaque_up = std::make_unique<Address>();
+}
+
+void SBAddress::SetAddress(lldb::SBSection section, lldb::addr_t offset) {
+ LLDB_INSTRUMENT_VA(this, section, offset);
+
+ Address &addr = ref();
+ addr.SetSection(section.GetSP());
+ addr.SetOffset(offset);
+}
+
+void SBAddress::SetAddress(const Address &address) { ref() = address; }
+
+lldb::addr_t SBAddress::GetFileAddress() const {
+ LLDB_INSTRUMENT_VA(this);
+
+ if (m_opaque_up->IsValid())
+ return m_opaque_up->GetFileAddress();
+ else
+ return LLDB_INVALID_ADDRESS;
+}
+
+lldb::addr_t SBAddress::GetLoadAddress(const SBTarget &target) const {
+ LLDB_INSTRUMENT_VA(this, target);
+
+ lldb::addr_t addr = LLDB_INVALID_ADDRESS;
+ TargetSP target_sp(target.GetSP());
+ if (target_sp) {
+ if (m_opaque_up->IsValid()) {
+ std::lock_guard<std::recursive_mutex> guard(target_sp->GetAPIMutex());
+ addr = m_opaque_up->GetLoadAddress(target_sp.get());
+ }
+ }
+
+ return addr;
+}
+
+void SBAddress::SetLoadAddress(lldb::addr_t load_addr, lldb::SBTarget &target) {
+ LLDB_INSTRUMENT_VA(this, load_addr, target);
+
+ // Create the address object if we don't already have one
+ ref();
+ if (target.IsValid())
+ *this = target.ResolveLoadAddress(load_addr);
+ else
+ m_opaque_up->Clear();
+
+ // 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
+ if (!m_opaque_up->IsValid())
+ m_opaque_up->SetOffset(load_addr);
+}
+
+bool SBAddress::OffsetAddress(addr_t offset) {
+ LLDB_INSTRUMENT_VA(this, offset);
+
+ if (m_opaque_up->IsValid()) {
+ addr_t addr_offset = m_opaque_up->GetOffset();
+ if (addr_offset != LLDB_INVALID_ADDRESS) {
+ m_opaque_up->SetOffset(addr_offset + offset);
+ return true;
+ }
+ }
+ return false;
+}
+
+lldb::SBSection SBAddress::GetSection() {
+ LLDB_INSTRUMENT_VA(this);
+
+ lldb::SBSection sb_section;
+ if (m_opaque_up->IsValid())
+ sb_section.SetSP(m_opaque_up->GetSection());
+ return sb_section;
+}
+
+lldb::addr_t SBAddress::GetOffset() {
+ LLDB_INSTRUMENT_VA(this);
+
+ if (m_opaque_up->IsValid())
+ return m_opaque_up->GetOffset();
+ return 0;
+}
+
+Address *SBAddress::operator->() { return m_opaque_up.get(); }
+
+const Address *SBAddress::operator->() const { return m_opaque_up.get(); }
+
+Address &SBAddress::ref() {
+ if (m_opaque_up == nullptr)
+ m_opaque_up = std::make_unique<Address>();
+ return *m_opaque_up;
+}
+
+const Address &SBAddress::ref() const {
+ // 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.
+ assert(m_opaque_up.get());
+ return *m_opaque_up;
+}
+
+Address *SBAddress::get() { return m_opaque_up.get(); }
+
+bool SBAddress::GetDescription(SBStream &description) {
+ LLDB_INSTRUMENT_VA(this, 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_up->IsValid()) {
+ m_opaque_up->Dump(&strm, nullptr, Address::DumpStyleResolvedDescription,
+ Address::DumpStyleModuleWithFileAddress, 4);
+ } else
+ strm.PutCString("No value");
+
+ return true;
+}
+
+SBModule SBAddress::GetModule() {
+ LLDB_INSTRUMENT_VA(this);
+
+ SBModule sb_module;
+ if (m_opaque_up->IsValid())
+ sb_module.SetSP(m_opaque_up->GetModule());
+ return sb_module;
+}
+
+SBSymbolContext SBAddress::GetSymbolContext(uint32_t resolve_scope) {
+ LLDB_INSTRUMENT_VA(this, resolve_scope);
+
+ SBSymbolContext sb_sc;
+ SymbolContextItem scope = static_cast<SymbolContextItem>(resolve_scope);
+ if (m_opaque_up->IsValid())
+ m_opaque_up->CalculateSymbolContext(&sb_sc.ref(), scope);
+ return sb_sc;
+}
+
+SBCompileUnit SBAddress::GetCompileUnit() {
+ LLDB_INSTRUMENT_VA(this);
+
+ SBCompileUnit sb_comp_unit;
+ if (m_opaque_up->IsValid())
+ sb_comp_unit.reset(m_opaque_up->CalculateSymbolContextCompileUnit());
+ return sb_comp_unit;
+}
+
+SBFunction SBAddress::GetFunction() {
+ LLDB_INSTRUMENT_VA(this);
+
+ SBFunction sb_function;
+ if (m_opaque_up->IsValid())
+ sb_function.reset(m_opaque_up->CalculateSymbolContextFunction());
+ return sb_function;
+}
+
+SBBlock SBAddress::GetBlock() {
+ LLDB_INSTRUMENT_VA(this);
+
+ SBBlock sb_block;
+ if (m_opaque_up->IsValid())
+ sb_block.SetPtr(m_opaque_up->CalculateSymbolContextBlock());
+ return sb_block;
+}
+
+SBSymbol SBAddress::GetSymbol() {
+ LLDB_INSTRUMENT_VA(this);
+
+ SBSymbol sb_symbol;
+ if (m_opaque_up->IsValid())
+ sb_symbol.reset(m_opaque_up->CalculateSymbolContextSymbol());
+ return sb_symbol;
+}
+
+SBLineEntry SBAddress::GetLineEntry() {
+ LLDB_INSTRUMENT_VA(this);
+
+ SBLineEntry sb_line_entry;
+ if (m_opaque_up->IsValid()) {
+ LineEntry line_entry;
+ if (m_opaque_up->CalculateSymbolContextLineEntry(line_entry))
+ sb_line_entry.SetLineEntry(line_entry);
+ }
+ return sb_line_entry;
+}