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
160a976141SEnrico Granata #include "lldb/Core/DataVisualization.h"
1730fdc8d8SChris Lattner #include "lldb/Core/Debugger.h"
186d56d2ceSJim Ingham #include "lldb/Core/Module.h"
196d56d2ceSJim Ingham #include "lldb/Core/StreamFile.h"
2030fdc8d8SChris Lattner #include "lldb/Core/Timer.h"
216d56d2ceSJim Ingham #include "lldb/Core/Value.h"
226d56d2ceSJim Ingham #include "lldb/Core/ValueObject.h"
236d56d2ceSJim Ingham #include "lldb/Core/ValueObjectVariable.h"
247fb56d0aSGreg Clayton #include "lldb/Host/Host.h"
256d56d2ceSJim Ingham #include "lldb/Interpreter/Args.h"
2630fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h"
2730fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h"
286d56d2ceSJim Ingham #include "lldb/Interpreter/Options.h"
292837b766SJim Ingham #include "lldb/Interpreter/OptionGroupValueObjectDisplay.h"
30715c2365SGreg Clayton #include "lldb/Interpreter/OptionGroupVariable.h"
316d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTType.h"
326d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTContext.h"
336d56d2ceSJim Ingham #include "lldb/Symbol/ObjectFile.h"
346d56d2ceSJim Ingham #include "lldb/Symbol/SymbolContext.h"
356d56d2ceSJim Ingham #include "lldb/Symbol/Type.h"
366d56d2ceSJim Ingham #include "lldb/Symbol/Variable.h"
376d56d2ceSJim Ingham #include "lldb/Symbol/VariableList.h"
3830fdc8d8SChris Lattner #include "lldb/Target/Process.h"
3930fdc8d8SChris Lattner #include "lldb/Target/StackFrame.h"
4030fdc8d8SChris Lattner #include "lldb/Target/Thread.h"
416d56d2ceSJim Ingham #include "lldb/Target/Target.h"
4230fdc8d8SChris Lattner 
4330fdc8d8SChris Lattner using namespace lldb;
4430fdc8d8SChris Lattner using namespace lldb_private;
4530fdc8d8SChris Lattner 
4630fdc8d8SChris Lattner #pragma mark CommandObjectFrameInfo
4730fdc8d8SChris Lattner 
4830fdc8d8SChris Lattner //-------------------------------------------------------------------------
4930fdc8d8SChris Lattner // CommandObjectFrameInfo
5030fdc8d8SChris Lattner //-------------------------------------------------------------------------
5130fdc8d8SChris Lattner 
5230fdc8d8SChris Lattner class CommandObjectFrameInfo : public CommandObject
5330fdc8d8SChris Lattner {
5430fdc8d8SChris Lattner public:
5530fdc8d8SChris Lattner 
56a7015092SGreg Clayton     CommandObjectFrameInfo (CommandInterpreter &interpreter) :
57a7015092SGreg Clayton         CommandObject (interpreter,
58a7015092SGreg Clayton                        "frame info",
59e3d26315SCaroline Tice                        "List information about the currently selected frame in the current thread.",
6030fdc8d8SChris Lattner                        "frame info",
6130fdc8d8SChris Lattner                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused)
6230fdc8d8SChris Lattner     {
6330fdc8d8SChris Lattner     }
6430fdc8d8SChris Lattner 
6530fdc8d8SChris Lattner     ~CommandObjectFrameInfo ()
6630fdc8d8SChris Lattner     {
6730fdc8d8SChris Lattner     }
6830fdc8d8SChris Lattner 
6930fdc8d8SChris Lattner     bool
70a7015092SGreg Clayton     Execute (Args& command,
7130fdc8d8SChris Lattner              CommandReturnObject &result)
7230fdc8d8SChris Lattner     {
738b82f087SGreg Clayton         ExecutionContext exe_ctx(m_interpreter.GetExecutionContext());
7430fdc8d8SChris Lattner         if (exe_ctx.frame)
7530fdc8d8SChris Lattner         {
760603aa9dSGreg Clayton             exe_ctx.frame->DumpUsingSettingsFormat (&result.GetOutputStream());
7730fdc8d8SChris Lattner             result.SetStatus (eReturnStatusSuccessFinishResult);
7830fdc8d8SChris Lattner         }
7930fdc8d8SChris Lattner         else
8030fdc8d8SChris Lattner         {
8130fdc8d8SChris Lattner             result.AppendError ("no current frame");
8230fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
8330fdc8d8SChris Lattner         }
8430fdc8d8SChris Lattner         return result.Succeeded();
8530fdc8d8SChris Lattner     }
8630fdc8d8SChris Lattner };
8730fdc8d8SChris Lattner 
8830fdc8d8SChris Lattner #pragma mark CommandObjectFrameSelect
8930fdc8d8SChris Lattner 
9030fdc8d8SChris Lattner //-------------------------------------------------------------------------
9130fdc8d8SChris Lattner // CommandObjectFrameSelect
9230fdc8d8SChris Lattner //-------------------------------------------------------------------------
9330fdc8d8SChris Lattner 
9430fdc8d8SChris Lattner class CommandObjectFrameSelect : public CommandObject
9530fdc8d8SChris Lattner {
9630fdc8d8SChris Lattner public:
9730fdc8d8SChris Lattner 
98864174e1SGreg Clayton    class CommandOptions : public Options
99864174e1SGreg Clayton     {
100864174e1SGreg Clayton     public:
101864174e1SGreg Clayton 
102eb0103f2SGreg Clayton         CommandOptions (CommandInterpreter &interpreter) :
103f16066e8SJohnny Chen             Options(interpreter)
104864174e1SGreg Clayton         {
105f6b8b581SGreg Clayton             OptionParsingStarting ();
106864174e1SGreg Clayton         }
107864174e1SGreg Clayton 
108864174e1SGreg Clayton         virtual
109864174e1SGreg Clayton         ~CommandOptions ()
110864174e1SGreg Clayton         {
111864174e1SGreg Clayton         }
112864174e1SGreg Clayton 
113864174e1SGreg Clayton         virtual Error
114f6b8b581SGreg Clayton         SetOptionValue (uint32_t option_idx, const char *option_arg)
115864174e1SGreg Clayton         {
116864174e1SGreg Clayton             Error error;
117864174e1SGreg Clayton             bool success = false;
118864174e1SGreg Clayton             char short_option = (char) m_getopt_table[option_idx].val;
119864174e1SGreg Clayton             switch (short_option)
120864174e1SGreg Clayton             {
121864174e1SGreg Clayton             case 'r':
122864174e1SGreg Clayton                 relative_frame_offset = Args::StringToSInt32 (option_arg, INT32_MIN, 0, &success);
123864174e1SGreg Clayton                 if (!success)
124864174e1SGreg Clayton                     error.SetErrorStringWithFormat ("invalid frame offset argument '%s'.\n", option_arg);
125864174e1SGreg Clayton                 break;
126864174e1SGreg Clayton 
127864174e1SGreg Clayton             default:
1282c88643aSBenjamin Kramer                 error.SetErrorStringWithFormat ("Invalid short option character '%c'.\n", short_option);
129864174e1SGreg Clayton                 break;
130864174e1SGreg Clayton             }
131864174e1SGreg Clayton 
132864174e1SGreg Clayton             return error;
133864174e1SGreg Clayton         }
134864174e1SGreg Clayton 
135864174e1SGreg Clayton         void
136f6b8b581SGreg Clayton         OptionParsingStarting ()
137864174e1SGreg Clayton         {
138864174e1SGreg Clayton             relative_frame_offset = INT32_MIN;
139864174e1SGreg Clayton         }
140864174e1SGreg Clayton 
141e0d378b3SGreg Clayton         const 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 
149e0d378b3SGreg Clayton         static 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,
158eb0103f2SGreg Clayton                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused),
159eb0103f2SGreg Clayton         m_options (interpreter)
16030fdc8d8SChris Lattner     {
161405fe67fSCaroline Tice         CommandArgumentEntry arg;
162405fe67fSCaroline Tice         CommandArgumentData index_arg;
163405fe67fSCaroline Tice 
164405fe67fSCaroline Tice         // Define the first (and only) variant of this arg.
165405fe67fSCaroline Tice         index_arg.arg_type = eArgTypeFrameIndex;
166864174e1SGreg Clayton         index_arg.arg_repetition = eArgRepeatOptional;
167405fe67fSCaroline Tice 
168405fe67fSCaroline Tice         // There is only one variant this argument could be; put it into the argument entry.
169405fe67fSCaroline Tice         arg.push_back (index_arg);
170405fe67fSCaroline Tice 
171405fe67fSCaroline Tice         // Push the data for the first argument into the m_arguments vector.
172405fe67fSCaroline Tice         m_arguments.push_back (arg);
17330fdc8d8SChris Lattner     }
17430fdc8d8SChris Lattner 
17530fdc8d8SChris Lattner     ~CommandObjectFrameSelect ()
17630fdc8d8SChris Lattner     {
17730fdc8d8SChris Lattner     }
17830fdc8d8SChris Lattner 
179864174e1SGreg Clayton     virtual
180864174e1SGreg Clayton     Options *
181864174e1SGreg Clayton     GetOptions ()
182864174e1SGreg Clayton     {
183864174e1SGreg Clayton         return &m_options;
184864174e1SGreg Clayton     }
185864174e1SGreg Clayton 
186864174e1SGreg Clayton 
18730fdc8d8SChris Lattner     bool
188a7015092SGreg Clayton     Execute (Args& command,
18930fdc8d8SChris Lattner              CommandReturnObject &result)
19030fdc8d8SChris Lattner     {
1918b82f087SGreg Clayton         ExecutionContext exe_ctx (m_interpreter.GetExecutionContext());
19230fdc8d8SChris Lattner         if (exe_ctx.thread)
19330fdc8d8SChris Lattner         {
194864174e1SGreg Clayton             const uint32_t num_frames = exe_ctx.thread->GetStackFrameCount();
195864174e1SGreg Clayton             uint32_t frame_idx = UINT32_MAX;
196864174e1SGreg Clayton             if (m_options.relative_frame_offset != INT32_MIN)
197864174e1SGreg Clayton             {
198864174e1SGreg Clayton                 // The one and only argument is a signed relative frame index
199864174e1SGreg Clayton                 frame_idx = exe_ctx.thread->GetSelectedFrameIndex ();
200864174e1SGreg Clayton                 if (frame_idx == UINT32_MAX)
201864174e1SGreg Clayton                     frame_idx = 0;
202864174e1SGreg Clayton 
203864174e1SGreg Clayton                 if (m_options.relative_frame_offset < 0)
204864174e1SGreg Clayton                 {
205864174e1SGreg Clayton                     if (frame_idx >= -m_options.relative_frame_offset)
206864174e1SGreg Clayton                         frame_idx += m_options.relative_frame_offset;
207864174e1SGreg Clayton                     else
208*213b4546SJim Ingham                     {
209*213b4546SJim Ingham                         if (frame_idx == 0)
210*213b4546SJim Ingham                         {
211*213b4546SJim Ingham                             //If you are already at the bottom of the stack, then just warn and don't reset the frame.
212*213b4546SJim Ingham                             result.AppendError("Already at the bottom of the stack");
213*213b4546SJim Ingham                             result.SetStatus(eReturnStatusFailed);
214*213b4546SJim Ingham                             return false;
215*213b4546SJim Ingham                         }
216*213b4546SJim Ingham                         else
217864174e1SGreg Clayton                             frame_idx = 0;
218864174e1SGreg Clayton                     }
219*213b4546SJim Ingham                 }
220864174e1SGreg Clayton                 else if (m_options.relative_frame_offset > 0)
221864174e1SGreg Clayton                 {
222864174e1SGreg Clayton                     if (num_frames - frame_idx > m_options.relative_frame_offset)
223864174e1SGreg Clayton                         frame_idx += m_options.relative_frame_offset;
224864174e1SGreg Clayton                     else
225*213b4546SJim Ingham                     {
226*213b4546SJim Ingham                         if (frame_idx == num_frames - 1)
227*213b4546SJim Ingham                         {
228*213b4546SJim Ingham                             //If we are already at the top of the stack, just warn and don't reset the frame.
229*213b4546SJim Ingham                             result.AppendError("Already at the top of the stack");
230*213b4546SJim Ingham                             result.SetStatus(eReturnStatusFailed);
231*213b4546SJim Ingham                             return false;
232*213b4546SJim Ingham                         }
233*213b4546SJim Ingham                         else
234864174e1SGreg Clayton                             frame_idx = num_frames - 1;
235864174e1SGreg Clayton                     }
236864174e1SGreg Clayton                 }
237*213b4546SJim Ingham             }
238864174e1SGreg Clayton             else
239864174e1SGreg Clayton             {
24030fdc8d8SChris Lattner                 if (command.GetArgumentCount() == 1)
24130fdc8d8SChris Lattner                 {
24230fdc8d8SChris Lattner                     const char *frame_idx_cstr = command.GetArgumentAtIndex(0);
243864174e1SGreg Clayton                     frame_idx = Args::StringToUInt32 (frame_idx_cstr, UINT32_MAX, 0);
244864174e1SGreg Clayton                 }
245864174e1SGreg Clayton                 else
246864174e1SGreg Clayton                 {
247864174e1SGreg Clayton                     result.AppendError ("invalid arguments.\n");
248eb0103f2SGreg Clayton                     m_options.GenerateOptionUsage (result.GetErrorStream(), this);
249864174e1SGreg Clayton                 }
250864174e1SGreg Clayton             }
25130fdc8d8SChris Lattner 
25230fdc8d8SChris Lattner             if (frame_idx < num_frames)
25330fdc8d8SChris Lattner             {
2542976d00aSJim Ingham                 exe_ctx.thread->SetSelectedFrameByIndex (frame_idx);
2552976d00aSJim Ingham                 exe_ctx.frame = exe_ctx.thread->GetSelectedFrame ().get();
25630fdc8d8SChris Lattner 
25730fdc8d8SChris Lattner                 if (exe_ctx.frame)
25830fdc8d8SChris Lattner                 {
259e40e4218SJim Ingham                     bool already_shown = false;
260e40e4218SJim Ingham                     SymbolContext frame_sc(exe_ctx.frame->GetSymbolContext(eSymbolContextLineEntry));
261daccaa9eSCaroline Tice                     if (m_interpreter.GetDebugger().GetUseExternalEditor() && frame_sc.line_entry.file && frame_sc.line_entry.line != 0)
262e40e4218SJim Ingham                     {
263e40e4218SJim Ingham                         already_shown = Host::OpenFileInExternalEditor (frame_sc.line_entry.file, frame_sc.line_entry.line);
264e40e4218SJim Ingham                     }
265e40e4218SJim Ingham 
2667260f620SGreg Clayton                     bool show_frame_info = true;
2677260f620SGreg Clayton                     bool show_source = !already_shown;
2687260f620SGreg Clayton                     uint32_t source_lines_before = 3;
2697260f620SGreg Clayton                     uint32_t source_lines_after = 3;
2707260f620SGreg Clayton                     if (exe_ctx.frame->GetStatus(result.GetOutputStream(),
2717260f620SGreg Clayton                                                  show_frame_info,
2727260f620SGreg Clayton                                                  show_source,
2737260f620SGreg Clayton                                                  source_lines_before,
2747260f620SGreg Clayton                                                  source_lines_after))
27530fdc8d8SChris Lattner                     {
27630fdc8d8SChris Lattner                         result.SetStatus (eReturnStatusSuccessFinishResult);
27730fdc8d8SChris Lattner                         return result.Succeeded();
27830fdc8d8SChris Lattner                     }
27930fdc8d8SChris Lattner                 }
28030fdc8d8SChris Lattner             }
28130fdc8d8SChris Lattner             result.AppendErrorWithFormat ("Frame index (%u) out of range.\n", frame_idx);
28230fdc8d8SChris Lattner         }
28330fdc8d8SChris Lattner         else
28430fdc8d8SChris Lattner         {
28530fdc8d8SChris Lattner             result.AppendError ("no current thread");
28630fdc8d8SChris Lattner         }
28730fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
28830fdc8d8SChris Lattner         return false;
28930fdc8d8SChris Lattner     }
290864174e1SGreg Clayton protected:
291864174e1SGreg Clayton 
292864174e1SGreg Clayton     CommandOptions m_options;
293864174e1SGreg Clayton };
294864174e1SGreg Clayton 
295e0d378b3SGreg Clayton OptionDefinition
296864174e1SGreg Clayton CommandObjectFrameSelect::CommandOptions::g_option_table[] =
297864174e1SGreg Clayton {
298864174e1SGreg Clayton { LLDB_OPT_SET_1, false, "relative", 'r', required_argument, NULL, 0, eArgTypeOffset, "A relative frame index offset from the current frame index."},
299864174e1SGreg Clayton { 0, false, NULL, 0, 0, NULL, NULL, eArgTypeNone, NULL }
30030fdc8d8SChris Lattner };
30130fdc8d8SChris Lattner 
3026d56d2ceSJim Ingham #pragma mark CommandObjectFrameVariable
3036d56d2ceSJim Ingham //----------------------------------------------------------------------
3046d56d2ceSJim Ingham // List images with associated information
3056d56d2ceSJim Ingham //----------------------------------------------------------------------
3066d56d2ceSJim Ingham class CommandObjectFrameVariable : public CommandObject
3076d56d2ceSJim Ingham {
3086d56d2ceSJim Ingham public:
3096d56d2ceSJim Ingham 
3102837b766SJim Ingham     class OptionGroupFrameVariable : public OptionGroup
3116d56d2ceSJim Ingham     {
3126d56d2ceSJim Ingham     public:
3136d56d2ceSJim Ingham 
3142837b766SJim Ingham         OptionGroupFrameVariable ()
3156d56d2ceSJim Ingham         {
3166d56d2ceSJim Ingham         }
3176d56d2ceSJim Ingham 
3186d56d2ceSJim Ingham         virtual
3192837b766SJim Ingham         ~OptionGroupFrameVariable ()
3206d56d2ceSJim Ingham         {
3216d56d2ceSJim Ingham         }
3226d56d2ceSJim Ingham 
3232837b766SJim Ingham         virtual uint32_t
3242837b766SJim Ingham         GetNumDefinitions ();
3252837b766SJim Ingham 
3262837b766SJim Ingham         virtual const OptionDefinition*
3272837b766SJim Ingham         GetDefinitions ()
3282837b766SJim Ingham         {
3292837b766SJim Ingham             return g_option_table;
3302837b766SJim Ingham         }
3312837b766SJim Ingham 
3326d56d2ceSJim Ingham         virtual Error
3332837b766SJim Ingham         SetOptionValue (CommandInterpreter &interpreter,
3342837b766SJim Ingham                         uint32_t option_idx,
3352837b766SJim Ingham                         const char *option_arg)
3366d56d2ceSJim Ingham         {
3376d56d2ceSJim Ingham             Error error;
3382837b766SJim Ingham             char short_option = (char) g_option_table[option_idx].short_option;
3396d56d2ceSJim Ingham             switch (short_option)
3406d56d2ceSJim Ingham             {
3416d56d2ceSJim Ingham             case 'r':   use_regex    = true;  break;
3426d56d2ceSJim Ingham             case 'a':   show_args    = false; break;
3436d56d2ceSJim Ingham             case 'l':   show_locals  = false; break;
344a134cc1bSGreg Clayton             case 'g':   show_globals = true;  break;
345a134cc1bSGreg Clayton             case 'c':   show_decl    = true;  break;
34668ebae61SGreg Clayton             case 'f':   error = Args::StringToFormat(option_arg, format, NULL); break;
3476d56d2ceSJim Ingham             case 'G':
348a134cc1bSGreg Clayton                 globals.push_back(ConstString (option_arg));
3496d56d2ceSJim Ingham                 break;
3506d56d2ceSJim Ingham             case 's':
3516d56d2ceSJim Ingham                 show_scope = true;
3526d56d2ceSJim Ingham                 break;
3536d56d2ceSJim Ingham 
3546d56d2ceSJim Ingham             default:
3556d56d2ceSJim Ingham                 error.SetErrorStringWithFormat("Invalid short option character '%c'.\n", short_option);
3566d56d2ceSJim Ingham                 break;
3576d56d2ceSJim Ingham             }
3586d56d2ceSJim Ingham 
3596d56d2ceSJim Ingham             return error;
3606d56d2ceSJim Ingham         }
3616d56d2ceSJim Ingham 
3622837b766SJim Ingham         virtual void
3632837b766SJim Ingham         OptionParsingStarting (CommandInterpreter &interpreter)
3646d56d2ceSJim Ingham         {
3656d56d2ceSJim Ingham             show_args     = true;
366a134cc1bSGreg Clayton             show_decl     = false;
367ded470d3SGreg Clayton             format        = eFormatDefault;
3682837b766SJim Ingham             show_globals  = false;
3692837b766SJim Ingham             show_locals   = true;
3702837b766SJim Ingham             use_regex     = false;
3712837b766SJim Ingham             show_scope    = false;
3726d56d2ceSJim Ingham             globals.clear();
3736d56d2ceSJim Ingham         }
3746d56d2ceSJim Ingham 
3756d56d2ceSJim Ingham         // Options table: Required for subclasses of Options.
3766d56d2ceSJim Ingham 
377e0d378b3SGreg Clayton         static OptionDefinition g_option_table[];
3782837b766SJim Ingham 
3792837b766SJim Ingham         bool use_regex:1,
380a134cc1bSGreg Clayton              show_args:1,
381a134cc1bSGreg Clayton              show_locals:1,
382a134cc1bSGreg Clayton              show_globals:1,
383a134cc1bSGreg Clayton              show_scope:1,
3842837b766SJim Ingham              show_decl:1;
385ded470d3SGreg Clayton         lldb::Format format; // The format to use when dumping variables or children of variables
3866d56d2ceSJim Ingham         std::vector<ConstString> globals;
3876d56d2ceSJim Ingham         // Instance variables to hold the values for command options.
3886d56d2ceSJim Ingham     };
3896d56d2ceSJim Ingham 
390a7015092SGreg Clayton     CommandObjectFrameVariable (CommandInterpreter &interpreter) :
391a7015092SGreg Clayton         CommandObject (interpreter,
3926d56d2ceSJim Ingham                        "frame variable",
393ed8a705cSGreg Clayton                        "Show frame variables. All argument and local variables "
394ed8a705cSGreg Clayton                        "that are in scope will be shown when no arguments are given. "
395ed8a705cSGreg Clayton                        "If any arguments are specified, they can be names of "
396ed8a705cSGreg Clayton                        "argument, local, file static and file global variables. "
397ed8a705cSGreg Clayton                        "Children of aggregate variables can be specified such as "
398ed8a705cSGreg Clayton                        "'var->child.x'.",
399ff471a94SJim Ingham                        NULL,
400eb0103f2SGreg Clayton                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused),
4012837b766SJim Ingham         m_option_group (interpreter),
402715c2365SGreg Clayton         m_option_variable(true), // Include the frame specific options by passing "true"
4032837b766SJim Ingham         m_varobj_options()
4046d56d2ceSJim Ingham     {
405405fe67fSCaroline Tice         CommandArgumentEntry arg;
406405fe67fSCaroline Tice         CommandArgumentData var_name_arg;
407405fe67fSCaroline Tice 
408405fe67fSCaroline Tice         // Define the first (and only) variant of this arg.
409405fe67fSCaroline Tice         var_name_arg.arg_type = eArgTypeVarName;
410405fe67fSCaroline Tice         var_name_arg.arg_repetition = eArgRepeatStar;
411405fe67fSCaroline Tice 
412405fe67fSCaroline Tice         // There is only one variant this argument could be; put it into the argument entry.
413405fe67fSCaroline Tice         arg.push_back (var_name_arg);
414405fe67fSCaroline Tice 
415405fe67fSCaroline Tice         // Push the data for the first argument into the m_arguments vector.
416405fe67fSCaroline Tice         m_arguments.push_back (arg);
4172837b766SJim Ingham 
418715c2365SGreg Clayton         m_option_group.Append (&m_option_variable, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1);
4192837b766SJim Ingham         m_option_group.Append (&m_varobj_options, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1);
4202837b766SJim Ingham         m_option_group.Finalize();
4216d56d2ceSJim Ingham     }
4226d56d2ceSJim Ingham 
4236d56d2ceSJim Ingham     virtual
4246d56d2ceSJim Ingham     ~CommandObjectFrameVariable ()
4256d56d2ceSJim Ingham     {
4266d56d2ceSJim Ingham     }
4276d56d2ceSJim Ingham 
4286d56d2ceSJim Ingham     virtual
4296d56d2ceSJim Ingham     Options *
4306d56d2ceSJim Ingham     GetOptions ()
4316d56d2ceSJim Ingham     {
4322837b766SJim Ingham         return &m_option_group;
4336d56d2ceSJim Ingham     }
4346d56d2ceSJim Ingham 
4356d56d2ceSJim Ingham 
4366d56d2ceSJim Ingham     virtual bool
4376d56d2ceSJim Ingham     Execute
4386d56d2ceSJim Ingham     (
4396d56d2ceSJim Ingham         Args& command,
4406d56d2ceSJim Ingham         CommandReturnObject &result
4416d56d2ceSJim Ingham     )
4426d56d2ceSJim Ingham     {
4438b82f087SGreg Clayton         ExecutionContext exe_ctx(m_interpreter.GetExecutionContext());
4446d56d2ceSJim Ingham         if (exe_ctx.frame == NULL)
4456d56d2ceSJim Ingham         {
446340b2baaSGreg Clayton             result.AppendError ("you must be stopped in a valid stack frame to view frame variables.");
4476d56d2ceSJim Ingham             result.SetStatus (eReturnStatusFailed);
4486d56d2ceSJim Ingham             return false;
4496d56d2ceSJim Ingham         }
4506d56d2ceSJim Ingham         else
4516d56d2ceSJim Ingham         {
452a134cc1bSGreg Clayton             Stream &s = result.GetOutputStream();
4536d56d2ceSJim Ingham 
454a134cc1bSGreg Clayton             bool get_file_globals = true;
455650543f9SJim Ingham 
456650543f9SJim Ingham             // Be careful about the stack frame, if any summary formatter runs code, it might clear the StackFrameList
457650543f9SJim Ingham             // for the thread.  So hold onto a shared pointer to the frame so it stays alive.
458650543f9SJim Ingham 
459650543f9SJim Ingham             StackFrameSP frame_sp = exe_ctx.frame->GetSP();
460650543f9SJim Ingham 
461650543f9SJim Ingham             VariableList *variable_list = frame_sp->GetVariableList (get_file_globals);
462a134cc1bSGreg Clayton 
4636d56d2ceSJim Ingham             VariableSP var_sp;
4646d56d2ceSJim Ingham             ValueObjectSP valobj_sp;
46578a685aaSJim Ingham 
4666d56d2ceSJim Ingham             const char *name_cstr = NULL;
4676d56d2ceSJim Ingham             size_t idx;
4686d56d2ceSJim Ingham 
469f9fa6ee5SEnrico Granata             SummaryFormatSP summary_format_sp;
470f9fa6ee5SEnrico Granata             if (!m_option_variable.summary.empty())
471c482a192SEnrico Granata                 DataVisualization::NamedSummaryFormats::Get(ConstString(m_option_variable.summary.c_str()), summary_format_sp);
472f9fa6ee5SEnrico Granata 
473379447a7SEnrico Granata             ValueObject::DumpValueObjectOptions options;
474379447a7SEnrico Granata 
475379447a7SEnrico Granata             options.SetPointerDepth(m_varobj_options.ptr_depth)
476379447a7SEnrico Granata                    .SetMaximumDepth(m_varobj_options.max_depth)
477379447a7SEnrico Granata                    .SetShowTypes(m_varobj_options.show_types)
478379447a7SEnrico Granata                    .SetShowLocation(m_varobj_options.show_location)
479379447a7SEnrico Granata                    .SetUseObjectiveC(m_varobj_options.use_objc)
480379447a7SEnrico Granata                    .SetUseDynamicType(m_varobj_options.use_dynamic)
481379447a7SEnrico Granata                    .SetUseSyntheticValue((lldb::SyntheticValueType)m_varobj_options.use_synth)
482379447a7SEnrico Granata                    .SetFlatOutput(m_varobj_options.flat_output)
483379447a7SEnrico Granata                    .SetOmitSummaryDepth(m_varobj_options.no_summary_depth)
484379447a7SEnrico Granata                    .SetIgnoreCap(m_varobj_options.ignore_cap);
485379447a7SEnrico Granata 
486379447a7SEnrico Granata             if (m_varobj_options.be_raw)
487379447a7SEnrico Granata                 options.SetRawDisplay(true);
488379447a7SEnrico Granata 
489715c2365SGreg Clayton             if (variable_list)
4909df87c17SGreg Clayton             {
4919df87c17SGreg Clayton                 if (command.GetArgumentCount() > 0)
4926d56d2ceSJim Ingham                 {
49346747022SGreg Clayton                     VariableList regex_var_list;
49446747022SGreg Clayton 
4956d56d2ceSJim Ingham                     // If we have any args to the variable command, we will make
4966d56d2ceSJim Ingham                     // variable objects from them...
4976d56d2ceSJim Ingham                     for (idx = 0; (name_cstr = command.GetArgumentAtIndex(idx)) != NULL; ++idx)
4986d56d2ceSJim Ingham                     {
499715c2365SGreg Clayton                         if (m_option_variable.use_regex)
50046747022SGreg Clayton                         {
50146747022SGreg Clayton                             const uint32_t regex_start_index = regex_var_list.GetSize();
50246747022SGreg Clayton                             RegularExpression regex (name_cstr);
50346747022SGreg Clayton                             if (regex.Compile(name_cstr))
50446747022SGreg Clayton                             {
50546747022SGreg Clayton                                 size_t num_matches = 0;
50678a685aaSJim Ingham                                 const size_t num_new_regex_vars = variable_list->AppendVariablesIfUnique(regex,
50778a685aaSJim Ingham                                                                                                          regex_var_list,
50878a685aaSJim Ingham                                                                                                          num_matches);
50946747022SGreg Clayton                                 if (num_new_regex_vars > 0)
51046747022SGreg Clayton                                 {
51146747022SGreg Clayton                                     for (uint32_t regex_idx = regex_start_index, end_index = regex_var_list.GetSize();
51246747022SGreg Clayton                                          regex_idx < end_index;
51346747022SGreg Clayton                                          ++regex_idx)
51446747022SGreg Clayton                                     {
51546747022SGreg Clayton                                         var_sp = regex_var_list.GetVariableAtIndex (regex_idx);
51646747022SGreg Clayton                                         if (var_sp)
51746747022SGreg Clayton                                         {
518650543f9SJim Ingham                                             valobj_sp = frame_sp->GetValueObjectForFrameVariable (var_sp, m_varobj_options.use_dynamic);
51946747022SGreg Clayton                                             if (valobj_sp)
52046747022SGreg Clayton                                             {
521715c2365SGreg Clayton                                                 if (m_option_variable.format != eFormatDefault)
522715c2365SGreg Clayton                                                     valobj_sp->SetFormat (m_option_variable.format);
523ded470d3SGreg Clayton 
524715c2365SGreg Clayton                                                 if (m_option_variable.show_decl && var_sp->GetDeclaration ().GetFile())
52546747022SGreg Clayton                                                 {
52645ba8543SGreg Clayton                                                     bool show_fullpaths = false;
52745ba8543SGreg Clayton                                                     bool show_module = true;
52845ba8543SGreg Clayton                                                     if (var_sp->DumpDeclaration(&s, show_fullpaths, show_module))
52946747022SGreg Clayton                                                         s.PutCString (": ");
53046747022SGreg Clayton                                                 }
531f9fa6ee5SEnrico Granata                                                 if (summary_format_sp)
532f9fa6ee5SEnrico Granata                                                     valobj_sp->SetCustomSummaryFormat(summary_format_sp);
53346747022SGreg Clayton                                                 ValueObject::DumpValueObject (result.GetOutputStream(),
53446747022SGreg Clayton                                                                               valobj_sp.get(),
535379447a7SEnrico Granata                                                                               options);
53646747022SGreg Clayton                                             }
53746747022SGreg Clayton                                         }
53846747022SGreg Clayton                                     }
53946747022SGreg Clayton                                 }
54046747022SGreg Clayton                                 else if (num_matches == 0)
54146747022SGreg Clayton                                 {
54246747022SGreg Clayton                                     result.GetErrorStream().Printf ("error: no variables matched the regular expression '%s'.\n", name_cstr);
54346747022SGreg Clayton                                 }
54446747022SGreg Clayton                             }
54546747022SGreg Clayton                             else
54646747022SGreg Clayton                             {
54746747022SGreg Clayton                                 char regex_error[1024];
54846747022SGreg Clayton                                 if (regex.GetErrorAsCString(regex_error, sizeof(regex_error)))
54946747022SGreg Clayton                                     result.GetErrorStream().Printf ("error: %s\n", regex_error);
55046747022SGreg Clayton                                 else
55146747022SGreg Clayton                                     result.GetErrorStream().Printf ("error: unkown regex error when compiling '%s'\n", name_cstr);
55246747022SGreg Clayton                             }
55346747022SGreg Clayton                         }
55446747022SGreg Clayton                         else
55546747022SGreg Clayton                         {
55654979cddSGreg Clayton                             Error error;
55778a685aaSJim Ingham                             uint32_t expr_path_options = StackFrame::eExpressionPathOptionCheckPtrVsMember;
5582837b766SJim Ingham                             lldb::VariableSP var_sp;
559650543f9SJim Ingham                             valobj_sp = frame_sp->GetValueForVariableExpressionPath (name_cstr,
5602837b766SJim Ingham                                                                                           m_varobj_options.use_dynamic,
5612837b766SJim Ingham                                                                                           expr_path_options,
5622837b766SJim Ingham                                                                                           var_sp,
5632837b766SJim Ingham                                                                                           error);
5646d56d2ceSJim Ingham                             if (valobj_sp)
5656d56d2ceSJim Ingham                             {
566715c2365SGreg Clayton                                 if (m_option_variable.format != eFormatDefault)
567715c2365SGreg Clayton                                     valobj_sp->SetFormat (m_option_variable.format);
568715c2365SGreg Clayton                                 if (m_option_variable.show_decl && var_sp && var_sp->GetDeclaration ().GetFile())
569a134cc1bSGreg Clayton                                 {
570a134cc1bSGreg Clayton                                     var_sp->GetDeclaration ().DumpStopContext (&s, false);
571a134cc1bSGreg Clayton                                     s.PutCString (": ");
572a134cc1bSGreg Clayton                                 }
573f9fa6ee5SEnrico Granata                                 if (summary_format_sp)
574f9fa6ee5SEnrico Granata                                     valobj_sp->SetCustomSummaryFormat(summary_format_sp);
5751d3afba3SGreg Clayton                                 ValueObject::DumpValueObject (result.GetOutputStream(),
576a134cc1bSGreg Clayton                                                               valobj_sp.get(),
57783c5cd9dSGreg Clayton                                                               valobj_sp->GetParent() ? name_cstr : NULL,
578379447a7SEnrico Granata                                                               options);
5796d56d2ceSJim Ingham                             }
5806d56d2ceSJim Ingham                             else
5816d56d2ceSJim Ingham                             {
58254979cddSGreg Clayton                                 const char *error_cstr = error.AsCString(NULL);
58354979cddSGreg Clayton                                 if (error_cstr)
58454979cddSGreg Clayton                                     result.GetErrorStream().Printf("error: %s\n", error_cstr);
58554979cddSGreg Clayton                                 else
58654979cddSGreg Clayton                                     result.GetErrorStream().Printf ("error: unable to find any variable expression path that matches '%s'\n", name_cstr);
5876d56d2ceSJim Ingham                             }
5886d56d2ceSJim Ingham                         }
5896d56d2ceSJim Ingham                     }
59046747022SGreg Clayton                 }
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                         {
5991a65ae11SGreg Clayton                             var_sp = variable_list->GetVariableAtIndex(i);
60078a685aaSJim Ingham 
6016d56d2ceSJim Ingham                             bool dump_variable = true;
6026d56d2ceSJim Ingham 
603a134cc1bSGreg Clayton                             switch (var_sp->GetScope())
6046d56d2ceSJim Ingham                             {
6056d56d2ceSJim Ingham                             case eValueTypeVariableGlobal:
606715c2365SGreg Clayton                                 dump_variable = m_option_variable.show_globals;
607715c2365SGreg Clayton                                 if (dump_variable && m_option_variable.show_scope)
608a134cc1bSGreg Clayton                                     s.PutCString("GLOBAL: ");
6096d56d2ceSJim Ingham                                 break;
6106d56d2ceSJim Ingham 
6116d56d2ceSJim Ingham                             case eValueTypeVariableStatic:
612715c2365SGreg Clayton                                 dump_variable = m_option_variable.show_globals;
613715c2365SGreg Clayton                                 if (dump_variable && m_option_variable.show_scope)
614a134cc1bSGreg Clayton                                     s.PutCString("STATIC: ");
6156d56d2ceSJim Ingham                                 break;
6166d56d2ceSJim Ingham 
6176d56d2ceSJim Ingham                             case eValueTypeVariableArgument:
618715c2365SGreg Clayton                                 dump_variable = m_option_variable.show_args;
619715c2365SGreg Clayton                                 if (dump_variable && m_option_variable.show_scope)
620a134cc1bSGreg Clayton                                     s.PutCString("   ARG: ");
6216d56d2ceSJim Ingham                                 break;
6226d56d2ceSJim Ingham 
6236d56d2ceSJim Ingham                             case eValueTypeVariableLocal:
624715c2365SGreg Clayton                                 dump_variable = m_option_variable.show_locals;
625715c2365SGreg Clayton                                 if (dump_variable && m_option_variable.show_scope)
626a134cc1bSGreg Clayton                                     s.PutCString(" LOCAL: ");
6276d56d2ceSJim Ingham                                 break;
6286d56d2ceSJim Ingham 
6296d56d2ceSJim Ingham                             default:
6306d56d2ceSJim Ingham                                 break;
6316d56d2ceSJim Ingham                             }
6326d56d2ceSJim Ingham 
6336d56d2ceSJim Ingham                             if (dump_variable)
634a134cc1bSGreg Clayton                             {
635a134cc1bSGreg Clayton 
636a134cc1bSGreg Clayton                                 // Use the variable object code to make sure we are
637a134cc1bSGreg Clayton                                 // using the same APIs as the the public API will be
638a134cc1bSGreg Clayton                                 // using...
639650543f9SJim Ingham                                 valobj_sp = frame_sp->GetValueObjectForFrameVariable (var_sp,
6402837b766SJim Ingham                                                                                            m_varobj_options.use_dynamic);
641a134cc1bSGreg Clayton                                 if (valobj_sp)
642a134cc1bSGreg Clayton                                 {
643715c2365SGreg Clayton                                     if (m_option_variable.format != eFormatDefault)
644715c2365SGreg Clayton                                         valobj_sp->SetFormat (m_option_variable.format);
645ded470d3SGreg Clayton 
6466f00abd5SGreg Clayton                                     // When dumping all variables, don't print any variables
6476f00abd5SGreg Clayton                                     // that are not in scope to avoid extra unneeded output
6486035b67dSJim Ingham                                     if (valobj_sp->IsInScope ())
6496f00abd5SGreg Clayton                                     {
650715c2365SGreg Clayton                                         if (m_option_variable.show_decl && var_sp->GetDeclaration ().GetFile())
651a134cc1bSGreg Clayton                                         {
652a134cc1bSGreg Clayton                                             var_sp->GetDeclaration ().DumpStopContext (&s, false);
653a134cc1bSGreg Clayton                                             s.PutCString (": ");
654a134cc1bSGreg Clayton                                         }
655f9fa6ee5SEnrico Granata                                         if (summary_format_sp)
656f9fa6ee5SEnrico Granata                                             valobj_sp->SetCustomSummaryFormat(summary_format_sp);
6571d3afba3SGreg Clayton                                         ValueObject::DumpValueObject (result.GetOutputStream(),
658a134cc1bSGreg Clayton                                                                       valobj_sp.get(),
659a134cc1bSGreg Clayton                                                                       name_cstr,
660379447a7SEnrico Granata                                                                       options);
661a134cc1bSGreg Clayton                                     }
662a134cc1bSGreg Clayton                                 }
6636d56d2ceSJim Ingham                             }
6646d56d2ceSJim Ingham                         }
6656d56d2ceSJim Ingham                     }
6666f00abd5SGreg Clayton                 }
6676d56d2ceSJim Ingham                 result.SetStatus (eReturnStatusSuccessFinishResult);
6686d56d2ceSJim Ingham             }
6699df87c17SGreg Clayton         }
67061a80ba6SEnrico Granata 
67161a80ba6SEnrico Granata         if (m_interpreter.TruncationWarningNecessary())
67261a80ba6SEnrico Granata         {
67361a80ba6SEnrico Granata             result.GetOutputStream().Printf(m_interpreter.TruncationWarningText(),
67461a80ba6SEnrico Granata                                             m_cmd_name.c_str());
67561a80ba6SEnrico Granata             m_interpreter.TruncationWarningGiven();
67661a80ba6SEnrico Granata         }
67761a80ba6SEnrico Granata 
6786d56d2ceSJim Ingham         return result.Succeeded();
6796d56d2ceSJim Ingham     }
6806d56d2ceSJim Ingham protected:
6816d56d2ceSJim Ingham 
6822837b766SJim Ingham     OptionGroupOptions m_option_group;
683715c2365SGreg Clayton     OptionGroupVariable m_option_variable;
6842837b766SJim Ingham     OptionGroupValueObjectDisplay m_varobj_options;
6856d56d2ceSJim Ingham };
6866d56d2ceSJim Ingham 
6872837b766SJim Ingham 
68830fdc8d8SChris Lattner #pragma mark CommandObjectMultiwordFrame
68930fdc8d8SChris Lattner 
69030fdc8d8SChris Lattner //-------------------------------------------------------------------------
69130fdc8d8SChris Lattner // CommandObjectMultiwordFrame
69230fdc8d8SChris Lattner //-------------------------------------------------------------------------
69330fdc8d8SChris Lattner 
6946611103cSGreg Clayton CommandObjectMultiwordFrame::CommandObjectMultiwordFrame (CommandInterpreter &interpreter) :
695a7015092SGreg Clayton     CommandObjectMultiword (interpreter,
696a7015092SGreg Clayton                             "frame",
69730fdc8d8SChris Lattner                             "A set of commands for operating on the current thread's frames.",
69830fdc8d8SChris Lattner                             "frame <subcommand> [<subcommand-options>]")
69930fdc8d8SChris Lattner {
700a7015092SGreg Clayton     LoadSubCommand ("info",   CommandObjectSP (new CommandObjectFrameInfo (interpreter)));
701a7015092SGreg Clayton     LoadSubCommand ("select", CommandObjectSP (new CommandObjectFrameSelect (interpreter)));
702a7015092SGreg Clayton     LoadSubCommand ("variable", CommandObjectSP (new CommandObjectFrameVariable (interpreter)));
70330fdc8d8SChris Lattner }
70430fdc8d8SChris Lattner 
70530fdc8d8SChris Lattner CommandObjectMultiwordFrame::~CommandObjectMultiwordFrame ()
70630fdc8d8SChris Lattner {
70730fdc8d8SChris Lattner }
70830fdc8d8SChris Lattner 
709