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