buffer.hpp 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. // (C) Copyright 2008 CodeRage, LLC (turkanis at coderage dot com)
  2. // (C) Copyright 2003-2007 Jonathan Turkanis
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
  5. // See http://www.boost.org/libs/iostreams for documentation.
  6. #ifndef BOOST_IOSTREAMS_DETAIL_BUFFERS_HPP_INCLUDED
  7. #define BOOST_IOSTREAMS_DETAIL_BUFFERS_HPP_INCLUDED
  8. #if defined(_MSC_VER) && (_MSC_VER >= 1020)
  9. # pragma once
  10. #endif
  11. #include <algorithm> // swap.
  12. #include <memory> // allocator.
  13. #include <boost/config.hpp> // member templates.
  14. #include <boost/iostreams/char_traits.hpp>
  15. #include <boost/iostreams/detail/ios.hpp> // streamsize.
  16. #include <boost/iostreams/read.hpp>
  17. #include <boost/iostreams/traits.hpp> // int_type_of.
  18. #include <boost/iostreams/checked_operations.hpp>
  19. #include <boost/mpl/if.hpp>
  20. #include <boost/type_traits/is_same.hpp>
  21. namespace boost { namespace iostreams { namespace detail {
  22. //----------------Buffers-----------------------------------------------------//
  23. //
  24. // Template name: buffer
  25. // Description: Character buffer.
  26. // Template parameters:
  27. // Ch - The character type.
  28. // Alloc - The Allocator type.
  29. //
  30. template< typename Ch,
  31. typename Alloc = std::allocator<Ch> >
  32. class basic_buffer {
  33. private:
  34. #ifndef BOOST_NO_STD_ALLOCATOR
  35. typedef typename Alloc::template rebind<Ch>::other allocator_type;
  36. #else
  37. typedef std::allocator<Ch> allocator_type;
  38. #endif
  39. public:
  40. basic_buffer();
  41. basic_buffer(int buffer_size);
  42. ~basic_buffer();
  43. void resize(int buffer_size);
  44. Ch* begin() const { return buf_; }
  45. Ch* end() const { return buf_ + size_; }
  46. Ch* data() const { return buf_; }
  47. std::streamsize size() const { return size_; }
  48. void swap(basic_buffer& rhs);
  49. private:
  50. // Disallow copying and assignment.
  51. basic_buffer(const basic_buffer&);
  52. basic_buffer& operator=(const basic_buffer&);
  53. Ch* buf_;
  54. std::streamsize size_;
  55. };
  56. template<typename Ch, typename Alloc>
  57. void swap(basic_buffer<Ch, Alloc>& lhs, basic_buffer<Ch, Alloc>& rhs)
  58. { lhs.swap(rhs); }
  59. //
  60. // Template name: buffer
  61. // Description: Character buffer with two pointers accessible via ptr() and
  62. // eptr().
  63. // Template parameters:
  64. // Ch - A character type.
  65. //
  66. template< typename Ch,
  67. typename Alloc = std::allocator<Ch> >
  68. class buffer : public basic_buffer<Ch, Alloc> {
  69. private:
  70. typedef basic_buffer<Ch, Alloc> base;
  71. public:
  72. typedef iostreams::char_traits<Ch> traits_type;
  73. using base::resize;
  74. using base::data;
  75. using base::size;
  76. typedef Ch* const const_pointer;
  77. buffer(int buffer_size);
  78. Ch* & ptr() { return ptr_; }
  79. const_pointer& ptr() const { return ptr_; }
  80. Ch* & eptr() { return eptr_; }
  81. const_pointer& eptr() const { return eptr_; }
  82. void set(std::streamsize ptr, std::streamsize end);
  83. void swap(buffer& rhs);
  84. // Returns an int_type as a status code.
  85. template<typename Source>
  86. typename int_type_of<Source>::type fill(Source& src)
  87. {
  88. using namespace std;
  89. std::streamsize keep;
  90. if ((keep = static_cast<std::streamsize>(eptr_ - ptr_)) > 0)
  91. traits_type::move(this->data(), ptr_, keep);
  92. set(0, keep);
  93. std::streamsize result =
  94. iostreams::read(src, this->data() + keep, this->size() - keep);
  95. if (result != -1)
  96. this->set(0, keep + result);
  97. return result == -1 ?
  98. traits_type::eof() :
  99. result == 0 ?
  100. traits_type::would_block() :
  101. traits_type::good();
  102. }
  103. // Returns true if one or more characters were written.
  104. template<typename Sink>
  105. bool flush(Sink& dest)
  106. {
  107. using namespace std;
  108. std::streamsize amt = static_cast<std::streamsize>(eptr_ - ptr_);
  109. std::streamsize result = iostreams::write_if(dest, ptr_, amt);
  110. if (result < amt) {
  111. traits_type::move( this->data(),
  112. ptr_ + result,
  113. amt - result );
  114. }
  115. this->set(0, amt - result);
  116. return result != 0;
  117. }
  118. private:
  119. Ch *ptr_, *eptr_;
  120. };
  121. template<typename Ch, typename Alloc>
  122. void swap(buffer<Ch, Alloc>& lhs, buffer<Ch, Alloc>& rhs)
  123. { lhs.swap(rhs); }
  124. //--------------Implementation of basic_buffer--------------------------------//
  125. template<typename Ch, typename Alloc>
  126. basic_buffer<Ch, Alloc>::basic_buffer() : buf_(0), size_(0) { }
  127. template<typename Ch, typename Alloc>
  128. basic_buffer<Ch, Alloc>::basic_buffer(int buffer_size)
  129. : buf_(static_cast<Ch*>(allocator_type().allocate(buffer_size, 0))),
  130. size_(buffer_size) // Cast for SunPro 5.3.
  131. { }
  132. template<typename Ch, typename Alloc>
  133. inline basic_buffer<Ch, Alloc>::~basic_buffer()
  134. {
  135. if (buf_) {
  136. allocator_type().deallocate(buf_,
  137. static_cast<BOOST_DEDUCED_TYPENAME Alloc::size_type>(size_));
  138. }
  139. }
  140. template<typename Ch, typename Alloc>
  141. inline void basic_buffer<Ch, Alloc>::resize(int buffer_size)
  142. {
  143. if (size_ != buffer_size) {
  144. basic_buffer<Ch, Alloc> temp(buffer_size);
  145. std::swap(size_, temp.size_);
  146. std::swap(buf_, temp.buf_);
  147. }
  148. }
  149. template<typename Ch, typename Alloc>
  150. void basic_buffer<Ch, Alloc>::swap(basic_buffer& rhs)
  151. {
  152. std::swap(buf_, rhs.buf_);
  153. std::swap(size_, rhs.size_);
  154. }
  155. //--------------Implementation of buffer--------------------------------------//
  156. template<typename Ch, typename Alloc>
  157. buffer<Ch, Alloc>::buffer(int buffer_size)
  158. : basic_buffer<Ch, Alloc>(buffer_size) { }
  159. template<typename Ch, typename Alloc>
  160. inline void buffer<Ch, Alloc>::set(std::streamsize ptr, std::streamsize end)
  161. {
  162. ptr_ = data() + ptr;
  163. eptr_ = data() + end;
  164. }
  165. template<typename Ch, typename Alloc>
  166. inline void buffer<Ch, Alloc>::swap(buffer& rhs)
  167. {
  168. base::swap(rhs);
  169. std::swap(ptr_, rhs.ptr_);
  170. std::swap(eptr_, rhs.eptr_);
  171. }
  172. //----------------------------------------------------------------------------//
  173. } } } // End namespaces detail, iostreams, boost.
  174. #endif // #ifndef BOOST_IOSTREAMS_DETAIL_BUFFERS_HPP_INCLUDED