// Copyright 2014 Renato Tegon Forti, Antony Polukhin. // Copyright 2015-2016 Antony Polukhin. // // 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_DLL_DETAIL_AGGRESSIVE_PTR_CAST_HPP #define BOOST_DLL_DETAIL_AGGRESSIVE_PTR_CAST_HPP #include #ifdef BOOST_HAS_PRAGMA_ONCE # pragma once #endif #include #include #include #include #include #include #include #include #include // boost::uintptr_t #include // std::memcpy namespace boost { namespace dll { namespace detail { // GCC warns when reinterpret_cast between function pointer and object pointer occur. // This method suppress the warnings and ensures that such casts are safe. template BOOST_FORCEINLINE typename boost::disable_if_c::value || boost::is_reference::value || boost::is_member_pointer::value, To>::type aggressive_ptr_cast(From v) BOOST_NOEXCEPT { BOOST_STATIC_ASSERT_MSG( boost::is_pointer::value && boost::is_pointer::value, "`agressive_ptr_cast` function must be used only for pointer casting." ); BOOST_STATIC_ASSERT_MSG( boost::is_void< typename boost::remove_pointer::type >::value || boost::is_void< typename boost::remove_pointer::type >::value, "`agressive_ptr_cast` function must be used only for casting to or from void pointers." ); BOOST_STATIC_ASSERT_MSG( sizeof(v) == sizeof(To), "Pointer to function and pointer to object differ in size on your platform." ); return reinterpret_cast( reinterpret_cast(v) ); } template BOOST_FORCEINLINE typename boost::disable_if_c::value || boost::is_member_pointer::value, To>::type aggressive_ptr_cast(From v) BOOST_NOEXCEPT { BOOST_STATIC_ASSERT_MSG( boost::is_pointer::value, "`agressive_ptr_cast` function must be used only for pointer casting." ); BOOST_STATIC_ASSERT_MSG( boost::is_void< typename boost::remove_pointer::type >::value, "`agressive_ptr_cast` function must be used only for casting to or from void pointers." ); BOOST_STATIC_ASSERT_MSG( sizeof(v) == sizeof(typename boost::remove_reference::type*), "Pointer to function and pointer to object differ in size on your platform." ); return static_cast( *reinterpret_cast::type*>( *reinterpret_cast( reinterpret_cast( v ) ) ) ); } template BOOST_FORCEINLINE typename boost::disable_if_c::value || boost::is_member_pointer::value, To>::type aggressive_ptr_cast(From v) BOOST_NOEXCEPT { BOOST_STATIC_ASSERT_MSG( boost::is_pointer::value, "`agressive_ptr_cast` function must be used only for pointer casting." ); BOOST_STATIC_ASSERT_MSG( boost::is_void< typename boost::remove_pointer::type >::value, "`agressive_ptr_cast` function must be used only for casting to or from void pointers." ); To res = 0; std::memcpy(&res, &v, sizeof(To)); return res; } template BOOST_FORCEINLINE typename boost::disable_if_c::value || !boost::is_member_pointer::value, To>::type aggressive_ptr_cast(From /* v */) BOOST_NOEXCEPT { BOOST_STATIC_ASSERT_MSG( boost::is_pointer::value, "`agressive_ptr_cast` function must be used only for pointer casting." ); BOOST_STATIC_ASSERT_MSG( boost::is_void< typename boost::remove_pointer::type >::value, "`agressive_ptr_cast` function must be used only for casting to or from void pointers." ); BOOST_STATIC_ASSERT_MSG( !sizeof(From), "Casting from member pointers to void pointer is not implemnted in `agressive_ptr_cast`." ); return 0; } }}} // boost::dll::detail #endif // BOOST_DLL_DETAIL_AGGRESSIVE_PTR_CAST_HPP