Coverage Report

Created: 2026-06-10 06:22

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