Coverage Report

Created: 2026-06-08 06:59

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libxml2/entities.c
Line
Count
Source
1
/*
2
 * entities.c : implementation for the XML entities handling
3
 *
4
 * See Copyright for the status of this software.
5
 *
6
 * daniel@veillard.com
7
 */
8
9
/* To avoid EBCDIC trouble when parsing on zOS */
10
#if defined(__MVS__)
11
#pragma convert("ISO8859-1")
12
#endif
13
14
#define IN_LIBXML
15
#include "libxml.h"
16
17
#include <string.h>
18
#include <stdlib.h>
19
20
#include <libxml/xmlmemory.h>
21
#include <libxml/hash.h>
22
#include <libxml/entities.h>
23
#include <libxml/parser.h>
24
#include <libxml/parserInternals.h>
25
#include <libxml/xmlerror.h>
26
#include <libxml/dict.h>
27
#include <libxml/xmlsave.h>
28
29
#include "private/entities.h"
30
#include "private/error.h"
31
32
#ifndef SIZE_MAX
33
  #define SIZE_MAX ((size_t) -1)
34
#endif
35
36
/*
37
 * The XML predefined entities.
38
 */
39
40
static xmlEntity xmlEntityLt = {
41
    NULL, XML_ENTITY_DECL, BAD_CAST "lt",
42
    NULL, NULL, NULL, NULL, NULL, NULL,
43
    BAD_CAST "<", BAD_CAST "<", 1,
44
    XML_INTERNAL_PREDEFINED_ENTITY,
45
    NULL, NULL, NULL, NULL, 0, 0, 0
46
};
47
static xmlEntity xmlEntityGt = {
48
    NULL, XML_ENTITY_DECL, BAD_CAST "gt",
49
    NULL, NULL, NULL, NULL, NULL, NULL,
50
    BAD_CAST ">", BAD_CAST ">", 1,
51
    XML_INTERNAL_PREDEFINED_ENTITY,
52
    NULL, NULL, NULL, NULL, 0, 0, 0
53
};
54
static xmlEntity xmlEntityAmp = {
55
    NULL, XML_ENTITY_DECL, BAD_CAST "amp",
56
    NULL, NULL, NULL, NULL, NULL, NULL,
57
    BAD_CAST "&", BAD_CAST "&", 1,
58
    XML_INTERNAL_PREDEFINED_ENTITY,
59
    NULL, NULL, NULL, NULL, 0, 0, 0
60
};
61
static xmlEntity xmlEntityQuot = {
62
    NULL, XML_ENTITY_DECL, BAD_CAST "quot",
63
    NULL, NULL, NULL, NULL, NULL, NULL,
64
    BAD_CAST "\"", BAD_CAST "\"", 1,
65
    XML_INTERNAL_PREDEFINED_ENTITY,
66
    NULL, NULL, NULL, NULL, 0, 0, 0
67
};
68
static xmlEntity xmlEntityApos = {
69
    NULL, XML_ENTITY_DECL, BAD_CAST "apos",
70
    NULL, NULL, NULL, NULL, NULL, NULL,
71
    BAD_CAST "'", BAD_CAST "'", 1,
72
    XML_INTERNAL_PREDEFINED_ENTITY,
73
    NULL, NULL, NULL, NULL, 0, 0, 0
74
};
75
76
/*
77
 * xmlFreeEntity:
78
 * @entity:  an entity
79
 *
80
 * Frees the entity.
81
 */
82
void
83
xmlFreeEntity(xmlEntityPtr entity)
84
85.9k
{
85
85.9k
    xmlDictPtr dict = NULL;
86
87
85.9k
    if (entity == NULL)
88
0
        return;
89
90
85.9k
    if (entity->doc != NULL)
91
85.8k
        dict = entity->doc->dict;
92
93
94
85.9k
    if ((entity->children) &&
95
1.52k
        (entity == (xmlEntityPtr) entity->children->parent))
96
1.52k
        xmlFreeNodeList(entity->children);
97
85.9k
    if ((entity->name != NULL) &&
98
85.9k
        ((dict == NULL) || (!xmlDictOwns(dict, entity->name))))
99
27.1k
        xmlFree((char *) entity->name);
100
85.9k
    if (entity->ExternalID != NULL)
101
7.71k
        xmlFree((char *) entity->ExternalID);
102
85.9k
    if (entity->SystemID != NULL)
103
32.6k
        xmlFree((char *) entity->SystemID);
104
85.9k
    if (entity->URI != NULL)
105
31.1k
        xmlFree((char *) entity->URI);
106
85.9k
    if (entity->content != NULL)
107
53.2k
        xmlFree((char *) entity->content);
108
85.9k
    if (entity->orig != NULL)
109
50.6k
        xmlFree((char *) entity->orig);
110
85.9k
    xmlFree(entity);
111
85.9k
}
112
113
/*
114
 * xmlCreateEntity:
115
 *
116
 * internal routine doing the entity node structures allocations
117
 */
118
static xmlEntityPtr
119
xmlCreateEntity(xmlDocPtr doc, const xmlChar *name, int type,
120
          const xmlChar *ExternalID, const xmlChar *SystemID,
121
66.8k
          const xmlChar *content) {
122
66.8k
    xmlEntityPtr ret;
123
124
66.8k
    ret = (xmlEntityPtr) xmlMalloc(sizeof(xmlEntity));
125
66.8k
    if (ret == NULL)
126
12
  return(NULL);
127
66.8k
    memset(ret, 0, sizeof(xmlEntity));
128
66.8k
    ret->doc = doc;
129
66.8k
    ret->type = XML_ENTITY_DECL;
130
131
    /*
132
     * fill the structure.
133
     */
134
66.8k
    ret->etype = (xmlEntityType) type;
135
66.8k
    if ((doc == NULL) || (doc->dict == NULL))
136
8.07k
  ret->name = xmlStrdup(name);
137
58.7k
    else
138
58.7k
        ret->name = xmlDictLookup(doc->dict, name, -1);
139
66.8k
    if (ret->name == NULL)
140
3
        goto error;
141
66.8k
    if (ExternalID != NULL) {
142
7.71k
        ret->ExternalID = xmlStrdup(ExternalID);
143
7.71k
        if (ret->ExternalID == NULL)
144
2
            goto error;
145
7.71k
    }
146
66.8k
    if (SystemID != NULL) {
147
30.7k
        ret->SystemID = xmlStrdup(SystemID);
148
30.7k
        if (ret->SystemID == NULL)
149
6
            goto error;
150
30.7k
    }
151
66.8k
    if (content != NULL) {
152
34.6k
        ret->length = xmlStrlen(content);
153
34.6k
  ret->content = xmlStrndup(content, ret->length);
154
34.6k
        if (ret->content == NULL)
155
6
            goto error;
156
34.6k
     } else {
157
32.2k
        ret->length = 0;
158
32.2k
        ret->content = NULL;
159
32.2k
    }
160
66.8k
    ret->URI = NULL; /* to be computed by the layer knowing
161
      the defining entity */
162
66.8k
    ret->orig = NULL;
163
164
66.8k
    return(ret);
165
166
17
error:
167
17
    xmlFreeEntity(ret);
168
17
    return(NULL);
169
66.8k
}
170
171
/**
172
 * xmlAddEntity:
173
 * @doc:  the document
174
 * @extSubset:  add to the external or internal subset
175
 * @name:  the entity name
176
 * @type:  the entity type XML_xxx_yyy_ENTITY
177
 * @ExternalID:  the entity external ID if available
178
 * @SystemID:  the entity system ID if available
179
 * @content:  the entity content
180
 * @out:  pointer to resulting entity (optional)
181
 *
182
 * Register a new entity for this document.
183
 *
184
 * Available since 2.13.0.
185
 *
186
 * Returns an xmlParserErrors error code.
187
 */
188
int
189
xmlAddEntity(xmlDocPtr doc, int extSubset, const xmlChar *name, int type,
190
    const xmlChar *ExternalID, const xmlChar *SystemID,
191
68.6k
    const xmlChar *content, xmlEntityPtr *out) {
192
68.6k
    xmlDtdPtr dtd;
193
68.6k
    xmlDictPtr dict = NULL;
194
68.6k
    xmlEntitiesTablePtr table = NULL;
195
68.6k
    xmlEntityPtr ret, predef;
196
68.6k
    int res;
197
198
68.6k
    if (out != NULL)
199
68.6k
        *out = NULL;
200
68.6k
    if ((doc == NULL) || (name == NULL))
201
0
  return(XML_ERR_ARGUMENT);
202
68.6k
    dict = doc->dict;
203
204
68.6k
    if (extSubset)
205
8.13k
        dtd = doc->extSubset;
206
60.5k
    else
207
60.5k
        dtd = doc->intSubset;
208
68.6k
    if (dtd == NULL)
209
0
        return(XML_DTD_NO_DTD);
210
211
68.6k
    switch (type) {
212
21.1k
        case XML_INTERNAL_GENERAL_ENTITY:
213
29.6k
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
214
29.9k
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
215
29.9k
            predef = xmlGetPredefinedEntity(name);
216
29.9k
            if (predef != NULL) {
217
2.73k
                int valid = 0;
218
219
                /* 4.6 Predefined Entities */
220
2.73k
                if ((type == XML_INTERNAL_GENERAL_ENTITY) &&
221
2.47k
                    (content != NULL)) {
222
2.14k
                    int c = predef->content[0];
223
224
2.14k
                    if (((content[0] == c) && (content[1] == 0)) &&
225
728
                        ((c == '>') || (c == '\'') || (c == '"'))) {
226
656
                        valid = 1;
227
1.48k
                    } else if ((content[0] == '&') && (content[1] == '#')) {
228
571
                        if (content[2] == 'x') {
229
69
                            xmlChar *hex = BAD_CAST "0123456789ABCDEF";
230
69
                            xmlChar ref[] = "00;";
231
232
69
                            ref[0] = hex[c / 16 % 16];
233
69
                            ref[1] = hex[c % 16];
234
69
                            if (xmlStrcasecmp(&content[3], ref) == 0)
235
0
                                valid = 1;
236
502
                        } else {
237
502
                            xmlChar ref[] = "00;";
238
239
502
                            ref[0] = '0' + c / 10 % 10;
240
502
                            ref[1] = '0' + c % 10;
241
502
                            if (xmlStrEqual(&content[2], ref))
242
337
                                valid = 1;
243
502
                        }
244
571
                    }
245
2.14k
                }
246
2.73k
                if (!valid)
247
1.74k
                    return(XML_ERR_REDECL_PREDEF_ENTITY);
248
2.73k
            }
249
28.1k
      if (dtd->entities == NULL) {
250
19.8k
    dtd->entities = xmlHashCreateDict(0, dict);
251
19.8k
                if (dtd->entities == NULL)
252
4
                    return(XML_ERR_NO_MEMORY);
253
19.8k
            }
254
28.1k
      table = dtd->entities;
255
28.1k
      break;
256
15.9k
        case XML_INTERNAL_PARAMETER_ENTITY:
257
38.7k
        case XML_EXTERNAL_PARAMETER_ENTITY:
258
38.7k
      if (dtd->pentities == NULL) {
259
19.3k
    dtd->pentities = xmlHashCreateDict(0, dict);
260
19.3k
                if (dtd->pentities == NULL)
261
1
                    return(XML_ERR_NO_MEMORY);
262
19.3k
            }
263
38.7k
      table = dtd->pentities;
264
38.7k
      break;
265
0
        default:
266
0
      return(XML_ERR_ARGUMENT);
267
68.6k
    }
268
66.8k
    ret = xmlCreateEntity(dtd->doc, name, type, ExternalID, SystemID, content);
269
66.8k
    if (ret == NULL)
270
29
        return(XML_ERR_NO_MEMORY);
271
272
66.8k
    res = xmlHashAdd(table, name, ret);
273
66.8k
    if (res < 0) {
274
6
        xmlFreeEntity(ret);
275
6
        return(XML_ERR_NO_MEMORY);
276
66.8k
    } else if (res == 0) {
277
  /*
278
   * entity was already defined at another level.
279
   */
280
1.17k
        xmlFreeEntity(ret);
281
1.17k
  return(XML_WAR_ENTITY_REDEFINED);
282
1.17k
    }
283
284
    /*
285
     * Link it to the DTD
286
     */
287
65.6k
    ret->parent = dtd;
288
65.6k
    ret->doc = dtd->doc;
289
65.6k
    if (dtd->last == NULL) {
290
28.0k
  dtd->children = dtd->last = (xmlNodePtr) ret;
291
37.6k
    } else {
292
37.6k
  dtd->last->next = (xmlNodePtr) ret;
293
37.6k
  ret->prev = dtd->last;
294
37.6k
  dtd->last = (xmlNodePtr) ret;
295
37.6k
    }
296
297
65.6k
    if (out != NULL)
298
65.6k
        *out = ret;
299
65.6k
    return(0);
300
66.8k
}
301
302
/**
303
 * xmlGetPredefinedEntity:
304
 * @name:  the entity name
305
 *
306
 * Check whether this name is an predefined entity.
307
 *
308
 * Returns NULL if not, otherwise the entity
309
 */
310
xmlEntityPtr
311
929k
xmlGetPredefinedEntity(const xmlChar *name) {
312
929k
    if (name == NULL) return(NULL);
313
929k
    switch (name[0]) {
314
13.1k
        case 'l':
315
13.1k
      if (xmlStrEqual(name, BAD_CAST "lt"))
316
11.4k
          return(&xmlEntityLt);
317
1.73k
      break;
318
26.7k
        case 'g':
319
26.7k
      if (xmlStrEqual(name, BAD_CAST "gt"))
320
23.9k
          return(&xmlEntityGt);
321
2.76k
      break;
322
379k
        case 'a':
323
379k
      if (xmlStrEqual(name, BAD_CAST "amp"))
324
45.5k
          return(&xmlEntityAmp);
325
334k
      if (xmlStrEqual(name, BAD_CAST "apos"))
326
1.09k
          return(&xmlEntityApos);
327
332k
      break;
328
332k
        case 'q':
329
254k
      if (xmlStrEqual(name, BAD_CAST "quot"))
330
502
          return(&xmlEntityQuot);
331
254k
      break;
332
254k
  default:
333
254k
      break;
334
929k
    }
335
846k
    return(NULL);
336
929k
}
337
338
/**
339
 * xmlAddDtdEntity:
340
 * @doc:  the document
341
 * @name:  the entity name
342
 * @type:  the entity type XML_xxx_yyy_ENTITY
343
 * @ExternalID:  the entity external ID if available
344
 * @SystemID:  the entity system ID if available
345
 * @content:  the entity content
346
 *
347
 * Register a new entity for this document DTD external subset.
348
 *
349
 * Returns a pointer to the entity or NULL in case of error
350
 */
351
xmlEntityPtr
352
xmlAddDtdEntity(xmlDocPtr doc, const xmlChar *name, int type,
353
          const xmlChar *ExternalID, const xmlChar *SystemID,
354
0
    const xmlChar *content) {
355
0
    xmlEntityPtr ret;
356
357
0
    xmlAddEntity(doc, 1, name, type, ExternalID, SystemID, content, &ret);
358
0
    return(ret);
359
0
}
360
361
/**
362
 * xmlAddDocEntity:
363
 * @doc:  the document
364
 * @name:  the entity name
365
 * @type:  the entity type XML_xxx_yyy_ENTITY
366
 * @ExternalID:  the entity external ID if available
367
 * @SystemID:  the entity system ID if available
368
 * @content:  the entity content
369
 *
370
 * Register a new entity for this document.
371
 *
372
 * Returns a pointer to the entity or NULL in case of error
373
 */
374
xmlEntityPtr
375
xmlAddDocEntity(xmlDocPtr doc, const xmlChar *name, int type,
376
          const xmlChar *ExternalID, const xmlChar *SystemID,
377
0
          const xmlChar *content) {
378
0
    xmlEntityPtr ret;
379
380
0
    xmlAddEntity(doc, 0, name, type, ExternalID, SystemID, content, &ret);
381
0
    return(ret);
382
0
}
383
384
/**
385
 * xmlNewEntity:
386
 * @doc:  the document
387
 * @name:  the entity name
388
 * @type:  the entity type XML_xxx_yyy_ENTITY
389
 * @ExternalID:  the entity external ID if available
390
 * @SystemID:  the entity system ID if available
391
 * @content:  the entity content
392
 *
393
 * Create a new entity, this differs from xmlAddDocEntity() that if
394
 * the document is NULL or has no internal subset defined, then an
395
 * unlinked entity structure will be returned, it is then the responsibility
396
 * of the caller to link it to the document later or free it when not needed
397
 * anymore.
398
 *
399
 * Returns a pointer to the entity or NULL in case of error
400
 */
401
xmlEntityPtr
402
xmlNewEntity(xmlDocPtr doc, const xmlChar *name, int type,
403
       const xmlChar *ExternalID, const xmlChar *SystemID,
404
0
       const xmlChar *content) {
405
0
    if ((doc != NULL) && (doc->intSubset != NULL)) {
406
0
  return(xmlAddDocEntity(doc, name, type, ExternalID, SystemID, content));
407
0
    }
408
0
    if (name == NULL)
409
0
        return(NULL);
410
0
    return(xmlCreateEntity(doc, name, type, ExternalID, SystemID, content));
411
0
}
412
413
/**
414
 * xmlGetEntityFromTable:
415
 * @table:  an entity table
416
 * @name:  the entity name
417
 * @parameter:  look for parameter entities
418
 *
419
 * Do an entity lookup in the table.
420
 * returns the corresponding parameter entity, if found.
421
 *
422
 * Returns A pointer to the entity structure or NULL if not found.
423
 */
424
static xmlEntityPtr
425
438k
xmlGetEntityFromTable(xmlEntitiesTablePtr table, const xmlChar *name) {
426
438k
    return((xmlEntityPtr) xmlHashLookup(table, name));
427
438k
}
428
429
/**
430
 * xmlGetParameterEntity:
431
 * @doc:  the document referencing the entity
432
 * @name:  the entity name
433
 *
434
 * Do an entity lookup in the internal and external subsets and
435
 * returns the corresponding parameter entity, if found.
436
 *
437
 * Returns A pointer to the entity structure or NULL if not found.
438
 */
439
xmlEntityPtr
440
84.6k
xmlGetParameterEntity(xmlDocPtr doc, const xmlChar *name) {
441
84.6k
    xmlEntitiesTablePtr table;
442
84.6k
    xmlEntityPtr ret;
443
444
84.6k
    if (doc == NULL)
445
673
  return(NULL);
446
83.9k
    if ((doc->intSubset != NULL) && (doc->intSubset->pentities != NULL)) {
447
78.9k
  table = (xmlEntitiesTablePtr) doc->intSubset->pentities;
448
78.9k
  ret = xmlGetEntityFromTable(table, name);
449
78.9k
  if (ret != NULL)
450
61.7k
      return(ret);
451
78.9k
    }
452
22.2k
    if ((doc->extSubset != NULL) && (doc->extSubset->pentities != NULL)) {
453
9.33k
  table = (xmlEntitiesTablePtr) doc->extSubset->pentities;
454
9.33k
  return(xmlGetEntityFromTable(table, name));
455
9.33k
    }
456
12.9k
    return(NULL);
457
22.2k
}
458
459
/**
460
 * xmlGetDtdEntity:
461
 * @doc:  the document referencing the entity
462
 * @name:  the entity name
463
 *
464
 * Do an entity lookup in the DTD entity hash table and
465
 * returns the corresponding entity, if found.
466
 * Note: the first argument is the document node, not the DTD node.
467
 *
468
 * Returns A pointer to the entity structure or NULL if not found.
469
 */
470
xmlEntityPtr
471
0
xmlGetDtdEntity(xmlDocPtr doc, const xmlChar *name) {
472
0
    xmlEntitiesTablePtr table;
473
474
0
    if (doc == NULL)
475
0
  return(NULL);
476
0
    if ((doc->extSubset != NULL) && (doc->extSubset->entities != NULL)) {
477
0
  table = (xmlEntitiesTablePtr) doc->extSubset->entities;
478
0
  return(xmlGetEntityFromTable(table, name));
479
0
    }
480
0
    return(NULL);
481
0
}
482
483
/**
484
 * xmlGetDocEntity:
485
 * @doc:  the document referencing the entity
486
 * @name:  the entity name
487
 *
488
 * Do an entity lookup in the document entity hash table and
489
 * returns the corresponding entity, otherwise a lookup is done
490
 * in the predefined entities too.
491
 *
492
 * Returns A pointer to the entity structure or NULL if not found.
493
 */
494
xmlEntityPtr
495
391k
xmlGetDocEntity(const xmlDoc *doc, const xmlChar *name) {
496
391k
    xmlEntityPtr cur;
497
391k
    xmlEntitiesTablePtr table;
498
499
391k
    if (doc != NULL) {
500
383k
  if ((doc->intSubset != NULL) && (doc->intSubset->entities != NULL)) {
501
318k
      table = (xmlEntitiesTablePtr) doc->intSubset->entities;
502
318k
      cur = xmlGetEntityFromTable(table, name);
503
318k
      if (cur != NULL)
504
277k
    return(cur);
505
318k
  }
506
106k
  if (doc->standalone != 1) {
507
105k
      if ((doc->extSubset != NULL) &&
508
32.8k
    (doc->extSubset->entities != NULL)) {
509
31.9k
    table = (xmlEntitiesTablePtr) doc->extSubset->entities;
510
31.9k
    cur = xmlGetEntityFromTable(table, name);
511
31.9k
    if (cur != NULL)
512
308
        return(cur);
513
31.9k
      }
514
105k
  }
515
106k
    }
516
113k
    return(xmlGetPredefinedEntity(name));
517
391k
}
518
519
/*
520
 * xmlSerializeHexCharRef:
521
 * @buf:  a char buffer
522
 * @val:  a codepoint
523
 *
524
 * Serializes a hex char ref like &#xA0;
525
 *
526
 * Writes at most 9 bytes. Does not include a terminating zero byte.
527
 *
528
 * Returns the number of bytes written.
529
 */
530
int
531
3.52k
xmlSerializeHexCharRef(char *buf, int val) {
532
3.52k
    char *out = buf;
533
3.52k
    int shift = 0, bits;
534
535
3.52k
    *out++ = '&';
536
3.52k
    *out++ = '#';
537
3.52k
    *out++ = 'x';
538
539
3.52k
    bits = val;
540
3.52k
    if (bits & 0xFF0000) {
541
87
        shift = 16;
542
87
        bits &= 0xFF0000;
543
3.44k
    } else if (bits & 0x00FF00) {
544
2.83k
        shift = 8;
545
2.83k
        bits &= 0x00FF00;
546
2.83k
    }
547
3.52k
    if (bits & 0xF0F0F0) {
548
1.05k
        shift += 4;
549
1.05k
    }
550
551
10.6k
    do {
552
10.6k
        int d = (val >> shift) & 0x0F;
553
554
10.6k
        if (d < 10)
555
5.84k
            *out++ = '0' + d;
556
4.76k
        else
557
4.76k
            *out++ = 'A' + (d - 10);
558
559
10.6k
  shift -= 4;
560
10.6k
    } while (shift >= 0);
561
562
3.52k
    *out++ = ';';
563
564
3.52k
    return(out - buf);
565
3.52k
}
566
567
/*
568
 * xmlSerializeDecCharRef:
569
 * @buf:  a char buffer
570
 * @val:  a codepoint
571
 *
572
 * Serializes a decimal char ref like &#38;
573
 *
574
 * Writes at most 10 bytes. Does not include a terminating zero byte.
575
 *
576
 * Returns the number of bytes written.
577
 */
578
int
579
51.2M
xmlSerializeDecCharRef(char *buf, int val) {
580
51.2M
    char *out = buf;
581
51.2M
    int len, i;
582
583
51.2M
    *out++ = '&';
584
51.2M
    *out++ = '#';
585
586
51.2M
    if (val < 100) {
587
475
        len = (val < 10) ? 1 : 2;
588
51.2M
    } else if (val < 10000) {
589
51.2M
        len = (val < 1000) ? 3 : 4;
590
51.2M
    } else if (val < 1000000) {
591
7.60k
        len = (val < 100000) ? 5 : 6;
592
7.60k
    } else {
593
228
        len = 7;
594
228
    }
595
596
211M
    for (i = len - 1; i >= 0; i--) {
597
160M
        out[i] = '0' + val % 10;
598
160M
        val /= 10;
599
160M
    }
600
601
51.2M
    out[len] = ';';
602
603
51.2M
    return(len + 3);
604
51.2M
}
605
606
static const char xmlEscapeSafe[128] = {
607
    0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0,
608
    0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
609
    1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1,
610
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1,
611
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
612
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
613
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
614
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
615
};
616
617
/*
618
 * xmlEscapeText:
619
 * @text:  input text
620
 * @flags:  XML_ESCAPE flags
621
 *
622
 * Escapes certain characters with char refs.
623
 *
624
 * XML_ESCAPE_ATTR: for attribute content.
625
 * XML_ESCAPE_NON_ASCII: escape non-ASCII chars.
626
 * XML_ESCAPE_HTML: for HTML content.
627
 * XML_ESCAPE_QUOT: escape double quotes.
628
 * XML_ESCAPE_ALLOW_INVALID: allow invalid characters.
629
 *
630
 * Returns an escaped string or NULL if a memory allocation failed.
631
 */
632
xmlChar *
633
23.0k
xmlEscapeText(const xmlChar *text, int flags) {
634
23.0k
    const xmlChar *cur;
635
23.0k
    xmlChar *buffer;
636
23.0k
    xmlChar *out;
637
23.0k
    const xmlChar *unescaped;
638
23.0k
    size_t size = 50;
639
640
23.0k
    buffer = xmlMalloc(size + 1);
641
23.0k
    if (buffer == NULL)
642
55
        return(NULL);
643
22.9k
    out = buffer;
644
645
22.9k
    cur = text;
646
22.9k
    unescaped = cur;
647
648
212k
    while (*cur != '\0') {
649
189k
        char buf[12];
650
189k
  const xmlChar *end;
651
189k
        const xmlChar *repl;
652
189k
        size_t used;
653
189k
        size_t replSize;
654
189k
        size_t unescapedSize;
655
189k
        size_t totalSize;
656
189k
        int chunkSize = 1;
657
189k
        int c;
658
659
        /* accelerator */
660
138M
  while (1) {
661
138M
            c = *cur;
662
663
138M
            if (c < 0x80) {
664
45.7M
                if (!xmlEscapeSafe[*cur])
665
189k
                    break;
666
92.9M
            } else {
667
92.9M
               if (flags & XML_ESCAPE_NON_ASCII)
668
0
                   break;
669
92.9M
            }
670
138M
            cur += 1;
671
138M
        }
672
673
189k
        if (c == 0) {
674
21.0k
            chunkSize = 0;
675
21.0k
            repl = BAD_CAST "";
676
21.0k
            replSize = 0;
677
168k
        } else if (c == '<') {
678
      /*
679
       * Special handling of server side include in HTML attributes
680
       */
681
19.8k
      if ((flags & XML_ESCAPE_HTML) && (flags & XML_ESCAPE_ATTR) &&
682
15.7k
          (cur[1] == '!') && (cur[2] == '-') && (cur[3] == '-') &&
683
1.46k
          ((end = xmlStrstr(cur, BAD_CAST "-->")) != NULL)) {
684
1.22k
                chunkSize = (end - cur) + 3;
685
1.22k
                repl = cur;
686
1.22k
                replSize = chunkSize;
687
18.6k
      } else {
688
18.6k
                repl = BAD_CAST "&lt;";
689
18.6k
                replSize = 4;
690
18.6k
            }
691
148k
  } else if (c == '>') {
692
88.8k
            repl = BAD_CAST "&gt;";
693
88.8k
            replSize = 4;
694
88.8k
  } else if (c == '&') {
695
      /*
696
       * Special handling of &{...} construct from HTML 4, see
697
       * http://www.w3.org/TR/html401/appendix/notes.html#h-B.7.1
698
       */
699
10.5k
      if ((flags & XML_ESCAPE_HTML) && (flags & XML_ESCAPE_ATTR) &&
700
10.1k
                (cur[1] == '{') && (end = xmlStrchr(cur, '}'))) {
701
2.57k
                chunkSize = (end - cur) + 1;
702
2.57k
                repl = cur;
703
2.57k
                replSize = chunkSize;
704
8.01k
      } else {
705
8.01k
                repl = BAD_CAST "&amp;";
706
8.01k
                replSize = 5;
707
8.01k
            }
708
48.8k
  } else if ((flags & XML_ESCAPE_QUOT) && (c == '"')) {
709
0
            repl = BAD_CAST "&quot;";
710
0
            replSize = 6;
711
48.8k
  } else if (((flags & XML_ESCAPE_HTML) == 0) && (c == '\r')) {
712
0
      repl = BAD_CAST "&#13;";
713
0
            replSize = 5;
714
48.8k
  } else if ((flags & XML_ESCAPE_NON_ASCII) && (c >= 0x80)) {
715
0
            int val;
716
717
0
            chunkSize = 4;
718
0
            val = xmlGetUTF8Char(cur, &chunkSize);
719
0
            if (val < 0) {
720
0
                val = 0xFFFD;
721
0
                chunkSize = 1;
722
0
            } else if (((flags & XML_ESCAPE_ALLOW_INVALID) == 0) &&
723
0
                       (!IS_CHAR(val))) {
724
0
                val = 0xFFFD;
725
0
            }
726
727
0
            replSize = xmlSerializeHexCharRef(buf, val);
728
0
            repl = BAD_CAST buf;
729
48.8k
  } else if ((flags & (XML_ESCAPE_ALLOW_INVALID | XML_ESCAPE_HTML)) ||
730
0
                   (c >= 0x20) ||
731
48.8k
             (c == '\n') || (c == '\t') || (c == '\r')) {
732
      /* default case, just copy */
733
48.8k
            cur += 1;
734
48.8k
            if (*cur != 0)
735
48.0k
                continue;
736
737
790
            chunkSize = 0;
738
790
            repl = BAD_CAST "";
739
790
            replSize = 0;
740
790
  } else {
741
            /* ignore */
742
0
            repl = BAD_CAST "";
743
0
            replSize = 0;
744
0
        }
745
746
141k
        used = out - buffer;
747
141k
        unescapedSize = cur - unescaped;
748
141k
        totalSize = unescapedSize + replSize;
749
750
141k
  cur += chunkSize;
751
752
141k
        if (totalSize > size - used) {
753
9.24k
            xmlChar *tmp;
754
9.24k
            int newSize;
755
756
9.24k
            if ((size > (SIZE_MAX - 1) / 2) ||
757
9.24k
                (totalSize > (SIZE_MAX - 1) / 2 - size)) {
758
0
                xmlFree(buffer);
759
0
                return(NULL);
760
0
            }
761
9.24k
            newSize = size + totalSize;
762
9.24k
            if (*cur != 0)
763
6.14k
                newSize *= 2;
764
9.24k
            tmp = xmlRealloc(buffer, newSize + 1);
765
9.24k
            if (tmp == NULL) {
766
4
                xmlFree(buffer);
767
4
                return(NULL);
768
4
            }
769
9.23k
            buffer = tmp;
770
9.23k
            size = newSize;
771
9.23k
            out = buffer + used;
772
9.23k
        }
773
774
141k
        memcpy(out, unescaped, unescapedSize);
775
141k
        out += unescapedSize;
776
141k
        memcpy(out, repl, replSize);
777
141k
        out += replSize;
778
779
141k
        unescaped = cur;
780
141k
    }
781
782
22.9k
    *out = 0;
783
22.9k
    return(buffer);
784
22.9k
}
785
786
/**
787
 * xmlEncodeEntitiesInternal:
788
 * @doc:  the document containing the string
789
 * @input:  A string to convert to XML.
790
 * @flags:  XML_ESCAPE flags
791
 *
792
 * Do a global encoding of a string, replacing the predefined entities
793
 * and non ASCII values with their entities and CharRef counterparts.
794
 * Contrary to xmlEncodeEntities, this routine is reentrant, and result
795
 * must be deallocated.
796
 *
797
 * Returns A newly allocated string with the substitution done.
798
 */
799
xmlChar *
800
xmlEncodeEntitiesInternal(xmlDocPtr doc, const xmlChar *input,
801
23.0k
                          unsigned flags) {
802
23.0k
    if (input == NULL)
803
0
        return(NULL);
804
805
23.0k
    if ((doc != NULL) && (doc->type == XML_HTML_DOCUMENT_NODE))
806
23.0k
        flags |= XML_ESCAPE_HTML;
807
0
    else if ((doc == NULL) || (doc->encoding == NULL))
808
0
        flags |= XML_ESCAPE_NON_ASCII;
809
810
23.0k
    return(xmlEscapeText(input, flags));
811
23.0k
}
812
813
/**
814
 * xmlEncodeEntitiesReentrant:
815
 * @doc:  the document containing the string
816
 * @input:  A string to convert to XML.
817
 *
818
 * Do a global encoding of a string, replacing the predefined entities
819
 * and non ASCII values with their entities and CharRef counterparts.
820
 * Contrary to xmlEncodeEntities, this routine is reentrant, and result
821
 * must be deallocated.
822
 *
823
 * This escapes '<', '>', '&' and '\r'. If the document has no encoding,
824
 * non-ASCII codepoints are escaped. There is some special handling for
825
 * HTML documents.
826
 *
827
 * Returns A newly allocated string with the substitution done.
828
 */
829
xmlChar *
830
15.3k
xmlEncodeEntitiesReentrant(xmlDocPtr doc, const xmlChar *input) {
831
15.3k
    return xmlEncodeEntitiesInternal(doc, input, 0);
832
15.3k
}
833
834
/**
835
 * xmlEncodeSpecialChars:
836
 * @doc:  unused
837
 * @input:  A string to convert to XML.
838
 *
839
 * Do a global encoding of a string, replacing the predefined entities
840
 * this routine is reentrant, and result must be deallocated.
841
 *
842
 * This escapes '<', '>', '&', '"' and '\r' chars.
843
 *
844
 * Returns A newly allocated string with the substitution done.
845
 */
846
xmlChar *
847
xmlEncodeSpecialChars(const xmlDoc *doc ATTRIBUTE_UNUSED,
848
0
                      const xmlChar *input) {
849
0
    if (input == NULL)
850
0
        return(NULL);
851
852
0
    return(xmlEscapeText(input, XML_ESCAPE_QUOT | XML_ESCAPE_ALLOW_INVALID));
853
0
}
854
855
/**
856
 * xmlCreateEntitiesTable:
857
 *
858
 * create and initialize an empty entities hash table.
859
 * This really doesn't make sense and should be deprecated
860
 *
861
 * Returns the xmlEntitiesTablePtr just created or NULL in case of error.
862
 */
863
xmlEntitiesTablePtr
864
0
xmlCreateEntitiesTable(void) {
865
0
    return((xmlEntitiesTablePtr) xmlHashCreate(0));
866
0
}
867
868
/**
869
 * xmlFreeEntityWrapper:
870
 * @entity:  An entity
871
 * @name:  its name
872
 *
873
 * Deallocate the memory used by an entities in the hash table.
874
 */
875
static void
876
84.7k
xmlFreeEntityWrapper(void *entity, const xmlChar *name ATTRIBUTE_UNUSED) {
877
84.7k
    if (entity != NULL)
878
84.7k
  xmlFreeEntity((xmlEntityPtr) entity);
879
84.7k
}
880
881
/**
882
 * xmlFreeEntitiesTable:
883
 * @table:  An entity table
884
 *
885
 * Deallocate the memory used by an entities hash table.
886
 */
887
void
888
45.9k
xmlFreeEntitiesTable(xmlEntitiesTablePtr table) {
889
45.9k
    xmlHashFree(table, xmlFreeEntityWrapper);
890
45.9k
}
891
892
/**
893
 * xmlCopyEntity:
894
 * @ent:  An entity
895
 *
896
 * Build a copy of an entity
897
 *
898
 * Returns the new xmlEntitiesPtr or NULL in case of error.
899
 */
900
static void *
901
19.1k
xmlCopyEntity(void *payload, const xmlChar *name ATTRIBUTE_UNUSED) {
902
19.1k
    xmlEntityPtr ent = (xmlEntityPtr) payload;
903
19.1k
    xmlEntityPtr cur;
904
905
19.1k
    cur = (xmlEntityPtr) xmlMalloc(sizeof(xmlEntity));
906
19.1k
    if (cur == NULL)
907
5
  return(NULL);
908
19.1k
    memset(cur, 0, sizeof(xmlEntity));
909
19.1k
    cur->type = XML_ENTITY_DECL;
910
911
19.1k
    cur->etype = ent->etype;
912
19.1k
    if (ent->name != NULL) {
913
19.1k
  cur->name = xmlStrdup(ent->name);
914
19.1k
        if (cur->name == NULL)
915
4
            goto error;
916
19.1k
    }
917
19.1k
    if (ent->ExternalID != NULL) {
918
0
  cur->ExternalID = xmlStrdup(ent->ExternalID);
919
0
        if (cur->ExternalID == NULL)
920
0
            goto error;
921
0
    }
922
19.1k
    if (ent->SystemID != NULL) {
923
1.86k
  cur->SystemID = xmlStrdup(ent->SystemID);
924
1.86k
        if (cur->SystemID == NULL)
925
2
            goto error;
926
1.86k
    }
927
19.1k
    if (ent->content != NULL) {
928
17.2k
  cur->content = xmlStrdup(ent->content);
929
17.2k
        if (cur->content == NULL)
930
4
            goto error;
931
17.2k
    }
932
19.1k
    if (ent->orig != NULL) {
933
17.2k
  cur->orig = xmlStrdup(ent->orig);
934
17.2k
        if (cur->orig == NULL)
935
1
            goto error;
936
17.2k
    }
937
19.1k
    if (ent->URI != NULL) {
938
1.07k
  cur->URI = xmlStrdup(ent->URI);
939
1.07k
        if (cur->URI == NULL)
940
1
            goto error;
941
1.07k
    }
942
19.1k
    return(cur);
943
944
12
error:
945
12
    xmlFreeEntity(cur);
946
12
    return(NULL);
947
19.1k
}
948
949
/**
950
 * xmlCopyEntitiesTable:
951
 * @table:  An entity table
952
 *
953
 * Build a copy of an entity table.
954
 *
955
 * Returns the new xmlEntitiesTablePtr or NULL in case of error.
956
 */
957
xmlEntitiesTablePtr
958
6.80k
xmlCopyEntitiesTable(xmlEntitiesTablePtr table) {
959
6.80k
    return(xmlHashCopySafe(table, xmlCopyEntity, xmlFreeEntityWrapper));
960
6.80k
}
961
962
#ifdef LIBXML_OUTPUT_ENABLED
963
964
/**
965
 * xmlDumpEntityDecl:
966
 * @buf:  An XML buffer.
967
 * @ent:  An entity table
968
 *
969
 * This will dump the content of the entity table as an XML DTD definition
970
 */
971
void
972
0
xmlDumpEntityDecl(xmlBufferPtr buf, xmlEntityPtr ent) {
973
0
    xmlSaveCtxtPtr save;
974
975
0
    if ((buf == NULL) || (ent == NULL))
976
0
        return;
977
978
0
    save = xmlSaveToBuffer(buf, NULL, 0);
979
0
    xmlSaveTree(save, (xmlNodePtr) ent);
980
0
    if (xmlSaveFinish(save) != XML_ERR_OK)
981
0
        xmlFree(xmlBufferDetach(buf));
982
0
}
983
984
/**
985
 * xmlDumpEntityDeclScan:
986
 * @ent:  An entity table
987
 * @buf:  An XML buffer.
988
 *
989
 * When using the hash table scan function, arguments need to be reversed
990
 */
991
static void
992
xmlDumpEntityDeclScan(void *ent, void *save,
993
0
                      const xmlChar *name ATTRIBUTE_UNUSED) {
994
0
    xmlSaveTree(save, ent);
995
0
}
996
997
/**
998
 * xmlDumpEntitiesTable:
999
 * @buf:  An XML buffer.
1000
 * @table:  An entity table
1001
 *
1002
 * This will dump the content of the entity table as an XML DTD definition
1003
 */
1004
void
1005
0
xmlDumpEntitiesTable(xmlBufferPtr buf, xmlEntitiesTablePtr table) {
1006
0
    xmlSaveCtxtPtr save;
1007
1008
0
    if ((buf == NULL) || (table == NULL))
1009
0
        return;
1010
1011
0
    save = xmlSaveToBuffer(buf, NULL, 0);
1012
0
    xmlHashScan(table, xmlDumpEntityDeclScan, save);
1013
0
    if (xmlSaveFinish(save) != XML_ERR_OK)
1014
0
        xmlFree(xmlBufferDetach(buf));
1015
0
}
1016
#endif /* LIBXML_OUTPUT_ENABLED */