123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274 |
- /*
- Copyright (C) 2002 Brad King ([email protected])
- Douglas Gregor ([email protected])
- Copyright (C) 2002, 2008, 2013 Peter Dimov
- Copyright (C) 2017 Glen Joseph Fernandes ([email protected])
- 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)
- */
- #ifndef BOOST_CORE_ADDRESSOF_HPP
- #define BOOST_CORE_ADDRESSOF_HPP
- #include <boost/config.hpp>
- #if defined(BOOST_MSVC_FULL_VER) && BOOST_MSVC_FULL_VER >= 190024215
- #define BOOST_CORE_HAS_BUILTIN_ADDRESSOF
- #elif defined(BOOST_GCC) && BOOST_GCC >= 70000
- #define BOOST_CORE_HAS_BUILTIN_ADDRESSOF
- #elif defined(__has_builtin)
- #if __has_builtin(__builtin_addressof)
- #define BOOST_CORE_HAS_BUILTIN_ADDRESSOF
- #endif
- #endif
- #if defined(BOOST_CORE_HAS_BUILTIN_ADDRESSOF)
- #if defined(BOOST_NO_CXX11_CONSTEXPR)
- #define BOOST_CORE_NO_CONSTEXPR_ADDRESSOF
- #endif
- namespace boost {
- template<class T>
- BOOST_CONSTEXPR inline T*
- addressof(T& o) BOOST_NOEXCEPT
- {
- return __builtin_addressof(o);
- }
- } /* boost */
- #else
- #include <boost/config/workaround.hpp>
- #include <cstddef>
- namespace boost {
- namespace detail {
- template<class T>
- class addrof_ref {
- public:
- BOOST_FORCEINLINE addrof_ref(T& o) BOOST_NOEXCEPT
- : o_(o) { }
- BOOST_FORCEINLINE operator T&() const BOOST_NOEXCEPT {
- return o_;
- }
- private:
- addrof_ref& operator=(const addrof_ref&);
- T& o_;
- };
- template<class T>
- struct addrof {
- static BOOST_FORCEINLINE T* get(T& o, long) BOOST_NOEXCEPT {
- return reinterpret_cast<T*>(&
- const_cast<char&>(reinterpret_cast<const volatile char&>(o)));
- }
- static BOOST_FORCEINLINE T* get(T* p, int) BOOST_NOEXCEPT {
- return p;
- }
- };
- #if !defined(BOOST_NO_CXX11_NULLPTR)
- #if !defined(BOOST_NO_CXX11_DECLTYPE) && \
- (defined(__INTEL_COMPILER) || \
- (defined(__clang__) && !defined(_LIBCPP_VERSION)))
- typedef decltype(nullptr) addrof_null_t;
- #else
- typedef std::nullptr_t addrof_null_t;
- #endif
- template<>
- struct addrof<addrof_null_t> {
- typedef addrof_null_t type;
- static BOOST_FORCEINLINE type* get(type& o, int) BOOST_NOEXCEPT {
- return &o;
- }
- };
- template<>
- struct addrof<const addrof_null_t> {
- typedef const addrof_null_t type;
- static BOOST_FORCEINLINE type* get(type& o, int) BOOST_NOEXCEPT {
- return &o;
- }
- };
- template<>
- struct addrof<volatile addrof_null_t> {
- typedef volatile addrof_null_t type;
- static BOOST_FORCEINLINE type* get(type& o, int) BOOST_NOEXCEPT {
- return &o;
- }
- };
- template<>
- struct addrof<const volatile addrof_null_t> {
- typedef const volatile addrof_null_t type;
- static BOOST_FORCEINLINE type* get(type& o, int) BOOST_NOEXCEPT {
- return &o;
- }
- };
- #endif
- } /* detail */
- #if defined(BOOST_NO_CXX11_SFINAE_EXPR) || \
- defined(BOOST_NO_CXX11_CONSTEXPR) || \
- defined(BOOST_NO_CXX11_DECLTYPE)
- #define BOOST_CORE_NO_CONSTEXPR_ADDRESSOF
- template<class T>
- BOOST_FORCEINLINE T*
- addressof(T& o) BOOST_NOEXCEPT
- {
- #if BOOST_WORKAROUND(BOOST_BORLANDC, BOOST_TESTED_AT(0x610)) || \
- BOOST_WORKAROUND(__SUNPRO_CC, <= 0x5120)
- return boost::detail::addrof<T>::get(o, 0);
- #else
- return boost::detail::addrof<T>::get(boost::detail::addrof_ref<T>(o), 0);
- #endif
- }
- #if BOOST_WORKAROUND(__SUNPRO_CC, BOOST_TESTED_AT(0x590))
- namespace detail {
- template<class T>
- struct addrof_result {
- typedef T* type;
- };
- } /* detail */
- template<class T, std::size_t N>
- BOOST_FORCEINLINE typename boost::detail::addrof_result<T[N]>::type
- addressof(T (&o)[N]) BOOST_NOEXCEPT
- {
- return &o;
- }
- #endif
- #if BOOST_WORKAROUND(BOOST_BORLANDC, BOOST_TESTED_AT(0x564))
- template<class T, std::size_t N>
- BOOST_FORCEINLINE
- T (*addressof(T (&o)[N]) BOOST_NOEXCEPT)[N]
- {
- return reinterpret_cast<T(*)[N]>(&o);
- }
- template<class T, std::size_t N>
- BOOST_FORCEINLINE
- const T (*addressof(const T (&o)[N]) BOOST_NOEXCEPT)[N]
- {
- return reinterpret_cast<const T(*)[N]>(&o);
- }
- #endif
- #else
- namespace detail {
- template<class T>
- T addrof_declval() BOOST_NOEXCEPT;
- template<class>
- struct addrof_void {
- typedef void type;
- };
- template<class T, class E = void>
- struct addrof_member_operator {
- static constexpr bool value = false;
- };
- template<class T>
- struct addrof_member_operator<T, typename
- addrof_void<decltype(addrof_declval<T&>().operator&())>::type> {
- static constexpr bool value = true;
- };
- #if BOOST_WORKAROUND(BOOST_INTEL, < 1600)
- struct addrof_addressable { };
- addrof_addressable*
- operator&(addrof_addressable&) BOOST_NOEXCEPT;
- #endif
- template<class T, class E = void>
- struct addrof_non_member_operator {
- static constexpr bool value = false;
- };
- template<class T>
- struct addrof_non_member_operator<T, typename
- addrof_void<decltype(operator&(addrof_declval<T&>()))>::type> {
- static constexpr bool value = true;
- };
- template<class T, class E = void>
- struct addrof_expression {
- static constexpr bool value = false;
- };
- template<class T>
- struct addrof_expression<T,
- typename addrof_void<decltype(&addrof_declval<T&>())>::type> {
- static constexpr bool value = true;
- };
- template<class T>
- struct addrof_is_constexpr {
- static constexpr bool value = addrof_expression<T>::value &&
- !addrof_member_operator<T>::value &&
- !addrof_non_member_operator<T>::value;
- };
- template<bool E, class T>
- struct addrof_if { };
- template<class T>
- struct addrof_if<true, T> {
- typedef T* type;
- };
- template<class T>
- BOOST_FORCEINLINE
- typename addrof_if<!addrof_is_constexpr<T>::value, T>::type
- addressof(T& o) BOOST_NOEXCEPT
- {
- return addrof<T>::get(addrof_ref<T>(o), 0);
- }
- template<class T>
- constexpr BOOST_FORCEINLINE
- typename addrof_if<addrof_is_constexpr<T>::value, T>::type
- addressof(T& o) BOOST_NOEXCEPT
- {
- return &o;
- }
- } /* detail */
- template<class T>
- constexpr BOOST_FORCEINLINE T*
- addressof(T& o) BOOST_NOEXCEPT
- {
- return boost::detail::addressof(o);
- }
- #endif
- } /* boost */
- #endif
- #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES) && \
- !defined(BOOST_NO_CXX11_DELETED_FUNCTIONS)
- namespace boost {
- template<class T>
- const T* addressof(const T&&) = delete;
- } /* boost */
- #endif
- #endif
|