130fdc8d8SChris Lattner //===-- SBThread.cpp --------------------------------------------*- C++ -*-===// 230fdc8d8SChris Lattner // 330fdc8d8SChris Lattner // The LLVM Compiler Infrastructure 430fdc8d8SChris Lattner // 530fdc8d8SChris Lattner // This file is distributed under the University of Illinois Open Source 630fdc8d8SChris Lattner // License. See LICENSE.TXT for details. 730fdc8d8SChris Lattner // 830fdc8d8SChris Lattner //===----------------------------------------------------------------------===// 930fdc8d8SChris Lattner 104c5de699SEli Friedman #include "lldb/API/SBThread.h" 1130fdc8d8SChris Lattner 1230fdc8d8SChris Lattner #include "lldb/API/SBSymbolContext.h" 1330fdc8d8SChris Lattner #include "lldb/API/SBFileSpec.h" 14dde9cff3SCaroline Tice #include "lldb/API/SBStream.h" 154e78f606SGreg Clayton #include "lldb/Breakpoint/BreakpointLocation.h" 166611103cSGreg Clayton #include "lldb/Core/Debugger.h" 1730fdc8d8SChris Lattner #include "lldb/Core/Stream.h" 1830fdc8d8SChris Lattner #include "lldb/Core/StreamFile.h" 196611103cSGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h" 2030fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 2130fdc8d8SChris Lattner #include "lldb/Target/Process.h" 2230fdc8d8SChris Lattner #include "lldb/Symbol/SymbolContext.h" 2330fdc8d8SChris Lattner #include "lldb/Symbol/CompileUnit.h" 24f4b47e15SGreg Clayton #include "lldb/Target/StopInfo.h" 2530fdc8d8SChris Lattner #include "lldb/Target/Target.h" 2630fdc8d8SChris Lattner #include "lldb/Target/ThreadPlan.h" 2730fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepInstruction.h" 2830fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepOut.h" 2930fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepRange.h" 3030fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepInRange.h" 3130fdc8d8SChris Lattner 3230fdc8d8SChris Lattner 334c5de699SEli Friedman #include "lldb/API/SBAddress.h" 344c5de699SEli Friedman #include "lldb/API/SBFrame.h" 35b7f6b2faSJim Ingham // DONT THINK THIS IS NECESSARY: #include "lldb/API/SBSourceManager.h" 364c5de699SEli Friedman #include "lldb/API/SBDebugger.h" 374c5de699SEli Friedman #include "lldb/API/SBProcess.h" 3830fdc8d8SChris Lattner 3930fdc8d8SChris Lattner using namespace lldb; 4030fdc8d8SChris Lattner using namespace lldb_private; 4130fdc8d8SChris Lattner 42cfd1acedSGreg Clayton //---------------------------------------------------------------------- 43cfd1acedSGreg Clayton // Constructors 44cfd1acedSGreg Clayton //---------------------------------------------------------------------- 4530fdc8d8SChris Lattner SBThread::SBThread () : 466611103cSGreg Clayton m_opaque_sp () 4730fdc8d8SChris Lattner { 4830fdc8d8SChris Lattner } 4930fdc8d8SChris Lattner 5030fdc8d8SChris Lattner SBThread::SBThread (const ThreadSP& lldb_object_sp) : 516611103cSGreg Clayton m_opaque_sp (lldb_object_sp) 5230fdc8d8SChris Lattner { 5330fdc8d8SChris Lattner } 5430fdc8d8SChris Lattner 5592ef5735SGreg Clayton SBThread::SBThread (const SBThread &rhs) : 5692ef5735SGreg Clayton m_opaque_sp (rhs.m_opaque_sp) 5730fdc8d8SChris Lattner { 5830fdc8d8SChris Lattner } 5930fdc8d8SChris Lattner 6030fdc8d8SChris Lattner //---------------------------------------------------------------------- 61cfd1acedSGreg Clayton // Assignment operator 62cfd1acedSGreg Clayton //---------------------------------------------------------------------- 63cfd1acedSGreg Clayton 64cfd1acedSGreg Clayton const lldb::SBThread & 65cfd1acedSGreg Clayton SBThread::operator = (const SBThread &rhs) 66cfd1acedSGreg Clayton { 67cfd1acedSGreg Clayton if (this != &rhs) 68cfd1acedSGreg Clayton m_opaque_sp = rhs.m_opaque_sp; 69cfd1acedSGreg Clayton return *this; 70cfd1acedSGreg Clayton } 71cfd1acedSGreg Clayton 72cfd1acedSGreg Clayton //---------------------------------------------------------------------- 7330fdc8d8SChris Lattner // Destructor 7430fdc8d8SChris Lattner //---------------------------------------------------------------------- 7530fdc8d8SChris Lattner SBThread::~SBThread() 7630fdc8d8SChris Lattner { 7730fdc8d8SChris Lattner } 7830fdc8d8SChris Lattner 7930fdc8d8SChris Lattner bool 8030fdc8d8SChris Lattner SBThread::IsValid() const 8130fdc8d8SChris Lattner { 82762f7135SGreg Clayton return m_opaque_sp; 8330fdc8d8SChris Lattner } 8430fdc8d8SChris Lattner 8548e42549SGreg Clayton void 8648e42549SGreg Clayton SBThread::Clear () 8748e42549SGreg Clayton { 8848e42549SGreg Clayton m_opaque_sp.reset(); 8948e42549SGreg Clayton } 9048e42549SGreg Clayton 9148e42549SGreg Clayton 9230fdc8d8SChris Lattner StopReason 9330fdc8d8SChris Lattner SBThread::GetStopReason() 9430fdc8d8SChris Lattner { 952d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 96ceb6b139SCaroline Tice 97ceb6b139SCaroline Tice StopReason reason = eStopReasonInvalid; 986611103cSGreg Clayton if (m_opaque_sp) 9930fdc8d8SChris Lattner { 100af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 101b15bfc75SJim Ingham StopInfoSP stop_info_sp = m_opaque_sp->GetStopInfo (); 102b15bfc75SJim Ingham if (stop_info_sp) 103ceb6b139SCaroline Tice reason = stop_info_sp->GetStopReason(); 10430fdc8d8SChris Lattner } 105ceb6b139SCaroline Tice 106ceb6b139SCaroline Tice if (log) 1074838131bSGreg Clayton log->Printf ("SBThread(%p)::GetStopReason () => %s", m_opaque_sp.get(), 108750cd175SCaroline Tice Thread::StopReasonAsCString (reason)); 109ceb6b139SCaroline Tice 110ceb6b139SCaroline Tice return reason; 11130fdc8d8SChris Lattner } 11230fdc8d8SChris Lattner 11330fdc8d8SChris Lattner size_t 1144e78f606SGreg Clayton SBThread::GetStopReasonDataCount () 1154e78f606SGreg Clayton { 1164e78f606SGreg Clayton if (m_opaque_sp) 1174e78f606SGreg Clayton { 118af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 1194e78f606SGreg Clayton StopInfoSP stop_info_sp = m_opaque_sp->GetStopInfo (); 1204e78f606SGreg Clayton if (stop_info_sp) 1214e78f606SGreg Clayton { 1224e78f606SGreg Clayton StopReason reason = stop_info_sp->GetStopReason(); 1234e78f606SGreg Clayton switch (reason) 1244e78f606SGreg Clayton { 1254e78f606SGreg Clayton case eStopReasonInvalid: 1264e78f606SGreg Clayton case eStopReasonNone: 1274e78f606SGreg Clayton case eStopReasonTrace: 1284e78f606SGreg Clayton case eStopReasonPlanComplete: 1294e78f606SGreg Clayton // There is no data for these stop reasons. 1304e78f606SGreg Clayton return 0; 1314e78f606SGreg Clayton 1324e78f606SGreg Clayton case eStopReasonBreakpoint: 1334e78f606SGreg Clayton { 1344e78f606SGreg Clayton break_id_t site_id = stop_info_sp->GetValue(); 1354e78f606SGreg Clayton lldb::BreakpointSiteSP bp_site_sp (m_opaque_sp->GetProcess().GetBreakpointSiteList().FindByID (site_id)); 1364e78f606SGreg Clayton if (bp_site_sp) 1374e78f606SGreg Clayton return bp_site_sp->GetNumberOfOwners () * 2; 1384e78f606SGreg Clayton else 1394e78f606SGreg Clayton return 0; // Breakpoint must have cleared itself... 1404e78f606SGreg Clayton } 1414e78f606SGreg Clayton break; 1424e78f606SGreg Clayton 1434e78f606SGreg Clayton case eStopReasonWatchpoint: 1444e78f606SGreg Clayton assert (!"implement watchpoint support in SBThread::GetStopReasonDataCount ()"); 1454e78f606SGreg Clayton return 0; // We don't have watchpoint support yet... 1464e78f606SGreg Clayton 1474e78f606SGreg Clayton case eStopReasonSignal: 1484e78f606SGreg Clayton return 1; 1494e78f606SGreg Clayton 1504e78f606SGreg Clayton case eStopReasonException: 1514e78f606SGreg Clayton return 1; 1524e78f606SGreg Clayton } 1534e78f606SGreg Clayton } 1544e78f606SGreg Clayton } 1554e78f606SGreg Clayton return 0; 1564e78f606SGreg Clayton } 1574e78f606SGreg Clayton 1584e78f606SGreg Clayton uint64_t 1594e78f606SGreg Clayton SBThread::GetStopReasonDataAtIndex (uint32_t idx) 1604e78f606SGreg Clayton { 1614e78f606SGreg Clayton if (m_opaque_sp) 1624e78f606SGreg Clayton { 163af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 1644e78f606SGreg Clayton StopInfoSP stop_info_sp = m_opaque_sp->GetStopInfo (); 1654e78f606SGreg Clayton if (stop_info_sp) 1664e78f606SGreg Clayton { 1674e78f606SGreg Clayton StopReason reason = stop_info_sp->GetStopReason(); 1684e78f606SGreg Clayton switch (reason) 1694e78f606SGreg Clayton { 1704e78f606SGreg Clayton case eStopReasonInvalid: 1714e78f606SGreg Clayton case eStopReasonNone: 1724e78f606SGreg Clayton case eStopReasonTrace: 1734e78f606SGreg Clayton case eStopReasonPlanComplete: 1744e78f606SGreg Clayton // There is no data for these stop reasons. 1754e78f606SGreg Clayton return 0; 1764e78f606SGreg Clayton 1774e78f606SGreg Clayton case eStopReasonBreakpoint: 1784e78f606SGreg Clayton { 1794e78f606SGreg Clayton break_id_t site_id = stop_info_sp->GetValue(); 1804e78f606SGreg Clayton lldb::BreakpointSiteSP bp_site_sp (m_opaque_sp->GetProcess().GetBreakpointSiteList().FindByID (site_id)); 1814e78f606SGreg Clayton if (bp_site_sp) 1824e78f606SGreg Clayton { 1834e78f606SGreg Clayton uint32_t bp_index = idx / 2; 1844e78f606SGreg Clayton BreakpointLocationSP bp_loc_sp (bp_site_sp->GetOwnerAtIndex (bp_index)); 1854e78f606SGreg Clayton if (bp_loc_sp) 1864e78f606SGreg Clayton { 1874e78f606SGreg Clayton if (bp_index & 1) 1884e78f606SGreg Clayton { 1894e78f606SGreg Clayton // Odd idx, return the breakpoint location ID 1904e78f606SGreg Clayton return bp_loc_sp->GetID(); 1914e78f606SGreg Clayton } 1924e78f606SGreg Clayton else 1934e78f606SGreg Clayton { 1944e78f606SGreg Clayton // Even idx, return the breakpoint ID 1954e78f606SGreg Clayton return bp_loc_sp->GetBreakpoint().GetID(); 1964e78f606SGreg Clayton } 1974e78f606SGreg Clayton } 1984e78f606SGreg Clayton } 1994e78f606SGreg Clayton return LLDB_INVALID_BREAK_ID; 2004e78f606SGreg Clayton } 2014e78f606SGreg Clayton break; 2024e78f606SGreg Clayton 2034e78f606SGreg Clayton case eStopReasonWatchpoint: 2044e78f606SGreg Clayton assert (!"implement watchpoint support in SBThread::GetStopReasonDataCount ()"); 2054e78f606SGreg Clayton return 0; // We don't have watchpoint support yet... 2064e78f606SGreg Clayton 2074e78f606SGreg Clayton case eStopReasonSignal: 2084e78f606SGreg Clayton return stop_info_sp->GetValue(); 2094e78f606SGreg Clayton 2104e78f606SGreg Clayton case eStopReasonException: 2114e78f606SGreg Clayton return stop_info_sp->GetValue(); 2124e78f606SGreg Clayton } 2134e78f606SGreg Clayton } 2144e78f606SGreg Clayton } 2154e78f606SGreg Clayton return 0; 2164e78f606SGreg Clayton } 2174e78f606SGreg Clayton 2184e78f606SGreg Clayton size_t 21930fdc8d8SChris Lattner SBThread::GetStopDescription (char *dst, size_t dst_len) 22030fdc8d8SChris Lattner { 2212d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 222ceb6b139SCaroline Tice 2236611103cSGreg Clayton if (m_opaque_sp) 22430fdc8d8SChris Lattner { 225af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 226b15bfc75SJim Ingham StopInfoSP stop_info_sp = m_opaque_sp->GetStopInfo (); 227b15bfc75SJim Ingham if (stop_info_sp) 22830fdc8d8SChris Lattner { 229b15bfc75SJim Ingham const char *stop_desc = stop_info_sp->GetDescription(); 23030fdc8d8SChris Lattner if (stop_desc) 23130fdc8d8SChris Lattner { 232ceb6b139SCaroline Tice if (log) 2334838131bSGreg Clayton log->Printf ("SBThread(%p)::GetStopDescription (dst, dst_len) => \"%s\"", 234750cd175SCaroline Tice m_opaque_sp.get(), stop_desc); 23530fdc8d8SChris Lattner if (dst) 23630fdc8d8SChris Lattner return ::snprintf (dst, dst_len, "%s", stop_desc); 23730fdc8d8SChris Lattner else 23830fdc8d8SChris Lattner { 23930fdc8d8SChris Lattner // NULL dst passed in, return the length needed to contain the description 24030fdc8d8SChris Lattner return ::strlen (stop_desc) + 1; // Include the NULL byte for size 24130fdc8d8SChris Lattner } 24230fdc8d8SChris Lattner } 24330fdc8d8SChris Lattner else 24430fdc8d8SChris Lattner { 24530fdc8d8SChris Lattner size_t stop_desc_len = 0; 246b15bfc75SJim Ingham switch (stop_info_sp->GetStopReason()) 24730fdc8d8SChris Lattner { 24830fdc8d8SChris Lattner case eStopReasonTrace: 24930fdc8d8SChris Lattner case eStopReasonPlanComplete: 25030fdc8d8SChris Lattner { 25130fdc8d8SChris Lattner static char trace_desc[] = "step"; 25230fdc8d8SChris Lattner stop_desc = trace_desc; 25330fdc8d8SChris Lattner stop_desc_len = sizeof(trace_desc); // Include the NULL byte for size 25430fdc8d8SChris Lattner } 25530fdc8d8SChris Lattner break; 25630fdc8d8SChris Lattner 25730fdc8d8SChris Lattner case eStopReasonBreakpoint: 25830fdc8d8SChris Lattner { 25930fdc8d8SChris Lattner static char bp_desc[] = "breakpoint hit"; 26030fdc8d8SChris Lattner stop_desc = bp_desc; 26130fdc8d8SChris Lattner stop_desc_len = sizeof(bp_desc); // Include the NULL byte for size 26230fdc8d8SChris Lattner } 26330fdc8d8SChris Lattner break; 26430fdc8d8SChris Lattner 26530fdc8d8SChris Lattner case eStopReasonWatchpoint: 26630fdc8d8SChris Lattner { 26730fdc8d8SChris Lattner static char wp_desc[] = "watchpoint hit"; 26830fdc8d8SChris Lattner stop_desc = wp_desc; 26930fdc8d8SChris Lattner stop_desc_len = sizeof(wp_desc); // Include the NULL byte for size 27030fdc8d8SChris Lattner } 27130fdc8d8SChris Lattner break; 27230fdc8d8SChris Lattner 27330fdc8d8SChris Lattner case eStopReasonSignal: 27430fdc8d8SChris Lattner { 275b15bfc75SJim Ingham stop_desc = m_opaque_sp->GetProcess().GetUnixSignals ().GetSignalAsCString (stop_info_sp->GetValue()); 27630fdc8d8SChris Lattner if (stop_desc == NULL || stop_desc[0] == '\0') 27730fdc8d8SChris Lattner { 27830fdc8d8SChris Lattner static char signal_desc[] = "signal"; 27930fdc8d8SChris Lattner stop_desc = signal_desc; 28030fdc8d8SChris Lattner stop_desc_len = sizeof(signal_desc); // Include the NULL byte for size 28130fdc8d8SChris Lattner } 28230fdc8d8SChris Lattner } 28330fdc8d8SChris Lattner break; 28430fdc8d8SChris Lattner 28530fdc8d8SChris Lattner case eStopReasonException: 28630fdc8d8SChris Lattner { 28730fdc8d8SChris Lattner char exc_desc[] = "exception"; 28830fdc8d8SChris Lattner stop_desc = exc_desc; 28930fdc8d8SChris Lattner stop_desc_len = sizeof(exc_desc); // Include the NULL byte for size 29030fdc8d8SChris Lattner } 29130fdc8d8SChris Lattner break; 292c982c768SGreg Clayton 293c982c768SGreg Clayton default: 294c982c768SGreg Clayton break; 29530fdc8d8SChris Lattner } 29630fdc8d8SChris Lattner 29730fdc8d8SChris Lattner if (stop_desc && stop_desc[0]) 29830fdc8d8SChris Lattner { 299ceb6b139SCaroline Tice if (log) 30093aa84e8SGreg Clayton log->Printf ("SBThread(%p)::GetStopDescription (dst, dst_len) => '%s'", 301750cd175SCaroline Tice m_opaque_sp.get(), stop_desc); 302ceb6b139SCaroline Tice 30330fdc8d8SChris Lattner if (dst) 30430fdc8d8SChris Lattner return ::snprintf (dst, dst_len, "%s", stop_desc) + 1; // Include the NULL byte 30530fdc8d8SChris Lattner 30630fdc8d8SChris Lattner if (stop_desc_len == 0) 30730fdc8d8SChris Lattner stop_desc_len = ::strlen (stop_desc) + 1; // Include the NULL byte 30830fdc8d8SChris Lattner 30930fdc8d8SChris Lattner return stop_desc_len; 31030fdc8d8SChris Lattner } 31130fdc8d8SChris Lattner } 31230fdc8d8SChris Lattner } 31330fdc8d8SChris Lattner } 31430fdc8d8SChris Lattner if (dst) 31530fdc8d8SChris Lattner *dst = 0; 31630fdc8d8SChris Lattner return 0; 31730fdc8d8SChris Lattner } 31830fdc8d8SChris Lattner 31930fdc8d8SChris Lattner void 32030fdc8d8SChris Lattner SBThread::SetThread (const ThreadSP& lldb_object_sp) 32130fdc8d8SChris Lattner { 3226611103cSGreg Clayton m_opaque_sp = lldb_object_sp; 32330fdc8d8SChris Lattner } 32430fdc8d8SChris Lattner 32530fdc8d8SChris Lattner 32630fdc8d8SChris Lattner lldb::tid_t 32730fdc8d8SChris Lattner SBThread::GetThreadID () const 32830fdc8d8SChris Lattner { 3292d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 330ceb6b139SCaroline Tice 331af67cecdSGreg Clayton lldb::tid_t tid = LLDB_INVALID_THREAD_ID; 3326611103cSGreg Clayton if (m_opaque_sp) 333af67cecdSGreg Clayton tid = m_opaque_sp->GetID(); 334ceb6b139SCaroline Tice 335ceb6b139SCaroline Tice if (log) 336af67cecdSGreg Clayton log->Printf ("SBThread(%p)::GetThreadID () => 0x%4.4x", m_opaque_sp.get(), tid); 337ceb6b139SCaroline Tice 338af67cecdSGreg Clayton return tid; 33930fdc8d8SChris Lattner } 34030fdc8d8SChris Lattner 34130fdc8d8SChris Lattner uint32_t 34230fdc8d8SChris Lattner SBThread::GetIndexID () const 34330fdc8d8SChris Lattner { 3446611103cSGreg Clayton if (m_opaque_sp) 3456611103cSGreg Clayton return m_opaque_sp->GetIndexID(); 34630fdc8d8SChris Lattner return LLDB_INVALID_INDEX32; 34730fdc8d8SChris Lattner } 34830fdc8d8SChris Lattner const char * 34930fdc8d8SChris Lattner SBThread::GetName () const 35030fdc8d8SChris Lattner { 3514838131bSGreg Clayton const char *name = NULL; 3526611103cSGreg Clayton if (m_opaque_sp) 353af67cecdSGreg Clayton { 354af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 3554838131bSGreg Clayton name = m_opaque_sp->GetName(); 356af67cecdSGreg Clayton } 357ceb6b139SCaroline Tice 3582d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 359ceb6b139SCaroline Tice if (log) 3604838131bSGreg Clayton log->Printf ("SBThread(%p)::GetName () => %s", m_opaque_sp.get(), name ? name : "NULL"); 361ceb6b139SCaroline Tice 3624838131bSGreg Clayton return name; 36330fdc8d8SChris Lattner } 36430fdc8d8SChris Lattner 36530fdc8d8SChris Lattner const char * 36630fdc8d8SChris Lattner SBThread::GetQueueName () const 36730fdc8d8SChris Lattner { 3684838131bSGreg Clayton const char *name = NULL; 3696611103cSGreg Clayton if (m_opaque_sp) 370af67cecdSGreg Clayton { 371af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 3724838131bSGreg Clayton name = m_opaque_sp->GetQueueName(); 373af67cecdSGreg Clayton } 374ceb6b139SCaroline Tice 3752d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 376ceb6b139SCaroline Tice if (log) 3774838131bSGreg Clayton log->Printf ("SBThread(%p)::GetQueueName () => %s", m_opaque_sp.get(), name ? name : "NULL"); 378ceb6b139SCaroline Tice 3794838131bSGreg Clayton return name; 38030fdc8d8SChris Lattner } 38130fdc8d8SChris Lattner 38230fdc8d8SChris Lattner 38330fdc8d8SChris Lattner void 38430fdc8d8SChris Lattner SBThread::StepOver (lldb::RunMode stop_other_threads) 38530fdc8d8SChris Lattner { 3862d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 387ceb6b139SCaroline Tice 388ceb6b139SCaroline Tice if (log) 38993aa84e8SGreg Clayton log->Printf ("SBThread(%p)::StepOver (stop_other_threads='%s')", m_opaque_sp.get(), 390ceb6b139SCaroline Tice Thread::RunModeAsCString (stop_other_threads)); 391ceb6b139SCaroline Tice 3926611103cSGreg Clayton if (m_opaque_sp) 39330fdc8d8SChris Lattner { 394af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 39530fdc8d8SChris Lattner bool abort_other_plans = true; 3966611103cSGreg Clayton StackFrameSP frame_sp(m_opaque_sp->GetStackFrameAtIndex (0)); 39730fdc8d8SChris Lattner 39830fdc8d8SChris Lattner if (frame_sp) 39930fdc8d8SChris Lattner { 40030fdc8d8SChris Lattner if (frame_sp->HasDebugInformation ()) 40130fdc8d8SChris Lattner { 40230fdc8d8SChris Lattner SymbolContext sc(frame_sp->GetSymbolContext(eSymbolContextEverything)); 4036611103cSGreg Clayton m_opaque_sp->QueueThreadPlanForStepRange (abort_other_plans, 40430fdc8d8SChris Lattner eStepTypeOver, 40530fdc8d8SChris Lattner sc.line_entry.range, 40630fdc8d8SChris Lattner sc, 407474966a4SGreg Clayton stop_other_threads, 408474966a4SGreg Clayton false); 40930fdc8d8SChris Lattner 41030fdc8d8SChris Lattner } 41130fdc8d8SChris Lattner else 41230fdc8d8SChris Lattner { 4136611103cSGreg Clayton m_opaque_sp->QueueThreadPlanForStepSingleInstruction (true, 41430fdc8d8SChris Lattner abort_other_plans, 41530fdc8d8SChris Lattner stop_other_threads); 41630fdc8d8SChris Lattner } 41730fdc8d8SChris Lattner } 41830fdc8d8SChris Lattner 4196611103cSGreg Clayton Process &process = m_opaque_sp->GetProcess(); 42030fdc8d8SChris Lattner // Why do we need to set the current thread by ID here??? 4212976d00aSJim Ingham process.GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID()); 4225d5028b5SGreg Clayton Error error (process.Resume()); 4235d5028b5SGreg Clayton if (error.Success()) 4245d5028b5SGreg Clayton { 4255d5028b5SGreg Clayton // If we are doing synchronous mode, then wait for the 4265d5028b5SGreg Clayton // process to stop yet again! 4275d5028b5SGreg Clayton if (process.GetTarget().GetDebugger().GetAsyncExecution () == false) 4285d5028b5SGreg Clayton process.WaitForProcessToStop (NULL); 4295d5028b5SGreg Clayton } 43030fdc8d8SChris Lattner } 43130fdc8d8SChris Lattner } 43230fdc8d8SChris Lattner 43330fdc8d8SChris Lattner void 43430fdc8d8SChris Lattner SBThread::StepInto (lldb::RunMode stop_other_threads) 43530fdc8d8SChris Lattner { 4362d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 437ceb6b139SCaroline Tice 438ceb6b139SCaroline Tice if (log) 43993aa84e8SGreg Clayton log->Printf ("SBThread(%p)::StepInto (stop_other_threads='%s')", m_opaque_sp.get(), 440ceb6b139SCaroline Tice Thread::RunModeAsCString (stop_other_threads)); 441ceb6b139SCaroline Tice 4426611103cSGreg Clayton if (m_opaque_sp) 44330fdc8d8SChris Lattner { 444af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 44530fdc8d8SChris Lattner bool abort_other_plans = true; 44630fdc8d8SChris Lattner 4476611103cSGreg Clayton StackFrameSP frame_sp(m_opaque_sp->GetStackFrameAtIndex (0)); 44830fdc8d8SChris Lattner 44930fdc8d8SChris Lattner if (frame_sp && frame_sp->HasDebugInformation ()) 45030fdc8d8SChris Lattner { 451474966a4SGreg Clayton bool avoid_code_without_debug_info = true; 45230fdc8d8SChris Lattner SymbolContext sc(frame_sp->GetSymbolContext(eSymbolContextEverything)); 4536611103cSGreg Clayton m_opaque_sp->QueueThreadPlanForStepRange (abort_other_plans, 45430fdc8d8SChris Lattner eStepTypeInto, 45530fdc8d8SChris Lattner sc.line_entry.range, 45630fdc8d8SChris Lattner sc, 457474966a4SGreg Clayton stop_other_threads, 458474966a4SGreg Clayton avoid_code_without_debug_info); 45930fdc8d8SChris Lattner } 46030fdc8d8SChris Lattner else 46130fdc8d8SChris Lattner { 4626611103cSGreg Clayton m_opaque_sp->QueueThreadPlanForStepSingleInstruction (false, 46330fdc8d8SChris Lattner abort_other_plans, 46430fdc8d8SChris Lattner stop_other_threads); 46530fdc8d8SChris Lattner } 46630fdc8d8SChris Lattner 4676611103cSGreg Clayton Process &process = m_opaque_sp->GetProcess(); 46830fdc8d8SChris Lattner // Why do we need to set the current thread by ID here??? 4692976d00aSJim Ingham process.GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID()); 4705d5028b5SGreg Clayton Error error (process.Resume()); 4715d5028b5SGreg Clayton if (error.Success()) 4725d5028b5SGreg Clayton { 4735d5028b5SGreg Clayton // If we are doing synchronous mode, then wait for the 4745d5028b5SGreg Clayton // process to stop yet again! 4755d5028b5SGreg Clayton if (process.GetTarget().GetDebugger().GetAsyncExecution () == false) 4765d5028b5SGreg Clayton process.WaitForProcessToStop (NULL); 4775d5028b5SGreg Clayton } 47830fdc8d8SChris Lattner } 47930fdc8d8SChris Lattner } 48030fdc8d8SChris Lattner 48130fdc8d8SChris Lattner void 48230fdc8d8SChris Lattner SBThread::StepOut () 48330fdc8d8SChris Lattner { 4842d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 485ceb6b139SCaroline Tice 486ceb6b139SCaroline Tice if (log) 4874838131bSGreg Clayton log->Printf ("SBThread(%p)::StepOut ()", m_opaque_sp.get()); 488ceb6b139SCaroline Tice 4896611103cSGreg Clayton if (m_opaque_sp) 49030fdc8d8SChris Lattner { 491af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 49230fdc8d8SChris Lattner bool abort_other_plans = true; 49330fdc8d8SChris Lattner bool stop_other_threads = true; 49430fdc8d8SChris Lattner 495481cef25SGreg Clayton m_opaque_sp->QueueThreadPlanForStepOut (abort_other_plans, 496481cef25SGreg Clayton NULL, 497481cef25SGreg Clayton false, 498481cef25SGreg Clayton stop_other_threads, 499481cef25SGreg Clayton eVoteYes, 500481cef25SGreg Clayton eVoteNoOpinion, 501481cef25SGreg Clayton 0); 502481cef25SGreg Clayton 503481cef25SGreg Clayton Process &process = m_opaque_sp->GetProcess(); 504481cef25SGreg Clayton process.GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID()); 505481cef25SGreg Clayton Error error (process.Resume()); 506481cef25SGreg Clayton if (error.Success()) 507481cef25SGreg Clayton { 508481cef25SGreg Clayton // If we are doing synchronous mode, then wait for the 509481cef25SGreg Clayton // process to stop yet again! 510481cef25SGreg Clayton if (process.GetTarget().GetDebugger().GetAsyncExecution () == false) 511481cef25SGreg Clayton process.WaitForProcessToStop (NULL); 512481cef25SGreg Clayton } 513481cef25SGreg Clayton } 514481cef25SGreg Clayton } 515481cef25SGreg Clayton 516481cef25SGreg Clayton void 517481cef25SGreg Clayton SBThread::StepOutOfFrame (lldb::SBFrame &sb_frame) 518481cef25SGreg Clayton { 519481cef25SGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 520481cef25SGreg Clayton 521481cef25SGreg Clayton if (log) 522481cef25SGreg Clayton { 523481cef25SGreg Clayton SBStream frame_desc_strm; 524481cef25SGreg Clayton sb_frame.GetDescription (frame_desc_strm); 525481cef25SGreg Clayton log->Printf ("SBThread(%p)::StepOutOfFrame (frame = SBFrame(%p): %s)", m_opaque_sp.get(), sb_frame.get(), frame_desc_strm.GetData()); 526481cef25SGreg Clayton } 527481cef25SGreg Clayton 528481cef25SGreg Clayton if (m_opaque_sp) 529481cef25SGreg Clayton { 530481cef25SGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 531481cef25SGreg Clayton bool abort_other_plans = true; 532481cef25SGreg Clayton bool stop_other_threads = true; 533481cef25SGreg Clayton 534481cef25SGreg Clayton m_opaque_sp->QueueThreadPlanForStepOut (abort_other_plans, 535481cef25SGreg Clayton NULL, 536481cef25SGreg Clayton false, 537481cef25SGreg Clayton stop_other_threads, 538481cef25SGreg Clayton eVoteYes, 539481cef25SGreg Clayton eVoteNoOpinion, 540481cef25SGreg Clayton sb_frame->GetFrameIndex()); 54130fdc8d8SChris Lattner 5426611103cSGreg Clayton Process &process = m_opaque_sp->GetProcess(); 5432976d00aSJim Ingham process.GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID()); 5445d5028b5SGreg Clayton Error error (process.Resume()); 5455d5028b5SGreg Clayton if (error.Success()) 5465d5028b5SGreg Clayton { 5475d5028b5SGreg Clayton // If we are doing synchronous mode, then wait for the 5485d5028b5SGreg Clayton // process to stop yet again! 5495d5028b5SGreg Clayton if (process.GetTarget().GetDebugger().GetAsyncExecution () == false) 5505d5028b5SGreg Clayton process.WaitForProcessToStop (NULL); 5515d5028b5SGreg Clayton } 55230fdc8d8SChris Lattner } 55330fdc8d8SChris Lattner } 55430fdc8d8SChris Lattner 55530fdc8d8SChris Lattner void 55630fdc8d8SChris Lattner SBThread::StepInstruction (bool step_over) 55730fdc8d8SChris Lattner { 5582d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 559ceb6b139SCaroline Tice 560ceb6b139SCaroline Tice if (log) 56193aa84e8SGreg Clayton log->Printf ("SBThread(%p)::StepInstruction (step_over=%i)", m_opaque_sp.get(), step_over); 562ceb6b139SCaroline Tice 5636611103cSGreg Clayton if (m_opaque_sp) 56430fdc8d8SChris Lattner { 565af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 5666611103cSGreg Clayton m_opaque_sp->QueueThreadPlanForStepSingleInstruction (step_over, true, true); 5676611103cSGreg Clayton Process &process = m_opaque_sp->GetProcess(); 5682976d00aSJim Ingham process.GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID()); 5695d5028b5SGreg Clayton Error error (process.Resume()); 5705d5028b5SGreg Clayton if (error.Success()) 5715d5028b5SGreg Clayton { 5725d5028b5SGreg Clayton // If we are doing synchronous mode, then wait for the 5735d5028b5SGreg Clayton // process to stop yet again! 5745d5028b5SGreg Clayton if (process.GetTarget().GetDebugger().GetAsyncExecution () == false) 5755d5028b5SGreg Clayton process.WaitForProcessToStop (NULL); 5765d5028b5SGreg Clayton } 57730fdc8d8SChris Lattner } 57830fdc8d8SChris Lattner } 57930fdc8d8SChris Lattner 58030fdc8d8SChris Lattner void 58130fdc8d8SChris Lattner SBThread::RunToAddress (lldb::addr_t addr) 58230fdc8d8SChris Lattner { 5832d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 584ceb6b139SCaroline Tice 585ceb6b139SCaroline Tice if (log) 58693aa84e8SGreg Clayton log->Printf ("SBThread(%p)::RunToAddress (addr=0x%llx)", m_opaque_sp.get(), addr); 587ceb6b139SCaroline Tice 5886611103cSGreg Clayton if (m_opaque_sp) 58930fdc8d8SChris Lattner { 59030fdc8d8SChris Lattner bool abort_other_plans = true; 59130fdc8d8SChris Lattner bool stop_other_threads = true; 59230fdc8d8SChris Lattner 59330fdc8d8SChris Lattner Address target_addr (NULL, addr); 59430fdc8d8SChris Lattner 5956611103cSGreg Clayton m_opaque_sp->QueueThreadPlanForRunToAddress (abort_other_plans, target_addr, stop_other_threads); 5966611103cSGreg Clayton Process &process = m_opaque_sp->GetProcess(); 5972976d00aSJim Ingham process.GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID()); 5985d5028b5SGreg Clayton Error error (process.Resume()); 5995d5028b5SGreg Clayton if (error.Success()) 6005d5028b5SGreg Clayton { 6015d5028b5SGreg Clayton // If we are doing synchronous mode, then wait for the 6025d5028b5SGreg Clayton // process to stop yet again! 6035d5028b5SGreg Clayton if (process.GetTarget().GetDebugger().GetAsyncExecution () == false) 6045d5028b5SGreg Clayton process.WaitForProcessToStop (NULL); 6055d5028b5SGreg Clayton } 60630fdc8d8SChris Lattner } 60730fdc8d8SChris Lattner } 60830fdc8d8SChris Lattner 609481cef25SGreg Clayton SBError 610481cef25SGreg Clayton SBThread::StepOverUntil (lldb::SBFrame &sb_frame, 611481cef25SGreg Clayton lldb::SBFileSpec &sb_file_spec, 612481cef25SGreg Clayton uint32_t line) 613481cef25SGreg Clayton { 614481cef25SGreg Clayton SBError sb_error; 615481cef25SGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 616481cef25SGreg Clayton char path[PATH_MAX]; 617481cef25SGreg Clayton 618481cef25SGreg Clayton if (log) 619481cef25SGreg Clayton { 620481cef25SGreg Clayton SBStream frame_desc_strm; 621481cef25SGreg Clayton sb_frame.GetDescription (frame_desc_strm); 622481cef25SGreg Clayton sb_file_spec->GetPath (path, sizeof(path)); 623481cef25SGreg Clayton log->Printf ("SBThread(%p)::StepOverUntil (frame = SBFrame(%p): %s, file+line = %s:%u)", 624481cef25SGreg Clayton m_opaque_sp.get(), 625481cef25SGreg Clayton sb_frame.get(), 626481cef25SGreg Clayton frame_desc_strm.GetData(), 627481cef25SGreg Clayton path, line); 628481cef25SGreg Clayton } 629481cef25SGreg Clayton 630481cef25SGreg Clayton if (m_opaque_sp) 631481cef25SGreg Clayton { 632481cef25SGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 633481cef25SGreg Clayton 634481cef25SGreg Clayton if (line == 0) 635481cef25SGreg Clayton { 636481cef25SGreg Clayton sb_error.SetErrorString("invalid line argument"); 637481cef25SGreg Clayton return sb_error; 638481cef25SGreg Clayton } 639481cef25SGreg Clayton 640481cef25SGreg Clayton StackFrameSP frame_sp; 641481cef25SGreg Clayton if (sb_frame.IsValid()) 642481cef25SGreg Clayton frame_sp = sb_frame.get_sp(); 643481cef25SGreg Clayton else 644481cef25SGreg Clayton { 645481cef25SGreg Clayton frame_sp = m_opaque_sp->GetSelectedFrame (); 646481cef25SGreg Clayton if (!frame_sp) 647481cef25SGreg Clayton frame_sp = m_opaque_sp->GetStackFrameAtIndex (0); 648481cef25SGreg Clayton } 649481cef25SGreg Clayton 650481cef25SGreg Clayton SymbolContext frame_sc; 651481cef25SGreg Clayton if (!frame_sp) 652481cef25SGreg Clayton { 653481cef25SGreg Clayton sb_error.SetErrorString("no valid frames in thread to step"); 654481cef25SGreg Clayton return sb_error; 655481cef25SGreg Clayton } 656481cef25SGreg Clayton 657481cef25SGreg Clayton // If we have a frame, get its line 658481cef25SGreg Clayton frame_sc = frame_sp->GetSymbolContext (eSymbolContextCompUnit | 659481cef25SGreg Clayton eSymbolContextFunction | 660481cef25SGreg Clayton eSymbolContextLineEntry | 661481cef25SGreg Clayton eSymbolContextSymbol ); 662481cef25SGreg Clayton 663481cef25SGreg Clayton if (frame_sc.comp_unit == NULL) 664481cef25SGreg Clayton { 665481cef25SGreg Clayton sb_error.SetErrorStringWithFormat("frame %u doesn't have debug information", frame_sp->GetFrameIndex()); 666481cef25SGreg Clayton return sb_error; 667481cef25SGreg Clayton } 668481cef25SGreg Clayton 669481cef25SGreg Clayton FileSpec step_file_spec; 670481cef25SGreg Clayton if (sb_file_spec.IsValid()) 671481cef25SGreg Clayton { 672481cef25SGreg Clayton // The file spec passed in was valid, so use it 673481cef25SGreg Clayton step_file_spec = sb_file_spec.ref(); 674481cef25SGreg Clayton } 675481cef25SGreg Clayton else 676481cef25SGreg Clayton { 677481cef25SGreg Clayton if (frame_sc.line_entry.IsValid()) 678481cef25SGreg Clayton step_file_spec = frame_sc.line_entry.file; 679481cef25SGreg Clayton else 680481cef25SGreg Clayton { 681481cef25SGreg Clayton sb_error.SetErrorString("invalid file argument or no file for frame"); 682481cef25SGreg Clayton return sb_error; 683481cef25SGreg Clayton } 684481cef25SGreg Clayton } 685481cef25SGreg Clayton 6869b70ddb3SJim Ingham // Grab the current function, then we will make sure the "until" address is 6879b70ddb3SJim Ingham // within the function. We discard addresses that are out of the current 6889b70ddb3SJim Ingham // function, and then if there are no addresses remaining, give an appropriate 6899b70ddb3SJim Ingham // error message. 6909b70ddb3SJim Ingham 6919b70ddb3SJim Ingham bool all_in_function = true; 6929b70ddb3SJim Ingham AddressRange fun_range = frame_sc.function->GetAddressRange(); 6939b70ddb3SJim Ingham 694481cef25SGreg Clayton std::vector<addr_t> step_over_until_addrs; 695481cef25SGreg Clayton const bool abort_other_plans = true; 696481cef25SGreg Clayton const bool stop_other_threads = true; 697481cef25SGreg Clayton const bool check_inlines = true; 698481cef25SGreg Clayton const bool exact = false; 6999b70ddb3SJim Ingham Target *target = &m_opaque_sp->GetProcess().GetTarget(); 700481cef25SGreg Clayton 701481cef25SGreg Clayton SymbolContextList sc_list; 7029b70ddb3SJim Ingham const uint32_t num_matches = frame_sc.comp_unit->ResolveSymbolContext (step_file_spec, 7039b70ddb3SJim Ingham line, 7049b70ddb3SJim Ingham check_inlines, 7059b70ddb3SJim Ingham exact, 7069b70ddb3SJim Ingham eSymbolContextLineEntry, 7079b70ddb3SJim Ingham sc_list); 708481cef25SGreg Clayton if (num_matches > 0) 709481cef25SGreg Clayton { 710481cef25SGreg Clayton SymbolContext sc; 711481cef25SGreg Clayton for (uint32_t i=0; i<num_matches; ++i) 712481cef25SGreg Clayton { 713481cef25SGreg Clayton if (sc_list.GetContextAtIndex(i, sc)) 714481cef25SGreg Clayton { 7159b70ddb3SJim Ingham addr_t step_addr = sc.line_entry.range.GetBaseAddress().GetLoadAddress(target); 716481cef25SGreg Clayton if (step_addr != LLDB_INVALID_ADDRESS) 717481cef25SGreg Clayton { 7189b70ddb3SJim Ingham if (fun_range.ContainsLoadAddress(step_addr, target)) 719481cef25SGreg Clayton step_over_until_addrs.push_back(step_addr); 7209b70ddb3SJim Ingham else 7219b70ddb3SJim Ingham all_in_function = false; 722481cef25SGreg Clayton } 723481cef25SGreg Clayton } 724481cef25SGreg Clayton } 725481cef25SGreg Clayton } 726481cef25SGreg Clayton 727481cef25SGreg Clayton if (step_over_until_addrs.empty()) 728481cef25SGreg Clayton { 7299b70ddb3SJim Ingham if (all_in_function) 7309b70ddb3SJim Ingham { 731481cef25SGreg Clayton step_file_spec.GetPath (path, sizeof(path)); 732fd54b368SJason Molenda sb_error.SetErrorStringWithFormat("No line entries for %s:%u", path, line); 733481cef25SGreg Clayton } 734481cef25SGreg Clayton else 73586edbf41SGreg Clayton sb_error.SetErrorString ("step until target not in current function"); 7369b70ddb3SJim Ingham } 7379b70ddb3SJim Ingham else 738481cef25SGreg Clayton { 739481cef25SGreg Clayton m_opaque_sp->QueueThreadPlanForStepUntil (abort_other_plans, 740481cef25SGreg Clayton &step_over_until_addrs[0], 741481cef25SGreg Clayton step_over_until_addrs.size(), 742481cef25SGreg Clayton stop_other_threads, 743481cef25SGreg Clayton frame_sp->GetFrameIndex()); 744481cef25SGreg Clayton 745481cef25SGreg Clayton m_opaque_sp->GetProcess().GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID()); 746481cef25SGreg Clayton sb_error.ref() = m_opaque_sp->GetProcess().Resume(); 747481cef25SGreg Clayton if (sb_error->Success()) 748481cef25SGreg Clayton { 749481cef25SGreg Clayton // If we are doing synchronous mode, then wait for the 750481cef25SGreg Clayton // process to stop yet again! 751481cef25SGreg Clayton if (m_opaque_sp->GetProcess().GetTarget().GetDebugger().GetAsyncExecution () == false) 752481cef25SGreg Clayton m_opaque_sp->GetProcess().WaitForProcessToStop (NULL); 753481cef25SGreg Clayton } 754481cef25SGreg Clayton } 755481cef25SGreg Clayton } 756481cef25SGreg Clayton else 757481cef25SGreg Clayton { 758481cef25SGreg Clayton sb_error.SetErrorString("this SBThread object is invalid"); 759481cef25SGreg Clayton } 760481cef25SGreg Clayton return sb_error; 761481cef25SGreg Clayton } 762481cef25SGreg Clayton 763481cef25SGreg Clayton 764722a0cdcSGreg Clayton bool 765722a0cdcSGreg Clayton SBThread::Suspend() 766722a0cdcSGreg Clayton { 767722a0cdcSGreg Clayton if (m_opaque_sp) 768722a0cdcSGreg Clayton { 769722a0cdcSGreg Clayton m_opaque_sp->SetResumeState (eStateSuspended); 770722a0cdcSGreg Clayton return true; 771722a0cdcSGreg Clayton } 772722a0cdcSGreg Clayton return false; 773722a0cdcSGreg Clayton } 774722a0cdcSGreg Clayton 775722a0cdcSGreg Clayton bool 776722a0cdcSGreg Clayton SBThread::Resume () 777722a0cdcSGreg Clayton { 778722a0cdcSGreg Clayton if (m_opaque_sp) 779722a0cdcSGreg Clayton { 780722a0cdcSGreg Clayton m_opaque_sp->SetResumeState (eStateRunning); 781722a0cdcSGreg Clayton return true; 782722a0cdcSGreg Clayton } 783722a0cdcSGreg Clayton return false; 784722a0cdcSGreg Clayton } 785722a0cdcSGreg Clayton 786722a0cdcSGreg Clayton bool 787722a0cdcSGreg Clayton SBThread::IsSuspended() 788722a0cdcSGreg Clayton { 789722a0cdcSGreg Clayton if (m_opaque_sp) 7909566143fSGreg Clayton return m_opaque_sp->GetResumeState () == eStateSuspended; 791722a0cdcSGreg Clayton return false; 792722a0cdcSGreg Clayton } 793722a0cdcSGreg Clayton 79430fdc8d8SChris Lattner SBProcess 79530fdc8d8SChris Lattner SBThread::GetProcess () 79630fdc8d8SChris Lattner { 797ceb6b139SCaroline Tice 79830fdc8d8SChris Lattner SBProcess process; 7996611103cSGreg Clayton if (m_opaque_sp) 80030fdc8d8SChris Lattner { 80130fdc8d8SChris Lattner // Have to go up to the target so we can get a shared pointer to our process... 8026611103cSGreg Clayton process.SetProcess(m_opaque_sp->GetProcess().GetTarget().GetProcessSP()); 80330fdc8d8SChris Lattner } 804ceb6b139SCaroline Tice 8052d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 806ceb6b139SCaroline Tice if (log) 807ceb6b139SCaroline Tice { 808481cef25SGreg Clayton SBStream frame_desc_strm; 809481cef25SGreg Clayton process.GetDescription (frame_desc_strm); 8104838131bSGreg Clayton log->Printf ("SBThread(%p)::GetProcess () => SBProcess(%p): %s", m_opaque_sp.get(), 811481cef25SGreg Clayton process.get(), frame_desc_strm.GetData()); 812ceb6b139SCaroline Tice } 813ceb6b139SCaroline Tice 81430fdc8d8SChris Lattner return process; 81530fdc8d8SChris Lattner } 81630fdc8d8SChris Lattner 81730fdc8d8SChris Lattner uint32_t 81830fdc8d8SChris Lattner SBThread::GetNumFrames () 81930fdc8d8SChris Lattner { 8202d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 821ceb6b139SCaroline Tice 822ceb6b139SCaroline Tice uint32_t num_frames = 0; 8236611103cSGreg Clayton if (m_opaque_sp) 824af67cecdSGreg Clayton { 825af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 826ceb6b139SCaroline Tice num_frames = m_opaque_sp->GetStackFrameCount(); 827af67cecdSGreg Clayton } 828ceb6b139SCaroline Tice 829ceb6b139SCaroline Tice if (log) 8304838131bSGreg Clayton log->Printf ("SBThread(%p)::GetNumFrames () => %u", m_opaque_sp.get(), num_frames); 831ceb6b139SCaroline Tice 832ceb6b139SCaroline Tice return num_frames; 83330fdc8d8SChris Lattner } 83430fdc8d8SChris Lattner 83530fdc8d8SChris Lattner SBFrame 83630fdc8d8SChris Lattner SBThread::GetFrameAtIndex (uint32_t idx) 83730fdc8d8SChris Lattner { 8382d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 839ceb6b139SCaroline Tice 84030fdc8d8SChris Lattner SBFrame sb_frame; 8416611103cSGreg Clayton if (m_opaque_sp) 842af67cecdSGreg Clayton { 843af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 8446611103cSGreg Clayton sb_frame.SetFrame (m_opaque_sp->GetStackFrameAtIndex (idx)); 845af67cecdSGreg Clayton } 846ceb6b139SCaroline Tice 847ceb6b139SCaroline Tice if (log) 848ceb6b139SCaroline Tice { 849481cef25SGreg Clayton SBStream frame_desc_strm; 850481cef25SGreg Clayton sb_frame.GetDescription (frame_desc_strm); 8514838131bSGreg Clayton log->Printf ("SBThread(%p)::GetFrameAtIndex (idx=%d) => SBFrame(%p): %s", 852481cef25SGreg Clayton m_opaque_sp.get(), idx, sb_frame.get(), frame_desc_strm.GetData()); 853ceb6b139SCaroline Tice } 854ceb6b139SCaroline Tice 85530fdc8d8SChris Lattner return sb_frame; 85630fdc8d8SChris Lattner } 85730fdc8d8SChris Lattner 858f028a1fbSGreg Clayton lldb::SBFrame 859f028a1fbSGreg Clayton SBThread::GetSelectedFrame () 860f028a1fbSGreg Clayton { 861f028a1fbSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 862f028a1fbSGreg Clayton 863f028a1fbSGreg Clayton SBFrame sb_frame; 864f028a1fbSGreg Clayton if (m_opaque_sp) 865af67cecdSGreg Clayton { 866af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 867f028a1fbSGreg Clayton sb_frame.SetFrame (m_opaque_sp->GetSelectedFrame ()); 868af67cecdSGreg Clayton } 869f028a1fbSGreg Clayton 870f028a1fbSGreg Clayton if (log) 871f028a1fbSGreg Clayton { 872481cef25SGreg Clayton SBStream frame_desc_strm; 873481cef25SGreg Clayton sb_frame.GetDescription (frame_desc_strm); 874f028a1fbSGreg Clayton log->Printf ("SBThread(%p)::GetSelectedFrame () => SBFrame(%p): %s", 875481cef25SGreg Clayton m_opaque_sp.get(), sb_frame.get(), frame_desc_strm.GetData()); 876f028a1fbSGreg Clayton } 877f028a1fbSGreg Clayton 878f028a1fbSGreg Clayton return sb_frame; 879f028a1fbSGreg Clayton } 880f028a1fbSGreg Clayton 881f028a1fbSGreg Clayton lldb::SBFrame 882f028a1fbSGreg Clayton SBThread::SetSelectedFrame (uint32_t idx) 883f028a1fbSGreg Clayton { 884f028a1fbSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); 885f028a1fbSGreg Clayton 886f028a1fbSGreg Clayton SBFrame sb_frame; 887f028a1fbSGreg Clayton if (m_opaque_sp) 888f028a1fbSGreg Clayton { 889af67cecdSGreg Clayton Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex()); 890f028a1fbSGreg Clayton lldb::StackFrameSP frame_sp (m_opaque_sp->GetStackFrameAtIndex (idx)); 891f028a1fbSGreg Clayton if (frame_sp) 892f028a1fbSGreg Clayton { 893f028a1fbSGreg Clayton m_opaque_sp->SetSelectedFrame (frame_sp.get()); 894f028a1fbSGreg Clayton sb_frame.SetFrame (frame_sp); 895f028a1fbSGreg Clayton } 896f028a1fbSGreg Clayton } 897f028a1fbSGreg Clayton 898f028a1fbSGreg Clayton if (log) 899f028a1fbSGreg Clayton { 900481cef25SGreg Clayton SBStream frame_desc_strm; 901481cef25SGreg Clayton sb_frame.GetDescription (frame_desc_strm); 902f028a1fbSGreg Clayton log->Printf ("SBThread(%p)::SetSelectedFrame (idx=%u) => SBFrame(%p): %s", 903481cef25SGreg Clayton m_opaque_sp.get(), idx, sb_frame.get(), frame_desc_strm.GetData()); 904f028a1fbSGreg Clayton } 905f028a1fbSGreg Clayton return sb_frame; 906f028a1fbSGreg Clayton } 907f028a1fbSGreg Clayton 908f028a1fbSGreg Clayton 90930fdc8d8SChris Lattner bool 91030fdc8d8SChris Lattner SBThread::operator == (const SBThread &rhs) const 91130fdc8d8SChris Lattner { 9126611103cSGreg Clayton return m_opaque_sp.get() == rhs.m_opaque_sp.get(); 91330fdc8d8SChris Lattner } 91430fdc8d8SChris Lattner 91530fdc8d8SChris Lattner bool 91630fdc8d8SChris Lattner SBThread::operator != (const SBThread &rhs) const 91730fdc8d8SChris Lattner { 9186611103cSGreg Clayton return m_opaque_sp.get() != rhs.m_opaque_sp.get(); 91930fdc8d8SChris Lattner } 92030fdc8d8SChris Lattner 92130fdc8d8SChris Lattner lldb_private::Thread * 9224838131bSGreg Clayton SBThread::get () 92330fdc8d8SChris Lattner { 9246611103cSGreg Clayton return m_opaque_sp.get(); 92530fdc8d8SChris Lattner } 92630fdc8d8SChris Lattner 92730fdc8d8SChris Lattner const lldb_private::Thread * 92830fdc8d8SChris Lattner SBThread::operator->() const 92930fdc8d8SChris Lattner { 9306611103cSGreg Clayton return m_opaque_sp.get(); 93130fdc8d8SChris Lattner } 93230fdc8d8SChris Lattner 93330fdc8d8SChris Lattner const lldb_private::Thread & 93430fdc8d8SChris Lattner SBThread::operator*() const 93530fdc8d8SChris Lattner { 9366611103cSGreg Clayton return *m_opaque_sp; 93730fdc8d8SChris Lattner } 93830fdc8d8SChris Lattner 93930fdc8d8SChris Lattner lldb_private::Thread * 94030fdc8d8SChris Lattner SBThread::operator->() 94130fdc8d8SChris Lattner { 9426611103cSGreg Clayton return m_opaque_sp.get(); 94330fdc8d8SChris Lattner } 94430fdc8d8SChris Lattner 94530fdc8d8SChris Lattner lldb_private::Thread & 94630fdc8d8SChris Lattner SBThread::operator*() 94730fdc8d8SChris Lattner { 9486611103cSGreg Clayton return *m_opaque_sp; 94930fdc8d8SChris Lattner } 950dde9cff3SCaroline Tice 951dde9cff3SCaroline Tice bool 952ceb6b139SCaroline Tice SBThread::GetDescription (SBStream &description) const 953ceb6b139SCaroline Tice { 954*da7bc7d0SGreg Clayton Stream &strm = description.ref(); 955*da7bc7d0SGreg Clayton 956ceb6b139SCaroline Tice if (m_opaque_sp) 957ceb6b139SCaroline Tice { 958*da7bc7d0SGreg Clayton strm.Printf("SBThread: tid = 0x%4.4llx", m_opaque_sp->GetID()); 959ceb6b139SCaroline Tice } 960ceb6b139SCaroline Tice else 961*da7bc7d0SGreg Clayton strm.PutCString ("No value"); 962ceb6b139SCaroline Tice 963ceb6b139SCaroline Tice return true; 964ceb6b139SCaroline Tice } 965