Coverage Report

Created: 2025-08-11 06:23

/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
28.5k
#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.64k
{
141
1.64k
    ctxt->errNo = XML_ERR_NO_MEMORY;
142
1.64k
    ctxt->fatalErr = 1;
143
1.64k
    ctxt->nbErrors++;
144
145
1.64k
    xmlRaiseMemoryError(ctxt->errorHandler, NULL, ctxt->errorCtxt,
146
1.64k
                        XML_FROM_XINCLUDE, NULL);
147
1.64k
}
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
34.3k
{
162
34.3k
    xmlStructuredErrorFunc schannel = NULL;
163
34.3k
    xmlGenericErrorFunc channel = NULL;
164
34.3k
    void *data = NULL;
165
34.3k
    int res;
166
167
34.3k
    if (error == XML_ERR_NO_MEMORY) {
168
88
        xmlXIncludeErrMemory(ctxt);
169
88
        return;
170
88
    }
171
172
34.2k
    if (ctxt->fatalErr != 0)
173
1.87k
        return;
174
32.3k
    ctxt->nbErrors++;
175
176
32.3k
    schannel = ctxt->errorHandler;
177
32.3k
    data = ctxt->errorCtxt;
178
179
32.3k
    if (schannel == NULL) {
180
32.3k
        channel = xmlGenericError;
181
32.3k
        data = xmlGenericErrorContext;
182
32.3k
    }
183
184
32.3k
    res = xmlRaiseError(schannel, channel, data, ctxt, node,
185
32.3k
                        XML_FROM_XINCLUDE, error, XML_ERR_ERROR,
186
32.3k
                        NULL, 0, (const char *) extra, NULL, NULL, 0, 0,
187
32.3k
                        msg, (const char *) extra);
188
32.3k
    if (res < 0) {
189
87
        ctxt->errNo = XML_ERR_NO_MEMORY;
190
87
        ctxt->fatalErr = 1;
191
32.2k
    } else {
192
32.2k
        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
32.2k
        if (xmlIsCatastrophicError(XML_ERR_FATAL, error))
200
3
            ctxt->fatalErr = 1;
201
32.2k
    }
202
32.3k
}
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
73.0k
                   const xmlChar *name) {
215
73.0k
    xmlChar *ret;
216
217
73.0k
    if (xmlNodeGetAttrValue(cur, name, XINCLUDE_NS, &ret) < 0)
218
1
        xmlXIncludeErrMemory(ctxt);
219
73.0k
    if (ret != NULL)
220
34
        return(ret);
221
222
73.0k
    if (ctxt->legacy != 0) {
223
64.7k
        if (xmlNodeGetAttrValue(cur, name, XINCLUDE_OLD_NS, &ret) < 0)
224
1
            xmlXIncludeErrMemory(ctxt);
225
64.7k
        if (ret != NULL)
226
88
            return(ret);
227
64.7k
    }
228
229
72.9k
    if (xmlNodeGetAttrValue(cur, name, NULL, &ret) < 0)
230
4
        xmlXIncludeErrMemory(ctxt);
231
72.9k
    return(ret);
232
73.0k
}
233
/**
234
 * Free an XInclude reference
235
 *
236
 * @param ref  the XInclude reference
237
 */
238
static void
239
47.4k
xmlXIncludeFreeRef(xmlXIncludeRefPtr ref) {
240
47.4k
    if (ref == NULL)
241
24.8k
  return;
242
22.5k
    if (ref->URI != NULL)
243
22.5k
  xmlFree(ref->URI);
244
22.5k
    if (ref->fragment != NULL)
245
17.1k
  xmlFree(ref->fragment);
246
22.5k
    if (ref->base != NULL)
247
12.3k
  xmlFree(ref->base);
248
22.5k
    xmlFree(ref);
249
22.5k
}
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.9k
xmlXIncludeNewContext(xmlDoc *doc) {
259
28.9k
    xmlXIncludeCtxtPtr ret;
260
261
28.9k
    if (doc == NULL)
262
9.14k
  return(NULL);
263
19.8k
    ret = (xmlXIncludeCtxtPtr) xmlMalloc(sizeof(xmlXIncludeCtxt));
264
19.8k
    if (ret == NULL)
265
55
  return(NULL);
266
19.7k
    memset(ret, 0, sizeof(xmlXIncludeCtxt));
267
19.7k
    ret->doc = doc;
268
19.7k
    ret->incNr = 0;
269
19.7k
    ret->incMax = 0;
270
19.7k
    ret->incTab = NULL;
271
19.7k
    ret->nbErrors = 0;
272
19.7k
    return(ret);
273
19.8k
}
274
275
/**
276
 * Free an XInclude context
277
 *
278
 * @param ctxt  the XInclude context
279
 */
280
void
281
28.9k
xmlXIncludeFreeContext(xmlXIncludeCtxt *ctxt) {
282
28.9k
    int i;
283
284
28.9k
    if (ctxt == NULL)
285
9.20k
  return;
286
19.7k
    if (ctxt->urlTab != NULL) {
287
6.82k
  for (i = 0; i < ctxt->urlNr; i++) {
288
4.43k
      xmlFreeDoc(ctxt->urlTab[i].doc);
289
4.43k
      xmlFree(ctxt->urlTab[i].url);
290
4.43k
  }
291
2.39k
  xmlFree(ctxt->urlTab);
292
2.39k
    }
293
42.0k
    for (i = 0;i < ctxt->incNr;i++) {
294
22.2k
  if (ctxt->incTab[i] != NULL)
295
22.2k
      xmlXIncludeFreeRef(ctxt->incTab[i]);
296
22.2k
    }
297
19.7k
    if (ctxt->incTab != NULL)
298
9.96k
  xmlFree(ctxt->incTab);
299
19.7k
    if (ctxt->txtTab != NULL) {
300
97
  for (i = 0;i < ctxt->txtNr;i++) {
301
50
      xmlFree(ctxt->txtTab[i].text);
302
50
      xmlFree(ctxt->txtTab[i].url);
303
50
  }
304
47
  xmlFree(ctxt->txtTab);
305
47
    }
306
19.7k
#ifdef LIBXML_XPTR_ENABLED
307
19.7k
    if (ctxt->xpctxt != NULL)
308
7.52k
  xmlXPathFreeContext(ctxt->xpctxt);
309
19.7k
#endif
310
19.7k
    xmlFree(ctxt);
311
19.7k
}
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.44k
xmlXIncludeParseFile(xmlXIncludeCtxtPtr ctxt, const char *URL) {
321
4.44k
    xmlDocPtr ret = NULL;
322
4.44k
    xmlParserCtxtPtr pctxt;
323
4.44k
    xmlParserInputPtr inputStream;
324
325
4.44k
    xmlInitParser();
326
327
4.44k
    pctxt = xmlNewParserCtxt();
328
4.44k
    if (pctxt == NULL) {
329
21
  xmlXIncludeErrMemory(ctxt);
330
21
  return(NULL);
331
21
    }
332
4.42k
    if (ctxt->errorHandler != NULL)
333
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
334
4.42k
    if (ctxt->resourceLoader != NULL)
335
4.42k
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
336
4.42k
                                 ctxt->resourceCtxt);
337
338
    /*
339
     * pass in the application data to the parser context.
340
     */
341
4.42k
    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
4.42k
    if ((ctxt->doc != NULL) && (ctxt->doc->dict != NULL)) {
348
2.99k
       if (pctxt->dict != NULL)
349
2.99k
            xmlDictFree(pctxt->dict);
350
2.99k
  pctxt->dict = ctxt->doc->dict;
351
2.99k
  xmlDictReference(pctxt->dict);
352
2.99k
    }
353
354
    /*
355
     * We set DTDLOAD to make sure that ID attributes declared in
356
     * external DTDs are detected.
357
     */
358
4.42k
    xmlCtxtUseOptions(pctxt, ctxt->parseFlags | XML_PARSE_DTDLOAD);
359
360
4.42k
    inputStream = xmlLoadResource(pctxt, URL, NULL, XML_RESOURCE_XINCLUDE);
361
4.42k
    if (inputStream == NULL)
362
2.79k
        goto error;
363
364
1.62k
    if (xmlCtxtPushInput(pctxt, inputStream) < 0) {
365
1
        xmlFreeInputStream(inputStream);
366
1
        goto error;
367
1
    }
368
369
1.62k
    xmlParseDocument(pctxt);
370
371
1.62k
    if (pctxt->wellFormed) {
372
145
        ret = pctxt->myDoc;
373
145
    }
374
1.48k
    else {
375
1.48k
        ret = NULL;
376
1.48k
  if (pctxt->myDoc != NULL)
377
1.47k
      xmlFreeDoc(pctxt->myDoc);
378
1.48k
        pctxt->myDoc = NULL;
379
1.48k
    }
380
381
4.42k
error:
382
4.42k
    if (xmlCtxtIsCatastrophicError(pctxt))
383
89
        xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "parser error", NULL);
384
4.42k
    xmlFreeParserCtxt(pctxt);
385
386
4.42k
    return(ret);
387
1.62k
}
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
24.8k
xmlXIncludeAddNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur) {
397
24.8k
    xmlXIncludeRefPtr ref = NULL;
398
24.8k
    xmlXIncludeRefPtr ret = NULL;
399
24.8k
    xmlURIPtr uri = NULL;
400
24.8k
    xmlChar *href = NULL;
401
24.8k
    xmlChar *parse = NULL;
402
24.8k
    xmlChar *fragment = NULL;
403
24.8k
    xmlChar *base = NULL;
404
24.8k
    xmlChar *tmp;
405
24.8k
    int xml = 1;
406
24.8k
    int local = 0;
407
24.8k
    int res;
408
409
24.8k
    if (ctxt == NULL)
410
0
  return(NULL);
411
24.8k
    if (cur == NULL)
412
0
  return(NULL);
413
414
    /*
415
     * read the attributes
416
     */
417
418
24.8k
    fragment = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE_XPOINTER);
419
420
24.8k
    href = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_HREF);
421
24.8k
    if (href == NULL) {
422
18.1k
        if (fragment == NULL) {
423
1.91k
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_HREF,
424
1.91k
                     "href or xpointer must be present\n", parse);
425
1.91k
      goto error;
426
1.91k
        }
427
428
16.2k
  href = xmlStrdup(BAD_CAST ""); /* @@@@ href is now optional */
429
16.2k
  if (href == NULL) {
430
5
            xmlXIncludeErrMemory(ctxt);
431
5
      goto error;
432
5
        }
433
16.2k
    } else if (xmlStrlen(href) > XML_MAX_URI_LENGTH) {
434
19
        xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI, "URI too long\n",
435
19
                       NULL);
436
19
        goto error;
437
19
    }
438
439
22.9k
    parse = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE);
440
22.9k
    if (parse != NULL) {
441
762
  if (xmlStrEqual(parse, XINCLUDE_PARSE_XML))
442
282
      xml = 1;
443
480
  else if (xmlStrEqual(parse, XINCLUDE_PARSE_TEXT))
444
445
      xml = 0;
445
35
  else {
446
35
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_PARSE_VALUE,
447
35
                     "invalid value %s for 'parse'\n", parse);
448
35
      goto error;
449
35
  }
450
762
    }
451
452
    /*
453
     * Check the URL and remove any fragment identifier
454
     */
455
22.9k
    res = xmlParseURISafe((const char *)href, &uri);
456
22.9k
    if (uri == NULL) {
457
177
        if (res < 0)
458
3
            xmlXIncludeErrMemory(ctxt);
459
174
        else
460
174
            xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI,
461
174
                           "invalid value href %s\n", href);
462
177
        goto error;
463
177
    }
464
465
22.7k
    if (uri->fragment != NULL) {
466
386
        if (ctxt->legacy != 0) {
467
352
      if (fragment == NULL) {
468
305
    fragment = (xmlChar *) uri->fragment;
469
305
      } else {
470
47
    xmlFree(uri->fragment);
471
47
      }
472
352
  } else {
473
34
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_FRAGMENT_ID,
474
34
       "Invalid fragment identifier in URI %s use the xpointer attribute\n",
475
34
                           href);
476
34
      goto error;
477
34
  }
478
352
  uri->fragment = NULL;
479
352
    }
480
22.6k
    tmp = xmlSaveUri(uri);
481
22.6k
    if (tmp == NULL) {
482
3
  xmlXIncludeErrMemory(ctxt);
483
3
  goto error;
484
3
    }
485
22.6k
    xmlFree(href);
486
22.6k
    href = tmp;
487
488
    /*
489
     * Resolve URI
490
     */
491
492
22.6k
    if (xmlNodeGetBaseSafe(ctxt->doc, cur, &base) < 0) {
493
9
        xmlXIncludeErrMemory(ctxt);
494
9
        goto error;
495
9
    }
496
497
22.6k
    if (href[0] != 0) {
498
6.17k
        if (xmlBuildURISafe(href, base, &tmp) < 0) {
499
10
            xmlXIncludeErrMemory(ctxt);
500
10
            goto error;
501
10
        }
502
6.16k
        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
6.12k
        xmlFree(href);
508
6.12k
        href = tmp;
509
510
6.12k
        if (xmlStrEqual(href, ctxt->doc->URL))
511
69
            local = 1;
512
16.5k
    } else {
513
16.5k
        local = 1;
514
16.5k
    }
515
516
    /*
517
     * If local and xml then we need a fragment
518
     */
519
22.6k
    if ((local == 1) && (xml == 1) &&
520
22.6k
        ((fragment == NULL) || (fragment[0] == 0))) {
521
104
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_RECURSION,
522
104
                 "detected a local recursion with no xpointer in %s\n",
523
104
           href);
524
104
  goto error;
525
104
    }
526
527
22.5k
    ref = (xmlXIncludeRefPtr) xmlMalloc(sizeof(xmlXIncludeRef));
528
22.5k
    if (ref == NULL) {
529
1
        xmlXIncludeErrMemory(ctxt);
530
1
        goto error;
531
1
    }
532
22.5k
    memset(ref, 0, sizeof(xmlXIncludeRef));
533
534
22.5k
    ref->elem = cur;
535
22.5k
    ref->xml = xml;
536
22.5k
    ref->URI = href;
537
22.5k
    href = NULL;
538
22.5k
    ref->fragment = fragment;
539
22.5k
    fragment = NULL;
540
541
    /*
542
     * xml:base fixup
543
     */
544
22.5k
    if (((ctxt->parseFlags & XML_PARSE_NOBASEFIX) == 0) &&
545
22.5k
        (cur->doc != NULL) &&
546
22.5k
        ((cur->doc->parseFlags & XML_PARSE_NOBASEFIX) == 0)) {
547
12.3k
        if (base != NULL) {
548
11.8k
            ref->base = base;
549
11.8k
            base = NULL;
550
11.8k
        } else {
551
475
            ref->base = xmlStrdup(BAD_CAST "");
552
475
            if (ref->base == NULL) {
553
1
          xmlXIncludeErrMemory(ctxt);
554
1
                goto error;
555
1
            }
556
475
        }
557
12.3k
    }
558
559
22.5k
    if (ctxt->incNr >= ctxt->incMax) {
560
18.2k
        xmlXIncludeRefPtr *table;
561
18.2k
        int newSize;
562
563
18.2k
        newSize = xmlGrowCapacity(ctxt->incMax, sizeof(table[0]),
564
18.2k
                                  4, XML_MAX_ITEMS);
565
18.2k
        if (newSize < 0) {
566
0
      xmlXIncludeErrMemory(ctxt);
567
0
      goto error;
568
0
  }
569
18.2k
        table = xmlRealloc(ctxt->incTab, newSize * sizeof(table[0]));
570
18.2k
        if (table == NULL) {
571
3
      xmlXIncludeErrMemory(ctxt);
572
3
      goto error;
573
3
  }
574
18.2k
        ctxt->incTab = table;
575
18.2k
        ctxt->incMax = newSize;
576
18.2k
    }
577
22.5k
    ctxt->incTab[ctxt->incNr++] = ref;
578
579
22.5k
    ret = ref;
580
22.5k
    ref = NULL;
581
582
24.8k
error:
583
24.8k
    xmlXIncludeFreeRef(ref);
584
24.8k
    xmlFreeURI(uri);
585
24.8k
    xmlFree(href);
586
24.8k
    xmlFree(parse);
587
24.8k
    xmlFree(fragment);
588
24.8k
    xmlFree(base);
589
24.8k
    return(ret);
590
22.5k
}
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
145
xmlXIncludeRecurseDoc(xmlXIncludeCtxtPtr ctxt, xmlDocPtr doc) {
600
145
    xmlDocPtr oldDoc;
601
145
    xmlXIncludeRefPtr *oldIncTab;
602
145
    int oldIncMax, oldIncNr, oldIsStream;
603
145
    int i;
604
605
145
    oldDoc = ctxt->doc;
606
145
    oldIncMax = ctxt->incMax;
607
145
    oldIncNr = ctxt->incNr;
608
145
    oldIncTab = ctxt->incTab;
609
145
    oldIsStream = ctxt->isStream;
610
145
    ctxt->doc = doc;
611
145
    ctxt->incMax = 0;
612
145
    ctxt->incNr = 0;
613
145
    ctxt->incTab = NULL;
614
145
    ctxt->isStream = 0;
615
616
145
    xmlXIncludeDoProcess(ctxt, xmlDocGetRootElement(doc));
617
618
145
    if (ctxt->incTab != NULL) {
619
359
        for (i = 0; i < ctxt->incNr; i++)
620
264
            xmlXIncludeFreeRef(ctxt->incTab[i]);
621
95
        xmlFree(ctxt->incTab);
622
95
    }
623
624
145
    ctxt->doc = oldDoc;
625
145
    ctxt->incMax = oldIncMax;
626
145
    ctxt->incNr = oldIncNr;
627
145
    ctxt->incTab = oldIncTab;
628
145
    ctxt->isStream = oldIsStream;
629
145
}
630
631
/************************************************************************
632
 *                  *
633
 *      Node copy with specific semantic    *
634
 *                  *
635
 ************************************************************************/
636
637
static void
638
xmlXIncludeBaseFixup(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur, xmlNodePtr copy,
639
80.4k
                     const xmlChar *targetBase) {
640
80.4k
    xmlChar *base = NULL;
641
80.4k
    xmlChar *relBase = NULL;
642
80.4k
    xmlNs ns;
643
80.4k
    int res;
644
645
80.4k
    if (cur->type != XML_ELEMENT_NODE)
646
16.0k
        return;
647
648
64.3k
    if (xmlNodeGetBaseSafe(cur->doc, cur, &base) < 0)
649
87
        xmlXIncludeErrMemory(ctxt);
650
651
64.3k
    if ((base != NULL) && !xmlStrEqual(base, targetBase)) {
652
21.6k
        if ((xmlStrlen(base) > XML_MAX_URI_LENGTH) ||
653
21.6k
            (xmlStrlen(targetBase) > XML_MAX_URI_LENGTH)) {
654
1.00k
            relBase = xmlStrdup(base);
655
1.00k
            if (relBase == NULL) {
656
1
                xmlXIncludeErrMemory(ctxt);
657
1
                goto done;
658
1
            }
659
20.6k
        } else if (xmlBuildRelativeURISafe(base, targetBase, &relBase) < 0) {
660
22
            xmlXIncludeErrMemory(ctxt);
661
22
            goto done;
662
22
        }
663
21.5k
        if (relBase == NULL) {
664
2.75k
            xmlXIncludeErr(ctxt, cur,
665
2.75k
                    XML_XINCLUDE_HREF_URI,
666
2.75k
                    "Building relative URI failed: %s\n",
667
2.75k
                    base);
668
2.75k
            goto done;
669
2.75k
        }
670
671
        /*
672
         * If the new base doesn't contain a slash, it can be omitted.
673
         */
674
18.8k
        if (xmlStrchr(relBase, '/') != NULL) {
675
16.0k
            res = xmlNodeSetBase(copy, relBase);
676
16.0k
            if (res < 0)
677
15
                xmlXIncludeErrMemory(ctxt);
678
16.0k
            goto done;
679
16.0k
        }
680
18.8k
    }
681
682
    /*
683
     * Delete existing xml:base if bases are equal
684
     */
685
45.5k
    memset(&ns, 0, sizeof(ns));
686
45.5k
    ns.href = XML_XML_NAMESPACE;
687
45.5k
    xmlUnsetNsProp(copy, &ns, BAD_CAST "base");
688
689
64.3k
done:
690
64.3k
    xmlFree(base);
691
64.3k
    xmlFree(relBase);
692
64.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
12.6k
                    int copyChildren, const xmlChar *targetBase) {
706
12.6k
    xmlNodePtr result = NULL;
707
12.6k
    xmlNodePtr insertParent = NULL;
708
12.6k
    xmlNodePtr insertLast = NULL;
709
12.6k
    xmlNodePtr cur;
710
12.6k
    xmlNodePtr item;
711
12.6k
    int depth = 0;
712
713
12.6k
    if (copyChildren) {
714
2.95k
        cur = elem->children;
715
2.95k
        if (cur == NULL)
716
0
            return(NULL);
717
9.64k
    } else {
718
9.64k
        cur = elem;
719
9.64k
    }
720
721
210k
    while (1) {
722
210k
        xmlNodePtr copy = NULL;
723
210k
        int recurse = 0;
724
725
210k
        if ((cur->type == XML_DOCUMENT_NODE) ||
726
210k
            (cur->type == XML_DTD_NODE)) {
727
0
            ;
728
210k
        } else if ((cur->type == XML_ELEMENT_NODE) &&
729
210k
                   (cur->ns != NULL) &&
730
210k
                   (xmlStrEqual(cur->name, XINCLUDE_NODE)) &&
731
210k
                   ((xmlStrEqual(cur->ns->href, XINCLUDE_NS)) ||
732
4.67k
                    (xmlStrEqual(cur->ns->href, XINCLUDE_OLD_NS)))) {
733
4.15k
            xmlXIncludeRefPtr ref = xmlXIncludeExpandNode(ctxt, cur);
734
735
4.15k
            if (ref == NULL)
736
1.18k
                goto error;
737
            /*
738
             * TODO: Insert XML_XINCLUDE_START and XML_XINCLUDE_END nodes
739
             */
740
44.1k
            for (item = ref->inc; item != NULL; item = item->next) {
741
41.1k
                copy = xmlStaticCopyNode(item, ctxt->doc, insertParent, 1);
742
41.1k
                if (copy == NULL) {
743
6
                    xmlXIncludeErrMemory(ctxt);
744
6
                    goto error;
745
6
                }
746
747
41.1k
                if (result == NULL)
748
70
                    result = copy;
749
41.1k
                if (insertLast != NULL) {
750
40.9k
                    insertLast->next = copy;
751
40.9k
                    copy->prev = insertLast;
752
40.9k
                } else if (insertParent != NULL) {
753
109
                    insertParent->children = copy;
754
109
                }
755
41.1k
                insertLast = copy;
756
757
41.1k
                if ((depth == 0) && (targetBase != NULL))
758
27.9k
                    xmlXIncludeBaseFixup(ctxt, item, copy, targetBase);
759
41.1k
            }
760
206k
        } else {
761
206k
            copy = xmlStaticCopyNode(cur, ctxt->doc, insertParent, 2);
762
206k
            if (copy == NULL) {
763
31
                xmlXIncludeErrMemory(ctxt);
764
31
                goto error;
765
31
            }
766
767
206k
            if (result == NULL)
768
12.4k
                result = copy;
769
206k
            if (insertLast != NULL) {
770
139k
                insertLast->next = copy;
771
139k
                copy->prev = insertLast;
772
139k
            } else if (insertParent != NULL) {
773
54.3k
                insertParent->children = copy;
774
54.3k
            }
775
206k
            insertLast = copy;
776
777
206k
            if ((depth == 0) && (targetBase != NULL))
778
52.3k
                xmlXIncludeBaseFixup(ctxt, cur, copy, targetBase);
779
780
206k
            recurse = (cur->type != XML_ENTITY_REF_NODE) &&
781
206k
                      (cur->children != NULL);
782
206k
        }
783
784
209k
        if (recurse) {
785
54.7k
            cur = cur->children;
786
54.7k
            insertParent = insertLast;
787
54.7k
            insertLast = NULL;
788
54.7k
            depth += 1;
789
54.7k
            continue;
790
54.7k
        }
791
792
154k
        if (cur == elem)
793
6.21k
            return(result);
794
795
187k
        while (cur->next == NULL) {
796
43.8k
            if (insertParent != NULL)
797
41.0k
                insertParent->last = insertLast;
798
43.8k
            cur = cur->parent;
799
43.8k
            if (cur == elem)
800
5.15k
                return(result);
801
38.7k
            insertLast = insertParent;
802
38.7k
            insertParent = insertParent->parent;
803
38.7k
            depth -= 1;
804
38.7k
        }
805
806
143k
        cur = cur->next;
807
143k
    }
808
809
1.22k
error:
810
1.22k
    xmlFreeNodeList(result);
811
1.22k
    return(NULL);
812
12.6k
}
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.89k
                        const xmlChar *targetBase) {
828
1.89k
    xmlNodePtr list = NULL, last = NULL, copy;
829
1.89k
    int i;
830
831
1.89k
    if ((ctxt == NULL) || (obj == NULL))
832
0
  return(NULL);
833
1.89k
    switch (obj->type) {
834
1.89k
        case XPATH_NODESET: {
835
1.89k
      xmlNodeSetPtr set = obj->nodesetval;
836
1.89k
      if (set == NULL)
837
0
    break;
838
10.3k
      for (i = 0;i < set->nodeNr;i++) {
839
9.64k
                xmlNodePtr node;
840
841
9.64k
    if (set->nodeTab[i] == NULL)
842
0
        continue;
843
9.64k
    switch (set->nodeTab[i]->type) {
844
67
        case XML_DOCUMENT_NODE:
845
67
        case XML_HTML_DOCUMENT_NODE:
846
67
                        node = xmlDocGetRootElement(
847
67
                                (xmlDocPtr) set->nodeTab[i]);
848
67
                        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
67
                        break;
855
1.47k
                    case XML_TEXT_NODE:
856
1.69k
        case XML_CDATA_SECTION_NODE:
857
9.14k
        case XML_ELEMENT_NODE:
858
9.35k
        case XML_PI_NODE:
859
9.57k
        case XML_COMMENT_NODE:
860
9.57k
                        node = set->nodeTab[i];
861
9.57k
      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
9.64k
    }
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
9.64k
    copy = xmlXIncludeCopyNode(ctxt, node, 0, targetBase);
876
9.64k
                if (copy == NULL) {
877
1.20k
                    xmlFreeNodeList(list);
878
1.20k
                    return(NULL);
879
1.20k
                }
880
8.44k
    if (last == NULL) {
881
684
                    list = copy;
882
7.75k
                } else {
883
8.17k
                    while (last->next != NULL)
884
416
                        last = last->next;
885
7.75k
                    copy->prev = last;
886
7.75k
                    last->next = copy;
887
7.75k
    }
888
8.44k
                last = copy;
889
8.44k
      }
890
691
      break;
891
1.89k
  }
892
691
  default:
893
0
      break;
894
1.89k
    }
895
691
    return(list);
896
1.89k
}
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
32
                 const xmlChar *name ATTRIBUTE_UNUSED) {
922
32
    xmlEntityPtr ent = (xmlEntityPtr) payload;
923
32
    xmlXIncludeMergeDataPtr data = (xmlXIncludeMergeDataPtr) vdata;
924
32
    xmlEntityPtr ret, prev;
925
32
    xmlDocPtr doc;
926
32
    xmlXIncludeCtxtPtr ctxt;
927
928
32
    if ((ent == NULL) || (data == NULL))
929
0
  return;
930
32
    ctxt = data->ctxt;
931
32
    doc = data->doc;
932
32
    if ((ctxt == NULL) || (doc == NULL))
933
0
  return;
934
32
    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
20
        case XML_INTERNAL_GENERAL_ENTITY:
940
29
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
941
32
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
942
32
      break;
943
32
    }
944
32
    prev = xmlGetDocEntity(doc, ent->name);
945
32
    if (prev == NULL) {
946
21
        ret = xmlAddDocEntity(doc, ent->name, ent->etype, ent->ExternalID,
947
21
                              ent->SystemID, ent->content);
948
21
        if (ret == NULL) {
949
1
            xmlXIncludeErrMemory(ctxt);
950
1
            return;
951
1
        }
952
20
  if (ent->URI != NULL) {
953
5
      ret->URI = xmlStrdup(ent->URI);
954
5
            if (ret->URI == 0)
955
1
                xmlXIncludeErrMemory(ctxt);
956
5
        }
957
20
    } else {
958
11
        if (ent->etype != prev->etype)
959
2
            goto error;
960
961
9
        if ((ent->SystemID != NULL) && (prev->SystemID != NULL)) {
962
2
            if (!xmlStrEqual(ent->SystemID, prev->SystemID))
963
1
                goto error;
964
7
        } else if ((ent->ExternalID != NULL) &&
965
7
                   (prev->ExternalID != NULL)) {
966
2
            if (!xmlStrEqual(ent->ExternalID, prev->ExternalID))
967
1
                goto error;
968
5
        } else if ((ent->content != NULL) && (prev->content != NULL)) {
969
4
            if (!xmlStrEqual(ent->content, prev->content))
970
1
                goto error;
971
4
        } else {
972
1
            goto error;
973
1
        }
974
9
    }
975
25
    return;
976
25
error:
977
6
    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
2
        case XML_INTERNAL_GENERAL_ENTITY:
982
5
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
983
5
      return;
984
1
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
985
1
      break;
986
6
    }
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
145
                   xmlDocPtr from) {
1003
145
    xmlNodePtr cur;
1004
145
    xmlDtdPtr target, source;
1005
1006
145
    if (ctxt == NULL)
1007
0
  return(-1);
1008
1009
145
    if ((from == NULL) || (from->intSubset == NULL))
1010
76
  return(0);
1011
1012
69
    target = doc->intSubset;
1013
69
    if (target == NULL) {
1014
32
  cur = xmlDocGetRootElement(doc);
1015
32
  if (cur == NULL)
1016
0
      return(-1);
1017
32
        target = xmlCreateIntSubset(doc, cur->name, NULL, NULL);
1018
32
  if (target == NULL) {
1019
1
            xmlXIncludeErrMemory(ctxt);
1020
1
      return(-1);
1021
1
        }
1022
32
    }
1023
1024
68
    source = from->intSubset;
1025
68
    if ((source != NULL) && (source->entities != NULL)) {
1026
21
  xmlXIncludeMergeData data;
1027
1028
21
  data.ctxt = ctxt;
1029
21
  data.doc = doc;
1030
1031
21
  xmlHashScan((xmlHashTablePtr) source->entities,
1032
21
        xmlXIncludeMergeEntity, &data);
1033
21
    }
1034
68
    source = from->extSubset;
1035
68
    if ((source != NULL) && (source->entities != NULL)) {
1036
1
  xmlXIncludeMergeData data;
1037
1038
1
  data.ctxt = ctxt;
1039
1
  data.doc = doc;
1040
1041
  /*
1042
   * don't duplicate existing stuff when external subsets are the same
1043
   */
1044
1
  if ((!xmlStrEqual(target->ExternalID, source->ExternalID)) &&
1045
1
      (!xmlStrEqual(target->SystemID, source->SystemID))) {
1046
0
      xmlHashScan((xmlHashTablePtr) source->entities,
1047
0
      xmlXIncludeMergeEntity, &data);
1048
0
  }
1049
1
    }
1050
68
    return(0);
1051
69
}
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
22.0k
xmlXIncludeLoadDoc(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1062
22.0k
    xmlXIncludeDocPtr cache;
1063
22.0k
    xmlDocPtr doc;
1064
22.0k
    const xmlChar *url = ref->URI;
1065
22.0k
    const xmlChar *fragment = ref->fragment;
1066
22.0k
    int i = 0;
1067
22.0k
    int ret = -1;
1068
22.0k
    int cacheNr;
1069
22.0k
#ifdef LIBXML_XPTR_ENABLED
1070
22.0k
    int saveFlags;
1071
22.0k
#endif
1072
1073
    /*
1074
     * Handling of references to the local document are done
1075
     * directly through ctxt->doc.
1076
     */
1077
22.0k
    if ((url[0] == 0) || (url[0] == '#') ||
1078
22.0k
  ((ctxt->doc != NULL) && (xmlStrEqual(url, ctxt->doc->URL)))) {
1079
16.5k
  doc = ctxt->doc;
1080
16.5k
        goto loaded;
1081
16.5k
    }
1082
1083
    /*
1084
     * Prevent reloading the document twice.
1085
     */
1086
11.1k
    for (i = 0; i < ctxt->urlNr; i++) {
1087
6.71k
  if (xmlStrEqual(url, ctxt->urlTab[i].url)) {
1088
1.06k
            if (ctxt->urlTab[i].expanding) {
1089
112
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_RECURSION,
1090
112
                               "inclusion loop detected\n", NULL);
1091
112
                goto error;
1092
112
            }
1093
951
      doc = ctxt->urlTab[i].doc;
1094
951
            if (doc == NULL)
1095
862
                goto error;
1096
89
      goto loaded;
1097
951
  }
1098
6.71k
    }
1099
1100
    /*
1101
     * Load it.
1102
     */
1103
4.44k
#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.44k
    saveFlags = ctxt->parseFlags;
1110
4.44k
    if (fragment != NULL) { /* if this is an XPointer eval */
1111
546
  ctxt->parseFlags |= XML_PARSE_NOENT;
1112
546
    }
1113
4.44k
#endif
1114
1115
4.44k
    doc = xmlXIncludeParseFile(ctxt, (const char *)url);
1116
4.44k
#ifdef LIBXML_XPTR_ENABLED
1117
4.44k
    ctxt->parseFlags = saveFlags;
1118
4.44k
#endif
1119
1120
    /* Also cache NULL docs */
1121
4.44k
    if (ctxt->urlNr >= ctxt->urlMax) {
1122
4.21k
        xmlXIncludeDoc *tmp;
1123
4.21k
        int newSize;
1124
1125
4.21k
        newSize = xmlGrowCapacity(ctxt->urlMax, sizeof(tmp[0]),
1126
4.21k
                                  8, XML_MAX_ITEMS);
1127
4.21k
        if (newSize < 0) {
1128
0
            xmlXIncludeErrMemory(ctxt);
1129
0
            xmlFreeDoc(doc);
1130
0
            goto error;
1131
0
        }
1132
4.21k
        tmp = xmlRealloc(ctxt->urlTab, newSize * sizeof(tmp[0]));
1133
4.21k
        if (tmp == NULL) {
1134
11
            xmlXIncludeErrMemory(ctxt);
1135
11
            xmlFreeDoc(doc);
1136
11
            goto error;
1137
11
        }
1138
4.19k
        ctxt->urlMax = newSize;
1139
4.19k
        ctxt->urlTab = tmp;
1140
4.19k
    }
1141
4.43k
    cache = &ctxt->urlTab[ctxt->urlNr];
1142
4.43k
    cache->doc = doc;
1143
4.43k
    cache->url = xmlStrdup(url);
1144
4.43k
    if (cache->url == NULL) {
1145
6
        xmlXIncludeErrMemory(ctxt);
1146
6
        xmlFreeDoc(doc);
1147
6
        goto error;
1148
6
    }
1149
4.43k
    cache->expanding = 0;
1150
4.43k
    cacheNr = ctxt->urlNr++;
1151
1152
4.43k
    if (doc == NULL)
1153
4.28k
        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
145
    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
145
    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
145
    cache->expanding = 1;
1182
145
    xmlXIncludeRecurseDoc(ctxt, doc);
1183
    /* urlTab might be reallocated. */
1184
145
    cache = &ctxt->urlTab[cacheNr];
1185
145
    cache->expanding = 0;
1186
1187
16.8k
loaded:
1188
16.8k
    if (fragment == NULL) {
1189
321
        xmlNodePtr root;
1190
1191
321
        root = xmlDocGetRootElement(doc);
1192
321
        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
321
        ref->inc = xmlDocCopyNode(root, ctxt->doc, 1);
1199
321
        if (ref->inc == NULL) {
1200
2
            xmlXIncludeErrMemory(ctxt);
1201
2
            goto error;
1202
2
        }
1203
1204
319
        if (ref->base != NULL)
1205
196
            xmlXIncludeBaseFixup(ctxt, root, ref->inc, ref->base);
1206
319
    }
1207
16.4k
#ifdef LIBXML_XPTR_ENABLED
1208
16.4k
    else {
1209
  /*
1210
   * Computes the XPointer expression and make a copy used
1211
   * as the replacement copy.
1212
   */
1213
16.4k
  xmlXPathObjectPtr xptr;
1214
16.4k
  xmlNodeSetPtr set;
1215
1216
16.4k
        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
16.4k
        if (ctxt->xpctxt == NULL) {
1224
7.53k
            ctxt->xpctxt = xmlXPathNewContext(doc);
1225
7.53k
            if (ctxt->xpctxt == NULL) {
1226
2
                xmlXIncludeErrMemory(ctxt);
1227
2
                goto error;
1228
2
            }
1229
7.52k
            if (ctxt->errorHandler != NULL)
1230
0
                xmlXPathSetErrorHandler(ctxt->xpctxt, ctxt->errorHandler,
1231
0
                                        ctxt->errorCtxt);
1232
7.52k
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
1233
7.52k
            ctxt->xpctxt->opLimit = 100000;
1234
7.52k
#endif
1235
8.95k
        } else {
1236
8.95k
            ctxt->xpctxt->doc = doc;
1237
8.95k
        }
1238
16.4k
  xptr = xmlXPtrEval(fragment, ctxt->xpctxt);
1239
16.4k
  if (ctxt->xpctxt->lastError.code != XML_ERR_OK) {
1240
11.6k
            if (ctxt->xpctxt->lastError.code == XML_ERR_NO_MEMORY)
1241
1.27k
                xmlXIncludeErrMemory(ctxt);
1242
10.3k
            else
1243
10.3k
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_FAILED,
1244
10.3k
                               "XPointer evaluation failed: #%s\n",
1245
10.3k
                               fragment);
1246
11.6k
            goto error;
1247
11.6k
  }
1248
4.84k
        if (xptr == NULL)
1249
2.75k
            goto done;
1250
2.09k
  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
2.09k
      case XPATH_NODESET:
1263
2.09k
                break;
1264
1265
2.09k
  }
1266
2.09k
  set = xptr->nodesetval;
1267
2.09k
  if (set != NULL) {
1268
816k
      for (i = 0;i < set->nodeNr;i++) {
1269
815k
    if (set->nodeTab[i] == NULL) /* shouldn't happen */
1270
0
        continue;
1271
815k
    switch (set->nodeTab[i]->type) {
1272
749k
        case XML_ELEMENT_NODE:
1273
813k
        case XML_TEXT_NODE:
1274
813k
        case XML_CDATA_SECTION_NODE:
1275
813k
        case XML_ENTITY_REF_NODE:
1276
813k
        case XML_ENTITY_NODE:
1277
814k
        case XML_PI_NODE:
1278
814k
        case XML_COMMENT_NODE:
1279
814k
        case XML_DOCUMENT_NODE:
1280
814k
        case XML_HTML_DOCUMENT_NODE:
1281
814k
      continue;
1282
1283
119
        case XML_ATTRIBUTE_NODE:
1284
119
      xmlXIncludeErr(ctxt, ref->elem,
1285
119
                     XML_XINCLUDE_XPTR_RESULT,
1286
119
               "XPointer selects an attribute: #%s\n",
1287
119
               fragment);
1288
119
      goto xptr_error;
1289
82
        case XML_NAMESPACE_DECL:
1290
82
      xmlXIncludeErr(ctxt, ref->elem,
1291
82
                     XML_XINCLUDE_XPTR_RESULT,
1292
82
               "XPointer selects a namespace: #%s\n",
1293
82
               fragment);
1294
82
      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
815k
    }
1311
815k
      }
1312
2.09k
  }
1313
1.89k
        ref->inc = xmlXIncludeCopyXPointer(ctxt, xptr, ref->base);
1314
2.09k
xptr_error:
1315
2.09k
        xmlXPathFreeObject(xptr);
1316
2.09k
    }
1317
1318
5.16k
done:
1319
5.16k
#endif
1320
1321
5.16k
    ret = 0;
1322
1323
22.0k
error:
1324
22.0k
    return(ret);
1325
5.16k
}
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
444
xmlXIncludeLoadTxt(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1336
444
    xmlParserInputBufferPtr buf;
1337
444
    xmlNodePtr node = NULL;
1338
444
    const xmlChar *url = ref->URI;
1339
444
    int i;
1340
444
    int ret = -1;
1341
444
    xmlChar *encoding = NULL;
1342
444
    xmlCharEncodingHandlerPtr handler = NULL;
1343
444
    xmlParserCtxtPtr pctxt = NULL;
1344
444
    xmlParserInputPtr inputStream = NULL;
1345
444
    int len;
1346
444
    int res;
1347
444
    const xmlChar *content;
1348
1349
    /*
1350
     * Handling of references to the local document are done
1351
     * directly through ctxt->doc.
1352
     */
1353
444
    if (url[0] == 0) {
1354
34
  xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_TEXT_DOCUMENT,
1355
34
           "text serialization of document not available\n", NULL);
1356
34
  goto error;
1357
34
    }
1358
1359
    /*
1360
     * Prevent reloading the document twice.
1361
     */
1362
474
    for (i = 0; i < ctxt->txtNr; i++) {
1363
112
  if (xmlStrEqual(url, ctxt->txtTab[i].url)) {
1364
48
            node = xmlNewDocText(ctxt->doc, ctxt->txtTab[i].text);
1365
48
            if (node == NULL)
1366
1
                xmlXIncludeErrMemory(ctxt);
1367
48
      goto loaded;
1368
48
  }
1369
112
    }
1370
1371
    /*
1372
     * Try to get the encoding if available
1373
     */
1374
362
    if (ref->elem != NULL) {
1375
362
  encoding = xmlXIncludeGetProp(ctxt, ref->elem, XINCLUDE_PARSE_ENCODING);
1376
362
    }
1377
362
    if (encoding != NULL) {
1378
141
        xmlParserErrors code;
1379
1380
141
        code = xmlOpenCharEncodingHandler((const char *) encoding,
1381
141
                                          /* output */ 0, &handler);
1382
1383
141
        if (code != XML_ERR_OK) {
1384
44
            if (code == XML_ERR_NO_MEMORY) {
1385
2
                xmlXIncludeErrMemory(ctxt);
1386
42
            } else if (code == XML_ERR_UNSUPPORTED_ENCODING) {
1387
42
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_UNKNOWN_ENCODING,
1388
42
                               "encoding %s not supported\n", encoding);
1389
42
                goto error;
1390
42
            } 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
141
    }
1397
1398
    /*
1399
     * Load it.
1400
     */
1401
320
    pctxt = xmlNewParserCtxt();
1402
320
    if (pctxt == NULL) {
1403
1
        xmlXIncludeErrMemory(ctxt);
1404
1
        goto error;
1405
1
    }
1406
319
    if (ctxt->errorHandler != NULL)
1407
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
1408
319
    if (ctxt->resourceLoader != NULL)
1409
319
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
1410
319
                                 ctxt->resourceCtxt);
1411
1412
319
    inputStream = xmlLoadResource(pctxt, (const char*) url, NULL,
1413
319
                                  XML_RESOURCE_XINCLUDE_TEXT);
1414
319
    if (inputStream == NULL) {
1415
        /*
1416
         * ENOENT only produces a warning which isn't reflected in errNo.
1417
         */
1418
100
        if (pctxt->errNo == XML_ERR_NO_MEMORY)
1419
1
            xmlXIncludeErrMemory(ctxt);
1420
99
        else if ((pctxt->errNo != XML_ERR_OK) &&
1421
99
                 (pctxt->errNo != XML_IO_ENOENT) &&
1422
99
                 (pctxt->errNo != XML_IO_UNKNOWN))
1423
2
            xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "load error", NULL);
1424
100
  goto error;
1425
100
    }
1426
219
    buf = inputStream->buf;
1427
219
    if (buf == NULL)
1428
0
  goto error;
1429
219
    if (buf->encoder)
1430
0
  xmlCharEncCloseFunc(buf->encoder);
1431
219
    buf->encoder = handler;
1432
219
    handler = NULL;
1433
1434
219
    node = xmlNewDocText(ctxt->doc, NULL);
1435
219
    if (node == NULL) {
1436
1
        xmlXIncludeErrMemory(ctxt);
1437
1
  goto error;
1438
1
    }
1439
1440
    /*
1441
     * Scan all chars from the resource and add the to the node
1442
     */
1443
218
    do {
1444
218
        res = xmlParserInputBufferRead(buf, 4096);
1445
218
    } while (res > 0);
1446
218
    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
218
    content = xmlBufContent(buf->buffer);
1455
218
    len = xmlBufUse(buf->buffer);
1456
17.6k
    for (i = 0; i < len;) {
1457
17.5k
        int cur;
1458
17.5k
        int l;
1459
1460
17.5k
        l = len - i;
1461
17.5k
        cur = xmlGetUTF8Char(&content[i], &l);
1462
17.5k
        if ((cur < 0) || (!IS_CHAR(cur))) {
1463
165
            xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_INVALID_CHAR,
1464
165
                           "%s contains invalid char\n", url);
1465
165
            goto error;
1466
165
        }
1467
1468
17.3k
        i += l;
1469
17.3k
    }
1470
1471
53
    if (xmlNodeAddContentLen(node, content, len) < 0)
1472
1
        xmlXIncludeErrMemory(ctxt);
1473
1474
53
    if (ctxt->txtNr >= ctxt->txtMax) {
1475
53
        xmlXIncludeTxt *tmp;
1476
53
        int newSize;
1477
1478
53
        newSize = xmlGrowCapacity(ctxt->txtMax, sizeof(tmp[0]),
1479
53
                                  8, XML_MAX_ITEMS);
1480
53
        if (newSize < 0) {
1481
0
            xmlXIncludeErrMemory(ctxt);
1482
0
      goto error;
1483
0
        }
1484
53
        tmp = xmlRealloc(ctxt->txtTab, newSize * sizeof(tmp[0]));
1485
53
        if (tmp == NULL) {
1486
1
            xmlXIncludeErrMemory(ctxt);
1487
1
      goto error;
1488
1
        }
1489
52
        ctxt->txtMax = newSize;
1490
52
        ctxt->txtTab = tmp;
1491
52
    }
1492
52
    ctxt->txtTab[ctxt->txtNr].text = xmlStrdup(node->content);
1493
52
    if ((node->content != NULL) &&
1494
52
        (ctxt->txtTab[ctxt->txtNr].text == NULL)) {
1495
1
        xmlXIncludeErrMemory(ctxt);
1496
1
        goto error;
1497
1
    }
1498
51
    ctxt->txtTab[ctxt->txtNr].url = xmlStrdup(url);
1499
51
    if (ctxt->txtTab[ctxt->txtNr].url == NULL) {
1500
1
        xmlXIncludeErrMemory(ctxt);
1501
1
        xmlFree(ctxt->txtTab[ctxt->txtNr].text);
1502
1
        goto error;
1503
1
    }
1504
50
    ctxt->txtNr++;
1505
1506
98
loaded:
1507
    /*
1508
     * Add the element as the replacement copy.
1509
     */
1510
98
    ref->inc = node;
1511
98
    node = NULL;
1512
98
    ret = 0;
1513
1514
444
error:
1515
444
    xmlFreeNode(node);
1516
444
    xmlFreeInputStream(inputStream);
1517
444
    xmlFreeParserCtxt(pctxt);
1518
444
    xmlCharEncCloseFunc(handler);
1519
444
    xmlFree(encoding);
1520
444
    return(ret);
1521
98
}
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
3.03k
                        xmlXIncludeRefPtr ref) {
1535
3.03k
    int ret = 0;
1536
3.03k
    int oldNbErrors;
1537
1538
3.03k
    if ((fallback == NULL) || (fallback->type == XML_NAMESPACE_DECL) ||
1539
3.03k
        (ctxt == NULL))
1540
0
  return(-1);
1541
3.03k
    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
2.95k
        oldNbErrors = ctxt->nbErrors;
1547
2.95k
  ref->inc = xmlXIncludeCopyNode(ctxt, fallback, 1, ref->base);
1548
2.95k
  if (ctxt->nbErrors > oldNbErrors)
1549
812
      ret = -1;
1550
2.95k
    } else {
1551
77
        ref->inc = NULL;
1552
77
    }
1553
3.03k
    ref->fallback = 1;
1554
3.03k
    return(ret);
1555
3.03k
}
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
29.0k
xmlXIncludeExpandNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1573
29.0k
    xmlXIncludeRefPtr ref;
1574
29.0k
    int i;
1575
1576
29.0k
    if (ctxt->fatalErr)
1577
506
        return(NULL);
1578
28.5k
    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
28.5k
#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
28.5k
    if (ctxt->incTotal >= 20)
1606
1.66k
        return(NULL);
1607
26.9k
    ctxt->incTotal++;
1608
26.9k
#endif
1609
1610
94.1k
    for (i = 0; i < ctxt->incNr; i++) {
1611
69.3k
        if (ctxt->incTab[i]->elem == node) {
1612
2.02k
            if (ctxt->incTab[i]->expanding) {
1613
961
                xmlXIncludeErr(ctxt, node, XML_XINCLUDE_RECURSION,
1614
961
                               "inclusion loop detected\n", NULL);
1615
961
                return(NULL);
1616
961
            }
1617
1.06k
            return(ctxt->incTab[i]);
1618
2.02k
        }
1619
69.3k
    }
1620
1621
24.8k
    ref = xmlXIncludeAddNode(ctxt, node);
1622
24.8k
    if (ref == NULL)
1623
2.35k
        return(NULL);
1624
22.5k
    ref->expanding = 1;
1625
22.5k
    ctxt->depth++;
1626
22.5k
    xmlXIncludeLoadNode(ctxt, ref);
1627
22.5k
    ctxt->depth--;
1628
22.5k
    ref->expanding = 0;
1629
1630
22.5k
    return(ref);
1631
24.8k
}
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
22.5k
xmlXIncludeLoadNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1642
22.5k
    xmlNodePtr cur;
1643
22.5k
    int ret;
1644
1645
22.5k
    if ((ctxt == NULL) || (ref == NULL))
1646
0
  return(-1);
1647
22.5k
    cur = ref->elem;
1648
22.5k
    if (cur == NULL)
1649
0
  return(-1);
1650
1651
22.5k
    if (ref->xml) {
1652
22.0k
  ret = xmlXIncludeLoadDoc(ctxt, ref);
1653
  /* xmlXIncludeGetFragment(ctxt, cur, URI); */
1654
22.0k
    } else {
1655
444
  ret = xmlXIncludeLoadTxt(ctxt, ref);
1656
444
    }
1657
1658
22.5k
    if (ret < 0) {
1659
17.2k
  xmlNodePtr children;
1660
1661
  /*
1662
   * Time to try a fallback if available
1663
   */
1664
17.2k
  children = cur->children;
1665
40.7k
  while (children != NULL) {
1666
26.5k
      if ((children->type == XML_ELEMENT_NODE) &&
1667
26.5k
    (children->ns != NULL) &&
1668
26.5k
    (xmlStrEqual(children->name, XINCLUDE_FALLBACK)) &&
1669
26.5k
    ((xmlStrEqual(children->ns->href, XINCLUDE_NS)) ||
1670
3.23k
     (xmlStrEqual(children->ns->href, XINCLUDE_OLD_NS)))) {
1671
3.03k
    ret = xmlXIncludeLoadFallback(ctxt, children, ref);
1672
3.03k
    break;
1673
3.03k
      }
1674
23.5k
      children = children->next;
1675
23.5k
  }
1676
17.2k
    }
1677
22.5k
    if (ret < 0) {
1678
15.0k
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_FALLBACK,
1679
15.0k
           "could not load %s, and no fallback was found\n",
1680
15.0k
           ref->URI);
1681
15.0k
    }
1682
1683
22.5k
    return(0);
1684
22.5k
}
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
20.6k
xmlXIncludeIncludeNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1695
20.6k
    xmlNodePtr cur, end, list, tmp;
1696
1697
20.6k
    if ((ctxt == NULL) || (ref == NULL))
1698
0
  return(-1);
1699
20.6k
    cur = ref->elem;
1700
20.6k
    if ((cur == NULL) || (cur->type == XML_NAMESPACE_DECL))
1701
0
  return(-1);
1702
1703
20.6k
    list = ref->inc;
1704
20.6k
    ref->inc = NULL;
1705
1706
    /*
1707
     * Check against the risk of generating a multi-rooted document
1708
     */
1709
20.6k
    if ((cur->parent != NULL) &&
1710
20.6k
  (cur->parent->type != XML_ELEMENT_NODE)) {
1711
80
  int nb_elem = 0;
1712
1713
80
  tmp = list;
1714
797
  while (tmp != NULL) {
1715
717
      if (tmp->type == XML_ELEMENT_NODE)
1716
312
    nb_elem++;
1717
717
      tmp = tmp->next;
1718
717
  }
1719
80
        if (nb_elem != 1) {
1720
62
            if (nb_elem > 1)
1721
15
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1722
15
                               "XInclude error: would result in multiple root "
1723
15
                               "nodes\n", NULL);
1724
47
            else
1725
47
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1726
47
                               "XInclude error: would result in no root "
1727
47
                               "node\n", NULL);
1728
62
            xmlFreeNodeList(list);
1729
62
      return(-1);
1730
62
  }
1731
80
    }
1732
1733
20.5k
    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
11.9k
  while (list != NULL) {
1740
7.66k
      end = list;
1741
7.66k
      list = list->next;
1742
1743
7.66k
      if (xmlAddPrevSibling(cur, end) == NULL) {
1744
0
                xmlUnlinkNode(end);
1745
0
                xmlFreeNode(end);
1746
0
                goto err_memory;
1747
0
            }
1748
7.66k
  }
1749
4.27k
  xmlUnlinkNode(cur);
1750
4.27k
  xmlFreeNode(cur);
1751
16.2k
    } else {
1752
16.2k
        xmlNodePtr child, next;
1753
1754
  /*
1755
   * Change the current node as an XInclude start one, and add an
1756
   * XInclude end one
1757
   */
1758
16.2k
        if (ref->fallback)
1759
1.39k
            xmlUnsetProp(cur, BAD_CAST "href");
1760
16.2k
  cur->type = XML_XINCLUDE_START;
1761
        /* Remove fallback children */
1762
44.3k
        for (child = cur->children; child != NULL; child = next) {
1763
28.0k
            next = child->next;
1764
28.0k
            xmlUnlinkNode(child);
1765
28.0k
            xmlFreeNode(child);
1766
28.0k
        }
1767
16.2k
  end = xmlNewDocNode(cur->doc, cur->ns, cur->name, NULL);
1768
16.2k
  if (end == NULL)
1769
27
            goto err_memory;
1770
16.2k
  end->type = XML_XINCLUDE_END;
1771
16.2k
  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
55.1k
  while (list != NULL) {
1780
38.9k
      cur = list;
1781
38.9k
      list = list->next;
1782
1783
38.9k
      if (xmlAddPrevSibling(end, cur) == NULL) {
1784
0
                xmlUnlinkNode(cur);
1785
0
                xmlFreeNode(cur);
1786
0
                goto err_memory;
1787
0
            }
1788
38.9k
  }
1789
16.2k
    }
1790
1791
1792
20.5k
    return(0);
1793
1794
27
err_memory:
1795
27
    xmlXIncludeErrMemory(ctxt);
1796
27
    xmlFreeNodeList(list);
1797
27
    return(-1);
1798
20.5k
}
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
1.16M
xmlXIncludeTestNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1809
1.16M
    if (node == NULL)
1810
0
  return(0);
1811
1.16M
    if (node->type != XML_ELEMENT_NODE)
1812
310k
  return(0);
1813
853k
    if (node->ns == NULL)
1814
429k
  return(0);
1815
423k
    if ((xmlStrEqual(node->ns->href, XINCLUDE_NS)) ||
1816
423k
        (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS))) {
1817
32.0k
  if (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS)) {
1818
28.7k
      if (ctxt->legacy == 0) {
1819
8.77k
          ctxt->legacy = 1;
1820
8.77k
      }
1821
28.7k
  }
1822
32.0k
  if (xmlStrEqual(node->name, XINCLUDE_NODE)) {
1823
25.7k
      xmlNodePtr child = node->children;
1824
25.7k
      int nb_fallback = 0;
1825
1826
78.0k
      while (child != NULL) {
1827
52.9k
    if ((child->type == XML_ELEMENT_NODE) &&
1828
52.9k
        (child->ns != NULL) &&
1829
52.9k
        ((xmlStrEqual(child->ns->href, XINCLUDE_NS)) ||
1830
16.3k
         (xmlStrEqual(child->ns->href, XINCLUDE_OLD_NS)))) {
1831
3.59k
        if (xmlStrEqual(child->name, XINCLUDE_NODE)) {
1832
636
      xmlXIncludeErr(ctxt, node,
1833
636
                     XML_XINCLUDE_INCLUDE_IN_INCLUDE,
1834
636
               "%s has an 'include' child\n",
1835
636
               XINCLUDE_NODE);
1836
636
      return(0);
1837
636
        }
1838
2.96k
        if (xmlStrEqual(child->name, XINCLUDE_FALLBACK)) {
1839
2.30k
      nb_fallback++;
1840
2.30k
        }
1841
2.96k
    }
1842
52.2k
    child = child->next;
1843
52.2k
      }
1844
25.1k
      if (nb_fallback > 1) {
1845
206
    xmlXIncludeErr(ctxt, node, XML_XINCLUDE_FALLBACKS_IN_INCLUDE,
1846
206
             "%s has multiple fallback children\n",
1847
206
                   XINCLUDE_NODE);
1848
206
    return(0);
1849
206
      }
1850
24.9k
      return(1);
1851
25.1k
  }
1852
6.32k
  if (xmlStrEqual(node->name, XINCLUDE_FALLBACK)) {
1853
1.94k
      if ((node->parent == NULL) ||
1854
1.94k
    (node->parent->type != XML_ELEMENT_NODE) ||
1855
1.94k
    (node->parent->ns == NULL) ||
1856
1.94k
    ((!xmlStrEqual(node->parent->ns->href, XINCLUDE_NS)) &&
1857
1.27k
     (!xmlStrEqual(node->parent->ns->href, XINCLUDE_OLD_NS))) ||
1858
1.94k
    (!xmlStrEqual(node->parent->name, XINCLUDE_NODE))) {
1859
1.33k
    xmlXIncludeErr(ctxt, node,
1860
1.33k
                   XML_XINCLUDE_FALLBACK_NOT_IN_INCLUDE,
1861
1.33k
             "%s is not the child of an 'include'\n",
1862
1.33k
             XINCLUDE_FALLBACK);
1863
1.33k
      }
1864
1.94k
  }
1865
6.32k
    }
1866
398k
    return(0);
1867
423k
}
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
19.9k
xmlXIncludeDoProcess(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1879
19.9k
    xmlXIncludeRefPtr ref;
1880
19.9k
    xmlNodePtr cur;
1881
19.9k
    int ret = 0;
1882
19.9k
    int i, start;
1883
1884
    /*
1885
     * First phase: lookup the elements in the document
1886
     */
1887
19.9k
    start = ctxt->incNr;
1888
19.9k
    cur = tree;
1889
1.16M
    do {
1890
  /* TODO: need to work on entities -> stack */
1891
1.16M
        if (xmlXIncludeTestNode(ctxt, cur) == 1) {
1892
24.9k
            ref = xmlXIncludeExpandNode(ctxt, cur);
1893
            /*
1894
             * Mark direct includes.
1895
             */
1896
24.9k
            if (ref != NULL)
1897
20.6k
                ref->replace = 1;
1898
1.13M
        } else if ((cur->children != NULL) &&
1899
1.13M
                   ((cur->type == XML_DOCUMENT_NODE) ||
1900
189k
                    (cur->type == XML_ELEMENT_NODE))) {
1901
181k
            cur = cur->children;
1902
181k
            continue;
1903
181k
        }
1904
1.16M
        do {
1905
1.16M
            if (cur == tree)
1906
19.9k
                break;
1907
1.14M
            if (cur->next != NULL) {
1908
962k
                cur = cur->next;
1909
962k
                break;
1910
962k
            }
1911
181k
            cur = cur->parent;
1912
181k
        } while (cur != NULL);
1913
1.16M
    } while ((cur != NULL) && (cur != tree));
1914
1915
    /*
1916
     * Second phase: extend the original document infoset.
1917
     */
1918
42.4k
    for (i = start; i < ctxt->incNr; i++) {
1919
22.5k
  if (ctxt->incTab[i]->replace != 0) {
1920
20.6k
            xmlXIncludeIncludeNode(ctxt, ctxt->incTab[i]);
1921
20.6k
            ctxt->incTab[i]->replace = 0;
1922
20.6k
        } else {
1923
            /*
1924
             * Ignore includes which were added indirectly, for example
1925
             * inside xi:fallback elements.
1926
             */
1927
1.90k
            if (ctxt->incTab[i]->inc != NULL) {
1928
1.36k
                xmlFreeNodeList(ctxt->incTab[i]->inc);
1929
1.36k
                ctxt->incTab[i]->inc = NULL;
1930
1.36k
            }
1931
1.90k
        }
1932
22.5k
  ret++;
1933
22.5k
    }
1934
1935
19.9k
    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
19.9k
    return(ret);
1948
19.9k
}
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
19.7k
xmlXIncludeDoProcessRoot(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1960
19.7k
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL))
1961
0
  return(-1);
1962
19.7k
    if (ctxt == NULL)
1963
0
  return(-1);
1964
1965
19.7k
    return(xmlXIncludeDoProcess(ctxt, tree));
1966
19.7k
}
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
39.5k
xmlXIncludeGetLastError(xmlXIncludeCtxt *ctxt) {
1976
39.5k
    if (ctxt == NULL)
1977
0
        return(XML_ERR_ARGUMENT);
1978
39.5k
    return(ctxt->errNo);
1979
39.5k
}
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
28.9k
                             xmlResourceLoader loader, void *data) {
2011
28.9k
    if (ctxt == NULL)
2012
9.20k
        return;
2013
19.7k
    ctxt->resourceLoader = loader;
2014
19.7k
    ctxt->resourceCtxt = data;
2015
19.7k
}
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
28.9k
xmlXIncludeSetFlags(xmlXIncludeCtxt *ctxt, int flags) {
2026
28.9k
    if (ctxt == NULL)
2027
9.20k
        return(-1);
2028
19.7k
    ctxt->parseFlags = flags;
2029
19.7k
    return(0);
2030
28.9k
}
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
28.9k
xmlXIncludeProcessNode(xmlXIncludeCtxt *ctxt, xmlNode *node) {
2178
28.9k
    int ret = 0;
2179
2180
28.9k
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL) ||
2181
28.9k
        (node->doc == NULL) || (ctxt == NULL))
2182
9.20k
  return(-1);
2183
19.7k
    ret = xmlXIncludeDoProcessRoot(ctxt, node);
2184
19.7k
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2185
7.13k
  ret = -1;
2186
19.7k
    return(ret);
2187
28.9k
}
2188
2189
#else /* !LIBXML_XINCLUDE_ENABLED */
2190
#endif