2019-08-24 13:39:04 +00:00
|
|
|
//
|
|
|
|
// async_result.hpp
|
|
|
|
// ~~~~~~~~~~~~~~~~
|
|
|
|
//
|
|
|
|
// Copyright (c) 2003-2019 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 BOOST_ASIO_ASYNC_RESULT_HPP
|
|
|
|
#define BOOST_ASIO_ASYNC_RESULT_HPP
|
|
|
|
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER >= 1200)
|
|
|
|
# pragma once
|
|
|
|
#endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
|
|
|
|
|
|
|
|
#include <boost/asio/detail/config.hpp>
|
|
|
|
#include <boost/asio/detail/type_traits.hpp>
|
|
|
|
#include <boost/asio/detail/variadic_templates.hpp>
|
|
|
|
|
|
|
|
#include <boost/asio/detail/push_options.hpp>
|
|
|
|
|
|
|
|
namespace boost {
|
|
|
|
namespace asio {
|
|
|
|
|
|
|
|
/// An interface for customising the behaviour of an initiating function.
|
|
|
|
/**
|
|
|
|
* The async_result traits class is used for determining:
|
|
|
|
*
|
|
|
|
* @li the concrete completion handler type to be called at the end of the
|
|
|
|
* asynchronous operation;
|
|
|
|
*
|
|
|
|
* @li the initiating function return type; and
|
|
|
|
*
|
|
|
|
* @li how the return value of the initiating function is obtained.
|
|
|
|
*
|
|
|
|
* The trait allows the handler and return types to be determined at the point
|
|
|
|
* where the specific completion handler signature is known.
|
|
|
|
*
|
|
|
|
* This template may be specialised for user-defined completion token types.
|
|
|
|
* The primary template assumes that the CompletionToken is the completion
|
|
|
|
* handler.
|
|
|
|
*/
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature>
|
2019-08-24 13:39:04 +00:00
|
|
|
class async_result
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
/// The concrete completion handler type for the specific signature.
|
|
|
|
typedef CompletionToken completion_handler_type;
|
|
|
|
|
|
|
|
/// The return type of the initiating function.
|
|
|
|
typedef void return_type;
|
|
|
|
|
|
|
|
/// Construct an async result from a given handler.
|
|
|
|
/**
|
|
|
|
* When using a specalised async_result, the constructor has an opportunity
|
|
|
|
* to initialise some state associated with the completion handler, which is
|
|
|
|
* then returned from the initiating function.
|
|
|
|
*/
|
|
|
|
explicit async_result(completion_handler_type& h)
|
|
|
|
{
|
|
|
|
(void)h;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Obtain the value to be returned from the initiating function.
|
|
|
|
return_type get()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-02-22 05:33:06 +00:00
|
|
|
#if defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES) \
|
|
|
|
|| defined(GENERATING_DOCUMENTATION)
|
2019-08-24 13:39:04 +00:00
|
|
|
|
|
|
|
/// Initiate the asynchronous operation that will produce the result, and
|
|
|
|
/// obtain the value to be returned from the initiating function.
|
|
|
|
template <typename Initiation, typename RawCompletionToken, typename... Args>
|
|
|
|
static return_type initiate(
|
|
|
|
BOOST_ASIO_MOVE_ARG(Initiation) initiation,
|
|
|
|
BOOST_ASIO_MOVE_ARG(RawCompletionToken) token,
|
|
|
|
BOOST_ASIO_MOVE_ARG(Args)... args)
|
|
|
|
{
|
|
|
|
BOOST_ASIO_MOVE_CAST(Initiation)(initiation)(
|
|
|
|
BOOST_ASIO_MOVE_CAST(RawCompletionToken)(token),
|
|
|
|
BOOST_ASIO_MOVE_CAST(Args)(args)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else // defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES)
|
2020-02-22 05:33:06 +00:00
|
|
|
// || defined(GENERATING_DOCUMENTATION)
|
2019-08-24 13:39:04 +00:00
|
|
|
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename Initiation, typename RawCompletionToken>
|
2019-08-24 13:39:04 +00:00
|
|
|
static return_type initiate(
|
|
|
|
BOOST_ASIO_MOVE_ARG(Initiation) initiation,
|
|
|
|
BOOST_ASIO_MOVE_ARG(RawCompletionToken) token)
|
|
|
|
{
|
|
|
|
BOOST_ASIO_MOVE_CAST(Initiation)(initiation)(
|
|
|
|
BOOST_ASIO_MOVE_CAST(RawCompletionToken)(token));
|
|
|
|
}
|
|
|
|
|
|
|
|
#define BOOST_ASIO_PRIVATE_INITIATE_DEF(n) \
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename Initiation, typename RawCompletionToken, \
|
2019-08-24 13:39:04 +00:00
|
|
|
BOOST_ASIO_VARIADIC_TPARAMS(n)> \
|
|
|
|
static return_type initiate( \
|
|
|
|
BOOST_ASIO_MOVE_ARG(Initiation) initiation, \
|
|
|
|
BOOST_ASIO_MOVE_ARG(RawCompletionToken) token, \
|
|
|
|
BOOST_ASIO_VARIADIC_MOVE_PARAMS(n)) \
|
|
|
|
{ \
|
|
|
|
BOOST_ASIO_MOVE_CAST(Initiation)(initiation)( \
|
|
|
|
BOOST_ASIO_MOVE_CAST(RawCompletionToken)(token), \
|
|
|
|
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)
|
2020-02-22 05:33:06 +00:00
|
|
|
// || defined(GENERATING_DOCUMENTATION)
|
2019-08-24 13:39:04 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
async_result(const async_result&) BOOST_ASIO_DELETED;
|
|
|
|
async_result& operator=(const async_result&) BOOST_ASIO_DELETED;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Helper template to deduce the handler type from a CompletionToken, capture
|
|
|
|
/// a local copy of the handler, and then create an async_result for the
|
|
|
|
/// handler.
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature>
|
2019-08-24 13:39:04 +00:00
|
|
|
struct async_completion
|
|
|
|
{
|
|
|
|
/// The real handler type to be used for the asynchronous operation.
|
|
|
|
typedef typename boost::asio::async_result<
|
|
|
|
typename decay<CompletionToken>::type,
|
|
|
|
Signature>::completion_handler_type completion_handler_type;
|
|
|
|
|
|
|
|
#if defined(BOOST_ASIO_HAS_MOVE) || defined(GENERATING_DOCUMENTATION)
|
|
|
|
/// Constructor.
|
|
|
|
/**
|
|
|
|
* The constructor creates the concrete completion handler and makes the link
|
|
|
|
* between the handler and the asynchronous result.
|
|
|
|
*/
|
|
|
|
explicit async_completion(CompletionToken& token)
|
|
|
|
: completion_handler(static_cast<typename conditional<
|
|
|
|
is_same<CompletionToken, completion_handler_type>::value,
|
|
|
|
completion_handler_type&, CompletionToken&&>::type>(token)),
|
|
|
|
result(completion_handler)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
#else // defined(BOOST_ASIO_HAS_MOVE) || defined(GENERATING_DOCUMENTATION)
|
|
|
|
explicit async_completion(typename decay<CompletionToken>::type& token)
|
|
|
|
: completion_handler(token),
|
|
|
|
result(completion_handler)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
explicit async_completion(const typename decay<CompletionToken>::type& token)
|
|
|
|
: completion_handler(token),
|
|
|
|
result(completion_handler)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
#endif // defined(BOOST_ASIO_HAS_MOVE) || defined(GENERATING_DOCUMENTATION)
|
|
|
|
|
|
|
|
/// A copy of, or reference to, a real handler object.
|
|
|
|
#if defined(BOOST_ASIO_HAS_MOVE) || defined(GENERATING_DOCUMENTATION)
|
|
|
|
typename conditional<
|
|
|
|
is_same<CompletionToken, completion_handler_type>::value,
|
|
|
|
completion_handler_type&, completion_handler_type>::type completion_handler;
|
|
|
|
#else // defined(BOOST_ASIO_HAS_MOVE) || defined(GENERATING_DOCUMENTATION)
|
|
|
|
completion_handler_type completion_handler;
|
|
|
|
#endif // defined(BOOST_ASIO_HAS_MOVE) || defined(GENERATING_DOCUMENTATION)
|
|
|
|
|
|
|
|
/// The result of the asynchronous operation's initiating function.
|
|
|
|
async_result<typename decay<CompletionToken>::type, Signature> result;
|
|
|
|
};
|
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
template <typename CompletionToken, typename Signature>
|
|
|
|
struct async_result_helper
|
|
|
|
: async_result<typename decay<CompletionToken>::type, Signature>
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
struct async_result_memfns_base
|
|
|
|
{
|
|
|
|
void initiate();
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
struct async_result_memfns_derived
|
|
|
|
: T, async_result_memfns_base
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T, T>
|
|
|
|
struct async_result_memfns_check
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename>
|
|
|
|
char (&async_result_initiate_memfn_helper(...))[2];
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
char async_result_initiate_memfn_helper(
|
|
|
|
async_result_memfns_check<
|
|
|
|
void (async_result_memfns_base::*)(),
|
|
|
|
&async_result_memfns_derived<T>::initiate>*);
|
|
|
|
|
|
|
|
template <typename CompletionToken, typename Signature>
|
|
|
|
struct async_result_has_initiate_memfn
|
|
|
|
: integral_constant<bool, sizeof(async_result_initiate_memfn_helper<
|
|
|
|
async_result<typename decay<CompletionToken>::type, Signature>
|
|
|
|
>(0)) != 1>
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace detail
|
|
|
|
|
|
|
|
#if defined(GENERATING_DOCUMENTATION)
|
|
|
|
# define BOOST_ASIO_INITFN_RESULT_TYPE(ct, sig) \
|
|
|
|
void_or_deduced
|
|
|
|
#elif defined(_MSC_VER) && (_MSC_VER < 1500)
|
|
|
|
# define BOOST_ASIO_INITFN_RESULT_TYPE(ct, sig) \
|
|
|
|
typename ::boost::asio::detail::async_result_helper< \
|
|
|
|
ct, sig>::return_type
|
|
|
|
#define BOOST_ASIO_HANDLER_TYPE(ct, sig) \
|
|
|
|
typename ::boost::asio::detail::async_result_helper< \
|
|
|
|
ct, sig>::completion_handler_type
|
|
|
|
#else
|
|
|
|
# define BOOST_ASIO_INITFN_RESULT_TYPE(ct, sig) \
|
|
|
|
typename ::boost::asio::async_result< \
|
|
|
|
typename ::boost::asio::decay<ct>::type, sig>::return_type
|
|
|
|
#define BOOST_ASIO_HANDLER_TYPE(ct, sig) \
|
|
|
|
typename ::boost::asio::async_result< \
|
|
|
|
typename ::boost::asio::decay<ct>::type, sig>::completion_handler_type
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(GENERATING_DOCUMENTATION)
|
|
|
|
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature,
|
2019-08-24 13:39:04 +00:00
|
|
|
typename Initiation, typename... Args>
|
2020-02-22 05:33:06 +00:00
|
|
|
BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, Signature)
|
|
|
|
async_initiate(BOOST_ASIO_MOVE_ARG(Initiation) initiation,
|
2019-08-24 13:39:04 +00:00
|
|
|
BOOST_ASIO_NONDEDUCED_MOVE_ARG(CompletionToken),
|
|
|
|
BOOST_ASIO_MOVE_ARG(Args)... args);
|
|
|
|
|
|
|
|
#elif defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES)
|
|
|
|
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature,
|
2019-08-24 13:39:04 +00:00
|
|
|
typename Initiation, typename... Args>
|
|
|
|
inline typename enable_if<
|
|
|
|
detail::async_result_has_initiate_memfn<CompletionToken, Signature>::value,
|
2020-02-22 05:33:06 +00:00
|
|
|
BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, Signature)>::type
|
2019-08-24 13:39:04 +00:00
|
|
|
async_initiate(BOOST_ASIO_MOVE_ARG(Initiation) initiation,
|
|
|
|
BOOST_ASIO_NONDEDUCED_MOVE_ARG(CompletionToken) token,
|
|
|
|
BOOST_ASIO_MOVE_ARG(Args)... args)
|
|
|
|
{
|
|
|
|
return async_result<typename decay<CompletionToken>::type,
|
|
|
|
Signature>::initiate(BOOST_ASIO_MOVE_CAST(Initiation)(initiation),
|
|
|
|
BOOST_ASIO_MOVE_CAST(CompletionToken)(token),
|
|
|
|
BOOST_ASIO_MOVE_CAST(Args)(args)...);
|
|
|
|
}
|
|
|
|
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature,
|
2019-08-24 13:39:04 +00:00
|
|
|
typename Initiation, typename... Args>
|
|
|
|
inline typename enable_if<
|
|
|
|
!detail::async_result_has_initiate_memfn<CompletionToken, Signature>::value,
|
|
|
|
BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, Signature)>::type
|
|
|
|
async_initiate(BOOST_ASIO_MOVE_ARG(Initiation) initiation,
|
|
|
|
BOOST_ASIO_NONDEDUCED_MOVE_ARG(CompletionToken) token,
|
|
|
|
BOOST_ASIO_MOVE_ARG(Args)... args)
|
|
|
|
{
|
|
|
|
async_completion<CompletionToken, Signature> completion(token);
|
|
|
|
|
|
|
|
BOOST_ASIO_MOVE_CAST(Initiation)(initiation)(
|
|
|
|
BOOST_ASIO_MOVE_CAST(BOOST_ASIO_HANDLER_TYPE(CompletionToken,
|
|
|
|
Signature))(completion.completion_handler),
|
|
|
|
BOOST_ASIO_MOVE_CAST(Args)(args)...);
|
|
|
|
|
|
|
|
return completion.result.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
#else // defined(BOOST_ASIO_HAS_VARIADIC_TEMPLATES)
|
|
|
|
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature, typename Initiation>
|
2019-08-24 13:39:04 +00:00
|
|
|
inline typename enable_if<
|
|
|
|
detail::async_result_has_initiate_memfn<CompletionToken, Signature>::value,
|
2020-02-22 05:33:06 +00:00
|
|
|
BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, Signature)>::type
|
2019-08-24 13:39:04 +00:00
|
|
|
async_initiate(BOOST_ASIO_MOVE_ARG(Initiation) initiation,
|
|
|
|
BOOST_ASIO_NONDEDUCED_MOVE_ARG(CompletionToken) token)
|
|
|
|
{
|
|
|
|
return async_result<typename decay<CompletionToken>::type,
|
|
|
|
Signature>::initiate(BOOST_ASIO_MOVE_CAST(Initiation)(initiation),
|
|
|
|
BOOST_ASIO_MOVE_CAST(CompletionToken)(token));
|
|
|
|
}
|
|
|
|
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature, typename Initiation>
|
2019-08-24 13:39:04 +00:00
|
|
|
inline typename enable_if<
|
|
|
|
!detail::async_result_has_initiate_memfn<CompletionToken, Signature>::value,
|
|
|
|
BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, Signature)>::type
|
|
|
|
async_initiate(BOOST_ASIO_MOVE_ARG(Initiation) initiation,
|
|
|
|
BOOST_ASIO_NONDEDUCED_MOVE_ARG(CompletionToken) token)
|
|
|
|
{
|
|
|
|
async_completion<CompletionToken, Signature> completion(token);
|
|
|
|
|
|
|
|
BOOST_ASIO_MOVE_CAST(Initiation)(initiation)(
|
|
|
|
BOOST_ASIO_MOVE_CAST(BOOST_ASIO_HANDLER_TYPE(CompletionToken,
|
|
|
|
Signature))(completion.completion_handler));
|
|
|
|
|
|
|
|
return completion.result.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
#define BOOST_ASIO_PRIVATE_INITIATE_DEF(n) \
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature, \
|
2019-08-24 13:39:04 +00:00
|
|
|
typename Initiation, BOOST_ASIO_VARIADIC_TPARAMS(n)> \
|
|
|
|
inline typename enable_if< \
|
|
|
|
detail::async_result_has_initiate_memfn< \
|
|
|
|
CompletionToken, Signature>::value, \
|
2020-02-22 05:33:06 +00:00
|
|
|
BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, Signature)>::type \
|
2019-08-24 13:39:04 +00:00
|
|
|
async_initiate(BOOST_ASIO_MOVE_ARG(Initiation) initiation, \
|
|
|
|
BOOST_ASIO_NONDEDUCED_MOVE_ARG(CompletionToken) token, \
|
|
|
|
BOOST_ASIO_VARIADIC_MOVE_PARAMS(n)) \
|
|
|
|
{ \
|
|
|
|
return async_result<typename decay<CompletionToken>::type, \
|
|
|
|
Signature>::initiate(BOOST_ASIO_MOVE_CAST(Initiation)(initiation), \
|
|
|
|
BOOST_ASIO_MOVE_CAST(CompletionToken)(token), \
|
|
|
|
BOOST_ASIO_VARIADIC_MOVE_ARGS(n)); \
|
|
|
|
} \
|
|
|
|
\
|
2020-02-22 05:33:06 +00:00
|
|
|
template <typename CompletionToken, typename Signature, \
|
2019-08-24 13:39:04 +00:00
|
|
|
typename Initiation, BOOST_ASIO_VARIADIC_TPARAMS(n)> \
|
|
|
|
inline typename enable_if< \
|
|
|
|
!detail::async_result_has_initiate_memfn< \
|
|
|
|
CompletionToken, Signature>::value, \
|
|
|
|
BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, Signature)>::type \
|
|
|
|
async_initiate(BOOST_ASIO_MOVE_ARG(Initiation) initiation, \
|
|
|
|
BOOST_ASIO_NONDEDUCED_MOVE_ARG(CompletionToken) token, \
|
|
|
|
BOOST_ASIO_VARIADIC_MOVE_PARAMS(n)) \
|
|
|
|
{ \
|
|
|
|
async_completion<CompletionToken, Signature> completion(token); \
|
|
|
|
\
|
|
|
|
BOOST_ASIO_MOVE_CAST(Initiation)(initiation)( \
|
|
|
|
BOOST_ASIO_MOVE_CAST(BOOST_ASIO_HANDLER_TYPE(CompletionToken, \
|
|
|
|
Signature))(completion.completion_handler), \
|
|
|
|
BOOST_ASIO_VARIADIC_MOVE_ARGS(n)); \
|
|
|
|
\
|
|
|
|
return completion.result.get(); \
|
|
|
|
} \
|
|
|
|
/**/
|
|
|
|
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
|
|
|
|
|
|
|
|
#include <boost/asio/detail/pop_options.hpp>
|
|
|
|
|
|
|
|
#endif // BOOST_ASIO_ASYNC_RESULT_HPP
|