blob: eea34e61d47b01599a310ef109878bbaf9023c55 [file] [log] [blame]
Eugene Zelenko26f34fb2015-11-07 00:28:50 +00001//===-- NSException.cpp -----------------------------------------*- C++ -*-===//
Enrico Granata16709ef2015-11-06 02:43:32 +00002//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// 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
Enrico Granata16709ef2015-11-06 02:43:32 +00006//
7//===----------------------------------------------------------------------===//
8
Eugene Zelenko26f34fb2015-11-07 00:28:50 +00009#include "clang/AST/DeclCXX.h"
10
Enrico Granata16709ef2015-11-06 02:43:32 +000011#include "Cocoa.h"
12
Enrico Granata16709ef2015-11-06 02:43:32 +000013#include "lldb/Core/ValueObject.h"
14#include "lldb/Core/ValueObjectConstResult.h"
15#include "lldb/DataFormatters/FormattersHelpers.h"
Enrico Granata16709ef2015-11-06 02:43:32 +000016#include "lldb/Symbol/ClangASTContext.h"
17#include "lldb/Target/ObjCLanguageRuntime.h"
Zachary Turner01c32432017-02-14 19:06:07 +000018#include "lldb/Target/ProcessStructReader.h"
Enrico Granata16709ef2015-11-06 02:43:32 +000019#include "lldb/Target/Target.h"
Zachary Turner666cc0b2017-03-04 01:30:05 +000020#include "lldb/Utility/DataBufferHeap.h"
Zachary Turner01c32432017-02-14 19:06:07 +000021#include "lldb/Utility/Endian.h"
Zachary Turner97206d52017-05-12 04:51:55 +000022#include "lldb/Utility/Status.h"
Zachary Turnerbf9a7732017-02-02 21:39:50 +000023#include "lldb/Utility/Stream.h"
Enrico Granata16709ef2015-11-06 02:43:32 +000024
Enrico Granata16709ef2015-11-06 02:43:32 +000025#include "Plugins/Language/ObjC/NSString.h"
26
27using namespace lldb;
28using namespace lldb_private;
29using namespace lldb_private::formatters;
30
Kuba Mracekb4ade532018-11-12 17:25:23 +000031static bool ExtractFields(ValueObject &valobj, ValueObjectSP *name_sp,
Kuba Mracekc3f1e622018-11-12 19:12:31 +000032 ValueObjectSP *reason_sp, ValueObjectSP *userinfo_sp,
33 ValueObjectSP *reserved_sp) {
Kate Stoneb9c1b512016-09-06 20:57:50 +000034 ProcessSP process_sp(valobj.GetProcessSP());
35 if (!process_sp)
36 return false;
37
Kuba Mracekb4ade532018-11-12 17:25:23 +000038 lldb::addr_t ptr = LLDB_INVALID_ADDRESS;
Kate Stoneb9c1b512016-09-06 20:57:50 +000039
40 CompilerType valobj_type(valobj.GetCompilerType());
41 Flags type_flags(valobj_type.GetTypeInfo());
42 if (type_flags.AllClear(eTypeHasValue)) {
43 if (valobj.IsBaseClass() && valobj.GetParent())
Kuba Mracekb4ade532018-11-12 17:25:23 +000044 ptr = valobj.GetParent()->GetValueAsUnsigned(LLDB_INVALID_ADDRESS);
45 } else {
46 ptr = valobj.GetValueAsUnsigned(LLDB_INVALID_ADDRESS);
47 }
Kate Stoneb9c1b512016-09-06 20:57:50 +000048
Kuba Mracekb4ade532018-11-12 17:25:23 +000049 if (ptr == LLDB_INVALID_ADDRESS)
Kate Stoneb9c1b512016-09-06 20:57:50 +000050 return false;
51 size_t ptr_size = process_sp->GetAddressByteSize();
Kate Stoneb9c1b512016-09-06 20:57:50 +000052
Zachary Turner97206d52017-05-12 04:51:55 +000053 Status error;
Kuba Mracekb4ade532018-11-12 17:25:23 +000054 auto name = process_sp->ReadPointerFromMemory(ptr + 1 * ptr_size, error);
Kate Stoneb9c1b512016-09-06 20:57:50 +000055 if (error.Fail() || name == LLDB_INVALID_ADDRESS)
56 return false;
Kuba Mracekb4ade532018-11-12 17:25:23 +000057 auto reason = process_sp->ReadPointerFromMemory(ptr + 2 * ptr_size, error);
Kate Stoneb9c1b512016-09-06 20:57:50 +000058 if (error.Fail() || reason == LLDB_INVALID_ADDRESS)
59 return false;
Kuba Mracekb4ade532018-11-12 17:25:23 +000060 auto userinfo = process_sp->ReadPointerFromMemory(ptr + 3 * ptr_size, error);
61 if (error.Fail() || userinfo == LLDB_INVALID_ADDRESS)
62 return false;
Kuba Mracekc3f1e622018-11-12 19:12:31 +000063 auto reserved = process_sp->ReadPointerFromMemory(ptr + 4 * ptr_size, error);
64 if (error.Fail() || reserved == LLDB_INVALID_ADDRESS)
65 return false;
Kate Stoneb9c1b512016-09-06 20:57:50 +000066
67 InferiorSizedWord name_isw(name, *process_sp);
68 InferiorSizedWord reason_isw(reason, *process_sp);
Kuba Mracekb4ade532018-11-12 17:25:23 +000069 InferiorSizedWord userinfo_isw(userinfo, *process_sp);
Kuba Mracekc3f1e622018-11-12 19:12:31 +000070 InferiorSizedWord reserved_isw(reserved, *process_sp);
Kate Stoneb9c1b512016-09-06 20:57:50 +000071
72 CompilerType voidstar = process_sp->GetTarget()
73 .GetScratchClangASTContext()
74 ->GetBasicType(lldb::eBasicTypeVoid)
75 .GetPointerType();
76
Kuba Mracekb4ade532018-11-12 17:25:23 +000077 if (name_sp)
78 *name_sp = ValueObject::CreateValueObjectFromData(
79 "name", name_isw.GetAsData(process_sp->GetByteOrder()),
80 valobj.GetExecutionContextRef(), voidstar);
81 if (reason_sp)
82 *reason_sp = ValueObject::CreateValueObjectFromData(
83 "reason", reason_isw.GetAsData(process_sp->GetByteOrder()),
84 valobj.GetExecutionContextRef(), voidstar);
85 if (userinfo_sp)
86 *userinfo_sp = ValueObject::CreateValueObjectFromData(
87 "userInfo", userinfo_isw.GetAsData(process_sp->GetByteOrder()),
88 valobj.GetExecutionContextRef(), voidstar);
Kuba Mracekc3f1e622018-11-12 19:12:31 +000089 if (reserved_sp)
90 *reserved_sp = ValueObject::CreateValueObjectFromData(
91 "reserved", reserved_isw.GetAsData(process_sp->GetByteOrder()),
92 valobj.GetExecutionContextRef(), voidstar);
Kuba Mracekb4ade532018-11-12 17:25:23 +000093
94 return true;
95}
96
97bool lldb_private::formatters::NSException_SummaryProvider(
98 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {
99 lldb::ValueObjectSP name_sp;
100 lldb::ValueObjectSP reason_sp;
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000101 if (!ExtractFields(valobj, &name_sp, &reason_sp, nullptr, nullptr))
Kuba Mracekb4ade532018-11-12 17:25:23 +0000102 return false;
Kate Stoneb9c1b512016-09-06 20:57:50 +0000103
104 if (!name_sp || !reason_sp)
105 return false;
106
107 StreamString name_str_summary;
108 StreamString reason_str_summary;
109 if (NSStringSummaryProvider(*name_sp, name_str_summary, options) &&
110 NSStringSummaryProvider(*reason_sp, reason_str_summary, options) &&
111 !name_str_summary.Empty() && !reason_str_summary.Empty()) {
112 stream.Printf("name: %s - reason: %s", name_str_summary.GetData(),
113 reason_str_summary.GetData());
114 return true;
115 } else
116 return false;
Enrico Granata16709ef2015-11-06 02:43:32 +0000117}
118
Kate Stoneb9c1b512016-09-06 20:57:50 +0000119class NSExceptionSyntheticFrontEnd : public SyntheticChildrenFrontEnd {
Enrico Granata16709ef2015-11-06 02:43:32 +0000120public:
Kate Stoneb9c1b512016-09-06 20:57:50 +0000121 NSExceptionSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)
122 : SyntheticChildrenFrontEnd(*valobj_sp) {}
Eugene Zelenko26f34fb2015-11-07 00:28:50 +0000123
Kate Stoneb9c1b512016-09-06 20:57:50 +0000124 ~NSExceptionSyntheticFrontEnd() override = default;
Eugene Zelenko26f34fb2015-11-07 00:28:50 +0000125
Kate Stoneb9c1b512016-09-06 20:57:50 +0000126 size_t CalculateNumChildren() override {
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000127 return 4;
Kate Stoneb9c1b512016-09-06 20:57:50 +0000128 }
129
130 lldb::ValueObjectSP GetChildAtIndex(size_t idx) override {
Kuba Mracekb4ade532018-11-12 17:25:23 +0000131 switch (idx) {
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000132 case 0: return m_name_sp;
133 case 1: return m_reason_sp;
134 case 2: return m_userinfo_sp;
135 case 3: return m_reserved_sp;
Kuba Mracekb4ade532018-11-12 17:25:23 +0000136 }
137 return lldb::ValueObjectSP();
Kate Stoneb9c1b512016-09-06 20:57:50 +0000138 }
139
140 bool Update() override {
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000141 m_name_sp.reset();
142 m_reason_sp.reset();
Kuba Mracekb4ade532018-11-12 17:25:23 +0000143 m_userinfo_sp.reset();
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000144 m_reserved_sp.reset();
145
Jonas Devliegherea6682a42018-12-15 00:15:33 +0000146 return ExtractFields(m_backend, &m_name_sp, &m_reason_sp, &m_userinfo_sp,
147 &m_reserved_sp);
Kate Stoneb9c1b512016-09-06 20:57:50 +0000148 }
149
150 bool MightHaveChildren() override { return true; }
151
Adrian Prantl0e4c4822019-03-06 21:22:25 +0000152 size_t GetIndexOfChildWithName(ConstString name) override {
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000153 // NSException has 4 members:
154 // NSString *name;
155 // NSString *reason;
156 // NSDictionary *userInfo;
157 // id reserved;
158 static ConstString g___name("name");
159 static ConstString g___reason("reason");
Kate Stoneb9c1b512016-09-06 20:57:50 +0000160 static ConstString g___userInfo("userInfo");
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000161 static ConstString g___reserved("reserved");
162 if (name == g___name) return 0;
163 if (name == g___reason) return 1;
164 if (name == g___userInfo) return 2;
165 if (name == g___reserved) return 3;
Kate Stoneb9c1b512016-09-06 20:57:50 +0000166 return UINT32_MAX;
167 }
Eugene Zelenko26f34fb2015-11-07 00:28:50 +0000168
Enrico Granata16709ef2015-11-06 02:43:32 +0000169private:
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000170 ValueObjectSP m_name_sp;
171 ValueObjectSP m_reason_sp;
Kuba Mracekb4ade532018-11-12 17:25:23 +0000172 ValueObjectSP m_userinfo_sp;
Kuba Mracekc3f1e622018-11-12 19:12:31 +0000173 ValueObjectSP m_reserved_sp;
Enrico Granata16709ef2015-11-06 02:43:32 +0000174};
175
Kate Stoneb9c1b512016-09-06 20:57:50 +0000176SyntheticChildrenFrontEnd *
177lldb_private::formatters::NSExceptionSyntheticFrontEndCreator(
178 CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) {
179 lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());
180 if (!process_sp)
Enrico Granata16709ef2015-11-06 02:43:32 +0000181 return nullptr;
Alex Langforde823bbe2019-06-10 20:53:23 +0000182 ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);
Kate Stoneb9c1b512016-09-06 20:57:50 +0000183 if (!runtime)
184 return nullptr;
185
186 ObjCLanguageRuntime::ClassDescriptorSP descriptor(
187 runtime->GetClassDescriptor(*valobj_sp.get()));
188
189 if (!descriptor.get() || !descriptor->IsValid())
190 return nullptr;
191
192 const char *class_name = descriptor->GetClassName().GetCString();
193
194 if (!class_name || !*class_name)
195 return nullptr;
196
197 if (!strcmp(class_name, "NSException"))
198 return (new NSExceptionSyntheticFrontEnd(valobj_sp));
199 else if (!strcmp(class_name, "NSCFException"))
200 return (new NSExceptionSyntheticFrontEnd(valobj_sp));
201 else if (!strcmp(class_name, "__NSCFException"))
202 return (new NSExceptionSyntheticFrontEnd(valobj_sp));
203
204 return nullptr;
Enrico Granata16709ef2015-11-06 02:43:32 +0000205}