#ifndef OPENMW_BARRIER_H #define OPENMW_BARRIER_H #include #include namespace Misc { /// @brief Synchronize several threads class Barrier { public: /// @param count number of threads to wait on explicit Barrier(unsigned count) : mThreadCount(count) , mRendezvousCount(0) , mGeneration(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 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