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@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
xmlGenericErrorDefaultFunc(void * ctx ATTRIBUTE_UNUSED,const char * msg,...)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
initGenericErrorDefaultFunc(xmlGenericErrorFunc * handler)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
xmlSetGenericErrorFunc(void * ctx,xmlGenericErrorFunc handler)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
xmlSetStructuredErrorFunc(void * ctx,xmlStructuredErrorFunc handler)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
xmlParserPrintFileInfo(xmlParserInputPtr input)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
xmlParserPrintFileContextInternal(xmlParserInputPtr input,xmlGenericErrorFunc channel,void * data)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
xmlParserPrintFileContext(xmlParserInputPtr input)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
xmlReportError(xmlErrorPtr err,xmlParserCtxtPtr ctxt,const char * str,xmlGenericErrorFunc channel,void * data)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
__xmlRaiseError(xmlStructuredErrorFunc schannel,xmlGenericErrorFunc channel,void * data,void * ctx,void * nod,int domain,int code,xmlErrorLevel level,const char * file,int line,const char * str1,const char * str2,const char * str3,int int1,int col,const char * msg,...)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
__xmlSimpleError(int domain,int code,xmlNodePtr node,const char * msg,const char * extra)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
xmlParserError(void * ctx,const char * msg,...)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
xmlParserWarning(void * ctx,const char * msg,...)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
xmlParserValidityError(void * ctx,const char * msg,...)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
xmlParserValidityWarning(void * ctx,const char * msg,...)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
xmlGetLastError(void)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
xmlResetError(xmlErrorPtr err)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
xmlResetLastError(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
xmlCtxtGetLastError(void * ctx)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
xmlCtxtResetLastError(void * ctx)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
xmlCopyError(xmlErrorPtr from,xmlErrorPtr to)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