buffer_sequence_adapter.hpp 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. //
  2. // detail/buffer_sequence_adapter.hpp
  3. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2011 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 ASIO_DETAIL_BUFFER_SEQUENCE_ADAPTER_HPP
  11. #define ASIO_DETAIL_BUFFER_SEQUENCE_ADAPTER_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include "asio/detail/config.hpp"
  16. #include "asio/buffer.hpp"
  17. #include "asio/detail/socket_types.hpp"
  18. #include "asio/detail/push_options.hpp"
  19. namespace asio {
  20. namespace detail {
  21. class buffer_sequence_adapter_base
  22. {
  23. protected:
  24. #if defined(BOOST_WINDOWS) || defined(__CYGWIN__)
  25. typedef WSABUF native_buffer_type;
  26. static void init_native_buffer(WSABUF& buf,
  27. const asio::mutable_buffer& buffer)
  28. {
  29. buf.buf = asio::buffer_cast<char*>(buffer);
  30. buf.len = static_cast<ULONG>(asio::buffer_size(buffer));
  31. }
  32. static void init_native_buffer(WSABUF& buf,
  33. const asio::const_buffer& buffer)
  34. {
  35. buf.buf = const_cast<char*>(asio::buffer_cast<const char*>(buffer));
  36. buf.len = static_cast<ULONG>(asio::buffer_size(buffer));
  37. }
  38. #else // defined(BOOST_WINDOWS) || defined(__CYGWIN__)
  39. typedef iovec native_buffer_type;
  40. static void init_iov_base(void*& base, void* addr)
  41. {
  42. base = addr;
  43. }
  44. template <typename T>
  45. static void init_iov_base(T& base, void* addr)
  46. {
  47. base = static_cast<T>(addr);
  48. }
  49. static void init_native_buffer(iovec& iov,
  50. const asio::mutable_buffer& buffer)
  51. {
  52. init_iov_base(iov.iov_base, asio::buffer_cast<void*>(buffer));
  53. iov.iov_len = asio::buffer_size(buffer);
  54. }
  55. static void init_native_buffer(iovec& iov,
  56. const asio::const_buffer& buffer)
  57. {
  58. init_iov_base(iov.iov_base, const_cast<void*>(
  59. asio::buffer_cast<const void*>(buffer)));
  60. iov.iov_len = asio::buffer_size(buffer);
  61. }
  62. #endif // defined(BOOST_WINDOWS) || defined(__CYGWIN__)
  63. };
  64. // Helper class to translate buffers into the native buffer representation.
  65. template <typename Buffer, typename Buffers>
  66. class buffer_sequence_adapter
  67. : buffer_sequence_adapter_base
  68. {
  69. public:
  70. explicit buffer_sequence_adapter(const Buffers& buffers)
  71. : count_(0), total_buffer_size_(0)
  72. {
  73. typename Buffers::const_iterator iter = buffers.begin();
  74. typename Buffers::const_iterator end = buffers.end();
  75. for (; iter != end && count_ < max_buffers; ++iter, ++count_)
  76. {
  77. Buffer buffer(*iter);
  78. init_native_buffer(buffers_[count_], buffer);
  79. total_buffer_size_ += asio::buffer_size(buffer);
  80. }
  81. }
  82. native_buffer_type* buffers()
  83. {
  84. return buffers_;
  85. }
  86. std::size_t count() const
  87. {
  88. return count_;
  89. }
  90. bool all_empty() const
  91. {
  92. return total_buffer_size_ == 0;
  93. }
  94. static bool all_empty(const Buffers& buffers)
  95. {
  96. typename Buffers::const_iterator iter = buffers.begin();
  97. typename Buffers::const_iterator end = buffers.end();
  98. std::size_t i = 0;
  99. for (; iter != end && i < max_buffers; ++iter, ++i)
  100. if (asio::buffer_size(Buffer(*iter)) > 0)
  101. return false;
  102. return true;
  103. }
  104. static void validate(const Buffers& buffers)
  105. {
  106. typename Buffers::const_iterator iter = buffers.begin();
  107. typename Buffers::const_iterator end = buffers.end();
  108. for (; iter != end; ++iter)
  109. {
  110. Buffer buffer(*iter);
  111. asio::buffer_cast<const void*>(buffer);
  112. }
  113. }
  114. static Buffer first(const Buffers& buffers)
  115. {
  116. typename Buffers::const_iterator iter = buffers.begin();
  117. typename Buffers::const_iterator end = buffers.end();
  118. for (; iter != end; ++iter)
  119. {
  120. Buffer buffer(*iter);
  121. if (asio::buffer_size(buffer) != 0)
  122. return buffer;
  123. }
  124. return Buffer();
  125. }
  126. private:
  127. // The maximum number of buffers to support in a single operation.
  128. enum { max_buffers = 64 < max_iov_len ? 64 : max_iov_len };
  129. native_buffer_type buffers_[max_buffers];
  130. std::size_t count_;
  131. std::size_t total_buffer_size_;
  132. };
  133. template <typename Buffer>
  134. class buffer_sequence_adapter<Buffer, asio::mutable_buffers_1>
  135. : buffer_sequence_adapter_base
  136. {
  137. public:
  138. explicit buffer_sequence_adapter(
  139. const asio::mutable_buffers_1& buffers)
  140. {
  141. init_native_buffer(buffer_, Buffer(buffers));
  142. total_buffer_size_ = asio::buffer_size(buffers);
  143. }
  144. native_buffer_type* buffers()
  145. {
  146. return &buffer_;
  147. }
  148. std::size_t count() const
  149. {
  150. return 1;
  151. }
  152. bool all_empty() const
  153. {
  154. return total_buffer_size_ == 0;
  155. }
  156. static bool all_empty(const asio::mutable_buffers_1& buffers)
  157. {
  158. return asio::buffer_size(buffers) == 0;
  159. }
  160. static void validate(const asio::mutable_buffers_1& buffers)
  161. {
  162. asio::buffer_cast<const void*>(buffers);
  163. }
  164. static Buffer first(const asio::mutable_buffers_1& buffers)
  165. {
  166. return Buffer(buffers);
  167. }
  168. private:
  169. native_buffer_type buffer_;
  170. std::size_t total_buffer_size_;
  171. };
  172. template <typename Buffer>
  173. class buffer_sequence_adapter<Buffer, asio::const_buffers_1>
  174. : buffer_sequence_adapter_base
  175. {
  176. public:
  177. explicit buffer_sequence_adapter(
  178. const asio::const_buffers_1& buffers)
  179. {
  180. init_native_buffer(buffer_, Buffer(buffers));
  181. total_buffer_size_ = asio::buffer_size(buffers);
  182. }
  183. native_buffer_type* buffers()
  184. {
  185. return &buffer_;
  186. }
  187. std::size_t count() const
  188. {
  189. return 1;
  190. }
  191. bool all_empty() const
  192. {
  193. return total_buffer_size_ == 0;
  194. }
  195. static bool all_empty(const asio::const_buffers_1& buffers)
  196. {
  197. return asio::buffer_size(buffers) == 0;
  198. }
  199. static void validate(const asio::const_buffers_1& buffers)
  200. {
  201. asio::buffer_cast<const void*>(buffers);
  202. }
  203. static Buffer first(const asio::const_buffers_1& buffers)
  204. {
  205. return Buffer(buffers);
  206. }
  207. private:
  208. native_buffer_type buffer_;
  209. std::size_t total_buffer_size_;
  210. };
  211. } // namespace detail
  212. } // namespace asio
  213. #include "asio/detail/pop_options.hpp"
  214. #endif // ASIO_DETAIL_BUFFER_SEQUENCE_ADAPTER_HPP