os/ossrv/ossrv_pub/boost_apis/boost/iostreams/invert.hpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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_INVERT_HPP_INCLUDED
sl@0
     8
#define BOOST_IOSTREAMS_INVERT_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>                             // copy, min.  
sl@0
    15
#include <cassert>
sl@0
    16
#include <boost/config.hpp>                      // BOOST_DEDUCED_TYPENAME.       
sl@0
    17
#include <boost/detail/workaround.hpp>           // default_filter_buffer_size.
sl@0
    18
#include <boost/iostreams/char_traits.hpp>
sl@0
    19
#include <boost/iostreams/compose.hpp>
sl@0
    20
#include <boost/iostreams/constants.hpp>
sl@0
    21
#include <boost/iostreams/device/array.hpp>
sl@0
    22
#include <boost/iostreams/detail/buffer.hpp>
sl@0
    23
#include <boost/iostreams/detail/counted_array.hpp>
sl@0
    24
#include <boost/mpl/if.hpp>
sl@0
    25
#include <boost/ref.hpp>
sl@0
    26
#include <boost/shared_ptr.hpp>
sl@0
    27
#include <boost/type_traits/is_convertible.hpp>
sl@0
    28
sl@0
    29
// Must come last.
sl@0
    30
#include <boost/iostreams/detail/config/disable_warnings.hpp>  // MSVC.
sl@0
    31
sl@0
    32
namespace boost { namespace iostreams {
sl@0
    33
sl@0
    34
//
sl@0
    35
// Template name: inverse.
sl@0
    36
// Template paramters:
sl@0
    37
//      Filter - A filter adapter which 
sl@0
    38
// Description: Returns an instance of an appropriate specialization of inverse.
sl@0
    39
//
sl@0
    40
template<typename Filter>
sl@0
    41
class inverse {
sl@0
    42
private:
sl@0
    43
    typedef typename category_of<Filter>::type   base_category;
sl@0
    44
    typedef reference_wrapper<Filter>            filter_ref;
sl@0
    45
public:
sl@0
    46
    typedef typename char_type_of<Filter>::type  char_type;
sl@0
    47
    typedef typename int_type_of<Filter>::type   int_type;
sl@0
    48
    typedef char_traits<char_type>               traits_type;
sl@0
    49
    typedef typename 
sl@0
    50
            mpl::if_<
sl@0
    51
                is_convertible<
sl@0
    52
                    base_category,
sl@0
    53
                    input
sl@0
    54
                >,
sl@0
    55
                output,
sl@0
    56
                input
sl@0
    57
            >::type                              mode;
sl@0
    58
    struct category 
sl@0
    59
        : mode, 
sl@0
    60
          filter_tag, 
sl@0
    61
          multichar_tag, 
sl@0
    62
          closable_tag 
sl@0
    63
        { };
sl@0
    64
    explicit inverse( const Filter& filter, 
sl@0
    65
                      std::streamsize buffer_size = 
sl@0
    66
                          default_filter_buffer_size) 
sl@0
    67
        : pimpl_(new impl(filter, buffer_size))
sl@0
    68
        { }
sl@0
    69
sl@0
    70
    template<typename Source>
sl@0
    71
    std::streamsize read(Source& src, char* s, std::streamsize n)
sl@0
    72
    {
sl@0
    73
        typedef detail::counted_array_sink<char_type>  array_sink;
sl@0
    74
        typedef composite<filter_ref, array_sink>      filtered_array_sink;
sl@0
    75
sl@0
    76
        assert((flags() & f_write) == 0);
sl@0
    77
        if (flags() == 0) {
sl@0
    78
            flags() = f_read;
sl@0
    79
            buf().set(0, 0);
sl@0
    80
        }
sl@0
    81
sl@0
    82
        filtered_array_sink snk(filter(), array_sink(s, n));
sl@0
    83
        int_type status;
sl@0
    84
        for ( status = traits_type::good();
sl@0
    85
              snk.second().count() < n && status == traits_type::good(); )
sl@0
    86
        {
sl@0
    87
            status = buf().fill(src);
sl@0
    88
            buf().flush(snk);
sl@0
    89
        }
sl@0
    90
        return snk.second().count() == 0 &&
sl@0
    91
               status == traits_type::eof() 
sl@0
    92
                   ? 
sl@0
    93
               -1
sl@0
    94
                   : 
sl@0
    95
               snk.second().count();
sl@0
    96
    }
sl@0
    97
sl@0
    98
    template<typename Sink>
sl@0
    99
    std::streamsize write(Sink& dest, const char* s, std::streamsize n)
sl@0
   100
    {
sl@0
   101
        typedef detail::counted_array_source<char_type>  array_source;
sl@0
   102
        typedef composite<filter_ref, array_source>      filtered_array_source;
sl@0
   103
sl@0
   104
        assert((flags() & f_read) == 0);
sl@0
   105
        if (flags() == 0) {
sl@0
   106
            flags() = f_write;
sl@0
   107
            buf().set(0, 0);
sl@0
   108
        }
sl@0
   109
        
sl@0
   110
        filtered_array_source src(filter(), array_source(s, n));
sl@0
   111
        for (bool good = true; src.second().count() < n && good; ) {
sl@0
   112
            buf().fill(src);
sl@0
   113
            good = buf().flush(dest);
sl@0
   114
        }
sl@0
   115
        return src.second().count();
sl@0
   116
    }
sl@0
   117
sl@0
   118
    template<typename Device>
sl@0
   119
    void close( Device& dev, 
sl@0
   120
                BOOST_IOS::openmode which = 
sl@0
   121
                    BOOST_IOS::in | BOOST_IOS::out )
sl@0
   122
    {
sl@0
   123
        if ((which & BOOST_IOS::out) != 0 && (flags() & f_write) != 0)
sl@0
   124
            buf().flush(dev);
sl@0
   125
        flags() = 0;
sl@0
   126
    }
sl@0
   127
private:
sl@0
   128
    filter_ref filter() { return boost::ref(pimpl_->filter_); }
sl@0
   129
    detail::buffer<char_type>& buf() { return pimpl_->buf_; }
sl@0
   130
    int& flags() { return pimpl_->flags_; }
sl@0
   131
    
sl@0
   132
    enum flags_ {
sl@0
   133
        f_read = 1, f_write = 2
sl@0
   134
    };
sl@0
   135
sl@0
   136
    struct impl {
sl@0
   137
        impl(const Filter& filter, std::streamsize n) 
sl@0
   138
            : filter_(filter), buf_(n), flags_(0)
sl@0
   139
        { buf_.set(0, 0); }
sl@0
   140
        Filter                     filter_;
sl@0
   141
        detail::buffer<char_type>  buf_;
sl@0
   142
        int                        flags_;
sl@0
   143
    };
sl@0
   144
    shared_ptr<impl> pimpl_;
sl@0
   145
};
sl@0
   146
sl@0
   147
//
sl@0
   148
// Template name: invert.
sl@0
   149
// Template paramters:
sl@0
   150
//      Filter - A model of InputFilter or OutputFilter.
sl@0
   151
// Description: Returns an instance of an appropriate specialization of inverse.
sl@0
   152
//
sl@0
   153
template<typename Filter>
sl@0
   154
inverse<Filter> invert(const Filter& f) { return inverse<Filter>(f); }
sl@0
   155
                    
sl@0
   156
//----------------------------------------------------------------------------//
sl@0
   157
sl@0
   158
} } // End namespaces iostreams, boost.
sl@0
   159
sl@0
   160
#include <boost/iostreams/detail/config/enable_warnings.hpp>  // MSVC.
sl@0
   161
sl@0
   162
#endif // #ifndef BOOST_IOSTREAMS_INVERT_HPP_INCLUDED