2010-06-09 00:52:24 +08:00
|
|
|
//===-- CommandObjectArgs.cpp -----------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-12-05 08:20:57 +08:00
|
|
|
#include "lldb/lldb-python.h"
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "CommandObjectArgs.h"
|
|
|
|
|
|
|
|
// C Includes
|
|
|
|
// C++ Includes
|
|
|
|
// Other libraries and framework includes
|
|
|
|
// Project includes
|
2010-06-16 03:49:27 +08:00
|
|
|
#include "lldb/Interpreter/Args.h"
|
<rdar://problem/11757916>
Make breakpoint setting by file and line much more efficient by only looking for inlined breakpoint locations if we are setting a breakpoint in anything but a source implementation file. Implementing this complex for a many reasons. Turns out that parsing compile units lazily had some issues with respect to how we need to do things with DWARF in .o files. So the fixes in the checkin for this makes these changes:
- Add a new setting called "target.inline-breakpoint-strategy" which can be set to "never", "always", or "headers". "never" will never try and set any inlined breakpoints (fastest). "always" always looks for inlined breakpoint locations (slowest, but most accurate). "headers", which is the default setting, will only look for inlined breakpoint locations if the breakpoint is set in what are consudered to be header files, which is realy defined as "not in an implementation source file".
- modify the breakpoint setting by file and line to check the current "target.inline-breakpoint-strategy" setting and act accordingly
- Modify compile units to be able to get their language and other info lazily. This allows us to create compile units from the debug map and not have to fill all of the details in, and then lazily discover this information as we go on debuggging. This is needed to avoid parsing all .o files when setting breakpoints in implementation only files (no inlines). Otherwise we would need to parse the .o file, the object file (mach-o in our case) and the symbol file (DWARF in the object file) just to see what the compile unit was.
- modify the "SymbolFileDWARFDebugMap" to subclass lldb_private::Module so that the virtual "GetObjectFile()" and "GetSymbolVendor()" functions can be intercepted when the .o file contenst are later lazilly needed. Prior to this fix, when we first instantiated the "SymbolFileDWARFDebugMap" class, we would also make modules, object files and symbol files for every .o file in the debug map because we needed to fix up the sections in the .o files with information that is in the executable debug map. Now we lazily do this in the DebugMapModule::GetObjectFile()
Cleaned up header includes a bit as well.
llvm-svn: 162860
2012-08-30 05:13:06 +08:00
|
|
|
#include "lldb/Core/Debugger.h"
|
|
|
|
#include "lldb/Core/Module.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Core/Value.h"
|
|
|
|
#include "lldb/Expression/ClangExpression.h"
|
|
|
|
#include "lldb/Expression/ClangExpressionVariable.h"
|
|
|
|
#include "lldb/Expression/ClangFunction.h"
|
|
|
|
#include "lldb/Host/Host.h"
|
|
|
|
#include "lldb/Interpreter/CommandInterpreter.h"
|
|
|
|
#include "lldb/Interpreter/CommandReturnObject.h"
|
|
|
|
#include "lldb/Symbol/ObjectFile.h"
|
|
|
|
#include "lldb/Symbol/Variable.h"
|
|
|
|
#include "lldb/Target/Process.h"
|
|
|
|
#include "lldb/Target/Target.h"
|
|
|
|
#include "lldb/Target/Thread.h"
|
|
|
|
#include "lldb/Target/StackFrame.h"
|
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
// This command is a toy. I'm just using it to have a way to construct the arguments to
|
|
|
|
// calling functions.
|
|
|
|
//
|
|
|
|
|
2011-04-08 06:46:35 +08:00
|
|
|
CommandObjectArgs::CommandOptions::CommandOptions (CommandInterpreter &interpreter) :
|
2011-04-09 06:39:17 +08:00
|
|
|
Options(interpreter)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
// Keep only one place to reset the values to their defaults
|
2011-04-13 08:18:08 +08:00
|
|
|
OptionParsingStarting();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
CommandObjectArgs::CommandOptions::~CommandOptions ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Error
|
2011-04-13 08:18:08 +08:00
|
|
|
CommandObjectArgs::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
Error error;
|
|
|
|
|
2012-12-04 08:32:51 +08:00
|
|
|
const int short_option = m_getopt_table[option_idx].val;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
switch (short_option)
|
|
|
|
{
|
|
|
|
default:
|
2011-10-26 08:56:27 +08:00
|
|
|
error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
|
2010-06-09 00:52:24 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-04-13 08:18:08 +08:00
|
|
|
CommandObjectArgs::CommandOptions::OptionParsingStarting ()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-03-25 05:19:54 +08:00
|
|
|
const OptionDefinition*
|
2010-06-09 00:52:24 +08:00
|
|
|
CommandObjectArgs::CommandOptions::GetDefinitions ()
|
|
|
|
{
|
|
|
|
return g_option_table;
|
|
|
|
}
|
|
|
|
|
2010-09-18 09:14:36 +08:00
|
|
|
CommandObjectArgs::CommandObjectArgs (CommandInterpreter &interpreter) :
|
2012-06-09 05:56:10 +08:00
|
|
|
CommandObjectParsed (interpreter,
|
|
|
|
"args",
|
|
|
|
"When stopped at the start of a function, reads function arguments of type (u?)int(8|16|32|64)_t, (void|char)*",
|
|
|
|
"args"),
|
2011-04-08 06:46:35 +08:00
|
|
|
m_options (interpreter)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
CommandObjectArgs::~CommandObjectArgs ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Options *
|
|
|
|
CommandObjectArgs::GetOptions ()
|
|
|
|
{
|
|
|
|
return &m_options;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2012-06-09 05:56:10 +08:00
|
|
|
CommandObjectArgs::DoExecute (Args& args, CommandReturnObject &result)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
ConstString target_triple;
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
|
Expanded the flags that can be set for a command object in lldb_private::CommandObject. This list of available flags are:
enum
{
//----------------------------------------------------------------------
// eFlagRequiresTarget
//
// Ensures a valid target is contained in m_exe_ctx prior to executing
// the command. If a target doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidTargetDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidTargetDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresTarget = (1u << 0),
//----------------------------------------------------------------------
// eFlagRequiresProcess
//
// Ensures a valid process is contained in m_exe_ctx prior to executing
// the command. If a process doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidProcessDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidProcessDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresProcess = (1u << 1),
//----------------------------------------------------------------------
// eFlagRequiresThread
//
// Ensures a valid thread is contained in m_exe_ctx prior to executing
// the command. If a thread doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidThreadDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidThreadDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresThread = (1u << 2),
//----------------------------------------------------------------------
// eFlagRequiresFrame
//
// Ensures a valid frame is contained in m_exe_ctx prior to executing
// the command. If a frame doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidFrameDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidFrameDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresFrame = (1u << 3),
//----------------------------------------------------------------------
// eFlagRequiresRegContext
//
// Ensures a valid register context (from the selected frame if there
// is a frame in m_exe_ctx, or from the selected thread from m_exe_ctx)
// is availble from m_exe_ctx prior to executing the command. If a
// target doesn't exist or is invalid, the command will fail and
// CommandObject::GetInvalidRegContextDescription() will be returned as
// the error. CommandObject subclasses can override the virtual function
// for GetInvalidRegContextDescription() to provide custom strings when
// needed.
//----------------------------------------------------------------------
eFlagRequiresRegContext = (1u << 4),
//----------------------------------------------------------------------
// eFlagTryTargetAPILock
//
// Attempts to acquire the target lock if a target is selected in the
// command interpreter. If the command object fails to acquire the API
// lock, the command will fail with an appropriate error message.
//----------------------------------------------------------------------
eFlagTryTargetAPILock = (1u << 5),
//----------------------------------------------------------------------
// eFlagProcessMustBeLaunched
//
// Verifies that there is a launched process in m_exe_ctx, if there
// isn't, the command will fail with an appropriate error message.
//----------------------------------------------------------------------
eFlagProcessMustBeLaunched = (1u << 6),
//----------------------------------------------------------------------
// eFlagProcessMustBePaused
//
// Verifies that there is a paused process in m_exe_ctx, if there
// isn't, the command will fail with an appropriate error message.
//----------------------------------------------------------------------
eFlagProcessMustBePaused = (1u << 7)
};
Now each command object contains a "ExecutionContext m_exe_ctx;" member variable that gets initialized prior to running the command. The validity of the target objects in m_exe_ctx are checked to ensure that any target/process/thread/frame/reg context that are required are valid prior to executing the command. Each command object also contains a Mutex::Locker m_api_locker which gets used if eFlagTryTargetAPILock is set. This centralizes a lot of checking code that was previously and inconsistently implemented across many commands.
llvm-svn: 171990
2013-01-10 03:44:40 +08:00
|
|
|
Process *process = m_exe_ctx.GetProcessPtr();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (!process)
|
|
|
|
{
|
|
|
|
result.AppendError ("Args found no process.");
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-05-12 02:39:18 +08:00
|
|
|
const ABI *abi = process->GetABI().get();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (!abi)
|
|
|
|
{
|
|
|
|
result.AppendError ("The current process has no ABI.");
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-01-26 02:06:21 +08:00
|
|
|
const size_t num_args = args.GetArgumentCount ();
|
2010-06-09 00:52:24 +08:00
|
|
|
int arg_index;
|
|
|
|
|
|
|
|
if (!num_args)
|
|
|
|
{
|
|
|
|
result.AppendError ("args requires at least one argument");
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
Expanded the flags that can be set for a command object in lldb_private::CommandObject. This list of available flags are:
enum
{
//----------------------------------------------------------------------
// eFlagRequiresTarget
//
// Ensures a valid target is contained in m_exe_ctx prior to executing
// the command. If a target doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidTargetDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidTargetDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresTarget = (1u << 0),
//----------------------------------------------------------------------
// eFlagRequiresProcess
//
// Ensures a valid process is contained in m_exe_ctx prior to executing
// the command. If a process doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidProcessDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidProcessDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresProcess = (1u << 1),
//----------------------------------------------------------------------
// eFlagRequiresThread
//
// Ensures a valid thread is contained in m_exe_ctx prior to executing
// the command. If a thread doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidThreadDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidThreadDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresThread = (1u << 2),
//----------------------------------------------------------------------
// eFlagRequiresFrame
//
// Ensures a valid frame is contained in m_exe_ctx prior to executing
// the command. If a frame doesn't exist or is invalid, the command
// will fail and CommandObject::GetInvalidFrameDescription() will be
// returned as the error. CommandObject subclasses can override the
// virtual function for GetInvalidFrameDescription() to provide custom
// strings when needed.
//----------------------------------------------------------------------
eFlagRequiresFrame = (1u << 3),
//----------------------------------------------------------------------
// eFlagRequiresRegContext
//
// Ensures a valid register context (from the selected frame if there
// is a frame in m_exe_ctx, or from the selected thread from m_exe_ctx)
// is availble from m_exe_ctx prior to executing the command. If a
// target doesn't exist or is invalid, the command will fail and
// CommandObject::GetInvalidRegContextDescription() will be returned as
// the error. CommandObject subclasses can override the virtual function
// for GetInvalidRegContextDescription() to provide custom strings when
// needed.
//----------------------------------------------------------------------
eFlagRequiresRegContext = (1u << 4),
//----------------------------------------------------------------------
// eFlagTryTargetAPILock
//
// Attempts to acquire the target lock if a target is selected in the
// command interpreter. If the command object fails to acquire the API
// lock, the command will fail with an appropriate error message.
//----------------------------------------------------------------------
eFlagTryTargetAPILock = (1u << 5),
//----------------------------------------------------------------------
// eFlagProcessMustBeLaunched
//
// Verifies that there is a launched process in m_exe_ctx, if there
// isn't, the command will fail with an appropriate error message.
//----------------------------------------------------------------------
eFlagProcessMustBeLaunched = (1u << 6),
//----------------------------------------------------------------------
// eFlagProcessMustBePaused
//
// Verifies that there is a paused process in m_exe_ctx, if there
// isn't, the command will fail with an appropriate error message.
//----------------------------------------------------------------------
eFlagProcessMustBePaused = (1u << 7)
};
Now each command object contains a "ExecutionContext m_exe_ctx;" member variable that gets initialized prior to running the command. The validity of the target objects in m_exe_ctx are checked to ensure that any target/process/thread/frame/reg context that are required are valid prior to executing the command. Each command object also contains a Mutex::Locker m_api_locker which gets used if eFlagTryTargetAPILock is set. This centralizes a lot of checking code that was previously and inconsistently implemented across many commands.
llvm-svn: 171990
2013-01-10 03:44:40 +08:00
|
|
|
Thread *thread = m_exe_ctx.GetThreadPtr();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
if (!thread)
|
|
|
|
{
|
|
|
|
result.AppendError ("args found no thread.");
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-08-27 05:32:51 +08:00
|
|
|
lldb::StackFrameSP thread_cur_frame = thread->GetSelectedFrame ();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (!thread_cur_frame)
|
|
|
|
{
|
|
|
|
result.AppendError ("The current thread has no current frame.");
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-02-24 09:59:29 +08:00
|
|
|
ModuleSP thread_module_sp (thread_cur_frame->GetFrameCodeAddress ().GetModule());
|
|
|
|
if (!thread_module_sp)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
result.AppendError ("The PC has no associated module.");
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-02-24 09:59:29 +08:00
|
|
|
ClangASTContext &ast_context = thread_module_sp->GetClangASTContext();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
ValueList value_list;
|
|
|
|
|
|
|
|
for (arg_index = 0; arg_index < num_args; ++arg_index)
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
const char *arg_type_cstr = args.GetArgumentAtIndex(arg_index);
|
2010-06-09 00:52:24 +08:00
|
|
|
Value value;
|
|
|
|
value.SetValueType(Value::eValueTypeScalar);
|
|
|
|
void *type;
|
|
|
|
|
|
|
|
char *int_pos;
|
2010-06-09 15:57:51 +08:00
|
|
|
if ((int_pos = strstr (const_cast<char*>(arg_type_cstr), "int")))
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
Encoding encoding = eEncodingSint;
|
|
|
|
|
|
|
|
int width = 0;
|
|
|
|
|
|
|
|
if (int_pos > arg_type_cstr + 1)
|
|
|
|
{
|
|
|
|
result.AppendErrorWithFormat ("Invalid format: %s.\n", arg_type_cstr);
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (int_pos == arg_type_cstr + 1 && arg_type_cstr[0] != 'u')
|
|
|
|
{
|
|
|
|
result.AppendErrorWithFormat ("Invalid format: %s.\n", arg_type_cstr);
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (arg_type_cstr[0] == 'u')
|
|
|
|
{
|
|
|
|
encoding = eEncodingUint;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *width_pos = int_pos + 3;
|
|
|
|
|
|
|
|
if (!strcmp (width_pos, "8_t"))
|
|
|
|
width = 8;
|
|
|
|
else if (!strcmp (width_pos, "16_t"))
|
|
|
|
width = 16;
|
|
|
|
else if (!strcmp (width_pos, "32_t"))
|
|
|
|
width = 32;
|
|
|
|
else if (!strcmp (width_pos, "64_t"))
|
|
|
|
width = 64;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
result.AppendErrorWithFormat ("Invalid format: %s.\n", arg_type_cstr);
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
type = ast_context.GetBuiltinTypeForEncodingAndBitSize(encoding, width);
|
|
|
|
|
|
|
|
if (!type)
|
|
|
|
{
|
|
|
|
result.AppendErrorWithFormat ("Couldn't get Clang type for format %s (%s integer, width %d).\n",
|
|
|
|
arg_type_cstr,
|
|
|
|
(encoding == eEncodingSint ? "signed" : "unsigned"),
|
|
|
|
width);
|
|
|
|
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (strchr (arg_type_cstr, '*'))
|
|
|
|
{
|
|
|
|
if (!strcmp (arg_type_cstr, "void*"))
|
2010-08-03 08:35:52 +08:00
|
|
|
type = ast_context.CreatePointerType (ast_context.GetBuiltInType_void ());
|
2010-06-09 00:52:24 +08:00
|
|
|
else if (!strcmp (arg_type_cstr, "char*"))
|
|
|
|
type = ast_context.GetCStringType (false);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
result.AppendErrorWithFormat ("Invalid format: %s.\n", arg_type_cstr);
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
result.AppendErrorWithFormat ("Invalid format: %s.\n", arg_type_cstr);
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-11-13 11:52:47 +08:00
|
|
|
value.SetContext (Value::eContextTypeClangType, type);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
value_list.PushValue(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!abi->GetArgumentValues (*thread, value_list))
|
|
|
|
{
|
|
|
|
result.AppendError ("Couldn't get argument values");
|
|
|
|
result.SetStatus (eReturnStatusFailed);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
result.GetOutputStream ().Printf("Arguments : \n");
|
|
|
|
|
|
|
|
for (arg_index = 0; arg_index < num_args; ++arg_index)
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
result.GetOutputStream ().Printf ("%d (%s): ", arg_index, args.GetArgumentAtIndex (arg_index));
|
2010-06-09 00:52:24 +08:00
|
|
|
value_list.GetValueAtIndex (arg_index)->Dump (&result.GetOutputStream ());
|
|
|
|
result.GetOutputStream ().Printf("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
return result.Succeeded();
|
|
|
|
}
|
|
|
|
|
2011-03-25 05:19:54 +08:00
|
|
|
OptionDefinition
|
2010-06-09 00:52:24 +08:00
|
|
|
CommandObjectArgs::CommandOptions::g_option_table[] =
|
|
|
|
{
|
2010-10-02 03:59:14 +08:00
|
|
|
{ LLDB_OPT_SET_1, false, "debug", 'g', no_argument, NULL, 0, eArgTypeNone, "Enable verbose debug logging of the expression parsing and evaluation."},
|
2012-09-12 00:09:27 +08:00
|
|
|
{ 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
|
2010-06-09 00:52:24 +08:00
|
|
|
};
|
|
|
|
|