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
|