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