Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
TaskGroup.hh
Go to the documentation of this file.
1//
2// MIT License
3// Copyright (c) 2020 Jonathan R. Madsen
4// Permission is hereby granted, free of charge, to any person obtaining a copy
5// of this software and associated documentation files (the "Software"), to deal
6// in the Software without restriction, including without limitation the rights
7// to use, copy, modify, merge, publish, distribute, sublicense, and
8// copies of the Software, and to permit persons to whom the Software is
9// furnished to do so, subject to the following conditions:
10// The above copyright notice and this permission notice shall be included in
11// all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED
12// "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT
13// LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
14// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
15// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
16// ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
17// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
18//
19//
20// ---------------------------------------------------------------
21// Tasking class header file
22//
23// Class Description:
24//
25// This file creates the a class for handling a group of tasks that
26// can be independently joined
27//
28// ---------------------------------------------------------------
29// Author: Jonathan Madsen (Feb 13th 2018)
30// ---------------------------------------------------------------
31
32#pragma once
33
34#include "PTL/AutoLock.hh"
35#ifndef G4GMAKE
36#include "PTL/Config.hh"
37#endif
38#include "PTL/JoinFunction.hh"
40#include "PTL/Task.hh"
41#include "PTL/ThreadData.hh"
42#include "PTL/ThreadPool.hh"
43#include "PTL/Types.hh"
44#include "PTL/VTask.hh"
45#include "PTL/VUserTaskQueue.hh"
47
48#include <atomic>
49#include <chrono>
50#include <cstdint>
51#include <cstdio>
52#include <functional>
53#include <future>
54#include <iostream>
55#include <memory>
56#include <mutex>
57#include <sstream> // IWYU pragma: keep
58#include <stdexcept>
59#include <thread>
60#include <type_traits>
61#include <utility>
62#include <vector>
63
64#if defined(PTL_USE_TBB)
65# include <tbb/task_group.h> // IWYU pragma: keep
66#endif
67
68namespace PTL
69{
70namespace internal
71{
72std::atomic_uintmax_t&
74
77
78intmax_t
80} // namespace internal
81
82template <typename Tp, typename Arg = Tp, intmax_t MaxDepth = 0>
84{
85public:
86 //------------------------------------------------------------------------//
87 template <typename Up>
88 using container_type = std::vector<Up>;
89
90 using tid_type = std::thread::id;
91 using size_type = uintmax_t;
92 using lock_t = Mutex;
93 using atomic_int = std::atomic_intmax_t;
94 using atomic_uint = std::atomic_uintmax_t;
95 using condition_t = Condition;
97 using result_type = Tp;
98 using task_pointer = std::shared_ptr<TaskFuture<ArgTp>>;
101 using promise_type = std::promise<ArgTp>;
102 using future_type = std::future<ArgTp>;
103 using packaged_task_type = std::packaged_task<ArgTp()>;
106 using iterator = typename future_list_t::iterator;
107 using reverse_iterator = typename future_list_t::reverse_iterator;
108 using const_iterator = typename future_list_t::const_iterator;
109 using const_reverse_iterator = typename future_list_t::const_reverse_iterator;
110 //------------------------------------------------------------------------//
111 template <typename... Args>
113 //------------------------------------------------------------------------//
114
115public:
116 // Constructor
117 template <typename Func>
119
120 template <typename Up = Tp>
122 enable_if_t<std::is_void<Up>::value, int> = 0);
123
124 // Destructor
126
127 // delete copy-construct
128 TaskGroup(const this_type&) = delete;
129 // define move-construct
130 // NOLINTNEXTLINE(performance-noexcept-move-constructor)
131 TaskGroup(this_type&& rhs) = default;
132 // delete copy-assign
133 TaskGroup& operator=(const this_type& rhs) = delete;
134 // define move-assign
135 // NOLINTNEXTLINE(performance-noexcept-move-constructor)
136 TaskGroup& operator=(this_type&& rhs) = default;
137
138public:
139 template <typename Up>
140 std::shared_ptr<Up> operator+=(std::shared_ptr<Up>&& _task);
141
142 // wait to finish
143 void wait();
144
145 // increment (prefix)
146 intmax_t operator++() { return ++(m_tot_task_count); }
147 intmax_t operator++(int) { return (m_tot_task_count)++; }
148 intmax_t operator--() { return --(m_tot_task_count); }
149 intmax_t operator--(int) { return (m_tot_task_count)--; }
150
151 // size
152 intmax_t size() const { return m_tot_task_count.load(); }
153
154 // get the locks/conditions
157
158 // identifier
159 uintmax_t id() const { return m_id; }
160
161 // thread pool
162 void set_pool(ThreadPool* tp) { m_pool = tp; }
163 ThreadPool*& pool() { return m_pool; }
164 ThreadPool* pool() const { return m_pool; }
165
166 bool is_native_task_group() const { return (m_tbb_task_group) == nullptr; }
167 bool is_main() const { return this_tid() == m_main_tid; }
168
169 // check if any tasks are still pending
170 intmax_t pending() { return m_tot_task_count.load(); }
171
172 static void set_verbose(int level) { f_verbose = level; }
173
175
176 void notify();
178
179 void reserve(size_t _n)
180 {
181 m_task_list.reserve(_n);
182 m_future_list.reserve(_n);
183 }
184
185public:
186 template <typename Func, typename... Args>
187 std::shared_ptr<task_type<Args...>> wrap(Func func, Args... args)
188 {
189 return operator+=(std::make_shared<task_type<Args...>>(
190 is_native_task_group(), m_depth, std::move(func), std::move(args)...));
191 }
192
193 template <typename Func, typename... Args, typename Up = Tp>
194 enable_if_t<std::is_void<Up>::value, void> exec(Func func, Args... args);
195
196 template <typename Func, typename... Args, typename Up = Tp>
197 enable_if_t<!std::is_void<Up>::value, void> exec(Func func, Args... args);
198
199 template <typename Func, typename... Args>
200 void run(Func func, Args... args)
201 {
202 exec(std::move(func), std::move(args)...);
203 }
204
205protected:
206 template <typename Up, typename Func, typename... Args>
207 enable_if_t<std::is_void<Up>::value, void> local_exec(Func func, Args... args);
208
209 template <typename Up, typename Func, typename... Args>
210 enable_if_t<!std::is_void<Up>::value, void> local_exec(Func func, Args... args);
211
212 // shorter typedefs
217
218public:
219 //------------------------------------------------------------------------//
220 // Get tasks with non-void return types
221 //
223 const future_list_t& get_tasks() const { return m_future_list; }
224
225 //------------------------------------------------------------------------//
226 // iterate over tasks with return type
227 //
228 itr_t begin() { return m_future_list.begin(); }
229 itr_t end() { return m_future_list.end(); }
230 citr_t begin() const { return m_future_list.begin(); }
231 citr_t end() const { return m_future_list.end(); }
232 citr_t cbegin() const { return m_future_list.begin(); }
233 citr_t cend() const { return m_future_list.end(); }
234 ritr_t rbegin() { return m_future_list.rbegin(); }
235 ritr_t rend() { return m_future_list.rend(); }
236 critr_t rbegin() const { return m_future_list.rbegin(); }
237 critr_t rend() const { return m_future_list.rend(); }
238
239 //------------------------------------------------------------------------//
240 // wait to finish
241 template <typename Up = Tp, enable_if_t<!std::is_void<Up>::value, int> = 0>
242 inline Up join(Up accum = {});
243 //------------------------------------------------------------------------//
244 // wait to finish
245 template <typename Up = Tp, typename Rp = Arg,
246 enable_if_t<std::is_void<Up>::value && std::is_void<Rp>::value, int> = 0>
247 inline void join();
248 //------------------------------------------------------------------------//
249 // wait to finish
250 template <typename Up = Tp, typename Rp = Arg,
251 enable_if_t<std::is_void<Up>::value && !std::is_void<Rp>::value, int> = 0>
252 inline void join();
253 //------------------------------------------------------------------------//
254 // clear the task result history
255 void clear();
256
257protected:
258 //------------------------------------------------------------------------//
259 // get the thread id
260 static tid_type this_tid() { return std::this_thread::get_id(); }
261
262 //------------------------------------------------------------------------//
263 // get the task count
265 const atomic_int& task_count() const { return m_tot_task_count; }
266
267protected:
268 static int f_verbose;
269 // Private variables
272 tid_type m_main_tid = std::this_thread::get_id();
281
282private:
283 void internal_update();
284};
285
286} // namespace PTL
287namespace PTL
288{
289template <typename Tp, typename Arg, intmax_t MaxDepth>
290template <typename Func>
292: m_join{ std::forward<Func>(_join) }
293, m_pool{ _tp }
294{
295 internal_update();
296}
297
298template <typename Tp, typename Arg, intmax_t MaxDepth>
299template <typename Up>
301 enable_if_t<std::is_void<Up>::value, int>)
302: m_join{ []() {} }
303, m_pool{ _tp }
304{
305 internal_update();
306}
307
308// Destructor
309template <typename Tp, typename Arg, intmax_t MaxDepth>
311{
312 {
313 // task will decrement counter and then acquire the lock to notify
314 // condition variable so acquiring lock here will prevent the
315 // task group from being destroyed before this is completed
316 AutoLock _lk{ m_task_lock, std::defer_lock };
317 if(!_lk.owns_lock())
318 _lk.lock();
319 }
320
322 {
323 auto* _arena = m_pool->get_task_arena();
324 _arena->execute([this]() { this->m_tbb_task_group->wait(); });
325 }
326 delete m_tbb_task_group;
327 this->clear();
328}
329
330template <typename Tp, typename Arg, intmax_t MaxDepth>
331template <typename Up>
332std::shared_ptr<Up>
334{
335 // thread-safe increment of tasks in task group
336 operator++();
337 // copy the shared pointer to abstract instance
338 m_task_list.push_back(_task);
339 // return the derived instance
340 return std::move(_task);
341}
342
343template <typename Tp, typename Arg, intmax_t MaxDepth>
344void
346{
347 auto _dtor = ScopeDestructor{ [&]() {
349 {
350 auto* _arena = m_pool->get_task_arena();
351 _arena->execute([this]() { this->m_tbb_task_group->wait(); });
352 }
353 } };
354
356 if(!data)
357 return;
358
359 // if no pool was initially present at creation
360 if(!m_pool)
361 {
362 // check for master MT run-manager
364
365 // if no thread pool created
366 if(!m_pool)
367 {
368 if(f_verbose > 0)
369 {
370 fprintf(stderr, "%s @ %i :: Warning! nullptr to thread-pool (%p)\n",
371 __FUNCTION__, __LINE__, static_cast<void*>(m_pool));
372 std::cerr << __FUNCTION__ << "@" << __LINE__ << " :: Warning! "
373 << "nullptr to thread pool!" << std::endl;
374 }
375 return;
376 }
377 }
378
379 ThreadPool* tpool = (m_pool) ? m_pool : data->thread_pool;
380 VUserTaskQueue* taskq = (tpool) ? tpool->get_queue() : data->current_queue;
381
382 bool _is_main = data->is_main;
383 bool _within_task = data->within_task;
384
385 auto is_active_state = [&]() {
386 return (tpool->state()->load(std::memory_order_relaxed) !=
387 thread_pool::state::STOPPED);
388 };
389
390 auto execute_this_threads_tasks = [&]() {
391 if(!taskq)
392 return;
393
394 // only want to process if within a task
395 if((!_is_main || tpool->size() < 2) && _within_task)
396 {
397 int bin = static_cast<int>(taskq->GetThreadBin());
398 // const auto nitr = (tpool) ? tpool->size() :
399 // Thread::hardware_concurrency();
400 while(this->pending() > 0)
401 {
402 if(!taskq->empty())
403 {
404 auto _task = taskq->GetTask(bin);
405 if(_task)
406 (*_task)();
407 }
408 }
409 }
410 };
411
412 // checks for validity
414 {
415 // for external threads
416 if(!_is_main || tpool->size() < 2)
417 return;
418 }
419 else if(f_verbose > 0)
420 {
421 if(!tpool || !taskq)
422 {
423 // something is wrong, didn't create thread-pool?
424 fprintf(stderr,
425 "%s @ %i :: Warning! nullptr to thread data (%p) or task-queue "
426 "(%p)\n",
427 __FUNCTION__, __LINE__, static_cast<void*>(tpool),
428 static_cast<void*>(taskq));
429 }
430 // return if thread pool isn't built
431 else if(is_native_task_group() && !tpool->is_alive())
432 {
433 fprintf(stderr, "%s @ %i :: Warning! thread-pool is not alive!\n",
434 __FUNCTION__, __LINE__);
435 }
436 else if(!is_active_state())
437 {
438 fprintf(stderr, "%s @ %i :: Warning! thread-pool is not active!\n",
439 __FUNCTION__, __LINE__);
440 }
441 }
442
443 intmax_t wake_size = 2;
444 AutoLock _lock(m_task_lock, std::defer_lock);
445
446 while(is_active_state())
447 {
448 execute_this_threads_tasks();
449
450 // while loop protects against spurious wake-ups
451 while(_is_main && pending() > 0 && is_active_state())
452 {
453 // auto _wake = [&]() { return (wake_size > pending() ||
454 // !is_active_state());
455 // };
456
457 // lock before sleeping on condition
458 if(!_lock.owns_lock())
459 _lock.lock();
460
461 // Wait until signaled that a task has been competed
462 // Unlock mutex while wait, then lock it back when signaled
463 // when true, this wakes the thread
464 if(pending() >= wake_size)
465 {
466 m_task_cond.wait(_lock);
467 }
468 else
469 {
470 m_task_cond.wait_for(_lock, std::chrono::microseconds(100));
471 }
472 // unlock
473 if(_lock.owns_lock())
474 _lock.unlock();
475 }
476
477 // if pending is not greater than zero, we are joined
478 if(pending() <= 0)
479 break;
480 }
481
482 if(_lock.owns_lock())
483 _lock.unlock();
484
485 intmax_t ntask = this->task_count().load();
486 if(ntask > 0)
487 {
488 std::stringstream ss;
489 ss << "\nWarning! Join operation issue! " << ntask << " tasks still "
490 << "are running!" << std::endl;
491 std::cerr << ss.str();
492 this->wait();
493 }
494}
495
496template <typename Tp, typename Arg, intmax_t MaxDepth>
499{
500 auto& _counter = m_tot_task_count;
501 auto& _task_cond = task_cond();
502 auto& _task_lock = task_lock();
503 return ScopeDestructor{ [&_task_cond, &_task_lock, &_counter]() {
504 auto _count = --(_counter);
505 if(_count < 1)
506 {
507 AutoLock _lk{ _task_lock };
508 _task_cond.notify_all();
509 }
510 } };
511}
512
513template <typename Tp, typename Arg, intmax_t MaxDepth>
514void
520
521template <typename Tp, typename Arg, intmax_t MaxDepth>
522void
528
529template <typename Tp, typename Arg, intmax_t MaxDepth>
530template <typename Func, typename... Args, typename Up>
533{
534 if(MaxDepth > 0 && !m_tbb_task_group && ThreadData::GetInstance() &&
535 ThreadData::GetInstance()->task_depth > MaxDepth)
536 {
537 local_exec<Tp>(std::move(func), std::move(args)...);
538 }
539 else
540 {
541 auto& _counter = m_tot_task_count;
542 auto& _task_cond = task_cond();
543 auto& _task_lock = task_lock();
544 auto _task = wrap([&_task_cond, &_task_lock, &_counter, func, args...]() {
545 auto* _tdata = ThreadData::GetInstance();
546 if(_tdata)
547 ++(_tdata->task_depth);
548 func(args...);
549 auto _count = --(_counter);
550 if(_tdata)
551 --(_tdata->task_depth);
552 if(_count < 1)
553 {
554 AutoLock _lk{ _task_lock };
555 _task_cond.notify_all();
556 }
557 });
558
560 {
561 auto* _arena = m_pool->get_task_arena();
562 auto* _tbb_task_group = m_tbb_task_group;
563 auto* _ptask = _task.get();
564 _arena->execute([_tbb_task_group, _ptask]() {
565 _tbb_task_group->run([_ptask]() { (*_ptask)(); });
566 });
567 }
568 else
569 {
570 m_pool->add_task(std::move(_task));
571 }
572 }
573}
574template <typename Tp, typename Arg, intmax_t MaxDepth>
575template <typename Func, typename... Args, typename Up>
576enable_if_t<!std::is_void<Up>::value, void>
578{
579 if(MaxDepth > 0 && !m_tbb_task_group && ThreadData::GetInstance() &&
580 ThreadData::GetInstance()->task_depth > MaxDepth)
581 {
582 local_exec<Tp>(std::move(func), std::move(args)...);
583 }
584 else
585 {
586 auto& _counter = m_tot_task_count;
587 auto& _task_cond = task_cond();
588 auto& _task_lock = task_lock();
589 auto _task = wrap([&_task_cond, &_task_lock, &_counter, func, args...]() {
590 auto* _tdata = ThreadData::GetInstance();
591 if(_tdata)
592 ++(_tdata->task_depth);
593 auto&& _ret = func(args...);
594 auto _count = --(_counter);
595 if(_tdata)
596 --(_tdata->task_depth);
597 if(_count < 1)
598 {
599 AutoLock _lk{ _task_lock };
600 _task_cond.notify_all();
601 }
602 return std::forward<decltype(_ret)>(_ret);
603 });
604
606 {
607 auto* _arena = m_pool->get_task_arena();
608 auto* _tbb_task_group = m_tbb_task_group;
609 auto* _ptask = _task.get();
610 _arena->execute([_tbb_task_group, _ptask]() {
611 _tbb_task_group->run([_ptask]() { (*_ptask)(); });
612 });
613 }
614 else
615 {
616 m_pool->add_task(std::move(_task));
617 }
618 }
619}
620
621template <typename Tp, typename Arg, intmax_t MaxDepth>
622template <typename Up, typename Func, typename... Args>
623enable_if_t<std::is_void<Up>::value, void>
625{
626 auto* _tdata = ThreadData::GetInstance();
627 if(_tdata)
628 ++(_tdata->task_depth);
629 promise_type _p{};
630 m_future_list.emplace_back(_p.get_future());
631 func(args...);
632 _p.set_value();
633 if(_tdata)
634 --(_tdata->task_depth);
635}
636
637template <typename Tp, typename Arg, intmax_t MaxDepth>
638template <typename Up, typename Func, typename... Args>
641{
642 auto* _tdata = ThreadData::GetInstance();
643 if(_tdata)
644 ++(_tdata->task_depth);
645 promise_type _p{};
646 m_future_list.emplace_back(_p.get_future());
647 _p.set_value(func(args...));
648 if(_tdata)
649 --(_tdata->task_depth);
650}
651
652template <typename Tp, typename Arg, intmax_t MaxDepth>
653template <typename Up, enable_if_t<!std::is_void<Up>::value, int>>
654inline Up
656{
657 this->wait();
658 for(auto& itr : m_task_list)
659 {
660 using RetT = decay_t<decltype(itr->get())>;
661 accum = std::move(m_join(std::ref(accum), std::forward<RetT>(itr->get())));
662 }
663 for(auto& itr : m_future_list)
664 {
665 using RetT = decay_t<decltype(itr.get())>;
666 accum = std::move(m_join(std::ref(accum), std::forward<RetT>(itr.get())));
667 }
668 this->clear();
669 return accum;
670}
671
672template <typename Tp, typename Arg, intmax_t MaxDepth>
673template <typename Up, typename Rp,
674 enable_if_t<std::is_void<Up>::value && std::is_void<Rp>::value, int>>
675inline void
677{
678 this->wait();
679 for(auto& itr : m_task_list)
680 itr->get();
681 for(auto& itr : m_future_list)
682 itr.get();
683 m_join();
684 this->clear();
685}
686
687template <typename Tp, typename Arg, intmax_t MaxDepth>
688template <typename Up, typename Rp,
689 enable_if_t<std::is_void<Up>::value && !std::is_void<Rp>::value, int>>
690inline void
692{
693 this->wait();
694 for(auto& itr : m_task_list)
695 {
696 using RetT = decay_t<decltype(itr->get())>;
697 m_join(std::forward<RetT>(itr->get()));
698 }
699 for(auto& itr : m_future_list)
700 {
701 using RetT = decay_t<decltype(itr.get())>;
702 m_join(std::forward<RetT>(itr.get()));
703 }
704 this->clear();
705}
706
707template <typename Tp, typename Arg, intmax_t MaxDepth>
708void
714
715template <typename Tp, typename Arg, intmax_t MaxDepth>
716void
717TaskGroup<Tp, Arg, MaxDepth>::internal_update()
718{
719 if(!m_pool)
721
722 if(!m_pool)
723 {
724 std::stringstream ss{};
725 ss << "[TaskGroup]> " << __FUNCTION__ << "@" << __LINE__
726 << " :: nullptr to thread pool";
727 throw std::runtime_error(ss.str());
728 }
729
730 if(m_pool->is_tbb_threadpool())
731 {
732 m_tbb_task_group = new tbb_task_group_t{};
733 }
734}
735
736template <typename Tp, typename Arg, intmax_t MaxDepth>
738
739} // namespace PTL
ritr_t rend()
Definition TaskGroup.hh:235
intmax_t operator++()
Definition TaskGroup.hh:146
TaskGroup & operator=(this_type &&rhs)=default
std::atomic_intmax_t atomic_int
Definition TaskGroup.hh:93
container_type< future_type > future_list_t
Definition TaskGroup.hh:104
typename future_list_t::reverse_iterator reverse_iterator
Definition TaskGroup.hh:107
ThreadPool * pool() const
Definition TaskGroup.hh:164
std::promise< ArgTp > promise_type
Definition TaskGroup.hh:101
intmax_t size() const
Definition TaskGroup.hh:152
Up join(Up accum={})
Definition TaskGroup.hh:655
tbb_task_group_t * m_tbb_task_group
Definition TaskGroup.hh:278
std::atomic_uintmax_t atomic_uint
Definition TaskGroup.hh:94
std::vector< Up > container_type
Definition TaskGroup.hh:88
decay_t< Arg > ArgTp
Definition TaskGroup.hh:96
ritr_t rbegin()
Definition TaskGroup.hh:234
TaskGroup(const this_type &)=delete
const atomic_int & task_count() const
Definition TaskGroup.hh:265
typename JoinFunction< Tp, Arg >::Type join_type
Definition TaskGroup.hh:105
citr_t end() const
Definition TaskGroup.hh:231
citr_t cbegin() const
Definition TaskGroup.hh:232
intmax_t operator++(int)
Definition TaskGroup.hh:147
TaskGroup(ThreadPool *_tp=internal::get_default_threadpool(), enable_if_t< std::is_void< Up >::value, int >=0)
Definition TaskGroup.hh:300
const future_list_t & get_tasks() const
Definition TaskGroup.hh:223
bool is_native_task_group() const
Definition TaskGroup.hh:166
citr_t cend() const
Definition TaskGroup.hh:233
typename future_list_t::const_iterator const_iterator
Definition TaskGroup.hh:108
intmax_t operator--()
Definition TaskGroup.hh:148
std::packaged_task< ArgTp()> packaged_task_type
Definition TaskGroup.hh:103
critr_t rend() const
Definition TaskGroup.hh:237
static void set_verbose(int level)
Definition TaskGroup.hh:172
enable_if_t< std::is_void< Up >::value, void > local_exec(Func func, Args... args)
Definition TaskGroup.hh:624
std::thread::id tid_type
Definition TaskGroup.hh:90
std::shared_ptr< TaskFuture< ArgTp > > task_pointer
Definition TaskGroup.hh:98
TaskGroup & operator=(const this_type &rhs)=delete
void notify_all()
Definition TaskGroup.hh:523
Condition condition_t
Definition TaskGroup.hh:95
static tid_type this_tid()
Definition TaskGroup.hh:260
atomic_int & task_count()
Definition TaskGroup.hh:264
const_reverse_iterator critr_t
Definition TaskGroup.hh:216
ThreadPool *& pool()
Definition TaskGroup.hh:163
condition_t & task_cond()
Definition TaskGroup.hh:156
intmax_t operator--(int)
Definition TaskGroup.hh:149
std::shared_ptr< task_type< Args... > > wrap(Func func, Args... args)
Definition TaskGroup.hh:187
void run(Func func, Args... args)
Definition TaskGroup.hh:200
Task< ArgTp, decay_t< Args >... > task_type
Definition TaskGroup.hh:112
future_list_t & get_tasks()
Definition TaskGroup.hh:222
bool is_main() const
Definition TaskGroup.hh:167
enable_if_t<!std::is_void< Up >::value, void > local_exec(Func func, Args... args)
Definition TaskGroup.hh:640
uintmax_t size_type
Definition TaskGroup.hh:91
typename future_list_t::iterator iterator
Definition TaskGroup.hh:106
TaskGroup(Func &&_join, ThreadPool *_tp=internal::get_default_threadpool())
Definition TaskGroup.hh:291
lock_t & task_lock()
Definition TaskGroup.hh:155
enable_if_t< std::is_void< Up >::value, void > exec(Func func, Args... args)
Definition TaskGroup.hh:532
uintmax_t id() const
Definition TaskGroup.hh:159
citr_t begin() const
Definition TaskGroup.hh:230
typename future_list_t::const_reverse_iterator const_reverse_iterator
Definition TaskGroup.hh:109
void set_pool(ThreadPool *tp)
Definition TaskGroup.hh:162
std::future< ArgTp > future_type
Definition TaskGroup.hh:102
void reserve(size_t _n)
Definition TaskGroup.hh:179
critr_t rbegin() const
Definition TaskGroup.hh:236
TaskGroup(this_type &&rhs)=default
TaskGroup< Tp, Arg, MaxDepth > this_type
Definition TaskGroup.hh:100
container_type< task_pointer > task_list_t
Definition TaskGroup.hh:99
intmax_t pending()
Definition TaskGroup.hh:170
enable_if_t<!std::is_void< Up >::value, void > exec(Func func, Args... args)
Definition TaskGroup.hh:577
ScopeDestructor get_scope_destructor()
Definition TaskGroup.hh:498
std::shared_ptr< Up > operator+=(std::shared_ptr< Up > &&_task)
Definition TaskGroup.hh:333
The task class is supplied to thread_pool.
Definition Task.hh:131
static ThreadData *& GetInstance()
Definition ThreadData.cc:31
VUserTaskQueue * current_queue
ThreadPool * thread_pool
const pool_state_type & state() const
task_queue_t * get_queue() const
size_type size() const
virtual task_pointer GetTask(intmax_t subq=-1, intmax_t nitr=-1)=0
virtual bool empty() const =0
virtual intmax_t GetThreadBin() const =0
intmax_t get_task_depth()
Definition TaskGroup.cc:63
ThreadPool * get_default_threadpool()
Definition TaskGroup.cc:50
std::atomic_uintmax_t & task_group_counter()
Definition TaskGroup.cc:43
Backports of C++ language features for use with C++11 compilers.
Definition AutoLock.hh:255
TemplateAutoLock< Mutex > AutoLock
Definition AutoLock.hh:479
typename std::enable_if< B, T >::type enable_if_t
tbb::task_group tbb_task_group_t
typename std::decay< T >::type decay_t
std::function< JoinT(JoinT &, JoinArg &&)> Type