]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/llvm/tools/lldb/source/API/SBAddress.cpp
Merge ^/head r320573 through r320970.
[FreeBSD/FreeBSD.git] / contrib / llvm / tools / lldb / source / API / SBAddress.cpp
1 //===-- SBAddress.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/API/SBAddress.h"
11 #include "lldb/API/SBProcess.h"
12 #include "lldb/API/SBSection.h"
13 #include "lldb/API/SBStream.h"
14 #include "lldb/Core/Address.h"
15 #include "lldb/Core/Module.h"
16 #include "lldb/Symbol/LineEntry.h"
17 #include "lldb/Target/Target.h"
18 #include "lldb/Utility/Log.h"
19 #include "lldb/Utility/StreamString.h"
20
21 using namespace lldb;
22 using namespace lldb_private;
23
24 SBAddress::SBAddress() : m_opaque_ap(new Address()) {}
25
26 SBAddress::SBAddress(const Address *lldb_object_ptr)
27     : m_opaque_ap(new Address()) {
28   if (lldb_object_ptr)
29     ref() = *lldb_object_ptr;
30 }
31
32 SBAddress::SBAddress(const SBAddress &rhs) : m_opaque_ap(new Address()) {
33   if (rhs.IsValid())
34     ref() = rhs.ref();
35 }
36
37 SBAddress::SBAddress(lldb::SBSection section, lldb::addr_t offset)
38     : m_opaque_ap(new Address(section.GetSP(), offset)) {}
39
40 // Create an address by resolving a load address using the supplied target
41 SBAddress::SBAddress(lldb::addr_t load_addr, lldb::SBTarget &target)
42     : m_opaque_ap(new Address()) {
43   SetLoadAddress(load_addr, target);
44 }
45
46 SBAddress::~SBAddress() {}
47
48 const SBAddress &SBAddress::operator=(const SBAddress &rhs) {
49   if (this != &rhs) {
50     if (rhs.IsValid())
51       ref() = rhs.ref();
52     else
53       m_opaque_ap.reset(new Address());
54   }
55   return *this;
56 }
57
58 bool lldb::operator==(const SBAddress &lhs, const SBAddress &rhs) {
59   if (lhs.IsValid() && rhs.IsValid())
60     return lhs.ref() == rhs.ref();
61   return false;
62 }
63
64 bool SBAddress::IsValid() const {
65   return m_opaque_ap.get() != NULL && m_opaque_ap->IsValid();
66 }
67
68 void SBAddress::Clear() { m_opaque_ap.reset(new Address()); }
69
70 void SBAddress::SetAddress(lldb::SBSection section, lldb::addr_t offset) {
71   Address &addr = ref();
72   addr.SetSection(section.GetSP());
73   addr.SetOffset(offset);
74 }
75
76 void SBAddress::SetAddress(const Address *lldb_object_ptr) {
77   if (lldb_object_ptr)
78     ref() = *lldb_object_ptr;
79   else
80     m_opaque_ap.reset(new Address());
81 }
82
83 lldb::addr_t SBAddress::GetFileAddress() const {
84   if (m_opaque_ap->IsValid())
85     return m_opaque_ap->GetFileAddress();
86   else
87     return LLDB_INVALID_ADDRESS;
88 }
89
90 lldb::addr_t SBAddress::GetLoadAddress(const SBTarget &target) const {
91   Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_API));
92
93   lldb::addr_t addr = LLDB_INVALID_ADDRESS;
94   TargetSP target_sp(target.GetSP());
95   if (target_sp) {
96     if (m_opaque_ap->IsValid()) {
97       std::lock_guard<std::recursive_mutex> guard(target_sp->GetAPIMutex());
98       addr = m_opaque_ap->GetLoadAddress(target_sp.get());
99     }
100   }
101
102   if (log) {
103     if (addr == LLDB_INVALID_ADDRESS)
104       log->Printf(
105           "SBAddress::GetLoadAddress (SBTarget(%p)) => LLDB_INVALID_ADDRESS",
106           static_cast<void *>(target_sp.get()));
107     else
108       log->Printf("SBAddress::GetLoadAddress (SBTarget(%p)) => 0x%" PRIx64,
109                   static_cast<void *>(target_sp.get()), addr);
110   }
111
112   return addr;
113 }
114
115 void SBAddress::SetLoadAddress(lldb::addr_t load_addr, lldb::SBTarget &target) {
116   // Create the address object if we don't already have one
117   ref();
118   if (target.IsValid())
119     *this = target.ResolveLoadAddress(load_addr);
120   else
121     m_opaque_ap->Clear();
122
123   // Check if we weren't were able to resolve a section offset address.
124   // If we weren't it is ok, the load address might be a location on the
125   // stack or heap, so we should just have an address with no section and
126   // a valid offset
127   if (!m_opaque_ap->IsValid())
128     m_opaque_ap->SetOffset(load_addr);
129 }
130
131 bool SBAddress::OffsetAddress(addr_t offset) {
132   if (m_opaque_ap->IsValid()) {
133     addr_t addr_offset = m_opaque_ap->GetOffset();
134     if (addr_offset != LLDB_INVALID_ADDRESS) {
135       m_opaque_ap->SetOffset(addr_offset + offset);
136       return true;
137     }
138   }
139   return false;
140 }
141
142 lldb::SBSection SBAddress::GetSection() {
143   lldb::SBSection sb_section;
144   if (m_opaque_ap->IsValid())
145     sb_section.SetSP(m_opaque_ap->GetSection());
146   return sb_section;
147 }
148
149 lldb::addr_t SBAddress::GetOffset() {
150   if (m_opaque_ap->IsValid())
151     return m_opaque_ap->GetOffset();
152   return 0;
153 }
154
155 Address *SBAddress::operator->() { return m_opaque_ap.get(); }
156
157 const Address *SBAddress::operator->() const { return m_opaque_ap.get(); }
158
159 Address &SBAddress::ref() {
160   if (m_opaque_ap.get() == NULL)
161     m_opaque_ap.reset(new Address());
162   return *m_opaque_ap;
163 }
164
165 const Address &SBAddress::ref() const {
166   // This object should already have checked with "IsValid()"
167   // prior to calling this function. In case you didn't we will assert
168   // and die to let you know.
169   assert(m_opaque_ap.get());
170   return *m_opaque_ap;
171 }
172
173 Address *SBAddress::get() { return m_opaque_ap.get(); }
174
175 bool SBAddress::GetDescription(SBStream &description) {
176   // Call "ref()" on the stream to make sure it creates a backing stream in
177   // case there isn't one already...
178   Stream &strm = description.ref();
179   if (m_opaque_ap->IsValid()) {
180     m_opaque_ap->Dump(&strm, NULL, Address::DumpStyleResolvedDescription,
181                       Address::DumpStyleModuleWithFileAddress, 4);
182     StreamString sstrm;
183     //        m_opaque_ap->Dump (&sstrm, NULL,
184     //        Address::DumpStyleResolvedDescription, Address::DumpStyleInvalid,
185     //        4);
186     //        if (sstrm.GetData())
187     //            strm.Printf (" (%s)", sstrm.GetData());
188   } else
189     strm.PutCString("No value");
190
191   return true;
192 }
193
194 SBModule SBAddress::GetModule() {
195   SBModule sb_module;
196   if (m_opaque_ap->IsValid())
197     sb_module.SetSP(m_opaque_ap->GetModule());
198   return sb_module;
199 }
200
201 SBSymbolContext SBAddress::GetSymbolContext(uint32_t resolve_scope) {
202   SBSymbolContext sb_sc;
203   if (m_opaque_ap->IsValid())
204     m_opaque_ap->CalculateSymbolContext(&sb_sc.ref(), resolve_scope);
205   return sb_sc;
206 }
207
208 SBCompileUnit SBAddress::GetCompileUnit() {
209   SBCompileUnit sb_comp_unit;
210   if (m_opaque_ap->IsValid())
211     sb_comp_unit.reset(m_opaque_ap->CalculateSymbolContextCompileUnit());
212   return sb_comp_unit;
213 }
214
215 SBFunction SBAddress::GetFunction() {
216   SBFunction sb_function;
217   if (m_opaque_ap->IsValid())
218     sb_function.reset(m_opaque_ap->CalculateSymbolContextFunction());
219   return sb_function;
220 }
221
222 SBBlock SBAddress::GetBlock() {
223   SBBlock sb_block;
224   if (m_opaque_ap->IsValid())
225     sb_block.SetPtr(m_opaque_ap->CalculateSymbolContextBlock());
226   return sb_block;
227 }
228
229 SBSymbol SBAddress::GetSymbol() {
230   SBSymbol sb_symbol;
231   if (m_opaque_ap->IsValid())
232     sb_symbol.reset(m_opaque_ap->CalculateSymbolContextSymbol());
233   return sb_symbol;
234 }
235
236 SBLineEntry SBAddress::GetLineEntry() {
237   SBLineEntry sb_line_entry;
238   if (m_opaque_ap->IsValid()) {
239     LineEntry line_entry;
240     if (m_opaque_ap->CalculateSymbolContextLineEntry(line_entry))
241       sb_line_entry.SetLineEntry(line_entry);
242   }
243   return sb_line_entry;
244 }
245
246 AddressClass SBAddress::GetAddressClass() {
247   if (m_opaque_ap->IsValid())
248     return m_opaque_ap->GetAddressClass();
249   return eAddressClassInvalid;
250 }