Asio Extensions
Additional functionality built on top of (Boost.)Asio
composed_operation.hpp
1 /// @file
2 /// Declares the asioext::composed_operation utility class.
3 ///
4 /// @copyright Copyright (c) 2016 Tim Niederhausen (tim@rnc-ag.de)
5 /// Distributed under the Boost Software License, Version 1.0.
6 /// (See accompanying file LICENSE_1_0.txt or copy at
7 /// http://www.boost.org/LICENSE_1_0.txt)
8 
9 #ifndef ASIOEXT_COMPOSEDOPERATION_HPP
10 #define ASIOEXT_COMPOSEDOPERATION_HPP
11 
12 #include "asioext/detail/config.hpp"
13 
14 #if ASIOEXT_HAS_PRAGMA_ONCE
15 # pragma once
16 #endif
17 
18 #include "asioext/detail/asio_version.hpp"
19 #include "asioext/detail/move_support.hpp"
20 
21 #if defined(ASIOEXT_USE_BOOST_ASIO)
22 # include <boost/asio/detail/handler_alloc_helpers.hpp>
23 # include <boost/asio/detail/handler_cont_helpers.hpp>
24 # include <boost/asio/detail/handler_invoke_helpers.hpp>
25 # define ASIOEXT_HANDLER_ALLOC_HELPERS_NS boost_asio_handler_alloc_helpers
26 # define ASIOEXT_HANDLER_CONT_HELPERS_NS boost_asio_handler_cont_helpers
27 # define ASIOEXT_HANDLER_INVOKE_HELPERS_NS boost_asio_handler_invoke_helpers
28 #else
29 # include <asio/detail/handler_alloc_helpers.hpp>
30 # include <asio/detail/handler_cont_helpers.hpp>
31 # include <asio/detail/handler_invoke_helpers.hpp>
32 # define ASIOEXT_HANDLER_ALLOC_HELPERS_NS asio_handler_alloc_helpers
33 # define ASIOEXT_HANDLER_CONT_HELPERS_NS asio_handler_cont_helpers
34 # define ASIOEXT_HANDLER_INVOKE_HELPERS_NS asio_handler_invoke_helpers
35 #endif
36 
37 #if (ASIOEXT_ASIO_VERSION >= 101100)
38 # if defined(ASIOEXT_USE_BOOST_ASIO)
39 # include <boost/asio/associated_allocator.hpp>
40 # include <boost/asio/associated_executor.hpp>
41 # else
42 # include <asio/associated_allocator.hpp>
43 # include <asio/associated_executor.hpp>
44 # endif
45 #endif
46 
47 ASIOEXT_NS_BEGIN
48 
49 /// @ingroup core
50 /// @brief Base class for composed operations.
51 ///
52 /// This class template is intended to be used as a base class for custom
53 /// composed operation types that wrap a user-provided handler.
54 /// As such, it expects child classes to implement <code>operator()</code>.
55 ///
56 /// It wraps the given handler object and provides
57 /// overloads for asio's hooks that forward to the real handler's hooks
58 /// (if implemented).
59 ///
60 /// For a detailed description of `composed operations`, see
61 /// [async.reqmts.async.composed] inside the Networking TS.
62 ///
63 /// Implemented hooks:
64 /// * asio_handler_allocate()
65 /// * asio_handler_deallocate()
66 /// * asio_handler_is_continuation()
67 /// * asio_handler_invoke()
68 ///
69 /// This class does **not** provide support for
70 /// @c asio::associated_allocator<> and
71 /// @c asio::associated_executor<> (Asio 1.11.0+). The user is required
72 /// to manually specialize these templates in the asio namespace.
73 ///
74 /// Example specializations:
75 /// @code
76 /// namespace asio {
77 ///
78 /// template <typename Handler, typename Allocator>
79 /// struct associated_allocator<myns::my_operation<Handler>, Allocator>
80 /// {
81 /// typedef typename associated_allocator<Handler, Allocator>::type type;
82 ///
83 /// static type get(const myns::my_operation<Handler>& h,
84 /// const Allocator& a = Allocator()) ASIOEXT_NOEXCEPT
85 /// {
86 /// return associated_allocator<Handler, Allocator>::get(h.handler_, a);
87 /// }
88 /// };
89 ///
90 /// template <typename Handler, typename Executor>
91 /// struct associated_executor<myns::my_operation<Handler>, Executor>
92 /// {
93 /// typedef typename associated_executor<Handler, Executor>::type type;
94 ///
95 /// static type get(const myns::my_operation<Handler>& h,
96 /// const Executor& ex = Executor()) ASIOEXT_NOEXCEPT
97 /// {
98 /// return associated_executor<Handler, Executor>::get(h.handler_, ex);
99 /// }
100 /// };
101 ///
102 /// }
103 /// @endcode
104 ///
105 ///
106 /// @note This type's <code>operator()</code> is executed by the
107 /// user-specified executor / invocation hook (see above).
108 /// It is not suitable for operations that are to be executed
109 /// in a service-provided context (e.g. a private io_service).
110 template <typename Handler>
112 {
113 public:
114  composed_operation(ASIOEXT_MOVE_ARG(Handler) handler)
115  : handler_(ASIOEXT_MOVE_CAST(Handler)(handler))
116  {
117  // ctor
118  }
119 
120 // TODO(tim): Make this protected.
121 //protected:
122  Handler handler_;
123 };
124 
125 #if !defined(ASIOEXT_IS_DOCUMENTATION)
126 template <typename Handler>
127 inline void* asio_handler_allocate(std::size_t size,
128  composed_operation<Handler>* this_handler)
129 {
130  return ASIOEXT_HANDLER_ALLOC_HELPERS_NS::allocate(
131  size, this_handler->handler_);
132 }
133 
134 template <typename Handler>
135 inline void asio_handler_deallocate(void* pointer, std::size_t size,
136  composed_operation<Handler>* this_handler)
137 {
138  ASIOEXT_HANDLER_ALLOC_HELPERS_NS::deallocate(
139  pointer, size, this_handler->handler_);
140 }
141 
142 template <typename Handler>
143 inline bool asio_handler_is_continuation(
144  composed_operation<Handler>* this_handler)
145 {
146  return ASIOEXT_HANDLER_CONT_HELPERS_NS::is_continuation(
147  this_handler->handler_);
148 }
149 
150 template <typename Function, typename Handler>
151 inline void asio_handler_invoke(Function& function,
152  composed_operation<Handler>* this_handler)
153 {
154  ASIOEXT_HANDLER_INVOKE_HELPERS_NS::invoke(
155  function, this_handler->handler_);
156 }
157 
158 template <typename Function, typename Handler>
159 inline void asio_handler_invoke(const Function& function,
160  composed_operation<Handler>* this_handler)
161 {
162  ASIOEXT_HANDLER_INVOKE_HELPERS_NS::invoke(
163  function, this_handler->handler_);
164 }
165 #endif
166 
167 ASIOEXT_NS_END
168 
169 #endif
composed_operation(Handler &&handler)
Definition: composed_operation.hpp:114
Base class for composed operations.
Definition: composed_operation.hpp:111
Handler handler_
Definition: composed_operation.hpp:122