Coverage Report

Created: 2026-01-10 06:08

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
86.2k
{
85
86.2k
    xmlDictPtr dict = NULL;
86
87
86.2k
    if (entity == NULL)
88
0
        return;
89
90
86.2k
    if (entity->doc != NULL)
91
86.2k
        dict = entity->doc->dict;
92
93
94
86.2k
    if ((entity->children) &&
95
963
        (entity == (xmlEntityPtr) entity->children->parent))
96
963
        xmlFreeNodeList(entity->children);
97
86.2k
    if ((entity->name != NULL) &&
98
86.2k
        ((dict == NULL) || (!xmlDictOwns(dict, entity->name))))
99
24.3k
        xmlFree((char *) entity->name);
100
86.2k
    if (entity->ExternalID != NULL)
101
5.93k
        xmlFree((char *) entity->ExternalID);
102
86.2k
    if (entity->SystemID != NULL)
103
32.5k
        xmlFree((char *) entity->SystemID);
104
86.2k
    if (entity->URI != NULL)
105
29.9k
        xmlFree((char *) entity->URI);
106
86.2k
    if (entity->content != NULL)
107
53.7k
        xmlFree((char *) entity->content);
108
86.2k
    if (entity->orig != NULL)
109
46.3k
        xmlFree((char *) entity->orig);
110
86.2k
    xmlFree(entity);
111
86.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
68.8k
          const xmlChar *content) {
122
68.8k
    xmlEntityPtr ret;
123
124
68.8k
    ret = (xmlEntityPtr) xmlMalloc(sizeof(xmlEntity));
125
68.8k
    if (ret == NULL)
126
10
  return(NULL);
127
68.8k
    memset(ret, 0, sizeof(xmlEntity));
128
68.8k
    ret->doc = doc;
129
68.8k
    ret->type = XML_ENTITY_DECL;
130
131
    /*
132
     * fill the structure.
133
     */
134
68.8k
    ret->etype = (xmlEntityType) type;
135
68.8k
    if ((doc == NULL) || (doc->dict == NULL))
136
6.94k
  ret->name = xmlStrdup(name);
137
61.8k
    else
138
61.8k
        ret->name = xmlDictLookup(doc->dict, name, -1);
139
68.8k
    if (ret->name == NULL)
140
1
        goto error;
141
68.8k
    if (ExternalID != NULL) {
142
5.93k
        ret->ExternalID = xmlStrdup(ExternalID);
143
5.93k
        if (ret->ExternalID == NULL)
144
1
            goto error;
145
5.93k
    }
146
68.8k
    if (SystemID != NULL) {
147
31.6k
        ret->SystemID = xmlStrdup(SystemID);
148
31.6k
        if (ret->SystemID == NULL)
149
3
            goto error;
150
31.6k
    }
151
68.8k
    if (content != NULL) {
152
36.1k
        ret->length = xmlStrlen(content);
153
36.1k
  ret->content = xmlStrndup(content, ret->length);
154
36.1k
        if (ret->content == NULL)
155
7
            goto error;
156
36.1k
     } else {
157
32.6k
        ret->length = 0;
158
32.6k
        ret->content = NULL;
159
32.6k
    }
160
68.8k
    ret->URI = NULL; /* to be computed by the layer knowing
161
      the defining entity */
162
68.8k
    ret->orig = NULL;
163
164
68.8k
    return(ret);
165
166
12
error:
167
12
    xmlFreeEntity(ret);
168
12
    return(NULL);
169
68.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
75.7k
    const xmlChar *content, xmlEntityPtr *out) {
192
75.7k
    xmlDtdPtr dtd;
193
75.7k
    xmlDictPtr dict = NULL;
194
75.7k
    xmlEntitiesTablePtr table = NULL;
195
75.7k
    xmlEntityPtr ret, predef;
196
75.7k
    int res;
197
198
75.7k
    if (out != NULL)
199
75.7k
        *out = NULL;
200
75.7k
    if ((doc == NULL) || (name == NULL))
201
0
  return(XML_ERR_ARGUMENT);
202
75.7k
    dict = doc->dict;
203
204
75.7k
    if (extSubset)
205
5.88k
        dtd = doc->extSubset;
206
69.8k
    else
207
69.8k
        dtd = doc->intSubset;
208
75.7k
    if (dtd == NULL)
209
0
        return(XML_DTD_NO_DTD);
210
211
75.7k
    switch (type) {
212
25.8k
        case XML_INTERNAL_GENERAL_ENTITY:
213
33.2k
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
214
33.5k
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
215
33.5k
            predef = xmlGetPredefinedEntity(name);
216
33.5k
            if (predef != NULL) {
217
9.37k
                int valid = 0;
218
219
                /* 4.6 Predefined Entities */
220
9.37k
                if ((type == XML_INTERNAL_GENERAL_ENTITY) &&
221
8.37k
                    (content != NULL)) {
222
8.33k
                    int c = predef->content[0];
223
224
8.33k
                    if (((content[0] == c) && (content[1] == 0)) &&
225
688
                        ((c == '>') || (c == '\'') || (c == '"'))) {
226
612
                        valid = 1;
227
7.72k
                    } else if ((content[0] == '&') && (content[1] == '#')) {
228
4.76k
                        if (content[2] == 'x') {
229
4.22k
                            xmlChar *hex = BAD_CAST "0123456789ABCDEF";
230
4.22k
                            xmlChar ref[] = "00;";
231
232
4.22k
                            ref[0] = hex[c / 16 % 16];
233
4.22k
                            ref[1] = hex[c % 16];
234
4.22k
                            if (xmlStrcasecmp(&content[3], ref) == 0)
235
1.65k
                                valid = 1;
236
4.22k
                        } else {
237
547
                            xmlChar ref[] = "00;";
238
239
547
                            ref[0] = '0' + c / 10 % 10;
240
547
                            ref[1] = '0' + c % 10;
241
547
                            if (xmlStrEqual(&content[2], ref))
242
266
                                valid = 1;
243
547
                        }
244
4.76k
                    }
245
8.33k
                }
246
9.37k
                if (!valid)
247
6.84k
                    return(XML_ERR_REDECL_PREDEF_ENTITY);
248
9.37k
            }
249
26.7k
      if (dtd->entities == NULL) {
250
15.1k
    dtd->entities = xmlHashCreateDict(0, dict);
251
15.1k
                if (dtd->entities == NULL)
252
3
                    return(XML_ERR_NO_MEMORY);
253
15.1k
            }
254
26.7k
      table = dtd->entities;
255
26.7k
      break;
256
16.3k
        case XML_INTERNAL_PARAMETER_ENTITY:
257
42.1k
        case XML_EXTERNAL_PARAMETER_ENTITY:
258
42.1k
      if (dtd->pentities == NULL) {
259
18.2k
    dtd->pentities = xmlHashCreateDict(0, dict);
260
18.2k
                if (dtd->pentities == NULL)
261
1
                    return(XML_ERR_NO_MEMORY);
262
18.2k
            }
263
42.1k
      table = dtd->pentities;
264
42.1k
      break;
265
0
        default:
266
0
      return(XML_ERR_ARGUMENT);
267
75.7k
    }
268
68.8k
    ret = xmlCreateEntity(dtd->doc, name, type, ExternalID, SystemID, content);
269
68.8k
    if (ret == NULL)
270
22
        return(XML_ERR_NO_MEMORY);
271
272
68.8k
    res = xmlHashAdd(table, name, ret);
273
68.8k
    if (res < 0) {
274
5
        xmlFreeEntity(ret);
275
5
        return(XML_ERR_NO_MEMORY);
276
68.8k
    } else if (res == 0) {
277
  /*
278
   * entity was already defined at another level.
279
   */
280
8.04k
        xmlFreeEntity(ret);
281
8.04k
  return(XML_WAR_ENTITY_REDEFINED);
282
8.04k
    }
283
284
    /*
285
     * Link it to the DTD
286
     */
287
60.7k
    ret->parent = dtd;
288
60.7k
    ret->doc = dtd->doc;
289
60.7k
    if (dtd->last == NULL) {
290
22.1k
  dtd->children = dtd->last = (xmlNodePtr) ret;
291
38.5k
    } else {
292
38.5k
  dtd->last->next = (xmlNodePtr) ret;
293
38.5k
  ret->prev = dtd->last;
294
38.5k
  dtd->last = (xmlNodePtr) ret;
295
38.5k
    }
296
297
60.7k
    if (out != NULL)
298
60.7k
        *out = ret;
299
60.7k
    return(0);
300
68.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
1.24M
xmlGetPredefinedEntity(const xmlChar *name) {
312
1.24M
    if (name == NULL) return(NULL);
313
1.24M
    switch (name[0]) {
314
17.6k
        case 'l':
315
17.6k
      if (xmlStrEqual(name, BAD_CAST "lt"))
316
9.21k
          return(&xmlEntityLt);
317
8.43k
      break;
318
41.9k
        case 'g':
319
41.9k
      if (xmlStrEqual(name, BAD_CAST "gt"))
320
40.1k
          return(&xmlEntityGt);
321
1.78k
      break;
322
471k
        case 'a':
323
471k
      if (xmlStrEqual(name, BAD_CAST "amp"))
324
68.9k
          return(&xmlEntityAmp);
325
402k
      if (xmlStrEqual(name, BAD_CAST "apos"))
326
2.47k
          return(&xmlEntityApos);
327
400k
      break;
328
400k
        case 'q':
329
363k
      if (xmlStrEqual(name, BAD_CAST "quot"))
330
3.62k
          return(&xmlEntityQuot);
331
360k
      break;
332
360k
  default:
333
348k
      break;
334
1.24M
    }
335
1.11M
    return(NULL);
336
1.24M
}
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
550k
xmlGetEntityFromTable(xmlEntitiesTablePtr table, const xmlChar *name) {
426
550k
    return((xmlEntityPtr) xmlHashLookup(table, name));
427
550k
}
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
78.1k
xmlGetParameterEntity(xmlDocPtr doc, const xmlChar *name) {
441
78.1k
    xmlEntitiesTablePtr table;
442
78.1k
    xmlEntityPtr ret;
443
444
78.1k
    if (doc == NULL)
445
593
  return(NULL);
446
77.5k
    if ((doc->intSubset != NULL) && (doc->intSubset->pentities != NULL)) {
447
74.2k
  table = (xmlEntitiesTablePtr) doc->intSubset->pentities;
448
74.2k
  ret = xmlGetEntityFromTable(table, name);
449
74.2k
  if (ret != NULL)
450
54.3k
      return(ret);
451
74.2k
    }
452
23.1k
    if ((doc->extSubset != NULL) && (doc->extSubset->pentities != NULL)) {
453
6.01k
  table = (xmlEntitiesTablePtr) doc->extSubset->pentities;
454
6.01k
  return(xmlGetEntityFromTable(table, name));
455
6.01k
    }
456
17.1k
    return(NULL);
457
23.1k
}
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
525k
xmlGetDocEntity(const xmlDoc *doc, const xmlChar *name) {
496
525k
    xmlEntityPtr cur;
497
525k
    xmlEntitiesTablePtr table;
498
499
525k
    if (doc != NULL) {
500
523k
  if ((doc->intSubset != NULL) && (doc->intSubset->entities != NULL)) {
501
430k
      table = (xmlEntitiesTablePtr) doc->intSubset->entities;
502
430k
      cur = xmlGetEntityFromTable(table, name);
503
430k
      if (cur != NULL)
504
336k
    return(cur);
505
430k
  }
506
186k
  if (doc->standalone != 1) {
507
186k
      if ((doc->extSubset != NULL) &&
508
40.9k
    (doc->extSubset->entities != NULL)) {
509
40.2k
    table = (xmlEntitiesTablePtr) doc->extSubset->entities;
510
40.2k
    cur = xmlGetEntityFromTable(table, name);
511
40.2k
    if (cur != NULL)
512
47
        return(cur);
513
40.2k
      }
514
186k
  }
515
186k
    }
516
188k
    return(xmlGetPredefinedEntity(name));
517
525k
}
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
799
xmlSerializeHexCharRef(char *buf, int val) {
532
799
    char *out = buf;
533
799
    int shift = 0, bits;
534
535
799
    *out++ = '&';
536
799
    *out++ = '#';
537
799
    *out++ = 'x';
538
539
799
    bits = val;
540
799
    if (bits & 0xFF0000) {
541
23
        shift = 16;
542
23
        bits &= 0xFF0000;
543
776
    } else if (bits & 0x00FF00) {
544
539
        shift = 8;
545
539
        bits &= 0x00FF00;
546
539
    }
547
799
    if (bits & 0xF0F0F0) {
548
763
        shift += 4;
549
763
    }
550
551
2.73k
    do {
552
2.73k
        int d = (val >> shift) & 0x0F;
553
554
2.73k
        if (d < 10)
555
218
            *out++ = '0' + d;
556
2.51k
        else
557
2.51k
            *out++ = 'A' + (d - 10);
558
559
2.73k
  shift -= 4;
560
2.73k
    } while (shift >= 0);
561
562
799
    *out++ = ';';
563
564
799
    return(out - buf);
565
799
}
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
62.3M
xmlSerializeDecCharRef(char *buf, int val) {
580
62.3M
    char *out = buf;
581
62.3M
    int len, i;
582
583
62.3M
    *out++ = '&';
584
62.3M
    *out++ = '#';
585
586
62.3M
    if (val < 100) {
587
705
        len = (val < 10) ? 1 : 2;
588
62.3M
    } else if (val < 10000) {
589
62.3M
        len = (val < 1000) ? 3 : 4;
590
62.3M
    } else if (val < 1000000) {
591
6.16k
        len = (val < 100000) ? 5 : 6;
592
6.16k
    } else {
593
253
        len = 7;
594
253
    }
595
596
255M
    for (i = len - 1; i >= 0; i--) {
597
192M
        out[i] = '0' + val % 10;
598
192M
        val /= 10;
599
192M
    }
600
601
62.3M
    out[len] = ';';
602
603
62.3M
    return(len + 3);
604
62.3M
}
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
18.2k
xmlEscapeText(const xmlChar *text, int flags) {
634
18.2k
    const xmlChar *cur;
635
18.2k
    xmlChar *buffer;
636
18.2k
    xmlChar *out;
637
18.2k
    const xmlChar *unescaped;
638
18.2k
    size_t size = 50;
639
640
18.2k
    buffer = xmlMalloc(size + 1);
641
18.2k
    if (buffer == NULL)
642
53
        return(NULL);
643
18.1k
    out = buffer;
644
645
18.1k
    cur = text;
646
18.1k
    unescaped = cur;
647
648
106k
    while (*cur != '\0') {
649
87.8k
        char buf[12];
650
87.8k
  const xmlChar *end;
651
87.8k
        const xmlChar *repl;
652
87.8k
        size_t used;
653
87.8k
        size_t replSize;
654
87.8k
        size_t unescapedSize;
655
87.8k
        size_t totalSize;
656
87.8k
        int chunkSize = 1;
657
87.8k
        int c;
658
659
        /* accelerator */
660
65.8M
  while (1) {
661
65.8M
            c = *cur;
662
663
65.8M
            if (c < 0x80) {
664
38.5M
                if (!xmlEscapeSafe[*cur])
665
87.8k
                    break;
666
38.5M
            } else {
667
27.3M
               if (flags & XML_ESCAPE_NON_ASCII)
668
0
                   break;
669
27.3M
            }
670
65.7M
            cur += 1;
671
65.7M
        }
672
673
87.8k
        if (c == 0) {
674
16.8k
            chunkSize = 0;
675
16.8k
            repl = BAD_CAST "";
676
16.8k
            replSize = 0;
677
70.9k
        } else if (c == '<') {
678
      /*
679
       * Special handling of server side include in HTML attributes
680
       */
681
13.1k
      if ((flags & XML_ESCAPE_HTML) && (flags & XML_ESCAPE_ATTR) &&
682
12.5k
          (cur[1] == '!') && (cur[2] == '-') && (cur[3] == '-') &&
683
1.67k
          ((end = xmlStrstr(cur, BAD_CAST "-->")) != NULL)) {
684
1.12k
                chunkSize = (end - cur) + 3;
685
1.12k
                repl = cur;
686
1.12k
                replSize = chunkSize;
687
12.0k
      } else {
688
12.0k
                repl = BAD_CAST "&lt;";
689
12.0k
                replSize = 4;
690
12.0k
            }
691
57.8k
  } else if (c == '>') {
692
17.4k
            repl = BAD_CAST "&gt;";
693
17.4k
            replSize = 4;
694
40.3k
  } 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
5.77k
      if ((flags & XML_ESCAPE_HTML) && (flags & XML_ESCAPE_ATTR) &&
700
5.22k
                (cur[1] == '{') && (end = xmlStrchr(cur, '}'))) {
701
1.17k
                chunkSize = (end - cur) + 1;
702
1.17k
                repl = cur;
703
1.17k
                replSize = chunkSize;
704
4.60k
      } else {
705
4.60k
                repl = BAD_CAST "&amp;";
706
4.60k
                replSize = 5;
707
4.60k
            }
708
34.5k
  } else if ((flags & XML_ESCAPE_QUOT) && (c == '"')) {
709
0
            repl = BAD_CAST "&quot;";
710
0
            replSize = 6;
711
34.5k
  } else if (((flags & XML_ESCAPE_HTML) == 0) && (c == '\r')) {
712
0
      repl = BAD_CAST "&#13;";
713
0
            replSize = 5;
714
34.5k
  } 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
34.5k
  } else if ((flags & (XML_ESCAPE_ALLOW_INVALID | XML_ESCAPE_HTML)) ||
730
0
                   (c >= 0x20) ||
731
34.5k
             (c == '\n') || (c == '\t') || (c == '\r')) {
732
      /* default case, just copy */
733
34.5k
            cur += 1;
734
34.5k
            if (*cur != 0)
735
34.3k
                continue;
736
737
190
            chunkSize = 0;
738
190
            repl = BAD_CAST "";
739
190
            replSize = 0;
740
190
  } else {
741
            /* ignore */
742
0
            repl = BAD_CAST "";
743
0
            replSize = 0;
744
0
        }
745
746
53.4k
        used = out - buffer;
747
53.4k
        unescapedSize = cur - unescaped;
748
53.4k
        totalSize = unescapedSize + replSize;
749
750
53.4k
  cur += chunkSize;
751
752
53.4k
        if (totalSize > size - used) {
753
4.60k
            xmlChar *tmp;
754
4.60k
            int newSize;
755
756
4.60k
            if ((size > (SIZE_MAX - 1) / 2) ||
757
4.60k
                (totalSize > (SIZE_MAX - 1) / 2 - size)) {
758
0
                xmlFree(buffer);
759
0
                return(NULL);
760
0
            }
761
4.60k
            newSize = size + totalSize;
762
4.60k
            if (*cur != 0)
763
2.14k
                newSize *= 2;
764
4.60k
            tmp = xmlRealloc(buffer, newSize + 1);
765
4.60k
            if (tmp == NULL) {
766
2
                xmlFree(buffer);
767
2
                return(NULL);
768
2
            }
769
4.60k
            buffer = tmp;
770
4.60k
            size = newSize;
771
4.60k
            out = buffer + used;
772
4.60k
        }
773
774
53.4k
        memcpy(out, unescaped, unescapedSize);
775
53.4k
        out += unescapedSize;
776
53.4k
        memcpy(out, repl, replSize);
777
53.4k
        out += replSize;
778
779
53.4k
        unescaped = cur;
780
53.4k
    }
781
782
18.1k
    *out = 0;
783
18.1k
    return(buffer);
784
18.1k
}
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
18.2k
                          unsigned flags) {
802
18.2k
    if (input == NULL)
803
0
        return(NULL);
804
805
18.2k
    if ((doc != NULL) && (doc->type == XML_HTML_DOCUMENT_NODE))
806
18.2k
        flags |= XML_ESCAPE_HTML;
807
0
    else if ((doc == NULL) || (doc->encoding == NULL))
808
0
        flags |= XML_ESCAPE_NON_ASCII;
809
810
18.2k
    return(xmlEscapeText(input, flags));
811
18.2k
}
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
13.2k
xmlEncodeEntitiesReentrant(xmlDocPtr doc, const xmlChar *input) {
831
13.2k
    return xmlEncodeEntitiesInternal(doc, input, 0);
832
13.2k
}
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
78.2k
xmlFreeEntityWrapper(void *entity, const xmlChar *name ATTRIBUTE_UNUSED) {
877
78.2k
    if (entity != NULL)
878
78.2k
  xmlFreeEntity((xmlEntityPtr) entity);
879
78.2k
}
880
881
/**
882
 * xmlFreeEntitiesTable:
883
 * @table:  An entity table
884
 *
885
 * Deallocate the memory used by an entities hash table.
886
 */
887
void
888
39.8k
xmlFreeEntitiesTable(xmlEntitiesTablePtr table) {
889
39.8k
    xmlHashFree(table, xmlFreeEntityWrapper);
890
39.8k
}
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
17.4k
xmlCopyEntity(void *payload, const xmlChar *name ATTRIBUTE_UNUSED) {
902
17.4k
    xmlEntityPtr ent = (xmlEntityPtr) payload;
903
17.4k
    xmlEntityPtr cur;
904
905
17.4k
    cur = (xmlEntityPtr) xmlMalloc(sizeof(xmlEntity));
906
17.4k
    if (cur == NULL)
907
3
  return(NULL);
908
17.4k
    memset(cur, 0, sizeof(xmlEntity));
909
17.4k
    cur->type = XML_ENTITY_DECL;
910
911
17.4k
    cur->etype = ent->etype;
912
17.4k
    if (ent->name != NULL) {
913
17.4k
  cur->name = xmlStrdup(ent->name);
914
17.4k
        if (cur->name == NULL)
915
3
            goto error;
916
17.4k
    }
917
17.4k
    if (ent->ExternalID != NULL) {
918
0
  cur->ExternalID = xmlStrdup(ent->ExternalID);
919
0
        if (cur->ExternalID == NULL)
920
0
            goto error;
921
0
    }
922
17.4k
    if (ent->SystemID != NULL) {
923
822
  cur->SystemID = xmlStrdup(ent->SystemID);
924
822
        if (cur->SystemID == NULL)
925
2
            goto error;
926
822
    }
927
17.4k
    if (ent->content != NULL) {
928
16.6k
  cur->content = xmlStrdup(ent->content);
929
16.6k
        if (cur->content == NULL)
930
2
            goto error;
931
16.6k
    }
932
17.4k
    if (ent->orig != NULL) {
933
16.6k
  cur->orig = xmlStrdup(ent->orig);
934
16.6k
        if (cur->orig == NULL)
935
3
            goto error;
936
16.6k
    }
937
17.4k
    if (ent->URI != NULL) {
938
619
  cur->URI = xmlStrdup(ent->URI);
939
619
        if (cur->URI == NULL)
940
1
            goto error;
941
619
    }
942
17.4k
    return(cur);
943
944
11
error:
945
11
    xmlFreeEntity(cur);
946
11
    return(NULL);
947
17.4k
}
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.59k
xmlCopyEntitiesTable(xmlEntitiesTablePtr table) {
959
6.59k
    return(xmlHashCopySafe(table, xmlCopyEntity, xmlFreeEntityWrapper));
960
6.59k
}
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 */