3 //==========================================================================
7 * @author Douglas C. Schmidt <d.schmidt@vanderbilt.edu>
9 //==========================================================================
13 #include /**/ "ace/pre.h"
15 #include /**/ "ace/ACE_export.h"
17 #if !defined (ACE_LACKS_PRAGMA_ONCE)
19 #endif /* ACE_LACKS_PRAGMA_ONCE */
21 #include /**/ "ace/config-all.h"
23 // ACE platform supports some form of threading.
24 #if !defined (ACE_HAS_THREADS)
26 #include "ace/OS_NS_errno.h"
28 ACE_BEGIN_VERSIONED_NAMESPACE_DECL
33 * @brief This is a no-op to make ACE "syntactically consistent."
35 class ACE_Export ACE_Barrier
38 ACE_Barrier (unsigned int, const ACE_TCHAR
* = 0, void * = 0) {}
39 ~ACE_Barrier () = default;
40 int wait () { ACE_NOTSUP_RETURN (-1); }
44 ACE_END_VERSIONED_NAMESPACE_DECL
46 #else /* ACE_HAS_THREADS */
48 #include "ace/Condition_Thread_Mutex.h"
50 ACE_BEGIN_VERSIONED_NAMESPACE_DECL
52 struct ACE_Export ACE_Sub_Barrier
55 ACE_Sub_Barrier (unsigned int count
,
56 ACE_Thread_Mutex
&lock
,
57 const ACE_TCHAR
*name
= 0,
60 ~ACE_Sub_Barrier () = default;
62 /// True if this generation of the barrier is done.
63 ACE_Condition_Thread_Mutex barrier_finished_
;
65 /// Number of threads that are still running.
68 /// Dump the state of an object.
71 /// Declare the dynamic allocation hooks.
72 ACE_ALLOC_HOOK_DECLARE
;
78 * @brief Implements "barrier synchronization".
80 * This class allows <count> number of threads to synchronize
81 * their completion of (one round of) a task, which is known as
82 * "barrier synchronization". After all the threads call <wait()>
83 * on the barrier they are all atomically released and can begin a new
86 * This implementation uses a "sub-barrier generation numbering"
87 * scheme to avoid overhead and to ensure that all threads wait to
88 * leave the barrier correct. This code is based on an article from
89 * SunOpsis Vol. 4, No. 1 by Richard Marejka
90 * (Richard.Marejka@canada.sun.com).
92 class ACE_Export ACE_Barrier
95 /// Initialize the barrier to synchronize @a count threads.
96 ACE_Barrier (unsigned int count
,
97 const ACE_TCHAR
*name
= 0,
100 /// Default destructor.
101 ~ACE_Barrier () = default;
103 /// Block the caller until all @c count threads have called @c wait and
104 /// then allow all the caller threads to continue in parallel.
106 /// @retval 0 after successfully waiting for all threads to wait.
107 /// @retval -1 if an error occurs or the barrier is shut
108 /// down (@sa shutdown ()).
111 /// Shut the barrier down, aborting the wait of all waiting threads.
112 /// Any threads waiting on the barrier when it is shut down will return with
113 /// value -1, errno ESHUTDOWN.
115 /// @retval 0 for success, -1 if already shut down.
117 /// @since ACE beta 5.4.9.
120 /// Dump the state of an object.
123 /// Declare the dynamic allocation hooks.
124 ACE_ALLOC_HOOK_DECLARE
;
127 /// Serialize access to the barrier state.
128 ACE_Thread_Mutex lock_
;
130 /// Either 0 or 1, depending on whether we are the first generation
131 /// of waiters or the next generation of waiters.
132 int current_generation_
;
134 /// Total number of threads that can be waiting at any one time.
138 * We keep two @c sub_barriers, one for the first "generation" of
139 * waiters, and one for the next "generation" of waiters. This
140 * efficiently solves the problem of what to do if all the first
141 * generation waiters don't leave the barrier before one of the
142 * threads calls wait() again (i.e., starts up the next generation
145 ACE_Sub_Barrier sub_barrier_1_
;
146 ACE_Sub_Barrier sub_barrier_2_
;
147 ACE_Sub_Barrier
*sub_barrier_
[2];
150 void operator= (const ACE_Barrier
&) = delete;
151 ACE_Barrier (const ACE_Barrier
&) = delete;
155 * @class ACE_Thread_Barrier
157 * @brief Implements "barrier synchronization" using ACE_Thread_Mutexes!
159 * This class is just a simple wrapper for ACE_Barrier that
160 * selects the USYNC_THREAD variant for the locks.
162 class ACE_Export ACE_Thread_Barrier
: public ACE_Barrier
165 /// Create a Thread_Barrier, passing in the optional @a name.
166 ACE_Thread_Barrier (unsigned int count
, const ACE_TCHAR
*name
= 0);
168 /// Default destructor.
169 ~ACE_Thread_Barrier () = default;
171 /// Dump the state of an object.
174 /// Declare the dynamic allocation hooks.
175 ACE_ALLOC_HOOK_DECLARE
;
178 ACE_END_VERSIONED_NAMESPACE_DECL
180 #endif /* !ACE_HAS_THREADS */
182 #include /**/ "ace/post.h"
183 #endif /* ACE_BARRIER_H */