130fdc8d8SChris Lattner //===-- CommandObjectExpression.cpp -----------------------------*- C++ -*-===//
230fdc8d8SChris Lattner //
330fdc8d8SChris Lattner //                     The LLVM Compiler Infrastructure
430fdc8d8SChris Lattner //
530fdc8d8SChris Lattner // This file is distributed under the University of Illinois Open Source
630fdc8d8SChris Lattner // License. See LICENSE.TXT for details.
730fdc8d8SChris Lattner //
830fdc8d8SChris Lattner //===----------------------------------------------------------------------===//
930fdc8d8SChris Lattner 
1093a64300SDaniel Malea #include "lldb/lldb-python.h"
1193a64300SDaniel Malea 
1230fdc8d8SChris Lattner #include "CommandObjectExpression.h"
1330fdc8d8SChris Lattner 
1430fdc8d8SChris Lattner // C Includes
1530fdc8d8SChris Lattner // C++ Includes
1630fdc8d8SChris Lattner // Other libraries and framework includes
1730fdc8d8SChris Lattner // Project includes
1830fdc8d8SChris Lattner #include "lldb/Core/Value.h"
196c68fb45SJim Ingham #include "lldb/Core/ValueObjectVariable.h"
204d93b8cdSEnrico Granata #include "lldb/DataFormatters/ValueObjectPrinter.h"
2130fdc8d8SChris Lattner #include "lldb/Expression/ClangExpressionVariable.h"
221a8d4093SSean Callanan #include "lldb/Expression/ClangUserExpression.h"
23ebb84b24SStephen Wilson #include "lldb/Expression/ClangFunction.h"
2430fdc8d8SChris Lattner #include "lldb/Expression/DWARFExpression.h"
2530fdc8d8SChris Lattner #include "lldb/Host/Host.h"
265275aaa0SVince Harron #include "lldb/Host/StringConvert.h"
27ebf7707eSSean Callanan #include "lldb/Core/Debugger.h"
286611103cSGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
2930fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h"
306c68fb45SJim Ingham #include "lldb/Target/ObjCLanguageRuntime.h"
3130fdc8d8SChris Lattner #include "lldb/Symbol/ObjectFile.h"
3230fdc8d8SChris Lattner #include "lldb/Symbol/Variable.h"
3330fdc8d8SChris Lattner #include "lldb/Target/Process.h"
34b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h"
3530fdc8d8SChris Lattner #include "lldb/Target/Target.h"
367260f620SGreg Clayton #include "lldb/Target/Thread.h"
3728606954SSaleem Abdulrasool #include "llvm/ADT/STLExtras.h"
38ebf7707eSSean Callanan #include "llvm/ADT/StringRef.h"
3930fdc8d8SChris Lattner 
4030fdc8d8SChris Lattner using namespace lldb;
4130fdc8d8SChris Lattner using namespace lldb_private;
4230fdc8d8SChris Lattner 
431deb7962SGreg Clayton CommandObjectExpression::CommandOptions::CommandOptions () :
441deb7962SGreg Clayton     OptionGroup()
4530fdc8d8SChris Lattner {
4630fdc8d8SChris Lattner }
4730fdc8d8SChris Lattner 
4830fdc8d8SChris Lattner 
4930fdc8d8SChris Lattner CommandObjectExpression::CommandOptions::~CommandOptions ()
5030fdc8d8SChris Lattner {
5130fdc8d8SChris Lattner }
5230fdc8d8SChris Lattner 
534d93b8cdSEnrico Granata static OptionEnumValueElement g_description_verbosity_type[] =
544d93b8cdSEnrico Granata {
554d93b8cdSEnrico Granata     { eLanguageRuntimeDescriptionDisplayVerbosityCompact,      "compact",       "Only show the description string"},
564d93b8cdSEnrico Granata     { eLanguageRuntimeDescriptionDisplayVerbosityFull,         "full",          "Show the full output, including persistent variable's name and type"},
574d93b8cdSEnrico Granata     { 0, NULL, NULL }
584d93b8cdSEnrico Granata };
594d93b8cdSEnrico Granata 
601deb7962SGreg Clayton OptionDefinition
611deb7962SGreg Clayton CommandObjectExpression::CommandOptions::g_option_table[] =
621deb7962SGreg Clayton {
63d37221dcSZachary Turner     { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "all-threads",        'a', OptionParser::eRequiredArgument, NULL, NULL, 0, eArgTypeBoolean,    "Should we run all threads if the execution doesn't complete on one thread."},
64d37221dcSZachary Turner     { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "ignore-breakpoints", 'i', OptionParser::eRequiredArgument, NULL, NULL, 0, eArgTypeBoolean,    "Ignore breakpoint hits while running expressions"},
65d37221dcSZachary Turner     { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "timeout",            't', OptionParser::eRequiredArgument, NULL, NULL, 0, eArgTypeUnsignedInteger,  "Timeout value (in microseconds) for running the expression."},
66d37221dcSZachary Turner     { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "unwind-on-error",    'u', OptionParser::eRequiredArgument, NULL, NULL, 0, eArgTypeBoolean,    "Clean up program state if the expression causes a crash, or raises a signal.  Note, unlike gdb hitting a breakpoint is controlled by another option (-i)."},
67d37221dcSZachary Turner     { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "debug",              'g', OptionParser::eNoArgument      , NULL, NULL, 0, eArgTypeNone,       "When specified, debug the JIT code by setting a breakpoint on the first instruction and forcing breakpoints to not be ignored (-i0) and no unwinding to happen on error (-u0)."},
68d37221dcSZachary Turner     { LLDB_OPT_SET_1, false, "description-verbosity", 'v', OptionParser::eOptionalArgument, NULL, g_description_verbosity_type, 0, eArgTypeDescriptionVerbosity,        "How verbose should the output of this expression be, if the object description is asked for."},
691deb7962SGreg Clayton };
701deb7962SGreg Clayton 
711deb7962SGreg Clayton 
721deb7962SGreg Clayton uint32_t
731deb7962SGreg Clayton CommandObjectExpression::CommandOptions::GetNumDefinitions ()
741deb7962SGreg Clayton {
7528606954SSaleem Abdulrasool     return llvm::array_lengthof(g_option_table);
761deb7962SGreg Clayton }
771deb7962SGreg Clayton 
7830fdc8d8SChris Lattner Error
791deb7962SGreg Clayton CommandObjectExpression::CommandOptions::SetOptionValue (CommandInterpreter &interpreter,
801deb7962SGreg Clayton                                                          uint32_t option_idx,
811deb7962SGreg Clayton                                                          const char *option_arg)
8230fdc8d8SChris Lattner {
8330fdc8d8SChris Lattner     Error error;
8430fdc8d8SChris Lattner 
853bcdfc0eSGreg Clayton     const int short_option = g_option_table[option_idx].short_option;
8630fdc8d8SChris Lattner 
8730fdc8d8SChris Lattner     switch (short_option)
8830fdc8d8SChris Lattner     {
893f4c09c1SCaroline Tice       //case 'l':
903f4c09c1SCaroline Tice       //if (language.SetLanguageFromCString (option_arg) == false)
913f4c09c1SCaroline Tice       //{
9286edbf41SGreg Clayton       //    error.SetErrorStringWithFormat("invalid language option argument '%s'", option_arg);
933f4c09c1SCaroline Tice       //}
943f4c09c1SCaroline Tice       //break;
9530fdc8d8SChris Lattner 
9635e1bda6SJim Ingham     case 'a':
9735e1bda6SJim Ingham         {
9835e1bda6SJim Ingham             bool success;
9935e1bda6SJim Ingham             bool result;
10035e1bda6SJim Ingham             result = Args::StringToBoolean(option_arg, true, &success);
10135e1bda6SJim Ingham             if (!success)
10235e1bda6SJim Ingham                 error.SetErrorStringWithFormat("invalid all-threads value setting: \"%s\"", option_arg);
10335e1bda6SJim Ingham             else
10435e1bda6SJim Ingham                 try_all_threads = result;
10535e1bda6SJim Ingham         }
1066c68fb45SJim Ingham         break;
1076c68fb45SJim Ingham 
108184e9811SJim Ingham     case 'i':
109184e9811SJim Ingham         {
110184e9811SJim Ingham             bool success;
111184e9811SJim Ingham             bool tmp_value = Args::StringToBoolean(option_arg, true, &success);
112184e9811SJim Ingham             if (success)
113184e9811SJim Ingham                 ignore_breakpoints = tmp_value;
114184e9811SJim Ingham             else
115184e9811SJim Ingham                 error.SetErrorStringWithFormat("could not convert \"%s\" to a boolean value.", option_arg);
116184e9811SJim Ingham             break;
117184e9811SJim Ingham         }
11835e1bda6SJim Ingham     case 't':
11935e1bda6SJim Ingham         {
12035e1bda6SJim Ingham             bool success;
12135e1bda6SJim Ingham             uint32_t result;
1225275aaa0SVince Harron             result = StringConvert::ToUInt32(option_arg, 0, 0, &success);
12335e1bda6SJim Ingham             if (success)
12435e1bda6SJim Ingham                 timeout = result;
12535e1bda6SJim Ingham             else
12635e1bda6SJim Ingham                 error.SetErrorStringWithFormat ("invalid timeout setting \"%s\"", option_arg);
12735e1bda6SJim Ingham         }
12835e1bda6SJim Ingham         break;
12935e1bda6SJim Ingham 
130399f1cafSJim Ingham     case 'u':
1313bfdaa2aSSean Callanan         {
132399f1cafSJim Ingham             bool success;
133184e9811SJim Ingham             bool tmp_value = Args::StringToBoolean(option_arg, true, &success);
134184e9811SJim Ingham             if (success)
135184e9811SJim Ingham                 unwind_on_error = tmp_value;
136184e9811SJim Ingham             else
13786edbf41SGreg Clayton                 error.SetErrorStringWithFormat("could not convert \"%s\" to a boolean value.", option_arg);
138399f1cafSJim Ingham             break;
1393bfdaa2aSSean Callanan         }
1404d93b8cdSEnrico Granata 
1414d93b8cdSEnrico Granata     case 'v':
1424d93b8cdSEnrico Granata         if (!option_arg)
1434d93b8cdSEnrico Granata         {
1444d93b8cdSEnrico Granata             m_verbosity = eLanguageRuntimeDescriptionDisplayVerbosityFull;
1454d93b8cdSEnrico Granata             break;
1464d93b8cdSEnrico Granata         }
1474d93b8cdSEnrico Granata         m_verbosity = (LanguageRuntimeDescriptionDisplayVerbosity) Args::StringToOptionEnum(option_arg, g_option_table[option_idx].enum_values, 0, error);
1484d93b8cdSEnrico Granata         if (!error.Success())
1494d93b8cdSEnrico Granata             error.SetErrorStringWithFormat ("unrecognized value for description-verbosity '%s'", option_arg);
1504d93b8cdSEnrico Granata         break;
1514d93b8cdSEnrico Granata 
15262afb9f6SGreg Clayton     case 'g':
15362afb9f6SGreg Clayton         debug = true;
15462afb9f6SGreg Clayton         unwind_on_error = false;
15562afb9f6SGreg Clayton         ignore_breakpoints = false;
15662afb9f6SGreg Clayton         break;
15762afb9f6SGreg Clayton 
15830fdc8d8SChris Lattner     default:
15986edbf41SGreg Clayton         error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
16030fdc8d8SChris Lattner         break;
16130fdc8d8SChris Lattner     }
16230fdc8d8SChris Lattner 
16330fdc8d8SChris Lattner     return error;
16430fdc8d8SChris Lattner }
16530fdc8d8SChris Lattner 
16630fdc8d8SChris Lattner void
1671deb7962SGreg Clayton CommandObjectExpression::CommandOptions::OptionParsingStarting (CommandInterpreter &interpreter)
16830fdc8d8SChris Lattner {
169184e9811SJim Ingham     Process *process = interpreter.GetExecutionContext().GetProcessPtr();
170184e9811SJim Ingham     if (process != NULL)
171184e9811SJim Ingham     {
172184e9811SJim Ingham         ignore_breakpoints = process->GetIgnoreBreakpointsInExpressions();
173184e9811SJim Ingham         unwind_on_error    = process->GetUnwindOnErrorInExpressions();
174184e9811SJim Ingham     }
175184e9811SJim Ingham     else
176184e9811SJim Ingham     {
177fc03f8fcSGreg Clayton         ignore_breakpoints = true;
178399f1cafSJim Ingham         unwind_on_error = true;
179184e9811SJim Ingham     }
180184e9811SJim Ingham 
18130fdc8d8SChris Lattner     show_summary = true;
18235e1bda6SJim Ingham     try_all_threads = true;
18335e1bda6SJim Ingham     timeout = 0;
18462afb9f6SGreg Clayton     debug = false;
1854d93b8cdSEnrico Granata     m_verbosity = eLanguageRuntimeDescriptionDisplayVerbosityCompact;
18630fdc8d8SChris Lattner }
18730fdc8d8SChris Lattner 
188e0d378b3SGreg Clayton const OptionDefinition*
18930fdc8d8SChris Lattner CommandObjectExpression::CommandOptions::GetDefinitions ()
19030fdc8d8SChris Lattner {
19130fdc8d8SChris Lattner     return g_option_table;
19230fdc8d8SChris Lattner }
19330fdc8d8SChris Lattner 
194a7015092SGreg Clayton CommandObjectExpression::CommandObjectExpression (CommandInterpreter &interpreter) :
1955a988416SJim Ingham     CommandObjectRaw (interpreter,
19630fdc8d8SChris Lattner                       "expression",
1975c48d5c5SJim Ingham                       "Evaluate a C/ObjC/C++ expression in the current program context, using user defined variables and variables currently in scope.",
1985a988416SJim Ingham                       NULL,
199*e87764f2SEnrico Granata                       eCommandProcessMustBePaused | eCommandTryTargetAPILock),
20044d93782SGreg Clayton     IOHandlerDelegate (IOHandlerDelegate::Completion::Expression),
2011deb7962SGreg Clayton     m_option_group (interpreter),
2021deb7962SGreg Clayton     m_format_options (eFormatDefault),
2031deb7962SGreg Clayton     m_command_options (),
20430fdc8d8SChris Lattner     m_expr_line_count (0),
20530fdc8d8SChris Lattner     m_expr_lines ()
20630fdc8d8SChris Lattner {
20730fdc8d8SChris Lattner   SetHelpLong(
20835e1bda6SJim Ingham "Timeouts:\n\
209d93c4a33SBruce Mitchener     If the expression can be evaluated statically (without running code) then it will be.\n\
21035e1bda6SJim Ingham     Otherwise, by default the expression will run on the current thread with a short timeout:\n\
21135e1bda6SJim Ingham     currently .25 seconds.  If it doesn't return in that time, the evaluation will be interrupted\n\
21235e1bda6SJim Ingham     and resumed with all threads running.  You can use the -a option to disable retrying on all\n\
21335e1bda6SJim Ingham     threads.  You can use the -t option to set a shorter timeout.\n\
2145c48d5c5SJim Ingham \n\
2155c48d5c5SJim Ingham User defined variables:\n\
2165c48d5c5SJim Ingham     You can define your own variables for convenience or to be used in subsequent expressions.\n\
2175c48d5c5SJim Ingham     You define them the same way you would define variables in C.  If the first character of \n\
2185c48d5c5SJim Ingham     your user defined variable is a $, then the variable's value will be available in future\n\
2195c48d5c5SJim Ingham     expressions, otherwise it will just be available in the current expression.\n\
2205c48d5c5SJim Ingham \n\
221c60b6eadSJim Ingham \n\
222c60b6eadSJim Ingham Continuing evaluation after a breakpoint:\n\
223c60b6eadSJim Ingham     If the \"-i false\" option is used, and execution is interrupted by a breakpoint hit, once\n\
224c60b6eadSJim Ingham     you are done with your investigation, you can either remove the expression execution frames\n\
225c60b6eadSJim Ingham     from the stack with \"thread return -x\" or if you are still interested in the expression result\n\
226c60b6eadSJim Ingham     you can issue the \"continue\" command and the expression evaluation will complete and the\n\
227c60b6eadSJim Ingham     expression result will be available using the \"thread.completed-expression\" key in the thread\n\
228c60b6eadSJim Ingham     format.\n\
229c60b6eadSJim Ingham \n\
23035e1bda6SJim Ingham Examples: \n\
23130fdc8d8SChris Lattner \n\
23230fdc8d8SChris Lattner    expr my_struct->a = my_array[3] \n\
23330fdc8d8SChris Lattner    expr -f bin -- (index * 8) + 5 \n\
2345c48d5c5SJim Ingham    expr unsigned int $foo = 5\n\
23530fdc8d8SChris Lattner    expr char c[] = \"foo\"; c[0]\n");
236405fe67fSCaroline Tice 
237405fe67fSCaroline Tice     CommandArgumentEntry arg;
238405fe67fSCaroline Tice     CommandArgumentData expression_arg;
239405fe67fSCaroline Tice 
240405fe67fSCaroline Tice     // Define the first (and only) variant of this arg.
241405fe67fSCaroline Tice     expression_arg.arg_type = eArgTypeExpression;
242405fe67fSCaroline Tice     expression_arg.arg_repetition = eArgRepeatPlain;
243405fe67fSCaroline Tice 
244405fe67fSCaroline Tice     // There is only one variant this argument could be; put it into the argument entry.
245405fe67fSCaroline Tice     arg.push_back (expression_arg);
246405fe67fSCaroline Tice 
247405fe67fSCaroline Tice     // Push the data for the first argument into the m_arguments vector.
248405fe67fSCaroline Tice     m_arguments.push_back (arg);
2491deb7962SGreg Clayton 
2505009f9d5SGreg Clayton     // Add the "--format" and "--gdb-format"
2515009f9d5SGreg Clayton     m_option_group.Append (&m_format_options, OptionGroupFormat::OPTION_GROUP_FORMAT | OptionGroupFormat::OPTION_GROUP_GDB_FMT, LLDB_OPT_SET_1);
2521deb7962SGreg Clayton     m_option_group.Append (&m_command_options);
253b576bba2SEnrico Granata     m_option_group.Append (&m_varobj_options, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1 | LLDB_OPT_SET_2);
2541deb7962SGreg Clayton     m_option_group.Finalize();
25530fdc8d8SChris Lattner }
25630fdc8d8SChris Lattner 
25730fdc8d8SChris Lattner CommandObjectExpression::~CommandObjectExpression ()
25830fdc8d8SChris Lattner {
25930fdc8d8SChris Lattner }
26030fdc8d8SChris Lattner 
26130fdc8d8SChris Lattner Options *
26230fdc8d8SChris Lattner CommandObjectExpression::GetOptions ()
26330fdc8d8SChris Lattner {
2641deb7962SGreg Clayton     return &m_option_group;
26530fdc8d8SChris Lattner }
26630fdc8d8SChris Lattner 
26730fdc8d8SChris Lattner bool
2681d3afba3SGreg Clayton CommandObjectExpression::EvaluateExpression
2691d3afba3SGreg Clayton (
2701d3afba3SGreg Clayton     const char *expr,
2716e8dc334SCaroline Tice     Stream *output_stream,
2726e8dc334SCaroline Tice     Stream *error_stream,
2731d3afba3SGreg Clayton     CommandReturnObject *result
2741d3afba3SGreg Clayton )
27530fdc8d8SChris Lattner {
276ba7b8e2cSGreg Clayton     // Don't use m_exe_ctx as this might be called asynchronously
277ba7b8e2cSGreg Clayton     // after the command object DoExecute has finished when doing
278ba7b8e2cSGreg Clayton     // multi-line expression that use an input reader...
279ba7b8e2cSGreg Clayton     ExecutionContext exe_ctx (m_interpreter.GetExecutionContext());
280ba7b8e2cSGreg Clayton 
281ba7b8e2cSGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
282c0a6e061SSean Callanan 
283c0a6e061SSean Callanan     if (!target)
284893c932aSJim Ingham         target = GetDummyTarget();
285c0a6e061SSean Callanan 
286c14ee32dSGreg Clayton     if (target)
2876961e878SSean Callanan     {
2888b2fe6dcSGreg Clayton         lldb::ValueObjectSP result_valobj_sp;
2898b2fe6dcSGreg Clayton 
29092adcac9SSean Callanan         bool keep_in_memory = true;
29192adcac9SSean Callanan 
29235e1bda6SJim Ingham         EvaluateExpressionOptions options;
2936fbc48bcSJim Ingham         options.SetCoerceToId(m_varobj_options.use_objc);
2946fbc48bcSJim Ingham         options.SetUnwindOnError(m_command_options.unwind_on_error);
2956fbc48bcSJim Ingham         options.SetIgnoreBreakpoints (m_command_options.ignore_breakpoints);
2966fbc48bcSJim Ingham         options.SetKeepInMemory(keep_in_memory);
2976fbc48bcSJim Ingham         options.SetUseDynamic(m_varobj_options.use_dynamic);
2986fbc48bcSJim Ingham         options.SetTryAllThreads(m_command_options.try_all_threads);
2996fbc48bcSJim Ingham         options.SetDebug(m_command_options.debug);
30062afb9f6SGreg Clayton 
30123f8c95aSGreg Clayton         // If there is any chance we are going to stop and want to see
30223f8c95aSGreg Clayton         // what went wrong with our expression, we should generate debug info
30323f8c95aSGreg Clayton         if (!m_command_options.ignore_breakpoints ||
30423f8c95aSGreg Clayton             !m_command_options.unwind_on_error)
30523f8c95aSGreg Clayton             options.SetGenerateDebugInfo(true);
30623f8c95aSGreg Clayton 
30762afb9f6SGreg Clayton         if (m_command_options.timeout > 0)
30862afb9f6SGreg Clayton             options.SetTimeoutUsec(m_command_options.timeout);
3096f78f386SJim Ingham         else
3106f78f386SJim Ingham             options.SetTimeoutUsec(0);
311d4439aa9SEnrico Granata 
31262e5f4deSArnaud A. de Grandmaison         target->EvaluateExpression(expr, exe_ctx.GetFramePtr(),
31362e5f4deSArnaud A. de Grandmaison                                    result_valobj_sp, options);
3148b2fe6dcSGreg Clayton 
3158b2fe6dcSGreg Clayton         if (result_valobj_sp)
3168b2fe6dcSGreg Clayton         {
317bf154daeSSean Callanan             Format format = m_format_options.GetFormat();
318bf154daeSSean Callanan 
319b71f3844SGreg Clayton             if (result_valobj_sp->GetError().Success())
32030fdc8d8SChris Lattner             {
321bf154daeSSean Callanan                 if (format != eFormatVoid)
322bf154daeSSean Callanan                 {
3231deb7962SGreg Clayton                     if (format != eFormatDefault)
3241deb7962SGreg Clayton                         result_valobj_sp->SetFormat (format);
32532c4085bSGreg Clayton 
3264d93b8cdSEnrico Granata                     DumpValueObjectOptions options(m_varobj_options.GetAsDumpOptions(m_command_options.m_verbosity,format));
327770eb05aSEnrico Granata 
3284d93b8cdSEnrico Granata                     result_valobj_sp->Dump(*output_stream,options);
3294d93b8cdSEnrico Granata 
330fcd43b71SJohnny Chen                     if (result)
331fcd43b71SJohnny Chen                         result->SetStatus (eReturnStatusSuccessFinishResult);
33230fdc8d8SChris Lattner                 }
333bf154daeSSean Callanan             }
33430fdc8d8SChris Lattner             else
33530fdc8d8SChris Lattner             {
336bccce813SSean Callanan                 if (result_valobj_sp->GetError().GetError() == ClangUserExpression::kNoResult)
337bccce813SSean Callanan                 {
338bcf897faSSean Callanan                     if (format != eFormatVoid && m_interpreter.GetDebugger().GetNotifyVoid())
339bf154daeSSean Callanan                     {
340bcf897faSSean Callanan                         error_stream->PutCString("(void)\n");
341bcf897faSSean Callanan                     }
342bccce813SSean Callanan 
343bccce813SSean Callanan                     if (result)
344bccce813SSean Callanan                         result->SetStatus (eReturnStatusSuccessFinishResult);
345bccce813SSean Callanan                 }
346bccce813SSean Callanan                 else
347bccce813SSean Callanan                 {
3485fd05903SGreg Clayton                     const char *error_cstr = result_valobj_sp->GetError().AsCString();
3495fd05903SGreg Clayton                     if (error_cstr && error_cstr[0])
3505fd05903SGreg Clayton                     {
351c7bece56SGreg Clayton                         const size_t error_cstr_len = strlen (error_cstr);
3525fd05903SGreg Clayton                         const bool ends_with_newline = error_cstr[error_cstr_len - 1] == '\n';
3535fd05903SGreg Clayton                         if (strstr(error_cstr, "error:") != error_cstr)
3545fd05903SGreg Clayton                             error_stream->PutCString ("error: ");
3555fd05903SGreg Clayton                         error_stream->Write(error_cstr, error_cstr_len);
3565fd05903SGreg Clayton                         if (!ends_with_newline)
3575fd05903SGreg Clayton                             error_stream->EOL();
3585fd05903SGreg Clayton                     }
3595fd05903SGreg Clayton                     else
3605fd05903SGreg Clayton                     {
3615fd05903SGreg Clayton                         error_stream->PutCString ("error: unknown error\n");
3625fd05903SGreg Clayton                     }
3635fd05903SGreg Clayton 
364fcd43b71SJohnny Chen                     if (result)
365b71f3844SGreg Clayton                         result->SetStatus (eReturnStatusFailed);
36630fdc8d8SChris Lattner                 }
3678b2fe6dcSGreg Clayton             }
3688b2fe6dcSGreg Clayton         }
369bccce813SSean Callanan     }
3708b2fe6dcSGreg Clayton     else
3718b2fe6dcSGreg Clayton     {
3726e8dc334SCaroline Tice         error_stream->Printf ("error: invalid execution context for expression\n");
3738b2fe6dcSGreg Clayton         return false;
3748b2fe6dcSGreg Clayton     }
37530fdc8d8SChris Lattner 
37616ad5faeSSean Callanan     return true;
37730fdc8d8SChris Lattner }
37830fdc8d8SChris Lattner 
37944d93782SGreg Clayton void
38044d93782SGreg Clayton CommandObjectExpression::IOHandlerInputComplete (IOHandler &io_handler, std::string &line)
38144d93782SGreg Clayton {
38244d93782SGreg Clayton     io_handler.SetIsDone(true);
38344d93782SGreg Clayton //    StreamSP output_stream = io_handler.GetDebugger().GetAsyncOutputStream();
38444d93782SGreg Clayton //    StreamSP error_stream = io_handler.GetDebugger().GetAsyncErrorStream();
38544d93782SGreg Clayton     StreamFileSP output_sp(io_handler.GetOutputStreamFile());
38644d93782SGreg Clayton     StreamFileSP error_sp(io_handler.GetErrorStreamFile());
38744d93782SGreg Clayton 
38844d93782SGreg Clayton     EvaluateExpression (line.c_str(),
38944d93782SGreg Clayton                         output_sp.get(),
39044d93782SGreg Clayton                         error_sp.get());
39144d93782SGreg Clayton     if (output_sp)
39244d93782SGreg Clayton         output_sp->Flush();
39344d93782SGreg Clayton     if (error_sp)
39444d93782SGreg Clayton         error_sp->Flush();
39544d93782SGreg Clayton }
39644d93782SGreg Clayton 
39744d93782SGreg Clayton LineStatus
39844d93782SGreg Clayton CommandObjectExpression::IOHandlerLinesUpdated (IOHandler &io_handler,
39944d93782SGreg Clayton                                                 StringList &lines,
40044d93782SGreg Clayton                                                 uint32_t line_idx,
40144d93782SGreg Clayton                                                 Error &error)
40244d93782SGreg Clayton {
40344d93782SGreg Clayton     if (line_idx == UINT32_MAX)
40444d93782SGreg Clayton     {
40544d93782SGreg Clayton         // Remove the last line from "lines" so it doesn't appear
40644d93782SGreg Clayton         // in our final expression
40744d93782SGreg Clayton         lines.PopBack();
40844d93782SGreg Clayton         error.Clear();
40944d93782SGreg Clayton         return LineStatus::Done;
41044d93782SGreg Clayton     }
41144d93782SGreg Clayton     else if (line_idx + 1 == lines.GetSize())
41244d93782SGreg Clayton     {
41344d93782SGreg Clayton         // The last line was edited, if this line is empty, then we are done
41444d93782SGreg Clayton         // getting our multiple lines.
41544d93782SGreg Clayton         if (lines[line_idx].empty())
41644d93782SGreg Clayton             return LineStatus::Done;
41744d93782SGreg Clayton     }
41844d93782SGreg Clayton     return LineStatus::Success;
41944d93782SGreg Clayton }
42044d93782SGreg Clayton 
421cf28a8b7SGreg Clayton void
422cf28a8b7SGreg Clayton CommandObjectExpression::GetMultilineExpression ()
42330fdc8d8SChris Lattner {
42430fdc8d8SChris Lattner     m_expr_lines.clear();
42530fdc8d8SChris Lattner     m_expr_line_count = 0;
42630fdc8d8SChris Lattner 
42744d93782SGreg Clayton     Debugger &debugger = GetCommandInterpreter().GetDebugger();
428e30f11d9SKate Stone     bool color_prompt = debugger.GetUseColor();
42944d93782SGreg Clayton     const bool multiple_lines = true; // Get multiple lines
43044d93782SGreg Clayton     IOHandlerSP io_handler_sp (new IOHandlerEditline (debugger,
431e30f11d9SKate Stone                                                       IOHandler::Type::Expression,
43244d93782SGreg Clayton                                                       "lldb-expr",      // Name of input reader for history
43344d93782SGreg Clayton                                                       NULL,             // No prompt
434e30f11d9SKate Stone                                                       NULL,             // Continuation prompt
43544d93782SGreg Clayton                                                       multiple_lines,
436e30f11d9SKate Stone                                                       color_prompt,
437f6913cd7SGreg Clayton                                                       1,                // Show line numbers starting at 1
43844d93782SGreg Clayton                                                       *this));
439b6892508SGreg Clayton 
440b6892508SGreg Clayton     StreamFileSP output_sp(io_handler_sp->GetOutputStreamFile());
441b6892508SGreg Clayton     if (output_sp)
442b6892508SGreg Clayton     {
443b6892508SGreg Clayton         output_sp->PutCString("Enter expressions, then terminate with an empty line to evaluate:\n");
444b6892508SGreg Clayton         output_sp->Flush();
445b6892508SGreg Clayton     }
44644d93782SGreg Clayton     debugger.PushIOHandler(io_handler_sp);
447cf28a8b7SGreg Clayton }
448cf28a8b7SGreg Clayton 
449cf28a8b7SGreg Clayton bool
450cf28a8b7SGreg Clayton CommandObjectExpression::DoExecute
451cf28a8b7SGreg Clayton (
452cf28a8b7SGreg Clayton     const char *command,
453cf28a8b7SGreg Clayton     CommandReturnObject &result
454cf28a8b7SGreg Clayton )
455cf28a8b7SGreg Clayton {
456cf28a8b7SGreg Clayton     m_option_group.NotifyOptionParsingStarting();
457cf28a8b7SGreg Clayton 
458cf28a8b7SGreg Clayton     const char * expr = NULL;
459cf28a8b7SGreg Clayton 
460cf28a8b7SGreg Clayton     if (command[0] == '\0')
461cf28a8b7SGreg Clayton     {
462cf28a8b7SGreg Clayton         GetMultilineExpression ();
46330fdc8d8SChris Lattner         return result.Succeeded();
46430fdc8d8SChris Lattner     }
46530fdc8d8SChris Lattner 
46630fdc8d8SChris Lattner     if (command[0] == '-')
46730fdc8d8SChris Lattner     {
46830fdc8d8SChris Lattner         // We have some options and these options MUST end with --.
46930fdc8d8SChris Lattner         const char *end_options = NULL;
47030fdc8d8SChris Lattner         const char *s = command;
47130fdc8d8SChris Lattner         while (s && s[0])
47230fdc8d8SChris Lattner         {
47330fdc8d8SChris Lattner             end_options = ::strstr (s, "--");
47430fdc8d8SChris Lattner             if (end_options)
47530fdc8d8SChris Lattner             {
47630fdc8d8SChris Lattner                 end_options += 2; // Get past the "--"
47730fdc8d8SChris Lattner                 if (::isspace (end_options[0]))
47830fdc8d8SChris Lattner                 {
47930fdc8d8SChris Lattner                     expr = end_options;
48030fdc8d8SChris Lattner                     while (::isspace (*expr))
48130fdc8d8SChris Lattner                         ++expr;
48230fdc8d8SChris Lattner                     break;
48330fdc8d8SChris Lattner                 }
48430fdc8d8SChris Lattner             }
48530fdc8d8SChris Lattner             s = end_options;
48630fdc8d8SChris Lattner         }
48730fdc8d8SChris Lattner 
48830fdc8d8SChris Lattner         if (end_options)
48930fdc8d8SChris Lattner         {
49000b7f95bSPavel Labath             Args args (llvm::StringRef(command, end_options - command));
491a7015092SGreg Clayton             if (!ParseOptions (args, result))
49230fdc8d8SChris Lattner                 return false;
493f6b8b581SGreg Clayton 
4941deb7962SGreg Clayton             Error error (m_option_group.NotifyOptionParsingFinished());
495f6b8b581SGreg Clayton             if (error.Fail())
496f6b8b581SGreg Clayton             {
497f6b8b581SGreg Clayton                 result.AppendError (error.AsCString());
498f6b8b581SGreg Clayton                 result.SetStatus (eReturnStatusFailed);
499f6b8b581SGreg Clayton                 return false;
500f6b8b581SGreg Clayton             }
501cf28a8b7SGreg Clayton 
502cf28a8b7SGreg Clayton             // No expression following options
50305da458cSGreg Clayton             if (expr == NULL || expr[0] == '\0')
504cf28a8b7SGreg Clayton             {
505cf28a8b7SGreg Clayton                 GetMultilineExpression ();
506cf28a8b7SGreg Clayton                 return result.Succeeded();
507cf28a8b7SGreg Clayton             }
50830fdc8d8SChris Lattner         }
50930fdc8d8SChris Lattner     }
51030fdc8d8SChris Lattner 
51130fdc8d8SChris Lattner     if (expr == NULL)
51230fdc8d8SChris Lattner         expr = command;
51330fdc8d8SChris Lattner 
5146e8dc334SCaroline Tice     if (EvaluateExpression (expr, &(result.GetOutputStream()), &(result.GetErrorStream()), &result))
515fcd43b71SJohnny Chen         return true;
516fcd43b71SJohnny Chen 
517fcd43b71SJohnny Chen     result.SetStatus (eReturnStatusFailed);
518fcd43b71SJohnny Chen     return false;
51930fdc8d8SChris Lattner }
52030fdc8d8SChris Lattner 
521