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 18ebf7707eSSean Callanan #include "lldb/Interpreter/Args.h" 1930fdc8d8SChris Lattner #include "lldb/Core/Value.h" 206c68fb45SJim Ingham #include "lldb/Core/ValueObjectVariable.h" 214d93b8cdSEnrico Granata #include "lldb/DataFormatters/ValueObjectPrinter.h" 2230fdc8d8SChris Lattner #include "lldb/Expression/ClangExpressionVariable.h" 231a8d4093SSean Callanan #include "lldb/Expression/ClangUserExpression.h" 24ebb84b24SStephen Wilson #include "lldb/Expression/ClangFunction.h" 2530fdc8d8SChris Lattner #include "lldb/Expression/DWARFExpression.h" 2630fdc8d8SChris Lattner #include "lldb/Host/Host.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 { 63*9734280fSTodd Fiala { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "all-threads", 'a', OptionParser::eRequiredArgument, NULL, 0, eArgTypeBoolean, "Should we run all threads if the execution doesn't complete on one thread."}, 64*9734280fSTodd Fiala { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "ignore-breakpoints", 'i', OptionParser::eRequiredArgument, NULL, 0, eArgTypeBoolean, "Ignore breakpoint hits while running expressions"}, 65*9734280fSTodd Fiala { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "timeout", 't', OptionParser::eRequiredArgument, NULL, 0, eArgTypeUnsignedInteger, "Timeout value (in microseconds) for running the expression."}, 66*9734280fSTodd Fiala { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "unwind-on-error", 'u', OptionParser::eRequiredArgument, 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)."}, 67*9734280fSTodd Fiala { LLDB_OPT_SET_1 | LLDB_OPT_SET_2, false, "debug", 'g', OptionParser::eNoArgument , 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)."}, 68*9734280fSTodd Fiala { LLDB_OPT_SET_1, false, "description-verbosity", 'v', OptionParser::eOptionalArgument, 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; 12235e1bda6SJim Ingham result = Args::StringToUInt32(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, 199f9fc609fSGreg Clayton eFlagProcessMustBePaused | eFlagTryTargetAPILock), 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\ 22135e1bda6SJim Ingham Examples: \n\ 22230fdc8d8SChris Lattner \n\ 22330fdc8d8SChris Lattner expr my_struct->a = my_array[3] \n\ 22430fdc8d8SChris Lattner expr -f bin -- (index * 8) + 5 \n\ 2255c48d5c5SJim Ingham expr unsigned int $foo = 5\n\ 22630fdc8d8SChris Lattner expr char c[] = \"foo\"; c[0]\n"); 227405fe67fSCaroline Tice 228405fe67fSCaroline Tice CommandArgumentEntry arg; 229405fe67fSCaroline Tice CommandArgumentData expression_arg; 230405fe67fSCaroline Tice 231405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 232405fe67fSCaroline Tice expression_arg.arg_type = eArgTypeExpression; 233405fe67fSCaroline Tice expression_arg.arg_repetition = eArgRepeatPlain; 234405fe67fSCaroline Tice 235405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 236405fe67fSCaroline Tice arg.push_back (expression_arg); 237405fe67fSCaroline Tice 238405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 239405fe67fSCaroline Tice m_arguments.push_back (arg); 2401deb7962SGreg Clayton 2415009f9d5SGreg Clayton // Add the "--format" and "--gdb-format" 2425009f9d5SGreg Clayton m_option_group.Append (&m_format_options, OptionGroupFormat::OPTION_GROUP_FORMAT | OptionGroupFormat::OPTION_GROUP_GDB_FMT, LLDB_OPT_SET_1); 2431deb7962SGreg Clayton m_option_group.Append (&m_command_options); 244b576bba2SEnrico Granata m_option_group.Append (&m_varobj_options, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1 | LLDB_OPT_SET_2); 2451deb7962SGreg Clayton m_option_group.Finalize(); 24630fdc8d8SChris Lattner } 24730fdc8d8SChris Lattner 24830fdc8d8SChris Lattner CommandObjectExpression::~CommandObjectExpression () 24930fdc8d8SChris Lattner { 25030fdc8d8SChris Lattner } 25130fdc8d8SChris Lattner 25230fdc8d8SChris Lattner Options * 25330fdc8d8SChris Lattner CommandObjectExpression::GetOptions () 25430fdc8d8SChris Lattner { 2551deb7962SGreg Clayton return &m_option_group; 25630fdc8d8SChris Lattner } 25730fdc8d8SChris Lattner 25830fdc8d8SChris Lattner bool 2591d3afba3SGreg Clayton CommandObjectExpression::EvaluateExpression 2601d3afba3SGreg Clayton ( 2611d3afba3SGreg Clayton const char *expr, 2626e8dc334SCaroline Tice Stream *output_stream, 2636e8dc334SCaroline Tice Stream *error_stream, 2641d3afba3SGreg Clayton CommandReturnObject *result 2651d3afba3SGreg Clayton ) 26630fdc8d8SChris Lattner { 267ba7b8e2cSGreg Clayton // Don't use m_exe_ctx as this might be called asynchronously 268ba7b8e2cSGreg Clayton // after the command object DoExecute has finished when doing 269ba7b8e2cSGreg Clayton // multi-line expression that use an input reader... 270ba7b8e2cSGreg Clayton ExecutionContext exe_ctx (m_interpreter.GetExecutionContext()); 271ba7b8e2cSGreg Clayton 272ba7b8e2cSGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 273c0a6e061SSean Callanan 274c0a6e061SSean Callanan if (!target) 275c0a6e061SSean Callanan target = Host::GetDummyTarget(m_interpreter.GetDebugger()).get(); 276c0a6e061SSean Callanan 277c14ee32dSGreg Clayton if (target) 2786961e878SSean Callanan { 2798b2fe6dcSGreg Clayton lldb::ValueObjectSP result_valobj_sp; 2808b2fe6dcSGreg Clayton 28192adcac9SSean Callanan bool keep_in_memory = true; 28292adcac9SSean Callanan 28335e1bda6SJim Ingham EvaluateExpressionOptions options; 2846fbc48bcSJim Ingham options.SetCoerceToId(m_varobj_options.use_objc); 2856fbc48bcSJim Ingham options.SetUnwindOnError(m_command_options.unwind_on_error); 2866fbc48bcSJim Ingham options.SetIgnoreBreakpoints (m_command_options.ignore_breakpoints); 2876fbc48bcSJim Ingham options.SetKeepInMemory(keep_in_memory); 2886fbc48bcSJim Ingham options.SetUseDynamic(m_varobj_options.use_dynamic); 2896fbc48bcSJim Ingham options.SetTryAllThreads(m_command_options.try_all_threads); 2906fbc48bcSJim Ingham options.SetDebug(m_command_options.debug); 29162afb9f6SGreg Clayton 29223f8c95aSGreg Clayton // If there is any chance we are going to stop and want to see 29323f8c95aSGreg Clayton // what went wrong with our expression, we should generate debug info 29423f8c95aSGreg Clayton if (!m_command_options.ignore_breakpoints || 29523f8c95aSGreg Clayton !m_command_options.unwind_on_error) 29623f8c95aSGreg Clayton options.SetGenerateDebugInfo(true); 29723f8c95aSGreg Clayton 29862afb9f6SGreg Clayton if (m_command_options.timeout > 0) 29962afb9f6SGreg Clayton options.SetTimeoutUsec(m_command_options.timeout); 3006f78f386SJim Ingham else 3016f78f386SJim Ingham options.SetTimeoutUsec(0); 302d4439aa9SEnrico Granata 30362e5f4deSArnaud A. de Grandmaison target->EvaluateExpression(expr, exe_ctx.GetFramePtr(), 30462e5f4deSArnaud A. de Grandmaison result_valobj_sp, options); 3058b2fe6dcSGreg Clayton 3068b2fe6dcSGreg Clayton if (result_valobj_sp) 3078b2fe6dcSGreg Clayton { 308bf154daeSSean Callanan Format format = m_format_options.GetFormat(); 309bf154daeSSean Callanan 310b71f3844SGreg Clayton if (result_valobj_sp->GetError().Success()) 31130fdc8d8SChris Lattner { 312bf154daeSSean Callanan if (format != eFormatVoid) 313bf154daeSSean Callanan { 3141deb7962SGreg Clayton if (format != eFormatDefault) 3151deb7962SGreg Clayton result_valobj_sp->SetFormat (format); 31632c4085bSGreg Clayton 3174d93b8cdSEnrico Granata DumpValueObjectOptions options(m_varobj_options.GetAsDumpOptions(m_command_options.m_verbosity,format)); 318770eb05aSEnrico Granata 3194d93b8cdSEnrico Granata result_valobj_sp->Dump(*output_stream,options); 3204d93b8cdSEnrico Granata 321fcd43b71SJohnny Chen if (result) 322fcd43b71SJohnny Chen result->SetStatus (eReturnStatusSuccessFinishResult); 32330fdc8d8SChris Lattner } 324bf154daeSSean Callanan } 32530fdc8d8SChris Lattner else 32630fdc8d8SChris Lattner { 327bccce813SSean Callanan if (result_valobj_sp->GetError().GetError() == ClangUserExpression::kNoResult) 328bccce813SSean Callanan { 329bcf897faSSean Callanan if (format != eFormatVoid && m_interpreter.GetDebugger().GetNotifyVoid()) 330bf154daeSSean Callanan { 331bcf897faSSean Callanan error_stream->PutCString("(void)\n"); 332bcf897faSSean Callanan } 333bccce813SSean Callanan 334bccce813SSean Callanan if (result) 335bccce813SSean Callanan result->SetStatus (eReturnStatusSuccessFinishResult); 336bccce813SSean Callanan } 337bccce813SSean Callanan else 338bccce813SSean Callanan { 3395fd05903SGreg Clayton const char *error_cstr = result_valobj_sp->GetError().AsCString(); 3405fd05903SGreg Clayton if (error_cstr && error_cstr[0]) 3415fd05903SGreg Clayton { 342c7bece56SGreg Clayton const size_t error_cstr_len = strlen (error_cstr); 3435fd05903SGreg Clayton const bool ends_with_newline = error_cstr[error_cstr_len - 1] == '\n'; 3445fd05903SGreg Clayton if (strstr(error_cstr, "error:") != error_cstr) 3455fd05903SGreg Clayton error_stream->PutCString ("error: "); 3465fd05903SGreg Clayton error_stream->Write(error_cstr, error_cstr_len); 3475fd05903SGreg Clayton if (!ends_with_newline) 3485fd05903SGreg Clayton error_stream->EOL(); 3495fd05903SGreg Clayton } 3505fd05903SGreg Clayton else 3515fd05903SGreg Clayton { 3525fd05903SGreg Clayton error_stream->PutCString ("error: unknown error\n"); 3535fd05903SGreg Clayton } 3545fd05903SGreg Clayton 355fcd43b71SJohnny Chen if (result) 356b71f3844SGreg Clayton result->SetStatus (eReturnStatusFailed); 35730fdc8d8SChris Lattner } 3588b2fe6dcSGreg Clayton } 3598b2fe6dcSGreg Clayton } 360bccce813SSean Callanan } 3618b2fe6dcSGreg Clayton else 3628b2fe6dcSGreg Clayton { 3636e8dc334SCaroline Tice error_stream->Printf ("error: invalid execution context for expression\n"); 3648b2fe6dcSGreg Clayton return false; 3658b2fe6dcSGreg Clayton } 36630fdc8d8SChris Lattner 36716ad5faeSSean Callanan return true; 36830fdc8d8SChris Lattner } 36930fdc8d8SChris Lattner 37044d93782SGreg Clayton void 37144d93782SGreg Clayton CommandObjectExpression::IOHandlerInputComplete (IOHandler &io_handler, std::string &line) 37244d93782SGreg Clayton { 37344d93782SGreg Clayton io_handler.SetIsDone(true); 37444d93782SGreg Clayton // StreamSP output_stream = io_handler.GetDebugger().GetAsyncOutputStream(); 37544d93782SGreg Clayton // StreamSP error_stream = io_handler.GetDebugger().GetAsyncErrorStream(); 37644d93782SGreg Clayton StreamFileSP output_sp(io_handler.GetOutputStreamFile()); 37744d93782SGreg Clayton StreamFileSP error_sp(io_handler.GetErrorStreamFile()); 37844d93782SGreg Clayton 37944d93782SGreg Clayton EvaluateExpression (line.c_str(), 38044d93782SGreg Clayton output_sp.get(), 38144d93782SGreg Clayton error_sp.get()); 38244d93782SGreg Clayton if (output_sp) 38344d93782SGreg Clayton output_sp->Flush(); 38444d93782SGreg Clayton if (error_sp) 38544d93782SGreg Clayton error_sp->Flush(); 38644d93782SGreg Clayton } 38744d93782SGreg Clayton 38844d93782SGreg Clayton LineStatus 38944d93782SGreg Clayton CommandObjectExpression::IOHandlerLinesUpdated (IOHandler &io_handler, 39044d93782SGreg Clayton StringList &lines, 39144d93782SGreg Clayton uint32_t line_idx, 39244d93782SGreg Clayton Error &error) 39344d93782SGreg Clayton { 39444d93782SGreg Clayton if (line_idx == UINT32_MAX) 39544d93782SGreg Clayton { 39644d93782SGreg Clayton // Remove the last line from "lines" so it doesn't appear 39744d93782SGreg Clayton // in our final expression 39844d93782SGreg Clayton lines.PopBack(); 39944d93782SGreg Clayton error.Clear(); 40044d93782SGreg Clayton return LineStatus::Done; 40144d93782SGreg Clayton } 40244d93782SGreg Clayton else if (line_idx + 1 == lines.GetSize()) 40344d93782SGreg Clayton { 40444d93782SGreg Clayton // The last line was edited, if this line is empty, then we are done 40544d93782SGreg Clayton // getting our multiple lines. 40644d93782SGreg Clayton if (lines[line_idx].empty()) 40744d93782SGreg Clayton return LineStatus::Done; 40844d93782SGreg Clayton } 40944d93782SGreg Clayton return LineStatus::Success; 41044d93782SGreg Clayton } 41144d93782SGreg Clayton 412cf28a8b7SGreg Clayton void 413cf28a8b7SGreg Clayton CommandObjectExpression::GetMultilineExpression () 41430fdc8d8SChris Lattner { 41530fdc8d8SChris Lattner m_expr_lines.clear(); 41630fdc8d8SChris Lattner m_expr_line_count = 0; 41730fdc8d8SChris Lattner 41844d93782SGreg Clayton Debugger &debugger = GetCommandInterpreter().GetDebugger(); 41944d93782SGreg Clayton const bool multiple_lines = true; // Get multiple lines 42044d93782SGreg Clayton IOHandlerSP io_handler_sp (new IOHandlerEditline (debugger, 42144d93782SGreg Clayton "lldb-expr", // Name of input reader for history 42244d93782SGreg Clayton NULL, // No prompt 42344d93782SGreg Clayton multiple_lines, 424f6913cd7SGreg Clayton 1, // Show line numbers starting at 1 42544d93782SGreg Clayton *this)); 426b6892508SGreg Clayton 427b6892508SGreg Clayton StreamFileSP output_sp(io_handler_sp->GetOutputStreamFile()); 428b6892508SGreg Clayton if (output_sp) 429b6892508SGreg Clayton { 430b6892508SGreg Clayton output_sp->PutCString("Enter expressions, then terminate with an empty line to evaluate:\n"); 431b6892508SGreg Clayton output_sp->Flush(); 432b6892508SGreg Clayton } 43344d93782SGreg Clayton debugger.PushIOHandler(io_handler_sp); 434cf28a8b7SGreg Clayton } 435cf28a8b7SGreg Clayton 436cf28a8b7SGreg Clayton bool 437cf28a8b7SGreg Clayton CommandObjectExpression::DoExecute 438cf28a8b7SGreg Clayton ( 439cf28a8b7SGreg Clayton const char *command, 440cf28a8b7SGreg Clayton CommandReturnObject &result 441cf28a8b7SGreg Clayton ) 442cf28a8b7SGreg Clayton { 443cf28a8b7SGreg Clayton m_option_group.NotifyOptionParsingStarting(); 444cf28a8b7SGreg Clayton 445cf28a8b7SGreg Clayton const char * expr = NULL; 446cf28a8b7SGreg Clayton 447cf28a8b7SGreg Clayton if (command[0] == '\0') 448cf28a8b7SGreg Clayton { 449cf28a8b7SGreg Clayton GetMultilineExpression (); 45030fdc8d8SChris Lattner return result.Succeeded(); 45130fdc8d8SChris Lattner } 45230fdc8d8SChris Lattner 45330fdc8d8SChris Lattner if (command[0] == '-') 45430fdc8d8SChris Lattner { 45530fdc8d8SChris Lattner // We have some options and these options MUST end with --. 45630fdc8d8SChris Lattner const char *end_options = NULL; 45730fdc8d8SChris Lattner const char *s = command; 45830fdc8d8SChris Lattner while (s && s[0]) 45930fdc8d8SChris Lattner { 46030fdc8d8SChris Lattner end_options = ::strstr (s, "--"); 46130fdc8d8SChris Lattner if (end_options) 46230fdc8d8SChris Lattner { 46330fdc8d8SChris Lattner end_options += 2; // Get past the "--" 46430fdc8d8SChris Lattner if (::isspace (end_options[0])) 46530fdc8d8SChris Lattner { 46630fdc8d8SChris Lattner expr = end_options; 46730fdc8d8SChris Lattner while (::isspace (*expr)) 46830fdc8d8SChris Lattner ++expr; 46930fdc8d8SChris Lattner break; 47030fdc8d8SChris Lattner } 47130fdc8d8SChris Lattner } 47230fdc8d8SChris Lattner s = end_options; 47330fdc8d8SChris Lattner } 47430fdc8d8SChris Lattner 47530fdc8d8SChris Lattner if (end_options) 47630fdc8d8SChris Lattner { 47730fdc8d8SChris Lattner Args args (command, end_options - command); 478a7015092SGreg Clayton if (!ParseOptions (args, result)) 47930fdc8d8SChris Lattner return false; 480f6b8b581SGreg Clayton 4811deb7962SGreg Clayton Error error (m_option_group.NotifyOptionParsingFinished()); 482f6b8b581SGreg Clayton if (error.Fail()) 483f6b8b581SGreg Clayton { 484f6b8b581SGreg Clayton result.AppendError (error.AsCString()); 485f6b8b581SGreg Clayton result.SetStatus (eReturnStatusFailed); 486f6b8b581SGreg Clayton return false; 487f6b8b581SGreg Clayton } 488cf28a8b7SGreg Clayton 489cf28a8b7SGreg Clayton // No expression following options 49005da458cSGreg Clayton if (expr == NULL || expr[0] == '\0') 491cf28a8b7SGreg Clayton { 492cf28a8b7SGreg Clayton GetMultilineExpression (); 493cf28a8b7SGreg Clayton return result.Succeeded(); 494cf28a8b7SGreg Clayton } 49530fdc8d8SChris Lattner } 49630fdc8d8SChris Lattner } 49730fdc8d8SChris Lattner 49830fdc8d8SChris Lattner if (expr == NULL) 49930fdc8d8SChris Lattner expr = command; 50030fdc8d8SChris Lattner 5016e8dc334SCaroline Tice if (EvaluateExpression (expr, &(result.GetOutputStream()), &(result.GetErrorStream()), &result)) 502fcd43b71SJohnny Chen return true; 503fcd43b71SJohnny Chen 504fcd43b71SJohnny Chen result.SetStatus (eReturnStatusFailed); 505fcd43b71SJohnny Chen return false; 50630fdc8d8SChris Lattner } 50730fdc8d8SChris Lattner 508