forked from OSchip/llvm-project
286 lines
7.6 KiB
C++
286 lines
7.6 KiB
C++
|
//===-- SBThread.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-python.h"
|
||
|
|
||
|
#include "lldb/API/SBThread.h"
|
||
|
|
||
|
#include "lldb/API/SBSymbolContext.h"
|
||
|
#include "lldb/API/SBFileSpec.h"
|
||
|
#include "lldb/API/SBStream.h"
|
||
|
#include "lldb/Breakpoint/BreakpointLocation.h"
|
||
|
#include "lldb/Core/Debugger.h"
|
||
|
#include "lldb/Core/State.h"
|
||
|
#include "lldb/Core/Stream.h"
|
||
|
#include "lldb/Core/StreamFile.h"
|
||
|
#include "lldb/Core/StructuredData.h"
|
||
|
#include "lldb/Interpreter/CommandInterpreter.h"
|
||
|
#include "lldb/Target/SystemRuntime.h"
|
||
|
#include "lldb/Target/Thread.h"
|
||
|
#include "lldb/Target/ThreadPlan.h"
|
||
|
#include "lldb/Target/Process.h"
|
||
|
#include "lldb/Target/Queue.h"
|
||
|
#include "lldb/Symbol/SymbolContext.h"
|
||
|
#include "lldb/Symbol/CompileUnit.h"
|
||
|
#include "lldb/Target/StopInfo.h"
|
||
|
#include "lldb/Target/Target.h"
|
||
|
#include "lldb/Target/ThreadPlan.h"
|
||
|
#include "lldb/Target/ThreadPlanPython.h"
|
||
|
#include "lldb/Target/ThreadPlanStepInstruction.h"
|
||
|
#include "lldb/Target/ThreadPlanStepOut.h"
|
||
|
#include "lldb/Target/ThreadPlanStepRange.h"
|
||
|
#include "lldb/Target/ThreadPlanStepInRange.h"
|
||
|
|
||
|
|
||
|
#include "lldb/API/SBAddress.h"
|
||
|
#include "lldb/API/SBDebugger.h"
|
||
|
#include "lldb/API/SBEvent.h"
|
||
|
#include "lldb/API/SBFrame.h"
|
||
|
#include "lldb/API/SBProcess.h"
|
||
|
#include "lldb/API/SBThreadPlan.h"
|
||
|
#include "lldb/API/SBValue.h"
|
||
|
|
||
|
using namespace lldb;
|
||
|
using namespace lldb_private;
|
||
|
|
||
|
//----------------------------------------------------------------------
|
||
|
// Constructors
|
||
|
//----------------------------------------------------------------------
|
||
|
SBThreadPlan::SBThreadPlan ()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
SBThreadPlan::SBThreadPlan (const ThreadPlanSP& lldb_object_sp) :
|
||
|
m_opaque_sp (lldb_object_sp)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
SBThreadPlan::SBThreadPlan (const SBThreadPlan &rhs) :
|
||
|
m_opaque_sp (rhs.m_opaque_sp)
|
||
|
{
|
||
|
|
||
|
}
|
||
|
|
||
|
SBThreadPlan::SBThreadPlan (lldb::SBThread &sb_thread, const char *class_name)
|
||
|
{
|
||
|
Thread *thread = sb_thread.get();
|
||
|
if (thread)
|
||
|
m_opaque_sp.reset(new ThreadPlanPython(*thread, class_name));
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------
|
||
|
// Assignment operator
|
||
|
//----------------------------------------------------------------------
|
||
|
|
||
|
const lldb::SBThreadPlan &
|
||
|
SBThreadPlan::operator = (const SBThreadPlan &rhs)
|
||
|
{
|
||
|
if (this != &rhs)
|
||
|
m_opaque_sp = rhs.m_opaque_sp;
|
||
|
return *this;
|
||
|
}
|
||
|
//----------------------------------------------------------------------
|
||
|
// Destructor
|
||
|
//----------------------------------------------------------------------
|
||
|
SBThreadPlan::~SBThreadPlan()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
lldb_private::ThreadPlan *
|
||
|
SBThreadPlan::get()
|
||
|
{
|
||
|
return m_opaque_sp.get();
|
||
|
}
|
||
|
|
||
|
bool
|
||
|
SBThreadPlan::IsValid() const
|
||
|
{
|
||
|
return m_opaque_sp.get() != NULL;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
SBThreadPlan::Clear ()
|
||
|
{
|
||
|
m_opaque_sp.reset();
|
||
|
}
|
||
|
|
||
|
lldb::StopReason
|
||
|
SBThreadPlan::GetStopReason()
|
||
|
{
|
||
|
return eStopReasonNone;
|
||
|
}
|
||
|
|
||
|
size_t
|
||
|
SBThreadPlan::GetStopReasonDataCount()
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
uint64_t
|
||
|
SBThreadPlan::GetStopReasonDataAtIndex(uint32_t idx)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
SBThread
|
||
|
SBThreadPlan::GetThread () const
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
{
|
||
|
return SBThread(m_opaque_sp->GetThread().shared_from_this());
|
||
|
}
|
||
|
else
|
||
|
return SBThread();
|
||
|
}
|
||
|
|
||
|
bool
|
||
|
SBThreadPlan::GetDescription (lldb::SBStream &description) const
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
{
|
||
|
m_opaque_sp->GetDescription(description.get(), eDescriptionLevelFull);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
description.Printf("Empty SBThreadPlan");
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
SBThreadPlan::SetThreadPlan (const ThreadPlanSP& lldb_object_sp)
|
||
|
{
|
||
|
m_opaque_sp = lldb_object_sp;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
SBThreadPlan::SetPlanComplete (bool success)
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
m_opaque_sp->SetPlanComplete (success);
|
||
|
}
|
||
|
|
||
|
bool
|
||
|
SBThreadPlan::IsPlanComplete()
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
return m_opaque_sp->IsPlanComplete();
|
||
|
else
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool
|
||
|
SBThreadPlan::IsValid()
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
return m_opaque_sp->ValidatePlan(nullptr);
|
||
|
else
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// This section allows an SBThreadPlan to push another of the common types of plans...
|
||
|
//
|
||
|
// FIXME, you should only be able to queue thread plans from inside the methods of a
|
||
|
// Scripted Thread Plan. Need a way to enforce that.
|
||
|
|
||
|
SBThreadPlan
|
||
|
SBThreadPlan::QueueThreadPlanForStepOverRange (SBAddress &sb_start_address,
|
||
|
lldb::addr_t size)
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
{
|
||
|
Address *start_address = sb_start_address.get();
|
||
|
if (!start_address)
|
||
|
{
|
||
|
return SBThreadPlan();
|
||
|
}
|
||
|
|
||
|
AddressRange range (*start_address, size);
|
||
|
SymbolContext sc;
|
||
|
start_address->CalculateSymbolContext(&sc);
|
||
|
return SBThreadPlan (m_opaque_sp->GetThread().QueueThreadPlanForStepOverRange (false,
|
||
|
range,
|
||
|
sc,
|
||
|
eAllThreads));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return SBThreadPlan();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
SBThreadPlan
|
||
|
SBThreadPlan::QueueThreadPlanForStepInRange (SBAddress &sb_start_address,
|
||
|
lldb::addr_t size)
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
{
|
||
|
Address *start_address = sb_start_address.get();
|
||
|
if (!start_address)
|
||
|
{
|
||
|
return SBThreadPlan();
|
||
|
}
|
||
|
|
||
|
AddressRange range (*start_address, size);
|
||
|
SymbolContext sc;
|
||
|
start_address->CalculateSymbolContext(&sc);
|
||
|
return SBThreadPlan (m_opaque_sp->GetThread().QueueThreadPlanForStepInRange (false,
|
||
|
range,
|
||
|
sc,
|
||
|
NULL,
|
||
|
eAllThreads));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return SBThreadPlan();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
SBThreadPlan
|
||
|
SBThreadPlan::QueueThreadPlanForStepOut (uint32_t frame_idx_to_step_to, bool first_insn)
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
{
|
||
|
SymbolContext sc;
|
||
|
sc = m_opaque_sp->GetThread().GetStackFrameAtIndex(0)->GetSymbolContext(lldb::eSymbolContextEverything);
|
||
|
return SBThreadPlan (m_opaque_sp->GetThread().QueueThreadPlanForStepOut (false,
|
||
|
&sc,
|
||
|
first_insn,
|
||
|
false,
|
||
|
eVoteYes,
|
||
|
eVoteNoOpinion,
|
||
|
frame_idx_to_step_to));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return SBThreadPlan();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
SBThreadPlan
|
||
|
SBThreadPlan::QueueThreadPlanForRunToAddress (SBAddress sb_address)
|
||
|
{
|
||
|
if (m_opaque_sp)
|
||
|
{
|
||
|
Address *address = sb_address.get();
|
||
|
if (!address)
|
||
|
return SBThreadPlan();
|
||
|
|
||
|
return SBThreadPlan (m_opaque_sp->GetThread().QueueThreadPlanForRunToAddress (false,
|
||
|
*address,
|
||
|
false));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return SBThreadPlan();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|