3 Copyright (C) 2019-2023 Free Software Foundation, Inc.
5 This file is part of GDB.
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
20 #ifndef GDBSUPPORT_THREAD_POOL_H
21 #define GDBSUPPORT_THREAD_POOL_H
30 #include <condition_variable>
33 #include "gdbsupport/gdb_optional.h"
40 /* Simply use the standard future. */
42 using future
= std::future
<T
>;
44 /* ... and the standard future_status. */
45 using future_status
= std::future_status
;
47 #else /* CXX_STD_THREAD */
49 /* A compatibility enum for std::future_status. This is just the
50 subset needed by gdb. */
51 enum class future_status
57 /* A compatibility wrapper for std::future. Once <thread> and
58 <future> are available in all GCC builds -- should that ever happen
59 -- this can be removed. GCC does not implement threading for
60 MinGW, see https://gcc.gnu.org/bugzilla/show_bug.cgi?id=93687.
62 Meanwhile, in this mode, there are no threads. Tasks submitted to
63 the thread pool are invoked immediately and their result is stored
64 here. The base template here simply wraps a T and provides some
65 std::future compatibility methods. The provided methods are chosen
66 based on what GDB needs presently. */
73 explicit future (T value
)
74 : m_value (std::move (value
))
79 future (future
&&other
) = default;
80 future (const future
&other
) = delete;
81 future
&operator= (future
&&other
) = default;
82 future
&operator= (const future
&other
) = delete;
84 void wait () const { }
86 template<class Rep
, class Period
>
87 future_status
wait_for (const std::chrono::duration
<Rep
,Period
> &duration
)
90 return future_status::ready
;
93 T
get () { return std::move (m_value
); }
100 /* A specialization for void. */
106 void wait () const { }
108 template<class Rep
, class Period
>
109 future_status
wait_for (const std::chrono::duration
<Rep
,Period
> &duration
)
112 return future_status::ready
;
118 #endif /* CXX_STD_THREAD */
123 There is a single global thread pool, see g_thread_pool. Tasks can
124 be submitted to the thread pool. They will be processed in worker
125 threads as time allows. */
129 /* The sole global thread pool. */
130 static thread_pool
*g_thread_pool
;
133 DISABLE_COPY_AND_ASSIGN (thread_pool
);
135 /* Set the thread count of this thread pool. By default, no threads
136 are created -- the thread count must be set first. */
137 void set_thread_count (size_t num_threads
);
139 /* Return the number of executing threads. */
140 size_t thread_count () const
143 return m_thread_count
;
149 /* Post a task to the thread pool. A future is returned, which can
150 be used to wait for the result. */
151 future
<void> post_task (std::function
<void ()> &&func
)
154 std::packaged_task
<void ()> task (std::move (func
));
155 future
<void> result
= task
.get_future ();
156 do_post_task (std::packaged_task
<void ()> (std::move (task
)));
161 #endif /* CXX_STD_THREAD */
164 /* Post a task to the thread pool. A future is returned, which can
165 be used to wait for the result. */
167 future
<T
> post_task (std::function
<T ()> &&func
)
170 std::packaged_task
<T ()> task (std::move (func
));
171 future
<T
> result
= task
.get_future ();
172 do_post_task (std::packaged_task
<void ()> (std::move (task
)));
175 return future
<T
> (func ());
176 #endif /* CXX_STD_THREAD */
181 thread_pool () = default;
184 /* The callback for each worker thread. */
185 void thread_function ();
187 /* Post a task to the thread pool. A future is returned, which can
188 be used to wait for the result. */
189 void do_post_task (std::packaged_task
<void ()> &&func
);
191 /* The current thread count. */
192 size_t m_thread_count
= 0;
194 /* A convenience typedef for the type of a task. */
195 typedef std::packaged_task
<void ()> task_t
;
197 /* The tasks that have not been processed yet. An optional is used
198 to represent a task. If the optional is empty, then this means
199 that the receiving thread should terminate. If the optional is
200 non-empty, then it is an actual task to evaluate. */
201 std::queue
<optional
<task_t
>> m_tasks
;
203 /* A condition variable and mutex that are used for communication
204 between the main thread and the worker threads. */
205 std::condition_variable m_tasks_cv
;
206 std::mutex m_tasks_mutex
;
207 #endif /* CXX_STD_THREAD */
212 #endif /* GDBSUPPORT_THREAD_POOL_H */