123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263 |
- /*
- * Distributed under the Boost Software License, Version 1.0.
- * (See accompanying file LICENSE_1_0.txt or copy at
- * http://www.boost.org/LICENSE_1_0.txt)
- *
- * Copyright (c) 2011 Helge Bahmann
- * Copyright (c) 2013 Tim Blechmann
- * Copyright (c) 2014, 2020 Andrey Semashev
- */
- /*!
- * \file atomic/atomic.hpp
- *
- * This header contains definition of \c atomic template.
- */
- #ifndef BOOST_ATOMIC_ATOMIC_HPP_INCLUDED_
- #define BOOST_ATOMIC_ATOMIC_HPP_INCLUDED_
- #include <cstddef>
- #include <boost/cstdint.hpp>
- #include <boost/memory_order.hpp>
- #include <boost/atomic/capabilities.hpp>
- #include <boost/atomic/detail/config.hpp>
- #include <boost/atomic/detail/classify.hpp>
- #include <boost/atomic/detail/atomic_impl.hpp>
- #include <boost/atomic/detail/type_traits/is_trivially_copyable.hpp>
- #include <boost/atomic/detail/type_traits/is_nothrow_default_constructible.hpp>
- #include <boost/atomic/detail/header.hpp>
- #ifdef BOOST_HAS_PRAGMA_ONCE
- #pragma once
- #endif
- namespace boost {
- namespace atomics {
- //! Atomic object
- template< typename T >
- class atomic :
- public atomics::detail::base_atomic< T, typename atomics::detail::classify< T >::type, false >
- {
- private:
- typedef atomics::detail::base_atomic< T, typename atomics::detail::classify< T >::type, false > base_type;
- typedef typename base_type::value_arg_type value_arg_type;
- public:
- typedef typename base_type::value_type value_type;
- static_assert(sizeof(value_type) > 0u, "boost::atomic<T> requires T to be a complete type");
- #if !defined(BOOST_ATOMIC_DETAIL_NO_CXX11_IS_TRIVIALLY_COPYABLE)
- static_assert(atomics::detail::is_trivially_copyable< value_type >::value, "boost::atomic<T> requires T to be a trivially copyable type");
- #endif
- public:
- BOOST_FORCEINLINE BOOST_ATOMIC_DETAIL_CONSTEXPR_UNION_INIT atomic() BOOST_NOEXCEPT_IF(atomics::detail::is_nothrow_default_constructible< value_type >::value) : base_type()
- {
- }
- BOOST_FORCEINLINE BOOST_ATOMIC_DETAIL_CONSTEXPR_UNION_INIT atomic(value_arg_type v) BOOST_NOEXCEPT : base_type(v)
- {
- }
- BOOST_FORCEINLINE value_type operator= (value_arg_type v) BOOST_NOEXCEPT
- {
- this->store(v);
- return v;
- }
- BOOST_FORCEINLINE value_type operator= (value_arg_type v) volatile BOOST_NOEXCEPT
- {
- this->store(v);
- return v;
- }
- BOOST_FORCEINLINE operator value_type() const volatile BOOST_NOEXCEPT
- {
- return this->load();
- }
- BOOST_DELETED_FUNCTION(atomic(atomic const&))
- BOOST_DELETED_FUNCTION(atomic& operator= (atomic const&))
- BOOST_DELETED_FUNCTION(atomic& operator= (atomic const&) volatile)
- };
- typedef atomic< char > atomic_char;
- typedef atomic< unsigned char > atomic_uchar;
- typedef atomic< signed char > atomic_schar;
- typedef atomic< uint8_t > atomic_uint8_t;
- typedef atomic< int8_t > atomic_int8_t;
- typedef atomic< unsigned short > atomic_ushort;
- typedef atomic< short > atomic_short;
- typedef atomic< uint16_t > atomic_uint16_t;
- typedef atomic< int16_t > atomic_int16_t;
- typedef atomic< unsigned int > atomic_uint;
- typedef atomic< int > atomic_int;
- typedef atomic< uint32_t > atomic_uint32_t;
- typedef atomic< int32_t > atomic_int32_t;
- typedef atomic< unsigned long > atomic_ulong;
- typedef atomic< long > atomic_long;
- typedef atomic< uint64_t > atomic_uint64_t;
- typedef atomic< int64_t > atomic_int64_t;
- #ifdef BOOST_HAS_LONG_LONG
- typedef atomic< boost::ulong_long_type > atomic_ullong;
- typedef atomic< boost::long_long_type > atomic_llong;
- #endif
- typedef atomic< void* > atomic_address;
- typedef atomic< bool > atomic_bool;
- typedef atomic< wchar_t > atomic_wchar_t;
- #if defined(__cpp_char8_t) && __cpp_char8_t >= 201811
- typedef atomic< char8_t > atomic_char8_t;
- #endif
- #if !defined(BOOST_NO_CXX11_CHAR16_T)
- typedef atomic< char16_t > atomic_char16_t;
- #endif
- #if !defined(BOOST_NO_CXX11_CHAR32_T)
- typedef atomic< char32_t > atomic_char32_t;
- #endif
- typedef atomic< int_least8_t > atomic_int_least8_t;
- typedef atomic< uint_least8_t > atomic_uint_least8_t;
- typedef atomic< int_least16_t > atomic_int_least16_t;
- typedef atomic< uint_least16_t > atomic_uint_least16_t;
- typedef atomic< int_least32_t > atomic_int_least32_t;
- typedef atomic< uint_least32_t > atomic_uint_least32_t;
- typedef atomic< int_least64_t > atomic_int_least64_t;
- typedef atomic< uint_least64_t > atomic_uint_least64_t;
- typedef atomic< int_fast8_t > atomic_int_fast8_t;
- typedef atomic< uint_fast8_t > atomic_uint_fast8_t;
- typedef atomic< int_fast16_t > atomic_int_fast16_t;
- typedef atomic< uint_fast16_t > atomic_uint_fast16_t;
- typedef atomic< int_fast32_t > atomic_int_fast32_t;
- typedef atomic< uint_fast32_t > atomic_uint_fast32_t;
- typedef atomic< int_fast64_t > atomic_int_fast64_t;
- typedef atomic< uint_fast64_t > atomic_uint_fast64_t;
- typedef atomic< intmax_t > atomic_intmax_t;
- typedef atomic< uintmax_t > atomic_uintmax_t;
- #if !defined(BOOST_ATOMIC_NO_FLOATING_POINT)
- typedef atomic< float > atomic_float_t;
- typedef atomic< double > atomic_double_t;
- typedef atomic< long double > atomic_long_double_t;
- #endif
- typedef atomic< std::size_t > atomic_size_t;
- typedef atomic< std::ptrdiff_t > atomic_ptrdiff_t;
- #if defined(BOOST_HAS_INTPTR_T)
- typedef atomic< boost::intptr_t > atomic_intptr_t;
- typedef atomic< boost::uintptr_t > atomic_uintptr_t;
- #endif
- // Select the lock-free atomic types that has natively supported waiting/notifying operations.
- // Prefer 32-bit types the most as those have the best performance on current 32 and 64-bit architectures.
- #if BOOST_ATOMIC_INT32_LOCK_FREE == 2 && BOOST_ATOMIC_HAS_NATIVE_INT32_WAIT_NOTIFY == 2
- typedef atomic< uint32_t > atomic_unsigned_lock_free;
- typedef atomic< int32_t > atomic_signed_lock_free;
- #elif BOOST_ATOMIC_INT64_LOCK_FREE == 2 && BOOST_ATOMIC_HAS_NATIVE_INT64_WAIT_NOTIFY == 2
- typedef atomic< uint64_t > atomic_unsigned_lock_free;
- typedef atomic< int64_t > atomic_signed_lock_free;
- #elif BOOST_ATOMIC_INT16_LOCK_FREE == 2 && BOOST_ATOMIC_HAS_NATIVE_INT16_WAIT_NOTIFY == 2
- typedef atomic< uint16_t > atomic_unsigned_lock_free;
- typedef atomic< int16_t > atomic_signed_lock_free;
- #elif BOOST_ATOMIC_INT8_LOCK_FREE == 2 && BOOST_ATOMIC_HAS_NATIVE_INT8_WAIT_NOTIFY == 2
- typedef atomic< uint8_t > atomic_unsigned_lock_free;
- typedef atomic< int8_t > atomic_signed_lock_free;
- #elif BOOST_ATOMIC_INT32_LOCK_FREE == 2
- typedef atomic< uint32_t > atomic_unsigned_lock_free;
- typedef atomic< int32_t > atomic_signed_lock_free;
- #elif BOOST_ATOMIC_INT64_LOCK_FREE == 2
- typedef atomic< uint64_t > atomic_unsigned_lock_free;
- typedef atomic< int64_t > atomic_signed_lock_free;
- #elif BOOST_ATOMIC_INT16_LOCK_FREE == 2
- typedef atomic< uint16_t > atomic_unsigned_lock_free;
- typedef atomic< int16_t > atomic_signed_lock_free;
- #elif BOOST_ATOMIC_INT8_LOCK_FREE == 2
- typedef atomic< uint8_t > atomic_unsigned_lock_free;
- typedef atomic< int8_t > atomic_signed_lock_free;
- #else
- #define BOOST_ATOMIC_DETAIL_NO_LOCK_FREE_TYPEDEFS
- #endif
- } // namespace atomics
- using atomics::atomic;
- using atomics::atomic_char;
- using atomics::atomic_uchar;
- using atomics::atomic_schar;
- using atomics::atomic_uint8_t;
- using atomics::atomic_int8_t;
- using atomics::atomic_ushort;
- using atomics::atomic_short;
- using atomics::atomic_uint16_t;
- using atomics::atomic_int16_t;
- using atomics::atomic_uint;
- using atomics::atomic_int;
- using atomics::atomic_uint32_t;
- using atomics::atomic_int32_t;
- using atomics::atomic_ulong;
- using atomics::atomic_long;
- using atomics::atomic_uint64_t;
- using atomics::atomic_int64_t;
- #ifdef BOOST_HAS_LONG_LONG
- using atomics::atomic_ullong;
- using atomics::atomic_llong;
- #endif
- using atomics::atomic_address;
- using atomics::atomic_bool;
- using atomics::atomic_wchar_t;
- #if defined(__cpp_char8_t) && __cpp_char8_t >= 201811
- using atomics::atomic_char8_t;
- #endif
- #if !defined(BOOST_NO_CXX11_CHAR16_T)
- using atomics::atomic_char16_t;
- #endif
- #if !defined(BOOST_NO_CXX11_CHAR32_T)
- using atomics::atomic_char32_t;
- #endif
- using atomics::atomic_int_least8_t;
- using atomics::atomic_uint_least8_t;
- using atomics::atomic_int_least16_t;
- using atomics::atomic_uint_least16_t;
- using atomics::atomic_int_least32_t;
- using atomics::atomic_uint_least32_t;
- using atomics::atomic_int_least64_t;
- using atomics::atomic_uint_least64_t;
- using atomics::atomic_int_fast8_t;
- using atomics::atomic_uint_fast8_t;
- using atomics::atomic_int_fast16_t;
- using atomics::atomic_uint_fast16_t;
- using atomics::atomic_int_fast32_t;
- using atomics::atomic_uint_fast32_t;
- using atomics::atomic_int_fast64_t;
- using atomics::atomic_uint_fast64_t;
- using atomics::atomic_intmax_t;
- using atomics::atomic_uintmax_t;
- #if !defined(BOOST_ATOMIC_NO_FLOATING_POINT)
- using atomics::atomic_float_t;
- using atomics::atomic_double_t;
- using atomics::atomic_long_double_t;
- #endif
- using atomics::atomic_size_t;
- using atomics::atomic_ptrdiff_t;
- #if defined(BOOST_HAS_INTPTR_T)
- using atomics::atomic_intptr_t;
- using atomics::atomic_uintptr_t;
- #endif
- #if !defined(BOOST_ATOMIC_DETAIL_NO_LOCK_FREE_TYPEDEFS)
- using atomics::atomic_unsigned_lock_free;
- using atomics::atomic_signed_lock_free;
- #endif
- #undef BOOST_ATOMIC_DETAIL_NO_LOCK_FREE_TYPEDEFS
- } // namespace boost
- #include <boost/atomic/detail/footer.hpp>
- #endif // BOOST_ATOMIC_ATOMIC_HPP_INCLUDED_
|