xref: /llvm-project/lldb/source/Commands/CommandObjectThreadUtil.cpp (revision 2d704f4bf2edb0f9343dac818ab4d29442be9968)
1fb19f11eSWalter Erquinigo //===-- CommandObjectThreadUtil.cpp -----------------------------*- C++ -*-===//
2fb19f11eSWalter Erquinigo //
3fb19f11eSWalter Erquinigo // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4fb19f11eSWalter Erquinigo // See https://llvm.org/LICENSE.txt for license information.
5fb19f11eSWalter Erquinigo // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6fb19f11eSWalter Erquinigo //
7fb19f11eSWalter Erquinigo //===----------------------------------------------------------------------===//
8fb19f11eSWalter Erquinigo 
9fb19f11eSWalter Erquinigo #include "CommandObjectThreadUtil.h"
10fb19f11eSWalter Erquinigo 
11fb19f11eSWalter Erquinigo #include "lldb/Interpreter/CommandReturnObject.h"
12fb19f11eSWalter Erquinigo #include "lldb/Target/Process.h"
13fb19f11eSWalter Erquinigo #include "lldb/Target/Thread.h"
14fb19f11eSWalter Erquinigo 
15fb19f11eSWalter Erquinigo using namespace lldb;
16fb19f11eSWalter Erquinigo using namespace lldb_private;
17fb19f11eSWalter Erquinigo using namespace llvm;
18fb19f11eSWalter Erquinigo 
CommandObjectIterateOverThreads(CommandInterpreter & interpreter,const char * name,const char * help,const char * syntax,uint32_t flags)19fb19f11eSWalter Erquinigo CommandObjectIterateOverThreads::CommandObjectIterateOverThreads(
20fb19f11eSWalter Erquinigo     CommandInterpreter &interpreter, const char *name, const char *help,
21fb19f11eSWalter Erquinigo     const char *syntax, uint32_t flags)
22c1b07d61SJim Ingham     : CommandObjectParsed(interpreter, name, help, syntax, flags) {
23c1b07d61SJim Ingham   // These commands all take thread ID's as arguments.
24*2d704f4bSjimingham   AddSimpleArgumentList(eArgTypeThreadIndex, eArgRepeatStar);
25c1b07d61SJim Ingham }
26fb19f11eSWalter Erquinigo 
CommandObjectMultipleThreads(CommandInterpreter & interpreter,const char * name,const char * help,const char * syntax,uint32_t flags)276564ca18SWalter Erquinigo CommandObjectMultipleThreads::CommandObjectMultipleThreads(
286564ca18SWalter Erquinigo     CommandInterpreter &interpreter, const char *name, const char *help,
296564ca18SWalter Erquinigo     const char *syntax, uint32_t flags)
306564ca18SWalter Erquinigo     : CommandObjectParsed(interpreter, name, help, syntax, flags) {
316564ca18SWalter Erquinigo   // These commands all take thread ID's as arguments.
32*2d704f4bSjimingham   AddSimpleArgumentList(eArgTypeThreadIndex, eArgRepeatStar);
336564ca18SWalter Erquinigo }
346564ca18SWalter Erquinigo 
DoExecute(Args & command,CommandReturnObject & result)3592d8a28cSPete Lawrence void CommandObjectIterateOverThreads::DoExecute(Args &command,
36fb19f11eSWalter Erquinigo                                                 CommandReturnObject &result) {
37fb19f11eSWalter Erquinigo   result.SetStatus(m_success_return);
38fb19f11eSWalter Erquinigo 
39fb19f11eSWalter Erquinigo   bool all_threads = false;
40fb19f11eSWalter Erquinigo   if (command.GetArgumentCount() == 0) {
41fb19f11eSWalter Erquinigo     Thread *thread = m_exe_ctx.GetThreadPtr();
4292d8a28cSPete Lawrence     if (thread)
4392d8a28cSPete Lawrence       HandleOneThread(thread->GetID(), result);
4492d8a28cSPete Lawrence     return;
45fb19f11eSWalter Erquinigo   } else if (command.GetArgumentCount() == 1) {
46fb19f11eSWalter Erquinigo     all_threads = ::strcmp(command.GetArgumentAtIndex(0), "all") == 0;
47fb19f11eSWalter Erquinigo     m_unique_stacks = ::strcmp(command.GetArgumentAtIndex(0), "unique") == 0;
48fb19f11eSWalter Erquinigo   }
49fb19f11eSWalter Erquinigo 
50fb19f11eSWalter Erquinigo   // Use tids instead of ThreadSPs to prevent deadlocking problems which
51fb19f11eSWalter Erquinigo   // result from JIT-ing code while iterating over the (locked) ThreadSP
52fb19f11eSWalter Erquinigo   // list.
53fb19f11eSWalter Erquinigo   std::vector<lldb::tid_t> tids;
54fb19f11eSWalter Erquinigo 
55fb19f11eSWalter Erquinigo   if (all_threads || m_unique_stacks) {
56fb19f11eSWalter Erquinigo     Process *process = m_exe_ctx.GetProcessPtr();
57fb19f11eSWalter Erquinigo 
58fb19f11eSWalter Erquinigo     for (ThreadSP thread_sp : process->Threads())
59fb19f11eSWalter Erquinigo       tids.push_back(thread_sp->GetID());
60fb19f11eSWalter Erquinigo   } else {
61fb19f11eSWalter Erquinigo     const size_t num_args = command.GetArgumentCount();
62fb19f11eSWalter Erquinigo     Process *process = m_exe_ctx.GetProcessPtr();
63fb19f11eSWalter Erquinigo 
64fb19f11eSWalter Erquinigo     std::lock_guard<std::recursive_mutex> guard(
65fb19f11eSWalter Erquinigo         process->GetThreadList().GetMutex());
66fb19f11eSWalter Erquinigo 
67fb19f11eSWalter Erquinigo     for (size_t i = 0; i < num_args; i++) {
68fb19f11eSWalter Erquinigo       uint32_t thread_idx;
69fb19f11eSWalter Erquinigo       if (!llvm::to_integer(command.GetArgumentAtIndex(i), thread_idx)) {
70fb19f11eSWalter Erquinigo         result.AppendErrorWithFormat("invalid thread specification: \"%s\"\n",
71fb19f11eSWalter Erquinigo                                      command.GetArgumentAtIndex(i));
7292d8a28cSPete Lawrence         return;
73fb19f11eSWalter Erquinigo       }
74fb19f11eSWalter Erquinigo 
75fb19f11eSWalter Erquinigo       ThreadSP thread =
76fb19f11eSWalter Erquinigo           process->GetThreadList().FindThreadByIndexID(thread_idx);
77fb19f11eSWalter Erquinigo 
78fb19f11eSWalter Erquinigo       if (!thread) {
79fb19f11eSWalter Erquinigo         result.AppendErrorWithFormat("no thread with index: \"%s\"\n",
80fb19f11eSWalter Erquinigo                                      command.GetArgumentAtIndex(i));
8192d8a28cSPete Lawrence         return;
82fb19f11eSWalter Erquinigo       }
83fb19f11eSWalter Erquinigo 
84fb19f11eSWalter Erquinigo       tids.push_back(thread->GetID());
85fb19f11eSWalter Erquinigo     }
86fb19f11eSWalter Erquinigo   }
87fb19f11eSWalter Erquinigo 
88fb19f11eSWalter Erquinigo   if (m_unique_stacks) {
89fb19f11eSWalter Erquinigo     // Iterate over threads, finding unique stack buckets.
90fb19f11eSWalter Erquinigo     std::set<UniqueStack> unique_stacks;
91fb19f11eSWalter Erquinigo     for (const lldb::tid_t &tid : tids) {
92fb19f11eSWalter Erquinigo       if (!BucketThread(tid, unique_stacks, result)) {
9392d8a28cSPete Lawrence         return;
94fb19f11eSWalter Erquinigo       }
95fb19f11eSWalter Erquinigo     }
96fb19f11eSWalter Erquinigo 
97fb19f11eSWalter Erquinigo     // Write the thread id's and unique call stacks to the output stream
98fb19f11eSWalter Erquinigo     Stream &strm = result.GetOutputStream();
99fb19f11eSWalter Erquinigo     Process *process = m_exe_ctx.GetProcessPtr();
100fb19f11eSWalter Erquinigo     for (const UniqueStack &stack : unique_stacks) {
101fb19f11eSWalter Erquinigo       // List the common thread ID's
102fb19f11eSWalter Erquinigo       const std::vector<uint32_t> &thread_index_ids =
103fb19f11eSWalter Erquinigo           stack.GetUniqueThreadIndexIDs();
104fb19f11eSWalter Erquinigo       strm.Format("{0} thread(s) ", thread_index_ids.size());
105fb19f11eSWalter Erquinigo       for (const uint32_t &thread_index_id : thread_index_ids) {
106fb19f11eSWalter Erquinigo         strm.Format("#{0} ", thread_index_id);
107fb19f11eSWalter Erquinigo       }
108fb19f11eSWalter Erquinigo       strm.EOL();
109fb19f11eSWalter Erquinigo 
110fb19f11eSWalter Erquinigo       // List the shared call stack for this set of threads
111fb19f11eSWalter Erquinigo       uint32_t representative_thread_id = stack.GetRepresentativeThread();
112fb19f11eSWalter Erquinigo       ThreadSP thread = process->GetThreadList().FindThreadByIndexID(
113fb19f11eSWalter Erquinigo           representative_thread_id);
114fb19f11eSWalter Erquinigo       if (!HandleOneThread(thread->GetID(), result)) {
11592d8a28cSPete Lawrence         return;
116fb19f11eSWalter Erquinigo       }
117fb19f11eSWalter Erquinigo     }
118fb19f11eSWalter Erquinigo   } else {
119fb19f11eSWalter Erquinigo     uint32_t idx = 0;
120fb19f11eSWalter Erquinigo     for (const lldb::tid_t &tid : tids) {
121fb19f11eSWalter Erquinigo       if (idx != 0 && m_add_return)
122fb19f11eSWalter Erquinigo         result.AppendMessage("");
123fb19f11eSWalter Erquinigo 
124fb19f11eSWalter Erquinigo       if (!HandleOneThread(tid, result))
12592d8a28cSPete Lawrence         return;
126fb19f11eSWalter Erquinigo 
127fb19f11eSWalter Erquinigo       ++idx;
128fb19f11eSWalter Erquinigo     }
129fb19f11eSWalter Erquinigo   }
130fb19f11eSWalter Erquinigo }
131fb19f11eSWalter Erquinigo 
BucketThread(lldb::tid_t tid,std::set<UniqueStack> & unique_stacks,CommandReturnObject & result)132fb19f11eSWalter Erquinigo bool CommandObjectIterateOverThreads::BucketThread(
133fb19f11eSWalter Erquinigo     lldb::tid_t tid, std::set<UniqueStack> &unique_stacks,
134fb19f11eSWalter Erquinigo     CommandReturnObject &result) {
135fb19f11eSWalter Erquinigo   // Grab the corresponding thread for the given thread id.
136fb19f11eSWalter Erquinigo   Process *process = m_exe_ctx.GetProcessPtr();
137fb19f11eSWalter Erquinigo   Thread *thread = process->GetThreadList().FindThreadByID(tid).get();
138fb19f11eSWalter Erquinigo   if (thread == nullptr) {
139fb19f11eSWalter Erquinigo     result.AppendErrorWithFormatv("Failed to process thread #{0}.\n", tid);
140fb19f11eSWalter Erquinigo     return false;
141fb19f11eSWalter Erquinigo   }
142fb19f11eSWalter Erquinigo 
143fb19f11eSWalter Erquinigo   // Collect the each frame's address for this call-stack
144fb19f11eSWalter Erquinigo   std::stack<lldb::addr_t> stack_frames;
145fb19f11eSWalter Erquinigo   const uint32_t frame_count = thread->GetStackFrameCount();
146fb19f11eSWalter Erquinigo   for (uint32_t frame_index = 0; frame_index < frame_count; frame_index++) {
147fb19f11eSWalter Erquinigo     const lldb::StackFrameSP frame_sp =
148fb19f11eSWalter Erquinigo         thread->GetStackFrameAtIndex(frame_index);
149fb19f11eSWalter Erquinigo     const lldb::addr_t pc = frame_sp->GetStackID().GetPC();
150fb19f11eSWalter Erquinigo     stack_frames.push(pc);
151fb19f11eSWalter Erquinigo   }
152fb19f11eSWalter Erquinigo 
153fb19f11eSWalter Erquinigo   uint32_t thread_index_id = thread->GetIndexID();
154fb19f11eSWalter Erquinigo   UniqueStack new_unique_stack(stack_frames, thread_index_id);
155fb19f11eSWalter Erquinigo 
156fb19f11eSWalter Erquinigo   // Try to match the threads stack to and existing entry.
157fb19f11eSWalter Erquinigo   std::set<UniqueStack>::iterator matching_stack =
158fb19f11eSWalter Erquinigo       unique_stacks.find(new_unique_stack);
159fb19f11eSWalter Erquinigo   if (matching_stack != unique_stacks.end()) {
160fb19f11eSWalter Erquinigo     matching_stack->AddThread(thread_index_id);
161fb19f11eSWalter Erquinigo   } else {
162fb19f11eSWalter Erquinigo     unique_stacks.insert(new_unique_stack);
163fb19f11eSWalter Erquinigo   }
164fb19f11eSWalter Erquinigo   return true;
165fb19f11eSWalter Erquinigo }
1660b697561SWalter Erquinigo 
DoExecute(Args & command,CommandReturnObject & result)16792d8a28cSPete Lawrence void CommandObjectMultipleThreads::DoExecute(Args &command,
1680b697561SWalter Erquinigo                                              CommandReturnObject &result) {
1690b697561SWalter Erquinigo   Process &process = m_exe_ctx.GetProcessRef();
1700b697561SWalter Erquinigo 
1710b697561SWalter Erquinigo   std::vector<lldb::tid_t> tids;
1720b697561SWalter Erquinigo   const size_t num_args = command.GetArgumentCount();
1730b697561SWalter Erquinigo 
1740b697561SWalter Erquinigo   std::lock_guard<std::recursive_mutex> guard(
1750b697561SWalter Erquinigo       process.GetThreadList().GetMutex());
1760b697561SWalter Erquinigo 
1770b697561SWalter Erquinigo   if (num_args > 0 && ::strcmp(command.GetArgumentAtIndex(0), "all") == 0) {
1780b697561SWalter Erquinigo     for (ThreadSP thread_sp : process.Threads())
1790b697561SWalter Erquinigo       tids.push_back(thread_sp->GetID());
1800b697561SWalter Erquinigo   } else {
1810b697561SWalter Erquinigo     if (num_args == 0) {
1820b697561SWalter Erquinigo       Thread &thread = m_exe_ctx.GetThreadRef();
1830b697561SWalter Erquinigo       tids.push_back(thread.GetID());
1840b697561SWalter Erquinigo     }
1850b697561SWalter Erquinigo 
1860b697561SWalter Erquinigo     for (size_t i = 0; i < num_args; i++) {
1870b697561SWalter Erquinigo       uint32_t thread_idx;
1880b697561SWalter Erquinigo       if (!llvm::to_integer(command.GetArgumentAtIndex(i), thread_idx)) {
1890b697561SWalter Erquinigo         result.AppendErrorWithFormat("invalid thread specification: \"%s\"\n",
1900b697561SWalter Erquinigo                                      command.GetArgumentAtIndex(i));
19192d8a28cSPete Lawrence         return;
1920b697561SWalter Erquinigo       }
1930b697561SWalter Erquinigo 
1940b697561SWalter Erquinigo       ThreadSP thread = process.GetThreadList().FindThreadByIndexID(thread_idx);
1950b697561SWalter Erquinigo 
1960b697561SWalter Erquinigo       if (!thread) {
1970b697561SWalter Erquinigo         result.AppendErrorWithFormat("no thread with index: \"%s\"\n",
1980b697561SWalter Erquinigo                                      command.GetArgumentAtIndex(i));
19992d8a28cSPete Lawrence         return;
2000b697561SWalter Erquinigo       }
2010b697561SWalter Erquinigo 
2020b697561SWalter Erquinigo       tids.push_back(thread->GetID());
2030b697561SWalter Erquinigo     }
2040b697561SWalter Erquinigo   }
2050b697561SWalter Erquinigo 
20692d8a28cSPete Lawrence   DoExecuteOnThreads(command, result, tids);
2070b697561SWalter Erquinigo }
208