First public contribution.
1 // (C) Copyright Jonathan Turkanis 2003.
2 // Distributed under the Boost Software License, Version 1.0. (See accompanying
3 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
5 // See http://www.boost.org/libs/iostreams for documentation.
7 // To do: add support for random-access.
9 #ifndef BOOST_IOSTREAMS_COMBINE_HPP_INCLUDED
10 #define BOOST_IOSTREAMS_COMBINE_HPP_INCLUDED
12 #if defined(_MSC_VER) && (_MSC_VER >= 1020)
16 #include <boost/config.hpp> // NO_STD_LOCALE, DEDUCED_TYPENAME.
17 #ifndef BOOST_NO_STD_LOCALE
20 #include <boost/iostreams/detail/ios.hpp>
21 #include <boost/iostreams/detail/wrap_unwrap.hpp>
22 #include <boost/iostreams/traits.hpp>
23 #include <boost/iostreams/operations.hpp>
24 #include <boost/mpl/if.hpp>
25 #include <boost/static_assert.hpp>
26 #include <boost/type_traits/is_convertible.hpp>
27 #include <boost/type_traits/is_same.hpp>
29 namespace boost { namespace iostreams {
34 // Template name: combined_device.
35 // Description: Model of Device defined in terms of a Source/Sink pair.
36 // Template paramters:
37 // Source - A model of Source, with the same char_type and traits_type
39 // Sink - A model of Sink, with the same char_type and traits_type
42 template<typename Source, typename Sink>
43 class combined_device {
45 typedef typename char_type_of<Source>::type char_type;
52 combined_device(const Source& src, const Sink& snk);
53 std::streamsize read(char_type* s, std::streamsize n);
54 std::streamsize write(const char_type* s, std::streamsize n);
55 void close(BOOST_IOS::openmode);
56 #ifndef BOOST_NO_STD_LOCALE
57 void imbue(const std::locale& loc);
60 typedef typename char_type_of<Sink>::type sink_char_type;
61 BOOST_STATIC_ASSERT((is_same<char_type, sink_char_type>::value));
67 // Template name: combined_filter.
68 // Description: Model of Device defined in terms of a Source/Sink pair.
69 // Template paramters:
70 // InputFilter - A model of InputFilter, with the same char_type as
72 // OutputFilter - A model of OutputFilter, with the same char_type as
75 template<typename InputFilter, typename OutputFilter>
76 class combined_filter {
78 typedef typename category_of<InputFilter>::type in_category;
79 typedef typename category_of<OutputFilter>::type out_category;
81 typedef typename char_type_of<InputFilter>::type char_type;
83 : multichar_bidirectional_filter_tag,
87 combined_filter(const InputFilter& in, const OutputFilter& out);
89 template<typename Source>
90 std::streamsize read(Source& src, char_type* s, std::streamsize n)
91 { return boost::iostreams::read(in_, src, s, n); }
93 template<typename Sink>
94 std::streamsize write(Sink& snk, const char_type* s, std::streamsize n)
95 { return boost::iostreams::write(out_, snk, s, n); }
97 template<typename Sink>
98 void close(Sink& snk, BOOST_IOS::openmode which)
100 if (which & BOOST_IOS::in)
101 iostreams::close(in_, snk, which);
102 if (which & BOOST_IOS::out)
103 iostreams::close(out_, snk, which);
105 #ifndef BOOST_NO_STD_LOCALE
106 void imbue(const std::locale& loc);
109 typedef typename char_type_of<OutputFilter>::type output_char_type;
110 BOOST_STATIC_ASSERT((is_same<char_type, output_char_type>::value));
115 template<typename In, typename Out>
116 struct combination_traits
120 typename wrapped_type<In>::type,
121 typename wrapped_type<Out>::type
124 typename wrapped_type<In>::type,
125 typename wrapped_type<Out>::type
130 } // End namespace detail.
132 template<typename In, typename Out>
133 struct combination : detail::combination_traits<In, Out>::type {
134 typedef typename detail::combination_traits<In, Out>::type base_type;
135 typedef typename detail::wrapped_type<In>::type in_type;
136 typedef typename detail::wrapped_type<Out>::type out_type;
137 combination(const in_type& in, const out_type& out)
138 : base_type(in, out) { }
143 // Workaround for VC6 ETI bug.
144 template<typename In, typename Out>
145 struct combine_traits {
147 BOOST_DEDUCED_TYPENAME detail::unwrapped_type<In>::type,
148 BOOST_DEDUCED_TYPENAME detail::unwrapped_type<Out>::type
152 } // End namespace detail.
155 // Template name: combine.
156 // Description: Takes a Source/Sink pair or InputFilter/OutputFilter pair and
157 // returns a Reource or Filter which performs input using the first member
158 // of the pair and output using the second member of the pair.
159 // Template paramters:
160 // In - A model of Source or InputFilter, with the same char_type as Out.
161 // Out - A model of Sink or OutputFilter, with the same char_type as In.
163 template<typename In, typename Out>
164 typename detail::combine_traits<In, Out>::type
165 combine(const In& in, const Out& out)
167 typedef typename detail::combine_traits<In, Out>::type return_type;
168 return return_type(in, out);
171 //----------------------------------------------------------------------------//
175 //--------------Implementation of combined_device-----------------------------//
177 template<typename Source, typename Sink>
178 inline combined_device<Source, Sink>::combined_device
179 (const Source& src, const Sink& snk)
180 : src_(src), sink_(snk) { }
182 template<typename Source, typename Sink>
183 inline std::streamsize
184 combined_device<Source, Sink>::read(char_type* s, std::streamsize n)
185 { return iostreams::read(src_, s, n); }
187 template<typename Source, typename Sink>
188 inline std::streamsize
189 combined_device<Source, Sink>::write(const char_type* s, std::streamsize n)
190 { return iostreams::write(sink_, s, n); }
192 template<typename Source, typename Sink>
194 combined_device<Source, Sink>::close(BOOST_IOS::openmode which)
196 if (which & BOOST_IOS::in)
197 iostreams::close(src_, which);
198 if (which & BOOST_IOS::out)
199 iostreams::close(sink_, which);
202 #ifndef BOOST_NO_STD_LOCALE
203 template<typename Source, typename Sink>
204 void combined_device<Source, Sink>::imbue(const std::locale& loc)
206 iostreams::imbue(src_, loc);
207 iostreams::imbue(sink_, loc);
211 //--------------Implementation of filter_pair---------------------------------//
213 template<typename InputFilter, typename OutputFilter>
214 inline combined_filter<InputFilter, OutputFilter>::combined_filter
215 (const InputFilter& in, const OutputFilter& out) : in_(in), out_(out)
218 #ifndef BOOST_NO_STD_LOCALE
219 template<typename InputFilter, typename OutputFilter>
220 void combined_filter<InputFilter, OutputFilter>::imbue
221 (const std::locale& loc)
223 iostreams::imbue(in_, loc);
224 iostreams::imbue(out_, loc);
229 } // End namespace detail.
231 } } // End namespaces iostreams, boost.
233 #endif // #ifndef BOOST_IOSTREAMS_COMBINE_HPP_INCLUDED