xref: /openbsd-src/gnu/llvm/lldb/source/Commands/CommandObjectThreadUtil.h (revision be691f3bb6417f04a68938fadbcaee2d5795e764)
1*be691f3bSpatrick //===-- CommandObjectThreadUtil.h -------------------------------*- C++ -*-===//
2*be691f3bSpatrick //
3*be691f3bSpatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*be691f3bSpatrick // See https://llvm.org/LICENSE.txt for license information.
5*be691f3bSpatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*be691f3bSpatrick //
7*be691f3bSpatrick //===----------------------------------------------------------------------===//
8*be691f3bSpatrick 
9*be691f3bSpatrick #ifndef LLDB_SOURCE_COMMANDS_COMMANDOBJECTTHREADUTIL_H
10*be691f3bSpatrick #define LLDB_SOURCE_COMMANDS_COMMANDOBJECTTHREADUTIL_H
11*be691f3bSpatrick 
12*be691f3bSpatrick #include "lldb/Interpreter/CommandObjectMultiword.h"
13*be691f3bSpatrick 
14*be691f3bSpatrick namespace lldb_private {
15*be691f3bSpatrick 
16*be691f3bSpatrick class CommandObjectIterateOverThreads : public CommandObjectParsed {
17*be691f3bSpatrick 
18*be691f3bSpatrick   class UniqueStack {
19*be691f3bSpatrick   public:
20*be691f3bSpatrick     UniqueStack(std::stack<lldb::addr_t> stack_frames, uint32_t thread_index_id)
21*be691f3bSpatrick         : m_stack_frames(stack_frames) {
22*be691f3bSpatrick       m_thread_index_ids.push_back(thread_index_id);
23*be691f3bSpatrick     }
24*be691f3bSpatrick 
25*be691f3bSpatrick     void AddThread(uint32_t thread_index_id) const {
26*be691f3bSpatrick       m_thread_index_ids.push_back(thread_index_id);
27*be691f3bSpatrick     }
28*be691f3bSpatrick 
29*be691f3bSpatrick     const std::vector<uint32_t> &GetUniqueThreadIndexIDs() const {
30*be691f3bSpatrick       return m_thread_index_ids;
31*be691f3bSpatrick     }
32*be691f3bSpatrick 
33*be691f3bSpatrick     lldb::tid_t GetRepresentativeThread() const {
34*be691f3bSpatrick       return m_thread_index_ids.front();
35*be691f3bSpatrick     }
36*be691f3bSpatrick 
37*be691f3bSpatrick     friend bool inline operator<(const UniqueStack &lhs,
38*be691f3bSpatrick                                  const UniqueStack &rhs) {
39*be691f3bSpatrick       return lhs.m_stack_frames < rhs.m_stack_frames;
40*be691f3bSpatrick     }
41*be691f3bSpatrick 
42*be691f3bSpatrick   protected:
43*be691f3bSpatrick     // Mark the thread index as mutable, as we don't care about it from a const
44*be691f3bSpatrick     // perspective, we only care about m_stack_frames so we keep our std::set
45*be691f3bSpatrick     // sorted.
46*be691f3bSpatrick     mutable std::vector<uint32_t> m_thread_index_ids;
47*be691f3bSpatrick     std::stack<lldb::addr_t> m_stack_frames;
48*be691f3bSpatrick   };
49*be691f3bSpatrick 
50*be691f3bSpatrick public:
51*be691f3bSpatrick   CommandObjectIterateOverThreads(CommandInterpreter &interpreter,
52*be691f3bSpatrick                                   const char *name, const char *help,
53*be691f3bSpatrick                                   const char *syntax, uint32_t flags);
54*be691f3bSpatrick 
55*be691f3bSpatrick   ~CommandObjectIterateOverThreads() override = default;
56*be691f3bSpatrick 
57*be691f3bSpatrick   bool DoExecute(Args &command, CommandReturnObject &result) override;
58*be691f3bSpatrick 
59*be691f3bSpatrick protected:
60*be691f3bSpatrick   // Override this to do whatever you need to do for one thread.
61*be691f3bSpatrick   //
62*be691f3bSpatrick   // If you return false, the iteration will stop, otherwise it will proceed.
63*be691f3bSpatrick   // The result is set to m_success_return (defaults to
64*be691f3bSpatrick   // eReturnStatusSuccessFinishResult) before the iteration, so you only need
65*be691f3bSpatrick   // to set the return status in HandleOneThread if you want to indicate an
66*be691f3bSpatrick   // error. If m_add_return is true, a blank line will be inserted between each
67*be691f3bSpatrick   // of the listings (except the last one.)
68*be691f3bSpatrick 
69*be691f3bSpatrick   virtual bool HandleOneThread(lldb::tid_t, CommandReturnObject &result) = 0;
70*be691f3bSpatrick 
71*be691f3bSpatrick   bool BucketThread(lldb::tid_t tid, std::set<UniqueStack> &unique_stacks,
72*be691f3bSpatrick                     CommandReturnObject &result);
73*be691f3bSpatrick 
74*be691f3bSpatrick   lldb::ReturnStatus m_success_return = lldb::eReturnStatusSuccessFinishResult;
75*be691f3bSpatrick   bool m_unique_stacks = false;
76*be691f3bSpatrick   bool m_add_return = true;
77*be691f3bSpatrick };
78*be691f3bSpatrick 
79*be691f3bSpatrick /// Class similar to \a CommandObjectIterateOverThreads, but which performs
80*be691f3bSpatrick /// an action on multiple threads at once instead of iterating over each thread.
81*be691f3bSpatrick class CommandObjectMultipleThreads : public CommandObjectParsed {
82*be691f3bSpatrick public:
83*be691f3bSpatrick   using CommandObjectParsed::CommandObjectParsed;
84*be691f3bSpatrick 
85*be691f3bSpatrick   bool DoExecute(Args &command, CommandReturnObject &result) override;
86*be691f3bSpatrick 
87*be691f3bSpatrick protected:
88*be691f3bSpatrick   /// Method that handles the command after the main arguments have been parsed.
89*be691f3bSpatrick   ///
90*be691f3bSpatrick   /// \param[in] tids
91*be691f3bSpatrick   ///     The thread ids passed as arguments.
92*be691f3bSpatrick   ///
93*be691f3bSpatrick   /// \return
94*be691f3bSpatrick   ///     A boolean result similar to the one expected from \a DoExecute.
95*be691f3bSpatrick   virtual bool DoExecuteOnThreads(Args &command, CommandReturnObject &result,
96*be691f3bSpatrick                                   llvm::ArrayRef<lldb::tid_t> tids) = 0;
97*be691f3bSpatrick };
98*be691f3bSpatrick 
99*be691f3bSpatrick } // namespace lldb_private
100*be691f3bSpatrick 
101*be691f3bSpatrick #endif // LLDB_SOURCE_COMMANDS_COMMANDOBJECTTHREADUTIL_H
102