mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 19:16:41 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			121 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			121 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // -*- C++ -*-
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | |
| // See https://llvm.org/LICENSE.txt for license information.
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #ifndef _LIBCPP_LATCH
 | |
| #define _LIBCPP_LATCH
 | |
| 
 | |
| /*
 | |
|     latch synopsis
 | |
| 
 | |
| namespace std
 | |
| {
 | |
| 
 | |
|   class latch
 | |
|   {
 | |
|   public:
 | |
|     static constexpr ptrdiff_t max() noexcept;
 | |
| 
 | |
|     constexpr explicit latch(ptrdiff_t __expected);
 | |
|     ~latch();
 | |
| 
 | |
|     latch(const latch&) = delete;
 | |
|     latch& operator=(const latch&) = delete;
 | |
| 
 | |
|     void count_down(ptrdiff_t __update = 1);
 | |
|     bool try_wait() const noexcept;
 | |
|     void wait() const;
 | |
|     void arrive_and_wait(ptrdiff_t __update = 1);
 | |
| 
 | |
|   private:
 | |
|     ptrdiff_t __counter; // exposition only
 | |
|   };
 | |
| 
 | |
| }
 | |
| 
 | |
| */
 | |
| 
 | |
| #include <__assert> // all public C++ headers provide the assertion handler
 | |
| #include <__atomic/atomic_base.h>
 | |
| #include <__atomic/atomic_sync.h>
 | |
| #include <__atomic/memory_order.h>
 | |
| #include <__availability>
 | |
| #include <__config>
 | |
| #include <cstddef>
 | |
| #include <limits>
 | |
| #include <version>
 | |
| 
 | |
| #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
 | |
| #  pragma GCC system_header
 | |
| #endif
 | |
| 
 | |
| #ifdef _LIBCPP_HAS_NO_THREADS
 | |
| # error "<latch> is not supported since libc++ has been configured without support for threads."
 | |
| #endif
 | |
| 
 | |
| _LIBCPP_PUSH_MACROS
 | |
| #include <__undef_macros>
 | |
| 
 | |
| #if _LIBCPP_STD_VER >= 14
 | |
| 
 | |
| _LIBCPP_BEGIN_NAMESPACE_STD
 | |
| 
 | |
| class latch
 | |
| {
 | |
|     __atomic_base<ptrdiff_t> __a_;
 | |
| 
 | |
| public:
 | |
|     static _LIBCPP_HIDE_FROM_ABI constexpr ptrdiff_t max() noexcept {
 | |
|         return numeric_limits<ptrdiff_t>::max();
 | |
|     }
 | |
| 
 | |
|     inline _LIBCPP_INLINE_VISIBILITY
 | |
|     constexpr explicit latch(ptrdiff_t __expected) : __a_(__expected) { }
 | |
| 
 | |
|     _LIBCPP_HIDE_FROM_ABI ~latch() = default;
 | |
|     latch(const latch&) = delete;
 | |
|     latch& operator=(const latch&) = delete;
 | |
| 
 | |
|     inline _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
 | |
|     void count_down(ptrdiff_t __update = 1)
 | |
|     {
 | |
|         auto const __old = __a_.fetch_sub(__update, memory_order_release);
 | |
|         if(__old == __update)
 | |
|             __a_.notify_all();
 | |
|     }
 | |
|     inline _LIBCPP_INLINE_VISIBILITY
 | |
|     bool try_wait() const noexcept
 | |
|     {
 | |
|         return 0 == __a_.load(memory_order_acquire);
 | |
|     }
 | |
|     inline _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
 | |
|     void wait() const
 | |
|     {
 | |
|         __cxx_atomic_wait(&__a_.__a_, [&]() -> bool {
 | |
|             return try_wait();
 | |
|         });
 | |
|     }
 | |
|     inline _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
 | |
|     void arrive_and_wait(ptrdiff_t __update = 1)
 | |
|     {
 | |
|         count_down(__update);
 | |
|         wait();
 | |
|     }
 | |
| };
 | |
| 
 | |
| _LIBCPP_END_NAMESPACE_STD
 | |
| 
 | |
| #endif // _LIBCPP_STD_VER >= 14
 | |
| 
 | |
| _LIBCPP_POP_MACROS
 | |
| 
 | |
| #if !defined(_LIBCPP_REMOVE_TRANSITIVE_INCLUDES) && _LIBCPP_STD_VER <= 20
 | |
| #  include <atomic>
 | |
| #endif
 | |
| 
 | |
| #endif //_LIBCPP_LATCH
 |