//===-- 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/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(); } }