AimRT/_deps/boost-src/libs/asio/test/bind_cancellation_slot.cpp
2025-01-12 20:37:50 +08:00

226 lines
5.3 KiB
C++

//
// bind_cancellation_slot.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 <boost/asio/bind_cancellation_slot.hpp>
#include <boost/asio/cancellation_signal.hpp>
#include <boost/asio/io_context.hpp>
#include <boost/asio/steady_timer.hpp>
#include "unit_test.hpp"
#if defined(BOOST_ASIO_HAS_BOOST_DATE_TIME)
# include <boost/asio/deadline_timer.hpp>
#else // defined(BOOST_ASIO_HAS_BOOST_DATE_TIME)
# include <boost/asio/steady_timer.hpp>
#endif // defined(BOOST_ASIO_HAS_BOOST_DATE_TIME)
#if defined(BOOST_ASIO_HAS_BOOST_BIND)
# include <boost/bind/bind.hpp>
#else // defined(BOOST_ASIO_HAS_BOOST_BIND)
# include <functional>
#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)
void increment_on_cancel(int* count, const boost::system::error_code& error)
{
if (error == boost::asio::error::operation_aborted)
++(*count);
}
void bind_cancellation_slot_to_function_object_test()
{
io_context ioc;
cancellation_signal sig;
int count = 0;
timer t(ioc, chronons::seconds(5));
t.async_wait(
bind_cancellation_slot(sig.slot(),
bindns::bind(&increment_on_cancel,
&count, bindns::placeholders::_1)));
ioc.poll();
BOOST_ASIO_CHECK(count == 0);
sig.emit(boost::asio::cancellation_type::all);
ioc.run();
BOOST_ASIO_CHECK(count == 1);
}
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 error)
{
increment_on_cancel(count, error);
}
int* count;
};
namespace boost {
namespace asio {
template <>
class async_result<incrementer_token_v1, void(boost::system::error_code)>
{
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_cancellation_slot_to_completion_token_v1_test()
{
io_context ioc;
cancellation_signal sig;
int count = 0;
timer t(ioc, chronons::seconds(5));
t.async_wait(
bind_cancellation_slot(sig.slot(),
incrementer_token_v1(&count)));
ioc.poll();
BOOST_ASIO_CHECK(count == 0);
sig.emit(boost::asio::cancellation_type::all);
ioc.run();
BOOST_ASIO_CHECK(count == 1);
}
struct incrementer_token_v2
{
explicit incrementer_token_v2(int* c) : count(c) {}
int* count;
};
namespace boost {
namespace asio {
template <>
class async_result<incrementer_token_v2, void(boost::system::error_code)>
{
public:
typedef void return_type;
#if defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES)
template <typename Initiation, typename... Args>
static void initiate(Initiation initiation,
incrementer_token_v2 token, BOOST_ASIO_MOVE_ARG(Args)... args)
{
initiation(
bindns::bind(&increment_on_cancel,
token.count, bindns::placeholders::_1),
BOOST_ASIO_MOVE_CAST(Args)(args)...);
}
#else // defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES)
template <typename Initiation>
static void initiate(Initiation initiation, incrementer_token_v2 token)
{
initiation(
bindns::bind(&increment_on_cancel,
token.count, bindns::placeholders::_1));
}
#define BOOST_ASIO_PRIVATE_INITIATE_DEF(n) \
template <typename Initiation, BOOST_ASIO_VARIADIC_TPARAMS(n)> \
static return_type initiate(Initiation initiation, \
incrementer_token_v2 token, BOOST_ASIO_VARIADIC_MOVE_PARAMS(n)) \
{ \
initiation( \
bindns::bind(&increment_on_cancel, \
token.count, bindns::placeholders::_1), \
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_cancellation_slot_to_completion_token_v2_test()
{
io_context ioc;
cancellation_signal sig;
int count = 0;
timer t(ioc, chronons::seconds(5));
t.async_wait(
bind_cancellation_slot(sig.slot(),
incrementer_token_v2(&count)));
ioc.poll();
BOOST_ASIO_CHECK(count == 0);
sig.emit(boost::asio::cancellation_type::all);
ioc.run();
BOOST_ASIO_CHECK(count == 1);
}
BOOST_ASIO_TEST_SUITE
(
"bind_cancellation_slot",
BOOST_ASIO_TEST_CASE(bind_cancellation_slot_to_function_object_test)
BOOST_ASIO_TEST_CASE(bind_cancellation_slot_to_completion_token_v1_test)
BOOST_ASIO_TEST_CASE(bind_cancellation_slot_to_completion_token_v2_test)
)