due to RTTI worries since llvm and clang don't use RTTI, but I was able to switch back with no issues as far as I can tell. Once the RTTI issue wasn't an issue, we were looking for a way to properly track weak pointers to objects to solve some of the threading issues we have been running into which naturally led us back to std::tr1::weak_ptr. We also wanted the ability to make a shared pointer from just a pointer, which is also easily solved using the std::tr1::enable_shared_from_this class. The main reason for this move back is so we can start properly having weak references to objects. Currently a lldb_private::Thread class has a refrence to its parent lldb_private::Process. This doesn't work well when we now hand out a SBThread object that contains a shared pointer to a lldb_private::Thread as this SBThread can be held onto by external clients and if they end up using one of these objects we can easily crash. So the next task is to start adopting std::tr1::weak_ptr where ever it makes sense which we can do with lldb_private::Debugger, lldb_private::Target, lldb_private::Process, lldb_private::Thread, lldb_private::StackFrame, and many more objects now that they are no longer using intrusive ref counted pointer objects (you can't do std::tr1::weak_ptr functionality with intrusive pointers). llvm-svn: 149207
275 lines
8.0 KiB
C++
275 lines
8.0 KiB
C++
//===-- CommandObjectArgs.cpp -----------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "CommandObjectArgs.h"
|
|
|
|
// C Includes
|
|
// C++ Includes
|
|
// Other libraries and framework includes
|
|
// Project includes
|
|
#include "lldb/Interpreter/Args.h"
|
|
#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/Core/Debugger.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.
|
|
//
|
|
|
|
CommandObjectArgs::CommandOptions::CommandOptions (CommandInterpreter &interpreter) :
|
|
Options(interpreter)
|
|
{
|
|
// Keep only one place to reset the values to their defaults
|
|
OptionParsingStarting();
|
|
}
|
|
|
|
|
|
CommandObjectArgs::CommandOptions::~CommandOptions ()
|
|
{
|
|
}
|
|
|
|
Error
|
|
CommandObjectArgs::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
|
|
{
|
|
Error error;
|
|
|
|
char short_option = (char) m_getopt_table[option_idx].val;
|
|
|
|
switch (short_option)
|
|
{
|
|
default:
|
|
error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
|
|
break;
|
|
}
|
|
|
|
return error;
|
|
}
|
|
|
|
void
|
|
CommandObjectArgs::CommandOptions::OptionParsingStarting ()
|
|
{
|
|
}
|
|
|
|
const OptionDefinition*
|
|
CommandObjectArgs::CommandOptions::GetDefinitions ()
|
|
{
|
|
return g_option_table;
|
|
}
|
|
|
|
CommandObjectArgs::CommandObjectArgs (CommandInterpreter &interpreter) :
|
|
CommandObject (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"),
|
|
m_options (interpreter)
|
|
{
|
|
}
|
|
|
|
CommandObjectArgs::~CommandObjectArgs ()
|
|
{
|
|
}
|
|
|
|
Options *
|
|
CommandObjectArgs::GetOptions ()
|
|
{
|
|
return &m_options;
|
|
}
|
|
|
|
bool
|
|
CommandObjectArgs::Execute
|
|
(
|
|
Args& args,
|
|
CommandReturnObject &result
|
|
)
|
|
{
|
|
ConstString target_triple;
|
|
|
|
|
|
Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
|
|
if (!process)
|
|
{
|
|
result.AppendError ("Args found no process.");
|
|
result.SetStatus (eReturnStatusFailed);
|
|
return false;
|
|
}
|
|
|
|
const ABI *abi = process->GetABI().get();
|
|
if (!abi)
|
|
{
|
|
result.AppendError ("The current process has no ABI.");
|
|
result.SetStatus (eReturnStatusFailed);
|
|
return false;
|
|
}
|
|
|
|
int num_args = args.GetArgumentCount ();
|
|
int arg_index;
|
|
|
|
if (!num_args)
|
|
{
|
|
result.AppendError ("args requires at least one argument");
|
|
result.SetStatus (eReturnStatusFailed);
|
|
return false;
|
|
}
|
|
|
|
Thread *thread = m_interpreter.GetExecutionContext ().GetThreadPtr();
|
|
|
|
if (!thread)
|
|
{
|
|
result.AppendError ("args found no thread.");
|
|
result.SetStatus (eReturnStatusFailed);
|
|
return false;
|
|
}
|
|
|
|
lldb::StackFrameSP thread_cur_frame = thread->GetSelectedFrame ();
|
|
if (!thread_cur_frame)
|
|
{
|
|
result.AppendError ("The current thread has no current frame.");
|
|
result.SetStatus (eReturnStatusFailed);
|
|
return false;
|
|
}
|
|
|
|
Module *thread_module = thread_cur_frame->GetFrameCodeAddress ().GetModulePtr ();
|
|
if (!thread_module)
|
|
{
|
|
result.AppendError ("The PC has no associated module.");
|
|
result.SetStatus (eReturnStatusFailed);
|
|
return false;
|
|
}
|
|
|
|
ClangASTContext &ast_context = thread_module->GetClangASTContext();
|
|
|
|
ValueList value_list;
|
|
|
|
for (arg_index = 0; arg_index < num_args; ++arg_index)
|
|
{
|
|
const char *arg_type_cstr = args.GetArgumentAtIndex(arg_index);
|
|
Value value;
|
|
value.SetValueType(Value::eValueTypeScalar);
|
|
void *type;
|
|
|
|
char *int_pos;
|
|
if ((int_pos = strstr (const_cast<char*>(arg_type_cstr), "int")))
|
|
{
|
|
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*"))
|
|
type = ast_context.CreatePointerType (ast_context.GetBuiltInType_void ());
|
|
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;
|
|
}
|
|
|
|
value.SetContext (Value::eContextTypeClangType, type);
|
|
|
|
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)
|
|
{
|
|
result.GetOutputStream ().Printf ("%d (%s): ", arg_index, args.GetArgumentAtIndex (arg_index));
|
|
value_list.GetValueAtIndex (arg_index)->Dump (&result.GetOutputStream ());
|
|
result.GetOutputStream ().Printf("\n");
|
|
}
|
|
|
|
return result.Succeeded();
|
|
}
|
|
|
|
OptionDefinition
|
|
CommandObjectArgs::CommandOptions::g_option_table[] =
|
|
{
|
|
{ LLDB_OPT_SET_1, false, "debug", 'g', no_argument, NULL, 0, eArgTypeNone, "Enable verbose debug logging of the expression parsing and evaluation."},
|
|
{ 0, false, NULL, 0, 0, NULL, NULL, eArgTypeNone, NULL }
|
|
};
|
|
|