1*fb19f11eSWalter Erquinigo //===-- CommandObjectThreadUtil.cpp -----------------------------*- C++ -*-===// 2*fb19f11eSWalter Erquinigo // 3*fb19f11eSWalter Erquinigo // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*fb19f11eSWalter Erquinigo // See https://llvm.org/LICENSE.txt for license information. 5*fb19f11eSWalter Erquinigo // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*fb19f11eSWalter Erquinigo // 7*fb19f11eSWalter Erquinigo //===----------------------------------------------------------------------===// 8*fb19f11eSWalter Erquinigo 9*fb19f11eSWalter Erquinigo #include "CommandObjectThreadUtil.h" 10*fb19f11eSWalter Erquinigo 11*fb19f11eSWalter Erquinigo #include "lldb/Interpreter/CommandReturnObject.h" 12*fb19f11eSWalter Erquinigo #include "lldb/Target/Process.h" 13*fb19f11eSWalter Erquinigo #include "lldb/Target/Thread.h" 14*fb19f11eSWalter Erquinigo 15*fb19f11eSWalter Erquinigo using namespace lldb; 16*fb19f11eSWalter Erquinigo using namespace lldb_private; 17*fb19f11eSWalter Erquinigo using namespace llvm; 18*fb19f11eSWalter Erquinigo 19*fb19f11eSWalter Erquinigo CommandObjectIterateOverThreads::CommandObjectIterateOverThreads( 20*fb19f11eSWalter Erquinigo CommandInterpreter &interpreter, const char *name, const char *help, 21*fb19f11eSWalter Erquinigo const char *syntax, uint32_t flags) 22*fb19f11eSWalter Erquinigo : CommandObjectParsed(interpreter, name, help, syntax, flags) {} 23*fb19f11eSWalter Erquinigo 24*fb19f11eSWalter Erquinigo bool CommandObjectIterateOverThreads::DoExecute(Args &command, 25*fb19f11eSWalter Erquinigo CommandReturnObject &result) { 26*fb19f11eSWalter Erquinigo result.SetStatus(m_success_return); 27*fb19f11eSWalter Erquinigo 28*fb19f11eSWalter Erquinigo bool all_threads = false; 29*fb19f11eSWalter Erquinigo if (command.GetArgumentCount() == 0) { 30*fb19f11eSWalter Erquinigo Thread *thread = m_exe_ctx.GetThreadPtr(); 31*fb19f11eSWalter Erquinigo if (!thread || !HandleOneThread(thread->GetID(), result)) 32*fb19f11eSWalter Erquinigo return false; 33*fb19f11eSWalter Erquinigo return result.Succeeded(); 34*fb19f11eSWalter Erquinigo } else if (command.GetArgumentCount() == 1) { 35*fb19f11eSWalter Erquinigo all_threads = ::strcmp(command.GetArgumentAtIndex(0), "all") == 0; 36*fb19f11eSWalter Erquinigo m_unique_stacks = ::strcmp(command.GetArgumentAtIndex(0), "unique") == 0; 37*fb19f11eSWalter Erquinigo } 38*fb19f11eSWalter Erquinigo 39*fb19f11eSWalter Erquinigo // Use tids instead of ThreadSPs to prevent deadlocking problems which 40*fb19f11eSWalter Erquinigo // result from JIT-ing code while iterating over the (locked) ThreadSP 41*fb19f11eSWalter Erquinigo // list. 42*fb19f11eSWalter Erquinigo std::vector<lldb::tid_t> tids; 43*fb19f11eSWalter Erquinigo 44*fb19f11eSWalter Erquinigo if (all_threads || m_unique_stacks) { 45*fb19f11eSWalter Erquinigo Process *process = m_exe_ctx.GetProcessPtr(); 46*fb19f11eSWalter Erquinigo 47*fb19f11eSWalter Erquinigo for (ThreadSP thread_sp : process->Threads()) 48*fb19f11eSWalter Erquinigo tids.push_back(thread_sp->GetID()); 49*fb19f11eSWalter Erquinigo } else { 50*fb19f11eSWalter Erquinigo const size_t num_args = command.GetArgumentCount(); 51*fb19f11eSWalter Erquinigo Process *process = m_exe_ctx.GetProcessPtr(); 52*fb19f11eSWalter Erquinigo 53*fb19f11eSWalter Erquinigo std::lock_guard<std::recursive_mutex> guard( 54*fb19f11eSWalter Erquinigo process->GetThreadList().GetMutex()); 55*fb19f11eSWalter Erquinigo 56*fb19f11eSWalter Erquinigo for (size_t i = 0; i < num_args; i++) { 57*fb19f11eSWalter Erquinigo uint32_t thread_idx; 58*fb19f11eSWalter Erquinigo if (!llvm::to_integer(command.GetArgumentAtIndex(i), thread_idx)) { 59*fb19f11eSWalter Erquinigo result.AppendErrorWithFormat("invalid thread specification: \"%s\"\n", 60*fb19f11eSWalter Erquinigo command.GetArgumentAtIndex(i)); 61*fb19f11eSWalter Erquinigo result.SetStatus(eReturnStatusFailed); 62*fb19f11eSWalter Erquinigo return false; 63*fb19f11eSWalter Erquinigo } 64*fb19f11eSWalter Erquinigo 65*fb19f11eSWalter Erquinigo ThreadSP thread = 66*fb19f11eSWalter Erquinigo process->GetThreadList().FindThreadByIndexID(thread_idx); 67*fb19f11eSWalter Erquinigo 68*fb19f11eSWalter Erquinigo if (!thread) { 69*fb19f11eSWalter Erquinigo result.AppendErrorWithFormat("no thread with index: \"%s\"\n", 70*fb19f11eSWalter Erquinigo command.GetArgumentAtIndex(i)); 71*fb19f11eSWalter Erquinigo result.SetStatus(eReturnStatusFailed); 72*fb19f11eSWalter Erquinigo return false; 73*fb19f11eSWalter Erquinigo } 74*fb19f11eSWalter Erquinigo 75*fb19f11eSWalter Erquinigo tids.push_back(thread->GetID()); 76*fb19f11eSWalter Erquinigo } 77*fb19f11eSWalter Erquinigo } 78*fb19f11eSWalter Erquinigo 79*fb19f11eSWalter Erquinigo if (m_unique_stacks) { 80*fb19f11eSWalter Erquinigo // Iterate over threads, finding unique stack buckets. 81*fb19f11eSWalter Erquinigo std::set<UniqueStack> unique_stacks; 82*fb19f11eSWalter Erquinigo for (const lldb::tid_t &tid : tids) { 83*fb19f11eSWalter Erquinigo if (!BucketThread(tid, unique_stacks, result)) { 84*fb19f11eSWalter Erquinigo return false; 85*fb19f11eSWalter Erquinigo } 86*fb19f11eSWalter Erquinigo } 87*fb19f11eSWalter Erquinigo 88*fb19f11eSWalter Erquinigo // Write the thread id's and unique call stacks to the output stream 89*fb19f11eSWalter Erquinigo Stream &strm = result.GetOutputStream(); 90*fb19f11eSWalter Erquinigo Process *process = m_exe_ctx.GetProcessPtr(); 91*fb19f11eSWalter Erquinigo for (const UniqueStack &stack : unique_stacks) { 92*fb19f11eSWalter Erquinigo // List the common thread ID's 93*fb19f11eSWalter Erquinigo const std::vector<uint32_t> &thread_index_ids = 94*fb19f11eSWalter Erquinigo stack.GetUniqueThreadIndexIDs(); 95*fb19f11eSWalter Erquinigo strm.Format("{0} thread(s) ", thread_index_ids.size()); 96*fb19f11eSWalter Erquinigo for (const uint32_t &thread_index_id : thread_index_ids) { 97*fb19f11eSWalter Erquinigo strm.Format("#{0} ", thread_index_id); 98*fb19f11eSWalter Erquinigo } 99*fb19f11eSWalter Erquinigo strm.EOL(); 100*fb19f11eSWalter Erquinigo 101*fb19f11eSWalter Erquinigo // List the shared call stack for this set of threads 102*fb19f11eSWalter Erquinigo uint32_t representative_thread_id = stack.GetRepresentativeThread(); 103*fb19f11eSWalter Erquinigo ThreadSP thread = process->GetThreadList().FindThreadByIndexID( 104*fb19f11eSWalter Erquinigo representative_thread_id); 105*fb19f11eSWalter Erquinigo if (!HandleOneThread(thread->GetID(), result)) { 106*fb19f11eSWalter Erquinigo return false; 107*fb19f11eSWalter Erquinigo } 108*fb19f11eSWalter Erquinigo } 109*fb19f11eSWalter Erquinigo } else { 110*fb19f11eSWalter Erquinigo uint32_t idx = 0; 111*fb19f11eSWalter Erquinigo for (const lldb::tid_t &tid : tids) { 112*fb19f11eSWalter Erquinigo if (idx != 0 && m_add_return) 113*fb19f11eSWalter Erquinigo result.AppendMessage(""); 114*fb19f11eSWalter Erquinigo 115*fb19f11eSWalter Erquinigo if (!HandleOneThread(tid, result)) 116*fb19f11eSWalter Erquinigo return false; 117*fb19f11eSWalter Erquinigo 118*fb19f11eSWalter Erquinigo ++idx; 119*fb19f11eSWalter Erquinigo } 120*fb19f11eSWalter Erquinigo } 121*fb19f11eSWalter Erquinigo return result.Succeeded(); 122*fb19f11eSWalter Erquinigo } 123*fb19f11eSWalter Erquinigo 124*fb19f11eSWalter Erquinigo bool CommandObjectIterateOverThreads::BucketThread( 125*fb19f11eSWalter Erquinigo lldb::tid_t tid, std::set<UniqueStack> &unique_stacks, 126*fb19f11eSWalter Erquinigo CommandReturnObject &result) { 127*fb19f11eSWalter Erquinigo // Grab the corresponding thread for the given thread id. 128*fb19f11eSWalter Erquinigo Process *process = m_exe_ctx.GetProcessPtr(); 129*fb19f11eSWalter Erquinigo Thread *thread = process->GetThreadList().FindThreadByID(tid).get(); 130*fb19f11eSWalter Erquinigo if (thread == nullptr) { 131*fb19f11eSWalter Erquinigo result.AppendErrorWithFormatv("Failed to process thread #{0}.\n", tid); 132*fb19f11eSWalter Erquinigo result.SetStatus(eReturnStatusFailed); 133*fb19f11eSWalter Erquinigo return false; 134*fb19f11eSWalter Erquinigo } 135*fb19f11eSWalter Erquinigo 136*fb19f11eSWalter Erquinigo // Collect the each frame's address for this call-stack 137*fb19f11eSWalter Erquinigo std::stack<lldb::addr_t> stack_frames; 138*fb19f11eSWalter Erquinigo const uint32_t frame_count = thread->GetStackFrameCount(); 139*fb19f11eSWalter Erquinigo for (uint32_t frame_index = 0; frame_index < frame_count; frame_index++) { 140*fb19f11eSWalter Erquinigo const lldb::StackFrameSP frame_sp = 141*fb19f11eSWalter Erquinigo thread->GetStackFrameAtIndex(frame_index); 142*fb19f11eSWalter Erquinigo const lldb::addr_t pc = frame_sp->GetStackID().GetPC(); 143*fb19f11eSWalter Erquinigo stack_frames.push(pc); 144*fb19f11eSWalter Erquinigo } 145*fb19f11eSWalter Erquinigo 146*fb19f11eSWalter Erquinigo uint32_t thread_index_id = thread->GetIndexID(); 147*fb19f11eSWalter Erquinigo UniqueStack new_unique_stack(stack_frames, thread_index_id); 148*fb19f11eSWalter Erquinigo 149*fb19f11eSWalter Erquinigo // Try to match the threads stack to and existing entry. 150*fb19f11eSWalter Erquinigo std::set<UniqueStack>::iterator matching_stack = 151*fb19f11eSWalter Erquinigo unique_stacks.find(new_unique_stack); 152*fb19f11eSWalter Erquinigo if (matching_stack != unique_stacks.end()) { 153*fb19f11eSWalter Erquinigo matching_stack->AddThread(thread_index_id); 154*fb19f11eSWalter Erquinigo } else { 155*fb19f11eSWalter Erquinigo unique_stacks.insert(new_unique_stack); 156*fb19f11eSWalter Erquinigo } 157*fb19f11eSWalter Erquinigo return true; 158*fb19f11eSWalter Erquinigo } 159