os/ossrv/ossrv_pub/boost_apis/boost/iostreams/restrict.hpp
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/ossrv/ossrv_pub/boost_apis/boost/iostreams/restrict.hpp	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,418 @@
     1.4 +// (C) Copyright Jonathan Turkanis 2005.
     1.5 +// Distributed under the Boost Software License, Version 1.0. (See accompanying
     1.6 +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
     1.7 +
     1.8 +// See http://www.boost.org/libs/iostreams for documentation.
     1.9 +
    1.10 +#ifndef BOOST_IOSTREAMS_RESTRICT_HPP_INCLUDED
    1.11 +#define BOOST_IOSTREAMS_RESTRICT_HPP_INCLUDED
    1.12 +
    1.13 +#if defined(_MSC_VER) && (_MSC_VER >= 1020)
    1.14 +# pragma once
    1.15 +#endif
    1.16 +
    1.17 +#include <algorithm>          // min.
    1.18 +#include <utility>            // pair.
    1.19 +#include <boost/cstdint.hpp>  // intmax_t.
    1.20 +#include <boost/config.hpp>   // DEDUCED_TYPENAME.
    1.21 +#include <boost/iostreams/categories.hpp>
    1.22 +#include <boost/iostreams/char_traits.hpp>
    1.23 +#include <boost/iostreams/detail/adapter/basic_adapter.hpp>
    1.24 +#include <boost/iostreams/detail/call_traits.hpp>
    1.25 +#include <boost/iostreams/detail/enable_if_stream.hpp>
    1.26 +#include <boost/iostreams/detail/error.hpp>
    1.27 +#include <boost/iostreams/detail/ios.hpp>     // failure.
    1.28 +#include <boost/iostreams/detail/select.hpp>
    1.29 +#include <boost/iostreams/operations.hpp>
    1.30 +#include <boost/iostreams/skip.hpp>
    1.31 +#include <boost/iostreams/traits.hpp>         // mode_of, is_direct.
    1.32 +#include <boost/mpl/bool.hpp>
    1.33 +#include <boost/static_assert.hpp>
    1.34 +#include <boost/type_traits/is_convertible.hpp>
    1.35 +
    1.36 +#include <boost/iostreams/detail/config/disable_warnings.hpp> // VC7.1 C4244.
    1.37 +
    1.38 +namespace boost { namespace iostreams {
    1.39 +
    1.40 +namespace detail {
    1.41 +
    1.42 +//
    1.43 +// Template name: restricted_indirect_device.
    1.44 +// Description: Provides an restricted view of an indirect Device.
    1.45 +// Template paramters:
    1.46 +//      Device - An indirect model of Device that models either Source or
    1.47 +//          SeekableDevice.
    1.48 +//
    1.49 +template<typename Device>
    1.50 +class restricted_indirect_device : public basic_adapter<Device> {
    1.51 +private:
    1.52 +    typedef typename detail::param_type<Device>::type  param_type;
    1.53 +public:
    1.54 +    typedef typename char_type_of<Device>::type        char_type;
    1.55 +    struct category
    1.56 +        : mode_of<Device>::type,
    1.57 +          device_tag,
    1.58 +          closable_tag,
    1.59 +          flushable_tag,
    1.60 +          localizable_tag,
    1.61 +          optimally_buffered_tag
    1.62 +        { };
    1.63 +    restricted_indirect_device( param_type dev, stream_offset off,
    1.64 +                                stream_offset len = -1 );
    1.65 +    std::streamsize read(char_type* s, std::streamsize n);
    1.66 +    std::streamsize write(const char_type* s, std::streamsize n);
    1.67 +    stream_offset seek(stream_offset off, BOOST_IOS::seekdir way);
    1.68 +private:
    1.69 +    stream_offset beg_, pos_, end_;
    1.70 +};
    1.71 +
    1.72 +//
    1.73 +// Template name: restricted_direct_device.
    1.74 +// Description: Provides an restricted view of a Direct Device.
    1.75 +// Template paramters:
    1.76 +//      Device - A model of Direct and Device.
    1.77 +//
    1.78 +template<typename Device>
    1.79 +class restricted_direct_device : public basic_adapter<Device> {
    1.80 +public:
    1.81 +    typedef typename char_type_of<Device>::type  char_type;
    1.82 +    typedef std::pair<char_type*, char_type*>    pair_type;
    1.83 +    struct category
    1.84 +        : mode_of<Device>::type,
    1.85 +          device_tag,
    1.86 +          direct_tag,
    1.87 +          closable_tag,
    1.88 +          localizable_tag
    1.89 +        { };
    1.90 +    restricted_direct_device( const Device& dev, stream_offset off,
    1.91 +                              stream_offset len = -1 );
    1.92 +    pair_type input_sequence();
    1.93 +    pair_type output_sequence();
    1.94 +private:
    1.95 +    pair_type sequence(mpl::true_);
    1.96 +    pair_type sequence(mpl::false_);
    1.97 +    char_type *beg_, *end_;
    1.98 +};
    1.99 +
   1.100 +//
   1.101 +// Template name: restricted_filter.
   1.102 +// Description: Provides an restricted view of a Filter.
   1.103 +// Template paramters:
   1.104 +//      Filter - An indirect model of Filter.
   1.105 +//
   1.106 +template<typename Filter>
   1.107 +class restricted_filter : public basic_adapter<Filter> {
   1.108 +public:
   1.109 +    typedef typename char_type_of<Filter>::type char_type;
   1.110 +    struct category
   1.111 +        : mode_of<Filter>::type,
   1.112 +          filter_tag,
   1.113 +          multichar_tag,
   1.114 +          closable_tag,
   1.115 +          localizable_tag,
   1.116 +          optimally_buffered_tag
   1.117 +        { };
   1.118 +    restricted_filter( const Filter& flt, stream_offset off, 
   1.119 +                       stream_offset len = -1 );
   1.120 +
   1.121 +    template<typename Source>
   1.122 +    std::streamsize read(Source& src, char_type* s, std::streamsize n)
   1.123 +    {
   1.124 +        using namespace std;
   1.125 +        if (!open_)
   1.126 +            open(src);
   1.127 +        streamsize amt =
   1.128 +            end_ != -1 ?
   1.129 +                (std::min) (n, static_cast<streamsize>(end_ - pos_)) :
   1.130 +                n;
   1.131 +        streamsize result = iostreams::read(this->component(), src, s, amt);
   1.132 +        if (result != -1)
   1.133 +            pos_ += result;
   1.134 +        return result;
   1.135 +    }
   1.136 +
   1.137 +    template<typename Sink>
   1.138 +    std::streamsize write(Sink& snk, const char_type* s, std::streamsize n)
   1.139 +    {
   1.140 +        if (!open_)
   1.141 +            open(snk);
   1.142 +        if (end_ != -1 && pos_ + n >= end_)
   1.143 +            bad_write();
   1.144 +        std::streamsize result = 
   1.145 +            iostreams::write(this->component(), snk, s, n);
   1.146 +        pos_ += result;
   1.147 +        return result;
   1.148 +    }
   1.149 +
   1.150 +    template<typename Device>
   1.151 +    stream_offset seek(Device& dev, stream_offset off, BOOST_IOS::seekdir way)
   1.152 +    {
   1.153 +        stream_offset next;
   1.154 +        if (way == BOOST_IOS::beg) {
   1.155 +            next = beg_ + off;
   1.156 +        } else if (way == BOOST_IOS::cur) {
   1.157 +            next = pos_ + off;
   1.158 +        } else if (end_ != -1) {
   1.159 +            next = end_ + off;
   1.160 +        } else {
   1.161 +            // Restriction is half-open; seek relative to the actual end.
   1.162 +            pos_ = this->component().seek(dev, off, BOOST_IOS::end);
   1.163 +            if (pos_ < beg_)
   1.164 +                bad_seek();
   1.165 +            return pos_ - beg_;
   1.166 +        }
   1.167 +        if (next < beg_ || end_ != -1 && next >= end_)
   1.168 +            bad_seek();
   1.169 +        pos_ = this->component().seek(dev, next, BOOST_IOS::cur);
   1.170 +        return pos_ - beg_;
   1.171 +    }
   1.172 +private:
   1.173 +    template<typename Device>
   1.174 +    void open(Device& dev)
   1.175 +    {
   1.176 +        open_ = true;
   1.177 +        iostreams::skip(this->component(), dev, beg_);
   1.178 +    }
   1.179 +    stream_offset  beg_, pos_, end_;
   1.180 +    bool           open_;
   1.181 +};
   1.182 +
   1.183 +template<typename T>
   1.184 +struct restriction_traits
   1.185 +    : iostreams::select<  // Disambiguation for Tru64.
   1.186 +          is_filter<T>,  restricted_filter<T>,
   1.187 +          is_direct<T>,  restricted_direct_device<T>,
   1.188 +          else_,         restricted_indirect_device<T>
   1.189 +      >
   1.190 +    { };
   1.191 +
   1.192 +} // End namespace detail.
   1.193 +
   1.194 +template<typename T>
   1.195 +struct restriction : public detail::restriction_traits<T>::type {
   1.196 +    typedef typename detail::param_type<T>::type          param_type;
   1.197 +    typedef typename detail::restriction_traits<T>::type  base_type;
   1.198 +    restriction(param_type t, stream_offset off, stream_offset len = -1)
   1.199 +        : base_type(t, off, len)
   1.200 +        { }
   1.201 +};
   1.202 +
   1.203 +//--------------Implementation of restrict------------------------------------//
   1.204 +
   1.205 +// Note: The following workarounds are patterned after resolve.hpp. It has not
   1.206 +// yet been confirmed that they are necessary.
   1.207 +
   1.208 +#ifndef BOOST_IOSTREAMS_BROKEN_OVERLOAD_RESOLUTION //-------------------------//
   1.209 +# ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //-------------------------------//
   1.210 +
   1.211 +template<typename T>
   1.212 +restriction<T> restrict( const T& t, stream_offset off, stream_offset len = -1
   1.213 +                         BOOST_IOSTREAMS_DISABLE_IF_STREAM(T) )
   1.214 +{ return restriction<T>(t, off, len); }
   1.215 +
   1.216 +template<typename Ch, typename Tr>
   1.217 +restriction< std::basic_streambuf<Ch, Tr> >
   1.218 +restrict(std::basic_streambuf<Ch, Tr>& sb, stream_offset off, stream_offset len = -1)
   1.219 +{ return restriction< std::basic_streambuf<Ch, Tr> >(sb, off, len); }
   1.220 +
   1.221 +template<typename Ch, typename Tr>
   1.222 +restriction< std::basic_istream<Ch, Tr> >
   1.223 +restrict(std::basic_istream<Ch, Tr>& is, stream_offset off, stream_offset len = -1)
   1.224 +{ return restriction< std::basic_istream<Ch, Tr> >(is, off, len); }
   1.225 +
   1.226 +template<typename Ch, typename Tr>
   1.227 +restriction< std::basic_ostream<Ch, Tr> >
   1.228 +restrict(std::basic_ostream<Ch, Tr>& os, stream_offset off, stream_offset len = -1)
   1.229 +{ return restriction< std::basic_ostream<Ch, Tr> >(os, off, len); }
   1.230 +
   1.231 +template<typename Ch, typename Tr>
   1.232 +restriction< std::basic_iostream<Ch, Tr> >
   1.233 +restrict(std::basic_iostream<Ch, Tr>& io, stream_offset off, stream_offset len = -1)
   1.234 +{ return restriction< std::basic_iostream<Ch, Tr> >(io, off, len); }
   1.235 +
   1.236 +# else // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //---------------------//
   1.237 +
   1.238 +template<typename T>
   1.239 +restriction<T> restrict( const T& t, stream_offset off, stream_offset len = -1
   1.240 +                         BOOST_IOSTREAMS_DISABLE_IF_STREAM(T) )
   1.241 +{ return restriction<T>(t, off, len); }
   1.242 +
   1.243 +restriction<std::streambuf> 
   1.244 +restrict(std::streambuf& sb, stream_offset off, stream_offset len = -1)
   1.245 +{ return restriction<std::streambuf>(sb, off, len); }
   1.246 +
   1.247 +restriction<std::istream> 
   1.248 +restrict(std::istream<Ch, Tr>& is, stream_offset off, stream_offset len = -1)
   1.249 +{ return restriction<std::istream>(is, off, len); }
   1.250 +
   1.251 +restriction<std::ostream> 
   1.252 +restrict(std::ostream<Ch, Tr>& os, stream_offset off, stream_offset len = -1)
   1.253 +{ return restriction<std::ostream>(os, off, len); }
   1.254 +
   1.255 +restriction<std::iostream> 
   1.256 +restrict(std::iostream<Ch, Tr>& io, stream_offset off, stream_offset len = -1)
   1.257 +{ return restriction<std::iostream>(io, off, len); }
   1.258 +
   1.259 +# endif // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //--------------------//
   1.260 +#else // #ifndef BOOST_IOSTREAMS_BROKEN_OVERLOAD_RESOLUTION //----------------//
   1.261 +
   1.262 +template<typename T>
   1.263 +restriction<T> 
   1.264 +restrict(const T& t, stream_offset off, stream_offset len, mpl::true_)
   1.265 +{   // Bad overload resolution.
   1.266 +    return restriction<T>(const_cast<T&>(t, off, len));
   1.267 +}
   1.268 +
   1.269 +template<typename T>
   1.270 +restriction<T> 
   1.271 +restrict(const T& t, stream_offset off, stream_offset len, mpl::false_)
   1.272 +{ return restriction<T>(t, off, len); }
   1.273 +
   1.274 +template<typename T>
   1.275 +restriction<T> 
   1.276 +restrict( const T& t, stream_offset off, stream_offset len = -1
   1.277 +          BOOST_IOSTREAMS_DISABLE_IF_STREAM(T) )
   1.278 +{ return restrict(t, off, len, is_std_io<T>()); }
   1.279 +
   1.280 +# if !BOOST_WORKAROUND(__BORLANDC__, < 0x600) && \
   1.281 +     !BOOST_WORKAROUND(BOOST_MSVC, <= 1300) && \
   1.282 +     !defined(__GNUC__) // ---------------------------------------------------//
   1.283 +
   1.284 +template<typename T>
   1.285 +restriction<T>
   1.286 +restrict(T& t, stream_offset off, stream_offset len = -1)
   1.287 +{ return restriction<T>(t, off, len); }
   1.288 +
   1.289 +# endif // Borland 5.x, VC6-7.0 or GCC 2.9x //--------------------------------//
   1.290 +#endif // #ifndef BOOST_IOSTREAMS_BROKEN_OVERLOAD_RESOLUTION //---------------//
   1.291 +//----------------------------------------------------------------------------//
   1.292 +
   1.293 +namespace detail {
   1.294 +
   1.295 +//--------------Implementation of restricted_indirect_device------------------//
   1.296 +
   1.297 +template<typename Device>
   1.298 +restricted_indirect_device<Device>::restricted_indirect_device
   1.299 +    (param_type dev, stream_offset off, stream_offset len)
   1.300 +    : basic_adapter<Device>(dev), beg_(off), pos_(off), 
   1.301 +      end_(len != -1 ? off + len : -1)
   1.302 +{
   1.303 +    if (len < -1 || off < 0)
   1.304 +        throw BOOST_IOSTREAMS_FAILURE("bad offset");
   1.305 +    iostreams::skip(this->component(), off);
   1.306 +}
   1.307 +
   1.308 +template<typename Device>
   1.309 +inline std::streamsize restricted_indirect_device<Device>::read
   1.310 +    (char_type* s, std::streamsize n)
   1.311 +{
   1.312 +    using namespace std;
   1.313 +    streamsize amt =
   1.314 +        end_ != -1 ?
   1.315 +            (std::min) (n, static_cast<streamsize>(end_ - pos_)) :
   1.316 +            n;
   1.317 +    streamsize result = iostreams::read(this->component(), s, amt);
   1.318 +    if (result != -1)
   1.319 +        pos_ += result;
   1.320 +    return result;
   1.321 +}
   1.322 +
   1.323 +template<typename Device>
   1.324 +inline std::streamsize restricted_indirect_device<Device>::write
   1.325 +    (const char_type* s, std::streamsize n)
   1.326 +{
   1.327 +    if (end_ != -1 && pos_ + n >= end_)
   1.328 +        bad_write();
   1.329 +    std::streamsize result = iostreams::write(this->component(), s, n);
   1.330 +    pos_ += result;
   1.331 +    return result;
   1.332 +}
   1.333 +
   1.334 +template<typename Device>
   1.335 +stream_offset restricted_indirect_device<Device>::seek
   1.336 +    (stream_offset off, BOOST_IOS::seekdir way)
   1.337 +{
   1.338 +    stream_offset next;
   1.339 +    if (way == BOOST_IOS::beg) {
   1.340 +        next = beg_ + off;
   1.341 +    } else if (way == BOOST_IOS::cur) {
   1.342 +        next = pos_ + off;
   1.343 +    } else if (end_ != -1) {
   1.344 +        next = end_ + off;
   1.345 +    } else {
   1.346 +        // Restriction is half-open; seek relative to the actual end.
   1.347 +        pos_ = iostreams::seek(this->component(), off, BOOST_IOS::end);
   1.348 +        if (pos_ < beg_)
   1.349 +            bad_seek();
   1.350 +        return pos_ - beg_;
   1.351 +    }
   1.352 +    if (next < beg_ || end_ != -1 && next >= end_)
   1.353 +        bad_seek();
   1.354 +    pos_ = iostreams::seek(this->component(), next - pos_, BOOST_IOS::cur);
   1.355 +    return pos_ - beg_;
   1.356 +}
   1.357 +
   1.358 +//--------------Implementation of restricted_direct_device--------------------//
   1.359 +
   1.360 +template<typename Device>
   1.361 +restricted_direct_device<Device>::restricted_direct_device
   1.362 +    (const Device& dev, stream_offset off, stream_offset len)
   1.363 +    : basic_adapter<Device>(dev), beg_(0), end_(0)
   1.364 +{
   1.365 +    std::pair<char_type*, char_type*> seq =
   1.366 +        sequence(is_convertible<category, input>());
   1.367 +    if ( off < 0 || len < -1 || 
   1.368 +         len != -1 && off + len > seq.second - seq.first )
   1.369 +    {
   1.370 +        throw BOOST_IOSTREAMS_FAILURE("bad offset");
   1.371 +    }
   1.372 +    beg_ = seq.first + off;
   1.373 +    end_ = len != -1 ? 
   1.374 +        seq.first + off + len :
   1.375 +        seq.second;
   1.376 +}
   1.377 +
   1.378 +template<typename Device>
   1.379 +typename restricted_direct_device<Device>::pair_type
   1.380 +restricted_direct_device<Device>::input_sequence()
   1.381 +{
   1.382 +    BOOST_STATIC_ASSERT((is_convertible<category, input>::value));
   1.383 +    return std::make_pair(beg_, end_);
   1.384 +}
   1.385 +
   1.386 +template<typename Device>
   1.387 +typename restricted_direct_device<Device>::pair_type
   1.388 +restricted_direct_device<Device>::output_sequence()
   1.389 +{
   1.390 +    BOOST_STATIC_ASSERT((is_convertible<category, output>::value));
   1.391 +    return std::make_pair(beg_, end_);
   1.392 +}
   1.393 +
   1.394 +template<typename Device>
   1.395 +typename restricted_direct_device<Device>::pair_type
   1.396 +restricted_direct_device<Device>::sequence(mpl::true_)
   1.397 +{ return iostreams::input_sequence(this->component()); }
   1.398 +
   1.399 +template<typename Device>
   1.400 +typename restricted_direct_device<Device>::pair_type
   1.401 +restricted_direct_device<Device>::sequence(mpl::false_)
   1.402 +{ return iostreams::output_sequence(this->component()); }
   1.403 +
   1.404 +//--------------Implementation of restricted_filter---------------------------//
   1.405 +
   1.406 +template<typename Filter>
   1.407 +restricted_filter<Filter>::restricted_filter
   1.408 +    (const Filter& flt, stream_offset off, stream_offset len)
   1.409 +    : basic_adapter<Filter>(flt), beg_(off),
   1.410 +      pos_(off), end_(len != -1 ? off + len : -1), open_(false)
   1.411 +{
   1.412 +    if (len < -1 || off < 0)
   1.413 +        throw BOOST_IOSTREAMS_FAILURE("bad offset");
   1.414 +}
   1.415 +
   1.416 +} // End namespace detail.
   1.417 +
   1.418 +} } // End namespaces iostreams, boost.
   1.419 +
   1.420 +
   1.421 +#endif // #ifndef BOOST_IOSTREAMS_RESTRICT_HPP_INCLUDED