200 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			200 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| 
								 | 
							
								/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
							 | 
						||
| 
								 | 
							
								// xml_iarchive_impl.cpp:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// (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)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//  See http://www.boost.org for updates, documentation, and revision history.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/config.hpp>
							 | 
						||
| 
								 | 
							
								#include <cstring> // memcpy
							 | 
						||
| 
								 | 
							
								#include <cstddef> // NULL
							 | 
						||
| 
								 | 
							
								#include <exception>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if defined(BOOST_NO_STDC_NAMESPACE)
							 | 
						||
| 
								 | 
							
								namespace std{ 
							 | 
						||
| 
								 | 
							
								    using ::memcpy;
							 | 
						||
| 
								 | 
							
								} // namespace std
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifndef BOOST_NO_CWCHAR
							 | 
						||
| 
								 | 
							
								#include <cwchar> // mbstate_t and mbrtowc
							 | 
						||
| 
								 | 
							
								#if defined(BOOST_NO_STDC_NAMESPACE)
							 | 
						||
| 
								 | 
							
								namespace std{ 
							 | 
						||
| 
								 | 
							
								    using ::mbstate_t;
							 | 
						||
| 
								 | 
							
								    using ::mbrtowc;
							 | 
						||
| 
								 | 
							
								 } // namespace std
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								#endif // BOOST_NO_CWCHAR
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/detail/workaround.hpp> // RogueWave and Dinkumware
							 | 
						||
| 
								 | 
							
								#if BOOST_WORKAROUND(BOOST_DINKUMWARE_STDLIB, == 1)
							 | 
						||
| 
								 | 
							
								#include <boost/archive/dinkumware.hpp>
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/core/no_exceptions_support.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/archive/xml_archive_exception.hpp>
							 | 
						||
| 
								 | 
							
								#include <boost/archive/iterators/dataflow_exception.hpp>
							 | 
						||
| 
								 | 
							
								#include <boost/archive/basic_xml_archive.hpp>
							 | 
						||
| 
								 | 
							
								#include <boost/archive/xml_iarchive.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "basic_xml_grammar.hpp"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace boost {
							 | 
						||
| 
								 | 
							
								namespace archive {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
							 | 
						||
| 
								 | 
							
								// implemenations of functions specific to char archives
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// wide char stuff used by char archives
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifndef BOOST_NO_CWCHAR
							 | 
						||
| 
								 | 
							
								#ifndef BOOST_NO_STD_WSTRING
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL void
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::load(std::wstring &ws){
							 | 
						||
| 
								 | 
							
								    std::string s;
							 | 
						||
| 
								 | 
							
								    bool result = gimpl->parse_string(is, s);
							 | 
						||
| 
								 | 
							
								    if(! result)
							 | 
						||
| 
								 | 
							
								        boost::serialization::throw_exception(
							 | 
						||
| 
								 | 
							
								            xml_archive_exception(xml_archive_exception::xml_archive_parsing_error)
							 | 
						||
| 
								 | 
							
								        );
							 | 
						||
| 
								 | 
							
								    
							 | 
						||
| 
								 | 
							
								    #if BOOST_WORKAROUND(_RWSTD_VER, BOOST_TESTED_AT(20101))
							 | 
						||
| 
								 | 
							
								    if(NULL != ws.data())
							 | 
						||
| 
								 | 
							
								    #endif
							 | 
						||
| 
								 | 
							
								    ws.resize(0);
							 | 
						||
| 
								 | 
							
								    std::mbstate_t mbs = std::mbstate_t();
							 | 
						||
| 
								 | 
							
								    const char * start = s.data();
							 | 
						||
| 
								 | 
							
								    const char * end = start + s.size();
							 | 
						||
| 
								 | 
							
								    while(start < end){
							 | 
						||
| 
								 | 
							
								        wchar_t wc;
							 | 
						||
| 
								 | 
							
								        std::size_t count = std::mbrtowc(&wc, start, end - start, &mbs);
							 | 
						||
| 
								 | 
							
								        if(count == static_cast<std::size_t>(-1))
							 | 
						||
| 
								 | 
							
								            boost::serialization::throw_exception(
							 | 
						||
| 
								 | 
							
								                iterators::dataflow_exception(
							 | 
						||
| 
								 | 
							
								                    iterators::dataflow_exception::invalid_conversion
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								            );
							 | 
						||
| 
								 | 
							
								        if(count == static_cast<std::size_t>(-2))
							 | 
						||
| 
								 | 
							
								            continue;
							 | 
						||
| 
								 | 
							
								        start += count;
							 | 
						||
| 
								 | 
							
								        ws += wc;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								#endif // BOOST_NO_STD_WSTRING
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifndef BOOST_NO_INTRINSIC_WCHAR_T
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL void
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::load(wchar_t * ws){
							 | 
						||
| 
								 | 
							
								    std::string s;
							 | 
						||
| 
								 | 
							
								    bool result = gimpl->parse_string(is, s);
							 | 
						||
| 
								 | 
							
								    if(! result)
							 | 
						||
| 
								 | 
							
								        boost::serialization::throw_exception(
							 | 
						||
| 
								 | 
							
								            xml_archive_exception(
							 | 
						||
| 
								 | 
							
								                xml_archive_exception::xml_archive_parsing_error
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								        );
							 | 
						||
| 
								 | 
							
								        
							 | 
						||
| 
								 | 
							
								    std::mbstate_t mbs = std::mbstate_t();
							 | 
						||
| 
								 | 
							
								    const char * start = s.data();
							 | 
						||
| 
								 | 
							
								    const char * end = start + s.size();
							 | 
						||
| 
								 | 
							
								    while(start < end){
							 | 
						||
| 
								 | 
							
								        wchar_t wc;
							 | 
						||
| 
								 | 
							
								        std::size_t length = std::mbrtowc(&wc, start, end - start, &mbs);
							 | 
						||
| 
								 | 
							
								        if(static_cast<std::size_t>(-1) == length)
							 | 
						||
| 
								 | 
							
								            boost::serialization::throw_exception(
							 | 
						||
| 
								 | 
							
								                iterators::dataflow_exception(
							 | 
						||
| 
								 | 
							
								                    iterators::dataflow_exception::invalid_conversion
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								            );
							 | 
						||
| 
								 | 
							
								        if(static_cast<std::size_t>(-2) == length)
							 | 
						||
| 
								 | 
							
								            continue;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        start += length;
							 | 
						||
| 
								 | 
							
								        *ws++ = wc;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    *ws = L'\0';
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								#endif // BOOST_NO_INTRINSIC_WCHAR_T
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif // BOOST_NO_CWCHAR
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL void
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::load(std::string &s){
							 | 
						||
| 
								 | 
							
								    bool result = gimpl->parse_string(is, s);
							 | 
						||
| 
								 | 
							
								    if(! result)
							 | 
						||
| 
								 | 
							
								        boost::serialization::throw_exception(
							 | 
						||
| 
								 | 
							
								            xml_archive_exception(xml_archive_exception::xml_archive_parsing_error)
							 | 
						||
| 
								 | 
							
								        );
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL void
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::load(char * s){
							 | 
						||
| 
								 | 
							
								    std::string tstring;
							 | 
						||
| 
								 | 
							
								    bool result = gimpl->parse_string(is, tstring);
							 | 
						||
| 
								 | 
							
								    if(! result)
							 | 
						||
| 
								 | 
							
								        boost::serialization::throw_exception(
							 | 
						||
| 
								 | 
							
								            xml_archive_exception(xml_archive_exception::xml_archive_parsing_error)
							 | 
						||
| 
								 | 
							
								        );
							 | 
						||
| 
								 | 
							
								    std::memcpy(s, tstring.data(), tstring.size());
							 | 
						||
| 
								 | 
							
								    s[tstring.size()] = 0;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL void
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::load_override(class_name_type & t){
							 | 
						||
| 
								 | 
							
								    const std::string & s = gimpl->rv.class_name;
							 | 
						||
| 
								 | 
							
								    if(s.size() > BOOST_SERIALIZATION_MAX_KEY_SIZE - 1)
							 | 
						||
| 
								 | 
							
								        boost::serialization::throw_exception(
							 | 
						||
| 
								 | 
							
								            archive_exception(archive_exception::invalid_class_name)
							 | 
						||
| 
								 | 
							
								       );
							 | 
						||
| 
								 | 
							
								    char * tptr = t;
							 | 
						||
| 
								 | 
							
								    std::memcpy(tptr, s.data(), s.size());
							 | 
						||
| 
								 | 
							
								    tptr[s.size()] = '\0';
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL void
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::init(){
							 | 
						||
| 
								 | 
							
								    gimpl->init(is);
							 | 
						||
| 
								 | 
							
								    this->set_library_version(
							 | 
						||
| 
								 | 
							
								        library_version_type(gimpl->rv.version)
							 | 
						||
| 
								 | 
							
								    );
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::xml_iarchive_impl(
							 | 
						||
| 
								 | 
							
								    std::istream &is_,
							 | 
						||
| 
								 | 
							
								    unsigned int flags
							 | 
						||
| 
								 | 
							
								) :
							 | 
						||
| 
								 | 
							
								    basic_text_iprimitive<std::istream>(
							 | 
						||
| 
								 | 
							
								        is_, 
							 | 
						||
| 
								 | 
							
								        0 != (flags & no_codecvt)
							 | 
						||
| 
								 | 
							
								    ),
							 | 
						||
| 
								 | 
							
								    basic_xml_iarchive<Archive>(flags),
							 | 
						||
| 
								 | 
							
								    gimpl(new xml_grammar())
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    if(0 == (flags & no_header))
							 | 
						||
| 
								 | 
							
								        init();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Archive>
							 | 
						||
| 
								 | 
							
								BOOST_ARCHIVE_DECL
							 | 
						||
| 
								 | 
							
								xml_iarchive_impl<Archive>::~xml_iarchive_impl(){
							 | 
						||
| 
								 | 
							
								    if(std::uncaught_exception())
							 | 
						||
| 
								 | 
							
								        return;
							 | 
						||
| 
								 | 
							
								    if(0 == (this->get_flags() & no_header)){
							 | 
						||
| 
								 | 
							
								        gimpl->windup(is);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								} // namespace archive
							 | 
						||
| 
								 | 
							
								} // namespace boost
							 |