// // detail/memory.hpp // ~~~~~~~~~~~~~~~~~ // // Copyright (c) 2003-2023 Christopher M. Kohlhoff (chris at kohlhoff dot com) // // 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 ASIO_DETAIL_MEMORY_HPP #define ASIO_DETAIL_MEMORY_HPP #if defined(_MSC_VER) && (_MSC_VER >= 1200) # pragma once #endif // defined(_MSC_VER) && (_MSC_VER >= 1200) #include "asio/detail/config.hpp" #include #include #include #include #include "asio/detail/cstdint.hpp" #include "asio/detail/throw_exception.hpp" #if !defined(ASIO_HAS_STD_SHARED_PTR) # include # include # include #endif // !defined(ASIO_HAS_STD_SHARED_PTR) #if !defined(ASIO_HAS_STD_ADDRESSOF) # include #endif // !defined(ASIO_HAS_STD_ADDRESSOF) #if !defined(ASIO_HAS_STD_ALIGNED_ALLOC) \ && defined(ASIO_HAS_BOOST_ALIGN) \ && defined(ASIO_HAS_ALIGNOF) # include #endif // !defined(ASIO_HAS_STD_ALIGNED_ALLOC) // && defined(ASIO_HAS_BOOST_ALIGN) // && defined(ASIO_HAS_ALIGNOF) namespace asio { namespace detail { #if defined(ASIO_HAS_STD_SHARED_PTR) using std::make_shared; using std::shared_ptr; using std::weak_ptr; #else // defined(ASIO_HAS_STD_SHARED_PTR) using boost::make_shared; using boost::shared_ptr; using boost::weak_ptr; #endif // defined(ASIO_HAS_STD_SHARED_PTR) #if defined(ASIO_HAS_STD_ADDRESSOF) using std::addressof; #else // defined(ASIO_HAS_STD_ADDRESSOF) using boost::addressof; #endif // defined(ASIO_HAS_STD_ADDRESSOF) #if defined(ASIO_HAS_STD_TO_ADDRESS) using std::to_address; #else // defined(ASIO_HAS_STD_TO_ADDRESS) template inline T* to_address(T* p) { return p; } template inline const T* to_address(const T* p) { return p; } template inline volatile T* to_address(volatile T* p) { return p; } template inline const volatile T* to_address(const volatile T* p) { return p; } #endif // defined(ASIO_HAS_STD_TO_ADDRESS) inline void* align(std::size_t alignment, std::size_t size, void*& ptr, std::size_t& space) { #if defined(ASIO_HAS_STD_ALIGN) return std::align(alignment, size, ptr, space); #else // defined(ASIO_HAS_STD_ALIGN) const uintptr_t intptr = reinterpret_cast(ptr); const uintptr_t aligned = (intptr - 1u + alignment) & -alignment; const std::size_t padding = aligned - intptr; if (size + padding > space) return 0; space -= padding; ptr = reinterpret_cast(aligned); return ptr; #endif // defined(ASIO_HAS_STD_ALIGN) } } // namespace detail #if defined(ASIO_HAS_CXX11_ALLOCATORS) using std::allocator_arg_t; # define ASIO_USES_ALLOCATOR(t) \ namespace std { \ template \ struct uses_allocator : true_type {}; \ } \ /**/ # define ASIO_REBIND_ALLOC(alloc, t) \ typename std::allocator_traits::template rebind_alloc /**/ #else // defined(ASIO_HAS_CXX11_ALLOCATORS) struct allocator_arg_t {}; # define ASIO_USES_ALLOCATOR(t) # define ASIO_REBIND_ALLOC(alloc, t) \ typename alloc::template rebind::other /**/ #endif // defined(ASIO_HAS_CXX11_ALLOCATORS) inline void* aligned_new(std::size_t align, std::size_t size) { #if defined(ASIO_HAS_STD_ALIGNED_ALLOC) && defined(ASIO_HAS_ALIGNOF) align = (align < ASIO_DEFAULT_ALIGN) ? ASIO_DEFAULT_ALIGN : align; size = (size % align == 0) ? size : size + (align - size % align); void* ptr = std::aligned_alloc(align, size); if (!ptr) { std::bad_alloc ex; asio::detail::throw_exception(ex); } return ptr; #elif defined(ASIO_HAS_BOOST_ALIGN) && defined(ASIO_HAS_ALIGNOF) align = (align < ASIO_DEFAULT_ALIGN) ? ASIO_DEFAULT_ALIGN : align; size = (size % align == 0) ? size : size + (align - size % align); void* ptr = boost::alignment::aligned_alloc(align, size); if (!ptr) { std::bad_alloc ex; asio::detail::throw_exception(ex); } return ptr; #elif defined(ASIO_MSVC) && defined(ASIO_HAS_ALIGNOF) align = (align < ASIO_DEFAULT_ALIGN) ? ASIO_DEFAULT_ALIGN : align; size = (size % align == 0) ? size : size + (align - size % align); void* ptr = _aligned_malloc(size, align); if (!ptr) { std::bad_alloc ex; asio::detail::throw_exception(ex); } return ptr; #else // defined(ASIO_MSVC) && defined(ASIO_HAS_ALIGNOF) (void)align; return ::operator new(size); #endif // defined(ASIO_MSVC) && defined(ASIO_HAS_ALIGNOF) } inline void aligned_delete(void* ptr) { #if defined(ASIO_HAS_STD_ALIGNED_ALLOC) && defined(ASIO_HAS_ALIGNOF) std::free(ptr); #elif defined(ASIO_HAS_BOOST_ALIGN) && defined(ASIO_HAS_ALIGNOF) boost::alignment::aligned_free(ptr); #elif defined(ASIO_MSVC) && defined(ASIO_HAS_ALIGNOF) _aligned_free(ptr); #else // defined(ASIO_MSVC) && defined(ASIO_HAS_ALIGNOF) ::operator delete(ptr); #endif // defined(ASIO_MSVC) && defined(ASIO_HAS_ALIGNOF) } } // namespace asio #endif // ASIO_DETAIL_MEMORY_HPP