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