Asio Extensions
Additional functionality built on top of (Boost.)Asio
associated_allocator.hpp
1 /// @file
2 /// Declares the asioext::hook_allocator class and the
3 /// asioext::associated_allocator trait.
4 ///
5 /// @copyright Copyright (c) 2016 Tim Niederhausen (tim@rnc-ag.de)
6 /// Distributed under the Boost Software License, Version 1.0.
7 /// (See accompanying file LICENSE_1_0.txt or copy at
8 /// http://www.boost.org/LICENSE_1_0.txt)
9 
10 #ifndef ASIOEXT_ASSOCIATEDALLOCATOR_HPP
11 #define ASIOEXT_ASSOCIATEDALLOCATOR_HPP
12 
13 #include "asioext/detail/config.hpp"
14 
15 #if ASIOEXT_HAS_PRAGMA_ONCE
16 # pragma once
17 #endif
18 
19 #include "asioext/detail/asio_version.hpp"
20 
21 #if defined(ASIOEXT_USE_BOOST_ASIO)
22 # include <boost/asio/detail/handler_alloc_helpers.hpp>
23 #else
24 # include <asio/detail/handler_alloc_helpers.hpp>
25 #endif
26 
27 #if (ASIOEXT_ASIO_VERSION >= 101100)
28 # if defined(ASIOEXT_USE_BOOST_ASIO)
29 # include <boost/asio/associated_allocator.hpp>
30 # else
31 # include <asio/associated_allocator.hpp>
32 # endif
33 #endif
34 
35 #if defined(ASIOEXT_USE_BOOST_ASIO)
36 # define ASIOEXT_HANDLER_ALLOC_HELPERS_NS boost_asio_handler_alloc_helpers
37 #else
38 # define ASIOEXT_HANDLER_ALLOC_HELPERS_NS asio_handler_alloc_helpers
39 #endif
40 
41 ASIOEXT_NS_BEGIN
42 
43 #if !defined(ASIOEXT_IS_DOCUMENTATION)
44 template <typename T, typename Handler>
45 class hook_allocator;
46 
47 namespace detail {
48 
49 template <typename Handler, typename Allocator>
50 struct associated_allocator_aux
51 {
52  typedef Allocator type;
53 
54  static type get(Handler&, const Allocator& a) ASIOEXT_NOEXCEPT
55  {
56  return a;
57  }
58 };
59 
60 template <typename Handler, typename T>
61 struct associated_allocator_aux<Handler, std::allocator<T> >
62 {
63  typedef hook_allocator<T, Handler> type;
64 
65  static type get(Handler& handler, const std::allocator<T>&) ASIOEXT_NOEXCEPT
66  {
67  return type(handler);
68  }
69 };
70 
71 }
72 #endif
73 
74 /// @ingroup compat
75 /// @brief Allocator that uses a handler's memory allocation hooks.
76 ///
77 /// Asio @c Handlers (before Asio 1.11.0+) use two hooks to allow memory
78 /// allocation to be customized:
79 ///
80 /// * asio_handler_allocate()
81 /// * asio_handler_deallocate()
82 ///
83 /// This stateless allocator uses these hooks to allocate/free memory.
84 /// @note The allocated memory's lifetime is dependent on the handler's
85 /// lifetime. All memory must be de-allocated before the user's handler
86 /// is invoked ([async.reqmts.async.alloc]).
87 template <typename T, typename Handler>
89 {
90  template <class U, typename Handler2>
91  friend class hook_allocator;
92 
93  template <class U>
94  friend bool operator==(const hook_allocator&,
96  ASIOEXT_NOEXCEPT;
97 
98  template <class U>
99  friend bool operator!=(const hook_allocator&,
101  ASIOEXT_NOEXCEPT;
102 
103 public:
104  typedef T value_type;
105 
106  explicit hook_allocator(Handler& h) ASIOEXT_NOEXCEPT
107  : handler_(h)
108  {
109  }
110 
111  template <typename U>
113  const hook_allocator<U, Handler>& a) ASIOEXT_NOEXCEPT
114  : handler_(a.handler_)
115  {
116  }
117 
119  {
120  return static_cast<T*>(
121  ASIOEXT_HANDLER_ALLOC_HELPERS_NS::allocate(sizeof(T) * n, handler_));
122  }
123 
124  void deallocate(T* p, std::size_t n)
125  {
126  ASIOEXT_HANDLER_ALLOC_HELPERS_NS::deallocate(p, sizeof(T) * n, handler_);
127  }
128 
129 private:
130  Handler& handler_;
131 };
132 
133 #if !defined(ASIOEXT_IS_DOCUMENTATION)
134 template <typename Handler>
135 class hook_allocator<void, Handler>
136 {
137  template <class U, typename Handler2>
138  friend class hook_allocator;
139 
140  template <class U>
141  friend bool operator==(const hook_allocator&,
142  const hook_allocator<U, Handler>&)
143  ASIOEXT_NOEXCEPT;
144 
145  template <class U>
146  friend bool operator!=(const hook_allocator&,
147  const hook_allocator<U, Handler>&)
148  ASIOEXT_NOEXCEPT;
149 
150 public:
151  typedef void value_type;
152 
153  explicit hook_allocator(Handler& h) ASIOEXT_NOEXCEPT
154  : handler_(h)
155  {
156  }
157 
158  template <typename U>
159  hook_allocator(const hook_allocator<U, Handler>& a) ASIOEXT_NOEXCEPT
160  : handler_(a.handler_)
161  {
162  }
163 
164 private:
165  Handler& handler_;
166 };
167 #endif
168 
169 template <typename Handler, class T, class U>
171  const hook_allocator<U, Handler>& b) ASIOEXT_NOEXCEPT
172 { return std::addressof(a.handler_) == std::addressof(b.handler_); }
173 
174 template <typename Handler, class T, class U>
175 bool operator!=(const hook_allocator<T, Handler>& a,
176  const hook_allocator<U, Handler>& b) ASIOEXT_NOEXCEPT
177 { return std::addressof(a.handler_) != std::addressof(b.handler_); }
178 
179 /// @ingroup compat
180 /// @brief Helper trait to obtain the allocator associated with a handler.
181 ///
182 /// Retrieves an @c Allocator that is used to allocate memory tied to the
183 /// handler's asynchronous operation.
184 ///
185 /// Supports the @c associated_allocator trait introduced by
186 /// the Networking TS / Asio 1.11.0+, as well as Asio's memory allocation
187 /// hooks (which this trait will wrap in a @c hook_allocator)
188 ///
189 /// @note This trait is useful for libraries / applications that need to
190 /// deal with both types of allocation customizations.
191 template <typename Handler>
193 {
194 #if (ASIOEXT_ASIO_VERSION >= 101100)
195  typedef typename asio::associated_allocator<Handler>::type
197 #else
199 #endif
200 
201  typedef detail::associated_allocator_aux<Handler,
202  associated_allocator_type> helper_type;
203 
204 public:
205 #if defined(ASIOEXT_IS_DOCUMENTATION)
206  /// @brief The Handler's ProtoAllocator.
207  ///
208  /// A type meeting ProtoAllocator requirements that if rebound
209  /// can be used to allocate memory using the handler's allocator.
210  typedef implementation_defined type;
211 #else
212  typedef typename helper_type::type type;
213 #endif
214 
215  /// @brief Get a ProtoAllocator for the given handler.
216  static type get(Handler& handler) ASIOEXT_NOEXCEPT
217  {
218 #if (ASIOEXT_ASIO_VERSION >= 101100)
219  return helper_type::get(handler, asio::get_associated_allocator(handler));
220 #else
221  return helper_type::get(handler, associated_allocator_type());
222 #endif
223  }
224 };
225 
226 /// @ingroup compat
227 /// @brief Get the handler's associated allocator.
228 ///
229 /// Calling this function is equivalent to calling
230 /// <code>associated_allocator<Handler>::get(handler)</code>.
231 /// See @c associated_allocator for details.
232 template <typename Handler>
233 typename associated_allocator<Handler>::type
234  get_associated_allocator(Handler& handler) ASIOEXT_NOEXCEPT
235 {
236  return associated_allocator<Handler>::get(handler);
237 }
238 
239 #if defined(ASIOEXT_HAS_ALIAS_TEMPLATES)
240 /// @ingroup compat
241 /// @brief Alias for associated_allocator::type
242 template <typename Handler>
244 #endif
245 
246 /// @}
247 
248 ASIOEXT_NS_END
249 
250 #endif
T value_type
Definition: associated_allocator.hpp:104
implementation_defined type
The Handler&#39;s ProtoAllocator.
Definition: associated_allocator.hpp:210
typename associated_allocator< Handler >::type associated_allocator_t
Alias for associated_allocator::type.
Definition: associated_allocator.hpp:243
Helper trait to obtain the allocator associated with a handler.
Definition: associated_allocator.hpp:192
STL namespace.
STL class.
associated_allocator< Handler >::type get_associated_allocator(Handler &handler) noexcept
Get the handler&#39;s associated allocator.
Definition: associated_allocator.hpp:234
bool operator==(const hook_allocator< T, Handler > &a, const hook_allocator< U, Handler > &b) noexcept
Definition: associated_allocator.hpp:170
Allocator that uses a handler&#39;s memory allocation hooks.
Definition: associated_allocator.hpp:88
hook_allocator(Handler &h) noexcept
Definition: associated_allocator.hpp:106
T * allocate(std::size_t n)
Definition: associated_allocator.hpp:118
void deallocate(T *p, std::size_t n)
Definition: associated_allocator.hpp:124
T addressof(T... args)
hook_allocator(const hook_allocator< U, Handler > &a) noexcept
Definition: associated_allocator.hpp:112