// xatomic_wait.h internal header // Copyright (c) Microsoft Corporation. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception #ifndef _XATOMIC_WAIT_H #define _XATOMIC_WAIT_H #include #if _STL_COMPILER_PREPROCESSOR #include #include #pragma pack(push, _CRT_PACKING) #pragma warning(push, _STL_WARNING_LEVEL) #pragma warning(disable : _STL_DISABLED_WARNINGS) _STL_DISABLE_CLANG_WARNINGS #pragma push_macro("new") #undef new extern "C" { inline constexpr unsigned long __std_atomic_wait_no_timeout = 0xFFFF'FFFF; // Pass as partial timeout enum class __std_atomic_api_level : unsigned long { __not_set, __detecting, __has_srwlock, __has_wait_on_address, }; // This function allows testing the atomic wait support while always using the APIs for a platform with fewer // capabilities; it attempts to lock the APIs used to the level `_Requested_api_level`, and returns the actual API level // in use. Once the API level has been set by calling this function (or detected by a call to one of the atomic wait // functions), it can no longer be changed. __std_atomic_api_level __stdcall __std_atomic_set_api_level(__std_atomic_api_level _Requested_api_level) noexcept; // Support for atomic waits. // The "direct" functions are used when the underlying infrastructure can use WaitOnAddress directly; that is, _Size is // 1, 2, 4, or 8. The contract is the same as the WaitOnAddress function from the Windows SDK. If WaitOnAddress is not // available on the current platform, falls back to a similar solution based on SRWLOCK and CONDITION_VARIABLE. int __stdcall __std_atomic_wait_direct( const void* _Storage, void* _Comparand, size_t _Size, unsigned long _Remaining_timeout) noexcept; void __stdcall __std_atomic_notify_one_direct(const void* _Storage) noexcept; void __stdcall __std_atomic_notify_all_direct(const void* _Storage) noexcept; // The "indirect" functions are used when the size is not 1, 2, 4, or 8; these notionally wait on another value which is // of one of those sizes whose value changes upon notify, hence "indirect". (As of 2020-07-24, this always uses the // fallback SRWLOCK and CONDITION_VARIABLE implementation but that is not contractual.) using _Atomic_wait_indirect_equal_callback_t = bool(__stdcall*)( const void* _Storage, void* _Comparand, size_t _Size, void* _Param) _NOEXCEPT_FNPTR; int __stdcall __std_atomic_wait_indirect(const void* _Storage, void* _Comparand, size_t _Size, void* _Param, _Atomic_wait_indirect_equal_callback_t _Are_equal, unsigned long _Remaining_timeout) noexcept; void __stdcall __std_atomic_notify_one_indirect(const void* _Storage) noexcept; void __stdcall __std_atomic_notify_all_indirect(const void* _Storage) noexcept; // These functions convert a duration into a time point in order to tolerate spurious wakes in atomic wait, and then // convert back from the time point to individual wait attempts (which are limited by DWORD milliseconds to a length of // ~49 days) unsigned long long __stdcall __std_atomic_wait_get_deadline(unsigned long long _Timeout) noexcept; unsigned long __stdcall __std_atomic_wait_get_remaining_timeout(unsigned long long _Deadline) noexcept; } // extern "C" #pragma pop_macro("new") _STL_RESTORE_CLANG_WARNINGS #pragma warning(pop) #pragma pack(pop) #endif // _STL_COMPILER_PREPROCESSOR #endif // _XATOMIC_WAIT_H