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
|