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