Coverage Report

Created: 2026-05-30 06:17

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libxml2/xinclude.c
Line
Count
Source
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
26.0k
#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
1.71k
{
141
1.71k
    ctxt->errNo = XML_ERR_NO_MEMORY;
142
1.71k
    ctxt->fatalErr = 1;
143
1.71k
    ctxt->nbErrors++;
144
145
1.71k
    xmlRaiseMemoryError(ctxt->errorHandler, NULL, ctxt->errorCtxt,
146
1.71k
                        XML_FROM_XINCLUDE, NULL);
147
1.71k
}
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
33.0k
{
162
33.0k
    xmlStructuredErrorFunc schannel = NULL;
163
33.0k
    xmlGenericErrorFunc channel = NULL;
164
33.0k
    void *data = NULL;
165
33.0k
    int res;
166
167
33.0k
    if (error == XML_ERR_NO_MEMORY) {
168
106
        xmlXIncludeErrMemory(ctxt);
169
106
        return;
170
106
    }
171
172
32.9k
    if (ctxt->fatalErr != 0)
173
2.04k
        return;
174
30.9k
    ctxt->nbErrors++;
175
176
30.9k
    schannel = ctxt->errorHandler;
177
30.9k
    data = ctxt->errorCtxt;
178
179
30.9k
    if (schannel == NULL) {
180
30.9k
        channel = xmlGenericError;
181
30.9k
        data = xmlGenericErrorContext;
182
30.9k
    }
183
184
30.9k
    res = xmlRaiseError(schannel, channel, data, ctxt, node,
185
30.9k
                        XML_FROM_XINCLUDE, error, XML_ERR_ERROR,
186
30.9k
                        NULL, 0, (const char *) extra, NULL, NULL, 0, 0,
187
30.9k
                        msg, (const char *) extra);
188
30.9k
    if (res < 0) {
189
98
        ctxt->errNo = XML_ERR_NO_MEMORY;
190
98
        ctxt->fatalErr = 1;
191
30.8k
    } else {
192
30.8k
        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
30.8k
        if (xmlIsCatastrophicError(XML_ERR_FATAL, error))
200
2
            ctxt->fatalErr = 1;
201
30.8k
    }
202
30.9k
}
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
67.5k
                   const xmlChar *name) {
215
67.5k
    xmlChar *ret;
216
217
67.5k
    if (xmlNodeGetAttrValue(cur, name, XINCLUDE_NS, &ret) < 0)
218
1
        xmlXIncludeErrMemory(ctxt);
219
67.5k
    if (ret != NULL)
220
35
        return(ret);
221
222
67.5k
    if (ctxt->legacy != 0) {
223
58.3k
        if (xmlNodeGetAttrValue(cur, name, XINCLUDE_OLD_NS, &ret) < 0)
224
1
            xmlXIncludeErrMemory(ctxt);
225
58.3k
        if (ret != NULL)
226
93
            return(ret);
227
58.3k
    }
228
229
67.4k
    if (xmlNodeGetAttrValue(cur, name, NULL, &ret) < 0)
230
7
        xmlXIncludeErrMemory(ctxt);
231
67.4k
    return(ret);
232
67.5k
}
233
/**
234
 * Free an XInclude reference
235
 *
236
 * @param ref  the XInclude reference
237
 */
238
static void
239
43.9k
xmlXIncludeFreeRef(xmlXIncludeRefPtr ref) {
240
43.9k
    if (ref == NULL)
241
22.8k
  return;
242
21.0k
    if (ref->URI != NULL)
243
21.0k
  xmlFree(ref->URI);
244
21.0k
    if (ref->fragment != NULL)
245
16.1k
  xmlFree(ref->fragment);
246
21.0k
    if (ref->base != NULL)
247
12.2k
  xmlFree(ref->base);
248
21.0k
    xmlFree(ref);
249
21.0k
}
250
251
/**
252
 * Creates a new XInclude context
253
 *
254
 * @param doc  an XML Document
255
 * @returns the new set
256
 */
257
xmlXIncludeCtxt *
258
28.4k
xmlXIncludeNewContext(xmlDoc *doc) {
259
28.4k
    xmlXIncludeCtxtPtr ret;
260
261
28.4k
    if (doc == NULL)
262
8.79k
  return(NULL);
263
19.6k
    ret = (xmlXIncludeCtxtPtr) xmlMalloc(sizeof(xmlXIncludeCtxt));
264
19.6k
    if (ret == NULL)
265
51
  return(NULL);
266
19.5k
    memset(ret, 0, sizeof(xmlXIncludeCtxt));
267
19.5k
    ret->doc = doc;
268
19.5k
    ret->incNr = 0;
269
19.5k
    ret->incMax = 0;
270
19.5k
    ret->incTab = NULL;
271
19.5k
    ret->nbErrors = 0;
272
19.5k
    return(ret);
273
19.6k
}
274
275
/**
276
 * Free an XInclude context
277
 *
278
 * @param ctxt  the XInclude context
279
 */
280
void
281
28.4k
xmlXIncludeFreeContext(xmlXIncludeCtxt *ctxt) {
282
28.4k
    int i;
283
284
28.4k
    if (ctxt == NULL)
285
8.85k
  return;
286
19.5k
    if (ctxt->urlTab != NULL) {
287
6.17k
  for (i = 0; i < ctxt->urlNr; i++) {
288
3.99k
      xmlFreeDoc(ctxt->urlTab[i].doc);
289
3.99k
      xmlFree(ctxt->urlTab[i].url);
290
3.99k
  }
291
2.17k
  xmlFree(ctxt->urlTab);
292
2.17k
    }
293
40.3k
    for (i = 0;i < ctxt->incNr;i++) {
294
20.8k
  if (ctxt->incTab[i] != NULL)
295
20.8k
      xmlXIncludeFreeRef(ctxt->incTab[i]);
296
20.8k
    }
297
19.5k
    if (ctxt->incTab != NULL)
298
9.66k
  xmlFree(ctxt->incTab);
299
19.5k
    if (ctxt->txtTab != NULL) {
300
90
  for (i = 0;i < ctxt->txtNr;i++) {
301
48
      xmlFree(ctxt->txtTab[i].text);
302
48
      xmlFree(ctxt->txtTab[i].url);
303
48
  }
304
42
  xmlFree(ctxt->txtTab);
305
42
    }
306
19.5k
#ifdef LIBXML_XPTR_ENABLED
307
19.5k
    if (ctxt->xpctxt != NULL)
308
7.47k
  xmlXPathFreeContext(ctxt->xpctxt);
309
19.5k
#endif
310
19.5k
    xmlFree(ctxt);
311
19.5k
}
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
4.00k
xmlXIncludeParseFile(xmlXIncludeCtxtPtr ctxt, const char *URL) {
321
4.00k
    xmlDocPtr ret = NULL;
322
4.00k
    xmlParserCtxtPtr pctxt;
323
4.00k
    xmlParserInputPtr inputStream;
324
325
4.00k
    xmlInitParser();
326
327
4.00k
    pctxt = xmlNewParserCtxt();
328
4.00k
    if (pctxt == NULL) {
329
25
  xmlXIncludeErrMemory(ctxt);
330
25
  return(NULL);
331
25
    }
332
3.97k
    if (ctxt->errorHandler != NULL)
333
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
334
3.97k
    if (ctxt->resourceLoader != NULL)
335
3.97k
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
336
3.97k
                                 ctxt->resourceCtxt);
337
338
    /*
339
     * pass in the application data to the parser context.
340
     */
341
3.97k
    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
3.97k
    if ((ctxt->doc != NULL) && (ctxt->doc->dict != NULL)) {
348
2.66k
       if (pctxt->dict != NULL)
349
2.66k
            xmlDictFree(pctxt->dict);
350
2.66k
  pctxt->dict = ctxt->doc->dict;
351
2.66k
  xmlDictReference(pctxt->dict);
352
2.66k
    }
353
354
    /*
355
     * We set DTDLOAD to make sure that ID attributes declared in
356
     * external DTDs are detected.
357
     */
358
3.97k
    xmlCtxtUseOptions(pctxt, ctxt->parseFlags | XML_PARSE_DTDLOAD);
359
360
3.97k
    inputStream = xmlLoadResource(pctxt, URL, NULL, XML_RESOURCE_XINCLUDE);
361
3.97k
    if (inputStream == NULL)
362
2.26k
        goto error;
363
364
1.71k
    if (xmlCtxtPushInput(pctxt, inputStream) < 0) {
365
1
        xmlFreeInputStream(inputStream);
366
1
        goto error;
367
1
    }
368
369
1.71k
    xmlParseDocument(pctxt);
370
371
1.71k
    if (pctxt->wellFormed) {
372
154
        ret = pctxt->myDoc;
373
154
    }
374
1.56k
    else {
375
1.56k
        ret = NULL;
376
1.56k
  if (pctxt->myDoc != NULL)
377
1.55k
      xmlFreeDoc(pctxt->myDoc);
378
1.56k
        pctxt->myDoc = NULL;
379
1.56k
    }
380
381
3.97k
error:
382
3.97k
    if (xmlCtxtIsCatastrophicError(pctxt))
383
107
        xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "parser error", NULL);
384
3.97k
    xmlFreeParserCtxt(pctxt);
385
386
3.97k
    return(ret);
387
1.71k
}
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
22.8k
xmlXIncludeAddNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur) {
397
22.8k
    xmlXIncludeRefPtr ref = NULL;
398
22.8k
    xmlXIncludeRefPtr ret = NULL;
399
22.8k
    xmlURIPtr uri = NULL;
400
22.8k
    xmlChar *href = NULL;
401
22.8k
    xmlChar *parse = NULL;
402
22.8k
    xmlChar *fragment = NULL;
403
22.8k
    xmlChar *base = NULL;
404
22.8k
    xmlChar *tmp;
405
22.8k
    int xml = 1;
406
22.8k
    int local = 0;
407
22.8k
    int res;
408
409
22.8k
    if (ctxt == NULL)
410
0
  return(NULL);
411
22.8k
    if (cur == NULL)
412
0
  return(NULL);
413
414
    /*
415
     * read the attributes
416
     */
417
418
22.8k
    fragment = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE_XPOINTER);
419
420
22.8k
    href = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_HREF);
421
22.8k
    if (href == NULL) {
422
16.7k
        if (fragment == NULL) {
423
1.43k
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_HREF,
424
1.43k
                     "href or xpointer must be present\n", parse);
425
1.43k
      goto error;
426
1.43k
        }
427
428
15.3k
  href = xmlStrdup(BAD_CAST ""); /* @@@@ href is now optional */
429
15.3k
  if (href == NULL) {
430
4
            xmlXIncludeErrMemory(ctxt);
431
4
      goto error;
432
4
        }
433
15.3k
    } else if (xmlStrlen(href) > XML_MAX_URI_LENGTH) {
434
15
        xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI, "URI too long\n",
435
15
                       NULL);
436
15
        goto error;
437
15
    }
438
439
21.4k
    parse = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE);
440
21.4k
    if (parse != NULL) {
441
793
  if (xmlStrEqual(parse, XINCLUDE_PARSE_XML))
442
242
      xml = 1;
443
551
  else if (xmlStrEqual(parse, XINCLUDE_PARSE_TEXT))
444
514
      xml = 0;
445
37
  else {
446
37
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_PARSE_VALUE,
447
37
                     "invalid value %s for 'parse'\n", parse);
448
37
      goto error;
449
37
  }
450
793
    }
451
452
    /*
453
     * Check the URL and remove any fragment identifier
454
     */
455
21.3k
    res = xmlParseURISafe((const char *)href, &uri);
456
21.3k
    if (uri == NULL) {
457
115
        if (res < 0)
458
3
            xmlXIncludeErrMemory(ctxt);
459
112
        else
460
112
            xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI,
461
112
                           "invalid value href %s\n", href);
462
115
        goto error;
463
115
    }
464
465
21.2k
    if (uri->fragment != NULL) {
466
364
        if (ctxt->legacy != 0) {
467
328
      if (fragment == NULL) {
468
283
    fragment = (xmlChar *) uri->fragment;
469
283
      } else {
470
45
    xmlFree(uri->fragment);
471
45
      }
472
328
  } else {
473
36
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_FRAGMENT_ID,
474
36
       "Invalid fragment identifier in URI %s use the xpointer attribute\n",
475
36
                           href);
476
36
      goto error;
477
36
  }
478
328
  uri->fragment = NULL;
479
328
    }
480
21.2k
    tmp = xmlSaveUri(uri);
481
21.2k
    if (tmp == NULL) {
482
3
  xmlXIncludeErrMemory(ctxt);
483
3
  goto error;
484
3
    }
485
21.2k
    xmlFree(href);
486
21.2k
    href = tmp;
487
488
    /*
489
     * Resolve URI
490
     */
491
492
21.2k
    if (xmlNodeGetBaseSafe(ctxt->doc, cur, &base) < 0) {
493
4
        xmlXIncludeErrMemory(ctxt);
494
4
        goto error;
495
4
    }
496
497
21.2k
    if (href[0] != 0) {
498
5.69k
        if (xmlBuildURISafe(href, base, &tmp) < 0) {
499
10
            xmlXIncludeErrMemory(ctxt);
500
10
            goto error;
501
10
        }
502
5.68k
        if (tmp == NULL) {
503
34
            xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI,
504
34
                           "failed build URL\n", NULL);
505
34
            goto error;
506
34
        }
507
5.65k
        xmlFree(href);
508
5.65k
        href = tmp;
509
510
5.65k
        if (xmlStrEqual(href, ctxt->doc->URL))
511
59
            local = 1;
512
15.5k
    } else {
513
15.5k
        local = 1;
514
15.5k
    }
515
516
    /*
517
     * If local and xml then we need a fragment
518
     */
519
21.1k
    if ((local == 1) && (xml == 1) &&
520
15.5k
        ((fragment == NULL) || (fragment[0] == 0))) {
521
92
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_RECURSION,
522
92
                 "detected a local recursion with no xpointer in %s\n",
523
92
           href);
524
92
  goto error;
525
92
    }
526
527
21.0k
    ref = (xmlXIncludeRefPtr) xmlMalloc(sizeof(xmlXIncludeRef));
528
21.0k
    if (ref == NULL) {
529
3
        xmlXIncludeErrMemory(ctxt);
530
3
        goto error;
531
3
    }
532
21.0k
    memset(ref, 0, sizeof(xmlXIncludeRef));
533
534
21.0k
    ref->elem = cur;
535
21.0k
    ref->xml = xml;
536
21.0k
    ref->URI = href;
537
21.0k
    href = NULL;
538
21.0k
    ref->fragment = fragment;
539
21.0k
    fragment = NULL;
540
541
    /*
542
     * xml:base fixup
543
     */
544
21.0k
    if (((ctxt->parseFlags & XML_PARSE_NOBASEFIX) == 0) &&
545
12.2k
        (cur->doc != NULL) &&
546
12.2k
        ((cur->doc->parseFlags & XML_PARSE_NOBASEFIX) == 0)) {
547
12.2k
        if (base != NULL) {
548
11.7k
            ref->base = base;
549
11.7k
            base = NULL;
550
11.7k
        } else {
551
453
            ref->base = xmlStrdup(BAD_CAST "");
552
453
            if (ref->base == NULL) {
553
1
          xmlXIncludeErrMemory(ctxt);
554
1
                goto error;
555
1
            }
556
453
        }
557
12.2k
    }
558
559
21.0k
    if (ctxt->incNr >= ctxt->incMax) {
560
17.3k
        xmlXIncludeRefPtr *table;
561
17.3k
        int newSize;
562
563
17.3k
        newSize = xmlGrowCapacity(ctxt->incMax, sizeof(table[0]),
564
17.3k
                                  4, XML_MAX_ITEMS);
565
17.3k
        if (newSize < 0) {
566
0
      xmlXIncludeErrMemory(ctxt);
567
0
      goto error;
568
0
  }
569
17.3k
        table = xmlRealloc(ctxt->incTab, newSize * sizeof(table[0]));
570
17.3k
        if (table == NULL) {
571
5
      xmlXIncludeErrMemory(ctxt);
572
5
      goto error;
573
5
  }
574
17.3k
        ctxt->incTab = table;
575
17.3k
        ctxt->incMax = newSize;
576
17.3k
    }
577
21.0k
    ctxt->incTab[ctxt->incNr++] = ref;
578
579
21.0k
    ret = ref;
580
21.0k
    ref = NULL;
581
582
22.8k
error:
583
22.8k
    xmlXIncludeFreeRef(ref);
584
22.8k
    xmlFreeURI(uri);
585
22.8k
    xmlFree(href);
586
22.8k
    xmlFree(parse);
587
22.8k
    xmlFree(fragment);
588
22.8k
    xmlFree(base);
589
22.8k
    return(ret);
590
21.0k
}
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
154
xmlXIncludeRecurseDoc(xmlXIncludeCtxtPtr ctxt, xmlDocPtr doc) {
600
154
    xmlDocPtr oldDoc;
601
154
    xmlXIncludeRefPtr *oldIncTab;
602
154
    int oldIncMax, oldIncNr, oldIsStream;
603
154
    int i;
604
605
154
    oldDoc = ctxt->doc;
606
154
    oldIncMax = ctxt->incMax;
607
154
    oldIncNr = ctxt->incNr;
608
154
    oldIncTab = ctxt->incTab;
609
154
    oldIsStream = ctxt->isStream;
610
154
    ctxt->doc = doc;
611
154
    ctxt->incMax = 0;
612
154
    ctxt->incNr = 0;
613
154
    ctxt->incTab = NULL;
614
154
    ctxt->isStream = 0;
615
616
154
    xmlXIncludeDoProcess(ctxt, xmlDocGetRootElement(doc));
617
618
154
    if (ctxt->incTab != NULL) {
619
340
        for (i = 0; i < ctxt->incNr; i++)
620
250
            xmlXIncludeFreeRef(ctxt->incTab[i]);
621
90
        xmlFree(ctxt->incTab);
622
90
    }
623
624
154
    ctxt->doc = oldDoc;
625
154
    ctxt->incMax = oldIncMax;
626
154
    ctxt->incNr = oldIncNr;
627
154
    ctxt->incTab = oldIncTab;
628
154
    ctxt->isStream = oldIsStream;
629
154
}
630
631
/************************************************************************
632
 *                  *
633
 *      Node copy with specific semantic    *
634
 *                  *
635
 ************************************************************************/
636
637
static void
638
xmlXIncludeBaseFixup(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur, xmlNodePtr copy,
639
95.0k
                     const xmlChar *targetBase) {
640
95.0k
    xmlChar *base = NULL;
641
95.0k
    xmlChar *relBase = NULL;
642
95.0k
    xmlNs ns;
643
95.0k
    int res;
644
645
95.0k
    if (cur->type != XML_ELEMENT_NODE)
646
29.6k
        return;
647
648
65.3k
    if (xmlNodeGetBaseSafe(cur->doc, cur, &base) < 0)
649
71
        xmlXIncludeErrMemory(ctxt);
650
651
65.3k
    if ((base != NULL) && !xmlStrEqual(base, targetBase)) {
652
32.3k
        if ((xmlStrlen(base) > XML_MAX_URI_LENGTH) ||
653
32.0k
            (xmlStrlen(targetBase) > XML_MAX_URI_LENGTH)) {
654
533
            relBase = xmlStrdup(base);
655
533
            if (relBase == NULL) {
656
1
                xmlXIncludeErrMemory(ctxt);
657
1
                goto done;
658
1
            }
659
31.8k
        } else if (xmlBuildRelativeURISafe(base, targetBase, &relBase) < 0) {
660
29
            xmlXIncludeErrMemory(ctxt);
661
29
            goto done;
662
29
        }
663
32.3k
        if (relBase == NULL) {
664
477
            xmlXIncludeErr(ctxt, cur,
665
477
                    XML_XINCLUDE_HREF_URI,
666
477
                    "Building relative URI failed: %s\n",
667
477
                    base);
668
477
            goto done;
669
477
        }
670
671
        /*
672
         * If the new base doesn't contain a slash, it can be omitted.
673
         */
674
31.8k
        if (xmlStrchr(relBase, '/') != NULL) {
675
29.4k
            res = xmlNodeSetBase(copy, relBase);
676
29.4k
            if (res < 0)
677
18
                xmlXIncludeErrMemory(ctxt);
678
29.4k
            goto done;
679
29.4k
        }
680
31.8k
    }
681
682
    /*
683
     * Delete existing xml:base if bases are equal
684
     */
685
35.3k
    memset(&ns, 0, sizeof(ns));
686
35.3k
    ns.href = XML_XML_NAMESPACE;
687
35.3k
    xmlUnsetNsProp(copy, &ns, BAD_CAST "base");
688
689
65.3k
done:
690
65.3k
    xmlFree(base);
691
65.3k
    xmlFree(relBase);
692
65.3k
}
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
10.9k
                    int copyChildren, const xmlChar *targetBase) {
706
10.9k
    xmlNodePtr result = NULL;
707
10.9k
    xmlNodePtr insertParent = NULL;
708
10.9k
    xmlNodePtr insertLast = NULL;
709
10.9k
    xmlNodePtr cur;
710
10.9k
    xmlNodePtr item;
711
10.9k
    int depth = 0;
712
713
10.9k
    if (copyChildren) {
714
2.28k
        cur = elem->children;
715
2.28k
        if (cur == NULL)
716
0
            return(NULL);
717
8.68k
    } else {
718
8.68k
        cur = elem;
719
8.68k
    }
720
721
155k
    while (1) {
722
155k
        xmlNodePtr copy = NULL;
723
155k
        int recurse = 0;
724
725
155k
        if ((cur->type == XML_DOCUMENT_NODE) ||
726
155k
            (cur->type == XML_DTD_NODE)) {
727
0
            ;
728
155k
        } else if ((cur->type == XML_ELEMENT_NODE) &&
729
125k
                   (cur->ns != NULL) &&
730
37.3k
                   (xmlStrEqual(cur->name, XINCLUDE_NODE)) &&
731
3.81k
                   ((xmlStrEqual(cur->ns->href, XINCLUDE_NS)) ||
732
3.25k
                    (xmlStrEqual(cur->ns->href, XINCLUDE_OLD_NS)))) {
733
3.25k
            xmlXIncludeRefPtr ref = xmlXIncludeExpandNode(ctxt, cur);
734
735
3.25k
            if (ref == NULL)
736
914
                goto error;
737
            /*
738
             * TODO: Insert XML_XINCLUDE_START and XML_XINCLUDE_END nodes
739
             */
740
66.7k
            for (item = ref->inc; item != NULL; item = item->next) {
741
64.4k
                copy = xmlStaticCopyNode(item, ctxt->doc, insertParent, 1);
742
64.4k
                if (copy == NULL) {
743
3
                    xmlXIncludeErrMemory(ctxt);
744
3
                    goto error;
745
3
                }
746
747
64.4k
                if (result == NULL)
748
73
                    result = copy;
749
64.4k
                if (insertLast != NULL) {
750
64.2k
                    insertLast->next = copy;
751
64.2k
                    copy->prev = insertLast;
752
64.2k
                } else if (insertParent != NULL) {
753
80
                    insertParent->children = copy;
754
80
                }
755
64.4k
                insertLast = copy;
756
757
64.4k
                if ((depth == 0) && (targetBase != NULL))
758
53.5k
                    xmlXIncludeBaseFixup(ctxt, item, copy, targetBase);
759
64.4k
            }
760
152k
        } else {
761
152k
            copy = xmlStaticCopyNode(cur, ctxt->doc, insertParent, 2);
762
152k
            if (copy == NULL) {
763
25
                xmlXIncludeErrMemory(ctxt);
764
25
                goto error;
765
25
            }
766
767
152k
            if (result == NULL)
768
10.7k
                result = copy;
769
152k
            if (insertLast != NULL) {
770
95.7k
                insertLast->next = copy;
771
95.7k
                copy->prev = insertLast;
772
95.7k
            } else if (insertParent != NULL) {
773
45.7k
                insertParent->children = copy;
774
45.7k
            }
775
152k
            insertLast = copy;
776
777
152k
            if ((depth == 0) && (targetBase != NULL))
778
41.3k
                xmlXIncludeBaseFixup(ctxt, cur, copy, targetBase);
779
780
152k
            recurse = (cur->type != XML_ENTITY_REF_NODE) &&
781
151k
                      (cur->children != NULL);
782
152k
        }
783
784
154k
        if (recurse) {
785
46.0k
            cur = cur->children;
786
46.0k
            insertParent = insertLast;
787
46.0k
            insertLast = NULL;
788
46.0k
            depth += 1;
789
46.0k
            continue;
790
46.0k
        }
791
792
108k
        if (cur == elem)
793
5.17k
            return(result);
794
795
136k
        while (cur->next == NULL) {
796
37.7k
            if (insertParent != NULL)
797
35.5k
                insertParent->last = insertLast;
798
37.7k
            cur = cur->parent;
799
37.7k
            if (cur == elem)
800
4.84k
                return(result);
801
32.9k
            insertLast = insertParent;
802
32.9k
            insertParent = insertParent->parent;
803
32.9k
            depth -= 1;
804
32.9k
        }
805
806
98.6k
        cur = cur->next;
807
98.6k
    }
808
809
942
error:
810
942
    xmlFreeNodeList(result);
811
942
    return(NULL);
812
10.9k
}
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
1.66k
                        const xmlChar *targetBase) {
828
1.66k
    xmlNodePtr list = NULL, last = NULL, copy;
829
1.66k
    int i;
830
831
1.66k
    if ((ctxt == NULL) || (obj == NULL))
832
0
  return(NULL);
833
1.66k
    switch (obj->type) {
834
1.66k
        case XPATH_NODESET: {
835
1.66k
      xmlNodeSetPtr set = obj->nodesetval;
836
1.66k
      if (set == NULL)
837
0
    break;
838
9.39k
      for (i = 0;i < set->nodeNr;i++) {
839
8.68k
                xmlNodePtr node;
840
841
8.68k
    if (set->nodeTab[i] == NULL)
842
0
        continue;
843
8.68k
    switch (set->nodeTab[i]->type) {
844
68
        case XML_DOCUMENT_NODE:
845
68
        case XML_HTML_DOCUMENT_NODE:
846
68
                        node = xmlDocGetRootElement(
847
68
                                (xmlDocPtr) set->nodeTab[i]);
848
68
                        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
68
                        break;
855
1.70k
                    case XML_TEXT_NODE:
856
1.90k
        case XML_CDATA_SECTION_NODE:
857
8.06k
        case XML_ELEMENT_NODE:
858
8.39k
        case XML_PI_NODE:
859
8.61k
        case XML_COMMENT_NODE:
860
8.61k
                        node = set->nodeTab[i];
861
8.61k
      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
8.68k
    }
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
8.68k
    copy = xmlXIncludeCopyNode(ctxt, node, 0, targetBase);
876
8.68k
                if (copy == NULL) {
877
954
                    xmlFreeNodeList(list);
878
954
                    return(NULL);
879
954
                }
880
7.72k
    if (last == NULL) {
881
643
                    list = copy;
882
7.08k
                } else {
883
7.90k
                    while (last->next != NULL)
884
826
                        last = last->next;
885
7.08k
                    copy->prev = last;
886
7.08k
                    last->next = copy;
887
7.08k
    }
888
7.72k
                last = copy;
889
7.72k
      }
890
715
      break;
891
1.66k
  }
892
715
  default:
893
0
      break;
894
1.66k
    }
895
715
    return(list);
896
1.66k
}
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
152
                 const xmlChar *name ATTRIBUTE_UNUSED) {
922
152
    xmlEntityPtr ent = (xmlEntityPtr) payload;
923
152
    xmlXIncludeMergeDataPtr data = (xmlXIncludeMergeDataPtr) vdata;
924
152
    xmlEntityPtr ret, prev;
925
152
    xmlDocPtr doc;
926
152
    xmlXIncludeCtxtPtr ctxt;
927
928
152
    if ((ent == NULL) || (data == NULL))
929
0
  return;
930
152
    ctxt = data->ctxt;
931
152
    doc = data->doc;
932
152
    if ((ctxt == NULL) || (doc == NULL))
933
0
  return;
934
152
    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
94
        case XML_INTERNAL_GENERAL_ENTITY:
940
150
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
941
152
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
942
152
      break;
943
152
    }
944
152
    prev = xmlGetDocEntity(doc, ent->name);
945
152
    if (prev == NULL) {
946
135
        ret = xmlAddDocEntity(doc, ent->name, ent->etype, ent->ExternalID,
947
135
                              ent->SystemID, ent->content);
948
135
        if (ret == NULL) {
949
5
            xmlXIncludeErrMemory(ctxt);
950
5
            return;
951
5
        }
952
130
  if (ent->URI != NULL) {
953
46
      ret->URI = xmlStrdup(ent->URI);
954
46
            if (ret->URI == 0)
955
2
                xmlXIncludeErrMemory(ctxt);
956
46
        }
957
130
    } else {
958
17
        if (ent->etype != prev->etype)
959
5
            goto error;
960
961
12
        if ((ent->SystemID != NULL) && (prev->SystemID != NULL)) {
962
2
            if (!xmlStrEqual(ent->SystemID, prev->SystemID))
963
1
                goto error;
964
10
        } else if ((ent->ExternalID != NULL) &&
965
4
                   (prev->ExternalID != NULL)) {
966
3
            if (!xmlStrEqual(ent->ExternalID, prev->ExternalID))
967
2
                goto error;
968
7
        } else if ((ent->content != NULL) && (prev->content != NULL)) {
969
5
            if (!xmlStrEqual(ent->content, prev->content))
970
3
                goto error;
971
5
        } else {
972
2
            goto error;
973
2
        }
974
12
    }
975
134
    return;
976
134
error:
977
13
    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
5
        case XML_INTERNAL_GENERAL_ENTITY:
982
12
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
983
12
      return;
984
1
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
985
1
      break;
986
13
    }
987
1
    xmlXIncludeErr(ctxt, (xmlNodePtr) ent, XML_XINCLUDE_ENTITY_DEF_MISMATCH,
988
1
                   "mismatch in redefinition of entity %s\n",
989
1
       ent->name);
990
1
}
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
154
                   xmlDocPtr from) {
1003
154
    xmlNodePtr cur;
1004
154
    xmlDtdPtr target, source;
1005
1006
154
    if (ctxt == NULL)
1007
0
  return(-1);
1008
1009
154
    if ((from == NULL) || (from->intSubset == NULL))
1010
72
  return(0);
1011
1012
82
    target = doc->intSubset;
1013
82
    if (target == NULL) {
1014
55
  cur = xmlDocGetRootElement(doc);
1015
55
  if (cur == NULL)
1016
0
      return(-1);
1017
55
        target = xmlCreateIntSubset(doc, cur->name, NULL, NULL);
1018
55
  if (target == NULL) {
1019
1
            xmlXIncludeErrMemory(ctxt);
1020
1
      return(-1);
1021
1
        }
1022
55
    }
1023
1024
81
    source = from->intSubset;
1025
81
    if ((source != NULL) && (source->entities != NULL)) {
1026
44
  xmlXIncludeMergeData data;
1027
1028
44
  data.ctxt = ctxt;
1029
44
  data.doc = doc;
1030
1031
44
  xmlHashScan((xmlHashTablePtr) source->entities,
1032
44
        xmlXIncludeMergeEntity, &data);
1033
44
    }
1034
81
    source = from->extSubset;
1035
81
    if ((source != NULL) && (source->entities != NULL)) {
1036
3
  xmlXIncludeMergeData data;
1037
1038
3
  data.ctxt = ctxt;
1039
3
  data.doc = doc;
1040
1041
  /*
1042
   * don't duplicate existing stuff when external subsets are the same
1043
   */
1044
3
  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
3
    }
1050
81
    return(0);
1051
82
}
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
20.5k
xmlXIncludeLoadDoc(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1062
20.5k
    xmlXIncludeDocPtr cache;
1063
20.5k
    xmlDocPtr doc;
1064
20.5k
    const xmlChar *url = ref->URI;
1065
20.5k
    const xmlChar *fragment = ref->fragment;
1066
20.5k
    int i = 0;
1067
20.5k
    int ret = -1;
1068
20.5k
    int cacheNr;
1069
20.5k
#ifdef LIBXML_XPTR_ENABLED
1070
20.5k
    int saveFlags;
1071
20.5k
#endif
1072
1073
    /*
1074
     * Handling of references to the local document are done
1075
     * directly through ctxt->doc.
1076
     */
1077
20.5k
    if ((url[0] == 0) || (url[0] == '#') ||
1078
15.5k
  ((ctxt->doc != NULL) && (xmlStrEqual(url, ctxt->doc->URL)))) {
1079
15.5k
  doc = ctxt->doc;
1080
15.5k
        goto loaded;
1081
15.5k
    }
1082
1083
    /*
1084
     * Prevent reloading the document twice.
1085
     */
1086
9.73k
    for (i = 0; i < ctxt->urlNr; i++) {
1087
5.73k
  if (xmlStrEqual(url, ctxt->urlTab[i].url)) {
1088
995
            if (ctxt->urlTab[i].expanding) {
1089
95
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_RECURSION,
1090
95
                               "inclusion loop detected\n", NULL);
1091
95
                goto error;
1092
95
            }
1093
900
      doc = ctxt->urlTab[i].doc;
1094
900
            if (doc == NULL)
1095
816
                goto error;
1096
84
      goto loaded;
1097
900
  }
1098
5.73k
    }
1099
1100
    /*
1101
     * Load it.
1102
     */
1103
4.00k
#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
4.00k
    saveFlags = ctxt->parseFlags;
1110
4.00k
    if (fragment != NULL) { /* if this is an XPointer eval */
1111
522
  ctxt->parseFlags |= XML_PARSE_NOENT;
1112
522
    }
1113
4.00k
#endif
1114
1115
4.00k
    doc = xmlXIncludeParseFile(ctxt, (const char *)url);
1116
4.00k
#ifdef LIBXML_XPTR_ENABLED
1117
4.00k
    ctxt->parseFlags = saveFlags;
1118
4.00k
#endif
1119
1120
    /* Also cache NULL docs */
1121
4.00k
    if (ctxt->urlNr >= ctxt->urlMax) {
1122
3.82k
        xmlXIncludeDoc *tmp;
1123
3.82k
        int newSize;
1124
1125
3.82k
        newSize = xmlGrowCapacity(ctxt->urlMax, sizeof(tmp[0]),
1126
3.82k
                                  8, XML_MAX_ITEMS);
1127
3.82k
        if (newSize < 0) {
1128
0
            xmlXIncludeErrMemory(ctxt);
1129
0
            xmlFreeDoc(doc);
1130
0
            goto error;
1131
0
        }
1132
3.82k
        tmp = xmlRealloc(ctxt->urlTab, newSize * sizeof(tmp[0]));
1133
3.82k
        if (tmp == NULL) {
1134
5
            xmlXIncludeErrMemory(ctxt);
1135
5
            xmlFreeDoc(doc);
1136
5
            goto error;
1137
5
        }
1138
3.81k
        ctxt->urlMax = newSize;
1139
3.81k
        ctxt->urlTab = tmp;
1140
3.81k
    }
1141
3.99k
    cache = &ctxt->urlTab[ctxt->urlNr];
1142
3.99k
    cache->doc = doc;
1143
3.99k
    cache->url = xmlStrdup(url);
1144
3.99k
    if (cache->url == NULL) {
1145
6
        xmlXIncludeErrMemory(ctxt);
1146
6
        xmlFreeDoc(doc);
1147
6
        goto error;
1148
6
    }
1149
3.99k
    cache->expanding = 0;
1150
3.99k
    cacheNr = ctxt->urlNr++;
1151
1152
3.99k
    if (doc == NULL)
1153
3.83k
        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
154
    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
154
    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
154
    cache->expanding = 1;
1182
154
    xmlXIncludeRecurseDoc(ctxt, doc);
1183
    /* urlTab might be reallocated. */
1184
154
    cache = &ctxt->urlTab[cacheNr];
1185
154
    cache->expanding = 0;
1186
1187
15.8k
loaded:
1188
15.8k
    if (fragment == NULL) {
1189
273
        xmlNodePtr root;
1190
1191
273
        root = xmlDocGetRootElement(doc);
1192
273
        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
273
        ref->inc = xmlDocCopyNode(root, ctxt->doc, 1);
1199
273
        if (ref->inc == NULL) {
1200
10
            xmlXIncludeErrMemory(ctxt);
1201
10
            goto error;
1202
10
        }
1203
1204
263
        if (ref->base != NULL)
1205
168
            xmlXIncludeBaseFixup(ctxt, root, ref->inc, ref->base);
1206
263
    }
1207
15.5k
#ifdef LIBXML_XPTR_ENABLED
1208
15.5k
    else {
1209
  /*
1210
   * Computes the XPointer expression and make a copy used
1211
   * as the replacement copy.
1212
   */
1213
15.5k
  xmlXPathObjectPtr xptr;
1214
15.5k
  xmlNodeSetPtr set;
1215
1216
15.5k
        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
15.5k
        if (ctxt->xpctxt == NULL) {
1224
7.47k
            ctxt->xpctxt = xmlXPathNewContext(doc);
1225
7.47k
            if (ctxt->xpctxt == NULL) {
1226
2
                xmlXIncludeErrMemory(ctxt);
1227
2
                goto error;
1228
2
            }
1229
7.47k
            if (ctxt->errorHandler != NULL)
1230
0
                xmlXPathSetErrorHandler(ctxt->xpctxt, ctxt->errorHandler,
1231
0
                                        ctxt->errorCtxt);
1232
7.47k
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
1233
7.47k
            ctxt->xpctxt->opLimit = 100000;
1234
7.47k
#endif
1235
8.06k
        } else {
1236
8.06k
            ctxt->xpctxt->doc = doc;
1237
8.06k
        }
1238
15.5k
  xptr = xmlXPtrEval(fragment, ctxt->xpctxt);
1239
15.5k
  if (ctxt->xpctxt->lastError.code != XML_ERR_OK) {
1240
11.0k
            if (ctxt->xpctxt->lastError.code == XML_ERR_NO_MEMORY)
1241
1.31k
                xmlXIncludeErrMemory(ctxt);
1242
9.77k
            else
1243
9.77k
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_FAILED,
1244
9.77k
                               "XPointer evaluation failed: #%s\n",
1245
9.77k
                               fragment);
1246
11.0k
            goto error;
1247
11.0k
  }
1248
4.44k
        if (xptr == NULL)
1249
2.64k
            goto done;
1250
1.80k
  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
1.80k
      case XPATH_NODESET:
1263
1.80k
                break;
1264
1265
1.80k
  }
1266
1.80k
  set = xptr->nodesetval;
1267
1.80k
  if (set != NULL) {
1268
225k
      for (i = 0;i < set->nodeNr;i++) {
1269
223k
    if (set->nodeTab[i] == NULL) /* shouldn't happen */
1270
0
        continue;
1271
223k
    switch (set->nodeTab[i]->type) {
1272
207k
        case XML_ELEMENT_NODE:
1273
222k
        case XML_TEXT_NODE:
1274
222k
        case XML_CDATA_SECTION_NODE:
1275
222k
        case XML_ENTITY_REF_NODE:
1276
222k
        case XML_ENTITY_NODE:
1277
222k
        case XML_PI_NODE:
1278
223k
        case XML_COMMENT_NODE:
1279
223k
        case XML_DOCUMENT_NODE:
1280
223k
        case XML_HTML_DOCUMENT_NODE:
1281
223k
      continue;
1282
1283
79
        case XML_ATTRIBUTE_NODE:
1284
79
      xmlXIncludeErr(ctxt, ref->elem,
1285
79
                     XML_XINCLUDE_XPTR_RESULT,
1286
79
               "XPointer selects an attribute: #%s\n",
1287
79
               fragment);
1288
79
      goto xptr_error;
1289
58
        case XML_NAMESPACE_DECL:
1290
58
      xmlXIncludeErr(ctxt, ref->elem,
1291
58
                     XML_XINCLUDE_XPTR_RESULT,
1292
58
               "XPointer selects a namespace: #%s\n",
1293
58
               fragment);
1294
58
      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
223k
    }
1311
223k
      }
1312
1.80k
  }
1313
1.66k
        ref->inc = xmlXIncludeCopyXPointer(ctxt, xptr, ref->base);
1314
1.80k
xptr_error:
1315
1.80k
        xmlXPathFreeObject(xptr);
1316
1.80k
    }
1317
1318
4.70k
done:
1319
4.70k
#endif
1320
1321
4.70k
    ret = 0;
1322
1323
20.5k
error:
1324
20.5k
    return(ret);
1325
4.70k
}
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
514
xmlXIncludeLoadTxt(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1336
514
    xmlParserInputBufferPtr buf;
1337
514
    xmlNodePtr node = NULL;
1338
514
    const xmlChar *url = ref->URI;
1339
514
    int i;
1340
514
    int ret = -1;
1341
514
    xmlChar *encoding = NULL;
1342
514
    xmlCharEncodingHandlerPtr handler = NULL;
1343
514
    xmlParserCtxtPtr pctxt = NULL;
1344
514
    xmlParserInputPtr inputStream = NULL;
1345
514
    int len;
1346
514
    int res;
1347
514
    const xmlChar *content;
1348
1349
    /*
1350
     * Handling of references to the local document are done
1351
     * directly through ctxt->doc.
1352
     */
1353
514
    if (url[0] == 0) {
1354
49
  xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_TEXT_DOCUMENT,
1355
49
           "text serialization of document not available\n", NULL);
1356
49
  goto error;
1357
49
    }
1358
1359
    /*
1360
     * Prevent reloading the document twice.
1361
     */
1362
575
    for (i = 0; i < ctxt->txtNr; i++) {
1363
160
  if (xmlStrEqual(url, ctxt->txtTab[i].url)) {
1364
50
            node = xmlNewDocText(ctxt->doc, ctxt->txtTab[i].text);
1365
50
            if (node == NULL)
1366
1
                xmlXIncludeErrMemory(ctxt);
1367
50
      goto loaded;
1368
50
  }
1369
160
    }
1370
1371
    /*
1372
     * Try to get the encoding if available
1373
     */
1374
415
    if (ref->elem != NULL) {
1375
415
  encoding = xmlXIncludeGetProp(ctxt, ref->elem, XINCLUDE_PARSE_ENCODING);
1376
415
    }
1377
415
    if (encoding != NULL) {
1378
151
        xmlParserErrors code;
1379
1380
151
        code = xmlOpenCharEncodingHandler((const char *) encoding,
1381
151
                                          /* output */ 0, &handler);
1382
1383
151
        if (code != XML_ERR_OK) {
1384
44
            if (code == XML_ERR_NO_MEMORY) {
1385
3
                xmlXIncludeErrMemory(ctxt);
1386
41
            } else if (code == XML_ERR_UNSUPPORTED_ENCODING) {
1387
41
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_UNKNOWN_ENCODING,
1388
41
                               "encoding %s not supported\n", encoding);
1389
41
                goto error;
1390
41
            } 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
44
        }
1396
151
    }
1397
1398
    /*
1399
     * Load it.
1400
     */
1401
374
    pctxt = xmlNewParserCtxt();
1402
374
    if (pctxt == NULL) {
1403
3
        xmlXIncludeErrMemory(ctxt);
1404
3
        goto error;
1405
3
    }
1406
371
    if (ctxt->errorHandler != NULL)
1407
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
1408
371
    if (ctxt->resourceLoader != NULL)
1409
371
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
1410
371
                                 ctxt->resourceCtxt);
1411
1412
371
    xmlCtxtUseOptions(pctxt, ctxt->parseFlags);
1413
1414
371
    inputStream = xmlLoadResource(pctxt, (const char*) url, NULL,
1415
371
                                  XML_RESOURCE_XINCLUDE_TEXT);
1416
371
    if (inputStream == NULL) {
1417
        /*
1418
         * ENOENT only produces a warning which isn't reflected in errNo.
1419
         */
1420
114
        if (pctxt->errNo == XML_ERR_NO_MEMORY)
1421
2
            xmlXIncludeErrMemory(ctxt);
1422
112
        else if ((pctxt->errNo != XML_ERR_OK) &&
1423
1
                 (pctxt->errNo != XML_IO_ENOENT) &&
1424
1
                 (pctxt->errNo != XML_IO_UNKNOWN) &&
1425
1
                 (pctxt->errNo != XML_IO_NETWORK_ATTEMPT))
1426
1
            xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "load error", NULL);
1427
114
  goto error;
1428
114
    }
1429
257
    buf = inputStream->buf;
1430
257
    if (buf == NULL)
1431
0
  goto error;
1432
257
    if (buf->encoder)
1433
0
  xmlCharEncCloseFunc(buf->encoder);
1434
257
    buf->encoder = handler;
1435
257
    handler = NULL;
1436
1437
257
    node = xmlNewDocText(ctxt->doc, NULL);
1438
257
    if (node == NULL) {
1439
2
        xmlXIncludeErrMemory(ctxt);
1440
2
  goto error;
1441
2
    }
1442
1443
    /*
1444
     * Scan all chars from the resource and add the to the node
1445
     */
1446
255
    do {
1447
255
        res = xmlParserInputBufferRead(buf, 4096);
1448
255
    } while (res > 0);
1449
255
    if (res < 0) {
1450
0
        if (buf->error == XML_ERR_NO_MEMORY)
1451
0
            xmlXIncludeErrMemory(ctxt);
1452
0
        else
1453
0
            xmlXIncludeErr(ctxt, NULL, buf->error, "read error", NULL);
1454
0
        goto error;
1455
0
    }
1456
1457
255
    content = xmlBufContent(buf->buffer);
1458
255
    len = xmlBufUse(buf->buffer);
1459
10.7k
    for (i = 0; i < len;) {
1460
10.6k
        int cur;
1461
10.6k
        int l;
1462
1463
10.6k
        l = len - i;
1464
10.6k
        cur = xmlGetUTF8Char(&content[i], &l);
1465
10.6k
        if ((cur < 0) || (!IS_CHAR(cur))) {
1466
204
            xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_INVALID_CHAR,
1467
204
                           "%s contains invalid char\n", url);
1468
204
            goto error;
1469
204
        }
1470
1471
10.4k
        i += l;
1472
10.4k
    }
1473
1474
51
    if (xmlNodeAddContentLen(node, content, len) < 0)
1475
1
        xmlXIncludeErrMemory(ctxt);
1476
1477
51
    if (ctxt->txtNr >= ctxt->txtMax) {
1478
51
        xmlXIncludeTxt *tmp;
1479
51
        int newSize;
1480
1481
51
        newSize = xmlGrowCapacity(ctxt->txtMax, sizeof(tmp[0]),
1482
51
                                  8, XML_MAX_ITEMS);
1483
51
        if (newSize < 0) {
1484
0
            xmlXIncludeErrMemory(ctxt);
1485
0
      goto error;
1486
0
        }
1487
51
        tmp = xmlRealloc(ctxt->txtTab, newSize * sizeof(tmp[0]));
1488
51
        if (tmp == NULL) {
1489
1
            xmlXIncludeErrMemory(ctxt);
1490
1
      goto error;
1491
1
        }
1492
50
        ctxt->txtMax = newSize;
1493
50
        ctxt->txtTab = tmp;
1494
50
    }
1495
50
    ctxt->txtTab[ctxt->txtNr].text = xmlStrdup(node->content);
1496
50
    if ((node->content != NULL) &&
1497
48
        (ctxt->txtTab[ctxt->txtNr].text == NULL)) {
1498
1
        xmlXIncludeErrMemory(ctxt);
1499
1
        goto error;
1500
1
    }
1501
49
    ctxt->txtTab[ctxt->txtNr].url = xmlStrdup(url);
1502
49
    if (ctxt->txtTab[ctxt->txtNr].url == NULL) {
1503
1
        xmlXIncludeErrMemory(ctxt);
1504
1
        xmlFree(ctxt->txtTab[ctxt->txtNr].text);
1505
1
        goto error;
1506
1
    }
1507
48
    ctxt->txtNr++;
1508
1509
98
loaded:
1510
    /*
1511
     * Add the element as the replacement copy.
1512
     */
1513
98
    ref->inc = node;
1514
98
    node = NULL;
1515
98
    ret = 0;
1516
1517
514
error:
1518
514
    xmlFreeNode(node);
1519
514
    xmlFreeInputStream(inputStream);
1520
514
    xmlFreeParserCtxt(pctxt);
1521
514
    xmlCharEncCloseFunc(handler);
1522
514
    xmlFree(encoding);
1523
514
    return(ret);
1524
98
}
1525
1526
/**
1527
 * Load the content of the fallback node, and store the result
1528
 * in the XInclude context
1529
 *
1530
 * @param ctxt  the XInclude context
1531
 * @param fallback  the fallback node
1532
 * @param ref  an XMLXincludeRefPtr
1533
 * @returns 0 in case of success, -1 in case of failure
1534
 */
1535
static int
1536
xmlXIncludeLoadFallback(xmlXIncludeCtxtPtr ctxt, xmlNodePtr fallback,
1537
2.35k
                        xmlXIncludeRefPtr ref) {
1538
2.35k
    int ret = 0;
1539
2.35k
    int oldNbErrors;
1540
1541
2.35k
    if ((fallback == NULL) || (fallback->type == XML_NAMESPACE_DECL) ||
1542
2.35k
        (ctxt == NULL))
1543
0
  return(-1);
1544
2.35k
    if (fallback->children != NULL) {
1545
  /*
1546
   * It's possible that the fallback also has 'includes'
1547
   * (Bug 129969), so we re-process the fallback just in case
1548
   */
1549
2.28k
        oldNbErrors = ctxt->nbErrors;
1550
2.28k
  ref->inc = xmlXIncludeCopyNode(ctxt, fallback, 1, ref->base);
1551
2.28k
  if (ctxt->nbErrors > oldNbErrors)
1552
483
      ret = -1;
1553
2.28k
    } else {
1554
74
        ref->inc = NULL;
1555
74
    }
1556
2.35k
    ref->fallback = 1;
1557
2.35k
    return(ret);
1558
2.35k
}
1559
1560
/************************************************************************
1561
 *                  *
1562
 *      XInclude Processing       *
1563
 *                  *
1564
 ************************************************************************/
1565
1566
/**
1567
 * If the XInclude node wasn't processed yet, create a new RefPtr,
1568
 * add it to ctxt->incTab and load the included items.
1569
 *
1570
 * @param ctxt  an XInclude context
1571
 * @param node  an XInclude node
1572
 * @returns the new or existing xmlXIncludeRef, or NULL in case of error.
1573
 */
1574
static xmlXIncludeRefPtr
1575
26.5k
xmlXIncludeExpandNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1576
26.5k
    xmlXIncludeRefPtr ref;
1577
26.5k
    int i;
1578
1579
26.5k
    if (ctxt->fatalErr)
1580
524
        return(NULL);
1581
26.0k
    if (ctxt->depth >= XINCLUDE_MAX_DEPTH) {
1582
0
        xmlXIncludeErr(ctxt, node, XML_XINCLUDE_RECURSION,
1583
0
                       "maximum recursion depth exceeded\n", NULL);
1584
0
        ctxt->fatalErr = 1;
1585
0
        return(NULL);
1586
0
    }
1587
1588
26.0k
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
1589
    /*
1590
     * The XInclude engine offers no protection against exponential
1591
     * expansion attacks similar to "billion laughs". Avoid timeouts by
1592
     * limiting the total number of replacements when fuzzing.
1593
     *
1594
     * Unfortuately, a single XInclude can already result in quadratic
1595
     * behavior:
1596
     *
1597
     *     <doc xmlns:xi="http://www.w3.org/2001/XInclude">
1598
     *       <xi:include xpointer="xpointer(//e)"/>
1599
     *       <e>
1600
     *         <e>
1601
     *           <e>
1602
     *             <!-- more nested elements -->
1603
     *           </e>
1604
     *         </e>
1605
     *       </e>
1606
     *     </doc>
1607
     */
1608
26.0k
    if (ctxt->incTotal >= 20)
1609
1.53k
        return(NULL);
1610
24.4k
    ctxt->incTotal++;
1611
24.4k
#endif
1612
1613
82.3k
    for (i = 0; i < ctxt->incNr; i++) {
1614
59.5k
        if (ctxt->incTab[i]->elem == node) {
1615
1.59k
            if (ctxt->incTab[i]->expanding) {
1616
764
                xmlXIncludeErr(ctxt, node, XML_XINCLUDE_RECURSION,
1617
764
                               "inclusion loop detected\n", NULL);
1618
764
                return(NULL);
1619
764
            }
1620
830
            return(ctxt->incTab[i]);
1621
1.59k
        }
1622
59.5k
    }
1623
1624
22.8k
    ref = xmlXIncludeAddNode(ctxt, node);
1625
22.8k
    if (ref == NULL)
1626
1.79k
        return(NULL);
1627
21.0k
    ref->expanding = 1;
1628
21.0k
    ctxt->depth++;
1629
21.0k
    xmlXIncludeLoadNode(ctxt, ref);
1630
21.0k
    ctxt->depth--;
1631
21.0k
    ref->expanding = 0;
1632
1633
21.0k
    return(ref);
1634
22.8k
}
1635
1636
/**
1637
 * Find and load the infoset replacement for the given node.
1638
 *
1639
 * @param ctxt  an XInclude context
1640
 * @param ref  an xmlXIncludeRef
1641
 * @returns 0 if substitution succeeded, -1 if some processing failed
1642
 */
1643
static int
1644
21.0k
xmlXIncludeLoadNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1645
21.0k
    xmlNodePtr cur;
1646
21.0k
    int ret;
1647
1648
21.0k
    if ((ctxt == NULL) || (ref == NULL))
1649
0
  return(-1);
1650
21.0k
    cur = ref->elem;
1651
21.0k
    if (cur == NULL)
1652
0
  return(-1);
1653
1654
21.0k
    if (ref->xml) {
1655
20.5k
  ret = xmlXIncludeLoadDoc(ctxt, ref);
1656
  /* xmlXIncludeGetFragment(ctxt, cur, URI); */
1657
20.5k
    } else {
1658
514
  ret = xmlXIncludeLoadTxt(ctxt, ref);
1659
514
    }
1660
1661
21.0k
    if (ret < 0) {
1662
16.2k
  xmlNodePtr children;
1663
1664
  /*
1665
   * Time to try a fallback if available
1666
   */
1667
16.2k
  children = cur->children;
1668
28.7k
  while (children != NULL) {
1669
14.8k
      if ((children->type == XML_ELEMENT_NODE) &&
1670
8.80k
    (children->ns != NULL) &&
1671
4.24k
    (xmlStrEqual(children->name, XINCLUDE_FALLBACK)) &&
1672
2.56k
    ((xmlStrEqual(children->ns->href, XINCLUDE_NS)) ||
1673
2.35k
     (xmlStrEqual(children->ns->href, XINCLUDE_OLD_NS)))) {
1674
2.35k
    ret = xmlXIncludeLoadFallback(ctxt, children, ref);
1675
2.35k
    break;
1676
2.35k
      }
1677
12.4k
      children = children->next;
1678
12.4k
  }
1679
16.2k
    }
1680
21.0k
    if (ret < 0) {
1681
14.4k
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_FALLBACK,
1682
14.4k
           "could not load %s, and no fallback was found\n",
1683
14.4k
           ref->URI);
1684
14.4k
    }
1685
1686
21.0k
    return(0);
1687
21.0k
}
1688
1689
/**
1690
 * Implement the infoset replacement for the given node
1691
 *
1692
 * @param ctxt  an XInclude context
1693
 * @param ref  an xmlXIncludeRef
1694
 * @returns 0 if substitution succeeded, -1 if some processing failed
1695
 */
1696
static int
1697
19.5k
xmlXIncludeIncludeNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1698
19.5k
    xmlNodePtr cur, end, list, tmp;
1699
1700
19.5k
    if ((ctxt == NULL) || (ref == NULL))
1701
0
  return(-1);
1702
19.5k
    cur = ref->elem;
1703
19.5k
    if ((cur == NULL) || (cur->type == XML_NAMESPACE_DECL))
1704
0
  return(-1);
1705
1706
19.5k
    list = ref->inc;
1707
19.5k
    ref->inc = NULL;
1708
1709
    /*
1710
     * Check against the risk of generating a multi-rooted document
1711
     */
1712
19.5k
    if ((cur->parent != NULL) &&
1713
19.5k
  (cur->parent->type != XML_ELEMENT_NODE)) {
1714
94
  int nb_elem = 0;
1715
1716
94
  tmp = list;
1717
398
  while (tmp != NULL) {
1718
304
      if (tmp->type == XML_ELEMENT_NODE)
1719
222
    nb_elem++;
1720
304
      tmp = tmp->next;
1721
304
  }
1722
94
        if (nb_elem != 1) {
1723
83
            if (nb_elem > 1)
1724
8
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1725
8
                               "XInclude error: would result in multiple root "
1726
8
                               "nodes\n", NULL);
1727
75
            else
1728
75
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1729
75
                               "XInclude error: would result in no root "
1730
75
                               "node\n", NULL);
1731
83
            xmlFreeNodeList(list);
1732
83
      return(-1);
1733
83
  }
1734
94
    }
1735
1736
19.4k
    if (ctxt->parseFlags & XML_PARSE_NOXINCNODE) {
1737
  /*
1738
   * Add the list of nodes
1739
         *
1740
         * TODO: Coalesce text nodes unless we are streaming mode.
1741
   */
1742
10.7k
  while (list != NULL) {
1743
6.42k
      end = list;
1744
6.42k
      list = list->next;
1745
1746
6.42k
      if (xmlAddPrevSibling(cur, end) == NULL) {
1747
0
                xmlUnlinkNode(end);
1748
0
                xmlFreeNode(end);
1749
0
                goto err_memory;
1750
0
            }
1751
6.42k
  }
1752
4.31k
  xmlUnlinkNode(cur);
1753
4.31k
  xmlFreeNode(cur);
1754
15.1k
    } else {
1755
15.1k
        xmlNodePtr child, next;
1756
1757
  /*
1758
   * Change the current node as an XInclude start one, and add an
1759
   * XInclude end one
1760
   */
1761
15.1k
        if (ref->fallback)
1762
1.15k
            xmlUnsetProp(cur, BAD_CAST "href");
1763
15.1k
  cur->type = XML_XINCLUDE_START;
1764
        /* Remove fallback children */
1765
29.9k
        for (child = cur->children; child != NULL; child = next) {
1766
14.7k
            next = child->next;
1767
14.7k
            xmlUnlinkNode(child);
1768
14.7k
            xmlFreeNode(child);
1769
14.7k
        }
1770
15.1k
  end = xmlNewDocNode(cur->doc, cur->ns, cur->name, NULL);
1771
15.1k
  if (end == NULL)
1772
37
            goto err_memory;
1773
15.1k
  end->type = XML_XINCLUDE_END;
1774
15.1k
  if (xmlAddNextSibling(cur, end) == NULL) {
1775
0
            xmlFreeNode(end);
1776
0
            goto err_memory;
1777
0
        }
1778
1779
  /*
1780
   * Add the list of nodes
1781
   */
1782
48.3k
  while (list != NULL) {
1783
33.2k
      cur = list;
1784
33.2k
      list = list->next;
1785
1786
33.2k
      if (xmlAddPrevSibling(end, cur) == NULL) {
1787
0
                xmlUnlinkNode(cur);
1788
0
                xmlFreeNode(cur);
1789
0
                goto err_memory;
1790
0
            }
1791
33.2k
  }
1792
15.1k
    }
1793
1794
1795
19.4k
    return(0);
1796
1797
37
err_memory:
1798
37
    xmlXIncludeErrMemory(ctxt);
1799
37
    xmlFreeNodeList(list);
1800
37
    return(-1);
1801
19.4k
}
1802
1803
/**
1804
 * test if the node is an XInclude node
1805
 *
1806
 * @param ctxt  the XInclude processing context
1807
 * @param node  an XInclude node
1808
 * @returns 1 true, 0 otherwise
1809
 */
1810
static int
1811
852k
xmlXIncludeTestNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1812
852k
    if (node == NULL)
1813
0
  return(0);
1814
852k
    if (node->type != XML_ELEMENT_NODE)
1815
322k
  return(0);
1816
529k
    if (node->ns == NULL)
1817
251k
  return(0);
1818
278k
    if ((xmlStrEqual(node->ns->href, XINCLUDE_NS)) ||
1819
274k
        (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS))) {
1820
34.9k
  if (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS)) {
1821
31.0k
      if (ctxt->legacy == 0) {
1822
8.38k
          ctxt->legacy = 1;
1823
8.38k
      }
1824
31.0k
  }
1825
34.9k
  if (xmlStrEqual(node->name, XINCLUDE_NODE)) {
1826
26.8k
      xmlNodePtr child = node->children;
1827
26.8k
      int nb_fallback = 0;
1828
1829
53.8k
      while (child != NULL) {
1830
30.3k
    if ((child->type == XML_ELEMENT_NODE) &&
1831
17.2k
        (child->ns != NULL) &&
1832
9.16k
        ((xmlStrEqual(child->ns->href, XINCLUDE_NS)) ||
1833
8.60k
         (xmlStrEqual(child->ns->href, XINCLUDE_OLD_NS)))) {
1834
6.09k
        if (xmlStrEqual(child->name, XINCLUDE_NODE)) {
1835
3.36k
      xmlXIncludeErr(ctxt, node,
1836
3.36k
                     XML_XINCLUDE_INCLUDE_IN_INCLUDE,
1837
3.36k
               "%s has an 'include' child\n",
1838
3.36k
               XINCLUDE_NODE);
1839
3.36k
      return(0);
1840
3.36k
        }
1841
2.73k
        if (xmlStrEqual(child->name, XINCLUDE_FALLBACK)) {
1842
2.09k
      nb_fallback++;
1843
2.09k
        }
1844
2.73k
    }
1845
27.0k
    child = child->next;
1846
27.0k
      }
1847
23.4k
      if (nb_fallback > 1) {
1848
207
    xmlXIncludeErr(ctxt, node, XML_XINCLUDE_FALLBACKS_IN_INCLUDE,
1849
207
             "%s has multiple fallback children\n",
1850
207
                   XINCLUDE_NODE);
1851
207
    return(0);
1852
207
      }
1853
23.2k
      return(1);
1854
23.4k
  }
1855
8.12k
  if (xmlStrEqual(node->name, XINCLUDE_FALLBACK)) {
1856
2.31k
      if ((node->parent == NULL) ||
1857
2.31k
    (node->parent->type != XML_ELEMENT_NODE) ||
1858
2.30k
    (node->parent->ns == NULL) ||
1859
1.46k
    ((!xmlStrEqual(node->parent->ns->href, XINCLUDE_NS)) &&
1860
1.25k
     (!xmlStrEqual(node->parent->ns->href, XINCLUDE_OLD_NS))) ||
1861
1.61k
    (!xmlStrEqual(node->parent->name, XINCLUDE_NODE))) {
1862
1.61k
    xmlXIncludeErr(ctxt, node,
1863
1.61k
                   XML_XINCLUDE_FALLBACK_NOT_IN_INCLUDE,
1864
1.61k
             "%s is not the child of an 'include'\n",
1865
1.61k
             XINCLUDE_FALLBACK);
1866
1.61k
      }
1867
2.31k
  }
1868
8.12k
    }
1869
251k
    return(0);
1870
278k
}
1871
1872
/**
1873
 * Implement the XInclude substitution on the XML document `doc`
1874
 *
1875
 * @param ctxt  the XInclude processing context
1876
 * @param tree  the top of the tree to process
1877
 * @returns 0 if no substitution were done, -1 if some processing failed
1878
 *    or the number of substitutions done.
1879
 */
1880
static int
1881
19.7k
xmlXIncludeDoProcess(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1882
19.7k
    xmlXIncludeRefPtr ref;
1883
19.7k
    xmlNodePtr cur;
1884
19.7k
    int ret = 0;
1885
19.7k
    int i, start;
1886
1887
    /*
1888
     * First phase: lookup the elements in the document
1889
     */
1890
19.7k
    start = ctxt->incNr;
1891
19.7k
    cur = tree;
1892
852k
    do {
1893
  /* TODO: need to work on entities -> stack */
1894
852k
        if (xmlXIncludeTestNode(ctxt, cur) == 1) {
1895
23.2k
            ref = xmlXIncludeExpandNode(ctxt, cur);
1896
            /*
1897
             * Mark direct includes.
1898
             */
1899
23.2k
            if (ref != NULL)
1900
19.5k
                ref->replace = 1;
1901
828k
        } else if ((cur->children != NULL) &&
1902
194k
                   ((cur->type == XML_DOCUMENT_NODE) ||
1903
186k
                    (cur->type == XML_ELEMENT_NODE))) {
1904
186k
            cur = cur->children;
1905
186k
            continue;
1906
186k
        }
1907
852k
        do {
1908
852k
            if (cur == tree)
1909
19.7k
                break;
1910
832k
            if (cur->next != NULL) {
1911
645k
                cur = cur->next;
1912
645k
                break;
1913
645k
            }
1914
186k
            cur = cur->parent;
1915
186k
        } while (cur != NULL);
1916
852k
    } while ((cur != NULL) && (cur != tree));
1917
1918
    /*
1919
     * Second phase: extend the original document infoset.
1920
     */
1921
40.7k
    for (i = start; i < ctxt->incNr; i++) {
1922
21.0k
  if (ctxt->incTab[i]->replace != 0) {
1923
19.5k
            xmlXIncludeIncludeNode(ctxt, ctxt->incTab[i]);
1924
19.5k
            ctxt->incTab[i]->replace = 0;
1925
19.5k
        } else {
1926
            /*
1927
             * Ignore includes which were added indirectly, for example
1928
             * inside xi:fallback elements.
1929
             */
1930
1.50k
            if (ctxt->incTab[i]->inc != NULL) {
1931
970
                xmlFreeNodeList(ctxt->incTab[i]->inc);
1932
970
                ctxt->incTab[i]->inc = NULL;
1933
970
            }
1934
1.50k
        }
1935
21.0k
  ret++;
1936
21.0k
    }
1937
1938
19.7k
    if (ctxt->isStream) {
1939
        /*
1940
         * incTab references nodes which will eventually be deleted in
1941
         * streaming mode. The table is only required for XPointer
1942
         * expressions which aren't allowed in streaming mode.
1943
         */
1944
0
        for (i = 0;i < ctxt->incNr;i++) {
1945
0
            xmlXIncludeFreeRef(ctxt->incTab[i]);
1946
0
        }
1947
0
        ctxt->incNr = 0;
1948
0
    }
1949
1950
19.7k
    return(ret);
1951
19.7k
}
1952
1953
/**
1954
 * Implement the XInclude substitution on the XML document `doc`
1955
 *
1956
 * @param ctxt  the XInclude processing context
1957
 * @param tree  the top of the tree to process
1958
 * @returns 0 if no substitution were done, -1 if some processing failed
1959
 *    or the number of substitutions done.
1960
 */
1961
static int
1962
19.5k
xmlXIncludeDoProcessRoot(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1963
19.5k
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL))
1964
0
  return(-1);
1965
19.5k
    if (ctxt == NULL)
1966
0
  return(-1);
1967
1968
19.5k
    return(xmlXIncludeDoProcess(ctxt, tree));
1969
19.5k
}
1970
1971
/**
1972
 * @since 2.13.0
1973
 *
1974
 * @param ctxt  an XInclude processing context
1975
 * @returns the last error code.
1976
 */
1977
int
1978
39.1k
xmlXIncludeGetLastError(xmlXIncludeCtxt *ctxt) {
1979
39.1k
    if (ctxt == NULL)
1980
0
        return(XML_ERR_ARGUMENT);
1981
39.1k
    return(ctxt->errNo);
1982
39.1k
}
1983
1984
/**
1985
 * Register a callback function that will be called on errors and
1986
 * warnings. If handler is NULL, the error handler will be deactivated.
1987
 *
1988
 * @since 2.13.0
1989
 * @param ctxt  an XInclude processing context
1990
 * @param handler  error handler
1991
 * @param data  user data which will be passed to the handler
1992
 */
1993
void
1994
xmlXIncludeSetErrorHandler(xmlXIncludeCtxt *ctxt,
1995
0
                           xmlStructuredErrorFunc handler, void *data) {
1996
0
    if (ctxt == NULL)
1997
0
        return;
1998
0
    ctxt->errorHandler = handler;
1999
0
    ctxt->errorCtxt = data;
2000
0
}
2001
2002
/**
2003
 * Register a callback function that will be called to load included
2004
 * documents.
2005
 *
2006
 * @since 2.14.0
2007
 * @param ctxt  an XInclude processing context
2008
 * @param loader  resource loader
2009
 * @param data  user data which will be passed to the loader
2010
 */
2011
void
2012
xmlXIncludeSetResourceLoader(xmlXIncludeCtxt *ctxt,
2013
28.4k
                             xmlResourceLoader loader, void *data) {
2014
28.4k
    if (ctxt == NULL)
2015
8.85k
        return;
2016
19.5k
    ctxt->resourceLoader = loader;
2017
19.5k
    ctxt->resourceCtxt = data;
2018
19.5k
}
2019
2020
/**
2021
 * Set the flags used for further processing of XML resources.
2022
 *
2023
 * @param ctxt  an XInclude processing context
2024
 * @param flags  a set of xmlParserOption used for parsing XML includes
2025
 * @returns 0 in case of success and -1 in case of error.
2026
 */
2027
int
2028
28.4k
xmlXIncludeSetFlags(xmlXIncludeCtxt *ctxt, int flags) {
2029
28.4k
    if (ctxt == NULL)
2030
8.85k
        return(-1);
2031
19.5k
    ctxt->parseFlags = flags;
2032
19.5k
    return(0);
2033
28.4k
}
2034
2035
/**
2036
 * In streaming mode, XPointer expressions aren't allowed.
2037
 *
2038
 * @param ctxt  an XInclude processing context
2039
 * @param mode  whether streaming mode should be enabled
2040
 * @returns 0 in case of success and -1 in case of error.
2041
 */
2042
int
2043
0
xmlXIncludeSetStreamingMode(xmlXIncludeCtxt *ctxt, int mode) {
2044
0
    if (ctxt == NULL)
2045
0
        return(-1);
2046
0
    ctxt->isStream = !!mode;
2047
0
    return(0);
2048
0
}
2049
2050
/**
2051
 * Implement the XInclude substitution on the XML node `tree`
2052
 *
2053
 * @param tree  an XML node
2054
 * @param flags  a set of xmlParserOption used for parsing XML includes
2055
 * @param data  application data that will be passed to the parser context
2056
 *        in the _private field of the parser context(s)
2057
 * @returns 0 if no substitution were done, -1 if some processing failed
2058
 *    or the number of substitutions done.
2059
 */
2060
2061
int
2062
0
xmlXIncludeProcessTreeFlagsData(xmlNode *tree, int flags, void *data) {
2063
0
    xmlXIncludeCtxtPtr ctxt;
2064
0
    int ret = 0;
2065
2066
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL) ||
2067
0
        (tree->doc == NULL))
2068
0
        return(-1);
2069
2070
0
    ctxt = xmlXIncludeNewContext(tree->doc);
2071
0
    if (ctxt == NULL)
2072
0
        return(-1);
2073
0
    ctxt->_private = data;
2074
0
    xmlXIncludeSetFlags(ctxt, flags);
2075
0
    ret = xmlXIncludeDoProcessRoot(ctxt, tree);
2076
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2077
0
        ret = -1;
2078
2079
0
    xmlXIncludeFreeContext(ctxt);
2080
0
    return(ret);
2081
0
}
2082
2083
/**
2084
 * Implement the XInclude substitution on the XML document `doc`
2085
 *
2086
 * @param doc  an XML document
2087
 * @param flags  a set of xmlParserOption used for parsing XML includes
2088
 * @param data  application data that will be passed to the parser context
2089
 *        in the _private field of the parser context(s)
2090
 * @returns 0 if no substitution were done, -1 if some processing failed
2091
 *    or the number of substitutions done.
2092
 */
2093
int
2094
0
xmlXIncludeProcessFlagsData(xmlDoc *doc, int flags, void *data) {
2095
0
    xmlNodePtr tree;
2096
2097
0
    if (doc == NULL)
2098
0
  return(-1);
2099
0
    tree = xmlDocGetRootElement(doc);
2100
0
    if (tree == NULL)
2101
0
  return(-1);
2102
0
    return(xmlXIncludeProcessTreeFlagsData(tree, flags, data));
2103
0
}
2104
2105
/**
2106
 * Implement the XInclude substitution on the XML document `doc`
2107
 *
2108
 * @param doc  an XML document
2109
 * @param flags  a set of xmlParserOption used for parsing XML includes
2110
 * @returns 0 if no substitution were done, -1 if some processing failed
2111
 *    or the number of substitutions done.
2112
 */
2113
int
2114
0
xmlXIncludeProcessFlags(xmlDoc *doc, int flags) {
2115
0
    return xmlXIncludeProcessFlagsData(doc, flags, NULL);
2116
0
}
2117
2118
/**
2119
 * Implement the XInclude substitution on the XML document `doc`
2120
 *
2121
 * @param doc  an XML document
2122
 * @returns 0 if no substitution were done, -1 if some processing failed
2123
 *    or the number of substitutions done.
2124
 */
2125
int
2126
0
xmlXIncludeProcess(xmlDoc *doc) {
2127
0
    return(xmlXIncludeProcessFlags(doc, doc ? doc->parseFlags : 0));
2128
0
}
2129
2130
/**
2131
 * Implement the XInclude substitution for the given subtree
2132
 *
2133
 * @param tree  a node in an XML document
2134
 * @param flags  a set of xmlParserOption used for parsing XML includes
2135
 * @returns 0 if no substitution were done, -1 if some processing failed
2136
 *    or the number of substitutions done.
2137
 */
2138
int
2139
0
xmlXIncludeProcessTreeFlags(xmlNode *tree, int flags) {
2140
0
    xmlXIncludeCtxtPtr ctxt;
2141
0
    int ret = 0;
2142
2143
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL) ||
2144
0
        (tree->doc == NULL))
2145
0
  return(-1);
2146
0
    ctxt = xmlXIncludeNewContext(tree->doc);
2147
0
    if (ctxt == NULL)
2148
0
  return(-1);
2149
0
    xmlXIncludeSetFlags(ctxt, flags);
2150
0
    ret = xmlXIncludeDoProcessRoot(ctxt, tree);
2151
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2152
0
  ret = -1;
2153
2154
0
    xmlXIncludeFreeContext(ctxt);
2155
0
    return(ret);
2156
0
}
2157
2158
/**
2159
 * Implement the XInclude substitution for the given subtree
2160
 *
2161
 * @param tree  a node in an XML document
2162
 * @returns 0 if no substitution were done, -1 if some processing failed
2163
 *    or the number of substitutions done.
2164
 */
2165
int
2166
0
xmlXIncludeProcessTree(xmlNode *tree) {
2167
0
    return(xmlXIncludeProcessTreeFlags(tree, (tree && tree->doc) ? tree->doc->parseFlags : 0));
2168
0
}
2169
2170
/**
2171
 * Implement the XInclude substitution for the given subtree reusing
2172
 * the information and data coming from the given context.
2173
 *
2174
 * @param ctxt  an existing XInclude context
2175
 * @param node  a node in an XML document
2176
 * @returns 0 if no substitution were done, -1 if some processing failed
2177
 *    or the number of substitutions done.
2178
 */
2179
int
2180
28.4k
xmlXIncludeProcessNode(xmlXIncludeCtxt *ctxt, xmlNode *node) {
2181
28.4k
    int ret = 0;
2182
2183
28.4k
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL) ||
2184
19.6k
        (node->doc == NULL) || (ctxt == NULL))
2185
8.85k
  return(-1);
2186
19.5k
    ret = xmlXIncludeDoProcessRoot(ctxt, node);
2187
19.5k
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2188
6.94k
  ret = -1;
2189
19.5k
    return(ret);
2190
28.4k
}
2191
2192
#else /* !LIBXML_XINCLUDE_ENABLED */
2193
#endif