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 24d5957336SEnrico Granata DumpValueObjectOptions::DumpValueObjectOptions (ValueObject& valobj) : 25d5957336SEnrico Granata DumpValueObjectOptions() 26d5957336SEnrico Granata { 27d5957336SEnrico Granata m_use_dynamic = valobj.GetDynamicValueType(); 28d5957336SEnrico Granata m_use_synthetic = valobj.IsSynthetic(); 29d5957336SEnrico Granata } 30d5957336SEnrico Granata 31d5957336SEnrico Granata ValueObjectPrinter::ValueObjectPrinter (ValueObject* valobj, 32d5957336SEnrico Granata Stream* s) 33d5957336SEnrico Granata { 34d5957336SEnrico Granata if (valobj) 35d5957336SEnrico Granata { 36d5957336SEnrico Granata DumpValueObjectOptions options(*valobj); 37d5957336SEnrico Granata Init (valobj,s,options,options.m_max_ptr_depth,0); 38d5957336SEnrico Granata } 39d5957336SEnrico Granata else 40d5957336SEnrico Granata { 41d5957336SEnrico Granata DumpValueObjectOptions options; 42d5957336SEnrico Granata Init (valobj,s,options,options.m_max_ptr_depth,0); 43d5957336SEnrico Granata } 44d5957336SEnrico Granata } 45d5957336SEnrico Granata 464d93b8cdSEnrico Granata ValueObjectPrinter::ValueObjectPrinter (ValueObject* valobj, 474d93b8cdSEnrico Granata Stream* s, 48938d1d67SEnrico Granata const DumpValueObjectOptions& options) 494d93b8cdSEnrico Granata { 50938d1d67SEnrico Granata Init(valobj,s,options,options.m_max_ptr_depth,0); 514d93b8cdSEnrico Granata } 524d93b8cdSEnrico Granata 534d93b8cdSEnrico Granata ValueObjectPrinter::ValueObjectPrinter (ValueObject* valobj, 544d93b8cdSEnrico Granata Stream* s, 554d93b8cdSEnrico Granata const DumpValueObjectOptions& options, 56c1b7c09aSEnrico Granata const DumpValueObjectOptions::PointerDepth& ptr_depth, 57938d1d67SEnrico Granata uint32_t curr_depth) 58938d1d67SEnrico Granata { 59938d1d67SEnrico Granata Init(valobj,s,options,ptr_depth,curr_depth); 60938d1d67SEnrico Granata } 61938d1d67SEnrico Granata 62938d1d67SEnrico Granata void 63938d1d67SEnrico Granata ValueObjectPrinter::Init (ValueObject* valobj, 64938d1d67SEnrico Granata Stream* s, 65938d1d67SEnrico Granata const DumpValueObjectOptions& options, 66c1b7c09aSEnrico Granata const DumpValueObjectOptions::PointerDepth& ptr_depth, 67938d1d67SEnrico Granata uint32_t curr_depth) 68938d1d67SEnrico Granata { 69938d1d67SEnrico Granata m_orig_valobj = valobj; 70938d1d67SEnrico Granata m_valobj = nullptr; 71938d1d67SEnrico Granata m_stream = s; 72938d1d67SEnrico Granata this->options = options; 73938d1d67SEnrico Granata m_ptr_depth = ptr_depth; 74938d1d67SEnrico Granata m_curr_depth = curr_depth; 75938d1d67SEnrico Granata assert (m_orig_valobj && "cannot print a NULL ValueObject"); 76938d1d67SEnrico Granata assert (m_stream && "cannot print to a NULL Stream"); 77938d1d67SEnrico Granata m_should_print = eLazyBoolCalculate; 78938d1d67SEnrico Granata m_is_nil = eLazyBoolCalculate; 79938d1d67SEnrico Granata m_is_ptr = eLazyBoolCalculate; 80938d1d67SEnrico Granata m_is_ref = eLazyBoolCalculate; 81938d1d67SEnrico Granata m_is_aggregate = eLazyBoolCalculate; 82938d1d67SEnrico Granata m_summary_formatter = {nullptr,false}; 83938d1d67SEnrico Granata m_value.assign(""); 84938d1d67SEnrico Granata m_summary.assign(""); 85938d1d67SEnrico Granata m_error.assign(""); 86938d1d67SEnrico Granata } 874d93b8cdSEnrico Granata 884d93b8cdSEnrico Granata bool 894d93b8cdSEnrico Granata ValueObjectPrinter::PrintValueObject () 904d93b8cdSEnrico Granata { 91d07cfd3aSEnrico Granata if (!GetMostSpecializedValue () || m_valobj == nullptr) 924d93b8cdSEnrico Granata return false; 934d93b8cdSEnrico Granata 944d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 954d93b8cdSEnrico Granata { 960f883ffbSEnrico Granata PrintValidationMarkerIfNeeded(); 970f883ffbSEnrico Granata 984d93b8cdSEnrico Granata PrintLocationIfNeeded(); 994d93b8cdSEnrico Granata m_stream->Indent(); 1004d93b8cdSEnrico Granata 1014d93b8cdSEnrico Granata bool show_type = PrintTypeIfNeeded(); 1024d93b8cdSEnrico Granata 1034d93b8cdSEnrico Granata PrintNameIfNeeded(show_type); 1044d93b8cdSEnrico Granata } 1054d93b8cdSEnrico Granata 1064d93b8cdSEnrico Granata bool value_printed = false; 1074d93b8cdSEnrico Granata bool summary_printed = false; 1084d93b8cdSEnrico Granata 1094d93b8cdSEnrico Granata bool val_summary_ok = PrintValueAndSummaryIfNeeded (value_printed,summary_printed); 1104d93b8cdSEnrico Granata 1114d93b8cdSEnrico Granata if (val_summary_ok) 1124d93b8cdSEnrico Granata PrintChildrenIfNeeded (value_printed, summary_printed); 11339938938SEnrico Granata else 1144d93b8cdSEnrico Granata m_stream->EOL(); 1154d93b8cdSEnrico Granata 1160f883ffbSEnrico Granata PrintValidationErrorIfNeeded(); 1170f883ffbSEnrico Granata 1184d93b8cdSEnrico Granata return true; 1194d93b8cdSEnrico Granata } 1204d93b8cdSEnrico Granata 1214d93b8cdSEnrico Granata bool 122d07cfd3aSEnrico Granata ValueObjectPrinter::GetMostSpecializedValue () 1234d93b8cdSEnrico Granata { 124a29cb0baSEnrico Granata if (m_valobj) 125a29cb0baSEnrico Granata return true; 1264d93b8cdSEnrico Granata bool update_success = m_orig_valobj->UpdateValueIfNeeded (true); 1274d93b8cdSEnrico Granata if (!update_success) 128106260c5SEnrico Granata { 129106260c5SEnrico Granata m_valobj = m_orig_valobj; 130106260c5SEnrico Granata } 131106260c5SEnrico Granata else 132106260c5SEnrico Granata { 133106260c5SEnrico Granata if (m_orig_valobj->IsDynamic()) 134106260c5SEnrico Granata { 135106260c5SEnrico Granata if (options.m_use_dynamic == eNoDynamicValues) 136106260c5SEnrico Granata { 137106260c5SEnrico Granata ValueObject *static_value = m_orig_valobj->GetStaticValue().get(); 138106260c5SEnrico Granata if (static_value) 139106260c5SEnrico Granata m_valobj = static_value; 140106260c5SEnrico Granata else 141106260c5SEnrico Granata m_valobj = m_orig_valobj; 142106260c5SEnrico Granata } 143106260c5SEnrico Granata else 144106260c5SEnrico Granata m_valobj = m_orig_valobj; 145106260c5SEnrico Granata } 146106260c5SEnrico Granata else 147106260c5SEnrico Granata { 1484d93b8cdSEnrico Granata if (options.m_use_dynamic != eNoDynamicValues) 1494d93b8cdSEnrico Granata { 1504d93b8cdSEnrico Granata ValueObject *dynamic_value = m_orig_valobj->GetDynamicValue(options.m_use_dynamic).get(); 1514d93b8cdSEnrico Granata if (dynamic_value) 1524d93b8cdSEnrico Granata m_valobj = dynamic_value; 1534d93b8cdSEnrico Granata else 1544d93b8cdSEnrico Granata m_valobj = m_orig_valobj; 1554d93b8cdSEnrico Granata } 1564d93b8cdSEnrico Granata else 1574d93b8cdSEnrico Granata m_valobj = m_orig_valobj; 158106260c5SEnrico Granata } 159d07cfd3aSEnrico Granata 160d07cfd3aSEnrico Granata if (m_valobj->IsSynthetic()) 161d07cfd3aSEnrico Granata { 162d07cfd3aSEnrico Granata if (options.m_use_synthetic == false) 163d07cfd3aSEnrico Granata { 164d07cfd3aSEnrico Granata ValueObject *non_synthetic = m_valobj->GetNonSyntheticValue().get(); 165d07cfd3aSEnrico Granata if (non_synthetic) 166d07cfd3aSEnrico Granata m_valobj = non_synthetic; 167d07cfd3aSEnrico Granata } 168d07cfd3aSEnrico Granata } 169d07cfd3aSEnrico Granata else 170d07cfd3aSEnrico Granata { 171d07cfd3aSEnrico Granata if (options.m_use_synthetic == true) 172d07cfd3aSEnrico Granata { 173d07cfd3aSEnrico Granata ValueObject *synthetic = m_valobj->GetSyntheticValue().get(); 174d07cfd3aSEnrico Granata if (synthetic) 175d07cfd3aSEnrico Granata m_valobj = synthetic; 176d07cfd3aSEnrico Granata } 177d07cfd3aSEnrico Granata } 178106260c5SEnrico Granata } 17999558cc4SGreg Clayton m_clang_type = m_valobj->GetCompilerType(); 1804d93b8cdSEnrico Granata m_type_flags = m_clang_type.GetTypeInfo (); 1814d93b8cdSEnrico Granata return true; 1824d93b8cdSEnrico Granata } 1834d93b8cdSEnrico Granata 1844d93b8cdSEnrico Granata const char* 1854d93b8cdSEnrico Granata ValueObjectPrinter::GetDescriptionForDisplay () 1864d93b8cdSEnrico Granata { 1874d93b8cdSEnrico Granata const char* str = m_valobj->GetObjectDescription(); 1884d93b8cdSEnrico Granata if (!str) 1894d93b8cdSEnrico Granata str = m_valobj->GetSummaryAsCString(); 1904d93b8cdSEnrico Granata if (!str) 1914d93b8cdSEnrico Granata str = m_valobj->GetValueAsCString(); 1924d93b8cdSEnrico Granata return str; 1934d93b8cdSEnrico Granata } 1944d93b8cdSEnrico Granata 1954d93b8cdSEnrico Granata const char* 1964d93b8cdSEnrico Granata ValueObjectPrinter::GetRootNameForDisplay (const char* if_fail) 1974d93b8cdSEnrico Granata { 1984d93b8cdSEnrico Granata const char *root_valobj_name = options.m_root_valobj_name.empty() ? 1994d93b8cdSEnrico Granata m_valobj->GetName().AsCString() : 2004d93b8cdSEnrico Granata options.m_root_valobj_name.c_str(); 2014d93b8cdSEnrico Granata return root_valobj_name ? root_valobj_name : if_fail; 2024d93b8cdSEnrico Granata } 2034d93b8cdSEnrico Granata 2044d93b8cdSEnrico Granata bool 2054d93b8cdSEnrico Granata ValueObjectPrinter::ShouldPrintValueObject () 2064d93b8cdSEnrico Granata { 2074d93b8cdSEnrico Granata if (m_should_print == eLazyBoolCalculate) 208622be238SEnrico Granata m_should_print = (options.m_flat_output == false || m_type_flags.Test (eTypeHasValue)) ? eLazyBoolYes : eLazyBoolNo; 2094d93b8cdSEnrico Granata return m_should_print == eLazyBoolYes; 2104d93b8cdSEnrico Granata } 2114d93b8cdSEnrico Granata 2124d93b8cdSEnrico Granata bool 2134d93b8cdSEnrico Granata ValueObjectPrinter::IsNil () 2144d93b8cdSEnrico Granata { 2154d93b8cdSEnrico Granata if (m_is_nil == eLazyBoolCalculate) 2164d93b8cdSEnrico Granata m_is_nil = m_valobj->IsObjCNil() ? eLazyBoolYes : eLazyBoolNo; 2174d93b8cdSEnrico Granata return m_is_nil == eLazyBoolYes; 2184d93b8cdSEnrico Granata } 2194d93b8cdSEnrico Granata 2204d93b8cdSEnrico Granata bool 2214d93b8cdSEnrico Granata ValueObjectPrinter::IsPtr () 2224d93b8cdSEnrico Granata { 2234d93b8cdSEnrico Granata if (m_is_ptr == eLazyBoolCalculate) 224622be238SEnrico Granata m_is_ptr = m_type_flags.Test (eTypeIsPointer) ? eLazyBoolYes : eLazyBoolNo; 2254d93b8cdSEnrico Granata return m_is_ptr == eLazyBoolYes; 2264d93b8cdSEnrico Granata } 2274d93b8cdSEnrico Granata 2284d93b8cdSEnrico Granata bool 2294d93b8cdSEnrico Granata ValueObjectPrinter::IsRef () 2304d93b8cdSEnrico Granata { 2314d93b8cdSEnrico Granata if (m_is_ref == eLazyBoolCalculate) 232622be238SEnrico Granata m_is_ref = m_type_flags.Test (eTypeIsReference) ? eLazyBoolYes : eLazyBoolNo; 2334d93b8cdSEnrico Granata return m_is_ref == eLazyBoolYes; 2344d93b8cdSEnrico Granata } 2354d93b8cdSEnrico Granata 2364d93b8cdSEnrico Granata bool 2374d93b8cdSEnrico Granata ValueObjectPrinter::IsAggregate () 2384d93b8cdSEnrico Granata { 2394d93b8cdSEnrico Granata if (m_is_aggregate == eLazyBoolCalculate) 240622be238SEnrico Granata m_is_aggregate = m_type_flags.Test (eTypeHasChildren) ? eLazyBoolYes : eLazyBoolNo; 2414d93b8cdSEnrico Granata return m_is_aggregate == eLazyBoolYes; 2424d93b8cdSEnrico Granata } 2434d93b8cdSEnrico Granata 2444d93b8cdSEnrico Granata bool 2454d93b8cdSEnrico Granata ValueObjectPrinter::PrintLocationIfNeeded () 2464d93b8cdSEnrico Granata { 2474d93b8cdSEnrico Granata if (options.m_show_location) 2484d93b8cdSEnrico Granata { 2494d93b8cdSEnrico Granata m_stream->Printf("%s: ", m_valobj->GetLocationAsCString()); 2504d93b8cdSEnrico Granata return true; 2514d93b8cdSEnrico Granata } 2524d93b8cdSEnrico Granata return false; 2534d93b8cdSEnrico Granata } 2544d93b8cdSEnrico Granata 2554d93b8cdSEnrico Granata bool 2564d93b8cdSEnrico Granata ValueObjectPrinter::PrintTypeIfNeeded () 2574d93b8cdSEnrico Granata { 2584d93b8cdSEnrico Granata bool show_type = true; 2594d93b8cdSEnrico Granata // if we are at the root-level and been asked to hide the root's type, then hide it 2604d93b8cdSEnrico Granata if (m_curr_depth == 0 && options.m_hide_root_type) 2614d93b8cdSEnrico Granata show_type = false; 2624d93b8cdSEnrico Granata else 2634d93b8cdSEnrico Granata // otherwise decide according to the usual rules (asked to show types - always at the root level) 2644d93b8cdSEnrico Granata show_type = options.m_show_types || (m_curr_depth == 0 && !options.m_flat_output); 2654d93b8cdSEnrico Granata 2664d93b8cdSEnrico Granata if (show_type) 2674d93b8cdSEnrico Granata { 2684d93b8cdSEnrico Granata // Some ValueObjects don't have types (like registers sets). Only print 2694d93b8cdSEnrico Granata // the type if there is one to print 270a126e462SEnrico Granata ConstString type_name; 271a126e462SEnrico Granata if (options.m_use_type_display_name) 272a126e462SEnrico Granata type_name = m_valobj->GetDisplayTypeName(); 273e8daa2f8SEnrico Granata else 274a126e462SEnrico Granata type_name = m_valobj->GetQualifiedTypeName(); 275a126e462SEnrico Granata if (type_name) 276a126e462SEnrico Granata m_stream->Printf("(%s) ", type_name.GetCString()); 2774d93b8cdSEnrico Granata else 2784d93b8cdSEnrico Granata show_type = false; 2794d93b8cdSEnrico Granata } 2804d93b8cdSEnrico Granata return show_type; 2814d93b8cdSEnrico Granata } 2824d93b8cdSEnrico Granata 2834d93b8cdSEnrico Granata bool 2844d93b8cdSEnrico Granata ValueObjectPrinter::PrintNameIfNeeded (bool show_type) 2854d93b8cdSEnrico Granata { 2864d93b8cdSEnrico Granata if (options.m_flat_output) 2874d93b8cdSEnrico Granata { 2884d93b8cdSEnrico Granata // If we are showing types, also qualify the C++ base classes 2894d93b8cdSEnrico Granata const bool qualify_cxx_base_classes = show_type; 2904d93b8cdSEnrico Granata if (!options.m_hide_name) 2914d93b8cdSEnrico Granata { 2924d93b8cdSEnrico Granata m_valobj->GetExpressionPath(*m_stream, qualify_cxx_base_classes); 2934d93b8cdSEnrico Granata m_stream->PutCString(" ="); 2944d93b8cdSEnrico Granata return true; 2954d93b8cdSEnrico Granata } 2964d93b8cdSEnrico Granata } 2974d93b8cdSEnrico Granata else if (!options.m_hide_name) 2984d93b8cdSEnrico Granata { 2994d93b8cdSEnrico Granata const char *name_cstr = GetRootNameForDisplay(""); 3004d93b8cdSEnrico Granata m_stream->Printf ("%s =", name_cstr); 3014d93b8cdSEnrico Granata return true; 3024d93b8cdSEnrico Granata } 3034d93b8cdSEnrico Granata return false; 3044d93b8cdSEnrico Granata } 3054d93b8cdSEnrico Granata 3064d93b8cdSEnrico Granata bool 3074d93b8cdSEnrico Granata ValueObjectPrinter::CheckScopeIfNeeded () 3084d93b8cdSEnrico Granata { 3094d93b8cdSEnrico Granata if (options.m_scope_already_checked) 3104d93b8cdSEnrico Granata return true; 3114d93b8cdSEnrico Granata return m_valobj->IsInScope(); 3124d93b8cdSEnrico Granata } 3134d93b8cdSEnrico Granata 3144d93b8cdSEnrico Granata TypeSummaryImpl* 3154d93b8cdSEnrico Granata ValueObjectPrinter::GetSummaryFormatter () 3164d93b8cdSEnrico Granata { 3174d93b8cdSEnrico Granata if (m_summary_formatter.second == false) 3184d93b8cdSEnrico Granata { 3194d93b8cdSEnrico Granata TypeSummaryImpl* entry = options.m_summary_sp ? options.m_summary_sp.get() : m_valobj->GetSummaryFormat().get(); 3204d93b8cdSEnrico Granata 3214d93b8cdSEnrico Granata if (options.m_omit_summary_depth > 0) 3224d93b8cdSEnrico Granata entry = NULL; 3234d93b8cdSEnrico Granata m_summary_formatter.first = entry; 3244d93b8cdSEnrico Granata m_summary_formatter.second = true; 3254d93b8cdSEnrico Granata } 3264d93b8cdSEnrico Granata return m_summary_formatter.first; 3274d93b8cdSEnrico Granata } 3284d93b8cdSEnrico Granata 3294d93b8cdSEnrico Granata void 3304d93b8cdSEnrico Granata ValueObjectPrinter::GetValueSummaryError (std::string& value, 3314d93b8cdSEnrico Granata std::string& summary, 3324d93b8cdSEnrico Granata std::string& error) 3334d93b8cdSEnrico Granata { 334465f4bc2SEnrico Granata if (options.m_format != eFormatDefault && options.m_format != m_valobj->GetFormat()) 3354d93b8cdSEnrico Granata { 336465f4bc2SEnrico Granata m_valobj->GetValueAsCString(options.m_format, 337465f4bc2SEnrico Granata value); 3384d93b8cdSEnrico Granata } 339465f4bc2SEnrico Granata else 340465f4bc2SEnrico Granata { 3414d93b8cdSEnrico Granata const char* val_cstr = m_valobj->GetValueAsCString(); 342465f4bc2SEnrico Granata if (val_cstr) 343465f4bc2SEnrico Granata value.assign(val_cstr); 344465f4bc2SEnrico Granata } 3454d93b8cdSEnrico Granata const char* err_cstr = m_valobj->GetError().AsCString(); 3464d93b8cdSEnrico Granata if (err_cstr) 3474d93b8cdSEnrico Granata error.assign(err_cstr); 3484d93b8cdSEnrico Granata 3494d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 3504d93b8cdSEnrico Granata { 3514d93b8cdSEnrico Granata if (IsNil()) 3524d93b8cdSEnrico Granata summary.assign("nil"); 3534d93b8cdSEnrico Granata else if (options.m_omit_summary_depth == 0) 3544d93b8cdSEnrico Granata { 3554d93b8cdSEnrico Granata TypeSummaryImpl* entry = GetSummaryFormatter(); 3564d93b8cdSEnrico Granata if (entry) 3574d93b8cdSEnrico Granata m_valobj->GetSummaryAsCString(entry, summary); 3584d93b8cdSEnrico Granata else 3594d93b8cdSEnrico Granata { 3604d93b8cdSEnrico Granata const char* sum_cstr = m_valobj->GetSummaryAsCString(); 3614d93b8cdSEnrico Granata if (sum_cstr) 3624d93b8cdSEnrico Granata summary.assign(sum_cstr); 3634d93b8cdSEnrico Granata } 3644d93b8cdSEnrico Granata } 3654d93b8cdSEnrico Granata } 3664d93b8cdSEnrico Granata } 3674d93b8cdSEnrico Granata 3684d93b8cdSEnrico Granata bool 3694d93b8cdSEnrico Granata ValueObjectPrinter::PrintValueAndSummaryIfNeeded (bool& value_printed, 3704d93b8cdSEnrico Granata bool& summary_printed) 3714d93b8cdSEnrico Granata { 3724d93b8cdSEnrico Granata bool error_printed = false; 3734d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 3744d93b8cdSEnrico Granata { 3754d93b8cdSEnrico Granata if (!CheckScopeIfNeeded()) 3764d93b8cdSEnrico Granata m_error.assign("out of scope"); 3774d93b8cdSEnrico Granata if (m_error.empty()) 3784d93b8cdSEnrico Granata { 3794d93b8cdSEnrico Granata GetValueSummaryError(m_value, m_summary, m_error); 3804d93b8cdSEnrico Granata } 3814d93b8cdSEnrico Granata if (m_error.size()) 3824d93b8cdSEnrico Granata { 3834d93b8cdSEnrico Granata error_printed = true; 3844d93b8cdSEnrico Granata m_stream->Printf (" <%s>\n", m_error.c_str()); 3854d93b8cdSEnrico Granata } 3864d93b8cdSEnrico Granata else 3874d93b8cdSEnrico Granata { 3884d93b8cdSEnrico Granata // Make sure we have a value and make sure the summary didn't 3894d93b8cdSEnrico Granata // specify that the value should not be printed - and do not print 3904d93b8cdSEnrico Granata // the value if this thing is nil 3914d93b8cdSEnrico Granata // (but show the value if the user passes a format explicitly) 3924d93b8cdSEnrico Granata TypeSummaryImpl* entry = GetSummaryFormatter(); 3938a068e6cSEnrico Granata if (!IsNil() && !m_value.empty() && (entry == NULL || (entry->DoesPrintValue(m_valobj) || options.m_format != eFormatDefault) || m_summary.empty()) && !options.m_hide_value) 3944d93b8cdSEnrico Granata { 3954d93b8cdSEnrico Granata m_stream->Printf(" %s", m_value.c_str()); 3964d93b8cdSEnrico Granata value_printed = true; 3974d93b8cdSEnrico Granata } 3984d93b8cdSEnrico Granata 3994d93b8cdSEnrico Granata if (m_summary.size()) 4004d93b8cdSEnrico Granata { 4014d93b8cdSEnrico Granata m_stream->Printf(" %s", m_summary.c_str()); 4024d93b8cdSEnrico Granata summary_printed = true; 4034d93b8cdSEnrico Granata } 4044d93b8cdSEnrico Granata } 4054d93b8cdSEnrico Granata } 4064d93b8cdSEnrico Granata return !error_printed; 4074d93b8cdSEnrico Granata } 4084d93b8cdSEnrico Granata 4094d93b8cdSEnrico Granata bool 4104d93b8cdSEnrico Granata ValueObjectPrinter::PrintObjectDescriptionIfNeeded (bool value_printed, 4114d93b8cdSEnrico Granata bool summary_printed) 4124d93b8cdSEnrico Granata { 4134d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 4144d93b8cdSEnrico Granata { 4154d93b8cdSEnrico Granata // let's avoid the overly verbose no description error for a nil thing 4164d93b8cdSEnrico Granata if (options.m_use_objc && !IsNil()) 4174d93b8cdSEnrico Granata { 4184d93b8cdSEnrico Granata if (!options.m_hide_value || !options.m_hide_name) 4194d93b8cdSEnrico Granata m_stream->Printf(" "); 4204d93b8cdSEnrico Granata const char *object_desc = nullptr; 4214d93b8cdSEnrico Granata if (value_printed || summary_printed) 4224d93b8cdSEnrico Granata object_desc = m_valobj->GetObjectDescription(); 4234d93b8cdSEnrico Granata else 4244d93b8cdSEnrico Granata object_desc = GetDescriptionForDisplay(); 4254d93b8cdSEnrico Granata if (object_desc && *object_desc) 4264d93b8cdSEnrico Granata { 4274d93b8cdSEnrico Granata m_stream->Printf("%s\n", object_desc); 4284d93b8cdSEnrico Granata return true; 4294d93b8cdSEnrico Granata } 4304d93b8cdSEnrico Granata else if (value_printed == false && summary_printed == false) 4314d93b8cdSEnrico Granata return true; 4324d93b8cdSEnrico Granata else 4334d93b8cdSEnrico Granata return false; 4344d93b8cdSEnrico Granata } 4354d93b8cdSEnrico Granata } 4364d93b8cdSEnrico Granata return true; 4374d93b8cdSEnrico Granata } 4384d93b8cdSEnrico Granata 4394d93b8cdSEnrico Granata bool 440c1b7c09aSEnrico Granata DumpValueObjectOptions::PointerDepth::CanAllowExpansion (bool is_root, 441c1b7c09aSEnrico Granata TypeSummaryImpl* entry, 442c1b7c09aSEnrico Granata ValueObject *valobj, 443c1b7c09aSEnrico Granata const std::string& summary) 444c1b7c09aSEnrico Granata { 445c1b7c09aSEnrico Granata switch (m_mode) 446c1b7c09aSEnrico Granata { 447c1b7c09aSEnrico Granata case Mode::Always: 448c1b7c09aSEnrico Granata return (m_count > 0); 449c1b7c09aSEnrico Granata case Mode::Never: 450c1b7c09aSEnrico Granata return false; 451c1b7c09aSEnrico Granata case Mode::Default: 452c1b7c09aSEnrico Granata if (is_root) 453c1b7c09aSEnrico Granata m_count = std::min<decltype(m_count)>(m_count,1); 454c1b7c09aSEnrico Granata return m_count > 0; 455c1b7c09aSEnrico Granata case Mode::Formatters: 456c1b7c09aSEnrico Granata if (!entry || entry->DoesPrintChildren(valobj) || summary.empty()) 457c1b7c09aSEnrico Granata return m_count > 0; 458c1b7c09aSEnrico Granata return false; 459c1b7c09aSEnrico Granata } 460*84f5b0dfSZachary Turner return false; 461c1b7c09aSEnrico Granata } 462c1b7c09aSEnrico Granata 463c1b7c09aSEnrico Granata bool 464c1b7c09aSEnrico Granata DumpValueObjectOptions::PointerDepth::CanAllowExpansion () const 465c1b7c09aSEnrico Granata { 466c1b7c09aSEnrico Granata switch (m_mode) 467c1b7c09aSEnrico Granata { 468c1b7c09aSEnrico Granata case Mode::Always: 469c1b7c09aSEnrico Granata case Mode::Default: 470c1b7c09aSEnrico Granata case Mode::Formatters: 471c1b7c09aSEnrico Granata return (m_count > 0); 472c1b7c09aSEnrico Granata case Mode::Never: 473c1b7c09aSEnrico Granata return false; 474c1b7c09aSEnrico Granata } 475*84f5b0dfSZachary Turner return false; 476c1b7c09aSEnrico Granata } 477c1b7c09aSEnrico Granata 478c1b7c09aSEnrico Granata bool 4794d93b8cdSEnrico Granata ValueObjectPrinter::ShouldPrintChildren (bool is_failed_description, 480c1b7c09aSEnrico Granata DumpValueObjectOptions::PointerDepth& curr_ptr_depth) 4814d93b8cdSEnrico Granata { 4824d93b8cdSEnrico Granata const bool is_ref = IsRef (); 4834d93b8cdSEnrico Granata const bool is_ptr = IsPtr (); 4844d93b8cdSEnrico Granata 485c1b7c09aSEnrico Granata TypeSummaryImpl* entry = GetSummaryFormatter(); 486c1b7c09aSEnrico Granata 4874d93b8cdSEnrico Granata if (is_failed_description || m_curr_depth < options.m_max_depth) 4884d93b8cdSEnrico Granata { 4894d93b8cdSEnrico Granata // We will show children for all concrete types. We won't show 4904d93b8cdSEnrico Granata // pointer contents unless a pointer depth has been specified. 4914d93b8cdSEnrico Granata // We won't reference contents unless the reference is the 4924d93b8cdSEnrico Granata // root object (depth of zero). 4934d93b8cdSEnrico Granata 4944d93b8cdSEnrico Granata // Use a new temporary pointer depth in case we override the 4954d93b8cdSEnrico Granata // current pointer depth below... 4964d93b8cdSEnrico Granata 4974d93b8cdSEnrico Granata if (is_ptr || is_ref) 4984d93b8cdSEnrico Granata { 4994d93b8cdSEnrico Granata // We have a pointer or reference whose value is an address. 5004d93b8cdSEnrico Granata // Make sure that address is not NULL 5014d93b8cdSEnrico Granata AddressType ptr_address_type; 5024d93b8cdSEnrico Granata if (m_valobj->GetPointerValue (&ptr_address_type) == 0) 5034d93b8cdSEnrico Granata return false; 5044d93b8cdSEnrico Granata 505c1b7c09aSEnrico Granata const bool is_root_level = m_curr_depth == 0; 506c1b7c09aSEnrico Granata 507c1b7c09aSEnrico Granata if (is_ref && 508c1b7c09aSEnrico Granata is_root_level) 5094d93b8cdSEnrico Granata { 5104d93b8cdSEnrico Granata // If this is the root object (depth is zero) that we are showing 5114d93b8cdSEnrico Granata // and it is a reference, and no pointer depth has been supplied 5124d93b8cdSEnrico Granata // print out what it references. Don't do this at deeper depths 5134d93b8cdSEnrico Granata // otherwise we can end up with infinite recursion... 514c1b7c09aSEnrico Granata return true; 5154d93b8cdSEnrico Granata } 5164d93b8cdSEnrico Granata 517c1b7c09aSEnrico Granata return curr_ptr_depth.CanAllowExpansion(false, entry, m_valobj, m_summary); 5184d93b8cdSEnrico Granata } 5194d93b8cdSEnrico Granata 5208a068e6cSEnrico Granata return (!entry || entry->DoesPrintChildren(m_valobj) || m_summary.empty()); 5214d93b8cdSEnrico Granata } 5224d93b8cdSEnrico Granata return false; 5234d93b8cdSEnrico Granata } 5244d93b8cdSEnrico Granata 525d26eb907SSiva Chandra bool 526d26eb907SSiva Chandra ValueObjectPrinter::ShouldExpandEmptyAggregates () 527d26eb907SSiva Chandra { 528d26eb907SSiva Chandra TypeSummaryImpl* entry = GetSummaryFormatter(); 529d26eb907SSiva Chandra 530d26eb907SSiva Chandra if (!entry) 531d26eb907SSiva Chandra return true; 532d26eb907SSiva Chandra 533d26eb907SSiva Chandra return entry->DoesPrintEmptyAggregates(); 534d26eb907SSiva Chandra } 535d26eb907SSiva Chandra 5364d93b8cdSEnrico Granata ValueObject* 5374d93b8cdSEnrico Granata ValueObjectPrinter::GetValueObjectForChildrenGeneration () 5384d93b8cdSEnrico Granata { 539d07cfd3aSEnrico Granata return m_valobj; 5404d93b8cdSEnrico Granata } 5414d93b8cdSEnrico Granata 5424d93b8cdSEnrico Granata void 5434d93b8cdSEnrico Granata ValueObjectPrinter::PrintChildrenPreamble () 5444d93b8cdSEnrico Granata { 5454d93b8cdSEnrico Granata if (options.m_flat_output) 5464d93b8cdSEnrico Granata { 5474d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 5484d93b8cdSEnrico Granata m_stream->EOL(); 5494d93b8cdSEnrico Granata } 5504d93b8cdSEnrico Granata else 5514d93b8cdSEnrico Granata { 5524d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 5534d93b8cdSEnrico Granata m_stream->PutCString(IsRef () ? ": {\n" : " {\n"); 5544d93b8cdSEnrico Granata m_stream->IndentMore(); 5554d93b8cdSEnrico Granata } 5564d93b8cdSEnrico Granata } 5574d93b8cdSEnrico Granata 5584d93b8cdSEnrico Granata void 5594d93b8cdSEnrico Granata ValueObjectPrinter::PrintChild (ValueObjectSP child_sp, 560c1b7c09aSEnrico Granata const DumpValueObjectOptions::PointerDepth& curr_ptr_depth) 5614d93b8cdSEnrico Granata { 5624d93b8cdSEnrico Granata DumpValueObjectOptions child_options(options); 563443844c4SEnrico Granata child_options.SetFormat(options.m_format).SetSummary().SetRootValueObjectName(); 5644d93b8cdSEnrico Granata child_options.SetScopeChecked(true).SetHideName(options.m_hide_name).SetHideValue(options.m_hide_value) 5654d93b8cdSEnrico Granata .SetOmitSummaryDepth(child_options.m_omit_summary_depth > 1 ? child_options.m_omit_summary_depth - 1 : 0); 5664d93b8cdSEnrico Granata if (child_sp.get()) 5674d93b8cdSEnrico Granata { 5684d93b8cdSEnrico Granata ValueObjectPrinter child_printer(child_sp.get(), 5694d93b8cdSEnrico Granata m_stream, 5704d93b8cdSEnrico Granata child_options, 571c1b7c09aSEnrico Granata (IsPtr() || IsRef()) ? --curr_ptr_depth : curr_ptr_depth, 5724d93b8cdSEnrico Granata m_curr_depth + 1); 5734d93b8cdSEnrico Granata child_printer.PrintValueObject(); 5744d93b8cdSEnrico Granata } 5754d93b8cdSEnrico Granata } 5764d93b8cdSEnrico Granata 5774d93b8cdSEnrico Granata uint32_t 5784d93b8cdSEnrico Granata ValueObjectPrinter::GetMaxNumChildrenToPrint (bool& print_dotdotdot) 5794d93b8cdSEnrico Granata { 5804d93b8cdSEnrico Granata ValueObject* synth_m_valobj = GetValueObjectForChildrenGeneration(); 5814d93b8cdSEnrico Granata 5824d93b8cdSEnrico Granata size_t num_children = synth_m_valobj->GetNumChildren(); 5834d93b8cdSEnrico Granata print_dotdotdot = false; 5844d93b8cdSEnrico Granata if (num_children) 5854d93b8cdSEnrico Granata { 5864d93b8cdSEnrico Granata const size_t max_num_children = m_valobj->GetTargetSP()->GetMaximumNumberOfChildrenToDisplay(); 5874d93b8cdSEnrico Granata 5884d93b8cdSEnrico Granata if (num_children > max_num_children && !options.m_ignore_cap) 5894d93b8cdSEnrico Granata { 5904d93b8cdSEnrico Granata print_dotdotdot = true; 5914d93b8cdSEnrico Granata return max_num_children; 5924d93b8cdSEnrico Granata } 5934d93b8cdSEnrico Granata } 5944d93b8cdSEnrico Granata return num_children; 5954d93b8cdSEnrico Granata } 5964d93b8cdSEnrico Granata 5974d93b8cdSEnrico Granata void 5984d93b8cdSEnrico Granata ValueObjectPrinter::PrintChildrenPostamble (bool print_dotdotdot) 5994d93b8cdSEnrico Granata { 6004d93b8cdSEnrico Granata if (!options.m_flat_output) 6014d93b8cdSEnrico Granata { 6024d93b8cdSEnrico Granata if (print_dotdotdot) 6034d93b8cdSEnrico Granata { 6044d93b8cdSEnrico Granata m_valobj->GetTargetSP()->GetDebugger().GetCommandInterpreter().ChildrenTruncated(); 6054d93b8cdSEnrico Granata m_stream->Indent("...\n"); 6064d93b8cdSEnrico Granata } 6074d93b8cdSEnrico Granata m_stream->IndentLess(); 6084d93b8cdSEnrico Granata m_stream->Indent("}\n"); 6094d93b8cdSEnrico Granata } 6104d93b8cdSEnrico Granata } 6114d93b8cdSEnrico Granata 6124d93b8cdSEnrico Granata void 613c1b7c09aSEnrico Granata ValueObjectPrinter::PrintChildren (bool value_printed, 614c1b7c09aSEnrico Granata bool summary_printed, 615c1b7c09aSEnrico Granata const DumpValueObjectOptions::PointerDepth& curr_ptr_depth) 6164d93b8cdSEnrico Granata { 6174d93b8cdSEnrico Granata ValueObject* synth_m_valobj = GetValueObjectForChildrenGeneration(); 6184d93b8cdSEnrico Granata 6194d93b8cdSEnrico Granata bool print_dotdotdot = false; 6204d93b8cdSEnrico Granata size_t num_children = GetMaxNumChildrenToPrint(print_dotdotdot); 6214d93b8cdSEnrico Granata if (num_children) 6224d93b8cdSEnrico Granata { 6234d93b8cdSEnrico Granata PrintChildrenPreamble (); 6244d93b8cdSEnrico Granata 6254d93b8cdSEnrico Granata for (size_t idx=0; idx<num_children; ++idx) 6264d93b8cdSEnrico Granata { 6274d93b8cdSEnrico Granata ValueObjectSP child_sp(synth_m_valobj->GetChildAtIndex(idx, true)); 6284d93b8cdSEnrico Granata PrintChild (child_sp, curr_ptr_depth); 6294d93b8cdSEnrico Granata } 6304d93b8cdSEnrico Granata 6314d93b8cdSEnrico Granata PrintChildrenPostamble (print_dotdotdot); 6324d93b8cdSEnrico Granata } 6334d93b8cdSEnrico Granata else if (IsAggregate()) 6344d93b8cdSEnrico Granata { 6354d93b8cdSEnrico Granata // Aggregate, no children... 6364d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 637d07cfd3aSEnrico Granata { 638d07cfd3aSEnrico Granata // if it has a synthetic value, then don't print {}, the synthetic children are probably only being used to vend a value 639d26eb907SSiva Chandra if (m_valobj->DoesProvideSyntheticValue() || !ShouldExpandEmptyAggregates()) 640d07cfd3aSEnrico Granata m_stream->PutCString( "\n"); 641d07cfd3aSEnrico Granata else 6424d93b8cdSEnrico Granata m_stream->PutCString(" {}\n"); 6434d93b8cdSEnrico Granata } 644d07cfd3aSEnrico Granata } 6454d93b8cdSEnrico Granata else 6464d93b8cdSEnrico Granata { 6474d93b8cdSEnrico Granata if (ShouldPrintValueObject()) 6484d93b8cdSEnrico Granata m_stream->EOL(); 6494d93b8cdSEnrico Granata } 6504d93b8cdSEnrico Granata } 6514d93b8cdSEnrico Granata 652a29cb0baSEnrico Granata bool 653a29cb0baSEnrico Granata ValueObjectPrinter::PrintChildrenOneLiner (bool hide_names) 654a29cb0baSEnrico Granata { 655d07cfd3aSEnrico Granata if (!GetMostSpecializedValue () || m_valobj == nullptr) 656a29cb0baSEnrico Granata return false; 657a29cb0baSEnrico Granata 658a29cb0baSEnrico Granata ValueObject* synth_m_valobj = GetValueObjectForChildrenGeneration(); 659a29cb0baSEnrico Granata 660a29cb0baSEnrico Granata bool print_dotdotdot = false; 661a29cb0baSEnrico Granata size_t num_children = GetMaxNumChildrenToPrint(print_dotdotdot); 662a29cb0baSEnrico Granata 663a29cb0baSEnrico Granata if (num_children) 664a29cb0baSEnrico Granata { 665a29cb0baSEnrico Granata m_stream->PutChar('('); 666a29cb0baSEnrico Granata 667a29cb0baSEnrico Granata for (uint32_t idx=0; idx<num_children; ++idx) 668a29cb0baSEnrico Granata { 669a29cb0baSEnrico Granata lldb::ValueObjectSP child_sp(synth_m_valobj->GetChildAtIndex(idx, true)); 670ddac7611SEnrico Granata if (child_sp) 671ddac7611SEnrico Granata child_sp = child_sp->GetQualifiedRepresentationIfAvailable(options.m_use_dynamic, options.m_use_synthetic); 672a29cb0baSEnrico Granata if (child_sp) 673a29cb0baSEnrico Granata { 674a29cb0baSEnrico Granata if (idx) 675a29cb0baSEnrico Granata m_stream->PutCString(", "); 676a29cb0baSEnrico Granata if (!hide_names) 677a29cb0baSEnrico Granata { 678a29cb0baSEnrico Granata const char* name = child_sp.get()->GetName().AsCString(); 679a29cb0baSEnrico Granata if (name && *name) 680a29cb0baSEnrico Granata { 681a29cb0baSEnrico Granata m_stream->PutCString(name); 682a29cb0baSEnrico Granata m_stream->PutCString(" = "); 683a29cb0baSEnrico Granata } 684a29cb0baSEnrico Granata } 685a29cb0baSEnrico Granata child_sp->DumpPrintableRepresentation(*m_stream, 686a29cb0baSEnrico Granata ValueObject::eValueObjectRepresentationStyleSummary, 687443844c4SEnrico Granata lldb::eFormatInvalid, 688a29cb0baSEnrico Granata ValueObject::ePrintableRepresentationSpecialCasesDisable); 689a29cb0baSEnrico Granata } 690a29cb0baSEnrico Granata } 691a29cb0baSEnrico Granata 692a29cb0baSEnrico Granata if (print_dotdotdot) 693a29cb0baSEnrico Granata m_stream->PutCString(", ...)"); 694a29cb0baSEnrico Granata else 695a29cb0baSEnrico Granata m_stream->PutChar(')'); 696a29cb0baSEnrico Granata } 697a29cb0baSEnrico Granata return true; 698a29cb0baSEnrico Granata } 699a29cb0baSEnrico Granata 7004d93b8cdSEnrico Granata void 7014d93b8cdSEnrico Granata ValueObjectPrinter::PrintChildrenIfNeeded (bool value_printed, 7024d93b8cdSEnrico Granata bool summary_printed) 7034d93b8cdSEnrico Granata { 7044d93b8cdSEnrico Granata // this flag controls whether we tried to display a description for this object and failed 7054d93b8cdSEnrico Granata // if that happens, we want to display the children, if any 7064d93b8cdSEnrico Granata bool is_failed_description = !PrintObjectDescriptionIfNeeded(value_printed, summary_printed); 7074d93b8cdSEnrico Granata 708c1b7c09aSEnrico Granata auto curr_ptr_depth = m_ptr_depth; 7094d93b8cdSEnrico Granata bool print_children = ShouldPrintChildren (is_failed_description,curr_ptr_depth); 710c1b7c09aSEnrico Granata bool print_oneline = (curr_ptr_depth.CanAllowExpansion() || 7113fa6dc90SEnrico Granata options.m_show_types || 7123fa6dc90SEnrico Granata !options.m_allow_oneliner_mode || 7133fa6dc90SEnrico Granata options.m_flat_output || 7143fa6dc90SEnrico Granata options.m_show_location) ? false : DataVisualization::ShouldPrintAsOneLiner(*m_valobj); 7154d93b8cdSEnrico Granata 7164d93b8cdSEnrico Granata if (print_children) 7174d93b8cdSEnrico Granata { 718a29cb0baSEnrico Granata if (print_oneline) 719a29cb0baSEnrico Granata { 720a29cb0baSEnrico Granata m_stream->PutChar(' '); 721a29cb0baSEnrico Granata PrintChildrenOneLiner (false); 722a29cb0baSEnrico Granata m_stream->EOL(); 723a29cb0baSEnrico Granata } 724a29cb0baSEnrico Granata else 725c1b7c09aSEnrico Granata PrintChildren (value_printed, summary_printed, curr_ptr_depth); 7264d93b8cdSEnrico Granata } 7274d93b8cdSEnrico Granata else if (m_curr_depth >= options.m_max_depth && IsAggregate() && ShouldPrintValueObject()) 7284d93b8cdSEnrico Granata { 7294d93b8cdSEnrico Granata m_stream->PutCString("{...}\n"); 7304d93b8cdSEnrico Granata } 731245b3caaSEnrico Granata else 732245b3caaSEnrico Granata m_stream->EOL(); 7334d93b8cdSEnrico Granata } 7340f883ffbSEnrico Granata 7350f883ffbSEnrico Granata bool 7360f883ffbSEnrico Granata ValueObjectPrinter::ShouldPrintValidation () 7370f883ffbSEnrico Granata { 7380f883ffbSEnrico Granata return options.m_run_validator; 7390f883ffbSEnrico Granata } 7400f883ffbSEnrico Granata 7410f883ffbSEnrico Granata bool 7420f883ffbSEnrico Granata ValueObjectPrinter::PrintValidationMarkerIfNeeded () 7430f883ffbSEnrico Granata { 7440f883ffbSEnrico Granata if (!ShouldPrintValidation()) 7450f883ffbSEnrico Granata return false; 7460f883ffbSEnrico Granata 7470f883ffbSEnrico Granata m_validation = m_valobj->GetValidationStatus(); 7480f883ffbSEnrico Granata 7490f883ffbSEnrico Granata if (TypeValidatorResult::Failure == m_validation.first) 7500f883ffbSEnrico Granata { 7510f883ffbSEnrico Granata m_stream->Printf("! "); 7520f883ffbSEnrico Granata return true; 7530f883ffbSEnrico Granata } 7540f883ffbSEnrico Granata 7550f883ffbSEnrico Granata return false; 7560f883ffbSEnrico Granata } 7570f883ffbSEnrico Granata 7580f883ffbSEnrico Granata bool 7590f883ffbSEnrico Granata ValueObjectPrinter::PrintValidationErrorIfNeeded () 7600f883ffbSEnrico Granata { 7610f883ffbSEnrico Granata if (!ShouldPrintValidation()) 7620f883ffbSEnrico Granata return false; 7630f883ffbSEnrico Granata 7640f883ffbSEnrico Granata if (TypeValidatorResult::Success == m_validation.first) 7650f883ffbSEnrico Granata return false; 7660f883ffbSEnrico Granata 7670f883ffbSEnrico Granata if (m_validation.second.empty()) 7680f883ffbSEnrico Granata m_validation.second.assign("unknown error"); 7690f883ffbSEnrico Granata 7700f883ffbSEnrico Granata m_stream->Printf(" ! validation error: %s", m_validation.second.c_str()); 7710f883ffbSEnrico Granata m_stream->EOL(); 7720f883ffbSEnrico Granata 7730f883ffbSEnrico Granata return true; 7740f883ffbSEnrico Granata } 775