basic_io_object.hpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. //
  2. // basic_io_object.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. #ifndef BOOST_ASIO_BASIC_IO_OBJECT_HPP
  11. #define BOOST_ASIO_BASIC_IO_OBJECT_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include <boost/asio/detail/config.hpp>
  16. #include <boost/asio/io_service.hpp>
  17. #include <boost/asio/detail/push_options.hpp>
  18. namespace boost {
  19. namespace asio {
  20. #if defined(BOOST_ASIO_HAS_MOVE)
  21. namespace detail
  22. {
  23. // Type trait used to determine whether a service supports move.
  24. template <typename IoObjectService>
  25. class service_has_move
  26. {
  27. private:
  28. typedef IoObjectService service_type;
  29. typedef typename service_type::implementation_type implementation_type;
  30. template <typename T, typename U>
  31. static auto eval(T* t, U* u) -> decltype(t->move_construct(*u, *u), char());
  32. static char (&eval(...))[2];
  33. public:
  34. static const bool value =
  35. sizeof(service_has_move::eval(
  36. static_cast<service_type*>(0),
  37. static_cast<implementation_type*>(0))) == 1;
  38. };
  39. }
  40. #endif // defined(BOOST_ASIO_HAS_MOVE)
  41. /// Base class for all I/O objects.
  42. /**
  43. * @note All I/O objects are non-copyable. However, when using C++0x, certain
  44. * I/O objects do support move construction and move assignment.
  45. */
  46. #if !defined(BOOST_ASIO_HAS_MOVE) || defined(GENERATING_DOCUMENTATION)
  47. template <typename IoObjectService>
  48. #else
  49. template <typename IoObjectService,
  50. bool Movable = detail::service_has_move<IoObjectService>::value>
  51. #endif
  52. class basic_io_object
  53. {
  54. public:
  55. /// The type of the service that will be used to provide I/O operations.
  56. typedef IoObjectService service_type;
  57. /// The underlying implementation type of I/O object.
  58. typedef typename service_type::implementation_type implementation_type;
  59. /// Get the io_service associated with the object.
  60. /**
  61. * This function may be used to obtain the io_service object that the I/O
  62. * object uses to dispatch handlers for asynchronous operations.
  63. *
  64. * @return A reference to the io_service object that the I/O object will use
  65. * to dispatch handlers. Ownership is not transferred to the caller.
  66. */
  67. boost::asio::io_service& get_io_service()
  68. {
  69. return service.get_io_service();
  70. }
  71. protected:
  72. /// Construct a basic_io_object.
  73. /**
  74. * Performs:
  75. * @code get_service().construct(get_implementation()); @endcode
  76. */
  77. explicit basic_io_object(boost::asio::io_service& io_service)
  78. : service(boost::asio::use_service<IoObjectService>(io_service))
  79. {
  80. service.construct(implementation);
  81. }
  82. #if defined(GENERATING_DOCUMENTATION)
  83. /// Move-construct a basic_io_object.
  84. /**
  85. * Performs:
  86. * @code get_service().move_construct(
  87. * get_implementation(), other.get_implementation()); @endcode
  88. *
  89. * @note Available only for services that support movability,
  90. */
  91. basic_io_object(basic_io_object&& other);
  92. /// Move-assign a basic_io_object.
  93. /**
  94. * Performs:
  95. * @code get_service().move_assign(get_implementation(),
  96. * other.get_service(), other.get_implementation()); @endcode
  97. *
  98. * @note Available only for services that support movability,
  99. */
  100. basic_io_object& operator=(basic_io_object&& other);
  101. #endif // defined(GENERATING_DOCUMENTATION)
  102. /// Protected destructor to prevent deletion through this type.
  103. /**
  104. * Performs:
  105. * @code get_service().destroy(get_implementation()); @endcode
  106. */
  107. ~basic_io_object()
  108. {
  109. service.destroy(implementation);
  110. }
  111. /// Get the service associated with the I/O object.
  112. service_type& get_service()
  113. {
  114. return service;
  115. }
  116. /// Get the service associated with the I/O object.
  117. const service_type& get_service() const
  118. {
  119. return service;
  120. }
  121. /// (Deprecated: Use get_service().) The service associated with the I/O
  122. /// object.
  123. /**
  124. * @note Available only for services that do not support movability.
  125. */
  126. service_type& service;
  127. /// Get the underlying implementation of the I/O object.
  128. implementation_type& get_implementation()
  129. {
  130. return implementation;
  131. }
  132. /// Get the underlying implementation of the I/O object.
  133. const implementation_type& get_implementation() const
  134. {
  135. return implementation;
  136. }
  137. /// (Deprecated: Use get_implementation().) The underlying implementation of
  138. /// the I/O object.
  139. implementation_type implementation;
  140. private:
  141. basic_io_object(const basic_io_object&);
  142. basic_io_object& operator=(const basic_io_object&);
  143. };
  144. #if defined(BOOST_ASIO_HAS_MOVE)
  145. // Specialisation for movable objects.
  146. template <typename IoObjectService>
  147. class basic_io_object<IoObjectService, true>
  148. {
  149. public:
  150. typedef IoObjectService service_type;
  151. typedef typename service_type::implementation_type implementation_type;
  152. boost::asio::io_service& get_io_service()
  153. {
  154. return service_->get_io_service();
  155. }
  156. protected:
  157. explicit basic_io_object(boost::asio::io_service& io_service)
  158. : service_(&boost::asio::use_service<IoObjectService>(io_service))
  159. {
  160. service_->construct(implementation);
  161. }
  162. basic_io_object(basic_io_object&& other)
  163. : service_(&other.get_service())
  164. {
  165. service_->move_construct(implementation, other.implementation);
  166. }
  167. ~basic_io_object()
  168. {
  169. service_->destroy(implementation);
  170. }
  171. basic_io_object& operator=(basic_io_object&& other)
  172. {
  173. service_->move_assign(implementation,
  174. *other.service_, other.implementation);
  175. service_ = other.service_;
  176. return *this;
  177. }
  178. service_type& get_service()
  179. {
  180. return *service_;
  181. }
  182. const service_type& get_service() const
  183. {
  184. return *service_;
  185. }
  186. implementation_type& get_implementation()
  187. {
  188. return implementation;
  189. }
  190. const implementation_type& get_implementation() const
  191. {
  192. return implementation;
  193. }
  194. implementation_type implementation;
  195. private:
  196. basic_io_object(const basic_io_object&);
  197. void operator=(const basic_io_object&);
  198. IoObjectService* service_;
  199. };
  200. #endif // defined(BOOST_ASIO_HAS_MOVE)
  201. } // namespace asio
  202. } // namespace boost
  203. #include <boost/asio/detail/pop_options.hpp>
  204. #endif // BOOST_ASIO_BASIC_IO_OBJECT_HPP