Home | History | Annotate | Download | only in libxml2
      1 /*
      2  * error.c: module displaying/handling XML parser errors
      3  *
      4  * See Copyright for the status of this software.
      5  *
      6  * Daniel Veillard <daniel (at) veillard.com>
      7  */
      8 
      9 #define IN_LIBXML
     10 #include "libxml.h"
     11 
     12 #include <string.h>
     13 #include <stdarg.h>
     14 #include <libxml/parser.h>
     15 #include <libxml/xmlerror.h>
     16 #include <libxml/xmlmemory.h>
     17 #include <libxml/globals.h>
     18 
     19 void XMLCDECL xmlGenericErrorDefaultFunc	(void *ctx ATTRIBUTE_UNUSED,
     20 				 const char *msg,
     21 				 ...);
     22 
     23 #define XML_GET_VAR_STR(msg, str) {				\
     24     int       size, prev_size = -1;				\
     25     int       chars;						\
     26     char      *larger;						\
     27     va_list   ap;						\
     28 								\
     29     str = (char *) xmlMalloc(150);				\
     30     if (str != NULL) {						\
     31 								\
     32     size = 150;							\
     33 								\
     34     while (size < 64000) {					\
     35 	va_start(ap, msg);					\
     36 	chars = vsnprintf(str, size, msg, ap);			\
     37 	va_end(ap);						\
     38 	if ((chars > -1) && (chars < size)) {			\
     39 	    if (prev_size == chars) {				\
     40 		break;						\
     41 	    } else {						\
     42 		prev_size = chars;				\
     43 	    }							\
     44 	}							\
     45 	if (chars > -1)						\
     46 	    size += chars + 1;					\
     47 	else							\
     48 	    size += 100;					\
     49 	if ((larger = (char *) xmlRealloc(str, size)) == NULL) {\
     50 	    break;						\
     51 	}							\
     52 	str = larger;						\
     53     }}								\
     54 }
     55 
     56 /************************************************************************
     57  *									*
     58  *			Handling of out of context errors		*
     59  *									*
     60  ************************************************************************/
     61 
     62 /**
     63  * xmlGenericErrorDefaultFunc:
     64  * @ctx:  an error context
     65  * @msg:  the message to display/transmit
     66  * @...:  extra parameters for the message display
     67  *
     68  * Default handler for out of context error messages.
     69  */
     70 void XMLCDECL
     71 xmlGenericErrorDefaultFunc(void *ctx ATTRIBUTE_UNUSED, const char *msg, ...) {
     72     va_list args;
     73 
     74     if (xmlGenericErrorContext == NULL)
     75 	xmlGenericErrorContext = (void *) stderr;
     76 
     77     va_start(args, msg);
     78     vfprintf((FILE *)xmlGenericErrorContext, msg, args);
     79     va_end(args);
     80 }
     81 
     82 /**
     83  * initGenericErrorDefaultFunc:
     84  * @handler:  the handler
     85  *
     86  * Set or reset (if NULL) the default handler for generic errors
     87  * to the builtin error function.
     88  */
     89 void
     90 initGenericErrorDefaultFunc(xmlGenericErrorFunc * handler)
     91 {
     92     if (handler == NULL)
     93         xmlGenericError = xmlGenericErrorDefaultFunc;
     94     else
     95         xmlGenericError = (*handler);
     96 }
     97 
     98 /**
     99  * xmlSetGenericErrorFunc:
    100  * @ctx:  the new error handling context
    101  * @handler:  the new handler function
    102  *
    103  * Function to reset the handler and the error context for out of
    104  * context error messages.
    105  * This simply means that @handler will be called for subsequent
    106  * error messages while not parsing nor validating. And @ctx will
    107  * be passed as first argument to @handler
    108  * One can simply force messages to be emitted to another FILE * than
    109  * stderr by setting @ctx to this file handle and @handler to NULL.
    110  * For multi-threaded applications, this must be set separately for each thread.
    111  */
    112 void
    113 xmlSetGenericErrorFunc(void *ctx, xmlGenericErrorFunc handler) {
    114     xmlGenericErrorContext = ctx;
    115     if (handler != NULL)
    116 	xmlGenericError = handler;
    117     else
    118 	xmlGenericError = xmlGenericErrorDefaultFunc;
    119 }
    120 
    121 /**
    122  * xmlSetStructuredErrorFunc:
    123  * @ctx:  the new error handling context
    124  * @handler:  the new handler function
    125  *
    126  * Function to reset the handler and the error context for out of
    127  * context structured error messages.
    128  * This simply means that @handler will be called for subsequent
    129  * error messages while not parsing nor validating. And @ctx will
    130  * be passed as first argument to @handler
    131  * For multi-threaded applications, this must be set separately for each thread.
    132  */
    133 void
    134 xmlSetStructuredErrorFunc(void *ctx, xmlStructuredErrorFunc handler) {
    135     xmlStructuredErrorContext = ctx;
    136     xmlStructuredError = handler;
    137 }
    138 
    139 /************************************************************************
    140  *									*
    141  *			Handling of parsing errors			*
    142  *									*
    143  ************************************************************************/
    144 
    145 /**
    146  * xmlParserPrintFileInfo:
    147  * @input:  an xmlParserInputPtr input
    148  *
    149  * Displays the associated file and line informations for the current input
    150  */
    151 
    152 void
    153 xmlParserPrintFileInfo(xmlParserInputPtr input) {
    154     if (input != NULL) {
    155 	if (input->filename)
    156 	    xmlGenericError(xmlGenericErrorContext,
    157 		    "%s:%d: ", input->filename,
    158 		    input->line);
    159 	else
    160 	    xmlGenericError(xmlGenericErrorContext,
    161 		    "Entity: line %d: ", input->line);
    162     }
    163 }
    164 
    165 /**
    166  * xmlParserPrintFileContext:
    167  * @input:  an xmlParserInputPtr input
    168  *
    169  * Displays current context within the input content for error tracking
    170  */
    171 
    172 static void
    173 xmlParserPrintFileContextInternal(xmlParserInputPtr input ,
    174 		xmlGenericErrorFunc channel, void *data ) {
    175     const xmlChar *cur, *base;
    176     unsigned int n, col;	/* GCC warns if signed, because compared with sizeof() */
    177     xmlChar  content[81]; /* space for 80 chars + line terminator */
    178     xmlChar *ctnt;
    179 
    180     if (input == NULL) return;
    181     cur = input->cur;
    182     base = input->base;
    183     /* skip backwards over any end-of-lines */
    184     while ((cur > base) && ((*(cur) == '\n') || (*(cur) == '\r'))) {
    185 	cur--;
    186     }
    187     n = 0;
    188     /* search backwards for beginning-of-line (to max buff size) */
    189     while ((n++ < (sizeof(content)-1)) && (cur > base) &&
    190 	   (*(cur) != '\n') && (*(cur) != '\r'))
    191         cur--;
    192     if ((*(cur) == '\n') || (*(cur) == '\r')) cur++;
    193     /* calculate the error position in terms of the current position */
    194     col = input->cur - cur;
    195     /* search forward for end-of-line (to max buff size) */
    196     n = 0;
    197     ctnt = content;
    198     /* copy selected text to our buffer */
    199     while ((*cur != 0) && (*(cur) != '\n') &&
    200 	   (*(cur) != '\r') && (n < sizeof(content)-1)) {
    201 		*ctnt++ = *cur++;
    202 	n++;
    203     }
    204     *ctnt = 0;
    205     /* print out the selected text */
    206     channel(data ,"%s\n", content);
    207     /* create blank line with problem pointer */
    208     n = 0;
    209     ctnt = content;
    210     /* (leave buffer space for pointer + line terminator) */
    211     while ((n<col) && (n++ < sizeof(content)-2) && (*ctnt != 0)) {
    212 	if (*(ctnt) != '\t')
    213 	    *(ctnt) = ' ';
    214 	ctnt++;
    215     }
    216     *ctnt++ = '^';
    217     *ctnt = 0;
    218     channel(data ,"%s\n", content);
    219 }
    220 
    221 /**
    222  * xmlParserPrintFileContext:
    223  * @input:  an xmlParserInputPtr input
    224  *
    225  * Displays current context within the input content for error tracking
    226  */
    227 void
    228 xmlParserPrintFileContext(xmlParserInputPtr input) {
    229    xmlParserPrintFileContextInternal(input, xmlGenericError,
    230                                      xmlGenericErrorContext);
    231 }
    232 
    233 /**
    234  * xmlReportError:
    235  * @err: the error
    236  * @ctx: the parser context or NULL
    237  * @str: the formatted error message
    238  *
    239  * Report an erro with its context, replace the 4 old error/warning
    240  * routines.
    241  */
    242 static void
    243 xmlReportError(xmlErrorPtr err, xmlParserCtxtPtr ctxt, const char *str,
    244                xmlGenericErrorFunc channel, void *data)
    245 {
    246     char *file = NULL;
    247     int line = 0;
    248     int code = -1;
    249     int domain;
    250     const xmlChar *name = NULL;
    251     xmlNodePtr node;
    252     xmlErrorLevel level;
    253     xmlParserInputPtr input = NULL;
    254     xmlParserInputPtr cur = NULL;
    255 
    256     if (err == NULL)
    257         return;
    258 
    259     if (channel == NULL) {
    260 	channel = xmlGenericError;
    261 	data = xmlGenericErrorContext;
    262     }
    263     file = err->file;
    264     line = err->line;
    265     code = err->code;
    266     domain = err->domain;
    267     level = err->level;
    268     node = err->node;
    269 
    270     if (code == XML_ERR_OK)
    271         return;
    272 
    273     if ((node != NULL) && (node->type == XML_ELEMENT_NODE))
    274         name = node->name;
    275 
    276     /*
    277      * Maintain the compatibility with the legacy error handling
    278      */
    279     if (ctxt != NULL) {
    280         input = ctxt->input;
    281         if ((input != NULL) && (input->filename == NULL) &&
    282             (ctxt->inputNr > 1)) {
    283             cur = input;
    284             input = ctxt->inputTab[ctxt->inputNr - 2];
    285         }
    286         if (input != NULL) {
    287             if (input->filename)
    288                 channel(data, "%s:%d: ", input->filename, input->line);
    289             else if ((line != 0) && (domain == XML_FROM_PARSER))
    290                 channel(data, "Entity: line %d: ", input->line);
    291         }
    292     } else {
    293         if (file != NULL)
    294             channel(data, "%s:%d: ", file, line);
    295         else if ((line != 0) &&
    296 	         ((domain == XML_FROM_PARSER) || (domain == XML_FROM_SCHEMASV)||
    297 		  (domain == XML_FROM_SCHEMASP)||(domain == XML_FROM_DTD) ||
    298 		  (domain == XML_FROM_RELAXNGP)||(domain == XML_FROM_RELAXNGV)))
    299             channel(data, "Entity: line %d: ", line);
    300     }
    301     if (name != NULL) {
    302         channel(data, "element %s: ", name);
    303     }
    304     switch (domain) {
    305         case XML_FROM_PARSER:
    306             channel(data, "parser ");
    307             break;
    308         case XML_FROM_NAMESPACE:
    309             channel(data, "namespace ");
    310             break;
    311         case XML_FROM_DTD:
    312         case XML_FROM_VALID:
    313             channel(data, "validity ");
    314             break;
    315         case XML_FROM_HTML:
    316             channel(data, "HTML parser ");
    317             break;
    318         case XML_FROM_MEMORY:
    319             channel(data, "memory ");
    320             break;
    321         case XML_FROM_OUTPUT:
    322             channel(data, "output ");
    323             break;
    324         case XML_FROM_IO:
    325             channel(data, "I/O ");
    326             break;
    327         case XML_FROM_XINCLUDE:
    328             channel(data, "XInclude ");
    329             break;
    330         case XML_FROM_XPATH:
    331             channel(data, "XPath ");
    332             break;
    333         case XML_FROM_XPOINTER:
    334             channel(data, "parser ");
    335             break;
    336         case XML_FROM_REGEXP:
    337             channel(data, "regexp ");
    338             break;
    339         case XML_FROM_MODULE:
    340             channel(data, "module ");
    341             break;
    342         case XML_FROM_SCHEMASV:
    343             channel(data, "Schemas validity ");
    344             break;
    345         case XML_FROM_SCHEMASP:
    346             channel(data, "Schemas parser ");
    347             break;
    348         case XML_FROM_RELAXNGP:
    349             channel(data, "Relax-NG parser ");
    350             break;
    351         case XML_FROM_RELAXNGV:
    352             channel(data, "Relax-NG validity ");
    353             break;
    354         case XML_FROM_CATALOG:
    355             channel(data, "Catalog ");
    356             break;
    357         case XML_FROM_C14N:
    358             channel(data, "C14N ");
    359             break;
    360         case XML_FROM_XSLT:
    361             channel(data, "XSLT ");
    362             break;
    363         case XML_FROM_I18N:
    364             channel(data, "encoding ");
    365             break;
    366         case XML_FROM_SCHEMATRONV:
    367             channel(data, "schematron ");
    368             break;
    369         case XML_FROM_BUFFER:
    370             channel(data, "internal buffer ");
    371             break;
    372         case XML_FROM_URI:
    373             channel(data, "URI ");
    374             break;
    375         default:
    376             break;
    377     }
    378     switch (level) {
    379         case XML_ERR_NONE:
    380             channel(data, ": ");
    381             break;
    382         case XML_ERR_WARNING:
    383             channel(data, "warning : ");
    384             break;
    385         case XML_ERR_ERROR:
    386             channel(data, "error : ");
    387             break;
    388         case XML_ERR_FATAL:
    389             channel(data, "error : ");
    390             break;
    391     }
    392     if (str != NULL) {
    393         int len;
    394 	len = xmlStrlen((const xmlChar *)str);
    395 	if ((len > 0) && (str[len - 1] != '\n'))
    396 	    channel(data, "%s\n", str);
    397 	else
    398 	    channel(data, "%s", str);
    399     } else {
    400         channel(data, "%s\n", "out of memory error");
    401     }
    402 
    403     if (ctxt != NULL) {
    404         xmlParserPrintFileContextInternal(input, channel, data);
    405         if (cur != NULL) {
    406             if (cur->filename)
    407                 channel(data, "%s:%d: \n", cur->filename, cur->line);
    408             else if ((line != 0) && (domain == XML_FROM_PARSER))
    409                 channel(data, "Entity: line %d: \n", cur->line);
    410             xmlParserPrintFileContextInternal(cur, channel, data);
    411         }
    412     }
    413     if ((domain == XML_FROM_XPATH) && (err->str1 != NULL) &&
    414         (err->int1 < 100) &&
    415 	(err->int1 < xmlStrlen((const xmlChar *)err->str1))) {
    416 	xmlChar buf[150];
    417 	int i;
    418 
    419 	channel(data, "%s\n", err->str1);
    420 	for (i=0;i < err->int1;i++)
    421 	     buf[i] = ' ';
    422 	buf[i++] = '^';
    423 	buf[i] = 0;
    424 	channel(data, "%s\n", buf);
    425     }
    426 }
    427 
    428 /**
    429  * __xmlRaiseError:
    430  * @schannel: the structured callback channel
    431  * @channel: the old callback channel
    432  * @data: the callback data
    433  * @ctx: the parser context or NULL
    434  * @ctx: the parser context or NULL
    435  * @domain: the domain for the error
    436  * @code: the code for the error
    437  * @level: the xmlErrorLevel for the error
    438  * @file: the file source of the error (or NULL)
    439  * @line: the line of the error or 0 if N/A
    440  * @str1: extra string info
    441  * @str2: extra string info
    442  * @str3: extra string info
    443  * @int1: extra int info
    444  * @col: column number of the error or 0 if N/A
    445  * @msg:  the message to display/transmit
    446  * @...:  extra parameters for the message display
    447  *
    448  * Update the appropriate global or contextual error structure,
    449  * then forward the error message down the parser or generic
    450  * error callback handler
    451  */
    452 void XMLCDECL
    453 __xmlRaiseError(xmlStructuredErrorFunc schannel,
    454               xmlGenericErrorFunc channel, void *data, void *ctx,
    455               void *nod, int domain, int code, xmlErrorLevel level,
    456               const char *file, int line, const char *str1,
    457               const char *str2, const char *str3, int int1, int col,
    458 	      const char *msg, ...)
    459 {
    460     xmlParserCtxtPtr ctxt = NULL;
    461     xmlNodePtr node = (xmlNodePtr) nod;
    462     char *str = NULL;
    463     xmlParserInputPtr input = NULL;
    464     xmlErrorPtr to = &xmlLastError;
    465     xmlNodePtr baseptr = NULL;
    466 
    467     if (code == XML_ERR_OK)
    468         return;
    469     if ((xmlGetWarningsDefaultValue == 0) && (level == XML_ERR_WARNING))
    470         return;
    471     if ((domain == XML_FROM_PARSER) || (domain == XML_FROM_HTML) ||
    472         (domain == XML_FROM_DTD) || (domain == XML_FROM_NAMESPACE) ||
    473 	(domain == XML_FROM_IO) || (domain == XML_FROM_VALID)) {
    474 	ctxt = (xmlParserCtxtPtr) ctx;
    475 	if ((schannel == NULL) && (ctxt != NULL) && (ctxt->sax != NULL) &&
    476 	    (ctxt->sax->initialized == XML_SAX2_MAGIC) &&
    477 	    (ctxt->sax->serror != NULL)) {
    478 	    schannel = ctxt->sax->serror;
    479 	    data = ctxt->userData;
    480 	}
    481     }
    482     /*
    483      * Check if structured error handler set
    484      */
    485     if (schannel == NULL) {
    486 	schannel = xmlStructuredError;
    487 	/*
    488 	 * if user has defined handler, change data ptr to user's choice
    489 	 */
    490 	if (schannel != NULL)
    491 	    data = xmlStructuredErrorContext;
    492     }
    493     /*
    494      * Formatting the message
    495      */
    496     if (msg == NULL) {
    497         str = (char *) xmlStrdup(BAD_CAST "No error message provided");
    498     } else {
    499         XML_GET_VAR_STR(msg, str);
    500     }
    501 
    502     /*
    503      * specific processing if a parser context is provided
    504      */
    505     if (ctxt != NULL) {
    506         if (file == NULL) {
    507             input = ctxt->input;
    508             if ((input != NULL) && (input->filename == NULL) &&
    509                 (ctxt->inputNr > 1)) {
    510                 input = ctxt->inputTab[ctxt->inputNr - 2];
    511             }
    512             if (input != NULL) {
    513                 file = input->filename;
    514                 line = input->line;
    515                 col = input->col;
    516             }
    517         }
    518         to = &ctxt->lastError;
    519     } else if ((node != NULL) && (file == NULL)) {
    520 	int i;
    521 
    522 	if ((node->doc != NULL) && (node->doc->URL != NULL)) {
    523 	    baseptr = node;
    524 /*	    file = (const char *) node->doc->URL; */
    525 	}
    526 	for (i = 0;
    527 	     ((i < 10) && (node != NULL) && (node->type != XML_ELEMENT_NODE));
    528 	     i++)
    529 	     node = node->parent;
    530         if ((baseptr == NULL) && (node != NULL) &&
    531 	    (node->doc != NULL) && (node->doc->URL != NULL))
    532 	    baseptr = node;
    533 
    534 	if ((node != NULL) && (node->type == XML_ELEMENT_NODE))
    535 	    line = node->line;
    536 	if ((line == 0) || (line == 65535))
    537 	    line = xmlGetLineNo(node);
    538     }
    539 
    540     /*
    541      * Save the information about the error
    542      */
    543     xmlResetError(to);
    544     to->domain = domain;
    545     to->code = code;
    546     to->message = str;
    547     to->level = level;
    548     if (file != NULL)
    549         to->file = (char *) xmlStrdup((const xmlChar *) file);
    550     else if (baseptr != NULL) {
    551 #ifdef LIBXML_XINCLUDE_ENABLED
    552 	/*
    553 	 * We check if the error is within an XInclude section and,
    554 	 * if so, attempt to print out the href of the XInclude instead
    555 	 * of the usual "base" (doc->URL) for the node (bug 152623).
    556 	 */
    557         xmlNodePtr prev = baseptr;
    558 	int inclcount = 0;
    559 	while (prev != NULL) {
    560 	    if (prev->prev == NULL)
    561 	        prev = prev->parent;
    562 	    else {
    563 	        prev = prev->prev;
    564 		if (prev->type == XML_XINCLUDE_START) {
    565 		    if (--inclcount < 0)
    566 		        break;
    567 		} else if (prev->type == XML_XINCLUDE_END)
    568 		    inclcount++;
    569 	    }
    570 	}
    571 	if (prev != NULL) {
    572 	    if (prev->type == XML_XINCLUDE_START) {
    573 		prev->type = XML_ELEMENT_NODE;
    574 		to->file = (char *) xmlGetProp(prev, BAD_CAST "href");
    575 		prev->type = XML_XINCLUDE_START;
    576 	    } else {
    577 		to->file = (char *) xmlGetProp(prev, BAD_CAST "href");
    578 	    }
    579 	} else
    580 #endif
    581 	    to->file = (char *) xmlStrdup(baseptr->doc->URL);
    582 	if ((to->file == NULL) && (node != NULL) && (node->doc != NULL)) {
    583 	    to->file = (char *) xmlStrdup(node->doc->URL);
    584 	}
    585     }
    586     to->line = line;
    587     if (str1 != NULL)
    588         to->str1 = (char *) xmlStrdup((const xmlChar *) str1);
    589     if (str2 != NULL)
    590         to->str2 = (char *) xmlStrdup((const xmlChar *) str2);
    591     if (str3 != NULL)
    592         to->str3 = (char *) xmlStrdup((const xmlChar *) str3);
    593     to->int1 = int1;
    594     to->int2 = col;
    595     to->node = node;
    596     to->ctxt = ctx;
    597 
    598     if (to != &xmlLastError)
    599         xmlCopyError(to,&xmlLastError);
    600 
    601     if (schannel != NULL) {
    602 	schannel(data, to);
    603 	return;
    604     }
    605 
    606     /*
    607      * Find the callback channel if channel param is NULL
    608      */
    609     if ((ctxt != NULL) && (channel == NULL) &&
    610         (xmlStructuredError == NULL) && (ctxt->sax != NULL)) {
    611         if (level == XML_ERR_WARNING)
    612 	    channel = ctxt->sax->warning;
    613         else
    614 	    channel = ctxt->sax->error;
    615 	data = ctxt->userData;
    616     } else if (channel == NULL) {
    617 	channel = xmlGenericError;
    618 	if (ctxt != NULL) {
    619 	    data = ctxt;
    620 	} else {
    621 	    data = xmlGenericErrorContext;
    622 	}
    623     }
    624     if (channel == NULL)
    625         return;
    626 
    627     if ((channel == xmlParserError) ||
    628         (channel == xmlParserWarning) ||
    629 	(channel == xmlParserValidityError) ||
    630 	(channel == xmlParserValidityWarning))
    631 	xmlReportError(to, ctxt, str, NULL, NULL);
    632     else if ((channel == (xmlGenericErrorFunc) fprintf) ||
    633              (channel == xmlGenericErrorDefaultFunc))
    634 	xmlReportError(to, ctxt, str, channel, data);
    635     else
    636 	channel(data, "%s", str);
    637 }
    638 
    639 /**
    640  * __xmlSimpleError:
    641  * @domain: where the error comes from
    642  * @code: the error code
    643  * @node: the context node
    644  * @extra:  extra informations
    645  *
    646  * Handle an out of memory condition
    647  */
    648 void
    649 __xmlSimpleError(int domain, int code, xmlNodePtr node,
    650                  const char *msg, const char *extra)
    651 {
    652 
    653     if (code == XML_ERR_NO_MEMORY) {
    654 	if (extra)
    655 	    __xmlRaiseError(NULL, NULL, NULL, NULL, node, domain,
    656 			    XML_ERR_NO_MEMORY, XML_ERR_FATAL, NULL, 0, extra,
    657 			    NULL, NULL, 0, 0,
    658 			    "Memory allocation failed : %s\n", extra);
    659 	else
    660 	    __xmlRaiseError(NULL, NULL, NULL, NULL, node, domain,
    661 			    XML_ERR_NO_MEMORY, XML_ERR_FATAL, NULL, 0, NULL,
    662 			    NULL, NULL, 0, 0, "Memory allocation failed\n");
    663     } else {
    664 	__xmlRaiseError(NULL, NULL, NULL, NULL, node, domain,
    665 			code, XML_ERR_ERROR, NULL, 0, extra,
    666 			NULL, NULL, 0, 0, msg, extra);
    667     }
    668 }
    669 /**
    670  * xmlParserError:
    671  * @ctx:  an XML parser context
    672  * @msg:  the message to display/transmit
    673  * @...:  extra parameters for the message display
    674  *
    675  * Display and format an error messages, gives file, line, position and
    676  * extra parameters.
    677  */
    678 void XMLCDECL
    679 xmlParserError(void *ctx, const char *msg, ...)
    680 {
    681     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
    682     xmlParserInputPtr input = NULL;
    683     xmlParserInputPtr cur = NULL;
    684     char * str;
    685 
    686     if (ctxt != NULL) {
    687 	input = ctxt->input;
    688 	if ((input != NULL) && (input->filename == NULL) &&
    689 	    (ctxt->inputNr > 1)) {
    690 	    cur = input;
    691 	    input = ctxt->inputTab[ctxt->inputNr - 2];
    692 	}
    693 	xmlParserPrintFileInfo(input);
    694     }
    695 
    696     xmlGenericError(xmlGenericErrorContext, "error: ");
    697     XML_GET_VAR_STR(msg, str);
    698     xmlGenericError(xmlGenericErrorContext, "%s", str);
    699     if (str != NULL)
    700 	xmlFree(str);
    701 
    702     if (ctxt != NULL) {
    703 	xmlParserPrintFileContext(input);
    704 	if (cur != NULL) {
    705 	    xmlParserPrintFileInfo(cur);
    706 	    xmlGenericError(xmlGenericErrorContext, "\n");
    707 	    xmlParserPrintFileContext(cur);
    708 	}
    709     }
    710 }
    711 
    712 /**
    713  * xmlParserWarning:
    714  * @ctx:  an XML parser context
    715  * @msg:  the message to display/transmit
    716  * @...:  extra parameters for the message display
    717  *
    718  * Display and format a warning messages, gives file, line, position and
    719  * extra parameters.
    720  */
    721 void XMLCDECL
    722 xmlParserWarning(void *ctx, const char *msg, ...)
    723 {
    724     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
    725     xmlParserInputPtr input = NULL;
    726     xmlParserInputPtr cur = NULL;
    727     char * str;
    728 
    729     if (ctxt != NULL) {
    730 	input = ctxt->input;
    731 	if ((input != NULL) && (input->filename == NULL) &&
    732 	    (ctxt->inputNr > 1)) {
    733 	    cur = input;
    734 	    input = ctxt->inputTab[ctxt->inputNr - 2];
    735 	}
    736 	xmlParserPrintFileInfo(input);
    737     }
    738 
    739     xmlGenericError(xmlGenericErrorContext, "warning: ");
    740     XML_GET_VAR_STR(msg, str);
    741     xmlGenericError(xmlGenericErrorContext, "%s", str);
    742     if (str != NULL)
    743 	xmlFree(str);
    744 
    745     if (ctxt != NULL) {
    746 	xmlParserPrintFileContext(input);
    747 	if (cur != NULL) {
    748 	    xmlParserPrintFileInfo(cur);
    749 	    xmlGenericError(xmlGenericErrorContext, "\n");
    750 	    xmlParserPrintFileContext(cur);
    751 	}
    752     }
    753 }
    754 
    755 /************************************************************************
    756  *									*
    757  *			Handling of validation errors			*
    758  *									*
    759  ************************************************************************/
    760 
    761 /**
    762  * xmlParserValidityError:
    763  * @ctx:  an XML parser context
    764  * @msg:  the message to display/transmit
    765  * @...:  extra parameters for the message display
    766  *
    767  * Display and format an validity error messages, gives file,
    768  * line, position and extra parameters.
    769  */
    770 void XMLCDECL
    771 xmlParserValidityError(void *ctx, const char *msg, ...)
    772 {
    773     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
    774     xmlParserInputPtr input = NULL;
    775     char * str;
    776     int len = xmlStrlen((const xmlChar *) msg);
    777     static int had_info = 0;
    778 
    779     if ((len > 1) && (msg[len - 2] != ':')) {
    780 	if (ctxt != NULL) {
    781 	    input = ctxt->input;
    782 	    if ((input->filename == NULL) && (ctxt->inputNr > 1))
    783 		input = ctxt->inputTab[ctxt->inputNr - 2];
    784 
    785 	    if (had_info == 0) {
    786 		xmlParserPrintFileInfo(input);
    787 	    }
    788 	}
    789 	xmlGenericError(xmlGenericErrorContext, "validity error: ");
    790 	had_info = 0;
    791     } else {
    792 	had_info = 1;
    793     }
    794 
    795     XML_GET_VAR_STR(msg, str);
    796     xmlGenericError(xmlGenericErrorContext, "%s", str);
    797     if (str != NULL)
    798 	xmlFree(str);
    799 
    800     if ((ctxt != NULL) && (input != NULL)) {
    801 	xmlParserPrintFileContext(input);
    802     }
    803 }
    804 
    805 /**
    806  * xmlParserValidityWarning:
    807  * @ctx:  an XML parser context
    808  * @msg:  the message to display/transmit
    809  * @...:  extra parameters for the message display
    810  *
    811  * Display and format a validity warning messages, gives file, line,
    812  * position and extra parameters.
    813  */
    814 void XMLCDECL
    815 xmlParserValidityWarning(void *ctx, const char *msg, ...)
    816 {
    817     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
    818     xmlParserInputPtr input = NULL;
    819     char * str;
    820     int len = xmlStrlen((const xmlChar *) msg);
    821 
    822     if ((ctxt != NULL) && (len != 0) && (msg[len - 1] != ':')) {
    823 	input = ctxt->input;
    824 	if ((input->filename == NULL) && (ctxt->inputNr > 1))
    825 	    input = ctxt->inputTab[ctxt->inputNr - 2];
    826 
    827 	xmlParserPrintFileInfo(input);
    828     }
    829 
    830     xmlGenericError(xmlGenericErrorContext, "validity warning: ");
    831     XML_GET_VAR_STR(msg, str);
    832     xmlGenericError(xmlGenericErrorContext, "%s", str);
    833     if (str != NULL)
    834 	xmlFree(str);
    835 
    836     if (ctxt != NULL) {
    837 	xmlParserPrintFileContext(input);
    838     }
    839 }
    840 
    841 
    842 /************************************************************************
    843  *									*
    844  *			Extended Error Handling				*
    845  *									*
    846  ************************************************************************/
    847 
    848 /**
    849  * xmlGetLastError:
    850  *
    851  * Get the last global error registered. This is per thread if compiled
    852  * with thread support.
    853  *
    854  * Returns NULL if no error occured or a pointer to the error
    855  */
    856 xmlErrorPtr
    857 xmlGetLastError(void)
    858 {
    859     if (xmlLastError.code == XML_ERR_OK)
    860         return (NULL);
    861     return (&xmlLastError);
    862 }
    863 
    864 /**
    865  * xmlResetError:
    866  * @err: pointer to the error.
    867  *
    868  * Cleanup the error.
    869  */
    870 void
    871 xmlResetError(xmlErrorPtr err)
    872 {
    873     if (err == NULL)
    874         return;
    875     if (err->code == XML_ERR_OK)
    876         return;
    877     if (err->message != NULL)
    878         xmlFree(err->message);
    879     if (err->file != NULL)
    880         xmlFree(err->file);
    881     if (err->str1 != NULL)
    882         xmlFree(err->str1);
    883     if (err->str2 != NULL)
    884         xmlFree(err->str2);
    885     if (err->str3 != NULL)
    886         xmlFree(err->str3);
    887     memset(err, 0, sizeof(xmlError));
    888     err->code = XML_ERR_OK;
    889 }
    890 
    891 /**
    892  * xmlResetLastError:
    893  *
    894  * Cleanup the last global error registered. For parsing error
    895  * this does not change the well-formedness result.
    896  */
    897 void
    898 xmlResetLastError(void)
    899 {
    900     if (xmlLastError.code == XML_ERR_OK)
    901         return;
    902     xmlResetError(&xmlLastError);
    903 }
    904 
    905 /**
    906  * xmlCtxtGetLastError:
    907  * @ctx:  an XML parser context
    908  *
    909  * Get the last parsing error registered.
    910  *
    911  * Returns NULL if no error occured or a pointer to the error
    912  */
    913 xmlErrorPtr
    914 xmlCtxtGetLastError(void *ctx)
    915 {
    916     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
    917 
    918     if (ctxt == NULL)
    919         return (NULL);
    920     if (ctxt->lastError.code == XML_ERR_OK)
    921         return (NULL);
    922     return (&ctxt->lastError);
    923 }
    924 
    925 /**
    926  * xmlCtxtResetLastError:
    927  * @ctx:  an XML parser context
    928  *
    929  * Cleanup the last global error registered. For parsing error
    930  * this does not change the well-formedness result.
    931  */
    932 void
    933 xmlCtxtResetLastError(void *ctx)
    934 {
    935     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
    936 
    937     if (ctxt == NULL)
    938         return;
    939     ctxt->errNo = XML_ERR_OK;
    940     if (ctxt->lastError.code == XML_ERR_OK)
    941         return;
    942     xmlResetError(&ctxt->lastError);
    943 }
    944 
    945 /**
    946  * xmlCopyError:
    947  * @from:  a source error
    948  * @to:  a target error
    949  *
    950  * Save the original error to the new place.
    951  *
    952  * Returns 0 in case of success and -1 in case of error.
    953  */
    954 int
    955 xmlCopyError(xmlErrorPtr from, xmlErrorPtr to) {
    956     char *message, *file, *str1, *str2, *str3;
    957 
    958     if ((from == NULL) || (to == NULL))
    959         return(-1);
    960 
    961     message = (char *) xmlStrdup((xmlChar *) from->message);
    962     file = (char *) xmlStrdup ((xmlChar *) from->file);
    963     str1 = (char *) xmlStrdup ((xmlChar *) from->str1);
    964     str2 = (char *) xmlStrdup ((xmlChar *) from->str2);
    965     str3 = (char *) xmlStrdup ((xmlChar *) from->str3);
    966 
    967     if (to->message != NULL)
    968         xmlFree(to->message);
    969     if (to->file != NULL)
    970         xmlFree(to->file);
    971     if (to->str1 != NULL)
    972         xmlFree(to->str1);
    973     if (to->str2 != NULL)
    974         xmlFree(to->str2);
    975     if (to->str3 != NULL)
    976         xmlFree(to->str3);
    977     to->domain = from->domain;
    978     to->code = from->code;
    979     to->level = from->level;
    980     to->line = from->line;
    981     to->node = from->node;
    982     to->int1 = from->int1;
    983     to->int2 = from->int2;
    984     to->node = from->node;
    985     to->ctxt = from->ctxt;
    986     to->message = message;
    987     to->file = file;
    988     to->str1 = str1;
    989     to->str2 = str2;
    990     to->str3 = str3;
    991 
    992     return 0;
    993 }
    994 
    995 #define bottom_error
    996 #include "elfgcchack.h"
    997