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 33b9c1b51eSKate Stone static 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"}, 72b9c1b51eSKate Stone {eFormatVoid, 'v', "void"}}; 735548cb50SEnrico Granata 7428606954SSaleem Abdulrasool static uint32_t g_num_format_infos = llvm::array_lengthof(g_format_infos); 755548cb50SEnrico Granata 76b9c1b51eSKate Stone static bool GetFormatFromFormatChar(char format_char, Format &format) { 77b9c1b51eSKate Stone for (uint32_t i = 0; i < g_num_format_infos; ++i) { 78b9c1b51eSKate Stone if (g_format_infos[i].format_char == format_char) { 795548cb50SEnrico Granata format = g_format_infos[i].format; 805548cb50SEnrico Granata return true; 815548cb50SEnrico Granata } 825548cb50SEnrico Granata } 835548cb50SEnrico Granata format = eFormatInvalid; 845548cb50SEnrico Granata return false; 855548cb50SEnrico Granata } 865548cb50SEnrico Granata 87b9c1b51eSKate Stone static bool GetFormatFromFormatName(const char *format_name, 88b9c1b51eSKate Stone bool partial_match_ok, Format &format) { 895548cb50SEnrico Granata uint32_t i; 90b9c1b51eSKate Stone for (i = 0; i < g_num_format_infos; ++i) { 91b9c1b51eSKate Stone if (strcasecmp(g_format_infos[i].format_name, format_name) == 0) { 925548cb50SEnrico Granata format = g_format_infos[i].format; 935548cb50SEnrico Granata return true; 945548cb50SEnrico Granata } 955548cb50SEnrico Granata } 965548cb50SEnrico Granata 97b9c1b51eSKate Stone if (partial_match_ok) { 98b9c1b51eSKate Stone for (i = 0; i < g_num_format_infos; ++i) { 99b9c1b51eSKate Stone if (strcasestr(g_format_infos[i].format_name, format_name) == 100b9c1b51eSKate Stone g_format_infos[i].format_name) { 1015548cb50SEnrico Granata format = g_format_infos[i].format; 1025548cb50SEnrico Granata return true; 1035548cb50SEnrico Granata } 1045548cb50SEnrico Granata } 1055548cb50SEnrico Granata } 1065548cb50SEnrico Granata format = eFormatInvalid; 1075548cb50SEnrico Granata return false; 1085548cb50SEnrico Granata } 1095548cb50SEnrico Granata 110b9c1b51eSKate Stone void FormatManager::Changed() { 111bd5eab82SEnrico Granata ++m_last_revision; 112bd5eab82SEnrico Granata m_format_cache.Clear(); 11316ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 114b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 115bd5eab82SEnrico Granata if (iter.second) 116bd5eab82SEnrico Granata iter.second->GetFormatCache().Clear(); 117bd5eab82SEnrico Granata } 118bd5eab82SEnrico Granata } 119bd5eab82SEnrico Granata 120b9c1b51eSKate Stone bool FormatManager::GetFormatFromCString(const char *format_cstr, 1215548cb50SEnrico Granata bool partial_match_ok, 122b9c1b51eSKate Stone lldb::Format &format) { 1235548cb50SEnrico Granata bool success = false; 124b9c1b51eSKate Stone if (format_cstr && format_cstr[0]) { 125b9c1b51eSKate Stone if (format_cstr[1] == '\0') { 1265548cb50SEnrico Granata success = GetFormatFromFormatChar(format_cstr[0], format); 1275548cb50SEnrico Granata if (success) 1285548cb50SEnrico Granata return true; 1295548cb50SEnrico Granata } 1305548cb50SEnrico Granata 1315548cb50SEnrico Granata success = GetFormatFromFormatName(format_cstr, partial_match_ok, format); 1325548cb50SEnrico Granata } 1335548cb50SEnrico Granata if (!success) 1345548cb50SEnrico Granata format = eFormatInvalid; 1355548cb50SEnrico Granata return success; 1365548cb50SEnrico Granata } 1375548cb50SEnrico Granata 138b9c1b51eSKate Stone char FormatManager::GetFormatAsFormatChar(lldb::Format format) { 139b9c1b51eSKate Stone for (uint32_t i = 0; i < g_num_format_infos; ++i) { 1405548cb50SEnrico Granata if (g_format_infos[i].format == format) 1415548cb50SEnrico Granata return g_format_infos[i].format_char; 1425548cb50SEnrico Granata } 1435548cb50SEnrico Granata return '\0'; 1445548cb50SEnrico Granata } 1455548cb50SEnrico Granata 146b9c1b51eSKate Stone const char *FormatManager::GetFormatAsCString(Format format) { 1475548cb50SEnrico Granata if (format >= eFormatDefault && format < kNumFormats) 1485548cb50SEnrico Granata return g_format_infos[format].format_name; 149248a1305SKonrad Kleine return nullptr; 1505548cb50SEnrico Granata } 1515548cb50SEnrico Granata 152b9c1b51eSKate Stone void FormatManager::EnableAllCategories() { 15333e97e63SEnrico Granata m_categories_map.EnableAllCategories(); 15416ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 155b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 15633e97e63SEnrico Granata if (iter.second) 15733e97e63SEnrico Granata iter.second->Enable(); 15833e97e63SEnrico Granata } 15933e97e63SEnrico Granata } 16033e97e63SEnrico Granata 161b9c1b51eSKate Stone void FormatManager::DisableAllCategories() { 16233e97e63SEnrico Granata m_categories_map.DisableAllCategories(); 16316ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 164b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 16533e97e63SEnrico Granata if (iter.second) 16633e97e63SEnrico Granata iter.second->Disable(); 16733e97e63SEnrico Granata } 16833e97e63SEnrico Granata } 16933e97e63SEnrico Granata 170b9c1b51eSKate Stone void FormatManager::GetPossibleMatches( 171b9c1b51eSKate Stone ValueObject &valobj, CompilerType compiler_type, uint32_t reason, 172b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic, FormattersMatchVector &entries, 173b9c1b51eSKate Stone bool did_strip_ptr, bool did_strip_ref, bool did_strip_typedef, 174b9c1b51eSKate Stone bool root_level) { 175c6bf2e2dSEnrico Granata compiler_type = compiler_type.GetTypeForFormatters(); 17659b5a37dSBruce Mitchener ConstString type_name(compiler_type.GetConstTypeName()); 177b9c1b51eSKate Stone if (valobj.GetBitfieldBitSize() > 0) { 178de61cecdSEnrico Granata StreamString sstring; 179de61cecdSEnrico Granata sstring.Printf("%s:%d", type_name.AsCString(), valobj.GetBitfieldBitSize()); 180c156427dSZachary Turner ConstString bitfieldname(sstring.GetString()); 181b9c1b51eSKate Stone entries.push_back( 182b9c1b51eSKate Stone {bitfieldname, 0, did_strip_ptr, did_strip_ref, did_strip_typedef}); 183de61cecdSEnrico Granata reason |= lldb_private::eFormatterChoiceCriterionStrippedBitField; 184de61cecdSEnrico Granata } 185b3f0c340SEnrico Granata 186b9c1b51eSKate Stone if (!compiler_type.IsMeaninglessWithoutDynamicResolution()) { 187b9c1b51eSKate Stone entries.push_back( 188b9c1b51eSKate Stone {type_name, reason, did_strip_ptr, did_strip_ref, did_strip_typedef}); 189de61cecdSEnrico Granata 19059b5a37dSBruce Mitchener ConstString display_type_name(compiler_type.GetDisplayTypeName()); 191e8daa2f8SEnrico Granata if (display_type_name != type_name) 192b9c1b51eSKate Stone entries.push_back({display_type_name, reason, did_strip_ptr, 193b9c1b51eSKate Stone did_strip_ref, did_strip_typedef}); 194b3f0c340SEnrico Granata } 195e8daa2f8SEnrico Granata 196b9c1b51eSKate Stone for (bool is_rvalue_ref = true, j = true; 197b9c1b51eSKate Stone j && compiler_type.IsReferenceType(nullptr, &is_rvalue_ref); j = false) { 19859b5a37dSBruce Mitchener CompilerType non_ref_type = compiler_type.GetNonReferenceType(); 199b9c1b51eSKate Stone GetPossibleMatches( 200b9c1b51eSKate Stone valobj, non_ref_type, 201b9c1b51eSKate Stone reason | 202b9c1b51eSKate Stone lldb_private::eFormatterChoiceCriterionStrippedPointerReference, 203b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, true, did_strip_typedef); 204b9c1b51eSKate Stone if (non_ref_type.IsTypedefType()) { 205a1e5dc86SGreg Clayton CompilerType deffed_referenced_type = non_ref_type.GetTypedefedType(); 206b9c1b51eSKate Stone deffed_referenced_type = 207b9c1b51eSKate Stone is_rvalue_ref ? deffed_referenced_type.GetRValueReferenceType() 208b9c1b51eSKate Stone : deffed_referenced_type.GetLValueReferenceType(); 209b9c1b51eSKate Stone GetPossibleMatches( 210b9c1b51eSKate Stone valobj, deffed_referenced_type, 2111ac62963SEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 212b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 2131ac62963SEnrico Granata true); // this is not exactly the usual meaning of stripping typedefs 214de61cecdSEnrico Granata } 2151ac62963SEnrico Granata } 2161ac62963SEnrico Granata 217b9c1b51eSKate Stone if (compiler_type.IsPointerType()) { 21859b5a37dSBruce Mitchener CompilerType non_ptr_type = compiler_type.GetPointeeType(); 219b9c1b51eSKate Stone GetPossibleMatches( 220b9c1b51eSKate Stone valobj, non_ptr_type, 221b9c1b51eSKate Stone reason | 222b9c1b51eSKate Stone lldb_private::eFormatterChoiceCriterionStrippedPointerReference, 223b9c1b51eSKate Stone use_dynamic, entries, true, did_strip_ref, did_strip_typedef); 224b9c1b51eSKate Stone if (non_ptr_type.IsTypedefType()) { 225b9c1b51eSKate Stone CompilerType deffed_pointed_type = 226b9c1b51eSKate Stone non_ptr_type.GetTypedefedType().GetPointerType(); 227b9c1b51eSKate Stone GetPossibleMatches( 228b9c1b51eSKate Stone valobj, deffed_pointed_type, 2291ac62963SEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 230b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 2311ac62963SEnrico Granata true); // this is not exactly the usual meaning of stripping typedefs 2321ac62963SEnrico Granata } 233de61cecdSEnrico Granata } 234de61cecdSEnrico Granata 235b9c1b51eSKate Stone for (lldb::LanguageType language_type : GetCandidateLanguages(valobj)) { 236b9c1b51eSKate Stone if (Language *language = Language::FindPlugin(language_type)) { 237b9c1b51eSKate Stone for (ConstString candidate : 238b9c1b51eSKate Stone language->GetPossibleFormattersMatches(valobj, use_dynamic)) { 239b9c1b51eSKate Stone entries.push_back( 240b9c1b51eSKate Stone {candidate, 241d3233c1eSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionLanguagePlugin, 242b9c1b51eSKate Stone did_strip_ptr, did_strip_ref, did_strip_typedef}); 243d3233c1eSEnrico Granata } 244d3233c1eSEnrico Granata } 245de61cecdSEnrico Granata } 246de61cecdSEnrico Granata 247de61cecdSEnrico Granata // try to strip typedef chains 248b9c1b51eSKate Stone if (compiler_type.IsTypedefType()) { 24959b5a37dSBruce Mitchener CompilerType deffed_type = compiler_type.GetTypedefedType(); 250b9c1b51eSKate Stone GetPossibleMatches( 251b9c1b51eSKate Stone valobj, deffed_type, 252de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 253b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, true); 254de61cecdSEnrico Granata } 255de61cecdSEnrico Granata 256b9c1b51eSKate Stone if (root_level) { 257de61cecdSEnrico Granata do { 25859b5a37dSBruce Mitchener if (!compiler_type.IsValid()) 259de61cecdSEnrico Granata break; 260de61cecdSEnrico Granata 261b9c1b51eSKate Stone CompilerType unqual_compiler_ast_type = 262b9c1b51eSKate Stone compiler_type.GetFullyUnqualifiedType(); 26359b5a37dSBruce Mitchener if (!unqual_compiler_ast_type.IsValid()) 264de61cecdSEnrico Granata break; 265b9c1b51eSKate Stone if (unqual_compiler_ast_type.GetOpaqueQualType() != 266b9c1b51eSKate Stone compiler_type.GetOpaqueQualType()) 267b9c1b51eSKate Stone GetPossibleMatches(valobj, unqual_compiler_ast_type, reason, 268b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 269de61cecdSEnrico Granata did_strip_typedef); 270de61cecdSEnrico Granata } while (false); 271de61cecdSEnrico Granata 272de61cecdSEnrico Granata // if all else fails, go to static type 273b9c1b51eSKate Stone if (valobj.IsDynamic()) { 274de61cecdSEnrico Granata lldb::ValueObjectSP static_value_sp(valobj.GetStaticValue()); 275de61cecdSEnrico Granata if (static_value_sp) 276b9c1b51eSKate Stone GetPossibleMatches( 277b9c1b51eSKate Stone *static_value_sp.get(), static_value_sp->GetCompilerType(), 278de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionWentToStaticValue, 279b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 280b9c1b51eSKate Stone did_strip_typedef, true); 281de61cecdSEnrico Granata } 282de61cecdSEnrico Granata } 283de61cecdSEnrico Granata } 284de61cecdSEnrico Granata 285852cc954SEnrico Granata lldb::TypeFormatImplSP 286b9c1b51eSKate Stone FormatManager::GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp) { 287852cc954SEnrico Granata if (!type_sp) 288852cc954SEnrico Granata return lldb::TypeFormatImplSP(); 289852cc954SEnrico Granata lldb::TypeFormatImplSP format_chosen_sp; 290852cc954SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 291852cc954SEnrico Granata lldb::TypeCategoryImplSP category_sp; 292852cc954SEnrico Granata uint32_t prio_category = UINT32_MAX; 293b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 294852cc954SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 295a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 296852cc954SEnrico Granata continue; 297b9c1b51eSKate Stone lldb::TypeFormatImplSP format_current_sp = 298b9c1b51eSKate Stone category_sp->GetFormatForType(type_sp); 299b9c1b51eSKate Stone if (format_current_sp && 300248a1305SKonrad Kleine (format_chosen_sp.get() == nullptr || 301b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 302852cc954SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 303852cc954SEnrico Granata format_chosen_sp = format_current_sp; 304852cc954SEnrico Granata } 305852cc954SEnrico Granata } 306852cc954SEnrico Granata return format_chosen_sp; 307852cc954SEnrico Granata } 308852cc954SEnrico Granata 3095548cb50SEnrico Granata lldb::TypeSummaryImplSP 310b9c1b51eSKate Stone FormatManager::GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp) { 3115548cb50SEnrico Granata if (!type_sp) 3125548cb50SEnrico Granata return lldb::TypeSummaryImplSP(); 3135548cb50SEnrico Granata lldb::TypeSummaryImplSP summary_chosen_sp; 3145548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3155548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3165548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 317b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3185548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 319a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3205548cb50SEnrico Granata continue; 321b9c1b51eSKate Stone lldb::TypeSummaryImplSP summary_current_sp = 322b9c1b51eSKate Stone category_sp->GetSummaryForType(type_sp); 323b9c1b51eSKate Stone if (summary_current_sp && 324248a1305SKonrad Kleine (summary_chosen_sp.get() == nullptr || 325b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3265548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3275548cb50SEnrico Granata summary_chosen_sp = summary_current_sp; 3285548cb50SEnrico Granata } 3295548cb50SEnrico Granata } 3305548cb50SEnrico Granata return summary_chosen_sp; 3315548cb50SEnrico Granata } 3325548cb50SEnrico Granata 3335548cb50SEnrico Granata lldb::TypeFilterImplSP 334b9c1b51eSKate Stone FormatManager::GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp) { 3355548cb50SEnrico Granata if (!type_sp) 3365548cb50SEnrico Granata return lldb::TypeFilterImplSP(); 3375548cb50SEnrico Granata lldb::TypeFilterImplSP filter_chosen_sp; 3385548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3395548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3405548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 341b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3425548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 343a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3445548cb50SEnrico Granata continue; 345b9c1b51eSKate Stone lldb::TypeFilterImplSP filter_current_sp( 346b9c1b51eSKate Stone (TypeFilterImpl *)category_sp->GetFilterForType(type_sp).get()); 347b9c1b51eSKate Stone if (filter_current_sp && 348248a1305SKonrad Kleine (filter_chosen_sp.get() == nullptr || 349b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3505548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3515548cb50SEnrico Granata filter_chosen_sp = filter_current_sp; 3525548cb50SEnrico Granata } 3535548cb50SEnrico Granata } 3545548cb50SEnrico Granata return filter_chosen_sp; 3555548cb50SEnrico Granata } 3565548cb50SEnrico Granata 3575548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP 358b9c1b51eSKate Stone FormatManager::GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp) { 3595548cb50SEnrico Granata if (!type_sp) 3605548cb50SEnrico Granata return lldb::ScriptedSyntheticChildrenSP(); 3615548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP synth_chosen_sp; 3625548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3635548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3645548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 365b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3665548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 367a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3685548cb50SEnrico Granata continue; 369b9c1b51eSKate Stone lldb::ScriptedSyntheticChildrenSP synth_current_sp( 370b9c1b51eSKate Stone (ScriptedSyntheticChildren *)category_sp->GetSyntheticForType(type_sp) 371b9c1b51eSKate Stone .get()); 372b9c1b51eSKate Stone if (synth_current_sp && 373248a1305SKonrad Kleine (synth_chosen_sp.get() == nullptr || 374b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3755548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3765548cb50SEnrico Granata synth_chosen_sp = synth_current_sp; 3775548cb50SEnrico Granata } 3785548cb50SEnrico Granata } 3795548cb50SEnrico Granata return synth_chosen_sp; 3805548cb50SEnrico Granata } 3815548cb50SEnrico Granata 382c582713cSEnrico Granata lldb::TypeValidatorImplSP 383b9c1b51eSKate Stone FormatManager::GetValidatorForType(lldb::TypeNameSpecifierImplSP type_sp) { 384c582713cSEnrico Granata if (!type_sp) 385c582713cSEnrico Granata return lldb::TypeValidatorImplSP(); 386c582713cSEnrico Granata lldb::TypeValidatorImplSP validator_chosen_sp; 387c582713cSEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 388c582713cSEnrico Granata lldb::TypeCategoryImplSP category_sp; 389c582713cSEnrico Granata uint32_t prio_category = UINT32_MAX; 390b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 391c582713cSEnrico Granata category_sp = GetCategoryAtIndex(category_id); 392a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 393c582713cSEnrico Granata continue; 394b9c1b51eSKate Stone lldb::TypeValidatorImplSP validator_current_sp( 395b9c1b51eSKate Stone category_sp->GetValidatorForType(type_sp).get()); 396b9c1b51eSKate Stone if (validator_current_sp && 397248a1305SKonrad Kleine (validator_chosen_sp.get() == nullptr || 398b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 399c582713cSEnrico Granata prio_category = category_sp->GetEnabledPosition(); 400c582713cSEnrico Granata validator_chosen_sp = validator_current_sp; 401c582713cSEnrico Granata } 402c582713cSEnrico Granata } 403c582713cSEnrico Granata return validator_chosen_sp; 404c582713cSEnrico Granata } 405c582713cSEnrico Granata 406b9c1b51eSKate Stone void FormatManager::ForEachCategory(TypeCategoryMap::ForEachCallback callback) { 407b56d0103SEnrico Granata m_categories_map.ForEach(callback); 40816ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 409b9c1b51eSKate Stone for (const auto &entry : m_language_categories_map) { 410b9c1b51eSKate Stone if (auto category_sp = entry.second->GetCategory()) { 411b56d0103SEnrico Granata if (!callback(category_sp)) 412b56d0103SEnrico Granata break; 413b56d0103SEnrico Granata } 414b56d0103SEnrico Granata } 415b56d0103SEnrico Granata } 416b56d0103SEnrico Granata 4175548cb50SEnrico Granata lldb::TypeCategoryImplSP 4180e4c4821SAdrian Prantl FormatManager::GetCategory(ConstString category_name, bool can_create) { 4195548cb50SEnrico Granata if (!category_name) 4205548cb50SEnrico Granata return GetCategory(m_default_category_name); 4215548cb50SEnrico Granata lldb::TypeCategoryImplSP category; 4225548cb50SEnrico Granata if (m_categories_map.Get(category_name, category)) 4235548cb50SEnrico Granata return category; 4245548cb50SEnrico Granata 4255548cb50SEnrico Granata if (!can_create) 4265548cb50SEnrico Granata return lldb::TypeCategoryImplSP(); 4275548cb50SEnrico Granata 428b9c1b51eSKate Stone m_categories_map.Add( 429b9c1b51eSKate Stone category_name, 430b9c1b51eSKate Stone lldb::TypeCategoryImplSP(new TypeCategoryImpl(this, category_name))); 4315548cb50SEnrico Granata return GetCategory(category_name); 4325548cb50SEnrico Granata } 4335548cb50SEnrico Granata 434b9c1b51eSKate Stone lldb::Format FormatManager::GetSingleItemFormat(lldb::Format vector_format) { 435b9c1b51eSKate Stone switch (vector_format) { 4365548cb50SEnrico Granata case eFormatVectorOfChar: 4375548cb50SEnrico Granata return eFormatCharArray; 4385548cb50SEnrico Granata 4395548cb50SEnrico Granata case eFormatVectorOfSInt8: 4405548cb50SEnrico Granata case eFormatVectorOfSInt16: 4415548cb50SEnrico Granata case eFormatVectorOfSInt32: 4425548cb50SEnrico Granata case eFormatVectorOfSInt64: 4435548cb50SEnrico Granata return eFormatDecimal; 4445548cb50SEnrico Granata 4455548cb50SEnrico Granata case eFormatVectorOfUInt8: 4465548cb50SEnrico Granata case eFormatVectorOfUInt16: 4475548cb50SEnrico Granata case eFormatVectorOfUInt32: 4485548cb50SEnrico Granata case eFormatVectorOfUInt64: 4495548cb50SEnrico Granata case eFormatVectorOfUInt128: 4505548cb50SEnrico Granata return eFormatHex; 4515548cb50SEnrico Granata 452a0f08674SEwan Crawford case eFormatVectorOfFloat16: 4535548cb50SEnrico Granata case eFormatVectorOfFloat32: 4545548cb50SEnrico Granata case eFormatVectorOfFloat64: 4555548cb50SEnrico Granata return eFormatFloat; 4565548cb50SEnrico Granata 4575548cb50SEnrico Granata default: 4585548cb50SEnrico Granata return lldb::eFormatInvalid; 4595548cb50SEnrico Granata } 4605548cb50SEnrico Granata } 4615548cb50SEnrico Granata 462b9c1b51eSKate Stone bool FormatManager::ShouldPrintAsOneLiner(ValueObject &valobj) { 463553fad5cSEnrico Granata // if settings say no oneline whatsoever 464b9c1b51eSKate Stone if (valobj.GetTargetSP().get() && 465a6682a41SJonas Devlieghere !valobj.GetTargetSP()->GetDebugger().GetAutoOneLineSummaries()) 466553fad5cSEnrico Granata return false; // then don't oneline 467553fad5cSEnrico Granata 46842fa4af8SEnrico Granata // if this object has a summary, then ask the summary 469a29cb0baSEnrico Granata if (valobj.GetSummaryFormat().get() != nullptr) 47042fa4af8SEnrico Granata return valobj.GetSummaryFormat()->IsOneLiner(); 471a29cb0baSEnrico Granata 472a29cb0baSEnrico Granata // no children, no party 473a29cb0baSEnrico Granata if (valobj.GetNumChildren() == 0) 474a29cb0baSEnrico Granata return false; 475a29cb0baSEnrico Granata 47605097246SAdrian Prantl // ask the type if it has any opinion about this eLazyBoolCalculate == no 47705097246SAdrian Prantl // opinion; other values should be self explanatory 4789c63f99aSEnrico Granata CompilerType compiler_type(valobj.GetCompilerType()); 479b9c1b51eSKate Stone if (compiler_type.IsValid()) { 480b9c1b51eSKate Stone switch (compiler_type.ShouldPrintAsOneLiner(&valobj)) { 4819c63f99aSEnrico Granata case eLazyBoolNo: 4829c63f99aSEnrico Granata return false; 4839c63f99aSEnrico Granata case eLazyBoolYes: 4849c63f99aSEnrico Granata return true; 4859c63f99aSEnrico Granata case eLazyBoolCalculate: 4869c63f99aSEnrico Granata break; 4879c63f99aSEnrico Granata } 4889c63f99aSEnrico Granata } 4899c63f99aSEnrico Granata 490a29cb0baSEnrico Granata size_t total_children_name_len = 0; 491a29cb0baSEnrico Granata 492b9c1b51eSKate Stone for (size_t idx = 0; idx < valobj.GetNumChildren(); idx++) { 493ddac7611SEnrico Granata bool is_synth_val = false; 494a29cb0baSEnrico Granata ValueObjectSP child_sp(valobj.GetChildAtIndex(idx, true)); 495a29cb0baSEnrico Granata // something is wrong here - bail out 496a29cb0baSEnrico Granata if (!child_sp) 497a29cb0baSEnrico Granata return false; 4986500061eSEnrico Granata 4996500061eSEnrico Granata // also ask the child's type if it has any opinion 5006500061eSEnrico Granata CompilerType child_compiler_type(child_sp->GetCompilerType()); 501b9c1b51eSKate Stone if (child_compiler_type.IsValid()) { 502b9c1b51eSKate Stone switch (child_compiler_type.ShouldPrintAsOneLiner(child_sp.get())) { 5036500061eSEnrico Granata case eLazyBoolYes: 5046500061eSEnrico Granata // an opinion of yes is only binding for the child, so keep going 5056500061eSEnrico Granata case eLazyBoolCalculate: 5066500061eSEnrico Granata break; 5076500061eSEnrico Granata case eLazyBoolNo: 5086500061eSEnrico Granata // but if the child says no, then it's a veto on the whole thing 5096500061eSEnrico Granata return false; 5106500061eSEnrico Granata } 5116500061eSEnrico Granata } 5126500061eSEnrico Granata 513b9c1b51eSKate Stone // if we decided to define synthetic children for a type, we probably care 51405097246SAdrian Prantl // enough to show them, but avoid nesting children in children 515b9c1b51eSKate Stone if (child_sp->GetSyntheticChildren().get() != nullptr) { 516ddac7611SEnrico Granata ValueObjectSP synth_sp(child_sp->GetSyntheticValue()); 517ddac7611SEnrico Granata // wait.. wat? just get out of here.. 518ddac7611SEnrico Granata if (!synth_sp) 519a29cb0baSEnrico Granata return false; 520ddac7611SEnrico Granata // but if we only have them to provide a value, keep going 521a6682a41SJonas Devlieghere if (!synth_sp->MightHaveChildren() && 522b9c1b51eSKate Stone synth_sp->DoesProvideSyntheticValue()) 523ddac7611SEnrico Granata is_synth_val = true; 524ddac7611SEnrico Granata else 525ddac7611SEnrico Granata return false; 526ddac7611SEnrico Granata } 527a29cb0baSEnrico Granata 528a29cb0baSEnrico Granata total_children_name_len += child_sp->GetName().GetLength(); 529a29cb0baSEnrico Granata 530a29cb0baSEnrico Granata // 50 itself is a "randomly" chosen number - the idea is that 531a29cb0baSEnrico Granata // overly long structs should not get this treatment 532a29cb0baSEnrico Granata // FIXME: maybe make this a user-tweakable setting? 533a29cb0baSEnrico Granata if (total_children_name_len > 50) 534a29cb0baSEnrico Granata return false; 535a29cb0baSEnrico Granata 536a29cb0baSEnrico Granata // if a summary is there.. 537b9c1b51eSKate Stone if (child_sp->GetSummaryFormat()) { 538a29cb0baSEnrico Granata // and it wants children, then bail out 5398a068e6cSEnrico Granata if (child_sp->GetSummaryFormat()->DoesPrintChildren(child_sp.get())) 540a29cb0baSEnrico Granata return false; 541a29cb0baSEnrico Granata } 542a29cb0baSEnrico Granata 543c89e4ca3SEnrico Granata // if this child has children.. 544b9c1b51eSKate Stone if (child_sp->GetNumChildren()) { 545a29cb0baSEnrico Granata // ...and no summary... 546b9c1b51eSKate Stone // (if it had a summary and the summary wanted children, we would have 547b9c1b51eSKate Stone // bailed out anyway 548b9c1b51eSKate Stone // so this only makes us bail out if this has no summary and we would 549b9c1b51eSKate Stone // then print children) 550b9c1b51eSKate Stone if (!child_sp->GetSummaryFormat() && !is_synth_val) // but again only do 551b9c1b51eSKate Stone // that if not a 552b9c1b51eSKate Stone // synthetic valued 553b9c1b51eSKate Stone // child 554a29cb0baSEnrico Granata return false; // then bail out 555a29cb0baSEnrico Granata } 556a29cb0baSEnrico Granata } 557a29cb0baSEnrico Granata return true; 558a29cb0baSEnrico Granata } 559a29cb0baSEnrico Granata 5600e4c4821SAdrian Prantl ConstString FormatManager::GetValidTypeName(ConstString type) { 5615548cb50SEnrico Granata return ::GetValidTypeName_Impl(type); 5625548cb50SEnrico Granata } 5635548cb50SEnrico Granata 564b9c1b51eSKate Stone ConstString FormatManager::GetTypeForCache(ValueObject &valobj, 565b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 566b9c1b51eSKate Stone ValueObjectSP valobj_sp = valobj.GetQualifiedRepresentationIfAvailable( 567b9c1b51eSKate Stone use_dynamic, valobj.IsSynthetic()); 568b9c1b51eSKate Stone if (valobj_sp && valobj_sp->GetCompilerType().IsValid()) { 569b3f0c340SEnrico Granata if (!valobj_sp->GetCompilerType().IsMeaninglessWithoutDynamicResolution()) 570b3f0c340SEnrico Granata return valobj_sp->GetQualifiedTypeName(); 5715548cb50SEnrico Granata } 5725548cb50SEnrico Granata return ConstString(); 5735548cb50SEnrico Granata } 5745548cb50SEnrico Granata 575d3233c1eSEnrico Granata std::vector<lldb::LanguageType> 576b9c1b51eSKate Stone FormatManager::GetCandidateLanguages(ValueObject &valobj) { 577980c0484SEnrico Granata lldb::LanguageType lang_type = valobj.GetObjectRuntimeLanguage(); 578ac49453bSEnrico Granata return GetCandidateLanguages(lang_type); 579ac49453bSEnrico Granata } 580ac49453bSEnrico Granata 581ac49453bSEnrico Granata std::vector<lldb::LanguageType> 582b9c1b51eSKate Stone FormatManager::GetCandidateLanguages(lldb::LanguageType lang_type) { 583b9c1b51eSKate Stone switch (lang_type) { 58433e97e63SEnrico Granata case lldb::eLanguageTypeC: 58533e97e63SEnrico Granata case lldb::eLanguageTypeC89: 58633e97e63SEnrico Granata case lldb::eLanguageTypeC99: 58733e97e63SEnrico Granata case lldb::eLanguageTypeC11: 58833e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus: 58933e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_03: 59033e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_11: 59133e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_14: 592170c395eSEnrico Granata return {lldb::eLanguageTypeC_plus_plus, lldb::eLanguageTypeObjC}; 593980c0484SEnrico Granata default: 594980c0484SEnrico Granata return {lang_type}; 595980c0484SEnrico Granata } 596980c0484SEnrico Granata } 597980c0484SEnrico Granata 598980c0484SEnrico Granata LanguageCategory * 599b9c1b51eSKate Stone FormatManager::GetCategoryForLanguage(lldb::LanguageType lang_type) { 60016ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 601b9c1b51eSKate Stone auto iter = m_language_categories_map.find(lang_type), 602b9c1b51eSKate Stone end = m_language_categories_map.end(); 603980c0484SEnrico Granata if (iter != end) 604980c0484SEnrico Granata return iter->second.get(); 605980c0484SEnrico Granata LanguageCategory *lang_category = new LanguageCategory(lang_type); 606b9c1b51eSKate Stone m_language_categories_map[lang_type] = 607b9c1b51eSKate Stone LanguageCategory::UniquePointer(lang_category); 608980c0484SEnrico Granata return lang_category; 609980c0484SEnrico Granata } 610980c0484SEnrico Granata 611ecd02bc1SEnrico Granata lldb::TypeFormatImplSP 612b9c1b51eSKate Stone FormatManager::GetHardcodedFormat(FormattersMatchData &match_data) { 6137cb59e1aSEnrico Granata TypeFormatImplSP retval_sp; 6147cb59e1aSEnrico Granata 615b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 616b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6178a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 6187cb59e1aSEnrico Granata break; 619ecd02bc1SEnrico Granata } 6207cb59e1aSEnrico Granata } 6217cb59e1aSEnrico Granata 6227cb59e1aSEnrico Granata return retval_sp; 623686f3debSEnrico Granata } 624686f3debSEnrico Granata 625852cc954SEnrico Granata lldb::TypeFormatImplSP 626852cc954SEnrico Granata FormatManager::GetFormat(ValueObject &valobj, 627b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 6288a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 6298a9a8f39SEnrico Granata 630852cc954SEnrico Granata TypeFormatImplSP retval; 6314edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 632b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 633*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 634b9c1b51eSKate Stone "\n\n[FormatManager::GetFormat] Looking into cache for type %s", 635b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 636b9c1b51eSKate Stone if (m_format_cache.GetFormat(match_data.GetTypeForCache(), retval)) { 637b9c1b51eSKate Stone if (log) { 638*63e5fb76SJonas Devlieghere LLDB_LOGF( 639*63e5fb76SJonas Devlieghere log, "[FormatManager::GetFormat] Cache search success. Returning."); 6406302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 641b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 642b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 64352b4b6cdSEnrico Granata } 64452b4b6cdSEnrico Granata return retval; 64552b4b6cdSEnrico Granata } 646*63e5fb76SJonas Devlieghere LLDB_LOGF( 647*63e5fb76SJonas Devlieghere log, 648b9c1b51eSKate Stone "[FormatManager::GetFormat] Cache search failed. Going normal route"); 64952b4b6cdSEnrico Granata } 650980c0484SEnrico Granata 6518a9a8f39SEnrico Granata retval = m_categories_map.GetFormat(match_data); 652b9c1b51eSKate Stone if (!retval) { 653*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 654*63e5fb76SJonas Devlieghere "[FormatManager::GetFormat] Search failed. Giving language a " 655b9c1b51eSKate Stone "chance."); 656b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 657b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6588a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 659980c0484SEnrico Granata break; 660980c0484SEnrico Granata } 661980c0484SEnrico Granata } 662b9c1b51eSKate Stone if (retval) { 663*63e5fb76SJonas Devlieghere LLDB_LOGF( 664*63e5fb76SJonas Devlieghere log, 665b9c1b51eSKate Stone "[FormatManager::GetFormat] Language search success. Returning."); 666980c0484SEnrico Granata return retval; 667980c0484SEnrico Granata } 668980c0484SEnrico Granata } 669b9c1b51eSKate Stone if (!retval) { 670*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetFormat] Search failed. Giving hardcoded " 671b9c1b51eSKate Stone "a chance."); 6728a9a8f39SEnrico Granata retval = GetHardcodedFormat(match_data); 673686f3debSEnrico Granata } 674d4cb1dddSEnrico Granata 675b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 676*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetFormat] Caching %p for type %s", 677324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 6788a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 6798a9a8f39SEnrico Granata m_format_cache.SetFormat(match_data.GetTypeForCache(), retval); 68052b4b6cdSEnrico Granata } 6816302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 682b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 683852cc954SEnrico Granata return retval; 684852cc954SEnrico Granata } 685852cc954SEnrico Granata 686ecd02bc1SEnrico Granata lldb::TypeSummaryImplSP 687b9c1b51eSKate Stone FormatManager::GetHardcodedSummaryFormat(FormattersMatchData &match_data) { 6887cb59e1aSEnrico Granata TypeSummaryImplSP retval_sp; 6897cb59e1aSEnrico Granata 690b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 691b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6928a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 6937cb59e1aSEnrico Granata break; 694ecd02bc1SEnrico Granata } 6957cb59e1aSEnrico Granata } 6967cb59e1aSEnrico Granata 6977cb59e1aSEnrico Granata return retval_sp; 698686f3debSEnrico Granata } 699686f3debSEnrico Granata 7005548cb50SEnrico Granata lldb::TypeSummaryImplSP 7015548cb50SEnrico Granata FormatManager::GetSummaryFormat(ValueObject &valobj, 702b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 7038a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 7048a9a8f39SEnrico Granata 7055548cb50SEnrico Granata TypeSummaryImplSP retval; 7064edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 707b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 708*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 709*63e5fb76SJonas Devlieghere "\n\n[FormatManager::GetSummaryFormat] Looking into cache " 710b9c1b51eSKate Stone "for type %s", 711b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 712b9c1b51eSKate Stone if (m_format_cache.GetSummary(match_data.GetTypeForCache(), retval)) { 713b9c1b51eSKate Stone if (log) { 714*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 715*63e5fb76SJonas Devlieghere "[FormatManager::GetSummaryFormat] Cache search success. " 716b9c1b51eSKate Stone "Returning."); 7176302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 718b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 719b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 72068ae4117SEnrico Granata } 7215548cb50SEnrico Granata return retval; 72268ae4117SEnrico Granata } 723*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Cache search failed. " 724b9c1b51eSKate Stone "Going normal route"); 7255548cb50SEnrico Granata } 726980c0484SEnrico Granata 7278a9a8f39SEnrico Granata retval = m_categories_map.GetSummaryFormat(match_data); 728b9c1b51eSKate Stone if (!retval) { 729*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Search failed. Giving " 730b9c1b51eSKate Stone "language a chance."); 731b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 732b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 7338a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 734980c0484SEnrico Granata break; 735980c0484SEnrico Granata } 736980c0484SEnrico Granata } 737b9c1b51eSKate Stone if (retval) { 738*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Language search " 739b9c1b51eSKate Stone "success. Returning."); 740980c0484SEnrico Granata return retval; 741980c0484SEnrico Granata } 742980c0484SEnrico Granata } 743b9c1b51eSKate Stone if (!retval) { 744*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Search failed. Giving " 745b9c1b51eSKate Stone "hardcoded a chance."); 7468a9a8f39SEnrico Granata retval = GetHardcodedSummaryFormat(match_data); 747686f3debSEnrico Granata } 748d4cb1dddSEnrico Granata 749b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 750*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Caching %p for type %s", 751324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 7528a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 7538a9a8f39SEnrico Granata m_format_cache.SetSummary(match_data.GetTypeForCache(), retval); 7545548cb50SEnrico Granata } 7556302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 756b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 7575548cb50SEnrico Granata return retval; 7585548cb50SEnrico Granata } 7595548cb50SEnrico Granata 760ecd02bc1SEnrico Granata lldb::SyntheticChildrenSP 761b9c1b51eSKate Stone FormatManager::GetHardcodedSyntheticChildren(FormattersMatchData &match_data) { 7627cb59e1aSEnrico Granata SyntheticChildrenSP retval_sp; 7637cb59e1aSEnrico Granata 764b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 765b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 7668a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 7677cb59e1aSEnrico Granata break; 768ecd02bc1SEnrico Granata } 7697cb59e1aSEnrico Granata } 7707cb59e1aSEnrico Granata 7717cb59e1aSEnrico Granata return retval_sp; 772686f3debSEnrico Granata } 773686f3debSEnrico Granata 7745548cb50SEnrico Granata lldb::SyntheticChildrenSP 7755548cb50SEnrico Granata FormatManager::GetSyntheticChildren(ValueObject &valobj, 776b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 7778a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 7788a9a8f39SEnrico Granata 7795548cb50SEnrico Granata SyntheticChildrenSP retval; 7804edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 781b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 782*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 783*63e5fb76SJonas Devlieghere "\n\n[FormatManager::GetSyntheticChildren] Looking into " 784b9c1b51eSKate Stone "cache for type %s", 785b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 786b9c1b51eSKate Stone if (m_format_cache.GetSynthetic(match_data.GetTypeForCache(), retval)) { 787b9c1b51eSKate Stone if (log) { 788*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Cache search " 789b9c1b51eSKate Stone "success. Returning."); 7906302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 7916302bf6aSPavel Labath m_format_cache.GetCacheHits(), 7926302bf6aSPavel Labath m_format_cache.GetCacheMisses()); 79368ae4117SEnrico Granata } 79468ae4117SEnrico Granata return retval; 79568ae4117SEnrico Granata } 796*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Cache search failed. " 797b9c1b51eSKate Stone "Going normal route"); 7985548cb50SEnrico Granata } 799980c0484SEnrico Granata 8008a9a8f39SEnrico Granata retval = m_categories_map.GetSyntheticChildren(match_data); 801b9c1b51eSKate Stone if (!retval) { 802*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 803*63e5fb76SJonas Devlieghere "[FormatManager::GetSyntheticChildren] Search failed. Giving " 804b9c1b51eSKate Stone "language a chance."); 805b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 806b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 8078a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 808980c0484SEnrico Granata break; 809980c0484SEnrico Granata } 810980c0484SEnrico Granata } 811b9c1b51eSKate Stone if (retval) { 812*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Language search " 813b9c1b51eSKate Stone "success. Returning."); 814980c0484SEnrico Granata return retval; 815980c0484SEnrico Granata } 816980c0484SEnrico Granata } 817b9c1b51eSKate Stone if (!retval) { 818*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 819*63e5fb76SJonas Devlieghere "[FormatManager::GetSyntheticChildren] Search failed. Giving " 820b9c1b51eSKate Stone "hardcoded a chance."); 8218a9a8f39SEnrico Granata retval = GetHardcodedSyntheticChildren(match_data); 822686f3debSEnrico Granata } 823d4cb1dddSEnrico Granata 824b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 825*63e5fb76SJonas Devlieghere LLDB_LOGF(log, 826b9c1b51eSKate Stone "[FormatManager::GetSyntheticChildren] Caching %p for type %s", 827324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 8288a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 8298a9a8f39SEnrico Granata m_format_cache.SetSynthetic(match_data.GetTypeForCache(), retval); 8305548cb50SEnrico Granata } 8316302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 832b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 8335548cb50SEnrico Granata return retval; 8345548cb50SEnrico Granata } 8355548cb50SEnrico Granata 836c582713cSEnrico Granata lldb::TypeValidatorImplSP 837c582713cSEnrico Granata FormatManager::GetValidator(ValueObject &valobj, 838b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 8398a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 8408a9a8f39SEnrico Granata 841c582713cSEnrico Granata TypeValidatorImplSP retval; 8424edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 843b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 844*63e5fb76SJonas Devlieghere LLDB_LOGF( 845*63e5fb76SJonas Devlieghere log, "\n\n[FormatManager::GetValidator] Looking into cache for type %s", 846b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 847b9c1b51eSKate Stone if (m_format_cache.GetValidator(match_data.GetTypeForCache(), retval)) { 848b9c1b51eSKate Stone if (log) { 849*63e5fb76SJonas Devlieghere LLDB_LOGF( 850*63e5fb76SJonas Devlieghere log, 851b9c1b51eSKate Stone "[FormatManager::GetValidator] Cache search success. Returning."); 8526302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 853b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 854b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 855c582713cSEnrico Granata } 856c582713cSEnrico Granata return retval; 857c582713cSEnrico Granata } 858*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Cache search failed. Going " 859b9c1b51eSKate Stone "normal route"); 860c582713cSEnrico Granata } 861980c0484SEnrico Granata 8628a9a8f39SEnrico Granata retval = m_categories_map.GetValidator(match_data); 863b9c1b51eSKate Stone if (!retval) { 864*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Search failed. Giving " 865b9c1b51eSKate Stone "language a chance."); 866b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 867b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 8688a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 869980c0484SEnrico Granata break; 870980c0484SEnrico Granata } 871980c0484SEnrico Granata } 872b9c1b51eSKate Stone if (retval) { 873*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Language search success. " 874b9c1b51eSKate Stone "Returning."); 875980c0484SEnrico Granata return retval; 876980c0484SEnrico Granata } 877980c0484SEnrico Granata } 878b9c1b51eSKate Stone if (!retval) { 879*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Search failed. Giving " 880b9c1b51eSKate Stone "hardcoded a chance."); 8818a9a8f39SEnrico Granata retval = GetHardcodedValidator(match_data); 882c582713cSEnrico Granata } 883d4cb1dddSEnrico Granata 884b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 885*63e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Caching %p for type %s", 886c582713cSEnrico Granata static_cast<void *>(retval.get()), 8878a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 8888a9a8f39SEnrico Granata m_format_cache.SetValidator(match_data.GetTypeForCache(), retval); 889c582713cSEnrico Granata } 8906302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 891b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 892c582713cSEnrico Granata return retval; 893c582713cSEnrico Granata } 894c582713cSEnrico Granata 895c582713cSEnrico Granata lldb::TypeValidatorImplSP 896b9c1b51eSKate Stone FormatManager::GetHardcodedValidator(FormattersMatchData &match_data) { 8977cb59e1aSEnrico Granata TypeValidatorImplSP retval_sp; 8987cb59e1aSEnrico Granata 899b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 900b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 9018a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 9027cb59e1aSEnrico Granata break; 903c582713cSEnrico Granata } 9047cb59e1aSEnrico Granata } 9057cb59e1aSEnrico Granata 9067cb59e1aSEnrico Granata return retval_sp; 907c582713cSEnrico Granata } 908c582713cSEnrico Granata 90916ff8604SSaleem Abdulrasool FormatManager::FormatManager() 910b9c1b51eSKate Stone : m_last_revision(0), m_format_cache(), m_language_categories_mutex(), 911b9c1b51eSKate Stone m_language_categories_map(), m_named_summaries_map(this), 912b9c1b51eSKate Stone m_categories_map(this), m_default_category_name(ConstString("default")), 9135548cb50SEnrico Granata m_system_category_name(ConstString("system")), 914b9c1b51eSKate Stone m_vectortypes_category_name(ConstString("VectorTypes")) { 9155548cb50SEnrico Granata LoadSystemFormatters(); 916170c395eSEnrico Granata LoadVectorFormatters(); 9175548cb50SEnrico Granata 918b9c1b51eSKate Stone EnableCategory(m_vectortypes_category_name, TypeCategoryMap::Last, 919b9c1b51eSKate Stone lldb::eLanguageTypeObjC_plus_plus); 920b9c1b51eSKate Stone EnableCategory(m_system_category_name, TypeCategoryMap::Last, 921b9c1b51eSKate Stone lldb::eLanguageTypeObjC_plus_plus); 9225548cb50SEnrico Granata } 9235548cb50SEnrico Granata 924b9c1b51eSKate Stone void FormatManager::LoadSystemFormatters() { 9255548cb50SEnrico Granata TypeSummaryImpl::Flags string_flags; 9260337c27fSEnrico Granata string_flags.SetCascades(true) 9275548cb50SEnrico Granata .SetSkipPointers(true) 9285548cb50SEnrico Granata .SetSkipReferences(false) 9295548cb50SEnrico Granata .SetDontShowChildren(true) 9305548cb50SEnrico Granata .SetDontShowValue(false) 9315548cb50SEnrico Granata .SetShowMembersOneLiner(false) 9325548cb50SEnrico Granata .SetHideItemNames(false); 9335548cb50SEnrico Granata 934bc2c2b01SEnrico Granata TypeSummaryImpl::Flags string_array_flags; 935d2911633SEnrico Granata string_array_flags.SetCascades(true) 9365548cb50SEnrico Granata .SetSkipPointers(true) 9375548cb50SEnrico Granata .SetSkipReferences(false) 938320dcf68SEnrico Granata .SetDontShowChildren(true) 9395548cb50SEnrico Granata .SetDontShowValue(true) 9405548cb50SEnrico Granata .SetShowMembersOneLiner(false) 941bc2c2b01SEnrico Granata .SetHideItemNames(false); 942bc2c2b01SEnrico Granata 943b9c1b51eSKate Stone lldb::TypeSummaryImplSP string_format( 944b9c1b51eSKate Stone new StringSummaryFormat(string_flags, "${var%s}")); 945bc2c2b01SEnrico Granata 946b9c1b51eSKate Stone lldb::TypeSummaryImplSP string_array_format( 947b9c1b51eSKate Stone new StringSummaryFormat(string_array_flags, "${var%s}")); 948bc2c2b01SEnrico Granata 949b9c1b51eSKate Stone lldb::RegularExpressionSP any_size_char_arr( 95095eae423SZachary Turner new RegularExpression(llvm::StringRef("char \\[[0-9]+\\]"))); 951b9c1b51eSKate Stone lldb::RegularExpressionSP any_size_wchar_arr( 95295eae423SZachary Turner new RegularExpression(llvm::StringRef("wchar_t \\[[0-9]+\\]"))); 9535548cb50SEnrico Granata 954b9c1b51eSKate Stone TypeCategoryImpl::SharedPointer sys_category_sp = 955b9c1b51eSKate Stone GetCategory(m_system_category_name); 9565548cb50SEnrico Granata 957b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("char *"), 958b9c1b51eSKate Stone string_format); 959b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add( 960b9c1b51eSKate Stone ConstString("unsigned char *"), string_format); 961b9c1b51eSKate Stone sys_category_sp->GetRegexTypeSummariesContainer()->Add(any_size_char_arr, 962b9c1b51eSKate Stone string_array_format); 9635548cb50SEnrico Granata 964b9c1b51eSKate Stone lldb::TypeSummaryImplSP ostype_summary( 965b9c1b51eSKate Stone new StringSummaryFormat(TypeSummaryImpl::Flags() 966b9c1b51eSKate Stone .SetCascades(false) 9675548cb50SEnrico Granata .SetSkipPointers(true) 9685548cb50SEnrico Granata .SetSkipReferences(true) 9695548cb50SEnrico Granata .SetDontShowChildren(true) 9705548cb50SEnrico Granata .SetDontShowValue(false) 9715548cb50SEnrico Granata .SetShowMembersOneLiner(false) 9725548cb50SEnrico Granata .SetHideItemNames(false), 9735548cb50SEnrico Granata "${var%O}")); 9745548cb50SEnrico Granata 975b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("OSType"), 976b9c1b51eSKate Stone ostype_summary); 9775548cb50SEnrico Granata 9784cc21772SEnrico Granata TypeFormatImpl::Flags fourchar_flags; 979b9c1b51eSKate Stone fourchar_flags.SetCascades(true).SetSkipPointers(true).SetSkipReferences( 980b9c1b51eSKate Stone true); 9814cc21772SEnrico Granata 982b9c1b51eSKate Stone AddFormat(sys_category_sp, lldb::eFormatOSType, ConstString("FourCharCode"), 983b9c1b51eSKate Stone fourchar_flags); 9845548cb50SEnrico Granata } 9855548cb50SEnrico Granata 986b9c1b51eSKate Stone void FormatManager::LoadVectorFormatters() { 987b9c1b51eSKate Stone TypeCategoryImpl::SharedPointer vectors_category_sp = 988b9c1b51eSKate Stone GetCategory(m_vectortypes_category_name); 9895548cb50SEnrico Granata 9905548cb50SEnrico Granata TypeSummaryImpl::Flags vector_flags; 9915548cb50SEnrico Granata vector_flags.SetCascades(true) 9925548cb50SEnrico Granata .SetSkipPointers(true) 9935548cb50SEnrico Granata .SetSkipReferences(false) 9945548cb50SEnrico Granata .SetDontShowChildren(true) 9955548cb50SEnrico Granata .SetDontShowValue(false) 9965548cb50SEnrico Granata .SetShowMembersOneLiner(true) 9975548cb50SEnrico Granata .SetHideItemNames(true); 9985548cb50SEnrico Granata 999b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "${var.uint128}", 1000b9c1b51eSKate Stone ConstString("builtin_type_vec128"), vector_flags); 10015548cb50SEnrico Granata 1002b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("float [4]"), 10035548cb50SEnrico Granata vector_flags); 1004b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("int32_t [4]"), 10055548cb50SEnrico Granata vector_flags); 1006b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("int16_t [8]"), 10075548cb50SEnrico Granata vector_flags); 1008b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vDouble"), 10095548cb50SEnrico Granata vector_flags); 1010b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vFloat"), 10115548cb50SEnrico Granata vector_flags); 1012b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt8"), 10135548cb50SEnrico Granata vector_flags); 1014b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt16"), 10155548cb50SEnrico Granata vector_flags); 1016b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt32"), 10175548cb50SEnrico Granata vector_flags); 1018b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"), 10195548cb50SEnrico Granata vector_flags); 1020b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt8"), 10215548cb50SEnrico Granata vector_flags); 1022b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"), 10235548cb50SEnrico Granata vector_flags); 1024b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt32"), 10255548cb50SEnrico Granata vector_flags); 1026b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vBool32"), 10275548cb50SEnrico Granata vector_flags); 10285548cb50SEnrico Granata } 1029