13#include "lldb/Host/Config.h"
41#pragma mark CommandObjectFrameDiagnose
47#define LLDB_OPTIONS_frame_diag
48#include "CommandOptions.inc"
62 switch (short_option) {
69 if (option_arg.getAsInteger(0, *
address)) {
71 error.SetErrorStringWithFormat(
"invalid address argument '%s'",
72 option_arg.str().c_str());
78 if (option_arg.getAsInteger(0, *
offset)) {
80 error.SetErrorStringWithFormat(
"invalid offset argument '%s'",
81 option_arg.str().c_str());
86 llvm_unreachable(
"Unimplemented option");
99 return llvm::ArrayRef(g_frame_diag_options);
104 std::optional<ConstString>
reg;
110 "Try to determine what path the current stop "
111 "location used to get to a register or address",
113 eCommandRequiresThread | eCommandTryTargetAPILock |
114 eCommandProcessMustBeLaunched |
115 eCommandProcessMustBePaused) {
133 "`frame diagnose --address` is incompatible with other arguments.");
138 valobj_sp = frame_sp->GuessValueForRegisterAndOffset(
143 result.
AppendError(
"No arguments provided, and no stop info.");
160 GetExpressionPathFormat::eGetExpressionPathFormatHonorPointers;
161 valobj_sp->GetExpressionPath(stream, format);
162 stream.PutCString(
" =");
170 assert(valobj_sp.get() &&
"Must have a valid ValueObject to print");
180#pragma mark CommandObjectFrameInfo
188 "List information about the current "
189 "stack frame in the current thread.",
191 eCommandRequiresFrame | eCommandTryTargetAPILock |
192 eCommandProcessMustBeLaunched |
193 eCommandProcessMustBePaused) {}
204#pragma mark CommandObjectFrameSelect
208#define LLDB_OPTIONS_frame_select
209#include "CommandOptions.inc"
223 switch (short_option) {
226 if (option_arg.getAsInteger(0, offset) || offset == INT32_MIN) {
227 error.SetErrorStringWithFormat(
"invalid frame offset argument '%s'",
228 option_arg.str().c_str());
235 llvm_unreachable(
"Unimplemented option");
246 return llvm::ArrayRef(g_frame_select_options);
254 "Select the current stack frame by "
255 "index from within the current thread "
256 "(see 'thread backtrace'.)",
258 eCommandRequiresThread | eCommandTryTargetAPILock |
259 eCommandProcessMustBeLaunched |
260 eCommandProcessMustBePaused) {
284 uint32_t candidate_idx = frame_idx;
285 const unsigned max_depth = 12;
286 for (
unsigned num_try = 0; num_try < max_depth; ++num_try) {
293 if (candidate_sp->IsHidden())
306 if (
static_cast<int32_t
>(frame_idx) >=
310 if (frame_idx == 0) {
313 result.
AppendError(
"Already at the bottom of the stack.");
323 const uint32_t frame_requested = frame_idx
327 frame_idx = frame_requested;
331 if (
static_cast<int32_t
>(num_frames - frame_idx) >
335 if (frame_idx == num_frames - 1) {
338 result.
AppendError(
"Already at the top of the stack.");
341 frame_idx = num_frames - 1;
348 "too many arguments; expected frame-index, saw '%s'.\n",
357 if (command[0].ref().getAsInteger(0, frame_idx)) {
384#pragma mark CommandObjectFrameVariable
390 interpreter,
"frame variable",
391 "Show variables for the current stack frame. Defaults to all "
392 "arguments and local variables in scope. Names of argument, "
393 "local, file static and file global variables can be specified.",
395 eCommandRequiresFrame | eCommandTryTargetAPILock |
396 eCommandProcessMustBeLaunched | eCommandProcessMustBePaused |
397 eCommandRequiresProcess),
402Children of aggregate variables can be specified such as 'var->child.x'. In
403'frame variable', the operators -> and [] do not invoke operator overloads if
404they exist, but directly access the specified element. If you want to trigger
405operator overloads use the expression command to print the variable instead.
407It is worth noting that except for overloaded operators, when printing local
408variables 'expr local_var' and 'frame var local_var' produce the same results.
409However, 'frame variable' is more efficient, since it uses debug information and
410memory reads directly, rather than parsing and evaluating an expression, which
411may even involve JITing and running code in the target program.)");
431 return llvm::StringRef();
433 switch (var_sp->GetScope()) {
448 return llvm::StringRef();
470 llvm_unreachable(
"Unexpected scope value");
479 std::optional<llvm::ArrayRef<VariableSP>>
483 bool any_matches =
false;
484 const size_t previous_num_vars = matches.
GetSize();
494 return matches.
toArrayRef().drop_front(previous_num_vars);
517 if (
error.Fail() && (!variable_list || variable_list->
GetSize() == 0)) {
529 summary_format_sp = std::make_shared<StringSummaryFormat>(
546 if (!command.
empty()) {
551 for (
auto &entry : command) {
553 llvm::StringRef name_str = entry.ref();
556 std::optional<llvm::ArrayRef<VariableSP>> results =
560 "no variables matched the regular expression '%s'.",
568 std::string scope_string;
572 if (!scope_string.empty())
576 var_sp->GetDeclaration().GetFile()) {
577 bool show_fullpaths =
false;
578 bool show_module =
true;
579 if (var_sp->DumpDeclaration(&s, show_fullpaths,
584 if (llvm::Error
error = valobj_sp->Dump(strm, options))
589 if (llvm::Error err = regex.
GetError())
590 result.
AppendError(llvm::toString(std::move(err)));
593 "unknown regex error when compiling '%s'", entry.c_str());
599 uint32_t expr_path_options =
608 std::string scope_string;
612 if (!scope_string.empty())
615 var_sp->GetDeclaration().GetFile()) {
616 var_sp->GetDeclaration().DumpStopContext(&s,
false);
622 valobj_sp->GetPreferredDisplayLanguage());
626 valobj_sp->GetParent() ? entry.c_str() :
nullptr);
627 if (llvm::Error
error = valobj_sp->Dump(output_stream, options))
630 if (
auto error_cstr =
error.AsCString(
nullptr))
634 "unable to find any variable expression path that matches "
642 const size_t num_variables = variable_list->
GetSize();
643 if (num_variables > 0) {
644 for (
size_t i = 0; i < num_variables; i++) {
648 std::string scope_string;
659 if (valobj_sp->IsInScope()) {
660 if (!valobj_sp->GetTargetSP()
661 ->GetDisplayRuntimeSupportValues() &&
662 valobj_sp->IsRuntimeSupportValue())
665 if (!scope_string.empty())
669 var_sp->GetDeclaration().GetFile()) {
670 var_sp->GetDeclaration().DumpStopContext(&s,
false);
676 valobj_sp->GetPreferredDisplayLanguage());
678 var_sp ? var_sp->GetName().AsCString() :
nullptr);
679 if (llvm::Error
error =
693 if (recognized_frame) {
695 recognized_frame->GetRecognizedArguments();
696 if (recognized_arg_list) {
697 for (
auto &rec_value_sp : recognized_arg_list->GetObjects()) {
700 rec_value_sp->GetPreferredDisplayLanguage());
702 if (llvm::Error
error =
727#pragma mark CommandObjectFrameRecognizer
729#define LLDB_OPTIONS_frame_recognizer_add
730#include "CommandOptions.inc"
744 switch (short_option) {
751 error.SetErrorStringWithFormat(
752 "invalid boolean value '%s' passed for -f option",
753 option_arg.str().c_str());
763 m_symbols.push_back(std::string(option_arg));
769 llvm_unreachable(
"Unimplemented option");
784 return llvm::ArrayRef(g_frame_recognizer_add_options);
805 "Add a new frame recognizer.", nullptr) {
807Frame recognizers allow for retrieving information about special frames based on
808ABI, arguments or other special properties of that frame, even without source
809code or debug info. Currently, one use case is to extract function arguments
810that would otherwise be unaccesible, or augment existing arguments.
812Adding a custom frame recognizer is possible by implementing a Python class
813and using the 'frame recognizer add' command. The Python class should have a
814'get_recognized_arguments' method and it will receive an argument of type
815lldb.SBFrame representing the current frame that we are trying to recognize.
816The method should return a (possibly empty) list of lldb.SBValue objects that
817represent the recognized arguments.
819An example of a recognizer that retrieves the file descriptor values from libc
820functions 'read', 'write' and 'close' follows:
822 class LibcFdRecognizer(object):
823 def get_recognized_arguments(self, frame):
824 if frame.name in ["read", "write", "close"]:
825 fd = frame.EvaluateExpression("$arg1").unsigned
826 target = frame.thread.process.target
827 value = target.CreateValueFromExpression("fd", "(int)%d" % fd)
831The file containing this implementation can be imported via 'command script
832import' and then we can register this recognizer with 'frame recognizer add'.
833It's important to restrict the recognizer to the libc library (which is
834libsystem_kernel.dylib on macOS) to avoid matching functions with the same name
837(lldb) command script import .../fd_recognizer.py
838(lldb) frame recognizer add -l fd_recognizer.LibcFdRecognizer -n read -s libsystem_kernel.dylib
840When the program is stopped at the beginning of the 'read' function in libc, we
841can view the recognizer arguments in 'frame variable':
846* thread #1, queue = 'com.apple.main-thread', stop reason = breakpoint 1.3
847 frame #0: 0x00007fff06013ca0 libsystem_kernel.dylib`read
858#if LLDB_ENABLE_PYTHON
861 "%s needs a Python class name (-l argument).\n",
m_cmd_name.c_str());
873 "%s needs at least one symbol name (-n argument).\n",
880 "%s needs only one symbol regular expression (-n argument).\n",
889 result.
AppendWarning(
"The provided class does not exist - please define it "
890 "before attempting to use this frame recognizer");
919 "Delete all frame recognizers.", nullptr) {}
934 "Delete an existing frame recognizer by id.",
948 [&request](uint32_t rid, std::string rname, std::string module,
949 llvm::ArrayRef<lldb_private::ConstString> symbols,
953 rname =
"(internal)";
957 strm <<
", module " << module;
958 if (!symbols.empty())
959 for (
auto &symbol : symbols)
960 strm <<
", symbol " << symbol;
972 "About to delete all frame recognizers, do you want to do that?",
989 uint32_t recognizer_id;
996 if (!
GetTarget().GetFrameRecognizerManager().RemoveRecognizerWithID(
1010 "Show a list of active frame recognizers.",
1017 bool any_printed =
false;
1019 [&result, &any_printed](
1020 uint32_t recognizer_id, std::string name, std::string module,
1021 llvm::ArrayRef<ConstString> symbols,
bool regexp) {
1025 name =
"(internal)";
1027 stream << std::to_string(recognizer_id) <<
": " << name;
1028 if (!module.empty())
1029 stream <<
", module " << module;
1030 if (!symbols.empty())
1031 for (
auto &symbol : symbols)
1032 stream <<
", symbol " << symbol;
1034 stream <<
" (regexp)";
1055 interpreter,
"frame recognizer info",
1056 "Show which frame recognizer is applied a stack frame (if any).",
1066 uint32_t frame_index;
1067 if (!llvm::to_integer(frame_index_str, frame_index)) {
1074 if (process ==
nullptr) {
1079 if (thread ==
nullptr) {
1085 "'%s' takes exactly one frame index argument.\n",
m_cmd_name.c_str());
1099 output_stream.
Printf(
"frame %d ", frame_index);
1101 output_stream <<
"is recognized by ";
1102 output_stream << recognizer->GetName();
1104 output_stream <<
"not recognized by any recognizer";
1106 output_stream.
EOL();
1115 interpreter,
"frame recognizer",
1116 "Commands for editing and viewing frame recognizers.",
1117 "frame recognizer [<sub-command-options>] ") {
1135#pragma mark CommandObjectMultiwordFrame
1142 "Commands for selecting and "
1143 "examing the current "
1144 "thread's stack frames.",
1145 "frame <subcommand> [<subcommand-options>]") {
1154#if LLDB_ENABLE_PYTHON
static llvm::raw_ostream & error(Stream &strm)
CommandObjectFrameInfo(CommandInterpreter &interpreter)
void DoExecute(Args &command, CommandReturnObject &result) override
~CommandObjectFrameInfo() override=default
void OptionParsingStarting(ExecutionContext *execution_context) override
std::vector< std::string > m_symbols
bool m_first_instruction_only
Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg, ExecutionContext *execution_context) override
Set the value of an option.
llvm::ArrayRef< OptionDefinition > GetDefinitions() override
~CommandOptions() override=default
CommandObjectFrameRecognizerAdd(CommandInterpreter &interpreter)
Options * GetOptions() override
void DoExecute(Args &command, CommandReturnObject &result) override
~CommandObjectFrameRecognizerAdd() override=default
CommandObjectFrameRecognizerClear(CommandInterpreter &interpreter)
~CommandObjectFrameRecognizerClear() override=default
void DoExecute(Args &command, CommandReturnObject &result) override
~CommandObjectFrameRecognizerDelete() override=default
void DoExecute(Args &command, CommandReturnObject &result) override
void HandleArgumentCompletion(CompletionRequest &request, OptionElementVector &opt_element_vector) override
The default version handles argument definitions that have only one argument type,...
CommandObjectFrameRecognizerDelete(CommandInterpreter &interpreter)
CommandObjectFrameRecognizerInfo(CommandInterpreter &interpreter)
~CommandObjectFrameRecognizerInfo() override=default
void DoExecute(Args &command, CommandReturnObject &result) override
CommandObjectFrameRecognizerList(CommandInterpreter &interpreter)
~CommandObjectFrameRecognizerList() override=default
void DoExecute(Args &command, CommandReturnObject &result) override
~CommandObjectFrameRecognizer() override=default
CommandObjectFrameRecognizer(CommandInterpreter &interpreter)
Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg, ExecutionContext *execution_context) override
Set the value of an option.
std::optional< int32_t > relative_frame_offset
void OptionParsingStarting(ExecutionContext *execution_context) override
llvm::ArrayRef< OptionDefinition > GetDefinitions() override
~CommandOptions() override=default
~CommandObjectFrameSelect() override=default
CommandObjectFrameSelect(CommandInterpreter &interpreter)
Options * GetOptions() override
void DoExecute(Args &command, CommandReturnObject &result) override
OptionGroupOptions m_option_group
OptionGroupFormat m_option_format
Options * GetOptions() override
OptionGroupVariable m_option_variable
OptionGroupValueObjectDisplay m_varobj_options
bool ScopeRequested(lldb::ValueType scope)
Returns true if scope matches any of the options in m_option_variable.
llvm::StringRef GetScopeString(VariableSP var_sp)
std::optional< llvm::ArrayRef< VariableSP > > findUniqueRegexMatches(RegularExpression ®ex, VariableList &matches, const VariableList &all_variables)
Finds all the variables in all_variables whose name matches regex, inserting them into matches.
~CommandObjectFrameVariable() override=default
CommandObjectFrameVariable(CommandInterpreter &interpreter)
void DoExecute(Args &command, CommandReturnObject &result) override
A command line argument class.
size_t GetArgumentCount() const
Gets the number of arguments left in this command object.
const char * GetArgumentAtIndex(size_t idx) const
Gets the NULL terminated C string argument pointer for the argument at index idx.
bool Confirm(llvm::StringRef message, bool default_answer)
void PrintWarningsIfNecessary(Stream &s, const std::string &cmd_name)
CommandObjectMultiwordFrame(CommandInterpreter &interpreter)
~CommandObjectMultiwordFrame() override
bool LoadSubCommand(llvm::StringRef cmd_name, const lldb::CommandObjectSP &command_obj) override
virtual void SetHelpLong(llvm::StringRef str)
void AddSimpleArgumentList(lldb::CommandArgumentType arg_type, ArgumentRepetitionType repetition_type=eArgRepeatPlain)
ExecutionContext m_exe_ctx
CommandInterpreter & GetCommandInterpreter()
CommandInterpreter & m_interpreter
void AppendMessage(llvm::StringRef in_string)
void void AppendError(llvm::StringRef in_string)
Stream & GetErrorStream()
void SetStatus(lldb::ReturnStatus status)
void AppendErrorWithFormat(const char *format,...) __attribute__((format(printf
lldb::ReturnStatus GetStatus() const
void void AppendWarning(llvm::StringRef in_string)
Stream & GetOutputStream()
"lldb/Utility/ArgCompletionRequest.h"
void TryCompleteCurrentArg(llvm::StringRef completion, llvm::StringRef description="")
Adds a possible completion string if the completion would complete the current argument.
size_t GetCursorIndex() const
A uniqued constant string class.
ScriptInterpreter * GetScriptInterpreter(bool can_create=true, std::optional< lldb::ScriptLanguage > language={})
DumpValueObjectOptions & SetVariableFormatDisplayLanguage(lldb::LanguageType lang=lldb::eLanguageTypeUnknown)
std::function< bool(ConstString, ConstString, const DumpValueObjectOptions &, Stream &)> DeclPrintingHelper
DumpValueObjectOptions & SetDeclPrintingHelper(DeclPrintingHelper helper)
DumpValueObjectOptions & SetRootValueObjectName(const char *name=nullptr)
DumpValueObjectOptions & SetFormat(lldb::Format format=lldb::eFormatDefault)
"lldb/Target/ExecutionContext.h" A class that contains an execution context.
void SetFrameSP(const lldb::StackFrameSP &frame_sp)
Set accessor to set only the frame shared pointer.
StackFrame * GetFramePtr() const
Returns a pointer to the frame object.
StackFrame & GetFrameRef() const
Returns a reference to the thread object.
Process * GetProcessPtr() const
Returns a pointer to the process object.
Thread * GetThreadPtr() const
Returns a pointer to the thread object.
bool IsTopLevelFunction()
Get whether this function represents a 'top-level' function.
void Append(OptionGroup *group)
Append options from a OptionGroup class.
DumpValueObjectOptions GetAsDumpOptions(LanguageRuntimeDescriptionDisplayVerbosity lang_descr_verbosity=eLanguageRuntimeDescriptionDisplayVerbosityFull, lldb::Format format=lldb::eFormatDefault, lldb::TypeSummaryImplSP summary_sp=lldb::TypeSummaryImplSP())
lldb::DynamicValueType use_dynamic
OptionValueString summary_string
OptionValueString summary
bool show_recognized_args
bool IsCurrentValueEmpty() const
const char * GetCurrentValue() const
A command line option parsing protocol class.
void GenerateOptionUsage(Stream &strm, CommandObject &cmd, uint32_t screen_width)
std::vector< Option > m_getopt_table
A plug-in interface definition class for debugging a process.
bool IsValid() const
Test if this object contains a valid regular expression.
llvm::Error GetError() const
Return an error if the regular expression failed to compile.
virtual bool CheckObjectExists(const char *name)
Python implementation for frame recognizers.
lldb::StackFrameRecognizerSP GetRecognizerForFrame(lldb::StackFrameSP frame)
void RemoveAllRecognizers()
void ForEach(std::function< void(uint32_t recognizer_id, std::string recognizer_name, std::string module, llvm::ArrayRef< ConstString > symbols, bool regexp)> const &callback)
void AddRecognizer(lldb::StackFrameRecognizerSP recognizer, ConstString module, llvm::ArrayRef< ConstString > symbols, bool first_instruction_only=true)
This base class provides an interface to stack frames.
@ eExpressionPathOptionCheckPtrVsMember
@ eExpressionPathOptionsInspectAnonymousUnions
@ eExpressionPathOptionsAllowDirectIVarAccess
VariableList * GetVariableList(bool get_file_globals, Status *error_ptr)
Retrieve the list of variables that are in scope at this StackFrame's pc.
void DumpUsingSettingsFormat(Stream *strm, bool show_unique=false, const char *frame_marker=nullptr)
Print a description for this frame using the frame-format formatter settings.
lldb::ValueObjectSP GetValueForVariableExpressionPath(llvm::StringRef var_expr, lldb::DynamicValueType use_dynamic, uint32_t options, lldb::VariableSP &var_sp, Status &error)
Create a ValueObject for a variable name / pathname, possibly including simple dereference/child sele...
lldb::ValueObjectSP GetValueObjectForFrameVariable(const lldb::VariableSP &variable_sp, lldb::DynamicValueType use_dynamic)
Create a ValueObject for a given Variable in this StackFrame.
const SymbolContext & GetSymbolContext(lldb::SymbolContextItem resolve_scope)
Provide a SymbolContext for this StackFrame's current pc value.
lldb::RecognizedStackFrameSP GetRecognizedFrame()
static lldb::ValueObjectSP GetCrashingDereference(lldb::StopInfoSP &stop_info_sp, lldb::addr_t *crashing_address=nullptr)
llvm::StringRef GetString() const
A stream class that can stream formatted output to a file.
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
size_t PutCString(llvm::StringRef cstr)
Output a C string to the stream.
virtual void Flush()=0
Flush the stream.
size_t EOL()
Output and End of Line character to the stream.
Defines a symbol context baton that can be handed other debug core functions.
Function * function
The Function for a given query.
A class that represents statistics for a since lldb_private::Target.
StatsSuccessFail & GetFrameVariableStats()
TargetStats & GetStatistics()
StackFrameRecognizerManager & GetFrameRecognizerManager()
virtual lldb::StackFrameSP GetStackFrameAtIndex(uint32_t idx)
lldb::StackFrameSP GetSelectedFrame(SelectMostRelevant select_most_relevant)
uint32_t GetSelectedFrameIndex(SelectMostRelevant select_most_relevant)
lldb::StopInfoSP GetStopInfo()
bool SetSelectedFrameByIndexNoisily(uint32_t frame_idx, Stream &output_stream)
virtual uint32_t GetStackFrameCount()
GetStackFrameCount can be expensive.
llvm::Error PrintValueObject()
bool AddVariableIfUnique(const lldb::VariableSP &var_sp)
lldb::VariableSP GetVariableAtIndex(size_t idx) const
llvm::ArrayRef< lldb::VariableSP > toArrayRef()
@ SelectMostRelevantFrame
A class that represents a running process on the host machine.
std::vector< OptionArgElement > OptionElementVector
const char * toString(AppleArm64ExceptionClass EC)
@ eLanguageRuntimeDescriptionDisplayVerbosityFull
std::shared_ptr< lldb_private::StackFrame > StackFrameSP
std::shared_ptr< lldb_private::TypeSummaryImpl > TypeSummaryImplSP
std::shared_ptr< lldb_private::CommandObject > CommandObjectSP
std::shared_ptr< lldb_private::ValueObject > ValueObjectSP
Format
Display format definitions.
std::shared_ptr< lldb_private::RegularExpression > RegularExpressionSP
std::shared_ptr< lldb_private::ValueObjectList > ValueObjectListSP
@ eReturnStatusSuccessFinishResult
@ eReturnStatusSuccessFinishNoResult
std::shared_ptr< lldb_private::Variable > VariableSP
std::shared_ptr< lldb_private::StackFrameRecognizer > StackFrameRecognizerSP
std::shared_ptr< lldb_private::StopInfo > StopInfoSP
@ eValueTypeVTableEntry
function pointer in virtual function table
@ eValueTypeVTable
virtual function table
@ eValueTypeVariableGlobal
globals variable
@ eValueTypeConstResult
constant result variables
@ eValueTypeVariableLocal
function local variables
@ eValueTypeVariableArgument
function argument variables
@ eValueTypeRegister
stack frame register value
@ eValueTypeVariableStatic
static variable
@ eValueTypeRegisterSet
A collection of stack frame register values.
@ eValueTypeVariableThreadLocal
thread local storage variable
static bool ToBoolean(llvm::StringRef s, bool fail_value, bool *success_ptr)