LLDB mainline
GenericInitializerList.cpp
Go to the documentation of this file.
1//===-- GenericInitializerList.cpp ----------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
12#include "llvm/Support/ErrorExtras.h"
13#include <cstddef>
14#include <optional>
15#include <type_traits>
16
17using namespace lldb;
18using namespace lldb_private;
19
20namespace generic_check {
21template <class T>
22using size_func = decltype(T::GetSizeMember(std::declval<ValueObject &>()));
23template <class T>
24using start_func = decltype(T::GetStartMember(std::declval<ValueObject &>()));
25namespace {
26template <typename...> struct check_func : std::true_type {};
27} // namespace
28
29template <typename T>
30using has_functions = check_func<size_func<T>, start_func<T>>;
31} // namespace generic_check
32
33struct LibCxx {
35 return backend.GetChildMemberWithName("__begin_");
36 }
37
39 return backend.GetChildMemberWithName("__size_");
40 }
41};
42
43struct LibStdcpp {
45 return backend.GetChildMemberWithName("_M_array");
46 }
47
49 return backend.GetChildMemberWithName("_M_len");
50 }
51};
52
54
55template <class StandardImpl>
58public:
60 "Missing Required Functions.");
61
67
69 // this needs to stay around because it's a child object who will follow its
70 // parent's life cycle
71 // delete m_start;
72 }
73
74 llvm::Expected<uint32_t> CalculateNumChildren() override {
76
77 const ValueObjectSP size_sp(StandardImpl::GetSizeMember(m_backend));
78 if (size_sp)
79 m_num_elements = size_sp->GetValueAsUnsigned(0);
80 return m_num_elements;
81 }
82
83 lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override {
84 if (!m_start)
85 return {};
86
87 uint64_t offset = static_cast<uint64_t>(idx) * m_element_size;
88 offset = offset + m_start->GetValueAsUnsigned(0);
89 StreamString name;
90 name.Printf("[%" PRIu64 "]", (uint64_t)idx);
91 return CreateValueObjectFromAddress(name.GetString(), offset,
92 m_backend.GetExecutionContextRef(),
94 }
95
97 m_start = nullptr;
99 m_element_type = m_backend.GetCompilerType().GetTypeTemplateArgument(0);
100 if (!m_element_type.IsValid())
102
103 llvm::Expected<uint64_t> size_or_err = m_element_type.GetByteSize(nullptr);
104 if (!size_or_err)
105 LLDB_LOG_ERRORV(GetLog(LLDBLog::DataFormatters), size_or_err.takeError(),
106 "{0}");
107 else {
108 m_element_size = *size_or_err;
109 // Store raw pointers or end up with a circular dependency.
110 m_start = StandardImpl::GetStartMember(m_backend).get();
111 }
112
114 }
115
116 llvm::Expected<size_t> GetIndexOfChildWithName(ConstString name) override {
117 if (!m_start) {
118 return llvm::createStringErrorV("type has no child named '{0}'", name);
119 }
120 auto optional_idx = formatters::ExtractIndexFromString(name.GetCString());
121 if (!optional_idx) {
122 return llvm::createStringErrorV("type has no child named '{0}'", name);
123 }
124 return *optional_idx;
125 }
126
127private:
130 uint32_t m_element_size = 0;
131 size_t m_num_elements = 0;
132};
133
135 CXXSyntheticChildren * /*unused*/, lldb::ValueObjectSP valobj_sp) {
136 if (!valobj_sp)
137 return nullptr;
138
139 if (LibCxx::GetStartMember(*valobj_sp) != nullptr)
141
143}
144} // namespace lldb_private::formatters
#define LLDB_LOG_ERRORV(log, error,...)
Definition Log.h:410
Generic representation of a type in a programming language.
A uniqued constant string class.
Definition ConstString.h:40
const char * GetCString() const
Get the string value as a C string.
llvm::StringRef GetString() const
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
Definition Stream.cpp:132
SyntheticChildrenFrontEnd(ValueObject &backend)
lldb::ValueObjectSP CreateValueObjectFromAddress(llvm::StringRef name, uint64_t address, const ExecutionContext &exe_ctx, CompilerType type, bool do_deref=true)
virtual lldb::ValueObjectSP GetChildMemberWithName(llvm::StringRef name, bool can_create=true)
lldb::ChildCacheState Update() override
This function is assumed to always succeed and if it fails, the front-end should know to deal with it...
llvm::Expected< size_t > GetIndexOfChildWithName(ConstString name) override
Determine the index of a named child.
decltype(T::GetSizeMember(std::declval< ValueObject & >())) size_func
decltype(T::GetStartMember(std::declval< ValueObject & >())) start_func
check_func< size_func< T >, start_func< T > > has_functions
std::optional< size_t > ExtractIndexFromString(const char *item_name)
SyntheticChildrenFrontEnd * GenericInitializerListSyntheticFrontEndCreator(CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp)
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
Definition Log.h:327
ChildCacheState
Specifies if children need to be re-computed after a call to SyntheticChildrenFrontEnd::Update.
@ eRefetch
Children need to be recomputed dynamically.
std::shared_ptr< lldb_private::ValueObject > ValueObjectSP
static ValueObjectSP GetSizeMember(ValueObject &backend)
static ValueObjectSP GetStartMember(ValueObject &backend)
static ValueObjectSP GetSizeMember(ValueObject &backend)
static ValueObjectSP GetStartMember(ValueObject &backend)