1c156427dSZachary Turner //===-- FormatManager.cpp ----------------------------------------*- C++-*-===//
25548cb50SEnrico Granata //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65548cb50SEnrico Granata //
75548cb50SEnrico Granata //===----------------------------------------------------------------------===//
85548cb50SEnrico Granata 
95548cb50SEnrico Granata #include "lldb/DataFormatters/FormatManager.h"
105548cb50SEnrico Granata 
11d717cc9fSEnrico Granata #include "llvm/ADT/STLExtras.h"
12d717cc9fSEnrico Granata 
135548cb50SEnrico Granata 
145548cb50SEnrico Granata #include "lldb/Core/Debugger.h"
15df7e79e6SEnrico Granata #include "lldb/DataFormatters/FormattersHelpers.h"
16980c0484SEnrico Granata #include "lldb/DataFormatters/LanguageCategory.h"
175548cb50SEnrico Granata #include "lldb/Target/ExecutionContext.h"
1833e97e63SEnrico Granata #include "lldb/Target/Language.h"
196f9e6901SZachary Turner #include "lldb/Utility/Log.h"
20980c0484SEnrico Granata 
215548cb50SEnrico Granata using namespace lldb;
225548cb50SEnrico Granata using namespace lldb_private;
23df7e79e6SEnrico Granata using namespace lldb_private::formatters;
245548cb50SEnrico Granata 
25b9c1b51eSKate Stone struct FormatInfo {
265548cb50SEnrico Granata   Format format;
27b9c1b51eSKate Stone   const char format_char;  // One or more format characters that can be used for
28b9c1b51eSKate Stone                            // this format.
29b9c1b51eSKate Stone   const char *format_name; // Long format name that can be used to specify the
30b9c1b51eSKate Stone                            // current format
315548cb50SEnrico Granata };
325548cb50SEnrico Granata 
3312002fbdSJonas Devlieghere static constexpr FormatInfo g_format_infos[] = {
345548cb50SEnrico Granata     {eFormatDefault, '\0', "default"},
355548cb50SEnrico Granata     {eFormatBoolean, 'B', "boolean"},
365548cb50SEnrico Granata     {eFormatBinary, 'b', "binary"},
375548cb50SEnrico Granata     {eFormatBytes, 'y', "bytes"},
385548cb50SEnrico Granata     {eFormatBytesWithASCII, 'Y', "bytes with ASCII"},
395548cb50SEnrico Granata     {eFormatChar, 'c', "character"},
405548cb50SEnrico Granata     {eFormatCharPrintable, 'C', "printable character"},
415548cb50SEnrico Granata     {eFormatComplexFloat, 'F', "complex float"},
425548cb50SEnrico Granata     {eFormatCString, 's', "c-string"},
435548cb50SEnrico Granata     {eFormatDecimal, 'd', "decimal"},
445548cb50SEnrico Granata     {eFormatEnum, 'E', "enumeration"},
455548cb50SEnrico Granata     {eFormatHex, 'x', "hex"},
465548cb50SEnrico Granata     {eFormatHexUppercase, 'X', "uppercase hex"},
475548cb50SEnrico Granata     {eFormatFloat, 'f', "float"},
485548cb50SEnrico Granata     {eFormatOctal, 'o', "octal"},
495548cb50SEnrico Granata     {eFormatOSType, 'O', "OSType"},
505548cb50SEnrico Granata     {eFormatUnicode16, 'U', "unicode16"},
515548cb50SEnrico Granata     {eFormatUnicode32, '\0', "unicode32"},
525548cb50SEnrico Granata     {eFormatUnsigned, 'u', "unsigned decimal"},
535548cb50SEnrico Granata     {eFormatPointer, 'p', "pointer"},
545548cb50SEnrico Granata     {eFormatVectorOfChar, '\0', "char[]"},
555548cb50SEnrico Granata     {eFormatVectorOfSInt8, '\0', "int8_t[]"},
565548cb50SEnrico Granata     {eFormatVectorOfUInt8, '\0', "uint8_t[]"},
575548cb50SEnrico Granata     {eFormatVectorOfSInt16, '\0', "int16_t[]"},
585548cb50SEnrico Granata     {eFormatVectorOfUInt16, '\0', "uint16_t[]"},
595548cb50SEnrico Granata     {eFormatVectorOfSInt32, '\0', "int32_t[]"},
605548cb50SEnrico Granata     {eFormatVectorOfUInt32, '\0', "uint32_t[]"},
615548cb50SEnrico Granata     {eFormatVectorOfSInt64, '\0', "int64_t[]"},
625548cb50SEnrico Granata     {eFormatVectorOfUInt64, '\0', "uint64_t[]"},
63a0f08674SEwan Crawford     {eFormatVectorOfFloat16, '\0', "float16[]"},
645548cb50SEnrico Granata     {eFormatVectorOfFloat32, '\0', "float32[]"},
655548cb50SEnrico Granata     {eFormatVectorOfFloat64, '\0', "float64[]"},
665548cb50SEnrico Granata     {eFormatVectorOfUInt128, '\0', "uint128_t[]"},
675548cb50SEnrico Granata     {eFormatComplexInteger, 'I', "complex integer"},
685548cb50SEnrico Granata     {eFormatCharArray, 'a', "character array"},
695548cb50SEnrico Granata     {eFormatAddressInfo, 'A', "address"},
705548cb50SEnrico Granata     {eFormatHexFloat, '\0', "hex float"},
715548cb50SEnrico Granata     {eFormatInstruction, 'i', "instruction"},
722621f7bdSJonas Devlieghere     {eFormatVoid, 'v', "void"},
732621f7bdSJonas Devlieghere     {eFormatUnicode8, 'u', "unicode8"},
742621f7bdSJonas Devlieghere };
755548cb50SEnrico Granata 
7612002fbdSJonas Devlieghere static_assert((sizeof(g_format_infos) / sizeof(g_format_infos[0])) ==
7712002fbdSJonas Devlieghere                   kNumFormats,
7812002fbdSJonas Devlieghere               "All formats must have a corresponding info entry.");
7912002fbdSJonas Devlieghere 
8028606954SSaleem Abdulrasool static uint32_t g_num_format_infos = llvm::array_lengthof(g_format_infos);
815548cb50SEnrico Granata 
82b9c1b51eSKate Stone static bool GetFormatFromFormatChar(char format_char, Format &format) {
83b9c1b51eSKate Stone   for (uint32_t i = 0; i < g_num_format_infos; ++i) {
84b9c1b51eSKate Stone     if (g_format_infos[i].format_char == format_char) {
855548cb50SEnrico Granata       format = g_format_infos[i].format;
865548cb50SEnrico Granata       return true;
875548cb50SEnrico Granata     }
885548cb50SEnrico Granata   }
895548cb50SEnrico Granata   format = eFormatInvalid;
905548cb50SEnrico Granata   return false;
915548cb50SEnrico Granata }
925548cb50SEnrico Granata 
93b9c1b51eSKate Stone static bool GetFormatFromFormatName(const char *format_name,
94b9c1b51eSKate Stone                                     bool partial_match_ok, Format &format) {
955548cb50SEnrico Granata   uint32_t i;
96b9c1b51eSKate Stone   for (i = 0; i < g_num_format_infos; ++i) {
97b9c1b51eSKate Stone     if (strcasecmp(g_format_infos[i].format_name, format_name) == 0) {
985548cb50SEnrico Granata       format = g_format_infos[i].format;
995548cb50SEnrico Granata       return true;
1005548cb50SEnrico Granata     }
1015548cb50SEnrico Granata   }
1025548cb50SEnrico Granata 
103b9c1b51eSKate Stone   if (partial_match_ok) {
104b9c1b51eSKate Stone     for (i = 0; i < g_num_format_infos; ++i) {
105b9c1b51eSKate Stone       if (strcasestr(g_format_infos[i].format_name, format_name) ==
106b9c1b51eSKate Stone           g_format_infos[i].format_name) {
1075548cb50SEnrico Granata         format = g_format_infos[i].format;
1085548cb50SEnrico Granata         return true;
1095548cb50SEnrico Granata       }
1105548cb50SEnrico Granata     }
1115548cb50SEnrico Granata   }
1125548cb50SEnrico Granata   format = eFormatInvalid;
1135548cb50SEnrico Granata   return false;
1145548cb50SEnrico Granata }
1155548cb50SEnrico Granata 
116b9c1b51eSKate Stone void FormatManager::Changed() {
117bd5eab82SEnrico Granata   ++m_last_revision;
118bd5eab82SEnrico Granata   m_format_cache.Clear();
11916ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
120b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
121bd5eab82SEnrico Granata     if (iter.second)
122bd5eab82SEnrico Granata       iter.second->GetFormatCache().Clear();
123bd5eab82SEnrico Granata   }
124bd5eab82SEnrico Granata }
125bd5eab82SEnrico Granata 
126b9c1b51eSKate Stone bool FormatManager::GetFormatFromCString(const char *format_cstr,
1275548cb50SEnrico Granata                                          bool partial_match_ok,
128b9c1b51eSKate Stone                                          lldb::Format &format) {
1295548cb50SEnrico Granata   bool success = false;
130b9c1b51eSKate Stone   if (format_cstr && format_cstr[0]) {
131b9c1b51eSKate Stone     if (format_cstr[1] == '\0') {
1325548cb50SEnrico Granata       success = GetFormatFromFormatChar(format_cstr[0], format);
1335548cb50SEnrico Granata       if (success)
1345548cb50SEnrico Granata         return true;
1355548cb50SEnrico Granata     }
1365548cb50SEnrico Granata 
1375548cb50SEnrico Granata     success = GetFormatFromFormatName(format_cstr, partial_match_ok, format);
1385548cb50SEnrico Granata   }
1395548cb50SEnrico Granata   if (!success)
1405548cb50SEnrico Granata     format = eFormatInvalid;
1415548cb50SEnrico Granata   return success;
1425548cb50SEnrico Granata }
1435548cb50SEnrico Granata 
144b9c1b51eSKate Stone char FormatManager::GetFormatAsFormatChar(lldb::Format format) {
145b9c1b51eSKate Stone   for (uint32_t i = 0; i < g_num_format_infos; ++i) {
1465548cb50SEnrico Granata     if (g_format_infos[i].format == format)
1475548cb50SEnrico Granata       return g_format_infos[i].format_char;
1485548cb50SEnrico Granata   }
1495548cb50SEnrico Granata   return '\0';
1505548cb50SEnrico Granata }
1515548cb50SEnrico Granata 
152b9c1b51eSKate Stone const char *FormatManager::GetFormatAsCString(Format format) {
1535548cb50SEnrico Granata   if (format >= eFormatDefault && format < kNumFormats)
1545548cb50SEnrico Granata     return g_format_infos[format].format_name;
155248a1305SKonrad Kleine   return nullptr;
1565548cb50SEnrico Granata }
1575548cb50SEnrico Granata 
158b9c1b51eSKate Stone void FormatManager::EnableAllCategories() {
15933e97e63SEnrico Granata   m_categories_map.EnableAllCategories();
16016ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
161b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
16233e97e63SEnrico Granata     if (iter.second)
16333e97e63SEnrico Granata       iter.second->Enable();
16433e97e63SEnrico Granata   }
16533e97e63SEnrico Granata }
16633e97e63SEnrico Granata 
167b9c1b51eSKate Stone void FormatManager::DisableAllCategories() {
16833e97e63SEnrico Granata   m_categories_map.DisableAllCategories();
16916ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
170b9c1b51eSKate Stone   for (auto &iter : m_language_categories_map) {
17133e97e63SEnrico Granata     if (iter.second)
17233e97e63SEnrico Granata       iter.second->Disable();
17333e97e63SEnrico Granata   }
17433e97e63SEnrico Granata }
17533e97e63SEnrico Granata 
176b9c1b51eSKate Stone void FormatManager::GetPossibleMatches(
177b9c1b51eSKate Stone     ValueObject &valobj, CompilerType compiler_type, uint32_t reason,
178b9c1b51eSKate Stone     lldb::DynamicValueType use_dynamic, FormattersMatchVector &entries,
179b9c1b51eSKate Stone     bool did_strip_ptr, bool did_strip_ref, bool did_strip_typedef,
180b9c1b51eSKate Stone     bool root_level) {
181c6bf2e2dSEnrico Granata   compiler_type = compiler_type.GetTypeForFormatters();
18259b5a37dSBruce Mitchener   ConstString type_name(compiler_type.GetConstTypeName());
183b9c1b51eSKate Stone   if (valobj.GetBitfieldBitSize() > 0) {
184de61cecdSEnrico Granata     StreamString sstring;
185de61cecdSEnrico Granata     sstring.Printf("%s:%d", type_name.AsCString(), valobj.GetBitfieldBitSize());
186c156427dSZachary Turner     ConstString bitfieldname(sstring.GetString());
187b9c1b51eSKate Stone     entries.push_back(
188b9c1b51eSKate Stone         {bitfieldname, 0, did_strip_ptr, did_strip_ref, did_strip_typedef});
189de61cecdSEnrico Granata     reason |= lldb_private::eFormatterChoiceCriterionStrippedBitField;
190de61cecdSEnrico Granata   }
191b3f0c340SEnrico Granata 
19278f05d35SDavide Italiano   if (!compiler_type.IsMeaninglessWithoutDynamicResolution()) {
193b9c1b51eSKate Stone     entries.push_back(
194b9c1b51eSKate Stone         {type_name, reason, did_strip_ptr, did_strip_ref, did_strip_typedef});
195de61cecdSEnrico Granata 
19659b5a37dSBruce Mitchener     ConstString display_type_name(compiler_type.GetDisplayTypeName());
197e8daa2f8SEnrico Granata     if (display_type_name != type_name)
19878f05d35SDavide Italiano       entries.push_back({display_type_name, reason, did_strip_ptr,
19978f05d35SDavide Italiano                          did_strip_ref, did_strip_typedef});
20078f05d35SDavide Italiano   }
201e8daa2f8SEnrico Granata 
202b9c1b51eSKate Stone   for (bool is_rvalue_ref = true, j = true;
203b9c1b51eSKate Stone        j && compiler_type.IsReferenceType(nullptr, &is_rvalue_ref); j = false) {
20459b5a37dSBruce Mitchener     CompilerType non_ref_type = compiler_type.GetNonReferenceType();
205b9c1b51eSKate Stone     GetPossibleMatches(
206b9c1b51eSKate Stone         valobj, non_ref_type,
207b9c1b51eSKate Stone         reason |
208b9c1b51eSKate Stone             lldb_private::eFormatterChoiceCriterionStrippedPointerReference,
209b9c1b51eSKate Stone         use_dynamic, entries, did_strip_ptr, true, did_strip_typedef);
210b9c1b51eSKate Stone     if (non_ref_type.IsTypedefType()) {
211a1e5dc86SGreg Clayton       CompilerType deffed_referenced_type = non_ref_type.GetTypedefedType();
212b9c1b51eSKate Stone       deffed_referenced_type =
213b9c1b51eSKate Stone           is_rvalue_ref ? deffed_referenced_type.GetRValueReferenceType()
214b9c1b51eSKate Stone                         : deffed_referenced_type.GetLValueReferenceType();
215b9c1b51eSKate Stone       GetPossibleMatches(
216b9c1b51eSKate Stone           valobj, deffed_referenced_type,
2171ac62963SEnrico Granata           reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs,
218b9c1b51eSKate Stone           use_dynamic, entries, did_strip_ptr, did_strip_ref,
2191ac62963SEnrico Granata           true); // this is not exactly the usual meaning of stripping typedefs
220de61cecdSEnrico Granata     }
2211ac62963SEnrico Granata   }
2221ac62963SEnrico Granata 
223b9c1b51eSKate Stone   if (compiler_type.IsPointerType()) {
22459b5a37dSBruce Mitchener     CompilerType non_ptr_type = compiler_type.GetPointeeType();
225b9c1b51eSKate Stone     GetPossibleMatches(
226b9c1b51eSKate Stone         valobj, non_ptr_type,
227b9c1b51eSKate Stone         reason |
228b9c1b51eSKate Stone             lldb_private::eFormatterChoiceCriterionStrippedPointerReference,
229b9c1b51eSKate Stone         use_dynamic, entries, true, did_strip_ref, did_strip_typedef);
230b9c1b51eSKate Stone     if (non_ptr_type.IsTypedefType()) {
231b9c1b51eSKate Stone       CompilerType deffed_pointed_type =
232b9c1b51eSKate Stone           non_ptr_type.GetTypedefedType().GetPointerType();
233b9c1b51eSKate Stone       GetPossibleMatches(
234b9c1b51eSKate Stone           valobj, deffed_pointed_type,
2351ac62963SEnrico Granata           reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs,
236b9c1b51eSKate Stone           use_dynamic, entries, did_strip_ptr, did_strip_ref,
2371ac62963SEnrico Granata           true); // this is not exactly the usual meaning of stripping typedefs
2381ac62963SEnrico Granata     }
239de61cecdSEnrico Granata   }
240de61cecdSEnrico Granata 
241bc69dd2cSDavide Italiano   for (lldb::LanguageType language_type :
242bc69dd2cSDavide Italiano        GetCandidateLanguages(valobj.GetObjectRuntimeLanguage())) {
243b9c1b51eSKate Stone     if (Language *language = Language::FindPlugin(language_type)) {
244b9c1b51eSKate Stone       for (ConstString candidate :
245b9c1b51eSKate Stone            language->GetPossibleFormattersMatches(valobj, use_dynamic)) {
246b9c1b51eSKate Stone         entries.push_back(
247b9c1b51eSKate Stone             {candidate,
248d3233c1eSEnrico Granata              reason | lldb_private::eFormatterChoiceCriterionLanguagePlugin,
249b9c1b51eSKate Stone              did_strip_ptr, did_strip_ref, did_strip_typedef});
250d3233c1eSEnrico Granata       }
251d3233c1eSEnrico Granata     }
252de61cecdSEnrico Granata   }
253de61cecdSEnrico Granata 
254de61cecdSEnrico Granata   // try to strip typedef chains
255b9c1b51eSKate Stone   if (compiler_type.IsTypedefType()) {
25659b5a37dSBruce Mitchener     CompilerType deffed_type = compiler_type.GetTypedefedType();
257b9c1b51eSKate Stone     GetPossibleMatches(
258b9c1b51eSKate Stone         valobj, deffed_type,
259de61cecdSEnrico Granata         reason | lldb_private::eFormatterChoiceCriterionNavigatedTypedefs,
260b9c1b51eSKate Stone         use_dynamic, entries, did_strip_ptr, did_strip_ref, true);
261de61cecdSEnrico Granata   }
262de61cecdSEnrico Granata 
263b9c1b51eSKate Stone   if (root_level) {
264de61cecdSEnrico Granata     do {
26559b5a37dSBruce Mitchener       if (!compiler_type.IsValid())
266de61cecdSEnrico Granata         break;
267de61cecdSEnrico Granata 
268b9c1b51eSKate Stone       CompilerType unqual_compiler_ast_type =
269b9c1b51eSKate Stone           compiler_type.GetFullyUnqualifiedType();
27059b5a37dSBruce Mitchener       if (!unqual_compiler_ast_type.IsValid())
271de61cecdSEnrico Granata         break;
272b9c1b51eSKate Stone       if (unqual_compiler_ast_type.GetOpaqueQualType() !=
273b9c1b51eSKate Stone           compiler_type.GetOpaqueQualType())
274b9c1b51eSKate Stone         GetPossibleMatches(valobj, unqual_compiler_ast_type, reason,
275b9c1b51eSKate Stone                            use_dynamic, entries, did_strip_ptr, did_strip_ref,
276de61cecdSEnrico Granata                            did_strip_typedef);
277de61cecdSEnrico Granata     } while (false);
278de61cecdSEnrico Granata 
279de61cecdSEnrico Granata     // if all else fails, go to static type
280b9c1b51eSKate Stone     if (valobj.IsDynamic()) {
281de61cecdSEnrico Granata       lldb::ValueObjectSP static_value_sp(valobj.GetStaticValue());
282de61cecdSEnrico Granata       if (static_value_sp)
283b9c1b51eSKate Stone         GetPossibleMatches(
284b9c1b51eSKate Stone             *static_value_sp.get(), static_value_sp->GetCompilerType(),
285de61cecdSEnrico Granata             reason | lldb_private::eFormatterChoiceCriterionWentToStaticValue,
286b9c1b51eSKate Stone             use_dynamic, entries, did_strip_ptr, did_strip_ref,
287b9c1b51eSKate Stone             did_strip_typedef, true);
288de61cecdSEnrico Granata     }
289de61cecdSEnrico Granata   }
290de61cecdSEnrico Granata }
291de61cecdSEnrico Granata 
292852cc954SEnrico Granata lldb::TypeFormatImplSP
293b9c1b51eSKate Stone FormatManager::GetFormatForType(lldb::TypeNameSpecifierImplSP type_sp) {
294852cc954SEnrico Granata   if (!type_sp)
295852cc954SEnrico Granata     return lldb::TypeFormatImplSP();
296852cc954SEnrico Granata   lldb::TypeFormatImplSP format_chosen_sp;
297852cc954SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
298852cc954SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
299852cc954SEnrico Granata   uint32_t prio_category = UINT32_MAX;
300b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
301852cc954SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
302a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
303852cc954SEnrico Granata       continue;
304b9c1b51eSKate Stone     lldb::TypeFormatImplSP format_current_sp =
305b9c1b51eSKate Stone         category_sp->GetFormatForType(type_sp);
306b9c1b51eSKate Stone     if (format_current_sp &&
307248a1305SKonrad Kleine         (format_chosen_sp.get() == nullptr ||
308b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
309852cc954SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
310852cc954SEnrico Granata       format_chosen_sp = format_current_sp;
311852cc954SEnrico Granata     }
312852cc954SEnrico Granata   }
313852cc954SEnrico Granata   return format_chosen_sp;
314852cc954SEnrico Granata }
315852cc954SEnrico Granata 
3165548cb50SEnrico Granata lldb::TypeSummaryImplSP
317b9c1b51eSKate Stone FormatManager::GetSummaryForType(lldb::TypeNameSpecifierImplSP type_sp) {
3185548cb50SEnrico Granata   if (!type_sp)
3195548cb50SEnrico Granata     return lldb::TypeSummaryImplSP();
3205548cb50SEnrico Granata   lldb::TypeSummaryImplSP summary_chosen_sp;
3215548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3225548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3235548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
324b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3255548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
326a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3275548cb50SEnrico Granata       continue;
328b9c1b51eSKate Stone     lldb::TypeSummaryImplSP summary_current_sp =
329b9c1b51eSKate Stone         category_sp->GetSummaryForType(type_sp);
330b9c1b51eSKate Stone     if (summary_current_sp &&
331248a1305SKonrad Kleine         (summary_chosen_sp.get() == nullptr ||
332b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3335548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3345548cb50SEnrico Granata       summary_chosen_sp = summary_current_sp;
3355548cb50SEnrico Granata     }
3365548cb50SEnrico Granata   }
3375548cb50SEnrico Granata   return summary_chosen_sp;
3385548cb50SEnrico Granata }
3395548cb50SEnrico Granata 
3405548cb50SEnrico Granata lldb::TypeFilterImplSP
341b9c1b51eSKate Stone FormatManager::GetFilterForType(lldb::TypeNameSpecifierImplSP type_sp) {
3425548cb50SEnrico Granata   if (!type_sp)
3435548cb50SEnrico Granata     return lldb::TypeFilterImplSP();
3445548cb50SEnrico Granata   lldb::TypeFilterImplSP filter_chosen_sp;
3455548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3465548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3475548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
348b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3495548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
350a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3515548cb50SEnrico Granata       continue;
352b9c1b51eSKate Stone     lldb::TypeFilterImplSP filter_current_sp(
353b9c1b51eSKate Stone         (TypeFilterImpl *)category_sp->GetFilterForType(type_sp).get());
354b9c1b51eSKate Stone     if (filter_current_sp &&
355248a1305SKonrad Kleine         (filter_chosen_sp.get() == nullptr ||
356b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3575548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3585548cb50SEnrico Granata       filter_chosen_sp = filter_current_sp;
3595548cb50SEnrico Granata     }
3605548cb50SEnrico Granata   }
3615548cb50SEnrico Granata   return filter_chosen_sp;
3625548cb50SEnrico Granata }
3635548cb50SEnrico Granata 
3645548cb50SEnrico Granata lldb::ScriptedSyntheticChildrenSP
365b9c1b51eSKate Stone FormatManager::GetSyntheticForType(lldb::TypeNameSpecifierImplSP type_sp) {
3665548cb50SEnrico Granata   if (!type_sp)
3675548cb50SEnrico Granata     return lldb::ScriptedSyntheticChildrenSP();
3685548cb50SEnrico Granata   lldb::ScriptedSyntheticChildrenSP synth_chosen_sp;
3695548cb50SEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
3705548cb50SEnrico Granata   lldb::TypeCategoryImplSP category_sp;
3715548cb50SEnrico Granata   uint32_t prio_category = UINT32_MAX;
372b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
3735548cb50SEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
374a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
3755548cb50SEnrico Granata       continue;
376b9c1b51eSKate Stone     lldb::ScriptedSyntheticChildrenSP synth_current_sp(
377b9c1b51eSKate Stone         (ScriptedSyntheticChildren *)category_sp->GetSyntheticForType(type_sp)
378b9c1b51eSKate Stone             .get());
379b9c1b51eSKate Stone     if (synth_current_sp &&
380248a1305SKonrad Kleine         (synth_chosen_sp.get() == nullptr ||
381b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
3825548cb50SEnrico Granata       prio_category = category_sp->GetEnabledPosition();
3835548cb50SEnrico Granata       synth_chosen_sp = synth_current_sp;
3845548cb50SEnrico Granata     }
3855548cb50SEnrico Granata   }
3865548cb50SEnrico Granata   return synth_chosen_sp;
3875548cb50SEnrico Granata }
3885548cb50SEnrico Granata 
389c582713cSEnrico Granata lldb::TypeValidatorImplSP
390b9c1b51eSKate Stone FormatManager::GetValidatorForType(lldb::TypeNameSpecifierImplSP type_sp) {
391c582713cSEnrico Granata   if (!type_sp)
392c582713cSEnrico Granata     return lldb::TypeValidatorImplSP();
393c582713cSEnrico Granata   lldb::TypeValidatorImplSP validator_chosen_sp;
394c582713cSEnrico Granata   uint32_t num_categories = m_categories_map.GetCount();
395c582713cSEnrico Granata   lldb::TypeCategoryImplSP category_sp;
396c582713cSEnrico Granata   uint32_t prio_category = UINT32_MAX;
397b9c1b51eSKate Stone   for (uint32_t category_id = 0; category_id < num_categories; category_id++) {
398c582713cSEnrico Granata     category_sp = GetCategoryAtIndex(category_id);
399a6682a41SJonas Devlieghere     if (!category_sp->IsEnabled())
400c582713cSEnrico Granata       continue;
401b9c1b51eSKate Stone     lldb::TypeValidatorImplSP validator_current_sp(
402b9c1b51eSKate Stone         category_sp->GetValidatorForType(type_sp).get());
403b9c1b51eSKate Stone     if (validator_current_sp &&
404248a1305SKonrad Kleine         (validator_chosen_sp.get() == nullptr ||
405b9c1b51eSKate Stone          (prio_category > category_sp->GetEnabledPosition()))) {
406c582713cSEnrico Granata       prio_category = category_sp->GetEnabledPosition();
407c582713cSEnrico Granata       validator_chosen_sp = validator_current_sp;
408c582713cSEnrico Granata     }
409c582713cSEnrico Granata   }
410c582713cSEnrico Granata   return validator_chosen_sp;
411c582713cSEnrico Granata }
412c582713cSEnrico Granata 
413b9c1b51eSKate Stone void FormatManager::ForEachCategory(TypeCategoryMap::ForEachCallback callback) {
414b56d0103SEnrico Granata   m_categories_map.ForEach(callback);
41516ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
416b9c1b51eSKate Stone   for (const auto &entry : m_language_categories_map) {
417b9c1b51eSKate Stone     if (auto category_sp = entry.second->GetCategory()) {
418b56d0103SEnrico Granata       if (!callback(category_sp))
419b56d0103SEnrico Granata         break;
420b56d0103SEnrico Granata     }
421b56d0103SEnrico Granata   }
422b56d0103SEnrico Granata }
423b56d0103SEnrico Granata 
4245548cb50SEnrico Granata lldb::TypeCategoryImplSP
4250e4c4821SAdrian Prantl FormatManager::GetCategory(ConstString category_name, bool can_create) {
4265548cb50SEnrico Granata   if (!category_name)
4275548cb50SEnrico Granata     return GetCategory(m_default_category_name);
4285548cb50SEnrico Granata   lldb::TypeCategoryImplSP category;
4295548cb50SEnrico Granata   if (m_categories_map.Get(category_name, category))
4305548cb50SEnrico Granata     return category;
4315548cb50SEnrico Granata 
4325548cb50SEnrico Granata   if (!can_create)
4335548cb50SEnrico Granata     return lldb::TypeCategoryImplSP();
4345548cb50SEnrico Granata 
435b9c1b51eSKate Stone   m_categories_map.Add(
436b9c1b51eSKate Stone       category_name,
437b9c1b51eSKate Stone       lldb::TypeCategoryImplSP(new TypeCategoryImpl(this, category_name)));
4385548cb50SEnrico Granata   return GetCategory(category_name);
4395548cb50SEnrico Granata }
4405548cb50SEnrico Granata 
441b9c1b51eSKate Stone lldb::Format FormatManager::GetSingleItemFormat(lldb::Format vector_format) {
442b9c1b51eSKate Stone   switch (vector_format) {
4435548cb50SEnrico Granata   case eFormatVectorOfChar:
4445548cb50SEnrico Granata     return eFormatCharArray;
4455548cb50SEnrico Granata 
4465548cb50SEnrico Granata   case eFormatVectorOfSInt8:
4475548cb50SEnrico Granata   case eFormatVectorOfSInt16:
4485548cb50SEnrico Granata   case eFormatVectorOfSInt32:
4495548cb50SEnrico Granata   case eFormatVectorOfSInt64:
4505548cb50SEnrico Granata     return eFormatDecimal;
4515548cb50SEnrico Granata 
4525548cb50SEnrico Granata   case eFormatVectorOfUInt8:
4535548cb50SEnrico Granata   case eFormatVectorOfUInt16:
4545548cb50SEnrico Granata   case eFormatVectorOfUInt32:
4555548cb50SEnrico Granata   case eFormatVectorOfUInt64:
4565548cb50SEnrico Granata   case eFormatVectorOfUInt128:
4575548cb50SEnrico Granata     return eFormatHex;
4585548cb50SEnrico Granata 
459a0f08674SEwan Crawford   case eFormatVectorOfFloat16:
4605548cb50SEnrico Granata   case eFormatVectorOfFloat32:
4615548cb50SEnrico Granata   case eFormatVectorOfFloat64:
4625548cb50SEnrico Granata     return eFormatFloat;
4635548cb50SEnrico Granata 
4645548cb50SEnrico Granata   default:
4655548cb50SEnrico Granata     return lldb::eFormatInvalid;
4665548cb50SEnrico Granata   }
4675548cb50SEnrico Granata }
4685548cb50SEnrico Granata 
469b9c1b51eSKate Stone bool FormatManager::ShouldPrintAsOneLiner(ValueObject &valobj) {
470553fad5cSEnrico Granata   // if settings say no oneline whatsoever
471b9c1b51eSKate Stone   if (valobj.GetTargetSP().get() &&
472a6682a41SJonas Devlieghere       !valobj.GetTargetSP()->GetDebugger().GetAutoOneLineSummaries())
473553fad5cSEnrico Granata     return false; // then don't oneline
474553fad5cSEnrico Granata 
47542fa4af8SEnrico Granata   // if this object has a summary, then ask the summary
476a29cb0baSEnrico Granata   if (valobj.GetSummaryFormat().get() != nullptr)
47742fa4af8SEnrico Granata     return valobj.GetSummaryFormat()->IsOneLiner();
478a29cb0baSEnrico Granata 
479a29cb0baSEnrico Granata   // no children, no party
480a29cb0baSEnrico Granata   if (valobj.GetNumChildren() == 0)
481a29cb0baSEnrico Granata     return false;
482a29cb0baSEnrico Granata 
48305097246SAdrian Prantl   // ask the type if it has any opinion about this eLazyBoolCalculate == no
48405097246SAdrian Prantl   // opinion; other values should be self explanatory
4859c63f99aSEnrico Granata   CompilerType compiler_type(valobj.GetCompilerType());
486b9c1b51eSKate Stone   if (compiler_type.IsValid()) {
487b9c1b51eSKate Stone     switch (compiler_type.ShouldPrintAsOneLiner(&valobj)) {
4889c63f99aSEnrico Granata     case eLazyBoolNo:
4899c63f99aSEnrico Granata       return false;
4909c63f99aSEnrico Granata     case eLazyBoolYes:
4919c63f99aSEnrico Granata       return true;
4929c63f99aSEnrico Granata     case eLazyBoolCalculate:
4939c63f99aSEnrico Granata       break;
4949c63f99aSEnrico Granata     }
4959c63f99aSEnrico Granata   }
4969c63f99aSEnrico Granata 
497a29cb0baSEnrico Granata   size_t total_children_name_len = 0;
498a29cb0baSEnrico Granata 
499b9c1b51eSKate Stone   for (size_t idx = 0; idx < valobj.GetNumChildren(); idx++) {
500ddac7611SEnrico Granata     bool is_synth_val = false;
501a29cb0baSEnrico Granata     ValueObjectSP child_sp(valobj.GetChildAtIndex(idx, true));
502a29cb0baSEnrico Granata     // something is wrong here - bail out
503a29cb0baSEnrico Granata     if (!child_sp)
504a29cb0baSEnrico Granata       return false;
5056500061eSEnrico Granata 
5066500061eSEnrico Granata     // also ask the child's type if it has any opinion
5076500061eSEnrico Granata     CompilerType child_compiler_type(child_sp->GetCompilerType());
508b9c1b51eSKate Stone     if (child_compiler_type.IsValid()) {
509b9c1b51eSKate Stone       switch (child_compiler_type.ShouldPrintAsOneLiner(child_sp.get())) {
5106500061eSEnrico Granata       case eLazyBoolYes:
5116500061eSEnrico Granata       // an opinion of yes is only binding for the child, so keep going
5126500061eSEnrico Granata       case eLazyBoolCalculate:
5136500061eSEnrico Granata         break;
5146500061eSEnrico Granata       case eLazyBoolNo:
5156500061eSEnrico Granata         // but if the child says no, then it's a veto on the whole thing
5166500061eSEnrico Granata         return false;
5176500061eSEnrico Granata       }
5186500061eSEnrico Granata     }
5196500061eSEnrico Granata 
520b9c1b51eSKate Stone     // if we decided to define synthetic children for a type, we probably care
52105097246SAdrian Prantl     // enough to show them, but avoid nesting children in children
522b9c1b51eSKate Stone     if (child_sp->GetSyntheticChildren().get() != nullptr) {
523ddac7611SEnrico Granata       ValueObjectSP synth_sp(child_sp->GetSyntheticValue());
524ddac7611SEnrico Granata       // wait.. wat? just get out of here..
525ddac7611SEnrico Granata       if (!synth_sp)
526a29cb0baSEnrico Granata         return false;
527ddac7611SEnrico Granata       // but if we only have them to provide a value, keep going
528a6682a41SJonas Devlieghere       if (!synth_sp->MightHaveChildren() &&
529b9c1b51eSKate Stone           synth_sp->DoesProvideSyntheticValue())
530ddac7611SEnrico Granata         is_synth_val = true;
531ddac7611SEnrico Granata       else
532ddac7611SEnrico Granata         return false;
533ddac7611SEnrico Granata     }
534a29cb0baSEnrico Granata 
535a29cb0baSEnrico Granata     total_children_name_len += child_sp->GetName().GetLength();
536a29cb0baSEnrico Granata 
537a29cb0baSEnrico Granata     // 50 itself is a "randomly" chosen number - the idea is that
538a29cb0baSEnrico Granata     // overly long structs should not get this treatment
539a29cb0baSEnrico Granata     // FIXME: maybe make this a user-tweakable setting?
540a29cb0baSEnrico Granata     if (total_children_name_len > 50)
541a29cb0baSEnrico Granata       return false;
542a29cb0baSEnrico Granata 
543a29cb0baSEnrico Granata     // if a summary is there..
544b9c1b51eSKate Stone     if (child_sp->GetSummaryFormat()) {
545a29cb0baSEnrico Granata       // and it wants children, then bail out
5468a068e6cSEnrico Granata       if (child_sp->GetSummaryFormat()->DoesPrintChildren(child_sp.get()))
547a29cb0baSEnrico Granata         return false;
548a29cb0baSEnrico Granata     }
549a29cb0baSEnrico Granata 
550c89e4ca3SEnrico Granata     // if this child has children..
551b9c1b51eSKate Stone     if (child_sp->GetNumChildren()) {
552a29cb0baSEnrico Granata       // ...and no summary...
553b9c1b51eSKate Stone       // (if it had a summary and the summary wanted children, we would have
554b9c1b51eSKate Stone       // bailed out anyway
555b9c1b51eSKate Stone       //  so this only makes us bail out if this has no summary and we would
556b9c1b51eSKate Stone       //  then print children)
557b9c1b51eSKate Stone       if (!child_sp->GetSummaryFormat() && !is_synth_val) // but again only do
558b9c1b51eSKate Stone                                                           // that if not a
559b9c1b51eSKate Stone                                                           // synthetic valued
560b9c1b51eSKate Stone                                                           // child
561a29cb0baSEnrico Granata         return false;                                     // then bail out
562a29cb0baSEnrico Granata     }
563a29cb0baSEnrico Granata   }
564a29cb0baSEnrico Granata   return true;
565a29cb0baSEnrico Granata }
566a29cb0baSEnrico Granata 
5670e4c4821SAdrian Prantl ConstString FormatManager::GetValidTypeName(ConstString type) {
5685548cb50SEnrico Granata   return ::GetValidTypeName_Impl(type);
5695548cb50SEnrico Granata }
5705548cb50SEnrico Granata 
571b9c1b51eSKate Stone ConstString FormatManager::GetTypeForCache(ValueObject &valobj,
572b9c1b51eSKate Stone                                            lldb::DynamicValueType use_dynamic) {
573b9c1b51eSKate Stone   ValueObjectSP valobj_sp = valobj.GetQualifiedRepresentationIfAvailable(
574b9c1b51eSKate Stone       use_dynamic, valobj.IsSynthetic());
57578f05d35SDavide Italiano   if (valobj_sp && valobj_sp->GetCompilerType().IsValid()) {
57678f05d35SDavide Italiano     if (!valobj_sp->GetCompilerType().IsMeaninglessWithoutDynamicResolution())
577b3f0c340SEnrico Granata       return valobj_sp->GetQualifiedTypeName();
57878f05d35SDavide Italiano   }
5795548cb50SEnrico Granata   return ConstString();
5805548cb50SEnrico Granata }
5815548cb50SEnrico Granata 
582d3233c1eSEnrico Granata std::vector<lldb::LanguageType>
583bc69dd2cSDavide Italiano FormatManager::GetCandidateLanguages(lldb::LanguageType lang_type) {
584b9c1b51eSKate Stone   switch (lang_type) {
58533e97e63SEnrico Granata   case lldb::eLanguageTypeC:
58633e97e63SEnrico Granata   case lldb::eLanguageTypeC89:
58733e97e63SEnrico Granata   case lldb::eLanguageTypeC99:
58833e97e63SEnrico Granata   case lldb::eLanguageTypeC11:
58933e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus:
59033e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_03:
59133e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_11:
59233e97e63SEnrico Granata   case lldb::eLanguageTypeC_plus_plus_14:
593170c395eSEnrico Granata     return {lldb::eLanguageTypeC_plus_plus, lldb::eLanguageTypeObjC};
594980c0484SEnrico Granata   default:
595980c0484SEnrico Granata     return {lang_type};
596980c0484SEnrico Granata   }
597295db41cSDavide Italiano   llvm_unreachable("Fully covered switch");
598980c0484SEnrico Granata }
599980c0484SEnrico Granata 
600980c0484SEnrico Granata LanguageCategory *
601b9c1b51eSKate Stone FormatManager::GetCategoryForLanguage(lldb::LanguageType lang_type) {
60216ff8604SSaleem Abdulrasool   std::lock_guard<std::recursive_mutex> guard(m_language_categories_mutex);
603b9c1b51eSKate Stone   auto iter = m_language_categories_map.find(lang_type),
604b9c1b51eSKate Stone        end = m_language_categories_map.end();
605980c0484SEnrico Granata   if (iter != end)
606980c0484SEnrico Granata     return iter->second.get();
607980c0484SEnrico Granata   LanguageCategory *lang_category = new LanguageCategory(lang_type);
608b9c1b51eSKate Stone   m_language_categories_map[lang_type] =
609b9c1b51eSKate Stone       LanguageCategory::UniquePointer(lang_category);
610980c0484SEnrico Granata   return lang_category;
611980c0484SEnrico Granata }
612980c0484SEnrico Granata 
6137034794bSAdrian Prantl template <typename ImplSP>
6147034794bSAdrian Prantl ImplSP FormatManager::GetHardcoded(FormattersMatchData &match_data) {
6157034794bSAdrian Prantl   ImplSP retval_sp;
616b9c1b51eSKate Stone   for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) {
617b9c1b51eSKate Stone     if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) {
6188a9a8f39SEnrico Granata       if (lang_category->GetHardcoded(*this, match_data, retval_sp))
6197034794bSAdrian Prantl         return retval_sp;
6207034794bSAdrian Prantl     }
6217034794bSAdrian Prantl   }
6227034794bSAdrian Prantl   return retval_sp;
6237034794bSAdrian Prantl }
6247034794bSAdrian Prantl 
62562a6d977SAdrian Prantl template <typename ImplSP>
62662a6d977SAdrian Prantl ImplSP FormatManager::Get(ValueObject &valobj,
6277034794bSAdrian Prantl                           lldb::DynamicValueType use_dynamic) {
6287034794bSAdrian Prantl   FormattersMatchData match_data(valobj, use_dynamic);
62962a6d977SAdrian Prantl   if (ImplSP retval_sp = GetCached<ImplSP>(match_data))
63062a6d977SAdrian Prantl     return retval_sp;
63162a6d977SAdrian Prantl 
63262a6d977SAdrian Prantl   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS));
633*70e3d0eaSAdrian Prantl 
634*70e3d0eaSAdrian Prantl   LLDB_LOGF(log, "[%s] Search failed. Giving language a chance.", __FUNCTION__);
635*70e3d0eaSAdrian Prantl   for (lldb::LanguageType lang_type : match_data.GetCandidateLanguages()) {
636*70e3d0eaSAdrian Prantl     if (LanguageCategory *lang_category = GetCategoryForLanguage(lang_type)) {
637*70e3d0eaSAdrian Prantl       ImplSP retval_sp;
638*70e3d0eaSAdrian Prantl       if (lang_category->Get(match_data, retval_sp))
639*70e3d0eaSAdrian Prantl         if (retval_sp) {
640*70e3d0eaSAdrian Prantl           LLDB_LOGF(log, "[%s] Language search success. Returning.",
641*70e3d0eaSAdrian Prantl                     __FUNCTION__);
642*70e3d0eaSAdrian Prantl           return retval_sp;
643*70e3d0eaSAdrian Prantl         }
644*70e3d0eaSAdrian Prantl     }
645*70e3d0eaSAdrian Prantl   }
646*70e3d0eaSAdrian Prantl 
647*70e3d0eaSAdrian Prantl   LLDB_LOGF(log, "[%s] Search failed. Giving hardcoded a chance.",
648*70e3d0eaSAdrian Prantl             __FUNCTION__);
64962a6d977SAdrian Prantl   return GetHardcoded<ImplSP>(match_data);
65062a6d977SAdrian Prantl }
65162a6d977SAdrian Prantl 
65262a6d977SAdrian Prantl template <typename ImplSP>
65362a6d977SAdrian Prantl ImplSP FormatManager::GetCached(FormattersMatchData &match_data) {
65462a6d977SAdrian Prantl   ImplSP retval_sp;
6557034794bSAdrian Prantl   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_DATAFORMATTERS));
6567034794bSAdrian Prantl   if (match_data.GetTypeForCache()) {
6577034794bSAdrian Prantl     LLDB_LOGF(log, "\n\n[%s] Looking into cache for type %s", __FUNCTION__,
6587034794bSAdrian Prantl               match_data.GetTypeForCache().AsCString("<invalid>"));
6597034794bSAdrian Prantl     if (m_format_cache.Get(match_data.GetTypeForCache(), retval_sp)) {
6607034794bSAdrian Prantl       if (log) {
6617034794bSAdrian Prantl         LLDB_LOGF(log, "[%s] Cache search success. Returning.", __FUNCTION__);
6627034794bSAdrian Prantl         LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}",
6637034794bSAdrian Prantl                   m_format_cache.GetCacheHits(),
6647034794bSAdrian Prantl                   m_format_cache.GetCacheMisses());
6657034794bSAdrian Prantl       }
6667034794bSAdrian Prantl       return retval_sp;
6677034794bSAdrian Prantl     }
6687034794bSAdrian Prantl     LLDB_LOGF(log, "[%s] Cache search failed. Going normal route",
6697034794bSAdrian Prantl               __FUNCTION__);
6707034794bSAdrian Prantl   }
6717034794bSAdrian Prantl 
6727034794bSAdrian Prantl   m_categories_map.Get(match_data, retval_sp);
6737034794bSAdrian Prantl   if (match_data.GetTypeForCache() && (!retval_sp || !retval_sp->NonCacheable())) {
6747034794bSAdrian Prantl     LLDB_LOGF(log, "[%s] Caching %p for type %s", __FUNCTION__,
6757034794bSAdrian Prantl               static_cast<void *>(retval_sp.get()),
6767034794bSAdrian Prantl               match_data.GetTypeForCache().AsCString("<invalid>"));
6777034794bSAdrian Prantl     m_format_cache.Set(match_data.GetTypeForCache(), retval_sp);
6787034794bSAdrian Prantl   }
6797034794bSAdrian Prantl   LLDB_LOGV(log, "Cache hits: {0} - Cache Misses: {1}",
6807034794bSAdrian Prantl             m_format_cache.GetCacheHits(), m_format_cache.GetCacheMisses());
6817cb59e1aSEnrico Granata   return retval_sp;
682686f3debSEnrico Granata }
683686f3debSEnrico Granata 
684852cc954SEnrico Granata lldb::TypeFormatImplSP
685852cc954SEnrico Granata FormatManager::GetFormat(ValueObject &valobj,
686b9c1b51eSKate Stone                          lldb::DynamicValueType use_dynamic) {
68762a6d977SAdrian Prantl   return Get<lldb::TypeFormatImplSP>(valobj, use_dynamic);
688686f3debSEnrico Granata }
689686f3debSEnrico Granata 
6905548cb50SEnrico Granata lldb::TypeSummaryImplSP
6915548cb50SEnrico Granata FormatManager::GetSummaryFormat(ValueObject &valobj,
692b9c1b51eSKate Stone                                 lldb::DynamicValueType use_dynamic) {
69362a6d977SAdrian Prantl   return Get<lldb::TypeSummaryImplSP>(valobj, use_dynamic);
694686f3debSEnrico Granata }
695686f3debSEnrico Granata 
6965548cb50SEnrico Granata lldb::SyntheticChildrenSP
6975548cb50SEnrico Granata FormatManager::GetSyntheticChildren(ValueObject &valobj,
698b9c1b51eSKate Stone                                     lldb::DynamicValueType use_dynamic) {
69962a6d977SAdrian Prantl   return Get<lldb::SyntheticChildrenSP>(valobj, use_dynamic);
7005548cb50SEnrico Granata }
7015548cb50SEnrico Granata 
702c582713cSEnrico Granata lldb::TypeValidatorImplSP
703c582713cSEnrico Granata FormatManager::GetValidator(ValueObject &valobj,
704b9c1b51eSKate Stone                             lldb::DynamicValueType use_dynamic) {
70562a6d977SAdrian Prantl   return Get<lldb::TypeValidatorImplSP>(valobj, use_dynamic);
706c582713cSEnrico Granata }
707c582713cSEnrico Granata 
70816ff8604SSaleem Abdulrasool FormatManager::FormatManager()
709b9c1b51eSKate Stone     : m_last_revision(0), m_format_cache(), m_language_categories_mutex(),
710b9c1b51eSKate Stone       m_language_categories_map(), m_named_summaries_map(this),
711b9c1b51eSKate Stone       m_categories_map(this), m_default_category_name(ConstString("default")),
7125548cb50SEnrico Granata       m_system_category_name(ConstString("system")),
713b9c1b51eSKate Stone       m_vectortypes_category_name(ConstString("VectorTypes")) {
7145548cb50SEnrico Granata   LoadSystemFormatters();
715170c395eSEnrico Granata   LoadVectorFormatters();
7165548cb50SEnrico Granata 
717b9c1b51eSKate Stone   EnableCategory(m_vectortypes_category_name, TypeCategoryMap::Last,
718b9c1b51eSKate Stone                  lldb::eLanguageTypeObjC_plus_plus);
719b9c1b51eSKate Stone   EnableCategory(m_system_category_name, TypeCategoryMap::Last,
720b9c1b51eSKate Stone                  lldb::eLanguageTypeObjC_plus_plus);
7215548cb50SEnrico Granata }
7225548cb50SEnrico Granata 
723b9c1b51eSKate Stone void FormatManager::LoadSystemFormatters() {
7245548cb50SEnrico Granata   TypeSummaryImpl::Flags string_flags;
7250337c27fSEnrico Granata   string_flags.SetCascades(true)
7265548cb50SEnrico Granata       .SetSkipPointers(true)
7275548cb50SEnrico Granata       .SetSkipReferences(false)
7285548cb50SEnrico Granata       .SetDontShowChildren(true)
7295548cb50SEnrico Granata       .SetDontShowValue(false)
7305548cb50SEnrico Granata       .SetShowMembersOneLiner(false)
7315548cb50SEnrico Granata       .SetHideItemNames(false);
7325548cb50SEnrico Granata 
733bc2c2b01SEnrico Granata   TypeSummaryImpl::Flags string_array_flags;
734d2911633SEnrico Granata   string_array_flags.SetCascades(true)
7355548cb50SEnrico Granata       .SetSkipPointers(true)
7365548cb50SEnrico Granata       .SetSkipReferences(false)
737320dcf68SEnrico Granata       .SetDontShowChildren(true)
7385548cb50SEnrico Granata       .SetDontShowValue(true)
7395548cb50SEnrico Granata       .SetShowMembersOneLiner(false)
740bc2c2b01SEnrico Granata       .SetHideItemNames(false);
741bc2c2b01SEnrico Granata 
742b9c1b51eSKate Stone   lldb::TypeSummaryImplSP string_format(
743b9c1b51eSKate Stone       new StringSummaryFormat(string_flags, "${var%s}"));
744bc2c2b01SEnrico Granata 
745b9c1b51eSKate Stone   lldb::TypeSummaryImplSP string_array_format(
746b9c1b51eSKate Stone       new StringSummaryFormat(string_array_flags, "${var%s}"));
747bc2c2b01SEnrico Granata 
7485aa1d819SJan Kratochvil   RegularExpression any_size_char_arr(llvm::StringRef("char \\[[0-9]+\\]"));
7495548cb50SEnrico Granata 
750b9c1b51eSKate Stone   TypeCategoryImpl::SharedPointer sys_category_sp =
751b9c1b51eSKate Stone       GetCategory(m_system_category_name);
7525548cb50SEnrico Granata 
753b9c1b51eSKate Stone   sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("char *"),
754b9c1b51eSKate Stone                                                     string_format);
755b9c1b51eSKate Stone   sys_category_sp->GetTypeSummariesContainer()->Add(
756b9c1b51eSKate Stone       ConstString("unsigned char *"), string_format);
7575aa1d819SJan Kratochvil   sys_category_sp->GetRegexTypeSummariesContainer()->Add(
7585aa1d819SJan Kratochvil       std::move(any_size_char_arr), string_array_format);
7595548cb50SEnrico Granata 
760b9c1b51eSKate Stone   lldb::TypeSummaryImplSP ostype_summary(
761b9c1b51eSKate Stone       new StringSummaryFormat(TypeSummaryImpl::Flags()
762b9c1b51eSKate Stone                                   .SetCascades(false)
7635548cb50SEnrico Granata                                   .SetSkipPointers(true)
7645548cb50SEnrico Granata                                   .SetSkipReferences(true)
7655548cb50SEnrico Granata                                   .SetDontShowChildren(true)
7665548cb50SEnrico Granata                                   .SetDontShowValue(false)
7675548cb50SEnrico Granata                                   .SetShowMembersOneLiner(false)
7685548cb50SEnrico Granata                                   .SetHideItemNames(false),
7695548cb50SEnrico Granata                               "${var%O}"));
7705548cb50SEnrico Granata 
771b9c1b51eSKate Stone   sys_category_sp->GetTypeSummariesContainer()->Add(ConstString("OSType"),
772b9c1b51eSKate Stone                                                     ostype_summary);
7735548cb50SEnrico Granata 
7744cc21772SEnrico Granata   TypeFormatImpl::Flags fourchar_flags;
775b9c1b51eSKate Stone   fourchar_flags.SetCascades(true).SetSkipPointers(true).SetSkipReferences(
776b9c1b51eSKate Stone       true);
7774cc21772SEnrico Granata 
778b9c1b51eSKate Stone   AddFormat(sys_category_sp, lldb::eFormatOSType, ConstString("FourCharCode"),
779b9c1b51eSKate Stone             fourchar_flags);
7805548cb50SEnrico Granata }
7815548cb50SEnrico Granata 
782b9c1b51eSKate Stone void FormatManager::LoadVectorFormatters() {
783b9c1b51eSKate Stone   TypeCategoryImpl::SharedPointer vectors_category_sp =
784b9c1b51eSKate Stone       GetCategory(m_vectortypes_category_name);
7855548cb50SEnrico Granata 
7865548cb50SEnrico Granata   TypeSummaryImpl::Flags vector_flags;
7875548cb50SEnrico Granata   vector_flags.SetCascades(true)
7885548cb50SEnrico Granata       .SetSkipPointers(true)
7895548cb50SEnrico Granata       .SetSkipReferences(false)
7905548cb50SEnrico Granata       .SetDontShowChildren(true)
7915548cb50SEnrico Granata       .SetDontShowValue(false)
7925548cb50SEnrico Granata       .SetShowMembersOneLiner(true)
7935548cb50SEnrico Granata       .SetHideItemNames(true);
7945548cb50SEnrico Granata 
795b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "${var.uint128}",
796b9c1b51eSKate Stone                    ConstString("builtin_type_vec128"), vector_flags);
7975548cb50SEnrico Granata 
798b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("float [4]"),
7995548cb50SEnrico Granata                    vector_flags);
800b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("int32_t [4]"),
8015548cb50SEnrico Granata                    vector_flags);
802b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("int16_t [8]"),
8035548cb50SEnrico Granata                    vector_flags);
804b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vDouble"),
8055548cb50SEnrico Granata                    vector_flags);
806b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vFloat"),
8075548cb50SEnrico Granata                    vector_flags);
808b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt8"),
8095548cb50SEnrico Granata                    vector_flags);
810b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt16"),
8115548cb50SEnrico Granata                    vector_flags);
812b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vSInt32"),
8135548cb50SEnrico Granata                    vector_flags);
814b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"),
8155548cb50SEnrico Granata                    vector_flags);
816b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt8"),
8175548cb50SEnrico Granata                    vector_flags);
818b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt16"),
8195548cb50SEnrico Granata                    vector_flags);
820b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vUInt32"),
8215548cb50SEnrico Granata                    vector_flags);
822b9c1b51eSKate Stone   AddStringSummary(vectors_category_sp, "", ConstString("vBool32"),
8235548cb50SEnrico Granata                    vector_flags);
8245548cb50SEnrico Granata }
825