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 // Contains machinery for performing code conversion.
9 #ifndef BOOST_IOSTREAMS_CODE_CONVERTER_HPP_INCLUDED
10 #define BOOST_IOSTREAMS_CODE_CONVERTER_HPP_INCLUDED
12 #if defined(_MSC_VER) && (_MSC_VER >= 1020)
16 #include <boost/iostreams/detail/config/wide_streams.hpp>
17 #if defined(BOOST_IOSTREAMS_NO_WIDE_STREAMS) || \
18 defined(BOOST_IOSTREAMS_NO_LOCALE) \
20 # error code conversion not supported on this platform
23 #include <algorithm> // max.
24 #include <cstring> // memcpy.
26 #include <boost/config.hpp> // DEDUCED_TYPENAME.
27 #include <boost/iostreams/char_traits.hpp>
28 #include <boost/iostreams/constants.hpp> // default_filter_buffer_size.
29 #include <boost/iostreams/detail/adapter/concept_adapter.hpp>
30 #include <boost/iostreams/detail/adapter/direct_adapter.hpp>
31 #include <boost/iostreams/detail/buffer.hpp>
32 #include <boost/iostreams/detail/call_traits.hpp>
33 #include <boost/iostreams/detail/codecvt_holder.hpp>
34 #include <boost/iostreams/detail/codecvt_helper.hpp>
35 #include <boost/iostreams/detail/double_object.hpp>
36 #include <boost/iostreams/detail/forward.hpp>
37 #include <boost/iostreams/detail/ios.hpp> // failure, openmode, int types.
38 #include <boost/iostreams/detail/select.hpp>
39 #include <boost/iostreams/traits.hpp>
40 #include <boost/iostreams/operations.hpp>
41 #include <boost/optional.hpp>
42 #include <boost/shared_ptr.hpp>
43 #include <boost/static_assert.hpp>
44 #include <boost/type_traits/is_convertible.hpp>
45 #include <boost/type_traits/is_same.hpp>
48 #include <boost/iostreams/detail/config/disable_warnings.hpp> // Borland 5.x
50 namespace boost { namespace iostreams {
52 struct code_conversion_error : BOOST_IOSTREAMS_FAILURE {
53 code_conversion_error()
54 : BOOST_IOSTREAMS_FAILURE("code conversion error")
60 //--------------Definition of conversion_buffer-------------------------------//
62 // Buffer and conversion state for reading.
63 template<typename Codecvt, typename Alloc>
64 class conversion_buffer
66 BOOST_DEDUCED_TYPENAME detail::codecvt_extern<Codecvt>::type,
71 typedef typename Codecvt::state_type state_type;
74 BOOST_DEDUCED_TYPENAME detail::codecvt_extern<Codecvt>::type,
80 state_type& state() { return state_; }
85 state_ = state_type();
91 //--------------Definition of converter_impl----------------------------------//
93 // Contains member data, open/is_open/close and buffer management functions.
94 template<typename Device, typename Codecvt, typename Alloc>
95 struct code_converter_impl {
96 typedef typename codecvt_extern<Codecvt>::type extern_type;
97 typedef typename category_of<Device>::type device_category;
98 typedef is_convertible<device_category, input> can_read;
99 typedef is_convertible<device_category, output> can_write;
100 typedef is_convertible<device_category, bidirectional> is_bidir;
102 iostreams::select< // Disambiguation for Tru64.
103 is_bidir, bidirectional,
110 direct_adapter<Device>,
113 typedef optional< concept_adapter<policy_type> > storage_type;
114 typedef is_convertible<device_category, two_sequence> is_double;
115 typedef conversion_buffer<Codecvt, Alloc> buffer_type;
117 code_converter_impl() : cvt_(), flags_(0) { }
119 ~code_converter_impl()
122 if (flags_ & f_open) close();
123 } catch (std::exception&) { /* */ }
126 void open(const Device& dev, int buffer_size)
129 throw BOOST_IOSTREAMS_FAILURE("already open");
130 if (buffer_size == -1)
131 buffer_size = default_filter_buffer_size;
132 int max_length = cvt_.get().max_length();
133 buffer_size = (std::max)(buffer_size, 2 * max_length);
134 if (can_read::value) {
135 buf_.first().resize(buffer_size);
136 buf_.first().set(0, 0);
138 if (can_write::value && !is_double::value) {
139 buf_.second().resize(buffer_size);
140 buf_.second().set(0, 0);
142 dev_.reset(concept_adapter<policy_type>(dev));
146 void close(BOOST_IOS::openmode which = BOOST_IOS::in | BOOST_IOS::out)
148 if (which & BOOST_IOS::in) {
149 iostreams::close(dev(), BOOST_IOS::in);
150 flags_ |= f_input_closed;
152 if (which & BOOST_IOS::out) {
153 buf_.second().flush(dev());
154 iostreams::close(dev(), BOOST_IOS::out);
155 flags_ |= f_output_closed;
157 if ( !is_double::value ||
158 (flags_ & f_input_closed) != 0 &&
159 (flags_ & f_output_closed) != 0 )
162 buf_.first().reset();
163 buf_.second().reset();
168 bool is_open() const { return (flags_ & f_open) != 0;}
170 policy_type& dev() { return **dev_; }
174 f_input_closed = f_open << 1,
175 f_output_closed = f_input_closed << 1
178 codecvt_holder<Codecvt> cvt_;
187 } // End namespace detail.
189 //--------------Definition of converter---------------------------------------//
191 #define BOOST_IOSTREAMS_CONVERTER_PARAMS() , int buffer_size = -1
192 #define BOOST_IOSTREAMS_CONVERTER_ARGS() , buffer_size
194 template<typename Device, typename Codecvt, typename Alloc>
195 struct code_converter_base {
196 typedef detail::code_converter_impl<
197 Device, Codecvt, Alloc
199 code_converter_base() : pimpl_(new impl_type) { }
200 shared_ptr<impl_type> pimpl_;
203 template< typename Device,
204 typename Codecvt = detail::default_codecvt,
205 typename Alloc = std::allocator<char> >
207 : protected code_converter_base<Device, Codecvt, Alloc>
210 typedef detail::code_converter_impl<
211 Device, Codecvt, Alloc
213 typedef typename impl_type::policy_type policy_type;
214 typedef typename impl_type::buffer_type buffer_type;
215 typedef typename detail::codecvt_holder<Codecvt>::codecvt_type codecvt_type;
216 typedef typename detail::codecvt_intern<Codecvt>::type intern_type;
217 typedef typename detail::codecvt_extern<Codecvt>::type extern_type;
218 typedef typename detail::codecvt_state<Codecvt>::type state_type;
220 typedef intern_type char_type;
222 : impl_type::mode, device_tag, closable_tag, localizable_tag
224 BOOST_STATIC_ASSERT((
227 BOOST_DEDUCED_TYPENAME char_type_of<Device>::type
232 #if BOOST_WORKAROUND(__GNUC__, < 3)
233 code_converter(code_converter& rhs)
234 : code_converter_base<Device, Codecvt, Alloc>(rhs)
236 code_converter(const code_converter& rhs)
237 : code_converter_base<Device, Codecvt, Alloc>(rhs)
240 BOOST_IOSTREAMS_FORWARD( code_converter, open_impl, Device,
241 BOOST_IOSTREAMS_CONVERTER_PARAMS,
242 BOOST_IOSTREAMS_CONVERTER_ARGS )
244 // fstream-like interface.
246 bool is_open() const { return this->pimpl_->is_open(); }
247 void close(BOOST_IOS::openmode which = BOOST_IOS::in | BOOST_IOS::out )
248 { impl().close(which); }
252 std::streamsize read(char_type*, std::streamsize);
253 std::streamsize write(const char_type*, std::streamsize);
254 void imbue(const std::locale& loc) { impl().cvt_.imbue(loc); }
256 // Direct device access.
258 Device& operator*() { return detail::unwrap_direct(dev()); }
259 Device* operator->() { return &detail::unwrap_direct(dev()); }
261 template<typename T> // Used for forwarding.
262 void open_impl(const T& t BOOST_IOSTREAMS_CONVERTER_PARAMS())
264 impl().open(t BOOST_IOSTREAMS_CONVERTER_ARGS());
267 const codecvt_type& cvt() { return impl().cvt_.get(); }
268 policy_type& dev() { return impl().dev(); }
269 buffer_type& in() { return impl().buf_.first(); }
270 buffer_type& out() { return impl().buf_.second(); }
271 impl_type& impl() { return *this->pimpl_; }
274 //--------------Implementation of converter-----------------------------------//
276 // Implementation note: if end of stream contains a partial character,
278 template<typename Device, typename Codevt, typename Alloc>
279 std::streamsize code_converter<Device, Codevt, Alloc>::read
280 (char_type* s, std::streamsize n)
283 const extern_type* next; // Next external char.
284 intern_type* nint; // Next internal char.
285 streamsize total = 0; // Characters read.
286 int status = iostreams::char_traits<char>::good();
287 bool partial = false;
288 buffer_type& buf = in();
293 if (buf.ptr() == buf.eptr() || partial) {
294 status = buf.fill(dev());
295 if (buf.ptr() == buf.eptr())
301 codecvt_base::result result =
302 cvt().in( buf.state(),
303 buf.ptr(), buf.eptr(), next,
304 s + total, s + n, nint );
305 buf.ptr() += next - buf.ptr();
306 total = static_cast<streamsize>(nint - s);
309 case codecvt_base::partial:
312 case codecvt_base::ok:
314 case codecvt_base::error:
315 buf.state() = state_type();
316 throw code_conversion_error();
317 case codecvt_base::noconv:
319 buf.state() = state_type();
320 intern_type c = intern_type();
321 memcpy(&c, (const void*) buf.ptr(), sizeof(extern_type));
326 } while (total < n && status != EOF && status != WOULD_BLOCK);
328 return total == 0 && status == EOF ? -1 : total;
331 template<typename Device, typename Codevt, typename Alloc>
332 std::streamsize code_converter<Device, Codevt, Alloc>::write
333 (const char_type* s, std::streamsize n)
336 buffer_type& buf = out();
337 extern_type* next; // Next external char.
338 const intern_type* nint; // Next internal char.
339 streamsize total = 0; // Characters written.
340 bool partial = false;
345 if (buf.eptr() == buf.end() || partial) {
346 if (!buf.flush(dev()))
352 codecvt_base::result result =
353 cvt().out( buf.state(),
354 s + total, s + n, nint,
355 buf.eptr(), buf.end(), next );
356 int progress = (int) (next - buf.eptr());
357 buf.eptr() += progress;
360 case codecvt_base::partial:
361 partial = true; // Fall through.
362 case codecvt_base::ok:
363 total = static_cast<streamsize>(nint - s);
365 case codecvt_base::error:
366 buf.state() = state_type();
367 throw code_conversion_error();
368 case codecvt_base::noconv:
370 // This can be shortened to two memcpy's.
371 const char* c = (const char*) (s + total);
372 for ( std::size_t index = 0;
373 index < sizeof(intern_type);
374 index += sizeof(extern_type),
377 memcpy(buf.ptr(), c + index, sizeof(extern_type));
378 if (buf.eptr() == buf.end())
388 //----------------------------------------------------------------------------//
390 } } // End namespaces iostreams, boost.
392 #include <boost/iostreams/detail/config/enable_warnings.hpp> // Borland 5.x
394 #endif // #ifndef BOOST_IOSTREAMS_CODE_CONVERTER_HPP_INCLUDED