123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127 |
- #ifndef BOOST_ARCHIVE_ITERATORS_XML_UNESCAPE_HPP
- #define BOOST_ARCHIVE_ITERATORS_XML_UNESCAPE_HPP
- // MS compatible compilers support #pragma once
- #if defined(_MSC_VER)
- # pragma once
- #endif
- /////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
- // xml_unescape.hpp
- // (C) Copyright 2002 Robert Ramey - http://www.rrsd.com .
- // Use, modification and distribution is subject to 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 <boost/detail/workaround.hpp>
- #include <boost/assert.hpp>
- #include <boost/serialization/throw_exception.hpp>
- #include <boost/archive/iterators/unescape.hpp>
- #include <boost/archive/iterators/dataflow_exception.hpp>
- namespace boost {
- namespace archive {
- namespace iterators {
- /////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
- // replace &??? xml escape sequences with the corresponding characters
- template<class Base>
- class xml_unescape
- : public unescape<xml_unescape<Base>, Base>
- {
- friend class boost::iterator_core_access;
- typedef xml_unescape<Base> this_t;
- typedef unescape<this_t, Base> super_t;
- typedef typename boost::iterator_reference<this_t> reference_type;
- reference_type dereference() const {
- return unescape<xml_unescape<Base>, Base>::dereference();
- }
- public:
- // msvc versions prior to 14.0 crash with and ICE
- #if BOOST_WORKAROUND(BOOST_MSVC, < 1900)
- typedef int value_type;
- #else
- typedef typename super_t::value_type value_type;
- #endif
- void drain_residue(const char *literal);
- value_type drain();
- template<class T>
- xml_unescape(T start) :
- super_t(Base(static_cast< T >(start)))
- {}
- // intel 7.1 doesn't like default copy constructor
- xml_unescape(const xml_unescape & rhs) :
- super_t(rhs.base_reference())
- {}
- };
- template<class Base>
- void xml_unescape<Base>::drain_residue(const char * literal){
- do{
- if(* literal != * ++(this->base_reference()))
- boost::serialization::throw_exception(
- dataflow_exception(
- dataflow_exception::invalid_xml_escape_sequence
- )
- );
- }
- while('\0' != * ++literal);
- }
- // note key constraint on this function is that can't "look ahead" any
- // more than necessary into base iterator. Doing so would alter the base
- // iterator refenence which would make subsequent iterator comparisons
- // incorrect and thereby break the composiblity of iterators.
- template<class Base>
- typename xml_unescape<Base>::value_type
- //int
- xml_unescape<Base>::drain(){
- value_type retval = * this->base_reference();
- if('&' != retval){
- return retval;
- }
- retval = * ++(this->base_reference());
- switch(retval){
- case 'l': // <
- drain_residue("t;");
- retval = '<';
- break;
- case 'g': // >
- drain_residue("t;");
- retval = '>';
- break;
- case 'a':
- retval = * ++(this->base_reference());
- switch(retval){
- case 'p': // '
- drain_residue("os;");
- retval = '\'';
- break;
- case 'm': // &
- drain_residue("p;");
- retval = '&';
- break;
- }
- break;
- case 'q':
- drain_residue("uot;");
- retval = '"';
- break;
- }
- return retval;
- }
- } // namespace iterators
- } // namespace archive
- } // namespace boost
- #endif // BOOST_ARCHIVE_ITERATORS_XML_UNESCAPE_HPP
|