281 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			281 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | /* | ||
|  |  * 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) 2009 Helge Bahmann | ||
|  |  * Copyright (c) 2012 Tim Blechmann | ||
|  |  * Copyright (c) 2013 - 2014 Andrey Semashev | ||
|  |  */ | ||
|  | /*! | ||
|  |  * \file   atomic/detail/storage_type.hpp | ||
|  |  * | ||
|  |  * This header defines underlying types used as storage | ||
|  |  */ | ||
|  | 
 | ||
|  | #ifndef BOOST_ATOMIC_DETAIL_STORAGE_TYPE_HPP_INCLUDED_ | ||
|  | #define BOOST_ATOMIC_DETAIL_STORAGE_TYPE_HPP_INCLUDED_ | ||
|  | 
 | ||
|  | #include <cstddef> | ||
|  | #include <boost/cstdint.hpp> | ||
|  | #include <boost/atomic/detail/config.hpp> | ||
|  | #if !defined(BOOST_ATOMIC_DETAIL_HAS_BUILTIN_MEMCMP) || !defined(BOOST_ATOMIC_DETAIL_HAS_BUILTIN_MEMCPY) | ||
|  | #include <cstring> | ||
|  | #endif | ||
|  | 
 | ||
|  | #ifdef BOOST_HAS_PRAGMA_ONCE | ||
|  | #pragma once | ||
|  | #endif | ||
|  | 
 | ||
|  | namespace boost { | ||
|  | namespace atomics { | ||
|  | namespace detail { | ||
|  | 
 | ||
|  | template< typename T > | ||
|  | BOOST_FORCEINLINE void non_atomic_load(T const volatile& from, T& to) BOOST_NOEXCEPT | ||
|  | { | ||
|  |     to = from; | ||
|  | } | ||
|  | 
 | ||
|  | template< std::size_t Size > | ||
|  | struct buffer_storage | ||
|  | { | ||
|  |     BOOST_ALIGNMENT(16) unsigned char data[Size]; | ||
|  | 
 | ||
|  |     BOOST_FORCEINLINE bool operator! () const BOOST_NOEXCEPT | ||
|  |     { | ||
|  |         return (data[0] == 0u && BOOST_ATOMIC_DETAIL_MEMCMP(data, data + 1, Size - 1) == 0); | ||
|  |     } | ||
|  | 
 | ||
|  |     BOOST_FORCEINLINE bool operator== (buffer_storage const& that) const BOOST_NOEXCEPT | ||
|  |     { | ||
|  |         return BOOST_ATOMIC_DETAIL_MEMCMP(data, that.data, Size) == 0; | ||
|  |     } | ||
|  | 
 | ||
|  |     BOOST_FORCEINLINE bool operator!= (buffer_storage const& that) const BOOST_NOEXCEPT | ||
|  |     { | ||
|  |         return BOOST_ATOMIC_DETAIL_MEMCMP(data, that.data, Size) != 0; | ||
|  |     } | ||
|  | }; | ||
|  | 
 | ||
|  | template< std::size_t Size > | ||
|  | BOOST_FORCEINLINE void non_atomic_load(buffer_storage< Size > const volatile& from, buffer_storage< Size >& to) BOOST_NOEXCEPT | ||
|  | { | ||
|  |     BOOST_ATOMIC_DETAIL_MEMCPY(to.data, const_cast< unsigned char const* >(from.data), Size); | ||
|  | } | ||
|  | 
 | ||
|  | template< std::size_t Size, bool Signed > | ||
|  | struct make_storage_type | ||
|  | { | ||
|  |     typedef buffer_storage< Size > type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type const& v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 1u, false > | ||
|  | { | ||
|  |     typedef boost::uint8_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 1u, true > | ||
|  | { | ||
|  |     typedef boost::int8_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 2u, false > | ||
|  | { | ||
|  |     typedef boost::uint16_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(2) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 2u, true > | ||
|  | { | ||
|  |     typedef boost::int16_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(2) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 4u, false > | ||
|  | { | ||
|  |     typedef boost::uint32_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(4) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 4u, true > | ||
|  | { | ||
|  |     typedef boost::int32_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(4) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 8u, false > | ||
|  | { | ||
|  |     typedef boost::uint64_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(8) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 8u, true > | ||
|  | { | ||
|  |     typedef boost::int64_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(8) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | #if defined(BOOST_HAS_INT128) | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 16u, false > | ||
|  | { | ||
|  |     typedef boost::uint128_type type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(16) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | template< > | ||
|  | struct make_storage_type< 16u, true > | ||
|  | { | ||
|  |     typedef boost::int128_type type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(16) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | #elif !defined(BOOST_NO_ALIGNMENT) | ||
|  | 
 | ||
|  | struct storage128_t | ||
|  | { | ||
|  |     boost::uint64_t data[2]; | ||
|  | 
 | ||
|  |     BOOST_FORCEINLINE bool operator! () const BOOST_NOEXCEPT | ||
|  |     { | ||
|  |         return data[0] == 0 && data[1] == 0; | ||
|  |     } | ||
|  | }; | ||
|  | 
 | ||
|  | BOOST_FORCEINLINE bool operator== (storage128_t const& left, storage128_t const& right) BOOST_NOEXCEPT | ||
|  | { | ||
|  |     return left.data[0] == right.data[0] && left.data[1] == right.data[1]; | ||
|  | } | ||
|  | BOOST_FORCEINLINE bool operator!= (storage128_t const& left, storage128_t const& right) BOOST_NOEXCEPT | ||
|  | { | ||
|  |     return !(left == right); | ||
|  | } | ||
|  | 
 | ||
|  | BOOST_FORCEINLINE void non_atomic_load(storage128_t const volatile& from, storage128_t& to) BOOST_NOEXCEPT | ||
|  | { | ||
|  |     to.data[0] = from.data[0]; | ||
|  |     to.data[1] = from.data[1]; | ||
|  | } | ||
|  | 
 | ||
|  | template< bool Signed > | ||
|  | struct make_storage_type< 16u, Signed > | ||
|  | { | ||
|  |     typedef storage128_t type; | ||
|  | 
 | ||
|  |     struct aligned | ||
|  |     { | ||
|  |         BOOST_ALIGNMENT(16) type value; | ||
|  | 
 | ||
|  |         BOOST_DEFAULTED_FUNCTION(aligned(), {}) | ||
|  |         BOOST_FORCEINLINE BOOST_CONSTEXPR explicit aligned(type const& v) BOOST_NOEXCEPT : value(v) {} | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | #endif | ||
|  | 
 | ||
|  | template< typename T > | ||
|  | struct storage_size_of | ||
|  | { | ||
|  |     enum _ | ||
|  |     { | ||
|  |         size = sizeof(T), | ||
|  |         value = (size == 3 ? 4 : (size >= 5 && size <= 7 ? 8 : (size >= 9 && size <= 15 ? 16 : size))) | ||
|  |     }; | ||
|  | }; | ||
|  | 
 | ||
|  | } // namespace detail | ||
|  | } // namespace atomics | ||
|  | } // namespace boost | ||
|  | 
 | ||
|  | #endif // BOOST_ATOMIC_DETAIL_STORAGE_TYPE_HPP_INCLUDED_ |