1 //===-- LibCxx.cpp ----------------------------------------------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 #include "lldb/DataFormatters/CXXFormatterFunctions.h"
12 #include "lldb/Core/DataBufferHeap.h"
13 #include "lldb/Core/Debugger.h"
14 #include "lldb/Core/Error.h"
15 #include "lldb/Core/FormatEntity.h"
16 #include "lldb/Core/Stream.h"
17 #include "lldb/Core/ValueObject.h"
18 #include "lldb/Core/ValueObjectConstResult.h"
19 #include "lldb/Host/Endian.h"
20 #include "lldb/Symbol/ClangASTContext.h"
21 #include "lldb/Target/ObjCLanguageRuntime.h"
22 #include "lldb/Target/Target.h"
25 using namespace lldb_private;
26 using namespace lldb_private::formatters;
29 lldb_private::formatters::LibcxxSmartPointerSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
31 ValueObjectSP valobj_sp(valobj.GetNonSyntheticValue());
34 ValueObjectSP ptr_sp(valobj_sp->GetChildMemberWithName(ConstString("__ptr_"), true));
35 ValueObjectSP count_sp(valobj_sp->GetChildAtNamePath( {ConstString("__cntrl_"),ConstString("__shared_owners_")} ));
36 ValueObjectSP weakcount_sp(valobj_sp->GetChildAtNamePath( {ConstString("__cntrl_"),ConstString("__shared_weak_owners_")} ));
41 if (ptr_sp->GetValueAsUnsigned(0) == 0)
43 stream.Printf("nullptr");
48 bool print_pointee = false;
50 ValueObjectSP pointee_sp = ptr_sp->Dereference(error);
51 if (pointee_sp && error.Success())
53 if (pointee_sp->DumpPrintableRepresentation(stream,
54 ValueObject::eValueObjectRepresentationStyleSummary,
56 ValueObject::ePrintableRepresentationSpecialCasesDisable,
61 stream.Printf("ptr = 0x%" PRIx64, ptr_sp->GetValueAsUnsigned(0));
65 stream.Printf(" strong=%" PRIu64, 1+count_sp->GetValueAsUnsigned(0));
68 stream.Printf(" weak=%" PRIu64, 1+weakcount_sp->GetValueAsUnsigned(0));
73 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::LibcxxVectorBoolSyntheticFrontEnd (lldb::ValueObjectSP valobj_sp) :
74 SyntheticChildrenFrontEnd(*valobj_sp.get()),
78 m_base_data_address(0),
84 m_bool_type = valobj_sp->GetClangType().GetBasicTypeFromAST(lldb::eBasicTypeBool);
89 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::CalculateNumChildren ()
95 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::GetChildAtIndex (size_t idx)
97 auto iter = m_children.find(idx),
98 end = m_children.end();
102 return ValueObjectSP();
103 if (m_base_data_address == 0 || m_count == 0)
104 return ValueObjectSP();
106 return ValueObjectSP();
107 size_t byte_idx = (idx >> 3); // divide by 8 to get byte index
108 size_t bit_index = (idx & 7); // efficient idx % 8 for bit index
109 lldb::addr_t byte_location = m_base_data_address + byte_idx;
110 ProcessSP process_sp(m_exe_ctx_ref.GetProcessSP());
112 return ValueObjectSP();
116 size_t bytes_read = process_sp->ReadMemory(byte_location, &byte, 1, err);
117 if (err.Fail() || bytes_read == 0)
118 return ValueObjectSP();
138 return ValueObjectSP();
140 bool bit_set = ((byte & mask) != 0);
141 DataBufferSP buffer_sp(new DataBufferHeap(m_bool_type.GetByteSize(nullptr),0));
142 if (bit_set && buffer_sp && buffer_sp->GetBytes())
143 *(buffer_sp->GetBytes()) = 1; // regardless of endianness, anything non-zero is true
144 StreamString name; name.Printf("[%" PRIu64 "]", (uint64_t)idx);
145 ValueObjectSP retval_sp(CreateValueObjectFromData(name.GetData(), DataExtractor(buffer_sp, process_sp->GetByteOrder(), process_sp->GetAddressByteSize()), m_exe_ctx_ref, m_bool_type));
147 m_children[idx] = retval_sp;
151 /*(std::__1::vector<std::__1::allocator<bool> >) vBool = {
152 __begin_ = 0x00000001001000e0
155 std::__1::__libcpp_compressed_pair_imp<unsigned long, std::__1::allocator<unsigned long> > = {
162 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::Update()
165 ValueObjectSP valobj_sp = m_backend.GetSP();
168 m_exe_ctx_ref = valobj_sp->GetExecutionContextRef();
169 ValueObjectSP size_sp(valobj_sp->GetChildMemberWithName(ConstString("__size_"), true));
172 m_count = size_sp->GetValueAsUnsigned(0);
175 ValueObjectSP begin_sp(valobj_sp->GetChildMemberWithName(ConstString("__begin_"), true));
181 m_base_data_address = begin_sp->GetValueAsUnsigned(0);
182 if (!m_base_data_address)
191 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::MightHaveChildren ()
197 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::GetIndexOfChildWithName (const ConstString &name)
199 if (!m_count || !m_base_data_address)
201 const char* item_name = name.GetCString();
202 uint32_t idx = ExtractIndexFromString(item_name);
203 if (idx < UINT32_MAX && idx >= CalculateNumChildren())
208 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::~LibcxxVectorBoolSyntheticFrontEnd ()
211 SyntheticChildrenFrontEnd*
212 lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
216 return (new LibcxxVectorBoolSyntheticFrontEnd(valobj_sp));
220 (lldb) fr var ibeg --raw --ptr-depth 1
221 (std::__1::__map_iterator<std::__1::__tree_iterator<std::__1::pair<int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > >, std::__1::__tree_node<std::__1::pair<int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > >, void *> *, long> >) ibeg = {
223 __ptr_ = 0x0000000100103870 {
224 std::__1::__tree_node_base<void *> = {
225 std::__1::__tree_end_node<std::__1::__tree_node_base<void *> *> = {
226 __left_ = 0x0000000000000000
228 __right_ = 0x0000000000000000
229 __parent_ = 0x00000001001038b0
234 second = { std::string }
237 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::LibCxxMapIteratorSyntheticFrontEnd (lldb::ValueObjectSP valobj_sp) :
238 SyntheticChildrenFrontEnd(*valobj_sp.get()),
246 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::Update()
248 ValueObjectSP valobj_sp = m_backend.GetSP();
252 TargetSP target_sp(valobj_sp->GetTargetSP());
260 // this must be a ValueObject* because it is a child of the ValueObject we are producing children for
261 // it if were a ValueObjectSP, we would end up with a loop (iterator -> synthetic -> child -> parent == iterator)
262 // and that would in turn leak memory by never allowing the ValueObjects to die and free their memory
263 m_pair_ptr = valobj_sp->GetValueForExpressionPath(".__i_.__ptr_->__value_",
267 ValueObject::GetValueForExpressionPathOptions().DontCheckDotVsArrowSyntax().SetSyntheticChildrenTraversal(ValueObject::GetValueForExpressionPathOptions::SyntheticChildrenTraversal::None),
274 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::CalculateNumChildren ()
280 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::GetChildAtIndex (size_t idx)
283 return lldb::ValueObjectSP();
284 return m_pair_ptr->GetChildAtIndex(idx, true);
288 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::MightHaveChildren ()
294 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::GetIndexOfChildWithName (const ConstString &name)
296 if (name == ConstString("first"))
298 if (name == ConstString("second"))
303 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::~LibCxxMapIteratorSyntheticFrontEnd ()
305 // this will be deleted when its parent dies (since it's a child object)
309 SyntheticChildrenFrontEnd*
310 lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
314 return (new LibCxxMapIteratorSyntheticFrontEnd(valobj_sp));
318 (lldb) fr var ibeg --raw --ptr-depth 1 -T
319 (std::__1::__wrap_iter<int *>) ibeg = {
320 (std::__1::__wrap_iter<int *>::iterator_type) __i = 0x00000001001037a0 {
326 SyntheticChildrenFrontEnd*
327 lldb_private::formatters::LibCxxVectorIteratorSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
329 static ConstString g_item_name;
331 g_item_name.SetCString("__i");
334 return (new VectorIteratorSyntheticFrontEnd(valobj_sp,g_item_name));
337 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::LibcxxSharedPtrSyntheticFrontEnd (lldb::ValueObjectSP valobj_sp) :
338 SyntheticChildrenFrontEnd(*valobj_sp.get()),
343 m_byte_order(lldb::eByteOrderInvalid)
350 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::CalculateNumChildren ()
352 return (m_cntrl ? 1 : 0);
356 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::GetChildAtIndex (size_t idx)
359 return lldb::ValueObjectSP();
361 ValueObjectSP valobj_sp = m_backend.GetSP();
363 return lldb::ValueObjectSP();
366 return valobj_sp->GetChildMemberWithName(ConstString("__ptr_"), true);
369 return lldb::ValueObjectSP();
375 ValueObjectSP shared_owners_sp(m_cntrl->GetChildMemberWithName(ConstString("__shared_owners_"),true));
376 if (!shared_owners_sp)
377 return lldb::ValueObjectSP();
378 uint64_t count = 1 + shared_owners_sp->GetValueAsUnsigned(0);
379 DataExtractor data(&count, 8, m_byte_order, m_ptr_size);
380 m_count_sp = CreateValueObjectFromData("count", data, valobj_sp->GetExecutionContextRef(), shared_owners_sp->GetClangType());
384 else /* if (idx == 2) */
386 if (!m_weak_count_sp)
388 ValueObjectSP shared_weak_owners_sp(m_cntrl->GetChildMemberWithName(ConstString("__shared_weak_owners_"),true));
389 if (!shared_weak_owners_sp)
390 return lldb::ValueObjectSP();
391 uint64_t count = 1 + shared_weak_owners_sp->GetValueAsUnsigned(0);
392 DataExtractor data(&count, 8, m_byte_order, m_ptr_size);
393 m_weak_count_sp = CreateValueObjectFromData("count", data, valobj_sp->GetExecutionContextRef(), shared_weak_owners_sp->GetClangType());
395 return m_weak_count_sp;
400 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::Update()
403 m_weak_count_sp.reset();
406 ValueObjectSP valobj_sp = m_backend.GetSP();
410 TargetSP target_sp(valobj_sp->GetTargetSP());
414 m_byte_order = target_sp->GetArchitecture().GetByteOrder();
415 m_ptr_size = target_sp->GetArchitecture().GetAddressByteSize();
417 lldb::ValueObjectSP cntrl_sp(valobj_sp->GetChildMemberWithName(ConstString("__cntrl_"),true));
419 m_cntrl = cntrl_sp.get(); // need to store the raw pointer to avoid a circular dependency
424 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::MightHaveChildren ()
430 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::GetIndexOfChildWithName (const ConstString &name)
432 if (name == ConstString("__ptr_"))
434 if (name == ConstString("count"))
436 if (name == ConstString("weak_count"))
441 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::~LibcxxSharedPtrSyntheticFrontEnd ()
444 SyntheticChildrenFrontEnd*
445 lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
449 return (new LibcxxSharedPtrSyntheticFrontEnd(valobj_sp));
453 lldb_private::formatters::LibcxxContainerSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
455 if (valobj.IsPointerType())
457 uint64_t value = valobj.GetValueAsUnsigned(0);
460 stream.Printf("0x%016" PRIx64 " ", value);
462 return FormatEntity::FormatStringRef("size=${svar%#}", stream, NULL, NULL, NULL, &valobj, false, false);