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 97a7015092SGreg Clayton CommandObjectFrameSelect (CommandInterpreter &interpreter) : 98a7015092SGreg Clayton CommandObject (interpreter, 99a7015092SGreg Clayton "frame select", 100e3d26315SCaroline Tice "Select a frame by index from within the current thread and make it the current frame.", 101405fe67fSCaroline Tice NULL, 10230fdc8d8SChris Lattner eFlagProcessMustBeLaunched | eFlagProcessMustBePaused) 10330fdc8d8SChris Lattner { 104405fe67fSCaroline Tice CommandArgumentEntry arg; 105405fe67fSCaroline Tice CommandArgumentData index_arg; 106405fe67fSCaroline Tice 107405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 108405fe67fSCaroline Tice index_arg.arg_type = eArgTypeFrameIndex; 109405fe67fSCaroline Tice index_arg.arg_repetition = eArgRepeatPlain; 110405fe67fSCaroline Tice 111405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 112405fe67fSCaroline Tice arg.push_back (index_arg); 113405fe67fSCaroline Tice 114405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 115405fe67fSCaroline Tice m_arguments.push_back (arg); 11630fdc8d8SChris Lattner } 11730fdc8d8SChris Lattner 11830fdc8d8SChris Lattner ~CommandObjectFrameSelect () 11930fdc8d8SChris Lattner { 12030fdc8d8SChris Lattner } 12130fdc8d8SChris Lattner 12230fdc8d8SChris Lattner bool 123a7015092SGreg Clayton Execute (Args& command, 12430fdc8d8SChris Lattner CommandReturnObject &result) 12530fdc8d8SChris Lattner { 126a7015092SGreg Clayton ExecutionContext exe_ctx (m_interpreter.GetDebugger().GetExecutionContext()); 12730fdc8d8SChris Lattner if (exe_ctx.thread) 12830fdc8d8SChris Lattner { 12930fdc8d8SChris Lattner if (command.GetArgumentCount() == 1) 13030fdc8d8SChris Lattner { 13130fdc8d8SChris Lattner const char *frame_idx_cstr = command.GetArgumentAtIndex(0); 13230fdc8d8SChris Lattner 13330fdc8d8SChris Lattner const uint32_t num_frames = exe_ctx.thread->GetStackFrameCount(); 13430fdc8d8SChris Lattner const uint32_t frame_idx = Args::StringToUInt32 (frame_idx_cstr, UINT32_MAX, 0); 13530fdc8d8SChris Lattner if (frame_idx < num_frames) 13630fdc8d8SChris Lattner { 1372976d00aSJim Ingham exe_ctx.thread->SetSelectedFrameByIndex (frame_idx); 1382976d00aSJim Ingham exe_ctx.frame = exe_ctx.thread->GetSelectedFrame ().get(); 13930fdc8d8SChris Lattner 14030fdc8d8SChris Lattner if (exe_ctx.frame) 14130fdc8d8SChris Lattner { 142e40e4218SJim Ingham bool already_shown = false; 143e40e4218SJim Ingham SymbolContext frame_sc(exe_ctx.frame->GetSymbolContext(eSymbolContextLineEntry)); 144daccaa9eSCaroline Tice if (m_interpreter.GetDebugger().GetUseExternalEditor() && frame_sc.line_entry.file && frame_sc.line_entry.line != 0) 145e40e4218SJim Ingham { 146e40e4218SJim Ingham already_shown = Host::OpenFileInExternalEditor (frame_sc.line_entry.file, frame_sc.line_entry.line); 147e40e4218SJim Ingham } 148e40e4218SJim Ingham 14930fdc8d8SChris Lattner if (DisplayFrameForExecutionContext (exe_ctx.thread, 15030fdc8d8SChris Lattner exe_ctx.frame, 151a7015092SGreg Clayton m_interpreter, 15230fdc8d8SChris Lattner result.GetOutputStream(), 15330fdc8d8SChris Lattner true, 154e40e4218SJim Ingham !already_shown, 15530fdc8d8SChris Lattner 3, 15630fdc8d8SChris Lattner 3)) 15730fdc8d8SChris Lattner { 15830fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 15930fdc8d8SChris Lattner return result.Succeeded(); 16030fdc8d8SChris Lattner } 16130fdc8d8SChris Lattner } 16230fdc8d8SChris Lattner } 16330fdc8d8SChris Lattner if (frame_idx == UINT32_MAX) 16430fdc8d8SChris Lattner result.AppendErrorWithFormat ("Invalid frame index: %s.\n", frame_idx_cstr); 16530fdc8d8SChris Lattner else 16630fdc8d8SChris Lattner result.AppendErrorWithFormat ("Frame index (%u) out of range.\n", frame_idx); 16730fdc8d8SChris Lattner } 16830fdc8d8SChris Lattner else 16930fdc8d8SChris Lattner { 17030fdc8d8SChris Lattner result.AppendError ("invalid arguments"); 17130fdc8d8SChris Lattner result.AppendErrorWithFormat ("Usage: %s\n", m_cmd_syntax.c_str()); 17230fdc8d8SChris Lattner } 17330fdc8d8SChris Lattner } 17430fdc8d8SChris Lattner else 17530fdc8d8SChris Lattner { 17630fdc8d8SChris Lattner result.AppendError ("no current thread"); 17730fdc8d8SChris Lattner } 17830fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 17930fdc8d8SChris Lattner return false; 18030fdc8d8SChris Lattner } 18130fdc8d8SChris Lattner }; 18230fdc8d8SChris Lattner 1836d56d2ceSJim Ingham #pragma mark CommandObjectFrameVariable 1846d56d2ceSJim Ingham //---------------------------------------------------------------------- 1856d56d2ceSJim Ingham // List images with associated information 1866d56d2ceSJim Ingham //---------------------------------------------------------------------- 1876d56d2ceSJim Ingham class CommandObjectFrameVariable : public CommandObject 1886d56d2ceSJim Ingham { 1896d56d2ceSJim Ingham public: 1906d56d2ceSJim Ingham 1916d56d2ceSJim Ingham class CommandOptions : public Options 1926d56d2ceSJim Ingham { 1936d56d2ceSJim Ingham public: 1946d56d2ceSJim Ingham 1956d56d2ceSJim Ingham CommandOptions () : 1966d56d2ceSJim Ingham Options() 1976d56d2ceSJim Ingham { 1986d56d2ceSJim Ingham ResetOptionValues (); 1996d56d2ceSJim Ingham } 2006d56d2ceSJim Ingham 2016d56d2ceSJim Ingham virtual 2026d56d2ceSJim Ingham ~CommandOptions () 2036d56d2ceSJim Ingham { 2046d56d2ceSJim Ingham } 2056d56d2ceSJim Ingham 2066d56d2ceSJim Ingham virtual Error 2076d56d2ceSJim Ingham SetOptionValue (int option_idx, const char *option_arg) 2086d56d2ceSJim Ingham { 2096d56d2ceSJim Ingham Error error; 2106d56d2ceSJim Ingham bool success; 2116d56d2ceSJim Ingham char short_option = (char) m_getopt_table[option_idx].val; 2126d56d2ceSJim Ingham switch (short_option) 2136d56d2ceSJim Ingham { 2146d56d2ceSJim Ingham case 'o': use_objc = true; break; 2156d56d2ceSJim Ingham case 'n': name = option_arg; break; 2166d56d2ceSJim Ingham case 'r': use_regex = true; break; 2176d56d2ceSJim Ingham case 'a': show_args = false; break; 2186d56d2ceSJim Ingham case 'l': show_locals = false; break; 219a134cc1bSGreg Clayton case 'g': show_globals = true; break; 2206d56d2ceSJim Ingham case 't': show_types = false; break; 2216d56d2ceSJim Ingham case 'y': show_summary = false; break; 2226d56d2ceSJim Ingham case 'L': show_location= true; break; 223a134cc1bSGreg Clayton case 'c': show_decl = true; break; 2246d56d2ceSJim Ingham case 'D': debug = true; break; 2256d56d2ceSJim Ingham case 'd': 2266d56d2ceSJim Ingham max_depth = Args::StringToUInt32 (option_arg, UINT32_MAX, 0, &success); 2276d56d2ceSJim Ingham if (!success) 2286d56d2ceSJim Ingham error.SetErrorStringWithFormat("Invalid max depth '%s'.\n", option_arg); 2296d56d2ceSJim Ingham break; 2306d56d2ceSJim Ingham 2316d56d2ceSJim Ingham case 'p': 2326d56d2ceSJim Ingham ptr_depth = Args::StringToUInt32 (option_arg, 0, 0, &success); 2336d56d2ceSJim Ingham if (!success) 2346d56d2ceSJim Ingham error.SetErrorStringWithFormat("Invalid pointer depth '%s'.\n", option_arg); 2356d56d2ceSJim Ingham break; 2366d56d2ceSJim Ingham 2376d56d2ceSJim Ingham case 'G': 238a134cc1bSGreg Clayton globals.push_back(ConstString (option_arg)); 2396d56d2ceSJim Ingham break; 2406d56d2ceSJim Ingham 2416d56d2ceSJim Ingham case 's': 2426d56d2ceSJim Ingham show_scope = true; 2436d56d2ceSJim Ingham break; 2446d56d2ceSJim Ingham 2456d56d2ceSJim Ingham default: 2466d56d2ceSJim Ingham error.SetErrorStringWithFormat("Invalid short option character '%c'.\n", short_option); 2476d56d2ceSJim Ingham break; 2486d56d2ceSJim Ingham } 2496d56d2ceSJim Ingham 2506d56d2ceSJim Ingham return error; 2516d56d2ceSJim Ingham } 2526d56d2ceSJim Ingham 2536d56d2ceSJim Ingham void 2546d56d2ceSJim Ingham ResetOptionValues () 2556d56d2ceSJim Ingham { 2566d56d2ceSJim Ingham Options::ResetOptionValues(); 2576d56d2ceSJim Ingham 2586d56d2ceSJim Ingham name.clear(); 2596d56d2ceSJim Ingham use_objc = false; 2606d56d2ceSJim Ingham use_regex = false; 2616d56d2ceSJim Ingham show_args = true; 2626d56d2ceSJim Ingham show_locals = true; 263a134cc1bSGreg Clayton show_globals = false; 2646d56d2ceSJim Ingham show_types = true; 2656d56d2ceSJim Ingham show_scope = false; 2666d56d2ceSJim Ingham show_summary = true; 2676d56d2ceSJim Ingham show_location = false; 268a134cc1bSGreg Clayton show_decl = false; 2696d56d2ceSJim Ingham debug = false; 2706d56d2ceSJim Ingham max_depth = UINT32_MAX; 2716d56d2ceSJim Ingham ptr_depth = 0; 2726d56d2ceSJim Ingham globals.clear(); 2736d56d2ceSJim Ingham } 2746d56d2ceSJim Ingham 2756d56d2ceSJim Ingham const lldb::OptionDefinition* 2766d56d2ceSJim Ingham GetDefinitions () 2776d56d2ceSJim Ingham { 2786d56d2ceSJim Ingham return g_option_table; 2796d56d2ceSJim Ingham } 2806d56d2ceSJim Ingham 2816d56d2ceSJim Ingham // Options table: Required for subclasses of Options. 2826d56d2ceSJim Ingham 2836d56d2ceSJim Ingham static lldb::OptionDefinition g_option_table[]; 2846d56d2ceSJim Ingham std::string name; 285a134cc1bSGreg Clayton bool use_objc:1, 286a134cc1bSGreg Clayton use_regex:1, 287a134cc1bSGreg Clayton show_args:1, 288a134cc1bSGreg Clayton show_locals:1, 289a134cc1bSGreg Clayton show_globals:1, 290a134cc1bSGreg Clayton show_types:1, 291a134cc1bSGreg Clayton show_scope:1, 292a134cc1bSGreg Clayton show_summary:1, 293a134cc1bSGreg Clayton show_location:1, 294a134cc1bSGreg Clayton show_decl:1, 295a134cc1bSGreg Clayton debug:1; 2966d56d2ceSJim Ingham uint32_t max_depth; // The depth to print when dumping concrete (not pointers) aggreate values 2976d56d2ceSJim Ingham uint32_t ptr_depth; // The default depth that is dumped when we find pointers 2986d56d2ceSJim Ingham std::vector<ConstString> globals; 2996d56d2ceSJim Ingham // Instance variables to hold the values for command options. 3006d56d2ceSJim Ingham }; 3016d56d2ceSJim Ingham 302a7015092SGreg Clayton CommandObjectFrameVariable (CommandInterpreter &interpreter) : 303a7015092SGreg Clayton CommandObject (interpreter, 3046d56d2ceSJim Ingham "frame variable", 305ed8a705cSGreg Clayton "Show frame variables. All argument and local variables " 306ed8a705cSGreg Clayton "that are in scope will be shown when no arguments are given. " 307ed8a705cSGreg Clayton "If any arguments are specified, they can be names of " 308ed8a705cSGreg Clayton "argument, local, file static and file global variables." 309ed8a705cSGreg Clayton "Children of aggregate variables can be specified such as " 310ed8a705cSGreg Clayton "'var->child.x'.", 311405fe67fSCaroline Tice NULL) 3126d56d2ceSJim Ingham { 313405fe67fSCaroline Tice CommandArgumentEntry arg; 314405fe67fSCaroline Tice CommandArgumentData var_name_arg; 315405fe67fSCaroline Tice 316405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 317405fe67fSCaroline Tice var_name_arg.arg_type = eArgTypeVarName; 318405fe67fSCaroline Tice var_name_arg.arg_repetition = eArgRepeatStar; 319405fe67fSCaroline Tice 320405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 321405fe67fSCaroline Tice arg.push_back (var_name_arg); 322405fe67fSCaroline Tice 323405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 324405fe67fSCaroline Tice m_arguments.push_back (arg); 3256d56d2ceSJim Ingham } 3266d56d2ceSJim Ingham 3276d56d2ceSJim Ingham virtual 3286d56d2ceSJim Ingham ~CommandObjectFrameVariable () 3296d56d2ceSJim Ingham { 3306d56d2ceSJim Ingham } 3316d56d2ceSJim Ingham 3326d56d2ceSJim Ingham virtual 3336d56d2ceSJim Ingham Options * 3346d56d2ceSJim Ingham GetOptions () 3356d56d2ceSJim Ingham { 3366d56d2ceSJim Ingham return &m_options; 3376d56d2ceSJim Ingham } 3386d56d2ceSJim Ingham 3396d56d2ceSJim Ingham 3406d56d2ceSJim Ingham virtual bool 3416d56d2ceSJim Ingham Execute 3426d56d2ceSJim Ingham ( 3436d56d2ceSJim Ingham Args& command, 3446d56d2ceSJim Ingham CommandReturnObject &result 3456d56d2ceSJim Ingham ) 3466d56d2ceSJim Ingham { 347a7015092SGreg Clayton ExecutionContext exe_ctx(m_interpreter.GetDebugger().GetExecutionContext()); 3486d56d2ceSJim Ingham if (exe_ctx.frame == NULL) 3496d56d2ceSJim Ingham { 350340b2baaSGreg Clayton result.AppendError ("you must be stopped in a valid stack frame to view frame variables."); 3516d56d2ceSJim Ingham result.SetStatus (eReturnStatusFailed); 3526d56d2ceSJim Ingham return false; 3536d56d2ceSJim Ingham } 3546d56d2ceSJim Ingham else 3556d56d2ceSJim Ingham { 356a134cc1bSGreg Clayton Stream &s = result.GetOutputStream(); 3576d56d2ceSJim Ingham 358a134cc1bSGreg Clayton bool get_file_globals = true; 359a134cc1bSGreg Clayton VariableList *variable_list = exe_ctx.frame->GetVariableList (get_file_globals); 360a134cc1bSGreg Clayton 3616d56d2ceSJim Ingham VariableSP var_sp; 3626d56d2ceSJim Ingham ValueObjectSP valobj_sp; 3636d56d2ceSJim Ingham //ValueObjectList &valobj_list = exe_ctx.frame->GetValueObjectList(); 3646d56d2ceSJim Ingham const char *name_cstr = NULL; 3656d56d2ceSJim Ingham size_t idx; 3666d56d2ceSJim Ingham if (!m_options.globals.empty()) 3676d56d2ceSJim Ingham { 3686d56d2ceSJim Ingham uint32_t fail_count = 0; 3696d56d2ceSJim Ingham if (exe_ctx.target) 3706d56d2ceSJim Ingham { 3716d56d2ceSJim Ingham const size_t num_globals = m_options.globals.size(); 3726d56d2ceSJim Ingham for (idx = 0; idx < num_globals; ++idx) 3736d56d2ceSJim Ingham { 3746d56d2ceSJim Ingham VariableList global_var_list; 3756d56d2ceSJim Ingham const uint32_t num_matching_globals = exe_ctx.target->GetImages().FindGlobalVariables (m_options.globals[idx], true, UINT32_MAX, global_var_list); 3766d56d2ceSJim Ingham 3776d56d2ceSJim Ingham if (num_matching_globals == 0) 3786d56d2ceSJim Ingham { 3796d56d2ceSJim Ingham ++fail_count; 3806d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: can't find global variable '%s'\n", m_options.globals[idx].AsCString()); 3816d56d2ceSJim Ingham } 3826d56d2ceSJim Ingham else 3836d56d2ceSJim Ingham { 3846d56d2ceSJim Ingham for (uint32_t global_idx=0; global_idx<num_matching_globals; ++global_idx) 3856d56d2ceSJim Ingham { 3866d56d2ceSJim Ingham var_sp = global_var_list.GetVariableAtIndex(global_idx); 3876d56d2ceSJim Ingham if (var_sp) 3886d56d2ceSJim Ingham { 389288bdf9cSGreg Clayton valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp); 3906d56d2ceSJim Ingham if (!valobj_sp) 391288bdf9cSGreg Clayton valobj_sp = exe_ctx.frame->TrackGlobalVariable (var_sp); 3926d56d2ceSJim Ingham 3936d56d2ceSJim Ingham if (valobj_sp) 3946d56d2ceSJim Ingham { 395a134cc1bSGreg Clayton if (m_options.show_decl && var_sp->GetDeclaration ().GetFile()) 396a134cc1bSGreg Clayton { 3976f00abd5SGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 3986f00abd5SGreg Clayton s.PutCString (": "); 399a134cc1bSGreg Clayton } 400a134cc1bSGreg Clayton 401*1d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 4026f00abd5SGreg Clayton exe_ctx.frame, 4036f00abd5SGreg Clayton valobj_sp.get(), 4046f00abd5SGreg Clayton name_cstr, 4056f00abd5SGreg Clayton m_options.ptr_depth, 4066f00abd5SGreg Clayton 0, 4076f00abd5SGreg Clayton m_options.max_depth, 408*1d3afba3SGreg Clayton m_options.show_types, 409*1d3afba3SGreg Clayton m_options.show_location, 4106f00abd5SGreg Clayton m_options.use_objc, 4116f00abd5SGreg Clayton false); 412a134cc1bSGreg Clayton s.EOL(); 4136d56d2ceSJim Ingham } 4146d56d2ceSJim Ingham } 4156d56d2ceSJim Ingham } 4166d56d2ceSJim Ingham } 4176d56d2ceSJim Ingham } 4186d56d2ceSJim Ingham } 4196d56d2ceSJim Ingham if (fail_count) 4206d56d2ceSJim Ingham result.SetStatus (eReturnStatusFailed); 4216d56d2ceSJim Ingham } 4229df87c17SGreg Clayton else if (variable_list) 4239df87c17SGreg Clayton { 4249df87c17SGreg Clayton if (command.GetArgumentCount() > 0) 4256d56d2ceSJim Ingham { 4266d56d2ceSJim Ingham // If we have any args to the variable command, we will make 4276d56d2ceSJim Ingham // variable objects from them... 4286d56d2ceSJim Ingham for (idx = 0; (name_cstr = command.GetArgumentAtIndex(idx)) != NULL; ++idx) 4296d56d2ceSJim Ingham { 4306d56d2ceSJim Ingham uint32_t ptr_depth = m_options.ptr_depth; 4316d56d2ceSJim Ingham // If first character is a '*', then show pointer contents 4326d56d2ceSJim Ingham if (name_cstr[0] == '*') 4336d56d2ceSJim Ingham { 4346d56d2ceSJim Ingham ++ptr_depth; 4356d56d2ceSJim Ingham name_cstr++; // Skip the '*' 4366d56d2ceSJim Ingham } 4376d56d2ceSJim Ingham 4386d56d2ceSJim Ingham std::string var_path (name_cstr); 4396d56d2ceSJim Ingham size_t separator_idx = var_path.find_first_of(".-["); 4406d56d2ceSJim Ingham 4416d56d2ceSJim Ingham ConstString name_const_string; 4426d56d2ceSJim Ingham if (separator_idx == std::string::npos) 4436d56d2ceSJim Ingham name_const_string.SetCString (var_path.c_str()); 4446d56d2ceSJim Ingham else 4456d56d2ceSJim Ingham name_const_string.SetCStringWithLength (var_path.c_str(), separator_idx); 4466d56d2ceSJim Ingham 447a134cc1bSGreg Clayton var_sp = variable_list->FindVariable(name_const_string); 4486d56d2ceSJim Ingham if (var_sp) 4496d56d2ceSJim Ingham { 450288bdf9cSGreg Clayton valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp); 4516d56d2ceSJim Ingham 4526d56d2ceSJim Ingham var_path.erase (0, name_const_string.GetLength ()); 4536d56d2ceSJim Ingham // We are dumping at least one child 4546d56d2ceSJim Ingham while (separator_idx != std::string::npos) 4556d56d2ceSJim Ingham { 4566d56d2ceSJim Ingham // Calculate the next separator index ahead of time 4576d56d2ceSJim Ingham ValueObjectSP child_valobj_sp; 4586d56d2ceSJim Ingham const char separator_type = var_path[0]; 4596d56d2ceSJim Ingham switch (separator_type) 4606d56d2ceSJim Ingham { 4616d56d2ceSJim Ingham 4626d56d2ceSJim Ingham case '-': 4636d56d2ceSJim Ingham if (var_path.size() >= 2 && var_path[1] != '>') 4646d56d2ceSJim Ingham { 4656d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: invalid character in variable path starting at '%s'\n", 4666d56d2ceSJim Ingham var_path.c_str()); 4676d56d2ceSJim Ingham var_path.clear(); 4686d56d2ceSJim Ingham valobj_sp.reset(); 4696d56d2ceSJim Ingham break; 4706d56d2ceSJim Ingham } 4716d56d2ceSJim Ingham var_path.erase (0, 1); // Remove the '-' 4726d56d2ceSJim Ingham // Fall through 4736d56d2ceSJim Ingham case '.': 4746d56d2ceSJim Ingham { 4756d56d2ceSJim Ingham var_path.erase (0, 1); // Remove the '.' or '>' 4766d56d2ceSJim Ingham separator_idx = var_path.find_first_of(".-["); 4776d56d2ceSJim Ingham ConstString child_name; 4786d56d2ceSJim Ingham if (separator_idx == std::string::npos) 4796d56d2ceSJim Ingham child_name.SetCString (var_path.c_str()); 4806d56d2ceSJim Ingham else 4816d56d2ceSJim Ingham child_name.SetCStringWithLength(var_path.c_str(), separator_idx); 4826d56d2ceSJim Ingham 4836d56d2ceSJim Ingham child_valobj_sp = valobj_sp->GetChildMemberWithName (child_name, true); 4846d56d2ceSJim Ingham if (!child_valobj_sp) 4856d56d2ceSJim Ingham { 4866d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: can't find child of '%s' named '%s'\n", 4876d56d2ceSJim Ingham valobj_sp->GetName().AsCString(), 4886d56d2ceSJim Ingham child_name.GetCString()); 4896d56d2ceSJim Ingham var_path.clear(); 4906d56d2ceSJim Ingham valobj_sp.reset(); 4916d56d2ceSJim Ingham break; 4926d56d2ceSJim Ingham } 4936d56d2ceSJim Ingham // Remove the child name from the path 4946d56d2ceSJim Ingham var_path.erase(0, child_name.GetLength()); 4956d56d2ceSJim Ingham } 4966d56d2ceSJim Ingham break; 4976d56d2ceSJim Ingham 4986d56d2ceSJim Ingham case '[': 4996d56d2ceSJim Ingham // Array member access, or treating pointer as an array 5006d56d2ceSJim Ingham if (var_path.size() > 2) // Need at least two brackets and a number 5016d56d2ceSJim Ingham { 5026d56d2ceSJim Ingham char *end = NULL; 5036d56d2ceSJim Ingham int32_t child_index = ::strtol (&var_path[1], &end, 0); 5046d56d2ceSJim Ingham if (end && *end == ']') 5056d56d2ceSJim Ingham { 5066d56d2ceSJim Ingham 5076d56d2ceSJim Ingham if (valobj_sp->IsPointerType ()) 5086d56d2ceSJim Ingham { 5096d56d2ceSJim Ingham child_valobj_sp = valobj_sp->GetSyntheticArrayMemberFromPointer (child_index, true); 5106d56d2ceSJim Ingham } 5116d56d2ceSJim Ingham else 5126d56d2ceSJim Ingham { 5136d56d2ceSJim Ingham child_valobj_sp = valobj_sp->GetChildAtIndex (child_index, true); 5146d56d2ceSJim Ingham } 5156d56d2ceSJim Ingham 5166d56d2ceSJim Ingham if (!child_valobj_sp) 5176d56d2ceSJim Ingham { 5186d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: invalid array index %u in '%s'\n", 5196d56d2ceSJim Ingham child_index, 5206d56d2ceSJim Ingham valobj_sp->GetName().AsCString()); 5216d56d2ceSJim Ingham var_path.clear(); 5226d56d2ceSJim Ingham valobj_sp.reset(); 5236d56d2ceSJim Ingham break; 5246d56d2ceSJim Ingham } 5256d56d2ceSJim Ingham 5266d56d2ceSJim Ingham // Erase the array member specification '[%i]' where %i is the array index 5276d56d2ceSJim Ingham var_path.erase(0, (end - var_path.c_str()) + 1); 5286d56d2ceSJim Ingham separator_idx = var_path.find_first_of(".-["); 5296d56d2ceSJim Ingham 5306d56d2ceSJim Ingham // Break out early from the switch since we were able to find the child member 5316d56d2ceSJim Ingham break; 5326d56d2ceSJim Ingham } 5336d56d2ceSJim Ingham } 5346d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: invalid array member specification for '%s' starting at '%s'\n", 5356d56d2ceSJim Ingham valobj_sp->GetName().AsCString(), 5366d56d2ceSJim Ingham var_path.c_str()); 5376d56d2ceSJim Ingham var_path.clear(); 5386d56d2ceSJim Ingham valobj_sp.reset(); 5396d56d2ceSJim Ingham break; 5406d56d2ceSJim Ingham 5416d56d2ceSJim Ingham break; 5426d56d2ceSJim Ingham 5436d56d2ceSJim Ingham default: 5446d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: invalid character in variable path starting at '%s'\n", 5456d56d2ceSJim Ingham var_path.c_str()); 5466d56d2ceSJim Ingham var_path.clear(); 5476d56d2ceSJim Ingham valobj_sp.reset(); 5486d56d2ceSJim Ingham separator_idx = std::string::npos; 5496d56d2ceSJim Ingham break; 5506d56d2ceSJim Ingham } 5516d56d2ceSJim Ingham 5526d56d2ceSJim Ingham if (child_valobj_sp) 5536d56d2ceSJim Ingham valobj_sp = child_valobj_sp; 5546d56d2ceSJim Ingham 5556d56d2ceSJim Ingham if (var_path.empty()) 5566d56d2ceSJim Ingham break; 5576d56d2ceSJim Ingham 5586d56d2ceSJim Ingham } 5596d56d2ceSJim Ingham 5606d56d2ceSJim Ingham if (valobj_sp) 5616d56d2ceSJim Ingham { 562a134cc1bSGreg Clayton if (m_options.show_decl && var_sp->GetDeclaration ().GetFile()) 563a134cc1bSGreg Clayton { 564a134cc1bSGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 565a134cc1bSGreg Clayton s.PutCString (": "); 566a134cc1bSGreg Clayton } 567a134cc1bSGreg Clayton 568*1d3afba3SGreg Clayton 569*1d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 570a134cc1bSGreg Clayton exe_ctx.frame, 571a134cc1bSGreg Clayton valobj_sp.get(), 572a134cc1bSGreg Clayton name_cstr, 573a134cc1bSGreg Clayton ptr_depth, 574a134cc1bSGreg Clayton 0, 575a134cc1bSGreg Clayton m_options.max_depth, 576*1d3afba3SGreg Clayton m_options.show_types, 577*1d3afba3SGreg Clayton m_options.show_location, 5786f00abd5SGreg Clayton m_options.use_objc, 5796f00abd5SGreg Clayton false); 580a134cc1bSGreg Clayton 581a134cc1bSGreg Clayton s.EOL(); 5826d56d2ceSJim Ingham } 5836d56d2ceSJim Ingham } 5846d56d2ceSJim Ingham else 5856d56d2ceSJim Ingham { 5866d56d2ceSJim Ingham result.GetErrorStream().Printf ("error: unable to find any variables named '%s'\n", name_cstr); 5876d56d2ceSJim Ingham var_path.clear(); 5886d56d2ceSJim Ingham } 5896d56d2ceSJim Ingham } 5906d56d2ceSJim Ingham } 5916d56d2ceSJim Ingham else 5926d56d2ceSJim Ingham { 593a134cc1bSGreg Clayton const uint32_t num_variables = variable_list->GetSize(); 5946d56d2ceSJim Ingham 5956d56d2ceSJim Ingham if (num_variables > 0) 5966d56d2ceSJim Ingham { 5976d56d2ceSJim Ingham for (uint32_t i=0; i<num_variables; i++) 5986d56d2ceSJim Ingham { 599a134cc1bSGreg Clayton VariableSP var_sp (variable_list->GetVariableAtIndex(i)); 6006d56d2ceSJim Ingham bool dump_variable = true; 6016d56d2ceSJim Ingham 602a134cc1bSGreg Clayton switch (var_sp->GetScope()) 6036d56d2ceSJim Ingham { 6046d56d2ceSJim Ingham case eValueTypeVariableGlobal: 6056d56d2ceSJim Ingham dump_variable = m_options.show_globals; 6066d56d2ceSJim Ingham if (dump_variable && m_options.show_scope) 607a134cc1bSGreg Clayton s.PutCString("GLOBAL: "); 6086d56d2ceSJim Ingham break; 6096d56d2ceSJim Ingham 6106d56d2ceSJim Ingham case eValueTypeVariableStatic: 6116d56d2ceSJim Ingham dump_variable = m_options.show_globals; 6126d56d2ceSJim Ingham if (dump_variable && m_options.show_scope) 613a134cc1bSGreg Clayton s.PutCString("STATIC: "); 6146d56d2ceSJim Ingham break; 6156d56d2ceSJim Ingham 6166d56d2ceSJim Ingham case eValueTypeVariableArgument: 6176d56d2ceSJim Ingham dump_variable = m_options.show_args; 6186d56d2ceSJim Ingham if (dump_variable && m_options.show_scope) 619a134cc1bSGreg Clayton s.PutCString(" ARG: "); 6206d56d2ceSJim Ingham break; 6216d56d2ceSJim Ingham 6226d56d2ceSJim Ingham case eValueTypeVariableLocal: 6236d56d2ceSJim Ingham dump_variable = m_options.show_locals; 6246d56d2ceSJim Ingham if (dump_variable && m_options.show_scope) 625a134cc1bSGreg Clayton s.PutCString(" LOCAL: "); 6266d56d2ceSJim Ingham break; 6276d56d2ceSJim Ingham 6286d56d2ceSJim Ingham default: 6296d56d2ceSJim Ingham break; 6306d56d2ceSJim Ingham } 6316d56d2ceSJim Ingham 6326d56d2ceSJim Ingham if (dump_variable) 633a134cc1bSGreg Clayton { 634a134cc1bSGreg Clayton 635a134cc1bSGreg Clayton // Use the variable object code to make sure we are 636a134cc1bSGreg Clayton // using the same APIs as the the public API will be 637a134cc1bSGreg Clayton // using... 638a134cc1bSGreg Clayton valobj_sp = exe_ctx.frame->GetValueObjectForFrameVariable (var_sp); 639a134cc1bSGreg Clayton if (valobj_sp) 640a134cc1bSGreg Clayton { 6416f00abd5SGreg Clayton // When dumping all variables, don't print any variables 6426f00abd5SGreg Clayton // that are not in scope to avoid extra unneeded output 6436f00abd5SGreg Clayton if (valobj_sp->IsInScope (exe_ctx.frame)) 6446f00abd5SGreg Clayton { 645a134cc1bSGreg Clayton if (m_options.show_decl && var_sp->GetDeclaration ().GetFile()) 646a134cc1bSGreg Clayton { 647a134cc1bSGreg Clayton var_sp->GetDeclaration ().DumpStopContext (&s, false); 648a134cc1bSGreg Clayton s.PutCString (": "); 649a134cc1bSGreg Clayton } 650*1d3afba3SGreg Clayton ValueObject::DumpValueObject (result.GetOutputStream(), 651a134cc1bSGreg Clayton exe_ctx.frame, 652a134cc1bSGreg Clayton valobj_sp.get(), 653a134cc1bSGreg Clayton name_cstr, 654a134cc1bSGreg Clayton m_options.ptr_depth, 655a134cc1bSGreg Clayton 0, 656a134cc1bSGreg Clayton m_options.max_depth, 657*1d3afba3SGreg Clayton m_options.show_types, 658*1d3afba3SGreg Clayton m_options.show_location, 6596f00abd5SGreg Clayton m_options.use_objc, 660*1d3afba3SGreg Clayton false); 661a134cc1bSGreg Clayton 662a134cc1bSGreg Clayton s.EOL(); 663a134cc1bSGreg Clayton } 664a134cc1bSGreg Clayton } 6656d56d2ceSJim Ingham } 6666d56d2ceSJim Ingham } 6676d56d2ceSJim Ingham } 6686f00abd5SGreg Clayton } 6696d56d2ceSJim Ingham result.SetStatus (eReturnStatusSuccessFinishResult); 6706d56d2ceSJim Ingham } 6719df87c17SGreg Clayton } 6726d56d2ceSJim Ingham return result.Succeeded(); 6736d56d2ceSJim Ingham } 6746d56d2ceSJim Ingham protected: 6756d56d2ceSJim Ingham 6766d56d2ceSJim Ingham CommandOptions m_options; 6776d56d2ceSJim Ingham }; 6786d56d2ceSJim Ingham 6796d56d2ceSJim Ingham lldb::OptionDefinition 6806d56d2ceSJim Ingham CommandObjectFrameVariable::CommandOptions::g_option_table[] = 6816d56d2ceSJim Ingham { 682deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "debug", 'D', no_argument, NULL, 0, eArgTypeNone, "Enable verbose debug information."}, 683deaab222SCaroline 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)."}, 684deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "show-globals",'g', no_argument, NULL, 0, eArgTypeNone, "Show the current frame source file global and static variables."}, 685deaab222SCaroline 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)."}, 686deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "location", 'L', no_argument, NULL, 0, eArgTypeNone, "Show variable location information."}, 687deaab222SCaroline 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)."}, 688deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "name", 'n', required_argument, NULL, 0, eArgTypeVarName, "Lookup a variable by name or regex (--regex) for the current execution context."}, 689deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-args", 'a', no_argument, NULL, 0, eArgTypeNone, "Omit function arguments."}, 690deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-locals", 'l', no_argument, NULL, 0, eArgTypeNone, "Omit local variables."}, 691deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-types", 't', no_argument, NULL, 0, eArgTypeNone, "Omit variable type names."}, 692deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "no-summary", 'y', no_argument, NULL, 0, eArgTypeNone, "Omit summary information."}, 693deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "scope", 's', no_argument, NULL, 0, eArgTypeNone, "Show variable scope (argument, local, global, static)."}, 694deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "objc", 'o', no_argument, NULL, 0, eArgTypeNone, "When looking up a variable by name (--name), print as an Objective-C object."}, 695deaab222SCaroline 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)."}, 696deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "regex", 'r', no_argument, NULL, 0, eArgTypeCount, "The <name> argument for name lookups are regular expressions."}, 697deaab222SCaroline Tice { 0, false, NULL, 0, 0, NULL, NULL, eArgTypeNone, NULL } 6986d56d2ceSJim Ingham }; 69930fdc8d8SChris Lattner #pragma mark CommandObjectMultiwordFrame 70030fdc8d8SChris Lattner 70130fdc8d8SChris Lattner //------------------------------------------------------------------------- 70230fdc8d8SChris Lattner // CommandObjectMultiwordFrame 70330fdc8d8SChris Lattner //------------------------------------------------------------------------- 70430fdc8d8SChris Lattner 7056611103cSGreg Clayton CommandObjectMultiwordFrame::CommandObjectMultiwordFrame (CommandInterpreter &interpreter) : 706a7015092SGreg Clayton CommandObjectMultiword (interpreter, 707a7015092SGreg Clayton "frame", 70830fdc8d8SChris Lattner "A set of commands for operating on the current thread's frames.", 70930fdc8d8SChris Lattner "frame <subcommand> [<subcommand-options>]") 71030fdc8d8SChris Lattner { 711a7015092SGreg Clayton LoadSubCommand ("info", CommandObjectSP (new CommandObjectFrameInfo (interpreter))); 712a7015092SGreg Clayton LoadSubCommand ("select", CommandObjectSP (new CommandObjectFrameSelect (interpreter))); 713a7015092SGreg Clayton LoadSubCommand ("variable", CommandObjectSP (new CommandObjectFrameVariable (interpreter))); 71430fdc8d8SChris Lattner } 71530fdc8d8SChris Lattner 71630fdc8d8SChris Lattner CommandObjectMultiwordFrame::~CommandObjectMultiwordFrame () 71730fdc8d8SChris Lattner { 71830fdc8d8SChris Lattner } 71930fdc8d8SChris Lattner 720