1c156427dSZachary Turner //===-- FormatManager.cpp ----------------------------------------*- C++-*-===// 25548cb50SEnrico Granata // 32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 65548cb50SEnrico Granata // 75548cb50SEnrico Granata //===----------------------------------------------------------------------===// 85548cb50SEnrico Granata 95548cb50SEnrico Granata #include "lldb/DataFormatters/FormatManager.h" 105548cb50SEnrico Granata 11d717cc9fSEnrico Granata #include "llvm/ADT/STLExtras.h" 12d717cc9fSEnrico Granata 135548cb50SEnrico Granata 145548cb50SEnrico Granata #include "lldb/Core/Debugger.h" 15df7e79e6SEnrico Granata #include "lldb/DataFormatters/FormattersHelpers.h" 16980c0484SEnrico Granata #include "lldb/DataFormatters/LanguageCategory.h" 175548cb50SEnrico Granata #include "lldb/Target/ExecutionContext.h" 1833e97e63SEnrico Granata #include "lldb/Target/Language.h" 196f9e6901SZachary Turner #include "lldb/Utility/Log.h" 20980c0484SEnrico Granata 215548cb50SEnrico Granata using namespace lldb; 225548cb50SEnrico Granata using namespace lldb_private; 23df7e79e6SEnrico Granata using namespace lldb_private::formatters; 245548cb50SEnrico Granata 25b9c1b51eSKate Stone struct FormatInfo { 265548cb50SEnrico Granata Format format; 27b9c1b51eSKate Stone const char format_char; // One or more format characters that can be used for 28b9c1b51eSKate Stone // this format. 29b9c1b51eSKate Stone const char *format_name; // Long format name that can be used to specify the 30b9c1b51eSKate Stone // current format 315548cb50SEnrico Granata }; 325548cb50SEnrico Granata 3312002fbdSJonas Devlieghere static constexpr FormatInfo g_format_infos[] = { 345548cb50SEnrico Granata {eFormatDefault, '\0', "default"}, 355548cb50SEnrico Granata {eFormatBoolean, 'B', "boolean"}, 365548cb50SEnrico Granata {eFormatBinary, 'b', "binary"}, 375548cb50SEnrico Granata {eFormatBytes, 'y', "bytes"}, 385548cb50SEnrico Granata {eFormatBytesWithASCII, 'Y', "bytes with ASCII"}, 395548cb50SEnrico Granata {eFormatChar, 'c', "character"}, 405548cb50SEnrico Granata {eFormatCharPrintable, 'C', "printable character"}, 415548cb50SEnrico Granata {eFormatComplexFloat, 'F', "complex float"}, 425548cb50SEnrico Granata {eFormatCString, 's', "c-string"}, 435548cb50SEnrico Granata {eFormatDecimal, 'd', "decimal"}, 445548cb50SEnrico Granata {eFormatEnum, 'E', "enumeration"}, 455548cb50SEnrico Granata {eFormatHex, 'x', "hex"}, 465548cb50SEnrico Granata {eFormatHexUppercase, 'X', "uppercase hex"}, 475548cb50SEnrico Granata {eFormatFloat, 'f', "float"}, 485548cb50SEnrico Granata {eFormatOctal, 'o', "octal"}, 495548cb50SEnrico Granata {eFormatOSType, 'O', "OSType"}, 505548cb50SEnrico Granata {eFormatUnicode16, 'U', "unicode16"}, 515548cb50SEnrico Granata {eFormatUnicode32, '\0', "unicode32"}, 525548cb50SEnrico Granata {eFormatUnsigned, 'u', "unsigned decimal"}, 535548cb50SEnrico Granata {eFormatPointer, 'p', "pointer"}, 545548cb50SEnrico Granata {eFormatVectorOfChar, '\0', "char[]"}, 555548cb50SEnrico Granata {eFormatVectorOfSInt8, '\0', "int8_t[]"}, 565548cb50SEnrico Granata {eFormatVectorOfUInt8, '\0', "uint8_t[]"}, 575548cb50SEnrico Granata {eFormatVectorOfSInt16, '\0', "int16_t[]"}, 585548cb50SEnrico Granata {eFormatVectorOfUInt16, '\0', "uint16_t[]"}, 595548cb50SEnrico Granata {eFormatVectorOfSInt32, '\0', "int32_t[]"}, 605548cb50SEnrico Granata {eFormatVectorOfUInt32, '\0', "uint32_t[]"}, 615548cb50SEnrico Granata {eFormatVectorOfSInt64, '\0', "int64_t[]"}, 625548cb50SEnrico Granata {eFormatVectorOfUInt64, '\0', "uint64_t[]"}, 63a0f08674SEwan Crawford {eFormatVectorOfFloat16, '\0', "float16[]"}, 645548cb50SEnrico Granata {eFormatVectorOfFloat32, '\0', "float32[]"}, 655548cb50SEnrico Granata {eFormatVectorOfFloat64, '\0', "float64[]"}, 665548cb50SEnrico Granata {eFormatVectorOfUInt128, '\0', "uint128_t[]"}, 675548cb50SEnrico Granata {eFormatComplexInteger, 'I', "complex integer"}, 685548cb50SEnrico Granata {eFormatCharArray, 'a', "character array"}, 695548cb50SEnrico Granata {eFormatAddressInfo, 'A', "address"}, 705548cb50SEnrico Granata {eFormatHexFloat, '\0', "hex float"}, 715548cb50SEnrico Granata {eFormatInstruction, 'i', "instruction"}, 722621f7bdSJonas Devlieghere {eFormatVoid, 'v', "void"}, 732621f7bdSJonas Devlieghere {eFormatUnicode8, 'u', "unicode8"}, 742621f7bdSJonas Devlieghere }; 755548cb50SEnrico Granata 7612002fbdSJonas Devlieghere static_assert((sizeof(g_format_infos) / sizeof(g_format_infos[0])) == 7712002fbdSJonas Devlieghere kNumFormats, 7812002fbdSJonas Devlieghere "All formats must have a corresponding info entry."); 7912002fbdSJonas Devlieghere 8028606954SSaleem Abdulrasool static uint32_t g_num_format_infos = llvm::array_lengthof(g_format_infos); 815548cb50SEnrico Granata 82b9c1b51eSKate Stone static bool GetFormatFromFormatChar(char format_char, Format &format) { 83b9c1b51eSKate Stone for (uint32_t i = 0; i < g_num_format_infos; ++i) { 84b9c1b51eSKate Stone if (g_format_infos[i].format_char == format_char) { 855548cb50SEnrico Granata format = g_format_infos[i].format; 865548cb50SEnrico Granata return true; 875548cb50SEnrico Granata } 885548cb50SEnrico Granata } 895548cb50SEnrico Granata format = eFormatInvalid; 905548cb50SEnrico Granata return false; 915548cb50SEnrico Granata } 925548cb50SEnrico Granata 93b9c1b51eSKate Stone static bool GetFormatFromFormatName(const char *format_name, 94b9c1b51eSKate Stone bool partial_match_ok, Format &format) { 955548cb50SEnrico Granata uint32_t i; 96b9c1b51eSKate Stone for (i = 0; i < g_num_format_infos; ++i) { 97b9c1b51eSKate Stone if (strcasecmp(g_format_infos[i].format_name, format_name) == 0) { 985548cb50SEnrico Granata format = g_format_infos[i].format; 995548cb50SEnrico Granata return true; 1005548cb50SEnrico Granata } 1015548cb50SEnrico Granata } 1025548cb50SEnrico Granata 103b9c1b51eSKate Stone if (partial_match_ok) { 104b9c1b51eSKate Stone for (i = 0; i < g_num_format_infos; ++i) { 105b9c1b51eSKate Stone if (strcasestr(g_format_infos[i].format_name, format_name) == 106b9c1b51eSKate Stone g_format_infos[i].format_name) { 1075548cb50SEnrico Granata format = g_format_infos[i].format; 1085548cb50SEnrico Granata return true; 1095548cb50SEnrico Granata } 1105548cb50SEnrico Granata } 1115548cb50SEnrico Granata } 1125548cb50SEnrico Granata format = eFormatInvalid; 1135548cb50SEnrico Granata return false; 1145548cb50SEnrico Granata } 1155548cb50SEnrico Granata 116b9c1b51eSKate Stone void FormatManager::Changed() { 117bd5eab82SEnrico Granata ++m_last_revision; 118bd5eab82SEnrico Granata m_format_cache.Clear(); 11916ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 120b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 121bd5eab82SEnrico Granata if (iter.second) 122bd5eab82SEnrico Granata iter.second->GetFormatCache().Clear(); 123bd5eab82SEnrico Granata } 124bd5eab82SEnrico Granata } 125bd5eab82SEnrico Granata 126b9c1b51eSKate Stone bool FormatManager::GetFormatFromCString(const char *format_cstr, 1275548cb50SEnrico Granata bool partial_match_ok, 128b9c1b51eSKate Stone lldb::Format &format) { 1295548cb50SEnrico Granata bool success = false; 130b9c1b51eSKate Stone if (format_cstr && format_cstr[0]) { 131b9c1b51eSKate Stone if (format_cstr[1] == '\0') { 1325548cb50SEnrico Granata success = GetFormatFromFormatChar(format_cstr[0], format); 1335548cb50SEnrico Granata if (success) 1345548cb50SEnrico Granata return true; 1355548cb50SEnrico Granata } 1365548cb50SEnrico Granata 1375548cb50SEnrico Granata success = GetFormatFromFormatName(format_cstr, partial_match_ok, format); 1385548cb50SEnrico Granata } 1395548cb50SEnrico Granata if (!success) 1405548cb50SEnrico Granata format = eFormatInvalid; 1415548cb50SEnrico Granata return success; 1425548cb50SEnrico Granata } 1435548cb50SEnrico Granata 144b9c1b51eSKate Stone char FormatManager::GetFormatAsFormatChar(lldb::Format format) { 145b9c1b51eSKate Stone for (uint32_t i = 0; i < g_num_format_infos; ++i) { 1465548cb50SEnrico Granata if (g_format_infos[i].format == format) 1475548cb50SEnrico Granata return g_format_infos[i].format_char; 1485548cb50SEnrico Granata } 1495548cb50SEnrico Granata return '\0'; 1505548cb50SEnrico Granata } 1515548cb50SEnrico Granata 152b9c1b51eSKate Stone const char *FormatManager::GetFormatAsCString(Format format) { 1535548cb50SEnrico Granata if (format >= eFormatDefault && format < kNumFormats) 1545548cb50SEnrico Granata return g_format_infos[format].format_name; 155248a1305SKonrad Kleine return nullptr; 1565548cb50SEnrico Granata } 1575548cb50SEnrico Granata 158b9c1b51eSKate Stone void FormatManager::EnableAllCategories() { 15933e97e63SEnrico Granata m_categories_map.EnableAllCategories(); 16016ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 161b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 16233e97e63SEnrico Granata if (iter.second) 16333e97e63SEnrico Granata iter.second->Enable(); 16433e97e63SEnrico Granata } 16533e97e63SEnrico Granata } 16633e97e63SEnrico Granata 167b9c1b51eSKate Stone void FormatManager::DisableAllCategories() { 16833e97e63SEnrico Granata m_categories_map.DisableAllCategories(); 16916ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 170b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 17133e97e63SEnrico Granata if (iter.second) 17233e97e63SEnrico Granata iter.second->Disable(); 17333e97e63SEnrico Granata } 17433e97e63SEnrico Granata } 17533e97e63SEnrico Granata 176b9c1b51eSKate Stone void FormatManager::GetPossibleMatches( 177b9c1b51eSKate Stone ValueObject &valobj, CompilerType compiler_type, uint32_t reason, 178b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic, FormattersMatchVector &entries, 179b9c1b51eSKate Stone bool did_strip_ptr, bool did_strip_ref, bool did_strip_typedef, 180b9c1b51eSKate Stone bool root_level) { 181c6bf2e2dSEnrico Granata compiler_type = compiler_type.GetTypeForFormatters(); 18259b5a37dSBruce Mitchener ConstString type_name(compiler_type.GetConstTypeName()); 183b9c1b51eSKate Stone if (valobj.GetBitfieldBitSize() > 0) { 184de61cecdSEnrico Granata StreamString sstring; 185de61cecdSEnrico Granata sstring.Printf("%s:%d", type_name.AsCString(), valobj.GetBitfieldBitSize()); 186c156427dSZachary Turner ConstString bitfieldname(sstring.GetString()); 187b9c1b51eSKate Stone entries.push_back( 188b9c1b51eSKate Stone {bitfieldname, 0, did_strip_ptr, did_strip_ref, did_strip_typedef}); 189de61cecdSEnrico Granata reason |= lldb_private::eFormatterChoiceCriterionStrippedBitField; 190de61cecdSEnrico Granata } 191b3f0c340SEnrico Granata 19278f05d35SDavide Italiano if (!compiler_type.IsMeaninglessWithoutDynamicResolution()) { 193b9c1b51eSKate Stone entries.push_back( 194b9c1b51eSKate Stone {type_name, reason, did_strip_ptr, did_strip_ref, did_strip_typedef}); 195de61cecdSEnrico Granata 19659b5a37dSBruce Mitchener ConstString display_type_name(compiler_type.GetDisplayTypeName()); 197e8daa2f8SEnrico Granata if (display_type_name != type_name) 19878f05d35SDavide Italiano entries.push_back({display_type_name, reason, did_strip_ptr, 19978f05d35SDavide Italiano did_strip_ref, did_strip_typedef}); 20078f05d35SDavide Italiano } 201e8daa2f8SEnrico Granata 202b9c1b51eSKate Stone for (bool is_rvalue_ref = true, j = true; 203b9c1b51eSKate Stone j && compiler_type.IsReferenceType(nullptr, &is_rvalue_ref); j = false) { 20459b5a37dSBruce Mitchener CompilerType non_ref_type = compiler_type.GetNonReferenceType(); 205b9c1b51eSKate Stone GetPossibleMatches( 206b9c1b51eSKate Stone valobj, non_ref_type, 207b9c1b51eSKate Stone reason | 208b9c1b51eSKate Stone lldb_private::eFormatterChoiceCriterionStrippedPointerReference, 209b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, true, did_strip_typedef); 210b9c1b51eSKate Stone if (non_ref_type.IsTypedefType()) { 211a1e5dc86SGreg Clayton CompilerType deffed_referenced_type = non_ref_type.GetTypedefedType(); 212b9c1b51eSKate Stone deffed_referenced_type = 213b9c1b51eSKate Stone is_rvalue_ref ? deffed_referenced_type.GetRValueReferenceType() 214b9c1b51eSKate Stone : deffed_referenced_type.GetLValueReferenceType(); 215b9c1b51eSKate Stone GetPossibleMatches( 216b9c1b51eSKate Stone valobj, deffed_referenced_type, 2171ac62963SEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 218b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 2191ac62963SEnrico Granata true); // this is not exactly the usual meaning of stripping typedefs 220de61cecdSEnrico Granata } 2211ac62963SEnrico Granata } 2221ac62963SEnrico Granata 223b9c1b51eSKate Stone if (compiler_type.IsPointerType()) { 22459b5a37dSBruce Mitchener CompilerType non_ptr_type = compiler_type.GetPointeeType(); 225b9c1b51eSKate Stone GetPossibleMatches( 226b9c1b51eSKate Stone valobj, non_ptr_type, 227b9c1b51eSKate Stone reason | 228b9c1b51eSKate Stone lldb_private::eFormatterChoiceCriterionStrippedPointerReference, 229b9c1b51eSKate Stone use_dynamic, entries, true, did_strip_ref, did_strip_typedef); 230b9c1b51eSKate Stone if (non_ptr_type.IsTypedefType()) { 231b9c1b51eSKate Stone CompilerType deffed_pointed_type = 232b9c1b51eSKate Stone non_ptr_type.GetTypedefedType().GetPointerType(); 233b9c1b51eSKate Stone GetPossibleMatches( 234b9c1b51eSKate Stone valobj, deffed_pointed_type, 2351ac62963SEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 236b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 2371ac62963SEnrico Granata true); // this is not exactly the usual meaning of stripping typedefs 2381ac62963SEnrico Granata } 239de61cecdSEnrico Granata } 240de61cecdSEnrico Granata 241*bc69dd2cSDavide Italiano for (lldb::LanguageType language_type : 242*bc69dd2cSDavide Italiano GetCandidateLanguages(valobj.GetObjectRuntimeLanguage())) { 243b9c1b51eSKate Stone if (Language *language = Language::FindPlugin(language_type)) { 244b9c1b51eSKate Stone for (ConstString candidate : 245b9c1b51eSKate Stone language->GetPossibleFormattersMatches(valobj, use_dynamic)) { 246b9c1b51eSKate Stone entries.push_back( 247b9c1b51eSKate Stone {candidate, 248d3233c1eSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionLanguagePlugin, 249b9c1b51eSKate Stone did_strip_ptr, did_strip_ref, did_strip_typedef}); 250d3233c1eSEnrico Granata } 251d3233c1eSEnrico Granata } 252de61cecdSEnrico Granata } 253de61cecdSEnrico Granata 254de61cecdSEnrico Granata // try to strip typedef chains 255b9c1b51eSKate Stone if (compiler_type.IsTypedefType()) { 25659b5a37dSBruce Mitchener CompilerType deffed_type = compiler_type.GetTypedefedType(); 257b9c1b51eSKate Stone GetPossibleMatches( 258b9c1b51eSKate Stone valobj, deffed_type, 259de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 260b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, true); 261de61cecdSEnrico Granata } 262de61cecdSEnrico Granata 263b9c1b51eSKate Stone if (root_level) { 264de61cecdSEnrico Granata do { 26559b5a37dSBruce Mitchener if (!compiler_type.IsValid()) 266de61cecdSEnrico Granata break; 267de61cecdSEnrico Granata 268b9c1b51eSKate Stone CompilerType unqual_compiler_ast_type = 269b9c1b51eSKate Stone compiler_type.GetFullyUnqualifiedType(); 27059b5a37dSBruce Mitchener if (!unqual_compiler_ast_type.IsValid()) 271de61cecdSEnrico Granata break; 272b9c1b51eSKate Stone if (unqual_compiler_ast_type.GetOpaqueQualType() != 273b9c1b51eSKate Stone compiler_type.GetOpaqueQualType()) 274b9c1b51eSKate Stone GetPossibleMatches(valobj, unqual_compiler_ast_type, reason, 275b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 276de61cecdSEnrico Granata did_strip_typedef); 277de61cecdSEnrico Granata } while (false); 278de61cecdSEnrico Granata 279de61cecdSEnrico Granata // if all else fails, go to static type 280b9c1b51eSKate Stone if (valobj.IsDynamic()) { 281de61cecdSEnrico Granata lldb::ValueObjectSP static_value_sp(valobj.GetStaticValue()); 282de61cecdSEnrico Granata if (static_value_sp) 283b9c1b51eSKate Stone GetPossibleMatches( 284b9c1b51eSKate Stone *static_value_sp.get(), static_value_sp->GetCompilerType(), 285de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionWentToStaticValue, 286b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 287b9c1b51eSKate Stone did_strip_typedef, true); 288de61cecdSEnrico Granata } 289de61cecdSEnrico Granata } 290de61cecdSEnrico Granata } 291de61cecdSEnrico Granata 292852cc954SEnrico Granata lldb::TypeFormatImplSP 293b9c1b51eSKate Stone FormatManager::GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp) { 294852cc954SEnrico Granata if (!type_sp) 295852cc954SEnrico Granata return lldb::TypeFormatImplSP(); 296852cc954SEnrico Granata lldb::TypeFormatImplSP format_chosen_sp; 297852cc954SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 298852cc954SEnrico Granata lldb::TypeCategoryImplSP category_sp; 299852cc954SEnrico Granata uint32_t prio_category = UINT32_MAX; 300b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 301852cc954SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 302a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 303852cc954SEnrico Granata continue; 304b9c1b51eSKate Stone lldb::TypeFormatImplSP format_current_sp = 305b9c1b51eSKate Stone category_sp->GetFormatForType(type_sp); 306b9c1b51eSKate Stone if (format_current_sp && 307248a1305SKonrad Kleine (format_chosen_sp.get() == nullptr || 308b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 309852cc954SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 310852cc954SEnrico Granata format_chosen_sp = format_current_sp; 311852cc954SEnrico Granata } 312852cc954SEnrico Granata } 313852cc954SEnrico Granata return format_chosen_sp; 314852cc954SEnrico Granata } 315852cc954SEnrico Granata 3165548cb50SEnrico Granata lldb::TypeSummaryImplSP 317b9c1b51eSKate Stone FormatManager::GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp) { 3185548cb50SEnrico Granata if (!type_sp) 3195548cb50SEnrico Granata return lldb::TypeSummaryImplSP(); 3205548cb50SEnrico Granata lldb::TypeSummaryImplSP summary_chosen_sp; 3215548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3225548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3235548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 324b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3255548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 326a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3275548cb50SEnrico Granata continue; 328b9c1b51eSKate Stone lldb::TypeSummaryImplSP summary_current_sp = 329b9c1b51eSKate Stone category_sp->GetSummaryForType(type_sp); 330b9c1b51eSKate Stone if (summary_current_sp && 331248a1305SKonrad Kleine (summary_chosen_sp.get() == nullptr || 332b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3335548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3345548cb50SEnrico Granata summary_chosen_sp = summary_current_sp; 3355548cb50SEnrico Granata } 3365548cb50SEnrico Granata } 3375548cb50SEnrico Granata return summary_chosen_sp; 3385548cb50SEnrico Granata } 3395548cb50SEnrico Granata 3405548cb50SEnrico Granata lldb::TypeFilterImplSP 341b9c1b51eSKate Stone FormatManager::GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp) { 3425548cb50SEnrico Granata if (!type_sp) 3435548cb50SEnrico Granata return lldb::TypeFilterImplSP(); 3445548cb50SEnrico Granata lldb::TypeFilterImplSP filter_chosen_sp; 3455548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3465548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3475548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 348b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3495548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 350a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3515548cb50SEnrico Granata continue; 352b9c1b51eSKate Stone lldb::TypeFilterImplSP filter_current_sp( 353b9c1b51eSKate Stone (TypeFilterImpl *)category_sp->GetFilterForType(type_sp).get()); 354b9c1b51eSKate Stone if (filter_current_sp && 355248a1305SKonrad Kleine (filter_chosen_sp.get() == nullptr || 356b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3575548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3585548cb50SEnrico Granata filter_chosen_sp = filter_current_sp; 3595548cb50SEnrico Granata } 3605548cb50SEnrico Granata } 3615548cb50SEnrico Granata return filter_chosen_sp; 3625548cb50SEnrico Granata } 3635548cb50SEnrico Granata 3645548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP 365b9c1b51eSKate Stone FormatManager::GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp) { 3665548cb50SEnrico Granata if (!type_sp) 3675548cb50SEnrico Granata return lldb::ScriptedSyntheticChildrenSP(); 3685548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP synth_chosen_sp; 3695548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3705548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3715548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 372b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3735548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 374a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3755548cb50SEnrico Granata continue; 376b9c1b51eSKate Stone lldb::ScriptedSyntheticChildrenSP synth_current_sp( 377b9c1b51eSKate Stone (ScriptedSyntheticChildren *)category_sp->GetSyntheticForType(type_sp) 378b9c1b51eSKate Stone .get()); 379b9c1b51eSKate Stone if (synth_current_sp && 380248a1305SKonrad Kleine (synth_chosen_sp.get() == nullptr || 381b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3825548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3835548cb50SEnrico Granata synth_chosen_sp = synth_current_sp; 3845548cb50SEnrico Granata } 3855548cb50SEnrico Granata } 3865548cb50SEnrico Granata return synth_chosen_sp; 3875548cb50SEnrico Granata } 3885548cb50SEnrico Granata 389c582713cSEnrico Granata lldb::TypeValidatorImplSP 390b9c1b51eSKate Stone FormatManager::GetValidatorForType(lldb::TypeNameSpecifierImplSP type_sp) { 391c582713cSEnrico Granata if (!type_sp) 392c582713cSEnrico Granata return lldb::TypeValidatorImplSP(); 393c582713cSEnrico Granata lldb::TypeValidatorImplSP validator_chosen_sp; 394c582713cSEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 395c582713cSEnrico Granata lldb::TypeCategoryImplSP category_sp; 396c582713cSEnrico Granata uint32_t prio_category = UINT32_MAX; 397b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 398c582713cSEnrico Granata category_sp = GetCategoryAtIndex(category_id); 399a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 400c582713cSEnrico Granata continue; 401b9c1b51eSKate Stone lldb::TypeValidatorImplSP validator_current_sp( 402b9c1b51eSKate Stone category_sp->GetValidatorForType(type_sp).get()); 403b9c1b51eSKate Stone if (validator_current_sp && 404248a1305SKonrad Kleine (validator_chosen_sp.get() == nullptr || 405b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 406c582713cSEnrico Granata prio_category = category_sp->GetEnabledPosition(); 407c582713cSEnrico Granata validator_chosen_sp = validator_current_sp; 408c582713cSEnrico Granata } 409c582713cSEnrico Granata } 410c582713cSEnrico Granata return validator_chosen_sp; 411c582713cSEnrico Granata } 412c582713cSEnrico Granata 413b9c1b51eSKate Stone void FormatManager::ForEachCategory(TypeCategoryMap::ForEachCallback callback) { 414b56d0103SEnrico Granata m_categories_map.ForEach(callback); 41516ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 416b9c1b51eSKate Stone for (const auto &entry : m_language_categories_map) { 417b9c1b51eSKate Stone if (auto category_sp = entry.second->GetCategory()) { 418b56d0103SEnrico Granata if (!callback(category_sp)) 419b56d0103SEnrico Granata break; 420b56d0103SEnrico Granata } 421b56d0103SEnrico Granata } 422b56d0103SEnrico Granata } 423b56d0103SEnrico Granata 4245548cb50SEnrico Granata lldb::TypeCategoryImplSP 4250e4c4821SAdrian Prantl FormatManager::GetCategory(ConstString category_name, bool can_create) { 4265548cb50SEnrico Granata if (!category_name) 4275548cb50SEnrico Granata return GetCategory(m_default_category_name); 4285548cb50SEnrico Granata lldb::TypeCategoryImplSP category; 4295548cb50SEnrico Granata if (m_categories_map.Get(category_name, category)) 4305548cb50SEnrico Granata return category; 4315548cb50SEnrico Granata 4325548cb50SEnrico Granata if (!can_create) 4335548cb50SEnrico Granata return lldb::TypeCategoryImplSP(); 4345548cb50SEnrico Granata 435b9c1b51eSKate Stone m_categories_map.Add( 436b9c1b51eSKate Stone category_name, 437b9c1b51eSKate Stone lldb::TypeCategoryImplSP(new TypeCategoryImpl(this, category_name))); 4385548cb50SEnrico Granata return GetCategory(category_name); 4395548cb50SEnrico Granata } 4405548cb50SEnrico Granata 441b9c1b51eSKate Stone lldb::Format FormatManager::GetSingleItemFormat(lldb::Format vector_format) { 442b9c1b51eSKate Stone switch (vector_format) { 4435548cb50SEnrico Granata case eFormatVectorOfChar: 4445548cb50SEnrico Granata return eFormatCharArray; 4455548cb50SEnrico Granata 4465548cb50SEnrico Granata case eFormatVectorOfSInt8: 4475548cb50SEnrico Granata case eFormatVectorOfSInt16: 4485548cb50SEnrico Granata case eFormatVectorOfSInt32: 4495548cb50SEnrico Granata case eFormatVectorOfSInt64: 4505548cb50SEnrico Granata return eFormatDecimal; 4515548cb50SEnrico Granata 4525548cb50SEnrico Granata case eFormatVectorOfUInt8: 4535548cb50SEnrico Granata case eFormatVectorOfUInt16: 4545548cb50SEnrico Granata case eFormatVectorOfUInt32: 4555548cb50SEnrico Granata case eFormatVectorOfUInt64: 4565548cb50SEnrico Granata case eFormatVectorOfUInt128: 4575548cb50SEnrico Granata return eFormatHex; 4585548cb50SEnrico Granata 459a0f08674SEwan Crawford case eFormatVectorOfFloat16: 4605548cb50SEnrico Granata case eFormatVectorOfFloat32: 4615548cb50SEnrico Granata case eFormatVectorOfFloat64: 4625548cb50SEnrico Granata return eFormatFloat; 4635548cb50SEnrico Granata 4645548cb50SEnrico Granata default: 4655548cb50SEnrico Granata return lldb::eFormatInvalid; 4665548cb50SEnrico Granata } 4675548cb50SEnrico Granata } 4685548cb50SEnrico Granata 469b9c1b51eSKate Stone bool FormatManager::ShouldPrintAsOneLiner(ValueObject &valobj) { 470553fad5cSEnrico Granata // if settings say no oneline whatsoever 471b9c1b51eSKate Stone if (valobj.GetTargetSP().get() && 472a6682a41SJonas Devlieghere !valobj.GetTargetSP()->GetDebugger().GetAutoOneLineSummaries()) 473553fad5cSEnrico Granata return false; // then don't oneline 474553fad5cSEnrico Granata 47542fa4af8SEnrico Granata // if this object has a summary, then ask the summary 476a29cb0baSEnrico Granata if (valobj.GetSummaryFormat().get() != nullptr) 47742fa4af8SEnrico Granata return valobj.GetSummaryFormat()->IsOneLiner(); 478a29cb0baSEnrico Granata 479a29cb0baSEnrico Granata // no children, no party 480a29cb0baSEnrico Granata if (valobj.GetNumChildren() == 0) 481a29cb0baSEnrico Granata return false; 482a29cb0baSEnrico Granata 48305097246SAdrian Prantl // ask the type if it has any opinion about this eLazyBoolCalculate == no 48405097246SAdrian Prantl // opinion; other values should be self explanatory 4859c63f99aSEnrico Granata CompilerType compiler_type(valobj.GetCompilerType()); 486b9c1b51eSKate Stone if (compiler_type.IsValid()) { 487b9c1b51eSKate Stone switch (compiler_type.ShouldPrintAsOneLiner(&valobj)) { 4889c63f99aSEnrico Granata case eLazyBoolNo: 4899c63f99aSEnrico Granata return false; 4909c63f99aSEnrico Granata case eLazyBoolYes: 4919c63f99aSEnrico Granata return true; 4929c63f99aSEnrico Granata case eLazyBoolCalculate: 4939c63f99aSEnrico Granata break; 4949c63f99aSEnrico Granata } 4959c63f99aSEnrico Granata } 4969c63f99aSEnrico Granata 497a29cb0baSEnrico Granata size_t total_children_name_len = 0; 498a29cb0baSEnrico Granata 499b9c1b51eSKate Stone for (size_t idx = 0; idx < valobj.GetNumChildren(); idx++) { 500ddac7611SEnrico Granata bool is_synth_val = false; 501a29cb0baSEnrico Granata ValueObjectSP child_sp(valobj.GetChildAtIndex(idx, true)); 502a29cb0baSEnrico Granata // something is wrong here - bail out 503a29cb0baSEnrico Granata if (!child_sp) 504a29cb0baSEnrico Granata return false; 5056500061eSEnrico Granata 5066500061eSEnrico Granata // also ask the child's type if it has any opinion 5076500061eSEnrico Granata CompilerType child_compiler_type(child_sp->GetCompilerType()); 508b9c1b51eSKate Stone if (child_compiler_type.IsValid()) { 509b9c1b51eSKate Stone switch (child_compiler_type.ShouldPrintAsOneLiner(child_sp.get())) { 5106500061eSEnrico Granata case eLazyBoolYes: 5116500061eSEnrico Granata // an opinion of yes is only binding for the child, so keep going 5126500061eSEnrico Granata case eLazyBoolCalculate: 5136500061eSEnrico Granata break; 5146500061eSEnrico Granata case eLazyBoolNo: 5156500061eSEnrico Granata // but if the child says no, then it's a veto on the whole thing 5166500061eSEnrico Granata return false; 5176500061eSEnrico Granata } 5186500061eSEnrico Granata } 5196500061eSEnrico Granata 520b9c1b51eSKate Stone // if we decided to define synthetic children for a type, we probably care 52105097246SAdrian Prantl // enough to show them, but avoid nesting children in children 522b9c1b51eSKate Stone if (child_sp->GetSyntheticChildren().get() != nullptr) { 523ddac7611SEnrico Granata ValueObjectSP synth_sp(child_sp->GetSyntheticValue()); 524ddac7611SEnrico Granata // wait.. wat? just get out of here.. 525ddac7611SEnrico Granata if (!synth_sp) 526a29cb0baSEnrico Granata return false; 527ddac7611SEnrico Granata // but if we only have them to provide a value, keep going 528a6682a41SJonas Devlieghere if (!synth_sp->MightHaveChildren() && 529b9c1b51eSKate Stone synth_sp->DoesProvideSyntheticValue()) 530ddac7611SEnrico Granata is_synth_val = true; 531ddac7611SEnrico Granata else 532ddac7611SEnrico Granata return false; 533ddac7611SEnrico Granata } 534a29cb0baSEnrico Granata 535a29cb0baSEnrico Granata total_children_name_len += child_sp->GetName().GetLength(); 536a29cb0baSEnrico Granata 537a29cb0baSEnrico Granata // 50 itself is a "randomly" chosen number - the idea is that 538a29cb0baSEnrico Granata // overly long structs should not get this treatment 539a29cb0baSEnrico Granata // FIXME: maybe make this a user-tweakable setting? 540a29cb0baSEnrico Granata if (total_children_name_len > 50) 541a29cb0baSEnrico Granata return false; 542a29cb0baSEnrico Granata 543a29cb0baSEnrico Granata // if a summary is there.. 544b9c1b51eSKate Stone if (child_sp->GetSummaryFormat()) { 545a29cb0baSEnrico Granata // and it wants children, then bail out 5468a068e6cSEnrico Granata if (child_sp->GetSummaryFormat()->DoesPrintChildren(child_sp.get())) 547a29cb0baSEnrico Granata return false; 548a29cb0baSEnrico Granata } 549a29cb0baSEnrico Granata 550c89e4ca3SEnrico Granata // if this child has children.. 551b9c1b51eSKate Stone if (child_sp->GetNumChildren()) { 552a29cb0baSEnrico Granata // ...and no summary... 553b9c1b51eSKate Stone // (if it had a summary and the summary wanted children, we would have 554b9c1b51eSKate Stone // bailed out anyway 555b9c1b51eSKate Stone // so this only makes us bail out if this has no summary and we would 556b9c1b51eSKate Stone // then print children) 557b9c1b51eSKate Stone if (!child_sp->GetSummaryFormat() && !is_synth_val) // but again only do 558b9c1b51eSKate Stone // that if not a 559b9c1b51eSKate Stone // synthetic valued 560b9c1b51eSKate Stone // child 561a29cb0baSEnrico Granata return false; // then bail out 562a29cb0baSEnrico Granata } 563a29cb0baSEnrico Granata } 564a29cb0baSEnrico Granata return true; 565a29cb0baSEnrico Granata } 566a29cb0baSEnrico Granata 5670e4c4821SAdrian Prantl ConstString FormatManager::GetValidTypeName(ConstString type) { 5685548cb50SEnrico Granata return ::GetValidTypeName_Impl(type); 5695548cb50SEnrico Granata } 5705548cb50SEnrico Granata 571b9c1b51eSKate Stone ConstString FormatManager::GetTypeForCache(ValueObject &valobj, 572b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 573b9c1b51eSKate Stone ValueObjectSP valobj_sp = valobj.GetQualifiedRepresentationIfAvailable( 574b9c1b51eSKate Stone use_dynamic, valobj.IsSynthetic()); 57578f05d35SDavide Italiano if (valobj_sp && valobj_sp->GetCompilerType().IsValid()) { 57678f05d35SDavide Italiano if (!valobj_sp->GetCompilerType().IsMeaninglessWithoutDynamicResolution()) 577b3f0c340SEnrico Granata return valobj_sp->GetQualifiedTypeName(); 57878f05d35SDavide Italiano } 5795548cb50SEnrico Granata return ConstString(); 5805548cb50SEnrico Granata } 5815548cb50SEnrico Granata 582d3233c1eSEnrico Granata std::vector<lldb::LanguageType> 583*bc69dd2cSDavide Italiano FormatManager::GetCandidateLanguages(lldb::LanguageType lang_type) { 584b9c1b51eSKate Stone switch (lang_type) { 58533e97e63SEnrico Granata case lldb::eLanguageTypeC: 58633e97e63SEnrico Granata case lldb::eLanguageTypeC89: 58733e97e63SEnrico Granata case lldb::eLanguageTypeC99: 58833e97e63SEnrico Granata case lldb::eLanguageTypeC11: 58933e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus: 59033e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_03: 59133e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_11: 59233e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_14: 593170c395eSEnrico Granata return {lldb::eLanguageTypeC_plus_plus, lldb::eLanguageTypeObjC}; 594980c0484SEnrico Granata default: 595980c0484SEnrico Granata return {lang_type}; 596980c0484SEnrico Granata } 597295db41cSDavide Italiano llvm_unreachable("Fully covered switch"); 598980c0484SEnrico Granata } 599980c0484SEnrico Granata 600980c0484SEnrico Granata LanguageCategory * 601b9c1b51eSKate Stone FormatManager::GetCategoryForLanguage(lldb::LanguageType lang_type) { 60216ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 603b9c1b51eSKate Stone auto iter = m_language_categories_map.find(lang_type), 604b9c1b51eSKate Stone end = m_language_categories_map.end(); 605980c0484SEnrico Granata if (iter != end) 606980c0484SEnrico Granata return iter->second.get(); 607980c0484SEnrico Granata LanguageCategory *lang_category = new LanguageCategory(lang_type); 608b9c1b51eSKate Stone m_language_categories_map[lang_type] = 609b9c1b51eSKate Stone LanguageCategory::UniquePointer(lang_category); 610980c0484SEnrico Granata return lang_category; 611980c0484SEnrico Granata } 612980c0484SEnrico Granata 613ecd02bc1SEnrico Granata lldb::TypeFormatImplSP 614b9c1b51eSKate Stone FormatManager::GetHardcodedFormat(FormattersMatchData &match_data) { 6157cb59e1aSEnrico Granata TypeFormatImplSP retval_sp; 6167cb59e1aSEnrico Granata 617b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 618b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6198a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 6207cb59e1aSEnrico Granata break; 621ecd02bc1SEnrico Granata } 6227cb59e1aSEnrico Granata } 6237cb59e1aSEnrico Granata 6247cb59e1aSEnrico Granata return retval_sp; 625686f3debSEnrico Granata } 626686f3debSEnrico Granata 627852cc954SEnrico Granata lldb::TypeFormatImplSP 628852cc954SEnrico Granata FormatManager::GetFormat(ValueObject &valobj, 629b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 6308a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 6318a9a8f39SEnrico Granata 632852cc954SEnrico Granata TypeFormatImplSP retval; 6334edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 634b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 63563e5fb76SJonas Devlieghere LLDB_LOGF(log, 636b9c1b51eSKate Stone "\n\n[FormatManager::GetFormat] Looking into cache for type %s", 637b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 638b9c1b51eSKate Stone if (m_format_cache.GetFormat(match_data.GetTypeForCache(), retval)) { 639b9c1b51eSKate Stone if (log) { 64063e5fb76SJonas Devlieghere LLDB_LOGF( 64163e5fb76SJonas Devlieghere log, "[FormatManager::GetFormat] Cache search success. Returning."); 6426302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 643b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 644b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 64552b4b6cdSEnrico Granata } 64652b4b6cdSEnrico Granata return retval; 64752b4b6cdSEnrico Granata } 64863e5fb76SJonas Devlieghere LLDB_LOGF( 64963e5fb76SJonas Devlieghere log, 650b9c1b51eSKate Stone "[FormatManager::GetFormat] Cache search failed. Going normal route"); 65152b4b6cdSEnrico Granata } 652980c0484SEnrico Granata 6538a9a8f39SEnrico Granata retval = m_categories_map.GetFormat(match_data); 654b9c1b51eSKate Stone if (!retval) { 65563e5fb76SJonas Devlieghere LLDB_LOGF(log, 65663e5fb76SJonas Devlieghere "[FormatManager::GetFormat] Search failed. Giving language a " 657b9c1b51eSKate Stone "chance."); 658b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 659b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6608a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 661980c0484SEnrico Granata break; 662980c0484SEnrico Granata } 663980c0484SEnrico Granata } 664b9c1b51eSKate Stone if (retval) { 66563e5fb76SJonas Devlieghere LLDB_LOGF( 66663e5fb76SJonas Devlieghere log, 667b9c1b51eSKate Stone "[FormatManager::GetFormat] Language search success. Returning."); 668980c0484SEnrico Granata return retval; 669980c0484SEnrico Granata } 670980c0484SEnrico Granata } 671b9c1b51eSKate Stone if (!retval) { 67263e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetFormat] Search failed. Giving hardcoded " 673b9c1b51eSKate Stone "a chance."); 6748a9a8f39SEnrico Granata retval = GetHardcodedFormat(match_data); 675686f3debSEnrico Granata } 676d4cb1dddSEnrico Granata 677b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 67863e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetFormat] Caching %p for type %s", 679324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 6808a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 6818a9a8f39SEnrico Granata m_format_cache.SetFormat(match_data.GetTypeForCache(), retval); 68252b4b6cdSEnrico Granata } 6836302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 684b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 685852cc954SEnrico Granata return retval; 686852cc954SEnrico Granata } 687852cc954SEnrico Granata 688ecd02bc1SEnrico Granata lldb::TypeSummaryImplSP 689b9c1b51eSKate Stone FormatManager::GetHardcodedSummaryFormat(FormattersMatchData &match_data) { 6907cb59e1aSEnrico Granata TypeSummaryImplSP retval_sp; 6917cb59e1aSEnrico Granata 692b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 693b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6948a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 6957cb59e1aSEnrico Granata break; 696ecd02bc1SEnrico Granata } 6977cb59e1aSEnrico Granata } 6987cb59e1aSEnrico Granata 6997cb59e1aSEnrico Granata return retval_sp; 700686f3debSEnrico Granata } 701686f3debSEnrico Granata 7025548cb50SEnrico Granata lldb::TypeSummaryImplSP 7035548cb50SEnrico Granata FormatManager::GetSummaryFormat(ValueObject &valobj, 704b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 7058a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 7068a9a8f39SEnrico Granata 7075548cb50SEnrico Granata TypeSummaryImplSP retval; 7084edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 709b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 71063e5fb76SJonas Devlieghere LLDB_LOGF(log, 71163e5fb76SJonas Devlieghere "\n\n[FormatManager::GetSummaryFormat] Looking into cache " 712b9c1b51eSKate Stone "for type %s", 713b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 714b9c1b51eSKate Stone if (m_format_cache.GetSummary(match_data.GetTypeForCache(), retval)) { 715b9c1b51eSKate Stone if (log) { 71663e5fb76SJonas Devlieghere LLDB_LOGF(log, 71763e5fb76SJonas Devlieghere "[FormatManager::GetSummaryFormat] Cache search success. " 718b9c1b51eSKate Stone "Returning."); 7196302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 720b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 721b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 72268ae4117SEnrico Granata } 7235548cb50SEnrico Granata return retval; 72468ae4117SEnrico Granata } 72563e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Cache search failed. " 726b9c1b51eSKate Stone "Going normal route"); 7275548cb50SEnrico Granata } 728980c0484SEnrico Granata 7298a9a8f39SEnrico Granata retval = m_categories_map.GetSummaryFormat(match_data); 730b9c1b51eSKate Stone if (!retval) { 73163e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Search failed. Giving " 732b9c1b51eSKate Stone "language a chance."); 733b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 734b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 7358a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 736980c0484SEnrico Granata break; 737980c0484SEnrico Granata } 738980c0484SEnrico Granata } 739b9c1b51eSKate Stone if (retval) { 74063e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Language search " 741b9c1b51eSKate Stone "success. Returning."); 742980c0484SEnrico Granata return retval; 743980c0484SEnrico Granata } 744980c0484SEnrico Granata } 745b9c1b51eSKate Stone if (!retval) { 74663e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Search failed. Giving " 747b9c1b51eSKate Stone "hardcoded a chance."); 7488a9a8f39SEnrico Granata retval = GetHardcodedSummaryFormat(match_data); 749686f3debSEnrico Granata } 750d4cb1dddSEnrico Granata 751b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 75263e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Caching %p for type %s", 753324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 7548a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 7558a9a8f39SEnrico Granata m_format_cache.SetSummary(match_data.GetTypeForCache(), retval); 7565548cb50SEnrico Granata } 7576302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 758b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 7595548cb50SEnrico Granata return retval; 7605548cb50SEnrico Granata } 7615548cb50SEnrico Granata 762ecd02bc1SEnrico Granata lldb::SyntheticChildrenSP 763b9c1b51eSKate Stone FormatManager::GetHardcodedSyntheticChildren(FormattersMatchData &match_data) { 7647cb59e1aSEnrico Granata SyntheticChildrenSP retval_sp; 7657cb59e1aSEnrico Granata 766b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 767b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 7688a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 7697cb59e1aSEnrico Granata break; 770ecd02bc1SEnrico Granata } 7717cb59e1aSEnrico Granata } 7727cb59e1aSEnrico Granata 7737cb59e1aSEnrico Granata return retval_sp; 774686f3debSEnrico Granata } 775686f3debSEnrico Granata 7765548cb50SEnrico Granata lldb::SyntheticChildrenSP 7775548cb50SEnrico Granata FormatManager::GetSyntheticChildren(ValueObject &valobj, 778b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 7798a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 7808a9a8f39SEnrico Granata 7815548cb50SEnrico Granata SyntheticChildrenSP retval; 7824edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 783b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 78463e5fb76SJonas Devlieghere LLDB_LOGF(log, 78563e5fb76SJonas Devlieghere "\n\n[FormatManager::GetSyntheticChildren] Looking into " 786b9c1b51eSKate Stone "cache for type %s", 787b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 788b9c1b51eSKate Stone if (m_format_cache.GetSynthetic(match_data.GetTypeForCache(), retval)) { 789b9c1b51eSKate Stone if (log) { 79063e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Cache search " 791b9c1b51eSKate Stone "success. Returning."); 7926302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 7936302bf6aSPavel Labath m_format_cache.GetCacheHits(), 7946302bf6aSPavel Labath m_format_cache.GetCacheMisses()); 79568ae4117SEnrico Granata } 79668ae4117SEnrico Granata return retval; 79768ae4117SEnrico Granata } 79863e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Cache search failed. " 799b9c1b51eSKate Stone "Going normal route"); 8005548cb50SEnrico Granata } 801980c0484SEnrico Granata 8028a9a8f39SEnrico Granata retval = m_categories_map.GetSyntheticChildren(match_data); 803b9c1b51eSKate Stone if (!retval) { 80463e5fb76SJonas Devlieghere LLDB_LOGF(log, 80563e5fb76SJonas Devlieghere "[FormatManager::GetSyntheticChildren] Search failed. Giving " 806b9c1b51eSKate Stone "language a chance."); 807b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 808b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 8098a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 810980c0484SEnrico Granata break; 811980c0484SEnrico Granata } 812980c0484SEnrico Granata } 813b9c1b51eSKate Stone if (retval) { 81463e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Language search " 815b9c1b51eSKate Stone "success. Returning."); 816980c0484SEnrico Granata return retval; 817980c0484SEnrico Granata } 818980c0484SEnrico Granata } 819b9c1b51eSKate Stone if (!retval) { 82063e5fb76SJonas Devlieghere LLDB_LOGF(log, 82163e5fb76SJonas Devlieghere "[FormatManager::GetSyntheticChildren] Search failed. Giving " 822b9c1b51eSKate Stone "hardcoded a chance."); 8238a9a8f39SEnrico Granata retval = GetHardcodedSyntheticChildren(match_data); 824686f3debSEnrico Granata } 825d4cb1dddSEnrico Granata 826b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 82763e5fb76SJonas Devlieghere LLDB_LOGF(log, 828b9c1b51eSKate Stone "[FormatManager::GetSyntheticChildren] Caching %p for type %s", 829324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 8308a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 8318a9a8f39SEnrico Granata m_format_cache.SetSynthetic(match_data.GetTypeForCache(), retval); 8325548cb50SEnrico Granata } 8336302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 834b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 8355548cb50SEnrico Granata return retval; 8365548cb50SEnrico Granata } 8375548cb50SEnrico Granata 838c582713cSEnrico Granata lldb::TypeValidatorImplSP 839c582713cSEnrico Granata FormatManager::GetValidator(ValueObject &valobj, 840b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 8418a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 8428a9a8f39SEnrico Granata 843c582713cSEnrico Granata TypeValidatorImplSP retval; 8444edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 845b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 84663e5fb76SJonas Devlieghere LLDB_LOGF( 84763e5fb76SJonas Devlieghere log, "\n\n[FormatManager::GetValidator] Looking into cache for type %s", 848b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 849b9c1b51eSKate Stone if (m_format_cache.GetValidator(match_data.GetTypeForCache(), retval)) { 850b9c1b51eSKate Stone if (log) { 85163e5fb76SJonas Devlieghere LLDB_LOGF( 85263e5fb76SJonas Devlieghere log, 853b9c1b51eSKate Stone "[FormatManager::GetValidator] Cache search success. Returning."); 8546302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 855b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 856b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 857c582713cSEnrico Granata } 858c582713cSEnrico Granata return retval; 859c582713cSEnrico Granata } 86063e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Cache search failed. Going " 861b9c1b51eSKate Stone "normal route"); 862c582713cSEnrico Granata } 863980c0484SEnrico Granata 8648a9a8f39SEnrico Granata retval = m_categories_map.GetValidator(match_data); 865b9c1b51eSKate Stone if (!retval) { 86663e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Search failed. Giving " 867b9c1b51eSKate Stone "language a chance."); 868b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 869b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 8708a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 871980c0484SEnrico Granata break; 872980c0484SEnrico Granata } 873980c0484SEnrico Granata } 874b9c1b51eSKate Stone if (retval) { 87563e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Language search success. " 876b9c1b51eSKate Stone "Returning."); 877980c0484SEnrico Granata return retval; 878980c0484SEnrico Granata } 879980c0484SEnrico Granata } 880b9c1b51eSKate Stone if (!retval) { 88163e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Search failed. Giving " 882b9c1b51eSKate Stone "hardcoded a chance."); 8838a9a8f39SEnrico Granata retval = GetHardcodedValidator(match_data); 884c582713cSEnrico Granata } 885d4cb1dddSEnrico Granata 886b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 88763e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Caching %p for type %s", 888c582713cSEnrico Granata static_cast<void *>(retval.get()), 8898a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 8908a9a8f39SEnrico Granata m_format_cache.SetValidator(match_data.GetTypeForCache(), retval); 891c582713cSEnrico Granata } 8926302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 893b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 894c582713cSEnrico Granata return retval; 895c582713cSEnrico Granata } 896c582713cSEnrico Granata 897c582713cSEnrico Granata lldb::TypeValidatorImplSP 898b9c1b51eSKate Stone FormatManager::GetHardcodedValidator(FormattersMatchData &match_data) { 8997cb59e1aSEnrico Granata TypeValidatorImplSP retval_sp; 9007cb59e1aSEnrico Granata 901b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 902b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 9038a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 9047cb59e1aSEnrico Granata break; 905c582713cSEnrico Granata } 9067cb59e1aSEnrico Granata } 9077cb59e1aSEnrico Granata 9087cb59e1aSEnrico Granata return retval_sp; 909c582713cSEnrico Granata } 910c582713cSEnrico Granata 91116ff8604SSaleem Abdulrasool FormatManager::FormatManager() 912b9c1b51eSKate Stone : m_last_revision(0), m_format_cache(), m_language_categories_mutex(), 913b9c1b51eSKate Stone m_language_categories_map(), m_named_summaries_map(this), 914b9c1b51eSKate Stone m_categories_map(this), m_default_category_name(ConstString("default")), 9155548cb50SEnrico Granata m_system_category_name(ConstString("system")), 916b9c1b51eSKate Stone m_vectortypes_category_name(ConstString("VectorTypes")) { 9175548cb50SEnrico Granata LoadSystemFormatters(); 918170c395eSEnrico Granata LoadVectorFormatters(); 9195548cb50SEnrico Granata 920b9c1b51eSKate Stone EnableCategory(m_vectortypes_category_name, TypeCategoryMap::Last, 921b9c1b51eSKate Stone lldb::eLanguageTypeObjC_plus_plus); 922b9c1b51eSKate Stone EnableCategory(m_system_category_name, TypeCategoryMap::Last, 923b9c1b51eSKate Stone lldb::eLanguageTypeObjC_plus_plus); 9245548cb50SEnrico Granata } 9255548cb50SEnrico Granata 926b9c1b51eSKate Stone void FormatManager::LoadSystemFormatters() { 9275548cb50SEnrico Granata TypeSummaryImpl::Flags string_flags; 9280337c27fSEnrico Granata string_flags.SetCascades(true) 9295548cb50SEnrico Granata .SetSkipPointers(true) 9305548cb50SEnrico Granata .SetSkipReferences(false) 9315548cb50SEnrico Granata .SetDontShowChildren(true) 9325548cb50SEnrico Granata .SetDontShowValue(false) 9335548cb50SEnrico Granata .SetShowMembersOneLiner(false) 9345548cb50SEnrico Granata .SetHideItemNames(false); 9355548cb50SEnrico Granata 936bc2c2b01SEnrico Granata TypeSummaryImpl::Flags string_array_flags; 937d2911633SEnrico Granata string_array_flags.SetCascades(true) 9385548cb50SEnrico Granata .SetSkipPointers(true) 9395548cb50SEnrico Granata .SetSkipReferences(false) 940320dcf68SEnrico Granata .SetDontShowChildren(true) 9415548cb50SEnrico Granata .SetDontShowValue(true) 9425548cb50SEnrico Granata .SetShowMembersOneLiner(false) 943bc2c2b01SEnrico Granata .SetHideItemNames(false); 944bc2c2b01SEnrico Granata 945b9c1b51eSKate Stone lldb::TypeSummaryImplSP string_format( 946b9c1b51eSKate Stone new StringSummaryFormat(string_flags, "${var%s}")); 947bc2c2b01SEnrico Granata 948b9c1b51eSKate Stone lldb::TypeSummaryImplSP string_array_format( 949b9c1b51eSKate Stone new StringSummaryFormat(string_array_flags, "${var%s}")); 950bc2c2b01SEnrico Granata 9515aa1d819SJan Kratochvil RegularExpression any_size_char_arr(llvm::StringRef("char \\[[0-9]+\\]")); 9525548cb50SEnrico Granata 953b9c1b51eSKate Stone TypeCategoryImpl::SharedPointer sys_category_sp = 954b9c1b51eSKate Stone GetCategory(m_system_category_name); 9555548cb50SEnrico Granata 956b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("char *"), 957b9c1b51eSKate Stone string_format); 958b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add( 959b9c1b51eSKate Stone ConstString("unsigned char *"), string_format); 9605aa1d819SJan Kratochvil sys_category_sp->GetRegexTypeSummariesContainer()->Add( 9615aa1d819SJan Kratochvil std::move(any_size_char_arr), string_array_format); 9625548cb50SEnrico Granata 963b9c1b51eSKate Stone lldb::TypeSummaryImplSP ostype_summary( 964b9c1b51eSKate Stone new StringSummaryFormat(TypeSummaryImpl::Flags() 965b9c1b51eSKate Stone .SetCascades(false) 9665548cb50SEnrico Granata .SetSkipPointers(true) 9675548cb50SEnrico Granata .SetSkipReferences(true) 9685548cb50SEnrico Granata .SetDontShowChildren(true) 9695548cb50SEnrico Granata .SetDontShowValue(false) 9705548cb50SEnrico Granata .SetShowMembersOneLiner(false) 9715548cb50SEnrico Granata .SetHideItemNames(false), 9725548cb50SEnrico Granata "${var%O}")); 9735548cb50SEnrico Granata 974b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("OSType"), 975b9c1b51eSKate Stone ostype_summary); 9765548cb50SEnrico Granata 9774cc21772SEnrico Granata TypeFormatImpl::Flags fourchar_flags; 978b9c1b51eSKate Stone fourchar_flags.SetCascades(true).SetSkipPointers(true).SetSkipReferences( 979b9c1b51eSKate Stone true); 9804cc21772SEnrico Granata 981b9c1b51eSKate Stone AddFormat(sys_category_sp, lldb::eFormatOSType, ConstString("FourCharCode"), 982b9c1b51eSKate Stone fourchar_flags); 9835548cb50SEnrico Granata } 9845548cb50SEnrico Granata 985b9c1b51eSKate Stone void FormatManager::LoadVectorFormatters() { 986b9c1b51eSKate Stone TypeCategoryImpl::SharedPointer vectors_category_sp = 987b9c1b51eSKate Stone GetCategory(m_vectortypes_category_name); 9885548cb50SEnrico Granata 9895548cb50SEnrico Granata TypeSummaryImpl::Flags vector_flags; 9905548cb50SEnrico Granata vector_flags.SetCascades(true) 9915548cb50SEnrico Granata .SetSkipPointers(true) 9925548cb50SEnrico Granata .SetSkipReferences(false) 9935548cb50SEnrico Granata .SetDontShowChildren(true) 9945548cb50SEnrico Granata .SetDontShowValue(false) 9955548cb50SEnrico Granata .SetShowMembersOneLiner(true) 9965548cb50SEnrico Granata .SetHideItemNames(true); 9975548cb50SEnrico Granata 998b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "${var.uint128}", 999b9c1b51eSKate Stone ConstString("builtin_type_vec128"), vector_flags); 10005548cb50SEnrico Granata 1001b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("float [4]"), 10025548cb50SEnrico Granata vector_flags); 1003b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("int32_t [4]"), 10045548cb50SEnrico Granata vector_flags); 1005b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("int16_t [8]"), 10065548cb50SEnrico Granata vector_flags); 1007b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vDouble"), 10085548cb50SEnrico Granata vector_flags); 1009b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vFloat"), 10105548cb50SEnrico Granata vector_flags); 1011b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt8"), 10125548cb50SEnrico Granata vector_flags); 1013b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt16"), 10145548cb50SEnrico Granata vector_flags); 1015b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt32"), 10165548cb50SEnrico Granata vector_flags); 1017b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"), 10185548cb50SEnrico Granata vector_flags); 1019b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt8"), 10205548cb50SEnrico Granata vector_flags); 1021b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"), 10225548cb50SEnrico Granata vector_flags); 1023b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt32"), 10245548cb50SEnrico Granata vector_flags); 1025b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vBool32"), 10265548cb50SEnrico Granata vector_flags); 10275548cb50SEnrico Granata } 1028