blob: 66be60ffcb9dd2f93d7ac1dc1437c0d25708f71e [file] [log] [blame]
Eugene Zelenko26f34fb2015-11-07 00:28:50 +00001//===-- NSException.cpp -----------------------------------------*- C++ -*-===//
Enrico Granata16709ef2015-11-06 02:43:32 +00002//
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
Eugene Zelenko26f34fb2015-11-07 00:28:50 +000010#include "clang/AST/DeclCXX.h"
11
Enrico Granata16709ef2015-11-06 02:43:32 +000012#include "Cocoa.h"
13
Enrico Granata16709ef2015-11-06 02:43:32 +000014#include "lldb/Core/ValueObject.h"
15#include "lldb/Core/ValueObjectConstResult.h"
16#include "lldb/DataFormatters/FormattersHelpers.h"
Enrico Granata16709ef2015-11-06 02:43:32 +000017#include "lldb/Symbol/ClangASTContext.h"
18#include "lldb/Target/ObjCLanguageRuntime.h"
Zachary Turner01c32432017-02-14 19:06:07 +000019#include "lldb/Target/ProcessStructReader.h"
Enrico Granata16709ef2015-11-06 02:43:32 +000020#include "lldb/Target/Target.h"
Zachary Turner666cc0b2017-03-04 01:30:05 +000021#include "lldb/Utility/DataBufferHeap.h"
Zachary Turner01c32432017-02-14 19:06:07 +000022#include "lldb/Utility/Endian.h"
Zachary Turner97206d52017-05-12 04:51:55 +000023#include "lldb/Utility/Status.h"
Zachary Turnerbf9a7732017-02-02 21:39:50 +000024#include "lldb/Utility/Stream.h"
Enrico Granata16709ef2015-11-06 02:43:32 +000025
Enrico Granata16709ef2015-11-06 02:43:32 +000026#include "Plugins/Language/ObjC/NSString.h"
27
28using namespace lldb;
29using namespace lldb_private;
30using namespace lldb_private::formatters;
31
Kuba Mracekb4ade532018-11-12 17:25:23 +000032static bool ExtractFields(ValueObject &valobj, ValueObjectSP *name_sp,
33 ValueObjectSP *reason_sp,
34 ValueObjectSP *userinfo_sp) {
Kate Stoneb9c1b512016-09-06 20:57:50 +000035 ProcessSP process_sp(valobj.GetProcessSP());
36 if (!process_sp)
37 return false;
38
Kuba Mracekb4ade532018-11-12 17:25:23 +000039 lldb::addr_t ptr = LLDB_INVALID_ADDRESS;
Kate Stoneb9c1b512016-09-06 20:57:50 +000040
41 CompilerType valobj_type(valobj.GetCompilerType());
42 Flags type_flags(valobj_type.GetTypeInfo());
43 if (type_flags.AllClear(eTypeHasValue)) {
44 if (valobj.IsBaseClass() && valobj.GetParent())
Kuba Mracekb4ade532018-11-12 17:25:23 +000045 ptr = valobj.GetParent()->GetValueAsUnsigned(LLDB_INVALID_ADDRESS);
46 } else {
47 ptr = valobj.GetValueAsUnsigned(LLDB_INVALID_ADDRESS);
48 }
Kate Stoneb9c1b512016-09-06 20:57:50 +000049
Kuba Mracekb4ade532018-11-12 17:25:23 +000050 if (ptr == LLDB_INVALID_ADDRESS)
Kate Stoneb9c1b512016-09-06 20:57:50 +000051 return false;
52 size_t ptr_size = process_sp->GetAddressByteSize();
Kate Stoneb9c1b512016-09-06 20:57:50 +000053
Zachary Turner97206d52017-05-12 04:51:55 +000054 Status error;
Kuba Mracekb4ade532018-11-12 17:25:23 +000055 auto name = process_sp->ReadPointerFromMemory(ptr + 1 * ptr_size, error);
Kate Stoneb9c1b512016-09-06 20:57:50 +000056 if (error.Fail() || name == LLDB_INVALID_ADDRESS)
57 return false;
Kuba Mracekb4ade532018-11-12 17:25:23 +000058 auto reason = process_sp->ReadPointerFromMemory(ptr + 2 * ptr_size, error);
Kate Stoneb9c1b512016-09-06 20:57:50 +000059 if (error.Fail() || reason == LLDB_INVALID_ADDRESS)
60 return false;
Kuba Mracekb4ade532018-11-12 17:25:23 +000061 auto userinfo = process_sp->ReadPointerFromMemory(ptr + 3 * ptr_size, error);
62 if (error.Fail() || userinfo == LLDB_INVALID_ADDRESS)
63 return false;
Kate Stoneb9c1b512016-09-06 20:57:50 +000064
65 InferiorSizedWord name_isw(name, *process_sp);
66 InferiorSizedWord reason_isw(reason, *process_sp);
Kuba Mracekb4ade532018-11-12 17:25:23 +000067 InferiorSizedWord userinfo_isw(userinfo, *process_sp);
Kate Stoneb9c1b512016-09-06 20:57:50 +000068
69 CompilerType voidstar = process_sp->GetTarget()
70 .GetScratchClangASTContext()
71 ->GetBasicType(lldb::eBasicTypeVoid)
72 .GetPointerType();
73
Kuba Mracekb4ade532018-11-12 17:25:23 +000074 if (name_sp)
75 *name_sp = ValueObject::CreateValueObjectFromData(
76 "name", name_isw.GetAsData(process_sp->GetByteOrder()),
77 valobj.GetExecutionContextRef(), voidstar);
78 if (reason_sp)
79 *reason_sp = ValueObject::CreateValueObjectFromData(
80 "reason", reason_isw.GetAsData(process_sp->GetByteOrder()),
81 valobj.GetExecutionContextRef(), voidstar);
82 if (userinfo_sp)
83 *userinfo_sp = ValueObject::CreateValueObjectFromData(
84 "userInfo", userinfo_isw.GetAsData(process_sp->GetByteOrder()),
85 valobj.GetExecutionContextRef(), voidstar);
86
87 return true;
88}
89
90bool lldb_private::formatters::NSException_SummaryProvider(
91 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {
92 lldb::ValueObjectSP name_sp;
93 lldb::ValueObjectSP reason_sp;
94 if (!ExtractFields(valobj, &name_sp, &reason_sp, nullptr))
95 return false;
Kate Stoneb9c1b512016-09-06 20:57:50 +000096
97 if (!name_sp || !reason_sp)
98 return false;
99
100 StreamString name_str_summary;
101 StreamString reason_str_summary;
102 if (NSStringSummaryProvider(*name_sp, name_str_summary, options) &&
103 NSStringSummaryProvider(*reason_sp, reason_str_summary, options) &&
104 !name_str_summary.Empty() && !reason_str_summary.Empty()) {
105 stream.Printf("name: %s - reason: %s", name_str_summary.GetData(),
106 reason_str_summary.GetData());
107 return true;
108 } else
109 return false;
Enrico Granata16709ef2015-11-06 02:43:32 +0000110}
111
Kate Stoneb9c1b512016-09-06 20:57:50 +0000112class NSExceptionSyntheticFrontEnd : public SyntheticChildrenFrontEnd {
Enrico Granata16709ef2015-11-06 02:43:32 +0000113public:
Kate Stoneb9c1b512016-09-06 20:57:50 +0000114 NSExceptionSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)
115 : SyntheticChildrenFrontEnd(*valobj_sp) {}
Eugene Zelenko26f34fb2015-11-07 00:28:50 +0000116
Kate Stoneb9c1b512016-09-06 20:57:50 +0000117 ~NSExceptionSyntheticFrontEnd() override = default;
Eugene Zelenko26f34fb2015-11-07 00:28:50 +0000118
Kate Stoneb9c1b512016-09-06 20:57:50 +0000119 size_t CalculateNumChildren() override {
Kuba Mracekb4ade532018-11-12 17:25:23 +0000120 return 1;
Kate Stoneb9c1b512016-09-06 20:57:50 +0000121 }
122
123 lldb::ValueObjectSP GetChildAtIndex(size_t idx) override {
Kuba Mracekb4ade532018-11-12 17:25:23 +0000124 switch (idx) {
125 case 0: return m_userinfo_sp;
126 }
127 return lldb::ValueObjectSP();
Kate Stoneb9c1b512016-09-06 20:57:50 +0000128 }
129
130 bool Update() override {
Kuba Mracekb4ade532018-11-12 17:25:23 +0000131 m_userinfo_sp.reset();
132 if (!ExtractFields(m_backend, nullptr, nullptr, &m_userinfo_sp)) {
Kate Stoneb9c1b512016-09-06 20:57:50 +0000133 return false;
Kuba Mracekb4ade532018-11-12 17:25:23 +0000134 }
135 return true;
Kate Stoneb9c1b512016-09-06 20:57:50 +0000136 }
137
138 bool MightHaveChildren() override { return true; }
139
140 size_t GetIndexOfChildWithName(const ConstString &name) override {
141 static ConstString g___userInfo("userInfo");
142 if (name == g___userInfo)
143 return 0;
144 return UINT32_MAX;
145 }
Eugene Zelenko26f34fb2015-11-07 00:28:50 +0000146
Enrico Granata16709ef2015-11-06 02:43:32 +0000147private:
Kuba Mracekb4ade532018-11-12 17:25:23 +0000148 ValueObjectSP m_userinfo_sp;
Enrico Granata16709ef2015-11-06 02:43:32 +0000149};
150
Kate Stoneb9c1b512016-09-06 20:57:50 +0000151SyntheticChildrenFrontEnd *
152lldb_private::formatters::NSExceptionSyntheticFrontEndCreator(
153 CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) {
154 lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());
155 if (!process_sp)
Enrico Granata16709ef2015-11-06 02:43:32 +0000156 return nullptr;
Kate Stoneb9c1b512016-09-06 20:57:50 +0000157 ObjCLanguageRuntime *runtime =
158 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime(
159 lldb::eLanguageTypeObjC);
160 if (!runtime)
161 return nullptr;
162
163 ObjCLanguageRuntime::ClassDescriptorSP descriptor(
164 runtime->GetClassDescriptor(*valobj_sp.get()));
165
166 if (!descriptor.get() || !descriptor->IsValid())
167 return nullptr;
168
169 const char *class_name = descriptor->GetClassName().GetCString();
170
171 if (!class_name || !*class_name)
172 return nullptr;
173
174 if (!strcmp(class_name, "NSException"))
175 return (new NSExceptionSyntheticFrontEnd(valobj_sp));
176 else if (!strcmp(class_name, "NSCFException"))
177 return (new NSExceptionSyntheticFrontEnd(valobj_sp));
178 else if (!strcmp(class_name, "__NSCFException"))
179 return (new NSExceptionSyntheticFrontEnd(valobj_sp));
180
181 return nullptr;
Enrico Granata16709ef2015-11-06 02:43:32 +0000182}