14d93b8cdSEnrico Granata //===-- ValueObjectPrinter.cpp -------------------------------------*- C++ -*-===//
24d93b8cdSEnrico Granata //
34d93b8cdSEnrico Granata //                     The LLVM Compiler Infrastructure
44d93b8cdSEnrico Granata //
54d93b8cdSEnrico Granata // This file is distributed under the University of Illinois Open Source
64d93b8cdSEnrico Granata // License. See LICENSE.TXT for details.
74d93b8cdSEnrico Granata //
84d93b8cdSEnrico Granata //===----------------------------------------------------------------------===//
94d93b8cdSEnrico Granata 
104d93b8cdSEnrico Granata #include "lldb/DataFormatters/ValueObjectPrinter.h"
114d93b8cdSEnrico Granata 
124d93b8cdSEnrico Granata // C Includes
134d93b8cdSEnrico Granata // C++ Includes
144d93b8cdSEnrico Granata // Other libraries and framework includes
154d93b8cdSEnrico Granata // Project includes
164d93b8cdSEnrico Granata #include "lldb/Core/Debugger.h"
17a29cb0baSEnrico Granata #include "lldb/DataFormatters/DataVisualization.h"
184d93b8cdSEnrico Granata #include "lldb/Interpreter/CommandInterpreter.h"
194d93b8cdSEnrico Granata #include "lldb/Target/Target.h"
204d93b8cdSEnrico Granata 
214d93b8cdSEnrico Granata using namespace lldb;
224d93b8cdSEnrico Granata using namespace lldb_private;
234d93b8cdSEnrico Granata 
244d93b8cdSEnrico Granata ValueObjectPrinter::ValueObjectPrinter (ValueObject* valobj,
254d93b8cdSEnrico Granata                                         Stream* s,
26938d1d67SEnrico Granata                                         const DumpValueObjectOptions& options)
274d93b8cdSEnrico Granata {
28938d1d67SEnrico Granata     Init(valobj,s,options,options.m_max_ptr_depth,0);
294d93b8cdSEnrico Granata }
304d93b8cdSEnrico Granata 
314d93b8cdSEnrico Granata ValueObjectPrinter::ValueObjectPrinter (ValueObject* valobj,
324d93b8cdSEnrico Granata                                         Stream* s,
334d93b8cdSEnrico Granata                                         const DumpValueObjectOptions& options,
344d93b8cdSEnrico Granata                                         uint32_t ptr_depth,
35938d1d67SEnrico Granata                                         uint32_t curr_depth)
36938d1d67SEnrico Granata {
37938d1d67SEnrico Granata     Init(valobj,s,options,ptr_depth,curr_depth);
38938d1d67SEnrico Granata }
39938d1d67SEnrico Granata 
40938d1d67SEnrico Granata void
41938d1d67SEnrico Granata ValueObjectPrinter::Init (ValueObject* valobj,
42938d1d67SEnrico Granata                           Stream* s,
43938d1d67SEnrico Granata                           const DumpValueObjectOptions& options,
44938d1d67SEnrico Granata                           uint32_t ptr_depth,
45938d1d67SEnrico Granata                           uint32_t curr_depth)
46938d1d67SEnrico Granata {
47938d1d67SEnrico Granata     m_orig_valobj = valobj;
48938d1d67SEnrico Granata     m_valobj = nullptr;
49938d1d67SEnrico Granata     m_stream = s;
50938d1d67SEnrico Granata     this->options = options;
51938d1d67SEnrico Granata     m_ptr_depth = ptr_depth;
52938d1d67SEnrico Granata     m_curr_depth = curr_depth;
53938d1d67SEnrico Granata     assert (m_orig_valobj && "cannot print a NULL ValueObject");
54938d1d67SEnrico Granata     assert (m_stream && "cannot print to a NULL Stream");
55938d1d67SEnrico Granata     m_should_print = eLazyBoolCalculate;
56938d1d67SEnrico Granata     m_is_nil = eLazyBoolCalculate;
57938d1d67SEnrico Granata     m_is_ptr = eLazyBoolCalculate;
58938d1d67SEnrico Granata     m_is_ref = eLazyBoolCalculate;
59938d1d67SEnrico Granata     m_is_aggregate = eLazyBoolCalculate;
60938d1d67SEnrico Granata     m_summary_formatter = {nullptr,false};
61938d1d67SEnrico Granata     m_value.assign("");
62938d1d67SEnrico Granata     m_summary.assign("");
63938d1d67SEnrico Granata     m_error.assign("");
64938d1d67SEnrico Granata }
654d93b8cdSEnrico Granata 
664d93b8cdSEnrico Granata bool
674d93b8cdSEnrico Granata ValueObjectPrinter::PrintValueObject ()
684d93b8cdSEnrico Granata {
69*d07cfd3aSEnrico Granata     if (!GetMostSpecializedValue () || m_valobj == nullptr)
704d93b8cdSEnrico Granata         return false;
714d93b8cdSEnrico Granata 
724d93b8cdSEnrico Granata     if (ShouldPrintValueObject())
734d93b8cdSEnrico Granata     {
740f883ffbSEnrico Granata         PrintValidationMarkerIfNeeded();
750f883ffbSEnrico Granata 
764d93b8cdSEnrico Granata         PrintLocationIfNeeded();
774d93b8cdSEnrico Granata         m_stream->Indent();
784d93b8cdSEnrico Granata 
794d93b8cdSEnrico Granata         bool show_type = PrintTypeIfNeeded();
804d93b8cdSEnrico Granata 
814d93b8cdSEnrico Granata         PrintNameIfNeeded(show_type);
824d93b8cdSEnrico Granata     }
834d93b8cdSEnrico Granata 
844d93b8cdSEnrico Granata     bool value_printed = false;
854d93b8cdSEnrico Granata     bool summary_printed = false;
864d93b8cdSEnrico Granata 
874d93b8cdSEnrico Granata     bool val_summary_ok = PrintValueAndSummaryIfNeeded (value_printed,summary_printed);
884d93b8cdSEnrico Granata 
894d93b8cdSEnrico Granata     if (val_summary_ok)
904d93b8cdSEnrico Granata         PrintChildrenIfNeeded (value_printed, summary_printed);
9139938938SEnrico Granata     else
924d93b8cdSEnrico Granata         m_stream->EOL();
934d93b8cdSEnrico Granata 
940f883ffbSEnrico Granata     PrintValidationErrorIfNeeded();
950f883ffbSEnrico Granata 
964d93b8cdSEnrico Granata     return true;
974d93b8cdSEnrico Granata }
984d93b8cdSEnrico Granata 
994d93b8cdSEnrico Granata bool
100*d07cfd3aSEnrico Granata ValueObjectPrinter::GetMostSpecializedValue ()
1014d93b8cdSEnrico Granata {
102a29cb0baSEnrico Granata     if (m_valobj)
103a29cb0baSEnrico Granata         return true;
1044d93b8cdSEnrico Granata     bool update_success = m_orig_valobj->UpdateValueIfNeeded (true);
1054d93b8cdSEnrico Granata     if (!update_success)
106106260c5SEnrico Granata     {
107106260c5SEnrico Granata         m_valobj = m_orig_valobj;
108106260c5SEnrico Granata     }
109106260c5SEnrico Granata     else
110106260c5SEnrico Granata     {
111106260c5SEnrico Granata         if (m_orig_valobj->IsDynamic())
112106260c5SEnrico Granata         {
113106260c5SEnrico Granata             if (options.m_use_dynamic == eNoDynamicValues)
114106260c5SEnrico Granata             {
115106260c5SEnrico Granata                 ValueObject *static_value = m_orig_valobj->GetStaticValue().get();
116106260c5SEnrico Granata                 if (static_value)
117106260c5SEnrico Granata                     m_valobj = static_value;
118106260c5SEnrico Granata                 else
119106260c5SEnrico Granata                     m_valobj = m_orig_valobj;
120106260c5SEnrico Granata             }
121106260c5SEnrico Granata             else
122106260c5SEnrico Granata                 m_valobj = m_orig_valobj;
123106260c5SEnrico Granata         }
124106260c5SEnrico Granata         else
125106260c5SEnrico Granata         {
1264d93b8cdSEnrico Granata             if (options.m_use_dynamic != eNoDynamicValues)
1274d93b8cdSEnrico Granata             {
1284d93b8cdSEnrico Granata                 ValueObject *dynamic_value = m_orig_valobj->GetDynamicValue(options.m_use_dynamic).get();
1294d93b8cdSEnrico Granata                 if (dynamic_value)
1304d93b8cdSEnrico Granata                     m_valobj = dynamic_value;
1314d93b8cdSEnrico Granata                 else
1324d93b8cdSEnrico Granata                     m_valobj = m_orig_valobj;
1334d93b8cdSEnrico Granata             }
1344d93b8cdSEnrico Granata             else
1354d93b8cdSEnrico Granata                 m_valobj = m_orig_valobj;
136106260c5SEnrico Granata         }
137*d07cfd3aSEnrico Granata 
138*d07cfd3aSEnrico Granata         if (m_valobj->IsSynthetic())
139*d07cfd3aSEnrico Granata         {
140*d07cfd3aSEnrico Granata             if (options.m_use_synthetic == false)
141*d07cfd3aSEnrico Granata             {
142*d07cfd3aSEnrico Granata                 ValueObject *non_synthetic = m_valobj->GetNonSyntheticValue().get();
143*d07cfd3aSEnrico Granata                 if (non_synthetic)
144*d07cfd3aSEnrico Granata                     m_valobj = non_synthetic;
145*d07cfd3aSEnrico Granata             }
146*d07cfd3aSEnrico Granata         }
147*d07cfd3aSEnrico Granata         else
148*d07cfd3aSEnrico Granata         {
149*d07cfd3aSEnrico Granata             if (options.m_use_synthetic == true)
150*d07cfd3aSEnrico Granata             {
151*d07cfd3aSEnrico Granata                 ValueObject *synthetic = m_valobj->GetSyntheticValue().get();
152*d07cfd3aSEnrico Granata                 if (synthetic)
153*d07cfd3aSEnrico Granata                     m_valobj = synthetic;
154*d07cfd3aSEnrico Granata             }
155*d07cfd3aSEnrico Granata         }
156106260c5SEnrico Granata     }
1574d93b8cdSEnrico Granata     m_clang_type = m_valobj->GetClangType();
1584d93b8cdSEnrico Granata     m_type_flags = m_clang_type.GetTypeInfo ();
1594d93b8cdSEnrico Granata     return true;
1604d93b8cdSEnrico Granata }
1614d93b8cdSEnrico Granata 
1624d93b8cdSEnrico Granata const char*
1634d93b8cdSEnrico Granata ValueObjectPrinter::GetDescriptionForDisplay ()
1644d93b8cdSEnrico Granata {
1654d93b8cdSEnrico Granata     const char* str = m_valobj->GetObjectDescription();
1664d93b8cdSEnrico Granata     if (!str)
1674d93b8cdSEnrico Granata         str = m_valobj->GetSummaryAsCString();
1684d93b8cdSEnrico Granata     if (!str)
1694d93b8cdSEnrico Granata         str = m_valobj->GetValueAsCString();
1704d93b8cdSEnrico Granata     return str;
1714d93b8cdSEnrico Granata }
1724d93b8cdSEnrico Granata 
1734d93b8cdSEnrico Granata const char*
1744d93b8cdSEnrico Granata ValueObjectPrinter::GetRootNameForDisplay (const char* if_fail)
1754d93b8cdSEnrico Granata {
1764d93b8cdSEnrico Granata     const char *root_valobj_name = options.m_root_valobj_name.empty() ?
1774d93b8cdSEnrico Granata         m_valobj->GetName().AsCString() :
1784d93b8cdSEnrico Granata         options.m_root_valobj_name.c_str();
1794d93b8cdSEnrico Granata     return root_valobj_name ? root_valobj_name : if_fail;
1804d93b8cdSEnrico Granata }
1814d93b8cdSEnrico Granata 
1824d93b8cdSEnrico Granata bool
1834d93b8cdSEnrico Granata ValueObjectPrinter::ShouldPrintValueObject ()
1844d93b8cdSEnrico Granata {
1854d93b8cdSEnrico Granata     if (m_should_print == eLazyBoolCalculate)
1864d93b8cdSEnrico Granata         m_should_print = (options.m_flat_output == false || m_type_flags.Test (ClangASTType::eTypeHasValue)) ? eLazyBoolYes : eLazyBoolNo;
1874d93b8cdSEnrico Granata     return m_should_print == eLazyBoolYes;
1884d93b8cdSEnrico Granata }
1894d93b8cdSEnrico Granata 
1904d93b8cdSEnrico Granata bool
1914d93b8cdSEnrico Granata ValueObjectPrinter::IsNil ()
1924d93b8cdSEnrico Granata {
1934d93b8cdSEnrico Granata     if (m_is_nil == eLazyBoolCalculate)
1944d93b8cdSEnrico Granata         m_is_nil = m_valobj->IsObjCNil() ? eLazyBoolYes : eLazyBoolNo;
1954d93b8cdSEnrico Granata     return m_is_nil == eLazyBoolYes;
1964d93b8cdSEnrico Granata }
1974d93b8cdSEnrico Granata 
1984d93b8cdSEnrico Granata bool
1994d93b8cdSEnrico Granata ValueObjectPrinter::IsPtr ()
2004d93b8cdSEnrico Granata {
2014d93b8cdSEnrico Granata     if (m_is_ptr == eLazyBoolCalculate)
2024d93b8cdSEnrico Granata         m_is_ptr = m_type_flags.Test (ClangASTType::eTypeIsPointer) ? eLazyBoolYes : eLazyBoolNo;
2034d93b8cdSEnrico Granata     return m_is_ptr == eLazyBoolYes;
2044d93b8cdSEnrico Granata }
2054d93b8cdSEnrico Granata 
2064d93b8cdSEnrico Granata bool
2074d93b8cdSEnrico Granata ValueObjectPrinter::IsRef ()
2084d93b8cdSEnrico Granata {
2094d93b8cdSEnrico Granata     if (m_is_ref == eLazyBoolCalculate)
2104d93b8cdSEnrico Granata         m_is_ref = m_type_flags.Test (ClangASTType::eTypeIsReference) ? eLazyBoolYes : eLazyBoolNo;
2114d93b8cdSEnrico Granata     return m_is_ref == eLazyBoolYes;
2124d93b8cdSEnrico Granata }
2134d93b8cdSEnrico Granata 
2144d93b8cdSEnrico Granata bool
2154d93b8cdSEnrico Granata ValueObjectPrinter::IsAggregate ()
2164d93b8cdSEnrico Granata {
2174d93b8cdSEnrico Granata     if (m_is_aggregate == eLazyBoolCalculate)
2184d93b8cdSEnrico Granata         m_is_aggregate = m_type_flags.Test (ClangASTType::eTypeHasChildren) ? eLazyBoolYes : eLazyBoolNo;
2194d93b8cdSEnrico Granata     return m_is_aggregate == eLazyBoolYes;
2204d93b8cdSEnrico Granata }
2214d93b8cdSEnrico Granata 
2224d93b8cdSEnrico Granata bool
2234d93b8cdSEnrico Granata ValueObjectPrinter::PrintLocationIfNeeded ()
2244d93b8cdSEnrico Granata {
2254d93b8cdSEnrico Granata     if (options.m_show_location)
2264d93b8cdSEnrico Granata     {
2274d93b8cdSEnrico Granata         m_stream->Printf("%s: ", m_valobj->GetLocationAsCString());
2284d93b8cdSEnrico Granata         return true;
2294d93b8cdSEnrico Granata     }
2304d93b8cdSEnrico Granata     return false;
2314d93b8cdSEnrico Granata }
2324d93b8cdSEnrico Granata 
2334d93b8cdSEnrico Granata bool
2344d93b8cdSEnrico Granata ValueObjectPrinter::PrintTypeIfNeeded ()
2354d93b8cdSEnrico Granata {
2364d93b8cdSEnrico Granata     bool show_type = true;
2374d93b8cdSEnrico Granata     // if we are at the root-level and been asked to hide the root's type, then hide it
2384d93b8cdSEnrico Granata     if (m_curr_depth == 0 && options.m_hide_root_type)
2394d93b8cdSEnrico Granata         show_type = false;
2404d93b8cdSEnrico Granata     else
2414d93b8cdSEnrico Granata         // otherwise decide according to the usual rules (asked to show types - always at the root level)
2424d93b8cdSEnrico Granata         show_type = options.m_show_types || (m_curr_depth == 0 && !options.m_flat_output);
2434d93b8cdSEnrico Granata 
2444d93b8cdSEnrico Granata     if (show_type)
2454d93b8cdSEnrico Granata     {
2464d93b8cdSEnrico Granata         // Some ValueObjects don't have types (like registers sets). Only print
2474d93b8cdSEnrico Granata         // the type if there is one to print
248e8daa2f8SEnrico Granata         ConstString qualified_type_name;
249e8daa2f8SEnrico Granata         if (options.m_be_raw)
250e8daa2f8SEnrico Granata             qualified_type_name = m_valobj->GetQualifiedTypeName();
251e8daa2f8SEnrico Granata         else
252e8daa2f8SEnrico Granata             qualified_type_name = m_valobj->GetDisplayTypeName();
2534d93b8cdSEnrico Granata         if (qualified_type_name)
2544d93b8cdSEnrico Granata             m_stream->Printf("(%s) ", qualified_type_name.GetCString());
2554d93b8cdSEnrico Granata         else
2564d93b8cdSEnrico Granata             show_type = false;
2574d93b8cdSEnrico Granata     }
2584d93b8cdSEnrico Granata     return show_type;
2594d93b8cdSEnrico Granata }
2604d93b8cdSEnrico Granata 
2614d93b8cdSEnrico Granata bool
2624d93b8cdSEnrico Granata ValueObjectPrinter::PrintNameIfNeeded (bool show_type)
2634d93b8cdSEnrico Granata {
2644d93b8cdSEnrico Granata     if (options.m_flat_output)
2654d93b8cdSEnrico Granata     {
2664d93b8cdSEnrico Granata         // If we are showing types, also qualify the C++ base classes
2674d93b8cdSEnrico Granata         const bool qualify_cxx_base_classes = show_type;
2684d93b8cdSEnrico Granata         if (!options.m_hide_name)
2694d93b8cdSEnrico Granata         {
2704d93b8cdSEnrico Granata             m_valobj->GetExpressionPath(*m_stream, qualify_cxx_base_classes);
2714d93b8cdSEnrico Granata             m_stream->PutCString(" =");
2724d93b8cdSEnrico Granata             return true;
2734d93b8cdSEnrico Granata         }
2744d93b8cdSEnrico Granata     }
2754d93b8cdSEnrico Granata     else if (!options.m_hide_name)
2764d93b8cdSEnrico Granata     {
2774d93b8cdSEnrico Granata         const char *name_cstr = GetRootNameForDisplay("");
2784d93b8cdSEnrico Granata         m_stream->Printf ("%s =", name_cstr);
2794d93b8cdSEnrico Granata         return true;
2804d93b8cdSEnrico Granata     }
2814d93b8cdSEnrico Granata     return false;
2824d93b8cdSEnrico Granata }
2834d93b8cdSEnrico Granata 
2844d93b8cdSEnrico Granata bool
2854d93b8cdSEnrico Granata ValueObjectPrinter::CheckScopeIfNeeded ()
2864d93b8cdSEnrico Granata {
2874d93b8cdSEnrico Granata     if (options.m_scope_already_checked)
2884d93b8cdSEnrico Granata         return true;
2894d93b8cdSEnrico Granata     return m_valobj->IsInScope();
2904d93b8cdSEnrico Granata }
2914d93b8cdSEnrico Granata 
2924d93b8cdSEnrico Granata TypeSummaryImpl*
2934d93b8cdSEnrico Granata ValueObjectPrinter::GetSummaryFormatter ()
2944d93b8cdSEnrico Granata {
2954d93b8cdSEnrico Granata     if (m_summary_formatter.second == false)
2964d93b8cdSEnrico Granata     {
2974d93b8cdSEnrico Granata         TypeSummaryImpl* entry = options.m_summary_sp ? options.m_summary_sp.get() : m_valobj->GetSummaryFormat().get();
2984d93b8cdSEnrico Granata 
2994d93b8cdSEnrico Granata         if (options.m_omit_summary_depth > 0)
3004d93b8cdSEnrico Granata             entry = NULL;
3014d93b8cdSEnrico Granata         m_summary_formatter.first = entry;
3024d93b8cdSEnrico Granata         m_summary_formatter.second = true;
3034d93b8cdSEnrico Granata     }
3044d93b8cdSEnrico Granata     return m_summary_formatter.first;
3054d93b8cdSEnrico Granata }
3064d93b8cdSEnrico Granata 
3074d93b8cdSEnrico Granata void
3084d93b8cdSEnrico Granata ValueObjectPrinter::GetValueSummaryError (std::string& value,
3094d93b8cdSEnrico Granata                                           std::string& summary,
3104d93b8cdSEnrico Granata                                           std::string& error)
3114d93b8cdSEnrico Granata {
312465f4bc2SEnrico Granata     if (options.m_format != eFormatDefault && options.m_format != m_valobj->GetFormat())
3134d93b8cdSEnrico Granata     {
314465f4bc2SEnrico Granata         m_valobj->GetValueAsCString(options.m_format,
315465f4bc2SEnrico Granata                                     value);
3164d93b8cdSEnrico Granata     }
317465f4bc2SEnrico Granata     else
318465f4bc2SEnrico Granata     {
3194d93b8cdSEnrico Granata         const char* val_cstr = m_valobj->GetValueAsCString();
320465f4bc2SEnrico Granata         if (val_cstr)
321465f4bc2SEnrico Granata             value.assign(val_cstr);
322465f4bc2SEnrico Granata     }
3234d93b8cdSEnrico Granata     const char* err_cstr = m_valobj->GetError().AsCString();
3244d93b8cdSEnrico Granata     if (err_cstr)
3254d93b8cdSEnrico Granata         error.assign(err_cstr);
3264d93b8cdSEnrico Granata 
3274d93b8cdSEnrico Granata     if (ShouldPrintValueObject())
3284d93b8cdSEnrico Granata     {
3294d93b8cdSEnrico Granata         if (IsNil())
3304d93b8cdSEnrico Granata             summary.assign("nil");
3314d93b8cdSEnrico Granata         else if (options.m_omit_summary_depth == 0)
3324d93b8cdSEnrico Granata         {
3334d93b8cdSEnrico Granata             TypeSummaryImpl* entry = GetSummaryFormatter();
3344d93b8cdSEnrico Granata             if (entry)
3354d93b8cdSEnrico Granata                 m_valobj->GetSummaryAsCString(entry, summary);
3364d93b8cdSEnrico Granata             else
3374d93b8cdSEnrico Granata             {
3384d93b8cdSEnrico Granata                 const char* sum_cstr = m_valobj->GetSummaryAsCString();
3394d93b8cdSEnrico Granata                 if (sum_cstr)
3404d93b8cdSEnrico Granata                     summary.assign(sum_cstr);
3414d93b8cdSEnrico Granata             }
3424d93b8cdSEnrico Granata         }
3434d93b8cdSEnrico Granata     }
3444d93b8cdSEnrico Granata }
3454d93b8cdSEnrico Granata 
3464d93b8cdSEnrico Granata bool
3474d93b8cdSEnrico Granata ValueObjectPrinter::PrintValueAndSummaryIfNeeded (bool& value_printed,
3484d93b8cdSEnrico Granata                                                   bool& summary_printed)
3494d93b8cdSEnrico Granata {
3504d93b8cdSEnrico Granata     bool error_printed = false;
3514d93b8cdSEnrico Granata     if (ShouldPrintValueObject())
3524d93b8cdSEnrico Granata     {
3534d93b8cdSEnrico Granata         if (!CheckScopeIfNeeded())
3544d93b8cdSEnrico Granata             m_error.assign("out of scope");
3554d93b8cdSEnrico Granata         if (m_error.empty())
3564d93b8cdSEnrico Granata         {
3574d93b8cdSEnrico Granata             GetValueSummaryError(m_value, m_summary, m_error);
3584d93b8cdSEnrico Granata         }
3594d93b8cdSEnrico Granata         if (m_error.size())
3604d93b8cdSEnrico Granata         {
3614d93b8cdSEnrico Granata             error_printed = true;
3624d93b8cdSEnrico Granata             m_stream->Printf (" <%s>\n", m_error.c_str());
3634d93b8cdSEnrico Granata         }
3644d93b8cdSEnrico Granata         else
3654d93b8cdSEnrico Granata         {
3664d93b8cdSEnrico Granata             // Make sure we have a value and make sure the summary didn't
3674d93b8cdSEnrico Granata             // specify that the value should not be printed - and do not print
3684d93b8cdSEnrico Granata             // the value if this thing is nil
3694d93b8cdSEnrico Granata             // (but show the value if the user passes a format explicitly)
3704d93b8cdSEnrico Granata             TypeSummaryImpl* entry = GetSummaryFormatter();
3718a068e6cSEnrico Granata             if (!IsNil() && !m_value.empty() && (entry == NULL || (entry->DoesPrintValue(m_valobj) || options.m_format != eFormatDefault) || m_summary.empty()) && !options.m_hide_value)
3724d93b8cdSEnrico Granata             {
3734d93b8cdSEnrico Granata                 m_stream->Printf(" %s", m_value.c_str());
3744d93b8cdSEnrico Granata                 value_printed = true;
3754d93b8cdSEnrico Granata             }
3764d93b8cdSEnrico Granata 
3774d93b8cdSEnrico Granata             if (m_summary.size())
3784d93b8cdSEnrico Granata             {
3794d93b8cdSEnrico Granata                 m_stream->Printf(" %s", m_summary.c_str());
3804d93b8cdSEnrico Granata                 summary_printed = true;
3814d93b8cdSEnrico Granata             }
3824d93b8cdSEnrico Granata         }
3834d93b8cdSEnrico Granata     }
3844d93b8cdSEnrico Granata     return !error_printed;
3854d93b8cdSEnrico Granata }
3864d93b8cdSEnrico Granata 
3874d93b8cdSEnrico Granata bool
3884d93b8cdSEnrico Granata ValueObjectPrinter::PrintObjectDescriptionIfNeeded (bool value_printed,
3894d93b8cdSEnrico Granata                                                     bool summary_printed)
3904d93b8cdSEnrico Granata {
3914d93b8cdSEnrico Granata     if (ShouldPrintValueObject())
3924d93b8cdSEnrico Granata     {
3934d93b8cdSEnrico Granata         // let's avoid the overly verbose no description error for a nil thing
3944d93b8cdSEnrico Granata         if (options.m_use_objc && !IsNil())
3954d93b8cdSEnrico Granata         {
3964d93b8cdSEnrico Granata             if (!options.m_hide_value || !options.m_hide_name)
3974d93b8cdSEnrico Granata                 m_stream->Printf(" ");
3984d93b8cdSEnrico Granata             const char *object_desc = nullptr;
3994d93b8cdSEnrico Granata             if (value_printed || summary_printed)
4004d93b8cdSEnrico Granata                 object_desc = m_valobj->GetObjectDescription();
4014d93b8cdSEnrico Granata             else
4024d93b8cdSEnrico Granata                 object_desc = GetDescriptionForDisplay();
4034d93b8cdSEnrico Granata             if (object_desc && *object_desc)
4044d93b8cdSEnrico Granata             {
4054d93b8cdSEnrico Granata                 m_stream->Printf("%s\n", object_desc);
4064d93b8cdSEnrico Granata                 return true;
4074d93b8cdSEnrico Granata             }
4084d93b8cdSEnrico Granata             else if (value_printed == false && summary_printed == false)
4094d93b8cdSEnrico Granata                 return true;
4104d93b8cdSEnrico Granata             else
4114d93b8cdSEnrico Granata                 return false;
4124d93b8cdSEnrico Granata         }
4134d93b8cdSEnrico Granata     }
4144d93b8cdSEnrico Granata     return true;
4154d93b8cdSEnrico Granata }
4164d93b8cdSEnrico Granata 
4174d93b8cdSEnrico Granata bool
4184d93b8cdSEnrico Granata ValueObjectPrinter::ShouldPrintChildren (bool is_failed_description,
4194d93b8cdSEnrico Granata                                          uint32_t& curr_ptr_depth)
4204d93b8cdSEnrico Granata {
4214d93b8cdSEnrico Granata     const bool is_ref = IsRef ();
4224d93b8cdSEnrico Granata     const bool is_ptr = IsPtr ();
4234d93b8cdSEnrico Granata 
4244d93b8cdSEnrico Granata     if (is_failed_description || m_curr_depth < options.m_max_depth)
4254d93b8cdSEnrico Granata     {
4264d93b8cdSEnrico Granata         // We will show children for all concrete types. We won't show
4274d93b8cdSEnrico Granata         // pointer contents unless a pointer depth has been specified.
4284d93b8cdSEnrico Granata         // We won't reference contents unless the reference is the
4294d93b8cdSEnrico Granata         // root object (depth of zero).
4304d93b8cdSEnrico Granata 
4314d93b8cdSEnrico Granata         // Use a new temporary pointer depth in case we override the
4324d93b8cdSEnrico Granata         // current pointer depth below...
4334d93b8cdSEnrico Granata 
4344d93b8cdSEnrico Granata         if (is_ptr || is_ref)
4354d93b8cdSEnrico Granata         {
4364d93b8cdSEnrico Granata             // We have a pointer or reference whose value is an address.
4374d93b8cdSEnrico Granata             // Make sure that address is not NULL
4384d93b8cdSEnrico Granata             AddressType ptr_address_type;
4394d93b8cdSEnrico Granata             if (m_valobj->GetPointerValue (&ptr_address_type) == 0)
4404d93b8cdSEnrico Granata                 return false;
4414d93b8cdSEnrico Granata 
4425000ee16SEnrico Granata             else if (is_ref && m_curr_depth == 0 && curr_ptr_depth == 0)
4434d93b8cdSEnrico Granata             {
4444d93b8cdSEnrico Granata                 // If this is the root object (depth is zero) that we are showing
4454d93b8cdSEnrico Granata                 // and it is a reference, and no pointer depth has been supplied
4464d93b8cdSEnrico Granata                 // print out what it references. Don't do this at deeper depths
4474d93b8cdSEnrico Granata                 // otherwise we can end up with infinite recursion...
4484d93b8cdSEnrico Granata                 curr_ptr_depth = 1;
4494d93b8cdSEnrico Granata             }
4504d93b8cdSEnrico Granata 
451fcf0c4e3SEnrico Granata             return (curr_ptr_depth > 0);
4524d93b8cdSEnrico Granata         }
4534d93b8cdSEnrico Granata 
4544d93b8cdSEnrico Granata         TypeSummaryImpl* entry = GetSummaryFormatter();
4554d93b8cdSEnrico Granata 
4568a068e6cSEnrico Granata         return (!entry || entry->DoesPrintChildren(m_valobj) || m_summary.empty());
4574d93b8cdSEnrico Granata     }
4584d93b8cdSEnrico Granata     return false;
4594d93b8cdSEnrico Granata }
4604d93b8cdSEnrico Granata 
4614d93b8cdSEnrico Granata ValueObject*
4624d93b8cdSEnrico Granata ValueObjectPrinter::GetValueObjectForChildrenGeneration ()
4634d93b8cdSEnrico Granata {
464*d07cfd3aSEnrico Granata     return m_valobj;
4654d93b8cdSEnrico Granata }
4664d93b8cdSEnrico Granata 
4674d93b8cdSEnrico Granata void
4684d93b8cdSEnrico Granata ValueObjectPrinter::PrintChildrenPreamble ()
4694d93b8cdSEnrico Granata {
4704d93b8cdSEnrico Granata     if (options.m_flat_output)
4714d93b8cdSEnrico Granata     {
4724d93b8cdSEnrico Granata         if (ShouldPrintValueObject())
4734d93b8cdSEnrico Granata             m_stream->EOL();
4744d93b8cdSEnrico Granata     }
4754d93b8cdSEnrico Granata     else
4764d93b8cdSEnrico Granata     {
4774d93b8cdSEnrico Granata         if (ShouldPrintValueObject())
4784d93b8cdSEnrico Granata             m_stream->PutCString(IsRef () ? ": {\n" : " {\n");
4794d93b8cdSEnrico Granata         m_stream->IndentMore();
4804d93b8cdSEnrico Granata     }
4814d93b8cdSEnrico Granata }
4824d93b8cdSEnrico Granata 
4834d93b8cdSEnrico Granata void
4844d93b8cdSEnrico Granata ValueObjectPrinter::PrintChild (ValueObjectSP child_sp,
4854d93b8cdSEnrico Granata                                 uint32_t curr_ptr_depth)
4864d93b8cdSEnrico Granata {
4874d93b8cdSEnrico Granata     DumpValueObjectOptions child_options(options);
4884d93b8cdSEnrico Granata     child_options.SetFormat(options.m_format).SetSummary().SetRootValueObjectName();
4894d93b8cdSEnrico Granata     child_options.SetScopeChecked(true).SetHideName(options.m_hide_name).SetHideValue(options.m_hide_value)
4904d93b8cdSEnrico Granata     .SetOmitSummaryDepth(child_options.m_omit_summary_depth > 1 ? child_options.m_omit_summary_depth - 1 : 0);
4914d93b8cdSEnrico Granata     if (child_sp.get())
4924d93b8cdSEnrico Granata     {
4934d93b8cdSEnrico Granata         ValueObjectPrinter child_printer(child_sp.get(),
4944d93b8cdSEnrico Granata                                          m_stream,
4954d93b8cdSEnrico Granata                                          child_options,
4965000ee16SEnrico Granata                                          (IsPtr() || IsRef()) && curr_ptr_depth >= 1 ? curr_ptr_depth - 1 : curr_ptr_depth,
4974d93b8cdSEnrico Granata                                          m_curr_depth + 1);
4984d93b8cdSEnrico Granata         child_printer.PrintValueObject();
4994d93b8cdSEnrico Granata     }
5004d93b8cdSEnrico Granata 
5014d93b8cdSEnrico Granata }
5024d93b8cdSEnrico Granata 
5034d93b8cdSEnrico Granata uint32_t
5044d93b8cdSEnrico Granata ValueObjectPrinter::GetMaxNumChildrenToPrint (bool& print_dotdotdot)
5054d93b8cdSEnrico Granata {
5064d93b8cdSEnrico Granata     ValueObject* synth_m_valobj = GetValueObjectForChildrenGeneration();
5074d93b8cdSEnrico Granata 
5084d93b8cdSEnrico Granata     size_t num_children = synth_m_valobj->GetNumChildren();
5094d93b8cdSEnrico Granata     print_dotdotdot = false;
5104d93b8cdSEnrico Granata     if (num_children)
5114d93b8cdSEnrico Granata     {
5124d93b8cdSEnrico Granata         const size_t max_num_children = m_valobj->GetTargetSP()->GetMaximumNumberOfChildrenToDisplay();
5134d93b8cdSEnrico Granata 
5144d93b8cdSEnrico Granata         if (num_children > max_num_children && !options.m_ignore_cap)
5154d93b8cdSEnrico Granata         {
5164d93b8cdSEnrico Granata             print_dotdotdot = true;
5174d93b8cdSEnrico Granata             return max_num_children;
5184d93b8cdSEnrico Granata         }
5194d93b8cdSEnrico Granata     }
5204d93b8cdSEnrico Granata     return num_children;
5214d93b8cdSEnrico Granata }
5224d93b8cdSEnrico Granata 
5234d93b8cdSEnrico Granata void
5244d93b8cdSEnrico Granata ValueObjectPrinter::PrintChildrenPostamble (bool print_dotdotdot)
5254d93b8cdSEnrico Granata {
5264d93b8cdSEnrico Granata     if (!options.m_flat_output)
5274d93b8cdSEnrico Granata     {
5284d93b8cdSEnrico Granata         if (print_dotdotdot)
5294d93b8cdSEnrico Granata         {
5304d93b8cdSEnrico Granata             m_valobj->GetTargetSP()->GetDebugger().GetCommandInterpreter().ChildrenTruncated();
5314d93b8cdSEnrico Granata             m_stream->Indent("...\n");
5324d93b8cdSEnrico Granata         }
5334d93b8cdSEnrico Granata         m_stream->IndentLess();
5344d93b8cdSEnrico Granata         m_stream->Indent("}\n");
5354d93b8cdSEnrico Granata     }
5364d93b8cdSEnrico Granata }
5374d93b8cdSEnrico Granata 
5384d93b8cdSEnrico Granata void
5394d93b8cdSEnrico Granata ValueObjectPrinter::PrintChildren (uint32_t curr_ptr_depth)
5404d93b8cdSEnrico Granata {
5414d93b8cdSEnrico Granata     ValueObject* synth_m_valobj = GetValueObjectForChildrenGeneration();
5424d93b8cdSEnrico Granata 
5434d93b8cdSEnrico Granata     bool print_dotdotdot = false;
5444d93b8cdSEnrico Granata     size_t num_children = GetMaxNumChildrenToPrint(print_dotdotdot);
5454d93b8cdSEnrico Granata     if (num_children)
5464d93b8cdSEnrico Granata     {
5474d93b8cdSEnrico Granata         PrintChildrenPreamble ();
5484d93b8cdSEnrico Granata 
5494d93b8cdSEnrico Granata         for (size_t idx=0; idx<num_children; ++idx)
5504d93b8cdSEnrico Granata         {
5514d93b8cdSEnrico Granata             ValueObjectSP child_sp(synth_m_valobj->GetChildAtIndex(idx, true));
5524d93b8cdSEnrico Granata             PrintChild (child_sp, curr_ptr_depth);
5534d93b8cdSEnrico Granata         }
5544d93b8cdSEnrico Granata 
5554d93b8cdSEnrico Granata         PrintChildrenPostamble (print_dotdotdot);
5564d93b8cdSEnrico Granata     }
5574d93b8cdSEnrico Granata     else if (IsAggregate())
5584d93b8cdSEnrico Granata     {
5594d93b8cdSEnrico Granata         // Aggregate, no children...
5604d93b8cdSEnrico Granata         if (ShouldPrintValueObject())
561*d07cfd3aSEnrico Granata         {
562*d07cfd3aSEnrico Granata             // if it has a synthetic value, then don't print {}, the synthetic children are probably only being used to vend a value
563*d07cfd3aSEnrico Granata             if (m_valobj->DoesProvideSyntheticValue())
564*d07cfd3aSEnrico Granata                 m_stream->PutCString( "\n");
565*d07cfd3aSEnrico Granata             else
5664d93b8cdSEnrico Granata                 m_stream->PutCString(" {}\n");
5674d93b8cdSEnrico Granata         }
568*d07cfd3aSEnrico Granata     }
5694d93b8cdSEnrico Granata     else
5704d93b8cdSEnrico Granata     {
5714d93b8cdSEnrico Granata         if (ShouldPrintValueObject())
5724d93b8cdSEnrico Granata             m_stream->EOL();
5734d93b8cdSEnrico Granata     }
5744d93b8cdSEnrico Granata }
5754d93b8cdSEnrico Granata 
576a29cb0baSEnrico Granata bool
577a29cb0baSEnrico Granata ValueObjectPrinter::PrintChildrenOneLiner (bool hide_names)
578a29cb0baSEnrico Granata {
579*d07cfd3aSEnrico Granata     if (!GetMostSpecializedValue () || m_valobj == nullptr)
580a29cb0baSEnrico Granata         return false;
581a29cb0baSEnrico Granata 
582a29cb0baSEnrico Granata     ValueObject* synth_m_valobj = GetValueObjectForChildrenGeneration();
583a29cb0baSEnrico Granata 
584a29cb0baSEnrico Granata     bool print_dotdotdot = false;
585a29cb0baSEnrico Granata     size_t num_children = GetMaxNumChildrenToPrint(print_dotdotdot);
586a29cb0baSEnrico Granata 
587a29cb0baSEnrico Granata     if (num_children)
588a29cb0baSEnrico Granata     {
589a29cb0baSEnrico Granata         m_stream->PutChar('(');
590a29cb0baSEnrico Granata 
591a29cb0baSEnrico Granata         for (uint32_t idx=0; idx<num_children; ++idx)
592a29cb0baSEnrico Granata         {
593a29cb0baSEnrico Granata             lldb::ValueObjectSP child_sp(synth_m_valobj->GetChildAtIndex(idx, true));
594a29cb0baSEnrico Granata             lldb::ValueObjectSP child_dyn_sp = child_sp.get() ? child_sp->GetDynamicValue(options.m_use_dynamic) : child_sp;
595a29cb0baSEnrico Granata             if (child_dyn_sp)
596a29cb0baSEnrico Granata                 child_sp = child_dyn_sp;
597a29cb0baSEnrico Granata             if (child_sp)
598a29cb0baSEnrico Granata             {
599a29cb0baSEnrico Granata                 if (idx)
600a29cb0baSEnrico Granata                     m_stream->PutCString(", ");
601a29cb0baSEnrico Granata                 if (!hide_names)
602a29cb0baSEnrico Granata                 {
603a29cb0baSEnrico Granata                     const char* name = child_sp.get()->GetName().AsCString();
604a29cb0baSEnrico Granata                     if (name && *name)
605a29cb0baSEnrico Granata                     {
606a29cb0baSEnrico Granata                         m_stream->PutCString(name);
607a29cb0baSEnrico Granata                         m_stream->PutCString(" = ");
608a29cb0baSEnrico Granata                     }
609a29cb0baSEnrico Granata                 }
610a29cb0baSEnrico Granata                 child_sp->DumpPrintableRepresentation(*m_stream,
611a29cb0baSEnrico Granata                                                       ValueObject::eValueObjectRepresentationStyleSummary,
612a29cb0baSEnrico Granata                                                       lldb::eFormatInvalid,
613a29cb0baSEnrico Granata                                                       ValueObject::ePrintableRepresentationSpecialCasesDisable);
614a29cb0baSEnrico Granata             }
615a29cb0baSEnrico Granata         }
616a29cb0baSEnrico Granata 
617a29cb0baSEnrico Granata         if (print_dotdotdot)
618a29cb0baSEnrico Granata             m_stream->PutCString(", ...)");
619a29cb0baSEnrico Granata         else
620a29cb0baSEnrico Granata             m_stream->PutChar(')');
621a29cb0baSEnrico Granata     }
622a29cb0baSEnrico Granata     return true;
623a29cb0baSEnrico Granata }
624a29cb0baSEnrico Granata 
6254d93b8cdSEnrico Granata void
6264d93b8cdSEnrico Granata ValueObjectPrinter::PrintChildrenIfNeeded (bool value_printed,
6274d93b8cdSEnrico Granata                                            bool summary_printed)
6284d93b8cdSEnrico Granata {
6294d93b8cdSEnrico Granata     // this flag controls whether we tried to display a description for this object and failed
6304d93b8cdSEnrico Granata     // if that happens, we want to display the children, if any
6314d93b8cdSEnrico Granata     bool is_failed_description = !PrintObjectDescriptionIfNeeded(value_printed, summary_printed);
6324d93b8cdSEnrico Granata 
6334d93b8cdSEnrico Granata     uint32_t curr_ptr_depth = m_ptr_depth;
6344d93b8cdSEnrico Granata     bool print_children = ShouldPrintChildren (is_failed_description,curr_ptr_depth);
63541b16533SEnrico Granata     bool print_oneline = (curr_ptr_depth > 0 || options.m_show_types || options.m_be_raw) ? false : DataVisualization::ShouldPrintAsOneLiner(*m_valobj);
6364d93b8cdSEnrico Granata 
6374d93b8cdSEnrico Granata     if (print_children)
6384d93b8cdSEnrico Granata     {
639a29cb0baSEnrico Granata         if (print_oneline)
640a29cb0baSEnrico Granata         {
641a29cb0baSEnrico Granata             m_stream->PutChar(' ');
642a29cb0baSEnrico Granata             PrintChildrenOneLiner (false);
643a29cb0baSEnrico Granata             m_stream->EOL();
644a29cb0baSEnrico Granata         }
645a29cb0baSEnrico Granata         else
6464d93b8cdSEnrico Granata             PrintChildren (curr_ptr_depth);
6474d93b8cdSEnrico Granata     }
6484d93b8cdSEnrico Granata     else if (m_curr_depth >= options.m_max_depth && IsAggregate() && ShouldPrintValueObject())
6494d93b8cdSEnrico Granata     {
6504d93b8cdSEnrico Granata             m_stream->PutCString("{...}\n");
6514d93b8cdSEnrico Granata     }
652245b3caaSEnrico Granata     else
653245b3caaSEnrico Granata         m_stream->EOL();
6544d93b8cdSEnrico Granata }
6550f883ffbSEnrico Granata 
6560f883ffbSEnrico Granata bool
6570f883ffbSEnrico Granata ValueObjectPrinter::ShouldPrintValidation ()
6580f883ffbSEnrico Granata {
6590f883ffbSEnrico Granata     return options.m_run_validator;
6600f883ffbSEnrico Granata }
6610f883ffbSEnrico Granata 
6620f883ffbSEnrico Granata bool
6630f883ffbSEnrico Granata ValueObjectPrinter::PrintValidationMarkerIfNeeded ()
6640f883ffbSEnrico Granata {
6650f883ffbSEnrico Granata     if (!ShouldPrintValidation())
6660f883ffbSEnrico Granata         return false;
6670f883ffbSEnrico Granata 
6680f883ffbSEnrico Granata     m_validation = m_valobj->GetValidationStatus();
6690f883ffbSEnrico Granata 
6700f883ffbSEnrico Granata     if (TypeValidatorResult::Failure == m_validation.first)
6710f883ffbSEnrico Granata     {
6720f883ffbSEnrico Granata         m_stream->Printf("! ");
6730f883ffbSEnrico Granata         return true;
6740f883ffbSEnrico Granata     }
6750f883ffbSEnrico Granata 
6760f883ffbSEnrico Granata     return false;
6770f883ffbSEnrico Granata }
6780f883ffbSEnrico Granata 
6790f883ffbSEnrico Granata bool
6800f883ffbSEnrico Granata ValueObjectPrinter::PrintValidationErrorIfNeeded ()
6810f883ffbSEnrico Granata {
6820f883ffbSEnrico Granata     if (!ShouldPrintValidation())
6830f883ffbSEnrico Granata         return false;
6840f883ffbSEnrico Granata 
6850f883ffbSEnrico Granata     if (TypeValidatorResult::Success == m_validation.first)
6860f883ffbSEnrico Granata         return false;
6870f883ffbSEnrico Granata 
6880f883ffbSEnrico Granata     if (m_validation.second.empty())
6890f883ffbSEnrico Granata         m_validation.second.assign("unknown error");
6900f883ffbSEnrico Granata 
6910f883ffbSEnrico Granata     m_stream->Printf(" ! validation error: %s", m_validation.second.c_str());
6920f883ffbSEnrico Granata     m_stream->EOL();
6930f883ffbSEnrico Granata 
6940f883ffbSEnrico Granata     return true;
6950f883ffbSEnrico Granata }
696