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 |
//
|
sl@0
|
8 |
// Contains metafunctions char_type_of, category_of and mode_of used for
|
sl@0
|
9 |
// deducing the i/o category and i/o mode of a model of Filter or Device.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Also contains several utility metafunctions, functions and macros.
|
sl@0
|
12 |
//
|
sl@0
|
13 |
|
sl@0
|
14 |
#ifndef BOOST_IOSTREAMS_IO_TRAITS_HPP_INCLUDED
|
sl@0
|
15 |
#define BOOST_IOSTREAMS_IO_TRAITS_HPP_INCLUDED
|
sl@0
|
16 |
|
sl@0
|
17 |
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
|
sl@0
|
18 |
# pragma once
|
sl@0
|
19 |
#endif
|
sl@0
|
20 |
|
sl@0
|
21 |
#include <iosfwd> // stream types, char_traits.
|
sl@0
|
22 |
#include <boost/config.hpp> // partial spec, deduced typename.
|
sl@0
|
23 |
#include <boost/iostreams/categories.hpp>
|
sl@0
|
24 |
#include <boost/iostreams/detail/bool_trait_def.hpp>
|
sl@0
|
25 |
#include <boost/iostreams/detail/config/wide_streams.hpp>
|
sl@0
|
26 |
#include <boost/iostreams/detail/is_iterator_range.hpp>
|
sl@0
|
27 |
#include <boost/iostreams/detail/select.hpp>
|
sl@0
|
28 |
#include <boost/iostreams/detail/select_by_size.hpp>
|
sl@0
|
29 |
#include <boost/iostreams/detail/wrap_unwrap.hpp>
|
sl@0
|
30 |
#include <boost/iostreams/traits_fwd.hpp>
|
sl@0
|
31 |
#include <boost/mpl/bool.hpp>
|
sl@0
|
32 |
#include <boost/mpl/eval_if.hpp>
|
sl@0
|
33 |
#include <boost/mpl/identity.hpp>
|
sl@0
|
34 |
#include <boost/mpl/int.hpp>
|
sl@0
|
35 |
#include <boost/mpl/or.hpp>
|
sl@0
|
36 |
#include <boost/range/iterator_range.hpp>
|
sl@0
|
37 |
#include <boost/range/value_type.hpp>
|
sl@0
|
38 |
#include <boost/type_traits/is_convertible.hpp>
|
sl@0
|
39 |
|
sl@0
|
40 |
namespace boost { namespace iostreams {
|
sl@0
|
41 |
|
sl@0
|
42 |
//------------------Definitions of predicates for streams and stream buffers--//
|
sl@0
|
43 |
|
sl@0
|
44 |
#ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //--------------------------------//
|
sl@0
|
45 |
|
sl@0
|
46 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_istream, std::basic_istream, 2)
|
sl@0
|
47 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_ostream, std::basic_ostream, 2)
|
sl@0
|
48 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_iostream, std::basic_iostream, 2)
|
sl@0
|
49 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_streambuf, std::basic_streambuf, 2)
|
sl@0
|
50 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_stringstream, std::basic_stringstream, 3)
|
sl@0
|
51 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_stringbuf, std::basic_stringbuf, 3)
|
sl@0
|
52 |
|
sl@0
|
53 |
#else // #ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //-----------------------//
|
sl@0
|
54 |
|
sl@0
|
55 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_istream, std::istream, 0)
|
sl@0
|
56 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_ostream, std::ostream, 0)
|
sl@0
|
57 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_iostream, std::iostream, 0)
|
sl@0
|
58 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_streambuf, std::streambuf, 0)
|
sl@0
|
59 |
|
sl@0
|
60 |
#endif // #ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //----------------------//
|
sl@0
|
61 |
|
sl@0
|
62 |
template<typename T>
|
sl@0
|
63 |
struct is_std_io
|
sl@0
|
64 |
: mpl::or_< is_istream<T>, is_ostream<T>, is_streambuf<T> >
|
sl@0
|
65 |
{ };
|
sl@0
|
66 |
|
sl@0
|
67 |
namespace detail {
|
sl@0
|
68 |
|
sl@0
|
69 |
template<typename T, typename Tr>
|
sl@0
|
70 |
class linked_streambuf;
|
sl@0
|
71 |
|
sl@0
|
72 |
BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_linked, linked_streambuf, 2)
|
sl@0
|
73 |
|
sl@0
|
74 |
} // End namespace detail.
|
sl@0
|
75 |
|
sl@0
|
76 |
//------------------Definitions of char_type_of-------------------------------//
|
sl@0
|
77 |
|
sl@0
|
78 |
namespace detail {
|
sl@0
|
79 |
|
sl@0
|
80 |
template<typename T>
|
sl@0
|
81 |
struct member_char_type { typedef typename T::char_type type; };
|
sl@0
|
82 |
|
sl@0
|
83 |
} // End namespace detail.
|
sl@0
|
84 |
|
sl@0
|
85 |
#ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //---------------------------//
|
sl@0
|
86 |
# ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //-------------------------------//
|
sl@0
|
87 |
|
sl@0
|
88 |
template<typename T>
|
sl@0
|
89 |
struct char_type_of
|
sl@0
|
90 |
: detail::member_char_type<
|
sl@0
|
91 |
typename detail::unwrapped_type<T>::type
|
sl@0
|
92 |
>
|
sl@0
|
93 |
{ };
|
sl@0
|
94 |
|
sl@0
|
95 |
# else // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //---------------------//
|
sl@0
|
96 |
|
sl@0
|
97 |
template<typename T>
|
sl@0
|
98 |
struct char_type_of {
|
sl@0
|
99 |
typedef typename detail::unwrapped_type<T>::type U;
|
sl@0
|
100 |
typedef typename
|
sl@0
|
101 |
mpl::eval_if<
|
sl@0
|
102 |
is_std_io<U>,
|
sl@0
|
103 |
mpl::identity<char>,
|
sl@0
|
104 |
detail::member_char_type<U>
|
sl@0
|
105 |
>::type type;
|
sl@0
|
106 |
};
|
sl@0
|
107 |
|
sl@0
|
108 |
# endif // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //--------------------//
|
sl@0
|
109 |
|
sl@0
|
110 |
template<typename Iter>
|
sl@0
|
111 |
struct char_type_of< iterator_range<Iter> > {
|
sl@0
|
112 |
typedef typename iterator_value<Iter>::type type;
|
sl@0
|
113 |
};
|
sl@0
|
114 |
|
sl@0
|
115 |
#else // #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //------------------//
|
sl@0
|
116 |
|
sl@0
|
117 |
template<typename T>
|
sl@0
|
118 |
struct char_type_of {
|
sl@0
|
119 |
template<typename U>
|
sl@0
|
120 |
struct get_value_type {
|
sl@0
|
121 |
typedef typename range_value<U>::type type;
|
sl@0
|
122 |
};
|
sl@0
|
123 |
typedef typename
|
sl@0
|
124 |
mpl::eval_if<
|
sl@0
|
125 |
is_iterator_range<T>,
|
sl@0
|
126 |
get_value_type<T>,
|
sl@0
|
127 |
detail::member_char_type<
|
sl@0
|
128 |
BOOST_DEDUCED_TYPENAME detail::unwrapped_type<T>::type
|
sl@0
|
129 |
>
|
sl@0
|
130 |
>::type type;
|
sl@0
|
131 |
};
|
sl@0
|
132 |
|
sl@0
|
133 |
#endif // #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //-----------------//
|
sl@0
|
134 |
|
sl@0
|
135 |
//------------------Definitions of category_of--------------------------------//
|
sl@0
|
136 |
|
sl@0
|
137 |
namespace detail {
|
sl@0
|
138 |
|
sl@0
|
139 |
template<typename T>
|
sl@0
|
140 |
struct member_category { typedef typename T::category type; };
|
sl@0
|
141 |
|
sl@0
|
142 |
} // End namespace detail.
|
sl@0
|
143 |
|
sl@0
|
144 |
template<typename T>
|
sl@0
|
145 |
struct category_of {
|
sl@0
|
146 |
template<typename U>
|
sl@0
|
147 |
struct member_category {
|
sl@0
|
148 |
typedef typename U::category type;
|
sl@0
|
149 |
};
|
sl@0
|
150 |
typedef typename detail::unwrapped_type<T>::type U;
|
sl@0
|
151 |
typedef typename
|
sl@0
|
152 |
mpl::eval_if<
|
sl@0
|
153 |
is_std_io<U>,
|
sl@0
|
154 |
iostreams::select< // Disambiguation for Tru64
|
sl@0
|
155 |
is_iostream<U>, iostream_tag,
|
sl@0
|
156 |
is_istream<U>, istream_tag,
|
sl@0
|
157 |
is_ostream<U>, ostream_tag,
|
sl@0
|
158 |
is_streambuf<U>, streambuf_tag
|
sl@0
|
159 |
>,
|
sl@0
|
160 |
detail::member_category<U>
|
sl@0
|
161 |
>::type type;
|
sl@0
|
162 |
};
|
sl@0
|
163 |
|
sl@0
|
164 |
//------------------Definition of get_category--------------------------------//
|
sl@0
|
165 |
|
sl@0
|
166 |
//
|
sl@0
|
167 |
// Returns an object of type category_of<T>::type.
|
sl@0
|
168 |
//
|
sl@0
|
169 |
template<typename T>
|
sl@0
|
170 |
inline typename category_of<T>::type get_category(const T&)
|
sl@0
|
171 |
{ typedef typename category_of<T>::type category; return category(); }
|
sl@0
|
172 |
|
sl@0
|
173 |
//------------------Definition of int_type_of---------------------------------//
|
sl@0
|
174 |
|
sl@0
|
175 |
template<typename T>
|
sl@0
|
176 |
struct int_type_of {
|
sl@0
|
177 |
#ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES
|
sl@0
|
178 |
typedef std::char_traits<
|
sl@0
|
179 |
BOOST_DEDUCED_TYPENAME char_type_of<T>::type
|
sl@0
|
180 |
> traits_type;
|
sl@0
|
181 |
typedef typename traits_type::int_type type;
|
sl@0
|
182 |
#else
|
sl@0
|
183 |
typedef int type;
|
sl@0
|
184 |
#endif
|
sl@0
|
185 |
};
|
sl@0
|
186 |
|
sl@0
|
187 |
//------------------Definition of mode----------------------------------------//
|
sl@0
|
188 |
|
sl@0
|
189 |
namespace detail {
|
sl@0
|
190 |
|
sl@0
|
191 |
template<int N> struct io_mode_impl;
|
sl@0
|
192 |
|
sl@0
|
193 |
#define BOOST_IOSTREAMS_MODE_HELPER(tag_, id_) \
|
sl@0
|
194 |
case_<id_> io_mode_impl_helper(tag_); \
|
sl@0
|
195 |
template<> struct io_mode_impl<id_> { typedef tag_ type; }; \
|
sl@0
|
196 |
/**/
|
sl@0
|
197 |
BOOST_IOSTREAMS_MODE_HELPER(input, 1)
|
sl@0
|
198 |
BOOST_IOSTREAMS_MODE_HELPER(output, 2)
|
sl@0
|
199 |
BOOST_IOSTREAMS_MODE_HELPER(bidirectional, 3)
|
sl@0
|
200 |
BOOST_IOSTREAMS_MODE_HELPER(input_seekable, 4)
|
sl@0
|
201 |
BOOST_IOSTREAMS_MODE_HELPER(output_seekable, 5)
|
sl@0
|
202 |
BOOST_IOSTREAMS_MODE_HELPER(seekable, 6)
|
sl@0
|
203 |
BOOST_IOSTREAMS_MODE_HELPER(dual_seekable, 7)
|
sl@0
|
204 |
BOOST_IOSTREAMS_MODE_HELPER(bidirectional_seekable, 8)
|
sl@0
|
205 |
BOOST_IOSTREAMS_MODE_HELPER(dual_use, 9)
|
sl@0
|
206 |
#undef BOOST_IOSTREAMS_MODE_HELPER
|
sl@0
|
207 |
|
sl@0
|
208 |
template<typename T>
|
sl@0
|
209 |
struct io_mode_id {
|
sl@0
|
210 |
typedef typename category_of<T>::type category;
|
sl@0
|
211 |
BOOST_SELECT_BY_SIZE(int, value, detail::io_mode_impl_helper(category()));
|
sl@0
|
212 |
};
|
sl@0
|
213 |
|
sl@0
|
214 |
} // End namespace detail.
|
sl@0
|
215 |
|
sl@0
|
216 |
template<typename T> // Borland 5.6.4 requires this circumlocution.
|
sl@0
|
217 |
struct mode_of : detail::io_mode_impl< detail::io_mode_id<T>::value > { };
|
sl@0
|
218 |
|
sl@0
|
219 |
//------------------Definition of is_device, is_filter and is_direct----------//
|
sl@0
|
220 |
|
sl@0
|
221 |
namespace detail {
|
sl@0
|
222 |
|
sl@0
|
223 |
template<typename T, typename Tag>
|
sl@0
|
224 |
struct has_trait_impl {
|
sl@0
|
225 |
typedef typename category_of<T>::type category;
|
sl@0
|
226 |
BOOST_STATIC_CONSTANT(bool, value = (is_convertible<category, Tag>::value));
|
sl@0
|
227 |
};
|
sl@0
|
228 |
|
sl@0
|
229 |
template<typename T, typename Tag>
|
sl@0
|
230 |
struct has_trait
|
sl@0
|
231 |
: mpl::bool_<has_trait_impl<T, Tag>::value>
|
sl@0
|
232 |
{ };
|
sl@0
|
233 |
|
sl@0
|
234 |
} // End namespace detail.
|
sl@0
|
235 |
|
sl@0
|
236 |
template<typename T>
|
sl@0
|
237 |
struct is_device : detail::has_trait<T, device_tag> { };
|
sl@0
|
238 |
|
sl@0
|
239 |
template<typename T>
|
sl@0
|
240 |
struct is_filter : detail::has_trait<T, filter_tag> { };
|
sl@0
|
241 |
|
sl@0
|
242 |
template<typename T>
|
sl@0
|
243 |
struct is_direct : detail::has_trait<T, direct_tag> { };
|
sl@0
|
244 |
|
sl@0
|
245 |
//------------------Definition of BOOST_IOSTREAMS_STREAMBUF_TYPEDEFS----------//
|
sl@0
|
246 |
|
sl@0
|
247 |
#define BOOST_IOSTREAMS_STREAMBUF_TYPEDEFS(Tr) \
|
sl@0
|
248 |
typedef Tr traits_type; \
|
sl@0
|
249 |
typedef typename traits_type::int_type int_type; \
|
sl@0
|
250 |
typedef typename traits_type::off_type off_type; \
|
sl@0
|
251 |
typedef typename traits_type::pos_type pos_type; \
|
sl@0
|
252 |
/**/
|
sl@0
|
253 |
|
sl@0
|
254 |
} } // End namespaces iostreams, boost.
|
sl@0
|
255 |
|
sl@0
|
256 |
#endif // #ifndef BOOST_IOSTREAMS_IO_TRAITS_HPP_INCLUDED
|