mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 22:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			52 lines
		
	
	
	
		
			1.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			52 lines
		
	
	
	
		
			1.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifndef OPENMW_BARRIER_H
 | |
| #define OPENMW_BARRIER_H
 | |
| 
 | |
| #include <condition_variable>
 | |
| #include <mutex>
 | |
| 
 | |
| 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 <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
 |