write.hpp 26 KB

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