Coverage Report

Created: 2025-06-22 06:55

/src/libxml2/xinclude.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * xinclude.c : Code to implement XInclude processing
3
 *
4
 * World Wide Web Consortium W3C Last Call Working Draft 10 November 2003
5
 * http://www.w3.org/TR/2003/WD-xinclude-20031110
6
 *
7
 * See Copyright for the status of this software.
8
 *
9
 * Author: Daniel Veillard
10
 */
11
12
#define IN_LIBXML
13
#include "libxml.h"
14
15
#include <string.h>
16
#include <libxml/xmlmemory.h>
17
#include <libxml/tree.h>
18
#include <libxml/parser.h>
19
#include <libxml/uri.h>
20
#include <libxml/xpath.h>
21
#include <libxml/xpointer.h>
22
#include <libxml/parserInternals.h>
23
#include <libxml/xmlerror.h>
24
#include <libxml/encoding.h>
25
26
#ifdef LIBXML_XINCLUDE_ENABLED
27
#include <libxml/xinclude.h>
28
29
#include "private/buf.h"
30
#include "private/error.h"
31
#include "private/memory.h"
32
#include "private/parser.h"
33
#include "private/tree.h"
34
#include "private/xinclude.h"
35
36
0
#define XINCLUDE_MAX_DEPTH 40
37
38
/************************************************************************
39
 *                  *
40
 *      XInclude context handling     *
41
 *                  *
42
 ************************************************************************/
43
44
/*
45
 * An XInclude context
46
 */
47
typedef xmlChar *xmlURL;
48
49
typedef struct _xmlXIncludeRef xmlXIncludeRef;
50
typedef xmlXIncludeRef *xmlXIncludeRefPtr;
51
struct _xmlXIncludeRef {
52
    xmlChar              *URI; /* the fully resolved resource URL */
53
    xmlChar         *fragment; /* the fragment in the URI */
54
    xmlChar             *base; /* base URI of xi:include element */
55
    xmlNodePtr           elem; /* the xi:include element */
56
    xmlNodePtr            inc; /* the included copy */
57
    int                   xml; /* xml or txt */
58
    int              fallback; /* fallback was loaded */
59
    int       expanding; /* flag to detect inclusion loops */
60
    int         replace; /* should the node be replaced? */
61
};
62
63
typedef struct _xmlXIncludeDoc xmlXIncludeDoc;
64
typedef xmlXIncludeDoc *xmlXIncludeDocPtr;
65
struct _xmlXIncludeDoc {
66
    xmlDocPtr             doc; /* the parsed document */
67
    xmlChar              *url; /* the URL */
68
    int             expanding; /* flag to detect inclusion loops */
69
};
70
71
typedef struct _xmlXIncludeTxt xmlXIncludeTxt;
72
typedef xmlXIncludeTxt *xmlXIncludeTxtPtr;
73
struct _xmlXIncludeTxt {
74
    xmlChar   *text; /* text string */
75
    xmlChar              *url; /* the URL */
76
};
77
78
struct _xmlXIncludeCtxt {
79
    xmlDocPtr             doc; /* the source document */
80
    int                 incNr; /* number of includes */
81
    int                incMax; /* size of includes tab */
82
    xmlXIncludeRefPtr *incTab; /* array of included references */
83
84
    int                 txtNr; /* number of unparsed documents */
85
    int                txtMax; /* size of unparsed documents tab */
86
    xmlXIncludeTxt    *txtTab; /* array of unparsed documents */
87
88
    int                 urlNr; /* number of documents stacked */
89
    int                urlMax; /* size of document stack */
90
    xmlXIncludeDoc    *urlTab; /* document stack */
91
92
    int              nbErrors; /* the number of errors detected */
93
    int              fatalErr; /* abort processing */
94
    int                 errNo; /* error code */
95
    int                legacy; /* using XINCLUDE_OLD_NS */
96
    int            parseFlags; /* the flags used for parsing XML documents */
97
98
    void            *_private; /* application data */
99
100
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
101
    unsigned long    incTotal; /* total number of processed inclusions */
102
#endif
103
    int     depth; /* recursion depth */
104
    int        isStream; /* streaming mode */
105
106
#ifdef LIBXML_XPTR_ENABLED
107
    xmlXPathContextPtr xpctxt;
108
#endif
109
110
    xmlStructuredErrorFunc errorHandler;
111
    void *errorCtxt;
112
113
    xmlResourceLoader resourceLoader;
114
    void *resourceCtxt;
115
};
116
117
static xmlXIncludeRefPtr
118
xmlXIncludeExpandNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node);
119
120
static int
121
xmlXIncludeLoadNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref);
122
123
static int
124
xmlXIncludeDoProcess(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree);
125
126
127
/************************************************************************
128
 *                  *
129
 *      XInclude error handler        *
130
 *                  *
131
 ************************************************************************/
132
133
/**
134
 * Handle an out of memory condition
135
 *
136
 * @param ctxt  an XInclude context
137
 */
138
static void
139
xmlXIncludeErrMemory(xmlXIncludeCtxtPtr ctxt)
140
0
{
141
0
    ctxt->errNo = XML_ERR_NO_MEMORY;
142
0
    ctxt->fatalErr = 1;
143
0
    ctxt->nbErrors++;
144
145
0
    xmlRaiseMemoryError(ctxt->errorHandler, NULL, ctxt->errorCtxt,
146
0
                        XML_FROM_XINCLUDE, NULL);
147
0
}
148
149
/**
150
 * Handle an XInclude error
151
 *
152
 * @param ctxt  the XInclude context
153
 * @param node  the context node
154
 * @param error  the error code
155
 * @param msg  the error message
156
 * @param extra  extra information
157
 */
158
static void LIBXML_ATTR_FORMAT(4,0)
159
xmlXIncludeErr(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node, int error,
160
               const char *msg, const xmlChar *extra)
161
0
{
162
0
    xmlStructuredErrorFunc schannel = NULL;
163
0
    xmlGenericErrorFunc channel = NULL;
164
0
    void *data = NULL;
165
0
    int res;
166
167
0
    if (error == XML_ERR_NO_MEMORY) {
168
0
        xmlXIncludeErrMemory(ctxt);
169
0
        return;
170
0
    }
171
172
0
    if (ctxt->fatalErr != 0)
173
0
        return;
174
0
    ctxt->nbErrors++;
175
176
0
    schannel = ctxt->errorHandler;
177
0
    data = ctxt->errorCtxt;
178
179
0
    if (schannel == NULL) {
180
0
        channel = xmlGenericError;
181
0
        data = xmlGenericErrorContext;
182
0
    }
183
184
0
    res = xmlRaiseError(schannel, channel, data, ctxt, node,
185
0
                        XML_FROM_XINCLUDE, error, XML_ERR_ERROR,
186
0
                        NULL, 0, (const char *) extra, NULL, NULL, 0, 0,
187
0
                        msg, (const char *) extra);
188
0
    if (res < 0) {
189
0
        ctxt->errNo = XML_ERR_NO_MEMORY;
190
0
        ctxt->fatalErr = 1;
191
0
    } else {
192
0
        ctxt->errNo = error;
193
        /*
194
         * Note that we treat IO errors except ENOENT as fatal
195
         * although the XInclude spec could be interpreted in a
196
         * way that at least some IO errors should be handled
197
         * gracefully.
198
         */
199
0
        if (xmlIsCatastrophicError(XML_ERR_FATAL, error))
200
0
            ctxt->fatalErr = 1;
201
0
    }
202
0
}
203
204
/**
205
 * Get an XInclude attribute
206
 *
207
 * @param ctxt  the XInclude context
208
 * @param cur  the node
209
 * @param name  the attribute name
210
 * @returns the value (to be freed) or NULL if not found
211
 */
212
static xmlChar *
213
xmlXIncludeGetProp(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur,
214
0
                   const xmlChar *name) {
215
0
    xmlChar *ret;
216
217
0
    if (xmlNodeGetAttrValue(cur, name, XINCLUDE_NS, &ret) < 0)
218
0
        xmlXIncludeErrMemory(ctxt);
219
0
    if (ret != NULL)
220
0
        return(ret);
221
222
0
    if (ctxt->legacy != 0) {
223
0
        if (xmlNodeGetAttrValue(cur, name, XINCLUDE_OLD_NS, &ret) < 0)
224
0
            xmlXIncludeErrMemory(ctxt);
225
0
        if (ret != NULL)
226
0
            return(ret);
227
0
    }
228
229
0
    if (xmlNodeGetAttrValue(cur, name, NULL, &ret) < 0)
230
0
        xmlXIncludeErrMemory(ctxt);
231
0
    return(ret);
232
0
}
233
/**
234
 * Free an XInclude reference
235
 *
236
 * @param ref  the XInclude reference
237
 */
238
static void
239
0
xmlXIncludeFreeRef(xmlXIncludeRefPtr ref) {
240
0
    if (ref == NULL)
241
0
  return;
242
0
    if (ref->URI != NULL)
243
0
  xmlFree(ref->URI);
244
0
    if (ref->fragment != NULL)
245
0
  xmlFree(ref->fragment);
246
0
    if (ref->base != NULL)
247
0
  xmlFree(ref->base);
248
0
    xmlFree(ref);
249
0
}
250
251
/**
252
 * Creates a new XInclude context
253
 *
254
 * @param doc  an XML Document
255
 * @returns the new set
256
 */
257
xmlXIncludeCtxt *
258
0
xmlXIncludeNewContext(xmlDoc *doc) {
259
0
    xmlXIncludeCtxtPtr ret;
260
261
0
    if (doc == NULL)
262
0
  return(NULL);
263
0
    ret = (xmlXIncludeCtxtPtr) xmlMalloc(sizeof(xmlXIncludeCtxt));
264
0
    if (ret == NULL)
265
0
  return(NULL);
266
0
    memset(ret, 0, sizeof(xmlXIncludeCtxt));
267
0
    ret->doc = doc;
268
0
    ret->incNr = 0;
269
0
    ret->incMax = 0;
270
0
    ret->incTab = NULL;
271
0
    ret->nbErrors = 0;
272
0
    return(ret);
273
0
}
274
275
/**
276
 * Free an XInclude context
277
 *
278
 * @param ctxt  the XInclude context
279
 */
280
void
281
0
xmlXIncludeFreeContext(xmlXIncludeCtxt *ctxt) {
282
0
    int i;
283
284
0
    if (ctxt == NULL)
285
0
  return;
286
0
    if (ctxt->urlTab != NULL) {
287
0
  for (i = 0; i < ctxt->urlNr; i++) {
288
0
      xmlFreeDoc(ctxt->urlTab[i].doc);
289
0
      xmlFree(ctxt->urlTab[i].url);
290
0
  }
291
0
  xmlFree(ctxt->urlTab);
292
0
    }
293
0
    for (i = 0;i < ctxt->incNr;i++) {
294
0
  if (ctxt->incTab[i] != NULL)
295
0
      xmlXIncludeFreeRef(ctxt->incTab[i]);
296
0
    }
297
0
    if (ctxt->incTab != NULL)
298
0
  xmlFree(ctxt->incTab);
299
0
    if (ctxt->txtTab != NULL) {
300
0
  for (i = 0;i < ctxt->txtNr;i++) {
301
0
      xmlFree(ctxt->txtTab[i].text);
302
0
      xmlFree(ctxt->txtTab[i].url);
303
0
  }
304
0
  xmlFree(ctxt->txtTab);
305
0
    }
306
0
#ifdef LIBXML_XPTR_ENABLED
307
0
    if (ctxt->xpctxt != NULL)
308
0
  xmlXPathFreeContext(ctxt->xpctxt);
309
0
#endif
310
0
    xmlFree(ctxt);
311
0
}
312
313
/**
314
 * parse a document for XInclude
315
 *
316
 * @param ctxt  the XInclude context
317
 * @param URL  the URL or file path
318
 */
319
static xmlDocPtr
320
0
xmlXIncludeParseFile(xmlXIncludeCtxtPtr ctxt, const char *URL) {
321
0
    xmlDocPtr ret = NULL;
322
0
    xmlParserCtxtPtr pctxt;
323
0
    xmlParserInputPtr inputStream;
324
325
0
    xmlInitParser();
326
327
0
    pctxt = xmlNewParserCtxt();
328
0
    if (pctxt == NULL) {
329
0
  xmlXIncludeErrMemory(ctxt);
330
0
  return(NULL);
331
0
    }
332
0
    if (ctxt->errorHandler != NULL)
333
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
334
0
    if (ctxt->resourceLoader != NULL)
335
0
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
336
0
                                 ctxt->resourceCtxt);
337
338
    /*
339
     * pass in the application data to the parser context.
340
     */
341
0
    pctxt->_private = ctxt->_private;
342
343
    /*
344
     * try to ensure that new documents included are actually
345
     * built with the same dictionary as the including document.
346
     */
347
0
    if ((ctxt->doc != NULL) && (ctxt->doc->dict != NULL)) {
348
0
       if (pctxt->dict != NULL)
349
0
            xmlDictFree(pctxt->dict);
350
0
  pctxt->dict = ctxt->doc->dict;
351
0
  xmlDictReference(pctxt->dict);
352
0
    }
353
354
    /*
355
     * We set DTDLOAD to make sure that ID attributes declared in
356
     * external DTDs are detected.
357
     */
358
0
    xmlCtxtUseOptions(pctxt, ctxt->parseFlags | XML_PARSE_DTDLOAD);
359
360
0
    inputStream = xmlLoadResource(pctxt, URL, NULL, XML_RESOURCE_XINCLUDE);
361
0
    if (inputStream == NULL)
362
0
        goto error;
363
364
0
    if (xmlCtxtPushInput(pctxt, inputStream) < 0) {
365
0
        xmlFreeInputStream(inputStream);
366
0
        goto error;
367
0
    }
368
369
0
    xmlParseDocument(pctxt);
370
371
0
    if (pctxt->wellFormed) {
372
0
        ret = pctxt->myDoc;
373
0
    }
374
0
    else {
375
0
        ret = NULL;
376
0
  if (pctxt->myDoc != NULL)
377
0
      xmlFreeDoc(pctxt->myDoc);
378
0
        pctxt->myDoc = NULL;
379
0
    }
380
381
0
error:
382
0
    if (xmlCtxtIsCatastrophicError(pctxt))
383
0
        xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "parser error", NULL);
384
0
    xmlFreeParserCtxt(pctxt);
385
386
0
    return(ret);
387
0
}
388
389
/**
390
 * Add a new node to process to an XInclude context
391
 *
392
 * @param ctxt  the XInclude context
393
 * @param cur  the new node
394
 */
395
static xmlXIncludeRefPtr
396
0
xmlXIncludeAddNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur) {
397
0
    xmlXIncludeRefPtr ref = NULL;
398
0
    xmlXIncludeRefPtr ret = NULL;
399
0
    xmlURIPtr uri = NULL;
400
0
    xmlChar *href = NULL;
401
0
    xmlChar *parse = NULL;
402
0
    xmlChar *fragment = NULL;
403
0
    xmlChar *base = NULL;
404
0
    xmlChar *tmp;
405
0
    int xml = 1;
406
0
    int local = 0;
407
0
    int res;
408
409
0
    if (ctxt == NULL)
410
0
  return(NULL);
411
0
    if (cur == NULL)
412
0
  return(NULL);
413
414
    /*
415
     * read the attributes
416
     */
417
418
0
    fragment = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE_XPOINTER);
419
420
0
    href = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_HREF);
421
0
    if (href == NULL) {
422
0
        if (fragment == NULL) {
423
0
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_HREF,
424
0
                     "href or xpointer must be present\n", parse);
425
0
      goto error;
426
0
        }
427
428
0
  href = xmlStrdup(BAD_CAST ""); /* @@@@ href is now optional */
429
0
  if (href == NULL) {
430
0
            xmlXIncludeErrMemory(ctxt);
431
0
      goto error;
432
0
        }
433
0
    } else if (xmlStrlen(href) > XML_MAX_URI_LENGTH) {
434
0
        xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI, "URI too long\n",
435
0
                       NULL);
436
0
        goto error;
437
0
    }
438
439
0
    parse = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE);
440
0
    if (parse != NULL) {
441
0
  if (xmlStrEqual(parse, XINCLUDE_PARSE_XML))
442
0
      xml = 1;
443
0
  else if (xmlStrEqual(parse, XINCLUDE_PARSE_TEXT))
444
0
      xml = 0;
445
0
  else {
446
0
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_PARSE_VALUE,
447
0
                     "invalid value %s for 'parse'\n", parse);
448
0
      goto error;
449
0
  }
450
0
    }
451
452
    /*
453
     * Check the URL and remove any fragment identifier
454
     */
455
0
    res = xmlParseURISafe((const char *)href, &uri);
456
0
    if (uri == NULL) {
457
0
        if (res < 0)
458
0
            xmlXIncludeErrMemory(ctxt);
459
0
        else
460
0
            xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI,
461
0
                           "invalid value href %s\n", href);
462
0
        goto error;
463
0
    }
464
465
0
    if (uri->fragment != NULL) {
466
0
        if (ctxt->legacy != 0) {
467
0
      if (fragment == NULL) {
468
0
    fragment = (xmlChar *) uri->fragment;
469
0
      } else {
470
0
    xmlFree(uri->fragment);
471
0
      }
472
0
  } else {
473
0
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_FRAGMENT_ID,
474
0
       "Invalid fragment identifier in URI %s use the xpointer attribute\n",
475
0
                           href);
476
0
      goto error;
477
0
  }
478
0
  uri->fragment = NULL;
479
0
    }
480
0
    tmp = xmlSaveUri(uri);
481
0
    if (tmp == NULL) {
482
0
  xmlXIncludeErrMemory(ctxt);
483
0
  goto error;
484
0
    }
485
0
    xmlFree(href);
486
0
    href = tmp;
487
488
    /*
489
     * Resolve URI
490
     */
491
492
0
    if (xmlNodeGetBaseSafe(ctxt->doc, cur, &base) < 0) {
493
0
        xmlXIncludeErrMemory(ctxt);
494
0
        goto error;
495
0
    }
496
497
0
    if (href[0] != 0) {
498
0
        if (xmlBuildURISafe(href, base, &tmp) < 0) {
499
0
            xmlXIncludeErrMemory(ctxt);
500
0
            goto error;
501
0
        }
502
0
        if (tmp == NULL) {
503
0
            xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI,
504
0
                           "failed build URL\n", NULL);
505
0
            goto error;
506
0
        }
507
0
        xmlFree(href);
508
0
        href = tmp;
509
510
0
        if (xmlStrEqual(href, ctxt->doc->URL))
511
0
            local = 1;
512
0
    } else {
513
0
        local = 1;
514
0
    }
515
516
    /*
517
     * If local and xml then we need a fragment
518
     */
519
0
    if ((local == 1) && (xml == 1) &&
520
0
        ((fragment == NULL) || (fragment[0] == 0))) {
521
0
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_RECURSION,
522
0
                 "detected a local recursion with no xpointer in %s\n",
523
0
           href);
524
0
  goto error;
525
0
    }
526
527
0
    ref = (xmlXIncludeRefPtr) xmlMalloc(sizeof(xmlXIncludeRef));
528
0
    if (ref == NULL) {
529
0
        xmlXIncludeErrMemory(ctxt);
530
0
        goto error;
531
0
    }
532
0
    memset(ref, 0, sizeof(xmlXIncludeRef));
533
534
0
    ref->elem = cur;
535
0
    ref->xml = xml;
536
0
    ref->URI = href;
537
0
    href = NULL;
538
0
    ref->fragment = fragment;
539
0
    fragment = NULL;
540
541
    /*
542
     * xml:base fixup
543
     */
544
0
    if (((ctxt->parseFlags & XML_PARSE_NOBASEFIX) == 0) &&
545
0
        (cur->doc != NULL) &&
546
0
        ((cur->doc->parseFlags & XML_PARSE_NOBASEFIX) == 0)) {
547
0
        if (base != NULL) {
548
0
            ref->base = base;
549
0
            base = NULL;
550
0
        } else {
551
0
            ref->base = xmlStrdup(BAD_CAST "");
552
0
            if (ref->base == NULL) {
553
0
          xmlXIncludeErrMemory(ctxt);
554
0
                goto error;
555
0
            }
556
0
        }
557
0
    }
558
559
0
    if (ctxt->incNr >= ctxt->incMax) {
560
0
        xmlXIncludeRefPtr *table;
561
0
        int newSize;
562
563
0
        newSize = xmlGrowCapacity(ctxt->incMax, sizeof(table[0]),
564
0
                                  4, XML_MAX_ITEMS);
565
0
        if (newSize < 0) {
566
0
      xmlXIncludeErrMemory(ctxt);
567
0
      goto error;
568
0
  }
569
0
        table = xmlRealloc(ctxt->incTab, newSize * sizeof(table[0]));
570
0
        if (table == NULL) {
571
0
      xmlXIncludeErrMemory(ctxt);
572
0
      goto error;
573
0
  }
574
0
        ctxt->incTab = table;
575
0
        ctxt->incMax = newSize;
576
0
    }
577
0
    ctxt->incTab[ctxt->incNr++] = ref;
578
579
0
    ret = ref;
580
0
    ref = NULL;
581
582
0
error:
583
0
    xmlXIncludeFreeRef(ref);
584
0
    xmlFreeURI(uri);
585
0
    xmlFree(href);
586
0
    xmlFree(parse);
587
0
    xmlFree(fragment);
588
0
    xmlFree(base);
589
0
    return(ret);
590
0
}
591
592
/**
593
 * The XInclude recursive nature is handled at this point.
594
 *
595
 * @param ctxt  the XInclude context
596
 * @param doc  the new document
597
 */
598
static void
599
0
xmlXIncludeRecurseDoc(xmlXIncludeCtxtPtr ctxt, xmlDocPtr doc) {
600
0
    xmlDocPtr oldDoc;
601
0
    xmlXIncludeRefPtr *oldIncTab;
602
0
    int oldIncMax, oldIncNr, oldIsStream;
603
0
    int i;
604
605
0
    oldDoc = ctxt->doc;
606
0
    oldIncMax = ctxt->incMax;
607
0
    oldIncNr = ctxt->incNr;
608
0
    oldIncTab = ctxt->incTab;
609
0
    oldIsStream = ctxt->isStream;
610
0
    ctxt->doc = doc;
611
0
    ctxt->incMax = 0;
612
0
    ctxt->incNr = 0;
613
0
    ctxt->incTab = NULL;
614
0
    ctxt->isStream = 0;
615
616
0
    xmlXIncludeDoProcess(ctxt, xmlDocGetRootElement(doc));
617
618
0
    if (ctxt->incTab != NULL) {
619
0
        for (i = 0; i < ctxt->incNr; i++)
620
0
            xmlXIncludeFreeRef(ctxt->incTab[i]);
621
0
        xmlFree(ctxt->incTab);
622
0
    }
623
624
0
    ctxt->doc = oldDoc;
625
0
    ctxt->incMax = oldIncMax;
626
0
    ctxt->incNr = oldIncNr;
627
0
    ctxt->incTab = oldIncTab;
628
0
    ctxt->isStream = oldIsStream;
629
0
}
630
631
/************************************************************************
632
 *                  *
633
 *      Node copy with specific semantic    *
634
 *                  *
635
 ************************************************************************/
636
637
static void
638
xmlXIncludeBaseFixup(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur, xmlNodePtr copy,
639
0
                     const xmlChar *targetBase) {
640
0
    xmlChar *base = NULL;
641
0
    xmlChar *relBase = NULL;
642
0
    xmlNs ns;
643
0
    int res;
644
645
0
    if (cur->type != XML_ELEMENT_NODE)
646
0
        return;
647
648
0
    if (xmlNodeGetBaseSafe(cur->doc, cur, &base) < 0)
649
0
        xmlXIncludeErrMemory(ctxt);
650
651
0
    if ((base != NULL) && !xmlStrEqual(base, targetBase)) {
652
0
        if ((xmlStrlen(base) > XML_MAX_URI_LENGTH) ||
653
0
            (xmlStrlen(targetBase) > XML_MAX_URI_LENGTH)) {
654
0
            relBase = xmlStrdup(base);
655
0
            if (relBase == NULL) {
656
0
                xmlXIncludeErrMemory(ctxt);
657
0
                goto done;
658
0
            }
659
0
        } else if (xmlBuildRelativeURISafe(base, targetBase, &relBase) < 0) {
660
0
            xmlXIncludeErrMemory(ctxt);
661
0
            goto done;
662
0
        }
663
0
        if (relBase == NULL) {
664
0
            xmlXIncludeErr(ctxt, cur,
665
0
                    XML_XINCLUDE_HREF_URI,
666
0
                    "Building relative URI failed: %s\n",
667
0
                    base);
668
0
            goto done;
669
0
        }
670
671
        /*
672
         * If the new base doesn't contain a slash, it can be omitted.
673
         */
674
0
        if (xmlStrchr(relBase, '/') != NULL) {
675
0
            res = xmlNodeSetBase(copy, relBase);
676
0
            if (res < 0)
677
0
                xmlXIncludeErrMemory(ctxt);
678
0
            goto done;
679
0
        }
680
0
    }
681
682
    /*
683
     * Delete existing xml:base if bases are equal
684
     */
685
0
    memset(&ns, 0, sizeof(ns));
686
0
    ns.href = XML_XML_NAMESPACE;
687
0
    xmlUnsetNsProp(copy, &ns, BAD_CAST "base");
688
689
0
done:
690
0
    xmlFree(base);
691
0
    xmlFree(relBase);
692
0
}
693
694
/**
695
 * Make a copy of the node while expanding nested XIncludes.
696
 *
697
 * @param ctxt  the XInclude context
698
 * @param elem  the element
699
 * @param copyChildren  copy children instead of node if true
700
 * @param targetBase  the xml:base of the target node
701
 * @returns a node list, not a single node.
702
 */
703
static xmlNodePtr
704
xmlXIncludeCopyNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr elem,
705
0
                    int copyChildren, const xmlChar *targetBase) {
706
0
    xmlNodePtr result = NULL;
707
0
    xmlNodePtr insertParent = NULL;
708
0
    xmlNodePtr insertLast = NULL;
709
0
    xmlNodePtr cur;
710
0
    xmlNodePtr item;
711
0
    int depth = 0;
712
713
0
    if (copyChildren) {
714
0
        cur = elem->children;
715
0
        if (cur == NULL)
716
0
            return(NULL);
717
0
    } else {
718
0
        cur = elem;
719
0
    }
720
721
0
    while (1) {
722
0
        xmlNodePtr copy = NULL;
723
0
        int recurse = 0;
724
725
0
        if ((cur->type == XML_DOCUMENT_NODE) ||
726
0
            (cur->type == XML_DTD_NODE)) {
727
0
            ;
728
0
        } else if ((cur->type == XML_ELEMENT_NODE) &&
729
0
                   (cur->ns != NULL) &&
730
0
                   (xmlStrEqual(cur->name, XINCLUDE_NODE)) &&
731
0
                   ((xmlStrEqual(cur->ns->href, XINCLUDE_NS)) ||
732
0
                    (xmlStrEqual(cur->ns->href, XINCLUDE_OLD_NS)))) {
733
0
            xmlXIncludeRefPtr ref = xmlXIncludeExpandNode(ctxt, cur);
734
735
0
            if (ref == NULL)
736
0
                goto error;
737
            /*
738
             * TODO: Insert XML_XINCLUDE_START and XML_XINCLUDE_END nodes
739
             */
740
0
            for (item = ref->inc; item != NULL; item = item->next) {
741
0
                copy = xmlStaticCopyNode(item, ctxt->doc, insertParent, 1);
742
0
                if (copy == NULL) {
743
0
                    xmlXIncludeErrMemory(ctxt);
744
0
                    goto error;
745
0
                }
746
747
0
                if (result == NULL)
748
0
                    result = copy;
749
0
                if (insertLast != NULL) {
750
0
                    insertLast->next = copy;
751
0
                    copy->prev = insertLast;
752
0
                } else if (insertParent != NULL) {
753
0
                    insertParent->children = copy;
754
0
                }
755
0
                insertLast = copy;
756
757
0
                if ((depth == 0) && (targetBase != NULL))
758
0
                    xmlXIncludeBaseFixup(ctxt, item, copy, targetBase);
759
0
            }
760
0
        } else {
761
0
            copy = xmlStaticCopyNode(cur, ctxt->doc, insertParent, 2);
762
0
            if (copy == NULL) {
763
0
                xmlXIncludeErrMemory(ctxt);
764
0
                goto error;
765
0
            }
766
767
0
            if (result == NULL)
768
0
                result = copy;
769
0
            if (insertLast != NULL) {
770
0
                insertLast->next = copy;
771
0
                copy->prev = insertLast;
772
0
            } else if (insertParent != NULL) {
773
0
                insertParent->children = copy;
774
0
            }
775
0
            insertLast = copy;
776
777
0
            if ((depth == 0) && (targetBase != NULL))
778
0
                xmlXIncludeBaseFixup(ctxt, cur, copy, targetBase);
779
780
0
            recurse = (cur->type != XML_ENTITY_REF_NODE) &&
781
0
                      (cur->children != NULL);
782
0
        }
783
784
0
        if (recurse) {
785
0
            cur = cur->children;
786
0
            insertParent = insertLast;
787
0
            insertLast = NULL;
788
0
            depth += 1;
789
0
            continue;
790
0
        }
791
792
0
        if (cur == elem)
793
0
            return(result);
794
795
0
        while (cur->next == NULL) {
796
0
            if (insertParent != NULL)
797
0
                insertParent->last = insertLast;
798
0
            cur = cur->parent;
799
0
            if (cur == elem)
800
0
                return(result);
801
0
            insertLast = insertParent;
802
0
            insertParent = insertParent->parent;
803
0
            depth -= 1;
804
0
        }
805
806
0
        cur = cur->next;
807
0
    }
808
809
0
error:
810
0
    xmlFreeNodeList(result);
811
0
    return(NULL);
812
0
}
813
814
#ifdef LIBXML_XPTR_ENABLED
815
/**
816
 * Build a node list tree copy of the XPointer result.
817
 * This will drop Attributes and Namespace declarations.
818
 *
819
 * @param ctxt  the XInclude context
820
 * @param obj  the XPointer result from the evaluation.
821
 * @param targetBase  the xml:base of the target node
822
 * @returns an xmlNode list or NULL.
823
 *         the caller has to free the node tree.
824
 */
825
static xmlNodePtr
826
xmlXIncludeCopyXPointer(xmlXIncludeCtxtPtr ctxt, xmlXPathObjectPtr obj,
827
0
                        const xmlChar *targetBase) {
828
0
    xmlNodePtr list = NULL, last = NULL, copy;
829
0
    int i;
830
831
0
    if ((ctxt == NULL) || (obj == NULL))
832
0
  return(NULL);
833
0
    switch (obj->type) {
834
0
        case XPATH_NODESET: {
835
0
      xmlNodeSetPtr set = obj->nodesetval;
836
0
      if (set == NULL)
837
0
    break;
838
0
      for (i = 0;i < set->nodeNr;i++) {
839
0
                xmlNodePtr node;
840
841
0
    if (set->nodeTab[i] == NULL)
842
0
        continue;
843
0
    switch (set->nodeTab[i]->type) {
844
0
        case XML_DOCUMENT_NODE:
845
0
        case XML_HTML_DOCUMENT_NODE:
846
0
                        node = xmlDocGetRootElement(
847
0
                                (xmlDocPtr) set->nodeTab[i]);
848
0
                        if (node == NULL) {
849
0
                            xmlXIncludeErr(ctxt, set->nodeTab[i],
850
0
                                           XML_ERR_INTERNAL_ERROR,
851
0
                                          "document without root\n", NULL);
852
0
                            continue;
853
0
                        }
854
0
                        break;
855
0
                    case XML_TEXT_NODE:
856
0
        case XML_CDATA_SECTION_NODE:
857
0
        case XML_ELEMENT_NODE:
858
0
        case XML_PI_NODE:
859
0
        case XML_COMMENT_NODE:
860
0
                        node = set->nodeTab[i];
861
0
      break;
862
0
                    default:
863
0
                        xmlXIncludeErr(ctxt, set->nodeTab[i],
864
0
                                       XML_XINCLUDE_XPTR_RESULT,
865
0
                                       "invalid node type in XPtr result\n",
866
0
                                       NULL);
867
0
      continue; /* for */
868
0
    }
869
                /*
870
                 * OPTIMIZE TODO: External documents should already be
871
                 * expanded, so xmlDocCopyNode should work as well.
872
                 * xmlXIncludeCopyNode is only required for the initial
873
                 * document.
874
                 */
875
0
    copy = xmlXIncludeCopyNode(ctxt, node, 0, targetBase);
876
0
                if (copy == NULL) {
877
0
                    xmlFreeNodeList(list);
878
0
                    return(NULL);
879
0
                }
880
0
    if (last == NULL) {
881
0
                    list = copy;
882
0
                } else {
883
0
                    while (last->next != NULL)
884
0
                        last = last->next;
885
0
                    copy->prev = last;
886
0
                    last->next = copy;
887
0
    }
888
0
                last = copy;
889
0
      }
890
0
      break;
891
0
  }
892
0
  default:
893
0
      break;
894
0
    }
895
0
    return(list);
896
0
}
897
#endif
898
899
/************************************************************************
900
 *                  *
901
 *      XInclude I/O handling       *
902
 *                  *
903
 ************************************************************************/
904
905
typedef struct _xmlXIncludeMergeData xmlXIncludeMergeData;
906
typedef xmlXIncludeMergeData *xmlXIncludeMergeDataPtr;
907
struct _xmlXIncludeMergeData {
908
    xmlDocPtr doc;
909
    xmlXIncludeCtxtPtr ctxt;
910
};
911
912
/**
913
 * Implements the merge of one entity
914
 *
915
 * @param payload  the entity
916
 * @param vdata  the merge data
917
 * @param name  unused
918
 */
919
static void
920
xmlXIncludeMergeEntity(void *payload, void *vdata,
921
0
                 const xmlChar *name ATTRIBUTE_UNUSED) {
922
0
    xmlEntityPtr ent = (xmlEntityPtr) payload;
923
0
    xmlXIncludeMergeDataPtr data = (xmlXIncludeMergeDataPtr) vdata;
924
0
    xmlEntityPtr ret, prev;
925
0
    xmlDocPtr doc;
926
0
    xmlXIncludeCtxtPtr ctxt;
927
928
0
    if ((ent == NULL) || (data == NULL))
929
0
  return;
930
0
    ctxt = data->ctxt;
931
0
    doc = data->doc;
932
0
    if ((ctxt == NULL) || (doc == NULL))
933
0
  return;
934
0
    switch (ent->etype) {
935
0
        case XML_INTERNAL_PARAMETER_ENTITY:
936
0
        case XML_EXTERNAL_PARAMETER_ENTITY:
937
0
        case XML_INTERNAL_PREDEFINED_ENTITY:
938
0
      return;
939
0
        case XML_INTERNAL_GENERAL_ENTITY:
940
0
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
941
0
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
942
0
      break;
943
0
    }
944
0
    prev = xmlGetDocEntity(doc, ent->name);
945
0
    if (prev == NULL) {
946
0
        ret = xmlAddDocEntity(doc, ent->name, ent->etype, ent->ExternalID,
947
0
                              ent->SystemID, ent->content);
948
0
        if (ret == NULL) {
949
0
            xmlXIncludeErrMemory(ctxt);
950
0
            return;
951
0
        }
952
0
  if (ent->URI != NULL) {
953
0
      ret->URI = xmlStrdup(ent->URI);
954
0
            if (ret->URI == 0)
955
0
                xmlXIncludeErrMemory(ctxt);
956
0
        }
957
0
    } else {
958
0
        if (ent->etype != prev->etype)
959
0
            goto error;
960
961
0
        if ((ent->SystemID != NULL) && (prev->SystemID != NULL)) {
962
0
            if (!xmlStrEqual(ent->SystemID, prev->SystemID))
963
0
                goto error;
964
0
        } else if ((ent->ExternalID != NULL) &&
965
0
                   (prev->ExternalID != NULL)) {
966
0
            if (!xmlStrEqual(ent->ExternalID, prev->ExternalID))
967
0
                goto error;
968
0
        } else if ((ent->content != NULL) && (prev->content != NULL)) {
969
0
            if (!xmlStrEqual(ent->content, prev->content))
970
0
                goto error;
971
0
        } else {
972
0
            goto error;
973
0
        }
974
0
    }
975
0
    return;
976
0
error:
977
0
    switch (ent->etype) {
978
0
        case XML_INTERNAL_PARAMETER_ENTITY:
979
0
        case XML_EXTERNAL_PARAMETER_ENTITY:
980
0
        case XML_INTERNAL_PREDEFINED_ENTITY:
981
0
        case XML_INTERNAL_GENERAL_ENTITY:
982
0
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
983
0
      return;
984
0
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
985
0
      break;
986
0
    }
987
0
    xmlXIncludeErr(ctxt, (xmlNodePtr) ent, XML_XINCLUDE_ENTITY_DEF_MISMATCH,
988
0
                   "mismatch in redefinition of entity %s\n",
989
0
       ent->name);
990
0
}
991
992
/**
993
 * Implements the entity merge
994
 *
995
 * @param ctxt  an XInclude context
996
 * @param doc  the including doc
997
 * @param from  the included doc
998
 * @returns 0 if merge succeeded, -1 if some processing failed
999
 */
1000
static int
1001
xmlXIncludeMergeEntities(xmlXIncludeCtxtPtr ctxt, xmlDocPtr doc,
1002
0
                   xmlDocPtr from) {
1003
0
    xmlNodePtr cur;
1004
0
    xmlDtdPtr target, source;
1005
1006
0
    if (ctxt == NULL)
1007
0
  return(-1);
1008
1009
0
    if ((from == NULL) || (from->intSubset == NULL))
1010
0
  return(0);
1011
1012
0
    target = doc->intSubset;
1013
0
    if (target == NULL) {
1014
0
  cur = xmlDocGetRootElement(doc);
1015
0
  if (cur == NULL)
1016
0
      return(-1);
1017
0
        target = xmlCreateIntSubset(doc, cur->name, NULL, NULL);
1018
0
  if (target == NULL) {
1019
0
            xmlXIncludeErrMemory(ctxt);
1020
0
      return(-1);
1021
0
        }
1022
0
    }
1023
1024
0
    source = from->intSubset;
1025
0
    if ((source != NULL) && (source->entities != NULL)) {
1026
0
  xmlXIncludeMergeData data;
1027
1028
0
  data.ctxt = ctxt;
1029
0
  data.doc = doc;
1030
1031
0
  xmlHashScan((xmlHashTablePtr) source->entities,
1032
0
        xmlXIncludeMergeEntity, &data);
1033
0
    }
1034
0
    source = from->extSubset;
1035
0
    if ((source != NULL) && (source->entities != NULL)) {
1036
0
  xmlXIncludeMergeData data;
1037
1038
0
  data.ctxt = ctxt;
1039
0
  data.doc = doc;
1040
1041
  /*
1042
   * don't duplicate existing stuff when external subsets are the same
1043
   */
1044
0
  if ((!xmlStrEqual(target->ExternalID, source->ExternalID)) &&
1045
0
      (!xmlStrEqual(target->SystemID, source->SystemID))) {
1046
0
      xmlHashScan((xmlHashTablePtr) source->entities,
1047
0
      xmlXIncludeMergeEntity, &data);
1048
0
  }
1049
0
    }
1050
0
    return(0);
1051
0
}
1052
1053
/**
1054
 * Load the document, and store the result in the XInclude context
1055
 *
1056
 * @param ctxt  the XInclude context
1057
 * @param ref  an XMLXincludeRefPtr
1058
 * @returns 0 in case of success, -1 in case of failure
1059
 */
1060
static int
1061
0
xmlXIncludeLoadDoc(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1062
0
    xmlXIncludeDocPtr cache;
1063
0
    xmlDocPtr doc;
1064
0
    const xmlChar *url = ref->URI;
1065
0
    const xmlChar *fragment = ref->fragment;
1066
0
    int i = 0;
1067
0
    int ret = -1;
1068
0
    int cacheNr;
1069
0
#ifdef LIBXML_XPTR_ENABLED
1070
0
    int saveFlags;
1071
0
#endif
1072
1073
    /*
1074
     * Handling of references to the local document are done
1075
     * directly through ctxt->doc.
1076
     */
1077
0
    if ((url[0] == 0) || (url[0] == '#') ||
1078
0
  ((ctxt->doc != NULL) && (xmlStrEqual(url, ctxt->doc->URL)))) {
1079
0
  doc = ctxt->doc;
1080
0
        goto loaded;
1081
0
    }
1082
1083
    /*
1084
     * Prevent reloading the document twice.
1085
     */
1086
0
    for (i = 0; i < ctxt->urlNr; i++) {
1087
0
  if (xmlStrEqual(url, ctxt->urlTab[i].url)) {
1088
0
            if (ctxt->urlTab[i].expanding) {
1089
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_RECURSION,
1090
0
                               "inclusion loop detected\n", NULL);
1091
0
                goto error;
1092
0
            }
1093
0
      doc = ctxt->urlTab[i].doc;
1094
0
            if (doc == NULL)
1095
0
                goto error;
1096
0
      goto loaded;
1097
0
  }
1098
0
    }
1099
1100
    /*
1101
     * Load it.
1102
     */
1103
0
#ifdef LIBXML_XPTR_ENABLED
1104
    /*
1105
     * If this is an XPointer evaluation, we want to assure that
1106
     * all entities have been resolved prior to processing the
1107
     * referenced document
1108
     */
1109
0
    saveFlags = ctxt->parseFlags;
1110
0
    if (fragment != NULL) { /* if this is an XPointer eval */
1111
0
  ctxt->parseFlags |= XML_PARSE_NOENT;
1112
0
    }
1113
0
#endif
1114
1115
0
    doc = xmlXIncludeParseFile(ctxt, (const char *)url);
1116
0
#ifdef LIBXML_XPTR_ENABLED
1117
0
    ctxt->parseFlags = saveFlags;
1118
0
#endif
1119
1120
    /* Also cache NULL docs */
1121
0
    if (ctxt->urlNr >= ctxt->urlMax) {
1122
0
        xmlXIncludeDoc *tmp;
1123
0
        int newSize;
1124
1125
0
        newSize = xmlGrowCapacity(ctxt->urlMax, sizeof(tmp[0]),
1126
0
                                  8, XML_MAX_ITEMS);
1127
0
        if (newSize < 0) {
1128
0
            xmlXIncludeErrMemory(ctxt);
1129
0
            xmlFreeDoc(doc);
1130
0
            goto error;
1131
0
        }
1132
0
        tmp = xmlRealloc(ctxt->urlTab, newSize * sizeof(tmp[0]));
1133
0
        if (tmp == NULL) {
1134
0
            xmlXIncludeErrMemory(ctxt);
1135
0
            xmlFreeDoc(doc);
1136
0
            goto error;
1137
0
        }
1138
0
        ctxt->urlMax = newSize;
1139
0
        ctxt->urlTab = tmp;
1140
0
    }
1141
0
    cache = &ctxt->urlTab[ctxt->urlNr];
1142
0
    cache->doc = doc;
1143
0
    cache->url = xmlStrdup(url);
1144
0
    if (cache->url == NULL) {
1145
0
        xmlXIncludeErrMemory(ctxt);
1146
0
        xmlFreeDoc(doc);
1147
0
        goto error;
1148
0
    }
1149
0
    cache->expanding = 0;
1150
0
    cacheNr = ctxt->urlNr++;
1151
1152
0
    if (doc == NULL)
1153
0
        goto error;
1154
    /*
1155
     * It's possible that the requested URL has been mapped to a
1156
     * completely different location (e.g. through a catalog entry).
1157
     * To check for this, we compare the URL with that of the doc
1158
     * and change it if they disagree (bug 146988).
1159
     */
1160
0
    if ((doc->URL != NULL) && (!xmlStrEqual(url, doc->URL)))
1161
0
        url = doc->URL;
1162
1163
    /*
1164
     * Make sure we have all entities fixed up
1165
     */
1166
0
    xmlXIncludeMergeEntities(ctxt, ctxt->doc, doc);
1167
1168
    /*
1169
     * We don't need the DTD anymore, free up space
1170
    if (doc->intSubset != NULL) {
1171
  xmlUnlinkNode((xmlNodePtr) doc->intSubset);
1172
  xmlFreeNode((xmlNodePtr) doc->intSubset);
1173
  doc->intSubset = NULL;
1174
    }
1175
    if (doc->extSubset != NULL) {
1176
  xmlUnlinkNode((xmlNodePtr) doc->extSubset);
1177
  xmlFreeNode((xmlNodePtr) doc->extSubset);
1178
  doc->extSubset = NULL;
1179
    }
1180
     */
1181
0
    cache->expanding = 1;
1182
0
    xmlXIncludeRecurseDoc(ctxt, doc);
1183
    /* urlTab might be reallocated. */
1184
0
    cache = &ctxt->urlTab[cacheNr];
1185
0
    cache->expanding = 0;
1186
1187
0
loaded:
1188
0
    if (fragment == NULL) {
1189
0
        xmlNodePtr root;
1190
1191
0
        root = xmlDocGetRootElement(doc);
1192
0
        if (root == NULL) {
1193
0
            xmlXIncludeErr(ctxt, ref->elem, XML_ERR_INTERNAL_ERROR,
1194
0
                           "document without root\n", NULL);
1195
0
            goto error;
1196
0
        }
1197
1198
0
        ref->inc = xmlDocCopyNode(root, ctxt->doc, 1);
1199
0
        if (ref->inc == NULL) {
1200
0
            xmlXIncludeErrMemory(ctxt);
1201
0
            goto error;
1202
0
        }
1203
1204
0
        if (ref->base != NULL)
1205
0
            xmlXIncludeBaseFixup(ctxt, root, ref->inc, ref->base);
1206
0
    }
1207
0
#ifdef LIBXML_XPTR_ENABLED
1208
0
    else {
1209
  /*
1210
   * Computes the XPointer expression and make a copy used
1211
   * as the replacement copy.
1212
   */
1213
0
  xmlXPathObjectPtr xptr;
1214
0
  xmlNodeSetPtr set;
1215
1216
0
        if (ctxt->isStream && doc == ctxt->doc) {
1217
0
      xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_FAILED,
1218
0
         "XPointer expressions not allowed in streaming"
1219
0
                           " mode\n", NULL);
1220
0
            goto error;
1221
0
        }
1222
1223
0
        if (ctxt->xpctxt == NULL) {
1224
0
            ctxt->xpctxt = xmlXPathNewContext(doc);
1225
0
            if (ctxt->xpctxt == NULL) {
1226
0
                xmlXIncludeErrMemory(ctxt);
1227
0
                goto error;
1228
0
            }
1229
0
            if (ctxt->errorHandler != NULL)
1230
0
                xmlXPathSetErrorHandler(ctxt->xpctxt, ctxt->errorHandler,
1231
0
                                        ctxt->errorCtxt);
1232
0
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
1233
0
            ctxt->xpctxt->opLimit = 100000;
1234
0
#endif
1235
0
        } else {
1236
0
            ctxt->xpctxt->doc = doc;
1237
0
        }
1238
0
  xptr = xmlXPtrEval(fragment, ctxt->xpctxt);
1239
0
  if (ctxt->xpctxt->lastError.code != XML_ERR_OK) {
1240
0
            if (ctxt->xpctxt->lastError.code == XML_ERR_NO_MEMORY)
1241
0
                xmlXIncludeErrMemory(ctxt);
1242
0
            else
1243
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_FAILED,
1244
0
                               "XPointer evaluation failed: #%s\n",
1245
0
                               fragment);
1246
0
            goto error;
1247
0
  }
1248
0
        if (xptr == NULL)
1249
0
            goto done;
1250
0
  switch (xptr->type) {
1251
0
      case XPATH_UNDEFINED:
1252
0
      case XPATH_BOOLEAN:
1253
0
      case XPATH_NUMBER:
1254
0
      case XPATH_STRING:
1255
0
      case XPATH_USERS:
1256
0
      case XPATH_XSLT_TREE:
1257
0
    xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_RESULT,
1258
0
             "XPointer is not a range: #%s\n",
1259
0
             fragment);
1260
0
                xmlXPathFreeObject(xptr);
1261
0
                goto error;
1262
0
      case XPATH_NODESET:
1263
0
                break;
1264
1265
0
  }
1266
0
  set = xptr->nodesetval;
1267
0
  if (set != NULL) {
1268
0
      for (i = 0;i < set->nodeNr;i++) {
1269
0
    if (set->nodeTab[i] == NULL) /* shouldn't happen */
1270
0
        continue;
1271
0
    switch (set->nodeTab[i]->type) {
1272
0
        case XML_ELEMENT_NODE:
1273
0
        case XML_TEXT_NODE:
1274
0
        case XML_CDATA_SECTION_NODE:
1275
0
        case XML_ENTITY_REF_NODE:
1276
0
        case XML_ENTITY_NODE:
1277
0
        case XML_PI_NODE:
1278
0
        case XML_COMMENT_NODE:
1279
0
        case XML_DOCUMENT_NODE:
1280
0
        case XML_HTML_DOCUMENT_NODE:
1281
0
      continue;
1282
1283
0
        case XML_ATTRIBUTE_NODE:
1284
0
      xmlXIncludeErr(ctxt, ref->elem,
1285
0
                     XML_XINCLUDE_XPTR_RESULT,
1286
0
               "XPointer selects an attribute: #%s\n",
1287
0
               fragment);
1288
0
      goto xptr_error;
1289
0
        case XML_NAMESPACE_DECL:
1290
0
      xmlXIncludeErr(ctxt, ref->elem,
1291
0
                     XML_XINCLUDE_XPTR_RESULT,
1292
0
               "XPointer selects a namespace: #%s\n",
1293
0
               fragment);
1294
0
      goto xptr_error;
1295
0
        case XML_DOCUMENT_TYPE_NODE:
1296
0
        case XML_DOCUMENT_FRAG_NODE:
1297
0
        case XML_NOTATION_NODE:
1298
0
        case XML_DTD_NODE:
1299
0
        case XML_ELEMENT_DECL:
1300
0
        case XML_ATTRIBUTE_DECL:
1301
0
        case XML_ENTITY_DECL:
1302
0
        case XML_XINCLUDE_START:
1303
0
        case XML_XINCLUDE_END:
1304
                        /* shouldn't happen */
1305
0
      xmlXIncludeErr(ctxt, ref->elem,
1306
0
                     XML_XINCLUDE_XPTR_RESULT,
1307
0
           "XPointer selects unexpected nodes: #%s\n",
1308
0
               fragment);
1309
0
      goto xptr_error;
1310
0
    }
1311
0
      }
1312
0
  }
1313
0
        ref->inc = xmlXIncludeCopyXPointer(ctxt, xptr, ref->base);
1314
0
xptr_error:
1315
0
        xmlXPathFreeObject(xptr);
1316
0
    }
1317
1318
0
done:
1319
0
#endif
1320
1321
0
    ret = 0;
1322
1323
0
error:
1324
0
    return(ret);
1325
0
}
1326
1327
/**
1328
 * Load the content, and store the result in the XInclude context
1329
 *
1330
 * @param ctxt  the XInclude context
1331
 * @param ref  an XMLXincludeRefPtr
1332
 * @returns 0 in case of success, -1 in case of failure
1333
 */
1334
static int
1335
0
xmlXIncludeLoadTxt(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1336
0
    xmlParserInputBufferPtr buf;
1337
0
    xmlNodePtr node = NULL;
1338
0
    const xmlChar *url = ref->URI;
1339
0
    int i;
1340
0
    int ret = -1;
1341
0
    xmlChar *encoding = NULL;
1342
0
    xmlCharEncodingHandlerPtr handler = NULL;
1343
0
    xmlParserCtxtPtr pctxt = NULL;
1344
0
    xmlParserInputPtr inputStream = NULL;
1345
0
    int len;
1346
0
    int res;
1347
0
    const xmlChar *content;
1348
1349
    /*
1350
     * Handling of references to the local document are done
1351
     * directly through ctxt->doc.
1352
     */
1353
0
    if (url[0] == 0) {
1354
0
  xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_TEXT_DOCUMENT,
1355
0
           "text serialization of document not available\n", NULL);
1356
0
  goto error;
1357
0
    }
1358
1359
    /*
1360
     * Prevent reloading the document twice.
1361
     */
1362
0
    for (i = 0; i < ctxt->txtNr; i++) {
1363
0
  if (xmlStrEqual(url, ctxt->txtTab[i].url)) {
1364
0
            node = xmlNewDocText(ctxt->doc, ctxt->txtTab[i].text);
1365
0
            if (node == NULL)
1366
0
                xmlXIncludeErrMemory(ctxt);
1367
0
      goto loaded;
1368
0
  }
1369
0
    }
1370
1371
    /*
1372
     * Try to get the encoding if available
1373
     */
1374
0
    if (ref->elem != NULL) {
1375
0
  encoding = xmlXIncludeGetProp(ctxt, ref->elem, XINCLUDE_PARSE_ENCODING);
1376
0
    }
1377
0
    if (encoding != NULL) {
1378
0
        xmlParserErrors code;
1379
1380
0
        code = xmlOpenCharEncodingHandler((const char *) encoding,
1381
0
                                          /* output */ 0, &handler);
1382
1383
0
        if (code != XML_ERR_OK) {
1384
0
            if (code == XML_ERR_NO_MEMORY) {
1385
0
                xmlXIncludeErrMemory(ctxt);
1386
0
            } else if (code == XML_ERR_UNSUPPORTED_ENCODING) {
1387
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_UNKNOWN_ENCODING,
1388
0
                               "encoding %s not supported\n", encoding);
1389
0
                goto error;
1390
0
            } else {
1391
0
                xmlXIncludeErr(ctxt, ref->elem, code,
1392
0
                               "unexpected error from iconv or ICU\n", NULL);
1393
0
                goto error;
1394
0
            }
1395
0
        }
1396
0
    }
1397
1398
    /*
1399
     * Load it.
1400
     */
1401
0
    pctxt = xmlNewParserCtxt();
1402
0
    if (pctxt == NULL) {
1403
0
        xmlXIncludeErrMemory(ctxt);
1404
0
        goto error;
1405
0
    }
1406
0
    if (ctxt->errorHandler != NULL)
1407
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
1408
0
    if (ctxt->resourceLoader != NULL)
1409
0
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
1410
0
                                 ctxt->resourceCtxt);
1411
1412
0
    inputStream = xmlLoadResource(pctxt, (const char*) url, NULL,
1413
0
                                  XML_RESOURCE_XINCLUDE_TEXT);
1414
0
    if (inputStream == NULL) {
1415
        /*
1416
         * ENOENT only produces a warning which isn't reflected in errNo.
1417
         */
1418
0
        if (pctxt->errNo == XML_ERR_NO_MEMORY)
1419
0
            xmlXIncludeErrMemory(ctxt);
1420
0
        else if ((pctxt->errNo != XML_ERR_OK) &&
1421
0
                 (pctxt->errNo != XML_IO_ENOENT) &&
1422
0
                 (pctxt->errNo != XML_IO_UNKNOWN))
1423
0
            xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "load error", NULL);
1424
0
  goto error;
1425
0
    }
1426
0
    buf = inputStream->buf;
1427
0
    if (buf == NULL)
1428
0
  goto error;
1429
0
    if (buf->encoder)
1430
0
  xmlCharEncCloseFunc(buf->encoder);
1431
0
    buf->encoder = handler;
1432
0
    handler = NULL;
1433
1434
0
    node = xmlNewDocText(ctxt->doc, NULL);
1435
0
    if (node == NULL) {
1436
0
        xmlXIncludeErrMemory(ctxt);
1437
0
  goto error;
1438
0
    }
1439
1440
    /*
1441
     * Scan all chars from the resource and add the to the node
1442
     */
1443
0
    do {
1444
0
        res = xmlParserInputBufferRead(buf, 4096);
1445
0
    } while (res > 0);
1446
0
    if (res < 0) {
1447
0
        if (buf->error == XML_ERR_NO_MEMORY)
1448
0
            xmlXIncludeErrMemory(ctxt);
1449
0
        else
1450
0
            xmlXIncludeErr(ctxt, NULL, buf->error, "read error", NULL);
1451
0
        goto error;
1452
0
    }
1453
1454
0
    content = xmlBufContent(buf->buffer);
1455
0
    len = xmlBufUse(buf->buffer);
1456
0
    for (i = 0; i < len;) {
1457
0
        int cur;
1458
0
        int l;
1459
1460
0
        l = len - i;
1461
0
        cur = xmlGetUTF8Char(&content[i], &l);
1462
0
        if ((cur < 0) || (!IS_CHAR(cur))) {
1463
0
            xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_INVALID_CHAR,
1464
0
                           "%s contains invalid char\n", url);
1465
0
            goto error;
1466
0
        }
1467
1468
0
        i += l;
1469
0
    }
1470
1471
0
    if (xmlNodeAddContentLen(node, content, len) < 0)
1472
0
        xmlXIncludeErrMemory(ctxt);
1473
1474
0
    if (ctxt->txtNr >= ctxt->txtMax) {
1475
0
        xmlXIncludeTxt *tmp;
1476
0
        int newSize;
1477
1478
0
        newSize = xmlGrowCapacity(ctxt->txtMax, sizeof(tmp[0]),
1479
0
                                  8, XML_MAX_ITEMS);
1480
0
        if (newSize < 0) {
1481
0
            xmlXIncludeErrMemory(ctxt);
1482
0
      goto error;
1483
0
        }
1484
0
        tmp = xmlRealloc(ctxt->txtTab, newSize * sizeof(tmp[0]));
1485
0
        if (tmp == NULL) {
1486
0
            xmlXIncludeErrMemory(ctxt);
1487
0
      goto error;
1488
0
        }
1489
0
        ctxt->txtMax = newSize;
1490
0
        ctxt->txtTab = tmp;
1491
0
    }
1492
0
    ctxt->txtTab[ctxt->txtNr].text = xmlStrdup(node->content);
1493
0
    if ((node->content != NULL) &&
1494
0
        (ctxt->txtTab[ctxt->txtNr].text == NULL)) {
1495
0
        xmlXIncludeErrMemory(ctxt);
1496
0
        goto error;
1497
0
    }
1498
0
    ctxt->txtTab[ctxt->txtNr].url = xmlStrdup(url);
1499
0
    if (ctxt->txtTab[ctxt->txtNr].url == NULL) {
1500
0
        xmlXIncludeErrMemory(ctxt);
1501
0
        xmlFree(ctxt->txtTab[ctxt->txtNr].text);
1502
0
        goto error;
1503
0
    }
1504
0
    ctxt->txtNr++;
1505
1506
0
loaded:
1507
    /*
1508
     * Add the element as the replacement copy.
1509
     */
1510
0
    ref->inc = node;
1511
0
    node = NULL;
1512
0
    ret = 0;
1513
1514
0
error:
1515
0
    xmlFreeNode(node);
1516
0
    xmlFreeInputStream(inputStream);
1517
0
    xmlFreeParserCtxt(pctxt);
1518
0
    xmlCharEncCloseFunc(handler);
1519
0
    xmlFree(encoding);
1520
0
    return(ret);
1521
0
}
1522
1523
/**
1524
 * Load the content of the fallback node, and store the result
1525
 * in the XInclude context
1526
 *
1527
 * @param ctxt  the XInclude context
1528
 * @param fallback  the fallback node
1529
 * @param ref  an XMLXincludeRefPtr
1530
 * @returns 0 in case of success, -1 in case of failure
1531
 */
1532
static int
1533
xmlXIncludeLoadFallback(xmlXIncludeCtxtPtr ctxt, xmlNodePtr fallback,
1534
0
                        xmlXIncludeRefPtr ref) {
1535
0
    int ret = 0;
1536
0
    int oldNbErrors;
1537
1538
0
    if ((fallback == NULL) || (fallback->type == XML_NAMESPACE_DECL) ||
1539
0
        (ctxt == NULL))
1540
0
  return(-1);
1541
0
    if (fallback->children != NULL) {
1542
  /*
1543
   * It's possible that the fallback also has 'includes'
1544
   * (Bug 129969), so we re-process the fallback just in case
1545
   */
1546
0
        oldNbErrors = ctxt->nbErrors;
1547
0
  ref->inc = xmlXIncludeCopyNode(ctxt, fallback, 1, ref->base);
1548
0
  if (ctxt->nbErrors > oldNbErrors)
1549
0
      ret = -1;
1550
0
    } else {
1551
0
        ref->inc = NULL;
1552
0
    }
1553
0
    ref->fallback = 1;
1554
0
    return(ret);
1555
0
}
1556
1557
/************************************************************************
1558
 *                  *
1559
 *      XInclude Processing       *
1560
 *                  *
1561
 ************************************************************************/
1562
1563
/**
1564
 * If the XInclude node wasn't processed yet, create a new RefPtr,
1565
 * add it to ctxt->incTab and load the included items.
1566
 *
1567
 * @param ctxt  an XInclude context
1568
 * @param node  an XInclude node
1569
 * @returns the new or existing xmlXIncludeRef, or NULL in case of error.
1570
 */
1571
static xmlXIncludeRefPtr
1572
0
xmlXIncludeExpandNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1573
0
    xmlXIncludeRefPtr ref;
1574
0
    int i;
1575
1576
0
    if (ctxt->fatalErr)
1577
0
        return(NULL);
1578
0
    if (ctxt->depth >= XINCLUDE_MAX_DEPTH) {
1579
0
        xmlXIncludeErr(ctxt, node, XML_XINCLUDE_RECURSION,
1580
0
                       "maximum recursion depth exceeded\n", NULL);
1581
0
        ctxt->fatalErr = 1;
1582
0
        return(NULL);
1583
0
    }
1584
1585
0
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
1586
    /*
1587
     * The XInclude engine offers no protection against exponential
1588
     * expansion attacks similar to "billion laughs". Avoid timeouts by
1589
     * limiting the total number of replacements when fuzzing.
1590
     *
1591
     * Unfortuately, a single XInclude can already result in quadratic
1592
     * behavior:
1593
     *
1594
     *     <doc xmlns:xi="http://www.w3.org/2001/XInclude">
1595
     *       <xi:include xpointer="xpointer(//e)"/>
1596
     *       <e>
1597
     *         <e>
1598
     *           <e>
1599
     *             <!-- more nested elements -->
1600
     *           </e>
1601
     *         </e>
1602
     *       </e>
1603
     *     </doc>
1604
     */
1605
0
    if (ctxt->incTotal >= 20)
1606
0
        return(NULL);
1607
0
    ctxt->incTotal++;
1608
0
#endif
1609
1610
0
    for (i = 0; i < ctxt->incNr; i++) {
1611
0
        if (ctxt->incTab[i]->elem == node) {
1612
0
            if (ctxt->incTab[i]->expanding) {
1613
0
                xmlXIncludeErr(ctxt, node, XML_XINCLUDE_RECURSION,
1614
0
                               "inclusion loop detected\n", NULL);
1615
0
                return(NULL);
1616
0
            }
1617
0
            return(ctxt->incTab[i]);
1618
0
        }
1619
0
    }
1620
1621
0
    ref = xmlXIncludeAddNode(ctxt, node);
1622
0
    if (ref == NULL)
1623
0
        return(NULL);
1624
0
    ref->expanding = 1;
1625
0
    ctxt->depth++;
1626
0
    xmlXIncludeLoadNode(ctxt, ref);
1627
0
    ctxt->depth--;
1628
0
    ref->expanding = 0;
1629
1630
0
    return(ref);
1631
0
}
1632
1633
/**
1634
 * Find and load the infoset replacement for the given node.
1635
 *
1636
 * @param ctxt  an XInclude context
1637
 * @param ref  an xmlXIncludeRef
1638
 * @returns 0 if substitution succeeded, -1 if some processing failed
1639
 */
1640
static int
1641
0
xmlXIncludeLoadNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1642
0
    xmlNodePtr cur;
1643
0
    int ret;
1644
1645
0
    if ((ctxt == NULL) || (ref == NULL))
1646
0
  return(-1);
1647
0
    cur = ref->elem;
1648
0
    if (cur == NULL)
1649
0
  return(-1);
1650
1651
0
    if (ref->xml) {
1652
0
  ret = xmlXIncludeLoadDoc(ctxt, ref);
1653
  /* xmlXIncludeGetFragment(ctxt, cur, URI); */
1654
0
    } else {
1655
0
  ret = xmlXIncludeLoadTxt(ctxt, ref);
1656
0
    }
1657
1658
0
    if (ret < 0) {
1659
0
  xmlNodePtr children;
1660
1661
  /*
1662
   * Time to try a fallback if available
1663
   */
1664
0
  children = cur->children;
1665
0
  while (children != NULL) {
1666
0
      if ((children->type == XML_ELEMENT_NODE) &&
1667
0
    (children->ns != NULL) &&
1668
0
    (xmlStrEqual(children->name, XINCLUDE_FALLBACK)) &&
1669
0
    ((xmlStrEqual(children->ns->href, XINCLUDE_NS)) ||
1670
0
     (xmlStrEqual(children->ns->href, XINCLUDE_OLD_NS)))) {
1671
0
    ret = xmlXIncludeLoadFallback(ctxt, children, ref);
1672
0
    break;
1673
0
      }
1674
0
      children = children->next;
1675
0
  }
1676
0
    }
1677
0
    if (ret < 0) {
1678
0
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_FALLBACK,
1679
0
           "could not load %s, and no fallback was found\n",
1680
0
           ref->URI);
1681
0
    }
1682
1683
0
    return(0);
1684
0
}
1685
1686
/**
1687
 * Implement the infoset replacement for the given node
1688
 *
1689
 * @param ctxt  an XInclude context
1690
 * @param ref  an xmlXIncludeRef
1691
 * @returns 0 if substitution succeeded, -1 if some processing failed
1692
 */
1693
static int
1694
0
xmlXIncludeIncludeNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1695
0
    xmlNodePtr cur, end, list, tmp;
1696
1697
0
    if ((ctxt == NULL) || (ref == NULL))
1698
0
  return(-1);
1699
0
    cur = ref->elem;
1700
0
    if ((cur == NULL) || (cur->type == XML_NAMESPACE_DECL))
1701
0
  return(-1);
1702
1703
0
    list = ref->inc;
1704
0
    ref->inc = NULL;
1705
1706
    /*
1707
     * Check against the risk of generating a multi-rooted document
1708
     */
1709
0
    if ((cur->parent != NULL) &&
1710
0
  (cur->parent->type != XML_ELEMENT_NODE)) {
1711
0
  int nb_elem = 0;
1712
1713
0
  tmp = list;
1714
0
  while (tmp != NULL) {
1715
0
      if (tmp->type == XML_ELEMENT_NODE)
1716
0
    nb_elem++;
1717
0
      tmp = tmp->next;
1718
0
  }
1719
0
        if (nb_elem != 1) {
1720
0
            if (nb_elem > 1)
1721
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1722
0
                               "XInclude error: would result in multiple root "
1723
0
                               "nodes\n", NULL);
1724
0
            else
1725
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1726
0
                               "XInclude error: would result in no root "
1727
0
                               "node\n", NULL);
1728
0
            xmlFreeNodeList(list);
1729
0
      return(-1);
1730
0
  }
1731
0
    }
1732
1733
0
    if (ctxt->parseFlags & XML_PARSE_NOXINCNODE) {
1734
  /*
1735
   * Add the list of nodes
1736
         *
1737
         * TODO: Coalesce text nodes unless we are streaming mode.
1738
   */
1739
0
  while (list != NULL) {
1740
0
      end = list;
1741
0
      list = list->next;
1742
1743
0
      if (xmlAddPrevSibling(cur, end) == NULL) {
1744
0
                xmlUnlinkNode(end);
1745
0
                xmlFreeNode(end);
1746
0
                goto err_memory;
1747
0
            }
1748
0
  }
1749
0
  xmlUnlinkNode(cur);
1750
0
  xmlFreeNode(cur);
1751
0
    } else {
1752
0
        xmlNodePtr child, next;
1753
1754
  /*
1755
   * Change the current node as an XInclude start one, and add an
1756
   * XInclude end one
1757
   */
1758
0
        if (ref->fallback)
1759
0
            xmlUnsetProp(cur, BAD_CAST "href");
1760
0
  cur->type = XML_XINCLUDE_START;
1761
        /* Remove fallback children */
1762
0
        for (child = cur->children; child != NULL; child = next) {
1763
0
            next = child->next;
1764
0
            xmlUnlinkNode(child);
1765
0
            xmlFreeNode(child);
1766
0
        }
1767
0
  end = xmlNewDocNode(cur->doc, cur->ns, cur->name, NULL);
1768
0
  if (end == NULL)
1769
0
            goto err_memory;
1770
0
  end->type = XML_XINCLUDE_END;
1771
0
  if (xmlAddNextSibling(cur, end) == NULL) {
1772
0
            xmlFreeNode(end);
1773
0
            goto err_memory;
1774
0
        }
1775
1776
  /*
1777
   * Add the list of nodes
1778
   */
1779
0
  while (list != NULL) {
1780
0
      cur = list;
1781
0
      list = list->next;
1782
1783
0
      if (xmlAddPrevSibling(end, cur) == NULL) {
1784
0
                xmlUnlinkNode(cur);
1785
0
                xmlFreeNode(cur);
1786
0
                goto err_memory;
1787
0
            }
1788
0
  }
1789
0
    }
1790
1791
1792
0
    return(0);
1793
1794
0
err_memory:
1795
0
    xmlXIncludeErrMemory(ctxt);
1796
0
    xmlFreeNodeList(list);
1797
0
    return(-1);
1798
0
}
1799
1800
/**
1801
 * test if the node is an XInclude node
1802
 *
1803
 * @param ctxt  the XInclude processing context
1804
 * @param node  an XInclude node
1805
 * @returns 1 true, 0 otherwise
1806
 */
1807
static int
1808
0
xmlXIncludeTestNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1809
0
    if (node == NULL)
1810
0
  return(0);
1811
0
    if (node->type != XML_ELEMENT_NODE)
1812
0
  return(0);
1813
0
    if (node->ns == NULL)
1814
0
  return(0);
1815
0
    if ((xmlStrEqual(node->ns->href, XINCLUDE_NS)) ||
1816
0
        (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS))) {
1817
0
  if (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS)) {
1818
0
      if (ctxt->legacy == 0) {
1819
0
          ctxt->legacy = 1;
1820
0
      }
1821
0
  }
1822
0
  if (xmlStrEqual(node->name, XINCLUDE_NODE)) {
1823
0
      xmlNodePtr child = node->children;
1824
0
      int nb_fallback = 0;
1825
1826
0
      while (child != NULL) {
1827
0
    if ((child->type == XML_ELEMENT_NODE) &&
1828
0
        (child->ns != NULL) &&
1829
0
        ((xmlStrEqual(child->ns->href, XINCLUDE_NS)) ||
1830
0
         (xmlStrEqual(child->ns->href, XINCLUDE_OLD_NS)))) {
1831
0
        if (xmlStrEqual(child->name, XINCLUDE_NODE)) {
1832
0
      xmlXIncludeErr(ctxt, node,
1833
0
                     XML_XINCLUDE_INCLUDE_IN_INCLUDE,
1834
0
               "%s has an 'include' child\n",
1835
0
               XINCLUDE_NODE);
1836
0
      return(0);
1837
0
        }
1838
0
        if (xmlStrEqual(child->name, XINCLUDE_FALLBACK)) {
1839
0
      nb_fallback++;
1840
0
        }
1841
0
    }
1842
0
    child = child->next;
1843
0
      }
1844
0
      if (nb_fallback > 1) {
1845
0
    xmlXIncludeErr(ctxt, node, XML_XINCLUDE_FALLBACKS_IN_INCLUDE,
1846
0
             "%s has multiple fallback children\n",
1847
0
                   XINCLUDE_NODE);
1848
0
    return(0);
1849
0
      }
1850
0
      return(1);
1851
0
  }
1852
0
  if (xmlStrEqual(node->name, XINCLUDE_FALLBACK)) {
1853
0
      if ((node->parent == NULL) ||
1854
0
    (node->parent->type != XML_ELEMENT_NODE) ||
1855
0
    (node->parent->ns == NULL) ||
1856
0
    ((!xmlStrEqual(node->parent->ns->href, XINCLUDE_NS)) &&
1857
0
     (!xmlStrEqual(node->parent->ns->href, XINCLUDE_OLD_NS))) ||
1858
0
    (!xmlStrEqual(node->parent->name, XINCLUDE_NODE))) {
1859
0
    xmlXIncludeErr(ctxt, node,
1860
0
                   XML_XINCLUDE_FALLBACK_NOT_IN_INCLUDE,
1861
0
             "%s is not the child of an 'include'\n",
1862
0
             XINCLUDE_FALLBACK);
1863
0
      }
1864
0
  }
1865
0
    }
1866
0
    return(0);
1867
0
}
1868
1869
/**
1870
 * Implement the XInclude substitution on the XML document `doc`
1871
 *
1872
 * @param ctxt  the XInclude processing context
1873
 * @param tree  the top of the tree to process
1874
 * @returns 0 if no substitution were done, -1 if some processing failed
1875
 *    or the number of substitutions done.
1876
 */
1877
static int
1878
0
xmlXIncludeDoProcess(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1879
0
    xmlXIncludeRefPtr ref;
1880
0
    xmlNodePtr cur;
1881
0
    int ret = 0;
1882
0
    int i, start;
1883
1884
    /*
1885
     * First phase: lookup the elements in the document
1886
     */
1887
0
    start = ctxt->incNr;
1888
0
    cur = tree;
1889
0
    do {
1890
  /* TODO: need to work on entities -> stack */
1891
0
        if (xmlXIncludeTestNode(ctxt, cur) == 1) {
1892
0
            ref = xmlXIncludeExpandNode(ctxt, cur);
1893
            /*
1894
             * Mark direct includes.
1895
             */
1896
0
            if (ref != NULL)
1897
0
                ref->replace = 1;
1898
0
        } else if ((cur->children != NULL) &&
1899
0
                   ((cur->type == XML_DOCUMENT_NODE) ||
1900
0
                    (cur->type == XML_ELEMENT_NODE))) {
1901
0
            cur = cur->children;
1902
0
            continue;
1903
0
        }
1904
0
        do {
1905
0
            if (cur == tree)
1906
0
                break;
1907
0
            if (cur->next != NULL) {
1908
0
                cur = cur->next;
1909
0
                break;
1910
0
            }
1911
0
            cur = cur->parent;
1912
0
        } while (cur != NULL);
1913
0
    } while ((cur != NULL) && (cur != tree));
1914
1915
    /*
1916
     * Second phase: extend the original document infoset.
1917
     */
1918
0
    for (i = start; i < ctxt->incNr; i++) {
1919
0
  if (ctxt->incTab[i]->replace != 0) {
1920
0
            xmlXIncludeIncludeNode(ctxt, ctxt->incTab[i]);
1921
0
            ctxt->incTab[i]->replace = 0;
1922
0
        } else {
1923
            /*
1924
             * Ignore includes which were added indirectly, for example
1925
             * inside xi:fallback elements.
1926
             */
1927
0
            if (ctxt->incTab[i]->inc != NULL) {
1928
0
                xmlFreeNodeList(ctxt->incTab[i]->inc);
1929
0
                ctxt->incTab[i]->inc = NULL;
1930
0
            }
1931
0
        }
1932
0
  ret++;
1933
0
    }
1934
1935
0
    if (ctxt->isStream) {
1936
        /*
1937
         * incTab references nodes which will eventually be deleted in
1938
         * streaming mode. The table is only required for XPointer
1939
         * expressions which aren't allowed in streaming mode.
1940
         */
1941
0
        for (i = 0;i < ctxt->incNr;i++) {
1942
0
            xmlXIncludeFreeRef(ctxt->incTab[i]);
1943
0
        }
1944
0
        ctxt->incNr = 0;
1945
0
    }
1946
1947
0
    return(ret);
1948
0
}
1949
1950
/**
1951
 * Implement the XInclude substitution on the XML document `doc`
1952
 *
1953
 * @param ctxt  the XInclude processing context
1954
 * @param tree  the top of the tree to process
1955
 * @returns 0 if no substitution were done, -1 if some processing failed
1956
 *    or the number of substitutions done.
1957
 */
1958
static int
1959
0
xmlXIncludeDoProcessRoot(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1960
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL))
1961
0
  return(-1);
1962
0
    if (ctxt == NULL)
1963
0
  return(-1);
1964
1965
0
    return(xmlXIncludeDoProcess(ctxt, tree));
1966
0
}
1967
1968
/**
1969
 * @since 2.13.0
1970
 *
1971
 * @param ctxt  an XInclude processing context
1972
 * @returns the last error code.
1973
 */
1974
int
1975
0
xmlXIncludeGetLastError(xmlXIncludeCtxt *ctxt) {
1976
0
    if (ctxt == NULL)
1977
0
        return(XML_ERR_ARGUMENT);
1978
0
    return(ctxt->errNo);
1979
0
}
1980
1981
/**
1982
 * Register a callback function that will be called on errors and
1983
 * warnings. If handler is NULL, the error handler will be deactivated.
1984
 *
1985
 * @since 2.13.0
1986
 * @param ctxt  an XInclude processing context
1987
 * @param handler  error handler
1988
 * @param data  user data which will be passed to the handler
1989
 */
1990
void
1991
xmlXIncludeSetErrorHandler(xmlXIncludeCtxt *ctxt,
1992
0
                           xmlStructuredErrorFunc handler, void *data) {
1993
0
    if (ctxt == NULL)
1994
0
        return;
1995
0
    ctxt->errorHandler = handler;
1996
0
    ctxt->errorCtxt = data;
1997
0
}
1998
1999
/**
2000
 * Register a callback function that will be called to load included
2001
 * documents.
2002
 *
2003
 * @since 2.14.0
2004
 * @param ctxt  an XInclude processing context
2005
 * @param loader  resource loader
2006
 * @param data  user data which will be passed to the loader
2007
 */
2008
void
2009
xmlXIncludeSetResourceLoader(xmlXIncludeCtxt *ctxt,
2010
0
                             xmlResourceLoader loader, void *data) {
2011
0
    if (ctxt == NULL)
2012
0
        return;
2013
0
    ctxt->resourceLoader = loader;
2014
0
    ctxt->resourceCtxt = data;
2015
0
}
2016
2017
/**
2018
 * Set the flags used for further processing of XML resources.
2019
 *
2020
 * @param ctxt  an XInclude processing context
2021
 * @param flags  a set of xmlParserOption used for parsing XML includes
2022
 * @returns 0 in case of success and -1 in case of error.
2023
 */
2024
int
2025
0
xmlXIncludeSetFlags(xmlXIncludeCtxt *ctxt, int flags) {
2026
0
    if (ctxt == NULL)
2027
0
        return(-1);
2028
0
    ctxt->parseFlags = flags;
2029
0
    return(0);
2030
0
}
2031
2032
/**
2033
 * In streaming mode, XPointer expressions aren't allowed.
2034
 *
2035
 * @param ctxt  an XInclude processing context
2036
 * @param mode  whether streaming mode should be enabled
2037
 * @returns 0 in case of success and -1 in case of error.
2038
 */
2039
int
2040
0
xmlXIncludeSetStreamingMode(xmlXIncludeCtxt *ctxt, int mode) {
2041
0
    if (ctxt == NULL)
2042
0
        return(-1);
2043
0
    ctxt->isStream = !!mode;
2044
0
    return(0);
2045
0
}
2046
2047
/**
2048
 * Implement the XInclude substitution on the XML node `tree`
2049
 *
2050
 * @param tree  an XML node
2051
 * @param flags  a set of xmlParserOption used for parsing XML includes
2052
 * @param data  application data that will be passed to the parser context
2053
 *        in the _private field of the parser context(s)
2054
 * @returns 0 if no substitution were done, -1 if some processing failed
2055
 *    or the number of substitutions done.
2056
 */
2057
2058
int
2059
0
xmlXIncludeProcessTreeFlagsData(xmlNode *tree, int flags, void *data) {
2060
0
    xmlXIncludeCtxtPtr ctxt;
2061
0
    int ret = 0;
2062
2063
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL) ||
2064
0
        (tree->doc == NULL))
2065
0
        return(-1);
2066
2067
0
    ctxt = xmlXIncludeNewContext(tree->doc);
2068
0
    if (ctxt == NULL)
2069
0
        return(-1);
2070
0
    ctxt->_private = data;
2071
0
    xmlXIncludeSetFlags(ctxt, flags);
2072
0
    ret = xmlXIncludeDoProcessRoot(ctxt, tree);
2073
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2074
0
        ret = -1;
2075
2076
0
    xmlXIncludeFreeContext(ctxt);
2077
0
    return(ret);
2078
0
}
2079
2080
/**
2081
 * Implement the XInclude substitution on the XML document `doc`
2082
 *
2083
 * @param doc  an XML document
2084
 * @param flags  a set of xmlParserOption used for parsing XML includes
2085
 * @param data  application data that will be passed to the parser context
2086
 *        in the _private field of the parser context(s)
2087
 * @returns 0 if no substitution were done, -1 if some processing failed
2088
 *    or the number of substitutions done.
2089
 */
2090
int
2091
0
xmlXIncludeProcessFlagsData(xmlDoc *doc, int flags, void *data) {
2092
0
    xmlNodePtr tree;
2093
2094
0
    if (doc == NULL)
2095
0
  return(-1);
2096
0
    tree = xmlDocGetRootElement(doc);
2097
0
    if (tree == NULL)
2098
0
  return(-1);
2099
0
    return(xmlXIncludeProcessTreeFlagsData(tree, flags, data));
2100
0
}
2101
2102
/**
2103
 * Implement the XInclude substitution on the XML document `doc`
2104
 *
2105
 * @param doc  an XML document
2106
 * @param flags  a set of xmlParserOption used for parsing XML includes
2107
 * @returns 0 if no substitution were done, -1 if some processing failed
2108
 *    or the number of substitutions done.
2109
 */
2110
int
2111
0
xmlXIncludeProcessFlags(xmlDoc *doc, int flags) {
2112
0
    return xmlXIncludeProcessFlagsData(doc, flags, NULL);
2113
0
}
2114
2115
/**
2116
 * Implement the XInclude substitution on the XML document `doc`
2117
 *
2118
 * @param doc  an XML document
2119
 * @returns 0 if no substitution were done, -1 if some processing failed
2120
 *    or the number of substitutions done.
2121
 */
2122
int
2123
0
xmlXIncludeProcess(xmlDoc *doc) {
2124
0
    return(xmlXIncludeProcessFlags(doc, 0));
2125
0
}
2126
2127
/**
2128
 * Implement the XInclude substitution for the given subtree
2129
 *
2130
 * @param tree  a node in an XML document
2131
 * @param flags  a set of xmlParserOption used for parsing XML includes
2132
 * @returns 0 if no substitution were done, -1 if some processing failed
2133
 *    or the number of substitutions done.
2134
 */
2135
int
2136
0
xmlXIncludeProcessTreeFlags(xmlNode *tree, int flags) {
2137
0
    xmlXIncludeCtxtPtr ctxt;
2138
0
    int ret = 0;
2139
2140
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL) ||
2141
0
        (tree->doc == NULL))
2142
0
  return(-1);
2143
0
    ctxt = xmlXIncludeNewContext(tree->doc);
2144
0
    if (ctxt == NULL)
2145
0
  return(-1);
2146
0
    xmlXIncludeSetFlags(ctxt, flags);
2147
0
    ret = xmlXIncludeDoProcessRoot(ctxt, tree);
2148
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2149
0
  ret = -1;
2150
2151
0
    xmlXIncludeFreeContext(ctxt);
2152
0
    return(ret);
2153
0
}
2154
2155
/**
2156
 * Implement the XInclude substitution for the given subtree
2157
 *
2158
 * @param tree  a node in an XML document
2159
 * @returns 0 if no substitution were done, -1 if some processing failed
2160
 *    or the number of substitutions done.
2161
 */
2162
int
2163
0
xmlXIncludeProcessTree(xmlNode *tree) {
2164
0
    return(xmlXIncludeProcessTreeFlags(tree, 0));
2165
0
}
2166
2167
/**
2168
 * Implement the XInclude substitution for the given subtree reusing
2169
 * the information and data coming from the given context.
2170
 *
2171
 * @param ctxt  an existing XInclude context
2172
 * @param node  a node in an XML document
2173
 * @returns 0 if no substitution were done, -1 if some processing failed
2174
 *    or the number of substitutions done.
2175
 */
2176
int
2177
0
xmlXIncludeProcessNode(xmlXIncludeCtxt *ctxt, xmlNode *node) {
2178
0
    int ret = 0;
2179
2180
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL) ||
2181
0
        (node->doc == NULL) || (ctxt == NULL))
2182
0
  return(-1);
2183
0
    ret = xmlXIncludeDoProcessRoot(ctxt, node);
2184
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2185
0
  ret = -1;
2186
0
    return(ret);
2187
0
}
2188
2189
#else /* !LIBXML_XINCLUDE_ENABLED */
2190
#endif