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 
11d717cc9fSEnrico Granata #include "llvm/ADT/STLExtras.h"
12d717cc9fSEnrico Granata 
135548cb50SEnrico Granata 
145548cb50SEnrico Granata #include "lldb/Core/Debugger.h"
15df7e79e6SEnrico Granata #include "lldb/DataFormatters/FormattersHelpers.h"
16980c0484SEnrico Granata #include "lldb/DataFormatters/LanguageCategory.h"
175548cb50SEnrico Granata #include "lldb/Target/ExecutionContext.h"
1833e97e63SEnrico Granata #include "lldb/Target/Language.h"
196f9e6901SZachary Turner #include "lldb/Utility/Log.h"
20980c0484SEnrico Granata 
215548cb50SEnrico Granata using namespace lldb;
225548cb50SEnrico Granata using namespace lldb_private;
23df7e79e6SEnrico Granata using namespace lldb_private::formatters;
245548cb50SEnrico Granata 
25b9c1b51eSKate Stone struct FormatInfo {
265548cb50SEnrico Granata   Format format;
27b9c1b51eSKate Stone   const char format_char;  // One or more format characters that can be used for
28b9c1b51eSKate Stone                            // this format.
29b9c1b51eSKate Stone   const char *format_name; // Long format name that can be used to specify the
30b9c1b51eSKate Stone                            // current format
315548cb50SEnrico Granata };
325548cb50SEnrico Granata 
3312002fbdSJonas Devlieghere static constexpr FormatInfo g_format_infos[] = {
345548cb50SEnrico Granata     {eFormatDefault, '\0', "default"},
355548cb50SEnrico Granata     {eFormatBoolean, 'B', "boolean"},
365548cb50SEnrico Granata     {eFormatBinary, 'b', "binary"},
375548cb50SEnrico Granata     {eFormatBytes, 'y', "bytes"},
385548cb50SEnrico Granata     {eFormatBytesWithASCII, 'Y', "bytes with ASCII"},
395548cb50SEnrico Granata     {eFormatChar, 'c', "character"},
405548cb50SEnrico Granata     {eFormatCharPrintable, 'C', "printable character"},
415548cb50SEnrico Granata     {eFormatComplexFloat, 'F', "complex float"},
425548cb50SEnrico Granata     {eFormatCString, 's', "c-string"},
435548cb50SEnrico Granata     {eFormatDecimal, 'd', "decimal"},
445548cb50SEnrico Granata     {eFormatEnum, 'E', "enumeration"},
455548cb50SEnrico Granata     {eFormatHex, 'x', "hex"},
465548cb50SEnrico Granata     {eFormatHexUppercase, 'X', "uppercase hex"},
475548cb50SEnrico Granata     {eFormatFloat, 'f', "float"},
485548cb50SEnrico Granata     {eFormatOctal, 'o', "octal"},
495548cb50SEnrico Granata     {eFormatOSType, 'O', "OSType"},
505548cb50SEnrico Granata     {eFormatUnicode16, 'U', "unicode16"},
515548cb50SEnrico Granata     {eFormatUnicode32, '\0', "unicode32"},
525548cb50SEnrico Granata     {eFormatUnsigned, 'u', "unsigned decimal"},
535548cb50SEnrico Granata     {eFormatPointer, 'p', "pointer"},
545548cb50SEnrico Granata     {eFormatVectorOfChar, '\0', "char[]"},
555548cb50SEnrico Granata     {eFormatVectorOfSInt8, '\0', "int8_t[]"},
565548cb50SEnrico Granata     {eFormatVectorOfUInt8, '\0', "uint8_t[]"},
575548cb50SEnrico Granata     {eFormatVectorOfSInt16, '\0', "int16_t[]"},
585548cb50SEnrico Granata     {eFormatVectorOfUInt16, '\0', "uint16_t[]"},
595548cb50SEnrico Granata     {eFormatVectorOfSInt32, '\0', "int32_t[]"},
605548cb50SEnrico Granata     {eFormatVectorOfUInt32, '\0', "uint32_t[]"},
615548cb50SEnrico Granata     {eFormatVectorOfSInt64, '\0', "int64_t[]"},
625548cb50SEnrico Granata     {eFormatVectorOfUInt64, '\0', "uint64_t[]"},
63a0f08674SEwan Crawford     {eFormatVectorOfFloat16, '\0', "float16[]"},
645548cb50SEnrico Granata     {eFormatVectorOfFloat32, '\0', "float32[]"},
655548cb50SEnrico Granata     {eFormatVectorOfFloat64, '\0', "float64[]"},
665548cb50SEnrico Granata     {eFormatVectorOfUInt128, '\0', "uint128_t[]"},
675548cb50SEnrico Granata     {eFormatComplexInteger, 'I', "complex integer"},
685548cb50SEnrico Granata     {eFormatCharArray, 'a', "character array"},
695548cb50SEnrico Granata     {eFormatAddressInfo, 'A', "address"},
705548cb50SEnrico Granata     {eFormatHexFloat, '\0', "hex float"},
715548cb50SEnrico Granata     {eFormatInstruction, 'i', "instruction"},
722621f7bdSJonas Devlieghere     {eFormatVoid, 'v', "void"},
732621f7bdSJonas Devlieghere     {eFormatUnicode8, 'u', "unicode8"},
742621f7bdSJonas Devlieghere };
755548cb50SEnrico Granata 
7612002fbdSJonas Devlieghere static_assert((sizeof(g_format_infos) / sizeof(g_format_infos[0])) ==
7712002fbdSJonas Devlieghere                   kNumFormats,
7812002fbdSJonas Devlieghere               "All formats must have a corresponding info entry.");
7912002fbdSJonas Devlieghere 
8028606954SSaleem Abdulrasool static uint32_t g_num_format_infos = llvm::array_lengthof(g_format_infos);
815548cb50SEnrico Granata 
82b9c1b51eSKate Stone static bool GetFormatFromFormatChar(char format_char, Format &format) {
83b9c1b51eSKate Stone   for (uint32_t i = 0; i < g_num_format_infos; ++i) {
84b9c1b51eSKate Stone     if (g_format_infos[i].format_char == format_char) {
855548cb50SEnrico Granata       format = g_format_infos[i].format;
865548cb50SEnrico Granata       return true;
875548cb50SEnrico Granata     }
885548cb50SEnrico Granata   }
895548cb50SEnrico Granata   format = eFormatInvalid;
905548cb50SEnrico Granata   return false;
915548cb50SEnrico Granata }
925548cb50SEnrico Granata 
93b9c1b51eSKate Stone static bool GetFormatFromFormatName(const char *format_name,
94b9c1b51eSKate Stone                                     bool partial_match_ok, Format &format) {
955548cb50SEnrico Granata   uint32_t i;
96b9c1b51eSKate Stone   for (i = 0; i < g_num_format_infos; ++i) {
97b9c1b51eSKate Stone     if (strcasecmp(g_format_infos[i].format_name, format_name) == 0) {
985548cb50SEnrico Granata       format = g_format_infos[i].format;
995548cb50SEnrico Granata       return true;
1005548cb50SEnrico Granata     }
1015548cb50SEnrico Granata   }
1025548cb50SEnrico Granata 
103b9c1b51eSKate Stone   if (partial_match_ok) {
104b9c1b51eSKate Stone     for (i = 0; i < g_num_format_infos; ++i) {
105b9c1b51eSKate Stone       if (strcasestr(g_format_infos[i].format_name, format_name) ==
106b9c1b51eSKate Stone           g_format_infos[i].format_name) {
1075548cb50SEnrico Granata         format = g_format_infos[i].format;
1085548cb50SEnrico Granata         return true;
1095548cb50SEnrico Granata       }
1105548cb50SEnrico Granata     }
1115548cb50SEnrico Granata   }
1125548cb50SEnrico Granata   format = eFormatInvalid;
1135548cb50SEnrico Granata   return false;
1145548cb50SEnrico Granata }
1155548cb50SEnrico Granata 
116b9c1b51eSKate Stone void FormatManager::Changed() {
117bd5eab82SEnrico Granata   ++m_last_revision;
118bd5eab82SEnrico Granata   m_format_cache.Clear();
11916ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
120b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
121bd5eab82SEnrico Granata     if (iter.second)
122bd5eab82SEnrico Granata       iter.second->GetFormatCache().Clear();
123bd5eab82SEnrico Granata   }
124bd5eab82SEnrico Granata }
125bd5eab82SEnrico Granata 
126b9c1b51eSKate Stone bool FormatManager::GetFormatFromCString(const char *format_cstr,
1275548cb50SEnrico Granata                                          bool partial_match_ok,
128b9c1b51eSKate Stone                                          lldb::Format &format) {
1295548cb50SEnrico Granata   bool success = false;
130b9c1b51eSKate Stone   if (format_cstr && format_cstr[0]) {
131b9c1b51eSKate Stone     if (format_cstr[1] == '\0') {
1325548cb50SEnrico Granata       success = GetFormatFromFormatChar(format_cstr[0], format);
1335548cb50SEnrico Granata       if (success)
1345548cb50SEnrico Granata         return true;
1355548cb50SEnrico Granata     }
1365548cb50SEnrico Granata 
1375548cb50SEnrico Granata     success = GetFormatFromFormatName(format_cstr, partial_match_ok, format);
1385548cb50SEnrico Granata   }
1395548cb50SEnrico Granata   if (!success)
1405548cb50SEnrico Granata     format = eFormatInvalid;
1415548cb50SEnrico Granata   return success;
1425548cb50SEnrico Granata }
1435548cb50SEnrico Granata 
144b9c1b51eSKate Stone char FormatManager::GetFormatAsFormatChar(lldb::Format format) {
145b9c1b51eSKate Stone   for (uint32_t i = 0; i < g_num_format_infos; ++i) {
1465548cb50SEnrico Granata     if (g_format_infos[i].format == format)
1475548cb50SEnrico Granata       return g_format_infos[i].format_char;
1485548cb50SEnrico Granata   }
1495548cb50SEnrico Granata   return '\0';
1505548cb50SEnrico Granata }
1515548cb50SEnrico Granata 
152b9c1b51eSKate Stone const char *FormatManager::GetFormatAsCString(Format format) {
1535548cb50SEnrico Granata   if (format >= eFormatDefault && format < kNumFormats)
1545548cb50SEnrico Granata     return g_format_infos[format].format_name;
155248a1305SKonrad Kleine   return nullptr;
1565548cb50SEnrico Granata }
1575548cb50SEnrico Granata 
158b9c1b51eSKate Stone void FormatManager::EnableAllCategories() {
15933e97e63SEnrico Granata   m_categories_map.EnableAllCategories();
16016ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
161b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
16233e97e63SEnrico Granata     if (iter.second)
16333e97e63SEnrico Granata       iter.second->Enable();
16433e97e63SEnrico Granata   }
16533e97e63SEnrico Granata }
16633e97e63SEnrico Granata 
167b9c1b51eSKate Stone void FormatManager::DisableAllCategories() {
16833e97e63SEnrico Granata   m_categories_map.DisableAllCategories();
16916ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
170b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
17133e97e63SEnrico Granata     if (iter.second)
17233e97e63SEnrico Granata       iter.second->Disable();
17333e97e63SEnrico Granata   }
17433e97e63SEnrico Granata }
17533e97e63SEnrico Granata 
176b9c1b51eSKate Stone void FormatManager::GetPossibleMatches(
177b9c1b51eSKate Stone     ValueObject &valobj, CompilerType compiler_type, uint32_t reason,
178b9c1b51eSKate Stone     lldb::DynamicValueType use_dynamic, FormattersMatchVector &entries,
179b9c1b51eSKate Stone     bool did_strip_ptr, bool did_strip_ref, bool did_strip_typedef,
180b9c1b51eSKate Stone     bool root_level) {
181c6bf2e2dSEnrico Granata   compiler_type = compiler_type.GetTypeForFormatters();
18230ce956aSRaphael Isemann   ConstString type_name(compiler_type.GetTypeName());
183b9c1b51eSKate Stone   if (valobj.GetBitfieldBitSize() > 0) {
184de61cecdSEnrico Granata     StreamString sstring;
185de61cecdSEnrico Granata     sstring.Printf("%s:%d", type_name.AsCString(), valobj.GetBitfieldBitSize());
186c156427dSZachary Turner     ConstString bitfieldname(sstring.GetString());
187b9c1b51eSKate Stone     entries.push_back(
188b9c1b51eSKate Stone         {bitfieldname, 0, did_strip_ptr, did_strip_ref, did_strip_typedef});
189de61cecdSEnrico Granata     reason |= lldb_private::eFormatterChoiceCriterionStrippedBitField;
190de61cecdSEnrico Granata   }
191b3f0c340SEnrico Granata 
19278f05d35SDavide Italiano   if (!compiler_type.IsMeaninglessWithoutDynamicResolution()) {
193b9c1b51eSKate Stone     entries.push_back(
194b9c1b51eSKate Stone         {type_name, reason, did_strip_ptr, did_strip_ref, did_strip_typedef});
195de61cecdSEnrico Granata 
196*785df616SRaphael Isemann     ConstString display_type_name(compiler_type.GetTypeName());
197e8daa2f8SEnrico Granata     if (display_type_name != type_name)
19878f05d35SDavide Italiano       entries.push_back({display_type_name, reason, did_strip_ptr,
19978f05d35SDavide Italiano                          did_strip_ref, did_strip_typedef});
20078f05d35SDavide Italiano   }
201e8daa2f8SEnrico Granata 
202b9c1b51eSKate Stone   for (bool is_rvalue_ref = true, j = true;
203b9c1b51eSKate Stone        j && compiler_type.IsReferenceType(nullptr, &is_rvalue_ref); j = false) {
20459b5a37dSBruce Mitchener     CompilerType non_ref_type = compiler_type.GetNonReferenceType();
205b9c1b51eSKate Stone     GetPossibleMatches(
206b9c1b51eSKate Stone         valobj, non_ref_type,
207b9c1b51eSKate Stone         reason |
208b9c1b51eSKate Stone             lldb_private::eFormatterChoiceCriterionStrippedPointerReference,
209b9c1b51eSKate Stone         use_dynamic, entries, did_strip_ptr, true, did_strip_typedef);
210b9c1b51eSKate Stone     if (non_ref_type.IsTypedefType()) {
211a1e5dc86SGreg Clayton       CompilerType deffed_referenced_type = non_ref_type.GetTypedefedType();
212b9c1b51eSKate Stone       deffed_referenced_type =
213b9c1b51eSKate Stone           is_rvalue_ref ? deffed_referenced_type.GetRValueReferenceType()
214b9c1b51eSKate Stone                         : deffed_referenced_type.GetLValueReferenceType();
215b9c1b51eSKate Stone       GetPossibleMatches(
216b9c1b51eSKate Stone           valobj, deffed_referenced_type,
2171ac62963SEnrico Granata           reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs,
218b9c1b51eSKate Stone           use_dynamic, entries, did_strip_ptr, did_strip_ref,
2191ac62963SEnrico Granata           true); // this is not exactly the usual meaning of stripping typedefs
220de61cecdSEnrico Granata     }
2211ac62963SEnrico Granata   }
2221ac62963SEnrico Granata 
223b9c1b51eSKate Stone   if (compiler_type.IsPointerType()) {
22459b5a37dSBruce Mitchener     CompilerType non_ptr_type = compiler_type.GetPointeeType();
225b9c1b51eSKate Stone     GetPossibleMatches(
226b9c1b51eSKate Stone         valobj, non_ptr_type,
227b9c1b51eSKate Stone         reason |
228b9c1b51eSKate Stone             lldb_private::eFormatterChoiceCriterionStrippedPointerReference,
229b9c1b51eSKate Stone         use_dynamic, entries, true, did_strip_ref, did_strip_typedef);
230b9c1b51eSKate Stone     if (non_ptr_type.IsTypedefType()) {
231b9c1b51eSKate Stone       CompilerType deffed_pointed_type =
232b9c1b51eSKate Stone           non_ptr_type.GetTypedefedType().GetPointerType();
23390297427SJaroslav Sevcik       const bool stripped_typedef = true;
234b9c1b51eSKate Stone       GetPossibleMatches(
235b9c1b51eSKate Stone           valobj, deffed_pointed_type,
2361ac62963SEnrico Granata           reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs,
237b9c1b51eSKate Stone           use_dynamic, entries, did_strip_ptr, did_strip_ref,
23890297427SJaroslav Sevcik           stripped_typedef); // this is not exactly the usual meaning of
23990297427SJaroslav Sevcik                              // stripping typedefs
24090297427SJaroslav Sevcik     }
24190297427SJaroslav Sevcik   }
24290297427SJaroslav Sevcik 
24390297427SJaroslav Sevcik   // For arrays with typedef-ed elements, we add a candidate with the typedef
24490297427SJaroslav Sevcik   // stripped.
24590297427SJaroslav Sevcik   uint64_t array_size;
24690297427SJaroslav Sevcik   if (compiler_type.IsArrayType(nullptr, &array_size, nullptr)) {
24790297427SJaroslav Sevcik     CompilerType element_type = compiler_type.GetArrayElementType();
24890297427SJaroslav Sevcik     if (element_type.IsTypedefType()) {
24990297427SJaroslav Sevcik       // Get the stripped element type and compute the stripped array type
25090297427SJaroslav Sevcik       // from it.
25190297427SJaroslav Sevcik       CompilerType deffed_array_type =
25290297427SJaroslav Sevcik           element_type.GetTypedefedType().GetArrayType(array_size);
25390297427SJaroslav Sevcik       const bool stripped_typedef = true;
25490297427SJaroslav Sevcik       GetPossibleMatches(
25590297427SJaroslav Sevcik           valobj, deffed_array_type,
25690297427SJaroslav Sevcik           reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs,
25790297427SJaroslav Sevcik           use_dynamic, entries, did_strip_ptr, did_strip_ref,
25890297427SJaroslav Sevcik           stripped_typedef); // this is not exactly the usual meaning of
25990297427SJaroslav Sevcik                              // stripping typedefs
2601ac62963SEnrico Granata     }
261de61cecdSEnrico Granata   }
262de61cecdSEnrico Granata 
263bc69dd2cSDavide Italiano   for (lldb::LanguageType language_type :
264bc69dd2cSDavide Italiano        GetCandidateLanguages(valobj.GetObjectRuntimeLanguage())) {
265b9c1b51eSKate Stone     if (Language *language = Language::FindPlugin(language_type)) {
266b9c1b51eSKate Stone       for (ConstString candidate :
267b9c1b51eSKate Stone            language->GetPossibleFormattersMatches(valobj, use_dynamic)) {
268b9c1b51eSKate Stone         entries.push_back(
269b9c1b51eSKate Stone             {candidate,
270d3233c1eSEnrico Granata              reason | lldb_private::eFormatterChoiceCriterionLanguagePlugin,
271b9c1b51eSKate Stone              did_strip_ptr, did_strip_ref, did_strip_typedef});
272d3233c1eSEnrico Granata       }
273d3233c1eSEnrico Granata     }
274de61cecdSEnrico Granata   }
275de61cecdSEnrico Granata 
276de61cecdSEnrico Granata   // try to strip typedef chains
277b9c1b51eSKate Stone   if (compiler_type.IsTypedefType()) {
27859b5a37dSBruce Mitchener     CompilerType deffed_type = compiler_type.GetTypedefedType();
279b9c1b51eSKate Stone     GetPossibleMatches(
280b9c1b51eSKate Stone         valobj, deffed_type,
281de61cecdSEnrico Granata         reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs,
282b9c1b51eSKate Stone         use_dynamic, entries, did_strip_ptr, did_strip_ref, true);
283de61cecdSEnrico Granata   }
284de61cecdSEnrico Granata 
285b9c1b51eSKate Stone   if (root_level) {
286de61cecdSEnrico Granata     do {
28759b5a37dSBruce Mitchener       if (!compiler_type.IsValid())
288de61cecdSEnrico Granata         break;
289de61cecdSEnrico Granata 
290b9c1b51eSKate Stone       CompilerType unqual_compiler_ast_type =
291b9c1b51eSKate Stone           compiler_type.GetFullyUnqualifiedType();
29259b5a37dSBruce Mitchener       if (!unqual_compiler_ast_type.IsValid())
293de61cecdSEnrico Granata         break;
294b9c1b51eSKate Stone       if (unqual_compiler_ast_type.GetOpaqueQualType() !=
295b9c1b51eSKate Stone           compiler_type.GetOpaqueQualType())
296b9c1b51eSKate Stone         GetPossibleMatches(valobj, unqual_compiler_ast_type, reason,
297b9c1b51eSKate Stone                            use_dynamic, entries, did_strip_ptr, did_strip_ref,
298de61cecdSEnrico Granata                            did_strip_typedef);
299de61cecdSEnrico Granata     } while (false);
300de61cecdSEnrico Granata 
301de61cecdSEnrico Granata     // if all else fails, go to static type
302b9c1b51eSKate Stone     if (valobj.IsDynamic()) {
303de61cecdSEnrico Granata       lldb::ValueObjectSP static_value_sp(valobj.GetStaticValue());
304de61cecdSEnrico Granata       if (static_value_sp)
305b9c1b51eSKate Stone         GetPossibleMatches(
306b9c1b51eSKate Stone             *static_value_sp.get(), static_value_sp->GetCompilerType(),
307de61cecdSEnrico Granata             reason | lldb_private::eFormatterChoiceCriterionWentToStaticValue,
308b9c1b51eSKate Stone             use_dynamic, entries, did_strip_ptr, did_strip_ref,
309b9c1b51eSKate Stone             did_strip_typedef, true);
310de61cecdSEnrico Granata     }
311de61cecdSEnrico Granata   }
312de61cecdSEnrico Granata }
313de61cecdSEnrico Granata 
314852cc954SEnrico Granata lldb::TypeFormatImplSP
315b9c1b51eSKate Stone FormatManager::GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp) {
316852cc954SEnrico Granata   if (!type_sp)
317852cc954SEnrico Granata     return lldb::TypeFormatImplSP();
318852cc954SEnrico Granata   lldb::TypeFormatImplSP format_chosen_sp;
319852cc954SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
320852cc954SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
321852cc954SEnrico Granata   uint32_t prio_category = UINT32_MAX;
322b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
323852cc954SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
324a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
325852cc954SEnrico Granata       continue;
326b9c1b51eSKate Stone     lldb::TypeFormatImplSP format_current_sp =
327b9c1b51eSKate Stone         category_sp->GetFormatForType(type_sp);
328b9c1b51eSKate Stone     if (format_current_sp &&
329248a1305SKonrad Kleine         (format_chosen_sp.get() == nullptr ||
330b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
331852cc954SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
332852cc954SEnrico Granata       format_chosen_sp = format_current_sp;
333852cc954SEnrico Granata     }
334852cc954SEnrico Granata   }
335852cc954SEnrico Granata   return format_chosen_sp;
336852cc954SEnrico Granata }
337852cc954SEnrico Granata 
3385548cb50SEnrico Granata lldb::TypeSummaryImplSP
339b9c1b51eSKate Stone FormatManager::GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp) {
3405548cb50SEnrico Granata   if (!type_sp)
3415548cb50SEnrico Granata     return lldb::TypeSummaryImplSP();
3425548cb50SEnrico Granata   lldb::TypeSummaryImplSP summary_chosen_sp;
3435548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3445548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3455548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
346b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3475548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
348a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3495548cb50SEnrico Granata       continue;
350b9c1b51eSKate Stone     lldb::TypeSummaryImplSP summary_current_sp =
351b9c1b51eSKate Stone         category_sp->GetSummaryForType(type_sp);
352b9c1b51eSKate Stone     if (summary_current_sp &&
353248a1305SKonrad Kleine         (summary_chosen_sp.get() == nullptr ||
354b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3555548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3565548cb50SEnrico Granata       summary_chosen_sp = summary_current_sp;
3575548cb50SEnrico Granata     }
3585548cb50SEnrico Granata   }
3595548cb50SEnrico Granata   return summary_chosen_sp;
3605548cb50SEnrico Granata }
3615548cb50SEnrico Granata 
3625548cb50SEnrico Granata lldb::TypeFilterImplSP
363b9c1b51eSKate Stone FormatManager::GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp) {
3645548cb50SEnrico Granata   if (!type_sp)
3655548cb50SEnrico Granata     return lldb::TypeFilterImplSP();
3665548cb50SEnrico Granata   lldb::TypeFilterImplSP filter_chosen_sp;
3675548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3685548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3695548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
370b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3715548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
372a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3735548cb50SEnrico Granata       continue;
374b9c1b51eSKate Stone     lldb::TypeFilterImplSP filter_current_sp(
375b9c1b51eSKate Stone         (TypeFilterImpl *)category_sp->GetFilterForType(type_sp).get());
376b9c1b51eSKate Stone     if (filter_current_sp &&
377248a1305SKonrad Kleine         (filter_chosen_sp.get() == nullptr ||
378b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3795548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3805548cb50SEnrico Granata       filter_chosen_sp = filter_current_sp;
3815548cb50SEnrico Granata     }
3825548cb50SEnrico Granata   }
3835548cb50SEnrico Granata   return filter_chosen_sp;
3845548cb50SEnrico Granata }
3855548cb50SEnrico Granata 
3865548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP
387b9c1b51eSKate Stone FormatManager::GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp) {
3885548cb50SEnrico Granata   if (!type_sp)
3895548cb50SEnrico Granata     return lldb::ScriptedSyntheticChildrenSP();
3905548cb50SEnrico Granata   lldb::ScriptedSyntheticChildrenSP synth_chosen_sp;
3915548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3925548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3935548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
394b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3955548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
396a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3975548cb50SEnrico Granata       continue;
398b9c1b51eSKate Stone     lldb::ScriptedSyntheticChildrenSP synth_current_sp(
399b9c1b51eSKate Stone         (ScriptedSyntheticChildren *)category_sp->GetSyntheticForType(type_sp)
400b9c1b51eSKate Stone             .get());
401b9c1b51eSKate Stone     if (synth_current_sp &&
402248a1305SKonrad Kleine         (synth_chosen_sp.get() == nullptr ||
403b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
4045548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
4055548cb50SEnrico Granata       synth_chosen_sp = synth_current_sp;
4065548cb50SEnrico Granata     }
4075548cb50SEnrico Granata   }
4085548cb50SEnrico Granata   return synth_chosen_sp;
4095548cb50SEnrico Granata }
4105548cb50SEnrico Granata 
411b9c1b51eSKate Stone void FormatManager::ForEachCategory(TypeCategoryMap::ForEachCallback callback) {
412b56d0103SEnrico Granata   m_categories_map.ForEach(callback);
41316ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
414b9c1b51eSKate Stone   for (const auto &entry : m_language_categories_map) {
415b9c1b51eSKate Stone     if (auto category_sp = entry.second->GetCategory()) {
416b56d0103SEnrico Granata       if (!callback(category_sp))
417b56d0103SEnrico Granata         break;
418b56d0103SEnrico Granata     }
419b56d0103SEnrico Granata   }
420b56d0103SEnrico Granata }
421b56d0103SEnrico Granata 
4225548cb50SEnrico Granata lldb::TypeCategoryImplSP
4230e4c4821SAdrian Prantl FormatManager::GetCategory(ConstString category_name, bool can_create) {
4245548cb50SEnrico Granata   if (!category_name)
4255548cb50SEnrico Granata     return GetCategory(m_default_category_name);
4265548cb50SEnrico Granata   lldb::TypeCategoryImplSP category;
4275548cb50SEnrico Granata   if (m_categories_map.Get(category_name, category))
4285548cb50SEnrico Granata     return category;
4295548cb50SEnrico Granata 
4305548cb50SEnrico Granata   if (!can_create)
4315548cb50SEnrico Granata     return lldb::TypeCategoryImplSP();
4325548cb50SEnrico Granata 
433b9c1b51eSKate Stone   m_categories_map.Add(
434b9c1b51eSKate Stone       category_name,
435b9c1b51eSKate Stone       lldb::TypeCategoryImplSP(new TypeCategoryImpl(this, category_name)));
4365548cb50SEnrico Granata   return GetCategory(category_name);
4375548cb50SEnrico Granata }
4385548cb50SEnrico Granata 
439b9c1b51eSKate Stone lldb::Format FormatManager::GetSingleItemFormat(lldb::Format vector_format) {
440b9c1b51eSKate Stone   switch (vector_format) {
4415548cb50SEnrico Granata   case eFormatVectorOfChar:
4425548cb50SEnrico Granata     return eFormatCharArray;
4435548cb50SEnrico Granata 
4445548cb50SEnrico Granata   case eFormatVectorOfSInt8:
4455548cb50SEnrico Granata   case eFormatVectorOfSInt16:
4465548cb50SEnrico Granata   case eFormatVectorOfSInt32:
4475548cb50SEnrico Granata   case eFormatVectorOfSInt64:
4485548cb50SEnrico Granata     return eFormatDecimal;
4495548cb50SEnrico Granata 
4505548cb50SEnrico Granata   case eFormatVectorOfUInt8:
4515548cb50SEnrico Granata   case eFormatVectorOfUInt16:
4525548cb50SEnrico Granata   case eFormatVectorOfUInt32:
4535548cb50SEnrico Granata   case eFormatVectorOfUInt64:
4545548cb50SEnrico Granata   case eFormatVectorOfUInt128:
4555548cb50SEnrico Granata     return eFormatHex;
4565548cb50SEnrico Granata 
457a0f08674SEwan Crawford   case eFormatVectorOfFloat16:
4585548cb50SEnrico Granata   case eFormatVectorOfFloat32:
4595548cb50SEnrico Granata   case eFormatVectorOfFloat64:
4605548cb50SEnrico Granata     return eFormatFloat;
4615548cb50SEnrico Granata 
4625548cb50SEnrico Granata   default:
4635548cb50SEnrico Granata     return lldb::eFormatInvalid;
4645548cb50SEnrico Granata   }
4655548cb50SEnrico Granata }
4665548cb50SEnrico Granata 
467b9c1b51eSKate Stone bool FormatManager::ShouldPrintAsOneLiner(ValueObject &valobj) {
468553fad5cSEnrico Granata   // if settings say no oneline whatsoever
469b9c1b51eSKate Stone   if (valobj.GetTargetSP().get() &&
470a6682a41SJonas Devlieghere       !valobj.GetTargetSP()->GetDebugger().GetAutoOneLineSummaries())
471553fad5cSEnrico Granata     return false; // then don't oneline
472553fad5cSEnrico Granata 
47342fa4af8SEnrico Granata   // if this object has a summary, then ask the summary
474a29cb0baSEnrico Granata   if (valobj.GetSummaryFormat().get() != nullptr)
47542fa4af8SEnrico Granata     return valobj.GetSummaryFormat()->IsOneLiner();
476a29cb0baSEnrico Granata 
477a29cb0baSEnrico Granata   // no children, no party
478a29cb0baSEnrico Granata   if (valobj.GetNumChildren() == 0)
479a29cb0baSEnrico Granata     return false;
480a29cb0baSEnrico Granata 
48105097246SAdrian Prantl   // ask the type if it has any opinion about this eLazyBoolCalculate == no
48205097246SAdrian Prantl   // opinion; other values should be self explanatory
4839c63f99aSEnrico Granata   CompilerType compiler_type(valobj.GetCompilerType());
484b9c1b51eSKate Stone   if (compiler_type.IsValid()) {
485b9c1b51eSKate Stone     switch (compiler_type.ShouldPrintAsOneLiner(&valobj)) {
4869c63f99aSEnrico Granata     case eLazyBoolNo:
4879c63f99aSEnrico Granata       return false;
4889c63f99aSEnrico Granata     case eLazyBoolYes:
4899c63f99aSEnrico Granata       return true;
4909c63f99aSEnrico Granata     case eLazyBoolCalculate:
4919c63f99aSEnrico Granata       break;
4929c63f99aSEnrico Granata     }
4939c63f99aSEnrico Granata   }
4949c63f99aSEnrico Granata 
495a29cb0baSEnrico Granata   size_t total_children_name_len = 0;
496a29cb0baSEnrico Granata 
497b9c1b51eSKate Stone   for (size_t idx = 0; idx < valobj.GetNumChildren(); idx++) {
498ddac7611SEnrico Granata     bool is_synth_val = false;
499a29cb0baSEnrico Granata     ValueObjectSP child_sp(valobj.GetChildAtIndex(idx, true));
500a29cb0baSEnrico Granata     // something is wrong here - bail out
501a29cb0baSEnrico Granata     if (!child_sp)
502a29cb0baSEnrico Granata       return false;
5036500061eSEnrico Granata 
5046500061eSEnrico Granata     // also ask the child's type if it has any opinion
5056500061eSEnrico Granata     CompilerType child_compiler_type(child_sp->GetCompilerType());
506b9c1b51eSKate Stone     if (child_compiler_type.IsValid()) {
507b9c1b51eSKate Stone       switch (child_compiler_type.ShouldPrintAsOneLiner(child_sp.get())) {
5086500061eSEnrico Granata       case eLazyBoolYes:
5096500061eSEnrico Granata       // an opinion of yes is only binding for the child, so keep going
5106500061eSEnrico Granata       case eLazyBoolCalculate:
5116500061eSEnrico Granata         break;
5126500061eSEnrico Granata       case eLazyBoolNo:
5136500061eSEnrico Granata         // but if the child says no, then it's a veto on the whole thing
5146500061eSEnrico Granata         return false;
5156500061eSEnrico Granata       }
5166500061eSEnrico Granata     }
5176500061eSEnrico Granata 
518b9c1b51eSKate Stone     // if we decided to define synthetic children for a type, we probably care
51905097246SAdrian Prantl     // enough to show them, but avoid nesting children in children
520b9c1b51eSKate Stone     if (child_sp->GetSyntheticChildren().get() != nullptr) {
521ddac7611SEnrico Granata       ValueObjectSP synth_sp(child_sp->GetSyntheticValue());
522ddac7611SEnrico Granata       // wait.. wat? just get out of here..
523ddac7611SEnrico Granata       if (!synth_sp)
524a29cb0baSEnrico Granata         return false;
525ddac7611SEnrico Granata       // but if we only have them to provide a value, keep going
526a6682a41SJonas Devlieghere       if (!synth_sp->MightHaveChildren() &&
527b9c1b51eSKate Stone           synth_sp->DoesProvideSyntheticValue())
528ddac7611SEnrico Granata         is_synth_val = true;
529ddac7611SEnrico Granata       else
530ddac7611SEnrico Granata         return false;
531ddac7611SEnrico Granata     }
532a29cb0baSEnrico Granata 
533a29cb0baSEnrico Granata     total_children_name_len += child_sp->GetName().GetLength();
534a29cb0baSEnrico Granata 
535a29cb0baSEnrico Granata     // 50 itself is a "randomly" chosen number - the idea is that
536a29cb0baSEnrico Granata     // overly long structs should not get this treatment
537a29cb0baSEnrico Granata     // FIXME: maybe make this a user-tweakable setting?
538a29cb0baSEnrico Granata     if (total_children_name_len > 50)
539a29cb0baSEnrico Granata       return false;
540a29cb0baSEnrico Granata 
541a29cb0baSEnrico Granata     // if a summary is there..
542b9c1b51eSKate Stone     if (child_sp->GetSummaryFormat()) {
543a29cb0baSEnrico Granata       // and it wants children, then bail out
5448a068e6cSEnrico Granata       if (child_sp->GetSummaryFormat()->DoesPrintChildren(child_sp.get()))
545a29cb0baSEnrico Granata         return false;
546a29cb0baSEnrico Granata     }
547a29cb0baSEnrico Granata 
548c89e4ca3SEnrico Granata     // if this child has children..
549b9c1b51eSKate Stone     if (child_sp->GetNumChildren()) {
550a29cb0baSEnrico Granata       // ...and no summary...
551b9c1b51eSKate Stone       // (if it had a summary and the summary wanted children, we would have
552b9c1b51eSKate Stone       // bailed out anyway
553b9c1b51eSKate Stone       //  so this only makes us bail out if this has no summary and we would
554b9c1b51eSKate Stone       //  then print children)
555b9c1b51eSKate Stone       if (!child_sp->GetSummaryFormat() && !is_synth_val) // but again only do
556b9c1b51eSKate Stone                                                           // that if not a
557b9c1b51eSKate Stone                                                           // synthetic valued
558b9c1b51eSKate Stone                                                           // child
559a29cb0baSEnrico Granata         return false;                                     // then bail out
560a29cb0baSEnrico Granata     }
561a29cb0baSEnrico Granata   }
562a29cb0baSEnrico Granata   return true;
563a29cb0baSEnrico Granata }
564a29cb0baSEnrico Granata 
5650e4c4821SAdrian Prantl ConstString FormatManager::GetValidTypeName(ConstString type) {
5665548cb50SEnrico Granata   return ::GetValidTypeName_Impl(type);
5675548cb50SEnrico Granata }
5685548cb50SEnrico Granata 
569b9c1b51eSKate Stone ConstString FormatManager::GetTypeForCache(ValueObject &valobj,
570b9c1b51eSKate Stone                                            lldb::DynamicValueType use_dynamic) {
571b9c1b51eSKate Stone   ValueObjectSP valobj_sp = valobj.GetQualifiedRepresentationIfAvailable(
572b9c1b51eSKate Stone       use_dynamic, valobj.IsSynthetic());
57378f05d35SDavide Italiano   if (valobj_sp && valobj_sp->GetCompilerType().IsValid()) {
57478f05d35SDavide Italiano     if (!valobj_sp->GetCompilerType().IsMeaninglessWithoutDynamicResolution())
575b3f0c340SEnrico Granata       return valobj_sp->GetQualifiedTypeName();
57678f05d35SDavide Italiano   }
5775548cb50SEnrico Granata   return ConstString();
5785548cb50SEnrico Granata }
5795548cb50SEnrico Granata 
580d3233c1eSEnrico Granata std::vector<lldb::LanguageType>
581bc69dd2cSDavide Italiano FormatManager::GetCandidateLanguages(lldb::LanguageType lang_type) {
582b9c1b51eSKate Stone   switch (lang_type) {
58333e97e63SEnrico Granata   case lldb::eLanguageTypeC:
58433e97e63SEnrico Granata   case lldb::eLanguageTypeC89:
58533e97e63SEnrico Granata   case lldb::eLanguageTypeC99:
58633e97e63SEnrico Granata   case lldb::eLanguageTypeC11:
58733e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus:
58833e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_03:
58933e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_11:
59033e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_14:
591170c395eSEnrico Granata     return {lldb::eLanguageTypeC_plus_plus, lldb::eLanguageTypeObjC};
592980c0484SEnrico Granata   default:
593980c0484SEnrico Granata     return {lang_type};
594980c0484SEnrico Granata   }
595295db41cSDavide Italiano   llvm_unreachable("Fully covered switch");
596980c0484SEnrico Granata }
597980c0484SEnrico Granata 
598980c0484SEnrico Granata LanguageCategory *
599b9c1b51eSKate Stone FormatManager::GetCategoryForLanguage(lldb::LanguageType lang_type) {
60016ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
601b9c1b51eSKate Stone   auto iter = m_language_categories_map.find(lang_type),
602b9c1b51eSKate Stone        end = m_language_categories_map.end();
603980c0484SEnrico Granata   if (iter != end)
604980c0484SEnrico Granata     return iter->second.get();
605980c0484SEnrico Granata   LanguageCategory *lang_category = new LanguageCategory(lang_type);
606b9c1b51eSKate Stone   m_language_categories_map[lang_type] =
607b9c1b51eSKate Stone       LanguageCategory::UniquePointer(lang_category);
608980c0484SEnrico Granata   return lang_category;
609980c0484SEnrico Granata }
610980c0484SEnrico Granata 
6117034794bSAdrian Prantl template <typename ImplSP>
6127034794bSAdrian Prantl ImplSP FormatManager::GetHardcoded(FormattersMatchData &match_data) {
6137034794bSAdrian Prantl   ImplSP retval_sp;
614b9c1b51eSKate Stone   for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) {
615b9c1b51eSKate Stone     if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) {
6168a9a8f39SEnrico Granata       if (lang_category->GetHardcoded(*this, match_data, retval_sp))
6177034794bSAdrian Prantl         return retval_sp;
6187034794bSAdrian Prantl     }
6197034794bSAdrian Prantl   }
6207034794bSAdrian Prantl   return retval_sp;
6217034794bSAdrian Prantl }
6227034794bSAdrian Prantl 
62362a6d977SAdrian Prantl template <typename ImplSP>
62462a6d977SAdrian Prantl ImplSP FormatManager::Get(ValueObject &valobj,
6257034794bSAdrian Prantl                           lldb::DynamicValueType use_dynamic) {
6267034794bSAdrian Prantl   FormattersMatchData match_data(valobj, use_dynamic);
62762a6d977SAdrian Prantl   if (ImplSP retval_sp = GetCached<ImplSP>(match_data))
62862a6d977SAdrian Prantl     return retval_sp;
62962a6d977SAdrian Prantl 
63062a6d977SAdrian Prantl   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS));
63170e3d0eaSAdrian Prantl 
63270e3d0eaSAdrian Prantl   LLDB_LOGF(log, "[%s] Search failed. Giving language a chance.", __FUNCTION__);
63370e3d0eaSAdrian Prantl   for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) {
63470e3d0eaSAdrian Prantl     if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) {
63570e3d0eaSAdrian Prantl       ImplSP retval_sp;
63670e3d0eaSAdrian Prantl       if (lang_category->Get(match_data, retval_sp))
63770e3d0eaSAdrian Prantl         if (retval_sp) {
63870e3d0eaSAdrian Prantl           LLDB_LOGF(log, "[%s] Language search success. Returning.",
63970e3d0eaSAdrian Prantl                     __FUNCTION__);
64070e3d0eaSAdrian Prantl           return retval_sp;
64170e3d0eaSAdrian Prantl         }
64270e3d0eaSAdrian Prantl     }
64370e3d0eaSAdrian Prantl   }
64470e3d0eaSAdrian Prantl 
64570e3d0eaSAdrian Prantl   LLDB_LOGF(log, "[%s] Search failed. Giving hardcoded a chance.",
64670e3d0eaSAdrian Prantl             __FUNCTION__);
64762a6d977SAdrian Prantl   return GetHardcoded<ImplSP>(match_data);
64862a6d977SAdrian Prantl }
64962a6d977SAdrian Prantl 
65062a6d977SAdrian Prantl template <typename ImplSP>
65162a6d977SAdrian Prantl ImplSP FormatManager::GetCached(FormattersMatchData &match_data) {
65262a6d977SAdrian Prantl   ImplSP retval_sp;
6537034794bSAdrian Prantl   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS));
6547034794bSAdrian Prantl   if (match_data.GetTypeForCache()) {
6557034794bSAdrian Prantl     LLDB_LOGF(log, "\n\n[%s] Looking into cache for type %s", __FUNCTION__,
6567034794bSAdrian Prantl               match_data.GetTypeForCache().AsCString("<invalid>"));
6577034794bSAdrian Prantl     if (m_format_cache.Get(match_data.GetTypeForCache(), retval_sp)) {
6587034794bSAdrian Prantl       if (log) {
6597034794bSAdrian Prantl         LLDB_LOGF(log, "[%s] Cache search success. Returning.", __FUNCTION__);
6607034794bSAdrian Prantl         LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}",
6617034794bSAdrian Prantl                   m_format_cache.GetCacheHits(),
6627034794bSAdrian Prantl                   m_format_cache.GetCacheMisses());
6637034794bSAdrian Prantl       }
6647034794bSAdrian Prantl       return retval_sp;
6657034794bSAdrian Prantl     }
6667034794bSAdrian Prantl     LLDB_LOGF(log, "[%s] Cache search failed. Going normal route",
6677034794bSAdrian Prantl               __FUNCTION__);
6687034794bSAdrian Prantl   }
6697034794bSAdrian Prantl 
6707034794bSAdrian Prantl   m_categories_map.Get(match_data, retval_sp);
6717034794bSAdrian Prantl   if (match_data.GetTypeForCache() && (!retval_sp || !retval_sp->NonCacheable())) {
6727034794bSAdrian Prantl     LLDB_LOGF(log, "[%s] Caching %p for type %s", __FUNCTION__,
6737034794bSAdrian Prantl               static_cast<void *>(retval_sp.get()),
6747034794bSAdrian Prantl               match_data.GetTypeForCache().AsCString("<invalid>"));
6757034794bSAdrian Prantl     m_format_cache.Set(match_data.GetTypeForCache(), retval_sp);
6767034794bSAdrian Prantl   }
6777034794bSAdrian Prantl   LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}",
6787034794bSAdrian Prantl             m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses());
6797cb59e1aSEnrico Granata   return retval_sp;
680686f3debSEnrico Granata }
681686f3debSEnrico Granata 
682852cc954SEnrico Granata lldb::TypeFormatImplSP
683852cc954SEnrico Granata FormatManager::GetFormat(ValueObject &valobj,
684b9c1b51eSKate Stone                          lldb::DynamicValueType use_dynamic) {
68562a6d977SAdrian Prantl   return Get<lldb::TypeFormatImplSP>(valobj, use_dynamic);
686686f3debSEnrico Granata }
687686f3debSEnrico Granata 
6885548cb50SEnrico Granata lldb::TypeSummaryImplSP
6895548cb50SEnrico Granata FormatManager::GetSummaryFormat(ValueObject &valobj,
690b9c1b51eSKate Stone                                 lldb::DynamicValueType use_dynamic) {
69162a6d977SAdrian Prantl   return Get<lldb::TypeSummaryImplSP>(valobj, use_dynamic);
692686f3debSEnrico Granata }
693686f3debSEnrico Granata 
6945548cb50SEnrico Granata lldb::SyntheticChildrenSP
6955548cb50SEnrico Granata FormatManager::GetSyntheticChildren(ValueObject &valobj,
696b9c1b51eSKate Stone                                     lldb::DynamicValueType use_dynamic) {
69762a6d977SAdrian Prantl   return Get<lldb::SyntheticChildrenSP>(valobj, use_dynamic);
6985548cb50SEnrico Granata }
6995548cb50SEnrico Granata 
70016ff8604SSaleem Abdulrasool FormatManager::FormatManager()
701b9c1b51eSKate Stone     : m_last_revision(0), m_format_cache(), m_language_categories_mutex(),
702b9c1b51eSKate Stone       m_language_categories_map(), m_named_summaries_map(this),
703b9c1b51eSKate Stone       m_categories_map(this), m_default_category_name(ConstString("default")),
7045548cb50SEnrico Granata       m_system_category_name(ConstString("system")),
705b9c1b51eSKate Stone       m_vectortypes_category_name(ConstString("VectorTypes")) {
7065548cb50SEnrico Granata   LoadSystemFormatters();
707170c395eSEnrico Granata   LoadVectorFormatters();
7085548cb50SEnrico Granata 
709b9c1b51eSKate Stone   EnableCategory(m_vectortypes_category_name, TypeCategoryMap::Last,
710b9c1b51eSKate Stone                  lldb::eLanguageTypeObjC_plus_plus);
711b9c1b51eSKate Stone   EnableCategory(m_system_category_name, TypeCategoryMap::Last,
712b9c1b51eSKate Stone                  lldb::eLanguageTypeObjC_plus_plus);
7135548cb50SEnrico Granata }
7145548cb50SEnrico Granata 
715b9c1b51eSKate Stone void FormatManager::LoadSystemFormatters() {
7165548cb50SEnrico Granata   TypeSummaryImpl::Flags string_flags;
7170337c27fSEnrico Granata   string_flags.SetCascades(true)
7185548cb50SEnrico Granata       .SetSkipPointers(true)
7195548cb50SEnrico Granata       .SetSkipReferences(false)
7205548cb50SEnrico Granata       .SetDontShowChildren(true)
7215548cb50SEnrico Granata       .SetDontShowValue(false)
7225548cb50SEnrico Granata       .SetShowMembersOneLiner(false)
7235548cb50SEnrico Granata       .SetHideItemNames(false);
7245548cb50SEnrico Granata 
725bc2c2b01SEnrico Granata   TypeSummaryImpl::Flags string_array_flags;
726d2911633SEnrico Granata   string_array_flags.SetCascades(true)
7275548cb50SEnrico Granata       .SetSkipPointers(true)
7285548cb50SEnrico Granata       .SetSkipReferences(false)
729320dcf68SEnrico Granata       .SetDontShowChildren(true)
7305548cb50SEnrico Granata       .SetDontShowValue(true)
7315548cb50SEnrico Granata       .SetShowMembersOneLiner(false)
732bc2c2b01SEnrico Granata       .SetHideItemNames(false);
733bc2c2b01SEnrico Granata 
734b9c1b51eSKate Stone   lldb::TypeSummaryImplSP string_format(
735b9c1b51eSKate Stone       new StringSummaryFormat(string_flags, "${var%s}"));
736bc2c2b01SEnrico Granata 
737b9c1b51eSKate Stone   lldb::TypeSummaryImplSP string_array_format(
738b9c1b51eSKate Stone       new StringSummaryFormat(string_array_flags, "${var%s}"));
739bc2c2b01SEnrico Granata 
7405aa1d819SJan Kratochvil   RegularExpression any_size_char_arr(llvm::StringRef("char \\[[0-9]+\\]"));
7415548cb50SEnrico Granata 
742b9c1b51eSKate Stone   TypeCategoryImpl::SharedPointer sys_category_sp =
743b9c1b51eSKate Stone       GetCategory(m_system_category_name);
7445548cb50SEnrico Granata 
745b9c1b51eSKate Stone   sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("char *"),
746b9c1b51eSKate Stone                                                     string_format);
747b9c1b51eSKate Stone   sys_category_sp->GetTypeSummariesContainer()->Add(
748b9c1b51eSKate Stone       ConstString("unsigned char *"), string_format);
7495aa1d819SJan Kratochvil   sys_category_sp->GetRegexTypeSummariesContainer()->Add(
7505aa1d819SJan Kratochvil       std::move(any_size_char_arr), string_array_format);
7515548cb50SEnrico Granata 
752b9c1b51eSKate Stone   lldb::TypeSummaryImplSP ostype_summary(
753b9c1b51eSKate Stone       new StringSummaryFormat(TypeSummaryImpl::Flags()
754b9c1b51eSKate Stone                                   .SetCascades(false)
7555548cb50SEnrico Granata                                   .SetSkipPointers(true)
7565548cb50SEnrico Granata                                   .SetSkipReferences(true)
7575548cb50SEnrico Granata                                   .SetDontShowChildren(true)
7585548cb50SEnrico Granata                                   .SetDontShowValue(false)
7595548cb50SEnrico Granata                                   .SetShowMembersOneLiner(false)
7605548cb50SEnrico Granata                                   .SetHideItemNames(false),
7615548cb50SEnrico Granata                               "${var%O}"));
7625548cb50SEnrico Granata 
763b9c1b51eSKate Stone   sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("OSType"),
764b9c1b51eSKate Stone                                                     ostype_summary);
7655548cb50SEnrico Granata 
7664cc21772SEnrico Granata   TypeFormatImpl::Flags fourchar_flags;
767b9c1b51eSKate Stone   fourchar_flags.SetCascades(true).SetSkipPointers(true).SetSkipReferences(
768b9c1b51eSKate Stone       true);
7694cc21772SEnrico Granata 
770b9c1b51eSKate Stone   AddFormat(sys_category_sp, lldb::eFormatOSType, ConstString("FourCharCode"),
771b9c1b51eSKate Stone             fourchar_flags);
7725548cb50SEnrico Granata }
7735548cb50SEnrico Granata 
774b9c1b51eSKate Stone void FormatManager::LoadVectorFormatters() {
775b9c1b51eSKate Stone   TypeCategoryImpl::SharedPointer vectors_category_sp =
776b9c1b51eSKate Stone       GetCategory(m_vectortypes_category_name);
7775548cb50SEnrico Granata 
7785548cb50SEnrico Granata   TypeSummaryImpl::Flags vector_flags;
7795548cb50SEnrico Granata   vector_flags.SetCascades(true)
7805548cb50SEnrico Granata       .SetSkipPointers(true)
7815548cb50SEnrico Granata       .SetSkipReferences(false)
7825548cb50SEnrico Granata       .SetDontShowChildren(true)
7835548cb50SEnrico Granata       .SetDontShowValue(false)
7845548cb50SEnrico Granata       .SetShowMembersOneLiner(true)
7855548cb50SEnrico Granata       .SetHideItemNames(true);
7865548cb50SEnrico Granata 
787b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "${var.uint128}",
788b9c1b51eSKate Stone                    ConstString("builtin_type_vec128"), vector_flags);
7895548cb50SEnrico Granata 
790b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("float [4]"),
7915548cb50SEnrico Granata                    vector_flags);
792b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("int32_t [4]"),
7935548cb50SEnrico Granata                    vector_flags);
794b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("int16_t [8]"),
7955548cb50SEnrico Granata                    vector_flags);
796b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vDouble"),
7975548cb50SEnrico Granata                    vector_flags);
798b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vFloat"),
7995548cb50SEnrico Granata                    vector_flags);
800b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt8"),
8015548cb50SEnrico Granata                    vector_flags);
802b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt16"),
8035548cb50SEnrico Granata                    vector_flags);
804b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt32"),
8055548cb50SEnrico Granata                    vector_flags);
806b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"),
8075548cb50SEnrico Granata                    vector_flags);
808b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt8"),
8095548cb50SEnrico Granata                    vector_flags);
810b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"),
8115548cb50SEnrico Granata                    vector_flags);
812b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt32"),
8135548cb50SEnrico Granata                    vector_flags);
814b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vBool32"),
8155548cb50SEnrico Granata                    vector_flags);
8165548cb50SEnrico Granata }
817