read.hpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755
  1. //
  2. // impl/read.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_READ_HPP
  11. #define BOOST_ASIO_IMPL_READ_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include <algorithm>
  16. #include <boost/asio/buffer.hpp>
  17. #include <boost/asio/completion_condition.hpp>
  18. #include <boost/asio/detail/array_fwd.hpp>
  19. #include <boost/asio/detail/base_from_completion_cond.hpp>
  20. #include <boost/asio/detail/bind_handler.hpp>
  21. #include <boost/asio/detail/consuming_buffers.hpp>
  22. #include <boost/asio/detail/dependent_type.hpp>
  23. #include <boost/asio/detail/handler_alloc_helpers.hpp>
  24. #include <boost/asio/detail/handler_cont_helpers.hpp>
  25. #include <boost/asio/detail/handler_invoke_helpers.hpp>
  26. #include <boost/asio/detail/handler_type_requirements.hpp>
  27. #include <boost/asio/detail/throw_error.hpp>
  28. #include <boost/asio/error.hpp>
  29. #include <boost/asio/detail/push_options.hpp>
  30. namespace boost {
  31. namespace asio {
  32. template <typename SyncReadStream, typename MutableBufferSequence,
  33. typename CompletionCondition>
  34. std::size_t read(SyncReadStream& s, const MutableBufferSequence& buffers,
  35. CompletionCondition completion_condition, boost::system::error_code& ec)
  36. {
  37. ec = boost::system::error_code();
  38. boost::asio::detail::consuming_buffers<
  39. mutable_buffer, MutableBufferSequence> tmp(buffers);
  40. std::size_t total_transferred = 0;
  41. tmp.prepare(detail::adapt_completion_condition_result(
  42. completion_condition(ec, total_transferred)));
  43. while (tmp.begin() != tmp.end())
  44. {
  45. std::size_t bytes_transferred = s.read_some(tmp, ec);
  46. tmp.consume(bytes_transferred);
  47. total_transferred += bytes_transferred;
  48. tmp.prepare(detail::adapt_completion_condition_result(
  49. completion_condition(ec, total_transferred)));
  50. }
  51. return total_transferred;
  52. }
  53. template <typename SyncReadStream, typename MutableBufferSequence>
  54. inline std::size_t read(SyncReadStream& s, const MutableBufferSequence& buffers)
  55. {
  56. boost::system::error_code ec;
  57. std::size_t bytes_transferred = read(s, buffers, transfer_all(), ec);
  58. boost::asio::detail::throw_error(ec, "read");
  59. return bytes_transferred;
  60. }
  61. template <typename SyncReadStream, typename MutableBufferSequence>
  62. inline std::size_t read(SyncReadStream& s, const MutableBufferSequence& buffers,
  63. boost::system::error_code& ec)
  64. {
  65. return read(s, buffers, transfer_all(), ec);
  66. }
  67. template <typename SyncReadStream, typename MutableBufferSequence,
  68. typename CompletionCondition>
  69. inline std::size_t read(SyncReadStream& s, const MutableBufferSequence& buffers,
  70. CompletionCondition completion_condition)
  71. {
  72. boost::system::error_code ec;
  73. std::size_t bytes_transferred = read(s, buffers, completion_condition, ec);
  74. boost::asio::detail::throw_error(ec, "read");
  75. return bytes_transferred;
  76. }
  77. #if !defined(BOOST_ASIO_NO_IOSTREAM)
  78. template <typename SyncReadStream, typename Allocator,
  79. typename CompletionCondition>
  80. std::size_t read(SyncReadStream& s,
  81. boost::asio::basic_streambuf<Allocator>& b,
  82. CompletionCondition completion_condition, boost::system::error_code& ec)
  83. {
  84. ec = boost::system::error_code();
  85. std::size_t total_transferred = 0;
  86. std::size_t max_size = detail::adapt_completion_condition_result(
  87. completion_condition(ec, total_transferred));
  88. std::size_t bytes_available = read_size_helper(b, max_size);
  89. while (bytes_available > 0)
  90. {
  91. std::size_t bytes_transferred = s.read_some(b.prepare(bytes_available), ec);
  92. b.commit(bytes_transferred);
  93. total_transferred += bytes_transferred;
  94. max_size = detail::adapt_completion_condition_result(
  95. completion_condition(ec, total_transferred));
  96. bytes_available = read_size_helper(b, max_size);
  97. }
  98. return total_transferred;
  99. }
  100. template <typename SyncReadStream, typename Allocator>
  101. inline std::size_t read(SyncReadStream& s,
  102. boost::asio::basic_streambuf<Allocator>& b)
  103. {
  104. boost::system::error_code ec;
  105. std::size_t bytes_transferred = read(s, b, transfer_all(), ec);
  106. boost::asio::detail::throw_error(ec, "read");
  107. return bytes_transferred;
  108. }
  109. template <typename SyncReadStream, typename Allocator>
  110. inline std::size_t read(SyncReadStream& s,
  111. boost::asio::basic_streambuf<Allocator>& b,
  112. boost::system::error_code& ec)
  113. {
  114. return read(s, b, transfer_all(), ec);
  115. }
  116. template <typename SyncReadStream, typename Allocator,
  117. typename CompletionCondition>
  118. inline std::size_t read(SyncReadStream& s,
  119. boost::asio::basic_streambuf<Allocator>& b,
  120. CompletionCondition completion_condition)
  121. {
  122. boost::system::error_code ec;
  123. std::size_t bytes_transferred = read(s, b, completion_condition, ec);
  124. boost::asio::detail::throw_error(ec, "read");
  125. return bytes_transferred;
  126. }
  127. #endif // !defined(BOOST_ASIO_NO_IOSTREAM)
  128. namespace detail
  129. {
  130. template <typename AsyncReadStream, typename MutableBufferSequence,
  131. typename CompletionCondition, typename ReadHandler>
  132. class read_op
  133. : detail::base_from_completion_cond<CompletionCondition>
  134. {
  135. public:
  136. read_op(AsyncReadStream& stream, const MutableBufferSequence& buffers,
  137. CompletionCondition completion_condition, ReadHandler& handler)
  138. : detail::base_from_completion_cond<
  139. CompletionCondition>(completion_condition),
  140. stream_(stream),
  141. buffers_(buffers),
  142. start_(0),
  143. total_transferred_(0),
  144. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  145. {
  146. }
  147. #if defined(BOOST_ASIO_HAS_MOVE)
  148. read_op(const read_op& other)
  149. : detail::base_from_completion_cond<CompletionCondition>(other),
  150. stream_(other.stream_),
  151. buffers_(other.buffers_),
  152. start_(other.start_),
  153. total_transferred_(other.total_transferred_),
  154. handler_(other.handler_)
  155. {
  156. }
  157. read_op(read_op&& other)
  158. : detail::base_from_completion_cond<CompletionCondition>(other),
  159. stream_(other.stream_),
  160. buffers_(other.buffers_),
  161. start_(other.start_),
  162. total_transferred_(other.total_transferred_),
  163. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  164. {
  165. }
  166. #endif // defined(BOOST_ASIO_HAS_MOVE)
  167. void operator()(const boost::system::error_code& ec,
  168. std::size_t bytes_transferred, int start = 0)
  169. {
  170. switch (start_ = start)
  171. {
  172. case 1:
  173. buffers_.prepare(this->check_for_completion(ec, total_transferred_));
  174. for (;;)
  175. {
  176. stream_.async_read_some(buffers_,
  177. BOOST_ASIO_MOVE_CAST(read_op)(*this));
  178. return; default:
  179. total_transferred_ += bytes_transferred;
  180. buffers_.consume(bytes_transferred);
  181. buffers_.prepare(this->check_for_completion(ec, total_transferred_));
  182. if ((!ec && bytes_transferred == 0)
  183. || buffers_.begin() == buffers_.end())
  184. break;
  185. }
  186. handler_(ec, static_cast<const std::size_t&>(total_transferred_));
  187. }
  188. }
  189. //private:
  190. AsyncReadStream& stream_;
  191. boost::asio::detail::consuming_buffers<
  192. mutable_buffer, MutableBufferSequence> buffers_;
  193. int start_;
  194. std::size_t total_transferred_;
  195. ReadHandler handler_;
  196. };
  197. template <typename AsyncReadStream,
  198. typename CompletionCondition, typename ReadHandler>
  199. class read_op<AsyncReadStream, boost::asio::mutable_buffers_1,
  200. CompletionCondition, ReadHandler>
  201. : detail::base_from_completion_cond<CompletionCondition>
  202. {
  203. public:
  204. read_op(AsyncReadStream& stream,
  205. const boost::asio::mutable_buffers_1& buffers,
  206. CompletionCondition completion_condition, ReadHandler& handler)
  207. : detail::base_from_completion_cond<
  208. CompletionCondition>(completion_condition),
  209. stream_(stream),
  210. buffer_(buffers),
  211. start_(0),
  212. total_transferred_(0),
  213. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  214. {
  215. }
  216. #if defined(BOOST_ASIO_HAS_MOVE)
  217. read_op(const read_op& other)
  218. : detail::base_from_completion_cond<CompletionCondition>(other),
  219. stream_(other.stream_),
  220. buffer_(other.buffer_),
  221. start_(other.start_),
  222. total_transferred_(other.total_transferred_),
  223. handler_(other.handler_)
  224. {
  225. }
  226. read_op(read_op&& other)
  227. : detail::base_from_completion_cond<CompletionCondition>(other),
  228. stream_(other.stream_),
  229. buffer_(other.buffer_),
  230. start_(other.start_),
  231. total_transferred_(other.total_transferred_),
  232. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  233. {
  234. }
  235. #endif // defined(BOOST_ASIO_HAS_MOVE)
  236. void operator()(const boost::system::error_code& ec,
  237. std::size_t bytes_transferred, int start = 0)
  238. {
  239. std::size_t n = 0;
  240. switch (start_ = start)
  241. {
  242. case 1:
  243. n = this->check_for_completion(ec, total_transferred_);
  244. for (;;)
  245. {
  246. stream_.async_read_some(
  247. boost::asio::buffer(buffer_ + total_transferred_, n),
  248. BOOST_ASIO_MOVE_CAST(read_op)(*this));
  249. return; default:
  250. total_transferred_ += bytes_transferred;
  251. if ((!ec && bytes_transferred == 0)
  252. || (n = this->check_for_completion(ec, total_transferred_)) == 0
  253. || total_transferred_ == boost::asio::buffer_size(buffer_))
  254. break;
  255. }
  256. handler_(ec, static_cast<const std::size_t&>(total_transferred_));
  257. }
  258. }
  259. //private:
  260. AsyncReadStream& stream_;
  261. boost::asio::mutable_buffer buffer_;
  262. int start_;
  263. std::size_t total_transferred_;
  264. ReadHandler handler_;
  265. };
  266. template <typename AsyncReadStream, typename Elem,
  267. typename CompletionCondition, typename ReadHandler>
  268. class read_op<AsyncReadStream, boost::array<Elem, 2>,
  269. CompletionCondition, ReadHandler>
  270. : detail::base_from_completion_cond<CompletionCondition>
  271. {
  272. public:
  273. read_op(AsyncReadStream& stream, const boost::array<Elem, 2>& buffers,
  274. CompletionCondition completion_condition, ReadHandler& handler)
  275. : detail::base_from_completion_cond<
  276. CompletionCondition>(completion_condition),
  277. stream_(stream),
  278. buffers_(buffers),
  279. start_(0),
  280. total_transferred_(0),
  281. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  282. {
  283. }
  284. #if defined(BOOST_ASIO_HAS_MOVE)
  285. read_op(const read_op& other)
  286. : detail::base_from_completion_cond<CompletionCondition>(other),
  287. stream_(other.stream_),
  288. buffers_(other.buffers_),
  289. start_(other.start_),
  290. total_transferred_(other.total_transferred_),
  291. handler_(other.handler_)
  292. {
  293. }
  294. read_op(read_op&& other)
  295. : detail::base_from_completion_cond<CompletionCondition>(other),
  296. stream_(other.stream_),
  297. buffers_(other.buffers_),
  298. start_(other.start_),
  299. total_transferred_(other.total_transferred_),
  300. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  301. {
  302. }
  303. #endif // defined(BOOST_ASIO_HAS_MOVE)
  304. void operator()(const boost::system::error_code& ec,
  305. std::size_t bytes_transferred, int start = 0)
  306. {
  307. typename boost::asio::detail::dependent_type<Elem,
  308. boost::array<boost::asio::mutable_buffer, 2> >::type bufs = {{
  309. boost::asio::mutable_buffer(buffers_[0]),
  310. boost::asio::mutable_buffer(buffers_[1]) }};
  311. std::size_t buffer_size0 = boost::asio::buffer_size(bufs[0]);
  312. std::size_t buffer_size1 = boost::asio::buffer_size(bufs[1]);
  313. std::size_t n = 0;
  314. switch (start_ = start)
  315. {
  316. case 1:
  317. n = this->check_for_completion(ec, total_transferred_);
  318. for (;;)
  319. {
  320. bufs[0] = boost::asio::buffer(bufs[0] + total_transferred_, n);
  321. bufs[1] = boost::asio::buffer(
  322. bufs[1] + (total_transferred_ < buffer_size0
  323. ? 0 : total_transferred_ - buffer_size0),
  324. n - boost::asio::buffer_size(bufs[0]));
  325. stream_.async_read_some(bufs, BOOST_ASIO_MOVE_CAST(read_op)(*this));
  326. return; default:
  327. total_transferred_ += bytes_transferred;
  328. if ((!ec && bytes_transferred == 0)
  329. || (n = this->check_for_completion(ec, total_transferred_)) == 0
  330. || total_transferred_ == buffer_size0 + buffer_size1)
  331. break;
  332. }
  333. handler_(ec, static_cast<const std::size_t&>(total_transferred_));
  334. }
  335. }
  336. //private:
  337. AsyncReadStream& stream_;
  338. boost::array<Elem, 2> buffers_;
  339. int start_;
  340. std::size_t total_transferred_;
  341. ReadHandler handler_;
  342. };
  343. #if defined(BOOST_ASIO_HAS_STD_ARRAY)
  344. template <typename AsyncReadStream, typename Elem,
  345. typename CompletionCondition, typename ReadHandler>
  346. class read_op<AsyncReadStream, std::array<Elem, 2>,
  347. CompletionCondition, ReadHandler>
  348. : detail::base_from_completion_cond<CompletionCondition>
  349. {
  350. public:
  351. read_op(AsyncReadStream& stream, const std::array<Elem, 2>& buffers,
  352. CompletionCondition completion_condition, ReadHandler& handler)
  353. : detail::base_from_completion_cond<
  354. CompletionCondition>(completion_condition),
  355. stream_(stream),
  356. buffers_(buffers),
  357. start_(0),
  358. total_transferred_(0),
  359. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  360. {
  361. }
  362. #if defined(BOOST_ASIO_HAS_MOVE)
  363. read_op(const read_op& other)
  364. : detail::base_from_completion_cond<CompletionCondition>(other),
  365. stream_(other.stream_),
  366. buffers_(other.buffers_),
  367. start_(other.start_),
  368. total_transferred_(other.total_transferred_),
  369. handler_(other.handler_)
  370. {
  371. }
  372. read_op(read_op&& other)
  373. : detail::base_from_completion_cond<CompletionCondition>(other),
  374. stream_(other.stream_),
  375. buffers_(other.buffers_),
  376. start_(other.start_),
  377. total_transferred_(other.total_transferred_),
  378. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  379. {
  380. }
  381. #endif // defined(BOOST_ASIO_HAS_MOVE)
  382. void operator()(const boost::system::error_code& ec,
  383. std::size_t bytes_transferred, int start = 0)
  384. {
  385. typename boost::asio::detail::dependent_type<Elem,
  386. std::array<boost::asio::mutable_buffer, 2> >::type bufs = {{
  387. boost::asio::mutable_buffer(buffers_[0]),
  388. boost::asio::mutable_buffer(buffers_[1]) }};
  389. std::size_t buffer_size0 = boost::asio::buffer_size(bufs[0]);
  390. std::size_t buffer_size1 = boost::asio::buffer_size(bufs[1]);
  391. std::size_t n = 0;
  392. switch (start_ = start)
  393. {
  394. case 1:
  395. n = this->check_for_completion(ec, total_transferred_);
  396. for (;;)
  397. {
  398. bufs[0] = boost::asio::buffer(bufs[0] + total_transferred_, n);
  399. bufs[1] = boost::asio::buffer(
  400. bufs[1] + (total_transferred_ < buffer_size0
  401. ? 0 : total_transferred_ - buffer_size0),
  402. n - boost::asio::buffer_size(bufs[0]));
  403. stream_.async_read_some(bufs, BOOST_ASIO_MOVE_CAST(read_op)(*this));
  404. return; default:
  405. total_transferred_ += bytes_transferred;
  406. if ((!ec && bytes_transferred == 0)
  407. || (n = this->check_for_completion(ec, total_transferred_)) == 0
  408. || total_transferred_ == buffer_size0 + buffer_size1)
  409. break;
  410. }
  411. handler_(ec, static_cast<const std::size_t&>(total_transferred_));
  412. }
  413. }
  414. //private:
  415. AsyncReadStream& stream_;
  416. std::array<Elem, 2> buffers_;
  417. int start_;
  418. std::size_t total_transferred_;
  419. ReadHandler handler_;
  420. };
  421. #endif // defined(BOOST_ASIO_HAS_STD_ARRAY)
  422. template <typename AsyncReadStream, typename MutableBufferSequence,
  423. typename CompletionCondition, typename ReadHandler>
  424. inline void* asio_handler_allocate(std::size_t size,
  425. read_op<AsyncReadStream, MutableBufferSequence,
  426. CompletionCondition, ReadHandler>* this_handler)
  427. {
  428. return boost_asio_handler_alloc_helpers::allocate(
  429. size, this_handler->handler_);
  430. }
  431. template <typename AsyncReadStream, typename MutableBufferSequence,
  432. typename CompletionCondition, typename ReadHandler>
  433. inline void asio_handler_deallocate(void* pointer, std::size_t size,
  434. read_op<AsyncReadStream, MutableBufferSequence,
  435. CompletionCondition, ReadHandler>* this_handler)
  436. {
  437. boost_asio_handler_alloc_helpers::deallocate(
  438. pointer, size, this_handler->handler_);
  439. }
  440. template <typename AsyncReadStream, typename MutableBufferSequence,
  441. typename CompletionCondition, typename ReadHandler>
  442. inline bool asio_handler_is_continuation(
  443. read_op<AsyncReadStream, MutableBufferSequence,
  444. CompletionCondition, ReadHandler>* this_handler)
  445. {
  446. return this_handler->start_ == 0 ? true
  447. : boost_asio_handler_cont_helpers::is_continuation(
  448. this_handler->handler_);
  449. }
  450. template <typename Function, typename AsyncReadStream,
  451. typename MutableBufferSequence, typename CompletionCondition,
  452. typename ReadHandler>
  453. inline void asio_handler_invoke(Function& function,
  454. read_op<AsyncReadStream, MutableBufferSequence,
  455. CompletionCondition, ReadHandler>* this_handler)
  456. {
  457. boost_asio_handler_invoke_helpers::invoke(
  458. function, this_handler->handler_);
  459. }
  460. template <typename Function, typename AsyncReadStream,
  461. typename MutableBufferSequence, typename CompletionCondition,
  462. typename ReadHandler>
  463. inline void asio_handler_invoke(const Function& function,
  464. read_op<AsyncReadStream, MutableBufferSequence,
  465. CompletionCondition, ReadHandler>* this_handler)
  466. {
  467. boost_asio_handler_invoke_helpers::invoke(
  468. function, this_handler->handler_);
  469. }
  470. } // namespace detail
  471. template <typename AsyncReadStream, typename MutableBufferSequence,
  472. typename CompletionCondition, typename ReadHandler>
  473. inline BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  474. void (boost::system::error_code, std::size_t))
  475. async_read(AsyncReadStream& s, const MutableBufferSequence& buffers,
  476. CompletionCondition completion_condition,
  477. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  478. {
  479. // If you get an error on the following line it means that your handler does
  480. // not meet the documented type requirements for a ReadHandler.
  481. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  482. detail::async_result_init<
  483. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  484. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  485. detail::read_op<AsyncReadStream, MutableBufferSequence,
  486. CompletionCondition, BOOST_ASIO_HANDLER_TYPE(
  487. ReadHandler, void (boost::system::error_code, std::size_t))>(
  488. s, buffers, completion_condition, init.handler)(
  489. boost::system::error_code(), 0, 1);
  490. return init.result.get();
  491. }
  492. template <typename AsyncReadStream, typename MutableBufferSequence,
  493. typename ReadHandler>
  494. inline BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  495. void (boost::system::error_code, std::size_t))
  496. async_read(AsyncReadStream& s, const MutableBufferSequence& buffers,
  497. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  498. {
  499. // If you get an error on the following line it means that your handler does
  500. // not meet the documented type requirements for a ReadHandler.
  501. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  502. detail::async_result_init<
  503. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  504. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  505. detail::read_op<AsyncReadStream, MutableBufferSequence,
  506. detail::transfer_all_t, BOOST_ASIO_HANDLER_TYPE(
  507. ReadHandler, void (boost::system::error_code, std::size_t))>(
  508. s, buffers, transfer_all(), init.handler)(
  509. boost::system::error_code(), 0, 1);
  510. return init.result.get();
  511. }
  512. #if !defined(BOOST_ASIO_NO_IOSTREAM)
  513. namespace detail
  514. {
  515. template <typename AsyncReadStream, typename Allocator,
  516. typename CompletionCondition, typename ReadHandler>
  517. class read_streambuf_op
  518. : detail::base_from_completion_cond<CompletionCondition>
  519. {
  520. public:
  521. read_streambuf_op(AsyncReadStream& stream,
  522. basic_streambuf<Allocator>& streambuf,
  523. CompletionCondition completion_condition, ReadHandler& handler)
  524. : detail::base_from_completion_cond<
  525. CompletionCondition>(completion_condition),
  526. stream_(stream),
  527. streambuf_(streambuf),
  528. start_(0),
  529. total_transferred_(0),
  530. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  531. {
  532. }
  533. #if defined(BOOST_ASIO_HAS_MOVE)
  534. read_streambuf_op(const read_streambuf_op& other)
  535. : detail::base_from_completion_cond<CompletionCondition>(other),
  536. stream_(other.stream_),
  537. streambuf_(other.streambuf_),
  538. start_(other.start_),
  539. total_transferred_(other.total_transferred_),
  540. handler_(other.handler_)
  541. {
  542. }
  543. read_streambuf_op(read_streambuf_op&& other)
  544. : detail::base_from_completion_cond<CompletionCondition>(other),
  545. stream_(other.stream_),
  546. streambuf_(other.streambuf_),
  547. start_(other.start_),
  548. total_transferred_(other.total_transferred_),
  549. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  550. {
  551. }
  552. #endif // defined(BOOST_ASIO_HAS_MOVE)
  553. void operator()(const boost::system::error_code& ec,
  554. std::size_t bytes_transferred, int start = 0)
  555. {
  556. std::size_t max_size, bytes_available;
  557. switch (start_ = start)
  558. {
  559. case 1:
  560. max_size = this->check_for_completion(ec, total_transferred_);
  561. bytes_available = read_size_helper(streambuf_, max_size);
  562. for (;;)
  563. {
  564. stream_.async_read_some(streambuf_.prepare(bytes_available),
  565. BOOST_ASIO_MOVE_CAST(read_streambuf_op)(*this));
  566. return; default:
  567. total_transferred_ += bytes_transferred;
  568. streambuf_.commit(bytes_transferred);
  569. max_size = this->check_for_completion(ec, total_transferred_);
  570. bytes_available = read_size_helper(streambuf_, max_size);
  571. if ((!ec && bytes_transferred == 0) || bytes_available == 0)
  572. break;
  573. }
  574. handler_(ec, static_cast<const std::size_t&>(total_transferred_));
  575. }
  576. }
  577. //private:
  578. AsyncReadStream& stream_;
  579. boost::asio::basic_streambuf<Allocator>& streambuf_;
  580. int start_;
  581. std::size_t total_transferred_;
  582. ReadHandler handler_;
  583. };
  584. template <typename AsyncReadStream, typename Allocator,
  585. typename CompletionCondition, typename ReadHandler>
  586. inline void* asio_handler_allocate(std::size_t size,
  587. read_streambuf_op<AsyncReadStream, Allocator,
  588. CompletionCondition, ReadHandler>* this_handler)
  589. {
  590. return boost_asio_handler_alloc_helpers::allocate(
  591. size, this_handler->handler_);
  592. }
  593. template <typename AsyncReadStream, typename Allocator,
  594. typename CompletionCondition, typename ReadHandler>
  595. inline void asio_handler_deallocate(void* pointer, std::size_t size,
  596. read_streambuf_op<AsyncReadStream, Allocator,
  597. CompletionCondition, ReadHandler>* this_handler)
  598. {
  599. boost_asio_handler_alloc_helpers::deallocate(
  600. pointer, size, this_handler->handler_);
  601. }
  602. template <typename AsyncReadStream, typename Allocator,
  603. typename CompletionCondition, typename ReadHandler>
  604. inline bool asio_handler_is_continuation(
  605. read_streambuf_op<AsyncReadStream, Allocator,
  606. CompletionCondition, ReadHandler>* this_handler)
  607. {
  608. return this_handler->start_ == 0 ? true
  609. : boost_asio_handler_cont_helpers::is_continuation(
  610. this_handler->handler_);
  611. }
  612. template <typename Function, typename AsyncReadStream,
  613. typename Allocator, typename CompletionCondition, typename ReadHandler>
  614. inline void asio_handler_invoke(Function& function,
  615. read_streambuf_op<AsyncReadStream, Allocator,
  616. CompletionCondition, ReadHandler>* this_handler)
  617. {
  618. boost_asio_handler_invoke_helpers::invoke(
  619. function, this_handler->handler_);
  620. }
  621. template <typename Function, typename AsyncReadStream,
  622. typename Allocator, typename CompletionCondition, typename ReadHandler>
  623. inline void asio_handler_invoke(const Function& function,
  624. read_streambuf_op<AsyncReadStream, Allocator,
  625. CompletionCondition, ReadHandler>* this_handler)
  626. {
  627. boost_asio_handler_invoke_helpers::invoke(
  628. function, this_handler->handler_);
  629. }
  630. } // namespace detail
  631. template <typename AsyncReadStream, typename Allocator,
  632. typename CompletionCondition, typename ReadHandler>
  633. inline BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  634. void (boost::system::error_code, std::size_t))
  635. async_read(AsyncReadStream& s,
  636. boost::asio::basic_streambuf<Allocator>& b,
  637. CompletionCondition completion_condition,
  638. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  639. {
  640. // If you get an error on the following line it means that your handler does
  641. // not meet the documented type requirements for a ReadHandler.
  642. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  643. detail::async_result_init<
  644. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  645. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  646. detail::read_streambuf_op<AsyncReadStream, Allocator,
  647. CompletionCondition, BOOST_ASIO_HANDLER_TYPE(
  648. ReadHandler, void (boost::system::error_code, std::size_t))>(
  649. s, b, completion_condition, init.handler)(
  650. boost::system::error_code(), 0, 1);
  651. return init.result.get();
  652. }
  653. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  654. inline BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  655. void (boost::system::error_code, std::size_t))
  656. async_read(AsyncReadStream& s,
  657. boost::asio::basic_streambuf<Allocator>& b,
  658. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  659. {
  660. // If you get an error on the following line it means that your handler does
  661. // not meet the documented type requirements for a ReadHandler.
  662. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  663. detail::async_result_init<
  664. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  665. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  666. detail::read_streambuf_op<AsyncReadStream, Allocator,
  667. detail::transfer_all_t, BOOST_ASIO_HANDLER_TYPE(
  668. ReadHandler, void (boost::system::error_code, std::size_t))>(
  669. s, b, transfer_all(), init.handler)(
  670. boost::system::error_code(), 0, 1);
  671. return init.result.get();
  672. }
  673. #endif // !defined(BOOST_ASIO_NO_IOSTREAM)
  674. } // namespace asio
  675. } // namespace boost
  676. #include <boost/asio/detail/pop_options.hpp>
  677. #endif // BOOST_ASIO_IMPL_READ_HPP