stream.hpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758
  1. //
  2. // ssl/stream.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_SSL_STREAM_HPP
  11. #define BOOST_ASIO_SSL_STREAM_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. #if defined(BOOST_ASIO_ENABLE_OLD_SSL)
  17. # include <boost/asio/ssl/old/stream.hpp>
  18. #else // defined(BOOST_ASIO_ENABLE_OLD_SSL)
  19. # include <boost/asio/async_result.hpp>
  20. # include <boost/asio/detail/buffer_sequence_adapter.hpp>
  21. # include <boost/asio/detail/handler_type_requirements.hpp>
  22. # include <boost/asio/detail/noncopyable.hpp>
  23. # include <boost/asio/detail/type_traits.hpp>
  24. # include <boost/asio/ssl/context.hpp>
  25. # include <boost/asio/ssl/detail/buffered_handshake_op.hpp>
  26. # include <boost/asio/ssl/detail/handshake_op.hpp>
  27. # include <boost/asio/ssl/detail/io.hpp>
  28. # include <boost/asio/ssl/detail/read_op.hpp>
  29. # include <boost/asio/ssl/detail/shutdown_op.hpp>
  30. # include <boost/asio/ssl/detail/stream_core.hpp>
  31. # include <boost/asio/ssl/detail/write_op.hpp>
  32. # include <boost/asio/ssl/stream_base.hpp>
  33. #endif // defined(BOOST_ASIO_ENABLE_OLD_SSL)
  34. #include <boost/asio/detail/push_options.hpp>
  35. namespace boost {
  36. namespace asio {
  37. namespace ssl {
  38. #if defined(BOOST_ASIO_ENABLE_OLD_SSL)
  39. using boost::asio::ssl::old::stream;
  40. #else // defined(BOOST_ASIO_ENABLE_OLD_SSL)
  41. /// Provides stream-oriented functionality using SSL.
  42. /**
  43. * The stream class template provides asynchronous and blocking stream-oriented
  44. * functionality using SSL.
  45. *
  46. * @par Thread Safety
  47. * @e Distinct @e objects: Safe.@n
  48. * @e Shared @e objects: Unsafe. The application must also ensure that all
  49. * asynchronous operations are performed within the same implicit or explicit
  50. * strand.
  51. *
  52. * @par Example
  53. * To use the SSL stream template with an ip::tcp::socket, you would write:
  54. * @code
  55. * boost::asio::io_service io_service;
  56. * boost::asio::ssl::context ctx(boost::asio::ssl::context::sslv23);
  57. * boost::asio::ssl::stream<asio:ip::tcp::socket> sock(io_service, ctx);
  58. * @endcode
  59. *
  60. * @par Concepts:
  61. * AsyncReadStream, AsyncWriteStream, Stream, SyncReadStream, SyncWriteStream.
  62. */
  63. template <typename Stream>
  64. class stream :
  65. public stream_base,
  66. private noncopyable
  67. {
  68. public:
  69. /// The native handle type of the SSL stream.
  70. typedef SSL* native_handle_type;
  71. /// Structure for use with deprecated impl_type.
  72. struct impl_struct
  73. {
  74. SSL* ssl;
  75. };
  76. /// (Deprecated: Use native_handle_type.) The underlying implementation type.
  77. typedef impl_struct* impl_type;
  78. /// The type of the next layer.
  79. typedef typename remove_reference<Stream>::type next_layer_type;
  80. /// The type of the lowest layer.
  81. typedef typename next_layer_type::lowest_layer_type lowest_layer_type;
  82. /// Construct a stream.
  83. /**
  84. * This constructor creates a stream and initialises the underlying stream
  85. * object.
  86. *
  87. * @param arg The argument to be passed to initialise the underlying stream.
  88. *
  89. * @param ctx The SSL context to be used for the stream.
  90. */
  91. template <typename Arg>
  92. stream(Arg& arg, context& ctx)
  93. : next_layer_(arg),
  94. core_(ctx.native_handle(), next_layer_.lowest_layer().get_io_service())
  95. {
  96. backwards_compatible_impl_.ssl = core_.engine_.native_handle();
  97. }
  98. /// Destructor.
  99. ~stream()
  100. {
  101. }
  102. /// Get the io_service associated with the object.
  103. /**
  104. * This function may be used to obtain the io_service object that the stream
  105. * uses to dispatch handlers for asynchronous operations.
  106. *
  107. * @return A reference to the io_service object that stream will use to
  108. * dispatch handlers. Ownership is not transferred to the caller.
  109. */
  110. boost::asio::io_service& get_io_service()
  111. {
  112. return next_layer_.lowest_layer().get_io_service();
  113. }
  114. /// Get the underlying implementation in the native type.
  115. /**
  116. * This function may be used to obtain the underlying implementation of the
  117. * context. This is intended to allow access to context functionality that is
  118. * not otherwise provided.
  119. *
  120. * @par Example
  121. * The native_handle() function returns a pointer of type @c SSL* that is
  122. * suitable for passing to functions such as @c SSL_get_verify_result and
  123. * @c SSL_get_peer_certificate:
  124. * @code
  125. * boost::asio::ssl::stream<asio:ip::tcp::socket> sock(io_service, ctx);
  126. *
  127. * // ... establish connection and perform handshake ...
  128. *
  129. * if (X509* cert = SSL_get_peer_certificate(sock.native_handle()))
  130. * {
  131. * if (SSL_get_verify_result(sock.native_handle()) == X509_V_OK)
  132. * {
  133. * // ...
  134. * }
  135. * }
  136. * @endcode
  137. */
  138. native_handle_type native_handle()
  139. {
  140. return core_.engine_.native_handle();
  141. }
  142. /// (Deprecated: Use native_handle().) Get the underlying implementation in
  143. /// the native type.
  144. /**
  145. * This function may be used to obtain the underlying implementation of the
  146. * context. This is intended to allow access to stream functionality that is
  147. * not otherwise provided.
  148. */
  149. impl_type impl()
  150. {
  151. return &backwards_compatible_impl_;
  152. }
  153. /// Get a reference to the next layer.
  154. /**
  155. * This function returns a reference to the next layer in a stack of stream
  156. * layers.
  157. *
  158. * @return A reference to the next layer in the stack of stream layers.
  159. * Ownership is not transferred to the caller.
  160. */
  161. const next_layer_type& next_layer() const
  162. {
  163. return next_layer_;
  164. }
  165. /// Get a reference to the next layer.
  166. /**
  167. * This function returns a reference to the next layer in a stack of stream
  168. * layers.
  169. *
  170. * @return A reference to the next layer in the stack of stream layers.
  171. * Ownership is not transferred to the caller.
  172. */
  173. next_layer_type& next_layer()
  174. {
  175. return next_layer_;
  176. }
  177. /// Get a reference to the lowest layer.
  178. /**
  179. * This function returns a reference to the lowest layer in a stack of
  180. * stream layers.
  181. *
  182. * @return A reference to the lowest layer in the stack of stream layers.
  183. * Ownership is not transferred to the caller.
  184. */
  185. lowest_layer_type& lowest_layer()
  186. {
  187. return next_layer_.lowest_layer();
  188. }
  189. /// Get a reference to the lowest layer.
  190. /**
  191. * This function returns a reference to the lowest layer in a stack of
  192. * stream layers.
  193. *
  194. * @return A reference to the lowest layer in the stack of stream layers.
  195. * Ownership is not transferred to the caller.
  196. */
  197. const lowest_layer_type& lowest_layer() const
  198. {
  199. return next_layer_.lowest_layer();
  200. }
  201. /// Set the peer verification mode.
  202. /**
  203. * This function may be used to configure the peer verification mode used by
  204. * the stream. The new mode will override the mode inherited from the context.
  205. *
  206. * @param v A bitmask of peer verification modes. See @ref verify_mode for
  207. * available values.
  208. *
  209. * @throws boost::system::system_error Thrown on failure.
  210. *
  211. * @note Calls @c SSL_set_verify.
  212. */
  213. void set_verify_mode(verify_mode v)
  214. {
  215. boost::system::error_code ec;
  216. set_verify_mode(v, ec);
  217. boost::asio::detail::throw_error(ec, "set_verify_mode");
  218. }
  219. /// Set the peer verification mode.
  220. /**
  221. * This function may be used to configure the peer verification mode used by
  222. * the stream. The new mode will override the mode inherited from the context.
  223. *
  224. * @param v A bitmask of peer verification modes. See @ref verify_mode for
  225. * available values.
  226. *
  227. * @param ec Set to indicate what error occurred, if any.
  228. *
  229. * @note Calls @c SSL_set_verify.
  230. */
  231. boost::system::error_code set_verify_mode(
  232. verify_mode v, boost::system::error_code& ec)
  233. {
  234. return core_.engine_.set_verify_mode(v, ec);
  235. }
  236. /// Set the peer verification depth.
  237. /**
  238. * This function may be used to configure the maximum verification depth
  239. * allowed by the stream.
  240. *
  241. * @param depth Maximum depth for the certificate chain verification that
  242. * shall be allowed.
  243. *
  244. * @throws boost::system::system_error Thrown on failure.
  245. *
  246. * @note Calls @c SSL_set_verify_depth.
  247. */
  248. void set_verify_depth(int depth)
  249. {
  250. boost::system::error_code ec;
  251. set_verify_depth(depth, ec);
  252. boost::asio::detail::throw_error(ec, "set_verify_depth");
  253. }
  254. /// Set the peer verification depth.
  255. /**
  256. * This function may be used to configure the maximum verification depth
  257. * allowed by the stream.
  258. *
  259. * @param depth Maximum depth for the certificate chain verification that
  260. * shall be allowed.
  261. *
  262. * @param ec Set to indicate what error occurred, if any.
  263. *
  264. * @note Calls @c SSL_set_verify_depth.
  265. */
  266. boost::system::error_code set_verify_depth(
  267. int depth, boost::system::error_code& ec)
  268. {
  269. return core_.engine_.set_verify_depth(depth, ec);
  270. }
  271. /// Set the callback used to verify peer certificates.
  272. /**
  273. * This function is used to specify a callback function that will be called
  274. * by the implementation when it needs to verify a peer certificate.
  275. *
  276. * @param callback The function object to be used for verifying a certificate.
  277. * The function signature of the handler must be:
  278. * @code bool verify_callback(
  279. * bool preverified, // True if the certificate passed pre-verification.
  280. * verify_context& ctx // The peer certificate and other context.
  281. * ); @endcode
  282. * The return value of the callback is true if the certificate has passed
  283. * verification, false otherwise.
  284. *
  285. * @throws boost::system::system_error Thrown on failure.
  286. *
  287. * @note Calls @c SSL_set_verify.
  288. */
  289. template <typename VerifyCallback>
  290. void set_verify_callback(VerifyCallback callback)
  291. {
  292. boost::system::error_code ec;
  293. this->set_verify_callback(callback, ec);
  294. boost::asio::detail::throw_error(ec, "set_verify_callback");
  295. }
  296. /// Set the callback used to verify peer certificates.
  297. /**
  298. * This function is used to specify a callback function that will be called
  299. * by the implementation when it needs to verify a peer certificate.
  300. *
  301. * @param callback The function object to be used for verifying a certificate.
  302. * The function signature of the handler must be:
  303. * @code bool verify_callback(
  304. * bool preverified, // True if the certificate passed pre-verification.
  305. * verify_context& ctx // The peer certificate and other context.
  306. * ); @endcode
  307. * The return value of the callback is true if the certificate has passed
  308. * verification, false otherwise.
  309. *
  310. * @param ec Set to indicate what error occurred, if any.
  311. *
  312. * @note Calls @c SSL_set_verify.
  313. */
  314. template <typename VerifyCallback>
  315. boost::system::error_code set_verify_callback(VerifyCallback callback,
  316. boost::system::error_code& ec)
  317. {
  318. return core_.engine_.set_verify_callback(
  319. new detail::verify_callback<VerifyCallback>(callback), ec);
  320. }
  321. /// Perform SSL handshaking.
  322. /**
  323. * This function is used to perform SSL handshaking on the stream. The
  324. * function call will block until handshaking is complete or an error occurs.
  325. *
  326. * @param type The type of handshaking to be performed, i.e. as a client or as
  327. * a server.
  328. *
  329. * @throws boost::system::system_error Thrown on failure.
  330. */
  331. void handshake(handshake_type type)
  332. {
  333. boost::system::error_code ec;
  334. handshake(type, ec);
  335. boost::asio::detail::throw_error(ec, "handshake");
  336. }
  337. /// Perform SSL handshaking.
  338. /**
  339. * This function is used to perform SSL handshaking on the stream. The
  340. * function call will block until handshaking is complete or an error occurs.
  341. *
  342. * @param type The type of handshaking to be performed, i.e. as a client or as
  343. * a server.
  344. *
  345. * @param ec Set to indicate what error occurred, if any.
  346. */
  347. boost::system::error_code handshake(handshake_type type,
  348. boost::system::error_code& ec)
  349. {
  350. detail::io(next_layer_, core_, detail::handshake_op(type), ec);
  351. return ec;
  352. }
  353. /// Perform SSL handshaking.
  354. /**
  355. * This function is used to perform SSL handshaking on the stream. The
  356. * function call will block until handshaking is complete or an error occurs.
  357. *
  358. * @param type The type of handshaking to be performed, i.e. as a client or as
  359. * a server.
  360. *
  361. * @param buffers The buffered data to be reused for the handshake.
  362. *
  363. * @throws boost::system::system_error Thrown on failure.
  364. */
  365. template <typename ConstBufferSequence>
  366. void handshake(handshake_type type, const ConstBufferSequence& buffers)
  367. {
  368. boost::system::error_code ec;
  369. handshake(type, buffers, ec);
  370. boost::asio::detail::throw_error(ec, "handshake");
  371. }
  372. /// Perform SSL handshaking.
  373. /**
  374. * This function is used to perform SSL handshaking on the stream. The
  375. * function call will block until handshaking is complete or an error occurs.
  376. *
  377. * @param type The type of handshaking to be performed, i.e. as a client or as
  378. * a server.
  379. *
  380. * @param buffers The buffered data to be reused for the handshake.
  381. *
  382. * @param ec Set to indicate what error occurred, if any.
  383. */
  384. template <typename ConstBufferSequence>
  385. boost::system::error_code handshake(handshake_type type,
  386. const ConstBufferSequence& buffers, boost::system::error_code& ec)
  387. {
  388. detail::io(next_layer_, core_,
  389. detail::buffered_handshake_op<ConstBufferSequence>(type, buffers), ec);
  390. return ec;
  391. }
  392. /// Start an asynchronous SSL handshake.
  393. /**
  394. * This function is used to asynchronously perform an SSL handshake on the
  395. * stream. This function call always returns immediately.
  396. *
  397. * @param type The type of handshaking to be performed, i.e. as a client or as
  398. * a server.
  399. *
  400. * @param handler The handler to be called when the handshake operation
  401. * completes. Copies will be made of the handler as required. The equivalent
  402. * function signature of the handler must be:
  403. * @code void handler(
  404. * const boost::system::error_code& error // Result of operation.
  405. * ); @endcode
  406. */
  407. template <typename HandshakeHandler>
  408. BOOST_ASIO_INITFN_RESULT_TYPE(HandshakeHandler,
  409. void (boost::system::error_code))
  410. async_handshake(handshake_type type,
  411. BOOST_ASIO_MOVE_ARG(HandshakeHandler) handler)
  412. {
  413. // If you get an error on the following line it means that your handler does
  414. // not meet the documented type requirements for a HandshakeHandler.
  415. BOOST_ASIO_HANDSHAKE_HANDLER_CHECK(HandshakeHandler, handler) type_check;
  416. boost::asio::detail::async_result_init<
  417. HandshakeHandler, void (boost::system::error_code)> init(
  418. BOOST_ASIO_MOVE_CAST(HandshakeHandler)(handler));
  419. detail::async_io(next_layer_, core_,
  420. detail::handshake_op(type), init.handler);
  421. return init.result.get();
  422. }
  423. /// Start an asynchronous SSL handshake.
  424. /**
  425. * This function is used to asynchronously perform an SSL handshake on the
  426. * stream. This function call always returns immediately.
  427. *
  428. * @param type The type of handshaking to be performed, i.e. as a client or as
  429. * a server.
  430. *
  431. * @param buffers The buffered data to be reused for the handshake. Although
  432. * the buffers object may be copied as necessary, ownership of the underlying
  433. * buffers is retained by the caller, which must guarantee that they remain
  434. * valid until the handler is called.
  435. *
  436. * @param handler The handler to be called when the handshake operation
  437. * completes. Copies will be made of the handler as required. The equivalent
  438. * function signature of the handler must be:
  439. * @code void handler(
  440. * const boost::system::error_code& error, // Result of operation.
  441. * std::size_t bytes_transferred // Amount of buffers used in handshake.
  442. * ); @endcode
  443. */
  444. template <typename ConstBufferSequence, typename BufferedHandshakeHandler>
  445. BOOST_ASIO_INITFN_RESULT_TYPE(BufferedHandshakeHandler,
  446. void (boost::system::error_code, std::size_t))
  447. async_handshake(handshake_type type, const ConstBufferSequence& buffers,
  448. BOOST_ASIO_MOVE_ARG(BufferedHandshakeHandler) handler)
  449. {
  450. // If you get an error on the following line it means that your handler does
  451. // not meet the documented type requirements for a BufferedHandshakeHandler.
  452. BOOST_ASIO_BUFFERED_HANDSHAKE_HANDLER_CHECK(
  453. BufferedHandshakeHandler, handler) type_check;
  454. boost::asio::detail::async_result_init<BufferedHandshakeHandler,
  455. void (boost::system::error_code, std::size_t)> init(
  456. BOOST_ASIO_MOVE_CAST(BufferedHandshakeHandler)(handler));
  457. detail::async_io(next_layer_, core_,
  458. detail::buffered_handshake_op<ConstBufferSequence>(type, buffers),
  459. init.handler);
  460. return init.result.get();
  461. }
  462. /// Shut down SSL on the stream.
  463. /**
  464. * This function is used to shut down SSL on the stream. The function call
  465. * will block until SSL has been shut down or an error occurs.
  466. *
  467. * @throws boost::system::system_error Thrown on failure.
  468. */
  469. void shutdown()
  470. {
  471. boost::system::error_code ec;
  472. shutdown(ec);
  473. boost::asio::detail::throw_error(ec, "shutdown");
  474. }
  475. /// Shut down SSL on the stream.
  476. /**
  477. * This function is used to shut down SSL on the stream. The function call
  478. * will block until SSL has been shut down or an error occurs.
  479. *
  480. * @param ec Set to indicate what error occurred, if any.
  481. */
  482. boost::system::error_code shutdown(boost::system::error_code& ec)
  483. {
  484. detail::io(next_layer_, core_, detail::shutdown_op(), ec);
  485. return ec;
  486. }
  487. /// Asynchronously shut down SSL on the stream.
  488. /**
  489. * This function is used to asynchronously shut down SSL on the stream. This
  490. * function call always returns immediately.
  491. *
  492. * @param handler The handler to be called when the handshake operation
  493. * completes. Copies will be made of the handler as required. The equivalent
  494. * function signature of the handler must be:
  495. * @code void handler(
  496. * const boost::system::error_code& error // Result of operation.
  497. * ); @endcode
  498. */
  499. template <typename ShutdownHandler>
  500. BOOST_ASIO_INITFN_RESULT_TYPE(ShutdownHandler,
  501. void (boost::system::error_code))
  502. async_shutdown(BOOST_ASIO_MOVE_ARG(ShutdownHandler) handler)
  503. {
  504. // If you get an error on the following line it means that your handler does
  505. // not meet the documented type requirements for a ShutdownHandler.
  506. BOOST_ASIO_SHUTDOWN_HANDLER_CHECK(ShutdownHandler, handler) type_check;
  507. boost::asio::detail::async_result_init<
  508. ShutdownHandler, void (boost::system::error_code)> init(
  509. BOOST_ASIO_MOVE_CAST(ShutdownHandler)(handler));
  510. detail::async_io(next_layer_, core_, detail::shutdown_op(), init.handler);
  511. return init.result.get();
  512. }
  513. /// Write some data to the stream.
  514. /**
  515. * This function is used to write data on the stream. The function call will
  516. * block until one or more bytes of data has been written successfully, or
  517. * until an error occurs.
  518. *
  519. * @param buffers The data to be written.
  520. *
  521. * @returns The number of bytes written.
  522. *
  523. * @throws boost::system::system_error Thrown on failure.
  524. *
  525. * @note The write_some operation may not transmit all of the data to the
  526. * peer. Consider using the @ref write function if you need to ensure that all
  527. * data is written before the blocking operation completes.
  528. */
  529. template <typename ConstBufferSequence>
  530. std::size_t write_some(const ConstBufferSequence& buffers)
  531. {
  532. boost::system::error_code ec;
  533. std::size_t n = write_some(buffers, ec);
  534. boost::asio::detail::throw_error(ec, "write_some");
  535. return n;
  536. }
  537. /// Write some data to the stream.
  538. /**
  539. * This function is used to write data on the stream. The function call will
  540. * block until one or more bytes of data has been written successfully, or
  541. * until an error occurs.
  542. *
  543. * @param buffers The data to be written to the stream.
  544. *
  545. * @param ec Set to indicate what error occurred, if any.
  546. *
  547. * @returns The number of bytes written. Returns 0 if an error occurred.
  548. *
  549. * @note The write_some operation may not transmit all of the data to the
  550. * peer. Consider using the @ref write function if you need to ensure that all
  551. * data is written before the blocking operation completes.
  552. */
  553. template <typename ConstBufferSequence>
  554. std::size_t write_some(const ConstBufferSequence& buffers,
  555. boost::system::error_code& ec)
  556. {
  557. return detail::io(next_layer_, core_,
  558. detail::write_op<ConstBufferSequence>(buffers), ec);
  559. }
  560. /// Start an asynchronous write.
  561. /**
  562. * This function is used to asynchronously write one or more bytes of data to
  563. * the stream. The function call always returns immediately.
  564. *
  565. * @param buffers The data to be written to the stream. Although the buffers
  566. * object may be copied as necessary, ownership of the underlying buffers is
  567. * retained by the caller, which must guarantee that they remain valid until
  568. * the handler is called.
  569. *
  570. * @param handler The handler to be called when the write operation completes.
  571. * Copies will be made of the handler as required. The equivalent function
  572. * signature of the handler must be:
  573. * @code void handler(
  574. * const boost::system::error_code& error, // Result of operation.
  575. * std::size_t bytes_transferred // Number of bytes written.
  576. * ); @endcode
  577. *
  578. * @note The async_write_some operation may not transmit all of the data to
  579. * the peer. Consider using the @ref async_write function if you need to
  580. * ensure that all data is written before the blocking operation completes.
  581. */
  582. template <typename ConstBufferSequence, typename WriteHandler>
  583. BOOST_ASIO_INITFN_RESULT_TYPE(WriteHandler,
  584. void (boost::system::error_code, std::size_t))
  585. async_write_some(const ConstBufferSequence& buffers,
  586. BOOST_ASIO_MOVE_ARG(WriteHandler) handler)
  587. {
  588. // If you get an error on the following line it means that your handler does
  589. // not meet the documented type requirements for a WriteHandler.
  590. BOOST_ASIO_WRITE_HANDLER_CHECK(WriteHandler, handler) type_check;
  591. boost::asio::detail::async_result_init<
  592. WriteHandler, void (boost::system::error_code, std::size_t)> init(
  593. BOOST_ASIO_MOVE_CAST(WriteHandler)(handler));
  594. detail::async_io(next_layer_, core_,
  595. detail::write_op<ConstBufferSequence>(buffers), init.handler);
  596. return init.result.get();
  597. }
  598. /// Read some data from the stream.
  599. /**
  600. * This function is used to read data from the stream. The function call will
  601. * block until one or more bytes of data has been read successfully, or until
  602. * an error occurs.
  603. *
  604. * @param buffers The buffers into which the data will be read.
  605. *
  606. * @returns The number of bytes read.
  607. *
  608. * @throws boost::system::system_error Thrown on failure.
  609. *
  610. * @note The read_some operation may not read all of the requested number of
  611. * bytes. Consider using the @ref read function if you need to ensure that the
  612. * requested amount of data is read before the blocking operation completes.
  613. */
  614. template <typename MutableBufferSequence>
  615. std::size_t read_some(const MutableBufferSequence& buffers)
  616. {
  617. boost::system::error_code ec;
  618. std::size_t n = read_some(buffers, ec);
  619. boost::asio::detail::throw_error(ec, "read_some");
  620. return n;
  621. }
  622. /// Read some data from the stream.
  623. /**
  624. * This function is used to read data from the stream. The function call will
  625. * block until one or more bytes of data has been read successfully, or until
  626. * an error occurs.
  627. *
  628. * @param buffers The buffers into which the data will be read.
  629. *
  630. * @param ec Set to indicate what error occurred, if any.
  631. *
  632. * @returns The number of bytes read. Returns 0 if an error occurred.
  633. *
  634. * @note The read_some operation may not read all of the requested number of
  635. * bytes. Consider using the @ref read function if you need to ensure that the
  636. * requested amount of data is read before the blocking operation completes.
  637. */
  638. template <typename MutableBufferSequence>
  639. std::size_t read_some(const MutableBufferSequence& buffers,
  640. boost::system::error_code& ec)
  641. {
  642. return detail::io(next_layer_, core_,
  643. detail::read_op<MutableBufferSequence>(buffers), ec);
  644. }
  645. /// Start an asynchronous read.
  646. /**
  647. * This function is used to asynchronously read one or more bytes of data from
  648. * the stream. The function call always returns immediately.
  649. *
  650. * @param buffers The buffers into which the data will be read. Although the
  651. * buffers object may be copied as necessary, ownership of the underlying
  652. * buffers is retained by the caller, which must guarantee that they remain
  653. * valid until the handler is called.
  654. *
  655. * @param handler The handler to be called when the read operation completes.
  656. * Copies will be made of the handler as required. The equivalent function
  657. * signature of the handler must be:
  658. * @code void handler(
  659. * const boost::system::error_code& error, // Result of operation.
  660. * std::size_t bytes_transferred // Number of bytes read.
  661. * ); @endcode
  662. *
  663. * @note The async_read_some operation may not read all of the requested
  664. * number of bytes. Consider using the @ref async_read function if you need to
  665. * ensure that the requested amount of data is read before the asynchronous
  666. * operation completes.
  667. */
  668. template <typename MutableBufferSequence, typename ReadHandler>
  669. BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  670. void (boost::system::error_code, std::size_t))
  671. async_read_some(const MutableBufferSequence& buffers,
  672. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  673. {
  674. // If you get an error on the following line it means that your handler does
  675. // not meet the documented type requirements for a ReadHandler.
  676. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  677. boost::asio::detail::async_result_init<
  678. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  679. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  680. detail::async_io(next_layer_, core_,
  681. detail::read_op<MutableBufferSequence>(buffers), init.handler);
  682. return init.result.get();
  683. }
  684. private:
  685. Stream next_layer_;
  686. detail::stream_core core_;
  687. impl_struct backwards_compatible_impl_;
  688. };
  689. #endif // defined(BOOST_ASIO_ENABLE_OLD_SSL)
  690. } // namespace ssl
  691. } // namespace asio
  692. } // namespace boost
  693. #include <boost/asio/detail/pop_options.hpp>
  694. #endif // BOOST_ASIO_SSL_STREAM_HPP