blob: ce63071d2ed236fb2f227ae0a721dc5447436847 [file] [log] [blame]
Owen Taylor3473f882001-02-23 17:55:21 +00001/*
2 * error.c: module displaying/handling XML parser errors
3 *
4 * See Copyright for the status of this software.
5 *
Daniel Veillardc5d64342001-06-24 12:13:24 +00006 * Daniel Veillard <daniel@veillard.com>
Owen Taylor3473f882001-02-23 17:55:21 +00007 */
8
Daniel Veillard34ce8be2002-03-18 19:37:11 +00009#define IN_LIBXML
Bjorn Reese70a9da52001-04-21 16:57:29 +000010#include "libxml.h"
Owen Taylor3473f882001-02-23 17:55:21 +000011
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000012#include <string.h>
Owen Taylor3473f882001-02-23 17:55:21 +000013#include <stdarg.h>
14#include <libxml/parser.h>
15#include <libxml/xmlerror.h>
Daniel Veillarde356c282001-03-10 12:32:04 +000016#include <libxml/xmlmemory.h>
Daniel Veillard3c01b1d2001-10-17 15:58:35 +000017#include <libxml/globals.h>
Owen Taylor3473f882001-02-23 17:55:21 +000018
Daniel Veillardffa3c742005-07-21 13:24:09 +000019void XMLCDECL xmlGenericErrorDefaultFunc (void *ctx ATTRIBUTE_UNUSED,
Daniel Veillard635ef722001-10-29 11:48:19 +000020 const char *msg,
21 ...);
22
Daniel Veillard1c43dbf2001-06-05 17:12:52 +000023#define XML_GET_VAR_STR(msg, str) { \
24 int size; \
25 int chars; \
26 char *larger; \
27 va_list ap; \
28 \
29 str = (char *) xmlMalloc(150); \
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000030 if (str != NULL) { \
Daniel Veillard1c43dbf2001-06-05 17:12:52 +000031 \
32 size = 150; \
33 \
34 while (1) { \
35 va_start(ap, msg); \
36 chars = vsnprintf(str, size, msg, ap); \
37 va_end(ap); \
38 if ((chars > -1) && (chars < size)) \
39 break; \
40 if (chars > -1) \
41 size += chars + 1; \
42 else \
43 size += 100; \
44 if ((larger = (char *) xmlRealloc(str, size)) == NULL) {\
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000045 break; \
Daniel Veillard1c43dbf2001-06-05 17:12:52 +000046 } \
47 str = larger; \
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000048 }} \
Daniel Veillard1c43dbf2001-06-05 17:12:52 +000049}
Bjorn Reese570ff082001-06-05 12:45:55 +000050
Owen Taylor3473f882001-02-23 17:55:21 +000051/************************************************************************
52 * *
53 * Handling of out of context errors *
54 * *
55 ************************************************************************/
56
57/**
58 * xmlGenericErrorDefaultFunc:
59 * @ctx: an error context
60 * @msg: the message to display/transmit
61 * @...: extra parameters for the message display
62 *
63 * Default handler for out of context error messages.
64 */
Daniel Veillardffa3c742005-07-21 13:24:09 +000065void XMLCDECL
Daniel Veillardc86a4fa2001-03-26 16:28:29 +000066xmlGenericErrorDefaultFunc(void *ctx ATTRIBUTE_UNUSED, const char *msg, ...) {
Owen Taylor3473f882001-02-23 17:55:21 +000067 va_list args;
68
69 if (xmlGenericErrorContext == NULL)
70 xmlGenericErrorContext = (void *) stderr;
71
72 va_start(args, msg);
73 vfprintf((FILE *)xmlGenericErrorContext, msg, args);
74 va_end(args);
75}
76
Daniel Veillard9d06d302002-01-22 18:15:52 +000077/**
78 * initGenericErrorDefaultFunc:
79 * @handler: the handler
80 *
81 * Set or reset (if NULL) the default handler for generic errors
Daniel Veillard7424eb62003-01-24 14:14:52 +000082 * to the builtin error function.
Daniel Veillard9d06d302002-01-22 18:15:52 +000083 */
Daniel Veillardd0463562001-10-13 09:15:48 +000084void
Daniel Veillarddb5850a2002-01-18 11:49:26 +000085initGenericErrorDefaultFunc(xmlGenericErrorFunc * handler)
Daniel Veillardd0463562001-10-13 09:15:48 +000086{
Daniel Veillarddb5850a2002-01-18 11:49:26 +000087 if (handler == NULL)
88 xmlGenericError = xmlGenericErrorDefaultFunc;
89 else
Daniel Veillardda0ff5d2004-04-20 09:45:26 +000090 xmlGenericError = (*handler);
Daniel Veillardd0463562001-10-13 09:15:48 +000091}
Owen Taylor3473f882001-02-23 17:55:21 +000092
93/**
94 * xmlSetGenericErrorFunc:
95 * @ctx: the new error handling context
96 * @handler: the new handler function
97 *
98 * Function to reset the handler and the error context for out of
99 * context error messages.
100 * This simply means that @handler will be called for subsequent
101 * error messages while not parsing nor validating. And @ctx will
102 * be passed as first argument to @handler
103 * One can simply force messages to be emitted to another FILE * than
104 * stderr by setting @ctx to this file handle and @handler to NULL.
Daniel Veillardda3b29a2004-08-14 11:15:13 +0000105 * For multi-threaded applications, this must be set separately for each thread.
Owen Taylor3473f882001-02-23 17:55:21 +0000106 */
107void
108xmlSetGenericErrorFunc(void *ctx, xmlGenericErrorFunc handler) {
109 xmlGenericErrorContext = ctx;
110 if (handler != NULL)
111 xmlGenericError = handler;
112 else
113 xmlGenericError = xmlGenericErrorDefaultFunc;
114}
115
Daniel Veillard659e71e2003-10-10 14:10:40 +0000116/**
117 * xmlSetStructuredErrorFunc:
118 * @ctx: the new error handling context
119 * @handler: the new handler function
120 *
121 * Function to reset the handler and the error context for out of
122 * context structured error messages.
123 * This simply means that @handler will be called for subsequent
124 * error messages while not parsing nor validating. And @ctx will
125 * be passed as first argument to @handler
Daniel Veillardda3b29a2004-08-14 11:15:13 +0000126 * For multi-threaded applications, this must be set separately for each thread.
Daniel Veillard659e71e2003-10-10 14:10:40 +0000127 */
128void
129xmlSetStructuredErrorFunc(void *ctx, xmlStructuredErrorFunc handler) {
130 xmlGenericErrorContext = ctx;
131 xmlStructuredError = handler;
132}
133
Owen Taylor3473f882001-02-23 17:55:21 +0000134/************************************************************************
135 * *
136 * Handling of parsing errors *
137 * *
138 ************************************************************************/
139
140/**
141 * xmlParserPrintFileInfo:
142 * @input: an xmlParserInputPtr input
143 *
144 * Displays the associated file and line informations for the current input
145 */
146
147void
148xmlParserPrintFileInfo(xmlParserInputPtr input) {
149 if (input != NULL) {
150 if (input->filename)
151 xmlGenericError(xmlGenericErrorContext,
152 "%s:%d: ", input->filename,
153 input->line);
154 else
155 xmlGenericError(xmlGenericErrorContext,
156 "Entity: line %d: ", input->line);
157 }
158}
159
160/**
161 * xmlParserPrintFileContext:
162 * @input: an xmlParserInputPtr input
163 *
164 * Displays current context within the input content for error tracking
165 */
166
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000167static void
168xmlParserPrintFileContextInternal(xmlParserInputPtr input ,
169 xmlGenericErrorFunc channel, void *data ) {
Daniel Veillard561b7f82002-03-20 21:55:57 +0000170 const xmlChar *cur, *base;
William M. Brackc1939562003-08-05 15:52:22 +0000171 unsigned int n, col; /* GCC warns if signed, because compared with sizeof() */
William M. Brack3dd57f72003-05-13 02:06:18 +0000172 xmlChar content[81]; /* space for 80 chars + line terminator */
Daniel Veillard2be30642001-03-27 00:32:28 +0000173 xmlChar *ctnt;
Owen Taylor3473f882001-02-23 17:55:21 +0000174
Daniel Veillard561b7f82002-03-20 21:55:57 +0000175 if (input == NULL) return;
Owen Taylor3473f882001-02-23 17:55:21 +0000176 cur = input->cur;
177 base = input->base;
Daniel Veillard2be30642001-03-27 00:32:28 +0000178 /* skip backwards over any end-of-lines */
William M. Brackc1939562003-08-05 15:52:22 +0000179 while ((cur > base) && ((*(cur) == '\n') || (*(cur) == '\r'))) {
Daniel Veillard561b7f82002-03-20 21:55:57 +0000180 cur--;
Owen Taylor3473f882001-02-23 17:55:21 +0000181 }
182 n = 0;
William M. Brack3dd57f72003-05-13 02:06:18 +0000183 /* search backwards for beginning-of-line (to max buff size) */
William M. Brackc1939562003-08-05 15:52:22 +0000184 while ((n++ < (sizeof(content)-1)) && (cur > base) &&
185 (*(cur) != '\n') && (*(cur) != '\r'))
Owen Taylor3473f882001-02-23 17:55:21 +0000186 cur--;
William M. Brackc1939562003-08-05 15:52:22 +0000187 if ((*(cur) == '\n') || (*(cur) == '\r')) cur++;
William M. Brack3dd57f72003-05-13 02:06:18 +0000188 /* calculate the error position in terms of the current position */
189 col = input->cur - cur;
190 /* search forward for end-of-line (to max buff size) */
Owen Taylor3473f882001-02-23 17:55:21 +0000191 n = 0;
Daniel Veillard2be30642001-03-27 00:32:28 +0000192 ctnt = content;
William M. Brack3dd57f72003-05-13 02:06:18 +0000193 /* copy selected text to our buffer */
William M. Brackc1939562003-08-05 15:52:22 +0000194 while ((*cur != 0) && (*(cur) != '\n') &&
195 (*(cur) != '\r') && (n < sizeof(content)-1)) {
Daniel Veillard561b7f82002-03-20 21:55:57 +0000196 *ctnt++ = *cur++;
197 n++;
Owen Taylor3473f882001-02-23 17:55:21 +0000198 }
Daniel Veillard2be30642001-03-27 00:32:28 +0000199 *ctnt = 0;
William M. Brack3dd57f72003-05-13 02:06:18 +0000200 /* print out the selected text */
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000201 channel(data ,"%s\n", content);
Daniel Veillard2be30642001-03-27 00:32:28 +0000202 /* create blank line with problem pointer */
Owen Taylor3473f882001-02-23 17:55:21 +0000203 n = 0;
Daniel Veillard7533cc82001-04-24 15:52:00 +0000204 ctnt = content;
William M. Brack3dd57f72003-05-13 02:06:18 +0000205 /* (leave buffer space for pointer + line terminator) */
206 while ((n<col) && (n++ < sizeof(content)-2) && (*ctnt != 0)) {
William M. Brackc1939562003-08-05 15:52:22 +0000207 if (*(ctnt) != '\t')
208 *(ctnt) = ' ';
William M. Brack69848302003-09-22 00:24:51 +0000209 ctnt++;
Owen Taylor3473f882001-02-23 17:55:21 +0000210 }
William M. Brack3dd57f72003-05-13 02:06:18 +0000211 *ctnt++ = '^';
212 *ctnt = 0;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000213 channel(data ,"%s\n", content);
Owen Taylor3473f882001-02-23 17:55:21 +0000214}
215
Daniel Veillard561b7f82002-03-20 21:55:57 +0000216/**
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000217 * xmlParserPrintFileContext:
218 * @input: an xmlParserInputPtr input
219 *
220 * Displays current context within the input content for error tracking
Daniel Veillard561b7f82002-03-20 21:55:57 +0000221 */
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000222void
223xmlParserPrintFileContext(xmlParserInputPtr input) {
224 xmlParserPrintFileContextInternal(input, xmlGenericError,
225 xmlGenericErrorContext);
Daniel Veillard561b7f82002-03-20 21:55:57 +0000226}
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000227
228/**
229 * xmlReportError:
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000230 * @err: the error
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000231 * @ctx: the parser context or NULL
232 * @str: the formatted error message
233 *
234 * Report an erro with its context, replace the 4 old error/warning
235 * routines.
236 */
237static void
Daniel Veillard4c004142003-10-07 11:33:24 +0000238xmlReportError(xmlErrorPtr err, xmlParserCtxtPtr ctxt, const char *str,
239 xmlGenericErrorFunc channel, void *data)
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000240{
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000241 char *file = NULL;
242 int line = 0;
243 int code = -1;
244 int domain;
Daniel Veillard4c004142003-10-07 11:33:24 +0000245 const xmlChar *name = NULL;
246 xmlNodePtr node;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000247 xmlErrorLevel level;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000248 xmlParserInputPtr input = NULL;
249 xmlParserInputPtr cur = NULL;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000250
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000251 if (err == NULL)
252 return;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000253
Daniel Veillard4c004142003-10-07 11:33:24 +0000254 if (channel == NULL) {
255 channel = xmlGenericError;
256 data = xmlGenericErrorContext;
257 }
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000258 file = err->file;
259 line = err->line;
260 code = err->code;
261 domain = err->domain;
262 level = err->level;
Daniel Veillard4c004142003-10-07 11:33:24 +0000263 node = err->node;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000264
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000265 if (code == XML_ERR_OK)
266 return;
267
Daniel Veillard4c004142003-10-07 11:33:24 +0000268 if ((node != NULL) && (node->type == XML_ELEMENT_NODE))
269 name = node->name;
270
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000271 /*
272 * Maintain the compatibility with the legacy error handling
273 */
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000274 if (ctxt != NULL) {
275 input = ctxt->input;
276 if ((input != NULL) && (input->filename == NULL) &&
277 (ctxt->inputNr > 1)) {
278 cur = input;
279 input = ctxt->inputTab[ctxt->inputNr - 2];
280 }
281 if (input != NULL) {
282 if (input->filename)
283 channel(data, "%s:%d: ", input->filename, input->line);
Daniel Veillardd96f6d32003-10-07 21:25:12 +0000284 else if ((line != 0) && (domain == XML_FROM_PARSER))
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000285 channel(data, "Entity: line %d: ", input->line);
286 }
287 } else {
288 if (file != NULL)
289 channel(data, "%s:%d: ", file, line);
Daniel Veillardd96f6d32003-10-07 21:25:12 +0000290 else if ((line != 0) && (domain == XML_FROM_PARSER))
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000291 channel(data, "Entity: line %d: ", line);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000292 }
Daniel Veillard4c004142003-10-07 11:33:24 +0000293 if (name != NULL) {
294 channel(data, "element %s: ", name);
295 }
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000296 if (code == XML_ERR_OK)
297 return;
298 switch (domain) {
299 case XML_FROM_PARSER:
300 channel(data, "parser ");
301 break;
302 case XML_FROM_NAMESPACE:
303 channel(data, "namespace ");
304 break;
305 case XML_FROM_DTD:
Daniel Veillard72b9e292003-10-28 15:44:17 +0000306 case XML_FROM_VALID:
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000307 channel(data, "validity ");
308 break;
309 case XML_FROM_HTML:
310 channel(data, "HTML parser ");
311 break;
312 case XML_FROM_MEMORY:
313 channel(data, "memory ");
314 break;
315 case XML_FROM_OUTPUT:
316 channel(data, "output ");
317 break;
318 case XML_FROM_IO:
319 channel(data, "I/O ");
320 break;
321 case XML_FROM_XINCLUDE:
322 channel(data, "XInclude ");
323 break;
324 case XML_FROM_XPATH:
325 channel(data, "XPath ");
326 break;
327 case XML_FROM_XPOINTER:
328 channel(data, "parser ");
329 break;
330 case XML_FROM_REGEXP:
331 channel(data, "regexp ");
332 break;
Daniel Veillardce1648b2005-01-04 15:10:22 +0000333 case XML_FROM_MODULE:
334 channel(data, "module ");
335 break;
Daniel Veillardd0c9c322003-10-10 00:49:42 +0000336 case XML_FROM_SCHEMASV:
Daniel Veillard87db3a82003-10-10 10:52:58 +0000337 channel(data, "Schemas validity ");
Daniel Veillardd0c9c322003-10-10 00:49:42 +0000338 break;
339 case XML_FROM_SCHEMASP:
Daniel Veillard87db3a82003-10-10 10:52:58 +0000340 channel(data, "Schemas parser ");
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000341 break;
Daniel Veillard4c004142003-10-07 11:33:24 +0000342 case XML_FROM_RELAXNGP:
343 channel(data, "Relax-NG parser ");
344 break;
345 case XML_FROM_RELAXNGV:
Daniel Veillardd0c9c322003-10-10 00:49:42 +0000346 channel(data, "Relax-NG validity ");
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000347 break;
348 case XML_FROM_CATALOG:
349 channel(data, "Catalog ");
350 break;
351 case XML_FROM_C14N:
352 channel(data, "C14N ");
353 break;
354 case XML_FROM_XSLT:
355 channel(data, "XSLT ");
356 break;
Daniel Veillard1fc3ed02005-08-24 12:46:09 +0000357 case XML_FROM_I18N:
358 channel(data, "encoding ");
359 break;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000360 default:
361 break;
362 }
363 if (code == XML_ERR_OK)
364 return;
365 switch (level) {
366 case XML_ERR_NONE:
367 channel(data, ": ");
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000368 break;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000369 case XML_ERR_WARNING:
370 channel(data, "warning : ");
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000371 break;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000372 case XML_ERR_ERROR:
373 channel(data, "error : ");
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000374 break;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000375 case XML_ERR_FATAL:
376 channel(data, "error : ");
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000377 break;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000378 }
379 if (code == XML_ERR_OK)
380 return;
381 if (str != NULL) {
Daniel Veillard828ce832003-10-08 19:19:10 +0000382 int len;
383 len = xmlStrlen((const xmlChar *)str);
384 if ((len > 0) && (str[len - 1] != '\n'))
Daniel Veillard828ce832003-10-08 19:19:10 +0000385 channel(data, "%s\n", str);
Daniel Veillarda8856222003-10-08 19:26:03 +0000386 else
387 channel(data, "%s", str);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000388 } else {
Daniel Veillard828ce832003-10-08 19:19:10 +0000389 channel(data, "%s\n", "out of memory error");
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000390 }
391 if (code == XML_ERR_OK)
392 return;
393
394 if (ctxt != NULL) {
395 xmlParserPrintFileContextInternal(input, channel, data);
396 if (cur != NULL) {
397 if (cur->filename)
398 channel(data, "%s:%d: \n", cur->filename, cur->line);
Daniel Veillardd96f6d32003-10-07 21:25:12 +0000399 else if ((line != 0) && (domain == XML_FROM_PARSER))
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000400 channel(data, "Entity: line %d: \n", cur->line);
401 xmlParserPrintFileContextInternal(cur, channel, data);
402 }
403 }
Daniel Veillardd96f6d32003-10-07 21:25:12 +0000404 if ((domain == XML_FROM_XPATH) && (err->str1 != NULL) &&
405 (err->int1 < 100) &&
406 (err->int1 < xmlStrlen((const xmlChar *)err->str1))) {
407 xmlChar buf[150];
408 int i;
409
410 channel(data, "%s\n", err->str1);
411 for (i=0;i < err->int1;i++)
412 buf[i] = ' ';
413 buf[i++] = '^';
414 buf[i] = 0;
415 channel(data, "%s\n", buf);
416 }
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000417}
418
419/**
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000420 * __xmlRaiseError:
William M. Brackd233e392004-05-16 03:12:08 +0000421 * @schannel: the structured callback channel
Daniel Veillard659e71e2003-10-10 14:10:40 +0000422 * @channel: the old callback channel
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000423 * @data: the callback data
424 * @ctx: the parser context or NULL
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000425 * @ctx: the parser context or NULL
426 * @domain: the domain for the error
427 * @code: the code for the error
428 * @level: the xmlErrorLevel for the error
429 * @file: the file source of the error (or NULL)
430 * @line: the line of the error or 0 if N/A
431 * @str1: extra string info
432 * @str2: extra string info
433 * @str3: extra string info
434 * @int1: extra int info
Aleksey Sanin8fdc32a2005-01-05 15:37:55 +0000435 * @col: column number of the error or 0 if N/A
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000436 * @msg: the message to display/transmit
437 * @...: extra parameters for the message display
438 *
William M. Brackd233e392004-05-16 03:12:08 +0000439 * Update the appropriate global or contextual error structure,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000440 * then forward the error message down the parser or generic
441 * error callback handler
442 */
Daniel Veillardffa3c742005-07-21 13:24:09 +0000443void XMLCDECL
Daniel Veillard659e71e2003-10-10 14:10:40 +0000444__xmlRaiseError(xmlStructuredErrorFunc schannel,
445 xmlGenericErrorFunc channel, void *data, void *ctx,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000446 void *nod, int domain, int code, xmlErrorLevel level,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000447 const char *file, int line, const char *str1,
Aleksey Sanin8fdc32a2005-01-05 15:37:55 +0000448 const char *str2, const char *str3, int int1, int col,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000449 const char *msg, ...)
450{
Daniel Veillardcd6ff282003-10-08 22:38:13 +0000451 xmlParserCtxtPtr ctxt = NULL;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000452 xmlNodePtr node = (xmlNodePtr) nod;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000453 char *str = NULL;
454 xmlParserInputPtr input = NULL;
455 xmlErrorPtr to = &xmlLastError;
William M. Brackd0407522004-10-02 03:54:00 +0000456 xmlNodePtr baseptr = NULL;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000457
Daniel Veillardb5fa0202003-12-08 17:41:29 +0000458 if ((xmlGetWarningsDefaultValue == 0) && (level == XML_ERR_WARNING))
459 return;
Daniel Veillardcd6ff282003-10-08 22:38:13 +0000460 if ((domain == XML_FROM_PARSER) || (domain == XML_FROM_HTML) ||
461 (domain == XML_FROM_DTD) || (domain == XML_FROM_NAMESPACE) ||
Daniel Veillard370ba3d2004-10-25 16:23:56 +0000462 (domain == XML_FROM_IO) || (domain == XML_FROM_VALID)) {
Daniel Veillardcd6ff282003-10-08 22:38:13 +0000463 ctxt = (xmlParserCtxtPtr) ctx;
Daniel Veillard659e71e2003-10-10 14:10:40 +0000464 if ((schannel == NULL) && (ctxt != NULL) && (ctxt->sax != NULL) &&
465 (ctxt->sax->initialized == XML_SAX2_MAGIC))
466 schannel = ctxt->sax->serror;
Daniel Veillardcd6ff282003-10-08 22:38:13 +0000467 }
William M. Brackcd3628b2004-07-25 21:07:29 +0000468 /*
469 * Check if structured error handler set
470 */
471 if (schannel == NULL) {
Daniel Veillardf88d8cf2003-12-08 10:25:02 +0000472 schannel = xmlStructuredError;
William M. Brackcd3628b2004-07-25 21:07:29 +0000473 /*
474 * if user has defined handler, change data ptr to user's choice
475 */
476 if (schannel != NULL)
477 data = xmlGenericErrorContext;
478 }
Daniel Veillard72b9e292003-10-28 15:44:17 +0000479 if ((domain == XML_FROM_VALID) &&
480 ((channel == xmlParserValidityError) ||
481 (channel == xmlParserValidityWarning))) {
482 ctxt = (xmlParserCtxtPtr) ctx;
483 if ((schannel == NULL) && (ctxt != NULL) && (ctxt->sax != NULL) &&
484 (ctxt->sax->initialized == XML_SAX2_MAGIC))
485 schannel = ctxt->sax->serror;
486 }
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000487 if (code == XML_ERR_OK)
488 return;
489 /*
490 * Formatting the message
491 */
492 if (msg == NULL) {
493 str = (char *) xmlStrdup(BAD_CAST "No error message provided");
494 } else {
495 XML_GET_VAR_STR(msg, str);
496 }
497
498 /*
499 * specific processing if a parser context is provided
500 */
501 if (ctxt != NULL) {
502 if (file == NULL) {
503 input = ctxt->input;
504 if ((input != NULL) && (input->filename == NULL) &&
505 (ctxt->inputNr > 1)) {
506 input = ctxt->inputTab[ctxt->inputNr - 2];
507 }
508 if (input != NULL) {
509 file = input->filename;
510 line = input->line;
Aleksey Sanin8fdc32a2005-01-05 15:37:55 +0000511 col = input->col;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000512 }
513 }
514 to = &ctxt->lastError;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000515 } else if ((node != NULL) && (file == NULL)) {
516 int i;
Daniel Veillard4c004142003-10-07 11:33:24 +0000517
Daniel Veillarddbee0f12005-06-27 13:42:57 +0000518 if ((node->doc != NULL) && (node->doc->URL != NULL)) {
William M. Brackd0407522004-10-02 03:54:00 +0000519 baseptr = node;
Daniel Veillard8ce01ce2005-08-25 20:14:38 +0000520/* file = (const char *) node->doc->URL; */
Daniel Veillarddbee0f12005-06-27 13:42:57 +0000521 }
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000522 for (i = 0;
523 ((i < 10) && (node != NULL) && (node->type != XML_ELEMENT_NODE));
524 i++)
525 node = node->parent;
William M. Brackd0407522004-10-02 03:54:00 +0000526 if ((baseptr == NULL) && (node != NULL) &&
Daniel Veillard4c004142003-10-07 11:33:24 +0000527 (node->doc != NULL) && (node->doc->URL != NULL))
William M. Brackd0407522004-10-02 03:54:00 +0000528 baseptr = node;
Daniel Veillard4c004142003-10-07 11:33:24 +0000529
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000530 if ((node != NULL) && (node->type == XML_ELEMENT_NODE))
Daniel Veillard3e35f8e2003-10-21 00:05:38 +0000531 line = node->line;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000532 }
533
534 /*
William M. Brackd233e392004-05-16 03:12:08 +0000535 * Save the information about the error
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000536 */
537 xmlResetError(to);
538 to->domain = domain;
539 to->code = code;
540 to->message = str;
541 to->level = level;
542 if (file != NULL)
543 to->file = (char *) xmlStrdup((const xmlChar *) file);
William M. Brackd0407522004-10-02 03:54:00 +0000544 else if (baseptr != NULL) {
545#ifdef LIBXML_XINCLUDE_ENABLED
546 /*
547 * We check if the error is within an XInclude section and,
548 * if so, attempt to print out the href of the XInclude instead
549 * of the usual "base" (doc->URL) for the node (bug 152623).
550 */
551 xmlNodePtr prev = baseptr;
552 int inclcount = 0;
553 while (prev != NULL) {
554 if (prev->prev == NULL)
555 prev = prev->parent;
556 else {
557 prev = prev->prev;
558 if (prev->type == XML_XINCLUDE_START) {
559 if (--inclcount < 0)
560 break;
561 } else if (prev->type == XML_XINCLUDE_END)
562 inclcount++;
563 }
564 }
565 if (prev != NULL) {
Daniel Veillard8ce01ce2005-08-25 20:14:38 +0000566 if (prev->type == XML_XINCLUDE_START) {
567 prev->type = XML_ELEMENT_NODE;
568 to->file = (char *) xmlGetProp(prev, BAD_CAST "href");
569 prev->type = XML_XINCLUDE_START;
570 } else {
571 to->file = (char *) xmlGetProp(prev, BAD_CAST "href");
572 }
William M. Brackd0407522004-10-02 03:54:00 +0000573 } else
574#endif
575 to->file = (char *) xmlStrdup(baseptr->doc->URL);
Daniel Veillard8ce01ce2005-08-25 20:14:38 +0000576 if ((to->file == NULL) && (node != NULL) && (node->doc != NULL)) {
577 to->file = (char *) xmlStrdup(node->doc->URL);
578 }
William M. Brackd0407522004-10-02 03:54:00 +0000579 file = to->file;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000580 }
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000581 to->line = line;
582 if (str1 != NULL)
583 to->str1 = (char *) xmlStrdup((const xmlChar *) str1);
584 if (str2 != NULL)
585 to->str2 = (char *) xmlStrdup((const xmlChar *) str2);
586 if (str3 != NULL)
587 to->str3 = (char *) xmlStrdup((const xmlChar *) str3);
588 to->int1 = int1;
Aleksey Sanin8fdc32a2005-01-05 15:37:55 +0000589 to->int2 = col;
Daniel Veillard4c004142003-10-07 11:33:24 +0000590 to->node = node;
Daniel Veillardcd6ff282003-10-08 22:38:13 +0000591 to->ctxt = ctx;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000592
Daniel Veillardd34b0b82004-01-02 20:26:01 +0000593 if (to != &xmlLastError)
594 xmlCopyError(to,&xmlLastError);
595
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000596 /*
William M. Brackcd3628b2004-07-25 21:07:29 +0000597 * Find the callback channel if channel param is NULL
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000598 */
William M. Brack9f797ab2004-07-28 07:40:12 +0000599 if ((ctxt != NULL) && (channel == NULL) && (xmlStructuredError == NULL) && (ctxt->sax != NULL)) {
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000600 if (level == XML_ERR_WARNING)
601 channel = ctxt->sax->warning;
602 else
603 channel = ctxt->sax->error;
Daniel Veillard4c004142003-10-07 11:33:24 +0000604 data = ctxt->userData;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000605 } else if (channel == NULL) {
Daniel Veillard659e71e2003-10-10 14:10:40 +0000606 if (xmlStructuredError != NULL)
607 schannel = xmlStructuredError;
608 else
609 channel = xmlGenericError;
Daniel Veillardda0aa4c2005-07-13 23:07:49 +0000610 if (!data) {
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000611 data = xmlGenericErrorContext;
612 }
Daniel Veillardda0aa4c2005-07-13 23:07:49 +0000613 }
Daniel Veillard659e71e2003-10-10 14:10:40 +0000614 if (schannel != NULL) {
615 schannel(data, to);
616 return;
617 }
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000618 if (channel == NULL)
619 return;
620
621 if ((channel == xmlParserError) ||
622 (channel == xmlParserWarning) ||
623 (channel == xmlParserValidityError) ||
624 (channel == xmlParserValidityWarning))
Daniel Veillard4c004142003-10-07 11:33:24 +0000625 xmlReportError(to, ctxt, str, NULL, NULL);
626 else if ((channel == (xmlGenericErrorFunc) fprintf) ||
627 (channel == xmlGenericErrorDefaultFunc))
628 xmlReportError(to, ctxt, str, channel, data);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000629 else
630 channel(data, "%s", str);
631}
Daniel Veillard561b7f82002-03-20 21:55:57 +0000632
Daniel Veillarde356c282001-03-10 12:32:04 +0000633/**
Daniel Veillard18ec16e2003-10-07 23:16:40 +0000634 * __xmlSimpleError:
635 * @domain: where the error comes from
636 * @code: the error code
637 * @node: the context node
638 * @extra: extra informations
639 *
640 * Handle an out of memory condition
641 */
642void
643__xmlSimpleError(int domain, int code, xmlNodePtr node,
644 const char *msg, const char *extra)
645{
646
647 if (code == XML_ERR_NO_MEMORY) {
648 if (extra)
Daniel Veillard659e71e2003-10-10 14:10:40 +0000649 __xmlRaiseError(NULL, NULL, NULL, NULL, node, domain,
Daniel Veillard18ec16e2003-10-07 23:16:40 +0000650 XML_ERR_NO_MEMORY, XML_ERR_FATAL, NULL, 0, extra,
651 NULL, NULL, 0, 0,
652 "Memory allocation failed : %s\n", extra);
653 else
Daniel Veillard659e71e2003-10-10 14:10:40 +0000654 __xmlRaiseError(NULL, NULL, NULL, NULL, node, domain,
Daniel Veillard18ec16e2003-10-07 23:16:40 +0000655 XML_ERR_NO_MEMORY, XML_ERR_FATAL, NULL, 0, NULL,
656 NULL, NULL, 0, 0, "Memory allocation failed\n");
657 } else {
Daniel Veillard659e71e2003-10-10 14:10:40 +0000658 __xmlRaiseError(NULL, NULL, NULL, NULL, node, domain,
Daniel Veillard18ec16e2003-10-07 23:16:40 +0000659 code, XML_ERR_ERROR, NULL, 0, extra,
660 NULL, NULL, 0, 0, msg, extra);
661 }
662}
663/**
Owen Taylor3473f882001-02-23 17:55:21 +0000664 * xmlParserError:
665 * @ctx: an XML parser context
666 * @msg: the message to display/transmit
667 * @...: extra parameters for the message display
668 *
669 * Display and format an error messages, gives file, line, position and
670 * extra parameters.
671 */
Daniel Veillardffa3c742005-07-21 13:24:09 +0000672void XMLCDECL
Owen Taylor3473f882001-02-23 17:55:21 +0000673xmlParserError(void *ctx, const char *msg, ...)
674{
675 xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
676 xmlParserInputPtr input = NULL;
677 xmlParserInputPtr cur = NULL;
Daniel Veillarde356c282001-03-10 12:32:04 +0000678 char * str;
Owen Taylor3473f882001-02-23 17:55:21 +0000679
680 if (ctxt != NULL) {
681 input = ctxt->input;
682 if ((input != NULL) && (input->filename == NULL) &&
683 (ctxt->inputNr > 1)) {
684 cur = input;
685 input = ctxt->inputTab[ctxt->inputNr - 2];
686 }
687 xmlParserPrintFileInfo(input);
688 }
689
690 xmlGenericError(xmlGenericErrorContext, "error: ");
Daniel Veillard1c43dbf2001-06-05 17:12:52 +0000691 XML_GET_VAR_STR(msg, str);
Daniel Veillard635ef722001-10-29 11:48:19 +0000692 xmlGenericError(xmlGenericErrorContext, "%s", str);
Daniel Veillard9e7160d2001-03-18 23:17:47 +0000693 if (str != NULL)
694 xmlFree(str);
Owen Taylor3473f882001-02-23 17:55:21 +0000695
696 if (ctxt != NULL) {
697 xmlParserPrintFileContext(input);
698 if (cur != NULL) {
699 xmlParserPrintFileInfo(cur);
700 xmlGenericError(xmlGenericErrorContext, "\n");
701 xmlParserPrintFileContext(cur);
702 }
703 }
704}
705
706/**
707 * xmlParserWarning:
708 * @ctx: an XML parser context
709 * @msg: the message to display/transmit
710 * @...: extra parameters for the message display
711 *
712 * Display and format a warning messages, gives file, line, position and
713 * extra parameters.
714 */
Daniel Veillardffa3c742005-07-21 13:24:09 +0000715void XMLCDECL
Owen Taylor3473f882001-02-23 17:55:21 +0000716xmlParserWarning(void *ctx, const char *msg, ...)
717{
718 xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
719 xmlParserInputPtr input = NULL;
720 xmlParserInputPtr cur = NULL;
Daniel Veillarde356c282001-03-10 12:32:04 +0000721 char * str;
Owen Taylor3473f882001-02-23 17:55:21 +0000722
723 if (ctxt != NULL) {
724 input = ctxt->input;
725 if ((input != NULL) && (input->filename == NULL) &&
726 (ctxt->inputNr > 1)) {
727 cur = input;
728 input = ctxt->inputTab[ctxt->inputNr - 2];
729 }
730 xmlParserPrintFileInfo(input);
731 }
732
733 xmlGenericError(xmlGenericErrorContext, "warning: ");
Daniel Veillard1c43dbf2001-06-05 17:12:52 +0000734 XML_GET_VAR_STR(msg, str);
Daniel Veillard635ef722001-10-29 11:48:19 +0000735 xmlGenericError(xmlGenericErrorContext, "%s", str);
Daniel Veillard9e7160d2001-03-18 23:17:47 +0000736 if (str != NULL)
737 xmlFree(str);
Owen Taylor3473f882001-02-23 17:55:21 +0000738
739 if (ctxt != NULL) {
740 xmlParserPrintFileContext(input);
741 if (cur != NULL) {
742 xmlParserPrintFileInfo(cur);
743 xmlGenericError(xmlGenericErrorContext, "\n");
744 xmlParserPrintFileContext(cur);
745 }
746 }
747}
748
749/************************************************************************
750 * *
751 * Handling of validation errors *
752 * *
753 ************************************************************************/
754
755/**
756 * xmlParserValidityError:
757 * @ctx: an XML parser context
758 * @msg: the message to display/transmit
759 * @...: extra parameters for the message display
760 *
761 * Display and format an validity error messages, gives file,
762 * line, position and extra parameters.
763 */
Daniel Veillardffa3c742005-07-21 13:24:09 +0000764void XMLCDECL
Owen Taylor3473f882001-02-23 17:55:21 +0000765xmlParserValidityError(void *ctx, const char *msg, ...)
766{
767 xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
768 xmlParserInputPtr input = NULL;
Daniel Veillard9e7160d2001-03-18 23:17:47 +0000769 char * str;
Daniel Veillard76575762002-09-05 14:21:15 +0000770 int len = xmlStrlen((const xmlChar *) msg);
771 static int had_info = 0;
Owen Taylor3473f882001-02-23 17:55:21 +0000772
Daniel Veillard76575762002-09-05 14:21:15 +0000773 if ((len > 1) && (msg[len - 2] != ':')) {
774 if (ctxt != NULL) {
775 input = ctxt->input;
776 if ((input->filename == NULL) && (ctxt->inputNr > 1))
777 input = ctxt->inputTab[ctxt->inputNr - 2];
778
779 if (had_info == 0) {
780 xmlParserPrintFileInfo(input);
781 }
782 }
783 xmlGenericError(xmlGenericErrorContext, "validity error: ");
Daniel Veillard76575762002-09-05 14:21:15 +0000784 had_info = 0;
785 } else {
786 had_info = 1;
Owen Taylor3473f882001-02-23 17:55:21 +0000787 }
788
Daniel Veillard1c43dbf2001-06-05 17:12:52 +0000789 XML_GET_VAR_STR(msg, str);
Daniel Veillard635ef722001-10-29 11:48:19 +0000790 xmlGenericError(xmlGenericErrorContext, "%s", str);
Daniel Veillard9e7160d2001-03-18 23:17:47 +0000791 if (str != NULL)
792 xmlFree(str);
Owen Taylor3473f882001-02-23 17:55:21 +0000793
Daniel Veillard76575762002-09-05 14:21:15 +0000794 if ((ctxt != NULL) && (input != NULL)) {
Owen Taylor3473f882001-02-23 17:55:21 +0000795 xmlParserPrintFileContext(input);
796 }
797}
798
799/**
800 * xmlParserValidityWarning:
801 * @ctx: an XML parser context
802 * @msg: the message to display/transmit
803 * @...: extra parameters for the message display
804 *
805 * Display and format a validity warning messages, gives file, line,
806 * position and extra parameters.
807 */
Daniel Veillardffa3c742005-07-21 13:24:09 +0000808void XMLCDECL
Owen Taylor3473f882001-02-23 17:55:21 +0000809xmlParserValidityWarning(void *ctx, const char *msg, ...)
810{
811 xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
812 xmlParserInputPtr input = NULL;
Daniel Veillard9e7160d2001-03-18 23:17:47 +0000813 char * str;
Daniel Veillard76575762002-09-05 14:21:15 +0000814 int len = xmlStrlen((const xmlChar *) msg);
Owen Taylor3473f882001-02-23 17:55:21 +0000815
Daniel Veillard76575762002-09-05 14:21:15 +0000816 if ((ctxt != NULL) && (len != 0) && (msg[len - 1] != ':')) {
Owen Taylor3473f882001-02-23 17:55:21 +0000817 input = ctxt->input;
818 if ((input->filename == NULL) && (ctxt->inputNr > 1))
819 input = ctxt->inputTab[ctxt->inputNr - 2];
820
821 xmlParserPrintFileInfo(input);
822 }
823
824 xmlGenericError(xmlGenericErrorContext, "validity warning: ");
Daniel Veillard1c43dbf2001-06-05 17:12:52 +0000825 XML_GET_VAR_STR(msg, str);
Daniel Veillard635ef722001-10-29 11:48:19 +0000826 xmlGenericError(xmlGenericErrorContext, "%s", str);
Daniel Veillard9e7160d2001-03-18 23:17:47 +0000827 if (str != NULL)
828 xmlFree(str);
Owen Taylor3473f882001-02-23 17:55:21 +0000829
830 if (ctxt != NULL) {
831 xmlParserPrintFileContext(input);
832 }
833}
834
835
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000836/************************************************************************
837 * *
838 * Extended Error Handling *
839 * *
840 ************************************************************************/
841
842/**
843 * xmlGetLastError:
844 *
845 * Get the last global error registered. This is per thread if compiled
846 * with thread support.
847 *
848 * Returns NULL if no error occured or a pointer to the error
849 */
850xmlErrorPtr
851xmlGetLastError(void)
852{
853 if (xmlLastError.code == XML_ERR_OK)
854 return (NULL);
855 return (&xmlLastError);
856}
857
858/**
859 * xmlResetError:
860 * @err: pointer to the error.
861 *
862 * Cleanup the error.
863 */
864void
865xmlResetError(xmlErrorPtr err)
866{
867 if (err == NULL)
868 return;
869 if (err->code == XML_ERR_OK)
870 return;
871 if (err->message != NULL)
872 xmlFree(err->message);
873 if (err->file != NULL)
874 xmlFree(err->file);
875 if (err->str1 != NULL)
876 xmlFree(err->str1);
877 if (err->str2 != NULL)
878 xmlFree(err->str2);
879 if (err->str3 != NULL)
880 xmlFree(err->str3);
881 memset(err, 0, sizeof(xmlError));
882 err->code = XML_ERR_OK;
883}
884
885/**
886 * xmlResetLastError:
887 *
888 * Cleanup the last global error registered. For parsing error
889 * this does not change the well-formedness result.
890 */
891void
892xmlResetLastError(void)
893{
894 if (xmlLastError.code == XML_ERR_OK)
895 return;
896 xmlResetError(&xmlLastError);
897}
898
899/**
900 * xmlCtxtGetLastError:
901 * @ctx: an XML parser context
902 *
903 * Get the last parsing error registered.
904 *
905 * Returns NULL if no error occured or a pointer to the error
906 */
907xmlErrorPtr
908xmlCtxtGetLastError(void *ctx)
909{
910 xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
911
912 if (ctxt == NULL)
913 return (NULL);
914 if (ctxt->lastError.code == XML_ERR_OK)
915 return (NULL);
916 return (&ctxt->lastError);
917}
918
919/**
920 * xmlCtxtResetLastError:
921 * @ctx: an XML parser context
922 *
923 * Cleanup the last global error registered. For parsing error
924 * this does not change the well-formedness result.
925 */
926void
927xmlCtxtResetLastError(void *ctx)
928{
929 xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
930
931 if (ctxt == NULL)
932 return;
933 if (ctxt->lastError.code == XML_ERR_OK)
934 return;
935 xmlResetError(&ctxt->lastError);
936}
937
938/**
939 * xmlCopyError:
940 * @from: a source error
941 * @to: a target error
942 *
943 * Save the original error to the new place.
944 *
945 * Returns 0 in case of success and -1 in case of error.
946 */
947int
948xmlCopyError(xmlErrorPtr from, xmlErrorPtr to) {
William M. Bracka3215c72004-07-31 16:24:01 +0000949 char *message, *file, *str1, *str2, *str3;
950
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000951 if ((from == NULL) || (to == NULL))
952 return(-1);
William M. Bracka3215c72004-07-31 16:24:01 +0000953
954 message = (char *) xmlStrdup((xmlChar *) from->message);
955 file = (char *) xmlStrdup ((xmlChar *) from->file);
956 str1 = (char *) xmlStrdup ((xmlChar *) from->str1);
957 str2 = (char *) xmlStrdup ((xmlChar *) from->str2);
958 str3 = (char *) xmlStrdup ((xmlChar *) from->str3);
959
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000960 if (to->message != NULL)
961 xmlFree(to->message);
962 if (to->file != NULL)
963 xmlFree(to->file);
964 if (to->str1 != NULL)
965 xmlFree(to->str1);
966 if (to->str2 != NULL)
967 xmlFree(to->str2);
968 if (to->str3 != NULL)
969 xmlFree(to->str3);
970 to->domain = from->domain;
971 to->code = from->code;
972 to->level = from->level;
973 to->line = from->line;
Daniel Veillard4c004142003-10-07 11:33:24 +0000974 to->node = from->node;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000975 to->int1 = from->int1;
976 to->int2 = from->int2;
Daniel Veillard4c004142003-10-07 11:33:24 +0000977 to->node = from->node;
978 to->ctxt = from->ctxt;
William M. Bracka3215c72004-07-31 16:24:01 +0000979 to->message = message;
980 to->file = file;
981 to->str1 = str1;
982 to->str2 = str2;
983 to->str3 = str3;
984
985 return 0;
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000986}
987
Daniel Veillard5d4644e2005-04-01 13:11:58 +0000988#define bottom_error
989#include "elfgcchack.h"