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"
354c5de699SEli Friedman #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 {
826611103cSGreg Clayton     return m_opaque_sp != NULL;
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 
495*481cef25SGreg Clayton         m_opaque_sp->QueueThreadPlanForStepOut (abort_other_plans,
496*481cef25SGreg Clayton                                                 NULL,
497*481cef25SGreg Clayton                                                 false,
498*481cef25SGreg Clayton                                                 stop_other_threads,
499*481cef25SGreg Clayton                                                 eVoteYes,
500*481cef25SGreg Clayton                                                 eVoteNoOpinion,
501*481cef25SGreg Clayton                                                 0);
502*481cef25SGreg Clayton 
503*481cef25SGreg Clayton         Process &process = m_opaque_sp->GetProcess();
504*481cef25SGreg Clayton         process.GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID());
505*481cef25SGreg Clayton         Error error (process.Resume());
506*481cef25SGreg Clayton         if (error.Success())
507*481cef25SGreg Clayton         {
508*481cef25SGreg Clayton             // If we are doing synchronous mode, then wait for the
509*481cef25SGreg Clayton             // process to stop yet again!
510*481cef25SGreg Clayton             if (process.GetTarget().GetDebugger().GetAsyncExecution () == false)
511*481cef25SGreg Clayton                 process.WaitForProcessToStop (NULL);
512*481cef25SGreg Clayton         }
513*481cef25SGreg Clayton     }
514*481cef25SGreg Clayton }
515*481cef25SGreg Clayton 
516*481cef25SGreg Clayton void
517*481cef25SGreg Clayton SBThread::StepOutOfFrame (lldb::SBFrame &sb_frame)
518*481cef25SGreg Clayton {
519*481cef25SGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));
520*481cef25SGreg Clayton 
521*481cef25SGreg Clayton     if (log)
522*481cef25SGreg Clayton     {
523*481cef25SGreg Clayton         SBStream frame_desc_strm;
524*481cef25SGreg Clayton         sb_frame.GetDescription (frame_desc_strm);
525*481cef25SGreg Clayton         log->Printf ("SBThread(%p)::StepOutOfFrame (frame = SBFrame(%p): %s)", m_opaque_sp.get(), sb_frame.get(), frame_desc_strm.GetData());
526*481cef25SGreg Clayton     }
527*481cef25SGreg Clayton 
528*481cef25SGreg Clayton     if (m_opaque_sp)
529*481cef25SGreg Clayton     {
530*481cef25SGreg Clayton         Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex());
531*481cef25SGreg Clayton         bool abort_other_plans = true;
532*481cef25SGreg Clayton         bool stop_other_threads = true;
533*481cef25SGreg Clayton 
534*481cef25SGreg Clayton         m_opaque_sp->QueueThreadPlanForStepOut (abort_other_plans,
535*481cef25SGreg Clayton                                                 NULL,
536*481cef25SGreg Clayton                                                 false,
537*481cef25SGreg Clayton                                                 stop_other_threads,
538*481cef25SGreg Clayton                                                 eVoteYes,
539*481cef25SGreg Clayton                                                 eVoteNoOpinion,
540*481cef25SGreg 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 
609*481cef25SGreg Clayton SBError
610*481cef25SGreg Clayton SBThread::StepOverUntil (lldb::SBFrame &sb_frame,
611*481cef25SGreg Clayton                          lldb::SBFileSpec &sb_file_spec,
612*481cef25SGreg Clayton                          uint32_t line)
613*481cef25SGreg Clayton {
614*481cef25SGreg Clayton     SBError sb_error;
615*481cef25SGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));
616*481cef25SGreg Clayton     char path[PATH_MAX];
617*481cef25SGreg Clayton 
618*481cef25SGreg Clayton     if (log)
619*481cef25SGreg Clayton     {
620*481cef25SGreg Clayton         SBStream frame_desc_strm;
621*481cef25SGreg Clayton         sb_frame.GetDescription (frame_desc_strm);
622*481cef25SGreg Clayton         sb_file_spec->GetPath (path, sizeof(path));
623*481cef25SGreg Clayton         log->Printf ("SBThread(%p)::StepOverUntil (frame = SBFrame(%p): %s, file+line = %s:%u)",
624*481cef25SGreg Clayton                      m_opaque_sp.get(),
625*481cef25SGreg Clayton                      sb_frame.get(),
626*481cef25SGreg Clayton                      frame_desc_strm.GetData(),
627*481cef25SGreg Clayton                      path, line);
628*481cef25SGreg Clayton     }
629*481cef25SGreg Clayton 
630*481cef25SGreg Clayton     if (m_opaque_sp)
631*481cef25SGreg Clayton     {
632*481cef25SGreg Clayton         Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex());
633*481cef25SGreg Clayton 
634*481cef25SGreg Clayton         if (line == 0)
635*481cef25SGreg Clayton         {
636*481cef25SGreg Clayton             sb_error.SetErrorString("invalid line argument");
637*481cef25SGreg Clayton             return sb_error;
638*481cef25SGreg Clayton         }
639*481cef25SGreg Clayton 
640*481cef25SGreg Clayton         StackFrameSP frame_sp;
641*481cef25SGreg Clayton         if (sb_frame.IsValid())
642*481cef25SGreg Clayton             frame_sp = sb_frame.get_sp();
643*481cef25SGreg Clayton         else
644*481cef25SGreg Clayton         {
645*481cef25SGreg Clayton             frame_sp = m_opaque_sp->GetSelectedFrame ();
646*481cef25SGreg Clayton             if (!frame_sp)
647*481cef25SGreg Clayton                 frame_sp = m_opaque_sp->GetStackFrameAtIndex (0);
648*481cef25SGreg Clayton         }
649*481cef25SGreg Clayton 
650*481cef25SGreg Clayton         SymbolContext frame_sc;
651*481cef25SGreg Clayton         if (!frame_sp)
652*481cef25SGreg Clayton         {
653*481cef25SGreg Clayton             sb_error.SetErrorString("no valid frames in thread to step");
654*481cef25SGreg Clayton             return sb_error;
655*481cef25SGreg Clayton         }
656*481cef25SGreg Clayton 
657*481cef25SGreg Clayton         // If we have a frame, get its line
658*481cef25SGreg Clayton         frame_sc = frame_sp->GetSymbolContext (eSymbolContextCompUnit  |
659*481cef25SGreg Clayton                                                eSymbolContextFunction  |
660*481cef25SGreg Clayton                                                eSymbolContextLineEntry |
661*481cef25SGreg Clayton                                                eSymbolContextSymbol    );
662*481cef25SGreg Clayton 
663*481cef25SGreg Clayton         if (frame_sc.comp_unit == NULL)
664*481cef25SGreg Clayton         {
665*481cef25SGreg Clayton             sb_error.SetErrorStringWithFormat("frame %u doesn't have debug information", frame_sp->GetFrameIndex());
666*481cef25SGreg Clayton             return sb_error;
667*481cef25SGreg Clayton         }
668*481cef25SGreg Clayton 
669*481cef25SGreg Clayton         FileSpec step_file_spec;
670*481cef25SGreg Clayton         if (sb_file_spec.IsValid())
671*481cef25SGreg Clayton         {
672*481cef25SGreg Clayton             // The file spec passed in was valid, so use it
673*481cef25SGreg Clayton             step_file_spec = sb_file_spec.ref();
674*481cef25SGreg Clayton         }
675*481cef25SGreg Clayton         else
676*481cef25SGreg Clayton         {
677*481cef25SGreg Clayton             if (frame_sc.line_entry.IsValid())
678*481cef25SGreg Clayton                 step_file_spec = frame_sc.line_entry.file;
679*481cef25SGreg Clayton             else
680*481cef25SGreg Clayton             {
681*481cef25SGreg Clayton                 sb_error.SetErrorString("invalid file argument or no file for frame");
682*481cef25SGreg Clayton                 return sb_error;
683*481cef25SGreg Clayton             }
684*481cef25SGreg Clayton         }
685*481cef25SGreg Clayton 
686*481cef25SGreg Clayton         std::vector<addr_t> step_over_until_addrs;
687*481cef25SGreg Clayton         const bool abort_other_plans = true;
688*481cef25SGreg Clayton         const bool stop_other_threads = true;
689*481cef25SGreg Clayton         const bool check_inlines = true;
690*481cef25SGreg Clayton         const bool exact = false;
691*481cef25SGreg Clayton 
692*481cef25SGreg Clayton         SymbolContextList sc_list;
693*481cef25SGreg Clayton         const uint32_t num_matches = frame_sc.comp_unit->ResolveSymbolContext (step_file_spec, line, check_inlines, exact, eSymbolContextLineEntry, sc_list);
694*481cef25SGreg Clayton         if (num_matches > 0)
695*481cef25SGreg Clayton         {
696*481cef25SGreg Clayton             SymbolContext sc;
697*481cef25SGreg Clayton             for (uint32_t i=0; i<num_matches; ++i)
698*481cef25SGreg Clayton             {
699*481cef25SGreg Clayton                 if (sc_list.GetContextAtIndex(i, sc))
700*481cef25SGreg Clayton                 {
701*481cef25SGreg Clayton                     addr_t step_addr = sc.line_entry.range.GetBaseAddress().GetLoadAddress(&m_opaque_sp->GetProcess().GetTarget());
702*481cef25SGreg Clayton                     if (step_addr != LLDB_INVALID_ADDRESS)
703*481cef25SGreg Clayton                     {
704*481cef25SGreg Clayton                         step_over_until_addrs.push_back(step_addr);
705*481cef25SGreg Clayton                     }
706*481cef25SGreg Clayton                 }
707*481cef25SGreg Clayton             }
708*481cef25SGreg Clayton         }
709*481cef25SGreg Clayton 
710*481cef25SGreg Clayton         if (step_over_until_addrs.empty())
711*481cef25SGreg Clayton         {
712*481cef25SGreg Clayton             step_file_spec.GetPath (path, sizeof(path));
713*481cef25SGreg Clayton             sb_error.SetErrorStringWithFormat("No line entries for %s:u", path, line);
714*481cef25SGreg Clayton         }
715*481cef25SGreg Clayton         else
716*481cef25SGreg Clayton         {
717*481cef25SGreg Clayton             m_opaque_sp->QueueThreadPlanForStepUntil (abort_other_plans,
718*481cef25SGreg Clayton                                                       &step_over_until_addrs[0],
719*481cef25SGreg Clayton                                                       step_over_until_addrs.size(),
720*481cef25SGreg Clayton                                                       stop_other_threads,
721*481cef25SGreg Clayton                                                       frame_sp->GetFrameIndex());
722*481cef25SGreg Clayton 
723*481cef25SGreg Clayton             m_opaque_sp->GetProcess().GetThreadList().SetSelectedThreadByID (m_opaque_sp->GetID());
724*481cef25SGreg Clayton             sb_error.ref() = m_opaque_sp->GetProcess().Resume();
725*481cef25SGreg Clayton             if (sb_error->Success())
726*481cef25SGreg Clayton             {
727*481cef25SGreg Clayton                 // If we are doing synchronous mode, then wait for the
728*481cef25SGreg Clayton                 // process to stop yet again!
729*481cef25SGreg Clayton                 if (m_opaque_sp->GetProcess().GetTarget().GetDebugger().GetAsyncExecution () == false)
730*481cef25SGreg Clayton                     m_opaque_sp->GetProcess().WaitForProcessToStop (NULL);
731*481cef25SGreg Clayton             }
732*481cef25SGreg Clayton         }
733*481cef25SGreg Clayton     }
734*481cef25SGreg Clayton     else
735*481cef25SGreg Clayton     {
736*481cef25SGreg Clayton         sb_error.SetErrorString("this SBThread object is invalid");
737*481cef25SGreg Clayton     }
738*481cef25SGreg Clayton     return sb_error;
739*481cef25SGreg Clayton }
740*481cef25SGreg Clayton 
741*481cef25SGreg Clayton 
742722a0cdcSGreg Clayton bool
743722a0cdcSGreg Clayton SBThread::Suspend()
744722a0cdcSGreg Clayton {
745722a0cdcSGreg Clayton     if (m_opaque_sp)
746722a0cdcSGreg Clayton     {
747722a0cdcSGreg Clayton         m_opaque_sp->SetResumeState (eStateSuspended);
748722a0cdcSGreg Clayton         return true;
749722a0cdcSGreg Clayton     }
750722a0cdcSGreg Clayton     return false;
751722a0cdcSGreg Clayton }
752722a0cdcSGreg Clayton 
753722a0cdcSGreg Clayton bool
754722a0cdcSGreg Clayton SBThread::Resume ()
755722a0cdcSGreg Clayton {
756722a0cdcSGreg Clayton     if (m_opaque_sp)
757722a0cdcSGreg Clayton     {
758722a0cdcSGreg Clayton         m_opaque_sp->SetResumeState (eStateRunning);
759722a0cdcSGreg Clayton         return true;
760722a0cdcSGreg Clayton     }
761722a0cdcSGreg Clayton     return false;
762722a0cdcSGreg Clayton }
763722a0cdcSGreg Clayton 
764722a0cdcSGreg Clayton bool
765722a0cdcSGreg Clayton SBThread::IsSuspended()
766722a0cdcSGreg Clayton {
767722a0cdcSGreg Clayton     if (m_opaque_sp)
7689566143fSGreg Clayton         return m_opaque_sp->GetResumeState () == eStateSuspended;
769722a0cdcSGreg Clayton     return false;
770722a0cdcSGreg Clayton }
771722a0cdcSGreg Clayton 
77230fdc8d8SChris Lattner SBProcess
77330fdc8d8SChris Lattner SBThread::GetProcess ()
77430fdc8d8SChris Lattner {
775ceb6b139SCaroline Tice 
77630fdc8d8SChris Lattner     SBProcess process;
7776611103cSGreg Clayton     if (m_opaque_sp)
77830fdc8d8SChris Lattner     {
77930fdc8d8SChris Lattner         // Have to go up to the target so we can get a shared pointer to our process...
7806611103cSGreg Clayton         process.SetProcess(m_opaque_sp->GetProcess().GetTarget().GetProcessSP());
78130fdc8d8SChris Lattner     }
782ceb6b139SCaroline Tice 
7832d4edfbcSGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));
784ceb6b139SCaroline Tice     if (log)
785ceb6b139SCaroline Tice     {
786*481cef25SGreg Clayton         SBStream frame_desc_strm;
787*481cef25SGreg Clayton         process.GetDescription (frame_desc_strm);
7884838131bSGreg Clayton         log->Printf ("SBThread(%p)::GetProcess () => SBProcess(%p): %s", m_opaque_sp.get(),
789*481cef25SGreg Clayton                      process.get(), frame_desc_strm.GetData());
790ceb6b139SCaroline Tice     }
791ceb6b139SCaroline Tice 
79230fdc8d8SChris Lattner     return process;
79330fdc8d8SChris Lattner }
79430fdc8d8SChris Lattner 
79530fdc8d8SChris Lattner uint32_t
79630fdc8d8SChris Lattner SBThread::GetNumFrames ()
79730fdc8d8SChris Lattner {
7982d4edfbcSGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));
799ceb6b139SCaroline Tice 
800ceb6b139SCaroline Tice     uint32_t num_frames = 0;
8016611103cSGreg Clayton     if (m_opaque_sp)
802af67cecdSGreg Clayton     {
803af67cecdSGreg Clayton         Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex());
804ceb6b139SCaroline Tice         num_frames = m_opaque_sp->GetStackFrameCount();
805af67cecdSGreg Clayton     }
806ceb6b139SCaroline Tice 
807ceb6b139SCaroline Tice     if (log)
8084838131bSGreg Clayton         log->Printf ("SBThread(%p)::GetNumFrames () => %u", m_opaque_sp.get(), num_frames);
809ceb6b139SCaroline Tice 
810ceb6b139SCaroline Tice     return num_frames;
81130fdc8d8SChris Lattner }
81230fdc8d8SChris Lattner 
81330fdc8d8SChris Lattner SBFrame
81430fdc8d8SChris Lattner SBThread::GetFrameAtIndex (uint32_t idx)
81530fdc8d8SChris Lattner {
8162d4edfbcSGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));
817ceb6b139SCaroline Tice 
81830fdc8d8SChris Lattner     SBFrame sb_frame;
8196611103cSGreg Clayton     if (m_opaque_sp)
820af67cecdSGreg Clayton     {
821af67cecdSGreg Clayton         Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex());
8226611103cSGreg Clayton         sb_frame.SetFrame (m_opaque_sp->GetStackFrameAtIndex (idx));
823af67cecdSGreg Clayton     }
824ceb6b139SCaroline Tice 
825ceb6b139SCaroline Tice     if (log)
826ceb6b139SCaroline Tice     {
827*481cef25SGreg Clayton         SBStream frame_desc_strm;
828*481cef25SGreg Clayton         sb_frame.GetDescription (frame_desc_strm);
8294838131bSGreg Clayton         log->Printf ("SBThread(%p)::GetFrameAtIndex (idx=%d) => SBFrame(%p): %s",
830*481cef25SGreg Clayton                      m_opaque_sp.get(), idx, sb_frame.get(), frame_desc_strm.GetData());
831ceb6b139SCaroline Tice     }
832ceb6b139SCaroline Tice 
83330fdc8d8SChris Lattner     return sb_frame;
83430fdc8d8SChris Lattner }
83530fdc8d8SChris Lattner 
836f028a1fbSGreg Clayton lldb::SBFrame
837f028a1fbSGreg Clayton SBThread::GetSelectedFrame ()
838f028a1fbSGreg Clayton {
839f028a1fbSGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));
840f028a1fbSGreg Clayton 
841f028a1fbSGreg Clayton     SBFrame sb_frame;
842f028a1fbSGreg Clayton     if (m_opaque_sp)
843af67cecdSGreg Clayton     {
844af67cecdSGreg Clayton         Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex());
845f028a1fbSGreg Clayton         sb_frame.SetFrame (m_opaque_sp->GetSelectedFrame ());
846af67cecdSGreg Clayton     }
847f028a1fbSGreg Clayton 
848f028a1fbSGreg Clayton     if (log)
849f028a1fbSGreg Clayton     {
850*481cef25SGreg Clayton         SBStream frame_desc_strm;
851*481cef25SGreg Clayton         sb_frame.GetDescription (frame_desc_strm);
852f028a1fbSGreg Clayton         log->Printf ("SBThread(%p)::GetSelectedFrame () => SBFrame(%p): %s",
853*481cef25SGreg Clayton                      m_opaque_sp.get(), sb_frame.get(), frame_desc_strm.GetData());
854f028a1fbSGreg Clayton     }
855f028a1fbSGreg Clayton 
856f028a1fbSGreg Clayton     return sb_frame;
857f028a1fbSGreg Clayton }
858f028a1fbSGreg Clayton 
859f028a1fbSGreg Clayton lldb::SBFrame
860f028a1fbSGreg Clayton SBThread::SetSelectedFrame (uint32_t idx)
861f028a1fbSGreg Clayton {
862f028a1fbSGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));
863f028a1fbSGreg Clayton 
864f028a1fbSGreg Clayton     SBFrame sb_frame;
865f028a1fbSGreg Clayton     if (m_opaque_sp)
866f028a1fbSGreg Clayton     {
867af67cecdSGreg Clayton         Mutex::Locker api_locker (m_opaque_sp->GetProcess().GetTarget().GetAPIMutex());
868f028a1fbSGreg Clayton         lldb::StackFrameSP frame_sp (m_opaque_sp->GetStackFrameAtIndex (idx));
869f028a1fbSGreg Clayton         if (frame_sp)
870f028a1fbSGreg Clayton         {
871f028a1fbSGreg Clayton             m_opaque_sp->SetSelectedFrame (frame_sp.get());
872f028a1fbSGreg Clayton             sb_frame.SetFrame (frame_sp);
873f028a1fbSGreg Clayton         }
874f028a1fbSGreg Clayton     }
875f028a1fbSGreg Clayton 
876f028a1fbSGreg Clayton     if (log)
877f028a1fbSGreg Clayton     {
878*481cef25SGreg Clayton         SBStream frame_desc_strm;
879*481cef25SGreg Clayton         sb_frame.GetDescription (frame_desc_strm);
880f028a1fbSGreg Clayton         log->Printf ("SBThread(%p)::SetSelectedFrame (idx=%u) => SBFrame(%p): %s",
881*481cef25SGreg Clayton                      m_opaque_sp.get(), idx, sb_frame.get(), frame_desc_strm.GetData());
882f028a1fbSGreg Clayton     }
883f028a1fbSGreg Clayton     return sb_frame;
884f028a1fbSGreg Clayton }
885f028a1fbSGreg Clayton 
886f028a1fbSGreg Clayton 
88730fdc8d8SChris Lattner bool
88830fdc8d8SChris Lattner SBThread::operator == (const SBThread &rhs) const
88930fdc8d8SChris Lattner {
8906611103cSGreg Clayton     return m_opaque_sp.get() == rhs.m_opaque_sp.get();
89130fdc8d8SChris Lattner }
89230fdc8d8SChris Lattner 
89330fdc8d8SChris Lattner bool
89430fdc8d8SChris Lattner SBThread::operator != (const SBThread &rhs) const
89530fdc8d8SChris Lattner {
8966611103cSGreg Clayton     return m_opaque_sp.get() != rhs.m_opaque_sp.get();
89730fdc8d8SChris Lattner }
89830fdc8d8SChris Lattner 
89930fdc8d8SChris Lattner lldb_private::Thread *
9004838131bSGreg Clayton SBThread::get ()
90130fdc8d8SChris Lattner {
9026611103cSGreg Clayton     return m_opaque_sp.get();
90330fdc8d8SChris Lattner }
90430fdc8d8SChris Lattner 
90530fdc8d8SChris Lattner const lldb_private::Thread *
90630fdc8d8SChris Lattner SBThread::operator->() const
90730fdc8d8SChris Lattner {
9086611103cSGreg Clayton     return m_opaque_sp.get();
90930fdc8d8SChris Lattner }
91030fdc8d8SChris Lattner 
91130fdc8d8SChris Lattner const lldb_private::Thread &
91230fdc8d8SChris Lattner SBThread::operator*() const
91330fdc8d8SChris Lattner {
9146611103cSGreg Clayton     return *m_opaque_sp;
91530fdc8d8SChris Lattner }
91630fdc8d8SChris Lattner 
91730fdc8d8SChris Lattner lldb_private::Thread *
91830fdc8d8SChris Lattner SBThread::operator->()
91930fdc8d8SChris Lattner {
9206611103cSGreg Clayton     return m_opaque_sp.get();
92130fdc8d8SChris Lattner }
92230fdc8d8SChris Lattner 
92330fdc8d8SChris Lattner lldb_private::Thread &
92430fdc8d8SChris Lattner SBThread::operator*()
92530fdc8d8SChris Lattner {
9266611103cSGreg Clayton     return *m_opaque_sp;
92730fdc8d8SChris Lattner }
928dde9cff3SCaroline Tice 
929dde9cff3SCaroline Tice bool
930ceb6b139SCaroline Tice SBThread::GetDescription (SBStream &description) const
931ceb6b139SCaroline Tice {
932ceb6b139SCaroline Tice     if (m_opaque_sp)
933ceb6b139SCaroline Tice     {
934ceb6b139SCaroline Tice         StreamString strm;
935ceb6b139SCaroline Tice         description.Printf("SBThread: tid = 0x%4.4x", m_opaque_sp->GetID());
936ceb6b139SCaroline Tice     }
937ceb6b139SCaroline Tice     else
938ceb6b139SCaroline Tice         description.Printf ("No value");
939ceb6b139SCaroline Tice 
940ceb6b139SCaroline Tice     return true;
941ceb6b139SCaroline Tice }
942