blob: 282a8de9cbb8ad88ec398749c6cefe24a43bd356 [file] [log] [blame]
Pavel Labath52698752017-11-14 11:15:03 +00001//===-- LibCxxBitset.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 "LibCxx.h"
11#include "lldb/DataFormatters/FormattersHelpers.h"
12#include "lldb/Symbol/ClangASTContext.h"
13#include "lldb/Target/Target.h"
14
15using namespace lldb;
16using namespace lldb_private;
17
18namespace {
19
20class BitsetFrontEnd : public SyntheticChildrenFrontEnd {
21public:
22 BitsetFrontEnd(ValueObject &valobj);
23
24 size_t GetIndexOfChildWithName(const ConstString &name) override {
25 return formatters::ExtractIndexFromString(name.GetCString());
26 }
27
28 bool MightHaveChildren() override { return true; }
29 bool Update() override;
30 size_t CalculateNumChildren() override { return m_elements.size(); }
31 ValueObjectSP GetChildAtIndex(size_t idx) override;
32
33private:
34 std::vector<ValueObjectSP> m_elements;
35 ValueObjectSP m_first;
36 CompilerType m_bool_type;
37 ByteOrder m_byte_order = eByteOrderInvalid;
38 uint8_t m_byte_size = 0;
39};
40} // namespace
41
42BitsetFrontEnd::BitsetFrontEnd(ValueObject &valobj)
43 : SyntheticChildrenFrontEnd(valobj) {
44 m_bool_type = valobj.GetCompilerType().GetBasicTypeFromAST(eBasicTypeBool);
45 if (auto target_sp = m_backend.GetTargetSP()) {
46 m_byte_order = target_sp->GetArchitecture().GetByteOrder();
47 m_byte_size = target_sp->GetArchitecture().GetAddressByteSize();
48 Update();
49 }
50}
51
52bool BitsetFrontEnd::Update() {
53 m_elements.clear();
54 m_first.reset();
55
56 TargetSP target_sp = m_backend.GetTargetSP();
57 if (!target_sp)
58 return false;
59 size_t capping_size = target_sp->GetMaximumNumberOfChildrenToDisplay();
60
61 size_t size = 0;
62 auto value_and_type =
63 m_backend.GetCompilerType().GetIntegralTemplateArgument(0);
64 if (value_and_type.second)
65 size = value_and_type.first.getLimitedValue(capping_size);
66
67 m_elements.assign(size, ValueObjectSP());
68
69 m_first = m_backend.GetChildMemberWithName(ConstString("__first_"), true);
70 return false;
71}
72
73ValueObjectSP BitsetFrontEnd::GetChildAtIndex(size_t idx) {
74 if (idx >= m_elements.size() || !m_first)
75 return ValueObjectSP();
76
77 if (m_elements[idx])
78 return m_elements[idx];
79
80 ExecutionContext ctx = m_backend.GetExecutionContextRef().Lock(false);
81 CompilerType type;
82 ValueObjectSP chunk;
83 // For small bitsets __first_ is not an array, but a plain size_t.
84 if (m_first->GetCompilerType().IsArrayType(&type, nullptr, nullptr))
85 chunk = m_first->GetChildAtIndex(
86 idx / type.GetBitSize(ctx.GetBestExecutionContextScope()), true);
87 else {
88 type = m_first->GetCompilerType();
89 chunk = m_first;
90 }
91 if (!type || !chunk)
92 return ValueObjectSP();
93
94 size_t chunk_idx = idx % type.GetBitSize(ctx.GetBestExecutionContextScope());
95 uint8_t value = !!(chunk->GetValueAsUnsigned(0) & (uint64_t(1) << chunk_idx));
96 DataExtractor data(&value, sizeof(value), m_byte_order, m_byte_size);
97
98 m_elements[idx] = CreateValueObjectFromData(llvm::formatv("[{0}]", idx).str(),
99 data, ctx, m_bool_type);
100
101 return m_elements[idx];
102}
103
104SyntheticChildrenFrontEnd *formatters::LibcxxBitsetSyntheticFrontEndCreator(
105 CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) {
106 if (valobj_sp)
107 return new BitsetFrontEnd(*valobj_sp);
108 return nullptr;
109}