xref: /reactos/sdk/tools/xml2sdb/tinyxml2.cpp (revision 34593d93)
1 /*
2 Original code by Lee Thomason (www.grinninglizard.com)
3 
4 This software is provided 'as-is', without any express or implied
5 warranty. In no event will the authors be held liable for any
6 damages arising from the use of this software.
7 
8 Permission is granted to anyone to use this software for any
9 purpose, including commercial applications, and to alter it and
10 redistribute it freely, subject to the following restrictions:
11 
12 1. The origin of this software must not be misrepresented; you must
13 not claim that you wrote the original software. If you use this
14 software in a product, an acknowledgment in the product documentation
15 would be appreciated but is not required.
16 
17 2. Altered source versions must be plainly marked as such, and
18 must not be misrepresented as being the original software.
19 
20 3. This notice may not be removed or altered from any source
21 distribution.
22 */
23 
24 #include "tinyxml2.h"
25 
26 #include <new>		// yes, this one new style header, is in the Android SDK.
27 #if defined(ANDROID_NDK) || defined(__BORLANDC__) || defined(__QNXNTO__)
28 #   include <stddef.h>
29 #   include <stdarg.h>
30 #else
31 #   include <cstddef>
32 #   include <cstdarg>
33 #endif
34 
35 #if defined(_MSC_VER) && (_MSC_VER >= 1400 ) && (!defined WINCE)
36 	// Microsoft Visual Studio, version 2005 and higher. Not WinCE.
37 	/*int _snprintf_s(
38 	   char *buffer,
39 	   size_t sizeOfBuffer,
40 	   size_t count,
41 	   const char *format [,
42 		  argument] ...
43 	);*/
TIXML_SNPRINTF(char * buffer,size_t size,const char * format,...)44 	static inline int TIXML_SNPRINTF( char* buffer, size_t size, const char* format, ... )
45 	{
46 		va_list va;
47 		va_start( va, format );
48 		int result = vsnprintf_s( buffer, size, _TRUNCATE, format, va );
49 		va_end( va );
50 		return result;
51 	}
52 
TIXML_VSNPRINTF(char * buffer,size_t size,const char * format,va_list va)53 	static inline int TIXML_VSNPRINTF( char* buffer, size_t size, const char* format, va_list va )
54 	{
55 		int result = vsnprintf_s( buffer, size, _TRUNCATE, format, va );
56 		return result;
57 	}
58 
59 	#define TIXML_VSCPRINTF	_vscprintf
60 	#define TIXML_SSCANF	sscanf_s
61 #elif defined _MSC_VER
62 	// Microsoft Visual Studio 2003 and earlier or WinCE
63 	#define TIXML_SNPRINTF	_snprintf
64 	#define TIXML_VSNPRINTF _vsnprintf
65 	#define TIXML_SSCANF	sscanf
66 	#if (_MSC_VER < 1400 ) && (!defined WINCE)
67 		// Microsoft Visual Studio 2003 and not WinCE.
68 		#define TIXML_VSCPRINTF   _vscprintf // VS2003's C runtime has this, but VC6 C runtime or WinCE SDK doesn't have.
69 	#else
70 		// Microsoft Visual Studio 2003 and earlier or WinCE.
TIXML_VSCPRINTF(const char * format,va_list va)71 		static inline int TIXML_VSCPRINTF( const char* format, va_list va )
72 		{
73 			int len = 512;
74 			for (;;) {
75 				len = len*2;
76 				char* str = new char[len]();
77 				const int required = _vsnprintf(str, len, format, va);
78 				delete[] str;
79 				if ( required != -1 ) {
80 					TIXMLASSERT( required >= 0 );
81 					len = required;
82 					break;
83 				}
84 			}
85 			TIXMLASSERT( len >= 0 );
86 			return len;
87 		}
88 	#endif
89 #else
90 	// GCC version 3 and higher
91 	//#warning( "Using sn* functions." )
92 	#define TIXML_SNPRINTF	snprintf
93 	#define TIXML_VSNPRINTF	vsnprintf
TIXML_VSCPRINTF(const char * format,va_list va)94 	static inline int TIXML_VSCPRINTF( const char* format, va_list va )
95 	{
96 		int len = vsnprintf( 0, 0, format, va );
97 		TIXMLASSERT( len >= 0 );
98 		return len;
99 	}
100 	#define TIXML_SSCANF   sscanf
101 #endif
102 
103 
104 static const char LINE_FEED				= (char)0x0a;			// all line endings are normalized to LF
105 static const char LF = LINE_FEED;
106 static const char CARRIAGE_RETURN		= (char)0x0d;			// CR gets filtered out
107 static const char CR = CARRIAGE_RETURN;
108 static const char SINGLE_QUOTE			= '\'';
109 static const char DOUBLE_QUOTE			= '\"';
110 
111 // Bunch of unicode info at:
112 //		http://www.unicode.org/faq/utf_bom.html
113 //	ef bb bf (Microsoft "lead bytes") - designates UTF-8
114 
115 static const unsigned char TIXML_UTF_LEAD_0 = 0xefU;
116 static const unsigned char TIXML_UTF_LEAD_1 = 0xbbU;
117 static const unsigned char TIXML_UTF_LEAD_2 = 0xbfU;
118 
119 namespace tinyxml2
120 {
121 
122 struct Entity {
123     const char* pattern;
124     int length;
125     char value;
126 };
127 
128 static const int NUM_ENTITIES = 5;
129 static const Entity entities[NUM_ENTITIES] = {
130     { "quot", 4,	DOUBLE_QUOTE },
131     { "amp", 3,		'&'  },
132     { "apos", 4,	SINGLE_QUOTE },
133     { "lt",	2, 		'<'	 },
134     { "gt",	2,		'>'	 }
135 };
136 
137 
~StrPair()138 StrPair::~StrPair()
139 {
140     Reset();
141 }
142 
143 
TransferTo(StrPair * other)144 void StrPair::TransferTo( StrPair* other )
145 {
146     if ( this == other ) {
147         return;
148     }
149     // This in effect implements the assignment operator by "moving"
150     // ownership (as in auto_ptr).
151 
152     TIXMLASSERT( other->_flags == 0 );
153     TIXMLASSERT( other->_start == 0 );
154     TIXMLASSERT( other->_end == 0 );
155 
156     other->Reset();
157 
158     other->_flags = _flags;
159     other->_start = _start;
160     other->_end = _end;
161 
162     _flags = 0;
163     _start = 0;
164     _end = 0;
165 }
166 
Reset()167 void StrPair::Reset()
168 {
169     if ( _flags & NEEDS_DELETE ) {
170         delete [] _start;
171     }
172     _flags = 0;
173     _start = 0;
174     _end = 0;
175 }
176 
177 
SetStr(const char * str,int flags)178 void StrPair::SetStr( const char* str, int flags )
179 {
180     TIXMLASSERT( str );
181     Reset();
182     size_t len = strlen( str );
183     TIXMLASSERT( _start == 0 );
184     _start = new char[ len+1 ];
185     memcpy( _start, str, len+1 );
186     _end = _start + len;
187     _flags = flags | NEEDS_DELETE;
188 }
189 
190 
ParseText(char * p,const char * endTag,int strFlags)191 char* StrPair::ParseText( char* p, const char* endTag, int strFlags )
192 {
193     TIXMLASSERT( endTag && *endTag );
194 
195     char* start = p;
196     char  endChar = *endTag;
197     size_t length = strlen( endTag );
198 
199     // Inner loop of text parsing.
200     while ( *p ) {
201         if ( *p == endChar && strncmp( p, endTag, length ) == 0 ) {
202             Set( start, p, strFlags );
203             return p + length;
204         }
205         ++p;
206     }
207     return 0;
208 }
209 
210 
ParseName(char * p)211 char* StrPair::ParseName( char* p )
212 {
213     if ( !p || !(*p) ) {
214         return 0;
215     }
216     if ( !XMLUtil::IsNameStartChar( *p ) ) {
217         return 0;
218     }
219 
220     char* const start = p;
221     ++p;
222     while ( *p && XMLUtil::IsNameChar( *p ) ) {
223         ++p;
224     }
225 
226     Set( start, p, 0 );
227     return p;
228 }
229 
230 
CollapseWhitespace()231 void StrPair::CollapseWhitespace()
232 {
233     // Adjusting _start would cause undefined behavior on delete[]
234     TIXMLASSERT( ( _flags & NEEDS_DELETE ) == 0 );
235     // Trim leading space.
236     _start = XMLUtil::SkipWhiteSpace( _start );
237 
238     if ( *_start ) {
239         char* p = _start;	// the read pointer
240         char* q = _start;	// the write pointer
241 
242         while( *p ) {
243             if ( XMLUtil::IsWhiteSpace( *p )) {
244                 p = XMLUtil::SkipWhiteSpace( p );
245                 if ( *p == 0 ) {
246                     break;    // don't write to q; this trims the trailing space.
247                 }
248                 *q = ' ';
249                 ++q;
250             }
251             *q = *p;
252             ++q;
253             ++p;
254         }
255         *q = 0;
256     }
257 }
258 
259 
GetStr()260 const char* StrPair::GetStr()
261 {
262     TIXMLASSERT( _start );
263     TIXMLASSERT( _end );
264     if ( _flags & NEEDS_FLUSH ) {
265         *_end = 0;
266         _flags ^= NEEDS_FLUSH;
267 
268         if ( _flags ) {
269             char* p = _start;	// the read pointer
270             char* q = _start;	// the write pointer
271 
272             while( p < _end ) {
273                 if ( (_flags & NEEDS_NEWLINE_NORMALIZATION) && *p == CR ) {
274                     // CR-LF pair becomes LF
275                     // CR alone becomes LF
276                     // LF-CR becomes LF
277                     if ( *(p+1) == LF ) {
278                         p += 2;
279                     }
280                     else {
281                         ++p;
282                     }
283                     *q++ = LF;
284                 }
285                 else if ( (_flags & NEEDS_NEWLINE_NORMALIZATION) && *p == LF ) {
286                     if ( *(p+1) == CR ) {
287                         p += 2;
288                     }
289                     else {
290                         ++p;
291                     }
292                     *q++ = LF;
293                 }
294                 else if ( (_flags & NEEDS_ENTITY_PROCESSING) && *p == '&' ) {
295                     // Entities handled by tinyXML2:
296                     // - special entities in the entity table [in/out]
297                     // - numeric character reference [in]
298                     //   &#20013; or &#x4e2d;
299 
300                     if ( *(p+1) == '#' ) {
301                         const int buflen = 10;
302                         char buf[buflen] = { 0 };
303                         int len = 0;
304                         char* adjusted = const_cast<char*>( XMLUtil::GetCharacterRef( p, buf, &len ) );
305                         if ( adjusted == 0 ) {
306                             *q = *p;
307                             ++p;
308                             ++q;
309                         }
310                         else {
311                             TIXMLASSERT( 0 <= len && len <= buflen );
312                             TIXMLASSERT( q + len <= adjusted );
313                             p = adjusted;
314                             memcpy( q, buf, len );
315                             q += len;
316                         }
317                     }
318                     else {
319                         bool entityFound = false;
320                         for( int i = 0; i < NUM_ENTITIES; ++i ) {
321                             const Entity& entity = entities[i];
322                             if ( strncmp( p + 1, entity.pattern, entity.length ) == 0
323                                     && *( p + entity.length + 1 ) == ';' ) {
324                                 // Found an entity - convert.
325                                 *q = entity.value;
326                                 ++q;
327                                 p += entity.length + 2;
328                                 entityFound = true;
329                                 break;
330                             }
331                         }
332                         if ( !entityFound ) {
333                             // fixme: treat as error?
334                             ++p;
335                             ++q;
336                         }
337                     }
338                 }
339                 else {
340                     *q = *p;
341                     ++p;
342                     ++q;
343                 }
344             }
345             *q = 0;
346         }
347         // The loop below has plenty going on, and this
348         // is a less useful mode. Break it out.
349         if ( _flags & NEEDS_WHITESPACE_COLLAPSING ) {
350             CollapseWhitespace();
351         }
352         _flags = (_flags & NEEDS_DELETE);
353     }
354     TIXMLASSERT( _start );
355     return _start;
356 }
357 
358 
359 
360 
361 // --------- XMLUtil ----------- //
362 
ReadBOM(const char * p,bool * bom)363 const char* XMLUtil::ReadBOM( const char* p, bool* bom )
364 {
365     TIXMLASSERT( p );
366     TIXMLASSERT( bom );
367     *bom = false;
368     const unsigned char* pu = reinterpret_cast<const unsigned char*>(p);
369     // Check for BOM:
370     if (    *(pu+0) == TIXML_UTF_LEAD_0
371             && *(pu+1) == TIXML_UTF_LEAD_1
372             && *(pu+2) == TIXML_UTF_LEAD_2 ) {
373         *bom = true;
374         p += 3;
375     }
376     TIXMLASSERT( p );
377     return p;
378 }
379 
380 
ConvertUTF32ToUTF8(unsigned long input,char * output,int * length)381 void XMLUtil::ConvertUTF32ToUTF8( unsigned long input, char* output, int* length )
382 {
383     const unsigned long BYTE_MASK = 0xBF;
384     const unsigned long BYTE_MARK = 0x80;
385     const unsigned long FIRST_BYTE_MARK[7] = { 0x00, 0x00, 0xC0, 0xE0, 0xF0, 0xF8, 0xFC };
386 
387     if (input < 0x80) {
388         *length = 1;
389     }
390     else if ( input < 0x800 ) {
391         *length = 2;
392     }
393     else if ( input < 0x10000 ) {
394         *length = 3;
395     }
396     else if ( input < 0x200000 ) {
397         *length = 4;
398     }
399     else {
400         *length = 0;    // This code won't convert this correctly anyway.
401         return;
402     }
403 
404     output += *length;
405 
406     // Scary scary fall throughs.
407     switch (*length) {
408         case 4:
409             --output;
410             *output = (char)((input | BYTE_MARK) & BYTE_MASK);
411             input >>= 6;
412         case 3:
413             --output;
414             *output = (char)((input | BYTE_MARK) & BYTE_MASK);
415             input >>= 6;
416         case 2:
417             --output;
418             *output = (char)((input | BYTE_MARK) & BYTE_MASK);
419             input >>= 6;
420         case 1:
421             --output;
422             *output = (char)(input | FIRST_BYTE_MARK[*length]);
423             break;
424         default:
425             TIXMLASSERT( false );
426     }
427 }
428 
429 
GetCharacterRef(const char * p,char * value,int * length)430 const char* XMLUtil::GetCharacterRef( const char* p, char* value, int* length )
431 {
432     // Presume an entity, and pull it out.
433     *length = 0;
434 
435     if ( *(p+1) == '#' && *(p+2) ) {
436         unsigned long ucs = 0;
437         TIXMLASSERT( sizeof( ucs ) >= 4 );
438         ptrdiff_t delta = 0;
439         unsigned mult = 1;
440         static const char SEMICOLON = ';';
441 
442         if ( *(p+2) == 'x' ) {
443             // Hexadecimal.
444             const char* q = p+3;
445             if ( !(*q) ) {
446                 return 0;
447             }
448 
449             q = strchr( q, SEMICOLON );
450 
451             if ( !q ) {
452                 return 0;
453             }
454             TIXMLASSERT( *q == SEMICOLON );
455 
456             delta = q-p;
457             --q;
458 
459             while ( *q != 'x' ) {
460                 unsigned int digit = 0;
461 
462                 if ( *q >= '0' && *q <= '9' ) {
463                     digit = *q - '0';
464                 }
465                 else if ( *q >= 'a' && *q <= 'f' ) {
466                     digit = *q - 'a' + 10;
467                 }
468                 else if ( *q >= 'A' && *q <= 'F' ) {
469                     digit = *q - 'A' + 10;
470                 }
471                 else {
472                     return 0;
473                 }
474                 TIXMLASSERT( digit < 16 );
475                 TIXMLASSERT( digit == 0 || mult <= UINT_MAX / digit );
476                 const unsigned int digitScaled = mult * digit;
477                 TIXMLASSERT( ucs <= ULONG_MAX - digitScaled );
478                 ucs += digitScaled;
479                 TIXMLASSERT( mult <= UINT_MAX / 16 );
480                 mult *= 16;
481                 --q;
482             }
483         }
484         else {
485             // Decimal.
486             const char* q = p+2;
487             if ( !(*q) ) {
488                 return 0;
489             }
490 
491             q = strchr( q, SEMICOLON );
492 
493             if ( !q ) {
494                 return 0;
495             }
496             TIXMLASSERT( *q == SEMICOLON );
497 
498             delta = q-p;
499             --q;
500 
501             while ( *q != '#' ) {
502                 if ( *q >= '0' && *q <= '9' ) {
503                     const unsigned int digit = *q - '0';
504                     TIXMLASSERT( digit < 10 );
505                     TIXMLASSERT( digit == 0 || mult <= UINT_MAX / digit );
506                     const unsigned int digitScaled = mult * digit;
507                     TIXMLASSERT( ucs <= ULONG_MAX - digitScaled );
508                     ucs += digitScaled;
509                 }
510                 else {
511                     return 0;
512                 }
513                 TIXMLASSERT( mult <= UINT_MAX / 10 );
514                 mult *= 10;
515                 --q;
516             }
517         }
518         // convert the UCS to UTF-8
519         ConvertUTF32ToUTF8( ucs, value, length );
520         return p + delta + 1;
521     }
522     return p+1;
523 }
524 
525 
ToStr(int v,char * buffer,int bufferSize)526 void XMLUtil::ToStr( int v, char* buffer, int bufferSize )
527 {
528     TIXML_SNPRINTF( buffer, bufferSize, "%d", v );
529 }
530 
531 
ToStr(unsigned v,char * buffer,int bufferSize)532 void XMLUtil::ToStr( unsigned v, char* buffer, int bufferSize )
533 {
534     TIXML_SNPRINTF( buffer, bufferSize, "%u", v );
535 }
536 
537 
ToStr(bool v,char * buffer,int bufferSize)538 void XMLUtil::ToStr( bool v, char* buffer, int bufferSize )
539 {
540     TIXML_SNPRINTF( buffer, bufferSize, "%d", v ? 1 : 0 );
541 }
542 
543 /*
544 	ToStr() of a number is a very tricky topic.
545 	https://github.com/leethomason/tinyxml2/issues/106
546 */
ToStr(float v,char * buffer,int bufferSize)547 void XMLUtil::ToStr( float v, char* buffer, int bufferSize )
548 {
549     TIXML_SNPRINTF( buffer, bufferSize, "%.8g", v );
550 }
551 
552 
ToStr(double v,char * buffer,int bufferSize)553 void XMLUtil::ToStr( double v, char* buffer, int bufferSize )
554 {
555     TIXML_SNPRINTF( buffer, bufferSize, "%.17g", v );
556 }
557 
558 
ToInt(const char * str,int * value)559 bool XMLUtil::ToInt( const char* str, int* value )
560 {
561     if ( TIXML_SSCANF( str, "%d", value ) == 1 ) {
562         return true;
563     }
564     return false;
565 }
566 
ToUnsigned(const char * str,unsigned * value)567 bool XMLUtil::ToUnsigned( const char* str, unsigned *value )
568 {
569     if ( TIXML_SSCANF( str, "%u", value ) == 1 ) {
570         return true;
571     }
572     return false;
573 }
574 
ToBool(const char * str,bool * value)575 bool XMLUtil::ToBool( const char* str, bool* value )
576 {
577     int ival = 0;
578     if ( ToInt( str, &ival )) {
579         *value = (ival==0) ? false : true;
580         return true;
581     }
582     if ( StringEqual( str, "true" ) ) {
583         *value = true;
584         return true;
585     }
586     else if ( StringEqual( str, "false" ) ) {
587         *value = false;
588         return true;
589     }
590     return false;
591 }
592 
593 
ToFloat(const char * str,float * value)594 bool XMLUtil::ToFloat( const char* str, float* value )
595 {
596     if ( TIXML_SSCANF( str, "%f", value ) == 1 ) {
597         return true;
598     }
599     return false;
600 }
601 
ToDouble(const char * str,double * value)602 bool XMLUtil::ToDouble( const char* str, double* value )
603 {
604     if ( TIXML_SSCANF( str, "%lf", value ) == 1 ) {
605         return true;
606     }
607     return false;
608 }
609 
610 
Identify(char * p,XMLNode ** node)611 char* XMLDocument::Identify( char* p, XMLNode** node )
612 {
613     TIXMLASSERT( node );
614     TIXMLASSERT( p );
615     char* const start = p;
616     p = XMLUtil::SkipWhiteSpace( p );
617     if( !*p ) {
618         *node = 0;
619         TIXMLASSERT( p );
620         return p;
621     }
622 
623     // These strings define the matching patterns:
624     static const char* xmlHeader		= { "<?" };
625     static const char* commentHeader	= { "<!--" };
626     static const char* cdataHeader		= { "<![CDATA[" };
627     static const char* dtdHeader		= { "<!" };
628     static const char* elementHeader	= { "<" };	// and a header for everything else; check last.
629 
630     static const int xmlHeaderLen		= 2;
631     static const int commentHeaderLen	= 4;
632     static const int cdataHeaderLen		= 9;
633     static const int dtdHeaderLen		= 2;
634     static const int elementHeaderLen	= 1;
635 
636     TIXMLASSERT( sizeof( XMLComment ) == sizeof( XMLUnknown ) );		// use same memory pool
637     TIXMLASSERT( sizeof( XMLComment ) == sizeof( XMLDeclaration ) );	// use same memory pool
638     XMLNode* returnNode = 0;
639     if ( XMLUtil::StringEqual( p, xmlHeader, xmlHeaderLen ) ) {
640         TIXMLASSERT( sizeof( XMLDeclaration ) == _commentPool.ItemSize() );
641         returnNode = new (_commentPool.Alloc()) XMLDeclaration( this );
642         returnNode->_memPool = &_commentPool;
643         p += xmlHeaderLen;
644     }
645     else if ( XMLUtil::StringEqual( p, commentHeader, commentHeaderLen ) ) {
646         TIXMLASSERT( sizeof( XMLComment ) == _commentPool.ItemSize() );
647         returnNode = new (_commentPool.Alloc()) XMLComment( this );
648         returnNode->_memPool = &_commentPool;
649         p += commentHeaderLen;
650     }
651     else if ( XMLUtil::StringEqual( p, cdataHeader, cdataHeaderLen ) ) {
652         TIXMLASSERT( sizeof( XMLText ) == _textPool.ItemSize() );
653         XMLText* text = new (_textPool.Alloc()) XMLText( this );
654         returnNode = text;
655         returnNode->_memPool = &_textPool;
656         p += cdataHeaderLen;
657         text->SetCData( true );
658     }
659     else if ( XMLUtil::StringEqual( p, dtdHeader, dtdHeaderLen ) ) {
660         TIXMLASSERT( sizeof( XMLUnknown ) == _commentPool.ItemSize() );
661         returnNode = new (_commentPool.Alloc()) XMLUnknown( this );
662         returnNode->_memPool = &_commentPool;
663         p += dtdHeaderLen;
664     }
665     else if ( XMLUtil::StringEqual( p, elementHeader, elementHeaderLen ) ) {
666         TIXMLASSERT( sizeof( XMLElement ) == _elementPool.ItemSize() );
667         returnNode = new (_elementPool.Alloc()) XMLElement( this );
668         returnNode->_memPool = &_elementPool;
669         p += elementHeaderLen;
670     }
671     else {
672         TIXMLASSERT( sizeof( XMLText ) == _textPool.ItemSize() );
673         returnNode = new (_textPool.Alloc()) XMLText( this );
674         returnNode->_memPool = &_textPool;
675         p = start;	// Back it up, all the text counts.
676     }
677 
678     TIXMLASSERT( returnNode );
679     TIXMLASSERT( p );
680     *node = returnNode;
681     return p;
682 }
683 
684 
Accept(XMLVisitor * visitor) const685 bool XMLDocument::Accept( XMLVisitor* visitor ) const
686 {
687     TIXMLASSERT( visitor );
688     if ( visitor->VisitEnter( *this ) ) {
689         for ( const XMLNode* node=FirstChild(); node; node=node->NextSibling() ) {
690             if ( !node->Accept( visitor ) ) {
691                 break;
692             }
693         }
694     }
695     return visitor->VisitExit( *this );
696 }
697 
698 
699 // --------- XMLNode ----------- //
700 
XMLNode(XMLDocument * doc)701 XMLNode::XMLNode( XMLDocument* doc ) :
702     _document( doc ),
703     _parent( 0 ),
704     _firstChild( 0 ), _lastChild( 0 ),
705     _prev( 0 ), _next( 0 ),
706     _memPool( 0 )
707 {
708 }
709 
710 
~XMLNode()711 XMLNode::~XMLNode()
712 {
713     DeleteChildren();
714     if ( _parent ) {
715         _parent->Unlink( this );
716     }
717 }
718 
Value() const719 const char* XMLNode::Value() const
720 {
721     // Catch an edge case: XMLDocuments don't have a a Value. Carefully return nullptr.
722     if ( this->ToDocument() )
723         return 0;
724     return _value.GetStr();
725 }
726 
SetValue(const char * str,bool staticMem)727 void XMLNode::SetValue( const char* str, bool staticMem )
728 {
729     if ( staticMem ) {
730         _value.SetInternedStr( str );
731     }
732     else {
733         _value.SetStr( str );
734     }
735 }
736 
737 
DeleteChildren()738 void XMLNode::DeleteChildren()
739 {
740     while( _firstChild ) {
741         TIXMLASSERT( _lastChild );
742         TIXMLASSERT( _firstChild->_document == _document );
743         XMLNode* node = _firstChild;
744         Unlink( node );
745 
746         DeleteNode( node );
747     }
748     _firstChild = _lastChild = 0;
749 }
750 
751 
Unlink(XMLNode * child)752 void XMLNode::Unlink( XMLNode* child )
753 {
754     TIXMLASSERT( child );
755     TIXMLASSERT( child->_document == _document );
756     TIXMLASSERT( child->_parent == this );
757     if ( child == _firstChild ) {
758         _firstChild = _firstChild->_next;
759     }
760     if ( child == _lastChild ) {
761         _lastChild = _lastChild->_prev;
762     }
763 
764     if ( child->_prev ) {
765         child->_prev->_next = child->_next;
766     }
767     if ( child->_next ) {
768         child->_next->_prev = child->_prev;
769     }
770 	child->_parent = 0;
771 }
772 
773 
DeleteChild(XMLNode * node)774 void XMLNode::DeleteChild( XMLNode* node )
775 {
776     TIXMLASSERT( node );
777     TIXMLASSERT( node->_document == _document );
778     TIXMLASSERT( node->_parent == this );
779     Unlink( node );
780     DeleteNode( node );
781 }
782 
783 
InsertEndChild(XMLNode * addThis)784 XMLNode* XMLNode::InsertEndChild( XMLNode* addThis )
785 {
786     TIXMLASSERT( addThis );
787     if ( addThis->_document != _document ) {
788         TIXMLASSERT( false );
789         return 0;
790     }
791     InsertChildPreamble( addThis );
792 
793     if ( _lastChild ) {
794         TIXMLASSERT( _firstChild );
795         TIXMLASSERT( _lastChild->_next == 0 );
796         _lastChild->_next = addThis;
797         addThis->_prev = _lastChild;
798         _lastChild = addThis;
799 
800         addThis->_next = 0;
801     }
802     else {
803         TIXMLASSERT( _firstChild == 0 );
804         _firstChild = _lastChild = addThis;
805 
806         addThis->_prev = 0;
807         addThis->_next = 0;
808     }
809     addThis->_parent = this;
810     return addThis;
811 }
812 
813 
InsertFirstChild(XMLNode * addThis)814 XMLNode* XMLNode::InsertFirstChild( XMLNode* addThis )
815 {
816     TIXMLASSERT( addThis );
817     if ( addThis->_document != _document ) {
818         TIXMLASSERT( false );
819         return 0;
820     }
821     InsertChildPreamble( addThis );
822 
823     if ( _firstChild ) {
824         TIXMLASSERT( _lastChild );
825         TIXMLASSERT( _firstChild->_prev == 0 );
826 
827         _firstChild->_prev = addThis;
828         addThis->_next = _firstChild;
829         _firstChild = addThis;
830 
831         addThis->_prev = 0;
832     }
833     else {
834         TIXMLASSERT( _lastChild == 0 );
835         _firstChild = _lastChild = addThis;
836 
837         addThis->_prev = 0;
838         addThis->_next = 0;
839     }
840     addThis->_parent = this;
841     return addThis;
842 }
843 
844 
InsertAfterChild(XMLNode * afterThis,XMLNode * addThis)845 XMLNode* XMLNode::InsertAfterChild( XMLNode* afterThis, XMLNode* addThis )
846 {
847     TIXMLASSERT( addThis );
848     if ( addThis->_document != _document ) {
849         TIXMLASSERT( false );
850         return 0;
851     }
852 
853     TIXMLASSERT( afterThis );
854 
855     if ( afterThis->_parent != this ) {
856         TIXMLASSERT( false );
857         return 0;
858     }
859 
860     if ( afterThis->_next == 0 ) {
861         // The last node or the only node.
862         return InsertEndChild( addThis );
863     }
864     InsertChildPreamble( addThis );
865     addThis->_prev = afterThis;
866     addThis->_next = afterThis->_next;
867     afterThis->_next->_prev = addThis;
868     afterThis->_next = addThis;
869     addThis->_parent = this;
870     return addThis;
871 }
872 
873 
874 
875 
FirstChildElement(const char * name) const876 const XMLElement* XMLNode::FirstChildElement( const char* name ) const
877 {
878     for( const XMLNode* node = _firstChild; node; node = node->_next ) {
879         const XMLElement* element = node->ToElement();
880         if ( element ) {
881             if ( !name || XMLUtil::StringEqual( element->Name(), name ) ) {
882                 return element;
883             }
884         }
885     }
886     return 0;
887 }
888 
889 
LastChildElement(const char * name) const890 const XMLElement* XMLNode::LastChildElement( const char* name ) const
891 {
892     for( const XMLNode* node = _lastChild; node; node = node->_prev ) {
893         const XMLElement* element = node->ToElement();
894         if ( element ) {
895             if ( !name || XMLUtil::StringEqual( element->Name(), name ) ) {
896                 return element;
897             }
898         }
899     }
900     return 0;
901 }
902 
903 
NextSiblingElement(const char * name) const904 const XMLElement* XMLNode::NextSiblingElement( const char* name ) const
905 {
906     for( const XMLNode* node = _next; node; node = node->_next ) {
907         const XMLElement* element = node->ToElement();
908         if ( element
909                 && (!name || XMLUtil::StringEqual( name, element->Name() ))) {
910             return element;
911         }
912     }
913     return 0;
914 }
915 
916 
PreviousSiblingElement(const char * name) const917 const XMLElement* XMLNode::PreviousSiblingElement( const char* name ) const
918 {
919     for( const XMLNode* node = _prev; node; node = node->_prev ) {
920         const XMLElement* element = node->ToElement();
921         if ( element
922                 && (!name || XMLUtil::StringEqual( name, element->Name() ))) {
923             return element;
924         }
925     }
926     return 0;
927 }
928 
929 
ParseDeep(char * p,StrPair * parentEnd)930 char* XMLNode::ParseDeep( char* p, StrPair* parentEnd )
931 {
932     // This is a recursive method, but thinking about it "at the current level"
933     // it is a pretty simple flat list:
934     //		<foo/>
935     //		<!-- comment -->
936     //
937     // With a special case:
938     //		<foo>
939     //		</foo>
940     //		<!-- comment -->
941     //
942     // Where the closing element (/foo) *must* be the next thing after the opening
943     // element, and the names must match. BUT the tricky bit is that the closing
944     // element will be read by the child.
945     //
946     // 'endTag' is the end tag for this node, it is returned by a call to a child.
947     // 'parentEnd' is the end tag for the parent, which is filled in and returned.
948 
949     while( p && *p ) {
950         XMLNode* node = 0;
951 
952         p = _document->Identify( p, &node );
953         if ( node == 0 ) {
954             break;
955         }
956 
957         StrPair endTag;
958         p = node->ParseDeep( p, &endTag );
959         if ( !p ) {
960             DeleteNode( node );
961             if ( !_document->Error() ) {
962                 _document->SetError( XML_ERROR_PARSING, 0, 0 );
963             }
964             break;
965         }
966 
967         XMLDeclaration* decl = node->ToDeclaration();
968         if ( decl ) {
969                 // A declaration can only be the first child of a document.
970                 // Set error, if document already has children.
971                 if ( !_document->NoChildren() ) {
972                         _document->SetError( XML_ERROR_PARSING_DECLARATION, decl->Value(), 0);
973                         DeleteNode( decl );
974                         break;
975                 }
976         }
977 
978         XMLElement* ele = node->ToElement();
979         if ( ele ) {
980             // We read the end tag. Return it to the parent.
981             if ( ele->ClosingType() == XMLElement::CLOSING ) {
982                 if ( parentEnd ) {
983                     ele->_value.TransferTo( parentEnd );
984                 }
985                 node->_memPool->SetTracked();   // created and then immediately deleted.
986                 DeleteNode( node );
987                 return p;
988             }
989 
990             // Handle an end tag returned to this level.
991             // And handle a bunch of annoying errors.
992             bool mismatch = false;
993             if ( endTag.Empty() ) {
994                 if ( ele->ClosingType() == XMLElement::OPEN ) {
995                     mismatch = true;
996                 }
997             }
998             else {
999                 if ( ele->ClosingType() != XMLElement::OPEN ) {
1000                     mismatch = true;
1001                 }
1002                 else if ( !XMLUtil::StringEqual( endTag.GetStr(), ele->Name() ) ) {
1003                     mismatch = true;
1004                 }
1005             }
1006             if ( mismatch ) {
1007                 _document->SetError( XML_ERROR_MISMATCHED_ELEMENT, ele->Name(), 0 );
1008                 DeleteNode( node );
1009                 break;
1010             }
1011         }
1012         InsertEndChild( node );
1013     }
1014     return 0;
1015 }
1016 
DeleteNode(XMLNode * node)1017 void XMLNode::DeleteNode( XMLNode* node )
1018 {
1019     if ( node == 0 ) {
1020         return;
1021     }
1022     MemPool* pool = node->_memPool;
1023     node->~XMLNode();
1024     pool->Free( node );
1025 }
1026 
InsertChildPreamble(XMLNode * insertThis) const1027 void XMLNode::InsertChildPreamble( XMLNode* insertThis ) const
1028 {
1029     TIXMLASSERT( insertThis );
1030     TIXMLASSERT( insertThis->_document == _document );
1031 
1032     if ( insertThis->_parent )
1033         insertThis->_parent->Unlink( insertThis );
1034     else
1035         insertThis->_memPool->SetTracked();
1036 }
1037 
1038 // --------- XMLText ---------- //
ParseDeep(char * p,StrPair *)1039 char* XMLText::ParseDeep( char* p, StrPair* )
1040 {
1041     const char* start = p;
1042     if ( this->CData() ) {
1043         p = _value.ParseText( p, "]]>", StrPair::NEEDS_NEWLINE_NORMALIZATION );
1044         if ( !p ) {
1045             _document->SetError( XML_ERROR_PARSING_CDATA, start, 0 );
1046         }
1047         return p;
1048     }
1049     else {
1050         int flags = _document->ProcessEntities() ? StrPair::TEXT_ELEMENT : StrPair::TEXT_ELEMENT_LEAVE_ENTITIES;
1051         if ( _document->WhitespaceMode() == COLLAPSE_WHITESPACE ) {
1052             flags |= StrPair::NEEDS_WHITESPACE_COLLAPSING;
1053         }
1054 
1055         p = _value.ParseText( p, "<", flags );
1056         if ( p && *p ) {
1057             return p-1;
1058         }
1059         if ( !p ) {
1060             _document->SetError( XML_ERROR_PARSING_TEXT, start, 0 );
1061         }
1062     }
1063     return 0;
1064 }
1065 
1066 
ShallowClone(XMLDocument * doc) const1067 XMLNode* XMLText::ShallowClone( XMLDocument* doc ) const
1068 {
1069     if ( !doc ) {
1070         doc = _document;
1071     }
1072     XMLText* text = doc->NewText( Value() );	// fixme: this will always allocate memory. Intern?
1073     text->SetCData( this->CData() );
1074     return text;
1075 }
1076 
1077 
ShallowEqual(const XMLNode * compare) const1078 bool XMLText::ShallowEqual( const XMLNode* compare ) const
1079 {
1080     const XMLText* text = compare->ToText();
1081     return ( text && XMLUtil::StringEqual( text->Value(), Value() ) );
1082 }
1083 
1084 
Accept(XMLVisitor * visitor) const1085 bool XMLText::Accept( XMLVisitor* visitor ) const
1086 {
1087     TIXMLASSERT( visitor );
1088     return visitor->Visit( *this );
1089 }
1090 
1091 
1092 // --------- XMLComment ---------- //
1093 
XMLComment(XMLDocument * doc)1094 XMLComment::XMLComment( XMLDocument* doc ) : XMLNode( doc )
1095 {
1096 }
1097 
1098 
~XMLComment()1099 XMLComment::~XMLComment()
1100 {
1101 }
1102 
1103 
ParseDeep(char * p,StrPair *)1104 char* XMLComment::ParseDeep( char* p, StrPair* )
1105 {
1106     // Comment parses as text.
1107     const char* start = p;
1108     p = _value.ParseText( p, "-->", StrPair::COMMENT );
1109     if ( p == 0 ) {
1110         _document->SetError( XML_ERROR_PARSING_COMMENT, start, 0 );
1111     }
1112     return p;
1113 }
1114 
1115 
ShallowClone(XMLDocument * doc) const1116 XMLNode* XMLComment::ShallowClone( XMLDocument* doc ) const
1117 {
1118     if ( !doc ) {
1119         doc = _document;
1120     }
1121     XMLComment* comment = doc->NewComment( Value() );	// fixme: this will always allocate memory. Intern?
1122     return comment;
1123 }
1124 
1125 
ShallowEqual(const XMLNode * compare) const1126 bool XMLComment::ShallowEqual( const XMLNode* compare ) const
1127 {
1128     TIXMLASSERT( compare );
1129     const XMLComment* comment = compare->ToComment();
1130     return ( comment && XMLUtil::StringEqual( comment->Value(), Value() ));
1131 }
1132 
1133 
Accept(XMLVisitor * visitor) const1134 bool XMLComment::Accept( XMLVisitor* visitor ) const
1135 {
1136     TIXMLASSERT( visitor );
1137     return visitor->Visit( *this );
1138 }
1139 
1140 
1141 // --------- XMLDeclaration ---------- //
1142 
XMLDeclaration(XMLDocument * doc)1143 XMLDeclaration::XMLDeclaration( XMLDocument* doc ) : XMLNode( doc )
1144 {
1145 }
1146 
1147 
~XMLDeclaration()1148 XMLDeclaration::~XMLDeclaration()
1149 {
1150     //printf( "~XMLDeclaration\n" );
1151 }
1152 
1153 
ParseDeep(char * p,StrPair *)1154 char* XMLDeclaration::ParseDeep( char* p, StrPair* )
1155 {
1156     // Declaration parses as text.
1157     const char* start = p;
1158     p = _value.ParseText( p, "?>", StrPair::NEEDS_NEWLINE_NORMALIZATION );
1159     if ( p == 0 ) {
1160         _document->SetError( XML_ERROR_PARSING_DECLARATION, start, 0 );
1161     }
1162     return p;
1163 }
1164 
1165 
ShallowClone(XMLDocument * doc) const1166 XMLNode* XMLDeclaration::ShallowClone( XMLDocument* doc ) const
1167 {
1168     if ( !doc ) {
1169         doc = _document;
1170     }
1171     XMLDeclaration* dec = doc->NewDeclaration( Value() );	// fixme: this will always allocate memory. Intern?
1172     return dec;
1173 }
1174 
1175 
ShallowEqual(const XMLNode * compare) const1176 bool XMLDeclaration::ShallowEqual( const XMLNode* compare ) const
1177 {
1178     TIXMLASSERT( compare );
1179     const XMLDeclaration* declaration = compare->ToDeclaration();
1180     return ( declaration && XMLUtil::StringEqual( declaration->Value(), Value() ));
1181 }
1182 
1183 
1184 
Accept(XMLVisitor * visitor) const1185 bool XMLDeclaration::Accept( XMLVisitor* visitor ) const
1186 {
1187     TIXMLASSERT( visitor );
1188     return visitor->Visit( *this );
1189 }
1190 
1191 // --------- XMLUnknown ---------- //
1192 
XMLUnknown(XMLDocument * doc)1193 XMLUnknown::XMLUnknown( XMLDocument* doc ) : XMLNode( doc )
1194 {
1195 }
1196 
1197 
~XMLUnknown()1198 XMLUnknown::~XMLUnknown()
1199 {
1200 }
1201 
1202 
ParseDeep(char * p,StrPair *)1203 char* XMLUnknown::ParseDeep( char* p, StrPair* )
1204 {
1205     // Unknown parses as text.
1206     const char* start = p;
1207 
1208     p = _value.ParseText( p, ">", StrPair::NEEDS_NEWLINE_NORMALIZATION );
1209     if ( !p ) {
1210         _document->SetError( XML_ERROR_PARSING_UNKNOWN, start, 0 );
1211     }
1212     return p;
1213 }
1214 
1215 
ShallowClone(XMLDocument * doc) const1216 XMLNode* XMLUnknown::ShallowClone( XMLDocument* doc ) const
1217 {
1218     if ( !doc ) {
1219         doc = _document;
1220     }
1221     XMLUnknown* text = doc->NewUnknown( Value() );	// fixme: this will always allocate memory. Intern?
1222     return text;
1223 }
1224 
1225 
ShallowEqual(const XMLNode * compare) const1226 bool XMLUnknown::ShallowEqual( const XMLNode* compare ) const
1227 {
1228     TIXMLASSERT( compare );
1229     const XMLUnknown* unknown = compare->ToUnknown();
1230     return ( unknown && XMLUtil::StringEqual( unknown->Value(), Value() ));
1231 }
1232 
1233 
Accept(XMLVisitor * visitor) const1234 bool XMLUnknown::Accept( XMLVisitor* visitor ) const
1235 {
1236     TIXMLASSERT( visitor );
1237     return visitor->Visit( *this );
1238 }
1239 
1240 // --------- XMLAttribute ---------- //
1241 
Name() const1242 const char* XMLAttribute::Name() const
1243 {
1244     return _name.GetStr();
1245 }
1246 
Value() const1247 const char* XMLAttribute::Value() const
1248 {
1249     return _value.GetStr();
1250 }
1251 
ParseDeep(char * p,bool processEntities)1252 char* XMLAttribute::ParseDeep( char* p, bool processEntities )
1253 {
1254     // Parse using the name rules: bug fix, was using ParseText before
1255     p = _name.ParseName( p );
1256     if ( !p || !*p ) {
1257         return 0;
1258     }
1259 
1260     // Skip white space before =
1261     p = XMLUtil::SkipWhiteSpace( p );
1262     if ( *p != '=' ) {
1263         return 0;
1264     }
1265 
1266     ++p;	// move up to opening quote
1267     p = XMLUtil::SkipWhiteSpace( p );
1268     if ( *p != '\"' && *p != '\'' ) {
1269         return 0;
1270     }
1271 
1272     char endTag[2] = { *p, 0 };
1273     ++p;	// move past opening quote
1274 
1275     p = _value.ParseText( p, endTag, processEntities ? StrPair::ATTRIBUTE_VALUE : StrPair::ATTRIBUTE_VALUE_LEAVE_ENTITIES );
1276     return p;
1277 }
1278 
1279 
SetName(const char * n)1280 void XMLAttribute::SetName( const char* n )
1281 {
1282     _name.SetStr( n );
1283 }
1284 
1285 
QueryIntValue(int * value) const1286 XMLError XMLAttribute::QueryIntValue( int* value ) const
1287 {
1288     if ( XMLUtil::ToInt( Value(), value )) {
1289         return XML_NO_ERROR;
1290     }
1291     return XML_WRONG_ATTRIBUTE_TYPE;
1292 }
1293 
1294 
QueryUnsignedValue(unsigned int * value) const1295 XMLError XMLAttribute::QueryUnsignedValue( unsigned int* value ) const
1296 {
1297     if ( XMLUtil::ToUnsigned( Value(), value )) {
1298         return XML_NO_ERROR;
1299     }
1300     return XML_WRONG_ATTRIBUTE_TYPE;
1301 }
1302 
1303 
QueryBoolValue(bool * value) const1304 XMLError XMLAttribute::QueryBoolValue( bool* value ) const
1305 {
1306     if ( XMLUtil::ToBool( Value(), value )) {
1307         return XML_NO_ERROR;
1308     }
1309     return XML_WRONG_ATTRIBUTE_TYPE;
1310 }
1311 
1312 
QueryFloatValue(float * value) const1313 XMLError XMLAttribute::QueryFloatValue( float* value ) const
1314 {
1315     if ( XMLUtil::ToFloat( Value(), value )) {
1316         return XML_NO_ERROR;
1317     }
1318     return XML_WRONG_ATTRIBUTE_TYPE;
1319 }
1320 
1321 
QueryDoubleValue(double * value) const1322 XMLError XMLAttribute::QueryDoubleValue( double* value ) const
1323 {
1324     if ( XMLUtil::ToDouble( Value(), value )) {
1325         return XML_NO_ERROR;
1326     }
1327     return XML_WRONG_ATTRIBUTE_TYPE;
1328 }
1329 
1330 
SetAttribute(const char * v)1331 void XMLAttribute::SetAttribute( const char* v )
1332 {
1333     _value.SetStr( v );
1334 }
1335 
1336 
SetAttribute(int v)1337 void XMLAttribute::SetAttribute( int v )
1338 {
1339     char buf[BUF_SIZE];
1340     XMLUtil::ToStr( v, buf, BUF_SIZE );
1341     _value.SetStr( buf );
1342 }
1343 
1344 
SetAttribute(unsigned v)1345 void XMLAttribute::SetAttribute( unsigned v )
1346 {
1347     char buf[BUF_SIZE];
1348     XMLUtil::ToStr( v, buf, BUF_SIZE );
1349     _value.SetStr( buf );
1350 }
1351 
1352 
SetAttribute(bool v)1353 void XMLAttribute::SetAttribute( bool v )
1354 {
1355     char buf[BUF_SIZE];
1356     XMLUtil::ToStr( v, buf, BUF_SIZE );
1357     _value.SetStr( buf );
1358 }
1359 
SetAttribute(double v)1360 void XMLAttribute::SetAttribute( double v )
1361 {
1362     char buf[BUF_SIZE];
1363     XMLUtil::ToStr( v, buf, BUF_SIZE );
1364     _value.SetStr( buf );
1365 }
1366 
SetAttribute(float v)1367 void XMLAttribute::SetAttribute( float v )
1368 {
1369     char buf[BUF_SIZE];
1370     XMLUtil::ToStr( v, buf, BUF_SIZE );
1371     _value.SetStr( buf );
1372 }
1373 
1374 
1375 // --------- XMLElement ---------- //
XMLElement(XMLDocument * doc)1376 XMLElement::XMLElement( XMLDocument* doc ) : XMLNode( doc ),
1377     _closingType( 0 ),
1378     _rootAttribute( 0 )
1379 {
1380 }
1381 
1382 
~XMLElement()1383 XMLElement::~XMLElement()
1384 {
1385     while( _rootAttribute ) {
1386         XMLAttribute* next = _rootAttribute->_next;
1387         DeleteAttribute( _rootAttribute );
1388         _rootAttribute = next;
1389     }
1390 }
1391 
1392 
FindAttribute(const char * name) const1393 const XMLAttribute* XMLElement::FindAttribute( const char* name ) const
1394 {
1395     for( XMLAttribute* a = _rootAttribute; a; a = a->_next ) {
1396         if ( XMLUtil::StringEqual( a->Name(), name ) ) {
1397             return a;
1398         }
1399     }
1400     return 0;
1401 }
1402 
1403 
Attribute(const char * name,const char * value) const1404 const char* XMLElement::Attribute( const char* name, const char* value ) const
1405 {
1406     const XMLAttribute* a = FindAttribute( name );
1407     if ( !a ) {
1408         return 0;
1409     }
1410     if ( !value || XMLUtil::StringEqual( a->Value(), value )) {
1411         return a->Value();
1412     }
1413     return 0;
1414 }
1415 
1416 
GetText() const1417 const char* XMLElement::GetText() const
1418 {
1419     if ( FirstChild() && FirstChild()->ToText() ) {
1420         return FirstChild()->Value();
1421     }
1422     return 0;
1423 }
1424 
1425 
SetText(const char * inText)1426 void	XMLElement::SetText( const char* inText )
1427 {
1428 	if ( FirstChild() && FirstChild()->ToText() )
1429 		FirstChild()->SetValue( inText );
1430 	else {
1431 		XMLText*	theText = GetDocument()->NewText( inText );
1432 		InsertFirstChild( theText );
1433 	}
1434 }
1435 
1436 
SetText(int v)1437 void XMLElement::SetText( int v )
1438 {
1439     char buf[BUF_SIZE];
1440     XMLUtil::ToStr( v, buf, BUF_SIZE );
1441     SetText( buf );
1442 }
1443 
1444 
SetText(unsigned v)1445 void XMLElement::SetText( unsigned v )
1446 {
1447     char buf[BUF_SIZE];
1448     XMLUtil::ToStr( v, buf, BUF_SIZE );
1449     SetText( buf );
1450 }
1451 
1452 
SetText(bool v)1453 void XMLElement::SetText( bool v )
1454 {
1455     char buf[BUF_SIZE];
1456     XMLUtil::ToStr( v, buf, BUF_SIZE );
1457     SetText( buf );
1458 }
1459 
1460 
SetText(float v)1461 void XMLElement::SetText( float v )
1462 {
1463     char buf[BUF_SIZE];
1464     XMLUtil::ToStr( v, buf, BUF_SIZE );
1465     SetText( buf );
1466 }
1467 
1468 
SetText(double v)1469 void XMLElement::SetText( double v )
1470 {
1471     char buf[BUF_SIZE];
1472     XMLUtil::ToStr( v, buf, BUF_SIZE );
1473     SetText( buf );
1474 }
1475 
1476 
QueryIntText(int * ival) const1477 XMLError XMLElement::QueryIntText( int* ival ) const
1478 {
1479     if ( FirstChild() && FirstChild()->ToText() ) {
1480         const char* t = FirstChild()->Value();
1481         if ( XMLUtil::ToInt( t, ival ) ) {
1482             return XML_SUCCESS;
1483         }
1484         return XML_CAN_NOT_CONVERT_TEXT;
1485     }
1486     return XML_NO_TEXT_NODE;
1487 }
1488 
1489 
QueryUnsignedText(unsigned * uval) const1490 XMLError XMLElement::QueryUnsignedText( unsigned* uval ) const
1491 {
1492     if ( FirstChild() && FirstChild()->ToText() ) {
1493         const char* t = FirstChild()->Value();
1494         if ( XMLUtil::ToUnsigned( t, uval ) ) {
1495             return XML_SUCCESS;
1496         }
1497         return XML_CAN_NOT_CONVERT_TEXT;
1498     }
1499     return XML_NO_TEXT_NODE;
1500 }
1501 
1502 
QueryBoolText(bool * bval) const1503 XMLError XMLElement::QueryBoolText( bool* bval ) const
1504 {
1505     if ( FirstChild() && FirstChild()->ToText() ) {
1506         const char* t = FirstChild()->Value();
1507         if ( XMLUtil::ToBool( t, bval ) ) {
1508             return XML_SUCCESS;
1509         }
1510         return XML_CAN_NOT_CONVERT_TEXT;
1511     }
1512     return XML_NO_TEXT_NODE;
1513 }
1514 
1515 
QueryDoubleText(double * dval) const1516 XMLError XMLElement::QueryDoubleText( double* dval ) const
1517 {
1518     if ( FirstChild() && FirstChild()->ToText() ) {
1519         const char* t = FirstChild()->Value();
1520         if ( XMLUtil::ToDouble( t, dval ) ) {
1521             return XML_SUCCESS;
1522         }
1523         return XML_CAN_NOT_CONVERT_TEXT;
1524     }
1525     return XML_NO_TEXT_NODE;
1526 }
1527 
1528 
QueryFloatText(float * fval) const1529 XMLError XMLElement::QueryFloatText( float* fval ) const
1530 {
1531     if ( FirstChild() && FirstChild()->ToText() ) {
1532         const char* t = FirstChild()->Value();
1533         if ( XMLUtil::ToFloat( t, fval ) ) {
1534             return XML_SUCCESS;
1535         }
1536         return XML_CAN_NOT_CONVERT_TEXT;
1537     }
1538     return XML_NO_TEXT_NODE;
1539 }
1540 
1541 
1542 
FindOrCreateAttribute(const char * name)1543 XMLAttribute* XMLElement::FindOrCreateAttribute( const char* name )
1544 {
1545     XMLAttribute* last = 0;
1546     XMLAttribute* attrib = 0;
1547     for( attrib = _rootAttribute;
1548             attrib;
1549             last = attrib, attrib = attrib->_next ) {
1550         if ( XMLUtil::StringEqual( attrib->Name(), name ) ) {
1551             break;
1552         }
1553     }
1554     if ( !attrib ) {
1555         TIXMLASSERT( sizeof( XMLAttribute ) == _document->_attributePool.ItemSize() );
1556         attrib = new (_document->_attributePool.Alloc() ) XMLAttribute();
1557         attrib->_memPool = &_document->_attributePool;
1558         if ( last ) {
1559             last->_next = attrib;
1560         }
1561         else {
1562             _rootAttribute = attrib;
1563         }
1564         attrib->SetName( name );
1565         attrib->_memPool->SetTracked(); // always created and linked.
1566     }
1567     return attrib;
1568 }
1569 
1570 
DeleteAttribute(const char * name)1571 void XMLElement::DeleteAttribute( const char* name )
1572 {
1573     XMLAttribute* prev = 0;
1574     for( XMLAttribute* a=_rootAttribute; a; a=a->_next ) {
1575         if ( XMLUtil::StringEqual( name, a->Name() ) ) {
1576             if ( prev ) {
1577                 prev->_next = a->_next;
1578             }
1579             else {
1580                 _rootAttribute = a->_next;
1581             }
1582             DeleteAttribute( a );
1583             break;
1584         }
1585         prev = a;
1586     }
1587 }
1588 
1589 
ParseAttributes(char * p)1590 char* XMLElement::ParseAttributes( char* p )
1591 {
1592     const char* start = p;
1593     XMLAttribute* prevAttribute = 0;
1594 
1595     // Read the attributes.
1596     while( p ) {
1597         p = XMLUtil::SkipWhiteSpace( p );
1598         if ( !(*p) ) {
1599             _document->SetError( XML_ERROR_PARSING_ELEMENT, start, Name() );
1600             return 0;
1601         }
1602 
1603         // attribute.
1604         if (XMLUtil::IsNameStartChar( *p ) ) {
1605             TIXMLASSERT( sizeof( XMLAttribute ) == _document->_attributePool.ItemSize() );
1606             XMLAttribute* attrib = new (_document->_attributePool.Alloc() ) XMLAttribute();
1607             attrib->_memPool = &_document->_attributePool;
1608 			attrib->_memPool->SetTracked();
1609 
1610             p = attrib->ParseDeep( p, _document->ProcessEntities() );
1611             if ( !p || Attribute( attrib->Name() ) ) {
1612                 DeleteAttribute( attrib );
1613                 _document->SetError( XML_ERROR_PARSING_ATTRIBUTE, start, p );
1614                 return 0;
1615             }
1616             // There is a minor bug here: if the attribute in the source xml
1617             // document is duplicated, it will not be detected and the
1618             // attribute will be doubly added. However, tracking the 'prevAttribute'
1619             // avoids re-scanning the attribute list. Preferring performance for
1620             // now, may reconsider in the future.
1621             if ( prevAttribute ) {
1622                 prevAttribute->_next = attrib;
1623             }
1624             else {
1625                 _rootAttribute = attrib;
1626             }
1627             prevAttribute = attrib;
1628         }
1629         // end of the tag
1630         else if ( *p == '>' ) {
1631             ++p;
1632             break;
1633         }
1634         // end of the tag
1635         else if ( *p == '/' && *(p+1) == '>' ) {
1636             _closingType = CLOSED;
1637             return p+2;	// done; sealed element.
1638         }
1639         else {
1640             _document->SetError( XML_ERROR_PARSING_ELEMENT, start, p );
1641             return 0;
1642         }
1643     }
1644     return p;
1645 }
1646 
DeleteAttribute(XMLAttribute * attribute)1647 void XMLElement::DeleteAttribute( XMLAttribute* attribute )
1648 {
1649     if ( attribute == 0 ) {
1650         return;
1651     }
1652     MemPool* pool = attribute->_memPool;
1653     attribute->~XMLAttribute();
1654     pool->Free( attribute );
1655 }
1656 
1657 //
1658 //	<ele></ele>
1659 //	<ele>foo<b>bar</b></ele>
1660 //
ParseDeep(char * p,StrPair * strPair)1661 char* XMLElement::ParseDeep( char* p, StrPair* strPair )
1662 {
1663     // Read the element name.
1664     p = XMLUtil::SkipWhiteSpace( p );
1665 
1666     // The closing element is the </element> form. It is
1667     // parsed just like a regular element then deleted from
1668     // the DOM.
1669     if ( *p == '/' ) {
1670         _closingType = CLOSING;
1671         ++p;
1672     }
1673 
1674     p = _value.ParseName( p );
1675     if ( _value.Empty() ) {
1676         return 0;
1677     }
1678 
1679     p = ParseAttributes( p );
1680     if ( !p || !*p || _closingType ) {
1681         return p;
1682     }
1683 
1684     p = XMLNode::ParseDeep( p, strPair );
1685     return p;
1686 }
1687 
1688 
1689 
ShallowClone(XMLDocument * doc) const1690 XMLNode* XMLElement::ShallowClone( XMLDocument* doc ) const
1691 {
1692     if ( !doc ) {
1693         doc = _document;
1694     }
1695     XMLElement* element = doc->NewElement( Value() );					// fixme: this will always allocate memory. Intern?
1696     for( const XMLAttribute* a=FirstAttribute(); a; a=a->Next() ) {
1697         element->SetAttribute( a->Name(), a->Value() );					// fixme: this will always allocate memory. Intern?
1698     }
1699     return element;
1700 }
1701 
1702 
ShallowEqual(const XMLNode * compare) const1703 bool XMLElement::ShallowEqual( const XMLNode* compare ) const
1704 {
1705     TIXMLASSERT( compare );
1706     const XMLElement* other = compare->ToElement();
1707     if ( other && XMLUtil::StringEqual( other->Name(), Name() )) {
1708 
1709         const XMLAttribute* a=FirstAttribute();
1710         const XMLAttribute* b=other->FirstAttribute();
1711 
1712         while ( a && b ) {
1713             if ( !XMLUtil::StringEqual( a->Value(), b->Value() ) ) {
1714                 return false;
1715             }
1716             a = a->Next();
1717             b = b->Next();
1718         }
1719         if ( a || b ) {
1720             // different count
1721             return false;
1722         }
1723         return true;
1724     }
1725     return false;
1726 }
1727 
1728 
Accept(XMLVisitor * visitor) const1729 bool XMLElement::Accept( XMLVisitor* visitor ) const
1730 {
1731     TIXMLASSERT( visitor );
1732     if ( visitor->VisitEnter( *this, _rootAttribute ) ) {
1733         for ( const XMLNode* node=FirstChild(); node; node=node->NextSibling() ) {
1734             if ( !node->Accept( visitor ) ) {
1735                 break;
1736             }
1737         }
1738     }
1739     return visitor->VisitExit( *this );
1740 }
1741 
1742 
1743 // --------- XMLDocument ----------- //
1744 
1745 // Warning: List must match 'enum XMLError'
1746 const char* XMLDocument::_errorNames[XML_ERROR_COUNT] = {
1747     "XML_SUCCESS",
1748     "XML_NO_ATTRIBUTE",
1749     "XML_WRONG_ATTRIBUTE_TYPE",
1750     "XML_ERROR_FILE_NOT_FOUND",
1751     "XML_ERROR_FILE_COULD_NOT_BE_OPENED",
1752     "XML_ERROR_FILE_READ_ERROR",
1753     "XML_ERROR_ELEMENT_MISMATCH",
1754     "XML_ERROR_PARSING_ELEMENT",
1755     "XML_ERROR_PARSING_ATTRIBUTE",
1756     "XML_ERROR_IDENTIFYING_TAG",
1757     "XML_ERROR_PARSING_TEXT",
1758     "XML_ERROR_PARSING_CDATA",
1759     "XML_ERROR_PARSING_COMMENT",
1760     "XML_ERROR_PARSING_DECLARATION",
1761     "XML_ERROR_PARSING_UNKNOWN",
1762     "XML_ERROR_EMPTY_DOCUMENT",
1763     "XML_ERROR_MISMATCHED_ELEMENT",
1764     "XML_ERROR_PARSING",
1765     "XML_CAN_NOT_CONVERT_TEXT",
1766     "XML_NO_TEXT_NODE"
1767 };
1768 
1769 
XMLDocument(bool processEntities,Whitespace whitespace)1770 XMLDocument::XMLDocument( bool processEntities, Whitespace whitespace ) :
1771     XMLNode( 0 ),
1772     _writeBOM( false ),
1773     _processEntities( processEntities ),
1774     _errorID( XML_NO_ERROR ),
1775     _whitespace( whitespace ),
1776     _errorStr1( 0 ),
1777     _errorStr2( 0 ),
1778     _charBuffer( 0 )
1779 {
1780     // avoid VC++ C4355 warning about 'this' in initializer list (C4355 is off by default in VS2012+)
1781     _document = this;
1782 }
1783 
1784 
~XMLDocument()1785 XMLDocument::~XMLDocument()
1786 {
1787     Clear();
1788 }
1789 
1790 
Clear()1791 void XMLDocument::Clear()
1792 {
1793     DeleteChildren();
1794 
1795 #ifdef DEBUG
1796     const bool hadError = Error();
1797 #endif
1798     _errorID = XML_NO_ERROR;
1799     _errorStr1 = 0;
1800     _errorStr2 = 0;
1801 
1802     delete [] _charBuffer;
1803     _charBuffer = 0;
1804 
1805 #if 0
1806     _textPool.Trace( "text" );
1807     _elementPool.Trace( "element" );
1808     _commentPool.Trace( "comment" );
1809     _attributePool.Trace( "attribute" );
1810 #endif
1811 
1812 #ifdef DEBUG
1813     if ( !hadError ) {
1814         TIXMLASSERT( _elementPool.CurrentAllocs()   == _elementPool.Untracked() );
1815         TIXMLASSERT( _attributePool.CurrentAllocs() == _attributePool.Untracked() );
1816         TIXMLASSERT( _textPool.CurrentAllocs()      == _textPool.Untracked() );
1817         TIXMLASSERT( _commentPool.CurrentAllocs()   == _commentPool.Untracked() );
1818     }
1819 #endif
1820 }
1821 
1822 
NewElement(const char * name)1823 XMLElement* XMLDocument::NewElement( const char* name )
1824 {
1825     TIXMLASSERT( sizeof( XMLElement ) == _elementPool.ItemSize() );
1826     XMLElement* ele = new (_elementPool.Alloc()) XMLElement( this );
1827     ele->_memPool = &_elementPool;
1828     ele->SetName( name );
1829     return ele;
1830 }
1831 
1832 
NewComment(const char * str)1833 XMLComment* XMLDocument::NewComment( const char* str )
1834 {
1835     TIXMLASSERT( sizeof( XMLComment ) == _commentPool.ItemSize() );
1836     XMLComment* comment = new (_commentPool.Alloc()) XMLComment( this );
1837     comment->_memPool = &_commentPool;
1838     comment->SetValue( str );
1839     return comment;
1840 }
1841 
1842 
NewText(const char * str)1843 XMLText* XMLDocument::NewText( const char* str )
1844 {
1845     TIXMLASSERT( sizeof( XMLText ) == _textPool.ItemSize() );
1846     XMLText* text = new (_textPool.Alloc()) XMLText( this );
1847     text->_memPool = &_textPool;
1848     text->SetValue( str );
1849     return text;
1850 }
1851 
1852 
NewDeclaration(const char * str)1853 XMLDeclaration* XMLDocument::NewDeclaration( const char* str )
1854 {
1855     TIXMLASSERT( sizeof( XMLDeclaration ) == _commentPool.ItemSize() );
1856     XMLDeclaration* dec = new (_commentPool.Alloc()) XMLDeclaration( this );
1857     dec->_memPool = &_commentPool;
1858     dec->SetValue( str ? str : "xml version=\"1.0\" encoding=\"UTF-8\"" );
1859     return dec;
1860 }
1861 
1862 
NewUnknown(const char * str)1863 XMLUnknown* XMLDocument::NewUnknown( const char* str )
1864 {
1865     TIXMLASSERT( sizeof( XMLUnknown ) == _commentPool.ItemSize() );
1866     XMLUnknown* unk = new (_commentPool.Alloc()) XMLUnknown( this );
1867     unk->_memPool = &_commentPool;
1868     unk->SetValue( str );
1869     return unk;
1870 }
1871 
callfopen(const char * filepath,const char * mode)1872 static FILE* callfopen( const char* filepath, const char* mode )
1873 {
1874     TIXMLASSERT( filepath );
1875     TIXMLASSERT( mode );
1876 #if defined(_MSC_VER) && (_MSC_VER >= 1400 ) && (!defined WINCE)
1877     FILE* fp = 0;
1878     errno_t err = fopen_s( &fp, filepath, mode );
1879     if ( err ) {
1880         return 0;
1881     }
1882 #else
1883     FILE* fp = fopen( filepath, mode );
1884 #endif
1885     return fp;
1886 }
1887 
DeleteNode(XMLNode * node)1888 void XMLDocument::DeleteNode( XMLNode* node )	{
1889     TIXMLASSERT( node );
1890     TIXMLASSERT(node->_document == this );
1891     if (node->_parent) {
1892         node->_parent->DeleteChild( node );
1893     }
1894     else {
1895         // Isn't in the tree.
1896         // Use the parent delete.
1897         // Also, we need to mark it tracked: we 'know'
1898         // it was never used.
1899         node->_memPool->SetTracked();
1900         // Call the static XMLNode version:
1901         XMLNode::DeleteNode(node);
1902     }
1903 }
1904 
1905 
LoadFile(const char * filename)1906 XMLError XMLDocument::LoadFile( const char* filename )
1907 {
1908     Clear();
1909     FILE* fp = callfopen( filename, "rb" );
1910     if ( !fp ) {
1911         SetError( XML_ERROR_FILE_NOT_FOUND, filename, 0 );
1912         return _errorID;
1913     }
1914     LoadFile( fp );
1915     fclose( fp );
1916     return _errorID;
1917 }
1918 
1919 // This is likely overengineered template art to have a check that unsigned long value incremented
1920 // by one still fits into size_t. If size_t type is larger than unsigned long type
1921 // (x86_64-w64-mingw32 target) then the check is redundant and gcc and clang emit
1922 // -Wtype-limits warning. This piece makes the compiler select code with a check when a check
1923 // is useful and code with no check when a check is redundant depending on how size_t and unsigned long
1924 // types sizes relate to each other.
1925 template
1926 <bool = (sizeof(unsigned long) >= sizeof(size_t))>
1927 struct LongFitsIntoSizeTMinusOne {
Fitstinyxml2::LongFitsIntoSizeTMinusOne1928     static bool Fits( unsigned long value )
1929     {
1930         return value < (size_t)-1;
1931     }
1932 };
1933 
1934 template <>
Fits(unsigned long)1935 bool LongFitsIntoSizeTMinusOne<false>::Fits( unsigned long /*value*/ )
1936 {
1937     return true;
1938 }
1939 
LoadFile(FILE * fp)1940 XMLError XMLDocument::LoadFile( FILE* fp )
1941 {
1942     Clear();
1943 
1944     fseek( fp, 0, SEEK_SET );
1945     if ( fgetc( fp ) == EOF && ferror( fp ) != 0 ) {
1946         SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );
1947         return _errorID;
1948     }
1949 
1950     fseek( fp, 0, SEEK_END );
1951     const long filelength = ftell( fp );
1952     fseek( fp, 0, SEEK_SET );
1953     if ( filelength == -1L ) {
1954         SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );
1955         return _errorID;
1956     }
1957     TIXMLASSERT( filelength >= 0 );
1958 
1959     if ( !LongFitsIntoSizeTMinusOne<>::Fits( filelength ) ) {
1960         // Cannot handle files which won't fit in buffer together with null terminator
1961         SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );
1962         return _errorID;
1963     }
1964 
1965     if ( filelength == 0 ) {
1966         SetError( XML_ERROR_EMPTY_DOCUMENT, 0, 0 );
1967         return _errorID;
1968     }
1969 
1970     const size_t size = filelength;
1971     TIXMLASSERT( _charBuffer == 0 );
1972     _charBuffer = new char[size+1];
1973     size_t read = fread( _charBuffer, 1, size, fp );
1974     if ( read != size ) {
1975         SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );
1976         return _errorID;
1977     }
1978 
1979     _charBuffer[size] = 0;
1980 
1981     Parse();
1982     return _errorID;
1983 }
1984 
1985 
SaveFile(const char * filename,bool compact)1986 XMLError XMLDocument::SaveFile( const char* filename, bool compact )
1987 {
1988     FILE* fp = callfopen( filename, "w" );
1989     if ( !fp ) {
1990         SetError( XML_ERROR_FILE_COULD_NOT_BE_OPENED, filename, 0 );
1991         return _errorID;
1992     }
1993     SaveFile(fp, compact);
1994     fclose( fp );
1995     return _errorID;
1996 }
1997 
1998 
SaveFile(FILE * fp,bool compact)1999 XMLError XMLDocument::SaveFile( FILE* fp, bool compact )
2000 {
2001     // Clear any error from the last save, otherwise it will get reported
2002     // for *this* call.
2003     SetError( XML_NO_ERROR, 0, 0 );
2004     XMLPrinter stream( fp, compact );
2005     Print( &stream );
2006     return _errorID;
2007 }
2008 
2009 
Parse(const char * p,size_t len)2010 XMLError XMLDocument::Parse( const char* p, size_t len )
2011 {
2012     Clear();
2013 
2014     if ( len == 0 || !p || !*p ) {
2015         SetError( XML_ERROR_EMPTY_DOCUMENT, 0, 0 );
2016         return _errorID;
2017     }
2018     if ( len == (size_t)(-1) ) {
2019         len = strlen( p );
2020     }
2021     TIXMLASSERT( _charBuffer == 0 );
2022     _charBuffer = new char[ len+1 ];
2023     memcpy( _charBuffer, p, len );
2024     _charBuffer[len] = 0;
2025 
2026     Parse();
2027     if ( Error() ) {
2028         // clean up now essentially dangling memory.
2029         // and the parse fail can put objects in the
2030         // pools that are dead and inaccessible.
2031         DeleteChildren();
2032         _elementPool.Clear();
2033         _attributePool.Clear();
2034         _textPool.Clear();
2035         _commentPool.Clear();
2036     }
2037     return _errorID;
2038 }
2039 
2040 
Print(XMLPrinter * streamer) const2041 void XMLDocument::Print( XMLPrinter* streamer ) const
2042 {
2043     if ( streamer ) {
2044         Accept( streamer );
2045     }
2046     else {
2047         XMLPrinter stdoutStreamer( stdout );
2048         Accept( &stdoutStreamer );
2049     }
2050 }
2051 
2052 
SetError(XMLError error,const char * str1,const char * str2)2053 void XMLDocument::SetError( XMLError error, const char* str1, const char* str2 )
2054 {
2055     TIXMLASSERT( error >= 0 && error < XML_ERROR_COUNT );
2056     _errorID = error;
2057     _errorStr1 = str1;
2058     _errorStr2 = str2;
2059 }
2060 
ErrorName() const2061 const char* XMLDocument::ErrorName() const
2062 {
2063 	TIXMLASSERT( _errorID >= 0 && _errorID < XML_ERROR_COUNT );
2064     const char* errorName = _errorNames[_errorID];
2065     TIXMLASSERT( errorName && errorName[0] );
2066     return errorName;
2067 }
2068 
PrintError() const2069 void XMLDocument::PrintError() const
2070 {
2071     if ( Error() ) {
2072         static const int LEN = 20;
2073         char buf1[LEN] = { 0 };
2074         char buf2[LEN] = { 0 };
2075 
2076         if ( _errorStr1 ) {
2077             TIXML_SNPRINTF( buf1, LEN, "%s", _errorStr1 );
2078         }
2079         if ( _errorStr2 ) {
2080             TIXML_SNPRINTF( buf2, LEN, "%s", _errorStr2 );
2081         }
2082 
2083         // Should check INT_MIN <= _errorID && _errorId <= INT_MAX, but that
2084         // causes a clang "always true" -Wtautological-constant-out-of-range-compare warning
2085         TIXMLASSERT( 0 <= _errorID && XML_ERROR_COUNT - 1 <= INT_MAX );
2086         printf( "XMLDocument error id=%d '%s' str1=%s str2=%s\n",
2087                 static_cast<int>( _errorID ), ErrorName(), buf1, buf2 );
2088     }
2089 }
2090 
Parse()2091 void XMLDocument::Parse()
2092 {
2093     TIXMLASSERT( NoChildren() ); // Clear() must have been called previously
2094     TIXMLASSERT( _charBuffer );
2095     char* p = _charBuffer;
2096     p = XMLUtil::SkipWhiteSpace( p );
2097     p = const_cast<char*>( XMLUtil::ReadBOM( p, &_writeBOM ) );
2098     if ( !*p ) {
2099         SetError( XML_ERROR_EMPTY_DOCUMENT, 0, 0 );
2100         return;
2101     }
2102     ParseDeep(p, 0 );
2103 }
2104 
XMLPrinter(FILE * file,bool compact,int depth)2105 XMLPrinter::XMLPrinter( FILE* file, bool compact, int depth ) :
2106     _elementJustOpened( false ),
2107     _firstElement( true ),
2108     _fp( file ),
2109     _depth( depth ),
2110     _textDepth( -1 ),
2111     _processEntities( true ),
2112     _compactMode( compact )
2113 {
2114     for( int i=0; i<ENTITY_RANGE; ++i ) {
2115         _entityFlag[i] = false;
2116         _restrictedEntityFlag[i] = false;
2117     }
2118     for( int i=0; i<NUM_ENTITIES; ++i ) {
2119         const char entityValue = entities[i].value;
2120         TIXMLASSERT( 0 <= entityValue && entityValue < ENTITY_RANGE );
2121         _entityFlag[ (unsigned char)entityValue ] = true;
2122     }
2123     _restrictedEntityFlag[(unsigned char)'&'] = true;
2124     _restrictedEntityFlag[(unsigned char)'<'] = true;
2125     _restrictedEntityFlag[(unsigned char)'>'] = true;	// not required, but consistency is nice
2126     _buffer.Push( 0 );
2127 }
2128 
2129 
Print(const char * format,...)2130 void XMLPrinter::Print( const char* format, ... )
2131 {
2132     va_list     va;
2133     va_start( va, format );
2134 
2135     if ( _fp ) {
2136         vfprintf( _fp, format, va );
2137     }
2138     else {
2139         const int len = TIXML_VSCPRINTF( format, va );
2140         // Close out and re-start the va-args
2141         va_end( va );
2142         TIXMLASSERT( len >= 0 );
2143         va_start( va, format );
2144         TIXMLASSERT( _buffer.Size() > 0 && _buffer[_buffer.Size() - 1] == 0 );
2145         char* p = _buffer.PushArr( len ) - 1;	// back up over the null terminator.
2146 		TIXML_VSNPRINTF( p, len+1, format, va );
2147     }
2148     va_end( va );
2149 }
2150 
2151 
PrintSpace(int depth)2152 void XMLPrinter::PrintSpace( int depth )
2153 {
2154     for( int i=0; i<depth; ++i ) {
2155         Print( "    " );
2156     }
2157 }
2158 
2159 
PrintString(const char * p,bool restricted)2160 void XMLPrinter::PrintString( const char* p, bool restricted )
2161 {
2162     // Look for runs of bytes between entities to print.
2163     const char* q = p;
2164 
2165     if ( _processEntities ) {
2166         const bool* flag = restricted ? _restrictedEntityFlag : _entityFlag;
2167         while ( *q ) {
2168             TIXMLASSERT( p <= q );
2169             // Remember, char is sometimes signed. (How many times has that bitten me?)
2170             if ( *q > 0 && *q < ENTITY_RANGE ) {
2171                 // Check for entities. If one is found, flush
2172                 // the stream up until the entity, write the
2173                 // entity, and keep looking.
2174                 if ( flag[(unsigned char)(*q)] ) {
2175                     while ( p < q ) {
2176                         const size_t delta = q - p;
2177                         // %.*s accepts type int as "precision"
2178                         const int toPrint = ( INT_MAX < delta ) ? INT_MAX : (int)delta;
2179                         Print( "%.*s", toPrint, p );
2180                         p += toPrint;
2181                     }
2182                     bool entityPatternPrinted = false;
2183                     for( int i=0; i<NUM_ENTITIES; ++i ) {
2184                         if ( entities[i].value == *q ) {
2185                             Print( "&%s;", entities[i].pattern );
2186                             entityPatternPrinted = true;
2187                             break;
2188                         }
2189                     }
2190                     if ( !entityPatternPrinted ) {
2191                         // TIXMLASSERT( entityPatternPrinted ) causes gcc -Wunused-but-set-variable in release
2192                         TIXMLASSERT( false );
2193                     }
2194                     ++p;
2195                 }
2196             }
2197             ++q;
2198             TIXMLASSERT( p <= q );
2199         }
2200     }
2201     // Flush the remaining string. This will be the entire
2202     // string if an entity wasn't found.
2203     TIXMLASSERT( p <= q );
2204     if ( !_processEntities || ( p < q ) ) {
2205         Print( "%s", p );
2206     }
2207 }
2208 
2209 
PushHeader(bool writeBOM,bool writeDec)2210 void XMLPrinter::PushHeader( bool writeBOM, bool writeDec )
2211 {
2212     if ( writeBOM ) {
2213         static const unsigned char bom[] = { TIXML_UTF_LEAD_0, TIXML_UTF_LEAD_1, TIXML_UTF_LEAD_2, 0 };
2214         Print( "%s", bom );
2215     }
2216     if ( writeDec ) {
2217         PushDeclaration( "xml version=\"1.0\"" );
2218     }
2219 }
2220 
2221 
OpenElement(const char * name,bool compactMode)2222 void XMLPrinter::OpenElement( const char* name, bool compactMode )
2223 {
2224     SealElementIfJustOpened();
2225     _stack.Push( name );
2226 
2227     if ( _textDepth < 0 && !_firstElement && !compactMode ) {
2228         Print( "\n" );
2229     }
2230     if ( !compactMode ) {
2231         PrintSpace( _depth );
2232     }
2233 
2234     Print( "<%s", name );
2235     _elementJustOpened = true;
2236     _firstElement = false;
2237     ++_depth;
2238 }
2239 
2240 
PushAttribute(const char * name,const char * value)2241 void XMLPrinter::PushAttribute( const char* name, const char* value )
2242 {
2243     TIXMLASSERT( _elementJustOpened );
2244     Print( " %s=\"", name );
2245     PrintString( value, false );
2246     Print( "\"" );
2247 }
2248 
2249 
PushAttribute(const char * name,int v)2250 void XMLPrinter::PushAttribute( const char* name, int v )
2251 {
2252     char buf[BUF_SIZE];
2253     XMLUtil::ToStr( v, buf, BUF_SIZE );
2254     PushAttribute( name, buf );
2255 }
2256 
2257 
PushAttribute(const char * name,unsigned v)2258 void XMLPrinter::PushAttribute( const char* name, unsigned v )
2259 {
2260     char buf[BUF_SIZE];
2261     XMLUtil::ToStr( v, buf, BUF_SIZE );
2262     PushAttribute( name, buf );
2263 }
2264 
2265 
PushAttribute(const char * name,bool v)2266 void XMLPrinter::PushAttribute( const char* name, bool v )
2267 {
2268     char buf[BUF_SIZE];
2269     XMLUtil::ToStr( v, buf, BUF_SIZE );
2270     PushAttribute( name, buf );
2271 }
2272 
2273 
PushAttribute(const char * name,double v)2274 void XMLPrinter::PushAttribute( const char* name, double v )
2275 {
2276     char buf[BUF_SIZE];
2277     XMLUtil::ToStr( v, buf, BUF_SIZE );
2278     PushAttribute( name, buf );
2279 }
2280 
2281 
CloseElement(bool compactMode)2282 void XMLPrinter::CloseElement( bool compactMode )
2283 {
2284     --_depth;
2285     const char* name = _stack.Pop();
2286 
2287     if ( _elementJustOpened ) {
2288         Print( "/>" );
2289     }
2290     else {
2291         if ( _textDepth < 0 && !compactMode) {
2292             Print( "\n" );
2293             PrintSpace( _depth );
2294         }
2295         Print( "</%s>", name );
2296     }
2297 
2298     if ( _textDepth == _depth ) {
2299         _textDepth = -1;
2300     }
2301     if ( _depth == 0 && !compactMode) {
2302         Print( "\n" );
2303     }
2304     _elementJustOpened = false;
2305 }
2306 
2307 
SealElementIfJustOpened()2308 void XMLPrinter::SealElementIfJustOpened()
2309 {
2310     if ( !_elementJustOpened ) {
2311         return;
2312     }
2313     _elementJustOpened = false;
2314     Print( ">" );
2315 }
2316 
2317 
PushText(const char * text,bool cdata)2318 void XMLPrinter::PushText( const char* text, bool cdata )
2319 {
2320     _textDepth = _depth-1;
2321 
2322     SealElementIfJustOpened();
2323     if ( cdata ) {
2324         Print( "<![CDATA[%s]]>", text );
2325     }
2326     else {
2327         PrintString( text, true );
2328     }
2329 }
2330 
PushText(int value)2331 void XMLPrinter::PushText( int value )
2332 {
2333     char buf[BUF_SIZE];
2334     XMLUtil::ToStr( value, buf, BUF_SIZE );
2335     PushText( buf, false );
2336 }
2337 
2338 
PushText(unsigned value)2339 void XMLPrinter::PushText( unsigned value )
2340 {
2341     char buf[BUF_SIZE];
2342     XMLUtil::ToStr( value, buf, BUF_SIZE );
2343     PushText( buf, false );
2344 }
2345 
2346 
PushText(bool value)2347 void XMLPrinter::PushText( bool value )
2348 {
2349     char buf[BUF_SIZE];
2350     XMLUtil::ToStr( value, buf, BUF_SIZE );
2351     PushText( buf, false );
2352 }
2353 
2354 
PushText(float value)2355 void XMLPrinter::PushText( float value )
2356 {
2357     char buf[BUF_SIZE];
2358     XMLUtil::ToStr( value, buf, BUF_SIZE );
2359     PushText( buf, false );
2360 }
2361 
2362 
PushText(double value)2363 void XMLPrinter::PushText( double value )
2364 {
2365     char buf[BUF_SIZE];
2366     XMLUtil::ToStr( value, buf, BUF_SIZE );
2367     PushText( buf, false );
2368 }
2369 
2370 
PushComment(const char * comment)2371 void XMLPrinter::PushComment( const char* comment )
2372 {
2373     SealElementIfJustOpened();
2374     if ( _textDepth < 0 && !_firstElement && !_compactMode) {
2375         Print( "\n" );
2376         PrintSpace( _depth );
2377     }
2378     _firstElement = false;
2379     Print( "<!--%s-->", comment );
2380 }
2381 
2382 
PushDeclaration(const char * value)2383 void XMLPrinter::PushDeclaration( const char* value )
2384 {
2385     SealElementIfJustOpened();
2386     if ( _textDepth < 0 && !_firstElement && !_compactMode) {
2387         Print( "\n" );
2388         PrintSpace( _depth );
2389     }
2390     _firstElement = false;
2391     Print( "<?%s?>", value );
2392 }
2393 
2394 
PushUnknown(const char * value)2395 void XMLPrinter::PushUnknown( const char* value )
2396 {
2397     SealElementIfJustOpened();
2398     if ( _textDepth < 0 && !_firstElement && !_compactMode) {
2399         Print( "\n" );
2400         PrintSpace( _depth );
2401     }
2402     _firstElement = false;
2403     Print( "<!%s>", value );
2404 }
2405 
2406 
VisitEnter(const XMLDocument & doc)2407 bool XMLPrinter::VisitEnter( const XMLDocument& doc )
2408 {
2409     _processEntities = doc.ProcessEntities();
2410     if ( doc.HasBOM() ) {
2411         PushHeader( true, false );
2412     }
2413     return true;
2414 }
2415 
2416 
VisitEnter(const XMLElement & element,const XMLAttribute * attribute)2417 bool XMLPrinter::VisitEnter( const XMLElement& element, const XMLAttribute* attribute )
2418 {
2419     const XMLElement* parentElem = 0;
2420     if ( element.Parent() ) {
2421         parentElem = element.Parent()->ToElement();
2422     }
2423     const bool compactMode = parentElem ? CompactMode( *parentElem ) : _compactMode;
2424     OpenElement( element.Name(), compactMode );
2425     while ( attribute ) {
2426         PushAttribute( attribute->Name(), attribute->Value() );
2427         attribute = attribute->Next();
2428     }
2429     return true;
2430 }
2431 
2432 
VisitExit(const XMLElement & element)2433 bool XMLPrinter::VisitExit( const XMLElement& element )
2434 {
2435     CloseElement( CompactMode(element) );
2436     return true;
2437 }
2438 
2439 
Visit(const XMLText & text)2440 bool XMLPrinter::Visit( const XMLText& text )
2441 {
2442     PushText( text.Value(), text.CData() );
2443     return true;
2444 }
2445 
2446 
Visit(const XMLComment & comment)2447 bool XMLPrinter::Visit( const XMLComment& comment )
2448 {
2449     PushComment( comment.Value() );
2450     return true;
2451 }
2452 
Visit(const XMLDeclaration & declaration)2453 bool XMLPrinter::Visit( const XMLDeclaration& declaration )
2454 {
2455     PushDeclaration( declaration.Value() );
2456     return true;
2457 }
2458 
2459 
Visit(const XMLUnknown & unknown)2460 bool XMLPrinter::Visit( const XMLUnknown& unknown )
2461 {
2462     PushUnknown( unknown.Value() );
2463     return true;
2464 }
2465 
2466 }   // namespace tinyxml2
2467 
2468