os/ossrv/ossrv_pub/boost_apis/boost/iostreams/restrict.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 2005.
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
#ifndef BOOST_IOSTREAMS_RESTRICT_HPP_INCLUDED
sl@0
     8
#define BOOST_IOSTREAMS_RESTRICT_HPP_INCLUDED
sl@0
     9
sl@0
    10
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
sl@0
    11
# pragma once
sl@0
    12
#endif
sl@0
    13
sl@0
    14
#include <algorithm>          // min.
sl@0
    15
#include <utility>            // pair.
sl@0
    16
#include <boost/cstdint.hpp>  // intmax_t.
sl@0
    17
#include <boost/config.hpp>   // DEDUCED_TYPENAME.
sl@0
    18
#include <boost/iostreams/categories.hpp>
sl@0
    19
#include <boost/iostreams/char_traits.hpp>
sl@0
    20
#include <boost/iostreams/detail/adapter/basic_adapter.hpp>
sl@0
    21
#include <boost/iostreams/detail/call_traits.hpp>
sl@0
    22
#include <boost/iostreams/detail/enable_if_stream.hpp>
sl@0
    23
#include <boost/iostreams/detail/error.hpp>
sl@0
    24
#include <boost/iostreams/detail/ios.hpp>     // failure.
sl@0
    25
#include <boost/iostreams/detail/select.hpp>
sl@0
    26
#include <boost/iostreams/operations.hpp>
sl@0
    27
#include <boost/iostreams/skip.hpp>
sl@0
    28
#include <boost/iostreams/traits.hpp>         // mode_of, is_direct.
sl@0
    29
#include <boost/mpl/bool.hpp>
sl@0
    30
#include <boost/static_assert.hpp>
sl@0
    31
#include <boost/type_traits/is_convertible.hpp>
sl@0
    32
sl@0
    33
#include <boost/iostreams/detail/config/disable_warnings.hpp> // VC7.1 C4244.
sl@0
    34
sl@0
    35
namespace boost { namespace iostreams {
sl@0
    36
sl@0
    37
namespace detail {
sl@0
    38
sl@0
    39
//
sl@0
    40
// Template name: restricted_indirect_device.
sl@0
    41
// Description: Provides an restricted view of an indirect Device.
sl@0
    42
// Template paramters:
sl@0
    43
//      Device - An indirect model of Device that models either Source or
sl@0
    44
//          SeekableDevice.
sl@0
    45
//
sl@0
    46
template<typename Device>
sl@0
    47
class restricted_indirect_device : public basic_adapter<Device> {
sl@0
    48
private:
sl@0
    49
    typedef typename detail::param_type<Device>::type  param_type;
sl@0
    50
public:
sl@0
    51
    typedef typename char_type_of<Device>::type        char_type;
sl@0
    52
    struct category
sl@0
    53
        : mode_of<Device>::type,
sl@0
    54
          device_tag,
sl@0
    55
          closable_tag,
sl@0
    56
          flushable_tag,
sl@0
    57
          localizable_tag,
sl@0
    58
          optimally_buffered_tag
sl@0
    59
        { };
sl@0
    60
    restricted_indirect_device( param_type dev, stream_offset off,
sl@0
    61
                                stream_offset len = -1 );
sl@0
    62
    std::streamsize read(char_type* s, std::streamsize n);
sl@0
    63
    std::streamsize write(const char_type* s, std::streamsize n);
sl@0
    64
    stream_offset seek(stream_offset off, BOOST_IOS::seekdir way);
sl@0
    65
private:
sl@0
    66
    stream_offset beg_, pos_, end_;
sl@0
    67
};
sl@0
    68
sl@0
    69
//
sl@0
    70
// Template name: restricted_direct_device.
sl@0
    71
// Description: Provides an restricted view of a Direct Device.
sl@0
    72
// Template paramters:
sl@0
    73
//      Device - A model of Direct and Device.
sl@0
    74
//
sl@0
    75
template<typename Device>
sl@0
    76
class restricted_direct_device : public basic_adapter<Device> {
sl@0
    77
public:
sl@0
    78
    typedef typename char_type_of<Device>::type  char_type;
sl@0
    79
    typedef std::pair<char_type*, char_type*>    pair_type;
sl@0
    80
    struct category
sl@0
    81
        : mode_of<Device>::type,
sl@0
    82
          device_tag,
sl@0
    83
          direct_tag,
sl@0
    84
          closable_tag,
sl@0
    85
          localizable_tag
sl@0
    86
        { };
sl@0
    87
    restricted_direct_device( const Device& dev, stream_offset off,
sl@0
    88
                              stream_offset len = -1 );
sl@0
    89
    pair_type input_sequence();
sl@0
    90
    pair_type output_sequence();
sl@0
    91
private:
sl@0
    92
    pair_type sequence(mpl::true_);
sl@0
    93
    pair_type sequence(mpl::false_);
sl@0
    94
    char_type *beg_, *end_;
sl@0
    95
};
sl@0
    96
sl@0
    97
//
sl@0
    98
// Template name: restricted_filter.
sl@0
    99
// Description: Provides an restricted view of a Filter.
sl@0
   100
// Template paramters:
sl@0
   101
//      Filter - An indirect model of Filter.
sl@0
   102
//
sl@0
   103
template<typename Filter>
sl@0
   104
class restricted_filter : public basic_adapter<Filter> {
sl@0
   105
public:
sl@0
   106
    typedef typename char_type_of<Filter>::type char_type;
sl@0
   107
    struct category
sl@0
   108
        : mode_of<Filter>::type,
sl@0
   109
          filter_tag,
sl@0
   110
          multichar_tag,
sl@0
   111
          closable_tag,
sl@0
   112
          localizable_tag,
sl@0
   113
          optimally_buffered_tag
sl@0
   114
        { };
sl@0
   115
    restricted_filter( const Filter& flt, stream_offset off, 
sl@0
   116
                       stream_offset len = -1 );
sl@0
   117
sl@0
   118
    template<typename Source>
sl@0
   119
    std::streamsize read(Source& src, char_type* s, std::streamsize n)
sl@0
   120
    {
sl@0
   121
        using namespace std;
sl@0
   122
        if (!open_)
sl@0
   123
            open(src);
sl@0
   124
        streamsize amt =
sl@0
   125
            end_ != -1 ?
sl@0
   126
                (std::min) (n, static_cast<streamsize>(end_ - pos_)) :
sl@0
   127
                n;
sl@0
   128
        streamsize result = iostreams::read(this->component(), src, s, amt);
sl@0
   129
        if (result != -1)
sl@0
   130
            pos_ += result;
sl@0
   131
        return result;
sl@0
   132
    }
sl@0
   133
sl@0
   134
    template<typename Sink>
sl@0
   135
    std::streamsize write(Sink& snk, const char_type* s, std::streamsize n)
sl@0
   136
    {
sl@0
   137
        if (!open_)
sl@0
   138
            open(snk);
sl@0
   139
        if (end_ != -1 && pos_ + n >= end_)
sl@0
   140
            bad_write();
sl@0
   141
        std::streamsize result = 
sl@0
   142
            iostreams::write(this->component(), snk, s, n);
sl@0
   143
        pos_ += result;
sl@0
   144
        return result;
sl@0
   145
    }
sl@0
   146
sl@0
   147
    template<typename Device>
sl@0
   148
    stream_offset seek(Device& dev, stream_offset off, BOOST_IOS::seekdir way)
sl@0
   149
    {
sl@0
   150
        stream_offset next;
sl@0
   151
        if (way == BOOST_IOS::beg) {
sl@0
   152
            next = beg_ + off;
sl@0
   153
        } else if (way == BOOST_IOS::cur) {
sl@0
   154
            next = pos_ + off;
sl@0
   155
        } else if (end_ != -1) {
sl@0
   156
            next = end_ + off;
sl@0
   157
        } else {
sl@0
   158
            // Restriction is half-open; seek relative to the actual end.
sl@0
   159
            pos_ = this->component().seek(dev, off, BOOST_IOS::end);
sl@0
   160
            if (pos_ < beg_)
sl@0
   161
                bad_seek();
sl@0
   162
            return pos_ - beg_;
sl@0
   163
        }
sl@0
   164
        if (next < beg_ || end_ != -1 && next >= end_)
sl@0
   165
            bad_seek();
sl@0
   166
        pos_ = this->component().seek(dev, next, BOOST_IOS::cur);
sl@0
   167
        return pos_ - beg_;
sl@0
   168
    }
sl@0
   169
private:
sl@0
   170
    template<typename Device>
sl@0
   171
    void open(Device& dev)
sl@0
   172
    {
sl@0
   173
        open_ = true;
sl@0
   174
        iostreams::skip(this->component(), dev, beg_);
sl@0
   175
    }
sl@0
   176
    stream_offset  beg_, pos_, end_;
sl@0
   177
    bool           open_;
sl@0
   178
};
sl@0
   179
sl@0
   180
template<typename T>
sl@0
   181
struct restriction_traits
sl@0
   182
    : iostreams::select<  // Disambiguation for Tru64.
sl@0
   183
          is_filter<T>,  restricted_filter<T>,
sl@0
   184
          is_direct<T>,  restricted_direct_device<T>,
sl@0
   185
          else_,         restricted_indirect_device<T>
sl@0
   186
      >
sl@0
   187
    { };
sl@0
   188
sl@0
   189
} // End namespace detail.
sl@0
   190
sl@0
   191
template<typename T>
sl@0
   192
struct restriction : public detail::restriction_traits<T>::type {
sl@0
   193
    typedef typename detail::param_type<T>::type          param_type;
sl@0
   194
    typedef typename detail::restriction_traits<T>::type  base_type;
sl@0
   195
    restriction(param_type t, stream_offset off, stream_offset len = -1)
sl@0
   196
        : base_type(t, off, len)
sl@0
   197
        { }
sl@0
   198
};
sl@0
   199
sl@0
   200
//--------------Implementation of restrict------------------------------------//
sl@0
   201
sl@0
   202
// Note: The following workarounds are patterned after resolve.hpp. It has not
sl@0
   203
// yet been confirmed that they are necessary.
sl@0
   204
sl@0
   205
#ifndef BOOST_IOSTREAMS_BROKEN_OVERLOAD_RESOLUTION //-------------------------//
sl@0
   206
# ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //-------------------------------//
sl@0
   207
sl@0
   208
template<typename T>
sl@0
   209
restriction<T> restrict( const T& t, stream_offset off, stream_offset len = -1
sl@0
   210
                         BOOST_IOSTREAMS_DISABLE_IF_STREAM(T) )
sl@0
   211
{ return restriction<T>(t, off, len); }
sl@0
   212
sl@0
   213
template<typename Ch, typename Tr>
sl@0
   214
restriction< std::basic_streambuf<Ch, Tr> >
sl@0
   215
restrict(std::basic_streambuf<Ch, Tr>& sb, stream_offset off, stream_offset len = -1)
sl@0
   216
{ return restriction< std::basic_streambuf<Ch, Tr> >(sb, off, len); }
sl@0
   217
sl@0
   218
template<typename Ch, typename Tr>
sl@0
   219
restriction< std::basic_istream<Ch, Tr> >
sl@0
   220
restrict(std::basic_istream<Ch, Tr>& is, stream_offset off, stream_offset len = -1)
sl@0
   221
{ return restriction< std::basic_istream<Ch, Tr> >(is, off, len); }
sl@0
   222
sl@0
   223
template<typename Ch, typename Tr>
sl@0
   224
restriction< std::basic_ostream<Ch, Tr> >
sl@0
   225
restrict(std::basic_ostream<Ch, Tr>& os, stream_offset off, stream_offset len = -1)
sl@0
   226
{ return restriction< std::basic_ostream<Ch, Tr> >(os, off, len); }
sl@0
   227
sl@0
   228
template<typename Ch, typename Tr>
sl@0
   229
restriction< std::basic_iostream<Ch, Tr> >
sl@0
   230
restrict(std::basic_iostream<Ch, Tr>& io, stream_offset off, stream_offset len = -1)
sl@0
   231
{ return restriction< std::basic_iostream<Ch, Tr> >(io, off, len); }
sl@0
   232
sl@0
   233
# else // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //---------------------//
sl@0
   234
sl@0
   235
template<typename T>
sl@0
   236
restriction<T> restrict( const T& t, stream_offset off, stream_offset len = -1
sl@0
   237
                         BOOST_IOSTREAMS_DISABLE_IF_STREAM(T) )
sl@0
   238
{ return restriction<T>(t, off, len); }
sl@0
   239
sl@0
   240
restriction<std::streambuf> 
sl@0
   241
restrict(std::streambuf& sb, stream_offset off, stream_offset len = -1)
sl@0
   242
{ return restriction<std::streambuf>(sb, off, len); }
sl@0
   243
sl@0
   244
restriction<std::istream> 
sl@0
   245
restrict(std::istream<Ch, Tr>& is, stream_offset off, stream_offset len = -1)
sl@0
   246
{ return restriction<std::istream>(is, off, len); }
sl@0
   247
sl@0
   248
restriction<std::ostream> 
sl@0
   249
restrict(std::ostream<Ch, Tr>& os, stream_offset off, stream_offset len = -1)
sl@0
   250
{ return restriction<std::ostream>(os, off, len); }
sl@0
   251
sl@0
   252
restriction<std::iostream> 
sl@0
   253
restrict(std::iostream<Ch, Tr>& io, stream_offset off, stream_offset len = -1)
sl@0
   254
{ return restriction<std::iostream>(io, off, len); }
sl@0
   255
sl@0
   256
# endif // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //--------------------//
sl@0
   257
#else // #ifndef BOOST_IOSTREAMS_BROKEN_OVERLOAD_RESOLUTION //----------------//
sl@0
   258
sl@0
   259
template<typename T>
sl@0
   260
restriction<T> 
sl@0
   261
restrict(const T& t, stream_offset off, stream_offset len, mpl::true_)
sl@0
   262
{   // Bad overload resolution.
sl@0
   263
    return restriction<T>(const_cast<T&>(t, off, len));
sl@0
   264
}
sl@0
   265
sl@0
   266
template<typename T>
sl@0
   267
restriction<T> 
sl@0
   268
restrict(const T& t, stream_offset off, stream_offset len, mpl::false_)
sl@0
   269
{ return restriction<T>(t, off, len); }
sl@0
   270
sl@0
   271
template<typename T>
sl@0
   272
restriction<T> 
sl@0
   273
restrict( const T& t, stream_offset off, stream_offset len = -1
sl@0
   274
          BOOST_IOSTREAMS_DISABLE_IF_STREAM(T) )
sl@0
   275
{ return restrict(t, off, len, is_std_io<T>()); }
sl@0
   276
sl@0
   277
# if !BOOST_WORKAROUND(__BORLANDC__, < 0x600) && \
sl@0
   278
     !BOOST_WORKAROUND(BOOST_MSVC, <= 1300) && \
sl@0
   279
     !defined(__GNUC__) // ---------------------------------------------------//
sl@0
   280
sl@0
   281
template<typename T>
sl@0
   282
restriction<T>
sl@0
   283
restrict(T& t, stream_offset off, stream_offset len = -1)
sl@0
   284
{ return restriction<T>(t, off, len); }
sl@0
   285
sl@0
   286
# endif // Borland 5.x, VC6-7.0 or GCC 2.9x //--------------------------------//
sl@0
   287
#endif // #ifndef BOOST_IOSTREAMS_BROKEN_OVERLOAD_RESOLUTION //---------------//
sl@0
   288
//----------------------------------------------------------------------------//
sl@0
   289
sl@0
   290
namespace detail {
sl@0
   291
sl@0
   292
//--------------Implementation of restricted_indirect_device------------------//
sl@0
   293
sl@0
   294
template<typename Device>
sl@0
   295
restricted_indirect_device<Device>::restricted_indirect_device
sl@0
   296
    (param_type dev, stream_offset off, stream_offset len)
sl@0
   297
    : basic_adapter<Device>(dev), beg_(off), pos_(off), 
sl@0
   298
      end_(len != -1 ? off + len : -1)
sl@0
   299
{
sl@0
   300
    if (len < -1 || off < 0)
sl@0
   301
        throw BOOST_IOSTREAMS_FAILURE("bad offset");
sl@0
   302
    iostreams::skip(this->component(), off);
sl@0
   303
}
sl@0
   304
sl@0
   305
template<typename Device>
sl@0
   306
inline std::streamsize restricted_indirect_device<Device>::read
sl@0
   307
    (char_type* s, std::streamsize n)
sl@0
   308
{
sl@0
   309
    using namespace std;
sl@0
   310
    streamsize amt =
sl@0
   311
        end_ != -1 ?
sl@0
   312
            (std::min) (n, static_cast<streamsize>(end_ - pos_)) :
sl@0
   313
            n;
sl@0
   314
    streamsize result = iostreams::read(this->component(), s, amt);
sl@0
   315
    if (result != -1)
sl@0
   316
        pos_ += result;
sl@0
   317
    return result;
sl@0
   318
}
sl@0
   319
sl@0
   320
template<typename Device>
sl@0
   321
inline std::streamsize restricted_indirect_device<Device>::write
sl@0
   322
    (const char_type* s, std::streamsize n)
sl@0
   323
{
sl@0
   324
    if (end_ != -1 && pos_ + n >= end_)
sl@0
   325
        bad_write();
sl@0
   326
    std::streamsize result = iostreams::write(this->component(), s, n);
sl@0
   327
    pos_ += result;
sl@0
   328
    return result;
sl@0
   329
}
sl@0
   330
sl@0
   331
template<typename Device>
sl@0
   332
stream_offset restricted_indirect_device<Device>::seek
sl@0
   333
    (stream_offset off, BOOST_IOS::seekdir way)
sl@0
   334
{
sl@0
   335
    stream_offset next;
sl@0
   336
    if (way == BOOST_IOS::beg) {
sl@0
   337
        next = beg_ + off;
sl@0
   338
    } else if (way == BOOST_IOS::cur) {
sl@0
   339
        next = pos_ + off;
sl@0
   340
    } else if (end_ != -1) {
sl@0
   341
        next = end_ + off;
sl@0
   342
    } else {
sl@0
   343
        // Restriction is half-open; seek relative to the actual end.
sl@0
   344
        pos_ = iostreams::seek(this->component(), off, BOOST_IOS::end);
sl@0
   345
        if (pos_ < beg_)
sl@0
   346
            bad_seek();
sl@0
   347
        return pos_ - beg_;
sl@0
   348
    }
sl@0
   349
    if (next < beg_ || end_ != -1 && next >= end_)
sl@0
   350
        bad_seek();
sl@0
   351
    pos_ = iostreams::seek(this->component(), next - pos_, BOOST_IOS::cur);
sl@0
   352
    return pos_ - beg_;
sl@0
   353
}
sl@0
   354
sl@0
   355
//--------------Implementation of restricted_direct_device--------------------//
sl@0
   356
sl@0
   357
template<typename Device>
sl@0
   358
restricted_direct_device<Device>::restricted_direct_device
sl@0
   359
    (const Device& dev, stream_offset off, stream_offset len)
sl@0
   360
    : basic_adapter<Device>(dev), beg_(0), end_(0)
sl@0
   361
{
sl@0
   362
    std::pair<char_type*, char_type*> seq =
sl@0
   363
        sequence(is_convertible<category, input>());
sl@0
   364
    if ( off < 0 || len < -1 || 
sl@0
   365
         len != -1 && off + len > seq.second - seq.first )
sl@0
   366
    {
sl@0
   367
        throw BOOST_IOSTREAMS_FAILURE("bad offset");
sl@0
   368
    }
sl@0
   369
    beg_ = seq.first + off;
sl@0
   370
    end_ = len != -1 ? 
sl@0
   371
        seq.first + off + len :
sl@0
   372
        seq.second;
sl@0
   373
}
sl@0
   374
sl@0
   375
template<typename Device>
sl@0
   376
typename restricted_direct_device<Device>::pair_type
sl@0
   377
restricted_direct_device<Device>::input_sequence()
sl@0
   378
{
sl@0
   379
    BOOST_STATIC_ASSERT((is_convertible<category, input>::value));
sl@0
   380
    return std::make_pair(beg_, end_);
sl@0
   381
}
sl@0
   382
sl@0
   383
template<typename Device>
sl@0
   384
typename restricted_direct_device<Device>::pair_type
sl@0
   385
restricted_direct_device<Device>::output_sequence()
sl@0
   386
{
sl@0
   387
    BOOST_STATIC_ASSERT((is_convertible<category, output>::value));
sl@0
   388
    return std::make_pair(beg_, end_);
sl@0
   389
}
sl@0
   390
sl@0
   391
template<typename Device>
sl@0
   392
typename restricted_direct_device<Device>::pair_type
sl@0
   393
restricted_direct_device<Device>::sequence(mpl::true_)
sl@0
   394
{ return iostreams::input_sequence(this->component()); }
sl@0
   395
sl@0
   396
template<typename Device>
sl@0
   397
typename restricted_direct_device<Device>::pair_type
sl@0
   398
restricted_direct_device<Device>::sequence(mpl::false_)
sl@0
   399
{ return iostreams::output_sequence(this->component()); }
sl@0
   400
sl@0
   401
//--------------Implementation of restricted_filter---------------------------//
sl@0
   402
sl@0
   403
template<typename Filter>
sl@0
   404
restricted_filter<Filter>::restricted_filter
sl@0
   405
    (const Filter& flt, stream_offset off, stream_offset len)
sl@0
   406
    : basic_adapter<Filter>(flt), beg_(off),
sl@0
   407
      pos_(off), end_(len != -1 ? off + len : -1), open_(false)
sl@0
   408
{
sl@0
   409
    if (len < -1 || off < 0)
sl@0
   410
        throw BOOST_IOSTREAMS_FAILURE("bad offset");
sl@0
   411
}
sl@0
   412
sl@0
   413
} // End namespace detail.
sl@0
   414
sl@0
   415
} } // End namespaces iostreams, boost.
sl@0
   416
sl@0
   417
sl@0
   418
#endif // #ifndef BOOST_IOSTREAMS_RESTRICT_HPP_INCLUDED