Cleanup ACE_HAS_PTHREAD_SIGMASK_PROTOTYPE, all platforms support it so far as I can...
[ACE_TAO.git] / ACE / ace / Barrier.h
blob33356c0b3225c553b41a3714bf79549e7a0d4bfa
1 // -*- C++ -*-
3 //==========================================================================
4 /**
5 * @file Barrier.h
7 * @author Douglas C. Schmidt <d.schmidt@vanderbilt.edu>
8 */
9 //==========================================================================
11 #ifndef ACE_BARRIER_H
12 #define ACE_BARRIER_H
13 #include /**/ "ace/pre.h"
15 #include /**/ "ace/ACE_export.h"
17 #if !defined (ACE_LACKS_PRAGMA_ONCE)
18 # 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
30 /**
31 * @class ACE_Barrier
33 * @brief This is a no-op to make ACE "syntactically consistent."
35 class ACE_Export ACE_Barrier
37 public:
38 ACE_Barrier (unsigned int, const ACE_TCHAR * = 0, void * = 0) {}
39 ~ACE_Barrier () = default;
40 int wait () { ACE_NOTSUP_RETURN (-1); }
41 void dump () const {}
44 ACE_END_VERSIONED_NAMESPACE_DECL
46 #else /* ACE_HAS_THREADS */
48 #include <atomic>
49 #include "ace/Condition_Thread_Mutex.h"
51 ACE_BEGIN_VERSIONED_NAMESPACE_DECL
53 struct ACE_Export ACE_Sub_Barrier
55 // = Initialization.
56 ACE_Sub_Barrier (unsigned int count,
57 ACE_Thread_Mutex &lock,
58 const ACE_TCHAR *name = 0,
59 void *arg = 0);
61 ~ACE_Sub_Barrier () = default;
63 /// True if this generation of the barrier is done.
64 ACE_Condition_Thread_Mutex barrier_finished_;
66 /// Number of threads that are still running.
67 std::atomic<int> running_threads_;
69 /// Dump the state of an object.
70 void dump () const;
72 /// Declare the dynamic allocation hooks.
73 ACE_ALLOC_HOOK_DECLARE;
76 /**
77 * @class ACE_Barrier
79 * @brief Implements "barrier synchronization".
81 * This class allows <count> number of threads to synchronize
82 * their completion of (one round of) a task, which is known as
83 * "barrier synchronization". After all the threads call <wait()>
84 * on the barrier they are all atomically released and can begin a new
85 * round.
87 * This implementation uses a "sub-barrier generation numbering"
88 * scheme to avoid overhead and to ensure that all threads wait to
89 * leave the barrier correct. This code is based on an article from
90 * SunOpsis Vol. 4, No. 1 by Richard Marejka
91 * (Richard.Marejka@canada.sun.com).
93 class ACE_Export ACE_Barrier
95 public:
96 /// Initialize the barrier to synchronize @a count threads.
97 ACE_Barrier (unsigned int count,
98 const ACE_TCHAR *name = 0,
99 void *arg = 0);
101 /// Default destructor.
102 ~ACE_Barrier () = default;
104 /// Block the caller until all @c count threads have called @c wait and
105 /// then allow all the caller threads to continue in parallel.
107 /// @retval 0 after successfully waiting for all threads to wait.
108 /// @retval -1 if an error occurs or the barrier is shut
109 /// down (@sa shutdown ()).
110 int wait ();
112 /// Shut the barrier down, aborting the wait of all waiting threads.
113 /// Any threads waiting on the barrier when it is shut down will return with
114 /// value -1, errno ESHUTDOWN.
116 /// @retval 0 for success, -1 if already shut down.
118 /// @since ACE beta 5.4.9.
119 int shutdown ();
121 /// Dump the state of an object.
122 void dump () const;
124 /// Declare the dynamic allocation hooks.
125 ACE_ALLOC_HOOK_DECLARE;
127 protected:
128 /// Serialize access to the barrier state.
129 ACE_Thread_Mutex lock_;
131 /// Either 0 or 1, depending on whether we are the first generation
132 /// of waiters or the next generation of waiters.
133 int current_generation_;
135 /// Total number of threads that can be waiting at any one time.
136 int count_;
139 * We keep two @c sub_barriers, one for the first "generation" of
140 * waiters, and one for the next "generation" of waiters. This
141 * efficiently solves the problem of what to do if all the first
142 * generation waiters don't leave the barrier before one of the
143 * threads calls wait() again (i.e., starts up the next generation
144 * barrier).
146 ACE_Sub_Barrier sub_barrier_1_;
147 ACE_Sub_Barrier sub_barrier_2_;
148 ACE_Sub_Barrier *sub_barrier_[2];
150 private:
151 void operator= (const ACE_Barrier &) = delete;
152 ACE_Barrier (const ACE_Barrier &) = delete;
156 * @class ACE_Thread_Barrier
158 * @brief Implements "barrier synchronization" using ACE_Thread_Mutexes!
160 * This class is just a simple wrapper for ACE_Barrier that
161 * selects the USYNC_THREAD variant for the locks.
163 class ACE_Export ACE_Thread_Barrier : public ACE_Barrier
165 public:
166 /// Create a Thread_Barrier, passing in the optional @a name.
167 ACE_Thread_Barrier (unsigned int count, const ACE_TCHAR *name = 0);
169 /// Default destructor.
170 ~ACE_Thread_Barrier () = default;
172 /// Dump the state of an object.
173 void dump () const;
175 /// Declare the dynamic allocation hooks.
176 ACE_ALLOC_HOOK_DECLARE;
179 ACE_END_VERSIONED_NAMESPACE_DECL
181 #endif /* !ACE_HAS_THREADS */
183 #include /**/ "ace/post.h"
184 #endif /* ACE_BARRIER_H */