os/ossrv/ossrv_pub/boost_apis/boost/iostreams/chain.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
#ifndef BOOST_IOSTREAMS_DETAIL_CHAIN_HPP_INCLUDED
sl@0
     8
#define BOOST_IOSTREAMS_DETAIL_CHAIN_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>                            // for_each.
sl@0
    15
#include <cassert>
sl@0
    16
#include <exception>
sl@0
    17
#include <functional>                           // unary_function.
sl@0
    18
#include <iterator>                             // advance.
sl@0
    19
#include <list>
sl@0
    20
#include <memory>                               // allocator, auto_ptr.
sl@0
    21
#include <typeinfo>
sl@0
    22
#include <stdexcept>                            // logic_error, out_of_range.
sl@0
    23
#include <boost/checked_delete.hpp>
sl@0
    24
#include <boost/config.hpp>                     // BOOST_MSVC, template friends,
sl@0
    25
#include <boost/detail/workaround.hpp>          // BOOST_NESTED_TEMPLATE 
sl@0
    26
#include <boost/iostreams/constants.hpp>
sl@0
    27
#include <boost/iostreams/detail/access_control.hpp>
sl@0
    28
#include <boost/iostreams/detail/char_traits.hpp>
sl@0
    29
#include <boost/iostreams/detail/push.hpp>
sl@0
    30
#include <boost/iostreams/detail/streambuf.hpp> // pubsync.
sl@0
    31
#include <boost/iostreams/detail/wrap_unwrap.hpp>
sl@0
    32
#include <boost/iostreams/device/null.hpp>
sl@0
    33
#include <boost/iostreams/positioning.hpp>
sl@0
    34
#include <boost/iostreams/traits.hpp>           // is_filter.
sl@0
    35
#include <boost/iostreams/stream_buffer.hpp>
sl@0
    36
#include <boost/next_prior.hpp>
sl@0
    37
#include <boost/shared_ptr.hpp>
sl@0
    38
#include <boost/static_assert.hpp>
sl@0
    39
#include <boost/type_traits/is_convertible.hpp>
sl@0
    40
#include <boost/type.hpp>
sl@0
    41
#if BOOST_WORKAROUND(BOOST_MSVC, < 1310)
sl@0
    42
# include <boost/mpl/int.hpp>
sl@0
    43
#endif
sl@0
    44
sl@0
    45
// Sometimes type_info objects must be compared by name. Borrowed from
sl@0
    46
// Boost.Python and Boost.Function.
sl@0
    47
#if (defined(__GNUC__) && __GNUC__ >= 3) || \
sl@0
    48
     defined(_AIX) || \
sl@0
    49
    (defined(__sgi) && defined(__host_mips)) || \
sl@0
    50
    (defined(linux) && defined(__INTEL_COMPILER) && defined(__ICC)) \
sl@0
    51
    /**/
sl@0
    52
# include <cstring>
sl@0
    53
# define BOOST_IOSTREAMS_COMPARE_TYPE_ID(X,Y) \
sl@0
    54
     (std::strcmp((X).name(),(Y).name()) == 0)
sl@0
    55
#else
sl@0
    56
# define BOOST_IOSTREAMS_COMPARE_TYPE_ID(X,Y) ((X)==(Y))
sl@0
    57
#endif
sl@0
    58
sl@0
    59
// Deprecated
sl@0
    60
#define BOOST_IOSTREAMS_COMPONENT_TYPE(chain, index) \
sl@0
    61
    chain.component_type( index ) \
sl@0
    62
    /**/
sl@0
    63
sl@0
    64
#if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
sl@0
    65
# define BOOST_IOSTREAMS_COMPONENT(chain, index, target) \
sl@0
    66
    chain.component< target >( index ) \
sl@0
    67
    /**/
sl@0
    68
#else
sl@0
    69
# define BOOST_IOSTREAMS_COMPONENT(chain, index, target) \
sl@0
    70
    chain.component( index, ::boost::type< target >() ) \
sl@0
    71
    /**/
sl@0
    72
#endif
sl@0
    73
sl@0
    74
namespace boost { namespace iostreams {
sl@0
    75
sl@0
    76
//--------------Definition of chain and wchain--------------------------------//
sl@0
    77
sl@0
    78
namespace detail {
sl@0
    79
sl@0
    80
template<typename Chain> class chain_client;
sl@0
    81
sl@0
    82
//
sl@0
    83
// Concept name: Chain.
sl@0
    84
// Description: Represents a chain of stream buffers which provides access
sl@0
    85
//     to the first buffer in the chain and send notifications when the
sl@0
    86
//     streambufs are added to or removed from chain.
sl@0
    87
// Refines: Closable device with mode equal to typename Chain::mode.
sl@0
    88
// Models: chain, converting_chain.
sl@0
    89
// Example:
sl@0
    90
//
sl@0
    91
//    class chain {
sl@0
    92
//    public:
sl@0
    93
//        typedef xxx chain_type;
sl@0
    94
//        typedef xxx client_type;
sl@0
    95
//        typedef xxx mode;
sl@0
    96
//        bool is_complete() const;                  // Ready for i/o.
sl@0
    97
//        template<typename T>
sl@0
    98
//        void push( const T& t,                     // Adds a stream buffer to
sl@0
    99
//                   streamsize,                     // chain, based on t, with
sl@0
   100
//                   streamsize );                   // given buffer and putback
sl@0
   101
//                                                   // buffer sizes. Pass -1 to
sl@0
   102
//                                                   // request default size.
sl@0
   103
//    protected:
sl@0
   104
//        void register_client(client_type* client); // Associate client.
sl@0
   105
//        void notify();                             // Notify client.
sl@0
   106
//    };
sl@0
   107
//
sl@0
   108
sl@0
   109
//
sl@0
   110
// Description: Represents a chain of filters with an optional device at the
sl@0
   111
//      end.
sl@0
   112
// Template parameters:
sl@0
   113
//      Self - A class deriving from the current instantiation of this template.
sl@0
   114
//          This is an example of the Curiously Recurring Template Pattern.
sl@0
   115
//      Ch - The character type.
sl@0
   116
//      Tr - The character traits type.
sl@0
   117
//      Alloc - The allocator type.
sl@0
   118
//      Mode - A mode tag.
sl@0
   119
//
sl@0
   120
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   121
class chain_base {
sl@0
   122
public:
sl@0
   123
    typedef Ch                                     char_type;
sl@0
   124
    BOOST_IOSTREAMS_STREAMBUF_TYPEDEFS(Tr)
sl@0
   125
    typedef Alloc                                  allocator_type;
sl@0
   126
    typedef Mode                                   mode;
sl@0
   127
    struct category
sl@0
   128
        : Mode,
sl@0
   129
          device_tag
sl@0
   130
        { };
sl@0
   131
    typedef chain_client<Self>                     client_type;
sl@0
   132
    friend class chain_client<Self>;
sl@0
   133
private:
sl@0
   134
    typedef linked_streambuf<Ch>                   streambuf_type;
sl@0
   135
    typedef std::list<streambuf_type*>             list_type;
sl@0
   136
    typedef chain_base<Self, Ch, Tr, Alloc, Mode>  my_type;
sl@0
   137
protected:
sl@0
   138
    chain_base() : pimpl_(new chain_impl) { }
sl@0
   139
    chain_base(const chain_base& rhs): pimpl_(rhs.pimpl_) { }
sl@0
   140
public:
sl@0
   141
sl@0
   142
    //----------Buffer sizing-------------------------------------------------//
sl@0
   143
sl@0
   144
    // Sets the size of the buffer created for the devices to be added to this
sl@0
   145
    // chain. Does not affect the size of the buffer for devices already
sl@0
   146
    // added.
sl@0
   147
    void set_device_buffer_size(int n) { pimpl_->device_buffer_size_ = n; }
sl@0
   148
sl@0
   149
    // Sets the size of the buffer created for the filters to be added
sl@0
   150
    // to this chain. Does not affect the size of the buffer for filters already
sl@0
   151
    // added.
sl@0
   152
    void set_filter_buffer_size(int n) { pimpl_->filter_buffer_size_ = n; }
sl@0
   153
sl@0
   154
    // Sets the size of the putback buffer for filters and devices to be added
sl@0
   155
    // to this chain. Does not affect the size of the buffer for filters or
sl@0
   156
    // devices already added.
sl@0
   157
    void set_pback_size(int n) { pimpl_->pback_size_ = n; }
sl@0
   158
sl@0
   159
    //----------Device interface----------------------------------------------//
sl@0
   160
sl@0
   161
    std::streamsize read(char_type* s, std::streamsize n);
sl@0
   162
    std::streamsize write(const char_type* s, std::streamsize n);
sl@0
   163
    std::streampos seek(stream_offset off, BOOST_IOS::seekdir way);
sl@0
   164
sl@0
   165
    //----------Direct component access---------------------------------------//
sl@0
   166
sl@0
   167
    const std::type_info& component_type(int n) const
sl@0
   168
    {
sl@0
   169
        if (static_cast<size_type>(n) >= size())
sl@0
   170
            throw std::out_of_range("bad chain offset");
sl@0
   171
        return (*boost::next(list().begin(), n))->component_type();
sl@0
   172
    }
sl@0
   173
sl@0
   174
#if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
sl@0
   175
    // Deprecated.
sl@0
   176
    template<int N>
sl@0
   177
    const std::type_info& component_type() const { return component_type(N); }
sl@0
   178
sl@0
   179
    template<typename T>
sl@0
   180
    T* component(int n) const { return component(n, boost::type<T>()); }
sl@0
   181
sl@0
   182
    // Deprecated.
sl@0
   183
    template<int N, typename T> 
sl@0
   184
    T* component() const { return component<T>(N); }
sl@0
   185
#endif
sl@0
   186
sl@0
   187
#if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
sl@0
   188
    private:
sl@0
   189
#endif
sl@0
   190
    template<typename T>
sl@0
   191
    T* component(int n, boost::type<T>) const
sl@0
   192
    {
sl@0
   193
        if (static_cast<size_type>(n) >= size())
sl@0
   194
            throw std::out_of_range("bad chain offset");
sl@0
   195
        streambuf_type* link = *boost::next(list().begin(), n);
sl@0
   196
        if (BOOST_IOSTREAMS_COMPARE_TYPE_ID(link->component_type(), typeid(T)))
sl@0
   197
            return static_cast<T*>(link->component_impl());
sl@0
   198
        else
sl@0
   199
            return 0;
sl@0
   200
    }
sl@0
   201
public:
sl@0
   202
sl@0
   203
    //----------Container-like interface--------------------------------------//
sl@0
   204
sl@0
   205
    typedef typename list_type::size_type size_type;
sl@0
   206
    streambuf_type& front() { return *list().front(); }
sl@0
   207
    BOOST_IOSTREAMS_DEFINE_PUSH(push, mode, char_type, push_impl)
sl@0
   208
    void pop();
sl@0
   209
    bool empty() const { return list().empty(); }
sl@0
   210
    size_type size() const { return list().size(); }
sl@0
   211
    void reset();
sl@0
   212
sl@0
   213
    //----------Additional i/o functions--------------------------------------//
sl@0
   214
sl@0
   215
    // Returns true if this chain is non-empty and its final link
sl@0
   216
    // is a source or sink, i.e., if it is ready to perform i/o.
sl@0
   217
    bool is_complete() const;
sl@0
   218
    bool auto_close() const;
sl@0
   219
    void set_auto_close(bool close);
sl@0
   220
    bool sync() { return front().BOOST_IOSTREAMS_PUBSYNC() != -1; }
sl@0
   221
    bool strict_sync();
sl@0
   222
private:
sl@0
   223
    template<typename T>
sl@0
   224
    void push_impl(const T& t, int buffer_size = -1, int pback_size = -1)
sl@0
   225
    {
sl@0
   226
        typedef typename iostreams::category_of<T>::type  category;
sl@0
   227
        typedef typename unwrap_ios<T>::type              policy_type;
sl@0
   228
        typedef stream_buffer<
sl@0
   229
                    policy_type,
sl@0
   230
                    BOOST_IOSTREAMS_CHAR_TRAITS(char_type),
sl@0
   231
                    Alloc, Mode
sl@0
   232
                >                                         facade_type;
sl@0
   233
        BOOST_STATIC_ASSERT((is_convertible<category, Mode>::value));
sl@0
   234
        if (is_complete())
sl@0
   235
            throw std::logic_error("chain complete");
sl@0
   236
        streambuf_type* prev = !empty() ? list().back() : 0;
sl@0
   237
        buffer_size =
sl@0
   238
            buffer_size != -1 ?
sl@0
   239
                buffer_size :
sl@0
   240
                iostreams::optimal_buffer_size(t);
sl@0
   241
        pback_size =
sl@0
   242
            pback_size != -1 ?
sl@0
   243
                pback_size :
sl@0
   244
                pimpl_->pback_size_;
sl@0
   245
        std::auto_ptr<facade_type>
sl@0
   246
            buf(new facade_type(t, buffer_size, pback_size));
sl@0
   247
        list().push_back(buf.get());
sl@0
   248
        buf.release();
sl@0
   249
        if (is_device<policy_type>::value)
sl@0
   250
            pimpl_->flags_ |= f_complete | f_open;
sl@0
   251
        if (prev) prev->set_next(list().back());
sl@0
   252
        notify();
sl@0
   253
    }
sl@0
   254
sl@0
   255
    list_type& list() { return pimpl_->links_; }
sl@0
   256
    const list_type& list() const { return pimpl_->links_; }
sl@0
   257
    void register_client(client_type* client) { pimpl_->client_ = client; }
sl@0
   258
    void notify() { if (pimpl_->client_) pimpl_->client_->notify(); }
sl@0
   259
sl@0
   260
    //----------Nested classes------------------------------------------------//
sl@0
   261
sl@0
   262
    static void close(streambuf_type* b, BOOST_IOS::openmode m)
sl@0
   263
    {
sl@0
   264
        if (m & BOOST_IOS::out)
sl@0
   265
            b->BOOST_IOSTREAMS_PUBSYNC();
sl@0
   266
        b->close(m);
sl@0
   267
    }
sl@0
   268
sl@0
   269
    static void set_next(streambuf_type* b, streambuf_type* next)
sl@0
   270
    { b->set_next(next); }
sl@0
   271
sl@0
   272
    static void set_auto_close(streambuf_type* b, bool close)
sl@0
   273
    { b->set_auto_close(close); }
sl@0
   274
sl@0
   275
    struct closer  : public std::unary_function<streambuf_type*, void>  {
sl@0
   276
        closer(BOOST_IOS::openmode m) : mode_(m) { }
sl@0
   277
        void operator() (streambuf_type* b)
sl@0
   278
        {
sl@0
   279
            close(b, mode_);
sl@0
   280
        }
sl@0
   281
        BOOST_IOS::openmode mode_;
sl@0
   282
    };
sl@0
   283
    friend struct closer;
sl@0
   284
sl@0
   285
    enum flags {
sl@0
   286
        f_complete = 1,
sl@0
   287
        f_open = 2,
sl@0
   288
        f_auto_close = 4
sl@0
   289
    };
sl@0
   290
sl@0
   291
    struct chain_impl {
sl@0
   292
        chain_impl()
sl@0
   293
            : client_(0), device_buffer_size_(default_device_buffer_size),
sl@0
   294
              filter_buffer_size_(default_filter_buffer_size),
sl@0
   295
              pback_size_(default_pback_buffer_size),
sl@0
   296
              flags_(f_auto_close)
sl@0
   297
            { }
sl@0
   298
        ~chain_impl() { try { close(); reset(); } catch (std::exception&) { } }
sl@0
   299
        void close()
sl@0
   300
            {
sl@0
   301
                if ((flags_ & f_open) != 0) {
sl@0
   302
                    stream_buffer< basic_null_device<Ch, Mode> > null;
sl@0
   303
                    if ((flags_ & f_complete) == 0) {
sl@0
   304
                        null.open(basic_null_device<Ch, Mode>());
sl@0
   305
                        set_next(links_.back(), &null);
sl@0
   306
                    }
sl@0
   307
                    links_.front()->BOOST_IOSTREAMS_PUBSYNC();
sl@0
   308
                    if (is_convertible<Mode, input>::value)
sl@0
   309
                        std::for_each( links_.rbegin(), links_.rend(),
sl@0
   310
                                       closer(BOOST_IOS::in) );
sl@0
   311
                    if (is_convertible<Mode, output>::value)
sl@0
   312
                        std::for_each( links_.begin(), links_.end(),
sl@0
   313
                                       closer(BOOST_IOS::out) );
sl@0
   314
                    flags_ &= ~f_open;
sl@0
   315
                }
sl@0
   316
            }
sl@0
   317
        void reset()
sl@0
   318
            {
sl@0
   319
                typedef typename list_type::iterator iterator;
sl@0
   320
                for ( iterator first = links_.begin(),
sl@0
   321
                               last = links_.end();
sl@0
   322
                      first != last;
sl@0
   323
                      ++first )
sl@0
   324
                {
sl@0
   325
                    if ( (flags_ & f_complete) == 0 ||
sl@0
   326
                         (flags_ & f_auto_close) == 0 )
sl@0
   327
                    {
sl@0
   328
                        set_auto_close(*first, false);
sl@0
   329
                    }
sl@0
   330
                    streambuf_type* buf = 0;
sl@0
   331
                    std::swap(buf, *first);
sl@0
   332
                    delete buf;
sl@0
   333
                }
sl@0
   334
                links_.clear();
sl@0
   335
                flags_ &= ~f_complete;
sl@0
   336
                flags_ &= ~f_open;
sl@0
   337
            }
sl@0
   338
        list_type     links_;
sl@0
   339
        client_type*  client_;
sl@0
   340
        int           device_buffer_size_,
sl@0
   341
                      filter_buffer_size_,
sl@0
   342
                      pback_size_;
sl@0
   343
        int           flags_;
sl@0
   344
    };
sl@0
   345
    friend struct chain_impl;
sl@0
   346
sl@0
   347
    //----------Member data---------------------------------------------------//
sl@0
   348
sl@0
   349
private:
sl@0
   350
    shared_ptr<chain_impl> pimpl_;
sl@0
   351
};
sl@0
   352
sl@0
   353
} // End namespace detail.
sl@0
   354
sl@0
   355
//
sl@0
   356
// Macro: BOOST_IOSTREAMS_DECL_CHAIN(name, category)
sl@0
   357
// Description: Defines a template derived from chain_base appropriate for a
sl@0
   358
//      particular i/o category. The template has the following parameters:
sl@0
   359
//      Ch - The character type.
sl@0
   360
//      Tr - The character traits type.
sl@0
   361
//      Alloc - The allocator type.
sl@0
   362
// Macro parameters:
sl@0
   363
//      name_ - The name of the template to be defined.
sl@0
   364
//      category_ - The i/o category of the template to be defined.
sl@0
   365
//
sl@0
   366
#define BOOST_IOSTREAMS_DECL_CHAIN(name_, default_char_) \
sl@0
   367
    template< typename Mode, typename Ch = default_char_, \
sl@0
   368
              typename Tr = BOOST_IOSTREAMS_CHAR_TRAITS(Ch), \
sl@0
   369
              typename Alloc = std::allocator<Ch> > \
sl@0
   370
    class name_ : public boost::iostreams::detail::chain_base< \
sl@0
   371
                            name_<Mode, Ch, Tr, Alloc>, \
sl@0
   372
                            Ch, Tr, Alloc, Mode \
sl@0
   373
                         > \
sl@0
   374
    { \
sl@0
   375
    public: \
sl@0
   376
        struct category : device_tag, Mode { }; \
sl@0
   377
        typedef Mode                                   mode; \
sl@0
   378
    private: \
sl@0
   379
        typedef boost::iostreams::detail::chain_base< \
sl@0
   380
                    name_<Mode, Ch, Tr, Alloc>, \
sl@0
   381
                    Ch, Tr, Alloc, Mode \
sl@0
   382
                >                                      base_type; \
sl@0
   383
    public: \
sl@0
   384
        typedef Ch                                     char_type; \
sl@0
   385
        typedef Tr                                     traits_type; \
sl@0
   386
        typedef typename traits_type::int_type         int_type; \
sl@0
   387
        typedef typename traits_type::off_type         off_type; \
sl@0
   388
        name_() { } \
sl@0
   389
        name_(const name_& rhs) { *this = rhs; } \
sl@0
   390
        name_& operator=(const name_& rhs) \
sl@0
   391
        { base_type::operator=(rhs); return *this; } \
sl@0
   392
    }; \
sl@0
   393
    /**/
sl@0
   394
BOOST_IOSTREAMS_DECL_CHAIN(chain, char)
sl@0
   395
BOOST_IOSTREAMS_DECL_CHAIN(wchain, wchar_t)
sl@0
   396
#undef BOOST_IOSTREAMS_DECL_CHAIN
sl@0
   397
sl@0
   398
//--------------Definition of chain_client------------------------------------//
sl@0
   399
sl@0
   400
namespace detail {
sl@0
   401
sl@0
   402
//
sl@0
   403
// Template name: chain_client
sl@0
   404
// Description: Class whose instances provide access to an underlying chain
sl@0
   405
//      using an interface similar to the chains.
sl@0
   406
// Subclasses: the various stream and stream buffer templates.
sl@0
   407
//
sl@0
   408
template<typename Chain>
sl@0
   409
class chain_client {
sl@0
   410
public:
sl@0
   411
    typedef Chain                             chain_type;
sl@0
   412
    typedef typename chain_type::char_type    char_type;
sl@0
   413
    typedef typename chain_type::traits_type  traits_type;
sl@0
   414
    typedef typename chain_type::size_type    size_type;
sl@0
   415
    typedef typename chain_type::mode         mode;
sl@0
   416
sl@0
   417
    chain_client(chain_type* chn = 0) : chain_(chn ) { }
sl@0
   418
    chain_client(chain_client* client) : chain_(client->chain_) { }
sl@0
   419
    virtual ~chain_client() { }
sl@0
   420
sl@0
   421
    const std::type_info& component_type(int n) const
sl@0
   422
    { return chain_->component_type(n); }
sl@0
   423
sl@0
   424
#if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
sl@0
   425
    // Deprecated.
sl@0
   426
    template<int N>
sl@0
   427
    const std::type_info& component_type() const
sl@0
   428
    { return chain_->BOOST_NESTED_TEMPLATE component_type<N>(); }
sl@0
   429
sl@0
   430
    template<typename T>
sl@0
   431
    T* component(int n) const
sl@0
   432
    { return chain_->BOOST_NESTED_TEMPLATE component<T>(n); }
sl@0
   433
sl@0
   434
    // Deprecated.
sl@0
   435
    template<int N, typename T>
sl@0
   436
    T* component() const
sl@0
   437
    { return chain_->BOOST_NESTED_TEMPLATE component<N, T>(); }
sl@0
   438
#else
sl@0
   439
    template<typename T>
sl@0
   440
    T* component(int n, boost::type<T> t) const
sl@0
   441
    { return chain_->component(n, t); }
sl@0
   442
#endif
sl@0
   443
sl@0
   444
    bool is_complete() const { return chain_->is_complete(); }
sl@0
   445
    bool auto_close() const { return chain_->auto_close(); }
sl@0
   446
    void set_auto_close(bool close) { chain_->set_auto_close(close); }
sl@0
   447
    bool strict_sync() { return chain_->strict_sync(); }
sl@0
   448
    void set_device_buffer_size(std::streamsize n)
sl@0
   449
        { chain_->set_device_buffer_size(n); }
sl@0
   450
    void set_filter_buffer_size(std::streamsize n)
sl@0
   451
        { chain_->set_filter_buffer_size(n); }
sl@0
   452
    void set_pback_size(std::streamsize n) { chain_->set_pback_size(n); }
sl@0
   453
    BOOST_IOSTREAMS_DEFINE_PUSH(push, mode, char_type, push_impl)
sl@0
   454
    void pop() { chain_->pop(); }
sl@0
   455
    bool empty() const { return chain_->empty(); }
sl@0
   456
    size_type size() { return chain_->size(); }
sl@0
   457
    void reset() { chain_->reset(); }
sl@0
   458
sl@0
   459
    // Returns a copy of the underlying chain.
sl@0
   460
    chain_type filters() { return *chain_; }
sl@0
   461
    chain_type filters() const { return *chain_; }
sl@0
   462
protected:
sl@0
   463
    template<typename T>
sl@0
   464
    void push_impl(const T& t BOOST_IOSTREAMS_PUSH_PARAMS())
sl@0
   465
    { chain_->push(t BOOST_IOSTREAMS_PUSH_ARGS()); }
sl@0
   466
    chain_type& ref() { return *chain_; }
sl@0
   467
    void set_chain(chain_type* c)
sl@0
   468
    { chain_ = c; chain_->register_client(this); }
sl@0
   469
#if !defined(BOOST_NO_MEMBER_TEMPLATE_FRIENDS) && \
sl@0
   470
    (!BOOST_WORKAROUND(__BORLANDC__, < 0x600))
sl@0
   471
    template<typename S, typename C, typename T, typename A, typename M>
sl@0
   472
    friend class chain_base;
sl@0
   473
#else
sl@0
   474
    public:
sl@0
   475
#endif
sl@0
   476
    virtual void notify() { }
sl@0
   477
private:
sl@0
   478
    chain_type* chain_;
sl@0
   479
};
sl@0
   480
sl@0
   481
//--------------Implementation of chain_base----------------------------------//
sl@0
   482
sl@0
   483
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   484
inline std::streamsize chain_base<Self, Ch, Tr, Alloc, Mode>::read
sl@0
   485
    (char_type* s, std::streamsize n)
sl@0
   486
{ return iostreams::read(*list().front(), s, n); }
sl@0
   487
sl@0
   488
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   489
inline std::streamsize chain_base<Self, Ch, Tr, Alloc, Mode>::write
sl@0
   490
    (const char_type* s, std::streamsize n)
sl@0
   491
{ return iostreams::write(*list().front(), s, n); }
sl@0
   492
sl@0
   493
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   494
inline std::streampos chain_base<Self, Ch, Tr, Alloc, Mode>::seek
sl@0
   495
    (stream_offset off, BOOST_IOS::seekdir way)
sl@0
   496
{ return iostreams::seek(*list().front(), off, way); }
sl@0
   497
sl@0
   498
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   499
void chain_base<Self, Ch, Tr, Alloc, Mode>::reset()
sl@0
   500
{
sl@0
   501
    using namespace std;
sl@0
   502
    pimpl_->close();
sl@0
   503
    pimpl_->reset();
sl@0
   504
}
sl@0
   505
sl@0
   506
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   507
bool chain_base<Self, Ch, Tr, Alloc, Mode>::is_complete() const
sl@0
   508
{
sl@0
   509
    return (pimpl_->flags_ & f_complete) != 0;
sl@0
   510
}
sl@0
   511
sl@0
   512
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   513
bool chain_base<Self, Ch, Tr, Alloc, Mode>::auto_close() const
sl@0
   514
{
sl@0
   515
    return (pimpl_->flags_ & f_auto_close) != 0;
sl@0
   516
}
sl@0
   517
sl@0
   518
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   519
void chain_base<Self, Ch, Tr, Alloc, Mode>::set_auto_close(bool close)
sl@0
   520
{
sl@0
   521
    pimpl_->flags_ =
sl@0
   522
        (pimpl_->flags_ & ~f_auto_close) |
sl@0
   523
        (close ? f_auto_close : 0);
sl@0
   524
}
sl@0
   525
sl@0
   526
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   527
bool chain_base<Self, Ch, Tr, Alloc, Mode>::strict_sync()
sl@0
   528
{
sl@0
   529
    typedef typename list_type::iterator iterator;
sl@0
   530
    bool result = true;
sl@0
   531
    for ( iterator first = list().begin(),
sl@0
   532
                   last = list().end();
sl@0
   533
          first != last;
sl@0
   534
          ++first )
sl@0
   535
    {
sl@0
   536
        bool s = (*first)->strict_sync();
sl@0
   537
        result = result && s;
sl@0
   538
    }
sl@0
   539
    return result;
sl@0
   540
}
sl@0
   541
sl@0
   542
template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
sl@0
   543
void chain_base<Self, Ch, Tr, Alloc, Mode>::pop()
sl@0
   544
{
sl@0
   545
    assert(!empty());
sl@0
   546
    if (auto_close())
sl@0
   547
        pimpl_->close();
sl@0
   548
    streambuf_type* buf = 0;
sl@0
   549
    std::swap(buf, list().back());
sl@0
   550
    buf->set_auto_close(false);
sl@0
   551
    buf->set_next(0);
sl@0
   552
    delete buf;
sl@0
   553
    list().pop_back();
sl@0
   554
    pimpl_->flags_ &= ~f_complete;
sl@0
   555
    if (auto_close() || list().empty())
sl@0
   556
        pimpl_->flags_ &= ~f_open;
sl@0
   557
}
sl@0
   558
sl@0
   559
} // End namespace detail.
sl@0
   560
sl@0
   561
} } // End namespaces iostreams, boost.
sl@0
   562
sl@0
   563
#endif // #ifndef BOOST_IOSTREAMS_DETAIL_CHAIN_HPP_INCLUDED