2010-06-09 00:52:24 +08:00
|
|
|
//===-- Debugger.cpp --------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "lldb/lldb-private.h"
|
|
|
|
#include "lldb/Core/ConnectionFileDescriptor.h"
|
|
|
|
#include "lldb/Core/Debugger.h"
|
|
|
|
#include "lldb/Core/InputReader.h"
|
|
|
|
#include "lldb/Core/State.h"
|
|
|
|
#include "lldb/Core/Timer.h"
|
2010-06-23 09:19:29 +08:00
|
|
|
#include "lldb/Interpreter/CommandInterpreter.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Target/TargetList.h"
|
|
|
|
#include "lldb/Target/Process.h"
|
|
|
|
#include "lldb/Target/Thread.h"
|
|
|
|
|
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
static uint32_t g_shared_debugger_refcount = 0;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-07-01 00:22:25 +08:00
|
|
|
static lldb::user_id_t g_unique_id = 1;
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
void
|
|
|
|
Debugger::Initialize ()
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
if (g_shared_debugger_refcount == 0)
|
2010-06-09 00:52:24 +08:00
|
|
|
lldb_private::Initialize();
|
2010-06-23 09:19:29 +08:00
|
|
|
g_shared_debugger_refcount++;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::Terminate ()
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
if (g_shared_debugger_refcount > 0)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
g_shared_debugger_refcount--;
|
|
|
|
if (g_shared_debugger_refcount == 0)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
lldb_private::WillTerminate();
|
|
|
|
lldb_private::Terminate();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
typedef std::vector<DebuggerSP> DebuggerList;
|
|
|
|
|
|
|
|
static Mutex &
|
|
|
|
GetDebuggerListMutex ()
|
|
|
|
{
|
|
|
|
static Mutex g_mutex(Mutex::eMutexTypeRecursive);
|
|
|
|
return g_mutex;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DebuggerList &
|
|
|
|
GetDebuggerList()
|
|
|
|
{
|
|
|
|
// hide the static debugger list inside a singleton accessor to avoid
|
|
|
|
// global init contructors
|
|
|
|
static DebuggerList g_list;
|
|
|
|
return g_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DebuggerSP
|
|
|
|
Debugger::CreateInstance ()
|
|
|
|
{
|
|
|
|
DebuggerSP debugger_sp (new Debugger);
|
|
|
|
// Scope for locker
|
|
|
|
{
|
|
|
|
Mutex::Locker locker (GetDebuggerListMutex ());
|
|
|
|
GetDebuggerList().push_back(debugger_sp);
|
|
|
|
}
|
|
|
|
return debugger_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
lldb::DebuggerSP
|
|
|
|
Debugger::GetSP ()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
lldb::DebuggerSP debugger_sp;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
Mutex::Locker locker (GetDebuggerListMutex ());
|
|
|
|
DebuggerList &debugger_list = GetDebuggerList();
|
|
|
|
DebuggerList::iterator pos, end = debugger_list.end();
|
|
|
|
for (pos = debugger_list.begin(); pos != end; ++pos)
|
|
|
|
{
|
|
|
|
if ((*pos).get() == this)
|
|
|
|
{
|
|
|
|
debugger_sp = *pos;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return debugger_sp;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2010-09-04 08:03:46 +08:00
|
|
|
lldb::DebuggerSP
|
|
|
|
Debugger::FindDebuggerWithInstanceName (const ConstString &instance_name)
|
|
|
|
{
|
|
|
|
lldb::DebuggerSP debugger_sp;
|
|
|
|
|
|
|
|
Mutex::Locker locker (GetDebuggerListMutex ());
|
|
|
|
DebuggerList &debugger_list = GetDebuggerList();
|
|
|
|
DebuggerList::iterator pos, end = debugger_list.end();
|
|
|
|
|
|
|
|
for (pos = debugger_list.begin(); pos != end; ++pos)
|
|
|
|
{
|
|
|
|
if ((*pos).get()->m_instance_name == instance_name)
|
|
|
|
{
|
|
|
|
debugger_sp = *pos;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return debugger_sp;
|
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
|
|
|
|
TargetSP
|
|
|
|
Debugger::FindTargetWithProcessID (lldb::pid_t pid)
|
|
|
|
{
|
|
|
|
lldb::TargetSP target_sp;
|
|
|
|
Mutex::Locker locker (GetDebuggerListMutex ());
|
|
|
|
DebuggerList &debugger_list = GetDebuggerList();
|
|
|
|
DebuggerList::iterator pos, end = debugger_list.end();
|
|
|
|
for (pos = debugger_list.begin(); pos != end; ++pos)
|
|
|
|
{
|
|
|
|
target_sp = (*pos)->GetTargetList().FindTargetWithProcessID (pid);
|
|
|
|
if (target_sp)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return target_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
Debugger::Debugger () :
|
2010-07-01 00:22:25 +08:00
|
|
|
UserID (g_unique_id++),
|
2010-09-04 08:03:46 +08:00
|
|
|
DebuggerInstanceSettings (*(Debugger::GetSettingsController().get())),
|
2010-06-09 00:52:24 +08:00
|
|
|
m_input_comm("debugger.input"),
|
|
|
|
m_input_file (),
|
|
|
|
m_output_file (),
|
|
|
|
m_error_file (),
|
|
|
|
m_target_list (),
|
|
|
|
m_listener ("lldb.Debugger"),
|
|
|
|
m_source_manager (),
|
2010-06-23 09:19:29 +08:00
|
|
|
m_command_interpreter_ap (new CommandInterpreter (*this, eScriptLanguageDefault, false)),
|
|
|
|
m_exe_ctx (),
|
2010-06-09 00:52:24 +08:00
|
|
|
m_input_readers (),
|
2010-08-31 03:44:40 +08:00
|
|
|
m_input_reader_data (),
|
|
|
|
m_use_external_editor(false)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
m_command_interpreter_ap->Initialize ();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Debugger::~Debugger ()
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
int num_targets = m_target_list.GetNumTargets();
|
|
|
|
for (int i = 0; i < num_targets; i++)
|
|
|
|
{
|
|
|
|
ProcessSP process_sp (m_target_list.GetTargetAtIndex (i)->GetProcessSP());
|
|
|
|
if (process_sp)
|
|
|
|
process_sp->Destroy();
|
|
|
|
}
|
|
|
|
DisconnectInput();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
Debugger::GetAsyncExecution ()
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
return !m_command_interpreter_ap->GetSynchronous();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::SetAsyncExecution (bool async_execution)
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
m_command_interpreter_ap->SetSynchronous (!async_execution);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::DisconnectInput()
|
|
|
|
{
|
|
|
|
m_input_comm.Clear ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::SetInputFileHandle (FILE *fh, bool tranfer_ownership)
|
|
|
|
{
|
|
|
|
m_input_file.SetFileHandle (fh, tranfer_ownership);
|
|
|
|
if (m_input_file.GetFileHandle() == NULL)
|
|
|
|
m_input_file.SetFileHandle (stdin, false);
|
|
|
|
|
|
|
|
// Disconnect from any old connection if we had one
|
|
|
|
m_input_comm.Disconnect ();
|
|
|
|
m_input_comm.SetConnection (new ConnectionFileDescriptor (::fileno (GetInputFileHandle()), true));
|
|
|
|
m_input_comm.SetReadThreadBytesReceivedCallback (Debugger::DispatchInputCallback, this);
|
|
|
|
|
|
|
|
Error error;
|
|
|
|
if (m_input_comm.StartReadThread (&error) == false)
|
|
|
|
{
|
|
|
|
FILE *err_fh = GetErrorFileHandle();
|
|
|
|
if (err_fh)
|
|
|
|
{
|
|
|
|
::fprintf (err_fh, "error: failed to main input read thread: %s", error.AsCString() ? error.AsCString() : "unkown error");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *
|
|
|
|
Debugger::GetInputFileHandle ()
|
|
|
|
{
|
|
|
|
return m_input_file.GetFileHandle();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::SetOutputFileHandle (FILE *fh, bool tranfer_ownership)
|
|
|
|
{
|
|
|
|
m_output_file.SetFileHandle (fh, tranfer_ownership);
|
|
|
|
if (m_output_file.GetFileHandle() == NULL)
|
|
|
|
m_output_file.SetFileHandle (stdin, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *
|
|
|
|
Debugger::GetOutputFileHandle ()
|
|
|
|
{
|
|
|
|
return m_output_file.GetFileHandle();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::SetErrorFileHandle (FILE *fh, bool tranfer_ownership)
|
|
|
|
{
|
|
|
|
m_error_file.SetFileHandle (fh, tranfer_ownership);
|
|
|
|
if (m_error_file.GetFileHandle() == NULL)
|
|
|
|
m_error_file.SetFileHandle (stdin, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FILE *
|
|
|
|
Debugger::GetErrorFileHandle ()
|
|
|
|
{
|
|
|
|
return m_error_file.GetFileHandle();
|
|
|
|
}
|
|
|
|
|
|
|
|
CommandInterpreter &
|
|
|
|
Debugger::GetCommandInterpreter ()
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
assert (m_command_interpreter_ap.get());
|
|
|
|
return *m_command_interpreter_ap;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Listener &
|
|
|
|
Debugger::GetListener ()
|
|
|
|
{
|
|
|
|
return m_listener;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TargetSP
|
2010-08-27 05:32:51 +08:00
|
|
|
Debugger::GetSelectedTarget ()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-08-27 05:32:51 +08:00
|
|
|
return m_target_list.GetSelectedTarget ();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ExecutionContext
|
2010-08-27 05:32:51 +08:00
|
|
|
Debugger::GetSelectedExecutionContext ()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
ExecutionContext exe_ctx;
|
|
|
|
exe_ctx.Clear();
|
|
|
|
|
2010-08-27 05:32:51 +08:00
|
|
|
lldb::TargetSP target_sp = GetSelectedTarget();
|
2010-06-09 00:52:24 +08:00
|
|
|
exe_ctx.target = target_sp.get();
|
|
|
|
|
|
|
|
if (target_sp)
|
|
|
|
{
|
|
|
|
exe_ctx.process = target_sp->GetProcessSP().get();
|
|
|
|
if (exe_ctx.process && exe_ctx.process->IsRunning() == false)
|
|
|
|
{
|
2010-08-27 05:32:51 +08:00
|
|
|
exe_ctx.thread = exe_ctx.process->GetThreadList().GetSelectedThread().get();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (exe_ctx.thread == NULL)
|
|
|
|
exe_ctx.thread = exe_ctx.process->GetThreadList().GetThreadAtIndex(0).get();
|
|
|
|
if (exe_ctx.thread)
|
|
|
|
{
|
2010-08-27 05:32:51 +08:00
|
|
|
exe_ctx.frame = exe_ctx.thread->GetSelectedFrame().get();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (exe_ctx.frame == NULL)
|
|
|
|
exe_ctx.frame = exe_ctx.thread->GetStackFrameAtIndex (0).get();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return exe_ctx;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceManager &
|
|
|
|
Debugger::GetSourceManager ()
|
|
|
|
{
|
|
|
|
return m_source_manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TargetList&
|
|
|
|
Debugger::GetTargetList ()
|
|
|
|
{
|
|
|
|
return m_target_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::DispatchInputCallback (void *baton, const void *bytes, size_t bytes_len)
|
|
|
|
{
|
|
|
|
((Debugger *)baton)->DispatchInput ((char *)bytes, bytes_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::DispatchInput (const char *bytes, size_t bytes_len)
|
|
|
|
{
|
|
|
|
if (bytes == NULL || bytes_len == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// TODO: implement the STDIO to the process as an input reader...
|
2010-08-27 05:32:51 +08:00
|
|
|
TargetSP target = GetSelectedTarget();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (target.get() != NULL)
|
|
|
|
{
|
|
|
|
ProcessSP process_sp = target->GetProcessSP();
|
|
|
|
if (process_sp.get() != NULL
|
|
|
|
&& StateIsRunningState (process_sp->GetState()))
|
|
|
|
{
|
|
|
|
Error error;
|
|
|
|
if (process_sp->PutSTDIN (bytes, bytes_len, error) == bytes_len)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteToDefaultReader (bytes, bytes_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::WriteToDefaultReader (const char *bytes, size_t bytes_len)
|
|
|
|
{
|
|
|
|
if (bytes && bytes_len)
|
|
|
|
m_input_reader_data.append (bytes, bytes_len);
|
|
|
|
|
|
|
|
if (m_input_reader_data.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
while (!m_input_readers.empty() && !m_input_reader_data.empty())
|
|
|
|
{
|
|
|
|
while (CheckIfTopInputReaderIsDone ())
|
|
|
|
/* Do nothing. */;
|
|
|
|
|
|
|
|
// Get the input reader from the top of the stack
|
|
|
|
InputReaderSP reader_sp(m_input_readers.top());
|
|
|
|
|
|
|
|
if (!reader_sp)
|
|
|
|
break;
|
|
|
|
|
2010-07-21 06:52:08 +08:00
|
|
|
size_t bytes_handled = reader_sp->HandleRawBytes (m_input_reader_data.c_str(),
|
2010-06-09 00:52:24 +08:00
|
|
|
m_input_reader_data.size());
|
|
|
|
if (bytes_handled)
|
|
|
|
{
|
|
|
|
m_input_reader_data.erase (0, bytes_handled);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// No bytes were handled, we might not have reached our
|
|
|
|
// granularity, just return and wait for more data
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flush out any input readers that are donesvn
|
|
|
|
while (CheckIfTopInputReaderIsDone ())
|
|
|
|
/* Do nothing. */;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::PushInputReader (const InputReaderSP& reader_sp)
|
|
|
|
{
|
|
|
|
if (!reader_sp)
|
|
|
|
return;
|
|
|
|
if (!m_input_readers.empty())
|
|
|
|
{
|
|
|
|
// Deactivate the old top reader
|
|
|
|
InputReaderSP top_reader_sp (m_input_readers.top());
|
|
|
|
if (top_reader_sp)
|
|
|
|
top_reader_sp->Notify (eInputReaderDeactivate);
|
|
|
|
}
|
|
|
|
m_input_readers.push (reader_sp);
|
|
|
|
reader_sp->Notify (eInputReaderActivate);
|
|
|
|
ActivateInputReader (reader_sp);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Debugger::PopInputReader (const lldb::InputReaderSP& pop_reader_sp)
|
|
|
|
{
|
|
|
|
bool result = false;
|
|
|
|
|
|
|
|
// The reader on the stop of the stack is done, so let the next
|
|
|
|
// read on the stack referesh its prompt and if there is one...
|
|
|
|
if (!m_input_readers.empty())
|
|
|
|
{
|
|
|
|
InputReaderSP reader_sp(m_input_readers.top());
|
|
|
|
|
|
|
|
if (!pop_reader_sp || pop_reader_sp.get() == reader_sp.get())
|
|
|
|
{
|
|
|
|
m_input_readers.pop ();
|
|
|
|
reader_sp->Notify (eInputReaderDeactivate);
|
|
|
|
reader_sp->Notify (eInputReaderDone);
|
|
|
|
result = true;
|
|
|
|
|
|
|
|
if (!m_input_readers.empty())
|
|
|
|
{
|
|
|
|
reader_sp = m_input_readers.top();
|
|
|
|
if (reader_sp)
|
|
|
|
{
|
|
|
|
ActivateInputReader (reader_sp);
|
|
|
|
reader_sp->Notify (eInputReaderReactivate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Debugger::CheckIfTopInputReaderIsDone ()
|
|
|
|
{
|
|
|
|
bool result = false;
|
|
|
|
if (!m_input_readers.empty())
|
|
|
|
{
|
|
|
|
InputReaderSP reader_sp(m_input_readers.top());
|
|
|
|
|
|
|
|
if (reader_sp && reader_sp->IsDone())
|
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
PopInputReader (reader_sp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::ActivateInputReader (const InputReaderSP &reader_sp)
|
|
|
|
{
|
|
|
|
FILE *in_fh = GetInputFileHandle();
|
|
|
|
|
|
|
|
if (in_fh)
|
|
|
|
{
|
|
|
|
struct termios in_fh_termios;
|
|
|
|
int in_fd = fileno (in_fh);
|
|
|
|
if (::tcgetattr(in_fd, &in_fh_termios) == 0)
|
|
|
|
{
|
|
|
|
if (reader_sp->GetEcho())
|
|
|
|
in_fh_termios.c_lflag |= ECHO; // Turn on echoing
|
|
|
|
else
|
|
|
|
in_fh_termios.c_lflag &= ~ECHO; // Turn off echoing
|
|
|
|
|
|
|
|
switch (reader_sp->GetGranularity())
|
|
|
|
{
|
|
|
|
case eInputReaderGranularityByte:
|
|
|
|
case eInputReaderGranularityWord:
|
|
|
|
in_fh_termios.c_lflag &= ~ICANON; // Get one char at a time
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eInputReaderGranularityLine:
|
|
|
|
case eInputReaderGranularityAll:
|
|
|
|
in_fh_termios.c_lflag |= ICANON; // Get lines at a time
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
::tcsetattr (in_fd, TCSANOW, &in_fh_termios);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::UpdateExecutionContext (ExecutionContext *override_context)
|
|
|
|
{
|
|
|
|
m_exe_ctx.Clear();
|
|
|
|
|
|
|
|
if (override_context != NULL)
|
|
|
|
{
|
|
|
|
m_exe_ctx.target = override_context->target;
|
|
|
|
m_exe_ctx.process = override_context->process;
|
|
|
|
m_exe_ctx.thread = override_context->thread;
|
|
|
|
m_exe_ctx.frame = override_context->frame;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-08-27 05:32:51 +08:00
|
|
|
TargetSP target_sp (GetSelectedTarget());
|
2010-06-23 09:19:29 +08:00
|
|
|
if (target_sp)
|
|
|
|
{
|
|
|
|
m_exe_ctx.target = target_sp.get();
|
|
|
|
m_exe_ctx.process = target_sp->GetProcessSP().get();
|
2010-09-04 06:35:47 +08:00
|
|
|
if (m_exe_ctx.process && m_exe_ctx.process->IsAlive() && !m_exe_ctx.process->IsRunning())
|
2010-06-23 09:19:29 +08:00
|
|
|
{
|
2010-08-27 05:32:51 +08:00
|
|
|
m_exe_ctx.thread = m_exe_ctx.process->GetThreadList().GetSelectedThread().get();
|
2010-06-23 09:19:29 +08:00
|
|
|
if (m_exe_ctx.thread == NULL)
|
2010-09-11 07:06:30 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
m_exe_ctx.thread = m_exe_ctx.process->GetThreadList().GetThreadAtIndex(0).get();
|
2010-09-11 07:06:30 +08:00
|
|
|
// If we didn't have a selected thread, select one here.
|
|
|
|
if (m_exe_ctx.thread != NULL)
|
2010-09-14 08:53:53 +08:00
|
|
|
m_exe_ctx.process->GetThreadList().SetSelectedThreadByID(m_exe_ctx.thread->GetID());
|
2010-09-11 07:06:30 +08:00
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
if (m_exe_ctx.thread)
|
|
|
|
{
|
2010-08-27 05:32:51 +08:00
|
|
|
m_exe_ctx.frame = m_exe_ctx.thread->GetSelectedFrame().get();
|
2010-06-23 09:19:29 +08:00
|
|
|
if (m_exe_ctx.frame == NULL)
|
2010-09-11 07:06:30 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
m_exe_ctx.frame = m_exe_ctx.thread->GetStackFrameAtIndex (0).get();
|
2010-09-11 07:06:30 +08:00
|
|
|
// If we didn't have a selected frame select one here.
|
|
|
|
if (m_exe_ctx.frame != NULL)
|
|
|
|
m_exe_ctx.thread->SetSelectedFrame(m_exe_ctx.frame);
|
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-01 00:22:25 +08:00
|
|
|
DebuggerSP
|
|
|
|
Debugger::FindDebuggerWithID (lldb::user_id_t id)
|
|
|
|
{
|
|
|
|
lldb::DebuggerSP debugger_sp;
|
|
|
|
|
|
|
|
Mutex::Locker locker (GetDebuggerListMutex ());
|
|
|
|
DebuggerList &debugger_list = GetDebuggerList();
|
|
|
|
DebuggerList::iterator pos, end = debugger_list.end();
|
|
|
|
for (pos = debugger_list.begin(); pos != end; ++pos)
|
|
|
|
{
|
|
|
|
if ((*pos).get()->GetID() == id)
|
|
|
|
{
|
|
|
|
debugger_sp = *pos;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return debugger_sp;
|
|
|
|
}
|
2010-09-04 08:03:46 +08:00
|
|
|
|
|
|
|
lldb::UserSettingsControllerSP &
|
|
|
|
Debugger::GetSettingsController (bool finish)
|
|
|
|
{
|
|
|
|
static lldb::UserSettingsControllerSP g_settings_controller (new DebuggerSettingsController);
|
|
|
|
static bool initialized = false;
|
|
|
|
|
|
|
|
if (!initialized)
|
|
|
|
{
|
2010-09-08 04:27:09 +08:00
|
|
|
initialized = UserSettingsController::InitializeSettingsController (g_settings_controller,
|
|
|
|
Debugger::DebuggerSettingsController::global_settings_table,
|
|
|
|
Debugger::DebuggerSettingsController::instance_settings_table);
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (finish)
|
|
|
|
{
|
|
|
|
UserSettingsControllerSP parent = g_settings_controller->GetParent();
|
|
|
|
if (parent)
|
|
|
|
parent->RemoveChild (g_settings_controller);
|
|
|
|
g_settings_controller.reset();
|
|
|
|
}
|
|
|
|
return g_settings_controller;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------
|
|
|
|
// class Debugger::DebuggerSettingsController
|
|
|
|
//--------------------------------------------------
|
|
|
|
|
|
|
|
Debugger::DebuggerSettingsController::DebuggerSettingsController () :
|
2010-09-09 14:25:08 +08:00
|
|
|
UserSettingsController ("", lldb::UserSettingsControllerSP())
|
2010-09-04 08:03:46 +08:00
|
|
|
{
|
2010-09-09 01:48:55 +08:00
|
|
|
m_default_settings.reset (new DebuggerInstanceSettings (*this, false,
|
|
|
|
InstanceSettings::GetDefaultName().AsCString()));
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Debugger::DebuggerSettingsController::~DebuggerSettingsController ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
lldb::InstanceSettingsSP
|
2010-09-09 01:48:55 +08:00
|
|
|
Debugger::DebuggerSettingsController::CreateNewInstanceSettings (const char *instance_name)
|
2010-09-04 08:03:46 +08:00
|
|
|
{
|
2010-09-09 01:48:55 +08:00
|
|
|
DebuggerInstanceSettings *new_settings = new DebuggerInstanceSettings (*(Debugger::GetSettingsController().get()),
|
|
|
|
false, instance_name);
|
2010-09-04 08:03:46 +08:00
|
|
|
lldb::InstanceSettingsSP new_settings_sp (new_settings);
|
|
|
|
return new_settings_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2010-09-09 14:25:08 +08:00
|
|
|
Debugger::DebuggerInstanceSettings::ValidTermWidthValue (const char *value, Error err)
|
2010-09-04 08:03:46 +08:00
|
|
|
{
|
|
|
|
bool valid = true;
|
|
|
|
|
|
|
|
// Verify we have a value string.
|
|
|
|
if (value == NULL
|
|
|
|
|| strlen (value) == 0)
|
|
|
|
{
|
|
|
|
valid = false;
|
|
|
|
err.SetErrorString ("Missing value. Can't set terminal width without a value.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify the string consists entirely of digits.
|
|
|
|
if (valid)
|
|
|
|
{
|
|
|
|
int len = strlen (value);
|
|
|
|
for (int i = 0; i < len; ++i)
|
|
|
|
if (! isdigit (value[i]))
|
|
|
|
{
|
|
|
|
valid = false;
|
2010-09-09 14:25:08 +08:00
|
|
|
err.SetErrorStringWithFormat ("'%s' is not a valid representation of an unsigned integer.\n", value);
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify the term-width is 'reasonable' (e.g. 10 <= width <= 250).
|
|
|
|
if (valid)
|
|
|
|
{
|
|
|
|
int width = atoi (value);
|
|
|
|
if (width < 10
|
|
|
|
|| width > 250)
|
|
|
|
{
|
|
|
|
valid = false;
|
|
|
|
err.SetErrorString ("Invalid term-width value; value must be between 10 and 250.\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//--------------------------------------------------
|
|
|
|
// class DebuggerInstanceSettings
|
|
|
|
//--------------------------------------------------
|
|
|
|
|
2010-09-09 01:48:55 +08:00
|
|
|
DebuggerInstanceSettings::DebuggerInstanceSettings (UserSettingsController &owner, bool live_instance,
|
|
|
|
const char *name) :
|
2010-09-17 03:05:55 +08:00
|
|
|
InstanceSettings (owner, (name == NULL ? InstanceSettings::InvalidName().AsCString() : name), live_instance),
|
2010-09-04 08:03:46 +08:00
|
|
|
m_prompt (),
|
|
|
|
m_script_lang ()
|
|
|
|
{
|
2010-09-10 02:26:37 +08:00
|
|
|
// CopyInstanceSettings is a pure virtual function in InstanceSettings; it therefore cannot be called
|
|
|
|
// until the vtables for DebuggerInstanceSettings are properly set up, i.e. AFTER all the initializers.
|
|
|
|
// For this reason it has to be called here, rather than in the initializer or in the parent constructor.
|
2010-09-17 03:05:55 +08:00
|
|
|
// The same is true of CreateInstanceName().
|
|
|
|
|
|
|
|
if (GetInstanceName() == InstanceSettings::InvalidName())
|
|
|
|
{
|
|
|
|
ChangeInstanceName (std::string (CreateInstanceName().AsCString()));
|
|
|
|
m_owner.RegisterInstanceSettings (this);
|
|
|
|
}
|
2010-09-10 02:26:37 +08:00
|
|
|
|
|
|
|
if (live_instance)
|
2010-09-04 08:03:46 +08:00
|
|
|
{
|
|
|
|
const lldb::InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name);
|
|
|
|
CopyInstanceSettings (pending_settings, false);
|
2010-09-10 02:26:37 +08:00
|
|
|
//m_owner.RemovePendingSettings (m_instance_name);
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DebuggerInstanceSettings::DebuggerInstanceSettings (const DebuggerInstanceSettings &rhs) :
|
|
|
|
InstanceSettings (*(Debugger::GetSettingsController().get()), CreateInstanceName ().AsCString()),
|
|
|
|
m_prompt (rhs.m_prompt),
|
|
|
|
m_script_lang (rhs.m_script_lang)
|
|
|
|
{
|
|
|
|
const lldb::InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name);
|
|
|
|
CopyInstanceSettings (pending_settings, false);
|
|
|
|
m_owner.RemovePendingSettings (m_instance_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
DebuggerInstanceSettings::~DebuggerInstanceSettings ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
DebuggerInstanceSettings&
|
|
|
|
DebuggerInstanceSettings::operator= (const DebuggerInstanceSettings &rhs)
|
|
|
|
{
|
|
|
|
if (this != &rhs)
|
|
|
|
{
|
|
|
|
m_prompt = rhs.m_prompt;
|
|
|
|
m_script_lang = rhs.m_script_lang;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
DebuggerInstanceSettings::UpdateInstanceSettingsVariable (const ConstString &var_name,
|
|
|
|
const char *index_value,
|
|
|
|
const char *value,
|
|
|
|
const ConstString &instance_name,
|
|
|
|
const SettingEntry &entry,
|
|
|
|
lldb::VarSetOperationType op,
|
|
|
|
Error &err,
|
|
|
|
bool pending)
|
|
|
|
{
|
|
|
|
if (var_name == PromptVarName())
|
|
|
|
{
|
2010-09-09 14:25:08 +08:00
|
|
|
UserSettingsController::UpdateStringVariable (op, m_prompt, value, err);
|
2010-09-04 08:03:46 +08:00
|
|
|
if (!pending)
|
|
|
|
{
|
2010-09-08 02:35:40 +08:00
|
|
|
// 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to
|
|
|
|
// strip off the brackets before passing it to BroadcastPromptChange.
|
|
|
|
|
|
|
|
std::string tmp_instance_name (instance_name.AsCString());
|
|
|
|
if ((tmp_instance_name[0] == '[')
|
|
|
|
&& (tmp_instance_name[instance_name.GetLength() - 1] == ']'))
|
|
|
|
tmp_instance_name = tmp_instance_name.substr (1, instance_name.GetLength() - 2);
|
|
|
|
ConstString new_name (tmp_instance_name.c_str());
|
|
|
|
|
|
|
|
BroadcastPromptChange (new_name, m_prompt.c_str());
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (var_name == ScriptLangVarName())
|
|
|
|
{
|
|
|
|
bool success;
|
|
|
|
m_script_lang = Args::StringToScriptLanguage (value, eScriptLanguageDefault,
|
|
|
|
&success);
|
|
|
|
}
|
2010-09-09 14:25:08 +08:00
|
|
|
else if (var_name == TermWidthVarName())
|
|
|
|
{
|
|
|
|
if (ValidTermWidthValue (value, err))
|
|
|
|
{
|
|
|
|
m_term_width = atoi (value);
|
|
|
|
}
|
|
|
|
}
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::DebuggerSettingsController::UpdateGlobalVariable (const ConstString &var_name,
|
|
|
|
const char *index_value,
|
|
|
|
const char *value,
|
|
|
|
const SettingEntry &entry,
|
|
|
|
lldb::VarSetOperationType op,
|
|
|
|
Error &err)
|
|
|
|
{
|
2010-09-09 14:25:08 +08:00
|
|
|
// There should not be any global variables at the Debugger level.
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
DebuggerInstanceSettings::GetInstanceSettingsValue (const SettingEntry &entry,
|
|
|
|
const ConstString &var_name,
|
|
|
|
StringList &value)
|
|
|
|
{
|
|
|
|
if (var_name == PromptVarName())
|
|
|
|
{
|
|
|
|
value.AppendString (m_prompt.c_str());
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (var_name == ScriptLangVarName())
|
|
|
|
{
|
|
|
|
value.AppendString (ScriptInterpreter::LanguageToString (m_script_lang).c_str());
|
|
|
|
}
|
2010-09-09 14:25:08 +08:00
|
|
|
else if (var_name == TermWidthVarName())
|
|
|
|
{
|
|
|
|
StreamString width_str;
|
|
|
|
width_str.Printf ("%d", m_term_width);
|
|
|
|
value.AppendString (width_str.GetData());
|
|
|
|
}
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
DebuggerInstanceSettings::CopyInstanceSettings (const lldb::InstanceSettingsSP &new_settings,
|
|
|
|
bool pending)
|
|
|
|
{
|
|
|
|
if (new_settings.get() == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DebuggerInstanceSettings *new_debugger_settings = (DebuggerInstanceSettings *) new_settings.get();
|
|
|
|
|
|
|
|
m_prompt = new_debugger_settings->m_prompt;
|
|
|
|
if (!pending)
|
2010-09-08 02:35:40 +08:00
|
|
|
{
|
|
|
|
// 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to
|
|
|
|
// strip off the brackets before passing it to BroadcastPromptChange.
|
|
|
|
|
|
|
|
std::string tmp_instance_name (m_instance_name.AsCString());
|
|
|
|
if ((tmp_instance_name[0] == '[')
|
|
|
|
&& (tmp_instance_name[m_instance_name.GetLength() - 1] == ']'))
|
|
|
|
tmp_instance_name = tmp_instance_name.substr (1, m_instance_name.GetLength() - 2);
|
|
|
|
ConstString new_name (tmp_instance_name.c_str());
|
|
|
|
|
|
|
|
BroadcastPromptChange (new_name, m_prompt.c_str());
|
|
|
|
}
|
2010-09-04 08:03:46 +08:00
|
|
|
|
|
|
|
m_script_lang = new_debugger_settings->m_script_lang;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Debugger::DebuggerSettingsController::GetGlobalSettingsValue (const ConstString &var_name,
|
|
|
|
StringList &value)
|
|
|
|
{
|
2010-09-09 14:25:08 +08:00
|
|
|
// There should not be any global variables at the Debugger level.
|
2010-09-04 08:03:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
DebuggerInstanceSettings::BroadcastPromptChange (const ConstString &instance_name, const char *new_prompt)
|
|
|
|
{
|
|
|
|
std::string tmp_prompt;
|
|
|
|
|
|
|
|
if (new_prompt != NULL)
|
|
|
|
{
|
|
|
|
tmp_prompt = new_prompt ;
|
|
|
|
int len = tmp_prompt.size();
|
|
|
|
if (len > 1
|
|
|
|
&& (tmp_prompt[0] == '\'' || tmp_prompt[0] == '"')
|
|
|
|
&& (tmp_prompt[len-1] == tmp_prompt[0]))
|
|
|
|
{
|
|
|
|
tmp_prompt = tmp_prompt.substr(1,len-2);
|
|
|
|
}
|
|
|
|
len = tmp_prompt.size();
|
|
|
|
if (tmp_prompt[len-1] != ' ')
|
|
|
|
tmp_prompt.append(" ");
|
|
|
|
}
|
|
|
|
EventSP new_event_sp;
|
|
|
|
new_event_sp.reset (new Event(CommandInterpreter::eBroadcastBitResetPrompt,
|
|
|
|
new EventDataBytes (tmp_prompt.c_str())));
|
|
|
|
|
|
|
|
if (instance_name.GetLength() != 0)
|
|
|
|
{
|
|
|
|
// Set prompt for a particular instance.
|
|
|
|
Debugger *dbg = Debugger::FindDebuggerWithInstanceName (instance_name).get();
|
|
|
|
if (dbg != NULL)
|
|
|
|
{
|
|
|
|
dbg->GetCommandInterpreter().BroadcastEvent (new_event_sp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ConstString
|
|
|
|
DebuggerInstanceSettings::CreateInstanceName ()
|
|
|
|
{
|
|
|
|
static int instance_count = 1;
|
|
|
|
StreamString sstr;
|
|
|
|
|
|
|
|
sstr.Printf ("debugger_%d", instance_count);
|
|
|
|
++instance_count;
|
|
|
|
|
|
|
|
const ConstString ret_val (sstr.GetData());
|
|
|
|
|
|
|
|
return ret_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ConstString &
|
|
|
|
DebuggerInstanceSettings::PromptVarName ()
|
|
|
|
{
|
|
|
|
static ConstString prompt_var_name ("prompt");
|
|
|
|
|
|
|
|
return prompt_var_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ConstString &
|
|
|
|
DebuggerInstanceSettings::ScriptLangVarName ()
|
|
|
|
{
|
|
|
|
static ConstString script_lang_var_name ("script-lang");
|
|
|
|
|
|
|
|
return script_lang_var_name;
|
|
|
|
}
|
|
|
|
|
2010-09-09 14:25:08 +08:00
|
|
|
const ConstString &
|
|
|
|
DebuggerInstanceSettings::TermWidthVarName ()
|
|
|
|
{
|
|
|
|
static ConstString term_width_var_name ("term-width");
|
|
|
|
|
|
|
|
return term_width_var_name;
|
|
|
|
}
|
|
|
|
|
2010-09-04 08:03:46 +08:00
|
|
|
//--------------------------------------------------
|
|
|
|
// DebuggerSettingsController Variable Tables
|
|
|
|
//--------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
SettingEntry
|
|
|
|
Debugger::DebuggerSettingsController::global_settings_table[] =
|
|
|
|
{
|
|
|
|
//{ "var-name", var-type, "default", enum-table, init'd, hidden, "help-text"},
|
2010-09-09 14:25:08 +08:00
|
|
|
// The Debugger level global table should always be empty; all Debugger settable variables should be instance
|
|
|
|
// variables.
|
2010-09-04 08:03:46 +08:00
|
|
|
{ NULL, eSetVarTypeNone, NULL, NULL, 0, 0, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
SettingEntry
|
|
|
|
Debugger::DebuggerSettingsController::instance_settings_table[] =
|
|
|
|
{
|
|
|
|
//{ "var-name", var-type , "default", enum-table, init'd, hidden, "help-text"},
|
2010-09-09 14:25:08 +08:00
|
|
|
{ "term-width" , eSetVarTypeInt, "80" , NULL, false , false , "The maximum number of columns to use for displaying text." },
|
2010-09-04 08:03:46 +08:00
|
|
|
{ "script-lang" , eSetVarTypeString, "python", NULL, false, false, "The script language to be used for evaluating user-written scripts." },
|
|
|
|
{ "prompt" , eSetVarTypeString, "(lldb)", NULL, false, false, "The debugger command line prompt displayed for the user." },
|
|
|
|
{ NULL, eSetVarTypeNone, NULL, NULL, 0, 0, NULL }
|
|
|
|
};
|