xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/Language/CPlusPlus/LibStdcppUniquePointer.cpp (revision be691f3bb6417f04a68938fadbcaee2d5795e764)
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