1dda28197Spatrick //===-- LibStdcppUniquePointer.cpp ----------------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick
9061da546Spatrick #include "LibStdcpp.h"
10061da546Spatrick
11061da546Spatrick #include "lldb/Core/ValueObject.h"
12061da546Spatrick #include "lldb/DataFormatters/FormattersHelpers.h"
13061da546Spatrick #include "lldb/DataFormatters/TypeSynthetic.h"
14061da546Spatrick #include "lldb/Utility/ConstString.h"
15061da546Spatrick
16061da546Spatrick #include <memory>
17061da546Spatrick #include <vector>
18061da546Spatrick
19061da546Spatrick using namespace lldb;
20061da546Spatrick using namespace lldb_private;
21061da546Spatrick using namespace lldb_private::formatters;
22061da546Spatrick
23061da546Spatrick namespace {
24061da546Spatrick
25061da546Spatrick class LibStdcppUniquePtrSyntheticFrontEnd : public SyntheticChildrenFrontEnd {
26061da546Spatrick public:
27061da546Spatrick explicit LibStdcppUniquePtrSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp);
28061da546Spatrick
29061da546Spatrick size_t CalculateNumChildren() override;
30061da546Spatrick
31061da546Spatrick lldb::ValueObjectSP GetChildAtIndex(size_t idx) override;
32061da546Spatrick
33061da546Spatrick bool Update() override;
34061da546Spatrick
35061da546Spatrick bool MightHaveChildren() override;
36061da546Spatrick
37061da546Spatrick size_t GetIndexOfChildWithName(ConstString name) override;
38061da546Spatrick
39061da546Spatrick bool GetSummary(Stream &stream, const TypeSummaryOptions &options);
40061da546Spatrick
41061da546Spatrick private:
42061da546Spatrick // The lifetime of a ValueObject and all its derivative ValueObjects
43061da546Spatrick // (children, clones, etc.) is managed by a ClusterManager. These
44061da546Spatrick // objects are only destroyed when every shared pointer to any of them
45061da546Spatrick // is destroyed, so we must not store a shared pointer to any ValueObject
46061da546Spatrick // derived from our backend ValueObject (since we're in the same cluster).
47061da546Spatrick ValueObject* m_ptr_obj = nullptr;
48061da546Spatrick ValueObject* m_obj_obj = nullptr;
49061da546Spatrick ValueObject* m_del_obj = nullptr;
50061da546Spatrick
51061da546Spatrick ValueObjectSP GetTuple();
52061da546Spatrick };
53061da546Spatrick
54061da546Spatrick } // end of anonymous namespace
55061da546Spatrick
LibStdcppUniquePtrSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)56061da546Spatrick LibStdcppUniquePtrSyntheticFrontEnd::LibStdcppUniquePtrSyntheticFrontEnd(
57061da546Spatrick lldb::ValueObjectSP valobj_sp)
58061da546Spatrick : SyntheticChildrenFrontEnd(*valobj_sp) {
59061da546Spatrick Update();
60061da546Spatrick }
61061da546Spatrick
GetTuple()62061da546Spatrick ValueObjectSP LibStdcppUniquePtrSyntheticFrontEnd::GetTuple() {
63061da546Spatrick ValueObjectSP valobj_backend_sp = m_backend.GetSP();
64061da546Spatrick
65061da546Spatrick if (!valobj_backend_sp)
66061da546Spatrick return nullptr;
67061da546Spatrick
68061da546Spatrick ValueObjectSP valobj_sp = valobj_backend_sp->GetNonSyntheticValue();
69061da546Spatrick if (!valobj_sp)
70061da546Spatrick return nullptr;
71061da546Spatrick
72061da546Spatrick ValueObjectSP obj_child_sp =
73061da546Spatrick valobj_sp->GetChildMemberWithName(ConstString("_M_t"), true);
74061da546Spatrick if (!obj_child_sp)
75061da546Spatrick return nullptr;
76061da546Spatrick
77061da546Spatrick ValueObjectSP obj_subchild_sp =
78061da546Spatrick obj_child_sp->GetChildMemberWithName(ConstString("_M_t"), true);
79061da546Spatrick
80061da546Spatrick // if there is a _M_t subchild, the tuple is found in the obj_subchild_sp
81061da546Spatrick // (for libstdc++ 6.0.23).
82061da546Spatrick if (obj_subchild_sp) {
83061da546Spatrick return obj_subchild_sp;
84061da546Spatrick }
85061da546Spatrick
86061da546Spatrick return obj_child_sp;
87061da546Spatrick }
88061da546Spatrick
Update()89061da546Spatrick bool LibStdcppUniquePtrSyntheticFrontEnd::Update() {
90061da546Spatrick ValueObjectSP tuple_sp = GetTuple();
91061da546Spatrick
92061da546Spatrick if (!tuple_sp)
93061da546Spatrick return false;
94061da546Spatrick
95061da546Spatrick std::unique_ptr<SyntheticChildrenFrontEnd> tuple_frontend(
96061da546Spatrick LibStdcppTupleSyntheticFrontEndCreator(nullptr, tuple_sp));
97061da546Spatrick
98061da546Spatrick ValueObjectSP ptr_obj = tuple_frontend->GetChildAtIndex(0);
99061da546Spatrick if (ptr_obj)
100061da546Spatrick m_ptr_obj = ptr_obj->Clone(ConstString("pointer")).get();
101061da546Spatrick
102*be691f3bSpatrick // Add a 'deleter' child if there was a non-empty deleter type specified.
103*be691f3bSpatrick //
104*be691f3bSpatrick // The object might have size=1 in the TypeSystem but occupies no dedicated
105*be691f3bSpatrick // storage due to no_unique_address, so infer the actual size from the total
106*be691f3bSpatrick // size of the unique_ptr class. If sizeof(unique_ptr) == sizeof(void*) then
107*be691f3bSpatrick // the deleter is empty and should be hidden.
108*be691f3bSpatrick if (tuple_sp->GetByteSize() > ptr_obj->GetByteSize()) {
109061da546Spatrick ValueObjectSP del_obj = tuple_frontend->GetChildAtIndex(1);
110061da546Spatrick if (del_obj)
111061da546Spatrick m_del_obj = del_obj->Clone(ConstString("deleter")).get();
112*be691f3bSpatrick }
113061da546Spatrick
114061da546Spatrick if (m_ptr_obj) {
115061da546Spatrick Status error;
116061da546Spatrick ValueObjectSP obj_obj = m_ptr_obj->Dereference(error);
117061da546Spatrick if (error.Success()) {
118061da546Spatrick m_obj_obj = obj_obj->Clone(ConstString("object")).get();
119061da546Spatrick }
120061da546Spatrick }
121061da546Spatrick
122061da546Spatrick return false;
123061da546Spatrick }
124061da546Spatrick
MightHaveChildren()125061da546Spatrick bool LibStdcppUniquePtrSyntheticFrontEnd::MightHaveChildren() { return true; }
126061da546Spatrick
127061da546Spatrick lldb::ValueObjectSP
GetChildAtIndex(size_t idx)128061da546Spatrick LibStdcppUniquePtrSyntheticFrontEnd::GetChildAtIndex(size_t idx) {
129061da546Spatrick if (idx == 0 && m_ptr_obj)
130061da546Spatrick return m_ptr_obj->GetSP();
131061da546Spatrick if (idx == 1 && m_del_obj)
132061da546Spatrick return m_del_obj->GetSP();
133061da546Spatrick if (idx == 2 && m_obj_obj)
134061da546Spatrick return m_obj_obj->GetSP();
135061da546Spatrick return lldb::ValueObjectSP();
136061da546Spatrick }
137061da546Spatrick
CalculateNumChildren()138061da546Spatrick size_t LibStdcppUniquePtrSyntheticFrontEnd::CalculateNumChildren() {
139061da546Spatrick if (m_del_obj)
140061da546Spatrick return 2;
141061da546Spatrick return 1;
142061da546Spatrick }
143061da546Spatrick
GetIndexOfChildWithName(ConstString name)144061da546Spatrick size_t LibStdcppUniquePtrSyntheticFrontEnd::GetIndexOfChildWithName(
145061da546Spatrick ConstString name) {
146061da546Spatrick if (name == "ptr" || name == "pointer")
147061da546Spatrick return 0;
148061da546Spatrick if (name == "del" || name == "deleter")
149061da546Spatrick return 1;
150061da546Spatrick if (name == "obj" || name == "object" || name == "$$dereference$$")
151061da546Spatrick return 2;
152061da546Spatrick return UINT32_MAX;
153061da546Spatrick }
154061da546Spatrick
GetSummary(Stream & stream,const TypeSummaryOptions & options)155061da546Spatrick bool LibStdcppUniquePtrSyntheticFrontEnd::GetSummary(
156061da546Spatrick Stream &stream, const TypeSummaryOptions &options) {
157061da546Spatrick if (!m_ptr_obj)
158061da546Spatrick return false;
159061da546Spatrick
160061da546Spatrick bool success;
161061da546Spatrick uint64_t ptr_value = m_ptr_obj->GetValueAsUnsigned(0, &success);
162061da546Spatrick if (!success)
163061da546Spatrick return false;
164061da546Spatrick if (ptr_value == 0)
165061da546Spatrick stream.Printf("nullptr");
166061da546Spatrick else
167061da546Spatrick stream.Printf("0x%" PRIx64, ptr_value);
168061da546Spatrick return true;
169061da546Spatrick }
170061da546Spatrick
171061da546Spatrick SyntheticChildrenFrontEnd *
LibStdcppUniquePtrSyntheticFrontEndCreator(CXXSyntheticChildren *,lldb::ValueObjectSP valobj_sp)172061da546Spatrick lldb_private::formatters::LibStdcppUniquePtrSyntheticFrontEndCreator(
173061da546Spatrick CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) {
174061da546Spatrick return (valobj_sp ? new LibStdcppUniquePtrSyntheticFrontEnd(valobj_sp)
175061da546Spatrick : nullptr);
176061da546Spatrick }
177061da546Spatrick
LibStdcppUniquePointerSummaryProvider(ValueObject & valobj,Stream & stream,const TypeSummaryOptions & options)178061da546Spatrick bool lldb_private::formatters::LibStdcppUniquePointerSummaryProvider(
179061da546Spatrick ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {
180061da546Spatrick LibStdcppUniquePtrSyntheticFrontEnd formatter(valobj.GetSP());
181061da546Spatrick return formatter.GetSummary(stream, options);
182061da546Spatrick }
183