| Line | Count | Source | 
| 1 |  | /* | 
| 2 |  |  * parser.c : an XML 1.0 parser, namespaces and validity support are mostly | 
| 3 |  |  *            implemented on top of the SAX interfaces | 
| 4 |  |  * | 
| 5 |  |  * References: | 
| 6 |  |  *   The XML specification: | 
| 7 |  |  *     http://www.w3.org/TR/REC-xml | 
| 8 |  |  *   Original 1.0 version: | 
| 9 |  |  *     http://www.w3.org/TR/1998/REC-xml-19980210 | 
| 10 |  |  *   XML second edition working draft | 
| 11 |  |  *     http://www.w3.org/TR/2000/WD-xml-2e-20000814 | 
| 12 |  |  * | 
| 13 |  |  * Okay this is a big file, the parser core is around 7000 lines, then it | 
| 14 |  |  * is followed by the progressive parser top routines, then the various | 
| 15 |  |  * high level APIs to call the parser and a few miscellaneous functions. | 
| 16 |  |  * A number of helper functions and deprecated ones have been moved to | 
| 17 |  |  * parserInternals.c to reduce this file size. | 
| 18 |  |  * As much as possible the functions are associated with their relative | 
| 19 |  |  * production in the XML specification. A few productions defining the | 
| 20 |  |  * different ranges of character are actually implanted either in | 
| 21 |  |  * parserInternals.h or parserInternals.c | 
| 22 |  |  * The DOM tree build is realized from the default SAX callbacks in | 
| 23 |  |  * the module SAX2.c. | 
| 24 |  |  * The routines doing the validation checks are in valid.c and called either | 
| 25 |  |  * from the SAX callbacks or as standalone functions using a preparsed | 
| 26 |  |  * document. | 
| 27 |  |  * | 
| 28 |  |  * See Copyright for the status of this software. | 
| 29 |  |  * | 
| 30 |  |  * Author: Daniel Veillard | 
| 31 |  |  */ | 
| 32 |  |  | 
| 33 |  | /* To avoid EBCDIC trouble when parsing on zOS */ | 
| 34 |  | #if defined(__MVS__) | 
| 35 |  | #pragma convert("ISO8859-1") | 
| 36 |  | #endif | 
| 37 |  |  | 
| 38 |  | #define IN_LIBXML | 
| 39 |  | #include "libxml.h" | 
| 40 |  |  | 
| 41 |  | #if defined(_WIN32) | 
| 42 |  | #define XML_DIR_SEP '\\' | 
| 43 |  | #else | 
| 44 |  | #define XML_DIR_SEP '/' | 
| 45 |  | #endif | 
| 46 |  |  | 
| 47 |  | #include <stdlib.h> | 
| 48 |  | #include <limits.h> | 
| 49 |  | #include <string.h> | 
| 50 |  | #include <stdarg.h> | 
| 51 |  | #include <stddef.h> | 
| 52 |  | #include <ctype.h> | 
| 53 |  | #include <stdlib.h> | 
| 54 |  | #include <libxml/parser.h> | 
| 55 |  | #include <libxml/xmlmemory.h> | 
| 56 |  | #include <libxml/tree.h> | 
| 57 |  | #include <libxml/parserInternals.h> | 
| 58 |  | #include <libxml/valid.h> | 
| 59 |  | #include <libxml/entities.h> | 
| 60 |  | #include <libxml/xmlerror.h> | 
| 61 |  | #include <libxml/encoding.h> | 
| 62 |  | #include <libxml/xmlIO.h> | 
| 63 |  | #include <libxml/uri.h> | 
| 64 |  | #include <libxml/SAX2.h> | 
| 65 |  | #include <libxml/HTMLparser.h> | 
| 66 |  | #ifdef LIBXML_CATALOG_ENABLED | 
| 67 |  | #include <libxml/catalog.h> | 
| 68 |  | #endif | 
| 69 |  |  | 
| 70 |  | #include "private/buf.h" | 
| 71 |  | #include "private/dict.h" | 
| 72 |  | #include "private/entities.h" | 
| 73 |  | #include "private/error.h" | 
| 74 |  | #include "private/html.h" | 
| 75 |  | #include "private/io.h" | 
| 76 |  | #include "private/memory.h" | 
| 77 |  | #include "private/parser.h" | 
| 78 |  | #include "private/tree.h" | 
| 79 |  |  | 
| 80 | 1.69M | #define NS_INDEX_EMPTY  INT_MAX | 
| 81 | 50.8k | #define NS_INDEX_XML    (INT_MAX - 1) | 
| 82 | 669k | #define URI_HASH_EMPTY  0xD943A04E | 
| 83 | 8.70k | #define URI_HASH_XML    0xF0451F02 | 
| 84 |  |  | 
| 85 |  | #ifndef STDIN_FILENO | 
| 86 | 0 |   #define STDIN_FILENO 0 | 
| 87 |  | #endif | 
| 88 |  |  | 
| 89 |  | #ifndef SIZE_MAX | 
| 90 |  |   #define SIZE_MAX ((size_t) -1) | 
| 91 |  | #endif | 
| 92 |  |  | 
| 93 | 245k | #define XML_MAX_ATTRS 100000000 /* 100 million */ | 
| 94 |  |  | 
| 95 | 624k | #define XML_SPECIAL_EXTERNAL    (1 << 20) | 
| 96 | 624k | #define XML_SPECIAL_TYPE_MASK   (XML_SPECIAL_EXTERNAL - 1) | 
| 97 |  |  | 
| 98 | 668k | #define XML_ATTVAL_ALLOC        (1 << 0) | 
| 99 | 72.6k | #define XML_ATTVAL_NORM_CHANGE  (1 << 1) | 
| 100 |  |  | 
| 101 |  | struct _xmlStartTag { | 
| 102 |  |     const xmlChar *prefix; | 
| 103 |  |     const xmlChar *URI; | 
| 104 |  |     int line; | 
| 105 |  |     int nsNr; | 
| 106 |  | }; | 
| 107 |  |  | 
| 108 |  | typedef struct { | 
| 109 |  |     void *saxData; | 
| 110 |  |     unsigned prefixHashValue; | 
| 111 |  |     unsigned uriHashValue; | 
| 112 |  |     unsigned elementId; | 
| 113 |  |     int oldIndex; | 
| 114 |  | } xmlParserNsExtra; | 
| 115 |  |  | 
| 116 |  | typedef struct { | 
| 117 |  |     unsigned hashValue; | 
| 118 |  |     int index; | 
| 119 |  | } xmlParserNsBucket; | 
| 120 |  |  | 
| 121 |  | struct _xmlParserNsData { | 
| 122 |  |     xmlParserNsExtra *extra; | 
| 123 |  |  | 
| 124 |  |     unsigned hashSize; | 
| 125 |  |     unsigned hashElems; | 
| 126 |  |     xmlParserNsBucket *hash; | 
| 127 |  |  | 
| 128 |  |     unsigned elementId; | 
| 129 |  |     int defaultNsIndex; | 
| 130 |  |     int minNsIndex; | 
| 131 |  | }; | 
| 132 |  |  | 
| 133 |  | static int | 
| 134 |  | xmlParseElementStart(xmlParserCtxtPtr ctxt); | 
| 135 |  |  | 
| 136 |  | static void | 
| 137 |  | xmlParseElementEnd(xmlParserCtxtPtr ctxt); | 
| 138 |  |  | 
| 139 |  | static xmlEntityPtr | 
| 140 |  | xmlLookupGeneralEntity(xmlParserCtxtPtr ctxt, const xmlChar *name, int inAttr); | 
| 141 |  |  | 
| 142 |  | static const xmlChar * | 
| 143 |  | xmlParseEntityRefInternal(xmlParserCtxtPtr ctxt); | 
| 144 |  |  | 
| 145 |  | /************************************************************************ | 
| 146 |  |  *                  * | 
| 147 |  |  *  Arbitrary limits set in the parser. See XML_PARSE_HUGE    * | 
| 148 |  |  *                  * | 
| 149 |  |  ************************************************************************/ | 
| 150 |  |  | 
| 151 |  | #define XML_PARSER_BIG_ENTITY 1000 | 
| 152 |  | #define XML_PARSER_LOT_ENTITY 5000 | 
| 153 |  |  | 
| 154 |  | /* | 
| 155 |  |  * Constants for protection against abusive entity expansion | 
| 156 |  |  * ("billion laughs"). | 
| 157 |  |  */ | 
| 158 |  |  | 
| 159 |  | /* | 
| 160 |  |  * A certain amount of entity expansion which is always allowed. | 
| 161 |  |  */ | 
| 162 | 1.78M | #define XML_PARSER_ALLOWED_EXPANSION 1000000 | 
| 163 |  |  | 
| 164 |  | /* | 
| 165 |  |  * Fixed cost for each entity reference. This crudely models processing time | 
| 166 |  |  * as well to protect, for example, against exponential expansion of empty | 
| 167 |  |  * or very short entities. | 
| 168 |  |  */ | 
| 169 | 1.78M | #define XML_ENT_FIXED_COST 20 | 
| 170 |  |  | 
| 171 | 33.3M | #define XML_PARSER_BIG_BUFFER_SIZE 300 | 
| 172 | 124k | #define XML_PARSER_BUFFER_SIZE 100 | 
| 173 | 36.8k | #define SAX_COMPAT_MODE BAD_CAST "SAX compatibility mode document" | 
| 174 |  |  | 
| 175 |  | /** | 
| 176 |  |  * XML_PARSER_CHUNK_SIZE | 
| 177 |  |  * | 
| 178 |  |  * When calling GROW that's the minimal amount of data | 
| 179 |  |  * the parser expected to have received. It is not a hard | 
| 180 |  |  * limit but an optimization when reading strings like Names | 
| 181 |  |  * It is not strictly needed as long as inputs available characters | 
| 182 |  |  * are followed by 0, which should be provided by the I/O level | 
| 183 |  |  */ | 
| 184 |  | #define XML_PARSER_CHUNK_SIZE 100 | 
| 185 |  |  | 
| 186 |  | /** | 
| 187 |  |  * Constant string describing the version of the library used at | 
| 188 |  |  * run-time. | 
| 189 |  |  */ | 
| 190 |  | const char *const | 
| 191 |  | xmlParserVersion = LIBXML_VERSION_STRING LIBXML_VERSION_EXTRA; | 
| 192 |  |  | 
| 193 |  | /* | 
| 194 |  |  * List of XML prefixed PI allowed by W3C specs | 
| 195 |  |  */ | 
| 196 |  |  | 
| 197 |  | static const char* const xmlW3CPIs[] = { | 
| 198 |  |     "xml-stylesheet", | 
| 199 |  |     "xml-model", | 
| 200 |  |     NULL | 
| 201 |  | }; | 
| 202 |  |  | 
| 203 |  |  | 
| 204 |  | /* DEPR void xmlParserHandleReference(xmlParserCtxtPtr ctxt); */ | 
| 205 |  | static xmlEntityPtr xmlParseStringPEReference(xmlParserCtxtPtr ctxt, | 
| 206 |  |                                               const xmlChar **str); | 
| 207 |  |  | 
| 208 |  | static void | 
| 209 |  | xmlCtxtParseEntity(xmlParserCtxtPtr ctxt, xmlEntityPtr ent); | 
| 210 |  |  | 
| 211 |  | static int | 
| 212 |  | xmlLoadEntityContent(xmlParserCtxtPtr ctxt, xmlEntityPtr entity); | 
| 213 |  |  | 
| 214 |  | static void | 
| 215 |  | xmlParsePERefInternal(xmlParserCtxt *ctxt, int markupDecl); | 
| 216 |  |  | 
| 217 |  | /************************************************************************ | 
| 218 |  |  *                  * | 
| 219 |  |  *    Some factorized error routines        * | 
| 220 |  |  *                  * | 
| 221 |  |  ************************************************************************/ | 
| 222 |  |  | 
| 223 |  | static void | 
| 224 | 0 | xmlErrMemory(xmlParserCtxtPtr ctxt) { | 
| 225 | 0 |     xmlCtxtErrMemory(ctxt); | 
| 226 | 0 | } | 
| 227 |  |  | 
| 228 |  | /** | 
| 229 |  |  * Handle a redefinition of attribute error | 
| 230 |  |  * | 
| 231 |  |  * @param ctxt  an XML parser context | 
| 232 |  |  * @param prefix  the attribute prefix | 
| 233 |  |  * @param localname  the attribute localname | 
| 234 |  |  */ | 
| 235 |  | static void | 
| 236 |  | xmlErrAttributeDup(xmlParserCtxtPtr ctxt, const xmlChar * prefix, | 
| 237 |  |                    const xmlChar * localname) | 
| 238 | 501k | { | 
| 239 | 501k |     if (prefix == NULL) | 
| 240 | 486k |         xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, XML_ERR_ATTRIBUTE_REDEFINED, | 
| 241 | 486k |                    XML_ERR_FATAL, localname, NULL, NULL, 0, | 
| 242 | 486k |                    "Attribute %s redefined\n", localname); | 
| 243 | 15.2k |     else | 
| 244 | 15.2k |         xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, XML_ERR_ATTRIBUTE_REDEFINED, | 
| 245 | 15.2k |                    XML_ERR_FATAL, prefix, localname, NULL, 0, | 
| 246 | 15.2k |                    "Attribute %s:%s redefined\n", prefix, localname); | 
| 247 | 501k | } | 
| 248 |  |  | 
| 249 |  | /** | 
| 250 |  |  * Handle a fatal parser error, i.e. violating Well-Formedness constraints | 
| 251 |  |  * | 
| 252 |  |  * @param ctxt  an XML parser context | 
| 253 |  |  * @param error  the error number | 
| 254 |  |  * @param msg  the error message | 
| 255 |  |  */ | 
| 256 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 257 |  | xmlFatalErrMsg(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 258 |  |                const char *msg) | 
| 259 | 7.50M | { | 
| 260 | 7.50M |     xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, error, XML_ERR_FATAL, | 
| 261 | 7.50M |                NULL, NULL, NULL, 0, "%s", msg); | 
| 262 | 7.50M | } | 
| 263 |  |  | 
| 264 |  | /** | 
| 265 |  |  * Handle a warning. | 
| 266 |  |  * | 
| 267 |  |  * @param ctxt  an XML parser context | 
| 268 |  |  * @param error  the error number | 
| 269 |  |  * @param msg  the error message | 
| 270 |  |  * @param str1  extra data | 
| 271 |  |  * @param str2  extra data | 
| 272 |  |  */ | 
| 273 |  | void LIBXML_ATTR_FORMAT(3,0) | 
| 274 |  | xmlWarningMsg(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 275 |  |               const char *msg, const xmlChar *str1, const xmlChar *str2) | 
| 276 | 3.24k | { | 
| 277 | 3.24k |     xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, error, XML_ERR_WARNING, | 
| 278 | 3.24k |                str1, str2, NULL, 0, msg, str1, str2); | 
| 279 | 3.24k | } | 
| 280 |  |  | 
| 281 |  | #ifdef LIBXML_VALID_ENABLED | 
| 282 |  | /** | 
| 283 |  |  * Handle a validity error. | 
| 284 |  |  * | 
| 285 |  |  * @param ctxt  an XML parser context | 
| 286 |  |  * @param error  the error number | 
| 287 |  |  * @param msg  the error message | 
| 288 |  |  * @param str1  extra data | 
| 289 |  |  * @param str2  extra data | 
| 290 |  |  */ | 
| 291 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 292 |  | xmlValidityError(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 293 |  |               const char *msg, const xmlChar *str1, const xmlChar *str2) | 
| 294 | 0 | { | 
| 295 | 0 |     ctxt->valid = 0; | 
| 296 |  | 
 | 
| 297 | 0 |     xmlCtxtErr(ctxt, NULL, XML_FROM_DTD, error, XML_ERR_ERROR, | 
| 298 | 0 |                str1, str2, NULL, 0, msg, str1, str2); | 
| 299 | 0 | } | 
| 300 |  | #endif | 
| 301 |  |  | 
| 302 |  | /** | 
| 303 |  |  * Handle a fatal parser error, i.e. violating Well-Formedness constraints | 
| 304 |  |  * | 
| 305 |  |  * @param ctxt  an XML parser context | 
| 306 |  |  * @param error  the error number | 
| 307 |  |  * @param msg  the error message | 
| 308 |  |  * @param val  an integer value | 
| 309 |  |  */ | 
| 310 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 311 |  | xmlFatalErrMsgInt(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 312 |  |                   const char *msg, int val) | 
| 313 | 1.47M | { | 
| 314 | 1.47M |     xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, error, XML_ERR_FATAL, | 
| 315 | 1.47M |                NULL, NULL, NULL, val, msg, val); | 
| 316 | 1.47M | } | 
| 317 |  |  | 
| 318 |  | /** | 
| 319 |  |  * Handle a fatal parser error, i.e. violating Well-Formedness constraints | 
| 320 |  |  * | 
| 321 |  |  * @param ctxt  an XML parser context | 
| 322 |  |  * @param error  the error number | 
| 323 |  |  * @param msg  the error message | 
| 324 |  |  * @param str1  an string info | 
| 325 |  |  * @param val  an integer value | 
| 326 |  |  * @param str2  an string info | 
| 327 |  |  */ | 
| 328 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 329 |  | xmlFatalErrMsgStrIntStr(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 330 |  |                   const char *msg, const xmlChar *str1, int val, | 
| 331 |  |       const xmlChar *str2) | 
| 332 | 258k | { | 
| 333 | 258k |     xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, error, XML_ERR_FATAL, | 
| 334 | 258k |                str1, str2, NULL, val, msg, str1, val, str2); | 
| 335 | 258k | } | 
| 336 |  |  | 
| 337 |  | /** | 
| 338 |  |  * Handle a fatal parser error, i.e. violating Well-Formedness constraints | 
| 339 |  |  * | 
| 340 |  |  * @param ctxt  an XML parser context | 
| 341 |  |  * @param error  the error number | 
| 342 |  |  * @param msg  the error message | 
| 343 |  |  * @param val  a string value | 
| 344 |  |  */ | 
| 345 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 346 |  | xmlFatalErrMsgStr(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 347 |  |                   const char *msg, const xmlChar * val) | 
| 348 | 400k | { | 
| 349 | 400k |     xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, error, XML_ERR_FATAL, | 
| 350 | 400k |                val, NULL, NULL, 0, msg, val); | 
| 351 | 400k | } | 
| 352 |  |  | 
| 353 |  | /** | 
| 354 |  |  * Handle a non fatal parser error | 
| 355 |  |  * | 
| 356 |  |  * @param ctxt  an XML parser context | 
| 357 |  |  * @param error  the error number | 
| 358 |  |  * @param msg  the error message | 
| 359 |  |  * @param val  a string value | 
| 360 |  |  */ | 
| 361 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 362 |  | xmlErrMsgStr(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 363 |  |                   const char *msg, const xmlChar * val) | 
| 364 | 19.6k | { | 
| 365 | 19.6k |     xmlCtxtErr(ctxt, NULL, XML_FROM_PARSER, error, XML_ERR_ERROR, | 
| 366 | 19.6k |                val, NULL, NULL, 0, msg, val); | 
| 367 | 19.6k | } | 
| 368 |  |  | 
| 369 |  | /** | 
| 370 |  |  * Handle a fatal parser error, i.e. violating Well-Formedness constraints | 
| 371 |  |  * | 
| 372 |  |  * @param ctxt  an XML parser context | 
| 373 |  |  * @param error  the error number | 
| 374 |  |  * @param msg  the message | 
| 375 |  |  * @param info1  extra information string | 
| 376 |  |  * @param info2  extra information string | 
| 377 |  |  * @param info3  extra information string | 
| 378 |  |  */ | 
| 379 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 380 |  | xmlNsErr(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 381 |  |          const char *msg, | 
| 382 |  |          const xmlChar * info1, const xmlChar * info2, | 
| 383 |  |          const xmlChar * info3) | 
| 384 | 251k | { | 
| 385 | 251k |     ctxt->nsWellFormed = 0; | 
| 386 |  |  | 
| 387 | 251k |     xmlCtxtErr(ctxt, NULL, XML_FROM_NAMESPACE, error, XML_ERR_ERROR, | 
| 388 | 251k |                info1, info2, info3, 0, msg, info1, info2, info3); | 
| 389 | 251k | } | 
| 390 |  |  | 
| 391 |  | /** | 
| 392 |  |  * Handle a namespace warning error | 
| 393 |  |  * | 
| 394 |  |  * @param ctxt  an XML parser context | 
| 395 |  |  * @param error  the error number | 
| 396 |  |  * @param msg  the message | 
| 397 |  |  * @param info1  extra information string | 
| 398 |  |  * @param info2  extra information string | 
| 399 |  |  * @param info3  extra information string | 
| 400 |  |  */ | 
| 401 |  | static void LIBXML_ATTR_FORMAT(3,0) | 
| 402 |  | xmlNsWarn(xmlParserCtxtPtr ctxt, xmlParserErrors error, | 
| 403 |  |          const char *msg, | 
| 404 |  |          const xmlChar * info1, const xmlChar * info2, | 
| 405 |  |          const xmlChar * info3) | 
| 406 | 18.0k | { | 
| 407 | 18.0k |     xmlCtxtErr(ctxt, NULL, XML_FROM_NAMESPACE, error, XML_ERR_WARNING, | 
| 408 | 18.0k |                info1, info2, info3, 0, msg, info1, info2, info3); | 
| 409 | 18.0k | } | 
| 410 |  |  | 
| 411 |  | /** | 
| 412 |  |  * Check for non-linear entity expansion behaviour. | 
| 413 |  |  * | 
| 414 |  |  * In some cases like xmlExpandEntityInAttValue, this function is called | 
| 415 |  |  * for each, possibly nested entity and its unexpanded content length. | 
| 416 |  |  * | 
| 417 |  |  * In other cases like #xmlParseReference, it's only called for each | 
| 418 |  |  * top-level entity with its unexpanded content length plus the sum of | 
| 419 |  |  * the unexpanded content lengths (plus fixed cost) of all nested | 
| 420 |  |  * entities. | 
| 421 |  |  * | 
| 422 |  |  * Summing the unexpanded lengths also adds the length of the reference. | 
| 423 |  |  * This is by design. Taking the length of the entity name into account | 
| 424 |  |  * discourages attacks that try to waste CPU time with abusively long | 
| 425 |  |  * entity names. See test/recurse/lol6.xml for example. Each call also | 
| 426 |  |  * adds some fixed cost XML_ENT_FIXED_COST to discourage attacks with | 
| 427 |  |  * short entities. | 
| 428 |  |  * | 
| 429 |  |  * @param ctxt  parser context | 
| 430 |  |  * @param extra  sum of unexpanded entity sizes | 
| 431 |  |  * @returns 1 on error, 0 on success. | 
| 432 |  |  */ | 
| 433 |  | static int | 
| 434 |  | xmlParserEntityCheck(xmlParserCtxtPtr ctxt, unsigned long extra) | 
| 435 | 1.78M | { | 
| 436 | 1.78M |     unsigned long consumed; | 
| 437 | 1.78M |     unsigned long *expandedSize; | 
| 438 | 1.78M |     xmlParserInputPtr input = ctxt->input; | 
| 439 | 1.78M |     xmlEntityPtr entity = input->entity; | 
| 440 |  |  | 
| 441 | 1.78M |     if ((entity) && (entity->flags & XML_ENT_CHECKED)) | 
| 442 | 935 |         return(0); | 
| 443 |  |  | 
| 444 |  |     /* | 
| 445 |  |      * Compute total consumed bytes so far, including input streams of | 
| 446 |  |      * external entities. | 
| 447 |  |      */ | 
| 448 | 1.78M |     consumed = input->consumed; | 
| 449 | 1.78M |     xmlSaturatedAddSizeT(&consumed, input->cur - input->base); | 
| 450 | 1.78M |     xmlSaturatedAdd(&consumed, ctxt->sizeentities); | 
| 451 |  |  | 
| 452 | 1.78M |     if (entity) | 
| 453 | 12.6k |         expandedSize = &entity->expandedSize; | 
| 454 | 1.76M |     else | 
| 455 | 1.76M |         expandedSize = &ctxt->sizeentcopy; | 
| 456 |  |  | 
| 457 |  |     /* | 
| 458 |  |      * Add extra cost and some fixed cost. | 
| 459 |  |      */ | 
| 460 | 1.78M |     xmlSaturatedAdd(expandedSize, extra); | 
| 461 | 1.78M |     xmlSaturatedAdd(expandedSize, XML_ENT_FIXED_COST); | 
| 462 |  |  | 
| 463 |  |     /* | 
| 464 |  |      * It's important to always use saturation arithmetic when tracking | 
| 465 |  |      * entity sizes to make the size checks reliable. If "sizeentcopy" | 
| 466 |  |      * overflows, we have to abort. | 
| 467 |  |      */ | 
| 468 | 1.78M |     if ((*expandedSize > XML_PARSER_ALLOWED_EXPANSION) && | 
| 469 | 324k |         ((*expandedSize >= ULONG_MAX) || | 
| 470 | 324k |          (*expandedSize / ctxt->maxAmpl > consumed))) { | 
| 471 | 348 |         xmlFatalErrMsg(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 472 | 348 |                        "Maximum entity amplification factor exceeded, see " | 
| 473 | 348 |                        "xmlCtxtSetMaxAmplification.\n"); | 
| 474 | 348 |         return(1); | 
| 475 | 348 |     } | 
| 476 |  |  | 
| 477 | 1.78M |     return(0); | 
| 478 | 1.78M | } | 
| 479 |  |  | 
| 480 |  | /************************************************************************ | 
| 481 |  |  *                  * | 
| 482 |  |  *    Library wide options          * | 
| 483 |  |  *                  * | 
| 484 |  |  ************************************************************************/ | 
| 485 |  |  | 
| 486 |  | /** | 
| 487 |  |  * Examines if the library has been compiled with a given feature. | 
| 488 |  |  * | 
| 489 |  |  * @param feature  the feature to be examined | 
| 490 |  |  * @returns zero (0) if the feature does not exist or an unknown | 
| 491 |  |  * feature is requested, non-zero otherwise. | 
| 492 |  |  */ | 
| 493 |  | int | 
| 494 |  | xmlHasFeature(xmlFeature feature) | 
| 495 | 0 | { | 
| 496 | 0 |     switch (feature) { | 
| 497 | 0 |   case XML_WITH_THREAD: | 
| 498 | 0 | #ifdef LIBXML_THREAD_ENABLED | 
| 499 | 0 |       return(1); | 
| 500 |  | #else | 
| 501 |  |       return(0); | 
| 502 |  | #endif | 
| 503 | 0 |         case XML_WITH_TREE: | 
| 504 | 0 |             return(1); | 
| 505 | 0 |         case XML_WITH_OUTPUT: | 
| 506 | 0 | #ifdef LIBXML_OUTPUT_ENABLED | 
| 507 | 0 |             return(1); | 
| 508 |  | #else | 
| 509 |  |             return(0); | 
| 510 |  | #endif | 
| 511 | 0 |         case XML_WITH_PUSH: | 
| 512 | 0 | #ifdef LIBXML_PUSH_ENABLED | 
| 513 | 0 |             return(1); | 
| 514 |  | #else | 
| 515 |  |             return(0); | 
| 516 |  | #endif | 
| 517 | 0 |         case XML_WITH_READER: | 
| 518 | 0 | #ifdef LIBXML_READER_ENABLED | 
| 519 | 0 |             return(1); | 
| 520 |  | #else | 
| 521 |  |             return(0); | 
| 522 |  | #endif | 
| 523 | 0 |         case XML_WITH_PATTERN: | 
| 524 | 0 | #ifdef LIBXML_PATTERN_ENABLED | 
| 525 | 0 |             return(1); | 
| 526 |  | #else | 
| 527 |  |             return(0); | 
| 528 |  | #endif | 
| 529 | 0 |         case XML_WITH_WRITER: | 
| 530 | 0 | #ifdef LIBXML_WRITER_ENABLED | 
| 531 | 0 |             return(1); | 
| 532 |  | #else | 
| 533 |  |             return(0); | 
| 534 |  | #endif | 
| 535 | 0 |         case XML_WITH_SAX1: | 
| 536 | 0 | #ifdef LIBXML_SAX1_ENABLED | 
| 537 | 0 |             return(1); | 
| 538 |  | #else | 
| 539 |  |             return(0); | 
| 540 |  | #endif | 
| 541 | 0 |         case XML_WITH_HTTP: | 
| 542 | 0 |             return(0); | 
| 543 | 0 |         case XML_WITH_VALID: | 
| 544 | 0 | #ifdef LIBXML_VALID_ENABLED | 
| 545 | 0 |             return(1); | 
| 546 |  | #else | 
| 547 |  |             return(0); | 
| 548 |  | #endif | 
| 549 | 0 |         case XML_WITH_HTML: | 
| 550 | 0 | #ifdef LIBXML_HTML_ENABLED | 
| 551 | 0 |             return(1); | 
| 552 |  | #else | 
| 553 |  |             return(0); | 
| 554 |  | #endif | 
| 555 | 0 |         case XML_WITH_LEGACY: | 
| 556 | 0 |             return(0); | 
| 557 | 0 |         case XML_WITH_C14N: | 
| 558 | 0 | #ifdef LIBXML_C14N_ENABLED | 
| 559 | 0 |             return(1); | 
| 560 |  | #else | 
| 561 |  |             return(0); | 
| 562 |  | #endif | 
| 563 | 0 |         case XML_WITH_CATALOG: | 
| 564 | 0 | #ifdef LIBXML_CATALOG_ENABLED | 
| 565 | 0 |             return(1); | 
| 566 |  | #else | 
| 567 |  |             return(0); | 
| 568 |  | #endif | 
| 569 | 0 |         case XML_WITH_XPATH: | 
| 570 | 0 | #ifdef LIBXML_XPATH_ENABLED | 
| 571 | 0 |             return(1); | 
| 572 |  | #else | 
| 573 |  |             return(0); | 
| 574 |  | #endif | 
| 575 | 0 |         case XML_WITH_XPTR: | 
| 576 | 0 | #ifdef LIBXML_XPTR_ENABLED | 
| 577 | 0 |             return(1); | 
| 578 |  | #else | 
| 579 |  |             return(0); | 
| 580 |  | #endif | 
| 581 | 0 |         case XML_WITH_XINCLUDE: | 
| 582 | 0 | #ifdef LIBXML_XINCLUDE_ENABLED | 
| 583 | 0 |             return(1); | 
| 584 |  | #else | 
| 585 |  |             return(0); | 
| 586 |  | #endif | 
| 587 | 0 |         case XML_WITH_ICONV: | 
| 588 | 0 | #ifdef LIBXML_ICONV_ENABLED | 
| 589 | 0 |             return(1); | 
| 590 |  | #else | 
| 591 |  |             return(0); | 
| 592 |  | #endif | 
| 593 | 0 |         case XML_WITH_ISO8859X: | 
| 594 | 0 | #ifdef LIBXML_ISO8859X_ENABLED | 
| 595 | 0 |             return(1); | 
| 596 |  | #else | 
| 597 |  |             return(0); | 
| 598 |  | #endif | 
| 599 | 0 |         case XML_WITH_UNICODE: | 
| 600 | 0 |             return(0); | 
| 601 | 0 |         case XML_WITH_REGEXP: | 
| 602 | 0 | #ifdef LIBXML_REGEXP_ENABLED | 
| 603 | 0 |             return(1); | 
| 604 |  | #else | 
| 605 |  |             return(0); | 
| 606 |  | #endif | 
| 607 | 0 |         case XML_WITH_AUTOMATA: | 
| 608 | 0 | #ifdef LIBXML_REGEXP_ENABLED | 
| 609 | 0 |             return(1); | 
| 610 |  | #else | 
| 611 |  |             return(0); | 
| 612 |  | #endif | 
| 613 | 0 |         case XML_WITH_EXPR: | 
| 614 | 0 |             return(0); | 
| 615 | 0 |         case XML_WITH_RELAXNG: | 
| 616 | 0 | #ifdef LIBXML_RELAXNG_ENABLED | 
| 617 | 0 |             return(1); | 
| 618 |  | #else | 
| 619 |  |             return(0); | 
| 620 |  | #endif | 
| 621 | 0 |         case XML_WITH_SCHEMAS: | 
| 622 | 0 | #ifdef LIBXML_SCHEMAS_ENABLED | 
| 623 | 0 |             return(1); | 
| 624 |  | #else | 
| 625 |  |             return(0); | 
| 626 |  | #endif | 
| 627 | 0 |         case XML_WITH_SCHEMATRON: | 
| 628 |  | #ifdef LIBXML_SCHEMATRON_ENABLED | 
| 629 |  |             return(1); | 
| 630 |  | #else | 
| 631 | 0 |             return(0); | 
| 632 | 0 | #endif | 
| 633 | 0 |         case XML_WITH_MODULES: | 
| 634 | 0 | #ifdef LIBXML_MODULES_ENABLED | 
| 635 | 0 |             return(1); | 
| 636 |  | #else | 
| 637 |  |             return(0); | 
| 638 |  | #endif | 
| 639 | 0 |         case XML_WITH_DEBUG: | 
| 640 | 0 | #ifdef LIBXML_DEBUG_ENABLED | 
| 641 | 0 |             return(1); | 
| 642 |  | #else | 
| 643 |  |             return(0); | 
| 644 |  | #endif | 
| 645 | 0 |         case XML_WITH_DEBUG_MEM: | 
| 646 | 0 |             return(0); | 
| 647 | 0 |         case XML_WITH_ZLIB: | 
| 648 |  | #ifdef LIBXML_ZLIB_ENABLED | 
| 649 |  |             return(1); | 
| 650 |  | #else | 
| 651 | 0 |             return(0); | 
| 652 | 0 | #endif | 
| 653 | 0 |         case XML_WITH_LZMA: | 
| 654 | 0 |             return(0); | 
| 655 | 0 |         case XML_WITH_ICU: | 
| 656 |  | #ifdef LIBXML_ICU_ENABLED | 
| 657 |  |             return(1); | 
| 658 |  | #else | 
| 659 | 0 |             return(0); | 
| 660 | 0 | #endif | 
| 661 | 0 |         default: | 
| 662 | 0 |       break; | 
| 663 | 0 |      } | 
| 664 | 0 |      return(0); | 
| 665 | 0 | } | 
| 666 |  |  | 
| 667 |  | /************************************************************************ | 
| 668 |  |  *                  * | 
| 669 |  |  *      Simple string buffer        * | 
| 670 |  |  *                  * | 
| 671 |  |  ************************************************************************/ | 
| 672 |  |  | 
| 673 |  | typedef struct { | 
| 674 |  |     xmlChar *mem; | 
| 675 |  |     unsigned size; | 
| 676 |  |     unsigned cap; /* size < cap */ | 
| 677 |  |     unsigned max; /* size <= max */ | 
| 678 |  |     xmlParserErrors code; | 
| 679 |  | } xmlSBuf; | 
| 680 |  |  | 
| 681 |  | static void | 
| 682 | 644k | xmlSBufInit(xmlSBuf *buf, unsigned max) { | 
| 683 | 644k |     buf->mem = NULL; | 
| 684 | 644k |     buf->size = 0; | 
| 685 | 644k |     buf->cap = 0; | 
| 686 | 644k |     buf->max = max; | 
| 687 | 644k |     buf->code = XML_ERR_OK; | 
| 688 | 644k | } | 
| 689 |  |  | 
| 690 |  | static int | 
| 691 | 107k | xmlSBufGrow(xmlSBuf *buf, unsigned len) { | 
| 692 | 107k |     xmlChar *mem; | 
| 693 | 107k |     unsigned cap; | 
| 694 |  |  | 
| 695 | 107k |     if (len >= UINT_MAX / 2 - buf->size) { | 
| 696 | 0 |         if (buf->code == XML_ERR_OK) | 
| 697 | 0 |             buf->code = XML_ERR_RESOURCE_LIMIT; | 
| 698 | 0 |         return(-1); | 
| 699 | 0 |     } | 
| 700 |  |  | 
| 701 | 107k |     cap = (buf->size + len) * 2; | 
| 702 | 107k |     if (cap < 240) | 
| 703 | 76.0k |         cap = 240; | 
| 704 |  |  | 
| 705 | 107k |     mem = xmlRealloc(buf->mem, cap); | 
| 706 | 107k |     if (mem == NULL) { | 
| 707 | 0 |         buf->code = XML_ERR_NO_MEMORY; | 
| 708 | 0 |         return(-1); | 
| 709 | 0 |     } | 
| 710 |  |  | 
| 711 | 107k |     buf->mem = mem; | 
| 712 | 107k |     buf->cap = cap; | 
| 713 |  |  | 
| 714 | 107k |     return(0); | 
| 715 | 107k | } | 
| 716 |  |  | 
| 717 |  | static void | 
| 718 | 52.0M | xmlSBufAddString(xmlSBuf *buf, const xmlChar *str, unsigned len) { | 
| 719 | 52.0M |     if (buf->max - buf->size < len) { | 
| 720 | 0 |         if (buf->code == XML_ERR_OK) | 
| 721 | 0 |             buf->code = XML_ERR_RESOURCE_LIMIT; | 
| 722 | 0 |         return; | 
| 723 | 0 |     } | 
| 724 |  |  | 
| 725 | 52.0M |     if (buf->cap - buf->size <= len) { | 
| 726 | 102k |         if (xmlSBufGrow(buf, len) < 0) | 
| 727 | 0 |             return; | 
| 728 | 102k |     } | 
| 729 |  |  | 
| 730 | 52.0M |     if (len > 0) | 
| 731 | 52.0M |         memcpy(buf->mem + buf->size, str, len); | 
| 732 | 52.0M |     buf->size += len; | 
| 733 | 52.0M | } | 
| 734 |  |  | 
| 735 |  | static void | 
| 736 | 50.2M | xmlSBufAddCString(xmlSBuf *buf, const char *str, unsigned len) { | 
| 737 | 50.2M |     xmlSBufAddString(buf, (const xmlChar *) str, len); | 
| 738 | 50.2M | } | 
| 739 |  |  | 
| 740 |  | static void | 
| 741 | 129k | xmlSBufAddChar(xmlSBuf *buf, int c) { | 
| 742 | 129k |     xmlChar *end; | 
| 743 |  |  | 
| 744 | 129k |     if (buf->max - buf->size < 4) { | 
| 745 | 0 |         if (buf->code == XML_ERR_OK) | 
| 746 | 0 |             buf->code = XML_ERR_RESOURCE_LIMIT; | 
| 747 | 0 |         return; | 
| 748 | 0 |     } | 
| 749 |  |  | 
| 750 | 129k |     if (buf->cap - buf->size <= 4) { | 
| 751 | 5.45k |         if (xmlSBufGrow(buf, 4) < 0) | 
| 752 | 0 |             return; | 
| 753 | 5.45k |     } | 
| 754 |  |  | 
| 755 | 129k |     end = buf->mem + buf->size; | 
| 756 |  |  | 
| 757 | 129k |     if (c < 0x80) { | 
| 758 | 23.1k |         *end = (xmlChar) c; | 
| 759 | 23.1k |         buf->size += 1; | 
| 760 | 106k |     } else { | 
| 761 | 106k |         buf->size += xmlCopyCharMultiByte(end, c); | 
| 762 | 106k |     } | 
| 763 | 129k | } | 
| 764 |  |  | 
| 765 |  | static void | 
| 766 | 44.3M | xmlSBufAddReplChar(xmlSBuf *buf) { | 
| 767 | 44.3M |     xmlSBufAddCString(buf, "\xEF\xBF\xBD", 3); | 
| 768 | 44.3M | } | 
| 769 |  |  | 
| 770 |  | static void | 
| 771 | 0 | xmlSBufReportError(xmlSBuf *buf, xmlParserCtxtPtr ctxt, const char *errMsg) { | 
| 772 | 0 |     if (buf->code == XML_ERR_NO_MEMORY) | 
| 773 | 0 |         xmlCtxtErrMemory(ctxt); | 
| 774 | 0 |     else | 
| 775 | 0 |         xmlFatalErr(ctxt, buf->code, errMsg); | 
| 776 | 0 | } | 
| 777 |  |  | 
| 778 |  | static xmlChar * | 
| 779 |  | xmlSBufFinish(xmlSBuf *buf, int *sizeOut, xmlParserCtxtPtr ctxt, | 
| 780 | 91.7k |               const char *errMsg) { | 
| 781 | 91.7k |     if (buf->mem == NULL) { | 
| 782 | 16.4k |         buf->mem = xmlMalloc(1); | 
| 783 | 16.4k |         if (buf->mem == NULL) { | 
| 784 | 0 |             buf->code = XML_ERR_NO_MEMORY; | 
| 785 | 16.4k |         } else { | 
| 786 | 16.4k |             buf->mem[0] = 0; | 
| 787 | 16.4k |         } | 
| 788 | 75.3k |     } else { | 
| 789 | 75.3k |         buf->mem[buf->size] = 0; | 
| 790 | 75.3k |     } | 
| 791 |  |  | 
| 792 | 91.7k |     if (buf->code == XML_ERR_OK) { | 
| 793 | 91.7k |         if (sizeOut != NULL) | 
| 794 | 55.3k |             *sizeOut = buf->size; | 
| 795 | 91.7k |         return(buf->mem); | 
| 796 | 91.7k |     } | 
| 797 |  |  | 
| 798 | 0 |     xmlSBufReportError(buf, ctxt, errMsg); | 
| 799 |  | 
 | 
| 800 | 0 |     xmlFree(buf->mem); | 
| 801 |  | 
 | 
| 802 | 0 |     if (sizeOut != NULL) | 
| 803 | 0 |         *sizeOut = 0; | 
| 804 | 0 |     return(NULL); | 
| 805 | 91.7k | } | 
| 806 |  |  | 
| 807 |  | static void | 
| 808 | 545k | xmlSBufCleanup(xmlSBuf *buf, xmlParserCtxtPtr ctxt, const char *errMsg) { | 
| 809 | 545k |     if (buf->code != XML_ERR_OK) | 
| 810 | 0 |         xmlSBufReportError(buf, ctxt, errMsg); | 
| 811 |  |  | 
| 812 | 545k |     xmlFree(buf->mem); | 
| 813 | 545k | } | 
| 814 |  |  | 
| 815 |  | static int | 
| 816 |  | xmlUTF8MultibyteLen(xmlParserCtxtPtr ctxt, const xmlChar *str, | 
| 817 | 94.8M |                     const char *errMsg) { | 
| 818 | 94.8M |     int c = str[0]; | 
| 819 | 94.8M |     int c1 = str[1]; | 
| 820 |  |  | 
| 821 | 94.8M |     if ((c1 & 0xC0) != 0x80) | 
| 822 | 10.3M |         goto encoding_error; | 
| 823 |  |  | 
| 824 | 84.4M |     if (c < 0xE0) { | 
| 825 |  |         /* 2-byte sequence */ | 
| 826 | 83.1M |         if (c < 0xC2) | 
| 827 | 29.5M |             goto encoding_error; | 
| 828 |  |  | 
| 829 | 53.5M |         return(2); | 
| 830 | 83.1M |     } else { | 
| 831 | 1.33M |         int c2 = str[2]; | 
| 832 |  |  | 
| 833 | 1.33M |         if ((c2 & 0xC0) != 0x80) | 
| 834 | 5.02k |             goto encoding_error; | 
| 835 |  |  | 
| 836 | 1.32M |         if (c < 0xF0) { | 
| 837 |  |             /* 3-byte sequence */ | 
| 838 | 1.31M |             if (c == 0xE0) { | 
| 839 |  |                 /* overlong */ | 
| 840 | 1.20k |                 if (c1 < 0xA0) | 
| 841 | 557 |                     goto encoding_error; | 
| 842 | 1.31M |             } else if (c == 0xED) { | 
| 843 |  |                 /* surrogate */ | 
| 844 | 984 |                 if (c1 >= 0xA0) | 
| 845 | 613 |                     goto encoding_error; | 
| 846 | 1.31M |             } else if (c == 0xEF) { | 
| 847 |  |                 /* U+FFFE and U+FFFF are invalid Chars */ | 
| 848 | 1.29M |                 if ((c1 == 0xBF) && (c2 >= 0xBE)) | 
| 849 | 645 |                     xmlFatalErrMsg(ctxt, XML_ERR_INVALID_CHAR, errMsg); | 
| 850 | 1.29M |             } | 
| 851 |  |  | 
| 852 | 1.31M |             return(3); | 
| 853 | 1.31M |         } else { | 
| 854 |  |             /* 4-byte sequence */ | 
| 855 | 14.2k |             if ((str[3] & 0xC0) != 0x80) | 
| 856 | 804 |                 goto encoding_error; | 
| 857 | 13.4k |             if (c == 0xF0) { | 
| 858 |  |                 /* overlong */ | 
| 859 | 667 |                 if (c1 < 0x90) | 
| 860 | 264 |                     goto encoding_error; | 
| 861 | 12.8k |             } else if (c >= 0xF4) { | 
| 862 |  |                 /* greater than 0x10FFFF */ | 
| 863 | 2.36k |                 if ((c > 0xF4) || (c1 >= 0x90)) | 
| 864 | 1.96k |                     goto encoding_error; | 
| 865 | 2.36k |             } | 
| 866 |  |  | 
| 867 | 11.2k |             return(4); | 
| 868 | 13.4k |         } | 
| 869 | 1.32M |     } | 
| 870 |  |  | 
| 871 | 39.9M | encoding_error: | 
| 872 |  |     /* Only report the first error */ | 
| 873 | 39.9M |     if ((ctxt->input->flags & XML_INPUT_ENCODING_ERROR) == 0) { | 
| 874 | 1.89k |         xmlCtxtErrIO(ctxt, XML_ERR_INVALID_ENCODING, NULL); | 
| 875 | 1.89k |         ctxt->input->flags |= XML_INPUT_ENCODING_ERROR; | 
| 876 | 1.89k |     } | 
| 877 |  |  | 
| 878 | 39.9M |     return(0); | 
| 879 | 84.4M | } | 
| 880 |  |  | 
| 881 |  | /************************************************************************ | 
| 882 |  |  *                  * | 
| 883 |  |  *    SAX2 defaulted attributes handling      * | 
| 884 |  |  *                  * | 
| 885 |  |  ************************************************************************/ | 
| 886 |  |  | 
| 887 |  | /** | 
| 888 |  |  * Final initialization of the parser context before starting to parse. | 
| 889 |  |  * | 
| 890 |  |  * This accounts for users modifying struct members of parser context | 
| 891 |  |  * directly. | 
| 892 |  |  * | 
| 893 |  |  * @param ctxt  an XML parser context | 
| 894 |  |  */ | 
| 895 |  | static void | 
| 896 | 17.4k | xmlCtxtInitializeLate(xmlParserCtxtPtr ctxt) { | 
| 897 | 17.4k |     xmlSAXHandlerPtr sax; | 
| 898 |  |  | 
| 899 |  |     /* Avoid unused variable warning if features are disabled. */ | 
| 900 | 17.4k |     (void) sax; | 
| 901 |  |  | 
| 902 |  |     /* | 
| 903 |  |      * Changing the SAX struct directly is still widespread practice | 
| 904 |  |      * in internal and external code. | 
| 905 |  |      */ | 
| 906 | 17.4k |     if (ctxt == NULL) return; | 
| 907 | 17.4k |     sax = ctxt->sax; | 
| 908 | 17.4k | #ifdef LIBXML_SAX1_ENABLED | 
| 909 |  |     /* | 
| 910 |  |      * Only enable SAX2 if there SAX2 element handlers, except when there | 
| 911 |  |      * are no element handlers at all. | 
| 912 |  |      */ | 
| 913 | 17.4k |     if (((ctxt->options & XML_PARSE_SAX1) == 0) && | 
| 914 | 17.4k |         (sax) && | 
| 915 | 17.4k |         (sax->initialized == XML_SAX2_MAGIC) && | 
| 916 | 17.4k |         ((sax->startElementNs != NULL) || | 
| 917 | 0 |          (sax->endElementNs != NULL) || | 
| 918 | 0 |          ((sax->startElement == NULL) && (sax->endElement == NULL)))) | 
| 919 | 17.4k |         ctxt->sax2 = 1; | 
| 920 |  | #else | 
| 921 |  |     ctxt->sax2 = 1; | 
| 922 |  | #endif /* LIBXML_SAX1_ENABLED */ | 
| 923 |  |  | 
| 924 |  |     /* | 
| 925 |  |      * Some users replace the dictionary directly in the context struct. | 
| 926 |  |      * We really need an API function to do that cleanly. | 
| 927 |  |      */ | 
| 928 | 17.4k |     ctxt->str_xml = xmlDictLookup(ctxt->dict, BAD_CAST "xml", 3); | 
| 929 | 17.4k |     ctxt->str_xmlns = xmlDictLookup(ctxt->dict, BAD_CAST "xmlns", 5); | 
| 930 | 17.4k |     ctxt->str_xml_ns = xmlDictLookup(ctxt->dict, XML_XML_NAMESPACE, 36); | 
| 931 | 17.4k |     if ((ctxt->str_xml==NULL) || (ctxt->str_xmlns==NULL) || | 
| 932 | 17.4k |     (ctxt->str_xml_ns == NULL)) { | 
| 933 | 0 |         xmlErrMemory(ctxt); | 
| 934 | 0 |     } | 
| 935 |  |  | 
| 936 | 17.4k |     xmlDictSetLimit(ctxt->dict, | 
| 937 | 17.4k |                     (ctxt->options & XML_PARSE_HUGE) ? | 
| 938 | 17.4k |                         0 : | 
| 939 | 17.4k |                         XML_MAX_DICTIONARY_LIMIT); | 
| 940 |  |  | 
| 941 | 17.4k | #ifdef LIBXML_VALID_ENABLED | 
| 942 | 17.4k |     if (ctxt->validate) | 
| 943 | 0 |         ctxt->vctxt.flags |= XML_VCTXT_VALIDATE; | 
| 944 | 17.4k |     else | 
| 945 | 17.4k |         ctxt->vctxt.flags &= ~XML_VCTXT_VALIDATE; | 
| 946 | 17.4k | #endif /* LIBXML_VALID_ENABLED */ | 
| 947 | 17.4k | } | 
| 948 |  |  | 
| 949 |  | typedef struct { | 
| 950 |  |     xmlHashedString prefix; | 
| 951 |  |     xmlHashedString name; | 
| 952 |  |     xmlHashedString value; | 
| 953 |  |     const xmlChar *valueEnd; | 
| 954 |  |     int external; | 
| 955 |  |     int expandedSize; | 
| 956 |  | } xmlDefAttr; | 
| 957 |  |  | 
| 958 |  | typedef struct _xmlDefAttrs xmlDefAttrs; | 
| 959 |  | typedef xmlDefAttrs *xmlDefAttrsPtr; | 
| 960 |  | struct _xmlDefAttrs { | 
| 961 |  |     int nbAttrs;  /* number of defaulted attributes on that element */ | 
| 962 |  |     int maxAttrs;       /* the size of the array */ | 
| 963 |  | #if __STDC_VERSION__ >= 199901L | 
| 964 |  |     /* Using a C99 flexible array member avoids UBSan errors. */ | 
| 965 |  |     xmlDefAttr attrs[] ATTRIBUTE_COUNTED_BY(maxAttrs); | 
| 966 |  | #else | 
| 967 |  |     xmlDefAttr attrs[1]; | 
| 968 |  | #endif | 
| 969 |  | }; | 
| 970 |  |  | 
| 971 |  | /** | 
| 972 |  |  * Normalize the space in non CDATA attribute values: | 
| 973 |  |  * If the attribute type is not CDATA, then the XML processor MUST further | 
| 974 |  |  * process the normalized attribute value by discarding any leading and | 
| 975 |  |  * trailing space (\#x20) characters, and by replacing sequences of space | 
| 976 |  |  * (\#x20) characters by a single space (\#x20) character. | 
| 977 |  |  * Note that the size of dst need to be at least src, and if one doesn't need | 
| 978 |  |  * to preserve dst (and it doesn't come from a dictionary or read-only) then | 
| 979 |  |  * passing src as dst is just fine. | 
| 980 |  |  * | 
| 981 |  |  * @param src  the source string | 
| 982 |  |  * @param dst  the target string | 
| 983 |  |  * @returns a pointer to the normalized value (dst) or NULL if no conversion | 
| 984 |  |  *         is needed. | 
| 985 |  |  */ | 
| 986 |  | static xmlChar * | 
| 987 |  | xmlAttrNormalizeSpace(const xmlChar *src, xmlChar *dst) | 
| 988 | 15.1k | { | 
| 989 | 15.1k |     if ((src == NULL) || (dst == NULL)) | 
| 990 | 0 |         return(NULL); | 
| 991 |  |  | 
| 992 | 15.6k |     while (*src == 0x20) src++; | 
| 993 | 4.25M |     while (*src != 0) { | 
| 994 | 4.23M |   if (*src == 0x20) { | 
| 995 | 57.6k |       while (*src == 0x20) src++; | 
| 996 | 4.59k |       if (*src != 0) | 
| 997 | 4.34k |     *dst++ = 0x20; | 
| 998 | 4.23M |   } else { | 
| 999 | 4.23M |       *dst++ = *src++; | 
| 1000 | 4.23M |   } | 
| 1001 | 4.23M |     } | 
| 1002 | 15.1k |     *dst = 0; | 
| 1003 | 15.1k |     if (dst == src) | 
| 1004 | 14.6k |        return(NULL); | 
| 1005 | 426 |     return(dst); | 
| 1006 | 15.1k | } | 
| 1007 |  |  | 
| 1008 |  | /** | 
| 1009 |  |  * Add a defaulted attribute for an element | 
| 1010 |  |  * | 
| 1011 |  |  * @param ctxt  an XML parser context | 
| 1012 |  |  * @param fullname  the element fullname | 
| 1013 |  |  * @param fullattr  the attribute fullname | 
| 1014 |  |  * @param value  the attribute value | 
| 1015 |  |  */ | 
| 1016 |  | static void | 
| 1017 |  | xmlAddDefAttrs(xmlParserCtxtPtr ctxt, | 
| 1018 |  |                const xmlChar *fullname, | 
| 1019 |  |                const xmlChar *fullattr, | 
| 1020 | 16.1k |                const xmlChar *value) { | 
| 1021 | 16.1k |     xmlDefAttrsPtr defaults; | 
| 1022 | 16.1k |     xmlDefAttr *attr; | 
| 1023 | 16.1k |     int len, expandedSize; | 
| 1024 | 16.1k |     xmlHashedString name; | 
| 1025 | 16.1k |     xmlHashedString prefix; | 
| 1026 | 16.1k |     xmlHashedString hvalue; | 
| 1027 | 16.1k |     const xmlChar *localname; | 
| 1028 |  |  | 
| 1029 |  |     /* | 
| 1030 |  |      * Allows to detect attribute redefinitions | 
| 1031 |  |      */ | 
| 1032 | 16.1k |     if (ctxt->attsSpecial != NULL) { | 
| 1033 | 14.2k |         if (xmlHashLookup2(ctxt->attsSpecial, fullname, fullattr) != NULL) | 
| 1034 | 3.74k |       return; | 
| 1035 | 14.2k |     } | 
| 1036 |  |  | 
| 1037 | 12.3k |     if (ctxt->attsDefault == NULL) { | 
| 1038 | 1.82k |         ctxt->attsDefault = xmlHashCreateDict(10, ctxt->dict); | 
| 1039 | 1.82k |   if (ctxt->attsDefault == NULL) | 
| 1040 | 0 |       goto mem_error; | 
| 1041 | 1.82k |     } | 
| 1042 |  |  | 
| 1043 |  |     /* | 
| 1044 |  |      * split the element name into prefix:localname , the string found | 
| 1045 |  |      * are within the DTD and then not associated to namespace names. | 
| 1046 |  |      */ | 
| 1047 | 12.3k |     localname = xmlSplitQName3(fullname, &len); | 
| 1048 | 12.3k |     if (localname == NULL) { | 
| 1049 | 10.3k |         name = xmlDictLookupHashed(ctxt->dict, fullname, -1); | 
| 1050 | 10.3k |   prefix.name = NULL; | 
| 1051 | 10.3k |     } else { | 
| 1052 | 2.01k |         name = xmlDictLookupHashed(ctxt->dict, localname, -1); | 
| 1053 | 2.01k |   prefix = xmlDictLookupHashed(ctxt->dict, fullname, len); | 
| 1054 | 2.01k |         if (prefix.name == NULL) | 
| 1055 | 0 |             goto mem_error; | 
| 1056 | 2.01k |     } | 
| 1057 | 12.3k |     if (name.name == NULL) | 
| 1058 | 0 |         goto mem_error; | 
| 1059 |  |  | 
| 1060 |  |     /* | 
| 1061 |  |      * make sure there is some storage | 
| 1062 |  |      */ | 
| 1063 | 12.3k |     defaults = xmlHashLookup2(ctxt->attsDefault, name.name, prefix.name); | 
| 1064 | 12.3k |     if ((defaults == NULL) || | 
| 1065 | 9.79k |         (defaults->nbAttrs >= defaults->maxAttrs)) { | 
| 1066 | 3.99k |         xmlDefAttrsPtr temp; | 
| 1067 | 3.99k |         int newSize; | 
| 1068 |  |  | 
| 1069 | 3.99k |         if (defaults == NULL) { | 
| 1070 | 2.58k |             newSize = 4; | 
| 1071 | 2.58k |         } else { | 
| 1072 | 1.40k |             if ((defaults->maxAttrs >= XML_MAX_ATTRS) || | 
| 1073 | 1.40k |                 ((size_t) defaults->maxAttrs > | 
| 1074 | 1.40k |                      SIZE_MAX / 2 / sizeof(temp[0]) - sizeof(*defaults))) | 
| 1075 | 0 |                 goto mem_error; | 
| 1076 |  |  | 
| 1077 | 1.40k |             if (defaults->maxAttrs > XML_MAX_ATTRS / 2) | 
| 1078 | 0 |                 newSize = XML_MAX_ATTRS; | 
| 1079 | 1.40k |             else | 
| 1080 | 1.40k |                 newSize = defaults->maxAttrs * 2; | 
| 1081 | 1.40k |         } | 
| 1082 | 3.99k |         temp = xmlRealloc(defaults, | 
| 1083 | 3.99k |                           sizeof(*defaults) + newSize * sizeof(xmlDefAttr)); | 
| 1084 | 3.99k |   if (temp == NULL) | 
| 1085 | 0 |       goto mem_error; | 
| 1086 | 3.99k |         if (defaults == NULL) | 
| 1087 | 2.58k |             temp->nbAttrs = 0; | 
| 1088 | 3.99k |   temp->maxAttrs = newSize; | 
| 1089 | 3.99k |         defaults = temp; | 
| 1090 | 3.99k |   if (xmlHashUpdateEntry2(ctxt->attsDefault, name.name, prefix.name, | 
| 1091 | 3.99k |                           defaults, NULL) < 0) { | 
| 1092 | 0 |       xmlFree(defaults); | 
| 1093 | 0 |       goto mem_error; | 
| 1094 | 0 |   } | 
| 1095 | 3.99k |     } | 
| 1096 |  |  | 
| 1097 |  |     /* | 
| 1098 |  |      * Split the attribute name into prefix:localname , the string found | 
| 1099 |  |      * are within the DTD and hen not associated to namespace names. | 
| 1100 |  |      */ | 
| 1101 | 12.3k |     localname = xmlSplitQName3(fullattr, &len); | 
| 1102 | 12.3k |     if (localname == NULL) { | 
| 1103 | 6.89k |         name = xmlDictLookupHashed(ctxt->dict, fullattr, -1); | 
| 1104 | 6.89k |   prefix.name = NULL; | 
| 1105 | 6.89k |     } else { | 
| 1106 | 5.47k |         name = xmlDictLookupHashed(ctxt->dict, localname, -1); | 
| 1107 | 5.47k |   prefix = xmlDictLookupHashed(ctxt->dict, fullattr, len); | 
| 1108 | 5.47k |         if (prefix.name == NULL) | 
| 1109 | 0 |             goto mem_error; | 
| 1110 | 5.47k |     } | 
| 1111 | 12.3k |     if (name.name == NULL) | 
| 1112 | 0 |         goto mem_error; | 
| 1113 |  |  | 
| 1114 |  |     /* intern the string and precompute the end */ | 
| 1115 | 12.3k |     len = strlen((const char *) value); | 
| 1116 | 12.3k |     hvalue = xmlDictLookupHashed(ctxt->dict, value, len); | 
| 1117 | 12.3k |     if (hvalue.name == NULL) | 
| 1118 | 0 |         goto mem_error; | 
| 1119 |  |  | 
| 1120 | 12.3k |     expandedSize = strlen((const char *) name.name); | 
| 1121 | 12.3k |     if (prefix.name != NULL) | 
| 1122 | 5.47k |         expandedSize += strlen((const char *) prefix.name); | 
| 1123 | 12.3k |     expandedSize += len; | 
| 1124 |  |  | 
| 1125 | 12.3k |     attr = &defaults->attrs[defaults->nbAttrs++]; | 
| 1126 | 12.3k |     attr->name = name; | 
| 1127 | 12.3k |     attr->prefix = prefix; | 
| 1128 | 12.3k |     attr->value = hvalue; | 
| 1129 | 12.3k |     attr->valueEnd = hvalue.name + len; | 
| 1130 | 12.3k |     attr->external = PARSER_EXTERNAL(ctxt); | 
| 1131 | 12.3k |     attr->expandedSize = expandedSize; | 
| 1132 |  |  | 
| 1133 | 12.3k |     return; | 
| 1134 |  |  | 
| 1135 | 0 | mem_error: | 
| 1136 | 0 |     xmlErrMemory(ctxt); | 
| 1137 | 0 | } | 
| 1138 |  |  | 
| 1139 |  | /** | 
| 1140 |  |  * Register this attribute type | 
| 1141 |  |  * | 
| 1142 |  |  * @param ctxt  an XML parser context | 
| 1143 |  |  * @param fullname  the element fullname | 
| 1144 |  |  * @param fullattr  the attribute fullname | 
| 1145 |  |  * @param type  the attribute type | 
| 1146 |  |  */ | 
| 1147 |  | static void | 
| 1148 |  | xmlAddSpecialAttr(xmlParserCtxtPtr ctxt, | 
| 1149 |  |       const xmlChar *fullname, | 
| 1150 |  |       const xmlChar *fullattr, | 
| 1151 |  |       int type) | 
| 1152 | 17.7k | { | 
| 1153 | 17.7k |     if (ctxt->attsSpecial == NULL) { | 
| 1154 | 2.18k |         ctxt->attsSpecial = xmlHashCreateDict(10, ctxt->dict); | 
| 1155 | 2.18k |   if (ctxt->attsSpecial == NULL) | 
| 1156 | 0 |       goto mem_error; | 
| 1157 | 2.18k |     } | 
| 1158 |  |  | 
| 1159 | 17.7k |     if (PARSER_EXTERNAL(ctxt)) | 
| 1160 | 0 |         type |= XML_SPECIAL_EXTERNAL; | 
| 1161 |  |  | 
| 1162 | 17.7k |     if (xmlHashAdd2(ctxt->attsSpecial, fullname, fullattr, | 
| 1163 | 17.7k |                     XML_INT_TO_PTR(type)) < 0) | 
| 1164 | 0 |         goto mem_error; | 
| 1165 | 17.7k |     return; | 
| 1166 |  |  | 
| 1167 | 17.7k | mem_error: | 
| 1168 | 0 |     xmlErrMemory(ctxt); | 
| 1169 | 0 | } | 
| 1170 |  |  | 
| 1171 |  | /** | 
| 1172 |  |  * Removes CDATA attributes from the special attribute table | 
| 1173 |  |  */ | 
| 1174 |  | static void | 
| 1175 |  | xmlCleanSpecialAttrCallback(void *payload, void *data, | 
| 1176 |  |                             const xmlChar *fullname, const xmlChar *fullattr, | 
| 1177 | 12.7k |                             const xmlChar *unused ATTRIBUTE_UNUSED) { | 
| 1178 | 12.7k |     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) data; | 
| 1179 |  |  | 
| 1180 | 12.7k |     if (XML_PTR_TO_INT(payload) == XML_ATTRIBUTE_CDATA) { | 
| 1181 | 1.01k |         xmlHashRemoveEntry2(ctxt->attsSpecial, fullname, fullattr, NULL); | 
| 1182 | 1.01k |     } | 
| 1183 | 12.7k | } | 
| 1184 |  |  | 
| 1185 |  | /** | 
| 1186 |  |  * Trim the list of attributes defined to remove all those of type | 
| 1187 |  |  * CDATA as they are not special. This call should be done when finishing | 
| 1188 |  |  * to parse the DTD and before starting to parse the document root. | 
| 1189 |  |  * | 
| 1190 |  |  * @param ctxt  an XML parser context | 
| 1191 |  |  */ | 
| 1192 |  | static void | 
| 1193 |  | xmlCleanSpecialAttr(xmlParserCtxtPtr ctxt) | 
| 1194 | 7.62k | { | 
| 1195 | 7.62k |     if (ctxt->attsSpecial == NULL) | 
| 1196 | 5.44k |         return; | 
| 1197 |  |  | 
| 1198 | 2.18k |     xmlHashScanFull(ctxt->attsSpecial, xmlCleanSpecialAttrCallback, ctxt); | 
| 1199 |  |  | 
| 1200 | 2.18k |     if (xmlHashSize(ctxt->attsSpecial) == 0) { | 
| 1201 | 50 |         xmlHashFree(ctxt->attsSpecial, NULL); | 
| 1202 | 50 |         ctxt->attsSpecial = NULL; | 
| 1203 | 50 |     } | 
| 1204 | 2.18k | } | 
| 1205 |  |  | 
| 1206 |  | /** | 
| 1207 |  |  * Checks that the value conforms to the LanguageID production: | 
| 1208 |  |  * | 
| 1209 |  |  * @deprecated Internal function, do not use. | 
| 1210 |  |  * | 
| 1211 |  |  * NOTE: this is somewhat deprecated, those productions were removed from | 
| 1212 |  |  * the XML Second edition. | 
| 1213 |  |  * | 
| 1214 |  |  *     [33] LanguageID ::= Langcode ('-' Subcode)* | 
| 1215 |  |  *     [34] Langcode ::= ISO639Code |  IanaCode |  UserCode | 
| 1216 |  |  *     [35] ISO639Code ::= ([a-z] | [A-Z]) ([a-z] | [A-Z]) | 
| 1217 |  |  *     [36] IanaCode ::= ('i' | 'I') '-' ([a-z] | [A-Z])+ | 
| 1218 |  |  *     [37] UserCode ::= ('x' | 'X') '-' ([a-z] | [A-Z])+ | 
| 1219 |  |  *     [38] Subcode ::= ([a-z] | [A-Z])+ | 
| 1220 |  |  * | 
| 1221 |  |  * The current REC reference the successors of RFC 1766, currently 5646 | 
| 1222 |  |  * | 
| 1223 |  |  * http://www.rfc-editor.org/rfc/rfc5646.txt | 
| 1224 |  |  * | 
| 1225 |  |  *     langtag       = language | 
| 1226 |  |  *                     ["-" script] | 
| 1227 |  |  *                     ["-" region] | 
| 1228 |  |  *                     *("-" variant) | 
| 1229 |  |  *                     *("-" extension) | 
| 1230 |  |  *                     ["-" privateuse] | 
| 1231 |  |  *     language      = 2*3ALPHA            ; shortest ISO 639 code | 
| 1232 |  |  *                     ["-" extlang]       ; sometimes followed by | 
| 1233 |  |  *                                         ; extended language subtags | 
| 1234 |  |  *                   / 4ALPHA              ; or reserved for future use | 
| 1235 |  |  *                   / 5*8ALPHA            ; or registered language subtag | 
| 1236 |  |  * | 
| 1237 |  |  *     extlang       = 3ALPHA              ; selected ISO 639 codes | 
| 1238 |  |  *                     *2("-" 3ALPHA)      ; permanently reserved | 
| 1239 |  |  * | 
| 1240 |  |  *     script        = 4ALPHA              ; ISO 15924 code | 
| 1241 |  |  * | 
| 1242 |  |  *     region        = 2ALPHA              ; ISO 3166-1 code | 
| 1243 |  |  *                   / 3DIGIT              ; UN M.49 code | 
| 1244 |  |  * | 
| 1245 |  |  *     variant       = 5*8alphanum         ; registered variants | 
| 1246 |  |  *                   / (DIGIT 3alphanum) | 
| 1247 |  |  * | 
| 1248 |  |  *     extension     = singleton 1*("-" (2*8alphanum)) | 
| 1249 |  |  * | 
| 1250 |  |  *                                         ; Single alphanumerics | 
| 1251 |  |  *                                         ; "x" reserved for private use | 
| 1252 |  |  *     singleton     = DIGIT               ; 0 - 9 | 
| 1253 |  |  *                   / %x41-57             ; A - W | 
| 1254 |  |  *                   / %x59-5A             ; Y - Z | 
| 1255 |  |  *                   / %x61-77             ; a - w | 
| 1256 |  |  *                   / %x79-7A             ; y - z | 
| 1257 |  |  * | 
| 1258 |  |  * it sounds right to still allow Irregular i-xxx IANA and user codes too | 
| 1259 |  |  * The parser below doesn't try to cope with extension or privateuse | 
| 1260 |  |  * that could be added but that's not interoperable anyway | 
| 1261 |  |  * | 
| 1262 |  |  * @param lang  pointer to the string value | 
| 1263 |  |  * @returns 1 if correct 0 otherwise | 
| 1264 |  |  **/ | 
| 1265 |  | int | 
| 1266 |  | xmlCheckLanguageID(const xmlChar * lang) | 
| 1267 | 0 | { | 
| 1268 | 0 |     const xmlChar *cur = lang, *nxt; | 
| 1269 |  | 
 | 
| 1270 | 0 |     if (cur == NULL) | 
| 1271 | 0 |         return (0); | 
| 1272 | 0 |     if (((cur[0] == 'i') && (cur[1] == '-')) || | 
| 1273 | 0 |         ((cur[0] == 'I') && (cur[1] == '-')) || | 
| 1274 | 0 |         ((cur[0] == 'x') && (cur[1] == '-')) || | 
| 1275 | 0 |         ((cur[0] == 'X') && (cur[1] == '-'))) { | 
| 1276 |  |         /* | 
| 1277 |  |          * Still allow IANA code and user code which were coming | 
| 1278 |  |          * from the previous version of the XML-1.0 specification | 
| 1279 |  |          * it's deprecated but we should not fail | 
| 1280 |  |          */ | 
| 1281 | 0 |         cur += 2; | 
| 1282 | 0 |         while (((cur[0] >= 'A') && (cur[0] <= 'Z')) || | 
| 1283 | 0 |                ((cur[0] >= 'a') && (cur[0] <= 'z'))) | 
| 1284 | 0 |             cur++; | 
| 1285 | 0 |         return(cur[0] == 0); | 
| 1286 | 0 |     } | 
| 1287 | 0 |     nxt = cur; | 
| 1288 | 0 |     while (((nxt[0] >= 'A') && (nxt[0] <= 'Z')) || | 
| 1289 | 0 |            ((nxt[0] >= 'a') && (nxt[0] <= 'z'))) | 
| 1290 | 0 |            nxt++; | 
| 1291 | 0 |     if (nxt - cur >= 4) { | 
| 1292 |  |         /* | 
| 1293 |  |          * Reserved | 
| 1294 |  |          */ | 
| 1295 | 0 |         if ((nxt - cur > 8) || (nxt[0] != 0)) | 
| 1296 | 0 |             return(0); | 
| 1297 | 0 |         return(1); | 
| 1298 | 0 |     } | 
| 1299 | 0 |     if (nxt - cur < 2) | 
| 1300 | 0 |         return(0); | 
| 1301 |  |     /* we got an ISO 639 code */ | 
| 1302 | 0 |     if (nxt[0] == 0) | 
| 1303 | 0 |         return(1); | 
| 1304 | 0 |     if (nxt[0] != '-') | 
| 1305 | 0 |         return(0); | 
| 1306 |  |  | 
| 1307 | 0 |     nxt++; | 
| 1308 | 0 |     cur = nxt; | 
| 1309 |  |     /* now we can have extlang or script or region or variant */ | 
| 1310 | 0 |     if ((nxt[0] >= '0') && (nxt[0] <= '9')) | 
| 1311 | 0 |         goto region_m49; | 
| 1312 |  |  | 
| 1313 | 0 |     while (((nxt[0] >= 'A') && (nxt[0] <= 'Z')) || | 
| 1314 | 0 |            ((nxt[0] >= 'a') && (nxt[0] <= 'z'))) | 
| 1315 | 0 |            nxt++; | 
| 1316 | 0 |     if (nxt - cur == 4) | 
| 1317 | 0 |         goto script; | 
| 1318 | 0 |     if (nxt - cur == 2) | 
| 1319 | 0 |         goto region; | 
| 1320 | 0 |     if ((nxt - cur >= 5) && (nxt - cur <= 8)) | 
| 1321 | 0 |         goto variant; | 
| 1322 | 0 |     if (nxt - cur != 3) | 
| 1323 | 0 |         return(0); | 
| 1324 |  |     /* we parsed an extlang */ | 
| 1325 | 0 |     if (nxt[0] == 0) | 
| 1326 | 0 |         return(1); | 
| 1327 | 0 |     if (nxt[0] != '-') | 
| 1328 | 0 |         return(0); | 
| 1329 |  |  | 
| 1330 | 0 |     nxt++; | 
| 1331 | 0 |     cur = nxt; | 
| 1332 |  |     /* now we can have script or region or variant */ | 
| 1333 | 0 |     if ((nxt[0] >= '0') && (nxt[0] <= '9')) | 
| 1334 | 0 |         goto region_m49; | 
| 1335 |  |  | 
| 1336 | 0 |     while (((nxt[0] >= 'A') && (nxt[0] <= 'Z')) || | 
| 1337 | 0 |            ((nxt[0] >= 'a') && (nxt[0] <= 'z'))) | 
| 1338 | 0 |            nxt++; | 
| 1339 | 0 |     if (nxt - cur == 2) | 
| 1340 | 0 |         goto region; | 
| 1341 | 0 |     if ((nxt - cur >= 5) && (nxt - cur <= 8)) | 
| 1342 | 0 |         goto variant; | 
| 1343 | 0 |     if (nxt - cur != 4) | 
| 1344 | 0 |         return(0); | 
| 1345 |  |     /* we parsed a script */ | 
| 1346 | 0 | script: | 
| 1347 | 0 |     if (nxt[0] == 0) | 
| 1348 | 0 |         return(1); | 
| 1349 | 0 |     if (nxt[0] != '-') | 
| 1350 | 0 |         return(0); | 
| 1351 |  |  | 
| 1352 | 0 |     nxt++; | 
| 1353 | 0 |     cur = nxt; | 
| 1354 |  |     /* now we can have region or variant */ | 
| 1355 | 0 |     if ((nxt[0] >= '0') && (nxt[0] <= '9')) | 
| 1356 | 0 |         goto region_m49; | 
| 1357 |  |  | 
| 1358 | 0 |     while (((nxt[0] >= 'A') && (nxt[0] <= 'Z')) || | 
| 1359 | 0 |            ((nxt[0] >= 'a') && (nxt[0] <= 'z'))) | 
| 1360 | 0 |            nxt++; | 
| 1361 |  | 
 | 
| 1362 | 0 |     if ((nxt - cur >= 5) && (nxt - cur <= 8)) | 
| 1363 | 0 |         goto variant; | 
| 1364 | 0 |     if (nxt - cur != 2) | 
| 1365 | 0 |         return(0); | 
| 1366 |  |     /* we parsed a region */ | 
| 1367 | 0 | region: | 
| 1368 | 0 |     if (nxt[0] == 0) | 
| 1369 | 0 |         return(1); | 
| 1370 | 0 |     if (nxt[0] != '-') | 
| 1371 | 0 |         return(0); | 
| 1372 |  |  | 
| 1373 | 0 |     nxt++; | 
| 1374 | 0 |     cur = nxt; | 
| 1375 |  |     /* now we can just have a variant */ | 
| 1376 | 0 |     while (((nxt[0] >= 'A') && (nxt[0] <= 'Z')) || | 
| 1377 | 0 |            ((nxt[0] >= 'a') && (nxt[0] <= 'z'))) | 
| 1378 | 0 |            nxt++; | 
| 1379 |  | 
 | 
| 1380 | 0 |     if ((nxt - cur < 5) || (nxt - cur > 8)) | 
| 1381 | 0 |         return(0); | 
| 1382 |  |  | 
| 1383 |  |     /* we parsed a variant */ | 
| 1384 | 0 | variant: | 
| 1385 | 0 |     if (nxt[0] == 0) | 
| 1386 | 0 |         return(1); | 
| 1387 | 0 |     if (nxt[0] != '-') | 
| 1388 | 0 |         return(0); | 
| 1389 |  |     /* extensions and private use subtags not checked */ | 
| 1390 | 0 |     return (1); | 
| 1391 |  |  | 
| 1392 | 0 | region_m49: | 
| 1393 | 0 |     if (((nxt[1] >= '0') && (nxt[1] <= '9')) && | 
| 1394 | 0 |         ((nxt[2] >= '0') && (nxt[2] <= '9'))) { | 
| 1395 | 0 |         nxt += 3; | 
| 1396 | 0 |         goto region; | 
| 1397 | 0 |     } | 
| 1398 | 0 |     return(0); | 
| 1399 | 0 | } | 
| 1400 |  |  | 
| 1401 |  | /************************************************************************ | 
| 1402 |  |  *                  * | 
| 1403 |  |  *    Parser stacks related functions and macros    * | 
| 1404 |  |  *                  * | 
| 1405 |  |  ************************************************************************/ | 
| 1406 |  |  | 
| 1407 |  | static xmlChar * | 
| 1408 |  | xmlParseStringEntityRef(xmlParserCtxtPtr ctxt, const xmlChar **str); | 
| 1409 |  |  | 
| 1410 |  | /** | 
| 1411 |  |  * Create a new namespace database. | 
| 1412 |  |  * | 
| 1413 |  |  * @returns the new obejct. | 
| 1414 |  |  */ | 
| 1415 |  | xmlParserNsData * | 
| 1416 | 17.4k | xmlParserNsCreate(void) { | 
| 1417 | 17.4k |     xmlParserNsData *nsdb = xmlMalloc(sizeof(*nsdb)); | 
| 1418 |  |  | 
| 1419 | 17.4k |     if (nsdb == NULL) | 
| 1420 | 0 |         return(NULL); | 
| 1421 | 17.4k |     memset(nsdb, 0, sizeof(*nsdb)); | 
| 1422 | 17.4k |     nsdb->defaultNsIndex = INT_MAX; | 
| 1423 |  |  | 
| 1424 | 17.4k |     return(nsdb); | 
| 1425 | 17.4k | } | 
| 1426 |  |  | 
| 1427 |  | /** | 
| 1428 |  |  * Free a namespace database. | 
| 1429 |  |  * | 
| 1430 |  |  * @param nsdb  namespace database | 
| 1431 |  |  */ | 
| 1432 |  | void | 
| 1433 | 17.4k | xmlParserNsFree(xmlParserNsData *nsdb) { | 
| 1434 | 17.4k |     if (nsdb == NULL) | 
| 1435 | 0 |         return; | 
| 1436 |  |  | 
| 1437 | 17.4k |     xmlFree(nsdb->extra); | 
| 1438 | 17.4k |     xmlFree(nsdb->hash); | 
| 1439 | 17.4k |     xmlFree(nsdb); | 
| 1440 | 17.4k | } | 
| 1441 |  |  | 
| 1442 |  | /** | 
| 1443 |  |  * Reset a namespace database. | 
| 1444 |  |  * | 
| 1445 |  |  * @param nsdb  namespace database | 
| 1446 |  |  */ | 
| 1447 |  | static void | 
| 1448 | 0 | xmlParserNsReset(xmlParserNsData *nsdb) { | 
| 1449 | 0 |     if (nsdb == NULL) | 
| 1450 | 0 |         return; | 
| 1451 |  |  | 
| 1452 | 0 |     nsdb->hashElems = 0; | 
| 1453 | 0 |     nsdb->elementId = 0; | 
| 1454 | 0 |     nsdb->defaultNsIndex = INT_MAX; | 
| 1455 |  | 
 | 
| 1456 | 0 |     if (nsdb->hash) | 
| 1457 | 0 |         memset(nsdb->hash, 0, nsdb->hashSize * sizeof(nsdb->hash[0])); | 
| 1458 | 0 | } | 
| 1459 |  |  | 
| 1460 |  | /** | 
| 1461 |  |  * Signal that a new element has started. | 
| 1462 |  |  * | 
| 1463 |  |  * @param nsdb  namespace database | 
| 1464 |  |  * @returns 0 on success, -1 if the element counter overflowed. | 
| 1465 |  |  */ | 
| 1466 |  | static int | 
| 1467 | 3.27M | xmlParserNsStartElement(xmlParserNsData *nsdb) { | 
| 1468 | 3.27M |     if (nsdb->elementId == UINT_MAX) | 
| 1469 | 0 |         return(-1); | 
| 1470 | 3.27M |     nsdb->elementId++; | 
| 1471 |  |  | 
| 1472 | 3.27M |     return(0); | 
| 1473 | 3.27M | } | 
| 1474 |  |  | 
| 1475 |  | /** | 
| 1476 |  |  * Lookup namespace with given prefix. If `bucketPtr` is non-NULL, it will | 
| 1477 |  |  * be set to the matching bucket, or the first empty bucket if no match | 
| 1478 |  |  * was found. | 
| 1479 |  |  * | 
| 1480 |  |  * @param ctxt  parser context | 
| 1481 |  |  * @param prefix  namespace prefix | 
| 1482 |  |  * @param bucketPtr  optional bucket (return value) | 
| 1483 |  |  * @returns the namespace index on success, INT_MAX if no namespace was | 
| 1484 |  |  * found. | 
| 1485 |  |  */ | 
| 1486 |  | static int | 
| 1487 |  | xmlParserNsLookup(xmlParserCtxtPtr ctxt, const xmlHashedString *prefix, | 
| 1488 | 2.93M |                   xmlParserNsBucket **bucketPtr) { | 
| 1489 | 2.93M |     xmlParserNsBucket *bucket, *tombstone; | 
| 1490 | 2.93M |     unsigned index, hashValue; | 
| 1491 |  |  | 
| 1492 | 2.93M |     if (prefix->name == NULL) | 
| 1493 | 477k |         return(ctxt->nsdb->defaultNsIndex); | 
| 1494 |  |  | 
| 1495 | 2.45M |     if (ctxt->nsdb->hashSize == 0) | 
| 1496 | 22.8k |         return(INT_MAX); | 
| 1497 |  |  | 
| 1498 | 2.42M |     hashValue = prefix->hashValue; | 
| 1499 | 2.42M |     index = hashValue & (ctxt->nsdb->hashSize - 1); | 
| 1500 | 2.42M |     bucket = &ctxt->nsdb->hash[index]; | 
| 1501 | 2.42M |     tombstone = NULL; | 
| 1502 |  |  | 
| 1503 | 3.54M |     while (bucket->hashValue) { | 
| 1504 | 2.97M |         if (bucket->index == INT_MAX) { | 
| 1505 | 432k |             if (tombstone == NULL) | 
| 1506 | 426k |                 tombstone = bucket; | 
| 1507 | 2.53M |         } else if (bucket->hashValue == hashValue) { | 
| 1508 | 1.85M |             if (ctxt->nsTab[bucket->index * 2] == prefix->name) { | 
| 1509 | 1.85M |                 if (bucketPtr != NULL) | 
| 1510 | 1.64M |                     *bucketPtr = bucket; | 
| 1511 | 1.85M |                 return(bucket->index); | 
| 1512 | 1.85M |             } | 
| 1513 | 1.85M |         } | 
| 1514 |  |  | 
| 1515 | 1.11M |         index++; | 
| 1516 | 1.11M |         bucket++; | 
| 1517 | 1.11M |         if (index == ctxt->nsdb->hashSize) { | 
| 1518 | 90.0k |             index = 0; | 
| 1519 | 90.0k |             bucket = ctxt->nsdb->hash; | 
| 1520 | 90.0k |         } | 
| 1521 | 1.11M |     } | 
| 1522 |  |  | 
| 1523 | 576k |     if (bucketPtr != NULL) | 
| 1524 | 425k |         *bucketPtr = tombstone ? tombstone : bucket; | 
| 1525 | 576k |     return(INT_MAX); | 
| 1526 | 2.42M | } | 
| 1527 |  |  | 
| 1528 |  | /** | 
| 1529 |  |  * Lookup namespace URI with given prefix. | 
| 1530 |  |  * | 
| 1531 |  |  * @param ctxt  parser context | 
| 1532 |  |  * @param prefix  namespace prefix | 
| 1533 |  |  * @returns the namespace URI on success, NULL if no namespace was found. | 
| 1534 |  |  */ | 
| 1535 |  | static const xmlChar * | 
| 1536 | 496k | xmlParserNsLookupUri(xmlParserCtxtPtr ctxt, const xmlHashedString *prefix) { | 
| 1537 | 496k |     const xmlChar *ret; | 
| 1538 | 496k |     int nsIndex; | 
| 1539 |  |  | 
| 1540 | 496k |     if (prefix->name == ctxt->str_xml) | 
| 1541 | 506 |         return(ctxt->str_xml_ns); | 
| 1542 |  |  | 
| 1543 |  |     /* | 
| 1544 |  |      * minNsIndex is used when building an entity tree. We must | 
| 1545 |  |      * ignore namespaces declared outside the entity. | 
| 1546 |  |      */ | 
| 1547 | 495k |     nsIndex = xmlParserNsLookup(ctxt, prefix, NULL); | 
| 1548 | 495k |     if ((nsIndex == INT_MAX) || (nsIndex < ctxt->nsdb->minNsIndex)) | 
| 1549 | 205k |         return(NULL); | 
| 1550 |  |  | 
| 1551 | 289k |     ret = ctxt->nsTab[nsIndex * 2 + 1]; | 
| 1552 | 289k |     if (ret[0] == 0) | 
| 1553 | 20.3k |         ret = NULL; | 
| 1554 | 289k |     return(ret); | 
| 1555 | 495k | } | 
| 1556 |  |  | 
| 1557 |  | /** | 
| 1558 |  |  * Lookup extra data for the given prefix. This returns data stored | 
| 1559 |  |  * with xmlParserNsUdpateSax. | 
| 1560 |  |  * | 
| 1561 |  |  * @param ctxt  parser context | 
| 1562 |  |  * @param prefix  namespace prefix | 
| 1563 |  |  * @returns the data on success, NULL if no namespace was found. | 
| 1564 |  |  */ | 
| 1565 |  | void * | 
| 1566 | 18.2k | xmlParserNsLookupSax(xmlParserCtxt *ctxt, const xmlChar *prefix) { | 
| 1567 | 18.2k |     xmlHashedString hprefix; | 
| 1568 | 18.2k |     int nsIndex; | 
| 1569 |  |  | 
| 1570 | 18.2k |     if (prefix == ctxt->str_xml) | 
| 1571 | 3.77k |         return(NULL); | 
| 1572 |  |  | 
| 1573 | 14.5k |     hprefix.name = prefix; | 
| 1574 | 14.5k |     if (prefix != NULL) | 
| 1575 | 10.0k |         hprefix.hashValue = xmlDictComputeHash(ctxt->dict, prefix); | 
| 1576 | 4.47k |     else | 
| 1577 | 4.47k |         hprefix.hashValue = 0; | 
| 1578 | 14.5k |     nsIndex = xmlParserNsLookup(ctxt, &hprefix, NULL); | 
| 1579 | 14.5k |     if ((nsIndex == INT_MAX) || (nsIndex < ctxt->nsdb->minNsIndex)) | 
| 1580 | 0 |         return(NULL); | 
| 1581 |  |  | 
| 1582 | 14.5k |     return(ctxt->nsdb->extra[nsIndex].saxData); | 
| 1583 | 14.5k | } | 
| 1584 |  |  | 
| 1585 |  | /** | 
| 1586 |  |  * Sets or updates extra data for the given prefix. This value will be | 
| 1587 |  |  * returned by xmlParserNsLookupSax as long as the namespace with the | 
| 1588 |  |  * given prefix is in scope. | 
| 1589 |  |  * | 
| 1590 |  |  * @param ctxt  parser context | 
| 1591 |  |  * @param prefix  namespace prefix | 
| 1592 |  |  * @param saxData  extra data for SAX handler | 
| 1593 |  |  * @returns the data on success, NULL if no namespace was found. | 
| 1594 |  |  */ | 
| 1595 |  | int | 
| 1596 |  | xmlParserNsUpdateSax(xmlParserCtxt *ctxt, const xmlChar *prefix, | 
| 1597 | 145k |                      void *saxData) { | 
| 1598 | 145k |     xmlHashedString hprefix; | 
| 1599 | 145k |     int nsIndex; | 
| 1600 |  |  | 
| 1601 | 145k |     if (prefix == ctxt->str_xml) | 
| 1602 | 0 |         return(-1); | 
| 1603 |  |  | 
| 1604 | 145k |     hprefix.name = prefix; | 
| 1605 | 145k |     if (prefix != NULL) | 
| 1606 | 143k |         hprefix.hashValue = xmlDictComputeHash(ctxt->dict, prefix); | 
| 1607 | 2.15k |     else | 
| 1608 | 2.15k |         hprefix.hashValue = 0; | 
| 1609 | 145k |     nsIndex = xmlParserNsLookup(ctxt, &hprefix, NULL); | 
| 1610 | 145k |     if ((nsIndex == INT_MAX) || (nsIndex < ctxt->nsdb->minNsIndex)) | 
| 1611 | 0 |         return(-1); | 
| 1612 |  |  | 
| 1613 | 145k |     ctxt->nsdb->extra[nsIndex].saxData = saxData; | 
| 1614 | 145k |     return(0); | 
| 1615 | 145k | } | 
| 1616 |  |  | 
| 1617 |  | /** | 
| 1618 |  |  * Grows the namespace tables. | 
| 1619 |  |  * | 
| 1620 |  |  * @param ctxt  parser context | 
| 1621 |  |  * @returns 0 on success, -1 if a memory allocation failed. | 
| 1622 |  |  */ | 
| 1623 |  | static int | 
| 1624 | 13.2k | xmlParserNsGrow(xmlParserCtxtPtr ctxt) { | 
| 1625 | 13.2k |     const xmlChar **table; | 
| 1626 | 13.2k |     xmlParserNsExtra *extra; | 
| 1627 | 13.2k |     int newSize; | 
| 1628 |  |  | 
| 1629 | 13.2k |     newSize = xmlGrowCapacity(ctxt->nsMax, | 
| 1630 | 13.2k |                               sizeof(table[0]) + sizeof(extra[0]), | 
| 1631 | 13.2k |                               16, XML_MAX_ITEMS); | 
| 1632 | 13.2k |     if (newSize < 0) | 
| 1633 | 0 |         goto error; | 
| 1634 |  |  | 
| 1635 | 13.2k |     table = xmlRealloc(ctxt->nsTab, 2 * newSize * sizeof(table[0])); | 
| 1636 | 13.2k |     if (table == NULL) | 
| 1637 | 0 |         goto error; | 
| 1638 | 13.2k |     ctxt->nsTab = table; | 
| 1639 |  |  | 
| 1640 | 13.2k |     extra = xmlRealloc(ctxt->nsdb->extra, newSize * sizeof(extra[0])); | 
| 1641 | 13.2k |     if (extra == NULL) | 
| 1642 | 0 |         goto error; | 
| 1643 | 13.2k |     ctxt->nsdb->extra = extra; | 
| 1644 |  |  | 
| 1645 | 13.2k |     ctxt->nsMax = newSize; | 
| 1646 | 13.2k |     return(0); | 
| 1647 |  |  | 
| 1648 | 0 | error: | 
| 1649 | 0 |     xmlErrMemory(ctxt); | 
| 1650 | 0 |     return(-1); | 
| 1651 | 13.2k | } | 
| 1652 |  |  | 
| 1653 |  | /** | 
| 1654 |  |  * Push a new namespace on the table. | 
| 1655 |  |  * | 
| 1656 |  |  * @param ctxt  parser context | 
| 1657 |  |  * @param prefix  prefix with hash value | 
| 1658 |  |  * @param uri  uri with hash value | 
| 1659 |  |  * @param saxData  extra data for SAX handler | 
| 1660 |  |  * @param defAttr  whether the namespace comes from a default attribute | 
| 1661 |  |  * @returns 1 if the namespace was pushed, 0 if the namespace was ignored, | 
| 1662 |  |  * -1 if a memory allocation failed. | 
| 1663 |  |  */ | 
| 1664 |  | static int | 
| 1665 |  | xmlParserNsPush(xmlParserCtxtPtr ctxt, const xmlHashedString *prefix, | 
| 1666 | 1.11M |                 const xmlHashedString *uri, void *saxData, int defAttr) { | 
| 1667 | 1.11M |     xmlParserNsBucket *bucket = NULL; | 
| 1668 | 1.11M |     xmlParserNsExtra *extra; | 
| 1669 | 1.11M |     const xmlChar **ns; | 
| 1670 | 1.11M |     unsigned hashValue, nsIndex, oldIndex; | 
| 1671 |  |  | 
| 1672 | 1.11M |     if ((prefix != NULL) && (prefix->name == ctxt->str_xml)) | 
| 1673 | 199 |         return(0); | 
| 1674 |  |  | 
| 1675 | 1.11M |     if ((ctxt->nsNr >= ctxt->nsMax) && (xmlParserNsGrow(ctxt) < 0)) { | 
| 1676 | 0 |         xmlErrMemory(ctxt); | 
| 1677 | 0 |         return(-1); | 
| 1678 | 0 |     } | 
| 1679 |  |  | 
| 1680 |  |     /* | 
| 1681 |  |      * Default namespace and 'xml' namespace | 
| 1682 |  |      */ | 
| 1683 | 1.11M |     if ((prefix == NULL) || (prefix->name == NULL)) { | 
| 1684 | 75.2k |         oldIndex = ctxt->nsdb->defaultNsIndex; | 
| 1685 |  |  | 
| 1686 | 75.2k |         if (oldIndex != INT_MAX) { | 
| 1687 | 70.6k |             extra = &ctxt->nsdb->extra[oldIndex]; | 
| 1688 |  |  | 
| 1689 | 70.6k |             if (extra->elementId == ctxt->nsdb->elementId) { | 
| 1690 | 19.2k |                 if (defAttr == 0) | 
| 1691 | 18.6k |                     xmlErrAttributeDup(ctxt, NULL, BAD_CAST "xmlns"); | 
| 1692 | 19.2k |                 return(0); | 
| 1693 | 19.2k |             } | 
| 1694 |  |  | 
| 1695 | 51.4k |             if ((ctxt->options & XML_PARSE_NSCLEAN) && | 
| 1696 | 0 |                 (uri->name == ctxt->nsTab[oldIndex * 2 + 1])) | 
| 1697 | 0 |                 return(0); | 
| 1698 | 51.4k |         } | 
| 1699 |  |  | 
| 1700 | 55.9k |         ctxt->nsdb->defaultNsIndex = ctxt->nsNr; | 
| 1701 | 55.9k |         goto populate_entry; | 
| 1702 | 75.2k |     } | 
| 1703 |  |  | 
| 1704 |  |     /* | 
| 1705 |  |      * Hash table lookup | 
| 1706 |  |      */ | 
| 1707 | 1.03M |     oldIndex = xmlParserNsLookup(ctxt, prefix, &bucket); | 
| 1708 | 1.03M |     if (oldIndex != INT_MAX) { | 
| 1709 | 613k |         extra = &ctxt->nsdb->extra[oldIndex]; | 
| 1710 |  |  | 
| 1711 |  |         /* | 
| 1712 |  |          * Check for duplicate definitions on the same element. | 
| 1713 |  |          */ | 
| 1714 | 613k |         if (extra->elementId == ctxt->nsdb->elementId) { | 
| 1715 | 892 |             if (defAttr == 0) | 
| 1716 | 625 |                 xmlErrAttributeDup(ctxt, BAD_CAST "xmlns", prefix->name); | 
| 1717 | 892 |             return(0); | 
| 1718 | 892 |         } | 
| 1719 |  |  | 
| 1720 | 612k |         if ((ctxt->options & XML_PARSE_NSCLEAN) && | 
| 1721 | 0 |             (uri->name == ctxt->nsTab[bucket->index * 2 + 1])) | 
| 1722 | 0 |             return(0); | 
| 1723 |  |  | 
| 1724 | 612k |         bucket->index = ctxt->nsNr; | 
| 1725 | 612k |         goto populate_entry; | 
| 1726 | 612k |     } | 
| 1727 |  |  | 
| 1728 |  |     /* | 
| 1729 |  |      * Insert new bucket | 
| 1730 |  |      */ | 
| 1731 |  |  | 
| 1732 | 426k |     hashValue = prefix->hashValue; | 
| 1733 |  |  | 
| 1734 |  |     /* | 
| 1735 |  |      * Grow hash table, 50% fill factor | 
| 1736 |  |      */ | 
| 1737 | 426k |     if (ctxt->nsdb->hashElems + 1 > ctxt->nsdb->hashSize / 2) { | 
| 1738 | 2.13k |         xmlParserNsBucket *newHash; | 
| 1739 | 2.13k |         unsigned newSize, i, index; | 
| 1740 |  |  | 
| 1741 | 2.13k |         if (ctxt->nsdb->hashSize > UINT_MAX / 2) { | 
| 1742 | 0 |             xmlErrMemory(ctxt); | 
| 1743 | 0 |             return(-1); | 
| 1744 | 0 |         } | 
| 1745 | 2.13k |         newSize = ctxt->nsdb->hashSize ? ctxt->nsdb->hashSize * 2 : 16; | 
| 1746 | 2.13k |         newHash = xmlMalloc(newSize * sizeof(newHash[0])); | 
| 1747 | 2.13k |         if (newHash == NULL) { | 
| 1748 | 0 |             xmlErrMemory(ctxt); | 
| 1749 | 0 |             return(-1); | 
| 1750 | 0 |         } | 
| 1751 | 2.13k |         memset(newHash, 0, newSize * sizeof(newHash[0])); | 
| 1752 |  |  | 
| 1753 | 1.66M |         for (i = 0; i < ctxt->nsdb->hashSize; i++) { | 
| 1754 | 1.66M |             unsigned hv = ctxt->nsdb->hash[i].hashValue; | 
| 1755 | 1.66M |             unsigned newIndex; | 
| 1756 |  |  | 
| 1757 | 1.66M |             if ((hv == 0) || (ctxt->nsdb->hash[i].index == INT_MAX)) | 
| 1758 | 1.65M |                 continue; | 
| 1759 | 4.42k |             newIndex = hv & (newSize - 1); | 
| 1760 |  |  | 
| 1761 | 5.32k |             while (newHash[newIndex].hashValue != 0) { | 
| 1762 | 893 |                 newIndex++; | 
| 1763 | 893 |                 if (newIndex == newSize) | 
| 1764 | 76 |                     newIndex = 0; | 
| 1765 | 893 |             } | 
| 1766 |  |  | 
| 1767 | 4.42k |             newHash[newIndex] = ctxt->nsdb->hash[i]; | 
| 1768 | 4.42k |         } | 
| 1769 |  |  | 
| 1770 | 2.13k |         xmlFree(ctxt->nsdb->hash); | 
| 1771 | 2.13k |         ctxt->nsdb->hash = newHash; | 
| 1772 | 2.13k |         ctxt->nsdb->hashSize = newSize; | 
| 1773 |  |  | 
| 1774 |  |         /* | 
| 1775 |  |          * Relookup | 
| 1776 |  |          */ | 
| 1777 | 2.13k |         index = hashValue & (newSize - 1); | 
| 1778 |  |  | 
| 1779 | 2.39k |         while (newHash[index].hashValue != 0) { | 
| 1780 | 261 |             index++; | 
| 1781 | 261 |             if (index == newSize) | 
| 1782 | 28 |                 index = 0; | 
| 1783 | 261 |         } | 
| 1784 |  |  | 
| 1785 | 2.13k |         bucket = &newHash[index]; | 
| 1786 | 2.13k |     } | 
| 1787 |  |  | 
| 1788 | 426k |     bucket->hashValue = hashValue; | 
| 1789 | 426k |     bucket->index = ctxt->nsNr; | 
| 1790 | 426k |     ctxt->nsdb->hashElems++; | 
| 1791 | 426k |     oldIndex = INT_MAX; | 
| 1792 |  |  | 
| 1793 | 1.09M | populate_entry: | 
| 1794 | 1.09M |     nsIndex = ctxt->nsNr; | 
| 1795 |  |  | 
| 1796 | 1.09M |     ns = &ctxt->nsTab[nsIndex * 2]; | 
| 1797 | 1.09M |     ns[0] = prefix ? prefix->name : NULL; | 
| 1798 | 1.09M |     ns[1] = uri->name; | 
| 1799 |  |  | 
| 1800 | 1.09M |     extra = &ctxt->nsdb->extra[nsIndex]; | 
| 1801 | 1.09M |     extra->saxData = saxData; | 
| 1802 | 1.09M |     extra->prefixHashValue = prefix ? prefix->hashValue : 0; | 
| 1803 | 1.09M |     extra->uriHashValue = uri->hashValue; | 
| 1804 | 1.09M |     extra->elementId = ctxt->nsdb->elementId; | 
| 1805 | 1.09M |     extra->oldIndex = oldIndex; | 
| 1806 |  |  | 
| 1807 | 1.09M |     ctxt->nsNr++; | 
| 1808 |  |  | 
| 1809 | 1.09M |     return(1); | 
| 1810 | 426k | } | 
| 1811 |  |  | 
| 1812 |  | /** | 
| 1813 |  |  * Pops the top `nr` namespaces and restores the hash table. | 
| 1814 |  |  * | 
| 1815 |  |  * @param ctxt  an XML parser context | 
| 1816 |  |  * @param nr  the number to pop | 
| 1817 |  |  * @returns the number of namespaces popped. | 
| 1818 |  |  */ | 
| 1819 |  | static int | 
| 1820 |  | xmlParserNsPop(xmlParserCtxtPtr ctxt, int nr) | 
| 1821 | 206k | { | 
| 1822 | 206k |     int i; | 
| 1823 |  |  | 
| 1824 |  |     /* assert(nr <= ctxt->nsNr); */ | 
| 1825 |  |  | 
| 1826 | 1.29M |     for (i = ctxt->nsNr - 1; i >= ctxt->nsNr - nr; i--) { | 
| 1827 | 1.09M |         const xmlChar *prefix = ctxt->nsTab[i * 2]; | 
| 1828 | 1.09M |         xmlParserNsExtra *extra = &ctxt->nsdb->extra[i]; | 
| 1829 |  |  | 
| 1830 | 1.09M |         if (prefix == NULL) { | 
| 1831 | 55.4k |             ctxt->nsdb->defaultNsIndex = extra->oldIndex; | 
| 1832 | 1.03M |         } else { | 
| 1833 | 1.03M |             xmlHashedString hprefix; | 
| 1834 | 1.03M |             xmlParserNsBucket *bucket = NULL; | 
| 1835 |  |  | 
| 1836 | 1.03M |             hprefix.name = prefix; | 
| 1837 | 1.03M |             hprefix.hashValue = extra->prefixHashValue; | 
| 1838 | 1.03M |             xmlParserNsLookup(ctxt, &hprefix, &bucket); | 
| 1839 |  |             /* assert(bucket && bucket->hashValue); */ | 
| 1840 | 1.03M |             bucket->index = extra->oldIndex; | 
| 1841 | 1.03M |         } | 
| 1842 | 1.09M |     } | 
| 1843 |  |  | 
| 1844 | 206k |     ctxt->nsNr -= nr; | 
| 1845 | 206k |     return(nr); | 
| 1846 | 206k | } | 
| 1847 |  |  | 
| 1848 |  | static int | 
| 1849 | 6.49k | xmlCtxtGrowAttrs(xmlParserCtxtPtr ctxt) { | 
| 1850 | 6.49k |     const xmlChar **atts; | 
| 1851 | 6.49k |     unsigned *attallocs; | 
| 1852 | 6.49k |     int newSize; | 
| 1853 |  |  | 
| 1854 | 6.49k |     newSize = xmlGrowCapacity(ctxt->maxatts / 5, | 
| 1855 | 6.49k |                               sizeof(atts[0]) * 5 + sizeof(attallocs[0]), | 
| 1856 | 6.49k |                               10, XML_MAX_ATTRS); | 
| 1857 | 6.49k |     if (newSize < 0) { | 
| 1858 | 0 |         xmlFatalErr(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 1859 | 0 |                     "Maximum number of attributes exceeded"); | 
| 1860 | 0 |         return(-1); | 
| 1861 | 0 |     } | 
| 1862 |  |  | 
| 1863 | 6.49k |     atts = xmlRealloc(ctxt->atts, newSize * sizeof(atts[0]) * 5); | 
| 1864 | 6.49k |     if (atts == NULL) | 
| 1865 | 0 |         goto mem_error; | 
| 1866 | 6.49k |     ctxt->atts = atts; | 
| 1867 |  |  | 
| 1868 | 6.49k |     attallocs = xmlRealloc(ctxt->attallocs, | 
| 1869 | 6.49k |                            newSize * sizeof(attallocs[0])); | 
| 1870 | 6.49k |     if (attallocs == NULL) | 
| 1871 | 0 |         goto mem_error; | 
| 1872 | 6.49k |     ctxt->attallocs = attallocs; | 
| 1873 |  |  | 
| 1874 | 6.49k |     ctxt->maxatts = newSize * 5; | 
| 1875 |  |  | 
| 1876 | 6.49k |     return(0); | 
| 1877 |  |  | 
| 1878 | 0 | mem_error: | 
| 1879 | 0 |     xmlErrMemory(ctxt); | 
| 1880 | 0 |     return(-1); | 
| 1881 | 6.49k | } | 
| 1882 |  |  | 
| 1883 |  | /** | 
| 1884 |  |  * Pushes a new parser input on top of the input stack | 
| 1885 |  |  * | 
| 1886 |  |  * @param ctxt  an XML parser context | 
| 1887 |  |  * @param value  the parser input | 
| 1888 |  |  * @returns -1 in case of error, the index in the stack otherwise | 
| 1889 |  |  */ | 
| 1890 |  | int | 
| 1891 |  | xmlCtxtPushInput(xmlParserCtxt *ctxt, xmlParserInput *value) | 
| 1892 | 68.8k | { | 
| 1893 | 68.8k |     char *directory = NULL; | 
| 1894 | 68.8k |     int maxDepth; | 
| 1895 |  |  | 
| 1896 | 68.8k |     if ((ctxt == NULL) || (value == NULL)) | 
| 1897 | 0 |         return(-1); | 
| 1898 |  |  | 
| 1899 | 68.8k |     maxDepth = (ctxt->options & XML_PARSE_HUGE) ? 40 : 20; | 
| 1900 |  |  | 
| 1901 | 68.8k |     if (ctxt->inputNr >= ctxt->inputMax) { | 
| 1902 | 2.40k |         xmlParserInputPtr *tmp; | 
| 1903 | 2.40k |         int newSize; | 
| 1904 |  |  | 
| 1905 | 2.40k |         newSize = xmlGrowCapacity(ctxt->inputMax, sizeof(tmp[0]), | 
| 1906 | 2.40k |                                   5, maxDepth); | 
| 1907 | 2.40k |         if (newSize < 0) { | 
| 1908 | 0 |             xmlFatalErrMsg(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 1909 | 0 |                            "Maximum entity nesting depth exceeded"); | 
| 1910 | 0 |             return(-1); | 
| 1911 | 0 |         } | 
| 1912 | 2.40k |         tmp = xmlRealloc(ctxt->inputTab, newSize * sizeof(tmp[0])); | 
| 1913 | 2.40k |         if (tmp == NULL) { | 
| 1914 | 0 |             xmlErrMemory(ctxt); | 
| 1915 | 0 |             return(-1); | 
| 1916 | 0 |         } | 
| 1917 | 2.40k |         ctxt->inputTab = tmp; | 
| 1918 | 2.40k |         ctxt->inputMax = newSize; | 
| 1919 | 2.40k |     } | 
| 1920 |  |  | 
| 1921 | 68.8k |     if ((ctxt->inputNr == 0) && (value->filename != NULL)) { | 
| 1922 | 0 |         directory = xmlParserGetDirectory(value->filename); | 
| 1923 | 0 |         if (directory == NULL) { | 
| 1924 | 0 |             xmlErrMemory(ctxt); | 
| 1925 | 0 |             return(-1); | 
| 1926 | 0 |         } | 
| 1927 | 0 |     } | 
| 1928 |  |  | 
| 1929 | 68.8k |     if (ctxt->input_id >= INT_MAX) { | 
| 1930 | 0 |         xmlFatalErrMsg(ctxt, XML_ERR_RESOURCE_LIMIT, "Input ID overflow\n"); | 
| 1931 | 0 |         return(-1); | 
| 1932 | 0 |     } | 
| 1933 |  |  | 
| 1934 | 68.8k |     ctxt->inputTab[ctxt->inputNr] = value; | 
| 1935 | 68.8k |     ctxt->input = value; | 
| 1936 |  |  | 
| 1937 | 68.8k |     if (ctxt->inputNr == 0) { | 
| 1938 | 17.4k |         xmlFree(ctxt->directory); | 
| 1939 | 17.4k |         ctxt->directory = directory; | 
| 1940 | 17.4k |     } | 
| 1941 |  |  | 
| 1942 |  |     /* | 
| 1943 |  |      * The input ID is unused internally, but there are entity | 
| 1944 |  |      * loaders in downstream code that detect the main document | 
| 1945 |  |      * by checking for "input_id == 1". | 
| 1946 |  |      */ | 
| 1947 | 68.8k |     value->id = ctxt->input_id++; | 
| 1948 |  |  | 
| 1949 | 68.8k |     return(ctxt->inputNr++); | 
| 1950 | 68.8k | } | 
| 1951 |  |  | 
| 1952 |  | /** | 
| 1953 |  |  * Pops the top parser input from the input stack | 
| 1954 |  |  * | 
| 1955 |  |  * @param ctxt  an XML parser context | 
| 1956 |  |  * @returns the input just removed | 
| 1957 |  |  */ | 
| 1958 |  | xmlParserInput * | 
| 1959 |  | xmlCtxtPopInput(xmlParserCtxt *ctxt) | 
| 1960 | 103k | { | 
| 1961 | 103k |     xmlParserInputPtr ret; | 
| 1962 |  |  | 
| 1963 | 103k |     if (ctxt == NULL) | 
| 1964 | 0 |         return(NULL); | 
| 1965 | 103k |     if (ctxt->inputNr <= 0) | 
| 1966 | 34.8k |         return (NULL); | 
| 1967 | 68.8k |     ctxt->inputNr--; | 
| 1968 | 68.8k |     if (ctxt->inputNr > 0) | 
| 1969 | 51.4k |         ctxt->input = ctxt->inputTab[ctxt->inputNr - 1]; | 
| 1970 | 17.4k |     else | 
| 1971 | 17.4k |         ctxt->input = NULL; | 
| 1972 | 68.8k |     ret = ctxt->inputTab[ctxt->inputNr]; | 
| 1973 | 68.8k |     ctxt->inputTab[ctxt->inputNr] = NULL; | 
| 1974 | 68.8k |     return (ret); | 
| 1975 | 103k | } | 
| 1976 |  |  | 
| 1977 |  | /** | 
| 1978 |  |  * Pushes a new element node on top of the node stack | 
| 1979 |  |  * | 
| 1980 |  |  * @deprecated Internal function, do not use. | 
| 1981 |  |  * | 
| 1982 |  |  * @param ctxt  an XML parser context | 
| 1983 |  |  * @param value  the element node | 
| 1984 |  |  * @returns -1 in case of error, the index in the stack otherwise | 
| 1985 |  |  */ | 
| 1986 |  | int | 
| 1987 |  | nodePush(xmlParserCtxt *ctxt, xmlNode *value) | 
| 1988 | 75.4k | { | 
| 1989 | 75.4k |     if (ctxt == NULL) | 
| 1990 | 0 |         return(0); | 
| 1991 |  |  | 
| 1992 | 75.4k |     if (ctxt->nodeNr >= ctxt->nodeMax) { | 
| 1993 | 5.82k |         int maxDepth = (ctxt->options & XML_PARSE_HUGE) ? 2048 : 256; | 
| 1994 | 5.82k |         xmlNodePtr *tmp; | 
| 1995 | 5.82k |         int newSize; | 
| 1996 |  |  | 
| 1997 | 5.82k |         newSize = xmlGrowCapacity(ctxt->nodeMax, sizeof(tmp[0]), | 
| 1998 | 5.82k |                                   10, maxDepth); | 
| 1999 | 5.82k |         if (newSize < 0) { | 
| 2000 | 9 |             xmlFatalErrMsgInt(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 2001 | 9 |                     "Excessive depth in document: %d," | 
| 2002 | 9 |                     " use XML_PARSE_HUGE option\n", | 
| 2003 | 9 |                     ctxt->nodeNr); | 
| 2004 | 9 |             return(-1); | 
| 2005 | 9 |         } | 
| 2006 |  |  | 
| 2007 | 5.81k |   tmp = xmlRealloc(ctxt->nodeTab, newSize * sizeof(tmp[0])); | 
| 2008 | 5.81k |         if (tmp == NULL) { | 
| 2009 | 0 |             xmlErrMemory(ctxt); | 
| 2010 | 0 |             return (-1); | 
| 2011 | 0 |         } | 
| 2012 | 5.81k |         ctxt->nodeTab = tmp; | 
| 2013 | 5.81k |   ctxt->nodeMax = newSize; | 
| 2014 | 5.81k |     } | 
| 2015 |  |  | 
| 2016 | 75.4k |     ctxt->nodeTab[ctxt->nodeNr] = value; | 
| 2017 | 75.4k |     ctxt->node = value; | 
| 2018 | 75.4k |     return (ctxt->nodeNr++); | 
| 2019 | 75.4k | } | 
| 2020 |  |  | 
| 2021 |  | /** | 
| 2022 |  |  * Pops the top element node from the node stack | 
| 2023 |  |  * | 
| 2024 |  |  * @deprecated Internal function, do not use. | 
| 2025 |  |  * | 
| 2026 |  |  * @param ctxt  an XML parser context | 
| 2027 |  |  * @returns the node just removed | 
| 2028 |  |  */ | 
| 2029 |  | xmlNode * | 
| 2030 |  | nodePop(xmlParserCtxt *ctxt) | 
| 2031 | 314k | { | 
| 2032 | 314k |     xmlNodePtr ret; | 
| 2033 |  |  | 
| 2034 | 314k |     if (ctxt == NULL) return(NULL); | 
| 2035 | 314k |     if (ctxt->nodeNr <= 0) | 
| 2036 | 241k |         return (NULL); | 
| 2037 | 72.2k |     ctxt->nodeNr--; | 
| 2038 | 72.2k |     if (ctxt->nodeNr > 0) | 
| 2039 | 70.6k |         ctxt->node = ctxt->nodeTab[ctxt->nodeNr - 1]; | 
| 2040 | 1.62k |     else | 
| 2041 | 1.62k |         ctxt->node = NULL; | 
| 2042 | 72.2k |     ret = ctxt->nodeTab[ctxt->nodeNr]; | 
| 2043 | 72.2k |     ctxt->nodeTab[ctxt->nodeNr] = NULL; | 
| 2044 | 72.2k |     return (ret); | 
| 2045 | 314k | } | 
| 2046 |  |  | 
| 2047 |  | /** | 
| 2048 |  |  * Pushes a new element name/prefix/URL on top of the name stack | 
| 2049 |  |  * | 
| 2050 |  |  * @param ctxt  an XML parser context | 
| 2051 |  |  * @param value  the element name | 
| 2052 |  |  * @param prefix  the element prefix | 
| 2053 |  |  * @param URI  the element namespace name | 
| 2054 |  |  * @param line  the current line number for error messages | 
| 2055 |  |  * @param nsNr  the number of namespaces pushed on the namespace table | 
| 2056 |  |  * @returns -1 in case of error, the index in the stack otherwise | 
| 2057 |  |  */ | 
| 2058 |  | static int | 
| 2059 |  | nameNsPush(xmlParserCtxtPtr ctxt, const xmlChar * value, | 
| 2060 |  |            const xmlChar *prefix, const xmlChar *URI, int line, int nsNr) | 
| 2061 | 498k | { | 
| 2062 | 498k |     xmlStartTag *tag; | 
| 2063 |  |  | 
| 2064 | 498k |     if (ctxt->nameNr >= ctxt->nameMax) { | 
| 2065 | 16.7k |         const xmlChar **tmp; | 
| 2066 | 16.7k |         xmlStartTag *tmp2; | 
| 2067 | 16.7k |         int newSize; | 
| 2068 |  |  | 
| 2069 | 16.7k |         newSize = xmlGrowCapacity(ctxt->nameMax, | 
| 2070 | 16.7k |                                   sizeof(tmp[0]) + sizeof(tmp2[0]), | 
| 2071 | 16.7k |                                   10, XML_MAX_ITEMS); | 
| 2072 | 16.7k |         if (newSize < 0) | 
| 2073 | 0 |             goto mem_error; | 
| 2074 |  |  | 
| 2075 | 16.7k |         tmp = xmlRealloc(ctxt->nameTab, newSize * sizeof(tmp[0])); | 
| 2076 | 16.7k |         if (tmp == NULL) | 
| 2077 | 0 |       goto mem_error; | 
| 2078 | 16.7k |   ctxt->nameTab = tmp; | 
| 2079 |  |  | 
| 2080 | 16.7k |         tmp2 = xmlRealloc(ctxt->pushTab, newSize * sizeof(tmp2[0])); | 
| 2081 | 16.7k |         if (tmp2 == NULL) | 
| 2082 | 0 |       goto mem_error; | 
| 2083 | 16.7k |   ctxt->pushTab = tmp2; | 
| 2084 |  |  | 
| 2085 | 16.7k |         ctxt->nameMax = newSize; | 
| 2086 | 481k |     } else if (ctxt->pushTab == NULL) { | 
| 2087 | 10.7k |         ctxt->pushTab = xmlMalloc(ctxt->nameMax * sizeof(ctxt->pushTab[0])); | 
| 2088 | 10.7k |         if (ctxt->pushTab == NULL) | 
| 2089 | 0 |             goto mem_error; | 
| 2090 | 10.7k |     } | 
| 2091 | 498k |     ctxt->nameTab[ctxt->nameNr] = value; | 
| 2092 | 498k |     ctxt->name = value; | 
| 2093 | 498k |     tag = &ctxt->pushTab[ctxt->nameNr]; | 
| 2094 | 498k |     tag->prefix = prefix; | 
| 2095 | 498k |     tag->URI = URI; | 
| 2096 | 498k |     tag->line = line; | 
| 2097 | 498k |     tag->nsNr = nsNr; | 
| 2098 | 498k |     return (ctxt->nameNr++); | 
| 2099 | 0 | mem_error: | 
| 2100 | 0 |     xmlErrMemory(ctxt); | 
| 2101 | 0 |     return (-1); | 
| 2102 | 498k | } | 
| 2103 |  | #ifdef LIBXML_PUSH_ENABLED | 
| 2104 |  | /** | 
| 2105 |  |  * Pops the top element/prefix/URI name from the name stack | 
| 2106 |  |  * | 
| 2107 |  |  * @param ctxt  an XML parser context | 
| 2108 |  |  * @returns the name just removed | 
| 2109 |  |  */ | 
| 2110 |  | static const xmlChar * | 
| 2111 |  | nameNsPop(xmlParserCtxtPtr ctxt) | 
| 2112 | 0 | { | 
| 2113 | 0 |     const xmlChar *ret; | 
| 2114 |  | 
 | 
| 2115 | 0 |     if (ctxt->nameNr <= 0) | 
| 2116 | 0 |         return (NULL); | 
| 2117 | 0 |     ctxt->nameNr--; | 
| 2118 | 0 |     if (ctxt->nameNr > 0) | 
| 2119 | 0 |         ctxt->name = ctxt->nameTab[ctxt->nameNr - 1]; | 
| 2120 | 0 |     else | 
| 2121 | 0 |         ctxt->name = NULL; | 
| 2122 | 0 |     ret = ctxt->nameTab[ctxt->nameNr]; | 
| 2123 | 0 |     ctxt->nameTab[ctxt->nameNr] = NULL; | 
| 2124 | 0 |     return (ret); | 
| 2125 | 0 | } | 
| 2126 |  | #endif /* LIBXML_PUSH_ENABLED */ | 
| 2127 |  |  | 
| 2128 |  | /** | 
| 2129 |  |  * Pops the top element name from the name stack | 
| 2130 |  |  * | 
| 2131 |  |  * @deprecated Internal function, do not use. | 
| 2132 |  |  * | 
| 2133 |  |  * @param ctxt  an XML parser context | 
| 2134 |  |  * @returns the name just removed | 
| 2135 |  |  */ | 
| 2136 |  | static const xmlChar * | 
| 2137 |  | namePop(xmlParserCtxtPtr ctxt) | 
| 2138 | 492k | { | 
| 2139 | 492k |     const xmlChar *ret; | 
| 2140 |  |  | 
| 2141 | 492k |     if ((ctxt == NULL) || (ctxt->nameNr <= 0)) | 
| 2142 | 0 |         return (NULL); | 
| 2143 | 492k |     ctxt->nameNr--; | 
| 2144 | 492k |     if (ctxt->nameNr > 0) | 
| 2145 | 487k |         ctxt->name = ctxt->nameTab[ctxt->nameNr - 1]; | 
| 2146 | 4.76k |     else | 
| 2147 | 4.76k |         ctxt->name = NULL; | 
| 2148 | 492k |     ret = ctxt->nameTab[ctxt->nameNr]; | 
| 2149 | 492k |     ctxt->nameTab[ctxt->nameNr] = NULL; | 
| 2150 | 492k |     return (ret); | 
| 2151 | 492k | } | 
| 2152 |  |  | 
| 2153 | 3.27M | static int spacePush(xmlParserCtxtPtr ctxt, int val) { | 
| 2154 | 3.27M |     if (ctxt->spaceNr >= ctxt->spaceMax) { | 
| 2155 | 26.5k |         int *tmp; | 
| 2156 | 26.5k |         int newSize; | 
| 2157 |  |  | 
| 2158 | 26.5k |         newSize = xmlGrowCapacity(ctxt->spaceMax, sizeof(tmp[0]), | 
| 2159 | 26.5k |                                   10, XML_MAX_ITEMS); | 
| 2160 | 26.5k |         if (newSize < 0) { | 
| 2161 | 0 |       xmlErrMemory(ctxt); | 
| 2162 | 0 |       return(-1); | 
| 2163 | 0 |         } | 
| 2164 |  |  | 
| 2165 | 26.5k |         tmp = xmlRealloc(ctxt->spaceTab, newSize * sizeof(tmp[0])); | 
| 2166 | 26.5k |         if (tmp == NULL) { | 
| 2167 | 0 |       xmlErrMemory(ctxt); | 
| 2168 | 0 |       return(-1); | 
| 2169 | 0 |   } | 
| 2170 | 26.5k |   ctxt->spaceTab = tmp; | 
| 2171 |  |  | 
| 2172 | 26.5k |         ctxt->spaceMax = newSize; | 
| 2173 | 26.5k |     } | 
| 2174 | 3.27M |     ctxt->spaceTab[ctxt->spaceNr] = val; | 
| 2175 | 3.27M |     ctxt->space = &ctxt->spaceTab[ctxt->spaceNr]; | 
| 2176 | 3.27M |     return(ctxt->spaceNr++); | 
| 2177 | 3.27M | } | 
| 2178 |  |  | 
| 2179 | 3.26M | static int spacePop(xmlParserCtxtPtr ctxt) { | 
| 2180 | 3.26M |     int ret; | 
| 2181 | 3.26M |     if (ctxt->spaceNr <= 0) return(0); | 
| 2182 | 3.26M |     ctxt->spaceNr--; | 
| 2183 | 3.26M |     if (ctxt->spaceNr > 0) | 
| 2184 | 3.26M |   ctxt->space = &ctxt->spaceTab[ctxt->spaceNr - 1]; | 
| 2185 | 0 |     else | 
| 2186 | 0 |         ctxt->space = &ctxt->spaceTab[0]; | 
| 2187 | 3.26M |     ret = ctxt->spaceTab[ctxt->spaceNr]; | 
| 2188 | 3.26M |     ctxt->spaceTab[ctxt->spaceNr] = -1; | 
| 2189 | 3.26M |     return(ret); | 
| 2190 | 3.26M | } | 
| 2191 |  |  | 
| 2192 |  | /* | 
| 2193 |  |  * Macros for accessing the content. Those should be used only by the parser, | 
| 2194 |  |  * and not exported. | 
| 2195 |  |  * | 
| 2196 |  |  * Dirty macros, i.e. one often need to make assumption on the context to | 
| 2197 |  |  * use them | 
| 2198 |  |  * | 
| 2199 |  |  *   CUR_PTR return the current pointer to the xmlChar to be parsed. | 
| 2200 |  |  *           To be used with extreme caution since operations consuming | 
| 2201 |  |  *           characters may move the input buffer to a different location ! | 
| 2202 |  |  *   CUR     returns the current xmlChar value, i.e. a 8 bit value if compiled | 
| 2203 |  |  *           This should be used internally by the parser | 
| 2204 |  |  *           only to compare to ASCII values otherwise it would break when | 
| 2205 |  |  *           running with UTF-8 encoding. | 
| 2206 |  |  *   RAW     same as CUR but in the input buffer, bypass any token | 
| 2207 |  |  *           extraction that may have been done | 
| 2208 |  |  *   NXT(n)  returns the n'th next xmlChar. Same as CUR is should be used only | 
| 2209 |  |  *           to compare on ASCII based substring. | 
| 2210 |  |  *   SKIP(n) Skip n xmlChar, and must also be used only to skip ASCII defined | 
| 2211 |  |  *           strings without newlines within the parser. | 
| 2212 |  |  *   NEXT1(l) Skip 1 xmlChar, and must also be used only to skip 1 non-newline ASCII | 
| 2213 |  |  *           defined char within the parser. | 
| 2214 |  |  * Clean macros, not dependent of an ASCII context, expect UTF-8 encoding | 
| 2215 |  |  * | 
| 2216 |  |  *   NEXT    Skip to the next character, this does the proper decoding | 
| 2217 |  |  *           in UTF-8 mode. It also pop-up unfinished entities on the fly. | 
| 2218 |  |  *   NEXTL(l) Skip the current unicode character of l xmlChars long. | 
| 2219 |  |  *   COPY_BUF  copy the current unicode char to the target buffer, increment | 
| 2220 |  |  *            the index | 
| 2221 |  |  *   GROW, SHRINK  handling of input buffers | 
| 2222 |  |  */ | 
| 2223 |  |  | 
| 2224 | 10.5M | #define RAW (*ctxt->input->cur) | 
| 2225 | 202M | #define CUR (*ctxt->input->cur) | 
| 2226 | 7.75M | #define NXT(val) ctxt->input->cur[(val)] | 
| 2227 | 383M | #define CUR_PTR ctxt->input->cur | 
| 2228 | 8.17M | #define BASE_PTR ctxt->input->base | 
| 2229 |  |  | 
| 2230 |  | #define CMP4( s, c1, c2, c3, c4 ) \ | 
| 2231 | 10.9M |   ( ((unsigned char *) s)[ 0 ] == c1 && ((unsigned char *) s)[ 1 ] == c2 && \ | 
| 2232 | 5.48M |     ((unsigned char *) s)[ 2 ] == c3 && ((unsigned char *) s)[ 3 ] == c4 ) | 
| 2233 |  | #define CMP5( s, c1, c2, c3, c4, c5 ) \ | 
| 2234 | 10.8M |   ( CMP4( s, c1, c2, c3, c4 ) && ((unsigned char *) s)[ 4 ] == c5 ) | 
| 2235 |  | #define CMP6( s, c1, c2, c3, c4, c5, c6 ) \ | 
| 2236 | 10.6M |   ( CMP5( s, c1, c2, c3, c4, c5 ) && ((unsigned char *) s)[ 5 ] == c6 ) | 
| 2237 |  | #define CMP7( s, c1, c2, c3, c4, c5, c6, c7 ) \ | 
| 2238 | 10.4M |   ( CMP6( s, c1, c2, c3, c4, c5, c6 ) && ((unsigned char *) s)[ 6 ] == c7 ) | 
| 2239 |  | #define CMP8( s, c1, c2, c3, c4, c5, c6, c7, c8 ) \ | 
| 2240 | 10.3M |   ( CMP7( s, c1, c2, c3, c4, c5, c6, c7 ) && ((unsigned char *) s)[ 7 ] == c8 ) | 
| 2241 |  | #define CMP9( s, c1, c2, c3, c4, c5, c6, c7, c8, c9 ) \ | 
| 2242 | 5.16M |   ( CMP8( s, c1, c2, c3, c4, c5, c6, c7, c8 ) && \ | 
| 2243 | 5.16M |     ((unsigned char *) s)[ 8 ] == c9 ) | 
| 2244 |  | #define CMP10( s, c1, c2, c3, c4, c5, c6, c7, c8, c9, c10 ) \ | 
| 2245 | 3.09k |   ( CMP9( s, c1, c2, c3, c4, c5, c6, c7, c8, c9 ) && \ | 
| 2246 | 3.09k |     ((unsigned char *) s)[ 9 ] == c10 ) | 
| 2247 |  |  | 
| 2248 | 397k | #define SKIP(val) do {             \ | 
| 2249 | 397k |     ctxt->input->cur += (val),ctxt->input->col+=(val);      \ | 
| 2250 | 397k |     if (*ctxt->input->cur == 0)           \ | 
| 2251 | 397k |         xmlParserGrow(ctxt);           \ | 
| 2252 | 397k |   } while (0) | 
| 2253 |  |  | 
| 2254 |  | #define SKIPL(val) do {             \ | 
| 2255 |  |     int skipl;                \ | 
| 2256 |  |     for(skipl=0; skipl<val; skipl++) {          \ | 
| 2257 |  |   if (*(ctxt->input->cur) == '\n') {        \ | 
| 2258 |  |   ctxt->input->line++; ctxt->input->col = 1;      \ | 
| 2259 |  |   } else ctxt->input->col++;          \ | 
| 2260 |  |   ctxt->input->cur++;           \ | 
| 2261 |  |     }                 \ | 
| 2262 |  |     if (*ctxt->input->cur == 0)           \ | 
| 2263 |  |         xmlParserGrow(ctxt);            \ | 
| 2264 |  |   } while (0) | 
| 2265 |  |  | 
| 2266 |  | #define SHRINK \ | 
| 2267 | 7.33M |     if (!PARSER_PROGRESSIVE(ctxt)) \ | 
| 2268 | 7.33M |   xmlParserShrink(ctxt); | 
| 2269 |  |  | 
| 2270 |  | #define GROW \ | 
| 2271 | 17.4M |     if ((!PARSER_PROGRESSIVE(ctxt)) && \ | 
| 2272 | 17.4M |         (ctxt->input->end - ctxt->input->cur < INPUT_CHUNK)) \ | 
| 2273 | 1.28M |   xmlParserGrow(ctxt); | 
| 2274 |  |  | 
| 2275 | 2.78M | #define SKIP_BLANKS xmlSkipBlankChars(ctxt) | 
| 2276 |  |  | 
| 2277 | 231k | #define SKIP_BLANKS_PE xmlSkipBlankCharsPE(ctxt) | 
| 2278 |  |  | 
| 2279 | 21.9M | #define NEXT xmlNextChar(ctxt) | 
| 2280 |  |  | 
| 2281 | 3.51M | #define NEXT1 {               \ | 
| 2282 | 3.51M |   ctxt->input->col++;           \ | 
| 2283 | 3.51M |   ctxt->input->cur++;           \ | 
| 2284 | 3.51M |   if (*ctxt->input->cur == 0)         \ | 
| 2285 | 3.51M |       xmlParserGrow(ctxt);           \ | 
| 2286 | 3.51M |     } | 
| 2287 |  |  | 
| 2288 | 239M | #define NEXTL(l) do {             \ | 
| 2289 | 239M |     if (*(ctxt->input->cur) == '\n') {         \ | 
| 2290 | 13.0M |   ctxt->input->line++; ctxt->input->col = 1;      \ | 
| 2291 | 226M |     } else ctxt->input->col++;           \ | 
| 2292 | 239M |     ctxt->input->cur += l;        \ | 
| 2293 | 239M |   } while (0) | 
| 2294 |  |  | 
| 2295 |  | #define COPY_BUF(b, i, v)           \ | 
| 2296 | 50.9M |     if (v < 0x80) b[i++] = v;           \ | 
| 2297 | 50.9M |     else i += xmlCopyCharMultiByte(&b[i],v) | 
| 2298 |  |  | 
| 2299 |  | static int | 
| 2300 | 48.3M | xmlCurrentCharRecover(xmlParserCtxtPtr ctxt, int *len) { | 
| 2301 | 48.3M |     int c = xmlCurrentChar(ctxt, len); | 
| 2302 |  |  | 
| 2303 | 48.3M |     if (c == XML_INVALID_CHAR) | 
| 2304 | 12.7M |         c = 0xFFFD; /* replacement character */ | 
| 2305 |  |  | 
| 2306 | 48.3M |     return(c); | 
| 2307 | 48.3M | } | 
| 2308 |  |  | 
| 2309 |  | /** | 
| 2310 |  |  * Skip whitespace in the input stream. | 
| 2311 |  |  * | 
| 2312 |  |  * @deprecated Internal function, do not use. | 
| 2313 |  |  * | 
| 2314 |  |  * @param ctxt  the XML parser context | 
| 2315 |  |  * @returns the number of space chars skipped | 
| 2316 |  |  */ | 
| 2317 |  | int | 
| 2318 | 2.89M | xmlSkipBlankChars(xmlParserCtxt *ctxt) { | 
| 2319 | 2.89M |     const xmlChar *cur; | 
| 2320 | 2.89M |     int res = 0; | 
| 2321 |  |  | 
| 2322 | 2.89M |     cur = ctxt->input->cur; | 
| 2323 | 4.36M |     while (IS_BLANK_CH(*cur)) { | 
| 2324 | 4.36M |         if (*cur == '\n') { | 
| 2325 | 3.59M |             ctxt->input->line++; ctxt->input->col = 1; | 
| 2326 | 3.59M |         } else { | 
| 2327 | 774k |             ctxt->input->col++; | 
| 2328 | 774k |         } | 
| 2329 | 4.36M |         cur++; | 
| 2330 | 4.36M |         if (res < INT_MAX) | 
| 2331 | 4.36M |             res++; | 
| 2332 | 4.36M |         if (*cur == 0) { | 
| 2333 | 2.11k |             ctxt->input->cur = cur; | 
| 2334 | 2.11k |             xmlParserGrow(ctxt); | 
| 2335 | 2.11k |             cur = ctxt->input->cur; | 
| 2336 | 2.11k |         } | 
| 2337 | 4.36M |     } | 
| 2338 | 2.89M |     ctxt->input->cur = cur; | 
| 2339 |  |  | 
| 2340 | 2.89M |     if (res > 4) | 
| 2341 | 5.07k |         GROW; | 
| 2342 |  |  | 
| 2343 | 2.89M |     return(res); | 
| 2344 | 2.89M | } | 
| 2345 |  |  | 
| 2346 |  | static void | 
| 2347 | 49.6k | xmlPopPE(xmlParserCtxtPtr ctxt) { | 
| 2348 | 49.6k |     unsigned long consumed; | 
| 2349 | 49.6k |     xmlEntityPtr ent; | 
| 2350 |  |  | 
| 2351 | 49.6k |     ent = ctxt->input->entity; | 
| 2352 |  |  | 
| 2353 | 49.6k |     ent->flags &= ~XML_ENT_EXPANDING; | 
| 2354 |  |  | 
| 2355 | 49.6k |     if ((ent->flags & XML_ENT_CHECKED) == 0) { | 
| 2356 | 1.55k |         int result; | 
| 2357 |  |  | 
| 2358 |  |         /* | 
| 2359 |  |          * Read the rest of the stream in case of errors. We want | 
| 2360 |  |          * to account for the whole entity size. | 
| 2361 |  |          */ | 
| 2362 | 1.55k |         do { | 
| 2363 | 1.55k |             ctxt->input->cur = ctxt->input->end; | 
| 2364 | 1.55k |             xmlParserShrink(ctxt); | 
| 2365 | 1.55k |             result = xmlParserGrow(ctxt); | 
| 2366 | 1.55k |         } while (result > 0); | 
| 2367 |  |  | 
| 2368 | 1.55k |         consumed = ctxt->input->consumed; | 
| 2369 | 1.55k |         xmlSaturatedAddSizeT(&consumed, | 
| 2370 | 1.55k |                              ctxt->input->end - ctxt->input->base); | 
| 2371 |  |  | 
| 2372 | 1.55k |         xmlSaturatedAdd(&ent->expandedSize, consumed); | 
| 2373 |  |  | 
| 2374 |  |         /* | 
| 2375 |  |          * Add to sizeentities when parsing an external entity | 
| 2376 |  |          * for the first time. | 
| 2377 |  |          */ | 
| 2378 | 1.55k |         if (ent->etype == XML_EXTERNAL_PARAMETER_ENTITY) { | 
| 2379 | 0 |             xmlSaturatedAdd(&ctxt->sizeentities, consumed); | 
| 2380 | 0 |         } | 
| 2381 |  |  | 
| 2382 | 1.55k |         ent->flags |= XML_ENT_CHECKED; | 
| 2383 | 1.55k |     } | 
| 2384 |  |  | 
| 2385 | 49.6k |     xmlFreeInputStream(xmlCtxtPopInput(ctxt)); | 
| 2386 |  |  | 
| 2387 | 49.6k |     xmlParserEntityCheck(ctxt, ent->expandedSize); | 
| 2388 |  |  | 
| 2389 | 49.6k |     GROW; | 
| 2390 | 49.6k | } | 
| 2391 |  |  | 
| 2392 |  | /** | 
| 2393 |  |  * Skip whitespace in the input stream, also handling parameter | 
| 2394 |  |  * entities. | 
| 2395 |  |  * | 
| 2396 |  |  * @param ctxt  the XML parser context | 
| 2397 |  |  * @returns the number of space chars skipped | 
| 2398 |  |  */ | 
| 2399 |  | static int | 
| 2400 | 231k | xmlSkipBlankCharsPE(xmlParserCtxtPtr ctxt) { | 
| 2401 | 231k |     int res = 0; | 
| 2402 | 231k |     int inParam; | 
| 2403 | 231k |     int expandParam; | 
| 2404 |  |  | 
| 2405 | 231k |     inParam = PARSER_IN_PE(ctxt); | 
| 2406 | 231k |     expandParam = PARSER_EXTERNAL(ctxt); | 
| 2407 |  |  | 
| 2408 | 231k |     if (!inParam && !expandParam) | 
| 2409 | 109k |         return(xmlSkipBlankChars(ctxt)); | 
| 2410 |  |  | 
| 2411 |  |     /* | 
| 2412 |  |      * It's Okay to use CUR/NEXT here since all the blanks are on | 
| 2413 |  |      * the ASCII range. | 
| 2414 |  |      */ | 
| 2415 | 2.82M |     while (PARSER_STOPPED(ctxt) == 0) { | 
| 2416 | 2.82M |         if (IS_BLANK_CH(CUR)) { /* CHECKED tstblanks.xml */ | 
| 2417 | 2.70M |             NEXT; | 
| 2418 | 2.70M |         } else if (CUR == '%') { | 
| 2419 | 1.33k |             if ((expandParam == 0) || | 
| 2420 | 0 |                 (IS_BLANK_CH(NXT(1))) || (NXT(1) == 0)) | 
| 2421 | 1.33k |                 break; | 
| 2422 |  |  | 
| 2423 |  |             /* | 
| 2424 |  |              * Expand parameter entity. We continue to consume | 
| 2425 |  |              * whitespace at the start of the entity and possible | 
| 2426 |  |              * even consume the whole entity and pop it. We might | 
| 2427 |  |              * even pop multiple PEs in this loop. | 
| 2428 |  |              */ | 
| 2429 | 0 |             xmlParsePERefInternal(ctxt, 0); | 
| 2430 |  | 
 | 
| 2431 | 0 |             inParam = PARSER_IN_PE(ctxt); | 
| 2432 | 0 |             expandParam = PARSER_EXTERNAL(ctxt); | 
| 2433 | 120k |         } else if (CUR == 0) { | 
| 2434 | 20.1k |             if (inParam == 0) | 
| 2435 | 4 |                 break; | 
| 2436 |  |  | 
| 2437 |  |             /* | 
| 2438 |  |              * Don't pop parameter entities that start a markup | 
| 2439 |  |              * declaration to detect Well-formedness constraint: | 
| 2440 |  |              * PE Between Declarations. | 
| 2441 |  |              */ | 
| 2442 | 20.1k |             if (ctxt->input->flags & XML_INPUT_MARKUP_DECL) | 
| 2443 | 19.6k |                 break; | 
| 2444 |  |  | 
| 2445 | 412 |             xmlPopPE(ctxt); | 
| 2446 |  |  | 
| 2447 | 412 |             inParam = PARSER_IN_PE(ctxt); | 
| 2448 | 412 |             expandParam = PARSER_EXTERNAL(ctxt); | 
| 2449 | 100k |         } else { | 
| 2450 | 100k |             break; | 
| 2451 | 100k |         } | 
| 2452 |  |  | 
| 2453 |  |         /* | 
| 2454 |  |          * Also increase the counter when entering or exiting a PERef. | 
| 2455 |  |          * The spec says: "When a parameter-entity reference is recognized | 
| 2456 |  |          * in the DTD and included, its replacement text MUST be enlarged | 
| 2457 |  |          * by the attachment of one leading and one following space (#x20) | 
| 2458 |  |          * character." | 
| 2459 |  |          */ | 
| 2460 | 2.70M |         if (res < INT_MAX) | 
| 2461 | 2.70M |             res++; | 
| 2462 | 2.70M |     } | 
| 2463 |  |  | 
| 2464 | 121k |     return(res); | 
| 2465 | 231k | } | 
| 2466 |  |  | 
| 2467 |  | /************************************************************************ | 
| 2468 |  |  *                  * | 
| 2469 |  |  *    Commodity functions to handle entities      * | 
| 2470 |  |  *                  * | 
| 2471 |  |  ************************************************************************/ | 
| 2472 |  |  | 
| 2473 |  | /** | 
| 2474 |  |  * @deprecated Internal function, don't use. | 
| 2475 |  |  * | 
| 2476 |  |  * @param ctxt  an XML parser context | 
| 2477 |  |  * @returns the current xmlChar in the parser context | 
| 2478 |  |  */ | 
| 2479 |  | xmlChar | 
| 2480 | 0 | xmlPopInput(xmlParserCtxt *ctxt) { | 
| 2481 | 0 |     xmlParserInputPtr input; | 
| 2482 |  | 
 | 
| 2483 | 0 |     if ((ctxt == NULL) || (ctxt->inputNr <= 1)) return(0); | 
| 2484 | 0 |     input = xmlCtxtPopInput(ctxt); | 
| 2485 | 0 |     xmlFreeInputStream(input); | 
| 2486 | 0 |     if (*ctxt->input->cur == 0) | 
| 2487 | 0 |         xmlParserGrow(ctxt); | 
| 2488 | 0 |     return(CUR); | 
| 2489 | 0 | } | 
| 2490 |  |  | 
| 2491 |  | /** | 
| 2492 |  |  * Push an input stream onto the stack. | 
| 2493 |  |  * | 
| 2494 |  |  * @deprecated Internal function, don't use. | 
| 2495 |  |  * | 
| 2496 |  |  * @param ctxt  an XML parser context | 
| 2497 |  |  * @param input  an XML parser input fragment (entity, XML fragment ...). | 
| 2498 |  |  * @returns -1 in case of error or the index in the input stack | 
| 2499 |  |  */ | 
| 2500 |  | int | 
| 2501 | 0 | xmlPushInput(xmlParserCtxt *ctxt, xmlParserInput *input) { | 
| 2502 | 0 |     int ret; | 
| 2503 |  | 
 | 
| 2504 | 0 |     if ((ctxt == NULL) || (input == NULL)) | 
| 2505 | 0 |         return(-1); | 
| 2506 |  |  | 
| 2507 | 0 |     ret = xmlCtxtPushInput(ctxt, input); | 
| 2508 | 0 |     if (ret >= 0) | 
| 2509 | 0 |         GROW; | 
| 2510 | 0 |     return(ret); | 
| 2511 | 0 | } | 
| 2512 |  |  | 
| 2513 |  | /** | 
| 2514 |  |  * Parse a numeric character reference. Always consumes '&'. | 
| 2515 |  |  * | 
| 2516 |  |  * @deprecated Internal function, don't use. | 
| 2517 |  |  * | 
| 2518 |  |  *     [66] CharRef ::= '&#' [0-9]+ ';' | | 
| 2519 |  |  *                      '&#x' [0-9a-fA-F]+ ';' | 
| 2520 |  |  * | 
| 2521 |  |  * [ WFC: Legal Character ] | 
| 2522 |  |  * Characters referred to using character references must match the | 
| 2523 |  |  * production for Char. | 
| 2524 |  |  * | 
| 2525 |  |  * @param ctxt  an XML parser context | 
| 2526 |  |  * @returns the value parsed (as an int), 0 in case of error | 
| 2527 |  |  */ | 
| 2528 |  | int | 
| 2529 | 34.8k | xmlParseCharRef(xmlParserCtxt *ctxt) { | 
| 2530 | 34.8k |     int val = 0; | 
| 2531 | 34.8k |     int count = 0; | 
| 2532 |  |  | 
| 2533 |  |     /* | 
| 2534 |  |      * Using RAW/CUR/NEXT is okay since we are working on ASCII range here | 
| 2535 |  |      */ | 
| 2536 | 34.8k |     if ((RAW == '&') && (NXT(1) == '#') && | 
| 2537 | 34.8k |         (NXT(2) == 'x')) { | 
| 2538 | 10.5k |   SKIP(3); | 
| 2539 | 10.5k |   GROW; | 
| 2540 | 36.9k |   while ((RAW != ';') && (PARSER_STOPPED(ctxt) == 0)) { | 
| 2541 | 28.6k |       if (count++ > 20) { | 
| 2542 | 694 |     count = 0; | 
| 2543 | 694 |     GROW; | 
| 2544 | 694 |       } | 
| 2545 | 28.6k |       if ((RAW >= '0') && (RAW <= '9')) | 
| 2546 | 10.4k |           val = val * 16 + (CUR - '0'); | 
| 2547 | 18.1k |       else if ((RAW >= 'a') && (RAW <= 'f') && (count < 20)) | 
| 2548 | 11.2k |           val = val * 16 + (CUR - 'a') + 10; | 
| 2549 | 6.95k |       else if ((RAW >= 'A') && (RAW <= 'F') && (count < 20)) | 
| 2550 | 4.68k |           val = val * 16 + (CUR - 'A') + 10; | 
| 2551 | 2.26k |       else { | 
| 2552 | 2.26k |     xmlFatalErr(ctxt, XML_ERR_INVALID_HEX_CHARREF, NULL); | 
| 2553 | 2.26k |     val = 0; | 
| 2554 | 2.26k |     break; | 
| 2555 | 2.26k |       } | 
| 2556 | 26.3k |       if (val > 0x110000) | 
| 2557 | 10.2k |           val = 0x110000; | 
| 2558 |  |  | 
| 2559 | 26.3k |       NEXT; | 
| 2560 | 26.3k |       count++; | 
| 2561 | 26.3k |   } | 
| 2562 | 10.5k |   if (RAW == ';') { | 
| 2563 |  |       /* on purpose to avoid reentrancy problems with NEXT and SKIP */ | 
| 2564 | 8.30k |       ctxt->input->col++; | 
| 2565 | 8.30k |       ctxt->input->cur++; | 
| 2566 | 8.30k |   } | 
| 2567 | 24.2k |     } else if  ((RAW == '&') && (NXT(1) == '#')) { | 
| 2568 | 24.2k |   SKIP(2); | 
| 2569 | 24.2k |   GROW; | 
| 2570 | 187k |   while (RAW != ';') { /* loop blocked by count */ | 
| 2571 | 168k |       if (count++ > 20) { | 
| 2572 | 10.6k |     count = 0; | 
| 2573 | 10.6k |     GROW; | 
| 2574 | 10.6k |       } | 
| 2575 | 168k |       if ((RAW >= '0') && (RAW <= '9')) | 
| 2576 | 163k |           val = val * 10 + (CUR - '0'); | 
| 2577 | 4.79k |       else { | 
| 2578 | 4.79k |     xmlFatalErr(ctxt, XML_ERR_INVALID_DEC_CHARREF, NULL); | 
| 2579 | 4.79k |     val = 0; | 
| 2580 | 4.79k |     break; | 
| 2581 | 4.79k |       } | 
| 2582 | 163k |       if (val > 0x110000) | 
| 2583 | 113k |           val = 0x110000; | 
| 2584 |  |  | 
| 2585 | 163k |       NEXT; | 
| 2586 | 163k |       count++; | 
| 2587 | 163k |   } | 
| 2588 | 24.2k |   if (RAW == ';') { | 
| 2589 |  |       /* on purpose to avoid reentrancy problems with NEXT and SKIP */ | 
| 2590 | 19.4k |       ctxt->input->col++; | 
| 2591 | 19.4k |       ctxt->input->cur++; | 
| 2592 | 19.4k |   } | 
| 2593 | 24.2k |     } else { | 
| 2594 | 0 |         if (RAW == '&') | 
| 2595 | 0 |             SKIP(1); | 
| 2596 | 0 |         xmlFatalErr(ctxt, XML_ERR_INVALID_CHARREF, NULL); | 
| 2597 | 0 |     } | 
| 2598 |  |  | 
| 2599 |  |     /* | 
| 2600 |  |      * [ WFC: Legal Character ] | 
| 2601 |  |      * Characters referred to using character references must match the | 
| 2602 |  |      * production for Char. | 
| 2603 |  |      */ | 
| 2604 | 34.8k |     if (val >= 0x110000) { | 
| 2605 | 1.33k |         xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 2606 | 1.33k |                 "xmlParseCharRef: character reference out of bounds\n", | 
| 2607 | 1.33k |           val); | 
| 2608 | 1.33k |         val = 0xFFFD; | 
| 2609 | 33.5k |     } else if (!IS_CHAR(val)) { | 
| 2610 | 9.34k |         xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 2611 | 9.34k |                           "xmlParseCharRef: invalid xmlChar value %d\n", | 
| 2612 | 9.34k |                     val); | 
| 2613 | 9.34k |     } | 
| 2614 | 34.8k |     return(val); | 
| 2615 | 34.8k | } | 
| 2616 |  |  | 
| 2617 |  | /** | 
| 2618 |  |  * Parse Reference declarations, variant parsing from a string rather | 
| 2619 |  |  * than an an input flow. | 
| 2620 |  |  * | 
| 2621 |  |  *     [66] CharRef ::= '&#' [0-9]+ ';' | | 
| 2622 |  |  *                      '&#x' [0-9a-fA-F]+ ';' | 
| 2623 |  |  * | 
| 2624 |  |  * [ WFC: Legal Character ] | 
| 2625 |  |  * Characters referred to using character references must match the | 
| 2626 |  |  * production for Char. | 
| 2627 |  |  * | 
| 2628 |  |  * @param ctxt  an XML parser context | 
| 2629 |  |  * @param str  a pointer to an index in the string | 
| 2630 |  |  * @returns the value parsed (as an int), 0 in case of error, str will be | 
| 2631 |  |  *         updated to the current value of the index | 
| 2632 |  |  */ | 
| 2633 |  | static int | 
| 2634 | 223k | xmlParseStringCharRef(xmlParserCtxtPtr ctxt, const xmlChar **str) { | 
| 2635 | 223k |     const xmlChar *ptr; | 
| 2636 | 223k |     xmlChar cur; | 
| 2637 | 223k |     int val = 0; | 
| 2638 |  |  | 
| 2639 | 223k |     if ((str == NULL) || (*str == NULL)) return(0); | 
| 2640 | 223k |     ptr = *str; | 
| 2641 | 223k |     cur = *ptr; | 
| 2642 | 223k |     if ((cur == '&') && (ptr[1] == '#') && (ptr[2] == 'x')) { | 
| 2643 | 21.6k |   ptr += 3; | 
| 2644 | 21.6k |   cur = *ptr; | 
| 2645 | 316k |   while (cur != ';') { /* Non input consuming loop */ | 
| 2646 | 295k |       if ((cur >= '0') && (cur <= '9')) | 
| 2647 | 262k |           val = val * 16 + (cur - '0'); | 
| 2648 | 32.4k |       else if ((cur >= 'a') && (cur <= 'f')) | 
| 2649 | 10.0k |           val = val * 16 + (cur - 'a') + 10; | 
| 2650 | 22.3k |       else if ((cur >= 'A') && (cur <= 'F')) | 
| 2651 | 21.7k |           val = val * 16 + (cur - 'A') + 10; | 
| 2652 | 539 |       else { | 
| 2653 | 539 |     xmlFatalErr(ctxt, XML_ERR_INVALID_HEX_CHARREF, NULL); | 
| 2654 | 539 |     val = 0; | 
| 2655 | 539 |     break; | 
| 2656 | 539 |       } | 
| 2657 | 294k |       if (val > 0x110000) | 
| 2658 | 1.21k |           val = 0x110000; | 
| 2659 |  |  | 
| 2660 | 294k |       ptr++; | 
| 2661 | 294k |       cur = *ptr; | 
| 2662 | 294k |   } | 
| 2663 | 21.6k |   if (cur == ';') | 
| 2664 | 21.0k |       ptr++; | 
| 2665 | 202k |     } else if  ((cur == '&') && (ptr[1] == '#')){ | 
| 2666 | 202k |   ptr += 2; | 
| 2667 | 202k |   cur = *ptr; | 
| 2668 | 728k |   while (cur != ';') { /* Non input consuming loops */ | 
| 2669 | 528k |       if ((cur >= '0') && (cur <= '9')) | 
| 2670 | 526k |           val = val * 10 + (cur - '0'); | 
| 2671 | 1.35k |       else { | 
| 2672 | 1.35k |     xmlFatalErr(ctxt, XML_ERR_INVALID_DEC_CHARREF, NULL); | 
| 2673 | 1.35k |     val = 0; | 
| 2674 | 1.35k |     break; | 
| 2675 | 1.35k |       } | 
| 2676 | 526k |       if (val > 0x110000) | 
| 2677 | 2.24k |           val = 0x110000; | 
| 2678 |  |  | 
| 2679 | 526k |       ptr++; | 
| 2680 | 526k |       cur = *ptr; | 
| 2681 | 526k |   } | 
| 2682 | 202k |   if (cur == ';') | 
| 2683 | 200k |       ptr++; | 
| 2684 | 202k |     } else { | 
| 2685 | 0 |   xmlFatalErr(ctxt, XML_ERR_INVALID_CHARREF, NULL); | 
| 2686 | 0 |   return(0); | 
| 2687 | 0 |     } | 
| 2688 | 223k |     *str = ptr; | 
| 2689 |  |  | 
| 2690 |  |     /* | 
| 2691 |  |      * [ WFC: Legal Character ] | 
| 2692 |  |      * Characters referred to using character references must match the | 
| 2693 |  |      * production for Char. | 
| 2694 |  |      */ | 
| 2695 | 223k |     if (val >= 0x110000) { | 
| 2696 | 396 |         xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 2697 | 396 |                 "xmlParseStringCharRef: character reference out of bounds\n", | 
| 2698 | 396 |                 val); | 
| 2699 | 223k |     } else if (IS_CHAR(val)) { | 
| 2700 | 220k |         return(val); | 
| 2701 | 220k |     } else { | 
| 2702 | 3.04k |         xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 2703 | 3.04k |         "xmlParseStringCharRef: invalid xmlChar value %d\n", | 
| 2704 | 3.04k |         val); | 
| 2705 | 3.04k |     } | 
| 2706 | 3.44k |     return(0); | 
| 2707 | 223k | } | 
| 2708 |  |  | 
| 2709 |  | /** | 
| 2710 |  |  *     [69] PEReference ::= '%' Name ';' | 
| 2711 |  |  * | 
| 2712 |  |  * @deprecated Internal function, do not use. | 
| 2713 |  |  * | 
| 2714 |  |  * [ WFC: No Recursion ] | 
| 2715 |  |  * A parsed entity must not contain a recursive | 
| 2716 |  |  * reference to itself, either directly or indirectly. | 
| 2717 |  |  * | 
| 2718 |  |  * [ WFC: Entity Declared ] | 
| 2719 |  |  * In a document without any DTD, a document with only an internal DTD | 
| 2720 |  |  * subset which contains no parameter entity references, or a document | 
| 2721 |  |  * with "standalone='yes'", ...  ... The declaration of a parameter | 
| 2722 |  |  * entity must precede any reference to it... | 
| 2723 |  |  * | 
| 2724 |  |  * [ VC: Entity Declared ] | 
| 2725 |  |  * In a document with an external subset or external parameter entities | 
| 2726 |  |  * with "standalone='no'", ...  ... The declaration of a parameter entity | 
| 2727 |  |  * must precede any reference to it... | 
| 2728 |  |  * | 
| 2729 |  |  * [ WFC: In DTD ] | 
| 2730 |  |  * Parameter-entity references may only appear in the DTD. | 
| 2731 |  |  * NOTE: misleading but this is handled. | 
| 2732 |  |  * | 
| 2733 |  |  * A PEReference may have been detected in the current input stream | 
| 2734 |  |  * the handling is done accordingly to | 
| 2735 |  |  *      http://www.w3.org/TR/REC-xml#entproc | 
| 2736 |  |  * i.e. | 
| 2737 |  |  *   - Included in literal in entity values | 
| 2738 |  |  *   - Included as Parameter Entity reference within DTDs | 
| 2739 |  |  * @param ctxt  the parser context | 
| 2740 |  |  */ | 
| 2741 |  | void | 
| 2742 | 0 | xmlParserHandlePEReference(xmlParserCtxt *ctxt) { | 
| 2743 | 0 |     xmlParsePERefInternal(ctxt, 0); | 
| 2744 | 0 | } | 
| 2745 |  |  | 
| 2746 |  | /** | 
| 2747 |  |  * @deprecated Internal function, don't use. | 
| 2748 |  |  * | 
| 2749 |  |  * @param ctxt  the parser context | 
| 2750 |  |  * @param str  the input string | 
| 2751 |  |  * @param len  the string length | 
| 2752 |  |  * @param what  combination of XML_SUBSTITUTE_REF and XML_SUBSTITUTE_PEREF | 
| 2753 |  |  * @param end  an end marker xmlChar, 0 if none | 
| 2754 |  |  * @param end2  an end marker xmlChar, 0 if none | 
| 2755 |  |  * @param end3  an end marker xmlChar, 0 if none | 
| 2756 |  |  * @returns A newly allocated string with the substitution done. The caller | 
| 2757 |  |  *      must deallocate it ! | 
| 2758 |  |  */ | 
| 2759 |  | xmlChar * | 
| 2760 |  | xmlStringLenDecodeEntities(xmlParserCtxt *ctxt, const xmlChar *str, int len, | 
| 2761 |  |                            int what ATTRIBUTE_UNUSED, | 
| 2762 | 0 |                            xmlChar end, xmlChar end2, xmlChar end3) { | 
| 2763 | 0 |     if ((ctxt == NULL) || (str == NULL) || (len < 0)) | 
| 2764 | 0 |         return(NULL); | 
| 2765 |  |  | 
| 2766 | 0 |     if ((str[len] != 0) || | 
| 2767 | 0 |         (end != 0) || (end2 != 0) || (end3 != 0)) | 
| 2768 | 0 |         return(NULL); | 
| 2769 |  |  | 
| 2770 | 0 |     return(xmlExpandEntitiesInAttValue(ctxt, str, 0)); | 
| 2771 | 0 | } | 
| 2772 |  |  | 
| 2773 |  | /** | 
| 2774 |  |  * @deprecated Internal function, don't use. | 
| 2775 |  |  * | 
| 2776 |  |  * @param ctxt  the parser context | 
| 2777 |  |  * @param str  the input string | 
| 2778 |  |  * @param what  combination of XML_SUBSTITUTE_REF and XML_SUBSTITUTE_PEREF | 
| 2779 |  |  * @param end  an end marker xmlChar, 0 if none | 
| 2780 |  |  * @param end2  an end marker xmlChar, 0 if none | 
| 2781 |  |  * @param end3  an end marker xmlChar, 0 if none | 
| 2782 |  |  * @returns A newly allocated string with the substitution done. The caller | 
| 2783 |  |  *      must deallocate it ! | 
| 2784 |  |  */ | 
| 2785 |  | xmlChar * | 
| 2786 |  | xmlStringDecodeEntities(xmlParserCtxt *ctxt, const xmlChar *str, | 
| 2787 |  |                         int what ATTRIBUTE_UNUSED, | 
| 2788 | 0 |             xmlChar end, xmlChar  end2, xmlChar end3) { | 
| 2789 | 0 |     if ((ctxt == NULL) || (str == NULL)) | 
| 2790 | 0 |         return(NULL); | 
| 2791 |  |  | 
| 2792 | 0 |     if ((end != 0) || (end2 != 0) || (end3 != 0)) | 
| 2793 | 0 |         return(NULL); | 
| 2794 |  |  | 
| 2795 | 0 |     return(xmlExpandEntitiesInAttValue(ctxt, str, 0)); | 
| 2796 | 0 | } | 
| 2797 |  |  | 
| 2798 |  | /************************************************************************ | 
| 2799 |  |  *                  * | 
| 2800 |  |  *    Commodity functions, cleanup needed ?     * | 
| 2801 |  |  *                  * | 
| 2802 |  |  ************************************************************************/ | 
| 2803 |  |  | 
| 2804 |  | /** | 
| 2805 |  |  * Is this a sequence of blank chars that one can ignore ? | 
| 2806 |  |  * | 
| 2807 |  |  * @param ctxt  an XML parser context | 
| 2808 |  |  * @param str  a xmlChar * | 
| 2809 |  |  * @param len  the size of `str` | 
| 2810 |  |  * @param blank_chars  we know the chars are blanks | 
| 2811 |  |  * @returns 1 if ignorable 0 otherwise. | 
| 2812 |  |  */ | 
| 2813 |  |  | 
| 2814 |  | static int areBlanks(xmlParserCtxtPtr ctxt, const xmlChar *str, int len, | 
| 2815 | 0 |                      int blank_chars) { | 
| 2816 | 0 |     int i; | 
| 2817 | 0 |     xmlNodePtr lastChild; | 
| 2818 |  |  | 
| 2819 |  |     /* | 
| 2820 |  |      * Check for xml:space value. | 
| 2821 |  |      */ | 
| 2822 | 0 |     if ((ctxt->space == NULL) || (*(ctxt->space) == 1) || | 
| 2823 | 0 |         (*(ctxt->space) == -2)) | 
| 2824 | 0 |   return(0); | 
| 2825 |  |  | 
| 2826 |  |     /* | 
| 2827 |  |      * Check that the string is made of blanks | 
| 2828 |  |      */ | 
| 2829 | 0 |     if (blank_chars == 0) { | 
| 2830 | 0 |   for (i = 0;i < len;i++) | 
| 2831 | 0 |       if (!(IS_BLANK_CH(str[i]))) return(0); | 
| 2832 | 0 |     } | 
| 2833 |  |  | 
| 2834 |  |     /* | 
| 2835 |  |      * Look if the element is mixed content in the DTD if available | 
| 2836 |  |      */ | 
| 2837 | 0 |     if (ctxt->node == NULL) return(0); | 
| 2838 | 0 |     if (ctxt->myDoc != NULL) { | 
| 2839 | 0 |         xmlElementPtr elemDecl = NULL; | 
| 2840 | 0 |         xmlDocPtr doc = ctxt->myDoc; | 
| 2841 | 0 |         const xmlChar *prefix = NULL; | 
| 2842 |  | 
 | 
| 2843 | 0 |         if (ctxt->node->ns) | 
| 2844 | 0 |             prefix = ctxt->node->ns->prefix; | 
| 2845 | 0 |         if (doc->intSubset != NULL) | 
| 2846 | 0 |             elemDecl = xmlHashLookup2(doc->intSubset->elements, ctxt->node->name, | 
| 2847 | 0 |                                       prefix); | 
| 2848 | 0 |         if ((elemDecl == NULL) && (doc->extSubset != NULL)) | 
| 2849 | 0 |             elemDecl = xmlHashLookup2(doc->extSubset->elements, ctxt->node->name, | 
| 2850 | 0 |                                       prefix); | 
| 2851 | 0 |         if (elemDecl != NULL) { | 
| 2852 | 0 |             if (elemDecl->etype == XML_ELEMENT_TYPE_ELEMENT) | 
| 2853 | 0 |                 return(1); | 
| 2854 | 0 |             if ((elemDecl->etype == XML_ELEMENT_TYPE_ANY) || | 
| 2855 | 0 |                 (elemDecl->etype == XML_ELEMENT_TYPE_MIXED)) | 
| 2856 | 0 |                 return(0); | 
| 2857 | 0 |         } | 
| 2858 | 0 |     } | 
| 2859 |  |  | 
| 2860 |  |     /* | 
| 2861 |  |      * Otherwise, heuristic :-\ | 
| 2862 |  |      * | 
| 2863 |  |      * When push parsing, we could be at the end of a chunk. | 
| 2864 |  |      * This makes the look-ahead and consequently the NOBLANKS | 
| 2865 |  |      * option unreliable. | 
| 2866 |  |      */ | 
| 2867 | 0 |     if ((RAW != '<') && (RAW != 0xD)) return(0); | 
| 2868 | 0 |     if ((ctxt->node->children == NULL) && | 
| 2869 | 0 |   (RAW == '<') && (NXT(1) == '/')) return(0); | 
| 2870 |  |  | 
| 2871 | 0 |     lastChild = xmlGetLastChild(ctxt->node); | 
| 2872 | 0 |     if (lastChild == NULL) { | 
| 2873 | 0 |         if ((ctxt->node->type != XML_ELEMENT_NODE) && | 
| 2874 | 0 |             (ctxt->node->content != NULL)) return(0); | 
| 2875 | 0 |     } else if (xmlNodeIsText(lastChild)) | 
| 2876 | 0 |         return(0); | 
| 2877 | 0 |     else if ((ctxt->node->children != NULL) && | 
| 2878 | 0 |              (xmlNodeIsText(ctxt->node->children))) | 
| 2879 | 0 |         return(0); | 
| 2880 | 0 |     return(1); | 
| 2881 | 0 | } | 
| 2882 |  |  | 
| 2883 |  | /************************************************************************ | 
| 2884 |  |  *                  * | 
| 2885 |  |  *    Extra stuff for namespace support     * | 
| 2886 |  |  *  Relates to http://www.w3.org/TR/WD-xml-names      * | 
| 2887 |  |  *                  * | 
| 2888 |  |  ************************************************************************/ | 
| 2889 |  |  | 
| 2890 |  | /** | 
| 2891 |  |  * Parse an UTF8 encoded XML qualified name string | 
| 2892 |  |  * | 
| 2893 |  |  * @deprecated Don't use. | 
| 2894 |  |  * | 
| 2895 |  |  * @param ctxt  an XML parser context | 
| 2896 |  |  * @param name  an XML parser context | 
| 2897 |  |  * @param prefixOut  a xmlChar ** | 
| 2898 |  |  * @returns the local part, and prefix is updated | 
| 2899 |  |  *   to get the Prefix if any. | 
| 2900 |  |  */ | 
| 2901 |  |  | 
| 2902 |  | xmlChar * | 
| 2903 | 0 | xmlSplitQName(xmlParserCtxt *ctxt, const xmlChar *name, xmlChar **prefixOut) { | 
| 2904 | 0 |     xmlChar *ret; | 
| 2905 | 0 |     const xmlChar *localname; | 
| 2906 |  | 
 | 
| 2907 | 0 |     localname = xmlSplitQName4(name, prefixOut); | 
| 2908 | 0 |     if (localname == NULL) { | 
| 2909 | 0 |         xmlCtxtErrMemory(ctxt); | 
| 2910 | 0 |         return(NULL); | 
| 2911 | 0 |     } | 
| 2912 |  |  | 
| 2913 | 0 |     ret = xmlStrdup(localname); | 
| 2914 | 0 |     if (ret == NULL) { | 
| 2915 | 0 |         xmlCtxtErrMemory(ctxt); | 
| 2916 | 0 |         xmlFree(*prefixOut); | 
| 2917 | 0 |     } | 
| 2918 |  | 
 | 
| 2919 | 0 |     return(ret); | 
| 2920 | 0 | } | 
| 2921 |  |  | 
| 2922 |  | /************************************************************************ | 
| 2923 |  |  *                  * | 
| 2924 |  |  *      The parser itself       * | 
| 2925 |  |  *  Relates to http://www.w3.org/TR/REC-xml       * | 
| 2926 |  |  *                  * | 
| 2927 |  |  ************************************************************************/ | 
| 2928 |  |  | 
| 2929 |  | /************************************************************************ | 
| 2930 |  |  *                  * | 
| 2931 |  |  *  Routines to parse Name, NCName and NmToken      * | 
| 2932 |  |  *                  * | 
| 2933 |  |  ************************************************************************/ | 
| 2934 |  |  | 
| 2935 |  | /* | 
| 2936 |  |  * The two following functions are related to the change of accepted | 
| 2937 |  |  * characters for Name and NmToken in the Revision 5 of XML-1.0 | 
| 2938 |  |  * They correspond to the modified production [4] and the new production [4a] | 
| 2939 |  |  * changes in that revision. Also note that the macros used for the | 
| 2940 |  |  * productions Letter, Digit, CombiningChar and Extender are not needed | 
| 2941 |  |  * anymore. | 
| 2942 |  |  * We still keep compatibility to pre-revision5 parsing semantic if the | 
| 2943 |  |  * new XML_PARSE_OLD10 option is given to the parser. | 
| 2944 |  |  */ | 
| 2945 |  |  | 
| 2946 |  | static int | 
| 2947 | 3.53M | xmlIsNameStartCharNew(int c) { | 
| 2948 |  |     /* | 
| 2949 |  |      * Use the new checks of production [4] [4a] amd [5] of the | 
| 2950 |  |      * Update 5 of XML-1.0 | 
| 2951 |  |      */ | 
| 2952 | 3.53M |     if ((c != ' ') && (c != '>') && (c != '/') && /* accelerators */ | 
| 2953 | 3.53M |         (((c >= 'a') && (c <= 'z')) || | 
| 2954 | 3.25M |          ((c >= 'A') && (c <= 'Z')) || | 
| 2955 | 3.22M |          (c == '_') || (c == ':') || | 
| 2956 | 3.04M |          ((c >= 0xC0) && (c <= 0xD6)) || | 
| 2957 | 3.04M |          ((c >= 0xD8) && (c <= 0xF6)) || | 
| 2958 | 3.04M |          ((c >= 0xF8) && (c <= 0x2FF)) || | 
| 2959 | 3.03M |          ((c >= 0x370) && (c <= 0x37D)) || | 
| 2960 | 3.03M |          ((c >= 0x37F) && (c <= 0x1FFF)) || | 
| 2961 | 3.03M |          ((c >= 0x200C) && (c <= 0x200D)) || | 
| 2962 | 3.03M |          ((c >= 0x2070) && (c <= 0x218F)) || | 
| 2963 | 3.03M |          ((c >= 0x2C00) && (c <= 0x2FEF)) || | 
| 2964 | 3.03M |          ((c >= 0x3001) && (c <= 0xD7FF)) || | 
| 2965 | 3.02M |          ((c >= 0xF900) && (c <= 0xFDCF)) || | 
| 2966 | 3.02M |          ((c >= 0xFDF0) && (c <= 0xFFFD)) || | 
| 2967 | 3.02M |          ((c >= 0x10000) && (c <= 0xEFFFF)))) | 
| 2968 | 513k |         return(1); | 
| 2969 | 3.02M |     return(0); | 
| 2970 | 3.53M | } | 
| 2971 |  |  | 
| 2972 |  | static int | 
| 2973 | 16.3M | xmlIsNameCharNew(int c) { | 
| 2974 |  |     /* | 
| 2975 |  |      * Use the new checks of production [4] [4a] amd [5] of the | 
| 2976 |  |      * Update 5 of XML-1.0 | 
| 2977 |  |      */ | 
| 2978 | 16.3M |     if ((c != ' ') && (c != '>') && (c != '/') && /* accelerators */ | 
| 2979 | 16.3M |         (((c >= 'a') && (c <= 'z')) || | 
| 2980 | 16.1M |          ((c >= 'A') && (c <= 'Z')) || | 
| 2981 | 13.7M |          ((c >= '0') && (c <= '9')) || /* !start */ | 
| 2982 | 13.5M |          (c == '_') || (c == ':') || | 
| 2983 | 13.5M |          (c == '-') || (c == '.') || (c == 0xB7) || /* !start */ | 
| 2984 | 13.5M |          ((c >= 0xC0) && (c <= 0xD6)) || | 
| 2985 | 13.5M |          ((c >= 0xD8) && (c <= 0xF6)) || | 
| 2986 | 13.5M |          ((c >= 0xF8) && (c <= 0x2FF)) || | 
| 2987 | 12.5M |          ((c >= 0x300) && (c <= 0x36F)) || /* !start */ | 
| 2988 | 12.5M |          ((c >= 0x370) && (c <= 0x37D)) || | 
| 2989 | 12.5M |          ((c >= 0x37F) && (c <= 0x1FFF)) || | 
| 2990 | 12.5M |          ((c >= 0x200C) && (c <= 0x200D)) || | 
| 2991 | 12.5M |          ((c >= 0x203F) && (c <= 0x2040)) || /* !start */ | 
| 2992 | 12.5M |          ((c >= 0x2070) && (c <= 0x218F)) || | 
| 2993 | 12.5M |          ((c >= 0x2C00) && (c <= 0x2FEF)) || | 
| 2994 | 12.5M |          ((c >= 0x3001) && (c <= 0xD7FF)) || | 
| 2995 | 11.7M |          ((c >= 0xF900) && (c <= 0xFDCF)) || | 
| 2996 | 11.7M |          ((c >= 0xFDF0) && (c <= 0xFFFD)) || | 
| 2997 | 524k |          ((c >= 0x10000) && (c <= 0xEFFFF)))) | 
| 2998 | 15.7M |          return(1); | 
| 2999 | 528k |     return(0); | 
| 3000 | 16.3M | } | 
| 3001 |  |  | 
| 3002 |  | static int | 
| 3003 | 0 | xmlIsNameStartCharOld(int c) { | 
| 3004 | 0 |     if ((c != ' ') && (c != '>') && (c != '/') && /* accelerators */ | 
| 3005 | 0 |         ((IS_LETTER(c) || (c == '_') || (c == ':')))) | 
| 3006 | 0 |         return(1); | 
| 3007 | 0 |     return(0); | 
| 3008 | 0 | } | 
| 3009 |  |  | 
| 3010 |  | static int | 
| 3011 | 0 | xmlIsNameCharOld(int c) { | 
| 3012 | 0 |     if ((c != ' ') && (c != '>') && (c != '/') && /* accelerators */ | 
| 3013 | 0 |         ((IS_LETTER(c)) || (IS_DIGIT(c)) || | 
| 3014 | 0 |          (c == '.') || (c == '-') || | 
| 3015 | 0 |          (c == '_') || (c == ':') || | 
| 3016 | 0 |          (IS_COMBINING(c)) || | 
| 3017 | 0 |          (IS_EXTENDER(c)))) | 
| 3018 | 0 |         return(1); | 
| 3019 | 0 |     return(0); | 
| 3020 | 0 | } | 
| 3021 |  |  | 
| 3022 |  | static int | 
| 3023 | 3.53M | xmlIsNameStartChar(int c, int old10) { | 
| 3024 | 3.53M |     if (!old10) | 
| 3025 | 3.53M |         return(xmlIsNameStartCharNew(c)); | 
| 3026 | 0 |     else | 
| 3027 | 0 |         return(xmlIsNameStartCharOld(c)); | 
| 3028 | 3.53M | } | 
| 3029 |  |  | 
| 3030 |  | static int | 
| 3031 | 16.3M | xmlIsNameChar(int c, int old10) { | 
| 3032 | 16.3M |     if (!old10) | 
| 3033 | 16.3M |         return(xmlIsNameCharNew(c)); | 
| 3034 | 0 |     else | 
| 3035 | 0 |         return(xmlIsNameCharOld(c)); | 
| 3036 | 16.3M | } | 
| 3037 |  |  | 
| 3038 |  | /* | 
| 3039 |  |  * Scan an XML Name, NCName or Nmtoken. | 
| 3040 |  |  * | 
| 3041 |  |  * Returns a pointer to the end of the name on success. If the | 
| 3042 |  |  * name is invalid, returns `ptr`. If the name is longer than | 
| 3043 |  |  * `maxSize` bytes, returns NULL. | 
| 3044 |  |  * | 
| 3045 |  |  * @param ptr  pointer to the start of the name | 
| 3046 |  |  * @param maxSize  maximum size in bytes | 
| 3047 |  |  * @param flags  XML_SCAN_* flags | 
| 3048 |  |  * @returns a pointer to the end of the name or NULL | 
| 3049 |  |  */ | 
| 3050 |  | const xmlChar * | 
| 3051 | 444k | xmlScanName(const xmlChar *ptr, size_t maxSize, int flags) { | 
| 3052 | 444k |     int stop = flags & XML_SCAN_NC ? ':' : 0; | 
| 3053 | 444k |     int old10 = flags & XML_SCAN_OLD10 ? 1 : 0; | 
| 3054 |  |  | 
| 3055 | 3.10M |     while (1) { | 
| 3056 | 3.10M |         int c, len; | 
| 3057 |  |  | 
| 3058 | 3.10M |         c = *ptr; | 
| 3059 | 3.10M |         if (c < 0x80) { | 
| 3060 | 1.28M |             if (c == stop) | 
| 3061 | 252 |                 break; | 
| 3062 | 1.28M |             len = 1; | 
| 3063 | 1.82M |         } else { | 
| 3064 | 1.82M |             len = 4; | 
| 3065 | 1.82M |             c = xmlGetUTF8Char(ptr, &len); | 
| 3066 | 1.82M |             if (c < 0) | 
| 3067 | 502 |                 break; | 
| 3068 | 1.82M |         } | 
| 3069 |  |  | 
| 3070 | 3.10M |         if (flags & XML_SCAN_NMTOKEN ? | 
| 3071 | 2.65M |                 !xmlIsNameChar(c, old10) : | 
| 3072 | 3.10M |                 !xmlIsNameStartChar(c, old10)) | 
| 3073 | 443k |             break; | 
| 3074 |  |  | 
| 3075 | 2.66M |         if ((size_t) len > maxSize) | 
| 3076 | 0 |             return(NULL); | 
| 3077 | 2.66M |         ptr += len; | 
| 3078 | 2.66M |         maxSize -= len; | 
| 3079 | 2.66M |         flags |= XML_SCAN_NMTOKEN; | 
| 3080 | 2.66M |     } | 
| 3081 |  |  | 
| 3082 | 444k |     return(ptr); | 
| 3083 | 444k | } | 
| 3084 |  |  | 
| 3085 |  | static const xmlChar * | 
| 3086 | 78.2k | xmlParseNameComplex(xmlParserCtxtPtr ctxt) { | 
| 3087 | 78.2k |     const xmlChar *ret; | 
| 3088 | 78.2k |     int len = 0, l; | 
| 3089 | 78.2k |     int c; | 
| 3090 | 78.2k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3091 | 78.2k |                     XML_MAX_TEXT_LENGTH : | 
| 3092 | 78.2k |                     XML_MAX_NAME_LENGTH; | 
| 3093 | 78.2k |     int old10 = (ctxt->options & XML_PARSE_OLD10) ? 1 : 0; | 
| 3094 |  |  | 
| 3095 |  |     /* | 
| 3096 |  |      * Handler for more complex cases | 
| 3097 |  |      */ | 
| 3098 | 78.2k |     c = xmlCurrentChar(ctxt, &l); | 
| 3099 | 78.2k |     if (!xmlIsNameStartChar(c, old10)) | 
| 3100 | 57.1k |         return(NULL); | 
| 3101 | 21.0k |     len += l; | 
| 3102 | 21.0k |     NEXTL(l); | 
| 3103 | 21.0k |     c = xmlCurrentChar(ctxt, &l); | 
| 3104 | 3.77M |     while (xmlIsNameChar(c, old10)) { | 
| 3105 | 3.75M |         if (len <= INT_MAX - l) | 
| 3106 | 3.75M |             len += l; | 
| 3107 | 3.75M |         NEXTL(l); | 
| 3108 | 3.75M |         c = xmlCurrentChar(ctxt, &l); | 
| 3109 | 3.75M |     } | 
| 3110 | 21.0k |     if (len > maxLength) { | 
| 3111 | 0 |         xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "Name"); | 
| 3112 | 0 |         return(NULL); | 
| 3113 | 0 |     } | 
| 3114 | 21.0k |     if (ctxt->input->cur - ctxt->input->base < len) { | 
| 3115 |  |         /* | 
| 3116 |  |          * There were a couple of bugs where PERefs lead to to a change | 
| 3117 |  |          * of the buffer. Check the buffer size to avoid passing an invalid | 
| 3118 |  |          * pointer to xmlDictLookup. | 
| 3119 |  |          */ | 
| 3120 | 0 |         xmlFatalErr(ctxt, XML_ERR_INTERNAL_ERROR, | 
| 3121 | 0 |                     "unexpected change of input buffer"); | 
| 3122 | 0 |         return (NULL); | 
| 3123 | 0 |     } | 
| 3124 | 21.0k |     if ((*ctxt->input->cur == '\n') && (ctxt->input->cur[-1] == '\r')) | 
| 3125 | 1.44k |         ret = xmlDictLookup(ctxt->dict, ctxt->input->cur - (len + 1), len); | 
| 3126 | 19.6k |     else | 
| 3127 | 19.6k |         ret = xmlDictLookup(ctxt->dict, ctxt->input->cur - len, len); | 
| 3128 | 21.0k |     if (ret == NULL) | 
| 3129 | 0 |         xmlErrMemory(ctxt); | 
| 3130 | 21.0k |     return(ret); | 
| 3131 | 21.0k | } | 
| 3132 |  |  | 
| 3133 |  | /** | 
| 3134 |  |  * Parse an XML name. | 
| 3135 |  |  * | 
| 3136 |  |  * @deprecated Internal function, don't use. | 
| 3137 |  |  * | 
| 3138 |  |  *     [4] NameChar ::= Letter | Digit | '.' | '-' | '_' | ':' | | 
| 3139 |  |  *                      CombiningChar | Extender | 
| 3140 |  |  * | 
| 3141 |  |  *     [5] Name ::= (Letter | '_' | ':') (NameChar)* | 
| 3142 |  |  * | 
| 3143 |  |  *     [6] Names ::= Name (#x20 Name)* | 
| 3144 |  |  * | 
| 3145 |  |  * @param ctxt  an XML parser context | 
| 3146 |  |  * @returns the Name parsed or NULL | 
| 3147 |  |  */ | 
| 3148 |  |  | 
| 3149 |  | const xmlChar * | 
| 3150 | 439k | xmlParseName(xmlParserCtxt *ctxt) { | 
| 3151 | 439k |     const xmlChar *in; | 
| 3152 | 439k |     const xmlChar *ret; | 
| 3153 | 439k |     size_t count = 0; | 
| 3154 | 439k |     size_t maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3155 | 439k |                        XML_MAX_TEXT_LENGTH : | 
| 3156 | 439k |                        XML_MAX_NAME_LENGTH; | 
| 3157 |  |  | 
| 3158 | 439k |     GROW; | 
| 3159 |  |  | 
| 3160 |  |     /* | 
| 3161 |  |      * Accelerator for simple ASCII names | 
| 3162 |  |      */ | 
| 3163 | 439k |     in = ctxt->input->cur; | 
| 3164 | 439k |     if (((*in >= 0x61) && (*in <= 0x7A)) || | 
| 3165 | 143k |   ((*in >= 0x41) && (*in <= 0x5A)) || | 
| 3166 | 375k |   (*in == '_') || (*in == ':')) { | 
| 3167 | 375k |   in++; | 
| 3168 | 5.21M |   while (((*in >= 0x61) && (*in <= 0x7A)) || | 
| 3169 | 4.85M |          ((*in >= 0x41) && (*in <= 0x5A)) || | 
| 3170 | 1.32M |          ((*in >= 0x30) && (*in <= 0x39)) || | 
| 3171 | 432k |          (*in == '_') || (*in == '-') || | 
| 3172 | 393k |          (*in == ':') || (*in == '.')) | 
| 3173 | 4.84M |       in++; | 
| 3174 | 375k |   if ((*in > 0) && (*in < 0x80)) { | 
| 3175 | 361k |       count = in - ctxt->input->cur; | 
| 3176 | 361k |             if (count > maxLength) { | 
| 3177 | 0 |                 xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "Name"); | 
| 3178 | 0 |                 return(NULL); | 
| 3179 | 0 |             } | 
| 3180 | 361k |       ret = xmlDictLookup(ctxt->dict, ctxt->input->cur, count); | 
| 3181 | 361k |       ctxt->input->cur = in; | 
| 3182 | 361k |       ctxt->input->col += count; | 
| 3183 | 361k |       if (ret == NULL) | 
| 3184 | 0 |           xmlErrMemory(ctxt); | 
| 3185 | 361k |       return(ret); | 
| 3186 | 361k |   } | 
| 3187 | 375k |     } | 
| 3188 |  |     /* accelerator for special cases */ | 
| 3189 | 78.2k |     return(xmlParseNameComplex(ctxt)); | 
| 3190 | 439k | } | 
| 3191 |  |  | 
| 3192 |  | static xmlHashedString | 
| 3193 | 3.02M | xmlParseNCNameComplex(xmlParserCtxtPtr ctxt) { | 
| 3194 | 3.02M |     xmlHashedString ret; | 
| 3195 | 3.02M |     int len = 0, l; | 
| 3196 | 3.02M |     int c; | 
| 3197 | 3.02M |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3198 | 3.02M |                     XML_MAX_TEXT_LENGTH : | 
| 3199 | 3.02M |                     XML_MAX_NAME_LENGTH; | 
| 3200 | 3.02M |     int old10 = (ctxt->options & XML_PARSE_OLD10) ? 1 : 0; | 
| 3201 | 3.02M |     size_t startPosition = 0; | 
| 3202 |  |  | 
| 3203 | 3.02M |     ret.name = NULL; | 
| 3204 | 3.02M |     ret.hashValue = 0; | 
| 3205 |  |  | 
| 3206 |  |     /* | 
| 3207 |  |      * Handler for more complex cases | 
| 3208 |  |      */ | 
| 3209 | 3.02M |     startPosition = CUR_PTR - BASE_PTR; | 
| 3210 | 3.02M |     c = xmlCurrentChar(ctxt, &l); | 
| 3211 | 3.02M |     if ((c == ' ') || (c == '>') || (c == '/') || /* accelerators */ | 
| 3212 | 3.01M |   (!xmlIsNameStartChar(c, old10) || (c == ':'))) { | 
| 3213 | 2.98M |   return(ret); | 
| 3214 | 2.98M |     } | 
| 3215 |  |  | 
| 3216 | 5.60M |     while ((c != ' ') && (c != '>') && (c != '/') && /* test bigname.xml */ | 
| 3217 | 5.59M |      (xmlIsNameChar(c, old10) && (c != ':'))) { | 
| 3218 | 5.55M |         if (len <= INT_MAX - l) | 
| 3219 | 5.55M |       len += l; | 
| 3220 | 5.55M |   NEXTL(l); | 
| 3221 | 5.55M |   c = xmlCurrentChar(ctxt, &l); | 
| 3222 | 5.55M |     } | 
| 3223 | 42.8k |     if (len > maxLength) { | 
| 3224 | 0 |         xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "NCName"); | 
| 3225 | 0 |         return(ret); | 
| 3226 | 0 |     } | 
| 3227 | 42.8k |     ret = xmlDictLookupHashed(ctxt->dict, (BASE_PTR + startPosition), len); | 
| 3228 | 42.8k |     if (ret.name == NULL) | 
| 3229 | 0 |         xmlErrMemory(ctxt); | 
| 3230 | 42.8k |     return(ret); | 
| 3231 | 42.8k | } | 
| 3232 |  |  | 
| 3233 |  | /** | 
| 3234 |  |  * Parse an XML name. | 
| 3235 |  |  * | 
| 3236 |  |  *     [4NS] NCNameChar ::= Letter | Digit | '.' | '-' | '_' | | 
| 3237 |  |  *                          CombiningChar | Extender | 
| 3238 |  |  * | 
| 3239 |  |  *     [5NS] NCName ::= (Letter | '_') (NCNameChar)* | 
| 3240 |  |  * | 
| 3241 |  |  * @param ctxt  an XML parser context | 
| 3242 |  |  * @returns the Name parsed or NULL | 
| 3243 |  |  */ | 
| 3244 |  |  | 
| 3245 |  | static xmlHashedString | 
| 3246 | 4.19M | xmlParseNCName(xmlParserCtxtPtr ctxt) { | 
| 3247 | 4.19M |     const xmlChar *in, *e; | 
| 3248 | 4.19M |     xmlHashedString ret; | 
| 3249 | 4.19M |     size_t count = 0; | 
| 3250 | 4.19M |     size_t maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3251 | 4.19M |                        XML_MAX_TEXT_LENGTH : | 
| 3252 | 4.19M |                        XML_MAX_NAME_LENGTH; | 
| 3253 |  |  | 
| 3254 | 4.19M |     ret.name = NULL; | 
| 3255 |  |  | 
| 3256 |  |     /* | 
| 3257 |  |      * Accelerator for simple ASCII names | 
| 3258 |  |      */ | 
| 3259 | 4.19M |     in = ctxt->input->cur; | 
| 3260 | 4.19M |     e = ctxt->input->end; | 
| 3261 | 4.19M |     if ((((*in >= 0x61) && (*in <= 0x7A)) || | 
| 3262 | 3.35M |    ((*in >= 0x41) && (*in <= 0x5A)) || | 
| 3263 | 3.00M |    (*in == '_')) && (in < e)) { | 
| 3264 | 1.19M |   in++; | 
| 3265 | 4.63M |   while ((((*in >= 0x61) && (*in <= 0x7A)) || | 
| 3266 | 3.98M |           ((*in >= 0x41) && (*in <= 0x5A)) || | 
| 3267 | 1.29M |           ((*in >= 0x30) && (*in <= 0x39)) || | 
| 3268 | 1.24M |           (*in == '_') || (*in == '-') || | 
| 3269 | 3.44M |           (*in == '.')) && (in < e)) | 
| 3270 | 3.44M |       in++; | 
| 3271 | 1.19M |   if (in >= e) | 
| 3272 | 1.24k |       goto complex; | 
| 3273 | 1.19M |   if ((*in > 0) && (*in < 0x80)) { | 
| 3274 | 1.16M |       count = in - ctxt->input->cur; | 
| 3275 | 1.16M |             if (count > maxLength) { | 
| 3276 | 0 |                 xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "NCName"); | 
| 3277 | 0 |                 return(ret); | 
| 3278 | 0 |             } | 
| 3279 | 1.16M |       ret = xmlDictLookupHashed(ctxt->dict, ctxt->input->cur, count); | 
| 3280 | 1.16M |       ctxt->input->cur = in; | 
| 3281 | 1.16M |       ctxt->input->col += count; | 
| 3282 | 1.16M |       if (ret.name == NULL) { | 
| 3283 | 0 |           xmlErrMemory(ctxt); | 
| 3284 | 0 |       } | 
| 3285 | 1.16M |       return(ret); | 
| 3286 | 1.16M |   } | 
| 3287 | 1.19M |     } | 
| 3288 | 3.02M | complex: | 
| 3289 | 3.02M |     return(xmlParseNCNameComplex(ctxt)); | 
| 3290 | 4.19M | } | 
| 3291 |  |  | 
| 3292 |  | /** | 
| 3293 |  |  * Parse an XML name and compares for match | 
| 3294 |  |  * (specialized for endtag parsing) | 
| 3295 |  |  * | 
| 3296 |  |  * @param ctxt  an XML parser context | 
| 3297 |  |  * @param other  the name to compare with | 
| 3298 |  |  * @returns NULL for an illegal name, (xmlChar*) 1 for success | 
| 3299 |  |  * and the name for mismatch | 
| 3300 |  |  */ | 
| 3301 |  |  | 
| 3302 |  | static const xmlChar * | 
| 3303 | 10.5k | xmlParseNameAndCompare(xmlParserCtxtPtr ctxt, xmlChar const *other) { | 
| 3304 | 10.5k |     register const xmlChar *cmp = other; | 
| 3305 | 10.5k |     register const xmlChar *in; | 
| 3306 | 10.5k |     const xmlChar *ret; | 
| 3307 |  |  | 
| 3308 | 10.5k |     GROW; | 
| 3309 |  |  | 
| 3310 | 10.5k |     in = ctxt->input->cur; | 
| 3311 | 17.2k |     while (*in != 0 && *in == *cmp) { | 
| 3312 | 6.73k |   ++in; | 
| 3313 | 6.73k |   ++cmp; | 
| 3314 | 6.73k |     } | 
| 3315 | 10.5k |     if (*cmp == 0 && (*in == '>' || IS_BLANK_CH (*in))) { | 
| 3316 |  |   /* success */ | 
| 3317 | 2.93k |   ctxt->input->col += in - ctxt->input->cur; | 
| 3318 | 2.93k |   ctxt->input->cur = in; | 
| 3319 | 2.93k |   return (const xmlChar*) 1; | 
| 3320 | 2.93k |     } | 
| 3321 |  |     /* failure (or end of input buffer), check with full function */ | 
| 3322 | 7.59k |     ret = xmlParseName (ctxt); | 
| 3323 |  |     /* strings coming from the dictionary direct compare possible */ | 
| 3324 | 7.59k |     if (ret == other) { | 
| 3325 | 2.65k |   return (const xmlChar*) 1; | 
| 3326 | 2.65k |     } | 
| 3327 | 4.94k |     return ret; | 
| 3328 | 7.59k | } | 
| 3329 |  |  | 
| 3330 |  | /** | 
| 3331 |  |  * Parse an XML name. | 
| 3332 |  |  * | 
| 3333 |  |  * @param ctxt  an XML parser context | 
| 3334 |  |  * @param str  a pointer to the string pointer (IN/OUT) | 
| 3335 |  |  * @returns the Name parsed or NULL. The `str` pointer | 
| 3336 |  |  * is updated to the current location in the string. | 
| 3337 |  |  */ | 
| 3338 |  |  | 
| 3339 |  | static xmlChar * | 
| 3340 | 441k | xmlParseStringName(xmlParserCtxtPtr ctxt, const xmlChar** str) { | 
| 3341 | 441k |     xmlChar *ret; | 
| 3342 | 441k |     const xmlChar *cur = *str; | 
| 3343 | 441k |     int flags = 0; | 
| 3344 | 441k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3345 | 441k |                     XML_MAX_TEXT_LENGTH : | 
| 3346 | 441k |                     XML_MAX_NAME_LENGTH; | 
| 3347 |  |  | 
| 3348 | 441k |     if (ctxt->options & XML_PARSE_OLD10) | 
| 3349 | 0 |         flags |= XML_SCAN_OLD10; | 
| 3350 |  |  | 
| 3351 | 441k |     cur = xmlScanName(*str, maxLength, flags); | 
| 3352 | 441k |     if (cur == NULL) { | 
| 3353 | 0 |         xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "NCName"); | 
| 3354 | 0 |         return(NULL); | 
| 3355 | 0 |     } | 
| 3356 | 441k |     if (cur == *str) | 
| 3357 | 1.67k |         return(NULL); | 
| 3358 |  |  | 
| 3359 | 439k |     ret = xmlStrndup(*str, cur - *str); | 
| 3360 | 439k |     if (ret == NULL) | 
| 3361 | 0 |         xmlErrMemory(ctxt); | 
| 3362 | 439k |     *str = cur; | 
| 3363 | 439k |     return(ret); | 
| 3364 | 441k | } | 
| 3365 |  |  | 
| 3366 |  | /** | 
| 3367 |  |  * Parse an XML Nmtoken. | 
| 3368 |  |  * | 
| 3369 |  |  * @deprecated Internal function, don't use. | 
| 3370 |  |  * | 
| 3371 |  |  *     [7] Nmtoken ::= (NameChar)+ | 
| 3372 |  |  * | 
| 3373 |  |  *     [8] Nmtokens ::= Nmtoken (#x20 Nmtoken)* | 
| 3374 |  |  * | 
| 3375 |  |  * @param ctxt  an XML parser context | 
| 3376 |  |  * @returns the Nmtoken parsed or NULL | 
| 3377 |  |  */ | 
| 3378 |  |  | 
| 3379 |  | xmlChar * | 
| 3380 | 27.1k | xmlParseNmtoken(xmlParserCtxt *ctxt) { | 
| 3381 | 27.1k |     xmlChar buf[XML_MAX_NAMELEN + 5]; | 
| 3382 | 27.1k |     xmlChar *ret; | 
| 3383 | 27.1k |     int len = 0, l; | 
| 3384 | 27.1k |     int c; | 
| 3385 | 27.1k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3386 | 27.1k |                     XML_MAX_TEXT_LENGTH : | 
| 3387 | 27.1k |                     XML_MAX_NAME_LENGTH; | 
| 3388 | 27.1k |     int old10 = (ctxt->options & XML_PARSE_OLD10) ? 1 : 0; | 
| 3389 |  |  | 
| 3390 | 27.1k |     c = xmlCurrentChar(ctxt, &l); | 
| 3391 |  |  | 
| 3392 | 122k |     while (xmlIsNameChar(c, old10)) { | 
| 3393 | 96.7k |   COPY_BUF(buf, len, c); | 
| 3394 | 96.7k |   NEXTL(l); | 
| 3395 | 96.7k |   c = xmlCurrentChar(ctxt, &l); | 
| 3396 | 96.7k |   if (len >= XML_MAX_NAMELEN) { | 
| 3397 |  |       /* | 
| 3398 |  |        * Okay someone managed to make a huge token, so he's ready to pay | 
| 3399 |  |        * for the processing speed. | 
| 3400 |  |        */ | 
| 3401 | 997 |       xmlChar *buffer; | 
| 3402 | 997 |       int max = len * 2; | 
| 3403 |  |  | 
| 3404 | 997 |       buffer = xmlMalloc(max); | 
| 3405 | 997 |       if (buffer == NULL) { | 
| 3406 | 0 |           xmlErrMemory(ctxt); | 
| 3407 | 0 |     return(NULL); | 
| 3408 | 0 |       } | 
| 3409 | 997 |       memcpy(buffer, buf, len); | 
| 3410 | 4.15M |       while (xmlIsNameChar(c, old10)) { | 
| 3411 | 4.15M |     if (len + 10 > max) { | 
| 3412 | 3.61k |         xmlChar *tmp; | 
| 3413 | 3.61k |                     int newSize; | 
| 3414 |  |  | 
| 3415 | 3.61k |                     newSize = xmlGrowCapacity(max, 1, 1, maxLength); | 
| 3416 | 3.61k |                     if (newSize < 0) { | 
| 3417 | 0 |                         xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "NmToken"); | 
| 3418 | 0 |                         xmlFree(buffer); | 
| 3419 | 0 |                         return(NULL); | 
| 3420 | 0 |                     } | 
| 3421 | 3.61k |         tmp = xmlRealloc(buffer, newSize); | 
| 3422 | 3.61k |         if (tmp == NULL) { | 
| 3423 | 0 |       xmlErrMemory(ctxt); | 
| 3424 | 0 |       xmlFree(buffer); | 
| 3425 | 0 |       return(NULL); | 
| 3426 | 0 |         } | 
| 3427 | 3.61k |         buffer = tmp; | 
| 3428 | 3.61k |                     max = newSize; | 
| 3429 | 3.61k |     } | 
| 3430 | 4.15M |     COPY_BUF(buffer, len, c); | 
| 3431 | 4.15M |     NEXTL(l); | 
| 3432 | 4.15M |     c = xmlCurrentChar(ctxt, &l); | 
| 3433 | 4.15M |       } | 
| 3434 | 997 |       buffer[len] = 0; | 
| 3435 | 997 |       return(buffer); | 
| 3436 | 997 |   } | 
| 3437 | 96.7k |     } | 
| 3438 | 26.1k |     if (len == 0) | 
| 3439 | 10.4k |         return(NULL); | 
| 3440 | 15.6k |     if (len > maxLength) { | 
| 3441 | 0 |         xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "NmToken"); | 
| 3442 | 0 |         return(NULL); | 
| 3443 | 0 |     } | 
| 3444 | 15.6k |     ret = xmlStrndup(buf, len); | 
| 3445 | 15.6k |     if (ret == NULL) | 
| 3446 | 0 |         xmlErrMemory(ctxt); | 
| 3447 | 15.6k |     return(ret); | 
| 3448 | 15.6k | } | 
| 3449 |  |  | 
| 3450 |  | /** | 
| 3451 |  |  * Validate an entity value and expand parameter entities. | 
| 3452 |  |  * | 
| 3453 |  |  * @param ctxt  parser context | 
| 3454 |  |  * @param buf  string buffer | 
| 3455 |  |  * @param str  entity value | 
| 3456 |  |  * @param length  size of entity value | 
| 3457 |  |  * @param depth  nesting depth | 
| 3458 |  |  */ | 
| 3459 |  | static void | 
| 3460 |  | xmlExpandPEsInEntityValue(xmlParserCtxtPtr ctxt, xmlSBuf *buf, | 
| 3461 | 19.9k |                           const xmlChar *str, int length, int depth) { | 
| 3462 | 19.9k |     int maxDepth = (ctxt->options & XML_PARSE_HUGE) ? 40 : 20; | 
| 3463 | 19.9k |     const xmlChar *end, *chunk; | 
| 3464 | 19.9k |     int c, l; | 
| 3465 |  |  | 
| 3466 | 19.9k |     if (str == NULL) | 
| 3467 | 0 |         return; | 
| 3468 |  |  | 
| 3469 | 19.9k |     depth += 1; | 
| 3470 | 19.9k |     if (depth > maxDepth) { | 
| 3471 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 3472 | 0 |                        "Maximum entity nesting depth exceeded"); | 
| 3473 | 0 |   return; | 
| 3474 | 0 |     } | 
| 3475 |  |  | 
| 3476 | 19.9k |     end = str + length; | 
| 3477 | 19.9k |     chunk = str; | 
| 3478 |  |  | 
| 3479 | 87.0M |     while ((str < end) && (!PARSER_STOPPED(ctxt))) { | 
| 3480 | 87.0M |         c = *str; | 
| 3481 |  |  | 
| 3482 | 87.0M |         if (c >= 0x80) { | 
| 3483 | 72.7M |             l = xmlUTF8MultibyteLen(ctxt, str, | 
| 3484 | 72.7M |                     "invalid character in entity value\n"); | 
| 3485 | 72.7M |             if (l == 0) { | 
| 3486 | 22.6M |                 if (chunk < str) | 
| 3487 | 113k |                     xmlSBufAddString(buf, chunk, str - chunk); | 
| 3488 | 22.6M |                 xmlSBufAddReplChar(buf); | 
| 3489 | 22.6M |                 str += 1; | 
| 3490 | 22.6M |                 chunk = str; | 
| 3491 | 50.1M |             } else { | 
| 3492 | 50.1M |                 str += l; | 
| 3493 | 50.1M |             } | 
| 3494 | 72.7M |         } else if (c == '&') { | 
| 3495 | 48.2k |             if (str[1] == '#') { | 
| 3496 | 19.1k |                 if (chunk < str) | 
| 3497 | 6.73k |                     xmlSBufAddString(buf, chunk, str - chunk); | 
| 3498 |  |  | 
| 3499 | 19.1k |                 c = xmlParseStringCharRef(ctxt, &str); | 
| 3500 | 19.1k |                 if (c == 0) | 
| 3501 | 3.42k |                     return; | 
| 3502 |  |  | 
| 3503 | 15.7k |                 xmlSBufAddChar(buf, c); | 
| 3504 |  |  | 
| 3505 | 15.7k |                 chunk = str; | 
| 3506 | 29.0k |             } else { | 
| 3507 | 29.0k |                 xmlChar *name; | 
| 3508 |  |  | 
| 3509 |  |                 /* | 
| 3510 |  |                  * General entity references are checked for | 
| 3511 |  |                  * syntactic validity. | 
| 3512 |  |                  */ | 
| 3513 | 29.0k |                 str++; | 
| 3514 | 29.0k |                 name = xmlParseStringName(ctxt, &str); | 
| 3515 |  |  | 
| 3516 | 29.0k |                 if ((name == NULL) || (*str++ != ';')) { | 
| 3517 | 1.37k |                     xmlFatalErrMsg(ctxt, XML_ERR_ENTITY_CHAR_ERROR, | 
| 3518 | 1.37k |                             "EntityValue: '&' forbidden except for entities " | 
| 3519 | 1.37k |                             "references\n"); | 
| 3520 | 1.37k |                     xmlFree(name); | 
| 3521 | 1.37k |                     return; | 
| 3522 | 1.37k |                 } | 
| 3523 |  |  | 
| 3524 | 27.6k |                 xmlFree(name); | 
| 3525 | 27.6k |             } | 
| 3526 | 14.2M |         } else if (c == '%') { | 
| 3527 | 2.16k |             xmlEntityPtr ent; | 
| 3528 |  |  | 
| 3529 | 2.16k |             if (chunk < str) | 
| 3530 | 1.25k |                 xmlSBufAddString(buf, chunk, str - chunk); | 
| 3531 |  |  | 
| 3532 | 2.16k |             ent = xmlParseStringPEReference(ctxt, &str); | 
| 3533 | 2.16k |             if (ent == NULL) | 
| 3534 | 1.67k |                 return; | 
| 3535 |  |  | 
| 3536 | 490 |             if (!PARSER_EXTERNAL(ctxt)) { | 
| 3537 | 490 |                 xmlFatalErr(ctxt, XML_ERR_ENTITY_PE_INTERNAL, NULL); | 
| 3538 | 490 |                 return; | 
| 3539 | 490 |             } | 
| 3540 |  |  | 
| 3541 | 0 |             if (ent->content == NULL) { | 
| 3542 |  |                 /* | 
| 3543 |  |                  * Note: external parsed entities will not be loaded, | 
| 3544 |  |                  * it is not required for a non-validating parser to | 
| 3545 |  |                  * complete external PEReferences coming from the | 
| 3546 |  |                  * internal subset | 
| 3547 |  |                  */ | 
| 3548 | 0 |                 if (((ctxt->options & XML_PARSE_NO_XXE) == 0) && | 
| 3549 | 0 |                     ((ctxt->replaceEntities) || | 
| 3550 | 0 |                      (ctxt->validate))) { | 
| 3551 | 0 |                     xmlLoadEntityContent(ctxt, ent); | 
| 3552 | 0 |                 } else { | 
| 3553 | 0 |                     xmlWarningMsg(ctxt, XML_ERR_ENTITY_PROCESSING, | 
| 3554 | 0 |                                   "not validating will not read content for " | 
| 3555 | 0 |                                   "PE entity %s\n", ent->name, NULL); | 
| 3556 | 0 |                 } | 
| 3557 | 0 |             } | 
| 3558 |  |  | 
| 3559 |  |             /* | 
| 3560 |  |              * TODO: Skip if ent->content is still NULL. | 
| 3561 |  |              */ | 
| 3562 |  | 
 | 
| 3563 | 0 |             if (xmlParserEntityCheck(ctxt, ent->length)) | 
| 3564 | 0 |                 return; | 
| 3565 |  |  | 
| 3566 | 0 |             if (ent->flags & XML_ENT_EXPANDING) { | 
| 3567 | 0 |                 xmlFatalErr(ctxt, XML_ERR_ENTITY_LOOP, NULL); | 
| 3568 | 0 |                 return; | 
| 3569 | 0 |             } | 
| 3570 |  |  | 
| 3571 | 0 |             ent->flags |= XML_ENT_EXPANDING; | 
| 3572 | 0 |             xmlExpandPEsInEntityValue(ctxt, buf, ent->content, ent->length, | 
| 3573 | 0 |                                       depth); | 
| 3574 | 0 |             ent->flags &= ~XML_ENT_EXPANDING; | 
| 3575 |  | 
 | 
| 3576 | 0 |             chunk = str; | 
| 3577 | 14.2M |         } else { | 
| 3578 |  |             /* Normal ASCII char */ | 
| 3579 | 14.2M |             if (!IS_BYTE_CHAR(c)) { | 
| 3580 | 4.05M |                 xmlFatalErrMsg(ctxt, XML_ERR_INVALID_CHAR, | 
| 3581 | 4.05M |                         "invalid character in entity value\n"); | 
| 3582 | 4.05M |                 if (chunk < str) | 
| 3583 | 52.6k |                     xmlSBufAddString(buf, chunk, str - chunk); | 
| 3584 | 4.05M |                 xmlSBufAddReplChar(buf); | 
| 3585 | 4.05M |                 str += 1; | 
| 3586 | 4.05M |                 chunk = str; | 
| 3587 | 10.1M |             } else { | 
| 3588 | 10.1M |                 str += 1; | 
| 3589 | 10.1M |             } | 
| 3590 | 14.2M |         } | 
| 3591 | 87.0M |     } | 
| 3592 |  |  | 
| 3593 | 12.9k |     if (chunk < str) | 
| 3594 | 10.7k |         xmlSBufAddString(buf, chunk, str - chunk); | 
| 3595 | 12.9k | } | 
| 3596 |  |  | 
| 3597 |  | /** | 
| 3598 |  |  * Parse a value for ENTITY declarations | 
| 3599 |  |  * | 
| 3600 |  |  * @deprecated Internal function, don't use. | 
| 3601 |  |  * | 
| 3602 |  |  *     [9] EntityValue ::= '"' ([^%&"] | PEReference | Reference)* '"' | | 
| 3603 |  |  *                         "'" ([^%&'] | PEReference | Reference)* "'" | 
| 3604 |  |  * | 
| 3605 |  |  * @param ctxt  an XML parser context | 
| 3606 |  |  * @param orig  if non-NULL store a copy of the original entity value | 
| 3607 |  |  * @returns the EntityValue parsed with reference substituted or NULL | 
| 3608 |  |  */ | 
| 3609 |  | xmlChar * | 
| 3610 | 20.3k | xmlParseEntityValue(xmlParserCtxt *ctxt, xmlChar **orig) { | 
| 3611 | 20.3k |     unsigned maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3612 | 20.3k |                          XML_MAX_HUGE_LENGTH : | 
| 3613 | 20.3k |                          XML_MAX_TEXT_LENGTH; | 
| 3614 | 20.3k |     xmlSBuf buf; | 
| 3615 | 20.3k |     const xmlChar *start; | 
| 3616 | 20.3k |     int quote, length; | 
| 3617 |  |  | 
| 3618 | 20.3k |     xmlSBufInit(&buf, maxLength); | 
| 3619 |  |  | 
| 3620 | 20.3k |     GROW; | 
| 3621 |  |  | 
| 3622 | 20.3k |     quote = CUR; | 
| 3623 | 20.3k |     if ((quote != '"') && (quote != '\'')) { | 
| 3624 | 0 |   xmlFatalErr(ctxt, XML_ERR_ATTRIBUTE_NOT_STARTED, NULL); | 
| 3625 | 0 |   return(NULL); | 
| 3626 | 0 |     } | 
| 3627 | 20.3k |     CUR_PTR++; | 
| 3628 |  |  | 
| 3629 | 20.3k |     length = 0; | 
| 3630 |  |  | 
| 3631 |  |     /* | 
| 3632 |  |      * Copy raw content of the entity into a buffer | 
| 3633 |  |      */ | 
| 3634 | 142M |     while (1) { | 
| 3635 | 142M |         int c; | 
| 3636 |  |  | 
| 3637 | 142M |         if (PARSER_STOPPED(ctxt)) | 
| 3638 | 1 |             goto error; | 
| 3639 |  |  | 
| 3640 | 142M |         if (CUR_PTR >= ctxt->input->end) { | 
| 3641 | 439 |             xmlFatalErrMsg(ctxt, XML_ERR_ENTITY_NOT_FINISHED, NULL); | 
| 3642 | 439 |             goto error; | 
| 3643 | 439 |         } | 
| 3644 |  |  | 
| 3645 | 142M |         c = CUR; | 
| 3646 |  |  | 
| 3647 | 142M |         if (c == 0) { | 
| 3648 | 7 |             xmlFatalErrMsg(ctxt, XML_ERR_INVALID_CHAR, | 
| 3649 | 7 |                     "invalid character in entity value\n"); | 
| 3650 | 7 |             goto error; | 
| 3651 | 7 |         } | 
| 3652 | 142M |         if (c == quote) | 
| 3653 | 19.9k |             break; | 
| 3654 | 142M |         NEXTL(1); | 
| 3655 | 142M |         length += 1; | 
| 3656 |  |  | 
| 3657 |  |         /* | 
| 3658 |  |          * TODO: Check growth threshold | 
| 3659 |  |          */ | 
| 3660 | 142M |         if (ctxt->input->end - CUR_PTR < 10) | 
| 3661 | 63.7k |             GROW; | 
| 3662 | 142M |     } | 
| 3663 |  |  | 
| 3664 | 19.9k |     start = CUR_PTR - length; | 
| 3665 |  |  | 
| 3666 | 19.9k |     if (orig != NULL) { | 
| 3667 | 19.9k |         *orig = xmlStrndup(start, length); | 
| 3668 | 19.9k |         if (*orig == NULL) | 
| 3669 | 0 |             xmlErrMemory(ctxt); | 
| 3670 | 19.9k |     } | 
| 3671 |  |  | 
| 3672 | 19.9k |     xmlExpandPEsInEntityValue(ctxt, &buf, start, length, ctxt->inputNr); | 
| 3673 |  |  | 
| 3674 | 19.9k |     NEXTL(1); | 
| 3675 |  |  | 
| 3676 | 19.9k |     return(xmlSBufFinish(&buf, NULL, ctxt, "entity length too long")); | 
| 3677 |  |  | 
| 3678 | 447 | error: | 
| 3679 | 447 |     xmlSBufCleanup(&buf, ctxt, "entity length too long"); | 
| 3680 | 447 |     return(NULL); | 
| 3681 | 20.3k | } | 
| 3682 |  |  | 
| 3683 |  | /** | 
| 3684 |  |  * Check an entity reference in an attribute value for validity | 
| 3685 |  |  * without expanding it. | 
| 3686 |  |  * | 
| 3687 |  |  * @param ctxt  parser context | 
| 3688 |  |  * @param pent  entity | 
| 3689 |  |  * @param depth  nesting depth | 
| 3690 |  |  */ | 
| 3691 |  | static void | 
| 3692 | 0 | xmlCheckEntityInAttValue(xmlParserCtxtPtr ctxt, xmlEntityPtr pent, int depth) { | 
| 3693 | 0 |     int maxDepth = (ctxt->options & XML_PARSE_HUGE) ? 40 : 20; | 
| 3694 | 0 |     const xmlChar *str; | 
| 3695 | 0 |     unsigned long expandedSize = pent->length; | 
| 3696 | 0 |     int c, flags; | 
| 3697 |  | 
 | 
| 3698 | 0 |     depth += 1; | 
| 3699 | 0 |     if (depth > maxDepth) { | 
| 3700 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 3701 | 0 |                        "Maximum entity nesting depth exceeded"); | 
| 3702 | 0 |   return; | 
| 3703 | 0 |     } | 
| 3704 |  |  | 
| 3705 | 0 |     if (pent->flags & XML_ENT_EXPANDING) { | 
| 3706 | 0 |         xmlFatalErr(ctxt, XML_ERR_ENTITY_LOOP, NULL); | 
| 3707 | 0 |         return; | 
| 3708 | 0 |     } | 
| 3709 |  |  | 
| 3710 |  |     /* | 
| 3711 |  |      * If we're parsing a default attribute value in DTD content, | 
| 3712 |  |      * the entity might reference other entities which weren't | 
| 3713 |  |      * defined yet, so the check isn't reliable. | 
| 3714 |  |      */ | 
| 3715 | 0 |     if (ctxt->inSubset == 0) | 
| 3716 | 0 |         flags = XML_ENT_CHECKED | XML_ENT_VALIDATED; | 
| 3717 | 0 |     else | 
| 3718 | 0 |         flags = XML_ENT_VALIDATED; | 
| 3719 |  | 
 | 
| 3720 | 0 |     str = pent->content; | 
| 3721 | 0 |     if (str == NULL) | 
| 3722 | 0 |         goto done; | 
| 3723 |  |  | 
| 3724 |  |     /* | 
| 3725 |  |      * Note that entity values are already validated. We only check | 
| 3726 |  |      * for illegal less-than signs and compute the expanded size | 
| 3727 |  |      * of the entity. No special handling for multi-byte characters | 
| 3728 |  |      * is needed. | 
| 3729 |  |      */ | 
| 3730 | 0 |     while (!PARSER_STOPPED(ctxt)) { | 
| 3731 | 0 |         c = *str; | 
| 3732 |  | 
 | 
| 3733 | 0 |   if (c != '&') { | 
| 3734 | 0 |             if (c == 0) | 
| 3735 | 0 |                 break; | 
| 3736 |  |  | 
| 3737 | 0 |             if (c == '<') | 
| 3738 | 0 |                 xmlFatalErrMsgStr(ctxt, XML_ERR_LT_IN_ATTRIBUTE, | 
| 3739 | 0 |                         "'<' in entity '%s' is not allowed in attributes " | 
| 3740 | 0 |                         "values\n", pent->name); | 
| 3741 |  | 
 | 
| 3742 | 0 |             str += 1; | 
| 3743 | 0 |         } else if (str[1] == '#') { | 
| 3744 | 0 |             int val; | 
| 3745 |  | 
 | 
| 3746 | 0 |       val = xmlParseStringCharRef(ctxt, &str); | 
| 3747 | 0 |       if (val == 0) { | 
| 3748 | 0 |                 pent->content[0] = 0; | 
| 3749 | 0 |                 break; | 
| 3750 | 0 |             } | 
| 3751 | 0 |   } else { | 
| 3752 | 0 |             xmlChar *name; | 
| 3753 | 0 |             xmlEntityPtr ent; | 
| 3754 |  | 
 | 
| 3755 | 0 |       name = xmlParseStringEntityRef(ctxt, &str); | 
| 3756 | 0 |       if (name == NULL) { | 
| 3757 | 0 |                 pent->content[0] = 0; | 
| 3758 | 0 |                 break; | 
| 3759 | 0 |             } | 
| 3760 |  |  | 
| 3761 | 0 |             ent = xmlLookupGeneralEntity(ctxt, name, /* inAttr */ 1); | 
| 3762 | 0 |             xmlFree(name); | 
| 3763 |  | 
 | 
| 3764 | 0 |             if ((ent != NULL) && | 
| 3765 | 0 |                 (ent->etype != XML_INTERNAL_PREDEFINED_ENTITY)) { | 
| 3766 | 0 |                 if ((ent->flags & flags) != flags) { | 
| 3767 | 0 |                     pent->flags |= XML_ENT_EXPANDING; | 
| 3768 | 0 |                     xmlCheckEntityInAttValue(ctxt, ent, depth); | 
| 3769 | 0 |                     pent->flags &= ~XML_ENT_EXPANDING; | 
| 3770 | 0 |                 } | 
| 3771 |  | 
 | 
| 3772 | 0 |                 xmlSaturatedAdd(&expandedSize, ent->expandedSize); | 
| 3773 | 0 |                 xmlSaturatedAdd(&expandedSize, XML_ENT_FIXED_COST); | 
| 3774 | 0 |             } | 
| 3775 | 0 |         } | 
| 3776 | 0 |     } | 
| 3777 |  | 
 | 
| 3778 | 0 | done: | 
| 3779 | 0 |     if (ctxt->inSubset == 0) | 
| 3780 | 0 |         pent->expandedSize = expandedSize; | 
| 3781 |  | 
 | 
| 3782 | 0 |     pent->flags |= flags; | 
| 3783 | 0 | } | 
| 3784 |  |  | 
| 3785 |  | /** | 
| 3786 |  |  * Expand general entity references in an entity or attribute value. | 
| 3787 |  |  * Perform attribute value normalization. | 
| 3788 |  |  * | 
| 3789 |  |  * @param ctxt  parser context | 
| 3790 |  |  * @param buf  string buffer | 
| 3791 |  |  * @param str  entity or attribute value | 
| 3792 |  |  * @param pent  entity for entity value, NULL for attribute values | 
| 3793 |  |  * @param normalize  whether to collapse whitespace | 
| 3794 |  |  * @param inSpace  whitespace state | 
| 3795 |  |  * @param depth  nesting depth | 
| 3796 |  |  * @param check  whether to check for amplification | 
| 3797 |  |  * @returns  whether there was a normalization change | 
| 3798 |  |  */ | 
| 3799 |  | static int | 
| 3800 |  | xmlExpandEntityInAttValue(xmlParserCtxtPtr ctxt, xmlSBuf *buf, | 
| 3801 |  |                           const xmlChar *str, xmlEntityPtr pent, int normalize, | 
| 3802 | 439k |                           int *inSpace, int depth, int check) { | 
| 3803 | 439k |     int maxDepth = (ctxt->options & XML_PARSE_HUGE) ? 40 : 20; | 
| 3804 | 439k |     int c, chunkSize; | 
| 3805 | 439k |     int normChange = 0; | 
| 3806 |  |  | 
| 3807 | 439k |     if (str == NULL) | 
| 3808 | 0 |         return(0); | 
| 3809 |  |  | 
| 3810 | 439k |     depth += 1; | 
| 3811 | 439k |     if (depth > maxDepth) { | 
| 3812 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 3813 | 0 |                        "Maximum entity nesting depth exceeded"); | 
| 3814 | 0 |   return(0); | 
| 3815 | 0 |     } | 
| 3816 |  |  | 
| 3817 | 439k |     if (pent != NULL) { | 
| 3818 | 439k |         if (pent->flags & XML_ENT_EXPANDING) { | 
| 3819 | 17 |             xmlFatalErr(ctxt, XML_ERR_ENTITY_LOOP, NULL); | 
| 3820 | 17 |             return(0); | 
| 3821 | 17 |         } | 
| 3822 |  |  | 
| 3823 | 439k |         if (check) { | 
| 3824 | 439k |             if (xmlParserEntityCheck(ctxt, pent->length)) | 
| 3825 | 68 |                 return(0); | 
| 3826 | 439k |         } | 
| 3827 | 439k |     } | 
| 3828 |  |  | 
| 3829 | 439k |     chunkSize = 0; | 
| 3830 |  |  | 
| 3831 |  |     /* | 
| 3832 |  |      * Note that entity values are already validated. No special | 
| 3833 |  |      * handling for multi-byte characters is needed. | 
| 3834 |  |      */ | 
| 3835 | 472M |     while (!PARSER_STOPPED(ctxt)) { | 
| 3836 | 472M |         c = *str; | 
| 3837 |  |  | 
| 3838 | 472M |   if (c != '&') { | 
| 3839 | 472M |             if (c == 0) | 
| 3840 | 368k |                 break; | 
| 3841 |  |  | 
| 3842 |  |             /* | 
| 3843 |  |              * If this function is called without an entity, it is used to | 
| 3844 |  |              * expand entities in an attribute content where less-than was | 
| 3845 |  |              * already unscaped and is allowed. | 
| 3846 |  |              */ | 
| 3847 | 471M |             if ((pent != NULL) && (c == '<')) { | 
| 3848 | 71.5k |                 xmlFatalErrMsgStr(ctxt, XML_ERR_LT_IN_ATTRIBUTE, | 
| 3849 | 71.5k |                         "'<' in entity '%s' is not allowed in attributes " | 
| 3850 | 71.5k |                         "values\n", pent->name); | 
| 3851 | 71.5k |                 break; | 
| 3852 | 71.5k |             } | 
| 3853 |  |  | 
| 3854 | 471M |             if (c <= 0x20) { | 
| 3855 | 1.42M |                 if ((normalize) && (*inSpace)) { | 
| 3856 |  |                     /* Skip char */ | 
| 3857 | 101k |                     if (chunkSize > 0) { | 
| 3858 | 45.8k |                         xmlSBufAddString(buf, str - chunkSize, chunkSize); | 
| 3859 | 45.8k |                         chunkSize = 0; | 
| 3860 | 45.8k |                     } | 
| 3861 | 101k |                     normChange = 1; | 
| 3862 | 1.32M |                 } else if (c < 0x20) { | 
| 3863 | 895k |                     if (chunkSize > 0) { | 
| 3864 | 395k |                         xmlSBufAddString(buf, str - chunkSize, chunkSize); | 
| 3865 | 395k |                         chunkSize = 0; | 
| 3866 | 395k |                     } | 
| 3867 |  |  | 
| 3868 | 895k |                     xmlSBufAddCString(buf, " ", 1); | 
| 3869 | 895k |                 } else { | 
| 3870 | 430k |                     chunkSize += 1; | 
| 3871 | 430k |                 } | 
| 3872 |  |  | 
| 3873 | 1.42M |                 *inSpace = 1; | 
| 3874 | 470M |             } else { | 
| 3875 | 470M |                 chunkSize += 1; | 
| 3876 | 470M |                 *inSpace = 0; | 
| 3877 | 470M |             } | 
| 3878 |  |  | 
| 3879 | 471M |             str += 1; | 
| 3880 | 471M |         } else if (str[1] == '#') { | 
| 3881 | 204k |             int val; | 
| 3882 |  |  | 
| 3883 | 204k |             if (chunkSize > 0) { | 
| 3884 | 194k |                 xmlSBufAddString(buf, str - chunkSize, chunkSize); | 
| 3885 | 194k |                 chunkSize = 0; | 
| 3886 | 194k |             } | 
| 3887 |  |  | 
| 3888 | 204k |       val = xmlParseStringCharRef(ctxt, &str); | 
| 3889 | 204k |       if (val == 0) { | 
| 3890 | 15 |                 if (pent != NULL) | 
| 3891 | 15 |                     pent->content[0] = 0; | 
| 3892 | 15 |                 break; | 
| 3893 | 15 |             } | 
| 3894 |  |  | 
| 3895 | 204k |             if (val == ' ') { | 
| 3896 | 101k |                 if ((normalize) && (*inSpace)) | 
| 3897 | 309 |                     normChange = 1; | 
| 3898 | 101k |                 else | 
| 3899 | 101k |                     xmlSBufAddCString(buf, " ", 1); | 
| 3900 | 101k |                 *inSpace = 1; | 
| 3901 | 102k |             } else { | 
| 3902 | 102k |                 xmlSBufAddChar(buf, val); | 
| 3903 | 102k |                 *inSpace = 0; | 
| 3904 | 102k |             } | 
| 3905 | 410k |   } else { | 
| 3906 | 410k |             xmlChar *name; | 
| 3907 | 410k |             xmlEntityPtr ent; | 
| 3908 |  |  | 
| 3909 | 410k |             if (chunkSize > 0) { | 
| 3910 | 377k |                 xmlSBufAddString(buf, str - chunkSize, chunkSize); | 
| 3911 | 377k |                 chunkSize = 0; | 
| 3912 | 377k |             } | 
| 3913 |  |  | 
| 3914 | 410k |       name = xmlParseStringEntityRef(ctxt, &str); | 
| 3915 | 410k |             if (name == NULL) { | 
| 3916 | 17 |                 if (pent != NULL) | 
| 3917 | 17 |                     pent->content[0] = 0; | 
| 3918 | 17 |                 break; | 
| 3919 | 17 |             } | 
| 3920 |  |  | 
| 3921 | 410k |             ent = xmlLookupGeneralEntity(ctxt, name, /* inAttr */ 1); | 
| 3922 | 410k |             xmlFree(name); | 
| 3923 |  |  | 
| 3924 | 410k |       if ((ent != NULL) && | 
| 3925 | 393k |     (ent->etype == XML_INTERNAL_PREDEFINED_ENTITY)) { | 
| 3926 | 20.7k |     if (ent->content == NULL) { | 
| 3927 | 0 |         xmlFatalErrMsg(ctxt, XML_ERR_INTERNAL_ERROR, | 
| 3928 | 0 |           "predefined entity has no content\n"); | 
| 3929 | 0 |                     break; | 
| 3930 | 0 |                 } | 
| 3931 |  |  | 
| 3932 | 20.7k |                 xmlSBufAddString(buf, ent->content, ent->length); | 
| 3933 |  |  | 
| 3934 | 20.7k |                 *inSpace = 0; | 
| 3935 | 389k |       } else if ((ent != NULL) && (ent->content != NULL)) { | 
| 3936 | 372k |                 if (pent != NULL) | 
| 3937 | 372k |                     pent->flags |= XML_ENT_EXPANDING; | 
| 3938 | 372k |     normChange |= xmlExpandEntityInAttValue(ctxt, buf, | 
| 3939 | 372k |                         ent->content, ent, normalize, inSpace, depth, check); | 
| 3940 | 372k |                 if (pent != NULL) | 
| 3941 | 372k |                     pent->flags &= ~XML_ENT_EXPANDING; | 
| 3942 | 372k |       } | 
| 3943 | 410k |         } | 
| 3944 | 472M |     } | 
| 3945 |  |  | 
| 3946 | 439k |     if (chunkSize > 0) | 
| 3947 | 400k |         xmlSBufAddString(buf, str - chunkSize, chunkSize); | 
| 3948 |  |  | 
| 3949 | 439k |     return(normChange); | 
| 3950 | 439k | } | 
| 3951 |  |  | 
| 3952 |  | /** | 
| 3953 |  |  * Expand general entity references in an entity or attribute value. | 
| 3954 |  |  * Perform attribute value normalization. | 
| 3955 |  |  * | 
| 3956 |  |  * @param ctxt  parser context | 
| 3957 |  |  * @param str  entity or attribute value | 
| 3958 |  |  * @param normalize  whether to collapse whitespace | 
| 3959 |  |  * @returns the expanded attribtue value. | 
| 3960 |  |  */ | 
| 3961 |  | xmlChar * | 
| 3962 |  | xmlExpandEntitiesInAttValue(xmlParserCtxt *ctxt, const xmlChar *str, | 
| 3963 | 0 |                             int normalize) { | 
| 3964 | 0 |     unsigned maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 3965 | 0 |                          XML_MAX_HUGE_LENGTH : | 
| 3966 | 0 |                          XML_MAX_TEXT_LENGTH; | 
| 3967 | 0 |     xmlSBuf buf; | 
| 3968 | 0 |     int inSpace = 1; | 
| 3969 |  | 
 | 
| 3970 | 0 |     xmlSBufInit(&buf, maxLength); | 
| 3971 |  | 
 | 
| 3972 | 0 |     xmlExpandEntityInAttValue(ctxt, &buf, str, NULL, normalize, &inSpace, | 
| 3973 | 0 |                               ctxt->inputNr, /* check */ 0); | 
| 3974 |  | 
 | 
| 3975 | 0 |     if ((normalize) && (inSpace) && (buf.size > 0)) | 
| 3976 | 0 |         buf.size--; | 
| 3977 |  | 
 | 
| 3978 | 0 |     return(xmlSBufFinish(&buf, NULL, ctxt, "AttValue length too long")); | 
| 3979 | 0 | } | 
| 3980 |  |  | 
| 3981 |  | /** | 
| 3982 |  |  * Parse a value for an attribute. | 
| 3983 |  |  * | 
| 3984 |  |  * NOTE: if no normalization is needed, the routine will return pointers | 
| 3985 |  |  * directly from the data buffer. | 
| 3986 |  |  * | 
| 3987 |  |  * 3.3.3 Attribute-Value Normalization: | 
| 3988 |  |  * | 
| 3989 |  |  * Before the value of an attribute is passed to the application or | 
| 3990 |  |  * checked for validity, the XML processor must normalize it as follows: | 
| 3991 |  |  * | 
| 3992 |  |  * - a character reference is processed by appending the referenced | 
| 3993 |  |  *   character to the attribute value | 
| 3994 |  |  * - an entity reference is processed by recursively processing the | 
| 3995 |  |  *   replacement text of the entity | 
| 3996 |  |  * - a whitespace character (\#x20, \#xD, \#xA, \#x9) is processed by | 
| 3997 |  |  *   appending \#x20 to the normalized value, except that only a single | 
| 3998 |  |  *   \#x20 is appended for a "#xD#xA" sequence that is part of an external | 
| 3999 |  |  *   parsed entity or the literal entity value of an internal parsed entity | 
| 4000 |  |  * - other characters are processed by appending them to the normalized value | 
| 4001 |  |  * | 
| 4002 |  |  * If the declared value is not CDATA, then the XML processor must further | 
| 4003 |  |  * process the normalized attribute value by discarding any leading and | 
| 4004 |  |  * trailing space (\#x20) characters, and by replacing sequences of space | 
| 4005 |  |  * (\#x20) characters by a single space (\#x20) character. | 
| 4006 |  |  * All attributes for which no declaration has been read should be treated | 
| 4007 |  |  * by a non-validating parser as if declared CDATA. | 
| 4008 |  |  * | 
| 4009 |  |  * @param ctxt  an XML parser context | 
| 4010 |  |  * @param attlen  attribute len result | 
| 4011 |  |  * @param outFlags  resulting XML_ATTVAL_* flags | 
| 4012 |  |  * @param special  value from attsSpecial | 
| 4013 |  |  * @param isNamespace  whether this is a namespace declaration | 
| 4014 |  |  * @returns the AttValue parsed or NULL. The value has to be freed by the | 
| 4015 |  |  *     caller if it was copied, this can be detected by val[*len] == 0. | 
| 4016 |  |  */ | 
| 4017 |  | static xmlChar * | 
| 4018 |  | xmlParseAttValueInternal(xmlParserCtxtPtr ctxt, int *attlen, int *outFlags, | 
| 4019 | 624k |                          int special, int isNamespace) { | 
| 4020 | 624k |     unsigned maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 4021 | 624k |                          XML_MAX_HUGE_LENGTH : | 
| 4022 | 624k |                          XML_MAX_TEXT_LENGTH; | 
| 4023 | 624k |     xmlSBuf buf; | 
| 4024 | 624k |     xmlChar *ret; | 
| 4025 | 624k |     int c, l, quote, entFlags, chunkSize; | 
| 4026 | 624k |     int inSpace = 1; | 
| 4027 | 624k |     int replaceEntities; | 
| 4028 | 624k |     int normalize = (special & XML_SPECIAL_TYPE_MASK) > XML_ATTRIBUTE_CDATA; | 
| 4029 | 624k |     int attvalFlags = 0; | 
| 4030 |  |  | 
| 4031 |  |     /* Always expand namespace URIs */ | 
| 4032 | 624k |     replaceEntities = (ctxt->replaceEntities) || (isNamespace); | 
| 4033 |  |  | 
| 4034 | 624k |     xmlSBufInit(&buf, maxLength); | 
| 4035 |  |  | 
| 4036 | 624k |     GROW; | 
| 4037 |  |  | 
| 4038 | 624k |     quote = CUR; | 
| 4039 | 624k |     if ((quote != '"') && (quote != '\'')) { | 
| 4040 | 7.04k |   xmlFatalErr(ctxt, XML_ERR_ATTRIBUTE_NOT_STARTED, NULL); | 
| 4041 | 7.04k |   return(NULL); | 
| 4042 | 7.04k |     } | 
| 4043 | 617k |     NEXTL(1); | 
| 4044 |  |  | 
| 4045 | 617k |     if (ctxt->inSubset == 0) | 
| 4046 | 598k |         entFlags = XML_ENT_CHECKED | XML_ENT_VALIDATED; | 
| 4047 | 18.7k |     else | 
| 4048 | 18.7k |         entFlags = XML_ENT_VALIDATED; | 
| 4049 |  |  | 
| 4050 | 617k |     inSpace = 1; | 
| 4051 | 617k |     chunkSize = 0; | 
| 4052 |  |  | 
| 4053 | 34.9M |     while (1) { | 
| 4054 | 34.9M |         if (PARSER_STOPPED(ctxt)) | 
| 4055 | 85 |             goto error; | 
| 4056 |  |  | 
| 4057 | 34.9M |         if (CUR_PTR >= ctxt->input->end) { | 
| 4058 | 3.31k |             xmlFatalErrMsg(ctxt, XML_ERR_ATTRIBUTE_NOT_FINISHED, | 
| 4059 | 3.31k |                            "AttValue: ' expected\n"); | 
| 4060 | 3.31k |             goto error; | 
| 4061 | 3.31k |         } | 
| 4062 |  |  | 
| 4063 |  |         /* | 
| 4064 |  |          * TODO: Check growth threshold | 
| 4065 |  |          */ | 
| 4066 | 34.9M |         if (ctxt->input->end - CUR_PTR < 10) | 
| 4067 | 37.8k |             GROW; | 
| 4068 |  |  | 
| 4069 | 34.9M |         c = CUR; | 
| 4070 |  |  | 
| 4071 | 34.9M |         if (c >= 0x80) { | 
| 4072 | 22.0M |             l = xmlUTF8MultibyteLen(ctxt, CUR_PTR, | 
| 4073 | 22.0M |                     "invalid character in attribute value\n"); | 
| 4074 | 22.0M |             if (l == 0) { | 
| 4075 | 17.3M |                 if (chunkSize > 0) { | 
| 4076 | 40.7k |                     xmlSBufAddString(&buf, CUR_PTR - chunkSize, chunkSize); | 
| 4077 | 40.7k |                     chunkSize = 0; | 
| 4078 | 40.7k |                 } | 
| 4079 | 17.3M |                 xmlSBufAddReplChar(&buf); | 
| 4080 | 17.3M |                 NEXTL(1); | 
| 4081 | 17.3M |             } else { | 
| 4082 | 4.72M |                 chunkSize += l; | 
| 4083 | 4.72M |                 NEXTL(l); | 
| 4084 | 4.72M |             } | 
| 4085 |  |  | 
| 4086 | 22.0M |             inSpace = 0; | 
| 4087 | 22.0M |         } else if (c != '&') { | 
| 4088 | 12.6M |             if (c > 0x20) { | 
| 4089 | 6.84M |                 if (c == quote) | 
| 4090 | 612k |                     break; | 
| 4091 |  |  | 
| 4092 | 6.23M |                 if (c == '<') | 
| 4093 | 184k |                     xmlFatalErr(ctxt, XML_ERR_LT_IN_ATTRIBUTE, NULL); | 
| 4094 |  |  | 
| 4095 | 6.23M |                 chunkSize += 1; | 
| 4096 | 6.23M |                 inSpace = 0; | 
| 4097 | 6.23M |             } else if (!IS_BYTE_CHAR(c)) { | 
| 4098 | 320k |                 xmlFatalErrMsg(ctxt, XML_ERR_INVALID_CHAR, | 
| 4099 | 320k |                         "invalid character in attribute value\n"); | 
| 4100 | 320k |                 if (chunkSize > 0) { | 
| 4101 | 17.2k |                     xmlSBufAddString(&buf, CUR_PTR - chunkSize, chunkSize); | 
| 4102 | 17.2k |                     chunkSize = 0; | 
| 4103 | 17.2k |                 } | 
| 4104 | 320k |                 xmlSBufAddReplChar(&buf); | 
| 4105 | 320k |                 inSpace = 0; | 
| 4106 | 5.51M |             } else { | 
| 4107 |  |                 /* Whitespace */ | 
| 4108 | 5.51M |                 if ((normalize) && (inSpace)) { | 
| 4109 |  |                     /* Skip char */ | 
| 4110 | 68.4k |                     if (chunkSize > 0) { | 
| 4111 | 1.65k |                         xmlSBufAddString(&buf, CUR_PTR - chunkSize, chunkSize); | 
| 4112 | 1.65k |                         chunkSize = 0; | 
| 4113 | 1.65k |                     } | 
| 4114 | 68.4k |                     attvalFlags |= XML_ATTVAL_NORM_CHANGE; | 
| 4115 | 5.45M |                 } else if (c < 0x20) { | 
| 4116 |  |                     /* Convert to space */ | 
| 4117 | 4.94M |                     if (chunkSize > 0) { | 
| 4118 | 34.3k |                         xmlSBufAddString(&buf, CUR_PTR - chunkSize, chunkSize); | 
| 4119 | 34.3k |                         chunkSize = 0; | 
| 4120 | 34.3k |                     } | 
| 4121 |  |  | 
| 4122 | 4.94M |                     xmlSBufAddCString(&buf, " ", 1); | 
| 4123 | 4.94M |                 } else { | 
| 4124 | 505k |                     chunkSize += 1; | 
| 4125 | 505k |                 } | 
| 4126 |  |  | 
| 4127 | 5.51M |                 inSpace = 1; | 
| 4128 |  |  | 
| 4129 | 5.51M |                 if ((c == 0xD) && (NXT(1) == 0xA)) | 
| 4130 | 2.66k |                     CUR_PTR++; | 
| 4131 | 5.51M |             } | 
| 4132 |  |  | 
| 4133 | 12.0M |             NEXTL(1); | 
| 4134 | 12.0M |         } else if (NXT(1) == '#') { | 
| 4135 | 15.0k |             int val; | 
| 4136 |  |  | 
| 4137 | 15.0k |             if (chunkSize > 0) { | 
| 4138 | 4.94k |                 xmlSBufAddString(&buf, CUR_PTR - chunkSize, chunkSize); | 
| 4139 | 4.94k |                 chunkSize = 0; | 
| 4140 | 4.94k |             } | 
| 4141 |  |  | 
| 4142 | 15.0k |             val = xmlParseCharRef(ctxt); | 
| 4143 | 15.0k |             if (val == 0) | 
| 4144 | 975 |                 goto error; | 
| 4145 |  |  | 
| 4146 | 14.0k |             if ((val == '&') && (!replaceEntities)) { | 
| 4147 |  |                 /* | 
| 4148 |  |                  * The reparsing will be done in xmlNodeParseContent() | 
| 4149 |  |                  * called from SAX2.c | 
| 4150 |  |                  */ | 
| 4151 | 0 |                 xmlSBufAddCString(&buf, "&", 5); | 
| 4152 | 0 |                 inSpace = 0; | 
| 4153 | 14.0k |             } else if (val == ' ') { | 
| 4154 | 2.93k |                 if ((normalize) && (inSpace)) | 
| 4155 | 267 |                     attvalFlags |= XML_ATTVAL_NORM_CHANGE; | 
| 4156 | 2.66k |                 else | 
| 4157 | 2.66k |                     xmlSBufAddCString(&buf, " ", 1); | 
| 4158 | 2.93k |                 inSpace = 1; | 
| 4159 | 11.1k |             } else { | 
| 4160 | 11.1k |                 xmlSBufAddChar(&buf, val); | 
| 4161 | 11.1k |                 inSpace = 0; | 
| 4162 | 11.1k |             } | 
| 4163 | 114k |         } else { | 
| 4164 | 114k |             const xmlChar *name; | 
| 4165 | 114k |             xmlEntityPtr ent; | 
| 4166 |  |  | 
| 4167 | 114k |             if (chunkSize > 0) { | 
| 4168 | 39.5k |                 xmlSBufAddString(&buf, CUR_PTR - chunkSize, chunkSize); | 
| 4169 | 39.5k |                 chunkSize = 0; | 
| 4170 | 39.5k |             } | 
| 4171 |  |  | 
| 4172 | 114k |             name = xmlParseEntityRefInternal(ctxt); | 
| 4173 | 114k |             if (name == NULL) { | 
| 4174 |  |                 /* | 
| 4175 |  |                  * Probably a literal '&' which wasn't escaped. | 
| 4176 |  |                  * TODO: Handle gracefully in recovery mode. | 
| 4177 |  |                  */ | 
| 4178 | 22.7k |                 continue; | 
| 4179 | 22.7k |             } | 
| 4180 |  |  | 
| 4181 | 91.6k |             ent = xmlLookupGeneralEntity(ctxt, name, /* isAttr */ 1); | 
| 4182 | 91.6k |             if (ent == NULL) | 
| 4183 | 21.3k |                 continue; | 
| 4184 |  |  | 
| 4185 | 70.3k |             if (ent->etype == XML_INTERNAL_PREDEFINED_ENTITY) { | 
| 4186 | 3.42k |                 if ((ent->content[0] == '&') && (!replaceEntities)) | 
| 4187 | 0 |                     xmlSBufAddCString(&buf, "&", 5); | 
| 4188 | 3.42k |                 else | 
| 4189 | 3.42k |                     xmlSBufAddString(&buf, ent->content, ent->length); | 
| 4190 | 3.42k |                 inSpace = 0; | 
| 4191 | 66.8k |             } else if (replaceEntities) { | 
| 4192 | 66.8k |                 if (xmlExpandEntityInAttValue(ctxt, &buf, | 
| 4193 | 66.8k |                         ent->content, ent, normalize, &inSpace, ctxt->inputNr, | 
| 4194 | 66.8k |                         /* check */ 1) > 0) | 
| 4195 | 2.91k |                     attvalFlags |= XML_ATTVAL_NORM_CHANGE; | 
| 4196 | 66.8k |             } else { | 
| 4197 | 0 |                 if ((ent->flags & entFlags) != entFlags) | 
| 4198 | 0 |                     xmlCheckEntityInAttValue(ctxt, ent, ctxt->inputNr); | 
| 4199 |  | 
 | 
| 4200 | 0 |                 if (xmlParserEntityCheck(ctxt, ent->expandedSize)) { | 
| 4201 | 0 |                     ent->content[0] = 0; | 
| 4202 | 0 |                     goto error; | 
| 4203 | 0 |                 } | 
| 4204 |  |  | 
| 4205 |  |                 /* | 
| 4206 |  |                  * Just output the reference | 
| 4207 |  |                  */ | 
| 4208 | 0 |                 xmlSBufAddCString(&buf, "&", 1); | 
| 4209 | 0 |                 xmlSBufAddString(&buf, ent->name, xmlStrlen(ent->name)); | 
| 4210 | 0 |                 xmlSBufAddCString(&buf, ";", 1); | 
| 4211 |  | 
 | 
| 4212 | 0 |                 inSpace = 0; | 
| 4213 | 0 |             } | 
| 4214 | 70.3k |   } | 
| 4215 | 34.9M |     } | 
| 4216 |  |  | 
| 4217 | 612k |     if ((buf.mem == NULL) && (outFlags != NULL)) { | 
| 4218 | 540k |         ret = (xmlChar *) CUR_PTR - chunkSize; | 
| 4219 |  |  | 
| 4220 | 540k |         if (attlen != NULL) | 
| 4221 | 540k |             *attlen = chunkSize; | 
| 4222 | 540k |         if ((normalize) && (inSpace) && (chunkSize > 0)) { | 
| 4223 | 228 |             attvalFlags |= XML_ATTVAL_NORM_CHANGE; | 
| 4224 | 228 |             *attlen -= 1; | 
| 4225 | 228 |         } | 
| 4226 |  |  | 
| 4227 |  |         /* Report potential error */ | 
| 4228 | 540k |         xmlSBufCleanup(&buf, ctxt, "AttValue length too long"); | 
| 4229 | 540k |     } else { | 
| 4230 | 71.8k |         if (chunkSize > 0) | 
| 4231 | 35.9k |             xmlSBufAddString(&buf, CUR_PTR - chunkSize, chunkSize); | 
| 4232 |  |  | 
| 4233 | 71.8k |         if ((normalize) && (inSpace) && (buf.size > 0)) { | 
| 4234 | 787 |             attvalFlags |= XML_ATTVAL_NORM_CHANGE; | 
| 4235 | 787 |             buf.size--; | 
| 4236 | 787 |         } | 
| 4237 |  |  | 
| 4238 | 71.8k |         ret = xmlSBufFinish(&buf, attlen, ctxt, "AttValue length too long"); | 
| 4239 | 71.8k |         attvalFlags |= XML_ATTVAL_ALLOC; | 
| 4240 |  |  | 
| 4241 | 71.8k |         if (ret != NULL) { | 
| 4242 | 71.8k |             if (attlen != NULL) | 
| 4243 | 55.3k |                 *attlen = buf.size; | 
| 4244 | 71.8k |         } | 
| 4245 | 71.8k |     } | 
| 4246 |  |  | 
| 4247 | 612k |     if (outFlags != NULL) | 
| 4248 | 596k |         *outFlags = attvalFlags; | 
| 4249 |  |  | 
| 4250 | 612k |     NEXTL(1); | 
| 4251 |  |  | 
| 4252 | 612k |     return(ret); | 
| 4253 |  |  | 
| 4254 | 4.37k | error: | 
| 4255 | 4.37k |     xmlSBufCleanup(&buf, ctxt, "AttValue length too long"); | 
| 4256 | 4.37k |     return(NULL); | 
| 4257 | 617k | } | 
| 4258 |  |  | 
| 4259 |  | /** | 
| 4260 |  |  * Parse a value for an attribute | 
| 4261 |  |  * Note: the parser won't do substitution of entities here, this | 
| 4262 |  |  * will be handled later in #xmlStringGetNodeList | 
| 4263 |  |  * | 
| 4264 |  |  * @deprecated Internal function, don't use. | 
| 4265 |  |  * | 
| 4266 |  |  *     [10] AttValue ::= '"' ([^<&"] | Reference)* '"' | | 
| 4267 |  |  *                       "'" ([^<&'] | Reference)* "'" | 
| 4268 |  |  * | 
| 4269 |  |  * 3.3.3 Attribute-Value Normalization: | 
| 4270 |  |  * | 
| 4271 |  |  * Before the value of an attribute is passed to the application or | 
| 4272 |  |  * checked for validity, the XML processor must normalize it as follows: | 
| 4273 |  |  * | 
| 4274 |  |  * - a character reference is processed by appending the referenced | 
| 4275 |  |  *   character to the attribute value | 
| 4276 |  |  * - an entity reference is processed by recursively processing the | 
| 4277 |  |  *   replacement text of the entity | 
| 4278 |  |  * - a whitespace character (\#x20, \#xD, \#xA, \#x9) is processed by | 
| 4279 |  |  *   appending \#x20 to the normalized value, except that only a single | 
| 4280 |  |  *   \#x20 is appended for a "#xD#xA" sequence that is part of an external | 
| 4281 |  |  *   parsed entity or the literal entity value of an internal parsed entity | 
| 4282 |  |  * - other characters are processed by appending them to the normalized value | 
| 4283 |  |  * | 
| 4284 |  |  * If the declared value is not CDATA, then the XML processor must further | 
| 4285 |  |  * process the normalized attribute value by discarding any leading and | 
| 4286 |  |  * trailing space (\#x20) characters, and by replacing sequences of space | 
| 4287 |  |  * (\#x20) characters by a single space (\#x20) character. | 
| 4288 |  |  * All attributes for which no declaration has been read should be treated | 
| 4289 |  |  * by a non-validating parser as if declared CDATA. | 
| 4290 |  |  * | 
| 4291 |  |  * @param ctxt  an XML parser context | 
| 4292 |  |  * @returns the AttValue parsed or NULL. The value has to be freed by the | 
| 4293 |  |  * caller. | 
| 4294 |  |  */ | 
| 4295 |  | xmlChar * | 
| 4296 | 19.8k | xmlParseAttValue(xmlParserCtxt *ctxt) { | 
| 4297 | 19.8k |     if ((ctxt == NULL) || (ctxt->input == NULL)) return(NULL); | 
| 4298 | 19.8k |     return(xmlParseAttValueInternal(ctxt, NULL, NULL, 0, 0)); | 
| 4299 | 19.8k | } | 
| 4300 |  |  | 
| 4301 |  | /** | 
| 4302 |  |  * Parse an XML Literal | 
| 4303 |  |  * | 
| 4304 |  |  * @deprecated Internal function, don't use. | 
| 4305 |  |  * | 
| 4306 |  |  *     [11] SystemLiteral ::= ('"' [^"]* '"') | ("'" [^']* "'") | 
| 4307 |  |  * | 
| 4308 |  |  * @param ctxt  an XML parser context | 
| 4309 |  |  * @returns the SystemLiteral parsed or NULL | 
| 4310 |  |  */ | 
| 4311 |  |  | 
| 4312 |  | xmlChar * | 
| 4313 | 4.82k | xmlParseSystemLiteral(xmlParserCtxt *ctxt) { | 
| 4314 | 4.82k |     xmlChar *buf = NULL; | 
| 4315 | 4.82k |     int len = 0; | 
| 4316 | 4.82k |     int size = XML_PARSER_BUFFER_SIZE; | 
| 4317 | 4.82k |     int cur, l; | 
| 4318 | 4.82k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 4319 | 4.82k |                     XML_MAX_TEXT_LENGTH : | 
| 4320 | 4.82k |                     XML_MAX_NAME_LENGTH; | 
| 4321 | 4.82k |     xmlChar stop; | 
| 4322 |  |  | 
| 4323 | 4.82k |     if (RAW == '"') { | 
| 4324 | 2.01k |         NEXT; | 
| 4325 | 2.01k |   stop = '"'; | 
| 4326 | 2.80k |     } else if (RAW == '\'') { | 
| 4327 | 1.22k |         NEXT; | 
| 4328 | 1.22k |   stop = '\''; | 
| 4329 | 1.58k |     } else { | 
| 4330 | 1.58k |   xmlFatalErr(ctxt, XML_ERR_LITERAL_NOT_STARTED, NULL); | 
| 4331 | 1.58k |   return(NULL); | 
| 4332 | 1.58k |     } | 
| 4333 |  |  | 
| 4334 | 3.24k |     buf = xmlMalloc(size); | 
| 4335 | 3.24k |     if (buf == NULL) { | 
| 4336 | 0 |         xmlErrMemory(ctxt); | 
| 4337 | 0 |   return(NULL); | 
| 4338 | 0 |     } | 
| 4339 | 3.24k |     cur = xmlCurrentCharRecover(ctxt, &l); | 
| 4340 | 4.89M |     while ((IS_CHAR(cur)) && (cur != stop)) { /* checked */ | 
| 4341 | 4.89M |   if (len + 5 >= size) { | 
| 4342 | 2.95k |       xmlChar *tmp; | 
| 4343 | 2.95k |             int newSize; | 
| 4344 |  |  | 
| 4345 | 2.95k |             newSize = xmlGrowCapacity(size, 1, 1, maxLength); | 
| 4346 | 2.95k |             if (newSize < 0) { | 
| 4347 | 0 |                 xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "SystemLiteral"); | 
| 4348 | 0 |                 xmlFree(buf); | 
| 4349 | 0 |                 return(NULL); | 
| 4350 | 0 |             } | 
| 4351 | 2.95k |       tmp = xmlRealloc(buf, newSize); | 
| 4352 | 2.95k |       if (tmp == NULL) { | 
| 4353 | 0 |           xmlFree(buf); | 
| 4354 | 0 |     xmlErrMemory(ctxt); | 
| 4355 | 0 |     return(NULL); | 
| 4356 | 0 |       } | 
| 4357 | 2.95k |       buf = tmp; | 
| 4358 | 2.95k |             size = newSize; | 
| 4359 | 2.95k |   } | 
| 4360 | 4.89M |   COPY_BUF(buf, len, cur); | 
| 4361 | 4.89M |   NEXTL(l); | 
| 4362 | 4.89M |   cur = xmlCurrentCharRecover(ctxt, &l); | 
| 4363 | 4.89M |     } | 
| 4364 | 3.24k |     buf[len] = 0; | 
| 4365 | 3.24k |     if (!IS_CHAR(cur)) { | 
| 4366 | 1.01k |   xmlFatalErr(ctxt, XML_ERR_LITERAL_NOT_FINISHED, NULL); | 
| 4367 | 2.23k |     } else { | 
| 4368 | 2.23k |   NEXT; | 
| 4369 | 2.23k |     } | 
| 4370 | 3.24k |     return(buf); | 
| 4371 | 3.24k | } | 
| 4372 |  |  | 
| 4373 |  | /** | 
| 4374 |  |  * Parse an XML public literal | 
| 4375 |  |  * | 
| 4376 |  |  * @deprecated Internal function, don't use. | 
| 4377 |  |  * | 
| 4378 |  |  *     [12] PubidLiteral ::= '"' PubidChar* '"' | "'" (PubidChar - "'")* "'" | 
| 4379 |  |  * | 
| 4380 |  |  * @param ctxt  an XML parser context | 
| 4381 |  |  * @returns the PubidLiteral parsed or NULL. | 
| 4382 |  |  */ | 
| 4383 |  |  | 
| 4384 |  | xmlChar * | 
| 4385 | 3.46k | xmlParsePubidLiteral(xmlParserCtxt *ctxt) { | 
| 4386 | 3.46k |     xmlChar *buf = NULL; | 
| 4387 | 3.46k |     int len = 0; | 
| 4388 | 3.46k |     int size = XML_PARSER_BUFFER_SIZE; | 
| 4389 | 3.46k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 4390 | 3.46k |                     XML_MAX_TEXT_LENGTH : | 
| 4391 | 3.46k |                     XML_MAX_NAME_LENGTH; | 
| 4392 | 3.46k |     xmlChar cur; | 
| 4393 | 3.46k |     xmlChar stop; | 
| 4394 |  |  | 
| 4395 | 3.46k |     if (RAW == '"') { | 
| 4396 | 1.70k |         NEXT; | 
| 4397 | 1.70k |   stop = '"'; | 
| 4398 | 1.76k |     } else if (RAW == '\'') { | 
| 4399 | 1.16k |         NEXT; | 
| 4400 | 1.16k |   stop = '\''; | 
| 4401 | 1.16k |     } else { | 
| 4402 | 593 |   xmlFatalErr(ctxt, XML_ERR_LITERAL_NOT_STARTED, NULL); | 
| 4403 | 593 |   return(NULL); | 
| 4404 | 593 |     } | 
| 4405 | 2.87k |     buf = xmlMalloc(size); | 
| 4406 | 2.87k |     if (buf == NULL) { | 
| 4407 | 0 |   xmlErrMemory(ctxt); | 
| 4408 | 0 |   return(NULL); | 
| 4409 | 0 |     } | 
| 4410 | 2.87k |     cur = CUR; | 
| 4411 | 14.3M |     while ((IS_PUBIDCHAR_CH(cur)) && (cur != stop) && | 
| 4412 | 14.3M |            (PARSER_STOPPED(ctxt) == 0)) { /* checked */ | 
| 4413 | 14.3M |   if (len + 1 >= size) { | 
| 4414 | 1.39k |       xmlChar *tmp; | 
| 4415 | 1.39k |             int newSize; | 
| 4416 |  |  | 
| 4417 | 1.39k |       newSize = xmlGrowCapacity(size, 1, 1, maxLength); | 
| 4418 | 1.39k |             if (newSize < 0) { | 
| 4419 | 0 |                 xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "Public ID"); | 
| 4420 | 0 |                 xmlFree(buf); | 
| 4421 | 0 |                 return(NULL); | 
| 4422 | 0 |             } | 
| 4423 | 1.39k |       tmp = xmlRealloc(buf, newSize); | 
| 4424 | 1.39k |       if (tmp == NULL) { | 
| 4425 | 0 |     xmlErrMemory(ctxt); | 
| 4426 | 0 |     xmlFree(buf); | 
| 4427 | 0 |     return(NULL); | 
| 4428 | 0 |       } | 
| 4429 | 1.39k |       buf = tmp; | 
| 4430 | 1.39k |             size = newSize; | 
| 4431 | 1.39k |   } | 
| 4432 | 14.3M |   buf[len++] = cur; | 
| 4433 | 14.3M |   NEXT; | 
| 4434 | 14.3M |   cur = CUR; | 
| 4435 | 14.3M |     } | 
| 4436 | 2.87k |     buf[len] = 0; | 
| 4437 | 2.87k |     if (cur != stop) { | 
| 4438 | 1.66k |   xmlFatalErr(ctxt, XML_ERR_LITERAL_NOT_FINISHED, NULL); | 
| 4439 | 1.66k |     } else { | 
| 4440 | 1.20k |   NEXTL(1); | 
| 4441 | 1.20k |     } | 
| 4442 | 2.87k |     return(buf); | 
| 4443 | 2.87k | } | 
| 4444 |  |  | 
| 4445 |  | static void xmlParseCharDataComplex(xmlParserCtxtPtr ctxt, int partial); | 
| 4446 |  |  | 
| 4447 |  | /* | 
| 4448 |  |  * used for the test in the inner loop of the char data testing | 
| 4449 |  |  */ | 
| 4450 |  | static const unsigned char test_char_data[256] = { | 
| 4451 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4452 |  |     0x00, 0x09, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 0x9, CR/LF separated */ | 
| 4453 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4454 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4455 |  |     0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x00, 0x27, /* & */ | 
| 4456 |  |     0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F, | 
| 4457 |  |     0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, | 
| 4458 |  |     0x38, 0x39, 0x3A, 0x3B, 0x00, 0x3D, 0x3E, 0x3F, /* < */ | 
| 4459 |  |     0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, | 
| 4460 |  |     0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F, | 
| 4461 |  |     0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, | 
| 4462 |  |     0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x00, 0x5E, 0x5F, /* ] */ | 
| 4463 |  |     0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, | 
| 4464 |  |     0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F, | 
| 4465 |  |     0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, | 
| 4466 |  |     0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F, | 
| 4467 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* non-ascii */ | 
| 4468 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4469 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4470 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4471 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4472 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4473 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4474 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4475 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4476 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4477 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4478 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4479 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4480 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4481 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
| 4482 |  |     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 | 
| 4483 |  | }; | 
| 4484 |  |  | 
| 4485 |  | static void | 
| 4486 |  | xmlCharacters(xmlParserCtxtPtr ctxt, const xmlChar *buf, int size, | 
| 4487 | 590k |               int isBlank) { | 
| 4488 | 590k |     int checkBlanks; | 
| 4489 |  |  | 
| 4490 | 590k |     if ((ctxt->sax == NULL) || (ctxt->disableSAX)) | 
| 4491 | 451k |         return; | 
| 4492 |  |  | 
| 4493 | 139k |     checkBlanks = (!ctxt->keepBlanks) || | 
| 4494 | 139k |                   (ctxt->sax->ignorableWhitespace != ctxt->sax->characters); | 
| 4495 |  |  | 
| 4496 |  |     /* | 
| 4497 |  |      * Calling areBlanks with only parts of a text node | 
| 4498 |  |      * is fundamentally broken, making the NOBLANKS option | 
| 4499 |  |      * essentially unusable. | 
| 4500 |  |      */ | 
| 4501 | 139k |     if ((checkBlanks) && | 
| 4502 | 0 |         (areBlanks(ctxt, buf, size, isBlank))) { | 
| 4503 | 0 |         if ((ctxt->sax->ignorableWhitespace != NULL) && | 
| 4504 | 0 |             (ctxt->keepBlanks)) | 
| 4505 | 0 |             ctxt->sax->ignorableWhitespace(ctxt->userData, buf, size); | 
| 4506 | 139k |     } else { | 
| 4507 | 139k |         if (ctxt->sax->characters != NULL) | 
| 4508 | 139k |             ctxt->sax->characters(ctxt->userData, buf, size); | 
| 4509 |  |  | 
| 4510 |  |         /* | 
| 4511 |  |          * The old code used to update this value for "complex" data | 
| 4512 |  |          * even if checkBlanks was false. This was probably a bug. | 
| 4513 |  |          */ | 
| 4514 | 139k |         if ((checkBlanks) && (*ctxt->space == -1)) | 
| 4515 | 0 |             *ctxt->space = -2; | 
| 4516 | 139k |     } | 
| 4517 | 139k | } | 
| 4518 |  |  | 
| 4519 |  | /** | 
| 4520 |  |  * Parse character data. Always makes progress if the first char isn't | 
| 4521 |  |  * '<' or '&'. | 
| 4522 |  |  * | 
| 4523 |  |  * The right angle bracket (>) may be represented using the string ">", | 
| 4524 |  |  * and must, for compatibility, be escaped using ">" or a character | 
| 4525 |  |  * reference when it appears in the string "]]>" in content, when that | 
| 4526 |  |  * string is not marking the end of a CDATA section. | 
| 4527 |  |  * | 
| 4528 |  |  *     [14] CharData ::= [^<&]* - ([^<&]* ']]>' [^<&]*) | 
| 4529 |  |  * @param ctxt  an XML parser context | 
| 4530 |  |  * @param partial  buffer may contain partial UTF-8 sequences | 
| 4531 |  |  */ | 
| 4532 |  | static void | 
| 4533 | 1.68M | xmlParseCharDataInternal(xmlParserCtxtPtr ctxt, int partial) { | 
| 4534 | 1.68M |     const xmlChar *in; | 
| 4535 | 1.68M |     int line = ctxt->input->line; | 
| 4536 | 1.68M |     int col = ctxt->input->col; | 
| 4537 | 1.68M |     int ccol; | 
| 4538 | 1.68M |     int terminate = 0; | 
| 4539 |  |  | 
| 4540 | 1.68M |     GROW; | 
| 4541 |  |     /* | 
| 4542 |  |      * Accelerated common case where input don't need to be | 
| 4543 |  |      * modified before passing it to the handler. | 
| 4544 |  |      */ | 
| 4545 | 1.68M |     in = ctxt->input->cur; | 
| 4546 | 1.68M |     do { | 
| 4547 | 1.70M | get_more_space: | 
| 4548 | 1.81M |         while (*in == 0x20) { in++; ctxt->input->col++; } | 
| 4549 | 1.70M |         if (*in == 0xA) { | 
| 4550 | 569k |             do { | 
| 4551 | 569k |                 ctxt->input->line++; ctxt->input->col = 1; | 
| 4552 | 569k |                 in++; | 
| 4553 | 569k |             } while (*in == 0xA); | 
| 4554 | 11.2k |             goto get_more_space; | 
| 4555 | 11.2k |         } | 
| 4556 | 1.68M |         if (*in == '<') { | 
| 4557 | 5.68k |             while (in > ctxt->input->cur) { | 
| 4558 | 2.83k |                 const xmlChar *tmp = ctxt->input->cur; | 
| 4559 | 2.83k |                 size_t nbchar = in - tmp; | 
| 4560 |  |  | 
| 4561 | 2.83k |                 if (nbchar > XML_MAX_ITEMS) | 
| 4562 | 0 |                     nbchar = XML_MAX_ITEMS; | 
| 4563 | 2.83k |                 ctxt->input->cur += nbchar; | 
| 4564 |  |  | 
| 4565 | 2.83k |                 xmlCharacters(ctxt, tmp, nbchar, 1); | 
| 4566 | 2.83k |             } | 
| 4567 | 2.84k |             return; | 
| 4568 | 2.84k |         } | 
| 4569 |  |  | 
| 4570 | 1.72M | get_more: | 
| 4571 | 1.72M |         ccol = ctxt->input->col; | 
| 4572 | 4.89M |         while (test_char_data[*in]) { | 
| 4573 | 3.17M |             in++; | 
| 4574 | 3.17M |             ccol++; | 
| 4575 | 3.17M |         } | 
| 4576 | 1.72M |         ctxt->input->col = ccol; | 
| 4577 | 1.72M |         if (*in == 0xA) { | 
| 4578 | 311k |             do { | 
| 4579 | 311k |                 ctxt->input->line++; ctxt->input->col = 1; | 
| 4580 | 311k |                 in++; | 
| 4581 | 311k |             } while (*in == 0xA); | 
| 4582 | 10.0k |             goto get_more; | 
| 4583 | 10.0k |         } | 
| 4584 | 1.71M |         if (*in == ']') { | 
| 4585 | 23.7k |             size_t avail = ctxt->input->end - in; | 
| 4586 |  |  | 
| 4587 | 23.7k |             if (partial && avail < 2) { | 
| 4588 | 0 |                 terminate = 1; | 
| 4589 | 0 |                 goto invoke_callback; | 
| 4590 | 0 |             } | 
| 4591 | 23.7k |             if (in[1] == ']') { | 
| 4592 | 13.8k |                 if (partial && avail < 3) { | 
| 4593 | 0 |                     terminate = 1; | 
| 4594 | 0 |                     goto invoke_callback; | 
| 4595 | 0 |                 } | 
| 4596 | 13.8k |                 if (in[2] == '>') | 
| 4597 | 1.93k |                     xmlFatalErr(ctxt, XML_ERR_MISPLACED_CDATA_END, NULL); | 
| 4598 | 13.8k |             } | 
| 4599 |  |  | 
| 4600 | 23.7k |             in++; | 
| 4601 | 23.7k |             ctxt->input->col++; | 
| 4602 | 23.7k |             goto get_more; | 
| 4603 | 23.7k |         } | 
| 4604 |  |  | 
| 4605 | 1.68M | invoke_callback: | 
| 4606 | 1.90M |         while (in > ctxt->input->cur) { | 
| 4607 | 219k |             const xmlChar *tmp = ctxt->input->cur; | 
| 4608 | 219k |             size_t nbchar = in - tmp; | 
| 4609 |  |  | 
| 4610 | 219k |             if (nbchar > XML_MAX_ITEMS) | 
| 4611 | 0 |                 nbchar = XML_MAX_ITEMS; | 
| 4612 | 219k |             ctxt->input->cur += nbchar; | 
| 4613 |  |  | 
| 4614 | 219k |             xmlCharacters(ctxt, tmp, nbchar, 0); | 
| 4615 |  |  | 
| 4616 | 219k |             line = ctxt->input->line; | 
| 4617 | 219k |             col = ctxt->input->col; | 
| 4618 | 219k |         } | 
| 4619 | 1.68M |         ctxt->input->cur = in; | 
| 4620 | 1.68M |         if (*in == 0xD) { | 
| 4621 | 14.1k |             in++; | 
| 4622 | 14.1k |             if (*in == 0xA) { | 
| 4623 | 632 |                 ctxt->input->cur = in; | 
| 4624 | 632 |                 in++; | 
| 4625 | 632 |                 ctxt->input->line++; ctxt->input->col = 1; | 
| 4626 | 632 |                 continue; /* while */ | 
| 4627 | 632 |             } | 
| 4628 | 13.4k |             in--; | 
| 4629 | 13.4k |         } | 
| 4630 | 1.68M |         if (*in == '<') { | 
| 4631 | 101k |             return; | 
| 4632 | 101k |         } | 
| 4633 | 1.58M |         if (*in == '&') { | 
| 4634 | 34.2k |             return; | 
| 4635 | 34.2k |         } | 
| 4636 | 1.55M |         if (terminate) { | 
| 4637 | 0 |             return; | 
| 4638 | 0 |         } | 
| 4639 | 1.55M |         SHRINK; | 
| 4640 | 1.55M |         GROW; | 
| 4641 | 1.55M |         in = ctxt->input->cur; | 
| 4642 | 1.55M |     } while (((*in >= 0x20) && (*in <= 0x7F)) || | 
| 4643 | 1.55M |              (*in == 0x09) || (*in == 0x0a)); | 
| 4644 | 1.55M |     ctxt->input->line = line; | 
| 4645 | 1.55M |     ctxt->input->col = col; | 
| 4646 | 1.55M |     xmlParseCharDataComplex(ctxt, partial); | 
| 4647 | 1.55M | } | 
| 4648 |  |  | 
| 4649 |  | /** | 
| 4650 |  |  * Always makes progress if the first char isn't '<' or '&'. | 
| 4651 |  |  * | 
| 4652 |  |  * parse a CharData section.this is the fallback function | 
| 4653 |  |  * of #xmlParseCharData when the parsing requires handling | 
| 4654 |  |  * of non-ASCII characters. | 
| 4655 |  |  * | 
| 4656 |  |  * @param ctxt  an XML parser context | 
| 4657 |  |  * @param partial  whether the input can end with truncated UTF-8 | 
| 4658 |  |  */ | 
| 4659 |  | static void | 
| 4660 | 1.55M | xmlParseCharDataComplex(xmlParserCtxtPtr ctxt, int partial) { | 
| 4661 | 1.55M |     xmlChar buf[XML_PARSER_BIG_BUFFER_SIZE + 5]; | 
| 4662 | 1.55M |     int nbchar = 0; | 
| 4663 | 1.55M |     int cur, l; | 
| 4664 |  |  | 
| 4665 | 1.55M |     cur = xmlCurrentCharRecover(ctxt, &l); | 
| 4666 | 34.8M |     while ((cur != '<') && /* checked */ | 
| 4667 | 34.7M |            (cur != '&') && | 
| 4668 | 34.7M |      (IS_CHAR(cur))) { | 
| 4669 | 33.3M |         if (cur == ']') { | 
| 4670 | 14.7k |             size_t avail = ctxt->input->end - ctxt->input->cur; | 
| 4671 |  |  | 
| 4672 | 14.7k |             if (partial && avail < 2) | 
| 4673 | 0 |                 break; | 
| 4674 | 14.7k |             if (NXT(1) == ']') { | 
| 4675 | 8.40k |                 if (partial && avail < 3) | 
| 4676 | 0 |                     break; | 
| 4677 | 8.40k |                 if (NXT(2) == '>') | 
| 4678 | 2.42k |                     xmlFatalErr(ctxt, XML_ERR_MISPLACED_CDATA_END, NULL); | 
| 4679 | 8.40k |             } | 
| 4680 | 14.7k |         } | 
| 4681 |  |  | 
| 4682 | 33.3M |   COPY_BUF(buf, nbchar, cur); | 
| 4683 |  |   /* move current position before possible calling of ctxt->sax->characters */ | 
| 4684 | 33.3M |   NEXTL(l); | 
| 4685 | 33.3M |   if (nbchar >= XML_PARSER_BIG_BUFFER_SIZE) { | 
| 4686 | 224k |       buf[nbchar] = 0; | 
| 4687 |  |  | 
| 4688 | 224k |             xmlCharacters(ctxt, buf, nbchar, 0); | 
| 4689 | 224k |       nbchar = 0; | 
| 4690 | 224k |             SHRINK; | 
| 4691 | 224k |   } | 
| 4692 | 33.3M |   cur = xmlCurrentCharRecover(ctxt, &l); | 
| 4693 | 33.3M |     } | 
| 4694 | 1.55M |     if (nbchar != 0) { | 
| 4695 | 143k |         buf[nbchar] = 0; | 
| 4696 |  |  | 
| 4697 | 143k |         xmlCharacters(ctxt, buf, nbchar, 0); | 
| 4698 | 143k |     } | 
| 4699 |  |     /* | 
| 4700 |  |      * cur == 0 can mean | 
| 4701 |  |      * | 
| 4702 |  |      * - End of buffer. | 
| 4703 |  |      * - An actual 0 character. | 
| 4704 |  |      * - An incomplete UTF-8 sequence. This is allowed if partial is set. | 
| 4705 |  |      */ | 
| 4706 | 1.55M |     if (ctxt->input->cur < ctxt->input->end) { | 
| 4707 | 1.54M |         if ((cur == 0) && (CUR != 0)) { | 
| 4708 | 404 |             if (partial == 0) { | 
| 4709 | 404 |                 xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 4710 | 404 |                         "Incomplete UTF-8 sequence starting with %02X\n", CUR); | 
| 4711 | 404 |                 NEXTL(1); | 
| 4712 | 404 |             } | 
| 4713 | 1.54M |         } else if ((cur != '<') && (cur != '&') && (cur != ']')) { | 
| 4714 |  |             /* Generate the error and skip the offending character */ | 
| 4715 | 1.44M |             xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 4716 | 1.44M |                               "PCDATA invalid Char value %d\n", cur); | 
| 4717 | 1.44M |             NEXTL(l); | 
| 4718 | 1.44M |         } | 
| 4719 | 1.54M |     } | 
| 4720 | 1.55M | } | 
| 4721 |  |  | 
| 4722 |  | /** | 
| 4723 |  |  * @deprecated Internal function, don't use. | 
| 4724 |  |  * @param ctxt  an XML parser context | 
| 4725 |  |  * @param cdata  unused | 
| 4726 |  |  */ | 
| 4727 |  | void | 
| 4728 | 0 | xmlParseCharData(xmlParserCtxt *ctxt, ATTRIBUTE_UNUSED int cdata) { | 
| 4729 | 0 |     xmlParseCharDataInternal(ctxt, 0); | 
| 4730 | 0 | } | 
| 4731 |  |  | 
| 4732 |  | /** | 
| 4733 |  |  * Parse an External ID or a Public ID | 
| 4734 |  |  * | 
| 4735 |  |  * @deprecated Internal function, don't use. | 
| 4736 |  |  * | 
| 4737 |  |  * NOTE: Productions [75] and [83] interact badly since [75] can generate | 
| 4738 |  |  * `'PUBLIC' S PubidLiteral S SystemLiteral` | 
| 4739 |  |  * | 
| 4740 |  |  *     [75] ExternalID ::= 'SYSTEM' S SystemLiteral | 
| 4741 |  |  *                       | 'PUBLIC' S PubidLiteral S SystemLiteral | 
| 4742 |  |  * | 
| 4743 |  |  *     [83] PublicID ::= 'PUBLIC' S PubidLiteral | 
| 4744 |  |  * | 
| 4745 |  |  * @param ctxt  an XML parser context | 
| 4746 |  |  * @param publicId  a xmlChar** receiving PubidLiteral | 
| 4747 |  |  * @param strict  indicate whether we should restrict parsing to only | 
| 4748 |  |  *          production [75], see NOTE below | 
| 4749 |  |  * @returns the function returns SystemLiteral and in the second | 
| 4750 |  |  *                case publicID receives PubidLiteral, is strict is off | 
| 4751 |  |  *                it is possible to return NULL and have publicID set. | 
| 4752 |  |  */ | 
| 4753 |  |  | 
| 4754 |  | xmlChar * | 
| 4755 | 16.0k | xmlParseExternalID(xmlParserCtxt *ctxt, xmlChar **publicId, int strict) { | 
| 4756 | 16.0k |     xmlChar *URI = NULL; | 
| 4757 |  |  | 
| 4758 | 16.0k |     *publicId = NULL; | 
| 4759 | 16.0k |     if (CMP6(CUR_PTR, 'S', 'Y', 'S', 'T', 'E', 'M')) { | 
| 4760 | 2.76k |         SKIP(6); | 
| 4761 | 2.76k |   if (SKIP_BLANKS == 0) { | 
| 4762 | 861 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 4763 | 861 |                      "Space required after 'SYSTEM'\n"); | 
| 4764 | 861 |   } | 
| 4765 | 2.76k |   URI = xmlParseSystemLiteral(ctxt); | 
| 4766 | 2.76k |   if (URI == NULL) { | 
| 4767 | 466 |       xmlFatalErr(ctxt, XML_ERR_URI_REQUIRED, NULL); | 
| 4768 | 466 |         } | 
| 4769 | 13.2k |     } else if (CMP6(CUR_PTR, 'P', 'U', 'B', 'L', 'I', 'C')) { | 
| 4770 | 3.46k |         SKIP(6); | 
| 4771 | 3.46k |   if (SKIP_BLANKS == 0) { | 
| 4772 | 2.02k |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 4773 | 2.02k |         "Space required after 'PUBLIC'\n"); | 
| 4774 | 2.02k |   } | 
| 4775 | 3.46k |   *publicId = xmlParsePubidLiteral(ctxt); | 
| 4776 | 3.46k |   if (*publicId == NULL) { | 
| 4777 | 593 |       xmlFatalErr(ctxt, XML_ERR_PUBID_REQUIRED, NULL); | 
| 4778 | 593 |   } | 
| 4779 | 3.46k |   if (strict) { | 
| 4780 |  |       /* | 
| 4781 |  |        * We don't handle [83] so "S SystemLiteral" is required. | 
| 4782 |  |        */ | 
| 4783 | 1.39k |       if (SKIP_BLANKS == 0) { | 
| 4784 | 1.15k |     xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 4785 | 1.15k |       "Space required after the Public Identifier\n"); | 
| 4786 | 1.15k |       } | 
| 4787 | 2.07k |   } else { | 
| 4788 |  |       /* | 
| 4789 |  |        * We handle [83] so we return immediately, if | 
| 4790 |  |        * "S SystemLiteral" is not detected. We skip blanks if no | 
| 4791 |  |              * system literal was found, but this is harmless since we must | 
| 4792 |  |              * be at the end of a NotationDecl. | 
| 4793 |  |        */ | 
| 4794 | 2.07k |       if (SKIP_BLANKS == 0) return(NULL); | 
| 4795 | 933 |       if ((CUR != '\'') && (CUR != '"')) return(NULL); | 
| 4796 | 933 |   } | 
| 4797 | 2.06k |   URI = xmlParseSystemLiteral(ctxt); | 
| 4798 | 2.06k |   if (URI == NULL) { | 
| 4799 | 1.11k |       xmlFatalErr(ctxt, XML_ERR_URI_REQUIRED, NULL); | 
| 4800 | 1.11k |         } | 
| 4801 | 2.06k |     } | 
| 4802 | 14.6k |     return(URI); | 
| 4803 | 16.0k | } | 
| 4804 |  |  | 
| 4805 |  | /** | 
| 4806 |  |  * Skip an XML (SGML) comment <!-- .... --> | 
| 4807 |  |  *  The spec says that "For compatibility, the string "--" (double-hyphen) | 
| 4808 |  |  *  must not occur within comments. " | 
| 4809 |  |  * This is the slow routine in case the accelerator for ascii didn't work | 
| 4810 |  |  * | 
| 4811 |  |  *     [15] Comment ::= '<!--' ((Char - '-') | ('-' (Char - '-')))* '-->' | 
| 4812 |  |  * @param ctxt  an XML parser context | 
| 4813 |  |  * @param buf  the already parsed part of the buffer | 
| 4814 |  |  * @param len  number of bytes in the buffer | 
| 4815 |  |  * @param size  allocated size of the buffer | 
| 4816 |  |  */ | 
| 4817 |  | static void | 
| 4818 |  | xmlParseCommentComplex(xmlParserCtxtPtr ctxt, xmlChar *buf, | 
| 4819 | 22.8k |                        size_t len, size_t size) { | 
| 4820 | 22.8k |     int q, ql; | 
| 4821 | 22.8k |     int r, rl; | 
| 4822 | 22.8k |     int cur, l; | 
| 4823 | 22.8k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 4824 | 22.8k |                     XML_MAX_HUGE_LENGTH : | 
| 4825 | 22.8k |                     XML_MAX_TEXT_LENGTH; | 
| 4826 |  |  | 
| 4827 | 22.8k |     if (buf == NULL) { | 
| 4828 | 11.2k |         len = 0; | 
| 4829 | 11.2k |   size = XML_PARSER_BUFFER_SIZE; | 
| 4830 | 11.2k |   buf = xmlMalloc(size); | 
| 4831 | 11.2k |   if (buf == NULL) { | 
| 4832 | 0 |       xmlErrMemory(ctxt); | 
| 4833 | 0 |       return; | 
| 4834 | 0 |   } | 
| 4835 | 11.2k |     } | 
| 4836 | 22.8k |     q = xmlCurrentCharRecover(ctxt, &ql); | 
| 4837 | 22.8k |     if (q == 0) | 
| 4838 | 1.95k |         goto not_terminated; | 
| 4839 | 20.9k |     if (!IS_CHAR(q)) { | 
| 4840 | 4.73k |         xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 4841 | 4.73k |                           "xmlParseComment: invalid xmlChar value %d\n", | 
| 4842 | 4.73k |                     q); | 
| 4843 | 4.73k |   xmlFree (buf); | 
| 4844 | 4.73k |   return; | 
| 4845 | 4.73k |     } | 
| 4846 | 16.1k |     NEXTL(ql); | 
| 4847 | 16.1k |     r = xmlCurrentCharRecover(ctxt, &rl); | 
| 4848 | 16.1k |     if (r == 0) | 
| 4849 | 1.60k |         goto not_terminated; | 
| 4850 | 14.5k |     if (!IS_CHAR(r)) { | 
| 4851 | 1.64k |         xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 4852 | 1.64k |                           "xmlParseComment: invalid xmlChar value %d\n", | 
| 4853 | 1.64k |                     r); | 
| 4854 | 1.64k |   xmlFree (buf); | 
| 4855 | 1.64k |   return; | 
| 4856 | 1.64k |     } | 
| 4857 | 12.9k |     NEXTL(rl); | 
| 4858 | 12.9k |     cur = xmlCurrentCharRecover(ctxt, &l); | 
| 4859 | 12.9k |     if (cur == 0) | 
| 4860 | 436 |         goto not_terminated; | 
| 4861 | 1.33M |     while (IS_CHAR(cur) && /* checked */ | 
| 4862 | 1.32M |            ((cur != '>') || | 
| 4863 | 1.31M |       (r != '-') || (q != '-'))) { | 
| 4864 | 1.31M |   if ((r == '-') && (q == '-')) { | 
| 4865 | 3.35k |       xmlFatalErr(ctxt, XML_ERR_HYPHEN_IN_COMMENT, NULL); | 
| 4866 | 3.35k |   } | 
| 4867 | 1.31M |   if (len + 5 >= size) { | 
| 4868 | 3.31k |       xmlChar *tmp; | 
| 4869 | 3.31k |             int newSize; | 
| 4870 |  |  | 
| 4871 | 3.31k |       newSize = xmlGrowCapacity(size, 1, 1, maxLength); | 
| 4872 | 3.31k |             if (newSize < 0) { | 
| 4873 | 0 |                 xmlFatalErrMsgStr(ctxt, XML_ERR_COMMENT_NOT_FINISHED, | 
| 4874 | 0 |                              "Comment too big found", NULL); | 
| 4875 | 0 |                 xmlFree (buf); | 
| 4876 | 0 |                 return; | 
| 4877 | 0 |             } | 
| 4878 | 3.31k |       tmp = xmlRealloc(buf, newSize); | 
| 4879 | 3.31k |       if (tmp == NULL) { | 
| 4880 | 0 |     xmlErrMemory(ctxt); | 
| 4881 | 0 |     xmlFree(buf); | 
| 4882 | 0 |     return; | 
| 4883 | 0 |       } | 
| 4884 | 3.31k |       buf = tmp; | 
| 4885 | 3.31k |             size = newSize; | 
| 4886 | 3.31k |   } | 
| 4887 | 1.31M |   COPY_BUF(buf, len, q); | 
| 4888 |  |  | 
| 4889 | 1.31M |   q = r; | 
| 4890 | 1.31M |   ql = rl; | 
| 4891 | 1.31M |   r = cur; | 
| 4892 | 1.31M |   rl = l; | 
| 4893 |  |  | 
| 4894 | 1.31M |   NEXTL(l); | 
| 4895 | 1.31M |   cur = xmlCurrentCharRecover(ctxt, &l); | 
| 4896 |  |  | 
| 4897 | 1.31M |     } | 
| 4898 | 12.4k |     buf[len] = 0; | 
| 4899 | 12.4k |     if (cur == 0) { | 
| 4900 | 1.89k |   xmlFatalErrMsgStr(ctxt, XML_ERR_COMMENT_NOT_FINISHED, | 
| 4901 | 1.89k |                        "Comment not terminated \n<!--%.50s\n", buf); | 
| 4902 | 10.6k |     } else if (!IS_CHAR(cur)) { | 
| 4903 | 7.48k |         xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 4904 | 7.48k |                           "xmlParseComment: invalid xmlChar value %d\n", | 
| 4905 | 7.48k |                     cur); | 
| 4906 | 7.48k |     } else { | 
| 4907 | 3.12k |         NEXT; | 
| 4908 | 3.12k |   if ((ctxt->sax != NULL) && (ctxt->sax->comment != NULL) && | 
| 4909 | 3.12k |       (!ctxt->disableSAX)) | 
| 4910 | 1.05k |       ctxt->sax->comment(ctxt->userData, buf); | 
| 4911 | 3.12k |     } | 
| 4912 | 12.4k |     xmlFree(buf); | 
| 4913 | 12.4k |     return; | 
| 4914 | 3.98k | not_terminated: | 
| 4915 | 3.98k |     xmlFatalErrMsgStr(ctxt, XML_ERR_COMMENT_NOT_FINISHED, | 
| 4916 | 3.98k |        "Comment not terminated\n", NULL); | 
| 4917 | 3.98k |     xmlFree(buf); | 
| 4918 | 3.98k | } | 
| 4919 |  |  | 
| 4920 |  | /** | 
| 4921 |  |  * Parse an XML (SGML) comment. Always consumes '<!'. | 
| 4922 |  |  * | 
| 4923 |  |  * @deprecated Internal function, don't use. | 
| 4924 |  |  * | 
| 4925 |  |  *  The spec says that "For compatibility, the string "--" (double-hyphen) | 
| 4926 |  |  *  must not occur within comments. " | 
| 4927 |  |  * | 
| 4928 |  |  *     [15] Comment ::= '<!--' ((Char - '-') | ('-' (Char - '-')))* '-->' | 
| 4929 |  |  * @param ctxt  an XML parser context | 
| 4930 |  |  */ | 
| 4931 |  | void | 
| 4932 | 33.7k | xmlParseComment(xmlParserCtxt *ctxt) { | 
| 4933 | 33.7k |     xmlChar *buf = NULL; | 
| 4934 | 33.7k |     size_t size = XML_PARSER_BUFFER_SIZE; | 
| 4935 | 33.7k |     size_t len = 0; | 
| 4936 | 33.7k |     size_t maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 4937 | 33.7k |                        XML_MAX_HUGE_LENGTH : | 
| 4938 | 33.7k |                        XML_MAX_TEXT_LENGTH; | 
| 4939 | 33.7k |     const xmlChar *in; | 
| 4940 | 33.7k |     size_t nbchar = 0; | 
| 4941 | 33.7k |     int ccol; | 
| 4942 |  |  | 
| 4943 |  |     /* | 
| 4944 |  |      * Check that there is a comment right here. | 
| 4945 |  |      */ | 
| 4946 | 33.7k |     if ((RAW != '<') || (NXT(1) != '!')) | 
| 4947 | 0 |         return; | 
| 4948 | 33.7k |     SKIP(2); | 
| 4949 | 33.7k |     if ((RAW != '-') || (NXT(1) != '-')) | 
| 4950 | 8 |         return; | 
| 4951 | 33.7k |     SKIP(2); | 
| 4952 | 33.7k |     GROW; | 
| 4953 |  |  | 
| 4954 |  |     /* | 
| 4955 |  |      * Accelerated common case where input don't need to be | 
| 4956 |  |      * modified before passing it to the handler. | 
| 4957 |  |      */ | 
| 4958 | 33.7k |     in = ctxt->input->cur; | 
| 4959 | 34.2k |     do { | 
| 4960 | 34.2k |   if (*in == 0xA) { | 
| 4961 | 3.12k |       do { | 
| 4962 | 3.12k |     ctxt->input->line++; ctxt->input->col = 1; | 
| 4963 | 3.12k |     in++; | 
| 4964 | 3.12k |       } while (*in == 0xA); | 
| 4965 | 546 |   } | 
| 4966 | 234k | get_more: | 
| 4967 | 234k |         ccol = ctxt->input->col; | 
| 4968 | 3.24M |   while (((*in > '-') && (*in <= 0x7F)) || | 
| 4969 | 772k |          ((*in >= 0x20) && (*in < '-')) || | 
| 4970 | 3.00M |          (*in == 0x09)) { | 
| 4971 | 3.00M |         in++; | 
| 4972 | 3.00M |         ccol++; | 
| 4973 | 3.00M |   } | 
| 4974 | 234k |   ctxt->input->col = ccol; | 
| 4975 | 234k |   if (*in == 0xA) { | 
| 4976 | 61.3k |       do { | 
| 4977 | 61.3k |     ctxt->input->line++; ctxt->input->col = 1; | 
| 4978 | 61.3k |     in++; | 
| 4979 | 61.3k |       } while (*in == 0xA); | 
| 4980 | 6.05k |       goto get_more; | 
| 4981 | 6.05k |   } | 
| 4982 | 228k |   nbchar = in - ctxt->input->cur; | 
| 4983 |  |   /* | 
| 4984 |  |    * save current set of data | 
| 4985 |  |    */ | 
| 4986 | 228k |   if (nbchar > 0) { | 
| 4987 | 203k |             if (nbchar > maxLength - len) { | 
| 4988 | 0 |                 xmlFatalErrMsgStr(ctxt, XML_ERR_COMMENT_NOT_FINISHED, | 
| 4989 | 0 |                                   "Comment too big found", NULL); | 
| 4990 | 0 |                 xmlFree(buf); | 
| 4991 | 0 |                 return; | 
| 4992 | 0 |             } | 
| 4993 | 203k |             if (buf == NULL) { | 
| 4994 | 15.3k |                 if ((*in == '-') && (in[1] == '-')) | 
| 4995 | 5.35k |                     size = nbchar + 1; | 
| 4996 | 9.97k |                 else | 
| 4997 | 9.97k |                     size = XML_PARSER_BUFFER_SIZE + nbchar; | 
| 4998 | 15.3k |                 buf = xmlMalloc(size); | 
| 4999 | 15.3k |                 if (buf == NULL) { | 
| 5000 | 0 |                     xmlErrMemory(ctxt); | 
| 5001 | 0 |                     return; | 
| 5002 | 0 |                 } | 
| 5003 | 15.3k |                 len = 0; | 
| 5004 | 187k |             } else if (len + nbchar + 1 >= size) { | 
| 5005 | 3.62k |                 xmlChar *new_buf; | 
| 5006 | 3.62k |                 size += len + nbchar + XML_PARSER_BUFFER_SIZE; | 
| 5007 | 3.62k |                 new_buf = xmlRealloc(buf, size); | 
| 5008 | 3.62k |                 if (new_buf == NULL) { | 
| 5009 | 0 |                     xmlErrMemory(ctxt); | 
| 5010 | 0 |                     xmlFree(buf); | 
| 5011 | 0 |                     return; | 
| 5012 | 0 |                 } | 
| 5013 | 3.62k |                 buf = new_buf; | 
| 5014 | 3.62k |             } | 
| 5015 | 203k |             memcpy(&buf[len], ctxt->input->cur, nbchar); | 
| 5016 | 203k |             len += nbchar; | 
| 5017 | 203k |             buf[len] = 0; | 
| 5018 | 203k |   } | 
| 5019 | 228k |   ctxt->input->cur = in; | 
| 5020 | 228k |   if (*in == 0xA) { | 
| 5021 | 0 |       in++; | 
| 5022 | 0 |       ctxt->input->line++; ctxt->input->col = 1; | 
| 5023 | 0 |   } | 
| 5024 | 228k |   if (*in == 0xD) { | 
| 5025 | 4.42k |       in++; | 
| 5026 | 4.42k |       if (*in == 0xA) { | 
| 5027 | 2.21k |     ctxt->input->cur = in; | 
| 5028 | 2.21k |     in++; | 
| 5029 | 2.21k |     ctxt->input->line++; ctxt->input->col = 1; | 
| 5030 | 2.21k |     goto get_more; | 
| 5031 | 2.21k |       } | 
| 5032 | 2.20k |       in--; | 
| 5033 | 2.20k |   } | 
| 5034 | 226k |   SHRINK; | 
| 5035 | 226k |   GROW; | 
| 5036 | 226k |   in = ctxt->input->cur; | 
| 5037 | 226k |   if (*in == '-') { | 
| 5038 | 202k |       if (in[1] == '-') { | 
| 5039 | 191k |           if (in[2] == '>') { | 
| 5040 | 10.8k |         SKIP(3); | 
| 5041 | 10.8k |         if ((ctxt->sax != NULL) && (ctxt->sax->comment != NULL) && | 
| 5042 | 10.8k |             (!ctxt->disableSAX)) { | 
| 5043 | 1.16k |       if (buf != NULL) | 
| 5044 | 499 |           ctxt->sax->comment(ctxt->userData, buf); | 
| 5045 | 661 |       else | 
| 5046 | 661 |           ctxt->sax->comment(ctxt->userData, BAD_CAST ""); | 
| 5047 | 1.16k |         } | 
| 5048 | 10.8k |         if (buf != NULL) | 
| 5049 | 3.65k |             xmlFree(buf); | 
| 5050 | 10.8k |         return; | 
| 5051 | 10.8k |     } | 
| 5052 | 180k |     if (buf != NULL) { | 
| 5053 | 178k |         xmlFatalErrMsgStr(ctxt, XML_ERR_HYPHEN_IN_COMMENT, | 
| 5054 | 178k |                           "Double hyphen within comment: " | 
| 5055 | 178k |                                       "<!--%.50s\n", | 
| 5056 | 178k |               buf); | 
| 5057 | 178k |     } else | 
| 5058 | 1.94k |         xmlFatalErrMsgStr(ctxt, XML_ERR_HYPHEN_IN_COMMENT, | 
| 5059 | 1.94k |                           "Double hyphen within comment\n", NULL); | 
| 5060 | 180k |     in++; | 
| 5061 | 180k |     ctxt->input->col++; | 
| 5062 | 180k |       } | 
| 5063 | 191k |       in++; | 
| 5064 | 191k |       ctxt->input->col++; | 
| 5065 | 191k |       goto get_more; | 
| 5066 | 202k |   } | 
| 5067 | 226k |     } while (((*in >= 0x20) && (*in <= 0x7F)) || (*in == 0x09) || (*in == 0x0a)); | 
| 5068 | 22.8k |     xmlParseCommentComplex(ctxt, buf, len, size); | 
| 5069 | 22.8k | } | 
| 5070 |  |  | 
| 5071 |  |  | 
| 5072 |  | /** | 
| 5073 |  |  * Parse the name of a PI | 
| 5074 |  |  * | 
| 5075 |  |  * @deprecated Internal function, don't use. | 
| 5076 |  |  * | 
| 5077 |  |  *     [17] PITarget ::= Name - (('X' | 'x') ('M' | 'm') ('L' | 'l')) | 
| 5078 |  |  * | 
| 5079 |  |  * @param ctxt  an XML parser context | 
| 5080 |  |  * @returns the PITarget name or NULL | 
| 5081 |  |  */ | 
| 5082 |  |  | 
| 5083 |  | const xmlChar * | 
| 5084 | 48.0k | xmlParsePITarget(xmlParserCtxt *ctxt) { | 
| 5085 | 48.0k |     const xmlChar *name; | 
| 5086 |  |  | 
| 5087 | 48.0k |     name = xmlParseName(ctxt); | 
| 5088 | 48.0k |     if ((name != NULL) && | 
| 5089 | 40.4k |         ((name[0] == 'x') || (name[0] == 'X')) && | 
| 5090 | 26.5k |         ((name[1] == 'm') || (name[1] == 'M')) && | 
| 5091 | 23.9k |         ((name[2] == 'l') || (name[2] == 'L'))) { | 
| 5092 | 22.6k |   int i; | 
| 5093 | 22.6k |   if ((name[0] == 'x') && (name[1] == 'm') && | 
| 5094 | 21.5k |       (name[2] == 'l') && (name[3] == 0)) { | 
| 5095 | 1.83k |       xmlFatalErrMsg(ctxt, XML_ERR_RESERVED_XML_NAME, | 
| 5096 | 1.83k |      "XML declaration allowed only at the start of the document\n"); | 
| 5097 | 1.83k |       return(name); | 
| 5098 | 20.8k |   } else if (name[3] == 0) { | 
| 5099 | 752 |       xmlFatalErr(ctxt, XML_ERR_RESERVED_XML_NAME, NULL); | 
| 5100 | 752 |       return(name); | 
| 5101 | 752 |   } | 
| 5102 | 42.6k |   for (i = 0;;i++) { | 
| 5103 | 42.6k |       if (xmlW3CPIs[i] == NULL) break; | 
| 5104 | 40.1k |       if (xmlStrEqual(name, (const xmlChar *)xmlW3CPIs[i])) | 
| 5105 | 17.6k |           return(name); | 
| 5106 | 40.1k |   } | 
| 5107 | 2.47k |   xmlWarningMsg(ctxt, XML_ERR_RESERVED_XML_NAME, | 
| 5108 | 2.47k |           "xmlParsePITarget: invalid name prefix 'xml'\n", | 
| 5109 | 2.47k |           NULL, NULL); | 
| 5110 | 2.47k |     } | 
| 5111 | 27.8k |     if ((name != NULL) && (xmlStrchr(name, ':') != NULL)) { | 
| 5112 | 1.38k |   xmlNsErr(ctxt, XML_NS_ERR_COLON, | 
| 5113 | 1.38k |      "colons are forbidden from PI names '%s'\n", name, NULL, NULL); | 
| 5114 | 1.38k |     } | 
| 5115 | 27.8k |     return(name); | 
| 5116 | 48.0k | } | 
| 5117 |  |  | 
| 5118 |  | #ifdef LIBXML_CATALOG_ENABLED | 
| 5119 |  | /** | 
| 5120 |  |  * Parse an XML Catalog Processing Instruction. | 
| 5121 |  |  * | 
| 5122 |  |  * <?oasis-xml-catalog catalog="http://example.com/catalog.xml"?> | 
| 5123 |  |  * | 
| 5124 |  |  * Occurs only if allowed by the user and if happening in the Misc | 
| 5125 |  |  * part of the document before any doctype information | 
| 5126 |  |  * This will add the given catalog to the parsing context in order | 
| 5127 |  |  * to be used if there is a resolution need further down in the document | 
| 5128 |  |  * | 
| 5129 |  |  * @param ctxt  an XML parser context | 
| 5130 |  |  * @param catalog  the PI value string | 
| 5131 |  |  */ | 
| 5132 |  |  | 
| 5133 |  | static void | 
| 5134 | 0 | xmlParseCatalogPI(xmlParserCtxtPtr ctxt, const xmlChar *catalog) { | 
| 5135 | 0 |     xmlChar *URL = NULL; | 
| 5136 | 0 |     const xmlChar *tmp, *base; | 
| 5137 | 0 |     xmlChar marker; | 
| 5138 |  | 
 | 
| 5139 | 0 |     tmp = catalog; | 
| 5140 | 0 |     while (IS_BLANK_CH(*tmp)) tmp++; | 
| 5141 | 0 |     if (xmlStrncmp(tmp, BAD_CAST"catalog", 7)) | 
| 5142 | 0 |   goto error; | 
| 5143 | 0 |     tmp += 7; | 
| 5144 | 0 |     while (IS_BLANK_CH(*tmp)) tmp++; | 
| 5145 | 0 |     if (*tmp != '=') { | 
| 5146 | 0 |   return; | 
| 5147 | 0 |     } | 
| 5148 | 0 |     tmp++; | 
| 5149 | 0 |     while (IS_BLANK_CH(*tmp)) tmp++; | 
| 5150 | 0 |     marker = *tmp; | 
| 5151 | 0 |     if ((marker != '\'') && (marker != '"')) | 
| 5152 | 0 |   goto error; | 
| 5153 | 0 |     tmp++; | 
| 5154 | 0 |     base = tmp; | 
| 5155 | 0 |     while ((*tmp != 0) && (*tmp != marker)) tmp++; | 
| 5156 | 0 |     if (*tmp == 0) | 
| 5157 | 0 |   goto error; | 
| 5158 | 0 |     URL = xmlStrndup(base, tmp - base); | 
| 5159 | 0 |     tmp++; | 
| 5160 | 0 |     while (IS_BLANK_CH(*tmp)) tmp++; | 
| 5161 | 0 |     if (*tmp != 0) | 
| 5162 | 0 |   goto error; | 
| 5163 |  |  | 
| 5164 | 0 |     if (URL != NULL) { | 
| 5165 |  |         /* | 
| 5166 |  |          * Unfortunately, the catalog API doesn't report OOM errors. | 
| 5167 |  |          * xmlGetLastError isn't very helpful since we don't know | 
| 5168 |  |          * where the last error came from. We'd have to reset it | 
| 5169 |  |          * before this call and restore it afterwards. | 
| 5170 |  |          */ | 
| 5171 | 0 |   ctxt->catalogs = xmlCatalogAddLocal(ctxt->catalogs, URL); | 
| 5172 | 0 |   xmlFree(URL); | 
| 5173 | 0 |     } | 
| 5174 | 0 |     return; | 
| 5175 |  |  | 
| 5176 | 0 | error: | 
| 5177 | 0 |     xmlWarningMsg(ctxt, XML_WAR_CATALOG_PI, | 
| 5178 | 0 |             "Catalog PI syntax error: %s\n", | 
| 5179 | 0 |       catalog, NULL); | 
| 5180 | 0 |     if (URL != NULL) | 
| 5181 | 0 |   xmlFree(URL); | 
| 5182 | 0 | } | 
| 5183 |  | #endif | 
| 5184 |  |  | 
| 5185 |  | /** | 
| 5186 |  |  * Parse an XML Processing Instruction. | 
| 5187 |  |  * | 
| 5188 |  |  * @deprecated Internal function, don't use. | 
| 5189 |  |  * | 
| 5190 |  |  *     [16] PI ::= '<?' PITarget (S (Char* - (Char* '?>' Char*)))? '?>' | 
| 5191 |  |  * | 
| 5192 |  |  * The processing is transferred to SAX once parsed. | 
| 5193 |  |  * | 
| 5194 |  |  * @param ctxt  an XML parser context | 
| 5195 |  |  */ | 
| 5196 |  |  | 
| 5197 |  | void | 
| 5198 | 48.0k | xmlParsePI(xmlParserCtxt *ctxt) { | 
| 5199 | 48.0k |     xmlChar *buf = NULL; | 
| 5200 | 48.0k |     size_t len = 0; | 
| 5201 | 48.0k |     size_t size = XML_PARSER_BUFFER_SIZE; | 
| 5202 | 48.0k |     size_t maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 5203 | 48.0k |                        XML_MAX_HUGE_LENGTH : | 
| 5204 | 48.0k |                        XML_MAX_TEXT_LENGTH; | 
| 5205 | 48.0k |     int cur, l; | 
| 5206 | 48.0k |     const xmlChar *target; | 
| 5207 |  |  | 
| 5208 | 48.0k |     if ((RAW == '<') && (NXT(1) == '?')) { | 
| 5209 |  |   /* | 
| 5210 |  |    * this is a Processing Instruction. | 
| 5211 |  |    */ | 
| 5212 | 48.0k |   SKIP(2); | 
| 5213 |  |  | 
| 5214 |  |   /* | 
| 5215 |  |    * Parse the target name and check for special support like | 
| 5216 |  |    * namespace. | 
| 5217 |  |    */ | 
| 5218 | 48.0k |         target = xmlParsePITarget(ctxt); | 
| 5219 | 48.0k |   if (target != NULL) { | 
| 5220 | 40.4k |       if ((RAW == '?') && (NXT(1) == '>')) { | 
| 5221 | 5.67k |     SKIP(2); | 
| 5222 |  |  | 
| 5223 |  |     /* | 
| 5224 |  |      * SAX: PI detected. | 
| 5225 |  |      */ | 
| 5226 | 5.67k |     if ((ctxt->sax) && (!ctxt->disableSAX) && | 
| 5227 | 1.38k |         (ctxt->sax->processingInstruction != NULL)) | 
| 5228 | 1.38k |         ctxt->sax->processingInstruction(ctxt->userData, | 
| 5229 | 1.38k |                                          target, NULL); | 
| 5230 | 5.67k |     return; | 
| 5231 | 5.67k |       } | 
| 5232 | 34.7k |       buf = xmlMalloc(size); | 
| 5233 | 34.7k |       if (buf == NULL) { | 
| 5234 | 0 |     xmlErrMemory(ctxt); | 
| 5235 | 0 |     return; | 
| 5236 | 0 |       } | 
| 5237 | 34.7k |       if (SKIP_BLANKS == 0) { | 
| 5238 | 28.2k |     xmlFatalErrMsgStr(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5239 | 28.2k |         "ParsePI: PI %s space expected\n", target); | 
| 5240 | 28.2k |       } | 
| 5241 | 34.7k |       cur = xmlCurrentCharRecover(ctxt, &l); | 
| 5242 | 4.51M |       while (IS_CHAR(cur) && /* checked */ | 
| 5243 | 4.48M |        ((cur != '?') || (NXT(1) != '>'))) { | 
| 5244 | 4.47M |     if (len + 5 >= size) { | 
| 5245 | 5.20k |         xmlChar *tmp; | 
| 5246 | 5.20k |                     int newSize; | 
| 5247 |  |  | 
| 5248 | 5.20k |                     newSize = xmlGrowCapacity(size, 1, 1, maxLength); | 
| 5249 | 5.20k |                     if (newSize < 0) { | 
| 5250 | 0 |                         xmlFatalErrMsgStr(ctxt, XML_ERR_PI_NOT_FINISHED, | 
| 5251 | 0 |                                           "PI %s too big found", target); | 
| 5252 | 0 |                         xmlFree(buf); | 
| 5253 | 0 |                         return; | 
| 5254 | 0 |                     } | 
| 5255 | 5.20k |         tmp = xmlRealloc(buf, newSize); | 
| 5256 | 5.20k |         if (tmp == NULL) { | 
| 5257 | 0 |       xmlErrMemory(ctxt); | 
| 5258 | 0 |       xmlFree(buf); | 
| 5259 | 0 |       return; | 
| 5260 | 0 |         } | 
| 5261 | 5.20k |         buf = tmp; | 
| 5262 | 5.20k |                     size = newSize; | 
| 5263 | 5.20k |     } | 
| 5264 | 4.47M |     COPY_BUF(buf, len, cur); | 
| 5265 | 4.47M |     NEXTL(l); | 
| 5266 | 4.47M |     cur = xmlCurrentCharRecover(ctxt, &l); | 
| 5267 | 4.47M |       } | 
| 5268 | 34.7k |       buf[len] = 0; | 
| 5269 | 34.7k |       if (cur != '?') { | 
| 5270 | 26.4k |     xmlFatalErrMsgStr(ctxt, XML_ERR_PI_NOT_FINISHED, | 
| 5271 | 26.4k |           "ParsePI: PI %s never end ...\n", target); | 
| 5272 | 26.4k |       } else { | 
| 5273 | 8.28k |     SKIP(2); | 
| 5274 |  |  | 
| 5275 | 8.28k | #ifdef LIBXML_CATALOG_ENABLED | 
| 5276 | 8.28k |     if ((ctxt->inSubset == 0) && | 
| 5277 | 5.13k |         (xmlStrEqual(target, XML_CATALOG_PI))) { | 
| 5278 | 1.56k |         xmlCatalogAllow allow = xmlCatalogGetDefaults(); | 
| 5279 |  |  | 
| 5280 | 1.56k |         if ((ctxt->options & XML_PARSE_CATALOG_PI) && | 
| 5281 | 0 |                         ((allow == XML_CATA_ALLOW_DOCUMENT) || | 
| 5282 | 0 |        (allow == XML_CATA_ALLOW_ALL))) | 
| 5283 | 0 |       xmlParseCatalogPI(ctxt, buf); | 
| 5284 | 1.56k |     } | 
| 5285 | 8.28k | #endif | 
| 5286 |  |  | 
| 5287 |  |     /* | 
| 5288 |  |      * SAX: PI detected. | 
| 5289 |  |      */ | 
| 5290 | 8.28k |     if ((ctxt->sax) && (!ctxt->disableSAX) && | 
| 5291 | 1.56k |         (ctxt->sax->processingInstruction != NULL)) | 
| 5292 | 1.56k |         ctxt->sax->processingInstruction(ctxt->userData, | 
| 5293 | 1.56k |                                          target, buf); | 
| 5294 | 8.28k |       } | 
| 5295 | 34.7k |       xmlFree(buf); | 
| 5296 | 34.7k |   } else { | 
| 5297 | 7.63k |       xmlFatalErr(ctxt, XML_ERR_PI_NOT_STARTED, NULL); | 
| 5298 | 7.63k |   } | 
| 5299 | 48.0k |     } | 
| 5300 | 48.0k | } | 
| 5301 |  |  | 
| 5302 |  | /** | 
| 5303 |  |  * Parse a notation declaration. Always consumes '<!'. | 
| 5304 |  |  * | 
| 5305 |  |  * @deprecated Internal function, don't use. | 
| 5306 |  |  * | 
| 5307 |  |  *     [82] NotationDecl ::= '<!NOTATION' S Name S (ExternalID |  PublicID) | 
| 5308 |  |  *                           S? '>' | 
| 5309 |  |  * | 
| 5310 |  |  * Hence there is actually 3 choices: | 
| 5311 |  |  * | 
| 5312 |  |  *     'PUBLIC' S PubidLiteral | 
| 5313 |  |  *     'PUBLIC' S PubidLiteral S SystemLiteral | 
| 5314 |  |  *     'SYSTEM' S SystemLiteral | 
| 5315 |  |  * | 
| 5316 |  |  * See the NOTE on #xmlParseExternalID. | 
| 5317 |  |  * | 
| 5318 |  |  * @param ctxt  an XML parser context | 
| 5319 |  |  */ | 
| 5320 |  |  | 
| 5321 |  | void | 
| 5322 | 4.06k | xmlParseNotationDecl(xmlParserCtxt *ctxt) { | 
| 5323 | 4.06k |     const xmlChar *name; | 
| 5324 | 4.06k |     xmlChar *Pubid; | 
| 5325 | 4.06k |     xmlChar *Systemid; | 
| 5326 |  |  | 
| 5327 | 4.06k |     if ((CUR != '<') || (NXT(1) != '!')) | 
| 5328 | 0 |         return; | 
| 5329 | 4.06k |     SKIP(2); | 
| 5330 |  |  | 
| 5331 | 4.06k |     if (CMP8(CUR_PTR, 'N', 'O', 'T', 'A', 'T', 'I', 'O', 'N')) { | 
| 5332 | 4.04k | #ifdef LIBXML_VALID_ENABLED | 
| 5333 | 4.04k |   int oldInputNr = ctxt->inputNr; | 
| 5334 | 4.04k | #endif | 
| 5335 |  |  | 
| 5336 | 4.04k |   SKIP(8); | 
| 5337 | 4.04k |   if (SKIP_BLANKS_PE == 0) { | 
| 5338 | 692 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5339 | 692 |          "Space required after '<!NOTATION'\n"); | 
| 5340 | 692 |       return; | 
| 5341 | 692 |   } | 
| 5342 |  |  | 
| 5343 | 3.34k |         name = xmlParseName(ctxt); | 
| 5344 | 3.34k |   if (name == NULL) { | 
| 5345 | 201 |       xmlFatalErr(ctxt, XML_ERR_NOTATION_NOT_STARTED, NULL); | 
| 5346 | 201 |       return; | 
| 5347 | 201 |   } | 
| 5348 | 3.14k |   if (xmlStrchr(name, ':') != NULL) { | 
| 5349 | 239 |       xmlNsErr(ctxt, XML_NS_ERR_COLON, | 
| 5350 | 239 |          "colons are forbidden from notation names '%s'\n", | 
| 5351 | 239 |          name, NULL, NULL); | 
| 5352 | 239 |   } | 
| 5353 | 3.14k |   if (SKIP_BLANKS_PE == 0) { | 
| 5354 | 356 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5355 | 356 |          "Space required after the NOTATION name'\n"); | 
| 5356 | 356 |       return; | 
| 5357 | 356 |   } | 
| 5358 |  |  | 
| 5359 |  |   /* | 
| 5360 |  |    * Parse the IDs. | 
| 5361 |  |    */ | 
| 5362 | 2.79k |   Systemid = xmlParseExternalID(ctxt, &Pubid, 0); | 
| 5363 | 2.79k |   SKIP_BLANKS_PE; | 
| 5364 |  |  | 
| 5365 | 2.79k |   if (RAW == '>') { | 
| 5366 | 1.20k | #ifdef LIBXML_VALID_ENABLED | 
| 5367 | 1.20k |       if ((ctxt->validate) && (ctxt->inputNr > oldInputNr)) { | 
| 5368 | 0 |     xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 5369 | 0 |                            "Notation declaration doesn't start and stop" | 
| 5370 | 0 |                                  " in the same entity\n", | 
| 5371 | 0 |                                  NULL, NULL); | 
| 5372 | 0 |       } | 
| 5373 | 1.20k | #endif | 
| 5374 | 1.20k |       NEXT; | 
| 5375 | 1.20k |       if ((ctxt->sax != NULL) && (!ctxt->disableSAX) && | 
| 5376 | 309 |     (ctxt->sax->notationDecl != NULL)) | 
| 5377 | 309 |     ctxt->sax->notationDecl(ctxt->userData, name, Pubid, Systemid); | 
| 5378 | 1.58k |   } else { | 
| 5379 | 1.58k |       xmlFatalErr(ctxt, XML_ERR_NOTATION_NOT_FINISHED, NULL); | 
| 5380 | 1.58k |   } | 
| 5381 | 2.79k |   if (Systemid != NULL) xmlFree(Systemid); | 
| 5382 | 2.79k |   if (Pubid != NULL) xmlFree(Pubid); | 
| 5383 | 2.79k |     } | 
| 5384 | 4.06k | } | 
| 5385 |  |  | 
| 5386 |  | /** | 
| 5387 |  |  * Parse an entity declaration. Always consumes '<!'. | 
| 5388 |  |  * | 
| 5389 |  |  * @deprecated Internal function, don't use. | 
| 5390 |  |  * | 
| 5391 |  |  *     [70] EntityDecl ::= GEDecl | PEDecl | 
| 5392 |  |  * | 
| 5393 |  |  *     [71] GEDecl ::= '<!ENTITY' S Name S EntityDef S? '>' | 
| 5394 |  |  * | 
| 5395 |  |  *     [72] PEDecl ::= '<!ENTITY' S '%' S Name S PEDef S? '>' | 
| 5396 |  |  * | 
| 5397 |  |  *     [73] EntityDef ::= EntityValue | (ExternalID NDataDecl?) | 
| 5398 |  |  * | 
| 5399 |  |  *     [74] PEDef ::= EntityValue | ExternalID | 
| 5400 |  |  * | 
| 5401 |  |  *     [76] NDataDecl ::= S 'NDATA' S Name | 
| 5402 |  |  * | 
| 5403 |  |  * [ VC: Notation Declared ] | 
| 5404 |  |  * The Name must match the declared name of a notation. | 
| 5405 |  |  * | 
| 5406 |  |  * @param ctxt  an XML parser context | 
| 5407 |  |  */ | 
| 5408 |  |  | 
| 5409 |  | void | 
| 5410 | 26.6k | xmlParseEntityDecl(xmlParserCtxt *ctxt) { | 
| 5411 | 26.6k |     const xmlChar *name = NULL; | 
| 5412 | 26.6k |     xmlChar *value = NULL; | 
| 5413 | 26.6k |     xmlChar *URI = NULL, *literal = NULL; | 
| 5414 | 26.6k |     const xmlChar *ndata = NULL; | 
| 5415 | 26.6k |     int isParameter = 0; | 
| 5416 | 26.6k |     xmlChar *orig = NULL; | 
| 5417 |  |  | 
| 5418 | 26.6k |     if ((CUR != '<') || (NXT(1) != '!')) | 
| 5419 | 0 |         return; | 
| 5420 | 26.6k |     SKIP(2); | 
| 5421 |  |  | 
| 5422 |  |     /* GROW; done in the caller */ | 
| 5423 | 26.6k |     if (CMP6(CUR_PTR, 'E', 'N', 'T', 'I', 'T', 'Y')) { | 
| 5424 | 26.6k | #ifdef LIBXML_VALID_ENABLED | 
| 5425 | 26.6k |   int oldInputNr = ctxt->inputNr; | 
| 5426 | 26.6k | #endif | 
| 5427 |  |  | 
| 5428 | 26.6k |   SKIP(6); | 
| 5429 | 26.6k |   if (SKIP_BLANKS_PE == 0) { | 
| 5430 | 15.4k |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5431 | 15.4k |          "Space required after '<!ENTITY'\n"); | 
| 5432 | 15.4k |   } | 
| 5433 |  |  | 
| 5434 | 26.6k |   if (RAW == '%') { | 
| 5435 | 4.61k |       NEXT; | 
| 5436 | 4.61k |       if (SKIP_BLANKS_PE == 0) { | 
| 5437 | 999 |     xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5438 | 999 |              "Space required after '%%'\n"); | 
| 5439 | 999 |       } | 
| 5440 | 4.61k |       isParameter = 1; | 
| 5441 | 4.61k |   } | 
| 5442 |  |  | 
| 5443 | 26.6k |         name = xmlParseName(ctxt); | 
| 5444 | 26.6k |   if (name == NULL) { | 
| 5445 | 644 |       xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 5446 | 644 |                      "xmlParseEntityDecl: no name\n"); | 
| 5447 | 644 |             return; | 
| 5448 | 644 |   } | 
| 5449 | 25.9k |   if (xmlStrchr(name, ':') != NULL) { | 
| 5450 | 894 |       xmlNsErr(ctxt, XML_NS_ERR_COLON, | 
| 5451 | 894 |          "colons are forbidden from entities names '%s'\n", | 
| 5452 | 894 |          name, NULL, NULL); | 
| 5453 | 894 |   } | 
| 5454 | 25.9k |   if (SKIP_BLANKS_PE == 0) { | 
| 5455 | 11.9k |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5456 | 11.9k |          "Space required after the entity name\n"); | 
| 5457 | 11.9k |   } | 
| 5458 |  |  | 
| 5459 |  |   /* | 
| 5460 |  |    * handle the various case of definitions... | 
| 5461 |  |    */ | 
| 5462 | 25.9k |   if (isParameter) { | 
| 5463 | 4.57k |       if ((RAW == '"') || (RAW == '\'')) { | 
| 5464 | 3.43k |           value = xmlParseEntityValue(ctxt, &orig); | 
| 5465 | 3.43k |     if (value) { | 
| 5466 | 3.22k |         if ((ctxt->sax != NULL) && | 
| 5467 | 3.22k |       (!ctxt->disableSAX) && (ctxt->sax->entityDecl != NULL)) | 
| 5468 | 2.10k |       ctxt->sax->entityDecl(ctxt->userData, name, | 
| 5469 | 2.10k |                         XML_INTERNAL_PARAMETER_ENTITY, | 
| 5470 | 2.10k |             NULL, NULL, value); | 
| 5471 | 3.22k |     } | 
| 5472 | 3.43k |       } else { | 
| 5473 | 1.13k |           URI = xmlParseExternalID(ctxt, &literal, 1); | 
| 5474 | 1.13k |     if ((URI == NULL) && (literal == NULL)) { | 
| 5475 | 211 |         xmlFatalErr(ctxt, XML_ERR_VALUE_REQUIRED, NULL); | 
| 5476 | 211 |     } | 
| 5477 | 1.13k |     if (URI) { | 
| 5478 | 728 |                     if (xmlStrchr(URI, '#')) { | 
| 5479 | 128 |                         xmlFatalErr(ctxt, XML_ERR_URI_FRAGMENT, NULL); | 
| 5480 | 600 |                     } else { | 
| 5481 | 600 |                         if ((ctxt->sax != NULL) && | 
| 5482 | 600 |                             (!ctxt->disableSAX) && | 
| 5483 | 241 |                             (ctxt->sax->entityDecl != NULL)) | 
| 5484 | 241 |                             ctxt->sax->entityDecl(ctxt->userData, name, | 
| 5485 | 241 |                                         XML_EXTERNAL_PARAMETER_ENTITY, | 
| 5486 | 241 |                                         literal, URI, NULL); | 
| 5487 | 600 |                     } | 
| 5488 | 728 |     } | 
| 5489 | 1.13k |       } | 
| 5490 | 21.3k |   } else { | 
| 5491 | 21.3k |       if ((RAW == '"') || (RAW == '\'')) { | 
| 5492 | 16.9k |           value = xmlParseEntityValue(ctxt, &orig); | 
| 5493 | 16.9k |     if ((ctxt->sax != NULL) && | 
| 5494 | 16.9k |         (!ctxt->disableSAX) && (ctxt->sax->entityDecl != NULL)) | 
| 5495 | 1.63k |         ctxt->sax->entityDecl(ctxt->userData, name, | 
| 5496 | 1.63k |         XML_INTERNAL_GENERAL_ENTITY, | 
| 5497 | 1.63k |         NULL, NULL, value); | 
| 5498 |  |     /* | 
| 5499 |  |      * For expat compatibility in SAX mode. | 
| 5500 |  |      */ | 
| 5501 | 16.9k |     if ((ctxt->myDoc == NULL) || | 
| 5502 | 15.4k |         (xmlStrEqual(ctxt->myDoc->version, SAX_COMPAT_MODE))) { | 
| 5503 | 8.41k |         if (ctxt->myDoc == NULL) { | 
| 5504 | 1.47k |       ctxt->myDoc = xmlNewDoc(SAX_COMPAT_MODE); | 
| 5505 | 1.47k |       if (ctxt->myDoc == NULL) { | 
| 5506 | 0 |           xmlErrMemory(ctxt); | 
| 5507 | 0 |           goto done; | 
| 5508 | 0 |       } | 
| 5509 | 1.47k |       ctxt->myDoc->properties = XML_DOC_INTERNAL; | 
| 5510 | 1.47k |         } | 
| 5511 | 8.41k |         if (ctxt->myDoc->intSubset == NULL) { | 
| 5512 | 1.47k |       ctxt->myDoc->intSubset = xmlNewDtd(ctxt->myDoc, | 
| 5513 | 1.47k |               BAD_CAST "fake", NULL, NULL); | 
| 5514 | 1.47k |                         if (ctxt->myDoc->intSubset == NULL) { | 
| 5515 | 0 |                             xmlErrMemory(ctxt); | 
| 5516 | 0 |                             goto done; | 
| 5517 | 0 |                         } | 
| 5518 | 1.47k |                     } | 
| 5519 |  |  | 
| 5520 | 8.41k |         xmlSAX2EntityDecl(ctxt, name, XML_INTERNAL_GENERAL_ENTITY, | 
| 5521 | 8.41k |                     NULL, NULL, value); | 
| 5522 | 8.41k |     } | 
| 5523 | 16.9k |       } else { | 
| 5524 | 4.45k |           URI = xmlParseExternalID(ctxt, &literal, 1); | 
| 5525 | 4.45k |     if ((URI == NULL) && (literal == NULL)) { | 
| 5526 | 2.49k |         xmlFatalErr(ctxt, XML_ERR_VALUE_REQUIRED, NULL); | 
| 5527 | 2.49k |     } | 
| 5528 | 4.45k |     if (URI) { | 
| 5529 | 1.16k |                     if (xmlStrchr(URI, '#')) { | 
| 5530 | 219 |                         xmlFatalErr(ctxt, XML_ERR_URI_FRAGMENT, NULL); | 
| 5531 | 219 |                     } | 
| 5532 | 1.16k |     } | 
| 5533 | 4.45k |     if ((RAW != '>') && (SKIP_BLANKS_PE == 0)) { | 
| 5534 | 2.98k |         xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5535 | 2.98k |            "Space required before 'NDATA'\n"); | 
| 5536 | 2.98k |     } | 
| 5537 | 4.45k |     if (CMP5(CUR_PTR, 'N', 'D', 'A', 'T', 'A')) { | 
| 5538 | 610 |         SKIP(5); | 
| 5539 | 610 |         if (SKIP_BLANKS_PE == 0) { | 
| 5540 | 225 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5541 | 225 |                "Space required after 'NDATA'\n"); | 
| 5542 | 225 |         } | 
| 5543 | 610 |         ndata = xmlParseName(ctxt); | 
| 5544 | 610 |         if ((ctxt->sax != NULL) && (!ctxt->disableSAX) && | 
| 5545 | 204 |             (ctxt->sax->unparsedEntityDecl != NULL)) | 
| 5546 | 204 |       ctxt->sax->unparsedEntityDecl(ctxt->userData, name, | 
| 5547 | 204 |             literal, URI, ndata); | 
| 5548 | 3.84k |     } else { | 
| 5549 | 3.84k |         if ((ctxt->sax != NULL) && | 
| 5550 | 3.84k |             (!ctxt->disableSAX) && (ctxt->sax->entityDecl != NULL)) | 
| 5551 | 341 |       ctxt->sax->entityDecl(ctxt->userData, name, | 
| 5552 | 341 |             XML_EXTERNAL_GENERAL_PARSED_ENTITY, | 
| 5553 | 341 |             literal, URI, NULL); | 
| 5554 |  |         /* | 
| 5555 |  |          * For expat compatibility in SAX mode. | 
| 5556 |  |          * assuming the entity replacement was asked for | 
| 5557 |  |          */ | 
| 5558 | 3.84k |         if ((ctxt->replaceEntities != 0) && | 
| 5559 | 3.84k |       ((ctxt->myDoc == NULL) || | 
| 5560 | 3.76k |       (xmlStrEqual(ctxt->myDoc->version, SAX_COMPAT_MODE)))) { | 
| 5561 | 930 |       if (ctxt->myDoc == NULL) { | 
| 5562 | 87 |           ctxt->myDoc = xmlNewDoc(SAX_COMPAT_MODE); | 
| 5563 | 87 |           if (ctxt->myDoc == NULL) { | 
| 5564 | 0 |               xmlErrMemory(ctxt); | 
| 5565 | 0 |         goto done; | 
| 5566 | 0 |           } | 
| 5567 | 87 |           ctxt->myDoc->properties = XML_DOC_INTERNAL; | 
| 5568 | 87 |       } | 
| 5569 |  |  | 
| 5570 | 930 |       if (ctxt->myDoc->intSubset == NULL) { | 
| 5571 | 87 |           ctxt->myDoc->intSubset = xmlNewDtd(ctxt->myDoc, | 
| 5572 | 87 |             BAD_CAST "fake", NULL, NULL); | 
| 5573 | 87 |                             if (ctxt->myDoc->intSubset == NULL) { | 
| 5574 | 0 |                                 xmlErrMemory(ctxt); | 
| 5575 | 0 |                                 goto done; | 
| 5576 | 0 |                             } | 
| 5577 | 87 |                         } | 
| 5578 | 930 |       xmlSAX2EntityDecl(ctxt, name, | 
| 5579 | 930 |                   XML_EXTERNAL_GENERAL_PARSED_ENTITY, | 
| 5580 | 930 |                   literal, URI, NULL); | 
| 5581 | 930 |         } | 
| 5582 | 3.84k |     } | 
| 5583 | 4.45k |       } | 
| 5584 | 21.3k |   } | 
| 5585 | 25.9k |   SKIP_BLANKS_PE; | 
| 5586 | 25.9k |   if (RAW != '>') { | 
| 5587 | 11.9k |       xmlFatalErrMsgStr(ctxt, XML_ERR_ENTITY_NOT_FINISHED, | 
| 5588 | 11.9k |               "xmlParseEntityDecl: entity %s not terminated\n", name); | 
| 5589 | 14.0k |   } else { | 
| 5590 | 14.0k | #ifdef LIBXML_VALID_ENABLED | 
| 5591 | 14.0k |       if ((ctxt->validate) && (ctxt->inputNr > oldInputNr)) { | 
| 5592 | 0 |     xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 5593 | 0 |                            "Entity declaration doesn't start and stop in" | 
| 5594 | 0 |                                  " the same entity\n", | 
| 5595 | 0 |                                  NULL, NULL); | 
| 5596 | 0 |       } | 
| 5597 | 14.0k | #endif | 
| 5598 | 14.0k |       NEXT; | 
| 5599 | 14.0k |   } | 
| 5600 | 25.9k |   if (orig != NULL) { | 
| 5601 |  |       /* | 
| 5602 |  |        * Ugly mechanism to save the raw entity value. | 
| 5603 |  |        */ | 
| 5604 | 19.9k |       xmlEntityPtr cur = NULL; | 
| 5605 |  |  | 
| 5606 | 19.9k |       if (isParameter) { | 
| 5607 | 3.22k |           if ((ctxt->sax != NULL) && | 
| 5608 | 3.22k |         (ctxt->sax->getParameterEntity != NULL)) | 
| 5609 | 3.22k |         cur = ctxt->sax->getParameterEntity(ctxt->userData, name); | 
| 5610 | 16.6k |       } else { | 
| 5611 | 16.6k |           if ((ctxt->sax != NULL) && | 
| 5612 | 16.6k |         (ctxt->sax->getEntity != NULL)) | 
| 5613 | 16.6k |         cur = ctxt->sax->getEntity(ctxt->userData, name); | 
| 5614 | 16.6k |     if ((cur == NULL) && (ctxt->userData==ctxt)) { | 
| 5615 | 6.48k |         cur = xmlSAX2GetEntity(ctxt, name); | 
| 5616 | 6.48k |     } | 
| 5617 | 16.6k |       } | 
| 5618 | 19.9k |             if ((cur != NULL) && (cur->orig == NULL)) { | 
| 5619 | 5.57k |     cur->orig = orig; | 
| 5620 | 5.57k |                 orig = NULL; | 
| 5621 | 5.57k |       } | 
| 5622 | 19.9k |   } | 
| 5623 |  |  | 
| 5624 | 25.9k | done: | 
| 5625 | 25.9k |   if (value != NULL) xmlFree(value); | 
| 5626 | 25.9k |   if (URI != NULL) xmlFree(URI); | 
| 5627 | 25.9k |   if (literal != NULL) xmlFree(literal); | 
| 5628 | 25.9k |         if (orig != NULL) xmlFree(orig); | 
| 5629 | 25.9k |     } | 
| 5630 | 26.6k | } | 
| 5631 |  |  | 
| 5632 |  | /** | 
| 5633 |  |  * Parse an attribute default declaration | 
| 5634 |  |  * | 
| 5635 |  |  * @deprecated Internal function, don't use. | 
| 5636 |  |  * | 
| 5637 |  |  *     [60] DefaultDecl ::= '#REQUIRED' | '#IMPLIED' | (('#FIXED' S)? AttValue) | 
| 5638 |  |  * | 
| 5639 |  |  * [ VC: Required Attribute ] | 
| 5640 |  |  * if the default declaration is the keyword \#REQUIRED, then the | 
| 5641 |  |  * attribute must be specified for all elements of the type in the | 
| 5642 |  |  * attribute-list declaration. | 
| 5643 |  |  * | 
| 5644 |  |  * [ VC: Attribute Default Legal ] | 
| 5645 |  |  * The declared default value must meet the lexical constraints of | 
| 5646 |  |  * the declared attribute type c.f. #xmlValidateAttributeDecl | 
| 5647 |  |  * | 
| 5648 |  |  * [ VC: Fixed Attribute Default ] | 
| 5649 |  |  * if an attribute has a default value declared with the \#FIXED | 
| 5650 |  |  * keyword, instances of that attribute must match the default value. | 
| 5651 |  |  * | 
| 5652 |  |  * [ WFC: No < in Attribute Values ] | 
| 5653 |  |  * handled in #xmlParseAttValue | 
| 5654 |  |  * | 
| 5655 |  |  * @param ctxt  an XML parser context | 
| 5656 |  |  * @param value  Receive a possible fixed default value for the attribute | 
| 5657 |  |  * @returns XML_ATTRIBUTE_NONE, XML_ATTRIBUTE_REQUIRED, XML_ATTRIBUTE_IMPLIED | 
| 5658 |  |  *          or XML_ATTRIBUTE_FIXED. | 
| 5659 |  |  */ | 
| 5660 |  |  | 
| 5661 |  | int | 
| 5662 | 21.5k | xmlParseDefaultDecl(xmlParserCtxt *ctxt, xmlChar **value) { | 
| 5663 | 21.5k |     int val; | 
| 5664 | 21.5k |     xmlChar *ret; | 
| 5665 |  |  | 
| 5666 | 21.5k |     *value = NULL; | 
| 5667 | 21.5k |     if (CMP9(CUR_PTR, '#', 'R', 'E', 'Q', 'U', 'I', 'R', 'E', 'D')) { | 
| 5668 | 1.39k |   SKIP(9); | 
| 5669 | 1.39k |   return(XML_ATTRIBUTE_REQUIRED); | 
| 5670 | 1.39k |     } | 
| 5671 | 20.1k |     if (CMP8(CUR_PTR, '#', 'I', 'M', 'P', 'L', 'I', 'E', 'D')) { | 
| 5672 | 290 |   SKIP(8); | 
| 5673 | 290 |   return(XML_ATTRIBUTE_IMPLIED); | 
| 5674 | 290 |     } | 
| 5675 | 19.8k |     val = XML_ATTRIBUTE_NONE; | 
| 5676 | 19.8k |     if (CMP6(CUR_PTR, '#', 'F', 'I', 'X', 'E', 'D')) { | 
| 5677 | 514 |   SKIP(6); | 
| 5678 | 514 |   val = XML_ATTRIBUTE_FIXED; | 
| 5679 | 514 |   if (SKIP_BLANKS_PE == 0) { | 
| 5680 | 263 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5681 | 263 |          "Space required after '#FIXED'\n"); | 
| 5682 | 263 |   } | 
| 5683 | 514 |     } | 
| 5684 | 19.8k |     ret = xmlParseAttValue(ctxt); | 
| 5685 | 19.8k |     if (ret == NULL) { | 
| 5686 | 3.35k |   xmlFatalErrMsg(ctxt, (xmlParserErrors)ctxt->errNo, | 
| 5687 | 3.35k |            "Attribute default value declaration error\n"); | 
| 5688 | 3.35k |     } else | 
| 5689 | 16.4k |         *value = ret; | 
| 5690 | 19.8k |     return(val); | 
| 5691 | 20.1k | } | 
| 5692 |  |  | 
| 5693 |  | /** | 
| 5694 |  |  * Parse an Notation attribute type. | 
| 5695 |  |  * | 
| 5696 |  |  * @deprecated Internal function, don't use. | 
| 5697 |  |  * | 
| 5698 |  |  * Note: the leading 'NOTATION' S part has already being parsed... | 
| 5699 |  |  * | 
| 5700 |  |  *     [58] NotationType ::= 'NOTATION' S '(' S? Name (S? '|' S? Name)* S? ')' | 
| 5701 |  |  * | 
| 5702 |  |  * [ VC: Notation Attributes ] | 
| 5703 |  |  * Values of this type must match one of the notation names included | 
| 5704 |  |  * in the declaration; all notation names in the declaration must be declared. | 
| 5705 |  |  * | 
| 5706 |  |  * @param ctxt  an XML parser context | 
| 5707 |  |  * @returns the notation attribute tree built while parsing | 
| 5708 |  |  */ | 
| 5709 |  |  | 
| 5710 |  | xmlEnumeration * | 
| 5711 | 1.11k | xmlParseNotationType(xmlParserCtxt *ctxt) { | 
| 5712 | 1.11k |     const xmlChar *name; | 
| 5713 | 1.11k |     xmlEnumerationPtr ret = NULL, last = NULL, cur, tmp; | 
| 5714 |  |  | 
| 5715 | 1.11k |     if (RAW != '(') { | 
| 5716 | 196 |   xmlFatalErr(ctxt, XML_ERR_NOTATION_NOT_STARTED, NULL); | 
| 5717 | 196 |   return(NULL); | 
| 5718 | 196 |     } | 
| 5719 | 1.30k |     do { | 
| 5720 | 1.30k |         NEXT; | 
| 5721 | 1.30k |   SKIP_BLANKS_PE; | 
| 5722 | 1.30k |         name = xmlParseName(ctxt); | 
| 5723 | 1.30k |   if (name == NULL) { | 
| 5724 | 358 |       xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 5725 | 358 |          "Name expected in NOTATION declaration\n"); | 
| 5726 | 358 |             xmlFreeEnumeration(ret); | 
| 5727 | 358 |       return(NULL); | 
| 5728 | 358 |   } | 
| 5729 | 945 |         tmp = NULL; | 
| 5730 | 945 | #ifdef LIBXML_VALID_ENABLED | 
| 5731 | 945 |         if (ctxt->validate) { | 
| 5732 | 0 |             tmp = ret; | 
| 5733 | 0 |             while (tmp != NULL) { | 
| 5734 | 0 |                 if (xmlStrEqual(name, tmp->name)) { | 
| 5735 | 0 |                     xmlValidityError(ctxt, XML_DTD_DUP_TOKEN, | 
| 5736 | 0 |               "standalone: attribute notation value token %s duplicated\n", | 
| 5737 | 0 |                                      name, NULL); | 
| 5738 | 0 |                     if (!xmlDictOwns(ctxt->dict, name)) | 
| 5739 | 0 |                         xmlFree((xmlChar *) name); | 
| 5740 | 0 |                     break; | 
| 5741 | 0 |                 } | 
| 5742 | 0 |                 tmp = tmp->next; | 
| 5743 | 0 |             } | 
| 5744 | 0 |         } | 
| 5745 | 945 | #endif /* LIBXML_VALID_ENABLED */ | 
| 5746 | 945 |   if (tmp == NULL) { | 
| 5747 | 945 |       cur = xmlCreateEnumeration(name); | 
| 5748 | 945 |       if (cur == NULL) { | 
| 5749 | 0 |                 xmlErrMemory(ctxt); | 
| 5750 | 0 |                 xmlFreeEnumeration(ret); | 
| 5751 | 0 |                 return(NULL); | 
| 5752 | 0 |             } | 
| 5753 | 945 |       if (last == NULL) ret = last = cur; | 
| 5754 | 386 |       else { | 
| 5755 | 386 |     last->next = cur; | 
| 5756 | 386 |     last = cur; | 
| 5757 | 386 |       } | 
| 5758 | 945 |   } | 
| 5759 | 945 |   SKIP_BLANKS_PE; | 
| 5760 | 945 |     } while (RAW == '|'); | 
| 5761 | 557 |     if (RAW != ')') { | 
| 5762 | 230 |   xmlFatalErr(ctxt, XML_ERR_NOTATION_NOT_FINISHED, NULL); | 
| 5763 | 230 |         xmlFreeEnumeration(ret); | 
| 5764 | 230 |   return(NULL); | 
| 5765 | 230 |     } | 
| 5766 | 327 |     NEXT; | 
| 5767 | 327 |     return(ret); | 
| 5768 | 557 | } | 
| 5769 |  |  | 
| 5770 |  | /** | 
| 5771 |  |  * Parse an Enumeration attribute type. | 
| 5772 |  |  * | 
| 5773 |  |  * @deprecated Internal function, don't use. | 
| 5774 |  |  * | 
| 5775 |  |  *     [59] Enumeration ::= '(' S? Nmtoken (S? '|' S? Nmtoken)* S? ')' | 
| 5776 |  |  * | 
| 5777 |  |  * [ VC: Enumeration ] | 
| 5778 |  |  * Values of this type must match one of the Nmtoken tokens in | 
| 5779 |  |  * the declaration | 
| 5780 |  |  * | 
| 5781 |  |  * @param ctxt  an XML parser context | 
| 5782 |  |  * @returns the enumeration attribute tree built while parsing | 
| 5783 |  |  */ | 
| 5784 |  |  | 
| 5785 |  | xmlEnumeration * | 
| 5786 | 3.44k | xmlParseEnumerationType(xmlParserCtxt *ctxt) { | 
| 5787 | 3.44k |     xmlChar *name; | 
| 5788 | 3.44k |     xmlEnumerationPtr ret = NULL, last = NULL, cur, tmp; | 
| 5789 |  |  | 
| 5790 | 3.44k |     if (RAW != '(') { | 
| 5791 | 499 |   xmlFatalErr(ctxt, XML_ERR_ATTLIST_NOT_STARTED, NULL); | 
| 5792 | 499 |   return(NULL); | 
| 5793 | 499 |     } | 
| 5794 | 5.28k |     do { | 
| 5795 | 5.28k |         NEXT; | 
| 5796 | 5.28k |   SKIP_BLANKS_PE; | 
| 5797 | 5.28k |         name = xmlParseNmtoken(ctxt); | 
| 5798 | 5.28k |   if (name == NULL) { | 
| 5799 | 319 |       xmlFatalErr(ctxt, XML_ERR_NMTOKEN_REQUIRED, NULL); | 
| 5800 | 319 |       return(ret); | 
| 5801 | 319 |   } | 
| 5802 | 4.96k |         tmp = NULL; | 
| 5803 | 4.96k | #ifdef LIBXML_VALID_ENABLED | 
| 5804 | 4.96k |         if (ctxt->validate) { | 
| 5805 | 0 |             tmp = ret; | 
| 5806 | 0 |             while (tmp != NULL) { | 
| 5807 | 0 |                 if (xmlStrEqual(name, tmp->name)) { | 
| 5808 | 0 |                     xmlValidityError(ctxt, XML_DTD_DUP_TOKEN, | 
| 5809 | 0 |               "standalone: attribute enumeration value token %s duplicated\n", | 
| 5810 | 0 |                                      name, NULL); | 
| 5811 | 0 |                     if (!xmlDictOwns(ctxt->dict, name)) | 
| 5812 | 0 |                         xmlFree(name); | 
| 5813 | 0 |                     break; | 
| 5814 | 0 |                 } | 
| 5815 | 0 |                 tmp = tmp->next; | 
| 5816 | 0 |             } | 
| 5817 | 0 |         } | 
| 5818 | 4.96k | #endif /* LIBXML_VALID_ENABLED */ | 
| 5819 | 4.96k |   if (tmp == NULL) { | 
| 5820 | 4.96k |       cur = xmlCreateEnumeration(name); | 
| 5821 | 4.96k |       if (!xmlDictOwns(ctxt->dict, name)) | 
| 5822 | 4.96k |     xmlFree(name); | 
| 5823 | 4.96k |       if (cur == NULL) { | 
| 5824 | 0 |                 xmlErrMemory(ctxt); | 
| 5825 | 0 |                 xmlFreeEnumeration(ret); | 
| 5826 | 0 |                 return(NULL); | 
| 5827 | 0 |             } | 
| 5828 | 4.96k |       if (last == NULL) ret = last = cur; | 
| 5829 | 2.06k |       else { | 
| 5830 | 2.06k |     last->next = cur; | 
| 5831 | 2.06k |     last = cur; | 
| 5832 | 2.06k |       } | 
| 5833 | 4.96k |   } | 
| 5834 | 4.96k |   SKIP_BLANKS_PE; | 
| 5835 | 4.96k |     } while (RAW == '|'); | 
| 5836 | 2.62k |     if (RAW != ')') { | 
| 5837 | 774 |   xmlFatalErr(ctxt, XML_ERR_ATTLIST_NOT_FINISHED, NULL); | 
| 5838 | 774 |   return(ret); | 
| 5839 | 774 |     } | 
| 5840 | 1.84k |     NEXT; | 
| 5841 | 1.84k |     return(ret); | 
| 5842 | 2.62k | } | 
| 5843 |  |  | 
| 5844 |  | /** | 
| 5845 |  |  * Parse an Enumerated attribute type. | 
| 5846 |  |  * | 
| 5847 |  |  * @deprecated Internal function, don't use. | 
| 5848 |  |  * | 
| 5849 |  |  *     [57] EnumeratedType ::= NotationType | Enumeration | 
| 5850 |  |  * | 
| 5851 |  |  *     [58] NotationType ::= 'NOTATION' S '(' S? Name (S? '|' S? Name)* S? ')' | 
| 5852 |  |  * | 
| 5853 |  |  * @param ctxt  an XML parser context | 
| 5854 |  |  * @param tree  the enumeration tree built while parsing | 
| 5855 |  |  * @returns XML_ATTRIBUTE_ENUMERATION or XML_ATTRIBUTE_NOTATION | 
| 5856 |  |  */ | 
| 5857 |  |  | 
| 5858 |  | int | 
| 5859 | 4.74k | xmlParseEnumeratedType(xmlParserCtxt *ctxt, xmlEnumeration **tree) { | 
| 5860 | 4.74k |     if (CMP8(CUR_PTR, 'N', 'O', 'T', 'A', 'T', 'I', 'O', 'N')) { | 
| 5861 | 1.30k |   SKIP(8); | 
| 5862 | 1.30k |   if (SKIP_BLANKS_PE == 0) { | 
| 5863 | 195 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5864 | 195 |          "Space required after 'NOTATION'\n"); | 
| 5865 | 195 |       return(0); | 
| 5866 | 195 |   } | 
| 5867 | 1.11k |   *tree = xmlParseNotationType(ctxt); | 
| 5868 | 1.11k |   if (*tree == NULL) return(0); | 
| 5869 | 327 |   return(XML_ATTRIBUTE_NOTATION); | 
| 5870 | 1.11k |     } | 
| 5871 | 3.44k |     *tree = xmlParseEnumerationType(ctxt); | 
| 5872 | 3.44k |     if (*tree == NULL) return(0); | 
| 5873 | 2.89k |     return(XML_ATTRIBUTE_ENUMERATION); | 
| 5874 | 3.44k | } | 
| 5875 |  |  | 
| 5876 |  | /** | 
| 5877 |  |  * Parse the Attribute list def for an element | 
| 5878 |  |  * | 
| 5879 |  |  * @deprecated Internal function, don't use. | 
| 5880 |  |  * | 
| 5881 |  |  *     [54] AttType ::= StringType | TokenizedType | EnumeratedType | 
| 5882 |  |  * | 
| 5883 |  |  *     [55] StringType ::= 'CDATA' | 
| 5884 |  |  * | 
| 5885 |  |  *     [56] TokenizedType ::= 'ID' | 'IDREF' | 'IDREFS' | 'ENTITY' | | 
| 5886 |  |  *                            'ENTITIES' | 'NMTOKEN' | 'NMTOKENS' | 
| 5887 |  |  * | 
| 5888 |  |  * Validity constraints for attribute values syntax are checked in | 
| 5889 |  |  * #xmlValidateAttributeValue | 
| 5890 |  |  * | 
| 5891 |  |  * [ VC: ID ] | 
| 5892 |  |  * Values of type ID must match the Name production. A name must not | 
| 5893 |  |  * appear more than once in an XML document as a value of this type; | 
| 5894 |  |  * i.e., ID values must uniquely identify the elements which bear them. | 
| 5895 |  |  * | 
| 5896 |  |  * [ VC: One ID per Element Type ] | 
| 5897 |  |  * No element type may have more than one ID attribute specified. | 
| 5898 |  |  * | 
| 5899 |  |  * [ VC: ID Attribute Default ] | 
| 5900 |  |  * An ID attribute must have a declared default of \#IMPLIED or \#REQUIRED. | 
| 5901 |  |  * | 
| 5902 |  |  * [ VC: IDREF ] | 
| 5903 |  |  * Values of type IDREF must match the Name production, and values | 
| 5904 |  |  * of type IDREFS must match Names; each IDREF Name must match the value | 
| 5905 |  |  * of an ID attribute on some element in the XML document; i.e. IDREF | 
| 5906 |  |  * values must match the value of some ID attribute. | 
| 5907 |  |  * | 
| 5908 |  |  * [ VC: Entity Name ] | 
| 5909 |  |  * Values of type ENTITY must match the Name production, values | 
| 5910 |  |  * of type ENTITIES must match Names; each Entity Name must match the | 
| 5911 |  |  * name of an unparsed entity declared in the DTD. | 
| 5912 |  |  * | 
| 5913 |  |  * [ VC: Name Token ] | 
| 5914 |  |  * Values of type NMTOKEN must match the Nmtoken production; values | 
| 5915 |  |  * of type NMTOKENS must match Nmtokens. | 
| 5916 |  |  * | 
| 5917 |  |  * @param ctxt  an XML parser context | 
| 5918 |  |  * @param tree  the enumeration tree built while parsing | 
| 5919 |  |  * @returns the attribute type | 
| 5920 |  |  */ | 
| 5921 |  | int | 
| 5922 | 25.9k | xmlParseAttributeType(xmlParserCtxt *ctxt, xmlEnumeration **tree) { | 
| 5923 | 25.9k |     if (CMP5(CUR_PTR, 'C', 'D', 'A', 'T', 'A')) { | 
| 5924 | 1.67k |   SKIP(5); | 
| 5925 | 1.67k |   return(XML_ATTRIBUTE_CDATA); | 
| 5926 | 24.2k |      } else if (CMP6(CUR_PTR, 'I', 'D', 'R', 'E', 'F', 'S')) { | 
| 5927 | 439 |   SKIP(6); | 
| 5928 | 439 |   return(XML_ATTRIBUTE_IDREFS); | 
| 5929 | 23.8k |      } else if (CMP5(CUR_PTR, 'I', 'D', 'R', 'E', 'F')) { | 
| 5930 | 1.56k |   SKIP(5); | 
| 5931 | 1.56k |   return(XML_ATTRIBUTE_IDREF); | 
| 5932 | 22.2k |      } else if ((RAW == 'I') && (NXT(1) == 'D')) { | 
| 5933 | 13.9k |         SKIP(2); | 
| 5934 | 13.9k |   return(XML_ATTRIBUTE_ID); | 
| 5935 | 13.9k |      } else if (CMP6(CUR_PTR, 'E', 'N', 'T', 'I', 'T', 'Y')) { | 
| 5936 | 428 |   SKIP(6); | 
| 5937 | 428 |   return(XML_ATTRIBUTE_ENTITY); | 
| 5938 | 7.88k |      } else if (CMP8(CUR_PTR, 'E', 'N', 'T', 'I', 'T', 'I', 'E', 'S')) { | 
| 5939 | 2.34k |   SKIP(8); | 
| 5940 | 2.34k |   return(XML_ATTRIBUTE_ENTITIES); | 
| 5941 | 5.54k |      } else if (CMP8(CUR_PTR, 'N', 'M', 'T', 'O', 'K', 'E', 'N', 'S')) { | 
| 5942 | 407 |   SKIP(8); | 
| 5943 | 407 |   return(XML_ATTRIBUTE_NMTOKENS); | 
| 5944 | 5.13k |      } else if (CMP7(CUR_PTR, 'N', 'M', 'T', 'O', 'K', 'E', 'N')) { | 
| 5945 | 388 |   SKIP(7); | 
| 5946 | 388 |   return(XML_ATTRIBUTE_NMTOKEN); | 
| 5947 | 388 |      } | 
| 5948 | 4.74k |      return(xmlParseEnumeratedType(ctxt, tree)); | 
| 5949 | 25.9k | } | 
| 5950 |  |  | 
| 5951 |  | /** | 
| 5952 |  |  * Parse an attribute list declaration for an element. Always consumes '<!'. | 
| 5953 |  |  * | 
| 5954 |  |  * @deprecated Internal function, don't use. | 
| 5955 |  |  * | 
| 5956 |  |  *     [52] AttlistDecl ::= '<!ATTLIST' S Name AttDef* S? '>' | 
| 5957 |  |  * | 
| 5958 |  |  *     [53] AttDef ::= S Name S AttType S DefaultDecl | 
| 5959 |  |  * @param ctxt  an XML parser context | 
| 5960 |  |  */ | 
| 5961 |  | void | 
| 5962 | 14.0k | xmlParseAttributeListDecl(xmlParserCtxt *ctxt) { | 
| 5963 | 14.0k |     const xmlChar *elemName; | 
| 5964 | 14.0k |     const xmlChar *attrName; | 
| 5965 | 14.0k |     xmlEnumerationPtr tree; | 
| 5966 |  |  | 
| 5967 | 14.0k |     if ((CUR != '<') || (NXT(1) != '!')) | 
| 5968 | 0 |         return; | 
| 5969 | 14.0k |     SKIP(2); | 
| 5970 |  |  | 
| 5971 | 14.0k |     if (CMP7(CUR_PTR, 'A', 'T', 'T', 'L', 'I', 'S', 'T')) { | 
| 5972 | 13.9k | #ifdef LIBXML_VALID_ENABLED | 
| 5973 | 13.9k |   int oldInputNr = ctxt->inputNr; | 
| 5974 | 13.9k | #endif | 
| 5975 |  |  | 
| 5976 | 13.9k |   SKIP(7); | 
| 5977 | 13.9k |   if (SKIP_BLANKS_PE == 0) { | 
| 5978 | 5.27k |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 5979 | 5.27k |                      "Space required after '<!ATTLIST'\n"); | 
| 5980 | 5.27k |   } | 
| 5981 | 13.9k |         elemName = xmlParseName(ctxt); | 
| 5982 | 13.9k |   if (elemName == NULL) { | 
| 5983 | 726 |       xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 5984 | 726 |          "ATTLIST: no name for Element\n"); | 
| 5985 | 726 |       return; | 
| 5986 | 726 |   } | 
| 5987 | 13.2k |   SKIP_BLANKS_PE; | 
| 5988 | 13.2k |   GROW; | 
| 5989 | 30.9k |   while ((RAW != '>') && (PARSER_STOPPED(ctxt) == 0)) { | 
| 5990 | 27.9k |       int type; | 
| 5991 | 27.9k |       int def; | 
| 5992 | 27.9k |       xmlChar *defaultValue = NULL; | 
| 5993 |  |  | 
| 5994 | 27.9k |       GROW; | 
| 5995 | 27.9k |             tree = NULL; | 
| 5996 | 27.9k |       attrName = xmlParseName(ctxt); | 
| 5997 | 27.9k |       if (attrName == NULL) { | 
| 5998 | 1.13k |     xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 5999 | 1.13k |              "ATTLIST: no name for Attribute\n"); | 
| 6000 | 1.13k |     break; | 
| 6001 | 1.13k |       } | 
| 6002 | 26.8k |       GROW; | 
| 6003 | 26.8k |       if (SKIP_BLANKS_PE == 0) { | 
| 6004 | 931 |     xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 6005 | 931 |             "Space required after the attribute name\n"); | 
| 6006 | 931 |     break; | 
| 6007 | 931 |       } | 
| 6008 |  |  | 
| 6009 | 25.9k |       type = xmlParseAttributeType(ctxt, &tree); | 
| 6010 | 25.9k |       if (type <= 0) { | 
| 6011 | 1.52k |           break; | 
| 6012 | 1.52k |       } | 
| 6013 |  |  | 
| 6014 | 24.4k |       GROW; | 
| 6015 | 24.4k |       if (SKIP_BLANKS_PE == 0) { | 
| 6016 | 2.88k |     xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 6017 | 2.88k |              "Space required after the attribute type\n"); | 
| 6018 | 2.88k |           if (tree != NULL) | 
| 6019 | 1.08k |         xmlFreeEnumeration(tree); | 
| 6020 | 2.88k |     break; | 
| 6021 | 2.88k |       } | 
| 6022 |  |  | 
| 6023 | 21.5k |       def = xmlParseDefaultDecl(ctxt, &defaultValue); | 
| 6024 | 21.5k |       if (def <= 0) { | 
| 6025 | 0 |                 if (defaultValue != NULL) | 
| 6026 | 0 |         xmlFree(defaultValue); | 
| 6027 | 0 |           if (tree != NULL) | 
| 6028 | 0 |         xmlFreeEnumeration(tree); | 
| 6029 | 0 |           break; | 
| 6030 | 0 |       } | 
| 6031 | 21.5k |       if ((type != XML_ATTRIBUTE_CDATA) && (defaultValue != NULL)) | 
| 6032 | 15.1k |           xmlAttrNormalizeSpace(defaultValue, defaultValue); | 
| 6033 |  |  | 
| 6034 | 21.5k |       GROW; | 
| 6035 | 21.5k |             if (RAW != '>') { | 
| 6036 | 18.7k |     if (SKIP_BLANKS_PE == 0) { | 
| 6037 | 3.79k |         xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 6038 | 3.79k |       "Space required after the attribute default value\n"); | 
| 6039 | 3.79k |         if (defaultValue != NULL) | 
| 6040 | 354 |       xmlFree(defaultValue); | 
| 6041 | 3.79k |         if (tree != NULL) | 
| 6042 | 323 |       xmlFreeEnumeration(tree); | 
| 6043 | 3.79k |         break; | 
| 6044 | 3.79k |     } | 
| 6045 | 18.7k |       } | 
| 6046 | 17.7k |       if ((ctxt->sax != NULL) && (!ctxt->disableSAX) && | 
| 6047 | 9.55k |     (ctxt->sax->attributeDecl != NULL)) | 
| 6048 | 9.55k |     ctxt->sax->attributeDecl(ctxt->userData, elemName, attrName, | 
| 6049 | 9.55k |                           type, def, defaultValue, tree); | 
| 6050 | 8.16k |       else if (tree != NULL) | 
| 6051 | 1.18k |     xmlFreeEnumeration(tree); | 
| 6052 |  |  | 
| 6053 | 17.7k |       if ((ctxt->sax2) && (defaultValue != NULL) && | 
| 6054 | 16.1k |           (def != XML_ATTRIBUTE_IMPLIED) && | 
| 6055 | 16.1k |     (def != XML_ATTRIBUTE_REQUIRED)) { | 
| 6056 | 16.1k |     xmlAddDefAttrs(ctxt, elemName, attrName, defaultValue); | 
| 6057 | 16.1k |       } | 
| 6058 | 17.7k |       if (ctxt->sax2) { | 
| 6059 | 17.7k |     xmlAddSpecialAttr(ctxt, elemName, attrName, type); | 
| 6060 | 17.7k |       } | 
| 6061 | 17.7k |       if (defaultValue != NULL) | 
| 6062 | 16.1k |           xmlFree(defaultValue); | 
| 6063 | 17.7k |       GROW; | 
| 6064 | 17.7k |   } | 
| 6065 | 13.2k |   if (RAW == '>') { | 
| 6066 | 3.31k | #ifdef LIBXML_VALID_ENABLED | 
| 6067 | 3.31k |       if ((ctxt->validate) && (ctxt->inputNr > oldInputNr)) { | 
| 6068 | 0 |     xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6069 | 0 |                                  "Attribute list declaration doesn't start and" | 
| 6070 | 0 |                                  " stop in the same entity\n", | 
| 6071 | 0 |                                  NULL, NULL); | 
| 6072 | 0 |       } | 
| 6073 | 3.31k | #endif | 
| 6074 | 3.31k |       NEXT; | 
| 6075 | 3.31k |   } | 
| 6076 | 13.2k |     } | 
| 6077 | 14.0k | } | 
| 6078 |  |  | 
| 6079 |  | /** | 
| 6080 |  |  * Handle PEs and check that we don't pop the entity that started | 
| 6081 |  |  * a balanced group. | 
| 6082 |  |  * | 
| 6083 |  |  * @param ctxt  parser context | 
| 6084 |  |  * @param openInputNr  input nr of the entity with opening '(' | 
| 6085 |  |  */ | 
| 6086 |  | static void | 
| 6087 | 104k | xmlSkipBlankCharsPEBalanced(xmlParserCtxt *ctxt, int openInputNr) { | 
| 6088 | 104k |     SKIP_BLANKS; | 
| 6089 | 104k |     GROW; | 
| 6090 |  |  | 
| 6091 | 104k |     (void) openInputNr; | 
| 6092 |  |  | 
| 6093 | 104k |     if (!PARSER_EXTERNAL(ctxt) && !PARSER_IN_PE(ctxt)) | 
| 6094 | 38.9k |         return; | 
| 6095 |  |  | 
| 6096 | 71.9k |     while (!PARSER_STOPPED(ctxt)) { | 
| 6097 | 71.7k |         if (ctxt->input->cur >= ctxt->input->end) { | 
| 6098 | 2.96k | #ifdef LIBXML_VALID_ENABLED | 
| 6099 | 2.96k |             if ((ctxt->validate) && (ctxt->inputNr <= openInputNr)) { | 
| 6100 | 0 |                 xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6101 | 0 |                                  "Element content declaration doesn't start " | 
| 6102 | 0 |                                  "and stop in the same entity\n", | 
| 6103 | 0 |                                  NULL, NULL); | 
| 6104 | 0 |             } | 
| 6105 | 2.96k | #endif | 
| 6106 | 2.96k |             if (PARSER_IN_PE(ctxt)) | 
| 6107 | 2.88k |                 xmlPopPE(ctxt); | 
| 6108 | 81 |             else | 
| 6109 | 81 |                 break; | 
| 6110 | 68.8k |         } else if (RAW == '%') { | 
| 6111 | 3.77k |             xmlParsePERefInternal(ctxt, 0); | 
| 6112 | 65.0k |         } else { | 
| 6113 | 65.0k |             break; | 
| 6114 | 65.0k |         } | 
| 6115 |  |  | 
| 6116 | 6.66k |         SKIP_BLANKS; | 
| 6117 | 6.66k |         GROW; | 
| 6118 | 6.66k |     } | 
| 6119 | 65.3k | } | 
| 6120 |  |  | 
| 6121 |  | /** | 
| 6122 |  |  * Parse the declaration for a Mixed Element content | 
| 6123 |  |  * The leading '(' and spaces have been skipped in #xmlParseElementContentDecl | 
| 6124 |  |  * | 
| 6125 |  |  * @deprecated Internal function, don't use. | 
| 6126 |  |  * | 
| 6127 |  |  *     [51] Mixed ::= '(' S? '#PCDATA' (S? '|' S? Name)* S? ')*' | | 
| 6128 |  |  *                    '(' S? '#PCDATA' S? ')' | 
| 6129 |  |  * | 
| 6130 |  |  * [ VC: Proper Group/PE Nesting ] applies to [51] too (see [49]) | 
| 6131 |  |  * | 
| 6132 |  |  * [ VC: No Duplicate Types ] | 
| 6133 |  |  * The same name must not appear more than once in a single | 
| 6134 |  |  * mixed-content declaration. | 
| 6135 |  |  * | 
| 6136 |  |  * @param ctxt  an XML parser context | 
| 6137 |  |  * @param openInputNr  the input used for the current entity, needed for | 
| 6138 |  |  * boundary checks | 
| 6139 |  |  * @returns the list of the xmlElementContent describing the element choices | 
| 6140 |  |  */ | 
| 6141 |  | xmlElementContent * | 
| 6142 | 1.63k | xmlParseElementMixedContentDecl(xmlParserCtxt *ctxt, int openInputNr) { | 
| 6143 | 1.63k |     xmlElementContentPtr ret = NULL, cur = NULL, n; | 
| 6144 | 1.63k |     const xmlChar *elem = NULL; | 
| 6145 |  |  | 
| 6146 | 1.63k |     GROW; | 
| 6147 | 1.63k |     if (CMP7(CUR_PTR, '#', 'P', 'C', 'D', 'A', 'T', 'A')) { | 
| 6148 | 1.63k |   SKIP(7); | 
| 6149 | 1.63k |         xmlSkipBlankCharsPEBalanced(ctxt, openInputNr); | 
| 6150 | 1.63k |   if (RAW == ')') { | 
| 6151 | 545 | #ifdef LIBXML_VALID_ENABLED | 
| 6152 | 545 |       if ((ctxt->validate) && (ctxt->inputNr > openInputNr)) { | 
| 6153 | 0 |     xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6154 | 0 |                                  "Element content declaration doesn't start " | 
| 6155 | 0 |                                  "and stop in the same entity\n", | 
| 6156 | 0 |                                  NULL, NULL); | 
| 6157 | 0 |       } | 
| 6158 | 545 | #endif | 
| 6159 | 545 |       NEXT; | 
| 6160 | 545 |       ret = xmlNewDocElementContent(ctxt->myDoc, NULL, XML_ELEMENT_CONTENT_PCDATA); | 
| 6161 | 545 |       if (ret == NULL) | 
| 6162 | 0 |                 goto mem_error; | 
| 6163 | 545 |       if (RAW == '*') { | 
| 6164 | 194 |     ret->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6165 | 194 |     NEXT; | 
| 6166 | 194 |       } | 
| 6167 | 545 |       return(ret); | 
| 6168 | 545 |   } | 
| 6169 | 1.08k |   if ((RAW == '(') || (RAW == '|')) { | 
| 6170 | 652 |       ret = cur = xmlNewDocElementContent(ctxt->myDoc, NULL, XML_ELEMENT_CONTENT_PCDATA); | 
| 6171 | 652 |       if (ret == NULL) | 
| 6172 | 0 |                 goto mem_error; | 
| 6173 | 652 |   } | 
| 6174 | 2.01k |   while ((RAW == '|') && (PARSER_STOPPED(ctxt) == 0)) { | 
| 6175 | 1.18k |       NEXT; | 
| 6176 | 1.18k |             n = xmlNewDocElementContent(ctxt->myDoc, NULL, XML_ELEMENT_CONTENT_OR); | 
| 6177 | 1.18k |             if (n == NULL) | 
| 6178 | 0 |                 goto mem_error; | 
| 6179 | 1.18k |       if (elem == NULL) { | 
| 6180 | 651 |     n->c1 = cur; | 
| 6181 | 651 |     if (cur != NULL) | 
| 6182 | 651 |         cur->parent = n; | 
| 6183 | 651 |     ret = cur = n; | 
| 6184 | 651 |       } else { | 
| 6185 | 537 |           cur->c2 = n; | 
| 6186 | 537 |     n->parent = cur; | 
| 6187 | 537 |     n->c1 = xmlNewDocElementContent(ctxt->myDoc, elem, XML_ELEMENT_CONTENT_ELEMENT); | 
| 6188 | 537 |                 if (n->c1 == NULL) | 
| 6189 | 0 |                     goto mem_error; | 
| 6190 | 537 |     n->c1->parent = n; | 
| 6191 | 537 |     cur = n; | 
| 6192 | 537 |       } | 
| 6193 | 1.18k |             xmlSkipBlankCharsPEBalanced(ctxt, openInputNr); | 
| 6194 | 1.18k |       elem = xmlParseName(ctxt); | 
| 6195 | 1.18k |       if (elem == NULL) { | 
| 6196 | 254 |     xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 6197 | 254 |       "xmlParseElementMixedContentDecl : Name expected\n"); | 
| 6198 | 254 |     xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6199 | 254 |     return(NULL); | 
| 6200 | 254 |       } | 
| 6201 | 934 |             xmlSkipBlankCharsPEBalanced(ctxt, openInputNr); | 
| 6202 | 934 |   } | 
| 6203 | 831 |   if ((RAW == ')') && (NXT(1) == '*')) { | 
| 6204 | 392 |       if (elem != NULL) { | 
| 6205 | 392 |     cur->c2 = xmlNewDocElementContent(ctxt->myDoc, elem, | 
| 6206 | 392 |                                    XML_ELEMENT_CONTENT_ELEMENT); | 
| 6207 | 392 |     if (cur->c2 == NULL) | 
| 6208 | 0 |                     goto mem_error; | 
| 6209 | 392 |     cur->c2->parent = cur; | 
| 6210 | 392 |             } | 
| 6211 | 392 |             if (ret != NULL) | 
| 6212 | 392 |                 ret->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6213 | 392 | #ifdef LIBXML_VALID_ENABLED | 
| 6214 | 392 |       if ((ctxt->validate) && (ctxt->inputNr > openInputNr)) { | 
| 6215 | 0 |     xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6216 | 0 |                                  "Element content declaration doesn't start " | 
| 6217 | 0 |                                  "and stop in the same entity\n", | 
| 6218 | 0 |                                  NULL, NULL); | 
| 6219 | 0 |       } | 
| 6220 | 392 | #endif | 
| 6221 | 392 |       SKIP(2); | 
| 6222 | 439 |   } else { | 
| 6223 | 439 |       xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6224 | 439 |       xmlFatalErr(ctxt, XML_ERR_MIXED_NOT_STARTED, NULL); | 
| 6225 | 439 |       return(NULL); | 
| 6226 | 439 |   } | 
| 6227 |  |  | 
| 6228 | 831 |     } else { | 
| 6229 | 0 |   xmlFatalErr(ctxt, XML_ERR_PCDATA_REQUIRED, NULL); | 
| 6230 | 0 |     } | 
| 6231 | 392 |     return(ret); | 
| 6232 |  |  | 
| 6233 | 0 | mem_error: | 
| 6234 | 0 |     xmlErrMemory(ctxt); | 
| 6235 | 0 |     xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6236 | 0 |     return(NULL); | 
| 6237 | 1.63k | } | 
| 6238 |  |  | 
| 6239 |  | /** | 
| 6240 |  |  * Parse the declaration for a Mixed Element content | 
| 6241 |  |  * The leading '(' and spaces have been skipped in #xmlParseElementContentDecl | 
| 6242 |  |  * | 
| 6243 |  |  *     [47] children ::= (choice | seq) ('?' | '*' | '+')? | 
| 6244 |  |  * | 
| 6245 |  |  *     [48] cp ::= (Name | choice | seq) ('?' | '*' | '+')? | 
| 6246 |  |  * | 
| 6247 |  |  *     [49] choice ::= '(' S? cp ( S? '|' S? cp )* S? ')' | 
| 6248 |  |  * | 
| 6249 |  |  *     [50] seq ::= '(' S? cp ( S? ',' S? cp )* S? ')' | 
| 6250 |  |  * | 
| 6251 |  |  * [ VC: Proper Group/PE Nesting ] applies to [49] and [50] | 
| 6252 |  |  * TODO Parameter-entity replacement text must be properly nested | 
| 6253 |  |  *  with parenthesized groups. That is to say, if either of the | 
| 6254 |  |  *  opening or closing parentheses in a choice, seq, or Mixed | 
| 6255 |  |  *  construct is contained in the replacement text for a parameter | 
| 6256 |  |  *  entity, both must be contained in the same replacement text. For | 
| 6257 |  |  *  interoperability, if a parameter-entity reference appears in a | 
| 6258 |  |  *  choice, seq, or Mixed construct, its replacement text should not | 
| 6259 |  |  *  be empty, and neither the first nor last non-blank character of | 
| 6260 |  |  *  the replacement text should be a connector (| or ,). | 
| 6261 |  |  * | 
| 6262 |  |  * @param ctxt  an XML parser context | 
| 6263 |  |  * @param openInputNr  the input used for the current entity, needed for | 
| 6264 |  |  * boundary checks | 
| 6265 |  |  * @param depth  the level of recursion | 
| 6266 |  |  * @returns the tree of xmlElementContent describing the element | 
| 6267 |  |  *          hierarchy. | 
| 6268 |  |  */ | 
| 6269 |  | static xmlElementContentPtr | 
| 6270 |  | xmlParseElementChildrenContentDeclPriv(xmlParserCtxtPtr ctxt, int openInputNr, | 
| 6271 | 45.8k |                                        int depth) { | 
| 6272 | 45.8k |     int maxDepth = (ctxt->options & XML_PARSE_HUGE) ? 2048 : 256; | 
| 6273 | 45.8k |     xmlElementContentPtr ret = NULL, cur = NULL, last = NULL, op = NULL; | 
| 6274 | 45.8k |     const xmlChar *elem; | 
| 6275 | 45.8k |     xmlChar type = 0; | 
| 6276 |  |  | 
| 6277 | 45.8k |     if (depth > maxDepth) { | 
| 6278 | 1 |         xmlFatalErrMsgInt(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 6279 | 1 |                 "xmlParseElementChildrenContentDecl : depth %d too deep, " | 
| 6280 | 1 |                 "use XML_PARSE_HUGE\n", depth); | 
| 6281 | 1 |   return(NULL); | 
| 6282 | 1 |     } | 
| 6283 | 45.8k |     xmlSkipBlankCharsPEBalanced(ctxt, openInputNr); | 
| 6284 | 45.8k |     if (RAW == '(') { | 
| 6285 | 32.2k |         int newInputNr = ctxt->inputNr; | 
| 6286 |  |  | 
| 6287 |  |         /* Recurse on first child */ | 
| 6288 | 32.2k |   NEXT; | 
| 6289 | 32.2k |         cur = ret = xmlParseElementChildrenContentDeclPriv(ctxt, newInputNr, | 
| 6290 | 32.2k |                                                            depth + 1); | 
| 6291 | 32.2k |         if (cur == NULL) | 
| 6292 | 28.8k |             return(NULL); | 
| 6293 | 32.2k |     } else { | 
| 6294 | 13.5k |   elem = xmlParseName(ctxt); | 
| 6295 | 13.5k |   if (elem == NULL) { | 
| 6296 | 1.08k |       xmlFatalErr(ctxt, XML_ERR_ELEMCONTENT_NOT_STARTED, NULL); | 
| 6297 | 1.08k |       return(NULL); | 
| 6298 | 1.08k |   } | 
| 6299 | 12.4k |         cur = ret = xmlNewDocElementContent(ctxt->myDoc, elem, XML_ELEMENT_CONTENT_ELEMENT); | 
| 6300 | 12.4k |   if (cur == NULL) { | 
| 6301 | 0 |       xmlErrMemory(ctxt); | 
| 6302 | 0 |       return(NULL); | 
| 6303 | 0 |   } | 
| 6304 | 12.4k |   GROW; | 
| 6305 | 12.4k |   if (RAW == '?') { | 
| 6306 | 3.43k |       cur->ocur = XML_ELEMENT_CONTENT_OPT; | 
| 6307 | 3.43k |       NEXT; | 
| 6308 | 9.02k |   } else if (RAW == '*') { | 
| 6309 | 462 |       cur->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6310 | 462 |       NEXT; | 
| 6311 | 8.56k |   } else if (RAW == '+') { | 
| 6312 | 2.65k |       cur->ocur = XML_ELEMENT_CONTENT_PLUS; | 
| 6313 | 2.65k |       NEXT; | 
| 6314 | 5.90k |   } else { | 
| 6315 | 5.90k |       cur->ocur = XML_ELEMENT_CONTENT_ONCE; | 
| 6316 | 5.90k |   } | 
| 6317 | 12.4k |   GROW; | 
| 6318 | 12.4k |     } | 
| 6319 | 30.9k |     while (!PARSER_STOPPED(ctxt)) { | 
| 6320 | 30.6k |         xmlSkipBlankCharsPEBalanced(ctxt, openInputNr); | 
| 6321 | 30.6k |         if (RAW == ')') | 
| 6322 | 10.7k |             break; | 
| 6323 |  |         /* | 
| 6324 |  |    * Each loop we parse one separator and one element. | 
| 6325 |  |    */ | 
| 6326 | 19.9k |         if (RAW == ',') { | 
| 6327 | 799 |       if (type == 0) type = CUR; | 
| 6328 |  |  | 
| 6329 |  |       /* | 
| 6330 |  |        * Detect "Name | Name , Name" error | 
| 6331 |  |        */ | 
| 6332 | 322 |       else if (type != CUR) { | 
| 6333 | 1 |     xmlFatalErrMsgInt(ctxt, XML_ERR_SEPARATOR_REQUIRED, | 
| 6334 | 1 |         "xmlParseElementChildrenContentDecl : '%c' expected\n", | 
| 6335 | 1 |                       type); | 
| 6336 | 1 |     if ((last != NULL) && (last != ret)) | 
| 6337 | 1 |         xmlFreeDocElementContent(ctxt->myDoc, last); | 
| 6338 | 1 |     if (ret != NULL) | 
| 6339 | 1 |         xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6340 | 1 |     return(NULL); | 
| 6341 | 1 |       } | 
| 6342 | 798 |       NEXT; | 
| 6343 |  |  | 
| 6344 | 798 |       op = xmlNewDocElementContent(ctxt->myDoc, NULL, XML_ELEMENT_CONTENT_SEQ); | 
| 6345 | 798 |       if (op == NULL) { | 
| 6346 | 0 |                 xmlErrMemory(ctxt); | 
| 6347 | 0 |     if ((last != NULL) && (last != ret)) | 
| 6348 | 0 |         xmlFreeDocElementContent(ctxt->myDoc, last); | 
| 6349 | 0 |           xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6350 | 0 |     return(NULL); | 
| 6351 | 0 |       } | 
| 6352 | 798 |       if (last == NULL) { | 
| 6353 | 477 |     op->c1 = ret; | 
| 6354 | 477 |     if (ret != NULL) | 
| 6355 | 477 |         ret->parent = op; | 
| 6356 | 477 |     ret = cur = op; | 
| 6357 | 477 |       } else { | 
| 6358 | 321 |           cur->c2 = op; | 
| 6359 | 321 |     if (op != NULL) | 
| 6360 | 321 |         op->parent = cur; | 
| 6361 | 321 |     op->c1 = last; | 
| 6362 | 321 |     if (last != NULL) | 
| 6363 | 321 |         last->parent = op; | 
| 6364 | 321 |     cur =op; | 
| 6365 | 321 |     last = NULL; | 
| 6366 | 321 |       } | 
| 6367 | 19.1k |   } else if (RAW == '|') { | 
| 6368 | 16.8k |       if (type == 0) type = CUR; | 
| 6369 |  |  | 
| 6370 |  |       /* | 
| 6371 |  |        * Detect "Name , Name | Name" error | 
| 6372 |  |        */ | 
| 6373 | 9.51k |       else if (type != CUR) { | 
| 6374 | 1 |     xmlFatalErrMsgInt(ctxt, XML_ERR_SEPARATOR_REQUIRED, | 
| 6375 | 1 |         "xmlParseElementChildrenContentDecl : '%c' expected\n", | 
| 6376 | 1 |           type); | 
| 6377 | 1 |     if ((last != NULL) && (last != ret)) | 
| 6378 | 1 |         xmlFreeDocElementContent(ctxt->myDoc, last); | 
| 6379 | 1 |     if (ret != NULL) | 
| 6380 | 1 |         xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6381 | 1 |     return(NULL); | 
| 6382 | 1 |       } | 
| 6383 | 16.8k |       NEXT; | 
| 6384 |  |  | 
| 6385 | 16.8k |       op = xmlNewDocElementContent(ctxt->myDoc, NULL, XML_ELEMENT_CONTENT_OR); | 
| 6386 | 16.8k |       if (op == NULL) { | 
| 6387 | 0 |                 xmlErrMemory(ctxt); | 
| 6388 | 0 |     if ((last != NULL) && (last != ret)) | 
| 6389 | 0 |         xmlFreeDocElementContent(ctxt->myDoc, last); | 
| 6390 | 0 |     if (ret != NULL) | 
| 6391 | 0 |         xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6392 | 0 |     return(NULL); | 
| 6393 | 0 |       } | 
| 6394 | 16.8k |       if (last == NULL) { | 
| 6395 | 7.37k |     op->c1 = ret; | 
| 6396 | 7.37k |     if (ret != NULL) | 
| 6397 | 7.37k |         ret->parent = op; | 
| 6398 | 7.37k |     ret = cur = op; | 
| 6399 | 9.51k |       } else { | 
| 6400 | 9.51k |           cur->c2 = op; | 
| 6401 | 9.51k |     if (op != NULL) | 
| 6402 | 9.51k |         op->parent = cur; | 
| 6403 | 9.51k |     op->c1 = last; | 
| 6404 | 9.51k |     if (last != NULL) | 
| 6405 | 9.51k |         last->parent = op; | 
| 6406 | 9.51k |     cur =op; | 
| 6407 | 9.51k |     last = NULL; | 
| 6408 | 9.51k |       } | 
| 6409 | 16.8k |   } else { | 
| 6410 | 2.27k |       xmlFatalErr(ctxt, XML_ERR_ELEMCONTENT_NOT_FINISHED, NULL); | 
| 6411 | 2.27k |       if ((last != NULL) && (last != ret)) | 
| 6412 | 1.31k |           xmlFreeDocElementContent(ctxt->myDoc, last); | 
| 6413 | 2.27k |       if (ret != NULL) | 
| 6414 | 2.27k |     xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6415 | 2.27k |       return(NULL); | 
| 6416 | 2.27k |   } | 
| 6417 | 17.6k |         xmlSkipBlankCharsPEBalanced(ctxt, openInputNr); | 
| 6418 | 17.6k |         if (RAW == '(') { | 
| 6419 | 8.81k |             int newInputNr = ctxt->inputNr; | 
| 6420 |  |  | 
| 6421 |  |       /* Recurse on second child */ | 
| 6422 | 8.81k |       NEXT; | 
| 6423 | 8.81k |       last = xmlParseElementChildrenContentDeclPriv(ctxt, newInputNr, | 
| 6424 | 8.81k |                                                           depth + 1); | 
| 6425 | 8.81k |             if (last == NULL) { | 
| 6426 | 2.11k |     if (ret != NULL) | 
| 6427 | 2.11k |         xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6428 | 2.11k |     return(NULL); | 
| 6429 | 2.11k |             } | 
| 6430 | 8.86k |   } else { | 
| 6431 | 8.86k |       elem = xmlParseName(ctxt); | 
| 6432 | 8.86k |       if (elem == NULL) { | 
| 6433 | 564 |     xmlFatalErr(ctxt, XML_ERR_ELEMCONTENT_NOT_STARTED, NULL); | 
| 6434 | 564 |     if (ret != NULL) | 
| 6435 | 564 |         xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6436 | 564 |     return(NULL); | 
| 6437 | 564 |       } | 
| 6438 | 8.30k |       last = xmlNewDocElementContent(ctxt->myDoc, elem, XML_ELEMENT_CONTENT_ELEMENT); | 
| 6439 | 8.30k |       if (last == NULL) { | 
| 6440 | 0 |                 xmlErrMemory(ctxt); | 
| 6441 | 0 |     if (ret != NULL) | 
| 6442 | 0 |         xmlFreeDocElementContent(ctxt->myDoc, ret); | 
| 6443 | 0 |     return(NULL); | 
| 6444 | 0 |       } | 
| 6445 | 8.30k |       if (RAW == '?') { | 
| 6446 | 2.94k |     last->ocur = XML_ELEMENT_CONTENT_OPT; | 
| 6447 | 2.94k |     NEXT; | 
| 6448 | 5.35k |       } else if (RAW == '*') { | 
| 6449 | 310 |     last->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6450 | 310 |     NEXT; | 
| 6451 | 5.04k |       } else if (RAW == '+') { | 
| 6452 | 269 |     last->ocur = XML_ELEMENT_CONTENT_PLUS; | 
| 6453 | 269 |     NEXT; | 
| 6454 | 4.77k |       } else { | 
| 6455 | 4.77k |     last->ocur = XML_ELEMENT_CONTENT_ONCE; | 
| 6456 | 4.77k |       } | 
| 6457 | 8.30k |   } | 
| 6458 | 17.6k |     } | 
| 6459 | 10.9k |     if ((cur != NULL) && (last != NULL)) { | 
| 6460 | 3.85k |         cur->c2 = last; | 
| 6461 | 3.85k |   if (last != NULL) | 
| 6462 | 3.85k |       last->parent = cur; | 
| 6463 | 3.85k |     } | 
| 6464 | 10.9k | #ifdef LIBXML_VALID_ENABLED | 
| 6465 | 10.9k |     if ((ctxt->validate) && (ctxt->inputNr > openInputNr)) { | 
| 6466 | 0 |         xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6467 | 0 |                          "Element content declaration doesn't start " | 
| 6468 | 0 |                          "and stop in the same entity\n", | 
| 6469 | 0 |                          NULL, NULL); | 
| 6470 | 0 |     } | 
| 6471 | 10.9k | #endif | 
| 6472 | 10.9k |     NEXT; | 
| 6473 | 10.9k |     if (RAW == '?') { | 
| 6474 | 3.18k |   if (ret != NULL) { | 
| 6475 | 3.18k |       if ((ret->ocur == XML_ELEMENT_CONTENT_PLUS) || | 
| 6476 | 981 |           (ret->ocur == XML_ELEMENT_CONTENT_MULT)) | 
| 6477 | 2.66k |           ret->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6478 | 519 |       else | 
| 6479 | 519 |           ret->ocur = XML_ELEMENT_CONTENT_OPT; | 
| 6480 | 3.18k |   } | 
| 6481 | 3.18k |   NEXT; | 
| 6482 | 7.77k |     } else if (RAW == '*') { | 
| 6483 | 812 |   if (ret != NULL) { | 
| 6484 | 812 |       ret->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6485 | 812 |       cur = ret; | 
| 6486 |  |       /* | 
| 6487 |  |        * Some normalization: | 
| 6488 |  |        * (a | b* | c?)* == (a | b | c)* | 
| 6489 |  |        */ | 
| 6490 | 3.68k |       while ((cur != NULL) && (cur->type == XML_ELEMENT_CONTENT_OR)) { | 
| 6491 | 2.87k |     if ((cur->c1 != NULL) && | 
| 6492 | 2.87k |               ((cur->c1->ocur == XML_ELEMENT_CONTENT_OPT) || | 
| 6493 | 2.21k |          (cur->c1->ocur == XML_ELEMENT_CONTENT_MULT))) | 
| 6494 | 1.30k |         cur->c1->ocur = XML_ELEMENT_CONTENT_ONCE; | 
| 6495 | 2.87k |     if ((cur->c2 != NULL) && | 
| 6496 | 2.87k |               ((cur->c2->ocur == XML_ELEMENT_CONTENT_OPT) || | 
| 6497 | 2.57k |          (cur->c2->ocur == XML_ELEMENT_CONTENT_MULT))) | 
| 6498 | 525 |         cur->c2->ocur = XML_ELEMENT_CONTENT_ONCE; | 
| 6499 | 2.87k |     cur = cur->c2; | 
| 6500 | 2.87k |       } | 
| 6501 | 812 |   } | 
| 6502 | 812 |   NEXT; | 
| 6503 | 6.96k |     } else if (RAW == '+') { | 
| 6504 | 3.04k |   if (ret != NULL) { | 
| 6505 | 3.04k |       int found = 0; | 
| 6506 |  |  | 
| 6507 | 3.04k |       if ((ret->ocur == XML_ELEMENT_CONTENT_OPT) || | 
| 6508 | 2.34k |           (ret->ocur == XML_ELEMENT_CONTENT_MULT)) | 
| 6509 | 1.29k |           ret->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6510 | 1.74k |       else | 
| 6511 | 1.74k |           ret->ocur = XML_ELEMENT_CONTENT_PLUS; | 
| 6512 |  |       /* | 
| 6513 |  |        * Some normalization: | 
| 6514 |  |        * (a | b*)+ == (a | b)* | 
| 6515 |  |        * (a | b?)+ == (a | b)* | 
| 6516 |  |        */ | 
| 6517 | 9.66k |       while ((cur != NULL) && (cur->type == XML_ELEMENT_CONTENT_OR)) { | 
| 6518 | 6.61k |     if ((cur->c1 != NULL) && | 
| 6519 | 6.61k |               ((cur->c1->ocur == XML_ELEMENT_CONTENT_OPT) || | 
| 6520 | 6.15k |          (cur->c1->ocur == XML_ELEMENT_CONTENT_MULT))) { | 
| 6521 | 754 |         cur->c1->ocur = XML_ELEMENT_CONTENT_ONCE; | 
| 6522 | 754 |         found = 1; | 
| 6523 | 754 |     } | 
| 6524 | 6.61k |     if ((cur->c2 != NULL) && | 
| 6525 | 6.61k |               ((cur->c2->ocur == XML_ELEMENT_CONTENT_OPT) || | 
| 6526 | 6.33k |          (cur->c2->ocur == XML_ELEMENT_CONTENT_MULT))) { | 
| 6527 | 1.46k |         cur->c2->ocur = XML_ELEMENT_CONTENT_ONCE; | 
| 6528 | 1.46k |         found = 1; | 
| 6529 | 1.46k |     } | 
| 6530 | 6.61k |     cur = cur->c2; | 
| 6531 | 6.61k |       } | 
| 6532 | 3.04k |       if (found) | 
| 6533 | 1.67k |     ret->ocur = XML_ELEMENT_CONTENT_MULT; | 
| 6534 | 3.04k |   } | 
| 6535 | 3.04k |   NEXT; | 
| 6536 | 3.04k |     } | 
| 6537 | 10.9k |     return(ret); | 
| 6538 | 15.9k | } | 
| 6539 |  |  | 
| 6540 |  | /** | 
| 6541 |  |  * Parse the declaration for a Mixed Element content | 
| 6542 |  |  * The leading '(' and spaces have been skipped in #xmlParseElementContentDecl | 
| 6543 |  |  * | 
| 6544 |  |  * @deprecated Internal function, don't use. | 
| 6545 |  |  * | 
| 6546 |  |  *     [47] children ::= (choice | seq) ('?' | '*' | '+')? | 
| 6547 |  |  * | 
| 6548 |  |  *     [48] cp ::= (Name | choice | seq) ('?' | '*' | '+')? | 
| 6549 |  |  * | 
| 6550 |  |  *     [49] choice ::= '(' S? cp ( S? '|' S? cp )* S? ')' | 
| 6551 |  |  * | 
| 6552 |  |  *     [50] seq ::= '(' S? cp ( S? ',' S? cp )* S? ')' | 
| 6553 |  |  * | 
| 6554 |  |  * [ VC: Proper Group/PE Nesting ] applies to [49] and [50] | 
| 6555 |  |  * TODO Parameter-entity replacement text must be properly nested | 
| 6556 |  |  *  with parenthesized groups. That is to say, if either of the | 
| 6557 |  |  *  opening or closing parentheses in a choice, seq, or Mixed | 
| 6558 |  |  *  construct is contained in the replacement text for a parameter | 
| 6559 |  |  *  entity, both must be contained in the same replacement text. For | 
| 6560 |  |  *  interoperability, if a parameter-entity reference appears in a | 
| 6561 |  |  *  choice, seq, or Mixed construct, its replacement text should not | 
| 6562 |  |  *  be empty, and neither the first nor last non-blank character of | 
| 6563 |  |  *  the replacement text should be a connector (| or ,). | 
| 6564 |  |  * | 
| 6565 |  |  * @param ctxt  an XML parser context | 
| 6566 |  |  * @param inputchk  the input used for the current entity, needed for boundary checks | 
| 6567 |  |  * @returns the tree of xmlElementContent describing the element | 
| 6568 |  |  *          hierarchy. | 
| 6569 |  |  */ | 
| 6570 |  | xmlElementContent * | 
| 6571 | 0 | xmlParseElementChildrenContentDecl(xmlParserCtxt *ctxt, int inputchk) { | 
| 6572 |  |     /* stub left for API/ABI compat */ | 
| 6573 | 0 |     return(xmlParseElementChildrenContentDeclPriv(ctxt, inputchk, 1)); | 
| 6574 | 0 | } | 
| 6575 |  |  | 
| 6576 |  | /** | 
| 6577 |  |  * Parse the declaration for an Element content either Mixed or Children, | 
| 6578 |  |  * the cases EMPTY and ANY are handled directly in #xmlParseElementDecl | 
| 6579 |  |  * | 
| 6580 |  |  * @deprecated Internal function, don't use. | 
| 6581 |  |  * | 
| 6582 |  |  *     [46] contentspec ::= 'EMPTY' | 'ANY' | Mixed | children | 
| 6583 |  |  * | 
| 6584 |  |  * @param ctxt  an XML parser context | 
| 6585 |  |  * @param name  the name of the element being defined. | 
| 6586 |  |  * @param result  the Element Content pointer will be stored here if any | 
| 6587 |  |  * @returns an xmlElementTypeVal value or -1 on error | 
| 6588 |  |  */ | 
| 6589 |  |  | 
| 6590 |  | int | 
| 6591 |  | xmlParseElementContentDecl(xmlParserCtxt *ctxt, const xmlChar *name, | 
| 6592 | 6.36k |                            xmlElementContent **result) { | 
| 6593 |  |  | 
| 6594 | 6.36k |     xmlElementContentPtr tree = NULL; | 
| 6595 | 6.36k |     int openInputNr = ctxt->inputNr; | 
| 6596 | 6.36k |     int res; | 
| 6597 |  |  | 
| 6598 | 6.36k |     *result = NULL; | 
| 6599 |  |  | 
| 6600 | 6.36k |     if (RAW != '(') { | 
| 6601 | 0 |   xmlFatalErrMsgStr(ctxt, XML_ERR_ELEMCONTENT_NOT_STARTED, | 
| 6602 | 0 |     "xmlParseElementContentDecl : %s '(' expected\n", name); | 
| 6603 | 0 |   return(-1); | 
| 6604 | 0 |     } | 
| 6605 | 6.36k |     NEXT; | 
| 6606 | 6.36k |     xmlSkipBlankCharsPEBalanced(ctxt, openInputNr); | 
| 6607 | 6.36k |     if (CMP7(CUR_PTR, '#', 'P', 'C', 'D', 'A', 'T', 'A')) { | 
| 6608 | 1.63k |         tree = xmlParseElementMixedContentDecl(ctxt, openInputNr); | 
| 6609 | 1.63k |   res = XML_ELEMENT_TYPE_MIXED; | 
| 6610 | 4.73k |     } else { | 
| 6611 | 4.73k |         tree = xmlParseElementChildrenContentDeclPriv(ctxt, openInputNr, 1); | 
| 6612 | 4.73k |   res = XML_ELEMENT_TYPE_ELEMENT; | 
| 6613 | 4.73k |     } | 
| 6614 | 6.36k |     if (tree == NULL) | 
| 6615 | 4.61k |         return(-1); | 
| 6616 | 1.74k |     SKIP_BLANKS_PE; | 
| 6617 | 1.74k |     *result = tree; | 
| 6618 | 1.74k |     return(res); | 
| 6619 | 6.36k | } | 
| 6620 |  |  | 
| 6621 |  | /** | 
| 6622 |  |  * Parse an element declaration. Always consumes '<!'. | 
| 6623 |  |  * | 
| 6624 |  |  * @deprecated Internal function, don't use. | 
| 6625 |  |  * | 
| 6626 |  |  *     [45] elementdecl ::= '<!ELEMENT' S Name S contentspec S? '>' | 
| 6627 |  |  * | 
| 6628 |  |  * [ VC: Unique Element Type Declaration ] | 
| 6629 |  |  * No element type may be declared more than once | 
| 6630 |  |  * | 
| 6631 |  |  * @param ctxt  an XML parser context | 
| 6632 |  |  * @returns the type of the element, or -1 in case of error | 
| 6633 |  |  */ | 
| 6634 |  | int | 
| 6635 | 9.09k | xmlParseElementDecl(xmlParserCtxt *ctxt) { | 
| 6636 | 9.09k |     const xmlChar *name; | 
| 6637 | 9.09k |     int ret = -1; | 
| 6638 | 9.09k |     xmlElementContentPtr content  = NULL; | 
| 6639 |  |  | 
| 6640 | 9.09k |     if ((CUR != '<') || (NXT(1) != '!')) | 
| 6641 | 0 |         return(ret); | 
| 6642 | 9.09k |     SKIP(2); | 
| 6643 |  |  | 
| 6644 |  |     /* GROW; done in the caller */ | 
| 6645 | 9.09k |     if (CMP7(CUR_PTR, 'E', 'L', 'E', 'M', 'E', 'N', 'T')) { | 
| 6646 | 9.07k | #ifdef LIBXML_VALID_ENABLED | 
| 6647 | 9.07k |   int oldInputNr = ctxt->inputNr; | 
| 6648 | 9.07k | #endif | 
| 6649 |  |  | 
| 6650 | 9.07k |   SKIP(7); | 
| 6651 | 9.07k |   if (SKIP_BLANKS_PE == 0) { | 
| 6652 | 393 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 6653 | 393 |                "Space required after 'ELEMENT'\n"); | 
| 6654 | 393 |       return(-1); | 
| 6655 | 393 |   } | 
| 6656 | 8.68k |         name = xmlParseName(ctxt); | 
| 6657 | 8.68k |   if (name == NULL) { | 
| 6658 | 263 |       xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 6659 | 263 |          "xmlParseElementDecl: no name for Element\n"); | 
| 6660 | 263 |       return(-1); | 
| 6661 | 263 |   } | 
| 6662 | 8.42k |   if (SKIP_BLANKS_PE == 0) { | 
| 6663 | 6.60k |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 6664 | 6.60k |          "Space required after the element name\n"); | 
| 6665 | 6.60k |   } | 
| 6666 | 8.42k |   if (CMP5(CUR_PTR, 'E', 'M', 'P', 'T', 'Y')) { | 
| 6667 | 843 |       SKIP(5); | 
| 6668 |  |       /* | 
| 6669 |  |        * Element must always be empty. | 
| 6670 |  |        */ | 
| 6671 | 843 |       ret = XML_ELEMENT_TYPE_EMPTY; | 
| 6672 | 7.57k |   } else if ((RAW == 'A') && (NXT(1) == 'N') && | 
| 6673 | 544 |              (NXT(2) == 'Y')) { | 
| 6674 | 543 |       SKIP(3); | 
| 6675 |  |       /* | 
| 6676 |  |        * Element is a generic container. | 
| 6677 |  |        */ | 
| 6678 | 543 |       ret = XML_ELEMENT_TYPE_ANY; | 
| 6679 | 7.03k |   } else if (RAW == '(') { | 
| 6680 | 6.36k |       ret = xmlParseElementContentDecl(ctxt, name, &content); | 
| 6681 | 6.36k |             if (ret <= 0) | 
| 6682 | 4.61k |                 return(-1); | 
| 6683 | 6.36k |   } else { | 
| 6684 |  |       /* | 
| 6685 |  |        * [ WFC: PEs in Internal Subset ] error handling. | 
| 6686 |  |        */ | 
| 6687 | 668 |             xmlFatalErrMsg(ctxt, XML_ERR_ELEMCONTENT_NOT_STARTED, | 
| 6688 | 668 |                   "xmlParseElementDecl: 'EMPTY', 'ANY' or '(' expected\n"); | 
| 6689 | 668 |       return(-1); | 
| 6690 | 668 |   } | 
| 6691 |  |  | 
| 6692 | 3.13k |   SKIP_BLANKS_PE; | 
| 6693 |  |  | 
| 6694 | 3.13k |   if (RAW != '>') { | 
| 6695 | 1.87k |       xmlFatalErr(ctxt, XML_ERR_GT_REQUIRED, NULL); | 
| 6696 | 1.87k |       if (content != NULL) { | 
| 6697 | 1.27k |     xmlFreeDocElementContent(ctxt->myDoc, content); | 
| 6698 | 1.27k |       } | 
| 6699 | 1.87k |   } else { | 
| 6700 | 1.26k | #ifdef LIBXML_VALID_ENABLED | 
| 6701 | 1.26k |       if ((ctxt->validate) && (ctxt->inputNr > oldInputNr)) { | 
| 6702 | 0 |     xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6703 | 0 |                                  "Element declaration doesn't start and stop in" | 
| 6704 | 0 |                                  " the same entity\n", | 
| 6705 | 0 |                                  NULL, NULL); | 
| 6706 | 0 |       } | 
| 6707 | 1.26k | #endif | 
| 6708 |  |  | 
| 6709 | 1.26k |       NEXT; | 
| 6710 | 1.26k |       if ((ctxt->sax != NULL) && (!ctxt->disableSAX) && | 
| 6711 | 727 |     (ctxt->sax->elementDecl != NULL)) { | 
| 6712 | 727 |     if (content != NULL) | 
| 6713 | 278 |         content->parent = NULL; | 
| 6714 | 727 |           ctxt->sax->elementDecl(ctxt->userData, name, ret, | 
| 6715 | 727 |                            content); | 
| 6716 | 727 |     if ((content != NULL) && (content->parent == NULL)) { | 
| 6717 |  |         /* | 
| 6718 |  |          * this is a trick: if xmlAddElementDecl is called, | 
| 6719 |  |          * instead of copying the full tree it is plugged directly | 
| 6720 |  |          * if called from the parser. Avoid duplicating the | 
| 6721 |  |          * interfaces or change the API/ABI | 
| 6722 |  |          */ | 
| 6723 | 184 |         xmlFreeDocElementContent(ctxt->myDoc, content); | 
| 6724 | 184 |     } | 
| 6725 | 727 |       } else if (content != NULL) { | 
| 6726 | 195 |     xmlFreeDocElementContent(ctxt->myDoc, content); | 
| 6727 | 195 |       } | 
| 6728 | 1.26k |   } | 
| 6729 | 3.13k |     } | 
| 6730 | 3.15k |     return(ret); | 
| 6731 | 9.09k | } | 
| 6732 |  |  | 
| 6733 |  | /** | 
| 6734 |  |  * Parse a conditional section. Always consumes '<!['. | 
| 6735 |  |  * | 
| 6736 |  |  *     [61] conditionalSect ::= includeSect | ignoreSect | 
| 6737 |  |  *     [62] includeSect ::= '<![' S? 'INCLUDE' S? '[' extSubsetDecl ']]>' | 
| 6738 |  |  *     [63] ignoreSect ::= '<![' S? 'IGNORE' S? '[' ignoreSectContents* ']]>' | 
| 6739 |  |  *     [64] ignoreSectContents ::= Ignore ('<![' ignoreSectContents ']]>' | 
| 6740 |  |  *                                 Ignore)* | 
| 6741 |  |  *     [65] Ignore ::= Char* - (Char* ('<![' | ']]>') Char*) | 
| 6742 |  |  * @param ctxt  an XML parser context | 
| 6743 |  |  */ | 
| 6744 |  |  | 
| 6745 |  | static void | 
| 6746 | 0 | xmlParseConditionalSections(xmlParserCtxtPtr ctxt) { | 
| 6747 | 0 |     size_t depth = 0; | 
| 6748 | 0 |     int isFreshPE = 0; | 
| 6749 | 0 |     int oldInputNr = ctxt->inputNr; | 
| 6750 | 0 |     int declInputNr = ctxt->inputNr; | 
| 6751 |  | 
 | 
| 6752 | 0 |     while (!PARSER_STOPPED(ctxt)) { | 
| 6753 | 0 |         if (ctxt->input->cur >= ctxt->input->end) { | 
| 6754 | 0 |             if (ctxt->inputNr <= oldInputNr) { | 
| 6755 | 0 |                 xmlFatalErr(ctxt, XML_ERR_EXT_SUBSET_NOT_FINISHED, NULL); | 
| 6756 | 0 |                 return; | 
| 6757 | 0 |             } | 
| 6758 |  |  | 
| 6759 | 0 |             xmlPopPE(ctxt); | 
| 6760 | 0 |             declInputNr = ctxt->inputNr; | 
| 6761 | 0 |         } else if ((RAW == '<') && (NXT(1) == '!') && (NXT(2) == '[')) { | 
| 6762 | 0 |             SKIP(3); | 
| 6763 | 0 |             SKIP_BLANKS_PE; | 
| 6764 |  | 
 | 
| 6765 | 0 |             isFreshPE = 0; | 
| 6766 |  | 
 | 
| 6767 | 0 |             if (CMP7(CUR_PTR, 'I', 'N', 'C', 'L', 'U', 'D', 'E')) { | 
| 6768 | 0 |                 SKIP(7); | 
| 6769 | 0 |                 SKIP_BLANKS_PE; | 
| 6770 | 0 |                 if (RAW != '[') { | 
| 6771 | 0 |                     xmlFatalErr(ctxt, XML_ERR_CONDSEC_INVALID, NULL); | 
| 6772 | 0 |                     return; | 
| 6773 | 0 |                 } | 
| 6774 | 0 | #ifdef LIBXML_VALID_ENABLED | 
| 6775 | 0 |                 if ((ctxt->validate) && (ctxt->inputNr > declInputNr)) { | 
| 6776 | 0 |         xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6777 | 0 |                                      "All markup of the conditional section is" | 
| 6778 | 0 |                                      " not in the same entity\n", | 
| 6779 | 0 |                                      NULL, NULL); | 
| 6780 | 0 |                 } | 
| 6781 | 0 | #endif | 
| 6782 | 0 |                 NEXT; | 
| 6783 |  | 
 | 
| 6784 | 0 |                 depth++; | 
| 6785 | 0 |             } else if (CMP6(CUR_PTR, 'I', 'G', 'N', 'O', 'R', 'E')) { | 
| 6786 | 0 |                 size_t ignoreDepth = 0; | 
| 6787 |  | 
 | 
| 6788 | 0 |                 SKIP(6); | 
| 6789 | 0 |                 SKIP_BLANKS_PE; | 
| 6790 | 0 |                 if (RAW != '[') { | 
| 6791 | 0 |                     xmlFatalErr(ctxt, XML_ERR_CONDSEC_INVALID, NULL); | 
| 6792 | 0 |                     return; | 
| 6793 | 0 |                 } | 
| 6794 | 0 | #ifdef LIBXML_VALID_ENABLED | 
| 6795 | 0 |                 if ((ctxt->validate) && (ctxt->inputNr > declInputNr)) { | 
| 6796 | 0 |         xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6797 | 0 |                                      "All markup of the conditional section is" | 
| 6798 | 0 |                                      " not in the same entity\n", | 
| 6799 | 0 |                                      NULL, NULL); | 
| 6800 | 0 |                 } | 
| 6801 | 0 | #endif | 
| 6802 | 0 |                 NEXT; | 
| 6803 |  | 
 | 
| 6804 | 0 |                 while (PARSER_STOPPED(ctxt) == 0) { | 
| 6805 | 0 |                     if (RAW == 0) { | 
| 6806 | 0 |                         xmlFatalErr(ctxt, XML_ERR_CONDSEC_NOT_FINISHED, NULL); | 
| 6807 | 0 |                         return; | 
| 6808 | 0 |                     } | 
| 6809 | 0 |                     if ((RAW == '<') && (NXT(1) == '!') && (NXT(2) == '[')) { | 
| 6810 | 0 |                         SKIP(3); | 
| 6811 | 0 |                         ignoreDepth++; | 
| 6812 |  |                         /* Check for integer overflow */ | 
| 6813 | 0 |                         if (ignoreDepth == 0) { | 
| 6814 | 0 |                             xmlErrMemory(ctxt); | 
| 6815 | 0 |                             return; | 
| 6816 | 0 |                         } | 
| 6817 | 0 |                     } else if ((RAW == ']') && (NXT(1) == ']') && | 
| 6818 | 0 |                                (NXT(2) == '>')) { | 
| 6819 | 0 |                         SKIP(3); | 
| 6820 | 0 |                         if (ignoreDepth == 0) | 
| 6821 | 0 |                             break; | 
| 6822 | 0 |                         ignoreDepth--; | 
| 6823 | 0 |                     } else { | 
| 6824 | 0 |                         NEXT; | 
| 6825 | 0 |                     } | 
| 6826 | 0 |                 } | 
| 6827 |  |  | 
| 6828 | 0 | #ifdef LIBXML_VALID_ENABLED | 
| 6829 | 0 |                 if ((ctxt->validate) && (ctxt->inputNr > declInputNr)) { | 
| 6830 | 0 |         xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6831 | 0 |                                      "All markup of the conditional section is" | 
| 6832 | 0 |                                      " not in the same entity\n", | 
| 6833 | 0 |                                      NULL, NULL); | 
| 6834 | 0 |                 } | 
| 6835 | 0 | #endif | 
| 6836 | 0 |             } else { | 
| 6837 | 0 |                 xmlFatalErr(ctxt, XML_ERR_CONDSEC_INVALID_KEYWORD, NULL); | 
| 6838 | 0 |                 return; | 
| 6839 | 0 |             } | 
| 6840 | 0 |         } else if ((depth > 0) && | 
| 6841 | 0 |                    (RAW == ']') && (NXT(1) == ']') && (NXT(2) == '>')) { | 
| 6842 | 0 |             if (isFreshPE) { | 
| 6843 | 0 |                 xmlFatalErrMsg(ctxt, XML_ERR_CONDSEC_INVALID, | 
| 6844 | 0 |                                "Parameter entity must match " | 
| 6845 | 0 |                                "extSubsetDecl\n"); | 
| 6846 | 0 |                 return; | 
| 6847 | 0 |             } | 
| 6848 |  |  | 
| 6849 | 0 |             depth--; | 
| 6850 | 0 | #ifdef LIBXML_VALID_ENABLED | 
| 6851 | 0 |             if ((ctxt->validate) && (ctxt->inputNr > declInputNr)) { | 
| 6852 | 0 |     xmlValidityError(ctxt, XML_ERR_ENTITY_BOUNDARY, | 
| 6853 | 0 |                                  "All markup of the conditional section is not" | 
| 6854 | 0 |                                  " in the same entity\n", | 
| 6855 | 0 |                                  NULL, NULL); | 
| 6856 | 0 |             } | 
| 6857 | 0 | #endif | 
| 6858 | 0 |             SKIP(3); | 
| 6859 | 0 |         } else if ((RAW == '<') && ((NXT(1) == '!') || (NXT(1) == '?'))) { | 
| 6860 | 0 |             isFreshPE = 0; | 
| 6861 | 0 |             xmlParseMarkupDecl(ctxt); | 
| 6862 | 0 |         } else if (RAW == '%') { | 
| 6863 | 0 |             xmlParsePERefInternal(ctxt, 1); | 
| 6864 | 0 |             if (ctxt->inputNr > declInputNr) { | 
| 6865 | 0 |                 isFreshPE = 1; | 
| 6866 | 0 |                 declInputNr = ctxt->inputNr; | 
| 6867 | 0 |             } | 
| 6868 | 0 |         } else { | 
| 6869 | 0 |             xmlFatalErr(ctxt, XML_ERR_EXT_SUBSET_NOT_FINISHED, NULL); | 
| 6870 | 0 |             return; | 
| 6871 | 0 |         } | 
| 6872 |  |  | 
| 6873 | 0 |         if (depth == 0) | 
| 6874 | 0 |             break; | 
| 6875 |  |  | 
| 6876 | 0 |         SKIP_BLANKS; | 
| 6877 | 0 |         SHRINK; | 
| 6878 | 0 |         GROW; | 
| 6879 | 0 |     } | 
| 6880 | 0 | } | 
| 6881 |  |  | 
| 6882 |  | /** | 
| 6883 |  |  * Parse markup declarations. Always consumes '<!' or '<?'. | 
| 6884 |  |  * | 
| 6885 |  |  * @deprecated Internal function, don't use. | 
| 6886 |  |  * | 
| 6887 |  |  *     [29] markupdecl ::= elementdecl | AttlistDecl | EntityDecl | | 
| 6888 |  |  *                         NotationDecl | PI | Comment | 
| 6889 |  |  * | 
| 6890 |  |  * [ VC: Proper Declaration/PE Nesting ] | 
| 6891 |  |  * Parameter-entity replacement text must be properly nested with | 
| 6892 |  |  * markup declarations. That is to say, if either the first character | 
| 6893 |  |  * or the last character of a markup declaration (markupdecl above) is | 
| 6894 |  |  * contained in the replacement text for a parameter-entity reference, | 
| 6895 |  |  * both must be contained in the same replacement text. | 
| 6896 |  |  * | 
| 6897 |  |  * [ WFC: PEs in Internal Subset ] | 
| 6898 |  |  * In the internal DTD subset, parameter-entity references can occur | 
| 6899 |  |  * only where markup declarations can occur, not within markup declarations. | 
| 6900 |  |  * (This does not apply to references that occur in external parameter | 
| 6901 |  |  * entities or to the external subset.) | 
| 6902 |  |  * | 
| 6903 |  |  * @param ctxt  an XML parser context | 
| 6904 |  |  */ | 
| 6905 |  | void | 
| 6906 | 89.9k | xmlParseMarkupDecl(xmlParserCtxt *ctxt) { | 
| 6907 | 89.9k |     GROW; | 
| 6908 | 89.9k |     if (CUR == '<') { | 
| 6909 | 89.9k |         if (NXT(1) == '!') { | 
| 6910 | 60.6k |       switch (NXT(2)) { | 
| 6911 | 35.7k |           case 'E': | 
| 6912 | 35.7k |         if (NXT(3) == 'L') | 
| 6913 | 9.09k |       xmlParseElementDecl(ctxt); | 
| 6914 | 26.6k |         else if (NXT(3) == 'N') | 
| 6915 | 26.6k |       xmlParseEntityDecl(ctxt); | 
| 6916 | 20 |                     else | 
| 6917 | 20 |                         SKIP(2); | 
| 6918 | 35.7k |         break; | 
| 6919 | 14.0k |           case 'A': | 
| 6920 | 14.0k |         xmlParseAttributeListDecl(ctxt); | 
| 6921 | 14.0k |         break; | 
| 6922 | 4.06k |           case 'N': | 
| 6923 | 4.06k |         xmlParseNotationDecl(ctxt); | 
| 6924 | 4.06k |         break; | 
| 6925 | 5.11k |           case '-': | 
| 6926 | 5.11k |         xmlParseComment(ctxt); | 
| 6927 | 5.11k |         break; | 
| 6928 | 1.72k |     default: | 
| 6929 | 1.72k |                     xmlFatalErr(ctxt, | 
| 6930 | 1.72k |                                 ctxt->inSubset == 2 ? | 
| 6931 | 0 |                                     XML_ERR_EXT_SUBSET_NOT_FINISHED : | 
| 6932 | 1.72k |                                     XML_ERR_INT_SUBSET_NOT_FINISHED, | 
| 6933 | 1.72k |                                 NULL); | 
| 6934 | 1.72k |                     SKIP(2); | 
| 6935 | 1.72k |         break; | 
| 6936 | 60.6k |       } | 
| 6937 | 60.6k |   } else if (NXT(1) == '?') { | 
| 6938 | 29.2k |       xmlParsePI(ctxt); | 
| 6939 | 29.2k |   } | 
| 6940 | 89.9k |     } | 
| 6941 | 89.9k | } | 
| 6942 |  |  | 
| 6943 |  | /** | 
| 6944 |  |  * Parse an XML declaration header for external entities | 
| 6945 |  |  * | 
| 6946 |  |  * @deprecated Internal function, don't use. | 
| 6947 |  |  * | 
| 6948 |  |  *     [77] TextDecl ::= '<?xml' VersionInfo? EncodingDecl S? '?>' | 
| 6949 |  |  * @param ctxt  an XML parser context | 
| 6950 |  |  */ | 
| 6951 |  |  | 
| 6952 |  | void | 
| 6953 | 0 | xmlParseTextDecl(xmlParserCtxt *ctxt) { | 
| 6954 | 0 |     xmlChar *version; | 
| 6955 |  |  | 
| 6956 |  |     /* | 
| 6957 |  |      * We know that '<?xml' is here. | 
| 6958 |  |      */ | 
| 6959 | 0 |     if ((CMP5(CUR_PTR, '<', '?', 'x', 'm', 'l')) && (IS_BLANK_CH(NXT(5)))) { | 
| 6960 | 0 |   SKIP(5); | 
| 6961 | 0 |     } else { | 
| 6962 | 0 |   xmlFatalErr(ctxt, XML_ERR_XMLDECL_NOT_STARTED, NULL); | 
| 6963 | 0 |   return; | 
| 6964 | 0 |     } | 
| 6965 |  |  | 
| 6966 | 0 |     if (SKIP_BLANKS == 0) { | 
| 6967 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 6968 | 0 |            "Space needed after '<?xml'\n"); | 
| 6969 | 0 |     } | 
| 6970 |  |  | 
| 6971 |  |     /* | 
| 6972 |  |      * We may have the VersionInfo here. | 
| 6973 |  |      */ | 
| 6974 | 0 |     version = xmlParseVersionInfo(ctxt); | 
| 6975 | 0 |     if (version == NULL) { | 
| 6976 | 0 |   version = xmlCharStrdup(XML_DEFAULT_VERSION); | 
| 6977 | 0 |         if (version == NULL) { | 
| 6978 | 0 |             xmlErrMemory(ctxt); | 
| 6979 | 0 |             return; | 
| 6980 | 0 |         } | 
| 6981 | 0 |     } else { | 
| 6982 | 0 |   if (SKIP_BLANKS == 0) { | 
| 6983 | 0 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 6984 | 0 |                "Space needed here\n"); | 
| 6985 | 0 |   } | 
| 6986 | 0 |     } | 
| 6987 | 0 |     ctxt->input->version = version; | 
| 6988 |  |  | 
| 6989 |  |     /* | 
| 6990 |  |      * We must have the encoding declaration | 
| 6991 |  |      */ | 
| 6992 | 0 |     xmlParseEncodingDecl(ctxt); | 
| 6993 |  | 
 | 
| 6994 | 0 |     SKIP_BLANKS; | 
| 6995 | 0 |     if ((RAW == '?') && (NXT(1) == '>')) { | 
| 6996 | 0 |         SKIP(2); | 
| 6997 | 0 |     } else if (RAW == '>') { | 
| 6998 |  |         /* Deprecated old WD ... */ | 
| 6999 | 0 |   xmlFatalErr(ctxt, XML_ERR_XMLDECL_NOT_FINISHED, NULL); | 
| 7000 | 0 |   NEXT; | 
| 7001 | 0 |     } else { | 
| 7002 | 0 |         int c; | 
| 7003 |  | 
 | 
| 7004 | 0 |   xmlFatalErr(ctxt, XML_ERR_XMLDECL_NOT_FINISHED, NULL); | 
| 7005 | 0 |         while ((PARSER_STOPPED(ctxt) == 0) && ((c = CUR) != 0)) { | 
| 7006 | 0 |             NEXT; | 
| 7007 | 0 |             if (c == '>') | 
| 7008 | 0 |                 break; | 
| 7009 | 0 |         } | 
| 7010 | 0 |     } | 
| 7011 | 0 | } | 
| 7012 |  |  | 
| 7013 |  | /** | 
| 7014 |  |  * Parse Markup declarations from an external subset | 
| 7015 |  |  * | 
| 7016 |  |  * @deprecated Internal function, don't use. | 
| 7017 |  |  * | 
| 7018 |  |  *     [30] extSubset ::= textDecl? extSubsetDecl | 
| 7019 |  |  * | 
| 7020 |  |  *     [31] extSubsetDecl ::= (markupdecl | conditionalSect | | 
| 7021 |  |  *                             PEReference | S) * | 
| 7022 |  |  * @param ctxt  an XML parser context | 
| 7023 |  |  * @param publicId  the public identifier | 
| 7024 |  |  * @param systemId  the system identifier (URL) | 
| 7025 |  |  */ | 
| 7026 |  | void | 
| 7027 |  | xmlParseExternalSubset(xmlParserCtxt *ctxt, const xmlChar *publicId, | 
| 7028 | 0 |                        const xmlChar *systemId) { | 
| 7029 | 0 |     int oldInputNr; | 
| 7030 |  | 
 | 
| 7031 | 0 |     xmlCtxtInitializeLate(ctxt); | 
| 7032 |  | 
 | 
| 7033 | 0 |     xmlDetectEncoding(ctxt); | 
| 7034 |  | 
 | 
| 7035 | 0 |     if (CMP5(CUR_PTR, '<', '?', 'x', 'm', 'l')) { | 
| 7036 | 0 |   xmlParseTextDecl(ctxt); | 
| 7037 | 0 |     } | 
| 7038 | 0 |     if (ctxt->myDoc == NULL) { | 
| 7039 | 0 |         ctxt->myDoc = xmlNewDoc(BAD_CAST "1.0"); | 
| 7040 | 0 |   if (ctxt->myDoc == NULL) { | 
| 7041 | 0 |       xmlErrMemory(ctxt); | 
| 7042 | 0 |       return; | 
| 7043 | 0 |   } | 
| 7044 | 0 |   ctxt->myDoc->properties = XML_DOC_INTERNAL; | 
| 7045 | 0 |     } | 
| 7046 | 0 |     if ((ctxt->myDoc->intSubset == NULL) && | 
| 7047 | 0 |         (xmlCreateIntSubset(ctxt->myDoc, NULL, publicId, systemId) == NULL)) { | 
| 7048 | 0 |         xmlErrMemory(ctxt); | 
| 7049 | 0 |     } | 
| 7050 |  | 
 | 
| 7051 | 0 |     ctxt->inSubset = 2; | 
| 7052 | 0 |     oldInputNr = ctxt->inputNr; | 
| 7053 |  | 
 | 
| 7054 | 0 |     SKIP_BLANKS; | 
| 7055 | 0 |     while (!PARSER_STOPPED(ctxt)) { | 
| 7056 | 0 |         if (ctxt->input->cur >= ctxt->input->end) { | 
| 7057 | 0 |             if (ctxt->inputNr <= oldInputNr) { | 
| 7058 | 0 |                 xmlParserCheckEOF(ctxt, XML_ERR_EXT_SUBSET_NOT_FINISHED); | 
| 7059 | 0 |                 break; | 
| 7060 | 0 |             } | 
| 7061 |  |  | 
| 7062 | 0 |             xmlPopPE(ctxt); | 
| 7063 | 0 |         } else if ((RAW == '<') && (NXT(1) == '!') && (NXT(2) == '[')) { | 
| 7064 | 0 |             xmlParseConditionalSections(ctxt); | 
| 7065 | 0 |         } else if ((RAW == '<') && ((NXT(1) == '!') || (NXT(1) == '?'))) { | 
| 7066 | 0 |             xmlParseMarkupDecl(ctxt); | 
| 7067 | 0 |         } else if (RAW == '%') { | 
| 7068 | 0 |             xmlParsePERefInternal(ctxt, 1); | 
| 7069 | 0 |         } else { | 
| 7070 | 0 |             xmlFatalErr(ctxt, XML_ERR_EXT_SUBSET_NOT_FINISHED, NULL); | 
| 7071 |  | 
 | 
| 7072 | 0 |             while (ctxt->inputNr > oldInputNr) | 
| 7073 | 0 |                 xmlPopPE(ctxt); | 
| 7074 | 0 |             break; | 
| 7075 | 0 |         } | 
| 7076 | 0 |         SKIP_BLANKS; | 
| 7077 | 0 |         SHRINK; | 
| 7078 | 0 |         GROW; | 
| 7079 | 0 |     } | 
| 7080 | 0 | } | 
| 7081 |  |  | 
| 7082 |  | /** | 
| 7083 |  |  * Parse and handle entity references in content, depending on the SAX | 
| 7084 |  |  * interface, this may end-up in a call to character() if this is a | 
| 7085 |  |  * CharRef, a predefined entity, if there is no reference() callback. | 
| 7086 |  |  * or if the parser was asked to switch to that mode. | 
| 7087 |  |  * | 
| 7088 |  |  * @deprecated Internal function, don't use. | 
| 7089 |  |  * | 
| 7090 |  |  * Always consumes '&'. | 
| 7091 |  |  * | 
| 7092 |  |  *     [67] Reference ::= EntityRef | CharRef | 
| 7093 |  |  * @param ctxt  an XML parser context | 
| 7094 |  |  */ | 
| 7095 |  | void | 
| 7096 | 110k | xmlParseReference(xmlParserCtxt *ctxt) { | 
| 7097 | 110k |     xmlEntityPtr ent = NULL; | 
| 7098 | 110k |     const xmlChar *name; | 
| 7099 | 110k |     xmlChar *val; | 
| 7100 |  |  | 
| 7101 | 110k |     if (RAW != '&') | 
| 7102 | 0 |         return; | 
| 7103 |  |  | 
| 7104 |  |     /* | 
| 7105 |  |      * Simple case of a CharRef | 
| 7106 |  |      */ | 
| 7107 | 110k |     if (NXT(1) == '#') { | 
| 7108 | 19.8k |   int i = 0; | 
| 7109 | 19.8k |   xmlChar out[16]; | 
| 7110 | 19.8k |   int value = xmlParseCharRef(ctxt); | 
| 7111 |  |  | 
| 7112 | 19.8k |   if (value == 0) | 
| 7113 | 6.59k |       return; | 
| 7114 |  |  | 
| 7115 |  |         /* | 
| 7116 |  |          * Just encode the value in UTF-8 | 
| 7117 |  |          */ | 
| 7118 | 13.2k |         COPY_BUF(out, i, value); | 
| 7119 | 13.2k |         out[i] = 0; | 
| 7120 | 13.2k |         if ((ctxt->sax != NULL) && (ctxt->sax->characters != NULL) && | 
| 7121 | 13.2k |             (!ctxt->disableSAX)) | 
| 7122 | 577 |             ctxt->sax->characters(ctxt->userData, out, i); | 
| 7123 | 13.2k |   return; | 
| 7124 | 19.8k |     } | 
| 7125 |  |  | 
| 7126 |  |     /* | 
| 7127 |  |      * We are seeing an entity reference | 
| 7128 |  |      */ | 
| 7129 | 90.7k |     name = xmlParseEntityRefInternal(ctxt); | 
| 7130 | 90.7k |     if (name == NULL) | 
| 7131 | 33.7k |         return; | 
| 7132 | 56.9k |     ent = xmlLookupGeneralEntity(ctxt, name, /* isAttr */ 0); | 
| 7133 | 56.9k |     if (ent == NULL) { | 
| 7134 |  |         /* | 
| 7135 |  |          * Create a reference for undeclared entities. | 
| 7136 |  |          */ | 
| 7137 | 24.9k |         if ((ctxt->replaceEntities == 0) && | 
| 7138 | 0 |             (ctxt->sax != NULL) && | 
| 7139 | 0 |             (ctxt->disableSAX == 0) && | 
| 7140 | 0 |             (ctxt->sax->reference != NULL)) { | 
| 7141 | 0 |             ctxt->sax->reference(ctxt->userData, name); | 
| 7142 | 0 |         } | 
| 7143 | 24.9k |         return; | 
| 7144 | 24.9k |     } | 
| 7145 | 32.0k |     if (!ctxt->wellFormed) | 
| 7146 | 18.6k |   return; | 
| 7147 |  |  | 
| 7148 |  |     /* special case of predefined entities */ | 
| 7149 | 13.3k |     if ((ent->name == NULL) || | 
| 7150 | 13.3k |         (ent->etype == XML_INTERNAL_PREDEFINED_ENTITY)) { | 
| 7151 | 711 |   val = ent->content; | 
| 7152 | 711 |   if (val == NULL) return; | 
| 7153 |  |   /* | 
| 7154 |  |    * inline the entity. | 
| 7155 |  |    */ | 
| 7156 | 711 |   if ((ctxt->sax != NULL) && (ctxt->sax->characters != NULL) && | 
| 7157 | 711 |       (!ctxt->disableSAX)) | 
| 7158 | 711 |       ctxt->sax->characters(ctxt->userData, val, xmlStrlen(val)); | 
| 7159 | 711 |   return; | 
| 7160 | 711 |     } | 
| 7161 |  |  | 
| 7162 |  |     /* | 
| 7163 |  |      * Some users try to parse entities on their own and used to set | 
| 7164 |  |      * the renamed "checked" member. Fix the flags to cover this | 
| 7165 |  |      * case. | 
| 7166 |  |      */ | 
| 7167 | 12.6k |     if (((ent->flags & XML_ENT_PARSED) == 0) && (ent->children != NULL)) | 
| 7168 | 0 |         ent->flags |= XML_ENT_PARSED; | 
| 7169 |  |  | 
| 7170 |  |     /* | 
| 7171 |  |      * The first reference to the entity trigger a parsing phase | 
| 7172 |  |      * where the ent->children is filled with the result from | 
| 7173 |  |      * the parsing. | 
| 7174 |  |      * Note: external parsed entities will not be loaded, it is not | 
| 7175 |  |      * required for a non-validating parser, unless the parsing option | 
| 7176 |  |      * of validating, or substituting entities were given. Doing so is | 
| 7177 |  |      * far more secure as the parser will only process data coming from | 
| 7178 |  |      * the document entity by default. | 
| 7179 |  |      * | 
| 7180 |  |      * FIXME: This doesn't work correctly since entities can be | 
| 7181 |  |      * expanded with different namespace declarations in scope. | 
| 7182 |  |      * For example: | 
| 7183 |  |      * | 
| 7184 |  |      * <!DOCTYPE doc [ | 
| 7185 |  |      *   <!ENTITY ent "<ns:elem/>"> | 
| 7186 |  |      * ]> | 
| 7187 |  |      * <doc> | 
| 7188 |  |      *   <decl1 xmlns:ns="urn:ns1"> | 
| 7189 |  |      *     &ent; | 
| 7190 |  |      *   </decl1> | 
| 7191 |  |      *   <decl2 xmlns:ns="urn:ns2"> | 
| 7192 |  |      *     &ent; | 
| 7193 |  |      *   </decl2> | 
| 7194 |  |      * </doc> | 
| 7195 |  |      * | 
| 7196 |  |      * Proposed fix: | 
| 7197 |  |      * | 
| 7198 |  |      * - Ignore current namespace declarations when parsing the | 
| 7199 |  |      *   entity. If a prefix can't be resolved, don't report an error | 
| 7200 |  |      *   but mark it as unresolved. | 
| 7201 |  |      * - Try to resolve these prefixes when expanding the entity. | 
| 7202 |  |      *   This will require a specialized version of xmlStaticCopyNode | 
| 7203 |  |      *   which can also make use of the namespace hash table to avoid | 
| 7204 |  |      *   quadratic behavior. | 
| 7205 |  |      * | 
| 7206 |  |      * Alternatively, we could simply reparse the entity on each | 
| 7207 |  |      * expansion like we already do with custom SAX callbacks. | 
| 7208 |  |      * External entity content should be cached in this case. | 
| 7209 |  |      */ | 
| 7210 | 12.6k |     if ((ent->etype == XML_INTERNAL_GENERAL_ENTITY) || | 
| 7211 | 66 |         (((ctxt->options & XML_PARSE_NO_XXE) == 0) && | 
| 7212 | 0 |          ((ctxt->replaceEntities) || | 
| 7213 | 12.5k |           (ctxt->validate)))) { | 
| 7214 | 12.5k |         if ((ent->flags & XML_ENT_PARSED) == 0) { | 
| 7215 | 1.02k |             xmlCtxtParseEntity(ctxt, ent); | 
| 7216 | 11.5k |         } else if (ent->children == NULL) { | 
| 7217 |  |             /* | 
| 7218 |  |              * Probably running in SAX mode and the callbacks don't | 
| 7219 |  |              * build the entity content. Parse the entity again. | 
| 7220 |  |              * | 
| 7221 |  |              * This will also be triggered in normal tree builder mode | 
| 7222 |  |              * if an entity happens to be empty, causing unnecessary | 
| 7223 |  |              * reloads. It's hard to come up with a reliable check in | 
| 7224 |  |              * which mode we're running. | 
| 7225 |  |              */ | 
| 7226 | 717 |             xmlCtxtParseEntity(ctxt, ent); | 
| 7227 | 717 |         } | 
| 7228 | 12.5k |     } | 
| 7229 |  |  | 
| 7230 |  |     /* | 
| 7231 |  |      * We also check for amplification if entities aren't substituted. | 
| 7232 |  |      * They might be expanded later. | 
| 7233 |  |      */ | 
| 7234 | 12.6k |     if (xmlParserEntityCheck(ctxt, ent->expandedSize)) | 
| 7235 | 77 |         return; | 
| 7236 |  |  | 
| 7237 | 12.5k |     if ((ctxt->sax == NULL) || (ctxt->disableSAX)) | 
| 7238 | 255 |         return; | 
| 7239 |  |  | 
| 7240 | 12.3k |     if (ctxt->replaceEntities == 0) { | 
| 7241 |  |   /* | 
| 7242 |  |    * Create a reference | 
| 7243 |  |    */ | 
| 7244 | 0 |         if (ctxt->sax->reference != NULL) | 
| 7245 | 0 |       ctxt->sax->reference(ctxt->userData, ent->name); | 
| 7246 | 12.3k |     } else if ((ent->children != NULL) && (ctxt->node != NULL)) { | 
| 7247 | 11.4k |         xmlNodePtr copy, cur; | 
| 7248 |  |  | 
| 7249 |  |         /* | 
| 7250 |  |          * Seems we are generating the DOM content, copy the tree | 
| 7251 |  |    */ | 
| 7252 | 11.4k |         cur = ent->children; | 
| 7253 |  |  | 
| 7254 |  |         /* | 
| 7255 |  |          * Handle first text node with SAX to coalesce text efficiently | 
| 7256 |  |          */ | 
| 7257 | 11.4k |         if ((cur->type == XML_TEXT_NODE) || | 
| 7258 | 10.3k |             (cur->type == XML_CDATA_SECTION_NODE)) { | 
| 7259 | 10.3k |             int len = xmlStrlen(cur->content); | 
| 7260 |  |  | 
| 7261 | 10.3k |             if ((cur->type == XML_TEXT_NODE) || | 
| 7262 | 9.72k |                 (ctxt->options & XML_PARSE_NOCDATA)) { | 
| 7263 | 9.72k |                 if (ctxt->sax->characters != NULL) | 
| 7264 | 9.72k |                     ctxt->sax->characters(ctxt, cur->content, len); | 
| 7265 | 9.72k |             } else { | 
| 7266 | 599 |                 if (ctxt->sax->cdataBlock != NULL) | 
| 7267 | 599 |                     ctxt->sax->cdataBlock(ctxt, cur->content, len); | 
| 7268 | 599 |             } | 
| 7269 |  |  | 
| 7270 | 10.3k |             cur = cur->next; | 
| 7271 | 10.3k |         } | 
| 7272 |  |  | 
| 7273 | 16.6k |         while (cur != NULL) { | 
| 7274 | 6.74k |             xmlNodePtr last; | 
| 7275 |  |  | 
| 7276 |  |             /* | 
| 7277 |  |              * Handle last text node with SAX to coalesce text efficiently | 
| 7278 |  |              */ | 
| 7279 | 6.74k |             if ((cur->next == NULL) && | 
| 7280 | 2.69k |                 ((cur->type == XML_TEXT_NODE) || | 
| 7281 | 1.55k |                  (cur->type == XML_CDATA_SECTION_NODE))) { | 
| 7282 | 1.55k |                 int len = xmlStrlen(cur->content); | 
| 7283 |  |  | 
| 7284 | 1.55k |                 if ((cur->type == XML_TEXT_NODE) || | 
| 7285 | 1.21k |                     (ctxt->options & XML_PARSE_NOCDATA)) { | 
| 7286 | 1.21k |                     if (ctxt->sax->characters != NULL) | 
| 7287 | 1.21k |                         ctxt->sax->characters(ctxt, cur->content, len); | 
| 7288 | 1.21k |                 } else { | 
| 7289 | 341 |                     if (ctxt->sax->cdataBlock != NULL) | 
| 7290 | 341 |                         ctxt->sax->cdataBlock(ctxt, cur->content, len); | 
| 7291 | 341 |                 } | 
| 7292 |  |  | 
| 7293 | 1.55k |                 break; | 
| 7294 | 1.55k |             } | 
| 7295 |  |  | 
| 7296 |  |             /* | 
| 7297 |  |              * Reset coalesce buffer stats only for non-text nodes. | 
| 7298 |  |              */ | 
| 7299 | 5.19k |             ctxt->nodemem = 0; | 
| 7300 | 5.19k |             ctxt->nodelen = 0; | 
| 7301 |  |  | 
| 7302 | 5.19k |             copy = xmlDocCopyNode(cur, ctxt->myDoc, 1); | 
| 7303 |  |  | 
| 7304 | 5.19k |             if (copy == NULL) { | 
| 7305 | 0 |                 xmlErrMemory(ctxt); | 
| 7306 | 0 |                 break; | 
| 7307 | 0 |             } | 
| 7308 |  |  | 
| 7309 | 5.19k |             if (ctxt->parseMode == XML_PARSE_READER) { | 
| 7310 |  |                 /* Needed for reader */ | 
| 7311 | 0 |                 copy->extra = cur->extra; | 
| 7312 |  |                 /* Maybe needed for reader */ | 
| 7313 | 0 |                 copy->_private = cur->_private; | 
| 7314 | 0 |             } | 
| 7315 |  |  | 
| 7316 | 5.19k |             copy->parent = ctxt->node; | 
| 7317 | 5.19k |             last = ctxt->node->last; | 
| 7318 | 5.19k |             if (last == NULL) { | 
| 7319 | 213 |                 ctxt->node->children = copy; | 
| 7320 | 4.97k |             } else { | 
| 7321 | 4.97k |                 last->next = copy; | 
| 7322 | 4.97k |                 copy->prev = last; | 
| 7323 | 4.97k |             } | 
| 7324 | 5.19k |             ctxt->node->last = copy; | 
| 7325 |  |  | 
| 7326 | 5.19k |             cur = cur->next; | 
| 7327 | 5.19k |         } | 
| 7328 | 11.4k |     } | 
| 7329 | 12.3k | } | 
| 7330 |  |  | 
| 7331 |  | static void | 
| 7332 | 67.9k | xmlHandleUndeclaredEntity(xmlParserCtxtPtr ctxt, const xmlChar *name) { | 
| 7333 |  |     /* | 
| 7334 |  |      * [ WFC: Entity Declared ] | 
| 7335 |  |      * In a document without any DTD, a document with only an | 
| 7336 |  |      * internal DTD subset which contains no parameter entity | 
| 7337 |  |      * references, or a document with "standalone='yes'", the | 
| 7338 |  |      * Name given in the entity reference must match that in an | 
| 7339 |  |      * entity declaration, except that well-formed documents | 
| 7340 |  |      * need not declare any of the following entities: amp, lt, | 
| 7341 |  |      * gt, apos, quot. | 
| 7342 |  |      * The declaration of a parameter entity must precede any | 
| 7343 |  |      * reference to it. | 
| 7344 |  |      * Similarly, the declaration of a general entity must | 
| 7345 |  |      * precede any reference to it which appears in a default | 
| 7346 |  |      * value in an attribute-list declaration. Note that if | 
| 7347 |  |      * entities are declared in the external subset or in | 
| 7348 |  |      * external parameter entities, a non-validating processor | 
| 7349 |  |      * is not obligated to read and process their declarations; | 
| 7350 |  |      * for such documents, the rule that an entity must be | 
| 7351 |  |      * declared is a well-formedness constraint only if | 
| 7352 |  |      * standalone='yes'. | 
| 7353 |  |      */ | 
| 7354 | 67.9k |     if ((ctxt->standalone == 1) || | 
| 7355 | 67.4k |         ((ctxt->hasExternalSubset == 0) && | 
| 7356 | 55.0k |          (ctxt->hasPErefs == 0))) { | 
| 7357 | 48.3k |         xmlFatalErrMsgStr(ctxt, XML_ERR_UNDECLARED_ENTITY, | 
| 7358 | 48.3k |                           "Entity '%s' not defined\n", name); | 
| 7359 | 48.3k | #ifdef LIBXML_VALID_ENABLED | 
| 7360 | 48.3k |     } else if (ctxt->validate) { | 
| 7361 |  |         /* | 
| 7362 |  |          * [ VC: Entity Declared ] | 
| 7363 |  |          * In a document with an external subset or external | 
| 7364 |  |          * parameter entities with "standalone='no'", ... | 
| 7365 |  |          * ... The declaration of a parameter entity must | 
| 7366 |  |          * precede any reference to it... | 
| 7367 |  |          */ | 
| 7368 | 0 |         xmlValidityError(ctxt, XML_ERR_UNDECLARED_ENTITY, | 
| 7369 | 0 |                          "Entity '%s' not defined\n", name, NULL); | 
| 7370 | 0 | #endif | 
| 7371 | 19.6k |     } else if ((ctxt->loadsubset & ~XML_SKIP_IDS) || | 
| 7372 | 0 |                ((ctxt->replaceEntities) && | 
| 7373 | 19.6k |                 ((ctxt->options & XML_PARSE_NO_XXE) == 0))) { | 
| 7374 |  |         /* | 
| 7375 |  |          * Also raise a non-fatal error | 
| 7376 |  |          * | 
| 7377 |  |          * - if the external subset is loaded and all entity declarations | 
| 7378 |  |          *   should be available, or | 
| 7379 |  |          * - entity substition was requested without restricting | 
| 7380 |  |          *   external entity access. | 
| 7381 |  |          */ | 
| 7382 | 19.6k |         xmlErrMsgStr(ctxt, XML_WAR_UNDECLARED_ENTITY, | 
| 7383 | 19.6k |                      "Entity '%s' not defined\n", name); | 
| 7384 | 19.6k |     } else { | 
| 7385 | 0 |         xmlWarningMsg(ctxt, XML_WAR_UNDECLARED_ENTITY, | 
| 7386 | 0 |                       "Entity '%s' not defined\n", name, NULL); | 
| 7387 | 0 |     } | 
| 7388 |  |  | 
| 7389 | 67.9k |     ctxt->valid = 0; | 
| 7390 | 67.9k | } | 
| 7391 |  |  | 
| 7392 |  | static xmlEntityPtr | 
| 7393 | 558k | xmlLookupGeneralEntity(xmlParserCtxtPtr ctxt, const xmlChar *name, int inAttr) { | 
| 7394 | 558k |     xmlEntityPtr ent = NULL; | 
| 7395 |  |  | 
| 7396 |  |     /* | 
| 7397 |  |      * Predefined entities override any extra definition | 
| 7398 |  |      */ | 
| 7399 | 558k |     if ((ctxt->options & XML_PARSE_OLDSAX) == 0) { | 
| 7400 | 558k |         ent = xmlGetPredefinedEntity(name); | 
| 7401 | 558k |         if (ent != NULL) | 
| 7402 | 27.0k |             return(ent); | 
| 7403 | 558k |     } | 
| 7404 |  |  | 
| 7405 |  |     /* | 
| 7406 |  |      * Ask first SAX for entity resolution, otherwise try the | 
| 7407 |  |      * entities which may have stored in the parser context. | 
| 7408 |  |      */ | 
| 7409 | 531k |     if (ctxt->sax != NULL) { | 
| 7410 | 531k |   if (ctxt->sax->getEntity != NULL) | 
| 7411 | 531k |       ent = ctxt->sax->getEntity(ctxt->userData, name); | 
| 7412 | 531k |   if ((ctxt->wellFormed == 1 ) && (ent == NULL) && | 
| 7413 | 2.90k |       (ctxt->options & XML_PARSE_OLDSAX)) | 
| 7414 | 0 |       ent = xmlGetPredefinedEntity(name); | 
| 7415 | 531k |   if ((ctxt->wellFormed == 1 ) && (ent == NULL) && | 
| 7416 | 2.90k |       (ctxt->userData==ctxt)) { | 
| 7417 | 2.90k |       ent = xmlSAX2GetEntity(ctxt, name); | 
| 7418 | 2.90k |   } | 
| 7419 | 531k |     } | 
| 7420 |  |  | 
| 7421 | 531k |     if (ent == NULL) { | 
| 7422 | 62.0k |         xmlHandleUndeclaredEntity(ctxt, name); | 
| 7423 | 62.0k |     } | 
| 7424 |  |  | 
| 7425 |  |     /* | 
| 7426 |  |      * [ WFC: Parsed Entity ] | 
| 7427 |  |      * An entity reference must not contain the name of an | 
| 7428 |  |      * unparsed entity | 
| 7429 |  |      */ | 
| 7430 | 469k |     else if (ent->etype == XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) { | 
| 7431 | 194 |   xmlFatalErrMsgStr(ctxt, XML_ERR_UNPARSED_ENTITY, | 
| 7432 | 194 |      "Entity reference to unparsed entity %s\n", name); | 
| 7433 | 194 |         ent = NULL; | 
| 7434 | 194 |     } | 
| 7435 |  |  | 
| 7436 |  |     /* | 
| 7437 |  |      * [ WFC: No External Entity References ] | 
| 7438 |  |      * Attribute values cannot contain direct or indirect | 
| 7439 |  |      * entity references to external entities. | 
| 7440 |  |      */ | 
| 7441 | 469k |     else if (ent->etype == XML_EXTERNAL_GENERAL_PARSED_ENTITY) { | 
| 7442 | 899 |         if (inAttr) { | 
| 7443 | 515 |             xmlFatalErrMsgStr(ctxt, XML_ERR_ENTITY_IS_EXTERNAL, | 
| 7444 | 515 |                  "Attribute references external entity '%s'\n", name); | 
| 7445 | 515 |             ent = NULL; | 
| 7446 | 515 |         } | 
| 7447 | 899 |     } | 
| 7448 |  |  | 
| 7449 | 531k |     return(ent); | 
| 7450 | 558k | } | 
| 7451 |  |  | 
| 7452 |  | /** | 
| 7453 |  |  * Parse an entity reference. Always consumes '&'. | 
| 7454 |  |  * | 
| 7455 |  |  *     [68] EntityRef ::= '&' Name ';' | 
| 7456 |  |  * | 
| 7457 |  |  * @param ctxt  an XML parser context | 
| 7458 |  |  * @returns the name, or NULL in case of error. | 
| 7459 |  |  */ | 
| 7460 |  | static const xmlChar * | 
| 7461 | 205k | xmlParseEntityRefInternal(xmlParserCtxtPtr ctxt) { | 
| 7462 | 205k |     const xmlChar *name; | 
| 7463 |  |  | 
| 7464 | 205k |     GROW; | 
| 7465 |  |  | 
| 7466 | 205k |     if (RAW != '&') | 
| 7467 | 0 |         return(NULL); | 
| 7468 | 205k |     NEXT; | 
| 7469 | 205k |     name = xmlParseName(ctxt); | 
| 7470 | 205k |     if (name == NULL) { | 
| 7471 | 33.8k |   xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 7472 | 33.8k |            "xmlParseEntityRef: no name\n"); | 
| 7473 | 33.8k |         return(NULL); | 
| 7474 | 33.8k |     } | 
| 7475 | 171k |     if (RAW != ';') { | 
| 7476 | 22.6k |   xmlFatalErr(ctxt, XML_ERR_ENTITYREF_SEMICOL_MISSING, NULL); | 
| 7477 | 22.6k |   return(NULL); | 
| 7478 | 22.6k |     } | 
| 7479 | 148k |     NEXT; | 
| 7480 |  |  | 
| 7481 | 148k |     return(name); | 
| 7482 | 171k | } | 
| 7483 |  |  | 
| 7484 |  | /** | 
| 7485 |  |  * @deprecated Internal function, don't use. | 
| 7486 |  |  * | 
| 7487 |  |  * @param ctxt  an XML parser context | 
| 7488 |  |  * @returns the xmlEntity if found, or NULL otherwise. | 
| 7489 |  |  */ | 
| 7490 |  | xmlEntity * | 
| 7491 | 0 | xmlParseEntityRef(xmlParserCtxt *ctxt) { | 
| 7492 | 0 |     const xmlChar *name; | 
| 7493 |  | 
 | 
| 7494 | 0 |     if (ctxt == NULL) | 
| 7495 | 0 |         return(NULL); | 
| 7496 |  |  | 
| 7497 | 0 |     name = xmlParseEntityRefInternal(ctxt); | 
| 7498 | 0 |     if (name == NULL) | 
| 7499 | 0 |         return(NULL); | 
| 7500 |  |  | 
| 7501 | 0 |     return(xmlLookupGeneralEntity(ctxt, name, /* inAttr */ 0)); | 
| 7502 | 0 | } | 
| 7503 |  |  | 
| 7504 |  | /** | 
| 7505 |  |  * Parse ENTITY references declarations, but this version parses it from | 
| 7506 |  |  * a string value. | 
| 7507 |  |  * | 
| 7508 |  |  *     [68] EntityRef ::= '&' Name ';' | 
| 7509 |  |  * | 
| 7510 |  |  * [ WFC: Entity Declared ] | 
| 7511 |  |  * In a document without any DTD, a document with only an internal DTD | 
| 7512 |  |  * subset which contains no parameter entity references, or a document | 
| 7513 |  |  * with "standalone='yes'", the Name given in the entity reference | 
| 7514 |  |  * must match that in an entity declaration, except that well-formed | 
| 7515 |  |  * documents need not declare any of the following entities: amp, lt, | 
| 7516 |  |  * gt, apos, quot.  The declaration of a parameter entity must precede | 
| 7517 |  |  * any reference to it.  Similarly, the declaration of a general entity | 
| 7518 |  |  * must precede any reference to it which appears in a default value in an | 
| 7519 |  |  * attribute-list declaration. Note that if entities are declared in the | 
| 7520 |  |  * external subset or in external parameter entities, a non-validating | 
| 7521 |  |  * processor is not obligated to read and process their declarations; | 
| 7522 |  |  * for such documents, the rule that an entity must be declared is a | 
| 7523 |  |  * well-formedness constraint only if standalone='yes'. | 
| 7524 |  |  * | 
| 7525 |  |  * [ WFC: Parsed Entity ] | 
| 7526 |  |  * An entity reference must not contain the name of an unparsed entity | 
| 7527 |  |  * | 
| 7528 |  |  * @param ctxt  an XML parser context | 
| 7529 |  |  * @param str  a pointer to an index in the string | 
| 7530 |  |  * @returns the xmlEntity if found, or NULL otherwise. The str pointer | 
| 7531 |  |  * is updated to the current location in the string. | 
| 7532 |  |  */ | 
| 7533 |  | static xmlChar * | 
| 7534 | 410k | xmlParseStringEntityRef(xmlParserCtxtPtr ctxt, const xmlChar ** str) { | 
| 7535 | 410k |     xmlChar *name; | 
| 7536 | 410k |     const xmlChar *ptr; | 
| 7537 | 410k |     xmlChar cur; | 
| 7538 |  |  | 
| 7539 | 410k |     if ((str == NULL) || (*str == NULL)) | 
| 7540 | 0 |         return(NULL); | 
| 7541 | 410k |     ptr = *str; | 
| 7542 | 410k |     cur = *ptr; | 
| 7543 | 410k |     if (cur != '&') | 
| 7544 | 0 |   return(NULL); | 
| 7545 |  |  | 
| 7546 | 410k |     ptr++; | 
| 7547 | 410k |     name = xmlParseStringName(ctxt, &ptr); | 
| 7548 | 410k |     if (name == NULL) { | 
| 7549 | 8 |   xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 7550 | 8 |            "xmlParseStringEntityRef: no name\n"); | 
| 7551 | 8 |   *str = ptr; | 
| 7552 | 8 |   return(NULL); | 
| 7553 | 8 |     } | 
| 7554 | 410k |     if (*ptr != ';') { | 
| 7555 | 9 |   xmlFatalErr(ctxt, XML_ERR_ENTITYREF_SEMICOL_MISSING, NULL); | 
| 7556 | 9 |         xmlFree(name); | 
| 7557 | 9 |   *str = ptr; | 
| 7558 | 9 |   return(NULL); | 
| 7559 | 9 |     } | 
| 7560 | 410k |     ptr++; | 
| 7561 |  |  | 
| 7562 | 410k |     *str = ptr; | 
| 7563 | 410k |     return(name); | 
| 7564 | 410k | } | 
| 7565 |  |  | 
| 7566 |  | /** | 
| 7567 |  |  * Parse a parameter entity reference. Always consumes '%'. | 
| 7568 |  |  * | 
| 7569 |  |  * The entity content is handled directly by pushing it's content as | 
| 7570 |  |  * a new input stream. | 
| 7571 |  |  * | 
| 7572 |  |  *     [69] PEReference ::= '%' Name ';' | 
| 7573 |  |  * | 
| 7574 |  |  * [ WFC: No Recursion ] | 
| 7575 |  |  * A parsed entity must not contain a recursive | 
| 7576 |  |  * reference to itself, either directly or indirectly. | 
| 7577 |  |  * | 
| 7578 |  |  * [ WFC: Entity Declared ] | 
| 7579 |  |  * In a document without any DTD, a document with only an internal DTD | 
| 7580 |  |  * subset which contains no parameter entity references, or a document | 
| 7581 |  |  * with "standalone='yes'", ...  ... The declaration of a parameter | 
| 7582 |  |  * entity must precede any reference to it... | 
| 7583 |  |  * | 
| 7584 |  |  * [ VC: Entity Declared ] | 
| 7585 |  |  * In a document with an external subset or external parameter entities | 
| 7586 |  |  * with "standalone='no'", ...  ... The declaration of a parameter entity | 
| 7587 |  |  * must precede any reference to it... | 
| 7588 |  |  * | 
| 7589 |  |  * [ WFC: In DTD ] | 
| 7590 |  |  * Parameter-entity references may only appear in the DTD. | 
| 7591 |  |  * NOTE: misleading but this is handled. | 
| 7592 |  |  * | 
| 7593 |  |  * @param ctxt  an XML parser context | 
| 7594 |  |  * @param markupDecl  whether the PERef starts a markup declaration | 
| 7595 |  |  */ | 
| 7596 |  | static void | 
| 7597 | 65.3k | xmlParsePERefInternal(xmlParserCtxt *ctxt, int markupDecl) { | 
| 7598 | 65.3k |     const xmlChar *name; | 
| 7599 | 65.3k |     xmlEntityPtr entity = NULL; | 
| 7600 | 65.3k |     xmlParserInputPtr input; | 
| 7601 |  |  | 
| 7602 | 65.3k |     if (RAW != '%') | 
| 7603 | 0 |         return; | 
| 7604 | 65.3k |     NEXT; | 
| 7605 | 65.3k |     name = xmlParseName(ctxt); | 
| 7606 | 65.3k |     if (name == NULL) { | 
| 7607 | 4.88k |   xmlFatalErrMsg(ctxt, XML_ERR_PEREF_NO_NAME, "PEReference: no name\n"); | 
| 7608 | 4.88k |   return; | 
| 7609 | 4.88k |     } | 
| 7610 | 60.4k |     if (RAW != ';') { | 
| 7611 | 4.63k |   xmlFatalErr(ctxt, XML_ERR_PEREF_SEMICOL_MISSING, NULL); | 
| 7612 | 4.63k |         return; | 
| 7613 | 4.63k |     } | 
| 7614 |  |  | 
| 7615 | 55.7k |     NEXT; | 
| 7616 |  |  | 
| 7617 |  |     /* Must be set before xmlHandleUndeclaredEntity */ | 
| 7618 | 55.7k |     ctxt->hasPErefs = 1; | 
| 7619 |  |  | 
| 7620 |  |     /* | 
| 7621 |  |      * Request the entity from SAX | 
| 7622 |  |      */ | 
| 7623 | 55.7k |     if ((ctxt->sax != NULL) && | 
| 7624 | 55.7k |   (ctxt->sax->getParameterEntity != NULL)) | 
| 7625 | 55.7k |   entity = ctxt->sax->getParameterEntity(ctxt->userData, name); | 
| 7626 |  |  | 
| 7627 | 55.7k |     if (entity == NULL) { | 
| 7628 | 5.23k |         xmlHandleUndeclaredEntity(ctxt, name); | 
| 7629 | 50.5k |     } else { | 
| 7630 |  |   /* | 
| 7631 |  |    * Internal checking in case the entity quest barfed | 
| 7632 |  |    */ | 
| 7633 | 50.5k |   if ((entity->etype != XML_INTERNAL_PARAMETER_ENTITY) && | 
| 7634 | 863 |       (entity->etype != XML_EXTERNAL_PARAMETER_ENTITY)) { | 
| 7635 | 0 |       xmlWarningMsg(ctxt, XML_WAR_UNDECLARED_ENTITY, | 
| 7636 | 0 |       "Internal: %%%s; is not a parameter entity\n", | 
| 7637 | 0 |         name, NULL); | 
| 7638 | 50.5k |   } else { | 
| 7639 | 50.5k |       if ((entity->etype == XML_EXTERNAL_PARAMETER_ENTITY) && | 
| 7640 | 863 |                 ((ctxt->options & XML_PARSE_NO_XXE) || | 
| 7641 | 0 |      (((ctxt->loadsubset & ~XML_SKIP_IDS) == 0) && | 
| 7642 | 0 |       (ctxt->replaceEntities == 0) && | 
| 7643 | 0 |       (ctxt->validate == 0)))) | 
| 7644 | 863 |     return; | 
| 7645 |  |  | 
| 7646 | 49.6k |             if (entity->flags & XML_ENT_EXPANDING) { | 
| 7647 | 20 |                 xmlFatalErr(ctxt, XML_ERR_ENTITY_LOOP, NULL); | 
| 7648 | 20 |                 return; | 
| 7649 | 20 |             } | 
| 7650 |  |  | 
| 7651 | 49.6k |       input = xmlNewEntityInputStream(ctxt, entity); | 
| 7652 | 49.6k |       if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 7653 | 0 |                 xmlFreeInputStream(input); | 
| 7654 | 0 |     return; | 
| 7655 | 0 |             } | 
| 7656 |  |  | 
| 7657 | 49.6k |             entity->flags |= XML_ENT_EXPANDING; | 
| 7658 |  |  | 
| 7659 | 49.6k |             if (markupDecl) | 
| 7660 | 46.7k |                 input->flags |= XML_INPUT_MARKUP_DECL; | 
| 7661 |  |  | 
| 7662 | 49.6k |             GROW; | 
| 7663 |  |  | 
| 7664 | 49.6k |       if (entity->etype == XML_EXTERNAL_PARAMETER_ENTITY) { | 
| 7665 | 0 |                 xmlDetectEncoding(ctxt); | 
| 7666 |  | 
 | 
| 7667 | 0 |                 if ((CMP5(CUR_PTR, '<', '?', 'x', 'm', 'l')) && | 
| 7668 | 0 |                     (IS_BLANK_CH(NXT(5)))) { | 
| 7669 | 0 |                     xmlParseTextDecl(ctxt); | 
| 7670 | 0 |                 } | 
| 7671 | 0 |             } | 
| 7672 | 49.6k |   } | 
| 7673 | 50.5k |     } | 
| 7674 | 55.7k | } | 
| 7675 |  |  | 
| 7676 |  | /** | 
| 7677 |  |  * Parse a parameter entity reference. | 
| 7678 |  |  * | 
| 7679 |  |  * @deprecated Internal function, don't use. | 
| 7680 |  |  * | 
| 7681 |  |  * @param ctxt  an XML parser context | 
| 7682 |  |  */ | 
| 7683 |  | void | 
| 7684 | 0 | xmlParsePEReference(xmlParserCtxt *ctxt) { | 
| 7685 | 0 |     xmlParsePERefInternal(ctxt, 0); | 
| 7686 | 0 | } | 
| 7687 |  |  | 
| 7688 |  | /** | 
| 7689 |  |  * Load the content of an entity. | 
| 7690 |  |  * | 
| 7691 |  |  * @param ctxt  an XML parser context | 
| 7692 |  |  * @param entity  an unloaded system entity | 
| 7693 |  |  * @returns 0 in case of success and -1 in case of failure | 
| 7694 |  |  */ | 
| 7695 |  | static int | 
| 7696 | 0 | xmlLoadEntityContent(xmlParserCtxtPtr ctxt, xmlEntityPtr entity) { | 
| 7697 | 0 |     xmlParserInputPtr oldinput, input = NULL; | 
| 7698 | 0 |     xmlParserInputPtr *oldinputTab; | 
| 7699 | 0 |     xmlChar *oldencoding; | 
| 7700 | 0 |     xmlChar *content = NULL; | 
| 7701 | 0 |     xmlResourceType rtype; | 
| 7702 | 0 |     size_t length, i; | 
| 7703 | 0 |     int oldinputNr, oldinputMax; | 
| 7704 | 0 |     int ret = -1; | 
| 7705 | 0 |     int res; | 
| 7706 |  | 
 | 
| 7707 | 0 |     if ((ctxt == NULL) || (entity == NULL) || | 
| 7708 | 0 |         ((entity->etype != XML_EXTERNAL_PARAMETER_ENTITY) && | 
| 7709 | 0 |    (entity->etype != XML_EXTERNAL_GENERAL_PARSED_ENTITY)) || | 
| 7710 | 0 |   (entity->content != NULL)) { | 
| 7711 | 0 |   xmlFatalErr(ctxt, XML_ERR_ARGUMENT, | 
| 7712 | 0 |               "xmlLoadEntityContent parameter error"); | 
| 7713 | 0 |         return(-1); | 
| 7714 | 0 |     } | 
| 7715 |  |  | 
| 7716 | 0 |     if (entity->etype == XML_EXTERNAL_PARAMETER_ENTITY) | 
| 7717 | 0 |         rtype = XML_RESOURCE_PARAMETER_ENTITY; | 
| 7718 | 0 |     else | 
| 7719 | 0 |         rtype = XML_RESOURCE_GENERAL_ENTITY; | 
| 7720 |  | 
 | 
| 7721 | 0 |     input = xmlLoadResource(ctxt, (char *) entity->URI, | 
| 7722 | 0 |                             (char *) entity->ExternalID, rtype); | 
| 7723 | 0 |     if (input == NULL) | 
| 7724 | 0 |         return(-1); | 
| 7725 |  |  | 
| 7726 | 0 |     oldinput = ctxt->input; | 
| 7727 | 0 |     oldinputNr = ctxt->inputNr; | 
| 7728 | 0 |     oldinputMax = ctxt->inputMax; | 
| 7729 | 0 |     oldinputTab = ctxt->inputTab; | 
| 7730 | 0 |     oldencoding = ctxt->encoding; | 
| 7731 |  | 
 | 
| 7732 | 0 |     ctxt->input = NULL; | 
| 7733 | 0 |     ctxt->inputNr = 0; | 
| 7734 | 0 |     ctxt->inputMax = 1; | 
| 7735 | 0 |     ctxt->encoding = NULL; | 
| 7736 | 0 |     ctxt->inputTab = xmlMalloc(sizeof(xmlParserInputPtr)); | 
| 7737 | 0 |     if (ctxt->inputTab == NULL) { | 
| 7738 | 0 |         xmlErrMemory(ctxt); | 
| 7739 | 0 |         xmlFreeInputStream(input); | 
| 7740 | 0 |         goto error; | 
| 7741 | 0 |     } | 
| 7742 |  |  | 
| 7743 | 0 |     xmlBufResetInput(input->buf->buffer, input); | 
| 7744 |  | 
 | 
| 7745 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 7746 | 0 |         xmlFreeInputStream(input); | 
| 7747 | 0 |         goto error; | 
| 7748 | 0 |     } | 
| 7749 |  |  | 
| 7750 | 0 |     xmlDetectEncoding(ctxt); | 
| 7751 |  |  | 
| 7752 |  |     /* | 
| 7753 |  |      * Parse a possible text declaration first | 
| 7754 |  |      */ | 
| 7755 | 0 |     if ((CMP5(CUR_PTR, '<', '?', 'x', 'm', 'l')) && (IS_BLANK_CH(NXT(5)))) { | 
| 7756 | 0 |   xmlParseTextDecl(ctxt); | 
| 7757 |  |         /* | 
| 7758 |  |          * An XML-1.0 document can't reference an entity not XML-1.0 | 
| 7759 |  |          */ | 
| 7760 | 0 |         if ((xmlStrEqual(ctxt->version, BAD_CAST "1.0")) && | 
| 7761 | 0 |             (!xmlStrEqual(ctxt->input->version, BAD_CAST "1.0"))) { | 
| 7762 | 0 |             xmlFatalErrMsg(ctxt, XML_ERR_VERSION_MISMATCH, | 
| 7763 | 0 |                            "Version mismatch between document and entity\n"); | 
| 7764 | 0 |         } | 
| 7765 | 0 |     } | 
| 7766 |  | 
 | 
| 7767 | 0 |     length = input->cur - input->base; | 
| 7768 | 0 |     xmlBufShrink(input->buf->buffer, length); | 
| 7769 | 0 |     xmlSaturatedAdd(&ctxt->sizeentities, length); | 
| 7770 |  | 
 | 
| 7771 | 0 |     while ((res = xmlParserInputBufferGrow(input->buf, 4096)) > 0) | 
| 7772 | 0 |         ; | 
| 7773 |  | 
 | 
| 7774 | 0 |     xmlBufResetInput(input->buf->buffer, input); | 
| 7775 |  | 
 | 
| 7776 | 0 |     if (res < 0) { | 
| 7777 | 0 |         xmlCtxtErrIO(ctxt, input->buf->error, NULL); | 
| 7778 | 0 |         goto error; | 
| 7779 | 0 |     } | 
| 7780 |  |  | 
| 7781 | 0 |     length = xmlBufUse(input->buf->buffer); | 
| 7782 | 0 |     if (length > INT_MAX) { | 
| 7783 | 0 |         xmlErrMemory(ctxt); | 
| 7784 | 0 |         goto error; | 
| 7785 | 0 |     } | 
| 7786 |  |  | 
| 7787 | 0 |     content = xmlStrndup(xmlBufContent(input->buf->buffer), length); | 
| 7788 | 0 |     if (content == NULL) { | 
| 7789 | 0 |         xmlErrMemory(ctxt); | 
| 7790 | 0 |         goto error; | 
| 7791 | 0 |     } | 
| 7792 |  |  | 
| 7793 | 0 |     for (i = 0; i < length; ) { | 
| 7794 | 0 |         int clen = length - i; | 
| 7795 | 0 |         int c = xmlGetUTF8Char(content + i, &clen); | 
| 7796 |  | 
 | 
| 7797 | 0 |         if ((c < 0) || (!IS_CHAR(c))) { | 
| 7798 | 0 |             xmlFatalErrMsgInt(ctxt, XML_ERR_INVALID_CHAR, | 
| 7799 | 0 |                               "xmlLoadEntityContent: invalid char value %d\n", | 
| 7800 | 0 |                               content[i]); | 
| 7801 | 0 |             goto error; | 
| 7802 | 0 |         } | 
| 7803 | 0 |         i += clen; | 
| 7804 | 0 |     } | 
| 7805 |  |  | 
| 7806 | 0 |     xmlSaturatedAdd(&ctxt->sizeentities, length); | 
| 7807 | 0 |     entity->content = content; | 
| 7808 | 0 |     entity->length = length; | 
| 7809 | 0 |     content = NULL; | 
| 7810 | 0 |     ret = 0; | 
| 7811 |  | 
 | 
| 7812 | 0 | error: | 
| 7813 | 0 |     while (ctxt->inputNr > 0) | 
| 7814 | 0 |         xmlFreeInputStream(xmlCtxtPopInput(ctxt)); | 
| 7815 | 0 |     xmlFree(ctxt->inputTab); | 
| 7816 | 0 |     xmlFree(ctxt->encoding); | 
| 7817 |  | 
 | 
| 7818 | 0 |     ctxt->input = oldinput; | 
| 7819 | 0 |     ctxt->inputNr = oldinputNr; | 
| 7820 | 0 |     ctxt->inputMax = oldinputMax; | 
| 7821 | 0 |     ctxt->inputTab = oldinputTab; | 
| 7822 | 0 |     ctxt->encoding = oldencoding; | 
| 7823 |  | 
 | 
| 7824 | 0 |     xmlFree(content); | 
| 7825 |  | 
 | 
| 7826 | 0 |     return(ret); | 
| 7827 | 0 | } | 
| 7828 |  |  | 
| 7829 |  | /** | 
| 7830 |  |  * Parse PEReference declarations | 
| 7831 |  |  * | 
| 7832 |  |  *     [69] PEReference ::= '%' Name ';' | 
| 7833 |  |  * | 
| 7834 |  |  * [ WFC: No Recursion ] | 
| 7835 |  |  * A parsed entity must not contain a recursive | 
| 7836 |  |  * reference to itself, either directly or indirectly. | 
| 7837 |  |  * | 
| 7838 |  |  * [ WFC: Entity Declared ] | 
| 7839 |  |  * In a document without any DTD, a document with only an internal DTD | 
| 7840 |  |  * subset which contains no parameter entity references, or a document | 
| 7841 |  |  * with "standalone='yes'", ...  ... The declaration of a parameter | 
| 7842 |  |  * entity must precede any reference to it... | 
| 7843 |  |  * | 
| 7844 |  |  * [ VC: Entity Declared ] | 
| 7845 |  |  * In a document with an external subset or external parameter entities | 
| 7846 |  |  * with "standalone='no'", ...  ... The declaration of a parameter entity | 
| 7847 |  |  * must precede any reference to it... | 
| 7848 |  |  * | 
| 7849 |  |  * [ WFC: In DTD ] | 
| 7850 |  |  * Parameter-entity references may only appear in the DTD. | 
| 7851 |  |  * NOTE: misleading but this is handled. | 
| 7852 |  |  * | 
| 7853 |  |  * @param ctxt  an XML parser context | 
| 7854 |  |  * @param str  a pointer to an index in the string | 
| 7855 |  |  * @returns the string of the entity content. | 
| 7856 |  |  *         str is updated to the current value of the index | 
| 7857 |  |  */ | 
| 7858 |  | static xmlEntityPtr | 
| 7859 | 2.16k | xmlParseStringPEReference(xmlParserCtxtPtr ctxt, const xmlChar **str) { | 
| 7860 | 2.16k |     const xmlChar *ptr; | 
| 7861 | 2.16k |     xmlChar cur; | 
| 7862 | 2.16k |     xmlChar *name; | 
| 7863 | 2.16k |     xmlEntityPtr entity = NULL; | 
| 7864 |  |  | 
| 7865 | 2.16k |     if ((str == NULL) || (*str == NULL)) return(NULL); | 
| 7866 | 2.16k |     ptr = *str; | 
| 7867 | 2.16k |     cur = *ptr; | 
| 7868 | 2.16k |     if (cur != '%') | 
| 7869 | 0 |         return(NULL); | 
| 7870 | 2.16k |     ptr++; | 
| 7871 | 2.16k |     name = xmlParseStringName(ctxt, &ptr); | 
| 7872 | 2.16k |     if (name == NULL) { | 
| 7873 | 684 |   xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 7874 | 684 |            "xmlParseStringPEReference: no name\n"); | 
| 7875 | 684 |   *str = ptr; | 
| 7876 | 684 |   return(NULL); | 
| 7877 | 684 |     } | 
| 7878 | 1.48k |     cur = *ptr; | 
| 7879 | 1.48k |     if (cur != ';') { | 
| 7880 | 262 |   xmlFatalErr(ctxt, XML_ERR_ENTITYREF_SEMICOL_MISSING, NULL); | 
| 7881 | 262 |   xmlFree(name); | 
| 7882 | 262 |   *str = ptr; | 
| 7883 | 262 |   return(NULL); | 
| 7884 | 262 |     } | 
| 7885 | 1.21k |     ptr++; | 
| 7886 |  |  | 
| 7887 |  |     /* Must be set before xmlHandleUndeclaredEntity */ | 
| 7888 | 1.21k |     ctxt->hasPErefs = 1; | 
| 7889 |  |  | 
| 7890 |  |     /* | 
| 7891 |  |      * Request the entity from SAX | 
| 7892 |  |      */ | 
| 7893 | 1.21k |     if ((ctxt->sax != NULL) && | 
| 7894 | 1.21k |   (ctxt->sax->getParameterEntity != NULL)) | 
| 7895 | 1.21k |   entity = ctxt->sax->getParameterEntity(ctxt->userData, name); | 
| 7896 |  |  | 
| 7897 | 1.21k |     if (entity == NULL) { | 
| 7898 | 729 |         xmlHandleUndeclaredEntity(ctxt, name); | 
| 7899 | 729 |     } else { | 
| 7900 |  |   /* | 
| 7901 |  |    * Internal checking in case the entity quest barfed | 
| 7902 |  |    */ | 
| 7903 | 490 |   if ((entity->etype != XML_INTERNAL_PARAMETER_ENTITY) && | 
| 7904 | 66 |       (entity->etype != XML_EXTERNAL_PARAMETER_ENTITY)) { | 
| 7905 | 0 |       xmlWarningMsg(ctxt, XML_WAR_UNDECLARED_ENTITY, | 
| 7906 | 0 |         "%%%s; is not a parameter entity\n", | 
| 7907 | 0 |         name, NULL); | 
| 7908 | 0 |   } | 
| 7909 | 490 |     } | 
| 7910 |  |  | 
| 7911 | 1.21k |     xmlFree(name); | 
| 7912 | 1.21k |     *str = ptr; | 
| 7913 | 1.21k |     return(entity); | 
| 7914 | 1.48k | } | 
| 7915 |  |  | 
| 7916 |  | /** | 
| 7917 |  |  * Parse a DOCTYPE declaration | 
| 7918 |  |  * | 
| 7919 |  |  * @deprecated Internal function, don't use. | 
| 7920 |  |  * | 
| 7921 |  |  *     [28] doctypedecl ::= '<!DOCTYPE' S Name (S ExternalID)? S? | 
| 7922 |  |  *                          ('[' (markupdecl | PEReference | S)* ']' S?)? '>' | 
| 7923 |  |  * | 
| 7924 |  |  * [ VC: Root Element Type ] | 
| 7925 |  |  * The Name in the document type declaration must match the element | 
| 7926 |  |  * type of the root element. | 
| 7927 |  |  * | 
| 7928 |  |  * @param ctxt  an XML parser context | 
| 7929 |  |  */ | 
| 7930 |  |  | 
| 7931 |  | void | 
| 7932 | 7.62k | xmlParseDocTypeDecl(xmlParserCtxt *ctxt) { | 
| 7933 | 7.62k |     const xmlChar *name = NULL; | 
| 7934 | 7.62k |     xmlChar *publicId = NULL; | 
| 7935 | 7.62k |     xmlChar *URI = NULL; | 
| 7936 |  |  | 
| 7937 |  |     /* | 
| 7938 |  |      * We know that '<!DOCTYPE' has been detected. | 
| 7939 |  |      */ | 
| 7940 | 7.62k |     SKIP(9); | 
| 7941 |  |  | 
| 7942 | 7.62k |     if (SKIP_BLANKS == 0) { | 
| 7943 | 3.85k |         xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 7944 | 3.85k |                        "Space required after 'DOCTYPE'\n"); | 
| 7945 | 3.85k |     } | 
| 7946 |  |  | 
| 7947 |  |     /* | 
| 7948 |  |      * Parse the DOCTYPE name. | 
| 7949 |  |      */ | 
| 7950 | 7.62k |     name = xmlParseName(ctxt); | 
| 7951 | 7.62k |     if (name == NULL) { | 
| 7952 | 2.48k |   xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 7953 | 2.48k |            "xmlParseDocTypeDecl : no DOCTYPE name !\n"); | 
| 7954 | 2.48k |     } | 
| 7955 | 7.62k |     ctxt->intSubName = name; | 
| 7956 |  |  | 
| 7957 | 7.62k |     SKIP_BLANKS; | 
| 7958 |  |  | 
| 7959 |  |     /* | 
| 7960 |  |      * Check for public and system identifier (URI) | 
| 7961 |  |      */ | 
| 7962 | 7.62k |     URI = xmlParseExternalID(ctxt, &publicId, 1); | 
| 7963 |  |  | 
| 7964 | 7.62k |     if ((URI != NULL) || (publicId != NULL)) { | 
| 7965 | 324 |         ctxt->hasExternalSubset = 1; | 
| 7966 | 324 |     } | 
| 7967 | 7.62k |     ctxt->extSubURI = URI; | 
| 7968 | 7.62k |     ctxt->extSubSystem = publicId; | 
| 7969 |  |  | 
| 7970 | 7.62k |     SKIP_BLANKS; | 
| 7971 |  |  | 
| 7972 |  |     /* | 
| 7973 |  |      * Create and update the internal subset. | 
| 7974 |  |      */ | 
| 7975 | 7.62k |     if ((ctxt->sax != NULL) && (ctxt->sax->internalSubset != NULL) && | 
| 7976 | 7.62k |   (!ctxt->disableSAX)) | 
| 7977 | 3.61k |   ctxt->sax->internalSubset(ctxt->userData, name, publicId, URI); | 
| 7978 |  |  | 
| 7979 | 7.62k |     if ((RAW != '[') && (RAW != '>')) { | 
| 7980 | 253 |   xmlFatalErr(ctxt, XML_ERR_DOCTYPE_NOT_FINISHED, NULL); | 
| 7981 | 253 |     } | 
| 7982 | 7.62k | } | 
| 7983 |  |  | 
| 7984 |  | /** | 
| 7985 |  |  * Parse the internal subset declaration | 
| 7986 |  |  * | 
| 7987 |  |  *     [28 end] ('[' (markupdecl | PEReference | S)* ']' S?)? '>' | 
| 7988 |  |  * @param ctxt  an XML parser context | 
| 7989 |  |  */ | 
| 7990 |  |  | 
| 7991 |  | static void | 
| 7992 | 7.27k | xmlParseInternalSubset(xmlParserCtxtPtr ctxt) { | 
| 7993 |  |     /* | 
| 7994 |  |      * Is there any DTD definition ? | 
| 7995 |  |      */ | 
| 7996 | 7.27k |     if (RAW == '[') { | 
| 7997 | 7.27k |         int oldInputNr = ctxt->inputNr; | 
| 7998 |  |  | 
| 7999 | 7.27k |         NEXT; | 
| 8000 |  |   /* | 
| 8001 |  |    * Parse the succession of Markup declarations and | 
| 8002 |  |    * PEReferences. | 
| 8003 |  |    * Subsequence (markupdecl | PEReference | S)* | 
| 8004 |  |    */ | 
| 8005 | 7.27k |   SKIP_BLANKS; | 
| 8006 | 205k |         while (1) { | 
| 8007 | 205k |             if (PARSER_STOPPED(ctxt)) { | 
| 8008 | 33 |                 return; | 
| 8009 | 204k |             } else if (ctxt->input->cur >= ctxt->input->end) { | 
| 8010 | 48.5k |                 if (ctxt->inputNr <= oldInputNr) { | 
| 8011 | 2.22k |                 xmlFatalErr(ctxt, XML_ERR_INT_SUBSET_NOT_FINISHED, NULL); | 
| 8012 | 2.22k |                     return; | 
| 8013 | 2.22k |                 } | 
| 8014 | 46.2k |                 xmlPopPE(ctxt); | 
| 8015 | 156k |             } else if ((RAW == ']') && (ctxt->inputNr <= oldInputNr)) { | 
| 8016 | 1.72k |                 NEXT; | 
| 8017 | 1.72k |                 SKIP_BLANKS; | 
| 8018 | 1.72k |                 break; | 
| 8019 | 154k |             } else if ((PARSER_EXTERNAL(ctxt)) && | 
| 8020 | 0 |                        (RAW == '<') && (NXT(1) == '!') && (NXT(2) == '[')) { | 
| 8021 |  |                 /* | 
| 8022 |  |                  * Conditional sections are allowed in external entities | 
| 8023 |  |                  * included by PE References in the internal subset. | 
| 8024 |  |                  */ | 
| 8025 | 0 |                 xmlParseConditionalSections(ctxt); | 
| 8026 | 154k |             } else if ((RAW == '<') && ((NXT(1) == '!') || (NXT(1) == '?'))) { | 
| 8027 | 89.9k |                 xmlParseMarkupDecl(ctxt); | 
| 8028 | 89.9k |             } else if (RAW == '%') { | 
| 8029 | 61.5k |                 xmlParsePERefInternal(ctxt, 1); | 
| 8030 | 61.5k |             } else { | 
| 8031 | 3.29k |                 xmlFatalErr(ctxt, XML_ERR_INT_SUBSET_NOT_FINISHED, NULL); | 
| 8032 |  |  | 
| 8033 | 3.37k |                 while (ctxt->inputNr > oldInputNr) | 
| 8034 | 85 |                     xmlPopPE(ctxt); | 
| 8035 | 3.29k |                 return; | 
| 8036 | 3.29k |             } | 
| 8037 | 197k |             SKIP_BLANKS; | 
| 8038 | 197k |             SHRINK; | 
| 8039 | 197k |             GROW; | 
| 8040 | 197k |         } | 
| 8041 | 7.27k |     } | 
| 8042 |  |  | 
| 8043 |  |     /* | 
| 8044 |  |      * We should be at the end of the DOCTYPE declaration. | 
| 8045 |  |      */ | 
| 8046 | 1.72k |     if (RAW != '>') { | 
| 8047 | 133 |         xmlFatalErr(ctxt, XML_ERR_DOCTYPE_NOT_FINISHED, NULL); | 
| 8048 | 133 |         return; | 
| 8049 | 133 |     } | 
| 8050 | 1.59k |     NEXT; | 
| 8051 | 1.59k | } | 
| 8052 |  |  | 
| 8053 |  | #ifdef LIBXML_SAX1_ENABLED | 
| 8054 |  | /** | 
| 8055 |  |  * Parse an attribute | 
| 8056 |  |  * | 
| 8057 |  |  * @deprecated Internal function, don't use. | 
| 8058 |  |  * | 
| 8059 |  |  *     [41] Attribute ::= Name Eq AttValue | 
| 8060 |  |  * | 
| 8061 |  |  * [ WFC: No External Entity References ] | 
| 8062 |  |  * Attribute values cannot contain direct or indirect entity references | 
| 8063 |  |  * to external entities. | 
| 8064 |  |  * | 
| 8065 |  |  * [ WFC: No < in Attribute Values ] | 
| 8066 |  |  * The replacement text of any entity referred to directly or indirectly in | 
| 8067 |  |  * an attribute value (other than "<") must not contain a <. | 
| 8068 |  |  * | 
| 8069 |  |  * [ VC: Attribute Value Type ] | 
| 8070 |  |  * The attribute must have been declared; the value must be of the type | 
| 8071 |  |  * declared for it. | 
| 8072 |  |  * | 
| 8073 |  |  *     [25] Eq ::= S? '=' S? | 
| 8074 |  |  * | 
| 8075 |  |  * With namespace: | 
| 8076 |  |  * | 
| 8077 |  |  *     [NS 11] Attribute ::= QName Eq AttValue | 
| 8078 |  |  * | 
| 8079 |  |  * Also the case QName == xmlns:??? is handled independently as a namespace | 
| 8080 |  |  * definition. | 
| 8081 |  |  * | 
| 8082 |  |  * @param ctxt  an XML parser context | 
| 8083 |  |  * @param value  a xmlChar ** used to store the value of the attribute | 
| 8084 |  |  * @returns the attribute name, and the value in *value. | 
| 8085 |  |  */ | 
| 8086 |  |  | 
| 8087 |  | const xmlChar * | 
| 8088 | 0 | xmlParseAttribute(xmlParserCtxt *ctxt, xmlChar **value) { | 
| 8089 | 0 |     const xmlChar *name; | 
| 8090 | 0 |     xmlChar *val; | 
| 8091 |  | 
 | 
| 8092 | 0 |     *value = NULL; | 
| 8093 | 0 |     GROW; | 
| 8094 | 0 |     name = xmlParseName(ctxt); | 
| 8095 | 0 |     if (name == NULL) { | 
| 8096 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 8097 | 0 |                  "error parsing attribute name\n"); | 
| 8098 | 0 |         return(NULL); | 
| 8099 | 0 |     } | 
| 8100 |  |  | 
| 8101 |  |     /* | 
| 8102 |  |      * read the value | 
| 8103 |  |      */ | 
| 8104 | 0 |     SKIP_BLANKS; | 
| 8105 | 0 |     if (RAW == '=') { | 
| 8106 | 0 |         NEXT; | 
| 8107 | 0 |   SKIP_BLANKS; | 
| 8108 | 0 |   val = xmlParseAttValue(ctxt); | 
| 8109 | 0 |     } else { | 
| 8110 | 0 |   xmlFatalErrMsgStr(ctxt, XML_ERR_ATTRIBUTE_WITHOUT_VALUE, | 
| 8111 | 0 |          "Specification mandates value for attribute %s\n", name); | 
| 8112 | 0 |   return(name); | 
| 8113 | 0 |     } | 
| 8114 |  |  | 
| 8115 |  |     /* | 
| 8116 |  |      * Check that xml:lang conforms to the specification | 
| 8117 |  |      * No more registered as an error, just generate a warning now | 
| 8118 |  |      * since this was deprecated in XML second edition | 
| 8119 |  |      */ | 
| 8120 | 0 |     if ((ctxt->pedantic) && (xmlStrEqual(name, BAD_CAST "xml:lang"))) { | 
| 8121 | 0 |   if (!xmlCheckLanguageID(val)) { | 
| 8122 | 0 |       xmlWarningMsg(ctxt, XML_WAR_LANG_VALUE, | 
| 8123 | 0 |               "Malformed value for xml:lang : %s\n", | 
| 8124 | 0 |         val, NULL); | 
| 8125 | 0 |   } | 
| 8126 | 0 |     } | 
| 8127 |  |  | 
| 8128 |  |     /* | 
| 8129 |  |      * Check that xml:space conforms to the specification | 
| 8130 |  |      */ | 
| 8131 | 0 |     if (xmlStrEqual(name, BAD_CAST "xml:space")) { | 
| 8132 | 0 |   if (xmlStrEqual(val, BAD_CAST "default")) | 
| 8133 | 0 |       *(ctxt->space) = 0; | 
| 8134 | 0 |   else if (xmlStrEqual(val, BAD_CAST "preserve")) | 
| 8135 | 0 |       *(ctxt->space) = 1; | 
| 8136 | 0 |   else { | 
| 8137 | 0 |     xmlWarningMsg(ctxt, XML_WAR_SPACE_VALUE, | 
| 8138 | 0 | "Invalid value \"%s\" for xml:space : \"default\" or \"preserve\" expected\n", | 
| 8139 | 0 |                                  val, NULL); | 
| 8140 | 0 |   } | 
| 8141 | 0 |     } | 
| 8142 |  | 
 | 
| 8143 | 0 |     *value = val; | 
| 8144 | 0 |     return(name); | 
| 8145 | 0 | } | 
| 8146 |  |  | 
| 8147 |  | /** | 
| 8148 |  |  * Parse a start tag. Always consumes '<'. | 
| 8149 |  |  * | 
| 8150 |  |  * @deprecated Internal function, don't use. | 
| 8151 |  |  * | 
| 8152 |  |  *     [40] STag ::= '<' Name (S Attribute)* S? '>' | 
| 8153 |  |  * | 
| 8154 |  |  * [ WFC: Unique Att Spec ] | 
| 8155 |  |  * No attribute name may appear more than once in the same start-tag or | 
| 8156 |  |  * empty-element tag. | 
| 8157 |  |  * | 
| 8158 |  |  *     [44] EmptyElemTag ::= '<' Name (S Attribute)* S? '/>' | 
| 8159 |  |  * | 
| 8160 |  |  * [ WFC: Unique Att Spec ] | 
| 8161 |  |  * No attribute name may appear more than once in the same start-tag or | 
| 8162 |  |  * empty-element tag. | 
| 8163 |  |  * | 
| 8164 |  |  * With namespace: | 
| 8165 |  |  * | 
| 8166 |  |  *     [NS 8] STag ::= '<' QName (S Attribute)* S? '>' | 
| 8167 |  |  * | 
| 8168 |  |  *     [NS 10] EmptyElement ::= '<' QName (S Attribute)* S? '/>' | 
| 8169 |  |  * | 
| 8170 |  |  * @param ctxt  an XML parser context | 
| 8171 |  |  * @returns the element name parsed | 
| 8172 |  |  */ | 
| 8173 |  |  | 
| 8174 |  | const xmlChar * | 
| 8175 | 0 | xmlParseStartTag(xmlParserCtxt *ctxt) { | 
| 8176 | 0 |     const xmlChar *name; | 
| 8177 | 0 |     const xmlChar *attname; | 
| 8178 | 0 |     xmlChar *attvalue; | 
| 8179 | 0 |     const xmlChar **atts = ctxt->atts; | 
| 8180 | 0 |     int nbatts = 0; | 
| 8181 | 0 |     int maxatts = ctxt->maxatts; | 
| 8182 | 0 |     int i; | 
| 8183 |  | 
 | 
| 8184 | 0 |     if (RAW != '<') return(NULL); | 
| 8185 | 0 |     NEXT1; | 
| 8186 |  | 
 | 
| 8187 | 0 |     name = xmlParseName(ctxt); | 
| 8188 | 0 |     if (name == NULL) { | 
| 8189 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 8190 | 0 |        "xmlParseStartTag: invalid element name\n"); | 
| 8191 | 0 |         return(NULL); | 
| 8192 | 0 |     } | 
| 8193 |  |  | 
| 8194 |  |     /* | 
| 8195 |  |      * Now parse the attributes, it ends up with the ending | 
| 8196 |  |      * | 
| 8197 |  |      * (S Attribute)* S? | 
| 8198 |  |      */ | 
| 8199 | 0 |     SKIP_BLANKS; | 
| 8200 | 0 |     GROW; | 
| 8201 |  | 
 | 
| 8202 | 0 |     while (((RAW != '>') && | 
| 8203 | 0 |      ((RAW != '/') || (NXT(1) != '>')) && | 
| 8204 | 0 |      (IS_BYTE_CHAR(RAW))) && (PARSER_STOPPED(ctxt) == 0)) { | 
| 8205 | 0 |   attname = xmlParseAttribute(ctxt, &attvalue); | 
| 8206 | 0 |         if (attname == NULL) | 
| 8207 | 0 |       break; | 
| 8208 | 0 |         if (attvalue != NULL) { | 
| 8209 |  |       /* | 
| 8210 |  |        * [ WFC: Unique Att Spec ] | 
| 8211 |  |        * No attribute name may appear more than once in the same | 
| 8212 |  |        * start-tag or empty-element tag. | 
| 8213 |  |        */ | 
| 8214 | 0 |       for (i = 0; i < nbatts;i += 2) { | 
| 8215 | 0 |           if (xmlStrEqual(atts[i], attname)) { | 
| 8216 | 0 |         xmlErrAttributeDup(ctxt, NULL, attname); | 
| 8217 | 0 |         goto failed; | 
| 8218 | 0 |     } | 
| 8219 | 0 |       } | 
| 8220 |  |       /* | 
| 8221 |  |        * Add the pair to atts | 
| 8222 |  |        */ | 
| 8223 | 0 |       if (nbatts + 4 > maxatts) { | 
| 8224 | 0 |           const xmlChar **n; | 
| 8225 | 0 |                 int newSize; | 
| 8226 |  | 
 | 
| 8227 | 0 |                 newSize = xmlGrowCapacity(maxatts, sizeof(n[0]) * 2, | 
| 8228 | 0 |                                           11, XML_MAX_ATTRS); | 
| 8229 | 0 |                 if (newSize < 0) { | 
| 8230 | 0 |         xmlErrMemory(ctxt); | 
| 8231 | 0 |         goto failed; | 
| 8232 | 0 |     } | 
| 8233 | 0 | #ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION | 
| 8234 | 0 |                 if (newSize < 2) | 
| 8235 | 0 |                     newSize = 2; | 
| 8236 | 0 | #endif | 
| 8237 | 0 |           n = xmlRealloc(atts, newSize * sizeof(n[0]) * 2); | 
| 8238 | 0 |     if (n == NULL) { | 
| 8239 | 0 |         xmlErrMemory(ctxt); | 
| 8240 | 0 |         goto failed; | 
| 8241 | 0 |     } | 
| 8242 | 0 |     atts = n; | 
| 8243 | 0 |                 maxatts = newSize * 2; | 
| 8244 | 0 |     ctxt->atts = atts; | 
| 8245 | 0 |     ctxt->maxatts = maxatts; | 
| 8246 | 0 |       } | 
| 8247 |  |  | 
| 8248 | 0 |       atts[nbatts++] = attname; | 
| 8249 | 0 |       atts[nbatts++] = attvalue; | 
| 8250 | 0 |       atts[nbatts] = NULL; | 
| 8251 | 0 |       atts[nbatts + 1] = NULL; | 
| 8252 |  | 
 | 
| 8253 | 0 |             attvalue = NULL; | 
| 8254 | 0 |   } | 
| 8255 |  |  | 
| 8256 | 0 | failed: | 
| 8257 |  | 
 | 
| 8258 | 0 |         if (attvalue != NULL) | 
| 8259 | 0 |             xmlFree(attvalue); | 
| 8260 |  | 
 | 
| 8261 | 0 |   GROW | 
| 8262 | 0 |   if ((RAW == '>') || (((RAW == '/') && (NXT(1) == '>')))) | 
| 8263 | 0 |       break; | 
| 8264 | 0 |   if (SKIP_BLANKS == 0) { | 
| 8265 | 0 |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 8266 | 0 |          "attributes construct error\n"); | 
| 8267 | 0 |   } | 
| 8268 | 0 |   SHRINK; | 
| 8269 | 0 |         GROW; | 
| 8270 | 0 |     } | 
| 8271 |  |  | 
| 8272 |  |     /* | 
| 8273 |  |      * SAX: Start of Element ! | 
| 8274 |  |      */ | 
| 8275 | 0 |     if ((ctxt->sax != NULL) && (ctxt->sax->startElement != NULL) && | 
| 8276 | 0 |   (!ctxt->disableSAX)) { | 
| 8277 | 0 |   if (nbatts > 0) | 
| 8278 | 0 |       ctxt->sax->startElement(ctxt->userData, name, atts); | 
| 8279 | 0 |   else | 
| 8280 | 0 |       ctxt->sax->startElement(ctxt->userData, name, NULL); | 
| 8281 | 0 |     } | 
| 8282 |  | 
 | 
| 8283 | 0 |     if (atts != NULL) { | 
| 8284 |  |         /* Free only the content strings */ | 
| 8285 | 0 |         for (i = 1;i < nbatts;i+=2) | 
| 8286 | 0 |       if (atts[i] != NULL) | 
| 8287 | 0 |          xmlFree((xmlChar *) atts[i]); | 
| 8288 | 0 |     } | 
| 8289 | 0 |     return(name); | 
| 8290 | 0 | } | 
| 8291 |  |  | 
| 8292 |  | /** | 
| 8293 |  |  * Parse an end tag. Always consumes '</'. | 
| 8294 |  |  * | 
| 8295 |  |  *     [42] ETag ::= '</' Name S? '>' | 
| 8296 |  |  * | 
| 8297 |  |  * With namespace | 
| 8298 |  |  * | 
| 8299 |  |  *     [NS 9] ETag ::= '</' QName S? '>' | 
| 8300 |  |  * @param ctxt  an XML parser context | 
| 8301 |  |  * @param line  line of the start tag | 
| 8302 |  |  */ | 
| 8303 |  |  | 
| 8304 |  | static void | 
| 8305 | 0 | xmlParseEndTag1(xmlParserCtxtPtr ctxt, int line) { | 
| 8306 | 0 |     const xmlChar *name; | 
| 8307 |  | 
 | 
| 8308 | 0 |     GROW; | 
| 8309 | 0 |     if ((RAW != '<') || (NXT(1) != '/')) { | 
| 8310 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_LTSLASH_REQUIRED, | 
| 8311 | 0 |            "xmlParseEndTag: '</' not found\n"); | 
| 8312 | 0 |   return; | 
| 8313 | 0 |     } | 
| 8314 | 0 |     SKIP(2); | 
| 8315 |  | 
 | 
| 8316 | 0 |     name = xmlParseNameAndCompare(ctxt,ctxt->name); | 
| 8317 |  |  | 
| 8318 |  |     /* | 
| 8319 |  |      * We should definitely be at the ending "S? '>'" part | 
| 8320 |  |      */ | 
| 8321 | 0 |     GROW; | 
| 8322 | 0 |     SKIP_BLANKS; | 
| 8323 | 0 |     if ((!IS_BYTE_CHAR(RAW)) || (RAW != '>')) { | 
| 8324 | 0 |   xmlFatalErr(ctxt, XML_ERR_GT_REQUIRED, NULL); | 
| 8325 | 0 |     } else | 
| 8326 | 0 |   NEXT1; | 
| 8327 |  |  | 
| 8328 |  |     /* | 
| 8329 |  |      * [ WFC: Element Type Match ] | 
| 8330 |  |      * The Name in an element's end-tag must match the element type in the | 
| 8331 |  |      * start-tag. | 
| 8332 |  |      * | 
| 8333 |  |      */ | 
| 8334 | 0 |     if (name != (xmlChar*)1) { | 
| 8335 | 0 |         if (name == NULL) name = BAD_CAST "unparsable"; | 
| 8336 | 0 |         xmlFatalErrMsgStrIntStr(ctxt, XML_ERR_TAG_NAME_MISMATCH, | 
| 8337 | 0 |          "Opening and ending tag mismatch: %s line %d and %s\n", | 
| 8338 | 0 |                     ctxt->name, line, name); | 
| 8339 | 0 |     } | 
| 8340 |  |  | 
| 8341 |  |     /* | 
| 8342 |  |      * SAX: End of Tag | 
| 8343 |  |      */ | 
| 8344 | 0 |     if ((ctxt->sax != NULL) && (ctxt->sax->endElement != NULL) && | 
| 8345 | 0 |   (!ctxt->disableSAX)) | 
| 8346 | 0 |         ctxt->sax->endElement(ctxt->userData, ctxt->name); | 
| 8347 |  | 
 | 
| 8348 | 0 |     namePop(ctxt); | 
| 8349 | 0 |     spacePop(ctxt); | 
| 8350 | 0 | } | 
| 8351 |  |  | 
| 8352 |  | /** | 
| 8353 |  |  * Parse an end of tag | 
| 8354 |  |  * | 
| 8355 |  |  * @deprecated Internal function, don't use. | 
| 8356 |  |  * | 
| 8357 |  |  *     [42] ETag ::= '</' Name S? '>' | 
| 8358 |  |  * | 
| 8359 |  |  * With namespace | 
| 8360 |  |  * | 
| 8361 |  |  *     [NS 9] ETag ::= '</' QName S? '>' | 
| 8362 |  |  * @param ctxt  an XML parser context | 
| 8363 |  |  */ | 
| 8364 |  |  | 
| 8365 |  | void | 
| 8366 | 0 | xmlParseEndTag(xmlParserCtxt *ctxt) { | 
| 8367 | 0 |     xmlParseEndTag1(ctxt, 0); | 
| 8368 | 0 | } | 
| 8369 |  | #endif /* LIBXML_SAX1_ENABLED */ | 
| 8370 |  |  | 
| 8371 |  | /************************************************************************ | 
| 8372 |  |  *                  * | 
| 8373 |  |  *          SAX 2 specific operations       * | 
| 8374 |  |  *                  * | 
| 8375 |  |  ************************************************************************/ | 
| 8376 |  |  | 
| 8377 |  | /** | 
| 8378 |  |  * Parse an XML Namespace QName | 
| 8379 |  |  * | 
| 8380 |  |  *     [6]  QName  ::= (Prefix ':')? LocalPart | 
| 8381 |  |  *     [7]  Prefix  ::= NCName | 
| 8382 |  |  *     [8]  LocalPart  ::= NCName | 
| 8383 |  |  * | 
| 8384 |  |  * @param ctxt  an XML parser context | 
| 8385 |  |  * @param prefix  pointer to store the prefix part | 
| 8386 |  |  * @returns the Name parsed or NULL | 
| 8387 |  |  */ | 
| 8388 |  |  | 
| 8389 |  | static xmlHashedString | 
| 8390 | 4.09M | xmlParseQNameHashed(xmlParserCtxtPtr ctxt, xmlHashedString *prefix) { | 
| 8391 | 4.09M |     xmlHashedString l, p; | 
| 8392 | 4.09M |     int start, isNCName = 0; | 
| 8393 |  |  | 
| 8394 | 4.09M |     l.name = NULL; | 
| 8395 | 4.09M |     p.name = NULL; | 
| 8396 |  |  | 
| 8397 | 4.09M |     GROW; | 
| 8398 | 4.09M |     start = CUR_PTR - BASE_PTR; | 
| 8399 |  |  | 
| 8400 | 4.09M |     l = xmlParseNCName(ctxt); | 
| 8401 | 4.09M |     if (l.name != NULL) { | 
| 8402 | 1.12M |         isNCName = 1; | 
| 8403 | 1.12M |         if (CUR == ':') { | 
| 8404 | 96.8k |             NEXT; | 
| 8405 | 96.8k |             p = l; | 
| 8406 | 96.8k |             l = xmlParseNCName(ctxt); | 
| 8407 | 96.8k |         } | 
| 8408 | 1.12M |     } | 
| 8409 | 4.09M |     if ((l.name == NULL) || (CUR == ':')) { | 
| 8410 | 2.98M |         xmlChar *tmp; | 
| 8411 |  |  | 
| 8412 | 2.98M |         l.name = NULL; | 
| 8413 | 2.98M |         p.name = NULL; | 
| 8414 | 2.98M |         if ((isNCName == 0) && (CUR != ':')) | 
| 8415 | 2.96M |             return(l); | 
| 8416 | 21.9k |         tmp = xmlParseNmtoken(ctxt); | 
| 8417 | 21.9k |         if (tmp != NULL) | 
| 8418 | 11.7k |             xmlFree(tmp); | 
| 8419 | 21.9k |         l = xmlDictLookupHashed(ctxt->dict, BASE_PTR + start, | 
| 8420 | 21.9k |                                 CUR_PTR - (BASE_PTR + start)); | 
| 8421 | 21.9k |         if (l.name == NULL) { | 
| 8422 | 0 |             xmlErrMemory(ctxt); | 
| 8423 | 0 |             return(l); | 
| 8424 | 0 |         } | 
| 8425 | 21.9k |         xmlNsErr(ctxt, XML_NS_ERR_QNAME, | 
| 8426 | 21.9k |                  "Failed to parse QName '%s'\n", l.name, NULL, NULL); | 
| 8427 | 21.9k |     } | 
| 8428 |  |  | 
| 8429 | 1.13M |     *prefix = p; | 
| 8430 | 1.13M |     return(l); | 
| 8431 | 4.09M | } | 
| 8432 |  |  | 
| 8433 |  | /** | 
| 8434 |  |  * Parse an XML Namespace QName | 
| 8435 |  |  * | 
| 8436 |  |  *     [6]  QName  ::= (Prefix ':')? LocalPart | 
| 8437 |  |  *     [7]  Prefix  ::= NCName | 
| 8438 |  |  *     [8]  LocalPart  ::= NCName | 
| 8439 |  |  * | 
| 8440 |  |  * @param ctxt  an XML parser context | 
| 8441 |  |  * @param prefix  pointer to store the prefix part | 
| 8442 |  |  * @returns the Name parsed or NULL | 
| 8443 |  |  */ | 
| 8444 |  |  | 
| 8445 |  | static const xmlChar * | 
| 8446 | 11.4k | xmlParseQName(xmlParserCtxtPtr ctxt, const xmlChar **prefix) { | 
| 8447 | 11.4k |     xmlHashedString n, p; | 
| 8448 |  |  | 
| 8449 | 11.4k |     n = xmlParseQNameHashed(ctxt, &p); | 
| 8450 | 11.4k |     if (n.name == NULL) | 
| 8451 | 4.54k |         return(NULL); | 
| 8452 | 6.89k |     *prefix = p.name; | 
| 8453 | 6.89k |     return(n.name); | 
| 8454 | 11.4k | } | 
| 8455 |  |  | 
| 8456 |  | /** | 
| 8457 |  |  * Parse an XML name and compares for match | 
| 8458 |  |  * (specialized for endtag parsing) | 
| 8459 |  |  * | 
| 8460 |  |  * @param ctxt  an XML parser context | 
| 8461 |  |  * @param name  the localname | 
| 8462 |  |  * @param prefix  the prefix, if any. | 
| 8463 |  |  * @returns NULL for an illegal name, (xmlChar*) 1 for success | 
| 8464 |  |  * and the name for mismatch | 
| 8465 |  |  */ | 
| 8466 |  |  | 
| 8467 |  | static const xmlChar * | 
| 8468 |  | xmlParseQNameAndCompare(xmlParserCtxtPtr ctxt, xmlChar const *name, | 
| 8469 | 15.5k |                         xmlChar const *prefix) { | 
| 8470 | 15.5k |     const xmlChar *cmp; | 
| 8471 | 15.5k |     const xmlChar *in; | 
| 8472 | 15.5k |     const xmlChar *ret; | 
| 8473 | 15.5k |     const xmlChar *prefix2; | 
| 8474 |  |  | 
| 8475 | 15.5k |     if (prefix == NULL) return(xmlParseNameAndCompare(ctxt, name)); | 
| 8476 |  |  | 
| 8477 | 15.5k |     GROW; | 
| 8478 | 15.5k |     in = ctxt->input->cur; | 
| 8479 |  |  | 
| 8480 | 15.5k |     cmp = prefix; | 
| 8481 | 26.4k |     while (*in != 0 && *in == *cmp) { | 
| 8482 | 10.9k |   ++in; | 
| 8483 | 10.9k |   ++cmp; | 
| 8484 | 10.9k |     } | 
| 8485 | 15.5k |     if ((*cmp == 0) && (*in == ':')) { | 
| 8486 | 8.93k |         in++; | 
| 8487 | 8.93k |   cmp = name; | 
| 8488 | 18.0k |   while (*in != 0 && *in == *cmp) { | 
| 8489 | 9.11k |       ++in; | 
| 8490 | 9.11k |       ++cmp; | 
| 8491 | 9.11k |   } | 
| 8492 | 8.93k |   if (*cmp == 0 && (*in == '>' || IS_BLANK_CH (*in))) { | 
| 8493 |  |       /* success */ | 
| 8494 | 4.08k |             ctxt->input->col += in - ctxt->input->cur; | 
| 8495 | 4.08k |       ctxt->input->cur = in; | 
| 8496 | 4.08k |       return((const xmlChar*) 1); | 
| 8497 | 4.08k |   } | 
| 8498 | 8.93k |     } | 
| 8499 |  |     /* | 
| 8500 |  |      * all strings coms from the dictionary, equality can be done directly | 
| 8501 |  |      */ | 
| 8502 | 11.4k |     ret = xmlParseQName (ctxt, &prefix2); | 
| 8503 | 11.4k |     if (ret == NULL) | 
| 8504 | 4.54k |         return(NULL); | 
| 8505 | 6.89k |     if ((ret == name) && (prefix == prefix2)) | 
| 8506 | 4.24k |   return((const xmlChar*) 1); | 
| 8507 | 2.65k |     return ret; | 
| 8508 | 6.89k | } | 
| 8509 |  |  | 
| 8510 |  | /** | 
| 8511 |  |  * Parse an attribute in the new SAX2 framework. | 
| 8512 |  |  * | 
| 8513 |  |  * @param ctxt  an XML parser context | 
| 8514 |  |  * @param pref  the element prefix | 
| 8515 |  |  * @param elem  the element name | 
| 8516 |  |  * @param hprefix  resulting attribute prefix | 
| 8517 |  |  * @param value  resulting value of the attribute | 
| 8518 |  |  * @param len  resulting length of the attribute | 
| 8519 |  |  * @param alloc  resulting indicator if the attribute was allocated | 
| 8520 |  |  * @returns the attribute name, and the value in *value, . | 
| 8521 |  |  */ | 
| 8522 |  |  | 
| 8523 |  | static xmlHashedString | 
| 8524 |  | xmlParseAttribute2(xmlParserCtxtPtr ctxt, | 
| 8525 |  |                    const xmlChar * pref, const xmlChar * elem, | 
| 8526 |  |                    xmlHashedString * hprefix, xmlChar ** value, | 
| 8527 |  |                    int *len, int *alloc) | 
| 8528 | 810k | { | 
| 8529 | 810k |     xmlHashedString hname; | 
| 8530 | 810k |     const xmlChar *prefix, *name; | 
| 8531 | 810k |     xmlChar *val = NULL, *internal_val = NULL; | 
| 8532 | 810k |     int special = 0; | 
| 8533 | 810k |     int isNamespace; | 
| 8534 | 810k |     int flags; | 
| 8535 |  |  | 
| 8536 | 810k |     *value = NULL; | 
| 8537 | 810k |     GROW; | 
| 8538 | 810k |     hname = xmlParseQNameHashed(ctxt, hprefix); | 
| 8539 | 810k |     if (hname.name == NULL) { | 
| 8540 | 181k |         xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 8541 | 181k |                        "error parsing attribute name\n"); | 
| 8542 | 181k |         return(hname); | 
| 8543 | 181k |     } | 
| 8544 | 628k |     name = hname.name; | 
| 8545 | 628k |     prefix = hprefix->name; | 
| 8546 |  |  | 
| 8547 |  |     /* | 
| 8548 |  |      * get the type if needed | 
| 8549 |  |      */ | 
| 8550 | 628k |     if (ctxt->attsSpecial != NULL) { | 
| 8551 | 76.8k |         special = XML_PTR_TO_INT(xmlHashQLookup2(ctxt->attsSpecial, pref, elem, | 
| 8552 | 76.8k |                                               prefix, name)); | 
| 8553 | 76.8k |     } | 
| 8554 |  |  | 
| 8555 |  |     /* | 
| 8556 |  |      * read the value | 
| 8557 |  |      */ | 
| 8558 | 628k |     SKIP_BLANKS; | 
| 8559 | 628k |     if (RAW != '=') { | 
| 8560 | 24.1k |         xmlFatalErrMsgStr(ctxt, XML_ERR_ATTRIBUTE_WITHOUT_VALUE, | 
| 8561 | 24.1k |                           "Specification mandates value for attribute %s\n", | 
| 8562 | 24.1k |                           name); | 
| 8563 | 24.1k |         goto error; | 
| 8564 | 24.1k |     } | 
| 8565 |  |  | 
| 8566 |  |  | 
| 8567 | 604k |     NEXT; | 
| 8568 | 604k |     SKIP_BLANKS; | 
| 8569 | 604k |     flags = 0; | 
| 8570 | 604k |     isNamespace = (((prefix == NULL) && (name == ctxt->str_xmlns)) || | 
| 8571 | 539k |                    (prefix == ctxt->str_xmlns)); | 
| 8572 | 604k |     val = xmlParseAttValueInternal(ctxt, len, &flags, special, | 
| 8573 | 604k |                                    isNamespace); | 
| 8574 | 604k |     if (val == NULL) | 
| 8575 | 8.05k |         goto error; | 
| 8576 |  |  | 
| 8577 | 596k |     *alloc = (flags & XML_ATTVAL_ALLOC) != 0; | 
| 8578 |  |  | 
| 8579 | 596k | #ifdef LIBXML_VALID_ENABLED | 
| 8580 | 596k |     if ((ctxt->validate) && | 
| 8581 | 0 |         (ctxt->standalone == 1) && | 
| 8582 | 0 |         (special & XML_SPECIAL_EXTERNAL) && | 
| 8583 | 0 |         (flags & XML_ATTVAL_NORM_CHANGE)) { | 
| 8584 | 0 |         xmlValidityError(ctxt, XML_DTD_NOT_STANDALONE, | 
| 8585 | 0 |                          "standalone: normalization of attribute %s on %s " | 
| 8586 | 0 |                          "by external subset declaration\n", | 
| 8587 | 0 |                          name, elem); | 
| 8588 | 0 |     } | 
| 8589 | 596k | #endif | 
| 8590 |  |  | 
| 8591 | 596k |     if (prefix == ctxt->str_xml) { | 
| 8592 |  |         /* | 
| 8593 |  |          * Check that xml:lang conforms to the specification | 
| 8594 |  |          * No more registered as an error, just generate a warning now | 
| 8595 |  |          * since this was deprecated in XML second edition | 
| 8596 |  |          */ | 
| 8597 | 9.47k |         if ((ctxt->pedantic) && (xmlStrEqual(name, BAD_CAST "lang"))) { | 
| 8598 | 0 |             internal_val = xmlStrndup(val, *len); | 
| 8599 | 0 |             if (internal_val == NULL) | 
| 8600 | 0 |                 goto mem_error; | 
| 8601 | 0 |             if (!xmlCheckLanguageID(internal_val)) { | 
| 8602 | 0 |                 xmlWarningMsg(ctxt, XML_WAR_LANG_VALUE, | 
| 8603 | 0 |                               "Malformed value for xml:lang : %s\n", | 
| 8604 | 0 |                               internal_val, NULL); | 
| 8605 | 0 |             } | 
| 8606 | 0 |         } | 
| 8607 |  |  | 
| 8608 |  |         /* | 
| 8609 |  |          * Check that xml:space conforms to the specification | 
| 8610 |  |          */ | 
| 8611 | 9.47k |         if (xmlStrEqual(name, BAD_CAST "space")) { | 
| 8612 | 1.13k |             internal_val = xmlStrndup(val, *len); | 
| 8613 | 1.13k |             if (internal_val == NULL) | 
| 8614 | 0 |                 goto mem_error; | 
| 8615 | 1.13k |             if (xmlStrEqual(internal_val, BAD_CAST "default")) | 
| 8616 | 495 |                 *(ctxt->space) = 0; | 
| 8617 | 643 |             else if (xmlStrEqual(internal_val, BAD_CAST "preserve")) | 
| 8618 | 195 |                 *(ctxt->space) = 1; | 
| 8619 | 448 |             else { | 
| 8620 | 448 |                 xmlWarningMsg(ctxt, XML_WAR_SPACE_VALUE, | 
| 8621 | 448 |                               "Invalid value \"%s\" for xml:space : \"default\" or \"preserve\" expected\n", | 
| 8622 | 448 |                               internal_val, NULL); | 
| 8623 | 448 |             } | 
| 8624 | 1.13k |         } | 
| 8625 | 9.47k |         if (internal_val) { | 
| 8626 | 1.13k |             xmlFree(internal_val); | 
| 8627 | 1.13k |         } | 
| 8628 | 9.47k |     } | 
| 8629 |  |  | 
| 8630 | 596k |     *value = val; | 
| 8631 | 596k |     return (hname); | 
| 8632 |  |  | 
| 8633 | 0 | mem_error: | 
| 8634 | 0 |     xmlErrMemory(ctxt); | 
| 8635 | 32.1k | error: | 
| 8636 | 32.1k |     if ((val != NULL) && (*alloc != 0)) | 
| 8637 | 0 |         xmlFree(val); | 
| 8638 | 32.1k |     return(hname); | 
| 8639 | 0 | } | 
| 8640 |  |  | 
| 8641 |  | /** | 
| 8642 |  |  * Inserts a new attribute into the hash table. | 
| 8643 |  |  * | 
| 8644 |  |  * @param ctxt  parser context | 
| 8645 |  |  * @param size  size of the hash table | 
| 8646 |  |  * @param name  attribute name | 
| 8647 |  |  * @param uri  namespace uri | 
| 8648 |  |  * @param hashValue  combined hash value of name and uri | 
| 8649 |  |  * @param aindex  attribute index (this is a multiple of 5) | 
| 8650 |  |  * @returns INT_MAX if no existing attribute was found, the attribute | 
| 8651 |  |  * index if an attribute was found, -1 if a memory allocation failed. | 
| 8652 |  |  */ | 
| 8653 |  | static int | 
| 8654 |  | xmlAttrHashInsert(xmlParserCtxtPtr ctxt, unsigned size, const xmlChar *name, | 
| 8655 | 690k |                   const xmlChar *uri, unsigned hashValue, int aindex) { | 
| 8656 | 690k |     xmlAttrHashBucket *table = ctxt->attrHash; | 
| 8657 | 690k |     xmlAttrHashBucket *bucket; | 
| 8658 | 690k |     unsigned hindex; | 
| 8659 |  |  | 
| 8660 | 690k |     hindex = hashValue & (size - 1); | 
| 8661 | 690k |     bucket = &table[hindex]; | 
| 8662 |  |  | 
| 8663 | 760k |     while (bucket->index >= 0) { | 
| 8664 | 555k |         const xmlChar **atts = &ctxt->atts[bucket->index]; | 
| 8665 |  |  | 
| 8666 | 555k |         if (name == atts[0]) { | 
| 8667 | 496k |             int nsIndex = XML_PTR_TO_INT(atts[2]); | 
| 8668 |  |  | 
| 8669 | 496k |             if ((nsIndex == NS_INDEX_EMPTY) ? (uri == NULL) : | 
| 8670 | 496k |                 (nsIndex == NS_INDEX_XML) ? (uri == ctxt->str_xml_ns) : | 
| 8671 | 20.5k |                 (uri == ctxt->nsTab[nsIndex * 2 + 1])) | 
| 8672 | 486k |                 return(bucket->index); | 
| 8673 | 496k |         } | 
| 8674 |  |  | 
| 8675 | 69.4k |         hindex++; | 
| 8676 | 69.4k |         bucket++; | 
| 8677 | 69.4k |         if (hindex >= size) { | 
| 8678 | 22.6k |             hindex = 0; | 
| 8679 | 22.6k |             bucket = table; | 
| 8680 | 22.6k |         } | 
| 8681 | 69.4k |     } | 
| 8682 |  |  | 
| 8683 | 204k |     bucket->index = aindex; | 
| 8684 |  |  | 
| 8685 | 204k |     return(INT_MAX); | 
| 8686 | 690k | } | 
| 8687 |  |  | 
| 8688 |  | static int | 
| 8689 |  | xmlAttrHashInsertQName(xmlParserCtxtPtr ctxt, unsigned size, | 
| 8690 |  |                        const xmlChar *name, const xmlChar *prefix, | 
| 8691 | 17.1k |                        unsigned hashValue, int aindex) { | 
| 8692 | 17.1k |     xmlAttrHashBucket *table = ctxt->attrHash; | 
| 8693 | 17.1k |     xmlAttrHashBucket *bucket; | 
| 8694 | 17.1k |     unsigned hindex; | 
| 8695 |  |  | 
| 8696 | 17.1k |     hindex = hashValue & (size - 1); | 
| 8697 | 17.1k |     bucket = &table[hindex]; | 
| 8698 |  |  | 
| 8699 | 24.4k |     while (bucket->index >= 0) { | 
| 8700 | 19.5k |         const xmlChar **atts = &ctxt->atts[bucket->index]; | 
| 8701 |  |  | 
| 8702 | 19.5k |         if ((name == atts[0]) && (prefix == atts[1])) | 
| 8703 | 12.2k |             return(bucket->index); | 
| 8704 |  |  | 
| 8705 | 7.31k |         hindex++; | 
| 8706 | 7.31k |         bucket++; | 
| 8707 | 7.31k |         if (hindex >= size) { | 
| 8708 | 897 |             hindex = 0; | 
| 8709 | 897 |             bucket = table; | 
| 8710 | 897 |         } | 
| 8711 | 7.31k |     } | 
| 8712 |  |  | 
| 8713 | 4.89k |     bucket->index = aindex; | 
| 8714 |  |  | 
| 8715 | 4.89k |     return(INT_MAX); | 
| 8716 | 17.1k | } | 
| 8717 |  | /** | 
| 8718 |  |  * Parse a start tag. Always consumes '<'. | 
| 8719 |  |  * | 
| 8720 |  |  * This routine is called when running SAX2 parsing | 
| 8721 |  |  * | 
| 8722 |  |  *     [40] STag ::= '<' Name (S Attribute)* S? '>' | 
| 8723 |  |  * | 
| 8724 |  |  * [ WFC: Unique Att Spec ] | 
| 8725 |  |  * No attribute name may appear more than once in the same start-tag or | 
| 8726 |  |  * empty-element tag. | 
| 8727 |  |  * | 
| 8728 |  |  *     [44] EmptyElemTag ::= '<' Name (S Attribute)* S? '/>' | 
| 8729 |  |  * | 
| 8730 |  |  * [ WFC: Unique Att Spec ] | 
| 8731 |  |  * No attribute name may appear more than once in the same start-tag or | 
| 8732 |  |  * empty-element tag. | 
| 8733 |  |  * | 
| 8734 |  |  * With namespace: | 
| 8735 |  |  * | 
| 8736 |  |  *     [NS 8] STag ::= '<' QName (S Attribute)* S? '>' | 
| 8737 |  |  * | 
| 8738 |  |  *     [NS 10] EmptyElement ::= '<' QName (S Attribute)* S? '/>' | 
| 8739 |  |  * | 
| 8740 |  |  * @param ctxt  an XML parser context | 
| 8741 |  |  * @param pref  resulting namespace prefix | 
| 8742 |  |  * @param URI  resulting namespace URI | 
| 8743 |  |  * @param nbNsPtr  resulting number of namespace declarations | 
| 8744 |  |  * @returns the element name parsed | 
| 8745 |  |  */ | 
| 8746 |  |  | 
| 8747 |  | static const xmlChar * | 
| 8748 |  | xmlParseStartTag2(xmlParserCtxtPtr ctxt, const xmlChar **pref, | 
| 8749 | 3.27M |                   const xmlChar **URI, int *nbNsPtr) { | 
| 8750 | 3.27M |     xmlHashedString hlocalname; | 
| 8751 | 3.27M |     xmlHashedString hprefix; | 
| 8752 | 3.27M |     xmlHashedString hattname; | 
| 8753 | 3.27M |     xmlHashedString haprefix; | 
| 8754 | 3.27M |     const xmlChar *localname; | 
| 8755 | 3.27M |     const xmlChar *prefix; | 
| 8756 | 3.27M |     const xmlChar *attname; | 
| 8757 | 3.27M |     const xmlChar *aprefix; | 
| 8758 | 3.27M |     const xmlChar *uri; | 
| 8759 | 3.27M |     xmlChar *attvalue = NULL; | 
| 8760 | 3.27M |     const xmlChar **atts = ctxt->atts; | 
| 8761 | 3.27M |     unsigned attrHashSize = 0; | 
| 8762 | 3.27M |     int maxatts = ctxt->maxatts; | 
| 8763 | 3.27M |     int nratts, nbatts, nbdef; | 
| 8764 | 3.27M |     int i, j, nbNs, nbTotalDef, attval, nsIndex, maxAtts; | 
| 8765 | 3.27M |     int alloc = 0; | 
| 8766 | 3.27M |     int numNsErr = 0; | 
| 8767 | 3.27M |     int numDupErr = 0; | 
| 8768 |  |  | 
| 8769 | 3.27M |     if (RAW != '<') return(NULL); | 
| 8770 | 3.27M |     NEXT1; | 
| 8771 |  |  | 
| 8772 | 3.27M |     nbatts = 0; | 
| 8773 | 3.27M |     nratts = 0; | 
| 8774 | 3.27M |     nbdef = 0; | 
| 8775 | 3.27M |     nbNs = 0; | 
| 8776 | 3.27M |     nbTotalDef = 0; | 
| 8777 | 3.27M |     attval = 0; | 
| 8778 |  |  | 
| 8779 | 3.27M |     if (xmlParserNsStartElement(ctxt->nsdb) < 0) { | 
| 8780 | 0 |         xmlErrMemory(ctxt); | 
| 8781 | 0 |         return(NULL); | 
| 8782 | 0 |     } | 
| 8783 |  |  | 
| 8784 | 3.27M |     hlocalname = xmlParseQNameHashed(ctxt, &hprefix); | 
| 8785 | 3.27M |     if (hlocalname.name == NULL) { | 
| 8786 | 2.77M |   xmlFatalErrMsg(ctxt, XML_ERR_NAME_REQUIRED, | 
| 8787 | 2.77M |            "StartTag: invalid element name\n"); | 
| 8788 | 2.77M |         return(NULL); | 
| 8789 | 2.77M |     } | 
| 8790 | 496k |     localname = hlocalname.name; | 
| 8791 | 496k |     prefix = hprefix.name; | 
| 8792 |  |  | 
| 8793 |  |     /* | 
| 8794 |  |      * Now parse the attributes, it ends up with the ending | 
| 8795 |  |      * | 
| 8796 |  |      * (S Attribute)* S? | 
| 8797 |  |      */ | 
| 8798 | 496k |     SKIP_BLANKS; | 
| 8799 | 496k |     GROW; | 
| 8800 |  |  | 
| 8801 |  |     /* | 
| 8802 |  |      * The ctxt->atts array will be ultimately passed to the SAX callback | 
| 8803 |  |      * containing five xmlChar pointers for each attribute: | 
| 8804 |  |      * | 
| 8805 |  |      * [0] attribute name | 
| 8806 |  |      * [1] attribute prefix | 
| 8807 |  |      * [2] namespace URI | 
| 8808 |  |      * [3] attribute value | 
| 8809 |  |      * [4] end of attribute value | 
| 8810 |  |      * | 
| 8811 |  |      * To save memory, we reuse this array temporarily and store integers | 
| 8812 |  |      * in these pointer variables. | 
| 8813 |  |      * | 
| 8814 |  |      * [0] attribute name | 
| 8815 |  |      * [1] attribute prefix | 
| 8816 |  |      * [2] hash value of attribute prefix, and later namespace index | 
| 8817 |  |      * [3] for non-allocated values: ptrdiff_t offset into input buffer | 
| 8818 |  |      * [4] for non-allocated values: ptrdiff_t offset into input buffer | 
| 8819 |  |      * | 
| 8820 |  |      * The ctxt->attallocs array contains an additional unsigned int for | 
| 8821 |  |      * each attribute, containing the hash value of the attribute name | 
| 8822 |  |      * and the alloc flag in bit 31. | 
| 8823 |  |      */ | 
| 8824 |  |  | 
| 8825 | 1.02M |     while (((RAW != '>') && | 
| 8826 | 839k |      ((RAW != '/') || (NXT(1) != '>')) && | 
| 8827 | 821k |      (IS_BYTE_CHAR(RAW))) && (PARSER_STOPPED(ctxt) == 0)) { | 
| 8828 | 810k |   int len = -1; | 
| 8829 |  |  | 
| 8830 | 810k |   hattname = xmlParseAttribute2(ctxt, prefix, localname, | 
| 8831 | 810k |                                           &haprefix, &attvalue, &len, | 
| 8832 | 810k |                                           &alloc); | 
| 8833 | 810k |         if (hattname.name == NULL) | 
| 8834 | 181k |       break; | 
| 8835 | 628k |         if (attvalue == NULL) | 
| 8836 | 32.1k |             goto next_attr; | 
| 8837 | 596k |         attname = hattname.name; | 
| 8838 | 596k |         aprefix = haprefix.name; | 
| 8839 | 596k |   if (len < 0) len = xmlStrlen(attvalue); | 
| 8840 |  |  | 
| 8841 | 596k |         if ((attname == ctxt->str_xmlns) && (aprefix == NULL)) { | 
| 8842 | 62.9k |             xmlHashedString huri; | 
| 8843 | 62.9k |             xmlURIPtr parsedUri; | 
| 8844 |  |  | 
| 8845 | 62.9k |             huri = xmlDictLookupHashed(ctxt->dict, attvalue, len); | 
| 8846 | 62.9k |             uri = huri.name; | 
| 8847 | 62.9k |             if (uri == NULL) { | 
| 8848 | 0 |                 xmlErrMemory(ctxt); | 
| 8849 | 0 |                 goto next_attr; | 
| 8850 | 0 |             } | 
| 8851 | 62.9k |             if (*uri != 0) { | 
| 8852 | 58.8k |                 if (xmlParseURISafe((const char *) uri, &parsedUri) < 0) { | 
| 8853 | 0 |                     xmlErrMemory(ctxt); | 
| 8854 | 0 |                     goto next_attr; | 
| 8855 | 0 |                 } | 
| 8856 | 58.8k |                 if (parsedUri == NULL) { | 
| 8857 | 32.6k |                     xmlNsErr(ctxt, XML_WAR_NS_URI, | 
| 8858 | 32.6k |                              "xmlns: '%s' is not a valid URI\n", | 
| 8859 | 32.6k |                                        uri, NULL, NULL); | 
| 8860 | 32.6k |                 } else { | 
| 8861 | 26.2k |                     if (parsedUri->scheme == NULL) { | 
| 8862 | 18.0k |                         xmlNsWarn(ctxt, XML_WAR_NS_URI_RELATIVE, | 
| 8863 | 18.0k |                                   "xmlns: URI %s is not absolute\n", | 
| 8864 | 18.0k |                                   uri, NULL, NULL); | 
| 8865 | 18.0k |                     } | 
| 8866 | 26.2k |                     xmlFreeURI(parsedUri); | 
| 8867 | 26.2k |                 } | 
| 8868 | 58.8k |                 if (uri == ctxt->str_xml_ns) { | 
| 8869 | 194 |                     if (attname != ctxt->str_xml) { | 
| 8870 | 194 |                         xmlNsErr(ctxt, XML_NS_ERR_XML_NAMESPACE, | 
| 8871 | 194 |                      "xml namespace URI cannot be the default namespace\n", | 
| 8872 | 194 |                                  NULL, NULL, NULL); | 
| 8873 | 194 |                     } | 
| 8874 | 194 |                     goto next_attr; | 
| 8875 | 194 |                 } | 
| 8876 | 58.7k |                 if ((len == 29) && | 
| 8877 | 1.33k |                     (xmlStrEqual(uri, | 
| 8878 | 1.33k |                              BAD_CAST "http://www.w3.org/2000/xmlns/"))) { | 
| 8879 | 689 |                     xmlNsErr(ctxt, XML_NS_ERR_XML_NAMESPACE, | 
| 8880 | 689 |                          "reuse of the xmlns namespace name is forbidden\n", | 
| 8881 | 689 |                              NULL, NULL, NULL); | 
| 8882 | 689 |                     goto next_attr; | 
| 8883 | 689 |                 } | 
| 8884 | 58.7k |             } | 
| 8885 |  |  | 
| 8886 | 62.0k |             if (xmlParserNsPush(ctxt, NULL, &huri, NULL, 0) > 0) | 
| 8887 | 43.4k |                 nbNs++; | 
| 8888 | 533k |         } else if (aprefix == ctxt->str_xmlns) { | 
| 8889 | 11.4k |             xmlHashedString huri; | 
| 8890 | 11.4k |             xmlURIPtr parsedUri; | 
| 8891 |  |  | 
| 8892 | 11.4k |             huri = xmlDictLookupHashed(ctxt->dict, attvalue, len); | 
| 8893 | 11.4k |             uri = huri.name; | 
| 8894 | 11.4k |             if (uri == NULL) { | 
| 8895 | 0 |                 xmlErrMemory(ctxt); | 
| 8896 | 0 |                 goto next_attr; | 
| 8897 | 0 |             } | 
| 8898 |  |  | 
| 8899 | 11.4k |             if (attname == ctxt->str_xml) { | 
| 8900 | 496 |                 if (uri != ctxt->str_xml_ns) { | 
| 8901 | 292 |                     xmlNsErr(ctxt, XML_NS_ERR_XML_NAMESPACE, | 
| 8902 | 292 |                              "xml namespace prefix mapped to wrong URI\n", | 
| 8903 | 292 |                              NULL, NULL, NULL); | 
| 8904 | 292 |                 } | 
| 8905 |  |                 /* | 
| 8906 |  |                  * Do not keep a namespace definition node | 
| 8907 |  |                  */ | 
| 8908 | 496 |                 goto next_attr; | 
| 8909 | 496 |             } | 
| 8910 | 10.9k |             if (uri == ctxt->str_xml_ns) { | 
| 8911 | 693 |                 if (attname != ctxt->str_xml) { | 
| 8912 | 693 |                     xmlNsErr(ctxt, XML_NS_ERR_XML_NAMESPACE, | 
| 8913 | 693 |                              "xml namespace URI mapped to wrong prefix\n", | 
| 8914 | 693 |                              NULL, NULL, NULL); | 
| 8915 | 693 |                 } | 
| 8916 | 693 |                 goto next_attr; | 
| 8917 | 693 |             } | 
| 8918 | 10.2k |             if (attname == ctxt->str_xmlns) { | 
| 8919 | 327 |                 xmlNsErr(ctxt, XML_NS_ERR_XML_NAMESPACE, | 
| 8920 | 327 |                          "redefinition of the xmlns prefix is forbidden\n", | 
| 8921 | 327 |                          NULL, NULL, NULL); | 
| 8922 | 327 |                 goto next_attr; | 
| 8923 | 327 |             } | 
| 8924 | 9.90k |             if ((len == 29) && | 
| 8925 | 423 |                 (xmlStrEqual(uri, | 
| 8926 | 423 |                              BAD_CAST "http://www.w3.org/2000/xmlns/"))) { | 
| 8927 | 194 |                 xmlNsErr(ctxt, XML_NS_ERR_XML_NAMESPACE, | 
| 8928 | 194 |                          "reuse of the xmlns namespace name is forbidden\n", | 
| 8929 | 194 |                          NULL, NULL, NULL); | 
| 8930 | 194 |                 goto next_attr; | 
| 8931 | 194 |             } | 
| 8932 | 9.71k |             if ((uri == NULL) || (uri[0] == 0)) { | 
| 8933 | 269 |                 xmlNsErr(ctxt, XML_NS_ERR_XML_NAMESPACE, | 
| 8934 | 269 |                          "xmlns:%s: Empty XML namespace is not allowed\n", | 
| 8935 | 269 |                               attname, NULL, NULL); | 
| 8936 | 269 |                 goto next_attr; | 
| 8937 | 9.44k |             } else { | 
| 8938 | 9.44k |                 if (xmlParseURISafe((const char *) uri, &parsedUri) < 0) { | 
| 8939 | 0 |                     xmlErrMemory(ctxt); | 
| 8940 | 0 |                     goto next_attr; | 
| 8941 | 0 |                 } | 
| 8942 | 9.44k |                 if (parsedUri == NULL) { | 
| 8943 | 3.64k |                     xmlNsErr(ctxt, XML_WAR_NS_URI, | 
| 8944 | 3.64k |                          "xmlns:%s: '%s' is not a valid URI\n", | 
| 8945 | 3.64k |                                        attname, uri, NULL); | 
| 8946 | 5.79k |                 } else { | 
| 8947 | 5.79k |                     if ((ctxt->pedantic) && (parsedUri->scheme == NULL)) { | 
| 8948 | 0 |                         xmlNsWarn(ctxt, XML_WAR_NS_URI_RELATIVE, | 
| 8949 | 0 |                                   "xmlns:%s: URI %s is not absolute\n", | 
| 8950 | 0 |                                   attname, uri, NULL); | 
| 8951 | 0 |                     } | 
| 8952 | 5.79k |                     xmlFreeURI(parsedUri); | 
| 8953 | 5.79k |                 } | 
| 8954 | 9.44k |             } | 
| 8955 |  |  | 
| 8956 | 9.44k |             if (xmlParserNsPush(ctxt, &hattname, &huri, NULL, 0) > 0) | 
| 8957 | 8.81k |                 nbNs++; | 
| 8958 | 522k |         } else { | 
| 8959 |  |             /* | 
| 8960 |  |              * Populate attributes array, see above for repurposing | 
| 8961 |  |              * of xmlChar pointers. | 
| 8962 |  |              */ | 
| 8963 | 522k |             if ((atts == NULL) || (nbatts + 5 > maxatts)) { | 
| 8964 | 4.77k |                 int res = xmlCtxtGrowAttrs(ctxt); | 
| 8965 |  |  | 
| 8966 | 4.77k |                 maxatts = ctxt->maxatts; | 
| 8967 | 4.77k |                 atts = ctxt->atts; | 
| 8968 |  |  | 
| 8969 | 4.77k |                 if (res < 0) | 
| 8970 | 0 |                     goto next_attr; | 
| 8971 | 4.77k |             } | 
| 8972 | 522k |             ctxt->attallocs[nratts++] = (hattname.hashValue & 0x7FFFFFFF) | | 
| 8973 | 522k |                                         ((unsigned) alloc << 31); | 
| 8974 | 522k |             atts[nbatts++] = attname; | 
| 8975 | 522k |             atts[nbatts++] = aprefix; | 
| 8976 | 522k |             atts[nbatts++] = XML_INT_TO_PTR(haprefix.hashValue); | 
| 8977 | 522k |             if (alloc) { | 
| 8978 | 34.1k |                 atts[nbatts++] = attvalue; | 
| 8979 | 34.1k |                 attvalue += len; | 
| 8980 | 34.1k |                 atts[nbatts++] = attvalue; | 
| 8981 | 487k |             } else { | 
| 8982 |  |                 /* | 
| 8983 |  |                  * attvalue points into the input buffer which can be | 
| 8984 |  |                  * reallocated. Store differences to input->base instead. | 
| 8985 |  |                  * The pointers will be reconstructed later. | 
| 8986 |  |                  */ | 
| 8987 | 487k |                 atts[nbatts++] = XML_INT_TO_PTR(attvalue - BASE_PTR); | 
| 8988 | 487k |                 attvalue += len; | 
| 8989 | 487k |                 atts[nbatts++] = XML_INT_TO_PTR(attvalue - BASE_PTR); | 
| 8990 | 487k |             } | 
| 8991 |  |             /* | 
| 8992 |  |              * tag if some deallocation is needed | 
| 8993 |  |              */ | 
| 8994 | 522k |             if (alloc != 0) attval = 1; | 
| 8995 | 522k |             attvalue = NULL; /* moved into atts */ | 
| 8996 | 522k |         } | 
| 8997 |  |  | 
| 8998 | 628k | next_attr: | 
| 8999 | 628k |         if ((attvalue != NULL) && (alloc != 0)) { | 
| 9000 | 21.2k |             xmlFree(attvalue); | 
| 9001 | 21.2k |             attvalue = NULL; | 
| 9002 | 21.2k |         } | 
| 9003 |  |  | 
| 9004 | 628k |   GROW | 
| 9005 | 628k |   if ((RAW == '>') || (((RAW == '/') && (NXT(1) == '>')))) | 
| 9006 | 44.8k |       break; | 
| 9007 | 583k |   if (SKIP_BLANKS == 0) { | 
| 9008 | 52.6k |       xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 9009 | 52.6k |          "attributes construct error\n"); | 
| 9010 | 52.6k |       break; | 
| 9011 | 52.6k |   } | 
| 9012 | 530k |         GROW; | 
| 9013 | 530k |     } | 
| 9014 |  |  | 
| 9015 |  |     /* | 
| 9016 |  |      * Namespaces from default attributes | 
| 9017 |  |      */ | 
| 9018 | 496k |     if (ctxt->attsDefault != NULL) { | 
| 9019 | 256k |         xmlDefAttrsPtr defaults; | 
| 9020 |  |  | 
| 9021 | 256k |   defaults = xmlHashLookup2(ctxt->attsDefault, localname, prefix); | 
| 9022 | 256k |   if (defaults != NULL) { | 
| 9023 | 1.45M |       for (i = 0; i < defaults->nbAttrs; i++) { | 
| 9024 | 1.28M |                 xmlDefAttr *attr = &defaults->attrs[i]; | 
| 9025 |  |  | 
| 9026 | 1.28M |           attname = attr->name.name; | 
| 9027 | 1.28M |     aprefix = attr->prefix.name; | 
| 9028 |  |  | 
| 9029 | 1.28M |     if ((attname == ctxt->str_xmlns) && (aprefix == NULL)) { | 
| 9030 | 13.1k |                     xmlParserEntityCheck(ctxt, attr->expandedSize); | 
| 9031 |  |  | 
| 9032 | 13.1k |                     if (xmlParserNsPush(ctxt, NULL, &attr->value, NULL, 1) > 0) | 
| 9033 | 12.5k |                         nbNs++; | 
| 9034 | 1.26M |     } else if (aprefix == ctxt->str_xmlns) { | 
| 9035 | 1.03M |                     xmlParserEntityCheck(ctxt, attr->expandedSize); | 
| 9036 |  |  | 
| 9037 | 1.03M |                     if (xmlParserNsPush(ctxt, &attr->name, &attr->value, | 
| 9038 | 1.03M |                                       NULL, 1) > 0) | 
| 9039 | 1.03M |                         nbNs++; | 
| 9040 | 1.03M |     } else { | 
| 9041 | 236k |                     if (nratts + nbTotalDef >= XML_MAX_ATTRS) { | 
| 9042 | 0 |                         xmlFatalErr(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 9043 | 0 |                                     "Maximum number of attributes exceeded"); | 
| 9044 | 0 |                         break; | 
| 9045 | 0 |                     } | 
| 9046 | 236k |                     nbTotalDef += 1; | 
| 9047 | 236k |                 } | 
| 9048 | 1.28M |       } | 
| 9049 | 177k |   } | 
| 9050 | 256k |     } | 
| 9051 |  |  | 
| 9052 |  |     /* | 
| 9053 |  |      * Resolve attribute namespaces | 
| 9054 |  |      */ | 
| 9055 | 1.01M |     for (i = 0; i < nbatts; i += 5) { | 
| 9056 | 522k |         attname = atts[i]; | 
| 9057 | 522k |         aprefix = atts[i+1]; | 
| 9058 |  |  | 
| 9059 |  |         /* | 
| 9060 |  |   * The default namespace does not apply to attribute names. | 
| 9061 |  |   */ | 
| 9062 | 522k |   if (aprefix == NULL) { | 
| 9063 | 483k |             nsIndex = NS_INDEX_EMPTY; | 
| 9064 | 483k |         } else if (aprefix == ctxt->str_xml) { | 
| 9065 | 9.47k |             nsIndex = NS_INDEX_XML; | 
| 9066 | 29.2k |         } else { | 
| 9067 | 29.2k |             haprefix.name = aprefix; | 
| 9068 | 29.2k |             haprefix.hashValue = (size_t) atts[i+2]; | 
| 9069 | 29.2k |             nsIndex = xmlParserNsLookup(ctxt, &haprefix, NULL); | 
| 9070 |  |  | 
| 9071 | 29.2k |       if ((nsIndex == INT_MAX) || (nsIndex < ctxt->nsdb->minNsIndex)) { | 
| 9072 | 16.9k |                 xmlNsErr(ctxt, XML_NS_ERR_UNDEFINED_NAMESPACE, | 
| 9073 | 16.9k |         "Namespace prefix %s for %s on %s is not defined\n", | 
| 9074 | 16.9k |         aprefix, attname, localname); | 
| 9075 | 16.9k |                 nsIndex = NS_INDEX_EMPTY; | 
| 9076 | 16.9k |             } | 
| 9077 | 29.2k |         } | 
| 9078 |  |  | 
| 9079 | 522k |         atts[i+2] = XML_INT_TO_PTR(nsIndex); | 
| 9080 | 522k |     } | 
| 9081 |  |  | 
| 9082 |  |     /* | 
| 9083 |  |      * Maximum number of attributes including default attributes. | 
| 9084 |  |      */ | 
| 9085 | 496k |     maxAtts = nratts + nbTotalDef; | 
| 9086 |  |  | 
| 9087 |  |     /* | 
| 9088 |  |      * Verify that attribute names are unique. | 
| 9089 |  |      */ | 
| 9090 | 496k |     if (maxAtts > 1) { | 
| 9091 | 54.2k |         attrHashSize = 4; | 
| 9092 | 109k |         while (attrHashSize / 2 < (unsigned) maxAtts) | 
| 9093 | 55.5k |             attrHashSize *= 2; | 
| 9094 |  |  | 
| 9095 | 54.2k |         if (attrHashSize > ctxt->attrHashMax) { | 
| 9096 | 1.35k |             xmlAttrHashBucket *tmp; | 
| 9097 |  |  | 
| 9098 | 1.35k |             tmp = xmlRealloc(ctxt->attrHash, attrHashSize * sizeof(tmp[0])); | 
| 9099 | 1.35k |             if (tmp == NULL) { | 
| 9100 | 0 |                 xmlErrMemory(ctxt); | 
| 9101 | 0 |                 goto done; | 
| 9102 | 0 |             } | 
| 9103 |  |  | 
| 9104 | 1.35k |             ctxt->attrHash = tmp; | 
| 9105 | 1.35k |             ctxt->attrHashMax = attrHashSize; | 
| 9106 | 1.35k |         } | 
| 9107 |  |  | 
| 9108 | 54.2k |         memset(ctxt->attrHash, -1, attrHashSize * sizeof(ctxt->attrHash[0])); | 
| 9109 |  |  | 
| 9110 | 559k |         for (i = 0, j = 0; j < nratts; i += 5, j++) { | 
| 9111 | 504k |             const xmlChar *nsuri; | 
| 9112 | 504k |             unsigned hashValue, nameHashValue, uriHashValue; | 
| 9113 | 504k |             int res; | 
| 9114 |  |  | 
| 9115 | 504k |             attname = atts[i]; | 
| 9116 | 504k |             aprefix = atts[i+1]; | 
| 9117 | 504k |             nsIndex = XML_PTR_TO_INT(atts[i+2]); | 
| 9118 |  |             /* Hash values always have bit 31 set, see dict.c */ | 
| 9119 | 504k |             nameHashValue = ctxt->attallocs[j] | 0x80000000; | 
| 9120 |  |  | 
| 9121 | 504k |             if (nsIndex == NS_INDEX_EMPTY) { | 
| 9122 |  |                 /* | 
| 9123 |  |                  * Prefix with empty namespace means an undeclared | 
| 9124 |  |                  * prefix which was already reported above. | 
| 9125 |  |                  */ | 
| 9126 | 491k |                 if (aprefix != NULL) | 
| 9127 | 16.2k |                     continue; | 
| 9128 | 474k |                 nsuri = NULL; | 
| 9129 | 474k |                 uriHashValue = URI_HASH_EMPTY; | 
| 9130 | 474k |             } else if (nsIndex == NS_INDEX_XML) { | 
| 9131 | 1.78k |                 nsuri = ctxt->str_xml_ns; | 
| 9132 | 1.78k |                 uriHashValue = URI_HASH_XML; | 
| 9133 | 12.1k |             } else { | 
| 9134 | 12.1k |                 nsuri = ctxt->nsTab[nsIndex * 2 + 1]; | 
| 9135 | 12.1k |                 uriHashValue = ctxt->nsdb->extra[nsIndex].uriHashValue; | 
| 9136 | 12.1k |             } | 
| 9137 |  |  | 
| 9138 | 488k |             hashValue = xmlDictCombineHash(nameHashValue, uriHashValue); | 
| 9139 | 488k |             res = xmlAttrHashInsert(ctxt, attrHashSize, attname, nsuri, | 
| 9140 | 488k |                                     hashValue, i); | 
| 9141 | 488k |             if (res < 0) | 
| 9142 | 0 |                 continue; | 
| 9143 |  |  | 
| 9144 |  |             /* | 
| 9145 |  |              * [ WFC: Unique Att Spec ] | 
| 9146 |  |              * No attribute name may appear more than once in the same | 
| 9147 |  |              * start-tag or empty-element tag. | 
| 9148 |  |              * As extended by the Namespace in XML REC. | 
| 9149 |  |              */ | 
| 9150 | 488k |             if (res < INT_MAX) { | 
| 9151 | 477k |                 if (aprefix == atts[res+1]) { | 
| 9152 | 470k |                     xmlErrAttributeDup(ctxt, aprefix, attname); | 
| 9153 | 470k |                     numDupErr += 1; | 
| 9154 | 470k |                 } else { | 
| 9155 | 7.36k |                     xmlNsErr(ctxt, XML_NS_ERR_ATTRIBUTE_REDEFINED, | 
| 9156 | 7.36k |                              "Namespaced Attribute %s in '%s' redefined\n", | 
| 9157 | 7.36k |                              attname, nsuri, NULL); | 
| 9158 | 7.36k |                     numNsErr += 1; | 
| 9159 | 7.36k |                 } | 
| 9160 | 477k |             } | 
| 9161 | 488k |         } | 
| 9162 | 54.2k |     } | 
| 9163 |  |  | 
| 9164 |  |     /* | 
| 9165 |  |      * Default attributes | 
| 9166 |  |      */ | 
| 9167 | 496k |     if (ctxt->attsDefault != NULL) { | 
| 9168 | 256k |         xmlDefAttrsPtr defaults; | 
| 9169 |  |  | 
| 9170 | 256k |   defaults = xmlHashLookup2(ctxt->attsDefault, localname, prefix); | 
| 9171 | 256k |   if (defaults != NULL) { | 
| 9172 | 1.45M |       for (i = 0; i < defaults->nbAttrs; i++) { | 
| 9173 | 1.28M |                 xmlDefAttr *attr = &defaults->attrs[i]; | 
| 9174 | 1.28M |                 const xmlChar *nsuri = NULL; | 
| 9175 | 1.28M |                 unsigned hashValue, uriHashValue = 0; | 
| 9176 | 1.28M |                 int res; | 
| 9177 |  |  | 
| 9178 | 1.28M |           attname = attr->name.name; | 
| 9179 | 1.28M |     aprefix = attr->prefix.name; | 
| 9180 |  |  | 
| 9181 | 1.28M |     if ((attname == ctxt->str_xmlns) && (aprefix == NULL)) | 
| 9182 | 13.1k |                     continue; | 
| 9183 | 1.26M |     if (aprefix == ctxt->str_xmlns) | 
| 9184 | 1.03M |                     continue; | 
| 9185 |  |  | 
| 9186 | 236k |                 if (aprefix == NULL) { | 
| 9187 | 60.3k |                     nsIndex = NS_INDEX_EMPTY; | 
| 9188 | 60.3k |                     nsuri = NULL; | 
| 9189 | 60.3k |                     uriHashValue = URI_HASH_EMPTY; | 
| 9190 | 176k |                 } else if (aprefix == ctxt->str_xml) { | 
| 9191 | 6.92k |                     nsIndex = NS_INDEX_XML; | 
| 9192 | 6.92k |                     nsuri = ctxt->str_xml_ns; | 
| 9193 | 6.92k |                     uriHashValue = URI_HASH_XML; | 
| 9194 | 169k |                 } else { | 
| 9195 | 169k |                     nsIndex = xmlParserNsLookup(ctxt, &attr->prefix, NULL); | 
| 9196 | 169k |                     if ((nsIndex == INT_MAX) || | 
| 9197 | 134k |                         (nsIndex < ctxt->nsdb->minNsIndex)) { | 
| 9198 | 134k |                         xmlNsErr(ctxt, XML_NS_ERR_UNDEFINED_NAMESPACE, | 
| 9199 | 134k |                                  "Namespace prefix %s for %s on %s is not " | 
| 9200 | 134k |                                  "defined\n", | 
| 9201 | 134k |                                  aprefix, attname, localname); | 
| 9202 | 134k |                         nsIndex = NS_INDEX_EMPTY; | 
| 9203 | 134k |                         nsuri = NULL; | 
| 9204 | 134k |                         uriHashValue = URI_HASH_EMPTY; | 
| 9205 | 134k |                     } else { | 
| 9206 | 34.9k |                         nsuri = ctxt->nsTab[nsIndex * 2 + 1]; | 
| 9207 | 34.9k |                         uriHashValue = ctxt->nsdb->extra[nsIndex].uriHashValue; | 
| 9208 | 34.9k |                     } | 
| 9209 | 169k |                 } | 
| 9210 |  |  | 
| 9211 |  |                 /* | 
| 9212 |  |                  * Check whether the attribute exists | 
| 9213 |  |                  */ | 
| 9214 | 236k |                 if (maxAtts > 1) { | 
| 9215 | 201k |                     hashValue = xmlDictCombineHash(attr->name.hashValue, | 
| 9216 | 201k |                                                    uriHashValue); | 
| 9217 | 201k |                     res = xmlAttrHashInsert(ctxt, attrHashSize, attname, nsuri, | 
| 9218 | 201k |                                             hashValue, nbatts); | 
| 9219 | 201k |                     if (res < 0) | 
| 9220 | 0 |                         continue; | 
| 9221 | 201k |                     if (res < INT_MAX) { | 
| 9222 | 8.31k |                         if (aprefix == atts[res+1]) | 
| 9223 | 753 |                             continue; | 
| 9224 | 7.56k |                         xmlNsErr(ctxt, XML_NS_ERR_ATTRIBUTE_REDEFINED, | 
| 9225 | 7.56k |                                  "Namespaced Attribute %s in '%s' redefined\n", | 
| 9226 | 7.56k |                                  attname, nsuri, NULL); | 
| 9227 | 7.56k |                     } | 
| 9228 | 201k |                 } | 
| 9229 |  |  | 
| 9230 | 235k |                 xmlParserEntityCheck(ctxt, attr->expandedSize); | 
| 9231 |  |  | 
| 9232 | 235k |                 if ((atts == NULL) || (nbatts + 5 > maxatts)) { | 
| 9233 | 1.71k |                     res = xmlCtxtGrowAttrs(ctxt); | 
| 9234 |  |  | 
| 9235 | 1.71k |                     maxatts = ctxt->maxatts; | 
| 9236 | 1.71k |                     atts = ctxt->atts; | 
| 9237 |  |  | 
| 9238 | 1.71k |                     if (res < 0) { | 
| 9239 | 0 |                         localname = NULL; | 
| 9240 | 0 |                         goto done; | 
| 9241 | 0 |                     } | 
| 9242 | 1.71k |                 } | 
| 9243 |  |  | 
| 9244 | 235k |                 atts[nbatts++] = attname; | 
| 9245 | 235k |                 atts[nbatts++] = aprefix; | 
| 9246 | 235k |                 atts[nbatts++] = XML_INT_TO_PTR(nsIndex); | 
| 9247 | 235k |                 atts[nbatts++] = attr->value.name; | 
| 9248 | 235k |                 atts[nbatts++] = attr->valueEnd; | 
| 9249 |  |  | 
| 9250 | 235k | #ifdef LIBXML_VALID_ENABLED | 
| 9251 |  |                 /* | 
| 9252 |  |                  * This should be moved to valid.c, but we don't keep track | 
| 9253 |  |                  * whether an attribute was defaulted. | 
| 9254 |  |                  */ | 
| 9255 | 235k |                 if ((ctxt->validate) && | 
| 9256 | 0 |                     (ctxt->standalone == 1) && | 
| 9257 | 0 |                     (attr->external != 0)) { | 
| 9258 | 0 |                     xmlValidityError(ctxt, XML_DTD_STANDALONE_DEFAULTED, | 
| 9259 | 0 |                             "standalone: attribute %s on %s defaulted " | 
| 9260 | 0 |                             "from external subset\n", | 
| 9261 | 0 |                             attname, localname); | 
| 9262 | 0 |                 } | 
| 9263 | 235k | #endif | 
| 9264 | 235k |                 nbdef++; | 
| 9265 | 235k |       } | 
| 9266 | 177k |   } | 
| 9267 | 256k |     } | 
| 9268 |  |  | 
| 9269 |  |     /* | 
| 9270 |  |      * Using a single hash table for nsUri/localName pairs cannot | 
| 9271 |  |      * detect duplicate QNames reliably. The following example will | 
| 9272 |  |      * only result in two namespace errors. | 
| 9273 |  |      * | 
| 9274 |  |      * <doc xmlns:a="a" xmlns:b="a"> | 
| 9275 |  |      *   <elem a:a="" b:a="" b:a=""/> | 
| 9276 |  |      * </doc> | 
| 9277 |  |      * | 
| 9278 |  |      * If we saw more than one namespace error but no duplicate QNames | 
| 9279 |  |      * were found, we have to scan for duplicate QNames. | 
| 9280 |  |      */ | 
| 9281 | 496k |     if ((numDupErr == 0) && (numNsErr > 1)) { | 
| 9282 | 977 |         memset(ctxt->attrHash, -1, | 
| 9283 | 977 |                attrHashSize * sizeof(ctxt->attrHash[0])); | 
| 9284 |  |  | 
| 9285 | 18.7k |         for (i = 0, j = 0; j < nratts; i += 5, j++) { | 
| 9286 | 17.7k |             unsigned hashValue, nameHashValue, prefixHashValue; | 
| 9287 | 17.7k |             int res; | 
| 9288 |  |  | 
| 9289 | 17.7k |             aprefix = atts[i+1]; | 
| 9290 | 17.7k |             if (aprefix == NULL) | 
| 9291 | 599 |                 continue; | 
| 9292 |  |  | 
| 9293 | 17.1k |             attname = atts[i]; | 
| 9294 |  |             /* Hash values always have bit 31 set, see dict.c */ | 
| 9295 | 17.1k |             nameHashValue = ctxt->attallocs[j] | 0x80000000; | 
| 9296 | 17.1k |             prefixHashValue = xmlDictComputeHash(ctxt->dict, aprefix); | 
| 9297 |  |  | 
| 9298 | 17.1k |             hashValue = xmlDictCombineHash(nameHashValue, prefixHashValue); | 
| 9299 | 17.1k |             res = xmlAttrHashInsertQName(ctxt, attrHashSize, attname, | 
| 9300 | 17.1k |                                          aprefix, hashValue, i); | 
| 9301 | 17.1k |             if (res < INT_MAX) | 
| 9302 | 12.2k |                 xmlErrAttributeDup(ctxt, aprefix, attname); | 
| 9303 | 17.1k |         } | 
| 9304 | 977 |     } | 
| 9305 |  |  | 
| 9306 |  |     /* | 
| 9307 |  |      * Reconstruct attribute pointers | 
| 9308 |  |      */ | 
| 9309 | 1.25M |     for (i = 0, j = 0; i < nbatts; i += 5, j++) { | 
| 9310 |  |         /* namespace URI */ | 
| 9311 | 757k |         nsIndex = XML_PTR_TO_INT(atts[i+2]); | 
| 9312 | 757k |         if (nsIndex == INT_MAX) | 
| 9313 | 694k |             atts[i+2] = NULL; | 
| 9314 | 63.1k |         else if (nsIndex == INT_MAX - 1) | 
| 9315 | 16.0k |             atts[i+2] = ctxt->str_xml_ns; | 
| 9316 | 47.1k |         else | 
| 9317 | 47.1k |             atts[i+2] = ctxt->nsTab[nsIndex * 2 + 1]; | 
| 9318 |  |  | 
| 9319 | 757k |         if ((j < nratts) && (ctxt->attallocs[j] & 0x80000000) == 0) { | 
| 9320 | 487k |             atts[i+3] = BASE_PTR + XML_PTR_TO_INT(atts[i+3]);  /* value */ | 
| 9321 | 487k |             atts[i+4] = BASE_PTR + XML_PTR_TO_INT(atts[i+4]);  /* valuend */ | 
| 9322 | 487k |         } | 
| 9323 | 757k |     } | 
| 9324 |  |  | 
| 9325 | 496k |     uri = xmlParserNsLookupUri(ctxt, &hprefix); | 
| 9326 | 496k |     if ((prefix != NULL) && (uri == NULL)) { | 
| 9327 | 22.0k |   xmlNsErr(ctxt, XML_NS_ERR_UNDEFINED_NAMESPACE, | 
| 9328 | 22.0k |            "Namespace prefix %s on %s is not defined\n", | 
| 9329 | 22.0k |      prefix, localname, NULL); | 
| 9330 | 22.0k |     } | 
| 9331 | 496k |     *pref = prefix; | 
| 9332 | 496k |     *URI = uri; | 
| 9333 |  |  | 
| 9334 |  |     /* | 
| 9335 |  |      * SAX callback | 
| 9336 |  |      */ | 
| 9337 | 496k |     if ((ctxt->sax != NULL) && (ctxt->sax->startElementNs != NULL) && | 
| 9338 | 496k |   (!ctxt->disableSAX)) { | 
| 9339 | 73.7k |   if (nbNs > 0) | 
| 9340 | 25.1k |       ctxt->sax->startElementNs(ctxt->userData, localname, prefix, uri, | 
| 9341 | 25.1k |                           nbNs, ctxt->nsTab + 2 * (ctxt->nsNr - nbNs), | 
| 9342 | 25.1k |         nbatts / 5, nbdef, atts); | 
| 9343 | 48.5k |   else | 
| 9344 | 48.5k |       ctxt->sax->startElementNs(ctxt->userData, localname, prefix, uri, | 
| 9345 | 48.5k |                           0, NULL, nbatts / 5, nbdef, atts); | 
| 9346 | 73.7k |     } | 
| 9347 |  |  | 
| 9348 | 496k | done: | 
| 9349 |  |     /* | 
| 9350 |  |      * Free allocated attribute values | 
| 9351 |  |      */ | 
| 9352 | 496k |     if (attval != 0) { | 
| 9353 | 502k |   for (i = 0, j = 0; j < nratts; i += 5, j++) | 
| 9354 | 489k |       if (ctxt->attallocs[j] & 0x80000000) | 
| 9355 | 34.1k |           xmlFree((xmlChar *) atts[i+3]); | 
| 9356 | 13.1k |     } | 
| 9357 |  |  | 
| 9358 | 496k |     *nbNsPtr = nbNs; | 
| 9359 | 496k |     return(localname); | 
| 9360 | 496k | } | 
| 9361 |  |  | 
| 9362 |  | /** | 
| 9363 |  |  * Parse an end tag. Always consumes '</'. | 
| 9364 |  |  * | 
| 9365 |  |  *     [42] ETag ::= '</' Name S? '>' | 
| 9366 |  |  * | 
| 9367 |  |  * With namespace | 
| 9368 |  |  * | 
| 9369 |  |  *     [NS 9] ETag ::= '</' QName S? '>' | 
| 9370 |  |  * @param ctxt  an XML parser context | 
| 9371 |  |  * @param tag  the corresponding start tag | 
| 9372 |  |  */ | 
| 9373 |  |  | 
| 9374 |  | static void | 
| 9375 | 26.2k | xmlParseEndTag2(xmlParserCtxtPtr ctxt, const xmlStartTag *tag) { | 
| 9376 | 26.2k |     const xmlChar *name; | 
| 9377 |  |  | 
| 9378 | 26.2k |     GROW; | 
| 9379 | 26.2k |     if ((RAW != '<') || (NXT(1) != '/')) { | 
| 9380 | 162 |   xmlFatalErr(ctxt, XML_ERR_LTSLASH_REQUIRED, NULL); | 
| 9381 | 162 |   return; | 
| 9382 | 162 |     } | 
| 9383 | 26.0k |     SKIP(2); | 
| 9384 |  |  | 
| 9385 | 26.0k |     if (tag->prefix == NULL) | 
| 9386 | 10.5k |         name = xmlParseNameAndCompare(ctxt, ctxt->name); | 
| 9387 | 15.5k |     else | 
| 9388 | 15.5k |         name = xmlParseQNameAndCompare(ctxt, ctxt->name, tag->prefix); | 
| 9389 |  |  | 
| 9390 |  |     /* | 
| 9391 |  |      * We should definitely be at the ending "S? '>'" part | 
| 9392 |  |      */ | 
| 9393 | 26.0k |     GROW; | 
| 9394 | 26.0k |     SKIP_BLANKS; | 
| 9395 | 26.0k |     if ((!IS_BYTE_CHAR(RAW)) || (RAW != '>')) { | 
| 9396 | 20.3k |   xmlFatalErr(ctxt, XML_ERR_GT_REQUIRED, NULL); | 
| 9397 | 20.3k |     } else | 
| 9398 | 5.70k |   NEXT1; | 
| 9399 |  |  | 
| 9400 |  |     /* | 
| 9401 |  |      * [ WFC: Element Type Match ] | 
| 9402 |  |      * The Name in an element's end-tag must match the element type in the | 
| 9403 |  |      * start-tag. | 
| 9404 |  |      * | 
| 9405 |  |      */ | 
| 9406 | 26.0k |     if (name != (xmlChar*)1) { | 
| 9407 | 12.1k |         if (name == NULL) name = BAD_CAST "unparsable"; | 
| 9408 | 12.1k |         xmlFatalErrMsgStrIntStr(ctxt, XML_ERR_TAG_NAME_MISMATCH, | 
| 9409 | 12.1k |          "Opening and ending tag mismatch: %s line %d and %s\n", | 
| 9410 | 12.1k |                     ctxt->name, tag->line, name); | 
| 9411 | 12.1k |     } | 
| 9412 |  |  | 
| 9413 |  |     /* | 
| 9414 |  |      * SAX: End of Tag | 
| 9415 |  |      */ | 
| 9416 | 26.0k |     if ((ctxt->sax != NULL) && (ctxt->sax->endElementNs != NULL) && | 
| 9417 | 26.0k |   (!ctxt->disableSAX)) | 
| 9418 | 582 |   ctxt->sax->endElementNs(ctxt->userData, ctxt->name, tag->prefix, | 
| 9419 | 582 |                                 tag->URI); | 
| 9420 |  |  | 
| 9421 | 26.0k |     spacePop(ctxt); | 
| 9422 | 26.0k |     if (tag->nsNr != 0) | 
| 9423 | 1.58k |   xmlParserNsPop(ctxt, tag->nsNr); | 
| 9424 | 26.0k | } | 
| 9425 |  |  | 
| 9426 |  | /** | 
| 9427 |  |  * Parse escaped pure raw content. Always consumes '<!['. | 
| 9428 |  |  * | 
| 9429 |  |  * @deprecated Internal function, don't use. | 
| 9430 |  |  * | 
| 9431 |  |  *     [18] CDSect ::= CDStart CData CDEnd | 
| 9432 |  |  * | 
| 9433 |  |  *     [19] CDStart ::= '<![CDATA[' | 
| 9434 |  |  * | 
| 9435 |  |  *     [20] Data ::= (Char* - (Char* ']]>' Char*)) | 
| 9436 |  |  * | 
| 9437 |  |  *     [21] CDEnd ::= ']]>' | 
| 9438 |  |  * @param ctxt  an XML parser context | 
| 9439 |  |  */ | 
| 9440 |  | void | 
| 9441 | 9.70k | xmlParseCDSect(xmlParserCtxt *ctxt) { | 
| 9442 | 9.70k |     xmlChar *buf = NULL; | 
| 9443 | 9.70k |     int len = 0; | 
| 9444 | 9.70k |     int size = XML_PARSER_BUFFER_SIZE; | 
| 9445 | 9.70k |     int r, rl; | 
| 9446 | 9.70k |     int s, sl; | 
| 9447 | 9.70k |     int cur, l; | 
| 9448 | 9.70k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 9449 | 9.70k |                     XML_MAX_HUGE_LENGTH : | 
| 9450 | 9.70k |                     XML_MAX_TEXT_LENGTH; | 
| 9451 |  |  | 
| 9452 | 9.70k |     if ((CUR != '<') || (NXT(1) != '!') || (NXT(2) != '[')) | 
| 9453 | 0 |         return; | 
| 9454 | 9.70k |     SKIP(3); | 
| 9455 |  |  | 
| 9456 | 9.70k |     if (!CMP6(CUR_PTR, 'C', 'D', 'A', 'T', 'A', '[')) | 
| 9457 | 0 |         return; | 
| 9458 | 9.70k |     SKIP(6); | 
| 9459 |  |  | 
| 9460 | 9.70k |     r = xmlCurrentCharRecover(ctxt, &rl); | 
| 9461 | 9.70k |     if (!IS_CHAR(r)) { | 
| 9462 | 1.46k |   xmlFatalErr(ctxt, XML_ERR_CDATA_NOT_FINISHED, NULL); | 
| 9463 | 1.46k |         goto out; | 
| 9464 | 1.46k |     } | 
| 9465 | 8.24k |     NEXTL(rl); | 
| 9466 | 8.24k |     s = xmlCurrentCharRecover(ctxt, &sl); | 
| 9467 | 8.24k |     if (!IS_CHAR(s)) { | 
| 9468 | 3.02k |   xmlFatalErr(ctxt, XML_ERR_CDATA_NOT_FINISHED, NULL); | 
| 9469 | 3.02k |         goto out; | 
| 9470 | 3.02k |     } | 
| 9471 | 5.21k |     NEXTL(sl); | 
| 9472 | 5.21k |     cur = xmlCurrentCharRecover(ctxt, &l); | 
| 9473 | 5.21k |     buf = xmlMalloc(size); | 
| 9474 | 5.21k |     if (buf == NULL) { | 
| 9475 | 0 |   xmlErrMemory(ctxt); | 
| 9476 | 0 |         goto out; | 
| 9477 | 0 |     } | 
| 9478 | 2.66M |     while (IS_CHAR(cur) && | 
| 9479 | 2.65M |            ((r != ']') || (s != ']') || (cur != '>'))) { | 
| 9480 | 2.65M |   if (len + 5 >= size) { | 
| 9481 | 4.23k |       xmlChar *tmp; | 
| 9482 | 4.23k |             int newSize; | 
| 9483 |  |  | 
| 9484 | 4.23k |             newSize = xmlGrowCapacity(size, 1, 1, maxLength); | 
| 9485 | 4.23k |             if (newSize < 0) { | 
| 9486 | 0 |                 xmlFatalErrMsg(ctxt, XML_ERR_CDATA_NOT_FINISHED, | 
| 9487 | 0 |                                "CData section too big found\n"); | 
| 9488 | 0 |                 goto out; | 
| 9489 | 0 |             } | 
| 9490 | 4.23k |       tmp = xmlRealloc(buf, newSize); | 
| 9491 | 4.23k |       if (tmp == NULL) { | 
| 9492 | 0 |     xmlErrMemory(ctxt); | 
| 9493 | 0 |                 goto out; | 
| 9494 | 0 |       } | 
| 9495 | 4.23k |       buf = tmp; | 
| 9496 | 4.23k |       size = newSize; | 
| 9497 | 4.23k |   } | 
| 9498 | 2.65M |   COPY_BUF(buf, len, r); | 
| 9499 | 2.65M |   r = s; | 
| 9500 | 2.65M |   rl = sl; | 
| 9501 | 2.65M |   s = cur; | 
| 9502 | 2.65M |   sl = l; | 
| 9503 | 2.65M |   NEXTL(l); | 
| 9504 | 2.65M |   cur = xmlCurrentCharRecover(ctxt, &l); | 
| 9505 | 2.65M |     } | 
| 9506 | 5.21k |     buf[len] = 0; | 
| 9507 | 5.21k |     if (cur != '>') { | 
| 9508 | 3.35k |   xmlFatalErrMsgStr(ctxt, XML_ERR_CDATA_NOT_FINISHED, | 
| 9509 | 3.35k |                        "CData section not finished\n%.50s\n", buf); | 
| 9510 | 3.35k |         goto out; | 
| 9511 | 3.35k |     } | 
| 9512 | 1.86k |     NEXTL(l); | 
| 9513 |  |  | 
| 9514 |  |     /* | 
| 9515 |  |      * OK the buffer is to be consumed as cdata. | 
| 9516 |  |      */ | 
| 9517 | 1.86k |     if ((ctxt->sax != NULL) && (!ctxt->disableSAX)) { | 
| 9518 | 1.17k |         if ((ctxt->sax->cdataBlock != NULL) && | 
| 9519 | 1.17k |             ((ctxt->options & XML_PARSE_NOCDATA) == 0)) { | 
| 9520 | 1.17k |             ctxt->sax->cdataBlock(ctxt->userData, buf, len); | 
| 9521 | 1.17k |         } else if (ctxt->sax->characters != NULL) { | 
| 9522 | 0 |             ctxt->sax->characters(ctxt->userData, buf, len); | 
| 9523 | 0 |         } | 
| 9524 | 1.17k |     } | 
| 9525 |  |  | 
| 9526 | 9.70k | out: | 
| 9527 | 9.70k |     xmlFree(buf); | 
| 9528 | 9.70k | } | 
| 9529 |  |  | 
| 9530 |  | /** | 
| 9531 |  |  * Parse a content sequence. Stops at EOF or '</'. Leaves checking of | 
| 9532 |  |  * unexpected EOF to the caller. | 
| 9533 |  |  * | 
| 9534 |  |  * @param ctxt  an XML parser context | 
| 9535 |  |  */ | 
| 9536 |  |  | 
| 9537 |  | static void | 
| 9538 | 8.18k | xmlParseContentInternal(xmlParserCtxtPtr ctxt) { | 
| 9539 | 8.18k |     int oldNameNr = ctxt->nameNr; | 
| 9540 | 8.18k |     int oldSpaceNr = ctxt->spaceNr; | 
| 9541 | 8.18k |     int oldNodeNr = ctxt->nodeNr; | 
| 9542 |  |  | 
| 9543 | 8.18k |     GROW; | 
| 9544 | 5.14M |     while ((ctxt->input->cur < ctxt->input->end) && | 
| 9545 | 5.14M |      (PARSER_STOPPED(ctxt) == 0)) { | 
| 9546 | 5.13M |   const xmlChar *cur = ctxt->input->cur; | 
| 9547 |  |  | 
| 9548 |  |   /* | 
| 9549 |  |    * First case : a Processing Instruction. | 
| 9550 |  |    */ | 
| 9551 | 5.13M |   if ((*cur == '<') && (cur[1] == '?')) { | 
| 9552 | 16.2k |       xmlParsePI(ctxt); | 
| 9553 | 16.2k |   } | 
| 9554 |  |  | 
| 9555 |  |   /* | 
| 9556 |  |    * Second case : a CDSection | 
| 9557 |  |    */ | 
| 9558 |  |   /* 2.6.0 test was *cur not RAW */ | 
| 9559 | 5.12M |   else if (CMP9(CUR_PTR, '<', '!', '[', 'C', 'D', 'A', 'T', 'A', '[')) { | 
| 9560 | 9.70k |       xmlParseCDSect(ctxt); | 
| 9561 | 9.70k |   } | 
| 9562 |  |  | 
| 9563 |  |   /* | 
| 9564 |  |    * Third case :  a comment | 
| 9565 |  |    */ | 
| 9566 | 5.11M |   else if ((*cur == '<') && (NXT(1) == '!') && | 
| 9567 | 53.2k |      (NXT(2) == '-') && (NXT(3) == '-')) { | 
| 9568 | 27.5k |       xmlParseComment(ctxt); | 
| 9569 | 27.5k |   } | 
| 9570 |  |  | 
| 9571 |  |   /* | 
| 9572 |  |    * Fourth case :  a sub-element. | 
| 9573 |  |    */ | 
| 9574 | 5.08M |   else if (*cur == '<') { | 
| 9575 | 3.28M |             if (NXT(1) == '/') { | 
| 9576 | 26.0k |                 if (ctxt->nameNr <= oldNameNr) | 
| 9577 | 292 |                     break; | 
| 9578 | 25.7k |           xmlParseElementEnd(ctxt); | 
| 9579 | 3.26M |             } else { | 
| 9580 | 3.26M |           xmlParseElementStart(ctxt); | 
| 9581 | 3.26M |             } | 
| 9582 | 3.28M |   } | 
| 9583 |  |  | 
| 9584 |  |   /* | 
| 9585 |  |    * Fifth case : a reference. If if has not been resolved, | 
| 9586 |  |    *    parsing returns it's Name, create the node | 
| 9587 |  |    */ | 
| 9588 |  |  | 
| 9589 | 1.79M |   else if (*cur == '&') { | 
| 9590 | 110k |       xmlParseReference(ctxt); | 
| 9591 | 110k |   } | 
| 9592 |  |  | 
| 9593 |  |   /* | 
| 9594 |  |    * Last case, text. Note that References are handled directly. | 
| 9595 |  |    */ | 
| 9596 | 1.68M |   else { | 
| 9597 | 1.68M |       xmlParseCharDataInternal(ctxt, 0); | 
| 9598 | 1.68M |   } | 
| 9599 |  |  | 
| 9600 | 5.13M |   SHRINK; | 
| 9601 | 5.13M |   GROW; | 
| 9602 | 5.13M |     } | 
| 9603 |  |  | 
| 9604 | 8.18k |     if ((ctxt->nameNr > oldNameNr) && | 
| 9605 | 2.60k |         (ctxt->input->cur >= ctxt->input->end) && | 
| 9606 | 2.46k |         (ctxt->wellFormed)) { | 
| 9607 | 217 |         const xmlChar *name = ctxt->nameTab[ctxt->nameNr - 1]; | 
| 9608 | 217 |         int line = ctxt->pushTab[ctxt->nameNr - 1].line; | 
| 9609 | 217 |         xmlFatalErrMsgStrIntStr(ctxt, XML_ERR_TAG_NOT_FINISHED, | 
| 9610 | 217 |                 "Premature end of data in tag %s line %d\n", | 
| 9611 | 217 |                 name, line, NULL); | 
| 9612 | 217 |     } | 
| 9613 |  |  | 
| 9614 |  |     /* | 
| 9615 |  |      * Clean up in error case | 
| 9616 |  |      */ | 
| 9617 |  |  | 
| 9618 | 70.0k |     while (ctxt->nodeNr > oldNodeNr) | 
| 9619 | 61.8k |         nodePop(ctxt); | 
| 9620 |  |  | 
| 9621 | 208k |     while (ctxt->nameNr > oldNameNr) { | 
| 9622 | 200k |         xmlStartTag *tag = &ctxt->pushTab[ctxt->nameNr - 1]; | 
| 9623 |  |  | 
| 9624 | 200k |         if (tag->nsNr != 0) | 
| 9625 | 86.8k |             xmlParserNsPop(ctxt, tag->nsNr); | 
| 9626 |  |  | 
| 9627 | 200k |         namePop(ctxt); | 
| 9628 | 200k |     } | 
| 9629 |  |  | 
| 9630 | 208k |     while (ctxt->spaceNr > oldSpaceNr) | 
| 9631 | 200k |         spacePop(ctxt); | 
| 9632 | 8.18k | } | 
| 9633 |  |  | 
| 9634 |  | /** | 
| 9635 |  |  * Parse XML element content. This is useful if you're only interested | 
| 9636 |  |  * in custom SAX callbacks. If you want a node list, use | 
| 9637 |  |  * #xmlCtxtParseContent. | 
| 9638 |  |  * | 
| 9639 |  |  * @param ctxt  an XML parser context | 
| 9640 |  |  */ | 
| 9641 |  | void | 
| 9642 | 0 | xmlParseContent(xmlParserCtxt *ctxt) { | 
| 9643 | 0 |     if ((ctxt == NULL) || (ctxt->input == NULL)) | 
| 9644 | 0 |         return; | 
| 9645 |  |  | 
| 9646 | 0 |     xmlCtxtInitializeLate(ctxt); | 
| 9647 |  | 
 | 
| 9648 | 0 |     xmlParseContentInternal(ctxt); | 
| 9649 |  | 
 | 
| 9650 | 0 |     xmlParserCheckEOF(ctxt, XML_ERR_NOT_WELL_BALANCED); | 
| 9651 | 0 | } | 
| 9652 |  |  | 
| 9653 |  | /** | 
| 9654 |  |  * Parse an XML element | 
| 9655 |  |  * | 
| 9656 |  |  * @deprecated Internal function, don't use. | 
| 9657 |  |  * | 
| 9658 |  |  *     [39] element ::= EmptyElemTag | STag content ETag | 
| 9659 |  |  * | 
| 9660 |  |  * [ WFC: Element Type Match ] | 
| 9661 |  |  * The Name in an element's end-tag must match the element type in the | 
| 9662 |  |  * start-tag. | 
| 9663 |  |  * | 
| 9664 |  |  * @param ctxt  an XML parser context | 
| 9665 |  |  */ | 
| 9666 |  |  | 
| 9667 |  | void | 
| 9668 | 11.1k | xmlParseElement(xmlParserCtxt *ctxt) { | 
| 9669 | 11.1k |     if (xmlParseElementStart(ctxt) != 0) | 
| 9670 | 4.72k |         return; | 
| 9671 |  |  | 
| 9672 | 6.45k |     xmlParseContentInternal(ctxt); | 
| 9673 |  |  | 
| 9674 | 6.45k |     if (ctxt->input->cur >= ctxt->input->end) { | 
| 9675 | 6.00k |         if (ctxt->wellFormed) { | 
| 9676 | 677 |             const xmlChar *name = ctxt->nameTab[ctxt->nameNr - 1]; | 
| 9677 | 677 |             int line = ctxt->pushTab[ctxt->nameNr - 1].line; | 
| 9678 | 677 |             xmlFatalErrMsgStrIntStr(ctxt, XML_ERR_TAG_NOT_FINISHED, | 
| 9679 | 677 |                     "Premature end of data in tag %s line %d\n", | 
| 9680 | 677 |                     name, line, NULL); | 
| 9681 | 677 |         } | 
| 9682 | 6.00k |         return; | 
| 9683 | 6.00k |     } | 
| 9684 |  |  | 
| 9685 | 446 |     xmlParseElementEnd(ctxt); | 
| 9686 | 446 | } | 
| 9687 |  |  | 
| 9688 |  | /** | 
| 9689 |  |  * Parse the start of an XML element. Returns -1 in case of error, 0 if an | 
| 9690 |  |  * opening tag was parsed, 1 if an empty element was parsed. | 
| 9691 |  |  * | 
| 9692 |  |  * Always consumes '<'. | 
| 9693 |  |  * | 
| 9694 |  |  * @param ctxt  an XML parser context | 
| 9695 |  |  */ | 
| 9696 |  | static int | 
| 9697 | 3.27M | xmlParseElementStart(xmlParserCtxtPtr ctxt) { | 
| 9698 | 3.27M |     int maxDepth = (ctxt->options & XML_PARSE_HUGE) ? 2048 : 256; | 
| 9699 | 3.27M |     const xmlChar *name; | 
| 9700 | 3.27M |     const xmlChar *prefix = NULL; | 
| 9701 | 3.27M |     const xmlChar *URI = NULL; | 
| 9702 | 3.27M |     xmlParserNodeInfo node_info; | 
| 9703 | 3.27M |     int line; | 
| 9704 | 3.27M |     xmlNodePtr cur; | 
| 9705 | 3.27M |     int nbNs = 0; | 
| 9706 |  |  | 
| 9707 | 3.27M |     if (ctxt->nameNr > maxDepth) { | 
| 9708 | 11 |         xmlFatalErrMsgInt(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 9709 | 11 |                 "Excessive depth in document: %d use XML_PARSE_HUGE option\n", | 
| 9710 | 11 |                 ctxt->nameNr); | 
| 9711 | 11 |   return(-1); | 
| 9712 | 11 |     } | 
| 9713 |  |  | 
| 9714 |  |     /* Capture start position */ | 
| 9715 | 3.27M |     if (ctxt->record_info) { | 
| 9716 | 0 |         node_info.begin_pos = ctxt->input->consumed + | 
| 9717 | 0 |                           (CUR_PTR - ctxt->input->base); | 
| 9718 | 0 |   node_info.begin_line = ctxt->input->line; | 
| 9719 | 0 |     } | 
| 9720 |  |  | 
| 9721 | 3.27M |     if (ctxt->spaceNr == 0) | 
| 9722 | 0 |   spacePush(ctxt, -1); | 
| 9723 | 3.27M |     else if (*ctxt->space == -2) | 
| 9724 | 0 |   spacePush(ctxt, -1); | 
| 9725 | 3.27M |     else | 
| 9726 | 3.27M |   spacePush(ctxt, *ctxt->space); | 
| 9727 |  |  | 
| 9728 | 3.27M |     line = ctxt->input->line; | 
| 9729 | 3.27M | #ifdef LIBXML_SAX1_ENABLED | 
| 9730 | 3.27M |     if (ctxt->sax2) | 
| 9731 | 3.27M | #endif /* LIBXML_SAX1_ENABLED */ | 
| 9732 | 3.27M |         name = xmlParseStartTag2(ctxt, &prefix, &URI, &nbNs); | 
| 9733 | 0 | #ifdef LIBXML_SAX1_ENABLED | 
| 9734 | 0 |     else | 
| 9735 | 0 |   name = xmlParseStartTag(ctxt); | 
| 9736 | 3.27M | #endif /* LIBXML_SAX1_ENABLED */ | 
| 9737 | 3.27M |     if (name == NULL) { | 
| 9738 | 2.77M |   spacePop(ctxt); | 
| 9739 | 2.77M |         return(-1); | 
| 9740 | 2.77M |     } | 
| 9741 | 496k |     nameNsPush(ctxt, name, prefix, URI, line, nbNs); | 
| 9742 | 496k |     cur = ctxt->node; | 
| 9743 |  |  | 
| 9744 | 496k | #ifdef LIBXML_VALID_ENABLED | 
| 9745 |  |     /* | 
| 9746 |  |      * [ VC: Root Element Type ] | 
| 9747 |  |      * The Name in the document type declaration must match the element | 
| 9748 |  |      * type of the root element. | 
| 9749 |  |      */ | 
| 9750 | 496k |     if (ctxt->validate && ctxt->wellFormed && ctxt->myDoc && | 
| 9751 | 0 |         ctxt->node && (ctxt->node == ctxt->myDoc->children)) | 
| 9752 | 0 |         ctxt->valid &= xmlValidateRoot(&ctxt->vctxt, ctxt->myDoc); | 
| 9753 | 496k | #endif /* LIBXML_VALID_ENABLED */ | 
| 9754 |  |  | 
| 9755 |  |     /* | 
| 9756 |  |      * Check for an Empty Element. | 
| 9757 |  |      */ | 
| 9758 | 496k |     if ((RAW == '/') && (NXT(1) == '>')) { | 
| 9759 | 18.7k |         SKIP(2); | 
| 9760 | 18.7k |   if (ctxt->sax2) { | 
| 9761 | 18.7k |       if ((ctxt->sax != NULL) && (ctxt->sax->endElementNs != NULL) && | 
| 9762 | 18.7k |     (!ctxt->disableSAX)) | 
| 9763 | 4.82k |     ctxt->sax->endElementNs(ctxt->userData, name, prefix, URI); | 
| 9764 | 18.7k | #ifdef LIBXML_SAX1_ENABLED | 
| 9765 | 18.7k |   } else { | 
| 9766 | 0 |       if ((ctxt->sax != NULL) && (ctxt->sax->endElement != NULL) && | 
| 9767 | 0 |     (!ctxt->disableSAX)) | 
| 9768 | 0 |     ctxt->sax->endElement(ctxt->userData, name); | 
| 9769 | 0 | #endif /* LIBXML_SAX1_ENABLED */ | 
| 9770 | 0 |   } | 
| 9771 | 18.7k |   namePop(ctxt); | 
| 9772 | 18.7k |   spacePop(ctxt); | 
| 9773 | 18.7k |   if (nbNs > 0) | 
| 9774 | 1.35k |       xmlParserNsPop(ctxt, nbNs); | 
| 9775 | 18.7k |   if (cur != NULL && ctxt->record_info) { | 
| 9776 | 0 |             node_info.node = cur; | 
| 9777 | 0 |             node_info.end_pos = ctxt->input->consumed + | 
| 9778 | 0 |                                 (CUR_PTR - ctxt->input->base); | 
| 9779 | 0 |             node_info.end_line = ctxt->input->line; | 
| 9780 | 0 |             xmlParserAddNodeInfo(ctxt, &node_info); | 
| 9781 | 0 |   } | 
| 9782 | 18.7k |   return(1); | 
| 9783 | 18.7k |     } | 
| 9784 | 477k |     if (RAW == '>') { | 
| 9785 | 232k |         NEXT1; | 
| 9786 | 232k |         if (cur != NULL && ctxt->record_info) { | 
| 9787 | 0 |             node_info.node = cur; | 
| 9788 | 0 |             node_info.end_pos = 0; | 
| 9789 | 0 |             node_info.end_line = 0; | 
| 9790 | 0 |             xmlParserAddNodeInfo(ctxt, &node_info); | 
| 9791 | 0 |         } | 
| 9792 | 245k |     } else { | 
| 9793 | 245k |         xmlFatalErrMsgStrIntStr(ctxt, XML_ERR_GT_REQUIRED, | 
| 9794 | 245k |          "Couldn't find end of Start Tag %s line %d\n", | 
| 9795 | 245k |                     name, line, NULL); | 
| 9796 |  |  | 
| 9797 |  |   /* | 
| 9798 |  |    * end of parsing of this node. | 
| 9799 |  |    */ | 
| 9800 | 245k |   nodePop(ctxt); | 
| 9801 | 245k |   namePop(ctxt); | 
| 9802 | 245k |   spacePop(ctxt); | 
| 9803 | 245k |   if (nbNs > 0) | 
| 9804 | 116k |       xmlParserNsPop(ctxt, nbNs); | 
| 9805 | 245k |   return(-1); | 
| 9806 | 245k |     } | 
| 9807 |  |  | 
| 9808 | 232k |     return(0); | 
| 9809 | 477k | } | 
| 9810 |  |  | 
| 9811 |  | /** | 
| 9812 |  |  * Parse the end of an XML element. Always consumes '</'. | 
| 9813 |  |  * | 
| 9814 |  |  * @param ctxt  an XML parser context | 
| 9815 |  |  */ | 
| 9816 |  | static void | 
| 9817 | 26.2k | xmlParseElementEnd(xmlParserCtxtPtr ctxt) { | 
| 9818 | 26.2k |     xmlNodePtr cur = ctxt->node; | 
| 9819 |  |  | 
| 9820 | 26.2k |     if (ctxt->nameNr <= 0) { | 
| 9821 | 0 |         if ((RAW == '<') && (NXT(1) == '/')) | 
| 9822 | 0 |             SKIP(2); | 
| 9823 | 0 |         return; | 
| 9824 | 0 |     } | 
| 9825 |  |  | 
| 9826 |  |     /* | 
| 9827 |  |      * parse the end of tag: '</' should be here. | 
| 9828 |  |      */ | 
| 9829 | 26.2k |     if (ctxt->sax2) { | 
| 9830 | 26.2k |   xmlParseEndTag2(ctxt, &ctxt->pushTab[ctxt->nameNr - 1]); | 
| 9831 | 26.2k |   namePop(ctxt); | 
| 9832 | 26.2k |     } | 
| 9833 | 0 | #ifdef LIBXML_SAX1_ENABLED | 
| 9834 | 0 |     else | 
| 9835 | 0 |   xmlParseEndTag1(ctxt, 0); | 
| 9836 | 26.2k | #endif /* LIBXML_SAX1_ENABLED */ | 
| 9837 |  |  | 
| 9838 |  |     /* | 
| 9839 |  |      * Capture end position | 
| 9840 |  |      */ | 
| 9841 | 26.2k |     if (cur != NULL && ctxt->record_info) { | 
| 9842 | 0 |         xmlParserNodeInfoPtr node_info; | 
| 9843 |  | 
 | 
| 9844 | 0 |         node_info = (xmlParserNodeInfoPtr) xmlParserFindNodeInfo(ctxt, cur); | 
| 9845 | 0 |         if (node_info != NULL) { | 
| 9846 | 0 |             node_info->end_pos = ctxt->input->consumed + | 
| 9847 | 0 |                                  (CUR_PTR - ctxt->input->base); | 
| 9848 | 0 |             node_info->end_line = ctxt->input->line; | 
| 9849 | 0 |         } | 
| 9850 | 0 |     } | 
| 9851 | 26.2k | } | 
| 9852 |  |  | 
| 9853 |  | /** | 
| 9854 |  |  * Parse the XML version value. | 
| 9855 |  |  * | 
| 9856 |  |  * @deprecated Internal function, don't use. | 
| 9857 |  |  * | 
| 9858 |  |  *     [26] VersionNum ::= '1.' [0-9]+ | 
| 9859 |  |  * | 
| 9860 |  |  * In practice allow [0-9].[0-9]+ at that level | 
| 9861 |  |  * | 
| 9862 |  |  * @param ctxt  an XML parser context | 
| 9863 |  |  * @returns the string giving the XML version number, or NULL | 
| 9864 |  |  */ | 
| 9865 |  | xmlChar * | 
| 9866 | 656 | xmlParseVersionNum(xmlParserCtxt *ctxt) { | 
| 9867 | 656 |     xmlChar *buf = NULL; | 
| 9868 | 656 |     int len = 0; | 
| 9869 | 656 |     int size = 10; | 
| 9870 | 656 |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 9871 | 656 |                     XML_MAX_TEXT_LENGTH : | 
| 9872 | 656 |                     XML_MAX_NAME_LENGTH; | 
| 9873 | 656 |     xmlChar cur; | 
| 9874 |  |  | 
| 9875 | 656 |     buf = xmlMalloc(size); | 
| 9876 | 656 |     if (buf == NULL) { | 
| 9877 | 0 |   xmlErrMemory(ctxt); | 
| 9878 | 0 |   return(NULL); | 
| 9879 | 0 |     } | 
| 9880 | 656 |     cur = CUR; | 
| 9881 | 656 |     if (!((cur >= '0') && (cur <= '9'))) { | 
| 9882 | 21 |   xmlFree(buf); | 
| 9883 | 21 |   return(NULL); | 
| 9884 | 21 |     } | 
| 9885 | 635 |     buf[len++] = cur; | 
| 9886 | 635 |     NEXT; | 
| 9887 | 635 |     cur=CUR; | 
| 9888 | 635 |     if (cur != '.') { | 
| 9889 | 7 |   xmlFree(buf); | 
| 9890 | 7 |   return(NULL); | 
| 9891 | 7 |     } | 
| 9892 | 628 |     buf[len++] = cur; | 
| 9893 | 628 |     NEXT; | 
| 9894 | 628 |     cur=CUR; | 
| 9895 | 692k |     while ((cur >= '0') && (cur <= '9')) { | 
| 9896 | 691k |   if (len + 1 >= size) { | 
| 9897 | 331 |       xmlChar *tmp; | 
| 9898 | 331 |             int newSize; | 
| 9899 |  |  | 
| 9900 | 331 |             newSize = xmlGrowCapacity(size, 1, 1, maxLength); | 
| 9901 | 331 |             if (newSize < 0) { | 
| 9902 | 0 |                 xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "VersionNum"); | 
| 9903 | 0 |                 xmlFree(buf); | 
| 9904 | 0 |                 return(NULL); | 
| 9905 | 0 |             } | 
| 9906 | 331 |       tmp = xmlRealloc(buf, newSize); | 
| 9907 | 331 |       if (tmp == NULL) { | 
| 9908 | 0 |     xmlErrMemory(ctxt); | 
| 9909 | 0 |           xmlFree(buf); | 
| 9910 | 0 |     return(NULL); | 
| 9911 | 0 |       } | 
| 9912 | 331 |       buf = tmp; | 
| 9913 | 331 |             size = newSize; | 
| 9914 | 331 |   } | 
| 9915 | 691k |   buf[len++] = cur; | 
| 9916 | 691k |   NEXT; | 
| 9917 | 691k |   cur=CUR; | 
| 9918 | 691k |     } | 
| 9919 | 628 |     buf[len] = 0; | 
| 9920 | 628 |     return(buf); | 
| 9921 | 628 | } | 
| 9922 |  |  | 
| 9923 |  | /** | 
| 9924 |  |  * Parse the XML version. | 
| 9925 |  |  * | 
| 9926 |  |  * @deprecated Internal function, don't use. | 
| 9927 |  |  * | 
| 9928 |  |  *     [24] VersionInfo ::= S 'version' Eq (' VersionNum ' | " VersionNum ") | 
| 9929 |  |  * | 
| 9930 |  |  *     [25] Eq ::= S? '=' S? | 
| 9931 |  |  * | 
| 9932 |  |  * @param ctxt  an XML parser context | 
| 9933 |  |  * @returns the version string, e.g. "1.0" | 
| 9934 |  |  */ | 
| 9935 |  |  | 
| 9936 |  | xmlChar * | 
| 9937 | 3.19k | xmlParseVersionInfo(xmlParserCtxt *ctxt) { | 
| 9938 | 3.19k |     xmlChar *version = NULL; | 
| 9939 |  |  | 
| 9940 | 3.19k |     if (CMP7(CUR_PTR, 'v', 'e', 'r', 's', 'i', 'o', 'n')) { | 
| 9941 | 685 |   SKIP(7); | 
| 9942 | 685 |   SKIP_BLANKS; | 
| 9943 | 685 |   if (RAW != '=') { | 
| 9944 | 15 |       xmlFatalErr(ctxt, XML_ERR_EQUAL_REQUIRED, NULL); | 
| 9945 | 15 |       return(NULL); | 
| 9946 | 15 |         } | 
| 9947 | 670 |   NEXT; | 
| 9948 | 670 |   SKIP_BLANKS; | 
| 9949 | 670 |   if (RAW == '"') { | 
| 9950 | 121 |       NEXT; | 
| 9951 | 121 |       version = xmlParseVersionNum(ctxt); | 
| 9952 | 121 |       if (RAW != '"') { | 
| 9953 | 32 |     xmlFatalErr(ctxt, XML_ERR_STRING_NOT_CLOSED, NULL); | 
| 9954 | 32 |       } else | 
| 9955 | 89 |           NEXT; | 
| 9956 | 549 |   } else if (RAW == '\''){ | 
| 9957 | 535 |       NEXT; | 
| 9958 | 535 |       version = xmlParseVersionNum(ctxt); | 
| 9959 | 535 |       if (RAW != '\'') { | 
| 9960 | 32 |     xmlFatalErr(ctxt, XML_ERR_STRING_NOT_CLOSED, NULL); | 
| 9961 | 32 |       } else | 
| 9962 | 503 |           NEXT; | 
| 9963 | 535 |   } else { | 
| 9964 | 14 |       xmlFatalErr(ctxt, XML_ERR_STRING_NOT_STARTED, NULL); | 
| 9965 | 14 |   } | 
| 9966 | 670 |     } | 
| 9967 | 3.18k |     return(version); | 
| 9968 | 3.19k | } | 
| 9969 |  |  | 
| 9970 |  | /** | 
| 9971 |  |  * Parse the XML encoding name | 
| 9972 |  |  * | 
| 9973 |  |  * @deprecated Internal function, don't use. | 
| 9974 |  |  * | 
| 9975 |  |  *     [81] EncName ::= [A-Za-z] ([A-Za-z0-9._] | '-')* | 
| 9976 |  |  * | 
| 9977 |  |  * @param ctxt  an XML parser context | 
| 9978 |  |  * @returns the encoding name value or NULL | 
| 9979 |  |  */ | 
| 9980 |  | xmlChar * | 
| 9981 | 1.35k | xmlParseEncName(xmlParserCtxt *ctxt) { | 
| 9982 | 1.35k |     xmlChar *buf = NULL; | 
| 9983 | 1.35k |     int len = 0; | 
| 9984 | 1.35k |     int size = 10; | 
| 9985 | 1.35k |     int maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 9986 | 1.35k |                     XML_MAX_TEXT_LENGTH : | 
| 9987 | 1.35k |                     XML_MAX_NAME_LENGTH; | 
| 9988 | 1.35k |     xmlChar cur; | 
| 9989 |  |  | 
| 9990 | 1.35k |     cur = CUR; | 
| 9991 | 1.35k |     if (((cur >= 'a') && (cur <= 'z')) || | 
| 9992 | 1.33k |         ((cur >= 'A') && (cur <= 'Z'))) { | 
| 9993 | 1.33k |   buf = xmlMalloc(size); | 
| 9994 | 1.33k |   if (buf == NULL) { | 
| 9995 | 0 |       xmlErrMemory(ctxt); | 
| 9996 | 0 |       return(NULL); | 
| 9997 | 0 |   } | 
| 9998 |  |  | 
| 9999 | 1.33k |   buf[len++] = cur; | 
| 10000 | 1.33k |   NEXT; | 
| 10001 | 1.33k |   cur = CUR; | 
| 10002 | 1.38M |   while (((cur >= 'a') && (cur <= 'z')) || | 
| 10003 | 1.21M |          ((cur >= 'A') && (cur <= 'Z')) || | 
| 10004 | 1.18M |          ((cur >= '0') && (cur <= '9')) || | 
| 10005 | 3.43k |          (cur == '.') || (cur == '_') || | 
| 10006 | 1.38M |          (cur == '-')) { | 
| 10007 | 1.38M |       if (len + 1 >= size) { | 
| 10008 | 796 |           xmlChar *tmp; | 
| 10009 | 796 |                 int newSize; | 
| 10010 |  |  | 
| 10011 | 796 |                 newSize = xmlGrowCapacity(size, 1, 1, maxLength); | 
| 10012 | 796 |                 if (newSize < 0) { | 
| 10013 | 0 |                     xmlFatalErr(ctxt, XML_ERR_NAME_TOO_LONG, "EncName"); | 
| 10014 | 0 |                     xmlFree(buf); | 
| 10015 | 0 |                     return(NULL); | 
| 10016 | 0 |                 } | 
| 10017 | 796 |     tmp = xmlRealloc(buf, newSize); | 
| 10018 | 796 |     if (tmp == NULL) { | 
| 10019 | 0 |         xmlErrMemory(ctxt); | 
| 10020 | 0 |         xmlFree(buf); | 
| 10021 | 0 |         return(NULL); | 
| 10022 | 0 |     } | 
| 10023 | 796 |     buf = tmp; | 
| 10024 | 796 |                 size = newSize; | 
| 10025 | 796 |       } | 
| 10026 | 1.38M |       buf[len++] = cur; | 
| 10027 | 1.38M |       NEXT; | 
| 10028 | 1.38M |       cur = CUR; | 
| 10029 | 1.38M |         } | 
| 10030 | 1.33k |   buf[len] = 0; | 
| 10031 | 1.33k |     } else { | 
| 10032 | 16 |   xmlFatalErr(ctxt, XML_ERR_ENCODING_NAME, NULL); | 
| 10033 | 16 |     } | 
| 10034 | 1.35k |     return(buf); | 
| 10035 | 1.35k | } | 
| 10036 |  |  | 
| 10037 |  | /** | 
| 10038 |  |  * Parse the XML encoding declaration | 
| 10039 |  |  * | 
| 10040 |  |  * @deprecated Internal function, don't use. | 
| 10041 |  |  * | 
| 10042 |  |  *     [80] EncodingDecl ::= S 'encoding' Eq ('"' EncName '"' |  | 
| 10043 |  |  *                           "'" EncName "'") | 
| 10044 |  |  * | 
| 10045 |  |  * this setups the conversion filters. | 
| 10046 |  |  * | 
| 10047 |  |  * @param ctxt  an XML parser context | 
| 10048 |  |  * @returns the encoding value or NULL | 
| 10049 |  |  */ | 
| 10050 |  |  | 
| 10051 |  | const xmlChar * | 
| 10052 | 3.18k | xmlParseEncodingDecl(xmlParserCtxt *ctxt) { | 
| 10053 | 3.18k |     xmlChar *encoding = NULL; | 
| 10054 |  |  | 
| 10055 | 3.18k |     SKIP_BLANKS; | 
| 10056 | 3.18k |     if (CMP8(CUR_PTR, 'e', 'n', 'c', 'o', 'd', 'i', 'n', 'g') == 0) | 
| 10057 | 1.82k |         return(NULL); | 
| 10058 |  |  | 
| 10059 | 1.36k |     SKIP(8); | 
| 10060 | 1.36k |     SKIP_BLANKS; | 
| 10061 | 1.36k |     if (RAW != '=') { | 
| 10062 | 4 |         xmlFatalErr(ctxt, XML_ERR_EQUAL_REQUIRED, NULL); | 
| 10063 | 4 |         return(NULL); | 
| 10064 | 4 |     } | 
| 10065 | 1.35k |     NEXT; | 
| 10066 | 1.35k |     SKIP_BLANKS; | 
| 10067 | 1.35k |     if (RAW == '"') { | 
| 10068 | 840 |         NEXT; | 
| 10069 | 840 |         encoding = xmlParseEncName(ctxt); | 
| 10070 | 840 |         if (RAW != '"') { | 
| 10071 | 83 |             xmlFatalErr(ctxt, XML_ERR_STRING_NOT_CLOSED, NULL); | 
| 10072 | 83 |             xmlFree(encoding); | 
| 10073 | 83 |             return(NULL); | 
| 10074 | 83 |         } else | 
| 10075 | 757 |             NEXT; | 
| 10076 | 840 |     } else if (RAW == '\''){ | 
| 10077 | 512 |         NEXT; | 
| 10078 | 512 |         encoding = xmlParseEncName(ctxt); | 
| 10079 | 512 |         if (RAW != '\'') { | 
| 10080 | 7 |             xmlFatalErr(ctxt, XML_ERR_STRING_NOT_CLOSED, NULL); | 
| 10081 | 7 |             xmlFree(encoding); | 
| 10082 | 7 |             return(NULL); | 
| 10083 | 7 |         } else | 
| 10084 | 505 |             NEXT; | 
| 10085 | 512 |     } else { | 
| 10086 | 5 |         xmlFatalErr(ctxt, XML_ERR_STRING_NOT_STARTED, NULL); | 
| 10087 | 5 |     } | 
| 10088 |  |  | 
| 10089 | 1.26k |     if (encoding == NULL) | 
| 10090 | 7 |         return(NULL); | 
| 10091 |  |  | 
| 10092 | 1.26k |     xmlSetDeclaredEncoding(ctxt, encoding); | 
| 10093 |  |  | 
| 10094 | 1.26k |     return(ctxt->encoding); | 
| 10095 | 1.26k | } | 
| 10096 |  |  | 
| 10097 |  | /** | 
| 10098 |  |  * Parse the XML standalone declaration | 
| 10099 |  |  * | 
| 10100 |  |  * @deprecated Internal function, don't use. | 
| 10101 |  |  * | 
| 10102 |  |  *     [32] SDDecl ::= S 'standalone' Eq | 
| 10103 |  |  *                     (("'" ('yes' | 'no') "'") | ('"' ('yes' | 'no')'"')) | 
| 10104 |  |  * | 
| 10105 |  |  * [ VC: Standalone Document Declaration ] | 
| 10106 |  |  * TODO The standalone document declaration must have the value "no" | 
| 10107 |  |  * if any external markup declarations contain declarations of: | 
| 10108 |  |  *  - attributes with default values, if elements to which these | 
| 10109 |  |  *    attributes apply appear in the document without specifications | 
| 10110 |  |  *    of values for these attributes, or | 
| 10111 |  |  *  - entities (other than amp, lt, gt, apos, quot), if references | 
| 10112 |  |  *    to those entities appear in the document, or | 
| 10113 |  |  *  - attributes with values subject to normalization, where the | 
| 10114 |  |  *    attribute appears in the document with a value which will change | 
| 10115 |  |  *    as a result of normalization, or | 
| 10116 |  |  *  - element types with element content, if white space occurs directly | 
| 10117 |  |  *    within any instance of those types. | 
| 10118 |  |  * | 
| 10119 |  |  * @param ctxt  an XML parser context | 
| 10120 |  |  * @returns | 
| 10121 |  |  *   1 if standalone="yes" | 
| 10122 |  |  *   0 if standalone="no" | 
| 10123 |  |  *  -2 if standalone attribute is missing or invalid | 
| 10124 |  |  *    (A standalone value of -2 means that the XML declaration was found, | 
| 10125 |  |  *     but no value was specified for the standalone attribute). | 
| 10126 |  |  */ | 
| 10127 |  |  | 
| 10128 |  | int | 
| 10129 | 3.09k | xmlParseSDDecl(xmlParserCtxt *ctxt) { | 
| 10130 | 3.09k |     int standalone = -2; | 
| 10131 |  |  | 
| 10132 | 3.09k |     SKIP_BLANKS; | 
| 10133 | 3.09k |     if (CMP10(CUR_PTR, 's', 't', 'a', 'n', 'd', 'a', 'l', 'o', 'n', 'e')) { | 
| 10134 | 399 |   SKIP(10); | 
| 10135 | 399 |         SKIP_BLANKS; | 
| 10136 | 399 |   if (RAW != '=') { | 
| 10137 | 1 |       xmlFatalErr(ctxt, XML_ERR_EQUAL_REQUIRED, NULL); | 
| 10138 | 1 |       return(standalone); | 
| 10139 | 1 |         } | 
| 10140 | 398 |   NEXT; | 
| 10141 | 398 |   SKIP_BLANKS; | 
| 10142 | 398 |         if (RAW == '\''){ | 
| 10143 | 349 |       NEXT; | 
| 10144 | 349 |       if ((RAW == 'n') && (NXT(1) == 'o')) { | 
| 10145 | 334 |           standalone = 0; | 
| 10146 | 334 |                 SKIP(2); | 
| 10147 | 334 |       } else if ((RAW == 'y') && (NXT(1) == 'e') && | 
| 10148 | 3 |                  (NXT(2) == 's')) { | 
| 10149 | 2 |           standalone = 1; | 
| 10150 | 2 |     SKIP(3); | 
| 10151 | 13 |             } else { | 
| 10152 | 13 |     xmlFatalErr(ctxt, XML_ERR_STANDALONE_VALUE, NULL); | 
| 10153 | 13 |       } | 
| 10154 | 349 |       if (RAW != '\'') { | 
| 10155 | 17 |     xmlFatalErr(ctxt, XML_ERR_STRING_NOT_CLOSED, NULL); | 
| 10156 | 17 |       } else | 
| 10157 | 332 |           NEXT; | 
| 10158 | 349 |   } else if (RAW == '"'){ | 
| 10159 | 47 |       NEXT; | 
| 10160 | 47 |       if ((RAW == 'n') && (NXT(1) == 'o')) { | 
| 10161 | 2 |           standalone = 0; | 
| 10162 | 2 |     SKIP(2); | 
| 10163 | 45 |       } else if ((RAW == 'y') && (NXT(1) == 'e') && | 
| 10164 | 33 |                  (NXT(2) == 's')) { | 
| 10165 | 32 |           standalone = 1; | 
| 10166 | 32 |                 SKIP(3); | 
| 10167 | 32 |             } else { | 
| 10168 | 13 |     xmlFatalErr(ctxt, XML_ERR_STANDALONE_VALUE, NULL); | 
| 10169 | 13 |       } | 
| 10170 | 47 |       if (RAW != '"') { | 
| 10171 | 21 |     xmlFatalErr(ctxt, XML_ERR_STRING_NOT_CLOSED, NULL); | 
| 10172 | 21 |       } else | 
| 10173 | 26 |           NEXT; | 
| 10174 | 47 |   } else { | 
| 10175 | 2 |       xmlFatalErr(ctxt, XML_ERR_STRING_NOT_STARTED, NULL); | 
| 10176 | 2 |         } | 
| 10177 | 398 |     } | 
| 10178 | 3.09k |     return(standalone); | 
| 10179 | 3.09k | } | 
| 10180 |  |  | 
| 10181 |  | /** | 
| 10182 |  |  * Parse an XML declaration header | 
| 10183 |  |  * | 
| 10184 |  |  * @deprecated Internal function, don't use. | 
| 10185 |  |  * | 
| 10186 |  |  *     [23] XMLDecl ::= '<?xml' VersionInfo EncodingDecl? SDDecl? S? '?>' | 
| 10187 |  |  * @param ctxt  an XML parser context | 
| 10188 |  |  */ | 
| 10189 |  |  | 
| 10190 |  | void | 
| 10191 | 3.19k | xmlParseXMLDecl(xmlParserCtxt *ctxt) { | 
| 10192 | 3.19k |     xmlChar *version; | 
| 10193 |  |  | 
| 10194 |  |     /* | 
| 10195 |  |      * This value for standalone indicates that the document has an | 
| 10196 |  |      * XML declaration but it does not have a standalone attribute. | 
| 10197 |  |      * It will be overwritten later if a standalone attribute is found. | 
| 10198 |  |      */ | 
| 10199 |  |  | 
| 10200 | 3.19k |     ctxt->standalone = -2; | 
| 10201 |  |  | 
| 10202 |  |     /* | 
| 10203 |  |      * We know that '<?xml' is here. | 
| 10204 |  |      */ | 
| 10205 | 3.19k |     SKIP(5); | 
| 10206 |  |  | 
| 10207 | 3.19k |     if (!IS_BLANK_CH(RAW)) { | 
| 10208 | 0 |   xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, | 
| 10209 | 0 |                  "Blank needed after '<?xml'\n"); | 
| 10210 | 0 |     } | 
| 10211 | 3.19k |     SKIP_BLANKS; | 
| 10212 |  |  | 
| 10213 |  |     /* | 
| 10214 |  |      * We must have the VersionInfo here. | 
| 10215 |  |      */ | 
| 10216 | 3.19k |     version = xmlParseVersionInfo(ctxt); | 
| 10217 | 3.19k |     if (version == NULL) { | 
| 10218 | 2.56k |   xmlFatalErr(ctxt, XML_ERR_VERSION_MISSING, NULL); | 
| 10219 | 2.56k |     } else { | 
| 10220 | 628 |   if (!xmlStrEqual(version, (const xmlChar *) XML_DEFAULT_VERSION)) { | 
| 10221 |  |       /* | 
| 10222 |  |        * Changed here for XML-1.0 5th edition | 
| 10223 |  |        */ | 
| 10224 | 344 |       if (ctxt->options & XML_PARSE_OLD10) { | 
| 10225 | 0 |     xmlFatalErrMsgStr(ctxt, XML_ERR_UNKNOWN_VERSION, | 
| 10226 | 0 |                 "Unsupported version '%s'\n", | 
| 10227 | 0 |                 version); | 
| 10228 | 344 |       } else { | 
| 10229 | 344 |           if ((version[0] == '1') && ((version[1] == '.'))) { | 
| 10230 | 312 |         xmlWarningMsg(ctxt, XML_WAR_UNKNOWN_VERSION, | 
| 10231 | 312 |                       "Unsupported version '%s'\n", | 
| 10232 | 312 |           version, NULL); | 
| 10233 | 312 |     } else { | 
| 10234 | 32 |         xmlFatalErrMsgStr(ctxt, XML_ERR_UNKNOWN_VERSION, | 
| 10235 | 32 |               "Unsupported version '%s'\n", | 
| 10236 | 32 |               version); | 
| 10237 | 32 |     } | 
| 10238 | 344 |       } | 
| 10239 | 344 |   } | 
| 10240 | 628 |   if (ctxt->version != NULL) | 
| 10241 | 0 |       xmlFree(ctxt->version); | 
| 10242 | 628 |   ctxt->version = version; | 
| 10243 | 628 |     } | 
| 10244 |  |  | 
| 10245 |  |     /* | 
| 10246 |  |      * We may have the encoding declaration | 
| 10247 |  |      */ | 
| 10248 | 3.19k |     if (!IS_BLANK_CH(RAW)) { | 
| 10249 | 2.60k |         if ((RAW == '?') && (NXT(1) == '>')) { | 
| 10250 | 16 |       SKIP(2); | 
| 10251 | 16 |       return; | 
| 10252 | 16 |   } | 
| 10253 | 2.58k |   xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, "Blank needed here\n"); | 
| 10254 | 2.58k |     } | 
| 10255 | 3.18k |     xmlParseEncodingDecl(ctxt); | 
| 10256 |  |  | 
| 10257 |  |     /* | 
| 10258 |  |      * We may have the standalone status. | 
| 10259 |  |      */ | 
| 10260 | 3.18k |     if ((ctxt->encoding != NULL) && (!IS_BLANK_CH(RAW))) { | 
| 10261 | 842 |         if ((RAW == '?') && (NXT(1) == '>')) { | 
| 10262 | 85 |       SKIP(2); | 
| 10263 | 85 |       return; | 
| 10264 | 85 |   } | 
| 10265 | 757 |   xmlFatalErrMsg(ctxt, XML_ERR_SPACE_REQUIRED, "Blank needed here\n"); | 
| 10266 | 757 |     } | 
| 10267 |  |  | 
| 10268 |  |     /* | 
| 10269 |  |      * We can grow the input buffer freely at that point | 
| 10270 |  |      */ | 
| 10271 | 3.09k |     GROW; | 
| 10272 |  |  | 
| 10273 | 3.09k |     SKIP_BLANKS; | 
| 10274 | 3.09k |     ctxt->standalone = xmlParseSDDecl(ctxt); | 
| 10275 |  |  | 
| 10276 | 3.09k |     SKIP_BLANKS; | 
| 10277 | 3.09k |     if ((RAW == '?') && (NXT(1) == '>')) { | 
| 10278 | 374 |         SKIP(2); | 
| 10279 | 2.72k |     } else if (RAW == '>') { | 
| 10280 |  |         /* Deprecated old WD ... */ | 
| 10281 | 1.35k |   xmlFatalErr(ctxt, XML_ERR_XMLDECL_NOT_FINISHED, NULL); | 
| 10282 | 1.35k |   NEXT; | 
| 10283 | 1.37k |     } else { | 
| 10284 | 1.37k |         int c; | 
| 10285 |  |  | 
| 10286 | 1.37k |   xmlFatalErr(ctxt, XML_ERR_XMLDECL_NOT_FINISHED, NULL); | 
| 10287 | 1.23M |         while ((PARSER_STOPPED(ctxt) == 0) && | 
| 10288 | 1.23M |                ((c = CUR) != 0)) { | 
| 10289 | 1.23M |             NEXT; | 
| 10290 | 1.23M |             if (c == '>') | 
| 10291 | 878 |                 break; | 
| 10292 | 1.23M |         } | 
| 10293 | 1.37k |     } | 
| 10294 | 3.09k | } | 
| 10295 |  |  | 
| 10296 |  | /** | 
| 10297 |  |  * @since 2.14.0 | 
| 10298 |  |  * | 
| 10299 |  |  * @param ctxt  parser context | 
| 10300 |  |  * @returns the version from the XML declaration. | 
| 10301 |  |  */ | 
| 10302 |  | const xmlChar * | 
| 10303 | 0 | xmlCtxtGetVersion(xmlParserCtxt *ctxt) { | 
| 10304 | 0 |     if (ctxt == NULL) | 
| 10305 | 0 |         return(NULL); | 
| 10306 |  |  | 
| 10307 | 0 |     return(ctxt->version); | 
| 10308 | 0 | } | 
| 10309 |  |  | 
| 10310 |  | /** | 
| 10311 |  |  * @since 2.14.0 | 
| 10312 |  |  * | 
| 10313 |  |  * @param ctxt  parser context | 
| 10314 |  |  * @returns the value from the standalone document declaration. | 
| 10315 |  |  */ | 
| 10316 |  | int | 
| 10317 | 0 | xmlCtxtGetStandalone(xmlParserCtxt *ctxt) { | 
| 10318 | 0 |     if (ctxt == NULL) | 
| 10319 | 0 |         return(0); | 
| 10320 |  |  | 
| 10321 | 0 |     return(ctxt->standalone); | 
| 10322 | 0 | } | 
| 10323 |  |  | 
| 10324 |  | /** | 
| 10325 |  |  * Parse an XML Misc* optional field. | 
| 10326 |  |  * | 
| 10327 |  |  * @deprecated Internal function, don't use. | 
| 10328 |  |  * | 
| 10329 |  |  *     [27] Misc ::= Comment | PI |  S | 
| 10330 |  |  * @param ctxt  an XML parser context | 
| 10331 |  |  */ | 
| 10332 |  |  | 
| 10333 |  | void | 
| 10334 | 36.1k | xmlParseMisc(xmlParserCtxt *ctxt) { | 
| 10335 | 39.8k |     while (PARSER_STOPPED(ctxt) == 0) { | 
| 10336 | 39.6k |         SKIP_BLANKS; | 
| 10337 | 39.6k |         GROW; | 
| 10338 | 39.6k |         if ((RAW == '<') && (NXT(1) == '?')) { | 
| 10339 | 2.61k |       xmlParsePI(ctxt); | 
| 10340 | 37.0k |         } else if (CMP4(CUR_PTR, '<', '!', '-', '-')) { | 
| 10341 | 1.08k |       xmlParseComment(ctxt); | 
| 10342 | 35.9k |         } else { | 
| 10343 | 35.9k |             break; | 
| 10344 | 35.9k |         } | 
| 10345 | 39.6k |     } | 
| 10346 | 36.1k | } | 
| 10347 |  |  | 
| 10348 |  | static void | 
| 10349 | 17.3k | xmlFinishDocument(xmlParserCtxtPtr ctxt) { | 
| 10350 | 17.3k |     xmlDocPtr doc; | 
| 10351 |  |  | 
| 10352 |  |     /* | 
| 10353 |  |      * SAX: end of the document processing. | 
| 10354 |  |      */ | 
| 10355 | 17.3k |     if ((ctxt->sax) && (ctxt->sax->endDocument != NULL)) | 
| 10356 | 17.3k |         ctxt->sax->endDocument(ctxt->userData); | 
| 10357 |  |  | 
| 10358 |  |     /* | 
| 10359 |  |      * Remove locally kept entity definitions if the tree was not built | 
| 10360 |  |      */ | 
| 10361 | 17.3k |     doc = ctxt->myDoc; | 
| 10362 | 17.3k |     if ((doc != NULL) && | 
| 10363 | 16.1k |         (xmlStrEqual(doc->version, SAX_COMPAT_MODE))) { | 
| 10364 | 1.56k |         xmlFreeDoc(doc); | 
| 10365 | 1.56k |         ctxt->myDoc = NULL; | 
| 10366 | 1.56k |     } | 
| 10367 | 17.3k | } | 
| 10368 |  |  | 
| 10369 |  | /** | 
| 10370 |  |  * Parse an XML document and invoke the SAX handlers. This is useful | 
| 10371 |  |  * if you're only interested in custom SAX callbacks. If you want a | 
| 10372 |  |  * document tree, use #xmlCtxtParseDocument. | 
| 10373 |  |  * | 
| 10374 |  |  * @param ctxt  an XML parser context | 
| 10375 |  |  * @returns 0, -1 in case of error. | 
| 10376 |  |  */ | 
| 10377 |  |  | 
| 10378 |  | int | 
| 10379 | 17.4k | xmlParseDocument(xmlParserCtxt *ctxt) { | 
| 10380 | 17.4k |     if ((ctxt == NULL) || (ctxt->input == NULL)) | 
| 10381 | 0 |         return(-1); | 
| 10382 |  |  | 
| 10383 | 17.4k |     GROW; | 
| 10384 |  |  | 
| 10385 |  |     /* | 
| 10386 |  |      * SAX: detecting the level. | 
| 10387 |  |      */ | 
| 10388 | 17.4k |     xmlCtxtInitializeLate(ctxt); | 
| 10389 |  |  | 
| 10390 | 17.4k |     if ((ctxt->sax) && (ctxt->sax->setDocumentLocator)) { | 
| 10391 | 17.4k |         ctxt->sax->setDocumentLocator(ctxt->userData, | 
| 10392 | 17.4k |                 (xmlSAXLocator *) &xmlDefaultSAXLocator); | 
| 10393 | 17.4k |     } | 
| 10394 |  |  | 
| 10395 | 17.4k |     xmlDetectEncoding(ctxt); | 
| 10396 |  |  | 
| 10397 | 17.4k |     if (CUR == 0) { | 
| 10398 | 58 |   xmlFatalErr(ctxt, XML_ERR_DOCUMENT_EMPTY, NULL); | 
| 10399 | 58 |   return(-1); | 
| 10400 | 58 |     } | 
| 10401 |  |  | 
| 10402 | 17.3k |     GROW; | 
| 10403 | 17.3k |     if ((CMP5(CUR_PTR, '<', '?', 'x', 'm', 'l')) && (IS_BLANK_CH(NXT(5)))) { | 
| 10404 |  |  | 
| 10405 |  |   /* | 
| 10406 |  |    * Note that we will switch encoding on the fly. | 
| 10407 |  |    */ | 
| 10408 | 3.19k |   xmlParseXMLDecl(ctxt); | 
| 10409 | 3.19k |   SKIP_BLANKS; | 
| 10410 | 14.1k |     } else { | 
| 10411 | 14.1k |   ctxt->version = xmlCharStrdup(XML_DEFAULT_VERSION); | 
| 10412 | 14.1k |         if (ctxt->version == NULL) { | 
| 10413 | 0 |             xmlErrMemory(ctxt); | 
| 10414 | 0 |             return(-1); | 
| 10415 | 0 |         } | 
| 10416 | 14.1k |     } | 
| 10417 | 17.3k |     if ((ctxt->sax) && (ctxt->sax->startDocument) && (!ctxt->disableSAX)) | 
| 10418 | 14.5k |         ctxt->sax->startDocument(ctxt->userData); | 
| 10419 | 17.3k |     if ((ctxt->myDoc != NULL) && (ctxt->input != NULL) && | 
| 10420 | 14.5k |         (ctxt->input->buf != NULL) && (ctxt->input->buf->compressed >= 0)) { | 
| 10421 | 0 |   ctxt->myDoc->compression = ctxt->input->buf->compressed; | 
| 10422 | 0 |     } | 
| 10423 |  |  | 
| 10424 |  |     /* | 
| 10425 |  |      * The Misc part of the Prolog | 
| 10426 |  |      */ | 
| 10427 | 17.3k |     xmlParseMisc(ctxt); | 
| 10428 |  |  | 
| 10429 |  |     /* | 
| 10430 |  |      * Then possibly doc type declaration(s) and more Misc | 
| 10431 |  |      * (doctypedecl Misc*)? | 
| 10432 |  |      */ | 
| 10433 | 17.3k |     GROW; | 
| 10434 | 17.3k |     if (CMP9(CUR_PTR, '<', '!', 'D', 'O', 'C', 'T', 'Y', 'P', 'E')) { | 
| 10435 |  |  | 
| 10436 | 7.62k |   ctxt->inSubset = 1; | 
| 10437 | 7.62k |   xmlParseDocTypeDecl(ctxt); | 
| 10438 | 7.62k |   if (RAW == '[') { | 
| 10439 | 7.27k |       xmlParseInternalSubset(ctxt); | 
| 10440 | 7.27k |   } else if (RAW == '>') { | 
| 10441 | 100 |             NEXT; | 
| 10442 | 100 |         } | 
| 10443 |  |  | 
| 10444 |  |   /* | 
| 10445 |  |    * Create and update the external subset. | 
| 10446 |  |    */ | 
| 10447 | 7.62k |   ctxt->inSubset = 2; | 
| 10448 | 7.62k |   if ((ctxt->sax != NULL) && (ctxt->sax->externalSubset != NULL) && | 
| 10449 | 7.62k |       (!ctxt->disableSAX)) | 
| 10450 | 1.03k |       ctxt->sax->externalSubset(ctxt->userData, ctxt->intSubName, | 
| 10451 | 1.03k |                                 ctxt->extSubSystem, ctxt->extSubURI); | 
| 10452 | 7.62k |   ctxt->inSubset = 0; | 
| 10453 |  |  | 
| 10454 | 7.62k |         xmlCleanSpecialAttr(ctxt); | 
| 10455 |  |  | 
| 10456 | 7.62k |   xmlParseMisc(ctxt); | 
| 10457 | 7.62k |     } | 
| 10458 |  |  | 
| 10459 |  |     /* | 
| 10460 |  |      * Time to start parsing the tree itself | 
| 10461 |  |      */ | 
| 10462 | 17.3k |     GROW; | 
| 10463 | 17.3k |     if (RAW != '<') { | 
| 10464 | 6.18k |         if (ctxt->wellFormed) | 
| 10465 | 334 |             xmlFatalErrMsg(ctxt, XML_ERR_DOCUMENT_EMPTY, | 
| 10466 | 334 |                            "Start tag expected, '<' not found\n"); | 
| 10467 | 11.1k |     } else { | 
| 10468 | 11.1k |   xmlParseElement(ctxt); | 
| 10469 |  |  | 
| 10470 |  |   /* | 
| 10471 |  |    * The Misc part at the end | 
| 10472 |  |    */ | 
| 10473 | 11.1k |   xmlParseMisc(ctxt); | 
| 10474 |  |  | 
| 10475 | 11.1k |         xmlParserCheckEOF(ctxt, XML_ERR_DOCUMENT_END); | 
| 10476 | 11.1k |     } | 
| 10477 |  |  | 
| 10478 | 17.3k |     ctxt->instate = XML_PARSER_EOF; | 
| 10479 | 17.3k |     xmlFinishDocument(ctxt); | 
| 10480 |  |  | 
| 10481 | 17.3k |     if (! ctxt->wellFormed) { | 
| 10482 | 17.3k |   ctxt->valid = 0; | 
| 10483 | 17.3k |   return(-1); | 
| 10484 | 17.3k |     } | 
| 10485 |  |  | 
| 10486 | 30 |     return(0); | 
| 10487 | 17.3k | } | 
| 10488 |  |  | 
| 10489 |  | /** | 
| 10490 |  |  * Parse a general parsed entity | 
| 10491 |  |  * An external general parsed entity is well-formed if it matches the | 
| 10492 |  |  * production labeled extParsedEnt. | 
| 10493 |  |  * | 
| 10494 |  |  * @deprecated Internal function, don't use. | 
| 10495 |  |  * | 
| 10496 |  |  *     [78] extParsedEnt ::= TextDecl? content | 
| 10497 |  |  * | 
| 10498 |  |  * @param ctxt  an XML parser context | 
| 10499 |  |  * @returns 0, -1 in case of error. the parser context is augmented | 
| 10500 |  |  *                as a result of the parsing. | 
| 10501 |  |  */ | 
| 10502 |  |  | 
| 10503 |  | int | 
| 10504 | 0 | xmlParseExtParsedEnt(xmlParserCtxt *ctxt) { | 
| 10505 | 0 |     if ((ctxt == NULL) || (ctxt->input == NULL)) | 
| 10506 | 0 |         return(-1); | 
| 10507 |  |  | 
| 10508 | 0 |     xmlCtxtInitializeLate(ctxt); | 
| 10509 |  | 
 | 
| 10510 | 0 |     if ((ctxt->sax) && (ctxt->sax->setDocumentLocator)) { | 
| 10511 | 0 |         ctxt->sax->setDocumentLocator(ctxt->userData, | 
| 10512 | 0 |                 (xmlSAXLocator *) &xmlDefaultSAXLocator); | 
| 10513 | 0 |     } | 
| 10514 |  | 
 | 
| 10515 | 0 |     xmlDetectEncoding(ctxt); | 
| 10516 |  | 
 | 
| 10517 | 0 |     if (CUR == 0) { | 
| 10518 | 0 |   xmlFatalErr(ctxt, XML_ERR_DOCUMENT_EMPTY, NULL); | 
| 10519 | 0 |     } | 
| 10520 |  |  | 
| 10521 |  |     /* | 
| 10522 |  |      * Check for the XMLDecl in the Prolog. | 
| 10523 |  |      */ | 
| 10524 | 0 |     GROW; | 
| 10525 | 0 |     if ((CMP5(CUR_PTR, '<', '?', 'x', 'm', 'l')) && (IS_BLANK_CH(NXT(5)))) { | 
| 10526 |  |  | 
| 10527 |  |   /* | 
| 10528 |  |    * Note that we will switch encoding on the fly. | 
| 10529 |  |    */ | 
| 10530 | 0 |   xmlParseXMLDecl(ctxt); | 
| 10531 | 0 |   SKIP_BLANKS; | 
| 10532 | 0 |     } else { | 
| 10533 | 0 |   ctxt->version = xmlCharStrdup(XML_DEFAULT_VERSION); | 
| 10534 | 0 |     } | 
| 10535 | 0 |     if ((ctxt->sax) && (ctxt->sax->startDocument) && (!ctxt->disableSAX)) | 
| 10536 | 0 |         ctxt->sax->startDocument(ctxt->userData); | 
| 10537 |  |  | 
| 10538 |  |     /* | 
| 10539 |  |      * Doing validity checking on chunk doesn't make sense | 
| 10540 |  |      */ | 
| 10541 | 0 |     ctxt->options &= ~XML_PARSE_DTDVALID; | 
| 10542 | 0 |     ctxt->validate = 0; | 
| 10543 | 0 |     ctxt->depth = 0; | 
| 10544 |  | 
 | 
| 10545 | 0 |     xmlParseContentInternal(ctxt); | 
| 10546 |  | 
 | 
| 10547 | 0 |     if (ctxt->input->cur < ctxt->input->end) | 
| 10548 | 0 |   xmlFatalErr(ctxt, XML_ERR_NOT_WELL_BALANCED, NULL); | 
| 10549 |  |  | 
| 10550 |  |     /* | 
| 10551 |  |      * SAX: end of the document processing. | 
| 10552 |  |      */ | 
| 10553 | 0 |     if ((ctxt->sax) && (ctxt->sax->endDocument != NULL)) | 
| 10554 | 0 |         ctxt->sax->endDocument(ctxt->userData); | 
| 10555 |  | 
 | 
| 10556 | 0 |     if (! ctxt->wellFormed) return(-1); | 
| 10557 | 0 |     return(0); | 
| 10558 | 0 | } | 
| 10559 |  |  | 
| 10560 |  | #ifdef LIBXML_PUSH_ENABLED | 
| 10561 |  | /************************************************************************ | 
| 10562 |  |  *                  * | 
| 10563 |  |  *    Progressive parsing interfaces        * | 
| 10564 |  |  *                  * | 
| 10565 |  |  ************************************************************************/ | 
| 10566 |  |  | 
| 10567 |  | /** | 
| 10568 |  |  * Check whether the input buffer contains a character. | 
| 10569 |  |  * | 
| 10570 |  |  * @param ctxt  an XML parser context | 
| 10571 |  |  * @param c  character | 
| 10572 |  |  */ | 
| 10573 |  | static int | 
| 10574 | 0 | xmlParseLookupChar(xmlParserCtxtPtr ctxt, int c) { | 
| 10575 | 0 |     const xmlChar *cur; | 
| 10576 |  | 
 | 
| 10577 | 0 |     if (ctxt->checkIndex == 0) { | 
| 10578 | 0 |         cur = ctxt->input->cur + 1; | 
| 10579 | 0 |     } else { | 
| 10580 | 0 |         cur = ctxt->input->cur + ctxt->checkIndex; | 
| 10581 | 0 |     } | 
| 10582 |  | 
 | 
| 10583 | 0 |     if (memchr(cur, c, ctxt->input->end - cur) == NULL) { | 
| 10584 | 0 |         size_t index = ctxt->input->end - ctxt->input->cur; | 
| 10585 |  | 
 | 
| 10586 | 0 |         if (index > LONG_MAX) { | 
| 10587 | 0 |             ctxt->checkIndex = 0; | 
| 10588 | 0 |             return(1); | 
| 10589 | 0 |         } | 
| 10590 | 0 |         ctxt->checkIndex = index; | 
| 10591 | 0 |         return(0); | 
| 10592 | 0 |     } else { | 
| 10593 | 0 |         ctxt->checkIndex = 0; | 
| 10594 | 0 |         return(1); | 
| 10595 | 0 |     } | 
| 10596 | 0 | } | 
| 10597 |  |  | 
| 10598 |  | /** | 
| 10599 |  |  * Check whether the input buffer contains a string. | 
| 10600 |  |  * | 
| 10601 |  |  * @param ctxt  an XML parser context | 
| 10602 |  |  * @param startDelta  delta to apply at the start | 
| 10603 |  |  * @param str  string | 
| 10604 |  |  * @param strLen  length of string | 
| 10605 |  |  */ | 
| 10606 |  | static const xmlChar * | 
| 10607 |  | xmlParseLookupString(xmlParserCtxtPtr ctxt, size_t startDelta, | 
| 10608 | 0 |                      const char *str, size_t strLen) { | 
| 10609 | 0 |     const xmlChar *cur, *term; | 
| 10610 |  | 
 | 
| 10611 | 0 |     if (ctxt->checkIndex == 0) { | 
| 10612 | 0 |         cur = ctxt->input->cur + startDelta; | 
| 10613 | 0 |     } else { | 
| 10614 | 0 |         cur = ctxt->input->cur + ctxt->checkIndex; | 
| 10615 | 0 |     } | 
| 10616 |  | 
 | 
| 10617 | 0 |     term = BAD_CAST strstr((const char *) cur, str); | 
| 10618 | 0 |     if (term == NULL) { | 
| 10619 | 0 |         const xmlChar *end = ctxt->input->end; | 
| 10620 | 0 |         size_t index; | 
| 10621 |  |  | 
| 10622 |  |         /* Rescan (strLen - 1) characters. */ | 
| 10623 | 0 |         if ((size_t) (end - cur) < strLen) | 
| 10624 | 0 |             end = cur; | 
| 10625 | 0 |         else | 
| 10626 | 0 |             end -= strLen - 1; | 
| 10627 | 0 |         index = end - ctxt->input->cur; | 
| 10628 | 0 |         if (index > LONG_MAX) { | 
| 10629 | 0 |             ctxt->checkIndex = 0; | 
| 10630 | 0 |             return(ctxt->input->end - strLen); | 
| 10631 | 0 |         } | 
| 10632 | 0 |         ctxt->checkIndex = index; | 
| 10633 | 0 |     } else { | 
| 10634 | 0 |         ctxt->checkIndex = 0; | 
| 10635 | 0 |     } | 
| 10636 |  |  | 
| 10637 | 0 |     return(term); | 
| 10638 | 0 | } | 
| 10639 |  |  | 
| 10640 |  | /** | 
| 10641 |  |  * Check whether the input buffer contains terminated char data. | 
| 10642 |  |  * | 
| 10643 |  |  * @param ctxt  an XML parser context | 
| 10644 |  |  */ | 
| 10645 |  | static int | 
| 10646 | 0 | xmlParseLookupCharData(xmlParserCtxtPtr ctxt) { | 
| 10647 | 0 |     const xmlChar *cur = ctxt->input->cur + ctxt->checkIndex; | 
| 10648 | 0 |     const xmlChar *end = ctxt->input->end; | 
| 10649 | 0 |     size_t index; | 
| 10650 |  | 
 | 
| 10651 | 0 |     while (cur < end) { | 
| 10652 | 0 |         if ((*cur == '<') || (*cur == '&')) { | 
| 10653 | 0 |             ctxt->checkIndex = 0; | 
| 10654 | 0 |             return(1); | 
| 10655 | 0 |         } | 
| 10656 | 0 |         cur++; | 
| 10657 | 0 |     } | 
| 10658 |  |  | 
| 10659 | 0 |     index = cur - ctxt->input->cur; | 
| 10660 | 0 |     if (index > LONG_MAX) { | 
| 10661 | 0 |         ctxt->checkIndex = 0; | 
| 10662 | 0 |         return(1); | 
| 10663 | 0 |     } | 
| 10664 | 0 |     ctxt->checkIndex = index; | 
| 10665 | 0 |     return(0); | 
| 10666 | 0 | } | 
| 10667 |  |  | 
| 10668 |  | /** | 
| 10669 |  |  * Check whether there's enough data in the input buffer to finish parsing | 
| 10670 |  |  * a start tag. This has to take quotes into account. | 
| 10671 |  |  * | 
| 10672 |  |  * @param ctxt  an XML parser context | 
| 10673 |  |  */ | 
| 10674 |  | static int | 
| 10675 | 0 | xmlParseLookupGt(xmlParserCtxtPtr ctxt) { | 
| 10676 | 0 |     const xmlChar *cur; | 
| 10677 | 0 |     const xmlChar *end = ctxt->input->end; | 
| 10678 | 0 |     int state = ctxt->endCheckState; | 
| 10679 | 0 |     size_t index; | 
| 10680 |  | 
 | 
| 10681 | 0 |     if (ctxt->checkIndex == 0) | 
| 10682 | 0 |         cur = ctxt->input->cur + 1; | 
| 10683 | 0 |     else | 
| 10684 | 0 |         cur = ctxt->input->cur + ctxt->checkIndex; | 
| 10685 |  | 
 | 
| 10686 | 0 |     while (cur < end) { | 
| 10687 | 0 |         if (state) { | 
| 10688 | 0 |             if (*cur == state) | 
| 10689 | 0 |                 state = 0; | 
| 10690 | 0 |         } else if (*cur == '\'' || *cur == '"') { | 
| 10691 | 0 |             state = *cur; | 
| 10692 | 0 |         } else if (*cur == '>') { | 
| 10693 | 0 |             ctxt->checkIndex = 0; | 
| 10694 | 0 |             ctxt->endCheckState = 0; | 
| 10695 | 0 |             return(1); | 
| 10696 | 0 |         } | 
| 10697 | 0 |         cur++; | 
| 10698 | 0 |     } | 
| 10699 |  |  | 
| 10700 | 0 |     index = cur - ctxt->input->cur; | 
| 10701 | 0 |     if (index > LONG_MAX) { | 
| 10702 | 0 |         ctxt->checkIndex = 0; | 
| 10703 | 0 |         ctxt->endCheckState = 0; | 
| 10704 | 0 |         return(1); | 
| 10705 | 0 |     } | 
| 10706 | 0 |     ctxt->checkIndex = index; | 
| 10707 | 0 |     ctxt->endCheckState = state; | 
| 10708 | 0 |     return(0); | 
| 10709 | 0 | } | 
| 10710 |  |  | 
| 10711 |  | /** | 
| 10712 |  |  * Check whether there's enough data in the input buffer to finish parsing | 
| 10713 |  |  * the internal subset. | 
| 10714 |  |  * | 
| 10715 |  |  * @param ctxt  an XML parser context | 
| 10716 |  |  */ | 
| 10717 |  | static int | 
| 10718 | 0 | xmlParseLookupInternalSubset(xmlParserCtxtPtr ctxt) { | 
| 10719 |  |     /* | 
| 10720 |  |      * Sorry, but progressive parsing of the internal subset is not | 
| 10721 |  |      * supported. We first check that the full content of the internal | 
| 10722 |  |      * subset is available and parsing is launched only at that point. | 
| 10723 |  |      * Internal subset ends with "']' S? '>'" in an unescaped section and | 
| 10724 |  |      * not in a ']]>' sequence which are conditional sections. | 
| 10725 |  |      */ | 
| 10726 | 0 |     const xmlChar *cur, *start; | 
| 10727 | 0 |     const xmlChar *end = ctxt->input->end; | 
| 10728 | 0 |     int state = ctxt->endCheckState; | 
| 10729 | 0 |     size_t index; | 
| 10730 |  | 
 | 
| 10731 | 0 |     if (ctxt->checkIndex == 0) { | 
| 10732 | 0 |         cur = ctxt->input->cur + 1; | 
| 10733 | 0 |     } else { | 
| 10734 | 0 |         cur = ctxt->input->cur + ctxt->checkIndex; | 
| 10735 | 0 |     } | 
| 10736 | 0 |     start = cur; | 
| 10737 |  | 
 | 
| 10738 | 0 |     while (cur < end) { | 
| 10739 | 0 |         if (state == '-') { | 
| 10740 | 0 |             if ((*cur == '-') && | 
| 10741 | 0 |                 (cur[1] == '-') && | 
| 10742 | 0 |                 (cur[2] == '>')) { | 
| 10743 | 0 |                 state = 0; | 
| 10744 | 0 |                 cur += 3; | 
| 10745 | 0 |                 start = cur; | 
| 10746 | 0 |                 continue; | 
| 10747 | 0 |             } | 
| 10748 | 0 |         } | 
| 10749 | 0 |         else if (state == ']') { | 
| 10750 | 0 |             if (*cur == '>') { | 
| 10751 | 0 |                 ctxt->checkIndex = 0; | 
| 10752 | 0 |                 ctxt->endCheckState = 0; | 
| 10753 | 0 |                 return(1); | 
| 10754 | 0 |             } | 
| 10755 | 0 |             if (IS_BLANK_CH(*cur)) { | 
| 10756 | 0 |                 state = ' '; | 
| 10757 | 0 |             } else if (*cur != ']') { | 
| 10758 | 0 |                 state = 0; | 
| 10759 | 0 |                 start = cur; | 
| 10760 | 0 |                 continue; | 
| 10761 | 0 |             } | 
| 10762 | 0 |         } | 
| 10763 | 0 |         else if (state == ' ') { | 
| 10764 | 0 |             if (*cur == '>') { | 
| 10765 | 0 |                 ctxt->checkIndex = 0; | 
| 10766 | 0 |                 ctxt->endCheckState = 0; | 
| 10767 | 0 |                 return(1); | 
| 10768 | 0 |             } | 
| 10769 | 0 |             if (!IS_BLANK_CH(*cur)) { | 
| 10770 | 0 |                 state = 0; | 
| 10771 | 0 |                 start = cur; | 
| 10772 | 0 |                 continue; | 
| 10773 | 0 |             } | 
| 10774 | 0 |         } | 
| 10775 | 0 |         else if (state != 0) { | 
| 10776 | 0 |             if (*cur == state) { | 
| 10777 | 0 |                 state = 0; | 
| 10778 | 0 |                 start = cur + 1; | 
| 10779 | 0 |             } | 
| 10780 | 0 |         } | 
| 10781 | 0 |         else if (*cur == '<') { | 
| 10782 | 0 |             if ((cur[1] == '!') && | 
| 10783 | 0 |                 (cur[2] == '-') && | 
| 10784 | 0 |                 (cur[3] == '-')) { | 
| 10785 | 0 |                 state = '-'; | 
| 10786 | 0 |                 cur += 4; | 
| 10787 |  |                 /* Don't treat <!--> as comment */ | 
| 10788 | 0 |                 start = cur; | 
| 10789 | 0 |                 continue; | 
| 10790 | 0 |             } | 
| 10791 | 0 |         } | 
| 10792 | 0 |         else if ((*cur == '"') || (*cur == '\'') || (*cur == ']')) { | 
| 10793 | 0 |             state = *cur; | 
| 10794 | 0 |         } | 
| 10795 |  |  | 
| 10796 | 0 |         cur++; | 
| 10797 | 0 |     } | 
| 10798 |  |  | 
| 10799 |  |     /* | 
| 10800 |  |      * Rescan the three last characters to detect "<!--" and "-->" | 
| 10801 |  |      * split across chunks. | 
| 10802 |  |      */ | 
| 10803 | 0 |     if ((state == 0) || (state == '-')) { | 
| 10804 | 0 |         if (cur - start < 3) | 
| 10805 | 0 |             cur = start; | 
| 10806 | 0 |         else | 
| 10807 | 0 |             cur -= 3; | 
| 10808 | 0 |     } | 
| 10809 | 0 |     index = cur - ctxt->input->cur; | 
| 10810 | 0 |     if (index > LONG_MAX) { | 
| 10811 | 0 |         ctxt->checkIndex = 0; | 
| 10812 | 0 |         ctxt->endCheckState = 0; | 
| 10813 | 0 |         return(1); | 
| 10814 | 0 |     } | 
| 10815 | 0 |     ctxt->checkIndex = index; | 
| 10816 | 0 |     ctxt->endCheckState = state; | 
| 10817 | 0 |     return(0); | 
| 10818 | 0 | } | 
| 10819 |  |  | 
| 10820 |  | /** | 
| 10821 |  |  * Try to progress on parsing | 
| 10822 |  |  * | 
| 10823 |  |  * @param ctxt  an XML parser context | 
| 10824 |  |  * @param terminate  last chunk indicator | 
| 10825 |  |  * @returns zero if no parsing was possible | 
| 10826 |  |  */ | 
| 10827 |  | static int | 
| 10828 | 0 | xmlParseTryOrFinish(xmlParserCtxtPtr ctxt, int terminate) { | 
| 10829 | 0 |     int ret = 0; | 
| 10830 | 0 |     size_t avail; | 
| 10831 | 0 |     xmlChar cur, next; | 
| 10832 |  | 
 | 
| 10833 | 0 |     if (ctxt->input == NULL) | 
| 10834 | 0 |         return(0); | 
| 10835 |  |  | 
| 10836 | 0 |     if ((ctxt->input != NULL) && | 
| 10837 | 0 |         (ctxt->input->cur - ctxt->input->base > 4096)) { | 
| 10838 | 0 |         xmlParserShrink(ctxt); | 
| 10839 | 0 |     } | 
| 10840 |  | 
 | 
| 10841 | 0 |     while (ctxt->disableSAX == 0) { | 
| 10842 | 0 |         avail = ctxt->input->end - ctxt->input->cur; | 
| 10843 | 0 |         if (avail < 1) | 
| 10844 | 0 |       goto done; | 
| 10845 | 0 |         switch (ctxt->instate) { | 
| 10846 | 0 |             case XML_PARSER_EOF: | 
| 10847 |  |           /* | 
| 10848 |  |      * Document parsing is done ! | 
| 10849 |  |      */ | 
| 10850 | 0 |           goto done; | 
| 10851 | 0 |             case XML_PARSER_START: | 
| 10852 |  |                 /* | 
| 10853 |  |                  * Very first chars read from the document flow. | 
| 10854 |  |                  */ | 
| 10855 | 0 |                 if ((!terminate) && (avail < 4)) | 
| 10856 | 0 |                     goto done; | 
| 10857 |  |  | 
| 10858 |  |                 /* | 
| 10859 |  |                  * We need more bytes to detect EBCDIC code pages. | 
| 10860 |  |                  * See xmlDetectEBCDIC. | 
| 10861 |  |                  */ | 
| 10862 | 0 |                 if ((CMP4(CUR_PTR, 0x4C, 0x6F, 0xA7, 0x94)) && | 
| 10863 | 0 |                     (!terminate) && (avail < 200)) | 
| 10864 | 0 |                     goto done; | 
| 10865 |  |  | 
| 10866 | 0 |                 xmlDetectEncoding(ctxt); | 
| 10867 | 0 |                 ctxt->instate = XML_PARSER_XML_DECL; | 
| 10868 | 0 |     break; | 
| 10869 |  |  | 
| 10870 | 0 |             case XML_PARSER_XML_DECL: | 
| 10871 | 0 |     if ((!terminate) && (avail < 2)) | 
| 10872 | 0 |         goto done; | 
| 10873 | 0 |     cur = ctxt->input->cur[0]; | 
| 10874 | 0 |     next = ctxt->input->cur[1]; | 
| 10875 | 0 |           if ((cur == '<') && (next == '?')) { | 
| 10876 |  |         /* PI or XML decl */ | 
| 10877 | 0 |         if ((!terminate) && | 
| 10878 | 0 |                         (!xmlParseLookupString(ctxt, 2, "?>", 2))) | 
| 10879 | 0 |       goto done; | 
| 10880 | 0 |         if ((ctxt->input->cur[2] == 'x') && | 
| 10881 | 0 |       (ctxt->input->cur[3] == 'm') && | 
| 10882 | 0 |       (ctxt->input->cur[4] == 'l') && | 
| 10883 | 0 |       (IS_BLANK_CH(ctxt->input->cur[5]))) { | 
| 10884 | 0 |       ret += 5; | 
| 10885 | 0 |       xmlParseXMLDecl(ctxt); | 
| 10886 | 0 |         } else { | 
| 10887 | 0 |       ctxt->version = xmlCharStrdup(XML_DEFAULT_VERSION); | 
| 10888 | 0 |                         if (ctxt->version == NULL) { | 
| 10889 | 0 |                             xmlErrMemory(ctxt); | 
| 10890 | 0 |                             break; | 
| 10891 | 0 |                         } | 
| 10892 | 0 |         } | 
| 10893 | 0 |     } else { | 
| 10894 | 0 |         ctxt->version = xmlCharStrdup(XML_DEFAULT_VERSION); | 
| 10895 | 0 |         if (ctxt->version == NULL) { | 
| 10896 | 0 |             xmlErrMemory(ctxt); | 
| 10897 | 0 |       break; | 
| 10898 | 0 |         } | 
| 10899 | 0 |     } | 
| 10900 | 0 |                 if ((ctxt->sax) && (ctxt->sax->setDocumentLocator)) { | 
| 10901 | 0 |                     ctxt->sax->setDocumentLocator(ctxt->userData, | 
| 10902 | 0 |                             (xmlSAXLocator *) &xmlDefaultSAXLocator); | 
| 10903 | 0 |                 } | 
| 10904 | 0 |                 if ((ctxt->sax) && (ctxt->sax->startDocument) && | 
| 10905 | 0 |                     (!ctxt->disableSAX)) | 
| 10906 | 0 |                     ctxt->sax->startDocument(ctxt->userData); | 
| 10907 | 0 |                 ctxt->instate = XML_PARSER_MISC; | 
| 10908 | 0 |     break; | 
| 10909 | 0 |             case XML_PARSER_START_TAG: { | 
| 10910 | 0 |           const xmlChar *name; | 
| 10911 | 0 |     const xmlChar *prefix = NULL; | 
| 10912 | 0 |     const xmlChar *URI = NULL; | 
| 10913 | 0 |                 int line = ctxt->input->line; | 
| 10914 | 0 |     int nbNs = 0; | 
| 10915 |  | 
 | 
| 10916 | 0 |     if ((!terminate) && (avail < 2)) | 
| 10917 | 0 |         goto done; | 
| 10918 | 0 |     cur = ctxt->input->cur[0]; | 
| 10919 | 0 |           if (cur != '<') { | 
| 10920 | 0 |         xmlFatalErrMsg(ctxt, XML_ERR_DOCUMENT_EMPTY, | 
| 10921 | 0 |                                    "Start tag expected, '<' not found"); | 
| 10922 | 0 |                     ctxt->instate = XML_PARSER_EOF; | 
| 10923 | 0 |                     xmlFinishDocument(ctxt); | 
| 10924 | 0 |         goto done; | 
| 10925 | 0 |     } | 
| 10926 | 0 |     if ((!terminate) && (!xmlParseLookupGt(ctxt))) | 
| 10927 | 0 |                     goto done; | 
| 10928 | 0 |     if (ctxt->spaceNr == 0) | 
| 10929 | 0 |         spacePush(ctxt, -1); | 
| 10930 | 0 |     else if (*ctxt->space == -2) | 
| 10931 | 0 |         spacePush(ctxt, -1); | 
| 10932 | 0 |     else | 
| 10933 | 0 |         spacePush(ctxt, *ctxt->space); | 
| 10934 | 0 | #ifdef LIBXML_SAX1_ENABLED | 
| 10935 | 0 |     if (ctxt->sax2) | 
| 10936 | 0 | #endif /* LIBXML_SAX1_ENABLED */ | 
| 10937 | 0 |         name = xmlParseStartTag2(ctxt, &prefix, &URI, &nbNs); | 
| 10938 | 0 | #ifdef LIBXML_SAX1_ENABLED | 
| 10939 | 0 |     else | 
| 10940 | 0 |         name = xmlParseStartTag(ctxt); | 
| 10941 | 0 | #endif /* LIBXML_SAX1_ENABLED */ | 
| 10942 | 0 |     if (name == NULL) { | 
| 10943 | 0 |         spacePop(ctxt); | 
| 10944 | 0 |                     ctxt->instate = XML_PARSER_EOF; | 
| 10945 | 0 |                     xmlFinishDocument(ctxt); | 
| 10946 | 0 |         goto done; | 
| 10947 | 0 |     } | 
| 10948 | 0 | #ifdef LIBXML_VALID_ENABLED | 
| 10949 |  |     /* | 
| 10950 |  |      * [ VC: Root Element Type ] | 
| 10951 |  |      * The Name in the document type declaration must match | 
| 10952 |  |      * the element type of the root element. | 
| 10953 |  |      */ | 
| 10954 | 0 |     if (ctxt->validate && ctxt->wellFormed && ctxt->myDoc && | 
| 10955 | 0 |         ctxt->node && (ctxt->node == ctxt->myDoc->children)) | 
| 10956 | 0 |         ctxt->valid &= xmlValidateRoot(&ctxt->vctxt, ctxt->myDoc); | 
| 10957 | 0 | #endif /* LIBXML_VALID_ENABLED */ | 
| 10958 |  |  | 
| 10959 |  |     /* | 
| 10960 |  |      * Check for an Empty Element. | 
| 10961 |  |      */ | 
| 10962 | 0 |     if ((RAW == '/') && (NXT(1) == '>')) { | 
| 10963 | 0 |         SKIP(2); | 
| 10964 |  | 
 | 
| 10965 | 0 |         if (ctxt->sax2) { | 
| 10966 | 0 |       if ((ctxt->sax != NULL) && | 
| 10967 | 0 |           (ctxt->sax->endElementNs != NULL) && | 
| 10968 | 0 |           (!ctxt->disableSAX)) | 
| 10969 | 0 |           ctxt->sax->endElementNs(ctxt->userData, name, | 
| 10970 | 0 |                                   prefix, URI); | 
| 10971 | 0 |       if (nbNs > 0) | 
| 10972 | 0 |           xmlParserNsPop(ctxt, nbNs); | 
| 10973 | 0 | #ifdef LIBXML_SAX1_ENABLED | 
| 10974 | 0 |         } else { | 
| 10975 | 0 |       if ((ctxt->sax != NULL) && | 
| 10976 | 0 |           (ctxt->sax->endElement != NULL) && | 
| 10977 | 0 |           (!ctxt->disableSAX)) | 
| 10978 | 0 |           ctxt->sax->endElement(ctxt->userData, name); | 
| 10979 | 0 | #endif /* LIBXML_SAX1_ENABLED */ | 
| 10980 | 0 |         } | 
| 10981 | 0 |         spacePop(ctxt); | 
| 10982 | 0 |     } else if (RAW == '>') { | 
| 10983 | 0 |         NEXT; | 
| 10984 | 0 |                     nameNsPush(ctxt, name, prefix, URI, line, nbNs); | 
| 10985 | 0 |     } else { | 
| 10986 | 0 |         xmlFatalErrMsgStr(ctxt, XML_ERR_GT_REQUIRED, | 
| 10987 | 0 |            "Couldn't find end of Start Tag %s\n", | 
| 10988 | 0 |            name); | 
| 10989 | 0 |         nodePop(ctxt); | 
| 10990 | 0 |         spacePop(ctxt); | 
| 10991 | 0 |                     if (nbNs > 0) | 
| 10992 | 0 |                         xmlParserNsPop(ctxt, nbNs); | 
| 10993 | 0 |     } | 
| 10994 |  | 
 | 
| 10995 | 0 |                 if (ctxt->nameNr == 0) | 
| 10996 | 0 |                     ctxt->instate = XML_PARSER_EPILOG; | 
| 10997 | 0 |                 else | 
| 10998 | 0 |                     ctxt->instate = XML_PARSER_CONTENT; | 
| 10999 | 0 |                 break; | 
| 11000 | 0 |       } | 
| 11001 | 0 |             case XML_PARSER_CONTENT: { | 
| 11002 | 0 |     cur = ctxt->input->cur[0]; | 
| 11003 |  | 
 | 
| 11004 | 0 |     if (cur == '<') { | 
| 11005 | 0 |                     if ((!terminate) && (avail < 2)) | 
| 11006 | 0 |                         goto done; | 
| 11007 | 0 |         next = ctxt->input->cur[1]; | 
| 11008 |  | 
 | 
| 11009 | 0 |                     if (next == '/') { | 
| 11010 | 0 |                         ctxt->instate = XML_PARSER_END_TAG; | 
| 11011 | 0 |                         break; | 
| 11012 | 0 |                     } else if (next == '?') { | 
| 11013 | 0 |                         if ((!terminate) && | 
| 11014 | 0 |                             (!xmlParseLookupString(ctxt, 2, "?>", 2))) | 
| 11015 | 0 |                             goto done; | 
| 11016 | 0 |                         xmlParsePI(ctxt); | 
| 11017 | 0 |                         ctxt->instate = XML_PARSER_CONTENT; | 
| 11018 | 0 |                         break; | 
| 11019 | 0 |                     } else if (next == '!') { | 
| 11020 | 0 |                         if ((!terminate) && (avail < 3)) | 
| 11021 | 0 |                             goto done; | 
| 11022 | 0 |                         next = ctxt->input->cur[2]; | 
| 11023 |  | 
 | 
| 11024 | 0 |                         if (next == '-') { | 
| 11025 | 0 |                             if ((!terminate) && (avail < 4)) | 
| 11026 | 0 |                                 goto done; | 
| 11027 | 0 |                             if (ctxt->input->cur[3] == '-') { | 
| 11028 | 0 |                                 if ((!terminate) && | 
| 11029 | 0 |                                     (!xmlParseLookupString(ctxt, 4, "-->", 3))) | 
| 11030 | 0 |                                     goto done; | 
| 11031 | 0 |                                 xmlParseComment(ctxt); | 
| 11032 | 0 |                                 ctxt->instate = XML_PARSER_CONTENT; | 
| 11033 | 0 |                                 break; | 
| 11034 | 0 |                             } | 
| 11035 | 0 |                         } else if (next == '[') { | 
| 11036 | 0 |                             if ((!terminate) && (avail < 9)) | 
| 11037 | 0 |                                 goto done; | 
| 11038 | 0 |                             if ((ctxt->input->cur[2] == '[') && | 
| 11039 | 0 |                                 (ctxt->input->cur[3] == 'C') && | 
| 11040 | 0 |                                 (ctxt->input->cur[4] == 'D') && | 
| 11041 | 0 |                                 (ctxt->input->cur[5] == 'A') && | 
| 11042 | 0 |                                 (ctxt->input->cur[6] == 'T') && | 
| 11043 | 0 |                                 (ctxt->input->cur[7] == 'A') && | 
| 11044 | 0 |                                 (ctxt->input->cur[8] == '[')) { | 
| 11045 | 0 |                                 if ((!terminate) && | 
| 11046 | 0 |                                     (!xmlParseLookupString(ctxt, 9, "]]>", 3))) | 
| 11047 | 0 |                                     goto done; | 
| 11048 | 0 |                                 ctxt->instate = XML_PARSER_CDATA_SECTION; | 
| 11049 | 0 |                                 xmlParseCDSect(ctxt); | 
| 11050 | 0 |                                 ctxt->instate = XML_PARSER_CONTENT; | 
| 11051 | 0 |                                 break; | 
| 11052 | 0 |                             } | 
| 11053 | 0 |                         } | 
| 11054 | 0 |                     } | 
| 11055 | 0 |     } else if (cur == '&') { | 
| 11056 | 0 |         if ((!terminate) && (!xmlParseLookupChar(ctxt, ';'))) | 
| 11057 | 0 |       goto done; | 
| 11058 | 0 |         xmlParseReference(ctxt); | 
| 11059 | 0 |                     break; | 
| 11060 | 0 |     } else { | 
| 11061 |  |         /* TODO Avoid the extra copy, handle directly !!! */ | 
| 11062 |  |         /* | 
| 11063 |  |          * Goal of the following test is: | 
| 11064 |  |          *  - minimize calls to the SAX 'character' callback | 
| 11065 |  |          *    when they are mergeable | 
| 11066 |  |          *  - handle an problem for isBlank when we only parse | 
| 11067 |  |          *    a sequence of blank chars and the next one is | 
| 11068 |  |          *    not available to check against '<' presence. | 
| 11069 |  |          *  - tries to homogenize the differences in SAX | 
| 11070 |  |          *    callbacks between the push and pull versions | 
| 11071 |  |          *    of the parser. | 
| 11072 |  |          */ | 
| 11073 | 0 |         if (avail < XML_PARSER_BIG_BUFFER_SIZE) { | 
| 11074 | 0 |       if ((!terminate) && (!xmlParseLookupCharData(ctxt))) | 
| 11075 | 0 |           goto done; | 
| 11076 | 0 |                     } | 
| 11077 | 0 |                     ctxt->checkIndex = 0; | 
| 11078 | 0 |         xmlParseCharDataInternal(ctxt, !terminate); | 
| 11079 | 0 |                     break; | 
| 11080 | 0 |     } | 
| 11081 |  |  | 
| 11082 | 0 |                 ctxt->instate = XML_PARSER_START_TAG; | 
| 11083 | 0 |     break; | 
| 11084 | 0 |       } | 
| 11085 | 0 |             case XML_PARSER_END_TAG: | 
| 11086 | 0 |     if ((!terminate) && (!xmlParseLookupChar(ctxt, '>'))) | 
| 11087 | 0 |         goto done; | 
| 11088 | 0 |     if (ctxt->sax2) { | 
| 11089 | 0 |               xmlParseEndTag2(ctxt, &ctxt->pushTab[ctxt->nameNr - 1]); | 
| 11090 | 0 |         nameNsPop(ctxt); | 
| 11091 | 0 |     } | 
| 11092 | 0 | #ifdef LIBXML_SAX1_ENABLED | 
| 11093 | 0 |       else | 
| 11094 | 0 |         xmlParseEndTag1(ctxt, 0); | 
| 11095 | 0 | #endif /* LIBXML_SAX1_ENABLED */ | 
| 11096 | 0 |     if (ctxt->nameNr == 0) { | 
| 11097 | 0 |         ctxt->instate = XML_PARSER_EPILOG; | 
| 11098 | 0 |     } else { | 
| 11099 | 0 |         ctxt->instate = XML_PARSER_CONTENT; | 
| 11100 | 0 |     } | 
| 11101 | 0 |     break; | 
| 11102 | 0 |             case XML_PARSER_MISC: | 
| 11103 | 0 |             case XML_PARSER_PROLOG: | 
| 11104 | 0 |             case XML_PARSER_EPILOG: | 
| 11105 | 0 |     SKIP_BLANKS; | 
| 11106 | 0 |                 avail = ctxt->input->end - ctxt->input->cur; | 
| 11107 | 0 |     if (avail < 1) | 
| 11108 | 0 |         goto done; | 
| 11109 | 0 |     if (ctxt->input->cur[0] == '<') { | 
| 11110 | 0 |                     if ((!terminate) && (avail < 2)) | 
| 11111 | 0 |                         goto done; | 
| 11112 | 0 |                     next = ctxt->input->cur[1]; | 
| 11113 | 0 |                     if (next == '?') { | 
| 11114 | 0 |                         if ((!terminate) && | 
| 11115 | 0 |                             (!xmlParseLookupString(ctxt, 2, "?>", 2))) | 
| 11116 | 0 |                             goto done; | 
| 11117 | 0 |                         xmlParsePI(ctxt); | 
| 11118 | 0 |                         break; | 
| 11119 | 0 |                     } else if (next == '!') { | 
| 11120 | 0 |                         if ((!terminate) && (avail < 3)) | 
| 11121 | 0 |                             goto done; | 
| 11122 |  |  | 
| 11123 | 0 |                         if (ctxt->input->cur[2] == '-') { | 
| 11124 | 0 |                             if ((!terminate) && (avail < 4)) | 
| 11125 | 0 |                                 goto done; | 
| 11126 | 0 |                             if (ctxt->input->cur[3] == '-') { | 
| 11127 | 0 |                                 if ((!terminate) && | 
| 11128 | 0 |                                     (!xmlParseLookupString(ctxt, 4, "-->", 3))) | 
| 11129 | 0 |                                     goto done; | 
| 11130 | 0 |                                 xmlParseComment(ctxt); | 
| 11131 | 0 |                                 break; | 
| 11132 | 0 |                             } | 
| 11133 | 0 |                         } else if (ctxt->instate == XML_PARSER_MISC) { | 
| 11134 | 0 |                             if ((!terminate) && (avail < 9)) | 
| 11135 | 0 |                                 goto done; | 
| 11136 | 0 |                             if ((ctxt->input->cur[2] == 'D') && | 
| 11137 | 0 |                                 (ctxt->input->cur[3] == 'O') && | 
| 11138 | 0 |                                 (ctxt->input->cur[4] == 'C') && | 
| 11139 | 0 |                                 (ctxt->input->cur[5] == 'T') && | 
| 11140 | 0 |                                 (ctxt->input->cur[6] == 'Y') && | 
| 11141 | 0 |                                 (ctxt->input->cur[7] == 'P') && | 
| 11142 | 0 |                                 (ctxt->input->cur[8] == 'E')) { | 
| 11143 | 0 |                                 if ((!terminate) && (!xmlParseLookupGt(ctxt))) | 
| 11144 | 0 |                                     goto done; | 
| 11145 | 0 |                                 ctxt->inSubset = 1; | 
| 11146 | 0 |                                 xmlParseDocTypeDecl(ctxt); | 
| 11147 | 0 |                                 if (RAW == '[') { | 
| 11148 | 0 |                                     ctxt->instate = XML_PARSER_DTD; | 
| 11149 | 0 |                                 } else { | 
| 11150 | 0 |                                     if (RAW == '>') | 
| 11151 | 0 |                                         NEXT; | 
| 11152 |  |                                     /* | 
| 11153 |  |                                      * Create and update the external subset. | 
| 11154 |  |                                      */ | 
| 11155 | 0 |                                     ctxt->inSubset = 2; | 
| 11156 | 0 |                                     if ((ctxt->sax != NULL) && | 
| 11157 | 0 |                                         (!ctxt->disableSAX) && | 
| 11158 | 0 |                                         (ctxt->sax->externalSubset != NULL)) | 
| 11159 | 0 |                                         ctxt->sax->externalSubset( | 
| 11160 | 0 |                                                 ctxt->userData, | 
| 11161 | 0 |                                                 ctxt->intSubName, | 
| 11162 | 0 |                                                 ctxt->extSubSystem, | 
| 11163 | 0 |                                                 ctxt->extSubURI); | 
| 11164 | 0 |                                     ctxt->inSubset = 0; | 
| 11165 | 0 |                                     xmlCleanSpecialAttr(ctxt); | 
| 11166 | 0 |                                     ctxt->instate = XML_PARSER_PROLOG; | 
| 11167 | 0 |                                 } | 
| 11168 | 0 |                                 break; | 
| 11169 | 0 |                             } | 
| 11170 | 0 |                         } | 
| 11171 | 0 |                     } | 
| 11172 | 0 |                 } | 
| 11173 |  |  | 
| 11174 | 0 |                 if (ctxt->instate == XML_PARSER_EPILOG) { | 
| 11175 | 0 |                     if (ctxt->errNo == XML_ERR_OK) | 
| 11176 | 0 |                         xmlFatalErr(ctxt, XML_ERR_DOCUMENT_END, NULL); | 
| 11177 | 0 |         ctxt->instate = XML_PARSER_EOF; | 
| 11178 | 0 |                     xmlFinishDocument(ctxt); | 
| 11179 | 0 |                 } else { | 
| 11180 | 0 |         ctxt->instate = XML_PARSER_START_TAG; | 
| 11181 | 0 |     } | 
| 11182 | 0 |     break; | 
| 11183 | 0 |             case XML_PARSER_DTD: { | 
| 11184 | 0 |                 if ((!terminate) && (!xmlParseLookupInternalSubset(ctxt))) | 
| 11185 | 0 |                     goto done; | 
| 11186 | 0 |     xmlParseInternalSubset(ctxt); | 
| 11187 | 0 |     ctxt->inSubset = 2; | 
| 11188 | 0 |     if ((ctxt->sax != NULL) && (!ctxt->disableSAX) && | 
| 11189 | 0 |         (ctxt->sax->externalSubset != NULL)) | 
| 11190 | 0 |         ctxt->sax->externalSubset(ctxt->userData, ctxt->intSubName, | 
| 11191 | 0 |           ctxt->extSubSystem, ctxt->extSubURI); | 
| 11192 | 0 |     ctxt->inSubset = 0; | 
| 11193 | 0 |     xmlCleanSpecialAttr(ctxt); | 
| 11194 | 0 |     ctxt->instate = XML_PARSER_PROLOG; | 
| 11195 | 0 |                 break; | 
| 11196 | 0 |       } | 
| 11197 | 0 |             default: | 
| 11198 | 0 |                 xmlFatalErrMsg(ctxt, XML_ERR_INTERNAL_ERROR, | 
| 11199 | 0 |       "PP: internal error\n"); | 
| 11200 | 0 |     ctxt->instate = XML_PARSER_EOF; | 
| 11201 | 0 |     break; | 
| 11202 | 0 |   } | 
| 11203 | 0 |     } | 
| 11204 | 0 | done: | 
| 11205 | 0 |     return(ret); | 
| 11206 | 0 | } | 
| 11207 |  |  | 
| 11208 |  | /** | 
| 11209 |  |  * Parse a chunk of memory in push parser mode. | 
| 11210 |  |  * | 
| 11211 |  |  * Assumes that the parser context was initialized with | 
| 11212 |  |  * #xmlCreatePushParserCtxt. | 
| 11213 |  |  * | 
| 11214 |  |  * The last chunk, which will often be empty, must be marked with | 
| 11215 |  |  * the `terminate` flag. With the default SAX callbacks, the resulting | 
| 11216 |  |  * document will be available in ctxt->myDoc. This pointer will not | 
| 11217 |  |  * be freed when calling #xmlFreeParserCtxt and must be freed by the | 
| 11218 |  |  * caller. If the document isn't well-formed, it will still be returned | 
| 11219 |  |  * in ctxt->myDoc. | 
| 11220 |  |  * | 
| 11221 |  |  * As an exception, #xmlCtxtResetPush will free the document in | 
| 11222 |  |  * ctxt->myDoc. So ctxt->myDoc should be set to NULL after extracting | 
| 11223 |  |  * the document. | 
| 11224 |  |  * | 
| 11225 |  |  * Since 2.14.0, #xmlCtxtGetDocument can be used to retrieve the | 
| 11226 |  |  * result document. | 
| 11227 |  |  * | 
| 11228 |  |  * @param ctxt  an XML parser context | 
| 11229 |  |  * @param chunk  chunk of memory | 
| 11230 |  |  * @param size  size of chunk in bytes | 
| 11231 |  |  * @param terminate  last chunk indicator | 
| 11232 |  |  * @returns an xmlParserErrors code (0 on success). | 
| 11233 |  |  */ | 
| 11234 |  | int | 
| 11235 |  | xmlParseChunk(xmlParserCtxt *ctxt, const char *chunk, int size, | 
| 11236 | 0 |               int terminate) { | 
| 11237 | 0 |     size_t curBase; | 
| 11238 | 0 |     size_t maxLength; | 
| 11239 | 0 |     size_t pos; | 
| 11240 | 0 |     int end_in_lf = 0; | 
| 11241 | 0 |     int res; | 
| 11242 |  | 
 | 
| 11243 | 0 |     if ((ctxt == NULL) || (size < 0)) | 
| 11244 | 0 |         return(XML_ERR_ARGUMENT); | 
| 11245 | 0 |     if ((chunk == NULL) && (size > 0)) | 
| 11246 | 0 |         return(XML_ERR_ARGUMENT); | 
| 11247 | 0 |     if ((ctxt->input == NULL) || (ctxt->input->buf == NULL)) | 
| 11248 | 0 |         return(XML_ERR_ARGUMENT); | 
| 11249 | 0 |     if (ctxt->disableSAX != 0) | 
| 11250 | 0 |         return(ctxt->errNo); | 
| 11251 |  |  | 
| 11252 | 0 |     ctxt->input->flags |= XML_INPUT_PROGRESSIVE; | 
| 11253 | 0 |     if (ctxt->instate == XML_PARSER_START) | 
| 11254 | 0 |         xmlCtxtInitializeLate(ctxt); | 
| 11255 | 0 |     if ((size > 0) && (chunk != NULL) && (!terminate) && | 
| 11256 | 0 |         (chunk[size - 1] == '\r')) { | 
| 11257 | 0 |   end_in_lf = 1; | 
| 11258 | 0 |   size--; | 
| 11259 | 0 |     } | 
| 11260 |  |  | 
| 11261 |  |     /* | 
| 11262 |  |      * Also push an empty chunk to make sure that the raw buffer | 
| 11263 |  |      * will be flushed if there is an encoder. | 
| 11264 |  |      */ | 
| 11265 | 0 |     pos = ctxt->input->cur - ctxt->input->base; | 
| 11266 | 0 |     res = xmlParserInputBufferPush(ctxt->input->buf, size, chunk); | 
| 11267 | 0 |     xmlBufUpdateInput(ctxt->input->buf->buffer, ctxt->input, pos); | 
| 11268 | 0 |     if (res < 0) { | 
| 11269 | 0 |         xmlCtxtErrIO(ctxt, ctxt->input->buf->error, NULL); | 
| 11270 | 0 |         return(ctxt->errNo); | 
| 11271 | 0 |     } | 
| 11272 |  |  | 
| 11273 | 0 |     xmlParseTryOrFinish(ctxt, terminate); | 
| 11274 |  | 
 | 
| 11275 | 0 |     curBase = ctxt->input->cur - ctxt->input->base; | 
| 11276 | 0 |     maxLength = (ctxt->options & XML_PARSE_HUGE) ? | 
| 11277 | 0 |                 XML_MAX_HUGE_LENGTH : | 
| 11278 | 0 |                 XML_MAX_LOOKUP_LIMIT; | 
| 11279 | 0 |     if (curBase > maxLength) { | 
| 11280 | 0 |         xmlFatalErr(ctxt, XML_ERR_RESOURCE_LIMIT, | 
| 11281 | 0 |                     "Buffer size limit exceeded, try XML_PARSE_HUGE\n"); | 
| 11282 | 0 |     } | 
| 11283 |  | 
 | 
| 11284 | 0 |     if ((ctxt->errNo != XML_ERR_OK) && (ctxt->disableSAX != 0)) | 
| 11285 | 0 |         return(ctxt->errNo); | 
| 11286 |  |  | 
| 11287 | 0 |     if (end_in_lf == 1) { | 
| 11288 | 0 |   pos = ctxt->input->cur - ctxt->input->base; | 
| 11289 | 0 |   res = xmlParserInputBufferPush(ctxt->input->buf, 1, "\r"); | 
| 11290 | 0 |   xmlBufUpdateInput(ctxt->input->buf->buffer, ctxt->input, pos); | 
| 11291 | 0 |         if (res < 0) { | 
| 11292 | 0 |             xmlCtxtErrIO(ctxt, ctxt->input->buf->error, NULL); | 
| 11293 | 0 |             return(ctxt->errNo); | 
| 11294 | 0 |         } | 
| 11295 | 0 |     } | 
| 11296 | 0 |     if (terminate) { | 
| 11297 |  |   /* | 
| 11298 |  |    * Check for termination | 
| 11299 |  |    */ | 
| 11300 | 0 |         if ((ctxt->instate != XML_PARSER_EOF) && | 
| 11301 | 0 |             (ctxt->instate != XML_PARSER_EPILOG)) { | 
| 11302 | 0 |             if (ctxt->nameNr > 0) { | 
| 11303 | 0 |                 const xmlChar *name = ctxt->nameTab[ctxt->nameNr - 1]; | 
| 11304 | 0 |                 int line = ctxt->pushTab[ctxt->nameNr - 1].line; | 
| 11305 | 0 |                 xmlFatalErrMsgStrIntStr(ctxt, XML_ERR_TAG_NOT_FINISHED, | 
| 11306 | 0 |                         "Premature end of data in tag %s line %d\n", | 
| 11307 | 0 |                         name, line, NULL); | 
| 11308 | 0 |             } else if (ctxt->instate == XML_PARSER_START) { | 
| 11309 | 0 |                 xmlFatalErr(ctxt, XML_ERR_DOCUMENT_EMPTY, NULL); | 
| 11310 | 0 |             } else { | 
| 11311 | 0 |                 xmlFatalErrMsg(ctxt, XML_ERR_DOCUMENT_EMPTY, | 
| 11312 | 0 |                                "Start tag expected, '<' not found\n"); | 
| 11313 | 0 |             } | 
| 11314 | 0 |         } else { | 
| 11315 | 0 |             xmlParserCheckEOF(ctxt, XML_ERR_DOCUMENT_END); | 
| 11316 | 0 |         } | 
| 11317 | 0 |   if (ctxt->instate != XML_PARSER_EOF) { | 
| 11318 | 0 |             ctxt->instate = XML_PARSER_EOF; | 
| 11319 | 0 |             xmlFinishDocument(ctxt); | 
| 11320 | 0 |   } | 
| 11321 | 0 |     } | 
| 11322 | 0 |     if (ctxt->wellFormed == 0) | 
| 11323 | 0 |   return((xmlParserErrors) ctxt->errNo); | 
| 11324 | 0 |     else | 
| 11325 | 0 |         return(0); | 
| 11326 | 0 | } | 
| 11327 |  |  | 
| 11328 |  | /************************************************************************ | 
| 11329 |  |  *                  * | 
| 11330 |  |  *    I/O front end functions to the parser     * | 
| 11331 |  |  *                  * | 
| 11332 |  |  ************************************************************************/ | 
| 11333 |  |  | 
| 11334 |  | /** | 
| 11335 |  |  * Create a parser context for using the XML parser in push mode. | 
| 11336 |  |  * See #xmlParseChunk. | 
| 11337 |  |  * | 
| 11338 |  |  * Passing an initial chunk is useless and deprecated. | 
| 11339 |  |  * | 
| 11340 |  |  * The push parser doesn't support recovery mode or the | 
| 11341 |  |  * XML_PARSE_NOBLANKS option. | 
| 11342 |  |  * | 
| 11343 |  |  * `filename` is used as base URI to fetch external entities and for | 
| 11344 |  |  * error reports. | 
| 11345 |  |  * | 
| 11346 |  |  * @param sax  a SAX handler (optional) | 
| 11347 |  |  * @param user_data  user data for SAX callbacks (optional) | 
| 11348 |  |  * @param chunk  initial chunk (optional, deprecated) | 
| 11349 |  |  * @param size  size of initial chunk in bytes | 
| 11350 |  |  * @param filename  file name or URI (optional) | 
| 11351 |  |  * @returns the new parser context or NULL if a memory allocation | 
| 11352 |  |  * failed. | 
| 11353 |  |  */ | 
| 11354 |  |  | 
| 11355 |  | xmlParserCtxt * | 
| 11356 |  | xmlCreatePushParserCtxt(xmlSAXHandler *sax, void *user_data, | 
| 11357 | 0 |                         const char *chunk, int size, const char *filename) { | 
| 11358 | 0 |     xmlParserCtxtPtr ctxt; | 
| 11359 | 0 |     xmlParserInputPtr input; | 
| 11360 |  | 
 | 
| 11361 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, user_data); | 
| 11362 | 0 |     if (ctxt == NULL) | 
| 11363 | 0 |   return(NULL); | 
| 11364 |  |  | 
| 11365 | 0 |     ctxt->options &= ~XML_PARSE_NODICT; | 
| 11366 | 0 |     ctxt->dictNames = 1; | 
| 11367 |  | 
 | 
| 11368 | 0 |     input = xmlNewPushInput(filename, chunk, size); | 
| 11369 | 0 |     if (input == NULL) { | 
| 11370 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 11371 | 0 |   return(NULL); | 
| 11372 | 0 |     } | 
| 11373 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 11374 | 0 |         xmlFreeInputStream(input); | 
| 11375 | 0 |         xmlFreeParserCtxt(ctxt); | 
| 11376 | 0 |         return(NULL); | 
| 11377 | 0 |     } | 
| 11378 |  |  | 
| 11379 | 0 |     return(ctxt); | 
| 11380 | 0 | } | 
| 11381 |  | #endif /* LIBXML_PUSH_ENABLED */ | 
| 11382 |  |  | 
| 11383 |  | /** | 
| 11384 |  |  * Blocks further parser processing | 
| 11385 |  |  * | 
| 11386 |  |  * @param ctxt  an XML parser context | 
| 11387 |  |  */ | 
| 11388 |  | void | 
| 11389 | 0 | xmlStopParser(xmlParserCtxt *ctxt) { | 
| 11390 | 0 |     if (ctxt == NULL) | 
| 11391 | 0 |         return; | 
| 11392 |  |  | 
| 11393 |  |     /* This stops the parser */ | 
| 11394 | 0 |     ctxt->disableSAX = 2; | 
| 11395 |  |  | 
| 11396 |  |     /* | 
| 11397 |  |      * xmlStopParser is often called from error handlers, | 
| 11398 |  |      * so we can't raise an error here to avoid infinite | 
| 11399 |  |      * loops. Just make sure that an error condition is | 
| 11400 |  |      * reported. | 
| 11401 |  |      */ | 
| 11402 | 0 |     if (ctxt->errNo == XML_ERR_OK) { | 
| 11403 | 0 |         ctxt->errNo = XML_ERR_USER_STOP; | 
| 11404 | 0 |         ctxt->lastError.code = XML_ERR_USER_STOP; | 
| 11405 | 0 |         ctxt->wellFormed = 0; | 
| 11406 | 0 |     } | 
| 11407 | 0 | } | 
| 11408 |  |  | 
| 11409 |  | /** | 
| 11410 |  |  * Create a parser context for using the XML parser with an existing | 
| 11411 |  |  * I/O stream | 
| 11412 |  |  * | 
| 11413 |  |  * @param sax  a SAX handler (optional) | 
| 11414 |  |  * @param user_data  user data for SAX callbacks (optional) | 
| 11415 |  |  * @param ioread  an I/O read function | 
| 11416 |  |  * @param ioclose  an I/O close function (optional) | 
| 11417 |  |  * @param ioctx  an I/O handler | 
| 11418 |  |  * @param enc  the charset encoding if known (deprecated) | 
| 11419 |  |  * @returns the new parser context or NULL | 
| 11420 |  |  */ | 
| 11421 |  | xmlParserCtxt * | 
| 11422 |  | xmlCreateIOParserCtxt(xmlSAXHandler *sax, void *user_data, | 
| 11423 |  |                       xmlInputReadCallback ioread, | 
| 11424 |  |                       xmlInputCloseCallback ioclose, | 
| 11425 | 0 |                       void *ioctx, xmlCharEncoding enc) { | 
| 11426 | 0 |     xmlParserCtxtPtr ctxt; | 
| 11427 | 0 |     xmlParserInputPtr input; | 
| 11428 | 0 |     const char *encoding; | 
| 11429 |  | 
 | 
| 11430 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, user_data); | 
| 11431 | 0 |     if (ctxt == NULL) | 
| 11432 | 0 |   return(NULL); | 
| 11433 |  |  | 
| 11434 | 0 |     encoding = xmlGetCharEncodingName(enc); | 
| 11435 | 0 |     input = xmlCtxtNewInputFromIO(ctxt, NULL, ioread, ioclose, ioctx, | 
| 11436 | 0 |                                   encoding, 0); | 
| 11437 | 0 |     if (input == NULL) { | 
| 11438 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 11439 | 0 |         return (NULL); | 
| 11440 | 0 |     } | 
| 11441 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 11442 | 0 |         xmlFreeInputStream(input); | 
| 11443 | 0 |         xmlFreeParserCtxt(ctxt); | 
| 11444 | 0 |         return(NULL); | 
| 11445 | 0 |     } | 
| 11446 |  |  | 
| 11447 | 0 |     return(ctxt); | 
| 11448 | 0 | } | 
| 11449 |  |  | 
| 11450 |  | #ifdef LIBXML_VALID_ENABLED | 
| 11451 |  | /************************************************************************ | 
| 11452 |  |  *                  * | 
| 11453 |  |  *    Front ends when parsing a DTD       * | 
| 11454 |  |  *                  * | 
| 11455 |  |  ************************************************************************/ | 
| 11456 |  |  | 
| 11457 |  | /** | 
| 11458 |  |  * Parse a DTD. | 
| 11459 |  |  * | 
| 11460 |  |  * Option XML_PARSE_DTDLOAD should be enabled in the parser context | 
| 11461 |  |  * to make external entities work. | 
| 11462 |  |  * | 
| 11463 |  |  * @since 2.14.0 | 
| 11464 |  |  * | 
| 11465 |  |  * @param ctxt  a parser context | 
| 11466 |  |  * @param input  a parser input | 
| 11467 |  |  * @param publicId  public ID of the DTD (optional) | 
| 11468 |  |  * @param systemId  system ID of the DTD (optional) | 
| 11469 |  |  * @returns the resulting xmlDtd or NULL in case of error. | 
| 11470 |  |  * `input` will be freed by the function in any case. | 
| 11471 |  |  */ | 
| 11472 |  | xmlDtd * | 
| 11473 |  | xmlCtxtParseDtd(xmlParserCtxt *ctxt, xmlParserInput *input, | 
| 11474 | 0 |                 const xmlChar *publicId, const xmlChar *systemId) { | 
| 11475 | 0 |     xmlDtdPtr ret = NULL; | 
| 11476 |  | 
 | 
| 11477 | 0 |     if ((ctxt == NULL) || (input == NULL)) { | 
| 11478 | 0 |         xmlFatalErr(ctxt, XML_ERR_ARGUMENT, NULL); | 
| 11479 | 0 |         xmlFreeInputStream(input); | 
| 11480 | 0 |         return(NULL); | 
| 11481 | 0 |     } | 
| 11482 |  |  | 
| 11483 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 11484 | 0 |         xmlFreeInputStream(input); | 
| 11485 | 0 |         return(NULL); | 
| 11486 | 0 |     } | 
| 11487 |  |  | 
| 11488 | 0 |     if (publicId == NULL) | 
| 11489 | 0 |         publicId = BAD_CAST "none"; | 
| 11490 | 0 |     if (systemId == NULL) | 
| 11491 | 0 |         systemId = BAD_CAST "none"; | 
| 11492 |  | 
 | 
| 11493 | 0 |     ctxt->myDoc = xmlNewDoc(BAD_CAST "1.0"); | 
| 11494 | 0 |     if (ctxt->myDoc == NULL) { | 
| 11495 | 0 |         xmlErrMemory(ctxt); | 
| 11496 | 0 |         goto error; | 
| 11497 | 0 |     } | 
| 11498 | 0 |     ctxt->myDoc->properties = XML_DOC_INTERNAL; | 
| 11499 | 0 |     ctxt->myDoc->extSubset = xmlNewDtd(ctxt->myDoc, BAD_CAST "none", | 
| 11500 | 0 |                                        publicId, systemId); | 
| 11501 | 0 |     if (ctxt->myDoc->extSubset == NULL) { | 
| 11502 | 0 |         xmlErrMemory(ctxt); | 
| 11503 | 0 |         xmlFreeDoc(ctxt->myDoc); | 
| 11504 | 0 |         goto error; | 
| 11505 | 0 |     } | 
| 11506 |  |  | 
| 11507 | 0 |     xmlParseExternalSubset(ctxt, publicId, systemId); | 
| 11508 |  | 
 | 
| 11509 | 0 |     if (ctxt->wellFormed) { | 
| 11510 | 0 |         ret = ctxt->myDoc->extSubset; | 
| 11511 | 0 |         ctxt->myDoc->extSubset = NULL; | 
| 11512 | 0 |         if (ret != NULL) { | 
| 11513 | 0 |             xmlNodePtr tmp; | 
| 11514 |  | 
 | 
| 11515 | 0 |             ret->doc = NULL; | 
| 11516 | 0 |             tmp = ret->children; | 
| 11517 | 0 |             while (tmp != NULL) { | 
| 11518 | 0 |                 tmp->doc = NULL; | 
| 11519 | 0 |                 tmp = tmp->next; | 
| 11520 | 0 |             } | 
| 11521 | 0 |         } | 
| 11522 | 0 |     } else { | 
| 11523 | 0 |         ret = NULL; | 
| 11524 | 0 |     } | 
| 11525 | 0 |     xmlFreeDoc(ctxt->myDoc); | 
| 11526 | 0 |     ctxt->myDoc = NULL; | 
| 11527 |  | 
 | 
| 11528 | 0 | error: | 
| 11529 | 0 |     xmlFreeInputStream(xmlCtxtPopInput(ctxt)); | 
| 11530 |  | 
 | 
| 11531 | 0 |     return(ret); | 
| 11532 | 0 | } | 
| 11533 |  |  | 
| 11534 |  | /** | 
| 11535 |  |  * Load and parse a DTD | 
| 11536 |  |  * | 
| 11537 |  |  * @deprecated Use #xmlCtxtParseDtd. | 
| 11538 |  |  * | 
| 11539 |  |  * @param sax  the SAX handler block or NULL | 
| 11540 |  |  * @param input  an Input Buffer | 
| 11541 |  |  * @param enc  the charset encoding if known | 
| 11542 |  |  * @returns the resulting xmlDtd or NULL in case of error. | 
| 11543 |  |  * `input` will be freed by the function in any case. | 
| 11544 |  |  */ | 
| 11545 |  |  | 
| 11546 |  | xmlDtd * | 
| 11547 |  | xmlIOParseDTD(xmlSAXHandler *sax, xmlParserInputBuffer *input, | 
| 11548 | 0 |         xmlCharEncoding enc) { | 
| 11549 | 0 |     xmlDtdPtr ret = NULL; | 
| 11550 | 0 |     xmlParserCtxtPtr ctxt; | 
| 11551 | 0 |     xmlParserInputPtr pinput = NULL; | 
| 11552 |  | 
 | 
| 11553 | 0 |     if (input == NULL) | 
| 11554 | 0 |   return(NULL); | 
| 11555 |  |  | 
| 11556 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, NULL); | 
| 11557 | 0 |     if (ctxt == NULL) { | 
| 11558 | 0 |         xmlFreeParserInputBuffer(input); | 
| 11559 | 0 |   return(NULL); | 
| 11560 | 0 |     } | 
| 11561 | 0 |     xmlCtxtSetOptions(ctxt, XML_PARSE_DTDLOAD); | 
| 11562 |  |  | 
| 11563 |  |     /* | 
| 11564 |  |      * generate a parser input from the I/O handler | 
| 11565 |  |      */ | 
| 11566 |  | 
 | 
| 11567 | 0 |     pinput = xmlNewIOInputStream(ctxt, input, XML_CHAR_ENCODING_NONE); | 
| 11568 | 0 |     if (pinput == NULL) { | 
| 11569 | 0 |         xmlFreeParserInputBuffer(input); | 
| 11570 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 11571 | 0 |   return(NULL); | 
| 11572 | 0 |     } | 
| 11573 |  |  | 
| 11574 | 0 |     if (enc != XML_CHAR_ENCODING_NONE) { | 
| 11575 | 0 |         xmlSwitchEncoding(ctxt, enc); | 
| 11576 | 0 |     } | 
| 11577 |  | 
 | 
| 11578 | 0 |     ret = xmlCtxtParseDtd(ctxt, pinput, NULL, NULL); | 
| 11579 |  | 
 | 
| 11580 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 11581 | 0 |     return(ret); | 
| 11582 | 0 | } | 
| 11583 |  |  | 
| 11584 |  | /** | 
| 11585 |  |  * Load and parse an external subset. | 
| 11586 |  |  * | 
| 11587 |  |  * @deprecated Use #xmlCtxtParseDtd. | 
| 11588 |  |  * | 
| 11589 |  |  * @param sax  the SAX handler block | 
| 11590 |  |  * @param publicId  public identifier of the DTD (optional) | 
| 11591 |  |  * @param systemId  system identifier (URL) of the DTD | 
| 11592 |  |  * @returns the resulting xmlDtd or NULL in case of error. | 
| 11593 |  |  */ | 
| 11594 |  |  | 
| 11595 |  | xmlDtd * | 
| 11596 |  | xmlSAXParseDTD(xmlSAXHandler *sax, const xmlChar *publicId, | 
| 11597 | 0 |                const xmlChar *systemId) { | 
| 11598 | 0 |     xmlDtdPtr ret = NULL; | 
| 11599 | 0 |     xmlParserCtxtPtr ctxt; | 
| 11600 | 0 |     xmlParserInputPtr input = NULL; | 
| 11601 | 0 |     xmlChar* systemIdCanonic; | 
| 11602 |  | 
 | 
| 11603 | 0 |     if ((publicId == NULL) && (systemId == NULL)) return(NULL); | 
| 11604 |  |  | 
| 11605 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, NULL); | 
| 11606 | 0 |     if (ctxt == NULL) { | 
| 11607 | 0 |   return(NULL); | 
| 11608 | 0 |     } | 
| 11609 | 0 |     xmlCtxtSetOptions(ctxt, XML_PARSE_DTDLOAD); | 
| 11610 |  |  | 
| 11611 |  |     /* | 
| 11612 |  |      * Canonicalise the system ID | 
| 11613 |  |      */ | 
| 11614 | 0 |     systemIdCanonic = xmlCanonicPath(systemId); | 
| 11615 | 0 |     if ((systemId != NULL) && (systemIdCanonic == NULL)) { | 
| 11616 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 11617 | 0 |   return(NULL); | 
| 11618 | 0 |     } | 
| 11619 |  |  | 
| 11620 |  |     /* | 
| 11621 |  |      * Ask the Entity resolver to load the damn thing | 
| 11622 |  |      */ | 
| 11623 |  |  | 
| 11624 | 0 |     if ((ctxt->sax != NULL) && (ctxt->sax->resolveEntity != NULL)) | 
| 11625 | 0 |   input = ctxt->sax->resolveEntity(ctxt->userData, publicId, | 
| 11626 | 0 |                                    systemIdCanonic); | 
| 11627 | 0 |     if (input == NULL) { | 
| 11628 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 11629 | 0 |   if (systemIdCanonic != NULL) | 
| 11630 | 0 |       xmlFree(systemIdCanonic); | 
| 11631 | 0 |   return(NULL); | 
| 11632 | 0 |     } | 
| 11633 |  |  | 
| 11634 | 0 |     if (input->filename == NULL) | 
| 11635 | 0 |   input->filename = (char *) systemIdCanonic; | 
| 11636 | 0 |     else | 
| 11637 | 0 |   xmlFree(systemIdCanonic); | 
| 11638 |  | 
 | 
| 11639 | 0 |     ret = xmlCtxtParseDtd(ctxt, input, publicId, systemId); | 
| 11640 |  | 
 | 
| 11641 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 11642 | 0 |     return(ret); | 
| 11643 | 0 | } | 
| 11644 |  |  | 
| 11645 |  |  | 
| 11646 |  | /** | 
| 11647 |  |  * Load and parse an external subset. | 
| 11648 |  |  * | 
| 11649 |  |  * @param publicId  public identifier of the DTD (optional) | 
| 11650 |  |  * @param systemId  system identifier (URL) of the DTD | 
| 11651 |  |  * @returns the resulting xmlDtd or NULL in case of error. | 
| 11652 |  |  */ | 
| 11653 |  |  | 
| 11654 |  | xmlDtd * | 
| 11655 | 0 | xmlParseDTD(const xmlChar *publicId, const xmlChar *systemId) { | 
| 11656 | 0 |     return(xmlSAXParseDTD(NULL, publicId, systemId)); | 
| 11657 | 0 | } | 
| 11658 |  | #endif /* LIBXML_VALID_ENABLED */ | 
| 11659 |  |  | 
| 11660 |  | /************************************************************************ | 
| 11661 |  |  *                  * | 
| 11662 |  |  *    Front ends when parsing an Entity     * | 
| 11663 |  |  *                  * | 
| 11664 |  |  ************************************************************************/ | 
| 11665 |  |  | 
| 11666 |  | static xmlNodePtr | 
| 11667 |  | xmlCtxtParseContentInternal(xmlParserCtxtPtr ctxt, xmlParserInputPtr input, | 
| 11668 | 1.73k |                             int hasTextDecl, int buildTree) { | 
| 11669 | 1.73k |     xmlNodePtr root = NULL; | 
| 11670 | 1.73k |     xmlNodePtr list = NULL; | 
| 11671 | 1.73k |     xmlChar *rootName = BAD_CAST "#root"; | 
| 11672 | 1.73k |     int result; | 
| 11673 |  |  | 
| 11674 | 1.73k |     if (buildTree) { | 
| 11675 | 1.73k |         root = xmlNewDocNode(ctxt->myDoc, NULL, rootName, NULL); | 
| 11676 | 1.73k |         if (root == NULL) { | 
| 11677 | 0 |             xmlErrMemory(ctxt); | 
| 11678 | 0 |             goto error; | 
| 11679 | 0 |         } | 
| 11680 | 1.73k |     } | 
| 11681 |  |  | 
| 11682 | 1.73k |     if (xmlCtxtPushInput(ctxt, input) < 0) | 
| 11683 | 0 |         goto error; | 
| 11684 |  |  | 
| 11685 | 1.73k |     nameNsPush(ctxt, rootName, NULL, NULL, 0, 0); | 
| 11686 | 1.73k |     spacePush(ctxt, -1); | 
| 11687 |  |  | 
| 11688 | 1.73k |     if (buildTree) | 
| 11689 | 1.73k |         nodePush(ctxt, root); | 
| 11690 |  |  | 
| 11691 | 1.73k |     if (hasTextDecl) { | 
| 11692 | 0 |         xmlDetectEncoding(ctxt); | 
| 11693 |  |  | 
| 11694 |  |         /* | 
| 11695 |  |          * Parse a possible text declaration first | 
| 11696 |  |          */ | 
| 11697 | 0 |         if ((CMP5(CUR_PTR, '<', '?', 'x', 'm', 'l')) && | 
| 11698 | 0 |             (IS_BLANK_CH(NXT(5)))) { | 
| 11699 | 0 |             xmlParseTextDecl(ctxt); | 
| 11700 |  |             /* | 
| 11701 |  |              * An XML-1.0 document can't reference an entity not XML-1.0 | 
| 11702 |  |              */ | 
| 11703 | 0 |             if ((xmlStrEqual(ctxt->version, BAD_CAST "1.0")) && | 
| 11704 | 0 |                 (!xmlStrEqual(ctxt->input->version, BAD_CAST "1.0"))) { | 
| 11705 | 0 |                 xmlFatalErrMsg(ctxt, XML_ERR_VERSION_MISMATCH, | 
| 11706 | 0 |                                "Version mismatch between document and " | 
| 11707 | 0 |                                "entity\n"); | 
| 11708 | 0 |             } | 
| 11709 | 0 |         } | 
| 11710 | 0 |     } | 
| 11711 |  |  | 
| 11712 | 1.73k |     xmlParseContentInternal(ctxt); | 
| 11713 |  |  | 
| 11714 | 1.73k |     if (ctxt->input->cur < ctxt->input->end) | 
| 11715 | 73 |   xmlFatalErr(ctxt, XML_ERR_NOT_WELL_BALANCED, NULL); | 
| 11716 |  |  | 
| 11717 | 1.73k |     if ((ctxt->wellFormed) || | 
| 11718 | 1.46k |         ((ctxt->recovery) && (!xmlCtxtIsCatastrophicError(ctxt)))) { | 
| 11719 | 1.46k |         if (root != NULL) { | 
| 11720 | 1.46k |             xmlNodePtr cur; | 
| 11721 |  |  | 
| 11722 |  |             /* | 
| 11723 |  |              * Unlink newly created node list. | 
| 11724 |  |              */ | 
| 11725 | 1.46k |             list = root->children; | 
| 11726 | 1.46k |             root->children = NULL; | 
| 11727 | 1.46k |             root->last = NULL; | 
| 11728 | 3.23k |             for (cur = list; cur != NULL; cur = cur->next) | 
| 11729 | 1.77k |                 cur->parent = NULL; | 
| 11730 | 1.46k |         } | 
| 11731 | 1.46k |     } | 
| 11732 |  |  | 
| 11733 |  |     /* | 
| 11734 |  |      * Read the rest of the stream in case of errors. We want | 
| 11735 |  |      * to account for the whole entity size. | 
| 11736 |  |      */ | 
| 11737 | 1.73k |     do { | 
| 11738 | 1.73k |         ctxt->input->cur = ctxt->input->end; | 
| 11739 | 1.73k |         xmlParserShrink(ctxt); | 
| 11740 | 1.73k |         result = xmlParserGrow(ctxt); | 
| 11741 | 1.73k |     } while (result > 0); | 
| 11742 |  |  | 
| 11743 | 1.73k |     if (buildTree) | 
| 11744 | 1.73k |         nodePop(ctxt); | 
| 11745 |  |  | 
| 11746 | 1.73k |     namePop(ctxt); | 
| 11747 | 1.73k |     spacePop(ctxt); | 
| 11748 |  |  | 
| 11749 | 1.73k |     xmlCtxtPopInput(ctxt); | 
| 11750 |  |  | 
| 11751 | 1.73k | error: | 
| 11752 | 1.73k |     xmlFreeNode(root); | 
| 11753 |  |  | 
| 11754 | 1.73k |     return(list); | 
| 11755 | 1.73k | } | 
| 11756 |  |  | 
| 11757 |  | static void | 
| 11758 | 1.74k | xmlCtxtParseEntity(xmlParserCtxtPtr ctxt, xmlEntityPtr ent) { | 
| 11759 | 1.74k |     xmlParserInputPtr input; | 
| 11760 | 1.74k |     xmlNodePtr list; | 
| 11761 | 1.74k |     unsigned long consumed; | 
| 11762 | 1.74k |     int isExternal; | 
| 11763 | 1.74k |     int buildTree; | 
| 11764 | 1.74k |     int oldMinNsIndex; | 
| 11765 | 1.74k |     int oldNodelen, oldNodemem; | 
| 11766 |  |  | 
| 11767 | 1.74k |     isExternal = (ent->etype == XML_EXTERNAL_GENERAL_PARSED_ENTITY); | 
| 11768 | 1.74k |     buildTree = (ctxt->node != NULL); | 
| 11769 |  |  | 
| 11770 |  |     /* | 
| 11771 |  |      * Recursion check | 
| 11772 |  |      */ | 
| 11773 | 1.74k |     if (ent->flags & XML_ENT_EXPANDING) { | 
| 11774 | 12 |         xmlFatalErr(ctxt, XML_ERR_ENTITY_LOOP, NULL); | 
| 11775 | 12 |         goto error; | 
| 11776 | 12 |     } | 
| 11777 |  |  | 
| 11778 |  |     /* | 
| 11779 |  |      * Load entity | 
| 11780 |  |      */ | 
| 11781 | 1.73k |     input = xmlNewEntityInputStream(ctxt, ent); | 
| 11782 | 1.73k |     if (input == NULL) | 
| 11783 | 0 |         goto error; | 
| 11784 |  |  | 
| 11785 |  |     /* | 
| 11786 |  |      * When building a tree, we need to limit the scope of namespace | 
| 11787 |  |      * declarations, so that entities don't reference xmlNs structs | 
| 11788 |  |      * from the parent of a reference. | 
| 11789 |  |      */ | 
| 11790 | 1.73k |     oldMinNsIndex = ctxt->nsdb->minNsIndex; | 
| 11791 | 1.73k |     if (buildTree) | 
| 11792 | 1.73k |         ctxt->nsdb->minNsIndex = ctxt->nsNr; | 
| 11793 |  |  | 
| 11794 | 1.73k |     oldNodelen = ctxt->nodelen; | 
| 11795 | 1.73k |     oldNodemem = ctxt->nodemem; | 
| 11796 | 1.73k |     ctxt->nodelen = 0; | 
| 11797 | 1.73k |     ctxt->nodemem = 0; | 
| 11798 |  |  | 
| 11799 |  |     /* | 
| 11800 |  |      * Parse content | 
| 11801 |  |      * | 
| 11802 |  |      * This initiates a recursive call chain: | 
| 11803 |  |      * | 
| 11804 |  |      * - xmlCtxtParseContentInternal | 
| 11805 |  |      * - xmlParseContentInternal | 
| 11806 |  |      * - xmlParseReference | 
| 11807 |  |      * - xmlCtxtParseEntity | 
| 11808 |  |      * | 
| 11809 |  |      * The nesting depth is limited by the maximum number of inputs, | 
| 11810 |  |      * see xmlCtxtPushInput. | 
| 11811 |  |      * | 
| 11812 |  |      * It's possible to make this non-recursive (minNsIndex must be | 
| 11813 |  |      * stored in the input struct) at the expense of code readability. | 
| 11814 |  |      */ | 
| 11815 |  |  | 
| 11816 | 1.73k |     ent->flags |= XML_ENT_EXPANDING; | 
| 11817 |  |  | 
| 11818 | 1.73k |     list = xmlCtxtParseContentInternal(ctxt, input, isExternal, buildTree); | 
| 11819 |  |  | 
| 11820 | 1.73k |     ent->flags &= ~XML_ENT_EXPANDING; | 
| 11821 |  |  | 
| 11822 | 1.73k |     ctxt->nsdb->minNsIndex = oldMinNsIndex; | 
| 11823 | 1.73k |     ctxt->nodelen = oldNodelen; | 
| 11824 | 1.73k |     ctxt->nodemem = oldNodemem; | 
| 11825 |  |  | 
| 11826 |  |     /* | 
| 11827 |  |      * Entity size accounting | 
| 11828 |  |      */ | 
| 11829 | 1.73k |     consumed = input->consumed; | 
| 11830 | 1.73k |     xmlSaturatedAddSizeT(&consumed, input->end - input->base); | 
| 11831 |  |  | 
| 11832 | 1.73k |     if ((ent->flags & XML_ENT_CHECKED) == 0) | 
| 11833 | 1.00k |         xmlSaturatedAdd(&ent->expandedSize, consumed); | 
| 11834 |  |  | 
| 11835 | 1.73k |     if ((ent->flags & XML_ENT_PARSED) == 0) { | 
| 11836 | 1.00k |         if (isExternal) | 
| 11837 | 0 |             xmlSaturatedAdd(&ctxt->sizeentities, consumed); | 
| 11838 |  |  | 
| 11839 | 1.00k |         ent->children = list; | 
| 11840 |  |  | 
| 11841 | 2.77k |         while (list != NULL) { | 
| 11842 | 1.77k |             list->parent = (xmlNodePtr) ent; | 
| 11843 |  |  | 
| 11844 |  |             /* | 
| 11845 |  |              * Downstream code like the nginx xslt module can set | 
| 11846 |  |              * ctxt->myDoc->extSubset to a separate DTD, so the entity | 
| 11847 |  |              * might have a different or a NULL document. | 
| 11848 |  |              */ | 
| 11849 | 1.77k |             if (list->doc != ent->doc) | 
| 11850 | 0 |                 xmlSetTreeDoc(list, ent->doc); | 
| 11851 |  |  | 
| 11852 | 1.77k |             if (list->next == NULL) | 
| 11853 | 679 |                 ent->last = list; | 
| 11854 | 1.77k |             list = list->next; | 
| 11855 | 1.77k |         } | 
| 11856 | 1.00k |     } else { | 
| 11857 | 729 |         xmlFreeNodeList(list); | 
| 11858 | 729 |     } | 
| 11859 |  |  | 
| 11860 | 1.73k |     xmlFreeInputStream(input); | 
| 11861 |  |  | 
| 11862 | 1.74k | error: | 
| 11863 | 1.74k |     ent->flags |= XML_ENT_PARSED | XML_ENT_CHECKED; | 
| 11864 | 1.74k | } | 
| 11865 |  |  | 
| 11866 |  | /** | 
| 11867 |  |  * Parse an external general entity within an existing parsing context | 
| 11868 |  |  * An external general parsed entity is well-formed if it matches the | 
| 11869 |  |  * production labeled extParsedEnt. | 
| 11870 |  |  * | 
| 11871 |  |  *     [78] extParsedEnt ::= TextDecl? content | 
| 11872 |  |  * | 
| 11873 |  |  * @param ctxt  the existing parsing context | 
| 11874 |  |  * @param URL  the URL for the entity to load | 
| 11875 |  |  * @param ID  the System ID for the entity to load | 
| 11876 |  |  * @param listOut  the return value for the set of parsed nodes | 
| 11877 |  |  * @returns 0 if the entity is well formed, -1 in case of args problem and | 
| 11878 |  |  *    the parser error code otherwise | 
| 11879 |  |  */ | 
| 11880 |  |  | 
| 11881 |  | int | 
| 11882 |  | xmlParseCtxtExternalEntity(xmlParserCtxt *ctxt, const xmlChar *URL, | 
| 11883 | 0 |                            const xmlChar *ID, xmlNode **listOut) { | 
| 11884 | 0 |     xmlParserInputPtr input; | 
| 11885 | 0 |     xmlNodePtr list; | 
| 11886 |  | 
 | 
| 11887 | 0 |     if (listOut != NULL) | 
| 11888 | 0 |         *listOut = NULL; | 
| 11889 |  | 
 | 
| 11890 | 0 |     if (ctxt == NULL) | 
| 11891 | 0 |         return(XML_ERR_ARGUMENT); | 
| 11892 |  |  | 
| 11893 | 0 |     input = xmlLoadResource(ctxt, (char *) URL, (char *) ID, | 
| 11894 | 0 |                             XML_RESOURCE_GENERAL_ENTITY); | 
| 11895 | 0 |     if (input == NULL) | 
| 11896 | 0 |         return(ctxt->errNo); | 
| 11897 |  |  | 
| 11898 | 0 |     xmlCtxtInitializeLate(ctxt); | 
| 11899 |  | 
 | 
| 11900 | 0 |     list = xmlCtxtParseContentInternal(ctxt, input, /* hasTextDecl */ 1, 1); | 
| 11901 | 0 |     if (listOut != NULL) | 
| 11902 | 0 |         *listOut = list; | 
| 11903 | 0 |     else | 
| 11904 | 0 |         xmlFreeNodeList(list); | 
| 11905 |  | 
 | 
| 11906 | 0 |     xmlFreeInputStream(input); | 
| 11907 | 0 |     return(ctxt->errNo); | 
| 11908 | 0 | } | 
| 11909 |  |  | 
| 11910 |  | #ifdef LIBXML_SAX1_ENABLED | 
| 11911 |  | /** | 
| 11912 |  |  * Parse an external general entity | 
| 11913 |  |  * An external general parsed entity is well-formed if it matches the | 
| 11914 |  |  * production labeled extParsedEnt. | 
| 11915 |  |  * | 
| 11916 |  |  * This function uses deprecated global variables to set parser options | 
| 11917 |  |  * which default to XML_PARSE_NODICT. | 
| 11918 |  |  * | 
| 11919 |  |  * @deprecated Use #xmlParseCtxtExternalEntity. | 
| 11920 |  |  * | 
| 11921 |  |  *     [78] extParsedEnt ::= TextDecl? content | 
| 11922 |  |  * | 
| 11923 |  |  * @param doc  the document the chunk pertains to | 
| 11924 |  |  * @param sax  the SAX handler block (possibly NULL) | 
| 11925 |  |  * @param user_data  The user data returned on SAX callbacks (possibly NULL) | 
| 11926 |  |  * @param depth  Used for loop detection, use 0 | 
| 11927 |  |  * @param URL  the URL for the entity to load | 
| 11928 |  |  * @param ID  the System ID for the entity to load | 
| 11929 |  |  * @param list  the return value for the set of parsed nodes | 
| 11930 |  |  * @returns 0 if the entity is well formed, -1 in case of args problem and | 
| 11931 |  |  *    the parser error code otherwise | 
| 11932 |  |  */ | 
| 11933 |  |  | 
| 11934 |  | int | 
| 11935 |  | xmlParseExternalEntity(xmlDoc *doc, xmlSAXHandler *sax, void *user_data, | 
| 11936 | 0 |     int depth, const xmlChar *URL, const xmlChar *ID, xmlNode **list) { | 
| 11937 | 0 |     xmlParserCtxtPtr ctxt; | 
| 11938 | 0 |     int ret; | 
| 11939 |  | 
 | 
| 11940 | 0 |     if (list != NULL) | 
| 11941 | 0 |         *list = NULL; | 
| 11942 |  | 
 | 
| 11943 | 0 |     if (doc == NULL) | 
| 11944 | 0 |         return(XML_ERR_ARGUMENT); | 
| 11945 |  |  | 
| 11946 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, user_data); | 
| 11947 | 0 |     if (ctxt == NULL) | 
| 11948 | 0 |         return(XML_ERR_NO_MEMORY); | 
| 11949 |  |  | 
| 11950 | 0 |     ctxt->depth = depth; | 
| 11951 | 0 |     ctxt->myDoc = doc; | 
| 11952 | 0 |     ret = xmlParseCtxtExternalEntity(ctxt, URL, ID, list); | 
| 11953 |  | 
 | 
| 11954 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 11955 | 0 |     return(ret); | 
| 11956 | 0 | } | 
| 11957 |  |  | 
| 11958 |  | /** | 
| 11959 |  |  * Parse a well-balanced chunk of an XML document | 
| 11960 |  |  * called by the parser | 
| 11961 |  |  * The allowed sequence for the Well Balanced Chunk is the one defined by | 
| 11962 |  |  * the content production in the XML grammar: | 
| 11963 |  |  * | 
| 11964 |  |  *     [43] content ::= (element | CharData | Reference | CDSect | PI | | 
| 11965 |  |  *                       Comment)* | 
| 11966 |  |  * | 
| 11967 |  |  * This function uses deprecated global variables to set parser options | 
| 11968 |  |  * which default to XML_PARSE_NODICT. | 
| 11969 |  |  * | 
| 11970 |  |  * @param doc  the document the chunk pertains to (must not be NULL) | 
| 11971 |  |  * @param sax  the SAX handler block (possibly NULL) | 
| 11972 |  |  * @param user_data  The user data returned on SAX callbacks (possibly NULL) | 
| 11973 |  |  * @param depth  Used for loop detection, use 0 | 
| 11974 |  |  * @param string  the input string in UTF8 or ISO-Latin (zero terminated) | 
| 11975 |  |  * @param lst  the return value for the set of parsed nodes | 
| 11976 |  |  * @returns 0 if the chunk is well balanced, -1 in case of args problem and | 
| 11977 |  |  *    the parser error code otherwise | 
| 11978 |  |  */ | 
| 11979 |  |  | 
| 11980 |  | int | 
| 11981 |  | xmlParseBalancedChunkMemory(xmlDoc *doc, xmlSAXHandler *sax, | 
| 11982 | 0 |      void *user_data, int depth, const xmlChar *string, xmlNode **lst) { | 
| 11983 | 0 |     return xmlParseBalancedChunkMemoryRecover( doc, sax, user_data, | 
| 11984 | 0 |                                                 depth, string, lst, 0 ); | 
| 11985 | 0 | } | 
| 11986 |  | #endif /* LIBXML_SAX1_ENABLED */ | 
| 11987 |  |  | 
| 11988 |  | /** | 
| 11989 |  |  * Parse a well-balanced chunk of XML matching the 'content' production. | 
| 11990 |  |  * | 
| 11991 |  |  * Namespaces in scope of `node` and entities of `node`'s document are | 
| 11992 |  |  * recognized. When validating, the DTD of `node`'s document is used. | 
| 11993 |  |  * | 
| 11994 |  |  * Always consumes `input` even in error case. | 
| 11995 |  |  * | 
| 11996 |  |  * @since 2.14.0 | 
| 11997 |  |  * | 
| 11998 |  |  * @param ctxt  parser context | 
| 11999 |  |  * @param input  parser input | 
| 12000 |  |  * @param node  target node or document | 
| 12001 |  |  * @param hasTextDecl  whether to parse text declaration | 
| 12002 |  |  * @returns a node list or NULL in case of error. | 
| 12003 |  |  */ | 
| 12004 |  | xmlNode * | 
| 12005 |  | xmlCtxtParseContent(xmlParserCtxt *ctxt, xmlParserInput *input, | 
| 12006 | 0 |                     xmlNode *node, int hasTextDecl) { | 
| 12007 | 0 |     xmlDocPtr doc; | 
| 12008 | 0 |     xmlNodePtr cur, list = NULL; | 
| 12009 | 0 |     int nsnr = 0; | 
| 12010 | 0 |     xmlDictPtr oldDict; | 
| 12011 | 0 |     int oldOptions, oldDictNames, oldLoadSubset; | 
| 12012 |  | 
 | 
| 12013 | 0 |     if ((ctxt == NULL) || (input == NULL) || (node == NULL)) { | 
| 12014 | 0 |         xmlFatalErr(ctxt, XML_ERR_ARGUMENT, NULL); | 
| 12015 | 0 |         goto exit; | 
| 12016 | 0 |     } | 
| 12017 |  |  | 
| 12018 | 0 |     doc = node->doc; | 
| 12019 | 0 |     if (doc == NULL) { | 
| 12020 | 0 |         xmlFatalErr(ctxt, XML_ERR_ARGUMENT, NULL); | 
| 12021 | 0 |         goto exit; | 
| 12022 | 0 |     } | 
| 12023 |  |  | 
| 12024 | 0 |     switch (node->type) { | 
| 12025 | 0 |         case XML_ELEMENT_NODE: | 
| 12026 | 0 |         case XML_DOCUMENT_NODE: | 
| 12027 | 0 |         case XML_HTML_DOCUMENT_NODE: | 
| 12028 | 0 |             break; | 
| 12029 |  |  | 
| 12030 | 0 |         case XML_ATTRIBUTE_NODE: | 
| 12031 | 0 |         case XML_TEXT_NODE: | 
| 12032 | 0 |         case XML_CDATA_SECTION_NODE: | 
| 12033 | 0 |         case XML_ENTITY_REF_NODE: | 
| 12034 | 0 |         case XML_PI_NODE: | 
| 12035 | 0 |         case XML_COMMENT_NODE: | 
| 12036 | 0 |             for (cur = node->parent; cur != NULL; cur = node->parent) { | 
| 12037 | 0 |                 if ((cur->type == XML_ELEMENT_NODE) || | 
| 12038 | 0 |                     (cur->type == XML_DOCUMENT_NODE) || | 
| 12039 | 0 |                     (cur->type == XML_HTML_DOCUMENT_NODE)) { | 
| 12040 | 0 |                     node = cur; | 
| 12041 | 0 |                     break; | 
| 12042 | 0 |                 } | 
| 12043 | 0 |             } | 
| 12044 | 0 |             break; | 
| 12045 |  |  | 
| 12046 | 0 |         default: | 
| 12047 | 0 |             xmlFatalErr(ctxt, XML_ERR_ARGUMENT, NULL); | 
| 12048 | 0 |             goto exit; | 
| 12049 | 0 |     } | 
| 12050 |  |  | 
| 12051 | 0 |     xmlCtxtReset(ctxt); | 
| 12052 |  | 
 | 
| 12053 | 0 |     oldDict = ctxt->dict; | 
| 12054 | 0 |     oldOptions = ctxt->options; | 
| 12055 | 0 |     oldDictNames = ctxt->dictNames; | 
| 12056 | 0 |     oldLoadSubset = ctxt->loadsubset; | 
| 12057 |  |  | 
| 12058 |  |     /* | 
| 12059 |  |      * Use input doc's dict if present, else assure XML_PARSE_NODICT is set. | 
| 12060 |  |      */ | 
| 12061 | 0 |     if (doc->dict != NULL) { | 
| 12062 | 0 |         ctxt->dict = doc->dict; | 
| 12063 | 0 |     } else { | 
| 12064 | 0 |         ctxt->options |= XML_PARSE_NODICT; | 
| 12065 | 0 |         ctxt->dictNames = 0; | 
| 12066 | 0 |     } | 
| 12067 |  |  | 
| 12068 |  |     /* | 
| 12069 |  |      * Disable IDs | 
| 12070 |  |      */ | 
| 12071 | 0 |     ctxt->loadsubset |= XML_SKIP_IDS; | 
| 12072 | 0 |     ctxt->options |= XML_PARSE_SKIP_IDS; | 
| 12073 |  | 
 | 
| 12074 | 0 |     ctxt->myDoc = doc; | 
| 12075 |  | 
 | 
| 12076 | 0 | #ifdef LIBXML_HTML_ENABLED | 
| 12077 | 0 |     if (ctxt->html) { | 
| 12078 |  |         /* | 
| 12079 |  |          * When parsing in context, it makes no sense to add implied | 
| 12080 |  |          * elements like html/body/etc... | 
| 12081 |  |          */ | 
| 12082 | 0 |         ctxt->options |= HTML_PARSE_NOIMPLIED; | 
| 12083 |  | 
 | 
| 12084 | 0 |         list = htmlCtxtParseContentInternal(ctxt, input); | 
| 12085 | 0 |     } else | 
| 12086 | 0 | #endif | 
| 12087 | 0 |     { | 
| 12088 | 0 |         xmlCtxtInitializeLate(ctxt); | 
| 12089 |  |  | 
| 12090 |  |         /* | 
| 12091 |  |          * initialize the SAX2 namespaces stack | 
| 12092 |  |          */ | 
| 12093 | 0 |         cur = node; | 
| 12094 | 0 |         while ((cur != NULL) && (cur->type == XML_ELEMENT_NODE)) { | 
| 12095 | 0 |             xmlNsPtr ns = cur->nsDef; | 
| 12096 | 0 |             xmlHashedString hprefix, huri; | 
| 12097 |  | 
 | 
| 12098 | 0 |             while (ns != NULL) { | 
| 12099 | 0 |                 hprefix = xmlDictLookupHashed(ctxt->dict, ns->prefix, -1); | 
| 12100 | 0 |                 huri = xmlDictLookupHashed(ctxt->dict, ns->href, -1); | 
| 12101 | 0 |                 if (xmlParserNsPush(ctxt, &hprefix, &huri, ns, 1) > 0) | 
| 12102 | 0 |                     nsnr++; | 
| 12103 | 0 |                 ns = ns->next; | 
| 12104 | 0 |             } | 
| 12105 | 0 |             cur = cur->parent; | 
| 12106 | 0 |         } | 
| 12107 |  | 
 | 
| 12108 | 0 |         list = xmlCtxtParseContentInternal(ctxt, input, hasTextDecl, 1); | 
| 12109 |  | 
 | 
| 12110 | 0 |         if (nsnr > 0) | 
| 12111 | 0 |             xmlParserNsPop(ctxt, nsnr); | 
| 12112 | 0 |     } | 
| 12113 |  | 
 | 
| 12114 | 0 |     ctxt->dict = oldDict; | 
| 12115 | 0 |     ctxt->options = oldOptions; | 
| 12116 | 0 |     ctxt->dictNames = oldDictNames; | 
| 12117 | 0 |     ctxt->loadsubset = oldLoadSubset; | 
| 12118 | 0 |     ctxt->myDoc = NULL; | 
| 12119 | 0 |     ctxt->node = NULL; | 
| 12120 |  | 
 | 
| 12121 | 0 | exit: | 
| 12122 | 0 |     xmlFreeInputStream(input); | 
| 12123 | 0 |     return(list); | 
| 12124 | 0 | } | 
| 12125 |  |  | 
| 12126 |  | /** | 
| 12127 |  |  * Parse a well-balanced chunk of an XML document | 
| 12128 |  |  * within the context (DTD, namespaces, etc ...) of the given node. | 
| 12129 |  |  * | 
| 12130 |  |  * The allowed sequence for the data is a Well Balanced Chunk defined by | 
| 12131 |  |  * the content production in the XML grammar: | 
| 12132 |  |  * | 
| 12133 |  |  *     [43] content ::= (element | CharData | Reference | CDSect | PI | | 
| 12134 |  |  *                       Comment)* | 
| 12135 |  |  * | 
| 12136 |  |  * This function assumes the encoding of `node`'s document which is | 
| 12137 |  |  * typically not what you want. A better alternative is | 
| 12138 |  |  * #xmlCtxtParseContent. | 
| 12139 |  |  * | 
| 12140 |  |  * @param node  the context node | 
| 12141 |  |  * @param data  the input string | 
| 12142 |  |  * @param datalen  the input string length in bytes | 
| 12143 |  |  * @param options  a combination of xmlParserOption | 
| 12144 |  |  * @param listOut  the return value for the set of parsed nodes | 
| 12145 |  |  * @returns XML_ERR_OK if the chunk is well balanced, and the parser | 
| 12146 |  |  * error code otherwise | 
| 12147 |  |  */ | 
| 12148 |  | xmlParserErrors | 
| 12149 |  | xmlParseInNodeContext(xmlNode *node, const char *data, int datalen, | 
| 12150 | 0 |                       int options, xmlNode **listOut) { | 
| 12151 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12152 | 0 |     xmlParserInputPtr input; | 
| 12153 | 0 |     xmlDocPtr doc; | 
| 12154 | 0 |     xmlNodePtr list; | 
| 12155 | 0 |     xmlParserErrors ret; | 
| 12156 |  | 
 | 
| 12157 | 0 |     if (listOut == NULL) | 
| 12158 | 0 |         return(XML_ERR_INTERNAL_ERROR); | 
| 12159 | 0 |     *listOut = NULL; | 
| 12160 |  | 
 | 
| 12161 | 0 |     if ((node == NULL) || (data == NULL) || (datalen < 0)) | 
| 12162 | 0 |         return(XML_ERR_INTERNAL_ERROR); | 
| 12163 |  |  | 
| 12164 | 0 |     doc = node->doc; | 
| 12165 | 0 |     if (doc == NULL) | 
| 12166 | 0 |         return(XML_ERR_INTERNAL_ERROR); | 
| 12167 |  |  | 
| 12168 | 0 | #ifdef LIBXML_HTML_ENABLED | 
| 12169 | 0 |     if (doc->type == XML_HTML_DOCUMENT_NODE) { | 
| 12170 | 0 |         ctxt = htmlNewParserCtxt(); | 
| 12171 | 0 |     } | 
| 12172 | 0 |     else | 
| 12173 | 0 | #endif | 
| 12174 | 0 |         ctxt = xmlNewParserCtxt(); | 
| 12175 |  | 
 | 
| 12176 | 0 |     if (ctxt == NULL) | 
| 12177 | 0 |         return(XML_ERR_NO_MEMORY); | 
| 12178 |  |  | 
| 12179 | 0 |     input = xmlCtxtNewInputFromMemory(ctxt, NULL, data, datalen, | 
| 12180 | 0 |                                       (const char *) doc->encoding, | 
| 12181 | 0 |                                       XML_INPUT_BUF_STATIC); | 
| 12182 | 0 |     if (input == NULL) { | 
| 12183 | 0 |         xmlFreeParserCtxt(ctxt); | 
| 12184 | 0 |         return(XML_ERR_NO_MEMORY); | 
| 12185 | 0 |     } | 
| 12186 |  |  | 
| 12187 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 12188 |  | 
 | 
| 12189 | 0 |     list = xmlCtxtParseContent(ctxt, input, node, /* hasTextDecl */ 0); | 
| 12190 |  | 
 | 
| 12191 | 0 |     if (list == NULL) { | 
| 12192 | 0 |         ret = ctxt->errNo; | 
| 12193 | 0 |         if (ret == XML_ERR_ARGUMENT) | 
| 12194 | 0 |             ret = XML_ERR_INTERNAL_ERROR; | 
| 12195 | 0 |     } else { | 
| 12196 | 0 |         ret = XML_ERR_OK; | 
| 12197 | 0 |         *listOut = list; | 
| 12198 | 0 |     } | 
| 12199 |  | 
 | 
| 12200 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12201 |  | 
 | 
| 12202 | 0 |     return(ret); | 
| 12203 | 0 | } | 
| 12204 |  |  | 
| 12205 |  | #ifdef LIBXML_SAX1_ENABLED | 
| 12206 |  | /** | 
| 12207 |  |  * Parse a well-balanced chunk of an XML document | 
| 12208 |  |  * | 
| 12209 |  |  * The allowed sequence for the Well Balanced Chunk is the one defined by | 
| 12210 |  |  * the content production in the XML grammar: | 
| 12211 |  |  * | 
| 12212 |  |  *     [43] content ::= (element | CharData | Reference | CDSect | PI | | 
| 12213 |  |  *                       Comment)* | 
| 12214 |  |  * | 
| 12215 |  |  * In case recover is set to 1, the nodelist will not be empty even if | 
| 12216 |  |  * the parsed chunk is not well balanced, assuming the parsing succeeded to | 
| 12217 |  |  * some extent. | 
| 12218 |  |  * | 
| 12219 |  |  * This function uses deprecated global variables to set parser options | 
| 12220 |  |  * which default to XML_PARSE_NODICT. | 
| 12221 |  |  * | 
| 12222 |  |  * @param doc  the document the chunk pertains to (must not be NULL) | 
| 12223 |  |  * @param sax  the SAX handler block (possibly NULL) | 
| 12224 |  |  * @param user_data  The user data returned on SAX callbacks (possibly NULL) | 
| 12225 |  |  * @param depth  Used for loop detection, use 0 | 
| 12226 |  |  * @param string  the input string in UTF8 or ISO-Latin (zero terminated) | 
| 12227 |  |  * @param listOut  the return value for the set of parsed nodes | 
| 12228 |  |  * @param recover  return nodes even if the data is broken (use 0) | 
| 12229 |  |  * @returns 0 if the chunk is well balanced, or thehe parser error code | 
| 12230 |  |  * otherwise. | 
| 12231 |  |  */ | 
| 12232 |  | int | 
| 12233 |  | xmlParseBalancedChunkMemoryRecover(xmlDoc *doc, xmlSAXHandler *sax, | 
| 12234 |  |      void *user_data, int depth, const xmlChar *string, xmlNode **listOut, | 
| 12235 | 0 |      int recover) { | 
| 12236 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12237 | 0 |     xmlParserInputPtr input; | 
| 12238 | 0 |     xmlNodePtr list; | 
| 12239 | 0 |     int ret; | 
| 12240 |  | 
 | 
| 12241 | 0 |     if (listOut != NULL) | 
| 12242 | 0 |         *listOut = NULL; | 
| 12243 |  | 
 | 
| 12244 | 0 |     if (string == NULL) | 
| 12245 | 0 |         return(XML_ERR_ARGUMENT); | 
| 12246 |  |  | 
| 12247 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, user_data); | 
| 12248 | 0 |     if (ctxt == NULL) | 
| 12249 | 0 |         return(XML_ERR_NO_MEMORY); | 
| 12250 |  |  | 
| 12251 | 0 |     xmlCtxtInitializeLate(ctxt); | 
| 12252 |  | 
 | 
| 12253 | 0 |     ctxt->depth = depth; | 
| 12254 | 0 |     ctxt->myDoc = doc; | 
| 12255 | 0 |     if (recover) { | 
| 12256 | 0 |         ctxt->options |= XML_PARSE_RECOVER; | 
| 12257 | 0 |         ctxt->recovery = 1; | 
| 12258 | 0 |     } | 
| 12259 |  | 
 | 
| 12260 | 0 |     input = xmlNewStringInputStream(ctxt, string); | 
| 12261 | 0 |     if (input == NULL) { | 
| 12262 | 0 |         ret = ctxt->errNo; | 
| 12263 | 0 |         goto error; | 
| 12264 | 0 |     } | 
| 12265 |  |  | 
| 12266 | 0 |     list = xmlCtxtParseContentInternal(ctxt, input, /* hasTextDecl */ 0, 1); | 
| 12267 | 0 |     if (listOut != NULL) | 
| 12268 | 0 |         *listOut = list; | 
| 12269 | 0 |     else | 
| 12270 | 0 |         xmlFreeNodeList(list); | 
| 12271 |  | 
 | 
| 12272 | 0 |     if (!ctxt->wellFormed) | 
| 12273 | 0 |         ret = ctxt->errNo; | 
| 12274 | 0 |     else | 
| 12275 | 0 |         ret = XML_ERR_OK; | 
| 12276 |  | 
 | 
| 12277 | 0 | error: | 
| 12278 | 0 |     xmlFreeInputStream(input); | 
| 12279 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12280 | 0 |     return(ret); | 
| 12281 | 0 | } | 
| 12282 |  |  | 
| 12283 |  | /** | 
| 12284 |  |  * Parse an XML external entity out of context and build a tree. | 
| 12285 |  |  * It use the given SAX function block to handle the parsing callback. | 
| 12286 |  |  * If sax is NULL, fallback to the default DOM tree building routines. | 
| 12287 |  |  * | 
| 12288 |  |  * @deprecated Don't use. | 
| 12289 |  |  * | 
| 12290 |  |  *     [78] extParsedEnt ::= TextDecl? content | 
| 12291 |  |  * | 
| 12292 |  |  * This correspond to a "Well Balanced" chunk | 
| 12293 |  |  * | 
| 12294 |  |  * This function uses deprecated global variables to set parser options | 
| 12295 |  |  * which default to XML_PARSE_NODICT. | 
| 12296 |  |  * | 
| 12297 |  |  * @param sax  the SAX handler block | 
| 12298 |  |  * @param filename  the filename | 
| 12299 |  |  * @returns the resulting document tree | 
| 12300 |  |  */ | 
| 12301 |  |  | 
| 12302 |  | xmlDoc * | 
| 12303 | 0 | xmlSAXParseEntity(xmlSAXHandler *sax, const char *filename) { | 
| 12304 | 0 |     xmlDocPtr ret; | 
| 12305 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12306 |  | 
 | 
| 12307 | 0 |     ctxt = xmlCreateFileParserCtxt(filename); | 
| 12308 | 0 |     if (ctxt == NULL) { | 
| 12309 | 0 |   return(NULL); | 
| 12310 | 0 |     } | 
| 12311 | 0 |     if (sax != NULL) { | 
| 12312 | 0 |         if (sax->initialized == XML_SAX2_MAGIC) { | 
| 12313 | 0 |             *ctxt->sax = *sax; | 
| 12314 | 0 |         } else { | 
| 12315 | 0 |             memset(ctxt->sax, 0, sizeof(*ctxt->sax)); | 
| 12316 | 0 |             memcpy(ctxt->sax, sax, sizeof(xmlSAXHandlerV1)); | 
| 12317 | 0 |         } | 
| 12318 | 0 |         ctxt->userData = NULL; | 
| 12319 | 0 |     } | 
| 12320 |  | 
 | 
| 12321 | 0 |     xmlParseExtParsedEnt(ctxt); | 
| 12322 |  | 
 | 
| 12323 | 0 |     if (ctxt->wellFormed) { | 
| 12324 | 0 |   ret = ctxt->myDoc; | 
| 12325 | 0 |     } else { | 
| 12326 | 0 |         ret = NULL; | 
| 12327 | 0 |         xmlFreeDoc(ctxt->myDoc); | 
| 12328 | 0 |     } | 
| 12329 |  | 
 | 
| 12330 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12331 |  | 
 | 
| 12332 | 0 |     return(ret); | 
| 12333 | 0 | } | 
| 12334 |  |  | 
| 12335 |  | /** | 
| 12336 |  |  * Parse an XML external entity out of context and build a tree. | 
| 12337 |  |  * | 
| 12338 |  |  *     [78] extParsedEnt ::= TextDecl? content | 
| 12339 |  |  * | 
| 12340 |  |  * This correspond to a "Well Balanced" chunk | 
| 12341 |  |  * | 
| 12342 |  |  * This function uses deprecated global variables to set parser options | 
| 12343 |  |  * which default to XML_PARSE_NODICT. | 
| 12344 |  |  * | 
| 12345 |  |  * @deprecated Don't use. | 
| 12346 |  |  * | 
| 12347 |  |  * @param filename  the filename | 
| 12348 |  |  * @returns the resulting document tree | 
| 12349 |  |  */ | 
| 12350 |  |  | 
| 12351 |  | xmlDoc * | 
| 12352 | 0 | xmlParseEntity(const char *filename) { | 
| 12353 | 0 |     return(xmlSAXParseEntity(NULL, filename)); | 
| 12354 | 0 | } | 
| 12355 |  | #endif /* LIBXML_SAX1_ENABLED */ | 
| 12356 |  |  | 
| 12357 |  | /** | 
| 12358 |  |  * Create a parser context for an external entity | 
| 12359 |  |  * Automatic support for ZLIB/Compress compressed document is provided | 
| 12360 |  |  * by default if found at compile-time. | 
| 12361 |  |  * | 
| 12362 |  |  * @deprecated Don't use. | 
| 12363 |  |  * | 
| 12364 |  |  * @param URL  the entity URL | 
| 12365 |  |  * @param ID  the entity PUBLIC ID | 
| 12366 |  |  * @param base  a possible base for the target URI | 
| 12367 |  |  * @returns the new parser context or NULL | 
| 12368 |  |  */ | 
| 12369 |  | xmlParserCtxt * | 
| 12370 |  | xmlCreateEntityParserCtxt(const xmlChar *URL, const xmlChar *ID, | 
| 12371 | 0 |                     const xmlChar *base) { | 
| 12372 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12373 | 0 |     xmlParserInputPtr input; | 
| 12374 | 0 |     xmlChar *uri = NULL; | 
| 12375 |  | 
 | 
| 12376 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 12377 | 0 |     if (ctxt == NULL) | 
| 12378 | 0 |   return(NULL); | 
| 12379 |  |  | 
| 12380 | 0 |     if (base != NULL) { | 
| 12381 | 0 |         if (xmlBuildURISafe(URL, base, &uri) < 0) | 
| 12382 | 0 |             goto error; | 
| 12383 | 0 |         if (uri != NULL) | 
| 12384 | 0 |             URL = uri; | 
| 12385 | 0 |     } | 
| 12386 |  |  | 
| 12387 | 0 |     input = xmlLoadResource(ctxt, (char *) URL, (char *) ID, | 
| 12388 | 0 |                             XML_RESOURCE_UNKNOWN); | 
| 12389 | 0 |     if (input == NULL) | 
| 12390 | 0 |         goto error; | 
| 12391 |  |  | 
| 12392 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 12393 | 0 |         xmlFreeInputStream(input); | 
| 12394 | 0 |         goto error; | 
| 12395 | 0 |     } | 
| 12396 |  |  | 
| 12397 | 0 |     xmlFree(uri); | 
| 12398 | 0 |     return(ctxt); | 
| 12399 |  |  | 
| 12400 | 0 | error: | 
| 12401 | 0 |     xmlFree(uri); | 
| 12402 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12403 | 0 |     return(NULL); | 
| 12404 | 0 | } | 
| 12405 |  |  | 
| 12406 |  | /************************************************************************ | 
| 12407 |  |  *                  * | 
| 12408 |  |  *    Front ends when parsing from a file     * | 
| 12409 |  |  *                  * | 
| 12410 |  |  ************************************************************************/ | 
| 12411 |  |  | 
| 12412 |  | /** | 
| 12413 |  |  * Create a parser context for a file or URL content. | 
| 12414 |  |  * Automatic support for ZLIB/Compress compressed document is provided | 
| 12415 |  |  * by default if found at compile-time and for file accesses | 
| 12416 |  |  * | 
| 12417 |  |  * @deprecated Use #xmlNewParserCtxt and #xmlCtxtReadFile. | 
| 12418 |  |  * | 
| 12419 |  |  * @param filename  the filename or URL | 
| 12420 |  |  * @param options  a combination of xmlParserOption | 
| 12421 |  |  * @returns the new parser context or NULL | 
| 12422 |  |  */ | 
| 12423 |  | xmlParserCtxt * | 
| 12424 |  | xmlCreateURLParserCtxt(const char *filename, int options) | 
| 12425 | 0 | { | 
| 12426 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12427 | 0 |     xmlParserInputPtr input; | 
| 12428 |  | 
 | 
| 12429 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 12430 | 0 |     if (ctxt == NULL) | 
| 12431 | 0 |   return(NULL); | 
| 12432 |  |  | 
| 12433 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 12434 |  | 
 | 
| 12435 | 0 |     input = xmlLoadResource(ctxt, filename, NULL, XML_RESOURCE_MAIN_DOCUMENT); | 
| 12436 | 0 |     if (input == NULL) { | 
| 12437 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 12438 | 0 |   return(NULL); | 
| 12439 | 0 |     } | 
| 12440 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 12441 | 0 |         xmlFreeInputStream(input); | 
| 12442 | 0 |         xmlFreeParserCtxt(ctxt); | 
| 12443 | 0 |         return(NULL); | 
| 12444 | 0 |     } | 
| 12445 |  |  | 
| 12446 | 0 |     return(ctxt); | 
| 12447 | 0 | } | 
| 12448 |  |  | 
| 12449 |  | /** | 
| 12450 |  |  * Create a parser context for a file content. | 
| 12451 |  |  * Automatic support for ZLIB/Compress compressed document is provided | 
| 12452 |  |  * by default if found at compile-time. | 
| 12453 |  |  * | 
| 12454 |  |  * @deprecated Use #xmlNewParserCtxt and #xmlCtxtReadFile. | 
| 12455 |  |  * | 
| 12456 |  |  * @param filename  the filename | 
| 12457 |  |  * @returns the new parser context or NULL | 
| 12458 |  |  */ | 
| 12459 |  | xmlParserCtxt * | 
| 12460 |  | xmlCreateFileParserCtxt(const char *filename) | 
| 12461 | 0 | { | 
| 12462 | 0 |     return(xmlCreateURLParserCtxt(filename, 0)); | 
| 12463 | 0 | } | 
| 12464 |  |  | 
| 12465 |  | #ifdef LIBXML_SAX1_ENABLED | 
| 12466 |  | /** | 
| 12467 |  |  * Parse an XML file and build a tree. Automatic support for ZLIB/Compress | 
| 12468 |  |  * compressed document is provided by default if found at compile-time. | 
| 12469 |  |  * It use the given SAX function block to handle the parsing callback. | 
| 12470 |  |  * If sax is NULL, fallback to the default DOM tree building routines. | 
| 12471 |  |  * | 
| 12472 |  |  * This function uses deprecated global variables to set parser options | 
| 12473 |  |  * which default to XML_PARSE_NODICT. | 
| 12474 |  |  * | 
| 12475 |  |  * @deprecated Use #xmlNewSAXParserCtxt and #xmlCtxtReadFile. | 
| 12476 |  |  * | 
| 12477 |  |  * User data (void *) is stored within the parser context in the | 
| 12478 |  |  * context's _private member, so it is available nearly everywhere in libxml | 
| 12479 |  |  * | 
| 12480 |  |  * @param sax  the SAX handler block | 
| 12481 |  |  * @param filename  the filename | 
| 12482 |  |  * @param recovery  work in recovery mode, i.e. tries to read no Well Formed | 
| 12483 |  |  *             documents | 
| 12484 |  |  * @param data  the userdata | 
| 12485 |  |  * @returns the resulting document tree | 
| 12486 |  |  */ | 
| 12487 |  |  | 
| 12488 |  | xmlDoc * | 
| 12489 |  | xmlSAXParseFileWithData(xmlSAXHandler *sax, const char *filename, | 
| 12490 | 0 |                         int recovery, void *data) { | 
| 12491 | 0 |     xmlDocPtr ret = NULL; | 
| 12492 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12493 | 0 |     xmlParserInputPtr input; | 
| 12494 |  | 
 | 
| 12495 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, NULL); | 
| 12496 | 0 |     if (ctxt == NULL) | 
| 12497 | 0 |   return(NULL); | 
| 12498 |  |  | 
| 12499 | 0 |     if (data != NULL) | 
| 12500 | 0 |   ctxt->_private = data; | 
| 12501 |  | 
 | 
| 12502 | 0 |     if (recovery) { | 
| 12503 | 0 |         ctxt->options |= XML_PARSE_RECOVER; | 
| 12504 | 0 |         ctxt->recovery = 1; | 
| 12505 | 0 |     } | 
| 12506 |  | 
 | 
| 12507 | 0 |     if ((filename != NULL) && (filename[0] == '-') && (filename[1] == 0)) | 
| 12508 | 0 |         input = xmlCtxtNewInputFromFd(ctxt, filename, STDIN_FILENO, NULL, 0); | 
| 12509 | 0 |     else | 
| 12510 | 0 |         input = xmlCtxtNewInputFromUrl(ctxt, filename, NULL, NULL, 0); | 
| 12511 |  | 
 | 
| 12512 | 0 |     if (input != NULL) | 
| 12513 | 0 |         ret = xmlCtxtParseDocument(ctxt, input); | 
| 12514 |  | 
 | 
| 12515 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12516 | 0 |     return(ret); | 
| 12517 | 0 | } | 
| 12518 |  |  | 
| 12519 |  | /** | 
| 12520 |  |  * Parse an XML file and build a tree. Automatic support for ZLIB/Compress | 
| 12521 |  |  * compressed document is provided by default if found at compile-time. | 
| 12522 |  |  * It use the given SAX function block to handle the parsing callback. | 
| 12523 |  |  * If sax is NULL, fallback to the default DOM tree building routines. | 
| 12524 |  |  * | 
| 12525 |  |  * This function uses deprecated global variables to set parser options | 
| 12526 |  |  * which default to XML_PARSE_NODICT. | 
| 12527 |  |  * | 
| 12528 |  |  * @deprecated Use #xmlNewSAXParserCtxt and #xmlCtxtReadFile. | 
| 12529 |  |  * | 
| 12530 |  |  * @param sax  the SAX handler block | 
| 12531 |  |  * @param filename  the filename | 
| 12532 |  |  * @param recovery  work in recovery mode, i.e. tries to read no Well Formed | 
| 12533 |  |  *             documents | 
| 12534 |  |  * @returns the resulting document tree | 
| 12535 |  |  */ | 
| 12536 |  |  | 
| 12537 |  | xmlDoc * | 
| 12538 |  | xmlSAXParseFile(xmlSAXHandler *sax, const char *filename, | 
| 12539 | 0 |                           int recovery) { | 
| 12540 | 0 |     return(xmlSAXParseFileWithData(sax,filename,recovery,NULL)); | 
| 12541 | 0 | } | 
| 12542 |  |  | 
| 12543 |  | /** | 
| 12544 |  |  * Parse an XML in-memory document and build a tree. | 
| 12545 |  |  * In the case the document is not Well Formed, a attempt to build a | 
| 12546 |  |  * tree is tried anyway | 
| 12547 |  |  * | 
| 12548 |  |  * This function uses deprecated global variables to set parser options | 
| 12549 |  |  * which default to XML_PARSE_NODICT | XML_PARSE_RECOVER. | 
| 12550 |  |  * | 
| 12551 |  |  * @deprecated Use #xmlReadDoc with XML_PARSE_RECOVER. | 
| 12552 |  |  * | 
| 12553 |  |  * @param cur  a pointer to an array of xmlChar | 
| 12554 |  |  * @returns the resulting document tree or NULL in case of failure | 
| 12555 |  |  */ | 
| 12556 |  |  | 
| 12557 |  | xmlDoc * | 
| 12558 | 0 | xmlRecoverDoc(const xmlChar *cur) { | 
| 12559 | 0 |     return(xmlSAXParseDoc(NULL, cur, 1)); | 
| 12560 | 0 | } | 
| 12561 |  |  | 
| 12562 |  | /** | 
| 12563 |  |  * Parse an XML file and build a tree. Automatic support for ZLIB/Compress | 
| 12564 |  |  * compressed document is provided by default if found at compile-time. | 
| 12565 |  |  * | 
| 12566 |  |  * This function uses deprecated global variables to set parser options | 
| 12567 |  |  * which default to XML_PARSE_NODICT. | 
| 12568 |  |  * | 
| 12569 |  |  * @deprecated Use #xmlReadFile. | 
| 12570 |  |  * | 
| 12571 |  |  * @param filename  the filename | 
| 12572 |  |  * @returns the resulting document tree if the file was wellformed, | 
| 12573 |  |  * NULL otherwise. | 
| 12574 |  |  */ | 
| 12575 |  |  | 
| 12576 |  | xmlDoc * | 
| 12577 | 0 | xmlParseFile(const char *filename) { | 
| 12578 | 0 |     return(xmlSAXParseFile(NULL, filename, 0)); | 
| 12579 | 0 | } | 
| 12580 |  |  | 
| 12581 |  | /** | 
| 12582 |  |  * Parse an XML file and build a tree. Automatic support for ZLIB/Compress | 
| 12583 |  |  * compressed document is provided by default if found at compile-time. | 
| 12584 |  |  * In the case the document is not Well Formed, it attempts to build | 
| 12585 |  |  * a tree anyway | 
| 12586 |  |  * | 
| 12587 |  |  * This function uses deprecated global variables to set parser options | 
| 12588 |  |  * which default to XML_PARSE_NODICT | XML_PARSE_RECOVER. | 
| 12589 |  |  * | 
| 12590 |  |  * @deprecated Use #xmlReadFile with XML_PARSE_RECOVER. | 
| 12591 |  |  * | 
| 12592 |  |  * @param filename  the filename | 
| 12593 |  |  * @returns the resulting document tree or NULL in case of failure | 
| 12594 |  |  */ | 
| 12595 |  |  | 
| 12596 |  | xmlDoc * | 
| 12597 | 0 | xmlRecoverFile(const char *filename) { | 
| 12598 | 0 |     return(xmlSAXParseFile(NULL, filename, 1)); | 
| 12599 | 0 | } | 
| 12600 |  |  | 
| 12601 |  |  | 
| 12602 |  | /** | 
| 12603 |  |  * Setup the parser context to parse a new buffer; Clears any prior | 
| 12604 |  |  * contents from the parser context. The buffer parameter must not be | 
| 12605 |  |  * NULL, but the filename parameter can be | 
| 12606 |  |  * | 
| 12607 |  |  * @deprecated Don't use. | 
| 12608 |  |  * | 
| 12609 |  |  * @param ctxt  an XML parser context | 
| 12610 |  |  * @param buffer  a xmlChar * buffer | 
| 12611 |  |  * @param filename  a file name | 
| 12612 |  |  */ | 
| 12613 |  | void | 
| 12614 |  | xmlSetupParserForBuffer(xmlParserCtxt *ctxt, const xmlChar* buffer, | 
| 12615 |  |                              const char* filename) | 
| 12616 | 0 | { | 
| 12617 | 0 |     xmlParserInputPtr input; | 
| 12618 |  | 
 | 
| 12619 | 0 |     if ((ctxt == NULL) || (buffer == NULL)) | 
| 12620 | 0 |         return; | 
| 12621 |  |  | 
| 12622 | 0 |     xmlCtxtReset(ctxt); | 
| 12623 |  | 
 | 
| 12624 | 0 |     input = xmlCtxtNewInputFromString(ctxt, filename, (const char *) buffer, | 
| 12625 | 0 |                                       NULL, 0); | 
| 12626 | 0 |     if (input == NULL) | 
| 12627 | 0 |         return; | 
| 12628 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) | 
| 12629 | 0 |         xmlFreeInputStream(input); | 
| 12630 | 0 | } | 
| 12631 |  |  | 
| 12632 |  | /** | 
| 12633 |  |  * Parse an XML file and call the given SAX handler routines. | 
| 12634 |  |  * Automatic support for ZLIB/Compress compressed document is provided | 
| 12635 |  |  * | 
| 12636 |  |  * This function uses deprecated global variables to set parser options | 
| 12637 |  |  * which default to XML_PARSE_NODICT. | 
| 12638 |  |  * | 
| 12639 |  |  * @deprecated Use #xmlNewSAXParserCtxt and #xmlCtxtReadFile. | 
| 12640 |  |  * | 
| 12641 |  |  * @param sax  a SAX handler | 
| 12642 |  |  * @param user_data  The user data returned on SAX callbacks | 
| 12643 |  |  * @param filename  a file name | 
| 12644 |  |  * @returns 0 in case of success or a error number otherwise | 
| 12645 |  |  */ | 
| 12646 |  | int | 
| 12647 |  | xmlSAXUserParseFile(xmlSAXHandler *sax, void *user_data, | 
| 12648 | 0 |                     const char *filename) { | 
| 12649 | 0 |     int ret = 0; | 
| 12650 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12651 |  | 
 | 
| 12652 | 0 |     ctxt = xmlCreateFileParserCtxt(filename); | 
| 12653 | 0 |     if (ctxt == NULL) return -1; | 
| 12654 | 0 |     if (sax != NULL) { | 
| 12655 | 0 |         if (sax->initialized == XML_SAX2_MAGIC) { | 
| 12656 | 0 |             *ctxt->sax = *sax; | 
| 12657 | 0 |         } else { | 
| 12658 | 0 |             memset(ctxt->sax, 0, sizeof(*ctxt->sax)); | 
| 12659 | 0 |             memcpy(ctxt->sax, sax, sizeof(xmlSAXHandlerV1)); | 
| 12660 | 0 |         } | 
| 12661 | 0 |   ctxt->userData = user_data; | 
| 12662 | 0 |     } | 
| 12663 |  | 
 | 
| 12664 | 0 |     xmlParseDocument(ctxt); | 
| 12665 |  | 
 | 
| 12666 | 0 |     if (ctxt->wellFormed) | 
| 12667 | 0 |   ret = 0; | 
| 12668 | 0 |     else { | 
| 12669 | 0 |         if (ctxt->errNo != 0) | 
| 12670 | 0 |       ret = ctxt->errNo; | 
| 12671 | 0 |   else | 
| 12672 | 0 |       ret = -1; | 
| 12673 | 0 |     } | 
| 12674 | 0 |     if (ctxt->myDoc != NULL) { | 
| 12675 | 0 |         xmlFreeDoc(ctxt->myDoc); | 
| 12676 | 0 |   ctxt->myDoc = NULL; | 
| 12677 | 0 |     } | 
| 12678 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12679 |  | 
 | 
| 12680 | 0 |     return ret; | 
| 12681 | 0 | } | 
| 12682 |  | #endif /* LIBXML_SAX1_ENABLED */ | 
| 12683 |  |  | 
| 12684 |  | /************************************************************************ | 
| 12685 |  |  *                  * | 
| 12686 |  |  *    Front ends when parsing from memory     * | 
| 12687 |  |  *                  * | 
| 12688 |  |  ************************************************************************/ | 
| 12689 |  |  | 
| 12690 |  | /** | 
| 12691 |  |  * Create a parser context for an XML in-memory document. The input buffer | 
| 12692 |  |  * must not contain a terminating null byte. | 
| 12693 |  |  * | 
| 12694 |  |  * @param buffer  a pointer to a char array | 
| 12695 |  |  * @param size  the size of the array | 
| 12696 |  |  * @returns the new parser context or NULL | 
| 12697 |  |  */ | 
| 12698 |  | xmlParserCtxt * | 
| 12699 | 17.4k | xmlCreateMemoryParserCtxt(const char *buffer, int size) { | 
| 12700 | 17.4k |     xmlParserCtxtPtr ctxt; | 
| 12701 | 17.4k |     xmlParserInputPtr input; | 
| 12702 |  |  | 
| 12703 | 17.4k |     if (size < 0) | 
| 12704 | 0 |   return(NULL); | 
| 12705 |  |  | 
| 12706 | 17.4k |     ctxt = xmlNewParserCtxt(); | 
| 12707 | 17.4k |     if (ctxt == NULL) | 
| 12708 | 0 |   return(NULL); | 
| 12709 |  |  | 
| 12710 | 17.4k |     input = xmlCtxtNewInputFromMemory(ctxt, NULL, buffer, size, NULL, 0); | 
| 12711 | 17.4k |     if (input == NULL) { | 
| 12712 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 12713 | 0 |   return(NULL); | 
| 12714 | 0 |     } | 
| 12715 | 17.4k |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 12716 | 0 |         xmlFreeInputStream(input); | 
| 12717 | 0 |         xmlFreeParserCtxt(ctxt); | 
| 12718 | 0 |         return(NULL); | 
| 12719 | 0 |     } | 
| 12720 |  |  | 
| 12721 | 17.4k |     return(ctxt); | 
| 12722 | 17.4k | } | 
| 12723 |  |  | 
| 12724 |  | #ifdef LIBXML_SAX1_ENABLED | 
| 12725 |  | /** | 
| 12726 |  |  * Parse an XML in-memory block and use the given SAX function block | 
| 12727 |  |  * to handle the parsing callback. If sax is NULL, fallback to the default | 
| 12728 |  |  * DOM tree building routines. | 
| 12729 |  |  * | 
| 12730 |  |  * This function uses deprecated global variables to set parser options | 
| 12731 |  |  * which default to XML_PARSE_NODICT. | 
| 12732 |  |  * | 
| 12733 |  |  * @deprecated Use #xmlNewSAXParserCtxt and #xmlCtxtReadMemory. | 
| 12734 |  |  * | 
| 12735 |  |  * User data (void *) is stored within the parser context in the | 
| 12736 |  |  * context's _private member, so it is available nearly everywhere in libxml | 
| 12737 |  |  * | 
| 12738 |  |  * @param sax  the SAX handler block | 
| 12739 |  |  * @param buffer  an pointer to a char array | 
| 12740 |  |  * @param size  the size of the array | 
| 12741 |  |  * @param recovery  work in recovery mode, i.e. tries to read no Well Formed | 
| 12742 |  |  *             documents | 
| 12743 |  |  * @param data  the userdata | 
| 12744 |  |  * @returns the resulting document tree | 
| 12745 |  |  */ | 
| 12746 |  |  | 
| 12747 |  | xmlDoc * | 
| 12748 |  | xmlSAXParseMemoryWithData(xmlSAXHandler *sax, const char *buffer, | 
| 12749 | 0 |                           int size, int recovery, void *data) { | 
| 12750 | 0 |     xmlDocPtr ret = NULL; | 
| 12751 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12752 | 0 |     xmlParserInputPtr input; | 
| 12753 |  | 
 | 
| 12754 | 0 |     if (size < 0) | 
| 12755 | 0 |         return(NULL); | 
| 12756 |  |  | 
| 12757 | 0 |     ctxt = xmlNewSAXParserCtxt(sax, NULL); | 
| 12758 | 0 |     if (ctxt == NULL) | 
| 12759 | 0 |         return(NULL); | 
| 12760 |  |  | 
| 12761 | 0 |     if (data != NULL) | 
| 12762 | 0 |   ctxt->_private=data; | 
| 12763 |  | 
 | 
| 12764 | 0 |     if (recovery) { | 
| 12765 | 0 |         ctxt->options |= XML_PARSE_RECOVER; | 
| 12766 | 0 |         ctxt->recovery = 1; | 
| 12767 | 0 |     } | 
| 12768 |  | 
 | 
| 12769 | 0 |     input = xmlCtxtNewInputFromMemory(ctxt, NULL, buffer, size, NULL, | 
| 12770 | 0 |                                       XML_INPUT_BUF_STATIC); | 
| 12771 |  | 
 | 
| 12772 | 0 |     if (input != NULL) | 
| 12773 | 0 |         ret = xmlCtxtParseDocument(ctxt, input); | 
| 12774 |  | 
 | 
| 12775 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12776 | 0 |     return(ret); | 
| 12777 | 0 | } | 
| 12778 |  |  | 
| 12779 |  | /** | 
| 12780 |  |  * Parse an XML in-memory block and use the given SAX function block | 
| 12781 |  |  * to handle the parsing callback. If sax is NULL, fallback to the default | 
| 12782 |  |  * DOM tree building routines. | 
| 12783 |  |  * | 
| 12784 |  |  * This function uses deprecated global variables to set parser options | 
| 12785 |  |  * which default to XML_PARSE_NODICT. | 
| 12786 |  |  * | 
| 12787 |  |  * @deprecated Use #xmlNewSAXParserCtxt and #xmlCtxtReadMemory. | 
| 12788 |  |  * | 
| 12789 |  |  * @param sax  the SAX handler block | 
| 12790 |  |  * @param buffer  an pointer to a char array | 
| 12791 |  |  * @param size  the size of the array | 
| 12792 |  |  * @param recovery  work in recovery mode, i.e. tries to read not Well Formed | 
| 12793 |  |  *             documents | 
| 12794 |  |  * @returns the resulting document tree | 
| 12795 |  |  */ | 
| 12796 |  | xmlDoc * | 
| 12797 |  | xmlSAXParseMemory(xmlSAXHandler *sax, const char *buffer, | 
| 12798 | 0 |             int size, int recovery) { | 
| 12799 | 0 |     return xmlSAXParseMemoryWithData(sax, buffer, size, recovery, NULL); | 
| 12800 | 0 | } | 
| 12801 |  |  | 
| 12802 |  | /** | 
| 12803 |  |  * Parse an XML in-memory block and build a tree. | 
| 12804 |  |  * | 
| 12805 |  |  * This function uses deprecated global variables to set parser options | 
| 12806 |  |  * which default to XML_PARSE_NODICT. | 
| 12807 |  |  * | 
| 12808 |  |  * @deprecated Use #xmlReadMemory. | 
| 12809 |  |  * | 
| 12810 |  |  * @param buffer  an pointer to a char array | 
| 12811 |  |  * @param size  the size of the array | 
| 12812 |  |  * @returns the resulting document tree | 
| 12813 |  |  */ | 
| 12814 |  |  | 
| 12815 | 0 | xmlDoc *xmlParseMemory(const char *buffer, int size) { | 
| 12816 | 0 |    return(xmlSAXParseMemory(NULL, buffer, size, 0)); | 
| 12817 | 0 | } | 
| 12818 |  |  | 
| 12819 |  | /** | 
| 12820 |  |  * Parse an XML in-memory block and build a tree. | 
| 12821 |  |  * In the case the document is not Well Formed, an attempt to | 
| 12822 |  |  * build a tree is tried anyway | 
| 12823 |  |  * | 
| 12824 |  |  * This function uses deprecated global variables to set parser options | 
| 12825 |  |  * which default to XML_PARSE_NODICT | XML_PARSE_RECOVER. | 
| 12826 |  |  * | 
| 12827 |  |  * @deprecated Use #xmlReadMemory with XML_PARSE_RECOVER. | 
| 12828 |  |  * | 
| 12829 |  |  * @param buffer  an pointer to a char array | 
| 12830 |  |  * @param size  the size of the array | 
| 12831 |  |  * @returns the resulting document tree or NULL in case of error | 
| 12832 |  |  */ | 
| 12833 |  |  | 
| 12834 | 0 | xmlDoc *xmlRecoverMemory(const char *buffer, int size) { | 
| 12835 | 0 |    return(xmlSAXParseMemory(NULL, buffer, size, 1)); | 
| 12836 | 0 | } | 
| 12837 |  |  | 
| 12838 |  | /** | 
| 12839 |  |  * Parse an XML in-memory buffer and call the given SAX handler routines. | 
| 12840 |  |  * | 
| 12841 |  |  * This function uses deprecated global variables to set parser options | 
| 12842 |  |  * which default to XML_PARSE_NODICT. | 
| 12843 |  |  * | 
| 12844 |  |  * @deprecated Use #xmlNewSAXParserCtxt and #xmlCtxtReadMemory. | 
| 12845 |  |  * | 
| 12846 |  |  * @param sax  a SAX handler | 
| 12847 |  |  * @param user_data  The user data returned on SAX callbacks | 
| 12848 |  |  * @param buffer  an in-memory XML document input | 
| 12849 |  |  * @param size  the length of the XML document in bytes | 
| 12850 |  |  * @returns 0 in case of success or a error number otherwise | 
| 12851 |  |  */ | 
| 12852 |  | int xmlSAXUserParseMemory(xmlSAXHandler *sax, void *user_data, | 
| 12853 | 0 |         const char *buffer, int size) { | 
| 12854 | 0 |     int ret = 0; | 
| 12855 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12856 |  | 
 | 
| 12857 | 0 |     ctxt = xmlCreateMemoryParserCtxt(buffer, size); | 
| 12858 | 0 |     if (ctxt == NULL) return -1; | 
| 12859 | 0 |     if (sax != NULL) { | 
| 12860 | 0 |         if (sax->initialized == XML_SAX2_MAGIC) { | 
| 12861 | 0 |             *ctxt->sax = *sax; | 
| 12862 | 0 |         } else { | 
| 12863 | 0 |             memset(ctxt->sax, 0, sizeof(*ctxt->sax)); | 
| 12864 | 0 |             memcpy(ctxt->sax, sax, sizeof(xmlSAXHandlerV1)); | 
| 12865 | 0 |         } | 
| 12866 | 0 |   ctxt->userData = user_data; | 
| 12867 | 0 |     } | 
| 12868 |  | 
 | 
| 12869 | 0 |     xmlParseDocument(ctxt); | 
| 12870 |  | 
 | 
| 12871 | 0 |     if (ctxt->wellFormed) | 
| 12872 | 0 |   ret = 0; | 
| 12873 | 0 |     else { | 
| 12874 | 0 |         if (ctxt->errNo != 0) | 
| 12875 | 0 |       ret = ctxt->errNo; | 
| 12876 | 0 |   else | 
| 12877 | 0 |       ret = -1; | 
| 12878 | 0 |     } | 
| 12879 | 0 |     if (ctxt->myDoc != NULL) { | 
| 12880 | 0 |         xmlFreeDoc(ctxt->myDoc); | 
| 12881 | 0 |   ctxt->myDoc = NULL; | 
| 12882 | 0 |     } | 
| 12883 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12884 |  | 
 | 
| 12885 | 0 |     return ret; | 
| 12886 | 0 | } | 
| 12887 |  | #endif /* LIBXML_SAX1_ENABLED */ | 
| 12888 |  |  | 
| 12889 |  | /** | 
| 12890 |  |  * Creates a parser context for an XML in-memory document. | 
| 12891 |  |  * | 
| 12892 |  |  * @param str  a pointer to an array of xmlChar | 
| 12893 |  |  * @returns the new parser context or NULL | 
| 12894 |  |  */ | 
| 12895 |  | xmlParserCtxt * | 
| 12896 | 0 | xmlCreateDocParserCtxt(const xmlChar *str) { | 
| 12897 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12898 | 0 |     xmlParserInputPtr input; | 
| 12899 |  | 
 | 
| 12900 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 12901 | 0 |     if (ctxt == NULL) | 
| 12902 | 0 |   return(NULL); | 
| 12903 |  |  | 
| 12904 | 0 |     input = xmlCtxtNewInputFromString(ctxt, NULL, (const char *) str, NULL, 0); | 
| 12905 | 0 |     if (input == NULL) { | 
| 12906 | 0 |   xmlFreeParserCtxt(ctxt); | 
| 12907 | 0 |   return(NULL); | 
| 12908 | 0 |     } | 
| 12909 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 12910 | 0 |         xmlFreeInputStream(input); | 
| 12911 | 0 |         xmlFreeParserCtxt(ctxt); | 
| 12912 | 0 |         return(NULL); | 
| 12913 | 0 |     } | 
| 12914 |  |  | 
| 12915 | 0 |     return(ctxt); | 
| 12916 | 0 | } | 
| 12917 |  |  | 
| 12918 |  | #ifdef LIBXML_SAX1_ENABLED | 
| 12919 |  | /** | 
| 12920 |  |  * Parse an XML in-memory document and build a tree. | 
| 12921 |  |  * It use the given SAX function block to handle the parsing callback. | 
| 12922 |  |  * If sax is NULL, fallback to the default DOM tree building routines. | 
| 12923 |  |  * | 
| 12924 |  |  * This function uses deprecated global variables to set parser options | 
| 12925 |  |  * which default to XML_PARSE_NODICT. | 
| 12926 |  |  * | 
| 12927 |  |  * @deprecated Use #xmlNewSAXParserCtxt and #xmlCtxtReadDoc. | 
| 12928 |  |  * | 
| 12929 |  |  * @param sax  the SAX handler block | 
| 12930 |  |  * @param cur  a pointer to an array of xmlChar | 
| 12931 |  |  * @param recovery  work in recovery mode, i.e. tries to read no Well Formed | 
| 12932 |  |  *             documents | 
| 12933 |  |  * @returns the resulting document tree | 
| 12934 |  |  */ | 
| 12935 |  |  | 
| 12936 |  | xmlDoc * | 
| 12937 | 0 | xmlSAXParseDoc(xmlSAXHandler *sax, const xmlChar *cur, int recovery) { | 
| 12938 | 0 |     xmlDocPtr ret; | 
| 12939 | 0 |     xmlParserCtxtPtr ctxt; | 
| 12940 | 0 |     xmlSAXHandlerPtr oldsax = NULL; | 
| 12941 |  | 
 | 
| 12942 | 0 |     if (cur == NULL) return(NULL); | 
| 12943 |  |  | 
| 12944 |  |  | 
| 12945 | 0 |     ctxt = xmlCreateDocParserCtxt(cur); | 
| 12946 | 0 |     if (ctxt == NULL) return(NULL); | 
| 12947 | 0 |     if (sax != NULL) { | 
| 12948 | 0 |         oldsax = ctxt->sax; | 
| 12949 | 0 |         ctxt->sax = sax; | 
| 12950 | 0 |         ctxt->userData = NULL; | 
| 12951 | 0 |     } | 
| 12952 |  | 
 | 
| 12953 | 0 |     xmlParseDocument(ctxt); | 
| 12954 | 0 |     if ((ctxt->wellFormed) || recovery) ret = ctxt->myDoc; | 
| 12955 | 0 |     else { | 
| 12956 | 0 |        ret = NULL; | 
| 12957 | 0 |        xmlFreeDoc(ctxt->myDoc); | 
| 12958 | 0 |        ctxt->myDoc = NULL; | 
| 12959 | 0 |     } | 
| 12960 | 0 |     if (sax != NULL) | 
| 12961 | 0 |   ctxt->sax = oldsax; | 
| 12962 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 12963 |  | 
 | 
| 12964 | 0 |     return(ret); | 
| 12965 | 0 | } | 
| 12966 |  |  | 
| 12967 |  | /** | 
| 12968 |  |  * Parse an XML in-memory document and build a tree. | 
| 12969 |  |  * | 
| 12970 |  |  * This function uses deprecated global variables to set parser options | 
| 12971 |  |  * which default to XML_PARSE_NODICT. | 
| 12972 |  |  * | 
| 12973 |  |  * @deprecated Use #xmlReadDoc. | 
| 12974 |  |  * | 
| 12975 |  |  * @param cur  a pointer to an array of xmlChar | 
| 12976 |  |  * @returns the resulting document tree | 
| 12977 |  |  */ | 
| 12978 |  |  | 
| 12979 |  | xmlDoc * | 
| 12980 | 0 | xmlParseDoc(const xmlChar *cur) { | 
| 12981 | 0 |     return(xmlSAXParseDoc(NULL, cur, 0)); | 
| 12982 | 0 | } | 
| 12983 |  | #endif /* LIBXML_SAX1_ENABLED */ | 
| 12984 |  |  | 
| 12985 |  | /************************************************************************ | 
| 12986 |  |  *                  * | 
| 12987 |  |  *  New set (2.6.0) of simpler and more flexible APIs   * | 
| 12988 |  |  *                  * | 
| 12989 |  |  ************************************************************************/ | 
| 12990 |  |  | 
| 12991 |  | /** | 
| 12992 |  |  * Reset a parser context | 
| 12993 |  |  * | 
| 12994 |  |  * @param ctxt  an XML parser context | 
| 12995 |  |  */ | 
| 12996 |  | void | 
| 12997 |  | xmlCtxtReset(xmlParserCtxt *ctxt) | 
| 12998 | 0 | { | 
| 12999 | 0 |     xmlParserInputPtr input; | 
| 13000 |  | 
 | 
| 13001 | 0 |     if (ctxt == NULL) | 
| 13002 | 0 |         return; | 
| 13003 |  |  | 
| 13004 | 0 |     while ((input = xmlCtxtPopInput(ctxt)) != NULL) { /* Non consuming */ | 
| 13005 | 0 |         xmlFreeInputStream(input); | 
| 13006 | 0 |     } | 
| 13007 | 0 |     ctxt->inputNr = 0; | 
| 13008 | 0 |     ctxt->input = NULL; | 
| 13009 |  | 
 | 
| 13010 | 0 |     ctxt->spaceNr = 0; | 
| 13011 | 0 |     if (ctxt->spaceTab != NULL) { | 
| 13012 | 0 |   ctxt->spaceTab[0] = -1; | 
| 13013 | 0 |   ctxt->space = &ctxt->spaceTab[0]; | 
| 13014 | 0 |     } else { | 
| 13015 | 0 |         ctxt->space = NULL; | 
| 13016 | 0 |     } | 
| 13017 |  |  | 
| 13018 |  | 
 | 
| 13019 | 0 |     ctxt->nodeNr = 0; | 
| 13020 | 0 |     ctxt->node = NULL; | 
| 13021 |  | 
 | 
| 13022 | 0 |     ctxt->nameNr = 0; | 
| 13023 | 0 |     ctxt->name = NULL; | 
| 13024 |  | 
 | 
| 13025 | 0 |     ctxt->nsNr = 0; | 
| 13026 | 0 |     xmlParserNsReset(ctxt->nsdb); | 
| 13027 |  | 
 | 
| 13028 | 0 |     if (ctxt->version != NULL) { | 
| 13029 | 0 |         xmlFree(ctxt->version); | 
| 13030 | 0 |         ctxt->version = NULL; | 
| 13031 | 0 |     } | 
| 13032 | 0 |     if (ctxt->encoding != NULL) { | 
| 13033 | 0 |         xmlFree(ctxt->encoding); | 
| 13034 | 0 |         ctxt->encoding = NULL; | 
| 13035 | 0 |     } | 
| 13036 | 0 |     if (ctxt->extSubURI != NULL) { | 
| 13037 | 0 |         xmlFree(ctxt->extSubURI); | 
| 13038 | 0 |         ctxt->extSubURI = NULL; | 
| 13039 | 0 |     } | 
| 13040 | 0 |     if (ctxt->extSubSystem != NULL) { | 
| 13041 | 0 |         xmlFree(ctxt->extSubSystem); | 
| 13042 | 0 |         ctxt->extSubSystem = NULL; | 
| 13043 | 0 |     } | 
| 13044 | 0 |     if (ctxt->directory != NULL) { | 
| 13045 | 0 |         xmlFree(ctxt->directory); | 
| 13046 | 0 |         ctxt->directory = NULL; | 
| 13047 | 0 |     } | 
| 13048 |  | 
 | 
| 13049 | 0 |     if (ctxt->myDoc != NULL) | 
| 13050 | 0 |         xmlFreeDoc(ctxt->myDoc); | 
| 13051 | 0 |     ctxt->myDoc = NULL; | 
| 13052 |  | 
 | 
| 13053 | 0 |     ctxt->standalone = -1; | 
| 13054 | 0 |     ctxt->hasExternalSubset = 0; | 
| 13055 | 0 |     ctxt->hasPErefs = 0; | 
| 13056 | 0 |     ctxt->html = ctxt->html ? 1 : 0; | 
| 13057 | 0 |     ctxt->instate = XML_PARSER_START; | 
| 13058 |  | 
 | 
| 13059 | 0 |     ctxt->wellFormed = 1; | 
| 13060 | 0 |     ctxt->nsWellFormed = 1; | 
| 13061 | 0 |     ctxt->disableSAX = 0; | 
| 13062 | 0 |     ctxt->valid = 1; | 
| 13063 | 0 |     ctxt->record_info = 0; | 
| 13064 | 0 |     ctxt->checkIndex = 0; | 
| 13065 | 0 |     ctxt->endCheckState = 0; | 
| 13066 | 0 |     ctxt->inSubset = 0; | 
| 13067 | 0 |     ctxt->errNo = XML_ERR_OK; | 
| 13068 | 0 |     ctxt->depth = 0; | 
| 13069 | 0 |     ctxt->catalogs = NULL; | 
| 13070 | 0 |     ctxt->sizeentities = 0; | 
| 13071 | 0 |     ctxt->sizeentcopy = 0; | 
| 13072 | 0 |     xmlInitNodeInfoSeq(&ctxt->node_seq); | 
| 13073 |  | 
 | 
| 13074 | 0 |     if (ctxt->attsDefault != NULL) { | 
| 13075 | 0 |         xmlHashFree(ctxt->attsDefault, xmlHashDefaultDeallocator); | 
| 13076 | 0 |         ctxt->attsDefault = NULL; | 
| 13077 | 0 |     } | 
| 13078 | 0 |     if (ctxt->attsSpecial != NULL) { | 
| 13079 | 0 |         xmlHashFree(ctxt->attsSpecial, NULL); | 
| 13080 | 0 |         ctxt->attsSpecial = NULL; | 
| 13081 | 0 |     } | 
| 13082 |  | 
 | 
| 13083 | 0 | #ifdef LIBXML_CATALOG_ENABLED | 
| 13084 | 0 |     if (ctxt->catalogs != NULL) | 
| 13085 | 0 |   xmlCatalogFreeLocal(ctxt->catalogs); | 
| 13086 | 0 | #endif | 
| 13087 | 0 |     ctxt->nbErrors = 0; | 
| 13088 | 0 |     ctxt->nbWarnings = 0; | 
| 13089 | 0 |     if (ctxt->lastError.code != XML_ERR_OK) | 
| 13090 | 0 |         xmlResetError(&ctxt->lastError); | 
| 13091 | 0 | } | 
| 13092 |  |  | 
| 13093 |  | /** | 
| 13094 |  |  * Reset a push parser context | 
| 13095 |  |  * | 
| 13096 |  |  * @param ctxt  an XML parser context | 
| 13097 |  |  * @param chunk  a pointer to an array of chars | 
| 13098 |  |  * @param size  number of chars in the array | 
| 13099 |  |  * @param filename  an optional file name or URI | 
| 13100 |  |  * @param encoding  the document encoding, or NULL | 
| 13101 |  |  * @returns 0 in case of success and 1 in case of error | 
| 13102 |  |  */ | 
| 13103 |  | int | 
| 13104 |  | xmlCtxtResetPush(xmlParserCtxt *ctxt, const char *chunk, | 
| 13105 |  |                  int size, const char *filename, const char *encoding) | 
| 13106 | 0 | { | 
| 13107 | 0 |     xmlParserInputPtr input; | 
| 13108 |  | 
 | 
| 13109 | 0 |     if (ctxt == NULL) | 
| 13110 | 0 |         return(1); | 
| 13111 |  |  | 
| 13112 | 0 |     xmlCtxtReset(ctxt); | 
| 13113 |  | 
 | 
| 13114 | 0 |     input = xmlNewPushInput(filename, chunk, size); | 
| 13115 | 0 |     if (input == NULL) | 
| 13116 | 0 |         return(1); | 
| 13117 |  |  | 
| 13118 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 13119 | 0 |         xmlFreeInputStream(input); | 
| 13120 | 0 |         return(1); | 
| 13121 | 0 |     } | 
| 13122 |  |  | 
| 13123 | 0 |     if (encoding != NULL) | 
| 13124 | 0 |         xmlSwitchEncodingName(ctxt, encoding); | 
| 13125 |  | 
 | 
| 13126 | 0 |     return(0); | 
| 13127 | 0 | } | 
| 13128 |  |  | 
| 13129 |  | static int | 
| 13130 |  | xmlCtxtSetOptionsInternal(xmlParserCtxtPtr ctxt, int options, int keepMask) | 
| 13131 | 34.8k | { | 
| 13132 | 34.8k |     int allMask; | 
| 13133 |  |  | 
| 13134 | 34.8k |     if (ctxt == NULL) | 
| 13135 | 0 |         return(-1); | 
| 13136 |  |  | 
| 13137 |  |     /* | 
| 13138 |  |      * XInclude options aren't handled by the parser. | 
| 13139 |  |      * | 
| 13140 |  |      * XML_PARSE_XINCLUDE | 
| 13141 |  |      * XML_PARSE_NOXINCNODE | 
| 13142 |  |      * XML_PARSE_NOBASEFIX | 
| 13143 |  |      */ | 
| 13144 | 34.8k |     allMask = XML_PARSE_RECOVER | | 
| 13145 | 34.8k |               XML_PARSE_NOENT | | 
| 13146 | 34.8k |               XML_PARSE_DTDLOAD | | 
| 13147 | 34.8k |               XML_PARSE_DTDATTR | | 
| 13148 | 34.8k |               XML_PARSE_DTDVALID | | 
| 13149 | 34.8k |               XML_PARSE_NOERROR | | 
| 13150 | 34.8k |               XML_PARSE_NOWARNING | | 
| 13151 | 34.8k |               XML_PARSE_PEDANTIC | | 
| 13152 | 34.8k |               XML_PARSE_NOBLANKS | | 
| 13153 | 34.8k | #ifdef LIBXML_SAX1_ENABLED | 
| 13154 | 34.8k |               XML_PARSE_SAX1 | | 
| 13155 | 34.8k | #endif | 
| 13156 | 34.8k |               XML_PARSE_NONET | | 
| 13157 | 34.8k |               XML_PARSE_NODICT | | 
| 13158 | 34.8k |               XML_PARSE_NSCLEAN | | 
| 13159 | 34.8k |               XML_PARSE_NOCDATA | | 
| 13160 | 34.8k |               XML_PARSE_COMPACT | | 
| 13161 | 34.8k |               XML_PARSE_OLD10 | | 
| 13162 | 34.8k |               XML_PARSE_HUGE | | 
| 13163 | 34.8k |               XML_PARSE_OLDSAX | | 
| 13164 | 34.8k |               XML_PARSE_IGNORE_ENC | | 
| 13165 | 34.8k |               XML_PARSE_BIG_LINES | | 
| 13166 | 34.8k |               XML_PARSE_NO_XXE | | 
| 13167 | 34.8k |               XML_PARSE_UNZIP | | 
| 13168 | 34.8k |               XML_PARSE_NO_SYS_CATALOG | | 
| 13169 | 34.8k |               XML_PARSE_CATALOG_PI; | 
| 13170 |  |  | 
| 13171 | 34.8k |     ctxt->options = (ctxt->options & keepMask) | (options & allMask); | 
| 13172 |  |  | 
| 13173 |  |     /* | 
| 13174 |  |      * For some options, struct members are historically the source | 
| 13175 |  |      * of truth. The values are initalized from global variables and | 
| 13176 |  |      * old code could also modify them directly. Several older API | 
| 13177 |  |      * functions that don't take an options argument rely on these | 
| 13178 |  |      * deprecated mechanisms. | 
| 13179 |  |      * | 
| 13180 |  |      * Once public access to struct members and the globals are | 
| 13181 |  |      * disabled, we can use the options bitmask as source of | 
| 13182 |  |      * truth, making all these struct members obsolete. | 
| 13183 |  |      * | 
| 13184 |  |      * The XML_DETECT_IDS flags is misnamed. It simply enables | 
| 13185 |  |      * loading of the external subset. | 
| 13186 |  |      */ | 
| 13187 | 34.8k |     ctxt->recovery = (options & XML_PARSE_RECOVER) ? 1 : 0; | 
| 13188 | 34.8k |     ctxt->replaceEntities = (options & XML_PARSE_NOENT) ? 1 : 0; | 
| 13189 | 34.8k |     ctxt->loadsubset = (options & XML_PARSE_DTDLOAD) ? XML_DETECT_IDS : 0; | 
| 13190 | 34.8k |     ctxt->loadsubset |= (options & XML_PARSE_DTDATTR) ? XML_COMPLETE_ATTRS : 0; | 
| 13191 | 34.8k |     ctxt->loadsubset |= (options & XML_PARSE_SKIP_IDS) ? XML_SKIP_IDS : 0; | 
| 13192 | 34.8k |     ctxt->validate = (options & XML_PARSE_DTDVALID) ? 1 : 0; | 
| 13193 | 34.8k |     ctxt->pedantic = (options & XML_PARSE_PEDANTIC) ? 1 : 0; | 
| 13194 | 34.8k |     ctxt->keepBlanks = (options & XML_PARSE_NOBLANKS) ? 0 : 1; | 
| 13195 | 34.8k |     ctxt->dictNames = (options & XML_PARSE_NODICT) ? 0 : 1; | 
| 13196 |  |  | 
| 13197 | 34.8k |     return(options & ~allMask); | 
| 13198 | 34.8k | } | 
| 13199 |  |  | 
| 13200 |  | /** | 
| 13201 |  |  * Applies the options to the parser context. Unset options are | 
| 13202 |  |  * cleared. | 
| 13203 |  |  * | 
| 13204 |  |  * @since 2.13.0 | 
| 13205 |  |  * | 
| 13206 |  |  * With older versions, you can use #xmlCtxtUseOptions. | 
| 13207 |  |  * | 
| 13208 |  |  * @param ctxt  an XML parser context | 
| 13209 |  |  * @param options  a bitmask of xmlParserOption values | 
| 13210 |  |  * @returns 0 in case of success, the set of unknown or unimplemented options | 
| 13211 |  |  *         in case of error. | 
| 13212 |  |  */ | 
| 13213 |  | int | 
| 13214 |  | xmlCtxtSetOptions(xmlParserCtxt *ctxt, int options) | 
| 13215 | 17.4k | { | 
| 13216 | 17.4k | #ifdef LIBXML_HTML_ENABLED | 
| 13217 | 17.4k |     if ((ctxt != NULL) && (ctxt->html)) | 
| 13218 | 0 |         return(htmlCtxtSetOptions(ctxt, options)); | 
| 13219 | 17.4k | #endif | 
| 13220 |  |  | 
| 13221 | 17.4k |     return(xmlCtxtSetOptionsInternal(ctxt, options, 0)); | 
| 13222 | 17.4k | } | 
| 13223 |  |  | 
| 13224 |  | /** | 
| 13225 |  |  * Get the current options of the parser context. | 
| 13226 |  |  * | 
| 13227 |  |  * @since 2.14.0 | 
| 13228 |  |  * | 
| 13229 |  |  * @param ctxt  an XML parser context | 
| 13230 |  |  * @returns the current options set in the parser context, or -1 if ctxt is NULL. | 
| 13231 |  |  */ | 
| 13232 |  | int | 
| 13233 |  | xmlCtxtGetOptions(xmlParserCtxt *ctxt) | 
| 13234 | 0 | { | 
| 13235 | 0 |     if (ctxt == NULL) | 
| 13236 | 0 |         return(-1); | 
| 13237 |  |  | 
| 13238 | 0 |     return(ctxt->options); | 
| 13239 | 0 | } | 
| 13240 |  |  | 
| 13241 |  | /** | 
| 13242 |  |  * Applies the options to the parser context. The following options | 
| 13243 |  |  * are never cleared and can only be enabled: | 
| 13244 |  |  * | 
| 13245 |  |  * - XML_PARSE_NOERROR | 
| 13246 |  |  * - XML_PARSE_NOWARNING | 
| 13247 |  |  * - XML_PARSE_NONET | 
| 13248 |  |  * - XML_PARSE_NSCLEAN | 
| 13249 |  |  * - XML_PARSE_NOCDATA | 
| 13250 |  |  * - XML_PARSE_COMPACT | 
| 13251 |  |  * - XML_PARSE_OLD10 | 
| 13252 |  |  * - XML_PARSE_HUGE | 
| 13253 |  |  * - XML_PARSE_OLDSAX | 
| 13254 |  |  * - XML_PARSE_IGNORE_ENC | 
| 13255 |  |  * - XML_PARSE_BIG_LINES | 
| 13256 |  |  * | 
| 13257 |  |  * @deprecated Use #xmlCtxtSetOptions. | 
| 13258 |  |  * | 
| 13259 |  |  * @param ctxt  an XML parser context | 
| 13260 |  |  * @param options  a combination of xmlParserOption | 
| 13261 |  |  * @returns 0 in case of success, the set of unknown or unimplemented options | 
| 13262 |  |  *         in case of error. | 
| 13263 |  |  */ | 
| 13264 |  | int | 
| 13265 |  | xmlCtxtUseOptions(xmlParserCtxt *ctxt, int options) | 
| 13266 | 17.4k | { | 
| 13267 | 17.4k |     int keepMask; | 
| 13268 |  |  | 
| 13269 | 17.4k | #ifdef LIBXML_HTML_ENABLED | 
| 13270 | 17.4k |     if ((ctxt != NULL) && (ctxt->html)) | 
| 13271 | 0 |         return(htmlCtxtUseOptions(ctxt, options)); | 
| 13272 | 17.4k | #endif | 
| 13273 |  |  | 
| 13274 |  |     /* | 
| 13275 |  |      * For historic reasons, some options can only be enabled. | 
| 13276 |  |      */ | 
| 13277 | 17.4k |     keepMask = XML_PARSE_NOERROR | | 
| 13278 | 17.4k |                XML_PARSE_NOWARNING | | 
| 13279 | 17.4k |                XML_PARSE_NONET | | 
| 13280 | 17.4k |                XML_PARSE_NSCLEAN | | 
| 13281 | 17.4k |                XML_PARSE_NOCDATA | | 
| 13282 | 17.4k |                XML_PARSE_COMPACT | | 
| 13283 | 17.4k |                XML_PARSE_OLD10 | | 
| 13284 | 17.4k |                XML_PARSE_HUGE | | 
| 13285 | 17.4k |                XML_PARSE_OLDSAX | | 
| 13286 | 17.4k |                XML_PARSE_IGNORE_ENC | | 
| 13287 | 17.4k |                XML_PARSE_BIG_LINES; | 
| 13288 |  |  | 
| 13289 | 17.4k |     return(xmlCtxtSetOptionsInternal(ctxt, options, keepMask)); | 
| 13290 | 17.4k | } | 
| 13291 |  |  | 
| 13292 |  | /** | 
| 13293 |  |  * To protect against exponential entity expansion ("billion laughs"), the | 
| 13294 |  |  * size of serialized output is (roughly) limited to the input size | 
| 13295 |  |  * multiplied by this factor. The default value is 5. | 
| 13296 |  |  * | 
| 13297 |  |  * When working with documents making heavy use of entity expansion, it can | 
| 13298 |  |  * be necessary to increase the value. For security reasons, this should only | 
| 13299 |  |  * be considered when processing trusted input. | 
| 13300 |  |  * | 
| 13301 |  |  * @param ctxt  an XML parser context | 
| 13302 |  |  * @param maxAmpl  maximum amplification factor | 
| 13303 |  |  */ | 
| 13304 |  | void | 
| 13305 |  | xmlCtxtSetMaxAmplification(xmlParserCtxt *ctxt, unsigned maxAmpl) | 
| 13306 | 0 | { | 
| 13307 | 0 |     if (ctxt == NULL) | 
| 13308 | 0 |         return; | 
| 13309 | 0 |     ctxt->maxAmpl = maxAmpl; | 
| 13310 | 0 | } | 
| 13311 |  |  | 
| 13312 |  | /** | 
| 13313 |  |  * Parse an XML document and return the resulting document tree. | 
| 13314 |  |  * Takes ownership of the input object. | 
| 13315 |  |  * | 
| 13316 |  |  * @since 2.13.0 | 
| 13317 |  |  * | 
| 13318 |  |  * @param ctxt  an XML parser context | 
| 13319 |  |  * @param input  parser input | 
| 13320 |  |  * @returns the resulting document tree or NULL | 
| 13321 |  |  */ | 
| 13322 |  | xmlDoc * | 
| 13323 |  | xmlCtxtParseDocument(xmlParserCtxt *ctxt, xmlParserInput *input) | 
| 13324 | 0 | { | 
| 13325 | 0 |     xmlDocPtr ret = NULL; | 
| 13326 |  | 
 | 
| 13327 | 0 |     if ((ctxt == NULL) || (input == NULL)) { | 
| 13328 | 0 |         xmlFatalErr(ctxt, XML_ERR_ARGUMENT, NULL); | 
| 13329 | 0 |         xmlFreeInputStream(input); | 
| 13330 | 0 |         return(NULL); | 
| 13331 | 0 |     } | 
| 13332 |  |  | 
| 13333 |  |     /* assert(ctxt->inputNr == 0); */ | 
| 13334 | 0 |     while (ctxt->inputNr > 0) | 
| 13335 | 0 |         xmlFreeInputStream(xmlCtxtPopInput(ctxt)); | 
| 13336 |  | 
 | 
| 13337 | 0 |     if (xmlCtxtPushInput(ctxt, input) < 0) { | 
| 13338 | 0 |         xmlFreeInputStream(input); | 
| 13339 | 0 |         return(NULL); | 
| 13340 | 0 |     } | 
| 13341 |  |  | 
| 13342 | 0 |     xmlParseDocument(ctxt); | 
| 13343 |  | 
 | 
| 13344 | 0 |     ret = xmlCtxtGetDocument(ctxt); | 
| 13345 |  |  | 
| 13346 |  |     /* assert(ctxt->inputNr == 1); */ | 
| 13347 | 0 |     while (ctxt->inputNr > 0) | 
| 13348 | 0 |         xmlFreeInputStream(xmlCtxtPopInput(ctxt)); | 
| 13349 |  | 
 | 
| 13350 | 0 |     return(ret); | 
| 13351 | 0 | } | 
| 13352 |  |  | 
| 13353 |  | /** | 
| 13354 |  |  * Convenience function to parse an XML document from a | 
| 13355 |  |  * zero-terminated string. | 
| 13356 |  |  * | 
| 13357 |  |  * See #xmlCtxtReadDoc for details. | 
| 13358 |  |  * | 
| 13359 |  |  * @param cur  a pointer to a zero terminated string | 
| 13360 |  |  * @param URL  base URL (optional) | 
| 13361 |  |  * @param encoding  the document encoding (optional) | 
| 13362 |  |  * @param options  a combination of xmlParserOption | 
| 13363 |  |  * @returns the resulting document tree | 
| 13364 |  |  */ | 
| 13365 |  | xmlDoc * | 
| 13366 |  | xmlReadDoc(const xmlChar *cur, const char *URL, const char *encoding, | 
| 13367 |  |            int options) | 
| 13368 | 0 | { | 
| 13369 | 0 |     xmlParserCtxtPtr ctxt; | 
| 13370 | 0 |     xmlParserInputPtr input; | 
| 13371 | 0 |     xmlDocPtr doc = NULL; | 
| 13372 |  | 
 | 
| 13373 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 13374 | 0 |     if (ctxt == NULL) | 
| 13375 | 0 |         return(NULL); | 
| 13376 |  |  | 
| 13377 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13378 |  | 
 | 
| 13379 | 0 |     input = xmlCtxtNewInputFromString(ctxt, URL, (const char *) cur, encoding, | 
| 13380 | 0 |                                       XML_INPUT_BUF_STATIC); | 
| 13381 |  | 
 | 
| 13382 | 0 |     if (input != NULL) | 
| 13383 | 0 |         doc = xmlCtxtParseDocument(ctxt, input); | 
| 13384 |  | 
 | 
| 13385 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 13386 | 0 |     return(doc); | 
| 13387 | 0 | } | 
| 13388 |  |  | 
| 13389 |  | /** | 
| 13390 |  |  * Convenience function to parse an XML file from the filesystem | 
| 13391 |  |  * or a global, user-defined resource loader. | 
| 13392 |  |  * | 
| 13393 |  |  * If a "-" filename is passed, the function will read from stdin. | 
| 13394 |  |  * This feature is potentially insecure and might be removed from | 
| 13395 |  |  * later versions. | 
| 13396 |  |  * | 
| 13397 |  |  * See #xmlCtxtReadFile for details. | 
| 13398 |  |  * | 
| 13399 |  |  * @param filename  a file or URL | 
| 13400 |  |  * @param encoding  the document encoding (optional) | 
| 13401 |  |  * @param options  a combination of xmlParserOption | 
| 13402 |  |  * @returns the resulting document tree | 
| 13403 |  |  */ | 
| 13404 |  | xmlDoc * | 
| 13405 |  | xmlReadFile(const char *filename, const char *encoding, int options) | 
| 13406 | 0 | { | 
| 13407 | 0 |     xmlParserCtxtPtr ctxt; | 
| 13408 | 0 |     xmlParserInputPtr input; | 
| 13409 | 0 |     xmlDocPtr doc = NULL; | 
| 13410 |  | 
 | 
| 13411 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 13412 | 0 |     if (ctxt == NULL) | 
| 13413 | 0 |         return(NULL); | 
| 13414 |  |  | 
| 13415 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13416 |  |  | 
| 13417 |  |     /* | 
| 13418 |  |      * Backward compatibility for users of command line utilities like | 
| 13419 |  |      * xmlstarlet expecting "-" to mean stdin. This is dangerous and | 
| 13420 |  |      * should be removed at some point. | 
| 13421 |  |      */ | 
| 13422 | 0 |     if ((filename != NULL) && (filename[0] == '-') && (filename[1] == 0)) | 
| 13423 | 0 |         input = xmlCtxtNewInputFromFd(ctxt, filename, STDIN_FILENO, | 
| 13424 | 0 |                                       encoding, 0); | 
| 13425 | 0 |     else | 
| 13426 | 0 |         input = xmlCtxtNewInputFromUrl(ctxt, filename, NULL, encoding, 0); | 
| 13427 |  | 
 | 
| 13428 | 0 |     if (input != NULL) | 
| 13429 | 0 |         doc = xmlCtxtParseDocument(ctxt, input); | 
| 13430 |  | 
 | 
| 13431 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 13432 | 0 |     return(doc); | 
| 13433 | 0 | } | 
| 13434 |  |  | 
| 13435 |  | /** | 
| 13436 |  |  * Parse an XML in-memory document and build a tree. The input buffer must | 
| 13437 |  |  * not contain a terminating null byte. | 
| 13438 |  |  * | 
| 13439 |  |  * See #xmlCtxtReadMemory for details. | 
| 13440 |  |  * | 
| 13441 |  |  * @param buffer  a pointer to a char array | 
| 13442 |  |  * @param size  the size of the array | 
| 13443 |  |  * @param url  base URL (optional) | 
| 13444 |  |  * @param encoding  the document encoding (optional) | 
| 13445 |  |  * @param options  a combination of xmlParserOption | 
| 13446 |  |  * @returns the resulting document tree | 
| 13447 |  |  */ | 
| 13448 |  | xmlDoc * | 
| 13449 |  | xmlReadMemory(const char *buffer, int size, const char *url, | 
| 13450 |  |               const char *encoding, int options) | 
| 13451 | 0 | { | 
| 13452 | 0 |     xmlParserCtxtPtr ctxt; | 
| 13453 | 0 |     xmlParserInputPtr input; | 
| 13454 | 0 |     xmlDocPtr doc = NULL; | 
| 13455 |  | 
 | 
| 13456 | 0 |     if (size < 0) | 
| 13457 | 0 |   return(NULL); | 
| 13458 |  |  | 
| 13459 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 13460 | 0 |     if (ctxt == NULL) | 
| 13461 | 0 |         return(NULL); | 
| 13462 |  |  | 
| 13463 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13464 |  | 
 | 
| 13465 | 0 |     input = xmlCtxtNewInputFromMemory(ctxt, url, buffer, size, encoding, | 
| 13466 | 0 |                                       XML_INPUT_BUF_STATIC); | 
| 13467 |  | 
 | 
| 13468 | 0 |     if (input != NULL) | 
| 13469 | 0 |         doc = xmlCtxtParseDocument(ctxt, input); | 
| 13470 |  | 
 | 
| 13471 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 13472 | 0 |     return(doc); | 
| 13473 | 0 | } | 
| 13474 |  |  | 
| 13475 |  | /** | 
| 13476 |  |  * Parse an XML from a file descriptor and build a tree. | 
| 13477 |  |  * | 
| 13478 |  |  * See #xmlCtxtReadFd for details. | 
| 13479 |  |  * | 
| 13480 |  |  * NOTE that the file descriptor will not be closed when the | 
| 13481 |  |  * context is freed or reset. | 
| 13482 |  |  * | 
| 13483 |  |  * @param fd  an open file descriptor | 
| 13484 |  |  * @param URL  base URL (optional) | 
| 13485 |  |  * @param encoding  the document encoding (optional) | 
| 13486 |  |  * @param options  a combination of xmlParserOption | 
| 13487 |  |  * @returns the resulting document tree | 
| 13488 |  |  */ | 
| 13489 |  | xmlDoc * | 
| 13490 |  | xmlReadFd(int fd, const char *URL, const char *encoding, int options) | 
| 13491 | 0 | { | 
| 13492 | 0 |     xmlParserCtxtPtr ctxt; | 
| 13493 | 0 |     xmlParserInputPtr input; | 
| 13494 | 0 |     xmlDocPtr doc = NULL; | 
| 13495 |  | 
 | 
| 13496 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 13497 | 0 |     if (ctxt == NULL) | 
| 13498 | 0 |         return(NULL); | 
| 13499 |  |  | 
| 13500 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13501 |  | 
 | 
| 13502 | 0 |     input = xmlCtxtNewInputFromFd(ctxt, URL, fd, encoding, 0); | 
| 13503 |  | 
 | 
| 13504 | 0 |     if (input != NULL) | 
| 13505 | 0 |         doc = xmlCtxtParseDocument(ctxt, input); | 
| 13506 |  | 
 | 
| 13507 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 13508 | 0 |     return(doc); | 
| 13509 | 0 | } | 
| 13510 |  |  | 
| 13511 |  | /** | 
| 13512 |  |  * Parse an XML document from I/O functions and context and build a tree. | 
| 13513 |  |  * | 
| 13514 |  |  * See #xmlCtxtReadIO for details. | 
| 13515 |  |  * | 
| 13516 |  |  * @param ioread  an I/O read function | 
| 13517 |  |  * @param ioclose  an I/O close function (optional) | 
| 13518 |  |  * @param ioctx  an I/O handler | 
| 13519 |  |  * @param URL  base URL (optional) | 
| 13520 |  |  * @param encoding  the document encoding (optional) | 
| 13521 |  |  * @param options  a combination of xmlParserOption | 
| 13522 |  |  * @returns the resulting document tree | 
| 13523 |  |  */ | 
| 13524 |  | xmlDoc * | 
| 13525 |  | xmlReadIO(xmlInputReadCallback ioread, xmlInputCloseCallback ioclose, | 
| 13526 |  |           void *ioctx, const char *URL, const char *encoding, int options) | 
| 13527 | 0 | { | 
| 13528 | 0 |     xmlParserCtxtPtr ctxt; | 
| 13529 | 0 |     xmlParserInputPtr input; | 
| 13530 | 0 |     xmlDocPtr doc = NULL; | 
| 13531 |  | 
 | 
| 13532 | 0 |     ctxt = xmlNewParserCtxt(); | 
| 13533 | 0 |     if (ctxt == NULL) | 
| 13534 | 0 |         return(NULL); | 
| 13535 |  |  | 
| 13536 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13537 |  | 
 | 
| 13538 | 0 |     input = xmlCtxtNewInputFromIO(ctxt, URL, ioread, ioclose, ioctx, | 
| 13539 | 0 |                                   encoding, 0); | 
| 13540 |  | 
 | 
| 13541 | 0 |     if (input != NULL) | 
| 13542 | 0 |         doc = xmlCtxtParseDocument(ctxt, input); | 
| 13543 |  | 
 | 
| 13544 | 0 |     xmlFreeParserCtxt(ctxt); | 
| 13545 | 0 |     return(doc); | 
| 13546 | 0 | } | 
| 13547 |  |  | 
| 13548 |  | /** | 
| 13549 |  |  * Parse an XML in-memory document and build a tree. | 
| 13550 |  |  * | 
| 13551 |  |  * `URL` is used as base to resolve external entities and for error | 
| 13552 |  |  * reporting. | 
| 13553 |  |  * | 
| 13554 |  |  * @param ctxt  an XML parser context | 
| 13555 |  |  * @param str  a pointer to a zero terminated string | 
| 13556 |  |  * @param URL  base URL (optional) | 
| 13557 |  |  * @param encoding  the document encoding (optional) | 
| 13558 |  |  * @param options  a combination of xmlParserOption | 
| 13559 |  |  * @returns the resulting document tree | 
| 13560 |  |  */ | 
| 13561 |  | xmlDoc * | 
| 13562 |  | xmlCtxtReadDoc(xmlParserCtxt *ctxt, const xmlChar *str, | 
| 13563 |  |                const char *URL, const char *encoding, int options) | 
| 13564 | 0 | { | 
| 13565 | 0 |     xmlParserInputPtr input; | 
| 13566 |  | 
 | 
| 13567 | 0 |     if (ctxt == NULL) | 
| 13568 | 0 |         return(NULL); | 
| 13569 |  |  | 
| 13570 | 0 |     xmlCtxtReset(ctxt); | 
| 13571 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13572 |  | 
 | 
| 13573 | 0 |     input = xmlCtxtNewInputFromString(ctxt, URL, (const char *) str, encoding, | 
| 13574 | 0 |                                       XML_INPUT_BUF_STATIC); | 
| 13575 | 0 |     if (input == NULL) | 
| 13576 | 0 |         return(NULL); | 
| 13577 |  |  | 
| 13578 | 0 |     return(xmlCtxtParseDocument(ctxt, input)); | 
| 13579 | 0 | } | 
| 13580 |  |  | 
| 13581 |  | /** | 
| 13582 |  |  * Parse an XML file from the filesystem or a global, user-defined | 
| 13583 |  |  * resource loader. | 
| 13584 |  |  * | 
| 13585 |  |  * @param ctxt  an XML parser context | 
| 13586 |  |  * @param filename  a file or URL | 
| 13587 |  |  * @param encoding  the document encoding (optional) | 
| 13588 |  |  * @param options  a combination of xmlParserOption | 
| 13589 |  |  * @returns the resulting document tree | 
| 13590 |  |  */ | 
| 13591 |  | xmlDoc * | 
| 13592 |  | xmlCtxtReadFile(xmlParserCtxt *ctxt, const char *filename, | 
| 13593 |  |                 const char *encoding, int options) | 
| 13594 | 0 | { | 
| 13595 | 0 |     xmlParserInputPtr input; | 
| 13596 |  | 
 | 
| 13597 | 0 |     if (ctxt == NULL) | 
| 13598 | 0 |         return(NULL); | 
| 13599 |  |  | 
| 13600 | 0 |     xmlCtxtReset(ctxt); | 
| 13601 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13602 |  | 
 | 
| 13603 | 0 |     input = xmlCtxtNewInputFromUrl(ctxt, filename, NULL, encoding, 0); | 
| 13604 | 0 |     if (input == NULL) | 
| 13605 | 0 |         return(NULL); | 
| 13606 |  |  | 
| 13607 | 0 |     return(xmlCtxtParseDocument(ctxt, input)); | 
| 13608 | 0 | } | 
| 13609 |  |  | 
| 13610 |  | /** | 
| 13611 |  |  * Parse an XML in-memory document and build a tree. The input buffer must | 
| 13612 |  |  * not contain a terminating null byte. | 
| 13613 |  |  * | 
| 13614 |  |  * `URL` is used as base to resolve external entities and for error | 
| 13615 |  |  * reporting. | 
| 13616 |  |  * | 
| 13617 |  |  * @param ctxt  an XML parser context | 
| 13618 |  |  * @param buffer  a pointer to a char array | 
| 13619 |  |  * @param size  the size of the array | 
| 13620 |  |  * @param URL  base URL (optional) | 
| 13621 |  |  * @param encoding  the document encoding (optional) | 
| 13622 |  |  * @param options  a combination of xmlParserOption | 
| 13623 |  |  * @returns the resulting document tree | 
| 13624 |  |  */ | 
| 13625 |  | xmlDoc * | 
| 13626 |  | xmlCtxtReadMemory(xmlParserCtxt *ctxt, const char *buffer, int size, | 
| 13627 |  |                   const char *URL, const char *encoding, int options) | 
| 13628 | 0 | { | 
| 13629 | 0 |     xmlParserInputPtr input; | 
| 13630 |  | 
 | 
| 13631 | 0 |     if ((ctxt == NULL) || (size < 0)) | 
| 13632 | 0 |         return(NULL); | 
| 13633 |  |  | 
| 13634 | 0 |     xmlCtxtReset(ctxt); | 
| 13635 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13636 |  | 
 | 
| 13637 | 0 |     input = xmlCtxtNewInputFromMemory(ctxt, URL, buffer, size, encoding, | 
| 13638 | 0 |                                       XML_INPUT_BUF_STATIC); | 
| 13639 | 0 |     if (input == NULL) | 
| 13640 | 0 |         return(NULL); | 
| 13641 |  |  | 
| 13642 | 0 |     return(xmlCtxtParseDocument(ctxt, input)); | 
| 13643 | 0 | } | 
| 13644 |  |  | 
| 13645 |  | /** | 
| 13646 |  |  * Parse an XML document from a file descriptor and build a tree. | 
| 13647 |  |  * | 
| 13648 |  |  * NOTE that the file descriptor will not be closed when the | 
| 13649 |  |  * context is freed or reset. | 
| 13650 |  |  * | 
| 13651 |  |  * `URL` is used as base to resolve external entities and for error | 
| 13652 |  |  * reporting. | 
| 13653 |  |  * | 
| 13654 |  |  * @param ctxt  an XML parser context | 
| 13655 |  |  * @param fd  an open file descriptor | 
| 13656 |  |  * @param URL  base URL (optional) | 
| 13657 |  |  * @param encoding  the document encoding (optional) | 
| 13658 |  |  * @param options  a combination of xmlParserOption | 
| 13659 |  |  * @returns the resulting document tree | 
| 13660 |  |  */ | 
| 13661 |  | xmlDoc * | 
| 13662 |  | xmlCtxtReadFd(xmlParserCtxt *ctxt, int fd, | 
| 13663 |  |               const char *URL, const char *encoding, int options) | 
| 13664 | 0 | { | 
| 13665 | 0 |     xmlParserInputPtr input; | 
| 13666 |  | 
 | 
| 13667 | 0 |     if (ctxt == NULL) | 
| 13668 | 0 |         return(NULL); | 
| 13669 |  |  | 
| 13670 | 0 |     xmlCtxtReset(ctxt); | 
| 13671 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13672 |  | 
 | 
| 13673 | 0 |     input = xmlCtxtNewInputFromFd(ctxt, URL, fd, encoding, 0); | 
| 13674 | 0 |     if (input == NULL) | 
| 13675 | 0 |         return(NULL); | 
| 13676 |  |  | 
| 13677 | 0 |     return(xmlCtxtParseDocument(ctxt, input)); | 
| 13678 | 0 | } | 
| 13679 |  |  | 
| 13680 |  | /** | 
| 13681 |  |  * Parse an XML document from I/O functions and source and build a tree. | 
| 13682 |  |  * This reuses the existing `ctxt` parser context | 
| 13683 |  |  * | 
| 13684 |  |  * `URL` is used as base to resolve external entities and for error | 
| 13685 |  |  * reporting. | 
| 13686 |  |  * | 
| 13687 |  |  * @param ctxt  an XML parser context | 
| 13688 |  |  * @param ioread  an I/O read function | 
| 13689 |  |  * @param ioclose  an I/O close function | 
| 13690 |  |  * @param ioctx  an I/O handler | 
| 13691 |  |  * @param URL  the base URL to use for the document | 
| 13692 |  |  * @param encoding  the document encoding, or NULL | 
| 13693 |  |  * @param options  a combination of xmlParserOption | 
| 13694 |  |  * @returns the resulting document tree | 
| 13695 |  |  */ | 
| 13696 |  | xmlDoc * | 
| 13697 |  | xmlCtxtReadIO(xmlParserCtxt *ctxt, xmlInputReadCallback ioread, | 
| 13698 |  |               xmlInputCloseCallback ioclose, void *ioctx, | 
| 13699 |  |         const char *URL, | 
| 13700 |  |               const char *encoding, int options) | 
| 13701 | 0 | { | 
| 13702 | 0 |     xmlParserInputPtr input; | 
| 13703 |  | 
 | 
| 13704 | 0 |     if (ctxt == NULL) | 
| 13705 | 0 |         return(NULL); | 
| 13706 |  |  | 
| 13707 | 0 |     xmlCtxtReset(ctxt); | 
| 13708 | 0 |     xmlCtxtUseOptions(ctxt, options); | 
| 13709 |  | 
 | 
| 13710 | 0 |     input = xmlCtxtNewInputFromIO(ctxt, URL, ioread, ioclose, ioctx, | 
| 13711 | 0 |                                   encoding, 0); | 
| 13712 | 0 |     if (input == NULL) | 
| 13713 | 0 |         return(NULL); | 
| 13714 |  |  | 
| 13715 | 0 |     return(xmlCtxtParseDocument(ctxt, input)); | 
| 13716 | 0 | } | 
| 13717 |  |  |