os/ossrv/ossrv_pub/boost_apis/boost/iostreams/code_converter.hpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // (C) Copyright Jonathan Turkanis 2003.
     2 // Distributed under the Boost Software License, Version 1.0. (See accompanying
     3 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
     4 
     5 // See http://www.boost.org/libs/iostreams for documentation.
     6 
     7 // Contains machinery for performing code conversion.
     8 
     9 #ifndef BOOST_IOSTREAMS_CODE_CONVERTER_HPP_INCLUDED
    10 #define BOOST_IOSTREAMS_CODE_CONVERTER_HPP_INCLUDED
    11 
    12 #if defined(_MSC_VER) && (_MSC_VER >= 1020)
    13 # pragma once
    14 #endif
    15 
    16 #include <boost/iostreams/detail/config/wide_streams.hpp>
    17 #if defined(BOOST_IOSTREAMS_NO_WIDE_STREAMS) || \
    18     defined(BOOST_IOSTREAMS_NO_LOCALE) \
    19     /**/
    20 # error code conversion not supported on this platform
    21 #endif
    22 
    23 #include <algorithm>                       // max.
    24 #include <cstring>                         // memcpy.
    25 #include <exception>
    26 #include <boost/config.hpp>                // DEDUCED_TYPENAME.
    27 #include <boost/iostreams/char_traits.hpp>
    28 #include <boost/iostreams/constants.hpp>   // default_filter_buffer_size.
    29 #include <boost/iostreams/detail/adapter/concept_adapter.hpp>
    30 #include <boost/iostreams/detail/adapter/direct_adapter.hpp>
    31 #include <boost/iostreams/detail/buffer.hpp>
    32 #include <boost/iostreams/detail/call_traits.hpp>
    33 #include <boost/iostreams/detail/codecvt_holder.hpp>
    34 #include <boost/iostreams/detail/codecvt_helper.hpp>
    35 #include <boost/iostreams/detail/double_object.hpp>
    36 #include <boost/iostreams/detail/forward.hpp>
    37 #include <boost/iostreams/detail/ios.hpp> // failure, openmode, int types.
    38 #include <boost/iostreams/detail/select.hpp>
    39 #include <boost/iostreams/traits.hpp>
    40 #include <boost/iostreams/operations.hpp>
    41 #include <boost/optional.hpp>
    42 #include <boost/shared_ptr.hpp>
    43 #include <boost/static_assert.hpp>
    44 #include <boost/type_traits/is_convertible.hpp>
    45 #include <boost/type_traits/is_same.hpp>
    46 
    47 // Must come last.
    48 #include <boost/iostreams/detail/config/disable_warnings.hpp> // Borland 5.x
    49 
    50 namespace boost { namespace iostreams {
    51 
    52 struct code_conversion_error : BOOST_IOSTREAMS_FAILURE {
    53     code_conversion_error() 
    54         : BOOST_IOSTREAMS_FAILURE("code conversion error")
    55         { }
    56 };
    57 
    58 namespace detail {
    59 
    60 //--------------Definition of conversion_buffer-------------------------------//
    61 
    62 // Buffer and conversion state for reading.
    63 template<typename Codecvt, typename Alloc>
    64 class conversion_buffer 
    65     : public buffer<
    66                  BOOST_DEDUCED_TYPENAME detail::codecvt_extern<Codecvt>::type,
    67                  Alloc
    68              > 
    69 {
    70 public:
    71     typedef typename Codecvt::state_type state_type;
    72     conversion_buffer() 
    73         : buffer<
    74               BOOST_DEDUCED_TYPENAME detail::codecvt_extern<Codecvt>::type,
    75               Alloc
    76           >(0) 
    77     { 
    78         reset(); 
    79     }
    80     state_type& state() { return state_; }
    81     void reset() 
    82     { 
    83         if (this->size()) 
    84             this->set(0, 0);
    85         state_ = state_type(); 
    86     }
    87 private:
    88     state_type state_;
    89 };
    90 
    91 //--------------Definition of converter_impl----------------------------------//
    92 
    93 // Contains member data, open/is_open/close and buffer management functions.
    94 template<typename Device, typename Codecvt, typename Alloc>
    95 struct code_converter_impl {
    96     typedef typename codecvt_extern<Codecvt>::type          extern_type;
    97     typedef typename category_of<Device>::type              device_category;
    98     typedef is_convertible<device_category, input>          can_read;
    99     typedef is_convertible<device_category, output>         can_write;
   100     typedef is_convertible<device_category, bidirectional>  is_bidir;
   101     typedef typename 
   102             iostreams::select<  // Disambiguation for Tru64.
   103                 is_bidir, bidirectional,
   104                 can_read, input,
   105                 can_write, output
   106             >::type                                         mode;      
   107     typedef typename
   108             mpl::if_<
   109                 is_direct<Device>,
   110                 direct_adapter<Device>,
   111                 Device
   112             >::type                                         policy_type;
   113     typedef optional< concept_adapter<policy_type> >        storage_type;
   114     typedef is_convertible<device_category, two_sequence>   is_double;
   115     typedef conversion_buffer<Codecvt, Alloc>               buffer_type;
   116 
   117     code_converter_impl() : cvt_(), flags_(0) { }
   118 
   119     ~code_converter_impl()
   120     { 
   121         try { 
   122             if (flags_ & f_open) close(); 
   123         } catch (std::exception&) { /* */ } 
   124     }
   125 
   126     void open(const Device& dev, int buffer_size)
   127     {
   128         if (flags_ & f_open)
   129             throw BOOST_IOSTREAMS_FAILURE("already open");
   130         if (buffer_size == -1)
   131             buffer_size = default_filter_buffer_size;
   132         int max_length = cvt_.get().max_length();
   133         buffer_size = (std::max)(buffer_size, 2 * max_length);
   134         if (can_read::value) {
   135             buf_.first().resize(buffer_size);
   136             buf_.first().set(0, 0);
   137         }
   138         if (can_write::value && !is_double::value) {
   139             buf_.second().resize(buffer_size);
   140             buf_.second().set(0, 0);
   141         }
   142         dev_.reset(concept_adapter<policy_type>(dev));
   143         flags_ |= f_open;
   144     }
   145 
   146     void close(BOOST_IOS::openmode which = BOOST_IOS::in | BOOST_IOS::out)
   147     {
   148         if (which & BOOST_IOS::in) {
   149             iostreams::close(dev(), BOOST_IOS::in);
   150             flags_ |= f_input_closed;
   151         }
   152         if (which & BOOST_IOS::out) {
   153             buf_.second().flush(dev());
   154             iostreams::close(dev(), BOOST_IOS::out);
   155             flags_ |= f_output_closed;
   156         }
   157         if ( !is_double::value || 
   158              (flags_ & f_input_closed) != 0 && 
   159              (flags_ & f_output_closed) != 0 )
   160         {
   161             dev_.reset();
   162             buf_.first().reset();
   163             buf_.second().reset();
   164             flags_ = 0;
   165         }
   166     }
   167 
   168     bool is_open() const { return (flags_ & f_open) != 0;}
   169 
   170     policy_type& dev() { return **dev_; }
   171 
   172     enum {
   173         f_open             = 1,
   174         f_input_closed     = f_open << 1,
   175         f_output_closed    = f_input_closed << 1
   176     };
   177 
   178     codecvt_holder<Codecvt>  cvt_;
   179     storage_type             dev_;
   180     double_object<
   181         buffer_type, 
   182         is_double
   183     >                        buf_;
   184     int                      flags_;
   185 };
   186 
   187 } // End namespace detail.
   188 
   189 //--------------Definition of converter---------------------------------------//
   190 
   191 #define BOOST_IOSTREAMS_CONVERTER_PARAMS() , int buffer_size = -1
   192 #define BOOST_IOSTREAMS_CONVERTER_ARGS() , buffer_size
   193 
   194 template<typename Device, typename Codecvt, typename Alloc>
   195 struct code_converter_base {
   196     typedef detail::code_converter_impl<
   197                 Device, Codecvt, Alloc
   198             > impl_type;
   199     code_converter_base() : pimpl_(new impl_type) { }
   200     shared_ptr<impl_type> pimpl_;
   201 };
   202 
   203 template< typename Device, 
   204           typename Codecvt = detail::default_codecvt, 
   205           typename Alloc = std::allocator<char> >
   206 class code_converter 
   207     : protected code_converter_base<Device, Codecvt, Alloc>
   208 {
   209 private:
   210     typedef detail::code_converter_impl<
   211                 Device, Codecvt, Alloc
   212             >                                                       impl_type;
   213     typedef typename impl_type::policy_type                         policy_type;
   214     typedef typename impl_type::buffer_type                         buffer_type;
   215     typedef typename detail::codecvt_holder<Codecvt>::codecvt_type  codecvt_type;
   216     typedef typename detail::codecvt_intern<Codecvt>::type          intern_type;
   217     typedef typename detail::codecvt_extern<Codecvt>::type          extern_type;
   218     typedef typename detail::codecvt_state<Codecvt>::type           state_type;
   219 public:
   220     typedef intern_type                                             char_type;    
   221     struct category 
   222         : impl_type::mode, device_tag, closable_tag, localizable_tag
   223         { };
   224     BOOST_STATIC_ASSERT((
   225         is_same<
   226             extern_type, 
   227             BOOST_DEDUCED_TYPENAME char_type_of<Device>::type
   228         >::value
   229     ));
   230 public:
   231     code_converter() { }
   232 #if BOOST_WORKAROUND(__GNUC__, < 3)
   233     code_converter(code_converter& rhs) 
   234         : code_converter_base<Device, Codecvt, Alloc>(rhs)
   235         { }
   236     code_converter(const code_converter& rhs) 
   237         : code_converter_base<Device, Codecvt, Alloc>(rhs)
   238         { }
   239 #endif
   240     BOOST_IOSTREAMS_FORWARD( code_converter, open_impl, Device,
   241                              BOOST_IOSTREAMS_CONVERTER_PARAMS, 
   242                              BOOST_IOSTREAMS_CONVERTER_ARGS )
   243 
   244         // fstream-like interface.
   245 
   246     bool is_open() const { return this->pimpl_->is_open(); }
   247     void close(BOOST_IOS::openmode which = BOOST_IOS::in | BOOST_IOS::out )
   248     { impl().close(which); }
   249 
   250         // Device interface.
   251 
   252     std::streamsize read(char_type*, std::streamsize);
   253     std::streamsize write(const char_type*, std::streamsize);
   254     void imbue(const std::locale& loc) { impl().cvt_.imbue(loc); }
   255 
   256         // Direct device access.
   257 
   258     Device& operator*() { return detail::unwrap_direct(dev()); }
   259     Device* operator->() { return &detail::unwrap_direct(dev()); }
   260 private:
   261     template<typename T> // Used for forwarding.
   262     void open_impl(const T& t BOOST_IOSTREAMS_CONVERTER_PARAMS()) 
   263     { 
   264         impl().open(t BOOST_IOSTREAMS_CONVERTER_ARGS()); 
   265     }
   266 
   267     const codecvt_type& cvt() { return impl().cvt_.get(); }
   268     policy_type& dev() { return impl().dev(); }
   269     buffer_type& in() { return impl().buf_.first(); }
   270     buffer_type& out() { return impl().buf_.second(); }
   271     impl_type& impl() { return *this->pimpl_; }
   272 };
   273 
   274 //--------------Implementation of converter-----------------------------------//
   275 
   276 // Implementation note: if end of stream contains a partial character,
   277 // it is ignored.
   278 template<typename Device, typename Codevt, typename Alloc>
   279 std::streamsize code_converter<Device, Codevt, Alloc>::read
   280     (char_type* s, std::streamsize n)
   281 {
   282     using namespace std;
   283     const extern_type*   next;        // Next external char.
   284     intern_type*         nint;        // Next internal char.
   285     streamsize           total = 0;   // Characters read.
   286     int                  status = iostreams::char_traits<char>::good();
   287     bool                 partial = false;
   288     buffer_type&         buf = in();
   289 
   290     do {
   291 
   292         // Fill buffer.
   293         if (buf.ptr() == buf.eptr() || partial) {
   294             status = buf.fill(dev());
   295             if (buf.ptr() == buf.eptr())
   296                 break;
   297             partial = false;
   298         }
   299 
   300         // Convert.
   301         codecvt_base::result result =
   302             cvt().in( buf.state(),
   303                       buf.ptr(), buf.eptr(), next,
   304                       s + total, s + n, nint );
   305         buf.ptr() += next - buf.ptr();
   306         total = static_cast<streamsize>(nint - s);
   307 
   308         switch (result) {
   309         case codecvt_base::partial:
   310             partial = true;
   311             break;
   312         case codecvt_base::ok:
   313             break;
   314         case codecvt_base::error:
   315             buf.state() = state_type();
   316             throw code_conversion_error();
   317         case codecvt_base::noconv:
   318         default:
   319             buf.state() = state_type();
   320             intern_type c = intern_type();
   321             memcpy(&c, (const void*) buf.ptr(), sizeof(extern_type));
   322             s[total++] = c;
   323             break;
   324         }
   325 
   326     } while (total < n && status != EOF && status != WOULD_BLOCK);
   327 
   328     return total == 0 && status == EOF ? -1 : total;
   329 }
   330 
   331 template<typename Device, typename Codevt, typename Alloc>
   332 std::streamsize code_converter<Device, Codevt, Alloc>::write
   333     (const char_type* s, std::streamsize n)
   334 {
   335     using namespace std;
   336     buffer_type&        buf = out();
   337     extern_type*        next;              // Next external char.
   338     const intern_type*  nint;              // Next internal char.
   339     streamsize          total = 0;         // Characters written.
   340     bool                partial = false;
   341 
   342     while (total < n) {
   343 
   344         // Empty buffer.
   345         if (buf.eptr() == buf.end() || partial) {
   346             if (!buf.flush(dev()))
   347                 break;
   348             partial = false;
   349         }
   350        
   351         // Convert.
   352         codecvt_base::result result =
   353             cvt().out( buf.state(),
   354                        s + total, s + n, nint,
   355                        buf.eptr(), buf.end(), next );
   356         int progress = (int) (next - buf.eptr());
   357         buf.eptr() += progress;
   358 
   359         switch (result) {
   360         case codecvt_base::partial:
   361             partial = true; // Fall through.
   362         case codecvt_base::ok:
   363             total = static_cast<streamsize>(nint - s);
   364             break;
   365         case codecvt_base::error:
   366             buf.state() = state_type();
   367             throw code_conversion_error();
   368         case codecvt_base::noconv:
   369             {
   370                 // This can be shortened to two memcpy's.
   371                 const char* c = (const char*) (s + total);
   372                 for ( std::size_t index = 0; 
   373                       index < sizeof(intern_type); 
   374                       index += sizeof(extern_type), 
   375                       ++buf.ptr() ) 
   376                 {
   377                     memcpy(buf.ptr(), c + index, sizeof(extern_type));
   378                     if (buf.eptr() == buf.end())
   379                         buf.flush(dev());
   380                 }
   381                 ++total;
   382             }
   383         }
   384     }
   385     return total;
   386 }
   387 
   388 //----------------------------------------------------------------------------//
   389 
   390 } } // End namespaces iostreams, boost.
   391 
   392 #include <boost/iostreams/detail/config/enable_warnings.hpp> // Borland 5.x
   393 
   394 #endif // #ifndef BOOST_IOSTREAMS_CODE_CONVERTER_HPP_INCLUDED