blob: 7395d93bdd31531f80e2797889a4d6454af955b6 [file] [log] [blame]
Greg Claytonbb1beaa2013-03-26 18:42:13 +00001//===-- LibCxx.cpp ----------------------------------------------*- C++ -*-===//
Enrico Granataea2bc0f2013-02-21 19:57:10 +00002//
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
Enrico Granataea2bc0f2013-02-21 19:57:10 +000010#include "lldb/DataFormatters/CXXFormatterFunctions.h"
Enrico Granatad87cc312015-09-03 01:29:42 +000011#include "lldb/DataFormatters/StringPrinter.h"
12#include "lldb/DataFormatters/TypeSummary.h"
Enrico Granataea2bc0f2013-02-21 19:57:10 +000013
14#include "lldb/Core/DataBufferHeap.h"
Enrico Granata14b74fd2013-05-06 18:55:52 +000015#include "lldb/Core/Debugger.h"
Enrico Granataea2bc0f2013-02-21 19:57:10 +000016#include "lldb/Core/Error.h"
Greg Clayton554f68d2015-02-04 22:00:53 +000017#include "lldb/Core/FormatEntity.h"
Enrico Granataea2bc0f2013-02-21 19:57:10 +000018#include "lldb/Core/Stream.h"
19#include "lldb/Core/ValueObject.h"
20#include "lldb/Core/ValueObjectConstResult.h"
21#include "lldb/Host/Endian.h"
22#include "lldb/Symbol/ClangASTContext.h"
23#include "lldb/Target/ObjCLanguageRuntime.h"
24#include "lldb/Target/Target.h"
25
26using namespace lldb;
27using namespace lldb_private;
28using namespace lldb_private::formatters;
29
Enrico Granata0dba9b32014-01-08 01:36:59 +000030bool
Enrico Granataf35bc632014-11-06 21:55:30 +000031lldb_private::formatters::LibcxxSmartPointerSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
Enrico Granata0dba9b32014-01-08 01:36:59 +000032{
33 ValueObjectSP valobj_sp(valobj.GetNonSyntheticValue());
34 if (!valobj_sp)
35 return false;
36 ValueObjectSP ptr_sp(valobj_sp->GetChildMemberWithName(ConstString("__ptr_"), true));
37 ValueObjectSP count_sp(valobj_sp->GetChildAtNamePath( {ConstString("__cntrl_"),ConstString("__shared_owners_")} ));
38 ValueObjectSP weakcount_sp(valobj_sp->GetChildAtNamePath( {ConstString("__cntrl_"),ConstString("__shared_weak_owners_")} ));
39
40 if (!ptr_sp)
41 return false;
42
43 if (ptr_sp->GetValueAsUnsigned(0) == 0)
44 {
45 stream.Printf("nullptr");
46 return true;
47 }
48 else
49 {
50 bool print_pointee = false;
51 Error error;
52 ValueObjectSP pointee_sp = ptr_sp->Dereference(error);
53 if (pointee_sp && error.Success())
54 {
55 if (pointee_sp->DumpPrintableRepresentation(stream,
56 ValueObject::eValueObjectRepresentationStyleSummary,
57 lldb::eFormatInvalid,
58 ValueObject::ePrintableRepresentationSpecialCasesDisable,
59 false))
60 print_pointee = true;
61 }
62 if (!print_pointee)
63 stream.Printf("ptr = 0x%" PRIx64, ptr_sp->GetValueAsUnsigned(0));
64 }
65
66 if (count_sp)
67 stream.Printf(" strong=%" PRIu64, 1+count_sp->GetValueAsUnsigned(0));
68
69 if (weakcount_sp)
70 stream.Printf(" weak=%" PRIu64, 1+weakcount_sp->GetValueAsUnsigned(0));
71
72 return true;
73}
74
Enrico Granataea2bc0f2013-02-21 19:57:10 +000075lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::LibcxxVectorBoolSyntheticFrontEnd (lldb::ValueObjectSP valobj_sp) :
76SyntheticChildrenFrontEnd(*valobj_sp.get()),
Enrico Granata473316f2013-12-21 08:09:49 +000077m_bool_type(),
Enrico Granataea2bc0f2013-02-21 19:57:10 +000078m_exe_ctx_ref(),
79m_count(0),
80m_base_data_address(0),
Enrico Granata473316f2013-12-21 08:09:49 +000081m_children()
Enrico Granataea2bc0f2013-02-21 19:57:10 +000082{
83 if (valobj_sp)
Enrico Granata473316f2013-12-21 08:09:49 +000084 {
Enrico Granataea2bc0f2013-02-21 19:57:10 +000085 Update();
Greg Clayton99558cc42015-08-24 23:46:31 +000086 m_bool_type = valobj_sp->GetCompilerType().GetBasicTypeFromAST(lldb::eBasicTypeBool);
Enrico Granata473316f2013-12-21 08:09:49 +000087 }
Enrico Granataea2bc0f2013-02-21 19:57:10 +000088}
89
90size_t
91lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::CalculateNumChildren ()
92{
93 return m_count;
94}
95
96lldb::ValueObjectSP
97lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::GetChildAtIndex (size_t idx)
98{
Enrico Granata473316f2013-12-21 08:09:49 +000099 auto iter = m_children.find(idx),
100 end = m_children.end();
101 if (iter != end)
102 return iter->second;
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000103 if (idx >= m_count)
104 return ValueObjectSP();
105 if (m_base_data_address == 0 || m_count == 0)
106 return ValueObjectSP();
Enrico Granata473316f2013-12-21 08:09:49 +0000107 if (!m_bool_type)
108 return ValueObjectSP();
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000109 size_t byte_idx = (idx >> 3); // divide by 8 to get byte index
110 size_t bit_index = (idx & 7); // efficient idx % 8 for bit index
111 lldb::addr_t byte_location = m_base_data_address + byte_idx;
112 ProcessSP process_sp(m_exe_ctx_ref.GetProcessSP());
113 if (!process_sp)
114 return ValueObjectSP();
115 uint8_t byte = 0;
116 uint8_t mask = 0;
117 Error err;
118 size_t bytes_read = process_sp->ReadMemory(byte_location, &byte, 1, err);
119 if (err.Fail() || bytes_read == 0)
120 return ValueObjectSP();
121 switch (bit_index)
122 {
123 case 0:
124 mask = 1; break;
125 case 1:
126 mask = 2; break;
127 case 2:
128 mask = 4; break;
129 case 3:
130 mask = 8; break;
131 case 4:
132 mask = 16; break;
133 case 5:
134 mask = 32; break;
135 case 6:
136 mask = 64; break;
137 case 7:
138 mask = 128; break;
139 default:
140 return ValueObjectSP();
141 }
142 bool bit_set = ((byte & mask) != 0);
Enrico Granata1cd5e922015-01-28 00:07:51 +0000143 DataBufferSP buffer_sp(new DataBufferHeap(m_bool_type.GetByteSize(nullptr),0));
Enrico Granata473316f2013-12-21 08:09:49 +0000144 if (bit_set && buffer_sp && buffer_sp->GetBytes())
145 *(buffer_sp->GetBytes()) = 1; // regardless of endianness, anything non-zero is true
Deepak Panickal99fbc072014-03-03 15:39:47 +0000146 StreamString name; name.Printf("[%" PRIu64 "]", (uint64_t)idx);
Enrico Granatae29df232014-12-09 19:51:20 +0000147 ValueObjectSP retval_sp(CreateValueObjectFromData(name.GetData(), DataExtractor(buffer_sp, process_sp->GetByteOrder(), process_sp->GetAddressByteSize()), m_exe_ctx_ref, m_bool_type));
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000148 if (retval_sp)
Enrico Granata473316f2013-12-21 08:09:49 +0000149 m_children[idx] = retval_sp;
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000150 return retval_sp;
151}
152
153/*(std::__1::vector<std::__1::allocator<bool> >) vBool = {
154 __begin_ = 0x00000001001000e0
155 __size_ = 56
156 __cap_alloc_ = {
157 std::__1::__libcpp_compressed_pair_imp<unsigned long, std::__1::allocator<unsigned long> > = {
158 __first_ = 1
159 }
160 }
161 }*/
162
163bool
164lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::Update()
165{
Enrico Granata473316f2013-12-21 08:09:49 +0000166 m_children.clear();
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000167 ValueObjectSP valobj_sp = m_backend.GetSP();
168 if (!valobj_sp)
169 return false;
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000170 m_exe_ctx_ref = valobj_sp->GetExecutionContextRef();
171 ValueObjectSP size_sp(valobj_sp->GetChildMemberWithName(ConstString("__size_"), true));
172 if (!size_sp)
173 return false;
174 m_count = size_sp->GetValueAsUnsigned(0);
175 if (!m_count)
176 return true;
177 ValueObjectSP begin_sp(valobj_sp->GetChildMemberWithName(ConstString("__begin_"), true));
178 if (!begin_sp)
179 {
180 m_count = 0;
181 return false;
182 }
183 m_base_data_address = begin_sp->GetValueAsUnsigned(0);
184 if (!m_base_data_address)
185 {
186 m_count = 0;
187 return false;
188 }
Enrico Granata92373532013-03-19 22:58:48 +0000189 return false;
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000190}
191
192bool
193lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::MightHaveChildren ()
194{
195 return true;
196}
197
198size_t
199lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::GetIndexOfChildWithName (const ConstString &name)
200{
201 if (!m_count || !m_base_data_address)
202 return UINT32_MAX;
203 const char* item_name = name.GetCString();
204 uint32_t idx = ExtractIndexFromString(item_name);
205 if (idx < UINT32_MAX && idx >= CalculateNumChildren())
206 return UINT32_MAX;
207 return idx;
208}
209
210lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEnd::~LibcxxVectorBoolSyntheticFrontEnd ()
211{}
212
213SyntheticChildrenFrontEnd*
214lldb_private::formatters::LibcxxVectorBoolSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
215{
216 if (!valobj_sp)
217 return NULL;
218 return (new LibcxxVectorBoolSyntheticFrontEnd(valobj_sp));
219}
220
221/*
222 (lldb) fr var ibeg --raw --ptr-depth 1
223 (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 = {
224 __i_ = {
225 __ptr_ = 0x0000000100103870 {
226 std::__1::__tree_node_base<void *> = {
227 std::__1::__tree_end_node<std::__1::__tree_node_base<void *> *> = {
228 __left_ = 0x0000000000000000
229 }
230 __right_ = 0x0000000000000000
231 __parent_ = 0x00000001001038b0
232 __is_black_ = true
233 }
234 __value_ = {
235 first = 0
236 second = { std::string }
237 */
238
239lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::LibCxxMapIteratorSyntheticFrontEnd (lldb::ValueObjectSP valobj_sp) :
240SyntheticChildrenFrontEnd(*valobj_sp.get()),
241m_pair_ptr()
242{
243 if (valobj_sp)
244 Update();
245}
246
247bool
248lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::Update()
249{
250 ValueObjectSP valobj_sp = m_backend.GetSP();
251 if (!valobj_sp)
252 return false;
253
254 TargetSP target_sp(valobj_sp->GetTargetSP());
255
256 if (!target_sp)
257 return false;
258
259 if (!valobj_sp)
260 return false;
261
262 // this must be a ValueObject* because it is a child of the ValueObject we are producing children for
263 // it if were a ValueObjectSP, we would end up with a loop (iterator -> synthetic -> child -> parent == iterator)
264 // and that would in turn leak memory by never allowing the ValueObjects to die and free their memory
265 m_pair_ptr = valobj_sp->GetValueForExpressionPath(".__i_.__ptr_->__value_",
266 NULL,
267 NULL,
268 NULL,
Enrico Granata495dccd2015-03-12 22:17:07 +0000269 ValueObject::GetValueForExpressionPathOptions().DontCheckDotVsArrowSyntax().SetSyntheticChildrenTraversal(ValueObject::GetValueForExpressionPathOptions::SyntheticChildrenTraversal::None),
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000270 NULL).get();
271
Enrico Granata92373532013-03-19 22:58:48 +0000272 return false;
Enrico Granataea2bc0f2013-02-21 19:57:10 +0000273}
274
275size_t
276lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::CalculateNumChildren ()
277{
278 return 2;
279}
280
281lldb::ValueObjectSP
282lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::GetChildAtIndex (size_t idx)
283{
284 if (!m_pair_ptr)
285 return lldb::ValueObjectSP();
286 return m_pair_ptr->GetChildAtIndex(idx, true);
287}
288
289bool
290lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::MightHaveChildren ()
291{
292 return true;
293}
294
295size_t
296lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::GetIndexOfChildWithName (const ConstString &name)
297{
298 if (name == ConstString("first"))
299 return 0;
300 if (name == ConstString("second"))
301 return 1;
302 return UINT32_MAX;
303}
304
305lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEnd::~LibCxxMapIteratorSyntheticFrontEnd ()
306{
307 // this will be deleted when its parent dies (since it's a child object)
308 //delete m_pair_ptr;
309}
310
311SyntheticChildrenFrontEnd*
312lldb_private::formatters::LibCxxMapIteratorSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
313{
314 if (!valobj_sp)
315 return NULL;
316 return (new LibCxxMapIteratorSyntheticFrontEnd(valobj_sp));
317}
318
319/*
320 (lldb) fr var ibeg --raw --ptr-depth 1 -T
321 (std::__1::__wrap_iter<int *>) ibeg = {
322 (std::__1::__wrap_iter<int *>::iterator_type) __i = 0x00000001001037a0 {
323 (int) *__i = 1
324 }
325 }
326*/
327
328SyntheticChildrenFrontEnd*
329lldb_private::formatters::LibCxxVectorIteratorSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
330{
331 static ConstString g_item_name;
332 if (!g_item_name)
333 g_item_name.SetCString("__i");
334 if (!valobj_sp)
335 return NULL;
336 return (new VectorIteratorSyntheticFrontEnd(valobj_sp,g_item_name));
337}
Enrico Granata92373532013-03-19 22:58:48 +0000338
339lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::LibcxxSharedPtrSyntheticFrontEnd (lldb::ValueObjectSP valobj_sp) :
340SyntheticChildrenFrontEnd(*valobj_sp.get()),
341m_cntrl(NULL),
342m_count_sp(),
343m_weak_count_sp(),
344m_ptr_size(0),
345m_byte_order(lldb::eByteOrderInvalid)
346{
347 if (valobj_sp)
348 Update();
349}
350
351size_t
352lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::CalculateNumChildren ()
353{
354 return (m_cntrl ? 1 : 0);
355}
356
357lldb::ValueObjectSP
358lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::GetChildAtIndex (size_t idx)
359{
360 if (!m_cntrl)
361 return lldb::ValueObjectSP();
362
363 ValueObjectSP valobj_sp = m_backend.GetSP();
364 if (!valobj_sp)
365 return lldb::ValueObjectSP();
366
367 if (idx == 0)
368 return valobj_sp->GetChildMemberWithName(ConstString("__ptr_"), true);
369
370 if (idx > 2)
371 return lldb::ValueObjectSP();
372
373 if (idx == 1)
374 {
375 if (!m_count_sp)
376 {
377 ValueObjectSP shared_owners_sp(m_cntrl->GetChildMemberWithName(ConstString("__shared_owners_"),true));
378 if (!shared_owners_sp)
379 return lldb::ValueObjectSP();
380 uint64_t count = 1 + shared_owners_sp->GetValueAsUnsigned(0);
381 DataExtractor data(&count, 8, m_byte_order, m_ptr_size);
Greg Clayton99558cc42015-08-24 23:46:31 +0000382 m_count_sp = CreateValueObjectFromData("count", data, valobj_sp->GetExecutionContextRef(), shared_owners_sp->GetCompilerType());
Enrico Granata92373532013-03-19 22:58:48 +0000383 }
384 return m_count_sp;
385 }
386 else /* if (idx == 2) */
387 {
388 if (!m_weak_count_sp)
389 {
390 ValueObjectSP shared_weak_owners_sp(m_cntrl->GetChildMemberWithName(ConstString("__shared_weak_owners_"),true));
391 if (!shared_weak_owners_sp)
392 return lldb::ValueObjectSP();
393 uint64_t count = 1 + shared_weak_owners_sp->GetValueAsUnsigned(0);
394 DataExtractor data(&count, 8, m_byte_order, m_ptr_size);
Greg Clayton99558cc42015-08-24 23:46:31 +0000395 m_weak_count_sp = CreateValueObjectFromData("count", data, valobj_sp->GetExecutionContextRef(), shared_weak_owners_sp->GetCompilerType());
Enrico Granata92373532013-03-19 22:58:48 +0000396 }
397 return m_weak_count_sp;
398 }
399}
400
401bool
402lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::Update()
403{
404 m_count_sp.reset();
405 m_weak_count_sp.reset();
406 m_cntrl = NULL;
407
408 ValueObjectSP valobj_sp = m_backend.GetSP();
409 if (!valobj_sp)
410 return false;
411
412 TargetSP target_sp(valobj_sp->GetTargetSP());
413 if (!target_sp)
414 return false;
415
416 m_byte_order = target_sp->GetArchitecture().GetByteOrder();
417 m_ptr_size = target_sp->GetArchitecture().GetAddressByteSize();
418
419 lldb::ValueObjectSP cntrl_sp(valobj_sp->GetChildMemberWithName(ConstString("__cntrl_"),true));
420
421 m_cntrl = cntrl_sp.get(); // need to store the raw pointer to avoid a circular dependency
422 return false;
423}
424
425bool
426lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::MightHaveChildren ()
427{
428 return true;
429}
430
431size_t
432lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::GetIndexOfChildWithName (const ConstString &name)
433{
434 if (name == ConstString("__ptr_"))
435 return 0;
436 if (name == ConstString("count"))
437 return 1;
438 if (name == ConstString("weak_count"))
439 return 2;
440 return UINT32_MAX;
441}
442
443lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEnd::~LibcxxSharedPtrSyntheticFrontEnd ()
444{}
445
446SyntheticChildrenFrontEnd*
447lldb_private::formatters::LibcxxSharedPtrSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
448{
449 if (!valobj_sp)
450 return NULL;
451 return (new LibcxxSharedPtrSyntheticFrontEnd(valobj_sp));
452}
453
Enrico Granata14b74fd2013-05-06 18:55:52 +0000454bool
Enrico Granataf35bc632014-11-06 21:55:30 +0000455lldb_private::formatters::LibcxxContainerSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
Enrico Granata14b74fd2013-05-06 18:55:52 +0000456{
457 if (valobj.IsPointerType())
458 {
459 uint64_t value = valobj.GetValueAsUnsigned(0);
460 if (!value)
461 return false;
Matt Kopecef143712013-06-03 18:00:07 +0000462 stream.Printf("0x%016" PRIx64 " ", value);
Enrico Granata14b74fd2013-05-06 18:55:52 +0000463 }
Greg Clayton554f68d2015-02-04 22:00:53 +0000464 return FormatEntity::FormatStringRef("size=${svar%#}", stream, NULL, NULL, NULL, &valobj, false, false);
Enrico Granata14b74fd2013-05-06 18:55:52 +0000465}
Enrico Granatad87cc312015-09-03 01:29:42 +0000466
467// the field layout in a libc++ string (cap, side, data or data, size, cap)
468enum LibcxxStringLayoutMode
469{
470 eLibcxxStringLayoutModeCSD = 0,
471 eLibcxxStringLayoutModeDSC = 1,
472 eLibcxxStringLayoutModeInvalid = 0xffff
473};
474
475// this function abstracts away the layout and mode details of a libc++ string
476// and returns the address of the data and the size ready for callers to consume
477static bool
478ExtractLibcxxStringInfo (ValueObject& valobj,
479 ValueObjectSP &location_sp,
480 uint64_t& size)
481{
482 ValueObjectSP D(valobj.GetChildAtIndexPath({0,0,0,0}));
483 if (!D)
484 return false;
485
486 ValueObjectSP layout_decider(D->GetChildAtIndexPath({0,0}));
487
488 // this child should exist
489 if (!layout_decider)
490 return false;
491
492 ConstString g_data_name("__data_");
493 ConstString g_size_name("__size_");
494 bool short_mode = false; // this means the string is in short-mode and the data is stored inline
495 LibcxxStringLayoutMode layout = (layout_decider->GetName() == g_data_name) ? eLibcxxStringLayoutModeDSC : eLibcxxStringLayoutModeCSD;
496 uint64_t size_mode_value = 0;
497
498 if (layout == eLibcxxStringLayoutModeDSC)
499 {
500 ValueObjectSP size_mode(D->GetChildAtIndexPath({1,1,0}));
501 if (!size_mode)
502 return false;
503
504 if (size_mode->GetName() != g_size_name)
505 {
506 // we are hitting the padding structure, move along
507 size_mode = D->GetChildAtIndexPath({1,1,1});
508 if (!size_mode)
509 return false;
510 }
511
512 size_mode_value = (size_mode->GetValueAsUnsigned(0));
513 short_mode = ((size_mode_value & 0x80) == 0);
514 }
515 else
516 {
517 ValueObjectSP size_mode(D->GetChildAtIndexPath({1,0,0}));
518 if (!size_mode)
519 return false;
520
521 size_mode_value = (size_mode->GetValueAsUnsigned(0));
522 short_mode = ((size_mode_value & 1) == 0);
523 }
524
525 if (short_mode)
526 {
527 ValueObjectSP s(D->GetChildAtIndex(1, true));
528 if (!s)
529 return false;
530 location_sp = s->GetChildAtIndex((layout == eLibcxxStringLayoutModeDSC) ? 0 : 1, true);
531 size = (layout == eLibcxxStringLayoutModeDSC) ? size_mode_value : ((size_mode_value >> 1) % 256);
532 return (location_sp.get() != nullptr);
533 }
534 else
535 {
536 ValueObjectSP l(D->GetChildAtIndex(0, true));
537 if (!l)
538 return false;
539 // we can use the layout_decider object as the data pointer
540 location_sp = (layout == eLibcxxStringLayoutModeDSC) ? layout_decider : l->GetChildAtIndex(2, true);
541 ValueObjectSP size_vo(l->GetChildAtIndex(1, true));
542 if (!size_vo || !location_sp)
543 return false;
544 size = size_vo->GetValueAsUnsigned(0);
545 return true;
546 }
547}
548
549bool
550lldb_private::formatters::LibcxxWStringSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& summary_options)
551{
552 uint64_t size = 0;
553 ValueObjectSP location_sp((ValueObject*)nullptr);
554 if (!ExtractLibcxxStringInfo(valobj, location_sp, size))
555 return false;
556 if (size == 0)
557 {
558 stream.Printf("L\"\"");
559 return true;
560 }
561 if (!location_sp)
562 return false;
563
564 DataExtractor extractor;
565 if (summary_options.GetCapping() == TypeSummaryCapping::eTypeSummaryCapped)
566 size = std::min<decltype(size)>(size, valobj.GetTargetSP()->GetMaximumSizeOfStringSummary());
567 location_sp->GetPointeeData(extractor, 0, size);
568
569 // std::wstring::size() is measured in 'characters', not bytes
570 auto wchar_t_size = valobj.GetTargetSP()->GetScratchClangASTContext()->GetBasicType(lldb::eBasicTypeWChar).GetByteSize(nullptr);
571
572 ReadBufferAndDumpToStreamOptions options(valobj);
573 options.SetData(extractor);
574 options.SetStream(&stream);
575 options.SetPrefixToken('L');
576 options.SetQuote('"');
577 options.SetSourceSize(size);
578 options.SetBinaryZeroIsTerminator(false);
579
580 switch (wchar_t_size)
581 {
582 case 1:
583 lldb_private::formatters::ReadBufferAndDumpToStream<lldb_private::formatters::StringElementType::UTF8>(options);
584 break;
585
586 case 2:
587 lldb_private::formatters::ReadBufferAndDumpToStream<lldb_private::formatters::StringElementType::UTF16>(options);
588 break;
589
590 case 4:
591 lldb_private::formatters::ReadBufferAndDumpToStream<lldb_private::formatters::StringElementType::UTF32>(options);
592 break;
593
594 default:
595 stream.Printf("size for wchar_t is not valid");
596 return true;
597 }
598
599 return true;
600}
601
602bool
603lldb_private::formatters::LibcxxStringSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& summary_options)
604{
605 uint64_t size = 0;
606 ValueObjectSP location_sp((ValueObject*)nullptr);
607
608 if (!ExtractLibcxxStringInfo(valobj, location_sp, size))
609 return false;
610
611 if (size == 0)
612 {
613 stream.Printf("\"\"");
614 return true;
615 }
616
617 if (!location_sp)
618 return false;
619
620 DataExtractor extractor;
621 if (summary_options.GetCapping() == TypeSummaryCapping::eTypeSummaryCapped)
622 size = std::min<decltype(size)>(size, valobj.GetTargetSP()->GetMaximumSizeOfStringSummary());
623 location_sp->GetPointeeData(extractor, 0, size);
624
625 ReadBufferAndDumpToStreamOptions options(valobj);
626 options.SetData(extractor);
627 options.SetStream(&stream);
628 options.SetPrefixToken(0);
629 options.SetQuote('"');
630 options.SetSourceSize(size);
631 options.SetBinaryZeroIsTerminator(false);
632 lldb_private::formatters::ReadBufferAndDumpToStream<lldb_private::formatters::StringElementType::ASCII>(options);
633
634 return true;
635}