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