channel_send_functions.hpp 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. //
  2. // experimental/detail/channel_send_functions.hpp
  3. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2023 Christopher M. Kohlhoff (chris at kohlhoff dot com)
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  9. //
  10. #ifndef ASIO_EXPERIMENTAL_DETAIL_CHANNEL_SEND_FUNCTIONS_HPP
  11. #define ASIO_EXPERIMENTAL_DETAIL_CHANNEL_SEND_FUNCTIONS_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include "asio/detail/config.hpp"
  16. #include "asio/async_result.hpp"
  17. #include "asio/detail/type_traits.hpp"
  18. #include "asio/error_code.hpp"
  19. #include "asio/experimental/detail/channel_message.hpp"
  20. #include "asio/detail/push_options.hpp"
  21. namespace asio {
  22. namespace experimental {
  23. namespace detail {
  24. template <typename Derived, typename Executor, typename... Signatures>
  25. class channel_send_functions;
  26. template <typename Derived, typename Executor, typename R, typename... Args>
  27. class channel_send_functions<Derived, Executor, R(Args...)>
  28. {
  29. public:
  30. template <typename... Args2>
  31. enable_if_t<
  32. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  33. bool
  34. > try_send(Args2&&... args)
  35. {
  36. typedef typename detail::channel_message<R(Args...)> message_type;
  37. Derived* self = static_cast<Derived*>(this);
  38. return self->service_->template try_send<message_type>(
  39. self->impl_, false, static_cast<Args2&&>(args)...);
  40. }
  41. template <typename... Args2>
  42. enable_if_t<
  43. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  44. bool
  45. > try_send_via_dispatch(Args2&&... args)
  46. {
  47. typedef typename detail::channel_message<R(Args...)> message_type;
  48. Derived* self = static_cast<Derived*>(this);
  49. return self->service_->template try_send<message_type>(
  50. self->impl_, true, static_cast<Args2&&>(args)...);
  51. }
  52. template <typename... Args2>
  53. enable_if_t<
  54. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  55. std::size_t
  56. > try_send_n(std::size_t count, Args2&&... args)
  57. {
  58. typedef typename detail::channel_message<R(Args...)> message_type;
  59. Derived* self = static_cast<Derived*>(this);
  60. return self->service_->template try_send_n<message_type>(
  61. self->impl_, count, false, static_cast<Args2&&>(args)...);
  62. }
  63. template <typename... Args2>
  64. enable_if_t<
  65. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  66. std::size_t
  67. > try_send_n_via_dispatch(std::size_t count, Args2&&... args)
  68. {
  69. typedef typename detail::channel_message<R(Args...)> message_type;
  70. Derived* self = static_cast<Derived*>(this);
  71. return self->service_->template try_send_n<message_type>(
  72. self->impl_, count, true, static_cast<Args2&&>(args)...);
  73. }
  74. template <
  75. ASIO_COMPLETION_TOKEN_FOR(void (asio::error_code))
  76. CompletionToken ASIO_DEFAULT_COMPLETION_TOKEN_TYPE(Executor)>
  77. auto async_send(Args... args,
  78. CompletionToken&& token
  79. ASIO_DEFAULT_COMPLETION_TOKEN(Executor))
  80. -> decltype(
  81. async_initiate<CompletionToken, void (asio::error_code)>(
  82. declval<typename conditional_t<false, CompletionToken,
  83. Derived>::initiate_async_send>(), token,
  84. declval<typename conditional_t<false, CompletionToken,
  85. Derived>::payload_type>()))
  86. {
  87. typedef typename Derived::payload_type payload_type;
  88. typedef typename detail::channel_message<R(Args...)> message_type;
  89. Derived* self = static_cast<Derived*>(this);
  90. return async_initiate<CompletionToken, void (asio::error_code)>(
  91. typename Derived::initiate_async_send(self), token,
  92. payload_type(message_type(0, static_cast<Args&&>(args)...)));
  93. }
  94. };
  95. template <typename Derived, typename Executor,
  96. typename R, typename... Args, typename... Signatures>
  97. class channel_send_functions<Derived, Executor, R(Args...), Signatures...> :
  98. public channel_send_functions<Derived, Executor, Signatures...>
  99. {
  100. public:
  101. using channel_send_functions<Derived, Executor, Signatures...>::try_send;
  102. using channel_send_functions<Derived, Executor, Signatures...>::async_send;
  103. template <typename... Args2>
  104. enable_if_t<
  105. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  106. bool
  107. > try_send(Args2&&... args)
  108. {
  109. typedef typename detail::channel_message<R(Args...)> message_type;
  110. Derived* self = static_cast<Derived*>(this);
  111. return self->service_->template try_send<message_type>(
  112. self->impl_, false, static_cast<Args2&&>(args)...);
  113. }
  114. template <typename... Args2>
  115. enable_if_t<
  116. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  117. bool
  118. > try_send_via_dispatch(Args2&&... args)
  119. {
  120. typedef typename detail::channel_message<R(Args...)> message_type;
  121. Derived* self = static_cast<Derived*>(this);
  122. return self->service_->template try_send<message_type>(
  123. self->impl_, true, static_cast<Args2&&>(args)...);
  124. }
  125. template <typename... Args2>
  126. enable_if_t<
  127. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  128. std::size_t
  129. > try_send_n(std::size_t count, Args2&&... args)
  130. {
  131. typedef typename detail::channel_message<R(Args...)> message_type;
  132. Derived* self = static_cast<Derived*>(this);
  133. return self->service_->template try_send_n<message_type>(
  134. self->impl_, count, false, static_cast<Args2&&>(args)...);
  135. }
  136. template <typename... Args2>
  137. enable_if_t<
  138. is_constructible<detail::channel_message<R(Args...)>, int, Args2...>::value,
  139. std::size_t
  140. > try_send_n_via_dispatch(std::size_t count, Args2&&... args)
  141. {
  142. typedef typename detail::channel_message<R(Args...)> message_type;
  143. Derived* self = static_cast<Derived*>(this);
  144. return self->service_->template try_send_n<message_type>(
  145. self->impl_, count, true, static_cast<Args2&&>(args)...);
  146. }
  147. template <
  148. ASIO_COMPLETION_TOKEN_FOR(void (asio::error_code))
  149. CompletionToken ASIO_DEFAULT_COMPLETION_TOKEN_TYPE(Executor)>
  150. auto async_send(Args... args,
  151. CompletionToken&& token
  152. ASIO_DEFAULT_COMPLETION_TOKEN(Executor))
  153. -> decltype(
  154. async_initiate<CompletionToken, void (asio::error_code)>(
  155. declval<typename conditional_t<false, CompletionToken,
  156. Derived>::initiate_async_send>(), token,
  157. declval<typename conditional_t<false, CompletionToken,
  158. Derived>::payload_type>()))
  159. {
  160. typedef typename Derived::payload_type payload_type;
  161. typedef typename detail::channel_message<R(Args...)> message_type;
  162. Derived* self = static_cast<Derived*>(this);
  163. return async_initiate<CompletionToken, void (asio::error_code)>(
  164. typename Derived::initiate_async_send(self), token,
  165. payload_type(message_type(0, static_cast<Args&&>(args)...)));
  166. }
  167. };
  168. } // namespace detail
  169. } // namespace experimental
  170. } // namespace asio
  171. #include "asio/detail/pop_options.hpp"
  172. #endif // ASIO_EXPERIMENTAL_DETAIL_CHANNEL_SEND_FUNCTIONS_HPP