1 //
2 // detail/wrapped_handler.hpp
3 // ~~~~~~~~~~~~~~~~~~~~~~~~~~
4 //
5 // Copyright (c) 2003-2015 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 
11 #ifndef ASIO_DETAIL_WRAPPED_HANDLER_HPP
12 #define ASIO_DETAIL_WRAPPED_HANDLER_HPP
13 
14 
15 #include "asio/detail/bind_handler.hpp"
16 #include "asio/detail/handler_alloc_helpers.hpp"
17 #include "asio/detail/handler_cont_helpers.hpp"
18 #include "asio/detail/handler_invoke_helpers.hpp"
19 
20 #include "asio/detail/push_options.hpp"
21 
22 namespace asio {
23 namespace detail {
24 
25 struct is_continuation_delegated
26 {
27   template <typename Dispatcher, typename Handler>
operator ()asio::detail::is_continuation_delegated28   bool operator()(Dispatcher&, Handler& handler) const
29   {
30     return asio_handler_cont_helpers::is_continuation(handler);
31   }
32 };
33 
34 struct is_continuation_if_running
35 {
36   template <typename Dispatcher, typename Handler>
operator ()asio::detail::is_continuation_if_running37   bool operator()(Dispatcher& dispatcher, Handler&) const
38   {
39     return dispatcher.running_in_this_thread();
40   }
41 };
42 
43 template <typename Dispatcher, typename Handler,
44     typename IsContinuation = is_continuation_delegated>
45 class wrapped_handler
46 {
47 public:
48   typedef void result_type;
49 
wrapped_handler(Dispatcher dispatcher,Handler & handler)50   wrapped_handler(Dispatcher dispatcher, Handler& handler)
51     : dispatcher_(dispatcher),
52       handler_(ASIO_MOVE_CAST(Handler)(handler))
53   {
54   }
55 
wrapped_handler(const wrapped_handler & other)56   wrapped_handler(const wrapped_handler& other)
57     : dispatcher_(other.dispatcher_),
58       handler_(other.handler_)
59   {
60   }
61 
wrapped_handler(wrapped_handler && other)62   wrapped_handler(wrapped_handler&& other)
63     : dispatcher_(other.dispatcher_),
64       handler_(ASIO_MOVE_CAST(Handler)(other.handler_))
65   {
66   }
67 
operator ()()68   void operator()()
69   {
70     dispatcher_.dispatch(ASIO_MOVE_CAST(Handler)(handler_));
71   }
72 
operator ()() const73   void operator()() const
74   {
75     dispatcher_.dispatch(handler_);
76   }
77 
78   template <typename Arg1>
operator ()(const Arg1 & arg1)79   void operator()(const Arg1& arg1)
80   {
81     dispatcher_.dispatch(detail::bind_handler(handler_, arg1));
82   }
83 
84   template <typename Arg1>
operator ()(const Arg1 & arg1) const85   void operator()(const Arg1& arg1) const
86   {
87     dispatcher_.dispatch(detail::bind_handler(handler_, arg1));
88   }
89 
90   template <typename Arg1, typename Arg2>
operator ()(const Arg1 & arg1,const Arg2 & arg2)91   void operator()(const Arg1& arg1, const Arg2& arg2)
92   {
93     dispatcher_.dispatch(detail::bind_handler(handler_, arg1, arg2));
94   }
95 
96   template <typename Arg1, typename Arg2>
operator ()(const Arg1 & arg1,const Arg2 & arg2) const97   void operator()(const Arg1& arg1, const Arg2& arg2) const
98   {
99     dispatcher_.dispatch(detail::bind_handler(handler_, arg1, arg2));
100   }
101 
102   template <typename Arg1, typename Arg2, typename Arg3>
operator ()(const Arg1 & arg1,const Arg2 & arg2,const Arg3 & arg3)103   void operator()(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3)
104   {
105     dispatcher_.dispatch(detail::bind_handler(handler_, arg1, arg2, arg3));
106   }
107 
108   template <typename Arg1, typename Arg2, typename Arg3>
operator ()(const Arg1 & arg1,const Arg2 & arg2,const Arg3 & arg3) const109   void operator()(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3) const
110   {
111     dispatcher_.dispatch(detail::bind_handler(handler_, arg1, arg2, arg3));
112   }
113 
114   template <typename Arg1, typename Arg2, typename Arg3, typename Arg4>
operator ()(const Arg1 & arg1,const Arg2 & arg2,const Arg3 & arg3,const Arg4 & arg4)115   void operator()(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3,
116       const Arg4& arg4)
117   {
118     dispatcher_.dispatch(
119         detail::bind_handler(handler_, arg1, arg2, arg3, arg4));
120   }
121 
122   template <typename Arg1, typename Arg2, typename Arg3, typename Arg4>
operator ()(const Arg1 & arg1,const Arg2 & arg2,const Arg3 & arg3,const Arg4 & arg4) const123   void operator()(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3,
124       const Arg4& arg4) const
125   {
126     dispatcher_.dispatch(
127         detail::bind_handler(handler_, arg1, arg2, arg3, arg4));
128   }
129 
130   template <typename Arg1, typename Arg2, typename Arg3, typename Arg4,
131       typename Arg5>
operator ()(const Arg1 & arg1,const Arg2 & arg2,const Arg3 & arg3,const Arg4 & arg4,const Arg5 & arg5)132   void operator()(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3,
133       const Arg4& arg4, const Arg5& arg5)
134   {
135     dispatcher_.dispatch(
136         detail::bind_handler(handler_, arg1, arg2, arg3, arg4, arg5));
137   }
138 
139   template <typename Arg1, typename Arg2, typename Arg3, typename Arg4,
140       typename Arg5>
operator ()(const Arg1 & arg1,const Arg2 & arg2,const Arg3 & arg3,const Arg4 & arg4,const Arg5 & arg5) const141   void operator()(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3,
142       const Arg4& arg4, const Arg5& arg5) const
143   {
144     dispatcher_.dispatch(
145         detail::bind_handler(handler_, arg1, arg2, arg3, arg4, arg5));
146   }
147 
148 //private:
149   Dispatcher dispatcher_;
150   Handler handler_;
151 };
152 
153 template <typename Handler, typename Context>
154 class rewrapped_handler
155 {
156 public:
rewrapped_handler(Handler & handler,const Context & context)157   explicit rewrapped_handler(Handler& handler, const Context& context)
158     : context_(context),
159       handler_(ASIO_MOVE_CAST(Handler)(handler))
160   {
161   }
162 
rewrapped_handler(const Handler & handler,const Context & context)163   explicit rewrapped_handler(const Handler& handler, const Context& context)
164     : context_(context),
165       handler_(handler)
166   {
167   }
168 
rewrapped_handler(const rewrapped_handler & other)169   rewrapped_handler(const rewrapped_handler& other)
170     : context_(other.context_),
171       handler_(other.handler_)
172   {
173   }
174 
rewrapped_handler(rewrapped_handler && other)175   rewrapped_handler(rewrapped_handler&& other)
176     : context_(ASIO_MOVE_CAST(Context)(other.context_)),
177       handler_(ASIO_MOVE_CAST(Handler)(other.handler_))
178   {
179   }
180 
operator ()()181   void operator()()
182   {
183     handler_();
184   }
185 
operator ()() const186   void operator()() const
187   {
188     handler_();
189   }
190 
191 //private:
192   Context context_;
193   Handler handler_;
194 };
195 
196 template <typename Dispatcher, typename Handler, typename IsContinuation>
asio_handler_allocate(std::size_t size,wrapped_handler<Dispatcher,Handler,IsContinuation> * this_handler)197 inline void* asio_handler_allocate(std::size_t size,
198     wrapped_handler<Dispatcher, Handler, IsContinuation>* this_handler)
199 {
200   return asio_handler_alloc_helpers::allocate(
201       size, this_handler->handler_);
202 }
203 
204 template <typename Dispatcher, typename Handler, typename IsContinuation>
asio_handler_deallocate(void * pointer,std::size_t size,wrapped_handler<Dispatcher,Handler,IsContinuation> * this_handler)205 inline void asio_handler_deallocate(void* pointer, std::size_t size,
206     wrapped_handler<Dispatcher, Handler, IsContinuation>* this_handler)
207 {
208   asio_handler_alloc_helpers::deallocate(
209       pointer, size, this_handler->handler_);
210 }
211 
212 template <typename Dispatcher, typename Handler, typename IsContinuation>
asio_handler_is_continuation(wrapped_handler<Dispatcher,Handler,IsContinuation> * this_handler)213 inline bool asio_handler_is_continuation(
214     wrapped_handler<Dispatcher, Handler, IsContinuation>* this_handler)
215 {
216   return IsContinuation()(this_handler->dispatcher_, this_handler->handler_);
217 }
218 
219 template <typename Function, typename Dispatcher,
220     typename Handler, typename IsContinuation>
asio_handler_invoke(Function & function,wrapped_handler<Dispatcher,Handler,IsContinuation> * this_handler)221 inline void asio_handler_invoke(Function& function,
222     wrapped_handler<Dispatcher, Handler, IsContinuation>* this_handler)
223 {
224   this_handler->dispatcher_.dispatch(
225       rewrapped_handler<Function, Handler>(
226         function, this_handler->handler_));
227 }
228 
229 template <typename Function, typename Dispatcher,
230     typename Handler, typename IsContinuation>
asio_handler_invoke(const Function & function,wrapped_handler<Dispatcher,Handler,IsContinuation> * this_handler)231 inline void asio_handler_invoke(const Function& function,
232     wrapped_handler<Dispatcher, Handler, IsContinuation>* this_handler)
233 {
234   this_handler->dispatcher_.dispatch(
235       rewrapped_handler<Function, Handler>(
236         function, this_handler->handler_));
237 }
238 
239 template <typename Handler, typename Context>
asio_handler_allocate(std::size_t size,rewrapped_handler<Handler,Context> * this_handler)240 inline void* asio_handler_allocate(std::size_t size,
241     rewrapped_handler<Handler, Context>* this_handler)
242 {
243   return asio_handler_alloc_helpers::allocate(
244       size, this_handler->context_);
245 }
246 
247 template <typename Handler, typename Context>
asio_handler_deallocate(void * pointer,std::size_t size,rewrapped_handler<Handler,Context> * this_handler)248 inline void asio_handler_deallocate(void* pointer, std::size_t size,
249     rewrapped_handler<Handler, Context>* this_handler)
250 {
251   asio_handler_alloc_helpers::deallocate(
252       pointer, size, this_handler->context_);
253 }
254 
255 template <typename Dispatcher, typename Context>
asio_handler_is_continuation(rewrapped_handler<Dispatcher,Context> * this_handler)256 inline bool asio_handler_is_continuation(
257     rewrapped_handler<Dispatcher, Context>* this_handler)
258 {
259   return asio_handler_cont_helpers::is_continuation(
260       this_handler->context_);
261 }
262 
263 template <typename Function, typename Handler, typename Context>
asio_handler_invoke(Function & function,rewrapped_handler<Handler,Context> * this_handler)264 inline void asio_handler_invoke(Function& function,
265     rewrapped_handler<Handler, Context>* this_handler)
266 {
267   asio_handler_invoke_helpers::invoke(
268       function, this_handler->context_);
269 }
270 
271 template <typename Function, typename Handler, typename Context>
asio_handler_invoke(const Function & function,rewrapped_handler<Handler,Context> * this_handler)272 inline void asio_handler_invoke(const Function& function,
273     rewrapped_handler<Handler, Context>* this_handler)
274 {
275   asio_handler_invoke_helpers::invoke(
276       function, this_handler->context_);
277 }
278 
279 } // namespace detail
280 } // namespace asio
281 
282 #include "asio/detail/pop_options.hpp"
283 
284 #endif // ASIO_DETAIL_WRAPPED_HANDLER_HPP
285