180814287SRaphael Isemann //===-- FormatManager.cpp -------------------------------------------------===//
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 
115548cb50SEnrico Granata #include "lldb/Core/Debugger.h"
12df7e79e6SEnrico Granata #include "lldb/DataFormatters/FormattersHelpers.h"
13980c0484SEnrico Granata #include "lldb/DataFormatters/LanguageCategory.h"
145548cb50SEnrico Granata #include "lldb/Target/ExecutionContext.h"
1533e97e63SEnrico Granata #include "lldb/Target/Language.h"
16c34698a8SPavel Labath #include "lldb/Utility/LLDBLog.h"
176f9e6901SZachary Turner #include "lldb/Utility/Log.h"
18c34698a8SPavel Labath #include "llvm/ADT/STLExtras.h"
19980c0484SEnrico Granata 
205548cb50SEnrico Granata using namespace lldb;
215548cb50SEnrico Granata using namespace lldb_private;
22df7e79e6SEnrico Granata using namespace lldb_private::formatters;
235548cb50SEnrico Granata 
24b9c1b51eSKate Stone struct FormatInfo {
255548cb50SEnrico Granata   Format format;
26b9c1b51eSKate Stone   const char format_char;  // One or more format characters that can be used for
27b9c1b51eSKate Stone                            // this format.
28b9c1b51eSKate Stone   const char *format_name; // Long format name that can be used to specify the
29b9c1b51eSKate Stone                            // current format
305548cb50SEnrico Granata };
315548cb50SEnrico Granata 
3212002fbdSJonas Devlieghere static constexpr FormatInfo g_format_infos[] = {
335548cb50SEnrico Granata     {eFormatDefault, '\0', "default"},
345548cb50SEnrico Granata     {eFormatBoolean, 'B', "boolean"},
355548cb50SEnrico Granata     {eFormatBinary, 'b', "binary"},
365548cb50SEnrico Granata     {eFormatBytes, 'y', "bytes"},
375548cb50SEnrico Granata     {eFormatBytesWithASCII, 'Y', "bytes with ASCII"},
385548cb50SEnrico Granata     {eFormatChar, 'c', "character"},
395548cb50SEnrico Granata     {eFormatCharPrintable, 'C', "printable character"},
405548cb50SEnrico Granata     {eFormatComplexFloat, 'F', "complex float"},
415548cb50SEnrico Granata     {eFormatCString, 's', "c-string"},
425548cb50SEnrico Granata     {eFormatDecimal, 'd', "decimal"},
435548cb50SEnrico Granata     {eFormatEnum, 'E', "enumeration"},
445548cb50SEnrico Granata     {eFormatHex, 'x', "hex"},
455548cb50SEnrico Granata     {eFormatHexUppercase, 'X', "uppercase hex"},
465548cb50SEnrico Granata     {eFormatFloat, 'f', "float"},
475548cb50SEnrico Granata     {eFormatOctal, 'o', "octal"},
485548cb50SEnrico Granata     {eFormatOSType, 'O', "OSType"},
495548cb50SEnrico Granata     {eFormatUnicode16, 'U', "unicode16"},
505548cb50SEnrico Granata     {eFormatUnicode32, '\0', "unicode32"},
515548cb50SEnrico Granata     {eFormatUnsigned, 'u', "unsigned decimal"},
525548cb50SEnrico Granata     {eFormatPointer, 'p', "pointer"},
535548cb50SEnrico Granata     {eFormatVectorOfChar, '\0', "char[]"},
545548cb50SEnrico Granata     {eFormatVectorOfSInt8, '\0', "int8_t[]"},
555548cb50SEnrico Granata     {eFormatVectorOfUInt8, '\0', "uint8_t[]"},
565548cb50SEnrico Granata     {eFormatVectorOfSInt16, '\0', "int16_t[]"},
575548cb50SEnrico Granata     {eFormatVectorOfUInt16, '\0', "uint16_t[]"},
585548cb50SEnrico Granata     {eFormatVectorOfSInt32, '\0', "int32_t[]"},
595548cb50SEnrico Granata     {eFormatVectorOfUInt32, '\0', "uint32_t[]"},
605548cb50SEnrico Granata     {eFormatVectorOfSInt64, '\0', "int64_t[]"},
615548cb50SEnrico Granata     {eFormatVectorOfUInt64, '\0', "uint64_t[]"},
62a0f08674SEwan Crawford     {eFormatVectorOfFloat16, '\0', "float16[]"},
635548cb50SEnrico Granata     {eFormatVectorOfFloat32, '\0', "float32[]"},
645548cb50SEnrico Granata     {eFormatVectorOfFloat64, '\0', "float64[]"},
655548cb50SEnrico Granata     {eFormatVectorOfUInt128, '\0', "uint128_t[]"},
665548cb50SEnrico Granata     {eFormatComplexInteger, 'I', "complex integer"},
675548cb50SEnrico Granata     {eFormatCharArray, 'a', "character array"},
685548cb50SEnrico Granata     {eFormatAddressInfo, 'A', "address"},
695548cb50SEnrico Granata     {eFormatHexFloat, '\0', "hex float"},
705548cb50SEnrico Granata     {eFormatInstruction, 'i', "instruction"},
712621f7bdSJonas Devlieghere     {eFormatVoid, 'v', "void"},
722621f7bdSJonas Devlieghere     {eFormatUnicode8, 'u', "unicode8"},
732621f7bdSJonas Devlieghere };
745548cb50SEnrico Granata 
7512002fbdSJonas Devlieghere static_assert((sizeof(g_format_infos) / sizeof(g_format_infos[0])) ==
7612002fbdSJonas Devlieghere                   kNumFormats,
7712002fbdSJonas Devlieghere               "All formats must have a corresponding info entry.");
7812002fbdSJonas Devlieghere 
79f15014ffSBenjamin Kramer static uint32_t g_num_format_infos = llvm::array_lengthof(g_format_infos);
805548cb50SEnrico Granata 
GetFormatFromFormatChar(char format_char,Format & format)81b9c1b51eSKate Stone static bool GetFormatFromFormatChar(char format_char, Format &format) {
82b9c1b51eSKate Stone   for (uint32_t i = 0; i < g_num_format_infos; ++i) {
83b9c1b51eSKate Stone     if (g_format_infos[i].format_char == format_char) {
845548cb50SEnrico Granata       format = g_format_infos[i].format;
855548cb50SEnrico Granata       return true;
865548cb50SEnrico Granata     }
875548cb50SEnrico Granata   }
885548cb50SEnrico Granata   format = eFormatInvalid;
895548cb50SEnrico Granata   return false;
905548cb50SEnrico Granata }
915548cb50SEnrico Granata 
GetFormatFromFormatName(llvm::StringRef format_name,bool partial_match_ok,Format & format)92*13a3b0bbSPavel Labath static bool GetFormatFromFormatName(llvm::StringRef format_name,
93b9c1b51eSKate Stone                                     bool partial_match_ok, Format &format) {
945548cb50SEnrico Granata   uint32_t i;
95b9c1b51eSKate Stone   for (i = 0; i < g_num_format_infos; ++i) {
96*13a3b0bbSPavel Labath     if (format_name.equals_insensitive(g_format_infos[i].format_name)) {
975548cb50SEnrico Granata       format = g_format_infos[i].format;
985548cb50SEnrico Granata       return true;
995548cb50SEnrico Granata     }
1005548cb50SEnrico Granata   }
1015548cb50SEnrico Granata 
102b9c1b51eSKate Stone   if (partial_match_ok) {
103b9c1b51eSKate Stone     for (i = 0; i < g_num_format_infos; ++i) {
104*13a3b0bbSPavel Labath       if (llvm::StringRef(g_format_infos[i].format_name)
105*13a3b0bbSPavel Labath               .startswith_insensitive(format_name)) {
1065548cb50SEnrico Granata         format = g_format_infos[i].format;
1075548cb50SEnrico Granata         return true;
1085548cb50SEnrico Granata       }
1095548cb50SEnrico Granata     }
1105548cb50SEnrico Granata   }
1115548cb50SEnrico Granata   format = eFormatInvalid;
1125548cb50SEnrico Granata   return false;
1135548cb50SEnrico Granata }
1145548cb50SEnrico Granata 
Changed()115b9c1b51eSKate Stone void FormatManager::Changed() {
116bd5eab82SEnrico Granata   ++m_last_revision;
117bd5eab82SEnrico Granata   m_format_cache.Clear();
11816ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
119b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
120bd5eab82SEnrico Granata     if (iter.second)
121bd5eab82SEnrico Granata       iter.second->GetFormatCache().Clear();
122bd5eab82SEnrico Granata   }
123bd5eab82SEnrico Granata }
124bd5eab82SEnrico Granata 
GetFormatFromCString(const char * format_cstr,bool partial_match_ok,lldb::Format & format)125b9c1b51eSKate Stone bool FormatManager::GetFormatFromCString(const char *format_cstr,
1265548cb50SEnrico Granata                                          bool partial_match_ok,
127b9c1b51eSKate Stone                                          lldb::Format &format) {
1285548cb50SEnrico Granata   bool success = false;
129b9c1b51eSKate Stone   if (format_cstr && format_cstr[0]) {
130b9c1b51eSKate Stone     if (format_cstr[1] == '\0') {
1315548cb50SEnrico Granata       success = GetFormatFromFormatChar(format_cstr[0], format);
1325548cb50SEnrico Granata       if (success)
1335548cb50SEnrico Granata         return true;
1345548cb50SEnrico Granata     }
1355548cb50SEnrico Granata 
1365548cb50SEnrico Granata     success = GetFormatFromFormatName(format_cstr, partial_match_ok, format);
1375548cb50SEnrico Granata   }
1385548cb50SEnrico Granata   if (!success)
1395548cb50SEnrico Granata     format = eFormatInvalid;
1405548cb50SEnrico Granata   return success;
1415548cb50SEnrico Granata }
1425548cb50SEnrico Granata 
GetFormatAsFormatChar(lldb::Format format)143b9c1b51eSKate Stone char FormatManager::GetFormatAsFormatChar(lldb::Format format) {
144b9c1b51eSKate Stone   for (uint32_t i = 0; i < g_num_format_infos; ++i) {
1455548cb50SEnrico Granata     if (g_format_infos[i].format == format)
1465548cb50SEnrico Granata       return g_format_infos[i].format_char;
1475548cb50SEnrico Granata   }
1485548cb50SEnrico Granata   return '\0';
1495548cb50SEnrico Granata }
1505548cb50SEnrico Granata 
GetFormatAsCString(Format format)151b9c1b51eSKate Stone const char *FormatManager::GetFormatAsCString(Format format) {
1525548cb50SEnrico Granata   if (format >= eFormatDefault && format < kNumFormats)
1535548cb50SEnrico Granata     return g_format_infos[format].format_name;
154248a1305SKonrad Kleine   return nullptr;
1555548cb50SEnrico Granata }
1565548cb50SEnrico Granata 
EnableAllCategories()157b9c1b51eSKate Stone void FormatManager::EnableAllCategories() {
15833e97e63SEnrico Granata   m_categories_map.EnableAllCategories();
15916ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
160b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
16133e97e63SEnrico Granata     if (iter.second)
16233e97e63SEnrico Granata       iter.second->Enable();
16333e97e63SEnrico Granata   }
16433e97e63SEnrico Granata }
16533e97e63SEnrico Granata 
DisableAllCategories()166b9c1b51eSKate Stone void FormatManager::DisableAllCategories() {
16733e97e63SEnrico Granata   m_categories_map.DisableAllCategories();
16816ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
169b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
17033e97e63SEnrico Granata     if (iter.second)
17133e97e63SEnrico Granata       iter.second->Disable();
17233e97e63SEnrico Granata   }
17333e97e63SEnrico Granata }
17433e97e63SEnrico Granata 
GetPossibleMatches(ValueObject & valobj,CompilerType compiler_type,lldb::DynamicValueType use_dynamic,FormattersMatchVector & entries,bool did_strip_ptr,bool did_strip_ref,bool did_strip_typedef,bool root_level)175b9c1b51eSKate Stone void FormatManager::GetPossibleMatches(
1763e3701f8SRaphael Isemann     ValueObject &valobj, CompilerType compiler_type,
177b9c1b51eSKate Stone     lldb::DynamicValueType use_dynamic, FormattersMatchVector &entries,
178b9c1b51eSKate Stone     bool did_strip_ptr, bool did_strip_ref, bool did_strip_typedef,
179b9c1b51eSKate Stone     bool root_level) {
180c6bf2e2dSEnrico Granata   compiler_type = compiler_type.GetTypeForFormatters();
18130ce956aSRaphael Isemann   ConstString type_name(compiler_type.GetTypeName());
182b9c1b51eSKate Stone   if (valobj.GetBitfieldBitSize() > 0) {
183de61cecdSEnrico Granata     StreamString sstring;
184de61cecdSEnrico Granata     sstring.Printf("%s:%d", type_name.AsCString(), valobj.GetBitfieldBitSize());
185c156427dSZachary Turner     ConstString bitfieldname(sstring.GetString());
186b9c1b51eSKate Stone     entries.push_back(
1873e3701f8SRaphael Isemann         {bitfieldname, did_strip_ptr, did_strip_ref, did_strip_typedef});
188de61cecdSEnrico Granata   }
189b3f0c340SEnrico Granata 
19078f05d35SDavide Italiano   if (!compiler_type.IsMeaninglessWithoutDynamicResolution()) {
191b9c1b51eSKate Stone     entries.push_back(
1923e3701f8SRaphael Isemann         {type_name, did_strip_ptr, did_strip_ref, did_strip_typedef});
193de61cecdSEnrico Granata 
194785df616SRaphael Isemann     ConstString display_type_name(compiler_type.GetTypeName());
195e8daa2f8SEnrico Granata     if (display_type_name != type_name)
1963e3701f8SRaphael Isemann       entries.push_back({display_type_name, did_strip_ptr,
19778f05d35SDavide Italiano                          did_strip_ref, did_strip_typedef});
19878f05d35SDavide Italiano   }
199e8daa2f8SEnrico Granata 
200b9c1b51eSKate Stone   for (bool is_rvalue_ref = true, j = true;
201b9c1b51eSKate Stone        j && compiler_type.IsReferenceType(nullptr, &is_rvalue_ref); j = false) {
20259b5a37dSBruce Mitchener     CompilerType non_ref_type = compiler_type.GetNonReferenceType();
203b9c1b51eSKate Stone     GetPossibleMatches(
204b9c1b51eSKate Stone         valobj, non_ref_type,
205b9c1b51eSKate Stone         use_dynamic, entries, did_strip_ptr, true, did_strip_typedef);
206b9c1b51eSKate Stone     if (non_ref_type.IsTypedefType()) {
207a1e5dc86SGreg Clayton       CompilerType deffed_referenced_type = non_ref_type.GetTypedefedType();
208b9c1b51eSKate Stone       deffed_referenced_type =
209b9c1b51eSKate Stone           is_rvalue_ref ? deffed_referenced_type.GetRValueReferenceType()
210b9c1b51eSKate Stone                         : deffed_referenced_type.GetLValueReferenceType();
211b9c1b51eSKate Stone       GetPossibleMatches(
212b9c1b51eSKate Stone           valobj, deffed_referenced_type,
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         use_dynamic, entries, true, did_strip_ref, did_strip_typedef);
223b9c1b51eSKate Stone     if (non_ptr_type.IsTypedefType()) {
224b9c1b51eSKate Stone       CompilerType deffed_pointed_type =
225b9c1b51eSKate Stone           non_ptr_type.GetTypedefedType().GetPointerType();
22690297427SJaroslav Sevcik       const bool stripped_typedef = true;
227b9c1b51eSKate Stone       GetPossibleMatches(
228b9c1b51eSKate Stone           valobj, deffed_pointed_type,
229b9c1b51eSKate Stone           use_dynamic, entries, did_strip_ptr, did_strip_ref,
23090297427SJaroslav Sevcik           stripped_typedef); // this is not exactly the usual meaning of
23190297427SJaroslav Sevcik                              // stripping typedefs
23290297427SJaroslav Sevcik     }
23390297427SJaroslav Sevcik   }
23490297427SJaroslav Sevcik 
23590297427SJaroslav Sevcik   // For arrays with typedef-ed elements, we add a candidate with the typedef
23690297427SJaroslav Sevcik   // stripped.
23790297427SJaroslav Sevcik   uint64_t array_size;
23890297427SJaroslav Sevcik   if (compiler_type.IsArrayType(nullptr, &array_size, nullptr)) {
23902f58373SAdrian Prantl     ExecutionContext exe_ctx(valobj.GetExecutionContextRef());
24002f58373SAdrian Prantl     CompilerType element_type = compiler_type.GetArrayElementType(
24102f58373SAdrian Prantl         exe_ctx.GetBestExecutionContextScope());
24290297427SJaroslav Sevcik     if (element_type.IsTypedefType()) {
24390297427SJaroslav Sevcik       // Get the stripped element type and compute the stripped array type
24490297427SJaroslav Sevcik       // from it.
24590297427SJaroslav Sevcik       CompilerType deffed_array_type =
24690297427SJaroslav Sevcik           element_type.GetTypedefedType().GetArrayType(array_size);
24790297427SJaroslav Sevcik       const bool stripped_typedef = true;
24890297427SJaroslav Sevcik       GetPossibleMatches(
24990297427SJaroslav Sevcik           valobj, deffed_array_type,
25090297427SJaroslav Sevcik           use_dynamic, entries, did_strip_ptr, did_strip_ref,
25190297427SJaroslav Sevcik           stripped_typedef); // this is not exactly the usual meaning of
25290297427SJaroslav Sevcik                              // stripping typedefs
2531ac62963SEnrico Granata     }
254de61cecdSEnrico Granata   }
255de61cecdSEnrico Granata 
256bc69dd2cSDavide Italiano   for (lldb::LanguageType language_type :
257bc69dd2cSDavide Italiano        GetCandidateLanguages(valobj.GetObjectRuntimeLanguage())) {
258b9c1b51eSKate Stone     if (Language *language = Language::FindPlugin(language_type)) {
259b9c1b51eSKate Stone       for (ConstString candidate :
260b9c1b51eSKate Stone            language->GetPossibleFormattersMatches(valobj, use_dynamic)) {
261b9c1b51eSKate Stone         entries.push_back(
262b9c1b51eSKate Stone             {candidate,
263b9c1b51eSKate Stone              did_strip_ptr, did_strip_ref, did_strip_typedef});
264d3233c1eSEnrico Granata       }
265d3233c1eSEnrico Granata     }
266de61cecdSEnrico Granata   }
267de61cecdSEnrico Granata 
268de61cecdSEnrico Granata   // try to strip typedef chains
269b9c1b51eSKate Stone   if (compiler_type.IsTypedefType()) {
27059b5a37dSBruce Mitchener     CompilerType deffed_type = compiler_type.GetTypedefedType();
271b9c1b51eSKate Stone     GetPossibleMatches(
272b9c1b51eSKate Stone         valobj, deffed_type,
273b9c1b51eSKate Stone         use_dynamic, entries, did_strip_ptr, did_strip_ref, true);
274de61cecdSEnrico Granata   }
275de61cecdSEnrico Granata 
276b9c1b51eSKate Stone   if (root_level) {
277de61cecdSEnrico Granata     do {
27859b5a37dSBruce Mitchener       if (!compiler_type.IsValid())
279de61cecdSEnrico Granata         break;
280de61cecdSEnrico Granata 
281b9c1b51eSKate Stone       CompilerType unqual_compiler_ast_type =
282b9c1b51eSKate Stone           compiler_type.GetFullyUnqualifiedType();
28359b5a37dSBruce Mitchener       if (!unqual_compiler_ast_type.IsValid())
284de61cecdSEnrico Granata         break;
285b9c1b51eSKate Stone       if (unqual_compiler_ast_type.GetOpaqueQualType() !=
286b9c1b51eSKate Stone           compiler_type.GetOpaqueQualType())
2873e3701f8SRaphael Isemann         GetPossibleMatches(valobj, unqual_compiler_ast_type,
288b9c1b51eSKate Stone                            use_dynamic, entries, did_strip_ptr, did_strip_ref,
289de61cecdSEnrico Granata                            did_strip_typedef);
290de61cecdSEnrico Granata     } while (false);
291de61cecdSEnrico Granata 
292de61cecdSEnrico Granata     // if all else fails, go to static type
293b9c1b51eSKate Stone     if (valobj.IsDynamic()) {
294de61cecdSEnrico Granata       lldb::ValueObjectSP static_value_sp(valobj.GetStaticValue());
295de61cecdSEnrico Granata       if (static_value_sp)
296b9c1b51eSKate Stone         GetPossibleMatches(
297b9c1b51eSKate Stone             *static_value_sp.get(), static_value_sp->GetCompilerType(),
298b9c1b51eSKate Stone             use_dynamic, entries, did_strip_ptr, did_strip_ref,
299b9c1b51eSKate Stone             did_strip_typedef, true);
300de61cecdSEnrico Granata     }
301de61cecdSEnrico Granata   }
302de61cecdSEnrico Granata }
303de61cecdSEnrico Granata 
304852cc954SEnrico Granata lldb::TypeFormatImplSP
GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp)305b9c1b51eSKate Stone FormatManager::GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp) {
306852cc954SEnrico Granata   if (!type_sp)
307852cc954SEnrico Granata     return lldb::TypeFormatImplSP();
308852cc954SEnrico Granata   lldb::TypeFormatImplSP format_chosen_sp;
309852cc954SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
310852cc954SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
311852cc954SEnrico Granata   uint32_t prio_category = UINT32_MAX;
312b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
313852cc954SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
314a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
315852cc954SEnrico Granata       continue;
316b9c1b51eSKate Stone     lldb::TypeFormatImplSP format_current_sp =
317b9c1b51eSKate Stone         category_sp->GetFormatForType(type_sp);
318b9c1b51eSKate Stone     if (format_current_sp &&
319248a1305SKonrad Kleine         (format_chosen_sp.get() == nullptr ||
320b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
321852cc954SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
322852cc954SEnrico Granata       format_chosen_sp = format_current_sp;
323852cc954SEnrico Granata     }
324852cc954SEnrico Granata   }
325852cc954SEnrico Granata   return format_chosen_sp;
326852cc954SEnrico Granata }
327852cc954SEnrico Granata 
3285548cb50SEnrico Granata lldb::TypeSummaryImplSP
GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp)329b9c1b51eSKate Stone FormatManager::GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp) {
3305548cb50SEnrico Granata   if (!type_sp)
3315548cb50SEnrico Granata     return lldb::TypeSummaryImplSP();
3325548cb50SEnrico Granata   lldb::TypeSummaryImplSP summary_chosen_sp;
3335548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3345548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3355548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
336b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3375548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
338a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3395548cb50SEnrico Granata       continue;
340b9c1b51eSKate Stone     lldb::TypeSummaryImplSP summary_current_sp =
341b9c1b51eSKate Stone         category_sp->GetSummaryForType(type_sp);
342b9c1b51eSKate Stone     if (summary_current_sp &&
343248a1305SKonrad Kleine         (summary_chosen_sp.get() == nullptr ||
344b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3455548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3465548cb50SEnrico Granata       summary_chosen_sp = summary_current_sp;
3475548cb50SEnrico Granata     }
3485548cb50SEnrico Granata   }
3495548cb50SEnrico Granata   return summary_chosen_sp;
3505548cb50SEnrico Granata }
3515548cb50SEnrico Granata 
3525548cb50SEnrico Granata lldb::TypeFilterImplSP
GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp)353b9c1b51eSKate Stone FormatManager::GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp) {
3545548cb50SEnrico Granata   if (!type_sp)
3555548cb50SEnrico Granata     return lldb::TypeFilterImplSP();
3565548cb50SEnrico Granata   lldb::TypeFilterImplSP filter_chosen_sp;
3575548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3585548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3595548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
360b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3615548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
362a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3635548cb50SEnrico Granata       continue;
364b9c1b51eSKate Stone     lldb::TypeFilterImplSP filter_current_sp(
365b9c1b51eSKate Stone         (TypeFilterImpl *)category_sp->GetFilterForType(type_sp).get());
366b9c1b51eSKate Stone     if (filter_current_sp &&
367248a1305SKonrad Kleine         (filter_chosen_sp.get() == nullptr ||
368b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3695548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3705548cb50SEnrico Granata       filter_chosen_sp = filter_current_sp;
3715548cb50SEnrico Granata     }
3725548cb50SEnrico Granata   }
3735548cb50SEnrico Granata   return filter_chosen_sp;
3745548cb50SEnrico Granata }
3755548cb50SEnrico Granata 
3765548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP
GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp)377b9c1b51eSKate Stone FormatManager::GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp) {
3785548cb50SEnrico Granata   if (!type_sp)
3795548cb50SEnrico Granata     return lldb::ScriptedSyntheticChildrenSP();
3805548cb50SEnrico Granata   lldb::ScriptedSyntheticChildrenSP synth_chosen_sp;
3815548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3825548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3835548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
384b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3855548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
386a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3875548cb50SEnrico Granata       continue;
388b9c1b51eSKate Stone     lldb::ScriptedSyntheticChildrenSP synth_current_sp(
389b9c1b51eSKate Stone         (ScriptedSyntheticChildren *)category_sp->GetSyntheticForType(type_sp)
390b9c1b51eSKate Stone             .get());
391b9c1b51eSKate Stone     if (synth_current_sp &&
392248a1305SKonrad Kleine         (synth_chosen_sp.get() == nullptr ||
393b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3945548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3955548cb50SEnrico Granata       synth_chosen_sp = synth_current_sp;
3965548cb50SEnrico Granata     }
3975548cb50SEnrico Granata   }
3985548cb50SEnrico Granata   return synth_chosen_sp;
3995548cb50SEnrico Granata }
4005548cb50SEnrico Granata 
ForEachCategory(TypeCategoryMap::ForEachCallback callback)401b9c1b51eSKate Stone void FormatManager::ForEachCategory(TypeCategoryMap::ForEachCallback callback) {
402b56d0103SEnrico Granata   m_categories_map.ForEach(callback);
40316ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
404b9c1b51eSKate Stone   for (const auto &entry : m_language_categories_map) {
405b9c1b51eSKate Stone     if (auto category_sp = entry.second->GetCategory()) {
406b56d0103SEnrico Granata       if (!callback(category_sp))
407b56d0103SEnrico Granata         break;
408b56d0103SEnrico Granata     }
409b56d0103SEnrico Granata   }
410b56d0103SEnrico Granata }
411b56d0103SEnrico Granata 
4125548cb50SEnrico Granata lldb::TypeCategoryImplSP
GetCategory(ConstString category_name,bool can_create)4130e4c4821SAdrian Prantl FormatManager::GetCategory(ConstString category_name, bool can_create) {
4145548cb50SEnrico Granata   if (!category_name)
4155548cb50SEnrico Granata     return GetCategory(m_default_category_name);
4165548cb50SEnrico Granata   lldb::TypeCategoryImplSP category;
4175548cb50SEnrico Granata   if (m_categories_map.Get(category_name, category))
4185548cb50SEnrico Granata     return category;
4195548cb50SEnrico Granata 
4205548cb50SEnrico Granata   if (!can_create)
4215548cb50SEnrico Granata     return lldb::TypeCategoryImplSP();
4225548cb50SEnrico Granata 
423b9c1b51eSKate Stone   m_categories_map.Add(
424b9c1b51eSKate Stone       category_name,
425b9c1b51eSKate Stone       lldb::TypeCategoryImplSP(new TypeCategoryImpl(this, category_name)));
4265548cb50SEnrico Granata   return GetCategory(category_name);
4275548cb50SEnrico Granata }
4285548cb50SEnrico Granata 
GetSingleItemFormat(lldb::Format vector_format)429b9c1b51eSKate Stone lldb::Format FormatManager::GetSingleItemFormat(lldb::Format vector_format) {
430b9c1b51eSKate Stone   switch (vector_format) {
4315548cb50SEnrico Granata   case eFormatVectorOfChar:
4325548cb50SEnrico Granata     return eFormatCharArray;
4335548cb50SEnrico Granata 
4345548cb50SEnrico Granata   case eFormatVectorOfSInt8:
4355548cb50SEnrico Granata   case eFormatVectorOfSInt16:
4365548cb50SEnrico Granata   case eFormatVectorOfSInt32:
4375548cb50SEnrico Granata   case eFormatVectorOfSInt64:
4385548cb50SEnrico Granata     return eFormatDecimal;
4395548cb50SEnrico Granata 
4405548cb50SEnrico Granata   case eFormatVectorOfUInt8:
4415548cb50SEnrico Granata   case eFormatVectorOfUInt16:
4425548cb50SEnrico Granata   case eFormatVectorOfUInt32:
4435548cb50SEnrico Granata   case eFormatVectorOfUInt64:
4445548cb50SEnrico Granata   case eFormatVectorOfUInt128:
4455548cb50SEnrico Granata     return eFormatHex;
4465548cb50SEnrico Granata 
447a0f08674SEwan Crawford   case eFormatVectorOfFloat16:
4485548cb50SEnrico Granata   case eFormatVectorOfFloat32:
4495548cb50SEnrico Granata   case eFormatVectorOfFloat64:
4505548cb50SEnrico Granata     return eFormatFloat;
4515548cb50SEnrico Granata 
4525548cb50SEnrico Granata   default:
4535548cb50SEnrico Granata     return lldb::eFormatInvalid;
4545548cb50SEnrico Granata   }
4555548cb50SEnrico Granata }
4565548cb50SEnrico Granata 
ShouldPrintAsOneLiner(ValueObject & valobj)457b9c1b51eSKate Stone bool FormatManager::ShouldPrintAsOneLiner(ValueObject &valobj) {
458553fad5cSEnrico Granata   // if settings say no oneline whatsoever
459b9c1b51eSKate Stone   if (valobj.GetTargetSP().get() &&
460a6682a41SJonas Devlieghere       !valobj.GetTargetSP()->GetDebugger().GetAutoOneLineSummaries())
461553fad5cSEnrico Granata     return false; // then don't oneline
462553fad5cSEnrico Granata 
46342fa4af8SEnrico Granata   // if this object has a summary, then ask the summary
464a29cb0baSEnrico Granata   if (valobj.GetSummaryFormat().get() != nullptr)
46542fa4af8SEnrico Granata     return valobj.GetSummaryFormat()->IsOneLiner();
466a29cb0baSEnrico Granata 
467a29cb0baSEnrico Granata   // no children, no party
468a29cb0baSEnrico Granata   if (valobj.GetNumChildren() == 0)
469a29cb0baSEnrico Granata     return false;
470a29cb0baSEnrico Granata 
47105097246SAdrian Prantl   // ask the type if it has any opinion about this eLazyBoolCalculate == no
47205097246SAdrian Prantl   // opinion; other values should be self explanatory
4739c63f99aSEnrico Granata   CompilerType compiler_type(valobj.GetCompilerType());
474b9c1b51eSKate Stone   if (compiler_type.IsValid()) {
475b9c1b51eSKate Stone     switch (compiler_type.ShouldPrintAsOneLiner(&valobj)) {
4769c63f99aSEnrico Granata     case eLazyBoolNo:
4779c63f99aSEnrico Granata       return false;
4789c63f99aSEnrico Granata     case eLazyBoolYes:
4799c63f99aSEnrico Granata       return true;
4809c63f99aSEnrico Granata     case eLazyBoolCalculate:
4819c63f99aSEnrico Granata       break;
4829c63f99aSEnrico Granata     }
4839c63f99aSEnrico Granata   }
4849c63f99aSEnrico Granata 
485a29cb0baSEnrico Granata   size_t total_children_name_len = 0;
486a29cb0baSEnrico Granata 
487b9c1b51eSKate Stone   for (size_t idx = 0; idx < valobj.GetNumChildren(); idx++) {
488ddac7611SEnrico Granata     bool is_synth_val = false;
489a29cb0baSEnrico Granata     ValueObjectSP child_sp(valobj.GetChildAtIndex(idx, true));
490a29cb0baSEnrico Granata     // something is wrong here - bail out
491a29cb0baSEnrico Granata     if (!child_sp)
492a29cb0baSEnrico Granata       return false;
4936500061eSEnrico Granata 
4946500061eSEnrico Granata     // also ask the child's type if it has any opinion
4956500061eSEnrico Granata     CompilerType child_compiler_type(child_sp->GetCompilerType());
496b9c1b51eSKate Stone     if (child_compiler_type.IsValid()) {
497b9c1b51eSKate Stone       switch (child_compiler_type.ShouldPrintAsOneLiner(child_sp.get())) {
4986500061eSEnrico Granata       case eLazyBoolYes:
4996500061eSEnrico Granata       // an opinion of yes is only binding for the child, so keep going
5006500061eSEnrico Granata       case eLazyBoolCalculate:
5016500061eSEnrico Granata         break;
5026500061eSEnrico Granata       case eLazyBoolNo:
5036500061eSEnrico Granata         // but if the child says no, then it's a veto on the whole thing
5046500061eSEnrico Granata         return false;
5056500061eSEnrico Granata       }
5066500061eSEnrico Granata     }
5076500061eSEnrico Granata 
508b9c1b51eSKate Stone     // if we decided to define synthetic children for a type, we probably care
50905097246SAdrian Prantl     // enough to show them, but avoid nesting children in children
510b9c1b51eSKate Stone     if (child_sp->GetSyntheticChildren().get() != nullptr) {
511ddac7611SEnrico Granata       ValueObjectSP synth_sp(child_sp->GetSyntheticValue());
512ddac7611SEnrico Granata       // wait.. wat? just get out of here..
513ddac7611SEnrico Granata       if (!synth_sp)
514a29cb0baSEnrico Granata         return false;
515ddac7611SEnrico Granata       // but if we only have them to provide a value, keep going
516a6682a41SJonas Devlieghere       if (!synth_sp->MightHaveChildren() &&
517b9c1b51eSKate Stone           synth_sp->DoesProvideSyntheticValue())
518ddac7611SEnrico Granata         is_synth_val = true;
519ddac7611SEnrico Granata       else
520ddac7611SEnrico Granata         return false;
521ddac7611SEnrico Granata     }
522a29cb0baSEnrico Granata 
523a29cb0baSEnrico Granata     total_children_name_len += child_sp->GetName().GetLength();
524a29cb0baSEnrico Granata 
525a29cb0baSEnrico Granata     // 50 itself is a "randomly" chosen number - the idea is that
526a29cb0baSEnrico Granata     // overly long structs should not get this treatment
527a29cb0baSEnrico Granata     // FIXME: maybe make this a user-tweakable setting?
528a29cb0baSEnrico Granata     if (total_children_name_len > 50)
529a29cb0baSEnrico Granata       return false;
530a29cb0baSEnrico Granata 
531a29cb0baSEnrico Granata     // if a summary is there..
532b9c1b51eSKate Stone     if (child_sp->GetSummaryFormat()) {
533a29cb0baSEnrico Granata       // and it wants children, then bail out
5348a068e6cSEnrico Granata       if (child_sp->GetSummaryFormat()->DoesPrintChildren(child_sp.get()))
535a29cb0baSEnrico Granata         return false;
536a29cb0baSEnrico Granata     }
537a29cb0baSEnrico Granata 
538c89e4ca3SEnrico Granata     // if this child has children..
539b9c1b51eSKate Stone     if (child_sp->GetNumChildren()) {
540a29cb0baSEnrico Granata       // ...and no summary...
541b9c1b51eSKate Stone       // (if it had a summary and the summary wanted children, we would have
542b9c1b51eSKate Stone       // bailed out anyway
543b9c1b51eSKate Stone       //  so this only makes us bail out if this has no summary and we would
544b9c1b51eSKate Stone       //  then print children)
545b9c1b51eSKate Stone       if (!child_sp->GetSummaryFormat() && !is_synth_val) // but again only do
546b9c1b51eSKate Stone                                                           // that if not a
547b9c1b51eSKate Stone                                                           // synthetic valued
548b9c1b51eSKate Stone                                                           // child
549a29cb0baSEnrico Granata         return false;                                     // then bail out
550a29cb0baSEnrico Granata     }
551a29cb0baSEnrico Granata   }
552a29cb0baSEnrico Granata   return true;
553a29cb0baSEnrico Granata }
554a29cb0baSEnrico Granata 
GetTypeForCache(ValueObject & valobj,lldb::DynamicValueType use_dynamic)555b9c1b51eSKate Stone ConstString FormatManager::GetTypeForCache(ValueObject &valobj,
556b9c1b51eSKate Stone                                            lldb::DynamicValueType use_dynamic) {
557b9c1b51eSKate Stone   ValueObjectSP valobj_sp = valobj.GetQualifiedRepresentationIfAvailable(
558b9c1b51eSKate Stone       use_dynamic, valobj.IsSynthetic());
55978f05d35SDavide Italiano   if (valobj_sp && valobj_sp->GetCompilerType().IsValid()) {
56078f05d35SDavide Italiano     if (!valobj_sp->GetCompilerType().IsMeaninglessWithoutDynamicResolution())
561b3f0c340SEnrico Granata       return valobj_sp->GetQualifiedTypeName();
56278f05d35SDavide Italiano   }
5635548cb50SEnrico Granata   return ConstString();
5645548cb50SEnrico Granata }
5655548cb50SEnrico Granata 
566d3233c1eSEnrico Granata std::vector<lldb::LanguageType>
GetCandidateLanguages(lldb::LanguageType lang_type)567bc69dd2cSDavide Italiano FormatManager::GetCandidateLanguages(lldb::LanguageType lang_type) {
568b9c1b51eSKate Stone   switch (lang_type) {
56933e97e63SEnrico Granata   case lldb::eLanguageTypeC:
57033e97e63SEnrico Granata   case lldb::eLanguageTypeC89:
57133e97e63SEnrico Granata   case lldb::eLanguageTypeC99:
57233e97e63SEnrico Granata   case lldb::eLanguageTypeC11:
57333e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus:
57433e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_03:
57533e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_11:
57633e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_14:
577170c395eSEnrico Granata     return {lldb::eLanguageTypeC_plus_plus, lldb::eLanguageTypeObjC};
578980c0484SEnrico Granata   default:
579980c0484SEnrico Granata     return {lang_type};
580980c0484SEnrico Granata   }
581295db41cSDavide Italiano   llvm_unreachable("Fully covered switch");
582980c0484SEnrico Granata }
583980c0484SEnrico Granata 
584980c0484SEnrico Granata LanguageCategory *
GetCategoryForLanguage(lldb::LanguageType lang_type)585b9c1b51eSKate Stone FormatManager::GetCategoryForLanguage(lldb::LanguageType lang_type) {
58616ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
587b9c1b51eSKate Stone   auto iter = m_language_categories_map.find(lang_type),
588b9c1b51eSKate Stone        end = m_language_categories_map.end();
589980c0484SEnrico Granata   if (iter != end)
590980c0484SEnrico Granata     return iter->second.get();
591980c0484SEnrico Granata   LanguageCategory *lang_category = new LanguageCategory(lang_type);
592b9c1b51eSKate Stone   m_language_categories_map[lang_type] =
593b9c1b51eSKate Stone       LanguageCategory::UniquePointer(lang_category);
594980c0484SEnrico Granata   return lang_category;
595980c0484SEnrico Granata }
596980c0484SEnrico Granata 
5977034794bSAdrian Prantl template <typename ImplSP>
GetHardcoded(FormattersMatchData & match_data)5987034794bSAdrian Prantl ImplSP FormatManager::GetHardcoded(FormattersMatchData &match_data) {
5997034794bSAdrian Prantl   ImplSP retval_sp;
600b9c1b51eSKate Stone   for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) {
601b9c1b51eSKate Stone     if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) {
6028a9a8f39SEnrico Granata       if (lang_category->GetHardcoded(*this, match_data, retval_sp))
6037034794bSAdrian Prantl         return retval_sp;
6047034794bSAdrian Prantl     }
6057034794bSAdrian Prantl   }
6067034794bSAdrian Prantl   return retval_sp;
6077034794bSAdrian Prantl }
6087034794bSAdrian Prantl 
60962a6d977SAdrian Prantl template <typename ImplSP>
Get(ValueObject & valobj,lldb::DynamicValueType use_dynamic)61062a6d977SAdrian Prantl ImplSP FormatManager::Get(ValueObject &valobj,
6117034794bSAdrian Prantl                           lldb::DynamicValueType use_dynamic) {
6127034794bSAdrian Prantl   FormattersMatchData match_data(valobj, use_dynamic);
61362a6d977SAdrian Prantl   if (ImplSP retval_sp = GetCached<ImplSP>(match_data))
61462a6d977SAdrian Prantl     return retval_sp;
61562a6d977SAdrian Prantl 
616a007a6d8SPavel Labath   Log *log = GetLog(LLDBLog::DataFormatters);
61770e3d0eaSAdrian Prantl 
61870e3d0eaSAdrian Prantl   LLDB_LOGF(log, "[%s] Search failed. Giving language a chance.", __FUNCTION__);
61970e3d0eaSAdrian Prantl   for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) {
62070e3d0eaSAdrian Prantl     if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) {
62170e3d0eaSAdrian Prantl       ImplSP retval_sp;
62270e3d0eaSAdrian Prantl       if (lang_category->Get(match_data, retval_sp))
62370e3d0eaSAdrian Prantl         if (retval_sp) {
62470e3d0eaSAdrian Prantl           LLDB_LOGF(log, "[%s] Language search success. Returning.",
62570e3d0eaSAdrian Prantl                     __FUNCTION__);
62670e3d0eaSAdrian Prantl           return retval_sp;
62770e3d0eaSAdrian Prantl         }
62870e3d0eaSAdrian Prantl     }
62970e3d0eaSAdrian Prantl   }
63070e3d0eaSAdrian Prantl 
63170e3d0eaSAdrian Prantl   LLDB_LOGF(log, "[%s] Search failed. Giving hardcoded a chance.",
63270e3d0eaSAdrian Prantl             __FUNCTION__);
63362a6d977SAdrian Prantl   return GetHardcoded<ImplSP>(match_data);
63462a6d977SAdrian Prantl }
63562a6d977SAdrian Prantl 
63662a6d977SAdrian Prantl template <typename ImplSP>
GetCached(FormattersMatchData & match_data)63762a6d977SAdrian Prantl ImplSP FormatManager::GetCached(FormattersMatchData &match_data) {
63862a6d977SAdrian Prantl   ImplSP retval_sp;
639a007a6d8SPavel Labath   Log *log = GetLog(LLDBLog::DataFormatters);
6407034794bSAdrian Prantl   if (match_data.GetTypeForCache()) {
6417034794bSAdrian Prantl     LLDB_LOGF(log, "\n\n[%s] Looking into cache for type %s", __FUNCTION__,
6427034794bSAdrian Prantl               match_data.GetTypeForCache().AsCString("<invalid>"));
6437034794bSAdrian Prantl     if (m_format_cache.Get(match_data.GetTypeForCache(), retval_sp)) {
6447034794bSAdrian Prantl       if (log) {
6457034794bSAdrian Prantl         LLDB_LOGF(log, "[%s] Cache search success. Returning.", __FUNCTION__);
6467034794bSAdrian Prantl         LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}",
6477034794bSAdrian Prantl                   m_format_cache.GetCacheHits(),
6487034794bSAdrian Prantl                   m_format_cache.GetCacheMisses());
6497034794bSAdrian Prantl       }
6507034794bSAdrian Prantl       return retval_sp;
6517034794bSAdrian Prantl     }
6527034794bSAdrian Prantl     LLDB_LOGF(log, "[%s] Cache search failed. Going normal route",
6537034794bSAdrian Prantl               __FUNCTION__);
6547034794bSAdrian Prantl   }
6557034794bSAdrian Prantl 
6567034794bSAdrian Prantl   m_categories_map.Get(match_data, retval_sp);
6577034794bSAdrian Prantl   if (match_data.GetTypeForCache() && (!retval_sp || !retval_sp->NonCacheable())) {
6587034794bSAdrian Prantl     LLDB_LOGF(log, "[%s] Caching %p for type %s", __FUNCTION__,
6597034794bSAdrian Prantl               static_cast<void *>(retval_sp.get()),
6607034794bSAdrian Prantl               match_data.GetTypeForCache().AsCString("<invalid>"));
6617034794bSAdrian Prantl     m_format_cache.Set(match_data.GetTypeForCache(), retval_sp);
6627034794bSAdrian Prantl   }
6637034794bSAdrian Prantl   LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}",
6647034794bSAdrian Prantl             m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses());
6657cb59e1aSEnrico Granata   return retval_sp;
666686f3debSEnrico Granata }
667686f3debSEnrico Granata 
668852cc954SEnrico Granata lldb::TypeFormatImplSP
GetFormat(ValueObject & valobj,lldb::DynamicValueType use_dynamic)669852cc954SEnrico Granata FormatManager::GetFormat(ValueObject &valobj,
670b9c1b51eSKate Stone                          lldb::DynamicValueType use_dynamic) {
67162a6d977SAdrian Prantl   return Get<lldb::TypeFormatImplSP>(valobj, use_dynamic);
672686f3debSEnrico Granata }
673686f3debSEnrico Granata 
6745548cb50SEnrico Granata lldb::TypeSummaryImplSP
GetSummaryFormat(ValueObject & valobj,lldb::DynamicValueType use_dynamic)6755548cb50SEnrico Granata FormatManager::GetSummaryFormat(ValueObject &valobj,
676b9c1b51eSKate Stone                                 lldb::DynamicValueType use_dynamic) {
67762a6d977SAdrian Prantl   return Get<lldb::TypeSummaryImplSP>(valobj, use_dynamic);
678686f3debSEnrico Granata }
679686f3debSEnrico Granata 
6805548cb50SEnrico Granata lldb::SyntheticChildrenSP
GetSyntheticChildren(ValueObject & valobj,lldb::DynamicValueType use_dynamic)6815548cb50SEnrico Granata FormatManager::GetSyntheticChildren(ValueObject &valobj,
682b9c1b51eSKate Stone                                     lldb::DynamicValueType use_dynamic) {
68362a6d977SAdrian Prantl   return Get<lldb::SyntheticChildrenSP>(valobj, use_dynamic);
6845548cb50SEnrico Granata }
6855548cb50SEnrico Granata 
FormatManager()68616ff8604SSaleem Abdulrasool FormatManager::FormatManager()
687b9c1b51eSKate Stone     : m_last_revision(0), m_format_cache(), m_language_categories_mutex(),
688b9c1b51eSKate Stone       m_language_categories_map(), m_named_summaries_map(this),
689b9c1b51eSKate Stone       m_categories_map(this), m_default_category_name(ConstString("default")),
6905548cb50SEnrico Granata       m_system_category_name(ConstString("system")),
691b9c1b51eSKate Stone       m_vectortypes_category_name(ConstString("VectorTypes")) {
6925548cb50SEnrico Granata   LoadSystemFormatters();
693170c395eSEnrico Granata   LoadVectorFormatters();
6945548cb50SEnrico Granata 
695b9c1b51eSKate Stone   EnableCategory(m_vectortypes_category_name, TypeCategoryMap::Last,
696b9c1b51eSKate Stone                  lldb::eLanguageTypeObjC_plus_plus);
697b9c1b51eSKate Stone   EnableCategory(m_system_category_name, TypeCategoryMap::Last,
698b9c1b51eSKate Stone                  lldb::eLanguageTypeObjC_plus_plus);
6995548cb50SEnrico Granata }
7005548cb50SEnrico Granata 
LoadSystemFormatters()701b9c1b51eSKate Stone void FormatManager::LoadSystemFormatters() {
7025548cb50SEnrico Granata   TypeSummaryImpl::Flags string_flags;
7030337c27fSEnrico Granata   string_flags.SetCascades(true)
7045548cb50SEnrico Granata       .SetSkipPointers(true)
7055548cb50SEnrico Granata       .SetSkipReferences(false)
7065548cb50SEnrico Granata       .SetDontShowChildren(true)
7075548cb50SEnrico Granata       .SetDontShowValue(false)
7085548cb50SEnrico Granata       .SetShowMembersOneLiner(false)
7095548cb50SEnrico Granata       .SetHideItemNames(false);
7105548cb50SEnrico Granata 
711bc2c2b01SEnrico Granata   TypeSummaryImpl::Flags string_array_flags;
712d2911633SEnrico Granata   string_array_flags.SetCascades(true)
7135548cb50SEnrico Granata       .SetSkipPointers(true)
7145548cb50SEnrico Granata       .SetSkipReferences(false)
715320dcf68SEnrico Granata       .SetDontShowChildren(true)
7165548cb50SEnrico Granata       .SetDontShowValue(true)
7175548cb50SEnrico Granata       .SetShowMembersOneLiner(false)
718bc2c2b01SEnrico Granata       .SetHideItemNames(false);
719bc2c2b01SEnrico Granata 
720b9c1b51eSKate Stone   lldb::TypeSummaryImplSP string_format(
721b9c1b51eSKate Stone       new StringSummaryFormat(string_flags, "${var%s}"));
722bc2c2b01SEnrico Granata 
723b9c1b51eSKate Stone   lldb::TypeSummaryImplSP string_array_format(
7248093c2eaSPavel Labath       new StringSummaryFormat(string_array_flags, "${var%char[]}"));
725bc2c2b01SEnrico Granata 
72611dc235cSPavel Labath   RegularExpression any_size_char_arr(R"(^((un)?signed )?char ?\[[0-9]+\]$)");
7275548cb50SEnrico Granata 
728b9c1b51eSKate Stone   TypeCategoryImpl::SharedPointer sys_category_sp =
729b9c1b51eSKate Stone       GetCategory(m_system_category_name);
7305548cb50SEnrico Granata 
73135870c44SPavel Labath   sys_category_sp->GetRegexTypeSummariesContainer()->Add(
73231c7165aSPavel Labath       RegularExpression(R"(^(unsigned )?char ?(\*|\[\])$)"), string_format);
73311dc235cSPavel Labath 
7345aa1d819SJan Kratochvil   sys_category_sp->GetRegexTypeSummariesContainer()->Add(
7355aa1d819SJan Kratochvil       std::move(any_size_char_arr), string_array_format);
7365548cb50SEnrico Granata 
737b9c1b51eSKate Stone   lldb::TypeSummaryImplSP ostype_summary(
738b9c1b51eSKate Stone       new StringSummaryFormat(TypeSummaryImpl::Flags()
739b9c1b51eSKate Stone                                   .SetCascades(false)
7405548cb50SEnrico Granata                                   .SetSkipPointers(true)
7415548cb50SEnrico Granata                                   .SetSkipReferences(true)
7425548cb50SEnrico Granata                                   .SetDontShowChildren(true)
7435548cb50SEnrico Granata                                   .SetDontShowValue(false)
7445548cb50SEnrico Granata                                   .SetShowMembersOneLiner(false)
7455548cb50SEnrico Granata                                   .SetHideItemNames(false),
7465548cb50SEnrico Granata                               "${var%O}"));
7475548cb50SEnrico Granata 
748b9c1b51eSKate Stone   sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("OSType"),
749b9c1b51eSKate Stone                                                     ostype_summary);
7505548cb50SEnrico Granata 
7514cc21772SEnrico Granata   TypeFormatImpl::Flags fourchar_flags;
752b9c1b51eSKate Stone   fourchar_flags.SetCascades(true).SetSkipPointers(true).SetSkipReferences(
753b9c1b51eSKate Stone       true);
7544cc21772SEnrico Granata 
755b9c1b51eSKate Stone   AddFormat(sys_category_sp, lldb::eFormatOSType, ConstString("FourCharCode"),
756b9c1b51eSKate Stone             fourchar_flags);
7575548cb50SEnrico Granata }
7585548cb50SEnrico Granata 
LoadVectorFormatters()759b9c1b51eSKate Stone void FormatManager::LoadVectorFormatters() {
760b9c1b51eSKate Stone   TypeCategoryImpl::SharedPointer vectors_category_sp =
761b9c1b51eSKate Stone       GetCategory(m_vectortypes_category_name);
7625548cb50SEnrico Granata 
7635548cb50SEnrico Granata   TypeSummaryImpl::Flags vector_flags;
7645548cb50SEnrico Granata   vector_flags.SetCascades(true)
7655548cb50SEnrico Granata       .SetSkipPointers(true)
7665548cb50SEnrico Granata       .SetSkipReferences(false)
7675548cb50SEnrico Granata       .SetDontShowChildren(true)
7685548cb50SEnrico Granata       .SetDontShowValue(false)
7695548cb50SEnrico Granata       .SetShowMembersOneLiner(true)
7705548cb50SEnrico Granata       .SetHideItemNames(true);
7715548cb50SEnrico Granata 
772b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "${var.uint128}",
773b9c1b51eSKate Stone                    ConstString("builtin_type_vec128"), vector_flags);
774b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("float[4]"),
7755548cb50SEnrico Granata                    vector_flags);
776b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("int32_t[4]"),
7775548cb50SEnrico Granata                    vector_flags);
778b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("int16_t[8]"),
7795548cb50SEnrico Granata                    vector_flags);
780b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vDouble"),
7815548cb50SEnrico Granata                    vector_flags);
782b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vFloat"),
7835548cb50SEnrico Granata                    vector_flags);
784b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt8"),
7855548cb50SEnrico Granata                    vector_flags);
786b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt16"),
7875548cb50SEnrico Granata                    vector_flags);
788b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt32"),
7895548cb50SEnrico Granata                    vector_flags);
790b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"),
7915548cb50SEnrico Granata                    vector_flags);
792b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt8"),
7935548cb50SEnrico Granata                    vector_flags);
794b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"),
7955548cb50SEnrico Granata                    vector_flags);
796b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt32"),
7975548cb50SEnrico Granata                    vector_flags);
798b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vBool32"),
7995548cb50SEnrico Granata                    vector_flags);
8005548cb50SEnrico Granata }
801