1 //===-- Cocoa.cpp -------------------------------------------------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 #include "lldb/DataFormatters/CXXFormatterFunctions.h"
12 #include "lldb/Core/DataBufferHeap.h"
13 #include "lldb/Core/Error.h"
14 #include "lldb/Core/Stream.h"
15 #include "lldb/Core/ValueObject.h"
16 #include "lldb/Core/ValueObjectConstResult.h"
17 #include "lldb/Host/Endian.h"
18 #include "lldb/Symbol/ClangASTContext.h"
19 #include "lldb/Target/ObjCLanguageRuntime.h"
20 #include "lldb/Target/Target.h"
23 using namespace lldb_private;
24 using namespace lldb_private::formatters;
27 lldb_private::formatters::NSBundleSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
29 ProcessSP process_sp = valobj.GetProcessSP();
33 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
38 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
40 if (!descriptor.get() || !descriptor->IsValid())
43 uint32_t ptr_size = process_sp->GetAddressByteSize();
45 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
50 const char* class_name = descriptor->GetClassName().GetCString();
52 if (!class_name || !*class_name)
55 if (!strcmp(class_name,"NSBundle"))
57 uint64_t offset = 5 * ptr_size;
58 ValueObjectSP text(valobj.GetSyntheticChildAtOffset(offset, valobj.GetClangType().GetBasicTypeFromAST(lldb::eBasicTypeObjCID), true));
60 StreamString summary_stream;
61 bool was_nsstring_ok = NSStringSummaryProvider(*text.get(), summary_stream, options);
62 if (was_nsstring_ok && summary_stream.GetSize() > 0)
64 stream.Printf("%s",summary_stream.GetData());
68 // this is either an unknown subclass or an NSBundle that comes from [NSBundle mainBundle]
69 // which is encoded differently and needs to be handled by running code
70 return ExtractSummaryFromObjCExpression(valobj, "NSString*", "bundlePath", stream);
74 lldb_private::formatters::NSTimeZoneSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
76 ProcessSP process_sp = valobj.GetProcessSP();
80 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
85 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
87 if (!descriptor.get() || !descriptor->IsValid())
90 uint32_t ptr_size = process_sp->GetAddressByteSize();
92 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
97 const char* class_name = descriptor->GetClassName().GetCString();
99 if (!class_name || !*class_name)
102 if (!strcmp(class_name,"__NSTimeZone"))
104 uint64_t offset = ptr_size;
105 ValueObjectSP text(valobj.GetSyntheticChildAtOffset(offset, valobj.GetClangType(), true));
106 StreamString summary_stream;
107 bool was_nsstring_ok = NSStringSummaryProvider(*text.get(), summary_stream, options);
108 if (was_nsstring_ok && summary_stream.GetSize() > 0)
110 stream.Printf("%s",summary_stream.GetData());
114 return ExtractSummaryFromObjCExpression(valobj, "NSString*", "name", stream);
118 lldb_private::formatters::NSNotificationSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
120 ProcessSP process_sp = valobj.GetProcessSP();
124 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
129 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
131 if (!descriptor.get() || !descriptor->IsValid())
134 uint32_t ptr_size = process_sp->GetAddressByteSize();
136 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
141 const char* class_name = descriptor->GetClassName().GetCString();
143 if (!class_name || !*class_name)
146 if (!strcmp(class_name,"NSConcreteNotification"))
148 uint64_t offset = ptr_size;
149 ValueObjectSP text(valobj.GetSyntheticChildAtOffset(offset, valobj.GetClangType(), true));
150 StreamString summary_stream;
151 bool was_nsstring_ok = NSStringSummaryProvider(*text.get(), summary_stream, options);
152 if (was_nsstring_ok && summary_stream.GetSize() > 0)
154 stream.Printf("%s",summary_stream.GetData());
158 // this is either an unknown subclass or an NSBundle that comes from [NSBundle mainBundle]
159 // which is encoded differently and needs to be handled by running code
160 return ExtractSummaryFromObjCExpression(valobj, "NSString*", "name", stream);
164 lldb_private::formatters::NSMachPortSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
166 ProcessSP process_sp = valobj.GetProcessSP();
170 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
175 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
177 if (!descriptor.get() || !descriptor->IsValid())
180 uint32_t ptr_size = process_sp->GetAddressByteSize();
182 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
187 const char* class_name = descriptor->GetClassName().GetCString();
189 if (!class_name || !*class_name)
192 uint64_t port_number = 0;
196 if (!strcmp(class_name,"NSMachPort"))
198 uint64_t offset = (ptr_size == 4 ? 12 : 20);
200 port_number = process_sp->ReadUnsignedIntegerFromMemory(offset+valobj_addr, 4, 0, error);
204 if (!ExtractValueFromObjCExpression(valobj, "int", "machPort", port_number))
208 stream.Printf("mach port: %u",(uint32_t)(port_number & 0x00000000FFFFFFFF));
213 lldb_private::formatters::NSIndexSetSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
215 ProcessSP process_sp = valobj.GetProcessSP();
219 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
224 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
226 if (!descriptor.get() || !descriptor->IsValid())
229 uint32_t ptr_size = process_sp->GetAddressByteSize();
231 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
236 const char* class_name = descriptor->GetClassName().GetCString();
238 if (!class_name || !*class_name)
244 if (!strcmp(class_name,"NSIndexSet") || !strcmp(class_name,"NSMutableIndexSet"))
247 uint32_t mode = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr+ptr_size, 4, 0, error);
250 // this means the set is empty - count = 0
257 mode = 1; // this means the set only has one range
259 mode = 2; // this means the set has multiple ranges
262 count = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr+3*ptr_size, ptr_size, 0, error);
268 // read a pointer to the data at 2*ptr_size
269 count = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr+2*ptr_size, ptr_size, 0, error);
272 // read the data at 2*ptr_size from the first location
273 count = process_sp->ReadUnsignedIntegerFromMemory(count+2*ptr_size, ptr_size, 0, error);
280 if (!ExtractValueFromObjCExpression(valobj, "unsigned long long int", "count", count))
284 stream.Printf("%" PRIu64 " index%s",
286 (count == 1 ? "" : "es"));
291 lldb_private::formatters::NSNumberSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
293 ProcessSP process_sp = valobj.GetProcessSP();
297 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
302 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
304 if (!descriptor.get() || !descriptor->IsValid())
307 uint32_t ptr_size = process_sp->GetAddressByteSize();
309 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
314 const char* class_name = descriptor->GetClassName().GetCString();
316 if (!class_name || !*class_name)
319 if (!strcmp(class_name,"NSNumber") || !strcmp(class_name,"__NSCFNumber"))
323 if (descriptor->GetTaggedPointerInfo(&i_bits,&value))
328 stream.Printf("(char)%hhd",(char)value);
332 stream.Printf("(short)%hd",(short)value);
336 stream.Printf("(int)%d",(int)value);
340 stream.Printf("(long)%" PRId64,value);
350 uint8_t data_type = (process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size, 1, 0, error) & 0x1F);
351 uint64_t data_location = valobj_addr + 2*ptr_size;
358 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 1, 0, error);
361 stream.Printf("(char)%hhd",(char)value);
364 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 2, 0, error);
367 stream.Printf("(short)%hd",(short)value);
370 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 4, 0, error);
373 stream.Printf("(int)%d",(int)value);
378 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 8, 0, error);
381 stream.Printf("(long)%" PRId64,value);
385 uint32_t flt_as_int = process_sp->ReadUnsignedIntegerFromMemory(data_location, 4, 0, error);
388 float flt_value = *((float*)&flt_as_int);
389 stream.Printf("(float)%f",flt_value);
394 uint64_t dbl_as_lng = process_sp->ReadUnsignedIntegerFromMemory(data_location, 8, 0, error);
397 double dbl_value = *((double*)&dbl_as_lng);
398 stream.Printf("(double)%g",dbl_value);
409 return ExtractSummaryFromObjCExpression(valobj, "NSString*", "stringValue", stream);
414 lldb_private::formatters::NSURLSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
416 ProcessSP process_sp = valobj.GetProcessSP();
420 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
425 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
427 if (!descriptor.get() || !descriptor->IsValid())
430 uint32_t ptr_size = process_sp->GetAddressByteSize();
432 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
437 const char* class_name = descriptor->GetClassName().GetCString();
439 if (!class_name || !*class_name)
442 if (strcmp(class_name, "NSURL") == 0)
444 uint64_t offset_text = ptr_size + ptr_size + 8; // ISA + pointer + 8 bytes of data (even on 32bit)
445 uint64_t offset_base = offset_text + ptr_size;
446 ClangASTType type(valobj.GetClangType());
447 ValueObjectSP text(valobj.GetSyntheticChildAtOffset(offset_text, type, true));
448 ValueObjectSP base(valobj.GetSyntheticChildAtOffset(offset_base, type, true));
451 if (text->GetValueAsUnsigned(0) == 0)
453 StreamString summary;
454 if (!NSStringSummaryProvider(*text, summary, options))
456 if (base && base->GetValueAsUnsigned(0))
458 if (summary.GetSize() > 0)
459 summary.GetString().resize(summary.GetSize()-1);
460 summary.Printf(" -- ");
461 StreamString base_summary;
462 if (NSURLSummaryProvider(*base, base_summary, options) && base_summary.GetSize() > 0)
463 summary.Printf("%s",base_summary.GetSize() > 2 ? base_summary.GetData() + 2 : base_summary.GetData());
465 if (summary.GetSize())
467 stream.Printf("%s",summary.GetData());
473 return ExtractSummaryFromObjCExpression(valobj, "NSString*", "description", stream);
479 lldb_private::formatters::NSDateSummaryProvider (ValueObject& valobj, Stream& stream, const TypeSummaryOptions& options)
481 ProcessSP process_sp = valobj.GetProcessSP();
485 ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
490 ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(valobj));
492 if (!descriptor.get() || !descriptor->IsValid())
495 uint32_t ptr_size = process_sp->GetAddressByteSize();
497 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
502 uint64_t date_value_bits = 0;
503 double date_value = 0.0;
505 const char* class_name = descriptor->GetClassName().GetCString();
507 if (!class_name || !*class_name)
510 if (strcmp(class_name,"NSDate") == 0 ||
511 strcmp(class_name,"__NSDate") == 0 ||
512 strcmp(class_name,"__NSTaggedDate") == 0)
514 uint64_t info_bits=0,value_bits = 0;
515 if (descriptor->GetTaggedPointerInfo(&info_bits,&value_bits))
517 date_value_bits = ((value_bits << 8) | (info_bits << 4));
518 date_value = *((double*)&date_value_bits);
523 date_value_bits = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr+ptr_size, 8, 0, error);
524 date_value = *((double*)&date_value_bits);
529 else if (!strcmp(class_name,"NSCalendarDate"))
532 date_value_bits = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr+2*ptr_size, 8, 0, error);
533 date_value = *((double*)&date_value_bits);
539 if (ExtractValueFromObjCExpression(valobj, "NSTimeInterval", "ExtractValueFromObjCExpression", date_value_bits) == false)
541 date_value = *((double*)&date_value_bits);
543 if (date_value == -63114076800)
545 stream.Printf("0001-12-30 00:00:00 +0000");
548 // this snippet of code assumes that time_t == seconds since Jan-1-1970
549 // this is generally true and POSIXly happy, but might break if a library
550 // vendor decides to get creative
551 time_t epoch = GetOSXEpoch();
552 epoch = epoch + (time_t)date_value;
553 tm *tm_date = gmtime(&epoch);
556 std::string buffer(1024,0);
557 if (strftime (&buffer[0], 1023, "%Z", tm_date) == 0)
559 stream.Printf("%04d-%02d-%02d %02d:%02d:%02d %s", tm_date->tm_year+1900, tm_date->tm_mon+1, tm_date->tm_mday, tm_date->tm_hour, tm_date->tm_min, tm_date->tm_sec, buffer.c_str());