Coverage Report

Created: 2026-03-07 06:46

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
98.2k
{
85
98.2k
    xmlDictPtr dict = NULL;
86
87
98.2k
    if (entity == NULL)
88
0
        return;
89
90
98.2k
    if (entity->doc != NULL)
91
98.1k
        dict = entity->doc->dict;
92
93
94
98.2k
    if ((entity->children) &&
95
3.16k
        (entity == (xmlEntityPtr) entity->children->parent))
96
3.16k
        xmlFreeNodeList(entity->children);
97
98.2k
    if ((entity->name != NULL) &&
98
98.2k
        ((dict == NULL) || (!xmlDictOwns(dict, entity->name))))
99
22.8k
        xmlFree((char *) entity->name);
100
98.2k
    if (entity->ExternalID != NULL)
101
10.5k
        xmlFree((char *) entity->ExternalID);
102
98.2k
    if (entity->SystemID != NULL)
103
34.4k
        xmlFree((char *) entity->SystemID);
104
98.2k
    if (entity->URI != NULL)
105
32.7k
        xmlFree((char *) entity->URI);
106
98.2k
    if (entity->content != NULL)
107
62.3k
        xmlFree((char *) entity->content);
108
98.2k
    if (entity->orig != NULL)
109
50.8k
        xmlFree((char *) entity->orig);
110
98.2k
    xmlFree(entity);
111
98.2k
}
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
86.7k
          const xmlChar *content) {
122
86.7k
    xmlEntityPtr ret;
123
124
86.7k
    ret = (xmlEntityPtr) xmlMalloc(sizeof(xmlEntity));
125
86.7k
    if (ret == NULL)
126
12
  return(NULL);
127
86.7k
    memset(ret, 0, sizeof(xmlEntity));
128
86.7k
    ret->doc = doc;
129
86.7k
    ret->type = XML_ENTITY_DECL;
130
131
    /*
132
     * fill the structure.
133
     */
134
86.7k
    ret->etype = (xmlEntityType) type;
135
86.7k
    if ((doc == NULL) || (doc->dict == NULL))
136
11.3k
  ret->name = xmlStrdup(name);
137
75.3k
    else
138
75.3k
        ret->name = xmlDictLookup(doc->dict, name, -1);
139
86.7k
    if (ret->name == NULL)
140
2
        goto error;
141
86.7k
    if (ExternalID != NULL) {
142
10.5k
        ret->ExternalID = xmlStrdup(ExternalID);
143
10.5k
        if (ret->ExternalID == NULL)
144
1
            goto error;
145
10.5k
    }
146
86.7k
    if (SystemID != NULL) {
147
33.9k
        ret->SystemID = xmlStrdup(SystemID);
148
33.9k
        if (ret->SystemID == NULL)
149
5
            goto error;
150
33.9k
    }
151
86.7k
    if (content != NULL) {
152
50.4k
        ret->length = xmlStrlen(content);
153
50.4k
  ret->content = xmlStrndup(content, ret->length);
154
50.4k
        if (ret->content == NULL)
155
6
            goto error;
156
50.4k
     } else {
157
36.2k
        ret->length = 0;
158
36.2k
        ret->content = NULL;
159
36.2k
    }
160
86.6k
    ret->URI = NULL; /* to be computed by the layer knowing
161
      the defining entity */
162
86.6k
    ret->orig = NULL;
163
164
86.6k
    return(ret);
165
166
14
error:
167
14
    xmlFreeEntity(ret);
168
14
    return(NULL);
169
86.7k
}
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
96.0k
    const xmlChar *content, xmlEntityPtr *out) {
192
96.0k
    xmlDtdPtr dtd;
193
96.0k
    xmlDictPtr dict = NULL;
194
96.0k
    xmlEntitiesTablePtr table = NULL;
195
96.0k
    xmlEntityPtr ret, predef;
196
96.0k
    int res;
197
198
96.0k
    if (out != NULL)
199
96.0k
        *out = NULL;
200
96.0k
    if ((doc == NULL) || (name == NULL))
201
0
  return(XML_ERR_ARGUMENT);
202
96.0k
    dict = doc->dict;
203
204
96.0k
    if (extSubset)
205
9.02k
        dtd = doc->extSubset;
206
87.0k
    else
207
87.0k
        dtd = doc->intSubset;
208
96.0k
    if (dtd == NULL)
209
0
        return(XML_DTD_NO_DTD);
210
211
96.0k
    switch (type) {
212
39.0k
        case XML_INTERNAL_GENERAL_ENTITY:
213
52.3k
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
214
52.6k
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
215
52.6k
            predef = xmlGetPredefinedEntity(name);
216
52.6k
            if (predef != NULL) {
217
12.5k
                int valid = 0;
218
219
                /* 4.6 Predefined Entities */
220
12.5k
                if ((type == XML_INTERNAL_GENERAL_ENTITY) &&
221
11.3k
                    (content != NULL)) {
222
11.0k
                    int c = predef->content[0];
223
224
11.0k
                    if (((content[0] == c) && (content[1] == 0)) &&
225
1.43k
                        ((c == '>') || (c == '\'') || (c == '"'))) {
226
1.20k
                        valid = 1;
227
9.89k
                    } else if ((content[0] == '&') && (content[1] == '#')) {
228
5.27k
                        if (content[2] == 'x') {
229
4.73k
                            xmlChar *hex = BAD_CAST "0123456789ABCDEF";
230
4.73k
                            xmlChar ref[] = "00;";
231
232
4.73k
                            ref[0] = hex[c / 16 % 16];
233
4.73k
                            ref[1] = hex[c % 16];
234
4.73k
                            if (xmlStrcasecmp(&content[3], ref) == 0)
235
1.87k
                                valid = 1;
236
4.73k
                        } else {
237
535
                            xmlChar ref[] = "00;";
238
239
535
                            ref[0] = '0' + c / 10 % 10;
240
535
                            ref[1] = '0' + c % 10;
241
535
                            if (xmlStrEqual(&content[2], ref))
242
145
                                valid = 1;
243
535
                        }
244
5.27k
                    }
245
11.0k
                }
246
12.5k
                if (!valid)
247
9.34k
                    return(XML_ERR_REDECL_PREDEF_ENTITY);
248
12.5k
            }
249
43.3k
      if (dtd->entities == NULL) {
250
25.2k
    dtd->entities = xmlHashCreateDict(0, dict);
251
25.2k
                if (dtd->entities == NULL)
252
6
                    return(XML_ERR_NO_MEMORY);
253
25.2k
            }
254
43.3k
      table = dtd->entities;
255
43.3k
      break;
256
19.8k
        case XML_INTERNAL_PARAMETER_ENTITY:
257
43.4k
        case XML_EXTERNAL_PARAMETER_ENTITY:
258
43.4k
      if (dtd->pentities == NULL) {
259
20.6k
    dtd->pentities = xmlHashCreateDict(0, dict);
260
20.6k
                if (dtd->pentities == NULL)
261
5
                    return(XML_ERR_NO_MEMORY);
262
20.6k
            }
263
43.4k
      table = dtd->pentities;
264
43.4k
      break;
265
0
        default:
266
0
      return(XML_ERR_ARGUMENT);
267
96.0k
    }
268
86.7k
    ret = xmlCreateEntity(dtd->doc, name, type, ExternalID, SystemID, content);
269
86.7k
    if (ret == NULL)
270
26
        return(XML_ERR_NO_MEMORY);
271
272
86.6k
    res = xmlHashAdd(table, name, ret);
273
86.6k
    if (res < 0) {
274
10
        xmlFreeEntity(ret);
275
10
        return(XML_ERR_NO_MEMORY);
276
86.6k
    } else if (res == 0) {
277
  /*
278
   * entity was already defined at another level.
279
   */
280
11.8k
        xmlFreeEntity(ret);
281
11.8k
  return(XML_WAR_ENTITY_REDEFINED);
282
11.8k
    }
283
284
    /*
285
     * Link it to the DTD
286
     */
287
74.8k
    ret->parent = dtd;
288
74.8k
    ret->doc = dtd->doc;
289
74.8k
    if (dtd->last == NULL) {
290
34.9k
  dtd->children = dtd->last = (xmlNodePtr) ret;
291
39.9k
    } else {
292
39.9k
  dtd->last->next = (xmlNodePtr) ret;
293
39.9k
  ret->prev = dtd->last;
294
39.9k
  dtd->last = (xmlNodePtr) ret;
295
39.9k
    }
296
297
74.8k
    if (out != NULL)
298
74.8k
        *out = ret;
299
74.8k
    return(0);
300
86.6k
}
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
2.42M
xmlGetPredefinedEntity(const xmlChar *name) {
312
2.42M
    if (name == NULL) return(NULL);
313
2.42M
    switch (name[0]) {
314
14.9k
        case 'l':
315
14.9k
      if (xmlStrEqual(name, BAD_CAST "lt"))
316
10.0k
          return(&xmlEntityLt);
317
4.95k
      break;
318
49.4k
        case 'g':
319
49.4k
      if (xmlStrEqual(name, BAD_CAST "gt"))
320
41.6k
          return(&xmlEntityGt);
321
7.83k
      break;
322
1.06M
        case 'a':
323
1.06M
      if (xmlStrEqual(name, BAD_CAST "amp"))
324
74.1k
          return(&xmlEntityAmp);
325
995k
      if (xmlStrEqual(name, BAD_CAST "apos"))
326
3.77k
          return(&xmlEntityApos);
327
992k
      break;
328
992k
        case 'q':
329
415k
      if (xmlStrEqual(name, BAD_CAST "quot"))
330
5.49k
          return(&xmlEntityQuot);
331
409k
      break;
332
871k
  default:
333
871k
      break;
334
2.42M
    }
335
2.28M
    return(NULL);
336
2.42M
}
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
1.15M
xmlGetEntityFromTable(xmlEntitiesTablePtr table, const xmlChar *name) {
426
1.15M
    return((xmlEntityPtr) xmlHashLookup(table, name));
427
1.15M
}
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
102k
xmlGetParameterEntity(xmlDocPtr doc, const xmlChar *name) {
441
102k
    xmlEntitiesTablePtr table;
442
102k
    xmlEntityPtr ret;
443
444
102k
    if (doc == NULL)
445
1.05k
  return(NULL);
446
101k
    if ((doc->intSubset != NULL) && (doc->intSubset->pentities != NULL)) {
447
98.5k
  table = (xmlEntitiesTablePtr) doc->intSubset->pentities;
448
98.5k
  ret = xmlGetEntityFromTable(table, name);
449
98.5k
  if (ret != NULL)
450
74.0k
      return(ret);
451
98.5k
    }
452
27.6k
    if ((doc->extSubset != NULL) && (doc->extSubset->pentities != NULL)) {
453
8.94k
  table = (xmlEntitiesTablePtr) doc->extSubset->pentities;
454
8.94k
  return(xmlGetEntityFromTable(table, name));
455
8.94k
    }
456
18.6k
    return(NULL);
457
27.6k
}
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
1.09M
xmlGetDocEntity(const xmlDoc *doc, const xmlChar *name) {
496
1.09M
    xmlEntityPtr cur;
497
1.09M
    xmlEntitiesTablePtr table;
498
499
1.09M
    if (doc != NULL) {
500
1.08M
  if ((doc->intSubset != NULL) && (doc->intSubset->entities != NULL)) {
501
989k
      table = (xmlEntitiesTablePtr) doc->intSubset->entities;
502
989k
      cur = xmlGetEntityFromTable(table, name);
503
989k
      if (cur != NULL)
504
879k
    return(cur);
505
989k
  }
506
202k
  if (doc->standalone != 1) {
507
201k
      if ((doc->extSubset != NULL) &&
508
60.4k
    (doc->extSubset->entities != NULL)) {
509
58.6k
    table = (xmlEntitiesTablePtr) doc->extSubset->entities;
510
58.6k
    cur = xmlGetEntityFromTable(table, name);
511
58.6k
    if (cur != NULL)
512
83
        return(cur);
513
58.6k
      }
514
201k
  }
515
202k
    }
516
218k
    return(xmlGetPredefinedEntity(name));
517
1.09M
}
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
2.14k
xmlSerializeHexCharRef(char *buf, int val) {
532
2.14k
    char *out = buf;
533
2.14k
    int shift = 0, bits;
534
535
2.14k
    *out++ = '&';
536
2.14k
    *out++ = '#';
537
2.14k
    *out++ = 'x';
538
539
2.14k
    bits = val;
540
2.14k
    if (bits & 0xFF0000) {
541
311
        shift = 16;
542
311
        bits &= 0xFF0000;
543
1.82k
    } else if (bits & 0x00FF00) {
544
1.41k
        shift = 8;
545
1.41k
        bits &= 0x00FF00;
546
1.41k
    }
547
2.14k
    if (bits & 0xF0F0F0) {
548
1.73k
        shift += 4;
549
1.73k
    }
550
551
7.94k
    do {
552
7.94k
        int d = (val >> shift) & 0x0F;
553
554
7.94k
        if (d < 10)
555
1.41k
            *out++ = '0' + d;
556
6.53k
        else
557
6.53k
            *out++ = 'A' + (d - 10);
558
559
7.94k
  shift -= 4;
560
7.94k
    } while (shift >= 0);
561
562
2.14k
    *out++ = ';';
563
564
2.14k
    return(out - buf);
565
2.14k
}
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
78.7M
xmlSerializeDecCharRef(char *buf, int val) {
580
78.7M
    char *out = buf;
581
78.7M
    int len, i;
582
583
78.7M
    *out++ = '&';
584
78.7M
    *out++ = '#';
585
586
78.7M
    if (val < 100) {
587
789
        len = (val < 10) ? 1 : 2;
588
78.7M
    } else if (val < 10000) {
589
78.6M
        len = (val < 1000) ? 3 : 4;
590
78.6M
    } else if (val < 1000000) {
591
10.4k
        len = (val < 100000) ? 5 : 6;
592
10.4k
    } else {
593
251
        len = 7;
594
251
    }
595
596
319M
    for (i = len - 1; i >= 0; i--) {
597
241M
        out[i] = '0' + val % 10;
598
241M
        val /= 10;
599
241M
    }
600
601
78.7M
    out[len] = ';';
602
603
78.7M
    return(len + 3);
604
78.7M
}
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
24.8k
xmlEscapeText(const xmlChar *text, int flags) {
634
24.8k
    const xmlChar *cur;
635
24.8k
    xmlChar *buffer;
636
24.8k
    xmlChar *out;
637
24.8k
    const xmlChar *unescaped;
638
24.8k
    size_t size = 50;
639
640
24.8k
    buffer = xmlMalloc(size + 1);
641
24.8k
    if (buffer == NULL)
642
67
        return(NULL);
643
24.7k
    out = buffer;
644
645
24.7k
    cur = text;
646
24.7k
    unescaped = cur;
647
648
395k
    while (*cur != '\0') {
649
371k
        char buf[12];
650
371k
  const xmlChar *end;
651
371k
        const xmlChar *repl;
652
371k
        size_t used;
653
371k
        size_t replSize;
654
371k
        size_t unescapedSize;
655
371k
        size_t totalSize;
656
371k
        int chunkSize = 1;
657
371k
        int c;
658
659
        /* accelerator */
660
134M
  while (1) {
661
134M
            c = *cur;
662
663
134M
            if (c < 0x80) {
664
42.6M
                if (!xmlEscapeSafe[*cur])
665
371k
                    break;
666
91.3M
            } else {
667
91.3M
               if (flags & XML_ESCAPE_NON_ASCII)
668
0
                   break;
669
91.3M
            }
670
133M
            cur += 1;
671
133M
        }
672
673
371k
        if (c == 0) {
674
22.8k
            chunkSize = 0;
675
22.8k
            repl = BAD_CAST "";
676
22.8k
            replSize = 0;
677
348k
        } else if (c == '<') {
678
      /*
679
       * Special handling of server side include in HTML attributes
680
       */
681
18.1k
      if ((flags & XML_ESCAPE_HTML) && (flags & XML_ESCAPE_ATTR) &&
682
17.7k
          (cur[1] == '!') && (cur[2] == '-') && (cur[3] == '-') &&
683
1.24k
          ((end = xmlStrstr(cur, BAD_CAST "-->")) != NULL)) {
684
1.06k
                chunkSize = (end - cur) + 3;
685
1.06k
                repl = cur;
686
1.06k
                replSize = chunkSize;
687
17.0k
      } else {
688
17.0k
                repl = BAD_CAST "&lt;";
689
17.0k
                replSize = 4;
690
17.0k
            }
691
329k
  } else if (c == '>') {
692
273k
            repl = BAD_CAST "&gt;";
693
273k
            replSize = 4;
694
273k
  } 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.3k
      if ((flags & XML_ESCAPE_HTML) && (flags & XML_ESCAPE_ATTR) &&
700
10.2k
                (cur[1] == '{') && (end = xmlStrchr(cur, '}'))) {
701
2.27k
                chunkSize = (end - cur) + 1;
702
2.27k
                repl = cur;
703
2.27k
                replSize = chunkSize;
704
8.06k
      } else {
705
8.06k
                repl = BAD_CAST "&amp;";
706
8.06k
                replSize = 5;
707
8.06k
            }
708
46.0k
  } else if ((flags & XML_ESCAPE_QUOT) && (c == '"')) {
709
0
            repl = BAD_CAST "&quot;";
710
0
            replSize = 6;
711
46.0k
  } else if (((flags & XML_ESCAPE_HTML) == 0) && (c == '\r')) {
712
0
      repl = BAD_CAST "&#13;";
713
0
            replSize = 5;
714
46.0k
  } 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
46.0k
  } else if ((flags & (XML_ESCAPE_ALLOW_INVALID | XML_ESCAPE_HTML)) ||
730
0
                   (c >= 0x20) ||
731
46.0k
             (c == '\n') || (c == '\t') || (c == '\r')) {
732
      /* default case, just copy */
733
46.0k
            cur += 1;
734
46.0k
            if (*cur != 0)
735
45.3k
                continue;
736
737
712
            chunkSize = 0;
738
712
            repl = BAD_CAST "";
739
712
            replSize = 0;
740
712
  } else {
741
            /* ignore */
742
0
            repl = BAD_CAST "";
743
0
            replSize = 0;
744
0
        }
745
746
325k
        used = out - buffer;
747
325k
        unescapedSize = cur - unescaped;
748
325k
        totalSize = unescapedSize + replSize;
749
750
325k
  cur += chunkSize;
751
752
325k
        if (totalSize > size - used) {
753
8.25k
            xmlChar *tmp;
754
8.25k
            int newSize;
755
756
8.25k
            if ((size > (SIZE_MAX - 1) / 2) ||
757
8.25k
                (totalSize > (SIZE_MAX - 1) / 2 - size)) {
758
0
                xmlFree(buffer);
759
0
                return(NULL);
760
0
            }
761
8.25k
            newSize = size + totalSize;
762
8.25k
            if (*cur != 0)
763
5.05k
                newSize *= 2;
764
8.25k
            tmp = xmlRealloc(buffer, newSize + 1);
765
8.25k
            if (tmp == NULL) {
766
5
                xmlFree(buffer);
767
5
                return(NULL);
768
5
            }
769
8.24k
            buffer = tmp;
770
8.24k
            size = newSize;
771
8.24k
            out = buffer + used;
772
8.24k
        }
773
774
325k
        memcpy(out, unescaped, unescapedSize);
775
325k
        out += unescapedSize;
776
325k
        memcpy(out, repl, replSize);
777
325k
        out += replSize;
778
779
325k
        unescaped = cur;
780
325k
    }
781
782
24.7k
    *out = 0;
783
24.7k
    return(buffer);
784
24.7k
}
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
24.8k
                          unsigned flags) {
802
24.8k
    if (input == NULL)
803
0
        return(NULL);
804
805
24.8k
    if ((doc != NULL) && (doc->type == XML_HTML_DOCUMENT_NODE))
806
24.8k
        flags |= XML_ESCAPE_HTML;
807
0
    else if ((doc == NULL) || (doc->encoding == NULL))
808
0
        flags |= XML_ESCAPE_NON_ASCII;
809
810
24.8k
    return(xmlEscapeText(input, flags));
811
24.8k
}
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.7k
xmlEncodeEntitiesReentrant(xmlDocPtr doc, const xmlChar *input) {
831
15.7k
    return xmlEncodeEntitiesInternal(doc, input, 0);
832
15.7k
}
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
86.4k
xmlFreeEntityWrapper(void *entity, const xmlChar *name ATTRIBUTE_UNUSED) {
877
86.4k
    if (entity != NULL)
878
86.4k
  xmlFreeEntity((xmlEntityPtr) entity);
879
86.4k
}
880
881
/**
882
 * xmlFreeEntitiesTable:
883
 * @table:  An entity table
884
 *
885
 * Deallocate the memory used by an entities hash table.
886
 */
887
void
888
50.4k
xmlFreeEntitiesTable(xmlEntitiesTablePtr table) {
889
50.4k
    xmlHashFree(table, xmlFreeEntityWrapper);
890
50.4k
}
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
11.5k
xmlCopyEntity(void *payload, const xmlChar *name ATTRIBUTE_UNUSED) {
902
11.5k
    xmlEntityPtr ent = (xmlEntityPtr) payload;
903
11.5k
    xmlEntityPtr cur;
904
905
11.5k
    cur = (xmlEntityPtr) xmlMalloc(sizeof(xmlEntity));
906
11.5k
    if (cur == NULL)
907
4
  return(NULL);
908
11.5k
    memset(cur, 0, sizeof(xmlEntity));
909
11.5k
    cur->type = XML_ENTITY_DECL;
910
911
11.5k
    cur->etype = ent->etype;
912
11.5k
    if (ent->name != NULL) {
913
11.5k
  cur->name = xmlStrdup(ent->name);
914
11.5k
        if (cur->name == NULL)
915
2
            goto error;
916
11.5k
    }
917
11.5k
    if (ent->ExternalID != NULL) {
918
0
  cur->ExternalID = xmlStrdup(ent->ExternalID);
919
0
        if (cur->ExternalID == NULL)
920
0
            goto error;
921
0
    }
922
11.5k
    if (ent->SystemID != NULL) {
923
497
  cur->SystemID = xmlStrdup(ent->SystemID);
924
497
        if (cur->SystemID == NULL)
925
1
            goto error;
926
497
    }
927
11.5k
    if (ent->content != NULL) {
928
11.0k
  cur->content = xmlStrdup(ent->content);
929
11.0k
        if (cur->content == NULL)
930
4
            goto error;
931
11.0k
    }
932
11.5k
    if (ent->orig != NULL) {
933
11.0k
  cur->orig = xmlStrdup(ent->orig);
934
11.0k
        if (cur->orig == NULL)
935
3
            goto error;
936
11.0k
    }
937
11.5k
    if (ent->URI != NULL) {
938
460
  cur->URI = xmlStrdup(ent->URI);
939
460
        if (cur->URI == NULL)
940
2
            goto error;
941
460
    }
942
11.5k
    return(cur);
943
944
12
error:
945
12
    xmlFreeEntity(cur);
946
12
    return(NULL);
947
11.5k
}
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
4.66k
xmlCopyEntitiesTable(xmlEntitiesTablePtr table) {
959
4.66k
    return(xmlHashCopySafe(table, xmlCopyEntity, xmlFreeEntityWrapper));
960
4.66k
}
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 */