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 241b9c1b51eSKate Stone for (lldb::LanguageType language_type : GetCandidateLanguages(valobj)) { 242b9c1b51eSKate Stone if (Language *language = Language::FindPlugin(language_type)) { 243b9c1b51eSKate Stone for (ConstString candidate : 244b9c1b51eSKate Stone language->GetPossibleFormattersMatches(valobj, use_dynamic)) { 245b9c1b51eSKate Stone entries.push_back( 246b9c1b51eSKate Stone {candidate, 247d3233c1eSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionLanguagePlugin, 248b9c1b51eSKate Stone did_strip_ptr, did_strip_ref, did_strip_typedef}); 249d3233c1eSEnrico Granata } 250d3233c1eSEnrico Granata } 251de61cecdSEnrico Granata } 252de61cecdSEnrico Granata 253de61cecdSEnrico Granata // try to strip typedef chains 254b9c1b51eSKate Stone if (compiler_type.IsTypedefType()) { 25559b5a37dSBruce Mitchener CompilerType deffed_type = compiler_type.GetTypedefedType(); 256b9c1b51eSKate Stone GetPossibleMatches( 257b9c1b51eSKate Stone valobj, deffed_type, 258de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 259b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, true); 260de61cecdSEnrico Granata } 261de61cecdSEnrico Granata 262b9c1b51eSKate Stone if (root_level) { 263de61cecdSEnrico Granata do { 26459b5a37dSBruce Mitchener if (!compiler_type.IsValid()) 265de61cecdSEnrico Granata break; 266de61cecdSEnrico Granata 267b9c1b51eSKate Stone CompilerType unqual_compiler_ast_type = 268b9c1b51eSKate Stone compiler_type.GetFullyUnqualifiedType(); 26959b5a37dSBruce Mitchener if (!unqual_compiler_ast_type.IsValid()) 270de61cecdSEnrico Granata break; 271b9c1b51eSKate Stone if (unqual_compiler_ast_type.GetOpaqueQualType() != 272b9c1b51eSKate Stone compiler_type.GetOpaqueQualType()) 273b9c1b51eSKate Stone GetPossibleMatches(valobj, unqual_compiler_ast_type, reason, 274b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 275de61cecdSEnrico Granata did_strip_typedef); 276de61cecdSEnrico Granata } while (false); 277de61cecdSEnrico Granata 278de61cecdSEnrico Granata // if all else fails, go to static type 279b9c1b51eSKate Stone if (valobj.IsDynamic()) { 280de61cecdSEnrico Granata lldb::ValueObjectSP static_value_sp(valobj.GetStaticValue()); 281de61cecdSEnrico Granata if (static_value_sp) 282b9c1b51eSKate Stone GetPossibleMatches( 283b9c1b51eSKate Stone *static_value_sp.get(), static_value_sp->GetCompilerType(), 284de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionWentToStaticValue, 285b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 286b9c1b51eSKate Stone did_strip_typedef, true); 287de61cecdSEnrico Granata } 288de61cecdSEnrico Granata } 289de61cecdSEnrico Granata } 290de61cecdSEnrico Granata 291852cc954SEnrico Granata lldb::TypeFormatImplSP 292b9c1b51eSKate Stone FormatManager::GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp) { 293852cc954SEnrico Granata if (!type_sp) 294852cc954SEnrico Granata return lldb::TypeFormatImplSP(); 295852cc954SEnrico Granata lldb::TypeFormatImplSP format_chosen_sp; 296852cc954SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 297852cc954SEnrico Granata lldb::TypeCategoryImplSP category_sp; 298852cc954SEnrico Granata uint32_t prio_category = UINT32_MAX; 299b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 300852cc954SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 301a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 302852cc954SEnrico Granata continue; 303b9c1b51eSKate Stone lldb::TypeFormatImplSP format_current_sp = 304b9c1b51eSKate Stone category_sp->GetFormatForType(type_sp); 305b9c1b51eSKate Stone if (format_current_sp && 306248a1305SKonrad Kleine (format_chosen_sp.get() == nullptr || 307b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 308852cc954SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 309852cc954SEnrico Granata format_chosen_sp = format_current_sp; 310852cc954SEnrico Granata } 311852cc954SEnrico Granata } 312852cc954SEnrico Granata return format_chosen_sp; 313852cc954SEnrico Granata } 314852cc954SEnrico Granata 3155548cb50SEnrico Granata lldb::TypeSummaryImplSP 316b9c1b51eSKate Stone FormatManager::GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp) { 3175548cb50SEnrico Granata if (!type_sp) 3185548cb50SEnrico Granata return lldb::TypeSummaryImplSP(); 3195548cb50SEnrico Granata lldb::TypeSummaryImplSP summary_chosen_sp; 3205548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3215548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3225548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 323b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3245548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 325a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3265548cb50SEnrico Granata continue; 327b9c1b51eSKate Stone lldb::TypeSummaryImplSP summary_current_sp = 328b9c1b51eSKate Stone category_sp->GetSummaryForType(type_sp); 329b9c1b51eSKate Stone if (summary_current_sp && 330248a1305SKonrad Kleine (summary_chosen_sp.get() == nullptr || 331b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3325548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3335548cb50SEnrico Granata summary_chosen_sp = summary_current_sp; 3345548cb50SEnrico Granata } 3355548cb50SEnrico Granata } 3365548cb50SEnrico Granata return summary_chosen_sp; 3375548cb50SEnrico Granata } 3385548cb50SEnrico Granata 3395548cb50SEnrico Granata lldb::TypeFilterImplSP 340b9c1b51eSKate Stone FormatManager::GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp) { 3415548cb50SEnrico Granata if (!type_sp) 3425548cb50SEnrico Granata return lldb::TypeFilterImplSP(); 3435548cb50SEnrico Granata lldb::TypeFilterImplSP filter_chosen_sp; 3445548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3455548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3465548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 347b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3485548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 349a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3505548cb50SEnrico Granata continue; 351b9c1b51eSKate Stone lldb::TypeFilterImplSP filter_current_sp( 352b9c1b51eSKate Stone (TypeFilterImpl *)category_sp->GetFilterForType(type_sp).get()); 353b9c1b51eSKate Stone if (filter_current_sp && 354248a1305SKonrad Kleine (filter_chosen_sp.get() == nullptr || 355b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3565548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3575548cb50SEnrico Granata filter_chosen_sp = filter_current_sp; 3585548cb50SEnrico Granata } 3595548cb50SEnrico Granata } 3605548cb50SEnrico Granata return filter_chosen_sp; 3615548cb50SEnrico Granata } 3625548cb50SEnrico Granata 3635548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP 364b9c1b51eSKate Stone FormatManager::GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp) { 3655548cb50SEnrico Granata if (!type_sp) 3665548cb50SEnrico Granata return lldb::ScriptedSyntheticChildrenSP(); 3675548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP synth_chosen_sp; 3685548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3695548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3705548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 371b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3725548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 373a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3745548cb50SEnrico Granata continue; 375b9c1b51eSKate Stone lldb::ScriptedSyntheticChildrenSP synth_current_sp( 376b9c1b51eSKate Stone (ScriptedSyntheticChildren *)category_sp->GetSyntheticForType(type_sp) 377b9c1b51eSKate Stone .get()); 378b9c1b51eSKate Stone if (synth_current_sp && 379248a1305SKonrad Kleine (synth_chosen_sp.get() == nullptr || 380b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3815548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3825548cb50SEnrico Granata synth_chosen_sp = synth_current_sp; 3835548cb50SEnrico Granata } 3845548cb50SEnrico Granata } 3855548cb50SEnrico Granata return synth_chosen_sp; 3865548cb50SEnrico Granata } 3875548cb50SEnrico Granata 388c582713cSEnrico Granata lldb::TypeValidatorImplSP 389b9c1b51eSKate Stone FormatManager::GetValidatorForType(lldb::TypeNameSpecifierImplSP type_sp) { 390c582713cSEnrico Granata if (!type_sp) 391c582713cSEnrico Granata return lldb::TypeValidatorImplSP(); 392c582713cSEnrico Granata lldb::TypeValidatorImplSP validator_chosen_sp; 393c582713cSEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 394c582713cSEnrico Granata lldb::TypeCategoryImplSP category_sp; 395c582713cSEnrico Granata uint32_t prio_category = UINT32_MAX; 396b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 397c582713cSEnrico Granata category_sp = GetCategoryAtIndex(category_id); 398a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 399c582713cSEnrico Granata continue; 400b9c1b51eSKate Stone lldb::TypeValidatorImplSP validator_current_sp( 401b9c1b51eSKate Stone category_sp->GetValidatorForType(type_sp).get()); 402b9c1b51eSKate Stone if (validator_current_sp && 403248a1305SKonrad Kleine (validator_chosen_sp.get() == nullptr || 404b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 405c582713cSEnrico Granata prio_category = category_sp->GetEnabledPosition(); 406c582713cSEnrico Granata validator_chosen_sp = validator_current_sp; 407c582713cSEnrico Granata } 408c582713cSEnrico Granata } 409c582713cSEnrico Granata return validator_chosen_sp; 410c582713cSEnrico Granata } 411c582713cSEnrico Granata 412b9c1b51eSKate Stone void FormatManager::ForEachCategory(TypeCategoryMap::ForEachCallback callback) { 413b56d0103SEnrico Granata m_categories_map.ForEach(callback); 41416ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 415b9c1b51eSKate Stone for (const auto &entry : m_language_categories_map) { 416b9c1b51eSKate Stone if (auto category_sp = entry.second->GetCategory()) { 417b56d0103SEnrico Granata if (!callback(category_sp)) 418b56d0103SEnrico Granata break; 419b56d0103SEnrico Granata } 420b56d0103SEnrico Granata } 421b56d0103SEnrico Granata } 422b56d0103SEnrico Granata 4235548cb50SEnrico Granata lldb::TypeCategoryImplSP 4240e4c4821SAdrian Prantl FormatManager::GetCategory(ConstString category_name, bool can_create) { 4255548cb50SEnrico Granata if (!category_name) 4265548cb50SEnrico Granata return GetCategory(m_default_category_name); 4275548cb50SEnrico Granata lldb::TypeCategoryImplSP category; 4285548cb50SEnrico Granata if (m_categories_map.Get(category_name, category)) 4295548cb50SEnrico Granata return category; 4305548cb50SEnrico Granata 4315548cb50SEnrico Granata if (!can_create) 4325548cb50SEnrico Granata return lldb::TypeCategoryImplSP(); 4335548cb50SEnrico Granata 434b9c1b51eSKate Stone m_categories_map.Add( 435b9c1b51eSKate Stone category_name, 436b9c1b51eSKate Stone lldb::TypeCategoryImplSP(new TypeCategoryImpl(this, category_name))); 4375548cb50SEnrico Granata return GetCategory(category_name); 4385548cb50SEnrico Granata } 4395548cb50SEnrico Granata 440b9c1b51eSKate Stone lldb::Format FormatManager::GetSingleItemFormat(lldb::Format vector_format) { 441b9c1b51eSKate Stone switch (vector_format) { 4425548cb50SEnrico Granata case eFormatVectorOfChar: 4435548cb50SEnrico Granata return eFormatCharArray; 4445548cb50SEnrico Granata 4455548cb50SEnrico Granata case eFormatVectorOfSInt8: 4465548cb50SEnrico Granata case eFormatVectorOfSInt16: 4475548cb50SEnrico Granata case eFormatVectorOfSInt32: 4485548cb50SEnrico Granata case eFormatVectorOfSInt64: 4495548cb50SEnrico Granata return eFormatDecimal; 4505548cb50SEnrico Granata 4515548cb50SEnrico Granata case eFormatVectorOfUInt8: 4525548cb50SEnrico Granata case eFormatVectorOfUInt16: 4535548cb50SEnrico Granata case eFormatVectorOfUInt32: 4545548cb50SEnrico Granata case eFormatVectorOfUInt64: 4555548cb50SEnrico Granata case eFormatVectorOfUInt128: 4565548cb50SEnrico Granata return eFormatHex; 4575548cb50SEnrico Granata 458a0f08674SEwan Crawford case eFormatVectorOfFloat16: 4595548cb50SEnrico Granata case eFormatVectorOfFloat32: 4605548cb50SEnrico Granata case eFormatVectorOfFloat64: 4615548cb50SEnrico Granata return eFormatFloat; 4625548cb50SEnrico Granata 4635548cb50SEnrico Granata default: 4645548cb50SEnrico Granata return lldb::eFormatInvalid; 4655548cb50SEnrico Granata } 4665548cb50SEnrico Granata } 4675548cb50SEnrico Granata 468b9c1b51eSKate Stone bool FormatManager::ShouldPrintAsOneLiner(ValueObject &valobj) { 469553fad5cSEnrico Granata // if settings say no oneline whatsoever 470b9c1b51eSKate Stone if (valobj.GetTargetSP().get() && 471a6682a41SJonas Devlieghere !valobj.GetTargetSP()->GetDebugger().GetAutoOneLineSummaries()) 472553fad5cSEnrico Granata return false; // then don't oneline 473553fad5cSEnrico Granata 47442fa4af8SEnrico Granata // if this object has a summary, then ask the summary 475a29cb0baSEnrico Granata if (valobj.GetSummaryFormat().get() != nullptr) 47642fa4af8SEnrico Granata return valobj.GetSummaryFormat()->IsOneLiner(); 477a29cb0baSEnrico Granata 478a29cb0baSEnrico Granata // no children, no party 479a29cb0baSEnrico Granata if (valobj.GetNumChildren() == 0) 480a29cb0baSEnrico Granata return false; 481a29cb0baSEnrico Granata 48205097246SAdrian Prantl // ask the type if it has any opinion about this eLazyBoolCalculate == no 48305097246SAdrian Prantl // opinion; other values should be self explanatory 4849c63f99aSEnrico Granata CompilerType compiler_type(valobj.GetCompilerType()); 485b9c1b51eSKate Stone if (compiler_type.IsValid()) { 486b9c1b51eSKate Stone switch (compiler_type.ShouldPrintAsOneLiner(&valobj)) { 4879c63f99aSEnrico Granata case eLazyBoolNo: 4889c63f99aSEnrico Granata return false; 4899c63f99aSEnrico Granata case eLazyBoolYes: 4909c63f99aSEnrico Granata return true; 4919c63f99aSEnrico Granata case eLazyBoolCalculate: 4929c63f99aSEnrico Granata break; 4939c63f99aSEnrico Granata } 4949c63f99aSEnrico Granata } 4959c63f99aSEnrico Granata 496a29cb0baSEnrico Granata size_t total_children_name_len = 0; 497a29cb0baSEnrico Granata 498b9c1b51eSKate Stone for (size_t idx = 0; idx < valobj.GetNumChildren(); idx++) { 499ddac7611SEnrico Granata bool is_synth_val = false; 500a29cb0baSEnrico Granata ValueObjectSP child_sp(valobj.GetChildAtIndex(idx, true)); 501a29cb0baSEnrico Granata // something is wrong here - bail out 502a29cb0baSEnrico Granata if (!child_sp) 503a29cb0baSEnrico Granata return false; 5046500061eSEnrico Granata 5056500061eSEnrico Granata // also ask the child's type if it has any opinion 5066500061eSEnrico Granata CompilerType child_compiler_type(child_sp->GetCompilerType()); 507b9c1b51eSKate Stone if (child_compiler_type.IsValid()) { 508b9c1b51eSKate Stone switch (child_compiler_type.ShouldPrintAsOneLiner(child_sp.get())) { 5096500061eSEnrico Granata case eLazyBoolYes: 5106500061eSEnrico Granata // an opinion of yes is only binding for the child, so keep going 5116500061eSEnrico Granata case eLazyBoolCalculate: 5126500061eSEnrico Granata break; 5136500061eSEnrico Granata case eLazyBoolNo: 5146500061eSEnrico Granata // but if the child says no, then it's a veto on the whole thing 5156500061eSEnrico Granata return false; 5166500061eSEnrico Granata } 5176500061eSEnrico Granata } 5186500061eSEnrico Granata 519b9c1b51eSKate Stone // if we decided to define synthetic children for a type, we probably care 52005097246SAdrian Prantl // enough to show them, but avoid nesting children in children 521b9c1b51eSKate Stone if (child_sp->GetSyntheticChildren().get() != nullptr) { 522ddac7611SEnrico Granata ValueObjectSP synth_sp(child_sp->GetSyntheticValue()); 523ddac7611SEnrico Granata // wait.. wat? just get out of here.. 524ddac7611SEnrico Granata if (!synth_sp) 525a29cb0baSEnrico Granata return false; 526ddac7611SEnrico Granata // but if we only have them to provide a value, keep going 527a6682a41SJonas Devlieghere if (!synth_sp->MightHaveChildren() && 528b9c1b51eSKate Stone synth_sp->DoesProvideSyntheticValue()) 529ddac7611SEnrico Granata is_synth_val = true; 530ddac7611SEnrico Granata else 531ddac7611SEnrico Granata return false; 532ddac7611SEnrico Granata } 533a29cb0baSEnrico Granata 534a29cb0baSEnrico Granata total_children_name_len += child_sp->GetName().GetLength(); 535a29cb0baSEnrico Granata 536a29cb0baSEnrico Granata // 50 itself is a "randomly" chosen number - the idea is that 537a29cb0baSEnrico Granata // overly long structs should not get this treatment 538a29cb0baSEnrico Granata // FIXME: maybe make this a user-tweakable setting? 539a29cb0baSEnrico Granata if (total_children_name_len > 50) 540a29cb0baSEnrico Granata return false; 541a29cb0baSEnrico Granata 542a29cb0baSEnrico Granata // if a summary is there.. 543b9c1b51eSKate Stone if (child_sp->GetSummaryFormat()) { 544a29cb0baSEnrico Granata // and it wants children, then bail out 5458a068e6cSEnrico Granata if (child_sp->GetSummaryFormat()->DoesPrintChildren(child_sp.get())) 546a29cb0baSEnrico Granata return false; 547a29cb0baSEnrico Granata } 548a29cb0baSEnrico Granata 549c89e4ca3SEnrico Granata // if this child has children.. 550b9c1b51eSKate Stone if (child_sp->GetNumChildren()) { 551a29cb0baSEnrico Granata // ...and no summary... 552b9c1b51eSKate Stone // (if it had a summary and the summary wanted children, we would have 553b9c1b51eSKate Stone // bailed out anyway 554b9c1b51eSKate Stone // so this only makes us bail out if this has no summary and we would 555b9c1b51eSKate Stone // then print children) 556b9c1b51eSKate Stone if (!child_sp->GetSummaryFormat() && !is_synth_val) // but again only do 557b9c1b51eSKate Stone // that if not a 558b9c1b51eSKate Stone // synthetic valued 559b9c1b51eSKate Stone // child 560a29cb0baSEnrico Granata return false; // then bail out 561a29cb0baSEnrico Granata } 562a29cb0baSEnrico Granata } 563a29cb0baSEnrico Granata return true; 564a29cb0baSEnrico Granata } 565a29cb0baSEnrico Granata 5660e4c4821SAdrian Prantl ConstString FormatManager::GetValidTypeName(ConstString type) { 5675548cb50SEnrico Granata return ::GetValidTypeName_Impl(type); 5685548cb50SEnrico Granata } 5695548cb50SEnrico Granata 570b9c1b51eSKate Stone ConstString FormatManager::GetTypeForCache(ValueObject &valobj, 571b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 572b9c1b51eSKate Stone ValueObjectSP valobj_sp = valobj.GetQualifiedRepresentationIfAvailable( 573b9c1b51eSKate Stone use_dynamic, valobj.IsSynthetic()); 57478f05d35SDavide Italiano if (valobj_sp && valobj_sp->GetCompilerType().IsValid()) { 57578f05d35SDavide Italiano if (!valobj_sp->GetCompilerType().IsMeaninglessWithoutDynamicResolution()) 576b3f0c340SEnrico Granata return valobj_sp->GetQualifiedTypeName(); 57778f05d35SDavide Italiano } 5785548cb50SEnrico Granata return ConstString(); 5795548cb50SEnrico Granata } 5805548cb50SEnrico Granata 581d3233c1eSEnrico Granata std::vector<lldb::LanguageType> 582b9c1b51eSKate Stone FormatManager::GetCandidateLanguages(ValueObject &valobj) { 583980c0484SEnrico Granata lldb::LanguageType lang_type = valobj.GetObjectRuntimeLanguage(); 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 } 597*295db41cSDavide 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