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" 236d56d2ceSJim Ingham #include "lldb/Interpreter/Args.h" 2430fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 2530fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 266d56d2ceSJim Ingham #include "lldb/Interpreter/Options.h" 276d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTType.h" 286d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTContext.h" 296d56d2ceSJim Ingham #include "lldb/Symbol/ObjectFile.h" 306d56d2ceSJim Ingham #include "lldb/Symbol/SymbolContext.h" 316d56d2ceSJim Ingham #include "lldb/Symbol/Type.h" 326d56d2ceSJim Ingham #include "lldb/Symbol/Variable.h" 336d56d2ceSJim Ingham #include "lldb/Symbol/VariableList.h" 3430fdc8d8SChris Lattner #include "lldb/Target/Process.h" 3530fdc8d8SChris Lattner #include "lldb/Target/StackFrame.h" 3630fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 376d56d2ceSJim Ingham #include "lldb/Target/Target.h" 3830fdc8d8SChris Lattner 3930fdc8d8SChris Lattner #include "CommandObjectThread.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 { 71a7015092SGreg Clayton ExecutionContext exe_ctx(m_interpreter.GetDebugger().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 101864174e1SGreg Clayton CommandOptions () : 102864174e1SGreg Clayton Options() 103864174e1SGreg Clayton { 104864174e1SGreg Clayton ResetOptionValues (); 105864174e1SGreg Clayton } 106864174e1SGreg Clayton 107864174e1SGreg Clayton virtual 108864174e1SGreg Clayton ~CommandOptions () 109864174e1SGreg Clayton { 110864174e1SGreg Clayton } 111864174e1SGreg Clayton 112864174e1SGreg Clayton virtual Error 113864174e1SGreg Clayton SetOptionValue (int 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 135864174e1SGreg Clayton ResetOptionValues () 136864174e1SGreg Clayton { 137864174e1SGreg Clayton Options::ResetOptionValues(); 138864174e1SGreg Clayton relative_frame_offset = INT32_MIN; 139864174e1SGreg Clayton } 140864174e1SGreg Clayton 141864174e1SGreg Clayton const lldb::OptionDefinition* 142864174e1SGreg Clayton GetDefinitions () 143864174e1SGreg Clayton { 144864174e1SGreg Clayton return g_option_table; 145864174e1SGreg Clayton } 146864174e1SGreg Clayton 147864174e1SGreg Clayton // Options table: Required for subclasses of Options. 148864174e1SGreg Clayton 149864174e1SGreg Clayton static lldb::OptionDefinition g_option_table[]; 150864174e1SGreg Clayton int32_t relative_frame_offset; 151864174e1SGreg Clayton }; 152864174e1SGreg Clayton 153a7015092SGreg Clayton CommandObjectFrameSelect (CommandInterpreter &interpreter) : 154a7015092SGreg Clayton CommandObject (interpreter, 155a7015092SGreg Clayton "frame select", 156e3d26315SCaroline Tice "Select a frame by index from within the current thread and make it the current frame.", 157405fe67fSCaroline Tice NULL, 15830fdc8d8SChris Lattner eFlagProcessMustBeLaunched | eFlagProcessMustBePaused) 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 { 190a7015092SGreg Clayton ExecutionContext exe_ctx (m_interpreter.GetDebugger().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"); 227864174e1SGreg Clayton m_options.GenerateOptionUsage (m_interpreter, 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 24530fdc8d8SChris Lattner if (DisplayFrameForExecutionContext (exe_ctx.thread, 24630fdc8d8SChris Lattner exe_ctx.frame, 247a7015092SGreg Clayton m_interpreter, 24830fdc8d8SChris Lattner result.GetOutputStream(), 24930fdc8d8SChris Lattner true, 250e40e4218SJim Ingham !already_shown, 25130fdc8d8SChris Lattner 3, 25230fdc8d8SChris Lattner 3)) 25330fdc8d8SChris Lattner { 25430fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 25530fdc8d8SChris Lattner return result.Succeeded(); 25630fdc8d8SChris Lattner } 25730fdc8d8SChris Lattner } 25830fdc8d8SChris Lattner } 25930fdc8d8SChris Lattner result.AppendErrorWithFormat ("Frame index (%u) out of range.\n", frame_idx); 26030fdc8d8SChris Lattner } 26130fdc8d8SChris Lattner else 26230fdc8d8SChris Lattner { 26330fdc8d8SChris Lattner result.AppendError ("no current thread"); 26430fdc8d8SChris Lattner } 26530fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 26630fdc8d8SChris Lattner return false; 26730fdc8d8SChris Lattner } 268864174e1SGreg Clayton protected: 269864174e1SGreg Clayton 270864174e1SGreg Clayton CommandOptions m_options; 271864174e1SGreg Clayton }; 272864174e1SGreg Clayton 273864174e1SGreg Clayton lldb::OptionDefinition 274864174e1SGreg Clayton CommandObjectFrameSelect::CommandOptions::g_option_table[] = 275864174e1SGreg Clayton { 276864174e1SGreg Clayton { LLDB_OPT_SET_1, false, "relative", 'r', required_argument, NULL, 0, eArgTypeOffset, "A relative frame index offset from the current frame index."}, 277864174e1SGreg Clayton { 0, false, NULL, 0, 0, NULL, NULL, eArgTypeNone, NULL } 27830fdc8d8SChris Lattner }; 27930fdc8d8SChris Lattner 2806d56d2ceSJim Ingham #pragma mark CommandObjectFrameVariable 2816d56d2ceSJim Ingham //---------------------------------------------------------------------- 2826d56d2ceSJim Ingham // List images with associated information 2836d56d2ceSJim Ingham //---------------------------------------------------------------------- 2846d56d2ceSJim Ingham class CommandObjectFrameVariable : public CommandObject 2856d56d2ceSJim Ingham { 2866d56d2ceSJim Ingham public: 2876d56d2ceSJim Ingham 2886d56d2ceSJim Ingham class CommandOptions : public Options 2896d56d2ceSJim Ingham { 2906d56d2ceSJim Ingham public: 2916d56d2ceSJim Ingham 2926d56d2ceSJim Ingham CommandOptions () : 2936d56d2ceSJim Ingham Options() 2946d56d2ceSJim Ingham { 2956d56d2ceSJim Ingham ResetOptionValues (); 2966d56d2ceSJim Ingham } 2976d56d2ceSJim Ingham 2986d56d2ceSJim Ingham virtual 2996d56d2ceSJim Ingham ~CommandOptions () 3006d56d2ceSJim Ingham { 3016d56d2ceSJim Ingham } 3026d56d2ceSJim Ingham 3036d56d2ceSJim Ingham virtual Error 3046d56d2ceSJim Ingham SetOptionValue (int option_idx, const char *option_arg) 3056d56d2ceSJim Ingham { 3066d56d2ceSJim Ingham Error error; 3076d56d2ceSJim Ingham bool success; 3086d56d2ceSJim Ingham char short_option = (char) m_getopt_table[option_idx].val; 3096d56d2ceSJim Ingham switch (short_option) 3106d56d2ceSJim Ingham { 3116d56d2ceSJim Ingham case 'o': use_objc = true; break; 3126d56d2ceSJim Ingham case 'r': use_regex = true; break; 3136d56d2ceSJim Ingham case 'a': show_args = false; break; 3146d56d2ceSJim Ingham case 'l': show_locals = false; break; 315a134cc1bSGreg Clayton case 'g': show_globals = true; break; 316b6e8cf96SGreg Clayton case 't': show_types = true; break; 3176d56d2ceSJim Ingham case 'y': show_summary = false; break; 3186d56d2ceSJim Ingham case 'L': show_location= true; break; 319a134cc1bSGreg Clayton case 'c': show_decl = true; break; 3206d56d2ceSJim Ingham case 'D': debug = true; break; 3218f92f0a3SGreg Clayton case 'f': flat_output = true; break; 3226d56d2ceSJim Ingham case 'd': 3236d56d2ceSJim Ingham max_depth = Args::StringToUInt32 (option_arg, UINT32_MAX, 0, &success); 3246d56d2ceSJim Ingham if (!success) 3256d56d2ceSJim Ingham error.SetErrorStringWithFormat("Invalid max depth '%s'.\n", option_arg); 3266d56d2ceSJim Ingham break; 3276d56d2ceSJim Ingham 3286d56d2ceSJim Ingham case 'p': 3296d56d2ceSJim Ingham ptr_depth = Args::StringToUInt32 (option_arg, 0, 0, &success); 3306d56d2ceSJim Ingham if (!success) 3316d56d2ceSJim Ingham error.SetErrorStringWithFormat("Invalid pointer depth '%s'.\n", option_arg); 3326d56d2ceSJim Ingham break; 3336d56d2ceSJim Ingham 3346d56d2ceSJim Ingham case 'G': 335a134cc1bSGreg Clayton globals.push_back(ConstString (option_arg)); 3366d56d2ceSJim Ingham break; 3376d56d2ceSJim Ingham 3386d56d2ceSJim Ingham case 's': 3396d56d2ceSJim Ingham show_scope = true; 3406d56d2ceSJim Ingham break; 3416d56d2ceSJim Ingham 3426d56d2ceSJim Ingham default: 3436d56d2ceSJim Ingham error.SetErrorStringWithFormat("Invalid short option character '%c'.\n", short_option); 3446d56d2ceSJim Ingham break; 3456d56d2ceSJim Ingham } 3466d56d2ceSJim Ingham 3476d56d2ceSJim Ingham return error; 3486d56d2ceSJim Ingham } 3496d56d2ceSJim Ingham 3506d56d2ceSJim Ingham void 3516d56d2ceSJim Ingham ResetOptionValues () 3526d56d2ceSJim Ingham { 3536d56d2ceSJim Ingham Options::ResetOptionValues(); 3546d56d2ceSJim Ingham 3556d56d2ceSJim Ingham use_objc = false; 3566d56d2ceSJim Ingham use_regex = false; 3576d56d2ceSJim Ingham show_args = true; 3586d56d2ceSJim Ingham show_locals = true; 359a134cc1bSGreg Clayton show_globals = false; 360b6e8cf96SGreg Clayton show_types = false; 3616d56d2ceSJim Ingham show_scope = false; 3626d56d2ceSJim Ingham show_summary = true; 3636d56d2ceSJim Ingham show_location = false; 364a134cc1bSGreg Clayton show_decl = false; 3656d56d2ceSJim Ingham debug = false; 3668f92f0a3SGreg Clayton flat_output = false; 3676d56d2ceSJim Ingham max_depth = UINT32_MAX; 3686d56d2ceSJim Ingham ptr_depth = 0; 3696d56d2ceSJim Ingham globals.clear(); 3706d56d2ceSJim Ingham } 3716d56d2ceSJim Ingham 3726d56d2ceSJim Ingham const lldb::OptionDefinition* 3736d56d2ceSJim Ingham GetDefinitions () 3746d56d2ceSJim Ingham { 3756d56d2ceSJim Ingham return g_option_table; 3766d56d2ceSJim Ingham } 3776d56d2ceSJim Ingham 3786d56d2ceSJim Ingham // Options table: Required for subclasses of Options. 3796d56d2ceSJim Ingham 3806d56d2ceSJim Ingham static lldb::OptionDefinition g_option_table[]; 381a134cc1bSGreg Clayton bool use_objc:1, 382a134cc1bSGreg Clayton use_regex:1, 383a134cc1bSGreg Clayton show_args:1, 384a134cc1bSGreg Clayton show_locals:1, 385a134cc1bSGreg Clayton show_globals:1, 386a134cc1bSGreg Clayton show_types:1, 387a134cc1bSGreg Clayton show_scope:1, 388a134cc1bSGreg Clayton show_summary:1, 389a134cc1bSGreg Clayton show_location:1, 390a134cc1bSGreg Clayton show_decl:1, 3918f92f0a3SGreg Clayton debug:1, 3928f92f0a3SGreg Clayton flat_output:1; 3936d56d2ceSJim Ingham uint32_t max_depth; // The depth to print when dumping concrete (not pointers) aggreate values 3946d56d2ceSJim Ingham uint32_t ptr_depth; // The default depth that is dumped when we find pointers 3956d56d2ceSJim Ingham std::vector<ConstString> globals; 3966d56d2ceSJim Ingham // Instance variables to hold the values for command options. 3976d56d2ceSJim Ingham }; 3986d56d2ceSJim Ingham 399a7015092SGreg Clayton CommandObjectFrameVariable (CommandInterpreter &interpreter) : 400a7015092SGreg Clayton CommandObject (interpreter, 4016d56d2ceSJim Ingham "frame variable", 402ed8a705cSGreg Clayton "Show frame variables. All argument and local variables " 403ed8a705cSGreg Clayton "that are in scope will be shown when no arguments are given. " 404ed8a705cSGreg Clayton "If any arguments are specified, they can be names of " 405ed8a705cSGreg Clayton "argument, local, file static and file global variables. " 406ed8a705cSGreg Clayton "Children of aggregate variables can be specified such as " 407ed8a705cSGreg Clayton "'var->child.x'.", 408ff471a94SJim Ingham NULL, 409ff471a94SJim Ingham eFlagProcessMustBeLaunched | eFlagProcessMustBePaused) 4106d56d2ceSJim Ingham { 411405fe67fSCaroline Tice CommandArgumentEntry arg; 412405fe67fSCaroline Tice CommandArgumentData var_name_arg; 413405fe67fSCaroline Tice 414405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 415405fe67fSCaroline Tice var_name_arg.arg_type = eArgTypeVarName; 416405fe67fSCaroline Tice var_name_arg.arg_repetition = eArgRepeatStar; 417405fe67fSCaroline Tice 418405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 419405fe67fSCaroline Tice arg.push_back (var_name_arg); 420405fe67fSCaroline Tice 421405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 422405fe67fSCaroline Tice m_arguments.push_back (arg); 4236d56d2ceSJim Ingham } 4246d56d2ceSJim Ingham 4256d56d2ceSJim Ingham virtual 4266d56d2ceSJim Ingham ~CommandObjectFrameVariable () 4276d56d2ceSJim Ingham { 4286d56d2ceSJim Ingham } 4296d56d2ceSJim Ingham 4306d56d2ceSJim Ingham virtual 4316d56d2ceSJim Ingham Options * 4326d56d2ceSJim Ingham GetOptions () 4336d56d2ceSJim Ingham { 4346d56d2ceSJim Ingham return &m_options; 4356d56d2ceSJim Ingham } 4366d56d2ceSJim Ingham 4376d56d2ceSJim Ingham 4386d56d2ceSJim Ingham virtual bool 4396d56d2ceSJim Ingham Execute 4406d56d2ceSJim Ingham ( 4416d56d2ceSJim Ingham Args& command, 4426d56d2ceSJim Ingham CommandReturnObject &result 4436d56d2ceSJim Ingham ) 4446d56d2ceSJim Ingham { 445a7015092SGreg Clayton ExecutionContext exe_ctx(m_interpreter.GetDebugger().GetExecutionContext()); 4466d56d2ceSJim Ingham if (exe_ctx.frame == NULL) 4476d56d2ceSJim Ingham { 448340b2baaSGreg Clayton result.AppendError ("you must be stopped in a valid stack frame to view frame variables."); 4496d56d2ceSJim Ingham result.SetStatus (eReturnStatusFailed); 4506d56d2ceSJim Ingham return false; 4516d56d2ceSJim Ingham } 4526d56d2ceSJim Ingham else 4536d56d2ceSJim Ingham { 454a134cc1bSGreg Clayton Stream &s = result.GetOutputStream(); 4556d56d2ceSJim Ingham 456a134cc1bSGreg Clayton bool get_file_globals = true; 457a134cc1bSGreg Clayton VariableList *variable_list = exe_ctx.frame->GetVariableList (get_file_globals); 458a134cc1bSGreg Clayton 4596d56d2ceSJim Ingham VariableSP var_sp; 4606d56d2ceSJim Ingham ValueObjectSP valobj_sp; 4616d56d2ceSJim Ingham //ValueObjectList &valobj_list = exe_ctx.frame->GetValueObjectList(); 4626d56d2ceSJim Ingham const char *name_cstr = NULL; 4636d56d2ceSJim Ingham size_t idx; 4646d56d2ceSJim Ingham if (!m_options.globals.empty()) 4656d56d2ceSJim Ingham { 4666d56d2ceSJim Ingham uint32_t fail_count = 0; 4676d56d2ceSJim Ingham if (exe_ctx.target) 4686d56d2ceSJim Ingham { 4696d56d2ceSJim Ingham const size_t num_globals = m_options.globals.size(); 4706d56d2ceSJim Ingham for (idx = 0; idx < num_globals; ++idx) 4716d56d2ceSJim Ingham { 4726d56d2ceSJim Ingham VariableList global_var_list; 4736d56d2ceSJim Ingham const uint32_t num_matching_globals = exe_ctx.target->GetImages().FindGlobalVariables (m_options.globals[idx], true, UINT32_MAX, global_var_list); 4746d56d2ceSJim Ingham 4756d56d2ceSJim Ingham if (num_matching_globals == 0) 4766d56d2ceSJim Ingham { 4776d56d2ceSJim Ingham ++fail_count; 4786d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: can't find global variable '%s'\n", m_options.globals[idx].AsCString()); 4796d56d2ceSJim Ingham } 4806d56d2ceSJim Ingham else 4816d56d2ceSJim Ingham { 4826d56d2ceSJim Ingham for (uint32_t global_idx=0; global_idx<num_matching_globals; ++global_idx) 4836d56d2ceSJim Ingham { 4846d56d2ceSJim Ingham var_sp = global_var_list.GetVariableAtIndex(global_idx); 4856d56d2ceSJim Ingham if (var_sp) 4866d56d2ceSJim Ingham { 487288bdf9cSGreg Clayton valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp); 4886d56d2ceSJim Ingham if (!valobj_sp) 489288bdf9cSGreg Clayton valobj_sp = exe_ctx.frame->TrackGlobalVariable (var_sp); 4906d56d2ceSJim Ingham 4916d56d2ceSJim Ingham if (valobj_sp) 4926d56d2ceSJim Ingham { 493a134cc1bSGreg Clayton if (m_options.show_decl && var_sp->GetDeclaration ().GetFile()) 494a134cc1bSGreg Clayton { 4956f00abd5SGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 4966f00abd5SGreg Clayton s.PutCString (": "); 497a134cc1bSGreg Clayton } 498a134cc1bSGreg Clayton 4991d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 5006f00abd5SGreg Clayton exe_ctx.frame, 5016f00abd5SGreg Clayton valobj_sp.get(), 5026f00abd5SGreg Clayton name_cstr, 5036f00abd5SGreg Clayton m_options.ptr_depth, 5046f00abd5SGreg Clayton 0, 5056f00abd5SGreg Clayton m_options.max_depth, 5061d3afba3SGreg Clayton m_options.show_types, 5071d3afba3SGreg Clayton m_options.show_location, 5086f00abd5SGreg Clayton m_options.use_objc, 5098f92f0a3SGreg Clayton false, 5108f92f0a3SGreg Clayton m_options.flat_output); 5116d56d2ceSJim Ingham } 5126d56d2ceSJim Ingham } 5136d56d2ceSJim Ingham } 5146d56d2ceSJim Ingham } 5156d56d2ceSJim Ingham } 5166d56d2ceSJim Ingham } 5176d56d2ceSJim Ingham if (fail_count) 5186d56d2ceSJim Ingham result.SetStatus (eReturnStatusFailed); 5196d56d2ceSJim Ingham } 5209df87c17SGreg Clayton else if (variable_list) 5219df87c17SGreg Clayton { 5229df87c17SGreg Clayton if (command.GetArgumentCount() > 0) 5236d56d2ceSJim Ingham { 52446747022SGreg Clayton VariableList regex_var_list; 52546747022SGreg Clayton 5266d56d2ceSJim Ingham // If we have any args to the variable command, we will make 5276d56d2ceSJim Ingham // variable objects from them... 5286d56d2ceSJim Ingham for (idx = 0; (name_cstr = command.GetArgumentAtIndex(idx)) != NULL; ++idx) 5296d56d2ceSJim Ingham { 5306d56d2ceSJim Ingham uint32_t ptr_depth = m_options.ptr_depth; 53146747022SGreg Clayton 53246747022SGreg Clayton if (m_options.use_regex) 53346747022SGreg Clayton { 53446747022SGreg Clayton const uint32_t regex_start_index = regex_var_list.GetSize(); 53546747022SGreg Clayton RegularExpression regex (name_cstr); 53646747022SGreg Clayton if (regex.Compile(name_cstr)) 53746747022SGreg Clayton { 53846747022SGreg Clayton size_t num_matches = 0; 53946747022SGreg Clayton const size_t num_new_regex_vars = variable_list->AppendVariablesIfUnique(regex, regex_var_list, num_matches); 54046747022SGreg Clayton if (num_new_regex_vars > 0) 54146747022SGreg Clayton { 54246747022SGreg Clayton for (uint32_t regex_idx = regex_start_index, end_index = regex_var_list.GetSize(); 54346747022SGreg Clayton regex_idx < end_index; 54446747022SGreg Clayton ++regex_idx) 54546747022SGreg Clayton { 54646747022SGreg Clayton var_sp = regex_var_list.GetVariableAtIndex (regex_idx); 54746747022SGreg Clayton if (var_sp) 54846747022SGreg Clayton { 54946747022SGreg Clayton valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp); 55046747022SGreg Clayton if (valobj_sp) 55146747022SGreg Clayton { 55246747022SGreg Clayton if (m_options.show_decl && var_sp->GetDeclaration ().GetFile()) 55346747022SGreg Clayton { 55446747022SGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 55546747022SGreg Clayton s.PutCString (": "); 55646747022SGreg Clayton } 55746747022SGreg Clayton 55846747022SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 55946747022SGreg Clayton exe_ctx.frame, 56046747022SGreg Clayton valobj_sp.get(), 56146747022SGreg Clayton var_sp->GetName().AsCString(), 56246747022SGreg Clayton m_options.ptr_depth, 56346747022SGreg Clayton 0, 56446747022SGreg Clayton m_options.max_depth, 56546747022SGreg Clayton m_options.show_types, 56646747022SGreg Clayton m_options.show_location, 56746747022SGreg Clayton m_options.use_objc, 5688f92f0a3SGreg Clayton false, 5698f92f0a3SGreg Clayton m_options.flat_output); 57046747022SGreg Clayton } 57146747022SGreg Clayton } 57246747022SGreg Clayton } 57346747022SGreg Clayton } 57446747022SGreg Clayton else if (num_matches == 0) 57546747022SGreg Clayton { 57646747022SGreg Clayton result.GetErrorStream().Printf ("error: no variables matched the regular expression '%s'.\n", name_cstr); 57746747022SGreg Clayton } 57846747022SGreg Clayton } 57946747022SGreg Clayton else 58046747022SGreg Clayton { 58146747022SGreg Clayton char regex_error[1024]; 58246747022SGreg Clayton if (regex.GetErrorAsCString(regex_error, sizeof(regex_error))) 58346747022SGreg Clayton result.GetErrorStream().Printf ("error: %s\n", regex_error); 58446747022SGreg Clayton else 58546747022SGreg Clayton result.GetErrorStream().Printf ("error: unkown regex error when compiling '%s'\n", name_cstr); 58646747022SGreg Clayton } 58746747022SGreg Clayton } 58846747022SGreg Clayton else 58946747022SGreg Clayton { 59054979cddSGreg Clayton Error error; 591*6d5e68eaSGreg Clayton const uint32_t expr_path_options = StackFrame::eExpressionPathOptionCheckPtrVsMember; 592*6d5e68eaSGreg Clayton valobj_sp = exe_ctx.frame->GetValueForVariableExpressionPath (name_cstr, expr_path_options, error); 5936d56d2ceSJim Ingham if (valobj_sp) 5946d56d2ceSJim Ingham { 595a134cc1bSGreg Clayton if (m_options.show_decl && var_sp->GetDeclaration ().GetFile()) 596a134cc1bSGreg Clayton { 597a134cc1bSGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 598a134cc1bSGreg Clayton s.PutCString (": "); 599a134cc1bSGreg Clayton } 6001d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 601a134cc1bSGreg Clayton exe_ctx.frame, 602a134cc1bSGreg Clayton valobj_sp.get(), 60383c5cd9dSGreg Clayton valobj_sp->GetParent() ? name_cstr : NULL, 604a134cc1bSGreg Clayton ptr_depth, 605a134cc1bSGreg Clayton 0, 606a134cc1bSGreg Clayton m_options.max_depth, 6071d3afba3SGreg Clayton m_options.show_types, 6081d3afba3SGreg Clayton m_options.show_location, 6096f00abd5SGreg Clayton m_options.use_objc, 6108f92f0a3SGreg Clayton false, 6118f92f0a3SGreg Clayton m_options.flat_output); 6126d56d2ceSJim Ingham } 6136d56d2ceSJim Ingham else 6146d56d2ceSJim Ingham { 61554979cddSGreg Clayton const char *error_cstr = error.AsCString(NULL); 61654979cddSGreg Clayton if (error_cstr) 61754979cddSGreg Clayton result.GetErrorStream().Printf("error: %s\n", error_cstr); 61854979cddSGreg Clayton else 61954979cddSGreg Clayton result.GetErrorStream().Printf ("error: unable to find any variable expression path that matches '%s'\n", name_cstr); 6206d56d2ceSJim Ingham } 6216d56d2ceSJim Ingham } 6226d56d2ceSJim Ingham } 62346747022SGreg Clayton } 6246d56d2ceSJim Ingham else 6256d56d2ceSJim Ingham { 626a134cc1bSGreg Clayton const uint32_t num_variables = variable_list->GetSize(); 6276d56d2ceSJim Ingham 6286d56d2ceSJim Ingham if (num_variables > 0) 6296d56d2ceSJim Ingham { 6306d56d2ceSJim Ingham for (uint32_t i=0; i<num_variables; i++) 6316d56d2ceSJim Ingham { 632a134cc1bSGreg Clayton VariableSP var_sp (variable_list->GetVariableAtIndex(i)); 6336d56d2ceSJim Ingham bool dump_variable = true; 6346d56d2ceSJim Ingham 635a134cc1bSGreg Clayton switch (var_sp->GetScope()) 6366d56d2ceSJim Ingham { 6376d56d2ceSJim Ingham case eValueTypeVariableGlobal: 6386d56d2ceSJim Ingham dump_variable = m_options.show_globals; 6396d56d2ceSJim Ingham if (dump_variable && m_options.show_scope) 640a134cc1bSGreg Clayton s.PutCString("GLOBAL: "); 6416d56d2ceSJim Ingham break; 6426d56d2ceSJim Ingham 6436d56d2ceSJim Ingham case eValueTypeVariableStatic: 6446d56d2ceSJim Ingham dump_variable = m_options.show_globals; 6456d56d2ceSJim Ingham if (dump_variable && m_options.show_scope) 646a134cc1bSGreg Clayton s.PutCString("STATIC: "); 6476d56d2ceSJim Ingham break; 6486d56d2ceSJim Ingham 6496d56d2ceSJim Ingham case eValueTypeVariableArgument: 6506d56d2ceSJim Ingham dump_variable = m_options.show_args; 6516d56d2ceSJim Ingham if (dump_variable && m_options.show_scope) 652a134cc1bSGreg Clayton s.PutCString(" ARG: "); 6536d56d2ceSJim Ingham break; 6546d56d2ceSJim Ingham 6556d56d2ceSJim Ingham case eValueTypeVariableLocal: 6566d56d2ceSJim Ingham dump_variable = m_options.show_locals; 6576d56d2ceSJim Ingham if (dump_variable && m_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... 671a134cc1bSGreg Clayton valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp); 672a134cc1bSGreg Clayton if (valobj_sp) 673a134cc1bSGreg Clayton { 6746f00abd5SGreg Clayton // When dumping all variables, don't print any variables 6756f00abd5SGreg Clayton // that are not in scope to avoid extra unneeded output 6766f00abd5SGreg Clayton if (valobj_sp->IsInScope (exe_ctx.frame)) 6776f00abd5SGreg Clayton { 678a134cc1bSGreg Clayton if (m_options.show_decl && var_sp->GetDeclaration ().GetFile()) 679a134cc1bSGreg Clayton { 680a134cc1bSGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 681a134cc1bSGreg Clayton s.PutCString (": "); 682a134cc1bSGreg Clayton } 6831d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 684a134cc1bSGreg Clayton exe_ctx.frame, 685a134cc1bSGreg Clayton valobj_sp.get(), 686a134cc1bSGreg Clayton name_cstr, 687a134cc1bSGreg Clayton m_options.ptr_depth, 688a134cc1bSGreg Clayton 0, 689a134cc1bSGreg Clayton m_options.max_depth, 6901d3afba3SGreg Clayton m_options.show_types, 6911d3afba3SGreg Clayton m_options.show_location, 6926f00abd5SGreg Clayton m_options.use_objc, 6938f92f0a3SGreg Clayton false, 6948f92f0a3SGreg Clayton m_options.flat_output); 695a134cc1bSGreg Clayton } 696a134cc1bSGreg Clayton } 6976d56d2ceSJim Ingham } 6986d56d2ceSJim Ingham } 6996d56d2ceSJim Ingham } 7006f00abd5SGreg Clayton } 7016d56d2ceSJim Ingham result.SetStatus (eReturnStatusSuccessFinishResult); 7026d56d2ceSJim Ingham } 7039df87c17SGreg Clayton } 7046d56d2ceSJim Ingham return result.Succeeded(); 7056d56d2ceSJim Ingham } 7066d56d2ceSJim Ingham protected: 7076d56d2ceSJim Ingham 7086d56d2ceSJim Ingham CommandOptions m_options; 7096d56d2ceSJim Ingham }; 7106d56d2ceSJim Ingham 7116d56d2ceSJim Ingham lldb::OptionDefinition 7126d56d2ceSJim Ingham CommandObjectFrameVariable::CommandOptions::g_option_table[] = 7136d56d2ceSJim Ingham { 714deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "debug", 'D', no_argument, NULL, 0, eArgTypeNone, "Enable verbose debug information."}, 715deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "depth", 'd', required_argument, NULL, 0, eArgTypeCount, "Set the max recurse depth when dumping aggregate types (default is infinity)."}, 716deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "show-globals",'g', no_argument, NULL, 0, eArgTypeNone, "Show the current frame source file global and static variables."}, 717deaab222SCaroline 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)."}, 718deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "location", 'L', no_argument, NULL, 0, eArgTypeNone, "Show variable location information."}, 719deaab222SCaroline Tice { 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)."}, 720deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-args", 'a', no_argument, NULL, 0, eArgTypeNone, "Omit function arguments."}, 721deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-locals", 'l', no_argument, NULL, 0, eArgTypeNone, "Omit local variables."}, 722b6e8cf96SGreg Clayton { LLDB_OPT_SET_1, false, "show-types", 't', no_argument, NULL, 0, eArgTypeNone, "Show variable types when dumping values."}, 723deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-summary", 'y', no_argument, NULL, 0, eArgTypeNone, "Omit summary information."}, 724deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "scope", 's', no_argument, NULL, 0, eArgTypeNone, "Show variable scope (argument, local, global, static)."}, 725bcf1217eSGreg Clayton { LLDB_OPT_SET_1, false, "objc", 'o', no_argument, NULL, 0, eArgTypeNone, "When looking up a variable by name, print as an Objective-C object."}, 726deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "ptr-depth", 'p', required_argument, NULL, 0, eArgTypeCount, "The number of pointers to be traversed when dumping values (default is zero)."}, 72746747022SGreg Clayton { LLDB_OPT_SET_1, false, "regex", 'r', no_argument, NULL, 0, eArgTypeRegularExpression, "The <variable-name> argument for name lookups are regular expressions."}, 7288f92f0a3SGreg Clayton { LLDB_OPT_SET_1, false, "flat", 'f', no_argument, NULL, 0, eArgTypeNone, "Display results in a flat format that uses expression paths for each variable or member."}, 729deaab222SCaroline Tice { 0, false, NULL, 0, 0, NULL, NULL, eArgTypeNone, NULL } 7306d56d2ceSJim Ingham }; 73130fdc8d8SChris Lattner #pragma mark CommandObjectMultiwordFrame 73230fdc8d8SChris Lattner 73330fdc8d8SChris Lattner //------------------------------------------------------------------------- 73430fdc8d8SChris Lattner // CommandObjectMultiwordFrame 73530fdc8d8SChris Lattner //------------------------------------------------------------------------- 73630fdc8d8SChris Lattner 7376611103cSGreg Clayton CommandObjectMultiwordFrame::CommandObjectMultiwordFrame (CommandInterpreter &interpreter) : 738a7015092SGreg Clayton CommandObjectMultiword (interpreter, 739a7015092SGreg Clayton "frame", 74030fdc8d8SChris Lattner "A set of commands for operating on the current thread's frames.", 74130fdc8d8SChris Lattner "frame <subcommand> [<subcommand-options>]") 74230fdc8d8SChris Lattner { 743a7015092SGreg Clayton LoadSubCommand ("info", CommandObjectSP (new CommandObjectFrameInfo (interpreter))); 744a7015092SGreg Clayton LoadSubCommand ("select", CommandObjectSP (new CommandObjectFrameSelect (interpreter))); 745a7015092SGreg Clayton LoadSubCommand ("variable", CommandObjectSP (new CommandObjectFrameVariable (interpreter))); 74630fdc8d8SChris Lattner } 74730fdc8d8SChris Lattner 74830fdc8d8SChris Lattner CommandObjectMultiwordFrame::~CommandObjectMultiwordFrame () 74930fdc8d8SChris Lattner { 75030fdc8d8SChris Lattner } 75130fdc8d8SChris Lattner 752