130fdc8d8SChris Lattner //===-- CommandObjectFrame.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 1030fdc8d8SChris Lattner #include "CommandObjectFrame.h" 1130fdc8d8SChris Lattner 1230fdc8d8SChris Lattner // C Includes 1330fdc8d8SChris Lattner // C++ Includes 1430fdc8d8SChris Lattner // Other libraries and framework includes 1530fdc8d8SChris Lattner // Project includes 1630fdc8d8SChris Lattner #include "lldb/Core/Debugger.h" 176d56d2ceSJim Ingham #include "lldb/Core/Module.h" 186d56d2ceSJim Ingham #include "lldb/Core/StreamFile.h" 1930fdc8d8SChris Lattner #include "lldb/Core/Timer.h" 206d56d2ceSJim Ingham #include "lldb/Core/Value.h" 216d56d2ceSJim Ingham #include "lldb/Core/ValueObject.h" 226d56d2ceSJim Ingham #include "lldb/Core/ValueObjectVariable.h" 237fb56d0aSGreg Clayton #include "lldb/Host/Host.h" 246d56d2ceSJim Ingham #include "lldb/Interpreter/Args.h" 2530fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 2630fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 276d56d2ceSJim Ingham #include "lldb/Interpreter/Options.h" 28*2837b766SJim Ingham #include "lldb/Interpreter/OptionGroupValueObjectDisplay.h" 296d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTType.h" 306d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTContext.h" 316d56d2ceSJim Ingham #include "lldb/Symbol/ObjectFile.h" 326d56d2ceSJim Ingham #include "lldb/Symbol/SymbolContext.h" 336d56d2ceSJim Ingham #include "lldb/Symbol/Type.h" 346d56d2ceSJim Ingham #include "lldb/Symbol/Variable.h" 356d56d2ceSJim Ingham #include "lldb/Symbol/VariableList.h" 3630fdc8d8SChris Lattner #include "lldb/Target/Process.h" 3730fdc8d8SChris Lattner #include "lldb/Target/StackFrame.h" 3830fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 396d56d2ceSJim Ingham #include "lldb/Target/Target.h" 4030fdc8d8SChris Lattner 4130fdc8d8SChris Lattner using namespace lldb; 4230fdc8d8SChris Lattner using namespace lldb_private; 4330fdc8d8SChris Lattner 4430fdc8d8SChris Lattner #pragma mark CommandObjectFrameInfo 4530fdc8d8SChris Lattner 4630fdc8d8SChris Lattner //------------------------------------------------------------------------- 4730fdc8d8SChris Lattner // CommandObjectFrameInfo 4830fdc8d8SChris Lattner //------------------------------------------------------------------------- 4930fdc8d8SChris Lattner 5030fdc8d8SChris Lattner class CommandObjectFrameInfo : public CommandObject 5130fdc8d8SChris Lattner { 5230fdc8d8SChris Lattner public: 5330fdc8d8SChris Lattner 54a7015092SGreg Clayton CommandObjectFrameInfo (CommandInterpreter &interpreter) : 55a7015092SGreg Clayton CommandObject (interpreter, 56a7015092SGreg Clayton "frame info", 57e3d26315SCaroline Tice "List information about the currently selected frame in the current thread.", 5830fdc8d8SChris Lattner "frame info", 5930fdc8d8SChris Lattner eFlagProcessMustBeLaunched | eFlagProcessMustBePaused) 6030fdc8d8SChris Lattner { 6130fdc8d8SChris Lattner } 6230fdc8d8SChris Lattner 6330fdc8d8SChris Lattner ~CommandObjectFrameInfo () 6430fdc8d8SChris Lattner { 6530fdc8d8SChris Lattner } 6630fdc8d8SChris Lattner 6730fdc8d8SChris Lattner bool 68a7015092SGreg Clayton Execute (Args& command, 6930fdc8d8SChris Lattner CommandReturnObject &result) 7030fdc8d8SChris Lattner { 718b82f087SGreg Clayton ExecutionContext exe_ctx(m_interpreter.GetExecutionContext()); 7230fdc8d8SChris Lattner if (exe_ctx.frame) 7330fdc8d8SChris Lattner { 740603aa9dSGreg Clayton exe_ctx.frame->DumpUsingSettingsFormat (&result.GetOutputStream()); 7530fdc8d8SChris Lattner result.GetOutputStream().EOL(); 7630fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 7730fdc8d8SChris Lattner } 7830fdc8d8SChris Lattner else 7930fdc8d8SChris Lattner { 8030fdc8d8SChris Lattner result.AppendError ("no current frame"); 8130fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 8230fdc8d8SChris Lattner } 8330fdc8d8SChris Lattner return result.Succeeded(); 8430fdc8d8SChris Lattner } 8530fdc8d8SChris Lattner }; 8630fdc8d8SChris Lattner 8730fdc8d8SChris Lattner #pragma mark CommandObjectFrameSelect 8830fdc8d8SChris Lattner 8930fdc8d8SChris Lattner //------------------------------------------------------------------------- 9030fdc8d8SChris Lattner // CommandObjectFrameSelect 9130fdc8d8SChris Lattner //------------------------------------------------------------------------- 9230fdc8d8SChris Lattner 9330fdc8d8SChris Lattner class CommandObjectFrameSelect : public CommandObject 9430fdc8d8SChris Lattner { 9530fdc8d8SChris Lattner public: 9630fdc8d8SChris Lattner 97864174e1SGreg Clayton class CommandOptions : public Options 98864174e1SGreg Clayton { 99864174e1SGreg Clayton public: 100864174e1SGreg Clayton 101eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 102f16066e8SJohnny Chen Options(interpreter) 103864174e1SGreg Clayton { 104f6b8b581SGreg Clayton OptionParsingStarting (); 105864174e1SGreg Clayton } 106864174e1SGreg Clayton 107864174e1SGreg Clayton virtual 108864174e1SGreg Clayton ~CommandOptions () 109864174e1SGreg Clayton { 110864174e1SGreg Clayton } 111864174e1SGreg Clayton 112864174e1SGreg Clayton virtual Error 113f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 114864174e1SGreg Clayton { 115864174e1SGreg Clayton Error error; 116864174e1SGreg Clayton bool success = false; 117864174e1SGreg Clayton char short_option = (char) m_getopt_table[option_idx].val; 118864174e1SGreg Clayton switch (short_option) 119864174e1SGreg Clayton { 120864174e1SGreg Clayton case 'r': 121864174e1SGreg Clayton relative_frame_offset = Args::StringToSInt32 (option_arg, INT32_MIN, 0, &success); 122864174e1SGreg Clayton if (!success) 123864174e1SGreg Clayton error.SetErrorStringWithFormat ("invalid frame offset argument '%s'.\n", option_arg); 124864174e1SGreg Clayton break; 125864174e1SGreg Clayton 126864174e1SGreg Clayton default: 1272c88643aSBenjamin Kramer error.SetErrorStringWithFormat ("Invalid short option character '%c'.\n", short_option); 128864174e1SGreg Clayton break; 129864174e1SGreg Clayton } 130864174e1SGreg Clayton 131864174e1SGreg Clayton return error; 132864174e1SGreg Clayton } 133864174e1SGreg Clayton 134864174e1SGreg Clayton void 135f6b8b581SGreg Clayton OptionParsingStarting () 136864174e1SGreg Clayton { 137864174e1SGreg Clayton relative_frame_offset = INT32_MIN; 138864174e1SGreg Clayton } 139864174e1SGreg Clayton 140e0d378b3SGreg Clayton const OptionDefinition* 141864174e1SGreg Clayton GetDefinitions () 142864174e1SGreg Clayton { 143864174e1SGreg Clayton return g_option_table; 144864174e1SGreg Clayton } 145864174e1SGreg Clayton 146864174e1SGreg Clayton // Options table: Required for subclasses of Options. 147864174e1SGreg Clayton 148e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 149864174e1SGreg Clayton int32_t relative_frame_offset; 150864174e1SGreg Clayton }; 151864174e1SGreg Clayton 152a7015092SGreg Clayton CommandObjectFrameSelect (CommandInterpreter &interpreter) : 153a7015092SGreg Clayton CommandObject (interpreter, 154a7015092SGreg Clayton "frame select", 155e3d26315SCaroline Tice "Select a frame by index from within the current thread and make it the current frame.", 156405fe67fSCaroline Tice NULL, 157eb0103f2SGreg Clayton eFlagProcessMustBeLaunched | eFlagProcessMustBePaused), 158eb0103f2SGreg Clayton m_options (interpreter) 15930fdc8d8SChris Lattner { 160405fe67fSCaroline Tice CommandArgumentEntry arg; 161405fe67fSCaroline Tice CommandArgumentData index_arg; 162405fe67fSCaroline Tice 163405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 164405fe67fSCaroline Tice index_arg.arg_type = eArgTypeFrameIndex; 165864174e1SGreg Clayton index_arg.arg_repetition = eArgRepeatOptional; 166405fe67fSCaroline Tice 167405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 168405fe67fSCaroline Tice arg.push_back (index_arg); 169405fe67fSCaroline Tice 170405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 171405fe67fSCaroline Tice m_arguments.push_back (arg); 17230fdc8d8SChris Lattner } 17330fdc8d8SChris Lattner 17430fdc8d8SChris Lattner ~CommandObjectFrameSelect () 17530fdc8d8SChris Lattner { 17630fdc8d8SChris Lattner } 17730fdc8d8SChris Lattner 178864174e1SGreg Clayton virtual 179864174e1SGreg Clayton Options * 180864174e1SGreg Clayton GetOptions () 181864174e1SGreg Clayton { 182864174e1SGreg Clayton return &m_options; 183864174e1SGreg Clayton } 184864174e1SGreg Clayton 185864174e1SGreg Clayton 18630fdc8d8SChris Lattner bool 187a7015092SGreg Clayton Execute (Args& command, 18830fdc8d8SChris Lattner CommandReturnObject &result) 18930fdc8d8SChris Lattner { 1908b82f087SGreg Clayton ExecutionContext exe_ctx (m_interpreter.GetExecutionContext()); 19130fdc8d8SChris Lattner if (exe_ctx.thread) 19230fdc8d8SChris Lattner { 193864174e1SGreg Clayton const uint32_t num_frames = exe_ctx.thread->GetStackFrameCount(); 194864174e1SGreg Clayton uint32_t frame_idx = UINT32_MAX; 195864174e1SGreg Clayton if (m_options.relative_frame_offset != INT32_MIN) 196864174e1SGreg Clayton { 197864174e1SGreg Clayton // The one and only argument is a signed relative frame index 198864174e1SGreg Clayton frame_idx = exe_ctx.thread->GetSelectedFrameIndex (); 199864174e1SGreg Clayton if (frame_idx == UINT32_MAX) 200864174e1SGreg Clayton frame_idx = 0; 201864174e1SGreg Clayton 202864174e1SGreg Clayton if (m_options.relative_frame_offset < 0) 203864174e1SGreg Clayton { 204864174e1SGreg Clayton if (frame_idx >= -m_options.relative_frame_offset) 205864174e1SGreg Clayton frame_idx += m_options.relative_frame_offset; 206864174e1SGreg Clayton else 207864174e1SGreg Clayton frame_idx = 0; 208864174e1SGreg Clayton } 209864174e1SGreg Clayton else if (m_options.relative_frame_offset > 0) 210864174e1SGreg Clayton { 211864174e1SGreg Clayton if (num_frames - frame_idx > m_options.relative_frame_offset) 212864174e1SGreg Clayton frame_idx += m_options.relative_frame_offset; 213864174e1SGreg Clayton else 214864174e1SGreg Clayton frame_idx = num_frames - 1; 215864174e1SGreg Clayton } 216864174e1SGreg Clayton } 217864174e1SGreg Clayton else 218864174e1SGreg Clayton { 21930fdc8d8SChris Lattner if (command.GetArgumentCount() == 1) 22030fdc8d8SChris Lattner { 22130fdc8d8SChris Lattner const char *frame_idx_cstr = command.GetArgumentAtIndex(0); 222864174e1SGreg Clayton frame_idx = Args::StringToUInt32 (frame_idx_cstr, UINT32_MAX, 0); 223864174e1SGreg Clayton } 224864174e1SGreg Clayton else 225864174e1SGreg Clayton { 226864174e1SGreg Clayton result.AppendError ("invalid arguments.\n"); 227eb0103f2SGreg Clayton m_options.GenerateOptionUsage (result.GetErrorStream(), this); 228864174e1SGreg Clayton } 229864174e1SGreg Clayton } 23030fdc8d8SChris Lattner 23130fdc8d8SChris Lattner if (frame_idx < num_frames) 23230fdc8d8SChris Lattner { 2332976d00aSJim Ingham exe_ctx.thread->SetSelectedFrameByIndex (frame_idx); 2342976d00aSJim Ingham exe_ctx.frame = exe_ctx.thread->GetSelectedFrame ().get(); 23530fdc8d8SChris Lattner 23630fdc8d8SChris Lattner if (exe_ctx.frame) 23730fdc8d8SChris Lattner { 238e40e4218SJim Ingham bool already_shown = false; 239e40e4218SJim Ingham SymbolContext frame_sc(exe_ctx.frame->GetSymbolContext(eSymbolContextLineEntry)); 240daccaa9eSCaroline Tice if (m_interpreter.GetDebugger().GetUseExternalEditor() && frame_sc.line_entry.file && frame_sc.line_entry.line != 0) 241e40e4218SJim Ingham { 242e40e4218SJim Ingham already_shown = Host::OpenFileInExternalEditor (frame_sc.line_entry.file, frame_sc.line_entry.line); 243e40e4218SJim Ingham } 244e40e4218SJim Ingham 2457260f620SGreg Clayton bool show_frame_info = true; 2467260f620SGreg Clayton bool show_source = !already_shown; 2477260f620SGreg Clayton uint32_t source_lines_before = 3; 2487260f620SGreg Clayton uint32_t source_lines_after = 3; 2497260f620SGreg Clayton if (exe_ctx.frame->GetStatus(result.GetOutputStream(), 2507260f620SGreg Clayton show_frame_info, 2517260f620SGreg Clayton show_source, 2527260f620SGreg Clayton source_lines_before, 2537260f620SGreg Clayton source_lines_after)) 25430fdc8d8SChris Lattner { 25530fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 25630fdc8d8SChris Lattner return result.Succeeded(); 25730fdc8d8SChris Lattner } 25830fdc8d8SChris Lattner } 25930fdc8d8SChris Lattner } 26030fdc8d8SChris Lattner result.AppendErrorWithFormat ("Frame index (%u) out of range.\n", frame_idx); 26130fdc8d8SChris Lattner } 26230fdc8d8SChris Lattner else 26330fdc8d8SChris Lattner { 26430fdc8d8SChris Lattner result.AppendError ("no current thread"); 26530fdc8d8SChris Lattner } 26630fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 26730fdc8d8SChris Lattner return false; 26830fdc8d8SChris Lattner } 269864174e1SGreg Clayton protected: 270864174e1SGreg Clayton 271864174e1SGreg Clayton CommandOptions m_options; 272864174e1SGreg Clayton }; 273864174e1SGreg Clayton 274e0d378b3SGreg Clayton OptionDefinition 275864174e1SGreg Clayton CommandObjectFrameSelect::CommandOptions::g_option_table[] = 276864174e1SGreg Clayton { 277864174e1SGreg Clayton { LLDB_OPT_SET_1, false, "relative", 'r', required_argument, NULL, 0, eArgTypeOffset, "A relative frame index offset from the current frame index."}, 278864174e1SGreg Clayton { 0, false, NULL, 0, 0, NULL, NULL, eArgTypeNone, NULL } 27930fdc8d8SChris Lattner }; 28030fdc8d8SChris Lattner 2816d56d2ceSJim Ingham #pragma mark CommandObjectFrameVariable 2826d56d2ceSJim Ingham //---------------------------------------------------------------------- 2836d56d2ceSJim Ingham // List images with associated information 2846d56d2ceSJim Ingham //---------------------------------------------------------------------- 2856d56d2ceSJim Ingham class CommandObjectFrameVariable : public CommandObject 2866d56d2ceSJim Ingham { 2876d56d2ceSJim Ingham public: 2886d56d2ceSJim Ingham 289*2837b766SJim Ingham class OptionGroupFrameVariable : public OptionGroup 2906d56d2ceSJim Ingham { 2916d56d2ceSJim Ingham public: 2926d56d2ceSJim Ingham 293*2837b766SJim Ingham OptionGroupFrameVariable () 2946d56d2ceSJim Ingham { 2956d56d2ceSJim Ingham } 2966d56d2ceSJim Ingham 2976d56d2ceSJim Ingham virtual 298*2837b766SJim Ingham ~OptionGroupFrameVariable () 2996d56d2ceSJim Ingham { 3006d56d2ceSJim Ingham } 3016d56d2ceSJim Ingham 302*2837b766SJim Ingham virtual uint32_t 303*2837b766SJim Ingham GetNumDefinitions (); 304*2837b766SJim Ingham 305*2837b766SJim Ingham virtual const OptionDefinition* 306*2837b766SJim Ingham GetDefinitions () 307*2837b766SJim Ingham { 308*2837b766SJim Ingham return g_option_table; 309*2837b766SJim Ingham } 310*2837b766SJim Ingham 3116d56d2ceSJim Ingham virtual Error 312*2837b766SJim Ingham SetOptionValue (CommandInterpreter &interpreter, 313*2837b766SJim Ingham uint32_t option_idx, 314*2837b766SJim Ingham const char *option_arg) 3156d56d2ceSJim Ingham { 3166d56d2ceSJim Ingham Error error; 317*2837b766SJim Ingham char short_option = (char) g_option_table[option_idx].short_option; 3186d56d2ceSJim Ingham switch (short_option) 3196d56d2ceSJim Ingham { 3206d56d2ceSJim Ingham case 'r': use_regex = true; break; 3216d56d2ceSJim Ingham case 'a': show_args = false; break; 3226d56d2ceSJim Ingham case 'l': show_locals = false; break; 323a134cc1bSGreg Clayton case 'g': show_globals = true; break; 324a134cc1bSGreg Clayton case 'c': show_decl = true; break; 32568ebae61SGreg Clayton case 'f': error = Args::StringToFormat(option_arg, format, NULL); break; 3266d56d2ceSJim Ingham case 'G': 327a134cc1bSGreg Clayton globals.push_back(ConstString (option_arg)); 3286d56d2ceSJim Ingham break; 3296d56d2ceSJim Ingham case 's': 3306d56d2ceSJim Ingham show_scope = true; 3316d56d2ceSJim Ingham break; 3326d56d2ceSJim Ingham 3336d56d2ceSJim Ingham default: 3346d56d2ceSJim Ingham error.SetErrorStringWithFormat("Invalid short option character '%c'.\n", short_option); 3356d56d2ceSJim Ingham break; 3366d56d2ceSJim Ingham } 3376d56d2ceSJim Ingham 3386d56d2ceSJim Ingham return error; 3396d56d2ceSJim Ingham } 3406d56d2ceSJim Ingham 341*2837b766SJim Ingham virtual void 342*2837b766SJim Ingham OptionParsingStarting (CommandInterpreter &interpreter) 3436d56d2ceSJim Ingham { 3446d56d2ceSJim Ingham show_args = true; 345a134cc1bSGreg Clayton show_decl = false; 346ded470d3SGreg Clayton format = eFormatDefault; 347*2837b766SJim Ingham show_globals = false; 348*2837b766SJim Ingham show_locals = true; 349*2837b766SJim Ingham use_regex = false; 350*2837b766SJim Ingham show_scope = false; 3516d56d2ceSJim Ingham globals.clear(); 3526d56d2ceSJim Ingham } 3536d56d2ceSJim Ingham 3546d56d2ceSJim Ingham // Options table: Required for subclasses of Options. 3556d56d2ceSJim Ingham 356e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 357*2837b766SJim Ingham 358*2837b766SJim Ingham bool use_regex:1, 359a134cc1bSGreg Clayton show_args:1, 360a134cc1bSGreg Clayton show_locals:1, 361a134cc1bSGreg Clayton show_globals:1, 362a134cc1bSGreg Clayton show_scope:1, 363*2837b766SJim Ingham show_decl:1; 364ded470d3SGreg Clayton lldb::Format format; // The format to use when dumping variables or children of variables 3656d56d2ceSJim Ingham std::vector<ConstString> globals; 3666d56d2ceSJim Ingham // Instance variables to hold the values for command options. 3676d56d2ceSJim Ingham }; 3686d56d2ceSJim Ingham 369a7015092SGreg Clayton CommandObjectFrameVariable (CommandInterpreter &interpreter) : 370a7015092SGreg Clayton CommandObject (interpreter, 3716d56d2ceSJim Ingham "frame variable", 372ed8a705cSGreg Clayton "Show frame variables. All argument and local variables " 373ed8a705cSGreg Clayton "that are in scope will be shown when no arguments are given. " 374ed8a705cSGreg Clayton "If any arguments are specified, they can be names of " 375ed8a705cSGreg Clayton "argument, local, file static and file global variables. " 376ed8a705cSGreg Clayton "Children of aggregate variables can be specified such as " 377ed8a705cSGreg Clayton "'var->child.x'.", 378ff471a94SJim Ingham NULL, 379eb0103f2SGreg Clayton eFlagProcessMustBeLaunched | eFlagProcessMustBePaused), 380*2837b766SJim Ingham m_option_group (interpreter), 381*2837b766SJim Ingham m_frame_var_options(), 382*2837b766SJim Ingham m_varobj_options() 3836d56d2ceSJim Ingham { 384405fe67fSCaroline Tice CommandArgumentEntry arg; 385405fe67fSCaroline Tice CommandArgumentData var_name_arg; 386405fe67fSCaroline Tice 387405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 388405fe67fSCaroline Tice var_name_arg.arg_type = eArgTypeVarName; 389405fe67fSCaroline Tice var_name_arg.arg_repetition = eArgRepeatStar; 390405fe67fSCaroline Tice 391405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 392405fe67fSCaroline Tice arg.push_back (var_name_arg); 393405fe67fSCaroline Tice 394405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 395405fe67fSCaroline Tice m_arguments.push_back (arg); 396*2837b766SJim Ingham 397*2837b766SJim Ingham m_option_group.Append (&m_frame_var_options, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 398*2837b766SJim Ingham m_option_group.Append (&m_varobj_options, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 399*2837b766SJim Ingham m_option_group.Finalize(); 4006d56d2ceSJim Ingham } 4016d56d2ceSJim Ingham 4026d56d2ceSJim Ingham virtual 4036d56d2ceSJim Ingham ~CommandObjectFrameVariable () 4046d56d2ceSJim Ingham { 4056d56d2ceSJim Ingham } 4066d56d2ceSJim Ingham 4076d56d2ceSJim Ingham virtual 4086d56d2ceSJim Ingham Options * 4096d56d2ceSJim Ingham GetOptions () 4106d56d2ceSJim Ingham { 411*2837b766SJim Ingham return &m_option_group; 4126d56d2ceSJim Ingham } 4136d56d2ceSJim Ingham 4146d56d2ceSJim Ingham 4156d56d2ceSJim Ingham virtual bool 4166d56d2ceSJim Ingham Execute 4176d56d2ceSJim Ingham ( 4186d56d2ceSJim Ingham Args& command, 4196d56d2ceSJim Ingham CommandReturnObject &result 4206d56d2ceSJim Ingham ) 4216d56d2ceSJim Ingham { 4228b82f087SGreg Clayton ExecutionContext exe_ctx(m_interpreter.GetExecutionContext()); 4236d56d2ceSJim Ingham if (exe_ctx.frame == NULL) 4246d56d2ceSJim Ingham { 425340b2baaSGreg Clayton result.AppendError ("you must be stopped in a valid stack frame to view frame variables."); 4266d56d2ceSJim Ingham result.SetStatus (eReturnStatusFailed); 4276d56d2ceSJim Ingham return false; 4286d56d2ceSJim Ingham } 4296d56d2ceSJim Ingham else 4306d56d2ceSJim Ingham { 431a134cc1bSGreg Clayton Stream &s = result.GetOutputStream(); 4326d56d2ceSJim Ingham 433a134cc1bSGreg Clayton bool get_file_globals = true; 434a134cc1bSGreg Clayton VariableList *variable_list = exe_ctx.frame->GetVariableList (get_file_globals); 435a134cc1bSGreg Clayton 4366d56d2ceSJim Ingham VariableSP var_sp; 4376d56d2ceSJim Ingham ValueObjectSP valobj_sp; 43878a685aaSJim Ingham 4396d56d2ceSJim Ingham const char *name_cstr = NULL; 4406d56d2ceSJim Ingham size_t idx; 441*2837b766SJim Ingham if (!m_frame_var_options.globals.empty()) 4426d56d2ceSJim Ingham { 4436d56d2ceSJim Ingham uint32_t fail_count = 0; 4446d56d2ceSJim Ingham if (exe_ctx.target) 4456d56d2ceSJim Ingham { 446*2837b766SJim Ingham const size_t num_globals = m_frame_var_options.globals.size(); 4476d56d2ceSJim Ingham for (idx = 0; idx < num_globals; ++idx) 4486d56d2ceSJim Ingham { 4496d56d2ceSJim Ingham VariableList global_var_list; 45078a685aaSJim Ingham const uint32_t num_matching_globals 451*2837b766SJim Ingham = exe_ctx.target->GetImages().FindGlobalVariables (m_frame_var_options.globals[idx], 45278a685aaSJim Ingham true, 45378a685aaSJim Ingham UINT32_MAX, 45478a685aaSJim Ingham global_var_list); 4556d56d2ceSJim Ingham 4566d56d2ceSJim Ingham if (num_matching_globals == 0) 4576d56d2ceSJim Ingham { 4586d56d2ceSJim Ingham ++fail_count; 45978a685aaSJim Ingham result.GetErrorStream().Printf ("error: can't find global variable '%s'\n", 460*2837b766SJim Ingham m_frame_var_options.globals[idx].AsCString()); 4616d56d2ceSJim Ingham } 4626d56d2ceSJim Ingham else 4636d56d2ceSJim Ingham { 4646d56d2ceSJim Ingham for (uint32_t global_idx=0; global_idx<num_matching_globals; ++global_idx) 4656d56d2ceSJim Ingham { 4666d56d2ceSJim Ingham var_sp = global_var_list.GetVariableAtIndex(global_idx); 4676d56d2ceSJim Ingham if (var_sp) 4686d56d2ceSJim Ingham { 469*2837b766SJim Ingham valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp, 470*2837b766SJim Ingham m_varobj_options.use_dynamic); 4716d56d2ceSJim Ingham if (!valobj_sp) 472*2837b766SJim Ingham valobj_sp = exe_ctx.frame->TrackGlobalVariable (var_sp, 473*2837b766SJim Ingham m_varobj_options.use_dynamic); 4746d56d2ceSJim Ingham 4756d56d2ceSJim Ingham if (valobj_sp) 4766d56d2ceSJim Ingham { 477*2837b766SJim Ingham if (m_frame_var_options.format != eFormatDefault) 478*2837b766SJim Ingham valobj_sp->SetFormat (m_frame_var_options.format); 479ded470d3SGreg Clayton 480*2837b766SJim Ingham if (m_frame_var_options.show_decl && var_sp->GetDeclaration ().GetFile()) 481a134cc1bSGreg Clayton { 4826f00abd5SGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 4836f00abd5SGreg Clayton s.PutCString (": "); 484a134cc1bSGreg Clayton } 485a134cc1bSGreg Clayton 4861d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 4876f00abd5SGreg Clayton valobj_sp.get(), 4886f00abd5SGreg Clayton name_cstr, 489*2837b766SJim Ingham m_varobj_options.ptr_depth, 4906f00abd5SGreg Clayton 0, 491*2837b766SJim Ingham m_varobj_options.max_depth, 492*2837b766SJim Ingham m_varobj_options.show_types, 493*2837b766SJim Ingham m_varobj_options.show_location, 494*2837b766SJim Ingham m_varobj_options.use_objc, 495*2837b766SJim Ingham m_varobj_options.use_dynamic, 4968f92f0a3SGreg Clayton false, 497*2837b766SJim Ingham m_varobj_options.flat_output); 4986d56d2ceSJim Ingham } 4996d56d2ceSJim Ingham } 5006d56d2ceSJim Ingham } 5016d56d2ceSJim Ingham } 5026d56d2ceSJim Ingham } 5036d56d2ceSJim Ingham } 5046d56d2ceSJim Ingham if (fail_count) 5056d56d2ceSJim Ingham result.SetStatus (eReturnStatusFailed); 5066d56d2ceSJim Ingham } 5079df87c17SGreg Clayton else if (variable_list) 5089df87c17SGreg Clayton { 5099df87c17SGreg Clayton if (command.GetArgumentCount() > 0) 5106d56d2ceSJim Ingham { 51146747022SGreg Clayton VariableList regex_var_list; 51246747022SGreg Clayton 5136d56d2ceSJim Ingham // If we have any args to the variable command, we will make 5146d56d2ceSJim Ingham // variable objects from them... 5156d56d2ceSJim Ingham for (idx = 0; (name_cstr = command.GetArgumentAtIndex(idx)) != NULL; ++idx) 5166d56d2ceSJim Ingham { 517*2837b766SJim Ingham uint32_t ptr_depth = m_varobj_options.ptr_depth; 51846747022SGreg Clayton 519*2837b766SJim Ingham if (m_frame_var_options.use_regex) 52046747022SGreg Clayton { 52146747022SGreg Clayton const uint32_t regex_start_index = regex_var_list.GetSize(); 52246747022SGreg Clayton RegularExpression regex (name_cstr); 52346747022SGreg Clayton if (regex.Compile(name_cstr)) 52446747022SGreg Clayton { 52546747022SGreg Clayton size_t num_matches = 0; 52678a685aaSJim Ingham const size_t num_new_regex_vars = variable_list->AppendVariablesIfUnique(regex, 52778a685aaSJim Ingham regex_var_list, 52878a685aaSJim Ingham num_matches); 52946747022SGreg Clayton if (num_new_regex_vars > 0) 53046747022SGreg Clayton { 53146747022SGreg Clayton for (uint32_t regex_idx = regex_start_index, end_index = regex_var_list.GetSize(); 53246747022SGreg Clayton regex_idx < end_index; 53346747022SGreg Clayton ++regex_idx) 53446747022SGreg Clayton { 53546747022SGreg Clayton var_sp = regex_var_list.GetVariableAtIndex (regex_idx); 53646747022SGreg Clayton if (var_sp) 53746747022SGreg Clayton { 538*2837b766SJim Ingham valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp, m_varobj_options.use_dynamic); 53946747022SGreg Clayton if (valobj_sp) 54046747022SGreg Clayton { 541*2837b766SJim Ingham if (m_frame_var_options.format != eFormatDefault) 542*2837b766SJim Ingham valobj_sp->SetFormat (m_frame_var_options.format); 543ded470d3SGreg Clayton 544*2837b766SJim Ingham if (m_frame_var_options.show_decl && var_sp->GetDeclaration ().GetFile()) 54546747022SGreg Clayton { 54646747022SGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 54746747022SGreg Clayton s.PutCString (": "); 54846747022SGreg Clayton } 54946747022SGreg Clayton 55046747022SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 55146747022SGreg Clayton valobj_sp.get(), 55246747022SGreg Clayton var_sp->GetName().AsCString(), 553*2837b766SJim Ingham m_varobj_options.ptr_depth, 55446747022SGreg Clayton 0, 555*2837b766SJim Ingham m_varobj_options.max_depth, 556*2837b766SJim Ingham m_varobj_options.show_types, 557*2837b766SJim Ingham m_varobj_options.show_location, 558*2837b766SJim Ingham m_varobj_options.use_objc, 559*2837b766SJim Ingham m_varobj_options.use_dynamic, 5608f92f0a3SGreg Clayton false, 561*2837b766SJim Ingham m_varobj_options.flat_output); 56246747022SGreg Clayton } 56346747022SGreg Clayton } 56446747022SGreg Clayton } 56546747022SGreg Clayton } 56646747022SGreg Clayton else if (num_matches == 0) 56746747022SGreg Clayton { 56846747022SGreg Clayton result.GetErrorStream().Printf ("error: no variables matched the regular expression '%s'.\n", name_cstr); 56946747022SGreg Clayton } 57046747022SGreg Clayton } 57146747022SGreg Clayton else 57246747022SGreg Clayton { 57346747022SGreg Clayton char regex_error[1024]; 57446747022SGreg Clayton if (regex.GetErrorAsCString(regex_error, sizeof(regex_error))) 57546747022SGreg Clayton result.GetErrorStream().Printf ("error: %s\n", regex_error); 57646747022SGreg Clayton else 57746747022SGreg Clayton result.GetErrorStream().Printf ("error: unkown regex error when compiling '%s'\n", name_cstr); 57846747022SGreg Clayton } 57946747022SGreg Clayton } 58046747022SGreg Clayton else 58146747022SGreg Clayton { 58254979cddSGreg Clayton Error error; 58378a685aaSJim Ingham uint32_t expr_path_options = StackFrame::eExpressionPathOptionCheckPtrVsMember; 584*2837b766SJim Ingham lldb::VariableSP var_sp; 585*2837b766SJim Ingham valobj_sp = exe_ctx.frame->GetValueForVariableExpressionPath (name_cstr, 586*2837b766SJim Ingham m_varobj_options.use_dynamic, 587*2837b766SJim Ingham expr_path_options, 588*2837b766SJim Ingham var_sp, 589*2837b766SJim Ingham error); 5906d56d2ceSJim Ingham if (valobj_sp) 5916d56d2ceSJim Ingham { 592*2837b766SJim Ingham if (m_frame_var_options.format != eFormatDefault) 593*2837b766SJim Ingham valobj_sp->SetFormat (m_frame_var_options.format); 594*2837b766SJim Ingham if (m_frame_var_options.show_decl && var_sp && var_sp->GetDeclaration ().GetFile()) 595a134cc1bSGreg Clayton { 596a134cc1bSGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 597a134cc1bSGreg Clayton s.PutCString (": "); 598a134cc1bSGreg Clayton } 5991d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 600a134cc1bSGreg Clayton valobj_sp.get(), 60183c5cd9dSGreg Clayton valobj_sp->GetParent() ? name_cstr : NULL, 602a134cc1bSGreg Clayton ptr_depth, 603a134cc1bSGreg Clayton 0, 604*2837b766SJim Ingham m_varobj_options.max_depth, 605*2837b766SJim Ingham m_varobj_options.show_types, 606*2837b766SJim Ingham m_varobj_options.show_location, 607*2837b766SJim Ingham m_varobj_options.use_objc, 608*2837b766SJim Ingham m_varobj_options.use_dynamic, 6098f92f0a3SGreg Clayton false, 610*2837b766SJim Ingham m_varobj_options.flat_output); 6116d56d2ceSJim Ingham } 6126d56d2ceSJim Ingham else 6136d56d2ceSJim Ingham { 61454979cddSGreg Clayton const char *error_cstr = error.AsCString(NULL); 61554979cddSGreg Clayton if (error_cstr) 61654979cddSGreg Clayton result.GetErrorStream().Printf("error: %s\n", error_cstr); 61754979cddSGreg Clayton else 61854979cddSGreg Clayton result.GetErrorStream().Printf ("error: unable to find any variable expression path that matches '%s'\n", name_cstr); 6196d56d2ceSJim Ingham } 6206d56d2ceSJim Ingham } 6216d56d2ceSJim Ingham } 62246747022SGreg Clayton } 6236d56d2ceSJim Ingham else 6246d56d2ceSJim Ingham { 625a134cc1bSGreg Clayton const uint32_t num_variables = variable_list->GetSize(); 6266d56d2ceSJim Ingham 6276d56d2ceSJim Ingham if (num_variables > 0) 6286d56d2ceSJim Ingham { 6296d56d2ceSJim Ingham for (uint32_t i=0; i<num_variables; i++) 6306d56d2ceSJim Ingham { 6311a65ae11SGreg Clayton var_sp = variable_list->GetVariableAtIndex(i); 63278a685aaSJim Ingham 6336d56d2ceSJim Ingham bool dump_variable = true; 6346d56d2ceSJim Ingham 635a134cc1bSGreg Clayton switch (var_sp->GetScope()) 6366d56d2ceSJim Ingham { 6376d56d2ceSJim Ingham case eValueTypeVariableGlobal: 638*2837b766SJim Ingham dump_variable = m_frame_var_options.show_globals; 639*2837b766SJim Ingham if (dump_variable && m_frame_var_options.show_scope) 640a134cc1bSGreg Clayton s.PutCString("GLOBAL: "); 6416d56d2ceSJim Ingham break; 6426d56d2ceSJim Ingham 6436d56d2ceSJim Ingham case eValueTypeVariableStatic: 644*2837b766SJim Ingham dump_variable = m_frame_var_options.show_globals; 645*2837b766SJim Ingham if (dump_variable && m_frame_var_options.show_scope) 646a134cc1bSGreg Clayton s.PutCString("STATIC: "); 6476d56d2ceSJim Ingham break; 6486d56d2ceSJim Ingham 6496d56d2ceSJim Ingham case eValueTypeVariableArgument: 650*2837b766SJim Ingham dump_variable = m_frame_var_options.show_args; 651*2837b766SJim Ingham if (dump_variable && m_frame_var_options.show_scope) 652a134cc1bSGreg Clayton s.PutCString(" ARG: "); 6536d56d2ceSJim Ingham break; 6546d56d2ceSJim Ingham 6556d56d2ceSJim Ingham case eValueTypeVariableLocal: 656*2837b766SJim Ingham dump_variable = m_frame_var_options.show_locals; 657*2837b766SJim Ingham if (dump_variable && m_frame_var_options.show_scope) 658a134cc1bSGreg Clayton s.PutCString(" LOCAL: "); 6596d56d2ceSJim Ingham break; 6606d56d2ceSJim Ingham 6616d56d2ceSJim Ingham default: 6626d56d2ceSJim Ingham break; 6636d56d2ceSJim Ingham } 6646d56d2ceSJim Ingham 6656d56d2ceSJim Ingham if (dump_variable) 666a134cc1bSGreg Clayton { 667a134cc1bSGreg Clayton 668a134cc1bSGreg Clayton // Use the variable object code to make sure we are 669a134cc1bSGreg Clayton // using the same APIs as the the public API will be 670a134cc1bSGreg Clayton // using... 671*2837b766SJim Ingham valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp, 672*2837b766SJim Ingham m_varobj_options.use_dynamic); 673a134cc1bSGreg Clayton if (valobj_sp) 674a134cc1bSGreg Clayton { 675*2837b766SJim Ingham if (m_frame_var_options.format != eFormatDefault) 676*2837b766SJim Ingham valobj_sp->SetFormat (m_frame_var_options.format); 677ded470d3SGreg Clayton 6786f00abd5SGreg Clayton // When dumping all variables, don't print any variables 6796f00abd5SGreg Clayton // that are not in scope to avoid extra unneeded output 6806035b67dSJim Ingham if (valobj_sp->IsInScope ()) 6816f00abd5SGreg Clayton { 682*2837b766SJim Ingham if (m_frame_var_options.show_decl && var_sp->GetDeclaration ().GetFile()) 683a134cc1bSGreg Clayton { 684a134cc1bSGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 685a134cc1bSGreg Clayton s.PutCString (": "); 686a134cc1bSGreg Clayton } 6871d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 688a134cc1bSGreg Clayton valobj_sp.get(), 689a134cc1bSGreg Clayton name_cstr, 690*2837b766SJim Ingham m_varobj_options.ptr_depth, 691a134cc1bSGreg Clayton 0, 692*2837b766SJim Ingham m_varobj_options.max_depth, 693*2837b766SJim Ingham m_varobj_options.show_types, 694*2837b766SJim Ingham m_varobj_options.show_location, 695*2837b766SJim Ingham m_varobj_options.use_objc, 696*2837b766SJim Ingham m_varobj_options.use_dynamic, 6978f92f0a3SGreg Clayton false, 698*2837b766SJim Ingham m_varobj_options.flat_output); 699a134cc1bSGreg Clayton } 700a134cc1bSGreg Clayton } 7016d56d2ceSJim Ingham } 7026d56d2ceSJim Ingham } 7036d56d2ceSJim Ingham } 7046f00abd5SGreg Clayton } 7056d56d2ceSJim Ingham result.SetStatus (eReturnStatusSuccessFinishResult); 7066d56d2ceSJim Ingham } 7079df87c17SGreg Clayton } 7086d56d2ceSJim Ingham return result.Succeeded(); 7096d56d2ceSJim Ingham } 7106d56d2ceSJim Ingham protected: 7116d56d2ceSJim Ingham 712*2837b766SJim Ingham OptionGroupOptions m_option_group; 713*2837b766SJim Ingham OptionGroupFrameVariable m_frame_var_options; 714*2837b766SJim Ingham OptionGroupValueObjectDisplay m_varobj_options; 7156d56d2ceSJim Ingham }; 7166d56d2ceSJim Ingham 717e0d378b3SGreg Clayton OptionDefinition 718*2837b766SJim Ingham CommandObjectFrameVariable::OptionGroupFrameVariable::g_option_table[] = 7196d56d2ceSJim Ingham { 72078a685aaSJim Ingham { LLDB_OPT_SET_1, false, "no-args", 'a', no_argument, NULL, 0, eArgTypeNone, "Omit function arguments."}, 72178a685aaSJim Ingham { LLDB_OPT_SET_1, false, "show-declaration",'c', no_argument, NULL, 0, eArgTypeNone, "Show variable declaration information (source file and line where the variable was declared)."}, 72278a685aaSJim Ingham { LLDB_OPT_SET_1, false, "format", 'f', required_argument, NULL, 0, eArgTypeExprFormat, "Specify the format that the variable output should use."}, 723deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "show-globals", 'g', no_argument, NULL, 0, eArgTypeNone, "Show the current frame source file global and static variables."}, 724deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "find-global", 'G', required_argument, NULL, 0, eArgTypeVarName, "Find a global variable by name (which might not be in the current stack frame source file)."}, 725deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-locals", 'l', no_argument, NULL, 0, eArgTypeNone, "Omit local variables."}, 72646747022SGreg Clayton { LLDB_OPT_SET_1, false, "regex", 'r', no_argument, NULL, 0, eArgTypeRegularExpression, "The <variable-name> argument for name lookups are regular expressions."}, 72778a685aaSJim Ingham { LLDB_OPT_SET_1, false, "scope", 's', no_argument, NULL, 0, eArgTypeNone, "Show variable scope (argument, local, global, static)."}, 728deaab222SCaroline Tice { 0, false, NULL, 0, 0, NULL, NULL, eArgTypeNone, NULL } 7296d56d2ceSJim Ingham }; 730*2837b766SJim Ingham 731*2837b766SJim Ingham uint32_t 732*2837b766SJim Ingham CommandObjectFrameVariable::OptionGroupFrameVariable::GetNumDefinitions () 733*2837b766SJim Ingham { 734*2837b766SJim Ingham return sizeof(CommandObjectFrameVariable::OptionGroupFrameVariable::g_option_table)/sizeof(OptionDefinition); 735*2837b766SJim Ingham } 736*2837b766SJim Ingham 737*2837b766SJim Ingham 73830fdc8d8SChris Lattner #pragma mark CommandObjectMultiwordFrame 73930fdc8d8SChris Lattner 74030fdc8d8SChris Lattner //------------------------------------------------------------------------- 74130fdc8d8SChris Lattner // CommandObjectMultiwordFrame 74230fdc8d8SChris Lattner //------------------------------------------------------------------------- 74330fdc8d8SChris Lattner 7446611103cSGreg Clayton CommandObjectMultiwordFrame::CommandObjectMultiwordFrame (CommandInterpreter &interpreter) : 745a7015092SGreg Clayton CommandObjectMultiword (interpreter, 746a7015092SGreg Clayton "frame", 74730fdc8d8SChris Lattner "A set of commands for operating on the current thread's frames.", 74830fdc8d8SChris Lattner "frame <subcommand> [<subcommand-options>]") 74930fdc8d8SChris Lattner { 750a7015092SGreg Clayton LoadSubCommand ("info", CommandObjectSP (new CommandObjectFrameInfo (interpreter))); 751a7015092SGreg Clayton LoadSubCommand ("select", CommandObjectSP (new CommandObjectFrameSelect (interpreter))); 752a7015092SGreg Clayton LoadSubCommand ("variable", CommandObjectSP (new CommandObjectFrameVariable (interpreter))); 75330fdc8d8SChris Lattner } 75430fdc8d8SChris Lattner 75530fdc8d8SChris Lattner CommandObjectMultiwordFrame::~CommandObjectMultiwordFrame () 75630fdc8d8SChris Lattner { 75730fdc8d8SChris Lattner } 75830fdc8d8SChris Lattner 759