123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- /////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
- // xml_woarchive_impl.ipp:
- // (C) Copyright 2002 Robert Ramey - http://www.rrsd.com .
- // Distributed under the Boost Software License, Version 1.0. (See
- // accompanying file LICENSE_1_0.txt or copy at
- // http://www.boost.org/LICENSE_1_0.txt)
- #include <boost/config.hpp>
- #ifndef BOOST_NO_STD_WSTREAMBUF
- #include <ostream>
- #include <string>
- #include <algorithm>
- #include <locale>
- #include <boost/config.hpp> // msvc 6.0 needs this to suppress warnings
- // for BOOST_DEDUCED_TYPENAME
- #include <cstring> // strlen
- #include <cstdlib> // mbtowc
- #include <cwchar> // wcslen
- #if defined(BOOST_NO_STDC_NAMESPACE)
- namespace std{
- using ::strlen;
- #if ! defined(BOOST_NO_INTRINSIC_WCHAR_T)
- using ::mbtowc;
- using ::wcslen;
- #endif
- } // namespace std
- #endif
- #include <boost/serialization/throw_exception.hpp>
- #include <boost/serialization/pfto.hpp>
- #include <boost/archive/iterators/xml_escape.hpp>
- #include <boost/archive/iterators/wchar_from_mb.hpp>
- #include <boost/archive/iterators/ostream_iterator.hpp>
- #include <boost/archive/iterators/dataflow_exception.hpp>
- #include <boost/archive/add_facet.hpp>
- #include <boost/archive/detail/utf8_codecvt_facet.hpp>
- namespace boost {
- namespace archive {
- /////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
- // implemenations of functions specific to wide char archives
- // copy chars to output escaping to xml and widening characters as we go
- template<class InputIterator>
- void save_iterator(std::wostream &os, InputIterator begin, InputIterator end){
- typedef iterators::wchar_from_mb<
- iterators::xml_escape<InputIterator>
- > xmbtows;
- std::copy(
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(begin)),
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(end)),
- boost::archive::iterators::ostream_iterator<wchar_t>(os)
- );
- }
- template<class Archive>
- BOOST_WARCHIVE_DECL(void)
- xml_woarchive_impl<Archive>::save(const std::string & s){
- // note: we don't use s.begin() and s.end() because dinkumware
- // doesn't have string::value_type defined. So use a wrapper
- // around these values to implement the definitions.
- const char * begin = s.data();
- const char * end = begin + s.size();
- save_iterator(os, begin, end);
- }
- #ifndef BOOST_NO_STD_WSTRING
- template<class Archive>
- BOOST_WARCHIVE_DECL(void)
- xml_woarchive_impl<Archive>::save(const std::wstring & ws){
- #if 0
- typedef iterators::xml_escape<std::wstring::const_iterator> xmbtows;
- std::copy(
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(ws.begin())),
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(ws.end())),
- boost::archive::iterators::ostream_iterator<wchar_t>(os)
- );
- #endif
- typedef iterators::xml_escape<const wchar_t *> xmbtows;
- std::copy(
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(ws.data())),
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(ws.data() + ws.size())),
- boost::archive::iterators::ostream_iterator<wchar_t>(os)
- );
- }
- #endif //BOOST_NO_STD_WSTRING
- template<class Archive>
- BOOST_WARCHIVE_DECL(void)
- xml_woarchive_impl<Archive>::save(const char * s){
- save_iterator(os, s, s + std::strlen(s));
- }
- #ifndef BOOST_NO_INTRINSIC_WCHAR_T
- template<class Archive>
- BOOST_WARCHIVE_DECL(void)
- xml_woarchive_impl<Archive>::save(const wchar_t * ws){
- os << ws;
- typedef iterators::xml_escape<const wchar_t *> xmbtows;
- std::copy(
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(ws)),
- xmbtows(BOOST_MAKE_PFTO_WRAPPER(ws + std::wcslen(ws))),
- boost::archive::iterators::ostream_iterator<wchar_t>(os)
- );
- }
- #endif
- template<class Archive>
- BOOST_WARCHIVE_DECL(BOOST_PP_EMPTY())
- xml_woarchive_impl<Archive>::xml_woarchive_impl(
- std::wostream & os_,
- unsigned int flags
- ) :
- basic_text_oprimitive<std::wostream>(
- os_,
- true // don't change the codecvt - use the one below
- ),
- basic_xml_oarchive<Archive>(flags)
- {
- // Standard behavior is that imbue can be called
- // a) before output is invoked or
- // b) after flush has been called. This prevents one-to-many
- // transforms (such as one to many transforms from getting
- // mixed up. Unfortunately, STLPort doesn't respect b) above
- // so the restoration of the original archive locale done by
- // the locale_saver doesn't get processed,
- // before the current one is destroyed.
- // so the codecvt doesn't get replaced with the orginal
- // so closing the stream invokes codecvt::do_unshift
- // so it crashes because the corresponding locale that contained
- // the codecvt isn't around any more.
- // we can hack around this by using a static codecvt that never
- // gets destroyed.
- if(0 == (flags & no_codecvt)){
- boost::archive::detail::utf8_codecvt_facet *pfacet;
- #if defined(__SGI_STL_PORT)
- static boost::archive::detail::utf8_codecvt_facet
- facet(static_cast<size_t>(1));
- pfacet = & facet;
- #else
- pfacet = new boost::archive::detail::utf8_codecvt_facet;
- #endif
- archive_locale.reset(add_facet(std::locale::classic(), pfacet));
- os.imbue(* archive_locale);
- }
- if(0 == (flags & no_header))
- this->init();
- }
- } // namespace archive
- } // namespace boost
- #endif //BOOST_NO_STD_WSTREAMBUF
|