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 // C Includes 1130fdc8d8SChris Lattner // C++ Includes 12de6bd243SJohnny Chen #include <string> 13c8ecc2a9SEugene Zelenko 1430fdc8d8SChris Lattner // Other libraries and framework includes 1530fdc8d8SChris Lattner // Project includes 16c8ecc2a9SEugene Zelenko #include "CommandObjectFrame.h" 1730fdc8d8SChris Lattner #include "lldb/Core/Debugger.h" 186d56d2ceSJim Ingham #include "lldb/Core/Module.h" 196d56d2ceSJim Ingham #include "lldb/Core/StreamFile.h" 206d56d2ceSJim Ingham #include "lldb/Core/Value.h" 216d56d2ceSJim Ingham #include "lldb/Core/ValueObject.h" 226d56d2ceSJim Ingham #include "lldb/Core/ValueObjectVariable.h" 235548cb50SEnrico Granata #include "lldb/DataFormatters/DataVisualization.h" 244d93b8cdSEnrico Granata #include "lldb/DataFormatters/ValueObjectPrinter.h" 257fb56d0aSGreg Clayton #include "lldb/Host/Host.h" 263eb2b44dSZachary Turner #include "lldb/Host/OptionParser.h" 2730fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 2830fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 291deb7962SGreg Clayton #include "lldb/Interpreter/OptionGroupFormat.h" 302837b766SJim Ingham #include "lldb/Interpreter/OptionGroupValueObjectDisplay.h" 31715c2365SGreg Clayton #include "lldb/Interpreter/OptionGroupVariable.h" 32b9c1b51eSKate Stone #include "lldb/Interpreter/Options.h" 336d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTContext.h" 34b9c1b51eSKate Stone #include "lldb/Symbol/CompilerType.h" 356754e04fSEnrico Granata #include "lldb/Symbol/Function.h" 366d56d2ceSJim Ingham #include "lldb/Symbol/ObjectFile.h" 376d56d2ceSJim Ingham #include "lldb/Symbol/SymbolContext.h" 386d56d2ceSJim Ingham #include "lldb/Symbol/Type.h" 396d56d2ceSJim Ingham #include "lldb/Symbol/Variable.h" 406d56d2ceSJim Ingham #include "lldb/Symbol/VariableList.h" 4130fdc8d8SChris Lattner #include "lldb/Target/Process.h" 42b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h" 434740a734SSean Callanan #include "lldb/Target/StopInfo.h" 446d56d2ceSJim Ingham #include "lldb/Target/Target.h" 45b9c1b51eSKate Stone #include "lldb/Target/Thread.h" 46145d95c9SPavel Labath #include "lldb/Utility/Args.h" 474740a734SSean Callanan #include "lldb/Utility/LLDBAssert.h" 48bf9a7730SZachary Turner #include "lldb/Utility/StreamString.h" 4938d0632eSPavel Labath #include "lldb/Utility/Timer.h" 5030fdc8d8SChris Lattner 5130fdc8d8SChris Lattner using namespace lldb; 5230fdc8d8SChris Lattner using namespace lldb_private; 5330fdc8d8SChris Lattner 544740a734SSean Callanan #pragma mark CommandObjectFrameDiagnose 554740a734SSean Callanan 564740a734SSean Callanan //------------------------------------------------------------------------- 574740a734SSean Callanan // CommandObjectFrameInfo 584740a734SSean Callanan //------------------------------------------------------------------------- 594740a734SSean Callanan 604740a734SSean Callanan //------------------------------------------------------------------------- 614740a734SSean Callanan // CommandObjectFrameDiagnose 624740a734SSean Callanan //------------------------------------------------------------------------- 634740a734SSean Callanan 648fe53c49STatyana Krasnukha static constexpr OptionDefinition g_frame_diag_options[] = { 651f0f5b5bSZachary Turner // clang-format off 668fe53c49STatyana Krasnukha { LLDB_OPT_SET_1, false, "register", 'r', OptionParser::eRequiredArgument, nullptr, {}, 0, eArgTypeRegisterName, "A register to diagnose." }, 678fe53c49STatyana Krasnukha { LLDB_OPT_SET_1, false, "address", 'a', OptionParser::eRequiredArgument, nullptr, {}, 0, eArgTypeAddress, "An address to diagnose." }, 688fe53c49STatyana Krasnukha { LLDB_OPT_SET_1, false, "offset", 'o', OptionParser::eRequiredArgument, nullptr, {}, 0, eArgTypeOffset, "An optional offset. Requires --register." } 691f0f5b5bSZachary Turner // clang-format on 701f0f5b5bSZachary Turner }; 711f0f5b5bSZachary Turner 72b9c1b51eSKate Stone class CommandObjectFrameDiagnose : public CommandObjectParsed { 734740a734SSean Callanan public: 74b9c1b51eSKate Stone class CommandOptions : public Options { 754740a734SSean Callanan public: 76b9c1b51eSKate Stone CommandOptions() : Options() { OptionParsingStarting(nullptr); } 774740a734SSean Callanan 784740a734SSean Callanan ~CommandOptions() override = default; 794740a734SSean Callanan 8097206d57SZachary Turner Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg, 81b9c1b51eSKate Stone ExecutionContext *execution_context) override { 8297206d57SZachary Turner Status error; 834740a734SSean Callanan const int short_option = m_getopt_table[option_idx].val; 84b9c1b51eSKate Stone switch (short_option) { 854740a734SSean Callanan case 'r': 864740a734SSean Callanan reg = ConstString(option_arg); 874740a734SSean Callanan break; 884740a734SSean Callanan 89b9c1b51eSKate Stone case 'a': { 90fe11483bSZachary Turner address.emplace(); 91fe11483bSZachary Turner if (option_arg.getAsInteger(0, *address)) { 924740a734SSean Callanan address.reset(); 93b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid address argument '%s'", 94fe11483bSZachary Turner option_arg.str().c_str()); 954740a734SSean Callanan } 96b9c1b51eSKate Stone } break; 974740a734SSean Callanan 98b9c1b51eSKate Stone case 'o': { 99fe11483bSZachary Turner offset.emplace(); 100fe11483bSZachary Turner if (option_arg.getAsInteger(0, *offset)) { 1014740a734SSean Callanan offset.reset(); 102b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid offset argument '%s'", 103fe11483bSZachary Turner option_arg.str().c_str()); 1044740a734SSean Callanan } 105b9c1b51eSKate Stone } break; 1064740a734SSean Callanan 1074740a734SSean Callanan default: 108b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid short option character '%c'", 109b9c1b51eSKate Stone short_option); 1104740a734SSean Callanan break; 1114740a734SSean Callanan } 1124740a734SSean Callanan 1134740a734SSean Callanan return error; 1144740a734SSean Callanan } 1154740a734SSean Callanan 116b9c1b51eSKate Stone void OptionParsingStarting(ExecutionContext *execution_context) override { 1174740a734SSean Callanan address.reset(); 1184740a734SSean Callanan reg.reset(); 1194740a734SSean Callanan offset.reset(); 1204740a734SSean Callanan } 1214740a734SSean Callanan 1221f0f5b5bSZachary Turner llvm::ArrayRef<OptionDefinition> GetDefinitions() override { 12370602439SZachary Turner return llvm::makeArrayRef(g_frame_diag_options); 1241f0f5b5bSZachary Turner } 1254740a734SSean Callanan 1264740a734SSean Callanan // Options. 1274740a734SSean Callanan llvm::Optional<lldb::addr_t> address; 1284740a734SSean Callanan llvm::Optional<ConstString> reg; 1294740a734SSean Callanan llvm::Optional<int64_t> offset; 1304740a734SSean Callanan }; 1314740a734SSean Callanan 1324740a734SSean Callanan CommandObjectFrameDiagnose(CommandInterpreter &interpreter) 1334740a734SSean Callanan : CommandObjectParsed(interpreter, "frame diagnose", 134b9c1b51eSKate Stone "Try to determine what path path the current stop " 135b9c1b51eSKate Stone "location used to get to a register or address", 136b9c1b51eSKate Stone nullptr, 137b9c1b51eSKate Stone eCommandRequiresThread | eCommandTryTargetAPILock | 138b9c1b51eSKate Stone eCommandProcessMustBeLaunched | 1394740a734SSean Callanan eCommandProcessMustBePaused), 140b9c1b51eSKate Stone m_options() { 1414740a734SSean Callanan CommandArgumentEntry arg; 1424740a734SSean Callanan CommandArgumentData index_arg; 1434740a734SSean Callanan 1444740a734SSean Callanan // Define the first (and only) variant of this arg. 1454740a734SSean Callanan index_arg.arg_type = eArgTypeFrameIndex; 1464740a734SSean Callanan index_arg.arg_repetition = eArgRepeatOptional; 1474740a734SSean Callanan 148b9c1b51eSKate Stone // There is only one variant this argument could be; put it into the 149b9c1b51eSKate Stone // argument entry. 1504740a734SSean Callanan arg.push_back(index_arg); 1514740a734SSean Callanan 1524740a734SSean Callanan // Push the data for the first argument into the m_arguments vector. 1534740a734SSean Callanan m_arguments.push_back(arg); 1544740a734SSean Callanan } 1554740a734SSean Callanan 1564740a734SSean Callanan ~CommandObjectFrameDiagnose() override = default; 1574740a734SSean Callanan 158b9c1b51eSKate Stone Options *GetOptions() override { return &m_options; } 1594740a734SSean Callanan 1604740a734SSean Callanan protected: 161b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 1624740a734SSean Callanan Thread *thread = m_exe_ctx.GetThreadPtr(); 1634740a734SSean Callanan StackFrameSP frame_sp = thread->GetSelectedFrame(); 1644740a734SSean Callanan 1654740a734SSean Callanan ValueObjectSP valobj_sp; 1664740a734SSean Callanan 167b9c1b51eSKate Stone if (m_options.address.hasValue()) { 168b9c1b51eSKate Stone if (m_options.reg.hasValue() || m_options.offset.hasValue()) { 169b9c1b51eSKate Stone result.AppendError( 170b9c1b51eSKate Stone "`frame diagnose --address` is incompatible with other arguments."); 1714740a734SSean Callanan result.SetStatus(eReturnStatusFailed); 1724740a734SSean Callanan return false; 1734740a734SSean Callanan } 1744740a734SSean Callanan valobj_sp = frame_sp->GuessValueForAddress(m_options.address.getValue()); 175b9c1b51eSKate Stone } else if (m_options.reg.hasValue()) { 176b9c1b51eSKate Stone valobj_sp = frame_sp->GuessValueForRegisterAndOffset( 177b9c1b51eSKate Stone m_options.reg.getValue(), m_options.offset.getValueOr(0)); 178b9c1b51eSKate Stone } else { 1794740a734SSean Callanan StopInfoSP stop_info_sp = thread->GetStopInfo(); 180b9c1b51eSKate Stone if (!stop_info_sp) { 1814740a734SSean Callanan result.AppendError("No arguments provided, and no stop info."); 1824740a734SSean Callanan result.SetStatus(eReturnStatusFailed); 1834740a734SSean Callanan return false; 1844740a734SSean Callanan } 1854740a734SSean Callanan 1864740a734SSean Callanan valobj_sp = StopInfo::GetCrashingDereference(stop_info_sp); 1874740a734SSean Callanan } 1884740a734SSean Callanan 189b9c1b51eSKate Stone if (!valobj_sp) { 1904740a734SSean Callanan result.AppendError("No diagnosis available."); 1914740a734SSean Callanan result.SetStatus(eReturnStatusFailed); 1924740a734SSean Callanan return false; 1934740a734SSean Callanan } 1944740a734SSean Callanan 1954740a734SSean Callanan 1963bc714b2SZachary Turner DumpValueObjectOptions::DeclPrintingHelper helper = [&valobj_sp]( 1973bc714b2SZachary Turner ConstString type, ConstString var, const DumpValueObjectOptions &opts, 1983bc714b2SZachary Turner Stream &stream) -> bool { 199b9c1b51eSKate Stone const ValueObject::GetExpressionPathFormat format = ValueObject:: 200b9c1b51eSKate Stone GetExpressionPathFormat::eGetExpressionPathFormatHonorPointers; 2015d1f711aSPavel Labath const bool qualify_cxx_base_classes = false; 2024740a734SSean Callanan valobj_sp->GetExpressionPath(stream, qualify_cxx_base_classes, format); 2034740a734SSean Callanan stream.PutCString(" ="); 2044740a734SSean Callanan return true; 2054740a734SSean Callanan }; 2064740a734SSean Callanan 2074740a734SSean Callanan DumpValueObjectOptions options; 2084740a734SSean Callanan options.SetDeclPrintingHelper(helper); 209b9c1b51eSKate Stone ValueObjectPrinter printer(valobj_sp.get(), &result.GetOutputStream(), 210b9c1b51eSKate Stone options); 2114740a734SSean Callanan printer.PrintValueObject(); 2124740a734SSean Callanan 2134740a734SSean Callanan return true; 2144740a734SSean Callanan } 2154740a734SSean Callanan 2164740a734SSean Callanan protected: 2174740a734SSean Callanan CommandOptions m_options; 2184740a734SSean Callanan }; 2194740a734SSean Callanan 22030fdc8d8SChris Lattner #pragma mark CommandObjectFrameInfo 22130fdc8d8SChris Lattner 22230fdc8d8SChris Lattner //------------------------------------------------------------------------- 22330fdc8d8SChris Lattner // CommandObjectFrameInfo 22430fdc8d8SChris Lattner //------------------------------------------------------------------------- 22530fdc8d8SChris Lattner 226b9c1b51eSKate Stone class CommandObjectFrameInfo : public CommandObjectParsed { 22730fdc8d8SChris Lattner public: 2287428a18cSKate Stone CommandObjectFrameInfo(CommandInterpreter &interpreter) 229b9c1b51eSKate Stone : CommandObjectParsed( 230b9c1b51eSKate Stone interpreter, "frame info", "List information about the current " 231b9c1b51eSKate Stone "stack frame in the current thread.", 232b9c1b51eSKate Stone "frame info", 233b9c1b51eSKate Stone eCommandRequiresFrame | eCommandTryTargetAPILock | 234b9c1b51eSKate Stone eCommandProcessMustBeLaunched | eCommandProcessMustBePaused) {} 23530fdc8d8SChris Lattner 236c8ecc2a9SEugene Zelenko ~CommandObjectFrameInfo() override = default; 23730fdc8d8SChris Lattner 2385a988416SJim Ingham protected: 239b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 240f9fc609fSGreg Clayton m_exe_ctx.GetFrameRef().DumpUsingSettingsFormat(&result.GetOutputStream()); 24130fdc8d8SChris Lattner result.SetStatus(eReturnStatusSuccessFinishResult); 24230fdc8d8SChris Lattner return result.Succeeded(); 24330fdc8d8SChris Lattner } 24430fdc8d8SChris Lattner }; 24530fdc8d8SChris Lattner 24630fdc8d8SChris Lattner #pragma mark CommandObjectFrameSelect 24730fdc8d8SChris Lattner 24830fdc8d8SChris Lattner //------------------------------------------------------------------------- 24930fdc8d8SChris Lattner // CommandObjectFrameSelect 25030fdc8d8SChris Lattner //------------------------------------------------------------------------- 25130fdc8d8SChris Lattner 2521f0f5b5bSZachary Turner static OptionDefinition g_frame_select_options[] = { 2531f0f5b5bSZachary Turner // clang-format off 2548fe53c49STatyana Krasnukha { LLDB_OPT_SET_1, false, "relative", 'r', OptionParser::eRequiredArgument, nullptr, {}, 0, eArgTypeOffset, "A relative frame index offset from the current frame index." }, 2551f0f5b5bSZachary Turner // clang-format on 2561f0f5b5bSZachary Turner }; 2571f0f5b5bSZachary Turner 258b9c1b51eSKate Stone class CommandObjectFrameSelect : public CommandObjectParsed { 25930fdc8d8SChris Lattner public: 260b9c1b51eSKate Stone class CommandOptions : public Options { 261864174e1SGreg Clayton public: 262b9c1b51eSKate Stone CommandOptions() : Options() { OptionParsingStarting(nullptr); } 263864174e1SGreg Clayton 264c8ecc2a9SEugene Zelenko ~CommandOptions() override = default; 265864174e1SGreg Clayton 26697206d57SZachary Turner Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg, 267b9c1b51eSKate Stone ExecutionContext *execution_context) override { 26897206d57SZachary Turner Status error; 2693bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 270b9c1b51eSKate Stone switch (short_option) { 271864174e1SGreg Clayton case 'r': 272fe11483bSZachary Turner if (option_arg.getAsInteger(0, relative_frame_offset)) { 273fe11483bSZachary Turner relative_frame_offset = INT32_MIN; 274b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid frame offset argument '%s'", 275fe11483bSZachary Turner option_arg.str().c_str()); 276fe11483bSZachary Turner } 277864174e1SGreg Clayton break; 278864174e1SGreg Clayton 279864174e1SGreg Clayton default: 280b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid short option character '%c'", 281b9c1b51eSKate Stone short_option); 282864174e1SGreg Clayton break; 283864174e1SGreg Clayton } 284864174e1SGreg Clayton 285864174e1SGreg Clayton return error; 286864174e1SGreg Clayton } 287864174e1SGreg Clayton 288b9c1b51eSKate Stone void OptionParsingStarting(ExecutionContext *execution_context) override { 289864174e1SGreg Clayton relative_frame_offset = INT32_MIN; 290864174e1SGreg Clayton } 291864174e1SGreg Clayton 2921f0f5b5bSZachary Turner llvm::ArrayRef<OptionDefinition> GetDefinitions() override { 29370602439SZachary Turner return llvm::makeArrayRef(g_frame_select_options); 2941f0f5b5bSZachary Turner } 295864174e1SGreg Clayton 296864174e1SGreg Clayton int32_t relative_frame_offset; 297864174e1SGreg Clayton }; 298864174e1SGreg Clayton 2997428a18cSKate Stone CommandObjectFrameSelect(CommandInterpreter &interpreter) 3007428a18cSKate Stone : CommandObjectParsed( 301b9c1b51eSKate Stone interpreter, "frame select", "Select the current stack frame by " 302b9c1b51eSKate Stone "index from within the current thread " 303b9c1b51eSKate Stone "(see 'thread backtrace'.)", 304b9c1b51eSKate Stone nullptr, 305b9c1b51eSKate Stone eCommandRequiresThread | eCommandTryTargetAPILock | 306b9c1b51eSKate Stone eCommandProcessMustBeLaunched | eCommandProcessMustBePaused), 307b9c1b51eSKate Stone m_options() { 308405fe67fSCaroline Tice CommandArgumentEntry arg; 309405fe67fSCaroline Tice CommandArgumentData index_arg; 310405fe67fSCaroline Tice 311405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 312405fe67fSCaroline Tice index_arg.arg_type = eArgTypeFrameIndex; 313864174e1SGreg Clayton index_arg.arg_repetition = eArgRepeatOptional; 314405fe67fSCaroline Tice 315b9c1b51eSKate Stone // There is only one variant this argument could be; put it into the 316b9c1b51eSKate Stone // argument entry. 317405fe67fSCaroline Tice arg.push_back(index_arg); 318405fe67fSCaroline Tice 319405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 320405fe67fSCaroline Tice m_arguments.push_back(arg); 32130fdc8d8SChris Lattner } 32230fdc8d8SChris Lattner 323c8ecc2a9SEugene Zelenko ~CommandObjectFrameSelect() override = default; 32430fdc8d8SChris Lattner 325b9c1b51eSKate Stone Options *GetOptions() override { return &m_options; } 326864174e1SGreg Clayton 3275a988416SJim Ingham protected: 328b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 329b9c1b51eSKate Stone // No need to check "thread" for validity as eCommandRequiresThread ensures 330b9c1b51eSKate Stone // it is valid 331f9fc609fSGreg Clayton Thread *thread = m_exe_ctx.GetThreadPtr(); 332f9fc609fSGreg Clayton 333864174e1SGreg Clayton uint32_t frame_idx = UINT32_MAX; 334b9c1b51eSKate Stone if (m_options.relative_frame_offset != INT32_MIN) { 335864174e1SGreg Clayton // The one and only argument is a signed relative frame index 336c14ee32dSGreg Clayton frame_idx = thread->GetSelectedFrameIndex(); 337864174e1SGreg Clayton if (frame_idx == UINT32_MAX) 338864174e1SGreg Clayton frame_idx = 0; 339864174e1SGreg Clayton 340b9c1b51eSKate Stone if (m_options.relative_frame_offset < 0) { 3413985c8c6SSaleem Abdulrasool if (static_cast<int32_t>(frame_idx) >= -m_options.relative_frame_offset) 342864174e1SGreg Clayton frame_idx += m_options.relative_frame_offset; 343b9c1b51eSKate Stone else { 344b9c1b51eSKate Stone if (frame_idx == 0) { 34505097246SAdrian Prantl // If you are already at the bottom of the stack, then just warn 34605097246SAdrian Prantl // and don't reset the frame. 3477428a18cSKate Stone result.AppendError("Already at the bottom of the stack."); 348213b4546SJim Ingham result.SetStatus(eReturnStatusFailed); 349213b4546SJim Ingham return false; 350b9c1b51eSKate Stone } else 351864174e1SGreg Clayton frame_idx = 0; 352864174e1SGreg Clayton } 353b9c1b51eSKate Stone } else if (m_options.relative_frame_offset > 0) { 354b9c1b51eSKate Stone // I don't want "up 20" where "20" takes you past the top of the stack 355b9c1b51eSKate Stone // to produce 356b9c1b51eSKate Stone // an error, but rather to just go to the top. So I have to count the 357b9c1b51eSKate Stone // stack here... 358b0c72a5fSJim Ingham const uint32_t num_frames = thread->GetStackFrameCount(); 359b9c1b51eSKate Stone if (static_cast<int32_t>(num_frames - frame_idx) > 360b9c1b51eSKate Stone m_options.relative_frame_offset) 361864174e1SGreg Clayton frame_idx += m_options.relative_frame_offset; 362b9c1b51eSKate Stone else { 363b9c1b51eSKate Stone if (frame_idx == num_frames - 1) { 364b9c1b51eSKate Stone // If we are already at the top of the stack, just warn and don't 365b9c1b51eSKate Stone // reset the frame. 3667428a18cSKate Stone result.AppendError("Already at the top of the stack."); 367213b4546SJim Ingham result.SetStatus(eReturnStatusFailed); 368213b4546SJim Ingham return false; 369b9c1b51eSKate Stone } else 370864174e1SGreg Clayton frame_idx = num_frames - 1; 371864174e1SGreg Clayton } 372864174e1SGreg Clayton } 373b9c1b51eSKate Stone } else { 374f965cc86SZachary Turner if (command.GetArgumentCount() > 1) { 375f965cc86SZachary Turner result.AppendErrorWithFormat( 376f965cc86SZachary Turner "too many arguments; expected frame-index, saw '%s'.\n", 377867e7d17SZachary Turner command[0].c_str()); 378f965cc86SZachary Turner m_options.GenerateOptionUsage( 379f965cc86SZachary Turner result.GetErrorStream(), this, 380f965cc86SZachary Turner GetCommandInterpreter().GetDebugger().GetTerminalWidth()); 381f965cc86SZachary Turner return false; 382f965cc86SZachary Turner } 383f965cc86SZachary Turner 384b9c1b51eSKate Stone if (command.GetArgumentCount() == 1) { 385f965cc86SZachary Turner if (command[0].ref.getAsInteger(0, frame_idx)) { 386b9c1b51eSKate Stone result.AppendErrorWithFormat("invalid frame index argument '%s'.", 387f965cc86SZachary Turner command[0].c_str()); 388afbb0af8SJim Ingham result.SetStatus(eReturnStatusFailed); 389afbb0af8SJim Ingham return false; 390afbb0af8SJim Ingham } 391b9c1b51eSKate Stone } else if (command.GetArgumentCount() == 0) { 39282d4a2b9SJason Molenda frame_idx = thread->GetSelectedFrameIndex(); 393b9c1b51eSKate Stone if (frame_idx == UINT32_MAX) { 39482d4a2b9SJason Molenda frame_idx = 0; 39582d4a2b9SJason Molenda } 396864174e1SGreg Clayton } 397864174e1SGreg Clayton } 39830fdc8d8SChris Lattner 399b9c1b51eSKate Stone bool success = thread->SetSelectedFrameByIndexNoisily( 400b9c1b51eSKate Stone frame_idx, result.GetOutputStream()); 401b9c1b51eSKate Stone if (success) { 402f9fc609fSGreg Clayton m_exe_ctx.SetFrameSP(thread->GetSelectedFrame()); 40330fdc8d8SChris Lattner result.SetStatus(eReturnStatusSuccessFinishResult); 404b9c1b51eSKate Stone } else { 405b9c1b51eSKate Stone result.AppendErrorWithFormat("Frame index (%u) out of range.\n", 406b9c1b51eSKate Stone frame_idx); 40730fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 40893208b86SJim Ingham } 40993208b86SJim Ingham 41093208b86SJim Ingham return result.Succeeded(); 41130fdc8d8SChris Lattner } 412864174e1SGreg Clayton 413c8ecc2a9SEugene Zelenko protected: 414864174e1SGreg Clayton CommandOptions m_options; 415864174e1SGreg Clayton }; 416864174e1SGreg Clayton 4176d56d2ceSJim Ingham #pragma mark CommandObjectFrameVariable 4186d56d2ceSJim Ingham //---------------------------------------------------------------------- 4196d56d2ceSJim Ingham // List images with associated information 4206d56d2ceSJim Ingham //---------------------------------------------------------------------- 421b9c1b51eSKate Stone class CommandObjectFrameVariable : public CommandObjectParsed { 4226d56d2ceSJim Ingham public: 4237428a18cSKate Stone CommandObjectFrameVariable(CommandInterpreter &interpreter) 4247428a18cSKate Stone : CommandObjectParsed( 425b9c1b51eSKate Stone interpreter, "frame variable", 426b9c1b51eSKate Stone "Show variables for the current stack frame. Defaults to all " 4277428a18cSKate Stone "arguments and local variables in scope. Names of argument, " 4287428a18cSKate Stone "local, file static and file global variables can be specified. " 429ed8a705cSGreg Clayton "Children of aggregate variables can be specified such as " 430*285ae0c0SJim Ingham "'var->child.x'. The -> and [] operators in 'frame variable' do " 431*285ae0c0SJim Ingham "not invoke operator overloads if they exist, but directly access " 432*285ae0c0SJim Ingham "the specified element. If you want to trigger operator overloads " 433*285ae0c0SJim Ingham "use the expression command to print the variable instead." 434*285ae0c0SJim Ingham "\nIt is worth noting that except for overloaded " 435*285ae0c0SJim Ingham "operators, when printing local variables 'expr local_var' and " 436*285ae0c0SJim Ingham "'frame var local_var' produce the same " 437*285ae0c0SJim Ingham "results. However, 'frame variable' is more efficient, since it " 438*285ae0c0SJim Ingham "uses debug information and memory reads directly, rather than " 439*285ae0c0SJim Ingham "parsing and evaluating an expression, which may even involve " 440*285ae0c0SJim Ingham "JITing and running code in the target program.", 441b9c1b51eSKate Stone nullptr, eCommandRequiresFrame | eCommandTryTargetAPILock | 442b9c1b51eSKate Stone eCommandProcessMustBeLaunched | 4437428a18cSKate Stone eCommandProcessMustBePaused | eCommandRequiresProcess), 444e1cfbc79STodd Fiala m_option_group(), 445b9c1b51eSKate Stone m_option_variable( 446b9c1b51eSKate Stone true), // Include the frame specific options by passing "true" 4471deb7962SGreg Clayton m_option_format(eFormatDefault), 448b9c1b51eSKate Stone m_varobj_options() { 449405fe67fSCaroline Tice CommandArgumentEntry arg; 450405fe67fSCaroline Tice CommandArgumentData var_name_arg; 451405fe67fSCaroline Tice 452405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 453405fe67fSCaroline Tice var_name_arg.arg_type = eArgTypeVarName; 454405fe67fSCaroline Tice var_name_arg.arg_repetition = eArgRepeatStar; 455405fe67fSCaroline Tice 456b9c1b51eSKate Stone // There is only one variant this argument could be; put it into the 457b9c1b51eSKate Stone // argument entry. 458405fe67fSCaroline Tice arg.push_back(var_name_arg); 459405fe67fSCaroline Tice 460405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 461405fe67fSCaroline Tice m_arguments.push_back(arg); 4622837b766SJim Ingham 463715c2365SGreg Clayton m_option_group.Append(&m_option_variable, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 464b9c1b51eSKate Stone m_option_group.Append(&m_option_format, 465b9c1b51eSKate Stone OptionGroupFormat::OPTION_GROUP_FORMAT | 466b9c1b51eSKate Stone OptionGroupFormat::OPTION_GROUP_GDB_FMT, 467b9c1b51eSKate Stone LLDB_OPT_SET_1); 4682837b766SJim Ingham m_option_group.Append(&m_varobj_options, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 4692837b766SJim Ingham m_option_group.Finalize(); 4706d56d2ceSJim Ingham } 4716d56d2ceSJim Ingham 472c8ecc2a9SEugene Zelenko ~CommandObjectFrameVariable() override = default; 4736d56d2ceSJim Ingham 474b9c1b51eSKate Stone Options *GetOptions() override { return &m_option_group; } 475f21feadcSGreg Clayton 4762443bbd4SRaphael Isemann int HandleArgumentCompletion( 4772443bbd4SRaphael Isemann CompletionRequest &request, 4782443bbd4SRaphael Isemann OptionElementVector &opt_element_vector) override { 479f21feadcSGreg Clayton // Arguments are the standard source file completer. 480b9c1b51eSKate Stone CommandCompletions::InvokeCommonCompletionCallbacks( 481b9c1b51eSKate Stone GetCommandInterpreter(), CommandCompletions::eVariablePathCompletion, 482a2e76c0bSRaphael Isemann request, nullptr); 4831a6d7ab5SRaphael Isemann return request.GetNumberOfMatches(); 484f21feadcSGreg Clayton } 4856d56d2ceSJim Ingham 4865a988416SJim Ingham protected: 48773418dfeSEnrico Granata llvm::StringRef GetScopeString(VariableSP var_sp) { 48873418dfeSEnrico Granata if (!var_sp) 48973418dfeSEnrico Granata return llvm::StringRef::withNullAsEmpty(nullptr); 49073418dfeSEnrico Granata 49173418dfeSEnrico Granata switch (var_sp->GetScope()) { 49273418dfeSEnrico Granata case eValueTypeVariableGlobal: 49373418dfeSEnrico Granata return "GLOBAL: "; 49473418dfeSEnrico Granata case eValueTypeVariableStatic: 49573418dfeSEnrico Granata return "STATIC: "; 49673418dfeSEnrico Granata case eValueTypeVariableArgument: 49773418dfeSEnrico Granata return "ARG: "; 49873418dfeSEnrico Granata case eValueTypeVariableLocal: 49973418dfeSEnrico Granata return "LOCAL: "; 50073418dfeSEnrico Granata case eValueTypeVariableThreadLocal: 50173418dfeSEnrico Granata return "THREAD: "; 50273418dfeSEnrico Granata default: 50373418dfeSEnrico Granata break; 50473418dfeSEnrico Granata } 50573418dfeSEnrico Granata 50673418dfeSEnrico Granata return llvm::StringRef::withNullAsEmpty(nullptr); 50773418dfeSEnrico Granata } 50873418dfeSEnrico Granata 509b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 51005097246SAdrian Prantl // No need to check "frame" for validity as eCommandRequiresFrame ensures 51105097246SAdrian Prantl // it is valid 512b57e4a1bSJason Molenda StackFrame *frame = m_exe_ctx.GetFramePtr(); 5131e49e5e7SJohnny Chen 514a134cc1bSGreg Clayton Stream &s = result.GetOutputStream(); 5156d56d2ceSJim Ingham 516b9c1b51eSKate Stone // Be careful about the stack frame, if any summary formatter runs code, it 51705097246SAdrian Prantl // might clear the StackFrameList for the thread. So hold onto a shared 51805097246SAdrian Prantl // pointer to the frame so it stays alive. 519650543f9SJim Ingham 520b9c1b51eSKate Stone VariableList *variable_list = 521b9c1b51eSKate Stone frame->GetVariableList(m_option_variable.show_globals); 522a134cc1bSGreg Clayton 5236d56d2ceSJim Ingham VariableSP var_sp; 5246d56d2ceSJim Ingham ValueObjectSP valobj_sp; 52578a685aaSJim Ingham 526061858ceSEnrico Granata TypeSummaryImplSP summary_format_sp; 52717b11749SEnrico Granata if (!m_option_variable.summary.IsCurrentValueEmpty()) 528b9c1b51eSKate Stone DataVisualization::NamedSummaryFormats::GetSummaryFormat( 529b9c1b51eSKate Stone ConstString(m_option_variable.summary.GetCurrentValue()), 530b9c1b51eSKate Stone summary_format_sp); 53117b11749SEnrico Granata else if (!m_option_variable.summary_string.IsCurrentValueEmpty()) 532b9c1b51eSKate Stone summary_format_sp.reset(new StringSummaryFormat( 533b9c1b51eSKate Stone TypeSummaryImpl::Flags(), 534b9c1b51eSKate Stone m_option_variable.summary_string.GetCurrentValue())); 535f9fa6ee5SEnrico Granata 536b9c1b51eSKate Stone DumpValueObjectOptions options(m_varobj_options.GetAsDumpOptions( 537b9c1b51eSKate Stone eLanguageRuntimeDescriptionDisplayVerbosityFull, eFormatDefault, 538b9c1b51eSKate Stone summary_format_sp)); 539379447a7SEnrico Granata 540b9c1b51eSKate Stone const SymbolContext &sym_ctx = 541b9c1b51eSKate Stone frame->GetSymbolContext(eSymbolContextFunction); 5426754e04fSEnrico Granata if (sym_ctx.function && sym_ctx.function->IsTopLevelFunction()) 5436754e04fSEnrico Granata m_option_variable.show_globals = true; 5446754e04fSEnrico Granata 545b9c1b51eSKate Stone if (variable_list) { 5461deb7962SGreg Clayton const Format format = m_option_format.GetFormat(); 5470c489f58SEnrico Granata options.SetFormat(format); 5481deb7962SGreg Clayton 54911eb9c64SZachary Turner if (!command.empty()) { 55046747022SGreg Clayton VariableList regex_var_list; 55146747022SGreg Clayton 55205097246SAdrian Prantl // If we have any args to the variable command, we will make variable 55305097246SAdrian Prantl // objects from them... 554f965cc86SZachary Turner for (auto &entry : command) { 555b9c1b51eSKate Stone if (m_option_variable.use_regex) { 556c7bece56SGreg Clayton const size_t regex_start_index = regex_var_list.GetSize(); 557f965cc86SZachary Turner llvm::StringRef name_str = entry.ref; 55895eae423SZachary Turner RegularExpression regex(name_str); 55995eae423SZachary Turner if (regex.Compile(name_str)) { 56046747022SGreg Clayton size_t num_matches = 0; 561b9c1b51eSKate Stone const size_t num_new_regex_vars = 562b9c1b51eSKate Stone variable_list->AppendVariablesIfUnique(regex, regex_var_list, 56378a685aaSJim Ingham num_matches); 564b9c1b51eSKate Stone if (num_new_regex_vars > 0) { 565b9c1b51eSKate Stone for (size_t regex_idx = regex_start_index, 566b9c1b51eSKate Stone end_index = regex_var_list.GetSize(); 567b9c1b51eSKate Stone regex_idx < end_index; ++regex_idx) { 56846747022SGreg Clayton var_sp = regex_var_list.GetVariableAtIndex(regex_idx); 569b9c1b51eSKate Stone if (var_sp) { 570b9c1b51eSKate Stone valobj_sp = frame->GetValueObjectForFrameVariable( 571b9c1b51eSKate Stone var_sp, m_varobj_options.use_dynamic); 572b9c1b51eSKate Stone if (valobj_sp) { 57373418dfeSEnrico Granata std::string scope_string; 57473418dfeSEnrico Granata if (m_option_variable.show_scope) 57573418dfeSEnrico Granata scope_string = GetScopeString(var_sp).str(); 57673418dfeSEnrico Granata 57773418dfeSEnrico Granata if (!scope_string.empty()) 578771ef6d4SMalcolm Parsons s.PutCString(scope_string); 57973418dfeSEnrico Granata 580b9c1b51eSKate Stone if (m_option_variable.show_decl && 581b9c1b51eSKate Stone var_sp->GetDeclaration().GetFile()) { 58245ba8543SGreg Clayton bool show_fullpaths = false; 58345ba8543SGreg Clayton bool show_module = true; 584b9c1b51eSKate Stone if (var_sp->DumpDeclaration(&s, show_fullpaths, 585b9c1b51eSKate Stone show_module)) 58646747022SGreg Clayton s.PutCString(": "); 58746747022SGreg Clayton } 5884d93b8cdSEnrico Granata valobj_sp->Dump(result.GetOutputStream(), options); 58946747022SGreg Clayton } 59046747022SGreg Clayton } 59146747022SGreg Clayton } 592b9c1b51eSKate Stone } else if (num_matches == 0) { 593b9c1b51eSKate Stone result.GetErrorStream().Printf("error: no variables matched " 594b9c1b51eSKate Stone "the regular expression '%s'.\n", 595f965cc86SZachary Turner entry.c_str()); 59646747022SGreg Clayton } 597b9c1b51eSKate Stone } else { 59846747022SGreg Clayton char regex_error[1024]; 59946747022SGreg Clayton if (regex.GetErrorAsCString(regex_error, sizeof(regex_error))) 60046747022SGreg Clayton result.GetErrorStream().Printf("error: %s\n", regex_error); 60146747022SGreg Clayton else 602b9c1b51eSKate Stone result.GetErrorStream().Printf( 603b9c1b51eSKate Stone "error: unknown regex error when compiling '%s'\n", 604f965cc86SZachary Turner entry.c_str()); 60546747022SGreg Clayton } 606b9c1b51eSKate Stone } else // No regex, either exact variable names or variable 607b9c1b51eSKate Stone // expressions. 60846747022SGreg Clayton { 60997206d57SZachary Turner Status error; 610b9c1b51eSKate Stone uint32_t expr_path_options = 611b9c1b51eSKate Stone StackFrame::eExpressionPathOptionCheckPtrVsMember | 61246252398SEnrico Granata StackFrame::eExpressionPathOptionsAllowDirectIVarAccess | 61346252398SEnrico Granata StackFrame::eExpressionPathOptionsInspectAnonymousUnions; 6142837b766SJim Ingham lldb::VariableSP var_sp; 615b9c1b51eSKate Stone valobj_sp = frame->GetValueForVariableExpressionPath( 616f965cc86SZachary Turner entry.ref, m_varobj_options.use_dynamic, expr_path_options, 617b9c1b51eSKate Stone var_sp, error); 618b9c1b51eSKate Stone if (valobj_sp) { 61973418dfeSEnrico Granata std::string scope_string; 62073418dfeSEnrico Granata if (m_option_variable.show_scope) 62173418dfeSEnrico Granata scope_string = GetScopeString(var_sp).str(); 62273418dfeSEnrico Granata 62373418dfeSEnrico Granata if (!scope_string.empty()) 624771ef6d4SMalcolm Parsons s.PutCString(scope_string); 625b9c1b51eSKate Stone if (m_option_variable.show_decl && var_sp && 626b9c1b51eSKate Stone var_sp->GetDeclaration().GetFile()) { 627a134cc1bSGreg Clayton var_sp->GetDeclaration().DumpStopContext(&s, false); 628a134cc1bSGreg Clayton s.PutCString(": "); 629a134cc1bSGreg Clayton } 6300c489f58SEnrico Granata 6310c489f58SEnrico Granata options.SetFormat(format); 632b9c1b51eSKate Stone options.SetVariableFormatDisplayLanguage( 633b9c1b51eSKate Stone valobj_sp->GetPreferredDisplayLanguage()); 634887062aeSJohnny Chen 635887062aeSJohnny Chen Stream &output_stream = result.GetOutputStream(); 636f965cc86SZachary Turner options.SetRootValueObjectName( 637f965cc86SZachary Turner valobj_sp->GetParent() ? entry.c_str() : nullptr); 6384d93b8cdSEnrico Granata valobj_sp->Dump(output_stream, options); 639b9c1b51eSKate Stone } else { 640c8ecc2a9SEugene Zelenko const char *error_cstr = error.AsCString(nullptr); 64154979cddSGreg Clayton if (error_cstr) 64254979cddSGreg Clayton result.GetErrorStream().Printf("error: %s\n", error_cstr); 64354979cddSGreg Clayton else 644b9c1b51eSKate Stone result.GetErrorStream().Printf("error: unable to find any " 645b9c1b51eSKate Stone "variable expression path that " 646b9c1b51eSKate Stone "matches '%s'.\n", 647f965cc86SZachary Turner entry.c_str()); 6486d56d2ceSJim Ingham } 6496d56d2ceSJim Ingham } 6506d56d2ceSJim Ingham } 651b9c1b51eSKate Stone } else // No command arg specified. Use variable_list, instead. 6526d56d2ceSJim Ingham { 653c7bece56SGreg Clayton const size_t num_variables = variable_list->GetSize(); 654b9c1b51eSKate Stone if (num_variables > 0) { 655b9c1b51eSKate Stone for (size_t i = 0; i < num_variables; i++) { 6561a65ae11SGreg Clayton var_sp = variable_list->GetVariableAtIndex(i); 657f955efc0SSylvestre Ledru switch (var_sp->GetScope()) { 658eb236735SJim Ingham case eValueTypeVariableGlobal: 659eb236735SJim Ingham if (!m_option_variable.show_globals) 660eb236735SJim Ingham continue; 661eb236735SJim Ingham break; 662eb236735SJim Ingham case eValueTypeVariableStatic: 663eb236735SJim Ingham if (!m_option_variable.show_globals) 664eb236735SJim Ingham continue; 665eb236735SJim Ingham break; 666eb236735SJim Ingham case eValueTypeVariableArgument: 667eb236735SJim Ingham if (!m_option_variable.show_args) 668eb236735SJim Ingham continue; 669eb236735SJim Ingham break; 670eb236735SJim Ingham case eValueTypeVariableLocal: 671eb236735SJim Ingham if (!m_option_variable.show_locals) 672eb236735SJim Ingham continue; 673eb236735SJim Ingham break; 674eb236735SJim Ingham default: 675eb236735SJim Ingham continue; 676eb236735SJim Ingham break; 677eb236735SJim Ingham } 678560558ebSEnrico Granata std::string scope_string; 679eb236735SJim Ingham if (m_option_variable.show_scope) 68073418dfeSEnrico Granata scope_string = GetScopeString(var_sp).str(); 6816d56d2ceSJim Ingham 68205097246SAdrian Prantl // Use the variable object code to make sure we are using the same 68305097246SAdrian Prantl // APIs as the public API will be using... 684b9c1b51eSKate Stone valobj_sp = frame->GetValueObjectForFrameVariable( 685b9c1b51eSKate Stone var_sp, m_varobj_options.use_dynamic); 686b9c1b51eSKate Stone if (valobj_sp) { 68705097246SAdrian Prantl // When dumping all variables, don't print any variables that are 68805097246SAdrian Prantl // not in scope to avoid extra unneeded output 689b9c1b51eSKate Stone if (valobj_sp->IsInScope()) { 690b9c1b51eSKate Stone if (!valobj_sp->GetTargetSP() 691b9c1b51eSKate Stone ->GetDisplayRuntimeSupportValues() && 692c8ecc2a9SEugene Zelenko valobj_sp->IsRuntimeSupportValue()) 693560558ebSEnrico Granata continue; 694560558ebSEnrico Granata 695560558ebSEnrico Granata if (!scope_string.empty()) 696771ef6d4SMalcolm Parsons s.PutCString(scope_string); 697560558ebSEnrico Granata 698b9c1b51eSKate Stone if (m_option_variable.show_decl && 699b9c1b51eSKate Stone var_sp->GetDeclaration().GetFile()) { 700a134cc1bSGreg Clayton var_sp->GetDeclaration().DumpStopContext(&s, false); 701a134cc1bSGreg Clayton s.PutCString(": "); 702a134cc1bSGreg Clayton } 7030c489f58SEnrico Granata 7040c489f58SEnrico Granata options.SetFormat(format); 705b9c1b51eSKate Stone options.SetVariableFormatDisplayLanguage( 706b9c1b51eSKate Stone valobj_sp->GetPreferredDisplayLanguage()); 707f965cc86SZachary Turner options.SetRootValueObjectName( 708f965cc86SZachary Turner var_sp ? var_sp->GetName().AsCString() : nullptr); 7094d93b8cdSEnrico Granata valobj_sp->Dump(result.GetOutputStream(), options); 710a134cc1bSGreg Clayton } 711a134cc1bSGreg Clayton } 7126d56d2ceSJim Ingham } 7136d56d2ceSJim Ingham } 7146d56d2ceSJim Ingham } 7156d56d2ceSJim Ingham result.SetStatus(eReturnStatusSuccessFinishResult); 7166d56d2ceSJim Ingham } 71761a80ba6SEnrico Granata 718b9c1b51eSKate Stone if (m_interpreter.TruncationWarningNecessary()) { 71961a80ba6SEnrico Granata result.GetOutputStream().Printf(m_interpreter.TruncationWarningText(), 72061a80ba6SEnrico Granata m_cmd_name.c_str()); 72161a80ba6SEnrico Granata m_interpreter.TruncationWarningGiven(); 72261a80ba6SEnrico Granata } 72361a80ba6SEnrico Granata 72424fff242SDavide Italiano // Increment statistics. 72524fff242SDavide Italiano bool res = result.Succeeded(); 7260df817aaSDavide Italiano Target *target = GetSelectedOrDummyTarget(); 72724fff242SDavide Italiano if (res) 72824fff242SDavide Italiano target->IncrementStats(StatisticKind::FrameVarSuccess); 72924fff242SDavide Italiano else 73024fff242SDavide Italiano target->IncrementStats(StatisticKind::FrameVarFailure); 73124fff242SDavide Italiano return res; 7326d56d2ceSJim Ingham } 7336d56d2ceSJim Ingham 734c8ecc2a9SEugene Zelenko protected: 7352837b766SJim Ingham OptionGroupOptions m_option_group; 736715c2365SGreg Clayton OptionGroupVariable m_option_variable; 7371deb7962SGreg Clayton OptionGroupFormat m_option_format; 7382837b766SJim Ingham OptionGroupValueObjectDisplay m_varobj_options; 7396d56d2ceSJim Ingham }; 7406d56d2ceSJim Ingham 74130fdc8d8SChris Lattner #pragma mark CommandObjectMultiwordFrame 74230fdc8d8SChris Lattner 74330fdc8d8SChris Lattner //------------------------------------------------------------------------- 74430fdc8d8SChris Lattner // CommandObjectMultiwordFrame 74530fdc8d8SChris Lattner //------------------------------------------------------------------------- 74630fdc8d8SChris Lattner 747b9c1b51eSKate Stone CommandObjectMultiwordFrame::CommandObjectMultiwordFrame( 748b9c1b51eSKate Stone CommandInterpreter &interpreter) 749b9c1b51eSKate Stone : CommandObjectMultiword(interpreter, "frame", "Commands for selecting and " 750b9c1b51eSKate Stone "examing the current " 751b9c1b51eSKate Stone "thread's stack frames.", 752b9c1b51eSKate Stone "frame <subcommand> [<subcommand-options>]") { 753b9c1b51eSKate Stone LoadSubCommand("diagnose", 754b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameDiagnose(interpreter))); 755b9c1b51eSKate Stone LoadSubCommand("info", 756b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameInfo(interpreter))); 757b9c1b51eSKate Stone LoadSubCommand("select", 758b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameSelect(interpreter))); 759b9c1b51eSKate Stone LoadSubCommand("variable", 760b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameVariable(interpreter))); 76130fdc8d8SChris Lattner } 76230fdc8d8SChris Lattner 763c8ecc2a9SEugene Zelenko CommandObjectMultiwordFrame::~CommandObjectMultiwordFrame() = default; 764