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" 20de6bd243SJohnny Chen #include "lldb/Core/StreamString.h" 2130fdc8d8SChris Lattner #include "lldb/Core/Timer.h" 226d56d2ceSJim Ingham #include "lldb/Core/Value.h" 236d56d2ceSJim Ingham #include "lldb/Core/ValueObject.h" 246d56d2ceSJim Ingham #include "lldb/Core/ValueObjectVariable.h" 255548cb50SEnrico Granata #include "lldb/DataFormatters/DataVisualization.h" 264d93b8cdSEnrico Granata #include "lldb/DataFormatters/ValueObjectPrinter.h" 277fb56d0aSGreg Clayton #include "lldb/Host/Host.h" 285275aaa0SVince Harron #include "lldb/Host/StringConvert.h" 296d56d2ceSJim Ingham #include "lldb/Interpreter/Args.h" 3030fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 3130fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 321deb7962SGreg Clayton #include "lldb/Interpreter/OptionGroupFormat.h" 332837b766SJim Ingham #include "lldb/Interpreter/OptionGroupValueObjectDisplay.h" 34715c2365SGreg Clayton #include "lldb/Interpreter/OptionGroupVariable.h" 35b9c1b51eSKate Stone #include "lldb/Interpreter/Options.h" 366d56d2ceSJim Ingham #include "lldb/Symbol/ClangASTContext.h" 37b9c1b51eSKate Stone #include "lldb/Symbol/CompilerType.h" 386754e04fSEnrico Granata #include "lldb/Symbol/Function.h" 396d56d2ceSJim Ingham #include "lldb/Symbol/ObjectFile.h" 406d56d2ceSJim Ingham #include "lldb/Symbol/SymbolContext.h" 416d56d2ceSJim Ingham #include "lldb/Symbol/Type.h" 426d56d2ceSJim Ingham #include "lldb/Symbol/Variable.h" 436d56d2ceSJim Ingham #include "lldb/Symbol/VariableList.h" 4430fdc8d8SChris Lattner #include "lldb/Target/Process.h" 45b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h" 464740a734SSean Callanan #include "lldb/Target/StopInfo.h" 476d56d2ceSJim Ingham #include "lldb/Target/Target.h" 48b9c1b51eSKate Stone #include "lldb/Target/Thread.h" 494740a734SSean Callanan #include "lldb/Utility/LLDBAssert.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 641f0f5b5bSZachary Turner static OptionDefinition g_frame_diag_options[] = { 651f0f5b5bSZachary Turner // clang-format off 661f0f5b5bSZachary Turner { LLDB_OPT_SET_1, false, "register", 'r', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeRegisterName, "A register to diagnose." }, 671f0f5b5bSZachary Turner { LLDB_OPT_SET_1, false, "address", 'a', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeAddress, "An address to diagnose." }, 681f0f5b5bSZachary Turner { LLDB_OPT_SET_1, false, "offset", 'o', OptionParser::eRequiredArgument, nullptr, 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 80b9c1b51eSKate Stone Error SetOptionValue(uint32_t option_idx, const char *option_arg, 81b9c1b51eSKate Stone ExecutionContext *execution_context) override { 824740a734SSean Callanan Error 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': { 904740a734SSean Callanan bool success = false; 914740a734SSean Callanan 924740a734SSean Callanan address = StringConvert::ToUInt64(option_arg, 0, 0, &success); 93b9c1b51eSKate Stone if (!success) { 944740a734SSean Callanan address.reset(); 95b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid address argument '%s'", 96b9c1b51eSKate Stone option_arg); 974740a734SSean Callanan } 98b9c1b51eSKate Stone } break; 994740a734SSean Callanan 100b9c1b51eSKate Stone case 'o': { 1014740a734SSean Callanan bool success = false; 1024740a734SSean Callanan 1034740a734SSean Callanan offset = StringConvert::ToSInt64(option_arg, 0, 0, &success); 104b9c1b51eSKate Stone if (!success) { 1054740a734SSean Callanan offset.reset(); 106b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid offset argument '%s'", 107b9c1b51eSKate Stone option_arg); 1084740a734SSean Callanan } 109b9c1b51eSKate Stone } break; 1104740a734SSean Callanan 1114740a734SSean Callanan default: 112b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid short option character '%c'", 113b9c1b51eSKate Stone short_option); 1144740a734SSean Callanan break; 1154740a734SSean Callanan } 1164740a734SSean Callanan 1174740a734SSean Callanan return error; 1184740a734SSean Callanan } 1194740a734SSean Callanan 120b9c1b51eSKate Stone void OptionParsingStarting(ExecutionContext *execution_context) override { 1214740a734SSean Callanan address.reset(); 1224740a734SSean Callanan reg.reset(); 1234740a734SSean Callanan offset.reset(); 1244740a734SSean Callanan } 1254740a734SSean Callanan 1261f0f5b5bSZachary Turner llvm::ArrayRef<OptionDefinition> GetDefinitions() override { 12770602439SZachary Turner return llvm::makeArrayRef(g_frame_diag_options); 1281f0f5b5bSZachary Turner } 1294740a734SSean Callanan 1304740a734SSean Callanan // Options. 1314740a734SSean Callanan llvm::Optional<lldb::addr_t> address; 1324740a734SSean Callanan llvm::Optional<ConstString> reg; 1334740a734SSean Callanan llvm::Optional<int64_t> offset; 1344740a734SSean Callanan }; 1354740a734SSean Callanan 1364740a734SSean Callanan CommandObjectFrameDiagnose(CommandInterpreter &interpreter) 1374740a734SSean Callanan : CommandObjectParsed(interpreter, "frame diagnose", 138b9c1b51eSKate Stone "Try to determine what path path the current stop " 139b9c1b51eSKate Stone "location used to get to a register or address", 140b9c1b51eSKate Stone nullptr, 141b9c1b51eSKate Stone eCommandRequiresThread | eCommandTryTargetAPILock | 142b9c1b51eSKate Stone eCommandProcessMustBeLaunched | 1434740a734SSean Callanan eCommandProcessMustBePaused), 144b9c1b51eSKate Stone m_options() { 1454740a734SSean Callanan CommandArgumentEntry arg; 1464740a734SSean Callanan CommandArgumentData index_arg; 1474740a734SSean Callanan 1484740a734SSean Callanan // Define the first (and only) variant of this arg. 1494740a734SSean Callanan index_arg.arg_type = eArgTypeFrameIndex; 1504740a734SSean Callanan index_arg.arg_repetition = eArgRepeatOptional; 1514740a734SSean Callanan 152b9c1b51eSKate Stone // There is only one variant this argument could be; put it into the 153b9c1b51eSKate Stone // argument entry. 1544740a734SSean Callanan arg.push_back(index_arg); 1554740a734SSean Callanan 1564740a734SSean Callanan // Push the data for the first argument into the m_arguments vector. 1574740a734SSean Callanan m_arguments.push_back(arg); 1584740a734SSean Callanan } 1594740a734SSean Callanan 1604740a734SSean Callanan ~CommandObjectFrameDiagnose() override = default; 1614740a734SSean Callanan 162b9c1b51eSKate Stone Options *GetOptions() override { return &m_options; } 1634740a734SSean Callanan 1644740a734SSean Callanan protected: 165b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 1664740a734SSean Callanan Thread *thread = m_exe_ctx.GetThreadPtr(); 1674740a734SSean Callanan StackFrameSP frame_sp = thread->GetSelectedFrame(); 1684740a734SSean Callanan 1694740a734SSean Callanan ValueObjectSP valobj_sp; 1704740a734SSean Callanan 171b9c1b51eSKate Stone if (m_options.address.hasValue()) { 172b9c1b51eSKate Stone if (m_options.reg.hasValue() || m_options.offset.hasValue()) { 173b9c1b51eSKate Stone result.AppendError( 174b9c1b51eSKate Stone "`frame diagnose --address` is incompatible with other arguments."); 1754740a734SSean Callanan result.SetStatus(eReturnStatusFailed); 1764740a734SSean Callanan return false; 1774740a734SSean Callanan } 1784740a734SSean Callanan valobj_sp = frame_sp->GuessValueForAddress(m_options.address.getValue()); 179b9c1b51eSKate Stone } else if (m_options.reg.hasValue()) { 180b9c1b51eSKate Stone valobj_sp = frame_sp->GuessValueForRegisterAndOffset( 181b9c1b51eSKate Stone m_options.reg.getValue(), m_options.offset.getValueOr(0)); 182b9c1b51eSKate Stone } else { 1834740a734SSean Callanan StopInfoSP stop_info_sp = thread->GetStopInfo(); 184b9c1b51eSKate Stone if (!stop_info_sp) { 1854740a734SSean Callanan result.AppendError("No arguments provided, and no stop info."); 1864740a734SSean Callanan result.SetStatus(eReturnStatusFailed); 1874740a734SSean Callanan return false; 1884740a734SSean Callanan } 1894740a734SSean Callanan 1904740a734SSean Callanan valobj_sp = StopInfo::GetCrashingDereference(stop_info_sp); 1914740a734SSean Callanan } 1924740a734SSean Callanan 193b9c1b51eSKate Stone if (!valobj_sp) { 1944740a734SSean Callanan result.AppendError("No diagnosis available."); 1954740a734SSean Callanan result.SetStatus(eReturnStatusFailed); 1964740a734SSean Callanan return false; 1974740a734SSean Callanan } 1984740a734SSean Callanan 1994740a734SSean Callanan const bool qualify_cxx_base_classes = false; 2004740a734SSean Callanan 201b9c1b51eSKate Stone DumpValueObjectOptions::DeclPrintingHelper helper = 202b9c1b51eSKate Stone [&valobj_sp, qualify_cxx_base_classes]( 203b9c1b51eSKate Stone ConstString type, ConstString var, 204b9c1b51eSKate Stone const DumpValueObjectOptions &opts, Stream &stream) -> bool { 205b9c1b51eSKate Stone const ValueObject::GetExpressionPathFormat format = ValueObject:: 206b9c1b51eSKate Stone GetExpressionPathFormat::eGetExpressionPathFormatHonorPointers; 2074740a734SSean Callanan valobj_sp->GetExpressionPath(stream, qualify_cxx_base_classes, format); 2084740a734SSean Callanan stream.PutCString(" ="); 2094740a734SSean Callanan return true; 2104740a734SSean Callanan }; 2114740a734SSean Callanan 2124740a734SSean Callanan DumpValueObjectOptions options; 2134740a734SSean Callanan options.SetDeclPrintingHelper(helper); 214b9c1b51eSKate Stone ValueObjectPrinter printer(valobj_sp.get(), &result.GetOutputStream(), 215b9c1b51eSKate Stone options); 2164740a734SSean Callanan printer.PrintValueObject(); 2174740a734SSean Callanan 2184740a734SSean Callanan return true; 2194740a734SSean Callanan } 2204740a734SSean Callanan 2214740a734SSean Callanan protected: 2224740a734SSean Callanan CommandOptions m_options; 2234740a734SSean Callanan }; 2244740a734SSean Callanan 22530fdc8d8SChris Lattner #pragma mark CommandObjectFrameInfo 22630fdc8d8SChris Lattner 22730fdc8d8SChris Lattner //------------------------------------------------------------------------- 22830fdc8d8SChris Lattner // CommandObjectFrameInfo 22930fdc8d8SChris Lattner //------------------------------------------------------------------------- 23030fdc8d8SChris Lattner 231b9c1b51eSKate Stone class CommandObjectFrameInfo : public CommandObjectParsed { 23230fdc8d8SChris Lattner public: 2337428a18cSKate Stone CommandObjectFrameInfo(CommandInterpreter &interpreter) 234b9c1b51eSKate Stone : CommandObjectParsed( 235b9c1b51eSKate Stone interpreter, "frame info", "List information about the current " 236b9c1b51eSKate Stone "stack frame in the current thread.", 237b9c1b51eSKate Stone "frame info", 238b9c1b51eSKate Stone eCommandRequiresFrame | eCommandTryTargetAPILock | 239b9c1b51eSKate Stone eCommandProcessMustBeLaunched | eCommandProcessMustBePaused) {} 24030fdc8d8SChris Lattner 241c8ecc2a9SEugene Zelenko ~CommandObjectFrameInfo() override = default; 24230fdc8d8SChris Lattner 2435a988416SJim Ingham protected: 244b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 245f9fc609fSGreg Clayton m_exe_ctx.GetFrameRef().DumpUsingSettingsFormat(&result.GetOutputStream()); 24630fdc8d8SChris Lattner result.SetStatus(eReturnStatusSuccessFinishResult); 24730fdc8d8SChris Lattner return result.Succeeded(); 24830fdc8d8SChris Lattner } 24930fdc8d8SChris Lattner }; 25030fdc8d8SChris Lattner 25130fdc8d8SChris Lattner #pragma mark CommandObjectFrameSelect 25230fdc8d8SChris Lattner 25330fdc8d8SChris Lattner //------------------------------------------------------------------------- 25430fdc8d8SChris Lattner // CommandObjectFrameSelect 25530fdc8d8SChris Lattner //------------------------------------------------------------------------- 25630fdc8d8SChris Lattner 2571f0f5b5bSZachary Turner static OptionDefinition g_frame_select_options[] = { 2581f0f5b5bSZachary Turner // clang-format off 2591f0f5b5bSZachary Turner { LLDB_OPT_SET_1, false, "relative", 'r', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeOffset, "A relative frame index offset from the current frame index." }, 2601f0f5b5bSZachary Turner // clang-format on 2611f0f5b5bSZachary Turner }; 2621f0f5b5bSZachary Turner 263b9c1b51eSKate Stone class CommandObjectFrameSelect : public CommandObjectParsed { 26430fdc8d8SChris Lattner public: 265b9c1b51eSKate Stone class CommandOptions : public Options { 266864174e1SGreg Clayton public: 267b9c1b51eSKate Stone CommandOptions() : Options() { OptionParsingStarting(nullptr); } 268864174e1SGreg Clayton 269c8ecc2a9SEugene Zelenko ~CommandOptions() override = default; 270864174e1SGreg Clayton 271b9c1b51eSKate Stone Error SetOptionValue(uint32_t option_idx, const char *option_arg, 272b9c1b51eSKate Stone ExecutionContext *execution_context) override { 273864174e1SGreg Clayton Error error; 274864174e1SGreg Clayton bool success = false; 2753bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 276b9c1b51eSKate Stone switch (short_option) { 277864174e1SGreg Clayton case 'r': 278b9c1b51eSKate Stone relative_frame_offset = 279b9c1b51eSKate Stone StringConvert::ToSInt32(option_arg, INT32_MIN, 0, &success); 280864174e1SGreg Clayton if (!success) 281b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid frame offset argument '%s'", 282b9c1b51eSKate Stone option_arg); 283864174e1SGreg Clayton break; 284864174e1SGreg Clayton 285864174e1SGreg Clayton default: 286b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid short option character '%c'", 287b9c1b51eSKate Stone short_option); 288864174e1SGreg Clayton break; 289864174e1SGreg Clayton } 290864174e1SGreg Clayton 291864174e1SGreg Clayton return error; 292864174e1SGreg Clayton } 293864174e1SGreg Clayton 294b9c1b51eSKate Stone void OptionParsingStarting(ExecutionContext *execution_context) override { 295864174e1SGreg Clayton relative_frame_offset = INT32_MIN; 296864174e1SGreg Clayton } 297864174e1SGreg Clayton 2981f0f5b5bSZachary Turner llvm::ArrayRef<OptionDefinition> GetDefinitions() override { 29970602439SZachary Turner return llvm::makeArrayRef(g_frame_select_options); 3001f0f5b5bSZachary Turner } 301864174e1SGreg Clayton 302864174e1SGreg Clayton int32_t relative_frame_offset; 303864174e1SGreg Clayton }; 304864174e1SGreg Clayton 3057428a18cSKate Stone CommandObjectFrameSelect(CommandInterpreter &interpreter) 3067428a18cSKate Stone : CommandObjectParsed( 307b9c1b51eSKate Stone interpreter, "frame select", "Select the current stack frame by " 308b9c1b51eSKate Stone "index from within the current thread " 309b9c1b51eSKate Stone "(see 'thread backtrace'.)", 310b9c1b51eSKate Stone nullptr, 311b9c1b51eSKate Stone eCommandRequiresThread | eCommandTryTargetAPILock | 312b9c1b51eSKate Stone eCommandProcessMustBeLaunched | eCommandProcessMustBePaused), 313b9c1b51eSKate Stone m_options() { 314405fe67fSCaroline Tice CommandArgumentEntry arg; 315405fe67fSCaroline Tice CommandArgumentData index_arg; 316405fe67fSCaroline Tice 317405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 318405fe67fSCaroline Tice index_arg.arg_type = eArgTypeFrameIndex; 319864174e1SGreg Clayton index_arg.arg_repetition = eArgRepeatOptional; 320405fe67fSCaroline Tice 321b9c1b51eSKate Stone // There is only one variant this argument could be; put it into the 322b9c1b51eSKate Stone // argument entry. 323405fe67fSCaroline Tice arg.push_back(index_arg); 324405fe67fSCaroline Tice 325405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 326405fe67fSCaroline Tice m_arguments.push_back(arg); 32730fdc8d8SChris Lattner } 32830fdc8d8SChris Lattner 329c8ecc2a9SEugene Zelenko ~CommandObjectFrameSelect() override = default; 33030fdc8d8SChris Lattner 331b9c1b51eSKate Stone Options *GetOptions() override { return &m_options; } 332864174e1SGreg Clayton 3335a988416SJim Ingham protected: 334b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 335b9c1b51eSKate Stone // No need to check "thread" for validity as eCommandRequiresThread ensures 336b9c1b51eSKate Stone // it is valid 337f9fc609fSGreg Clayton Thread *thread = m_exe_ctx.GetThreadPtr(); 338f9fc609fSGreg Clayton 339864174e1SGreg Clayton uint32_t frame_idx = UINT32_MAX; 340b9c1b51eSKate Stone if (m_options.relative_frame_offset != INT32_MIN) { 341864174e1SGreg Clayton // The one and only argument is a signed relative frame index 342c14ee32dSGreg Clayton frame_idx = thread->GetSelectedFrameIndex(); 343864174e1SGreg Clayton if (frame_idx == UINT32_MAX) 344864174e1SGreg Clayton frame_idx = 0; 345864174e1SGreg Clayton 346b9c1b51eSKate Stone if (m_options.relative_frame_offset < 0) { 3473985c8c6SSaleem Abdulrasool if (static_cast<int32_t>(frame_idx) >= -m_options.relative_frame_offset) 348864174e1SGreg Clayton frame_idx += m_options.relative_frame_offset; 349b9c1b51eSKate Stone else { 350b9c1b51eSKate Stone if (frame_idx == 0) { 351b9c1b51eSKate Stone // If you are already at the bottom of the stack, then just warn and 352b9c1b51eSKate Stone // don't reset the frame. 3537428a18cSKate Stone result.AppendError("Already at the bottom of the stack."); 354213b4546SJim Ingham result.SetStatus(eReturnStatusFailed); 355213b4546SJim Ingham return false; 356b9c1b51eSKate Stone } else 357864174e1SGreg Clayton frame_idx = 0; 358864174e1SGreg Clayton } 359b9c1b51eSKate Stone } else if (m_options.relative_frame_offset > 0) { 360b9c1b51eSKate Stone // I don't want "up 20" where "20" takes you past the top of the stack 361b9c1b51eSKate Stone // to produce 362b9c1b51eSKate Stone // an error, but rather to just go to the top. So I have to count the 363b9c1b51eSKate Stone // stack here... 364b0c72a5fSJim Ingham const uint32_t num_frames = thread->GetStackFrameCount(); 365b9c1b51eSKate Stone if (static_cast<int32_t>(num_frames - frame_idx) > 366b9c1b51eSKate Stone m_options.relative_frame_offset) 367864174e1SGreg Clayton frame_idx += m_options.relative_frame_offset; 368b9c1b51eSKate Stone else { 369b9c1b51eSKate Stone if (frame_idx == num_frames - 1) { 370b9c1b51eSKate Stone // If we are already at the top of the stack, just warn and don't 371b9c1b51eSKate Stone // reset the frame. 3727428a18cSKate Stone result.AppendError("Already at the top of the stack."); 373213b4546SJim Ingham result.SetStatus(eReturnStatusFailed); 374213b4546SJim Ingham return false; 375b9c1b51eSKate Stone } else 376864174e1SGreg Clayton frame_idx = num_frames - 1; 377864174e1SGreg Clayton } 378864174e1SGreg Clayton } 379b9c1b51eSKate Stone } else { 380b9c1b51eSKate Stone if (command.GetArgumentCount() == 1) { 38130fdc8d8SChris Lattner const char *frame_idx_cstr = command.GetArgumentAtIndex(0); 382afbb0af8SJim Ingham bool success = false; 383b9c1b51eSKate Stone frame_idx = 384b9c1b51eSKate Stone StringConvert::ToUInt32(frame_idx_cstr, UINT32_MAX, 0, &success); 385b9c1b51eSKate Stone if (!success) { 386b9c1b51eSKate Stone result.AppendErrorWithFormat("invalid frame index argument '%s'.", 387b9c1b51eSKate Stone frame_idx_cstr); 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 } 396b9c1b51eSKate Stone } else { 397b9c1b51eSKate Stone result.AppendErrorWithFormat( 398b9c1b51eSKate Stone "too many arguments; expected frame-index, saw '%s'.\n", 399c2961ab8SAdrian McCarthy command.GetArgumentAtIndex(0)); 400b9c1b51eSKate Stone m_options.GenerateOptionUsage( 401b9c1b51eSKate Stone result.GetErrorStream(), this, 402b9c1b51eSKate Stone GetCommandInterpreter().GetDebugger().GetTerminalWidth()); 403c2961ab8SAdrian McCarthy return false; 404864174e1SGreg Clayton } 405864174e1SGreg Clayton } 40630fdc8d8SChris Lattner 407b9c1b51eSKate Stone bool success = thread->SetSelectedFrameByIndexNoisily( 408b9c1b51eSKate Stone frame_idx, result.GetOutputStream()); 409b9c1b51eSKate Stone if (success) { 410f9fc609fSGreg Clayton m_exe_ctx.SetFrameSP(thread->GetSelectedFrame()); 41130fdc8d8SChris Lattner result.SetStatus(eReturnStatusSuccessFinishResult); 412b9c1b51eSKate Stone } else { 413b9c1b51eSKate Stone result.AppendErrorWithFormat("Frame index (%u) out of range.\n", 414b9c1b51eSKate Stone frame_idx); 41530fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 41693208b86SJim Ingham } 41793208b86SJim Ingham 41893208b86SJim Ingham return result.Succeeded(); 41930fdc8d8SChris Lattner } 420864174e1SGreg Clayton 421c8ecc2a9SEugene Zelenko protected: 422864174e1SGreg Clayton CommandOptions m_options; 423864174e1SGreg Clayton }; 424864174e1SGreg Clayton 4256d56d2ceSJim Ingham #pragma mark CommandObjectFrameVariable 4266d56d2ceSJim Ingham //---------------------------------------------------------------------- 4276d56d2ceSJim Ingham // List images with associated information 4286d56d2ceSJim Ingham //---------------------------------------------------------------------- 429b9c1b51eSKate Stone class CommandObjectFrameVariable : public CommandObjectParsed { 4306d56d2ceSJim Ingham public: 4317428a18cSKate Stone CommandObjectFrameVariable(CommandInterpreter &interpreter) 4327428a18cSKate Stone : CommandObjectParsed( 433b9c1b51eSKate Stone interpreter, "frame variable", 434b9c1b51eSKate Stone "Show variables for the current stack frame. Defaults to all " 4357428a18cSKate Stone "arguments and local variables in scope. Names of argument, " 4367428a18cSKate Stone "local, file static and file global variables can be specified. " 437ed8a705cSGreg Clayton "Children of aggregate variables can be specified such as " 438973cf9e8SJohnny Chen "'var->child.x'.", 439b9c1b51eSKate Stone nullptr, eCommandRequiresFrame | eCommandTryTargetAPILock | 440b9c1b51eSKate Stone eCommandProcessMustBeLaunched | 4417428a18cSKate Stone eCommandProcessMustBePaused | eCommandRequiresProcess), 442e1cfbc79STodd Fiala m_option_group(), 443b9c1b51eSKate Stone m_option_variable( 444b9c1b51eSKate Stone true), // Include the frame specific options by passing "true" 4451deb7962SGreg Clayton m_option_format(eFormatDefault), 446b9c1b51eSKate Stone m_varobj_options() { 447405fe67fSCaroline Tice CommandArgumentEntry arg; 448405fe67fSCaroline Tice CommandArgumentData var_name_arg; 449405fe67fSCaroline Tice 450405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 451405fe67fSCaroline Tice var_name_arg.arg_type = eArgTypeVarName; 452405fe67fSCaroline Tice var_name_arg.arg_repetition = eArgRepeatStar; 453405fe67fSCaroline Tice 454b9c1b51eSKate Stone // There is only one variant this argument could be; put it into the 455b9c1b51eSKate Stone // argument entry. 456405fe67fSCaroline Tice arg.push_back(var_name_arg); 457405fe67fSCaroline Tice 458405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 459405fe67fSCaroline Tice m_arguments.push_back(arg); 4602837b766SJim Ingham 461715c2365SGreg Clayton m_option_group.Append(&m_option_variable, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 462b9c1b51eSKate Stone m_option_group.Append(&m_option_format, 463b9c1b51eSKate Stone OptionGroupFormat::OPTION_GROUP_FORMAT | 464b9c1b51eSKate Stone OptionGroupFormat::OPTION_GROUP_GDB_FMT, 465b9c1b51eSKate Stone LLDB_OPT_SET_1); 4662837b766SJim Ingham m_option_group.Append(&m_varobj_options, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 4672837b766SJim Ingham m_option_group.Finalize(); 4686d56d2ceSJim Ingham } 4696d56d2ceSJim Ingham 470c8ecc2a9SEugene Zelenko ~CommandObjectFrameVariable() override = default; 4716d56d2ceSJim Ingham 472b9c1b51eSKate Stone Options *GetOptions() override { return &m_option_group; } 473f21feadcSGreg Clayton 474b9c1b51eSKate Stone int HandleArgumentCompletion(Args &input, int &cursor_index, 475f21feadcSGreg Clayton int &cursor_char_position, 476f21feadcSGreg Clayton OptionElementVector &opt_element_vector, 477b9c1b51eSKate Stone int match_start_point, int max_return_elements, 478f21feadcSGreg Clayton bool &word_complete, 479b9c1b51eSKate Stone StringList &matches) override { 480f21feadcSGreg Clayton // Arguments are the standard source file completer. 481f21feadcSGreg Clayton std::string completion_str(input.GetArgumentAtIndex(cursor_index)); 482f21feadcSGreg Clayton completion_str.erase(cursor_char_position); 483f21feadcSGreg Clayton 484b9c1b51eSKate Stone CommandCompletions::InvokeCommonCompletionCallbacks( 485b9c1b51eSKate Stone GetCommandInterpreter(), CommandCompletions::eVariablePathCompletion, 486b9c1b51eSKate Stone completion_str.c_str(), match_start_point, max_return_elements, nullptr, 487b9c1b51eSKate Stone word_complete, matches); 488f21feadcSGreg Clayton return matches.GetSize(); 489f21feadcSGreg Clayton } 4906d56d2ceSJim Ingham 4915a988416SJim Ingham protected: 492*73418dfeSEnrico Granata llvm::StringRef GetScopeString(VariableSP var_sp) { 493*73418dfeSEnrico Granata if (!var_sp) 494*73418dfeSEnrico Granata return llvm::StringRef::withNullAsEmpty(nullptr); 495*73418dfeSEnrico Granata 496*73418dfeSEnrico Granata switch (var_sp->GetScope()) { 497*73418dfeSEnrico Granata case eValueTypeVariableGlobal: 498*73418dfeSEnrico Granata return "GLOBAL: "; 499*73418dfeSEnrico Granata case eValueTypeVariableStatic: 500*73418dfeSEnrico Granata return "STATIC: "; 501*73418dfeSEnrico Granata case eValueTypeVariableArgument: 502*73418dfeSEnrico Granata return "ARG: "; 503*73418dfeSEnrico Granata case eValueTypeVariableLocal: 504*73418dfeSEnrico Granata return "LOCAL: "; 505*73418dfeSEnrico Granata case eValueTypeVariableThreadLocal: 506*73418dfeSEnrico Granata return "THREAD: "; 507*73418dfeSEnrico Granata default: 508*73418dfeSEnrico Granata break; 509*73418dfeSEnrico Granata } 510*73418dfeSEnrico Granata 511*73418dfeSEnrico Granata return llvm::StringRef::withNullAsEmpty(nullptr); 512*73418dfeSEnrico Granata } 513*73418dfeSEnrico Granata 514b9c1b51eSKate Stone bool DoExecute(Args &command, CommandReturnObject &result) override { 515b9c1b51eSKate Stone // No need to check "frame" for validity as eCommandRequiresFrame ensures it 516b9c1b51eSKate Stone // is valid 517b57e4a1bSJason Molenda StackFrame *frame = m_exe_ctx.GetFramePtr(); 5181e49e5e7SJohnny Chen 519a134cc1bSGreg Clayton Stream &s = result.GetOutputStream(); 5206d56d2ceSJim Ingham 521b9c1b51eSKate Stone // Be careful about the stack frame, if any summary formatter runs code, it 522b9c1b51eSKate Stone // might clear the StackFrameList 523b9c1b51eSKate Stone // for the thread. So hold onto a shared pointer to the frame so it stays 524b9c1b51eSKate Stone // alive. 525650543f9SJim Ingham 526b9c1b51eSKate Stone VariableList *variable_list = 527b9c1b51eSKate Stone frame->GetVariableList(m_option_variable.show_globals); 528a134cc1bSGreg Clayton 5296d56d2ceSJim Ingham VariableSP var_sp; 5306d56d2ceSJim Ingham ValueObjectSP valobj_sp; 53178a685aaSJim Ingham 532c8ecc2a9SEugene Zelenko const char *name_cstr = nullptr; 5336d56d2ceSJim Ingham size_t idx; 5346d56d2ceSJim Ingham 535061858ceSEnrico Granata TypeSummaryImplSP summary_format_sp; 53617b11749SEnrico Granata if (!m_option_variable.summary.IsCurrentValueEmpty()) 537b9c1b51eSKate Stone DataVisualization::NamedSummaryFormats::GetSummaryFormat( 538b9c1b51eSKate Stone ConstString(m_option_variable.summary.GetCurrentValue()), 539b9c1b51eSKate Stone summary_format_sp); 54017b11749SEnrico Granata else if (!m_option_variable.summary_string.IsCurrentValueEmpty()) 541b9c1b51eSKate Stone summary_format_sp.reset(new StringSummaryFormat( 542b9c1b51eSKate Stone TypeSummaryImpl::Flags(), 543b9c1b51eSKate Stone m_option_variable.summary_string.GetCurrentValue())); 544f9fa6ee5SEnrico Granata 545b9c1b51eSKate Stone DumpValueObjectOptions options(m_varobj_options.GetAsDumpOptions( 546b9c1b51eSKate Stone eLanguageRuntimeDescriptionDisplayVerbosityFull, eFormatDefault, 547b9c1b51eSKate Stone summary_format_sp)); 548379447a7SEnrico Granata 549b9c1b51eSKate Stone const SymbolContext &sym_ctx = 550b9c1b51eSKate Stone frame->GetSymbolContext(eSymbolContextFunction); 5516754e04fSEnrico Granata if (sym_ctx.function && sym_ctx.function->IsTopLevelFunction()) 5526754e04fSEnrico Granata m_option_variable.show_globals = true; 5536754e04fSEnrico Granata 554b9c1b51eSKate Stone if (variable_list) { 5551deb7962SGreg Clayton const Format format = m_option_format.GetFormat(); 5560c489f58SEnrico Granata options.SetFormat(format); 5571deb7962SGreg Clayton 55811eb9c64SZachary Turner if (!command.empty()) { 55946747022SGreg Clayton VariableList regex_var_list; 56046747022SGreg Clayton 5616d56d2ceSJim Ingham // If we have any args to the variable command, we will make 5626d56d2ceSJim Ingham // variable objects from them... 563b9c1b51eSKate Stone for (idx = 0; (name_cstr = command.GetArgumentAtIndex(idx)) != nullptr; 564b9c1b51eSKate Stone ++idx) { 565b9c1b51eSKate Stone if (m_option_variable.use_regex) { 566c7bece56SGreg Clayton const size_t regex_start_index = regex_var_list.GetSize(); 56795eae423SZachary Turner llvm::StringRef name_str(name_cstr); 56895eae423SZachary Turner RegularExpression regex(name_str); 56995eae423SZachary Turner if (regex.Compile(name_str)) { 57046747022SGreg Clayton size_t num_matches = 0; 571b9c1b51eSKate Stone const size_t num_new_regex_vars = 572b9c1b51eSKate Stone variable_list->AppendVariablesIfUnique(regex, regex_var_list, 57378a685aaSJim Ingham num_matches); 574b9c1b51eSKate Stone if (num_new_regex_vars > 0) { 575b9c1b51eSKate Stone for (size_t regex_idx = regex_start_index, 576b9c1b51eSKate Stone end_index = regex_var_list.GetSize(); 577b9c1b51eSKate Stone regex_idx < end_index; ++regex_idx) { 57846747022SGreg Clayton var_sp = regex_var_list.GetVariableAtIndex(regex_idx); 579b9c1b51eSKate Stone if (var_sp) { 580b9c1b51eSKate Stone valobj_sp = frame->GetValueObjectForFrameVariable( 581b9c1b51eSKate Stone var_sp, m_varobj_options.use_dynamic); 582b9c1b51eSKate Stone if (valobj_sp) { 583b9c1b51eSKate Stone // if (format 584b9c1b51eSKate Stone // != 585b9c1b51eSKate Stone // eFormatDefault) 586b9c1b51eSKate Stone // valobj_sp->SetFormat 587b9c1b51eSKate Stone // (format); 588ded470d3SGreg Clayton 589*73418dfeSEnrico Granata std::string scope_string; 590*73418dfeSEnrico Granata if (m_option_variable.show_scope) 591*73418dfeSEnrico Granata scope_string = GetScopeString(var_sp).str(); 592*73418dfeSEnrico Granata 593*73418dfeSEnrico Granata if (!scope_string.empty()) 594*73418dfeSEnrico Granata s.PutCString(scope_string.c_str()); 595*73418dfeSEnrico Granata 596b9c1b51eSKate Stone if (m_option_variable.show_decl && 597b9c1b51eSKate Stone var_sp->GetDeclaration().GetFile()) { 59845ba8543SGreg Clayton bool show_fullpaths = false; 59945ba8543SGreg Clayton bool show_module = true; 600b9c1b51eSKate Stone if (var_sp->DumpDeclaration(&s, show_fullpaths, 601b9c1b51eSKate Stone show_module)) 60246747022SGreg Clayton s.PutCString(": "); 60346747022SGreg Clayton } 6044d93b8cdSEnrico Granata valobj_sp->Dump(result.GetOutputStream(), options); 60546747022SGreg Clayton } 60646747022SGreg Clayton } 60746747022SGreg Clayton } 608b9c1b51eSKate Stone } else if (num_matches == 0) { 609b9c1b51eSKate Stone result.GetErrorStream().Printf("error: no variables matched " 610b9c1b51eSKate Stone "the regular expression '%s'.\n", 611b9c1b51eSKate Stone name_cstr); 61246747022SGreg Clayton } 613b9c1b51eSKate Stone } else { 61446747022SGreg Clayton char regex_error[1024]; 61546747022SGreg Clayton if (regex.GetErrorAsCString(regex_error, sizeof(regex_error))) 61646747022SGreg Clayton result.GetErrorStream().Printf("error: %s\n", regex_error); 61746747022SGreg Clayton else 618b9c1b51eSKate Stone result.GetErrorStream().Printf( 619b9c1b51eSKate Stone "error: unknown regex error when compiling '%s'\n", 620b9c1b51eSKate Stone name_cstr); 62146747022SGreg Clayton } 622b9c1b51eSKate Stone } else // No regex, either exact variable names or variable 623b9c1b51eSKate Stone // expressions. 62446747022SGreg Clayton { 62554979cddSGreg Clayton Error error; 626b9c1b51eSKate Stone uint32_t expr_path_options = 627b9c1b51eSKate Stone StackFrame::eExpressionPathOptionCheckPtrVsMember | 62846252398SEnrico Granata StackFrame::eExpressionPathOptionsAllowDirectIVarAccess | 62946252398SEnrico Granata StackFrame::eExpressionPathOptionsInspectAnonymousUnions; 6302837b766SJim Ingham lldb::VariableSP var_sp; 631b9c1b51eSKate Stone valobj_sp = frame->GetValueForVariableExpressionPath( 632b9c1b51eSKate Stone name_cstr, m_varobj_options.use_dynamic, expr_path_options, 633b9c1b51eSKate Stone var_sp, error); 634b9c1b51eSKate Stone if (valobj_sp) { 635*73418dfeSEnrico Granata std::string scope_string; 636*73418dfeSEnrico Granata if (m_option_variable.show_scope) 637*73418dfeSEnrico Granata scope_string = GetScopeString(var_sp).str(); 638*73418dfeSEnrico Granata 639*73418dfeSEnrico Granata if (!scope_string.empty()) 640*73418dfeSEnrico Granata s.PutCString(scope_string.c_str()); 641*73418dfeSEnrico Granata 6426efba4fcSGreg Clayton // if (format != eFormatDefault) 6436efba4fcSGreg Clayton // valobj_sp->SetFormat (format); 644b9c1b51eSKate Stone if (m_option_variable.show_decl && var_sp && 645b9c1b51eSKate Stone var_sp->GetDeclaration().GetFile()) { 646a134cc1bSGreg Clayton var_sp->GetDeclaration().DumpStopContext(&s, false); 647a134cc1bSGreg Clayton s.PutCString(": "); 648a134cc1bSGreg Clayton } 6490c489f58SEnrico Granata 6500c489f58SEnrico Granata options.SetFormat(format); 651b9c1b51eSKate Stone options.SetVariableFormatDisplayLanguage( 652b9c1b51eSKate Stone valobj_sp->GetPreferredDisplayLanguage()); 653887062aeSJohnny Chen 654887062aeSJohnny Chen Stream &output_stream = result.GetOutputStream(); 655b9c1b51eSKate Stone options.SetRootValueObjectName(valobj_sp->GetParent() ? name_cstr 656b9c1b51eSKate Stone : nullptr); 6574d93b8cdSEnrico Granata valobj_sp->Dump(output_stream, options); 658b9c1b51eSKate Stone } else { 659c8ecc2a9SEugene Zelenko const char *error_cstr = error.AsCString(nullptr); 66054979cddSGreg Clayton if (error_cstr) 66154979cddSGreg Clayton result.GetErrorStream().Printf("error: %s\n", error_cstr); 66254979cddSGreg Clayton else 663b9c1b51eSKate Stone result.GetErrorStream().Printf("error: unable to find any " 664b9c1b51eSKate Stone "variable expression path that " 665b9c1b51eSKate Stone "matches '%s'.\n", 6667428a18cSKate Stone name_cstr); 6676d56d2ceSJim Ingham } 6686d56d2ceSJim Ingham } 6696d56d2ceSJim Ingham } 670b9c1b51eSKate Stone } else // No command arg specified. Use variable_list, instead. 6716d56d2ceSJim Ingham { 672c7bece56SGreg Clayton const size_t num_variables = variable_list->GetSize(); 673b9c1b51eSKate Stone if (num_variables > 0) { 674b9c1b51eSKate Stone for (size_t i = 0; i < num_variables; i++) { 6751a65ae11SGreg Clayton var_sp = variable_list->GetVariableAtIndex(i); 6766d56d2ceSJim Ingham bool dump_variable = true; 677560558ebSEnrico Granata std::string scope_string; 678715c2365SGreg Clayton if (dump_variable && m_option_variable.show_scope) 679*73418dfeSEnrico Granata scope_string = GetScopeString(var_sp).str(); 6806d56d2ceSJim Ingham 681b9c1b51eSKate Stone if (dump_variable) { 682a134cc1bSGreg Clayton // Use the variable object code to make sure we are 68358ef391fSBruce Mitchener // using the same APIs as the public API will be 684a134cc1bSGreg Clayton // using... 685b9c1b51eSKate Stone valobj_sp = frame->GetValueObjectForFrameVariable( 686b9c1b51eSKate Stone var_sp, m_varobj_options.use_dynamic); 687b9c1b51eSKate Stone if (valobj_sp) { 6886efba4fcSGreg Clayton // if (format != eFormatDefault) 689b9c1b51eSKate Stone // valobj_sp->SetFormat 690b9c1b51eSKate Stone // (format); 691ded470d3SGreg Clayton 6926f00abd5SGreg Clayton // When dumping all variables, don't print any variables 6936f00abd5SGreg Clayton // that are not in scope to avoid extra unneeded output 694b9c1b51eSKate Stone if (valobj_sp->IsInScope()) { 695b9c1b51eSKate Stone if (!valobj_sp->GetTargetSP() 696b9c1b51eSKate Stone ->GetDisplayRuntimeSupportValues() && 697c8ecc2a9SEugene Zelenko valobj_sp->IsRuntimeSupportValue()) 698560558ebSEnrico Granata continue; 699560558ebSEnrico Granata 700560558ebSEnrico Granata if (!scope_string.empty()) 701560558ebSEnrico Granata s.PutCString(scope_string.c_str()); 702560558ebSEnrico Granata 703b9c1b51eSKate Stone if (m_option_variable.show_decl && 704b9c1b51eSKate Stone var_sp->GetDeclaration().GetFile()) { 705a134cc1bSGreg Clayton var_sp->GetDeclaration().DumpStopContext(&s, false); 706a134cc1bSGreg Clayton s.PutCString(": "); 707a134cc1bSGreg Clayton } 7080c489f58SEnrico Granata 7090c489f58SEnrico Granata options.SetFormat(format); 710b9c1b51eSKate Stone options.SetVariableFormatDisplayLanguage( 711b9c1b51eSKate Stone valobj_sp->GetPreferredDisplayLanguage()); 7120c489f58SEnrico Granata options.SetRootValueObjectName(name_cstr); 7134d93b8cdSEnrico Granata valobj_sp->Dump(result.GetOutputStream(), options); 714a134cc1bSGreg Clayton } 715a134cc1bSGreg Clayton } 7166d56d2ceSJim Ingham } 7176d56d2ceSJim Ingham } 7186d56d2ceSJim Ingham } 7196f00abd5SGreg Clayton } 7206d56d2ceSJim Ingham result.SetStatus(eReturnStatusSuccessFinishResult); 7216d56d2ceSJim Ingham } 72261a80ba6SEnrico Granata 723b9c1b51eSKate Stone if (m_interpreter.TruncationWarningNecessary()) { 72461a80ba6SEnrico Granata result.GetOutputStream().Printf(m_interpreter.TruncationWarningText(), 72561a80ba6SEnrico Granata m_cmd_name.c_str()); 72661a80ba6SEnrico Granata m_interpreter.TruncationWarningGiven(); 72761a80ba6SEnrico Granata } 72861a80ba6SEnrico Granata 7296d56d2ceSJim Ingham return result.Succeeded(); 7306d56d2ceSJim Ingham } 7316d56d2ceSJim Ingham 732c8ecc2a9SEugene Zelenko protected: 7332837b766SJim Ingham OptionGroupOptions m_option_group; 734715c2365SGreg Clayton OptionGroupVariable m_option_variable; 7351deb7962SGreg Clayton OptionGroupFormat m_option_format; 7362837b766SJim Ingham OptionGroupValueObjectDisplay m_varobj_options; 7376d56d2ceSJim Ingham }; 7386d56d2ceSJim Ingham 73930fdc8d8SChris Lattner #pragma mark CommandObjectMultiwordFrame 74030fdc8d8SChris Lattner 74130fdc8d8SChris Lattner //------------------------------------------------------------------------- 74230fdc8d8SChris Lattner // CommandObjectMultiwordFrame 74330fdc8d8SChris Lattner //------------------------------------------------------------------------- 74430fdc8d8SChris Lattner 745b9c1b51eSKate Stone CommandObjectMultiwordFrame::CommandObjectMultiwordFrame( 746b9c1b51eSKate Stone CommandInterpreter &interpreter) 747b9c1b51eSKate Stone : CommandObjectMultiword(interpreter, "frame", "Commands for selecting and " 748b9c1b51eSKate Stone "examing the current " 749b9c1b51eSKate Stone "thread's stack frames.", 750b9c1b51eSKate Stone "frame <subcommand> [<subcommand-options>]") { 751b9c1b51eSKate Stone LoadSubCommand("diagnose", 752b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameDiagnose(interpreter))); 753b9c1b51eSKate Stone LoadSubCommand("info", 754b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameInfo(interpreter))); 755b9c1b51eSKate Stone LoadSubCommand("select", 756b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameSelect(interpreter))); 757b9c1b51eSKate Stone LoadSubCommand("variable", 758b9c1b51eSKate Stone CommandObjectSP(new CommandObjectFrameVariable(interpreter))); 75930fdc8d8SChris Lattner } 76030fdc8d8SChris Lattner 761c8ecc2a9SEugene Zelenko CommandObjectMultiwordFrame::~CommandObjectMultiwordFrame() = default; 762