1 //===-- Cocoa.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 // C Includes 11 // C++ Includes 12 // Other libraries and framework includes 13 // Project includes 14 #include "Cocoa.h" 15 16 #include "lldb/Core/DataBufferHeap.h" 17 #include "lldb/Core/Error.h" 18 #include "lldb/Core/Mangled.h" 19 #include "lldb/Core/Stream.h" 20 #include "lldb/Core/ValueObject.h" 21 #include "lldb/Core/ValueObjectConstResult.h" 22 #include "lldb/DataFormatters/FormattersHelpers.h" 23 #include "lldb/DataFormatters/StringPrinter.h" 24 #include "lldb/DataFormatters/TypeSummary.h" 25 #include "lldb/Host/Endian.h" 26 #include "lldb/Symbol/ClangASTContext.h" 27 #include "lldb/Target/Language.h" 28 #include "lldb/Target/ObjCLanguageRuntime.h" 29 #include "lldb/Target/Process.h" 30 #include "lldb/Target/Target.h" 31 #include "lldb/Utility/ProcessStructReader.h" 32 33 #include "Plugins/LanguageRuntime/ObjC/AppleObjCRuntime/AppleObjCRuntime.h" 34 35 #include "NSString.h" 36 37 using namespace lldb; 38 using namespace lldb_private; 39 using namespace lldb_private::formatters; 40 41 bool lldb_private::formatters::NSBundleSummaryProvider( 42 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 43 ProcessSP process_sp = valobj.GetProcessSP(); 44 if (!process_sp) 45 return false; 46 47 ObjCLanguageRuntime *runtime = 48 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 49 lldb::eLanguageTypeObjC); 50 51 if (!runtime) 52 return false; 53 54 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 55 runtime->GetClassDescriptor(valobj)); 56 57 if (!descriptor || !descriptor->IsValid()) 58 return false; 59 60 uint32_t ptr_size = process_sp->GetAddressByteSize(); 61 62 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 63 64 if (!valobj_addr) 65 return false; 66 67 const char *class_name = descriptor->GetClassName().GetCString(); 68 69 if (!class_name || !*class_name) 70 return false; 71 72 if (!strcmp(class_name, "NSBundle")) { 73 uint64_t offset = 5 * ptr_size; 74 ValueObjectSP text(valobj.GetSyntheticChildAtOffset( 75 offset, 76 valobj.GetCompilerType().GetBasicTypeFromAST(lldb::eBasicTypeObjCID), 77 true)); 78 79 StreamString summary_stream; 80 bool was_nsstring_ok = 81 NSStringSummaryProvider(*text, summary_stream, options); 82 if (was_nsstring_ok && summary_stream.GetSize() > 0) { 83 stream.Printf("%s", summary_stream.GetData()); 84 return true; 85 } 86 } 87 88 return false; 89 } 90 91 bool lldb_private::formatters::NSTimeZoneSummaryProvider( 92 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 93 ProcessSP process_sp = valobj.GetProcessSP(); 94 if (!process_sp) 95 return false; 96 97 ObjCLanguageRuntime *runtime = 98 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 99 lldb::eLanguageTypeObjC); 100 101 if (!runtime) 102 return false; 103 104 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 105 runtime->GetClassDescriptor(valobj)); 106 107 if (!descriptor || !descriptor->IsValid()) 108 return false; 109 110 uint32_t ptr_size = process_sp->GetAddressByteSize(); 111 112 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 113 114 if (!valobj_addr) 115 return false; 116 117 const char *class_name = descriptor->GetClassName().GetCString(); 118 119 if (!class_name || !*class_name) 120 return false; 121 122 if (!strcmp(class_name, "__NSTimeZone")) { 123 uint64_t offset = ptr_size; 124 ValueObjectSP text(valobj.GetSyntheticChildAtOffset( 125 offset, valobj.GetCompilerType(), true)); 126 StreamString summary_stream; 127 bool was_nsstring_ok = 128 NSStringSummaryProvider(*text, summary_stream, options); 129 if (was_nsstring_ok && summary_stream.GetSize() > 0) { 130 stream.Printf("%s", summary_stream.GetData()); 131 return true; 132 } 133 } 134 135 return false; 136 } 137 138 bool lldb_private::formatters::NSNotificationSummaryProvider( 139 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 140 ProcessSP process_sp = valobj.GetProcessSP(); 141 if (!process_sp) 142 return false; 143 144 ObjCLanguageRuntime *runtime = 145 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 146 lldb::eLanguageTypeObjC); 147 148 if (!runtime) 149 return false; 150 151 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 152 runtime->GetClassDescriptor(valobj)); 153 154 if (!descriptor || !descriptor->IsValid()) 155 return false; 156 157 uint32_t ptr_size = process_sp->GetAddressByteSize(); 158 159 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 160 161 if (!valobj_addr) 162 return false; 163 164 const char *class_name = descriptor->GetClassName().GetCString(); 165 166 if (!class_name || !*class_name) 167 return false; 168 169 if (!strcmp(class_name, "NSConcreteNotification")) { 170 uint64_t offset = ptr_size; 171 ValueObjectSP text(valobj.GetSyntheticChildAtOffset( 172 offset, valobj.GetCompilerType(), true)); 173 StreamString summary_stream; 174 bool was_nsstring_ok = 175 NSStringSummaryProvider(*text, summary_stream, options); 176 if (was_nsstring_ok && summary_stream.GetSize() > 0) { 177 stream.Printf("%s", summary_stream.GetData()); 178 return true; 179 } 180 } 181 182 return false; 183 } 184 185 bool lldb_private::formatters::NSMachPortSummaryProvider( 186 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 187 ProcessSP process_sp = valobj.GetProcessSP(); 188 if (!process_sp) 189 return false; 190 191 ObjCLanguageRuntime *runtime = 192 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 193 lldb::eLanguageTypeObjC); 194 195 if (!runtime) 196 return false; 197 198 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 199 runtime->GetClassDescriptor(valobj)); 200 201 if (!descriptor || !descriptor->IsValid()) 202 return false; 203 204 uint32_t ptr_size = process_sp->GetAddressByteSize(); 205 206 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 207 208 if (!valobj_addr) 209 return false; 210 211 const char *class_name = descriptor->GetClassName().GetCString(); 212 213 if (!class_name || !*class_name) 214 return false; 215 216 uint64_t port_number = 0; 217 218 if (!strcmp(class_name, "NSMachPort")) { 219 uint64_t offset = (ptr_size == 4 ? 12 : 20); 220 Error error; 221 port_number = process_sp->ReadUnsignedIntegerFromMemory( 222 offset + valobj_addr, 4, 0, error); 223 if (error.Success()) { 224 stream.Printf("mach port: %u", 225 (uint32_t)(port_number & 0x00000000FFFFFFFF)); 226 return true; 227 } 228 } 229 230 return false; 231 } 232 233 bool lldb_private::formatters::NSIndexSetSummaryProvider( 234 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 235 ProcessSP process_sp = valobj.GetProcessSP(); 236 if (!process_sp) 237 return false; 238 239 ObjCLanguageRuntime *runtime = 240 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 241 lldb::eLanguageTypeObjC); 242 243 if (!runtime) 244 return false; 245 246 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 247 runtime->GetClassDescriptor(valobj)); 248 249 if (!descriptor || !descriptor->IsValid()) 250 return false; 251 252 uint32_t ptr_size = process_sp->GetAddressByteSize(); 253 254 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 255 256 if (!valobj_addr) 257 return false; 258 259 const char *class_name = descriptor->GetClassName().GetCString(); 260 261 if (!class_name || !*class_name) 262 return false; 263 264 uint64_t count = 0; 265 266 do { 267 if (!strcmp(class_name, "NSIndexSet") || 268 !strcmp(class_name, "NSMutableIndexSet")) { 269 Error error; 270 uint32_t mode = process_sp->ReadUnsignedIntegerFromMemory( 271 valobj_addr + ptr_size, 4, 0, error); 272 if (error.Fail()) 273 return false; 274 // this means the set is empty - count = 0 275 if ((mode & 1) == 1) { 276 count = 0; 277 break; 278 } 279 if ((mode & 2) == 2) 280 mode = 1; // this means the set only has one range 281 else 282 mode = 2; // this means the set has multiple ranges 283 if (mode == 1) { 284 count = process_sp->ReadUnsignedIntegerFromMemory( 285 valobj_addr + 3 * ptr_size, ptr_size, 0, error); 286 if (error.Fail()) 287 return false; 288 } else { 289 // read a pointer to the data at 2*ptr_size 290 count = process_sp->ReadUnsignedIntegerFromMemory( 291 valobj_addr + 2 * ptr_size, ptr_size, 0, error); 292 if (error.Fail()) 293 return false; 294 // read the data at 2*ptr_size from the first location 295 count = process_sp->ReadUnsignedIntegerFromMemory(count + 2 * ptr_size, 296 ptr_size, 0, error); 297 if (error.Fail()) 298 return false; 299 } 300 } else 301 return false; 302 } while (false); 303 stream.Printf("%" PRIu64 " index%s", count, (count == 1 ? "" : "es")); 304 return true; 305 } 306 307 static void NSNumber_FormatChar(ValueObject &valobj, Stream &stream, char value, 308 lldb::LanguageType lang) { 309 static ConstString g_TypeHint("NSNumber:char"); 310 311 std::string prefix, suffix; 312 if (Language *language = Language::FindPlugin(lang)) { 313 if (!language->GetFormatterPrefixSuffix(valobj, g_TypeHint, prefix, 314 suffix)) { 315 prefix.clear(); 316 suffix.clear(); 317 } 318 } 319 320 stream.Printf("%s%hhd%s", prefix.c_str(), value, suffix.c_str()); 321 } 322 323 static void NSNumber_FormatShort(ValueObject &valobj, Stream &stream, 324 short value, lldb::LanguageType lang) { 325 static ConstString g_TypeHint("NSNumber:short"); 326 327 std::string prefix, suffix; 328 if (Language *language = Language::FindPlugin(lang)) { 329 if (!language->GetFormatterPrefixSuffix(valobj, g_TypeHint, prefix, 330 suffix)) { 331 prefix.clear(); 332 suffix.clear(); 333 } 334 } 335 336 stream.Printf("%s%hd%s", prefix.c_str(), value, suffix.c_str()); 337 } 338 339 static void NSNumber_FormatInt(ValueObject &valobj, Stream &stream, int value, 340 lldb::LanguageType lang) { 341 static ConstString g_TypeHint("NSNumber:int"); 342 343 std::string prefix, suffix; 344 if (Language *language = Language::FindPlugin(lang)) { 345 if (!language->GetFormatterPrefixSuffix(valobj, g_TypeHint, prefix, 346 suffix)) { 347 prefix.clear(); 348 suffix.clear(); 349 } 350 } 351 352 stream.Printf("%s%d%s", prefix.c_str(), value, suffix.c_str()); 353 } 354 355 static void NSNumber_FormatLong(ValueObject &valobj, Stream &stream, 356 uint64_t value, lldb::LanguageType lang) { 357 static ConstString g_TypeHint("NSNumber:long"); 358 359 std::string prefix, suffix; 360 if (Language *language = Language::FindPlugin(lang)) { 361 if (!language->GetFormatterPrefixSuffix(valobj, g_TypeHint, prefix, 362 suffix)) { 363 prefix.clear(); 364 suffix.clear(); 365 } 366 } 367 368 stream.Printf("%s%" PRId64 "%s", prefix.c_str(), value, suffix.c_str()); 369 } 370 371 static void NSNumber_FormatFloat(ValueObject &valobj, Stream &stream, 372 float value, lldb::LanguageType lang) { 373 static ConstString g_TypeHint("NSNumber:float"); 374 375 std::string prefix, suffix; 376 if (Language *language = Language::FindPlugin(lang)) { 377 if (!language->GetFormatterPrefixSuffix(valobj, g_TypeHint, prefix, 378 suffix)) { 379 prefix.clear(); 380 suffix.clear(); 381 } 382 } 383 384 stream.Printf("%s%f%s", prefix.c_str(), value, suffix.c_str()); 385 } 386 387 static void NSNumber_FormatDouble(ValueObject &valobj, Stream &stream, 388 double value, lldb::LanguageType lang) { 389 static ConstString g_TypeHint("NSNumber:double"); 390 391 std::string prefix, suffix; 392 if (Language *language = Language::FindPlugin(lang)) { 393 if (!language->GetFormatterPrefixSuffix(valobj, g_TypeHint, prefix, 394 suffix)) { 395 prefix.clear(); 396 suffix.clear(); 397 } 398 } 399 400 stream.Printf("%s%g%s", prefix.c_str(), value, suffix.c_str()); 401 } 402 403 bool lldb_private::formatters::NSNumberSummaryProvider( 404 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 405 ProcessSP process_sp = valobj.GetProcessSP(); 406 if (!process_sp) 407 return false; 408 409 ObjCLanguageRuntime *runtime = 410 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 411 lldb::eLanguageTypeObjC); 412 413 if (!runtime) 414 return false; 415 416 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 417 runtime->GetClassDescriptor(valobj)); 418 419 if (!descriptor || !descriptor->IsValid()) 420 return false; 421 422 uint32_t ptr_size = process_sp->GetAddressByteSize(); 423 424 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 425 426 if (!valobj_addr) 427 return false; 428 429 const char *class_name = descriptor->GetClassName().GetCString(); 430 431 if (!class_name || !*class_name) 432 return false; 433 434 if (!strcmp(class_name, "__NSCFBoolean")) 435 return ObjCBooleanSummaryProvider(valobj, stream, options); 436 437 if (!strcmp(class_name, "NSNumber") || !strcmp(class_name, "__NSCFNumber")) { 438 uint64_t value = 0; 439 uint64_t i_bits = 0; 440 if (descriptor->GetTaggedPointerInfo(&i_bits, &value)) { 441 switch (i_bits) { 442 case 0: 443 NSNumber_FormatChar(valobj, stream, (char)value, options.GetLanguage()); 444 break; 445 case 1: 446 case 4: 447 NSNumber_FormatShort(valobj, stream, (short)value, 448 options.GetLanguage()); 449 break; 450 case 2: 451 case 8: 452 NSNumber_FormatInt(valobj, stream, (int)value, options.GetLanguage()); 453 break; 454 case 3: 455 case 12: 456 NSNumber_FormatLong(valobj, stream, value, options.GetLanguage()); 457 break; 458 default: 459 return false; 460 } 461 return true; 462 } else { 463 Error error; 464 uint8_t data_type = (process_sp->ReadUnsignedIntegerFromMemory( 465 valobj_addr + ptr_size, 1, 0, error) & 466 0x1F); 467 uint64_t data_location = valobj_addr + 2 * ptr_size; 468 uint64_t value = 0; 469 if (error.Fail()) 470 return false; 471 switch (data_type) { 472 case 1: // 0B00001 473 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 1, 0, 474 error); 475 if (error.Fail()) 476 return false; 477 NSNumber_FormatChar(valobj, stream, (char)value, options.GetLanguage()); 478 break; 479 case 2: // 0B0010 480 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 2, 0, 481 error); 482 if (error.Fail()) 483 return false; 484 NSNumber_FormatShort(valobj, stream, (short)value, 485 options.GetLanguage()); 486 break; 487 case 3: // 0B0011 488 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 4, 0, 489 error); 490 if (error.Fail()) 491 return false; 492 NSNumber_FormatInt(valobj, stream, (int)value, options.GetLanguage()); 493 break; 494 case 17: // 0B10001 495 data_location += 8; 496 LLVM_FALLTHROUGH; 497 case 4: // 0B0100 498 value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 8, 0, 499 error); 500 if (error.Fail()) 501 return false; 502 NSNumber_FormatLong(valobj, stream, value, options.GetLanguage()); 503 break; 504 case 5: // 0B0101 505 { 506 uint32_t flt_as_int = process_sp->ReadUnsignedIntegerFromMemory( 507 data_location, 4, 0, error); 508 if (error.Fail()) 509 return false; 510 float flt_value = 0.0f; 511 memcpy(&flt_value, &flt_as_int, sizeof(flt_as_int)); 512 NSNumber_FormatFloat(valobj, stream, flt_value, options.GetLanguage()); 513 break; 514 } 515 case 6: // 0B0110 516 { 517 uint64_t dbl_as_lng = process_sp->ReadUnsignedIntegerFromMemory( 518 data_location, 8, 0, error); 519 if (error.Fail()) 520 return false; 521 double dbl_value = 0.0; 522 memcpy(&dbl_value, &dbl_as_lng, sizeof(dbl_as_lng)); 523 NSNumber_FormatDouble(valobj, stream, dbl_value, options.GetLanguage()); 524 break; 525 } 526 default: 527 return false; 528 } 529 return true; 530 } 531 } 532 533 return false; 534 } 535 536 bool lldb_private::formatters::NSURLSummaryProvider( 537 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 538 ProcessSP process_sp = valobj.GetProcessSP(); 539 if (!process_sp) 540 return false; 541 542 ObjCLanguageRuntime *runtime = 543 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 544 lldb::eLanguageTypeObjC); 545 546 if (!runtime) 547 return false; 548 549 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 550 runtime->GetClassDescriptor(valobj)); 551 552 if (!descriptor || !descriptor->IsValid()) 553 return false; 554 555 uint32_t ptr_size = process_sp->GetAddressByteSize(); 556 557 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 558 559 if (!valobj_addr) 560 return false; 561 562 const char *class_name = descriptor->GetClassName().GetCString(); 563 564 if (!class_name || !*class_name) 565 return false; 566 567 if (strcmp(class_name, "NSURL") == 0) { 568 uint64_t offset_text = ptr_size + ptr_size + 569 8; // ISA + pointer + 8 bytes of data (even on 32bit) 570 uint64_t offset_base = offset_text + ptr_size; 571 CompilerType type(valobj.GetCompilerType()); 572 ValueObjectSP text( 573 valobj.GetSyntheticChildAtOffset(offset_text, type, true)); 574 ValueObjectSP base( 575 valobj.GetSyntheticChildAtOffset(offset_base, type, true)); 576 if (!text) 577 return false; 578 if (text->GetValueAsUnsigned(0) == 0) 579 return false; 580 StreamString summary; 581 if (!NSStringSummaryProvider(*text, summary, options)) 582 return false; 583 if (base && base->GetValueAsUnsigned(0)) { 584 if (summary.GetSize() > 0) 585 summary.GetString().resize(summary.GetSize() - 1); 586 summary.Printf(" -- "); 587 StreamString base_summary; 588 if (NSURLSummaryProvider(*base, base_summary, options) && 589 base_summary.GetSize() > 0) 590 summary.Printf("%s", base_summary.GetSize() > 2 591 ? base_summary.GetData() + 2 592 : base_summary.GetData()); 593 } 594 if (summary.GetSize()) { 595 stream.Printf("%s", summary.GetData()); 596 return true; 597 } 598 } 599 600 return false; 601 } 602 603 bool lldb_private::formatters::NSDateSummaryProvider( 604 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 605 ProcessSP process_sp = valobj.GetProcessSP(); 606 if (!process_sp) 607 return false; 608 609 ObjCLanguageRuntime *runtime = 610 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 611 lldb::eLanguageTypeObjC); 612 613 if (!runtime) 614 return false; 615 616 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 617 runtime->GetClassDescriptor(valobj)); 618 619 if (!descriptor || !descriptor->IsValid()) 620 return false; 621 622 uint32_t ptr_size = process_sp->GetAddressByteSize(); 623 624 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 625 626 if (!valobj_addr) 627 return false; 628 629 uint64_t date_value_bits = 0; 630 double date_value = 0.0; 631 632 ConstString class_name = descriptor->GetClassName(); 633 634 static const ConstString g_NSDate("NSDate"); 635 static const ConstString g___NSDate("__NSDate"); 636 static const ConstString g___NSTaggedDate("__NSTaggedDate"); 637 static const ConstString g_NSCalendarDate("NSCalendarDate"); 638 639 if (class_name.IsEmpty()) 640 return false; 641 642 if ((class_name == g_NSDate) || (class_name == g___NSDate) || 643 (class_name == g___NSTaggedDate)) { 644 uint64_t info_bits = 0, value_bits = 0; 645 if (descriptor->GetTaggedPointerInfo(&info_bits, &value_bits)) { 646 date_value_bits = ((value_bits << 8) | (info_bits << 4)); 647 memcpy(&date_value, &date_value_bits, sizeof(date_value_bits)); 648 } else { 649 llvm::Triple triple( 650 process_sp->GetTarget().GetArchitecture().GetTriple()); 651 uint32_t delta = 652 (triple.isWatchOS() && triple.isWatchABI()) ? 8 : ptr_size; 653 Error error; 654 date_value_bits = process_sp->ReadUnsignedIntegerFromMemory( 655 valobj_addr + delta, 8, 0, error); 656 memcpy(&date_value, &date_value_bits, sizeof(date_value_bits)); 657 if (error.Fail()) 658 return false; 659 } 660 } else if (class_name == g_NSCalendarDate) { 661 Error error; 662 date_value_bits = process_sp->ReadUnsignedIntegerFromMemory( 663 valobj_addr + 2 * ptr_size, 8, 0, error); 664 memcpy(&date_value, &date_value_bits, sizeof(date_value_bits)); 665 if (error.Fail()) 666 return false; 667 } else 668 return false; 669 670 if (date_value == -63114076800) { 671 stream.Printf("0001-12-30 00:00:00 +0000"); 672 return true; 673 } 674 // this snippet of code assumes that time_t == seconds since Jan-1-1970 675 // this is generally true and POSIXly happy, but might break if a library 676 // vendor decides to get creative 677 time_t epoch = GetOSXEpoch(); 678 epoch = epoch + (time_t)date_value; 679 tm *tm_date = gmtime(&epoch); 680 if (!tm_date) 681 return false; 682 std::string buffer(1024, 0); 683 if (strftime(&buffer[0], 1023, "%Z", tm_date) == 0) 684 return false; 685 stream.Printf("%04d-%02d-%02d %02d:%02d:%02d %s", tm_date->tm_year + 1900, 686 tm_date->tm_mon + 1, tm_date->tm_mday, tm_date->tm_hour, 687 tm_date->tm_min, tm_date->tm_sec, buffer.c_str()); 688 return true; 689 } 690 691 bool lldb_private::formatters::ObjCClassSummaryProvider( 692 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 693 ProcessSP process_sp = valobj.GetProcessSP(); 694 if (!process_sp) 695 return false; 696 697 ObjCLanguageRuntime *runtime = 698 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 699 lldb::eLanguageTypeObjC); 700 701 if (!runtime) 702 return false; 703 704 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 705 runtime->GetClassDescriptorFromISA(valobj.GetValueAsUnsigned(0))); 706 707 if (!descriptor || !descriptor->IsValid()) 708 return false; 709 710 ConstString class_name = descriptor->GetClassName(); 711 712 if (class_name.IsEmpty()) 713 return false; 714 715 if (ConstString cs = 716 Mangled(class_name).GetDemangledName(lldb::eLanguageTypeUnknown)) 717 class_name = cs; 718 719 stream.Printf("%s", class_name.AsCString("<unknown class>")); 720 return true; 721 } 722 723 class ObjCClassSyntheticChildrenFrontEnd : public SyntheticChildrenFrontEnd { 724 public: 725 ObjCClassSyntheticChildrenFrontEnd(lldb::ValueObjectSP valobj_sp) 726 : SyntheticChildrenFrontEnd(*valobj_sp) {} 727 728 ~ObjCClassSyntheticChildrenFrontEnd() override = default; 729 730 size_t CalculateNumChildren() override { return 0; } 731 732 lldb::ValueObjectSP GetChildAtIndex(size_t idx) override { 733 return lldb::ValueObjectSP(); 734 } 735 736 bool Update() override { return false; } 737 738 bool MightHaveChildren() override { return false; } 739 740 size_t GetIndexOfChildWithName(const ConstString &name) override { 741 return UINT32_MAX; 742 } 743 }; 744 745 SyntheticChildrenFrontEnd * 746 lldb_private::formatters::ObjCClassSyntheticFrontEndCreator( 747 CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) { 748 return new ObjCClassSyntheticChildrenFrontEnd(valobj_sp); 749 } 750 751 template <bool needs_at> 752 bool lldb_private::formatters::NSDataSummaryProvider( 753 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 754 ProcessSP process_sp = valobj.GetProcessSP(); 755 if (!process_sp) 756 return false; 757 758 ObjCLanguageRuntime *runtime = 759 (ObjCLanguageRuntime *)process_sp->GetLanguageRuntime( 760 lldb::eLanguageTypeObjC); 761 762 if (!runtime) 763 return false; 764 765 ObjCLanguageRuntime::ClassDescriptorSP descriptor( 766 runtime->GetClassDescriptor(valobj)); 767 768 if (!descriptor || !descriptor->IsValid()) 769 return false; 770 771 bool is_64bit = (process_sp->GetAddressByteSize() == 8); 772 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0); 773 774 if (!valobj_addr) 775 return false; 776 777 uint64_t value = 0; 778 779 const char *class_name = descriptor->GetClassName().GetCString(); 780 781 if (!class_name || !*class_name) 782 return false; 783 784 if (!strcmp(class_name, "NSConcreteData") || 785 !strcmp(class_name, "NSConcreteMutableData") || 786 !strcmp(class_name, "__NSCFData")) { 787 uint32_t offset = (is_64bit ? 16 : 8); 788 Error error; 789 value = process_sp->ReadUnsignedIntegerFromMemory( 790 valobj_addr + offset, is_64bit ? 8 : 4, 0, error); 791 if (error.Fail()) 792 return false; 793 } else if (!strcmp(class_name, "_NSInlineData")) { 794 uint32_t offset = (is_64bit ? 8 : 4); 795 Error error; 796 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + offset, 2, 797 0, error); 798 if (error.Fail()) 799 return false; 800 } else if (!strcmp(class_name, "_NSZeroData")) { 801 value = 0; 802 } else 803 return false; 804 805 stream.Printf("%s%" PRIu64 " byte%s%s", (needs_at ? "@\"" : ""), value, 806 (value != 1 ? "s" : ""), (needs_at ? "\"" : "")); 807 808 return true; 809 } 810 811 bool lldb_private::formatters::ObjCBOOLSummaryProvider( 812 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 813 const uint32_t type_info = valobj.GetCompilerType().GetTypeInfo(); 814 815 ValueObjectSP real_guy_sp = valobj.GetSP(); 816 817 if (type_info & eTypeIsPointer) { 818 Error err; 819 real_guy_sp = valobj.Dereference(err); 820 if (err.Fail() || !real_guy_sp) 821 return false; 822 } else if (type_info & eTypeIsReference) { 823 real_guy_sp = valobj.GetChildAtIndex(0, true); 824 if (!real_guy_sp) 825 return false; 826 } 827 uint8_t value = (real_guy_sp->GetValueAsUnsigned(0) & 0xFF); 828 switch (value) { 829 case 0: 830 stream.Printf("NO"); 831 break; 832 case 1: 833 stream.Printf("YES"); 834 break; 835 default: 836 stream.Printf("%u", value); 837 break; 838 } 839 return true; 840 } 841 842 bool lldb_private::formatters::ObjCBooleanSummaryProvider( 843 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 844 lldb::addr_t valobj_ptr_value = 845 valobj.GetValueAsUnsigned(LLDB_INVALID_ADDRESS); 846 if (valobj_ptr_value == LLDB_INVALID_ADDRESS) 847 return false; 848 849 ProcessSP process_sp(valobj.GetProcessSP()); 850 if (!process_sp) 851 return false; 852 853 if (AppleObjCRuntime *objc_runtime = 854 (AppleObjCRuntime *)process_sp->GetObjCLanguageRuntime()) { 855 lldb::addr_t cf_true = LLDB_INVALID_ADDRESS, 856 cf_false = LLDB_INVALID_ADDRESS; 857 objc_runtime->GetValuesForGlobalCFBooleans(cf_true, cf_false); 858 if (valobj_ptr_value == cf_true) { 859 stream.PutCString("YES"); 860 return true; 861 } 862 if (valobj_ptr_value == cf_false) { 863 stream.PutCString("NO"); 864 return true; 865 } 866 } 867 868 return false; 869 } 870 871 template <bool is_sel_ptr> 872 bool lldb_private::formatters::ObjCSELSummaryProvider( 873 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { 874 lldb::ValueObjectSP valobj_sp; 875 876 CompilerType charstar(valobj.GetCompilerType() 877 .GetBasicTypeFromAST(eBasicTypeChar) 878 .GetPointerType()); 879 880 if (!charstar) 881 return false; 882 883 ExecutionContext exe_ctx(valobj.GetExecutionContextRef()); 884 885 if (is_sel_ptr) { 886 lldb::addr_t data_address = valobj.GetValueAsUnsigned(LLDB_INVALID_ADDRESS); 887 if (data_address == LLDB_INVALID_ADDRESS) 888 return false; 889 valobj_sp = ValueObject::CreateValueObjectFromAddress("text", data_address, 890 exe_ctx, charstar); 891 } else { 892 DataExtractor data; 893 Error error; 894 valobj.GetData(data, error); 895 if (error.Fail()) 896 return false; 897 valobj_sp = 898 ValueObject::CreateValueObjectFromData("text", data, exe_ctx, charstar); 899 } 900 901 if (!valobj_sp) 902 return false; 903 904 stream.Printf("%s", valobj_sp->GetSummaryAsCString()); 905 return true; 906 } 907 908 // POSIX has an epoch on Jan-1-1970, but Cocoa prefers Jan-1-2001 909 // this call gives the POSIX equivalent of the Cocoa epoch 910 time_t lldb_private::formatters::GetOSXEpoch() { 911 static time_t epoch = 0; 912 if (!epoch) { 913 #ifndef _WIN32 914 tzset(); 915 tm tm_epoch; 916 tm_epoch.tm_sec = 0; 917 tm_epoch.tm_hour = 0; 918 tm_epoch.tm_min = 0; 919 tm_epoch.tm_mon = 0; 920 tm_epoch.tm_mday = 1; 921 tm_epoch.tm_year = 2001 - 1900; 922 tm_epoch.tm_isdst = -1; 923 tm_epoch.tm_gmtoff = 0; 924 tm_epoch.tm_zone = nullptr; 925 epoch = timegm(&tm_epoch); 926 #endif 927 } 928 return epoch; 929 } 930 931 template bool lldb_private::formatters::NSDataSummaryProvider<true>( 932 ValueObject &, Stream &, const TypeSummaryOptions &); 933 934 template bool lldb_private::formatters::NSDataSummaryProvider<false>( 935 ValueObject &, Stream &, const TypeSummaryOptions &); 936 937 template bool lldb_private::formatters::ObjCSELSummaryProvider<true>( 938 ValueObject &, Stream &, const TypeSummaryOptions &); 939 940 template bool lldb_private::formatters::ObjCSELSummaryProvider<false>( 941 ValueObject &, Stream &, const TypeSummaryOptions &); 942