xref: /llvm-project/lldb/source/Commands/CommandObjectThreadUtil.h (revision 9c4944095db919580bdc698273065d1c91a98ed8)
1fb19f11eSWalter Erquinigo //===-- CommandObjectThreadUtil.h -------------------------------*- 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 #ifndef LLDB_SOURCE_COMMANDS_COMMANDOBJECTTHREADUTIL_H
10fb19f11eSWalter Erquinigo #define LLDB_SOURCE_COMMANDS_COMMANDOBJECTTHREADUTIL_H
11fb19f11eSWalter Erquinigo 
12fb19f11eSWalter Erquinigo #include "lldb/Interpreter/CommandObjectMultiword.h"
13*9c494409SNikita Popov #include <stack>
14fb19f11eSWalter Erquinigo 
15fb19f11eSWalter Erquinigo namespace lldb_private {
16fb19f11eSWalter Erquinigo 
17fb19f11eSWalter Erquinigo class CommandObjectIterateOverThreads : public CommandObjectParsed {
18fb19f11eSWalter Erquinigo 
19fb19f11eSWalter Erquinigo   class UniqueStack {
20fb19f11eSWalter Erquinigo   public:
UniqueStack(std::stack<lldb::addr_t> stack_frames,uint32_t thread_index_id)21fb19f11eSWalter Erquinigo     UniqueStack(std::stack<lldb::addr_t> stack_frames, uint32_t thread_index_id)
22fb19f11eSWalter Erquinigo         : m_stack_frames(stack_frames) {
23fb19f11eSWalter Erquinigo       m_thread_index_ids.push_back(thread_index_id);
24fb19f11eSWalter Erquinigo     }
25fb19f11eSWalter Erquinigo 
AddThread(uint32_t thread_index_id)26fb19f11eSWalter Erquinigo     void AddThread(uint32_t thread_index_id) const {
27fb19f11eSWalter Erquinigo       m_thread_index_ids.push_back(thread_index_id);
28fb19f11eSWalter Erquinigo     }
29fb19f11eSWalter Erquinigo 
GetUniqueThreadIndexIDs()30fb19f11eSWalter Erquinigo     const std::vector<uint32_t> &GetUniqueThreadIndexIDs() const {
31fb19f11eSWalter Erquinigo       return m_thread_index_ids;
32fb19f11eSWalter Erquinigo     }
33fb19f11eSWalter Erquinigo 
GetRepresentativeThread()34fb19f11eSWalter Erquinigo     lldb::tid_t GetRepresentativeThread() const {
35fb19f11eSWalter Erquinigo       return m_thread_index_ids.front();
36fb19f11eSWalter Erquinigo     }
37fb19f11eSWalter Erquinigo 
38fb19f11eSWalter Erquinigo     friend bool inline operator<(const UniqueStack &lhs,
39fb19f11eSWalter Erquinigo                                  const UniqueStack &rhs) {
40fb19f11eSWalter Erquinigo       return lhs.m_stack_frames < rhs.m_stack_frames;
41fb19f11eSWalter Erquinigo     }
42fb19f11eSWalter Erquinigo 
43fb19f11eSWalter Erquinigo   protected:
44fb19f11eSWalter Erquinigo     // Mark the thread index as mutable, as we don't care about it from a const
45fb19f11eSWalter Erquinigo     // perspective, we only care about m_stack_frames so we keep our std::set
46fb19f11eSWalter Erquinigo     // sorted.
47fb19f11eSWalter Erquinigo     mutable std::vector<uint32_t> m_thread_index_ids;
48fb19f11eSWalter Erquinigo     std::stack<lldb::addr_t> m_stack_frames;
49fb19f11eSWalter Erquinigo   };
50fb19f11eSWalter Erquinigo 
51fb19f11eSWalter Erquinigo public:
52fb19f11eSWalter Erquinigo   CommandObjectIterateOverThreads(CommandInterpreter &interpreter,
53fb19f11eSWalter Erquinigo                                   const char *name, const char *help,
54fb19f11eSWalter Erquinigo                                   const char *syntax, uint32_t flags);
55fb19f11eSWalter Erquinigo 
56fb19f11eSWalter Erquinigo   ~CommandObjectIterateOverThreads() override = default;
57fb19f11eSWalter Erquinigo 
5892d8a28cSPete Lawrence   void DoExecute(Args &command, CommandReturnObject &result) override;
59fb19f11eSWalter Erquinigo 
60fb19f11eSWalter Erquinigo protected:
61fb19f11eSWalter Erquinigo   // Override this to do whatever you need to do for one thread.
62fb19f11eSWalter Erquinigo   //
63fb19f11eSWalter Erquinigo   // If you return false, the iteration will stop, otherwise it will proceed.
64fb19f11eSWalter Erquinigo   // The result is set to m_success_return (defaults to
65fb19f11eSWalter Erquinigo   // eReturnStatusSuccessFinishResult) before the iteration, so you only need
66fb19f11eSWalter Erquinigo   // to set the return status in HandleOneThread if you want to indicate an
67fb19f11eSWalter Erquinigo   // error. If m_add_return is true, a blank line will be inserted between each
68fb19f11eSWalter Erquinigo   // of the listings (except the last one.)
69fb19f11eSWalter Erquinigo 
70fb19f11eSWalter Erquinigo   virtual bool HandleOneThread(lldb::tid_t, CommandReturnObject &result) = 0;
71fb19f11eSWalter Erquinigo 
72fb19f11eSWalter Erquinigo   bool BucketThread(lldb::tid_t tid, std::set<UniqueStack> &unique_stacks,
73fb19f11eSWalter Erquinigo                     CommandReturnObject &result);
74fb19f11eSWalter Erquinigo 
75fb19f11eSWalter Erquinigo   lldb::ReturnStatus m_success_return = lldb::eReturnStatusSuccessFinishResult;
76fb19f11eSWalter Erquinigo   bool m_unique_stacks = false;
77fb19f11eSWalter Erquinigo   bool m_add_return = true;
78fb19f11eSWalter Erquinigo };
79fb19f11eSWalter Erquinigo 
800b697561SWalter Erquinigo /// Class similar to \a CommandObjectIterateOverThreads, but which performs
810b697561SWalter Erquinigo /// an action on multiple threads at once instead of iterating over each thread.
820b697561SWalter Erquinigo class CommandObjectMultipleThreads : public CommandObjectParsed {
830b697561SWalter Erquinigo public:
846564ca18SWalter Erquinigo   CommandObjectMultipleThreads(CommandInterpreter &interpreter,
856564ca18SWalter Erquinigo                                const char *name, const char *help,
866564ca18SWalter Erquinigo                                const char *syntax, uint32_t flags);
870b697561SWalter Erquinigo 
8892d8a28cSPete Lawrence   void DoExecute(Args &command, CommandReturnObject &result) override;
890b697561SWalter Erquinigo 
900b697561SWalter Erquinigo protected:
910b697561SWalter Erquinigo   /// Method that handles the command after the main arguments have been parsed.
920b697561SWalter Erquinigo   ///
930b697561SWalter Erquinigo   /// \param[in] tids
940b697561SWalter Erquinigo   ///     The thread ids passed as arguments.
950b697561SWalter Erquinigo   ///
960b697561SWalter Erquinigo   /// \return
970b697561SWalter Erquinigo   ///     A boolean result similar to the one expected from \a DoExecute.
980b697561SWalter Erquinigo   virtual bool DoExecuteOnThreads(Args &command, CommandReturnObject &result,
99bf9f21a2SWalter Erquinigo                                   llvm::ArrayRef<lldb::tid_t> tids) = 0;
1000b697561SWalter Erquinigo };
1010b697561SWalter Erquinigo 
102fb19f11eSWalter Erquinigo } // namespace lldb_private
103fb19f11eSWalter Erquinigo 
104fb19f11eSWalter Erquinigo #endif // LLDB_SOURCE_COMMANDS_COMMANDOBJECTTHREADUTIL_H
105