connect.hpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430
  1. //
  2. // impl/connect.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_IMPL_CONNECT_HPP
  11. #define BOOST_ASIO_IMPL_CONNECT_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/bind_handler.hpp>
  16. #include <boost/asio/detail/consuming_buffers.hpp>
  17. #include <boost/asio/detail/handler_alloc_helpers.hpp>
  18. #include <boost/asio/detail/handler_cont_helpers.hpp>
  19. #include <boost/asio/detail/handler_invoke_helpers.hpp>
  20. #include <boost/asio/detail/handler_type_requirements.hpp>
  21. #include <boost/asio/detail/throw_error.hpp>
  22. #include <boost/asio/error.hpp>
  23. #include <boost/asio/detail/push_options.hpp>
  24. namespace boost {
  25. namespace asio {
  26. namespace detail
  27. {
  28. struct default_connect_condition
  29. {
  30. template <typename Iterator>
  31. Iterator operator()(const boost::system::error_code&, Iterator next)
  32. {
  33. return next;
  34. }
  35. };
  36. }
  37. template <typename Protocol, typename SocketService, typename Iterator>
  38. Iterator connect(basic_socket<Protocol, SocketService>& s, Iterator begin)
  39. {
  40. boost::system::error_code ec;
  41. Iterator result = connect(s, begin, ec);
  42. boost::asio::detail::throw_error(ec, "connect");
  43. return result;
  44. }
  45. template <typename Protocol, typename SocketService, typename Iterator>
  46. inline Iterator connect(basic_socket<Protocol, SocketService>& s,
  47. Iterator begin, boost::system::error_code& ec)
  48. {
  49. return connect(s, begin, Iterator(), detail::default_connect_condition(), ec);
  50. }
  51. template <typename Protocol, typename SocketService, typename Iterator>
  52. Iterator connect(basic_socket<Protocol, SocketService>& s,
  53. Iterator begin, Iterator end)
  54. {
  55. boost::system::error_code ec;
  56. Iterator result = connect(s, begin, end, ec);
  57. boost::asio::detail::throw_error(ec, "connect");
  58. return result;
  59. }
  60. template <typename Protocol, typename SocketService, typename Iterator>
  61. inline Iterator connect(basic_socket<Protocol, SocketService>& s,
  62. Iterator begin, Iterator end, boost::system::error_code& ec)
  63. {
  64. return connect(s, begin, end, detail::default_connect_condition(), ec);
  65. }
  66. template <typename Protocol, typename SocketService,
  67. typename Iterator, typename ConnectCondition>
  68. Iterator connect(basic_socket<Protocol, SocketService>& s,
  69. Iterator begin, ConnectCondition connect_condition)
  70. {
  71. boost::system::error_code ec;
  72. Iterator result = connect(s, begin, connect_condition, ec);
  73. boost::asio::detail::throw_error(ec, "connect");
  74. return result;
  75. }
  76. template <typename Protocol, typename SocketService,
  77. typename Iterator, typename ConnectCondition>
  78. inline Iterator connect(basic_socket<Protocol, SocketService>& s,
  79. Iterator begin, ConnectCondition connect_condition,
  80. boost::system::error_code& ec)
  81. {
  82. return connect(s, begin, Iterator(), connect_condition, ec);
  83. }
  84. template <typename Protocol, typename SocketService,
  85. typename Iterator, typename ConnectCondition>
  86. Iterator connect(basic_socket<Protocol, SocketService>& s,
  87. Iterator begin, Iterator end, ConnectCondition connect_condition)
  88. {
  89. boost::system::error_code ec;
  90. Iterator result = connect(s, begin, end, connect_condition, ec);
  91. boost::asio::detail::throw_error(ec, "connect");
  92. return result;
  93. }
  94. template <typename Protocol, typename SocketService,
  95. typename Iterator, typename ConnectCondition>
  96. Iterator connect(basic_socket<Protocol, SocketService>& s,
  97. Iterator begin, Iterator end, ConnectCondition connect_condition,
  98. boost::system::error_code& ec)
  99. {
  100. ec = boost::system::error_code();
  101. for (Iterator iter = begin; iter != end; ++iter)
  102. {
  103. iter = connect_condition(ec, iter);
  104. if (iter != end)
  105. {
  106. s.close(ec);
  107. s.connect(*iter, ec);
  108. if (!ec)
  109. return iter;
  110. }
  111. }
  112. if (!ec)
  113. ec = boost::asio::error::not_found;
  114. return end;
  115. }
  116. namespace detail
  117. {
  118. // Enable the empty base class optimisation for the connect condition.
  119. template <typename ConnectCondition>
  120. class base_from_connect_condition
  121. {
  122. protected:
  123. explicit base_from_connect_condition(
  124. const ConnectCondition& connect_condition)
  125. : connect_condition_(connect_condition)
  126. {
  127. }
  128. template <typename Iterator>
  129. void check_condition(const boost::system::error_code& ec,
  130. Iterator& iter, Iterator& end)
  131. {
  132. if (iter != end)
  133. iter = connect_condition_(ec, static_cast<const Iterator&>(iter));
  134. }
  135. private:
  136. ConnectCondition connect_condition_;
  137. };
  138. // The default_connect_condition implementation is essentially a no-op. This
  139. // template specialisation lets us eliminate all costs associated with it.
  140. template <>
  141. class base_from_connect_condition<default_connect_condition>
  142. {
  143. protected:
  144. explicit base_from_connect_condition(const default_connect_condition&)
  145. {
  146. }
  147. template <typename Iterator>
  148. void check_condition(const boost::system::error_code&, Iterator&, Iterator&)
  149. {
  150. }
  151. };
  152. template <typename Protocol, typename SocketService, typename Iterator,
  153. typename ConnectCondition, typename ComposedConnectHandler>
  154. class connect_op : base_from_connect_condition<ConnectCondition>
  155. {
  156. public:
  157. connect_op(basic_socket<Protocol, SocketService>& sock,
  158. const Iterator& begin, const Iterator& end,
  159. const ConnectCondition& connect_condition,
  160. ComposedConnectHandler& handler)
  161. : base_from_connect_condition<ConnectCondition>(connect_condition),
  162. socket_(sock),
  163. iter_(begin),
  164. end_(end),
  165. start_(0),
  166. handler_(BOOST_ASIO_MOVE_CAST(ComposedConnectHandler)(handler))
  167. {
  168. }
  169. #if defined(BOOST_ASIO_HAS_MOVE)
  170. connect_op(const connect_op& other)
  171. : base_from_connect_condition<ConnectCondition>(other),
  172. socket_(other.socket_),
  173. iter_(other.iter_),
  174. end_(other.end_),
  175. start_(other.start_),
  176. handler_(other.handler_)
  177. {
  178. }
  179. connect_op(connect_op&& other)
  180. : base_from_connect_condition<ConnectCondition>(other),
  181. socket_(other.socket_),
  182. iter_(other.iter_),
  183. end_(other.end_),
  184. start_(other.start_),
  185. handler_(BOOST_ASIO_MOVE_CAST(ComposedConnectHandler)(other.handler_))
  186. {
  187. }
  188. #endif // defined(BOOST_ASIO_HAS_MOVE)
  189. void operator()(boost::system::error_code ec, int start = 0)
  190. {
  191. switch (start_ = start)
  192. {
  193. case 1:
  194. for (;;)
  195. {
  196. this->check_condition(ec, iter_, end_);
  197. if (iter_ != end_)
  198. {
  199. socket_.close(ec);
  200. socket_.async_connect(*iter_,
  201. BOOST_ASIO_MOVE_CAST(connect_op)(*this));
  202. return;
  203. }
  204. if (start)
  205. {
  206. ec = boost::asio::error::not_found;
  207. socket_.get_io_service().post(detail::bind_handler(*this, ec));
  208. return;
  209. }
  210. default:
  211. if (iter_ == end_)
  212. break;
  213. if (!socket_.is_open())
  214. {
  215. ec = boost::asio::error::operation_aborted;
  216. break;
  217. }
  218. if (!ec)
  219. break;
  220. ++iter_;
  221. }
  222. handler_(static_cast<const boost::system::error_code&>(ec),
  223. static_cast<const Iterator&>(iter_));
  224. }
  225. }
  226. //private:
  227. basic_socket<Protocol, SocketService>& socket_;
  228. Iterator iter_;
  229. Iterator end_;
  230. int start_;
  231. ComposedConnectHandler handler_;
  232. };
  233. template <typename Protocol, typename SocketService, typename Iterator,
  234. typename ConnectCondition, typename ComposedConnectHandler>
  235. inline void* asio_handler_allocate(std::size_t size,
  236. connect_op<Protocol, SocketService, Iterator,
  237. ConnectCondition, ComposedConnectHandler>* this_handler)
  238. {
  239. return boost_asio_handler_alloc_helpers::allocate(
  240. size, this_handler->handler_);
  241. }
  242. template <typename Protocol, typename SocketService, typename Iterator,
  243. typename ConnectCondition, typename ComposedConnectHandler>
  244. inline void asio_handler_deallocate(void* pointer, std::size_t size,
  245. connect_op<Protocol, SocketService, Iterator,
  246. ConnectCondition, ComposedConnectHandler>* this_handler)
  247. {
  248. boost_asio_handler_alloc_helpers::deallocate(
  249. pointer, size, this_handler->handler_);
  250. }
  251. template <typename Protocol, typename SocketService, typename Iterator,
  252. typename ConnectCondition, typename ComposedConnectHandler>
  253. inline bool asio_handler_is_continuation(
  254. connect_op<Protocol, SocketService, Iterator,
  255. ConnectCondition, ComposedConnectHandler>* this_handler)
  256. {
  257. return boost_asio_handler_cont_helpers::is_continuation(
  258. this_handler->handler_);
  259. }
  260. template <typename Function, typename Protocol,
  261. typename SocketService, typename Iterator,
  262. typename ConnectCondition, typename ComposedConnectHandler>
  263. inline void asio_handler_invoke(Function& function,
  264. connect_op<Protocol, SocketService, Iterator,
  265. ConnectCondition, ComposedConnectHandler>* this_handler)
  266. {
  267. boost_asio_handler_invoke_helpers::invoke(
  268. function, this_handler->handler_);
  269. }
  270. template <typename Function, typename Protocol,
  271. typename SocketService, typename Iterator,
  272. typename ConnectCondition, typename ComposedConnectHandler>
  273. inline void asio_handler_invoke(const Function& function,
  274. connect_op<Protocol, SocketService, Iterator,
  275. ConnectCondition, ComposedConnectHandler>* this_handler)
  276. {
  277. boost_asio_handler_invoke_helpers::invoke(
  278. function, this_handler->handler_);
  279. }
  280. } // namespace detail
  281. template <typename Protocol, typename SocketService,
  282. typename Iterator, typename ComposedConnectHandler>
  283. inline BOOST_ASIO_INITFN_RESULT_TYPE(ComposedConnectHandler,
  284. void (boost::system::error_code, Iterator))
  285. async_connect(basic_socket<Protocol, SocketService>& s,
  286. Iterator begin, BOOST_ASIO_MOVE_ARG(ComposedConnectHandler) handler)
  287. {
  288. // If you get an error on the following line it means that your handler does
  289. // not meet the documented type requirements for a ComposedConnectHandler.
  290. BOOST_ASIO_COMPOSED_CONNECT_HANDLER_CHECK(
  291. ComposedConnectHandler, handler, Iterator) type_check;
  292. detail::async_result_init<ComposedConnectHandler,
  293. void (boost::system::error_code, Iterator)> init(
  294. BOOST_ASIO_MOVE_CAST(ComposedConnectHandler)(handler));
  295. detail::connect_op<Protocol, SocketService, Iterator,
  296. detail::default_connect_condition, BOOST_ASIO_HANDLER_TYPE(
  297. ComposedConnectHandler, void (boost::system::error_code, Iterator))>(s,
  298. begin, Iterator(), detail::default_connect_condition(), init.handler)(
  299. boost::system::error_code(), 1);
  300. return init.result.get();
  301. }
  302. template <typename Protocol, typename SocketService,
  303. typename Iterator, typename ComposedConnectHandler>
  304. inline BOOST_ASIO_INITFN_RESULT_TYPE(ComposedConnectHandler,
  305. void (boost::system::error_code, Iterator))
  306. async_connect(basic_socket<Protocol, SocketService>& s,
  307. Iterator begin, Iterator end,
  308. BOOST_ASIO_MOVE_ARG(ComposedConnectHandler) handler)
  309. {
  310. // If you get an error on the following line it means that your handler does
  311. // not meet the documented type requirements for a ComposedConnectHandler.
  312. BOOST_ASIO_COMPOSED_CONNECT_HANDLER_CHECK(
  313. ComposedConnectHandler, handler, Iterator) type_check;
  314. detail::async_result_init<ComposedConnectHandler,
  315. void (boost::system::error_code, Iterator)> init(
  316. BOOST_ASIO_MOVE_CAST(ComposedConnectHandler)(handler));
  317. detail::connect_op<Protocol, SocketService, Iterator,
  318. detail::default_connect_condition, BOOST_ASIO_HANDLER_TYPE(
  319. ComposedConnectHandler, void (boost::system::error_code, Iterator))>(s,
  320. begin, end, detail::default_connect_condition(), init.handler)(
  321. boost::system::error_code(), 1);
  322. return init.result.get();
  323. }
  324. template <typename Protocol, typename SocketService, typename Iterator,
  325. typename ConnectCondition, typename ComposedConnectHandler>
  326. inline BOOST_ASIO_INITFN_RESULT_TYPE(ComposedConnectHandler,
  327. void (boost::system::error_code, Iterator))
  328. async_connect(basic_socket<Protocol, SocketService>& s,
  329. Iterator begin, ConnectCondition connect_condition,
  330. BOOST_ASIO_MOVE_ARG(ComposedConnectHandler) handler)
  331. {
  332. // If you get an error on the following line it means that your handler does
  333. // not meet the documented type requirements for a ComposedConnectHandler.
  334. BOOST_ASIO_COMPOSED_CONNECT_HANDLER_CHECK(
  335. ComposedConnectHandler, handler, Iterator) type_check;
  336. detail::async_result_init<ComposedConnectHandler,
  337. void (boost::system::error_code, Iterator)> init(
  338. BOOST_ASIO_MOVE_CAST(ComposedConnectHandler)(handler));
  339. detail::connect_op<Protocol, SocketService, Iterator,
  340. ConnectCondition, BOOST_ASIO_HANDLER_TYPE(
  341. ComposedConnectHandler, void (boost::system::error_code, Iterator))>(s,
  342. begin, Iterator(), connect_condition, init.handler)(
  343. boost::system::error_code(), 1);
  344. return init.result.get();
  345. }
  346. template <typename Protocol, typename SocketService, typename Iterator,
  347. typename ConnectCondition, typename ComposedConnectHandler>
  348. inline BOOST_ASIO_INITFN_RESULT_TYPE(ComposedConnectHandler,
  349. void (boost::system::error_code, Iterator))
  350. async_connect(basic_socket<Protocol, SocketService>& s,
  351. Iterator begin, Iterator end, ConnectCondition connect_condition,
  352. BOOST_ASIO_MOVE_ARG(ComposedConnectHandler) handler)
  353. {
  354. // If you get an error on the following line it means that your handler does
  355. // not meet the documented type requirements for a ComposedConnectHandler.
  356. BOOST_ASIO_COMPOSED_CONNECT_HANDLER_CHECK(
  357. ComposedConnectHandler, handler, Iterator) type_check;
  358. detail::async_result_init<ComposedConnectHandler,
  359. void (boost::system::error_code, Iterator)> init(
  360. BOOST_ASIO_MOVE_CAST(ComposedConnectHandler)(handler));
  361. detail::connect_op<Protocol, SocketService, Iterator,
  362. ConnectCondition, BOOST_ASIO_HANDLER_TYPE(
  363. ComposedConnectHandler, void (boost::system::error_code, Iterator))>(s,
  364. begin, end, connect_condition, init.handler)(
  365. boost::system::error_code(), 1);
  366. return init.result.get();
  367. }
  368. } // namespace asio
  369. } // namespace boost
  370. #include <boost/asio/detail/pop_options.hpp>
  371. #endif // BOOST_ASIO_IMPL_CONNECT_HPP