Path: blob/main/contrib/llvm-project/lldb/source/Plugins/Language/ObjC/Cocoa.cpp
39644 views
//===-- Cocoa.cpp ---------------------------------------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#include "Cocoa.h"9#include "NSString.h"10#include "ObjCConstants.h"1112#include "Plugins/LanguageRuntime/ObjC/AppleObjCRuntime/AppleObjCRuntime.h"13#include "Plugins/TypeSystem/Clang/TypeSystemClang.h"14#include "lldb/Core/Mangled.h"15#include "lldb/Core/ValueObject.h"16#include "lldb/Core/ValueObjectConstResult.h"17#include "lldb/DataFormatters/FormattersHelpers.h"18#include "lldb/DataFormatters/StringPrinter.h"19#include "lldb/DataFormatters/TypeSummary.h"20#include "lldb/Host/Time.h"21#include "lldb/Target/Language.h"22#include "lldb/Target/Process.h"23#include "lldb/Target/Target.h"24#include "lldb/Utility/DataBufferHeap.h"25#include "lldb/Utility/Endian.h"26#include "lldb/Utility/LLDBLog.h"27#include "lldb/Utility/Status.h"28#include "lldb/Utility/Stream.h"2930#include "llvm/ADT/APInt.h"31#include "llvm/ADT/bit.h"323334using namespace lldb;35using namespace lldb_private;36using namespace lldb_private::formatters;3738bool lldb_private::formatters::NSBundleSummaryProvider(39ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {40ProcessSP process_sp = valobj.GetProcessSP();41if (!process_sp)42return false;4344ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);4546if (!runtime)47return false;4849ObjCLanguageRuntime::ClassDescriptorSP descriptor(50runtime->GetClassDescriptor(valobj));5152if (!descriptor || !descriptor->IsValid())53return false;5455uint32_t ptr_size = process_sp->GetAddressByteSize();5657lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);5859if (!valobj_addr)60return false;6162llvm::StringRef class_name(descriptor->GetClassName().GetCString());6364if (class_name.empty())65return false;6667if (class_name == "NSBundle") {68uint64_t offset = 5 * ptr_size;69ValueObjectSP text(valobj.GetSyntheticChildAtOffset(70offset,71valobj.GetCompilerType().GetBasicTypeFromAST(lldb::eBasicTypeObjCID),72true));7374if (!text)75return false;7677StreamString summary_stream;78bool was_nsstring_ok =79NSStringSummaryProvider(*text, summary_stream, options);80if (was_nsstring_ok && summary_stream.GetSize() > 0) {81stream.Printf("%s", summary_stream.GetData());82return true;83}84}8586return false;87}8889bool lldb_private::formatters::NSTimeZoneSummaryProvider(90ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {91ProcessSP process_sp = valobj.GetProcessSP();92if (!process_sp)93return false;9495ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);9697if (!runtime)98return false;99100ObjCLanguageRuntime::ClassDescriptorSP descriptor(101runtime->GetClassDescriptor(valobj));102103if (!descriptor || !descriptor->IsValid())104return false;105106uint32_t ptr_size = process_sp->GetAddressByteSize();107108lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);109110if (!valobj_addr)111return false;112113llvm::StringRef class_name(descriptor->GetClassName().GetCString());114115if (class_name.empty())116return false;117118if (class_name == "__NSTimeZone") {119uint64_t offset = ptr_size;120ValueObjectSP text(valobj.GetSyntheticChildAtOffset(121offset, valobj.GetCompilerType(), true));122123if (!text)124return false;125126StreamString summary_stream;127bool was_nsstring_ok =128NSStringSummaryProvider(*text, summary_stream, options);129if (was_nsstring_ok && summary_stream.GetSize() > 0) {130stream.Printf("%s", summary_stream.GetData());131return true;132}133}134135return false;136}137138bool lldb_private::formatters::NSNotificationSummaryProvider(139ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {140ProcessSP process_sp = valobj.GetProcessSP();141if (!process_sp)142return false;143144ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);145146if (!runtime)147return false;148149ObjCLanguageRuntime::ClassDescriptorSP descriptor(150runtime->GetClassDescriptor(valobj));151152if (!descriptor || !descriptor->IsValid())153return false;154155uint32_t ptr_size = process_sp->GetAddressByteSize();156157lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);158159if (!valobj_addr)160return false;161162llvm::StringRef class_name(descriptor->GetClassName().GetCString());163164if (class_name.empty())165return false;166167if (class_name == "NSConcreteNotification") {168uint64_t offset = ptr_size;169ValueObjectSP text(valobj.GetSyntheticChildAtOffset(170offset, valobj.GetCompilerType(), true));171172if (!text)173return false;174175StreamString summary_stream;176bool was_nsstring_ok =177NSStringSummaryProvider(*text, summary_stream, options);178if (was_nsstring_ok && summary_stream.GetSize() > 0) {179stream.Printf("%s", summary_stream.GetData());180return true;181}182}183184return false;185}186187bool lldb_private::formatters::NSMachPortSummaryProvider(188ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {189ProcessSP process_sp = valobj.GetProcessSP();190if (!process_sp)191return false;192193ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);194195if (!runtime)196return false;197198ObjCLanguageRuntime::ClassDescriptorSP descriptor(199runtime->GetClassDescriptor(valobj));200201if (!descriptor || !descriptor->IsValid())202return false;203204uint32_t ptr_size = process_sp->GetAddressByteSize();205206lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);207208if (!valobj_addr)209return false;210211llvm::StringRef class_name(descriptor->GetClassName().GetCString());212213if (class_name.empty())214return false;215216uint64_t port_number = 0;217218if (class_name == "NSMachPort") {219uint64_t offset = (ptr_size == 4 ? 12 : 20);220Status error;221port_number = process_sp->ReadUnsignedIntegerFromMemory(222offset + valobj_addr, 4, 0, error);223if (error.Success()) {224stream.Printf("mach port: %u",225(uint32_t)(port_number & 0x00000000FFFFFFFF));226return true;227}228}229230return false;231}232233bool lldb_private::formatters::NSIndexSetSummaryProvider(234ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {235ProcessSP process_sp = valobj.GetProcessSP();236if (!process_sp)237return false;238239AppleObjCRuntime *runtime = llvm::dyn_cast_or_null<AppleObjCRuntime>(240ObjCLanguageRuntime::Get(*process_sp));241242if (!runtime)243return false;244245ObjCLanguageRuntime::ClassDescriptorSP descriptor(246runtime->GetClassDescriptor(valobj));247248if (!descriptor || !descriptor->IsValid())249return false;250251uint32_t ptr_size = process_sp->GetAddressByteSize();252253lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);254255if (!valobj_addr)256return false;257258llvm::StringRef class_name(descriptor->GetClassName().GetCString());259260if (class_name.empty())261return false;262263uint64_t count = 0;264265do {266if (class_name == "NSIndexSet" || class_name == "NSMutableIndexSet") {267// Foundation version 2000 added a bitmask if the index set fit in 64 bits268// and a Tagged Pointer version if the bitmask is small enough to fit in269// the tagged pointer payload.270// It also changed the layout (but not the size) of the set descriptor.271272// First check whether this is a tagged pointer. The bitmask will be in273// the payload of the tagged pointer.274uint64_t payload;275if (runtime->GetFoundationVersion() >= 2000276&& descriptor->GetTaggedPointerInfo(nullptr, nullptr, &payload)) {277count = llvm::popcount(payload);278break;279}280// The first 32 bits describe the index set in all cases:281Status error;282uint32_t mode = process_sp->ReadUnsignedIntegerFromMemory(283valobj_addr + ptr_size, 4, 0, error);284if (error.Fail())285return false;286// Now check if the index is held in a bitmask in the object:287if (runtime->GetFoundationVersion() >= 2000) {288// The first two bits are "isSingleRange" and "isBitfield". If this is289// a bitfield we handle it here, otherwise set mode appropriately and290// the rest of the treatment is in common.291if ((mode & 2) == 2) {292// The bitfield is a 64 bit uint at the beginning of the data var.293uint64_t bitfield = process_sp->ReadUnsignedIntegerFromMemory(294valobj_addr + 2 * ptr_size, 8, 0, error);295if (error.Fail())296return false;297count = llvm::popcount(bitfield);298break;299}300// It wasn't a bitfield, so read the isSingleRange from its new loc:301if ((mode & 1) == 1)302mode = 1; // this means the set only has one range303else304mode = 2; // this means the set has multiple ranges305} else {306// this means the set is empty - count = 0307if ((mode & 1) == 1) {308count = 0;309break;310}311312if ((mode & 2) == 2)313mode = 1; // this means the set only has one range314else315mode = 2; // this means the set has multiple ranges316}317if (mode == 1) {318count = process_sp->ReadUnsignedIntegerFromMemory(319valobj_addr + 3 * ptr_size, ptr_size, 0, error);320if (error.Fail())321return false;322} else {323// read a pointer to the data at 2*ptr_size324count = process_sp->ReadUnsignedIntegerFromMemory(325valobj_addr + 2 * ptr_size, ptr_size, 0, error);326if (error.Fail())327return false;328// read the data at 2*ptr_size from the first location329count = process_sp->ReadUnsignedIntegerFromMemory(count + 2 * ptr_size,330ptr_size, 0, error);331if (error.Fail())332return false;333}334} else335return false;336} while (false);337stream.Printf("%" PRIu64 " index%s", count, (count == 1 ? "" : "es"));338return true;339}340341static void NSNumber_FormatChar(ValueObject &valobj, Stream &stream, char value,342lldb::LanguageType lang) {343static constexpr llvm::StringLiteral g_TypeHint("NSNumber:char");344345llvm::StringRef prefix, suffix;346if (Language *language = Language::FindPlugin(lang))347std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);348349stream << prefix;350stream.Printf("%hhd", value);351stream << suffix;352}353354static void NSNumber_FormatShort(ValueObject &valobj, Stream &stream,355short value, lldb::LanguageType lang) {356static constexpr llvm::StringLiteral g_TypeHint("NSNumber:short");357358llvm::StringRef prefix, suffix;359if (Language *language = Language::FindPlugin(lang))360std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);361362stream << prefix;363stream.Printf("%hd", value);364stream << suffix;365}366367static void NSNumber_FormatInt(ValueObject &valobj, Stream &stream, int value,368lldb::LanguageType lang) {369static constexpr llvm::StringLiteral g_TypeHint("NSNumber:int");370371llvm::StringRef prefix, suffix;372if (Language *language = Language::FindPlugin(lang))373std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);374375stream << prefix;376stream.Printf("%d", value);377stream << suffix;378}379380static void NSNumber_FormatLong(ValueObject &valobj, Stream &stream,381int64_t value, lldb::LanguageType lang) {382static constexpr llvm::StringLiteral g_TypeHint("NSNumber:long");383384llvm::StringRef prefix, suffix;385if (Language *language = Language::FindPlugin(lang))386std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);387388stream << prefix;389stream.Printf("%" PRId64 "", value);390stream << suffix;391}392393static void NSNumber_FormatInt128(ValueObject &valobj, Stream &stream,394const llvm::APInt &value,395lldb::LanguageType lang) {396static constexpr llvm::StringLiteral g_TypeHint("NSNumber:int128_t");397398llvm::StringRef prefix, suffix;399if (Language *language = Language::FindPlugin(lang))400std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);401402stream << prefix;403const int radix = 10;404const bool isSigned = true;405std::string str = llvm::toString(value, radix, isSigned);406stream.PutCString(str.c_str());407stream << suffix;408}409410static void NSNumber_FormatFloat(ValueObject &valobj, Stream &stream,411float value, lldb::LanguageType lang) {412static constexpr llvm::StringLiteral g_TypeHint("NSNumber:float");413414llvm::StringRef prefix, suffix;415if (Language *language = Language::FindPlugin(lang))416std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);417418stream << prefix;419stream.Printf("%f", value);420stream << suffix;421}422423static void NSNumber_FormatDouble(ValueObject &valobj, Stream &stream,424double value, lldb::LanguageType lang) {425static constexpr llvm::StringLiteral g_TypeHint("NSNumber:double");426427llvm::StringRef prefix, suffix;428if (Language *language = Language::FindPlugin(lang))429std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);430431stream << prefix;432stream.Printf("%g", value);433stream << suffix;434}435436bool lldb_private::formatters::NSNumberSummaryProvider(437ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {438ProcessSP process_sp = valobj.GetProcessSP();439if (!process_sp)440return false;441442Log *log = GetLog(LLDBLog::DataFormatters);443ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);444445if (!runtime)446return false;447448ObjCLanguageRuntime::ClassDescriptorSP descriptor(449runtime->GetClassDescriptor(valobj));450451if (!descriptor || !descriptor->IsValid())452return false;453454uint32_t ptr_size = process_sp->GetAddressByteSize();455456lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);457458if (!valobj_addr)459return false;460461llvm::StringRef class_name(descriptor->GetClassName().GetCString());462463if (class_name.empty())464return false;465466if (class_name == "__NSCFBoolean")467return ObjCBooleanSummaryProvider(valobj, stream, options);468469if (class_name == "NSDecimalNumber")470return NSDecimalNumberSummaryProvider(valobj, stream, options);471472if (class_name == "NSConstantIntegerNumber") {473Status error;474int64_t value = process_sp->ReadSignedIntegerFromMemory(475valobj_addr + 2 * ptr_size, 8, 0, error);476if (error.Fail())477return false;478uint64_t encoding_addr = process_sp->ReadUnsignedIntegerFromMemory(479valobj_addr + ptr_size, ptr_size, 0, error);480if (error.Fail())481return false;482char encoding =483process_sp->ReadUnsignedIntegerFromMemory(encoding_addr, 1, 0, error);484if (error.Fail())485return false;486487switch (encoding) {488case _C_CHR:489NSNumber_FormatChar(valobj, stream, (char)value, options.GetLanguage());490return true;491case _C_SHT:492NSNumber_FormatShort(valobj, stream, (short)value, options.GetLanguage());493return true;494case _C_INT:495NSNumber_FormatInt(valobj, stream, (int)value, options.GetLanguage());496return true;497case _C_LNG:498case _C_LNG_LNG:499NSNumber_FormatLong(valobj, stream, value, options.GetLanguage());500return true;501502case _C_UCHR:503case _C_USHT:504case _C_UINT:505case _C_ULNG:506case _C_ULNG_LNG:507stream.Printf("%" PRIu64, value);508return true;509}510511return false;512}513514if (class_name == "NSConstantFloatNumber") {515Status error;516uint32_t flt_as_int = process_sp->ReadUnsignedIntegerFromMemory(517valobj_addr + ptr_size, 4, 0, error);518if (error.Fail())519return false;520float flt_value = 0.0f;521memcpy(&flt_value, &flt_as_int, sizeof(flt_as_int));522NSNumber_FormatFloat(valobj, stream, flt_value, options.GetLanguage());523return true;524}525526if (class_name == "NSConstantDoubleNumber") {527Status error;528uint64_t dbl_as_lng = process_sp->ReadUnsignedIntegerFromMemory(529valobj_addr + ptr_size, 8, 0, error);530if (error.Fail())531return false;532double dbl_value = 0.0;533memcpy(&dbl_value, &dbl_as_lng, sizeof(dbl_as_lng));534NSNumber_FormatDouble(valobj, stream, dbl_value, options.GetLanguage());535return true;536}537538if (class_name == "NSNumber" || class_name == "__NSCFNumber") {539int64_t value = 0;540uint64_t i_bits = 0;541if (descriptor->GetTaggedPointerInfoSigned(&i_bits, &value)) {542// Check for "preserved" numbers. We still don't support them yet.543if (i_bits & 0x8) {544if (log)545log->Printf(546"Unsupported (preserved) NSNumber tagged pointer 0x%" PRIu64,547valobj_addr);548return false;549}550551switch (i_bits) {552case 0:553NSNumber_FormatChar(valobj, stream, (char)value, options.GetLanguage());554break;555case 1:556case 4:557NSNumber_FormatShort(valobj, stream, (short)value,558options.GetLanguage());559break;560case 2:561case 8:562NSNumber_FormatInt(valobj, stream, (int)value, options.GetLanguage());563break;564case 3:565case 12:566NSNumber_FormatLong(valobj, stream, value, options.GetLanguage());567break;568default:569return false;570}571return true;572} else {573Status error;574575AppleObjCRuntime *runtime = llvm::dyn_cast_or_null<AppleObjCRuntime>(576ObjCLanguageRuntime::Get(*process_sp));577578const bool new_format =579(runtime && runtime->GetFoundationVersion() >= 1400);580581enum class TypeCodes : int {582sint8 = 0x0,583sint16 = 0x1,584sint32 = 0x2,585sint64 = 0x3,586f32 = 0x4,587f64 = 0x5,588sint128 = 0x6589};590591uint64_t data_location = valobj_addr + 2 * ptr_size;592TypeCodes type_code;593594if (new_format) {595uint64_t cfinfoa = process_sp->ReadUnsignedIntegerFromMemory(596valobj_addr + ptr_size, ptr_size, 0, error);597598if (error.Fail())599return false;600601bool is_preserved_number = cfinfoa & 0x8;602if (is_preserved_number) {603if (log)604log->Printf(605"Unsupported preserved NSNumber tagged pointer 0x%" PRIu64,606valobj_addr);607return false;608}609610type_code = static_cast<TypeCodes>(cfinfoa & 0x7);611} else {612uint8_t data_type = process_sp->ReadUnsignedIntegerFromMemory(613valobj_addr + ptr_size, 1, 0, error) &6140x1F;615616if (error.Fail())617return false;618619switch (data_type) {620case 1:621type_code = TypeCodes::sint8;622break;623case 2:624type_code = TypeCodes::sint16;625break;626case 3:627type_code = TypeCodes::sint32;628break;629case 17:630data_location += 8;631[[fallthrough]];632case 4:633type_code = TypeCodes::sint64;634break;635case 5:636type_code = TypeCodes::f32;637break;638case 6:639type_code = TypeCodes::f64;640break;641default:642return false;643}644}645646uint64_t value = 0;647bool success = false;648switch (type_code) {649case TypeCodes::sint8:650value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 1, 0,651error);652if (error.Fail())653return false;654NSNumber_FormatChar(valobj, stream, (char)value, options.GetLanguage());655success = true;656break;657case TypeCodes::sint16:658value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 2, 0,659error);660if (error.Fail())661return false;662NSNumber_FormatShort(valobj, stream, (short)value,663options.GetLanguage());664success = true;665break;666case TypeCodes::sint32:667value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 4, 0,668error);669if (error.Fail())670return false;671NSNumber_FormatInt(valobj, stream, (int)value, options.GetLanguage());672success = true;673break;674case TypeCodes::sint64:675value = process_sp->ReadUnsignedIntegerFromMemory(data_location, 8, 0,676error);677if (error.Fail())678return false;679NSNumber_FormatLong(valobj, stream, value, options.GetLanguage());680success = true;681break;682case TypeCodes::f32: {683uint32_t flt_as_int = process_sp->ReadUnsignedIntegerFromMemory(684data_location, 4, 0, error);685if (error.Fail())686return false;687float flt_value = 0.0f;688memcpy(&flt_value, &flt_as_int, sizeof(flt_as_int));689NSNumber_FormatFloat(valobj, stream, flt_value, options.GetLanguage());690success = true;691break;692}693case TypeCodes::f64: {694uint64_t dbl_as_lng = process_sp->ReadUnsignedIntegerFromMemory(695data_location, 8, 0, error);696if (error.Fail())697return false;698double dbl_value = 0.0;699memcpy(&dbl_value, &dbl_as_lng, sizeof(dbl_as_lng));700NSNumber_FormatDouble(valobj, stream, dbl_value, options.GetLanguage());701success = true;702break;703}704case TypeCodes::sint128: // internally, this is the same705{706uint64_t words[2];707words[1] = process_sp->ReadUnsignedIntegerFromMemory(data_location, 8,7080, error);709if (error.Fail())710return false;711words[0] = process_sp->ReadUnsignedIntegerFromMemory(data_location + 8,7128, 0, error);713if (error.Fail())714return false;715llvm::APInt i128_value(128, words);716NSNumber_FormatInt128(valobj, stream, i128_value,717options.GetLanguage());718success = true;719break;720}721}722return success;723}724}725726return false;727}728729bool lldb_private::formatters::NSDecimalNumberSummaryProvider(730ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {731ProcessSP process_sp = valobj.GetProcessSP();732if (!process_sp)733return false;734735lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);736uint32_t ptr_size = process_sp->GetAddressByteSize();737738Status error;739int8_t exponent = process_sp->ReadUnsignedIntegerFromMemory(740valobj_addr + ptr_size, 1, 0, error);741if (error.Fail())742return false;743744uint8_t length_and_negative = process_sp->ReadUnsignedIntegerFromMemory(745valobj_addr + ptr_size + 1, 1, 0, error);746if (error.Fail())747return false;748749// Fifth bit marks negativity.750const bool is_negative = (length_and_negative >> 4) & 1;751752// Zero length and negative means NaN.753uint8_t length = length_and_negative & 0xf;754const bool is_nan = is_negative && (length == 0);755756if (is_nan) {757stream.Printf("NaN");758return true;759}760761if (length == 0) {762stream.Printf("0");763return true;764}765766uint64_t mantissa = process_sp->ReadUnsignedIntegerFromMemory(767valobj_addr + ptr_size + 4, 8, 0, error);768if (error.Fail())769return false;770771if (is_negative)772stream.Printf("-");773774stream.Printf("%" PRIu64 " x 10^%" PRIi8, mantissa, exponent);775return true;776}777778bool lldb_private::formatters::NSURLSummaryProvider(779ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {780ProcessSP process_sp = valobj.GetProcessSP();781if (!process_sp)782return false;783784ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);785786if (!runtime)787return false;788789ObjCLanguageRuntime::ClassDescriptorSP descriptor(790runtime->GetClassDescriptor(valobj));791792if (!descriptor || !descriptor->IsValid())793return false;794795uint32_t ptr_size = process_sp->GetAddressByteSize();796797lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);798799if (!valobj_addr)800return false;801802llvm::StringRef class_name = descriptor->GetClassName().GetStringRef();803804if (class_name != "NSURL")805return false;806807uint64_t offset_text = ptr_size + ptr_size +8088; // ISA + pointer + 8 bytes of data (even on 32bit)809uint64_t offset_base = offset_text + ptr_size;810CompilerType type(valobj.GetCompilerType());811ValueObjectSP text(valobj.GetSyntheticChildAtOffset(offset_text, type, true));812ValueObjectSP base(valobj.GetSyntheticChildAtOffset(offset_base, type, true));813if (!text || text->GetValueAsUnsigned(0) == 0)814return false;815816StreamString base_summary;817if (base && base->GetValueAsUnsigned(0)) {818if (!NSURLSummaryProvider(*base, base_summary, options))819base_summary.Clear();820}821if (base_summary.Empty())822return NSStringSummaryProvider(*text, stream, options);823824StreamString summary;825if (!NSStringSummaryProvider(*text, summary, options) || summary.Empty())826return false;827828static constexpr llvm::StringLiteral quote_char("\"");829static constexpr llvm::StringLiteral g_TypeHint("NSString");830llvm::StringRef prefix, suffix;831if (Language *language = Language::FindPlugin(options.GetLanguage()))832std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);833834// @"A" -> @"A835llvm::StringRef summary_str = summary.GetString();836bool back_consumed =837summary_str.consume_back(suffix) && summary_str.consume_back(quote_char);838assert(back_consumed);839UNUSED_IF_ASSERT_DISABLED(back_consumed);840// @"B" -> B"841llvm::StringRef base_summary_str = base_summary.GetString();842bool front_consumed = base_summary_str.consume_front(prefix) &&843base_summary_str.consume_front(quote_char);844assert(front_consumed);845UNUSED_IF_ASSERT_DISABLED(front_consumed);846// @"A -- B"847if (!summary_str.empty() && !base_summary_str.empty()) {848stream << summary_str << " -- " << base_summary_str;849return true;850}851852return false;853}854855/// Bias value for tagged pointer exponents.856/// Recommended values:857/// 0x3e3: encodes all dates between distantPast and distantFuture858/// except for the range within about 1e-28 second of the reference date.859/// 0x3ef: encodes all dates for a few million years beyond distantPast and860/// distantFuture, except within about 1e-25 second of the reference date.861const int TAGGED_DATE_EXPONENT_BIAS = 0x3ef;862863struct DoubleBits {864uint64_t fraction : 52; // unsigned865uint64_t exponent : 11; // signed866uint64_t sign : 1;867};868869struct TaggedDoubleBits {870uint64_t fraction : 52; // unsigned871uint64_t exponent : 7; // signed872uint64_t sign : 1;873uint64_t unused : 4; // placeholder for pointer tag bits874};875876static uint64_t decodeExponent(uint64_t exp) {877// Tagged exponent field is 7-bit signed. Sign-extend the value to 64 bits878// before performing arithmetic.879return llvm::SignExtend64<7>(exp) + TAGGED_DATE_EXPONENT_BIAS;880}881882static double decodeTaggedTimeInterval(uint64_t encodedTimeInterval) {883if (encodedTimeInterval == 0)884return 0.0;885if (encodedTimeInterval == std::numeric_limits<uint64_t>::max())886return (uint64_t)-0.0;887888TaggedDoubleBits encodedBits =889llvm::bit_cast<TaggedDoubleBits>(encodedTimeInterval);890assert(encodedBits.unused == 0);891892// Sign and fraction are represented exactly.893// Exponent is encoded.894DoubleBits decodedBits;895decodedBits.sign = encodedBits.sign;896decodedBits.fraction = encodedBits.fraction;897decodedBits.exponent = decodeExponent(encodedBits.exponent);898899return llvm::bit_cast<double>(decodedBits);900}901902bool lldb_private::formatters::NSDateSummaryProvider(903ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {904ProcessSP process_sp = valobj.GetProcessSP();905if (!process_sp)906return false;907908ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);909910if (!runtime)911return false;912913ObjCLanguageRuntime::ClassDescriptorSP descriptor(914runtime->GetClassDescriptor(valobj));915916if (!descriptor || !descriptor->IsValid())917return false;918919uint32_t ptr_size = process_sp->GetAddressByteSize();920921lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);922923if (!valobj_addr)924return false;925926uint64_t date_value_bits = 0;927double date_value = 0.0;928929ConstString class_name = descriptor->GetClassName();930931static const ConstString g_NSDate("NSDate");932static const ConstString g_dunder_NSDate("__NSDate");933static const ConstString g_NSTaggedDate("__NSTaggedDate");934static const ConstString g_NSCalendarDate("NSCalendarDate");935static const ConstString g_NSConstantDate("NSConstantDate");936937if (class_name.IsEmpty())938return false;939940uint64_t info_bits = 0, value_bits = 0;941if ((class_name == g_NSDate) || (class_name == g_dunder_NSDate) ||942(class_name == g_NSTaggedDate) || (class_name == g_NSConstantDate)) {943if (descriptor->GetTaggedPointerInfo(&info_bits, &value_bits)) {944date_value_bits = ((value_bits << 8) | (info_bits << 4));945memcpy(&date_value, &date_value_bits, sizeof(date_value_bits));946} else {947llvm::Triple triple(948process_sp->GetTarget().GetArchitecture().GetTriple());949uint32_t delta =950(triple.isWatchOS() && triple.isWatchABI()) ? 8 : ptr_size;951Status error;952date_value_bits = process_sp->ReadUnsignedIntegerFromMemory(953valobj_addr + delta, 8, 0, error);954memcpy(&date_value, &date_value_bits, sizeof(date_value_bits));955if (error.Fail())956return false;957}958} else if (class_name == g_NSCalendarDate) {959Status error;960date_value_bits = process_sp->ReadUnsignedIntegerFromMemory(961valobj_addr + 2 * ptr_size, 8, 0, error);962memcpy(&date_value, &date_value_bits, sizeof(date_value_bits));963if (error.Fail())964return false;965} else966return false;967968// FIXME: It seems old dates are not formatted according to NSDate's calendar969// so we hardcode distantPast's value so that it looks like LLDB is doing970// the right thing.971972// The relative time in seconds from Cocoa Epoch to [NSDate distantPast].973const double RelSecondsFromCocoaEpochToNSDateDistantPast = -63114076800;974if (date_value == RelSecondsFromCocoaEpochToNSDateDistantPast) {975stream.Printf("0001-01-01 00:00:00 UTC");976return true;977}978979// Accomodate for the __NSTaggedDate format introduced in Foundation 1600.980if (class_name == g_NSTaggedDate) {981auto *runtime = llvm::dyn_cast_or_null<AppleObjCRuntime>(982ObjCLanguageRuntime::Get(*process_sp));983if (runtime && runtime->GetFoundationVersion() >= 1600)984date_value = decodeTaggedTimeInterval(value_bits << 4);985}986987// this snippet of code assumes that time_t == seconds since Jan-1-1970 this988// is generally true and POSIXly happy, but might break if a library vendor989// decides to get creative990time_t epoch = GetOSXEpoch();991epoch = epoch + static_cast<time_t>(std::floor(date_value));992tm *tm_date = gmtime(&epoch);993if (!tm_date)994return false;995std::string buffer(1024, 0);996if (strftime(&buffer[0], 1023, "%Z", tm_date) == 0)997return false;998stream.Printf("%04d-%02d-%02d %02d:%02d:%02d %s", tm_date->tm_year + 1900,999tm_date->tm_mon + 1, tm_date->tm_mday, tm_date->tm_hour,1000tm_date->tm_min, tm_date->tm_sec, buffer.c_str());1001return true;1002}10031004bool lldb_private::formatters::ObjCClassSummaryProvider(1005ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {1006ProcessSP process_sp = valobj.GetProcessSP();1007if (!process_sp)1008return false;10091010ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);10111012if (!runtime)1013return false;10141015ObjCLanguageRuntime::ClassDescriptorSP descriptor(1016runtime->GetClassDescriptorFromISA(valobj.GetValueAsUnsigned(0)));10171018if (!descriptor || !descriptor->IsValid())1019return false;10201021ConstString class_name = descriptor->GetClassName();10221023if (class_name.IsEmpty())1024return false;10251026if (ConstString cs = Mangled(class_name).GetDemangledName())1027class_name = cs;10281029stream.Printf("%s", class_name.AsCString("<unknown class>"));1030return true;1031}10321033class ObjCClassSyntheticChildrenFrontEnd : public SyntheticChildrenFrontEnd {1034public:1035ObjCClassSyntheticChildrenFrontEnd(lldb::ValueObjectSP valobj_sp)1036: SyntheticChildrenFrontEnd(*valobj_sp) {}10371038~ObjCClassSyntheticChildrenFrontEnd() override = default;10391040llvm::Expected<uint32_t> CalculateNumChildren() override { return 0; }10411042lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override {1043return lldb::ValueObjectSP();1044}10451046lldb::ChildCacheState Update() override {1047return lldb::ChildCacheState::eRefetch;1048}10491050bool MightHaveChildren() override { return false; }10511052size_t GetIndexOfChildWithName(ConstString name) override {1053return UINT32_MAX;1054}1055};10561057SyntheticChildrenFrontEnd *1058lldb_private::formatters::ObjCClassSyntheticFrontEndCreator(1059CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) {1060return new ObjCClassSyntheticChildrenFrontEnd(valobj_sp);1061}10621063template <bool needs_at>1064bool lldb_private::formatters::NSDataSummaryProvider(1065ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {1066ProcessSP process_sp = valobj.GetProcessSP();1067if (!process_sp)1068return false;10691070ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);10711072if (!runtime)1073return false;10741075ObjCLanguageRuntime::ClassDescriptorSP descriptor(1076runtime->GetClassDescriptor(valobj));10771078if (!descriptor || !descriptor->IsValid())1079return false;10801081bool is_64bit = (process_sp->GetAddressByteSize() == 8);1082lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);10831084if (!valobj_addr)1085return false;10861087uint64_t value = 0;10881089llvm::StringRef class_name = descriptor->GetClassName().GetCString();10901091if (class_name.empty())1092return false;10931094bool isNSConcreteData = class_name == "NSConcreteData";1095bool isNSConcreteMutableData = class_name == "NSConcreteMutableData";1096bool isNSCFData = class_name == "__NSCFData";1097if (isNSConcreteData || isNSConcreteMutableData || isNSCFData) {1098uint32_t offset;1099if (isNSConcreteData)1100offset = is_64bit ? 8 : 4;1101else1102offset = is_64bit ? 16 : 8;11031104Status error;1105value = process_sp->ReadUnsignedIntegerFromMemory(1106valobj_addr + offset, is_64bit ? 8 : 4, 0, error);1107if (error.Fail())1108return false;1109} else if (class_name == "_NSInlineData") {1110uint32_t offset = (is_64bit ? 8 : 4);1111Status error;1112value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + offset, 2,11130, error);1114if (error.Fail())1115return false;1116} else if (class_name == "_NSZeroData") {1117value = 0;1118} else1119return false;11201121stream.Printf("%s%" PRIu64 " byte%s%s", (needs_at ? "@\"" : ""), value,1122(value != 1 ? "s" : ""), (needs_at ? "\"" : ""));11231124return true;1125}11261127bool lldb_private::formatters::ObjCBOOLSummaryProvider(1128ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {1129const uint32_t type_info = valobj.GetCompilerType().GetTypeInfo();11301131ValueObjectSP real_guy_sp = valobj.GetSP();11321133if (type_info & eTypeIsPointer) {1134Status err;1135real_guy_sp = valobj.Dereference(err);1136if (err.Fail() || !real_guy_sp)1137return false;1138} else if (type_info & eTypeIsReference) {1139real_guy_sp = valobj.GetChildAtIndex(0);1140if (!real_guy_sp)1141return false;1142}1143int8_t value = (real_guy_sp->GetValueAsSigned(0) & 0xFF);1144switch (value) {1145case 0:1146stream.Printf("NO");1147break;1148case 1:1149stream.Printf("YES");1150break;1151default:1152stream.Printf("%d", value);1153break;1154}1155return true;1156}11571158bool lldb_private::formatters::ObjCBooleanSummaryProvider(1159ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {1160lldb::addr_t valobj_ptr_value =1161valobj.GetValueAsUnsigned(LLDB_INVALID_ADDRESS);1162if (valobj_ptr_value == LLDB_INVALID_ADDRESS)1163return false;11641165ProcessSP process_sp(valobj.GetProcessSP());1166if (!process_sp)1167return false;11681169if (AppleObjCRuntime *objc_runtime = llvm::dyn_cast_or_null<AppleObjCRuntime>(1170ObjCLanguageRuntime::Get(*process_sp))) {1171lldb::addr_t cf_true = LLDB_INVALID_ADDRESS,1172cf_false = LLDB_INVALID_ADDRESS;1173objc_runtime->GetValuesForGlobalCFBooleans(cf_true, cf_false);1174if (valobj_ptr_value == cf_true) {1175stream.PutCString("YES");1176return true;1177}1178if (valobj_ptr_value == cf_false) {1179stream.PutCString("NO");1180return true;1181}1182}11831184return false;1185}11861187template <bool is_sel_ptr>1188bool lldb_private::formatters::ObjCSELSummaryProvider(1189ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {1190lldb::ValueObjectSP valobj_sp;11911192CompilerType charstar(valobj.GetCompilerType()1193.GetBasicTypeFromAST(eBasicTypeChar)1194.GetPointerType());11951196if (!charstar)1197return false;11981199ExecutionContext exe_ctx(valobj.GetExecutionContextRef());12001201if (is_sel_ptr) {1202lldb::addr_t data_address = valobj.GetValueAsUnsigned(LLDB_INVALID_ADDRESS);1203if (data_address == LLDB_INVALID_ADDRESS)1204return false;1205valobj_sp = ValueObject::CreateValueObjectFromAddress("text", data_address,1206exe_ctx, charstar);1207} else {1208DataExtractor data;1209Status error;1210valobj.GetData(data, error);1211if (error.Fail())1212return false;1213valobj_sp =1214ValueObject::CreateValueObjectFromData("text", data, exe_ctx, charstar);1215}12161217if (!valobj_sp)1218return false;12191220stream.Printf("%s", valobj_sp->GetSummaryAsCString());1221return true;1222}12231224// POSIX has an epoch on Jan-1-1970, but Cocoa prefers Jan-1-20011225// this call gives the POSIX equivalent of the Cocoa epoch1226time_t lldb_private::formatters::GetOSXEpoch() {1227static time_t epoch = 0;1228if (!epoch) {1229#ifndef _WIN321230tzset();1231tm tm_epoch;1232tm_epoch.tm_sec = 0;1233tm_epoch.tm_hour = 0;1234tm_epoch.tm_min = 0;1235tm_epoch.tm_mon = 0;1236tm_epoch.tm_mday = 1;1237tm_epoch.tm_year = 2001 - 1900;1238tm_epoch.tm_isdst = -1;1239tm_epoch.tm_gmtoff = 0;1240tm_epoch.tm_zone = nullptr;1241epoch = timegm(&tm_epoch);1242#endif1243}1244return epoch;1245}12461247template bool lldb_private::formatters::NSDataSummaryProvider<true>(1248ValueObject &, Stream &, const TypeSummaryOptions &);12491250template bool lldb_private::formatters::NSDataSummaryProvider<false>(1251ValueObject &, Stream &, const TypeSummaryOptions &);12521253template bool lldb_private::formatters::ObjCSELSummaryProvider<true>(1254ValueObject &, Stream &, const TypeSummaryOptions &);12551256template bool lldb_private::formatters::ObjCSELSummaryProvider<false>(1257ValueObject &, Stream &, const TypeSummaryOptions &);125812591260