// // bind_allocator.cpp // ~~~~~~~~~~~~~~~~~~ // // 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) // // Disable autolinking for unit tests. #if !defined(BOOST_ALL_NO_LIB) #define BOOST_ALL_NO_LIB 1 #endif // !defined(BOOST_ALL_NO_LIB) // Test that header file is self-contained. #include #include #include #include "unit_test.hpp" #if defined(BOOST_ASIO_HAS_BOOST_DATE_TIME) # include #else // defined(BOOST_ASIO_HAS_BOOST_DATE_TIME) # include #endif // defined(BOOST_ASIO_HAS_BOOST_DATE_TIME) #if defined(BOOST_ASIO_HAS_BOOST_BIND) # include #else // defined(BOOST_ASIO_HAS_BOOST_BIND) # include #endif // defined(BOOST_ASIO_HAS_BOOST_BIND) using namespace boost::asio; #if defined(BOOST_ASIO_HAS_BOOST_BIND) namespace bindns = boost; #else // defined(BOOST_ASIO_HAS_BOOST_BIND) namespace bindns = std; #endif #if defined(BOOST_ASIO_HAS_BOOST_DATE_TIME) typedef deadline_timer timer; namespace chronons = boost::posix_time; #elif defined(BOOST_ASIO_HAS_CHRONO) typedef steady_timer timer; namespace chronons = boost::asio::chrono; #endif // defined(BOOST_ASIO_HAS_BOOST_DATE_TIME) template class test_allocator { public: typedef T value_type; explicit test_allocator(int* allocations) : allocations_(allocations) { } template test_allocator(const test_allocator& other) : allocations_(other.allocations_) { } template struct rebind { typedef test_allocator other; }; bool operator==(const test_allocator&) const { return true; } bool operator!=(const test_allocator&) const { return false; } T* allocate(std::size_t n) const { ++(*allocations_); return static_cast(::operator new(sizeof(T) * n)); } void deallocate(T* p, std::size_t /*n*/) const { --(*allocations_); ::operator delete(p); } //private: int* allocations_; }; void increment(int* count) { ++(*count); } void bind_allocator_to_function_object_test() { io_context ioc; int count = 0; int allocations = 0; timer t(ioc, chronons::seconds(1)); t.async_wait( bind_allocator( test_allocator(&allocations), bindns::bind(&increment, &count))); BOOST_ASIO_CHECK(count == 0); BOOST_ASIO_CHECK(allocations == 1); ioc.run(); BOOST_ASIO_CHECK(count == 1); BOOST_ASIO_CHECK(allocations == 0); } struct incrementer_token_v1 { explicit incrementer_token_v1(int* c) : count(c) {} int* count; }; struct incrementer_handler_v1 { explicit incrementer_handler_v1(incrementer_token_v1 t) : count(t.count) {} void operator()(boost::system::error_code){ increment(count); } int* count; }; namespace boost { namespace asio { template <> class async_result { public: typedef incrementer_handler_v1 completion_handler_type; typedef void return_type; explicit async_result(completion_handler_type&) {} return_type get() {} }; } // namespace asio } // namespace boost void bind_allocator_to_completion_token_v1_test() { io_context ioc; int count = 0; int allocations = 0; timer t(ioc, chronons::seconds(1)); t.async_wait( bind_allocator( test_allocator(&allocations), incrementer_token_v1(&count))); BOOST_ASIO_CHECK(count == 0); BOOST_ASIO_CHECK(allocations == 1); ioc.run(); BOOST_ASIO_CHECK(count == 1); BOOST_ASIO_CHECK(allocations == 0); } struct incrementer_token_v2 { explicit incrementer_token_v2(int* c) : count(c) {} int* count; }; namespace boost { namespace asio { template <> class async_result { public: typedef void return_type; #if defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES) template static void initiate(Initiation initiation, incrementer_token_v2 token, BOOST_ASIO_MOVE_ARG(Args)... args) { initiation(bindns::bind(&increment, token.count), BOOST_ASIO_MOVE_CAST(Args)(args)...); } #else // defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES) template static void initiate(Initiation initiation, incrementer_token_v2 token) { initiation(bindns::bind(&increment, token.count)); } #define BOOST_ASIO_PRIVATE_INITIATE_DEF(n) \ template \ static return_type initiate(Initiation initiation, \ incrementer_token_v2 token, BOOST_ASIO_VARIADIC_MOVE_PARAMS(n)) \ { \ initiation(bindns::bind(&increment, token.count), \ BOOST_ASIO_VARIADIC_MOVE_ARGS(n)); \ } \ /**/ BOOST_ASIO_VARIADIC_GENERATE(BOOST_ASIO_PRIVATE_INITIATE_DEF) #undef BOOST_ASIO_PRIVATE_INITIATE_DEF #endif // defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES) }; } // namespace asio } // namespace boost void bind_allocator_to_completion_token_v2_test() { io_context ioc; int count = 0; int allocations = 0; timer t(ioc, chronons::seconds(1)); t.async_wait( bind_allocator( test_allocator(&allocations), incrementer_token_v2(&count))); BOOST_ASIO_CHECK(count == 0); BOOST_ASIO_CHECK(allocations == 1); ioc.run(); BOOST_ASIO_CHECK(count == 1); BOOST_ASIO_CHECK(allocations == 0); } BOOST_ASIO_TEST_SUITE ( "bind_allocator", BOOST_ASIO_TEST_CASE(bind_allocator_to_function_object_test) BOOST_ASIO_TEST_CASE(bind_allocator_to_completion_token_v1_test) BOOST_ASIO_TEST_CASE(bind_allocator_to_completion_token_v2_test) )