mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-26 09:35:28 +00:00
21dbe314bf
same flow, synchronous simulation is just a special case.
52 lines
1.5 KiB
C++
52 lines
1.5 KiB
C++
#ifndef OPENMW_BARRIER_H
|
|
#define OPENMW_BARRIER_H
|
|
|
|
#include <cassert>
|
|
#include <condition_variable>
|
|
#include <mutex>
|
|
|
|
namespace Misc
|
|
{
|
|
/// @brief Synchronize several threads
|
|
class Barrier
|
|
{
|
|
public:
|
|
/// @param count number of threads to wait on
|
|
explicit Barrier(int count) : mThreadCount(count), mRendezvousCount(0), mGeneration(0)
|
|
{
|
|
assert(count >= 0);
|
|
}
|
|
|
|
/// @brief stop execution of threads until count distinct threads reach this point
|
|
/// @param func callable to be executed once after all threads have met
|
|
template <class Callback>
|
|
void wait(Callback&& func)
|
|
{
|
|
std::unique_lock lock(mMutex);
|
|
|
|
++mRendezvousCount;
|
|
const unsigned int currentGeneration = mGeneration;
|
|
if (mRendezvousCount == mThreadCount || mThreadCount == 0)
|
|
{
|
|
++mGeneration;
|
|
mRendezvousCount = 0;
|
|
func();
|
|
mRendezvous.notify_all();
|
|
}
|
|
else
|
|
{
|
|
mRendezvous.wait(lock, [&]() { return mGeneration != currentGeneration; });
|
|
}
|
|
}
|
|
|
|
private:
|
|
unsigned int mThreadCount;
|
|
unsigned int mRendezvousCount;
|
|
unsigned int mGeneration;
|
|
mutable std::mutex mMutex;
|
|
std::condition_variable mRendezvous;
|
|
};
|
|
}
|
|
|
|
#endif
|