|  | //===-- SBThread.cpp --------------------------------------------*- C++ -*-===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "lldb/API/SBThread.h" | 
|  | #include "SBReproducerPrivate.h" | 
|  | #include "Utils.h" | 
|  | #include "lldb/API/SBAddress.h" | 
|  | #include "lldb/API/SBDebugger.h" | 
|  | #include "lldb/API/SBEvent.h" | 
|  | #include "lldb/API/SBFileSpec.h" | 
|  | #include "lldb/API/SBFrame.h" | 
|  | #include "lldb/API/SBProcess.h" | 
|  | #include "lldb/API/SBStream.h" | 
|  | #include "lldb/API/SBSymbolContext.h" | 
|  | #include "lldb/API/SBThreadCollection.h" | 
|  | #include "lldb/API/SBThreadPlan.h" | 
|  | #include "lldb/API/SBValue.h" | 
|  | #include "lldb/Breakpoint/BreakpointLocation.h" | 
|  | #include "lldb/Core/Debugger.h" | 
|  | #include "lldb/Core/StreamFile.h" | 
|  | #include "lldb/Core/ValueObject.h" | 
|  | #include "lldb/Interpreter/CommandInterpreter.h" | 
|  | #include "lldb/Symbol/CompileUnit.h" | 
|  | #include "lldb/Symbol/SymbolContext.h" | 
|  | #include "lldb/Target/Process.h" | 
|  | #include "lldb/Target/Queue.h" | 
|  | #include "lldb/Target/StopInfo.h" | 
|  | #include "lldb/Target/SystemRuntime.h" | 
|  | #include "lldb/Target/Target.h" | 
|  | #include "lldb/Target/Thread.h" | 
|  | #include "lldb/Target/ThreadPlan.h" | 
|  | #include "lldb/Target/ThreadPlanStepInRange.h" | 
|  | #include "lldb/Target/ThreadPlanStepInstruction.h" | 
|  | #include "lldb/Target/ThreadPlanStepOut.h" | 
|  | #include "lldb/Target/ThreadPlanStepRange.h" | 
|  | #include "lldb/Target/UnixSignals.h" | 
|  | #include "lldb/Utility/State.h" | 
|  | #include "lldb/Utility/Stream.h" | 
|  | #include "lldb/Utility/StructuredData.h" | 
|  | #include "lldb/lldb-enumerations.h" | 
|  |  | 
|  | #include <memory> | 
|  |  | 
|  | using namespace lldb; | 
|  | using namespace lldb_private; | 
|  |  | 
|  | const char *SBThread::GetBroadcasterClassName() { | 
|  | LLDB_RECORD_STATIC_METHOD_NO_ARGS(const char *, SBThread, | 
|  | GetBroadcasterClassName); | 
|  |  | 
|  | return Thread::GetStaticBroadcasterClass().AsCString(); | 
|  | } | 
|  |  | 
|  | // Constructors | 
|  | SBThread::SBThread() : m_opaque_sp(new ExecutionContextRef()) { | 
|  | LLDB_RECORD_CONSTRUCTOR_NO_ARGS(SBThread); | 
|  | } | 
|  |  | 
|  | SBThread::SBThread(const ThreadSP &lldb_object_sp) | 
|  | : m_opaque_sp(new ExecutionContextRef(lldb_object_sp)) { | 
|  | LLDB_RECORD_CONSTRUCTOR(SBThread, (const lldb::ThreadSP &), lldb_object_sp); | 
|  | } | 
|  |  | 
|  | SBThread::SBThread(const SBThread &rhs) : m_opaque_sp() { | 
|  | LLDB_RECORD_CONSTRUCTOR(SBThread, (const lldb::SBThread &), rhs); | 
|  |  | 
|  | m_opaque_sp = clone(rhs.m_opaque_sp); | 
|  | } | 
|  |  | 
|  | // Assignment operator | 
|  |  | 
|  | const lldb::SBThread &SBThread::operator=(const SBThread &rhs) { | 
|  | LLDB_RECORD_METHOD(const lldb::SBThread &, | 
|  | SBThread, operator=,(const lldb::SBThread &), rhs); | 
|  |  | 
|  | if (this != &rhs) | 
|  | m_opaque_sp = clone(rhs.m_opaque_sp); | 
|  | return LLDB_RECORD_RESULT(*this); | 
|  | } | 
|  |  | 
|  | // Destructor | 
|  | SBThread::~SBThread() {} | 
|  |  | 
|  | lldb::SBQueue SBThread::GetQueue() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(lldb::SBQueue, SBThread, GetQueue); | 
|  |  | 
|  | SBQueue sb_queue; | 
|  | QueueSP queue_sp; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | queue_sp = exe_ctx.GetThreadPtr()->GetQueue(); | 
|  | if (queue_sp) { | 
|  | sb_queue.SetQueue(queue_sp); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_queue); | 
|  | } | 
|  |  | 
|  | bool SBThread::IsValid() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBThread, IsValid); | 
|  | return this->operator bool(); | 
|  | } | 
|  | SBThread::operator bool() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBThread, operator bool); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | Target *target = exe_ctx.GetTargetPtr(); | 
|  | Process *process = exe_ctx.GetProcessPtr(); | 
|  | if (target && process) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&process->GetRunLock())) | 
|  | return m_opaque_sp->GetThreadSP().get() != nullptr; | 
|  | } | 
|  | // Without a valid target & process, this thread can't be valid. | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void SBThread::Clear() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(void, SBThread, Clear); | 
|  |  | 
|  | m_opaque_sp->Clear(); | 
|  | } | 
|  |  | 
|  | StopReason SBThread::GetStopReason() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::StopReason, SBThread, GetStopReason); | 
|  |  | 
|  | StopReason reason = eStopReasonInvalid; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | return exe_ctx.GetThreadPtr()->GetStopReason(); | 
|  | } | 
|  | } | 
|  |  | 
|  | return reason; | 
|  | } | 
|  |  | 
|  | size_t SBThread::GetStopReasonDataCount() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(size_t, SBThread, GetStopReasonDataCount); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | StopInfoSP stop_info_sp = exe_ctx.GetThreadPtr()->GetStopInfo(); | 
|  | if (stop_info_sp) { | 
|  | StopReason reason = stop_info_sp->GetStopReason(); | 
|  | switch (reason) { | 
|  | case eStopReasonInvalid: | 
|  | case eStopReasonNone: | 
|  | case eStopReasonTrace: | 
|  | case eStopReasonExec: | 
|  | case eStopReasonPlanComplete: | 
|  | case eStopReasonThreadExiting: | 
|  | case eStopReasonInstrumentation: | 
|  | // There is no data for these stop reasons. | 
|  | return 0; | 
|  |  | 
|  | case eStopReasonBreakpoint: { | 
|  | break_id_t site_id = stop_info_sp->GetValue(); | 
|  | lldb::BreakpointSiteSP bp_site_sp( | 
|  | exe_ctx.GetProcessPtr()->GetBreakpointSiteList().FindByID( | 
|  | site_id)); | 
|  | if (bp_site_sp) | 
|  | return bp_site_sp->GetNumberOfOwners() * 2; | 
|  | else | 
|  | return 0; // Breakpoint must have cleared itself... | 
|  | } break; | 
|  |  | 
|  | case eStopReasonWatchpoint: | 
|  | return 1; | 
|  |  | 
|  | case eStopReasonSignal: | 
|  | return 1; | 
|  |  | 
|  | case eStopReasonException: | 
|  | return 1; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | uint64_t SBThread::GetStopReasonDataAtIndex(uint32_t idx) { | 
|  | LLDB_RECORD_METHOD(uint64_t, SBThread, GetStopReasonDataAtIndex, (uint32_t), | 
|  | idx); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | StopInfoSP stop_info_sp = thread->GetStopInfo(); | 
|  | if (stop_info_sp) { | 
|  | StopReason reason = stop_info_sp->GetStopReason(); | 
|  | switch (reason) { | 
|  | case eStopReasonInvalid: | 
|  | case eStopReasonNone: | 
|  | case eStopReasonTrace: | 
|  | case eStopReasonExec: | 
|  | case eStopReasonPlanComplete: | 
|  | case eStopReasonThreadExiting: | 
|  | case eStopReasonInstrumentation: | 
|  | // There is no data for these stop reasons. | 
|  | return 0; | 
|  |  | 
|  | case eStopReasonBreakpoint: { | 
|  | break_id_t site_id = stop_info_sp->GetValue(); | 
|  | lldb::BreakpointSiteSP bp_site_sp( | 
|  | exe_ctx.GetProcessPtr()->GetBreakpointSiteList().FindByID( | 
|  | site_id)); | 
|  | if (bp_site_sp) { | 
|  | uint32_t bp_index = idx / 2; | 
|  | BreakpointLocationSP bp_loc_sp( | 
|  | bp_site_sp->GetOwnerAtIndex(bp_index)); | 
|  | if (bp_loc_sp) { | 
|  | if (idx & 1) { | 
|  | // Odd idx, return the breakpoint location ID | 
|  | return bp_loc_sp->GetID(); | 
|  | } else { | 
|  | // Even idx, return the breakpoint ID | 
|  | return bp_loc_sp->GetBreakpoint().GetID(); | 
|  | } | 
|  | } | 
|  | } | 
|  | return LLDB_INVALID_BREAK_ID; | 
|  | } break; | 
|  |  | 
|  | case eStopReasonWatchpoint: | 
|  | return stop_info_sp->GetValue(); | 
|  |  | 
|  | case eStopReasonSignal: | 
|  | return stop_info_sp->GetValue(); | 
|  |  | 
|  | case eStopReasonException: | 
|  | return stop_info_sp->GetValue(); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | bool SBThread::GetStopReasonExtendedInfoAsJSON(lldb::SBStream &stream) { | 
|  | LLDB_RECORD_METHOD(bool, SBThread, GetStopReasonExtendedInfoAsJSON, | 
|  | (lldb::SBStream &), stream); | 
|  |  | 
|  | Stream &strm = stream.ref(); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) | 
|  | return false; | 
|  |  | 
|  | StopInfoSP stop_info = exe_ctx.GetThreadPtr()->GetStopInfo(); | 
|  | StructuredData::ObjectSP info = stop_info->GetExtendedInfo(); | 
|  | if (!info) | 
|  | return false; | 
|  |  | 
|  | info->Dump(strm); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | SBThreadCollection | 
|  | SBThread::GetStopReasonExtendedBacktraces(InstrumentationRuntimeType type) { | 
|  | LLDB_RECORD_METHOD(lldb::SBThreadCollection, SBThread, | 
|  | GetStopReasonExtendedBacktraces, | 
|  | (lldb::InstrumentationRuntimeType), type); | 
|  |  | 
|  | ThreadCollectionSP threads; | 
|  | threads = std::make_shared<ThreadCollection>(); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) | 
|  | return LLDB_RECORD_RESULT(threads); | 
|  |  | 
|  | ProcessSP process_sp = exe_ctx.GetProcessSP(); | 
|  |  | 
|  | StopInfoSP stop_info = exe_ctx.GetThreadPtr()->GetStopInfo(); | 
|  | StructuredData::ObjectSP info = stop_info->GetExtendedInfo(); | 
|  | if (!info) | 
|  | return LLDB_RECORD_RESULT(threads); | 
|  |  | 
|  | return LLDB_RECORD_RESULT(process_sp->GetInstrumentationRuntime(type) | 
|  | ->GetBacktracesFromExtendedStopInfo(info)); | 
|  | } | 
|  |  | 
|  | size_t SBThread::GetStopDescription(char *dst, size_t dst_len) { | 
|  | LLDB_RECORD_METHOD(size_t, SBThread, GetStopDescription, (char *, size_t), | 
|  | dst, dst_len); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  |  | 
|  | StopInfoSP stop_info_sp = exe_ctx.GetThreadPtr()->GetStopInfo(); | 
|  | if (stop_info_sp) { | 
|  | const char *stop_desc = stop_info_sp->GetDescription(); | 
|  | if (stop_desc) { | 
|  | if (dst) | 
|  | return ::snprintf(dst, dst_len, "%s", stop_desc); | 
|  | else { | 
|  | // NULL dst passed in, return the length needed to contain the | 
|  | // description | 
|  | return ::strlen(stop_desc) + 1; // Include the NULL byte for size | 
|  | } | 
|  | } else { | 
|  | size_t stop_desc_len = 0; | 
|  | switch (stop_info_sp->GetStopReason()) { | 
|  | case eStopReasonTrace: | 
|  | case eStopReasonPlanComplete: { | 
|  | static char trace_desc[] = "step"; | 
|  | stop_desc = trace_desc; | 
|  | stop_desc_len = | 
|  | sizeof(trace_desc); // Include the NULL byte for size | 
|  | } break; | 
|  |  | 
|  | case eStopReasonBreakpoint: { | 
|  | static char bp_desc[] = "breakpoint hit"; | 
|  | stop_desc = bp_desc; | 
|  | stop_desc_len = sizeof(bp_desc); // Include the NULL byte for size | 
|  | } break; | 
|  |  | 
|  | case eStopReasonWatchpoint: { | 
|  | static char wp_desc[] = "watchpoint hit"; | 
|  | stop_desc = wp_desc; | 
|  | stop_desc_len = sizeof(wp_desc); // Include the NULL byte for size | 
|  | } break; | 
|  |  | 
|  | case eStopReasonSignal: { | 
|  | stop_desc = | 
|  | exe_ctx.GetProcessPtr()->GetUnixSignals()->GetSignalAsCString( | 
|  | stop_info_sp->GetValue()); | 
|  | if (stop_desc == nullptr || stop_desc[0] == '\0') { | 
|  | static char signal_desc[] = "signal"; | 
|  | stop_desc = signal_desc; | 
|  | stop_desc_len = | 
|  | sizeof(signal_desc); // Include the NULL byte for size | 
|  | } | 
|  | } break; | 
|  |  | 
|  | case eStopReasonException: { | 
|  | char exc_desc[] = "exception"; | 
|  | stop_desc = exc_desc; | 
|  | stop_desc_len = sizeof(exc_desc); // Include the NULL byte for size | 
|  | } break; | 
|  |  | 
|  | case eStopReasonExec: { | 
|  | char exc_desc[] = "exec"; | 
|  | stop_desc = exc_desc; | 
|  | stop_desc_len = sizeof(exc_desc); // Include the NULL byte for size | 
|  | } break; | 
|  |  | 
|  | case eStopReasonThreadExiting: { | 
|  | char limbo_desc[] = "thread exiting"; | 
|  | stop_desc = limbo_desc; | 
|  | stop_desc_len = sizeof(limbo_desc); | 
|  | } break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (stop_desc && stop_desc[0]) { | 
|  | if (dst) | 
|  | return ::snprintf(dst, dst_len, "%s", stop_desc) + | 
|  | 1; // Include the NULL byte | 
|  |  | 
|  | if (stop_desc_len == 0) | 
|  | stop_desc_len = ::strlen(stop_desc) + 1; // Include the NULL byte | 
|  |  | 
|  | return stop_desc_len; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | if (dst) | 
|  | *dst = 0; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | SBValue SBThread::GetStopReturnValue() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBValue, SBThread, GetStopReturnValue); | 
|  |  | 
|  | ValueObjectSP return_valobj_sp; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | StopInfoSP stop_info_sp = exe_ctx.GetThreadPtr()->GetStopInfo(); | 
|  | if (stop_info_sp) { | 
|  | return_valobj_sp = StopInfo::GetReturnValueObject(stop_info_sp); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(SBValue(return_valobj_sp)); | 
|  | } | 
|  |  | 
|  | void SBThread::SetThread(const ThreadSP &lldb_object_sp) { | 
|  | m_opaque_sp->SetThreadSP(lldb_object_sp); | 
|  | } | 
|  |  | 
|  | lldb::tid_t SBThread::GetThreadID() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(lldb::tid_t, SBThread, GetThreadID); | 
|  |  | 
|  | ThreadSP thread_sp(m_opaque_sp->GetThreadSP()); | 
|  | if (thread_sp) | 
|  | return thread_sp->GetID(); | 
|  | return LLDB_INVALID_THREAD_ID; | 
|  | } | 
|  |  | 
|  | uint32_t SBThread::GetIndexID() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(uint32_t, SBThread, GetIndexID); | 
|  |  | 
|  | ThreadSP thread_sp(m_opaque_sp->GetThreadSP()); | 
|  | if (thread_sp) | 
|  | return thread_sp->GetIndexID(); | 
|  | return LLDB_INVALID_INDEX32; | 
|  | } | 
|  |  | 
|  | const char *SBThread::GetName() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(const char *, SBThread, GetName); | 
|  |  | 
|  | const char *name = nullptr; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | name = exe_ctx.GetThreadPtr()->GetName(); | 
|  | } | 
|  | } | 
|  |  | 
|  | return name; | 
|  | } | 
|  |  | 
|  | const char *SBThread::GetQueueName() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(const char *, SBThread, GetQueueName); | 
|  |  | 
|  | const char *name = nullptr; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | name = exe_ctx.GetThreadPtr()->GetQueueName(); | 
|  | } | 
|  | } | 
|  |  | 
|  | return name; | 
|  | } | 
|  |  | 
|  | lldb::queue_id_t SBThread::GetQueueID() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(lldb::queue_id_t, SBThread, GetQueueID); | 
|  |  | 
|  | queue_id_t id = LLDB_INVALID_QUEUE_ID; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | id = exe_ctx.GetThreadPtr()->GetQueueID(); | 
|  | } | 
|  | } | 
|  |  | 
|  | return id; | 
|  | } | 
|  |  | 
|  | bool SBThread::GetInfoItemByPathAsString(const char *path, SBStream &strm) { | 
|  | LLDB_RECORD_METHOD(bool, SBThread, GetInfoItemByPathAsString, | 
|  | (const char *, lldb::SBStream &), path, strm); | 
|  |  | 
|  | bool success = false; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | StructuredData::ObjectSP info_root_sp = thread->GetExtendedInfo(); | 
|  | if (info_root_sp) { | 
|  | StructuredData::ObjectSP node = | 
|  | info_root_sp->GetObjectForDotSeparatedPath(path); | 
|  | if (node) { | 
|  | if (node->GetType() == eStructuredDataTypeString) { | 
|  | strm.Printf("%s", node->GetAsString()->GetValue().str().c_str()); | 
|  | success = true; | 
|  | } | 
|  | if (node->GetType() == eStructuredDataTypeInteger) { | 
|  | strm.Printf("0x%" PRIx64, node->GetAsInteger()->GetValue()); | 
|  | success = true; | 
|  | } | 
|  | if (node->GetType() == eStructuredDataTypeFloat) { | 
|  | strm.Printf("0x%f", node->GetAsFloat()->GetValue()); | 
|  | success = true; | 
|  | } | 
|  | if (node->GetType() == eStructuredDataTypeBoolean) { | 
|  | if (node->GetAsBoolean()->GetValue()) | 
|  | strm.Printf("true"); | 
|  | else | 
|  | strm.Printf("false"); | 
|  | success = true; | 
|  | } | 
|  | if (node->GetType() == eStructuredDataTypeNull) { | 
|  | strm.Printf("null"); | 
|  | success = true; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return success; | 
|  | } | 
|  |  | 
|  | SBError SBThread::ResumeNewPlan(ExecutionContext &exe_ctx, | 
|  | ThreadPlan *new_plan) { | 
|  | SBError sb_error; | 
|  |  | 
|  | Process *process = exe_ctx.GetProcessPtr(); | 
|  | if (!process) { | 
|  | sb_error.SetErrorString("No process in SBThread::ResumeNewPlan"); | 
|  | return sb_error; | 
|  | } | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | if (!thread) { | 
|  | sb_error.SetErrorString("No thread in SBThread::ResumeNewPlan"); | 
|  | return sb_error; | 
|  | } | 
|  |  | 
|  | // User level plans should be Master Plans so they can be interrupted, other | 
|  | // plans executed, and then a "continue" will resume the plan. | 
|  | if (new_plan != nullptr) { | 
|  | new_plan->SetIsMasterPlan(true); | 
|  | new_plan->SetOkayToDiscard(false); | 
|  | } | 
|  |  | 
|  | // Why do we need to set the current thread by ID here??? | 
|  | process->GetThreadList().SetSelectedThreadByID(thread->GetID()); | 
|  |  | 
|  | if (process->GetTarget().GetDebugger().GetAsyncExecution()) | 
|  | sb_error.ref() = process->Resume(); | 
|  | else | 
|  | sb_error.ref() = process->ResumeSynchronous(nullptr); | 
|  |  | 
|  | return sb_error; | 
|  | } | 
|  |  | 
|  | void SBThread::StepOver(lldb::RunMode stop_other_threads) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepOver, (lldb::RunMode), | 
|  | stop_other_threads); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | StepOver(stop_other_threads, error); | 
|  | } | 
|  |  | 
|  | void SBThread::StepOver(lldb::RunMode stop_other_threads, SBError &error) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepOver, (lldb::RunMode, lldb::SBError &), | 
|  | stop_other_threads, error); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | bool abort_other_plans = false; | 
|  | StackFrameSP frame_sp(thread->GetStackFrameAtIndex(0)); | 
|  |  | 
|  | Status new_plan_status; | 
|  | ThreadPlanSP new_plan_sp; | 
|  | if (frame_sp) { | 
|  | if (frame_sp->HasDebugInformation()) { | 
|  | const LazyBool avoid_no_debug = eLazyBoolCalculate; | 
|  | SymbolContext sc(frame_sp->GetSymbolContext(eSymbolContextEverything)); | 
|  | new_plan_sp = thread->QueueThreadPlanForStepOverRange( | 
|  | abort_other_plans, sc.line_entry, sc, stop_other_threads, | 
|  | new_plan_status, avoid_no_debug); | 
|  | } else { | 
|  | new_plan_sp = thread->QueueThreadPlanForStepSingleInstruction( | 
|  | true, abort_other_plans, stop_other_threads, new_plan_status); | 
|  | } | 
|  | } | 
|  | error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | } | 
|  |  | 
|  | void SBThread::StepInto(lldb::RunMode stop_other_threads) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepInto, (lldb::RunMode), | 
|  | stop_other_threads); | 
|  |  | 
|  | StepInto(nullptr, stop_other_threads); | 
|  | } | 
|  |  | 
|  | void SBThread::StepInto(const char *target_name, | 
|  | lldb::RunMode stop_other_threads) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepInto, (const char *, lldb::RunMode), | 
|  | target_name, stop_other_threads); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | StepInto(target_name, LLDB_INVALID_LINE_NUMBER, error, stop_other_threads); | 
|  | } | 
|  |  | 
|  | void SBThread::StepInto(const char *target_name, uint32_t end_line, | 
|  | SBError &error, lldb::RunMode stop_other_threads) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepInto, | 
|  | (const char *, uint32_t, lldb::SBError &, lldb::RunMode), | 
|  | target_name, end_line, error, stop_other_threads); | 
|  |  | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | bool abort_other_plans = false; | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | StackFrameSP frame_sp(thread->GetStackFrameAtIndex(0)); | 
|  | ThreadPlanSP new_plan_sp; | 
|  | Status new_plan_status; | 
|  |  | 
|  | if (frame_sp && frame_sp->HasDebugInformation()) { | 
|  | SymbolContext sc(frame_sp->GetSymbolContext(eSymbolContextEverything)); | 
|  | AddressRange range; | 
|  | if (end_line == LLDB_INVALID_LINE_NUMBER) | 
|  | range = sc.line_entry.range; | 
|  | else { | 
|  | if (!sc.GetAddressRangeFromHereToEndLine(end_line, range, error.ref())) | 
|  | return; | 
|  | } | 
|  |  | 
|  | const LazyBool step_out_avoids_code_without_debug_info = | 
|  | eLazyBoolCalculate; | 
|  | const LazyBool step_in_avoids_code_without_debug_info = | 
|  | eLazyBoolCalculate; | 
|  | new_plan_sp = thread->QueueThreadPlanForStepInRange( | 
|  | abort_other_plans, range, sc, target_name, stop_other_threads, | 
|  | new_plan_status, step_in_avoids_code_without_debug_info, | 
|  | step_out_avoids_code_without_debug_info); | 
|  | } else { | 
|  | new_plan_sp = thread->QueueThreadPlanForStepSingleInstruction( | 
|  | false, abort_other_plans, stop_other_threads, new_plan_status); | 
|  | } | 
|  |  | 
|  | if (new_plan_status.Success()) | 
|  | error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | else | 
|  | error.SetErrorString(new_plan_status.AsCString()); | 
|  | } | 
|  |  | 
|  | void SBThread::StepOut() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(void, SBThread, StepOut); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | StepOut(error); | 
|  | } | 
|  |  | 
|  | void SBThread::StepOut(SBError &error) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepOut, (lldb::SBError &), error); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | bool abort_other_plans = false; | 
|  | bool stop_other_threads = false; | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  |  | 
|  | const LazyBool avoid_no_debug = eLazyBoolCalculate; | 
|  | Status new_plan_status; | 
|  | ThreadPlanSP new_plan_sp(thread->QueueThreadPlanForStepOut( | 
|  | abort_other_plans, nullptr, false, stop_other_threads, eVoteYes, | 
|  | eVoteNoOpinion, 0, new_plan_status, avoid_no_debug)); | 
|  |  | 
|  | if (new_plan_status.Success()) | 
|  | error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | else | 
|  | error.SetErrorString(new_plan_status.AsCString()); | 
|  | } | 
|  |  | 
|  | void SBThread::StepOutOfFrame(SBFrame &sb_frame) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepOutOfFrame, (lldb::SBFrame &), | 
|  | sb_frame); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | StepOutOfFrame(sb_frame, error); | 
|  | } | 
|  |  | 
|  | void SBThread::StepOutOfFrame(SBFrame &sb_frame, SBError &error) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepOutOfFrame, | 
|  | (lldb::SBFrame &, lldb::SBError &), sb_frame, error); | 
|  |  | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!sb_frame.IsValid()) { | 
|  | error.SetErrorString("passed invalid SBFrame object"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | StackFrameSP frame_sp(sb_frame.GetFrameSP()); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | bool abort_other_plans = false; | 
|  | bool stop_other_threads = false; | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | if (sb_frame.GetThread().GetThreadID() != thread->GetID()) { | 
|  | error.SetErrorString("passed a frame from another thread"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | Status new_plan_status; | 
|  | ThreadPlanSP new_plan_sp(thread->QueueThreadPlanForStepOut( | 
|  | abort_other_plans, nullptr, false, stop_other_threads, eVoteYes, | 
|  | eVoteNoOpinion, frame_sp->GetFrameIndex(), new_plan_status)); | 
|  |  | 
|  | if (new_plan_status.Success()) | 
|  | error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | else | 
|  | error.SetErrorString(new_plan_status.AsCString()); | 
|  | } | 
|  |  | 
|  | void SBThread::StepInstruction(bool step_over) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepInstruction, (bool), step_over); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | StepInstruction(step_over, error); | 
|  | } | 
|  |  | 
|  | void SBThread::StepInstruction(bool step_over, SBError &error) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, StepInstruction, (bool, lldb::SBError &), | 
|  | step_over, error); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | Status new_plan_status; | 
|  | ThreadPlanSP new_plan_sp(thread->QueueThreadPlanForStepSingleInstruction( | 
|  | step_over, true, true, new_plan_status)); | 
|  |  | 
|  | if (new_plan_status.Success()) | 
|  | error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | else | 
|  | error.SetErrorString(new_plan_status.AsCString()); | 
|  | } | 
|  |  | 
|  | void SBThread::RunToAddress(lldb::addr_t addr) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, RunToAddress, (lldb::addr_t), addr); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | RunToAddress(addr, error); | 
|  | } | 
|  |  | 
|  | void SBThread::RunToAddress(lldb::addr_t addr, SBError &error) { | 
|  | LLDB_RECORD_METHOD(void, SBThread, RunToAddress, | 
|  | (lldb::addr_t, lldb::SBError &), addr, error); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | bool abort_other_plans = false; | 
|  | bool stop_other_threads = true; | 
|  |  | 
|  | Address target_addr(addr); | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  |  | 
|  | Status new_plan_status; | 
|  | ThreadPlanSP new_plan_sp(thread->QueueThreadPlanForRunToAddress( | 
|  | abort_other_plans, target_addr, stop_other_threads, new_plan_status)); | 
|  |  | 
|  | if (new_plan_status.Success()) | 
|  | error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | else | 
|  | error.SetErrorString(new_plan_status.AsCString()); | 
|  | } | 
|  |  | 
|  | SBError SBThread::StepOverUntil(lldb::SBFrame &sb_frame, | 
|  | lldb::SBFileSpec &sb_file_spec, uint32_t line) { | 
|  | LLDB_RECORD_METHOD(lldb::SBError, SBThread, StepOverUntil, | 
|  | (lldb::SBFrame &, lldb::SBFileSpec &, uint32_t), sb_frame, | 
|  | sb_file_spec, line); | 
|  |  | 
|  | SBError sb_error; | 
|  | char path[PATH_MAX]; | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | StackFrameSP frame_sp(sb_frame.GetFrameSP()); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Target *target = exe_ctx.GetTargetPtr(); | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  |  | 
|  | if (line == 0) { | 
|  | sb_error.SetErrorString("invalid line argument"); | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | if (!frame_sp) { | 
|  | frame_sp = thread->GetSelectedFrame(); | 
|  | if (!frame_sp) | 
|  | frame_sp = thread->GetStackFrameAtIndex(0); | 
|  | } | 
|  |  | 
|  | SymbolContext frame_sc; | 
|  | if (!frame_sp) { | 
|  | sb_error.SetErrorString("no valid frames in thread to step"); | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | // If we have a frame, get its line | 
|  | frame_sc = frame_sp->GetSymbolContext( | 
|  | eSymbolContextCompUnit | eSymbolContextFunction | | 
|  | eSymbolContextLineEntry | eSymbolContextSymbol); | 
|  |  | 
|  | if (frame_sc.comp_unit == nullptr) { | 
|  | sb_error.SetErrorStringWithFormat( | 
|  | "frame %u doesn't have debug information", frame_sp->GetFrameIndex()); | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | FileSpec step_file_spec; | 
|  | if (sb_file_spec.IsValid()) { | 
|  | // The file spec passed in was valid, so use it | 
|  | step_file_spec = sb_file_spec.ref(); | 
|  | } else { | 
|  | if (frame_sc.line_entry.IsValid()) | 
|  | step_file_spec = frame_sc.line_entry.file; | 
|  | else { | 
|  | sb_error.SetErrorString("invalid file argument or no file for frame"); | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Grab the current function, then we will make sure the "until" address is | 
|  | // within the function.  We discard addresses that are out of the current | 
|  | // function, and then if there are no addresses remaining, give an | 
|  | // appropriate error message. | 
|  |  | 
|  | bool all_in_function = true; | 
|  | AddressRange fun_range = frame_sc.function->GetAddressRange(); | 
|  |  | 
|  | std::vector<addr_t> step_over_until_addrs; | 
|  | const bool abort_other_plans = false; | 
|  | const bool stop_other_threads = false; | 
|  | const bool check_inlines = true; | 
|  | const bool exact = false; | 
|  |  | 
|  | SymbolContextList sc_list; | 
|  | const uint32_t num_matches = frame_sc.comp_unit->ResolveSymbolContext( | 
|  | step_file_spec, line, check_inlines, exact, eSymbolContextLineEntry, | 
|  | sc_list); | 
|  | if (num_matches > 0) { | 
|  | SymbolContext sc; | 
|  | for (uint32_t i = 0; i < num_matches; ++i) { | 
|  | if (sc_list.GetContextAtIndex(i, sc)) { | 
|  | addr_t step_addr = | 
|  | sc.line_entry.range.GetBaseAddress().GetLoadAddress(target); | 
|  | if (step_addr != LLDB_INVALID_ADDRESS) { | 
|  | if (fun_range.ContainsLoadAddress(step_addr, target)) | 
|  | step_over_until_addrs.push_back(step_addr); | 
|  | else | 
|  | all_in_function = false; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (step_over_until_addrs.empty()) { | 
|  | if (all_in_function) { | 
|  | step_file_spec.GetPath(path, sizeof(path)); | 
|  | sb_error.SetErrorStringWithFormat("No line entries for %s:%u", path, | 
|  | line); | 
|  | } else | 
|  | sb_error.SetErrorString("step until target not in current function"); | 
|  | } else { | 
|  | Status new_plan_status; | 
|  | ThreadPlanSP new_plan_sp(thread->QueueThreadPlanForStepUntil( | 
|  | abort_other_plans, &step_over_until_addrs[0], | 
|  | step_over_until_addrs.size(), stop_other_threads, | 
|  | frame_sp->GetFrameIndex(), new_plan_status)); | 
|  |  | 
|  | if (new_plan_status.Success()) | 
|  | sb_error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | else | 
|  | sb_error.SetErrorString(new_plan_status.AsCString()); | 
|  | } | 
|  | } else { | 
|  | sb_error.SetErrorString("this SBThread object is invalid"); | 
|  | } | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | SBError SBThread::StepUsingScriptedThreadPlan(const char *script_class_name) { | 
|  | LLDB_RECORD_METHOD(lldb::SBError, SBThread, StepUsingScriptedThreadPlan, | 
|  | (const char *), script_class_name); | 
|  |  | 
|  | return LLDB_RECORD_RESULT( | 
|  | StepUsingScriptedThreadPlan(script_class_name, true)); | 
|  | } | 
|  |  | 
|  | SBError SBThread::StepUsingScriptedThreadPlan(const char *script_class_name, | 
|  | bool resume_immediately) { | 
|  | LLDB_RECORD_METHOD(lldb::SBError, SBThread, StepUsingScriptedThreadPlan, | 
|  | (const char *, bool), script_class_name, | 
|  | resume_immediately); | 
|  |  | 
|  | SBError error; | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return LLDB_RECORD_RESULT(error); | 
|  | } | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | Status new_plan_status; | 
|  | ThreadPlanSP new_plan_sp = thread->QueueThreadPlanForStepScripted( | 
|  | false, script_class_name, false, new_plan_status); | 
|  |  | 
|  | if (new_plan_status.Fail()) { | 
|  | error.SetErrorString(new_plan_status.AsCString()); | 
|  | return LLDB_RECORD_RESULT(error); | 
|  | } | 
|  |  | 
|  | if (!resume_immediately) | 
|  | return LLDB_RECORD_RESULT(error); | 
|  |  | 
|  | if (new_plan_status.Success()) | 
|  | error = ResumeNewPlan(exe_ctx, new_plan_sp.get()); | 
|  | else | 
|  | error.SetErrorString(new_plan_status.AsCString()); | 
|  |  | 
|  | return LLDB_RECORD_RESULT(error); | 
|  | } | 
|  |  | 
|  | SBError SBThread::JumpToLine(lldb::SBFileSpec &file_spec, uint32_t line) { | 
|  | LLDB_RECORD_METHOD(lldb::SBError, SBThread, JumpToLine, | 
|  | (lldb::SBFileSpec &, uint32_t), file_spec, line); | 
|  |  | 
|  | SBError sb_error; | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (!exe_ctx.HasThreadScope()) { | 
|  | sb_error.SetErrorString("this SBThread object is invalid"); | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  |  | 
|  | Status err = thread->JumpToLine(file_spec.get(), line, true); | 
|  | sb_error.SetError(err); | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | SBError SBThread::ReturnFromFrame(SBFrame &frame, SBValue &return_value) { | 
|  | LLDB_RECORD_METHOD(lldb::SBError, SBThread, ReturnFromFrame, | 
|  | (lldb::SBFrame &, lldb::SBValue &), frame, return_value); | 
|  |  | 
|  | SBError sb_error; | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | sb_error.SetError( | 
|  | thread->ReturnFromFrame(frame.GetFrameSP(), return_value.GetSP())); | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | SBError SBThread::UnwindInnermostExpression() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBError, SBThread, | 
|  | UnwindInnermostExpression); | 
|  |  | 
|  | SBError sb_error; | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | sb_error.SetError(thread->UnwindInnermostExpression()); | 
|  | if (sb_error.Success()) | 
|  | thread->SetSelectedFrameByIndex(0, false); | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_error); | 
|  | } | 
|  |  | 
|  | bool SBThread::Suspend() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(bool, SBThread, Suspend); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | return Suspend(error); | 
|  | } | 
|  |  | 
|  | bool SBThread::Suspend(SBError &error) { | 
|  | LLDB_RECORD_METHOD(bool, SBThread, Suspend, (lldb::SBError &), error); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | bool result = false; | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | exe_ctx.GetThreadPtr()->SetResumeState(eStateSuspended); | 
|  | result = true; | 
|  | } else { | 
|  | error.SetErrorString("process is running"); | 
|  | } | 
|  | } else | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool SBThread::Resume() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(bool, SBThread, Resume); | 
|  |  | 
|  | SBError error; // Ignored | 
|  | return Resume(error); | 
|  | } | 
|  |  | 
|  | bool SBThread::Resume(SBError &error) { | 
|  | LLDB_RECORD_METHOD(bool, SBThread, Resume, (lldb::SBError &), error); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | bool result = false; | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | const bool override_suspend = true; | 
|  | exe_ctx.GetThreadPtr()->SetResumeState(eStateRunning, override_suspend); | 
|  | result = true; | 
|  | } else { | 
|  | error.SetErrorString("process is running"); | 
|  | } | 
|  | } else | 
|  | error.SetErrorString("this SBThread object is invalid"); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool SBThread::IsSuspended() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(bool, SBThread, IsSuspended); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) | 
|  | return exe_ctx.GetThreadPtr()->GetResumeState() == eStateSuspended; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool SBThread::IsStopped() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(bool, SBThread, IsStopped); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) | 
|  | return StateIsStoppedState(exe_ctx.GetThreadPtr()->GetState(), true); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | SBProcess SBThread::GetProcess() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBProcess, SBThread, GetProcess); | 
|  |  | 
|  | SBProcess sb_process; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | // Have to go up to the target so we can get a shared pointer to our | 
|  | // process... | 
|  | sb_process.SetSP(exe_ctx.GetProcessSP()); | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_process); | 
|  | } | 
|  |  | 
|  | uint32_t SBThread::GetNumFrames() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBThread, GetNumFrames); | 
|  |  | 
|  | uint32_t num_frames = 0; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | num_frames = exe_ctx.GetThreadPtr()->GetStackFrameCount(); | 
|  | } | 
|  | } | 
|  |  | 
|  | return num_frames; | 
|  | } | 
|  |  | 
|  | SBFrame SBThread::GetFrameAtIndex(uint32_t idx) { | 
|  | LLDB_RECORD_METHOD(lldb::SBFrame, SBThread, GetFrameAtIndex, (uint32_t), idx); | 
|  |  | 
|  | SBFrame sb_frame; | 
|  | StackFrameSP frame_sp; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | frame_sp = exe_ctx.GetThreadPtr()->GetStackFrameAtIndex(idx); | 
|  | sb_frame.SetFrameSP(frame_sp); | 
|  | } | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_frame); | 
|  | } | 
|  |  | 
|  | lldb::SBFrame SBThread::GetSelectedFrame() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBFrame, SBThread, GetSelectedFrame); | 
|  |  | 
|  | SBFrame sb_frame; | 
|  | StackFrameSP frame_sp; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | frame_sp = exe_ctx.GetThreadPtr()->GetSelectedFrame(); | 
|  | sb_frame.SetFrameSP(frame_sp); | 
|  | } | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_frame); | 
|  | } | 
|  |  | 
|  | lldb::SBFrame SBThread::SetSelectedFrame(uint32_t idx) { | 
|  | LLDB_RECORD_METHOD(lldb::SBFrame, SBThread, SetSelectedFrame, (uint32_t), | 
|  | idx); | 
|  |  | 
|  | SBFrame sb_frame; | 
|  | StackFrameSP frame_sp; | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | Thread *thread = exe_ctx.GetThreadPtr(); | 
|  | frame_sp = thread->GetStackFrameAtIndex(idx); | 
|  | if (frame_sp) { | 
|  | thread->SetSelectedFrame(frame_sp.get()); | 
|  | sb_frame.SetFrameSP(frame_sp); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_frame); | 
|  | } | 
|  |  | 
|  | bool SBThread::EventIsThreadEvent(const SBEvent &event) { | 
|  | LLDB_RECORD_STATIC_METHOD(bool, SBThread, EventIsThreadEvent, | 
|  | (const lldb::SBEvent &), event); | 
|  |  | 
|  | return Thread::ThreadEventData::GetEventDataFromEvent(event.get()) != nullptr; | 
|  | } | 
|  |  | 
|  | SBFrame SBThread::GetStackFrameFromEvent(const SBEvent &event) { | 
|  | LLDB_RECORD_STATIC_METHOD(lldb::SBFrame, SBThread, GetStackFrameFromEvent, | 
|  | (const lldb::SBEvent &), event); | 
|  |  | 
|  | return LLDB_RECORD_RESULT( | 
|  | Thread::ThreadEventData::GetStackFrameFromEvent(event.get())); | 
|  | } | 
|  |  | 
|  | SBThread SBThread::GetThreadFromEvent(const SBEvent &event) { | 
|  | LLDB_RECORD_STATIC_METHOD(lldb::SBThread, SBThread, GetThreadFromEvent, | 
|  | (const lldb::SBEvent &), event); | 
|  |  | 
|  | return LLDB_RECORD_RESULT( | 
|  | Thread::ThreadEventData::GetThreadFromEvent(event.get())); | 
|  | } | 
|  |  | 
|  | bool SBThread::operator==(const SBThread &rhs) const { | 
|  | LLDB_RECORD_METHOD_CONST(bool, SBThread, operator==,(const lldb::SBThread &), | 
|  | rhs); | 
|  |  | 
|  | return m_opaque_sp->GetThreadSP().get() == | 
|  | rhs.m_opaque_sp->GetThreadSP().get(); | 
|  | } | 
|  |  | 
|  | bool SBThread::operator!=(const SBThread &rhs) const { | 
|  | LLDB_RECORD_METHOD_CONST(bool, SBThread, operator!=,(const lldb::SBThread &), | 
|  | rhs); | 
|  |  | 
|  | return m_opaque_sp->GetThreadSP().get() != | 
|  | rhs.m_opaque_sp->GetThreadSP().get(); | 
|  | } | 
|  |  | 
|  | bool SBThread::GetStatus(SBStream &status) const { | 
|  | LLDB_RECORD_METHOD_CONST(bool, SBThread, GetStatus, (lldb::SBStream &), | 
|  | status); | 
|  |  | 
|  | Stream &strm = status.ref(); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | exe_ctx.GetThreadPtr()->GetStatus(strm, 0, 1, 1, true); | 
|  | } else | 
|  | strm.PutCString("No status"); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool SBThread::GetDescription(SBStream &description) const { | 
|  | LLDB_RECORD_METHOD_CONST(bool, SBThread, GetDescription, (lldb::SBStream &), | 
|  | description); | 
|  |  | 
|  | return GetDescription(description, false); | 
|  | } | 
|  |  | 
|  | bool SBThread::GetDescription(SBStream &description, bool stop_format) const { | 
|  | LLDB_RECORD_METHOD_CONST(bool, SBThread, GetDescription, | 
|  | (lldb::SBStream &, bool), description, stop_format); | 
|  |  | 
|  | Stream &strm = description.ref(); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  |  | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | exe_ctx.GetThreadPtr()->DumpUsingSettingsFormat(strm, | 
|  | LLDB_INVALID_THREAD_ID, | 
|  | stop_format); | 
|  | // strm.Printf("SBThread: tid = 0x%4.4" PRIx64, | 
|  | // exe_ctx.GetThreadPtr()->GetID()); | 
|  | } else | 
|  | strm.PutCString("No value"); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | SBThread SBThread::GetExtendedBacktraceThread(const char *type) { | 
|  | LLDB_RECORD_METHOD(lldb::SBThread, SBThread, GetExtendedBacktraceThread, | 
|  | (const char *), type); | 
|  |  | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ExecutionContext exe_ctx(m_opaque_sp.get(), lock); | 
|  | SBThread sb_origin_thread; | 
|  |  | 
|  | Process::StopLocker stop_locker; | 
|  | if (stop_locker.TryLock(&exe_ctx.GetProcessPtr()->GetRunLock())) { | 
|  | if (exe_ctx.HasThreadScope()) { | 
|  | ThreadSP real_thread(exe_ctx.GetThreadSP()); | 
|  | if (real_thread) { | 
|  | ConstString type_const(type); | 
|  | Process *process = exe_ctx.GetProcessPtr(); | 
|  | if (process) { | 
|  | SystemRuntime *runtime = process->GetSystemRuntime(); | 
|  | if (runtime) { | 
|  | ThreadSP new_thread_sp( | 
|  | runtime->GetExtendedBacktraceThread(real_thread, type_const)); | 
|  | if (new_thread_sp) { | 
|  | // Save this in the Process' ExtendedThreadList so a strong | 
|  | // pointer retains the object. | 
|  | process->GetExtendedThreadList().AddThread(new_thread_sp); | 
|  | sb_origin_thread.SetThread(new_thread_sp); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return LLDB_RECORD_RESULT(sb_origin_thread); | 
|  | } | 
|  |  | 
|  | uint32_t SBThread::GetExtendedBacktraceOriginatingIndexID() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBThread, | 
|  | GetExtendedBacktraceOriginatingIndexID); | 
|  |  | 
|  | ThreadSP thread_sp(m_opaque_sp->GetThreadSP()); | 
|  | if (thread_sp) | 
|  | return thread_sp->GetExtendedBacktraceOriginatingIndexID(); | 
|  | return LLDB_INVALID_INDEX32; | 
|  | } | 
|  |  | 
|  | SBValue SBThread::GetCurrentException() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBValue, SBThread, GetCurrentException); | 
|  |  | 
|  | ThreadSP thread_sp(m_opaque_sp->GetThreadSP()); | 
|  | if (!thread_sp) | 
|  | return LLDB_RECORD_RESULT(SBValue()); | 
|  |  | 
|  | return LLDB_RECORD_RESULT(SBValue(thread_sp->GetCurrentException())); | 
|  | } | 
|  |  | 
|  | SBThread SBThread::GetCurrentExceptionBacktrace() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBThread, SBThread, | 
|  | GetCurrentExceptionBacktrace); | 
|  |  | 
|  | ThreadSP thread_sp(m_opaque_sp->GetThreadSP()); | 
|  | if (!thread_sp) | 
|  | return LLDB_RECORD_RESULT(SBThread()); | 
|  |  | 
|  | return LLDB_RECORD_RESULT( | 
|  | SBThread(thread_sp->GetCurrentExceptionBacktrace())); | 
|  | } | 
|  |  | 
|  | bool SBThread::SafeToCallFunctions() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(bool, SBThread, SafeToCallFunctions); | 
|  |  | 
|  | ThreadSP thread_sp(m_opaque_sp->GetThreadSP()); | 
|  | if (thread_sp) | 
|  | return thread_sp->SafeToCallFunctions(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | lldb_private::Thread *SBThread::operator->() { | 
|  | return get(); | 
|  | } | 
|  |  | 
|  | lldb_private::Thread *SBThread::get() { | 
|  | return m_opaque_sp->GetThreadSP().get(); | 
|  | } | 
|  |  | 
|  | namespace lldb_private { | 
|  | namespace repro { | 
|  |  | 
|  | template <> | 
|  | void RegisterMethods<SBThread>(Registry &R) { | 
|  | LLDB_REGISTER_STATIC_METHOD(const char *, SBThread, GetBroadcasterClassName, | 
|  | ()); | 
|  | LLDB_REGISTER_CONSTRUCTOR(SBThread, ()); | 
|  | LLDB_REGISTER_CONSTRUCTOR(SBThread, (const lldb::ThreadSP &)); | 
|  | LLDB_REGISTER_CONSTRUCTOR(SBThread, (const lldb::SBThread &)); | 
|  | LLDB_REGISTER_METHOD(const lldb::SBThread &, | 
|  | SBThread, operator=,(const lldb::SBThread &)); | 
|  | LLDB_REGISTER_METHOD_CONST(lldb::SBQueue, SBThread, GetQueue, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, SBThread, IsValid, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, SBThread, operator bool, ()); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, Clear, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::StopReason, SBThread, GetStopReason, ()); | 
|  | LLDB_REGISTER_METHOD(size_t, SBThread, GetStopReasonDataCount, ()); | 
|  | LLDB_REGISTER_METHOD(uint64_t, SBThread, GetStopReasonDataAtIndex, | 
|  | (uint32_t)); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, GetStopReasonExtendedInfoAsJSON, | 
|  | (lldb::SBStream &)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBThreadCollection, SBThread, | 
|  | GetStopReasonExtendedBacktraces, | 
|  | (lldb::InstrumentationRuntimeType)); | 
|  | LLDB_REGISTER_METHOD(size_t, SBThread, GetStopDescription, | 
|  | (char *, size_t)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBValue, SBThread, GetStopReturnValue, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(lldb::tid_t, SBThread, GetThreadID, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(uint32_t, SBThread, GetIndexID, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(const char *, SBThread, GetName, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(const char *, SBThread, GetQueueName, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(lldb::queue_id_t, SBThread, GetQueueID, ()); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, GetInfoItemByPathAsString, | 
|  | (const char *, lldb::SBStream &)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepOver, (lldb::RunMode)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepOver, | 
|  | (lldb::RunMode, lldb::SBError &)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepInto, (lldb::RunMode)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepInto, | 
|  | (const char *, lldb::RunMode)); | 
|  | LLDB_REGISTER_METHOD( | 
|  | void, SBThread, StepInto, | 
|  | (const char *, uint32_t, lldb::SBError &, lldb::RunMode)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepOut, ()); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepOut, (lldb::SBError &)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepOutOfFrame, (lldb::SBFrame &)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepOutOfFrame, | 
|  | (lldb::SBFrame &, lldb::SBError &)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepInstruction, (bool)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, StepInstruction, | 
|  | (bool, lldb::SBError &)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, RunToAddress, (lldb::addr_t)); | 
|  | LLDB_REGISTER_METHOD(void, SBThread, RunToAddress, | 
|  | (lldb::addr_t, lldb::SBError &)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBError, SBThread, StepOverUntil, | 
|  | (lldb::SBFrame &, lldb::SBFileSpec &, uint32_t)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBError, SBThread, StepUsingScriptedThreadPlan, | 
|  | (const char *)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBError, SBThread, StepUsingScriptedThreadPlan, | 
|  | (const char *, bool)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBError, SBThread, JumpToLine, | 
|  | (lldb::SBFileSpec &, uint32_t)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBError, SBThread, ReturnFromFrame, | 
|  | (lldb::SBFrame &, lldb::SBValue &)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBError, SBThread, UnwindInnermostExpression, | 
|  | ()); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, Suspend, ()); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, Suspend, (lldb::SBError &)); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, Resume, ()); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, Resume, (lldb::SBError &)); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, IsSuspended, ()); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, IsStopped, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBProcess, SBThread, GetProcess, ()); | 
|  | LLDB_REGISTER_METHOD(uint32_t, SBThread, GetNumFrames, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBFrame, SBThread, GetFrameAtIndex, (uint32_t)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBFrame, SBThread, GetSelectedFrame, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBFrame, SBThread, SetSelectedFrame, (uint32_t)); | 
|  | LLDB_REGISTER_STATIC_METHOD(bool, SBThread, EventIsThreadEvent, | 
|  | (const lldb::SBEvent &)); | 
|  | LLDB_REGISTER_STATIC_METHOD(lldb::SBFrame, SBThread, GetStackFrameFromEvent, | 
|  | (const lldb::SBEvent &)); | 
|  | LLDB_REGISTER_STATIC_METHOD(lldb::SBThread, SBThread, GetThreadFromEvent, | 
|  | (const lldb::SBEvent &)); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, | 
|  | SBThread, operator==,(const lldb::SBThread &)); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, | 
|  | SBThread, operator!=,(const lldb::SBThread &)); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, SBThread, GetStatus, (lldb::SBStream &)); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, SBThread, GetDescription, | 
|  | (lldb::SBStream &)); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, SBThread, GetDescription, | 
|  | (lldb::SBStream &, bool)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBThread, SBThread, GetExtendedBacktraceThread, | 
|  | (const char *)); | 
|  | LLDB_REGISTER_METHOD(uint32_t, SBThread, | 
|  | GetExtendedBacktraceOriginatingIndexID, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBValue, SBThread, GetCurrentException, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBThread, SBThread, GetCurrentExceptionBacktrace, | 
|  | ()); | 
|  | LLDB_REGISTER_METHOD(bool, SBThread, SafeToCallFunctions, ()); | 
|  | } | 
|  |  | 
|  | } | 
|  | } |