blob: 9d46e3e3ee58cd8affe8d964feb6125530a84fd6 [file] [log] [blame]
Tamas Berghammerd161b212016-10-21 15:02:44 +00001//===-- LibStdcppUniquePointer.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 "LibStdcpp.h"
11
Tamas Berghammerd161b212016-10-21 15:02:44 +000012#include "lldb/Core/ValueObject.h"
13#include "lldb/DataFormatters/FormattersHelpers.h"
14#include "lldb/DataFormatters/TypeSynthetic.h"
Zachary Turnerbf9a7732017-02-02 21:39:50 +000015#include "lldb/Utility/ConstString.h"
Tamas Berghammerd161b212016-10-21 15:02:44 +000016
Tamas Berghammerde2cc6e2016-10-21 15:05:03 +000017#include <memory>
18#include <vector>
19
Tamas Berghammerd161b212016-10-21 15:02:44 +000020using namespace lldb;
21using namespace lldb_private;
22using namespace lldb_private::formatters;
23
24namespace {
25
26class LibStdcppUniquePtrSyntheticFrontEnd : public SyntheticChildrenFrontEnd {
27public:
28 explicit LibStdcppUniquePtrSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp);
29
30 size_t CalculateNumChildren() override;
31
32 lldb::ValueObjectSP GetChildAtIndex(size_t idx) override;
33
34 bool Update() override;
35
36 bool MightHaveChildren() override;
37
38 size_t GetIndexOfChildWithName(const ConstString &name) override;
39
40 bool GetSummary(Stream &stream, const TypeSummaryOptions &options);
41
42private:
43 ValueObjectSP m_ptr_obj;
44 ValueObjectSP m_obj_obj;
45 ValueObjectSP m_del_obj;
Pavel Labathb312b132018-03-08 16:03:09 +000046
47 ValueObjectSP GetTuple();
Tamas Berghammerd161b212016-10-21 15:02:44 +000048};
49
50} // end of anonymous namespace
51
52LibStdcppUniquePtrSyntheticFrontEnd::LibStdcppUniquePtrSyntheticFrontEnd(
53 lldb::ValueObjectSP valobj_sp)
54 : SyntheticChildrenFrontEnd(*valobj_sp) {
55 Update();
56}
57
Pavel Labathb312b132018-03-08 16:03:09 +000058ValueObjectSP LibStdcppUniquePtrSyntheticFrontEnd::GetTuple() {
Tamas Berghammerd161b212016-10-21 15:02:44 +000059 ValueObjectSP valobj_backend_sp = m_backend.GetSP();
Pavel Labathb312b132018-03-08 16:03:09 +000060
Tamas Berghammerd161b212016-10-21 15:02:44 +000061 if (!valobj_backend_sp)
Pavel Labathb312b132018-03-08 16:03:09 +000062 return nullptr;
Tamas Berghammerd161b212016-10-21 15:02:44 +000063
64 ValueObjectSP valobj_sp = valobj_backend_sp->GetNonSyntheticValue();
65 if (!valobj_sp)
Pavel Labathb312b132018-03-08 16:03:09 +000066 return nullptr;
Tamas Berghammerd161b212016-10-21 15:02:44 +000067
Pavel Labathb312b132018-03-08 16:03:09 +000068 ValueObjectSP obj_child_sp =
Tamas Berghammerd161b212016-10-21 15:02:44 +000069 valobj_sp->GetChildMemberWithName(ConstString("_M_t"), true);
Pavel Labathcb512a32018-06-11 14:52:52 +000070 if (!obj_child_sp)
71 return nullptr;
Pavel Labathb312b132018-03-08 16:03:09 +000072
73 ValueObjectSP obj_subchild_sp =
74 obj_child_sp->GetChildMemberWithName(ConstString("_M_t"), true);
75
Adrian Prantl05097242018-04-30 16:49:04 +000076 // if there is a _M_t subchild, the tuple is found in the obj_subchild_sp
77 // (for libstdc++ 6.0.23).
Pavel Labathb312b132018-03-08 16:03:09 +000078 if (obj_subchild_sp) {
79 return obj_subchild_sp;
80 }
81
82 return obj_child_sp;
83}
84
85bool LibStdcppUniquePtrSyntheticFrontEnd::Update() {
86 ValueObjectSP tuple_sp = GetTuple();
87
Tamas Berghammerd161b212016-10-21 15:02:44 +000088 if (!tuple_sp)
89 return false;
90
91 std::unique_ptr<SyntheticChildrenFrontEnd> tuple_frontend(
92 LibStdcppTupleSyntheticFrontEndCreator(nullptr, tuple_sp));
93
Tamas Berghammer4fbb55b2017-03-31 20:48:00 +000094 ValueObjectSP ptr_obj = tuple_frontend->GetChildAtIndex(0);
95 if (ptr_obj)
96 m_ptr_obj = ptr_obj->Clone(ConstString("pointer"));
Tamas Berghammerd161b212016-10-21 15:02:44 +000097
Tamas Berghammer4fbb55b2017-03-31 20:48:00 +000098 ValueObjectSP del_obj = tuple_frontend->GetChildAtIndex(1);
99 if (del_obj)
100 m_del_obj = del_obj->Clone(ConstString("deleter"));
Tamas Berghammerd161b212016-10-21 15:02:44 +0000101
102 if (m_ptr_obj) {
Zachary Turner97206d52017-05-12 04:51:55 +0000103 Status error;
Tamas Berghammer4fbb55b2017-03-31 20:48:00 +0000104 ValueObjectSP obj_obj = m_ptr_obj->Dereference(error);
Tamas Berghammerd161b212016-10-21 15:02:44 +0000105 if (error.Success()) {
Tamas Berghammer4fbb55b2017-03-31 20:48:00 +0000106 m_obj_obj = obj_obj->Clone(ConstString("object"));
Tamas Berghammerd161b212016-10-21 15:02:44 +0000107 }
108 }
109
110 return false;
111}
112
113bool LibStdcppUniquePtrSyntheticFrontEnd::MightHaveChildren() { return true; }
114
115lldb::ValueObjectSP
116LibStdcppUniquePtrSyntheticFrontEnd::GetChildAtIndex(size_t idx) {
117 if (idx == 0)
Tamas Berghammeraf8953a2017-03-31 20:07:20 +0000118 return m_ptr_obj;
Tamas Berghammerd161b212016-10-21 15:02:44 +0000119 if (idx == 1)
120 return m_del_obj;
121 if (idx == 2)
Tamas Berghammeraf8953a2017-03-31 20:07:20 +0000122 return m_obj_obj;
Tamas Berghammerd161b212016-10-21 15:02:44 +0000123 return lldb::ValueObjectSP();
124}
125
126size_t LibStdcppUniquePtrSyntheticFrontEnd::CalculateNumChildren() {
127 if (m_del_obj)
128 return 2;
Tamas Berghammeraf8953a2017-03-31 20:07:20 +0000129 return 1;
Tamas Berghammerd161b212016-10-21 15:02:44 +0000130}
131
132size_t LibStdcppUniquePtrSyntheticFrontEnd::GetIndexOfChildWithName(
133 const ConstString &name) {
Tamas Berghammeraf8953a2017-03-31 20:07:20 +0000134 if (name == ConstString("ptr") || name == ConstString("pointer"))
Tamas Berghammerd161b212016-10-21 15:02:44 +0000135 return 0;
136 if (name == ConstString("del") || name == ConstString("deleter"))
137 return 1;
Tamas Berghammer4c08fe22017-03-31 20:23:22 +0000138 if (name == ConstString("obj") || name == ConstString("object") ||
139 name == ConstString("$$dereference$$"))
Tamas Berghammerd161b212016-10-21 15:02:44 +0000140 return 2;
141 return UINT32_MAX;
142}
143
144bool LibStdcppUniquePtrSyntheticFrontEnd::GetSummary(
145 Stream &stream, const TypeSummaryOptions &options) {
146 if (!m_ptr_obj)
147 return false;
148
Pavel Labath833e3d12016-11-09 10:42:29 +0000149 bool success;
150 uint64_t ptr_value = m_ptr_obj->GetValueAsUnsigned(0, &success);
151 if (!success)
152 return false;
153 if (ptr_value == 0)
Tamas Berghammerd161b212016-10-21 15:02:44 +0000154 stream.Printf("nullptr");
Pavel Labath833e3d12016-11-09 10:42:29 +0000155 else
156 stream.Printf("0x%" PRIx64, ptr_value);
Tamas Berghammerd161b212016-10-21 15:02:44 +0000157 return true;
158}
159
160SyntheticChildrenFrontEnd *
161lldb_private::formatters::LibStdcppUniquePtrSyntheticFrontEndCreator(
162 CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) {
163 return (valobj_sp ? new LibStdcppUniquePtrSyntheticFrontEnd(valobj_sp)
164 : nullptr);
165}
166
167bool lldb_private::formatters::LibStdcppUniquePointerSummaryProvider(
168 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {
169 LibStdcppUniquePtrSyntheticFrontEnd formatter(valobj.GetSP());
170 return formatter.GetSummary(stream, options);
171}