Coverage Report

Created: 2025-07-11 06:36

/src/libxml2/xmlreader.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * xmlreader.c: implements the xmlTextReader streaming node API
3
 *
4
 * NOTE:
5
 *   XmlTextReader.Normalization Property won't be supported, since
6
 *     it makes the parser non compliant to the XML recommendation
7
 *
8
 * See Copyright for the status of this software.
9
 *
10
 * Author: Daniel Veillard
11
 */
12
13
/*
14
 * TODOs:
15
 *   - XML Schemas validation
16
 */
17
#define IN_LIBXML
18
#include "libxml.h"
19
20
#ifdef LIBXML_READER_ENABLED
21
#include <string.h> /* for memset() only ! */
22
#include <stdarg.h>
23
#include <ctype.h>
24
#include <stdlib.h>
25
26
#include <libxml/xmlmemory.h>
27
#include <libxml/xmlIO.h>
28
#include <libxml/xmlreader.h>
29
#include <libxml/parserInternals.h>
30
#ifdef LIBXML_RELAXNG_ENABLED
31
#include <libxml/relaxng.h>
32
#endif
33
#ifdef LIBXML_SCHEMAS_ENABLED
34
#include <libxml/xmlschemas.h>
35
#endif
36
#include <libxml/uri.h>
37
#ifdef LIBXML_XINCLUDE_ENABLED
38
#include <libxml/xinclude.h>
39
#endif
40
#ifdef LIBXML_PATTERN_ENABLED
41
#include <libxml/pattern.h>
42
#endif
43
44
#include "private/buf.h"
45
#include "private/error.h"
46
#include "private/io.h"
47
#include "private/memory.h"
48
#include "private/parser.h"
49
#include "private/tree.h"
50
#ifdef LIBXML_XINCLUDE_ENABLED
51
#include "private/xinclude.h"
52
#endif
53
54
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
55
/* Keeping free objects can hide memory errors. */
56
3.42M
#define MAX_FREE_NODES 1
57
#else
58
#define MAX_FREE_NODES 100
59
#endif
60
61
#ifndef va_copy
62
  #ifdef __va_copy
63
    #define va_copy(dest, src) __va_copy(dest, src)
64
  #else
65
    #define va_copy(dest, src) memcpy(&(dest), &(src), sizeof(va_list))
66
  #endif
67
#endif
68
69
1.38M
#define CHUNK_SIZE 512
70
/************************************************************************
71
 *                  *
72
 *  The parser: maps the Text Reader API on top of the existing *
73
 *    parsing routines building a tree      *
74
 *                  *
75
 ************************************************************************/
76
77
111k
#define XML_TEXTREADER_INPUT  1
78
75.3k
#define XML_TEXTREADER_CTXT 2
79
80
typedef enum {
81
    XML_TEXTREADER_NONE = -1,
82
    XML_TEXTREADER_START= 0,
83
    XML_TEXTREADER_ELEMENT= 1,
84
    XML_TEXTREADER_END= 2,
85
    XML_TEXTREADER_EMPTY= 3,
86
    XML_TEXTREADER_BACKTRACK= 4,
87
    XML_TEXTREADER_DONE= 5,
88
    XML_TEXTREADER_ERROR= 6
89
} xmlTextReaderState;
90
91
typedef enum {
92
    XML_TEXTREADER_NOT_VALIDATE = 0,
93
    XML_TEXTREADER_VALIDATE_DTD = 1,
94
    XML_TEXTREADER_VALIDATE_RNG = 2,
95
    XML_TEXTREADER_VALIDATE_XSD = 4
96
} xmlTextReaderValidate;
97
98
struct _xmlTextReader {
99
    int       mode; /* the parsing mode */
100
    xmlDocPtr     doc;    /* when walking an existing doc */
101
    xmlTextReaderValidate       validate;/* is there any validation */
102
    int       allocs; /* what structure were deallocated */
103
    xmlTextReaderState    state;
104
    xmlParserCtxtPtr    ctxt; /* the parser context */
105
    xmlSAXHandlerPtr    sax;  /* the parser SAX callbacks */
106
    xmlParserInputBufferPtr input;  /* the input */
107
    startElementSAXFunc   startElement;/* initial SAX callbacks */
108
    endElementSAXFunc   endElement;  /* idem */
109
    startElementNsSAX2Func  startElementNs;/* idem */
110
    endElementNsSAX2Func  endElementNs;  /* idem */
111
    charactersSAXFunc   characters;
112
    cdataBlockSAXFunc   cdataBlock;
113
    unsigned int    base; /* base of the segment in the input */
114
    unsigned int    cur;  /* current position in the input */
115
    xmlNodePtr      node; /* current node */
116
    xmlNodePtr      curnode;/* current attribute node */
117
    int       depth;  /* depth of the current node */
118
    xmlNodePtr      faketext;/* fake xmlNs chld */
119
    int       preserve;/* preserve the resulting document */
120
    xmlBufPtr           buffer; /* used to return const xmlChar * */
121
    xmlDictPtr      dict; /* the context dictionary */
122
123
    /* entity stack when traversing entities content */
124
    xmlNodePtr         ent;          /* Current Entity Ref Node */
125
    int                entNr;        /* Depth of the entities stack */
126
    int                entMax;       /* Max depth of the entities stack */
127
    xmlNodePtr        *entTab;       /* array of entities */
128
129
    /* error handling */
130
    xmlTextReaderErrorFunc errorFunc;    /* callback function */
131
    void                  *errorFuncArg; /* callback function user argument */
132
133
#ifdef LIBXML_RELAXNG_ENABLED
134
    /* Handling of RelaxNG validation */
135
    xmlRelaxNGPtr          rngSchemas;  /* The Relax NG schemas */
136
    xmlRelaxNGValidCtxtPtr rngValidCtxt;/* The Relax NG validation context */
137
    int                    rngPreserveCtxt; /* 1 if the context was provided by the user */
138
    int                    rngValidErrors;/* The number of errors detected */
139
    xmlNodePtr             rngFullNode; /* the node if RNG not progressive */
140
#endif
141
#ifdef LIBXML_SCHEMAS_ENABLED
142
    /* Handling of Schemas validation */
143
    xmlSchemaPtr          xsdSchemas; /* The Schemas schemas */
144
    xmlSchemaValidCtxtPtr xsdValidCtxt;/* The Schemas validation context */
145
    int                   xsdPreserveCtxt; /* 1 if the context was provided by the user */
146
    int                   xsdValidErrors;/* The number of errors detected */
147
    xmlSchemaSAXPlugPtr   xsdPlug;  /* the schemas plug in SAX pipeline */
148
#endif
149
#ifdef LIBXML_XINCLUDE_ENABLED
150
    /* Handling of XInclude processing */
151
    int                xinclude;  /* is xinclude asked for */
152
    xmlXIncludeCtxtPtr xincctxt;  /* the xinclude context */
153
    int                in_xinclude; /* counts for xinclude */
154
#endif
155
#ifdef LIBXML_PATTERN_ENABLED
156
    int                patternNr;       /* number of preserve patterns */
157
    int                patternMax;      /* max preserve patterns */
158
    xmlPatternPtr     *patternTab;      /* array of preserve patterns */
159
#endif
160
    int                preserves; /* level of preserves */
161
    int                parserFlags; /* the set of options set */
162
    /* Structured error handling */
163
    xmlStructuredErrorFunc sErrorFunc;  /* callback function */
164
165
    xmlResourceLoader resourceLoader;
166
    void *resourceCtxt;
167
};
168
169
151k
#define NODE_IS_EMPTY   0x1
170
800k
#define NODE_IS_PRESERVED 0x2
171
24.4k
#define NODE_IS_SPRESERVED  0x4
172
173
static int xmlTextReaderReadTree(xmlTextReaderPtr reader);
174
static int xmlTextReaderNextTree(xmlTextReaderPtr reader);
175
176
/**
177
 * Free a string if it is not owned by the "dict" dictionary in the
178
 * current scope
179
 *
180
 * @param str  a string
181
 */
182
#define DICT_FREE(str)            \
183
4.24M
  if ((str) && ((!dict) ||       \
184
4.19M
      (xmlDictOwns(dict, (const xmlChar *)(str)) == 0))) \
185
3.13M
      xmlFree((char *)(str));
186
187
static void xmlTextReaderFreeNode(xmlTextReaderPtr reader, xmlNodePtr cur);
188
static void xmlTextReaderFreeNodeList(xmlTextReaderPtr reader, xmlNodePtr cur);
189
190
static void
191
3.05k
xmlTextReaderErr(xmlParserErrors code, const char *msg, ...) {
192
3.05k
    va_list ap;
193
3.05k
    int res;
194
195
3.05k
    va_start(ap, msg);
196
3.05k
    res = xmlVRaiseError(NULL, NULL, NULL, NULL, NULL,
197
3.05k
                         XML_FROM_PARSER, code, XML_ERR_FATAL,
198
3.05k
                         NULL, 0, NULL, NULL, NULL, 0, 0,
199
3.05k
                         msg, ap);
200
3.05k
    va_end(ap);
201
3.05k
    if (res < 0)
202
78
        xmlRaiseMemoryError(NULL, NULL, NULL, XML_FROM_PARSER, NULL);
203
3.05k
}
204
205
static void
206
3.66k
xmlTextReaderErrMemory(xmlTextReaderPtr reader) {
207
3.66k
    if (reader == NULL) {
208
14
        xmlRaiseMemoryError(NULL, NULL, NULL, XML_FROM_PARSER, NULL);
209
14
        return;
210
14
    }
211
212
3.65k
    if (reader->ctxt != NULL)
213
3.65k
        xmlCtxtErrMemory(reader->ctxt);
214
0
    else
215
0
        xmlRaiseMemoryError(NULL, NULL, NULL, XML_FROM_PARSER, NULL);
216
217
3.65k
    reader->mode = XML_TEXTREADER_MODE_ERROR;
218
3.65k
    reader->state = XML_TEXTREADER_ERROR;
219
3.65k
}
220
221
static xmlChar *
222
13.7k
readerStrdup(xmlTextReaderPtr reader, const xmlChar *string) {
223
13.7k
    xmlChar *copy;
224
225
13.7k
    if (string == NULL)
226
1.25k
        return(NULL);
227
228
12.4k
    copy = xmlStrdup(string);
229
12.4k
    if (copy == NULL)
230
47
        xmlTextReaderErrMemory(reader);
231
232
12.4k
    return(copy);
233
13.7k
}
234
235
static const xmlChar *
236
65.9k
constString(xmlTextReaderPtr reader, const xmlChar *string) {
237
65.9k
    const xmlChar *dictString;
238
239
65.9k
    if (string == NULL)
240
2.38k
        return(NULL);
241
242
63.5k
    dictString = xmlDictLookup(reader->dict, string, -1);
243
63.5k
    if (dictString == NULL)
244
15
        xmlTextReaderErrMemory(reader);
245
246
63.5k
    return(dictString);
247
65.9k
}
248
249
static const xmlChar *
250
constQString(xmlTextReaderPtr reader, const xmlChar *prefix,
251
3.42k
             const xmlChar *name) {
252
3.42k
    const xmlChar *dictString;
253
254
3.42k
    if (name == NULL)
255
0
        return(NULL);
256
257
3.42k
    dictString = xmlDictQLookup(reader->dict, prefix, name);
258
3.42k
    if (dictString == NULL)
259
4
        xmlTextReaderErrMemory(reader);
260
261
3.42k
    return(dictString);
262
3.42k
}
263
264
/************************************************************************
265
 *                  *
266
 *  Our own version of the freeing routines as we recycle nodes *
267
 *                  *
268
 ************************************************************************/
269
270
/**
271
 * Free a node.
272
 *
273
 * @param reader  the xmlTextReader used
274
 * @param cur  the node
275
 */
276
static void
277
270k
xmlTextReaderFreeProp(xmlTextReaderPtr reader, xmlAttrPtr cur) {
278
270k
    xmlDictPtr dict;
279
280
270k
    if ((reader != NULL) && (reader->ctxt != NULL))
281
270k
  dict = reader->ctxt->dict;
282
0
    else
283
0
        dict = NULL;
284
270k
    if (cur == NULL) return;
285
286
270k
    if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
287
0
  xmlDeregisterNodeDefaultValue((xmlNodePtr) cur);
288
289
270k
    if (cur->children != NULL)
290
265k
        xmlTextReaderFreeNodeList(reader, cur->children);
291
292
270k
    if (cur->id != NULL) {
293
        /*
294
         * Operating in streaming mode, attr is gonna disappear
295
         */
296
171
        cur->id->attr = NULL;
297
171
        if (cur->id->name != NULL)
298
0
            DICT_FREE(cur->id->name);
299
171
        cur->id->name = cur->name;
300
171
        cur->name = NULL;
301
270k
    } else {
302
270k
        DICT_FREE(cur->name);
303
270k
    }
304
305
270k
    if ((reader != NULL) && (reader->ctxt != NULL) &&
306
270k
        (reader->ctxt->freeAttrsNr < MAX_FREE_NODES)) {
307
15.4k
        cur->next = reader->ctxt->freeAttrs;
308
15.4k
  reader->ctxt->freeAttrs = cur;
309
15.4k
  reader->ctxt->freeAttrsNr++;
310
254k
    } else {
311
254k
  xmlFree(cur);
312
254k
    }
313
270k
}
314
315
/**
316
 * Free a property and all its siblings, all the children are freed too.
317
 *
318
 * @param reader  the xmlTextReader used
319
 * @param cur  the first property in the list
320
 */
321
static void
322
110k
xmlTextReaderFreePropList(xmlTextReaderPtr reader, xmlAttrPtr cur) {
323
110k
    xmlAttrPtr next;
324
325
380k
    while (cur != NULL) {
326
270k
        next = cur->next;
327
270k
        xmlTextReaderFreeProp(reader, cur);
328
270k
  cur = next;
329
270k
    }
330
110k
}
331
332
/**
333
 * Free a node and all its siblings, this is a recursive behaviour, all
334
 * the children are freed too.
335
 *
336
 * @param reader  the xmlTextReader used
337
 * @param cur  the first node in the list
338
 */
339
static void
340
288k
xmlTextReaderFreeNodeList(xmlTextReaderPtr reader, xmlNodePtr cur) {
341
288k
    xmlNodePtr next;
342
288k
    xmlNodePtr parent;
343
288k
    xmlDictPtr dict;
344
288k
    size_t depth = 0;
345
346
288k
    if ((reader != NULL) && (reader->ctxt != NULL))
347
288k
  dict = reader->ctxt->dict;
348
0
    else
349
0
        dict = NULL;
350
288k
    if (cur == NULL) return;
351
288k
    if (cur->type == XML_NAMESPACE_DECL) {
352
0
  xmlFreeNsList((xmlNsPtr) cur);
353
0
  return;
354
0
    }
355
288k
    if ((cur->type == XML_DOCUMENT_NODE) ||
356
288k
  (cur->type == XML_HTML_DOCUMENT_NODE)) {
357
0
  xmlFreeDoc((xmlDocPtr) cur);
358
0
  return;
359
0
    }
360
3.78M
    while (1) {
361
5.48M
        while ((cur->type != XML_DTD_NODE) &&
362
5.48M
               (cur->type != XML_ENTITY_REF_NODE) &&
363
5.48M
               (cur->children != NULL) &&
364
5.48M
               (cur->children->parent == cur)) {
365
1.70M
            cur = cur->children;
366
1.70M
            depth += 1;
367
1.70M
        }
368
369
3.78M
        next = cur->next;
370
3.78M
        parent = cur->parent;
371
372
  /* unroll to speed up freeing the document */
373
3.78M
  if (cur->type != XML_DTD_NODE) {
374
375
3.78M
      if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
376
0
    xmlDeregisterNodeDefaultValue(cur);
377
378
3.78M
      if (((cur->type == XML_ELEMENT_NODE) ||
379
3.78M
     (cur->type == XML_XINCLUDE_START) ||
380
3.78M
     (cur->type == XML_XINCLUDE_END)) &&
381
3.78M
    (cur->properties != NULL))
382
89.0k
    xmlTextReaderFreePropList(reader, cur->properties);
383
3.78M
      if ((cur->content != (xmlChar *) &(cur->properties)) &&
384
3.78M
          (cur->type != XML_ELEMENT_NODE) &&
385
3.78M
    (cur->type != XML_XINCLUDE_START) &&
386
3.78M
    (cur->type != XML_XINCLUDE_END) &&
387
3.78M
    (cur->type != XML_ENTITY_REF_NODE)) {
388
1.09M
    DICT_FREE(cur->content);
389
1.09M
      }
390
3.78M
      if (((cur->type == XML_ELEMENT_NODE) ||
391
3.78M
           (cur->type == XML_XINCLUDE_START) ||
392
3.78M
     (cur->type == XML_XINCLUDE_END)) &&
393
3.78M
    (cur->nsDef != NULL))
394
777k
    xmlFreeNsList(cur->nsDef);
395
396
      /*
397
       * we don't free element names here they are interned now
398
       */
399
3.78M
      if ((cur->type != XML_TEXT_NODE) &&
400
3.78M
    (cur->type != XML_COMMENT_NODE))
401
2.61M
    DICT_FREE(cur->name);
402
3.78M
      if (((cur->type == XML_ELEMENT_NODE) ||
403
3.78M
     (cur->type == XML_TEXT_NODE)) &&
404
3.78M
          (reader != NULL) && (reader->ctxt != NULL) &&
405
3.78M
    (reader->ctxt->freeElemsNr < MAX_FREE_NODES)) {
406
18.7k
          cur->next = reader->ctxt->freeElems;
407
18.7k
    reader->ctxt->freeElems = cur;
408
18.7k
    reader->ctxt->freeElemsNr++;
409
3.76M
      } else {
410
3.76M
    xmlFree(cur);
411
3.76M
      }
412
3.78M
  }
413
414
3.78M
        if (next != NULL) {
415
1.79M
      cur = next;
416
1.98M
        } else {
417
1.98M
            if ((depth == 0) || (parent == NULL))
418
288k
                break;
419
1.70M
            depth -= 1;
420
1.70M
            cur = parent;
421
1.70M
            cur->children = NULL;
422
1.70M
        }
423
3.78M
    }
424
288k
}
425
426
/**
427
 * Free a node, this is a recursive behaviour, all the children are freed too.
428
 * This doesn't unlink the child from the list, use #xmlUnlinkNode first.
429
 *
430
 * @param reader  the xmlTextReader used
431
 * @param cur  the node
432
 */
433
static void
434
233k
xmlTextReaderFreeNode(xmlTextReaderPtr reader, xmlNodePtr cur) {
435
233k
    xmlDictPtr dict;
436
437
233k
    if ((reader != NULL) && (reader->ctxt != NULL))
438
233k
  dict = reader->ctxt->dict;
439
0
    else
440
0
        dict = NULL;
441
233k
    if (cur->type == XML_DTD_NODE) {
442
0
  xmlFreeDtd((xmlDtdPtr) cur);
443
0
  return;
444
0
    }
445
233k
    if (cur->type == XML_NAMESPACE_DECL) {
446
0
  xmlFreeNs((xmlNsPtr) cur);
447
0
        return;
448
0
    }
449
233k
    if (cur->type == XML_ATTRIBUTE_NODE) {
450
0
  xmlTextReaderFreeProp(reader, (xmlAttrPtr) cur);
451
0
  return;
452
0
    }
453
454
233k
    if ((cur->children != NULL) &&
455
233k
  (cur->type != XML_ENTITY_REF_NODE)) {
456
646
  if (cur->children->parent == cur)
457
646
      xmlTextReaderFreeNodeList(reader, cur->children);
458
646
  cur->children = NULL;
459
646
    }
460
461
233k
    if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
462
0
  xmlDeregisterNodeDefaultValue(cur);
463
464
233k
    if (((cur->type == XML_ELEMENT_NODE) ||
465
233k
   (cur->type == XML_XINCLUDE_START) ||
466
233k
   (cur->type == XML_XINCLUDE_END)) &&
467
233k
  (cur->properties != NULL))
468
21.1k
  xmlTextReaderFreePropList(reader, cur->properties);
469
233k
    if ((cur->content != (xmlChar *) &(cur->properties)) &&
470
233k
        (cur->type != XML_ELEMENT_NODE) &&
471
233k
  (cur->type != XML_XINCLUDE_START) &&
472
233k
  (cur->type != XML_XINCLUDE_END) &&
473
233k
  (cur->type != XML_ENTITY_REF_NODE)) {
474
140k
  DICT_FREE(cur->content);
475
140k
    }
476
233k
    if (((cur->type == XML_ELEMENT_NODE) ||
477
233k
   (cur->type == XML_XINCLUDE_START) ||
478
233k
   (cur->type == XML_XINCLUDE_END)) &&
479
233k
  (cur->nsDef != NULL))
480
6.53k
  xmlFreeNsList(cur->nsDef);
481
482
    /*
483
     * we don't free names here they are interned now
484
     */
485
233k
    if ((cur->type != XML_TEXT_NODE) &&
486
233k
        (cur->type != XML_COMMENT_NODE))
487
132k
  DICT_FREE(cur->name);
488
489
233k
    if (((cur->type == XML_ELEMENT_NODE) ||
490
233k
   (cur->type == XML_TEXT_NODE)) &&
491
233k
  (reader != NULL) && (reader->ctxt != NULL) &&
492
233k
  (reader->ctxt->freeElemsNr < MAX_FREE_NODES)) {
493
11.5k
  cur->next = reader->ctxt->freeElems;
494
11.5k
  reader->ctxt->freeElems = cur;
495
11.5k
  reader->ctxt->freeElemsNr++;
496
222k
    } else {
497
222k
  xmlFree(cur);
498
222k
    }
499
233k
}
500
501
/**
502
 * Free up all the structures used by a document, tree included.
503
 *
504
 * @param reader  the xmlTextReader used
505
 * @param cur  pointer to the document
506
 */
507
static void
508
32.0k
xmlTextReaderFreeDoc(xmlTextReaderPtr reader, xmlDocPtr cur) {
509
32.0k
    xmlDtdPtr extSubset, intSubset;
510
511
32.0k
    if (cur == NULL) return;
512
513
32.0k
    if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
514
0
  xmlDeregisterNodeDefaultValue((xmlNodePtr) cur);
515
516
    /*
517
     * Do this before freeing the children list to avoid ID lookups
518
     */
519
32.0k
    if (cur->ids != NULL) xmlFreeIDTable((xmlIDTablePtr) cur->ids);
520
32.0k
    cur->ids = NULL;
521
32.0k
    if (cur->refs != NULL) xmlFreeRefTable((xmlRefTablePtr) cur->refs);
522
32.0k
    cur->refs = NULL;
523
32.0k
    extSubset = cur->extSubset;
524
32.0k
    intSubset = cur->intSubset;
525
32.0k
    if (intSubset == extSubset)
526
17.5k
  extSubset = NULL;
527
32.0k
    if (extSubset != NULL) {
528
1.29k
  xmlUnlinkNode((xmlNodePtr) cur->extSubset);
529
1.29k
  cur->extSubset = NULL;
530
1.29k
  xmlFreeDtd(extSubset);
531
1.29k
    }
532
32.0k
    if (intSubset != NULL) {
533
14.4k
  xmlUnlinkNode((xmlNodePtr) cur->intSubset);
534
14.4k
  cur->intSubset = NULL;
535
14.4k
  xmlFreeDtd(intSubset);
536
14.4k
    }
537
538
32.0k
    if (cur->children != NULL) xmlTextReaderFreeNodeList(reader, cur->children);
539
540
32.0k
    if (cur->version != NULL) xmlFree(cur->version);
541
32.0k
    if (cur->name != NULL) xmlFree((char *) cur->name);
542
32.0k
    if (cur->encoding != NULL) xmlFree(cur->encoding);
543
32.0k
    if (cur->oldNs != NULL) xmlFreeNsList(cur->oldNs);
544
32.0k
    if (cur->URL != NULL) xmlFree(cur->URL);
545
32.0k
    if (cur->dict != NULL) xmlDictFree(cur->dict);
546
547
32.0k
    xmlFree(cur);
548
32.0k
}
549
550
/************************************************************************
551
 *                  *
552
 *      The reader core parser        *
553
 *                  *
554
 ************************************************************************/
555
556
static void
557
xmlTextReaderStructuredRelay(void *userData, const xmlError *error)
558
263k
{
559
263k
    xmlTextReaderPtr reader = (xmlTextReaderPtr) userData;
560
561
263k
    if (reader->sErrorFunc != NULL) {
562
263k
        reader->sErrorFunc(reader->errorFuncArg, error);
563
263k
    } else if (reader->errorFunc != NULL) {
564
0
        xmlParserSeverities severity;
565
566
0
        if ((error->domain == XML_FROM_VALID) ||
567
0
            (error->domain == XML_FROM_DTD)) {
568
0
            if (error->level == XML_ERR_WARNING)
569
0
                severity = XML_PARSER_SEVERITY_VALIDITY_WARNING;
570
0
            else
571
0
                severity = XML_PARSER_SEVERITY_VALIDITY_ERROR;
572
0
        } else {
573
0
            if (error->level == XML_ERR_WARNING)
574
0
                severity = XML_PARSER_SEVERITY_WARNING;
575
0
            else
576
0
                severity = XML_PARSER_SEVERITY_ERROR;
577
0
        }
578
579
0
        reader->errorFunc(reader->errorFuncArg, error->message, severity,
580
0
                          reader->ctxt);
581
0
    }
582
263k
}
583
584
/**
585
 * Pushes a new entity reference node on top of the entities stack
586
 *
587
 * @param reader  the xmlTextReader used
588
 * @param value  the entity reference node
589
 * @returns -1 in case of error, the index in the stack otherwise
590
 */
591
static int
592
xmlTextReaderEntPush(xmlTextReaderPtr reader, xmlNodePtr value)
593
11.6k
{
594
11.6k
    if (reader->entNr >= reader->entMax) {
595
522
        xmlNodePtr *tmp;
596
522
        int newSize;
597
598
522
        newSize = xmlGrowCapacity(reader->entMax, sizeof(tmp[0]),
599
522
                                  10, XML_MAX_ITEMS);
600
522
        if (newSize < 0) {
601
0
            xmlTextReaderErrMemory(reader);
602
0
            return (-1);
603
0
        }
604
522
        tmp = xmlRealloc(reader->entTab, newSize * sizeof(tmp[0]));
605
522
        if (tmp == NULL) {
606
2
            xmlTextReaderErrMemory(reader);
607
2
            return (-1);
608
2
        }
609
520
        reader->entTab = tmp;
610
520
        reader->entMax = newSize;
611
520
    }
612
11.5k
    reader->entTab[reader->entNr] = value;
613
11.5k
    reader->ent = value;
614
11.5k
    return (reader->entNr++);
615
11.6k
}
616
617
/**
618
 * Pops the top element entity from the entities stack
619
 *
620
 * @param reader  the xmlTextReader used
621
 * @returns the entity just removed
622
 */
623
static xmlNodePtr
624
xmlTextReaderEntPop(xmlTextReaderPtr reader)
625
11.5k
{
626
11.5k
    xmlNodePtr ret;
627
628
11.5k
    if (reader->entNr <= 0)
629
0
        return (NULL);
630
11.5k
    reader->entNr--;
631
11.5k
    if (reader->entNr > 0)
632
9.79k
        reader->ent = reader->entTab[reader->entNr - 1];
633
1.76k
    else
634
1.76k
        reader->ent = NULL;
635
11.5k
    ret = reader->entTab[reader->entNr];
636
11.5k
    reader->entTab[reader->entNr] = NULL;
637
11.5k
    return (ret);
638
11.5k
}
639
640
/**
641
 * called when an opening tag has been processed.
642
 *
643
 * @param ctx  the user data (XML parser context)
644
 * @param fullname  The element name, including namespace prefix
645
 * @param atts  An array of name/value attributes pairs, NULL terminated
646
 */
647
static void
648
xmlTextReaderStartElement(void *ctx, const xmlChar *fullname,
649
135k
                    const xmlChar **atts) {
650
135k
    xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
651
135k
    xmlTextReaderPtr reader = ctxt->_private;
652
653
135k
    if ((reader != NULL) && (reader->startElement != NULL)) {
654
135k
  reader->startElement(ctx, fullname, atts);
655
135k
  if ((ctxt->node != NULL) && (ctxt->input != NULL) &&
656
135k
      (ctxt->input->cur != NULL) && (ctxt->input->cur[0] == '/') &&
657
135k
      (ctxt->input->cur[1] == '>'))
658
26.0k
      ctxt->node->extra = NODE_IS_EMPTY;
659
135k
    }
660
135k
    if (reader != NULL)
661
135k
  reader->state = XML_TEXTREADER_ELEMENT;
662
135k
}
663
664
/**
665
 * called when an ending tag has been processed.
666
 *
667
 * @param ctx  the user data (XML parser context)
668
 * @param fullname  The element name, including namespace prefix
669
 */
670
static void
671
49.6k
xmlTextReaderEndElement(void *ctx, const xmlChar *fullname) {
672
49.6k
    xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
673
49.6k
    xmlTextReaderPtr reader = ctxt->_private;
674
675
49.6k
    if ((reader != NULL) && (reader->endElement != NULL)) {
676
49.6k
  reader->endElement(ctx, fullname);
677
49.6k
    }
678
49.6k
}
679
680
/**
681
 * called when an opening tag has been processed.
682
 *
683
 * @param ctx  the user data (XML parser context)
684
 * @param localname  the local name of the element
685
 * @param prefix  the element namespace prefix if available
686
 * @param URI  the element namespace name if available
687
 * @param nb_namespaces  number of namespace definitions on that node
688
 * @param namespaces  pointer to the array of prefix/URI pairs namespace definitions
689
 * @param nb_attributes  the number of attributes on that node
690
 * @param nb_defaulted  the number of defaulted attributes.
691
 * @param attributes  pointer to the array of (localname/prefix/URI/value/end)
692
 *               attribute values.
693
 */
694
static void
695
xmlTextReaderStartElementNs(void *ctx,
696
                      const xmlChar *localname,
697
          const xmlChar *prefix,
698
          const xmlChar *URI,
699
          int nb_namespaces,
700
          const xmlChar **namespaces,
701
          int nb_attributes,
702
          int nb_defaulted,
703
          const xmlChar **attributes)
704
220k
{
705
220k
    xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
706
220k
    xmlTextReaderPtr reader = ctxt->_private;
707
708
220k
    if ((reader != NULL) && (reader->startElementNs != NULL)) {
709
220k
  reader->startElementNs(ctx, localname, prefix, URI, nb_namespaces,
710
220k
                         namespaces, nb_attributes, nb_defaulted,
711
220k
             attributes);
712
220k
  if ((ctxt->node != NULL) && (ctxt->input != NULL) &&
713
220k
      (ctxt->input->cur != NULL) && (ctxt->input->cur[0] == '/') &&
714
220k
      (ctxt->input->cur[1] == '>'))
715
53.1k
      ctxt->node->extra = NODE_IS_EMPTY;
716
220k
    }
717
220k
    if (reader != NULL)
718
220k
  reader->state = XML_TEXTREADER_ELEMENT;
719
220k
}
720
721
/**
722
 * called when an ending tag has been processed.
723
 *
724
 * @param ctx  the user data (XML parser context)
725
 * @param localname  the local name of the element
726
 * @param prefix  the element namespace prefix if available
727
 * @param URI  the element namespace name if available
728
 */
729
static void
730
xmlTextReaderEndElementNs(void *ctx,
731
                          const xmlChar * localname,
732
                          const xmlChar * prefix,
733
              const xmlChar * URI)
734
85.8k
{
735
85.8k
    xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
736
85.8k
    xmlTextReaderPtr reader = ctxt->_private;
737
738
85.8k
    if ((reader != NULL) && (reader->endElementNs != NULL)) {
739
85.8k
  reader->endElementNs(ctx, localname, prefix, URI);
740
85.8k
    }
741
85.8k
}
742
743
744
/**
745
 * receiving some chars from the parser.
746
 *
747
 * @param ctx  the user data (XML parser context)
748
 * @param ch  a xmlChar string
749
 * @param len  the number of xmlChar
750
 */
751
static void
752
xmlTextReaderCharacters(void *ctx, const xmlChar *ch, int len)
753
390k
{
754
390k
    xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
755
390k
    xmlTextReaderPtr reader = ctxt->_private;
756
757
390k
    if ((reader != NULL) && (reader->characters != NULL)) {
758
390k
  reader->characters(ctx, ch, len);
759
390k
    }
760
390k
}
761
762
/**
763
 * called when a pcdata block has been parsed
764
 *
765
 * @param ctx  the user data (XML parser context)
766
 * @param ch  The pcdata content
767
 * @param len  the block length
768
 */
769
static void
770
xmlTextReaderCDataBlock(void *ctx, const xmlChar *ch, int len)
771
24.2k
{
772
24.2k
    xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
773
24.2k
    xmlTextReaderPtr reader = ctxt->_private;
774
775
24.2k
    if ((reader != NULL) && (reader->cdataBlock != NULL)) {
776
24.2k
  reader->cdataBlock(ctx, ch, len);
777
24.2k
    }
778
24.2k
}
779
780
/**
781
 * Push data down the progressive parser until a significant callback
782
 * got raised.
783
 *
784
 * @param reader  the xmlTextReader used
785
 * @returns -1 in case of failure, 0 otherwise
786
 */
787
static int
788
78.9k
xmlTextReaderPushData(xmlTextReaderPtr reader) {
789
78.9k
    xmlBufPtr inbuf;
790
78.9k
    int val, s;
791
78.9k
    xmlTextReaderState oldstate;
792
793
78.9k
    if ((reader->input == NULL) || (reader->input->buffer == NULL))
794
0
  return(-1);
795
796
78.9k
    oldstate = reader->state;
797
78.9k
    reader->state = XML_TEXTREADER_NONE;
798
78.9k
    inbuf = reader->input->buffer;
799
800
398k
    while (reader->state == XML_TEXTREADER_NONE) {
801
388k
  if (xmlBufUse(inbuf) < reader->cur + CHUNK_SIZE) {
802
      /*
803
       * Refill the buffer unless we are at the end of the stream
804
       */
805
66.9k
      if (reader->mode != XML_TEXTREADER_MODE_EOF) {
806
66.9k
    val = xmlParserInputBufferRead(reader->input, 4096);
807
66.9k
    if (val == 0) {
808
66.9k
        if (xmlBufUse(inbuf) == reader->cur) {
809
31.6k
      reader->mode = XML_TEXTREADER_MODE_EOF;
810
31.6k
                        break;
811
31.6k
        }
812
66.9k
    } else if (val < 0) {
813
0
                    xmlCtxtErrIO(reader->ctxt, reader->input->error, NULL);
814
0
                    reader->mode = XML_TEXTREADER_MODE_ERROR;
815
0
                    reader->state = XML_TEXTREADER_ERROR;
816
0
                    return(-1);
817
0
    }
818
819
66.9k
      } else
820
0
    break;
821
66.9k
  }
822
  /*
823
   * parse by block of CHUNK_SIZE bytes, various tests show that
824
   * it's the best tradeoff at least on a 1.2GH Duron
825
   */
826
357k
  if (xmlBufUse(inbuf) >= reader->cur + CHUNK_SIZE) {
827
321k
      val = xmlParseChunk(reader->ctxt,
828
321k
                 (const char *) xmlBufContent(inbuf) + reader->cur,
829
321k
                                CHUNK_SIZE, 0);
830
321k
      reader->cur += CHUNK_SIZE;
831
321k
      if (val != 0)
832
2.60k
    reader->ctxt->wellFormed = 0;
833
321k
      if (reader->ctxt->wellFormed == 0)
834
2.60k
    break;
835
321k
  } else {
836
35.2k
      s = xmlBufUse(inbuf) - reader->cur;
837
35.2k
      val = xmlParseChunk(reader->ctxt,
838
35.2k
     (const char *) xmlBufContent(inbuf) + reader->cur,
839
35.2k
              s, 0);
840
35.2k
      reader->cur += s;
841
35.2k
      if (val != 0)
842
6.42k
    reader->ctxt->wellFormed = 0;
843
35.2k
      break;
844
35.2k
  }
845
357k
    }
846
78.9k
    reader->state = oldstate;
847
848
    /*
849
     * Discard the consumed input when needed and possible
850
     */
851
78.9k
    if (reader->mode == XML_TEXTREADER_MODE_INTERACTIVE) {
852
44.9k
        if (reader->cur > 80 /* LINE_LEN */) {
853
28.8k
            val = xmlBufShrink(inbuf, reader->cur - 80);
854
28.8k
            if (val >= 0) {
855
28.8k
                reader->cur -= val;
856
28.8k
            }
857
28.8k
        }
858
44.9k
    }
859
860
    /*
861
     * At the end of the stream signal that the work is done to the Push
862
     * parser.
863
     */
864
33.9k
    else if (reader->mode == XML_TEXTREADER_MODE_EOF) {
865
31.6k
  if (reader->state != XML_TEXTREADER_DONE) {
866
31.6k
      s = xmlBufUse(inbuf) - reader->cur;
867
31.6k
      val = xmlParseChunk(reader->ctxt,
868
31.6k
     (const char *) xmlBufContent(inbuf) + reader->cur,
869
31.6k
              s, 1);
870
31.6k
      reader->cur = xmlBufUse(inbuf);
871
31.6k
      reader->state  = XML_TEXTREADER_DONE;
872
31.6k
      if (val != 0) {
873
28.1k
          if (reader->ctxt->wellFormed)
874
0
        reader->ctxt->wellFormed = 0;
875
28.1k
    else
876
28.1k
        return(-1);
877
28.1k
      }
878
31.6k
  }
879
31.6k
    }
880
50.7k
    if (reader->ctxt->wellFormed == 0) {
881
9.03k
  reader->mode = XML_TEXTREADER_MODE_EOF;
882
9.03k
        return(-1);
883
9.03k
    }
884
885
41.7k
    return(0);
886
50.7k
}
887
888
#ifdef LIBXML_REGEXP_ENABLED
889
/**
890
 * Push the current node for validation
891
 *
892
 * @param reader  the xmlTextReader used
893
 */
894
static int
895
109k
xmlTextReaderValidatePush(xmlTextReaderPtr reader) {
896
109k
    xmlNodePtr node = reader->node;
897
898
109k
#ifdef LIBXML_VALID_ENABLED
899
109k
    if ((reader->validate == XML_TEXTREADER_VALIDATE_DTD) &&
900
109k
        (reader->ctxt != NULL) && (reader->ctxt->validate == 1)) {
901
83.0k
  if ((node->ns == NULL) || (node->ns->prefix == NULL)) {
902
72.3k
      reader->ctxt->valid &= xmlValidatePushElement(&reader->ctxt->vctxt,
903
72.3k
            reader->ctxt->myDoc, node, node->name);
904
72.3k
  } else {
905
10.6k
            xmlChar buf[50];
906
10.6k
      xmlChar *qname;
907
908
10.6k
      qname = xmlBuildQName(node->name, node->ns->prefix, buf, 50);
909
10.6k
            if (qname == NULL) {
910
9
                xmlTextReaderErrMemory(reader);
911
9
                return(-1);
912
9
            }
913
10.6k
      reader->ctxt->valid &= xmlValidatePushElement(&reader->ctxt->vctxt,
914
10.6k
            reader->ctxt->myDoc, node, qname);
915
10.6k
            if (qname != buf)
916
1.96k
          xmlFree(qname);
917
10.6k
  }
918
        /*if (reader->ctxt->errNo == XML_ERR_NO_MEMORY) {
919
            reader->mode = XML_TEXTREADER_MODE_ERROR;
920
            reader->state = XML_TEXTREADER_ERROR;
921
            return(-1);
922
        }*/
923
83.0k
    }
924
109k
#endif /* LIBXML_VALID_ENABLED */
925
109k
#ifdef LIBXML_RELAXNG_ENABLED
926
109k
    if ((reader->validate == XML_TEXTREADER_VALIDATE_RNG) &&
927
109k
               (reader->rngValidCtxt != NULL)) {
928
0
  int ret;
929
930
0
  if (reader->rngFullNode != NULL) return(0);
931
0
  ret = xmlRelaxNGValidatePushElement(reader->rngValidCtxt,
932
0
                                      reader->ctxt->myDoc,
933
0
              node);
934
0
  if (ret == 0) {
935
      /*
936
       * this element requires a full tree
937
       */
938
0
      node = xmlTextReaderExpand(reader);
939
0
      if (node == NULL) {
940
0
          ret = -1;
941
0
      } else {
942
0
    ret = xmlRelaxNGValidateFullElement(reader->rngValidCtxt,
943
0
                reader->ctxt->myDoc,
944
0
                node);
945
0
    reader->rngFullNode = node;
946
0
      }
947
0
  }
948
0
  if (ret != 1)
949
0
      reader->rngValidErrors++;
950
0
    }
951
109k
#endif
952
953
109k
    return(0);
954
109k
}
955
956
/**
957
 * Push some CData for validation
958
 *
959
 * @param reader  the xmlTextReader used
960
 * @param data  pointer to the CData
961
 * @param len  length of the CData block in bytes.
962
 */
963
static void
964
xmlTextReaderValidateCData(xmlTextReaderPtr reader,
965
76.4k
                           const xmlChar *data, int len) {
966
76.4k
#ifdef LIBXML_VALID_ENABLED
967
76.4k
    if ((reader->validate == XML_TEXTREADER_VALIDATE_DTD) &&
968
76.4k
        (reader->ctxt != NULL) && (reader->ctxt->validate == 1)) {
969
49.7k
  reader->ctxt->valid &= xmlValidatePushCData(&reader->ctxt->vctxt,
970
49.7k
                                              data, len);
971
49.7k
    }
972
76.4k
#endif /* LIBXML_VALID_ENABLED */
973
76.4k
#ifdef LIBXML_RELAXNG_ENABLED
974
76.4k
    if ((reader->validate == XML_TEXTREADER_VALIDATE_RNG) &&
975
76.4k
               (reader->rngValidCtxt != NULL)) {
976
0
  int ret;
977
978
0
  if (reader->rngFullNode != NULL) return;
979
0
  ret = xmlRelaxNGValidatePushCData(reader->rngValidCtxt, data, len);
980
0
  if (ret != 1)
981
0
      reader->rngValidErrors++;
982
0
    }
983
76.4k
#endif
984
76.4k
}
985
986
/**
987
 * Pop the current node from validation
988
 *
989
 * @param reader  the xmlTextReader used
990
 */
991
static int
992
72.8k
xmlTextReaderValidatePop(xmlTextReaderPtr reader) {
993
72.8k
    xmlNodePtr node = reader->node;
994
995
72.8k
#ifdef LIBXML_VALID_ENABLED
996
72.8k
    if ((reader->validate == XML_TEXTREADER_VALIDATE_DTD) &&
997
72.8k
        (reader->ctxt != NULL) && (reader->ctxt->validate == 1)) {
998
61.9k
  if ((node->ns == NULL) || (node->ns->prefix == NULL)) {
999
52.9k
      reader->ctxt->valid &= xmlValidatePopElement(&reader->ctxt->vctxt,
1000
52.9k
            reader->ctxt->myDoc, node, node->name);
1001
52.9k
  } else {
1002
8.99k
            xmlChar buf[50];
1003
8.99k
      xmlChar *qname;
1004
1005
8.99k
      qname = xmlBuildQName(node->name, node->ns->prefix, buf, 50);
1006
8.99k
            if (qname == NULL) {
1007
10
                xmlTextReaderErrMemory(reader);
1008
10
                return(-1);
1009
10
            }
1010
8.98k
      reader->ctxt->valid &= xmlValidatePopElement(&reader->ctxt->vctxt,
1011
8.98k
            reader->ctxt->myDoc, node, qname);
1012
8.98k
            if (qname != buf)
1013
1.64k
          xmlFree(qname);
1014
8.98k
  }
1015
        /*if (reader->ctxt->errNo == XML_ERR_NO_MEMORY) {
1016
            reader->mode = XML_TEXTREADER_MODE_ERROR;
1017
            reader->state = XML_TEXTREADER_ERROR;
1018
            return(-1);
1019
        }*/
1020
61.9k
    }
1021
72.8k
#endif /* LIBXML_VALID_ENABLED */
1022
72.8k
#ifdef LIBXML_RELAXNG_ENABLED
1023
72.8k
    if ((reader->validate == XML_TEXTREADER_VALIDATE_RNG) &&
1024
72.8k
               (reader->rngValidCtxt != NULL)) {
1025
0
  int ret;
1026
1027
0
  if (reader->rngFullNode != NULL) {
1028
0
      if (node == reader->rngFullNode)
1029
0
          reader->rngFullNode = NULL;
1030
0
      return(0);
1031
0
  }
1032
0
  ret = xmlRelaxNGValidatePopElement(reader->rngValidCtxt,
1033
0
                                     reader->ctxt->myDoc,
1034
0
             node);
1035
0
  if (ret != 1)
1036
0
      reader->rngValidErrors++;
1037
0
    }
1038
72.8k
#endif
1039
1040
72.8k
    return(0);
1041
72.8k
}
1042
1043
/**
1044
 * Handle the validation when an entity reference is encountered and
1045
 * entity substitution is not activated. As a result the parser interface
1046
 * must walk through the entity and do the validation calls
1047
 *
1048
 * @param reader  the xmlTextReader used
1049
 */
1050
static int
1051
2.36k
xmlTextReaderValidateEntity(xmlTextReaderPtr reader) {
1052
2.36k
    xmlNodePtr oldnode = reader->node;
1053
2.36k
    xmlNodePtr node = reader->node;
1054
1055
40.7k
    do {
1056
40.7k
  if (node->type == XML_ENTITY_REF_NODE) {
1057
16.6k
      if ((node->children != NULL) &&
1058
16.6k
    (node->children->type == XML_ENTITY_DECL) &&
1059
16.6k
    (node->children->children != NULL)) {
1060
11.6k
    if (xmlTextReaderEntPush(reader, node) < 0) {
1061
2
                    if (node == oldnode)
1062
1
                        break;
1063
1
                    goto skip_children;
1064
2
                }
1065
11.5k
    node = node->children->children;
1066
11.5k
    continue;
1067
11.6k
      } else {
1068
    /*
1069
     * The error has probably been raised already.
1070
     */
1071
5.09k
    if (node == oldnode)
1072
586
        break;
1073
4.50k
                goto skip_children;
1074
5.09k
      }
1075
16.6k
#ifdef LIBXML_REGEXP_ENABLED
1076
24.0k
  } else if (node->type == XML_ELEMENT_NODE) {
1077
7.01k
      reader->node = node;
1078
7.01k
      if (xmlTextReaderValidatePush(reader) < 0)
1079
8
                return(-1);
1080
17.0k
  } else if ((node->type == XML_TEXT_NODE) ||
1081
17.0k
       (node->type == XML_CDATA_SECTION_NODE)) {
1082
16.1k
            xmlTextReaderValidateCData(reader, node->content,
1083
16.1k
                                 xmlStrlen(node->content));
1084
16.1k
#endif
1085
16.1k
  }
1086
1087
  /*
1088
   * go to next node
1089
   */
1090
24.0k
  if (node->children != NULL) {
1091
2.27k
      node = node->children;
1092
2.27k
      continue;
1093
21.7k
  } else if (node->type == XML_ELEMENT_NODE) {
1094
4.72k
      if (xmlTextReaderValidatePop(reader) < 0)
1095
6
                return(-1);
1096
4.72k
  }
1097
26.2k
skip_children:
1098
26.2k
  if (node->next != NULL) {
1099
16.1k
      node = node->next;
1100
16.1k
      continue;
1101
16.1k
  }
1102
13.8k
  do {
1103
13.8k
      node = node->parent;
1104
13.8k
      if (node->type == XML_ELEMENT_NODE) {
1105
2.27k
          xmlNodePtr tmp;
1106
2.27k
    if (reader->entNr == 0) {
1107
0
        while ((tmp = node->last) != NULL) {
1108
0
      if ((tmp->extra & NODE_IS_PRESERVED) == 0) {
1109
0
          xmlUnlinkNode(tmp);
1110
0
          xmlTextReaderFreeNode(reader, tmp);
1111
0
      } else
1112
0
          break;
1113
0
        }
1114
0
    }
1115
2.27k
    reader->node = node;
1116
2.27k
    if (xmlTextReaderValidatePop(reader) < 0)
1117
0
                    return(-1);
1118
2.27k
      }
1119
13.8k
      if ((node->type == XML_ENTITY_DECL) &&
1120
13.8k
    (reader->ent != NULL) && (reader->ent->children == node)) {
1121
11.5k
    node = xmlTextReaderEntPop(reader);
1122
11.5k
      }
1123
13.8k
      if (node == oldnode)
1124
1.76k
    break;
1125
12.0k
      if (node->next != NULL) {
1126
8.33k
    node = node->next;
1127
8.33k
    break;
1128
8.33k
      }
1129
12.0k
  } while ((node != NULL) && (node != oldnode));
1130
40.1k
    } while ((node != NULL) && (node != oldnode));
1131
2.35k
    reader->node = oldnode;
1132
1133
2.35k
    return(0);
1134
2.36k
}
1135
#endif /* LIBXML_REGEXP_ENABLED */
1136
1137
1138
/**
1139
 * Get the successor of a node if available.
1140
 *
1141
 * @param cur  the current node
1142
 * @returns the successor node or NULL
1143
 */
1144
static xmlNodePtr
1145
129k
xmlTextReaderGetSuccessor(xmlNodePtr cur) {
1146
129k
    if (cur == NULL) return(NULL) ; /* ERROR */
1147
129k
    if (cur->next != NULL) return(cur->next) ;
1148
525k
    do {
1149
525k
        cur = cur->parent;
1150
525k
        if (cur == NULL) break;
1151
501k
        if (cur->next != NULL) return(cur->next);
1152
501k
    } while (cur != NULL);
1153
23.6k
    return(cur);
1154
108k
}
1155
1156
/**
1157
 * Makes sure that the current node is fully read as well as all its
1158
 * descendant. It means the full DOM subtree must be available at the
1159
 * end of the call.
1160
 *
1161
 * @param reader  the xmlTextReader used
1162
 * @returns 1 if the node was expanded successfully, 0 if there is no more
1163
 *          nodes to read, or -1 in case of error
1164
 */
1165
static int
1166
150k
xmlTextReaderDoExpand(xmlTextReaderPtr reader) {
1167
150k
    int val;
1168
1169
150k
    if ((reader == NULL) || (reader->node == NULL) || (reader->ctxt == NULL))
1170
0
        return(-1);
1171
153k
    do {
1172
153k
  if (PARSER_STOPPED(reader->ctxt))
1173
23.7k
            return(1);
1174
1175
129k
        if (xmlTextReaderGetSuccessor(reader->node) != NULL)
1176
106k
      return(1);
1177
23.6k
  if (reader->ctxt->nodeNr < reader->depth)
1178
1.95k
      return(1);
1179
21.7k
  if (reader->mode == XML_TEXTREADER_MODE_EOF)
1180
3.85k
      return(1);
1181
17.8k
  val = xmlTextReaderPushData(reader);
1182
17.8k
  if (val < 0){
1183
14.8k
      reader->mode = XML_TEXTREADER_MODE_ERROR;
1184
14.8k
            reader->state = XML_TEXTREADER_ERROR;
1185
14.8k
      return(-1);
1186
14.8k
  }
1187
17.8k
    } while(reader->mode != XML_TEXTREADER_MODE_EOF);
1188
43
    return(1);
1189
150k
}
1190
1191
/**
1192
 *  Moves the position of the current instance to the next node in
1193
 *  the stream, exposing its properties.
1194
 *
1195
 * @param reader  the xmlTextReader used
1196
 * @returns 1 if the node was read successfully, 0 if there is no more
1197
 *          nodes to read, or -1 in case of error
1198
 */
1199
int
1200
2.24M
xmlTextReaderRead(xmlTextReader *reader) {
1201
2.24M
    int val, olddepth = 0;
1202
2.24M
    xmlTextReaderState oldstate = XML_TEXTREADER_START;
1203
2.24M
    xmlNodePtr oldnode = NULL;
1204
1205
2.24M
    if (reader == NULL)
1206
0
  return(-1);
1207
2.24M
    if (reader->state == XML_TEXTREADER_ERROR)
1208
189k
        return(-1);
1209
1210
2.05M
    reader->curnode = NULL;
1211
2.05M
    if (reader->doc != NULL)
1212
6.58k
        return(xmlTextReaderReadTree(reader));
1213
2.04M
    if (reader->ctxt == NULL)
1214
0
  return(-1);
1215
1216
2.04M
    if (reader->mode == XML_TEXTREADER_MODE_INITIAL) {
1217
36.1k
  reader->mode = XML_TEXTREADER_MODE_INTERACTIVE;
1218
  /*
1219
   * Initial state
1220
   */
1221
50.6k
  do {
1222
50.6k
      val = xmlTextReaderPushData(reader);
1223
50.6k
            if (val < 0) {
1224
17.4k
                reader->mode = XML_TEXTREADER_MODE_ERROR;
1225
17.4k
                reader->state = XML_TEXTREADER_ERROR;
1226
17.4k
                return(-1);
1227
17.4k
            }
1228
50.6k
  } while ((reader->ctxt->node == NULL) &&
1229
33.1k
     ((reader->mode != XML_TEXTREADER_MODE_EOF) &&
1230
17.7k
      (reader->state != XML_TEXTREADER_DONE)));
1231
18.7k
  if (reader->ctxt->node == NULL) {
1232
3.37k
      if (reader->ctxt->myDoc != NULL) {
1233
3.37k
    reader->node = reader->ctxt->myDoc->children;
1234
3.37k
      }
1235
3.37k
      if (reader->node == NULL) {
1236
0
                reader->mode = XML_TEXTREADER_MODE_ERROR;
1237
0
                reader->state = XML_TEXTREADER_ERROR;
1238
0
    return(-1);
1239
0
      }
1240
3.37k
      reader->state = XML_TEXTREADER_ELEMENT;
1241
15.3k
  } else {
1242
15.3k
      if (reader->ctxt->myDoc != NULL) {
1243
15.3k
    reader->node = reader->ctxt->myDoc->children;
1244
15.3k
      }
1245
15.3k
      if (reader->node == NULL)
1246
0
    reader->node = reader->ctxt->nodeTab[0];
1247
15.3k
      reader->state = XML_TEXTREADER_ELEMENT;
1248
15.3k
  }
1249
18.7k
  reader->depth = 0;
1250
18.7k
  reader->ctxt->parseMode = XML_PARSE_READER;
1251
18.7k
  goto node_found;
1252
18.7k
    }
1253
2.01M
    oldstate = reader->state;
1254
2.01M
    olddepth = reader->ctxt->nodeNr;
1255
2.01M
    oldnode = reader->node;
1256
1257
2.09M
get_next_node:
1258
2.09M
    if (reader->node == NULL) {
1259
8.03k
  if (reader->mode == XML_TEXTREADER_MODE_EOF) {
1260
7.98k
      return(0);
1261
7.98k
        } else {
1262
57
            reader->mode = XML_TEXTREADER_MODE_ERROR;
1263
57
            reader->state = XML_TEXTREADER_ERROR;
1264
57
      return(-1);
1265
57
        }
1266
8.03k
    }
1267
1268
    /*
1269
     * If we are not backtracking on ancestors or examined nodes,
1270
     * that the parser didn't finished or that we aren't at the end
1271
     * of stream, continue processing.
1272
     */
1273
2.08M
    while ((reader->node != NULL) && (reader->node->next == NULL) &&
1274
2.08M
     (reader->ctxt->nodeNr == olddepth) &&
1275
2.08M
           ((oldstate == XML_TEXTREADER_BACKTRACK) ||
1276
949k
            (reader->node->children == NULL) ||
1277
949k
      (reader->node->type == XML_ENTITY_REF_NODE) ||
1278
949k
      ((reader->node->children != NULL) &&
1279
766k
       (reader->node->children->type == XML_TEXT_NODE) &&
1280
766k
       (reader->node->children->next == NULL)) ||
1281
949k
      (reader->node->type == XML_DTD_NODE) ||
1282
949k
      (reader->node->type == XML_DOCUMENT_NODE) ||
1283
949k
      (reader->node->type == XML_HTML_DOCUMENT_NODE)) &&
1284
2.08M
     ((reader->ctxt->node == NULL) ||
1285
187k
      (reader->ctxt->node == reader->node) ||
1286
187k
      (reader->ctxt->node == reader->node->parent)) &&
1287
2.08M
     (reader->ctxt->instate != XML_PARSER_EOF) &&
1288
2.08M
     (PARSER_STOPPED(reader->ctxt) == 0)) {
1289
10.4k
  val = xmlTextReaderPushData(reader);
1290
10.4k
  if (val < 0) {
1291
4.87k
            reader->mode = XML_TEXTREADER_MODE_ERROR;
1292
4.87k
            reader->state = XML_TEXTREADER_ERROR;
1293
4.87k
      return(-1);
1294
4.87k
        }
1295
5.56k
  if (reader->node == NULL)
1296
0
      goto node_end;
1297
5.56k
    }
1298
2.07M
    if (oldstate != XML_TEXTREADER_BACKTRACK) {
1299
1.83M
  if ((reader->node->children != NULL) &&
1300
1.83M
      (reader->node->type != XML_ENTITY_REF_NODE) &&
1301
1.83M
      (reader->node->type != XML_XINCLUDE_START) &&
1302
1.83M
      (reader->node->type != XML_DTD_NODE)) {
1303
1.48M
      reader->node = reader->node->children;
1304
1.48M
      reader->depth++;
1305
1.48M
      reader->state = XML_TEXTREADER_ELEMENT;
1306
1.48M
      goto node_found;
1307
1.48M
  }
1308
1.83M
    }
1309
590k
    if (reader->node->next != NULL) {
1310
414k
  if ((oldstate == XML_TEXTREADER_ELEMENT) &&
1311
414k
            (reader->node->type == XML_ELEMENT_NODE) &&
1312
414k
      (reader->node->children == NULL) &&
1313
414k
      ((reader->node->extra & NODE_IS_EMPTY) == 0)
1314
414k
#ifdef LIBXML_XINCLUDE_ENABLED
1315
414k
      && (reader->in_xinclude <= 0)
1316
414k
#endif
1317
414k
      ) {
1318
13.9k
      reader->state = XML_TEXTREADER_END;
1319
13.9k
      goto node_found;
1320
13.9k
  }
1321
400k
#ifdef LIBXML_REGEXP_ENABLED
1322
400k
  if ((reader->validate) &&
1323
400k
      (reader->node->type == XML_ELEMENT_NODE))
1324
54.0k
      if (xmlTextReaderValidatePop(reader) < 0)
1325
3
                return(-1);
1326
400k
#endif /* LIBXML_REGEXP_ENABLED */
1327
400k
        if ((reader->preserves > 0) &&
1328
400k
      (reader->node->extra & NODE_IS_SPRESERVED))
1329
581
      reader->preserves--;
1330
400k
  reader->node = reader->node->next;
1331
400k
  reader->state = XML_TEXTREADER_ELEMENT;
1332
1333
  /*
1334
   * Cleanup of the old node
1335
   */
1336
400k
  if ((reader->preserves == 0) &&
1337
400k
#ifdef LIBXML_XINCLUDE_ENABLED
1338
400k
      (reader->in_xinclude == 0) &&
1339
400k
#endif
1340
400k
      (reader->entNr == 0) &&
1341
400k
      (reader->node->prev != NULL) &&
1342
400k
            (reader->node->prev->type != XML_DTD_NODE)) {
1343
215k
      xmlNodePtr tmp = reader->node->prev;
1344
215k
      if ((tmp->extra & NODE_IS_PRESERVED) == 0) {
1345
215k
                if (oldnode == tmp)
1346
213k
                    oldnode = NULL;
1347
215k
    xmlUnlinkNode(tmp);
1348
215k
    xmlTextReaderFreeNode(reader, tmp);
1349
215k
      }
1350
215k
  }
1351
1352
400k
  goto node_found;
1353
400k
    }
1354
175k
    if ((oldstate == XML_TEXTREADER_ELEMENT) &&
1355
175k
  (reader->node->type == XML_ELEMENT_NODE) &&
1356
175k
  (reader->node->children == NULL) &&
1357
175k
  ((reader->node->extra & NODE_IS_EMPTY) == 0)) {;
1358
1.27k
  reader->state = XML_TEXTREADER_END;
1359
1.27k
  goto node_found;
1360
1.27k
    }
1361
174k
#ifdef LIBXML_REGEXP_ENABLED
1362
174k
    if ((reader->validate != XML_TEXTREADER_NOT_VALIDATE) &&
1363
174k
        (reader->node->type == XML_ELEMENT_NODE)) {
1364
11.7k
        if (xmlTextReaderValidatePop(reader) < 0)
1365
1
            return(-1);
1366
11.7k
    }
1367
174k
#endif /* LIBXML_REGEXP_ENABLED */
1368
174k
    if ((reader->preserves > 0) &&
1369
174k
  (reader->node->extra & NODE_IS_SPRESERVED))
1370
182
  reader->preserves--;
1371
174k
    reader->node = reader->node->parent;
1372
174k
    if ((reader->node == NULL) ||
1373
174k
  (reader->node->type == XML_DOCUMENT_NODE) ||
1374
174k
  (reader->node->type == XML_HTML_DOCUMENT_NODE)) {
1375
1.83k
  if (reader->mode != XML_TEXTREADER_MODE_EOF) {
1376
756
      val = xmlParseChunk(reader->ctxt, "", 0, 1);
1377
756
      reader->state = XML_TEXTREADER_DONE;
1378
756
      if (val != 0) {
1379
741
                reader->mode = XML_TEXTREADER_MODE_ERROR;
1380
741
                reader->state = XML_TEXTREADER_ERROR;
1381
741
          return(-1);
1382
741
            }
1383
756
  }
1384
1.08k
  reader->node = NULL;
1385
1.08k
  reader->depth = -1;
1386
1387
  /*
1388
   * Cleanup of the old node
1389
   */
1390
1.08k
  if ((oldnode != NULL) && (reader->preserves == 0) &&
1391
1.08k
#ifdef LIBXML_XINCLUDE_ENABLED
1392
1.08k
      (reader->in_xinclude == 0) &&
1393
1.08k
#endif
1394
1.08k
      (reader->entNr == 0) &&
1395
1.08k
      (oldnode->type != XML_DTD_NODE) &&
1396
1.08k
      ((oldnode->extra & NODE_IS_PRESERVED) == 0)) {
1397
894
      xmlUnlinkNode(oldnode);
1398
894
      xmlTextReaderFreeNode(reader, oldnode);
1399
894
  }
1400
1401
1.08k
  goto node_end;
1402
1.83k
    }
1403
172k
    if ((reader->preserves == 0) &&
1404
172k
#ifdef LIBXML_XINCLUDE_ENABLED
1405
172k
        (reader->in_xinclude == 0) &&
1406
172k
#endif
1407
172k
  (reader->entNr == 0) &&
1408
172k
        (reader->node->last != NULL) &&
1409
172k
        ((reader->node->last->extra & NODE_IS_PRESERVED) == 0)) {
1410
17.5k
  xmlNodePtr tmp = reader->node->last;
1411
17.5k
  xmlUnlinkNode(tmp);
1412
17.5k
  xmlTextReaderFreeNode(reader, tmp);
1413
17.5k
    }
1414
172k
    reader->depth--;
1415
172k
    reader->state = XML_TEXTREADER_BACKTRACK;
1416
1417
2.09M
node_found:
1418
    /*
1419
     * If we are in the middle of a piece of CDATA make sure it's finished
1420
     */
1421
2.09M
    if ((reader->node != NULL) &&
1422
2.09M
        (reader->node->next == NULL) &&
1423
2.09M
        ((reader->node->type == XML_TEXT_NODE) ||
1424
959k
   (reader->node->type == XML_CDATA_SECTION_NODE))) {
1425
87.9k
            if (xmlTextReaderExpand(reader) == NULL)
1426
2.53k
          return -1;
1427
87.9k
    }
1428
1429
2.09M
#ifdef LIBXML_XINCLUDE_ENABLED
1430
    /*
1431
     * Handle XInclude if asked for
1432
     */
1433
2.09M
    if ((reader->xinclude) && (reader->in_xinclude == 0) &&
1434
2.09M
        (reader->state != XML_TEXTREADER_BACKTRACK) &&
1435
2.09M
        (reader->node != NULL) &&
1436
2.09M
  (reader->node->type == XML_ELEMENT_NODE) &&
1437
2.09M
  (reader->node->ns != NULL) &&
1438
2.09M
  ((xmlStrEqual(reader->node->ns->href, XINCLUDE_NS)) ||
1439
22.8k
   (xmlStrEqual(reader->node->ns->href, XINCLUDE_OLD_NS)))) {
1440
12.9k
  if (reader->xincctxt == NULL) {
1441
6.88k
      reader->xincctxt = xmlXIncludeNewContext(reader->ctxt->myDoc);
1442
6.88k
            if (reader->xincctxt == NULL) {
1443
1
                xmlTextReaderErrMemory(reader);
1444
1
                return(-1);
1445
1
            }
1446
6.88k
      xmlXIncludeSetFlags(reader->xincctxt,
1447
6.88k
                          reader->parserFlags & (~XML_PARSE_NOXINCNODE));
1448
6.88k
            xmlXIncludeSetStreamingMode(reader->xincctxt, 1);
1449
6.88k
            if ((reader->errorFunc != NULL) || (reader->sErrorFunc != NULL))
1450
6.88k
                xmlXIncludeSetErrorHandler(reader->xincctxt,
1451
6.88k
                        xmlTextReaderStructuredRelay, reader);
1452
6.88k
            if (reader->resourceLoader != NULL)
1453
6.88k
                xmlXIncludeSetResourceLoader(reader->xincctxt,
1454
6.88k
                        reader->resourceLoader, reader->resourceCtxt);
1455
6.88k
  }
1456
  /*
1457
   * expand that node and process it
1458
   */
1459
12.9k
  if (xmlTextReaderExpand(reader) == NULL)
1460
11
      return(-1);
1461
12.9k
        if (xmlXIncludeProcessNode(reader->xincctxt, reader->node) < 0) {
1462
10.6k
            int err = xmlXIncludeGetLastError(reader->xincctxt);
1463
1464
10.6k
            if (xmlIsCatastrophicError(XML_ERR_FATAL, err)) {
1465
2.10k
                xmlFatalErr(reader->ctxt, err, NULL);
1466
2.10k
                reader->mode = XML_TEXTREADER_MODE_ERROR;
1467
2.10k
                reader->state = XML_TEXTREADER_ERROR;
1468
2.10k
            }
1469
10.6k
            return(-1);
1470
10.6k
        }
1471
12.9k
    }
1472
2.08M
    if ((reader->node != NULL) && (reader->node->type == XML_XINCLUDE_START)) {
1473
2.24k
        reader->in_xinclude++;
1474
2.24k
  goto get_next_node;
1475
2.24k
    }
1476
2.08M
    if ((reader->node != NULL) && (reader->node->type == XML_XINCLUDE_END)) {
1477
79.8k
        reader->in_xinclude--;
1478
79.8k
  goto get_next_node;
1479
79.8k
    }
1480
2.00M
#endif
1481
    /*
1482
     * Handle entities enter and exit when in entity replacement mode
1483
     */
1484
2.00M
    if ((reader->node != NULL) &&
1485
2.00M
  (reader->node->type == XML_ENTITY_REF_NODE) &&
1486
2.00M
  (reader->ctxt != NULL) && (reader->ctxt->replaceEntities == 1)) {
1487
0
  if ((reader->node->children != NULL) &&
1488
0
      (reader->node->children->type == XML_ENTITY_DECL) &&
1489
0
      (reader->node->children->children != NULL)) {
1490
0
      if (xmlTextReaderEntPush(reader, reader->node) < 0)
1491
0
                goto get_next_node;
1492
0
      reader->node = reader->node->children->children;
1493
0
  }
1494
0
#ifdef LIBXML_REGEXP_ENABLED
1495
2.00M
    } else if ((reader->node != NULL) &&
1496
2.00M
         (reader->node->type == XML_ENTITY_REF_NODE) &&
1497
2.00M
         (reader->ctxt != NULL) && (reader->validate)) {
1498
2.36k
  if (xmlTextReaderValidateEntity(reader) < 0)
1499
14
            return(-1);
1500
2.36k
#endif /* LIBXML_REGEXP_ENABLED */
1501
2.36k
    }
1502
2.00M
    if ((reader->node != NULL) &&
1503
2.00M
  (reader->node->type == XML_ENTITY_DECL) &&
1504
2.00M
  (reader->ent != NULL) && (reader->ent->children == reader->node)) {
1505
0
  reader->node = xmlTextReaderEntPop(reader);
1506
0
  reader->depth++;
1507
0
        goto get_next_node;
1508
0
    }
1509
2.00M
#ifdef LIBXML_REGEXP_ENABLED
1510
2.00M
    if ((reader->validate != XML_TEXTREADER_NOT_VALIDATE) && (reader->node != NULL)) {
1511
216k
  xmlNodePtr node = reader->node;
1512
1513
216k
  if ((node->type == XML_ELEMENT_NODE) &&
1514
216k
            ((reader->state != XML_TEXTREADER_END) &&
1515
133k
       (reader->state != XML_TEXTREADER_BACKTRACK))) {
1516
102k
      if (xmlTextReaderValidatePush(reader) < 0)
1517
1
                return(-1);
1518
113k
  } else if ((node->type == XML_TEXT_NODE) ||
1519
113k
       (node->type == XML_CDATA_SECTION_NODE)) {
1520
60.2k
            xmlTextReaderValidateCData(reader, node->content,
1521
60.2k
                                 xmlStrlen(node->content));
1522
60.2k
  }
1523
216k
    }
1524
2.00M
#endif /* LIBXML_REGEXP_ENABLED */
1525
2.00M
#ifdef LIBXML_PATTERN_ENABLED
1526
2.00M
    if ((reader->patternNr > 0) && (reader->state != XML_TEXTREADER_END) &&
1527
2.00M
        (reader->state != XML_TEXTREADER_BACKTRACK)) {
1528
0
        int i;
1529
0
  for (i = 0;i < reader->patternNr;i++) {
1530
0
       if (xmlPatternMatch(reader->patternTab[i], reader->node) == 1) {
1531
0
           xmlTextReaderPreserve(reader);
1532
0
     break;
1533
0
             }
1534
0
  }
1535
0
    }
1536
2.00M
#endif /* LIBXML_PATTERN_ENABLED */
1537
2.00M
#ifdef LIBXML_SCHEMAS_ENABLED
1538
2.00M
    if ((reader->validate == XML_TEXTREADER_VALIDATE_XSD) &&
1539
2.00M
        (reader->xsdValidErrors == 0) &&
1540
2.00M
  (reader->xsdValidCtxt != NULL)) {
1541
0
  reader->xsdValidErrors = !xmlSchemaIsValid(reader->xsdValidCtxt);
1542
0
    }
1543
2.00M
#endif /* LIBXML_PATTERN_ENABLED */
1544
2.00M
    return(1);
1545
1.08k
node_end:
1546
1.08k
    reader->state = XML_TEXTREADER_DONE;
1547
1.08k
    return(0);
1548
2.00M
}
1549
1550
/**
1551
 * Gets the read state of the reader.
1552
 *
1553
 * @param reader  the xmlTextReader used
1554
 * @returns the state value, or -1 in case of error
1555
 */
1556
int
1557
7.22k
xmlTextReaderReadState(xmlTextReader *reader) {
1558
7.22k
    if (reader == NULL)
1559
0
  return(-1);
1560
7.22k
    return(reader->mode);
1561
7.22k
}
1562
1563
/**
1564
 * Reads the contents of the current node and the full subtree. It then makes
1565
 * the subtree available until the next #xmlTextReaderRead call
1566
 *
1567
 * @param reader  the xmlTextReader used
1568
 * @returns a node pointer valid until the next #xmlTextReaderRead call
1569
 *         or NULL in case of error.
1570
 */
1571
xmlNode *
1572
155k
xmlTextReaderExpand(xmlTextReader *reader) {
1573
155k
    if ((reader == NULL) || (reader->node == NULL))
1574
9.22k
        return(NULL);
1575
145k
    if (reader->doc != NULL)
1576
0
        return(reader->node);
1577
145k
    if (reader->ctxt == NULL)
1578
0
        return(NULL);
1579
145k
    if (xmlTextReaderDoExpand(reader) < 0)
1580
12.8k
        return(NULL);
1581
133k
    return(reader->node);
1582
145k
}
1583
1584
/**
1585
 * Skip to the node following the current one in document order while
1586
 * avoiding the subtree if any.
1587
 *
1588
 * @param reader  the xmlTextReader used
1589
 * @returns 1 if the node was read successfully, 0 if there is no more
1590
 *          nodes to read, or -1 in case of error
1591
 */
1592
int
1593
25.3k
xmlTextReaderNext(xmlTextReader *reader) {
1594
25.3k
    int ret;
1595
25.3k
    xmlNodePtr cur;
1596
1597
25.3k
    if (reader == NULL)
1598
0
  return(-1);
1599
25.3k
    if (reader->doc != NULL)
1600
0
        return(xmlTextReaderNextTree(reader));
1601
25.3k
    cur = reader->node;
1602
25.3k
    if ((cur == NULL) || (cur->type != XML_ELEMENT_NODE))
1603
14.1k
        return(xmlTextReaderRead(reader));
1604
11.2k
    if (reader->state == XML_TEXTREADER_END || reader->state == XML_TEXTREADER_BACKTRACK)
1605
1.30k
        return(xmlTextReaderRead(reader));
1606
9.94k
    if (cur->extra & NODE_IS_EMPTY)
1607
610
        return(xmlTextReaderRead(reader));
1608
1.93M
    do {
1609
1.93M
        ret = xmlTextReaderRead(reader);
1610
1.93M
  if (ret != 1)
1611
8.36k
      return(ret);
1612
1.93M
    } while (reader->node != cur);
1613
969
    return(xmlTextReaderRead(reader));
1614
9.33k
}
1615
1616
#ifdef LIBXML_WRITER_ENABLED
1617
static void
1618
xmlTextReaderDumpCopy(xmlTextReaderPtr reader, xmlOutputBufferPtr output,
1619
35.8k
                      xmlNodePtr node) {
1620
35.8k
    if ((node->type == XML_DTD_NODE) ||
1621
35.8k
        (node->type == XML_ELEMENT_DECL) ||
1622
35.8k
        (node->type == XML_ATTRIBUTE_DECL) ||
1623
35.8k
        (node->type == XML_ENTITY_DECL))
1624
1.89k
        return;
1625
1626
33.9k
    if ((node->type == XML_DOCUMENT_NODE) ||
1627
33.9k
        (node->type == XML_HTML_DOCUMENT_NODE)) {
1628
4.75k
        xmlNodeDumpOutput(output, node->doc, node, 0, 0, NULL);
1629
29.2k
    } else {
1630
29.2k
        xmlNodePtr copy;
1631
1632
        /*
1633
         * Create a copy to make sure that namespace declarations from
1634
         * ancestors are added.
1635
         */
1636
29.2k
        copy = xmlDocCopyNode(node, node->doc, 1);
1637
29.2k
        if (copy == NULL) {
1638
2.59k
            xmlTextReaderErrMemory(reader);
1639
2.59k
            return;
1640
2.59k
        }
1641
1642
26.6k
        xmlNodeDumpOutput(output, copy->doc, copy, 0, 0, NULL);
1643
1644
26.6k
        xmlFreeNode(copy);
1645
26.6k
    }
1646
33.9k
}
1647
1648
/**
1649
 * Reads the contents of the current node, including child nodes and markup.
1650
 *
1651
 * @param reader  the xmlTextReader used
1652
 * @returns a string containing the XML content, or NULL if the current node
1653
 *         is neither an element nor attribute, or has no child nodes. The
1654
 *         string must be deallocated by the caller.
1655
 */
1656
xmlChar *
1657
xmlTextReaderReadInnerXml(xmlTextReader *reader)
1658
12.5k
{
1659
12.5k
    xmlOutputBufferPtr output;
1660
12.5k
    xmlNodePtr cur;
1661
12.5k
    xmlChar *ret;
1662
1663
12.5k
    if (xmlTextReaderExpand(reader) == NULL)
1664
5.14k
        return(NULL);
1665
1666
7.38k
    if (reader->node == NULL)
1667
0
        return(NULL);
1668
1669
7.38k
    output = xmlAllocOutputBuffer(NULL);
1670
7.38k
    if (output == NULL) {
1671
20
        xmlTextReaderErrMemory(reader);
1672
20
        return(NULL);
1673
20
    }
1674
1675
20.1k
    for (cur = reader->node->children; cur != NULL; cur = cur->next)
1676
12.8k
        xmlTextReaderDumpCopy(reader, output, cur);
1677
1678
7.36k
    if (output->error)
1679
6
        xmlCtxtErrIO(reader->ctxt, output->error, NULL);
1680
1681
7.36k
    ret = xmlBufDetach(output->buffer);
1682
7.36k
    xmlOutputBufferClose(output);
1683
1684
7.36k
    return(ret);
1685
7.38k
}
1686
1687
/**
1688
 * Reads the contents of the current node, including child nodes and markup.
1689
 *
1690
 * @param reader  the xmlTextReader used
1691
 * @returns a string containing the node and any XML content, or NULL if the
1692
 *         current node cannot be serialized. The string must be deallocated
1693
 *         by the caller.
1694
 */
1695
xmlChar *
1696
xmlTextReaderReadOuterXml(xmlTextReader *reader)
1697
32.1k
{
1698
32.1k
    xmlOutputBufferPtr output;
1699
32.1k
    xmlNodePtr node;
1700
32.1k
    xmlChar *ret;
1701
1702
32.1k
    if (xmlTextReaderExpand(reader) == NULL)
1703
9.00k
        return(NULL);
1704
1705
23.1k
    node = reader->node;
1706
23.1k
    if (node == NULL)
1707
0
        return(NULL);
1708
1709
23.1k
    output = xmlAllocOutputBuffer(NULL);
1710
23.1k
    if (output == NULL) {
1711
53
        xmlTextReaderErrMemory(reader);
1712
53
        return(NULL);
1713
53
    }
1714
1715
23.0k
    xmlTextReaderDumpCopy(reader, output, node);
1716
23.0k
    if (output->error)
1717
14
        xmlCtxtErrIO(reader->ctxt, output->error, NULL);
1718
1719
23.0k
    ret = xmlBufDetach(output->buffer);
1720
23.0k
    xmlOutputBufferClose(output);
1721
1722
23.0k
    return(ret);
1723
23.1k
}
1724
#endif
1725
1726
/**
1727
 * Reads the contents of an element or a text node as a string.
1728
 *
1729
 * @param reader  the xmlTextReader used
1730
 * @returns a string containing the contents of the non-empty Element or
1731
 *         Text node (including CDATA sections), or NULL if the reader
1732
 *         is positioned on any other type of node.
1733
 *         The string must be deallocated by the caller.
1734
 */
1735
xmlChar *
1736
xmlTextReaderReadString(xmlTextReader *reader)
1737
12.0k
{
1738
12.0k
    xmlNodePtr node, cur;
1739
12.0k
    xmlBufPtr buf;
1740
12.0k
    xmlChar *ret;
1741
1742
12.0k
    if ((reader == NULL) || (reader->node == NULL))
1743
3.39k
       return(NULL);
1744
1745
8.64k
    node = (reader->curnode != NULL) ? reader->curnode : reader->node;
1746
8.64k
    switch (node->type) {
1747
1.82k
        case XML_TEXT_NODE:
1748
2.21k
        case XML_CDATA_SECTION_NODE:
1749
2.21k
            break;
1750
4.49k
        case XML_ELEMENT_NODE:
1751
4.49k
            if ((xmlTextReaderDoExpand(reader) == -1) ||
1752
4.49k
                (node->children == NULL))
1753
3.57k
                return(NULL);
1754
922
            break;
1755
1.93k
        default:
1756
1.93k
            return(NULL);
1757
8.64k
    }
1758
1759
3.14k
    buf = xmlBufCreate(50);
1760
3.14k
    if (buf == NULL) {
1761
11
        xmlTextReaderErrMemory(reader);
1762
11
        return(NULL);
1763
11
    }
1764
1765
3.12k
    cur = node;
1766
858k
    while (cur != NULL) {
1767
858k
        switch (cur->type) {
1768
223k
            case XML_TEXT_NODE:
1769
224k
            case XML_CDATA_SECTION_NODE:
1770
224k
                xmlBufCat(buf, cur->content);
1771
224k
                break;
1772
1773
430k
            case XML_ELEMENT_NODE:
1774
430k
                if (cur->children != NULL) {
1775
412k
                    cur = cur->children;
1776
412k
                    continue;
1777
412k
                }
1778
18.3k
                break;
1779
1780
203k
            default:
1781
203k
                break;
1782
858k
        }
1783
1784
445k
        if (cur == node)
1785
2.21k
            goto done;
1786
1787
854k
        while (cur->next == NULL) {
1788
412k
            cur = cur->parent;
1789
412k
            if (cur == node)
1790
919
                goto done;
1791
412k
        }
1792
442k
        cur = cur->next;
1793
442k
    }
1794
1795
3.12k
done:
1796
3.12k
    ret = xmlBufDetach(buf);
1797
3.12k
    if (ret == NULL)
1798
3
        xmlTextReaderErrMemory(reader);
1799
1800
3.12k
    xmlBufFree(buf);
1801
3.12k
    return(ret);
1802
3.12k
}
1803
1804
/************************************************************************
1805
 *                  *
1806
 *      Operating on a preparsed tree     *
1807
 *                  *
1808
 ************************************************************************/
1809
static int
1810
xmlTextReaderNextTree(xmlTextReaderPtr reader)
1811
0
{
1812
0
    if (reader == NULL)
1813
0
        return(-1);
1814
1815
0
    if (reader->state == XML_TEXTREADER_END)
1816
0
        return(0);
1817
1818
0
    if (reader->node == NULL) {
1819
0
        if (reader->doc->children == NULL) {
1820
0
            reader->state = XML_TEXTREADER_END;
1821
0
            return(0);
1822
0
        }
1823
1824
0
        reader->node = reader->doc->children;
1825
0
        reader->state = XML_TEXTREADER_START;
1826
0
        return(1);
1827
0
    }
1828
1829
0
    if (reader->state != XML_TEXTREADER_BACKTRACK) {
1830
  /* Here removed traversal to child, because we want to skip the subtree,
1831
  replace with traversal to sibling to skip subtree */
1832
0
        if (reader->node->next != 0) {
1833
      /* Move to sibling if present,skipping sub-tree */
1834
0
            reader->node = reader->node->next;
1835
0
            reader->state = XML_TEXTREADER_START;
1836
0
            return(1);
1837
0
        }
1838
1839
  /* if reader->node->next is NULL mean no subtree for current node,
1840
  so need to move to sibling of parent node if present */
1841
0
  reader->state = XML_TEXTREADER_BACKTRACK;
1842
  /* This will move to parent if present */
1843
0
  xmlTextReaderRead(reader);
1844
0
    }
1845
1846
0
    if (reader->node->next != 0) {
1847
0
        reader->node = reader->node->next;
1848
0
        reader->state = XML_TEXTREADER_START;
1849
0
        return(1);
1850
0
    }
1851
1852
0
    if (reader->node->parent != 0) {
1853
0
        if (reader->node->parent->type == XML_DOCUMENT_NODE) {
1854
0
            reader->state = XML_TEXTREADER_END;
1855
0
            return(0);
1856
0
        }
1857
1858
0
        reader->node = reader->node->parent;
1859
0
        reader->depth--;
1860
0
        reader->state = XML_TEXTREADER_BACKTRACK;
1861
  /* Repeat process to move to sibling of parent node if present */
1862
0
        xmlTextReaderNextTree(reader);
1863
0
    }
1864
1865
0
    reader->state = XML_TEXTREADER_END;
1866
1867
0
    return(1);
1868
0
}
1869
1870
/**
1871
 *  Moves the position of the current instance to the next node in
1872
 *  the stream, exposing its properties.
1873
 *
1874
 * @param reader  the xmlTextReader used
1875
 * @returns 1 if the node was read successfully, 0 if there is no more
1876
 *          nodes to read, or -1 in case of error
1877
 */
1878
static int
1879
6.58k
xmlTextReaderReadTree(xmlTextReaderPtr reader) {
1880
6.58k
    if (reader->state == XML_TEXTREADER_END)
1881
0
        return(0);
1882
1883
6.58k
next_node:
1884
6.58k
    if (reader->node == NULL) {
1885
1.64k
        if (reader->doc->children == NULL) {
1886
0
            reader->state = XML_TEXTREADER_END;
1887
0
            return(0);
1888
0
        }
1889
1890
1.64k
        reader->node = reader->doc->children;
1891
1.64k
        reader->state = XML_TEXTREADER_START;
1892
1.64k
        goto found_node;
1893
1.64k
    }
1894
1895
4.94k
    if ((reader->state != XML_TEXTREADER_BACKTRACK) &&
1896
4.94k
        (reader->node->type != XML_DTD_NODE) &&
1897
4.94k
        (reader->node->type != XML_XINCLUDE_START) &&
1898
4.94k
  (reader->node->type != XML_ENTITY_REF_NODE)) {
1899
3.29k
        if (reader->node->children != NULL) {
1900
1.64k
            reader->node = reader->node->children;
1901
1.64k
            reader->depth++;
1902
1.64k
            reader->state = XML_TEXTREADER_START;
1903
1.64k
            goto found_node;
1904
1.64k
        }
1905
1906
1.64k
        if (reader->node->type == XML_ATTRIBUTE_NODE) {
1907
0
            reader->state = XML_TEXTREADER_BACKTRACK;
1908
0
            goto found_node;
1909
0
        }
1910
1.64k
    }
1911
1912
3.29k
    if (reader->node->next != NULL) {
1913
0
        reader->node = reader->node->next;
1914
0
        reader->state = XML_TEXTREADER_START;
1915
0
        goto found_node;
1916
0
    }
1917
1918
3.29k
    if (reader->node->parent != NULL) {
1919
3.29k
        if ((reader->node->parent->type == XML_DOCUMENT_NODE) ||
1920
3.29k
      (reader->node->parent->type == XML_HTML_DOCUMENT_NODE)) {
1921
1.64k
            reader->state = XML_TEXTREADER_END;
1922
1.64k
            return(0);
1923
1.64k
        }
1924
1925
1.64k
        reader->node = reader->node->parent;
1926
1.64k
        reader->depth--;
1927
1.64k
        reader->state = XML_TEXTREADER_BACKTRACK;
1928
1.64k
        goto found_node;
1929
3.29k
    }
1930
1931
0
    reader->state = XML_TEXTREADER_END;
1932
1933
4.94k
found_node:
1934
4.94k
    if ((reader->node->type == XML_XINCLUDE_START) ||
1935
4.94k
        (reader->node->type == XML_XINCLUDE_END))
1936
0
  goto next_node;
1937
1938
4.94k
    return(1);
1939
4.94k
}
1940
1941
/**
1942
 * Skip to the node following the current one in document order while
1943
 * avoiding the subtree if any.
1944
 * Currently implemented only for Readers built on a document
1945
 *
1946
 * @param reader  the xmlTextReader used
1947
 * @returns 1 if the node was read successfully, 0 if there is no more
1948
 *          nodes to read, or -1 in case of error
1949
 */
1950
int
1951
10.4k
xmlTextReaderNextSibling(xmlTextReader *reader) {
1952
10.4k
    if (reader == NULL)
1953
0
        return(-1);
1954
10.4k
    if (reader->doc == NULL) {
1955
        /* TODO */
1956
10.4k
  return(-1);
1957
10.4k
    }
1958
1959
0
    if (reader->state == XML_TEXTREADER_END)
1960
0
        return(0);
1961
1962
0
    if (reader->node == NULL)
1963
0
        return(xmlTextReaderNextTree(reader));
1964
1965
0
    if (reader->node->next != NULL) {
1966
0
        reader->node = reader->node->next;
1967
0
        reader->state = XML_TEXTREADER_START;
1968
0
        return(1);
1969
0
    }
1970
1971
0
    return(0);
1972
0
}
1973
1974
/************************************************************************
1975
 *                  *
1976
 *      Constructor and destructors     *
1977
 *                  *
1978
 ************************************************************************/
1979
/**
1980
 * Create an xmlTextReader structure fed with `input`
1981
 *
1982
 * @param input  the xmlParserInputBuffer used to read data
1983
 * @param URI  the URI information for the source if available
1984
 * @returns the new xmlTextReader or NULL in case of error
1985
 */
1986
xmlTextReader *
1987
36.9k
xmlNewTextReader(xmlParserInputBuffer *input, const char *URI) {
1988
36.9k
    xmlTextReaderPtr ret;
1989
1990
36.9k
    if (input == NULL)
1991
0
  return(NULL);
1992
36.9k
    ret = xmlMalloc(sizeof(xmlTextReader));
1993
36.9k
    if (ret == NULL)
1994
3
  return(NULL);
1995
36.9k
    memset(ret, 0, sizeof(xmlTextReader));
1996
36.9k
    ret->doc = NULL;
1997
36.9k
    ret->entTab = NULL;
1998
36.9k
    ret->entMax = 0;
1999
36.9k
    ret->entNr = 0;
2000
36.9k
    ret->input = input;
2001
36.9k
    ret->buffer = xmlBufCreate(50);
2002
36.9k
    if (ret->buffer == NULL) {
2003
5
        xmlFree(ret);
2004
5
  return(NULL);
2005
5
    }
2006
36.8k
    ret->sax = (xmlSAXHandler *) xmlMalloc(sizeof(xmlSAXHandler));
2007
36.8k
    if (ret->sax == NULL) {
2008
2
  xmlBufFree(ret->buffer);
2009
2
  xmlFree(ret);
2010
2
  return(NULL);
2011
2
    }
2012
36.8k
    xmlSAXVersion(ret->sax, 2);
2013
36.8k
    ret->startElement = ret->sax->startElement;
2014
36.8k
    ret->sax->startElement = xmlTextReaderStartElement;
2015
36.8k
    ret->endElement = ret->sax->endElement;
2016
36.8k
    ret->sax->endElement = xmlTextReaderEndElement;
2017
36.8k
#ifdef LIBXML_SAX1_ENABLED
2018
36.8k
    if (ret->sax->initialized == XML_SAX2_MAGIC) {
2019
36.8k
#endif /* LIBXML_SAX1_ENABLED */
2020
36.8k
  ret->startElementNs = ret->sax->startElementNs;
2021
36.8k
  ret->sax->startElementNs = xmlTextReaderStartElementNs;
2022
36.8k
  ret->endElementNs = ret->sax->endElementNs;
2023
36.8k
  ret->sax->endElementNs = xmlTextReaderEndElementNs;
2024
36.8k
#ifdef LIBXML_SAX1_ENABLED
2025
36.8k
    } else {
2026
0
  ret->startElementNs = NULL;
2027
0
  ret->endElementNs = NULL;
2028
0
    }
2029
36.8k
#endif /* LIBXML_SAX1_ENABLED */
2030
36.8k
    ret->characters = ret->sax->characters;
2031
36.8k
    ret->sax->characters = xmlTextReaderCharacters;
2032
36.8k
    ret->sax->ignorableWhitespace = xmlTextReaderCharacters;
2033
36.8k
    ret->cdataBlock = ret->sax->cdataBlock;
2034
36.8k
    ret->sax->cdataBlock = xmlTextReaderCDataBlock;
2035
2036
36.8k
    ret->mode = XML_TEXTREADER_MODE_INITIAL;
2037
36.8k
    ret->node = NULL;
2038
36.8k
    ret->curnode = NULL;
2039
36.8k
    if (xmlBufUse(ret->input->buffer) < 4) {
2040
1.15k
  xmlParserInputBufferRead(input, 4);
2041
1.15k
    }
2042
36.8k
    if (xmlBufUse(ret->input->buffer) >= 4) {
2043
35.7k
  ret->ctxt = xmlCreatePushParserCtxt(ret->sax, NULL,
2044
35.7k
           (const char *) xmlBufContent(ret->input->buffer),
2045
35.7k
                                            4, URI);
2046
35.7k
  ret->base = 0;
2047
35.7k
  ret->cur = 4;
2048
35.7k
    } else {
2049
1.15k
  ret->ctxt = xmlCreatePushParserCtxt(ret->sax, NULL, NULL, 0, URI);
2050
1.15k
  ret->base = 0;
2051
1.15k
  ret->cur = 0;
2052
1.15k
    }
2053
2054
36.8k
    if (ret->ctxt == NULL) {
2055
58
  xmlBufFree(ret->buffer);
2056
58
  xmlFree(ret->sax);
2057
58
  xmlFree(ret);
2058
58
  return(NULL);
2059
58
    }
2060
36.8k
    ret->ctxt->parseMode = XML_PARSE_READER;
2061
36.8k
    ret->ctxt->_private = ret;
2062
36.8k
    ret->ctxt->dictNames = 1;
2063
36.8k
    ret->allocs = XML_TEXTREADER_CTXT;
2064
    /*
2065
     * use the parser dictionary to allocate all elements and attributes names
2066
     */
2067
36.8k
    ret->dict = ret->ctxt->dict;
2068
36.8k
#ifdef LIBXML_XINCLUDE_ENABLED
2069
36.8k
    ret->xinclude = 0;
2070
36.8k
#endif
2071
36.8k
#ifdef LIBXML_PATTERN_ENABLED
2072
36.8k
    ret->patternMax = 0;
2073
36.8k
    ret->patternTab = NULL;
2074
36.8k
#endif
2075
36.8k
    return(ret);
2076
36.8k
}
2077
2078
/**
2079
 * Create an xmlTextReader structure fed with the resource at `URI`
2080
 *
2081
 * @param URI  the URI of the resource to process
2082
 * @returns the new xmlTextReader or NULL in case of error
2083
 */
2084
xmlTextReader *
2085
3.07k
xmlNewTextReaderFilename(const char *URI) {
2086
3.07k
    xmlParserInputBufferPtr input;
2087
3.07k
    xmlTextReaderPtr ret;
2088
2089
3.07k
    if (xmlParserInputBufferCreateFilenameValue != NULL) {
2090
0
        input = xmlParserInputBufferCreateFilenameValue(URI,
2091
0
                XML_CHAR_ENCODING_NONE);
2092
0
        if (input == NULL) {
2093
0
            xmlTextReaderErr(XML_IO_ENOENT, "filaed to open %s", URI);
2094
0
            return(NULL);
2095
0
        }
2096
3.07k
    } else {
2097
3.07k
        xmlParserErrors code;
2098
2099
        /*
2100
         * TODO: Remove XML_INPUT_UNZIP
2101
         */
2102
3.07k
        code = xmlParserInputBufferCreateUrl(URI, XML_CHAR_ENCODING_NONE,
2103
3.07k
                                             XML_INPUT_UNZIP, &input);
2104
3.07k
        if (code != XML_ERR_OK) {
2105
3.05k
            xmlTextReaderErr(code, "failed to open %s", URI);
2106
3.05k
            return(NULL);
2107
3.05k
        }
2108
3.07k
    }
2109
2110
14
    ret = xmlNewTextReader(input, URI);
2111
14
    if (ret == NULL) {
2112
14
        xmlTextReaderErrMemory(NULL);
2113
14
  xmlFreeParserInputBuffer(input);
2114
14
  return(NULL);
2115
14
    }
2116
0
    ret->allocs |= XML_TEXTREADER_INPUT;
2117
0
    return(ret);
2118
14
}
2119
2120
/**
2121
 * Deallocate all the resources associated to the reader
2122
 *
2123
 * @param reader  the xmlTextReader
2124
 */
2125
void
2126
38.4k
xmlFreeTextReader(xmlTextReader *reader) {
2127
38.4k
    if (reader == NULL)
2128
0
  return;
2129
38.4k
#ifdef LIBXML_RELAXNG_ENABLED
2130
38.4k
    if (reader->rngSchemas != NULL) {
2131
0
  xmlRelaxNGFree(reader->rngSchemas);
2132
0
  reader->rngSchemas = NULL;
2133
0
    }
2134
38.4k
    if (reader->rngValidCtxt != NULL) {
2135
0
  if (! reader->rngPreserveCtxt)
2136
0
      xmlRelaxNGFreeValidCtxt(reader->rngValidCtxt);
2137
0
  reader->rngValidCtxt = NULL;
2138
0
    }
2139
38.4k
#endif
2140
38.4k
#ifdef LIBXML_SCHEMAS_ENABLED
2141
38.4k
    if (reader->xsdPlug != NULL) {
2142
0
  xmlSchemaSAXUnplug(reader->xsdPlug);
2143
0
  reader->xsdPlug = NULL;
2144
0
    }
2145
38.4k
    if (reader->xsdValidCtxt != NULL) {
2146
0
  if (! reader->xsdPreserveCtxt)
2147
0
      xmlSchemaFreeValidCtxt(reader->xsdValidCtxt);
2148
0
  reader->xsdValidCtxt = NULL;
2149
0
    }
2150
38.4k
    if (reader->xsdSchemas != NULL) {
2151
0
  xmlSchemaFree(reader->xsdSchemas);
2152
0
  reader->xsdSchemas = NULL;
2153
0
    }
2154
38.4k
#endif
2155
38.4k
#ifdef LIBXML_XINCLUDE_ENABLED
2156
38.4k
    if (reader->xincctxt != NULL)
2157
6.88k
  xmlXIncludeFreeContext(reader->xincctxt);
2158
38.4k
#endif
2159
38.4k
#ifdef LIBXML_PATTERN_ENABLED
2160
38.4k
    if (reader->patternTab != NULL) {
2161
0
        int i;
2162
0
  for (i = 0;i < reader->patternNr;i++) {
2163
0
      if (reader->patternTab[i] != NULL)
2164
0
          xmlFreePattern(reader->patternTab[i]);
2165
0
  }
2166
0
  xmlFree(reader->patternTab);
2167
0
    }
2168
38.4k
#endif
2169
38.4k
    if (reader->mode != XML_TEXTREADER_MODE_CLOSED)
2170
38.1k
        xmlTextReaderClose(reader);
2171
38.4k
    if (reader->ctxt != NULL) {
2172
36.8k
        if (reader->dict == reader->ctxt->dict)
2173
36.8k
      reader->dict = NULL;
2174
36.8k
  if (reader->allocs & XML_TEXTREADER_CTXT)
2175
36.8k
      xmlFreeParserCtxt(reader->ctxt);
2176
36.8k
    }
2177
38.4k
    if (reader->sax != NULL)
2178
36.8k
  xmlFree(reader->sax);
2179
38.4k
    if (reader->buffer != NULL)
2180
36.8k
        xmlBufFree(reader->buffer);
2181
38.4k
    if (reader->entTab != NULL)
2182
238
  xmlFree(reader->entTab);
2183
38.4k
    if (reader->dict != NULL)
2184
1.64k
        xmlDictFree(reader->dict);
2185
38.4k
    xmlFree(reader);
2186
38.4k
}
2187
2188
/************************************************************************
2189
 *                  *
2190
 *      Methods for XmlTextReader     *
2191
 *                  *
2192
 ************************************************************************/
2193
2194
/**
2195
 * This method releases any resources allocated by the current instance
2196
 * changes the state to Closed and close any underlying input.
2197
 *
2198
 * @param reader  the xmlTextReader used
2199
 * @returns 0 or -1 in case of error
2200
 */
2201
int
2202
39.6k
xmlTextReaderClose(xmlTextReader *reader) {
2203
39.6k
    if (reader == NULL)
2204
0
  return(-1);
2205
39.6k
    reader->node = NULL;
2206
39.6k
    reader->curnode = NULL;
2207
39.6k
    reader->mode = XML_TEXTREADER_MODE_CLOSED;
2208
39.6k
    if (reader->faketext != NULL) {
2209
151
        xmlFreeNode(reader->faketext);
2210
151
        reader->faketext = NULL;
2211
151
    }
2212
39.6k
    if (reader->ctxt != NULL) {
2213
38.0k
#ifdef LIBXML_VALID_ENABLED
2214
38.0k
  if ((reader->ctxt->vctxt.vstateTab != NULL) &&
2215
38.0k
      (reader->ctxt->vctxt.vstateMax > 0)){
2216
3.00k
#ifdef LIBXML_REGEXP_ENABLED
2217
24.0k
            while (reader->ctxt->vctxt.vstateNr > 0)
2218
21.0k
                xmlValidatePopElement(&reader->ctxt->vctxt, NULL, NULL, NULL);
2219
3.00k
#endif /* LIBXML_REGEXP_ENABLED */
2220
3.00k
      xmlFree(reader->ctxt->vctxt.vstateTab);
2221
3.00k
      reader->ctxt->vctxt.vstateTab = NULL;
2222
3.00k
      reader->ctxt->vctxt.vstateMax = 0;
2223
3.00k
  }
2224
38.0k
#endif /* LIBXML_VALID_ENABLED */
2225
38.0k
  xmlStopParser(reader->ctxt);
2226
38.0k
  if (reader->ctxt->myDoc != NULL) {
2227
34.8k
      if (reader->preserve == 0)
2228
32.0k
    xmlTextReaderFreeDoc(reader, reader->ctxt->myDoc);
2229
34.8k
      reader->ctxt->myDoc = NULL;
2230
34.8k
  }
2231
38.0k
    }
2232
39.6k
    if ((reader->input != NULL)  && (reader->allocs & XML_TEXTREADER_INPUT)) {
2233
35.8k
  xmlFreeParserInputBuffer(reader->input);
2234
35.8k
  reader->allocs -= XML_TEXTREADER_INPUT;
2235
35.8k
    }
2236
39.6k
    return(0);
2237
39.6k
}
2238
2239
/**
2240
 * Provides the value of the attribute with the specified index relative
2241
 * to the containing element.
2242
 *
2243
 * @param reader  the xmlTextReader used
2244
 * @param no  the zero-based index of the attribute relative to the containing element
2245
 * @returns a string containing the value of the specified attribute, or NULL
2246
 *    in case of error. The string must be deallocated by the caller.
2247
 */
2248
xmlChar *
2249
4.37k
xmlTextReaderGetAttributeNo(xmlTextReader *reader, int no) {
2250
4.37k
    xmlChar *ret;
2251
4.37k
    int i;
2252
4.37k
    xmlAttrPtr cur;
2253
4.37k
    xmlNsPtr ns;
2254
2255
4.37k
    if (reader == NULL)
2256
0
  return(NULL);
2257
4.37k
    if (reader->node == NULL)
2258
1.01k
  return(NULL);
2259
3.36k
    if (reader->curnode != NULL)
2260
277
  return(NULL);
2261
    /* TODO: handle the xmlDecl */
2262
3.08k
    if (reader->node->type != XML_ELEMENT_NODE)
2263
637
  return(NULL);
2264
2265
2.45k
    ns = reader->node->nsDef;
2266
2.98k
    for (i = 0;(i < no) && (ns != NULL);i++) {
2267
534
  ns = ns->next;
2268
534
    }
2269
2.45k
    if (ns != NULL)
2270
575
  return(readerStrdup(reader, ns->href));
2271
2272
1.87k
    cur = reader->node->properties;
2273
1.87k
    if (cur == NULL)
2274
734
  return(NULL);
2275
2.96k
    for (;i < no;i++) {
2276
2.33k
  cur = cur->next;
2277
2.33k
  if (cur == NULL)
2278
508
      return(NULL);
2279
2.33k
    }
2280
    /* TODO walk the DTD if present */
2281
2282
635
    if (cur->children == NULL)
2283
384
        return(NULL);
2284
251
    ret = xmlNodeListGetString(reader->node->doc, cur->children, 1);
2285
251
    if (ret == NULL)
2286
1
        xmlTextReaderErrMemory(reader);
2287
251
    return(ret);
2288
635
}
2289
2290
/**
2291
 * Provides the value of the attribute with the specified qualified name.
2292
 *
2293
 * @param reader  the xmlTextReader used
2294
 * @param name  the qualified name of the attribute.
2295
 * @returns a string containing the value of the specified attribute, or NULL
2296
 *    in case of error. The string must be deallocated by the caller.
2297
 */
2298
xmlChar *
2299
34.6k
xmlTextReaderGetAttribute(xmlTextReader *reader, const xmlChar *name) {
2300
34.6k
    xmlChar *prefix = NULL;
2301
34.6k
    const xmlChar *localname;
2302
34.6k
    xmlNsPtr ns;
2303
34.6k
    xmlChar *ret = NULL;
2304
34.6k
    int result;
2305
2306
34.6k
    if ((reader == NULL) || (name == NULL))
2307
16.7k
  return(NULL);
2308
17.8k
    if (reader->node == NULL)
2309
0
  return(NULL);
2310
17.8k
    if (reader->curnode != NULL)
2311
1.69k
  return(NULL);
2312
2313
    /* TODO: handle the xmlDecl */
2314
16.1k
    if (reader->node->type != XML_ELEMENT_NODE)
2315
6.31k
  return(NULL);
2316
2317
9.82k
    localname = xmlSplitQName4(name, &prefix);
2318
9.82k
    if (localname == NULL) {
2319
12
        xmlTextReaderErrMemory(reader);
2320
12
        return(NULL);
2321
12
    }
2322
9.81k
    if (prefix == NULL) {
2323
        /*
2324
         * Namespace default decl
2325
         */
2326
6.80k
        if (xmlStrEqual(name, BAD_CAST "xmlns")) {
2327
507
            ns = reader->node->nsDef;
2328
833
            while (ns != NULL) {
2329
521
                if (ns->prefix == NULL) {
2330
195
                    return(readerStrdup(reader, ns->href));
2331
195
                }
2332
326
                ns = ns->next;
2333
326
            }
2334
312
            return NULL;
2335
507
        }
2336
2337
6.30k
        result = xmlNodeGetAttrValue(reader->node, name, NULL, &ret);
2338
6.30k
        if (result < 0)
2339
11
            xmlTextReaderErrMemory(reader);
2340
6.30k
        return(ret);
2341
6.80k
    }
2342
2343
    /*
2344
     * Namespace default decl
2345
     */
2346
3.00k
    if (xmlStrEqual(prefix, BAD_CAST "xmlns")) {
2347
781
        ns = reader->node->nsDef;
2348
1.31k
        while (ns != NULL) {
2349
777
            if ((ns->prefix != NULL) && (xmlStrEqual(ns->prefix, localname))) {
2350
247
                ret = readerStrdup(reader, ns->href);
2351
247
                break;
2352
247
            }
2353
530
            ns = ns->next;
2354
530
        }
2355
2.22k
    } else {
2356
2.22k
        result = xmlSearchNsSafe(reader->node, prefix, &ns);
2357
2.22k
        if (result < 0)
2358
1
            xmlTextReaderErrMemory(reader);
2359
2.22k
        if (ns != NULL) {
2360
1.45k
            result = xmlNodeGetAttrValue(reader->node, localname, ns->href,
2361
1.45k
                                         &ret);
2362
1.45k
            if (result < 0)
2363
3
                xmlTextReaderErrMemory(reader);
2364
1.45k
        }
2365
2.22k
    }
2366
2367
3.00k
    if (prefix != NULL)
2368
3.00k
        xmlFree(prefix);
2369
3.00k
    return(ret);
2370
9.81k
}
2371
2372
2373
/**
2374
 * Provides the value of the specified attribute
2375
 *
2376
 * @param reader  the xmlTextReader used
2377
 * @param localName  the local name of the attribute.
2378
 * @param namespaceURI  the namespace URI of the attribute.
2379
 * @returns a string containing the value of the specified attribute, or NULL
2380
 *    in case of error. The string must be deallocated by the caller.
2381
 */
2382
xmlChar *
2383
xmlTextReaderGetAttributeNs(xmlTextReader *reader, const xmlChar *localName,
2384
24.5k
          const xmlChar *namespaceURI) {
2385
24.5k
    xmlChar *ret = NULL;
2386
24.5k
    xmlChar *prefix = NULL;
2387
24.5k
    xmlNsPtr ns;
2388
24.5k
    int result;
2389
2390
24.5k
    if ((reader == NULL) || (localName == NULL))
2391
9.79k
  return(NULL);
2392
14.7k
    if (reader->node == NULL)
2393
0
  return(NULL);
2394
14.7k
    if (reader->curnode != NULL)
2395
2.30k
  return(NULL);
2396
2397
    /* TODO: handle the xmlDecl */
2398
12.4k
    if (reader->node->type != XML_ELEMENT_NODE)
2399
4.95k
  return(NULL);
2400
2401
7.48k
    if (xmlStrEqual(namespaceURI, BAD_CAST "http://www.w3.org/2000/xmlns/")) {
2402
658
        if (! xmlStrEqual(localName, BAD_CAST "xmlns")) {
2403
653
            prefix = BAD_CAST localName;
2404
653
        }
2405
658
        ns = reader->node->nsDef;
2406
1.37k
        while (ns != NULL) {
2407
724
            if ((prefix == NULL && ns->prefix == NULL) ||
2408
724
                ((ns->prefix != NULL) && (xmlStrEqual(ns->prefix, localName)))) {
2409
5
                return readerStrdup(reader, ns->href);
2410
5
            }
2411
719
            ns = ns->next;
2412
719
        }
2413
653
        return NULL;
2414
658
    }
2415
2416
6.82k
    result = xmlNodeGetAttrValue(reader->node, localName, namespaceURI, &ret);
2417
6.82k
    if (result < 0)
2418
6
        xmlTextReaderErrMemory(reader);
2419
2420
6.82k
    return(ret);
2421
7.48k
}
2422
2423
/**
2424
 * Method to get the remainder of the buffered XML. this method stops the
2425
 * parser, set its state to End Of File and return the input stream with
2426
 * what is left that the parser did not use.
2427
 *
2428
 * The implementation is not good, the parser certainly progressed past
2429
 * what's left in reader->input, and there is an allocation problem. Best
2430
 * would be to rewrite it differently.
2431
 *
2432
 * @param reader  the xmlTextReader used
2433
 * @returns the xmlParserInputBuffer attached to the XML or NULL
2434
 *    in case of error.
2435
 */
2436
xmlParserInputBuffer *
2437
3.75k
xmlTextReaderGetRemainder(xmlTextReader *reader) {
2438
3.75k
    xmlParserInputBufferPtr ret = NULL;
2439
2440
3.75k
    if (reader == NULL)
2441
0
  return(NULL);
2442
3.75k
    if (reader->node == NULL)
2443
2.80k
  return(NULL);
2444
2445
954
    reader->node = NULL;
2446
954
    reader->curnode = NULL;
2447
954
    reader->mode = XML_TEXTREADER_MODE_EOF;
2448
954
    if (reader->ctxt != NULL) {
2449
954
  xmlStopParser(reader->ctxt);
2450
954
  if (reader->ctxt->myDoc != NULL) {
2451
954
      if (reader->preserve == 0)
2452
0
    xmlTextReaderFreeDoc(reader, reader->ctxt->myDoc);
2453
954
      reader->ctxt->myDoc = NULL;
2454
954
  }
2455
954
    }
2456
954
    if (reader->allocs & XML_TEXTREADER_INPUT) {
2457
954
  ret = reader->input;
2458
954
  reader->input = NULL;
2459
954
  reader->allocs -= XML_TEXTREADER_INPUT;
2460
954
    } else {
2461
  /*
2462
   * Hum, one may need to duplicate the data structure because
2463
   * without reference counting the input may be freed twice:
2464
   *   - by the layer which allocated it.
2465
   *   - by the layer to which would have been returned to.
2466
   */
2467
0
  return(NULL);
2468
0
    }
2469
954
    return(ret);
2470
954
}
2471
2472
/**
2473
 * Resolves a namespace prefix in the scope of the current element.
2474
 *
2475
 * @param reader  the xmlTextReader used
2476
 * @param prefix  the prefix whose namespace URI is to be resolved. To return
2477
 *          the default namespace, specify NULL
2478
 * @returns a string containing the namespace URI to which the prefix maps
2479
 *    or NULL in case of error. The string must be deallocated by the caller.
2480
 */
2481
xmlChar *
2482
8.96k
xmlTextReaderLookupNamespace(xmlTextReader *reader, const xmlChar *prefix) {
2483
8.96k
    xmlNsPtr ns;
2484
8.96k
    int result;
2485
2486
8.96k
    if (reader == NULL)
2487
0
  return(NULL);
2488
8.96k
    if (reader->node == NULL)
2489
3.02k
  return(NULL);
2490
2491
5.93k
    result = xmlSearchNsSafe(reader->node, prefix, &ns);
2492
5.93k
    if (result < 0) {
2493
0
        xmlTextReaderErrMemory(reader);
2494
0
        return(NULL);
2495
0
    }
2496
5.93k
    if (ns == NULL)
2497
5.06k
  return(NULL);
2498
868
    return(readerStrdup(reader, ns->href));
2499
5.93k
}
2500
2501
/**
2502
 * Moves the position of the current instance to the attribute with
2503
 * the specified index relative to the containing element.
2504
 *
2505
 * @param reader  the xmlTextReader used
2506
 * @param no  the zero-based index of the attribute relative to the containing
2507
 *      element.
2508
 * @returns 1 in case of success, -1 in case of error, 0 if not found
2509
 */
2510
int
2511
8.39k
xmlTextReaderMoveToAttributeNo(xmlTextReader *reader, int no) {
2512
8.39k
    int i;
2513
8.39k
    xmlAttrPtr cur;
2514
8.39k
    xmlNsPtr ns;
2515
2516
8.39k
    if (reader == NULL)
2517
0
  return(-1);
2518
8.39k
    if (reader->node == NULL)
2519
3.30k
  return(-1);
2520
    /* TODO: handle the xmlDecl */
2521
5.09k
    if (reader->node->type != XML_ELEMENT_NODE)
2522
1.40k
  return(-1);
2523
2524
3.68k
    reader->curnode = NULL;
2525
2526
3.68k
    ns = reader->node->nsDef;
2527
5.82k
    for (i = 0;(i < no) && (ns != NULL);i++) {
2528
2.13k
  ns = ns->next;
2529
2.13k
    }
2530
3.68k
    if (ns != NULL) {
2531
552
  reader->curnode = (xmlNodePtr) ns;
2532
552
  return(1);
2533
552
    }
2534
2535
3.13k
    cur = reader->node->properties;
2536
3.13k
    if (cur == NULL)
2537
1.26k
  return(0);
2538
10.9k
    for (;i < no;i++) {
2539
10.4k
  cur = cur->next;
2540
10.4k
  if (cur == NULL)
2541
1.30k
      return(0);
2542
10.4k
    }
2543
    /* TODO walk the DTD if present */
2544
2545
559
    reader->curnode = (xmlNodePtr) cur;
2546
559
    return(1);
2547
1.86k
}
2548
2549
/**
2550
 * Moves the position of the current instance to the attribute with
2551
 * the specified qualified name.
2552
 *
2553
 * @param reader  the xmlTextReader used
2554
 * @param name  the qualified name of the attribute.
2555
 * @returns 1 in case of success, -1 in case of error, 0 if not found
2556
 */
2557
int
2558
12.5k
xmlTextReaderMoveToAttribute(xmlTextReader *reader, const xmlChar *name) {
2559
12.5k
    xmlChar *prefix = NULL;
2560
12.5k
    const xmlChar *localname;
2561
12.5k
    xmlNsPtr ns;
2562
12.5k
    xmlAttrPtr prop;
2563
2564
12.5k
    if ((reader == NULL) || (name == NULL))
2565
4.40k
  return(-1);
2566
8.13k
    if (reader->node == NULL)
2567
0
  return(-1);
2568
2569
    /* TODO: handle the xmlDecl */
2570
8.13k
    if (reader->node->type != XML_ELEMENT_NODE)
2571
1.44k
  return(0);
2572
2573
6.69k
    localname = xmlSplitQName4(name, &prefix);
2574
6.69k
    if (localname == NULL) {
2575
5
        xmlTextReaderErrMemory(reader);
2576
5
        return(-1);
2577
5
    }
2578
6.69k
    if (prefix == NULL) {
2579
  /*
2580
   * Namespace default decl
2581
   */
2582
4.59k
  if (xmlStrEqual(name, BAD_CAST "xmlns")) {
2583
1.00k
      ns = reader->node->nsDef;
2584
1.32k
      while (ns != NULL) {
2585
1.00k
    if (ns->prefix == NULL) {
2586
686
        reader->curnode = (xmlNodePtr) ns;
2587
686
        return(1);
2588
686
    }
2589
322
    ns = ns->next;
2590
322
      }
2591
319
      return(0);
2592
1.00k
  }
2593
2594
3.59k
  prop = reader->node->properties;
2595
6.88k
  while (prop != NULL) {
2596
      /*
2597
       * One need to have
2598
       *   - same attribute names
2599
       *   - and the attribute carrying that namespace
2600
       */
2601
3.88k
      if ((xmlStrEqual(prop->name, name)) &&
2602
3.88k
    ((prop->ns == NULL) || (prop->ns->prefix == NULL))) {
2603
592
    reader->curnode = (xmlNodePtr) prop;
2604
592
    return(1);
2605
592
      }
2606
3.29k
      prop = prop->next;
2607
3.29k
  }
2608
3.00k
  return(0);
2609
3.59k
    }
2610
2611
    /*
2612
     * Namespace default decl
2613
     */
2614
2.09k
    if (xmlStrEqual(prefix, BAD_CAST "xmlns")) {
2615
841
  ns = reader->node->nsDef;
2616
1.31k
  while (ns != NULL) {
2617
855
      if ((ns->prefix != NULL) && (xmlStrEqual(ns->prefix, localname))) {
2618
382
    reader->curnode = (xmlNodePtr) ns;
2619
382
    goto found;
2620
382
      }
2621
473
      ns = ns->next;
2622
473
  }
2623
459
  goto not_found;
2624
841
    }
2625
1.25k
    prop = reader->node->properties;
2626
2.33k
    while (prop != NULL) {
2627
  /*
2628
   * One need to have
2629
   *   - same attribute names
2630
   *   - and the attribute carrying that namespace
2631
   */
2632
1.32k
  if ((xmlStrEqual(prop->name, localname)) &&
2633
1.32k
      (prop->ns != NULL) && (xmlStrEqual(prop->ns->prefix, prefix))) {
2634
244
      reader->curnode = (xmlNodePtr) prop;
2635
244
      goto found;
2636
244
  }
2637
1.07k
  prop = prop->next;
2638
1.07k
    }
2639
1.46k
not_found:
2640
1.46k
    if (prefix != NULL)
2641
1.46k
        xmlFree(prefix);
2642
1.46k
    return(0);
2643
2644
626
found:
2645
626
    if (prefix != NULL)
2646
626
        xmlFree(prefix);
2647
626
    return(1);
2648
1.25k
}
2649
2650
/**
2651
 * Moves the position of the current instance to the attribute with the
2652
 * specified local name and namespace URI.
2653
 *
2654
 * @param reader  the xmlTextReader used
2655
 * @param localName  the local name of the attribute.
2656
 * @param namespaceURI  the namespace URI of the attribute.
2657
 * @returns 1 in case of success, -1 in case of error, 0 if not found
2658
 */
2659
int
2660
xmlTextReaderMoveToAttributeNs(xmlTextReader *reader,
2661
10.4k
  const xmlChar *localName, const xmlChar *namespaceURI) {
2662
10.4k
    xmlAttrPtr prop;
2663
10.4k
    xmlNodePtr node;
2664
10.4k
    xmlNsPtr ns;
2665
10.4k
    xmlChar *prefix = NULL;
2666
2667
10.4k
    if ((reader == NULL) || (localName == NULL) || (namespaceURI == NULL))
2668
7.06k
  return(-1);
2669
3.42k
    if (reader->node == NULL)
2670
0
  return(-1);
2671
3.42k
    if (reader->node->type != XML_ELEMENT_NODE)
2672
0
  return(0);
2673
3.42k
    node = reader->node;
2674
2675
3.42k
    if (xmlStrEqual(namespaceURI, BAD_CAST "http://www.w3.org/2000/xmlns/")) {
2676
2.01k
    if (! xmlStrEqual(localName, BAD_CAST "xmlns")) {
2677
1.15k
      prefix = BAD_CAST localName;
2678
1.15k
    }
2679
2.01k
    ns = reader->node->nsDef;
2680
2.69k
    while (ns != NULL) {
2681
2.30k
      if ((prefix == NULL && ns->prefix == NULL) ||
2682
2.30k
        ((ns->prefix != NULL) && (xmlStrEqual(ns->prefix, localName)))) {
2683
1.62k
        reader->curnode = (xmlNodePtr) ns;
2684
1.62k
        return(1);
2685
1.62k
      }
2686
676
      ns = ns->next;
2687
676
    }
2688
394
    return(0);
2689
2.01k
    }
2690
2691
1.40k
    prop = node->properties;
2692
10.8k
    while (prop != NULL) {
2693
  /*
2694
   * One need to have
2695
   *   - same attribute names
2696
   *   - and the attribute carrying that namespace
2697
   */
2698
9.81k
        if (xmlStrEqual(prop->name, localName) &&
2699
9.81k
      ((prop->ns != NULL) &&
2700
918
       (xmlStrEqual(prop->ns->href, namespaceURI)))) {
2701
348
      reader->curnode = (xmlNodePtr) prop;
2702
348
      return(1);
2703
348
        }
2704
9.46k
  prop = prop->next;
2705
9.46k
    }
2706
1.05k
    return(0);
2707
1.40k
}
2708
2709
/**
2710
 * Moves the position of the current instance to the first attribute
2711
 * associated with the current node.
2712
 *
2713
 * @param reader  the xmlTextReader used
2714
 * @returns 1 in case of success, -1 in case of error, 0 if not found
2715
 */
2716
int
2717
15.0k
xmlTextReaderMoveToFirstAttribute(xmlTextReader *reader) {
2718
15.0k
    if (reader == NULL)
2719
0
  return(-1);
2720
15.0k
    if (reader->node == NULL)
2721
4.56k
  return(-1);
2722
10.4k
    if (reader->node->type != XML_ELEMENT_NODE)
2723
2.62k
  return(0);
2724
2725
7.85k
    if (reader->node->nsDef != NULL) {
2726
3.17k
  reader->curnode = (xmlNodePtr) reader->node->nsDef;
2727
3.17k
  return(1);
2728
3.17k
    }
2729
4.68k
    if (reader->node->properties != NULL) {
2730
1.63k
  reader->curnode = (xmlNodePtr) reader->node->properties;
2731
1.63k
  return(1);
2732
1.63k
    }
2733
3.04k
    return(0);
2734
4.68k
}
2735
2736
/**
2737
 * Moves the position of the current instance to the next attribute
2738
 * associated with the current node.
2739
 *
2740
 * @param reader  the xmlTextReader used
2741
 * @returns 1 in case of success, -1 in case of error, 0 if not found
2742
 */
2743
int
2744
8.90k
xmlTextReaderMoveToNextAttribute(xmlTextReader *reader) {
2745
8.90k
    if (reader == NULL)
2746
0
  return(-1);
2747
8.90k
    if (reader->node == NULL)
2748
2.24k
  return(-1);
2749
6.66k
    if (reader->node->type != XML_ELEMENT_NODE)
2750
1.72k
  return(0);
2751
4.93k
    if (reader->curnode == NULL)
2752
2.48k
  return(xmlTextReaderMoveToFirstAttribute(reader));
2753
2754
2.45k
    if (reader->curnode->type == XML_NAMESPACE_DECL) {
2755
1.04k
  xmlNsPtr ns = (xmlNsPtr) reader->curnode;
2756
1.04k
  if (ns->next != NULL) {
2757
234
      reader->curnode = (xmlNodePtr) ns->next;
2758
234
      return(1);
2759
234
  }
2760
807
  if (reader->node->properties != NULL) {
2761
275
      reader->curnode = (xmlNodePtr) reader->node->properties;
2762
275
      return(1);
2763
275
  }
2764
532
  return(0);
2765
1.41k
    } else if ((reader->curnode->type == XML_ATTRIBUTE_NODE) &&
2766
1.41k
         (reader->curnode->next != NULL)) {
2767
284
  reader->curnode = reader->curnode->next;
2768
284
  return(1);
2769
284
    }
2770
1.12k
    return(0);
2771
2.45k
}
2772
2773
/**
2774
 * Moves the position of the current instance to the node that
2775
 * contains the current Attribute  node.
2776
 *
2777
 * @param reader  the xmlTextReader used
2778
 * @returns 1 in case of success, -1 in case of error, 0 if not moved
2779
 */
2780
int
2781
5.32k
xmlTextReaderMoveToElement(xmlTextReader *reader) {
2782
5.32k
    if (reader == NULL)
2783
0
  return(-1);
2784
5.32k
    if (reader->node == NULL)
2785
2.38k
  return(-1);
2786
2.94k
    if (reader->node->type != XML_ELEMENT_NODE)
2787
1.19k
  return(0);
2788
1.74k
    if (reader->curnode != NULL) {
2789
423
  reader->curnode = NULL;
2790
423
  return(1);
2791
423
    }
2792
1.32k
    return(0);
2793
1.74k
}
2794
2795
/**
2796
 * Parses an attribute value into one or more Text and EntityReference nodes.
2797
 *
2798
 * @param reader  the xmlTextReader used
2799
 * @returns 1 in case of success, 0 if the reader was not positioned on an
2800
 *         attribute node or all the attribute values have been read, or -1
2801
 *         in case of error.
2802
 */
2803
int
2804
10.3k
xmlTextReaderReadAttributeValue(xmlTextReader *reader) {
2805
10.3k
    if (reader == NULL)
2806
0
  return(-1);
2807
10.3k
    if (reader->node == NULL)
2808
3.60k
  return(-1);
2809
6.72k
    if (reader->curnode == NULL)
2810
4.65k
  return(0);
2811
2.06k
    if (reader->curnode->type == XML_ATTRIBUTE_NODE) {
2812
558
  if (reader->curnode->children == NULL)
2813
222
      return(0);
2814
336
  reader->curnode = reader->curnode->children;
2815
1.51k
    } else if (reader->curnode->type == XML_NAMESPACE_DECL) {
2816
721
  xmlNsPtr ns = (xmlNsPtr) reader->curnode;
2817
2818
721
  if (reader->faketext == NULL) {
2819
154
      reader->faketext = xmlNewDocText(reader->node->doc,
2820
154
                                 ns->href);
2821
154
            if (reader->faketext == NULL) {
2822
3
                xmlTextReaderErrMemory(reader);
2823
3
                return(-1);
2824
3
            }
2825
567
  } else {
2826
567
            if ((reader->faketext->content != NULL) &&
2827
567
          (reader->faketext->content !=
2828
231
     (xmlChar *) &(reader->faketext->properties)))
2829
231
    xmlFree(reader->faketext->content);
2830
567
            if (ns->href == NULL) {
2831
335
                reader->faketext->content = NULL;
2832
335
            } else {
2833
232
                reader->faketext->content = xmlStrdup(ns->href);
2834
232
                if (reader->faketext->content == NULL) {
2835
1
                    xmlTextReaderErrMemory(reader);
2836
1
                    return(-1);
2837
1
                }
2838
232
            }
2839
567
  }
2840
717
  reader->curnode = reader->faketext;
2841
789
    } else {
2842
789
  if (reader->curnode->next == NULL)
2843
587
      return(0);
2844
202
  reader->curnode = reader->curnode->next;
2845
202
    }
2846
1.25k
    return(1);
2847
2.06k
}
2848
2849
/**
2850
 * Determine the encoding of the document being read.
2851
 *
2852
 * @param reader  the xmlTextReader used
2853
 * @returns a string containing the encoding of the document or NULL in
2854
 * case of error.  The string is deallocated with the reader.
2855
 */
2856
const xmlChar *
2857
3.06k
xmlTextReaderConstEncoding(xmlTextReader *reader) {
2858
3.06k
    const xmlChar *encoding = NULL;
2859
2860
3.06k
    if (reader == NULL)
2861
0
        return(NULL);
2862
2863
3.06k
    if (reader->ctxt != NULL)
2864
3.06k
        encoding = xmlGetActualEncoding(reader->ctxt);
2865
0
    else if (reader->doc != NULL)
2866
0
        encoding = reader->doc->encoding;
2867
2868
3.06k
    return(constString(reader, encoding));
2869
3.06k
}
2870
2871
2872
/************************************************************************
2873
 *                  *
2874
 *      Access API to the current node      *
2875
 *                  *
2876
 ************************************************************************/
2877
/**
2878
 * Provides the number of attributes of the current node
2879
 *
2880
 * @param reader  the xmlTextReader used
2881
 * @returns 0 i no attributes, -1 in case of error or the attribute count
2882
 */
2883
int
2884
9.69k
xmlTextReaderAttributeCount(xmlTextReader *reader) {
2885
9.69k
    int ret;
2886
9.69k
    xmlAttrPtr attr;
2887
9.69k
    xmlNsPtr ns;
2888
9.69k
    xmlNodePtr node;
2889
2890
9.69k
    if (reader == NULL)
2891
0
  return(-1);
2892
9.69k
    if (reader->node == NULL)
2893
4.29k
  return(0);
2894
2895
5.39k
    if (reader->curnode != NULL)
2896
627
  node = reader->curnode;
2897
4.76k
    else
2898
4.76k
  node = reader->node;
2899
2900
5.39k
    if (node->type != XML_ELEMENT_NODE)
2901
1.70k
  return(0);
2902
3.68k
    if ((reader->state == XML_TEXTREADER_END) ||
2903
3.68k
  (reader->state == XML_TEXTREADER_BACKTRACK))
2904
537
  return(0);
2905
3.14k
    ret = 0;
2906
3.14k
    attr = node->properties;
2907
8.86k
    while (attr != NULL) {
2908
5.71k
  ret++;
2909
5.71k
  attr = attr->next;
2910
5.71k
    }
2911
3.14k
    ns = node->nsDef;
2912
5.29k
    while (ns != NULL) {
2913
2.14k
  ret++;
2914
2.14k
  ns = ns->next;
2915
2.14k
    }
2916
3.14k
    return(ret);
2917
3.68k
}
2918
2919
/**
2920
 * Get the node type of the current node
2921
 * Reference:
2922
 * http://www.gnu.org/software/dotgnu/pnetlib-doc/System/Xml/XmlNodeType.html
2923
 *
2924
 * @param reader  the xmlTextReader used
2925
 * @returns the xmlReaderTypes of the current node or -1 in case of error
2926
 */
2927
int
2928
14.8k
xmlTextReaderNodeType(xmlTextReader *reader) {
2929
14.8k
    xmlNodePtr node;
2930
2931
14.8k
    if (reader == NULL)
2932
0
  return(-1);
2933
14.8k
    if (reader->node == NULL)
2934
3.54k
  return(XML_READER_TYPE_NONE);
2935
11.2k
    if (reader->curnode != NULL)
2936
784
  node = reader->curnode;
2937
10.4k
    else
2938
10.4k
  node = reader->node;
2939
11.2k
    switch (node->type) {
2940
3.87k
        case XML_ELEMENT_NODE:
2941
3.87k
      if ((reader->state == XML_TEXTREADER_END) ||
2942
3.87k
    (reader->state == XML_TEXTREADER_BACKTRACK))
2943
1.32k
    return(XML_READER_TYPE_END_ELEMENT);
2944
2.54k
      return(XML_READER_TYPE_ELEMENT);
2945
276
        case XML_NAMESPACE_DECL:
2946
552
        case XML_ATTRIBUTE_NODE:
2947
552
      return(XML_READER_TYPE_ATTRIBUTE);
2948
3.64k
        case XML_TEXT_NODE:
2949
3.64k
      if (xmlIsBlankNode(reader->node)) {
2950
1.21k
    if (xmlNodeGetSpacePreserve(reader->node))
2951
1.21k
        return(XML_READER_TYPE_SIGNIFICANT_WHITESPACE);
2952
0
    else
2953
0
        return(XML_READER_TYPE_WHITESPACE);
2954
2.43k
      } else {
2955
2.43k
    return(XML_READER_TYPE_TEXT);
2956
2.43k
      }
2957
228
        case XML_CDATA_SECTION_NODE:
2958
228
      return(XML_READER_TYPE_CDATA);
2959
225
        case XML_ENTITY_REF_NODE:
2960
225
      return(XML_READER_TYPE_ENTITY_REFERENCE);
2961
0
        case XML_ENTITY_NODE:
2962
0
      return(XML_READER_TYPE_ENTITY);
2963
243
        case XML_PI_NODE:
2964
243
      return(XML_READER_TYPE_PROCESSING_INSTRUCTION);
2965
310
        case XML_COMMENT_NODE:
2966
310
      return(XML_READER_TYPE_COMMENT);
2967
703
        case XML_DOCUMENT_NODE:
2968
703
        case XML_HTML_DOCUMENT_NODE:
2969
703
      return(XML_READER_TYPE_DOCUMENT);
2970
0
        case XML_DOCUMENT_FRAG_NODE:
2971
0
      return(XML_READER_TYPE_DOCUMENT_FRAGMENT);
2972
0
        case XML_NOTATION_NODE:
2973
0
      return(XML_READER_TYPE_NOTATION);
2974
0
        case XML_DOCUMENT_TYPE_NODE:
2975
338
        case XML_DTD_NODE:
2976
338
      return(XML_READER_TYPE_DOCUMENT_TYPE);
2977
2978
0
        case XML_ELEMENT_DECL:
2979
0
        case XML_ATTRIBUTE_DECL:
2980
0
        case XML_ENTITY_DECL:
2981
1.16k
        case XML_XINCLUDE_START:
2982
1.16k
        case XML_XINCLUDE_END:
2983
1.16k
      return(XML_READER_TYPE_NONE);
2984
11.2k
    }
2985
0
    return(-1);
2986
11.2k
}
2987
2988
/**
2989
 * Check if the current node is empty
2990
 *
2991
 * @param reader  the xmlTextReader used
2992
 * @returns 1 if empty, 0 if not and -1 in case of error
2993
 */
2994
int
2995
13.2k
xmlTextReaderIsEmptyElement(xmlTextReader *reader) {
2996
13.2k
    if ((reader == NULL) || (reader->node == NULL))
2997
2.58k
  return(-1);
2998
10.6k
    if (reader->node->type != XML_ELEMENT_NODE)
2999
4.79k
  return(0);
3000
5.86k
    if (reader->curnode != NULL)
3001
534
  return(0);
3002
5.32k
    if (reader->node->children != NULL)
3003
4.31k
  return(0);
3004
1.01k
    if (reader->state == XML_TEXTREADER_END)
3005
243
  return(0);
3006
771
    if (reader->doc != NULL)
3007
0
        return(1);
3008
771
#ifdef LIBXML_XINCLUDE_ENABLED
3009
771
    if (reader->in_xinclude > 0)
3010
238
        return(1);
3011
533
#endif
3012
533
    return((reader->node->extra & NODE_IS_EMPTY) != 0);
3013
771
}
3014
3015
/**
3016
 * The local name of the node.
3017
 *
3018
 * @param reader  the xmlTextReader used
3019
 * @returns the local name or NULL if not available,
3020
 *   if non NULL it need to be freed by the caller.
3021
 */
3022
xmlChar *
3023
5.39k
xmlTextReaderLocalName(xmlTextReader *reader) {
3024
5.39k
    xmlNodePtr node;
3025
5.39k
    if ((reader == NULL) || (reader->node == NULL))
3026
1.69k
  return(NULL);
3027
3.70k
    if (reader->curnode != NULL)
3028
834
  node = reader->curnode;
3029
2.87k
    else
3030
2.87k
  node = reader->node;
3031
3.70k
    if (node->type == XML_NAMESPACE_DECL) {
3032
432
  xmlNsPtr ns = (xmlNsPtr) node;
3033
432
  if (ns->prefix == NULL)
3034
201
      return(readerStrdup(reader, BAD_CAST "xmlns"));
3035
231
  else
3036
231
      return(readerStrdup(reader, ns->prefix));
3037
432
    }
3038
3.27k
    if ((node->type != XML_ELEMENT_NODE) &&
3039
3.27k
  (node->type != XML_ATTRIBUTE_NODE))
3040
1.41k
  return(xmlTextReaderName(reader));
3041
1.85k
    return(readerStrdup(reader, node->name));
3042
3.27k
}
3043
3044
/**
3045
 * The local name of the node.
3046
 *
3047
 * @param reader  the xmlTextReader used
3048
 * @returns the local name or NULL if not available, the
3049
 *         string will be deallocated with the reader.
3050
 */
3051
const xmlChar *
3052
38.4k
xmlTextReaderConstLocalName(xmlTextReader *reader) {
3053
38.4k
    xmlNodePtr node;
3054
38.4k
    if ((reader == NULL) || (reader->node == NULL))
3055
13.0k
  return(NULL);
3056
25.4k
    if (reader->curnode != NULL)
3057
4.77k
  node = reader->curnode;
3058
20.6k
    else
3059
20.6k
  node = reader->node;
3060
25.4k
    if (node->type == XML_NAMESPACE_DECL) {
3061
2.86k
  xmlNsPtr ns = (xmlNsPtr) node;
3062
2.86k
  if (ns->prefix == NULL)
3063
817
      return(constString(reader, BAD_CAST "xmlns"));
3064
2.05k
  else
3065
2.05k
      return(ns->prefix);
3066
2.86k
    }
3067
22.5k
    if ((node->type != XML_ELEMENT_NODE) &&
3068
22.5k
  (node->type != XML_ATTRIBUTE_NODE))
3069
9.74k
  return(xmlTextReaderConstName(reader));
3070
12.8k
    return(node->name);
3071
22.5k
}
3072
3073
/**
3074
 * The qualified name of the node, equal to Prefix :LocalName.
3075
 *
3076
 * @param reader  the xmlTextReader used
3077
 * @returns the local name or NULL if not available,
3078
 *   if non NULL it need to be freed by the caller.
3079
 */
3080
xmlChar *
3081
8.74k
xmlTextReaderName(xmlTextReader *reader) {
3082
8.74k
    xmlNodePtr node;
3083
8.74k
    xmlChar *ret;
3084
3085
8.74k
    if ((reader == NULL) || (reader->node == NULL))
3086
1.99k
  return(NULL);
3087
6.75k
    if (reader->curnode != NULL)
3088
1.01k
  node = reader->curnode;
3089
5.73k
    else
3090
5.73k
  node = reader->node;
3091
6.75k
    switch (node->type) {
3092
1.79k
        case XML_ELEMENT_NODE:
3093
2.07k
        case XML_ATTRIBUTE_NODE:
3094
2.07k
      if ((node->ns == NULL) ||
3095
2.07k
    (node->ns->prefix == NULL))
3096
1.78k
    return(readerStrdup(reader, node->name));
3097
3098
289
            ret = xmlBuildQName(node->name, node->ns->prefix, NULL, 0);
3099
289
            if (ret == NULL)
3100
1
                xmlTextReaderErrMemory(reader);
3101
289
      return(ret);
3102
1.28k
        case XML_TEXT_NODE:
3103
1.28k
      return(readerStrdup(reader, BAD_CAST "#text"));
3104
210
        case XML_CDATA_SECTION_NODE:
3105
210
      return(readerStrdup(reader, BAD_CAST "#cdata-section"));
3106
0
        case XML_ENTITY_NODE:
3107
237
        case XML_ENTITY_REF_NODE:
3108
237
      return(readerStrdup(reader, node->name));
3109
776
        case XML_PI_NODE:
3110
776
      return(readerStrdup(reader, node->name));
3111
264
        case XML_COMMENT_NODE:
3112
264
      return(readerStrdup(reader, BAD_CAST "#comment"));
3113
499
        case XML_DOCUMENT_NODE:
3114
499
        case XML_HTML_DOCUMENT_NODE:
3115
499
      return(readerStrdup(reader, BAD_CAST "#document"));
3116
0
        case XML_DOCUMENT_FRAG_NODE:
3117
0
      return(readerStrdup(reader, BAD_CAST "#document-fragment"));
3118
0
        case XML_NOTATION_NODE:
3119
0
      return(readerStrdup(reader, node->name));
3120
0
        case XML_DOCUMENT_TYPE_NODE:
3121
429
        case XML_DTD_NODE:
3122
429
      return(readerStrdup(reader, node->name));
3123
439
        case XML_NAMESPACE_DECL: {
3124
439
      xmlNsPtr ns = (xmlNsPtr) node;
3125
3126
439
      if (ns->prefix == NULL)
3127
208
    return(readerStrdup(reader, BAD_CAST "xmlns"));
3128
231
            ret = xmlBuildQName(ns->prefix, BAD_CAST "xmlns", NULL, 0);
3129
231
            if (ret == NULL)
3130
1
                xmlTextReaderErrMemory(reader);
3131
231
      return(ret);
3132
439
  }
3133
3134
0
        case XML_ELEMENT_DECL:
3135
0
        case XML_ATTRIBUTE_DECL:
3136
0
        case XML_ENTITY_DECL:
3137
551
        case XML_XINCLUDE_START:
3138
551
        case XML_XINCLUDE_END:
3139
551
      return(NULL);
3140
6.75k
    }
3141
0
    return(NULL);
3142
6.75k
}
3143
3144
/**
3145
 * The qualified name of the node, equal to Prefix :LocalName.
3146
 *
3147
 * @param reader  the xmlTextReader used
3148
 * @returns the local name or NULL if not available, the string is
3149
 *         deallocated with the reader.
3150
 */
3151
const xmlChar *
3152
62.5k
xmlTextReaderConstName(xmlTextReader *reader) {
3153
62.5k
    xmlNodePtr node;
3154
3155
62.5k
    if ((reader == NULL) || (reader->node == NULL))
3156
21.6k
  return(NULL);
3157
40.9k
    if (reader->curnode != NULL)
3158
4.22k
  node = reader->curnode;
3159
36.7k
    else
3160
36.7k
  node = reader->node;
3161
40.9k
    switch (node->type) {
3162
16.5k
        case XML_ELEMENT_NODE:
3163
18.0k
        case XML_ATTRIBUTE_NODE:
3164
18.0k
      if ((node->ns == NULL) ||
3165
18.0k
    (node->ns->prefix == NULL))
3166
15.7k
    return(node->name);
3167
2.33k
      return(constQString(reader, node->ns->prefix, node->name));
3168
8.90k
        case XML_TEXT_NODE:
3169
8.90k
      return(constString(reader, BAD_CAST "#text"));
3170
635
        case XML_CDATA_SECTION_NODE:
3171
635
      return(constString(reader, BAD_CAST "#cdata-section"));
3172
0
        case XML_ENTITY_NODE:
3173
472
        case XML_ENTITY_REF_NODE:
3174
472
      return(constString(reader, node->name));
3175
2.19k
        case XML_PI_NODE:
3176
2.19k
      return(constString(reader, node->name));
3177
1.04k
        case XML_COMMENT_NODE:
3178
1.04k
      return(constString(reader, BAD_CAST "#comment"));
3179
2.97k
        case XML_DOCUMENT_NODE:
3180
2.97k
        case XML_HTML_DOCUMENT_NODE:
3181
2.97k
      return(constString(reader, BAD_CAST "#document"));
3182
0
        case XML_DOCUMENT_FRAG_NODE:
3183
0
      return(constString(reader, BAD_CAST "#document-fragment"));
3184
0
        case XML_NOTATION_NODE:
3185
0
      return(constString(reader, node->name));
3186
0
        case XML_DOCUMENT_TYPE_NODE:
3187
1.81k
        case XML_DTD_NODE:
3188
1.81k
      return(constString(reader, node->name));
3189
1.96k
        case XML_NAMESPACE_DECL: {
3190
1.96k
      xmlNsPtr ns = (xmlNsPtr) node;
3191
3192
1.96k
      if (ns->prefix == NULL)
3193
887
    return(constString(reader, BAD_CAST "xmlns"));
3194
1.08k
      return(constQString(reader, BAD_CAST "xmlns", ns->prefix));
3195
1.96k
  }
3196
3197
0
        case XML_ELEMENT_DECL:
3198
0
        case XML_ATTRIBUTE_DECL:
3199
0
        case XML_ENTITY_DECL:
3200
2.89k
        case XML_XINCLUDE_START:
3201
2.89k
        case XML_XINCLUDE_END:
3202
2.89k
      return(NULL);
3203
40.9k
    }
3204
0
    return(NULL);
3205
40.9k
}
3206
3207
/**
3208
 * A shorthand reference to the namespace associated with the node.
3209
 *
3210
 * @param reader  the xmlTextReader used
3211
 * @returns the prefix or NULL if not available,
3212
 *    if non NULL it need to be freed by the caller.
3213
 */
3214
xmlChar *
3215
3.95k
xmlTextReaderPrefix(xmlTextReader *reader) {
3216
3.95k
    xmlNodePtr node;
3217
3.95k
    if ((reader == NULL) || (reader->node == NULL))
3218
1.16k
  return(NULL);
3219
2.79k
    if (reader->curnode != NULL)
3220
750
  node = reader->curnode;
3221
2.04k
    else
3222
2.04k
  node = reader->node;
3223
2.79k
    if (node->type == XML_NAMESPACE_DECL) {
3224
441
  xmlNsPtr ns = (xmlNsPtr) node;
3225
441
  if (ns->prefix == NULL)
3226
202
      return(NULL);
3227
239
  return(readerStrdup(reader, BAD_CAST "xmlns"));
3228
441
    }
3229
2.35k
    if ((node->type != XML_ELEMENT_NODE) &&
3230
2.35k
  (node->type != XML_ATTRIBUTE_NODE))
3231
670
  return(NULL);
3232
1.68k
    if ((node->ns != NULL) && (node->ns->prefix != NULL))
3233
305
  return(readerStrdup(reader, node->ns->prefix));
3234
1.37k
    return(NULL);
3235
1.68k
}
3236
3237
/**
3238
 * A shorthand reference to the namespace associated with the node.
3239
 *
3240
 * @param reader  the xmlTextReader used
3241
 * @returns the prefix or NULL if not available, the string is deallocated
3242
 *         with the reader.
3243
 */
3244
const xmlChar *
3245
23.7k
xmlTextReaderConstPrefix(xmlTextReader *reader) {
3246
23.7k
    xmlNodePtr node;
3247
23.7k
    if ((reader == NULL) || (reader->node == NULL))
3248
9.09k
  return(NULL);
3249
14.6k
    if (reader->curnode != NULL)
3250
2.19k
  node = reader->curnode;
3251
12.5k
    else
3252
12.5k
  node = reader->node;
3253
14.6k
    if (node->type == XML_NAMESPACE_DECL) {
3254
1.12k
  xmlNsPtr ns = (xmlNsPtr) node;
3255
1.12k
  if (ns->prefix == NULL)
3256
291
      return(NULL);
3257
835
  return(constString(reader, BAD_CAST "xmlns"));
3258
1.12k
    }
3259
13.5k
    if ((node->type != XML_ELEMENT_NODE) &&
3260
13.5k
  (node->type != XML_ATTRIBUTE_NODE))
3261
6.82k
  return(NULL);
3262
6.74k
    if ((node->ns != NULL) && (node->ns->prefix != NULL))
3263
1.12k
  return(constString(reader, node->ns->prefix));
3264
5.61k
    return(NULL);
3265
6.74k
}
3266
3267
/**
3268
 * The URI defining the namespace associated with the node.
3269
 *
3270
 * @param reader  the xmlTextReader used
3271
 * @returns the namespace URI or NULL if not available,
3272
 *    if non NULL it need to be freed by the caller.
3273
 */
3274
xmlChar *
3275
4.29k
xmlTextReaderNamespaceUri(xmlTextReader *reader) {
3276
4.29k
    xmlNodePtr node;
3277
4.29k
    if ((reader == NULL) || (reader->node == NULL))
3278
2.68k
  return(NULL);
3279
1.61k
    if (reader->curnode != NULL)
3280
467
  node = reader->curnode;
3281
1.14k
    else
3282
1.14k
  node = reader->node;
3283
1.61k
    if (node->type == XML_NAMESPACE_DECL)
3284
198
  return(readerStrdup(reader, BAD_CAST "http://www.w3.org/2000/xmlns/"));
3285
1.41k
    if ((node->type != XML_ELEMENT_NODE) &&
3286
1.41k
  (node->type != XML_ATTRIBUTE_NODE))
3287
453
  return(NULL);
3288
963
    if (node->ns != NULL)
3289
247
  return(readerStrdup(reader, node->ns->href));
3290
716
    return(NULL);
3291
963
}
3292
3293
/**
3294
 * The URI defining the namespace associated with the node.
3295
 *
3296
 * @param reader  the xmlTextReader used
3297
 * @returns the namespace URI or NULL if not available, the string
3298
 *         will be deallocated with the reader
3299
 */
3300
const xmlChar *
3301
41.3k
xmlTextReaderConstNamespaceUri(xmlTextReader *reader) {
3302
41.3k
    xmlNodePtr node;
3303
41.3k
    if ((reader == NULL) || (reader->node == NULL))
3304
15.2k
  return(NULL);
3305
26.1k
    if (reader->curnode != NULL)
3306
4.86k
  node = reader->curnode;
3307
21.2k
    else
3308
21.2k
  node = reader->node;
3309
26.1k
    if (node->type == XML_NAMESPACE_DECL)
3310
2.82k
  return(constString(reader, BAD_CAST "http://www.w3.org/2000/xmlns/"));
3311
23.2k
    if ((node->type != XML_ELEMENT_NODE) &&
3312
23.2k
  (node->type != XML_ATTRIBUTE_NODE))
3313
9.43k
  return(NULL);
3314
13.8k
    if (node->ns != NULL)
3315
4.65k
  return(constString(reader, node->ns->href));
3316
9.20k
    return(NULL);
3317
13.8k
}
3318
3319
/**
3320
 * The base URI of the node.
3321
 *
3322
 * @param reader  the xmlTextReader used
3323
 * @returns the base URI or NULL if not available,
3324
 *    if non NULL it need to be freed by the caller.
3325
 */
3326
xmlChar *
3327
9.51k
xmlTextReaderBaseUri(xmlTextReader *reader) {
3328
9.51k
    xmlChar *ret = NULL;
3329
9.51k
    int result;
3330
3331
9.51k
    if ((reader == NULL) || (reader->node == NULL))
3332
1.69k
  return(NULL);
3333
7.82k
    result = xmlNodeGetBaseSafe(NULL, reader->node, &ret);
3334
7.82k
    if (result < 0)
3335
156
        xmlTextReaderErrMemory(reader);
3336
3337
7.82k
    return(ret);
3338
9.51k
}
3339
3340
/**
3341
 * The base URI of the node.
3342
 *
3343
 * @param reader  the xmlTextReader used
3344
 * @returns the base URI or NULL if not available, the string
3345
 *         will be deallocated with the reader
3346
 */
3347
const xmlChar *
3348
43.8k
xmlTextReaderConstBaseUri(xmlTextReader *reader) {
3349
43.8k
    xmlChar *tmp;
3350
43.8k
    const xmlChar *ret;
3351
43.8k
    int result;
3352
3353
43.8k
    if ((reader == NULL) || (reader->node == NULL))
3354
5.21k
  return(NULL);
3355
38.6k
    result = xmlNodeGetBaseSafe(NULL, reader->node, &tmp);
3356
38.6k
    if (result < 0)
3357
672
        xmlTextReaderErrMemory(reader);
3358
38.6k
    if (tmp == NULL)
3359
10.3k
        return(NULL);
3360
28.3k
    ret = constString(reader, tmp);
3361
28.3k
    xmlFree(tmp);
3362
28.3k
    return(ret);
3363
38.6k
}
3364
3365
/**
3366
 * The depth of the node in the tree.
3367
 *
3368
 * @param reader  the xmlTextReader used
3369
 * @returns the depth or -1 in case of error
3370
 */
3371
int
3372
5.36k
xmlTextReaderDepth(xmlTextReader *reader) {
3373
5.36k
    if (reader == NULL)
3374
0
  return(-1);
3375
5.36k
    if (reader->node == NULL)
3376
992
  return(0);
3377
3378
4.37k
    if (reader->curnode != NULL) {
3379
717
  if ((reader->curnode->type == XML_ATTRIBUTE_NODE) ||
3380
717
      (reader->curnode->type == XML_NAMESPACE_DECL))
3381
512
      return(reader->depth + 1);
3382
205
  return(reader->depth + 2);
3383
717
    }
3384
3.65k
    return(reader->depth);
3385
4.37k
}
3386
3387
/**
3388
 * Whether the node has attributes.
3389
 *
3390
 * @param reader  the xmlTextReader used
3391
 * @returns 1 if true, 0 if false, and -1 in case or error
3392
 */
3393
int
3394
3.26k
xmlTextReaderHasAttributes(xmlTextReader *reader) {
3395
3.26k
    xmlNodePtr node;
3396
3.26k
    if (reader == NULL)
3397
0
  return(-1);
3398
3.26k
    if (reader->node == NULL)
3399
1.29k
  return(0);
3400
1.96k
    if (reader->curnode != NULL)
3401
303
  node = reader->curnode;
3402
1.65k
    else
3403
1.65k
  node = reader->node;
3404
3405
1.96k
    if ((node->type == XML_ELEMENT_NODE) &&
3406
1.96k
  ((node->properties != NULL) || (node->nsDef != NULL)))
3407
605
  return(1);
3408
    /* TODO: handle the xmlDecl */
3409
1.35k
    return(0);
3410
1.96k
}
3411
3412
/**
3413
 * Whether the node can have a text value.
3414
 *
3415
 * @param reader  the xmlTextReader used
3416
 * @returns 1 if true, 0 if false, and -1 in case or error
3417
 */
3418
int
3419
15.4k
xmlTextReaderHasValue(xmlTextReader *reader) {
3420
15.4k
    xmlNodePtr node;
3421
15.4k
    if (reader == NULL)
3422
0
  return(-1);
3423
15.4k
    if (reader->node == NULL)
3424
6.06k
  return(0);
3425
9.37k
    if (reader->curnode != NULL)
3426
1.25k
  node = reader->curnode;
3427
8.12k
    else
3428
8.12k
  node = reader->node;
3429
3430
9.37k
    switch (node->type) {
3431
474
        case XML_ATTRIBUTE_NODE:
3432
2.47k
        case XML_TEXT_NODE:
3433
2.70k
        case XML_CDATA_SECTION_NODE:
3434
3.01k
        case XML_PI_NODE:
3435
3.32k
        case XML_COMMENT_NODE:
3436
3.75k
        case XML_NAMESPACE_DECL:
3437
3.75k
      return(1);
3438
5.62k
  default:
3439
5.62k
      break;
3440
9.37k
    }
3441
5.62k
    return(0);
3442
9.37k
}
3443
3444
/**
3445
 * Provides the text value of the node if present
3446
 *
3447
 * @param reader  the xmlTextReader used
3448
 * @returns the string or NULL if not available. The result must be deallocated
3449
 *     with #xmlFree
3450
 */
3451
xmlChar *
3452
11.4k
xmlTextReaderValue(xmlTextReader *reader) {
3453
11.4k
    xmlNodePtr node;
3454
11.4k
    if (reader == NULL)
3455
0
  return(NULL);
3456
11.4k
    if (reader->node == NULL)
3457
2.73k
  return(NULL);
3458
8.69k
    if (reader->curnode != NULL)
3459
2.09k
  node = reader->curnode;
3460
6.60k
    else
3461
6.60k
  node = reader->node;
3462
3463
8.69k
    switch (node->type) {
3464
509
        case XML_NAMESPACE_DECL:
3465
509
      return(readerStrdup(reader, ((xmlNsPtr) node)->href));
3466
1.03k
        case XML_ATTRIBUTE_NODE:{
3467
1.03k
      xmlAttrPtr attr = (xmlAttrPtr) node;
3468
1.03k
            xmlDocPtr doc = NULL;
3469
1.03k
            xmlChar *ret;
3470
3471
1.03k
            if (attr->children == NULL)
3472
209
                return(NULL);
3473
826
      if (attr->parent != NULL)
3474
826
                doc = attr->parent->doc;
3475
826
      ret = xmlNodeListGetString(doc, attr->children, 1);
3476
826
            if (ret == NULL)
3477
7
                xmlTextReaderErrMemory(reader);
3478
826
      return(ret);
3479
1.03k
  }
3480
1.52k
        case XML_TEXT_NODE:
3481
1.73k
        case XML_CDATA_SECTION_NODE:
3482
2.14k
        case XML_PI_NODE:
3483
2.38k
        case XML_COMMENT_NODE:
3484
2.38k
            return(readerStrdup(reader, node->content));
3485
4.76k
  default:
3486
4.76k
      break;
3487
8.69k
    }
3488
4.76k
    return(NULL);
3489
8.69k
}
3490
3491
/**
3492
 * Provides the text value of the node if present
3493
 *
3494
 * @param reader  the xmlTextReader used
3495
 * @returns the string or NULL if not available. The result will be
3496
 *     deallocated on the next Read() operation.
3497
 */
3498
const xmlChar *
3499
7.28k
xmlTextReaderConstValue(xmlTextReader *reader) {
3500
7.28k
    xmlNodePtr node;
3501
7.28k
    if (reader == NULL)
3502
0
  return(NULL);
3503
7.28k
    if (reader->node == NULL)
3504
1.35k
  return(NULL);
3505
5.93k
    if (reader->curnode != NULL)
3506
1.31k
  node = reader->curnode;
3507
4.62k
    else
3508
4.62k
  node = reader->node;
3509
3510
5.93k
    switch (node->type) {
3511
264
        case XML_NAMESPACE_DECL:
3512
264
      return(((xmlNsPtr) node)->href);
3513
1.01k
        case XML_ATTRIBUTE_NODE:{
3514
1.01k
      xmlAttrPtr attr = (xmlAttrPtr) node;
3515
1.01k
      const xmlChar *ret;
3516
3517
1.01k
      if ((attr->children != NULL) &&
3518
1.01k
          (attr->children->type == XML_TEXT_NODE) &&
3519
1.01k
    (attr->children->next == NULL))
3520
242
    return(attr->children->content);
3521
774
      else {
3522
774
    if (reader->buffer == NULL) {
3523
0
        reader->buffer = xmlBufCreate(50);
3524
0
                    if (reader->buffer == NULL)
3525
0
                        return (NULL);
3526
0
                } else
3527
774
                    xmlBufEmpty(reader->buffer);
3528
774
          xmlBufGetNodeContent(reader->buffer, node);
3529
774
    ret = xmlBufContent(reader->buffer);
3530
774
    if (ret == NULL) {
3531
1
                    xmlTextReaderErrMemory(reader);
3532
        /* error on the buffer best to reallocate */
3533
1
        xmlBufFree(reader->buffer);
3534
1
        reader->buffer = xmlBufCreate(50);
3535
1
    }
3536
774
    return(ret);
3537
774
      }
3538
0
      break;
3539
1.01k
  }
3540
1.10k
        case XML_TEXT_NODE:
3541
1.30k
        case XML_CDATA_SECTION_NODE:
3542
1.53k
        case XML_PI_NODE:
3543
1.75k
        case XML_COMMENT_NODE:
3544
1.75k
      return(node->content);
3545
2.89k
  default:
3546
2.89k
      break;
3547
5.93k
    }
3548
2.89k
    return(NULL);
3549
5.93k
}
3550
3551
/**
3552
 * Whether an Attribute  node was generated from the default value
3553
 * defined in the DTD or schema.
3554
 *
3555
 * @param reader  the xmlTextReader used
3556
 * @returns 0 if not defaulted, 1 if defaulted, and -1 in case of error
3557
 */
3558
int
3559
5.70k
xmlTextReaderIsDefault(xmlTextReader *reader) {
3560
5.70k
    if (reader == NULL)
3561
0
  return(-1);
3562
5.70k
    return(0);
3563
5.70k
}
3564
3565
/**
3566
 * The quotation mark character used to enclose the value of an attribute.
3567
 *
3568
 * @param reader  the xmlTextReader used
3569
 * @returns " or ' and -1 in case of error
3570
 */
3571
int
3572
3.35k
xmlTextReaderQuoteChar(xmlTextReader *reader) {
3573
3.35k
    if (reader == NULL)
3574
0
  return(-1);
3575
    /* TODO maybe lookup the attribute value for " first */
3576
3.35k
    return('"');
3577
3.35k
}
3578
3579
/**
3580
 * The xml:lang scope within which the node resides.
3581
 *
3582
 * @param reader  the xmlTextReader used
3583
 * @returns the xml:lang value or NULL if none exists.,
3584
 *    if non NULL it need to be freed by the caller.
3585
 */
3586
xmlChar *
3587
3.73k
xmlTextReaderXmlLang(xmlTextReader *reader) {
3588
3.73k
    if (reader == NULL)
3589
0
  return(NULL);
3590
3.73k
    if (reader->node == NULL)
3591
1.28k
  return(NULL);
3592
2.44k
    return(xmlNodeGetLang(reader->node));
3593
3.73k
}
3594
3595
/**
3596
 * The xml:lang scope within which the node resides.
3597
 *
3598
 * @param reader  the xmlTextReader used
3599
 * @returns the xml:lang value or NULL if none exists.
3600
 */
3601
const xmlChar *
3602
6.63k
xmlTextReaderConstXmlLang(xmlTextReader *reader) {
3603
6.63k
    xmlChar *tmp;
3604
6.63k
    const xmlChar *ret;
3605
3606
6.63k
    if (reader == NULL)
3607
0
  return(NULL);
3608
6.63k
    if (reader->node == NULL)
3609
2.61k
  return(NULL);
3610
4.01k
    tmp = xmlNodeGetLang(reader->node);
3611
4.01k
    if (tmp == NULL)
3612
3.71k
        return(NULL);
3613
299
    ret = constString(reader, tmp);
3614
299
    xmlFree(tmp);
3615
299
    return(ret);
3616
4.01k
}
3617
3618
/**
3619
 * Get an interned string from the reader, allows for example to
3620
 * speedup string name comparisons
3621
 *
3622
 * @param reader  the xmlTextReader used
3623
 * @param str  the string to intern.
3624
 * @returns an interned copy of the string or NULL in case of error. The
3625
 *         string will be deallocated with the reader.
3626
 */
3627
const xmlChar *
3628
0
xmlTextReaderConstString(xmlTextReader *reader, const xmlChar *str) {
3629
0
    if (reader == NULL)
3630
0
  return(NULL);
3631
0
    return(constString(reader, str));
3632
0
}
3633
3634
/**
3635
 * The value indicating whether to normalize white space and attribute values.
3636
 * Since attribute value and end of line normalizations are a MUST in the XML
3637
 * specification only the value true is accepted. The broken behaviour of
3638
 * accepting out of range character entities like &\#0; is of course not
3639
 * supported either.
3640
 *
3641
 * @param reader  the xmlTextReader used
3642
 * @returns 1 or -1 in case of error.
3643
 */
3644
int
3645
1.90k
xmlTextReaderNormalization(xmlTextReader *reader) {
3646
1.90k
    if (reader == NULL)
3647
0
  return(-1);
3648
1.90k
    return(1);
3649
1.90k
}
3650
3651
/************************************************************************
3652
 *                  *
3653
 *      Extensions to the base APIs     *
3654
 *                  *
3655
 ************************************************************************/
3656
3657
/**
3658
 * Change the parser processing behaviour by changing some of its internal
3659
 * properties. Note that some properties can only be changed before any
3660
 * read has been done.
3661
 *
3662
 * @param reader  the xmlTextReader used
3663
 * @param prop  the xmlParserProperties to set
3664
 * @param value  usually 0 or 1 to (de)activate it
3665
 * @returns 0 if the call was successful, or -1 in case of error
3666
 */
3667
int
3668
0
xmlTextReaderSetParserProp(xmlTextReader *reader, int prop, int value) {
3669
0
    xmlParserProperties p = (xmlParserProperties) prop;
3670
0
    xmlParserCtxtPtr ctxt;
3671
3672
0
    if ((reader == NULL) || (reader->ctxt == NULL))
3673
0
  return(-1);
3674
0
    ctxt = reader->ctxt;
3675
3676
0
    switch (p) {
3677
0
        case XML_PARSER_LOADDTD:
3678
0
      if (value != 0) {
3679
0
    if (ctxt->loadsubset == 0) {
3680
0
        if (reader->mode != XML_TEXTREADER_MODE_INITIAL)
3681
0
      return(-1);
3682
0
                    ctxt->options |= XML_PARSE_DTDLOAD;
3683
0
        ctxt->loadsubset |= XML_DETECT_IDS;
3684
0
    }
3685
0
      } else {
3686
0
                ctxt->options &= ~XML_PARSE_DTDLOAD;
3687
0
    ctxt->loadsubset &= ~XML_DETECT_IDS;
3688
0
      }
3689
0
      return(0);
3690
0
        case XML_PARSER_DEFAULTATTRS:
3691
0
      if (value != 0) {
3692
0
                ctxt->options |= XML_PARSE_DTDATTR;
3693
0
    ctxt->loadsubset |= XML_COMPLETE_ATTRS;
3694
0
      } else {
3695
0
                ctxt->options &= ~XML_PARSE_DTDATTR;
3696
0
    ctxt->loadsubset &= ~XML_COMPLETE_ATTRS;
3697
0
      }
3698
0
      return(0);
3699
0
        case XML_PARSER_VALIDATE:
3700
0
      if (value != 0) {
3701
0
                ctxt->options |= XML_PARSE_DTDVALID;
3702
0
    ctxt->validate = 1;
3703
0
    reader->validate = XML_TEXTREADER_VALIDATE_DTD;
3704
0
      } else {
3705
0
                ctxt->options &= ~XML_PARSE_DTDVALID;
3706
0
    ctxt->validate = 0;
3707
0
      }
3708
0
      return(0);
3709
0
        case XML_PARSER_SUBST_ENTITIES:
3710
0
      if (value != 0) {
3711
0
                ctxt->options |= XML_PARSE_NOENT;
3712
0
    ctxt->replaceEntities = 1;
3713
0
      } else {
3714
0
                ctxt->options &= ~XML_PARSE_NOENT;
3715
0
    ctxt->replaceEntities = 0;
3716
0
      }
3717
0
      return(0);
3718
0
    }
3719
0
    return(-1);
3720
0
}
3721
3722
/**
3723
 * Read the parser internal property.
3724
 *
3725
 * @param reader  the xmlTextReader used
3726
 * @param prop  the xmlParserProperties to get
3727
 * @returns the value, usually 0 or 1, or -1 in case of error.
3728
 */
3729
int
3730
7.66k
xmlTextReaderGetParserProp(xmlTextReader *reader, int prop) {
3731
7.66k
    xmlParserProperties p = (xmlParserProperties) prop;
3732
7.66k
    xmlParserCtxtPtr ctxt;
3733
3734
7.66k
    if ((reader == NULL) || (reader->ctxt == NULL))
3735
0
  return(-1);
3736
7.66k
    ctxt = reader->ctxt;
3737
3738
7.66k
    switch (p) {
3739
736
        case XML_PARSER_LOADDTD:
3740
736
      if ((ctxt->loadsubset != 0) || (ctxt->validate != 0))
3741
509
    return(1);
3742
227
      return(0);
3743
413
        case XML_PARSER_DEFAULTATTRS:
3744
413
      if (ctxt->loadsubset & XML_COMPLETE_ATTRS)
3745
210
    return(1);
3746
203
      return(0);
3747
256
        case XML_PARSER_VALIDATE:
3748
256
      return(reader->validate);
3749
213
  case XML_PARSER_SUBST_ENTITIES:
3750
213
      return(ctxt->replaceEntities);
3751
7.66k
    }
3752
6.04k
    return(-1);
3753
7.66k
}
3754
3755
3756
/**
3757
 * Provide the line number of the current parsing point.
3758
 *
3759
 * @param reader  the user data (XML reader context)
3760
 * @returns an int or 0 if not available
3761
 */
3762
int
3763
xmlTextReaderGetParserLineNumber(xmlTextReader *reader)
3764
6.67k
{
3765
6.67k
    if ((reader == NULL) || (reader->ctxt == NULL) ||
3766
6.67k
        (reader->ctxt->input == NULL)) {
3767
0
        return (0);
3768
0
    }
3769
6.67k
    return (reader->ctxt->input->line);
3770
6.67k
}
3771
3772
/**
3773
 * Provide the column number of the current parsing point.
3774
 *
3775
 * @param reader  the user data (XML reader context)
3776
 * @returns an int or 0 if not available
3777
 */
3778
int
3779
xmlTextReaderGetParserColumnNumber(xmlTextReader *reader)
3780
8.50k
{
3781
8.50k
    if ((reader == NULL) || (reader->ctxt == NULL) ||
3782
8.50k
        (reader->ctxt->input == NULL)) {
3783
0
        return (0);
3784
0
    }
3785
8.50k
    return (reader->ctxt->input->col);
3786
8.50k
}
3787
3788
/**
3789
 * Hacking interface allowing to get the xmlNode corresponding to the
3790
 * current node being accessed by the xmlTextReader. This is dangerous
3791
 * because the underlying node may be destroyed on the next Reads.
3792
 *
3793
 * @param reader  the xmlTextReader used
3794
 * @returns the xmlNode or NULL in case of error.
3795
 */
3796
xmlNode *
3797
12.3k
xmlTextReaderCurrentNode(xmlTextReader *reader) {
3798
12.3k
    if (reader == NULL)
3799
0
  return(NULL);
3800
3801
12.3k
    if (reader->curnode != NULL)
3802
662
  return(reader->curnode);
3803
11.6k
    return(reader->node);
3804
12.3k
}
3805
3806
/**
3807
 * This tells the XML Reader to preserve the current node.
3808
 * The caller must also use #xmlTextReaderCurrentDoc to
3809
 * keep an handle on the resulting document once parsing has finished
3810
 *
3811
 * @param reader  the xmlTextReader used
3812
 * @returns the xmlNode or NULL in case of error.
3813
 */
3814
xmlNode *
3815
9.21k
xmlTextReaderPreserve(xmlTextReader *reader) {
3816
9.21k
    xmlNodePtr cur, parent;
3817
3818
9.21k
    if (reader == NULL)
3819
0
  return(NULL);
3820
3821
9.21k
    cur = reader->node;
3822
9.21k
    if (cur == NULL)
3823
4.26k
        return(NULL);
3824
3825
4.95k
    if ((cur->type != XML_DOCUMENT_NODE) && (cur->type != XML_DTD_NODE)) {
3826
4.12k
  cur->extra |= NODE_IS_PRESERVED;
3827
4.12k
  cur->extra |= NODE_IS_SPRESERVED;
3828
4.12k
    }
3829
4.95k
    reader->preserves++;
3830
3831
4.95k
    parent = cur->parent;;
3832
571k
    while (parent != NULL) {
3833
566k
        if (parent->type == XML_ELEMENT_NODE)
3834
561k
      parent->extra |= NODE_IS_PRESERVED;
3835
566k
  parent = parent->parent;
3836
566k
    }
3837
4.95k
    return(cur);
3838
9.21k
}
3839
3840
#ifdef LIBXML_PATTERN_ENABLED
3841
/**
3842
 * This tells the XML Reader to preserve all nodes matched by the
3843
 * pattern. The caller must also use #xmlTextReaderCurrentDoc to
3844
 * keep an handle on the resulting document once parsing has finished
3845
 *
3846
 * @param reader  the xmlTextReader used
3847
 * @param pattern  an XPath subset pattern
3848
 * @param namespaces  the prefix definitions, array of [URI, prefix] or NULL
3849
 * @returns a non-negative number in case of success and -1 in case of error
3850
 */
3851
int
3852
xmlTextReaderPreservePattern(xmlTextReader *reader, const xmlChar *pattern,
3853
                             const xmlChar **namespaces)
3854
0
{
3855
0
    xmlPatternPtr comp;
3856
3857
0
    if ((reader == NULL) || (pattern == NULL))
3858
0
  return(-1);
3859
3860
0
    comp = xmlPatterncompile(pattern, reader->dict, 0, namespaces);
3861
0
    if (comp == NULL)
3862
0
        return(-1);
3863
3864
0
    if (reader->patternNr >= reader->patternMax) {
3865
0
        xmlPatternPtr *tmp;
3866
0
        int newSize;
3867
3868
0
        newSize = xmlGrowCapacity(reader->patternMax, sizeof(tmp[0]),
3869
0
                                  4, XML_MAX_ITEMS);
3870
0
        if (newSize < 0) {
3871
0
            xmlTextReaderErrMemory(reader);
3872
0
            return(-1);
3873
0
        }
3874
0
  tmp = xmlRealloc(reader->patternTab, newSize * sizeof(tmp[0]));
3875
0
        if (tmp == NULL) {
3876
0
            xmlTextReaderErrMemory(reader);
3877
0
            return(-1);
3878
0
        }
3879
0
  reader->patternTab = tmp;
3880
0
        reader->patternMax = newSize;
3881
0
    }
3882
0
    reader->patternTab[reader->patternNr] = comp;
3883
0
    return(reader->patternNr++);
3884
0
}
3885
#endif
3886
3887
/**
3888
 * Hacking interface allowing to get the xmlDoc corresponding to the
3889
 * current document being accessed by the xmlTextReader.
3890
 * NOTE: as a result of this call, the reader will not destroy the
3891
 *       associated XML document and calling #xmlFreeDoc on the result
3892
 *       is needed once the reader parsing has finished.
3893
 *
3894
 * @param reader  the xmlTextReader used
3895
 * @returns the xmlDoc or NULL in case of error.
3896
 */
3897
xmlDoc *
3898
10.0k
xmlTextReaderCurrentDoc(xmlTextReader *reader) {
3899
10.0k
    if (reader == NULL)
3900
0
  return(NULL);
3901
10.0k
    if (reader->doc != NULL)
3902
0
        return(reader->doc);
3903
10.0k
    if ((reader->ctxt == NULL) || (reader->ctxt->myDoc == NULL))
3904
1.69k
  return(NULL);
3905
3906
8.40k
    reader->preserve = 1;
3907
8.40k
    return(reader->ctxt->myDoc);
3908
10.0k
}
3909
3910
#ifdef LIBXML_RELAXNG_ENABLED
3911
/**
3912
 * Use RelaxNG to validate the document as it is processed.
3913
 * Activation is only possible before the first Read().
3914
 * if `schema` is NULL, then RelaxNG validation is deactivated.
3915
 * The `schema` should not be freed until the reader is deallocated
3916
 * or its use has been deactivated.
3917
 *
3918
 * @param reader  the xmlTextReader used
3919
 * @param schema  a precompiled RelaxNG schema
3920
 * @returns 0 in case the RelaxNG validation could be (de)activated and
3921
 *         -1 in case of error.
3922
 */
3923
int
3924
0
xmlTextReaderRelaxNGSetSchema(xmlTextReader *reader, xmlRelaxNG *schema) {
3925
0
    if (reader == NULL)
3926
0
        return(-1);
3927
0
    if (schema == NULL) {
3928
0
        if (reader->rngSchemas != NULL) {
3929
0
      xmlRelaxNGFree(reader->rngSchemas);
3930
0
      reader->rngSchemas = NULL;
3931
0
  }
3932
0
        if (reader->rngValidCtxt != NULL) {
3933
0
      if (! reader->rngPreserveCtxt)
3934
0
    xmlRelaxNGFreeValidCtxt(reader->rngValidCtxt);
3935
0
      reader->rngValidCtxt = NULL;
3936
0
        }
3937
0
  reader->rngPreserveCtxt = 0;
3938
0
  return(0);
3939
0
    }
3940
0
    if (reader->mode != XML_TEXTREADER_MODE_INITIAL)
3941
0
  return(-1);
3942
0
    if (reader->rngSchemas != NULL) {
3943
0
  xmlRelaxNGFree(reader->rngSchemas);
3944
0
  reader->rngSchemas = NULL;
3945
0
    }
3946
0
    if (reader->rngValidCtxt != NULL) {
3947
0
  if (! reader->rngPreserveCtxt)
3948
0
      xmlRelaxNGFreeValidCtxt(reader->rngValidCtxt);
3949
0
  reader->rngValidCtxt = NULL;
3950
0
    }
3951
0
    reader->rngPreserveCtxt = 0;
3952
0
    reader->rngValidCtxt = xmlRelaxNGNewValidCtxt(schema);
3953
0
    if (reader->rngValidCtxt == NULL)
3954
0
        return(-1);
3955
0
    if ((reader->errorFunc != NULL) || (reader->sErrorFunc != NULL))
3956
0
  xmlRelaxNGSetValidStructuredErrors(reader->rngValidCtxt,
3957
0
      xmlTextReaderStructuredRelay, reader);
3958
0
    reader->rngValidErrors = 0;
3959
0
    reader->rngFullNode = NULL;
3960
0
    reader->validate = XML_TEXTREADER_VALIDATE_RNG;
3961
0
    return(0);
3962
0
}
3963
#endif /* LIBXML_RELAXNG_ENABLED */
3964
3965
#ifdef LIBXML_SCHEMAS_ENABLED
3966
/**
3967
 * Internal locator function for the readers
3968
 *
3969
 * @param ctx  the xmlTextReader used
3970
 * @param file  returned file information
3971
 * @param line  returned line information
3972
 * @returns 0 in case the Schema validation could be (de)activated and
3973
 *         -1 in case of error.
3974
 */
3975
static int
3976
0
xmlTextReaderLocator(void *ctx, const char **file, unsigned long *line) {
3977
0
    xmlTextReaderPtr reader;
3978
3979
0
    if ((ctx == NULL) || ((file == NULL) && (line == NULL)))
3980
0
        return(-1);
3981
3982
0
    if (file != NULL)
3983
0
        *file = NULL;
3984
0
    if (line != NULL)
3985
0
        *line = 0;
3986
3987
0
    reader = (xmlTextReaderPtr) ctx;
3988
0
    if ((reader->ctxt != NULL) && (reader->ctxt->input != NULL)) {
3989
0
  if (file != NULL)
3990
0
      *file = reader->ctxt->input->filename;
3991
0
  if (line != NULL)
3992
0
      *line = reader->ctxt->input->line;
3993
0
  return(0);
3994
0
    }
3995
0
    if (reader->node != NULL) {
3996
0
        long res;
3997
0
  int ret = 0;
3998
3999
0
  if (line != NULL) {
4000
0
      res = xmlGetLineNo(reader->node);
4001
0
      if (res > 0)
4002
0
          *line = (unsigned long) res;
4003
0
      else
4004
0
                ret = -1;
4005
0
  }
4006
0
        if (file != NULL) {
4007
0
      xmlDocPtr doc = reader->node->doc;
4008
0
      if ((doc != NULL) && (doc->URL != NULL))
4009
0
          *file = (const char *) doc->URL;
4010
0
      else
4011
0
                ret = -1;
4012
0
  }
4013
0
  return(ret);
4014
0
    }
4015
0
    return(-1);
4016
0
}
4017
4018
/**
4019
 * Use XSD Schema to validate the document as it is processed.
4020
 * Activation is only possible before the first Read().
4021
 * if `schema` is NULL, then Schema validation is deactivated.
4022
 * The `schema` should not be freed until the reader is deallocated
4023
 * or its use has been deactivated.
4024
 *
4025
 * @param reader  the xmlTextReader used
4026
 * @param schema  a precompiled Schema schema
4027
 * @returns 0 in case the Schema validation could be (de)activated and
4028
 *         -1 in case of error.
4029
 */
4030
int
4031
0
xmlTextReaderSetSchema(xmlTextReader *reader, xmlSchema *schema) {
4032
0
    if (reader == NULL)
4033
0
        return(-1);
4034
0
    if (schema == NULL) {
4035
0
  if (reader->xsdPlug != NULL) {
4036
0
      xmlSchemaSAXUnplug(reader->xsdPlug);
4037
0
      reader->xsdPlug = NULL;
4038
0
  }
4039
0
        if (reader->xsdValidCtxt != NULL) {
4040
0
      if (! reader->xsdPreserveCtxt)
4041
0
    xmlSchemaFreeValidCtxt(reader->xsdValidCtxt);
4042
0
      reader->xsdValidCtxt = NULL;
4043
0
        }
4044
0
  reader->xsdPreserveCtxt = 0;
4045
0
        if (reader->xsdSchemas != NULL) {
4046
0
      xmlSchemaFree(reader->xsdSchemas);
4047
0
      reader->xsdSchemas = NULL;
4048
0
  }
4049
0
  return(0);
4050
0
    }
4051
0
    if (reader->mode != XML_TEXTREADER_MODE_INITIAL)
4052
0
  return(-1);
4053
0
    if (reader->xsdPlug != NULL) {
4054
0
  xmlSchemaSAXUnplug(reader->xsdPlug);
4055
0
  reader->xsdPlug = NULL;
4056
0
    }
4057
0
    if (reader->xsdValidCtxt != NULL) {
4058
0
  if (! reader->xsdPreserveCtxt)
4059
0
      xmlSchemaFreeValidCtxt(reader->xsdValidCtxt);
4060
0
  reader->xsdValidCtxt = NULL;
4061
0
    }
4062
0
    reader->xsdPreserveCtxt = 0;
4063
0
    if (reader->xsdSchemas != NULL) {
4064
0
  xmlSchemaFree(reader->xsdSchemas);
4065
0
  reader->xsdSchemas = NULL;
4066
0
    }
4067
0
    reader->xsdValidCtxt = xmlSchemaNewValidCtxt(schema);
4068
0
    if (reader->xsdValidCtxt == NULL) {
4069
0
  xmlSchemaFree(reader->xsdSchemas);
4070
0
  reader->xsdSchemas = NULL;
4071
0
        return(-1);
4072
0
    }
4073
0
    reader->xsdPlug = xmlSchemaSAXPlug(reader->xsdValidCtxt,
4074
0
                                       &(reader->ctxt->sax),
4075
0
               &(reader->ctxt->userData));
4076
0
    if (reader->xsdPlug == NULL) {
4077
0
  xmlSchemaFree(reader->xsdSchemas);
4078
0
  reader->xsdSchemas = NULL;
4079
0
  xmlSchemaFreeValidCtxt(reader->xsdValidCtxt);
4080
0
  reader->xsdValidCtxt = NULL;
4081
0
  return(-1);
4082
0
    }
4083
0
    xmlSchemaValidateSetLocator(reader->xsdValidCtxt,
4084
0
                                xmlTextReaderLocator,
4085
0
        (void *) reader);
4086
4087
0
    if ((reader->errorFunc != NULL) || (reader->sErrorFunc != NULL))
4088
0
  xmlSchemaSetValidStructuredErrors(reader->xsdValidCtxt,
4089
0
      xmlTextReaderStructuredRelay, reader);
4090
0
    reader->xsdValidErrors = 0;
4091
0
    reader->validate = XML_TEXTREADER_VALIDATE_XSD;
4092
0
    return(0);
4093
0
}
4094
#endif /* LIBXML_SCHEMAS_ENABLED */
4095
4096
#ifdef LIBXML_RELAXNG_ENABLED
4097
/**
4098
 * Use RelaxNG to validate the document as it is processed.
4099
 * Activation is only possible before the first Read().
4100
 * If both `rng` and `ctxt` are NULL, then RelaxNG validation is deactivated.
4101
 *
4102
 * @param reader  the xmlTextReader used
4103
 * @param rng  the path to a RelaxNG schema or NULL
4104
 * @param ctxt  the RelaxNG schema validation context or NULL
4105
 * @param options  options (not yet used)
4106
 * @returns 0 in case the RelaxNG validation could be (de)activated and
4107
 *     -1 in case of error.
4108
 */
4109
static int
4110
xmlTextReaderRelaxNGValidateInternal(xmlTextReaderPtr reader,
4111
             const char *rng,
4112
             xmlRelaxNGValidCtxtPtr ctxt,
4113
             int options ATTRIBUTE_UNUSED)
4114
0
{
4115
0
    if (reader == NULL)
4116
0
  return(-1);
4117
4118
0
    if ((rng != NULL) && (ctxt != NULL))
4119
0
  return (-1);
4120
4121
0
    if (((rng != NULL) || (ctxt != NULL)) &&
4122
0
  ((reader->mode != XML_TEXTREADER_MODE_INITIAL) ||
4123
0
   (reader->ctxt == NULL)))
4124
0
  return(-1);
4125
4126
    /* Cleanup previous validation stuff. */
4127
0
    if (reader->rngValidCtxt != NULL) {
4128
0
  if ( !reader->rngPreserveCtxt)
4129
0
      xmlRelaxNGFreeValidCtxt(reader->rngValidCtxt);
4130
0
  reader->rngValidCtxt = NULL;
4131
0
    }
4132
0
    reader->rngPreserveCtxt = 0;
4133
0
    if (reader->rngSchemas != NULL) {
4134
0
  xmlRelaxNGFree(reader->rngSchemas);
4135
0
  reader->rngSchemas = NULL;
4136
0
    }
4137
4138
0
    if ((rng == NULL) && (ctxt == NULL)) {
4139
  /* We just want to deactivate the validation, so get out. */
4140
0
  return(0);
4141
0
    }
4142
4143
4144
0
    if (rng != NULL) {
4145
0
  xmlRelaxNGParserCtxtPtr pctxt;
4146
  /* Parse the schema and create validation environment. */
4147
4148
0
  pctxt = xmlRelaxNGNewParserCtxt(rng);
4149
0
  if ((reader->errorFunc != NULL) || (reader->sErrorFunc != NULL))
4150
0
      xmlRelaxNGSetParserStructuredErrors(pctxt,
4151
0
                    xmlTextReaderStructuredRelay, reader);
4152
0
  reader->rngSchemas = xmlRelaxNGParse(pctxt);
4153
0
  xmlRelaxNGFreeParserCtxt(pctxt);
4154
0
  if (reader->rngSchemas == NULL)
4155
0
      return(-1);
4156
4157
0
  reader->rngValidCtxt = xmlRelaxNGNewValidCtxt(reader->rngSchemas);
4158
0
  if (reader->rngValidCtxt == NULL) {
4159
0
      xmlRelaxNGFree(reader->rngSchemas);
4160
0
      reader->rngSchemas = NULL;
4161
0
      return(-1);
4162
0
  }
4163
0
    } else {
4164
  /* Use the given validation context. */
4165
0
  reader->rngValidCtxt = ctxt;
4166
0
  reader->rngPreserveCtxt = 1;
4167
0
    }
4168
    /*
4169
    * Redirect the validation context's error channels to use
4170
    * the reader channels.
4171
    * TODO: In case the user provides the validation context we
4172
    * could make this redirection optional.
4173
    */
4174
0
    if ((reader->errorFunc != NULL) || (reader->sErrorFunc != NULL))
4175
0
        xmlRelaxNGSetValidStructuredErrors(reader->rngValidCtxt,
4176
0
                xmlTextReaderStructuredRelay, reader);
4177
0
    reader->rngValidErrors = 0;
4178
0
    reader->rngFullNode = NULL;
4179
0
    reader->validate = XML_TEXTREADER_VALIDATE_RNG;
4180
0
    return(0);
4181
0
}
4182
#endif /* LIBXML_RELAXNG_ENABLED */
4183
4184
#ifdef LIBXML_SCHEMAS_ENABLED
4185
/**
4186
 * Validate the document as it is processed using XML Schema.
4187
 * Activation is only possible before the first Read().
4188
 * If both `xsd` and `ctxt` are NULL then XML Schema validation is deactivated.
4189
 *
4190
 * @param reader  the xmlTextReader used
4191
 * @param xsd  the path to a W3C XSD schema or NULL
4192
 * @param ctxt  the XML Schema validation context or NULL
4193
 * @param options  options (not used yet)
4194
 * @returns 0 in case the schemas validation could be (de)activated and
4195
 *         -1 in case of error.
4196
 */
4197
static int
4198
xmlTextReaderSchemaValidateInternal(xmlTextReaderPtr reader,
4199
            const char *xsd,
4200
            xmlSchemaValidCtxtPtr ctxt,
4201
            int options ATTRIBUTE_UNUSED)
4202
0
{
4203
0
    if (reader == NULL)
4204
0
        return(-1);
4205
4206
0
    if ((xsd != NULL) && (ctxt != NULL))
4207
0
  return(-1);
4208
4209
0
    if (((xsd != NULL) || (ctxt != NULL)) &&
4210
0
  ((reader->mode != XML_TEXTREADER_MODE_INITIAL) ||
4211
0
        (reader->ctxt == NULL)))
4212
0
  return(-1);
4213
4214
    /* Cleanup previous validation stuff. */
4215
0
    if (reader->xsdPlug != NULL) {
4216
0
  xmlSchemaSAXUnplug(reader->xsdPlug);
4217
0
  reader->xsdPlug = NULL;
4218
0
    }
4219
0
    if (reader->xsdValidCtxt != NULL) {
4220
0
  if (! reader->xsdPreserveCtxt)
4221
0
      xmlSchemaFreeValidCtxt(reader->xsdValidCtxt);
4222
0
  reader->xsdValidCtxt = NULL;
4223
0
    }
4224
0
    reader->xsdPreserveCtxt = 0;
4225
0
    if (reader->xsdSchemas != NULL) {
4226
0
  xmlSchemaFree(reader->xsdSchemas);
4227
0
  reader->xsdSchemas = NULL;
4228
0
    }
4229
4230
0
    if ((xsd == NULL) && (ctxt == NULL)) {
4231
  /* We just want to deactivate the validation, so get out. */
4232
0
  return(0);
4233
0
    }
4234
4235
0
    if (xsd != NULL) {
4236
0
  xmlSchemaParserCtxtPtr pctxt;
4237
  /* Parse the schema and create validation environment. */
4238
0
  pctxt = xmlSchemaNewParserCtxt(xsd);
4239
0
  if ((reader->errorFunc != NULL) || (reader->sErrorFunc != NULL))
4240
0
      xmlSchemaSetParserStructuredErrors(pctxt,
4241
0
                    xmlTextReaderStructuredRelay, reader);
4242
0
  reader->xsdSchemas = xmlSchemaParse(pctxt);
4243
0
  xmlSchemaFreeParserCtxt(pctxt);
4244
0
  if (reader->xsdSchemas == NULL)
4245
0
      return(-1);
4246
0
  reader->xsdValidCtxt = xmlSchemaNewValidCtxt(reader->xsdSchemas);
4247
0
  if (reader->xsdValidCtxt == NULL) {
4248
0
      xmlSchemaFree(reader->xsdSchemas);
4249
0
      reader->xsdSchemas = NULL;
4250
0
      return(-1);
4251
0
  }
4252
0
  reader->xsdPlug = xmlSchemaSAXPlug(reader->xsdValidCtxt,
4253
0
      &(reader->ctxt->sax),
4254
0
      &(reader->ctxt->userData));
4255
0
  if (reader->xsdPlug == NULL) {
4256
0
      xmlSchemaFree(reader->xsdSchemas);
4257
0
      reader->xsdSchemas = NULL;
4258
0
      xmlSchemaFreeValidCtxt(reader->xsdValidCtxt);
4259
0
      reader->xsdValidCtxt = NULL;
4260
0
      return(-1);
4261
0
  }
4262
0
    } else {
4263
  /* Use the given validation context. */
4264
0
  reader->xsdValidCtxt = ctxt;
4265
0
  reader->xsdPreserveCtxt = 1;
4266
0
  reader->xsdPlug = xmlSchemaSAXPlug(reader->xsdValidCtxt,
4267
0
      &(reader->ctxt->sax),
4268
0
      &(reader->ctxt->userData));
4269
0
  if (reader->xsdPlug == NULL) {
4270
0
      reader->xsdValidCtxt = NULL;
4271
0
      reader->xsdPreserveCtxt = 0;
4272
0
      return(-1);
4273
0
  }
4274
0
    }
4275
0
    xmlSchemaValidateSetLocator(reader->xsdValidCtxt,
4276
0
                                xmlTextReaderLocator,
4277
0
        (void *) reader);
4278
    /*
4279
    * Redirect the validation context's error channels to use
4280
    * the reader channels.
4281
    * TODO: In case the user provides the validation context we
4282
    *   could make this redirection optional.
4283
    */
4284
0
    if ((reader->errorFunc != NULL) || (reader->sErrorFunc != NULL))
4285
0
  xmlSchemaSetValidStructuredErrors(reader->xsdValidCtxt,
4286
0
      xmlTextReaderStructuredRelay, reader);
4287
0
    reader->xsdValidErrors = 0;
4288
0
    reader->validate = XML_TEXTREADER_VALIDATE_XSD;
4289
0
    return(0);
4290
0
}
4291
4292
/**
4293
 * Use W3C XSD schema context to validate the document as it is processed.
4294
 * Activation is only possible before the first Read().
4295
 * If `ctxt` is NULL, then XML Schema validation is deactivated.
4296
 *
4297
 * @param reader  the xmlTextReader used
4298
 * @param ctxt  the XML Schema validation context or NULL
4299
 * @param options  options (not used yet)
4300
 * @returns 0 in case the schemas validation could be (de)activated and
4301
 *         -1 in case of error.
4302
 */
4303
int
4304
xmlTextReaderSchemaValidateCtxt(xmlTextReader *reader,
4305
            xmlSchemaValidCtxt *ctxt,
4306
            int options)
4307
0
{
4308
0
    return(xmlTextReaderSchemaValidateInternal(reader, NULL, ctxt, options));
4309
0
}
4310
4311
/**
4312
 * Use W3C XSD schema to validate the document as it is processed.
4313
 * Activation is only possible before the first Read().
4314
 * If `xsd` is NULL, then XML Schema validation is deactivated.
4315
 *
4316
 * @param reader  the xmlTextReader used
4317
 * @param xsd  the path to a W3C XSD schema or NULL
4318
 * @returns 0 in case the schemas validation could be (de)activated and
4319
 *         -1 in case of error.
4320
 */
4321
int
4322
xmlTextReaderSchemaValidate(xmlTextReader *reader, const char *xsd)
4323
0
{
4324
0
    return(xmlTextReaderSchemaValidateInternal(reader, xsd, NULL, 0));
4325
0
}
4326
#endif /* LIBXML_SCHEMAS_ENABLED */
4327
4328
#ifdef LIBXML_RELAXNG_ENABLED
4329
/**
4330
 * Use RelaxNG schema context to validate the document as it is processed.
4331
 * Activation is only possible before the first Read().
4332
 * If `ctxt` is NULL, then RelaxNG schema validation is deactivated.
4333
 *
4334
 * @param reader  the xmlTextReader used
4335
 * @param ctxt  the RelaxNG schema validation context or NULL
4336
 * @param options  options (not used yet)
4337
 * @returns 0 in case the schemas validation could be (de)activated and
4338
 *         -1 in case of error.
4339
 */
4340
int
4341
xmlTextReaderRelaxNGValidateCtxt(xmlTextReader *reader,
4342
         xmlRelaxNGValidCtxt *ctxt,
4343
         int options)
4344
0
{
4345
0
    return(xmlTextReaderRelaxNGValidateInternal(reader, NULL, ctxt, options));
4346
0
}
4347
4348
/**
4349
 * Use RelaxNG schema to validate the document as it is processed.
4350
 * Activation is only possible before the first Read().
4351
 * If `rng` is NULL, then RelaxNG schema validation is deactivated.
4352
 *
4353
 * @param reader  the xmlTextReader used
4354
 * @param rng  the path to a RelaxNG schema or NULL
4355
 * @returns 0 in case the schemas validation could be (de)activated and
4356
 *         -1 in case of error.
4357
 */
4358
int
4359
xmlTextReaderRelaxNGValidate(xmlTextReader *reader, const char *rng)
4360
0
{
4361
0
    return(xmlTextReaderRelaxNGValidateInternal(reader, rng, NULL, 0));
4362
0
}
4363
#endif /* LIBXML_RELAXNG_ENABLED */
4364
4365
/**
4366
 * Determine whether the current node is a namespace declaration
4367
 * rather than a regular attribute.
4368
 *
4369
 * @param reader  the xmlTextReader used
4370
 * @returns 1 if the current node is a namespace declaration, 0 if it
4371
 * is a regular attribute or other type of node, or -1 in case of
4372
 * error.
4373
 */
4374
int
4375
3.82k
xmlTextReaderIsNamespaceDecl(xmlTextReader *reader) {
4376
3.82k
    xmlNodePtr node;
4377
3.82k
    if (reader == NULL)
4378
0
  return(-1);
4379
3.82k
    if (reader->node == NULL)
4380
1.61k
  return(-1);
4381
2.21k
    if (reader->curnode != NULL)
4382
419
  node = reader->curnode;
4383
1.79k
    else
4384
1.79k
  node = reader->node;
4385
4386
2.21k
    if (XML_NAMESPACE_DECL == node->type)
4387
252
  return(1);
4388
1.95k
    else
4389
1.95k
  return(0);
4390
2.21k
}
4391
4392
/**
4393
 * Determine the XML version of the document being read.
4394
 *
4395
 * @param reader  the xmlTextReader used
4396
 * @returns a string containing the XML version of the document or NULL
4397
 * in case of error.  The string is deallocated with the reader.
4398
 */
4399
const xmlChar *
4400
5.82k
xmlTextReaderConstXmlVersion(xmlTextReader *reader) {
4401
5.82k
    xmlDocPtr doc = NULL;
4402
5.82k
    if (reader == NULL)
4403
0
  return(NULL);
4404
5.82k
    if (reader->doc != NULL)
4405
0
        doc = reader->doc;
4406
5.82k
    else if (reader->ctxt != NULL)
4407
5.82k
  doc = reader->ctxt->myDoc;
4408
5.82k
    if (doc == NULL)
4409
710
  return(NULL);
4410
4411
5.11k
    if (doc->version == NULL)
4412
0
  return(NULL);
4413
5.11k
    else
4414
5.11k
      return(constString(reader, doc->version));
4415
5.11k
}
4416
4417
/**
4418
 * Determine the standalone status of the document being read.
4419
 *
4420
 * @param reader  the xmlTextReader used
4421
 * @returns 1 if the document was declared to be standalone, 0 if it
4422
 * was declared to be not standalone, or -1 if the document did not
4423
 * specify its standalone status or in case of error.
4424
 */
4425
int
4426
4.55k
xmlTextReaderStandalone(xmlTextReader *reader) {
4427
4.55k
    xmlDocPtr doc = NULL;
4428
4.55k
    if (reader == NULL)
4429
0
  return(-1);
4430
4.55k
    if (reader->doc != NULL)
4431
0
        doc = reader->doc;
4432
4.55k
    else if (reader->ctxt != NULL)
4433
4.55k
  doc = reader->ctxt->myDoc;
4434
4.55k
    if (doc == NULL)
4435
944
  return(-1);
4436
4437
3.60k
    return(doc->standalone);
4438
4.55k
}
4439
4440
/************************************************************************
4441
 *                  *
4442
 *      Error Handling Extensions                       *
4443
 *                  *
4444
 ************************************************************************/
4445
4446
/**
4447
 * Obtain the line number for the given locator.
4448
 *
4449
 * @param locator  the void used
4450
 * @returns the line number or -1 in case of error.
4451
 */
4452
int
4453
0
xmlTextReaderLocatorLineNumber(xmlTextReaderLocatorPtr locator) {
4454
    /* we know that locator is a xmlParserCtxtPtr */
4455
0
    xmlParserCtxtPtr ctx = (xmlParserCtxtPtr)locator;
4456
0
    int ret = -1;
4457
4458
0
    if (locator == NULL)
4459
0
        return(-1);
4460
0
    if (ctx->node != NULL) {
4461
0
  ret = xmlGetLineNo(ctx->node);
4462
0
    }
4463
0
    else {
4464
  /* inspired from error.c */
4465
0
  xmlParserInputPtr input;
4466
0
  input = ctx->input;
4467
0
  if ((input->filename == NULL) && (ctx->inputNr > 1))
4468
0
      input = ctx->inputTab[ctx->inputNr - 2];
4469
0
  if (input != NULL) {
4470
0
      ret = input->line;
4471
0
  }
4472
0
  else {
4473
0
      ret = -1;
4474
0
  }
4475
0
    }
4476
4477
0
    return ret;
4478
0
}
4479
4480
/**
4481
 * Obtain the base URI for the given locator.
4482
 *
4483
 * @param locator  the void used
4484
 * @returns the base URI or NULL in case of error,
4485
 *    if non NULL it need to be freed by the caller.
4486
 */
4487
xmlChar *
4488
0
xmlTextReaderLocatorBaseURI(xmlTextReaderLocatorPtr locator) {
4489
    /* we know that locator is a xmlParserCtxtPtr */
4490
0
    xmlParserCtxtPtr ctx = (xmlParserCtxtPtr)locator;
4491
0
    xmlChar *ret = NULL;
4492
4493
0
    if (locator == NULL)
4494
0
        return(NULL);
4495
0
    if (ctx->node != NULL) {
4496
0
  ret = xmlNodeGetBase(NULL,ctx->node);
4497
0
    }
4498
0
    else {
4499
  /* inspired from error.c */
4500
0
  xmlParserInputPtr input;
4501
0
  input = ctx->input;
4502
0
  if ((input->filename == NULL) && (ctx->inputNr > 1))
4503
0
      input = ctx->inputTab[ctx->inputNr - 2];
4504
0
  if (input != NULL) {
4505
0
      ret = xmlStrdup(BAD_CAST input->filename);
4506
0
  }
4507
0
  else {
4508
0
      ret = NULL;
4509
0
  }
4510
0
    }
4511
4512
0
    return ret;
4513
0
}
4514
4515
/**
4516
 * Register a callback function that will be called on error and warnings.
4517
 *
4518
 * @deprecated Use #xmlTextReaderSetStructuredErrorHandler.
4519
 *
4520
 * If `f` is NULL, the default error and warning handlers are restored.
4521
 *
4522
 * @param reader  the xmlTextReader used
4523
 * @param f   the callback function to call on error and warnings
4524
 * @param arg  a user argument to pass to the callback function
4525
 */
4526
void
4527
xmlTextReaderSetErrorHandler(xmlTextReader *reader,
4528
                             xmlTextReaderErrorFunc f, void *arg)
4529
0
{
4530
0
    if (reader == NULL)
4531
0
        return;
4532
4533
0
    if (f != NULL) {
4534
0
        reader->errorFunc = f;
4535
0
        reader->sErrorFunc = NULL;
4536
0
        reader->errorFuncArg = arg;
4537
0
        xmlCtxtSetErrorHandler(reader->ctxt,
4538
0
                xmlTextReaderStructuredRelay, reader);
4539
0
#ifdef LIBXML_RELAXNG_ENABLED
4540
0
        if (reader->rngValidCtxt) {
4541
0
            xmlRelaxNGSetValidStructuredErrors(reader->rngValidCtxt,
4542
0
                    xmlTextReaderStructuredRelay, reader);
4543
0
        }
4544
0
#endif
4545
0
#ifdef LIBXML_SCHEMAS_ENABLED
4546
0
        if (reader->xsdValidCtxt) {
4547
0
            xmlSchemaSetValidStructuredErrors(reader->xsdValidCtxt,
4548
0
                    xmlTextReaderStructuredRelay, reader);
4549
0
        }
4550
0
#endif
4551
0
    } else {
4552
        /* restore defaults */
4553
0
        reader->errorFunc = NULL;
4554
0
        reader->sErrorFunc = NULL;
4555
0
        reader->errorFuncArg = NULL;
4556
0
        xmlCtxtSetErrorHandler(reader->ctxt, NULL, NULL);
4557
0
#ifdef LIBXML_RELAXNG_ENABLED
4558
0
        if (reader->rngValidCtxt) {
4559
0
            xmlRelaxNGSetValidStructuredErrors(reader->rngValidCtxt, NULL,
4560
0
                                               NULL);
4561
0
        }
4562
0
#endif
4563
0
#ifdef LIBXML_SCHEMAS_ENABLED
4564
0
        if (reader->xsdValidCtxt) {
4565
0
            xmlSchemaSetValidStructuredErrors(reader->xsdValidCtxt, NULL,
4566
0
                                              NULL);
4567
0
        }
4568
0
#endif
4569
0
    }
4570
0
}
4571
4572
/**
4573
* xmlTextReaderSetStructuredErrorHandler:
4574
 *
4575
 * Register a callback function that will be called on error and warnings.
4576
 *
4577
 * If `f` is NULL, the default error and warning handlers are restored.
4578
 *
4579
 * @param reader  the xmlTextReader used
4580
 * @param f   the callback function to call on error and warnings
4581
 * @param arg  a user argument to pass to the callback function
4582
 */
4583
void
4584
xmlTextReaderSetStructuredErrorHandler(xmlTextReader *reader,
4585
                                       xmlStructuredErrorFunc f, void *arg)
4586
36.8k
{
4587
36.8k
    if (reader == NULL)
4588
0
        return;
4589
4590
36.8k
    if (f != NULL) {
4591
36.8k
        reader->sErrorFunc = f;
4592
36.8k
        reader->errorFunc = NULL;
4593
36.8k
        reader->errorFuncArg = arg;
4594
36.8k
        xmlCtxtSetErrorHandler(reader->ctxt,
4595
36.8k
                xmlTextReaderStructuredRelay, reader);
4596
36.8k
#ifdef LIBXML_RELAXNG_ENABLED
4597
36.8k
        if (reader->rngValidCtxt) {
4598
0
            xmlRelaxNGSetValidStructuredErrors(reader->rngValidCtxt,
4599
0
                    xmlTextReaderStructuredRelay, reader);
4600
0
        }
4601
36.8k
#endif
4602
36.8k
#ifdef LIBXML_SCHEMAS_ENABLED
4603
36.8k
        if (reader->xsdValidCtxt) {
4604
0
            xmlSchemaSetValidStructuredErrors(reader->xsdValidCtxt,
4605
0
                    xmlTextReaderStructuredRelay, reader);
4606
0
        }
4607
36.8k
#endif
4608
36.8k
    } else {
4609
        /* restore defaults */
4610
0
        reader->errorFunc = NULL;
4611
0
        reader->sErrorFunc = NULL;
4612
0
        reader->errorFuncArg = NULL;
4613
0
        xmlCtxtSetErrorHandler(reader->ctxt, NULL, NULL);
4614
0
#ifdef LIBXML_RELAXNG_ENABLED
4615
0
        if (reader->rngValidCtxt) {
4616
0
            xmlRelaxNGSetValidStructuredErrors(reader->rngValidCtxt, NULL,
4617
0
                                               NULL);
4618
0
        }
4619
0
#endif
4620
0
#ifdef LIBXML_SCHEMAS_ENABLED
4621
0
        if (reader->xsdValidCtxt) {
4622
0
            xmlSchemaSetValidStructuredErrors(reader->xsdValidCtxt, NULL,
4623
0
                                              NULL);
4624
0
        }
4625
0
#endif
4626
0
    }
4627
36.8k
}
4628
4629
/**
4630
 * Retrieve the error callback function and user argument.
4631
 *
4632
 * @param reader  the xmlTextReader used
4633
 * @param f   the callback function or NULL is no callback has been registered
4634
 * @param arg  a user argument
4635
 */
4636
void
4637
xmlTextReaderGetErrorHandler(xmlTextReader *reader,
4638
                             xmlTextReaderErrorFunc * f, void **arg)
4639
0
{
4640
0
    if (f != NULL)
4641
0
        *f = reader->errorFunc;
4642
0
    if (arg != NULL)
4643
0
        *arg = reader->errorFuncArg;
4644
0
}
4645
4646
/**
4647
 * Register a callback function that will be called to load external
4648
 * resources like entities.
4649
 *
4650
 * @since 2.14.0
4651
 * @param reader  thr reader
4652
 * @param loader  resource loader
4653
 * @param data  user data which will be passed to the loader
4654
 */
4655
void
4656
xmlTextReaderSetResourceLoader(xmlTextReader *reader,
4657
36.8k
                               xmlResourceLoader loader, void *data) {
4658
36.8k
    if ((reader == NULL) || (reader->ctxt == NULL))
4659
0
        return;
4660
4661
36.8k
    reader->resourceLoader = loader;
4662
36.8k
    reader->resourceCtxt = data;
4663
4664
36.8k
    xmlCtxtSetResourceLoader(reader->ctxt, loader, data);
4665
36.8k
}
4666
4667
/**
4668
 * Retrieve the validity status from the parser context
4669
 *
4670
 * @param reader  the xmlTextReader used
4671
 * @returns the flag value 1 if valid, 0 if no, and -1 in case of error
4672
 */
4673
int
4674
xmlTextReaderIsValid(xmlTextReader *reader)
4675
4.35k
{
4676
4.35k
    if (reader == NULL)
4677
0
        return (-1);
4678
4.35k
#ifdef LIBXML_RELAXNG_ENABLED
4679
4.35k
    if (reader->validate == XML_TEXTREADER_VALIDATE_RNG)
4680
0
        return (reader->rngValidErrors == 0);
4681
4.35k
#endif
4682
4.35k
#ifdef LIBXML_SCHEMAS_ENABLED
4683
4.35k
    if (reader->validate == XML_TEXTREADER_VALIDATE_XSD)
4684
0
        return (reader->xsdValidErrors == 0);
4685
4.35k
#endif
4686
4.35k
    if ((reader->ctxt != NULL) && (reader->ctxt->validate == 1))
4687
1.79k
        return (reader->ctxt->valid);
4688
2.55k
    return (0);
4689
4.35k
}
4690
4691
/************************************************************************
4692
 *                  *
4693
 *  New set (2.6.0) of simpler and more flexible APIs   *
4694
 *                  *
4695
 ************************************************************************/
4696
4697
/**
4698
 * Setup an XML reader with new options
4699
 *
4700
 * @param reader  an XML reader
4701
 * @param input  xmlParserInputBuffer used to feed the reader, will
4702
 *         be destroyed with it.
4703
 * @param URL  the base URL to use for the document
4704
 * @param encoding  the document encoding, or NULL
4705
 * @param options  a combination of xmlParserOption
4706
 * @returns 0 in case of success and -1 in case of error.
4707
 */
4708
int
4709
xmlTextReaderSetup(xmlTextReader *reader,
4710
                   xmlParserInputBuffer *input, const char *URL,
4711
                   const char *encoding, int options)
4712
36.8k
{
4713
36.8k
    if (reader == NULL) {
4714
0
        if (input != NULL)
4715
0
      xmlFreeParserInputBuffer(input);
4716
0
        return (-1);
4717
0
    }
4718
4719
    /*
4720
     * we force the generation of compact text nodes on the reader
4721
     * since usr applications should never modify the tree
4722
     */
4723
36.8k
    options |= XML_PARSE_COMPACT;
4724
4725
36.8k
    reader->doc = NULL;
4726
36.8k
    reader->entNr = 0;
4727
36.8k
    reader->parserFlags = options;
4728
36.8k
    reader->validate = XML_TEXTREADER_NOT_VALIDATE;
4729
36.8k
    if ((input != NULL) && (reader->input != NULL) &&
4730
36.8k
        (reader->allocs & XML_TEXTREADER_INPUT)) {
4731
0
  xmlFreeParserInputBuffer(reader->input);
4732
0
  reader->input = NULL;
4733
0
  reader->allocs -= XML_TEXTREADER_INPUT;
4734
0
    }
4735
36.8k
    if (input != NULL) {
4736
0
  reader->input = input;
4737
0
  reader->allocs |= XML_TEXTREADER_INPUT;
4738
0
    }
4739
36.8k
    if (reader->buffer == NULL)
4740
0
        reader->buffer = xmlBufCreate(50);
4741
36.8k
    if (reader->buffer == NULL) {
4742
0
        return (-1);
4743
0
    }
4744
36.8k
    if (reader->sax == NULL)
4745
0
  reader->sax = (xmlSAXHandler *) xmlMalloc(sizeof(xmlSAXHandler));
4746
36.8k
    if (reader->sax == NULL) {
4747
0
        return (-1);
4748
0
    }
4749
36.8k
    xmlSAXVersion(reader->sax, 2);
4750
36.8k
    reader->startElement = reader->sax->startElement;
4751
36.8k
    reader->sax->startElement = xmlTextReaderStartElement;
4752
36.8k
    reader->endElement = reader->sax->endElement;
4753
36.8k
    reader->sax->endElement = xmlTextReaderEndElement;
4754
36.8k
#ifdef LIBXML_SAX1_ENABLED
4755
36.8k
    if (reader->sax->initialized == XML_SAX2_MAGIC) {
4756
36.8k
#endif /* LIBXML_SAX1_ENABLED */
4757
36.8k
        reader->startElementNs = reader->sax->startElementNs;
4758
36.8k
        reader->sax->startElementNs = xmlTextReaderStartElementNs;
4759
36.8k
        reader->endElementNs = reader->sax->endElementNs;
4760
36.8k
        reader->sax->endElementNs = xmlTextReaderEndElementNs;
4761
36.8k
#ifdef LIBXML_SAX1_ENABLED
4762
36.8k
    } else {
4763
0
        reader->startElementNs = NULL;
4764
0
        reader->endElementNs = NULL;
4765
0
    }
4766
36.8k
#endif /* LIBXML_SAX1_ENABLED */
4767
36.8k
    reader->characters = reader->sax->characters;
4768
36.8k
    reader->sax->characters = xmlTextReaderCharacters;
4769
36.8k
    reader->sax->ignorableWhitespace = xmlTextReaderCharacters;
4770
36.8k
    reader->cdataBlock = reader->sax->cdataBlock;
4771
36.8k
    reader->sax->cdataBlock = xmlTextReaderCDataBlock;
4772
4773
36.8k
    reader->mode = XML_TEXTREADER_MODE_INITIAL;
4774
36.8k
    reader->node = NULL;
4775
36.8k
    reader->curnode = NULL;
4776
36.8k
    if (input != NULL) {
4777
0
        if (xmlBufUse(reader->input->buffer) < 4) {
4778
0
            xmlParserInputBufferRead(input, 4);
4779
0
        }
4780
0
        if (reader->ctxt == NULL) {
4781
0
            if (xmlBufUse(reader->input->buffer) >= 4) {
4782
0
                reader->ctxt = xmlCreatePushParserCtxt(reader->sax, NULL,
4783
0
           (const char *) xmlBufContent(reader->input->buffer),
4784
0
                                      4, URL);
4785
0
                reader->base = 0;
4786
0
                reader->cur = 4;
4787
0
            } else {
4788
0
                reader->ctxt =
4789
0
                    xmlCreatePushParserCtxt(reader->sax, NULL, NULL, 0, URL);
4790
0
                reader->base = 0;
4791
0
                reader->cur = 0;
4792
0
            }
4793
0
            if (reader->ctxt == NULL) {
4794
0
                return (-1);
4795
0
            }
4796
0
        } else {
4797
0
      xmlParserInputPtr inputStream;
4798
0
      xmlParserInputBufferPtr buf;
4799
4800
0
      xmlCtxtReset(reader->ctxt);
4801
0
      buf = xmlAllocParserInputBuffer(XML_CHAR_ENCODING_NONE);
4802
0
      if (buf == NULL) return(-1);
4803
0
      inputStream = xmlNewInputStream(reader->ctxt);
4804
0
      if (inputStream == NULL) {
4805
0
    xmlFreeParserInputBuffer(buf);
4806
0
    return(-1);
4807
0
      }
4808
4809
0
      if (URL == NULL)
4810
0
    inputStream->filename = NULL;
4811
0
      else
4812
0
    inputStream->filename = (char *)
4813
0
        xmlCanonicPath((const xmlChar *) URL);
4814
0
      inputStream->buf = buf;
4815
0
            xmlBufResetInput(buf->buffer, inputStream);
4816
4817
0
            if (xmlCtxtPushInput(reader->ctxt, inputStream) < 0) {
4818
0
                xmlFreeInputStream(inputStream);
4819
0
                return(-1);
4820
0
            }
4821
0
      reader->cur = 0;
4822
0
  }
4823
0
    }
4824
36.8k
    if (reader->dict != NULL) {
4825
36.8k
        if (reader->ctxt->dict != NULL) {
4826
36.8k
      if (reader->dict != reader->ctxt->dict) {
4827
0
    xmlDictFree(reader->dict);
4828
0
    reader->dict = reader->ctxt->dict;
4829
0
      }
4830
36.8k
  } else {
4831
0
      reader->ctxt->dict = reader->dict;
4832
0
  }
4833
36.8k
    } else {
4834
0
  if (reader->ctxt->dict == NULL)
4835
0
      reader->ctxt->dict = xmlDictCreate();
4836
0
        reader->dict = reader->ctxt->dict;
4837
0
    }
4838
36.8k
    reader->ctxt->_private = reader;
4839
36.8k
    reader->ctxt->dictNames = 1;
4840
    /*
4841
     * use the parser dictionary to allocate all elements and attributes names
4842
     */
4843
36.8k
    reader->ctxt->parseMode = XML_PARSE_READER;
4844
4845
36.8k
#ifdef LIBXML_XINCLUDE_ENABLED
4846
36.8k
    if (reader->xincctxt != NULL) {
4847
0
  xmlXIncludeFreeContext(reader->xincctxt);
4848
0
  reader->xincctxt = NULL;
4849
0
    }
4850
36.8k
    if (options & XML_PARSE_XINCLUDE) {
4851
20.3k
        reader->xinclude = 1;
4852
20.3k
  options -= XML_PARSE_XINCLUDE;
4853
20.3k
    } else
4854
16.5k
        reader->xinclude = 0;
4855
36.8k
    reader->in_xinclude = 0;
4856
36.8k
#endif
4857
36.8k
#ifdef LIBXML_PATTERN_ENABLED
4858
36.8k
    if (reader->patternTab == NULL) {
4859
36.8k
        reader->patternNr = 0;
4860
36.8k
  reader->patternMax = 0;
4861
36.8k
    }
4862
36.8k
    while (reader->patternNr > 0) {
4863
0
        reader->patternNr--;
4864
0
  if (reader->patternTab[reader->patternNr] != NULL) {
4865
0
      xmlFreePattern(reader->patternTab[reader->patternNr]);
4866
0
            reader->patternTab[reader->patternNr] = NULL;
4867
0
  }
4868
0
    }
4869
36.8k
#endif
4870
4871
36.8k
    if (options & XML_PARSE_DTDVALID)
4872
17.9k
        reader->validate = XML_TEXTREADER_VALIDATE_DTD;
4873
4874
36.8k
    xmlCtxtUseOptions(reader->ctxt, options);
4875
36.8k
    if (encoding != NULL)
4876
0
        xmlSwitchEncodingName(reader->ctxt, encoding);
4877
36.8k
    if ((URL != NULL) && (reader->ctxt->input != NULL) &&
4878
36.8k
        (reader->ctxt->input->filename == NULL)) {
4879
0
        reader->ctxt->input->filename = (char *)
4880
0
            xmlStrdup((const xmlChar *) URL);
4881
0
        if (reader->ctxt->input->filename == NULL)
4882
0
            return(-1);
4883
0
    }
4884
4885
36.8k
    reader->doc = NULL;
4886
4887
36.8k
    return (0);
4888
36.8k
}
4889
4890
/**
4891
 * Set the maximum amplification factor. See #xmlCtxtSetMaxAmplification.
4892
 *
4893
 * @param reader  an XML reader
4894
 * @param maxAmpl  maximum amplification factor
4895
 */
4896
void
4897
xmlTextReaderSetMaxAmplification(xmlTextReader *reader, unsigned maxAmpl)
4898
0
{
4899
0
    if (reader == NULL)
4900
0
        return;
4901
0
    xmlCtxtSetMaxAmplification(reader->ctxt, maxAmpl);
4902
0
}
4903
4904
/**
4905
 * @since 2.13.0
4906
 *
4907
 * @param reader  an XML reader
4908
 * @returns the last error.
4909
 */
4910
const xmlError *
4911
xmlTextReaderGetLastError(xmlTextReader *reader)
4912
36.8k
{
4913
36.8k
    if ((reader == NULL) || (reader->ctxt == NULL))
4914
0
        return(NULL);
4915
36.8k
    return(&reader->ctxt->lastError);
4916
36.8k
}
4917
4918
/**
4919
 * This function provides the current index of the parser used
4920
 * by the reader, relative to the start of the current entity.
4921
 * This function actually just wraps a call to #xmlByteConsumed
4922
 * for the parser context associated with the reader.
4923
 * See #xmlByteConsumed for more information.
4924
 *
4925
 * @deprecated The returned value is mostly random and useless.
4926
 * It reflects the parser reading ahead and is in no way related to
4927
 * the current node.
4928
 *
4929
 * @param reader  an XML reader
4930
 * @returns the index in bytes from the beginning of the entity or -1
4931
 *         in case the index could not be computed.
4932
 */
4933
long
4934
6.12k
xmlTextReaderByteConsumed(xmlTextReader *reader) {
4935
6.12k
    xmlParserInputPtr in;
4936
4937
6.12k
    if ((reader == NULL) || (reader->ctxt == NULL))
4938
0
        return(-1);
4939
6.12k
    in = reader->ctxt->input;
4940
6.12k
    if (in == NULL)
4941
0
        return(-1);
4942
6.12k
    return(in->consumed + (in->cur - in->base));
4943
6.12k
}
4944
4945
4946
/**
4947
 * Create an xmltextReader for a preparsed document.
4948
 *
4949
 * @param doc  a preparsed document
4950
 * @returns the new reader or NULL in case of error.
4951
 */
4952
xmlTextReader *
4953
xmlReaderWalker(xmlDoc *doc)
4954
1.64k
{
4955
1.64k
    xmlTextReaderPtr ret;
4956
4957
1.64k
    if (doc == NULL)
4958
0
        return(NULL);
4959
4960
1.64k
    ret = xmlMalloc(sizeof(xmlTextReader));
4961
1.64k
    if (ret == NULL) {
4962
0
  return(NULL);
4963
0
    }
4964
1.64k
    memset(ret, 0, sizeof(xmlTextReader));
4965
1.64k
    ret->entNr = 0;
4966
1.64k
    ret->input = NULL;
4967
1.64k
    ret->mode = XML_TEXTREADER_MODE_INITIAL;
4968
1.64k
    ret->node = NULL;
4969
1.64k
    ret->curnode = NULL;
4970
1.64k
    ret->base = 0;
4971
1.64k
    ret->cur = 0;
4972
1.64k
    ret->allocs = XML_TEXTREADER_CTXT;
4973
1.64k
    ret->doc = doc;
4974
1.64k
    ret->state = XML_TEXTREADER_START;
4975
1.64k
    ret->dict = xmlDictCreate();
4976
1.64k
    return(ret);
4977
1.64k
}
4978
4979
/**
4980
 * Create an xmltextReader for an XML in-memory document.
4981
 * The parsing flags `options` are a combination of xmlParserOption.
4982
 *
4983
 * @param cur  a pointer to a zero terminated string
4984
 * @param URL  the base URL to use for the document
4985
 * @param encoding  the document encoding, or NULL
4986
 * @param options  a combination of xmlParserOption
4987
 * @returns the new reader or NULL in case of error.
4988
 */
4989
xmlTextReader *
4990
xmlReaderForDoc(const xmlChar * cur, const char *URL, const char *encoding,
4991
                int options)
4992
0
{
4993
0
    int len;
4994
4995
0
    if (cur == NULL)
4996
0
        return (NULL);
4997
0
    len = xmlStrlen(cur);
4998
4999
0
    return (xmlReaderForMemory
5000
0
            ((const char *) cur, len, URL, encoding, options));
5001
0
}
5002
5003
/**
5004
 * parse an XML file from the filesystem or the network.
5005
 * The parsing flags `options` are a combination of xmlParserOption.
5006
 *
5007
 * @param filename  a file or URL
5008
 * @param encoding  the document encoding, or NULL
5009
 * @param options  a combination of xmlParserOption
5010
 * @returns the new reader or NULL in case of error.
5011
 */
5012
xmlTextReader *
5013
xmlReaderForFile(const char *filename, const char *encoding, int options)
5014
3.07k
{
5015
3.07k
    xmlTextReaderPtr reader;
5016
5017
3.07k
    reader = xmlNewTextReaderFilename(filename);
5018
3.07k
    if (reader == NULL)
5019
3.07k
        return (NULL);
5020
0
    if (xmlTextReaderSetup(reader, NULL, NULL, encoding, options) < 0) {
5021
0
        xmlFreeTextReader(reader);
5022
0
        return (NULL);
5023
0
    }
5024
0
    return (reader);
5025
0
}
5026
5027
/**
5028
 * Create an xmltextReader for an XML in-memory document.
5029
 * The parsing flags `options` are a combination of xmlParserOption.
5030
 *
5031
 * @param buffer  a pointer to a char array
5032
 * @param size  the size of the array
5033
 * @param URL  the base URL to use for the document
5034
 * @param encoding  the document encoding, or NULL
5035
 * @param options  a combination of xmlParserOption
5036
 * @returns the new reader or NULL in case of error.
5037
 */
5038
xmlTextReader *
5039
xmlReaderForMemory(const char *buffer, int size, const char *URL,
5040
                   const char *encoding, int options)
5041
36.9k
{
5042
36.9k
    xmlTextReaderPtr reader;
5043
36.9k
    xmlParserInputBufferPtr buf;
5044
5045
36.9k
    buf = xmlParserInputBufferCreateMem(buffer, size, XML_CHAR_ENCODING_NONE);
5046
36.9k
    if (buf == NULL) {
5047
11
        return (NULL);
5048
11
    }
5049
36.8k
    reader = xmlNewTextReader(buf, URL);
5050
36.8k
    if (reader == NULL) {
5051
54
        xmlFreeParserInputBuffer(buf);
5052
54
        return (NULL);
5053
54
    }
5054
36.8k
    reader->allocs |= XML_TEXTREADER_INPUT;
5055
36.8k
    if (xmlTextReaderSetup(reader, NULL, URL, encoding, options) < 0) {
5056
0
        xmlFreeTextReader(reader);
5057
0
        return (NULL);
5058
0
    }
5059
36.8k
    return (reader);
5060
36.8k
}
5061
5062
/**
5063
 * Create an xmltextReader for an XML from a file descriptor.
5064
 * The parsing flags `options` are a combination of xmlParserOption.
5065
 * NOTE that the file descriptor will not be closed when the
5066
 *      reader is closed or reset.
5067
 *
5068
 * @param fd  an open file descriptor
5069
 * @param URL  the base URL to use for the document
5070
 * @param encoding  the document encoding, or NULL
5071
 * @param options  a combination of xmlParserOption
5072
 * @returns the new reader or NULL in case of error.
5073
 */
5074
xmlTextReader *
5075
xmlReaderForFd(int fd, const char *URL, const char *encoding, int options)
5076
0
{
5077
0
    xmlTextReaderPtr reader;
5078
0
    xmlParserInputBufferPtr input;
5079
0
    xmlParserErrors code;
5080
5081
0
    if (fd < 0) {
5082
0
        xmlTextReaderErr(XML_ERR_ARGUMENT, "invalid argument");
5083
0
        return(NULL);
5084
0
    }
5085
5086
0
    input = xmlAllocParserInputBuffer(XML_CHAR_ENCODING_NONE);
5087
0
    if (input == NULL) {
5088
0
        xmlTextReaderErrMemory(NULL);
5089
0
        return(NULL);
5090
0
    }
5091
    /*
5092
     * TODO: Remove XML_INPUT_UNZIP
5093
     */
5094
0
    code = xmlInputFromFd(input, fd, XML_INPUT_UNZIP);
5095
0
    if (code != XML_ERR_OK) {
5096
0
        xmlTextReaderErr(code, "failed to open fd");
5097
0
        return(NULL);
5098
0
    }
5099
0
    input->closecallback = NULL;
5100
5101
0
    reader = xmlNewTextReader(input, URL);
5102
0
    if (reader == NULL) {
5103
0
        xmlTextReaderErrMemory(NULL);
5104
0
        xmlFreeParserInputBuffer(input);
5105
0
        return(NULL);
5106
0
    }
5107
0
    reader->allocs |= XML_TEXTREADER_INPUT;
5108
0
    if (xmlTextReaderSetup(reader, NULL, URL, encoding, options) < 0) {
5109
0
        xmlTextReaderErrMemory(NULL);
5110
0
        xmlFreeTextReader(reader);
5111
0
        return(NULL);
5112
0
    }
5113
0
    return (reader);
5114
0
}
5115
5116
/**
5117
 * Create an xmltextReader for an XML document from I/O functions and source.
5118
 * The parsing flags `options` are a combination of xmlParserOption.
5119
 *
5120
 * @param ioread  an I/O read function
5121
 * @param ioclose  an I/O close function
5122
 * @param ioctx  an I/O handler
5123
 * @param URL  the base URL to use for the document
5124
 * @param encoding  the document encoding, or NULL
5125
 * @param options  a combination of xmlParserOption
5126
 * @returns the new reader or NULL in case of error.
5127
 */
5128
xmlTextReader *
5129
xmlReaderForIO(xmlInputReadCallback ioread, xmlInputCloseCallback ioclose,
5130
               void *ioctx, const char *URL, const char *encoding,
5131
               int options)
5132
0
{
5133
0
    xmlTextReaderPtr reader;
5134
0
    xmlParserInputBufferPtr input;
5135
5136
0
    if (ioread == NULL)
5137
0
        return (NULL);
5138
5139
0
    input = xmlParserInputBufferCreateIO(ioread, ioclose, ioctx,
5140
0
                                         XML_CHAR_ENCODING_NONE);
5141
0
    if (input == NULL) {
5142
0
        if (ioclose != NULL)
5143
0
            ioclose(ioctx);
5144
0
        return (NULL);
5145
0
    }
5146
0
    reader = xmlNewTextReader(input, URL);
5147
0
    if (reader == NULL) {
5148
0
        xmlFreeParserInputBuffer(input);
5149
0
        return (NULL);
5150
0
    }
5151
0
    reader->allocs |= XML_TEXTREADER_INPUT;
5152
0
    if (xmlTextReaderSetup(reader, NULL, URL, encoding, options) < 0) {
5153
0
        xmlFreeTextReader(reader);
5154
0
        return (NULL);
5155
0
    }
5156
0
    return (reader);
5157
0
}
5158
5159
/**
5160
 * Setup an xmltextReader to parse a preparsed XML document.
5161
 * This reuses the existing `reader` xmlTextReader.
5162
 *
5163
 * @param reader  an XML reader
5164
 * @param doc  a preparsed document
5165
 * @returns 0 in case of success and -1 in case of error
5166
 */
5167
int
5168
xmlReaderNewWalker(xmlTextReader *reader, xmlDoc *doc)
5169
0
{
5170
0
    if (doc == NULL)
5171
0
        return (-1);
5172
0
    if (reader == NULL)
5173
0
        return (-1);
5174
5175
0
    if (reader->input != NULL) {
5176
0
        xmlFreeParserInputBuffer(reader->input);
5177
0
    }
5178
0
    if (reader->ctxt != NULL) {
5179
0
  xmlCtxtReset(reader->ctxt);
5180
0
    }
5181
5182
0
    reader->entNr = 0;
5183
0
    reader->input = NULL;
5184
0
    reader->mode = XML_TEXTREADER_MODE_INITIAL;
5185
0
    reader->node = NULL;
5186
0
    reader->curnode = NULL;
5187
0
    reader->base = 0;
5188
0
    reader->cur = 0;
5189
0
    reader->allocs = XML_TEXTREADER_CTXT;
5190
0
    reader->doc = doc;
5191
0
    reader->state = XML_TEXTREADER_START;
5192
0
    if (reader->dict == NULL) {
5193
0
        if ((reader->ctxt != NULL) && (reader->ctxt->dict != NULL))
5194
0
      reader->dict = reader->ctxt->dict;
5195
0
  else
5196
0
      reader->dict = xmlDictCreate();
5197
0
    }
5198
0
    return(0);
5199
0
}
5200
5201
/**
5202
 * Setup an xmltextReader to parse an XML in-memory document.
5203
 * The parsing flags `options` are a combination of xmlParserOption.
5204
 * This reuses the existing `reader` xmlTextReader.
5205
 *
5206
 * @param reader  an XML reader
5207
 * @param cur  a pointer to a zero terminated string
5208
 * @param URL  the base URL to use for the document
5209
 * @param encoding  the document encoding, or NULL
5210
 * @param options  a combination of xmlParserOption
5211
 * @returns 0 in case of success and -1 in case of error
5212
 */
5213
int
5214
xmlReaderNewDoc(xmlTextReader *reader, const xmlChar * cur,
5215
                const char *URL, const char *encoding, int options)
5216
0
{
5217
5218
0
    int len;
5219
5220
0
    if (cur == NULL)
5221
0
        return (-1);
5222
0
    if (reader == NULL)
5223
0
        return (-1);
5224
5225
0
    len = xmlStrlen(cur);
5226
0
    return (xmlReaderNewMemory(reader, (const char *)cur, len,
5227
0
                               URL, encoding, options));
5228
0
}
5229
5230
/**
5231
 * parse an XML file from the filesystem or the network.
5232
 * The parsing flags `options` are a combination of xmlParserOption.
5233
 * This reuses the existing `reader` xmlTextReader.
5234
 *
5235
 * @param reader  an XML reader
5236
 * @param filename  a file or URL
5237
 * @param encoding  the document encoding, or NULL
5238
 * @param options  a combination of xmlParserOption
5239
 * @returns 0 in case of success and -1 in case of error
5240
 */
5241
int
5242
xmlReaderNewFile(xmlTextReader *reader, const char *filename,
5243
                 const char *encoding, int options)
5244
0
{
5245
0
    xmlParserInputBufferPtr input;
5246
5247
0
    if ((filename == NULL) || (reader == NULL)) {
5248
0
        xmlTextReaderErr(XML_ERR_ARGUMENT, "invalid argument");
5249
0
        return(-1);
5250
0
    }
5251
5252
0
    if (xmlParserInputBufferCreateFilenameValue != NULL) {
5253
0
        input = xmlParserInputBufferCreateFilenameValue(filename,
5254
0
                XML_CHAR_ENCODING_NONE);
5255
0
        if (input == NULL) {
5256
0
            xmlTextReaderErr(XML_IO_ENOENT, "failed to open %s", filename);
5257
0
            return(-1);
5258
0
        }
5259
0
    } else {
5260
        /*
5261
         * TODO: Remove XML_INPUT_UNZIP
5262
         */
5263
0
        xmlParserInputFlags flags = XML_INPUT_UNZIP;
5264
0
        xmlParserErrors code;
5265
5266
0
        if ((options & XML_PARSE_NONET) == 0)
5267
0
            flags |= XML_INPUT_NETWORK;
5268
5269
0
        code = xmlParserInputBufferCreateUrl(filename, XML_CHAR_ENCODING_NONE,
5270
0
                                             flags, &input);
5271
0
        if (code != XML_ERR_OK) {
5272
0
            xmlTextReaderErr(code, "failed to open %s", filename);
5273
0
            return(-1);
5274
0
        }
5275
0
    }
5276
5277
0
    if (xmlTextReaderSetup(reader, input, filename, encoding, options) < 0) {
5278
0
        xmlTextReaderErrMemory(NULL);
5279
0
        return(-1);
5280
0
    }
5281
5282
0
    return(0);
5283
0
}
5284
5285
/**
5286
 * Setup an xmltextReader to parse an XML in-memory document.
5287
 * The parsing flags `options` are a combination of xmlParserOption.
5288
 * This reuses the existing `reader` xmlTextReader.
5289
 *
5290
 * @param reader  an XML reader
5291
 * @param buffer  a pointer to a char array
5292
 * @param size  the size of the array
5293
 * @param URL  the base URL to use for the document
5294
 * @param encoding  the document encoding, or NULL
5295
 * @param options  a combination of xmlParserOption
5296
 * @returns 0 in case of success and -1 in case of error
5297
 */
5298
int
5299
xmlReaderNewMemory(xmlTextReader *reader, const char *buffer, int size,
5300
                   const char *URL, const char *encoding, int options)
5301
0
{
5302
0
    xmlParserInputBufferPtr input;
5303
5304
0
    if (reader == NULL)
5305
0
        return (-1);
5306
0
    if (buffer == NULL)
5307
0
        return (-1);
5308
5309
0
    input = xmlParserInputBufferCreateMem(buffer, size,
5310
0
                                      XML_CHAR_ENCODING_NONE);
5311
0
    if (input == NULL) {
5312
0
        return (-1);
5313
0
    }
5314
0
    return (xmlTextReaderSetup(reader, input, URL, encoding, options));
5315
0
}
5316
5317
/**
5318
 * Setup an xmltextReader to parse an XML from a file descriptor.
5319
 * NOTE that the file descriptor will not be closed when the
5320
 *      reader is closed or reset.
5321
 * The parsing flags `options` are a combination of xmlParserOption.
5322
 * This reuses the existing `reader` xmlTextReader.
5323
 *
5324
 * @param reader  an XML reader
5325
 * @param fd  an open file descriptor
5326
 * @param URL  the base URL to use for the document
5327
 * @param encoding  the document encoding, or NULL
5328
 * @param options  a combination of xmlParserOption
5329
 * @returns 0 in case of success and -1 in case of error
5330
 */
5331
int
5332
xmlReaderNewFd(xmlTextReader *reader, int fd,
5333
               const char *URL, const char *encoding, int options)
5334
0
{
5335
0
    xmlParserInputBufferPtr input;
5336
0
    xmlParserErrors code;
5337
5338
0
    if ((fd < 0) || (reader == NULL)) {
5339
0
        xmlTextReaderErr(XML_ERR_ARGUMENT, "invalid argument");
5340
0
        return(-1);
5341
0
    }
5342
5343
0
    input = xmlAllocParserInputBuffer(XML_CHAR_ENCODING_NONE);
5344
0
    if (input == NULL) {
5345
0
        xmlTextReaderErrMemory(NULL);
5346
0
        return(-1);
5347
0
    }
5348
    /*
5349
     * TODO: Remove XML_INPUT_UNZIP
5350
     */
5351
0
    code = xmlInputFromFd(input, fd, XML_INPUT_UNZIP);
5352
0
    if (code != XML_ERR_OK) {
5353
0
        xmlTextReaderErr(code, "failed to open fd");
5354
0
        return(-1);
5355
0
    }
5356
0
    input->closecallback = NULL;
5357
5358
0
    if (xmlTextReaderSetup(reader, input, URL, encoding, options) < 0) {
5359
0
        xmlTextReaderErrMemory(NULL);
5360
0
        return(-1);
5361
0
    }
5362
5363
0
    return(0);
5364
0
}
5365
5366
/**
5367
 * Setup an xmltextReader to parse an XML document from I/O functions
5368
 * and source.
5369
 * The parsing flags `options` are a combination of xmlParserOption.
5370
 * This reuses the existing `reader` xmlTextReader.
5371
 *
5372
 * @param reader  an XML reader
5373
 * @param ioread  an I/O read function
5374
 * @param ioclose  an I/O close function
5375
 * @param ioctx  an I/O handler
5376
 * @param URL  the base URL to use for the document
5377
 * @param encoding  the document encoding, or NULL
5378
 * @param options  a combination of xmlParserOption
5379
 * @returns 0 in case of success and -1 in case of error
5380
 */
5381
int
5382
xmlReaderNewIO(xmlTextReader *reader, xmlInputReadCallback ioread,
5383
               xmlInputCloseCallback ioclose, void *ioctx,
5384
               const char *URL, const char *encoding, int options)
5385
0
{
5386
0
    xmlParserInputBufferPtr input;
5387
5388
0
    if (ioread == NULL)
5389
0
        return (-1);
5390
0
    if (reader == NULL)
5391
0
        return (-1);
5392
5393
0
    input = xmlParserInputBufferCreateIO(ioread, ioclose, ioctx,
5394
0
                                         XML_CHAR_ENCODING_NONE);
5395
0
    if (input == NULL) {
5396
0
        if (ioclose != NULL)
5397
0
            ioclose(ioctx);
5398
0
        return (-1);
5399
0
    }
5400
0
    return (xmlTextReaderSetup(reader, input, URL, encoding, options));
5401
0
}
5402
5403
#endif /* LIBXML_READER_ENABLED */