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"}, 50*ca4409b4SJason Molenda {eFormatUnicode8, 'u', "unicode8"}, 515548cb50SEnrico Granata {eFormatUnicode16, 'U', "unicode16"}, 525548cb50SEnrico Granata {eFormatUnicode32, '\0', "unicode32"}, 535548cb50SEnrico Granata {eFormatUnsigned, 'u', "unsigned decimal"}, 545548cb50SEnrico Granata {eFormatPointer, 'p', "pointer"}, 555548cb50SEnrico Granata {eFormatVectorOfChar, '\0', "char[]"}, 565548cb50SEnrico Granata {eFormatVectorOfSInt8, '\0', "int8_t[]"}, 575548cb50SEnrico Granata {eFormatVectorOfUInt8, '\0', "uint8_t[]"}, 585548cb50SEnrico Granata {eFormatVectorOfSInt16, '\0', "int16_t[]"}, 595548cb50SEnrico Granata {eFormatVectorOfUInt16, '\0', "uint16_t[]"}, 605548cb50SEnrico Granata {eFormatVectorOfSInt32, '\0', "int32_t[]"}, 615548cb50SEnrico Granata {eFormatVectorOfUInt32, '\0', "uint32_t[]"}, 625548cb50SEnrico Granata {eFormatVectorOfSInt64, '\0', "int64_t[]"}, 635548cb50SEnrico Granata {eFormatVectorOfUInt64, '\0', "uint64_t[]"}, 64a0f08674SEwan Crawford {eFormatVectorOfFloat16, '\0', "float16[]"}, 655548cb50SEnrico Granata {eFormatVectorOfFloat32, '\0', "float32[]"}, 665548cb50SEnrico Granata {eFormatVectorOfFloat64, '\0', "float64[]"}, 675548cb50SEnrico Granata {eFormatVectorOfUInt128, '\0', "uint128_t[]"}, 685548cb50SEnrico Granata {eFormatComplexInteger, 'I', "complex integer"}, 695548cb50SEnrico Granata {eFormatCharArray, 'a', "character array"}, 705548cb50SEnrico Granata {eFormatAddressInfo, 'A', "address"}, 715548cb50SEnrico Granata {eFormatHexFloat, '\0', "hex float"}, 725548cb50SEnrico Granata {eFormatInstruction, 'i', "instruction"}, 73b9c1b51eSKate Stone {eFormatVoid, 'v', "void"}}; 745548cb50SEnrico Granata 7528606954SSaleem Abdulrasool static uint32_t g_num_format_infos = llvm::array_lengthof(g_format_infos); 765548cb50SEnrico Granata 77b9c1b51eSKate Stone static bool GetFormatFromFormatChar(char format_char, Format &format) { 78b9c1b51eSKate Stone for (uint32_t i = 0; i < g_num_format_infos; ++i) { 79b9c1b51eSKate Stone if (g_format_infos[i].format_char == format_char) { 805548cb50SEnrico Granata format = g_format_infos[i].format; 815548cb50SEnrico Granata return true; 825548cb50SEnrico Granata } 835548cb50SEnrico Granata } 845548cb50SEnrico Granata format = eFormatInvalid; 855548cb50SEnrico Granata return false; 865548cb50SEnrico Granata } 875548cb50SEnrico Granata 88b9c1b51eSKate Stone static bool GetFormatFromFormatName(const char *format_name, 89b9c1b51eSKate Stone bool partial_match_ok, Format &format) { 905548cb50SEnrico Granata uint32_t i; 91b9c1b51eSKate Stone for (i = 0; i < g_num_format_infos; ++i) { 92b9c1b51eSKate Stone if (strcasecmp(g_format_infos[i].format_name, format_name) == 0) { 935548cb50SEnrico Granata format = g_format_infos[i].format; 945548cb50SEnrico Granata return true; 955548cb50SEnrico Granata } 965548cb50SEnrico Granata } 975548cb50SEnrico Granata 98b9c1b51eSKate Stone if (partial_match_ok) { 99b9c1b51eSKate Stone for (i = 0; i < g_num_format_infos; ++i) { 100b9c1b51eSKate Stone if (strcasestr(g_format_infos[i].format_name, format_name) == 101b9c1b51eSKate Stone g_format_infos[i].format_name) { 1025548cb50SEnrico Granata format = g_format_infos[i].format; 1035548cb50SEnrico Granata return true; 1045548cb50SEnrico Granata } 1055548cb50SEnrico Granata } 1065548cb50SEnrico Granata } 1075548cb50SEnrico Granata format = eFormatInvalid; 1085548cb50SEnrico Granata return false; 1095548cb50SEnrico Granata } 1105548cb50SEnrico Granata 111b9c1b51eSKate Stone void FormatManager::Changed() { 112bd5eab82SEnrico Granata ++m_last_revision; 113bd5eab82SEnrico Granata m_format_cache.Clear(); 11416ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 115b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 116bd5eab82SEnrico Granata if (iter.second) 117bd5eab82SEnrico Granata iter.second->GetFormatCache().Clear(); 118bd5eab82SEnrico Granata } 119bd5eab82SEnrico Granata } 120bd5eab82SEnrico Granata 121b9c1b51eSKate Stone bool FormatManager::GetFormatFromCString(const char *format_cstr, 1225548cb50SEnrico Granata bool partial_match_ok, 123b9c1b51eSKate Stone lldb::Format &format) { 1245548cb50SEnrico Granata bool success = false; 125b9c1b51eSKate Stone if (format_cstr && format_cstr[0]) { 126b9c1b51eSKate Stone if (format_cstr[1] == '\0') { 1275548cb50SEnrico Granata success = GetFormatFromFormatChar(format_cstr[0], format); 1285548cb50SEnrico Granata if (success) 1295548cb50SEnrico Granata return true; 1305548cb50SEnrico Granata } 1315548cb50SEnrico Granata 1325548cb50SEnrico Granata success = GetFormatFromFormatName(format_cstr, partial_match_ok, format); 1335548cb50SEnrico Granata } 1345548cb50SEnrico Granata if (!success) 1355548cb50SEnrico Granata format = eFormatInvalid; 1365548cb50SEnrico Granata return success; 1375548cb50SEnrico Granata } 1385548cb50SEnrico Granata 139b9c1b51eSKate Stone char FormatManager::GetFormatAsFormatChar(lldb::Format format) { 140b9c1b51eSKate Stone for (uint32_t i = 0; i < g_num_format_infos; ++i) { 1415548cb50SEnrico Granata if (g_format_infos[i].format == format) 1425548cb50SEnrico Granata return g_format_infos[i].format_char; 1435548cb50SEnrico Granata } 1445548cb50SEnrico Granata return '\0'; 1455548cb50SEnrico Granata } 1465548cb50SEnrico Granata 147b9c1b51eSKate Stone const char *FormatManager::GetFormatAsCString(Format format) { 1485548cb50SEnrico Granata if (format >= eFormatDefault && format < kNumFormats) 1495548cb50SEnrico Granata return g_format_infos[format].format_name; 150248a1305SKonrad Kleine return nullptr; 1515548cb50SEnrico Granata } 1525548cb50SEnrico Granata 153b9c1b51eSKate Stone void FormatManager::EnableAllCategories() { 15433e97e63SEnrico Granata m_categories_map.EnableAllCategories(); 15516ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 156b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 15733e97e63SEnrico Granata if (iter.second) 15833e97e63SEnrico Granata iter.second->Enable(); 15933e97e63SEnrico Granata } 16033e97e63SEnrico Granata } 16133e97e63SEnrico Granata 162b9c1b51eSKate Stone void FormatManager::DisableAllCategories() { 16333e97e63SEnrico Granata m_categories_map.DisableAllCategories(); 16416ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 165b9c1b51eSKate Stone for (auto &iter : m_language_categories_map) { 16633e97e63SEnrico Granata if (iter.second) 16733e97e63SEnrico Granata iter.second->Disable(); 16833e97e63SEnrico Granata } 16933e97e63SEnrico Granata } 17033e97e63SEnrico Granata 171b9c1b51eSKate Stone void FormatManager::GetPossibleMatches( 172b9c1b51eSKate Stone ValueObject &valobj, CompilerType compiler_type, uint32_t reason, 173b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic, FormattersMatchVector &entries, 174b9c1b51eSKate Stone bool did_strip_ptr, bool did_strip_ref, bool did_strip_typedef, 175b9c1b51eSKate Stone bool root_level) { 176c6bf2e2dSEnrico Granata compiler_type = compiler_type.GetTypeForFormatters(); 17759b5a37dSBruce Mitchener ConstString type_name(compiler_type.GetConstTypeName()); 178b9c1b51eSKate Stone if (valobj.GetBitfieldBitSize() > 0) { 179de61cecdSEnrico Granata StreamString sstring; 180de61cecdSEnrico Granata sstring.Printf("%s:%d", type_name.AsCString(), valobj.GetBitfieldBitSize()); 181c156427dSZachary Turner ConstString bitfieldname(sstring.GetString()); 182b9c1b51eSKate Stone entries.push_back( 183b9c1b51eSKate Stone {bitfieldname, 0, did_strip_ptr, did_strip_ref, did_strip_typedef}); 184de61cecdSEnrico Granata reason |= lldb_private::eFormatterChoiceCriterionStrippedBitField; 185de61cecdSEnrico Granata } 186b3f0c340SEnrico Granata 18778f05d35SDavide Italiano if (!compiler_type.IsMeaninglessWithoutDynamicResolution()) { 188b9c1b51eSKate Stone entries.push_back( 189b9c1b51eSKate Stone {type_name, reason, did_strip_ptr, did_strip_ref, did_strip_typedef}); 190de61cecdSEnrico Granata 19159b5a37dSBruce Mitchener ConstString display_type_name(compiler_type.GetDisplayTypeName()); 192e8daa2f8SEnrico Granata if (display_type_name != type_name) 19378f05d35SDavide Italiano entries.push_back({display_type_name, reason, did_strip_ptr, 19478f05d35SDavide Italiano did_strip_ref, did_strip_typedef}); 19578f05d35SDavide Italiano } 196e8daa2f8SEnrico Granata 197b9c1b51eSKate Stone for (bool is_rvalue_ref = true, j = true; 198b9c1b51eSKate Stone j && compiler_type.IsReferenceType(nullptr, &is_rvalue_ref); j = false) { 19959b5a37dSBruce Mitchener CompilerType non_ref_type = compiler_type.GetNonReferenceType(); 200b9c1b51eSKate Stone GetPossibleMatches( 201b9c1b51eSKate Stone valobj, non_ref_type, 202b9c1b51eSKate Stone reason | 203b9c1b51eSKate Stone lldb_private::eFormatterChoiceCriterionStrippedPointerReference, 204b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, true, did_strip_typedef); 205b9c1b51eSKate Stone if (non_ref_type.IsTypedefType()) { 206a1e5dc86SGreg Clayton CompilerType deffed_referenced_type = non_ref_type.GetTypedefedType(); 207b9c1b51eSKate Stone deffed_referenced_type = 208b9c1b51eSKate Stone is_rvalue_ref ? deffed_referenced_type.GetRValueReferenceType() 209b9c1b51eSKate Stone : deffed_referenced_type.GetLValueReferenceType(); 210b9c1b51eSKate Stone GetPossibleMatches( 211b9c1b51eSKate Stone valobj, deffed_referenced_type, 2121ac62963SEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 213b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 2141ac62963SEnrico Granata true); // this is not exactly the usual meaning of stripping typedefs 215de61cecdSEnrico Granata } 2161ac62963SEnrico Granata } 2171ac62963SEnrico Granata 218b9c1b51eSKate Stone if (compiler_type.IsPointerType()) { 21959b5a37dSBruce Mitchener CompilerType non_ptr_type = compiler_type.GetPointeeType(); 220b9c1b51eSKate Stone GetPossibleMatches( 221b9c1b51eSKate Stone valobj, non_ptr_type, 222b9c1b51eSKate Stone reason | 223b9c1b51eSKate Stone lldb_private::eFormatterChoiceCriterionStrippedPointerReference, 224b9c1b51eSKate Stone use_dynamic, entries, true, did_strip_ref, did_strip_typedef); 225b9c1b51eSKate Stone if (non_ptr_type.IsTypedefType()) { 226b9c1b51eSKate Stone CompilerType deffed_pointed_type = 227b9c1b51eSKate Stone non_ptr_type.GetTypedefedType().GetPointerType(); 228b9c1b51eSKate Stone GetPossibleMatches( 229b9c1b51eSKate Stone valobj, deffed_pointed_type, 2301ac62963SEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 231b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 2321ac62963SEnrico Granata true); // this is not exactly the usual meaning of stripping typedefs 2331ac62963SEnrico Granata } 234de61cecdSEnrico Granata } 235de61cecdSEnrico Granata 236b9c1b51eSKate Stone for (lldb::LanguageType language_type : GetCandidateLanguages(valobj)) { 237b9c1b51eSKate Stone if (Language *language = Language::FindPlugin(language_type)) { 238b9c1b51eSKate Stone for (ConstString candidate : 239b9c1b51eSKate Stone language->GetPossibleFormattersMatches(valobj, use_dynamic)) { 240b9c1b51eSKate Stone entries.push_back( 241b9c1b51eSKate Stone {candidate, 242d3233c1eSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionLanguagePlugin, 243b9c1b51eSKate Stone did_strip_ptr, did_strip_ref, did_strip_typedef}); 244d3233c1eSEnrico Granata } 245d3233c1eSEnrico Granata } 246de61cecdSEnrico Granata } 247de61cecdSEnrico Granata 248de61cecdSEnrico Granata // try to strip typedef chains 249b9c1b51eSKate Stone if (compiler_type.IsTypedefType()) { 25059b5a37dSBruce Mitchener CompilerType deffed_type = compiler_type.GetTypedefedType(); 251b9c1b51eSKate Stone GetPossibleMatches( 252b9c1b51eSKate Stone valobj, deffed_type, 253de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs, 254b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, true); 255de61cecdSEnrico Granata } 256de61cecdSEnrico Granata 257b9c1b51eSKate Stone if (root_level) { 258de61cecdSEnrico Granata do { 25959b5a37dSBruce Mitchener if (!compiler_type.IsValid()) 260de61cecdSEnrico Granata break; 261de61cecdSEnrico Granata 262b9c1b51eSKate Stone CompilerType unqual_compiler_ast_type = 263b9c1b51eSKate Stone compiler_type.GetFullyUnqualifiedType(); 26459b5a37dSBruce Mitchener if (!unqual_compiler_ast_type.IsValid()) 265de61cecdSEnrico Granata break; 266b9c1b51eSKate Stone if (unqual_compiler_ast_type.GetOpaqueQualType() != 267b9c1b51eSKate Stone compiler_type.GetOpaqueQualType()) 268b9c1b51eSKate Stone GetPossibleMatches(valobj, unqual_compiler_ast_type, reason, 269b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 270de61cecdSEnrico Granata did_strip_typedef); 271de61cecdSEnrico Granata } while (false); 272de61cecdSEnrico Granata 273de61cecdSEnrico Granata // if all else fails, go to static type 274b9c1b51eSKate Stone if (valobj.IsDynamic()) { 275de61cecdSEnrico Granata lldb::ValueObjectSP static_value_sp(valobj.GetStaticValue()); 276de61cecdSEnrico Granata if (static_value_sp) 277b9c1b51eSKate Stone GetPossibleMatches( 278b9c1b51eSKate Stone *static_value_sp.get(), static_value_sp->GetCompilerType(), 279de61cecdSEnrico Granata reason | lldb_private::eFormatterChoiceCriterionWentToStaticValue, 280b9c1b51eSKate Stone use_dynamic, entries, did_strip_ptr, did_strip_ref, 281b9c1b51eSKate Stone did_strip_typedef, true); 282de61cecdSEnrico Granata } 283de61cecdSEnrico Granata } 284de61cecdSEnrico Granata } 285de61cecdSEnrico Granata 286852cc954SEnrico Granata lldb::TypeFormatImplSP 287b9c1b51eSKate Stone FormatManager::GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp) { 288852cc954SEnrico Granata if (!type_sp) 289852cc954SEnrico Granata return lldb::TypeFormatImplSP(); 290852cc954SEnrico Granata lldb::TypeFormatImplSP format_chosen_sp; 291852cc954SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 292852cc954SEnrico Granata lldb::TypeCategoryImplSP category_sp; 293852cc954SEnrico Granata uint32_t prio_category = UINT32_MAX; 294b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 295852cc954SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 296a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 297852cc954SEnrico Granata continue; 298b9c1b51eSKate Stone lldb::TypeFormatImplSP format_current_sp = 299b9c1b51eSKate Stone category_sp->GetFormatForType(type_sp); 300b9c1b51eSKate Stone if (format_current_sp && 301248a1305SKonrad Kleine (format_chosen_sp.get() == nullptr || 302b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 303852cc954SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 304852cc954SEnrico Granata format_chosen_sp = format_current_sp; 305852cc954SEnrico Granata } 306852cc954SEnrico Granata } 307852cc954SEnrico Granata return format_chosen_sp; 308852cc954SEnrico Granata } 309852cc954SEnrico Granata 3105548cb50SEnrico Granata lldb::TypeSummaryImplSP 311b9c1b51eSKate Stone FormatManager::GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp) { 3125548cb50SEnrico Granata if (!type_sp) 3135548cb50SEnrico Granata return lldb::TypeSummaryImplSP(); 3145548cb50SEnrico Granata lldb::TypeSummaryImplSP summary_chosen_sp; 3155548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3165548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3175548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 318b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3195548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 320a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3215548cb50SEnrico Granata continue; 322b9c1b51eSKate Stone lldb::TypeSummaryImplSP summary_current_sp = 323b9c1b51eSKate Stone category_sp->GetSummaryForType(type_sp); 324b9c1b51eSKate Stone if (summary_current_sp && 325248a1305SKonrad Kleine (summary_chosen_sp.get() == nullptr || 326b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3275548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3285548cb50SEnrico Granata summary_chosen_sp = summary_current_sp; 3295548cb50SEnrico Granata } 3305548cb50SEnrico Granata } 3315548cb50SEnrico Granata return summary_chosen_sp; 3325548cb50SEnrico Granata } 3335548cb50SEnrico Granata 3345548cb50SEnrico Granata lldb::TypeFilterImplSP 335b9c1b51eSKate Stone FormatManager::GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp) { 3365548cb50SEnrico Granata if (!type_sp) 3375548cb50SEnrico Granata return lldb::TypeFilterImplSP(); 3385548cb50SEnrico Granata lldb::TypeFilterImplSP filter_chosen_sp; 3395548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3405548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3415548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 342b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3435548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 344a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3455548cb50SEnrico Granata continue; 346b9c1b51eSKate Stone lldb::TypeFilterImplSP filter_current_sp( 347b9c1b51eSKate Stone (TypeFilterImpl *)category_sp->GetFilterForType(type_sp).get()); 348b9c1b51eSKate Stone if (filter_current_sp && 349248a1305SKonrad Kleine (filter_chosen_sp.get() == nullptr || 350b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3515548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3525548cb50SEnrico Granata filter_chosen_sp = filter_current_sp; 3535548cb50SEnrico Granata } 3545548cb50SEnrico Granata } 3555548cb50SEnrico Granata return filter_chosen_sp; 3565548cb50SEnrico Granata } 3575548cb50SEnrico Granata 3585548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP 359b9c1b51eSKate Stone FormatManager::GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp) { 3605548cb50SEnrico Granata if (!type_sp) 3615548cb50SEnrico Granata return lldb::ScriptedSyntheticChildrenSP(); 3625548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP synth_chosen_sp; 3635548cb50SEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 3645548cb50SEnrico Granata lldb::TypeCategoryImplSP category_sp; 3655548cb50SEnrico Granata uint32_t prio_category = UINT32_MAX; 366b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 3675548cb50SEnrico Granata category_sp = GetCategoryAtIndex(category_id); 368a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 3695548cb50SEnrico Granata continue; 370b9c1b51eSKate Stone lldb::ScriptedSyntheticChildrenSP synth_current_sp( 371b9c1b51eSKate Stone (ScriptedSyntheticChildren *)category_sp->GetSyntheticForType(type_sp) 372b9c1b51eSKate Stone .get()); 373b9c1b51eSKate Stone if (synth_current_sp && 374248a1305SKonrad Kleine (synth_chosen_sp.get() == nullptr || 375b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 3765548cb50SEnrico Granata prio_category = category_sp->GetEnabledPosition(); 3775548cb50SEnrico Granata synth_chosen_sp = synth_current_sp; 3785548cb50SEnrico Granata } 3795548cb50SEnrico Granata } 3805548cb50SEnrico Granata return synth_chosen_sp; 3815548cb50SEnrico Granata } 3825548cb50SEnrico Granata 383c582713cSEnrico Granata lldb::TypeValidatorImplSP 384b9c1b51eSKate Stone FormatManager::GetValidatorForType(lldb::TypeNameSpecifierImplSP type_sp) { 385c582713cSEnrico Granata if (!type_sp) 386c582713cSEnrico Granata return lldb::TypeValidatorImplSP(); 387c582713cSEnrico Granata lldb::TypeValidatorImplSP validator_chosen_sp; 388c582713cSEnrico Granata uint32_t num_categories = m_categories_map.GetCount(); 389c582713cSEnrico Granata lldb::TypeCategoryImplSP category_sp; 390c582713cSEnrico Granata uint32_t prio_category = UINT32_MAX; 391b9c1b51eSKate Stone for (uint32_t category_id = 0; category_id < num_categories; category_id++) { 392c582713cSEnrico Granata category_sp = GetCategoryAtIndex(category_id); 393a6682a41SJonas Devlieghere if (!category_sp->IsEnabled()) 394c582713cSEnrico Granata continue; 395b9c1b51eSKate Stone lldb::TypeValidatorImplSP validator_current_sp( 396b9c1b51eSKate Stone category_sp->GetValidatorForType(type_sp).get()); 397b9c1b51eSKate Stone if (validator_current_sp && 398248a1305SKonrad Kleine (validator_chosen_sp.get() == nullptr || 399b9c1b51eSKate Stone (prio_category > category_sp->GetEnabledPosition()))) { 400c582713cSEnrico Granata prio_category = category_sp->GetEnabledPosition(); 401c582713cSEnrico Granata validator_chosen_sp = validator_current_sp; 402c582713cSEnrico Granata } 403c582713cSEnrico Granata } 404c582713cSEnrico Granata return validator_chosen_sp; 405c582713cSEnrico Granata } 406c582713cSEnrico Granata 407b9c1b51eSKate Stone void FormatManager::ForEachCategory(TypeCategoryMap::ForEachCallback callback) { 408b56d0103SEnrico Granata m_categories_map.ForEach(callback); 40916ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 410b9c1b51eSKate Stone for (const auto &entry : m_language_categories_map) { 411b9c1b51eSKate Stone if (auto category_sp = entry.second->GetCategory()) { 412b56d0103SEnrico Granata if (!callback(category_sp)) 413b56d0103SEnrico Granata break; 414b56d0103SEnrico Granata } 415b56d0103SEnrico Granata } 416b56d0103SEnrico Granata } 417b56d0103SEnrico Granata 4185548cb50SEnrico Granata lldb::TypeCategoryImplSP 4190e4c4821SAdrian Prantl FormatManager::GetCategory(ConstString category_name, bool can_create) { 4205548cb50SEnrico Granata if (!category_name) 4215548cb50SEnrico Granata return GetCategory(m_default_category_name); 4225548cb50SEnrico Granata lldb::TypeCategoryImplSP category; 4235548cb50SEnrico Granata if (m_categories_map.Get(category_name, category)) 4245548cb50SEnrico Granata return category; 4255548cb50SEnrico Granata 4265548cb50SEnrico Granata if (!can_create) 4275548cb50SEnrico Granata return lldb::TypeCategoryImplSP(); 4285548cb50SEnrico Granata 429b9c1b51eSKate Stone m_categories_map.Add( 430b9c1b51eSKate Stone category_name, 431b9c1b51eSKate Stone lldb::TypeCategoryImplSP(new TypeCategoryImpl(this, category_name))); 4325548cb50SEnrico Granata return GetCategory(category_name); 4335548cb50SEnrico Granata } 4345548cb50SEnrico Granata 435b9c1b51eSKate Stone lldb::Format FormatManager::GetSingleItemFormat(lldb::Format vector_format) { 436b9c1b51eSKate Stone switch (vector_format) { 4375548cb50SEnrico Granata case eFormatVectorOfChar: 4385548cb50SEnrico Granata return eFormatCharArray; 4395548cb50SEnrico Granata 4405548cb50SEnrico Granata case eFormatVectorOfSInt8: 4415548cb50SEnrico Granata case eFormatVectorOfSInt16: 4425548cb50SEnrico Granata case eFormatVectorOfSInt32: 4435548cb50SEnrico Granata case eFormatVectorOfSInt64: 4445548cb50SEnrico Granata return eFormatDecimal; 4455548cb50SEnrico Granata 4465548cb50SEnrico Granata case eFormatVectorOfUInt8: 4475548cb50SEnrico Granata case eFormatVectorOfUInt16: 4485548cb50SEnrico Granata case eFormatVectorOfUInt32: 4495548cb50SEnrico Granata case eFormatVectorOfUInt64: 4505548cb50SEnrico Granata case eFormatVectorOfUInt128: 4515548cb50SEnrico Granata return eFormatHex; 4525548cb50SEnrico Granata 453a0f08674SEwan Crawford case eFormatVectorOfFloat16: 4545548cb50SEnrico Granata case eFormatVectorOfFloat32: 4555548cb50SEnrico Granata case eFormatVectorOfFloat64: 4565548cb50SEnrico Granata return eFormatFloat; 4575548cb50SEnrico Granata 4585548cb50SEnrico Granata default: 4595548cb50SEnrico Granata return lldb::eFormatInvalid; 4605548cb50SEnrico Granata } 4615548cb50SEnrico Granata } 4625548cb50SEnrico Granata 463b9c1b51eSKate Stone bool FormatManager::ShouldPrintAsOneLiner(ValueObject &valobj) { 464553fad5cSEnrico Granata // if settings say no oneline whatsoever 465b9c1b51eSKate Stone if (valobj.GetTargetSP().get() && 466a6682a41SJonas Devlieghere !valobj.GetTargetSP()->GetDebugger().GetAutoOneLineSummaries()) 467553fad5cSEnrico Granata return false; // then don't oneline 468553fad5cSEnrico Granata 46942fa4af8SEnrico Granata // if this object has a summary, then ask the summary 470a29cb0baSEnrico Granata if (valobj.GetSummaryFormat().get() != nullptr) 47142fa4af8SEnrico Granata return valobj.GetSummaryFormat()->IsOneLiner(); 472a29cb0baSEnrico Granata 473a29cb0baSEnrico Granata // no children, no party 474a29cb0baSEnrico Granata if (valobj.GetNumChildren() == 0) 475a29cb0baSEnrico Granata return false; 476a29cb0baSEnrico Granata 47705097246SAdrian Prantl // ask the type if it has any opinion about this eLazyBoolCalculate == no 47805097246SAdrian Prantl // opinion; other values should be self explanatory 4799c63f99aSEnrico Granata CompilerType compiler_type(valobj.GetCompilerType()); 480b9c1b51eSKate Stone if (compiler_type.IsValid()) { 481b9c1b51eSKate Stone switch (compiler_type.ShouldPrintAsOneLiner(&valobj)) { 4829c63f99aSEnrico Granata case eLazyBoolNo: 4839c63f99aSEnrico Granata return false; 4849c63f99aSEnrico Granata case eLazyBoolYes: 4859c63f99aSEnrico Granata return true; 4869c63f99aSEnrico Granata case eLazyBoolCalculate: 4879c63f99aSEnrico Granata break; 4889c63f99aSEnrico Granata } 4899c63f99aSEnrico Granata } 4909c63f99aSEnrico Granata 491a29cb0baSEnrico Granata size_t total_children_name_len = 0; 492a29cb0baSEnrico Granata 493b9c1b51eSKate Stone for (size_t idx = 0; idx < valobj.GetNumChildren(); idx++) { 494ddac7611SEnrico Granata bool is_synth_val = false; 495a29cb0baSEnrico Granata ValueObjectSP child_sp(valobj.GetChildAtIndex(idx, true)); 496a29cb0baSEnrico Granata // something is wrong here - bail out 497a29cb0baSEnrico Granata if (!child_sp) 498a29cb0baSEnrico Granata return false; 4996500061eSEnrico Granata 5006500061eSEnrico Granata // also ask the child's type if it has any opinion 5016500061eSEnrico Granata CompilerType child_compiler_type(child_sp->GetCompilerType()); 502b9c1b51eSKate Stone if (child_compiler_type.IsValid()) { 503b9c1b51eSKate Stone switch (child_compiler_type.ShouldPrintAsOneLiner(child_sp.get())) { 5046500061eSEnrico Granata case eLazyBoolYes: 5056500061eSEnrico Granata // an opinion of yes is only binding for the child, so keep going 5066500061eSEnrico Granata case eLazyBoolCalculate: 5076500061eSEnrico Granata break; 5086500061eSEnrico Granata case eLazyBoolNo: 5096500061eSEnrico Granata // but if the child says no, then it's a veto on the whole thing 5106500061eSEnrico Granata return false; 5116500061eSEnrico Granata } 5126500061eSEnrico Granata } 5136500061eSEnrico Granata 514b9c1b51eSKate Stone // if we decided to define synthetic children for a type, we probably care 51505097246SAdrian Prantl // enough to show them, but avoid nesting children in children 516b9c1b51eSKate Stone if (child_sp->GetSyntheticChildren().get() != nullptr) { 517ddac7611SEnrico Granata ValueObjectSP synth_sp(child_sp->GetSyntheticValue()); 518ddac7611SEnrico Granata // wait.. wat? just get out of here.. 519ddac7611SEnrico Granata if (!synth_sp) 520a29cb0baSEnrico Granata return false; 521ddac7611SEnrico Granata // but if we only have them to provide a value, keep going 522a6682a41SJonas Devlieghere if (!synth_sp->MightHaveChildren() && 523b9c1b51eSKate Stone synth_sp->DoesProvideSyntheticValue()) 524ddac7611SEnrico Granata is_synth_val = true; 525ddac7611SEnrico Granata else 526ddac7611SEnrico Granata return false; 527ddac7611SEnrico Granata } 528a29cb0baSEnrico Granata 529a29cb0baSEnrico Granata total_children_name_len += child_sp->GetName().GetLength(); 530a29cb0baSEnrico Granata 531a29cb0baSEnrico Granata // 50 itself is a "randomly" chosen number - the idea is that 532a29cb0baSEnrico Granata // overly long structs should not get this treatment 533a29cb0baSEnrico Granata // FIXME: maybe make this a user-tweakable setting? 534a29cb0baSEnrico Granata if (total_children_name_len > 50) 535a29cb0baSEnrico Granata return false; 536a29cb0baSEnrico Granata 537a29cb0baSEnrico Granata // if a summary is there.. 538b9c1b51eSKate Stone if (child_sp->GetSummaryFormat()) { 539a29cb0baSEnrico Granata // and it wants children, then bail out 5408a068e6cSEnrico Granata if (child_sp->GetSummaryFormat()->DoesPrintChildren(child_sp.get())) 541a29cb0baSEnrico Granata return false; 542a29cb0baSEnrico Granata } 543a29cb0baSEnrico Granata 544c89e4ca3SEnrico Granata // if this child has children.. 545b9c1b51eSKate Stone if (child_sp->GetNumChildren()) { 546a29cb0baSEnrico Granata // ...and no summary... 547b9c1b51eSKate Stone // (if it had a summary and the summary wanted children, we would have 548b9c1b51eSKate Stone // bailed out anyway 549b9c1b51eSKate Stone // so this only makes us bail out if this has no summary and we would 550b9c1b51eSKate Stone // then print children) 551b9c1b51eSKate Stone if (!child_sp->GetSummaryFormat() && !is_synth_val) // but again only do 552b9c1b51eSKate Stone // that if not a 553b9c1b51eSKate Stone // synthetic valued 554b9c1b51eSKate Stone // child 555a29cb0baSEnrico Granata return false; // then bail out 556a29cb0baSEnrico Granata } 557a29cb0baSEnrico Granata } 558a29cb0baSEnrico Granata return true; 559a29cb0baSEnrico Granata } 560a29cb0baSEnrico Granata 5610e4c4821SAdrian Prantl ConstString FormatManager::GetValidTypeName(ConstString type) { 5625548cb50SEnrico Granata return ::GetValidTypeName_Impl(type); 5635548cb50SEnrico Granata } 5645548cb50SEnrico Granata 565b9c1b51eSKate Stone ConstString FormatManager::GetTypeForCache(ValueObject &valobj, 566b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 567b9c1b51eSKate Stone ValueObjectSP valobj_sp = valobj.GetQualifiedRepresentationIfAvailable( 568b9c1b51eSKate Stone use_dynamic, valobj.IsSynthetic()); 56978f05d35SDavide Italiano if (valobj_sp && valobj_sp->GetCompilerType().IsValid()) { 57078f05d35SDavide Italiano if (!valobj_sp->GetCompilerType().IsMeaninglessWithoutDynamicResolution()) 571b3f0c340SEnrico Granata return valobj_sp->GetQualifiedTypeName(); 57278f05d35SDavide Italiano } 5735548cb50SEnrico Granata return ConstString(); 5745548cb50SEnrico Granata } 5755548cb50SEnrico Granata 576d3233c1eSEnrico Granata std::vector<lldb::LanguageType> 577b9c1b51eSKate Stone FormatManager::GetCandidateLanguages(ValueObject &valobj) { 578980c0484SEnrico Granata lldb::LanguageType lang_type = valobj.GetObjectRuntimeLanguage(); 579ac49453bSEnrico Granata return GetCandidateLanguages(lang_type); 580ac49453bSEnrico Granata } 581ac49453bSEnrico Granata 582ac49453bSEnrico Granata std::vector<lldb::LanguageType> 583b9c1b51eSKate Stone FormatManager::GetCandidateLanguages(lldb::LanguageType lang_type) { 584b9c1b51eSKate Stone switch (lang_type) { 58533e97e63SEnrico Granata case lldb::eLanguageTypeC: 58633e97e63SEnrico Granata case lldb::eLanguageTypeC89: 58733e97e63SEnrico Granata case lldb::eLanguageTypeC99: 58833e97e63SEnrico Granata case lldb::eLanguageTypeC11: 58933e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus: 59033e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_03: 59133e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_11: 59233e97e63SEnrico Granata case lldb::eLanguageTypeC_plus_plus_14: 593170c395eSEnrico Granata return {lldb::eLanguageTypeC_plus_plus, lldb::eLanguageTypeObjC}; 594980c0484SEnrico Granata default: 595980c0484SEnrico Granata return {lang_type}; 596980c0484SEnrico Granata } 597980c0484SEnrico Granata } 598980c0484SEnrico Granata 599980c0484SEnrico Granata LanguageCategory * 600b9c1b51eSKate Stone FormatManager::GetCategoryForLanguage(lldb::LanguageType lang_type) { 60116ff8604SSaleem Abdulrasool std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex); 602b9c1b51eSKate Stone auto iter = m_language_categories_map.find(lang_type), 603b9c1b51eSKate Stone end = m_language_categories_map.end(); 604980c0484SEnrico Granata if (iter != end) 605980c0484SEnrico Granata return iter->second.get(); 606980c0484SEnrico Granata LanguageCategory *lang_category = new LanguageCategory(lang_type); 607b9c1b51eSKate Stone m_language_categories_map[lang_type] = 608b9c1b51eSKate Stone LanguageCategory::UniquePointer(lang_category); 609980c0484SEnrico Granata return lang_category; 610980c0484SEnrico Granata } 611980c0484SEnrico Granata 612ecd02bc1SEnrico Granata lldb::TypeFormatImplSP 613b9c1b51eSKate Stone FormatManager::GetHardcodedFormat(FormattersMatchData &match_data) { 6147cb59e1aSEnrico Granata TypeFormatImplSP retval_sp; 6157cb59e1aSEnrico Granata 616b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 617b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6188a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 6197cb59e1aSEnrico Granata break; 620ecd02bc1SEnrico Granata } 6217cb59e1aSEnrico Granata } 6227cb59e1aSEnrico Granata 6237cb59e1aSEnrico Granata return retval_sp; 624686f3debSEnrico Granata } 625686f3debSEnrico Granata 626852cc954SEnrico Granata lldb::TypeFormatImplSP 627852cc954SEnrico Granata FormatManager::GetFormat(ValueObject &valobj, 628b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 6298a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 6308a9a8f39SEnrico Granata 631852cc954SEnrico Granata TypeFormatImplSP retval; 6324edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 633b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 63463e5fb76SJonas Devlieghere LLDB_LOGF(log, 635b9c1b51eSKate Stone "\n\n[FormatManager::GetFormat] Looking into cache for type %s", 636b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 637b9c1b51eSKate Stone if (m_format_cache.GetFormat(match_data.GetTypeForCache(), retval)) { 638b9c1b51eSKate Stone if (log) { 63963e5fb76SJonas Devlieghere LLDB_LOGF( 64063e5fb76SJonas Devlieghere log, "[FormatManager::GetFormat] Cache search success. Returning."); 6416302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 642b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 643b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 64452b4b6cdSEnrico Granata } 64552b4b6cdSEnrico Granata return retval; 64652b4b6cdSEnrico Granata } 64763e5fb76SJonas Devlieghere LLDB_LOGF( 64863e5fb76SJonas Devlieghere log, 649b9c1b51eSKate Stone "[FormatManager::GetFormat] Cache search failed. Going normal route"); 65052b4b6cdSEnrico Granata } 651980c0484SEnrico Granata 6528a9a8f39SEnrico Granata retval = m_categories_map.GetFormat(match_data); 653b9c1b51eSKate Stone if (!retval) { 65463e5fb76SJonas Devlieghere LLDB_LOGF(log, 65563e5fb76SJonas Devlieghere "[FormatManager::GetFormat] Search failed. Giving language a " 656b9c1b51eSKate Stone "chance."); 657b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 658b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6598a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 660980c0484SEnrico Granata break; 661980c0484SEnrico Granata } 662980c0484SEnrico Granata } 663b9c1b51eSKate Stone if (retval) { 66463e5fb76SJonas Devlieghere LLDB_LOGF( 66563e5fb76SJonas Devlieghere log, 666b9c1b51eSKate Stone "[FormatManager::GetFormat] Language search success. Returning."); 667980c0484SEnrico Granata return retval; 668980c0484SEnrico Granata } 669980c0484SEnrico Granata } 670b9c1b51eSKate Stone if (!retval) { 67163e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetFormat] Search failed. Giving hardcoded " 672b9c1b51eSKate Stone "a chance."); 6738a9a8f39SEnrico Granata retval = GetHardcodedFormat(match_data); 674686f3debSEnrico Granata } 675d4cb1dddSEnrico Granata 676b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 67763e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetFormat] Caching %p for type %s", 678324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 6798a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 6808a9a8f39SEnrico Granata m_format_cache.SetFormat(match_data.GetTypeForCache(), retval); 68152b4b6cdSEnrico Granata } 6826302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 683b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 684852cc954SEnrico Granata return retval; 685852cc954SEnrico Granata } 686852cc954SEnrico Granata 687ecd02bc1SEnrico Granata lldb::TypeSummaryImplSP 688b9c1b51eSKate Stone FormatManager::GetHardcodedSummaryFormat(FormattersMatchData &match_data) { 6897cb59e1aSEnrico Granata TypeSummaryImplSP retval_sp; 6907cb59e1aSEnrico Granata 691b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 692b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 6938a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 6947cb59e1aSEnrico Granata break; 695ecd02bc1SEnrico Granata } 6967cb59e1aSEnrico Granata } 6977cb59e1aSEnrico Granata 6987cb59e1aSEnrico Granata return retval_sp; 699686f3debSEnrico Granata } 700686f3debSEnrico Granata 7015548cb50SEnrico Granata lldb::TypeSummaryImplSP 7025548cb50SEnrico Granata FormatManager::GetSummaryFormat(ValueObject &valobj, 703b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 7048a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 7058a9a8f39SEnrico Granata 7065548cb50SEnrico Granata TypeSummaryImplSP retval; 7074edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 708b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 70963e5fb76SJonas Devlieghere LLDB_LOGF(log, 71063e5fb76SJonas Devlieghere "\n\n[FormatManager::GetSummaryFormat] Looking into cache " 711b9c1b51eSKate Stone "for type %s", 712b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 713b9c1b51eSKate Stone if (m_format_cache.GetSummary(match_data.GetTypeForCache(), retval)) { 714b9c1b51eSKate Stone if (log) { 71563e5fb76SJonas Devlieghere LLDB_LOGF(log, 71663e5fb76SJonas Devlieghere "[FormatManager::GetSummaryFormat] Cache search success. " 717b9c1b51eSKate Stone "Returning."); 7186302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 719b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 720b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 72168ae4117SEnrico Granata } 7225548cb50SEnrico Granata return retval; 72368ae4117SEnrico Granata } 72463e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Cache search failed. " 725b9c1b51eSKate Stone "Going normal route"); 7265548cb50SEnrico Granata } 727980c0484SEnrico Granata 7288a9a8f39SEnrico Granata retval = m_categories_map.GetSummaryFormat(match_data); 729b9c1b51eSKate Stone if (!retval) { 73063e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Search failed. Giving " 731b9c1b51eSKate Stone "language a chance."); 732b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 733b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 7348a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 735980c0484SEnrico Granata break; 736980c0484SEnrico Granata } 737980c0484SEnrico Granata } 738b9c1b51eSKate Stone if (retval) { 73963e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Language search " 740b9c1b51eSKate Stone "success. Returning."); 741980c0484SEnrico Granata return retval; 742980c0484SEnrico Granata } 743980c0484SEnrico Granata } 744b9c1b51eSKate Stone if (!retval) { 74563e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Search failed. Giving " 746b9c1b51eSKate Stone "hardcoded a chance."); 7478a9a8f39SEnrico Granata retval = GetHardcodedSummaryFormat(match_data); 748686f3debSEnrico Granata } 749d4cb1dddSEnrico Granata 750b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 75163e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSummaryFormat] Caching %p for type %s", 752324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 7538a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 7548a9a8f39SEnrico Granata m_format_cache.SetSummary(match_data.GetTypeForCache(), retval); 7555548cb50SEnrico Granata } 7566302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 757b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 7585548cb50SEnrico Granata return retval; 7595548cb50SEnrico Granata } 7605548cb50SEnrico Granata 761ecd02bc1SEnrico Granata lldb::SyntheticChildrenSP 762b9c1b51eSKate Stone FormatManager::GetHardcodedSyntheticChildren(FormattersMatchData &match_data) { 7637cb59e1aSEnrico Granata SyntheticChildrenSP retval_sp; 7647cb59e1aSEnrico Granata 765b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 766b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 7678a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 7687cb59e1aSEnrico Granata break; 769ecd02bc1SEnrico Granata } 7707cb59e1aSEnrico Granata } 7717cb59e1aSEnrico Granata 7727cb59e1aSEnrico Granata return retval_sp; 773686f3debSEnrico Granata } 774686f3debSEnrico Granata 7755548cb50SEnrico Granata lldb::SyntheticChildrenSP 7765548cb50SEnrico Granata FormatManager::GetSyntheticChildren(ValueObject &valobj, 777b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 7788a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 7798a9a8f39SEnrico Granata 7805548cb50SEnrico Granata SyntheticChildrenSP retval; 7814edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 782b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 78363e5fb76SJonas Devlieghere LLDB_LOGF(log, 78463e5fb76SJonas Devlieghere "\n\n[FormatManager::GetSyntheticChildren] Looking into " 785b9c1b51eSKate Stone "cache for type %s", 786b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 787b9c1b51eSKate Stone if (m_format_cache.GetSynthetic(match_data.GetTypeForCache(), retval)) { 788b9c1b51eSKate Stone if (log) { 78963e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Cache search " 790b9c1b51eSKate Stone "success. Returning."); 7916302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 7926302bf6aSPavel Labath m_format_cache.GetCacheHits(), 7936302bf6aSPavel Labath m_format_cache.GetCacheMisses()); 79468ae4117SEnrico Granata } 79568ae4117SEnrico Granata return retval; 79668ae4117SEnrico Granata } 79763e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Cache search failed. " 798b9c1b51eSKate Stone "Going normal route"); 7995548cb50SEnrico Granata } 800980c0484SEnrico Granata 8018a9a8f39SEnrico Granata retval = m_categories_map.GetSyntheticChildren(match_data); 802b9c1b51eSKate Stone if (!retval) { 80363e5fb76SJonas Devlieghere LLDB_LOGF(log, 80463e5fb76SJonas Devlieghere "[FormatManager::GetSyntheticChildren] Search failed. Giving " 805b9c1b51eSKate Stone "language a chance."); 806b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 807b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 8088a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 809980c0484SEnrico Granata break; 810980c0484SEnrico Granata } 811980c0484SEnrico Granata } 812b9c1b51eSKate Stone if (retval) { 81363e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetSyntheticChildren] Language search " 814b9c1b51eSKate Stone "success. Returning."); 815980c0484SEnrico Granata return retval; 816980c0484SEnrico Granata } 817980c0484SEnrico Granata } 818b9c1b51eSKate Stone if (!retval) { 81963e5fb76SJonas Devlieghere LLDB_LOGF(log, 82063e5fb76SJonas Devlieghere "[FormatManager::GetSyntheticChildren] Search failed. Giving " 821b9c1b51eSKate Stone "hardcoded a chance."); 8228a9a8f39SEnrico Granata retval = GetHardcodedSyntheticChildren(match_data); 823686f3debSEnrico Granata } 824d4cb1dddSEnrico Granata 825b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 82663e5fb76SJonas Devlieghere LLDB_LOGF(log, 827b9c1b51eSKate Stone "[FormatManager::GetSyntheticChildren] Caching %p for type %s", 828324a1036SSaleem Abdulrasool static_cast<void *>(retval.get()), 8298a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 8308a9a8f39SEnrico Granata m_format_cache.SetSynthetic(match_data.GetTypeForCache(), retval); 8315548cb50SEnrico Granata } 8326302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 833b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 8345548cb50SEnrico Granata return retval; 8355548cb50SEnrico Granata } 8365548cb50SEnrico Granata 837c582713cSEnrico Granata lldb::TypeValidatorImplSP 838c582713cSEnrico Granata FormatManager::GetValidator(ValueObject &valobj, 839b9c1b51eSKate Stone lldb::DynamicValueType use_dynamic) { 8408a9a8f39SEnrico Granata FormattersMatchData match_data(valobj, use_dynamic); 8418a9a8f39SEnrico Granata 842c582713cSEnrico Granata TypeValidatorImplSP retval; 8434edfef45SEnrico Granata Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS)); 844b9c1b51eSKate Stone if (match_data.GetTypeForCache()) { 84563e5fb76SJonas Devlieghere LLDB_LOGF( 84663e5fb76SJonas Devlieghere log, "\n\n[FormatManager::GetValidator] Looking into cache for type %s", 847b9c1b51eSKate Stone match_data.GetTypeForCache().AsCString("<invalid>")); 848b9c1b51eSKate Stone if (m_format_cache.GetValidator(match_data.GetTypeForCache(), retval)) { 849b9c1b51eSKate Stone if (log) { 85063e5fb76SJonas Devlieghere LLDB_LOGF( 85163e5fb76SJonas Devlieghere log, 852b9c1b51eSKate Stone "[FormatManager::GetValidator] Cache search success. Returning."); 8536302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 854b9c1b51eSKate Stone m_format_cache.GetCacheHits(), 855b9c1b51eSKate Stone m_format_cache.GetCacheMisses()); 856c582713cSEnrico Granata } 857c582713cSEnrico Granata return retval; 858c582713cSEnrico Granata } 85963e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Cache search failed. Going " 860b9c1b51eSKate Stone "normal route"); 861c582713cSEnrico Granata } 862980c0484SEnrico Granata 8638a9a8f39SEnrico Granata retval = m_categories_map.GetValidator(match_data); 864b9c1b51eSKate Stone if (!retval) { 86563e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Search failed. Giving " 866b9c1b51eSKate Stone "language a chance."); 867b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 868b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 8698a9a8f39SEnrico Granata if (lang_category->Get(match_data, retval)) 870980c0484SEnrico Granata break; 871980c0484SEnrico Granata } 872980c0484SEnrico Granata } 873b9c1b51eSKate Stone if (retval) { 87463e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Language search success. " 875b9c1b51eSKate Stone "Returning."); 876980c0484SEnrico Granata return retval; 877980c0484SEnrico Granata } 878980c0484SEnrico Granata } 879b9c1b51eSKate Stone if (!retval) { 88063e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Search failed. Giving " 881b9c1b51eSKate Stone "hardcoded a chance."); 8828a9a8f39SEnrico Granata retval = GetHardcodedValidator(match_data); 883c582713cSEnrico Granata } 884d4cb1dddSEnrico Granata 885b9c1b51eSKate Stone if (match_data.GetTypeForCache() && (!retval || !retval->NonCacheable())) { 88663e5fb76SJonas Devlieghere LLDB_LOGF(log, "[FormatManager::GetValidator] Caching %p for type %s", 887c582713cSEnrico Granata static_cast<void *>(retval.get()), 8888a9a8f39SEnrico Granata match_data.GetTypeForCache().AsCString("<invalid>")); 8898a9a8f39SEnrico Granata m_format_cache.SetValidator(match_data.GetTypeForCache(), retval); 890c582713cSEnrico Granata } 8916302bf6aSPavel Labath LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}", 892b9c1b51eSKate Stone m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses()); 893c582713cSEnrico Granata return retval; 894c582713cSEnrico Granata } 895c582713cSEnrico Granata 896c582713cSEnrico Granata lldb::TypeValidatorImplSP 897b9c1b51eSKate Stone FormatManager::GetHardcodedValidator(FormattersMatchData &match_data) { 8987cb59e1aSEnrico Granata TypeValidatorImplSP retval_sp; 8997cb59e1aSEnrico Granata 900b9c1b51eSKate Stone for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) { 901b9c1b51eSKate Stone if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) { 9028a9a8f39SEnrico Granata if (lang_category->GetHardcoded(*this, match_data, retval_sp)) 9037cb59e1aSEnrico Granata break; 904c582713cSEnrico Granata } 9057cb59e1aSEnrico Granata } 9067cb59e1aSEnrico Granata 9077cb59e1aSEnrico Granata return retval_sp; 908c582713cSEnrico Granata } 909c582713cSEnrico Granata 91016ff8604SSaleem Abdulrasool FormatManager::FormatManager() 911b9c1b51eSKate Stone : m_last_revision(0), m_format_cache(), m_language_categories_mutex(), 912b9c1b51eSKate Stone m_language_categories_map(), m_named_summaries_map(this), 913b9c1b51eSKate Stone m_categories_map(this), m_default_category_name(ConstString("default")), 9145548cb50SEnrico Granata m_system_category_name(ConstString("system")), 915b9c1b51eSKate Stone m_vectortypes_category_name(ConstString("VectorTypes")) { 9165548cb50SEnrico Granata LoadSystemFormatters(); 917170c395eSEnrico Granata LoadVectorFormatters(); 9185548cb50SEnrico Granata 919b9c1b51eSKate Stone EnableCategory(m_vectortypes_category_name, TypeCategoryMap::Last, 920b9c1b51eSKate Stone lldb::eLanguageTypeObjC_plus_plus); 921b9c1b51eSKate Stone EnableCategory(m_system_category_name, TypeCategoryMap::Last, 922b9c1b51eSKate Stone lldb::eLanguageTypeObjC_plus_plus); 9235548cb50SEnrico Granata } 9245548cb50SEnrico Granata 925b9c1b51eSKate Stone void FormatManager::LoadSystemFormatters() { 9265548cb50SEnrico Granata TypeSummaryImpl::Flags string_flags; 9270337c27fSEnrico Granata string_flags.SetCascades(true) 9285548cb50SEnrico Granata .SetSkipPointers(true) 9295548cb50SEnrico Granata .SetSkipReferences(false) 9305548cb50SEnrico Granata .SetDontShowChildren(true) 9315548cb50SEnrico Granata .SetDontShowValue(false) 9325548cb50SEnrico Granata .SetShowMembersOneLiner(false) 9335548cb50SEnrico Granata .SetHideItemNames(false); 9345548cb50SEnrico Granata 935bc2c2b01SEnrico Granata TypeSummaryImpl::Flags string_array_flags; 936d2911633SEnrico Granata string_array_flags.SetCascades(true) 9375548cb50SEnrico Granata .SetSkipPointers(true) 9385548cb50SEnrico Granata .SetSkipReferences(false) 939320dcf68SEnrico Granata .SetDontShowChildren(true) 9405548cb50SEnrico Granata .SetDontShowValue(true) 9415548cb50SEnrico Granata .SetShowMembersOneLiner(false) 942bc2c2b01SEnrico Granata .SetHideItemNames(false); 943bc2c2b01SEnrico Granata 944b9c1b51eSKate Stone lldb::TypeSummaryImplSP string_format( 945b9c1b51eSKate Stone new StringSummaryFormat(string_flags, "${var%s}")); 946bc2c2b01SEnrico Granata 947b9c1b51eSKate Stone lldb::TypeSummaryImplSP string_array_format( 948b9c1b51eSKate Stone new StringSummaryFormat(string_array_flags, "${var%s}")); 949bc2c2b01SEnrico Granata 950b9c1b51eSKate Stone lldb::RegularExpressionSP any_size_char_arr( 95195eae423SZachary Turner new RegularExpression(llvm::StringRef("char \\[[0-9]+\\]"))); 952b9c1b51eSKate Stone lldb::RegularExpressionSP any_size_wchar_arr( 95395eae423SZachary Turner new RegularExpression(llvm::StringRef("wchar_t \\[[0-9]+\\]"))); 9545548cb50SEnrico Granata 955b9c1b51eSKate Stone TypeCategoryImpl::SharedPointer sys_category_sp = 956b9c1b51eSKate Stone GetCategory(m_system_category_name); 9575548cb50SEnrico Granata 958b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("char *"), 959b9c1b51eSKate Stone string_format); 960b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add( 961b9c1b51eSKate Stone ConstString("unsigned char *"), string_format); 962b9c1b51eSKate Stone sys_category_sp->GetRegexTypeSummariesContainer()->Add(any_size_char_arr, 963b9c1b51eSKate Stone string_array_format); 9645548cb50SEnrico Granata 965b9c1b51eSKate Stone lldb::TypeSummaryImplSP ostype_summary( 966b9c1b51eSKate Stone new StringSummaryFormat(TypeSummaryImpl::Flags() 967b9c1b51eSKate Stone .SetCascades(false) 9685548cb50SEnrico Granata .SetSkipPointers(true) 9695548cb50SEnrico Granata .SetSkipReferences(true) 9705548cb50SEnrico Granata .SetDontShowChildren(true) 9715548cb50SEnrico Granata .SetDontShowValue(false) 9725548cb50SEnrico Granata .SetShowMembersOneLiner(false) 9735548cb50SEnrico Granata .SetHideItemNames(false), 9745548cb50SEnrico Granata "${var%O}")); 9755548cb50SEnrico Granata 976b9c1b51eSKate Stone sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("OSType"), 977b9c1b51eSKate Stone ostype_summary); 9785548cb50SEnrico Granata 9794cc21772SEnrico Granata TypeFormatImpl::Flags fourchar_flags; 980b9c1b51eSKate Stone fourchar_flags.SetCascades(true).SetSkipPointers(true).SetSkipReferences( 981b9c1b51eSKate Stone true); 9824cc21772SEnrico Granata 983b9c1b51eSKate Stone AddFormat(sys_category_sp, lldb::eFormatOSType, ConstString("FourCharCode"), 984b9c1b51eSKate Stone fourchar_flags); 9855548cb50SEnrico Granata } 9865548cb50SEnrico Granata 987b9c1b51eSKate Stone void FormatManager::LoadVectorFormatters() { 988b9c1b51eSKate Stone TypeCategoryImpl::SharedPointer vectors_category_sp = 989b9c1b51eSKate Stone GetCategory(m_vectortypes_category_name); 9905548cb50SEnrico Granata 9915548cb50SEnrico Granata TypeSummaryImpl::Flags vector_flags; 9925548cb50SEnrico Granata vector_flags.SetCascades(true) 9935548cb50SEnrico Granata .SetSkipPointers(true) 9945548cb50SEnrico Granata .SetSkipReferences(false) 9955548cb50SEnrico Granata .SetDontShowChildren(true) 9965548cb50SEnrico Granata .SetDontShowValue(false) 9975548cb50SEnrico Granata .SetShowMembersOneLiner(true) 9985548cb50SEnrico Granata .SetHideItemNames(true); 9995548cb50SEnrico Granata 1000b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "${var.uint128}", 1001b9c1b51eSKate Stone ConstString("builtin_type_vec128"), vector_flags); 10025548cb50SEnrico Granata 1003b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("float [4]"), 10045548cb50SEnrico Granata vector_flags); 1005b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("int32_t [4]"), 10065548cb50SEnrico Granata vector_flags); 1007b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("int16_t [8]"), 10085548cb50SEnrico Granata vector_flags); 1009b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vDouble"), 10105548cb50SEnrico Granata vector_flags); 1011b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vFloat"), 10125548cb50SEnrico Granata vector_flags); 1013b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt8"), 10145548cb50SEnrico Granata vector_flags); 1015b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt16"), 10165548cb50SEnrico Granata vector_flags); 1017b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vSInt32"), 10185548cb50SEnrico Granata vector_flags); 1019b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"), 10205548cb50SEnrico Granata vector_flags); 1021b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt8"), 10225548cb50SEnrico Granata vector_flags); 1023b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"), 10245548cb50SEnrico Granata vector_flags); 1025b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vUInt32"), 10265548cb50SEnrico Granata vector_flags); 1027b9c1b51eSKate Stone AddStringSummary(vectors_category_sp, "", ConstString("vBool32"), 10285548cb50SEnrico Granata vector_flags); 10295548cb50SEnrico Granata } 1030