Mercurial > minori
comparison dep/pugixml/src/pugixml.cpp @ 55:d10b6c6b432e
add xml lib, we will need to use it eventually
author | Paper <mrpapersonic@gmail.com> |
---|---|
date | Tue, 26 Sep 2023 12:37:08 -0400 (16 months ago) |
parents | |
children | a45edd073f9e |
comparison
equal
deleted
inserted
replaced
54:466ac9870df9 | 55:d10b6c6b432e |
---|---|
1 /** | |
2 * pugixml parser - version 1.13 | |
3 * -------------------------------------------------------- | |
4 * Copyright (C) 2006-2022, by Arseny Kapoulkine (arseny.kapoulkine@gmail.com) | |
5 * Report bugs and download new versions at https://pugixml.org/ | |
6 * | |
7 * This library is distributed under the MIT License. See notice at the end | |
8 * of this file. | |
9 * | |
10 * This work is based on the pugxml parser, which is: | |
11 * Copyright (C) 2003, by Kristen Wegner (kristen@tima.net) | |
12 */ | |
13 | |
14 #ifndef SOURCE_PUGIXML_CPP | |
15 #define SOURCE_PUGIXML_CPP | |
16 | |
17 #include "pugixml.hpp" | |
18 | |
19 #include <stdlib.h> | |
20 #include <stdio.h> | |
21 #include <string.h> | |
22 #include <assert.h> | |
23 #include <limits.h> | |
24 | |
25 #ifdef PUGIXML_WCHAR_MODE | |
26 # include <wchar.h> | |
27 #endif | |
28 | |
29 #ifndef PUGIXML_NO_XPATH | |
30 # include <math.h> | |
31 # include <float.h> | |
32 #endif | |
33 | |
34 #ifndef PUGIXML_NO_STL | |
35 # include <istream> | |
36 # include <ostream> | |
37 # include <string> | |
38 #endif | |
39 | |
40 // For placement new | |
41 #include <new> | |
42 | |
43 #ifdef _MSC_VER | |
44 # pragma warning(push) | |
45 # pragma warning(disable: 4127) // conditional expression is constant | |
46 # pragma warning(disable: 4324) // structure was padded due to __declspec(align()) | |
47 # pragma warning(disable: 4702) // unreachable code | |
48 # pragma warning(disable: 4996) // this function or variable may be unsafe | |
49 #endif | |
50 | |
51 #if defined(_MSC_VER) && defined(__c2__) | |
52 # pragma clang diagnostic push | |
53 # pragma clang diagnostic ignored "-Wdeprecated" // this function or variable may be unsafe | |
54 #endif | |
55 | |
56 #ifdef __INTEL_COMPILER | |
57 # pragma warning(disable: 177) // function was declared but never referenced | |
58 # pragma warning(disable: 279) // controlling expression is constant | |
59 # pragma warning(disable: 1478 1786) // function was declared "deprecated" | |
60 # pragma warning(disable: 1684) // conversion from pointer to same-sized integral type | |
61 #endif | |
62 | |
63 #if defined(__BORLANDC__) && defined(PUGIXML_HEADER_ONLY) | |
64 # pragma warn -8080 // symbol is declared but never used; disabling this inside push/pop bracket does not make the warning go away | |
65 #endif | |
66 | |
67 #ifdef __BORLANDC__ | |
68 # pragma option push | |
69 # pragma warn -8008 // condition is always false | |
70 # pragma warn -8066 // unreachable code | |
71 #endif | |
72 | |
73 #ifdef __SNC__ | |
74 // Using diag_push/diag_pop does not disable the warnings inside templates due to a compiler bug | |
75 # pragma diag_suppress=178 // function was declared but never referenced | |
76 # pragma diag_suppress=237 // controlling expression is constant | |
77 #endif | |
78 | |
79 #ifdef __TI_COMPILER_VERSION__ | |
80 # pragma diag_suppress 179 // function was declared but never referenced | |
81 #endif | |
82 | |
83 // Inlining controls | |
84 #if defined(_MSC_VER) && _MSC_VER >= 1300 | |
85 # define PUGI__NO_INLINE __declspec(noinline) | |
86 #elif defined(__GNUC__) | |
87 # define PUGI__NO_INLINE __attribute__((noinline)) | |
88 #else | |
89 # define PUGI__NO_INLINE | |
90 #endif | |
91 | |
92 // Branch weight controls | |
93 #if defined(__GNUC__) && !defined(__c2__) | |
94 # define PUGI__UNLIKELY(cond) __builtin_expect(cond, 0) | |
95 #else | |
96 # define PUGI__UNLIKELY(cond) (cond) | |
97 #endif | |
98 | |
99 // Simple static assertion | |
100 #define PUGI__STATIC_ASSERT(cond) { static const char condition_failed[(cond) ? 1 : -1] = {0}; (void)condition_failed[0]; } | |
101 | |
102 // Digital Mars C++ bug workaround for passing char loaded from memory via stack | |
103 #ifdef __DMC__ | |
104 # define PUGI__DMC_VOLATILE volatile | |
105 #else | |
106 # define PUGI__DMC_VOLATILE | |
107 #endif | |
108 | |
109 // Integer sanitizer workaround; we only apply this for clang since gcc8 has no_sanitize but not unsigned-integer-overflow and produces "attribute directive ignored" warnings | |
110 #if defined(__clang__) && defined(__has_attribute) | |
111 # if __has_attribute(no_sanitize) | |
112 # define PUGI__UNSIGNED_OVERFLOW __attribute__((no_sanitize("unsigned-integer-overflow"))) | |
113 # else | |
114 # define PUGI__UNSIGNED_OVERFLOW | |
115 # endif | |
116 #else | |
117 # define PUGI__UNSIGNED_OVERFLOW | |
118 #endif | |
119 | |
120 // Borland C++ bug workaround for not defining ::memcpy depending on header include order (can't always use std::memcpy because some compilers don't have it at all) | |
121 #if defined(__BORLANDC__) && !defined(__MEM_H_USING_LIST) | |
122 using std::memcpy; | |
123 using std::memmove; | |
124 using std::memset; | |
125 #endif | |
126 | |
127 // Some MinGW/GCC versions have headers that erroneously omit LLONG_MIN/LLONG_MAX/ULLONG_MAX definitions from limits.h in some configurations | |
128 #if defined(PUGIXML_HAS_LONG_LONG) && defined(__GNUC__) && !defined(LLONG_MAX) && !defined(LLONG_MIN) && !defined(ULLONG_MAX) | |
129 # define LLONG_MIN (-LLONG_MAX - 1LL) | |
130 # define LLONG_MAX __LONG_LONG_MAX__ | |
131 # define ULLONG_MAX (LLONG_MAX * 2ULL + 1ULL) | |
132 #endif | |
133 | |
134 // In some environments MSVC is a compiler but the CRT lacks certain MSVC-specific features | |
135 #if defined(_MSC_VER) && !defined(__S3E__) && !defined(_WIN32_WCE) | |
136 # define PUGI__MSVC_CRT_VERSION _MSC_VER | |
137 #elif defined(_WIN32_WCE) | |
138 # define PUGI__MSVC_CRT_VERSION 1310 // MSVC7.1 | |
139 #endif | |
140 | |
141 // Not all platforms have snprintf; we define a wrapper that uses snprintf if possible. This only works with buffers with a known size. | |
142 #if __cplusplus >= 201103 | |
143 # define PUGI__SNPRINTF(buf, ...) snprintf(buf, sizeof(buf), __VA_ARGS__) | |
144 #elif defined(PUGI__MSVC_CRT_VERSION) && PUGI__MSVC_CRT_VERSION >= 1400 | |
145 # define PUGI__SNPRINTF(buf, ...) _snprintf_s(buf, _countof(buf), _TRUNCATE, __VA_ARGS__) | |
146 #else | |
147 # define PUGI__SNPRINTF sprintf | |
148 #endif | |
149 | |
150 // We put implementation details into an anonymous namespace in source mode, but have to keep it in non-anonymous namespace in header-only mode to prevent binary bloat. | |
151 #ifdef PUGIXML_HEADER_ONLY | |
152 # define PUGI__NS_BEGIN namespace pugi { namespace impl { | |
153 # define PUGI__NS_END } } | |
154 # define PUGI__FN inline | |
155 # define PUGI__FN_NO_INLINE inline | |
156 #else | |
157 # if defined(_MSC_VER) && _MSC_VER < 1300 // MSVC6 seems to have an amusing bug with anonymous namespaces inside namespaces | |
158 # define PUGI__NS_BEGIN namespace pugi { namespace impl { | |
159 # define PUGI__NS_END } } | |
160 # else | |
161 # define PUGI__NS_BEGIN namespace pugi { namespace impl { namespace { | |
162 # define PUGI__NS_END } } } | |
163 # endif | |
164 # define PUGI__FN | |
165 # define PUGI__FN_NO_INLINE PUGI__NO_INLINE | |
166 #endif | |
167 | |
168 // uintptr_t | |
169 #if (defined(_MSC_VER) && _MSC_VER < 1600) || (defined(__BORLANDC__) && __BORLANDC__ < 0x561) | |
170 namespace pugi | |
171 { | |
172 # ifndef _UINTPTR_T_DEFINED | |
173 typedef size_t uintptr_t; | |
174 # endif | |
175 | |
176 typedef unsigned __int8 uint8_t; | |
177 typedef unsigned __int16 uint16_t; | |
178 typedef unsigned __int32 uint32_t; | |
179 } | |
180 #else | |
181 # include <stdint.h> | |
182 #endif | |
183 | |
184 // Memory allocation | |
185 PUGI__NS_BEGIN | |
186 PUGI__FN void* default_allocate(size_t size) | |
187 { | |
188 return malloc(size); | |
189 } | |
190 | |
191 PUGI__FN void default_deallocate(void* ptr) | |
192 { | |
193 free(ptr); | |
194 } | |
195 | |
196 template <typename T> | |
197 struct xml_memory_management_function_storage | |
198 { | |
199 static allocation_function allocate; | |
200 static deallocation_function deallocate; | |
201 }; | |
202 | |
203 // Global allocation functions are stored in class statics so that in header mode linker deduplicates them | |
204 // Without a template<> we'll get multiple definitions of the same static | |
205 template <typename T> allocation_function xml_memory_management_function_storage<T>::allocate = default_allocate; | |
206 template <typename T> deallocation_function xml_memory_management_function_storage<T>::deallocate = default_deallocate; | |
207 | |
208 typedef xml_memory_management_function_storage<int> xml_memory; | |
209 PUGI__NS_END | |
210 | |
211 // String utilities | |
212 PUGI__NS_BEGIN | |
213 // Get string length | |
214 PUGI__FN size_t strlength(const char_t* s) | |
215 { | |
216 assert(s); | |
217 | |
218 #ifdef PUGIXML_WCHAR_MODE | |
219 return wcslen(s); | |
220 #else | |
221 return strlen(s); | |
222 #endif | |
223 } | |
224 | |
225 // Compare two strings | |
226 PUGI__FN bool strequal(const char_t* src, const char_t* dst) | |
227 { | |
228 assert(src && dst); | |
229 | |
230 #ifdef PUGIXML_WCHAR_MODE | |
231 return wcscmp(src, dst) == 0; | |
232 #else | |
233 return strcmp(src, dst) == 0; | |
234 #endif | |
235 } | |
236 | |
237 // Compare lhs with [rhs_begin, rhs_end) | |
238 PUGI__FN bool strequalrange(const char_t* lhs, const char_t* rhs, size_t count) | |
239 { | |
240 for (size_t i = 0; i < count; ++i) | |
241 if (lhs[i] != rhs[i]) | |
242 return false; | |
243 | |
244 return lhs[count] == 0; | |
245 } | |
246 | |
247 // Get length of wide string, even if CRT lacks wide character support | |
248 PUGI__FN size_t strlength_wide(const wchar_t* s) | |
249 { | |
250 assert(s); | |
251 | |
252 #ifdef PUGIXML_WCHAR_MODE | |
253 return wcslen(s); | |
254 #else | |
255 const wchar_t* end = s; | |
256 while (*end) end++; | |
257 return static_cast<size_t>(end - s); | |
258 #endif | |
259 } | |
260 PUGI__NS_END | |
261 | |
262 // auto_ptr-like object for exception recovery | |
263 PUGI__NS_BEGIN | |
264 template <typename T> struct auto_deleter | |
265 { | |
266 typedef void (*D)(T*); | |
267 | |
268 T* data; | |
269 D deleter; | |
270 | |
271 auto_deleter(T* data_, D deleter_): data(data_), deleter(deleter_) | |
272 { | |
273 } | |
274 | |
275 ~auto_deleter() | |
276 { | |
277 if (data) deleter(data); | |
278 } | |
279 | |
280 T* release() | |
281 { | |
282 T* result = data; | |
283 data = 0; | |
284 return result; | |
285 } | |
286 }; | |
287 PUGI__NS_END | |
288 | |
289 #ifdef PUGIXML_COMPACT | |
290 PUGI__NS_BEGIN | |
291 class compact_hash_table | |
292 { | |
293 public: | |
294 compact_hash_table(): _items(0), _capacity(0), _count(0) | |
295 { | |
296 } | |
297 | |
298 void clear() | |
299 { | |
300 if (_items) | |
301 { | |
302 xml_memory::deallocate(_items); | |
303 _items = 0; | |
304 _capacity = 0; | |
305 _count = 0; | |
306 } | |
307 } | |
308 | |
309 void* find(const void* key) | |
310 { | |
311 if (_capacity == 0) return 0; | |
312 | |
313 item_t* item = get_item(key); | |
314 assert(item); | |
315 assert(item->key == key || (item->key == 0 && item->value == 0)); | |
316 | |
317 return item->value; | |
318 } | |
319 | |
320 void insert(const void* key, void* value) | |
321 { | |
322 assert(_capacity != 0 && _count < _capacity - _capacity / 4); | |
323 | |
324 item_t* item = get_item(key); | |
325 assert(item); | |
326 | |
327 if (item->key == 0) | |
328 { | |
329 _count++; | |
330 item->key = key; | |
331 } | |
332 | |
333 item->value = value; | |
334 } | |
335 | |
336 bool reserve(size_t extra = 16) | |
337 { | |
338 if (_count + extra >= _capacity - _capacity / 4) | |
339 return rehash(_count + extra); | |
340 | |
341 return true; | |
342 } | |
343 | |
344 private: | |
345 struct item_t | |
346 { | |
347 const void* key; | |
348 void* value; | |
349 }; | |
350 | |
351 item_t* _items; | |
352 size_t _capacity; | |
353 | |
354 size_t _count; | |
355 | |
356 bool rehash(size_t count); | |
357 | |
358 item_t* get_item(const void* key) | |
359 { | |
360 assert(key); | |
361 assert(_capacity > 0); | |
362 | |
363 size_t hashmod = _capacity - 1; | |
364 size_t bucket = hash(key) & hashmod; | |
365 | |
366 for (size_t probe = 0; probe <= hashmod; ++probe) | |
367 { | |
368 item_t& probe_item = _items[bucket]; | |
369 | |
370 if (probe_item.key == key || probe_item.key == 0) | |
371 return &probe_item; | |
372 | |
373 // hash collision, quadratic probing | |
374 bucket = (bucket + probe + 1) & hashmod; | |
375 } | |
376 | |
377 assert(false && "Hash table is full"); // unreachable | |
378 return 0; | |
379 } | |
380 | |
381 static PUGI__UNSIGNED_OVERFLOW unsigned int hash(const void* key) | |
382 { | |
383 unsigned int h = static_cast<unsigned int>(reinterpret_cast<uintptr_t>(key) & 0xffffffff); | |
384 | |
385 // MurmurHash3 32-bit finalizer | |
386 h ^= h >> 16; | |
387 h *= 0x85ebca6bu; | |
388 h ^= h >> 13; | |
389 h *= 0xc2b2ae35u; | |
390 h ^= h >> 16; | |
391 | |
392 return h; | |
393 } | |
394 }; | |
395 | |
396 PUGI__FN_NO_INLINE bool compact_hash_table::rehash(size_t count) | |
397 { | |
398 size_t capacity = 32; | |
399 while (count >= capacity - capacity / 4) | |
400 capacity *= 2; | |
401 | |
402 compact_hash_table rt; | |
403 rt._capacity = capacity; | |
404 rt._items = static_cast<item_t*>(xml_memory::allocate(sizeof(item_t) * capacity)); | |
405 | |
406 if (!rt._items) | |
407 return false; | |
408 | |
409 memset(rt._items, 0, sizeof(item_t) * capacity); | |
410 | |
411 for (size_t i = 0; i < _capacity; ++i) | |
412 if (_items[i].key) | |
413 rt.insert(_items[i].key, _items[i].value); | |
414 | |
415 if (_items) | |
416 xml_memory::deallocate(_items); | |
417 | |
418 _capacity = capacity; | |
419 _items = rt._items; | |
420 | |
421 assert(_count == rt._count); | |
422 | |
423 return true; | |
424 } | |
425 | |
426 PUGI__NS_END | |
427 #endif | |
428 | |
429 PUGI__NS_BEGIN | |
430 #ifdef PUGIXML_COMPACT | |
431 static const uintptr_t xml_memory_block_alignment = 4; | |
432 #else | |
433 static const uintptr_t xml_memory_block_alignment = sizeof(void*); | |
434 #endif | |
435 | |
436 // extra metadata bits | |
437 static const uintptr_t xml_memory_page_contents_shared_mask = 64; | |
438 static const uintptr_t xml_memory_page_name_allocated_mask = 32; | |
439 static const uintptr_t xml_memory_page_value_allocated_mask = 16; | |
440 static const uintptr_t xml_memory_page_type_mask = 15; | |
441 | |
442 // combined masks for string uniqueness | |
443 static const uintptr_t xml_memory_page_name_allocated_or_shared_mask = xml_memory_page_name_allocated_mask | xml_memory_page_contents_shared_mask; | |
444 static const uintptr_t xml_memory_page_value_allocated_or_shared_mask = xml_memory_page_value_allocated_mask | xml_memory_page_contents_shared_mask; | |
445 | |
446 #ifdef PUGIXML_COMPACT | |
447 #define PUGI__GETHEADER_IMPL(object, page, flags) // unused | |
448 #define PUGI__GETPAGE_IMPL(header) (header).get_page() | |
449 #else | |
450 #define PUGI__GETHEADER_IMPL(object, page, flags) (((reinterpret_cast<char*>(object) - reinterpret_cast<char*>(page)) << 8) | (flags)) | |
451 // this macro casts pointers through void* to avoid 'cast increases required alignment of target type' warnings | |
452 #define PUGI__GETPAGE_IMPL(header) static_cast<impl::xml_memory_page*>(const_cast<void*>(static_cast<const void*>(reinterpret_cast<const char*>(&header) - (header >> 8)))) | |
453 #endif | |
454 | |
455 #define PUGI__GETPAGE(n) PUGI__GETPAGE_IMPL((n)->header) | |
456 #define PUGI__NODETYPE(n) static_cast<xml_node_type>((n)->header & impl::xml_memory_page_type_mask) | |
457 | |
458 struct xml_allocator; | |
459 | |
460 struct xml_memory_page | |
461 { | |
462 static xml_memory_page* construct(void* memory) | |
463 { | |
464 xml_memory_page* result = static_cast<xml_memory_page*>(memory); | |
465 | |
466 result->allocator = 0; | |
467 result->prev = 0; | |
468 result->next = 0; | |
469 result->busy_size = 0; | |
470 result->freed_size = 0; | |
471 | |
472 #ifdef PUGIXML_COMPACT | |
473 result->compact_string_base = 0; | |
474 result->compact_shared_parent = 0; | |
475 result->compact_page_marker = 0; | |
476 #endif | |
477 | |
478 return result; | |
479 } | |
480 | |
481 xml_allocator* allocator; | |
482 | |
483 xml_memory_page* prev; | |
484 xml_memory_page* next; | |
485 | |
486 size_t busy_size; | |
487 size_t freed_size; | |
488 | |
489 #ifdef PUGIXML_COMPACT | |
490 char_t* compact_string_base; | |
491 void* compact_shared_parent; | |
492 uint32_t* compact_page_marker; | |
493 #endif | |
494 }; | |
495 | |
496 static const size_t xml_memory_page_size = | |
497 #ifdef PUGIXML_MEMORY_PAGE_SIZE | |
498 (PUGIXML_MEMORY_PAGE_SIZE) | |
499 #else | |
500 32768 | |
501 #endif | |
502 - sizeof(xml_memory_page); | |
503 | |
504 struct xml_memory_string_header | |
505 { | |
506 uint16_t page_offset; // offset from page->data | |
507 uint16_t full_size; // 0 if string occupies whole page | |
508 }; | |
509 | |
510 struct xml_allocator | |
511 { | |
512 xml_allocator(xml_memory_page* root): _root(root), _busy_size(root->busy_size) | |
513 { | |
514 #ifdef PUGIXML_COMPACT | |
515 _hash = 0; | |
516 #endif | |
517 } | |
518 | |
519 xml_memory_page* allocate_page(size_t data_size) | |
520 { | |
521 size_t size = sizeof(xml_memory_page) + data_size; | |
522 | |
523 // allocate block with some alignment, leaving memory for worst-case padding | |
524 void* memory = xml_memory::allocate(size); | |
525 if (!memory) return 0; | |
526 | |
527 // prepare page structure | |
528 xml_memory_page* page = xml_memory_page::construct(memory); | |
529 assert(page); | |
530 | |
531 assert(this == _root->allocator); | |
532 page->allocator = this; | |
533 | |
534 return page; | |
535 } | |
536 | |
537 static void deallocate_page(xml_memory_page* page) | |
538 { | |
539 xml_memory::deallocate(page); | |
540 } | |
541 | |
542 void* allocate_memory_oob(size_t size, xml_memory_page*& out_page); | |
543 | |
544 void* allocate_memory(size_t size, xml_memory_page*& out_page) | |
545 { | |
546 if (PUGI__UNLIKELY(_busy_size + size > xml_memory_page_size)) | |
547 return allocate_memory_oob(size, out_page); | |
548 | |
549 void* buf = reinterpret_cast<char*>(_root) + sizeof(xml_memory_page) + _busy_size; | |
550 | |
551 _busy_size += size; | |
552 | |
553 out_page = _root; | |
554 | |
555 return buf; | |
556 } | |
557 | |
558 #ifdef PUGIXML_COMPACT | |
559 void* allocate_object(size_t size, xml_memory_page*& out_page) | |
560 { | |
561 void* result = allocate_memory(size + sizeof(uint32_t), out_page); | |
562 if (!result) return 0; | |
563 | |
564 // adjust for marker | |
565 ptrdiff_t offset = static_cast<char*>(result) - reinterpret_cast<char*>(out_page->compact_page_marker); | |
566 | |
567 if (PUGI__UNLIKELY(static_cast<uintptr_t>(offset) >= 256 * xml_memory_block_alignment)) | |
568 { | |
569 // insert new marker | |
570 uint32_t* marker = static_cast<uint32_t*>(result); | |
571 | |
572 *marker = static_cast<uint32_t>(reinterpret_cast<char*>(marker) - reinterpret_cast<char*>(out_page)); | |
573 out_page->compact_page_marker = marker; | |
574 | |
575 // since we don't reuse the page space until we reallocate it, we can just pretend that we freed the marker block | |
576 // this will make sure deallocate_memory correctly tracks the size | |
577 out_page->freed_size += sizeof(uint32_t); | |
578 | |
579 return marker + 1; | |
580 } | |
581 else | |
582 { | |
583 // roll back uint32_t part | |
584 _busy_size -= sizeof(uint32_t); | |
585 | |
586 return result; | |
587 } | |
588 } | |
589 #else | |
590 void* allocate_object(size_t size, xml_memory_page*& out_page) | |
591 { | |
592 return allocate_memory(size, out_page); | |
593 } | |
594 #endif | |
595 | |
596 void deallocate_memory(void* ptr, size_t size, xml_memory_page* page) | |
597 { | |
598 if (page == _root) page->busy_size = _busy_size; | |
599 | |
600 assert(ptr >= reinterpret_cast<char*>(page) + sizeof(xml_memory_page) && ptr < reinterpret_cast<char*>(page) + sizeof(xml_memory_page) + page->busy_size); | |
601 (void)!ptr; | |
602 | |
603 page->freed_size += size; | |
604 assert(page->freed_size <= page->busy_size); | |
605 | |
606 if (page->freed_size == page->busy_size) | |
607 { | |
608 if (page->next == 0) | |
609 { | |
610 assert(_root == page); | |
611 | |
612 // top page freed, just reset sizes | |
613 page->busy_size = 0; | |
614 page->freed_size = 0; | |
615 | |
616 #ifdef PUGIXML_COMPACT | |
617 // reset compact state to maximize efficiency | |
618 page->compact_string_base = 0; | |
619 page->compact_shared_parent = 0; | |
620 page->compact_page_marker = 0; | |
621 #endif | |
622 | |
623 _busy_size = 0; | |
624 } | |
625 else | |
626 { | |
627 assert(_root != page); | |
628 assert(page->prev); | |
629 | |
630 // remove from the list | |
631 page->prev->next = page->next; | |
632 page->next->prev = page->prev; | |
633 | |
634 // deallocate | |
635 deallocate_page(page); | |
636 } | |
637 } | |
638 } | |
639 | |
640 char_t* allocate_string(size_t length) | |
641 { | |
642 static const size_t max_encoded_offset = (1 << 16) * xml_memory_block_alignment; | |
643 | |
644 PUGI__STATIC_ASSERT(xml_memory_page_size <= max_encoded_offset); | |
645 | |
646 // allocate memory for string and header block | |
647 size_t size = sizeof(xml_memory_string_header) + length * sizeof(char_t); | |
648 | |
649 // round size up to block alignment boundary | |
650 size_t full_size = (size + (xml_memory_block_alignment - 1)) & ~(xml_memory_block_alignment - 1); | |
651 | |
652 xml_memory_page* page; | |
653 xml_memory_string_header* header = static_cast<xml_memory_string_header*>(allocate_memory(full_size, page)); | |
654 | |
655 if (!header) return 0; | |
656 | |
657 // setup header | |
658 ptrdiff_t page_offset = reinterpret_cast<char*>(header) - reinterpret_cast<char*>(page) - sizeof(xml_memory_page); | |
659 | |
660 assert(page_offset % xml_memory_block_alignment == 0); | |
661 assert(page_offset >= 0 && static_cast<size_t>(page_offset) < max_encoded_offset); | |
662 header->page_offset = static_cast<uint16_t>(static_cast<size_t>(page_offset) / xml_memory_block_alignment); | |
663 | |
664 // full_size == 0 for large strings that occupy the whole page | |
665 assert(full_size % xml_memory_block_alignment == 0); | |
666 assert(full_size < max_encoded_offset || (page->busy_size == full_size && page_offset == 0)); | |
667 header->full_size = static_cast<uint16_t>(full_size < max_encoded_offset ? full_size / xml_memory_block_alignment : 0); | |
668 | |
669 // round-trip through void* to avoid 'cast increases required alignment of target type' warning | |
670 // header is guaranteed a pointer-sized alignment, which should be enough for char_t | |
671 return static_cast<char_t*>(static_cast<void*>(header + 1)); | |
672 } | |
673 | |
674 void deallocate_string(char_t* string) | |
675 { | |
676 // this function casts pointers through void* to avoid 'cast increases required alignment of target type' warnings | |
677 // we're guaranteed the proper (pointer-sized) alignment on the input string if it was allocated via allocate_string | |
678 | |
679 // get header | |
680 xml_memory_string_header* header = static_cast<xml_memory_string_header*>(static_cast<void*>(string)) - 1; | |
681 assert(header); | |
682 | |
683 // deallocate | |
684 size_t page_offset = sizeof(xml_memory_page) + header->page_offset * xml_memory_block_alignment; | |
685 xml_memory_page* page = reinterpret_cast<xml_memory_page*>(static_cast<void*>(reinterpret_cast<char*>(header) - page_offset)); | |
686 | |
687 // if full_size == 0 then this string occupies the whole page | |
688 size_t full_size = header->full_size == 0 ? page->busy_size : header->full_size * xml_memory_block_alignment; | |
689 | |
690 deallocate_memory(header, full_size, page); | |
691 } | |
692 | |
693 bool reserve() | |
694 { | |
695 #ifdef PUGIXML_COMPACT | |
696 return _hash->reserve(); | |
697 #else | |
698 return true; | |
699 #endif | |
700 } | |
701 | |
702 xml_memory_page* _root; | |
703 size_t _busy_size; | |
704 | |
705 #ifdef PUGIXML_COMPACT | |
706 compact_hash_table* _hash; | |
707 #endif | |
708 }; | |
709 | |
710 PUGI__FN_NO_INLINE void* xml_allocator::allocate_memory_oob(size_t size, xml_memory_page*& out_page) | |
711 { | |
712 const size_t large_allocation_threshold = xml_memory_page_size / 4; | |
713 | |
714 xml_memory_page* page = allocate_page(size <= large_allocation_threshold ? xml_memory_page_size : size); | |
715 out_page = page; | |
716 | |
717 if (!page) return 0; | |
718 | |
719 if (size <= large_allocation_threshold) | |
720 { | |
721 _root->busy_size = _busy_size; | |
722 | |
723 // insert page at the end of linked list | |
724 page->prev = _root; | |
725 _root->next = page; | |
726 _root = page; | |
727 | |
728 _busy_size = size; | |
729 } | |
730 else | |
731 { | |
732 // insert page before the end of linked list, so that it is deleted as soon as possible | |
733 // the last page is not deleted even if it's empty (see deallocate_memory) | |
734 assert(_root->prev); | |
735 | |
736 page->prev = _root->prev; | |
737 page->next = _root; | |
738 | |
739 _root->prev->next = page; | |
740 _root->prev = page; | |
741 | |
742 page->busy_size = size; | |
743 } | |
744 | |
745 return reinterpret_cast<char*>(page) + sizeof(xml_memory_page); | |
746 } | |
747 PUGI__NS_END | |
748 | |
749 #ifdef PUGIXML_COMPACT | |
750 PUGI__NS_BEGIN | |
751 static const uintptr_t compact_alignment_log2 = 2; | |
752 static const uintptr_t compact_alignment = 1 << compact_alignment_log2; | |
753 | |
754 class compact_header | |
755 { | |
756 public: | |
757 compact_header(xml_memory_page* page, unsigned int flags) | |
758 { | |
759 PUGI__STATIC_ASSERT(xml_memory_block_alignment == compact_alignment); | |
760 | |
761 ptrdiff_t offset = (reinterpret_cast<char*>(this) - reinterpret_cast<char*>(page->compact_page_marker)); | |
762 assert(offset % compact_alignment == 0 && static_cast<uintptr_t>(offset) < 256 * compact_alignment); | |
763 | |
764 _page = static_cast<unsigned char>(offset >> compact_alignment_log2); | |
765 _flags = static_cast<unsigned char>(flags); | |
766 } | |
767 | |
768 void operator&=(uintptr_t mod) | |
769 { | |
770 _flags &= static_cast<unsigned char>(mod); | |
771 } | |
772 | |
773 void operator|=(uintptr_t mod) | |
774 { | |
775 _flags |= static_cast<unsigned char>(mod); | |
776 } | |
777 | |
778 uintptr_t operator&(uintptr_t mod) const | |
779 { | |
780 return _flags & mod; | |
781 } | |
782 | |
783 xml_memory_page* get_page() const | |
784 { | |
785 // round-trip through void* to silence 'cast increases required alignment of target type' warnings | |
786 const char* page_marker = reinterpret_cast<const char*>(this) - (_page << compact_alignment_log2); | |
787 const char* page = page_marker - *reinterpret_cast<const uint32_t*>(static_cast<const void*>(page_marker)); | |
788 | |
789 return const_cast<xml_memory_page*>(reinterpret_cast<const xml_memory_page*>(static_cast<const void*>(page))); | |
790 } | |
791 | |
792 private: | |
793 unsigned char _page; | |
794 unsigned char _flags; | |
795 }; | |
796 | |
797 PUGI__FN xml_memory_page* compact_get_page(const void* object, int header_offset) | |
798 { | |
799 const compact_header* header = reinterpret_cast<const compact_header*>(static_cast<const char*>(object) - header_offset); | |
800 | |
801 return header->get_page(); | |
802 } | |
803 | |
804 template <int header_offset, typename T> PUGI__FN_NO_INLINE T* compact_get_value(const void* object) | |
805 { | |
806 return static_cast<T*>(compact_get_page(object, header_offset)->allocator->_hash->find(object)); | |
807 } | |
808 | |
809 template <int header_offset, typename T> PUGI__FN_NO_INLINE void compact_set_value(const void* object, T* value) | |
810 { | |
811 compact_get_page(object, header_offset)->allocator->_hash->insert(object, value); | |
812 } | |
813 | |
814 template <typename T, int header_offset, int start = -126> class compact_pointer | |
815 { | |
816 public: | |
817 compact_pointer(): _data(0) | |
818 { | |
819 } | |
820 | |
821 void operator=(const compact_pointer& rhs) | |
822 { | |
823 *this = rhs + 0; | |
824 } | |
825 | |
826 void operator=(T* value) | |
827 { | |
828 if (value) | |
829 { | |
830 // value is guaranteed to be compact-aligned; 'this' is not | |
831 // our decoding is based on 'this' aligned to compact alignment downwards (see operator T*) | |
832 // so for negative offsets (e.g. -3) we need to adjust the diff by compact_alignment - 1 to | |
833 // compensate for arithmetic shift rounding for negative values | |
834 ptrdiff_t diff = reinterpret_cast<char*>(value) - reinterpret_cast<char*>(this); | |
835 ptrdiff_t offset = ((diff + int(compact_alignment - 1)) >> compact_alignment_log2) - start; | |
836 | |
837 if (static_cast<uintptr_t>(offset) <= 253) | |
838 _data = static_cast<unsigned char>(offset + 1); | |
839 else | |
840 { | |
841 compact_set_value<header_offset>(this, value); | |
842 | |
843 _data = 255; | |
844 } | |
845 } | |
846 else | |
847 _data = 0; | |
848 } | |
849 | |
850 operator T*() const | |
851 { | |
852 if (_data) | |
853 { | |
854 if (_data < 255) | |
855 { | |
856 uintptr_t base = reinterpret_cast<uintptr_t>(this) & ~(compact_alignment - 1); | |
857 | |
858 return reinterpret_cast<T*>(base + (_data - 1 + start) * compact_alignment); | |
859 } | |
860 else | |
861 return compact_get_value<header_offset, T>(this); | |
862 } | |
863 else | |
864 return 0; | |
865 } | |
866 | |
867 T* operator->() const | |
868 { | |
869 return *this; | |
870 } | |
871 | |
872 private: | |
873 unsigned char _data; | |
874 }; | |
875 | |
876 template <typename T, int header_offset> class compact_pointer_parent | |
877 { | |
878 public: | |
879 compact_pointer_parent(): _data(0) | |
880 { | |
881 } | |
882 | |
883 void operator=(const compact_pointer_parent& rhs) | |
884 { | |
885 *this = rhs + 0; | |
886 } | |
887 | |
888 void operator=(T* value) | |
889 { | |
890 if (value) | |
891 { | |
892 // value is guaranteed to be compact-aligned; 'this' is not | |
893 // our decoding is based on 'this' aligned to compact alignment downwards (see operator T*) | |
894 // so for negative offsets (e.g. -3) we need to adjust the diff by compact_alignment - 1 to | |
895 // compensate for arithmetic shift behavior for negative values | |
896 ptrdiff_t diff = reinterpret_cast<char*>(value) - reinterpret_cast<char*>(this); | |
897 ptrdiff_t offset = ((diff + int(compact_alignment - 1)) >> compact_alignment_log2) + 65533; | |
898 | |
899 if (static_cast<uintptr_t>(offset) <= 65533) | |
900 { | |
901 _data = static_cast<unsigned short>(offset + 1); | |
902 } | |
903 else | |
904 { | |
905 xml_memory_page* page = compact_get_page(this, header_offset); | |
906 | |
907 if (PUGI__UNLIKELY(page->compact_shared_parent == 0)) | |
908 page->compact_shared_parent = value; | |
909 | |
910 if (page->compact_shared_parent == value) | |
911 { | |
912 _data = 65534; | |
913 } | |
914 else | |
915 { | |
916 compact_set_value<header_offset>(this, value); | |
917 | |
918 _data = 65535; | |
919 } | |
920 } | |
921 } | |
922 else | |
923 { | |
924 _data = 0; | |
925 } | |
926 } | |
927 | |
928 operator T*() const | |
929 { | |
930 if (_data) | |
931 { | |
932 if (_data < 65534) | |
933 { | |
934 uintptr_t base = reinterpret_cast<uintptr_t>(this) & ~(compact_alignment - 1); | |
935 | |
936 return reinterpret_cast<T*>(base + (_data - 1 - 65533) * compact_alignment); | |
937 } | |
938 else if (_data == 65534) | |
939 return static_cast<T*>(compact_get_page(this, header_offset)->compact_shared_parent); | |
940 else | |
941 return compact_get_value<header_offset, T>(this); | |
942 } | |
943 else | |
944 return 0; | |
945 } | |
946 | |
947 T* operator->() const | |
948 { | |
949 return *this; | |
950 } | |
951 | |
952 private: | |
953 uint16_t _data; | |
954 }; | |
955 | |
956 template <int header_offset, int base_offset> class compact_string | |
957 { | |
958 public: | |
959 compact_string(): _data(0) | |
960 { | |
961 } | |
962 | |
963 void operator=(const compact_string& rhs) | |
964 { | |
965 *this = rhs + 0; | |
966 } | |
967 | |
968 void operator=(char_t* value) | |
969 { | |
970 if (value) | |
971 { | |
972 xml_memory_page* page = compact_get_page(this, header_offset); | |
973 | |
974 if (PUGI__UNLIKELY(page->compact_string_base == 0)) | |
975 page->compact_string_base = value; | |
976 | |
977 ptrdiff_t offset = value - page->compact_string_base; | |
978 | |
979 if (static_cast<uintptr_t>(offset) < (65535 << 7)) | |
980 { | |
981 // round-trip through void* to silence 'cast increases required alignment of target type' warnings | |
982 uint16_t* base = reinterpret_cast<uint16_t*>(static_cast<void*>(reinterpret_cast<char*>(this) - base_offset)); | |
983 | |
984 if (*base == 0) | |
985 { | |
986 *base = static_cast<uint16_t>((offset >> 7) + 1); | |
987 _data = static_cast<unsigned char>((offset & 127) + 1); | |
988 } | |
989 else | |
990 { | |
991 ptrdiff_t remainder = offset - ((*base - 1) << 7); | |
992 | |
993 if (static_cast<uintptr_t>(remainder) <= 253) | |
994 { | |
995 _data = static_cast<unsigned char>(remainder + 1); | |
996 } | |
997 else | |
998 { | |
999 compact_set_value<header_offset>(this, value); | |
1000 | |
1001 _data = 255; | |
1002 } | |
1003 } | |
1004 } | |
1005 else | |
1006 { | |
1007 compact_set_value<header_offset>(this, value); | |
1008 | |
1009 _data = 255; | |
1010 } | |
1011 } | |
1012 else | |
1013 { | |
1014 _data = 0; | |
1015 } | |
1016 } | |
1017 | |
1018 operator char_t*() const | |
1019 { | |
1020 if (_data) | |
1021 { | |
1022 if (_data < 255) | |
1023 { | |
1024 xml_memory_page* page = compact_get_page(this, header_offset); | |
1025 | |
1026 // round-trip through void* to silence 'cast increases required alignment of target type' warnings | |
1027 const uint16_t* base = reinterpret_cast<const uint16_t*>(static_cast<const void*>(reinterpret_cast<const char*>(this) - base_offset)); | |
1028 assert(*base); | |
1029 | |
1030 ptrdiff_t offset = ((*base - 1) << 7) + (_data - 1); | |
1031 | |
1032 return page->compact_string_base + offset; | |
1033 } | |
1034 else | |
1035 { | |
1036 return compact_get_value<header_offset, char_t>(this); | |
1037 } | |
1038 } | |
1039 else | |
1040 return 0; | |
1041 } | |
1042 | |
1043 private: | |
1044 unsigned char _data; | |
1045 }; | |
1046 PUGI__NS_END | |
1047 #endif | |
1048 | |
1049 #ifdef PUGIXML_COMPACT | |
1050 namespace pugi | |
1051 { | |
1052 struct xml_attribute_struct | |
1053 { | |
1054 xml_attribute_struct(impl::xml_memory_page* page): header(page, 0), namevalue_base(0) | |
1055 { | |
1056 PUGI__STATIC_ASSERT(sizeof(xml_attribute_struct) == 8); | |
1057 } | |
1058 | |
1059 impl::compact_header header; | |
1060 | |
1061 uint16_t namevalue_base; | |
1062 | |
1063 impl::compact_string<4, 2> name; | |
1064 impl::compact_string<5, 3> value; | |
1065 | |
1066 impl::compact_pointer<xml_attribute_struct, 6> prev_attribute_c; | |
1067 impl::compact_pointer<xml_attribute_struct, 7, 0> next_attribute; | |
1068 }; | |
1069 | |
1070 struct xml_node_struct | |
1071 { | |
1072 xml_node_struct(impl::xml_memory_page* page, xml_node_type type): header(page, type), namevalue_base(0) | |
1073 { | |
1074 PUGI__STATIC_ASSERT(sizeof(xml_node_struct) == 12); | |
1075 } | |
1076 | |
1077 impl::compact_header header; | |
1078 | |
1079 uint16_t namevalue_base; | |
1080 | |
1081 impl::compact_string<4, 2> name; | |
1082 impl::compact_string<5, 3> value; | |
1083 | |
1084 impl::compact_pointer_parent<xml_node_struct, 6> parent; | |
1085 | |
1086 impl::compact_pointer<xml_node_struct, 8, 0> first_child; | |
1087 | |
1088 impl::compact_pointer<xml_node_struct, 9> prev_sibling_c; | |
1089 impl::compact_pointer<xml_node_struct, 10, 0> next_sibling; | |
1090 | |
1091 impl::compact_pointer<xml_attribute_struct, 11, 0> first_attribute; | |
1092 }; | |
1093 } | |
1094 #else | |
1095 namespace pugi | |
1096 { | |
1097 struct xml_attribute_struct | |
1098 { | |
1099 xml_attribute_struct(impl::xml_memory_page* page): name(0), value(0), prev_attribute_c(0), next_attribute(0) | |
1100 { | |
1101 header = PUGI__GETHEADER_IMPL(this, page, 0); | |
1102 } | |
1103 | |
1104 uintptr_t header; | |
1105 | |
1106 char_t* name; | |
1107 char_t* value; | |
1108 | |
1109 xml_attribute_struct* prev_attribute_c; | |
1110 xml_attribute_struct* next_attribute; | |
1111 }; | |
1112 | |
1113 struct xml_node_struct | |
1114 { | |
1115 xml_node_struct(impl::xml_memory_page* page, xml_node_type type): name(0), value(0), parent(0), first_child(0), prev_sibling_c(0), next_sibling(0), first_attribute(0) | |
1116 { | |
1117 header = PUGI__GETHEADER_IMPL(this, page, type); | |
1118 } | |
1119 | |
1120 uintptr_t header; | |
1121 | |
1122 char_t* name; | |
1123 char_t* value; | |
1124 | |
1125 xml_node_struct* parent; | |
1126 | |
1127 xml_node_struct* first_child; | |
1128 | |
1129 xml_node_struct* prev_sibling_c; | |
1130 xml_node_struct* next_sibling; | |
1131 | |
1132 xml_attribute_struct* first_attribute; | |
1133 }; | |
1134 } | |
1135 #endif | |
1136 | |
1137 PUGI__NS_BEGIN | |
1138 struct xml_extra_buffer | |
1139 { | |
1140 char_t* buffer; | |
1141 xml_extra_buffer* next; | |
1142 }; | |
1143 | |
1144 struct xml_document_struct: public xml_node_struct, public xml_allocator | |
1145 { | |
1146 xml_document_struct(xml_memory_page* page): xml_node_struct(page, node_document), xml_allocator(page), buffer(0), extra_buffers(0) | |
1147 { | |
1148 } | |
1149 | |
1150 const char_t* buffer; | |
1151 | |
1152 xml_extra_buffer* extra_buffers; | |
1153 | |
1154 #ifdef PUGIXML_COMPACT | |
1155 compact_hash_table hash; | |
1156 #endif | |
1157 }; | |
1158 | |
1159 template <typename Object> inline xml_allocator& get_allocator(const Object* object) | |
1160 { | |
1161 assert(object); | |
1162 | |
1163 return *PUGI__GETPAGE(object)->allocator; | |
1164 } | |
1165 | |
1166 template <typename Object> inline xml_document_struct& get_document(const Object* object) | |
1167 { | |
1168 assert(object); | |
1169 | |
1170 return *static_cast<xml_document_struct*>(PUGI__GETPAGE(object)->allocator); | |
1171 } | |
1172 PUGI__NS_END | |
1173 | |
1174 // Low-level DOM operations | |
1175 PUGI__NS_BEGIN | |
1176 inline xml_attribute_struct* allocate_attribute(xml_allocator& alloc) | |
1177 { | |
1178 xml_memory_page* page; | |
1179 void* memory = alloc.allocate_object(sizeof(xml_attribute_struct), page); | |
1180 if (!memory) return 0; | |
1181 | |
1182 return new (memory) xml_attribute_struct(page); | |
1183 } | |
1184 | |
1185 inline xml_node_struct* allocate_node(xml_allocator& alloc, xml_node_type type) | |
1186 { | |
1187 xml_memory_page* page; | |
1188 void* memory = alloc.allocate_object(sizeof(xml_node_struct), page); | |
1189 if (!memory) return 0; | |
1190 | |
1191 return new (memory) xml_node_struct(page, type); | |
1192 } | |
1193 | |
1194 inline void destroy_attribute(xml_attribute_struct* a, xml_allocator& alloc) | |
1195 { | |
1196 if (a->header & impl::xml_memory_page_name_allocated_mask) | |
1197 alloc.deallocate_string(a->name); | |
1198 | |
1199 if (a->header & impl::xml_memory_page_value_allocated_mask) | |
1200 alloc.deallocate_string(a->value); | |
1201 | |
1202 alloc.deallocate_memory(a, sizeof(xml_attribute_struct), PUGI__GETPAGE(a)); | |
1203 } | |
1204 | |
1205 inline void destroy_node(xml_node_struct* n, xml_allocator& alloc) | |
1206 { | |
1207 if (n->header & impl::xml_memory_page_name_allocated_mask) | |
1208 alloc.deallocate_string(n->name); | |
1209 | |
1210 if (n->header & impl::xml_memory_page_value_allocated_mask) | |
1211 alloc.deallocate_string(n->value); | |
1212 | |
1213 for (xml_attribute_struct* attr = n->first_attribute; attr; ) | |
1214 { | |
1215 xml_attribute_struct* next = attr->next_attribute; | |
1216 | |
1217 destroy_attribute(attr, alloc); | |
1218 | |
1219 attr = next; | |
1220 } | |
1221 | |
1222 for (xml_node_struct* child = n->first_child; child; ) | |
1223 { | |
1224 xml_node_struct* next = child->next_sibling; | |
1225 | |
1226 destroy_node(child, alloc); | |
1227 | |
1228 child = next; | |
1229 } | |
1230 | |
1231 alloc.deallocate_memory(n, sizeof(xml_node_struct), PUGI__GETPAGE(n)); | |
1232 } | |
1233 | |
1234 inline void append_node(xml_node_struct* child, xml_node_struct* node) | |
1235 { | |
1236 child->parent = node; | |
1237 | |
1238 xml_node_struct* head = node->first_child; | |
1239 | |
1240 if (head) | |
1241 { | |
1242 xml_node_struct* tail = head->prev_sibling_c; | |
1243 | |
1244 tail->next_sibling = child; | |
1245 child->prev_sibling_c = tail; | |
1246 head->prev_sibling_c = child; | |
1247 } | |
1248 else | |
1249 { | |
1250 node->first_child = child; | |
1251 child->prev_sibling_c = child; | |
1252 } | |
1253 } | |
1254 | |
1255 inline void prepend_node(xml_node_struct* child, xml_node_struct* node) | |
1256 { | |
1257 child->parent = node; | |
1258 | |
1259 xml_node_struct* head = node->first_child; | |
1260 | |
1261 if (head) | |
1262 { | |
1263 child->prev_sibling_c = head->prev_sibling_c; | |
1264 head->prev_sibling_c = child; | |
1265 } | |
1266 else | |
1267 child->prev_sibling_c = child; | |
1268 | |
1269 child->next_sibling = head; | |
1270 node->first_child = child; | |
1271 } | |
1272 | |
1273 inline void insert_node_after(xml_node_struct* child, xml_node_struct* node) | |
1274 { | |
1275 xml_node_struct* parent = node->parent; | |
1276 | |
1277 child->parent = parent; | |
1278 | |
1279 xml_node_struct* next = node->next_sibling; | |
1280 | |
1281 if (next) | |
1282 next->prev_sibling_c = child; | |
1283 else | |
1284 parent->first_child->prev_sibling_c = child; | |
1285 | |
1286 child->next_sibling = next; | |
1287 child->prev_sibling_c = node; | |
1288 | |
1289 node->next_sibling = child; | |
1290 } | |
1291 | |
1292 inline void insert_node_before(xml_node_struct* child, xml_node_struct* node) | |
1293 { | |
1294 xml_node_struct* parent = node->parent; | |
1295 | |
1296 child->parent = parent; | |
1297 | |
1298 xml_node_struct* prev = node->prev_sibling_c; | |
1299 | |
1300 if (prev->next_sibling) | |
1301 prev->next_sibling = child; | |
1302 else | |
1303 parent->first_child = child; | |
1304 | |
1305 child->prev_sibling_c = prev; | |
1306 child->next_sibling = node; | |
1307 | |
1308 node->prev_sibling_c = child; | |
1309 } | |
1310 | |
1311 inline void remove_node(xml_node_struct* node) | |
1312 { | |
1313 xml_node_struct* parent = node->parent; | |
1314 | |
1315 xml_node_struct* next = node->next_sibling; | |
1316 xml_node_struct* prev = node->prev_sibling_c; | |
1317 | |
1318 if (next) | |
1319 next->prev_sibling_c = prev; | |
1320 else | |
1321 parent->first_child->prev_sibling_c = prev; | |
1322 | |
1323 if (prev->next_sibling) | |
1324 prev->next_sibling = next; | |
1325 else | |
1326 parent->first_child = next; | |
1327 | |
1328 node->parent = 0; | |
1329 node->prev_sibling_c = 0; | |
1330 node->next_sibling = 0; | |
1331 } | |
1332 | |
1333 inline void append_attribute(xml_attribute_struct* attr, xml_node_struct* node) | |
1334 { | |
1335 xml_attribute_struct* head = node->first_attribute; | |
1336 | |
1337 if (head) | |
1338 { | |
1339 xml_attribute_struct* tail = head->prev_attribute_c; | |
1340 | |
1341 tail->next_attribute = attr; | |
1342 attr->prev_attribute_c = tail; | |
1343 head->prev_attribute_c = attr; | |
1344 } | |
1345 else | |
1346 { | |
1347 node->first_attribute = attr; | |
1348 attr->prev_attribute_c = attr; | |
1349 } | |
1350 } | |
1351 | |
1352 inline void prepend_attribute(xml_attribute_struct* attr, xml_node_struct* node) | |
1353 { | |
1354 xml_attribute_struct* head = node->first_attribute; | |
1355 | |
1356 if (head) | |
1357 { | |
1358 attr->prev_attribute_c = head->prev_attribute_c; | |
1359 head->prev_attribute_c = attr; | |
1360 } | |
1361 else | |
1362 attr->prev_attribute_c = attr; | |
1363 | |
1364 attr->next_attribute = head; | |
1365 node->first_attribute = attr; | |
1366 } | |
1367 | |
1368 inline void insert_attribute_after(xml_attribute_struct* attr, xml_attribute_struct* place, xml_node_struct* node) | |
1369 { | |
1370 xml_attribute_struct* next = place->next_attribute; | |
1371 | |
1372 if (next) | |
1373 next->prev_attribute_c = attr; | |
1374 else | |
1375 node->first_attribute->prev_attribute_c = attr; | |
1376 | |
1377 attr->next_attribute = next; | |
1378 attr->prev_attribute_c = place; | |
1379 place->next_attribute = attr; | |
1380 } | |
1381 | |
1382 inline void insert_attribute_before(xml_attribute_struct* attr, xml_attribute_struct* place, xml_node_struct* node) | |
1383 { | |
1384 xml_attribute_struct* prev = place->prev_attribute_c; | |
1385 | |
1386 if (prev->next_attribute) | |
1387 prev->next_attribute = attr; | |
1388 else | |
1389 node->first_attribute = attr; | |
1390 | |
1391 attr->prev_attribute_c = prev; | |
1392 attr->next_attribute = place; | |
1393 place->prev_attribute_c = attr; | |
1394 } | |
1395 | |
1396 inline void remove_attribute(xml_attribute_struct* attr, xml_node_struct* node) | |
1397 { | |
1398 xml_attribute_struct* next = attr->next_attribute; | |
1399 xml_attribute_struct* prev = attr->prev_attribute_c; | |
1400 | |
1401 if (next) | |
1402 next->prev_attribute_c = prev; | |
1403 else | |
1404 node->first_attribute->prev_attribute_c = prev; | |
1405 | |
1406 if (prev->next_attribute) | |
1407 prev->next_attribute = next; | |
1408 else | |
1409 node->first_attribute = next; | |
1410 | |
1411 attr->prev_attribute_c = 0; | |
1412 attr->next_attribute = 0; | |
1413 } | |
1414 | |
1415 PUGI__FN_NO_INLINE xml_node_struct* append_new_node(xml_node_struct* node, xml_allocator& alloc, xml_node_type type = node_element) | |
1416 { | |
1417 if (!alloc.reserve()) return 0; | |
1418 | |
1419 xml_node_struct* child = allocate_node(alloc, type); | |
1420 if (!child) return 0; | |
1421 | |
1422 append_node(child, node); | |
1423 | |
1424 return child; | |
1425 } | |
1426 | |
1427 PUGI__FN_NO_INLINE xml_attribute_struct* append_new_attribute(xml_node_struct* node, xml_allocator& alloc) | |
1428 { | |
1429 if (!alloc.reserve()) return 0; | |
1430 | |
1431 xml_attribute_struct* attr = allocate_attribute(alloc); | |
1432 if (!attr) return 0; | |
1433 | |
1434 append_attribute(attr, node); | |
1435 | |
1436 return attr; | |
1437 } | |
1438 PUGI__NS_END | |
1439 | |
1440 // Helper classes for code generation | |
1441 PUGI__NS_BEGIN | |
1442 struct opt_false | |
1443 { | |
1444 enum { value = 0 }; | |
1445 }; | |
1446 | |
1447 struct opt_true | |
1448 { | |
1449 enum { value = 1 }; | |
1450 }; | |
1451 PUGI__NS_END | |
1452 | |
1453 // Unicode utilities | |
1454 PUGI__NS_BEGIN | |
1455 inline uint16_t endian_swap(uint16_t value) | |
1456 { | |
1457 return static_cast<uint16_t>(((value & 0xff) << 8) | (value >> 8)); | |
1458 } | |
1459 | |
1460 inline uint32_t endian_swap(uint32_t value) | |
1461 { | |
1462 return ((value & 0xff) << 24) | ((value & 0xff00) << 8) | ((value & 0xff0000) >> 8) | (value >> 24); | |
1463 } | |
1464 | |
1465 struct utf8_counter | |
1466 { | |
1467 typedef size_t value_type; | |
1468 | |
1469 static value_type low(value_type result, uint32_t ch) | |
1470 { | |
1471 // U+0000..U+007F | |
1472 if (ch < 0x80) return result + 1; | |
1473 // U+0080..U+07FF | |
1474 else if (ch < 0x800) return result + 2; | |
1475 // U+0800..U+FFFF | |
1476 else return result + 3; | |
1477 } | |
1478 | |
1479 static value_type high(value_type result, uint32_t) | |
1480 { | |
1481 // U+10000..U+10FFFF | |
1482 return result + 4; | |
1483 } | |
1484 }; | |
1485 | |
1486 struct utf8_writer | |
1487 { | |
1488 typedef uint8_t* value_type; | |
1489 | |
1490 static value_type low(value_type result, uint32_t ch) | |
1491 { | |
1492 // U+0000..U+007F | |
1493 if (ch < 0x80) | |
1494 { | |
1495 *result = static_cast<uint8_t>(ch); | |
1496 return result + 1; | |
1497 } | |
1498 // U+0080..U+07FF | |
1499 else if (ch < 0x800) | |
1500 { | |
1501 result[0] = static_cast<uint8_t>(0xC0 | (ch >> 6)); | |
1502 result[1] = static_cast<uint8_t>(0x80 | (ch & 0x3F)); | |
1503 return result + 2; | |
1504 } | |
1505 // U+0800..U+FFFF | |
1506 else | |
1507 { | |
1508 result[0] = static_cast<uint8_t>(0xE0 | (ch >> 12)); | |
1509 result[1] = static_cast<uint8_t>(0x80 | ((ch >> 6) & 0x3F)); | |
1510 result[2] = static_cast<uint8_t>(0x80 | (ch & 0x3F)); | |
1511 return result + 3; | |
1512 } | |
1513 } | |
1514 | |
1515 static value_type high(value_type result, uint32_t ch) | |
1516 { | |
1517 // U+10000..U+10FFFF | |
1518 result[0] = static_cast<uint8_t>(0xF0 | (ch >> 18)); | |
1519 result[1] = static_cast<uint8_t>(0x80 | ((ch >> 12) & 0x3F)); | |
1520 result[2] = static_cast<uint8_t>(0x80 | ((ch >> 6) & 0x3F)); | |
1521 result[3] = static_cast<uint8_t>(0x80 | (ch & 0x3F)); | |
1522 return result + 4; | |
1523 } | |
1524 | |
1525 static value_type any(value_type result, uint32_t ch) | |
1526 { | |
1527 return (ch < 0x10000) ? low(result, ch) : high(result, ch); | |
1528 } | |
1529 }; | |
1530 | |
1531 struct utf16_counter | |
1532 { | |
1533 typedef size_t value_type; | |
1534 | |
1535 static value_type low(value_type result, uint32_t) | |
1536 { | |
1537 return result + 1; | |
1538 } | |
1539 | |
1540 static value_type high(value_type result, uint32_t) | |
1541 { | |
1542 return result + 2; | |
1543 } | |
1544 }; | |
1545 | |
1546 struct utf16_writer | |
1547 { | |
1548 typedef uint16_t* value_type; | |
1549 | |
1550 static value_type low(value_type result, uint32_t ch) | |
1551 { | |
1552 *result = static_cast<uint16_t>(ch); | |
1553 | |
1554 return result + 1; | |
1555 } | |
1556 | |
1557 static value_type high(value_type result, uint32_t ch) | |
1558 { | |
1559 uint32_t msh = static_cast<uint32_t>(ch - 0x10000) >> 10; | |
1560 uint32_t lsh = static_cast<uint32_t>(ch - 0x10000) & 0x3ff; | |
1561 | |
1562 result[0] = static_cast<uint16_t>(0xD800 + msh); | |
1563 result[1] = static_cast<uint16_t>(0xDC00 + lsh); | |
1564 | |
1565 return result + 2; | |
1566 } | |
1567 | |
1568 static value_type any(value_type result, uint32_t ch) | |
1569 { | |
1570 return (ch < 0x10000) ? low(result, ch) : high(result, ch); | |
1571 } | |
1572 }; | |
1573 | |
1574 struct utf32_counter | |
1575 { | |
1576 typedef size_t value_type; | |
1577 | |
1578 static value_type low(value_type result, uint32_t) | |
1579 { | |
1580 return result + 1; | |
1581 } | |
1582 | |
1583 static value_type high(value_type result, uint32_t) | |
1584 { | |
1585 return result + 1; | |
1586 } | |
1587 }; | |
1588 | |
1589 struct utf32_writer | |
1590 { | |
1591 typedef uint32_t* value_type; | |
1592 | |
1593 static value_type low(value_type result, uint32_t ch) | |
1594 { | |
1595 *result = ch; | |
1596 | |
1597 return result + 1; | |
1598 } | |
1599 | |
1600 static value_type high(value_type result, uint32_t ch) | |
1601 { | |
1602 *result = ch; | |
1603 | |
1604 return result + 1; | |
1605 } | |
1606 | |
1607 static value_type any(value_type result, uint32_t ch) | |
1608 { | |
1609 *result = ch; | |
1610 | |
1611 return result + 1; | |
1612 } | |
1613 }; | |
1614 | |
1615 struct latin1_writer | |
1616 { | |
1617 typedef uint8_t* value_type; | |
1618 | |
1619 static value_type low(value_type result, uint32_t ch) | |
1620 { | |
1621 *result = static_cast<uint8_t>(ch > 255 ? '?' : ch); | |
1622 | |
1623 return result + 1; | |
1624 } | |
1625 | |
1626 static value_type high(value_type result, uint32_t ch) | |
1627 { | |
1628 (void)ch; | |
1629 | |
1630 *result = '?'; | |
1631 | |
1632 return result + 1; | |
1633 } | |
1634 }; | |
1635 | |
1636 struct utf8_decoder | |
1637 { | |
1638 typedef uint8_t type; | |
1639 | |
1640 template <typename Traits> static inline typename Traits::value_type process(const uint8_t* data, size_t size, typename Traits::value_type result, Traits) | |
1641 { | |
1642 const uint8_t utf8_byte_mask = 0x3f; | |
1643 | |
1644 while (size) | |
1645 { | |
1646 uint8_t lead = *data; | |
1647 | |
1648 // 0xxxxxxx -> U+0000..U+007F | |
1649 if (lead < 0x80) | |
1650 { | |
1651 result = Traits::low(result, lead); | |
1652 data += 1; | |
1653 size -= 1; | |
1654 | |
1655 // process aligned single-byte (ascii) blocks | |
1656 if ((reinterpret_cast<uintptr_t>(data) & 3) == 0) | |
1657 { | |
1658 // round-trip through void* to silence 'cast increases required alignment of target type' warnings | |
1659 while (size >= 4 && (*static_cast<const uint32_t*>(static_cast<const void*>(data)) & 0x80808080) == 0) | |
1660 { | |
1661 result = Traits::low(result, data[0]); | |
1662 result = Traits::low(result, data[1]); | |
1663 result = Traits::low(result, data[2]); | |
1664 result = Traits::low(result, data[3]); | |
1665 data += 4; | |
1666 size -= 4; | |
1667 } | |
1668 } | |
1669 } | |
1670 // 110xxxxx -> U+0080..U+07FF | |
1671 else if (static_cast<unsigned int>(lead - 0xC0) < 0x20 && size >= 2 && (data[1] & 0xc0) == 0x80) | |
1672 { | |
1673 result = Traits::low(result, ((lead & ~0xC0) << 6) | (data[1] & utf8_byte_mask)); | |
1674 data += 2; | |
1675 size -= 2; | |
1676 } | |
1677 // 1110xxxx -> U+0800-U+FFFF | |
1678 else if (static_cast<unsigned int>(lead - 0xE0) < 0x10 && size >= 3 && (data[1] & 0xc0) == 0x80 && (data[2] & 0xc0) == 0x80) | |
1679 { | |
1680 result = Traits::low(result, ((lead & ~0xE0) << 12) | ((data[1] & utf8_byte_mask) << 6) | (data[2] & utf8_byte_mask)); | |
1681 data += 3; | |
1682 size -= 3; | |
1683 } | |
1684 // 11110xxx -> U+10000..U+10FFFF | |
1685 else if (static_cast<unsigned int>(lead - 0xF0) < 0x08 && size >= 4 && (data[1] & 0xc0) == 0x80 && (data[2] & 0xc0) == 0x80 && (data[3] & 0xc0) == 0x80) | |
1686 { | |
1687 result = Traits::high(result, ((lead & ~0xF0) << 18) | ((data[1] & utf8_byte_mask) << 12) | ((data[2] & utf8_byte_mask) << 6) | (data[3] & utf8_byte_mask)); | |
1688 data += 4; | |
1689 size -= 4; | |
1690 } | |
1691 // 10xxxxxx or 11111xxx -> invalid | |
1692 else | |
1693 { | |
1694 data += 1; | |
1695 size -= 1; | |
1696 } | |
1697 } | |
1698 | |
1699 return result; | |
1700 } | |
1701 }; | |
1702 | |
1703 template <typename opt_swap> struct utf16_decoder | |
1704 { | |
1705 typedef uint16_t type; | |
1706 | |
1707 template <typename Traits> static inline typename Traits::value_type process(const uint16_t* data, size_t size, typename Traits::value_type result, Traits) | |
1708 { | |
1709 while (size) | |
1710 { | |
1711 uint16_t lead = opt_swap::value ? endian_swap(*data) : *data; | |
1712 | |
1713 // U+0000..U+D7FF | |
1714 if (lead < 0xD800) | |
1715 { | |
1716 result = Traits::low(result, lead); | |
1717 data += 1; | |
1718 size -= 1; | |
1719 } | |
1720 // U+E000..U+FFFF | |
1721 else if (static_cast<unsigned int>(lead - 0xE000) < 0x2000) | |
1722 { | |
1723 result = Traits::low(result, lead); | |
1724 data += 1; | |
1725 size -= 1; | |
1726 } | |
1727 // surrogate pair lead | |
1728 else if (static_cast<unsigned int>(lead - 0xD800) < 0x400 && size >= 2) | |
1729 { | |
1730 uint16_t next = opt_swap::value ? endian_swap(data[1]) : data[1]; | |
1731 | |
1732 if (static_cast<unsigned int>(next - 0xDC00) < 0x400) | |
1733 { | |
1734 result = Traits::high(result, 0x10000 + ((lead & 0x3ff) << 10) + (next & 0x3ff)); | |
1735 data += 2; | |
1736 size -= 2; | |
1737 } | |
1738 else | |
1739 { | |
1740 data += 1; | |
1741 size -= 1; | |
1742 } | |
1743 } | |
1744 else | |
1745 { | |
1746 data += 1; | |
1747 size -= 1; | |
1748 } | |
1749 } | |
1750 | |
1751 return result; | |
1752 } | |
1753 }; | |
1754 | |
1755 template <typename opt_swap> struct utf32_decoder | |
1756 { | |
1757 typedef uint32_t type; | |
1758 | |
1759 template <typename Traits> static inline typename Traits::value_type process(const uint32_t* data, size_t size, typename Traits::value_type result, Traits) | |
1760 { | |
1761 while (size) | |
1762 { | |
1763 uint32_t lead = opt_swap::value ? endian_swap(*data) : *data; | |
1764 | |
1765 // U+0000..U+FFFF | |
1766 if (lead < 0x10000) | |
1767 { | |
1768 result = Traits::low(result, lead); | |
1769 data += 1; | |
1770 size -= 1; | |
1771 } | |
1772 // U+10000..U+10FFFF | |
1773 else | |
1774 { | |
1775 result = Traits::high(result, lead); | |
1776 data += 1; | |
1777 size -= 1; | |
1778 } | |
1779 } | |
1780 | |
1781 return result; | |
1782 } | |
1783 }; | |
1784 | |
1785 struct latin1_decoder | |
1786 { | |
1787 typedef uint8_t type; | |
1788 | |
1789 template <typename Traits> static inline typename Traits::value_type process(const uint8_t* data, size_t size, typename Traits::value_type result, Traits) | |
1790 { | |
1791 while (size) | |
1792 { | |
1793 result = Traits::low(result, *data); | |
1794 data += 1; | |
1795 size -= 1; | |
1796 } | |
1797 | |
1798 return result; | |
1799 } | |
1800 }; | |
1801 | |
1802 template <size_t size> struct wchar_selector; | |
1803 | |
1804 template <> struct wchar_selector<2> | |
1805 { | |
1806 typedef uint16_t type; | |
1807 typedef utf16_counter counter; | |
1808 typedef utf16_writer writer; | |
1809 typedef utf16_decoder<opt_false> decoder; | |
1810 }; | |
1811 | |
1812 template <> struct wchar_selector<4> | |
1813 { | |
1814 typedef uint32_t type; | |
1815 typedef utf32_counter counter; | |
1816 typedef utf32_writer writer; | |
1817 typedef utf32_decoder<opt_false> decoder; | |
1818 }; | |
1819 | |
1820 typedef wchar_selector<sizeof(wchar_t)>::counter wchar_counter; | |
1821 typedef wchar_selector<sizeof(wchar_t)>::writer wchar_writer; | |
1822 | |
1823 struct wchar_decoder | |
1824 { | |
1825 typedef wchar_t type; | |
1826 | |
1827 template <typename Traits> static inline typename Traits::value_type process(const wchar_t* data, size_t size, typename Traits::value_type result, Traits traits) | |
1828 { | |
1829 typedef wchar_selector<sizeof(wchar_t)>::decoder decoder; | |
1830 | |
1831 return decoder::process(reinterpret_cast<const typename decoder::type*>(data), size, result, traits); | |
1832 } | |
1833 }; | |
1834 | |
1835 #ifdef PUGIXML_WCHAR_MODE | |
1836 PUGI__FN void convert_wchar_endian_swap(wchar_t* result, const wchar_t* data, size_t length) | |
1837 { | |
1838 for (size_t i = 0; i < length; ++i) | |
1839 result[i] = static_cast<wchar_t>(endian_swap(static_cast<wchar_selector<sizeof(wchar_t)>::type>(data[i]))); | |
1840 } | |
1841 #endif | |
1842 PUGI__NS_END | |
1843 | |
1844 PUGI__NS_BEGIN | |
1845 enum chartype_t | |
1846 { | |
1847 ct_parse_pcdata = 1, // \0, &, \r, < | |
1848 ct_parse_attr = 2, // \0, &, \r, ', " | |
1849 ct_parse_attr_ws = 4, // \0, &, \r, ', ", \n, tab | |
1850 ct_space = 8, // \r, \n, space, tab | |
1851 ct_parse_cdata = 16, // \0, ], >, \r | |
1852 ct_parse_comment = 32, // \0, -, >, \r | |
1853 ct_symbol = 64, // Any symbol > 127, a-z, A-Z, 0-9, _, :, -, . | |
1854 ct_start_symbol = 128 // Any symbol > 127, a-z, A-Z, _, : | |
1855 }; | |
1856 | |
1857 static const unsigned char chartype_table[256] = | |
1858 { | |
1859 55, 0, 0, 0, 0, 0, 0, 0, 0, 12, 12, 0, 0, 63, 0, 0, // 0-15 | |
1860 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 16-31 | |
1861 8, 0, 6, 0, 0, 0, 7, 6, 0, 0, 0, 0, 0, 96, 64, 0, // 32-47 | |
1862 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 192, 0, 1, 0, 48, 0, // 48-63 | |
1863 0, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, // 64-79 | |
1864 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 0, 0, 16, 0, 192, // 80-95 | |
1865 0, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, // 96-111 | |
1866 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 0, 0, 0, 0, 0, // 112-127 | |
1867 | |
1868 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, // 128+ | |
1869 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, | |
1870 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, | |
1871 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, | |
1872 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, | |
1873 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, | |
1874 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, | |
1875 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192, 192 | |
1876 }; | |
1877 | |
1878 enum chartypex_t | |
1879 { | |
1880 ctx_special_pcdata = 1, // Any symbol >= 0 and < 32 (except \t, \r, \n), &, <, > | |
1881 ctx_special_attr = 2, // Any symbol >= 0 and < 32, &, <, ", ' | |
1882 ctx_start_symbol = 4, // Any symbol > 127, a-z, A-Z, _ | |
1883 ctx_digit = 8, // 0-9 | |
1884 ctx_symbol = 16 // Any symbol > 127, a-z, A-Z, 0-9, _, -, . | |
1885 }; | |
1886 | |
1887 static const unsigned char chartypex_table[256] = | |
1888 { | |
1889 3, 3, 3, 3, 3, 3, 3, 3, 3, 2, 2, 3, 3, 2, 3, 3, // 0-15 | |
1890 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, // 16-31 | |
1891 0, 0, 2, 0, 0, 0, 3, 2, 0, 0, 0, 0, 0, 16, 16, 0, // 32-47 | |
1892 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 0, 0, 3, 0, 1, 0, // 48-63 | |
1893 | |
1894 0, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, // 64-79 | |
1895 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 0, 0, 0, 0, 20, // 80-95 | |
1896 0, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, // 96-111 | |
1897 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 0, 0, 0, 0, 0, // 112-127 | |
1898 | |
1899 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, // 128+ | |
1900 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, | |
1901 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, | |
1902 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, | |
1903 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, | |
1904 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, | |
1905 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, | |
1906 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20 | |
1907 }; | |
1908 | |
1909 #ifdef PUGIXML_WCHAR_MODE | |
1910 #define PUGI__IS_CHARTYPE_IMPL(c, ct, table) ((static_cast<unsigned int>(c) < 128 ? table[static_cast<unsigned int>(c)] : table[128]) & (ct)) | |
1911 #else | |
1912 #define PUGI__IS_CHARTYPE_IMPL(c, ct, table) (table[static_cast<unsigned char>(c)] & (ct)) | |
1913 #endif | |
1914 | |
1915 #define PUGI__IS_CHARTYPE(c, ct) PUGI__IS_CHARTYPE_IMPL(c, ct, chartype_table) | |
1916 #define PUGI__IS_CHARTYPEX(c, ct) PUGI__IS_CHARTYPE_IMPL(c, ct, chartypex_table) | |
1917 | |
1918 PUGI__FN bool is_little_endian() | |
1919 { | |
1920 unsigned int ui = 1; | |
1921 | |
1922 return *reinterpret_cast<unsigned char*>(&ui) == 1; | |
1923 } | |
1924 | |
1925 PUGI__FN xml_encoding get_wchar_encoding() | |
1926 { | |
1927 PUGI__STATIC_ASSERT(sizeof(wchar_t) == 2 || sizeof(wchar_t) == 4); | |
1928 | |
1929 if (sizeof(wchar_t) == 2) | |
1930 return is_little_endian() ? encoding_utf16_le : encoding_utf16_be; | |
1931 else | |
1932 return is_little_endian() ? encoding_utf32_le : encoding_utf32_be; | |
1933 } | |
1934 | |
1935 PUGI__FN bool parse_declaration_encoding(const uint8_t* data, size_t size, const uint8_t*& out_encoding, size_t& out_length) | |
1936 { | |
1937 #define PUGI__SCANCHAR(ch) { if (offset >= size || data[offset] != ch) return false; offset++; } | |
1938 #define PUGI__SCANCHARTYPE(ct) { while (offset < size && PUGI__IS_CHARTYPE(data[offset], ct)) offset++; } | |
1939 | |
1940 // check if we have a non-empty XML declaration | |
1941 if (size < 6 || !((data[0] == '<') & (data[1] == '?') & (data[2] == 'x') & (data[3] == 'm') & (data[4] == 'l') && PUGI__IS_CHARTYPE(data[5], ct_space))) | |
1942 return false; | |
1943 | |
1944 // scan XML declaration until the encoding field | |
1945 for (size_t i = 6; i + 1 < size; ++i) | |
1946 { | |
1947 // declaration can not contain ? in quoted values | |
1948 if (data[i] == '?') | |
1949 return false; | |
1950 | |
1951 if (data[i] == 'e' && data[i + 1] == 'n') | |
1952 { | |
1953 size_t offset = i; | |
1954 | |
1955 // encoding follows the version field which can't contain 'en' so this has to be the encoding if XML is well formed | |
1956 PUGI__SCANCHAR('e'); PUGI__SCANCHAR('n'); PUGI__SCANCHAR('c'); PUGI__SCANCHAR('o'); | |
1957 PUGI__SCANCHAR('d'); PUGI__SCANCHAR('i'); PUGI__SCANCHAR('n'); PUGI__SCANCHAR('g'); | |
1958 | |
1959 // S? = S? | |
1960 PUGI__SCANCHARTYPE(ct_space); | |
1961 PUGI__SCANCHAR('='); | |
1962 PUGI__SCANCHARTYPE(ct_space); | |
1963 | |
1964 // the only two valid delimiters are ' and " | |
1965 uint8_t delimiter = (offset < size && data[offset] == '"') ? '"' : '\''; | |
1966 | |
1967 PUGI__SCANCHAR(delimiter); | |
1968 | |
1969 size_t start = offset; | |
1970 | |
1971 out_encoding = data + offset; | |
1972 | |
1973 PUGI__SCANCHARTYPE(ct_symbol); | |
1974 | |
1975 out_length = offset - start; | |
1976 | |
1977 PUGI__SCANCHAR(delimiter); | |
1978 | |
1979 return true; | |
1980 } | |
1981 } | |
1982 | |
1983 return false; | |
1984 | |
1985 #undef PUGI__SCANCHAR | |
1986 #undef PUGI__SCANCHARTYPE | |
1987 } | |
1988 | |
1989 PUGI__FN xml_encoding guess_buffer_encoding(const uint8_t* data, size_t size) | |
1990 { | |
1991 // skip encoding autodetection if input buffer is too small | |
1992 if (size < 4) return encoding_utf8; | |
1993 | |
1994 uint8_t d0 = data[0], d1 = data[1], d2 = data[2], d3 = data[3]; | |
1995 | |
1996 // look for BOM in first few bytes | |
1997 if (d0 == 0 && d1 == 0 && d2 == 0xfe && d3 == 0xff) return encoding_utf32_be; | |
1998 if (d0 == 0xff && d1 == 0xfe && d2 == 0 && d3 == 0) return encoding_utf32_le; | |
1999 if (d0 == 0xfe && d1 == 0xff) return encoding_utf16_be; | |
2000 if (d0 == 0xff && d1 == 0xfe) return encoding_utf16_le; | |
2001 if (d0 == 0xef && d1 == 0xbb && d2 == 0xbf) return encoding_utf8; | |
2002 | |
2003 // look for <, <? or <?xm in various encodings | |
2004 if (d0 == 0 && d1 == 0 && d2 == 0 && d3 == 0x3c) return encoding_utf32_be; | |
2005 if (d0 == 0x3c && d1 == 0 && d2 == 0 && d3 == 0) return encoding_utf32_le; | |
2006 if (d0 == 0 && d1 == 0x3c && d2 == 0 && d3 == 0x3f) return encoding_utf16_be; | |
2007 if (d0 == 0x3c && d1 == 0 && d2 == 0x3f && d3 == 0) return encoding_utf16_le; | |
2008 | |
2009 // look for utf16 < followed by node name (this may fail, but is better than utf8 since it's zero terminated so early) | |
2010 if (d0 == 0 && d1 == 0x3c) return encoding_utf16_be; | |
2011 if (d0 == 0x3c && d1 == 0) return encoding_utf16_le; | |
2012 | |
2013 // no known BOM detected; parse declaration | |
2014 const uint8_t* enc = 0; | |
2015 size_t enc_length = 0; | |
2016 | |
2017 if (d0 == 0x3c && d1 == 0x3f && d2 == 0x78 && d3 == 0x6d && parse_declaration_encoding(data, size, enc, enc_length)) | |
2018 { | |
2019 // iso-8859-1 (case-insensitive) | |
2020 if (enc_length == 10 | |
2021 && (enc[0] | ' ') == 'i' && (enc[1] | ' ') == 's' && (enc[2] | ' ') == 'o' | |
2022 && enc[3] == '-' && enc[4] == '8' && enc[5] == '8' && enc[6] == '5' && enc[7] == '9' | |
2023 && enc[8] == '-' && enc[9] == '1') | |
2024 return encoding_latin1; | |
2025 | |
2026 // latin1 (case-insensitive) | |
2027 if (enc_length == 6 | |
2028 && (enc[0] | ' ') == 'l' && (enc[1] | ' ') == 'a' && (enc[2] | ' ') == 't' | |
2029 && (enc[3] | ' ') == 'i' && (enc[4] | ' ') == 'n' | |
2030 && enc[5] == '1') | |
2031 return encoding_latin1; | |
2032 } | |
2033 | |
2034 return encoding_utf8; | |
2035 } | |
2036 | |
2037 PUGI__FN xml_encoding get_buffer_encoding(xml_encoding encoding, const void* contents, size_t size) | |
2038 { | |
2039 // replace wchar encoding with utf implementation | |
2040 if (encoding == encoding_wchar) return get_wchar_encoding(); | |
2041 | |
2042 // replace utf16 encoding with utf16 with specific endianness | |
2043 if (encoding == encoding_utf16) return is_little_endian() ? encoding_utf16_le : encoding_utf16_be; | |
2044 | |
2045 // replace utf32 encoding with utf32 with specific endianness | |
2046 if (encoding == encoding_utf32) return is_little_endian() ? encoding_utf32_le : encoding_utf32_be; | |
2047 | |
2048 // only do autodetection if no explicit encoding is requested | |
2049 if (encoding != encoding_auto) return encoding; | |
2050 | |
2051 // try to guess encoding (based on XML specification, Appendix F.1) | |
2052 const uint8_t* data = static_cast<const uint8_t*>(contents); | |
2053 | |
2054 return guess_buffer_encoding(data, size); | |
2055 } | |
2056 | |
2057 PUGI__FN bool get_mutable_buffer(char_t*& out_buffer, size_t& out_length, const void* contents, size_t size, bool is_mutable) | |
2058 { | |
2059 size_t length = size / sizeof(char_t); | |
2060 | |
2061 if (is_mutable) | |
2062 { | |
2063 out_buffer = static_cast<char_t*>(const_cast<void*>(contents)); | |
2064 out_length = length; | |
2065 } | |
2066 else | |
2067 { | |
2068 char_t* buffer = static_cast<char_t*>(xml_memory::allocate((length + 1) * sizeof(char_t))); | |
2069 if (!buffer) return false; | |
2070 | |
2071 if (contents) | |
2072 memcpy(buffer, contents, length * sizeof(char_t)); | |
2073 else | |
2074 assert(length == 0); | |
2075 | |
2076 buffer[length] = 0; | |
2077 | |
2078 out_buffer = buffer; | |
2079 out_length = length + 1; | |
2080 } | |
2081 | |
2082 return true; | |
2083 } | |
2084 | |
2085 #ifdef PUGIXML_WCHAR_MODE | |
2086 PUGI__FN bool need_endian_swap_utf(xml_encoding le, xml_encoding re) | |
2087 { | |
2088 return (le == encoding_utf16_be && re == encoding_utf16_le) || (le == encoding_utf16_le && re == encoding_utf16_be) || | |
2089 (le == encoding_utf32_be && re == encoding_utf32_le) || (le == encoding_utf32_le && re == encoding_utf32_be); | |
2090 } | |
2091 | |
2092 PUGI__FN bool convert_buffer_endian_swap(char_t*& out_buffer, size_t& out_length, const void* contents, size_t size, bool is_mutable) | |
2093 { | |
2094 const char_t* data = static_cast<const char_t*>(contents); | |
2095 size_t length = size / sizeof(char_t); | |
2096 | |
2097 if (is_mutable) | |
2098 { | |
2099 char_t* buffer = const_cast<char_t*>(data); | |
2100 | |
2101 convert_wchar_endian_swap(buffer, data, length); | |
2102 | |
2103 out_buffer = buffer; | |
2104 out_length = length; | |
2105 } | |
2106 else | |
2107 { | |
2108 char_t* buffer = static_cast<char_t*>(xml_memory::allocate((length + 1) * sizeof(char_t))); | |
2109 if (!buffer) return false; | |
2110 | |
2111 convert_wchar_endian_swap(buffer, data, length); | |
2112 buffer[length] = 0; | |
2113 | |
2114 out_buffer = buffer; | |
2115 out_length = length + 1; | |
2116 } | |
2117 | |
2118 return true; | |
2119 } | |
2120 | |
2121 template <typename D> PUGI__FN bool convert_buffer_generic(char_t*& out_buffer, size_t& out_length, const void* contents, size_t size, D) | |
2122 { | |
2123 const typename D::type* data = static_cast<const typename D::type*>(contents); | |
2124 size_t data_length = size / sizeof(typename D::type); | |
2125 | |
2126 // first pass: get length in wchar_t units | |
2127 size_t length = D::process(data, data_length, 0, wchar_counter()); | |
2128 | |
2129 // allocate buffer of suitable length | |
2130 char_t* buffer = static_cast<char_t*>(xml_memory::allocate((length + 1) * sizeof(char_t))); | |
2131 if (!buffer) return false; | |
2132 | |
2133 // second pass: convert utf16 input to wchar_t | |
2134 wchar_writer::value_type obegin = reinterpret_cast<wchar_writer::value_type>(buffer); | |
2135 wchar_writer::value_type oend = D::process(data, data_length, obegin, wchar_writer()); | |
2136 | |
2137 assert(oend == obegin + length); | |
2138 *oend = 0; | |
2139 | |
2140 out_buffer = buffer; | |
2141 out_length = length + 1; | |
2142 | |
2143 return true; | |
2144 } | |
2145 | |
2146 PUGI__FN bool convert_buffer(char_t*& out_buffer, size_t& out_length, xml_encoding encoding, const void* contents, size_t size, bool is_mutable) | |
2147 { | |
2148 // get native encoding | |
2149 xml_encoding wchar_encoding = get_wchar_encoding(); | |
2150 | |
2151 // fast path: no conversion required | |
2152 if (encoding == wchar_encoding) | |
2153 return get_mutable_buffer(out_buffer, out_length, contents, size, is_mutable); | |
2154 | |
2155 // only endian-swapping is required | |
2156 if (need_endian_swap_utf(encoding, wchar_encoding)) | |
2157 return convert_buffer_endian_swap(out_buffer, out_length, contents, size, is_mutable); | |
2158 | |
2159 // source encoding is utf8 | |
2160 if (encoding == encoding_utf8) | |
2161 return convert_buffer_generic(out_buffer, out_length, contents, size, utf8_decoder()); | |
2162 | |
2163 // source encoding is utf16 | |
2164 if (encoding == encoding_utf16_be || encoding == encoding_utf16_le) | |
2165 { | |
2166 xml_encoding native_encoding = is_little_endian() ? encoding_utf16_le : encoding_utf16_be; | |
2167 | |
2168 return (native_encoding == encoding) ? | |
2169 convert_buffer_generic(out_buffer, out_length, contents, size, utf16_decoder<opt_false>()) : | |
2170 convert_buffer_generic(out_buffer, out_length, contents, size, utf16_decoder<opt_true>()); | |
2171 } | |
2172 | |
2173 // source encoding is utf32 | |
2174 if (encoding == encoding_utf32_be || encoding == encoding_utf32_le) | |
2175 { | |
2176 xml_encoding native_encoding = is_little_endian() ? encoding_utf32_le : encoding_utf32_be; | |
2177 | |
2178 return (native_encoding == encoding) ? | |
2179 convert_buffer_generic(out_buffer, out_length, contents, size, utf32_decoder<opt_false>()) : | |
2180 convert_buffer_generic(out_buffer, out_length, contents, size, utf32_decoder<opt_true>()); | |
2181 } | |
2182 | |
2183 // source encoding is latin1 | |
2184 if (encoding == encoding_latin1) | |
2185 return convert_buffer_generic(out_buffer, out_length, contents, size, latin1_decoder()); | |
2186 | |
2187 assert(false && "Invalid encoding"); // unreachable | |
2188 return false; | |
2189 } | |
2190 #else | |
2191 template <typename D> PUGI__FN bool convert_buffer_generic(char_t*& out_buffer, size_t& out_length, const void* contents, size_t size, D) | |
2192 { | |
2193 const typename D::type* data = static_cast<const typename D::type*>(contents); | |
2194 size_t data_length = size / sizeof(typename D::type); | |
2195 | |
2196 // first pass: get length in utf8 units | |
2197 size_t length = D::process(data, data_length, 0, utf8_counter()); | |
2198 | |
2199 // allocate buffer of suitable length | |
2200 char_t* buffer = static_cast<char_t*>(xml_memory::allocate((length + 1) * sizeof(char_t))); | |
2201 if (!buffer) return false; | |
2202 | |
2203 // second pass: convert utf16 input to utf8 | |
2204 uint8_t* obegin = reinterpret_cast<uint8_t*>(buffer); | |
2205 uint8_t* oend = D::process(data, data_length, obegin, utf8_writer()); | |
2206 | |
2207 assert(oend == obegin + length); | |
2208 *oend = 0; | |
2209 | |
2210 out_buffer = buffer; | |
2211 out_length = length + 1; | |
2212 | |
2213 return true; | |
2214 } | |
2215 | |
2216 PUGI__FN size_t get_latin1_7bit_prefix_length(const uint8_t* data, size_t size) | |
2217 { | |
2218 for (size_t i = 0; i < size; ++i) | |
2219 if (data[i] > 127) | |
2220 return i; | |
2221 | |
2222 return size; | |
2223 } | |
2224 | |
2225 PUGI__FN bool convert_buffer_latin1(char_t*& out_buffer, size_t& out_length, const void* contents, size_t size, bool is_mutable) | |
2226 { | |
2227 const uint8_t* data = static_cast<const uint8_t*>(contents); | |
2228 size_t data_length = size; | |
2229 | |
2230 // get size of prefix that does not need utf8 conversion | |
2231 size_t prefix_length = get_latin1_7bit_prefix_length(data, data_length); | |
2232 assert(prefix_length <= data_length); | |
2233 | |
2234 const uint8_t* postfix = data + prefix_length; | |
2235 size_t postfix_length = data_length - prefix_length; | |
2236 | |
2237 // if no conversion is needed, just return the original buffer | |
2238 if (postfix_length == 0) return get_mutable_buffer(out_buffer, out_length, contents, size, is_mutable); | |
2239 | |
2240 // first pass: get length in utf8 units | |
2241 size_t length = prefix_length + latin1_decoder::process(postfix, postfix_length, 0, utf8_counter()); | |
2242 | |
2243 // allocate buffer of suitable length | |
2244 char_t* buffer = static_cast<char_t*>(xml_memory::allocate((length + 1) * sizeof(char_t))); | |
2245 if (!buffer) return false; | |
2246 | |
2247 // second pass: convert latin1 input to utf8 | |
2248 memcpy(buffer, data, prefix_length); | |
2249 | |
2250 uint8_t* obegin = reinterpret_cast<uint8_t*>(buffer); | |
2251 uint8_t* oend = latin1_decoder::process(postfix, postfix_length, obegin + prefix_length, utf8_writer()); | |
2252 | |
2253 assert(oend == obegin + length); | |
2254 *oend = 0; | |
2255 | |
2256 out_buffer = buffer; | |
2257 out_length = length + 1; | |
2258 | |
2259 return true; | |
2260 } | |
2261 | |
2262 PUGI__FN bool convert_buffer(char_t*& out_buffer, size_t& out_length, xml_encoding encoding, const void* contents, size_t size, bool is_mutable) | |
2263 { | |
2264 // fast path: no conversion required | |
2265 if (encoding == encoding_utf8) | |
2266 return get_mutable_buffer(out_buffer, out_length, contents, size, is_mutable); | |
2267 | |
2268 // source encoding is utf16 | |
2269 if (encoding == encoding_utf16_be || encoding == encoding_utf16_le) | |
2270 { | |
2271 xml_encoding native_encoding = is_little_endian() ? encoding_utf16_le : encoding_utf16_be; | |
2272 | |
2273 return (native_encoding == encoding) ? | |
2274 convert_buffer_generic(out_buffer, out_length, contents, size, utf16_decoder<opt_false>()) : | |
2275 convert_buffer_generic(out_buffer, out_length, contents, size, utf16_decoder<opt_true>()); | |
2276 } | |
2277 | |
2278 // source encoding is utf32 | |
2279 if (encoding == encoding_utf32_be || encoding == encoding_utf32_le) | |
2280 { | |
2281 xml_encoding native_encoding = is_little_endian() ? encoding_utf32_le : encoding_utf32_be; | |
2282 | |
2283 return (native_encoding == encoding) ? | |
2284 convert_buffer_generic(out_buffer, out_length, contents, size, utf32_decoder<opt_false>()) : | |
2285 convert_buffer_generic(out_buffer, out_length, contents, size, utf32_decoder<opt_true>()); | |
2286 } | |
2287 | |
2288 // source encoding is latin1 | |
2289 if (encoding == encoding_latin1) | |
2290 return convert_buffer_latin1(out_buffer, out_length, contents, size, is_mutable); | |
2291 | |
2292 assert(false && "Invalid encoding"); // unreachable | |
2293 return false; | |
2294 } | |
2295 #endif | |
2296 | |
2297 PUGI__FN size_t as_utf8_begin(const wchar_t* str, size_t length) | |
2298 { | |
2299 // get length in utf8 characters | |
2300 return wchar_decoder::process(str, length, 0, utf8_counter()); | |
2301 } | |
2302 | |
2303 PUGI__FN void as_utf8_end(char* buffer, size_t size, const wchar_t* str, size_t length) | |
2304 { | |
2305 // convert to utf8 | |
2306 uint8_t* begin = reinterpret_cast<uint8_t*>(buffer); | |
2307 uint8_t* end = wchar_decoder::process(str, length, begin, utf8_writer()); | |
2308 | |
2309 assert(begin + size == end); | |
2310 (void)!end; | |
2311 (void)!size; | |
2312 } | |
2313 | |
2314 #ifndef PUGIXML_NO_STL | |
2315 PUGI__FN std::string as_utf8_impl(const wchar_t* str, size_t length) | |
2316 { | |
2317 // first pass: get length in utf8 characters | |
2318 size_t size = as_utf8_begin(str, length); | |
2319 | |
2320 // allocate resulting string | |
2321 std::string result; | |
2322 result.resize(size); | |
2323 | |
2324 // second pass: convert to utf8 | |
2325 if (size > 0) as_utf8_end(&result[0], size, str, length); | |
2326 | |
2327 return result; | |
2328 } | |
2329 | |
2330 PUGI__FN std::basic_string<wchar_t> as_wide_impl(const char* str, size_t size) | |
2331 { | |
2332 const uint8_t* data = reinterpret_cast<const uint8_t*>(str); | |
2333 | |
2334 // first pass: get length in wchar_t units | |
2335 size_t length = utf8_decoder::process(data, size, 0, wchar_counter()); | |
2336 | |
2337 // allocate resulting string | |
2338 std::basic_string<wchar_t> result; | |
2339 result.resize(length); | |
2340 | |
2341 // second pass: convert to wchar_t | |
2342 if (length > 0) | |
2343 { | |
2344 wchar_writer::value_type begin = reinterpret_cast<wchar_writer::value_type>(&result[0]); | |
2345 wchar_writer::value_type end = utf8_decoder::process(data, size, begin, wchar_writer()); | |
2346 | |
2347 assert(begin + length == end); | |
2348 (void)!end; | |
2349 } | |
2350 | |
2351 return result; | |
2352 } | |
2353 #endif | |
2354 | |
2355 template <typename Header> | |
2356 inline bool strcpy_insitu_allow(size_t length, const Header& header, uintptr_t header_mask, char_t* target) | |
2357 { | |
2358 // never reuse shared memory | |
2359 if (header & xml_memory_page_contents_shared_mask) return false; | |
2360 | |
2361 size_t target_length = strlength(target); | |
2362 | |
2363 // always reuse document buffer memory if possible | |
2364 if ((header & header_mask) == 0) return target_length >= length; | |
2365 | |
2366 // reuse heap memory if waste is not too great | |
2367 const size_t reuse_threshold = 32; | |
2368 | |
2369 return target_length >= length && (target_length < reuse_threshold || target_length - length < target_length / 2); | |
2370 } | |
2371 | |
2372 template <typename String, typename Header> | |
2373 PUGI__FN bool strcpy_insitu(String& dest, Header& header, uintptr_t header_mask, const char_t* source, size_t source_length) | |
2374 { | |
2375 if (source_length == 0) | |
2376 { | |
2377 // empty string and null pointer are equivalent, so just deallocate old memory | |
2378 xml_allocator* alloc = PUGI__GETPAGE_IMPL(header)->allocator; | |
2379 | |
2380 if (header & header_mask) alloc->deallocate_string(dest); | |
2381 | |
2382 // mark the string as not allocated | |
2383 dest = 0; | |
2384 header &= ~header_mask; | |
2385 | |
2386 return true; | |
2387 } | |
2388 else if (dest && strcpy_insitu_allow(source_length, header, header_mask, dest)) | |
2389 { | |
2390 // we can reuse old buffer, so just copy the new data (including zero terminator) | |
2391 memcpy(dest, source, source_length * sizeof(char_t)); | |
2392 dest[source_length] = 0; | |
2393 | |
2394 return true; | |
2395 } | |
2396 else | |
2397 { | |
2398 xml_allocator* alloc = PUGI__GETPAGE_IMPL(header)->allocator; | |
2399 | |
2400 if (!alloc->reserve()) return false; | |
2401 | |
2402 // allocate new buffer | |
2403 char_t* buf = alloc->allocate_string(source_length + 1); | |
2404 if (!buf) return false; | |
2405 | |
2406 // copy the string (including zero terminator) | |
2407 memcpy(buf, source, source_length * sizeof(char_t)); | |
2408 buf[source_length] = 0; | |
2409 | |
2410 // deallocate old buffer (*after* the above to protect against overlapping memory and/or allocation failures) | |
2411 if (header & header_mask) alloc->deallocate_string(dest); | |
2412 | |
2413 // the string is now allocated, so set the flag | |
2414 dest = buf; | |
2415 header |= header_mask; | |
2416 | |
2417 return true; | |
2418 } | |
2419 } | |
2420 | |
2421 struct gap | |
2422 { | |
2423 char_t* end; | |
2424 size_t size; | |
2425 | |
2426 gap(): end(0), size(0) | |
2427 { | |
2428 } | |
2429 | |
2430 // Push new gap, move s count bytes further (skipping the gap). | |
2431 // Collapse previous gap. | |
2432 void push(char_t*& s, size_t count) | |
2433 { | |
2434 if (end) // there was a gap already; collapse it | |
2435 { | |
2436 // Move [old_gap_end, new_gap_start) to [old_gap_start, ...) | |
2437 assert(s >= end); | |
2438 memmove(end - size, end, reinterpret_cast<char*>(s) - reinterpret_cast<char*>(end)); | |
2439 } | |
2440 | |
2441 s += count; // end of current gap | |
2442 | |
2443 // "merge" two gaps | |
2444 end = s; | |
2445 size += count; | |
2446 } | |
2447 | |
2448 // Collapse all gaps, return past-the-end pointer | |
2449 char_t* flush(char_t* s) | |
2450 { | |
2451 if (end) | |
2452 { | |
2453 // Move [old_gap_end, current_pos) to [old_gap_start, ...) | |
2454 assert(s >= end); | |
2455 memmove(end - size, end, reinterpret_cast<char*>(s) - reinterpret_cast<char*>(end)); | |
2456 | |
2457 return s - size; | |
2458 } | |
2459 else return s; | |
2460 } | |
2461 }; | |
2462 | |
2463 PUGI__FN char_t* strconv_escape(char_t* s, gap& g) | |
2464 { | |
2465 char_t* stre = s + 1; | |
2466 | |
2467 switch (*stre) | |
2468 { | |
2469 case '#': // &#... | |
2470 { | |
2471 unsigned int ucsc = 0; | |
2472 | |
2473 if (stre[1] == 'x') // &#x... (hex code) | |
2474 { | |
2475 stre += 2; | |
2476 | |
2477 char_t ch = *stre; | |
2478 | |
2479 if (ch == ';') return stre; | |
2480 | |
2481 for (;;) | |
2482 { | |
2483 if (static_cast<unsigned int>(ch - '0') <= 9) | |
2484 ucsc = 16 * ucsc + (ch - '0'); | |
2485 else if (static_cast<unsigned int>((ch | ' ') - 'a') <= 5) | |
2486 ucsc = 16 * ucsc + ((ch | ' ') - 'a' + 10); | |
2487 else if (ch == ';') | |
2488 break; | |
2489 else // cancel | |
2490 return stre; | |
2491 | |
2492 ch = *++stre; | |
2493 } | |
2494 | |
2495 ++stre; | |
2496 } | |
2497 else // &#... (dec code) | |
2498 { | |
2499 char_t ch = *++stre; | |
2500 | |
2501 if (ch == ';') return stre; | |
2502 | |
2503 for (;;) | |
2504 { | |
2505 if (static_cast<unsigned int>(ch - '0') <= 9) | |
2506 ucsc = 10 * ucsc + (ch - '0'); | |
2507 else if (ch == ';') | |
2508 break; | |
2509 else // cancel | |
2510 return stre; | |
2511 | |
2512 ch = *++stre; | |
2513 } | |
2514 | |
2515 ++stre; | |
2516 } | |
2517 | |
2518 #ifdef PUGIXML_WCHAR_MODE | |
2519 s = reinterpret_cast<char_t*>(wchar_writer::any(reinterpret_cast<wchar_writer::value_type>(s), ucsc)); | |
2520 #else | |
2521 s = reinterpret_cast<char_t*>(utf8_writer::any(reinterpret_cast<uint8_t*>(s), ucsc)); | |
2522 #endif | |
2523 | |
2524 g.push(s, stre - s); | |
2525 return stre; | |
2526 } | |
2527 | |
2528 case 'a': // &a | |
2529 { | |
2530 ++stre; | |
2531 | |
2532 if (*stre == 'm') // &am | |
2533 { | |
2534 if (*++stre == 'p' && *++stre == ';') // & | |
2535 { | |
2536 *s++ = '&'; | |
2537 ++stre; | |
2538 | |
2539 g.push(s, stre - s); | |
2540 return stre; | |
2541 } | |
2542 } | |
2543 else if (*stre == 'p') // &ap | |
2544 { | |
2545 if (*++stre == 'o' && *++stre == 's' && *++stre == ';') // ' | |
2546 { | |
2547 *s++ = '\''; | |
2548 ++stre; | |
2549 | |
2550 g.push(s, stre - s); | |
2551 return stre; | |
2552 } | |
2553 } | |
2554 break; | |
2555 } | |
2556 | |
2557 case 'g': // &g | |
2558 { | |
2559 if (*++stre == 't' && *++stre == ';') // > | |
2560 { | |
2561 *s++ = '>'; | |
2562 ++stre; | |
2563 | |
2564 g.push(s, stre - s); | |
2565 return stre; | |
2566 } | |
2567 break; | |
2568 } | |
2569 | |
2570 case 'l': // &l | |
2571 { | |
2572 if (*++stre == 't' && *++stre == ';') // < | |
2573 { | |
2574 *s++ = '<'; | |
2575 ++stre; | |
2576 | |
2577 g.push(s, stre - s); | |
2578 return stre; | |
2579 } | |
2580 break; | |
2581 } | |
2582 | |
2583 case 'q': // &q | |
2584 { | |
2585 if (*++stre == 'u' && *++stre == 'o' && *++stre == 't' && *++stre == ';') // " | |
2586 { | |
2587 *s++ = '"'; | |
2588 ++stre; | |
2589 | |
2590 g.push(s, stre - s); | |
2591 return stre; | |
2592 } | |
2593 break; | |
2594 } | |
2595 | |
2596 default: | |
2597 break; | |
2598 } | |
2599 | |
2600 return stre; | |
2601 } | |
2602 | |
2603 // Parser utilities | |
2604 #define PUGI__ENDSWITH(c, e) ((c) == (e) || ((c) == 0 && endch == (e))) | |
2605 #define PUGI__SKIPWS() { while (PUGI__IS_CHARTYPE(*s, ct_space)) ++s; } | |
2606 #define PUGI__OPTSET(OPT) ( optmsk & (OPT) ) | |
2607 #define PUGI__PUSHNODE(TYPE) { cursor = append_new_node(cursor, *alloc, TYPE); if (!cursor) PUGI__THROW_ERROR(status_out_of_memory, s); } | |
2608 #define PUGI__POPNODE() { cursor = cursor->parent; } | |
2609 #define PUGI__SCANFOR(X) { while (*s != 0 && !(X)) ++s; } | |
2610 #define PUGI__SCANWHILE(X) { while (X) ++s; } | |
2611 #define PUGI__SCANWHILE_UNROLL(X) { for (;;) { char_t ss = s[0]; if (PUGI__UNLIKELY(!(X))) { break; } ss = s[1]; if (PUGI__UNLIKELY(!(X))) { s += 1; break; } ss = s[2]; if (PUGI__UNLIKELY(!(X))) { s += 2; break; } ss = s[3]; if (PUGI__UNLIKELY(!(X))) { s += 3; break; } s += 4; } } | |
2612 #define PUGI__ENDSEG() { ch = *s; *s = 0; ++s; } | |
2613 #define PUGI__THROW_ERROR(err, m) return error_offset = m, error_status = err, static_cast<char_t*>(0) | |
2614 #define PUGI__CHECK_ERROR(err, m) { if (*s == 0) PUGI__THROW_ERROR(err, m); } | |
2615 | |
2616 PUGI__FN char_t* strconv_comment(char_t* s, char_t endch) | |
2617 { | |
2618 gap g; | |
2619 | |
2620 while (true) | |
2621 { | |
2622 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPE(ss, ct_parse_comment)); | |
2623 | |
2624 if (*s == '\r') // Either a single 0x0d or 0x0d 0x0a pair | |
2625 { | |
2626 *s++ = '\n'; // replace first one with 0x0a | |
2627 | |
2628 if (*s == '\n') g.push(s, 1); | |
2629 } | |
2630 else if (s[0] == '-' && s[1] == '-' && PUGI__ENDSWITH(s[2], '>')) // comment ends here | |
2631 { | |
2632 *g.flush(s) = 0; | |
2633 | |
2634 return s + (s[2] == '>' ? 3 : 2); | |
2635 } | |
2636 else if (*s == 0) | |
2637 { | |
2638 return 0; | |
2639 } | |
2640 else ++s; | |
2641 } | |
2642 } | |
2643 | |
2644 PUGI__FN char_t* strconv_cdata(char_t* s, char_t endch) | |
2645 { | |
2646 gap g; | |
2647 | |
2648 while (true) | |
2649 { | |
2650 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPE(ss, ct_parse_cdata)); | |
2651 | |
2652 if (*s == '\r') // Either a single 0x0d or 0x0d 0x0a pair | |
2653 { | |
2654 *s++ = '\n'; // replace first one with 0x0a | |
2655 | |
2656 if (*s == '\n') g.push(s, 1); | |
2657 } | |
2658 else if (s[0] == ']' && s[1] == ']' && PUGI__ENDSWITH(s[2], '>')) // CDATA ends here | |
2659 { | |
2660 *g.flush(s) = 0; | |
2661 | |
2662 return s + 1; | |
2663 } | |
2664 else if (*s == 0) | |
2665 { | |
2666 return 0; | |
2667 } | |
2668 else ++s; | |
2669 } | |
2670 } | |
2671 | |
2672 typedef char_t* (*strconv_pcdata_t)(char_t*); | |
2673 | |
2674 template <typename opt_trim, typename opt_eol, typename opt_escape> struct strconv_pcdata_impl | |
2675 { | |
2676 static char_t* parse(char_t* s) | |
2677 { | |
2678 gap g; | |
2679 | |
2680 char_t* begin = s; | |
2681 | |
2682 while (true) | |
2683 { | |
2684 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPE(ss, ct_parse_pcdata)); | |
2685 | |
2686 if (*s == '<') // PCDATA ends here | |
2687 { | |
2688 char_t* end = g.flush(s); | |
2689 | |
2690 if (opt_trim::value) | |
2691 while (end > begin && PUGI__IS_CHARTYPE(end[-1], ct_space)) | |
2692 --end; | |
2693 | |
2694 *end = 0; | |
2695 | |
2696 return s + 1; | |
2697 } | |
2698 else if (opt_eol::value && *s == '\r') // Either a single 0x0d or 0x0d 0x0a pair | |
2699 { | |
2700 *s++ = '\n'; // replace first one with 0x0a | |
2701 | |
2702 if (*s == '\n') g.push(s, 1); | |
2703 } | |
2704 else if (opt_escape::value && *s == '&') | |
2705 { | |
2706 s = strconv_escape(s, g); | |
2707 } | |
2708 else if (*s == 0) | |
2709 { | |
2710 char_t* end = g.flush(s); | |
2711 | |
2712 if (opt_trim::value) | |
2713 while (end > begin && PUGI__IS_CHARTYPE(end[-1], ct_space)) | |
2714 --end; | |
2715 | |
2716 *end = 0; | |
2717 | |
2718 return s; | |
2719 } | |
2720 else ++s; | |
2721 } | |
2722 } | |
2723 }; | |
2724 | |
2725 PUGI__FN strconv_pcdata_t get_strconv_pcdata(unsigned int optmask) | |
2726 { | |
2727 PUGI__STATIC_ASSERT(parse_escapes == 0x10 && parse_eol == 0x20 && parse_trim_pcdata == 0x0800); | |
2728 | |
2729 switch (((optmask >> 4) & 3) | ((optmask >> 9) & 4)) // get bitmask for flags (trim eol escapes); this simultaneously checks 3 options from assertion above | |
2730 { | |
2731 case 0: return strconv_pcdata_impl<opt_false, opt_false, opt_false>::parse; | |
2732 case 1: return strconv_pcdata_impl<opt_false, opt_false, opt_true>::parse; | |
2733 case 2: return strconv_pcdata_impl<opt_false, opt_true, opt_false>::parse; | |
2734 case 3: return strconv_pcdata_impl<opt_false, opt_true, opt_true>::parse; | |
2735 case 4: return strconv_pcdata_impl<opt_true, opt_false, opt_false>::parse; | |
2736 case 5: return strconv_pcdata_impl<opt_true, opt_false, opt_true>::parse; | |
2737 case 6: return strconv_pcdata_impl<opt_true, opt_true, opt_false>::parse; | |
2738 case 7: return strconv_pcdata_impl<opt_true, opt_true, opt_true>::parse; | |
2739 default: assert(false); return 0; // unreachable | |
2740 } | |
2741 } | |
2742 | |
2743 typedef char_t* (*strconv_attribute_t)(char_t*, char_t); | |
2744 | |
2745 template <typename opt_escape> struct strconv_attribute_impl | |
2746 { | |
2747 static char_t* parse_wnorm(char_t* s, char_t end_quote) | |
2748 { | |
2749 gap g; | |
2750 | |
2751 // trim leading whitespaces | |
2752 if (PUGI__IS_CHARTYPE(*s, ct_space)) | |
2753 { | |
2754 char_t* str = s; | |
2755 | |
2756 do ++str; | |
2757 while (PUGI__IS_CHARTYPE(*str, ct_space)); | |
2758 | |
2759 g.push(s, str - s); | |
2760 } | |
2761 | |
2762 while (true) | |
2763 { | |
2764 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPE(ss, ct_parse_attr_ws | ct_space)); | |
2765 | |
2766 if (*s == end_quote) | |
2767 { | |
2768 char_t* str = g.flush(s); | |
2769 | |
2770 do *str-- = 0; | |
2771 while (PUGI__IS_CHARTYPE(*str, ct_space)); | |
2772 | |
2773 return s + 1; | |
2774 } | |
2775 else if (PUGI__IS_CHARTYPE(*s, ct_space)) | |
2776 { | |
2777 *s++ = ' '; | |
2778 | |
2779 if (PUGI__IS_CHARTYPE(*s, ct_space)) | |
2780 { | |
2781 char_t* str = s + 1; | |
2782 while (PUGI__IS_CHARTYPE(*str, ct_space)) ++str; | |
2783 | |
2784 g.push(s, str - s); | |
2785 } | |
2786 } | |
2787 else if (opt_escape::value && *s == '&') | |
2788 { | |
2789 s = strconv_escape(s, g); | |
2790 } | |
2791 else if (!*s) | |
2792 { | |
2793 return 0; | |
2794 } | |
2795 else ++s; | |
2796 } | |
2797 } | |
2798 | |
2799 static char_t* parse_wconv(char_t* s, char_t end_quote) | |
2800 { | |
2801 gap g; | |
2802 | |
2803 while (true) | |
2804 { | |
2805 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPE(ss, ct_parse_attr_ws)); | |
2806 | |
2807 if (*s == end_quote) | |
2808 { | |
2809 *g.flush(s) = 0; | |
2810 | |
2811 return s + 1; | |
2812 } | |
2813 else if (PUGI__IS_CHARTYPE(*s, ct_space)) | |
2814 { | |
2815 if (*s == '\r') | |
2816 { | |
2817 *s++ = ' '; | |
2818 | |
2819 if (*s == '\n') g.push(s, 1); | |
2820 } | |
2821 else *s++ = ' '; | |
2822 } | |
2823 else if (opt_escape::value && *s == '&') | |
2824 { | |
2825 s = strconv_escape(s, g); | |
2826 } | |
2827 else if (!*s) | |
2828 { | |
2829 return 0; | |
2830 } | |
2831 else ++s; | |
2832 } | |
2833 } | |
2834 | |
2835 static char_t* parse_eol(char_t* s, char_t end_quote) | |
2836 { | |
2837 gap g; | |
2838 | |
2839 while (true) | |
2840 { | |
2841 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPE(ss, ct_parse_attr)); | |
2842 | |
2843 if (*s == end_quote) | |
2844 { | |
2845 *g.flush(s) = 0; | |
2846 | |
2847 return s + 1; | |
2848 } | |
2849 else if (*s == '\r') | |
2850 { | |
2851 *s++ = '\n'; | |
2852 | |
2853 if (*s == '\n') g.push(s, 1); | |
2854 } | |
2855 else if (opt_escape::value && *s == '&') | |
2856 { | |
2857 s = strconv_escape(s, g); | |
2858 } | |
2859 else if (!*s) | |
2860 { | |
2861 return 0; | |
2862 } | |
2863 else ++s; | |
2864 } | |
2865 } | |
2866 | |
2867 static char_t* parse_simple(char_t* s, char_t end_quote) | |
2868 { | |
2869 gap g; | |
2870 | |
2871 while (true) | |
2872 { | |
2873 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPE(ss, ct_parse_attr)); | |
2874 | |
2875 if (*s == end_quote) | |
2876 { | |
2877 *g.flush(s) = 0; | |
2878 | |
2879 return s + 1; | |
2880 } | |
2881 else if (opt_escape::value && *s == '&') | |
2882 { | |
2883 s = strconv_escape(s, g); | |
2884 } | |
2885 else if (!*s) | |
2886 { | |
2887 return 0; | |
2888 } | |
2889 else ++s; | |
2890 } | |
2891 } | |
2892 }; | |
2893 | |
2894 PUGI__FN strconv_attribute_t get_strconv_attribute(unsigned int optmask) | |
2895 { | |
2896 PUGI__STATIC_ASSERT(parse_escapes == 0x10 && parse_eol == 0x20 && parse_wconv_attribute == 0x40 && parse_wnorm_attribute == 0x80); | |
2897 | |
2898 switch ((optmask >> 4) & 15) // get bitmask for flags (wnorm wconv eol escapes); this simultaneously checks 4 options from assertion above | |
2899 { | |
2900 case 0: return strconv_attribute_impl<opt_false>::parse_simple; | |
2901 case 1: return strconv_attribute_impl<opt_true>::parse_simple; | |
2902 case 2: return strconv_attribute_impl<opt_false>::parse_eol; | |
2903 case 3: return strconv_attribute_impl<opt_true>::parse_eol; | |
2904 case 4: return strconv_attribute_impl<opt_false>::parse_wconv; | |
2905 case 5: return strconv_attribute_impl<opt_true>::parse_wconv; | |
2906 case 6: return strconv_attribute_impl<opt_false>::parse_wconv; | |
2907 case 7: return strconv_attribute_impl<opt_true>::parse_wconv; | |
2908 case 8: return strconv_attribute_impl<opt_false>::parse_wnorm; | |
2909 case 9: return strconv_attribute_impl<opt_true>::parse_wnorm; | |
2910 case 10: return strconv_attribute_impl<opt_false>::parse_wnorm; | |
2911 case 11: return strconv_attribute_impl<opt_true>::parse_wnorm; | |
2912 case 12: return strconv_attribute_impl<opt_false>::parse_wnorm; | |
2913 case 13: return strconv_attribute_impl<opt_true>::parse_wnorm; | |
2914 case 14: return strconv_attribute_impl<opt_false>::parse_wnorm; | |
2915 case 15: return strconv_attribute_impl<opt_true>::parse_wnorm; | |
2916 default: assert(false); return 0; // unreachable | |
2917 } | |
2918 } | |
2919 | |
2920 inline xml_parse_result make_parse_result(xml_parse_status status, ptrdiff_t offset = 0) | |
2921 { | |
2922 xml_parse_result result; | |
2923 result.status = status; | |
2924 result.offset = offset; | |
2925 | |
2926 return result; | |
2927 } | |
2928 | |
2929 struct xml_parser | |
2930 { | |
2931 xml_allocator* alloc; | |
2932 char_t* error_offset; | |
2933 xml_parse_status error_status; | |
2934 | |
2935 xml_parser(xml_allocator* alloc_): alloc(alloc_), error_offset(0), error_status(status_ok) | |
2936 { | |
2937 } | |
2938 | |
2939 // DOCTYPE consists of nested sections of the following possible types: | |
2940 // <!-- ... -->, <? ... ?>, "...", '...' | |
2941 // <![...]]> | |
2942 // <!...> | |
2943 // First group can not contain nested groups | |
2944 // Second group can contain nested groups of the same type | |
2945 // Third group can contain all other groups | |
2946 char_t* parse_doctype_primitive(char_t* s) | |
2947 { | |
2948 if (*s == '"' || *s == '\'') | |
2949 { | |
2950 // quoted string | |
2951 char_t ch = *s++; | |
2952 PUGI__SCANFOR(*s == ch); | |
2953 if (!*s) PUGI__THROW_ERROR(status_bad_doctype, s); | |
2954 | |
2955 s++; | |
2956 } | |
2957 else if (s[0] == '<' && s[1] == '?') | |
2958 { | |
2959 // <? ... ?> | |
2960 s += 2; | |
2961 PUGI__SCANFOR(s[0] == '?' && s[1] == '>'); // no need for ENDSWITH because ?> can't terminate proper doctype | |
2962 if (!*s) PUGI__THROW_ERROR(status_bad_doctype, s); | |
2963 | |
2964 s += 2; | |
2965 } | |
2966 else if (s[0] == '<' && s[1] == '!' && s[2] == '-' && s[3] == '-') | |
2967 { | |
2968 s += 4; | |
2969 PUGI__SCANFOR(s[0] == '-' && s[1] == '-' && s[2] == '>'); // no need for ENDSWITH because --> can't terminate proper doctype | |
2970 if (!*s) PUGI__THROW_ERROR(status_bad_doctype, s); | |
2971 | |
2972 s += 3; | |
2973 } | |
2974 else PUGI__THROW_ERROR(status_bad_doctype, s); | |
2975 | |
2976 return s; | |
2977 } | |
2978 | |
2979 char_t* parse_doctype_ignore(char_t* s) | |
2980 { | |
2981 size_t depth = 0; | |
2982 | |
2983 assert(s[0] == '<' && s[1] == '!' && s[2] == '['); | |
2984 s += 3; | |
2985 | |
2986 while (*s) | |
2987 { | |
2988 if (s[0] == '<' && s[1] == '!' && s[2] == '[') | |
2989 { | |
2990 // nested ignore section | |
2991 s += 3; | |
2992 depth++; | |
2993 } | |
2994 else if (s[0] == ']' && s[1] == ']' && s[2] == '>') | |
2995 { | |
2996 // ignore section end | |
2997 s += 3; | |
2998 | |
2999 if (depth == 0) | |
3000 return s; | |
3001 | |
3002 depth--; | |
3003 } | |
3004 else s++; | |
3005 } | |
3006 | |
3007 PUGI__THROW_ERROR(status_bad_doctype, s); | |
3008 } | |
3009 | |
3010 char_t* parse_doctype_group(char_t* s, char_t endch) | |
3011 { | |
3012 size_t depth = 0; | |
3013 | |
3014 assert((s[0] == '<' || s[0] == 0) && s[1] == '!'); | |
3015 s += 2; | |
3016 | |
3017 while (*s) | |
3018 { | |
3019 if (s[0] == '<' && s[1] == '!' && s[2] != '-') | |
3020 { | |
3021 if (s[2] == '[') | |
3022 { | |
3023 // ignore | |
3024 s = parse_doctype_ignore(s); | |
3025 if (!s) return s; | |
3026 } | |
3027 else | |
3028 { | |
3029 // some control group | |
3030 s += 2; | |
3031 depth++; | |
3032 } | |
3033 } | |
3034 else if (s[0] == '<' || s[0] == '"' || s[0] == '\'') | |
3035 { | |
3036 // unknown tag (forbidden), or some primitive group | |
3037 s = parse_doctype_primitive(s); | |
3038 if (!s) return s; | |
3039 } | |
3040 else if (*s == '>') | |
3041 { | |
3042 if (depth == 0) | |
3043 return s; | |
3044 | |
3045 depth--; | |
3046 s++; | |
3047 } | |
3048 else s++; | |
3049 } | |
3050 | |
3051 if (depth != 0 || endch != '>') PUGI__THROW_ERROR(status_bad_doctype, s); | |
3052 | |
3053 return s; | |
3054 } | |
3055 | |
3056 char_t* parse_exclamation(char_t* s, xml_node_struct* cursor, unsigned int optmsk, char_t endch) | |
3057 { | |
3058 // parse node contents, starting with exclamation mark | |
3059 ++s; | |
3060 | |
3061 if (*s == '-') // '<!-...' | |
3062 { | |
3063 ++s; | |
3064 | |
3065 if (*s == '-') // '<!--...' | |
3066 { | |
3067 ++s; | |
3068 | |
3069 if (PUGI__OPTSET(parse_comments)) | |
3070 { | |
3071 PUGI__PUSHNODE(node_comment); // Append a new node on the tree. | |
3072 cursor->value = s; // Save the offset. | |
3073 } | |
3074 | |
3075 if (PUGI__OPTSET(parse_eol) && PUGI__OPTSET(parse_comments)) | |
3076 { | |
3077 s = strconv_comment(s, endch); | |
3078 | |
3079 if (!s) PUGI__THROW_ERROR(status_bad_comment, cursor->value); | |
3080 } | |
3081 else | |
3082 { | |
3083 // Scan for terminating '-->'. | |
3084 PUGI__SCANFOR(s[0] == '-' && s[1] == '-' && PUGI__ENDSWITH(s[2], '>')); | |
3085 PUGI__CHECK_ERROR(status_bad_comment, s); | |
3086 | |
3087 if (PUGI__OPTSET(parse_comments)) | |
3088 *s = 0; // Zero-terminate this segment at the first terminating '-'. | |
3089 | |
3090 s += (s[2] == '>' ? 3 : 2); // Step over the '\0->'. | |
3091 } | |
3092 } | |
3093 else PUGI__THROW_ERROR(status_bad_comment, s); | |
3094 } | |
3095 else if (*s == '[') | |
3096 { | |
3097 // '<![CDATA[...' | |
3098 if (*++s=='C' && *++s=='D' && *++s=='A' && *++s=='T' && *++s=='A' && *++s == '[') | |
3099 { | |
3100 ++s; | |
3101 | |
3102 if (PUGI__OPTSET(parse_cdata)) | |
3103 { | |
3104 PUGI__PUSHNODE(node_cdata); // Append a new node on the tree. | |
3105 cursor->value = s; // Save the offset. | |
3106 | |
3107 if (PUGI__OPTSET(parse_eol)) | |
3108 { | |
3109 s = strconv_cdata(s, endch); | |
3110 | |
3111 if (!s) PUGI__THROW_ERROR(status_bad_cdata, cursor->value); | |
3112 } | |
3113 else | |
3114 { | |
3115 // Scan for terminating ']]>'. | |
3116 PUGI__SCANFOR(s[0] == ']' && s[1] == ']' && PUGI__ENDSWITH(s[2], '>')); | |
3117 PUGI__CHECK_ERROR(status_bad_cdata, s); | |
3118 | |
3119 *s++ = 0; // Zero-terminate this segment. | |
3120 } | |
3121 } | |
3122 else // Flagged for discard, but we still have to scan for the terminator. | |
3123 { | |
3124 // Scan for terminating ']]>'. | |
3125 PUGI__SCANFOR(s[0] == ']' && s[1] == ']' && PUGI__ENDSWITH(s[2], '>')); | |
3126 PUGI__CHECK_ERROR(status_bad_cdata, s); | |
3127 | |
3128 ++s; | |
3129 } | |
3130 | |
3131 s += (s[1] == '>' ? 2 : 1); // Step over the last ']>'. | |
3132 } | |
3133 else PUGI__THROW_ERROR(status_bad_cdata, s); | |
3134 } | |
3135 else if (s[0] == 'D' && s[1] == 'O' && s[2] == 'C' && s[3] == 'T' && s[4] == 'Y' && s[5] == 'P' && PUGI__ENDSWITH(s[6], 'E')) | |
3136 { | |
3137 s -= 2; | |
3138 | |
3139 if (cursor->parent) PUGI__THROW_ERROR(status_bad_doctype, s); | |
3140 | |
3141 char_t* mark = s + 9; | |
3142 | |
3143 s = parse_doctype_group(s, endch); | |
3144 if (!s) return s; | |
3145 | |
3146 assert((*s == 0 && endch == '>') || *s == '>'); | |
3147 if (*s) *s++ = 0; | |
3148 | |
3149 if (PUGI__OPTSET(parse_doctype)) | |
3150 { | |
3151 while (PUGI__IS_CHARTYPE(*mark, ct_space)) ++mark; | |
3152 | |
3153 PUGI__PUSHNODE(node_doctype); | |
3154 | |
3155 cursor->value = mark; | |
3156 } | |
3157 } | |
3158 else if (*s == 0 && endch == '-') PUGI__THROW_ERROR(status_bad_comment, s); | |
3159 else if (*s == 0 && endch == '[') PUGI__THROW_ERROR(status_bad_cdata, s); | |
3160 else PUGI__THROW_ERROR(status_unrecognized_tag, s); | |
3161 | |
3162 return s; | |
3163 } | |
3164 | |
3165 char_t* parse_question(char_t* s, xml_node_struct*& ref_cursor, unsigned int optmsk, char_t endch) | |
3166 { | |
3167 // load into registers | |
3168 xml_node_struct* cursor = ref_cursor; | |
3169 char_t ch = 0; | |
3170 | |
3171 // parse node contents, starting with question mark | |
3172 ++s; | |
3173 | |
3174 // read PI target | |
3175 char_t* target = s; | |
3176 | |
3177 if (!PUGI__IS_CHARTYPE(*s, ct_start_symbol)) PUGI__THROW_ERROR(status_bad_pi, s); | |
3178 | |
3179 PUGI__SCANWHILE(PUGI__IS_CHARTYPE(*s, ct_symbol)); | |
3180 PUGI__CHECK_ERROR(status_bad_pi, s); | |
3181 | |
3182 // determine node type; stricmp / strcasecmp is not portable | |
3183 bool declaration = (target[0] | ' ') == 'x' && (target[1] | ' ') == 'm' && (target[2] | ' ') == 'l' && target + 3 == s; | |
3184 | |
3185 if (declaration ? PUGI__OPTSET(parse_declaration) : PUGI__OPTSET(parse_pi)) | |
3186 { | |
3187 if (declaration) | |
3188 { | |
3189 // disallow non top-level declarations | |
3190 if (cursor->parent) PUGI__THROW_ERROR(status_bad_pi, s); | |
3191 | |
3192 PUGI__PUSHNODE(node_declaration); | |
3193 } | |
3194 else | |
3195 { | |
3196 PUGI__PUSHNODE(node_pi); | |
3197 } | |
3198 | |
3199 cursor->name = target; | |
3200 | |
3201 PUGI__ENDSEG(); | |
3202 | |
3203 // parse value/attributes | |
3204 if (ch == '?') | |
3205 { | |
3206 // empty node | |
3207 if (!PUGI__ENDSWITH(*s, '>')) PUGI__THROW_ERROR(status_bad_pi, s); | |
3208 s += (*s == '>'); | |
3209 | |
3210 PUGI__POPNODE(); | |
3211 } | |
3212 else if (PUGI__IS_CHARTYPE(ch, ct_space)) | |
3213 { | |
3214 PUGI__SKIPWS(); | |
3215 | |
3216 // scan for tag end | |
3217 char_t* value = s; | |
3218 | |
3219 PUGI__SCANFOR(s[0] == '?' && PUGI__ENDSWITH(s[1], '>')); | |
3220 PUGI__CHECK_ERROR(status_bad_pi, s); | |
3221 | |
3222 if (declaration) | |
3223 { | |
3224 // replace ending ? with / so that 'element' terminates properly | |
3225 *s = '/'; | |
3226 | |
3227 // we exit from this function with cursor at node_declaration, which is a signal to parse() to go to LOC_ATTRIBUTES | |
3228 s = value; | |
3229 } | |
3230 else | |
3231 { | |
3232 // store value and step over > | |
3233 cursor->value = value; | |
3234 | |
3235 PUGI__POPNODE(); | |
3236 | |
3237 PUGI__ENDSEG(); | |
3238 | |
3239 s += (*s == '>'); | |
3240 } | |
3241 } | |
3242 else PUGI__THROW_ERROR(status_bad_pi, s); | |
3243 } | |
3244 else | |
3245 { | |
3246 // scan for tag end | |
3247 PUGI__SCANFOR(s[0] == '?' && PUGI__ENDSWITH(s[1], '>')); | |
3248 PUGI__CHECK_ERROR(status_bad_pi, s); | |
3249 | |
3250 s += (s[1] == '>' ? 2 : 1); | |
3251 } | |
3252 | |
3253 // store from registers | |
3254 ref_cursor = cursor; | |
3255 | |
3256 return s; | |
3257 } | |
3258 | |
3259 char_t* parse_tree(char_t* s, xml_node_struct* root, unsigned int optmsk, char_t endch) | |
3260 { | |
3261 strconv_attribute_t strconv_attribute = get_strconv_attribute(optmsk); | |
3262 strconv_pcdata_t strconv_pcdata = get_strconv_pcdata(optmsk); | |
3263 | |
3264 char_t ch = 0; | |
3265 xml_node_struct* cursor = root; | |
3266 char_t* mark = s; | |
3267 | |
3268 while (*s != 0) | |
3269 { | |
3270 if (*s == '<') | |
3271 { | |
3272 ++s; | |
3273 | |
3274 LOC_TAG: | |
3275 if (PUGI__IS_CHARTYPE(*s, ct_start_symbol)) // '<#...' | |
3276 { | |
3277 PUGI__PUSHNODE(node_element); // Append a new node to the tree. | |
3278 | |
3279 cursor->name = s; | |
3280 | |
3281 PUGI__SCANWHILE_UNROLL(PUGI__IS_CHARTYPE(ss, ct_symbol)); // Scan for a terminator. | |
3282 PUGI__ENDSEG(); // Save char in 'ch', terminate & step over. | |
3283 | |
3284 if (ch == '>') | |
3285 { | |
3286 // end of tag | |
3287 } | |
3288 else if (PUGI__IS_CHARTYPE(ch, ct_space)) | |
3289 { | |
3290 LOC_ATTRIBUTES: | |
3291 while (true) | |
3292 { | |
3293 PUGI__SKIPWS(); // Eat any whitespace. | |
3294 | |
3295 if (PUGI__IS_CHARTYPE(*s, ct_start_symbol)) // <... #... | |
3296 { | |
3297 xml_attribute_struct* a = append_new_attribute(cursor, *alloc); // Make space for this attribute. | |
3298 if (!a) PUGI__THROW_ERROR(status_out_of_memory, s); | |
3299 | |
3300 a->name = s; // Save the offset. | |
3301 | |
3302 PUGI__SCANWHILE_UNROLL(PUGI__IS_CHARTYPE(ss, ct_symbol)); // Scan for a terminator. | |
3303 PUGI__ENDSEG(); // Save char in 'ch', terminate & step over. | |
3304 | |
3305 if (PUGI__IS_CHARTYPE(ch, ct_space)) | |
3306 { | |
3307 PUGI__SKIPWS(); // Eat any whitespace. | |
3308 | |
3309 ch = *s; | |
3310 ++s; | |
3311 } | |
3312 | |
3313 if (ch == '=') // '<... #=...' | |
3314 { | |
3315 PUGI__SKIPWS(); // Eat any whitespace. | |
3316 | |
3317 if (*s == '"' || *s == '\'') // '<... #="...' | |
3318 { | |
3319 ch = *s; // Save quote char to avoid breaking on "''" -or- '""'. | |
3320 ++s; // Step over the quote. | |
3321 a->value = s; // Save the offset. | |
3322 | |
3323 s = strconv_attribute(s, ch); | |
3324 | |
3325 if (!s) PUGI__THROW_ERROR(status_bad_attribute, a->value); | |
3326 | |
3327 // After this line the loop continues from the start; | |
3328 // Whitespaces, / and > are ok, symbols and EOF are wrong, | |
3329 // everything else will be detected | |
3330 if (PUGI__IS_CHARTYPE(*s, ct_start_symbol)) PUGI__THROW_ERROR(status_bad_attribute, s); | |
3331 } | |
3332 else PUGI__THROW_ERROR(status_bad_attribute, s); | |
3333 } | |
3334 else PUGI__THROW_ERROR(status_bad_attribute, s); | |
3335 } | |
3336 else if (*s == '/') | |
3337 { | |
3338 ++s; | |
3339 | |
3340 if (*s == '>') | |
3341 { | |
3342 PUGI__POPNODE(); | |
3343 s++; | |
3344 break; | |
3345 } | |
3346 else if (*s == 0 && endch == '>') | |
3347 { | |
3348 PUGI__POPNODE(); | |
3349 break; | |
3350 } | |
3351 else PUGI__THROW_ERROR(status_bad_start_element, s); | |
3352 } | |
3353 else if (*s == '>') | |
3354 { | |
3355 ++s; | |
3356 | |
3357 break; | |
3358 } | |
3359 else if (*s == 0 && endch == '>') | |
3360 { | |
3361 break; | |
3362 } | |
3363 else PUGI__THROW_ERROR(status_bad_start_element, s); | |
3364 } | |
3365 | |
3366 // !!! | |
3367 } | |
3368 else if (ch == '/') // '<#.../' | |
3369 { | |
3370 if (!PUGI__ENDSWITH(*s, '>')) PUGI__THROW_ERROR(status_bad_start_element, s); | |
3371 | |
3372 PUGI__POPNODE(); // Pop. | |
3373 | |
3374 s += (*s == '>'); | |
3375 } | |
3376 else if (ch == 0) | |
3377 { | |
3378 // we stepped over null terminator, backtrack & handle closing tag | |
3379 --s; | |
3380 | |
3381 if (endch != '>') PUGI__THROW_ERROR(status_bad_start_element, s); | |
3382 } | |
3383 else PUGI__THROW_ERROR(status_bad_start_element, s); | |
3384 } | |
3385 else if (*s == '/') | |
3386 { | |
3387 ++s; | |
3388 | |
3389 mark = s; | |
3390 | |
3391 char_t* name = cursor->name; | |
3392 if (!name) PUGI__THROW_ERROR(status_end_element_mismatch, mark); | |
3393 | |
3394 while (PUGI__IS_CHARTYPE(*s, ct_symbol)) | |
3395 { | |
3396 if (*s++ != *name++) PUGI__THROW_ERROR(status_end_element_mismatch, mark); | |
3397 } | |
3398 | |
3399 if (*name) | |
3400 { | |
3401 if (*s == 0 && name[0] == endch && name[1] == 0) PUGI__THROW_ERROR(status_bad_end_element, s); | |
3402 else PUGI__THROW_ERROR(status_end_element_mismatch, mark); | |
3403 } | |
3404 | |
3405 PUGI__POPNODE(); // Pop. | |
3406 | |
3407 PUGI__SKIPWS(); | |
3408 | |
3409 if (*s == 0) | |
3410 { | |
3411 if (endch != '>') PUGI__THROW_ERROR(status_bad_end_element, s); | |
3412 } | |
3413 else | |
3414 { | |
3415 if (*s != '>') PUGI__THROW_ERROR(status_bad_end_element, s); | |
3416 ++s; | |
3417 } | |
3418 } | |
3419 else if (*s == '?') // '<?...' | |
3420 { | |
3421 s = parse_question(s, cursor, optmsk, endch); | |
3422 if (!s) return s; | |
3423 | |
3424 assert(cursor); | |
3425 if (PUGI__NODETYPE(cursor) == node_declaration) goto LOC_ATTRIBUTES; | |
3426 } | |
3427 else if (*s == '!') // '<!...' | |
3428 { | |
3429 s = parse_exclamation(s, cursor, optmsk, endch); | |
3430 if (!s) return s; | |
3431 } | |
3432 else if (*s == 0 && endch == '?') PUGI__THROW_ERROR(status_bad_pi, s); | |
3433 else PUGI__THROW_ERROR(status_unrecognized_tag, s); | |
3434 } | |
3435 else | |
3436 { | |
3437 mark = s; // Save this offset while searching for a terminator. | |
3438 | |
3439 PUGI__SKIPWS(); // Eat whitespace if no genuine PCDATA here. | |
3440 | |
3441 if (*s == '<' || !*s) | |
3442 { | |
3443 // We skipped some whitespace characters because otherwise we would take the tag branch instead of PCDATA one | |
3444 assert(mark != s); | |
3445 | |
3446 if (!PUGI__OPTSET(parse_ws_pcdata | parse_ws_pcdata_single) || PUGI__OPTSET(parse_trim_pcdata)) | |
3447 { | |
3448 continue; | |
3449 } | |
3450 else if (PUGI__OPTSET(parse_ws_pcdata_single)) | |
3451 { | |
3452 if (s[0] != '<' || s[1] != '/' || cursor->first_child) continue; | |
3453 } | |
3454 } | |
3455 | |
3456 if (!PUGI__OPTSET(parse_trim_pcdata)) | |
3457 s = mark; | |
3458 | |
3459 if (cursor->parent || PUGI__OPTSET(parse_fragment)) | |
3460 { | |
3461 if (PUGI__OPTSET(parse_embed_pcdata) && cursor->parent && !cursor->first_child && !cursor->value) | |
3462 { | |
3463 cursor->value = s; // Save the offset. | |
3464 } | |
3465 else | |
3466 { | |
3467 PUGI__PUSHNODE(node_pcdata); // Append a new node on the tree. | |
3468 | |
3469 cursor->value = s; // Save the offset. | |
3470 | |
3471 PUGI__POPNODE(); // Pop since this is a standalone. | |
3472 } | |
3473 | |
3474 s = strconv_pcdata(s); | |
3475 | |
3476 if (!*s) break; | |
3477 } | |
3478 else | |
3479 { | |
3480 PUGI__SCANFOR(*s == '<'); // '...<' | |
3481 if (!*s) break; | |
3482 | |
3483 ++s; | |
3484 } | |
3485 | |
3486 // We're after '<' | |
3487 goto LOC_TAG; | |
3488 } | |
3489 } | |
3490 | |
3491 // check that last tag is closed | |
3492 if (cursor != root) PUGI__THROW_ERROR(status_end_element_mismatch, s); | |
3493 | |
3494 return s; | |
3495 } | |
3496 | |
3497 #ifdef PUGIXML_WCHAR_MODE | |
3498 static char_t* parse_skip_bom(char_t* s) | |
3499 { | |
3500 unsigned int bom = 0xfeff; | |
3501 return (s[0] == static_cast<wchar_t>(bom)) ? s + 1 : s; | |
3502 } | |
3503 #else | |
3504 static char_t* parse_skip_bom(char_t* s) | |
3505 { | |
3506 return (s[0] == '\xef' && s[1] == '\xbb' && s[2] == '\xbf') ? s + 3 : s; | |
3507 } | |
3508 #endif | |
3509 | |
3510 static bool has_element_node_siblings(xml_node_struct* node) | |
3511 { | |
3512 while (node) | |
3513 { | |
3514 if (PUGI__NODETYPE(node) == node_element) return true; | |
3515 | |
3516 node = node->next_sibling; | |
3517 } | |
3518 | |
3519 return false; | |
3520 } | |
3521 | |
3522 static xml_parse_result parse(char_t* buffer, size_t length, xml_document_struct* xmldoc, xml_node_struct* root, unsigned int optmsk) | |
3523 { | |
3524 // early-out for empty documents | |
3525 if (length == 0) | |
3526 return make_parse_result(PUGI__OPTSET(parse_fragment) ? status_ok : status_no_document_element); | |
3527 | |
3528 // get last child of the root before parsing | |
3529 xml_node_struct* last_root_child = root->first_child ? root->first_child->prev_sibling_c + 0 : 0; | |
3530 | |
3531 // create parser on stack | |
3532 xml_parser parser(static_cast<xml_allocator*>(xmldoc)); | |
3533 | |
3534 // save last character and make buffer zero-terminated (speeds up parsing) | |
3535 char_t endch = buffer[length - 1]; | |
3536 buffer[length - 1] = 0; | |
3537 | |
3538 // skip BOM to make sure it does not end up as part of parse output | |
3539 char_t* buffer_data = parse_skip_bom(buffer); | |
3540 | |
3541 // perform actual parsing | |
3542 parser.parse_tree(buffer_data, root, optmsk, endch); | |
3543 | |
3544 xml_parse_result result = make_parse_result(parser.error_status, parser.error_offset ? parser.error_offset - buffer : 0); | |
3545 assert(result.offset >= 0 && static_cast<size_t>(result.offset) <= length); | |
3546 | |
3547 if (result) | |
3548 { | |
3549 // since we removed last character, we have to handle the only possible false positive (stray <) | |
3550 if (endch == '<') | |
3551 return make_parse_result(status_unrecognized_tag, length - 1); | |
3552 | |
3553 // check if there are any element nodes parsed | |
3554 xml_node_struct* first_root_child_parsed = last_root_child ? last_root_child->next_sibling + 0 : root->first_child+ 0; | |
3555 | |
3556 if (!PUGI__OPTSET(parse_fragment) && !has_element_node_siblings(first_root_child_parsed)) | |
3557 return make_parse_result(status_no_document_element, length - 1); | |
3558 } | |
3559 else | |
3560 { | |
3561 // roll back offset if it occurs on a null terminator in the source buffer | |
3562 if (result.offset > 0 && static_cast<size_t>(result.offset) == length - 1 && endch == 0) | |
3563 result.offset--; | |
3564 } | |
3565 | |
3566 return result; | |
3567 } | |
3568 }; | |
3569 | |
3570 // Output facilities | |
3571 PUGI__FN xml_encoding get_write_native_encoding() | |
3572 { | |
3573 #ifdef PUGIXML_WCHAR_MODE | |
3574 return get_wchar_encoding(); | |
3575 #else | |
3576 return encoding_utf8; | |
3577 #endif | |
3578 } | |
3579 | |
3580 PUGI__FN xml_encoding get_write_encoding(xml_encoding encoding) | |
3581 { | |
3582 // replace wchar encoding with utf implementation | |
3583 if (encoding == encoding_wchar) return get_wchar_encoding(); | |
3584 | |
3585 // replace utf16 encoding with utf16 with specific endianness | |
3586 if (encoding == encoding_utf16) return is_little_endian() ? encoding_utf16_le : encoding_utf16_be; | |
3587 | |
3588 // replace utf32 encoding with utf32 with specific endianness | |
3589 if (encoding == encoding_utf32) return is_little_endian() ? encoding_utf32_le : encoding_utf32_be; | |
3590 | |
3591 // only do autodetection if no explicit encoding is requested | |
3592 if (encoding != encoding_auto) return encoding; | |
3593 | |
3594 // assume utf8 encoding | |
3595 return encoding_utf8; | |
3596 } | |
3597 | |
3598 template <typename D, typename T> PUGI__FN size_t convert_buffer_output_generic(typename T::value_type dest, const char_t* data, size_t length, D, T) | |
3599 { | |
3600 PUGI__STATIC_ASSERT(sizeof(char_t) == sizeof(typename D::type)); | |
3601 | |
3602 typename T::value_type end = D::process(reinterpret_cast<const typename D::type*>(data), length, dest, T()); | |
3603 | |
3604 return static_cast<size_t>(end - dest) * sizeof(*dest); | |
3605 } | |
3606 | |
3607 template <typename D, typename T> PUGI__FN size_t convert_buffer_output_generic(typename T::value_type dest, const char_t* data, size_t length, D, T, bool opt_swap) | |
3608 { | |
3609 PUGI__STATIC_ASSERT(sizeof(char_t) == sizeof(typename D::type)); | |
3610 | |
3611 typename T::value_type end = D::process(reinterpret_cast<const typename D::type*>(data), length, dest, T()); | |
3612 | |
3613 if (opt_swap) | |
3614 { | |
3615 for (typename T::value_type i = dest; i != end; ++i) | |
3616 *i = endian_swap(*i); | |
3617 } | |
3618 | |
3619 return static_cast<size_t>(end - dest) * sizeof(*dest); | |
3620 } | |
3621 | |
3622 #ifdef PUGIXML_WCHAR_MODE | |
3623 PUGI__FN size_t get_valid_length(const char_t* data, size_t length) | |
3624 { | |
3625 if (length < 1) return 0; | |
3626 | |
3627 // discard last character if it's the lead of a surrogate pair | |
3628 return (sizeof(wchar_t) == 2 && static_cast<unsigned int>(static_cast<uint16_t>(data[length - 1]) - 0xD800) < 0x400) ? length - 1 : length; | |
3629 } | |
3630 | |
3631 PUGI__FN size_t convert_buffer_output(char_t* r_char, uint8_t* r_u8, uint16_t* r_u16, uint32_t* r_u32, const char_t* data, size_t length, xml_encoding encoding) | |
3632 { | |
3633 // only endian-swapping is required | |
3634 if (need_endian_swap_utf(encoding, get_wchar_encoding())) | |
3635 { | |
3636 convert_wchar_endian_swap(r_char, data, length); | |
3637 | |
3638 return length * sizeof(char_t); | |
3639 } | |
3640 | |
3641 // convert to utf8 | |
3642 if (encoding == encoding_utf8) | |
3643 return convert_buffer_output_generic(r_u8, data, length, wchar_decoder(), utf8_writer()); | |
3644 | |
3645 // convert to utf16 | |
3646 if (encoding == encoding_utf16_be || encoding == encoding_utf16_le) | |
3647 { | |
3648 xml_encoding native_encoding = is_little_endian() ? encoding_utf16_le : encoding_utf16_be; | |
3649 | |
3650 return convert_buffer_output_generic(r_u16, data, length, wchar_decoder(), utf16_writer(), native_encoding != encoding); | |
3651 } | |
3652 | |
3653 // convert to utf32 | |
3654 if (encoding == encoding_utf32_be || encoding == encoding_utf32_le) | |
3655 { | |
3656 xml_encoding native_encoding = is_little_endian() ? encoding_utf32_le : encoding_utf32_be; | |
3657 | |
3658 return convert_buffer_output_generic(r_u32, data, length, wchar_decoder(), utf32_writer(), native_encoding != encoding); | |
3659 } | |
3660 | |
3661 // convert to latin1 | |
3662 if (encoding == encoding_latin1) | |
3663 return convert_buffer_output_generic(r_u8, data, length, wchar_decoder(), latin1_writer()); | |
3664 | |
3665 assert(false && "Invalid encoding"); // unreachable | |
3666 return 0; | |
3667 } | |
3668 #else | |
3669 PUGI__FN size_t get_valid_length(const char_t* data, size_t length) | |
3670 { | |
3671 if (length < 5) return 0; | |
3672 | |
3673 for (size_t i = 1; i <= 4; ++i) | |
3674 { | |
3675 uint8_t ch = static_cast<uint8_t>(data[length - i]); | |
3676 | |
3677 // either a standalone character or a leading one | |
3678 if ((ch & 0xc0) != 0x80) return length - i; | |
3679 } | |
3680 | |
3681 // there are four non-leading characters at the end, sequence tail is broken so might as well process the whole chunk | |
3682 return length; | |
3683 } | |
3684 | |
3685 PUGI__FN size_t convert_buffer_output(char_t* /* r_char */, uint8_t* r_u8, uint16_t* r_u16, uint32_t* r_u32, const char_t* data, size_t length, xml_encoding encoding) | |
3686 { | |
3687 if (encoding == encoding_utf16_be || encoding == encoding_utf16_le) | |
3688 { | |
3689 xml_encoding native_encoding = is_little_endian() ? encoding_utf16_le : encoding_utf16_be; | |
3690 | |
3691 return convert_buffer_output_generic(r_u16, data, length, utf8_decoder(), utf16_writer(), native_encoding != encoding); | |
3692 } | |
3693 | |
3694 if (encoding == encoding_utf32_be || encoding == encoding_utf32_le) | |
3695 { | |
3696 xml_encoding native_encoding = is_little_endian() ? encoding_utf32_le : encoding_utf32_be; | |
3697 | |
3698 return convert_buffer_output_generic(r_u32, data, length, utf8_decoder(), utf32_writer(), native_encoding != encoding); | |
3699 } | |
3700 | |
3701 if (encoding == encoding_latin1) | |
3702 return convert_buffer_output_generic(r_u8, data, length, utf8_decoder(), latin1_writer()); | |
3703 | |
3704 assert(false && "Invalid encoding"); // unreachable | |
3705 return 0; | |
3706 } | |
3707 #endif | |
3708 | |
3709 class xml_buffered_writer | |
3710 { | |
3711 xml_buffered_writer(const xml_buffered_writer&); | |
3712 xml_buffered_writer& operator=(const xml_buffered_writer&); | |
3713 | |
3714 public: | |
3715 xml_buffered_writer(xml_writer& writer_, xml_encoding user_encoding): writer(writer_), bufsize(0), encoding(get_write_encoding(user_encoding)) | |
3716 { | |
3717 PUGI__STATIC_ASSERT(bufcapacity >= 8); | |
3718 } | |
3719 | |
3720 size_t flush() | |
3721 { | |
3722 flush(buffer, bufsize); | |
3723 bufsize = 0; | |
3724 return 0; | |
3725 } | |
3726 | |
3727 void flush(const char_t* data, size_t size) | |
3728 { | |
3729 if (size == 0) return; | |
3730 | |
3731 // fast path, just write data | |
3732 if (encoding == get_write_native_encoding()) | |
3733 writer.write(data, size * sizeof(char_t)); | |
3734 else | |
3735 { | |
3736 // convert chunk | |
3737 size_t result = convert_buffer_output(scratch.data_char, scratch.data_u8, scratch.data_u16, scratch.data_u32, data, size, encoding); | |
3738 assert(result <= sizeof(scratch)); | |
3739 | |
3740 // write data | |
3741 writer.write(scratch.data_u8, result); | |
3742 } | |
3743 } | |
3744 | |
3745 void write_direct(const char_t* data, size_t length) | |
3746 { | |
3747 // flush the remaining buffer contents | |
3748 flush(); | |
3749 | |
3750 // handle large chunks | |
3751 if (length > bufcapacity) | |
3752 { | |
3753 if (encoding == get_write_native_encoding()) | |
3754 { | |
3755 // fast path, can just write data chunk | |
3756 writer.write(data, length * sizeof(char_t)); | |
3757 return; | |
3758 } | |
3759 | |
3760 // need to convert in suitable chunks | |
3761 while (length > bufcapacity) | |
3762 { | |
3763 // get chunk size by selecting such number of characters that are guaranteed to fit into scratch buffer | |
3764 // and form a complete codepoint sequence (i.e. discard start of last codepoint if necessary) | |
3765 size_t chunk_size = get_valid_length(data, bufcapacity); | |
3766 assert(chunk_size); | |
3767 | |
3768 // convert chunk and write | |
3769 flush(data, chunk_size); | |
3770 | |
3771 // iterate | |
3772 data += chunk_size; | |
3773 length -= chunk_size; | |
3774 } | |
3775 | |
3776 // small tail is copied below | |
3777 bufsize = 0; | |
3778 } | |
3779 | |
3780 memcpy(buffer + bufsize, data, length * sizeof(char_t)); | |
3781 bufsize += length; | |
3782 } | |
3783 | |
3784 void write_buffer(const char_t* data, size_t length) | |
3785 { | |
3786 size_t offset = bufsize; | |
3787 | |
3788 if (offset + length <= bufcapacity) | |
3789 { | |
3790 memcpy(buffer + offset, data, length * sizeof(char_t)); | |
3791 bufsize = offset + length; | |
3792 } | |
3793 else | |
3794 { | |
3795 write_direct(data, length); | |
3796 } | |
3797 } | |
3798 | |
3799 void write_string(const char_t* data) | |
3800 { | |
3801 // write the part of the string that fits in the buffer | |
3802 size_t offset = bufsize; | |
3803 | |
3804 while (*data && offset < bufcapacity) | |
3805 buffer[offset++] = *data++; | |
3806 | |
3807 // write the rest | |
3808 if (offset < bufcapacity) | |
3809 { | |
3810 bufsize = offset; | |
3811 } | |
3812 else | |
3813 { | |
3814 // backtrack a bit if we have split the codepoint | |
3815 size_t length = offset - bufsize; | |
3816 size_t extra = length - get_valid_length(data - length, length); | |
3817 | |
3818 bufsize = offset - extra; | |
3819 | |
3820 write_direct(data - extra, strlength(data) + extra); | |
3821 } | |
3822 } | |
3823 | |
3824 void write(char_t d0) | |
3825 { | |
3826 size_t offset = bufsize; | |
3827 if (offset > bufcapacity - 1) offset = flush(); | |
3828 | |
3829 buffer[offset + 0] = d0; | |
3830 bufsize = offset + 1; | |
3831 } | |
3832 | |
3833 void write(char_t d0, char_t d1) | |
3834 { | |
3835 size_t offset = bufsize; | |
3836 if (offset > bufcapacity - 2) offset = flush(); | |
3837 | |
3838 buffer[offset + 0] = d0; | |
3839 buffer[offset + 1] = d1; | |
3840 bufsize = offset + 2; | |
3841 } | |
3842 | |
3843 void write(char_t d0, char_t d1, char_t d2) | |
3844 { | |
3845 size_t offset = bufsize; | |
3846 if (offset > bufcapacity - 3) offset = flush(); | |
3847 | |
3848 buffer[offset + 0] = d0; | |
3849 buffer[offset + 1] = d1; | |
3850 buffer[offset + 2] = d2; | |
3851 bufsize = offset + 3; | |
3852 } | |
3853 | |
3854 void write(char_t d0, char_t d1, char_t d2, char_t d3) | |
3855 { | |
3856 size_t offset = bufsize; | |
3857 if (offset > bufcapacity - 4) offset = flush(); | |
3858 | |
3859 buffer[offset + 0] = d0; | |
3860 buffer[offset + 1] = d1; | |
3861 buffer[offset + 2] = d2; | |
3862 buffer[offset + 3] = d3; | |
3863 bufsize = offset + 4; | |
3864 } | |
3865 | |
3866 void write(char_t d0, char_t d1, char_t d2, char_t d3, char_t d4) | |
3867 { | |
3868 size_t offset = bufsize; | |
3869 if (offset > bufcapacity - 5) offset = flush(); | |
3870 | |
3871 buffer[offset + 0] = d0; | |
3872 buffer[offset + 1] = d1; | |
3873 buffer[offset + 2] = d2; | |
3874 buffer[offset + 3] = d3; | |
3875 buffer[offset + 4] = d4; | |
3876 bufsize = offset + 5; | |
3877 } | |
3878 | |
3879 void write(char_t d0, char_t d1, char_t d2, char_t d3, char_t d4, char_t d5) | |
3880 { | |
3881 size_t offset = bufsize; | |
3882 if (offset > bufcapacity - 6) offset = flush(); | |
3883 | |
3884 buffer[offset + 0] = d0; | |
3885 buffer[offset + 1] = d1; | |
3886 buffer[offset + 2] = d2; | |
3887 buffer[offset + 3] = d3; | |
3888 buffer[offset + 4] = d4; | |
3889 buffer[offset + 5] = d5; | |
3890 bufsize = offset + 6; | |
3891 } | |
3892 | |
3893 // utf8 maximum expansion: x4 (-> utf32) | |
3894 // utf16 maximum expansion: x2 (-> utf32) | |
3895 // utf32 maximum expansion: x1 | |
3896 enum | |
3897 { | |
3898 bufcapacitybytes = | |
3899 #ifdef PUGIXML_MEMORY_OUTPUT_STACK | |
3900 PUGIXML_MEMORY_OUTPUT_STACK | |
3901 #else | |
3902 10240 | |
3903 #endif | |
3904 , | |
3905 bufcapacity = bufcapacitybytes / (sizeof(char_t) + 4) | |
3906 }; | |
3907 | |
3908 char_t buffer[bufcapacity]; | |
3909 | |
3910 union | |
3911 { | |
3912 uint8_t data_u8[4 * bufcapacity]; | |
3913 uint16_t data_u16[2 * bufcapacity]; | |
3914 uint32_t data_u32[bufcapacity]; | |
3915 char_t data_char[bufcapacity]; | |
3916 } scratch; | |
3917 | |
3918 xml_writer& writer; | |
3919 size_t bufsize; | |
3920 xml_encoding encoding; | |
3921 }; | |
3922 | |
3923 PUGI__FN void text_output_escaped(xml_buffered_writer& writer, const char_t* s, chartypex_t type, unsigned int flags) | |
3924 { | |
3925 while (*s) | |
3926 { | |
3927 const char_t* prev = s; | |
3928 | |
3929 // While *s is a usual symbol | |
3930 PUGI__SCANWHILE_UNROLL(!PUGI__IS_CHARTYPEX(ss, type)); | |
3931 | |
3932 writer.write_buffer(prev, static_cast<size_t>(s - prev)); | |
3933 | |
3934 switch (*s) | |
3935 { | |
3936 case 0: break; | |
3937 case '&': | |
3938 writer.write('&', 'a', 'm', 'p', ';'); | |
3939 ++s; | |
3940 break; | |
3941 case '<': | |
3942 writer.write('&', 'l', 't', ';'); | |
3943 ++s; | |
3944 break; | |
3945 case '>': | |
3946 writer.write('&', 'g', 't', ';'); | |
3947 ++s; | |
3948 break; | |
3949 case '"': | |
3950 if (flags & format_attribute_single_quote) | |
3951 writer.write('"'); | |
3952 else | |
3953 writer.write('&', 'q', 'u', 'o', 't', ';'); | |
3954 ++s; | |
3955 break; | |
3956 case '\'': | |
3957 if (flags & format_attribute_single_quote) | |
3958 writer.write('&', 'a', 'p', 'o', 's', ';'); | |
3959 else | |
3960 writer.write('\''); | |
3961 ++s; | |
3962 break; | |
3963 default: // s is not a usual symbol | |
3964 { | |
3965 unsigned int ch = static_cast<unsigned int>(*s++); | |
3966 assert(ch < 32); | |
3967 | |
3968 if (!(flags & format_skip_control_chars)) | |
3969 writer.write('&', '#', static_cast<char_t>((ch / 10) + '0'), static_cast<char_t>((ch % 10) + '0'), ';'); | |
3970 } | |
3971 } | |
3972 } | |
3973 } | |
3974 | |
3975 PUGI__FN void text_output(xml_buffered_writer& writer, const char_t* s, chartypex_t type, unsigned int flags) | |
3976 { | |
3977 if (flags & format_no_escapes) | |
3978 writer.write_string(s); | |
3979 else | |
3980 text_output_escaped(writer, s, type, flags); | |
3981 } | |
3982 | |
3983 PUGI__FN void text_output_cdata(xml_buffered_writer& writer, const char_t* s) | |
3984 { | |
3985 do | |
3986 { | |
3987 writer.write('<', '!', '[', 'C', 'D'); | |
3988 writer.write('A', 'T', 'A', '['); | |
3989 | |
3990 const char_t* prev = s; | |
3991 | |
3992 // look for ]]> sequence - we can't output it as is since it terminates CDATA | |
3993 while (*s && !(s[0] == ']' && s[1] == ']' && s[2] == '>')) ++s; | |
3994 | |
3995 // skip ]] if we stopped at ]]>, > will go to the next CDATA section | |
3996 if (*s) s += 2; | |
3997 | |
3998 writer.write_buffer(prev, static_cast<size_t>(s - prev)); | |
3999 | |
4000 writer.write(']', ']', '>'); | |
4001 } | |
4002 while (*s); | |
4003 } | |
4004 | |
4005 PUGI__FN void text_output_indent(xml_buffered_writer& writer, const char_t* indent, size_t indent_length, unsigned int depth) | |
4006 { | |
4007 switch (indent_length) | |
4008 { | |
4009 case 1: | |
4010 { | |
4011 for (unsigned int i = 0; i < depth; ++i) | |
4012 writer.write(indent[0]); | |
4013 break; | |
4014 } | |
4015 | |
4016 case 2: | |
4017 { | |
4018 for (unsigned int i = 0; i < depth; ++i) | |
4019 writer.write(indent[0], indent[1]); | |
4020 break; | |
4021 } | |
4022 | |
4023 case 3: | |
4024 { | |
4025 for (unsigned int i = 0; i < depth; ++i) | |
4026 writer.write(indent[0], indent[1], indent[2]); | |
4027 break; | |
4028 } | |
4029 | |
4030 case 4: | |
4031 { | |
4032 for (unsigned int i = 0; i < depth; ++i) | |
4033 writer.write(indent[0], indent[1], indent[2], indent[3]); | |
4034 break; | |
4035 } | |
4036 | |
4037 default: | |
4038 { | |
4039 for (unsigned int i = 0; i < depth; ++i) | |
4040 writer.write_buffer(indent, indent_length); | |
4041 } | |
4042 } | |
4043 } | |
4044 | |
4045 PUGI__FN void node_output_comment(xml_buffered_writer& writer, const char_t* s) | |
4046 { | |
4047 writer.write('<', '!', '-', '-'); | |
4048 | |
4049 while (*s) | |
4050 { | |
4051 const char_t* prev = s; | |
4052 | |
4053 // look for -\0 or -- sequence - we can't output it since -- is illegal in comment body | |
4054 while (*s && !(s[0] == '-' && (s[1] == '-' || s[1] == 0))) ++s; | |
4055 | |
4056 writer.write_buffer(prev, static_cast<size_t>(s - prev)); | |
4057 | |
4058 if (*s) | |
4059 { | |
4060 assert(*s == '-'); | |
4061 | |
4062 writer.write('-', ' '); | |
4063 ++s; | |
4064 } | |
4065 } | |
4066 | |
4067 writer.write('-', '-', '>'); | |
4068 } | |
4069 | |
4070 PUGI__FN void node_output_pi_value(xml_buffered_writer& writer, const char_t* s) | |
4071 { | |
4072 while (*s) | |
4073 { | |
4074 const char_t* prev = s; | |
4075 | |
4076 // look for ?> sequence - we can't output it since ?> terminates PI | |
4077 while (*s && !(s[0] == '?' && s[1] == '>')) ++s; | |
4078 | |
4079 writer.write_buffer(prev, static_cast<size_t>(s - prev)); | |
4080 | |
4081 if (*s) | |
4082 { | |
4083 assert(s[0] == '?' && s[1] == '>'); | |
4084 | |
4085 writer.write('?', ' ', '>'); | |
4086 s += 2; | |
4087 } | |
4088 } | |
4089 } | |
4090 | |
4091 PUGI__FN void node_output_attributes(xml_buffered_writer& writer, xml_node_struct* node, const char_t* indent, size_t indent_length, unsigned int flags, unsigned int depth) | |
4092 { | |
4093 const char_t* default_name = PUGIXML_TEXT(":anonymous"); | |
4094 const char_t enquotation_char = (flags & format_attribute_single_quote) ? '\'' : '"'; | |
4095 | |
4096 for (xml_attribute_struct* a = node->first_attribute; a; a = a->next_attribute) | |
4097 { | |
4098 if ((flags & (format_indent_attributes | format_raw)) == format_indent_attributes) | |
4099 { | |
4100 writer.write('\n'); | |
4101 | |
4102 text_output_indent(writer, indent, indent_length, depth + 1); | |
4103 } | |
4104 else | |
4105 { | |
4106 writer.write(' '); | |
4107 } | |
4108 | |
4109 writer.write_string(a->name ? a->name + 0 : default_name); | |
4110 writer.write('=', enquotation_char); | |
4111 | |
4112 if (a->value) | |
4113 text_output(writer, a->value, ctx_special_attr, flags); | |
4114 | |
4115 writer.write(enquotation_char); | |
4116 } | |
4117 } | |
4118 | |
4119 PUGI__FN bool node_output_start(xml_buffered_writer& writer, xml_node_struct* node, const char_t* indent, size_t indent_length, unsigned int flags, unsigned int depth) | |
4120 { | |
4121 const char_t* default_name = PUGIXML_TEXT(":anonymous"); | |
4122 const char_t* name = node->name ? node->name + 0 : default_name; | |
4123 | |
4124 writer.write('<'); | |
4125 writer.write_string(name); | |
4126 | |
4127 if (node->first_attribute) | |
4128 node_output_attributes(writer, node, indent, indent_length, flags, depth); | |
4129 | |
4130 // element nodes can have value if parse_embed_pcdata was used | |
4131 if (!node->value) | |
4132 { | |
4133 if (!node->first_child) | |
4134 { | |
4135 if (flags & format_no_empty_element_tags) | |
4136 { | |
4137 writer.write('>', '<', '/'); | |
4138 writer.write_string(name); | |
4139 writer.write('>'); | |
4140 | |
4141 return false; | |
4142 } | |
4143 else | |
4144 { | |
4145 if ((flags & format_raw) == 0) | |
4146 writer.write(' '); | |
4147 | |
4148 writer.write('/', '>'); | |
4149 | |
4150 return false; | |
4151 } | |
4152 } | |
4153 else | |
4154 { | |
4155 writer.write('>'); | |
4156 | |
4157 return true; | |
4158 } | |
4159 } | |
4160 else | |
4161 { | |
4162 writer.write('>'); | |
4163 | |
4164 text_output(writer, node->value, ctx_special_pcdata, flags); | |
4165 | |
4166 if (!node->first_child) | |
4167 { | |
4168 writer.write('<', '/'); | |
4169 writer.write_string(name); | |
4170 writer.write('>'); | |
4171 | |
4172 return false; | |
4173 } | |
4174 else | |
4175 { | |
4176 return true; | |
4177 } | |
4178 } | |
4179 } | |
4180 | |
4181 PUGI__FN void node_output_end(xml_buffered_writer& writer, xml_node_struct* node) | |
4182 { | |
4183 const char_t* default_name = PUGIXML_TEXT(":anonymous"); | |
4184 const char_t* name = node->name ? node->name + 0 : default_name; | |
4185 | |
4186 writer.write('<', '/'); | |
4187 writer.write_string(name); | |
4188 writer.write('>'); | |
4189 } | |
4190 | |
4191 PUGI__FN void node_output_simple(xml_buffered_writer& writer, xml_node_struct* node, unsigned int flags) | |
4192 { | |
4193 const char_t* default_name = PUGIXML_TEXT(":anonymous"); | |
4194 | |
4195 switch (PUGI__NODETYPE(node)) | |
4196 { | |
4197 case node_pcdata: | |
4198 text_output(writer, node->value ? node->value + 0 : PUGIXML_TEXT(""), ctx_special_pcdata, flags); | |
4199 break; | |
4200 | |
4201 case node_cdata: | |
4202 text_output_cdata(writer, node->value ? node->value + 0 : PUGIXML_TEXT("")); | |
4203 break; | |
4204 | |
4205 case node_comment: | |
4206 node_output_comment(writer, node->value ? node->value + 0 : PUGIXML_TEXT("")); | |
4207 break; | |
4208 | |
4209 case node_pi: | |
4210 writer.write('<', '?'); | |
4211 writer.write_string(node->name ? node->name + 0 : default_name); | |
4212 | |
4213 if (node->value) | |
4214 { | |
4215 writer.write(' '); | |
4216 node_output_pi_value(writer, node->value); | |
4217 } | |
4218 | |
4219 writer.write('?', '>'); | |
4220 break; | |
4221 | |
4222 case node_declaration: | |
4223 writer.write('<', '?'); | |
4224 writer.write_string(node->name ? node->name + 0 : default_name); | |
4225 node_output_attributes(writer, node, PUGIXML_TEXT(""), 0, flags | format_raw, 0); | |
4226 writer.write('?', '>'); | |
4227 break; | |
4228 | |
4229 case node_doctype: | |
4230 writer.write('<', '!', 'D', 'O', 'C'); | |
4231 writer.write('T', 'Y', 'P', 'E'); | |
4232 | |
4233 if (node->value) | |
4234 { | |
4235 writer.write(' '); | |
4236 writer.write_string(node->value); | |
4237 } | |
4238 | |
4239 writer.write('>'); | |
4240 break; | |
4241 | |
4242 default: | |
4243 assert(false && "Invalid node type"); // unreachable | |
4244 } | |
4245 } | |
4246 | |
4247 enum indent_flags_t | |
4248 { | |
4249 indent_newline = 1, | |
4250 indent_indent = 2 | |
4251 }; | |
4252 | |
4253 PUGI__FN void node_output(xml_buffered_writer& writer, xml_node_struct* root, const char_t* indent, unsigned int flags, unsigned int depth) | |
4254 { | |
4255 size_t indent_length = ((flags & (format_indent | format_indent_attributes)) && (flags & format_raw) == 0) ? strlength(indent) : 0; | |
4256 unsigned int indent_flags = indent_indent; | |
4257 | |
4258 xml_node_struct* node = root; | |
4259 | |
4260 do | |
4261 { | |
4262 assert(node); | |
4263 | |
4264 // begin writing current node | |
4265 if (PUGI__NODETYPE(node) == node_pcdata || PUGI__NODETYPE(node) == node_cdata) | |
4266 { | |
4267 node_output_simple(writer, node, flags); | |
4268 | |
4269 indent_flags = 0; | |
4270 } | |
4271 else | |
4272 { | |
4273 if ((indent_flags & indent_newline) && (flags & format_raw) == 0) | |
4274 writer.write('\n'); | |
4275 | |
4276 if ((indent_flags & indent_indent) && indent_length) | |
4277 text_output_indent(writer, indent, indent_length, depth); | |
4278 | |
4279 if (PUGI__NODETYPE(node) == node_element) | |
4280 { | |
4281 indent_flags = indent_newline | indent_indent; | |
4282 | |
4283 if (node_output_start(writer, node, indent, indent_length, flags, depth)) | |
4284 { | |
4285 // element nodes can have value if parse_embed_pcdata was used | |
4286 if (node->value) | |
4287 indent_flags = 0; | |
4288 | |
4289 node = node->first_child; | |
4290 depth++; | |
4291 continue; | |
4292 } | |
4293 } | |
4294 else if (PUGI__NODETYPE(node) == node_document) | |
4295 { | |
4296 indent_flags = indent_indent; | |
4297 | |
4298 if (node->first_child) | |
4299 { | |
4300 node = node->first_child; | |
4301 continue; | |
4302 } | |
4303 } | |
4304 else | |
4305 { | |
4306 node_output_simple(writer, node, flags); | |
4307 | |
4308 indent_flags = indent_newline | indent_indent; | |
4309 } | |
4310 } | |
4311 | |
4312 // continue to the next node | |
4313 while (node != root) | |
4314 { | |
4315 if (node->next_sibling) | |
4316 { | |
4317 node = node->next_sibling; | |
4318 break; | |
4319 } | |
4320 | |
4321 node = node->parent; | |
4322 | |
4323 // write closing node | |
4324 if (PUGI__NODETYPE(node) == node_element) | |
4325 { | |
4326 depth--; | |
4327 | |
4328 if ((indent_flags & indent_newline) && (flags & format_raw) == 0) | |
4329 writer.write('\n'); | |
4330 | |
4331 if ((indent_flags & indent_indent) && indent_length) | |
4332 text_output_indent(writer, indent, indent_length, depth); | |
4333 | |
4334 node_output_end(writer, node); | |
4335 | |
4336 indent_flags = indent_newline | indent_indent; | |
4337 } | |
4338 } | |
4339 } | |
4340 while (node != root); | |
4341 | |
4342 if ((indent_flags & indent_newline) && (flags & format_raw) == 0) | |
4343 writer.write('\n'); | |
4344 } | |
4345 | |
4346 PUGI__FN bool has_declaration(xml_node_struct* node) | |
4347 { | |
4348 for (xml_node_struct* child = node->first_child; child; child = child->next_sibling) | |
4349 { | |
4350 xml_node_type type = PUGI__NODETYPE(child); | |
4351 | |
4352 if (type == node_declaration) return true; | |
4353 if (type == node_element) return false; | |
4354 } | |
4355 | |
4356 return false; | |
4357 } | |
4358 | |
4359 PUGI__FN bool is_attribute_of(xml_attribute_struct* attr, xml_node_struct* node) | |
4360 { | |
4361 for (xml_attribute_struct* a = node->first_attribute; a; a = a->next_attribute) | |
4362 if (a == attr) | |
4363 return true; | |
4364 | |
4365 return false; | |
4366 } | |
4367 | |
4368 PUGI__FN bool allow_insert_attribute(xml_node_type parent) | |
4369 { | |
4370 return parent == node_element || parent == node_declaration; | |
4371 } | |
4372 | |
4373 PUGI__FN bool allow_insert_child(xml_node_type parent, xml_node_type child) | |
4374 { | |
4375 if (parent != node_document && parent != node_element) return false; | |
4376 if (child == node_document || child == node_null) return false; | |
4377 if (parent != node_document && (child == node_declaration || child == node_doctype)) return false; | |
4378 | |
4379 return true; | |
4380 } | |
4381 | |
4382 PUGI__FN bool allow_move(xml_node parent, xml_node child) | |
4383 { | |
4384 // check that child can be a child of parent | |
4385 if (!allow_insert_child(parent.type(), child.type())) | |
4386 return false; | |
4387 | |
4388 // check that node is not moved between documents | |
4389 if (parent.root() != child.root()) | |
4390 return false; | |
4391 | |
4392 // check that new parent is not in the child subtree | |
4393 xml_node cur = parent; | |
4394 | |
4395 while (cur) | |
4396 { | |
4397 if (cur == child) | |
4398 return false; | |
4399 | |
4400 cur = cur.parent(); | |
4401 } | |
4402 | |
4403 return true; | |
4404 } | |
4405 | |
4406 template <typename String, typename Header> | |
4407 PUGI__FN void node_copy_string(String& dest, Header& header, uintptr_t header_mask, char_t* source, Header& source_header, xml_allocator* alloc) | |
4408 { | |
4409 assert(!dest && (header & header_mask) == 0); | |
4410 | |
4411 if (source) | |
4412 { | |
4413 if (alloc && (source_header & header_mask) == 0) | |
4414 { | |
4415 dest = source; | |
4416 | |
4417 // since strcpy_insitu can reuse document buffer memory we need to mark both source and dest as shared | |
4418 header |= xml_memory_page_contents_shared_mask; | |
4419 source_header |= xml_memory_page_contents_shared_mask; | |
4420 } | |
4421 else | |
4422 strcpy_insitu(dest, header, header_mask, source, strlength(source)); | |
4423 } | |
4424 } | |
4425 | |
4426 PUGI__FN void node_copy_contents(xml_node_struct* dn, xml_node_struct* sn, xml_allocator* shared_alloc) | |
4427 { | |
4428 node_copy_string(dn->name, dn->header, xml_memory_page_name_allocated_mask, sn->name, sn->header, shared_alloc); | |
4429 node_copy_string(dn->value, dn->header, xml_memory_page_value_allocated_mask, sn->value, sn->header, shared_alloc); | |
4430 | |
4431 for (xml_attribute_struct* sa = sn->first_attribute; sa; sa = sa->next_attribute) | |
4432 { | |
4433 xml_attribute_struct* da = append_new_attribute(dn, get_allocator(dn)); | |
4434 | |
4435 if (da) | |
4436 { | |
4437 node_copy_string(da->name, da->header, xml_memory_page_name_allocated_mask, sa->name, sa->header, shared_alloc); | |
4438 node_copy_string(da->value, da->header, xml_memory_page_value_allocated_mask, sa->value, sa->header, shared_alloc); | |
4439 } | |
4440 } | |
4441 } | |
4442 | |
4443 PUGI__FN void node_copy_tree(xml_node_struct* dn, xml_node_struct* sn) | |
4444 { | |
4445 xml_allocator& alloc = get_allocator(dn); | |
4446 xml_allocator* shared_alloc = (&alloc == &get_allocator(sn)) ? &alloc : 0; | |
4447 | |
4448 node_copy_contents(dn, sn, shared_alloc); | |
4449 | |
4450 xml_node_struct* dit = dn; | |
4451 xml_node_struct* sit = sn->first_child; | |
4452 | |
4453 while (sit && sit != sn) | |
4454 { | |
4455 // loop invariant: dit is inside the subtree rooted at dn | |
4456 assert(dit); | |
4457 | |
4458 // when a tree is copied into one of the descendants, we need to skip that subtree to avoid an infinite loop | |
4459 if (sit != dn) | |
4460 { | |
4461 xml_node_struct* copy = append_new_node(dit, alloc, PUGI__NODETYPE(sit)); | |
4462 | |
4463 if (copy) | |
4464 { | |
4465 node_copy_contents(copy, sit, shared_alloc); | |
4466 | |
4467 if (sit->first_child) | |
4468 { | |
4469 dit = copy; | |
4470 sit = sit->first_child; | |
4471 continue; | |
4472 } | |
4473 } | |
4474 } | |
4475 | |
4476 // continue to the next node | |
4477 do | |
4478 { | |
4479 if (sit->next_sibling) | |
4480 { | |
4481 sit = sit->next_sibling; | |
4482 break; | |
4483 } | |
4484 | |
4485 sit = sit->parent; | |
4486 dit = dit->parent; | |
4487 | |
4488 // loop invariant: dit is inside the subtree rooted at dn while sit is inside sn | |
4489 assert(sit == sn || dit); | |
4490 } | |
4491 while (sit != sn); | |
4492 } | |
4493 | |
4494 assert(!sit || dit == dn->parent); | |
4495 } | |
4496 | |
4497 PUGI__FN void node_copy_attribute(xml_attribute_struct* da, xml_attribute_struct* sa) | |
4498 { | |
4499 xml_allocator& alloc = get_allocator(da); | |
4500 xml_allocator* shared_alloc = (&alloc == &get_allocator(sa)) ? &alloc : 0; | |
4501 | |
4502 node_copy_string(da->name, da->header, xml_memory_page_name_allocated_mask, sa->name, sa->header, shared_alloc); | |
4503 node_copy_string(da->value, da->header, xml_memory_page_value_allocated_mask, sa->value, sa->header, shared_alloc); | |
4504 } | |
4505 | |
4506 inline bool is_text_node(xml_node_struct* node) | |
4507 { | |
4508 xml_node_type type = PUGI__NODETYPE(node); | |
4509 | |
4510 return type == node_pcdata || type == node_cdata; | |
4511 } | |
4512 | |
4513 // get value with conversion functions | |
4514 template <typename U> PUGI__FN PUGI__UNSIGNED_OVERFLOW U string_to_integer(const char_t* value, U minv, U maxv) | |
4515 { | |
4516 U result = 0; | |
4517 const char_t* s = value; | |
4518 | |
4519 while (PUGI__IS_CHARTYPE(*s, ct_space)) | |
4520 s++; | |
4521 | |
4522 bool negative = (*s == '-'); | |
4523 | |
4524 s += (*s == '+' || *s == '-'); | |
4525 | |
4526 bool overflow = false; | |
4527 | |
4528 if (s[0] == '0' && (s[1] | ' ') == 'x') | |
4529 { | |
4530 s += 2; | |
4531 | |
4532 // since overflow detection relies on length of the sequence skip leading zeros | |
4533 while (*s == '0') | |
4534 s++; | |
4535 | |
4536 const char_t* start = s; | |
4537 | |
4538 for (;;) | |
4539 { | |
4540 if (static_cast<unsigned>(*s - '0') < 10) | |
4541 result = result * 16 + (*s - '0'); | |
4542 else if (static_cast<unsigned>((*s | ' ') - 'a') < 6) | |
4543 result = result * 16 + ((*s | ' ') - 'a' + 10); | |
4544 else | |
4545 break; | |
4546 | |
4547 s++; | |
4548 } | |
4549 | |
4550 size_t digits = static_cast<size_t>(s - start); | |
4551 | |
4552 overflow = digits > sizeof(U) * 2; | |
4553 } | |
4554 else | |
4555 { | |
4556 // since overflow detection relies on length of the sequence skip leading zeros | |
4557 while (*s == '0') | |
4558 s++; | |
4559 | |
4560 const char_t* start = s; | |
4561 | |
4562 for (;;) | |
4563 { | |
4564 if (static_cast<unsigned>(*s - '0') < 10) | |
4565 result = result * 10 + (*s - '0'); | |
4566 else | |
4567 break; | |
4568 | |
4569 s++; | |
4570 } | |
4571 | |
4572 size_t digits = static_cast<size_t>(s - start); | |
4573 | |
4574 PUGI__STATIC_ASSERT(sizeof(U) == 8 || sizeof(U) == 4 || sizeof(U) == 2); | |
4575 | |
4576 const size_t max_digits10 = sizeof(U) == 8 ? 20 : sizeof(U) == 4 ? 10 : 5; | |
4577 const char_t max_lead = sizeof(U) == 8 ? '1' : sizeof(U) == 4 ? '4' : '6'; | |
4578 const size_t high_bit = sizeof(U) * 8 - 1; | |
4579 | |
4580 overflow = digits >= max_digits10 && !(digits == max_digits10 && (*start < max_lead || (*start == max_lead && result >> high_bit))); | |
4581 } | |
4582 | |
4583 if (negative) | |
4584 { | |
4585 // Workaround for crayc++ CC-3059: Expected no overflow in routine. | |
4586 #ifdef _CRAYC | |
4587 return (overflow || result > ~minv + 1) ? minv : ~result + 1; | |
4588 #else | |
4589 return (overflow || result > 0 - minv) ? minv : 0 - result; | |
4590 #endif | |
4591 } | |
4592 else | |
4593 return (overflow || result > maxv) ? maxv : result; | |
4594 } | |
4595 | |
4596 PUGI__FN int get_value_int(const char_t* value) | |
4597 { | |
4598 return string_to_integer<unsigned int>(value, static_cast<unsigned int>(INT_MIN), INT_MAX); | |
4599 } | |
4600 | |
4601 PUGI__FN unsigned int get_value_uint(const char_t* value) | |
4602 { | |
4603 return string_to_integer<unsigned int>(value, 0, UINT_MAX); | |
4604 } | |
4605 | |
4606 PUGI__FN double get_value_double(const char_t* value) | |
4607 { | |
4608 #ifdef PUGIXML_WCHAR_MODE | |
4609 return wcstod(value, 0); | |
4610 #else | |
4611 return strtod(value, 0); | |
4612 #endif | |
4613 } | |
4614 | |
4615 PUGI__FN float get_value_float(const char_t* value) | |
4616 { | |
4617 #ifdef PUGIXML_WCHAR_MODE | |
4618 return static_cast<float>(wcstod(value, 0)); | |
4619 #else | |
4620 return static_cast<float>(strtod(value, 0)); | |
4621 #endif | |
4622 } | |
4623 | |
4624 PUGI__FN bool get_value_bool(const char_t* value) | |
4625 { | |
4626 // only look at first char | |
4627 char_t first = *value; | |
4628 | |
4629 // 1*, t* (true), T* (True), y* (yes), Y* (YES) | |
4630 return (first == '1' || first == 't' || first == 'T' || first == 'y' || first == 'Y'); | |
4631 } | |
4632 | |
4633 #ifdef PUGIXML_HAS_LONG_LONG | |
4634 PUGI__FN long long get_value_llong(const char_t* value) | |
4635 { | |
4636 return string_to_integer<unsigned long long>(value, static_cast<unsigned long long>(LLONG_MIN), LLONG_MAX); | |
4637 } | |
4638 | |
4639 PUGI__FN unsigned long long get_value_ullong(const char_t* value) | |
4640 { | |
4641 return string_to_integer<unsigned long long>(value, 0, ULLONG_MAX); | |
4642 } | |
4643 #endif | |
4644 | |
4645 template <typename U> PUGI__FN PUGI__UNSIGNED_OVERFLOW char_t* integer_to_string(char_t* begin, char_t* end, U value, bool negative) | |
4646 { | |
4647 char_t* result = end - 1; | |
4648 U rest = negative ? 0 - value : value; | |
4649 | |
4650 do | |
4651 { | |
4652 *result-- = static_cast<char_t>('0' + (rest % 10)); | |
4653 rest /= 10; | |
4654 } | |
4655 while (rest); | |
4656 | |
4657 assert(result >= begin); | |
4658 (void)begin; | |
4659 | |
4660 *result = '-'; | |
4661 | |
4662 return result + !negative; | |
4663 } | |
4664 | |
4665 // set value with conversion functions | |
4666 template <typename String, typename Header> | |
4667 PUGI__FN bool set_value_ascii(String& dest, Header& header, uintptr_t header_mask, char* buf) | |
4668 { | |
4669 #ifdef PUGIXML_WCHAR_MODE | |
4670 char_t wbuf[128]; | |
4671 assert(strlen(buf) < sizeof(wbuf) / sizeof(wbuf[0])); | |
4672 | |
4673 size_t offset = 0; | |
4674 for (; buf[offset]; ++offset) wbuf[offset] = buf[offset]; | |
4675 | |
4676 return strcpy_insitu(dest, header, header_mask, wbuf, offset); | |
4677 #else | |
4678 return strcpy_insitu(dest, header, header_mask, buf, strlen(buf)); | |
4679 #endif | |
4680 } | |
4681 | |
4682 template <typename U, typename String, typename Header> | |
4683 PUGI__FN bool set_value_integer(String& dest, Header& header, uintptr_t header_mask, U value, bool negative) | |
4684 { | |
4685 char_t buf[64]; | |
4686 char_t* end = buf + sizeof(buf) / sizeof(buf[0]); | |
4687 char_t* begin = integer_to_string(buf, end, value, negative); | |
4688 | |
4689 return strcpy_insitu(dest, header, header_mask, begin, end - begin); | |
4690 } | |
4691 | |
4692 template <typename String, typename Header> | |
4693 PUGI__FN bool set_value_convert(String& dest, Header& header, uintptr_t header_mask, float value, int precision) | |
4694 { | |
4695 char buf[128]; | |
4696 PUGI__SNPRINTF(buf, "%.*g", precision, double(value)); | |
4697 | |
4698 return set_value_ascii(dest, header, header_mask, buf); | |
4699 } | |
4700 | |
4701 template <typename String, typename Header> | |
4702 PUGI__FN bool set_value_convert(String& dest, Header& header, uintptr_t header_mask, double value, int precision) | |
4703 { | |
4704 char buf[128]; | |
4705 PUGI__SNPRINTF(buf, "%.*g", precision, value); | |
4706 | |
4707 return set_value_ascii(dest, header, header_mask, buf); | |
4708 } | |
4709 | |
4710 template <typename String, typename Header> | |
4711 PUGI__FN bool set_value_bool(String& dest, Header& header, uintptr_t header_mask, bool value) | |
4712 { | |
4713 return strcpy_insitu(dest, header, header_mask, value ? PUGIXML_TEXT("true") : PUGIXML_TEXT("false"), value ? 4 : 5); | |
4714 } | |
4715 | |
4716 PUGI__FN xml_parse_result load_buffer_impl(xml_document_struct* doc, xml_node_struct* root, void* contents, size_t size, unsigned int options, xml_encoding encoding, bool is_mutable, bool own, char_t** out_buffer) | |
4717 { | |
4718 // check input buffer | |
4719 if (!contents && size) return make_parse_result(status_io_error); | |
4720 | |
4721 // get actual encoding | |
4722 xml_encoding buffer_encoding = impl::get_buffer_encoding(encoding, contents, size); | |
4723 | |
4724 // if convert_buffer below throws bad_alloc, we still need to deallocate contents if we own it | |
4725 auto_deleter<void> contents_guard(own ? contents : 0, xml_memory::deallocate); | |
4726 | |
4727 // get private buffer | |
4728 char_t* buffer = 0; | |
4729 size_t length = 0; | |
4730 | |
4731 // coverity[var_deref_model] | |
4732 if (!impl::convert_buffer(buffer, length, buffer_encoding, contents, size, is_mutable)) return impl::make_parse_result(status_out_of_memory); | |
4733 | |
4734 // after this we either deallocate contents (below) or hold on to it via doc->buffer, so we don't need to guard it | |
4735 contents_guard.release(); | |
4736 | |
4737 // delete original buffer if we performed a conversion | |
4738 if (own && buffer != contents && contents) impl::xml_memory::deallocate(contents); | |
4739 | |
4740 // grab onto buffer if it's our buffer, user is responsible for deallocating contents himself | |
4741 if (own || buffer != contents) *out_buffer = buffer; | |
4742 | |
4743 // store buffer for offset_debug | |
4744 doc->buffer = buffer; | |
4745 | |
4746 // parse | |
4747 xml_parse_result res = impl::xml_parser::parse(buffer, length, doc, root, options); | |
4748 | |
4749 // remember encoding | |
4750 res.encoding = buffer_encoding; | |
4751 | |
4752 return res; | |
4753 } | |
4754 | |
4755 // we need to get length of entire file to load it in memory; the only (relatively) sane way to do it is via seek/tell trick | |
4756 PUGI__FN xml_parse_status get_file_size(FILE* file, size_t& out_result) | |
4757 { | |
4758 #if defined(PUGI__MSVC_CRT_VERSION) && PUGI__MSVC_CRT_VERSION >= 1400 | |
4759 // there are 64-bit versions of fseek/ftell, let's use them | |
4760 typedef __int64 length_type; | |
4761 | |
4762 _fseeki64(file, 0, SEEK_END); | |
4763 length_type length = _ftelli64(file); | |
4764 _fseeki64(file, 0, SEEK_SET); | |
4765 #elif defined(__MINGW32__) && !defined(__NO_MINGW_LFS) && (!defined(__STRICT_ANSI__) || defined(__MINGW64_VERSION_MAJOR)) | |
4766 // there are 64-bit versions of fseek/ftell, let's use them | |
4767 typedef off64_t length_type; | |
4768 | |
4769 fseeko64(file, 0, SEEK_END); | |
4770 length_type length = ftello64(file); | |
4771 fseeko64(file, 0, SEEK_SET); | |
4772 #else | |
4773 // if this is a 32-bit OS, long is enough; if this is a unix system, long is 64-bit, which is enough; otherwise we can't do anything anyway. | |
4774 typedef long length_type; | |
4775 | |
4776 fseek(file, 0, SEEK_END); | |
4777 length_type length = ftell(file); | |
4778 fseek(file, 0, SEEK_SET); | |
4779 #endif | |
4780 | |
4781 // check for I/O errors | |
4782 if (length < 0) return status_io_error; | |
4783 | |
4784 // check for overflow | |
4785 size_t result = static_cast<size_t>(length); | |
4786 | |
4787 if (static_cast<length_type>(result) != length) return status_out_of_memory; | |
4788 | |
4789 // finalize | |
4790 out_result = result; | |
4791 | |
4792 return status_ok; | |
4793 } | |
4794 | |
4795 // This function assumes that buffer has extra sizeof(char_t) writable bytes after size | |
4796 PUGI__FN size_t zero_terminate_buffer(void* buffer, size_t size, xml_encoding encoding) | |
4797 { | |
4798 // We only need to zero-terminate if encoding conversion does not do it for us | |
4799 #ifdef PUGIXML_WCHAR_MODE | |
4800 xml_encoding wchar_encoding = get_wchar_encoding(); | |
4801 | |
4802 if (encoding == wchar_encoding || need_endian_swap_utf(encoding, wchar_encoding)) | |
4803 { | |
4804 size_t length = size / sizeof(char_t); | |
4805 | |
4806 static_cast<char_t*>(buffer)[length] = 0; | |
4807 return (length + 1) * sizeof(char_t); | |
4808 } | |
4809 #else | |
4810 if (encoding == encoding_utf8) | |
4811 { | |
4812 static_cast<char*>(buffer)[size] = 0; | |
4813 return size + 1; | |
4814 } | |
4815 #endif | |
4816 | |
4817 return size; | |
4818 } | |
4819 | |
4820 PUGI__FN xml_parse_result load_file_impl(xml_document_struct* doc, FILE* file, unsigned int options, xml_encoding encoding, char_t** out_buffer) | |
4821 { | |
4822 if (!file) return make_parse_result(status_file_not_found); | |
4823 | |
4824 // get file size (can result in I/O errors) | |
4825 size_t size = 0; | |
4826 xml_parse_status size_status = get_file_size(file, size); | |
4827 if (size_status != status_ok) return make_parse_result(size_status); | |
4828 | |
4829 size_t max_suffix_size = sizeof(char_t); | |
4830 | |
4831 // allocate buffer for the whole file | |
4832 char* contents = static_cast<char*>(xml_memory::allocate(size + max_suffix_size)); | |
4833 if (!contents) return make_parse_result(status_out_of_memory); | |
4834 | |
4835 // read file in memory | |
4836 size_t read_size = fread(contents, 1, size, file); | |
4837 | |
4838 if (read_size != size) | |
4839 { | |
4840 xml_memory::deallocate(contents); | |
4841 return make_parse_result(status_io_error); | |
4842 } | |
4843 | |
4844 xml_encoding real_encoding = get_buffer_encoding(encoding, contents, size); | |
4845 | |
4846 return load_buffer_impl(doc, doc, contents, zero_terminate_buffer(contents, size, real_encoding), options, real_encoding, true, true, out_buffer); | |
4847 } | |
4848 | |
4849 PUGI__FN void close_file(FILE* file) | |
4850 { | |
4851 fclose(file); | |
4852 } | |
4853 | |
4854 #ifndef PUGIXML_NO_STL | |
4855 template <typename T> struct xml_stream_chunk | |
4856 { | |
4857 static xml_stream_chunk* create() | |
4858 { | |
4859 void* memory = xml_memory::allocate(sizeof(xml_stream_chunk)); | |
4860 if (!memory) return 0; | |
4861 | |
4862 return new (memory) xml_stream_chunk(); | |
4863 } | |
4864 | |
4865 static void destroy(xml_stream_chunk* chunk) | |
4866 { | |
4867 // free chunk chain | |
4868 while (chunk) | |
4869 { | |
4870 xml_stream_chunk* next_ = chunk->next; | |
4871 | |
4872 xml_memory::deallocate(chunk); | |
4873 | |
4874 chunk = next_; | |
4875 } | |
4876 } | |
4877 | |
4878 xml_stream_chunk(): next(0), size(0) | |
4879 { | |
4880 } | |
4881 | |
4882 xml_stream_chunk* next; | |
4883 size_t size; | |
4884 | |
4885 T data[xml_memory_page_size / sizeof(T)]; | |
4886 }; | |
4887 | |
4888 template <typename T> PUGI__FN xml_parse_status load_stream_data_noseek(std::basic_istream<T>& stream, void** out_buffer, size_t* out_size) | |
4889 { | |
4890 auto_deleter<xml_stream_chunk<T> > chunks(0, xml_stream_chunk<T>::destroy); | |
4891 | |
4892 // read file to a chunk list | |
4893 size_t total = 0; | |
4894 xml_stream_chunk<T>* last = 0; | |
4895 | |
4896 while (!stream.eof()) | |
4897 { | |
4898 // allocate new chunk | |
4899 xml_stream_chunk<T>* chunk = xml_stream_chunk<T>::create(); | |
4900 if (!chunk) return status_out_of_memory; | |
4901 | |
4902 // append chunk to list | |
4903 if (last) last = last->next = chunk; | |
4904 else chunks.data = last = chunk; | |
4905 | |
4906 // read data to chunk | |
4907 stream.read(chunk->data, static_cast<std::streamsize>(sizeof(chunk->data) / sizeof(T))); | |
4908 chunk->size = static_cast<size_t>(stream.gcount()) * sizeof(T); | |
4909 | |
4910 // read may set failbit | eofbit in case gcount() is less than read length, so check for other I/O errors | |
4911 if (stream.bad() || (!stream.eof() && stream.fail())) return status_io_error; | |
4912 | |
4913 // guard against huge files (chunk size is small enough to make this overflow check work) | |
4914 if (total + chunk->size < total) return status_out_of_memory; | |
4915 total += chunk->size; | |
4916 } | |
4917 | |
4918 size_t max_suffix_size = sizeof(char_t); | |
4919 | |
4920 // copy chunk list to a contiguous buffer | |
4921 char* buffer = static_cast<char*>(xml_memory::allocate(total + max_suffix_size)); | |
4922 if (!buffer) return status_out_of_memory; | |
4923 | |
4924 char* write = buffer; | |
4925 | |
4926 for (xml_stream_chunk<T>* chunk = chunks.data; chunk; chunk = chunk->next) | |
4927 { | |
4928 assert(write + chunk->size <= buffer + total); | |
4929 memcpy(write, chunk->data, chunk->size); | |
4930 write += chunk->size; | |
4931 } | |
4932 | |
4933 assert(write == buffer + total); | |
4934 | |
4935 // return buffer | |
4936 *out_buffer = buffer; | |
4937 *out_size = total; | |
4938 | |
4939 return status_ok; | |
4940 } | |
4941 | |
4942 template <typename T> PUGI__FN xml_parse_status load_stream_data_seek(std::basic_istream<T>& stream, void** out_buffer, size_t* out_size) | |
4943 { | |
4944 // get length of remaining data in stream | |
4945 typename std::basic_istream<T>::pos_type pos = stream.tellg(); | |
4946 stream.seekg(0, std::ios::end); | |
4947 std::streamoff length = stream.tellg() - pos; | |
4948 stream.seekg(pos); | |
4949 | |
4950 if (stream.fail() || pos < 0) return status_io_error; | |
4951 | |
4952 // guard against huge files | |
4953 size_t read_length = static_cast<size_t>(length); | |
4954 | |
4955 if (static_cast<std::streamsize>(read_length) != length || length < 0) return status_out_of_memory; | |
4956 | |
4957 size_t max_suffix_size = sizeof(char_t); | |
4958 | |
4959 // read stream data into memory (guard against stream exceptions with buffer holder) | |
4960 auto_deleter<void> buffer(xml_memory::allocate(read_length * sizeof(T) + max_suffix_size), xml_memory::deallocate); | |
4961 if (!buffer.data) return status_out_of_memory; | |
4962 | |
4963 stream.read(static_cast<T*>(buffer.data), static_cast<std::streamsize>(read_length)); | |
4964 | |
4965 // read may set failbit | eofbit in case gcount() is less than read_length (i.e. line ending conversion), so check for other I/O errors | |
4966 if (stream.bad() || (!stream.eof() && stream.fail())) return status_io_error; | |
4967 | |
4968 // return buffer | |
4969 size_t actual_length = static_cast<size_t>(stream.gcount()); | |
4970 assert(actual_length <= read_length); | |
4971 | |
4972 *out_buffer = buffer.release(); | |
4973 *out_size = actual_length * sizeof(T); | |
4974 | |
4975 return status_ok; | |
4976 } | |
4977 | |
4978 template <typename T> PUGI__FN xml_parse_result load_stream_impl(xml_document_struct* doc, std::basic_istream<T>& stream, unsigned int options, xml_encoding encoding, char_t** out_buffer) | |
4979 { | |
4980 void* buffer = 0; | |
4981 size_t size = 0; | |
4982 xml_parse_status status = status_ok; | |
4983 | |
4984 // if stream has an error bit set, bail out (otherwise tellg() can fail and we'll clear error bits) | |
4985 if (stream.fail()) return make_parse_result(status_io_error); | |
4986 | |
4987 // load stream to memory (using seek-based implementation if possible, since it's faster and takes less memory) | |
4988 if (stream.tellg() < 0) | |
4989 { | |
4990 stream.clear(); // clear error flags that could be set by a failing tellg | |
4991 status = load_stream_data_noseek(stream, &buffer, &size); | |
4992 } | |
4993 else | |
4994 status = load_stream_data_seek(stream, &buffer, &size); | |
4995 | |
4996 if (status != status_ok) return make_parse_result(status); | |
4997 | |
4998 xml_encoding real_encoding = get_buffer_encoding(encoding, buffer, size); | |
4999 | |
5000 return load_buffer_impl(doc, doc, buffer, zero_terminate_buffer(buffer, size, real_encoding), options, real_encoding, true, true, out_buffer); | |
5001 } | |
5002 #endif | |
5003 | |
5004 #if defined(PUGI__MSVC_CRT_VERSION) || defined(__BORLANDC__) || (defined(__MINGW32__) && (!defined(__STRICT_ANSI__) || defined(__MINGW64_VERSION_MAJOR))) | |
5005 PUGI__FN FILE* open_file_wide(const wchar_t* path, const wchar_t* mode) | |
5006 { | |
5007 #if defined(PUGI__MSVC_CRT_VERSION) && PUGI__MSVC_CRT_VERSION >= 1400 | |
5008 FILE* file = 0; | |
5009 return _wfopen_s(&file, path, mode) == 0 ? file : 0; | |
5010 #else | |
5011 return _wfopen(path, mode); | |
5012 #endif | |
5013 } | |
5014 #else | |
5015 PUGI__FN char* convert_path_heap(const wchar_t* str) | |
5016 { | |
5017 assert(str); | |
5018 | |
5019 // first pass: get length in utf8 characters | |
5020 size_t length = strlength_wide(str); | |
5021 size_t size = as_utf8_begin(str, length); | |
5022 | |
5023 // allocate resulting string | |
5024 char* result = static_cast<char*>(xml_memory::allocate(size + 1)); | |
5025 if (!result) return 0; | |
5026 | |
5027 // second pass: convert to utf8 | |
5028 as_utf8_end(result, size, str, length); | |
5029 | |
5030 // zero-terminate | |
5031 result[size] = 0; | |
5032 | |
5033 return result; | |
5034 } | |
5035 | |
5036 PUGI__FN FILE* open_file_wide(const wchar_t* path, const wchar_t* mode) | |
5037 { | |
5038 // there is no standard function to open wide paths, so our best bet is to try utf8 path | |
5039 char* path_utf8 = convert_path_heap(path); | |
5040 if (!path_utf8) return 0; | |
5041 | |
5042 // convert mode to ASCII (we mirror _wfopen interface) | |
5043 char mode_ascii[4] = {0}; | |
5044 for (size_t i = 0; mode[i]; ++i) mode_ascii[i] = static_cast<char>(mode[i]); | |
5045 | |
5046 // try to open the utf8 path | |
5047 FILE* result = fopen(path_utf8, mode_ascii); | |
5048 | |
5049 // free dummy buffer | |
5050 xml_memory::deallocate(path_utf8); | |
5051 | |
5052 return result; | |
5053 } | |
5054 #endif | |
5055 | |
5056 PUGI__FN FILE* open_file(const char* path, const char* mode) | |
5057 { | |
5058 #if defined(PUGI__MSVC_CRT_VERSION) && PUGI__MSVC_CRT_VERSION >= 1400 | |
5059 FILE* file = 0; | |
5060 return fopen_s(&file, path, mode) == 0 ? file : 0; | |
5061 #else | |
5062 return fopen(path, mode); | |
5063 #endif | |
5064 } | |
5065 | |
5066 PUGI__FN bool save_file_impl(const xml_document& doc, FILE* file, const char_t* indent, unsigned int flags, xml_encoding encoding) | |
5067 { | |
5068 if (!file) return false; | |
5069 | |
5070 xml_writer_file writer(file); | |
5071 doc.save(writer, indent, flags, encoding); | |
5072 | |
5073 return fflush(file) == 0 && ferror(file) == 0; | |
5074 } | |
5075 | |
5076 struct name_null_sentry | |
5077 { | |
5078 xml_node_struct* node; | |
5079 char_t* name; | |
5080 | |
5081 name_null_sentry(xml_node_struct* node_): node(node_), name(node_->name) | |
5082 { | |
5083 node->name = 0; | |
5084 } | |
5085 | |
5086 ~name_null_sentry() | |
5087 { | |
5088 node->name = name; | |
5089 } | |
5090 }; | |
5091 PUGI__NS_END | |
5092 | |
5093 namespace pugi | |
5094 { | |
5095 PUGI__FN xml_writer_file::xml_writer_file(void* file_): file(file_) | |
5096 { | |
5097 } | |
5098 | |
5099 PUGI__FN void xml_writer_file::write(const void* data, size_t size) | |
5100 { | |
5101 size_t result = fwrite(data, 1, size, static_cast<FILE*>(file)); | |
5102 (void)!result; // unfortunately we can't do proper error handling here | |
5103 } | |
5104 | |
5105 #ifndef PUGIXML_NO_STL | |
5106 PUGI__FN xml_writer_stream::xml_writer_stream(std::basic_ostream<char, std::char_traits<char> >& stream): narrow_stream(&stream), wide_stream(0) | |
5107 { | |
5108 } | |
5109 | |
5110 PUGI__FN xml_writer_stream::xml_writer_stream(std::basic_ostream<wchar_t, std::char_traits<wchar_t> >& stream): narrow_stream(0), wide_stream(&stream) | |
5111 { | |
5112 } | |
5113 | |
5114 PUGI__FN void xml_writer_stream::write(const void* data, size_t size) | |
5115 { | |
5116 if (narrow_stream) | |
5117 { | |
5118 assert(!wide_stream); | |
5119 narrow_stream->write(reinterpret_cast<const char*>(data), static_cast<std::streamsize>(size)); | |
5120 } | |
5121 else | |
5122 { | |
5123 assert(wide_stream); | |
5124 assert(size % sizeof(wchar_t) == 0); | |
5125 | |
5126 wide_stream->write(reinterpret_cast<const wchar_t*>(data), static_cast<std::streamsize>(size / sizeof(wchar_t))); | |
5127 } | |
5128 } | |
5129 #endif | |
5130 | |
5131 PUGI__FN xml_tree_walker::xml_tree_walker(): _depth(0) | |
5132 { | |
5133 } | |
5134 | |
5135 PUGI__FN xml_tree_walker::~xml_tree_walker() | |
5136 { | |
5137 } | |
5138 | |
5139 PUGI__FN int xml_tree_walker::depth() const | |
5140 { | |
5141 return _depth; | |
5142 } | |
5143 | |
5144 PUGI__FN bool xml_tree_walker::begin(xml_node&) | |
5145 { | |
5146 return true; | |
5147 } | |
5148 | |
5149 PUGI__FN bool xml_tree_walker::end(xml_node&) | |
5150 { | |
5151 return true; | |
5152 } | |
5153 | |
5154 PUGI__FN xml_attribute::xml_attribute(): _attr(0) | |
5155 { | |
5156 } | |
5157 | |
5158 PUGI__FN xml_attribute::xml_attribute(xml_attribute_struct* attr): _attr(attr) | |
5159 { | |
5160 } | |
5161 | |
5162 PUGI__FN static void unspecified_bool_xml_attribute(xml_attribute***) | |
5163 { | |
5164 } | |
5165 | |
5166 PUGI__FN xml_attribute::operator xml_attribute::unspecified_bool_type() const | |
5167 { | |
5168 return _attr ? unspecified_bool_xml_attribute : 0; | |
5169 } | |
5170 | |
5171 PUGI__FN bool xml_attribute::operator!() const | |
5172 { | |
5173 return !_attr; | |
5174 } | |
5175 | |
5176 PUGI__FN bool xml_attribute::operator==(const xml_attribute& r) const | |
5177 { | |
5178 return (_attr == r._attr); | |
5179 } | |
5180 | |
5181 PUGI__FN bool xml_attribute::operator!=(const xml_attribute& r) const | |
5182 { | |
5183 return (_attr != r._attr); | |
5184 } | |
5185 | |
5186 PUGI__FN bool xml_attribute::operator<(const xml_attribute& r) const | |
5187 { | |
5188 return (_attr < r._attr); | |
5189 } | |
5190 | |
5191 PUGI__FN bool xml_attribute::operator>(const xml_attribute& r) const | |
5192 { | |
5193 return (_attr > r._attr); | |
5194 } | |
5195 | |
5196 PUGI__FN bool xml_attribute::operator<=(const xml_attribute& r) const | |
5197 { | |
5198 return (_attr <= r._attr); | |
5199 } | |
5200 | |
5201 PUGI__FN bool xml_attribute::operator>=(const xml_attribute& r) const | |
5202 { | |
5203 return (_attr >= r._attr); | |
5204 } | |
5205 | |
5206 PUGI__FN xml_attribute xml_attribute::next_attribute() const | |
5207 { | |
5208 if (!_attr) return xml_attribute(); | |
5209 return xml_attribute(_attr->next_attribute); | |
5210 } | |
5211 | |
5212 PUGI__FN xml_attribute xml_attribute::previous_attribute() const | |
5213 { | |
5214 if (!_attr) return xml_attribute(); | |
5215 xml_attribute_struct* prev = _attr->prev_attribute_c; | |
5216 return prev->next_attribute ? xml_attribute(prev) : xml_attribute(); | |
5217 } | |
5218 | |
5219 PUGI__FN const char_t* xml_attribute::as_string(const char_t* def) const | |
5220 { | |
5221 if (!_attr) return def; | |
5222 const char_t* value = _attr->value; | |
5223 return value ? value : def; | |
5224 } | |
5225 | |
5226 PUGI__FN int xml_attribute::as_int(int def) const | |
5227 { | |
5228 if (!_attr) return def; | |
5229 const char_t* value = _attr->value; | |
5230 return value ? impl::get_value_int(value) : def; | |
5231 } | |
5232 | |
5233 PUGI__FN unsigned int xml_attribute::as_uint(unsigned int def) const | |
5234 { | |
5235 if (!_attr) return def; | |
5236 const char_t* value = _attr->value; | |
5237 return value ? impl::get_value_uint(value) : def; | |
5238 } | |
5239 | |
5240 PUGI__FN double xml_attribute::as_double(double def) const | |
5241 { | |
5242 if (!_attr) return def; | |
5243 const char_t* value = _attr->value; | |
5244 return value ? impl::get_value_double(value) : def; | |
5245 } | |
5246 | |
5247 PUGI__FN float xml_attribute::as_float(float def) const | |
5248 { | |
5249 if (!_attr) return def; | |
5250 const char_t* value = _attr->value; | |
5251 return value ? impl::get_value_float(value) : def; | |
5252 } | |
5253 | |
5254 PUGI__FN bool xml_attribute::as_bool(bool def) const | |
5255 { | |
5256 if (!_attr) return def; | |
5257 const char_t* value = _attr->value; | |
5258 return value ? impl::get_value_bool(value) : def; | |
5259 } | |
5260 | |
5261 #ifdef PUGIXML_HAS_LONG_LONG | |
5262 PUGI__FN long long xml_attribute::as_llong(long long def) const | |
5263 { | |
5264 if (!_attr) return def; | |
5265 const char_t* value = _attr->value; | |
5266 return value ? impl::get_value_llong(value) : def; | |
5267 } | |
5268 | |
5269 PUGI__FN unsigned long long xml_attribute::as_ullong(unsigned long long def) const | |
5270 { | |
5271 if (!_attr) return def; | |
5272 const char_t* value = _attr->value; | |
5273 return value ? impl::get_value_ullong(value) : def; | |
5274 } | |
5275 #endif | |
5276 | |
5277 PUGI__FN bool xml_attribute::empty() const | |
5278 { | |
5279 return !_attr; | |
5280 } | |
5281 | |
5282 PUGI__FN const char_t* xml_attribute::name() const | |
5283 { | |
5284 if (!_attr) return PUGIXML_TEXT(""); | |
5285 const char_t* name = _attr->name; | |
5286 return name ? name : PUGIXML_TEXT(""); | |
5287 } | |
5288 | |
5289 PUGI__FN const char_t* xml_attribute::value() const | |
5290 { | |
5291 if (!_attr) return PUGIXML_TEXT(""); | |
5292 const char_t* value = _attr->value; | |
5293 return value ? value : PUGIXML_TEXT(""); | |
5294 } | |
5295 | |
5296 PUGI__FN size_t xml_attribute::hash_value() const | |
5297 { | |
5298 return static_cast<size_t>(reinterpret_cast<uintptr_t>(_attr) / sizeof(xml_attribute_struct)); | |
5299 } | |
5300 | |
5301 PUGI__FN xml_attribute_struct* xml_attribute::internal_object() const | |
5302 { | |
5303 return _attr; | |
5304 } | |
5305 | |
5306 PUGI__FN xml_attribute& xml_attribute::operator=(const char_t* rhs) | |
5307 { | |
5308 set_value(rhs); | |
5309 return *this; | |
5310 } | |
5311 | |
5312 PUGI__FN xml_attribute& xml_attribute::operator=(int rhs) | |
5313 { | |
5314 set_value(rhs); | |
5315 return *this; | |
5316 } | |
5317 | |
5318 PUGI__FN xml_attribute& xml_attribute::operator=(unsigned int rhs) | |
5319 { | |
5320 set_value(rhs); | |
5321 return *this; | |
5322 } | |
5323 | |
5324 PUGI__FN xml_attribute& xml_attribute::operator=(long rhs) | |
5325 { | |
5326 set_value(rhs); | |
5327 return *this; | |
5328 } | |
5329 | |
5330 PUGI__FN xml_attribute& xml_attribute::operator=(unsigned long rhs) | |
5331 { | |
5332 set_value(rhs); | |
5333 return *this; | |
5334 } | |
5335 | |
5336 PUGI__FN xml_attribute& xml_attribute::operator=(double rhs) | |
5337 { | |
5338 set_value(rhs); | |
5339 return *this; | |
5340 } | |
5341 | |
5342 PUGI__FN xml_attribute& xml_attribute::operator=(float rhs) | |
5343 { | |
5344 set_value(rhs); | |
5345 return *this; | |
5346 } | |
5347 | |
5348 PUGI__FN xml_attribute& xml_attribute::operator=(bool rhs) | |
5349 { | |
5350 set_value(rhs); | |
5351 return *this; | |
5352 } | |
5353 | |
5354 #ifdef PUGIXML_HAS_LONG_LONG | |
5355 PUGI__FN xml_attribute& xml_attribute::operator=(long long rhs) | |
5356 { | |
5357 set_value(rhs); | |
5358 return *this; | |
5359 } | |
5360 | |
5361 PUGI__FN xml_attribute& xml_attribute::operator=(unsigned long long rhs) | |
5362 { | |
5363 set_value(rhs); | |
5364 return *this; | |
5365 } | |
5366 #endif | |
5367 | |
5368 PUGI__FN bool xml_attribute::set_name(const char_t* rhs) | |
5369 { | |
5370 if (!_attr) return false; | |
5371 | |
5372 return impl::strcpy_insitu(_attr->name, _attr->header, impl::xml_memory_page_name_allocated_mask, rhs, impl::strlength(rhs)); | |
5373 } | |
5374 | |
5375 PUGI__FN bool xml_attribute::set_value(const char_t* rhs, size_t sz) | |
5376 { | |
5377 if (!_attr) return false; | |
5378 | |
5379 return impl::strcpy_insitu(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, sz); | |
5380 } | |
5381 | |
5382 PUGI__FN bool xml_attribute::set_value(const char_t* rhs) | |
5383 { | |
5384 if (!_attr) return false; | |
5385 | |
5386 return impl::strcpy_insitu(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, impl::strlength(rhs)); | |
5387 } | |
5388 | |
5389 PUGI__FN bool xml_attribute::set_value(int rhs) | |
5390 { | |
5391 if (!_attr) return false; | |
5392 | |
5393 return impl::set_value_integer<unsigned int>(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, rhs < 0); | |
5394 } | |
5395 | |
5396 PUGI__FN bool xml_attribute::set_value(unsigned int rhs) | |
5397 { | |
5398 if (!_attr) return false; | |
5399 | |
5400 return impl::set_value_integer<unsigned int>(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, false); | |
5401 } | |
5402 | |
5403 PUGI__FN bool xml_attribute::set_value(long rhs) | |
5404 { | |
5405 if (!_attr) return false; | |
5406 | |
5407 return impl::set_value_integer<unsigned long>(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, rhs < 0); | |
5408 } | |
5409 | |
5410 PUGI__FN bool xml_attribute::set_value(unsigned long rhs) | |
5411 { | |
5412 if (!_attr) return false; | |
5413 | |
5414 return impl::set_value_integer<unsigned long>(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, false); | |
5415 } | |
5416 | |
5417 PUGI__FN bool xml_attribute::set_value(double rhs) | |
5418 { | |
5419 if (!_attr) return false; | |
5420 | |
5421 return impl::set_value_convert(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, default_double_precision); | |
5422 } | |
5423 | |
5424 PUGI__FN bool xml_attribute::set_value(double rhs, int precision) | |
5425 { | |
5426 if (!_attr) return false; | |
5427 | |
5428 return impl::set_value_convert(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, precision); | |
5429 } | |
5430 | |
5431 PUGI__FN bool xml_attribute::set_value(float rhs) | |
5432 { | |
5433 if (!_attr) return false; | |
5434 | |
5435 return impl::set_value_convert(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, default_float_precision); | |
5436 } | |
5437 | |
5438 PUGI__FN bool xml_attribute::set_value(float rhs, int precision) | |
5439 { | |
5440 if (!_attr) return false; | |
5441 | |
5442 return impl::set_value_convert(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, precision); | |
5443 } | |
5444 | |
5445 PUGI__FN bool xml_attribute::set_value(bool rhs) | |
5446 { | |
5447 if (!_attr) return false; | |
5448 | |
5449 return impl::set_value_bool(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs); | |
5450 } | |
5451 | |
5452 #ifdef PUGIXML_HAS_LONG_LONG | |
5453 PUGI__FN bool xml_attribute::set_value(long long rhs) | |
5454 { | |
5455 if (!_attr) return false; | |
5456 | |
5457 return impl::set_value_integer<unsigned long long>(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, rhs < 0); | |
5458 } | |
5459 | |
5460 PUGI__FN bool xml_attribute::set_value(unsigned long long rhs) | |
5461 { | |
5462 if (!_attr) return false; | |
5463 | |
5464 return impl::set_value_integer<unsigned long long>(_attr->value, _attr->header, impl::xml_memory_page_value_allocated_mask, rhs, false); | |
5465 } | |
5466 #endif | |
5467 | |
5468 #ifdef __BORLANDC__ | |
5469 PUGI__FN bool operator&&(const xml_attribute& lhs, bool rhs) | |
5470 { | |
5471 return (bool)lhs && rhs; | |
5472 } | |
5473 | |
5474 PUGI__FN bool operator||(const xml_attribute& lhs, bool rhs) | |
5475 { | |
5476 return (bool)lhs || rhs; | |
5477 } | |
5478 #endif | |
5479 | |
5480 PUGI__FN xml_node::xml_node(): _root(0) | |
5481 { | |
5482 } | |
5483 | |
5484 PUGI__FN xml_node::xml_node(xml_node_struct* p): _root(p) | |
5485 { | |
5486 } | |
5487 | |
5488 PUGI__FN static void unspecified_bool_xml_node(xml_node***) | |
5489 { | |
5490 } | |
5491 | |
5492 PUGI__FN xml_node::operator xml_node::unspecified_bool_type() const | |
5493 { | |
5494 return _root ? unspecified_bool_xml_node : 0; | |
5495 } | |
5496 | |
5497 PUGI__FN bool xml_node::operator!() const | |
5498 { | |
5499 return !_root; | |
5500 } | |
5501 | |
5502 PUGI__FN xml_node::iterator xml_node::begin() const | |
5503 { | |
5504 return iterator(_root ? _root->first_child + 0 : 0, _root); | |
5505 } | |
5506 | |
5507 PUGI__FN xml_node::iterator xml_node::end() const | |
5508 { | |
5509 return iterator(0, _root); | |
5510 } | |
5511 | |
5512 PUGI__FN xml_node::attribute_iterator xml_node::attributes_begin() const | |
5513 { | |
5514 return attribute_iterator(_root ? _root->first_attribute + 0 : 0, _root); | |
5515 } | |
5516 | |
5517 PUGI__FN xml_node::attribute_iterator xml_node::attributes_end() const | |
5518 { | |
5519 return attribute_iterator(0, _root); | |
5520 } | |
5521 | |
5522 PUGI__FN xml_object_range<xml_node_iterator> xml_node::children() const | |
5523 { | |
5524 return xml_object_range<xml_node_iterator>(begin(), end()); | |
5525 } | |
5526 | |
5527 PUGI__FN xml_object_range<xml_named_node_iterator> xml_node::children(const char_t* name_) const | |
5528 { | |
5529 return xml_object_range<xml_named_node_iterator>(xml_named_node_iterator(child(name_)._root, _root, name_), xml_named_node_iterator(0, _root, name_)); | |
5530 } | |
5531 | |
5532 PUGI__FN xml_object_range<xml_attribute_iterator> xml_node::attributes() const | |
5533 { | |
5534 return xml_object_range<xml_attribute_iterator>(attributes_begin(), attributes_end()); | |
5535 } | |
5536 | |
5537 PUGI__FN bool xml_node::operator==(const xml_node& r) const | |
5538 { | |
5539 return (_root == r._root); | |
5540 } | |
5541 | |
5542 PUGI__FN bool xml_node::operator!=(const xml_node& r) const | |
5543 { | |
5544 return (_root != r._root); | |
5545 } | |
5546 | |
5547 PUGI__FN bool xml_node::operator<(const xml_node& r) const | |
5548 { | |
5549 return (_root < r._root); | |
5550 } | |
5551 | |
5552 PUGI__FN bool xml_node::operator>(const xml_node& r) const | |
5553 { | |
5554 return (_root > r._root); | |
5555 } | |
5556 | |
5557 PUGI__FN bool xml_node::operator<=(const xml_node& r) const | |
5558 { | |
5559 return (_root <= r._root); | |
5560 } | |
5561 | |
5562 PUGI__FN bool xml_node::operator>=(const xml_node& r) const | |
5563 { | |
5564 return (_root >= r._root); | |
5565 } | |
5566 | |
5567 PUGI__FN bool xml_node::empty() const | |
5568 { | |
5569 return !_root; | |
5570 } | |
5571 | |
5572 PUGI__FN const char_t* xml_node::name() const | |
5573 { | |
5574 if (!_root) return PUGIXML_TEXT(""); | |
5575 const char_t* name = _root->name; | |
5576 return name ? name : PUGIXML_TEXT(""); | |
5577 } | |
5578 | |
5579 PUGI__FN xml_node_type xml_node::type() const | |
5580 { | |
5581 return _root ? PUGI__NODETYPE(_root) : node_null; | |
5582 } | |
5583 | |
5584 PUGI__FN const char_t* xml_node::value() const | |
5585 { | |
5586 if (!_root) return PUGIXML_TEXT(""); | |
5587 const char_t* value = _root->value; | |
5588 return value ? value : PUGIXML_TEXT(""); | |
5589 } | |
5590 | |
5591 PUGI__FN xml_node xml_node::child(const char_t* name_) const | |
5592 { | |
5593 if (!_root) return xml_node(); | |
5594 | |
5595 for (xml_node_struct* i = _root->first_child; i; i = i->next_sibling) | |
5596 { | |
5597 const char_t* iname = i->name; | |
5598 if (iname && impl::strequal(name_, iname)) | |
5599 return xml_node(i); | |
5600 } | |
5601 | |
5602 return xml_node(); | |
5603 } | |
5604 | |
5605 PUGI__FN xml_attribute xml_node::attribute(const char_t* name_) const | |
5606 { | |
5607 if (!_root) return xml_attribute(); | |
5608 | |
5609 for (xml_attribute_struct* i = _root->first_attribute; i; i = i->next_attribute) | |
5610 { | |
5611 const char_t* iname = i->name; | |
5612 if (iname && impl::strequal(name_, iname)) | |
5613 return xml_attribute(i); | |
5614 } | |
5615 | |
5616 return xml_attribute(); | |
5617 } | |
5618 | |
5619 PUGI__FN xml_node xml_node::next_sibling(const char_t* name_) const | |
5620 { | |
5621 if (!_root) return xml_node(); | |
5622 | |
5623 for (xml_node_struct* i = _root->next_sibling; i; i = i->next_sibling) | |
5624 { | |
5625 const char_t* iname = i->name; | |
5626 if (iname && impl::strequal(name_, iname)) | |
5627 return xml_node(i); | |
5628 } | |
5629 | |
5630 return xml_node(); | |
5631 } | |
5632 | |
5633 PUGI__FN xml_node xml_node::next_sibling() const | |
5634 { | |
5635 return _root ? xml_node(_root->next_sibling) : xml_node(); | |
5636 } | |
5637 | |
5638 PUGI__FN xml_node xml_node::previous_sibling(const char_t* name_) const | |
5639 { | |
5640 if (!_root) return xml_node(); | |
5641 | |
5642 for (xml_node_struct* i = _root->prev_sibling_c; i->next_sibling; i = i->prev_sibling_c) | |
5643 { | |
5644 const char_t* iname = i->name; | |
5645 if (iname && impl::strequal(name_, iname)) | |
5646 return xml_node(i); | |
5647 } | |
5648 | |
5649 return xml_node(); | |
5650 } | |
5651 | |
5652 PUGI__FN xml_attribute xml_node::attribute(const char_t* name_, xml_attribute& hint_) const | |
5653 { | |
5654 xml_attribute_struct* hint = hint_._attr; | |
5655 | |
5656 // if hint is not an attribute of node, behavior is not defined | |
5657 assert(!hint || (_root && impl::is_attribute_of(hint, _root))); | |
5658 | |
5659 if (!_root) return xml_attribute(); | |
5660 | |
5661 // optimistically search from hint up until the end | |
5662 for (xml_attribute_struct* i = hint; i; i = i->next_attribute) | |
5663 { | |
5664 const char_t* iname = i->name; | |
5665 if (iname && impl::strequal(name_, iname)) | |
5666 { | |
5667 // update hint to maximize efficiency of searching for consecutive attributes | |
5668 hint_._attr = i->next_attribute; | |
5669 | |
5670 return xml_attribute(i); | |
5671 } | |
5672 } | |
5673 | |
5674 // wrap around and search from the first attribute until the hint | |
5675 // 'j' null pointer check is technically redundant, but it prevents a crash in case the assertion above fails | |
5676 for (xml_attribute_struct* j = _root->first_attribute; j && j != hint; j = j->next_attribute) | |
5677 { | |
5678 const char_t* jname = j->name; | |
5679 if (jname && impl::strequal(name_, jname)) | |
5680 { | |
5681 // update hint to maximize efficiency of searching for consecutive attributes | |
5682 hint_._attr = j->next_attribute; | |
5683 | |
5684 return xml_attribute(j); | |
5685 } | |
5686 } | |
5687 | |
5688 return xml_attribute(); | |
5689 } | |
5690 | |
5691 PUGI__FN xml_node xml_node::previous_sibling() const | |
5692 { | |
5693 if (!_root) return xml_node(); | |
5694 xml_node_struct* prev = _root->prev_sibling_c; | |
5695 return prev->next_sibling ? xml_node(prev) : xml_node(); | |
5696 } | |
5697 | |
5698 PUGI__FN xml_node xml_node::parent() const | |
5699 { | |
5700 return _root ? xml_node(_root->parent) : xml_node(); | |
5701 } | |
5702 | |
5703 PUGI__FN xml_node xml_node::root() const | |
5704 { | |
5705 return _root ? xml_node(&impl::get_document(_root)) : xml_node(); | |
5706 } | |
5707 | |
5708 PUGI__FN xml_text xml_node::text() const | |
5709 { | |
5710 return xml_text(_root); | |
5711 } | |
5712 | |
5713 PUGI__FN const char_t* xml_node::child_value() const | |
5714 { | |
5715 if (!_root) return PUGIXML_TEXT(""); | |
5716 | |
5717 // element nodes can have value if parse_embed_pcdata was used | |
5718 if (PUGI__NODETYPE(_root) == node_element && _root->value) | |
5719 return _root->value; | |
5720 | |
5721 for (xml_node_struct* i = _root->first_child; i; i = i->next_sibling) | |
5722 { | |
5723 const char_t* ivalue = i->value; | |
5724 if (impl::is_text_node(i) && ivalue) | |
5725 return ivalue; | |
5726 } | |
5727 | |
5728 return PUGIXML_TEXT(""); | |
5729 } | |
5730 | |
5731 PUGI__FN const char_t* xml_node::child_value(const char_t* name_) const | |
5732 { | |
5733 return child(name_).child_value(); | |
5734 } | |
5735 | |
5736 PUGI__FN xml_attribute xml_node::first_attribute() const | |
5737 { | |
5738 if (!_root) return xml_attribute(); | |
5739 return xml_attribute(_root->first_attribute); | |
5740 } | |
5741 | |
5742 PUGI__FN xml_attribute xml_node::last_attribute() const | |
5743 { | |
5744 if (!_root) return xml_attribute(); | |
5745 xml_attribute_struct* first = _root->first_attribute; | |
5746 return first ? xml_attribute(first->prev_attribute_c) : xml_attribute(); | |
5747 } | |
5748 | |
5749 PUGI__FN xml_node xml_node::first_child() const | |
5750 { | |
5751 if (!_root) return xml_node(); | |
5752 return xml_node(_root->first_child); | |
5753 } | |
5754 | |
5755 PUGI__FN xml_node xml_node::last_child() const | |
5756 { | |
5757 if (!_root) return xml_node(); | |
5758 xml_node_struct* first = _root->first_child; | |
5759 return first ? xml_node(first->prev_sibling_c) : xml_node(); | |
5760 } | |
5761 | |
5762 PUGI__FN bool xml_node::set_name(const char_t* rhs) | |
5763 { | |
5764 xml_node_type type_ = _root ? PUGI__NODETYPE(_root) : node_null; | |
5765 | |
5766 if (type_ != node_element && type_ != node_pi && type_ != node_declaration) | |
5767 return false; | |
5768 | |
5769 return impl::strcpy_insitu(_root->name, _root->header, impl::xml_memory_page_name_allocated_mask, rhs, impl::strlength(rhs)); | |
5770 } | |
5771 | |
5772 PUGI__FN bool xml_node::set_value(const char_t* rhs, size_t sz) | |
5773 { | |
5774 xml_node_type type_ = _root ? PUGI__NODETYPE(_root) : node_null; | |
5775 | |
5776 if (type_ != node_pcdata && type_ != node_cdata && type_ != node_comment && type_ != node_pi && type_ != node_doctype) | |
5777 return false; | |
5778 | |
5779 return impl::strcpy_insitu(_root->value, _root->header, impl::xml_memory_page_value_allocated_mask, rhs, sz); | |
5780 } | |
5781 | |
5782 PUGI__FN bool xml_node::set_value(const char_t* rhs) | |
5783 { | |
5784 xml_node_type type_ = _root ? PUGI__NODETYPE(_root) : node_null; | |
5785 | |
5786 if (type_ != node_pcdata && type_ != node_cdata && type_ != node_comment && type_ != node_pi && type_ != node_doctype) | |
5787 return false; | |
5788 | |
5789 return impl::strcpy_insitu(_root->value, _root->header, impl::xml_memory_page_value_allocated_mask, rhs, impl::strlength(rhs)); | |
5790 } | |
5791 | |
5792 PUGI__FN xml_attribute xml_node::append_attribute(const char_t* name_) | |
5793 { | |
5794 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5795 | |
5796 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5797 if (!alloc.reserve()) return xml_attribute(); | |
5798 | |
5799 xml_attribute a(impl::allocate_attribute(alloc)); | |
5800 if (!a) return xml_attribute(); | |
5801 | |
5802 impl::append_attribute(a._attr, _root); | |
5803 | |
5804 a.set_name(name_); | |
5805 | |
5806 return a; | |
5807 } | |
5808 | |
5809 PUGI__FN xml_attribute xml_node::prepend_attribute(const char_t* name_) | |
5810 { | |
5811 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5812 | |
5813 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5814 if (!alloc.reserve()) return xml_attribute(); | |
5815 | |
5816 xml_attribute a(impl::allocate_attribute(alloc)); | |
5817 if (!a) return xml_attribute(); | |
5818 | |
5819 impl::prepend_attribute(a._attr, _root); | |
5820 | |
5821 a.set_name(name_); | |
5822 | |
5823 return a; | |
5824 } | |
5825 | |
5826 PUGI__FN xml_attribute xml_node::insert_attribute_after(const char_t* name_, const xml_attribute& attr) | |
5827 { | |
5828 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5829 if (!attr || !impl::is_attribute_of(attr._attr, _root)) return xml_attribute(); | |
5830 | |
5831 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5832 if (!alloc.reserve()) return xml_attribute(); | |
5833 | |
5834 xml_attribute a(impl::allocate_attribute(alloc)); | |
5835 if (!a) return xml_attribute(); | |
5836 | |
5837 impl::insert_attribute_after(a._attr, attr._attr, _root); | |
5838 | |
5839 a.set_name(name_); | |
5840 | |
5841 return a; | |
5842 } | |
5843 | |
5844 PUGI__FN xml_attribute xml_node::insert_attribute_before(const char_t* name_, const xml_attribute& attr) | |
5845 { | |
5846 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5847 if (!attr || !impl::is_attribute_of(attr._attr, _root)) return xml_attribute(); | |
5848 | |
5849 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5850 if (!alloc.reserve()) return xml_attribute(); | |
5851 | |
5852 xml_attribute a(impl::allocate_attribute(alloc)); | |
5853 if (!a) return xml_attribute(); | |
5854 | |
5855 impl::insert_attribute_before(a._attr, attr._attr, _root); | |
5856 | |
5857 a.set_name(name_); | |
5858 | |
5859 return a; | |
5860 } | |
5861 | |
5862 PUGI__FN xml_attribute xml_node::append_copy(const xml_attribute& proto) | |
5863 { | |
5864 if (!proto) return xml_attribute(); | |
5865 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5866 | |
5867 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5868 if (!alloc.reserve()) return xml_attribute(); | |
5869 | |
5870 xml_attribute a(impl::allocate_attribute(alloc)); | |
5871 if (!a) return xml_attribute(); | |
5872 | |
5873 impl::append_attribute(a._attr, _root); | |
5874 impl::node_copy_attribute(a._attr, proto._attr); | |
5875 | |
5876 return a; | |
5877 } | |
5878 | |
5879 PUGI__FN xml_attribute xml_node::prepend_copy(const xml_attribute& proto) | |
5880 { | |
5881 if (!proto) return xml_attribute(); | |
5882 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5883 | |
5884 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5885 if (!alloc.reserve()) return xml_attribute(); | |
5886 | |
5887 xml_attribute a(impl::allocate_attribute(alloc)); | |
5888 if (!a) return xml_attribute(); | |
5889 | |
5890 impl::prepend_attribute(a._attr, _root); | |
5891 impl::node_copy_attribute(a._attr, proto._attr); | |
5892 | |
5893 return a; | |
5894 } | |
5895 | |
5896 PUGI__FN xml_attribute xml_node::insert_copy_after(const xml_attribute& proto, const xml_attribute& attr) | |
5897 { | |
5898 if (!proto) return xml_attribute(); | |
5899 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5900 if (!attr || !impl::is_attribute_of(attr._attr, _root)) return xml_attribute(); | |
5901 | |
5902 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5903 if (!alloc.reserve()) return xml_attribute(); | |
5904 | |
5905 xml_attribute a(impl::allocate_attribute(alloc)); | |
5906 if (!a) return xml_attribute(); | |
5907 | |
5908 impl::insert_attribute_after(a._attr, attr._attr, _root); | |
5909 impl::node_copy_attribute(a._attr, proto._attr); | |
5910 | |
5911 return a; | |
5912 } | |
5913 | |
5914 PUGI__FN xml_attribute xml_node::insert_copy_before(const xml_attribute& proto, const xml_attribute& attr) | |
5915 { | |
5916 if (!proto) return xml_attribute(); | |
5917 if (!impl::allow_insert_attribute(type())) return xml_attribute(); | |
5918 if (!attr || !impl::is_attribute_of(attr._attr, _root)) return xml_attribute(); | |
5919 | |
5920 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5921 if (!alloc.reserve()) return xml_attribute(); | |
5922 | |
5923 xml_attribute a(impl::allocate_attribute(alloc)); | |
5924 if (!a) return xml_attribute(); | |
5925 | |
5926 impl::insert_attribute_before(a._attr, attr._attr, _root); | |
5927 impl::node_copy_attribute(a._attr, proto._attr); | |
5928 | |
5929 return a; | |
5930 } | |
5931 | |
5932 PUGI__FN xml_node xml_node::append_child(xml_node_type type_) | |
5933 { | |
5934 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
5935 | |
5936 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5937 if (!alloc.reserve()) return xml_node(); | |
5938 | |
5939 xml_node n(impl::allocate_node(alloc, type_)); | |
5940 if (!n) return xml_node(); | |
5941 | |
5942 impl::append_node(n._root, _root); | |
5943 | |
5944 if (type_ == node_declaration) n.set_name(PUGIXML_TEXT("xml")); | |
5945 | |
5946 return n; | |
5947 } | |
5948 | |
5949 PUGI__FN xml_node xml_node::prepend_child(xml_node_type type_) | |
5950 { | |
5951 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
5952 | |
5953 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5954 if (!alloc.reserve()) return xml_node(); | |
5955 | |
5956 xml_node n(impl::allocate_node(alloc, type_)); | |
5957 if (!n) return xml_node(); | |
5958 | |
5959 impl::prepend_node(n._root, _root); | |
5960 | |
5961 if (type_ == node_declaration) n.set_name(PUGIXML_TEXT("xml")); | |
5962 | |
5963 return n; | |
5964 } | |
5965 | |
5966 PUGI__FN xml_node xml_node::insert_child_before(xml_node_type type_, const xml_node& node) | |
5967 { | |
5968 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
5969 if (!node._root || node._root->parent != _root) return xml_node(); | |
5970 | |
5971 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5972 if (!alloc.reserve()) return xml_node(); | |
5973 | |
5974 xml_node n(impl::allocate_node(alloc, type_)); | |
5975 if (!n) return xml_node(); | |
5976 | |
5977 impl::insert_node_before(n._root, node._root); | |
5978 | |
5979 if (type_ == node_declaration) n.set_name(PUGIXML_TEXT("xml")); | |
5980 | |
5981 return n; | |
5982 } | |
5983 | |
5984 PUGI__FN xml_node xml_node::insert_child_after(xml_node_type type_, const xml_node& node) | |
5985 { | |
5986 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
5987 if (!node._root || node._root->parent != _root) return xml_node(); | |
5988 | |
5989 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
5990 if (!alloc.reserve()) return xml_node(); | |
5991 | |
5992 xml_node n(impl::allocate_node(alloc, type_)); | |
5993 if (!n) return xml_node(); | |
5994 | |
5995 impl::insert_node_after(n._root, node._root); | |
5996 | |
5997 if (type_ == node_declaration) n.set_name(PUGIXML_TEXT("xml")); | |
5998 | |
5999 return n; | |
6000 } | |
6001 | |
6002 PUGI__FN xml_node xml_node::append_child(const char_t* name_) | |
6003 { | |
6004 xml_node result = append_child(node_element); | |
6005 | |
6006 result.set_name(name_); | |
6007 | |
6008 return result; | |
6009 } | |
6010 | |
6011 PUGI__FN xml_node xml_node::prepend_child(const char_t* name_) | |
6012 { | |
6013 xml_node result = prepend_child(node_element); | |
6014 | |
6015 result.set_name(name_); | |
6016 | |
6017 return result; | |
6018 } | |
6019 | |
6020 PUGI__FN xml_node xml_node::insert_child_after(const char_t* name_, const xml_node& node) | |
6021 { | |
6022 xml_node result = insert_child_after(node_element, node); | |
6023 | |
6024 result.set_name(name_); | |
6025 | |
6026 return result; | |
6027 } | |
6028 | |
6029 PUGI__FN xml_node xml_node::insert_child_before(const char_t* name_, const xml_node& node) | |
6030 { | |
6031 xml_node result = insert_child_before(node_element, node); | |
6032 | |
6033 result.set_name(name_); | |
6034 | |
6035 return result; | |
6036 } | |
6037 | |
6038 PUGI__FN xml_node xml_node::append_copy(const xml_node& proto) | |
6039 { | |
6040 xml_node_type type_ = proto.type(); | |
6041 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
6042 | |
6043 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6044 if (!alloc.reserve()) return xml_node(); | |
6045 | |
6046 xml_node n(impl::allocate_node(alloc, type_)); | |
6047 if (!n) return xml_node(); | |
6048 | |
6049 impl::append_node(n._root, _root); | |
6050 impl::node_copy_tree(n._root, proto._root); | |
6051 | |
6052 return n; | |
6053 } | |
6054 | |
6055 PUGI__FN xml_node xml_node::prepend_copy(const xml_node& proto) | |
6056 { | |
6057 xml_node_type type_ = proto.type(); | |
6058 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
6059 | |
6060 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6061 if (!alloc.reserve()) return xml_node(); | |
6062 | |
6063 xml_node n(impl::allocate_node(alloc, type_)); | |
6064 if (!n) return xml_node(); | |
6065 | |
6066 impl::prepend_node(n._root, _root); | |
6067 impl::node_copy_tree(n._root, proto._root); | |
6068 | |
6069 return n; | |
6070 } | |
6071 | |
6072 PUGI__FN xml_node xml_node::insert_copy_after(const xml_node& proto, const xml_node& node) | |
6073 { | |
6074 xml_node_type type_ = proto.type(); | |
6075 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
6076 if (!node._root || node._root->parent != _root) return xml_node(); | |
6077 | |
6078 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6079 if (!alloc.reserve()) return xml_node(); | |
6080 | |
6081 xml_node n(impl::allocate_node(alloc, type_)); | |
6082 if (!n) return xml_node(); | |
6083 | |
6084 impl::insert_node_after(n._root, node._root); | |
6085 impl::node_copy_tree(n._root, proto._root); | |
6086 | |
6087 return n; | |
6088 } | |
6089 | |
6090 PUGI__FN xml_node xml_node::insert_copy_before(const xml_node& proto, const xml_node& node) | |
6091 { | |
6092 xml_node_type type_ = proto.type(); | |
6093 if (!impl::allow_insert_child(type(), type_)) return xml_node(); | |
6094 if (!node._root || node._root->parent != _root) return xml_node(); | |
6095 | |
6096 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6097 if (!alloc.reserve()) return xml_node(); | |
6098 | |
6099 xml_node n(impl::allocate_node(alloc, type_)); | |
6100 if (!n) return xml_node(); | |
6101 | |
6102 impl::insert_node_before(n._root, node._root); | |
6103 impl::node_copy_tree(n._root, proto._root); | |
6104 | |
6105 return n; | |
6106 } | |
6107 | |
6108 PUGI__FN xml_node xml_node::append_move(const xml_node& moved) | |
6109 { | |
6110 if (!impl::allow_move(*this, moved)) return xml_node(); | |
6111 | |
6112 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6113 if (!alloc.reserve()) return xml_node(); | |
6114 | |
6115 // disable document_buffer_order optimization since moving nodes around changes document order without changing buffer pointers | |
6116 impl::get_document(_root).header |= impl::xml_memory_page_contents_shared_mask; | |
6117 | |
6118 impl::remove_node(moved._root); | |
6119 impl::append_node(moved._root, _root); | |
6120 | |
6121 return moved; | |
6122 } | |
6123 | |
6124 PUGI__FN xml_node xml_node::prepend_move(const xml_node& moved) | |
6125 { | |
6126 if (!impl::allow_move(*this, moved)) return xml_node(); | |
6127 | |
6128 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6129 if (!alloc.reserve()) return xml_node(); | |
6130 | |
6131 // disable document_buffer_order optimization since moving nodes around changes document order without changing buffer pointers | |
6132 impl::get_document(_root).header |= impl::xml_memory_page_contents_shared_mask; | |
6133 | |
6134 impl::remove_node(moved._root); | |
6135 impl::prepend_node(moved._root, _root); | |
6136 | |
6137 return moved; | |
6138 } | |
6139 | |
6140 PUGI__FN xml_node xml_node::insert_move_after(const xml_node& moved, const xml_node& node) | |
6141 { | |
6142 if (!impl::allow_move(*this, moved)) return xml_node(); | |
6143 if (!node._root || node._root->parent != _root) return xml_node(); | |
6144 if (moved._root == node._root) return xml_node(); | |
6145 | |
6146 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6147 if (!alloc.reserve()) return xml_node(); | |
6148 | |
6149 // disable document_buffer_order optimization since moving nodes around changes document order without changing buffer pointers | |
6150 impl::get_document(_root).header |= impl::xml_memory_page_contents_shared_mask; | |
6151 | |
6152 impl::remove_node(moved._root); | |
6153 impl::insert_node_after(moved._root, node._root); | |
6154 | |
6155 return moved; | |
6156 } | |
6157 | |
6158 PUGI__FN xml_node xml_node::insert_move_before(const xml_node& moved, const xml_node& node) | |
6159 { | |
6160 if (!impl::allow_move(*this, moved)) return xml_node(); | |
6161 if (!node._root || node._root->parent != _root) return xml_node(); | |
6162 if (moved._root == node._root) return xml_node(); | |
6163 | |
6164 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6165 if (!alloc.reserve()) return xml_node(); | |
6166 | |
6167 // disable document_buffer_order optimization since moving nodes around changes document order without changing buffer pointers | |
6168 impl::get_document(_root).header |= impl::xml_memory_page_contents_shared_mask; | |
6169 | |
6170 impl::remove_node(moved._root); | |
6171 impl::insert_node_before(moved._root, node._root); | |
6172 | |
6173 return moved; | |
6174 } | |
6175 | |
6176 PUGI__FN bool xml_node::remove_attribute(const char_t* name_) | |
6177 { | |
6178 return remove_attribute(attribute(name_)); | |
6179 } | |
6180 | |
6181 PUGI__FN bool xml_node::remove_attribute(const xml_attribute& a) | |
6182 { | |
6183 if (!_root || !a._attr) return false; | |
6184 if (!impl::is_attribute_of(a._attr, _root)) return false; | |
6185 | |
6186 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6187 if (!alloc.reserve()) return false; | |
6188 | |
6189 impl::remove_attribute(a._attr, _root); | |
6190 impl::destroy_attribute(a._attr, alloc); | |
6191 | |
6192 return true; | |
6193 } | |
6194 | |
6195 PUGI__FN bool xml_node::remove_attributes() | |
6196 { | |
6197 if (!_root) return false; | |
6198 | |
6199 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6200 if (!alloc.reserve()) return false; | |
6201 | |
6202 for (xml_attribute_struct* attr = _root->first_attribute; attr; ) | |
6203 { | |
6204 xml_attribute_struct* next = attr->next_attribute; | |
6205 | |
6206 impl::destroy_attribute(attr, alloc); | |
6207 | |
6208 attr = next; | |
6209 } | |
6210 | |
6211 _root->first_attribute = 0; | |
6212 | |
6213 return true; | |
6214 } | |
6215 | |
6216 PUGI__FN bool xml_node::remove_child(const char_t* name_) | |
6217 { | |
6218 return remove_child(child(name_)); | |
6219 } | |
6220 | |
6221 PUGI__FN bool xml_node::remove_child(const xml_node& n) | |
6222 { | |
6223 if (!_root || !n._root || n._root->parent != _root) return false; | |
6224 | |
6225 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6226 if (!alloc.reserve()) return false; | |
6227 | |
6228 impl::remove_node(n._root); | |
6229 impl::destroy_node(n._root, alloc); | |
6230 | |
6231 return true; | |
6232 } | |
6233 | |
6234 PUGI__FN bool xml_node::remove_children() | |
6235 { | |
6236 if (!_root) return false; | |
6237 | |
6238 impl::xml_allocator& alloc = impl::get_allocator(_root); | |
6239 if (!alloc.reserve()) return false; | |
6240 | |
6241 for (xml_node_struct* cur = _root->first_child; cur; ) | |
6242 { | |
6243 xml_node_struct* next = cur->next_sibling; | |
6244 | |
6245 impl::destroy_node(cur, alloc); | |
6246 | |
6247 cur = next; | |
6248 } | |
6249 | |
6250 _root->first_child = 0; | |
6251 | |
6252 return true; | |
6253 } | |
6254 | |
6255 PUGI__FN xml_parse_result xml_node::append_buffer(const void* contents, size_t size, unsigned int options, xml_encoding encoding) | |
6256 { | |
6257 // append_buffer is only valid for elements/documents | |
6258 if (!impl::allow_insert_child(type(), node_element)) return impl::make_parse_result(status_append_invalid_root); | |
6259 | |
6260 // get document node | |
6261 impl::xml_document_struct* doc = &impl::get_document(_root); | |
6262 | |
6263 // disable document_buffer_order optimization since in a document with multiple buffers comparing buffer pointers does not make sense | |
6264 doc->header |= impl::xml_memory_page_contents_shared_mask; | |
6265 | |
6266 // get extra buffer element (we'll store the document fragment buffer there so that we can deallocate it later) | |
6267 impl::xml_memory_page* page = 0; | |
6268 impl::xml_extra_buffer* extra = static_cast<impl::xml_extra_buffer*>(doc->allocate_memory(sizeof(impl::xml_extra_buffer) + sizeof(void*), page)); | |
6269 (void)page; | |
6270 | |
6271 if (!extra) return impl::make_parse_result(status_out_of_memory); | |
6272 | |
6273 #ifdef PUGIXML_COMPACT | |
6274 // align the memory block to a pointer boundary; this is required for compact mode where memory allocations are only 4b aligned | |
6275 // note that this requires up to sizeof(void*)-1 additional memory, which the allocation above takes into account | |
6276 extra = reinterpret_cast<impl::xml_extra_buffer*>((reinterpret_cast<uintptr_t>(extra) + (sizeof(void*) - 1)) & ~(sizeof(void*) - 1)); | |
6277 #endif | |
6278 | |
6279 // add extra buffer to the list | |
6280 extra->buffer = 0; | |
6281 extra->next = doc->extra_buffers; | |
6282 doc->extra_buffers = extra; | |
6283 | |
6284 // name of the root has to be NULL before parsing - otherwise closing node mismatches will not be detected at the top level | |
6285 impl::name_null_sentry sentry(_root); | |
6286 | |
6287 return impl::load_buffer_impl(doc, _root, const_cast<void*>(contents), size, options, encoding, false, false, &extra->buffer); | |
6288 } | |
6289 | |
6290 PUGI__FN xml_node xml_node::find_child_by_attribute(const char_t* name_, const char_t* attr_name, const char_t* attr_value) const | |
6291 { | |
6292 if (!_root) return xml_node(); | |
6293 | |
6294 for (xml_node_struct* i = _root->first_child; i; i = i->next_sibling) | |
6295 { | |
6296 const char_t* iname = i->name; | |
6297 if (iname && impl::strequal(name_, iname)) | |
6298 { | |
6299 for (xml_attribute_struct* a = i->first_attribute; a; a = a->next_attribute) | |
6300 { | |
6301 const char_t* aname = a->name; | |
6302 if (aname && impl::strequal(attr_name, aname)) | |
6303 { | |
6304 const char_t* avalue = a->value; | |
6305 if (impl::strequal(attr_value, avalue ? avalue : PUGIXML_TEXT(""))) | |
6306 return xml_node(i); | |
6307 } | |
6308 } | |
6309 } | |
6310 } | |
6311 | |
6312 return xml_node(); | |
6313 } | |
6314 | |
6315 PUGI__FN xml_node xml_node::find_child_by_attribute(const char_t* attr_name, const char_t* attr_value) const | |
6316 { | |
6317 if (!_root) return xml_node(); | |
6318 | |
6319 for (xml_node_struct* i = _root->first_child; i; i = i->next_sibling) | |
6320 for (xml_attribute_struct* a = i->first_attribute; a; a = a->next_attribute) | |
6321 { | |
6322 const char_t* aname = a->name; | |
6323 if (aname && impl::strequal(attr_name, aname)) | |
6324 { | |
6325 const char_t* avalue = a->value; | |
6326 if (impl::strequal(attr_value, avalue ? avalue : PUGIXML_TEXT(""))) | |
6327 return xml_node(i); | |
6328 } | |
6329 } | |
6330 | |
6331 return xml_node(); | |
6332 } | |
6333 | |
6334 #ifndef PUGIXML_NO_STL | |
6335 PUGI__FN string_t xml_node::path(char_t delimiter) const | |
6336 { | |
6337 if (!_root) return string_t(); | |
6338 | |
6339 size_t offset = 0; | |
6340 | |
6341 for (xml_node_struct* i = _root; i; i = i->parent) | |
6342 { | |
6343 const char_t* iname = i->name; | |
6344 offset += (i != _root); | |
6345 offset += iname ? impl::strlength(iname) : 0; | |
6346 } | |
6347 | |
6348 string_t result; | |
6349 result.resize(offset); | |
6350 | |
6351 for (xml_node_struct* j = _root; j; j = j->parent) | |
6352 { | |
6353 if (j != _root) | |
6354 result[--offset] = delimiter; | |
6355 | |
6356 const char_t* jname = j->name; | |
6357 if (jname) | |
6358 { | |
6359 size_t length = impl::strlength(jname); | |
6360 | |
6361 offset -= length; | |
6362 memcpy(&result[offset], jname, length * sizeof(char_t)); | |
6363 } | |
6364 } | |
6365 | |
6366 assert(offset == 0); | |
6367 | |
6368 return result; | |
6369 } | |
6370 #endif | |
6371 | |
6372 PUGI__FN xml_node xml_node::first_element_by_path(const char_t* path_, char_t delimiter) const | |
6373 { | |
6374 xml_node context = path_[0] == delimiter ? root() : *this; | |
6375 | |
6376 if (!context._root) return xml_node(); | |
6377 | |
6378 const char_t* path_segment = path_; | |
6379 | |
6380 while (*path_segment == delimiter) ++path_segment; | |
6381 | |
6382 const char_t* path_segment_end = path_segment; | |
6383 | |
6384 while (*path_segment_end && *path_segment_end != delimiter) ++path_segment_end; | |
6385 | |
6386 if (path_segment == path_segment_end) return context; | |
6387 | |
6388 const char_t* next_segment = path_segment_end; | |
6389 | |
6390 while (*next_segment == delimiter) ++next_segment; | |
6391 | |
6392 if (*path_segment == '.' && path_segment + 1 == path_segment_end) | |
6393 return context.first_element_by_path(next_segment, delimiter); | |
6394 else if (*path_segment == '.' && *(path_segment+1) == '.' && path_segment + 2 == path_segment_end) | |
6395 return context.parent().first_element_by_path(next_segment, delimiter); | |
6396 else | |
6397 { | |
6398 for (xml_node_struct* j = context._root->first_child; j; j = j->next_sibling) | |
6399 { | |
6400 const char_t* jname = j->name; | |
6401 if (jname && impl::strequalrange(jname, path_segment, static_cast<size_t>(path_segment_end - path_segment))) | |
6402 { | |
6403 xml_node subsearch = xml_node(j).first_element_by_path(next_segment, delimiter); | |
6404 | |
6405 if (subsearch) return subsearch; | |
6406 } | |
6407 } | |
6408 | |
6409 return xml_node(); | |
6410 } | |
6411 } | |
6412 | |
6413 PUGI__FN bool xml_node::traverse(xml_tree_walker& walker) | |
6414 { | |
6415 walker._depth = -1; | |
6416 | |
6417 xml_node arg_begin(_root); | |
6418 if (!walker.begin(arg_begin)) return false; | |
6419 | |
6420 xml_node_struct* cur = _root ? _root->first_child + 0 : 0; | |
6421 | |
6422 if (cur) | |
6423 { | |
6424 ++walker._depth; | |
6425 | |
6426 do | |
6427 { | |
6428 xml_node arg_for_each(cur); | |
6429 if (!walker.for_each(arg_for_each)) | |
6430 return false; | |
6431 | |
6432 if (cur->first_child) | |
6433 { | |
6434 ++walker._depth; | |
6435 cur = cur->first_child; | |
6436 } | |
6437 else if (cur->next_sibling) | |
6438 cur = cur->next_sibling; | |
6439 else | |
6440 { | |
6441 while (!cur->next_sibling && cur != _root && cur->parent) | |
6442 { | |
6443 --walker._depth; | |
6444 cur = cur->parent; | |
6445 } | |
6446 | |
6447 if (cur != _root) | |
6448 cur = cur->next_sibling; | |
6449 } | |
6450 } | |
6451 while (cur && cur != _root); | |
6452 } | |
6453 | |
6454 assert(walker._depth == -1); | |
6455 | |
6456 xml_node arg_end(_root); | |
6457 return walker.end(arg_end); | |
6458 } | |
6459 | |
6460 PUGI__FN size_t xml_node::hash_value() const | |
6461 { | |
6462 return static_cast<size_t>(reinterpret_cast<uintptr_t>(_root) / sizeof(xml_node_struct)); | |
6463 } | |
6464 | |
6465 PUGI__FN xml_node_struct* xml_node::internal_object() const | |
6466 { | |
6467 return _root; | |
6468 } | |
6469 | |
6470 PUGI__FN void xml_node::print(xml_writer& writer, const char_t* indent, unsigned int flags, xml_encoding encoding, unsigned int depth) const | |
6471 { | |
6472 if (!_root) return; | |
6473 | |
6474 impl::xml_buffered_writer buffered_writer(writer, encoding); | |
6475 | |
6476 impl::node_output(buffered_writer, _root, indent, flags, depth); | |
6477 | |
6478 buffered_writer.flush(); | |
6479 } | |
6480 | |
6481 #ifndef PUGIXML_NO_STL | |
6482 PUGI__FN void xml_node::print(std::basic_ostream<char, std::char_traits<char> >& stream, const char_t* indent, unsigned int flags, xml_encoding encoding, unsigned int depth) const | |
6483 { | |
6484 xml_writer_stream writer(stream); | |
6485 | |
6486 print(writer, indent, flags, encoding, depth); | |
6487 } | |
6488 | |
6489 PUGI__FN void xml_node::print(std::basic_ostream<wchar_t, std::char_traits<wchar_t> >& stream, const char_t* indent, unsigned int flags, unsigned int depth) const | |
6490 { | |
6491 xml_writer_stream writer(stream); | |
6492 | |
6493 print(writer, indent, flags, encoding_wchar, depth); | |
6494 } | |
6495 #endif | |
6496 | |
6497 PUGI__FN ptrdiff_t xml_node::offset_debug() const | |
6498 { | |
6499 if (!_root) return -1; | |
6500 | |
6501 impl::xml_document_struct& doc = impl::get_document(_root); | |
6502 | |
6503 // we can determine the offset reliably only if there is exactly once parse buffer | |
6504 if (!doc.buffer || doc.extra_buffers) return -1; | |
6505 | |
6506 switch (type()) | |
6507 { | |
6508 case node_document: | |
6509 return 0; | |
6510 | |
6511 case node_element: | |
6512 case node_declaration: | |
6513 case node_pi: | |
6514 return _root->name && (_root->header & impl::xml_memory_page_name_allocated_or_shared_mask) == 0 ? _root->name - doc.buffer : -1; | |
6515 | |
6516 case node_pcdata: | |
6517 case node_cdata: | |
6518 case node_comment: | |
6519 case node_doctype: | |
6520 return _root->value && (_root->header & impl::xml_memory_page_value_allocated_or_shared_mask) == 0 ? _root->value - doc.buffer : -1; | |
6521 | |
6522 default: | |
6523 assert(false && "Invalid node type"); // unreachable | |
6524 return -1; | |
6525 } | |
6526 } | |
6527 | |
6528 #ifdef __BORLANDC__ | |
6529 PUGI__FN bool operator&&(const xml_node& lhs, bool rhs) | |
6530 { | |
6531 return (bool)lhs && rhs; | |
6532 } | |
6533 | |
6534 PUGI__FN bool operator||(const xml_node& lhs, bool rhs) | |
6535 { | |
6536 return (bool)lhs || rhs; | |
6537 } | |
6538 #endif | |
6539 | |
6540 PUGI__FN xml_text::xml_text(xml_node_struct* root): _root(root) | |
6541 { | |
6542 } | |
6543 | |
6544 PUGI__FN xml_node_struct* xml_text::_data() const | |
6545 { | |
6546 if (!_root || impl::is_text_node(_root)) return _root; | |
6547 | |
6548 // element nodes can have value if parse_embed_pcdata was used | |
6549 if (PUGI__NODETYPE(_root) == node_element && _root->value) | |
6550 return _root; | |
6551 | |
6552 for (xml_node_struct* node = _root->first_child; node; node = node->next_sibling) | |
6553 if (impl::is_text_node(node)) | |
6554 return node; | |
6555 | |
6556 return 0; | |
6557 } | |
6558 | |
6559 PUGI__FN xml_node_struct* xml_text::_data_new() | |
6560 { | |
6561 xml_node_struct* d = _data(); | |
6562 if (d) return d; | |
6563 | |
6564 return xml_node(_root).append_child(node_pcdata).internal_object(); | |
6565 } | |
6566 | |
6567 PUGI__FN xml_text::xml_text(): _root(0) | |
6568 { | |
6569 } | |
6570 | |
6571 PUGI__FN static void unspecified_bool_xml_text(xml_text***) | |
6572 { | |
6573 } | |
6574 | |
6575 PUGI__FN xml_text::operator xml_text::unspecified_bool_type() const | |
6576 { | |
6577 return _data() ? unspecified_bool_xml_text : 0; | |
6578 } | |
6579 | |
6580 PUGI__FN bool xml_text::operator!() const | |
6581 { | |
6582 return !_data(); | |
6583 } | |
6584 | |
6585 PUGI__FN bool xml_text::empty() const | |
6586 { | |
6587 return _data() == 0; | |
6588 } | |
6589 | |
6590 PUGI__FN const char_t* xml_text::get() const | |
6591 { | |
6592 xml_node_struct* d = _data(); | |
6593 if (!d) return PUGIXML_TEXT(""); | |
6594 const char_t* value = d->value; | |
6595 return value ? value : PUGIXML_TEXT(""); | |
6596 } | |
6597 | |
6598 PUGI__FN const char_t* xml_text::as_string(const char_t* def) const | |
6599 { | |
6600 xml_node_struct* d = _data(); | |
6601 if (!d) return def; | |
6602 const char_t* value = d->value; | |
6603 return value ? value : def; | |
6604 } | |
6605 | |
6606 PUGI__FN int xml_text::as_int(int def) const | |
6607 { | |
6608 xml_node_struct* d = _data(); | |
6609 if (!d) return def; | |
6610 const char_t* value = d->value; | |
6611 return value ? impl::get_value_int(value) : def; | |
6612 } | |
6613 | |
6614 PUGI__FN unsigned int xml_text::as_uint(unsigned int def) const | |
6615 { | |
6616 xml_node_struct* d = _data(); | |
6617 if (!d) return def; | |
6618 const char_t* value = d->value; | |
6619 return value ? impl::get_value_uint(value) : def; | |
6620 } | |
6621 | |
6622 PUGI__FN double xml_text::as_double(double def) const | |
6623 { | |
6624 xml_node_struct* d = _data(); | |
6625 if (!d) return def; | |
6626 const char_t* value = d->value; | |
6627 return value ? impl::get_value_double(value) : def; | |
6628 } | |
6629 | |
6630 PUGI__FN float xml_text::as_float(float def) const | |
6631 { | |
6632 xml_node_struct* d = _data(); | |
6633 if (!d) return def; | |
6634 const char_t* value = d->value; | |
6635 return value ? impl::get_value_float(value) : def; | |
6636 } | |
6637 | |
6638 PUGI__FN bool xml_text::as_bool(bool def) const | |
6639 { | |
6640 xml_node_struct* d = _data(); | |
6641 if (!d) return def; | |
6642 const char_t* value = d->value; | |
6643 return value ? impl::get_value_bool(value) : def; | |
6644 } | |
6645 | |
6646 #ifdef PUGIXML_HAS_LONG_LONG | |
6647 PUGI__FN long long xml_text::as_llong(long long def) const | |
6648 { | |
6649 xml_node_struct* d = _data(); | |
6650 if (!d) return def; | |
6651 const char_t* value = d->value; | |
6652 return value ? impl::get_value_llong(value) : def; | |
6653 } | |
6654 | |
6655 PUGI__FN unsigned long long xml_text::as_ullong(unsigned long long def) const | |
6656 { | |
6657 xml_node_struct* d = _data(); | |
6658 if (!d) return def; | |
6659 const char_t* value = d->value; | |
6660 return value ? impl::get_value_ullong(value) : def; | |
6661 } | |
6662 #endif | |
6663 | |
6664 PUGI__FN bool xml_text::set(const char_t* rhs, size_t sz) | |
6665 { | |
6666 xml_node_struct* dn = _data_new(); | |
6667 | |
6668 return dn ? impl::strcpy_insitu(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, sz) : false; | |
6669 } | |
6670 | |
6671 PUGI__FN bool xml_text::set(const char_t* rhs) | |
6672 { | |
6673 xml_node_struct* dn = _data_new(); | |
6674 | |
6675 return dn ? impl::strcpy_insitu(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, impl::strlength(rhs)) : false; | |
6676 } | |
6677 | |
6678 PUGI__FN bool xml_text::set(int rhs) | |
6679 { | |
6680 xml_node_struct* dn = _data_new(); | |
6681 | |
6682 return dn ? impl::set_value_integer<unsigned int>(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, rhs < 0) : false; | |
6683 } | |
6684 | |
6685 PUGI__FN bool xml_text::set(unsigned int rhs) | |
6686 { | |
6687 xml_node_struct* dn = _data_new(); | |
6688 | |
6689 return dn ? impl::set_value_integer<unsigned int>(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, false) : false; | |
6690 } | |
6691 | |
6692 PUGI__FN bool xml_text::set(long rhs) | |
6693 { | |
6694 xml_node_struct* dn = _data_new(); | |
6695 | |
6696 return dn ? impl::set_value_integer<unsigned long>(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, rhs < 0) : false; | |
6697 } | |
6698 | |
6699 PUGI__FN bool xml_text::set(unsigned long rhs) | |
6700 { | |
6701 xml_node_struct* dn = _data_new(); | |
6702 | |
6703 return dn ? impl::set_value_integer<unsigned long>(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, false) : false; | |
6704 } | |
6705 | |
6706 PUGI__FN bool xml_text::set(float rhs) | |
6707 { | |
6708 xml_node_struct* dn = _data_new(); | |
6709 | |
6710 return dn ? impl::set_value_convert(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, default_float_precision) : false; | |
6711 } | |
6712 | |
6713 PUGI__FN bool xml_text::set(float rhs, int precision) | |
6714 { | |
6715 xml_node_struct* dn = _data_new(); | |
6716 | |
6717 return dn ? impl::set_value_convert(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, precision) : false; | |
6718 } | |
6719 | |
6720 PUGI__FN bool xml_text::set(double rhs) | |
6721 { | |
6722 xml_node_struct* dn = _data_new(); | |
6723 | |
6724 return dn ? impl::set_value_convert(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, default_double_precision) : false; | |
6725 } | |
6726 | |
6727 PUGI__FN bool xml_text::set(double rhs, int precision) | |
6728 { | |
6729 xml_node_struct* dn = _data_new(); | |
6730 | |
6731 return dn ? impl::set_value_convert(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, precision) : false; | |
6732 } | |
6733 | |
6734 PUGI__FN bool xml_text::set(bool rhs) | |
6735 { | |
6736 xml_node_struct* dn = _data_new(); | |
6737 | |
6738 return dn ? impl::set_value_bool(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs) : false; | |
6739 } | |
6740 | |
6741 #ifdef PUGIXML_HAS_LONG_LONG | |
6742 PUGI__FN bool xml_text::set(long long rhs) | |
6743 { | |
6744 xml_node_struct* dn = _data_new(); | |
6745 | |
6746 return dn ? impl::set_value_integer<unsigned long long>(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, rhs < 0) : false; | |
6747 } | |
6748 | |
6749 PUGI__FN bool xml_text::set(unsigned long long rhs) | |
6750 { | |
6751 xml_node_struct* dn = _data_new(); | |
6752 | |
6753 return dn ? impl::set_value_integer<unsigned long long>(dn->value, dn->header, impl::xml_memory_page_value_allocated_mask, rhs, false) : false; | |
6754 } | |
6755 #endif | |
6756 | |
6757 PUGI__FN xml_text& xml_text::operator=(const char_t* rhs) | |
6758 { | |
6759 set(rhs); | |
6760 return *this; | |
6761 } | |
6762 | |
6763 PUGI__FN xml_text& xml_text::operator=(int rhs) | |
6764 { | |
6765 set(rhs); | |
6766 return *this; | |
6767 } | |
6768 | |
6769 PUGI__FN xml_text& xml_text::operator=(unsigned int rhs) | |
6770 { | |
6771 set(rhs); | |
6772 return *this; | |
6773 } | |
6774 | |
6775 PUGI__FN xml_text& xml_text::operator=(long rhs) | |
6776 { | |
6777 set(rhs); | |
6778 return *this; | |
6779 } | |
6780 | |
6781 PUGI__FN xml_text& xml_text::operator=(unsigned long rhs) | |
6782 { | |
6783 set(rhs); | |
6784 return *this; | |
6785 } | |
6786 | |
6787 PUGI__FN xml_text& xml_text::operator=(double rhs) | |
6788 { | |
6789 set(rhs); | |
6790 return *this; | |
6791 } | |
6792 | |
6793 PUGI__FN xml_text& xml_text::operator=(float rhs) | |
6794 { | |
6795 set(rhs); | |
6796 return *this; | |
6797 } | |
6798 | |
6799 PUGI__FN xml_text& xml_text::operator=(bool rhs) | |
6800 { | |
6801 set(rhs); | |
6802 return *this; | |
6803 } | |
6804 | |
6805 #ifdef PUGIXML_HAS_LONG_LONG | |
6806 PUGI__FN xml_text& xml_text::operator=(long long rhs) | |
6807 { | |
6808 set(rhs); | |
6809 return *this; | |
6810 } | |
6811 | |
6812 PUGI__FN xml_text& xml_text::operator=(unsigned long long rhs) | |
6813 { | |
6814 set(rhs); | |
6815 return *this; | |
6816 } | |
6817 #endif | |
6818 | |
6819 PUGI__FN xml_node xml_text::data() const | |
6820 { | |
6821 return xml_node(_data()); | |
6822 } | |
6823 | |
6824 #ifdef __BORLANDC__ | |
6825 PUGI__FN bool operator&&(const xml_text& lhs, bool rhs) | |
6826 { | |
6827 return (bool)lhs && rhs; | |
6828 } | |
6829 | |
6830 PUGI__FN bool operator||(const xml_text& lhs, bool rhs) | |
6831 { | |
6832 return (bool)lhs || rhs; | |
6833 } | |
6834 #endif | |
6835 | |
6836 PUGI__FN xml_node_iterator::xml_node_iterator() | |
6837 { | |
6838 } | |
6839 | |
6840 PUGI__FN xml_node_iterator::xml_node_iterator(const xml_node& node): _wrap(node), _parent(node.parent()) | |
6841 { | |
6842 } | |
6843 | |
6844 PUGI__FN xml_node_iterator::xml_node_iterator(xml_node_struct* ref, xml_node_struct* parent): _wrap(ref), _parent(parent) | |
6845 { | |
6846 } | |
6847 | |
6848 PUGI__FN bool xml_node_iterator::operator==(const xml_node_iterator& rhs) const | |
6849 { | |
6850 return _wrap._root == rhs._wrap._root && _parent._root == rhs._parent._root; | |
6851 } | |
6852 | |
6853 PUGI__FN bool xml_node_iterator::operator!=(const xml_node_iterator& rhs) const | |
6854 { | |
6855 return _wrap._root != rhs._wrap._root || _parent._root != rhs._parent._root; | |
6856 } | |
6857 | |
6858 PUGI__FN xml_node& xml_node_iterator::operator*() const | |
6859 { | |
6860 assert(_wrap._root); | |
6861 return _wrap; | |
6862 } | |
6863 | |
6864 PUGI__FN xml_node* xml_node_iterator::operator->() const | |
6865 { | |
6866 assert(_wrap._root); | |
6867 return const_cast<xml_node*>(&_wrap); // BCC5 workaround | |
6868 } | |
6869 | |
6870 PUGI__FN xml_node_iterator& xml_node_iterator::operator++() | |
6871 { | |
6872 assert(_wrap._root); | |
6873 _wrap._root = _wrap._root->next_sibling; | |
6874 return *this; | |
6875 } | |
6876 | |
6877 PUGI__FN xml_node_iterator xml_node_iterator::operator++(int) | |
6878 { | |
6879 xml_node_iterator temp = *this; | |
6880 ++*this; | |
6881 return temp; | |
6882 } | |
6883 | |
6884 PUGI__FN xml_node_iterator& xml_node_iterator::operator--() | |
6885 { | |
6886 _wrap = _wrap._root ? _wrap.previous_sibling() : _parent.last_child(); | |
6887 return *this; | |
6888 } | |
6889 | |
6890 PUGI__FN xml_node_iterator xml_node_iterator::operator--(int) | |
6891 { | |
6892 xml_node_iterator temp = *this; | |
6893 --*this; | |
6894 return temp; | |
6895 } | |
6896 | |
6897 PUGI__FN xml_attribute_iterator::xml_attribute_iterator() | |
6898 { | |
6899 } | |
6900 | |
6901 PUGI__FN xml_attribute_iterator::xml_attribute_iterator(const xml_attribute& attr, const xml_node& parent): _wrap(attr), _parent(parent) | |
6902 { | |
6903 } | |
6904 | |
6905 PUGI__FN xml_attribute_iterator::xml_attribute_iterator(xml_attribute_struct* ref, xml_node_struct* parent): _wrap(ref), _parent(parent) | |
6906 { | |
6907 } | |
6908 | |
6909 PUGI__FN bool xml_attribute_iterator::operator==(const xml_attribute_iterator& rhs) const | |
6910 { | |
6911 return _wrap._attr == rhs._wrap._attr && _parent._root == rhs._parent._root; | |
6912 } | |
6913 | |
6914 PUGI__FN bool xml_attribute_iterator::operator!=(const xml_attribute_iterator& rhs) const | |
6915 { | |
6916 return _wrap._attr != rhs._wrap._attr || _parent._root != rhs._parent._root; | |
6917 } | |
6918 | |
6919 PUGI__FN xml_attribute& xml_attribute_iterator::operator*() const | |
6920 { | |
6921 assert(_wrap._attr); | |
6922 return _wrap; | |
6923 } | |
6924 | |
6925 PUGI__FN xml_attribute* xml_attribute_iterator::operator->() const | |
6926 { | |
6927 assert(_wrap._attr); | |
6928 return const_cast<xml_attribute*>(&_wrap); // BCC5 workaround | |
6929 } | |
6930 | |
6931 PUGI__FN xml_attribute_iterator& xml_attribute_iterator::operator++() | |
6932 { | |
6933 assert(_wrap._attr); | |
6934 _wrap._attr = _wrap._attr->next_attribute; | |
6935 return *this; | |
6936 } | |
6937 | |
6938 PUGI__FN xml_attribute_iterator xml_attribute_iterator::operator++(int) | |
6939 { | |
6940 xml_attribute_iterator temp = *this; | |
6941 ++*this; | |
6942 return temp; | |
6943 } | |
6944 | |
6945 PUGI__FN xml_attribute_iterator& xml_attribute_iterator::operator--() | |
6946 { | |
6947 _wrap = _wrap._attr ? _wrap.previous_attribute() : _parent.last_attribute(); | |
6948 return *this; | |
6949 } | |
6950 | |
6951 PUGI__FN xml_attribute_iterator xml_attribute_iterator::operator--(int) | |
6952 { | |
6953 xml_attribute_iterator temp = *this; | |
6954 --*this; | |
6955 return temp; | |
6956 } | |
6957 | |
6958 PUGI__FN xml_named_node_iterator::xml_named_node_iterator(): _name(0) | |
6959 { | |
6960 } | |
6961 | |
6962 PUGI__FN xml_named_node_iterator::xml_named_node_iterator(const xml_node& node, const char_t* name): _wrap(node), _parent(node.parent()), _name(name) | |
6963 { | |
6964 } | |
6965 | |
6966 PUGI__FN xml_named_node_iterator::xml_named_node_iterator(xml_node_struct* ref, xml_node_struct* parent, const char_t* name): _wrap(ref), _parent(parent), _name(name) | |
6967 { | |
6968 } | |
6969 | |
6970 PUGI__FN bool xml_named_node_iterator::operator==(const xml_named_node_iterator& rhs) const | |
6971 { | |
6972 return _wrap._root == rhs._wrap._root && _parent._root == rhs._parent._root; | |
6973 } | |
6974 | |
6975 PUGI__FN bool xml_named_node_iterator::operator!=(const xml_named_node_iterator& rhs) const | |
6976 { | |
6977 return _wrap._root != rhs._wrap._root || _parent._root != rhs._parent._root; | |
6978 } | |
6979 | |
6980 PUGI__FN xml_node& xml_named_node_iterator::operator*() const | |
6981 { | |
6982 assert(_wrap._root); | |
6983 return _wrap; | |
6984 } | |
6985 | |
6986 PUGI__FN xml_node* xml_named_node_iterator::operator->() const | |
6987 { | |
6988 assert(_wrap._root); | |
6989 return const_cast<xml_node*>(&_wrap); // BCC5 workaround | |
6990 } | |
6991 | |
6992 PUGI__FN xml_named_node_iterator& xml_named_node_iterator::operator++() | |
6993 { | |
6994 assert(_wrap._root); | |
6995 _wrap = _wrap.next_sibling(_name); | |
6996 return *this; | |
6997 } | |
6998 | |
6999 PUGI__FN xml_named_node_iterator xml_named_node_iterator::operator++(int) | |
7000 { | |
7001 xml_named_node_iterator temp = *this; | |
7002 ++*this; | |
7003 return temp; | |
7004 } | |
7005 | |
7006 PUGI__FN xml_named_node_iterator& xml_named_node_iterator::operator--() | |
7007 { | |
7008 if (_wrap._root) | |
7009 _wrap = _wrap.previous_sibling(_name); | |
7010 else | |
7011 { | |
7012 _wrap = _parent.last_child(); | |
7013 | |
7014 if (!impl::strequal(_wrap.name(), _name)) | |
7015 _wrap = _wrap.previous_sibling(_name); | |
7016 } | |
7017 | |
7018 return *this; | |
7019 } | |
7020 | |
7021 PUGI__FN xml_named_node_iterator xml_named_node_iterator::operator--(int) | |
7022 { | |
7023 xml_named_node_iterator temp = *this; | |
7024 --*this; | |
7025 return temp; | |
7026 } | |
7027 | |
7028 PUGI__FN xml_parse_result::xml_parse_result(): status(status_internal_error), offset(0), encoding(encoding_auto) | |
7029 { | |
7030 } | |
7031 | |
7032 PUGI__FN xml_parse_result::operator bool() const | |
7033 { | |
7034 return status == status_ok; | |
7035 } | |
7036 | |
7037 PUGI__FN const char* xml_parse_result::description() const | |
7038 { | |
7039 switch (status) | |
7040 { | |
7041 case status_ok: return "No error"; | |
7042 | |
7043 case status_file_not_found: return "File was not found"; | |
7044 case status_io_error: return "Error reading from file/stream"; | |
7045 case status_out_of_memory: return "Could not allocate memory"; | |
7046 case status_internal_error: return "Internal error occurred"; | |
7047 | |
7048 case status_unrecognized_tag: return "Could not determine tag type"; | |
7049 | |
7050 case status_bad_pi: return "Error parsing document declaration/processing instruction"; | |
7051 case status_bad_comment: return "Error parsing comment"; | |
7052 case status_bad_cdata: return "Error parsing CDATA section"; | |
7053 case status_bad_doctype: return "Error parsing document type declaration"; | |
7054 case status_bad_pcdata: return "Error parsing PCDATA section"; | |
7055 case status_bad_start_element: return "Error parsing start element tag"; | |
7056 case status_bad_attribute: return "Error parsing element attribute"; | |
7057 case status_bad_end_element: return "Error parsing end element tag"; | |
7058 case status_end_element_mismatch: return "Start-end tags mismatch"; | |
7059 | |
7060 case status_append_invalid_root: return "Unable to append nodes: root is not an element or document"; | |
7061 | |
7062 case status_no_document_element: return "No document element found"; | |
7063 | |
7064 default: return "Unknown error"; | |
7065 } | |
7066 } | |
7067 | |
7068 PUGI__FN xml_document::xml_document(): _buffer(0) | |
7069 { | |
7070 _create(); | |
7071 } | |
7072 | |
7073 PUGI__FN xml_document::~xml_document() | |
7074 { | |
7075 _destroy(); | |
7076 } | |
7077 | |
7078 #ifdef PUGIXML_HAS_MOVE | |
7079 PUGI__FN xml_document::xml_document(xml_document&& rhs) PUGIXML_NOEXCEPT_IF_NOT_COMPACT: _buffer(0) | |
7080 { | |
7081 _create(); | |
7082 _move(rhs); | |
7083 } | |
7084 | |
7085 PUGI__FN xml_document& xml_document::operator=(xml_document&& rhs) PUGIXML_NOEXCEPT_IF_NOT_COMPACT | |
7086 { | |
7087 if (this == &rhs) return *this; | |
7088 | |
7089 _destroy(); | |
7090 _create(); | |
7091 _move(rhs); | |
7092 | |
7093 return *this; | |
7094 } | |
7095 #endif | |
7096 | |
7097 PUGI__FN void xml_document::reset() | |
7098 { | |
7099 _destroy(); | |
7100 _create(); | |
7101 } | |
7102 | |
7103 PUGI__FN void xml_document::reset(const xml_document& proto) | |
7104 { | |
7105 reset(); | |
7106 | |
7107 impl::node_copy_tree(_root, proto._root); | |
7108 } | |
7109 | |
7110 PUGI__FN void xml_document::_create() | |
7111 { | |
7112 assert(!_root); | |
7113 | |
7114 #ifdef PUGIXML_COMPACT | |
7115 // space for page marker for the first page (uint32_t), rounded up to pointer size; assumes pointers are at least 32-bit | |
7116 const size_t page_offset = sizeof(void*); | |
7117 #else | |
7118 const size_t page_offset = 0; | |
7119 #endif | |
7120 | |
7121 // initialize sentinel page | |
7122 PUGI__STATIC_ASSERT(sizeof(impl::xml_memory_page) + sizeof(impl::xml_document_struct) + page_offset <= sizeof(_memory)); | |
7123 | |
7124 // prepare page structure | |
7125 impl::xml_memory_page* page = impl::xml_memory_page::construct(_memory); | |
7126 assert(page); | |
7127 | |
7128 page->busy_size = impl::xml_memory_page_size; | |
7129 | |
7130 // setup first page marker | |
7131 #ifdef PUGIXML_COMPACT | |
7132 // round-trip through void* to avoid 'cast increases required alignment of target type' warning | |
7133 page->compact_page_marker = reinterpret_cast<uint32_t*>(static_cast<void*>(reinterpret_cast<char*>(page) + sizeof(impl::xml_memory_page))); | |
7134 *page->compact_page_marker = sizeof(impl::xml_memory_page); | |
7135 #endif | |
7136 | |
7137 // allocate new root | |
7138 _root = new (reinterpret_cast<char*>(page) + sizeof(impl::xml_memory_page) + page_offset) impl::xml_document_struct(page); | |
7139 _root->prev_sibling_c = _root; | |
7140 | |
7141 // setup sentinel page | |
7142 page->allocator = static_cast<impl::xml_document_struct*>(_root); | |
7143 | |
7144 // setup hash table pointer in allocator | |
7145 #ifdef PUGIXML_COMPACT | |
7146 page->allocator->_hash = &static_cast<impl::xml_document_struct*>(_root)->hash; | |
7147 #endif | |
7148 | |
7149 // verify the document allocation | |
7150 assert(reinterpret_cast<char*>(_root) + sizeof(impl::xml_document_struct) <= _memory + sizeof(_memory)); | |
7151 } | |
7152 | |
7153 PUGI__FN void xml_document::_destroy() | |
7154 { | |
7155 assert(_root); | |
7156 | |
7157 // destroy static storage | |
7158 if (_buffer) | |
7159 { | |
7160 impl::xml_memory::deallocate(_buffer); | |
7161 _buffer = 0; | |
7162 } | |
7163 | |
7164 // destroy extra buffers (note: no need to destroy linked list nodes, they're allocated using document allocator) | |
7165 for (impl::xml_extra_buffer* extra = static_cast<impl::xml_document_struct*>(_root)->extra_buffers; extra; extra = extra->next) | |
7166 { | |
7167 if (extra->buffer) impl::xml_memory::deallocate(extra->buffer); | |
7168 } | |
7169 | |
7170 // destroy dynamic storage, leave sentinel page (it's in static memory) | |
7171 impl::xml_memory_page* root_page = PUGI__GETPAGE(_root); | |
7172 assert(root_page && !root_page->prev); | |
7173 assert(reinterpret_cast<char*>(root_page) >= _memory && reinterpret_cast<char*>(root_page) < _memory + sizeof(_memory)); | |
7174 | |
7175 for (impl::xml_memory_page* page = root_page->next; page; ) | |
7176 { | |
7177 impl::xml_memory_page* next = page->next; | |
7178 | |
7179 impl::xml_allocator::deallocate_page(page); | |
7180 | |
7181 page = next; | |
7182 } | |
7183 | |
7184 #ifdef PUGIXML_COMPACT | |
7185 // destroy hash table | |
7186 static_cast<impl::xml_document_struct*>(_root)->hash.clear(); | |
7187 #endif | |
7188 | |
7189 _root = 0; | |
7190 } | |
7191 | |
7192 #ifdef PUGIXML_HAS_MOVE | |
7193 PUGI__FN void xml_document::_move(xml_document& rhs) PUGIXML_NOEXCEPT_IF_NOT_COMPACT | |
7194 { | |
7195 impl::xml_document_struct* doc = static_cast<impl::xml_document_struct*>(_root); | |
7196 impl::xml_document_struct* other = static_cast<impl::xml_document_struct*>(rhs._root); | |
7197 | |
7198 // save first child pointer for later; this needs hash access | |
7199 xml_node_struct* other_first_child = other->first_child; | |
7200 | |
7201 #ifdef PUGIXML_COMPACT | |
7202 // reserve space for the hash table up front; this is the only operation that can fail | |
7203 // if it does, we have no choice but to throw (if we have exceptions) | |
7204 if (other_first_child) | |
7205 { | |
7206 size_t other_children = 0; | |
7207 for (xml_node_struct* node = other_first_child; node; node = node->next_sibling) | |
7208 other_children++; | |
7209 | |
7210 // in compact mode, each pointer assignment could result in a hash table request | |
7211 // during move, we have to relocate document first_child and parents of all children | |
7212 // normally there's just one child and its parent has a pointerless encoding but | |
7213 // we assume the worst here | |
7214 if (!other->_hash->reserve(other_children + 1)) | |
7215 { | |
7216 #ifdef PUGIXML_NO_EXCEPTIONS | |
7217 return; | |
7218 #else | |
7219 throw std::bad_alloc(); | |
7220 #endif | |
7221 } | |
7222 } | |
7223 #endif | |
7224 | |
7225 // move allocation state | |
7226 // note that other->_root may point to the embedded document page, in which case we should keep original (empty) state | |
7227 if (other->_root != PUGI__GETPAGE(other)) | |
7228 { | |
7229 doc->_root = other->_root; | |
7230 doc->_busy_size = other->_busy_size; | |
7231 } | |
7232 | |
7233 // move buffer state | |
7234 doc->buffer = other->buffer; | |
7235 doc->extra_buffers = other->extra_buffers; | |
7236 _buffer = rhs._buffer; | |
7237 | |
7238 #ifdef PUGIXML_COMPACT | |
7239 // move compact hash; note that the hash table can have pointers to other but they will be "inactive", similarly to nodes removed with remove_child | |
7240 doc->hash = other->hash; | |
7241 doc->_hash = &doc->hash; | |
7242 | |
7243 // make sure we don't access other hash up until the end when we reinitialize other document | |
7244 other->_hash = 0; | |
7245 #endif | |
7246 | |
7247 // move page structure | |
7248 impl::xml_memory_page* doc_page = PUGI__GETPAGE(doc); | |
7249 assert(doc_page && !doc_page->prev && !doc_page->next); | |
7250 | |
7251 impl::xml_memory_page* other_page = PUGI__GETPAGE(other); | |
7252 assert(other_page && !other_page->prev); | |
7253 | |
7254 // relink pages since root page is embedded into xml_document | |
7255 if (impl::xml_memory_page* page = other_page->next) | |
7256 { | |
7257 assert(page->prev == other_page); | |
7258 | |
7259 page->prev = doc_page; | |
7260 | |
7261 doc_page->next = page; | |
7262 other_page->next = 0; | |
7263 } | |
7264 | |
7265 // make sure pages point to the correct document state | |
7266 for (impl::xml_memory_page* page = doc_page->next; page; page = page->next) | |
7267 { | |
7268 assert(page->allocator == other); | |
7269 | |
7270 page->allocator = doc; | |
7271 | |
7272 #ifdef PUGIXML_COMPACT | |
7273 // this automatically migrates most children between documents and prevents ->parent assignment from allocating | |
7274 if (page->compact_shared_parent == other) | |
7275 page->compact_shared_parent = doc; | |
7276 #endif | |
7277 } | |
7278 | |
7279 // move tree structure | |
7280 assert(!doc->first_child); | |
7281 | |
7282 doc->first_child = other_first_child; | |
7283 | |
7284 for (xml_node_struct* node = other_first_child; node; node = node->next_sibling) | |
7285 { | |
7286 #ifdef PUGIXML_COMPACT | |
7287 // most children will have migrated when we reassigned compact_shared_parent | |
7288 assert(node->parent == other || node->parent == doc); | |
7289 | |
7290 node->parent = doc; | |
7291 #else | |
7292 assert(node->parent == other); | |
7293 node->parent = doc; | |
7294 #endif | |
7295 } | |
7296 | |
7297 // reset other document | |
7298 new (other) impl::xml_document_struct(PUGI__GETPAGE(other)); | |
7299 rhs._buffer = 0; | |
7300 } | |
7301 #endif | |
7302 | |
7303 #ifndef PUGIXML_NO_STL | |
7304 PUGI__FN xml_parse_result xml_document::load(std::basic_istream<char, std::char_traits<char> >& stream, unsigned int options, xml_encoding encoding) | |
7305 { | |
7306 reset(); | |
7307 | |
7308 return impl::load_stream_impl(static_cast<impl::xml_document_struct*>(_root), stream, options, encoding, &_buffer); | |
7309 } | |
7310 | |
7311 PUGI__FN xml_parse_result xml_document::load(std::basic_istream<wchar_t, std::char_traits<wchar_t> >& stream, unsigned int options) | |
7312 { | |
7313 reset(); | |
7314 | |
7315 return impl::load_stream_impl(static_cast<impl::xml_document_struct*>(_root), stream, options, encoding_wchar, &_buffer); | |
7316 } | |
7317 #endif | |
7318 | |
7319 PUGI__FN xml_parse_result xml_document::load_string(const char_t* contents, unsigned int options) | |
7320 { | |
7321 // Force native encoding (skip autodetection) | |
7322 #ifdef PUGIXML_WCHAR_MODE | |
7323 xml_encoding encoding = encoding_wchar; | |
7324 #else | |
7325 xml_encoding encoding = encoding_utf8; | |
7326 #endif | |
7327 | |
7328 return load_buffer(contents, impl::strlength(contents) * sizeof(char_t), options, encoding); | |
7329 } | |
7330 | |
7331 PUGI__FN xml_parse_result xml_document::load(const char_t* contents, unsigned int options) | |
7332 { | |
7333 return load_string(contents, options); | |
7334 } | |
7335 | |
7336 PUGI__FN xml_parse_result xml_document::load_file(const char* path_, unsigned int options, xml_encoding encoding) | |
7337 { | |
7338 reset(); | |
7339 | |
7340 using impl::auto_deleter; // MSVC7 workaround | |
7341 auto_deleter<FILE> file(impl::open_file(path_, "rb"), impl::close_file); | |
7342 | |
7343 return impl::load_file_impl(static_cast<impl::xml_document_struct*>(_root), file.data, options, encoding, &_buffer); | |
7344 } | |
7345 | |
7346 PUGI__FN xml_parse_result xml_document::load_file(const wchar_t* path_, unsigned int options, xml_encoding encoding) | |
7347 { | |
7348 reset(); | |
7349 | |
7350 using impl::auto_deleter; // MSVC7 workaround | |
7351 auto_deleter<FILE> file(impl::open_file_wide(path_, L"rb"), impl::close_file); | |
7352 | |
7353 return impl::load_file_impl(static_cast<impl::xml_document_struct*>(_root), file.data, options, encoding, &_buffer); | |
7354 } | |
7355 | |
7356 PUGI__FN xml_parse_result xml_document::load_buffer(const void* contents, size_t size, unsigned int options, xml_encoding encoding) | |
7357 { | |
7358 reset(); | |
7359 | |
7360 return impl::load_buffer_impl(static_cast<impl::xml_document_struct*>(_root), _root, const_cast<void*>(contents), size, options, encoding, false, false, &_buffer); | |
7361 } | |
7362 | |
7363 PUGI__FN xml_parse_result xml_document::load_buffer_inplace(void* contents, size_t size, unsigned int options, xml_encoding encoding) | |
7364 { | |
7365 reset(); | |
7366 | |
7367 return impl::load_buffer_impl(static_cast<impl::xml_document_struct*>(_root), _root, contents, size, options, encoding, true, false, &_buffer); | |
7368 } | |
7369 | |
7370 PUGI__FN xml_parse_result xml_document::load_buffer_inplace_own(void* contents, size_t size, unsigned int options, xml_encoding encoding) | |
7371 { | |
7372 reset(); | |
7373 | |
7374 return impl::load_buffer_impl(static_cast<impl::xml_document_struct*>(_root), _root, contents, size, options, encoding, true, true, &_buffer); | |
7375 } | |
7376 | |
7377 PUGI__FN void xml_document::save(xml_writer& writer, const char_t* indent, unsigned int flags, xml_encoding encoding) const | |
7378 { | |
7379 impl::xml_buffered_writer buffered_writer(writer, encoding); | |
7380 | |
7381 if ((flags & format_write_bom) && encoding != encoding_latin1) | |
7382 { | |
7383 // BOM always represents the codepoint U+FEFF, so just write it in native encoding | |
7384 #ifdef PUGIXML_WCHAR_MODE | |
7385 unsigned int bom = 0xfeff; | |
7386 buffered_writer.write(static_cast<wchar_t>(bom)); | |
7387 #else | |
7388 buffered_writer.write('\xef', '\xbb', '\xbf'); | |
7389 #endif | |
7390 } | |
7391 | |
7392 if (!(flags & format_no_declaration) && !impl::has_declaration(_root)) | |
7393 { | |
7394 buffered_writer.write_string(PUGIXML_TEXT("<?xml version=\"1.0\"")); | |
7395 if (encoding == encoding_latin1) buffered_writer.write_string(PUGIXML_TEXT(" encoding=\"ISO-8859-1\"")); | |
7396 buffered_writer.write('?', '>'); | |
7397 if (!(flags & format_raw)) buffered_writer.write('\n'); | |
7398 } | |
7399 | |
7400 impl::node_output(buffered_writer, _root, indent, flags, 0); | |
7401 | |
7402 buffered_writer.flush(); | |
7403 } | |
7404 | |
7405 #ifndef PUGIXML_NO_STL | |
7406 PUGI__FN void xml_document::save(std::basic_ostream<char, std::char_traits<char> >& stream, const char_t* indent, unsigned int flags, xml_encoding encoding) const | |
7407 { | |
7408 xml_writer_stream writer(stream); | |
7409 | |
7410 save(writer, indent, flags, encoding); | |
7411 } | |
7412 | |
7413 PUGI__FN void xml_document::save(std::basic_ostream<wchar_t, std::char_traits<wchar_t> >& stream, const char_t* indent, unsigned int flags) const | |
7414 { | |
7415 xml_writer_stream writer(stream); | |
7416 | |
7417 save(writer, indent, flags, encoding_wchar); | |
7418 } | |
7419 #endif | |
7420 | |
7421 PUGI__FN bool xml_document::save_file(const char* path_, const char_t* indent, unsigned int flags, xml_encoding encoding) const | |
7422 { | |
7423 using impl::auto_deleter; // MSVC7 workaround | |
7424 auto_deleter<FILE> file(impl::open_file(path_, (flags & format_save_file_text) ? "w" : "wb"), impl::close_file); | |
7425 | |
7426 return impl::save_file_impl(*this, file.data, indent, flags, encoding) && fclose(file.release()) == 0; | |
7427 } | |
7428 | |
7429 PUGI__FN bool xml_document::save_file(const wchar_t* path_, const char_t* indent, unsigned int flags, xml_encoding encoding) const | |
7430 { | |
7431 using impl::auto_deleter; // MSVC7 workaround | |
7432 auto_deleter<FILE> file(impl::open_file_wide(path_, (flags & format_save_file_text) ? L"w" : L"wb"), impl::close_file); | |
7433 | |
7434 return impl::save_file_impl(*this, file.data, indent, flags, encoding) && fclose(file.release()) == 0; | |
7435 } | |
7436 | |
7437 PUGI__FN xml_node xml_document::document_element() const | |
7438 { | |
7439 assert(_root); | |
7440 | |
7441 for (xml_node_struct* i = _root->first_child; i; i = i->next_sibling) | |
7442 if (PUGI__NODETYPE(i) == node_element) | |
7443 return xml_node(i); | |
7444 | |
7445 return xml_node(); | |
7446 } | |
7447 | |
7448 #ifndef PUGIXML_NO_STL | |
7449 PUGI__FN std::string PUGIXML_FUNCTION as_utf8(const wchar_t* str) | |
7450 { | |
7451 assert(str); | |
7452 | |
7453 return impl::as_utf8_impl(str, impl::strlength_wide(str)); | |
7454 } | |
7455 | |
7456 PUGI__FN std::string PUGIXML_FUNCTION as_utf8(const std::basic_string<wchar_t>& str) | |
7457 { | |
7458 return impl::as_utf8_impl(str.c_str(), str.size()); | |
7459 } | |
7460 | |
7461 PUGI__FN std::basic_string<wchar_t> PUGIXML_FUNCTION as_wide(const char* str) | |
7462 { | |
7463 assert(str); | |
7464 | |
7465 return impl::as_wide_impl(str, strlen(str)); | |
7466 } | |
7467 | |
7468 PUGI__FN std::basic_string<wchar_t> PUGIXML_FUNCTION as_wide(const std::string& str) | |
7469 { | |
7470 return impl::as_wide_impl(str.c_str(), str.size()); | |
7471 } | |
7472 #endif | |
7473 | |
7474 PUGI__FN void PUGIXML_FUNCTION set_memory_management_functions(allocation_function allocate, deallocation_function deallocate) | |
7475 { | |
7476 impl::xml_memory::allocate = allocate; | |
7477 impl::xml_memory::deallocate = deallocate; | |
7478 } | |
7479 | |
7480 PUGI__FN allocation_function PUGIXML_FUNCTION get_memory_allocation_function() | |
7481 { | |
7482 return impl::xml_memory::allocate; | |
7483 } | |
7484 | |
7485 PUGI__FN deallocation_function PUGIXML_FUNCTION get_memory_deallocation_function() | |
7486 { | |
7487 return impl::xml_memory::deallocate; | |
7488 } | |
7489 } | |
7490 | |
7491 #if !defined(PUGIXML_NO_STL) && (defined(_MSC_VER) || defined(__ICC)) | |
7492 namespace std | |
7493 { | |
7494 // Workarounds for (non-standard) iterator category detection for older versions (MSVC7/IC8 and earlier) | |
7495 PUGI__FN std::bidirectional_iterator_tag _Iter_cat(const pugi::xml_node_iterator&) | |
7496 { | |
7497 return std::bidirectional_iterator_tag(); | |
7498 } | |
7499 | |
7500 PUGI__FN std::bidirectional_iterator_tag _Iter_cat(const pugi::xml_attribute_iterator&) | |
7501 { | |
7502 return std::bidirectional_iterator_tag(); | |
7503 } | |
7504 | |
7505 PUGI__FN std::bidirectional_iterator_tag _Iter_cat(const pugi::xml_named_node_iterator&) | |
7506 { | |
7507 return std::bidirectional_iterator_tag(); | |
7508 } | |
7509 } | |
7510 #endif | |
7511 | |
7512 #if !defined(PUGIXML_NO_STL) && defined(__SUNPRO_CC) | |
7513 namespace std | |
7514 { | |
7515 // Workarounds for (non-standard) iterator category detection | |
7516 PUGI__FN std::bidirectional_iterator_tag __iterator_category(const pugi::xml_node_iterator&) | |
7517 { | |
7518 return std::bidirectional_iterator_tag(); | |
7519 } | |
7520 | |
7521 PUGI__FN std::bidirectional_iterator_tag __iterator_category(const pugi::xml_attribute_iterator&) | |
7522 { | |
7523 return std::bidirectional_iterator_tag(); | |
7524 } | |
7525 | |
7526 PUGI__FN std::bidirectional_iterator_tag __iterator_category(const pugi::xml_named_node_iterator&) | |
7527 { | |
7528 return std::bidirectional_iterator_tag(); | |
7529 } | |
7530 } | |
7531 #endif | |
7532 | |
7533 #ifndef PUGIXML_NO_XPATH | |
7534 // STL replacements | |
7535 PUGI__NS_BEGIN | |
7536 struct equal_to | |
7537 { | |
7538 template <typename T> bool operator()(const T& lhs, const T& rhs) const | |
7539 { | |
7540 return lhs == rhs; | |
7541 } | |
7542 }; | |
7543 | |
7544 struct not_equal_to | |
7545 { | |
7546 template <typename T> bool operator()(const T& lhs, const T& rhs) const | |
7547 { | |
7548 return lhs != rhs; | |
7549 } | |
7550 }; | |
7551 | |
7552 struct less | |
7553 { | |
7554 template <typename T> bool operator()(const T& lhs, const T& rhs) const | |
7555 { | |
7556 return lhs < rhs; | |
7557 } | |
7558 }; | |
7559 | |
7560 struct less_equal | |
7561 { | |
7562 template <typename T> bool operator()(const T& lhs, const T& rhs) const | |
7563 { | |
7564 return lhs <= rhs; | |
7565 } | |
7566 }; | |
7567 | |
7568 template <typename T> inline void swap(T& lhs, T& rhs) | |
7569 { | |
7570 T temp = lhs; | |
7571 lhs = rhs; | |
7572 rhs = temp; | |
7573 } | |
7574 | |
7575 template <typename I, typename Pred> PUGI__FN I min_element(I begin, I end, const Pred& pred) | |
7576 { | |
7577 I result = begin; | |
7578 | |
7579 for (I it = begin + 1; it != end; ++it) | |
7580 if (pred(*it, *result)) | |
7581 result = it; | |
7582 | |
7583 return result; | |
7584 } | |
7585 | |
7586 template <typename I> PUGI__FN void reverse(I begin, I end) | |
7587 { | |
7588 while (end - begin > 1) | |
7589 swap(*begin++, *--end); | |
7590 } | |
7591 | |
7592 template <typename I> PUGI__FN I unique(I begin, I end) | |
7593 { | |
7594 // fast skip head | |
7595 while (end - begin > 1 && *begin != *(begin + 1)) | |
7596 begin++; | |
7597 | |
7598 if (begin == end) | |
7599 return begin; | |
7600 | |
7601 // last written element | |
7602 I write = begin++; | |
7603 | |
7604 // merge unique elements | |
7605 while (begin != end) | |
7606 { | |
7607 if (*begin != *write) | |
7608 *++write = *begin++; | |
7609 else | |
7610 begin++; | |
7611 } | |
7612 | |
7613 // past-the-end (write points to live element) | |
7614 return write + 1; | |
7615 } | |
7616 | |
7617 template <typename T, typename Pred> PUGI__FN void insertion_sort(T* begin, T* end, const Pred& pred) | |
7618 { | |
7619 if (begin == end) | |
7620 return; | |
7621 | |
7622 for (T* it = begin + 1; it != end; ++it) | |
7623 { | |
7624 T val = *it; | |
7625 T* hole = it; | |
7626 | |
7627 // move hole backwards | |
7628 while (hole > begin && pred(val, *(hole - 1))) | |
7629 { | |
7630 *hole = *(hole - 1); | |
7631 hole--; | |
7632 } | |
7633 | |
7634 // fill hole with element | |
7635 *hole = val; | |
7636 } | |
7637 } | |
7638 | |
7639 template <typename I, typename Pred> inline I median3(I first, I middle, I last, const Pred& pred) | |
7640 { | |
7641 if (pred(*middle, *first)) | |
7642 swap(middle, first); | |
7643 if (pred(*last, *middle)) | |
7644 swap(last, middle); | |
7645 if (pred(*middle, *first)) | |
7646 swap(middle, first); | |
7647 | |
7648 return middle; | |
7649 } | |
7650 | |
7651 template <typename T, typename Pred> PUGI__FN void partition3(T* begin, T* end, T pivot, const Pred& pred, T** out_eqbeg, T** out_eqend) | |
7652 { | |
7653 // invariant: array is split into 4 groups: = < ? > (each variable denotes the boundary between the groups) | |
7654 T* eq = begin; | |
7655 T* lt = begin; | |
7656 T* gt = end; | |
7657 | |
7658 while (lt < gt) | |
7659 { | |
7660 if (pred(*lt, pivot)) | |
7661 lt++; | |
7662 else if (*lt == pivot) | |
7663 swap(*eq++, *lt++); | |
7664 else | |
7665 swap(*lt, *--gt); | |
7666 } | |
7667 | |
7668 // we now have just 4 groups: = < >; move equal elements to the middle | |
7669 T* eqbeg = gt; | |
7670 | |
7671 for (T* it = begin; it != eq; ++it) | |
7672 swap(*it, *--eqbeg); | |
7673 | |
7674 *out_eqbeg = eqbeg; | |
7675 *out_eqend = gt; | |
7676 } | |
7677 | |
7678 template <typename I, typename Pred> PUGI__FN void sort(I begin, I end, const Pred& pred) | |
7679 { | |
7680 // sort large chunks | |
7681 while (end - begin > 16) | |
7682 { | |
7683 // find median element | |
7684 I middle = begin + (end - begin) / 2; | |
7685 I median = median3(begin, middle, end - 1, pred); | |
7686 | |
7687 // partition in three chunks (< = >) | |
7688 I eqbeg, eqend; | |
7689 partition3(begin, end, *median, pred, &eqbeg, &eqend); | |
7690 | |
7691 // loop on larger half | |
7692 if (eqbeg - begin > end - eqend) | |
7693 { | |
7694 sort(eqend, end, pred); | |
7695 end = eqbeg; | |
7696 } | |
7697 else | |
7698 { | |
7699 sort(begin, eqbeg, pred); | |
7700 begin = eqend; | |
7701 } | |
7702 } | |
7703 | |
7704 // insertion sort small chunk | |
7705 insertion_sort(begin, end, pred); | |
7706 } | |
7707 | |
7708 PUGI__FN bool hash_insert(const void** table, size_t size, const void* key) | |
7709 { | |
7710 assert(key); | |
7711 | |
7712 unsigned int h = static_cast<unsigned int>(reinterpret_cast<uintptr_t>(key)); | |
7713 | |
7714 // MurmurHash3 32-bit finalizer | |
7715 h ^= h >> 16; | |
7716 h *= 0x85ebca6bu; | |
7717 h ^= h >> 13; | |
7718 h *= 0xc2b2ae35u; | |
7719 h ^= h >> 16; | |
7720 | |
7721 size_t hashmod = size - 1; | |
7722 size_t bucket = h & hashmod; | |
7723 | |
7724 for (size_t probe = 0; probe <= hashmod; ++probe) | |
7725 { | |
7726 if (table[bucket] == 0) | |
7727 { | |
7728 table[bucket] = key; | |
7729 return true; | |
7730 } | |
7731 | |
7732 if (table[bucket] == key) | |
7733 return false; | |
7734 | |
7735 // hash collision, quadratic probing | |
7736 bucket = (bucket + probe + 1) & hashmod; | |
7737 } | |
7738 | |
7739 assert(false && "Hash table is full"); // unreachable | |
7740 return false; | |
7741 } | |
7742 PUGI__NS_END | |
7743 | |
7744 // Allocator used for AST and evaluation stacks | |
7745 PUGI__NS_BEGIN | |
7746 static const size_t xpath_memory_page_size = | |
7747 #ifdef PUGIXML_MEMORY_XPATH_PAGE_SIZE | |
7748 PUGIXML_MEMORY_XPATH_PAGE_SIZE | |
7749 #else | |
7750 4096 | |
7751 #endif | |
7752 ; | |
7753 | |
7754 static const uintptr_t xpath_memory_block_alignment = sizeof(double) > sizeof(void*) ? sizeof(double) : sizeof(void*); | |
7755 | |
7756 struct xpath_memory_block | |
7757 { | |
7758 xpath_memory_block* next; | |
7759 size_t capacity; | |
7760 | |
7761 union | |
7762 { | |
7763 char data[xpath_memory_page_size]; | |
7764 double alignment; | |
7765 }; | |
7766 }; | |
7767 | |
7768 struct xpath_allocator | |
7769 { | |
7770 xpath_memory_block* _root; | |
7771 size_t _root_size; | |
7772 bool* _error; | |
7773 | |
7774 xpath_allocator(xpath_memory_block* root, bool* error = 0): _root(root), _root_size(0), _error(error) | |
7775 { | |
7776 } | |
7777 | |
7778 void* allocate(size_t size) | |
7779 { | |
7780 // round size up to block alignment boundary | |
7781 size = (size + xpath_memory_block_alignment - 1) & ~(xpath_memory_block_alignment - 1); | |
7782 | |
7783 if (_root_size + size <= _root->capacity) | |
7784 { | |
7785 void* buf = &_root->data[0] + _root_size; | |
7786 _root_size += size; | |
7787 return buf; | |
7788 } | |
7789 else | |
7790 { | |
7791 // make sure we have at least 1/4th of the page free after allocation to satisfy subsequent allocation requests | |
7792 size_t block_capacity_base = sizeof(_root->data); | |
7793 size_t block_capacity_req = size + block_capacity_base / 4; | |
7794 size_t block_capacity = (block_capacity_base > block_capacity_req) ? block_capacity_base : block_capacity_req; | |
7795 | |
7796 size_t block_size = block_capacity + offsetof(xpath_memory_block, data); | |
7797 | |
7798 xpath_memory_block* block = static_cast<xpath_memory_block*>(xml_memory::allocate(block_size)); | |
7799 if (!block) | |
7800 { | |
7801 if (_error) *_error = true; | |
7802 return 0; | |
7803 } | |
7804 | |
7805 block->next = _root; | |
7806 block->capacity = block_capacity; | |
7807 | |
7808 _root = block; | |
7809 _root_size = size; | |
7810 | |
7811 return block->data; | |
7812 } | |
7813 } | |
7814 | |
7815 void* reallocate(void* ptr, size_t old_size, size_t new_size) | |
7816 { | |
7817 // round size up to block alignment boundary | |
7818 old_size = (old_size + xpath_memory_block_alignment - 1) & ~(xpath_memory_block_alignment - 1); | |
7819 new_size = (new_size + xpath_memory_block_alignment - 1) & ~(xpath_memory_block_alignment - 1); | |
7820 | |
7821 // we can only reallocate the last object | |
7822 assert(ptr == 0 || static_cast<char*>(ptr) + old_size == &_root->data[0] + _root_size); | |
7823 | |
7824 // try to reallocate the object inplace | |
7825 if (ptr && _root_size - old_size + new_size <= _root->capacity) | |
7826 { | |
7827 _root_size = _root_size - old_size + new_size; | |
7828 return ptr; | |
7829 } | |
7830 | |
7831 // allocate a new block | |
7832 void* result = allocate(new_size); | |
7833 if (!result) return 0; | |
7834 | |
7835 // we have a new block | |
7836 if (ptr) | |
7837 { | |
7838 // copy old data (we only support growing) | |
7839 assert(new_size >= old_size); | |
7840 memcpy(result, ptr, old_size); | |
7841 | |
7842 // free the previous page if it had no other objects | |
7843 assert(_root->data == result); | |
7844 assert(_root->next); | |
7845 | |
7846 if (_root->next->data == ptr) | |
7847 { | |
7848 // deallocate the whole page, unless it was the first one | |
7849 xpath_memory_block* next = _root->next->next; | |
7850 | |
7851 if (next) | |
7852 { | |
7853 xml_memory::deallocate(_root->next); | |
7854 _root->next = next; | |
7855 } | |
7856 } | |
7857 } | |
7858 | |
7859 return result; | |
7860 } | |
7861 | |
7862 void revert(const xpath_allocator& state) | |
7863 { | |
7864 // free all new pages | |
7865 xpath_memory_block* cur = _root; | |
7866 | |
7867 while (cur != state._root) | |
7868 { | |
7869 xpath_memory_block* next = cur->next; | |
7870 | |
7871 xml_memory::deallocate(cur); | |
7872 | |
7873 cur = next; | |
7874 } | |
7875 | |
7876 // restore state | |
7877 _root = state._root; | |
7878 _root_size = state._root_size; | |
7879 } | |
7880 | |
7881 void release() | |
7882 { | |
7883 xpath_memory_block* cur = _root; | |
7884 assert(cur); | |
7885 | |
7886 while (cur->next) | |
7887 { | |
7888 xpath_memory_block* next = cur->next; | |
7889 | |
7890 xml_memory::deallocate(cur); | |
7891 | |
7892 cur = next; | |
7893 } | |
7894 } | |
7895 }; | |
7896 | |
7897 struct xpath_allocator_capture | |
7898 { | |
7899 xpath_allocator_capture(xpath_allocator* alloc): _target(alloc), _state(*alloc) | |
7900 { | |
7901 } | |
7902 | |
7903 ~xpath_allocator_capture() | |
7904 { | |
7905 _target->revert(_state); | |
7906 } | |
7907 | |
7908 xpath_allocator* _target; | |
7909 xpath_allocator _state; | |
7910 }; | |
7911 | |
7912 struct xpath_stack | |
7913 { | |
7914 xpath_allocator* result; | |
7915 xpath_allocator* temp; | |
7916 }; | |
7917 | |
7918 struct xpath_stack_data | |
7919 { | |
7920 xpath_memory_block blocks[2]; | |
7921 xpath_allocator result; | |
7922 xpath_allocator temp; | |
7923 xpath_stack stack; | |
7924 bool oom; | |
7925 | |
7926 xpath_stack_data(): result(blocks + 0, &oom), temp(blocks + 1, &oom), oom(false) | |
7927 { | |
7928 blocks[0].next = blocks[1].next = 0; | |
7929 blocks[0].capacity = blocks[1].capacity = sizeof(blocks[0].data); | |
7930 | |
7931 stack.result = &result; | |
7932 stack.temp = &temp; | |
7933 } | |
7934 | |
7935 ~xpath_stack_data() | |
7936 { | |
7937 result.release(); | |
7938 temp.release(); | |
7939 } | |
7940 }; | |
7941 PUGI__NS_END | |
7942 | |
7943 // String class | |
7944 PUGI__NS_BEGIN | |
7945 class xpath_string | |
7946 { | |
7947 const char_t* _buffer; | |
7948 bool _uses_heap; | |
7949 size_t _length_heap; | |
7950 | |
7951 static char_t* duplicate_string(const char_t* string, size_t length, xpath_allocator* alloc) | |
7952 { | |
7953 char_t* result = static_cast<char_t*>(alloc->allocate((length + 1) * sizeof(char_t))); | |
7954 if (!result) return 0; | |
7955 | |
7956 memcpy(result, string, length * sizeof(char_t)); | |
7957 result[length] = 0; | |
7958 | |
7959 return result; | |
7960 } | |
7961 | |
7962 xpath_string(const char_t* buffer, bool uses_heap_, size_t length_heap): _buffer(buffer), _uses_heap(uses_heap_), _length_heap(length_heap) | |
7963 { | |
7964 } | |
7965 | |
7966 public: | |
7967 static xpath_string from_const(const char_t* str) | |
7968 { | |
7969 return xpath_string(str, false, 0); | |
7970 } | |
7971 | |
7972 static xpath_string from_heap_preallocated(const char_t* begin, const char_t* end) | |
7973 { | |
7974 assert(begin <= end && *end == 0); | |
7975 | |
7976 return xpath_string(begin, true, static_cast<size_t>(end - begin)); | |
7977 } | |
7978 | |
7979 static xpath_string from_heap(const char_t* begin, const char_t* end, xpath_allocator* alloc) | |
7980 { | |
7981 assert(begin <= end); | |
7982 | |
7983 if (begin == end) | |
7984 return xpath_string(); | |
7985 | |
7986 size_t length = static_cast<size_t>(end - begin); | |
7987 const char_t* data = duplicate_string(begin, length, alloc); | |
7988 | |
7989 return data ? xpath_string(data, true, length) : xpath_string(); | |
7990 } | |
7991 | |
7992 xpath_string(): _buffer(PUGIXML_TEXT("")), _uses_heap(false), _length_heap(0) | |
7993 { | |
7994 } | |
7995 | |
7996 void append(const xpath_string& o, xpath_allocator* alloc) | |
7997 { | |
7998 // skip empty sources | |
7999 if (!*o._buffer) return; | |
8000 | |
8001 // fast append for constant empty target and constant source | |
8002 if (!*_buffer && !_uses_heap && !o._uses_heap) | |
8003 { | |
8004 _buffer = o._buffer; | |
8005 } | |
8006 else | |
8007 { | |
8008 // need to make heap copy | |
8009 size_t target_length = length(); | |
8010 size_t source_length = o.length(); | |
8011 size_t result_length = target_length + source_length; | |
8012 | |
8013 // allocate new buffer | |
8014 char_t* result = static_cast<char_t*>(alloc->reallocate(_uses_heap ? const_cast<char_t*>(_buffer) : 0, (target_length + 1) * sizeof(char_t), (result_length + 1) * sizeof(char_t))); | |
8015 if (!result) return; | |
8016 | |
8017 // append first string to the new buffer in case there was no reallocation | |
8018 if (!_uses_heap) memcpy(result, _buffer, target_length * sizeof(char_t)); | |
8019 | |
8020 // append second string to the new buffer | |
8021 memcpy(result + target_length, o._buffer, source_length * sizeof(char_t)); | |
8022 result[result_length] = 0; | |
8023 | |
8024 // finalize | |
8025 _buffer = result; | |
8026 _uses_heap = true; | |
8027 _length_heap = result_length; | |
8028 } | |
8029 } | |
8030 | |
8031 const char_t* c_str() const | |
8032 { | |
8033 return _buffer; | |
8034 } | |
8035 | |
8036 size_t length() const | |
8037 { | |
8038 return _uses_heap ? _length_heap : strlength(_buffer); | |
8039 } | |
8040 | |
8041 char_t* data(xpath_allocator* alloc) | |
8042 { | |
8043 // make private heap copy | |
8044 if (!_uses_heap) | |
8045 { | |
8046 size_t length_ = strlength(_buffer); | |
8047 const char_t* data_ = duplicate_string(_buffer, length_, alloc); | |
8048 | |
8049 if (!data_) return 0; | |
8050 | |
8051 _buffer = data_; | |
8052 _uses_heap = true; | |
8053 _length_heap = length_; | |
8054 } | |
8055 | |
8056 return const_cast<char_t*>(_buffer); | |
8057 } | |
8058 | |
8059 bool empty() const | |
8060 { | |
8061 return *_buffer == 0; | |
8062 } | |
8063 | |
8064 bool operator==(const xpath_string& o) const | |
8065 { | |
8066 return strequal(_buffer, o._buffer); | |
8067 } | |
8068 | |
8069 bool operator!=(const xpath_string& o) const | |
8070 { | |
8071 return !strequal(_buffer, o._buffer); | |
8072 } | |
8073 | |
8074 bool uses_heap() const | |
8075 { | |
8076 return _uses_heap; | |
8077 } | |
8078 }; | |
8079 PUGI__NS_END | |
8080 | |
8081 PUGI__NS_BEGIN | |
8082 PUGI__FN bool starts_with(const char_t* string, const char_t* pattern) | |
8083 { | |
8084 while (*pattern && *string == *pattern) | |
8085 { | |
8086 string++; | |
8087 pattern++; | |
8088 } | |
8089 | |
8090 return *pattern == 0; | |
8091 } | |
8092 | |
8093 PUGI__FN const char_t* find_char(const char_t* s, char_t c) | |
8094 { | |
8095 #ifdef PUGIXML_WCHAR_MODE | |
8096 return wcschr(s, c); | |
8097 #else | |
8098 return strchr(s, c); | |
8099 #endif | |
8100 } | |
8101 | |
8102 PUGI__FN const char_t* find_substring(const char_t* s, const char_t* p) | |
8103 { | |
8104 #ifdef PUGIXML_WCHAR_MODE | |
8105 // MSVC6 wcsstr bug workaround (if s is empty it always returns 0) | |
8106 return (*p == 0) ? s : wcsstr(s, p); | |
8107 #else | |
8108 return strstr(s, p); | |
8109 #endif | |
8110 } | |
8111 | |
8112 // Converts symbol to lower case, if it is an ASCII one | |
8113 PUGI__FN char_t tolower_ascii(char_t ch) | |
8114 { | |
8115 return static_cast<unsigned int>(ch - 'A') < 26 ? static_cast<char_t>(ch | ' ') : ch; | |
8116 } | |
8117 | |
8118 PUGI__FN xpath_string string_value(const xpath_node& na, xpath_allocator* alloc) | |
8119 { | |
8120 if (na.attribute()) | |
8121 return xpath_string::from_const(na.attribute().value()); | |
8122 else | |
8123 { | |
8124 xml_node n = na.node(); | |
8125 | |
8126 switch (n.type()) | |
8127 { | |
8128 case node_pcdata: | |
8129 case node_cdata: | |
8130 case node_comment: | |
8131 case node_pi: | |
8132 return xpath_string::from_const(n.value()); | |
8133 | |
8134 case node_document: | |
8135 case node_element: | |
8136 { | |
8137 xpath_string result; | |
8138 | |
8139 // element nodes can have value if parse_embed_pcdata was used | |
8140 if (n.value()[0]) | |
8141 result.append(xpath_string::from_const(n.value()), alloc); | |
8142 | |
8143 xml_node cur = n.first_child(); | |
8144 | |
8145 while (cur && cur != n) | |
8146 { | |
8147 if (cur.type() == node_pcdata || cur.type() == node_cdata) | |
8148 result.append(xpath_string::from_const(cur.value()), alloc); | |
8149 | |
8150 if (cur.first_child()) | |
8151 cur = cur.first_child(); | |
8152 else if (cur.next_sibling()) | |
8153 cur = cur.next_sibling(); | |
8154 else | |
8155 { | |
8156 while (!cur.next_sibling() && cur != n) | |
8157 cur = cur.parent(); | |
8158 | |
8159 if (cur != n) cur = cur.next_sibling(); | |
8160 } | |
8161 } | |
8162 | |
8163 return result; | |
8164 } | |
8165 | |
8166 default: | |
8167 return xpath_string(); | |
8168 } | |
8169 } | |
8170 } | |
8171 | |
8172 PUGI__FN bool node_is_before_sibling(xml_node_struct* ln, xml_node_struct* rn) | |
8173 { | |
8174 assert(ln->parent == rn->parent); | |
8175 | |
8176 // there is no common ancestor (the shared parent is null), nodes are from different documents | |
8177 if (!ln->parent) return ln < rn; | |
8178 | |
8179 // determine sibling order | |
8180 xml_node_struct* ls = ln; | |
8181 xml_node_struct* rs = rn; | |
8182 | |
8183 while (ls && rs) | |
8184 { | |
8185 if (ls == rn) return true; | |
8186 if (rs == ln) return false; | |
8187 | |
8188 ls = ls->next_sibling; | |
8189 rs = rs->next_sibling; | |
8190 } | |
8191 | |
8192 // if rn sibling chain ended ln must be before rn | |
8193 return !rs; | |
8194 } | |
8195 | |
8196 PUGI__FN bool node_is_before(xml_node_struct* ln, xml_node_struct* rn) | |
8197 { | |
8198 // find common ancestor at the same depth, if any | |
8199 xml_node_struct* lp = ln; | |
8200 xml_node_struct* rp = rn; | |
8201 | |
8202 while (lp && rp && lp->parent != rp->parent) | |
8203 { | |
8204 lp = lp->parent; | |
8205 rp = rp->parent; | |
8206 } | |
8207 | |
8208 // parents are the same! | |
8209 if (lp && rp) return node_is_before_sibling(lp, rp); | |
8210 | |
8211 // nodes are at different depths, need to normalize heights | |
8212 bool left_higher = !lp; | |
8213 | |
8214 while (lp) | |
8215 { | |
8216 lp = lp->parent; | |
8217 ln = ln->parent; | |
8218 } | |
8219 | |
8220 while (rp) | |
8221 { | |
8222 rp = rp->parent; | |
8223 rn = rn->parent; | |
8224 } | |
8225 | |
8226 // one node is the ancestor of the other | |
8227 if (ln == rn) return left_higher; | |
8228 | |
8229 // find common ancestor... again | |
8230 while (ln->parent != rn->parent) | |
8231 { | |
8232 ln = ln->parent; | |
8233 rn = rn->parent; | |
8234 } | |
8235 | |
8236 return node_is_before_sibling(ln, rn); | |
8237 } | |
8238 | |
8239 PUGI__FN bool node_is_ancestor(xml_node_struct* parent, xml_node_struct* node) | |
8240 { | |
8241 while (node && node != parent) node = node->parent; | |
8242 | |
8243 return parent && node == parent; | |
8244 } | |
8245 | |
8246 PUGI__FN const void* document_buffer_order(const xpath_node& xnode) | |
8247 { | |
8248 xml_node_struct* node = xnode.node().internal_object(); | |
8249 | |
8250 if (node) | |
8251 { | |
8252 if ((get_document(node).header & xml_memory_page_contents_shared_mask) == 0) | |
8253 { | |
8254 if (node->name && (node->header & impl::xml_memory_page_name_allocated_or_shared_mask) == 0) return node->name; | |
8255 if (node->value && (node->header & impl::xml_memory_page_value_allocated_or_shared_mask) == 0) return node->value; | |
8256 } | |
8257 | |
8258 return 0; | |
8259 } | |
8260 | |
8261 xml_attribute_struct* attr = xnode.attribute().internal_object(); | |
8262 | |
8263 if (attr) | |
8264 { | |
8265 if ((get_document(attr).header & xml_memory_page_contents_shared_mask) == 0) | |
8266 { | |
8267 if ((attr->header & impl::xml_memory_page_name_allocated_or_shared_mask) == 0) return attr->name; | |
8268 if ((attr->header & impl::xml_memory_page_value_allocated_or_shared_mask) == 0) return attr->value; | |
8269 } | |
8270 | |
8271 return 0; | |
8272 } | |
8273 | |
8274 return 0; | |
8275 } | |
8276 | |
8277 struct document_order_comparator | |
8278 { | |
8279 bool operator()(const xpath_node& lhs, const xpath_node& rhs) const | |
8280 { | |
8281 // optimized document order based check | |
8282 const void* lo = document_buffer_order(lhs); | |
8283 const void* ro = document_buffer_order(rhs); | |
8284 | |
8285 if (lo && ro) return lo < ro; | |
8286 | |
8287 // slow comparison | |
8288 xml_node ln = lhs.node(), rn = rhs.node(); | |
8289 | |
8290 // compare attributes | |
8291 if (lhs.attribute() && rhs.attribute()) | |
8292 { | |
8293 // shared parent | |
8294 if (lhs.parent() == rhs.parent()) | |
8295 { | |
8296 // determine sibling order | |
8297 for (xml_attribute a = lhs.attribute(); a; a = a.next_attribute()) | |
8298 if (a == rhs.attribute()) | |
8299 return true; | |
8300 | |
8301 return false; | |
8302 } | |
8303 | |
8304 // compare attribute parents | |
8305 ln = lhs.parent(); | |
8306 rn = rhs.parent(); | |
8307 } | |
8308 else if (lhs.attribute()) | |
8309 { | |
8310 // attributes go after the parent element | |
8311 if (lhs.parent() == rhs.node()) return false; | |
8312 | |
8313 ln = lhs.parent(); | |
8314 } | |
8315 else if (rhs.attribute()) | |
8316 { | |
8317 // attributes go after the parent element | |
8318 if (rhs.parent() == lhs.node()) return true; | |
8319 | |
8320 rn = rhs.parent(); | |
8321 } | |
8322 | |
8323 if (ln == rn) return false; | |
8324 | |
8325 if (!ln || !rn) return ln < rn; | |
8326 | |
8327 return node_is_before(ln.internal_object(), rn.internal_object()); | |
8328 } | |
8329 }; | |
8330 | |
8331 PUGI__FN double gen_nan() | |
8332 { | |
8333 #if defined(__STDC_IEC_559__) || ((FLT_RADIX - 0 == 2) && (FLT_MAX_EXP - 0 == 128) && (FLT_MANT_DIG - 0 == 24)) | |
8334 PUGI__STATIC_ASSERT(sizeof(float) == sizeof(uint32_t)); | |
8335 typedef uint32_t UI; // BCC5 workaround | |
8336 union { float f; UI i; } u; | |
8337 u.i = 0x7fc00000; | |
8338 return double(u.f); | |
8339 #else | |
8340 // fallback | |
8341 const volatile double zero = 0.0; | |
8342 return zero / zero; | |
8343 #endif | |
8344 } | |
8345 | |
8346 PUGI__FN bool is_nan(double value) | |
8347 { | |
8348 #if defined(PUGI__MSVC_CRT_VERSION) || defined(__BORLANDC__) | |
8349 return !!_isnan(value); | |
8350 #elif defined(fpclassify) && defined(FP_NAN) | |
8351 return fpclassify(value) == FP_NAN; | |
8352 #else | |
8353 // fallback | |
8354 const volatile double v = value; | |
8355 return v != v; | |
8356 #endif | |
8357 } | |
8358 | |
8359 PUGI__FN const char_t* convert_number_to_string_special(double value) | |
8360 { | |
8361 #if defined(PUGI__MSVC_CRT_VERSION) || defined(__BORLANDC__) | |
8362 if (_finite(value)) return (value == 0) ? PUGIXML_TEXT("0") : 0; | |
8363 if (_isnan(value)) return PUGIXML_TEXT("NaN"); | |
8364 return value > 0 ? PUGIXML_TEXT("Infinity") : PUGIXML_TEXT("-Infinity"); | |
8365 #elif defined(fpclassify) && defined(FP_NAN) && defined(FP_INFINITE) && defined(FP_ZERO) | |
8366 switch (fpclassify(value)) | |
8367 { | |
8368 case FP_NAN: | |
8369 return PUGIXML_TEXT("NaN"); | |
8370 | |
8371 case FP_INFINITE: | |
8372 return value > 0 ? PUGIXML_TEXT("Infinity") : PUGIXML_TEXT("-Infinity"); | |
8373 | |
8374 case FP_ZERO: | |
8375 return PUGIXML_TEXT("0"); | |
8376 | |
8377 default: | |
8378 return 0; | |
8379 } | |
8380 #else | |
8381 // fallback | |
8382 const volatile double v = value; | |
8383 | |
8384 if (v == 0) return PUGIXML_TEXT("0"); | |
8385 if (v != v) return PUGIXML_TEXT("NaN"); | |
8386 if (v * 2 == v) return value > 0 ? PUGIXML_TEXT("Infinity") : PUGIXML_TEXT("-Infinity"); | |
8387 return 0; | |
8388 #endif | |
8389 } | |
8390 | |
8391 PUGI__FN bool convert_number_to_boolean(double value) | |
8392 { | |
8393 return (value != 0 && !is_nan(value)); | |
8394 } | |
8395 | |
8396 PUGI__FN void truncate_zeros(char* begin, char* end) | |
8397 { | |
8398 while (begin != end && end[-1] == '0') end--; | |
8399 | |
8400 *end = 0; | |
8401 } | |
8402 | |
8403 // gets mantissa digits in the form of 0.xxxxx with 0. implied and the exponent | |
8404 #if defined(PUGI__MSVC_CRT_VERSION) && PUGI__MSVC_CRT_VERSION >= 1400 | |
8405 PUGI__FN void convert_number_to_mantissa_exponent(double value, char (&buffer)[32], char** out_mantissa, int* out_exponent) | |
8406 { | |
8407 // get base values | |
8408 int sign, exponent; | |
8409 _ecvt_s(buffer, sizeof(buffer), value, DBL_DIG + 1, &exponent, &sign); | |
8410 | |
8411 // truncate redundant zeros | |
8412 truncate_zeros(buffer, buffer + strlen(buffer)); | |
8413 | |
8414 // fill results | |
8415 *out_mantissa = buffer; | |
8416 *out_exponent = exponent; | |
8417 } | |
8418 #else | |
8419 PUGI__FN void convert_number_to_mantissa_exponent(double value, char (&buffer)[32], char** out_mantissa, int* out_exponent) | |
8420 { | |
8421 // get a scientific notation value with IEEE DBL_DIG decimals | |
8422 PUGI__SNPRINTF(buffer, "%.*e", DBL_DIG, value); | |
8423 | |
8424 // get the exponent (possibly negative) | |
8425 char* exponent_string = strchr(buffer, 'e'); | |
8426 assert(exponent_string); | |
8427 | |
8428 int exponent = atoi(exponent_string + 1); | |
8429 | |
8430 // extract mantissa string: skip sign | |
8431 char* mantissa = buffer[0] == '-' ? buffer + 1 : buffer; | |
8432 assert(mantissa[0] != '0' && mantissa[1] == '.'); | |
8433 | |
8434 // divide mantissa by 10 to eliminate integer part | |
8435 mantissa[1] = mantissa[0]; | |
8436 mantissa++; | |
8437 exponent++; | |
8438 | |
8439 // remove extra mantissa digits and zero-terminate mantissa | |
8440 truncate_zeros(mantissa, exponent_string); | |
8441 | |
8442 // fill results | |
8443 *out_mantissa = mantissa; | |
8444 *out_exponent = exponent; | |
8445 } | |
8446 #endif | |
8447 | |
8448 PUGI__FN xpath_string convert_number_to_string(double value, xpath_allocator* alloc) | |
8449 { | |
8450 // try special number conversion | |
8451 const char_t* special = convert_number_to_string_special(value); | |
8452 if (special) return xpath_string::from_const(special); | |
8453 | |
8454 // get mantissa + exponent form | |
8455 char mantissa_buffer[32]; | |
8456 | |
8457 char* mantissa; | |
8458 int exponent; | |
8459 convert_number_to_mantissa_exponent(value, mantissa_buffer, &mantissa, &exponent); | |
8460 | |
8461 // allocate a buffer of suitable length for the number | |
8462 size_t result_size = strlen(mantissa_buffer) + (exponent > 0 ? exponent : -exponent) + 4; | |
8463 char_t* result = static_cast<char_t*>(alloc->allocate(sizeof(char_t) * result_size)); | |
8464 if (!result) return xpath_string(); | |
8465 | |
8466 // make the number! | |
8467 char_t* s = result; | |
8468 | |
8469 // sign | |
8470 if (value < 0) *s++ = '-'; | |
8471 | |
8472 // integer part | |
8473 if (exponent <= 0) | |
8474 { | |
8475 *s++ = '0'; | |
8476 } | |
8477 else | |
8478 { | |
8479 while (exponent > 0) | |
8480 { | |
8481 assert(*mantissa == 0 || static_cast<unsigned int>(*mantissa - '0') <= 9); | |
8482 *s++ = *mantissa ? *mantissa++ : '0'; | |
8483 exponent--; | |
8484 } | |
8485 } | |
8486 | |
8487 // fractional part | |
8488 if (*mantissa) | |
8489 { | |
8490 // decimal point | |
8491 *s++ = '.'; | |
8492 | |
8493 // extra zeroes from negative exponent | |
8494 while (exponent < 0) | |
8495 { | |
8496 *s++ = '0'; | |
8497 exponent++; | |
8498 } | |
8499 | |
8500 // extra mantissa digits | |
8501 while (*mantissa) | |
8502 { | |
8503 assert(static_cast<unsigned int>(*mantissa - '0') <= 9); | |
8504 *s++ = *mantissa++; | |
8505 } | |
8506 } | |
8507 | |
8508 // zero-terminate | |
8509 assert(s < result + result_size); | |
8510 *s = 0; | |
8511 | |
8512 return xpath_string::from_heap_preallocated(result, s); | |
8513 } | |
8514 | |
8515 PUGI__FN bool check_string_to_number_format(const char_t* string) | |
8516 { | |
8517 // parse leading whitespace | |
8518 while (PUGI__IS_CHARTYPE(*string, ct_space)) ++string; | |
8519 | |
8520 // parse sign | |
8521 if (*string == '-') ++string; | |
8522 | |
8523 if (!*string) return false; | |
8524 | |
8525 // if there is no integer part, there should be a decimal part with at least one digit | |
8526 if (!PUGI__IS_CHARTYPEX(string[0], ctx_digit) && (string[0] != '.' || !PUGI__IS_CHARTYPEX(string[1], ctx_digit))) return false; | |
8527 | |
8528 // parse integer part | |
8529 while (PUGI__IS_CHARTYPEX(*string, ctx_digit)) ++string; | |
8530 | |
8531 // parse decimal part | |
8532 if (*string == '.') | |
8533 { | |
8534 ++string; | |
8535 | |
8536 while (PUGI__IS_CHARTYPEX(*string, ctx_digit)) ++string; | |
8537 } | |
8538 | |
8539 // parse trailing whitespace | |
8540 while (PUGI__IS_CHARTYPE(*string, ct_space)) ++string; | |
8541 | |
8542 return *string == 0; | |
8543 } | |
8544 | |
8545 PUGI__FN double convert_string_to_number(const char_t* string) | |
8546 { | |
8547 // check string format | |
8548 if (!check_string_to_number_format(string)) return gen_nan(); | |
8549 | |
8550 // parse string | |
8551 #ifdef PUGIXML_WCHAR_MODE | |
8552 return wcstod(string, 0); | |
8553 #else | |
8554 return strtod(string, 0); | |
8555 #endif | |
8556 } | |
8557 | |
8558 PUGI__FN bool convert_string_to_number_scratch(char_t (&buffer)[32], const char_t* begin, const char_t* end, double* out_result) | |
8559 { | |
8560 size_t length = static_cast<size_t>(end - begin); | |
8561 char_t* scratch = buffer; | |
8562 | |
8563 if (length >= sizeof(buffer) / sizeof(buffer[0])) | |
8564 { | |
8565 // need to make dummy on-heap copy | |
8566 scratch = static_cast<char_t*>(xml_memory::allocate((length + 1) * sizeof(char_t))); | |
8567 if (!scratch) return false; | |
8568 } | |
8569 | |
8570 // copy string to zero-terminated buffer and perform conversion | |
8571 memcpy(scratch, begin, length * sizeof(char_t)); | |
8572 scratch[length] = 0; | |
8573 | |
8574 *out_result = convert_string_to_number(scratch); | |
8575 | |
8576 // free dummy buffer | |
8577 if (scratch != buffer) xml_memory::deallocate(scratch); | |
8578 | |
8579 return true; | |
8580 } | |
8581 | |
8582 PUGI__FN double round_nearest(double value) | |
8583 { | |
8584 return floor(value + 0.5); | |
8585 } | |
8586 | |
8587 PUGI__FN double round_nearest_nzero(double value) | |
8588 { | |
8589 // same as round_nearest, but returns -0 for [-0.5, -0] | |
8590 // ceil is used to differentiate between +0 and -0 (we return -0 for [-0.5, -0] and +0 for +0) | |
8591 return (value >= -0.5 && value <= 0) ? ceil(value) : floor(value + 0.5); | |
8592 } | |
8593 | |
8594 PUGI__FN const char_t* qualified_name(const xpath_node& node) | |
8595 { | |
8596 return node.attribute() ? node.attribute().name() : node.node().name(); | |
8597 } | |
8598 | |
8599 PUGI__FN const char_t* local_name(const xpath_node& node) | |
8600 { | |
8601 const char_t* name = qualified_name(node); | |
8602 const char_t* p = find_char(name, ':'); | |
8603 | |
8604 return p ? p + 1 : name; | |
8605 } | |
8606 | |
8607 struct namespace_uri_predicate | |
8608 { | |
8609 const char_t* prefix; | |
8610 size_t prefix_length; | |
8611 | |
8612 namespace_uri_predicate(const char_t* name) | |
8613 { | |
8614 const char_t* pos = find_char(name, ':'); | |
8615 | |
8616 prefix = pos ? name : 0; | |
8617 prefix_length = pos ? static_cast<size_t>(pos - name) : 0; | |
8618 } | |
8619 | |
8620 bool operator()(xml_attribute a) const | |
8621 { | |
8622 const char_t* name = a.name(); | |
8623 | |
8624 if (!starts_with(name, PUGIXML_TEXT("xmlns"))) return false; | |
8625 | |
8626 return prefix ? name[5] == ':' && strequalrange(name + 6, prefix, prefix_length) : name[5] == 0; | |
8627 } | |
8628 }; | |
8629 | |
8630 PUGI__FN const char_t* namespace_uri(xml_node node) | |
8631 { | |
8632 namespace_uri_predicate pred = node.name(); | |
8633 | |
8634 xml_node p = node; | |
8635 | |
8636 while (p) | |
8637 { | |
8638 xml_attribute a = p.find_attribute(pred); | |
8639 | |
8640 if (a) return a.value(); | |
8641 | |
8642 p = p.parent(); | |
8643 } | |
8644 | |
8645 return PUGIXML_TEXT(""); | |
8646 } | |
8647 | |
8648 PUGI__FN const char_t* namespace_uri(xml_attribute attr, xml_node parent) | |
8649 { | |
8650 namespace_uri_predicate pred = attr.name(); | |
8651 | |
8652 // Default namespace does not apply to attributes | |
8653 if (!pred.prefix) return PUGIXML_TEXT(""); | |
8654 | |
8655 xml_node p = parent; | |
8656 | |
8657 while (p) | |
8658 { | |
8659 xml_attribute a = p.find_attribute(pred); | |
8660 | |
8661 if (a) return a.value(); | |
8662 | |
8663 p = p.parent(); | |
8664 } | |
8665 | |
8666 return PUGIXML_TEXT(""); | |
8667 } | |
8668 | |
8669 PUGI__FN const char_t* namespace_uri(const xpath_node& node) | |
8670 { | |
8671 return node.attribute() ? namespace_uri(node.attribute(), node.parent()) : namespace_uri(node.node()); | |
8672 } | |
8673 | |
8674 PUGI__FN char_t* normalize_space(char_t* buffer) | |
8675 { | |
8676 char_t* write = buffer; | |
8677 | |
8678 for (char_t* it = buffer; *it; ) | |
8679 { | |
8680 char_t ch = *it++; | |
8681 | |
8682 if (PUGI__IS_CHARTYPE(ch, ct_space)) | |
8683 { | |
8684 // replace whitespace sequence with single space | |
8685 while (PUGI__IS_CHARTYPE(*it, ct_space)) it++; | |
8686 | |
8687 // avoid leading spaces | |
8688 if (write != buffer) *write++ = ' '; | |
8689 } | |
8690 else *write++ = ch; | |
8691 } | |
8692 | |
8693 // remove trailing space | |
8694 if (write != buffer && PUGI__IS_CHARTYPE(write[-1], ct_space)) write--; | |
8695 | |
8696 // zero-terminate | |
8697 *write = 0; | |
8698 | |
8699 return write; | |
8700 } | |
8701 | |
8702 PUGI__FN char_t* translate(char_t* buffer, const char_t* from, const char_t* to, size_t to_length) | |
8703 { | |
8704 char_t* write = buffer; | |
8705 | |
8706 while (*buffer) | |
8707 { | |
8708 PUGI__DMC_VOLATILE char_t ch = *buffer++; | |
8709 | |
8710 const char_t* pos = find_char(from, ch); | |
8711 | |
8712 if (!pos) | |
8713 *write++ = ch; // do not process | |
8714 else if (static_cast<size_t>(pos - from) < to_length) | |
8715 *write++ = to[pos - from]; // replace | |
8716 } | |
8717 | |
8718 // zero-terminate | |
8719 *write = 0; | |
8720 | |
8721 return write; | |
8722 } | |
8723 | |
8724 PUGI__FN unsigned char* translate_table_generate(xpath_allocator* alloc, const char_t* from, const char_t* to) | |
8725 { | |
8726 unsigned char table[128] = {0}; | |
8727 | |
8728 while (*from) | |
8729 { | |
8730 unsigned int fc = static_cast<unsigned int>(*from); | |
8731 unsigned int tc = static_cast<unsigned int>(*to); | |
8732 | |
8733 if (fc >= 128 || tc >= 128) | |
8734 return 0; | |
8735 | |
8736 // code=128 means "skip character" | |
8737 if (!table[fc]) | |
8738 table[fc] = static_cast<unsigned char>(tc ? tc : 128); | |
8739 | |
8740 from++; | |
8741 if (tc) to++; | |
8742 } | |
8743 | |
8744 for (int i = 0; i < 128; ++i) | |
8745 if (!table[i]) | |
8746 table[i] = static_cast<unsigned char>(i); | |
8747 | |
8748 void* result = alloc->allocate(sizeof(table)); | |
8749 if (!result) return 0; | |
8750 | |
8751 memcpy(result, table, sizeof(table)); | |
8752 | |
8753 return static_cast<unsigned char*>(result); | |
8754 } | |
8755 | |
8756 PUGI__FN char_t* translate_table(char_t* buffer, const unsigned char* table) | |
8757 { | |
8758 char_t* write = buffer; | |
8759 | |
8760 while (*buffer) | |
8761 { | |
8762 char_t ch = *buffer++; | |
8763 unsigned int index = static_cast<unsigned int>(ch); | |
8764 | |
8765 if (index < 128) | |
8766 { | |
8767 unsigned char code = table[index]; | |
8768 | |
8769 // code=128 means "skip character" (table size is 128 so 128 can be a special value) | |
8770 // this code skips these characters without extra branches | |
8771 *write = static_cast<char_t>(code); | |
8772 write += 1 - (code >> 7); | |
8773 } | |
8774 else | |
8775 { | |
8776 *write++ = ch; | |
8777 } | |
8778 } | |
8779 | |
8780 // zero-terminate | |
8781 *write = 0; | |
8782 | |
8783 return write; | |
8784 } | |
8785 | |
8786 inline bool is_xpath_attribute(const char_t* name) | |
8787 { | |
8788 return !(starts_with(name, PUGIXML_TEXT("xmlns")) && (name[5] == 0 || name[5] == ':')); | |
8789 } | |
8790 | |
8791 struct xpath_variable_boolean: xpath_variable | |
8792 { | |
8793 xpath_variable_boolean(): xpath_variable(xpath_type_boolean), value(false) | |
8794 { | |
8795 } | |
8796 | |
8797 bool value; | |
8798 char_t name[1]; | |
8799 }; | |
8800 | |
8801 struct xpath_variable_number: xpath_variable | |
8802 { | |
8803 xpath_variable_number(): xpath_variable(xpath_type_number), value(0) | |
8804 { | |
8805 } | |
8806 | |
8807 double value; | |
8808 char_t name[1]; | |
8809 }; | |
8810 | |
8811 struct xpath_variable_string: xpath_variable | |
8812 { | |
8813 xpath_variable_string(): xpath_variable(xpath_type_string), value(0) | |
8814 { | |
8815 } | |
8816 | |
8817 ~xpath_variable_string() | |
8818 { | |
8819 if (value) xml_memory::deallocate(value); | |
8820 } | |
8821 | |
8822 char_t* value; | |
8823 char_t name[1]; | |
8824 }; | |
8825 | |
8826 struct xpath_variable_node_set: xpath_variable | |
8827 { | |
8828 xpath_variable_node_set(): xpath_variable(xpath_type_node_set) | |
8829 { | |
8830 } | |
8831 | |
8832 xpath_node_set value; | |
8833 char_t name[1]; | |
8834 }; | |
8835 | |
8836 static const xpath_node_set dummy_node_set; | |
8837 | |
8838 PUGI__FN PUGI__UNSIGNED_OVERFLOW unsigned int hash_string(const char_t* str) | |
8839 { | |
8840 // Jenkins one-at-a-time hash (http://en.wikipedia.org/wiki/Jenkins_hash_function#one-at-a-time) | |
8841 unsigned int result = 0; | |
8842 | |
8843 while (*str) | |
8844 { | |
8845 result += static_cast<unsigned int>(*str++); | |
8846 result += result << 10; | |
8847 result ^= result >> 6; | |
8848 } | |
8849 | |
8850 result += result << 3; | |
8851 result ^= result >> 11; | |
8852 result += result << 15; | |
8853 | |
8854 return result; | |
8855 } | |
8856 | |
8857 template <typename T> PUGI__FN T* new_xpath_variable(const char_t* name) | |
8858 { | |
8859 size_t length = strlength(name); | |
8860 if (length == 0) return 0; // empty variable names are invalid | |
8861 | |
8862 // $$ we can't use offsetof(T, name) because T is non-POD, so we just allocate additional length characters | |
8863 void* memory = xml_memory::allocate(sizeof(T) + length * sizeof(char_t)); | |
8864 if (!memory) return 0; | |
8865 | |
8866 T* result = new (memory) T(); | |
8867 | |
8868 memcpy(result->name, name, (length + 1) * sizeof(char_t)); | |
8869 | |
8870 return result; | |
8871 } | |
8872 | |
8873 PUGI__FN xpath_variable* new_xpath_variable(xpath_value_type type, const char_t* name) | |
8874 { | |
8875 switch (type) | |
8876 { | |
8877 case xpath_type_node_set: | |
8878 return new_xpath_variable<xpath_variable_node_set>(name); | |
8879 | |
8880 case xpath_type_number: | |
8881 return new_xpath_variable<xpath_variable_number>(name); | |
8882 | |
8883 case xpath_type_string: | |
8884 return new_xpath_variable<xpath_variable_string>(name); | |
8885 | |
8886 case xpath_type_boolean: | |
8887 return new_xpath_variable<xpath_variable_boolean>(name); | |
8888 | |
8889 default: | |
8890 return 0; | |
8891 } | |
8892 } | |
8893 | |
8894 template <typename T> PUGI__FN void delete_xpath_variable(T* var) | |
8895 { | |
8896 var->~T(); | |
8897 xml_memory::deallocate(var); | |
8898 } | |
8899 | |
8900 PUGI__FN void delete_xpath_variable(xpath_value_type type, xpath_variable* var) | |
8901 { | |
8902 switch (type) | |
8903 { | |
8904 case xpath_type_node_set: | |
8905 delete_xpath_variable(static_cast<xpath_variable_node_set*>(var)); | |
8906 break; | |
8907 | |
8908 case xpath_type_number: | |
8909 delete_xpath_variable(static_cast<xpath_variable_number*>(var)); | |
8910 break; | |
8911 | |
8912 case xpath_type_string: | |
8913 delete_xpath_variable(static_cast<xpath_variable_string*>(var)); | |
8914 break; | |
8915 | |
8916 case xpath_type_boolean: | |
8917 delete_xpath_variable(static_cast<xpath_variable_boolean*>(var)); | |
8918 break; | |
8919 | |
8920 default: | |
8921 assert(false && "Invalid variable type"); // unreachable | |
8922 } | |
8923 } | |
8924 | |
8925 PUGI__FN bool copy_xpath_variable(xpath_variable* lhs, const xpath_variable* rhs) | |
8926 { | |
8927 switch (rhs->type()) | |
8928 { | |
8929 case xpath_type_node_set: | |
8930 return lhs->set(static_cast<const xpath_variable_node_set*>(rhs)->value); | |
8931 | |
8932 case xpath_type_number: | |
8933 return lhs->set(static_cast<const xpath_variable_number*>(rhs)->value); | |
8934 | |
8935 case xpath_type_string: | |
8936 return lhs->set(static_cast<const xpath_variable_string*>(rhs)->value); | |
8937 | |
8938 case xpath_type_boolean: | |
8939 return lhs->set(static_cast<const xpath_variable_boolean*>(rhs)->value); | |
8940 | |
8941 default: | |
8942 assert(false && "Invalid variable type"); // unreachable | |
8943 return false; | |
8944 } | |
8945 } | |
8946 | |
8947 PUGI__FN bool get_variable_scratch(char_t (&buffer)[32], xpath_variable_set* set, const char_t* begin, const char_t* end, xpath_variable** out_result) | |
8948 { | |
8949 size_t length = static_cast<size_t>(end - begin); | |
8950 char_t* scratch = buffer; | |
8951 | |
8952 if (length >= sizeof(buffer) / sizeof(buffer[0])) | |
8953 { | |
8954 // need to make dummy on-heap copy | |
8955 scratch = static_cast<char_t*>(xml_memory::allocate((length + 1) * sizeof(char_t))); | |
8956 if (!scratch) return false; | |
8957 } | |
8958 | |
8959 // copy string to zero-terminated buffer and perform lookup | |
8960 memcpy(scratch, begin, length * sizeof(char_t)); | |
8961 scratch[length] = 0; | |
8962 | |
8963 *out_result = set->get(scratch); | |
8964 | |
8965 // free dummy buffer | |
8966 if (scratch != buffer) xml_memory::deallocate(scratch); | |
8967 | |
8968 return true; | |
8969 } | |
8970 PUGI__NS_END | |
8971 | |
8972 // Internal node set class | |
8973 PUGI__NS_BEGIN | |
8974 PUGI__FN xpath_node_set::type_t xpath_get_order(const xpath_node* begin, const xpath_node* end) | |
8975 { | |
8976 if (end - begin < 2) | |
8977 return xpath_node_set::type_sorted; | |
8978 | |
8979 document_order_comparator cmp; | |
8980 | |
8981 bool first = cmp(begin[0], begin[1]); | |
8982 | |
8983 for (const xpath_node* it = begin + 1; it + 1 < end; ++it) | |
8984 if (cmp(it[0], it[1]) != first) | |
8985 return xpath_node_set::type_unsorted; | |
8986 | |
8987 return first ? xpath_node_set::type_sorted : xpath_node_set::type_sorted_reverse; | |
8988 } | |
8989 | |
8990 PUGI__FN xpath_node_set::type_t xpath_sort(xpath_node* begin, xpath_node* end, xpath_node_set::type_t type, bool rev) | |
8991 { | |
8992 xpath_node_set::type_t order = rev ? xpath_node_set::type_sorted_reverse : xpath_node_set::type_sorted; | |
8993 | |
8994 if (type == xpath_node_set::type_unsorted) | |
8995 { | |
8996 xpath_node_set::type_t sorted = xpath_get_order(begin, end); | |
8997 | |
8998 if (sorted == xpath_node_set::type_unsorted) | |
8999 { | |
9000 sort(begin, end, document_order_comparator()); | |
9001 | |
9002 type = xpath_node_set::type_sorted; | |
9003 } | |
9004 else | |
9005 type = sorted; | |
9006 } | |
9007 | |
9008 if (type != order) reverse(begin, end); | |
9009 | |
9010 return order; | |
9011 } | |
9012 | |
9013 PUGI__FN xpath_node xpath_first(const xpath_node* begin, const xpath_node* end, xpath_node_set::type_t type) | |
9014 { | |
9015 if (begin == end) return xpath_node(); | |
9016 | |
9017 switch (type) | |
9018 { | |
9019 case xpath_node_set::type_sorted: | |
9020 return *begin; | |
9021 | |
9022 case xpath_node_set::type_sorted_reverse: | |
9023 return *(end - 1); | |
9024 | |
9025 case xpath_node_set::type_unsorted: | |
9026 return *min_element(begin, end, document_order_comparator()); | |
9027 | |
9028 default: | |
9029 assert(false && "Invalid node set type"); // unreachable | |
9030 return xpath_node(); | |
9031 } | |
9032 } | |
9033 | |
9034 class xpath_node_set_raw | |
9035 { | |
9036 xpath_node_set::type_t _type; | |
9037 | |
9038 xpath_node* _begin; | |
9039 xpath_node* _end; | |
9040 xpath_node* _eos; | |
9041 | |
9042 public: | |
9043 xpath_node_set_raw(): _type(xpath_node_set::type_unsorted), _begin(0), _end(0), _eos(0) | |
9044 { | |
9045 } | |
9046 | |
9047 xpath_node* begin() const | |
9048 { | |
9049 return _begin; | |
9050 } | |
9051 | |
9052 xpath_node* end() const | |
9053 { | |
9054 return _end; | |
9055 } | |
9056 | |
9057 bool empty() const | |
9058 { | |
9059 return _begin == _end; | |
9060 } | |
9061 | |
9062 size_t size() const | |
9063 { | |
9064 return static_cast<size_t>(_end - _begin); | |
9065 } | |
9066 | |
9067 xpath_node first() const | |
9068 { | |
9069 return xpath_first(_begin, _end, _type); | |
9070 } | |
9071 | |
9072 void push_back_grow(const xpath_node& node, xpath_allocator* alloc); | |
9073 | |
9074 void push_back(const xpath_node& node, xpath_allocator* alloc) | |
9075 { | |
9076 if (_end != _eos) | |
9077 *_end++ = node; | |
9078 else | |
9079 push_back_grow(node, alloc); | |
9080 } | |
9081 | |
9082 void append(const xpath_node* begin_, const xpath_node* end_, xpath_allocator* alloc) | |
9083 { | |
9084 if (begin_ == end_) return; | |
9085 | |
9086 size_t size_ = static_cast<size_t>(_end - _begin); | |
9087 size_t capacity = static_cast<size_t>(_eos - _begin); | |
9088 size_t count = static_cast<size_t>(end_ - begin_); | |
9089 | |
9090 if (size_ + count > capacity) | |
9091 { | |
9092 // reallocate the old array or allocate a new one | |
9093 xpath_node* data = static_cast<xpath_node*>(alloc->reallocate(_begin, capacity * sizeof(xpath_node), (size_ + count) * sizeof(xpath_node))); | |
9094 if (!data) return; | |
9095 | |
9096 // finalize | |
9097 _begin = data; | |
9098 _end = data + size_; | |
9099 _eos = data + size_ + count; | |
9100 } | |
9101 | |
9102 memcpy(_end, begin_, count * sizeof(xpath_node)); | |
9103 _end += count; | |
9104 } | |
9105 | |
9106 void sort_do() | |
9107 { | |
9108 _type = xpath_sort(_begin, _end, _type, false); | |
9109 } | |
9110 | |
9111 void truncate(xpath_node* pos) | |
9112 { | |
9113 assert(_begin <= pos && pos <= _end); | |
9114 | |
9115 _end = pos; | |
9116 } | |
9117 | |
9118 void remove_duplicates(xpath_allocator* alloc) | |
9119 { | |
9120 if (_type == xpath_node_set::type_unsorted && _end - _begin > 2) | |
9121 { | |
9122 xpath_allocator_capture cr(alloc); | |
9123 | |
9124 size_t size_ = static_cast<size_t>(_end - _begin); | |
9125 | |
9126 size_t hash_size = 1; | |
9127 while (hash_size < size_ + size_ / 2) hash_size *= 2; | |
9128 | |
9129 const void** hash_data = static_cast<const void**>(alloc->allocate(hash_size * sizeof(void**))); | |
9130 if (!hash_data) return; | |
9131 | |
9132 memset(hash_data, 0, hash_size * sizeof(const void**)); | |
9133 | |
9134 xpath_node* write = _begin; | |
9135 | |
9136 for (xpath_node* it = _begin; it != _end; ++it) | |
9137 { | |
9138 const void* attr = it->attribute().internal_object(); | |
9139 const void* node = it->node().internal_object(); | |
9140 const void* key = attr ? attr : node; | |
9141 | |
9142 if (key && hash_insert(hash_data, hash_size, key)) | |
9143 { | |
9144 *write++ = *it; | |
9145 } | |
9146 } | |
9147 | |
9148 _end = write; | |
9149 } | |
9150 else | |
9151 { | |
9152 _end = unique(_begin, _end); | |
9153 } | |
9154 } | |
9155 | |
9156 xpath_node_set::type_t type() const | |
9157 { | |
9158 return _type; | |
9159 } | |
9160 | |
9161 void set_type(xpath_node_set::type_t value) | |
9162 { | |
9163 _type = value; | |
9164 } | |
9165 }; | |
9166 | |
9167 PUGI__FN_NO_INLINE void xpath_node_set_raw::push_back_grow(const xpath_node& node, xpath_allocator* alloc) | |
9168 { | |
9169 size_t capacity = static_cast<size_t>(_eos - _begin); | |
9170 | |
9171 // get new capacity (1.5x rule) | |
9172 size_t new_capacity = capacity + capacity / 2 + 1; | |
9173 | |
9174 // reallocate the old array or allocate a new one | |
9175 xpath_node* data = static_cast<xpath_node*>(alloc->reallocate(_begin, capacity * sizeof(xpath_node), new_capacity * sizeof(xpath_node))); | |
9176 if (!data) return; | |
9177 | |
9178 // finalize | |
9179 _begin = data; | |
9180 _end = data + capacity; | |
9181 _eos = data + new_capacity; | |
9182 | |
9183 // push | |
9184 *_end++ = node; | |
9185 } | |
9186 PUGI__NS_END | |
9187 | |
9188 PUGI__NS_BEGIN | |
9189 struct xpath_context | |
9190 { | |
9191 xpath_node n; | |
9192 size_t position, size; | |
9193 | |
9194 xpath_context(const xpath_node& n_, size_t position_, size_t size_): n(n_), position(position_), size(size_) | |
9195 { | |
9196 } | |
9197 }; | |
9198 | |
9199 enum lexeme_t | |
9200 { | |
9201 lex_none = 0, | |
9202 lex_equal, | |
9203 lex_not_equal, | |
9204 lex_less, | |
9205 lex_greater, | |
9206 lex_less_or_equal, | |
9207 lex_greater_or_equal, | |
9208 lex_plus, | |
9209 lex_minus, | |
9210 lex_multiply, | |
9211 lex_union, | |
9212 lex_var_ref, | |
9213 lex_open_brace, | |
9214 lex_close_brace, | |
9215 lex_quoted_string, | |
9216 lex_number, | |
9217 lex_slash, | |
9218 lex_double_slash, | |
9219 lex_open_square_brace, | |
9220 lex_close_square_brace, | |
9221 lex_string, | |
9222 lex_comma, | |
9223 lex_axis_attribute, | |
9224 lex_dot, | |
9225 lex_double_dot, | |
9226 lex_double_colon, | |
9227 lex_eof | |
9228 }; | |
9229 | |
9230 struct xpath_lexer_string | |
9231 { | |
9232 const char_t* begin; | |
9233 const char_t* end; | |
9234 | |
9235 xpath_lexer_string(): begin(0), end(0) | |
9236 { | |
9237 } | |
9238 | |
9239 bool operator==(const char_t* other) const | |
9240 { | |
9241 size_t length = static_cast<size_t>(end - begin); | |
9242 | |
9243 return strequalrange(other, begin, length); | |
9244 } | |
9245 }; | |
9246 | |
9247 class xpath_lexer | |
9248 { | |
9249 const char_t* _cur; | |
9250 const char_t* _cur_lexeme_pos; | |
9251 xpath_lexer_string _cur_lexeme_contents; | |
9252 | |
9253 lexeme_t _cur_lexeme; | |
9254 | |
9255 public: | |
9256 explicit xpath_lexer(const char_t* query): _cur(query) | |
9257 { | |
9258 next(); | |
9259 } | |
9260 | |
9261 const char_t* state() const | |
9262 { | |
9263 return _cur; | |
9264 } | |
9265 | |
9266 void next() | |
9267 { | |
9268 const char_t* cur = _cur; | |
9269 | |
9270 while (PUGI__IS_CHARTYPE(*cur, ct_space)) ++cur; | |
9271 | |
9272 // save lexeme position for error reporting | |
9273 _cur_lexeme_pos = cur; | |
9274 | |
9275 switch (*cur) | |
9276 { | |
9277 case 0: | |
9278 _cur_lexeme = lex_eof; | |
9279 break; | |
9280 | |
9281 case '>': | |
9282 if (*(cur+1) == '=') | |
9283 { | |
9284 cur += 2; | |
9285 _cur_lexeme = lex_greater_or_equal; | |
9286 } | |
9287 else | |
9288 { | |
9289 cur += 1; | |
9290 _cur_lexeme = lex_greater; | |
9291 } | |
9292 break; | |
9293 | |
9294 case '<': | |
9295 if (*(cur+1) == '=') | |
9296 { | |
9297 cur += 2; | |
9298 _cur_lexeme = lex_less_or_equal; | |
9299 } | |
9300 else | |
9301 { | |
9302 cur += 1; | |
9303 _cur_lexeme = lex_less; | |
9304 } | |
9305 break; | |
9306 | |
9307 case '!': | |
9308 if (*(cur+1) == '=') | |
9309 { | |
9310 cur += 2; | |
9311 _cur_lexeme = lex_not_equal; | |
9312 } | |
9313 else | |
9314 { | |
9315 _cur_lexeme = lex_none; | |
9316 } | |
9317 break; | |
9318 | |
9319 case '=': | |
9320 cur += 1; | |
9321 _cur_lexeme = lex_equal; | |
9322 | |
9323 break; | |
9324 | |
9325 case '+': | |
9326 cur += 1; | |
9327 _cur_lexeme = lex_plus; | |
9328 | |
9329 break; | |
9330 | |
9331 case '-': | |
9332 cur += 1; | |
9333 _cur_lexeme = lex_minus; | |
9334 | |
9335 break; | |
9336 | |
9337 case '*': | |
9338 cur += 1; | |
9339 _cur_lexeme = lex_multiply; | |
9340 | |
9341 break; | |
9342 | |
9343 case '|': | |
9344 cur += 1; | |
9345 _cur_lexeme = lex_union; | |
9346 | |
9347 break; | |
9348 | |
9349 case '$': | |
9350 cur += 1; | |
9351 | |
9352 if (PUGI__IS_CHARTYPEX(*cur, ctx_start_symbol)) | |
9353 { | |
9354 _cur_lexeme_contents.begin = cur; | |
9355 | |
9356 while (PUGI__IS_CHARTYPEX(*cur, ctx_symbol)) cur++; | |
9357 | |
9358 if (cur[0] == ':' && PUGI__IS_CHARTYPEX(cur[1], ctx_symbol)) // qname | |
9359 { | |
9360 cur++; // : | |
9361 | |
9362 while (PUGI__IS_CHARTYPEX(*cur, ctx_symbol)) cur++; | |
9363 } | |
9364 | |
9365 _cur_lexeme_contents.end = cur; | |
9366 | |
9367 _cur_lexeme = lex_var_ref; | |
9368 } | |
9369 else | |
9370 { | |
9371 _cur_lexeme = lex_none; | |
9372 } | |
9373 | |
9374 break; | |
9375 | |
9376 case '(': | |
9377 cur += 1; | |
9378 _cur_lexeme = lex_open_brace; | |
9379 | |
9380 break; | |
9381 | |
9382 case ')': | |
9383 cur += 1; | |
9384 _cur_lexeme = lex_close_brace; | |
9385 | |
9386 break; | |
9387 | |
9388 case '[': | |
9389 cur += 1; | |
9390 _cur_lexeme = lex_open_square_brace; | |
9391 | |
9392 break; | |
9393 | |
9394 case ']': | |
9395 cur += 1; | |
9396 _cur_lexeme = lex_close_square_brace; | |
9397 | |
9398 break; | |
9399 | |
9400 case ',': | |
9401 cur += 1; | |
9402 _cur_lexeme = lex_comma; | |
9403 | |
9404 break; | |
9405 | |
9406 case '/': | |
9407 if (*(cur+1) == '/') | |
9408 { | |
9409 cur += 2; | |
9410 _cur_lexeme = lex_double_slash; | |
9411 } | |
9412 else | |
9413 { | |
9414 cur += 1; | |
9415 _cur_lexeme = lex_slash; | |
9416 } | |
9417 break; | |
9418 | |
9419 case '.': | |
9420 if (*(cur+1) == '.') | |
9421 { | |
9422 cur += 2; | |
9423 _cur_lexeme = lex_double_dot; | |
9424 } | |
9425 else if (PUGI__IS_CHARTYPEX(*(cur+1), ctx_digit)) | |
9426 { | |
9427 _cur_lexeme_contents.begin = cur; // . | |
9428 | |
9429 ++cur; | |
9430 | |
9431 while (PUGI__IS_CHARTYPEX(*cur, ctx_digit)) cur++; | |
9432 | |
9433 _cur_lexeme_contents.end = cur; | |
9434 | |
9435 _cur_lexeme = lex_number; | |
9436 } | |
9437 else | |
9438 { | |
9439 cur += 1; | |
9440 _cur_lexeme = lex_dot; | |
9441 } | |
9442 break; | |
9443 | |
9444 case '@': | |
9445 cur += 1; | |
9446 _cur_lexeme = lex_axis_attribute; | |
9447 | |
9448 break; | |
9449 | |
9450 case '"': | |
9451 case '\'': | |
9452 { | |
9453 char_t terminator = *cur; | |
9454 | |
9455 ++cur; | |
9456 | |
9457 _cur_lexeme_contents.begin = cur; | |
9458 while (*cur && *cur != terminator) cur++; | |
9459 _cur_lexeme_contents.end = cur; | |
9460 | |
9461 if (!*cur) | |
9462 _cur_lexeme = lex_none; | |
9463 else | |
9464 { | |
9465 cur += 1; | |
9466 _cur_lexeme = lex_quoted_string; | |
9467 } | |
9468 | |
9469 break; | |
9470 } | |
9471 | |
9472 case ':': | |
9473 if (*(cur+1) == ':') | |
9474 { | |
9475 cur += 2; | |
9476 _cur_lexeme = lex_double_colon; | |
9477 } | |
9478 else | |
9479 { | |
9480 _cur_lexeme = lex_none; | |
9481 } | |
9482 break; | |
9483 | |
9484 default: | |
9485 if (PUGI__IS_CHARTYPEX(*cur, ctx_digit)) | |
9486 { | |
9487 _cur_lexeme_contents.begin = cur; | |
9488 | |
9489 while (PUGI__IS_CHARTYPEX(*cur, ctx_digit)) cur++; | |
9490 | |
9491 if (*cur == '.') | |
9492 { | |
9493 cur++; | |
9494 | |
9495 while (PUGI__IS_CHARTYPEX(*cur, ctx_digit)) cur++; | |
9496 } | |
9497 | |
9498 _cur_lexeme_contents.end = cur; | |
9499 | |
9500 _cur_lexeme = lex_number; | |
9501 } | |
9502 else if (PUGI__IS_CHARTYPEX(*cur, ctx_start_symbol)) | |
9503 { | |
9504 _cur_lexeme_contents.begin = cur; | |
9505 | |
9506 while (PUGI__IS_CHARTYPEX(*cur, ctx_symbol)) cur++; | |
9507 | |
9508 if (cur[0] == ':') | |
9509 { | |
9510 if (cur[1] == '*') // namespace test ncname:* | |
9511 { | |
9512 cur += 2; // :* | |
9513 } | |
9514 else if (PUGI__IS_CHARTYPEX(cur[1], ctx_symbol)) // namespace test qname | |
9515 { | |
9516 cur++; // : | |
9517 | |
9518 while (PUGI__IS_CHARTYPEX(*cur, ctx_symbol)) cur++; | |
9519 } | |
9520 } | |
9521 | |
9522 _cur_lexeme_contents.end = cur; | |
9523 | |
9524 _cur_lexeme = lex_string; | |
9525 } | |
9526 else | |
9527 { | |
9528 _cur_lexeme = lex_none; | |
9529 } | |
9530 } | |
9531 | |
9532 _cur = cur; | |
9533 } | |
9534 | |
9535 lexeme_t current() const | |
9536 { | |
9537 return _cur_lexeme; | |
9538 } | |
9539 | |
9540 const char_t* current_pos() const | |
9541 { | |
9542 return _cur_lexeme_pos; | |
9543 } | |
9544 | |
9545 const xpath_lexer_string& contents() const | |
9546 { | |
9547 assert(_cur_lexeme == lex_var_ref || _cur_lexeme == lex_number || _cur_lexeme == lex_string || _cur_lexeme == lex_quoted_string); | |
9548 | |
9549 return _cur_lexeme_contents; | |
9550 } | |
9551 }; | |
9552 | |
9553 enum ast_type_t | |
9554 { | |
9555 ast_unknown, | |
9556 ast_op_or, // left or right | |
9557 ast_op_and, // left and right | |
9558 ast_op_equal, // left = right | |
9559 ast_op_not_equal, // left != right | |
9560 ast_op_less, // left < right | |
9561 ast_op_greater, // left > right | |
9562 ast_op_less_or_equal, // left <= right | |
9563 ast_op_greater_or_equal, // left >= right | |
9564 ast_op_add, // left + right | |
9565 ast_op_subtract, // left - right | |
9566 ast_op_multiply, // left * right | |
9567 ast_op_divide, // left / right | |
9568 ast_op_mod, // left % right | |
9569 ast_op_negate, // left - right | |
9570 ast_op_union, // left | right | |
9571 ast_predicate, // apply predicate to set; next points to next predicate | |
9572 ast_filter, // select * from left where right | |
9573 ast_string_constant, // string constant | |
9574 ast_number_constant, // number constant | |
9575 ast_variable, // variable | |
9576 ast_func_last, // last() | |
9577 ast_func_position, // position() | |
9578 ast_func_count, // count(left) | |
9579 ast_func_id, // id(left) | |
9580 ast_func_local_name_0, // local-name() | |
9581 ast_func_local_name_1, // local-name(left) | |
9582 ast_func_namespace_uri_0, // namespace-uri() | |
9583 ast_func_namespace_uri_1, // namespace-uri(left) | |
9584 ast_func_name_0, // name() | |
9585 ast_func_name_1, // name(left) | |
9586 ast_func_string_0, // string() | |
9587 ast_func_string_1, // string(left) | |
9588 ast_func_concat, // concat(left, right, siblings) | |
9589 ast_func_starts_with, // starts_with(left, right) | |
9590 ast_func_contains, // contains(left, right) | |
9591 ast_func_substring_before, // substring-before(left, right) | |
9592 ast_func_substring_after, // substring-after(left, right) | |
9593 ast_func_substring_2, // substring(left, right) | |
9594 ast_func_substring_3, // substring(left, right, third) | |
9595 ast_func_string_length_0, // string-length() | |
9596 ast_func_string_length_1, // string-length(left) | |
9597 ast_func_normalize_space_0, // normalize-space() | |
9598 ast_func_normalize_space_1, // normalize-space(left) | |
9599 ast_func_translate, // translate(left, right, third) | |
9600 ast_func_boolean, // boolean(left) | |
9601 ast_func_not, // not(left) | |
9602 ast_func_true, // true() | |
9603 ast_func_false, // false() | |
9604 ast_func_lang, // lang(left) | |
9605 ast_func_number_0, // number() | |
9606 ast_func_number_1, // number(left) | |
9607 ast_func_sum, // sum(left) | |
9608 ast_func_floor, // floor(left) | |
9609 ast_func_ceiling, // ceiling(left) | |
9610 ast_func_round, // round(left) | |
9611 ast_step, // process set left with step | |
9612 ast_step_root, // select root node | |
9613 | |
9614 ast_opt_translate_table, // translate(left, right, third) where right/third are constants | |
9615 ast_opt_compare_attribute // @name = 'string' | |
9616 }; | |
9617 | |
9618 enum axis_t | |
9619 { | |
9620 axis_ancestor, | |
9621 axis_ancestor_or_self, | |
9622 axis_attribute, | |
9623 axis_child, | |
9624 axis_descendant, | |
9625 axis_descendant_or_self, | |
9626 axis_following, | |
9627 axis_following_sibling, | |
9628 axis_namespace, | |
9629 axis_parent, | |
9630 axis_preceding, | |
9631 axis_preceding_sibling, | |
9632 axis_self | |
9633 }; | |
9634 | |
9635 enum nodetest_t | |
9636 { | |
9637 nodetest_none, | |
9638 nodetest_name, | |
9639 nodetest_type_node, | |
9640 nodetest_type_comment, | |
9641 nodetest_type_pi, | |
9642 nodetest_type_text, | |
9643 nodetest_pi, | |
9644 nodetest_all, | |
9645 nodetest_all_in_namespace | |
9646 }; | |
9647 | |
9648 enum predicate_t | |
9649 { | |
9650 predicate_default, | |
9651 predicate_posinv, | |
9652 predicate_constant, | |
9653 predicate_constant_one | |
9654 }; | |
9655 | |
9656 enum nodeset_eval_t | |
9657 { | |
9658 nodeset_eval_all, | |
9659 nodeset_eval_any, | |
9660 nodeset_eval_first | |
9661 }; | |
9662 | |
9663 template <axis_t N> struct axis_to_type | |
9664 { | |
9665 static const axis_t axis; | |
9666 }; | |
9667 | |
9668 template <axis_t N> const axis_t axis_to_type<N>::axis = N; | |
9669 | |
9670 class xpath_ast_node | |
9671 { | |
9672 private: | |
9673 // node type | |
9674 char _type; | |
9675 char _rettype; | |
9676 | |
9677 // for ast_step | |
9678 char _axis; | |
9679 | |
9680 // for ast_step/ast_predicate/ast_filter | |
9681 char _test; | |
9682 | |
9683 // tree node structure | |
9684 xpath_ast_node* _left; | |
9685 xpath_ast_node* _right; | |
9686 xpath_ast_node* _next; | |
9687 | |
9688 union | |
9689 { | |
9690 // value for ast_string_constant | |
9691 const char_t* string; | |
9692 // value for ast_number_constant | |
9693 double number; | |
9694 // variable for ast_variable | |
9695 xpath_variable* variable; | |
9696 // node test for ast_step (node name/namespace/node type/pi target) | |
9697 const char_t* nodetest; | |
9698 // table for ast_opt_translate_table | |
9699 const unsigned char* table; | |
9700 } _data; | |
9701 | |
9702 xpath_ast_node(const xpath_ast_node&); | |
9703 xpath_ast_node& operator=(const xpath_ast_node&); | |
9704 | |
9705 template <class Comp> static bool compare_eq(xpath_ast_node* lhs, xpath_ast_node* rhs, const xpath_context& c, const xpath_stack& stack, const Comp& comp) | |
9706 { | |
9707 xpath_value_type lt = lhs->rettype(), rt = rhs->rettype(); | |
9708 | |
9709 if (lt != xpath_type_node_set && rt != xpath_type_node_set) | |
9710 { | |
9711 if (lt == xpath_type_boolean || rt == xpath_type_boolean) | |
9712 return comp(lhs->eval_boolean(c, stack), rhs->eval_boolean(c, stack)); | |
9713 else if (lt == xpath_type_number || rt == xpath_type_number) | |
9714 return comp(lhs->eval_number(c, stack), rhs->eval_number(c, stack)); | |
9715 else if (lt == xpath_type_string || rt == xpath_type_string) | |
9716 { | |
9717 xpath_allocator_capture cr(stack.result); | |
9718 | |
9719 xpath_string ls = lhs->eval_string(c, stack); | |
9720 xpath_string rs = rhs->eval_string(c, stack); | |
9721 | |
9722 return comp(ls, rs); | |
9723 } | |
9724 } | |
9725 else if (lt == xpath_type_node_set && rt == xpath_type_node_set) | |
9726 { | |
9727 xpath_allocator_capture cr(stack.result); | |
9728 | |
9729 xpath_node_set_raw ls = lhs->eval_node_set(c, stack, nodeset_eval_all); | |
9730 xpath_node_set_raw rs = rhs->eval_node_set(c, stack, nodeset_eval_all); | |
9731 | |
9732 for (const xpath_node* li = ls.begin(); li != ls.end(); ++li) | |
9733 for (const xpath_node* ri = rs.begin(); ri != rs.end(); ++ri) | |
9734 { | |
9735 xpath_allocator_capture cri(stack.result); | |
9736 | |
9737 if (comp(string_value(*li, stack.result), string_value(*ri, stack.result))) | |
9738 return true; | |
9739 } | |
9740 | |
9741 return false; | |
9742 } | |
9743 else | |
9744 { | |
9745 if (lt == xpath_type_node_set) | |
9746 { | |
9747 swap(lhs, rhs); | |
9748 swap(lt, rt); | |
9749 } | |
9750 | |
9751 if (lt == xpath_type_boolean) | |
9752 return comp(lhs->eval_boolean(c, stack), rhs->eval_boolean(c, stack)); | |
9753 else if (lt == xpath_type_number) | |
9754 { | |
9755 xpath_allocator_capture cr(stack.result); | |
9756 | |
9757 double l = lhs->eval_number(c, stack); | |
9758 xpath_node_set_raw rs = rhs->eval_node_set(c, stack, nodeset_eval_all); | |
9759 | |
9760 for (const xpath_node* ri = rs.begin(); ri != rs.end(); ++ri) | |
9761 { | |
9762 xpath_allocator_capture cri(stack.result); | |
9763 | |
9764 if (comp(l, convert_string_to_number(string_value(*ri, stack.result).c_str()))) | |
9765 return true; | |
9766 } | |
9767 | |
9768 return false; | |
9769 } | |
9770 else if (lt == xpath_type_string) | |
9771 { | |
9772 xpath_allocator_capture cr(stack.result); | |
9773 | |
9774 xpath_string l = lhs->eval_string(c, stack); | |
9775 xpath_node_set_raw rs = rhs->eval_node_set(c, stack, nodeset_eval_all); | |
9776 | |
9777 for (const xpath_node* ri = rs.begin(); ri != rs.end(); ++ri) | |
9778 { | |
9779 xpath_allocator_capture cri(stack.result); | |
9780 | |
9781 if (comp(l, string_value(*ri, stack.result))) | |
9782 return true; | |
9783 } | |
9784 | |
9785 return false; | |
9786 } | |
9787 } | |
9788 | |
9789 assert(false && "Wrong types"); // unreachable | |
9790 return false; | |
9791 } | |
9792 | |
9793 static bool eval_once(xpath_node_set::type_t type, nodeset_eval_t eval) | |
9794 { | |
9795 return type == xpath_node_set::type_sorted ? eval != nodeset_eval_all : eval == nodeset_eval_any; | |
9796 } | |
9797 | |
9798 template <class Comp> static bool compare_rel(xpath_ast_node* lhs, xpath_ast_node* rhs, const xpath_context& c, const xpath_stack& stack, const Comp& comp) | |
9799 { | |
9800 xpath_value_type lt = lhs->rettype(), rt = rhs->rettype(); | |
9801 | |
9802 if (lt != xpath_type_node_set && rt != xpath_type_node_set) | |
9803 return comp(lhs->eval_number(c, stack), rhs->eval_number(c, stack)); | |
9804 else if (lt == xpath_type_node_set && rt == xpath_type_node_set) | |
9805 { | |
9806 xpath_allocator_capture cr(stack.result); | |
9807 | |
9808 xpath_node_set_raw ls = lhs->eval_node_set(c, stack, nodeset_eval_all); | |
9809 xpath_node_set_raw rs = rhs->eval_node_set(c, stack, nodeset_eval_all); | |
9810 | |
9811 for (const xpath_node* li = ls.begin(); li != ls.end(); ++li) | |
9812 { | |
9813 xpath_allocator_capture cri(stack.result); | |
9814 | |
9815 double l = convert_string_to_number(string_value(*li, stack.result).c_str()); | |
9816 | |
9817 for (const xpath_node* ri = rs.begin(); ri != rs.end(); ++ri) | |
9818 { | |
9819 xpath_allocator_capture crii(stack.result); | |
9820 | |
9821 if (comp(l, convert_string_to_number(string_value(*ri, stack.result).c_str()))) | |
9822 return true; | |
9823 } | |
9824 } | |
9825 | |
9826 return false; | |
9827 } | |
9828 else if (lt != xpath_type_node_set && rt == xpath_type_node_set) | |
9829 { | |
9830 xpath_allocator_capture cr(stack.result); | |
9831 | |
9832 double l = lhs->eval_number(c, stack); | |
9833 xpath_node_set_raw rs = rhs->eval_node_set(c, stack, nodeset_eval_all); | |
9834 | |
9835 for (const xpath_node* ri = rs.begin(); ri != rs.end(); ++ri) | |
9836 { | |
9837 xpath_allocator_capture cri(stack.result); | |
9838 | |
9839 if (comp(l, convert_string_to_number(string_value(*ri, stack.result).c_str()))) | |
9840 return true; | |
9841 } | |
9842 | |
9843 return false; | |
9844 } | |
9845 else if (lt == xpath_type_node_set && rt != xpath_type_node_set) | |
9846 { | |
9847 xpath_allocator_capture cr(stack.result); | |
9848 | |
9849 xpath_node_set_raw ls = lhs->eval_node_set(c, stack, nodeset_eval_all); | |
9850 double r = rhs->eval_number(c, stack); | |
9851 | |
9852 for (const xpath_node* li = ls.begin(); li != ls.end(); ++li) | |
9853 { | |
9854 xpath_allocator_capture cri(stack.result); | |
9855 | |
9856 if (comp(convert_string_to_number(string_value(*li, stack.result).c_str()), r)) | |
9857 return true; | |
9858 } | |
9859 | |
9860 return false; | |
9861 } | |
9862 else | |
9863 { | |
9864 assert(false && "Wrong types"); // unreachable | |
9865 return false; | |
9866 } | |
9867 } | |
9868 | |
9869 static void apply_predicate_boolean(xpath_node_set_raw& ns, size_t first, xpath_ast_node* expr, const xpath_stack& stack, bool once) | |
9870 { | |
9871 assert(ns.size() >= first); | |
9872 assert(expr->rettype() != xpath_type_number); | |
9873 | |
9874 size_t i = 1; | |
9875 size_t size = ns.size() - first; | |
9876 | |
9877 xpath_node* last = ns.begin() + first; | |
9878 | |
9879 // remove_if... or well, sort of | |
9880 for (xpath_node* it = last; it != ns.end(); ++it, ++i) | |
9881 { | |
9882 xpath_context c(*it, i, size); | |
9883 | |
9884 if (expr->eval_boolean(c, stack)) | |
9885 { | |
9886 *last++ = *it; | |
9887 | |
9888 if (once) break; | |
9889 } | |
9890 } | |
9891 | |
9892 ns.truncate(last); | |
9893 } | |
9894 | |
9895 static void apply_predicate_number(xpath_node_set_raw& ns, size_t first, xpath_ast_node* expr, const xpath_stack& stack, bool once) | |
9896 { | |
9897 assert(ns.size() >= first); | |
9898 assert(expr->rettype() == xpath_type_number); | |
9899 | |
9900 size_t i = 1; | |
9901 size_t size = ns.size() - first; | |
9902 | |
9903 xpath_node* last = ns.begin() + first; | |
9904 | |
9905 // remove_if... or well, sort of | |
9906 for (xpath_node* it = last; it != ns.end(); ++it, ++i) | |
9907 { | |
9908 xpath_context c(*it, i, size); | |
9909 | |
9910 if (expr->eval_number(c, stack) == static_cast<double>(i)) | |
9911 { | |
9912 *last++ = *it; | |
9913 | |
9914 if (once) break; | |
9915 } | |
9916 } | |
9917 | |
9918 ns.truncate(last); | |
9919 } | |
9920 | |
9921 static void apply_predicate_number_const(xpath_node_set_raw& ns, size_t first, xpath_ast_node* expr, const xpath_stack& stack) | |
9922 { | |
9923 assert(ns.size() >= first); | |
9924 assert(expr->rettype() == xpath_type_number); | |
9925 | |
9926 size_t size = ns.size() - first; | |
9927 | |
9928 xpath_node* last = ns.begin() + first; | |
9929 | |
9930 xpath_context c(xpath_node(), 1, size); | |
9931 | |
9932 double er = expr->eval_number(c, stack); | |
9933 | |
9934 if (er >= 1.0 && er <= static_cast<double>(size)) | |
9935 { | |
9936 size_t eri = static_cast<size_t>(er); | |
9937 | |
9938 if (er == static_cast<double>(eri)) | |
9939 { | |
9940 xpath_node r = last[eri - 1]; | |
9941 | |
9942 *last++ = r; | |
9943 } | |
9944 } | |
9945 | |
9946 ns.truncate(last); | |
9947 } | |
9948 | |
9949 void apply_predicate(xpath_node_set_raw& ns, size_t first, const xpath_stack& stack, bool once) | |
9950 { | |
9951 if (ns.size() == first) return; | |
9952 | |
9953 assert(_type == ast_filter || _type == ast_predicate); | |
9954 | |
9955 if (_test == predicate_constant || _test == predicate_constant_one) | |
9956 apply_predicate_number_const(ns, first, _right, stack); | |
9957 else if (_right->rettype() == xpath_type_number) | |
9958 apply_predicate_number(ns, first, _right, stack, once); | |
9959 else | |
9960 apply_predicate_boolean(ns, first, _right, stack, once); | |
9961 } | |
9962 | |
9963 void apply_predicates(xpath_node_set_raw& ns, size_t first, const xpath_stack& stack, nodeset_eval_t eval) | |
9964 { | |
9965 if (ns.size() == first) return; | |
9966 | |
9967 bool last_once = eval_once(ns.type(), eval); | |
9968 | |
9969 for (xpath_ast_node* pred = _right; pred; pred = pred->_next) | |
9970 pred->apply_predicate(ns, first, stack, !pred->_next && last_once); | |
9971 } | |
9972 | |
9973 bool step_push(xpath_node_set_raw& ns, xml_attribute_struct* a, xml_node_struct* parent, xpath_allocator* alloc) | |
9974 { | |
9975 assert(a); | |
9976 | |
9977 const char_t* name = a->name ? a->name + 0 : PUGIXML_TEXT(""); | |
9978 | |
9979 switch (_test) | |
9980 { | |
9981 case nodetest_name: | |
9982 if (strequal(name, _data.nodetest) && is_xpath_attribute(name)) | |
9983 { | |
9984 ns.push_back(xpath_node(xml_attribute(a), xml_node(parent)), alloc); | |
9985 return true; | |
9986 } | |
9987 break; | |
9988 | |
9989 case nodetest_type_node: | |
9990 case nodetest_all: | |
9991 if (is_xpath_attribute(name)) | |
9992 { | |
9993 ns.push_back(xpath_node(xml_attribute(a), xml_node(parent)), alloc); | |
9994 return true; | |
9995 } | |
9996 break; | |
9997 | |
9998 case nodetest_all_in_namespace: | |
9999 if (starts_with(name, _data.nodetest) && is_xpath_attribute(name)) | |
10000 { | |
10001 ns.push_back(xpath_node(xml_attribute(a), xml_node(parent)), alloc); | |
10002 return true; | |
10003 } | |
10004 break; | |
10005 | |
10006 default: | |
10007 ; | |
10008 } | |
10009 | |
10010 return false; | |
10011 } | |
10012 | |
10013 bool step_push(xpath_node_set_raw& ns, xml_node_struct* n, xpath_allocator* alloc) | |
10014 { | |
10015 assert(n); | |
10016 | |
10017 xml_node_type type = PUGI__NODETYPE(n); | |
10018 | |
10019 switch (_test) | |
10020 { | |
10021 case nodetest_name: | |
10022 if (type == node_element && n->name && strequal(n->name, _data.nodetest)) | |
10023 { | |
10024 ns.push_back(xml_node(n), alloc); | |
10025 return true; | |
10026 } | |
10027 break; | |
10028 | |
10029 case nodetest_type_node: | |
10030 ns.push_back(xml_node(n), alloc); | |
10031 return true; | |
10032 | |
10033 case nodetest_type_comment: | |
10034 if (type == node_comment) | |
10035 { | |
10036 ns.push_back(xml_node(n), alloc); | |
10037 return true; | |
10038 } | |
10039 break; | |
10040 | |
10041 case nodetest_type_text: | |
10042 if (type == node_pcdata || type == node_cdata) | |
10043 { | |
10044 ns.push_back(xml_node(n), alloc); | |
10045 return true; | |
10046 } | |
10047 break; | |
10048 | |
10049 case nodetest_type_pi: | |
10050 if (type == node_pi) | |
10051 { | |
10052 ns.push_back(xml_node(n), alloc); | |
10053 return true; | |
10054 } | |
10055 break; | |
10056 | |
10057 case nodetest_pi: | |
10058 if (type == node_pi && n->name && strequal(n->name, _data.nodetest)) | |
10059 { | |
10060 ns.push_back(xml_node(n), alloc); | |
10061 return true; | |
10062 } | |
10063 break; | |
10064 | |
10065 case nodetest_all: | |
10066 if (type == node_element) | |
10067 { | |
10068 ns.push_back(xml_node(n), alloc); | |
10069 return true; | |
10070 } | |
10071 break; | |
10072 | |
10073 case nodetest_all_in_namespace: | |
10074 if (type == node_element && n->name && starts_with(n->name, _data.nodetest)) | |
10075 { | |
10076 ns.push_back(xml_node(n), alloc); | |
10077 return true; | |
10078 } | |
10079 break; | |
10080 | |
10081 default: | |
10082 assert(false && "Unknown axis"); // unreachable | |
10083 } | |
10084 | |
10085 return false; | |
10086 } | |
10087 | |
10088 template <class T> void step_fill(xpath_node_set_raw& ns, xml_node_struct* n, xpath_allocator* alloc, bool once, T) | |
10089 { | |
10090 const axis_t axis = T::axis; | |
10091 | |
10092 switch (axis) | |
10093 { | |
10094 case axis_attribute: | |
10095 { | |
10096 for (xml_attribute_struct* a = n->first_attribute; a; a = a->next_attribute) | |
10097 if (step_push(ns, a, n, alloc) & once) | |
10098 return; | |
10099 | |
10100 break; | |
10101 } | |
10102 | |
10103 case axis_child: | |
10104 { | |
10105 for (xml_node_struct* c = n->first_child; c; c = c->next_sibling) | |
10106 if (step_push(ns, c, alloc) & once) | |
10107 return; | |
10108 | |
10109 break; | |
10110 } | |
10111 | |
10112 case axis_descendant: | |
10113 case axis_descendant_or_self: | |
10114 { | |
10115 if (axis == axis_descendant_or_self) | |
10116 if (step_push(ns, n, alloc) & once) | |
10117 return; | |
10118 | |
10119 xml_node_struct* cur = n->first_child; | |
10120 | |
10121 while (cur) | |
10122 { | |
10123 if (step_push(ns, cur, alloc) & once) | |
10124 return; | |
10125 | |
10126 if (cur->first_child) | |
10127 cur = cur->first_child; | |
10128 else | |
10129 { | |
10130 while (!cur->next_sibling) | |
10131 { | |
10132 cur = cur->parent; | |
10133 | |
10134 if (cur == n) return; | |
10135 } | |
10136 | |
10137 cur = cur->next_sibling; | |
10138 } | |
10139 } | |
10140 | |
10141 break; | |
10142 } | |
10143 | |
10144 case axis_following_sibling: | |
10145 { | |
10146 for (xml_node_struct* c = n->next_sibling; c; c = c->next_sibling) | |
10147 if (step_push(ns, c, alloc) & once) | |
10148 return; | |
10149 | |
10150 break; | |
10151 } | |
10152 | |
10153 case axis_preceding_sibling: | |
10154 { | |
10155 for (xml_node_struct* c = n->prev_sibling_c; c->next_sibling; c = c->prev_sibling_c) | |
10156 if (step_push(ns, c, alloc) & once) | |
10157 return; | |
10158 | |
10159 break; | |
10160 } | |
10161 | |
10162 case axis_following: | |
10163 { | |
10164 xml_node_struct* cur = n; | |
10165 | |
10166 // exit from this node so that we don't include descendants | |
10167 while (!cur->next_sibling) | |
10168 { | |
10169 cur = cur->parent; | |
10170 | |
10171 if (!cur) return; | |
10172 } | |
10173 | |
10174 cur = cur->next_sibling; | |
10175 | |
10176 while (cur) | |
10177 { | |
10178 if (step_push(ns, cur, alloc) & once) | |
10179 return; | |
10180 | |
10181 if (cur->first_child) | |
10182 cur = cur->first_child; | |
10183 else | |
10184 { | |
10185 while (!cur->next_sibling) | |
10186 { | |
10187 cur = cur->parent; | |
10188 | |
10189 if (!cur) return; | |
10190 } | |
10191 | |
10192 cur = cur->next_sibling; | |
10193 } | |
10194 } | |
10195 | |
10196 break; | |
10197 } | |
10198 | |
10199 case axis_preceding: | |
10200 { | |
10201 xml_node_struct* cur = n; | |
10202 | |
10203 // exit from this node so that we don't include descendants | |
10204 while (!cur->prev_sibling_c->next_sibling) | |
10205 { | |
10206 cur = cur->parent; | |
10207 | |
10208 if (!cur) return; | |
10209 } | |
10210 | |
10211 cur = cur->prev_sibling_c; | |
10212 | |
10213 while (cur) | |
10214 { | |
10215 if (cur->first_child) | |
10216 cur = cur->first_child->prev_sibling_c; | |
10217 else | |
10218 { | |
10219 // leaf node, can't be ancestor | |
10220 if (step_push(ns, cur, alloc) & once) | |
10221 return; | |
10222 | |
10223 while (!cur->prev_sibling_c->next_sibling) | |
10224 { | |
10225 cur = cur->parent; | |
10226 | |
10227 if (!cur) return; | |
10228 | |
10229 if (!node_is_ancestor(cur, n)) | |
10230 if (step_push(ns, cur, alloc) & once) | |
10231 return; | |
10232 } | |
10233 | |
10234 cur = cur->prev_sibling_c; | |
10235 } | |
10236 } | |
10237 | |
10238 break; | |
10239 } | |
10240 | |
10241 case axis_ancestor: | |
10242 case axis_ancestor_or_self: | |
10243 { | |
10244 if (axis == axis_ancestor_or_self) | |
10245 if (step_push(ns, n, alloc) & once) | |
10246 return; | |
10247 | |
10248 xml_node_struct* cur = n->parent; | |
10249 | |
10250 while (cur) | |
10251 { | |
10252 if (step_push(ns, cur, alloc) & once) | |
10253 return; | |
10254 | |
10255 cur = cur->parent; | |
10256 } | |
10257 | |
10258 break; | |
10259 } | |
10260 | |
10261 case axis_self: | |
10262 { | |
10263 step_push(ns, n, alloc); | |
10264 | |
10265 break; | |
10266 } | |
10267 | |
10268 case axis_parent: | |
10269 { | |
10270 if (n->parent) | |
10271 step_push(ns, n->parent, alloc); | |
10272 | |
10273 break; | |
10274 } | |
10275 | |
10276 default: | |
10277 assert(false && "Unimplemented axis"); // unreachable | |
10278 } | |
10279 } | |
10280 | |
10281 template <class T> void step_fill(xpath_node_set_raw& ns, xml_attribute_struct* a, xml_node_struct* p, xpath_allocator* alloc, bool once, T v) | |
10282 { | |
10283 const axis_t axis = T::axis; | |
10284 | |
10285 switch (axis) | |
10286 { | |
10287 case axis_ancestor: | |
10288 case axis_ancestor_or_self: | |
10289 { | |
10290 if (axis == axis_ancestor_or_self && _test == nodetest_type_node) // reject attributes based on principal node type test | |
10291 if (step_push(ns, a, p, alloc) & once) | |
10292 return; | |
10293 | |
10294 xml_node_struct* cur = p; | |
10295 | |
10296 while (cur) | |
10297 { | |
10298 if (step_push(ns, cur, alloc) & once) | |
10299 return; | |
10300 | |
10301 cur = cur->parent; | |
10302 } | |
10303 | |
10304 break; | |
10305 } | |
10306 | |
10307 case axis_descendant_or_self: | |
10308 case axis_self: | |
10309 { | |
10310 if (_test == nodetest_type_node) // reject attributes based on principal node type test | |
10311 step_push(ns, a, p, alloc); | |
10312 | |
10313 break; | |
10314 } | |
10315 | |
10316 case axis_following: | |
10317 { | |
10318 xml_node_struct* cur = p; | |
10319 | |
10320 while (cur) | |
10321 { | |
10322 if (cur->first_child) | |
10323 cur = cur->first_child; | |
10324 else | |
10325 { | |
10326 while (!cur->next_sibling) | |
10327 { | |
10328 cur = cur->parent; | |
10329 | |
10330 if (!cur) return; | |
10331 } | |
10332 | |
10333 cur = cur->next_sibling; | |
10334 } | |
10335 | |
10336 if (step_push(ns, cur, alloc) & once) | |
10337 return; | |
10338 } | |
10339 | |
10340 break; | |
10341 } | |
10342 | |
10343 case axis_parent: | |
10344 { | |
10345 step_push(ns, p, alloc); | |
10346 | |
10347 break; | |
10348 } | |
10349 | |
10350 case axis_preceding: | |
10351 { | |
10352 // preceding:: axis does not include attribute nodes and attribute ancestors (they are the same as parent's ancestors), so we can reuse node preceding | |
10353 step_fill(ns, p, alloc, once, v); | |
10354 break; | |
10355 } | |
10356 | |
10357 default: | |
10358 assert(false && "Unimplemented axis"); // unreachable | |
10359 } | |
10360 } | |
10361 | |
10362 template <class T> void step_fill(xpath_node_set_raw& ns, const xpath_node& xn, xpath_allocator* alloc, bool once, T v) | |
10363 { | |
10364 const axis_t axis = T::axis; | |
10365 const bool axis_has_attributes = (axis == axis_ancestor || axis == axis_ancestor_or_self || axis == axis_descendant_or_self || axis == axis_following || axis == axis_parent || axis == axis_preceding || axis == axis_self); | |
10366 | |
10367 if (xn.node()) | |
10368 step_fill(ns, xn.node().internal_object(), alloc, once, v); | |
10369 else if (axis_has_attributes && xn.attribute() && xn.parent()) | |
10370 step_fill(ns, xn.attribute().internal_object(), xn.parent().internal_object(), alloc, once, v); | |
10371 } | |
10372 | |
10373 template <class T> xpath_node_set_raw step_do(const xpath_context& c, const xpath_stack& stack, nodeset_eval_t eval, T v) | |
10374 { | |
10375 const axis_t axis = T::axis; | |
10376 const bool axis_reverse = (axis == axis_ancestor || axis == axis_ancestor_or_self || axis == axis_preceding || axis == axis_preceding_sibling); | |
10377 const xpath_node_set::type_t axis_type = axis_reverse ? xpath_node_set::type_sorted_reverse : xpath_node_set::type_sorted; | |
10378 | |
10379 bool once = | |
10380 (axis == axis_attribute && _test == nodetest_name) || | |
10381 (!_right && eval_once(axis_type, eval)) || | |
10382 // coverity[mixed_enums] | |
10383 (_right && !_right->_next && _right->_test == predicate_constant_one); | |
10384 | |
10385 xpath_node_set_raw ns; | |
10386 ns.set_type(axis_type); | |
10387 | |
10388 if (_left) | |
10389 { | |
10390 xpath_node_set_raw s = _left->eval_node_set(c, stack, nodeset_eval_all); | |
10391 | |
10392 // self axis preserves the original order | |
10393 if (axis == axis_self) ns.set_type(s.type()); | |
10394 | |
10395 for (const xpath_node* it = s.begin(); it != s.end(); ++it) | |
10396 { | |
10397 size_t size = ns.size(); | |
10398 | |
10399 // in general, all axes generate elements in a particular order, but there is no order guarantee if axis is applied to two nodes | |
10400 if (axis != axis_self && size != 0) ns.set_type(xpath_node_set::type_unsorted); | |
10401 | |
10402 step_fill(ns, *it, stack.result, once, v); | |
10403 if (_right) apply_predicates(ns, size, stack, eval); | |
10404 } | |
10405 } | |
10406 else | |
10407 { | |
10408 step_fill(ns, c.n, stack.result, once, v); | |
10409 if (_right) apply_predicates(ns, 0, stack, eval); | |
10410 } | |
10411 | |
10412 // child, attribute and self axes always generate unique set of nodes | |
10413 // for other axis, if the set stayed sorted, it stayed unique because the traversal algorithms do not visit the same node twice | |
10414 if (axis != axis_child && axis != axis_attribute && axis != axis_self && ns.type() == xpath_node_set::type_unsorted) | |
10415 ns.remove_duplicates(stack.temp); | |
10416 | |
10417 return ns; | |
10418 } | |
10419 | |
10420 public: | |
10421 xpath_ast_node(ast_type_t type, xpath_value_type rettype_, const char_t* value): | |
10422 _type(static_cast<char>(type)), _rettype(static_cast<char>(rettype_)), _axis(0), _test(0), _left(0), _right(0), _next(0) | |
10423 { | |
10424 assert(type == ast_string_constant); | |
10425 _data.string = value; | |
10426 } | |
10427 | |
10428 xpath_ast_node(ast_type_t type, xpath_value_type rettype_, double value): | |
10429 _type(static_cast<char>(type)), _rettype(static_cast<char>(rettype_)), _axis(0), _test(0), _left(0), _right(0), _next(0) | |
10430 { | |
10431 assert(type == ast_number_constant); | |
10432 _data.number = value; | |
10433 } | |
10434 | |
10435 xpath_ast_node(ast_type_t type, xpath_value_type rettype_, xpath_variable* value): | |
10436 _type(static_cast<char>(type)), _rettype(static_cast<char>(rettype_)), _axis(0), _test(0), _left(0), _right(0), _next(0) | |
10437 { | |
10438 assert(type == ast_variable); | |
10439 _data.variable = value; | |
10440 } | |
10441 | |
10442 xpath_ast_node(ast_type_t type, xpath_value_type rettype_, xpath_ast_node* left = 0, xpath_ast_node* right = 0): | |
10443 _type(static_cast<char>(type)), _rettype(static_cast<char>(rettype_)), _axis(0), _test(0), _left(left), _right(right), _next(0) | |
10444 { | |
10445 } | |
10446 | |
10447 xpath_ast_node(ast_type_t type, xpath_ast_node* left, axis_t axis, nodetest_t test, const char_t* contents): | |
10448 _type(static_cast<char>(type)), _rettype(xpath_type_node_set), _axis(static_cast<char>(axis)), _test(static_cast<char>(test)), _left(left), _right(0), _next(0) | |
10449 { | |
10450 assert(type == ast_step); | |
10451 _data.nodetest = contents; | |
10452 } | |
10453 | |
10454 xpath_ast_node(ast_type_t type, xpath_ast_node* left, xpath_ast_node* right, predicate_t test): | |
10455 _type(static_cast<char>(type)), _rettype(xpath_type_node_set), _axis(0), _test(static_cast<char>(test)), _left(left), _right(right), _next(0) | |
10456 { | |
10457 assert(type == ast_filter || type == ast_predicate); | |
10458 } | |
10459 | |
10460 void set_next(xpath_ast_node* value) | |
10461 { | |
10462 _next = value; | |
10463 } | |
10464 | |
10465 void set_right(xpath_ast_node* value) | |
10466 { | |
10467 _right = value; | |
10468 } | |
10469 | |
10470 bool eval_boolean(const xpath_context& c, const xpath_stack& stack) | |
10471 { | |
10472 switch (_type) | |
10473 { | |
10474 case ast_op_or: | |
10475 return _left->eval_boolean(c, stack) || _right->eval_boolean(c, stack); | |
10476 | |
10477 case ast_op_and: | |
10478 return _left->eval_boolean(c, stack) && _right->eval_boolean(c, stack); | |
10479 | |
10480 case ast_op_equal: | |
10481 return compare_eq(_left, _right, c, stack, equal_to()); | |
10482 | |
10483 case ast_op_not_equal: | |
10484 return compare_eq(_left, _right, c, stack, not_equal_to()); | |
10485 | |
10486 case ast_op_less: | |
10487 return compare_rel(_left, _right, c, stack, less()); | |
10488 | |
10489 case ast_op_greater: | |
10490 return compare_rel(_right, _left, c, stack, less()); | |
10491 | |
10492 case ast_op_less_or_equal: | |
10493 return compare_rel(_left, _right, c, stack, less_equal()); | |
10494 | |
10495 case ast_op_greater_or_equal: | |
10496 return compare_rel(_right, _left, c, stack, less_equal()); | |
10497 | |
10498 case ast_func_starts_with: | |
10499 { | |
10500 xpath_allocator_capture cr(stack.result); | |
10501 | |
10502 xpath_string lr = _left->eval_string(c, stack); | |
10503 xpath_string rr = _right->eval_string(c, stack); | |
10504 | |
10505 return starts_with(lr.c_str(), rr.c_str()); | |
10506 } | |
10507 | |
10508 case ast_func_contains: | |
10509 { | |
10510 xpath_allocator_capture cr(stack.result); | |
10511 | |
10512 xpath_string lr = _left->eval_string(c, stack); | |
10513 xpath_string rr = _right->eval_string(c, stack); | |
10514 | |
10515 return find_substring(lr.c_str(), rr.c_str()) != 0; | |
10516 } | |
10517 | |
10518 case ast_func_boolean: | |
10519 return _left->eval_boolean(c, stack); | |
10520 | |
10521 case ast_func_not: | |
10522 return !_left->eval_boolean(c, stack); | |
10523 | |
10524 case ast_func_true: | |
10525 return true; | |
10526 | |
10527 case ast_func_false: | |
10528 return false; | |
10529 | |
10530 case ast_func_lang: | |
10531 { | |
10532 if (c.n.attribute()) return false; | |
10533 | |
10534 xpath_allocator_capture cr(stack.result); | |
10535 | |
10536 xpath_string lang = _left->eval_string(c, stack); | |
10537 | |
10538 for (xml_node n = c.n.node(); n; n = n.parent()) | |
10539 { | |
10540 xml_attribute a = n.attribute(PUGIXML_TEXT("xml:lang")); | |
10541 | |
10542 if (a) | |
10543 { | |
10544 const char_t* value = a.value(); | |
10545 | |
10546 // strnicmp / strncasecmp is not portable | |
10547 for (const char_t* lit = lang.c_str(); *lit; ++lit) | |
10548 { | |
10549 if (tolower_ascii(*lit) != tolower_ascii(*value)) return false; | |
10550 ++value; | |
10551 } | |
10552 | |
10553 return *value == 0 || *value == '-'; | |
10554 } | |
10555 } | |
10556 | |
10557 return false; | |
10558 } | |
10559 | |
10560 case ast_opt_compare_attribute: | |
10561 { | |
10562 const char_t* value = (_right->_type == ast_string_constant) ? _right->_data.string : _right->_data.variable->get_string(); | |
10563 | |
10564 xml_attribute attr = c.n.node().attribute(_left->_data.nodetest); | |
10565 | |
10566 return attr && strequal(attr.value(), value) && is_xpath_attribute(attr.name()); | |
10567 } | |
10568 | |
10569 case ast_variable: | |
10570 { | |
10571 assert(_rettype == _data.variable->type()); | |
10572 | |
10573 if (_rettype == xpath_type_boolean) | |
10574 return _data.variable->get_boolean(); | |
10575 | |
10576 // variable needs to be converted to the correct type, this is handled by the fallthrough block below | |
10577 break; | |
10578 } | |
10579 | |
10580 default: | |
10581 ; | |
10582 } | |
10583 | |
10584 // none of the ast types that return the value directly matched, we need to perform type conversion | |
10585 switch (_rettype) | |
10586 { | |
10587 case xpath_type_number: | |
10588 return convert_number_to_boolean(eval_number(c, stack)); | |
10589 | |
10590 case xpath_type_string: | |
10591 { | |
10592 xpath_allocator_capture cr(stack.result); | |
10593 | |
10594 return !eval_string(c, stack).empty(); | |
10595 } | |
10596 | |
10597 case xpath_type_node_set: | |
10598 { | |
10599 xpath_allocator_capture cr(stack.result); | |
10600 | |
10601 return !eval_node_set(c, stack, nodeset_eval_any).empty(); | |
10602 } | |
10603 | |
10604 default: | |
10605 assert(false && "Wrong expression for return type boolean"); // unreachable | |
10606 return false; | |
10607 } | |
10608 } | |
10609 | |
10610 double eval_number(const xpath_context& c, const xpath_stack& stack) | |
10611 { | |
10612 switch (_type) | |
10613 { | |
10614 case ast_op_add: | |
10615 return _left->eval_number(c, stack) + _right->eval_number(c, stack); | |
10616 | |
10617 case ast_op_subtract: | |
10618 return _left->eval_number(c, stack) - _right->eval_number(c, stack); | |
10619 | |
10620 case ast_op_multiply: | |
10621 return _left->eval_number(c, stack) * _right->eval_number(c, stack); | |
10622 | |
10623 case ast_op_divide: | |
10624 return _left->eval_number(c, stack) / _right->eval_number(c, stack); | |
10625 | |
10626 case ast_op_mod: | |
10627 return fmod(_left->eval_number(c, stack), _right->eval_number(c, stack)); | |
10628 | |
10629 case ast_op_negate: | |
10630 return -_left->eval_number(c, stack); | |
10631 | |
10632 case ast_number_constant: | |
10633 return _data.number; | |
10634 | |
10635 case ast_func_last: | |
10636 return static_cast<double>(c.size); | |
10637 | |
10638 case ast_func_position: | |
10639 return static_cast<double>(c.position); | |
10640 | |
10641 case ast_func_count: | |
10642 { | |
10643 xpath_allocator_capture cr(stack.result); | |
10644 | |
10645 return static_cast<double>(_left->eval_node_set(c, stack, nodeset_eval_all).size()); | |
10646 } | |
10647 | |
10648 case ast_func_string_length_0: | |
10649 { | |
10650 xpath_allocator_capture cr(stack.result); | |
10651 | |
10652 return static_cast<double>(string_value(c.n, stack.result).length()); | |
10653 } | |
10654 | |
10655 case ast_func_string_length_1: | |
10656 { | |
10657 xpath_allocator_capture cr(stack.result); | |
10658 | |
10659 return static_cast<double>(_left->eval_string(c, stack).length()); | |
10660 } | |
10661 | |
10662 case ast_func_number_0: | |
10663 { | |
10664 xpath_allocator_capture cr(stack.result); | |
10665 | |
10666 return convert_string_to_number(string_value(c.n, stack.result).c_str()); | |
10667 } | |
10668 | |
10669 case ast_func_number_1: | |
10670 return _left->eval_number(c, stack); | |
10671 | |
10672 case ast_func_sum: | |
10673 { | |
10674 xpath_allocator_capture cr(stack.result); | |
10675 | |
10676 double r = 0; | |
10677 | |
10678 xpath_node_set_raw ns = _left->eval_node_set(c, stack, nodeset_eval_all); | |
10679 | |
10680 for (const xpath_node* it = ns.begin(); it != ns.end(); ++it) | |
10681 { | |
10682 xpath_allocator_capture cri(stack.result); | |
10683 | |
10684 r += convert_string_to_number(string_value(*it, stack.result).c_str()); | |
10685 } | |
10686 | |
10687 return r; | |
10688 } | |
10689 | |
10690 case ast_func_floor: | |
10691 { | |
10692 double r = _left->eval_number(c, stack); | |
10693 | |
10694 return r == r ? floor(r) : r; | |
10695 } | |
10696 | |
10697 case ast_func_ceiling: | |
10698 { | |
10699 double r = _left->eval_number(c, stack); | |
10700 | |
10701 return r == r ? ceil(r) : r; | |
10702 } | |
10703 | |
10704 case ast_func_round: | |
10705 return round_nearest_nzero(_left->eval_number(c, stack)); | |
10706 | |
10707 case ast_variable: | |
10708 { | |
10709 assert(_rettype == _data.variable->type()); | |
10710 | |
10711 if (_rettype == xpath_type_number) | |
10712 return _data.variable->get_number(); | |
10713 | |
10714 // variable needs to be converted to the correct type, this is handled by the fallthrough block below | |
10715 break; | |
10716 } | |
10717 | |
10718 default: | |
10719 ; | |
10720 } | |
10721 | |
10722 // none of the ast types that return the value directly matched, we need to perform type conversion | |
10723 switch (_rettype) | |
10724 { | |
10725 case xpath_type_boolean: | |
10726 return eval_boolean(c, stack) ? 1 : 0; | |
10727 | |
10728 case xpath_type_string: | |
10729 { | |
10730 xpath_allocator_capture cr(stack.result); | |
10731 | |
10732 return convert_string_to_number(eval_string(c, stack).c_str()); | |
10733 } | |
10734 | |
10735 case xpath_type_node_set: | |
10736 { | |
10737 xpath_allocator_capture cr(stack.result); | |
10738 | |
10739 return convert_string_to_number(eval_string(c, stack).c_str()); | |
10740 } | |
10741 | |
10742 default: | |
10743 assert(false && "Wrong expression for return type number"); // unreachable | |
10744 return 0; | |
10745 } | |
10746 } | |
10747 | |
10748 xpath_string eval_string_concat(const xpath_context& c, const xpath_stack& stack) | |
10749 { | |
10750 assert(_type == ast_func_concat); | |
10751 | |
10752 xpath_allocator_capture ct(stack.temp); | |
10753 | |
10754 // count the string number | |
10755 size_t count = 1; | |
10756 for (xpath_ast_node* nc = _right; nc; nc = nc->_next) count++; | |
10757 | |
10758 // allocate a buffer for temporary string objects | |
10759 xpath_string* buffer = static_cast<xpath_string*>(stack.temp->allocate(count * sizeof(xpath_string))); | |
10760 if (!buffer) return xpath_string(); | |
10761 | |
10762 // evaluate all strings to temporary stack | |
10763 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
10764 | |
10765 buffer[0] = _left->eval_string(c, swapped_stack); | |
10766 | |
10767 size_t pos = 1; | |
10768 for (xpath_ast_node* n = _right; n; n = n->_next, ++pos) buffer[pos] = n->eval_string(c, swapped_stack); | |
10769 assert(pos == count); | |
10770 | |
10771 // get total length | |
10772 size_t length = 0; | |
10773 for (size_t i = 0; i < count; ++i) length += buffer[i].length(); | |
10774 | |
10775 // create final string | |
10776 char_t* result = static_cast<char_t*>(stack.result->allocate((length + 1) * sizeof(char_t))); | |
10777 if (!result) return xpath_string(); | |
10778 | |
10779 char_t* ri = result; | |
10780 | |
10781 for (size_t j = 0; j < count; ++j) | |
10782 for (const char_t* bi = buffer[j].c_str(); *bi; ++bi) | |
10783 *ri++ = *bi; | |
10784 | |
10785 *ri = 0; | |
10786 | |
10787 return xpath_string::from_heap_preallocated(result, ri); | |
10788 } | |
10789 | |
10790 xpath_string eval_string(const xpath_context& c, const xpath_stack& stack) | |
10791 { | |
10792 switch (_type) | |
10793 { | |
10794 case ast_string_constant: | |
10795 return xpath_string::from_const(_data.string); | |
10796 | |
10797 case ast_func_local_name_0: | |
10798 { | |
10799 xpath_node na = c.n; | |
10800 | |
10801 return xpath_string::from_const(local_name(na)); | |
10802 } | |
10803 | |
10804 case ast_func_local_name_1: | |
10805 { | |
10806 xpath_allocator_capture cr(stack.result); | |
10807 | |
10808 xpath_node_set_raw ns = _left->eval_node_set(c, stack, nodeset_eval_first); | |
10809 xpath_node na = ns.first(); | |
10810 | |
10811 return xpath_string::from_const(local_name(na)); | |
10812 } | |
10813 | |
10814 case ast_func_name_0: | |
10815 { | |
10816 xpath_node na = c.n; | |
10817 | |
10818 return xpath_string::from_const(qualified_name(na)); | |
10819 } | |
10820 | |
10821 case ast_func_name_1: | |
10822 { | |
10823 xpath_allocator_capture cr(stack.result); | |
10824 | |
10825 xpath_node_set_raw ns = _left->eval_node_set(c, stack, nodeset_eval_first); | |
10826 xpath_node na = ns.first(); | |
10827 | |
10828 return xpath_string::from_const(qualified_name(na)); | |
10829 } | |
10830 | |
10831 case ast_func_namespace_uri_0: | |
10832 { | |
10833 xpath_node na = c.n; | |
10834 | |
10835 return xpath_string::from_const(namespace_uri(na)); | |
10836 } | |
10837 | |
10838 case ast_func_namespace_uri_1: | |
10839 { | |
10840 xpath_allocator_capture cr(stack.result); | |
10841 | |
10842 xpath_node_set_raw ns = _left->eval_node_set(c, stack, nodeset_eval_first); | |
10843 xpath_node na = ns.first(); | |
10844 | |
10845 return xpath_string::from_const(namespace_uri(na)); | |
10846 } | |
10847 | |
10848 case ast_func_string_0: | |
10849 return string_value(c.n, stack.result); | |
10850 | |
10851 case ast_func_string_1: | |
10852 return _left->eval_string(c, stack); | |
10853 | |
10854 case ast_func_concat: | |
10855 return eval_string_concat(c, stack); | |
10856 | |
10857 case ast_func_substring_before: | |
10858 { | |
10859 xpath_allocator_capture cr(stack.temp); | |
10860 | |
10861 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
10862 | |
10863 xpath_string s = _left->eval_string(c, swapped_stack); | |
10864 xpath_string p = _right->eval_string(c, swapped_stack); | |
10865 | |
10866 const char_t* pos = find_substring(s.c_str(), p.c_str()); | |
10867 | |
10868 return pos ? xpath_string::from_heap(s.c_str(), pos, stack.result) : xpath_string(); | |
10869 } | |
10870 | |
10871 case ast_func_substring_after: | |
10872 { | |
10873 xpath_allocator_capture cr(stack.temp); | |
10874 | |
10875 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
10876 | |
10877 xpath_string s = _left->eval_string(c, swapped_stack); | |
10878 xpath_string p = _right->eval_string(c, swapped_stack); | |
10879 | |
10880 const char_t* pos = find_substring(s.c_str(), p.c_str()); | |
10881 if (!pos) return xpath_string(); | |
10882 | |
10883 const char_t* rbegin = pos + p.length(); | |
10884 const char_t* rend = s.c_str() + s.length(); | |
10885 | |
10886 return s.uses_heap() ? xpath_string::from_heap(rbegin, rend, stack.result) : xpath_string::from_const(rbegin); | |
10887 } | |
10888 | |
10889 case ast_func_substring_2: | |
10890 { | |
10891 xpath_allocator_capture cr(stack.temp); | |
10892 | |
10893 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
10894 | |
10895 xpath_string s = _left->eval_string(c, swapped_stack); | |
10896 size_t s_length = s.length(); | |
10897 | |
10898 double first = round_nearest(_right->eval_number(c, stack)); | |
10899 | |
10900 if (is_nan(first)) return xpath_string(); // NaN | |
10901 else if (first >= static_cast<double>(s_length + 1)) return xpath_string(); | |
10902 | |
10903 size_t pos = first < 1 ? 1 : static_cast<size_t>(first); | |
10904 assert(1 <= pos && pos <= s_length + 1); | |
10905 | |
10906 const char_t* rbegin = s.c_str() + (pos - 1); | |
10907 const char_t* rend = s.c_str() + s.length(); | |
10908 | |
10909 return s.uses_heap() ? xpath_string::from_heap(rbegin, rend, stack.result) : xpath_string::from_const(rbegin); | |
10910 } | |
10911 | |
10912 case ast_func_substring_3: | |
10913 { | |
10914 xpath_allocator_capture cr(stack.temp); | |
10915 | |
10916 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
10917 | |
10918 xpath_string s = _left->eval_string(c, swapped_stack); | |
10919 size_t s_length = s.length(); | |
10920 | |
10921 double first = round_nearest(_right->eval_number(c, stack)); | |
10922 double last = first + round_nearest(_right->_next->eval_number(c, stack)); | |
10923 | |
10924 if (is_nan(first) || is_nan(last)) return xpath_string(); | |
10925 else if (first >= static_cast<double>(s_length + 1)) return xpath_string(); | |
10926 else if (first >= last) return xpath_string(); | |
10927 else if (last < 1) return xpath_string(); | |
10928 | |
10929 size_t pos = first < 1 ? 1 : static_cast<size_t>(first); | |
10930 size_t end = last >= static_cast<double>(s_length + 1) ? s_length + 1 : static_cast<size_t>(last); | |
10931 | |
10932 assert(1 <= pos && pos <= end && end <= s_length + 1); | |
10933 const char_t* rbegin = s.c_str() + (pos - 1); | |
10934 const char_t* rend = s.c_str() + (end - 1); | |
10935 | |
10936 return (end == s_length + 1 && !s.uses_heap()) ? xpath_string::from_const(rbegin) : xpath_string::from_heap(rbegin, rend, stack.result); | |
10937 } | |
10938 | |
10939 case ast_func_normalize_space_0: | |
10940 { | |
10941 xpath_string s = string_value(c.n, stack.result); | |
10942 | |
10943 char_t* begin = s.data(stack.result); | |
10944 if (!begin) return xpath_string(); | |
10945 | |
10946 char_t* end = normalize_space(begin); | |
10947 | |
10948 return xpath_string::from_heap_preallocated(begin, end); | |
10949 } | |
10950 | |
10951 case ast_func_normalize_space_1: | |
10952 { | |
10953 xpath_string s = _left->eval_string(c, stack); | |
10954 | |
10955 char_t* begin = s.data(stack.result); | |
10956 if (!begin) return xpath_string(); | |
10957 | |
10958 char_t* end = normalize_space(begin); | |
10959 | |
10960 return xpath_string::from_heap_preallocated(begin, end); | |
10961 } | |
10962 | |
10963 case ast_func_translate: | |
10964 { | |
10965 xpath_allocator_capture cr(stack.temp); | |
10966 | |
10967 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
10968 | |
10969 xpath_string s = _left->eval_string(c, stack); | |
10970 xpath_string from = _right->eval_string(c, swapped_stack); | |
10971 xpath_string to = _right->_next->eval_string(c, swapped_stack); | |
10972 | |
10973 char_t* begin = s.data(stack.result); | |
10974 if (!begin) return xpath_string(); | |
10975 | |
10976 char_t* end = translate(begin, from.c_str(), to.c_str(), to.length()); | |
10977 | |
10978 return xpath_string::from_heap_preallocated(begin, end); | |
10979 } | |
10980 | |
10981 case ast_opt_translate_table: | |
10982 { | |
10983 xpath_string s = _left->eval_string(c, stack); | |
10984 | |
10985 char_t* begin = s.data(stack.result); | |
10986 if (!begin) return xpath_string(); | |
10987 | |
10988 char_t* end = translate_table(begin, _data.table); | |
10989 | |
10990 return xpath_string::from_heap_preallocated(begin, end); | |
10991 } | |
10992 | |
10993 case ast_variable: | |
10994 { | |
10995 assert(_rettype == _data.variable->type()); | |
10996 | |
10997 if (_rettype == xpath_type_string) | |
10998 return xpath_string::from_const(_data.variable->get_string()); | |
10999 | |
11000 // variable needs to be converted to the correct type, this is handled by the fallthrough block below | |
11001 break; | |
11002 } | |
11003 | |
11004 default: | |
11005 ; | |
11006 } | |
11007 | |
11008 // none of the ast types that return the value directly matched, we need to perform type conversion | |
11009 switch (_rettype) | |
11010 { | |
11011 case xpath_type_boolean: | |
11012 return xpath_string::from_const(eval_boolean(c, stack) ? PUGIXML_TEXT("true") : PUGIXML_TEXT("false")); | |
11013 | |
11014 case xpath_type_number: | |
11015 return convert_number_to_string(eval_number(c, stack), stack.result); | |
11016 | |
11017 case xpath_type_node_set: | |
11018 { | |
11019 xpath_allocator_capture cr(stack.temp); | |
11020 | |
11021 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
11022 | |
11023 xpath_node_set_raw ns = eval_node_set(c, swapped_stack, nodeset_eval_first); | |
11024 return ns.empty() ? xpath_string() : string_value(ns.first(), stack.result); | |
11025 } | |
11026 | |
11027 default: | |
11028 assert(false && "Wrong expression for return type string"); // unreachable | |
11029 return xpath_string(); | |
11030 } | |
11031 } | |
11032 | |
11033 xpath_node_set_raw eval_node_set(const xpath_context& c, const xpath_stack& stack, nodeset_eval_t eval) | |
11034 { | |
11035 switch (_type) | |
11036 { | |
11037 case ast_op_union: | |
11038 { | |
11039 xpath_allocator_capture cr(stack.temp); | |
11040 | |
11041 xpath_stack swapped_stack = {stack.temp, stack.result}; | |
11042 | |
11043 xpath_node_set_raw ls = _left->eval_node_set(c, stack, eval); | |
11044 xpath_node_set_raw rs = _right->eval_node_set(c, swapped_stack, eval); | |
11045 | |
11046 // we can optimize merging two sorted sets, but this is a very rare operation, so don't bother | |
11047 ls.set_type(xpath_node_set::type_unsorted); | |
11048 | |
11049 ls.append(rs.begin(), rs.end(), stack.result); | |
11050 ls.remove_duplicates(stack.temp); | |
11051 | |
11052 return ls; | |
11053 } | |
11054 | |
11055 case ast_filter: | |
11056 { | |
11057 xpath_node_set_raw set = _left->eval_node_set(c, stack, _test == predicate_constant_one ? nodeset_eval_first : nodeset_eval_all); | |
11058 | |
11059 // either expression is a number or it contains position() call; sort by document order | |
11060 if (_test != predicate_posinv) set.sort_do(); | |
11061 | |
11062 bool once = eval_once(set.type(), eval); | |
11063 | |
11064 apply_predicate(set, 0, stack, once); | |
11065 | |
11066 return set; | |
11067 } | |
11068 | |
11069 case ast_func_id: | |
11070 return xpath_node_set_raw(); | |
11071 | |
11072 case ast_step: | |
11073 { | |
11074 switch (_axis) | |
11075 { | |
11076 case axis_ancestor: | |
11077 return step_do(c, stack, eval, axis_to_type<axis_ancestor>()); | |
11078 | |
11079 case axis_ancestor_or_self: | |
11080 return step_do(c, stack, eval, axis_to_type<axis_ancestor_or_self>()); | |
11081 | |
11082 case axis_attribute: | |
11083 return step_do(c, stack, eval, axis_to_type<axis_attribute>()); | |
11084 | |
11085 case axis_child: | |
11086 return step_do(c, stack, eval, axis_to_type<axis_child>()); | |
11087 | |
11088 case axis_descendant: | |
11089 return step_do(c, stack, eval, axis_to_type<axis_descendant>()); | |
11090 | |
11091 case axis_descendant_or_self: | |
11092 return step_do(c, stack, eval, axis_to_type<axis_descendant_or_self>()); | |
11093 | |
11094 case axis_following: | |
11095 return step_do(c, stack, eval, axis_to_type<axis_following>()); | |
11096 | |
11097 case axis_following_sibling: | |
11098 return step_do(c, stack, eval, axis_to_type<axis_following_sibling>()); | |
11099 | |
11100 case axis_namespace: | |
11101 // namespaced axis is not supported | |
11102 return xpath_node_set_raw(); | |
11103 | |
11104 case axis_parent: | |
11105 return step_do(c, stack, eval, axis_to_type<axis_parent>()); | |
11106 | |
11107 case axis_preceding: | |
11108 return step_do(c, stack, eval, axis_to_type<axis_preceding>()); | |
11109 | |
11110 case axis_preceding_sibling: | |
11111 return step_do(c, stack, eval, axis_to_type<axis_preceding_sibling>()); | |
11112 | |
11113 case axis_self: | |
11114 return step_do(c, stack, eval, axis_to_type<axis_self>()); | |
11115 | |
11116 default: | |
11117 assert(false && "Unknown axis"); // unreachable | |
11118 return xpath_node_set_raw(); | |
11119 } | |
11120 } | |
11121 | |
11122 case ast_step_root: | |
11123 { | |
11124 assert(!_right); // root step can't have any predicates | |
11125 | |
11126 xpath_node_set_raw ns; | |
11127 | |
11128 ns.set_type(xpath_node_set::type_sorted); | |
11129 | |
11130 if (c.n.node()) ns.push_back(c.n.node().root(), stack.result); | |
11131 else if (c.n.attribute()) ns.push_back(c.n.parent().root(), stack.result); | |
11132 | |
11133 return ns; | |
11134 } | |
11135 | |
11136 case ast_variable: | |
11137 { | |
11138 assert(_rettype == _data.variable->type()); | |
11139 | |
11140 if (_rettype == xpath_type_node_set) | |
11141 { | |
11142 const xpath_node_set& s = _data.variable->get_node_set(); | |
11143 | |
11144 xpath_node_set_raw ns; | |
11145 | |
11146 ns.set_type(s.type()); | |
11147 ns.append(s.begin(), s.end(), stack.result); | |
11148 | |
11149 return ns; | |
11150 } | |
11151 | |
11152 // variable needs to be converted to the correct type, this is handled by the fallthrough block below | |
11153 break; | |
11154 } | |
11155 | |
11156 default: | |
11157 ; | |
11158 } | |
11159 | |
11160 // none of the ast types that return the value directly matched, but conversions to node set are invalid | |
11161 assert(false && "Wrong expression for return type node set"); // unreachable | |
11162 return xpath_node_set_raw(); | |
11163 } | |
11164 | |
11165 void optimize(xpath_allocator* alloc) | |
11166 { | |
11167 if (_left) | |
11168 _left->optimize(alloc); | |
11169 | |
11170 if (_right) | |
11171 _right->optimize(alloc); | |
11172 | |
11173 if (_next) | |
11174 _next->optimize(alloc); | |
11175 | |
11176 // coverity[var_deref_model] | |
11177 optimize_self(alloc); | |
11178 } | |
11179 | |
11180 void optimize_self(xpath_allocator* alloc) | |
11181 { | |
11182 // Rewrite [position()=expr] with [expr] | |
11183 // Note that this step has to go before classification to recognize [position()=1] | |
11184 if ((_type == ast_filter || _type == ast_predicate) && | |
11185 _right && // workaround for clang static analyzer (_right is never null for ast_filter/ast_predicate) | |
11186 _right->_type == ast_op_equal && _right->_left->_type == ast_func_position && _right->_right->_rettype == xpath_type_number) | |
11187 { | |
11188 _right = _right->_right; | |
11189 } | |
11190 | |
11191 // Classify filter/predicate ops to perform various optimizations during evaluation | |
11192 if ((_type == ast_filter || _type == ast_predicate) && _right) // workaround for clang static analyzer (_right is never null for ast_filter/ast_predicate) | |
11193 { | |
11194 assert(_test == predicate_default); | |
11195 | |
11196 if (_right->_type == ast_number_constant && _right->_data.number == 1.0) | |
11197 _test = predicate_constant_one; | |
11198 else if (_right->_rettype == xpath_type_number && (_right->_type == ast_number_constant || _right->_type == ast_variable || _right->_type == ast_func_last)) | |
11199 _test = predicate_constant; | |
11200 else if (_right->_rettype != xpath_type_number && _right->is_posinv_expr()) | |
11201 _test = predicate_posinv; | |
11202 } | |
11203 | |
11204 // Rewrite descendant-or-self::node()/child::foo with descendant::foo | |
11205 // The former is a full form of //foo, the latter is much faster since it executes the node test immediately | |
11206 // Do a similar kind of rewrite for self/descendant/descendant-or-self axes | |
11207 // Note that we only rewrite positionally invariant steps (//foo[1] != /descendant::foo[1]) | |
11208 if (_type == ast_step && (_axis == axis_child || _axis == axis_self || _axis == axis_descendant || _axis == axis_descendant_or_self) && | |
11209 _left && _left->_type == ast_step && _left->_axis == axis_descendant_or_self && _left->_test == nodetest_type_node && !_left->_right && | |
11210 is_posinv_step()) | |
11211 { | |
11212 if (_axis == axis_child || _axis == axis_descendant) | |
11213 _axis = axis_descendant; | |
11214 else | |
11215 _axis = axis_descendant_or_self; | |
11216 | |
11217 _left = _left->_left; | |
11218 } | |
11219 | |
11220 // Use optimized lookup table implementation for translate() with constant arguments | |
11221 if (_type == ast_func_translate && | |
11222 _right && // workaround for clang static analyzer (_right is never null for ast_func_translate) | |
11223 _right->_type == ast_string_constant && _right->_next->_type == ast_string_constant) | |
11224 { | |
11225 unsigned char* table = translate_table_generate(alloc, _right->_data.string, _right->_next->_data.string); | |
11226 | |
11227 if (table) | |
11228 { | |
11229 _type = ast_opt_translate_table; | |
11230 _data.table = table; | |
11231 } | |
11232 } | |
11233 | |
11234 // Use optimized path for @attr = 'value' or @attr = $value | |
11235 if (_type == ast_op_equal && | |
11236 _left && _right && // workaround for clang static analyzer and Coverity (_left and _right are never null for ast_op_equal) | |
11237 // coverity[mixed_enums] | |
11238 _left->_type == ast_step && _left->_axis == axis_attribute && _left->_test == nodetest_name && !_left->_left && !_left->_right && | |
11239 (_right->_type == ast_string_constant || (_right->_type == ast_variable && _right->_rettype == xpath_type_string))) | |
11240 { | |
11241 _type = ast_opt_compare_attribute; | |
11242 } | |
11243 } | |
11244 | |
11245 bool is_posinv_expr() const | |
11246 { | |
11247 switch (_type) | |
11248 { | |
11249 case ast_func_position: | |
11250 case ast_func_last: | |
11251 return false; | |
11252 | |
11253 case ast_string_constant: | |
11254 case ast_number_constant: | |
11255 case ast_variable: | |
11256 return true; | |
11257 | |
11258 case ast_step: | |
11259 case ast_step_root: | |
11260 return true; | |
11261 | |
11262 case ast_predicate: | |
11263 case ast_filter: | |
11264 return true; | |
11265 | |
11266 default: | |
11267 if (_left && !_left->is_posinv_expr()) return false; | |
11268 | |
11269 for (xpath_ast_node* n = _right; n; n = n->_next) | |
11270 if (!n->is_posinv_expr()) return false; | |
11271 | |
11272 return true; | |
11273 } | |
11274 } | |
11275 | |
11276 bool is_posinv_step() const | |
11277 { | |
11278 assert(_type == ast_step); | |
11279 | |
11280 for (xpath_ast_node* n = _right; n; n = n->_next) | |
11281 { | |
11282 assert(n->_type == ast_predicate); | |
11283 | |
11284 if (n->_test != predicate_posinv) | |
11285 return false; | |
11286 } | |
11287 | |
11288 return true; | |
11289 } | |
11290 | |
11291 xpath_value_type rettype() const | |
11292 { | |
11293 return static_cast<xpath_value_type>(_rettype); | |
11294 } | |
11295 }; | |
11296 | |
11297 static const size_t xpath_ast_depth_limit = | |
11298 #ifdef PUGIXML_XPATH_DEPTH_LIMIT | |
11299 PUGIXML_XPATH_DEPTH_LIMIT | |
11300 #else | |
11301 1024 | |
11302 #endif | |
11303 ; | |
11304 | |
11305 struct xpath_parser | |
11306 { | |
11307 xpath_allocator* _alloc; | |
11308 xpath_lexer _lexer; | |
11309 | |
11310 const char_t* _query; | |
11311 xpath_variable_set* _variables; | |
11312 | |
11313 xpath_parse_result* _result; | |
11314 | |
11315 char_t _scratch[32]; | |
11316 | |
11317 size_t _depth; | |
11318 | |
11319 xpath_ast_node* error(const char* message) | |
11320 { | |
11321 _result->error = message; | |
11322 _result->offset = _lexer.current_pos() - _query; | |
11323 | |
11324 return 0; | |
11325 } | |
11326 | |
11327 xpath_ast_node* error_oom() | |
11328 { | |
11329 assert(_alloc->_error); | |
11330 *_alloc->_error = true; | |
11331 | |
11332 return 0; | |
11333 } | |
11334 | |
11335 xpath_ast_node* error_rec() | |
11336 { | |
11337 return error("Exceeded maximum allowed query depth"); | |
11338 } | |
11339 | |
11340 void* alloc_node() | |
11341 { | |
11342 return _alloc->allocate(sizeof(xpath_ast_node)); | |
11343 } | |
11344 | |
11345 xpath_ast_node* alloc_node(ast_type_t type, xpath_value_type rettype, const char_t* value) | |
11346 { | |
11347 void* memory = alloc_node(); | |
11348 return memory ? new (memory) xpath_ast_node(type, rettype, value) : 0; | |
11349 } | |
11350 | |
11351 xpath_ast_node* alloc_node(ast_type_t type, xpath_value_type rettype, double value) | |
11352 { | |
11353 void* memory = alloc_node(); | |
11354 return memory ? new (memory) xpath_ast_node(type, rettype, value) : 0; | |
11355 } | |
11356 | |
11357 xpath_ast_node* alloc_node(ast_type_t type, xpath_value_type rettype, xpath_variable* value) | |
11358 { | |
11359 void* memory = alloc_node(); | |
11360 return memory ? new (memory) xpath_ast_node(type, rettype, value) : 0; | |
11361 } | |
11362 | |
11363 xpath_ast_node* alloc_node(ast_type_t type, xpath_value_type rettype, xpath_ast_node* left = 0, xpath_ast_node* right = 0) | |
11364 { | |
11365 void* memory = alloc_node(); | |
11366 return memory ? new (memory) xpath_ast_node(type, rettype, left, right) : 0; | |
11367 } | |
11368 | |
11369 xpath_ast_node* alloc_node(ast_type_t type, xpath_ast_node* left, axis_t axis, nodetest_t test, const char_t* contents) | |
11370 { | |
11371 void* memory = alloc_node(); | |
11372 return memory ? new (memory) xpath_ast_node(type, left, axis, test, contents) : 0; | |
11373 } | |
11374 | |
11375 xpath_ast_node* alloc_node(ast_type_t type, xpath_ast_node* left, xpath_ast_node* right, predicate_t test) | |
11376 { | |
11377 void* memory = alloc_node(); | |
11378 return memory ? new (memory) xpath_ast_node(type, left, right, test) : 0; | |
11379 } | |
11380 | |
11381 const char_t* alloc_string(const xpath_lexer_string& value) | |
11382 { | |
11383 if (!value.begin) | |
11384 return PUGIXML_TEXT(""); | |
11385 | |
11386 size_t length = static_cast<size_t>(value.end - value.begin); | |
11387 | |
11388 char_t* c = static_cast<char_t*>(_alloc->allocate((length + 1) * sizeof(char_t))); | |
11389 if (!c) return 0; | |
11390 | |
11391 memcpy(c, value.begin, length * sizeof(char_t)); | |
11392 c[length] = 0; | |
11393 | |
11394 return c; | |
11395 } | |
11396 | |
11397 xpath_ast_node* parse_function(const xpath_lexer_string& name, size_t argc, xpath_ast_node* args[2]) | |
11398 { | |
11399 switch (name.begin[0]) | |
11400 { | |
11401 case 'b': | |
11402 if (name == PUGIXML_TEXT("boolean") && argc == 1) | |
11403 return alloc_node(ast_func_boolean, xpath_type_boolean, args[0]); | |
11404 | |
11405 break; | |
11406 | |
11407 case 'c': | |
11408 if (name == PUGIXML_TEXT("count") && argc == 1) | |
11409 { | |
11410 if (args[0]->rettype() != xpath_type_node_set) return error("Function has to be applied to node set"); | |
11411 return alloc_node(ast_func_count, xpath_type_number, args[0]); | |
11412 } | |
11413 else if (name == PUGIXML_TEXT("contains") && argc == 2) | |
11414 return alloc_node(ast_func_contains, xpath_type_boolean, args[0], args[1]); | |
11415 else if (name == PUGIXML_TEXT("concat") && argc >= 2) | |
11416 return alloc_node(ast_func_concat, xpath_type_string, args[0], args[1]); | |
11417 else if (name == PUGIXML_TEXT("ceiling") && argc == 1) | |
11418 return alloc_node(ast_func_ceiling, xpath_type_number, args[0]); | |
11419 | |
11420 break; | |
11421 | |
11422 case 'f': | |
11423 if (name == PUGIXML_TEXT("false") && argc == 0) | |
11424 return alloc_node(ast_func_false, xpath_type_boolean); | |
11425 else if (name == PUGIXML_TEXT("floor") && argc == 1) | |
11426 return alloc_node(ast_func_floor, xpath_type_number, args[0]); | |
11427 | |
11428 break; | |
11429 | |
11430 case 'i': | |
11431 if (name == PUGIXML_TEXT("id") && argc == 1) | |
11432 return alloc_node(ast_func_id, xpath_type_node_set, args[0]); | |
11433 | |
11434 break; | |
11435 | |
11436 case 'l': | |
11437 if (name == PUGIXML_TEXT("last") && argc == 0) | |
11438 return alloc_node(ast_func_last, xpath_type_number); | |
11439 else if (name == PUGIXML_TEXT("lang") && argc == 1) | |
11440 return alloc_node(ast_func_lang, xpath_type_boolean, args[0]); | |
11441 else if (name == PUGIXML_TEXT("local-name") && argc <= 1) | |
11442 { | |
11443 if (argc == 1 && args[0]->rettype() != xpath_type_node_set) return error("Function has to be applied to node set"); | |
11444 return alloc_node(argc == 0 ? ast_func_local_name_0 : ast_func_local_name_1, xpath_type_string, args[0]); | |
11445 } | |
11446 | |
11447 break; | |
11448 | |
11449 case 'n': | |
11450 if (name == PUGIXML_TEXT("name") && argc <= 1) | |
11451 { | |
11452 if (argc == 1 && args[0]->rettype() != xpath_type_node_set) return error("Function has to be applied to node set"); | |
11453 return alloc_node(argc == 0 ? ast_func_name_0 : ast_func_name_1, xpath_type_string, args[0]); | |
11454 } | |
11455 else if (name == PUGIXML_TEXT("namespace-uri") && argc <= 1) | |
11456 { | |
11457 if (argc == 1 && args[0]->rettype() != xpath_type_node_set) return error("Function has to be applied to node set"); | |
11458 return alloc_node(argc == 0 ? ast_func_namespace_uri_0 : ast_func_namespace_uri_1, xpath_type_string, args[0]); | |
11459 } | |
11460 else if (name == PUGIXML_TEXT("normalize-space") && argc <= 1) | |
11461 return alloc_node(argc == 0 ? ast_func_normalize_space_0 : ast_func_normalize_space_1, xpath_type_string, args[0], args[1]); | |
11462 else if (name == PUGIXML_TEXT("not") && argc == 1) | |
11463 return alloc_node(ast_func_not, xpath_type_boolean, args[0]); | |
11464 else if (name == PUGIXML_TEXT("number") && argc <= 1) | |
11465 return alloc_node(argc == 0 ? ast_func_number_0 : ast_func_number_1, xpath_type_number, args[0]); | |
11466 | |
11467 break; | |
11468 | |
11469 case 'p': | |
11470 if (name == PUGIXML_TEXT("position") && argc == 0) | |
11471 return alloc_node(ast_func_position, xpath_type_number); | |
11472 | |
11473 break; | |
11474 | |
11475 case 'r': | |
11476 if (name == PUGIXML_TEXT("round") && argc == 1) | |
11477 return alloc_node(ast_func_round, xpath_type_number, args[0]); | |
11478 | |
11479 break; | |
11480 | |
11481 case 's': | |
11482 if (name == PUGIXML_TEXT("string") && argc <= 1) | |
11483 return alloc_node(argc == 0 ? ast_func_string_0 : ast_func_string_1, xpath_type_string, args[0]); | |
11484 else if (name == PUGIXML_TEXT("string-length") && argc <= 1) | |
11485 return alloc_node(argc == 0 ? ast_func_string_length_0 : ast_func_string_length_1, xpath_type_number, args[0]); | |
11486 else if (name == PUGIXML_TEXT("starts-with") && argc == 2) | |
11487 return alloc_node(ast_func_starts_with, xpath_type_boolean, args[0], args[1]); | |
11488 else if (name == PUGIXML_TEXT("substring-before") && argc == 2) | |
11489 return alloc_node(ast_func_substring_before, xpath_type_string, args[0], args[1]); | |
11490 else if (name == PUGIXML_TEXT("substring-after") && argc == 2) | |
11491 return alloc_node(ast_func_substring_after, xpath_type_string, args[0], args[1]); | |
11492 else if (name == PUGIXML_TEXT("substring") && (argc == 2 || argc == 3)) | |
11493 return alloc_node(argc == 2 ? ast_func_substring_2 : ast_func_substring_3, xpath_type_string, args[0], args[1]); | |
11494 else if (name == PUGIXML_TEXT("sum") && argc == 1) | |
11495 { | |
11496 if (args[0]->rettype() != xpath_type_node_set) return error("Function has to be applied to node set"); | |
11497 return alloc_node(ast_func_sum, xpath_type_number, args[0]); | |
11498 } | |
11499 | |
11500 break; | |
11501 | |
11502 case 't': | |
11503 if (name == PUGIXML_TEXT("translate") && argc == 3) | |
11504 return alloc_node(ast_func_translate, xpath_type_string, args[0], args[1]); | |
11505 else if (name == PUGIXML_TEXT("true") && argc == 0) | |
11506 return alloc_node(ast_func_true, xpath_type_boolean); | |
11507 | |
11508 break; | |
11509 | |
11510 default: | |
11511 break; | |
11512 } | |
11513 | |
11514 return error("Unrecognized function or wrong parameter count"); | |
11515 } | |
11516 | |
11517 axis_t parse_axis_name(const xpath_lexer_string& name, bool& specified) | |
11518 { | |
11519 specified = true; | |
11520 | |
11521 switch (name.begin[0]) | |
11522 { | |
11523 case 'a': | |
11524 if (name == PUGIXML_TEXT("ancestor")) | |
11525 return axis_ancestor; | |
11526 else if (name == PUGIXML_TEXT("ancestor-or-self")) | |
11527 return axis_ancestor_or_self; | |
11528 else if (name == PUGIXML_TEXT("attribute")) | |
11529 return axis_attribute; | |
11530 | |
11531 break; | |
11532 | |
11533 case 'c': | |
11534 if (name == PUGIXML_TEXT("child")) | |
11535 return axis_child; | |
11536 | |
11537 break; | |
11538 | |
11539 case 'd': | |
11540 if (name == PUGIXML_TEXT("descendant")) | |
11541 return axis_descendant; | |
11542 else if (name == PUGIXML_TEXT("descendant-or-self")) | |
11543 return axis_descendant_or_self; | |
11544 | |
11545 break; | |
11546 | |
11547 case 'f': | |
11548 if (name == PUGIXML_TEXT("following")) | |
11549 return axis_following; | |
11550 else if (name == PUGIXML_TEXT("following-sibling")) | |
11551 return axis_following_sibling; | |
11552 | |
11553 break; | |
11554 | |
11555 case 'n': | |
11556 if (name == PUGIXML_TEXT("namespace")) | |
11557 return axis_namespace; | |
11558 | |
11559 break; | |
11560 | |
11561 case 'p': | |
11562 if (name == PUGIXML_TEXT("parent")) | |
11563 return axis_parent; | |
11564 else if (name == PUGIXML_TEXT("preceding")) | |
11565 return axis_preceding; | |
11566 else if (name == PUGIXML_TEXT("preceding-sibling")) | |
11567 return axis_preceding_sibling; | |
11568 | |
11569 break; | |
11570 | |
11571 case 's': | |
11572 if (name == PUGIXML_TEXT("self")) | |
11573 return axis_self; | |
11574 | |
11575 break; | |
11576 | |
11577 default: | |
11578 break; | |
11579 } | |
11580 | |
11581 specified = false; | |
11582 return axis_child; | |
11583 } | |
11584 | |
11585 nodetest_t parse_node_test_type(const xpath_lexer_string& name) | |
11586 { | |
11587 switch (name.begin[0]) | |
11588 { | |
11589 case 'c': | |
11590 if (name == PUGIXML_TEXT("comment")) | |
11591 return nodetest_type_comment; | |
11592 | |
11593 break; | |
11594 | |
11595 case 'n': | |
11596 if (name == PUGIXML_TEXT("node")) | |
11597 return nodetest_type_node; | |
11598 | |
11599 break; | |
11600 | |
11601 case 'p': | |
11602 if (name == PUGIXML_TEXT("processing-instruction")) | |
11603 return nodetest_type_pi; | |
11604 | |
11605 break; | |
11606 | |
11607 case 't': | |
11608 if (name == PUGIXML_TEXT("text")) | |
11609 return nodetest_type_text; | |
11610 | |
11611 break; | |
11612 | |
11613 default: | |
11614 break; | |
11615 } | |
11616 | |
11617 return nodetest_none; | |
11618 } | |
11619 | |
11620 // PrimaryExpr ::= VariableReference | '(' Expr ')' | Literal | Number | FunctionCall | |
11621 xpath_ast_node* parse_primary_expression() | |
11622 { | |
11623 switch (_lexer.current()) | |
11624 { | |
11625 case lex_var_ref: | |
11626 { | |
11627 xpath_lexer_string name = _lexer.contents(); | |
11628 | |
11629 if (!_variables) | |
11630 return error("Unknown variable: variable set is not provided"); | |
11631 | |
11632 xpath_variable* var = 0; | |
11633 if (!get_variable_scratch(_scratch, _variables, name.begin, name.end, &var)) | |
11634 return error_oom(); | |
11635 | |
11636 if (!var) | |
11637 return error("Unknown variable: variable set does not contain the given name"); | |
11638 | |
11639 _lexer.next(); | |
11640 | |
11641 return alloc_node(ast_variable, var->type(), var); | |
11642 } | |
11643 | |
11644 case lex_open_brace: | |
11645 { | |
11646 _lexer.next(); | |
11647 | |
11648 xpath_ast_node* n = parse_expression(); | |
11649 if (!n) return 0; | |
11650 | |
11651 if (_lexer.current() != lex_close_brace) | |
11652 return error("Expected ')' to match an opening '('"); | |
11653 | |
11654 _lexer.next(); | |
11655 | |
11656 return n; | |
11657 } | |
11658 | |
11659 case lex_quoted_string: | |
11660 { | |
11661 const char_t* value = alloc_string(_lexer.contents()); | |
11662 if (!value) return 0; | |
11663 | |
11664 _lexer.next(); | |
11665 | |
11666 return alloc_node(ast_string_constant, xpath_type_string, value); | |
11667 } | |
11668 | |
11669 case lex_number: | |
11670 { | |
11671 double value = 0; | |
11672 | |
11673 if (!convert_string_to_number_scratch(_scratch, _lexer.contents().begin, _lexer.contents().end, &value)) | |
11674 return error_oom(); | |
11675 | |
11676 _lexer.next(); | |
11677 | |
11678 return alloc_node(ast_number_constant, xpath_type_number, value); | |
11679 } | |
11680 | |
11681 case lex_string: | |
11682 { | |
11683 xpath_ast_node* args[2] = {0}; | |
11684 size_t argc = 0; | |
11685 | |
11686 xpath_lexer_string function = _lexer.contents(); | |
11687 _lexer.next(); | |
11688 | |
11689 xpath_ast_node* last_arg = 0; | |
11690 | |
11691 if (_lexer.current() != lex_open_brace) | |
11692 return error("Unrecognized function call"); | |
11693 _lexer.next(); | |
11694 | |
11695 size_t old_depth = _depth; | |
11696 | |
11697 while (_lexer.current() != lex_close_brace) | |
11698 { | |
11699 if (argc > 0) | |
11700 { | |
11701 if (_lexer.current() != lex_comma) | |
11702 return error("No comma between function arguments"); | |
11703 _lexer.next(); | |
11704 } | |
11705 | |
11706 if (++_depth > xpath_ast_depth_limit) | |
11707 return error_rec(); | |
11708 | |
11709 xpath_ast_node* n = parse_expression(); | |
11710 if (!n) return 0; | |
11711 | |
11712 if (argc < 2) args[argc] = n; | |
11713 else last_arg->set_next(n); | |
11714 | |
11715 argc++; | |
11716 last_arg = n; | |
11717 } | |
11718 | |
11719 _lexer.next(); | |
11720 | |
11721 _depth = old_depth; | |
11722 | |
11723 return parse_function(function, argc, args); | |
11724 } | |
11725 | |
11726 default: | |
11727 return error("Unrecognizable primary expression"); | |
11728 } | |
11729 } | |
11730 | |
11731 // FilterExpr ::= PrimaryExpr | FilterExpr Predicate | |
11732 // Predicate ::= '[' PredicateExpr ']' | |
11733 // PredicateExpr ::= Expr | |
11734 xpath_ast_node* parse_filter_expression() | |
11735 { | |
11736 xpath_ast_node* n = parse_primary_expression(); | |
11737 if (!n) return 0; | |
11738 | |
11739 size_t old_depth = _depth; | |
11740 | |
11741 while (_lexer.current() == lex_open_square_brace) | |
11742 { | |
11743 _lexer.next(); | |
11744 | |
11745 if (++_depth > xpath_ast_depth_limit) | |
11746 return error_rec(); | |
11747 | |
11748 if (n->rettype() != xpath_type_node_set) | |
11749 return error("Predicate has to be applied to node set"); | |
11750 | |
11751 xpath_ast_node* expr = parse_expression(); | |
11752 if (!expr) return 0; | |
11753 | |
11754 n = alloc_node(ast_filter, n, expr, predicate_default); | |
11755 if (!n) return 0; | |
11756 | |
11757 if (_lexer.current() != lex_close_square_brace) | |
11758 return error("Expected ']' to match an opening '['"); | |
11759 | |
11760 _lexer.next(); | |
11761 } | |
11762 | |
11763 _depth = old_depth; | |
11764 | |
11765 return n; | |
11766 } | |
11767 | |
11768 // Step ::= AxisSpecifier NodeTest Predicate* | AbbreviatedStep | |
11769 // AxisSpecifier ::= AxisName '::' | '@'? | |
11770 // NodeTest ::= NameTest | NodeType '(' ')' | 'processing-instruction' '(' Literal ')' | |
11771 // NameTest ::= '*' | NCName ':' '*' | QName | |
11772 // AbbreviatedStep ::= '.' | '..' | |
11773 xpath_ast_node* parse_step(xpath_ast_node* set) | |
11774 { | |
11775 if (set && set->rettype() != xpath_type_node_set) | |
11776 return error("Step has to be applied to node set"); | |
11777 | |
11778 bool axis_specified = false; | |
11779 axis_t axis = axis_child; // implied child axis | |
11780 | |
11781 if (_lexer.current() == lex_axis_attribute) | |
11782 { | |
11783 axis = axis_attribute; | |
11784 axis_specified = true; | |
11785 | |
11786 _lexer.next(); | |
11787 } | |
11788 else if (_lexer.current() == lex_dot) | |
11789 { | |
11790 _lexer.next(); | |
11791 | |
11792 if (_lexer.current() == lex_open_square_brace) | |
11793 return error("Predicates are not allowed after an abbreviated step"); | |
11794 | |
11795 return alloc_node(ast_step, set, axis_self, nodetest_type_node, 0); | |
11796 } | |
11797 else if (_lexer.current() == lex_double_dot) | |
11798 { | |
11799 _lexer.next(); | |
11800 | |
11801 if (_lexer.current() == lex_open_square_brace) | |
11802 return error("Predicates are not allowed after an abbreviated step"); | |
11803 | |
11804 return alloc_node(ast_step, set, axis_parent, nodetest_type_node, 0); | |
11805 } | |
11806 | |
11807 nodetest_t nt_type = nodetest_none; | |
11808 xpath_lexer_string nt_name; | |
11809 | |
11810 if (_lexer.current() == lex_string) | |
11811 { | |
11812 // node name test | |
11813 nt_name = _lexer.contents(); | |
11814 _lexer.next(); | |
11815 | |
11816 // was it an axis name? | |
11817 if (_lexer.current() == lex_double_colon) | |
11818 { | |
11819 // parse axis name | |
11820 if (axis_specified) | |
11821 return error("Two axis specifiers in one step"); | |
11822 | |
11823 axis = parse_axis_name(nt_name, axis_specified); | |
11824 | |
11825 if (!axis_specified) | |
11826 return error("Unknown axis"); | |
11827 | |
11828 // read actual node test | |
11829 _lexer.next(); | |
11830 | |
11831 if (_lexer.current() == lex_multiply) | |
11832 { | |
11833 nt_type = nodetest_all; | |
11834 nt_name = xpath_lexer_string(); | |
11835 _lexer.next(); | |
11836 } | |
11837 else if (_lexer.current() == lex_string) | |
11838 { | |
11839 nt_name = _lexer.contents(); | |
11840 _lexer.next(); | |
11841 } | |
11842 else | |
11843 { | |
11844 return error("Unrecognized node test"); | |
11845 } | |
11846 } | |
11847 | |
11848 if (nt_type == nodetest_none) | |
11849 { | |
11850 // node type test or processing-instruction | |
11851 if (_lexer.current() == lex_open_brace) | |
11852 { | |
11853 _lexer.next(); | |
11854 | |
11855 if (_lexer.current() == lex_close_brace) | |
11856 { | |
11857 _lexer.next(); | |
11858 | |
11859 nt_type = parse_node_test_type(nt_name); | |
11860 | |
11861 if (nt_type == nodetest_none) | |
11862 return error("Unrecognized node type"); | |
11863 | |
11864 nt_name = xpath_lexer_string(); | |
11865 } | |
11866 else if (nt_name == PUGIXML_TEXT("processing-instruction")) | |
11867 { | |
11868 if (_lexer.current() != lex_quoted_string) | |
11869 return error("Only literals are allowed as arguments to processing-instruction()"); | |
11870 | |
11871 nt_type = nodetest_pi; | |
11872 nt_name = _lexer.contents(); | |
11873 _lexer.next(); | |
11874 | |
11875 if (_lexer.current() != lex_close_brace) | |
11876 return error("Unmatched brace near processing-instruction()"); | |
11877 _lexer.next(); | |
11878 } | |
11879 else | |
11880 { | |
11881 return error("Unmatched brace near node type test"); | |
11882 } | |
11883 } | |
11884 // QName or NCName:* | |
11885 else | |
11886 { | |
11887 if (nt_name.end - nt_name.begin > 2 && nt_name.end[-2] == ':' && nt_name.end[-1] == '*') // NCName:* | |
11888 { | |
11889 nt_name.end--; // erase * | |
11890 | |
11891 nt_type = nodetest_all_in_namespace; | |
11892 } | |
11893 else | |
11894 { | |
11895 nt_type = nodetest_name; | |
11896 } | |
11897 } | |
11898 } | |
11899 } | |
11900 else if (_lexer.current() == lex_multiply) | |
11901 { | |
11902 nt_type = nodetest_all; | |
11903 _lexer.next(); | |
11904 } | |
11905 else | |
11906 { | |
11907 return error("Unrecognized node test"); | |
11908 } | |
11909 | |
11910 const char_t* nt_name_copy = alloc_string(nt_name); | |
11911 if (!nt_name_copy) return 0; | |
11912 | |
11913 xpath_ast_node* n = alloc_node(ast_step, set, axis, nt_type, nt_name_copy); | |
11914 if (!n) return 0; | |
11915 | |
11916 size_t old_depth = _depth; | |
11917 | |
11918 xpath_ast_node* last = 0; | |
11919 | |
11920 while (_lexer.current() == lex_open_square_brace) | |
11921 { | |
11922 _lexer.next(); | |
11923 | |
11924 if (++_depth > xpath_ast_depth_limit) | |
11925 return error_rec(); | |
11926 | |
11927 xpath_ast_node* expr = parse_expression(); | |
11928 if (!expr) return 0; | |
11929 | |
11930 xpath_ast_node* pred = alloc_node(ast_predicate, 0, expr, predicate_default); | |
11931 if (!pred) return 0; | |
11932 | |
11933 if (_lexer.current() != lex_close_square_brace) | |
11934 return error("Expected ']' to match an opening '['"); | |
11935 _lexer.next(); | |
11936 | |
11937 if (last) last->set_next(pred); | |
11938 else n->set_right(pred); | |
11939 | |
11940 last = pred; | |
11941 } | |
11942 | |
11943 _depth = old_depth; | |
11944 | |
11945 return n; | |
11946 } | |
11947 | |
11948 // RelativeLocationPath ::= Step | RelativeLocationPath '/' Step | RelativeLocationPath '//' Step | |
11949 xpath_ast_node* parse_relative_location_path(xpath_ast_node* set) | |
11950 { | |
11951 xpath_ast_node* n = parse_step(set); | |
11952 if (!n) return 0; | |
11953 | |
11954 size_t old_depth = _depth; | |
11955 | |
11956 while (_lexer.current() == lex_slash || _lexer.current() == lex_double_slash) | |
11957 { | |
11958 lexeme_t l = _lexer.current(); | |
11959 _lexer.next(); | |
11960 | |
11961 if (l == lex_double_slash) | |
11962 { | |
11963 n = alloc_node(ast_step, n, axis_descendant_or_self, nodetest_type_node, 0); | |
11964 if (!n) return 0; | |
11965 | |
11966 ++_depth; | |
11967 } | |
11968 | |
11969 if (++_depth > xpath_ast_depth_limit) | |
11970 return error_rec(); | |
11971 | |
11972 n = parse_step(n); | |
11973 if (!n) return 0; | |
11974 } | |
11975 | |
11976 _depth = old_depth; | |
11977 | |
11978 return n; | |
11979 } | |
11980 | |
11981 // LocationPath ::= RelativeLocationPath | AbsoluteLocationPath | |
11982 // AbsoluteLocationPath ::= '/' RelativeLocationPath? | '//' RelativeLocationPath | |
11983 xpath_ast_node* parse_location_path() | |
11984 { | |
11985 if (_lexer.current() == lex_slash) | |
11986 { | |
11987 _lexer.next(); | |
11988 | |
11989 xpath_ast_node* n = alloc_node(ast_step_root, xpath_type_node_set); | |
11990 if (!n) return 0; | |
11991 | |
11992 // relative location path can start from axis_attribute, dot, double_dot, multiply and string lexemes; any other lexeme means standalone root path | |
11993 lexeme_t l = _lexer.current(); | |
11994 | |
11995 if (l == lex_string || l == lex_axis_attribute || l == lex_dot || l == lex_double_dot || l == lex_multiply) | |
11996 return parse_relative_location_path(n); | |
11997 else | |
11998 return n; | |
11999 } | |
12000 else if (_lexer.current() == lex_double_slash) | |
12001 { | |
12002 _lexer.next(); | |
12003 | |
12004 xpath_ast_node* n = alloc_node(ast_step_root, xpath_type_node_set); | |
12005 if (!n) return 0; | |
12006 | |
12007 n = alloc_node(ast_step, n, axis_descendant_or_self, nodetest_type_node, 0); | |
12008 if (!n) return 0; | |
12009 | |
12010 return parse_relative_location_path(n); | |
12011 } | |
12012 | |
12013 // else clause moved outside of if because of bogus warning 'control may reach end of non-void function being inlined' in gcc 4.0.1 | |
12014 return parse_relative_location_path(0); | |
12015 } | |
12016 | |
12017 // PathExpr ::= LocationPath | |
12018 // | FilterExpr | |
12019 // | FilterExpr '/' RelativeLocationPath | |
12020 // | FilterExpr '//' RelativeLocationPath | |
12021 // UnionExpr ::= PathExpr | UnionExpr '|' PathExpr | |
12022 // UnaryExpr ::= UnionExpr | '-' UnaryExpr | |
12023 xpath_ast_node* parse_path_or_unary_expression() | |
12024 { | |
12025 // Clarification. | |
12026 // PathExpr begins with either LocationPath or FilterExpr. | |
12027 // FilterExpr begins with PrimaryExpr | |
12028 // PrimaryExpr begins with '$' in case of it being a variable reference, | |
12029 // '(' in case of it being an expression, string literal, number constant or | |
12030 // function call. | |
12031 if (_lexer.current() == lex_var_ref || _lexer.current() == lex_open_brace || | |
12032 _lexer.current() == lex_quoted_string || _lexer.current() == lex_number || | |
12033 _lexer.current() == lex_string) | |
12034 { | |
12035 if (_lexer.current() == lex_string) | |
12036 { | |
12037 // This is either a function call, or not - if not, we shall proceed with location path | |
12038 const char_t* state = _lexer.state(); | |
12039 | |
12040 while (PUGI__IS_CHARTYPE(*state, ct_space)) ++state; | |
12041 | |
12042 if (*state != '(') | |
12043 return parse_location_path(); | |
12044 | |
12045 // This looks like a function call; however this still can be a node-test. Check it. | |
12046 if (parse_node_test_type(_lexer.contents()) != nodetest_none) | |
12047 return parse_location_path(); | |
12048 } | |
12049 | |
12050 xpath_ast_node* n = parse_filter_expression(); | |
12051 if (!n) return 0; | |
12052 | |
12053 if (_lexer.current() == lex_slash || _lexer.current() == lex_double_slash) | |
12054 { | |
12055 lexeme_t l = _lexer.current(); | |
12056 _lexer.next(); | |
12057 | |
12058 if (l == lex_double_slash) | |
12059 { | |
12060 if (n->rettype() != xpath_type_node_set) | |
12061 return error("Step has to be applied to node set"); | |
12062 | |
12063 n = alloc_node(ast_step, n, axis_descendant_or_self, nodetest_type_node, 0); | |
12064 if (!n) return 0; | |
12065 } | |
12066 | |
12067 // select from location path | |
12068 return parse_relative_location_path(n); | |
12069 } | |
12070 | |
12071 return n; | |
12072 } | |
12073 else if (_lexer.current() == lex_minus) | |
12074 { | |
12075 _lexer.next(); | |
12076 | |
12077 // precedence 7+ - only parses union expressions | |
12078 xpath_ast_node* n = parse_expression(7); | |
12079 if (!n) return 0; | |
12080 | |
12081 return alloc_node(ast_op_negate, xpath_type_number, n); | |
12082 } | |
12083 else | |
12084 { | |
12085 return parse_location_path(); | |
12086 } | |
12087 } | |
12088 | |
12089 struct binary_op_t | |
12090 { | |
12091 ast_type_t asttype; | |
12092 xpath_value_type rettype; | |
12093 int precedence; | |
12094 | |
12095 binary_op_t(): asttype(ast_unknown), rettype(xpath_type_none), precedence(0) | |
12096 { | |
12097 } | |
12098 | |
12099 binary_op_t(ast_type_t asttype_, xpath_value_type rettype_, int precedence_): asttype(asttype_), rettype(rettype_), precedence(precedence_) | |
12100 { | |
12101 } | |
12102 | |
12103 static binary_op_t parse(xpath_lexer& lexer) | |
12104 { | |
12105 switch (lexer.current()) | |
12106 { | |
12107 case lex_string: | |
12108 if (lexer.contents() == PUGIXML_TEXT("or")) | |
12109 return binary_op_t(ast_op_or, xpath_type_boolean, 1); | |
12110 else if (lexer.contents() == PUGIXML_TEXT("and")) | |
12111 return binary_op_t(ast_op_and, xpath_type_boolean, 2); | |
12112 else if (lexer.contents() == PUGIXML_TEXT("div")) | |
12113 return binary_op_t(ast_op_divide, xpath_type_number, 6); | |
12114 else if (lexer.contents() == PUGIXML_TEXT("mod")) | |
12115 return binary_op_t(ast_op_mod, xpath_type_number, 6); | |
12116 else | |
12117 return binary_op_t(); | |
12118 | |
12119 case lex_equal: | |
12120 return binary_op_t(ast_op_equal, xpath_type_boolean, 3); | |
12121 | |
12122 case lex_not_equal: | |
12123 return binary_op_t(ast_op_not_equal, xpath_type_boolean, 3); | |
12124 | |
12125 case lex_less: | |
12126 return binary_op_t(ast_op_less, xpath_type_boolean, 4); | |
12127 | |
12128 case lex_greater: | |
12129 return binary_op_t(ast_op_greater, xpath_type_boolean, 4); | |
12130 | |
12131 case lex_less_or_equal: | |
12132 return binary_op_t(ast_op_less_or_equal, xpath_type_boolean, 4); | |
12133 | |
12134 case lex_greater_or_equal: | |
12135 return binary_op_t(ast_op_greater_or_equal, xpath_type_boolean, 4); | |
12136 | |
12137 case lex_plus: | |
12138 return binary_op_t(ast_op_add, xpath_type_number, 5); | |
12139 | |
12140 case lex_minus: | |
12141 return binary_op_t(ast_op_subtract, xpath_type_number, 5); | |
12142 | |
12143 case lex_multiply: | |
12144 return binary_op_t(ast_op_multiply, xpath_type_number, 6); | |
12145 | |
12146 case lex_union: | |
12147 return binary_op_t(ast_op_union, xpath_type_node_set, 7); | |
12148 | |
12149 default: | |
12150 return binary_op_t(); | |
12151 } | |
12152 } | |
12153 }; | |
12154 | |
12155 xpath_ast_node* parse_expression_rec(xpath_ast_node* lhs, int limit) | |
12156 { | |
12157 binary_op_t op = binary_op_t::parse(_lexer); | |
12158 | |
12159 while (op.asttype != ast_unknown && op.precedence >= limit) | |
12160 { | |
12161 _lexer.next(); | |
12162 | |
12163 if (++_depth > xpath_ast_depth_limit) | |
12164 return error_rec(); | |
12165 | |
12166 xpath_ast_node* rhs = parse_path_or_unary_expression(); | |
12167 if (!rhs) return 0; | |
12168 | |
12169 binary_op_t nextop = binary_op_t::parse(_lexer); | |
12170 | |
12171 while (nextop.asttype != ast_unknown && nextop.precedence > op.precedence) | |
12172 { | |
12173 rhs = parse_expression_rec(rhs, nextop.precedence); | |
12174 if (!rhs) return 0; | |
12175 | |
12176 nextop = binary_op_t::parse(_lexer); | |
12177 } | |
12178 | |
12179 if (op.asttype == ast_op_union && (lhs->rettype() != xpath_type_node_set || rhs->rettype() != xpath_type_node_set)) | |
12180 return error("Union operator has to be applied to node sets"); | |
12181 | |
12182 lhs = alloc_node(op.asttype, op.rettype, lhs, rhs); | |
12183 if (!lhs) return 0; | |
12184 | |
12185 op = binary_op_t::parse(_lexer); | |
12186 } | |
12187 | |
12188 return lhs; | |
12189 } | |
12190 | |
12191 // Expr ::= OrExpr | |
12192 // OrExpr ::= AndExpr | OrExpr 'or' AndExpr | |
12193 // AndExpr ::= EqualityExpr | AndExpr 'and' EqualityExpr | |
12194 // EqualityExpr ::= RelationalExpr | |
12195 // | EqualityExpr '=' RelationalExpr | |
12196 // | EqualityExpr '!=' RelationalExpr | |
12197 // RelationalExpr ::= AdditiveExpr | |
12198 // | RelationalExpr '<' AdditiveExpr | |
12199 // | RelationalExpr '>' AdditiveExpr | |
12200 // | RelationalExpr '<=' AdditiveExpr | |
12201 // | RelationalExpr '>=' AdditiveExpr | |
12202 // AdditiveExpr ::= MultiplicativeExpr | |
12203 // | AdditiveExpr '+' MultiplicativeExpr | |
12204 // | AdditiveExpr '-' MultiplicativeExpr | |
12205 // MultiplicativeExpr ::= UnaryExpr | |
12206 // | MultiplicativeExpr '*' UnaryExpr | |
12207 // | MultiplicativeExpr 'div' UnaryExpr | |
12208 // | MultiplicativeExpr 'mod' UnaryExpr | |
12209 xpath_ast_node* parse_expression(int limit = 0) | |
12210 { | |
12211 size_t old_depth = _depth; | |
12212 | |
12213 if (++_depth > xpath_ast_depth_limit) | |
12214 return error_rec(); | |
12215 | |
12216 xpath_ast_node* n = parse_path_or_unary_expression(); | |
12217 if (!n) return 0; | |
12218 | |
12219 n = parse_expression_rec(n, limit); | |
12220 | |
12221 _depth = old_depth; | |
12222 | |
12223 return n; | |
12224 } | |
12225 | |
12226 xpath_parser(const char_t* query, xpath_variable_set* variables, xpath_allocator* alloc, xpath_parse_result* result): _alloc(alloc), _lexer(query), _query(query), _variables(variables), _result(result), _depth(0) | |
12227 { | |
12228 } | |
12229 | |
12230 xpath_ast_node* parse() | |
12231 { | |
12232 xpath_ast_node* n = parse_expression(); | |
12233 if (!n) return 0; | |
12234 | |
12235 assert(_depth == 0); | |
12236 | |
12237 // check if there are unparsed tokens left | |
12238 if (_lexer.current() != lex_eof) | |
12239 return error("Incorrect query"); | |
12240 | |
12241 return n; | |
12242 } | |
12243 | |
12244 static xpath_ast_node* parse(const char_t* query, xpath_variable_set* variables, xpath_allocator* alloc, xpath_parse_result* result) | |
12245 { | |
12246 xpath_parser parser(query, variables, alloc, result); | |
12247 | |
12248 return parser.parse(); | |
12249 } | |
12250 }; | |
12251 | |
12252 struct xpath_query_impl | |
12253 { | |
12254 static xpath_query_impl* create() | |
12255 { | |
12256 void* memory = xml_memory::allocate(sizeof(xpath_query_impl)); | |
12257 if (!memory) return 0; | |
12258 | |
12259 return new (memory) xpath_query_impl(); | |
12260 } | |
12261 | |
12262 static void destroy(xpath_query_impl* impl) | |
12263 { | |
12264 // free all allocated pages | |
12265 impl->alloc.release(); | |
12266 | |
12267 // free allocator memory (with the first page) | |
12268 xml_memory::deallocate(impl); | |
12269 } | |
12270 | |
12271 xpath_query_impl(): root(0), alloc(&block, &oom), oom(false) | |
12272 { | |
12273 block.next = 0; | |
12274 block.capacity = sizeof(block.data); | |
12275 } | |
12276 | |
12277 xpath_ast_node* root; | |
12278 xpath_allocator alloc; | |
12279 xpath_memory_block block; | |
12280 bool oom; | |
12281 }; | |
12282 | |
12283 PUGI__FN impl::xpath_ast_node* evaluate_node_set_prepare(xpath_query_impl* impl) | |
12284 { | |
12285 if (!impl) return 0; | |
12286 | |
12287 if (impl->root->rettype() != xpath_type_node_set) | |
12288 { | |
12289 #ifdef PUGIXML_NO_EXCEPTIONS | |
12290 return 0; | |
12291 #else | |
12292 xpath_parse_result res; | |
12293 res.error = "Expression does not evaluate to node set"; | |
12294 | |
12295 throw xpath_exception(res); | |
12296 #endif | |
12297 } | |
12298 | |
12299 return impl->root; | |
12300 } | |
12301 PUGI__NS_END | |
12302 | |
12303 namespace pugi | |
12304 { | |
12305 #ifndef PUGIXML_NO_EXCEPTIONS | |
12306 PUGI__FN xpath_exception::xpath_exception(const xpath_parse_result& result_): _result(result_) | |
12307 { | |
12308 assert(_result.error); | |
12309 } | |
12310 | |
12311 PUGI__FN const char* xpath_exception::what() const throw() | |
12312 { | |
12313 return _result.error; | |
12314 } | |
12315 | |
12316 PUGI__FN const xpath_parse_result& xpath_exception::result() const | |
12317 { | |
12318 return _result; | |
12319 } | |
12320 #endif | |
12321 | |
12322 PUGI__FN xpath_node::xpath_node() | |
12323 { | |
12324 } | |
12325 | |
12326 PUGI__FN xpath_node::xpath_node(const xml_node& node_): _node(node_) | |
12327 { | |
12328 } | |
12329 | |
12330 PUGI__FN xpath_node::xpath_node(const xml_attribute& attribute_, const xml_node& parent_): _node(attribute_ ? parent_ : xml_node()), _attribute(attribute_) | |
12331 { | |
12332 } | |
12333 | |
12334 PUGI__FN xml_node xpath_node::node() const | |
12335 { | |
12336 return _attribute ? xml_node() : _node; | |
12337 } | |
12338 | |
12339 PUGI__FN xml_attribute xpath_node::attribute() const | |
12340 { | |
12341 return _attribute; | |
12342 } | |
12343 | |
12344 PUGI__FN xml_node xpath_node::parent() const | |
12345 { | |
12346 return _attribute ? _node : _node.parent(); | |
12347 } | |
12348 | |
12349 PUGI__FN static void unspecified_bool_xpath_node(xpath_node***) | |
12350 { | |
12351 } | |
12352 | |
12353 PUGI__FN xpath_node::operator xpath_node::unspecified_bool_type() const | |
12354 { | |
12355 return (_node || _attribute) ? unspecified_bool_xpath_node : 0; | |
12356 } | |
12357 | |
12358 PUGI__FN bool xpath_node::operator!() const | |
12359 { | |
12360 return !(_node || _attribute); | |
12361 } | |
12362 | |
12363 PUGI__FN bool xpath_node::operator==(const xpath_node& n) const | |
12364 { | |
12365 return _node == n._node && _attribute == n._attribute; | |
12366 } | |
12367 | |
12368 PUGI__FN bool xpath_node::operator!=(const xpath_node& n) const | |
12369 { | |
12370 return _node != n._node || _attribute != n._attribute; | |
12371 } | |
12372 | |
12373 #ifdef __BORLANDC__ | |
12374 PUGI__FN bool operator&&(const xpath_node& lhs, bool rhs) | |
12375 { | |
12376 return (bool)lhs && rhs; | |
12377 } | |
12378 | |
12379 PUGI__FN bool operator||(const xpath_node& lhs, bool rhs) | |
12380 { | |
12381 return (bool)lhs || rhs; | |
12382 } | |
12383 #endif | |
12384 | |
12385 PUGI__FN void xpath_node_set::_assign(const_iterator begin_, const_iterator end_, type_t type_) | |
12386 { | |
12387 assert(begin_ <= end_); | |
12388 | |
12389 size_t size_ = static_cast<size_t>(end_ - begin_); | |
12390 | |
12391 // use internal buffer for 0 or 1 elements, heap buffer otherwise | |
12392 xpath_node* storage = (size_ <= 1) ? _storage : static_cast<xpath_node*>(impl::xml_memory::allocate(size_ * sizeof(xpath_node))); | |
12393 | |
12394 if (!storage) | |
12395 { | |
12396 #ifdef PUGIXML_NO_EXCEPTIONS | |
12397 return; | |
12398 #else | |
12399 throw std::bad_alloc(); | |
12400 #endif | |
12401 } | |
12402 | |
12403 // deallocate old buffer | |
12404 if (_begin != _storage) | |
12405 impl::xml_memory::deallocate(_begin); | |
12406 | |
12407 // size check is necessary because for begin_ = end_ = nullptr, memcpy is UB | |
12408 if (size_) | |
12409 memcpy(storage, begin_, size_ * sizeof(xpath_node)); | |
12410 | |
12411 _begin = storage; | |
12412 _end = storage + size_; | |
12413 _type = type_; | |
12414 } | |
12415 | |
12416 #ifdef PUGIXML_HAS_MOVE | |
12417 PUGI__FN void xpath_node_set::_move(xpath_node_set& rhs) PUGIXML_NOEXCEPT | |
12418 { | |
12419 _type = rhs._type; | |
12420 _storage[0] = rhs._storage[0]; | |
12421 _begin = (rhs._begin == rhs._storage) ? _storage : rhs._begin; | |
12422 _end = _begin + (rhs._end - rhs._begin); | |
12423 | |
12424 rhs._type = type_unsorted; | |
12425 rhs._begin = rhs._storage; | |
12426 rhs._end = rhs._storage; | |
12427 } | |
12428 #endif | |
12429 | |
12430 PUGI__FN xpath_node_set::xpath_node_set(): _type(type_unsorted), _begin(_storage), _end(_storage) | |
12431 { | |
12432 } | |
12433 | |
12434 PUGI__FN xpath_node_set::xpath_node_set(const_iterator begin_, const_iterator end_, type_t type_): _type(type_unsorted), _begin(_storage), _end(_storage) | |
12435 { | |
12436 _assign(begin_, end_, type_); | |
12437 } | |
12438 | |
12439 PUGI__FN xpath_node_set::~xpath_node_set() | |
12440 { | |
12441 if (_begin != _storage) | |
12442 impl::xml_memory::deallocate(_begin); | |
12443 } | |
12444 | |
12445 PUGI__FN xpath_node_set::xpath_node_set(const xpath_node_set& ns): _type(type_unsorted), _begin(_storage), _end(_storage) | |
12446 { | |
12447 _assign(ns._begin, ns._end, ns._type); | |
12448 } | |
12449 | |
12450 PUGI__FN xpath_node_set& xpath_node_set::operator=(const xpath_node_set& ns) | |
12451 { | |
12452 if (this == &ns) return *this; | |
12453 | |
12454 _assign(ns._begin, ns._end, ns._type); | |
12455 | |
12456 return *this; | |
12457 } | |
12458 | |
12459 #ifdef PUGIXML_HAS_MOVE | |
12460 PUGI__FN xpath_node_set::xpath_node_set(xpath_node_set&& rhs) PUGIXML_NOEXCEPT: _type(type_unsorted), _begin(_storage), _end(_storage) | |
12461 { | |
12462 _move(rhs); | |
12463 } | |
12464 | |
12465 PUGI__FN xpath_node_set& xpath_node_set::operator=(xpath_node_set&& rhs) PUGIXML_NOEXCEPT | |
12466 { | |
12467 if (this == &rhs) return *this; | |
12468 | |
12469 if (_begin != _storage) | |
12470 impl::xml_memory::deallocate(_begin); | |
12471 | |
12472 _move(rhs); | |
12473 | |
12474 return *this; | |
12475 } | |
12476 #endif | |
12477 | |
12478 PUGI__FN xpath_node_set::type_t xpath_node_set::type() const | |
12479 { | |
12480 return _type; | |
12481 } | |
12482 | |
12483 PUGI__FN size_t xpath_node_set::size() const | |
12484 { | |
12485 return _end - _begin; | |
12486 } | |
12487 | |
12488 PUGI__FN bool xpath_node_set::empty() const | |
12489 { | |
12490 return _begin == _end; | |
12491 } | |
12492 | |
12493 PUGI__FN const xpath_node& xpath_node_set::operator[](size_t index) const | |
12494 { | |
12495 assert(index < size()); | |
12496 return _begin[index]; | |
12497 } | |
12498 | |
12499 PUGI__FN xpath_node_set::const_iterator xpath_node_set::begin() const | |
12500 { | |
12501 return _begin; | |
12502 } | |
12503 | |
12504 PUGI__FN xpath_node_set::const_iterator xpath_node_set::end() const | |
12505 { | |
12506 return _end; | |
12507 } | |
12508 | |
12509 PUGI__FN void xpath_node_set::sort(bool reverse) | |
12510 { | |
12511 _type = impl::xpath_sort(_begin, _end, _type, reverse); | |
12512 } | |
12513 | |
12514 PUGI__FN xpath_node xpath_node_set::first() const | |
12515 { | |
12516 return impl::xpath_first(_begin, _end, _type); | |
12517 } | |
12518 | |
12519 PUGI__FN xpath_parse_result::xpath_parse_result(): error("Internal error"), offset(0) | |
12520 { | |
12521 } | |
12522 | |
12523 PUGI__FN xpath_parse_result::operator bool() const | |
12524 { | |
12525 return error == 0; | |
12526 } | |
12527 | |
12528 PUGI__FN const char* xpath_parse_result::description() const | |
12529 { | |
12530 return error ? error : "No error"; | |
12531 } | |
12532 | |
12533 PUGI__FN xpath_variable::xpath_variable(xpath_value_type type_): _type(type_), _next(0) | |
12534 { | |
12535 } | |
12536 | |
12537 PUGI__FN const char_t* xpath_variable::name() const | |
12538 { | |
12539 switch (_type) | |
12540 { | |
12541 case xpath_type_node_set: | |
12542 return static_cast<const impl::xpath_variable_node_set*>(this)->name; | |
12543 | |
12544 case xpath_type_number: | |
12545 return static_cast<const impl::xpath_variable_number*>(this)->name; | |
12546 | |
12547 case xpath_type_string: | |
12548 return static_cast<const impl::xpath_variable_string*>(this)->name; | |
12549 | |
12550 case xpath_type_boolean: | |
12551 return static_cast<const impl::xpath_variable_boolean*>(this)->name; | |
12552 | |
12553 default: | |
12554 assert(false && "Invalid variable type"); // unreachable | |
12555 return 0; | |
12556 } | |
12557 } | |
12558 | |
12559 PUGI__FN xpath_value_type xpath_variable::type() const | |
12560 { | |
12561 return _type; | |
12562 } | |
12563 | |
12564 PUGI__FN bool xpath_variable::get_boolean() const | |
12565 { | |
12566 return (_type == xpath_type_boolean) ? static_cast<const impl::xpath_variable_boolean*>(this)->value : false; | |
12567 } | |
12568 | |
12569 PUGI__FN double xpath_variable::get_number() const | |
12570 { | |
12571 return (_type == xpath_type_number) ? static_cast<const impl::xpath_variable_number*>(this)->value : impl::gen_nan(); | |
12572 } | |
12573 | |
12574 PUGI__FN const char_t* xpath_variable::get_string() const | |
12575 { | |
12576 const char_t* value = (_type == xpath_type_string) ? static_cast<const impl::xpath_variable_string*>(this)->value : 0; | |
12577 return value ? value : PUGIXML_TEXT(""); | |
12578 } | |
12579 | |
12580 PUGI__FN const xpath_node_set& xpath_variable::get_node_set() const | |
12581 { | |
12582 return (_type == xpath_type_node_set) ? static_cast<const impl::xpath_variable_node_set*>(this)->value : impl::dummy_node_set; | |
12583 } | |
12584 | |
12585 PUGI__FN bool xpath_variable::set(bool value) | |
12586 { | |
12587 if (_type != xpath_type_boolean) return false; | |
12588 | |
12589 static_cast<impl::xpath_variable_boolean*>(this)->value = value; | |
12590 return true; | |
12591 } | |
12592 | |
12593 PUGI__FN bool xpath_variable::set(double value) | |
12594 { | |
12595 if (_type != xpath_type_number) return false; | |
12596 | |
12597 static_cast<impl::xpath_variable_number*>(this)->value = value; | |
12598 return true; | |
12599 } | |
12600 | |
12601 PUGI__FN bool xpath_variable::set(const char_t* value) | |
12602 { | |
12603 if (_type != xpath_type_string) return false; | |
12604 | |
12605 impl::xpath_variable_string* var = static_cast<impl::xpath_variable_string*>(this); | |
12606 | |
12607 // duplicate string | |
12608 size_t size = (impl::strlength(value) + 1) * sizeof(char_t); | |
12609 | |
12610 char_t* copy = static_cast<char_t*>(impl::xml_memory::allocate(size)); | |
12611 if (!copy) return false; | |
12612 | |
12613 memcpy(copy, value, size); | |
12614 | |
12615 // replace old string | |
12616 if (var->value) impl::xml_memory::deallocate(var->value); | |
12617 var->value = copy; | |
12618 | |
12619 return true; | |
12620 } | |
12621 | |
12622 PUGI__FN bool xpath_variable::set(const xpath_node_set& value) | |
12623 { | |
12624 if (_type != xpath_type_node_set) return false; | |
12625 | |
12626 static_cast<impl::xpath_variable_node_set*>(this)->value = value; | |
12627 return true; | |
12628 } | |
12629 | |
12630 PUGI__FN xpath_variable_set::xpath_variable_set() | |
12631 { | |
12632 for (size_t i = 0; i < sizeof(_data) / sizeof(_data[0]); ++i) | |
12633 _data[i] = 0; | |
12634 } | |
12635 | |
12636 PUGI__FN xpath_variable_set::~xpath_variable_set() | |
12637 { | |
12638 for (size_t i = 0; i < sizeof(_data) / sizeof(_data[0]); ++i) | |
12639 _destroy(_data[i]); | |
12640 } | |
12641 | |
12642 PUGI__FN xpath_variable_set::xpath_variable_set(const xpath_variable_set& rhs) | |
12643 { | |
12644 for (size_t i = 0; i < sizeof(_data) / sizeof(_data[0]); ++i) | |
12645 _data[i] = 0; | |
12646 | |
12647 _assign(rhs); | |
12648 } | |
12649 | |
12650 PUGI__FN xpath_variable_set& xpath_variable_set::operator=(const xpath_variable_set& rhs) | |
12651 { | |
12652 if (this == &rhs) return *this; | |
12653 | |
12654 _assign(rhs); | |
12655 | |
12656 return *this; | |
12657 } | |
12658 | |
12659 #ifdef PUGIXML_HAS_MOVE | |
12660 PUGI__FN xpath_variable_set::xpath_variable_set(xpath_variable_set&& rhs) PUGIXML_NOEXCEPT | |
12661 { | |
12662 for (size_t i = 0; i < sizeof(_data) / sizeof(_data[0]); ++i) | |
12663 { | |
12664 _data[i] = rhs._data[i]; | |
12665 rhs._data[i] = 0; | |
12666 } | |
12667 } | |
12668 | |
12669 PUGI__FN xpath_variable_set& xpath_variable_set::operator=(xpath_variable_set&& rhs) PUGIXML_NOEXCEPT | |
12670 { | |
12671 for (size_t i = 0; i < sizeof(_data) / sizeof(_data[0]); ++i) | |
12672 { | |
12673 _destroy(_data[i]); | |
12674 | |
12675 _data[i] = rhs._data[i]; | |
12676 rhs._data[i] = 0; | |
12677 } | |
12678 | |
12679 return *this; | |
12680 } | |
12681 #endif | |
12682 | |
12683 PUGI__FN void xpath_variable_set::_assign(const xpath_variable_set& rhs) | |
12684 { | |
12685 xpath_variable_set temp; | |
12686 | |
12687 for (size_t i = 0; i < sizeof(_data) / sizeof(_data[0]); ++i) | |
12688 if (rhs._data[i] && !_clone(rhs._data[i], &temp._data[i])) | |
12689 return; | |
12690 | |
12691 _swap(temp); | |
12692 } | |
12693 | |
12694 PUGI__FN void xpath_variable_set::_swap(xpath_variable_set& rhs) | |
12695 { | |
12696 for (size_t i = 0; i < sizeof(_data) / sizeof(_data[0]); ++i) | |
12697 { | |
12698 xpath_variable* chain = _data[i]; | |
12699 | |
12700 _data[i] = rhs._data[i]; | |
12701 rhs._data[i] = chain; | |
12702 } | |
12703 } | |
12704 | |
12705 PUGI__FN xpath_variable* xpath_variable_set::_find(const char_t* name) const | |
12706 { | |
12707 const size_t hash_size = sizeof(_data) / sizeof(_data[0]); | |
12708 size_t hash = impl::hash_string(name) % hash_size; | |
12709 | |
12710 // look for existing variable | |
12711 for (xpath_variable* var = _data[hash]; var; var = var->_next) | |
12712 if (impl::strequal(var->name(), name)) | |
12713 return var; | |
12714 | |
12715 return 0; | |
12716 } | |
12717 | |
12718 PUGI__FN bool xpath_variable_set::_clone(xpath_variable* var, xpath_variable** out_result) | |
12719 { | |
12720 xpath_variable* last = 0; | |
12721 | |
12722 while (var) | |
12723 { | |
12724 // allocate storage for new variable | |
12725 xpath_variable* nvar = impl::new_xpath_variable(var->_type, var->name()); | |
12726 if (!nvar) return false; | |
12727 | |
12728 // link the variable to the result immediately to handle failures gracefully | |
12729 if (last) | |
12730 last->_next = nvar; | |
12731 else | |
12732 *out_result = nvar; | |
12733 | |
12734 last = nvar; | |
12735 | |
12736 // copy the value; this can fail due to out-of-memory conditions | |
12737 if (!impl::copy_xpath_variable(nvar, var)) return false; | |
12738 | |
12739 var = var->_next; | |
12740 } | |
12741 | |
12742 return true; | |
12743 } | |
12744 | |
12745 PUGI__FN void xpath_variable_set::_destroy(xpath_variable* var) | |
12746 { | |
12747 while (var) | |
12748 { | |
12749 xpath_variable* next = var->_next; | |
12750 | |
12751 impl::delete_xpath_variable(var->_type, var); | |
12752 | |
12753 var = next; | |
12754 } | |
12755 } | |
12756 | |
12757 PUGI__FN xpath_variable* xpath_variable_set::add(const char_t* name, xpath_value_type type) | |
12758 { | |
12759 const size_t hash_size = sizeof(_data) / sizeof(_data[0]); | |
12760 size_t hash = impl::hash_string(name) % hash_size; | |
12761 | |
12762 // look for existing variable | |
12763 for (xpath_variable* var = _data[hash]; var; var = var->_next) | |
12764 if (impl::strequal(var->name(), name)) | |
12765 return var->type() == type ? var : 0; | |
12766 | |
12767 // add new variable | |
12768 xpath_variable* result = impl::new_xpath_variable(type, name); | |
12769 | |
12770 if (result) | |
12771 { | |
12772 result->_next = _data[hash]; | |
12773 | |
12774 _data[hash] = result; | |
12775 } | |
12776 | |
12777 return result; | |
12778 } | |
12779 | |
12780 PUGI__FN bool xpath_variable_set::set(const char_t* name, bool value) | |
12781 { | |
12782 xpath_variable* var = add(name, xpath_type_boolean); | |
12783 return var ? var->set(value) : false; | |
12784 } | |
12785 | |
12786 PUGI__FN bool xpath_variable_set::set(const char_t* name, double value) | |
12787 { | |
12788 xpath_variable* var = add(name, xpath_type_number); | |
12789 return var ? var->set(value) : false; | |
12790 } | |
12791 | |
12792 PUGI__FN bool xpath_variable_set::set(const char_t* name, const char_t* value) | |
12793 { | |
12794 xpath_variable* var = add(name, xpath_type_string); | |
12795 return var ? var->set(value) : false; | |
12796 } | |
12797 | |
12798 PUGI__FN bool xpath_variable_set::set(const char_t* name, const xpath_node_set& value) | |
12799 { | |
12800 xpath_variable* var = add(name, xpath_type_node_set); | |
12801 return var ? var->set(value) : false; | |
12802 } | |
12803 | |
12804 PUGI__FN xpath_variable* xpath_variable_set::get(const char_t* name) | |
12805 { | |
12806 return _find(name); | |
12807 } | |
12808 | |
12809 PUGI__FN const xpath_variable* xpath_variable_set::get(const char_t* name) const | |
12810 { | |
12811 return _find(name); | |
12812 } | |
12813 | |
12814 PUGI__FN xpath_query::xpath_query(const char_t* query, xpath_variable_set* variables): _impl(0) | |
12815 { | |
12816 impl::xpath_query_impl* qimpl = impl::xpath_query_impl::create(); | |
12817 | |
12818 if (!qimpl) | |
12819 { | |
12820 #ifdef PUGIXML_NO_EXCEPTIONS | |
12821 _result.error = "Out of memory"; | |
12822 #else | |
12823 throw std::bad_alloc(); | |
12824 #endif | |
12825 } | |
12826 else | |
12827 { | |
12828 using impl::auto_deleter; // MSVC7 workaround | |
12829 auto_deleter<impl::xpath_query_impl> impl(qimpl, impl::xpath_query_impl::destroy); | |
12830 | |
12831 qimpl->root = impl::xpath_parser::parse(query, variables, &qimpl->alloc, &_result); | |
12832 | |
12833 if (qimpl->root) | |
12834 { | |
12835 qimpl->root->optimize(&qimpl->alloc); | |
12836 | |
12837 _impl = impl.release(); | |
12838 _result.error = 0; | |
12839 } | |
12840 else | |
12841 { | |
12842 #ifdef PUGIXML_NO_EXCEPTIONS | |
12843 if (qimpl->oom) _result.error = "Out of memory"; | |
12844 #else | |
12845 if (qimpl->oom) throw std::bad_alloc(); | |
12846 throw xpath_exception(_result); | |
12847 #endif | |
12848 } | |
12849 } | |
12850 } | |
12851 | |
12852 PUGI__FN xpath_query::xpath_query(): _impl(0) | |
12853 { | |
12854 } | |
12855 | |
12856 PUGI__FN xpath_query::~xpath_query() | |
12857 { | |
12858 if (_impl) | |
12859 impl::xpath_query_impl::destroy(static_cast<impl::xpath_query_impl*>(_impl)); | |
12860 } | |
12861 | |
12862 #ifdef PUGIXML_HAS_MOVE | |
12863 PUGI__FN xpath_query::xpath_query(xpath_query&& rhs) PUGIXML_NOEXCEPT | |
12864 { | |
12865 _impl = rhs._impl; | |
12866 _result = rhs._result; | |
12867 rhs._impl = 0; | |
12868 rhs._result = xpath_parse_result(); | |
12869 } | |
12870 | |
12871 PUGI__FN xpath_query& xpath_query::operator=(xpath_query&& rhs) PUGIXML_NOEXCEPT | |
12872 { | |
12873 if (this == &rhs) return *this; | |
12874 | |
12875 if (_impl) | |
12876 impl::xpath_query_impl::destroy(static_cast<impl::xpath_query_impl*>(_impl)); | |
12877 | |
12878 _impl = rhs._impl; | |
12879 _result = rhs._result; | |
12880 rhs._impl = 0; | |
12881 rhs._result = xpath_parse_result(); | |
12882 | |
12883 return *this; | |
12884 } | |
12885 #endif | |
12886 | |
12887 PUGI__FN xpath_value_type xpath_query::return_type() const | |
12888 { | |
12889 if (!_impl) return xpath_type_none; | |
12890 | |
12891 return static_cast<impl::xpath_query_impl*>(_impl)->root->rettype(); | |
12892 } | |
12893 | |
12894 PUGI__FN bool xpath_query::evaluate_boolean(const xpath_node& n) const | |
12895 { | |
12896 if (!_impl) return false; | |
12897 | |
12898 impl::xpath_context c(n, 1, 1); | |
12899 impl::xpath_stack_data sd; | |
12900 | |
12901 bool r = static_cast<impl::xpath_query_impl*>(_impl)->root->eval_boolean(c, sd.stack); | |
12902 | |
12903 if (sd.oom) | |
12904 { | |
12905 #ifdef PUGIXML_NO_EXCEPTIONS | |
12906 return false; | |
12907 #else | |
12908 throw std::bad_alloc(); | |
12909 #endif | |
12910 } | |
12911 | |
12912 return r; | |
12913 } | |
12914 | |
12915 PUGI__FN double xpath_query::evaluate_number(const xpath_node& n) const | |
12916 { | |
12917 if (!_impl) return impl::gen_nan(); | |
12918 | |
12919 impl::xpath_context c(n, 1, 1); | |
12920 impl::xpath_stack_data sd; | |
12921 | |
12922 double r = static_cast<impl::xpath_query_impl*>(_impl)->root->eval_number(c, sd.stack); | |
12923 | |
12924 if (sd.oom) | |
12925 { | |
12926 #ifdef PUGIXML_NO_EXCEPTIONS | |
12927 return impl::gen_nan(); | |
12928 #else | |
12929 throw std::bad_alloc(); | |
12930 #endif | |
12931 } | |
12932 | |
12933 return r; | |
12934 } | |
12935 | |
12936 #ifndef PUGIXML_NO_STL | |
12937 PUGI__FN string_t xpath_query::evaluate_string(const xpath_node& n) const | |
12938 { | |
12939 if (!_impl) return string_t(); | |
12940 | |
12941 impl::xpath_context c(n, 1, 1); | |
12942 impl::xpath_stack_data sd; | |
12943 | |
12944 impl::xpath_string r = static_cast<impl::xpath_query_impl*>(_impl)->root->eval_string(c, sd.stack); | |
12945 | |
12946 if (sd.oom) | |
12947 { | |
12948 #ifdef PUGIXML_NO_EXCEPTIONS | |
12949 return string_t(); | |
12950 #else | |
12951 throw std::bad_alloc(); | |
12952 #endif | |
12953 } | |
12954 | |
12955 return string_t(r.c_str(), r.length()); | |
12956 } | |
12957 #endif | |
12958 | |
12959 PUGI__FN size_t xpath_query::evaluate_string(char_t* buffer, size_t capacity, const xpath_node& n) const | |
12960 { | |
12961 impl::xpath_context c(n, 1, 1); | |
12962 impl::xpath_stack_data sd; | |
12963 | |
12964 impl::xpath_string r = _impl ? static_cast<impl::xpath_query_impl*>(_impl)->root->eval_string(c, sd.stack) : impl::xpath_string(); | |
12965 | |
12966 if (sd.oom) | |
12967 { | |
12968 #ifdef PUGIXML_NO_EXCEPTIONS | |
12969 r = impl::xpath_string(); | |
12970 #else | |
12971 throw std::bad_alloc(); | |
12972 #endif | |
12973 } | |
12974 | |
12975 size_t full_size = r.length() + 1; | |
12976 | |
12977 if (capacity > 0) | |
12978 { | |
12979 size_t size = (full_size < capacity) ? full_size : capacity; | |
12980 assert(size > 0); | |
12981 | |
12982 memcpy(buffer, r.c_str(), (size - 1) * sizeof(char_t)); | |
12983 buffer[size - 1] = 0; | |
12984 } | |
12985 | |
12986 return full_size; | |
12987 } | |
12988 | |
12989 PUGI__FN xpath_node_set xpath_query::evaluate_node_set(const xpath_node& n) const | |
12990 { | |
12991 impl::xpath_ast_node* root = impl::evaluate_node_set_prepare(static_cast<impl::xpath_query_impl*>(_impl)); | |
12992 if (!root) return xpath_node_set(); | |
12993 | |
12994 impl::xpath_context c(n, 1, 1); | |
12995 impl::xpath_stack_data sd; | |
12996 | |
12997 impl::xpath_node_set_raw r = root->eval_node_set(c, sd.stack, impl::nodeset_eval_all); | |
12998 | |
12999 if (sd.oom) | |
13000 { | |
13001 #ifdef PUGIXML_NO_EXCEPTIONS | |
13002 return xpath_node_set(); | |
13003 #else | |
13004 throw std::bad_alloc(); | |
13005 #endif | |
13006 } | |
13007 | |
13008 return xpath_node_set(r.begin(), r.end(), r.type()); | |
13009 } | |
13010 | |
13011 PUGI__FN xpath_node xpath_query::evaluate_node(const xpath_node& n) const | |
13012 { | |
13013 impl::xpath_ast_node* root = impl::evaluate_node_set_prepare(static_cast<impl::xpath_query_impl*>(_impl)); | |
13014 if (!root) return xpath_node(); | |
13015 | |
13016 impl::xpath_context c(n, 1, 1); | |
13017 impl::xpath_stack_data sd; | |
13018 | |
13019 impl::xpath_node_set_raw r = root->eval_node_set(c, sd.stack, impl::nodeset_eval_first); | |
13020 | |
13021 if (sd.oom) | |
13022 { | |
13023 #ifdef PUGIXML_NO_EXCEPTIONS | |
13024 return xpath_node(); | |
13025 #else | |
13026 throw std::bad_alloc(); | |
13027 #endif | |
13028 } | |
13029 | |
13030 return r.first(); | |
13031 } | |
13032 | |
13033 PUGI__FN const xpath_parse_result& xpath_query::result() const | |
13034 { | |
13035 return _result; | |
13036 } | |
13037 | |
13038 PUGI__FN static void unspecified_bool_xpath_query(xpath_query***) | |
13039 { | |
13040 } | |
13041 | |
13042 PUGI__FN xpath_query::operator xpath_query::unspecified_bool_type() const | |
13043 { | |
13044 return _impl ? unspecified_bool_xpath_query : 0; | |
13045 } | |
13046 | |
13047 PUGI__FN bool xpath_query::operator!() const | |
13048 { | |
13049 return !_impl; | |
13050 } | |
13051 | |
13052 PUGI__FN xpath_node xml_node::select_node(const char_t* query, xpath_variable_set* variables) const | |
13053 { | |
13054 xpath_query q(query, variables); | |
13055 return q.evaluate_node(*this); | |
13056 } | |
13057 | |
13058 PUGI__FN xpath_node xml_node::select_node(const xpath_query& query) const | |
13059 { | |
13060 return query.evaluate_node(*this); | |
13061 } | |
13062 | |
13063 PUGI__FN xpath_node_set xml_node::select_nodes(const char_t* query, xpath_variable_set* variables) const | |
13064 { | |
13065 xpath_query q(query, variables); | |
13066 return q.evaluate_node_set(*this); | |
13067 } | |
13068 | |
13069 PUGI__FN xpath_node_set xml_node::select_nodes(const xpath_query& query) const | |
13070 { | |
13071 return query.evaluate_node_set(*this); | |
13072 } | |
13073 | |
13074 PUGI__FN xpath_node xml_node::select_single_node(const char_t* query, xpath_variable_set* variables) const | |
13075 { | |
13076 xpath_query q(query, variables); | |
13077 return q.evaluate_node(*this); | |
13078 } | |
13079 | |
13080 PUGI__FN xpath_node xml_node::select_single_node(const xpath_query& query) const | |
13081 { | |
13082 return query.evaluate_node(*this); | |
13083 } | |
13084 } | |
13085 | |
13086 #endif | |
13087 | |
13088 #ifdef __BORLANDC__ | |
13089 # pragma option pop | |
13090 #endif | |
13091 | |
13092 // Intel C++ does not properly keep warning state for function templates, | |
13093 // so popping warning state at the end of translation unit leads to warnings in the middle. | |
13094 #if defined(_MSC_VER) && !defined(__INTEL_COMPILER) | |
13095 # pragma warning(pop) | |
13096 #endif | |
13097 | |
13098 #if defined(_MSC_VER) && defined(__c2__) | |
13099 # pragma clang diagnostic pop | |
13100 #endif | |
13101 | |
13102 // Undefine all local macros (makes sure we're not leaking macros in header-only mode) | |
13103 #undef PUGI__NO_INLINE | |
13104 #undef PUGI__UNLIKELY | |
13105 #undef PUGI__STATIC_ASSERT | |
13106 #undef PUGI__DMC_VOLATILE | |
13107 #undef PUGI__UNSIGNED_OVERFLOW | |
13108 #undef PUGI__MSVC_CRT_VERSION | |
13109 #undef PUGI__SNPRINTF | |
13110 #undef PUGI__NS_BEGIN | |
13111 #undef PUGI__NS_END | |
13112 #undef PUGI__FN | |
13113 #undef PUGI__FN_NO_INLINE | |
13114 #undef PUGI__GETHEADER_IMPL | |
13115 #undef PUGI__GETPAGE_IMPL | |
13116 #undef PUGI__GETPAGE | |
13117 #undef PUGI__NODETYPE | |
13118 #undef PUGI__IS_CHARTYPE_IMPL | |
13119 #undef PUGI__IS_CHARTYPE | |
13120 #undef PUGI__IS_CHARTYPEX | |
13121 #undef PUGI__ENDSWITH | |
13122 #undef PUGI__SKIPWS | |
13123 #undef PUGI__OPTSET | |
13124 #undef PUGI__PUSHNODE | |
13125 #undef PUGI__POPNODE | |
13126 #undef PUGI__SCANFOR | |
13127 #undef PUGI__SCANWHILE | |
13128 #undef PUGI__SCANWHILE_UNROLL | |
13129 #undef PUGI__ENDSEG | |
13130 #undef PUGI__THROW_ERROR | |
13131 #undef PUGI__CHECK_ERROR | |
13132 | |
13133 #endif | |
13134 | |
13135 /** | |
13136 * Copyright (c) 2006-2022 Arseny Kapoulkine | |
13137 * | |
13138 * Permission is hereby granted, free of charge, to any person | |
13139 * obtaining a copy of this software and associated documentation | |
13140 * files (the "Software"), to deal in the Software without | |
13141 * restriction, including without limitation the rights to use, | |
13142 * copy, modify, merge, publish, distribute, sublicense, and/or sell | |
13143 * copies of the Software, and to permit persons to whom the | |
13144 * Software is furnished to do so, subject to the following | |
13145 * conditions: | |
13146 * | |
13147 * The above copyright notice and this permission notice shall be | |
13148 * included in all copies or substantial portions of the Software. | |
13149 * | |
13150 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | |
13151 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES | |
13152 * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | |
13153 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT | |
13154 * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | |
13155 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | |
13156 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR | |
13157 * OTHER DEALINGS IN THE SOFTWARE. | |
13158 */ |