45a3f703a24caa3458bceaad1150b25f6c27c1f7
[reactos.git] / reactos / sdk / lib / 3rdparty / libxml2 / valid.c
1 /*
2 * valid.c : part of the code use to do the DTD handling and the validity
3 * checking
4 *
5 * See Copyright for the status of this software.
6 *
7 * daniel@veillard.com
8 */
9
10 #define IN_LIBXML
11 #include "libxml.h"
12
13 #include <string.h>
14
15 #ifdef HAVE_STDLIB_H
16 #include <stdlib.h>
17 #endif
18
19 #include <libxml/xmlmemory.h>
20 #include <libxml/hash.h>
21 #include <libxml/uri.h>
22 #include <libxml/valid.h>
23 #include <libxml/parser.h>
24 #include <libxml/parserInternals.h>
25 #include <libxml/xmlerror.h>
26 #include <libxml/list.h>
27 #include <libxml/globals.h>
28
29 static xmlElementPtr xmlGetDtdElementDesc2(xmlDtdPtr dtd, const xmlChar *name,
30 int create);
31 /* #define DEBUG_VALID_ALGO */
32 /* #define DEBUG_REGEXP_ALGO */
33
34 #define TODO \
35 xmlGenericError(xmlGenericErrorContext, \
36 "Unimplemented block at %s:%d\n", \
37 __FILE__, __LINE__);
38
39 #ifdef LIBXML_VALID_ENABLED
40 static int
41 xmlValidateAttributeValueInternal(xmlDocPtr doc, xmlAttributeType type,
42 const xmlChar *value);
43 #endif
44 /************************************************************************
45 * *
46 * Error handling routines *
47 * *
48 ************************************************************************/
49
50 /**
51 * xmlVErrMemory:
52 * @ctxt: an XML validation parser context
53 * @extra: extra informations
54 *
55 * Handle an out of memory error
56 */
57 static void
58 xmlVErrMemory(xmlValidCtxtPtr ctxt, const char *extra)
59 {
60 xmlGenericErrorFunc channel = NULL;
61 xmlParserCtxtPtr pctxt = NULL;
62 void *data = NULL;
63
64 if (ctxt != NULL) {
65 channel = ctxt->error;
66 data = ctxt->userData;
67 /* Use the special values to detect if it is part of a parsing
68 context */
69 if ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
70 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1)) {
71 long delta = (char *) ctxt - (char *) ctxt->userData;
72 if ((delta > 0) && (delta < 250))
73 pctxt = ctxt->userData;
74 }
75 }
76 if (extra)
77 __xmlRaiseError(NULL, channel, data,
78 pctxt, NULL, XML_FROM_VALID, XML_ERR_NO_MEMORY,
79 XML_ERR_FATAL, NULL, 0, extra, NULL, NULL, 0, 0,
80 "Memory allocation failed : %s\n", extra);
81 else
82 __xmlRaiseError(NULL, channel, data,
83 pctxt, NULL, XML_FROM_VALID, XML_ERR_NO_MEMORY,
84 XML_ERR_FATAL, NULL, 0, NULL, NULL, NULL, 0, 0,
85 "Memory allocation failed\n");
86 }
87
88 /**
89 * xmlErrValid:
90 * @ctxt: an XML validation parser context
91 * @error: the error number
92 * @extra: extra informations
93 *
94 * Handle a validation error
95 */
96 static void
97 xmlErrValid(xmlValidCtxtPtr ctxt, xmlParserErrors error,
98 const char *msg, const char *extra)
99 {
100 xmlGenericErrorFunc channel = NULL;
101 xmlParserCtxtPtr pctxt = NULL;
102 void *data = NULL;
103
104 if (ctxt != NULL) {
105 channel = ctxt->error;
106 data = ctxt->userData;
107 /* Use the special values to detect if it is part of a parsing
108 context */
109 if ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
110 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1)) {
111 long delta = (char *) ctxt - (char *) ctxt->userData;
112 if ((delta > 0) && (delta < 250))
113 pctxt = ctxt->userData;
114 }
115 }
116 if (extra)
117 __xmlRaiseError(NULL, channel, data,
118 pctxt, NULL, XML_FROM_VALID, error,
119 XML_ERR_ERROR, NULL, 0, extra, NULL, NULL, 0, 0,
120 msg, extra);
121 else
122 __xmlRaiseError(NULL, channel, data,
123 pctxt, NULL, XML_FROM_VALID, error,
124 XML_ERR_ERROR, NULL, 0, NULL, NULL, NULL, 0, 0,
125 "%s", msg);
126 }
127
128 #if defined(LIBXML_VALID_ENABLED) || defined(LIBXML_SCHEMAS_ENABLED)
129 /**
130 * xmlErrValidNode:
131 * @ctxt: an XML validation parser context
132 * @node: the node raising the error
133 * @error: the error number
134 * @str1: extra informations
135 * @str2: extra informations
136 * @str3: extra informations
137 *
138 * Handle a validation error, provide contextual informations
139 */
140 static void
141 xmlErrValidNode(xmlValidCtxtPtr ctxt,
142 xmlNodePtr node, xmlParserErrors error,
143 const char *msg, const xmlChar * str1,
144 const xmlChar * str2, const xmlChar * str3)
145 {
146 xmlStructuredErrorFunc schannel = NULL;
147 xmlGenericErrorFunc channel = NULL;
148 xmlParserCtxtPtr pctxt = NULL;
149 void *data = NULL;
150
151 if (ctxt != NULL) {
152 channel = ctxt->error;
153 data = ctxt->userData;
154 /* Use the special values to detect if it is part of a parsing
155 context */
156 if ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
157 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1)) {
158 long delta = (char *) ctxt - (char *) ctxt->userData;
159 if ((delta > 0) && (delta < 250))
160 pctxt = ctxt->userData;
161 }
162 }
163 __xmlRaiseError(schannel, channel, data, pctxt, node, XML_FROM_VALID, error,
164 XML_ERR_ERROR, NULL, 0,
165 (const char *) str1,
166 (const char *) str1,
167 (const char *) str3, 0, 0, msg, str1, str2, str3);
168 }
169 #endif /* LIBXML_VALID_ENABLED or LIBXML_SCHEMAS_ENABLED */
170
171 #ifdef LIBXML_VALID_ENABLED
172 /**
173 * xmlErrValidNodeNr:
174 * @ctxt: an XML validation parser context
175 * @node: the node raising the error
176 * @error: the error number
177 * @str1: extra informations
178 * @int2: extra informations
179 * @str3: extra informations
180 *
181 * Handle a validation error, provide contextual informations
182 */
183 static void
184 xmlErrValidNodeNr(xmlValidCtxtPtr ctxt,
185 xmlNodePtr node, xmlParserErrors error,
186 const char *msg, const xmlChar * str1,
187 int int2, const xmlChar * str3)
188 {
189 xmlStructuredErrorFunc schannel = NULL;
190 xmlGenericErrorFunc channel = NULL;
191 xmlParserCtxtPtr pctxt = NULL;
192 void *data = NULL;
193
194 if (ctxt != NULL) {
195 channel = ctxt->error;
196 data = ctxt->userData;
197 /* Use the special values to detect if it is part of a parsing
198 context */
199 if ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
200 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1)) {
201 long delta = (char *) ctxt - (char *) ctxt->userData;
202 if ((delta > 0) && (delta < 250))
203 pctxt = ctxt->userData;
204 }
205 }
206 __xmlRaiseError(schannel, channel, data, pctxt, node, XML_FROM_VALID, error,
207 XML_ERR_ERROR, NULL, 0,
208 (const char *) str1,
209 (const char *) str3,
210 NULL, int2, 0, msg, str1, int2, str3);
211 }
212
213 /**
214 * xmlErrValidWarning:
215 * @ctxt: an XML validation parser context
216 * @node: the node raising the error
217 * @error: the error number
218 * @str1: extra information
219 * @str2: extra information
220 * @str3: extra information
221 *
222 * Handle a validation error, provide contextual information
223 */
224 static void
225 xmlErrValidWarning(xmlValidCtxtPtr ctxt,
226 xmlNodePtr node, xmlParserErrors error,
227 const char *msg, const xmlChar * str1,
228 const xmlChar * str2, const xmlChar * str3)
229 {
230 xmlStructuredErrorFunc schannel = NULL;
231 xmlGenericErrorFunc channel = NULL;
232 xmlParserCtxtPtr pctxt = NULL;
233 void *data = NULL;
234
235 if (ctxt != NULL) {
236 channel = ctxt->warning;
237 data = ctxt->userData;
238 /* Use the special values to detect if it is part of a parsing
239 context */
240 if ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
241 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1)) {
242 long delta = (char *) ctxt - (char *) ctxt->userData;
243 if ((delta > 0) && (delta < 250))
244 pctxt = ctxt->userData;
245 }
246 }
247 __xmlRaiseError(schannel, channel, data, pctxt, node, XML_FROM_VALID, error,
248 XML_ERR_WARNING, NULL, 0,
249 (const char *) str1,
250 (const char *) str1,
251 (const char *) str3, 0, 0, msg, str1, str2, str3);
252 }
253
254
255
256 #ifdef LIBXML_REGEXP_ENABLED
257 /*
258 * If regexp are enabled we can do continuous validation without the
259 * need of a tree to validate the content model. this is done in each
260 * callbacks.
261 * Each xmlValidState represent the validation state associated to the
262 * set of nodes currently open from the document root to the current element.
263 */
264
265
266 typedef struct _xmlValidState {
267 xmlElementPtr elemDecl; /* pointer to the content model */
268 xmlNodePtr node; /* pointer to the current node */
269 xmlRegExecCtxtPtr exec; /* regexp runtime */
270 } _xmlValidState;
271
272
273 static int
274 vstateVPush(xmlValidCtxtPtr ctxt, xmlElementPtr elemDecl, xmlNodePtr node) {
275 if ((ctxt->vstateMax == 0) || (ctxt->vstateTab == NULL)) {
276 ctxt->vstateMax = 10;
277 ctxt->vstateTab = (xmlValidState *) xmlMalloc(ctxt->vstateMax *
278 sizeof(ctxt->vstateTab[0]));
279 if (ctxt->vstateTab == NULL) {
280 xmlVErrMemory(ctxt, "malloc failed");
281 return(-1);
282 }
283 }
284
285 if (ctxt->vstateNr >= ctxt->vstateMax) {
286 xmlValidState *tmp;
287
288 tmp = (xmlValidState *) xmlRealloc(ctxt->vstateTab,
289 2 * ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
290 if (tmp == NULL) {
291 xmlVErrMemory(ctxt, "realloc failed");
292 return(-1);
293 }
294 ctxt->vstateMax *= 2;
295 ctxt->vstateTab = tmp;
296 }
297 ctxt->vstate = &ctxt->vstateTab[ctxt->vstateNr];
298 ctxt->vstateTab[ctxt->vstateNr].elemDecl = elemDecl;
299 ctxt->vstateTab[ctxt->vstateNr].node = node;
300 if ((elemDecl != NULL) && (elemDecl->etype == XML_ELEMENT_TYPE_ELEMENT)) {
301 if (elemDecl->contModel == NULL)
302 xmlValidBuildContentModel(ctxt, elemDecl);
303 if (elemDecl->contModel != NULL) {
304 ctxt->vstateTab[ctxt->vstateNr].exec =
305 xmlRegNewExecCtxt(elemDecl->contModel, NULL, NULL);
306 } else {
307 ctxt->vstateTab[ctxt->vstateNr].exec = NULL;
308 xmlErrValidNode(ctxt, (xmlNodePtr) elemDecl,
309 XML_ERR_INTERNAL_ERROR,
310 "Failed to build content model regexp for %s\n",
311 node->name, NULL, NULL);
312 }
313 }
314 return(ctxt->vstateNr++);
315 }
316
317 static int
318 vstateVPop(xmlValidCtxtPtr ctxt) {
319 xmlElementPtr elemDecl;
320
321 if (ctxt->vstateNr < 1) return(-1);
322 ctxt->vstateNr--;
323 elemDecl = ctxt->vstateTab[ctxt->vstateNr].elemDecl;
324 ctxt->vstateTab[ctxt->vstateNr].elemDecl = NULL;
325 ctxt->vstateTab[ctxt->vstateNr].node = NULL;
326 if ((elemDecl != NULL) && (elemDecl->etype == XML_ELEMENT_TYPE_ELEMENT)) {
327 xmlRegFreeExecCtxt(ctxt->vstateTab[ctxt->vstateNr].exec);
328 }
329 ctxt->vstateTab[ctxt->vstateNr].exec = NULL;
330 if (ctxt->vstateNr >= 1)
331 ctxt->vstate = &ctxt->vstateTab[ctxt->vstateNr - 1];
332 else
333 ctxt->vstate = NULL;
334 return(ctxt->vstateNr);
335 }
336
337 #else /* not LIBXML_REGEXP_ENABLED */
338 /*
339 * If regexp are not enabled, it uses a home made algorithm less
340 * complex and easier to
341 * debug/maintain than a generic NFA -> DFA state based algo. The
342 * only restriction is on the deepness of the tree limited by the
343 * size of the occurs bitfield
344 *
345 * this is the content of a saved state for rollbacks
346 */
347
348 #define ROLLBACK_OR 0
349 #define ROLLBACK_PARENT 1
350
351 typedef struct _xmlValidState {
352 xmlElementContentPtr cont; /* pointer to the content model subtree */
353 xmlNodePtr node; /* pointer to the current node in the list */
354 long occurs;/* bitfield for multiple occurrences */
355 unsigned char depth; /* current depth in the overall tree */
356 unsigned char state; /* ROLLBACK_XXX */
357 } _xmlValidState;
358
359 #define MAX_RECURSE 25000
360 #define MAX_DEPTH ((sizeof(_xmlValidState.occurs)) * 8)
361 #define CONT ctxt->vstate->cont
362 #define NODE ctxt->vstate->node
363 #define DEPTH ctxt->vstate->depth
364 #define OCCURS ctxt->vstate->occurs
365 #define STATE ctxt->vstate->state
366
367 #define OCCURRENCE (ctxt->vstate->occurs & (1 << DEPTH))
368 #define PARENT_OCCURRENCE (ctxt->vstate->occurs & ((1 << DEPTH) - 1))
369
370 #define SET_OCCURRENCE ctxt->vstate->occurs |= (1 << DEPTH)
371 #define RESET_OCCURRENCE ctxt->vstate->occurs &= ((1 << DEPTH) - 1)
372
373 static int
374 vstateVPush(xmlValidCtxtPtr ctxt, xmlElementContentPtr cont,
375 xmlNodePtr node, unsigned char depth, long occurs,
376 unsigned char state) {
377 int i = ctxt->vstateNr - 1;
378
379 if (ctxt->vstateNr > MAX_RECURSE) {
380 return(-1);
381 }
382 if (ctxt->vstateTab == NULL) {
383 ctxt->vstateMax = 8;
384 ctxt->vstateTab = (xmlValidState *) xmlMalloc(
385 ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
386 if (ctxt->vstateTab == NULL) {
387 xmlVErrMemory(ctxt, "malloc failed");
388 return(-1);
389 }
390 }
391 if (ctxt->vstateNr >= ctxt->vstateMax) {
392 xmlValidState *tmp;
393
394 tmp = (xmlValidState *) xmlRealloc(ctxt->vstateTab,
395 2 * ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
396 if (tmp == NULL) {
397 xmlVErrMemory(ctxt, "malloc failed");
398 return(-1);
399 }
400 ctxt->vstateMax *= 2;
401 ctxt->vstateTab = tmp;
402 ctxt->vstate = &ctxt->vstateTab[0];
403 }
404 /*
405 * Don't push on the stack a state already here
406 */
407 if ((i >= 0) && (ctxt->vstateTab[i].cont == cont) &&
408 (ctxt->vstateTab[i].node == node) &&
409 (ctxt->vstateTab[i].depth == depth) &&
410 (ctxt->vstateTab[i].occurs == occurs) &&
411 (ctxt->vstateTab[i].state == state))
412 return(ctxt->vstateNr);
413 ctxt->vstateTab[ctxt->vstateNr].cont = cont;
414 ctxt->vstateTab[ctxt->vstateNr].node = node;
415 ctxt->vstateTab[ctxt->vstateNr].depth = depth;
416 ctxt->vstateTab[ctxt->vstateNr].occurs = occurs;
417 ctxt->vstateTab[ctxt->vstateNr].state = state;
418 return(ctxt->vstateNr++);
419 }
420
421 static int
422 vstateVPop(xmlValidCtxtPtr ctxt) {
423 if (ctxt->vstateNr <= 1) return(-1);
424 ctxt->vstateNr--;
425 ctxt->vstate = &ctxt->vstateTab[0];
426 ctxt->vstate->cont = ctxt->vstateTab[ctxt->vstateNr].cont;
427 ctxt->vstate->node = ctxt->vstateTab[ctxt->vstateNr].node;
428 ctxt->vstate->depth = ctxt->vstateTab[ctxt->vstateNr].depth;
429 ctxt->vstate->occurs = ctxt->vstateTab[ctxt->vstateNr].occurs;
430 ctxt->vstate->state = ctxt->vstateTab[ctxt->vstateNr].state;
431 return(ctxt->vstateNr);
432 }
433
434 #endif /* LIBXML_REGEXP_ENABLED */
435
436 static int
437 nodeVPush(xmlValidCtxtPtr ctxt, xmlNodePtr value)
438 {
439 if (ctxt->nodeMax <= 0) {
440 ctxt->nodeMax = 4;
441 ctxt->nodeTab =
442 (xmlNodePtr *) xmlMalloc(ctxt->nodeMax *
443 sizeof(ctxt->nodeTab[0]));
444 if (ctxt->nodeTab == NULL) {
445 xmlVErrMemory(ctxt, "malloc failed");
446 ctxt->nodeMax = 0;
447 return (0);
448 }
449 }
450 if (ctxt->nodeNr >= ctxt->nodeMax) {
451 xmlNodePtr *tmp;
452 tmp = (xmlNodePtr *) xmlRealloc(ctxt->nodeTab,
453 ctxt->nodeMax * 2 * sizeof(ctxt->nodeTab[0]));
454 if (tmp == NULL) {
455 xmlVErrMemory(ctxt, "realloc failed");
456 return (0);
457 }
458 ctxt->nodeMax *= 2;
459 ctxt->nodeTab = tmp;
460 }
461 ctxt->nodeTab[ctxt->nodeNr] = value;
462 ctxt->node = value;
463 return (ctxt->nodeNr++);
464 }
465 static xmlNodePtr
466 nodeVPop(xmlValidCtxtPtr ctxt)
467 {
468 xmlNodePtr ret;
469
470 if (ctxt->nodeNr <= 0)
471 return (NULL);
472 ctxt->nodeNr--;
473 if (ctxt->nodeNr > 0)
474 ctxt->node = ctxt->nodeTab[ctxt->nodeNr - 1];
475 else
476 ctxt->node = NULL;
477 ret = ctxt->nodeTab[ctxt->nodeNr];
478 ctxt->nodeTab[ctxt->nodeNr] = NULL;
479 return (ret);
480 }
481
482 #ifdef DEBUG_VALID_ALGO
483 static void
484 xmlValidPrintNode(xmlNodePtr cur) {
485 if (cur == NULL) {
486 xmlGenericError(xmlGenericErrorContext, "null");
487 return;
488 }
489 switch (cur->type) {
490 case XML_ELEMENT_NODE:
491 xmlGenericError(xmlGenericErrorContext, "%s ", cur->name);
492 break;
493 case XML_TEXT_NODE:
494 xmlGenericError(xmlGenericErrorContext, "text ");
495 break;
496 case XML_CDATA_SECTION_NODE:
497 xmlGenericError(xmlGenericErrorContext, "cdata ");
498 break;
499 case XML_ENTITY_REF_NODE:
500 xmlGenericError(xmlGenericErrorContext, "&%s; ", cur->name);
501 break;
502 case XML_PI_NODE:
503 xmlGenericError(xmlGenericErrorContext, "pi(%s) ", cur->name);
504 break;
505 case XML_COMMENT_NODE:
506 xmlGenericError(xmlGenericErrorContext, "comment ");
507 break;
508 case XML_ATTRIBUTE_NODE:
509 xmlGenericError(xmlGenericErrorContext, "?attr? ");
510 break;
511 case XML_ENTITY_NODE:
512 xmlGenericError(xmlGenericErrorContext, "?ent? ");
513 break;
514 case XML_DOCUMENT_NODE:
515 xmlGenericError(xmlGenericErrorContext, "?doc? ");
516 break;
517 case XML_DOCUMENT_TYPE_NODE:
518 xmlGenericError(xmlGenericErrorContext, "?doctype? ");
519 break;
520 case XML_DOCUMENT_FRAG_NODE:
521 xmlGenericError(xmlGenericErrorContext, "?frag? ");
522 break;
523 case XML_NOTATION_NODE:
524 xmlGenericError(xmlGenericErrorContext, "?nota? ");
525 break;
526 case XML_HTML_DOCUMENT_NODE:
527 xmlGenericError(xmlGenericErrorContext, "?html? ");
528 break;
529 #ifdef LIBXML_DOCB_ENABLED
530 case XML_DOCB_DOCUMENT_NODE:
531 xmlGenericError(xmlGenericErrorContext, "?docb? ");
532 break;
533 #endif
534 case XML_DTD_NODE:
535 xmlGenericError(xmlGenericErrorContext, "?dtd? ");
536 break;
537 case XML_ELEMENT_DECL:
538 xmlGenericError(xmlGenericErrorContext, "?edecl? ");
539 break;
540 case XML_ATTRIBUTE_DECL:
541 xmlGenericError(xmlGenericErrorContext, "?adecl? ");
542 break;
543 case XML_ENTITY_DECL:
544 xmlGenericError(xmlGenericErrorContext, "?entdecl? ");
545 break;
546 case XML_NAMESPACE_DECL:
547 xmlGenericError(xmlGenericErrorContext, "?nsdecl? ");
548 break;
549 case XML_XINCLUDE_START:
550 xmlGenericError(xmlGenericErrorContext, "incstart ");
551 break;
552 case XML_XINCLUDE_END:
553 xmlGenericError(xmlGenericErrorContext, "incend ");
554 break;
555 }
556 }
557
558 static void
559 xmlValidPrintNodeList(xmlNodePtr cur) {
560 if (cur == NULL)
561 xmlGenericError(xmlGenericErrorContext, "null ");
562 while (cur != NULL) {
563 xmlValidPrintNode(cur);
564 cur = cur->next;
565 }
566 }
567
568 static void
569 xmlValidDebug(xmlNodePtr cur, xmlElementContentPtr cont) {
570 char expr[5000];
571
572 expr[0] = 0;
573 xmlGenericError(xmlGenericErrorContext, "valid: ");
574 xmlValidPrintNodeList(cur);
575 xmlGenericError(xmlGenericErrorContext, "against ");
576 xmlSnprintfElementContent(expr, 5000, cont, 1);
577 xmlGenericError(xmlGenericErrorContext, "%s\n", expr);
578 }
579
580 static void
581 xmlValidDebugState(xmlValidStatePtr state) {
582 xmlGenericError(xmlGenericErrorContext, "(");
583 if (state->cont == NULL)
584 xmlGenericError(xmlGenericErrorContext, "null,");
585 else
586 switch (state->cont->type) {
587 case XML_ELEMENT_CONTENT_PCDATA:
588 xmlGenericError(xmlGenericErrorContext, "pcdata,");
589 break;
590 case XML_ELEMENT_CONTENT_ELEMENT:
591 xmlGenericError(xmlGenericErrorContext, "%s,",
592 state->cont->name);
593 break;
594 case XML_ELEMENT_CONTENT_SEQ:
595 xmlGenericError(xmlGenericErrorContext, "seq,");
596 break;
597 case XML_ELEMENT_CONTENT_OR:
598 xmlGenericError(xmlGenericErrorContext, "or,");
599 break;
600 }
601 xmlValidPrintNode(state->node);
602 xmlGenericError(xmlGenericErrorContext, ",%d,%X,%d)",
603 state->depth, state->occurs, state->state);
604 }
605
606 static void
607 xmlValidStateDebug(xmlValidCtxtPtr ctxt) {
608 int i, j;
609
610 xmlGenericError(xmlGenericErrorContext, "state: ");
611 xmlValidDebugState(ctxt->vstate);
612 xmlGenericError(xmlGenericErrorContext, " stack: %d ",
613 ctxt->vstateNr - 1);
614 for (i = 0, j = ctxt->vstateNr - 1;(i < 3) && (j > 0);i++,j--)
615 xmlValidDebugState(&ctxt->vstateTab[j]);
616 xmlGenericError(xmlGenericErrorContext, "\n");
617 }
618
619 /*****
620 #define DEBUG_VALID_STATE(n,c) xmlValidDebug(n,c);
621 *****/
622
623 #define DEBUG_VALID_STATE(n,c) xmlValidStateDebug(ctxt);
624 #define DEBUG_VALID_MSG(m) \
625 xmlGenericError(xmlGenericErrorContext, "%s\n", m);
626
627 #else
628 #define DEBUG_VALID_STATE(n,c)
629 #define DEBUG_VALID_MSG(m)
630 #endif
631
632 /* TODO: use hash table for accesses to elem and attribute definitions */
633
634
635 #define CHECK_DTD \
636 if (doc == NULL) return(0); \
637 else if ((doc->intSubset == NULL) && \
638 (doc->extSubset == NULL)) return(0)
639
640 #ifdef LIBXML_REGEXP_ENABLED
641
642 /************************************************************************
643 * *
644 * Content model validation based on the regexps *
645 * *
646 ************************************************************************/
647
648 /**
649 * xmlValidBuildAContentModel:
650 * @content: the content model
651 * @ctxt: the schema parser context
652 * @name: the element name whose content is being built
653 *
654 * Generate the automata sequence needed for that type
655 *
656 * Returns 1 if successful or 0 in case of error.
657 */
658 static int
659 xmlValidBuildAContentModel(xmlElementContentPtr content,
660 xmlValidCtxtPtr ctxt,
661 const xmlChar *name) {
662 if (content == NULL) {
663 xmlErrValidNode(ctxt, NULL, XML_ERR_INTERNAL_ERROR,
664 "Found NULL content in content model of %s\n",
665 name, NULL, NULL);
666 return(0);
667 }
668 switch (content->type) {
669 case XML_ELEMENT_CONTENT_PCDATA:
670 xmlErrValidNode(ctxt, NULL, XML_ERR_INTERNAL_ERROR,
671 "Found PCDATA in content model of %s\n",
672 name, NULL, NULL);
673 return(0);
674 break;
675 case XML_ELEMENT_CONTENT_ELEMENT: {
676 xmlAutomataStatePtr oldstate = ctxt->state;
677 xmlChar fn[50];
678 xmlChar *fullname;
679
680 fullname = xmlBuildQName(content->name, content->prefix, fn, 50);
681 if (fullname == NULL) {
682 xmlVErrMemory(ctxt, "Building content model");
683 return(0);
684 }
685
686 switch (content->ocur) {
687 case XML_ELEMENT_CONTENT_ONCE:
688 ctxt->state = xmlAutomataNewTransition(ctxt->am,
689 ctxt->state, NULL, fullname, NULL);
690 break;
691 case XML_ELEMENT_CONTENT_OPT:
692 ctxt->state = xmlAutomataNewTransition(ctxt->am,
693 ctxt->state, NULL, fullname, NULL);
694 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
695 break;
696 case XML_ELEMENT_CONTENT_PLUS:
697 ctxt->state = xmlAutomataNewTransition(ctxt->am,
698 ctxt->state, NULL, fullname, NULL);
699 xmlAutomataNewTransition(ctxt->am, ctxt->state,
700 ctxt->state, fullname, NULL);
701 break;
702 case XML_ELEMENT_CONTENT_MULT:
703 ctxt->state = xmlAutomataNewEpsilon(ctxt->am,
704 ctxt->state, NULL);
705 xmlAutomataNewTransition(ctxt->am,
706 ctxt->state, ctxt->state, fullname, NULL);
707 break;
708 }
709 if ((fullname != fn) && (fullname != content->name))
710 xmlFree(fullname);
711 break;
712 }
713 case XML_ELEMENT_CONTENT_SEQ: {
714 xmlAutomataStatePtr oldstate, oldend;
715 xmlElementContentOccur ocur;
716
717 /*
718 * Simply iterate over the content
719 */
720 oldstate = ctxt->state;
721 ocur = content->ocur;
722 if (ocur != XML_ELEMENT_CONTENT_ONCE) {
723 ctxt->state = xmlAutomataNewEpsilon(ctxt->am, oldstate, NULL);
724 oldstate = ctxt->state;
725 }
726 do {
727 xmlValidBuildAContentModel(content->c1, ctxt, name);
728 content = content->c2;
729 } while ((content->type == XML_ELEMENT_CONTENT_SEQ) &&
730 (content->ocur == XML_ELEMENT_CONTENT_ONCE));
731 xmlValidBuildAContentModel(content, ctxt, name);
732 oldend = ctxt->state;
733 ctxt->state = xmlAutomataNewEpsilon(ctxt->am, oldend, NULL);
734 switch (ocur) {
735 case XML_ELEMENT_CONTENT_ONCE:
736 break;
737 case XML_ELEMENT_CONTENT_OPT:
738 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
739 break;
740 case XML_ELEMENT_CONTENT_MULT:
741 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
742 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
743 break;
744 case XML_ELEMENT_CONTENT_PLUS:
745 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
746 break;
747 }
748 break;
749 }
750 case XML_ELEMENT_CONTENT_OR: {
751 xmlAutomataStatePtr oldstate, oldend;
752 xmlElementContentOccur ocur;
753
754 ocur = content->ocur;
755 if ((ocur == XML_ELEMENT_CONTENT_PLUS) ||
756 (ocur == XML_ELEMENT_CONTENT_MULT)) {
757 ctxt->state = xmlAutomataNewEpsilon(ctxt->am,
758 ctxt->state, NULL);
759 }
760 oldstate = ctxt->state;
761 oldend = xmlAutomataNewState(ctxt->am);
762
763 /*
764 * iterate over the subtypes and remerge the end with an
765 * epsilon transition
766 */
767 do {
768 ctxt->state = oldstate;
769 xmlValidBuildAContentModel(content->c1, ctxt, name);
770 xmlAutomataNewEpsilon(ctxt->am, ctxt->state, oldend);
771 content = content->c2;
772 } while ((content->type == XML_ELEMENT_CONTENT_OR) &&
773 (content->ocur == XML_ELEMENT_CONTENT_ONCE));
774 ctxt->state = oldstate;
775 xmlValidBuildAContentModel(content, ctxt, name);
776 xmlAutomataNewEpsilon(ctxt->am, ctxt->state, oldend);
777 ctxt->state = xmlAutomataNewEpsilon(ctxt->am, oldend, NULL);
778 switch (ocur) {
779 case XML_ELEMENT_CONTENT_ONCE:
780 break;
781 case XML_ELEMENT_CONTENT_OPT:
782 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
783 break;
784 case XML_ELEMENT_CONTENT_MULT:
785 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
786 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
787 break;
788 case XML_ELEMENT_CONTENT_PLUS:
789 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
790 break;
791 }
792 break;
793 }
794 default:
795 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
796 "ContentModel broken for element %s\n",
797 (const char *) name);
798 return(0);
799 }
800 return(1);
801 }
802 /**
803 * xmlValidBuildContentModel:
804 * @ctxt: a validation context
805 * @elem: an element declaration node
806 *
807 * (Re)Build the automata associated to the content model of this
808 * element
809 *
810 * Returns 1 in case of success, 0 in case of error
811 */
812 int
813 xmlValidBuildContentModel(xmlValidCtxtPtr ctxt, xmlElementPtr elem) {
814
815 if ((ctxt == NULL) || (elem == NULL))
816 return(0);
817 if (elem->type != XML_ELEMENT_DECL)
818 return(0);
819 if (elem->etype != XML_ELEMENT_TYPE_ELEMENT)
820 return(1);
821 /* TODO: should we rebuild in this case ? */
822 if (elem->contModel != NULL) {
823 if (!xmlRegexpIsDeterminist(elem->contModel)) {
824 ctxt->valid = 0;
825 return(0);
826 }
827 return(1);
828 }
829
830 ctxt->am = xmlNewAutomata();
831 if (ctxt->am == NULL) {
832 xmlErrValidNode(ctxt, (xmlNodePtr) elem,
833 XML_ERR_INTERNAL_ERROR,
834 "Cannot create automata for element %s\n",
835 elem->name, NULL, NULL);
836 return(0);
837 }
838 ctxt->state = xmlAutomataGetInitState(ctxt->am);
839 xmlValidBuildAContentModel(elem->content, ctxt, elem->name);
840 xmlAutomataSetFinalState(ctxt->am, ctxt->state);
841 elem->contModel = xmlAutomataCompile(ctxt->am);
842 if (xmlRegexpIsDeterminist(elem->contModel) != 1) {
843 char expr[5000];
844 expr[0] = 0;
845 xmlSnprintfElementContent(expr, 5000, elem->content, 1);
846 xmlErrValidNode(ctxt, (xmlNodePtr) elem,
847 XML_DTD_CONTENT_NOT_DETERMINIST,
848 "Content model of %s is not determinist: %s\n",
849 elem->name, BAD_CAST expr, NULL);
850 #ifdef DEBUG_REGEXP_ALGO
851 xmlRegexpPrint(stderr, elem->contModel);
852 #endif
853 ctxt->valid = 0;
854 ctxt->state = NULL;
855 xmlFreeAutomata(ctxt->am);
856 ctxt->am = NULL;
857 return(0);
858 }
859 ctxt->state = NULL;
860 xmlFreeAutomata(ctxt->am);
861 ctxt->am = NULL;
862 return(1);
863 }
864
865 #endif /* LIBXML_REGEXP_ENABLED */
866
867 /****************************************************************
868 * *
869 * Util functions for data allocation/deallocation *
870 * *
871 ****************************************************************/
872
873 /**
874 * xmlNewValidCtxt:
875 *
876 * Allocate a validation context structure.
877 *
878 * Returns NULL if not, otherwise the new validation context structure
879 */
880 xmlValidCtxtPtr xmlNewValidCtxt(void) {
881 xmlValidCtxtPtr ret;
882
883 if ((ret = xmlMalloc(sizeof (xmlValidCtxt))) == NULL) {
884 xmlVErrMemory(NULL, "malloc failed");
885 return (NULL);
886 }
887
888 (void) memset(ret, 0, sizeof (xmlValidCtxt));
889
890 return (ret);
891 }
892
893 /**
894 * xmlFreeValidCtxt:
895 * @cur: the validation context to free
896 *
897 * Free a validation context structure.
898 */
899 void
900 xmlFreeValidCtxt(xmlValidCtxtPtr cur) {
901 if (cur->vstateTab != NULL)
902 xmlFree(cur->vstateTab);
903 if (cur->nodeTab != NULL)
904 xmlFree(cur->nodeTab);
905 xmlFree(cur);
906 }
907
908 #endif /* LIBXML_VALID_ENABLED */
909
910 /**
911 * xmlNewDocElementContent:
912 * @doc: the document
913 * @name: the subelement name or NULL
914 * @type: the type of element content decl
915 *
916 * Allocate an element content structure for the document.
917 *
918 * Returns NULL if not, otherwise the new element content structure
919 */
920 xmlElementContentPtr
921 xmlNewDocElementContent(xmlDocPtr doc, const xmlChar *name,
922 xmlElementContentType type) {
923 xmlElementContentPtr ret;
924 xmlDictPtr dict = NULL;
925
926 if (doc != NULL)
927 dict = doc->dict;
928
929 switch(type) {
930 case XML_ELEMENT_CONTENT_ELEMENT:
931 if (name == NULL) {
932 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
933 "xmlNewElementContent : name == NULL !\n",
934 NULL);
935 }
936 break;
937 case XML_ELEMENT_CONTENT_PCDATA:
938 case XML_ELEMENT_CONTENT_SEQ:
939 case XML_ELEMENT_CONTENT_OR:
940 if (name != NULL) {
941 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
942 "xmlNewElementContent : name != NULL !\n",
943 NULL);
944 }
945 break;
946 default:
947 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
948 "Internal: ELEMENT content corrupted invalid type\n",
949 NULL);
950 return(NULL);
951 }
952 ret = (xmlElementContentPtr) xmlMalloc(sizeof(xmlElementContent));
953 if (ret == NULL) {
954 xmlVErrMemory(NULL, "malloc failed");
955 return(NULL);
956 }
957 memset(ret, 0, sizeof(xmlElementContent));
958 ret->type = type;
959 ret->ocur = XML_ELEMENT_CONTENT_ONCE;
960 if (name != NULL) {
961 int l;
962 const xmlChar *tmp;
963
964 tmp = xmlSplitQName3(name, &l);
965 if (tmp == NULL) {
966 if (dict == NULL)
967 ret->name = xmlStrdup(name);
968 else
969 ret->name = xmlDictLookup(dict, name, -1);
970 } else {
971 if (dict == NULL) {
972 ret->prefix = xmlStrndup(name, l);
973 ret->name = xmlStrdup(tmp);
974 } else {
975 ret->prefix = xmlDictLookup(dict, name, l);
976 ret->name = xmlDictLookup(dict, tmp, -1);
977 }
978 }
979 }
980 return(ret);
981 }
982
983 /**
984 * xmlNewElementContent:
985 * @name: the subelement name or NULL
986 * @type: the type of element content decl
987 *
988 * Allocate an element content structure.
989 * Deprecated in favor of xmlNewDocElementContent
990 *
991 * Returns NULL if not, otherwise the new element content structure
992 */
993 xmlElementContentPtr
994 xmlNewElementContent(const xmlChar *name, xmlElementContentType type) {
995 return(xmlNewDocElementContent(NULL, name, type));
996 }
997
998 /**
999 * xmlCopyDocElementContent:
1000 * @doc: the document owning the element declaration
1001 * @cur: An element content pointer.
1002 *
1003 * Build a copy of an element content description.
1004 *
1005 * Returns the new xmlElementContentPtr or NULL in case of error.
1006 */
1007 xmlElementContentPtr
1008 xmlCopyDocElementContent(xmlDocPtr doc, xmlElementContentPtr cur) {
1009 xmlElementContentPtr ret = NULL, prev = NULL, tmp;
1010 xmlDictPtr dict = NULL;
1011
1012 if (cur == NULL) return(NULL);
1013
1014 if (doc != NULL)
1015 dict = doc->dict;
1016
1017 ret = (xmlElementContentPtr) xmlMalloc(sizeof(xmlElementContent));
1018 if (ret == NULL) {
1019 xmlVErrMemory(NULL, "malloc failed");
1020 return(NULL);
1021 }
1022 memset(ret, 0, sizeof(xmlElementContent));
1023 ret->type = cur->type;
1024 ret->ocur = cur->ocur;
1025 if (cur->name != NULL) {
1026 if (dict)
1027 ret->name = xmlDictLookup(dict, cur->name, -1);
1028 else
1029 ret->name = xmlStrdup(cur->name);
1030 }
1031
1032 if (cur->prefix != NULL) {
1033 if (dict)
1034 ret->prefix = xmlDictLookup(dict, cur->prefix, -1);
1035 else
1036 ret->prefix = xmlStrdup(cur->prefix);
1037 }
1038 if (cur->c1 != NULL)
1039 ret->c1 = xmlCopyDocElementContent(doc, cur->c1);
1040 if (ret->c1 != NULL)
1041 ret->c1->parent = ret;
1042 if (cur->c2 != NULL) {
1043 prev = ret;
1044 cur = cur->c2;
1045 while (cur != NULL) {
1046 tmp = (xmlElementContentPtr) xmlMalloc(sizeof(xmlElementContent));
1047 if (tmp == NULL) {
1048 xmlVErrMemory(NULL, "malloc failed");
1049 return(ret);
1050 }
1051 memset(tmp, 0, sizeof(xmlElementContent));
1052 tmp->type = cur->type;
1053 tmp->ocur = cur->ocur;
1054 prev->c2 = tmp;
1055 if (cur->name != NULL) {
1056 if (dict)
1057 tmp->name = xmlDictLookup(dict, cur->name, -1);
1058 else
1059 tmp->name = xmlStrdup(cur->name);
1060 }
1061
1062 if (cur->prefix != NULL) {
1063 if (dict)
1064 tmp->prefix = xmlDictLookup(dict, cur->prefix, -1);
1065 else
1066 tmp->prefix = xmlStrdup(cur->prefix);
1067 }
1068 if (cur->c1 != NULL)
1069 tmp->c1 = xmlCopyDocElementContent(doc,cur->c1);
1070 if (tmp->c1 != NULL)
1071 tmp->c1->parent = ret;
1072 prev = tmp;
1073 cur = cur->c2;
1074 }
1075 }
1076 return(ret);
1077 }
1078
1079 /**
1080 * xmlCopyElementContent:
1081 * @cur: An element content pointer.
1082 *
1083 * Build a copy of an element content description.
1084 * Deprecated, use xmlCopyDocElementContent instead
1085 *
1086 * Returns the new xmlElementContentPtr or NULL in case of error.
1087 */
1088 xmlElementContentPtr
1089 xmlCopyElementContent(xmlElementContentPtr cur) {
1090 return(xmlCopyDocElementContent(NULL, cur));
1091 }
1092
1093 /**
1094 * xmlFreeDocElementContent:
1095 * @doc: the document owning the element declaration
1096 * @cur: the element content tree to free
1097 *
1098 * Free an element content structure. The whole subtree is removed.
1099 */
1100 void
1101 xmlFreeDocElementContent(xmlDocPtr doc, xmlElementContentPtr cur) {
1102 xmlElementContentPtr next;
1103 xmlDictPtr dict = NULL;
1104
1105 if (doc != NULL)
1106 dict = doc->dict;
1107
1108 while (cur != NULL) {
1109 next = cur->c2;
1110 switch (cur->type) {
1111 case XML_ELEMENT_CONTENT_PCDATA:
1112 case XML_ELEMENT_CONTENT_ELEMENT:
1113 case XML_ELEMENT_CONTENT_SEQ:
1114 case XML_ELEMENT_CONTENT_OR:
1115 break;
1116 default:
1117 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
1118 "Internal: ELEMENT content corrupted invalid type\n",
1119 NULL);
1120 return;
1121 }
1122 if (cur->c1 != NULL) xmlFreeDocElementContent(doc, cur->c1);
1123 if (dict) {
1124 if ((cur->name != NULL) && (!xmlDictOwns(dict, cur->name)))
1125 xmlFree((xmlChar *) cur->name);
1126 if ((cur->prefix != NULL) && (!xmlDictOwns(dict, cur->prefix)))
1127 xmlFree((xmlChar *) cur->prefix);
1128 } else {
1129 if (cur->name != NULL) xmlFree((xmlChar *) cur->name);
1130 if (cur->prefix != NULL) xmlFree((xmlChar *) cur->prefix);
1131 }
1132 xmlFree(cur);
1133 cur = next;
1134 }
1135 }
1136
1137 /**
1138 * xmlFreeElementContent:
1139 * @cur: the element content tree to free
1140 *
1141 * Free an element content structure. The whole subtree is removed.
1142 * Deprecated, use xmlFreeDocElementContent instead
1143 */
1144 void
1145 xmlFreeElementContent(xmlElementContentPtr cur) {
1146 xmlFreeDocElementContent(NULL, cur);
1147 }
1148
1149 #ifdef LIBXML_OUTPUT_ENABLED
1150 /**
1151 * xmlDumpElementContent:
1152 * @buf: An XML buffer
1153 * @content: An element table
1154 * @glob: 1 if one must print the englobing parenthesis, 0 otherwise
1155 *
1156 * This will dump the content of the element table as an XML DTD definition
1157 */
1158 static void
1159 xmlDumpElementContent(xmlBufferPtr buf, xmlElementContentPtr content, int glob) {
1160 if (content == NULL) return;
1161
1162 if (glob) xmlBufferWriteChar(buf, "(");
1163 switch (content->type) {
1164 case XML_ELEMENT_CONTENT_PCDATA:
1165 xmlBufferWriteChar(buf, "#PCDATA");
1166 break;
1167 case XML_ELEMENT_CONTENT_ELEMENT:
1168 if (content->prefix != NULL) {
1169 xmlBufferWriteCHAR(buf, content->prefix);
1170 xmlBufferWriteChar(buf, ":");
1171 }
1172 xmlBufferWriteCHAR(buf, content->name);
1173 break;
1174 case XML_ELEMENT_CONTENT_SEQ:
1175 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1176 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
1177 xmlDumpElementContent(buf, content->c1, 1);
1178 else
1179 xmlDumpElementContent(buf, content->c1, 0);
1180 xmlBufferWriteChar(buf, " , ");
1181 if ((content->c2->type == XML_ELEMENT_CONTENT_OR) ||
1182 ((content->c2->type == XML_ELEMENT_CONTENT_SEQ) &&
1183 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)))
1184 xmlDumpElementContent(buf, content->c2, 1);
1185 else
1186 xmlDumpElementContent(buf, content->c2, 0);
1187 break;
1188 case XML_ELEMENT_CONTENT_OR:
1189 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1190 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
1191 xmlDumpElementContent(buf, content->c1, 1);
1192 else
1193 xmlDumpElementContent(buf, content->c1, 0);
1194 xmlBufferWriteChar(buf, " | ");
1195 if ((content->c2->type == XML_ELEMENT_CONTENT_SEQ) ||
1196 ((content->c2->type == XML_ELEMENT_CONTENT_OR) &&
1197 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)))
1198 xmlDumpElementContent(buf, content->c2, 1);
1199 else
1200 xmlDumpElementContent(buf, content->c2, 0);
1201 break;
1202 default:
1203 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
1204 "Internal: ELEMENT content corrupted invalid type\n",
1205 NULL);
1206 }
1207 if (glob)
1208 xmlBufferWriteChar(buf, ")");
1209 switch (content->ocur) {
1210 case XML_ELEMENT_CONTENT_ONCE:
1211 break;
1212 case XML_ELEMENT_CONTENT_OPT:
1213 xmlBufferWriteChar(buf, "?");
1214 break;
1215 case XML_ELEMENT_CONTENT_MULT:
1216 xmlBufferWriteChar(buf, "*");
1217 break;
1218 case XML_ELEMENT_CONTENT_PLUS:
1219 xmlBufferWriteChar(buf, "+");
1220 break;
1221 }
1222 }
1223
1224 /**
1225 * xmlSprintfElementContent:
1226 * @buf: an output buffer
1227 * @content: An element table
1228 * @englob: 1 if one must print the englobing parenthesis, 0 otherwise
1229 *
1230 * Deprecated, unsafe, use xmlSnprintfElementContent
1231 */
1232 void
1233 xmlSprintfElementContent(char *buf ATTRIBUTE_UNUSED,
1234 xmlElementContentPtr content ATTRIBUTE_UNUSED,
1235 int englob ATTRIBUTE_UNUSED) {
1236 }
1237 #endif /* LIBXML_OUTPUT_ENABLED */
1238
1239 /**
1240 * xmlSnprintfElementContent:
1241 * @buf: an output buffer
1242 * @size: the buffer size
1243 * @content: An element table
1244 * @englob: 1 if one must print the englobing parenthesis, 0 otherwise
1245 *
1246 * This will dump the content of the element content definition
1247 * Intended just for the debug routine
1248 */
1249 void
1250 xmlSnprintfElementContent(char *buf, int size, xmlElementContentPtr content, int englob) {
1251 int len;
1252
1253 if (content == NULL) return;
1254 len = strlen(buf);
1255 if (size - len < 50) {
1256 if ((size - len > 4) && (buf[len - 1] != '.'))
1257 strcat(buf, " ...");
1258 return;
1259 }
1260 if (englob) strcat(buf, "(");
1261 switch (content->type) {
1262 case XML_ELEMENT_CONTENT_PCDATA:
1263 strcat(buf, "#PCDATA");
1264 break;
1265 case XML_ELEMENT_CONTENT_ELEMENT:
1266 if (content->prefix != NULL) {
1267 if (size - len < xmlStrlen(content->prefix) + 10) {
1268 strcat(buf, " ...");
1269 return;
1270 }
1271 strcat(buf, (char *) content->prefix);
1272 strcat(buf, ":");
1273 }
1274 if (size - len < xmlStrlen(content->name) + 10) {
1275 strcat(buf, " ...");
1276 return;
1277 }
1278 if (content->name != NULL)
1279 strcat(buf, (char *) content->name);
1280 break;
1281 case XML_ELEMENT_CONTENT_SEQ:
1282 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1283 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
1284 xmlSnprintfElementContent(buf, size, content->c1, 1);
1285 else
1286 xmlSnprintfElementContent(buf, size, content->c1, 0);
1287 len = strlen(buf);
1288 if (size - len < 50) {
1289 if ((size - len > 4) && (buf[len - 1] != '.'))
1290 strcat(buf, " ...");
1291 return;
1292 }
1293 strcat(buf, " , ");
1294 if (((content->c2->type == XML_ELEMENT_CONTENT_OR) ||
1295 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)) &&
1296 (content->c2->type != XML_ELEMENT_CONTENT_ELEMENT))
1297 xmlSnprintfElementContent(buf, size, content->c2, 1);
1298 else
1299 xmlSnprintfElementContent(buf, size, content->c2, 0);
1300 break;
1301 case XML_ELEMENT_CONTENT_OR:
1302 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1303 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
1304 xmlSnprintfElementContent(buf, size, content->c1, 1);
1305 else
1306 xmlSnprintfElementContent(buf, size, content->c1, 0);
1307 len = strlen(buf);
1308 if (size - len < 50) {
1309 if ((size - len > 4) && (buf[len - 1] != '.'))
1310 strcat(buf, " ...");
1311 return;
1312 }
1313 strcat(buf, " | ");
1314 if (((content->c2->type == XML_ELEMENT_CONTENT_SEQ) ||
1315 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)) &&
1316 (content->c2->type != XML_ELEMENT_CONTENT_ELEMENT))
1317 xmlSnprintfElementContent(buf, size, content->c2, 1);
1318 else
1319 xmlSnprintfElementContent(buf, size, content->c2, 0);
1320 break;
1321 }
1322 if (englob)
1323 strcat(buf, ")");
1324 switch (content->ocur) {
1325 case XML_ELEMENT_CONTENT_ONCE:
1326 break;
1327 case XML_ELEMENT_CONTENT_OPT:
1328 strcat(buf, "?");
1329 break;
1330 case XML_ELEMENT_CONTENT_MULT:
1331 strcat(buf, "*");
1332 break;
1333 case XML_ELEMENT_CONTENT_PLUS:
1334 strcat(buf, "+");
1335 break;
1336 }
1337 }
1338
1339 /****************************************************************
1340 * *
1341 * Registration of DTD declarations *
1342 * *
1343 ****************************************************************/
1344
1345 /**
1346 * xmlFreeElement:
1347 * @elem: An element
1348 *
1349 * Deallocate the memory used by an element definition
1350 */
1351 static void
1352 xmlFreeElement(xmlElementPtr elem) {
1353 if (elem == NULL) return;
1354 xmlUnlinkNode((xmlNodePtr) elem);
1355 xmlFreeDocElementContent(elem->doc, elem->content);
1356 if (elem->name != NULL)
1357 xmlFree((xmlChar *) elem->name);
1358 if (elem->prefix != NULL)
1359 xmlFree((xmlChar *) elem->prefix);
1360 #ifdef LIBXML_REGEXP_ENABLED
1361 if (elem->contModel != NULL)
1362 xmlRegFreeRegexp(elem->contModel);
1363 #endif
1364 xmlFree(elem);
1365 }
1366
1367
1368 /**
1369 * xmlAddElementDecl:
1370 * @ctxt: the validation context
1371 * @dtd: pointer to the DTD
1372 * @name: the entity name
1373 * @type: the element type
1374 * @content: the element content tree or NULL
1375 *
1376 * Register a new element declaration
1377 *
1378 * Returns NULL if not, otherwise the entity
1379 */
1380 xmlElementPtr
1381 xmlAddElementDecl(xmlValidCtxtPtr ctxt,
1382 xmlDtdPtr dtd, const xmlChar *name,
1383 xmlElementTypeVal type,
1384 xmlElementContentPtr content) {
1385 xmlElementPtr ret;
1386 xmlElementTablePtr table;
1387 xmlAttributePtr oldAttributes = NULL;
1388 xmlChar *ns, *uqname;
1389
1390 if (dtd == NULL) {
1391 return(NULL);
1392 }
1393 if (name == NULL) {
1394 return(NULL);
1395 }
1396
1397 switch (type) {
1398 case XML_ELEMENT_TYPE_EMPTY:
1399 if (content != NULL) {
1400 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
1401 "xmlAddElementDecl: content != NULL for EMPTY\n",
1402 NULL);
1403 return(NULL);
1404 }
1405 break;
1406 case XML_ELEMENT_TYPE_ANY:
1407 if (content != NULL) {
1408 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
1409 "xmlAddElementDecl: content != NULL for ANY\n",
1410 NULL);
1411 return(NULL);
1412 }
1413 break;
1414 case XML_ELEMENT_TYPE_MIXED:
1415 if (content == NULL) {
1416 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
1417 "xmlAddElementDecl: content == NULL for MIXED\n",
1418 NULL);
1419 return(NULL);
1420 }
1421 break;
1422 case XML_ELEMENT_TYPE_ELEMENT:
1423 if (content == NULL) {
1424 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
1425 "xmlAddElementDecl: content == NULL for ELEMENT\n",
1426 NULL);
1427 return(NULL);
1428 }
1429 break;
1430 default:
1431 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
1432 "Internal: ELEMENT decl corrupted invalid type\n",
1433 NULL);
1434 return(NULL);
1435 }
1436
1437 /*
1438 * check if name is a QName
1439 */
1440 uqname = xmlSplitQName2(name, &ns);
1441 if (uqname != NULL)
1442 name = uqname;
1443
1444 /*
1445 * Create the Element table if needed.
1446 */
1447 table = (xmlElementTablePtr) dtd->elements;
1448 if (table == NULL) {
1449 xmlDictPtr dict = NULL;
1450
1451 if (dtd->doc != NULL)
1452 dict = dtd->doc->dict;
1453 table = xmlHashCreateDict(0, dict);
1454 dtd->elements = (void *) table;
1455 }
1456 if (table == NULL) {
1457 xmlVErrMemory(ctxt,
1458 "xmlAddElementDecl: Table creation failed!\n");
1459 if (uqname != NULL)
1460 xmlFree(uqname);
1461 if (ns != NULL)
1462 xmlFree(ns);
1463 return(NULL);
1464 }
1465
1466 /*
1467 * lookup old attributes inserted on an undefined element in the
1468 * internal subset.
1469 */
1470 if ((dtd->doc != NULL) && (dtd->doc->intSubset != NULL)) {
1471 ret = xmlHashLookup2(dtd->doc->intSubset->elements, name, ns);
1472 if ((ret != NULL) && (ret->etype == XML_ELEMENT_TYPE_UNDEFINED)) {
1473 oldAttributes = ret->attributes;
1474 ret->attributes = NULL;
1475 xmlHashRemoveEntry2(dtd->doc->intSubset->elements, name, ns, NULL);
1476 xmlFreeElement(ret);
1477 }
1478 }
1479
1480 /*
1481 * The element may already be present if one of its attribute
1482 * was registered first
1483 */
1484 ret = xmlHashLookup2(table, name, ns);
1485 if (ret != NULL) {
1486 if (ret->etype != XML_ELEMENT_TYPE_UNDEFINED) {
1487 #ifdef LIBXML_VALID_ENABLED
1488 /*
1489 * The element is already defined in this DTD.
1490 */
1491 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ELEM_REDEFINED,
1492 "Redefinition of element %s\n",
1493 name, NULL, NULL);
1494 #endif /* LIBXML_VALID_ENABLED */
1495 if (uqname != NULL)
1496 xmlFree(uqname);
1497 if (ns != NULL)
1498 xmlFree(ns);
1499 return(NULL);
1500 }
1501 if (ns != NULL) {
1502 xmlFree(ns);
1503 ns = NULL;
1504 }
1505 } else {
1506 ret = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
1507 if (ret == NULL) {
1508 xmlVErrMemory(ctxt, "malloc failed");
1509 if (uqname != NULL)
1510 xmlFree(uqname);
1511 if (ns != NULL)
1512 xmlFree(ns);
1513 return(NULL);
1514 }
1515 memset(ret, 0, sizeof(xmlElement));
1516 ret->type = XML_ELEMENT_DECL;
1517
1518 /*
1519 * fill the structure.
1520 */
1521 ret->name = xmlStrdup(name);
1522 if (ret->name == NULL) {
1523 xmlVErrMemory(ctxt, "malloc failed");
1524 if (uqname != NULL)
1525 xmlFree(uqname);
1526 if (ns != NULL)
1527 xmlFree(ns);
1528 xmlFree(ret);
1529 return(NULL);
1530 }
1531 ret->prefix = ns;
1532
1533 /*
1534 * Validity Check:
1535 * Insertion must not fail
1536 */
1537 if (xmlHashAddEntry2(table, name, ns, ret)) {
1538 #ifdef LIBXML_VALID_ENABLED
1539 /*
1540 * The element is already defined in this DTD.
1541 */
1542 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ELEM_REDEFINED,
1543 "Redefinition of element %s\n",
1544 name, NULL, NULL);
1545 #endif /* LIBXML_VALID_ENABLED */
1546 xmlFreeElement(ret);
1547 if (uqname != NULL)
1548 xmlFree(uqname);
1549 return(NULL);
1550 }
1551 /*
1552 * For new element, may have attributes from earlier
1553 * definition in internal subset
1554 */
1555 ret->attributes = oldAttributes;
1556 }
1557
1558 /*
1559 * Finish to fill the structure.
1560 */
1561 ret->etype = type;
1562 /*
1563 * Avoid a stupid copy when called by the parser
1564 * and flag it by setting a special parent value
1565 * so the parser doesn't unallocate it.
1566 */
1567 if ((ctxt != NULL) &&
1568 ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
1569 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1))) {
1570 ret->content = content;
1571 if (content != NULL)
1572 content->parent = (xmlElementContentPtr) 1;
1573 } else {
1574 ret->content = xmlCopyDocElementContent(dtd->doc, content);
1575 }
1576
1577 /*
1578 * Link it to the DTD
1579 */
1580 ret->parent = dtd;
1581 ret->doc = dtd->doc;
1582 if (dtd->last == NULL) {
1583 dtd->children = dtd->last = (xmlNodePtr) ret;
1584 } else {
1585 dtd->last->next = (xmlNodePtr) ret;
1586 ret->prev = dtd->last;
1587 dtd->last = (xmlNodePtr) ret;
1588 }
1589 if (uqname != NULL)
1590 xmlFree(uqname);
1591 return(ret);
1592 }
1593
1594 /**
1595 * xmlFreeElementTable:
1596 * @table: An element table
1597 *
1598 * Deallocate the memory used by an element hash table.
1599 */
1600 void
1601 xmlFreeElementTable(xmlElementTablePtr table) {
1602 xmlHashFree(table, (xmlHashDeallocator) xmlFreeElement);
1603 }
1604
1605 #ifdef LIBXML_TREE_ENABLED
1606 /**
1607 * xmlCopyElement:
1608 * @elem: An element
1609 *
1610 * Build a copy of an element.
1611 *
1612 * Returns the new xmlElementPtr or NULL in case of error.
1613 */
1614 static xmlElementPtr
1615 xmlCopyElement(xmlElementPtr elem) {
1616 xmlElementPtr cur;
1617
1618 cur = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
1619 if (cur == NULL) {
1620 xmlVErrMemory(NULL, "malloc failed");
1621 return(NULL);
1622 }
1623 memset(cur, 0, sizeof(xmlElement));
1624 cur->type = XML_ELEMENT_DECL;
1625 cur->etype = elem->etype;
1626 if (elem->name != NULL)
1627 cur->name = xmlStrdup(elem->name);
1628 else
1629 cur->name = NULL;
1630 if (elem->prefix != NULL)
1631 cur->prefix = xmlStrdup(elem->prefix);
1632 else
1633 cur->prefix = NULL;
1634 cur->content = xmlCopyElementContent(elem->content);
1635 /* TODO : rebuild the attribute list on the copy */
1636 cur->attributes = NULL;
1637 return(cur);
1638 }
1639
1640 /**
1641 * xmlCopyElementTable:
1642 * @table: An element table
1643 *
1644 * Build a copy of an element table.
1645 *
1646 * Returns the new xmlElementTablePtr or NULL in case of error.
1647 */
1648 xmlElementTablePtr
1649 xmlCopyElementTable(xmlElementTablePtr table) {
1650 return((xmlElementTablePtr) xmlHashCopy(table,
1651 (xmlHashCopier) xmlCopyElement));
1652 }
1653 #endif /* LIBXML_TREE_ENABLED */
1654
1655 #ifdef LIBXML_OUTPUT_ENABLED
1656 /**
1657 * xmlDumpElementDecl:
1658 * @buf: the XML buffer output
1659 * @elem: An element table
1660 *
1661 * This will dump the content of the element declaration as an XML
1662 * DTD definition
1663 */
1664 void
1665 xmlDumpElementDecl(xmlBufferPtr buf, xmlElementPtr elem) {
1666 if ((buf == NULL) || (elem == NULL))
1667 return;
1668 switch (elem->etype) {
1669 case XML_ELEMENT_TYPE_EMPTY:
1670 xmlBufferWriteChar(buf, "<!ELEMENT ");
1671 if (elem->prefix != NULL) {
1672 xmlBufferWriteCHAR(buf, elem->prefix);
1673 xmlBufferWriteChar(buf, ":");
1674 }
1675 xmlBufferWriteCHAR(buf, elem->name);
1676 xmlBufferWriteChar(buf, " EMPTY>\n");
1677 break;
1678 case XML_ELEMENT_TYPE_ANY:
1679 xmlBufferWriteChar(buf, "<!ELEMENT ");
1680 if (elem->prefix != NULL) {
1681 xmlBufferWriteCHAR(buf, elem->prefix);
1682 xmlBufferWriteChar(buf, ":");
1683 }
1684 xmlBufferWriteCHAR(buf, elem->name);
1685 xmlBufferWriteChar(buf, " ANY>\n");
1686 break;
1687 case XML_ELEMENT_TYPE_MIXED:
1688 xmlBufferWriteChar(buf, "<!ELEMENT ");
1689 if (elem->prefix != NULL) {
1690 xmlBufferWriteCHAR(buf, elem->prefix);
1691 xmlBufferWriteChar(buf, ":");
1692 }
1693 xmlBufferWriteCHAR(buf, elem->name);
1694 xmlBufferWriteChar(buf, " ");
1695 xmlDumpElementContent(buf, elem->content, 1);
1696 xmlBufferWriteChar(buf, ">\n");
1697 break;
1698 case XML_ELEMENT_TYPE_ELEMENT:
1699 xmlBufferWriteChar(buf, "<!ELEMENT ");
1700 if (elem->prefix != NULL) {
1701 xmlBufferWriteCHAR(buf, elem->prefix);
1702 xmlBufferWriteChar(buf, ":");
1703 }
1704 xmlBufferWriteCHAR(buf, elem->name);
1705 xmlBufferWriteChar(buf, " ");
1706 xmlDumpElementContent(buf, elem->content, 1);
1707 xmlBufferWriteChar(buf, ">\n");
1708 break;
1709 default:
1710 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
1711 "Internal: ELEMENT struct corrupted invalid type\n",
1712 NULL);
1713 }
1714 }
1715
1716 /**
1717 * xmlDumpElementDeclScan:
1718 * @elem: An element table
1719 * @buf: the XML buffer output
1720 *
1721 * This routine is used by the hash scan function. It just reverses
1722 * the arguments.
1723 */
1724 static void
1725 xmlDumpElementDeclScan(xmlElementPtr elem, xmlBufferPtr buf) {
1726 xmlDumpElementDecl(buf, elem);
1727 }
1728
1729 /**
1730 * xmlDumpElementTable:
1731 * @buf: the XML buffer output
1732 * @table: An element table
1733 *
1734 * This will dump the content of the element table as an XML DTD definition
1735 */
1736 void
1737 xmlDumpElementTable(xmlBufferPtr buf, xmlElementTablePtr table) {
1738 if ((buf == NULL) || (table == NULL))
1739 return;
1740 xmlHashScan(table, (xmlHashScanner) xmlDumpElementDeclScan, buf);
1741 }
1742 #endif /* LIBXML_OUTPUT_ENABLED */
1743
1744 /**
1745 * xmlCreateEnumeration:
1746 * @name: the enumeration name or NULL
1747 *
1748 * create and initialize an enumeration attribute node.
1749 *
1750 * Returns the xmlEnumerationPtr just created or NULL in case
1751 * of error.
1752 */
1753 xmlEnumerationPtr
1754 xmlCreateEnumeration(const xmlChar *name) {
1755 xmlEnumerationPtr ret;
1756
1757 ret = (xmlEnumerationPtr) xmlMalloc(sizeof(xmlEnumeration));
1758 if (ret == NULL) {
1759 xmlVErrMemory(NULL, "malloc failed");
1760 return(NULL);
1761 }
1762 memset(ret, 0, sizeof(xmlEnumeration));
1763
1764 if (name != NULL)
1765 ret->name = xmlStrdup(name);
1766 return(ret);
1767 }
1768
1769 /**
1770 * xmlFreeEnumeration:
1771 * @cur: the tree to free.
1772 *
1773 * free an enumeration attribute node (recursive).
1774 */
1775 void
1776 xmlFreeEnumeration(xmlEnumerationPtr cur) {
1777 if (cur == NULL) return;
1778
1779 if (cur->next != NULL) xmlFreeEnumeration(cur->next);
1780
1781 if (cur->name != NULL) xmlFree((xmlChar *) cur->name);
1782 xmlFree(cur);
1783 }
1784
1785 #ifdef LIBXML_TREE_ENABLED
1786 /**
1787 * xmlCopyEnumeration:
1788 * @cur: the tree to copy.
1789 *
1790 * Copy an enumeration attribute node (recursive).
1791 *
1792 * Returns the xmlEnumerationPtr just created or NULL in case
1793 * of error.
1794 */
1795 xmlEnumerationPtr
1796 xmlCopyEnumeration(xmlEnumerationPtr cur) {
1797 xmlEnumerationPtr ret;
1798
1799 if (cur == NULL) return(NULL);
1800 ret = xmlCreateEnumeration((xmlChar *) cur->name);
1801 if (ret == NULL) return(NULL);
1802
1803 if (cur->next != NULL) ret->next = xmlCopyEnumeration(cur->next);
1804 else ret->next = NULL;
1805
1806 return(ret);
1807 }
1808 #endif /* LIBXML_TREE_ENABLED */
1809
1810 #ifdef LIBXML_OUTPUT_ENABLED
1811 /**
1812 * xmlDumpEnumeration:
1813 * @buf: the XML buffer output
1814 * @enum: An enumeration
1815 *
1816 * This will dump the content of the enumeration
1817 */
1818 static void
1819 xmlDumpEnumeration(xmlBufferPtr buf, xmlEnumerationPtr cur) {
1820 if ((buf == NULL) || (cur == NULL))
1821 return;
1822
1823 xmlBufferWriteCHAR(buf, cur->name);
1824 if (cur->next == NULL)
1825 xmlBufferWriteChar(buf, ")");
1826 else {
1827 xmlBufferWriteChar(buf, " | ");
1828 xmlDumpEnumeration(buf, cur->next);
1829 }
1830 }
1831 #endif /* LIBXML_OUTPUT_ENABLED */
1832
1833 #ifdef LIBXML_VALID_ENABLED
1834 /**
1835 * xmlScanIDAttributeDecl:
1836 * @ctxt: the validation context
1837 * @elem: the element name
1838 * @err: whether to raise errors here
1839 *
1840 * Verify that the element don't have too many ID attributes
1841 * declared.
1842 *
1843 * Returns the number of ID attributes found.
1844 */
1845 static int
1846 xmlScanIDAttributeDecl(xmlValidCtxtPtr ctxt, xmlElementPtr elem, int err) {
1847 xmlAttributePtr cur;
1848 int ret = 0;
1849
1850 if (elem == NULL) return(0);
1851 cur = elem->attributes;
1852 while (cur != NULL) {
1853 if (cur->atype == XML_ATTRIBUTE_ID) {
1854 ret ++;
1855 if ((ret > 1) && (err))
1856 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_MULTIPLE_ID,
1857 "Element %s has too many ID attributes defined : %s\n",
1858 elem->name, cur->name, NULL);
1859 }
1860 cur = cur->nexth;
1861 }
1862 return(ret);
1863 }
1864 #endif /* LIBXML_VALID_ENABLED */
1865
1866 /**
1867 * xmlFreeAttribute:
1868 * @elem: An attribute
1869 *
1870 * Deallocate the memory used by an attribute definition
1871 */
1872 static void
1873 xmlFreeAttribute(xmlAttributePtr attr) {
1874 xmlDictPtr dict;
1875
1876 if (attr == NULL) return;
1877 if (attr->doc != NULL)
1878 dict = attr->doc->dict;
1879 else
1880 dict = NULL;
1881 xmlUnlinkNode((xmlNodePtr) attr);
1882 if (attr->tree != NULL)
1883 xmlFreeEnumeration(attr->tree);
1884 if (dict) {
1885 if ((attr->elem != NULL) && (!xmlDictOwns(dict, attr->elem)))
1886 xmlFree((xmlChar *) attr->elem);
1887 if ((attr->name != NULL) && (!xmlDictOwns(dict, attr->name)))
1888 xmlFree((xmlChar *) attr->name);
1889 if ((attr->prefix != NULL) && (!xmlDictOwns(dict, attr->prefix)))
1890 xmlFree((xmlChar *) attr->prefix);
1891 if ((attr->defaultValue != NULL) &&
1892 (!xmlDictOwns(dict, attr->defaultValue)))
1893 xmlFree((xmlChar *) attr->defaultValue);
1894 } else {
1895 if (attr->elem != NULL)
1896 xmlFree((xmlChar *) attr->elem);
1897 if (attr->name != NULL)
1898 xmlFree((xmlChar *) attr->name);
1899 if (attr->defaultValue != NULL)
1900 xmlFree((xmlChar *) attr->defaultValue);
1901 if (attr->prefix != NULL)
1902 xmlFree((xmlChar *) attr->prefix);
1903 }
1904 xmlFree(attr);
1905 }
1906
1907
1908 /**
1909 * xmlAddAttributeDecl:
1910 * @ctxt: the validation context
1911 * @dtd: pointer to the DTD
1912 * @elem: the element name
1913 * @name: the attribute name
1914 * @ns: the attribute namespace prefix
1915 * @type: the attribute type
1916 * @def: the attribute default type
1917 * @defaultValue: the attribute default value
1918 * @tree: if it's an enumeration, the associated list
1919 *
1920 * Register a new attribute declaration
1921 * Note that @tree becomes the ownership of the DTD
1922 *
1923 * Returns NULL if not new, otherwise the attribute decl
1924 */
1925 xmlAttributePtr
1926 xmlAddAttributeDecl(xmlValidCtxtPtr ctxt,
1927 xmlDtdPtr dtd, const xmlChar *elem,
1928 const xmlChar *name, const xmlChar *ns,
1929 xmlAttributeType type, xmlAttributeDefault def,
1930 const xmlChar *defaultValue, xmlEnumerationPtr tree) {
1931 xmlAttributePtr ret;
1932 xmlAttributeTablePtr table;
1933 xmlElementPtr elemDef;
1934 xmlDictPtr dict = NULL;
1935
1936 if (dtd == NULL) {
1937 xmlFreeEnumeration(tree);
1938 return(NULL);
1939 }
1940 if (name == NULL) {
1941 xmlFreeEnumeration(tree);
1942 return(NULL);
1943 }
1944 if (elem == NULL) {
1945 xmlFreeEnumeration(tree);
1946 return(NULL);
1947 }
1948 if (dtd->doc != NULL)
1949 dict = dtd->doc->dict;
1950
1951 #ifdef LIBXML_VALID_ENABLED
1952 /*
1953 * Check the type and possibly the default value.
1954 */
1955 switch (type) {
1956 case XML_ATTRIBUTE_CDATA:
1957 break;
1958 case XML_ATTRIBUTE_ID:
1959 break;
1960 case XML_ATTRIBUTE_IDREF:
1961 break;
1962 case XML_ATTRIBUTE_IDREFS:
1963 break;
1964 case XML_ATTRIBUTE_ENTITY:
1965 break;
1966 case XML_ATTRIBUTE_ENTITIES:
1967 break;
1968 case XML_ATTRIBUTE_NMTOKEN:
1969 break;
1970 case XML_ATTRIBUTE_NMTOKENS:
1971 break;
1972 case XML_ATTRIBUTE_ENUMERATION:
1973 break;
1974 case XML_ATTRIBUTE_NOTATION:
1975 break;
1976 default:
1977 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
1978 "Internal: ATTRIBUTE struct corrupted invalid type\n",
1979 NULL);
1980 xmlFreeEnumeration(tree);
1981 return(NULL);
1982 }
1983 if ((defaultValue != NULL) &&
1984 (!xmlValidateAttributeValueInternal(dtd->doc, type, defaultValue))) {
1985 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ATTRIBUTE_DEFAULT,
1986 "Attribute %s of %s: invalid default value\n",
1987 elem, name, defaultValue);
1988 defaultValue = NULL;
1989 if (ctxt != NULL)
1990 ctxt->valid = 0;
1991 }
1992 #endif /* LIBXML_VALID_ENABLED */
1993
1994 /*
1995 * Check first that an attribute defined in the external subset wasn't
1996 * already defined in the internal subset
1997 */
1998 if ((dtd->doc != NULL) && (dtd->doc->extSubset == dtd) &&
1999 (dtd->doc->intSubset != NULL) &&
2000 (dtd->doc->intSubset->attributes != NULL)) {
2001 ret = xmlHashLookup3(dtd->doc->intSubset->attributes, name, ns, elem);
2002 if (ret != NULL) {
2003 xmlFreeEnumeration(tree);
2004 return(NULL);
2005 }
2006 }
2007
2008 /*
2009 * Create the Attribute table if needed.
2010 */
2011 table = (xmlAttributeTablePtr) dtd->attributes;
2012 if (table == NULL) {
2013 table = xmlHashCreateDict(0, dict);
2014 dtd->attributes = (void *) table;
2015 }
2016 if (table == NULL) {
2017 xmlVErrMemory(ctxt,
2018 "xmlAddAttributeDecl: Table creation failed!\n");
2019 xmlFreeEnumeration(tree);
2020 return(NULL);
2021 }
2022
2023
2024 ret = (xmlAttributePtr) xmlMalloc(sizeof(xmlAttribute));
2025 if (ret == NULL) {
2026 xmlVErrMemory(ctxt, "malloc failed");
2027 xmlFreeEnumeration(tree);
2028 return(NULL);
2029 }
2030 memset(ret, 0, sizeof(xmlAttribute));
2031 ret->type = XML_ATTRIBUTE_DECL;
2032
2033 /*
2034 * fill the structure.
2035 */
2036 ret->atype = type;
2037 /*
2038 * doc must be set before possible error causes call
2039 * to xmlFreeAttribute (because it's used to check on
2040 * dict use)
2041 */
2042 ret->doc = dtd->doc;
2043 if (dict) {
2044 ret->name = xmlDictLookup(dict, name, -1);
2045 ret->prefix = xmlDictLookup(dict, ns, -1);
2046 ret->elem = xmlDictLookup(dict, elem, -1);
2047 } else {
2048 ret->name = xmlStrdup(name);
2049 ret->prefix = xmlStrdup(ns);
2050 ret->elem = xmlStrdup(elem);
2051 }
2052 ret->def = def;
2053 ret->tree = tree;
2054 if (defaultValue != NULL) {
2055 if (dict)
2056 ret->defaultValue = xmlDictLookup(dict, defaultValue, -1);
2057 else
2058 ret->defaultValue = xmlStrdup(defaultValue);
2059 }
2060
2061 /*
2062 * Validity Check:
2063 * Search the DTD for previous declarations of the ATTLIST
2064 */
2065 if (xmlHashAddEntry3(table, ret->name, ret->prefix, ret->elem, ret) < 0) {
2066 #ifdef LIBXML_VALID_ENABLED
2067 /*
2068 * The attribute is already defined in this DTD.
2069 */
2070 xmlErrValidWarning(ctxt, (xmlNodePtr) dtd, XML_DTD_ATTRIBUTE_REDEFINED,
2071 "Attribute %s of element %s: already defined\n",
2072 name, elem, NULL);
2073 #endif /* LIBXML_VALID_ENABLED */
2074 xmlFreeAttribute(ret);
2075 return(NULL);
2076 }
2077
2078 /*
2079 * Validity Check:
2080 * Multiple ID per element
2081 */
2082 elemDef = xmlGetDtdElementDesc2(dtd, elem, 1);
2083 if (elemDef != NULL) {
2084
2085 #ifdef LIBXML_VALID_ENABLED
2086 if ((type == XML_ATTRIBUTE_ID) &&
2087 (xmlScanIDAttributeDecl(NULL, elemDef, 1) != 0)) {
2088 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_MULTIPLE_ID,
2089 "Element %s has too may ID attributes defined : %s\n",
2090 elem, name, NULL);
2091 if (ctxt != NULL)
2092 ctxt->valid = 0;
2093 }
2094 #endif /* LIBXML_VALID_ENABLED */
2095
2096 /*
2097 * Insert namespace default def first they need to be
2098 * processed first.
2099 */
2100 if ((xmlStrEqual(ret->name, BAD_CAST "xmlns")) ||
2101 ((ret->prefix != NULL &&
2102 (xmlStrEqual(ret->prefix, BAD_CAST "xmlns"))))) {
2103 ret->nexth = elemDef->attributes;
2104 elemDef->attributes = ret;
2105 } else {
2106 xmlAttributePtr tmp = elemDef->attributes;
2107
2108 while ((tmp != NULL) &&
2109 ((xmlStrEqual(tmp->name, BAD_CAST "xmlns")) ||
2110 ((ret->prefix != NULL &&
2111 (xmlStrEqual(ret->prefix, BAD_CAST "xmlns")))))) {
2112 if (tmp->nexth == NULL)
2113 break;
2114 tmp = tmp->nexth;
2115 }
2116 if (tmp != NULL) {
2117 ret->nexth = tmp->nexth;
2118 tmp->nexth = ret;
2119 } else {
2120 ret->nexth = elemDef->attributes;
2121 elemDef->attributes = ret;
2122 }
2123 }
2124 }
2125
2126 /*
2127 * Link it to the DTD
2128 */
2129 ret->parent = dtd;
2130 if (dtd->last == NULL) {
2131 dtd->children = dtd->last = (xmlNodePtr) ret;
2132 } else {
2133 dtd->last->next = (xmlNodePtr) ret;
2134 ret->prev = dtd->last;
2135 dtd->last = (xmlNodePtr) ret;
2136 }
2137 return(ret);
2138 }
2139
2140 /**
2141 * xmlFreeAttributeTable:
2142 * @table: An attribute table
2143 *
2144 * Deallocate the memory used by an entities hash table.
2145 */
2146 void
2147 xmlFreeAttributeTable(xmlAttributeTablePtr table) {
2148 xmlHashFree(table, (xmlHashDeallocator) xmlFreeAttribute);
2149 }
2150
2151 #ifdef LIBXML_TREE_ENABLED
2152 /**
2153 * xmlCopyAttribute:
2154 * @attr: An attribute
2155 *
2156 * Build a copy of an attribute.
2157 *
2158 * Returns the new xmlAttributePtr or NULL in case of error.
2159 */
2160 static xmlAttributePtr
2161 xmlCopyAttribute(xmlAttributePtr attr) {
2162 xmlAttributePtr cur;
2163
2164 cur = (xmlAttributePtr) xmlMalloc(sizeof(xmlAttribute));
2165 if (cur == NULL) {
2166 xmlVErrMemory(NULL, "malloc failed");
2167 return(NULL);
2168 }
2169 memset(cur, 0, sizeof(xmlAttribute));
2170 cur->type = XML_ATTRIBUTE_DECL;
2171 cur->atype = attr->atype;
2172 cur->def = attr->def;
2173 cur->tree = xmlCopyEnumeration(attr->tree);
2174 if (attr->elem != NULL)
2175 cur->elem = xmlStrdup(attr->elem);
2176 if (attr->name != NULL)
2177 cur->name = xmlStrdup(attr->name);
2178 if (attr->prefix != NULL)
2179 cur->prefix = xmlStrdup(attr->prefix);
2180 if (attr->defaultValue != NULL)
2181 cur->defaultValue = xmlStrdup(attr->defaultValue);
2182 return(cur);
2183 }
2184
2185 /**
2186 * xmlCopyAttributeTable:
2187 * @table: An attribute table
2188 *
2189 * Build a copy of an attribute table.
2190 *
2191 * Returns the new xmlAttributeTablePtr or NULL in case of error.
2192 */
2193 xmlAttributeTablePtr
2194 xmlCopyAttributeTable(xmlAttributeTablePtr table) {
2195 return((xmlAttributeTablePtr) xmlHashCopy(table,
2196 (xmlHashCopier) xmlCopyAttribute));
2197 }
2198 #endif /* LIBXML_TREE_ENABLED */
2199
2200 #ifdef LIBXML_OUTPUT_ENABLED
2201 /**
2202 * xmlDumpAttributeDecl:
2203 * @buf: the XML buffer output
2204 * @attr: An attribute declaration
2205 *
2206 * This will dump the content of the attribute declaration as an XML
2207 * DTD definition
2208 */
2209 void
2210 xmlDumpAttributeDecl(xmlBufferPtr buf, xmlAttributePtr attr) {
2211 if ((buf == NULL) || (attr == NULL))
2212 return;
2213 xmlBufferWriteChar(buf, "<!ATTLIST ");
2214 xmlBufferWriteCHAR(buf, attr->elem);
2215 xmlBufferWriteChar(buf, " ");
2216 if (attr->prefix != NULL) {
2217 xmlBufferWriteCHAR(buf, attr->prefix);
2218 xmlBufferWriteChar(buf, ":");
2219 }
2220 xmlBufferWriteCHAR(buf, attr->name);
2221 switch (attr->atype) {
2222 case XML_ATTRIBUTE_CDATA:
2223 xmlBufferWriteChar(buf, " CDATA");
2224 break;
2225 case XML_ATTRIBUTE_ID:
2226 xmlBufferWriteChar(buf, " ID");
2227 break;
2228 case XML_ATTRIBUTE_IDREF:
2229 xmlBufferWriteChar(buf, " IDREF");
2230 break;
2231 case XML_ATTRIBUTE_IDREFS:
2232 xmlBufferWriteChar(buf, " IDREFS");
2233 break;
2234 case XML_ATTRIBUTE_ENTITY:
2235 xmlBufferWriteChar(buf, " ENTITY");
2236 break;
2237 case XML_ATTRIBUTE_ENTITIES:
2238 xmlBufferWriteChar(buf, " ENTITIES");
2239 break;
2240 case XML_ATTRIBUTE_NMTOKEN:
2241 xmlBufferWriteChar(buf, " NMTOKEN");
2242 break;
2243 case XML_ATTRIBUTE_NMTOKENS:
2244 xmlBufferWriteChar(buf, " NMTOKENS");
2245 break;
2246 case XML_ATTRIBUTE_ENUMERATION:
2247 xmlBufferWriteChar(buf, " (");
2248 xmlDumpEnumeration(buf, attr->tree);
2249 break;
2250 case XML_ATTRIBUTE_NOTATION:
2251 xmlBufferWriteChar(buf, " NOTATION (");
2252 xmlDumpEnumeration(buf, attr->tree);
2253 break;
2254 default:
2255 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2256 "Internal: ATTRIBUTE struct corrupted invalid type\n",
2257 NULL);
2258 }
2259 switch (attr->def) {
2260 case XML_ATTRIBUTE_NONE:
2261 break;
2262 case XML_ATTRIBUTE_REQUIRED:
2263 xmlBufferWriteChar(buf, " #REQUIRED");
2264 break;
2265 case XML_ATTRIBUTE_IMPLIED:
2266 xmlBufferWriteChar(buf, " #IMPLIED");
2267 break;
2268 case XML_ATTRIBUTE_FIXED:
2269 xmlBufferWriteChar(buf, " #FIXED");
2270 break;
2271 default:
2272 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2273 "Internal: ATTRIBUTE struct corrupted invalid def\n",
2274 NULL);
2275 }
2276 if (attr->defaultValue != NULL) {
2277 xmlBufferWriteChar(buf, " ");
2278 xmlBufferWriteQuotedString(buf, attr->defaultValue);
2279 }
2280 xmlBufferWriteChar(buf, ">\n");
2281 }
2282
2283 /**
2284 * xmlDumpAttributeDeclScan:
2285 * @attr: An attribute declaration
2286 * @buf: the XML buffer output
2287 *
2288 * This is used with the hash scan function - just reverses arguments
2289 */
2290 static void
2291 xmlDumpAttributeDeclScan(xmlAttributePtr attr, xmlBufferPtr buf) {
2292 xmlDumpAttributeDecl(buf, attr);
2293 }
2294
2295 /**
2296 * xmlDumpAttributeTable:
2297 * @buf: the XML buffer output
2298 * @table: An attribute table
2299 *
2300 * This will dump the content of the attribute table as an XML DTD definition
2301 */
2302 void
2303 xmlDumpAttributeTable(xmlBufferPtr buf, xmlAttributeTablePtr table) {
2304 if ((buf == NULL) || (table == NULL))
2305 return;
2306 xmlHashScan(table, (xmlHashScanner) xmlDumpAttributeDeclScan, buf);
2307 }
2308 #endif /* LIBXML_OUTPUT_ENABLED */
2309
2310 /************************************************************************
2311 * *
2312 * NOTATIONs *
2313 * *
2314 ************************************************************************/
2315 /**
2316 * xmlFreeNotation:
2317 * @not: A notation
2318 *
2319 * Deallocate the memory used by an notation definition
2320 */
2321 static void
2322 xmlFreeNotation(xmlNotationPtr nota) {
2323 if (nota == NULL) return;
2324 if (nota->name != NULL)
2325 xmlFree((xmlChar *) nota->name);
2326 if (nota->PublicID != NULL)
2327 xmlFree((xmlChar *) nota->PublicID);
2328 if (nota->SystemID != NULL)
2329 xmlFree((xmlChar *) nota->SystemID);
2330 xmlFree(nota);
2331 }
2332
2333
2334 /**
2335 * xmlAddNotationDecl:
2336 * @dtd: pointer to the DTD
2337 * @ctxt: the validation context
2338 * @name: the entity name
2339 * @PublicID: the public identifier or NULL
2340 * @SystemID: the system identifier or NULL
2341 *
2342 * Register a new notation declaration
2343 *
2344 * Returns NULL if not, otherwise the entity
2345 */
2346 xmlNotationPtr
2347 xmlAddNotationDecl(xmlValidCtxtPtr ctxt, xmlDtdPtr dtd,
2348 const xmlChar *name,
2349 const xmlChar *PublicID, const xmlChar *SystemID) {
2350 xmlNotationPtr ret;
2351 xmlNotationTablePtr table;
2352
2353 if (dtd == NULL) {
2354 return(NULL);
2355 }
2356 if (name == NULL) {
2357 return(NULL);
2358 }
2359 if ((PublicID == NULL) && (SystemID == NULL)) {
2360 return(NULL);
2361 }
2362
2363 /*
2364 * Create the Notation table if needed.
2365 */
2366 table = (xmlNotationTablePtr) dtd->notations;
2367 if (table == NULL) {
2368 xmlDictPtr dict = NULL;
2369 if (dtd->doc != NULL)
2370 dict = dtd->doc->dict;
2371
2372 dtd->notations = table = xmlHashCreateDict(0, dict);
2373 }
2374 if (table == NULL) {
2375 xmlVErrMemory(ctxt,
2376 "xmlAddNotationDecl: Table creation failed!\n");
2377 return(NULL);
2378 }
2379
2380 ret = (xmlNotationPtr) xmlMalloc(sizeof(xmlNotation));
2381 if (ret == NULL) {
2382 xmlVErrMemory(ctxt, "malloc failed");
2383 return(NULL);
2384 }
2385 memset(ret, 0, sizeof(xmlNotation));
2386
2387 /*
2388 * fill the structure.
2389 */
2390 ret->name = xmlStrdup(name);
2391 if (SystemID != NULL)
2392 ret->SystemID = xmlStrdup(SystemID);
2393 if (PublicID != NULL)
2394 ret->PublicID = xmlStrdup(PublicID);
2395
2396 /*
2397 * Validity Check:
2398 * Check the DTD for previous declarations of the ATTLIST
2399 */
2400 if (xmlHashAddEntry(table, name, ret)) {
2401 #ifdef LIBXML_VALID_ENABLED
2402 xmlErrValid(NULL, XML_DTD_NOTATION_REDEFINED,
2403 "xmlAddNotationDecl: %s already defined\n",
2404 (const char *) name);
2405 #endif /* LIBXML_VALID_ENABLED */
2406 xmlFreeNotation(ret);
2407 return(NULL);
2408 }
2409 return(ret);
2410 }
2411
2412 /**
2413 * xmlFreeNotationTable:
2414 * @table: An notation table
2415 *
2416 * Deallocate the memory used by an entities hash table.
2417 */
2418 void
2419 xmlFreeNotationTable(xmlNotationTablePtr table) {
2420 xmlHashFree(table, (xmlHashDeallocator) xmlFreeNotation);
2421 }
2422
2423 #ifdef LIBXML_TREE_ENABLED
2424 /**
2425 * xmlCopyNotation:
2426 * @nota: A notation
2427 *
2428 * Build a copy of a notation.
2429 *
2430 * Returns the new xmlNotationPtr or NULL in case of error.
2431 */
2432 static xmlNotationPtr
2433 xmlCopyNotation(xmlNotationPtr nota) {
2434 xmlNotationPtr cur;
2435
2436 cur = (xmlNotationPtr) xmlMalloc(sizeof(xmlNotation));
2437 if (cur == NULL) {
2438 xmlVErrMemory(NULL, "malloc failed");
2439 return(NULL);
2440 }
2441 if (nota->name != NULL)
2442 cur->name = xmlStrdup(nota->name);
2443 else
2444 cur->name = NULL;
2445 if (nota->PublicID != NULL)
2446 cur->PublicID = xmlStrdup(nota->PublicID);
2447 else
2448 cur->PublicID = NULL;
2449 if (nota->SystemID != NULL)
2450 cur->SystemID = xmlStrdup(nota->SystemID);
2451 else
2452 cur->SystemID = NULL;
2453 return(cur);
2454 }
2455
2456 /**
2457 * xmlCopyNotationTable:
2458 * @table: A notation table
2459 *
2460 * Build a copy of a notation table.
2461 *
2462 * Returns the new xmlNotationTablePtr or NULL in case of error.
2463 */
2464 xmlNotationTablePtr
2465 xmlCopyNotationTable(xmlNotationTablePtr table) {
2466 return((xmlNotationTablePtr) xmlHashCopy(table,
2467 (xmlHashCopier) xmlCopyNotation));
2468 }
2469 #endif /* LIBXML_TREE_ENABLED */
2470
2471 #ifdef LIBXML_OUTPUT_ENABLED
2472 /**
2473 * xmlDumpNotationDecl:
2474 * @buf: the XML buffer output
2475 * @nota: A notation declaration
2476 *
2477 * This will dump the content the notation declaration as an XML DTD definition
2478 */
2479 void
2480 xmlDumpNotationDecl(xmlBufferPtr buf, xmlNotationPtr nota) {
2481 if ((buf == NULL) || (nota == NULL))
2482 return;
2483 xmlBufferWriteChar(buf, "<!NOTATION ");
2484 xmlBufferWriteCHAR(buf, nota->name);
2485 if (nota->PublicID != NULL) {
2486 xmlBufferWriteChar(buf, " PUBLIC ");
2487 xmlBufferWriteQuotedString(buf, nota->PublicID);
2488 if (nota->SystemID != NULL) {
2489 xmlBufferWriteChar(buf, " ");
2490 xmlBufferWriteQuotedString(buf, nota->SystemID);
2491 }
2492 } else {
2493 xmlBufferWriteChar(buf, " SYSTEM ");
2494 xmlBufferWriteQuotedString(buf, nota->SystemID);
2495 }
2496 xmlBufferWriteChar(buf, " >\n");
2497 }
2498
2499 /**
2500 * xmlDumpNotationDeclScan:
2501 * @nota: A notation declaration
2502 * @buf: the XML buffer output
2503 *
2504 * This is called with the hash scan function, and just reverses args
2505 */
2506 static void
2507 xmlDumpNotationDeclScan(xmlNotationPtr nota, xmlBufferPtr buf) {
2508 xmlDumpNotationDecl(buf, nota);
2509 }
2510
2511 /**
2512 * xmlDumpNotationTable:
2513 * @buf: the XML buffer output
2514 * @table: A notation table
2515 *
2516 * This will dump the content of the notation table as an XML DTD definition
2517 */
2518 void
2519 xmlDumpNotationTable(xmlBufferPtr buf, xmlNotationTablePtr table) {
2520 if ((buf == NULL) || (table == NULL))
2521 return;
2522 xmlHashScan(table, (xmlHashScanner) xmlDumpNotationDeclScan, buf);
2523 }
2524 #endif /* LIBXML_OUTPUT_ENABLED */
2525
2526 /************************************************************************
2527 * *
2528 * IDs *
2529 * *
2530 ************************************************************************/
2531 /**
2532 * DICT_FREE:
2533 * @str: a string
2534 *
2535 * Free a string if it is not owned by the "dict" dictionnary in the
2536 * current scope
2537 */
2538 #define DICT_FREE(str) \
2539 if ((str) && ((!dict) || \
2540 (xmlDictOwns(dict, (const xmlChar *)(str)) == 0))) \
2541 xmlFree((char *)(str));
2542
2543 /**
2544 * xmlFreeID:
2545 * @not: A id
2546 *
2547 * Deallocate the memory used by an id definition
2548 */
2549 static void
2550 xmlFreeID(xmlIDPtr id) {
2551 xmlDictPtr dict = NULL;
2552
2553 if (id == NULL) return;
2554
2555 if (id->doc != NULL)
2556 dict = id->doc->dict;
2557
2558 if (id->value != NULL)
2559 DICT_FREE(id->value)
2560 if (id->name != NULL)
2561 DICT_FREE(id->name)
2562 xmlFree(id);
2563 }
2564
2565
2566 /**
2567 * xmlAddID:
2568 * @ctxt: the validation context
2569 * @doc: pointer to the document
2570 * @value: the value name
2571 * @attr: the attribute holding the ID
2572 *
2573 * Register a new id declaration
2574 *
2575 * Returns NULL if not, otherwise the new xmlIDPtr
2576 */
2577 xmlIDPtr
2578 xmlAddID(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value,
2579 xmlAttrPtr attr) {
2580 xmlIDPtr ret;
2581 xmlIDTablePtr table;
2582
2583 if (doc == NULL) {
2584 return(NULL);
2585 }
2586 if (value == NULL) {
2587 return(NULL);
2588 }
2589 if (attr == NULL) {
2590 return(NULL);
2591 }
2592
2593 /*
2594 * Create the ID table if needed.
2595 */
2596 table = (xmlIDTablePtr) doc->ids;
2597 if (table == NULL) {
2598 doc->ids = table = xmlHashCreateDict(0, doc->dict);
2599 }
2600 if (table == NULL) {
2601 xmlVErrMemory(ctxt,
2602 "xmlAddID: Table creation failed!\n");
2603 return(NULL);
2604 }
2605
2606 ret = (xmlIDPtr) xmlMalloc(sizeof(xmlID));
2607 if (ret == NULL) {
2608 xmlVErrMemory(ctxt, "malloc failed");
2609 return(NULL);
2610 }
2611
2612 /*
2613 * fill the structure.
2614 */
2615 ret->value = xmlStrdup(value);
2616 ret->doc = doc;
2617 if ((ctxt != NULL) && (ctxt->vstateNr != 0)) {
2618 /*
2619 * Operating in streaming mode, attr is gonna disapear
2620 */
2621 if (doc->dict != NULL)
2622 ret->name = xmlDictLookup(doc->dict, attr->name, -1);
2623 else
2624 ret->name = xmlStrdup(attr->name);
2625 ret->attr = NULL;
2626 } else {
2627 ret->attr = attr;
2628 ret->name = NULL;
2629 }
2630 ret->lineno = xmlGetLineNo(attr->parent);
2631
2632 if (xmlHashAddEntry(table, value, ret) < 0) {
2633 #ifdef LIBXML_VALID_ENABLED
2634 /*
2635 * The id is already defined in this DTD.
2636 */
2637 if (ctxt != NULL) {
2638 xmlErrValidNode(ctxt, attr->parent, XML_DTD_ID_REDEFINED,
2639 "ID %s already defined\n", value, NULL, NULL);
2640 }
2641 #endif /* LIBXML_VALID_ENABLED */
2642 xmlFreeID(ret);
2643 return(NULL);
2644 }
2645 if (attr != NULL)
2646 attr->atype = XML_ATTRIBUTE_ID;
2647 return(ret);
2648 }
2649
2650 /**
2651 * xmlFreeIDTable:
2652 * @table: An id table
2653 *
2654 * Deallocate the memory used by an ID hash table.
2655 */
2656 void
2657 xmlFreeIDTable(xmlIDTablePtr table) {
2658 xmlHashFree(table, (xmlHashDeallocator) xmlFreeID);
2659 }
2660
2661 /**
2662 * xmlIsID:
2663 * @doc: the document
2664 * @elem: the element carrying the attribute
2665 * @attr: the attribute
2666 *
2667 * Determine whether an attribute is of type ID. In case we have DTD(s)
2668 * then this is done if DTD loading has been requested. In the case
2669 * of HTML documents parsed with the HTML parser, then ID detection is
2670 * done systematically.
2671 *
2672 * Returns 0 or 1 depending on the lookup result
2673 */
2674 int
2675 xmlIsID(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) {
2676 if ((attr == NULL) || (attr->name == NULL)) return(0);
2677 if ((attr->ns != NULL) && (attr->ns->prefix != NULL) &&
2678 (!strcmp((char *) attr->name, "id")) &&
2679 (!strcmp((char *) attr->ns->prefix, "xml")))
2680 return(1);
2681 if (doc == NULL) return(0);
2682 if ((doc->intSubset == NULL) && (doc->extSubset == NULL) &&
2683 (doc->type != XML_HTML_DOCUMENT_NODE)) {
2684 return(0);
2685 } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
2686 if ((xmlStrEqual(BAD_CAST "id", attr->name)) ||
2687 ((xmlStrEqual(BAD_CAST "name", attr->name)) &&
2688 ((elem == NULL) || (xmlStrEqual(elem->name, BAD_CAST "a")))))
2689 return(1);
2690 return(0);
2691 } else if (elem == NULL) {
2692 return(0);
2693 } else {
2694 xmlAttributePtr attrDecl = NULL;
2695
2696 xmlChar felem[50], fattr[50];
2697 xmlChar *fullelemname, *fullattrname;
2698
2699 fullelemname = (elem->ns != NULL && elem->ns->prefix != NULL) ?
2700 xmlBuildQName(elem->name, elem->ns->prefix, felem, 50) :
2701 (xmlChar *)elem->name;
2702
2703 fullattrname = (attr->ns != NULL && attr->ns->prefix != NULL) ?
2704 xmlBuildQName(attr->name, attr->ns->prefix, fattr, 50) :
2705 (xmlChar *)attr->name;
2706
2707 if (fullelemname != NULL && fullattrname != NULL) {
2708 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullelemname,
2709 fullattrname);
2710 if ((attrDecl == NULL) && (doc->extSubset != NULL))
2711 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullelemname,
2712 fullattrname);
2713 }
2714
2715 if ((fullattrname != fattr) && (fullattrname != attr->name))
2716 xmlFree(fullattrname);
2717 if ((fullelemname != felem) && (fullelemname != elem->name))
2718 xmlFree(fullelemname);
2719
2720 if ((attrDecl != NULL) && (attrDecl->atype == XML_ATTRIBUTE_ID))
2721 return(1);
2722 }
2723 return(0);
2724 }
2725
2726 /**
2727 * xmlRemoveID:
2728 * @doc: the document
2729 * @attr: the attribute
2730 *
2731 * Remove the given attribute from the ID table maintained internally.
2732 *
2733 * Returns -1 if the lookup failed and 0 otherwise
2734 */
2735 int
2736 xmlRemoveID(xmlDocPtr doc, xmlAttrPtr attr) {
2737 xmlIDTablePtr table;
2738 xmlIDPtr id;
2739 xmlChar *ID;
2740
2741 if (doc == NULL) return(-1);
2742 if (attr == NULL) return(-1);
2743
2744 table = (xmlIDTablePtr) doc->ids;
2745 if (table == NULL)
2746 return(-1);
2747
2748 ID = xmlNodeListGetString(doc, attr->children, 1);
2749 if (ID == NULL)
2750 return(-1);
2751
2752 id = xmlHashLookup(table, ID);
2753 if (id == NULL || id->attr != attr) {
2754 xmlFree(ID);
2755 return(-1);
2756 }
2757
2758 xmlHashRemoveEntry(table, ID, (xmlHashDeallocator) xmlFreeID);
2759 xmlFree(ID);
2760 attr->atype = 0;
2761 return(0);
2762 }
2763
2764 /**
2765 * xmlGetID:
2766 * @doc: pointer to the document
2767 * @ID: the ID value
2768 *
2769 * Search the attribute declaring the given ID
2770 *
2771 * Returns NULL if not found, otherwise the xmlAttrPtr defining the ID
2772 */
2773 xmlAttrPtr
2774 xmlGetID(xmlDocPtr doc, const xmlChar *ID) {
2775 xmlIDTablePtr table;
2776 xmlIDPtr id;
2777
2778 if (doc == NULL) {
2779 return(NULL);
2780 }
2781
2782 if (ID == NULL) {
2783 return(NULL);
2784 }
2785
2786 table = (xmlIDTablePtr) doc->ids;
2787 if (table == NULL)
2788 return(NULL);
2789
2790 id = xmlHashLookup(table, ID);
2791 if (id == NULL)
2792 return(NULL);
2793 if (id->attr == NULL) {
2794 /*
2795 * We are operating on a stream, return a well known reference
2796 * since the attribute node doesn't exist anymore
2797 */
2798 return((xmlAttrPtr) doc);
2799 }
2800 return(id->attr);
2801 }
2802
2803 /************************************************************************
2804 * *
2805 * Refs *
2806 * *
2807 ************************************************************************/
2808 typedef struct xmlRemoveMemo_t
2809 {
2810 xmlListPtr l;
2811 xmlAttrPtr ap;
2812 } xmlRemoveMemo;
2813
2814 typedef xmlRemoveMemo *xmlRemoveMemoPtr;
2815
2816 typedef struct xmlValidateMemo_t
2817 {
2818 xmlValidCtxtPtr ctxt;
2819 const xmlChar *name;
2820 } xmlValidateMemo;
2821
2822 typedef xmlValidateMemo *xmlValidateMemoPtr;
2823
2824 /**
2825 * xmlFreeRef:
2826 * @lk: A list link
2827 *
2828 * Deallocate the memory used by a ref definition
2829 */
2830 static void
2831 xmlFreeRef(xmlLinkPtr lk) {
2832 xmlRefPtr ref = (xmlRefPtr)xmlLinkGetData(lk);
2833 if (ref == NULL) return;
2834 if (ref->value != NULL)
2835 xmlFree((xmlChar *)ref->value);
2836 if (ref->name != NULL)
2837 xmlFree((xmlChar *)ref->name);
2838 xmlFree(ref);
2839 }
2840
2841 /**
2842 * xmlFreeRefList:
2843 * @list_ref: A list of references.
2844 *
2845 * Deallocate the memory used by a list of references
2846 */
2847 static void
2848 xmlFreeRefList(xmlListPtr list_ref) {
2849 if (list_ref == NULL) return;
2850 xmlListDelete(list_ref);
2851 }
2852
2853 /**
2854 * xmlWalkRemoveRef:
2855 * @data: Contents of current link
2856 * @user: Value supplied by the user
2857 *
2858 * Returns 0 to abort the walk or 1 to continue
2859 */
2860 static int
2861 xmlWalkRemoveRef(const void *data, const void *user)
2862 {
2863 xmlAttrPtr attr0 = ((xmlRefPtr)data)->attr;
2864 xmlAttrPtr attr1 = ((xmlRemoveMemoPtr)user)->ap;
2865 xmlListPtr ref_list = ((xmlRemoveMemoPtr)user)->l;
2866
2867 if (attr0 == attr1) { /* Matched: remove and terminate walk */
2868 xmlListRemoveFirst(ref_list, (void *)data);
2869 return 0;
2870 }
2871 return 1;
2872 }
2873
2874 /**
2875 * xmlDummyCompare
2876 * @data0: Value supplied by the user
2877 * @data1: Value supplied by the user
2878 *
2879 * Do nothing, return 0. Used to create unordered lists.
2880 */
2881 static int
2882 xmlDummyCompare(const void *data0 ATTRIBUTE_UNUSED,
2883 const void *data1 ATTRIBUTE_UNUSED)
2884 {
2885 return (0);
2886 }
2887
2888 /**
2889 * xmlAddRef:
2890 * @ctxt: the validation context
2891 * @doc: pointer to the document
2892 * @value: the value name
2893 * @attr: the attribute holding the Ref
2894 *
2895 * Register a new ref declaration
2896 *
2897 * Returns NULL if not, otherwise the new xmlRefPtr
2898 */
2899 xmlRefPtr
2900 xmlAddRef(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value,
2901 xmlAttrPtr attr) {
2902 xmlRefPtr ret;
2903 xmlRefTablePtr table;
2904 xmlListPtr ref_list;
2905
2906 if (doc == NULL) {
2907 return(NULL);
2908 }
2909 if (value == NULL) {
2910 return(NULL);
2911 }
2912 if (attr == NULL) {
2913 return(NULL);
2914 }
2915
2916 /*
2917 * Create the Ref table if needed.
2918 */
2919 table = (xmlRefTablePtr) doc->refs;
2920 if (table == NULL) {
2921 doc->refs = table = xmlHashCreateDict(0, doc->dict);
2922 }
2923 if (table == NULL) {
2924 xmlVErrMemory(ctxt,
2925 "xmlAddRef: Table creation failed!\n");
2926 return(NULL);
2927 }
2928
2929 ret = (xmlRefPtr) xmlMalloc(sizeof(xmlRef));
2930 if (ret == NULL) {
2931 xmlVErrMemory(ctxt, "malloc failed");
2932 return(NULL);
2933 }
2934
2935 /*
2936 * fill the structure.
2937 */
2938 ret->value = xmlStrdup(value);
2939 if ((ctxt != NULL) && (ctxt->vstateNr != 0)) {
2940 /*
2941 * Operating in streaming mode, attr is gonna disapear
2942 */
2943 ret->name = xmlStrdup(attr->name);
2944 ret->attr = NULL;
2945 } else {
2946 ret->name = NULL;
2947 ret->attr = attr;
2948 }
2949 ret->lineno = xmlGetLineNo(attr->parent);
2950
2951 /* To add a reference :-
2952 * References are maintained as a list of references,
2953 * Lookup the entry, if no entry create new nodelist
2954 * Add the owning node to the NodeList
2955 * Return the ref
2956 */
2957
2958 if (NULL == (ref_list = xmlHashLookup(table, value))) {
2959 if (NULL == (ref_list = xmlListCreate(xmlFreeRef, xmlDummyCompare))) {
2960 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2961 "xmlAddRef: Reference list creation failed!\n",
2962 NULL);
2963 goto failed;
2964 }
2965 if (xmlHashAddEntry(table, value, ref_list) < 0) {
2966 xmlListDelete(ref_list);
2967 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2968 "xmlAddRef: Reference list insertion failed!\n",
2969 NULL);
2970 goto failed;
2971 }
2972 }
2973 if (xmlListAppend(ref_list, ret) != 0) {
2974 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2975 "xmlAddRef: Reference list insertion failed!\n",
2976 NULL);
2977 goto failed;
2978 }
2979 return(ret);
2980 failed:
2981 if (ret != NULL) {
2982 if (ret->value != NULL)
2983 xmlFree((char *)ret->value);
2984 if (ret->name != NULL)
2985 xmlFree((char *)ret->name);
2986 xmlFree(ret);
2987 }
2988 return(NULL);
2989 }
2990
2991 /**
2992 * xmlFreeRefTable:
2993 * @table: An ref table
2994 *
2995 * Deallocate the memory used by an Ref hash table.
2996 */
2997 void
2998 xmlFreeRefTable(xmlRefTablePtr table) {
2999 xmlHashFree(table, (xmlHashDeallocator) xmlFreeRefList);
3000 }
3001
3002 /**
3003 * xmlIsRef:
3004 * @doc: the document
3005 * @elem: the element carrying the attribute
3006 * @attr: the attribute
3007 *
3008 * Determine whether an attribute is of type Ref. In case we have DTD(s)
3009 * then this is simple, otherwise we use an heuristic: name Ref (upper
3010 * or lowercase).
3011 *
3012 * Returns 0 or 1 depending on the lookup result
3013 */
3014 int
3015 xmlIsRef(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) {
3016 if (attr == NULL)
3017 return(0);
3018 if (doc == NULL) {
3019 doc = attr->doc;
3020 if (doc == NULL) return(0);
3021 }
3022
3023 if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
3024 return(0);
3025 } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
3026 /* TODO @@@ */
3027 return(0);
3028 } else {
3029 xmlAttributePtr attrDecl;
3030
3031 if (elem == NULL) return(0);
3032 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, attr->name);
3033 if ((attrDecl == NULL) && (doc->extSubset != NULL))
3034 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
3035 elem->name, attr->name);
3036
3037 if ((attrDecl != NULL) &&
3038 (attrDecl->atype == XML_ATTRIBUTE_IDREF ||
3039 attrDecl->atype == XML_ATTRIBUTE_IDREFS))
3040 return(1);
3041 }
3042 return(0);
3043 }
3044
3045 /**
3046 * xmlRemoveRef:
3047 * @doc: the document
3048 * @attr: the attribute
3049 *
3050 * Remove the given attribute from the Ref table maintained internally.
3051 *
3052 * Returns -1 if the lookup failed and 0 otherwise
3053 */
3054 int
3055 xmlRemoveRef(xmlDocPtr doc, xmlAttrPtr attr) {
3056 xmlListPtr ref_list;
3057 xmlRefTablePtr table;
3058 xmlChar *ID;
3059 xmlRemoveMemo target;
3060
3061 if (doc == NULL) return(-1);
3062 if (attr == NULL) return(-1);
3063
3064 table = (xmlRefTablePtr) doc->refs;
3065 if (table == NULL)
3066 return(-1);
3067
3068 ID = xmlNodeListGetString(doc, attr->children, 1);
3069 if (ID == NULL)
3070 return(-1);
3071
3072 ref_list = xmlHashLookup(table, ID);
3073 if(ref_list == NULL) {
3074 xmlFree(ID);
3075 return (-1);
3076 }
3077
3078 /* At this point, ref_list refers to a list of references which
3079 * have the same key as the supplied attr. Our list of references
3080 * is ordered by reference address and we don't have that information
3081 * here to use when removing. We'll have to walk the list and
3082 * check for a matching attribute, when we find one stop the walk
3083 * and remove the entry.
3084 * The list is ordered by reference, so that means we don't have the
3085 * key. Passing the list and the reference to the walker means we
3086 * will have enough data to be able to remove the entry.
3087 */
3088 target.l = ref_list;
3089 target.ap = attr;
3090
3091 /* Remove the supplied attr from our list */
3092 xmlListWalk(ref_list, xmlWalkRemoveRef, &target);
3093
3094 /*If the list is empty then remove the list entry in the hash */
3095 if (xmlListEmpty(ref_list))
3096 xmlHashUpdateEntry(table, ID, NULL, (xmlHashDeallocator)
3097 xmlFreeRefList);
3098 xmlFree(ID);
3099 return(0);
3100 }
3101
3102 /**
3103 * xmlGetRefs:
3104 * @doc: pointer to the document
3105 * @ID: the ID value
3106 *
3107 * Find the set of references for the supplied ID.
3108 *
3109 * Returns NULL if not found, otherwise node set for the ID.
3110 */
3111 xmlListPtr
3112 xmlGetRefs(xmlDocPtr doc, const xmlChar *ID) {
3113 xmlRefTablePtr table;
3114
3115 if (doc == NULL) {
3116 return(NULL);
3117 }
3118
3119 if (ID == NULL) {
3120 return(NULL);
3121 }
3122
3123 table = (xmlRefTablePtr) doc->refs;
3124 if (table == NULL)
3125 return(NULL);
3126
3127 return (xmlHashLookup(table, ID));
3128 }
3129
3130 /************************************************************************
3131 * *
3132 * Routines for validity checking *
3133 * *
3134 ************************************************************************/
3135
3136 /**
3137 * xmlGetDtdElementDesc:
3138 * @dtd: a pointer to the DtD to search
3139 * @name: the element name
3140 *
3141 * Search the DTD for the description of this element
3142 *
3143 * returns the xmlElementPtr if found or NULL
3144 */
3145
3146 xmlElementPtr
3147 xmlGetDtdElementDesc(xmlDtdPtr dtd, const xmlChar *name) {
3148 xmlElementTablePtr table;
3149 xmlElementPtr cur;
3150 xmlChar *uqname = NULL, *prefix = NULL;
3151
3152 if ((dtd == NULL) || (name == NULL)) return(NULL);
3153 if (dtd->elements == NULL)
3154 return(NULL);
3155 table = (xmlElementTablePtr) dtd->elements;
3156
3157 uqname = xmlSplitQName2(name, &prefix);
3158 if (uqname != NULL)
3159 name = uqname;
3160 cur = xmlHashLookup2(table, name, prefix);
3161 if (prefix != NULL) xmlFree(prefix);
3162 if (uqname != NULL) xmlFree(uqname);
3163 return(cur);
3164 }
3165 /**
3166 * xmlGetDtdElementDesc2:
3167 * @dtd: a pointer to the DtD to search
3168 * @name: the element name
3169 * @create: create an empty description if not found
3170 *
3171 * Search the DTD for the description of this element
3172 *
3173 * returns the xmlElementPtr if found or NULL
3174 */
3175
3176 static xmlElementPtr
3177 xmlGetDtdElementDesc2(xmlDtdPtr dtd, const xmlChar *name, int create) {
3178 xmlElementTablePtr table;
3179 xmlElementPtr cur;
3180 xmlChar *uqname = NULL, *prefix = NULL;
3181
3182 if (dtd == NULL) return(NULL);
3183 if (dtd->elements == NULL) {
3184 xmlDictPtr dict = NULL;
3185
3186 if (dtd->doc != NULL)
3187 dict = dtd->doc->dict;
3188
3189 if (!create)
3190 return(NULL);
3191 /*
3192 * Create the Element table if needed.
3193 */
3194 table = (xmlElementTablePtr) dtd->elements;
3195 if (table == NULL) {
3196 table = xmlHashCreateDict(0, dict);
3197 dtd->elements = (void *) table;
3198 }
3199 if (table == NULL) {
3200 xmlVErrMemory(NULL, "element table allocation failed");
3201 return(NULL);
3202 }
3203 }
3204 table = (xmlElementTablePtr) dtd->elements;
3205
3206 uqname = xmlSplitQName2(name, &prefix);
3207 if (uqname != NULL)
3208 name = uqname;
3209 cur = xmlHashLookup2(table, name, prefix);
3210 if ((cur == NULL) && (create)) {
3211 cur = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
3212 if (cur == NULL) {
3213 xmlVErrMemory(NULL, "malloc failed");
3214 return(NULL);
3215 }
3216 memset(cur, 0, sizeof(xmlElement));
3217 cur->type = XML_ELEMENT_DECL;
3218
3219 /*
3220 * fill the structure.
3221 */
3222 cur->name = xmlStrdup(name);
3223 cur->prefix = xmlStrdup(prefix);
3224 cur->etype = XML_ELEMENT_TYPE_UNDEFINED;
3225
3226 xmlHashAddEntry2(table, name, prefix, cur);
3227 }
3228 if (prefix != NULL) xmlFree(prefix);
3229 if (uqname != NULL) xmlFree(uqname);
3230 return(cur);
3231 }
3232
3233 /**
3234 * xmlGetDtdQElementDesc:
3235 * @dtd: a pointer to the DtD to search
3236 * @name: the element name
3237 * @prefix: the element namespace prefix
3238 *
3239 * Search the DTD for the description of this element
3240 *
3241 * returns the xmlElementPtr if found or NULL
3242 */
3243
3244 xmlElementPtr
3245 xmlGetDtdQElementDesc(xmlDtdPtr dtd, const xmlChar *name,
3246 const xmlChar *prefix) {
3247 xmlElementTablePtr table;
3248
3249 if (dtd == NULL) return(NULL);
3250 if (dtd->elements == NULL) return(NULL);
3251 table = (xmlElementTablePtr) dtd->elements;
3252
3253 return(xmlHashLookup2(table, name, prefix));
3254 }
3255
3256 /**
3257 * xmlGetDtdAttrDesc:
3258 * @dtd: a pointer to the DtD to search
3259 * @elem: the element name
3260 * @name: the attribute name
3261 *
3262 * Search the DTD for the description of this attribute on
3263 * this element.
3264 *
3265 * returns the xmlAttributePtr if found or NULL
3266 */
3267
3268 xmlAttributePtr
3269 xmlGetDtdAttrDesc(xmlDtdPtr dtd, const xmlChar *elem, const xmlChar *name) {
3270 xmlAttributeTablePtr table;
3271 xmlAttributePtr cur;
3272 xmlChar *uqname = NULL, *prefix = NULL;
3273
3274 if (dtd == NULL) return(NULL);
3275 if (dtd->attributes == NULL) return(NULL);
3276
3277 table = (xmlAttributeTablePtr) dtd->attributes;
3278 if (table == NULL)
3279 return(NULL);
3280
3281 uqname = xmlSplitQName2(name, &prefix);
3282
3283 if (uqname != NULL) {
3284 cur = xmlHashLookup3(table, uqname, prefix, elem);
3285 if (prefix != NULL) xmlFree(prefix);
3286 if (uqname != NULL) xmlFree(uqname);
3287 } else
3288 cur = xmlHashLookup3(table, name, NULL, elem);
3289 return(cur);
3290 }
3291
3292 /**
3293 * xmlGetDtdQAttrDesc:
3294 * @dtd: a pointer to the DtD to search
3295 * @elem: the element name
3296 * @name: the attribute name
3297 * @prefix: the attribute namespace prefix
3298 *
3299 * Search the DTD for the description of this qualified attribute on
3300 * this element.
3301 *
3302 * returns the xmlAttributePtr if found or NULL
3303 */
3304
3305 xmlAttributePtr
3306 xmlGetDtdQAttrDesc(xmlDtdPtr dtd, const xmlChar *elem, const xmlChar *name,
3307 const xmlChar *prefix) {
3308 xmlAttributeTablePtr table;
3309
3310 if (dtd == NULL) return(NULL);
3311 if (dtd->attributes == NULL) return(NULL);
3312 table = (xmlAttributeTablePtr) dtd->attributes;
3313
3314 return(xmlHashLookup3(table, name, prefix, elem));
3315 }
3316
3317 /**
3318 * xmlGetDtdNotationDesc:
3319 * @dtd: a pointer to the DtD to search
3320 * @name: the notation name
3321 *
3322 * Search the DTD for the description of this notation
3323 *
3324 * returns the xmlNotationPtr if found or NULL
3325 */
3326
3327 xmlNotationPtr
3328 xmlGetDtdNotationDesc(xmlDtdPtr dtd, const xmlChar *name) {
3329 xmlNotationTablePtr table;
3330
3331 if (dtd == NULL) return(NULL);
3332 if (dtd->notations == NULL) return(NULL);
3333 table = (xmlNotationTablePtr) dtd->notations;
3334
3335 return(xmlHashLookup(table, name));
3336 }
3337
3338 #if defined(LIBXML_VALID_ENABLED) || defined(LIBXML_SCHEMAS_ENABLED)
3339 /**
3340 * xmlValidateNotationUse:
3341 * @ctxt: the validation context
3342 * @doc: the document
3343 * @notationName: the notation name to check
3344 *
3345 * Validate that the given name match a notation declaration.
3346 * - [ VC: Notation Declared ]
3347 *
3348 * returns 1 if valid or 0 otherwise
3349 */
3350
3351 int
3352 xmlValidateNotationUse(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3353 const xmlChar *notationName) {
3354 xmlNotationPtr notaDecl;
3355 if ((doc == NULL) || (doc->intSubset == NULL) ||
3356 (notationName == NULL)) return(-1);
3357
3358 notaDecl = xmlGetDtdNotationDesc(doc->intSubset, notationName);
3359 if ((notaDecl == NULL) && (doc->extSubset != NULL))
3360 notaDecl = xmlGetDtdNotationDesc(doc->extSubset, notationName);
3361
3362 if ((notaDecl == NULL) && (ctxt != NULL)) {
3363 xmlErrValidNode(ctxt, (xmlNodePtr) doc, XML_DTD_UNKNOWN_NOTATION,
3364 "NOTATION %s is not declared\n",
3365 notationName, NULL, NULL);
3366 return(0);
3367 }
3368 return(1);
3369 }
3370 #endif /* LIBXML_VALID_ENABLED or LIBXML_SCHEMAS_ENABLED */
3371
3372 /**
3373 * xmlIsMixedElement:
3374 * @doc: the document
3375 * @name: the element name
3376 *
3377 * Search in the DtDs whether an element accept Mixed content (or ANY)
3378 * basically if it is supposed to accept text childs
3379 *
3380 * returns 0 if no, 1 if yes, and -1 if no element description is available
3381 */
3382
3383 int
3384 xmlIsMixedElement(xmlDocPtr doc, const xmlChar *name) {
3385 xmlElementPtr elemDecl;
3386
3387 if ((doc == NULL) || (doc->intSubset == NULL)) return(-1);
3388
3389 elemDecl = xmlGetDtdElementDesc(doc->intSubset, name);
3390 if ((elemDecl == NULL) && (doc->extSubset != NULL))
3391 elemDecl = xmlGetDtdElementDesc(doc->extSubset, name);
3392 if (elemDecl == NULL) return(-1);
3393 switch (elemDecl->etype) {
3394 case XML_ELEMENT_TYPE_UNDEFINED:
3395 return(-1);
3396 case XML_ELEMENT_TYPE_ELEMENT:
3397 return(0);
3398 case XML_ELEMENT_TYPE_EMPTY:
3399 /*
3400 * return 1 for EMPTY since we want VC error to pop up
3401 * on <empty> </empty> for example
3402 */
3403 case XML_ELEMENT_TYPE_ANY:
3404 case XML_ELEMENT_TYPE_MIXED:
3405 return(1);
3406 }
3407 return(1);
3408 }
3409
3410 #ifdef LIBXML_VALID_ENABLED
3411
3412 static int
3413 xmlIsDocNameStartChar(xmlDocPtr doc, int c) {
3414 if ((doc == NULL) || (doc->properties & XML_DOC_OLD10) == 0) {
3415 /*
3416 * Use the new checks of production [4] [4a] amd [5] of the
3417 * Update 5 of XML-1.0
3418 */
3419 if (((c >= 'a') && (c <= 'z')) ||
3420 ((c >= 'A') && (c <= 'Z')) ||
3421 (c == '_') || (c == ':') ||
3422 ((c >= 0xC0) && (c <= 0xD6)) ||
3423 ((c >= 0xD8) && (c <= 0xF6)) ||
3424 ((c >= 0xF8) && (c <= 0x2FF)) ||
3425 ((c >= 0x370) && (c <= 0x37D)) ||
3426 ((c >= 0x37F) && (c <= 0x1FFF)) ||
3427 ((c >= 0x200C) && (c <= 0x200D)) ||
3428 ((c >= 0x2070) && (c <= 0x218F)) ||
3429 ((c >= 0x2C00) && (c <= 0x2FEF)) ||
3430 ((c >= 0x3001) && (c <= 0xD7FF)) ||
3431 ((c >= 0xF900) && (c <= 0xFDCF)) ||
3432 ((c >= 0xFDF0) && (c <= 0xFFFD)) ||
3433 ((c >= 0x10000) && (c <= 0xEFFFF)))
3434 return(1);
3435 } else {
3436 if (IS_LETTER(c) || (c == '_') || (c == ':'))
3437 return(1);
3438 }
3439 return(0);
3440 }
3441
3442 static int
3443 xmlIsDocNameChar(xmlDocPtr doc, int c) {
3444 if ((doc == NULL) || (doc->properties & XML_DOC_OLD10) == 0) {
3445 /*
3446 * Use the new checks of production [4] [4a] amd [5] of the
3447 * Update 5 of XML-1.0
3448 */
3449 if (((c >= 'a') && (c <= 'z')) ||
3450 ((c >= 'A') && (c <= 'Z')) ||
3451 ((c >= '0') && (c <= '9')) || /* !start */
3452 (c == '_') || (c == ':') ||
3453 (c == '-') || (c == '.') || (c == 0xB7) || /* !start */
3454 ((c >= 0xC0) && (c <= 0xD6)) ||
3455 ((c >= 0xD8) && (c <= 0xF6)) ||
3456 ((c >= 0xF8) && (c <= 0x2FF)) ||
3457 ((c >= 0x300) && (c <= 0x36F)) || /* !start */
3458 ((c >= 0x370) && (c <= 0x37D)) ||
3459 ((c >= 0x37F) && (c <= 0x1FFF)) ||
3460 ((c >= 0x200C) && (c <= 0x200D)) ||
3461 ((c >= 0x203F) && (c <= 0x2040)) || /* !start */
3462 ((c >= 0x2070) && (c <= 0x218F)) ||
3463 ((c >= 0x2C00) && (c <= 0x2FEF)) ||
3464 ((c >= 0x3001) && (c <= 0xD7FF)) ||
3465 ((c >= 0xF900) && (c <= 0xFDCF)) ||
3466 ((c >= 0xFDF0) && (c <= 0xFFFD)) ||
3467 ((c >= 0x10000) && (c <= 0xEFFFF)))
3468 return(1);
3469 } else {
3470 if ((IS_LETTER(c)) || (IS_DIGIT(c)) ||
3471 (c == '.') || (c == '-') ||
3472 (c == '_') || (c == ':') ||
3473 (IS_COMBINING(c)) ||
3474 (IS_EXTENDER(c)))
3475 return(1);
3476 }
3477 return(0);
3478 }
3479
3480 /**
3481 * xmlValidateNameValue:
3482 * @doc: pointer to the document or NULL
3483 * @value: an Name value
3484 *
3485 * Validate that the given value match Name production
3486 *
3487 * returns 1 if valid or 0 otherwise
3488 */
3489
3490 static int
3491 xmlValidateNameValueInternal(xmlDocPtr doc, const xmlChar *value) {
3492 const xmlChar *cur;
3493 int val, len;
3494
3495 if (value == NULL) return(0);
3496 cur = value;
3497 val = xmlStringCurrentChar(NULL, cur, &len);
3498 cur += len;
3499 if (!xmlIsDocNameStartChar(doc, val))
3500 return(0);
3501
3502 val = xmlStringCurrentChar(NULL, cur, &len);
3503 cur += len;
3504 while (xmlIsDocNameChar(doc, val)) {
3505 val = xmlStringCurrentChar(NULL, cur, &len);
3506 cur += len;
3507 }
3508
3509 if (val != 0) return(0);
3510
3511 return(1);
3512 }
3513
3514 /**
3515 * xmlValidateNameValue:
3516 * @value: an Name value
3517 *
3518 * Validate that the given value match Name production
3519 *
3520 * returns 1 if valid or 0 otherwise
3521 */
3522
3523 int
3524 xmlValidateNameValue(const xmlChar *value) {
3525 return(xmlValidateNameValueInternal(NULL, value));
3526 }
3527
3528 /**
3529 * xmlValidateNamesValueInternal:
3530 * @doc: pointer to the document or NULL
3531 * @value: an Names value
3532 *
3533 * Validate that the given value match Names production
3534 *
3535 * returns 1 if valid or 0 otherwise
3536 */
3537
3538 static int
3539 xmlValidateNamesValueInternal(xmlDocPtr doc, const xmlChar *value) {
3540 const xmlChar *cur;
3541 int val, len;
3542
3543 if (value == NULL) return(0);
3544 cur = value;
3545 val = xmlStringCurrentChar(NULL, cur, &len);
3546 cur += len;
3547
3548 if (!xmlIsDocNameStartChar(doc, val))
3549 return(0);
3550
3551 val = xmlStringCurrentChar(NULL, cur, &len);
3552 cur += len;
3553 while (xmlIsDocNameChar(doc, val)) {
3554 val = xmlStringCurrentChar(NULL, cur, &len);
3555 cur += len;
3556 }
3557
3558 /* Should not test IS_BLANK(val) here -- see erratum E20*/
3559 while (val == 0x20) {
3560 while (val == 0x20) {
3561 val = xmlStringCurrentChar(NULL, cur, &len);
3562 cur += len;
3563 }
3564
3565 if (!xmlIsDocNameStartChar(doc, val))
3566 return(0);
3567
3568 val = xmlStringCurrentChar(NULL, cur, &len);
3569 cur += len;
3570
3571 while (xmlIsDocNameChar(doc, val)) {
3572 val = xmlStringCurrentChar(NULL, cur, &len);
3573 cur += len;
3574 }
3575 }
3576
3577 if (val != 0) return(0);
3578
3579 return(1);
3580 }
3581
3582 /**
3583 * xmlValidateNamesValue:
3584 * @value: an Names value
3585 *
3586 * Validate that the given value match Names production
3587 *
3588 * returns 1 if valid or 0 otherwise
3589 */
3590
3591 int
3592 xmlValidateNamesValue(const xmlChar *value) {
3593 return(xmlValidateNamesValueInternal(NULL, value));
3594 }
3595
3596 /**
3597 * xmlValidateNmtokenValueInternal:
3598 * @doc: pointer to the document or NULL
3599 * @value: an Nmtoken value
3600 *
3601 * Validate that the given value match Nmtoken production
3602 *
3603 * [ VC: Name Token ]
3604 *
3605 * returns 1 if valid or 0 otherwise
3606 */
3607
3608 static int
3609 xmlValidateNmtokenValueInternal(xmlDocPtr doc, const xmlChar *value) {
3610 const xmlChar *cur;
3611 int val, len;
3612
3613 if (value == NULL) return(0);
3614 cur = value;
3615 val = xmlStringCurrentChar(NULL, cur, &len);
3616 cur += len;
3617
3618 if (!xmlIsDocNameChar(doc, val))
3619 return(0);
3620
3621 val = xmlStringCurrentChar(NULL, cur, &len);
3622 cur += len;
3623 while (xmlIsDocNameChar(doc, val)) {
3624 val = xmlStringCurrentChar(NULL, cur, &len);
3625 cur += len;
3626 }
3627
3628 if (val != 0) return(0);
3629
3630 return(1);
3631 }
3632
3633 /**
3634 * xmlValidateNmtokenValue:
3635 * @value: an Nmtoken value
3636 *
3637 * Validate that the given value match Nmtoken production
3638 *
3639 * [ VC: Name Token ]
3640 *
3641 * returns 1 if valid or 0 otherwise
3642 */
3643
3644 int
3645 xmlValidateNmtokenValue(const xmlChar *value) {
3646 return(xmlValidateNmtokenValueInternal(NULL, value));
3647 }
3648
3649 /**
3650 * xmlValidateNmtokensValueInternal:
3651 * @doc: pointer to the document or NULL
3652 * @value: an Nmtokens value
3653 *
3654 * Validate that the given value match Nmtokens production
3655 *
3656 * [ VC: Name Token ]
3657 *
3658 * returns 1 if valid or 0 otherwise
3659 */
3660
3661 static int
3662 xmlValidateNmtokensValueInternal(xmlDocPtr doc, const xmlChar *value) {
3663 const xmlChar *cur;
3664 int val, len;
3665
3666 if (value == NULL) return(0);
3667 cur = value;
3668 val = xmlStringCurrentChar(NULL, cur, &len);
3669 cur += len;
3670
3671 while (IS_BLANK(val)) {
3672 val = xmlStringCurrentChar(NULL, cur, &len);
3673 cur += len;
3674 }
3675
3676 if (!xmlIsDocNameChar(doc, val))
3677 return(0);
3678
3679 while (xmlIsDocNameChar(doc, val)) {
3680 val = xmlStringCurrentChar(NULL, cur, &len);
3681 cur += len;
3682 }
3683
3684 /* Should not test IS_BLANK(val) here -- see erratum E20*/
3685 while (val == 0x20) {
3686 while (val == 0x20) {
3687 val = xmlStringCurrentChar(NULL, cur, &len);
3688 cur += len;
3689 }
3690 if (val == 0) return(1);
3691
3692 if (!xmlIsDocNameChar(doc, val))
3693 return(0);
3694
3695 val = xmlStringCurrentChar(NULL, cur, &len);
3696 cur += len;
3697
3698 while (xmlIsDocNameChar(doc, val)) {
3699 val = xmlStringCurrentChar(NULL, cur, &len);
3700 cur += len;
3701 }
3702 }
3703
3704 if (val != 0) return(0);
3705
3706 return(1);
3707 }
3708
3709 /**
3710 * xmlValidateNmtokensValue:
3711 * @value: an Nmtokens value
3712 *
3713 * Validate that the given value match Nmtokens production
3714 *
3715 * [ VC: Name Token ]
3716 *
3717 * returns 1 if valid or 0 otherwise
3718 */
3719
3720 int
3721 xmlValidateNmtokensValue(const xmlChar *value) {
3722 return(xmlValidateNmtokensValueInternal(NULL, value));
3723 }
3724
3725 /**
3726 * xmlValidateNotationDecl:
3727 * @ctxt: the validation context
3728 * @doc: a document instance
3729 * @nota: a notation definition
3730 *
3731 * Try to validate a single notation definition
3732 * basically it does the following checks as described by the
3733 * XML-1.0 recommendation:
3734 * - it seems that no validity constraint exists on notation declarations
3735 * But this function get called anyway ...
3736 *
3737 * returns 1 if valid or 0 otherwise
3738 */
3739
3740 int
3741 xmlValidateNotationDecl(xmlValidCtxtPtr ctxt ATTRIBUTE_UNUSED, xmlDocPtr doc ATTRIBUTE_UNUSED,
3742 xmlNotationPtr nota ATTRIBUTE_UNUSED) {
3743 int ret = 1;
3744
3745 return(ret);
3746 }
3747
3748 /**
3749 * xmlValidateAttributeValueInternal:
3750 * @doc: the document
3751 * @type: an attribute type
3752 * @value: an attribute value
3753 *
3754 * Validate that the given attribute value match the proper production
3755 *
3756 * returns 1 if valid or 0 otherwise
3757 */
3758
3759 static int
3760 xmlValidateAttributeValueInternal(xmlDocPtr doc, xmlAttributeType type,
3761 const xmlChar *value) {
3762 switch (type) {
3763 case XML_ATTRIBUTE_ENTITIES:
3764 case XML_ATTRIBUTE_IDREFS:
3765 return(xmlValidateNamesValueInternal(doc, value));
3766 case XML_ATTRIBUTE_ENTITY:
3767 case XML_ATTRIBUTE_IDREF:
3768 case XML_ATTRIBUTE_ID:
3769 case XML_ATTRIBUTE_NOTATION:
3770 return(xmlValidateNameValueInternal(doc, value));
3771 case XML_ATTRIBUTE_NMTOKENS:
3772 case XML_ATTRIBUTE_ENUMERATION:
3773 return(xmlValidateNmtokensValueInternal(doc, value));
3774 case XML_ATTRIBUTE_NMTOKEN:
3775 return(xmlValidateNmtokenValueInternal(doc, value));
3776 case XML_ATTRIBUTE_CDATA:
3777 break;
3778 }
3779 return(1);
3780 }
3781
3782 /**
3783 * xmlValidateAttributeValue:
3784 * @type: an attribute type
3785 * @value: an attribute value
3786 *
3787 * Validate that the given attribute value match the proper production
3788 *
3789 * [ VC: ID ]
3790 * Values of type ID must match the Name production....
3791 *
3792 * [ VC: IDREF ]
3793 * Values of type IDREF must match the Name production, and values
3794 * of type IDREFS must match Names ...
3795 *
3796 * [ VC: Entity Name ]
3797 * Values of type ENTITY must match the Name production, values
3798 * of type ENTITIES must match Names ...
3799 *
3800 * [ VC: Name Token ]
3801 * Values of type NMTOKEN must match the Nmtoken production; values
3802 * of type NMTOKENS must match Nmtokens.
3803 *
3804 * returns 1 if valid or 0 otherwise
3805 */
3806 int
3807 xmlValidateAttributeValue(xmlAttributeType type, const xmlChar *value) {
3808 return(xmlValidateAttributeValueInternal(NULL, type, value));
3809 }
3810
3811 /**
3812 * xmlValidateAttributeValue2:
3813 * @ctxt: the validation context
3814 * @doc: the document
3815 * @name: the attribute name (used for error reporting only)
3816 * @type: the attribute type
3817 * @value: the attribute value
3818 *
3819 * Validate that the given attribute value match a given type.
3820 * This typically cannot be done before having finished parsing
3821 * the subsets.
3822 *
3823 * [ VC: IDREF ]
3824 * Values of type IDREF must match one of the declared IDs
3825 * Values of type IDREFS must match a sequence of the declared IDs
3826 * each Name must match the value of an ID attribute on some element
3827 * in the XML document; i.e. IDREF values must match the value of
3828 * some ID attribute
3829 *
3830 * [ VC: Entity Name ]
3831 * Values of type ENTITY must match one declared entity
3832 * Values of type ENTITIES must match a sequence of declared entities
3833 *
3834 * [ VC: Notation Attributes ]
3835 * all notation names in the declaration must be declared.
3836 *
3837 * returns 1 if valid or 0 otherwise
3838 */
3839
3840 static int
3841 xmlValidateAttributeValue2(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3842 const xmlChar *name, xmlAttributeType type, const xmlChar *value) {
3843 int ret = 1;
3844 switch (type) {
3845 case XML_ATTRIBUTE_IDREFS:
3846 case XML_ATTRIBUTE_IDREF:
3847 case XML_ATTRIBUTE_ID:
3848 case XML_ATTRIBUTE_NMTOKENS:
3849 case XML_ATTRIBUTE_ENUMERATION:
3850 case XML_ATTRIBUTE_NMTOKEN:
3851 case XML_ATTRIBUTE_CDATA:
3852 break;
3853 case XML_ATTRIBUTE_ENTITY: {
3854 xmlEntityPtr ent;
3855
3856 ent = xmlGetDocEntity(doc, value);
3857 /* yeah it's a bit messy... */
3858 if ((ent == NULL) && (doc->standalone == 1)) {
3859 doc->standalone = 0;
3860 ent = xmlGetDocEntity(doc, value);
3861 }
3862 if (ent == NULL) {
3863 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3864 XML_DTD_UNKNOWN_ENTITY,
3865 "ENTITY attribute %s reference an unknown entity \"%s\"\n",
3866 name, value, NULL);
3867 ret = 0;
3868 } else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
3869 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3870 XML_DTD_ENTITY_TYPE,
3871 "ENTITY attribute %s reference an entity \"%s\" of wrong type\n",
3872 name, value, NULL);
3873 ret = 0;
3874 }
3875 break;
3876 }
3877 case XML_ATTRIBUTE_ENTITIES: {
3878 xmlChar *dup, *nam = NULL, *cur, save;
3879 xmlEntityPtr ent;
3880
3881 dup = xmlStrdup(value);
3882 if (dup == NULL)
3883 return(0);
3884 cur = dup;
3885 while (*cur != 0) {
3886 nam = cur;
3887 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
3888 save = *cur;
3889 *cur = 0;
3890 ent = xmlGetDocEntity(doc, nam);
3891 if (ent == NULL) {
3892 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3893 XML_DTD_UNKNOWN_ENTITY,
3894 "ENTITIES attribute %s reference an unknown entity \"%s\"\n",
3895 name, nam, NULL);
3896 ret = 0;
3897 } else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
3898 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3899 XML_DTD_ENTITY_TYPE,
3900 "ENTITIES attribute %s reference an entity \"%s\" of wrong type\n",
3901 name, nam, NULL);
3902 ret = 0;
3903 }
3904 if (save == 0)
3905 break;
3906 *cur = save;
3907 while (IS_BLANK_CH(*cur)) cur++;
3908 }
3909 xmlFree(dup);
3910 break;
3911 }
3912 case XML_ATTRIBUTE_NOTATION: {
3913 xmlNotationPtr nota;
3914
3915 nota = xmlGetDtdNotationDesc(doc->intSubset, value);
3916 if ((nota == NULL) && (doc->extSubset != NULL))
3917 nota = xmlGetDtdNotationDesc(doc->extSubset, value);
3918
3919 if (nota == NULL) {
3920 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3921 XML_DTD_UNKNOWN_NOTATION,
3922 "NOTATION attribute %s reference an unknown notation \"%s\"\n",
3923 name, value, NULL);
3924 ret = 0;
3925 }
3926 break;
3927 }
3928 }
3929 return(ret);
3930 }
3931
3932 /**
3933 * xmlValidCtxtNormalizeAttributeValue:
3934 * @ctxt: the validation context
3935 * @doc: the document
3936 * @elem: the parent
3937 * @name: the attribute name
3938 * @value: the attribute value
3939 * @ctxt: the validation context or NULL
3940 *
3941 * Does the validation related extra step of the normalization of attribute
3942 * values:
3943 *
3944 * If the declared value is not CDATA, then the XML processor must further
3945 * process the normalized attribute value by discarding any leading and
3946 * trailing space (#x20) characters, and by replacing sequences of space
3947 * (#x20) characters by single space (#x20) character.
3948 *
3949 * Also check VC: Standalone Document Declaration in P32, and update
3950 * ctxt->valid accordingly
3951 *
3952 * returns a new normalized string if normalization is needed, NULL otherwise
3953 * the caller must free the returned value.
3954 */
3955
3956 xmlChar *
3957 xmlValidCtxtNormalizeAttributeValue(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3958 xmlNodePtr elem, const xmlChar *name, const xmlChar *value) {
3959 xmlChar *ret, *dst;
3960 const xmlChar *src;
3961 xmlAttributePtr attrDecl = NULL;
3962 int extsubset = 0;
3963
3964 if (doc == NULL) return(NULL);
3965 if (elem == NULL) return(NULL);
3966 if (name == NULL) return(NULL);
3967 if (value == NULL) return(NULL);
3968
3969 if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
3970 xmlChar fn[50];
3971 xmlChar *fullname;
3972
3973 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
3974 if (fullname == NULL)
3975 return(NULL);
3976 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, name);
3977 if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
3978 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname, name);
3979 if (attrDecl != NULL)
3980 extsubset = 1;
3981 }
3982 if ((fullname != fn) && (fullname != elem->name))
3983 xmlFree(fullname);
3984 }
3985 if ((attrDecl == NULL) && (doc->intSubset != NULL))
3986 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, name);
3987 if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
3988 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, elem->name, name);
3989 if (attrDecl != NULL)
3990 extsubset = 1;
3991 }
3992
3993 if (attrDecl == NULL)
3994 return(NULL);
3995 if (attrDecl->atype == XML_ATTRIBUTE_CDATA)
3996 return(NULL);
3997
3998 ret = xmlStrdup(value);
3999 if (ret == NULL)
4000 return(NULL);
4001 src = value;
4002 dst = ret;
4003 while (*src == 0x20) src++;
4004 while (*src != 0) {
4005 if (*src == 0x20) {
4006 while (*src == 0x20) src++;
4007 if (*src != 0)
4008 *dst++ = 0x20;
4009 } else {
4010 *dst++ = *src++;
4011 }
4012 }
4013 *dst = 0;
4014 if ((doc->standalone) && (extsubset == 1) && (!xmlStrEqual(value, ret))) {
4015 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_STANDALONE,
4016 "standalone: %s on %s value had to be normalized based on external subset declaration\n",
4017 name, elem->name, NULL);
4018 ctxt->valid = 0;
4019 }
4020 return(ret);
4021 }
4022
4023 /**
4024 * xmlValidNormalizeAttributeValue:
4025 * @doc: the document
4026 * @elem: the parent
4027 * @name: the attribute name
4028 * @value: the attribute value
4029 *
4030 * Does the validation related extra step of the normalization of attribute
4031 * values:
4032 *
4033 * If the declared value is not CDATA, then the XML processor must further
4034 * process the normalized attribute value by discarding any leading and
4035 * trailing space (#x20) characters, and by replacing sequences of space
4036 * (#x20) characters by single space (#x20) character.
4037 *
4038 * Returns a new normalized string if normalization is needed, NULL otherwise
4039 * the caller must free the returned value.
4040 */
4041
4042 xmlChar *
4043 xmlValidNormalizeAttributeValue(xmlDocPtr doc, xmlNodePtr elem,
4044 const xmlChar *name, const xmlChar *value) {
4045 xmlChar *ret, *dst;
4046 const xmlChar *src;
4047 xmlAttributePtr attrDecl = NULL;
4048
4049 if (doc == NULL) return(NULL);
4050 if (elem == NULL) return(NULL);
4051 if (name == NULL) return(NULL);
4052 if (value == NULL) return(NULL);
4053
4054 if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
4055 xmlChar fn[50];
4056 xmlChar *fullname;
4057
4058 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4059 if (fullname == NULL)
4060 return(NULL);
4061 if ((fullname != fn) && (fullname != elem->name))
4062 xmlFree(fullname);
4063 }
4064 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, name);
4065 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4066 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, elem->name, name);
4067
4068 if (attrDecl == NULL)
4069 return(NULL);
4070 if (attrDecl->atype == XML_ATTRIBUTE_CDATA)
4071 return(NULL);
4072
4073 ret = xmlStrdup(value);
4074 if (ret == NULL)
4075 return(NULL);
4076 src = value;
4077 dst = ret;
4078 while (*src == 0x20) src++;
4079 while (*src != 0) {
4080 if (*src == 0x20) {
4081 while (*src == 0x20) src++;
4082 if (*src != 0)
4083 *dst++ = 0x20;
4084 } else {
4085 *dst++ = *src++;
4086 }
4087 }
4088 *dst = 0;
4089 return(ret);
4090 }
4091
4092 static void
4093 xmlValidateAttributeIdCallback(xmlAttributePtr attr, int *count,
4094 const xmlChar* name ATTRIBUTE_UNUSED) {
4095 if (attr->atype == XML_ATTRIBUTE_ID) (*count)++;
4096 }
4097
4098 /**
4099 * xmlValidateAttributeDecl:
4100 * @ctxt: the validation context
4101 * @doc: a document instance
4102 * @attr: an attribute definition
4103 *
4104 * Try to validate a single attribute definition
4105 * basically it does the following checks as described by the
4106 * XML-1.0 recommendation:
4107 * - [ VC: Attribute Default Legal ]
4108 * - [ VC: Enumeration ]
4109 * - [ VC: ID Attribute Default ]
4110 *
4111 * The ID/IDREF uniqueness and matching are done separately
4112 *
4113 * returns 1 if valid or 0 otherwise
4114 */
4115
4116 int
4117 xmlValidateAttributeDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4118 xmlAttributePtr attr) {
4119 int ret = 1;
4120 int val;
4121 CHECK_DTD;
4122 if(attr == NULL) return(1);
4123
4124 /* Attribute Default Legal */
4125 /* Enumeration */
4126 if (attr->defaultValue != NULL) {
4127 val = xmlValidateAttributeValueInternal(doc, attr->atype,
4128 attr->defaultValue);
4129 if (val == 0) {
4130 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_DEFAULT,
4131 "Syntax of default value for attribute %s of %s is not valid\n",
4132 attr->name, attr->elem, NULL);
4133 }
4134 ret &= val;
4135 }
4136
4137 /* ID Attribute Default */
4138 if ((attr->atype == XML_ATTRIBUTE_ID)&&
4139 (attr->def != XML_ATTRIBUTE_IMPLIED) &&
4140 (attr->def != XML_ATTRIBUTE_REQUIRED)) {
4141 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_FIXED,
4142 "ID attribute %s of %s is not valid must be #IMPLIED or #REQUIRED\n",
4143 attr->name, attr->elem, NULL);
4144 ret = 0;
4145 }
4146
4147 /* One ID per Element Type */
4148 if (attr->atype == XML_ATTRIBUTE_ID) {
4149 int nbId;
4150
4151 /* the trick is that we parse DtD as their own internal subset */
4152 xmlElementPtr elem = xmlGetDtdElementDesc(doc->intSubset,
4153 attr->elem);
4154 if (elem != NULL) {
4155 nbId = xmlScanIDAttributeDecl(NULL, elem, 0);
4156 } else {
4157 xmlAttributeTablePtr table;
4158
4159 /*
4160 * The attribute may be declared in the internal subset and the
4161 * element in the external subset.
4162 */
4163 nbId = 0;
4164 if (doc->intSubset != NULL) {
4165 table = (xmlAttributeTablePtr) doc->intSubset->attributes;
4166 xmlHashScan3(table, NULL, NULL, attr->elem, (xmlHashScanner)
4167 xmlValidateAttributeIdCallback, &nbId);
4168 }
4169 }
4170 if (nbId > 1) {
4171
4172 xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
4173 "Element %s has %d ID attribute defined in the internal subset : %s\n",
4174 attr->elem, nbId, attr->name);
4175 } else if (doc->extSubset != NULL) {
4176 int extId = 0;
4177 elem = xmlGetDtdElementDesc(doc->extSubset, attr->elem);
4178 if (elem != NULL) {
4179 extId = xmlScanIDAttributeDecl(NULL, elem, 0);
4180 }
4181 if (extId > 1) {
4182 xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
4183 "Element %s has %d ID attribute defined in the external subset : %s\n",
4184 attr->elem, extId, attr->name);
4185 } else if (extId + nbId > 1) {
4186 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
4187 "Element %s has ID attributes defined in the internal and external subset : %s\n",
4188 attr->elem, attr->name, NULL);
4189 }
4190 }
4191 }
4192
4193 /* Validity Constraint: Enumeration */
4194 if ((attr->defaultValue != NULL) && (attr->tree != NULL)) {
4195 xmlEnumerationPtr tree = attr->tree;
4196 while (tree != NULL) {
4197 if (xmlStrEqual(tree->name, attr->defaultValue)) break;
4198 tree = tree->next;
4199 }
4200 if (tree == NULL) {
4201 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_VALUE,
4202 "Default value \"%s\" for attribute %s of %s is not among the enumerated set\n",
4203 attr->defaultValue, attr->name, attr->elem);
4204 ret = 0;
4205 }
4206 }
4207
4208 return(ret);
4209 }
4210
4211 /**
4212 * xmlValidateElementDecl:
4213 * @ctxt: the validation context
4214 * @doc: a document instance
4215 * @elem: an element definition
4216 *
4217 * Try to validate a single element definition
4218 * basically it does the following checks as described by the
4219 * XML-1.0 recommendation:
4220 * - [ VC: One ID per Element Type ]
4221 * - [ VC: No Duplicate Types ]
4222 * - [ VC: Unique Element Type Declaration ]
4223 *
4224 * returns 1 if valid or 0 otherwise
4225 */
4226
4227 int
4228 xmlValidateElementDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4229 xmlElementPtr elem) {
4230 int ret = 1;
4231 xmlElementPtr tst;
4232
4233 CHECK_DTD;
4234
4235 if (elem == NULL) return(1);
4236
4237 #if 0
4238 #ifdef LIBXML_REGEXP_ENABLED
4239 /* Build the regexp associated to the content model */
4240 ret = xmlValidBuildContentModel(ctxt, elem);
4241 #endif
4242 #endif
4243
4244 /* No Duplicate Types */
4245 if (elem->etype == XML_ELEMENT_TYPE_MIXED) {
4246 xmlElementContentPtr cur, next;
4247 const xmlChar *name;
4248
4249 cur = elem->content;
4250 while (cur != NULL) {
4251 if (cur->type != XML_ELEMENT_CONTENT_OR) break;
4252 if (cur->c1 == NULL) break;
4253 if (cur->c1->type == XML_ELEMENT_CONTENT_ELEMENT) {
4254 name = cur->c1->name;
4255 next = cur->c2;
4256 while (next != NULL) {
4257 if (next->type == XML_ELEMENT_CONTENT_ELEMENT) {
4258 if ((xmlStrEqual(next->name, name)) &&
4259 (xmlStrEqual(next->prefix, cur->c1->prefix))) {
4260 if (cur->c1->prefix == NULL) {
4261 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4262 "Definition of %s has duplicate references of %s\n",
4263 elem->name, name, NULL);
4264 } else {
4265 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4266 "Definition of %s has duplicate references of %s:%s\n",
4267 elem->name, cur->c1->prefix, name);
4268 }
4269 ret = 0;
4270 }
4271 break;
4272 }
4273 if (next->c1 == NULL) break;
4274 if (next->c1->type != XML_ELEMENT_CONTENT_ELEMENT) break;
4275 if ((xmlStrEqual(next->c1->name, name)) &&
4276 (xmlStrEqual(next->c1->prefix, cur->c1->prefix))) {
4277 if (cur->c1->prefix == NULL) {
4278 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4279 "Definition of %s has duplicate references to %s\n",
4280 elem->name, name, NULL);
4281 } else {
4282 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4283 "Definition of %s has duplicate references to %s:%s\n",
4284 elem->name, cur->c1->prefix, name);
4285 }
4286 ret = 0;
4287 }
4288 next = next->c2;
4289 }
4290 }
4291 cur = cur->c2;
4292 }
4293 }
4294
4295 /* VC: Unique Element Type Declaration */
4296 tst = xmlGetDtdElementDesc(doc->intSubset, elem->name);
4297 if ((tst != NULL ) && (tst != elem) &&
4298 ((tst->prefix == elem->prefix) ||
4299 (xmlStrEqual(tst->prefix, elem->prefix))) &&
4300 (tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
4301 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4302 "Redefinition of element %s\n",
4303 elem->name, NULL, NULL);
4304 ret = 0;
4305 }
4306 tst = xmlGetDtdElementDesc(doc->extSubset, elem->name);
4307 if ((tst != NULL ) && (tst != elem) &&
4308 ((tst->prefix == elem->prefix) ||
4309 (xmlStrEqual(tst->prefix, elem->prefix))) &&
4310 (tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
4311 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4312 "Redefinition of element %s\n",
4313 elem->name, NULL, NULL);
4314 ret = 0;
4315 }
4316 /* One ID per Element Type
4317 * already done when registering the attribute
4318 if (xmlScanIDAttributeDecl(ctxt, elem) > 1) {
4319 ret = 0;
4320 } */
4321 return(ret);
4322 }
4323
4324 /**
4325 * xmlValidateOneAttribute:
4326 * @ctxt: the validation context
4327 * @doc: a document instance
4328 * @elem: an element instance
4329 * @attr: an attribute instance
4330 * @value: the attribute value (without entities processing)
4331 *
4332 * Try to validate a single attribute for an element
4333 * basically it does the following checks as described by the
4334 * XML-1.0 recommendation:
4335 * - [ VC: Attribute Value Type ]
4336 * - [ VC: Fixed Attribute Default ]
4337 * - [ VC: Entity Name ]
4338 * - [ VC: Name Token ]
4339 * - [ VC: ID ]
4340 * - [ VC: IDREF ]
4341 * - [ VC: Entity Name ]
4342 * - [ VC: Notation Attributes ]
4343 *
4344 * The ID/IDREF uniqueness and matching are done separately
4345 *
4346 * returns 1 if valid or 0 otherwise
4347 */
4348
4349 int
4350 xmlValidateOneAttribute(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4351 xmlNodePtr elem, xmlAttrPtr attr, const xmlChar *value)
4352 {
4353 xmlAttributePtr attrDecl = NULL;
4354 int val;
4355 int ret = 1;
4356
4357 CHECK_DTD;
4358 if ((elem == NULL) || (elem->name == NULL)) return(0);
4359 if ((attr == NULL) || (attr->name == NULL)) return(0);
4360
4361 if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
4362 xmlChar fn[50];
4363 xmlChar *fullname;
4364
4365 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4366 if (fullname == NULL)
4367 return(0);
4368 if (attr->ns != NULL) {
4369 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
4370 attr->name, attr->ns->prefix);
4371 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4372 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
4373 attr->name, attr->ns->prefix);
4374 } else {
4375 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, attr->name);
4376 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4377 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4378 fullname, attr->name);
4379 }
4380 if ((fullname != fn) && (fullname != elem->name))
4381 xmlFree(fullname);
4382 }
4383 if (attrDecl == NULL) {
4384 if (attr->ns != NULL) {
4385 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, elem->name,
4386 attr->name, attr->ns->prefix);
4387 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4388 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, elem->name,
4389 attr->name, attr->ns->prefix);
4390 } else {
4391 attrDecl = xmlGetDtdAttrDesc(doc->intSubset,
4392 elem->name, attr->name);
4393 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4394 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4395 elem->name, attr->name);
4396 }
4397 }
4398
4399
4400 /* Validity Constraint: Attribute Value Type */
4401 if (attrDecl == NULL) {
4402 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
4403 "No declaration for attribute %s of element %s\n",
4404 attr->name, elem->name, NULL);
4405 return(0);
4406 }
4407 attr->atype = attrDecl->atype;
4408
4409 val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
4410 if (val == 0) {
4411 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4412 "Syntax of value for attribute %s of %s is not valid\n",
4413 attr->name, elem->name, NULL);
4414 ret = 0;
4415 }
4416
4417 /* Validity constraint: Fixed Attribute Default */
4418 if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4419 if (!xmlStrEqual(value, attrDecl->defaultValue)) {
4420 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
4421 "Value for attribute %s of %s is different from default \"%s\"\n",
4422 attr->name, elem->name, attrDecl->defaultValue);
4423 ret = 0;
4424 }
4425 }
4426
4427 /* Validity Constraint: ID uniqueness */
4428 if (attrDecl->atype == XML_ATTRIBUTE_ID) {
4429 if (xmlAddID(ctxt, doc, value, attr) == NULL)
4430 ret = 0;
4431 }
4432
4433 if ((attrDecl->atype == XML_ATTRIBUTE_IDREF) ||
4434 (attrDecl->atype == XML_ATTRIBUTE_IDREFS)) {
4435 if (xmlAddRef(ctxt, doc, value, attr) == NULL)
4436 ret = 0;
4437 }
4438
4439 /* Validity Constraint: Notation Attributes */
4440 if (attrDecl->atype == XML_ATTRIBUTE_NOTATION) {
4441 xmlEnumerationPtr tree = attrDecl->tree;
4442 xmlNotationPtr nota;
4443
4444 /* First check that the given NOTATION was declared */
4445 nota = xmlGetDtdNotationDesc(doc->intSubset, value);
4446 if (nota == NULL)
4447 nota = xmlGetDtdNotationDesc(doc->extSubset, value);
4448
4449 if (nota == NULL) {
4450 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
4451 "Value \"%s\" for attribute %s of %s is not a declared Notation\n",
4452 value, attr->name, elem->name);
4453 ret = 0;
4454 }
4455
4456 /* Second, verify that it's among the list */
4457 while (tree != NULL) {
4458 if (xmlStrEqual(tree->name, value)) break;
4459 tree = tree->next;
4460 }
4461 if (tree == NULL) {
4462 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
4463 "Value \"%s\" for attribute %s of %s is not among the enumerated notations\n",
4464 value, attr->name, elem->name);
4465 ret = 0;
4466 }
4467 }
4468
4469 /* Validity Constraint: Enumeration */
4470 if (attrDecl->atype == XML_ATTRIBUTE_ENUMERATION) {
4471 xmlEnumerationPtr tree = attrDecl->tree;
4472 while (tree != NULL) {
4473 if (xmlStrEqual(tree->name, value)) break;
4474 tree = tree->next;
4475 }
4476 if (tree == NULL) {
4477 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4478 "Value \"%s\" for attribute %s of %s is not among the enumerated set\n",
4479 value, attr->name, elem->name);
4480 ret = 0;
4481 }
4482 }
4483
4484 /* Fixed Attribute Default */
4485 if ((attrDecl->def == XML_ATTRIBUTE_FIXED) &&
4486 (!xmlStrEqual(attrDecl->defaultValue, value))) {
4487 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4488 "Value for attribute %s of %s must be \"%s\"\n",
4489 attr->name, elem->name, attrDecl->defaultValue);
4490 ret = 0;
4491 }
4492
4493 /* Extra check for the attribute value */
4494 ret &= xmlValidateAttributeValue2(ctxt, doc, attr->name,
4495 attrDecl->atype, value);
4496
4497 return(ret);
4498 }
4499
4500 /**
4501 * xmlValidateOneNamespace:
4502 * @ctxt: the validation context
4503 * @doc: a document instance
4504 * @elem: an element instance
4505 * @prefix: the namespace prefix
4506 * @ns: an namespace declaration instance
4507 * @value: the attribute value (without entities processing)
4508 *
4509 * Try to validate a single namespace declaration for an element
4510 * basically it does the following checks as described by the
4511 * XML-1.0 recommendation:
4512 * - [ VC: Attribute Value Type ]
4513 * - [ VC: Fixed Attribute Default ]
4514 * - [ VC: Entity Name ]
4515 * - [ VC: Name Token ]
4516 * - [ VC: ID ]
4517 * - [ VC: IDREF ]
4518 * - [ VC: Entity Name ]
4519 * - [ VC: Notation Attributes ]
4520 *
4521 * The ID/IDREF uniqueness and matching are done separately
4522 *
4523 * returns 1 if valid or 0 otherwise
4524 */
4525
4526 int
4527 xmlValidateOneNamespace(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4528 xmlNodePtr elem, const xmlChar *prefix, xmlNsPtr ns, const xmlChar *value) {
4529 /* xmlElementPtr elemDecl; */
4530 xmlAttributePtr attrDecl = NULL;
4531 int val;
4532 int ret = 1;
4533
4534 CHECK_DTD;
4535 if ((elem == NULL) || (elem->name == NULL)) return(0);
4536 if ((ns == NULL) || (ns->href == NULL)) return(0);
4537
4538 if (prefix != NULL) {
4539 xmlChar fn[50];
4540 xmlChar *fullname;
4541
4542 fullname = xmlBuildQName(elem->name, prefix, fn, 50);
4543 if (fullname == NULL) {
4544 xmlVErrMemory(ctxt, "Validating namespace");
4545 return(0);
4546 }
4547 if (ns->prefix != NULL) {
4548 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
4549 ns->prefix, BAD_CAST "xmlns");
4550 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4551 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
4552 ns->prefix, BAD_CAST "xmlns");
4553 } else {
4554 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname,
4555 BAD_CAST "xmlns");
4556 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4557 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname,
4558 BAD_CAST "xmlns");
4559 }
4560 if ((fullname != fn) && (fullname != elem->name))
4561 xmlFree(fullname);
4562 }
4563 if (attrDecl == NULL) {
4564 if (ns->prefix != NULL) {
4565 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, elem->name,
4566 ns->prefix, BAD_CAST "xmlns");
4567 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4568 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, elem->name,
4569 ns->prefix, BAD_CAST "xmlns");
4570 } else {
4571 attrDecl = xmlGetDtdAttrDesc(doc->intSubset,
4572 elem->name, BAD_CAST "xmlns");
4573 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4574 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4575 elem->name, BAD_CAST "xmlns");
4576 }
4577 }
4578
4579
4580 /* Validity Constraint: Attribute Value Type */
4581 if (attrDecl == NULL) {
4582 if (ns->prefix != NULL) {
4583 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
4584 "No declaration for attribute xmlns:%s of element %s\n",
4585 ns->prefix, elem->name, NULL);
4586 } else {
4587 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
4588 "No declaration for attribute xmlns of element %s\n",
4589 elem->name, NULL, NULL);
4590 }
4591 return(0);
4592 }
4593
4594 val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
4595 if (val == 0) {
4596 if (ns->prefix != NULL) {
4597 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
4598 "Syntax of value for attribute xmlns:%s of %s is not valid\n",
4599 ns->prefix, elem->name, NULL);
4600 } else {
4601 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
4602 "Syntax of value for attribute xmlns of %s is not valid\n",
4603 elem->name, NULL, NULL);
4604 }
4605 ret = 0;
4606 }
4607
4608 /* Validity constraint: Fixed Attribute Default */
4609 if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4610 if (!xmlStrEqual(value, attrDecl->defaultValue)) {
4611 if (ns->prefix != NULL) {
4612 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
4613 "Value for attribute xmlns:%s of %s is different from default \"%s\"\n",
4614 ns->prefix, elem->name, attrDecl->defaultValue);
4615 } else {
4616 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
4617 "Value for attribute xmlns of %s is different from default \"%s\"\n",
4618 elem->name, attrDecl->defaultValue, NULL);
4619 }
4620 ret = 0;
4621 }
4622 }
4623
4624 /* Validity Constraint: ID uniqueness */
4625 if (attrDecl->atype == XML_ATTRIBUTE_ID) {
4626 if (xmlAddID(ctxt, doc, value, (xmlAttrPtr) ns) == NULL)
4627 ret = 0;
4628 }
4629
4630 if ((attrDecl->atype == XML_ATTRIBUTE_IDREF) ||
4631 (attrDecl->atype == XML_ATTRIBUTE_IDREFS)) {
4632 if (xmlAddRef(ctxt, doc, value, (xmlAttrPtr) ns) == NULL)
4633 ret = 0;
4634 }
4635
4636 /* Validity Constraint: Notation Attributes */
4637 if (attrDecl->atype == XML_ATTRIBUTE_NOTATION) {
4638 xmlEnumerationPtr tree = attrDecl->tree;
4639 xmlNotationPtr nota;
4640
4641 /* First check that the given NOTATION was declared */
4642 nota = xmlGetDtdNotationDesc(doc->intSubset, value);
4643 if (nota == NULL)
4644 nota = xmlGetDtdNotationDesc(doc->extSubset, value);
4645
4646 if (nota == NULL) {
4647 if (ns->prefix != NULL) {
4648 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
4649 "Value \"%s\" for attribute xmlns:%s of %s is not a declared Notation\n",
4650 value, ns->prefix, elem->name);
4651 } else {
4652 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
4653 "Value \"%s\" for attribute xmlns of %s is not a declared Notation\n",
4654 value, elem->name, NULL);
4655 }
4656 ret = 0;
4657 }
4658
4659 /* Second, verify that it's among the list */
4660 while (tree != NULL) {
4661 if (xmlStrEqual(tree->name, value)) break;
4662 tree = tree->next;
4663 }
4664 if (tree == NULL) {
4665 if (ns->prefix != NULL) {
4666 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
4667 "Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated notations\n",
4668 value, ns->prefix, elem->name);
4669 } else {
4670 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
4671 "Value \"%s\" for attribute xmlns of %s is not among the enumerated notations\n",
4672 value, elem->name, NULL);
4673 }
4674 ret = 0;
4675 }
4676 }
4677
4678 /* Validity Constraint: Enumeration */
4679 if (attrDecl->atype == XML_ATTRIBUTE_ENUMERATION) {
4680 xmlEnumerationPtr tree = attrDecl->tree;
4681 while (tree != NULL) {
4682 if (xmlStrEqual(tree->name, value)) break;
4683 tree = tree->next;
4684 }
4685 if (tree == NULL) {
4686 if (ns->prefix != NULL) {
4687 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4688 "Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated set\n",
4689 value, ns->prefix, elem->name);
4690 } else {
4691 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4692 "Value \"%s\" for attribute xmlns of %s is not among the enumerated set\n",
4693 value, elem->name, NULL);
4694 }
4695 ret = 0;
4696 }
4697 }
4698
4699 /* Fixed Attribute Default */
4700 if ((attrDecl->def == XML_ATTRIBUTE_FIXED) &&
4701 (!xmlStrEqual(attrDecl->defaultValue, value))) {
4702 if (ns->prefix != NULL) {
4703 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
4704 "Value for attribute xmlns:%s of %s must be \"%s\"\n",
4705 ns->prefix, elem->name, attrDecl->defaultValue);
4706 } else {
4707 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
4708 "Value for attribute xmlns of %s must be \"%s\"\n",
4709 elem->name, attrDecl->defaultValue, NULL);
4710 }
4711 ret = 0;
4712 }
4713
4714 /* Extra check for the attribute value */
4715 if (ns->prefix != NULL) {
4716 ret &= xmlValidateAttributeValue2(ctxt, doc, ns->prefix,
4717 attrDecl->atype, value);
4718 } else {
4719 ret &= xmlValidateAttributeValue2(ctxt, doc, BAD_CAST "xmlns",
4720 attrDecl->atype, value);
4721 }
4722
4723 return(ret);
4724 }
4725
4726 #ifndef LIBXML_REGEXP_ENABLED
4727 /**
4728 * xmlValidateSkipIgnorable:
4729 * @ctxt: the validation context
4730 * @child: the child list
4731 *
4732 * Skip ignorable elements w.r.t. the validation process
4733 *
4734 * returns the first element to consider for validation of the content model
4735 */
4736
4737 static xmlNodePtr
4738 xmlValidateSkipIgnorable(xmlNodePtr child) {
4739 while (child != NULL) {
4740 switch (child->type) {
4741 /* These things are ignored (skipped) during validation. */
4742 case XML_PI_NODE:
4743 case XML_COMMENT_NODE:
4744 case XML_XINCLUDE_START:
4745 case XML_XINCLUDE_END:
4746 child = child->next;
4747 break;
4748 case XML_TEXT_NODE:
4749 if (xmlIsBlankNode(child))
4750 child = child->next;
4751 else
4752 return(child);
4753 break;
4754 /* keep current node */
4755 default:
4756 return(child);
4757 }
4758 }
4759 return(child);
4760 }
4761
4762 /**
4763 * xmlValidateElementType:
4764 * @ctxt: the validation context
4765 *
4766 * Try to validate the content model of an element internal function
4767 *
4768 * returns 1 if valid or 0 ,-1 in case of error, -2 if an entity
4769 * reference is found and -3 if the validation succeeded but
4770 * the content model is not determinist.
4771 */
4772
4773 static int
4774 xmlValidateElementType(xmlValidCtxtPtr ctxt) {
4775 int ret = -1;
4776 int determinist = 1;
4777
4778 NODE = xmlValidateSkipIgnorable(NODE);
4779 if ((NODE == NULL) && (CONT == NULL))
4780 return(1);
4781 if ((NODE == NULL) &&
4782 ((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
4783 (CONT->ocur == XML_ELEMENT_CONTENT_OPT))) {
4784 return(1);
4785 }
4786 if (CONT == NULL) return(-1);
4787 if ((NODE != NULL) && (NODE->type == XML_ENTITY_REF_NODE))
4788 return(-2);
4789
4790 /*
4791 * We arrive here when more states need to be examined
4792 */
4793 cont:
4794
4795 /*
4796 * We just recovered from a rollback generated by a possible
4797 * epsilon transition, go directly to the analysis phase
4798 */
4799 if (STATE == ROLLBACK_PARENT) {
4800 DEBUG_VALID_MSG("restored parent branch");
4801 DEBUG_VALID_STATE(NODE, CONT)
4802 ret = 1;
4803 goto analyze;
4804 }
4805
4806 DEBUG_VALID_STATE(NODE, CONT)
4807 /*
4808 * we may have to save a backup state here. This is the equivalent
4809 * of handling epsilon transition in NFAs.
4810 */
4811 if ((CONT != NULL) &&
4812 ((CONT->parent == NULL) ||
4813 (CONT->parent->type != XML_ELEMENT_CONTENT_OR)) &&
4814 ((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
4815 (CONT->ocur == XML_ELEMENT_CONTENT_OPT) ||
4816 ((CONT->ocur == XML_ELEMENT_CONTENT_PLUS) && (OCCURRENCE)))) {
4817 DEBUG_VALID_MSG("saving parent branch");
4818 if (vstateVPush(ctxt, CONT, NODE, DEPTH, OCCURS, ROLLBACK_PARENT) < 0)
4819 return(0);
4820 }
4821
4822
4823 /*
4824 * Check first if the content matches
4825 */
4826 switch (CONT->type) {
4827 case XML_ELEMENT_CONTENT_PCDATA:
4828 if (NODE == NULL) {
4829 DEBUG_VALID_MSG("pcdata failed no node");
4830 ret = 0;
4831 break;
4832 }
4833 if (NODE->type == XML_TEXT_NODE) {
4834 DEBUG_VALID_MSG("pcdata found, skip to next");
4835 /*
4836 * go to next element in the content model
4837 * skipping ignorable elems
4838 */
4839 do {
4840 NODE = NODE->next;
4841 NODE = xmlValidateSkipIgnorable(NODE);
4842 if ((NODE != NULL) &&
4843 (NODE->type == XML_ENTITY_REF_NODE))
4844 return(-2);
4845 } while ((NODE != NULL) &&
4846 ((NODE->type != XML_ELEMENT_NODE) &&
4847 (NODE->type != XML_TEXT_NODE) &&
4848 (NODE->type != XML_CDATA_SECTION_NODE)));
4849 ret = 1;
4850 break;
4851 } else {
4852 DEBUG_VALID_MSG("pcdata failed");
4853 ret = 0;
4854 break;
4855 }
4856 break;
4857 case XML_ELEMENT_CONTENT_ELEMENT:
4858 if (NODE == NULL) {
4859 DEBUG_VALID_MSG("element failed no node");
4860 ret = 0;
4861 break;
4862 }
4863 ret = ((NODE->type == XML_ELEMENT_NODE) &&
4864 (xmlStrEqual(NODE->name, CONT->name)));
4865 if (ret == 1) {
4866 if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4867 ret = (CONT->prefix == NULL);
4868 } else if (CONT->prefix == NULL) {
4869 ret = 0;
4870 } else {
4871 ret = xmlStrEqual(NODE->ns->prefix, CONT->prefix);
4872 }
4873 }
4874 if (ret == 1) {
4875 DEBUG_VALID_MSG("element found, skip to next");
4876 /*
4877 * go to next element in the content model
4878 * skipping ignorable elems
4879 */
4880 do {
4881 NODE = NODE->next;
4882 NODE = xmlValidateSkipIgnorable(NODE);
4883 if ((NODE != NULL) &&
4884 (NODE->type == XML_ENTITY_REF_NODE))
4885 return(-2);
4886 } while ((NODE != NULL) &&
4887 ((NODE->type != XML_ELEMENT_NODE) &&
4888 (NODE->type != XML_TEXT_NODE) &&
4889 (NODE->type != XML_CDATA_SECTION_NODE)));
4890 } else {
4891 DEBUG_VALID_MSG("element failed");
4892 ret = 0;
4893 break;
4894 }
4895 break;
4896 case XML_ELEMENT_CONTENT_OR:
4897 /*
4898 * Small optimization.
4899 */
4900 if (CONT->c1->type == XML_ELEMENT_CONTENT_ELEMENT) {
4901 if ((NODE == NULL) ||
4902 (!xmlStrEqual(NODE->name, CONT->c1->name))) {
4903 DEPTH++;
4904 CONT = CONT->c2;
4905 goto cont;
4906 }
4907 if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4908 ret = (CONT->c1->prefix == NULL);
4909 } else if (CONT->c1->prefix == NULL) {
4910 ret = 0;
4911 } else {
4912 ret = xmlStrEqual(NODE->ns->prefix, CONT->c1->prefix);
4913 }
4914 if (ret == 0) {
4915 DEPTH++;
4916 CONT = CONT->c2;
4917 goto cont;
4918 }
4919 }
4920
4921 /*
4922 * save the second branch 'or' branch
4923 */
4924 DEBUG_VALID_MSG("saving 'or' branch");
4925 if (vstateVPush(ctxt, CONT->c2, NODE, (unsigned char)(DEPTH + 1),
4926 OCCURS, ROLLBACK_OR) < 0)
4927 return(-1);
4928 DEPTH++;
4929 CONT = CONT->c1;
4930 goto cont;
4931 case XML_ELEMENT_CONTENT_SEQ:
4932 /*
4933 * Small optimization.
4934 */
4935 if ((CONT->c1->type == XML_ELEMENT_CONTENT_ELEMENT) &&
4936 ((CONT->c1->ocur == XML_ELEMENT_CONTENT_OPT) ||
4937 (CONT->c1->ocur == XML_ELEMENT_CONTENT_MULT))) {
4938 if ((NODE == NULL) ||
4939 (!xmlStrEqual(NODE->name, CONT->c1->name))) {
4940 DEPTH++;
4941 CONT = CONT->c2;
4942 goto cont;
4943 }
4944 if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4945 ret = (CONT->c1->prefix == NULL);
4946 } else if (CONT->c1->prefix == NULL) {
4947 ret = 0;
4948 } else {
4949 ret = xmlStrEqual(NODE->ns->prefix, CONT->c1->prefix);
4950 }
4951 if (ret == 0) {
4952 DEPTH++;
4953 CONT = CONT->c2;
4954 goto cont;
4955 }
4956 }
4957 DEPTH++;
4958 CONT = CONT->c1;
4959 goto cont;
4960 }
4961
4962 /*
4963 * At this point handle going up in the tree
4964 */
4965 if (ret == -1) {
4966 DEBUG_VALID_MSG("error found returning");
4967 return(ret);
4968 }
4969 analyze:
4970 while (CONT != NULL) {
4971 /*
4972 * First do the analysis depending on the occurrence model at
4973 * this level.
4974 */
4975 if (ret == 0) {
4976 switch (CONT->ocur) {
4977 xmlNodePtr cur;
4978
4979 case XML_ELEMENT_CONTENT_ONCE:
4980 cur = ctxt->vstate->node;
4981 DEBUG_VALID_MSG("Once branch failed, rollback");
4982 if (vstateVPop(ctxt) < 0 ) {
4983 DEBUG_VALID_MSG("exhaustion, failed");
4984 return(0);
4985 }
4986 if (cur != ctxt->vstate->node)
4987 determinist = -3;
4988 goto cont;
4989 case XML_ELEMENT_CONTENT_PLUS:
4990 if (OCCURRENCE == 0) {
4991 cur = ctxt->vstate->node;
4992 DEBUG_VALID_MSG("Plus branch failed, rollback");
4993 if (vstateVPop(ctxt) < 0 ) {
4994 DEBUG_VALID_MSG("exhaustion, failed");
4995 return(0);
4996 }
4997 if (cur != ctxt->vstate->node)
4998 determinist = -3;
4999 goto cont;
5000 }
5001 DEBUG_VALID_MSG("Plus branch found");
5002 ret = 1;
5003 break;
5004 case XML_ELEMENT_CONTENT_MULT:
5005 #ifdef DEBUG_VALID_ALGO
5006 if (OCCURRENCE == 0) {
5007 DEBUG_VALID_MSG("Mult branch failed");
5008 } else {
5009 DEBUG_VALID_MSG("Mult branch found");
5010 }
5011 #endif
5012 ret = 1;
5013 break;
5014 case XML_ELEMENT_CONTENT_OPT:
5015 DEBUG_VALID_MSG("Option branch failed");
5016 ret = 1;
5017 break;
5018 }
5019 } else {
5020 switch (CONT->ocur) {
5021 case XML_ELEMENT_CONTENT_OPT:
5022 DEBUG_VALID_MSG("Option branch succeeded");
5023 ret = 1;
5024 break;
5025 case XML_ELEMENT_CONTENT_ONCE:
5026 DEBUG_VALID_MSG("Once branch succeeded");
5027 ret = 1;
5028 break;
5029 case XML_ELEMENT_CONTENT_PLUS:
5030 if (STATE == ROLLBACK_PARENT) {
5031 DEBUG_VALID_MSG("Plus branch rollback");
5032 ret = 1;
5033 break;
5034 }
5035 if (NODE == NULL) {
5036 DEBUG_VALID_MSG("Plus branch exhausted");
5037 ret = 1;
5038 break;
5039 }
5040 DEBUG_VALID_MSG("Plus branch succeeded, continuing");
5041 SET_OCCURRENCE;
5042 goto cont;
5043 case XML_ELEMENT_CONTENT_MULT:
5044 if (STATE == ROLLBACK_PARENT) {
5045 DEBUG_VALID_MSG("Mult branch rollback");
5046 ret = 1;
5047 break;
5048 }
5049 if (NODE == NULL) {
5050 DEBUG_VALID_MSG("Mult branch exhausted");
5051 ret = 1;
5052 break;
5053 }
5054 DEBUG_VALID_MSG("Mult branch succeeded, continuing");
5055 /* SET_OCCURRENCE; */
5056 goto cont;
5057 }
5058 }
5059 STATE = 0;
5060
5061 /*
5062 * Then act accordingly at the parent level
5063 */
5064 RESET_OCCURRENCE;
5065 if (CONT->parent == NULL)
5066 break;
5067
5068 switch (CONT->parent->type) {
5069 case XML_ELEMENT_CONTENT_PCDATA:
5070 DEBUG_VALID_MSG("Error: parent pcdata");
5071 return(-1);
5072 case XML_ELEMENT_CONTENT_ELEMENT:
5073 DEBUG_VALID_MSG("Error: parent element");
5074 return(-1);
5075 case XML_ELEMENT_CONTENT_OR:
5076 if (ret == 1) {
5077 DEBUG_VALID_MSG("Or succeeded");
5078 CONT = CONT->parent;
5079 DEPTH--;
5080 } else {
5081 DEBUG_VALID_MSG("Or failed");
5082 CONT = CONT->parent;
5083 DEPTH--;
5084 }
5085 break;
5086 case XML_ELEMENT_CONTENT_SEQ:
5087 if (ret == 0) {
5088 DEBUG_VALID_MSG("Sequence failed");
5089 CONT = CONT->parent;
5090 DEPTH--;
5091 } else if (CONT == CONT->parent->c1) {
5092 DEBUG_VALID_MSG("Sequence testing 2nd branch");
5093 CONT = CONT->parent->c2;
5094 goto cont;
5095 } else {
5096 DEBUG_VALID_MSG("Sequence succeeded");
5097 CONT = CONT->parent;
5098 DEPTH--;
5099 }
5100 }
5101 }
5102 if (NODE != NULL) {
5103 xmlNodePtr cur;
5104
5105 cur = ctxt->vstate->node;
5106 DEBUG_VALID_MSG("Failed, remaining input, rollback");
5107 if (vstateVPop(ctxt) < 0 ) {
5108 DEBUG_VALID_MSG("exhaustion, failed");
5109 return(0);
5110 }
5111 if (cur != ctxt->vstate->node)
5112 determinist = -3;
5113 goto cont;
5114 }
5115 if (ret == 0) {
5116 xmlNodePtr cur;
5117
5118 cur = ctxt->vstate->node;
5119 DEBUG_VALID_MSG("Failure, rollback");
5120 if (vstateVPop(ctxt) < 0 ) {
5121 DEBUG_VALID_MSG("exhaustion, failed");
5122 return(0);
5123 }
5124 if (cur != ctxt->vstate->node)
5125 determinist = -3;
5126 goto cont;
5127 }
5128 return(determinist);
5129 }
5130 #endif
5131
5132 /**
5133 * xmlSnprintfElements:
5134 * @buf: an output buffer
5135 * @size: the size of the buffer
5136 * @content: An element
5137 * @glob: 1 if one must print the englobing parenthesis, 0 otherwise
5138 *
5139 * This will dump the list of elements to the buffer
5140 * Intended just for the debug routine
5141 */
5142 static void
5143 xmlSnprintfElements(char *buf, int size, xmlNodePtr node, int glob) {
5144 xmlNodePtr cur;
5145 int len;
5146
5147 if (node == NULL) return;
5148 if (glob) strcat(buf, "(");
5149 cur = node;
5150 while (cur != NULL) {
5151 len = strlen(buf);
5152 if (size - len < 50) {
5153 if ((size - len > 4) && (buf[len - 1] != '.'))
5154 strcat(buf, " ...");
5155 return;
5156 }
5157 switch (cur->type) {
5158 case XML_ELEMENT_NODE:
5159 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
5160 if (size - len < xmlStrlen(cur->ns->prefix) + 10) {
5161 if ((size - len > 4) && (buf[len - 1] != '.'))
5162 strcat(buf, " ...");
5163 return;
5164 }
5165 strcat(buf, (char *) cur->ns->prefix);
5166 strcat(buf, ":");
5167 }
5168 if (size - len < xmlStrlen(cur->name) + 10) {
5169 if ((size - len > 4) && (buf[len - 1] != '.'))
5170 strcat(buf, " ...");
5171 return;
5172 }
5173 strcat(buf, (char *) cur->name);
5174 if (cur->next != NULL)
5175 strcat(buf, " ");
5176 break;
5177 case XML_TEXT_NODE:
5178 if (xmlIsBlankNode(cur))
5179 break;
5180 case XML_CDATA_SECTION_NODE:
5181 case XML_ENTITY_REF_NODE:
5182 strcat(buf, "CDATA");
5183 if (cur->next != NULL)
5184 strcat(buf, " ");
5185 break;
5186 case XML_ATTRIBUTE_NODE:
5187 case XML_DOCUMENT_NODE:
5188 #ifdef LIBXML_DOCB_ENABLED
5189 case XML_DOCB_DOCUMENT_NODE:
5190 #endif
5191 case XML_HTML_DOCUMENT_NODE:
5192 case XML_DOCUMENT_TYPE_NODE:
5193 case XML_DOCUMENT_FRAG_NODE:
5194 case XML_NOTATION_NODE:
5195 case XML_NAMESPACE_DECL:
5196 strcat(buf, "???");
5197 if (cur->next != NULL)
5198 strcat(buf, " ");
5199 break;
5200 case XML_ENTITY_NODE:
5201 case XML_PI_NODE:
5202 case XML_DTD_NODE:
5203 case XML_COMMENT_NODE:
5204 case XML_ELEMENT_DECL:
5205 case XML_ATTRIBUTE_DECL:
5206 case XML_ENTITY_DECL:
5207 case XML_XINCLUDE_START:
5208 case XML_XINCLUDE_END:
5209 break;
5210 }
5211 cur = cur->next;
5212 }
5213 if (glob) strcat(buf, ")");
5214 }
5215
5216 /**
5217 * xmlValidateElementContent:
5218 * @ctxt: the validation context
5219 * @child: the child list
5220 * @elemDecl: pointer to the element declaration
5221 * @warn: emit the error message
5222 * @parent: the parent element (for error reporting)
5223 *
5224 * Try to validate the content model of an element
5225 *
5226 * returns 1 if valid or 0 if not and -1 in case of error
5227 */
5228
5229 static int
5230 xmlValidateElementContent(xmlValidCtxtPtr ctxt, xmlNodePtr child,
5231 xmlElementPtr elemDecl, int warn, xmlNodePtr parent) {
5232 int ret = 1;
5233 #ifndef LIBXML_REGEXP_ENABLED
5234 xmlNodePtr repl = NULL, last = NULL, tmp;
5235 #endif
5236 xmlNodePtr cur;
5237 xmlElementContentPtr cont;
5238 const xmlChar *name;
5239
5240 if ((elemDecl == NULL) || (parent == NULL) || (ctxt == NULL))
5241 return(-1);
5242 cont = elemDecl->content;
5243 name = elemDecl->name;
5244
5245 #ifdef LIBXML_REGEXP_ENABLED
5246 /* Build the regexp associated to the content model */
5247 if (elemDecl->contModel == NULL)
5248 ret = xmlValidBuildContentModel(ctxt, elemDecl);
5249 if (elemDecl->contModel == NULL) {
5250 return(-1);
5251 } else {
5252 xmlRegExecCtxtPtr exec;
5253
5254 if (!xmlRegexpIsDeterminist(elemDecl->contModel)) {
5255 return(-1);
5256 }
5257 ctxt->nodeMax = 0;
5258 ctxt->nodeNr = 0;
5259 ctxt->nodeTab = NULL;
5260 exec = xmlRegNewExecCtxt(elemDecl->contModel, NULL, NULL);
5261 if (exec != NULL) {
5262 cur = child;
5263 while (cur != NULL) {
5264 switch (cur->type) {
5265 case XML_ENTITY_REF_NODE:
5266 /*
5267 * Push the current node to be able to roll back
5268 * and process within the entity
5269 */
5270 if ((cur->children != NULL) &&
5271 (cur->children->children != NULL)) {
5272 nodeVPush(ctxt, cur);
5273 cur = cur->children->children;
5274 continue;
5275 }
5276 break;
5277 case XML_TEXT_NODE:
5278 if (xmlIsBlankNode(cur))
5279 break;
5280 ret = 0;
5281 goto fail;
5282 case XML_CDATA_SECTION_NODE:
5283 /* TODO */
5284 ret = 0;
5285 goto fail;
5286 case XML_ELEMENT_NODE:
5287 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
5288 xmlChar fn[50];
5289 xmlChar *fullname;
5290
5291 fullname = xmlBuildQName(cur->name,
5292 cur->ns->prefix, fn, 50);
5293 if (fullname == NULL) {
5294 ret = -1;
5295 goto fail;
5296 }
5297 ret = xmlRegExecPushString(exec, fullname, NULL);
5298 if ((fullname != fn) && (fullname != cur->name))
5299 xmlFree(fullname);
5300 } else {
5301 ret = xmlRegExecPushString(exec, cur->name, NULL);
5302 }
5303 break;
5304 default:
5305 break;
5306 }
5307 /*
5308 * Switch to next element
5309 */
5310 cur = cur->next;
5311 while (cur == NULL) {
5312 cur = nodeVPop(ctxt);
5313 if (cur == NULL)
5314 break;
5315 cur = cur->next;
5316 }
5317 }
5318 ret = xmlRegExecPushString(exec, NULL, NULL);
5319 fail:
5320 xmlRegFreeExecCtxt(exec);
5321 }
5322 }
5323 #else /* LIBXML_REGEXP_ENABLED */
5324 /*
5325 * Allocate the stack
5326 */
5327 ctxt->vstateMax = 8;
5328 ctxt->vstateTab = (xmlValidState *) xmlMalloc(
5329 ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
5330 if (ctxt->vstateTab == NULL) {
5331 xmlVErrMemory(ctxt, "malloc failed");
5332 return(-1);
5333 }
5334 /*
5335 * The first entry in the stack is reserved to the current state
5336 */
5337 ctxt->nodeMax = 0;
5338 ctxt->nodeNr = 0;
5339 ctxt->nodeTab = NULL;
5340 ctxt->vstate = &ctxt->vstateTab[0];
5341 ctxt->vstateNr = 1;
5342 CONT = cont;
5343 NODE = child;
5344 DEPTH = 0;
5345 OCCURS = 0;
5346 STATE = 0;
5347 ret = xmlValidateElementType(ctxt);
5348 if ((ret == -3) && (warn)) {
5349 xmlErrValidWarning(ctxt, child, XML_DTD_CONTENT_NOT_DETERMINIST,
5350 "Content model for Element %s is ambiguous\n",
5351 name, NULL, NULL);
5352 } else if (ret == -2) {
5353 /*
5354 * An entities reference appeared at this level.
5355 * Buid a minimal representation of this node content
5356 * sufficient to run the validation process on it
5357 */
5358 DEBUG_VALID_MSG("Found an entity reference, linearizing");
5359 cur = child;
5360 while (cur != NULL) {
5361 switch (cur->type) {
5362 case XML_ENTITY_REF_NODE:
5363 /*
5364 * Push the current node to be able to roll back
5365 * and process within the entity
5366 */
5367 if ((cur->children != NULL) &&
5368 (cur->children->children != NULL)) {
5369 nodeVPush(ctxt, cur);
5370 cur = cur->children->children;
5371 continue;
5372 }
5373 break;
5374 case XML_TEXT_NODE:
5375 if (xmlIsBlankNode(cur))
5376 break;
5377 /* no break on purpose */
5378 case XML_CDATA_SECTION_NODE:
5379 /* no break on purpose */
5380 case XML_ELEMENT_NODE:
5381 /*
5382 * Allocate a new node and minimally fills in
5383 * what's required
5384 */
5385 tmp = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
5386 if (tmp == NULL) {
5387 xmlVErrMemory(ctxt, "malloc failed");
5388 xmlFreeNodeList(repl);
5389 ret = -1;
5390 goto done;
5391 }
5392 tmp->type = cur->type;
5393 tmp->name = cur->name;
5394 tmp->ns = cur->ns;
5395 tmp->next = NULL;
5396 tmp->content = NULL;
5397 if (repl == NULL)
5398 repl = last = tmp;
5399 else {
5400 last->next = tmp;
5401 last = tmp;
5402 }
5403 if (cur->type == XML_CDATA_SECTION_NODE) {
5404 /*
5405 * E59 spaces in CDATA does not match the
5406 * nonterminal S
5407 */
5408 tmp->content = xmlStrdup(BAD_CAST "CDATA");
5409 }
5410 break;
5411 default:
5412 break;
5413 }
5414 /*
5415 * Switch to next element
5416 */
5417 cur = cur->next;
5418 while (cur == NULL) {
5419 cur = nodeVPop(ctxt);
5420 if (cur == NULL)
5421 break;
5422 cur = cur->next;
5423 }
5424 }
5425
5426 /*
5427 * Relaunch the validation
5428 */
5429 ctxt->vstate = &ctxt->vstateTab[0];
5430 ctxt->vstateNr = 1;
5431 CONT = cont;
5432 NODE = repl;
5433 DEPTH = 0;
5434 OCCURS = 0;
5435 STATE = 0;
5436 ret = xmlValidateElementType(ctxt);
5437 }
5438 #endif /* LIBXML_REGEXP_ENABLED */
5439 if ((warn) && ((ret != 1) && (ret != -3))) {
5440 if (ctxt != NULL) {
5441 char expr[5000];
5442 char list[5000];
5443
5444 expr[0] = 0;
5445 xmlSnprintfElementContent(&expr[0], 5000, cont, 1);
5446 list[0] = 0;
5447 #ifndef LIBXML_REGEXP_ENABLED
5448 if (repl != NULL)
5449 xmlSnprintfElements(&list[0], 5000, repl, 1);
5450 else
5451 #endif /* LIBXML_REGEXP_ENABLED */
5452 xmlSnprintfElements(&list[0], 5000, child, 1);
5453
5454 if (name != NULL) {
5455 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5456 "Element %s content does not follow the DTD, expecting %s, got %s\n",
5457 name, BAD_CAST expr, BAD_CAST list);
5458 } else {
5459 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5460 "Element content does not follow the DTD, expecting %s, got %s\n",
5461 BAD_CAST expr, BAD_CAST list, NULL);
5462 }
5463 } else {
5464 if (name != NULL) {
5465 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5466 "Element %s content does not follow the DTD\n",
5467 name, NULL, NULL);
5468 } else {
5469 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5470 "Element content does not follow the DTD\n",
5471 NULL, NULL, NULL);
5472 }
5473 }
5474 ret = 0;
5475 }
5476 if (ret == -3)
5477 ret = 1;
5478
5479 #ifndef LIBXML_REGEXP_ENABLED
5480 done:
5481 /*
5482 * Deallocate the copy if done, and free up the validation stack
5483 */
5484 while (repl != NULL) {
5485 tmp = repl->next;
5486 xmlFree(repl);
5487 repl = tmp;
5488 }
5489 ctxt->vstateMax = 0;
5490 if (ctxt->vstateTab != NULL) {
5491 xmlFree(ctxt->vstateTab);
5492 ctxt->vstateTab = NULL;
5493 }
5494 #endif
5495 ctxt->nodeMax = 0;
5496 ctxt->nodeNr = 0;
5497 if (ctxt->nodeTab != NULL) {
5498 xmlFree(ctxt->nodeTab);
5499 ctxt->nodeTab = NULL;
5500 }
5501 return(ret);
5502
5503 }
5504
5505 /**
5506 * xmlValidateCdataElement:
5507 * @ctxt: the validation context
5508 * @doc: a document instance
5509 * @elem: an element instance
5510 *
5511 * Check that an element follows #CDATA
5512 *
5513 * returns 1 if valid or 0 otherwise
5514 */
5515 static int
5516 xmlValidateOneCdataElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5517 xmlNodePtr elem) {
5518 int ret = 1;
5519 xmlNodePtr cur, child;
5520
5521 if ((ctxt == NULL) || (doc == NULL) || (elem == NULL) ||
5522 (elem->type != XML_ELEMENT_NODE))
5523 return(0);
5524
5525 child = elem->children;
5526
5527 cur = child;
5528 while (cur != NULL) {
5529 switch (cur->type) {
5530 case XML_ENTITY_REF_NODE:
5531 /*
5532 * Push the current node to be able to roll back
5533 * and process within the entity
5534 */
5535 if ((cur->children != NULL) &&
5536 (cur->children->children != NULL)) {
5537 nodeVPush(ctxt, cur);
5538 cur = cur->children->children;
5539 continue;
5540 }
5541 break;
5542 case XML_COMMENT_NODE:
5543 case XML_PI_NODE:
5544 case XML_TEXT_NODE:
5545 case XML_CDATA_SECTION_NODE:
5546 break;
5547 default:
5548 ret = 0;
5549 goto done;
5550 }
5551 /*
5552 * Switch to next element
5553 */
5554 cur = cur->next;
5555 while (cur == NULL) {
5556 cur = nodeVPop(ctxt);
5557 if (cur == NULL)
5558 break;
5559 cur = cur->next;
5560 }
5561 }
5562 done:
5563 ctxt->nodeMax = 0;
5564 ctxt->nodeNr = 0;
5565 if (ctxt->nodeTab != NULL) {
5566 xmlFree(ctxt->nodeTab);
5567 ctxt->nodeTab = NULL;
5568 }
5569 return(ret);
5570 }
5571
5572 /**
5573 * xmlValidateCheckMixed:
5574 * @ctxt: the validation context
5575 * @cont: the mixed content model
5576 * @qname: the qualified name as appearing in the serialization
5577 *
5578 * Check if the given node is part of the content model.
5579 *
5580 * Returns 1 if yes, 0 if no, -1 in case of error
5581 */
5582 static int
5583 xmlValidateCheckMixed(xmlValidCtxtPtr ctxt,
5584 xmlElementContentPtr cont, const xmlChar *qname) {
5585 const xmlChar *name;
5586 int plen;
5587 name = xmlSplitQName3(qname, &plen);
5588
5589 if (name == NULL) {
5590 while (cont != NULL) {
5591 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
5592 if ((cont->prefix == NULL) && (xmlStrEqual(cont->name, qname)))
5593 return(1);
5594 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
5595 (cont->c1 != NULL) &&
5596 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
5597 if ((cont->c1->prefix == NULL) &&
5598 (xmlStrEqual(cont->c1->name, qname)))
5599 return(1);
5600 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
5601 (cont->c1 == NULL) ||
5602 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
5603 xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
5604 "Internal: MIXED struct corrupted\n",
5605 NULL);
5606 break;
5607 }
5608 cont = cont->c2;
5609 }
5610 } else {
5611 while (cont != NULL) {
5612 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
5613 if ((cont->prefix != NULL) &&
5614 (xmlStrncmp(cont->prefix, qname, plen) == 0) &&
5615 (xmlStrEqual(cont->name, name)))
5616 return(1);
5617 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
5618 (cont->c1 != NULL) &&
5619 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
5620 if ((cont->c1->prefix != NULL) &&
5621 (xmlStrncmp(cont->c1->prefix, qname, plen) == 0) &&
5622 (xmlStrEqual(cont->c1->name, name)))
5623 return(1);
5624 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
5625 (cont->c1 == NULL) ||
5626 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
5627 xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
5628 "Internal: MIXED struct corrupted\n",
5629 NULL);
5630 break;
5631 }
5632 cont = cont->c2;
5633 }
5634 }
5635 return(0);
5636 }
5637
5638 /**
5639 * xmlValidGetElemDecl:
5640 * @ctxt: the validation context
5641 * @doc: a document instance
5642 * @elem: an element instance
5643 * @extsubset: pointer, (out) indicate if the declaration was found
5644 * in the external subset.
5645 *
5646 * Finds a declaration associated to an element in the document.
5647 *
5648 * returns the pointer to the declaration or NULL if not found.
5649 */
5650 static xmlElementPtr
5651 xmlValidGetElemDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5652 xmlNodePtr elem, int *extsubset) {
5653 xmlElementPtr elemDecl = NULL;
5654 const xmlChar *prefix = NULL;
5655
5656 if ((ctxt == NULL) || (doc == NULL) ||
5657 (elem == NULL) || (elem->name == NULL))
5658 return(NULL);
5659 if (extsubset != NULL)
5660 *extsubset = 0;
5661
5662 /*
5663 * Fetch the declaration for the qualified name
5664 */
5665 if ((elem->ns != NULL) && (elem->ns->prefix != NULL))
5666 prefix = elem->ns->prefix;
5667
5668 if (prefix != NULL) {
5669 elemDecl = xmlGetDtdQElementDesc(doc->intSubset,
5670 elem->name, prefix);
5671 if ((elemDecl == NULL) && (doc->extSubset != NULL)) {
5672 elemDecl = xmlGetDtdQElementDesc(doc->extSubset,
5673 elem->name, prefix);
5674 if ((elemDecl != NULL) && (extsubset != NULL))
5675 *extsubset = 1;
5676 }
5677 }
5678
5679 /*
5680 * Fetch the declaration for the non qualified name
5681 * This is "non-strict" validation should be done on the
5682 * full QName but in that case being flexible makes sense.
5683 */
5684 if (elemDecl == NULL) {
5685 elemDecl = xmlGetDtdElementDesc(doc->intSubset, elem->name);
5686 if ((elemDecl == NULL) && (doc->extSubset != NULL)) {
5687 elemDecl = xmlGetDtdElementDesc(doc->extSubset, elem->name);
5688 if ((elemDecl != NULL) && (extsubset != NULL))
5689 *extsubset = 1;
5690 }
5691 }
5692 if (elemDecl == NULL) {
5693 xmlErrValidNode(ctxt, elem,
5694 XML_DTD_UNKNOWN_ELEM,
5695 "No declaration for element %s\n",
5696 elem->name, NULL, NULL);
5697 }
5698 return(elemDecl);
5699 }
5700
5701 #ifdef LIBXML_REGEXP_ENABLED
5702 /**
5703 * xmlValidatePushElement:
5704 * @ctxt: the validation context
5705 * @doc: a document instance
5706 * @elem: an element instance
5707 * @qname: the qualified name as appearing in the serialization
5708 *
5709 * Push a new element start on the validation stack.
5710 *
5711 * returns 1 if no validation problem was found or 0 otherwise
5712 */
5713 int
5714 xmlValidatePushElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5715 xmlNodePtr elem, const xmlChar *qname) {
5716 int ret = 1;
5717 xmlElementPtr eDecl;
5718 int extsubset = 0;
5719
5720 if (ctxt == NULL)
5721 return(0);
5722 /* printf("PushElem %s\n", qname); */
5723 if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5724 xmlValidStatePtr state = ctxt->vstate;
5725 xmlElementPtr elemDecl;
5726
5727 /*
5728 * Check the new element agaisnt the content model of the new elem.
5729 */
5730 if (state->elemDecl != NULL) {
5731 elemDecl = state->elemDecl;
5732
5733 switch(elemDecl->etype) {
5734 case XML_ELEMENT_TYPE_UNDEFINED:
5735 ret = 0;
5736 break;
5737 case XML_ELEMENT_TYPE_EMPTY:
5738 xmlErrValidNode(ctxt, state->node,
5739 XML_DTD_NOT_EMPTY,
5740 "Element %s was declared EMPTY this one has content\n",
5741 state->node->name, NULL, NULL);
5742 ret = 0;
5743 break;
5744 case XML_ELEMENT_TYPE_ANY:
5745 /* I don't think anything is required then */
5746 break;
5747 case XML_ELEMENT_TYPE_MIXED:
5748 /* simple case of declared as #PCDATA */
5749 if ((elemDecl->content != NULL) &&
5750 (elemDecl->content->type ==
5751 XML_ELEMENT_CONTENT_PCDATA)) {
5752 xmlErrValidNode(ctxt, state->node,
5753 XML_DTD_NOT_PCDATA,
5754 "Element %s was declared #PCDATA but contains non text nodes\n",
5755 state->node->name, NULL, NULL);
5756 ret = 0;
5757 } else {
5758 ret = xmlValidateCheckMixed(ctxt, elemDecl->content,
5759 qname);
5760 if (ret != 1) {
5761 xmlErrValidNode(ctxt, state->node,
5762 XML_DTD_INVALID_CHILD,
5763 "Element %s is not declared in %s list of possible children\n",
5764 qname, state->node->name, NULL);
5765 }
5766 }
5767 break;
5768 case XML_ELEMENT_TYPE_ELEMENT:
5769 /*
5770 * TODO:
5771 * VC: Standalone Document Declaration
5772 * - element types with element content, if white space
5773 * occurs directly within any instance of those types.
5774 */
5775 if (state->exec != NULL) {
5776 ret = xmlRegExecPushString(state->exec, qname, NULL);
5777 if (ret < 0) {
5778 xmlErrValidNode(ctxt, state->node,
5779 XML_DTD_CONTENT_MODEL,
5780 "Element %s content does not follow the DTD, Misplaced %s\n",
5781 state->node->name, qname, NULL);
5782 ret = 0;
5783 } else {
5784 ret = 1;
5785 }
5786 }
5787 break;
5788 }
5789 }
5790 }
5791 eDecl = xmlValidGetElemDecl(ctxt, doc, elem, &extsubset);
5792 vstateVPush(ctxt, eDecl, elem);
5793 return(ret);
5794 }
5795
5796 /**
5797 * xmlValidatePushCData:
5798 * @ctxt: the validation context
5799 * @data: some character data read
5800 * @len: the length of the data
5801 *
5802 * check the CData parsed for validation in the current stack
5803 *
5804 * returns 1 if no validation problem was found or 0 otherwise
5805 */
5806 int
5807 xmlValidatePushCData(xmlValidCtxtPtr ctxt, const xmlChar *data, int len) {
5808 int ret = 1;
5809
5810 /* printf("CDATA %s %d\n", data, len); */
5811 if (ctxt == NULL)
5812 return(0);
5813 if (len <= 0)
5814 return(ret);
5815 if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5816 xmlValidStatePtr state = ctxt->vstate;
5817 xmlElementPtr elemDecl;
5818
5819 /*
5820 * Check the new element agaisnt the content model of the new elem.
5821 */
5822 if (state->elemDecl != NULL) {
5823 elemDecl = state->elemDecl;
5824
5825 switch(elemDecl->etype) {
5826 case XML_ELEMENT_TYPE_UNDEFINED:
5827 ret = 0;
5828 break;
5829 case XML_ELEMENT_TYPE_EMPTY:
5830 xmlErrValidNode(ctxt, state->node,
5831 XML_DTD_NOT_EMPTY,
5832 "Element %s was declared EMPTY this one has content\n",
5833 state->node->name, NULL, NULL);
5834 ret = 0;
5835 break;
5836 case XML_ELEMENT_TYPE_ANY:
5837 break;
5838 case XML_ELEMENT_TYPE_MIXED:
5839 break;
5840 case XML_ELEMENT_TYPE_ELEMENT:
5841 if (len > 0) {
5842 int i;
5843
5844 for (i = 0;i < len;i++) {
5845 if (!IS_BLANK_CH(data[i])) {
5846 xmlErrValidNode(ctxt, state->node,
5847 XML_DTD_CONTENT_MODEL,
5848 "Element %s content does not follow the DTD, Text not allowed\n",
5849 state->node->name, NULL, NULL);
5850 ret = 0;
5851 goto done;
5852 }
5853 }
5854 /*
5855 * TODO:
5856 * VC: Standalone Document Declaration
5857 * element types with element content, if white space
5858 * occurs directly within any instance of those types.
5859 */
5860 }
5861 break;
5862 }
5863 }
5864 }
5865 done:
5866 return(ret);
5867 }
5868
5869 /**
5870 * xmlValidatePopElement:
5871 * @ctxt: the validation context
5872 * @doc: a document instance
5873 * @elem: an element instance
5874 * @qname: the qualified name as appearing in the serialization
5875 *
5876 * Pop the element end from the validation stack.
5877 *
5878 * returns 1 if no validation problem was found or 0 otherwise
5879 */
5880 int
5881 xmlValidatePopElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc ATTRIBUTE_UNUSED,
5882 xmlNodePtr elem ATTRIBUTE_UNUSED,
5883 const xmlChar *qname ATTRIBUTE_UNUSED) {
5884 int ret = 1;
5885
5886 if (ctxt == NULL)
5887 return(0);
5888 /* printf("PopElem %s\n", qname); */
5889 if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5890 xmlValidStatePtr state = ctxt->vstate;
5891 xmlElementPtr elemDecl;
5892
5893 /*
5894 * Check the new element agaisnt the content model of the new elem.
5895 */
5896 if (state->elemDecl != NULL) {
5897 elemDecl = state->elemDecl;
5898
5899 if (elemDecl->etype == XML_ELEMENT_TYPE_ELEMENT) {
5900 if (state->exec != NULL) {
5901 ret = xmlRegExecPushString(state->exec, NULL, NULL);
5902 if (ret == 0) {
5903 xmlErrValidNode(ctxt, state->node,
5904 XML_DTD_CONTENT_MODEL,
5905 "Element %s content does not follow the DTD, Expecting more child\n",
5906 state->node->name, NULL,NULL);
5907 } else {
5908 /*
5909 * previous validation errors should not generate
5910 * a new one here
5911 */
5912 ret = 1;
5913 }
5914 }
5915 }
5916 }
5917 vstateVPop(ctxt);
5918 }
5919 return(ret);
5920 }
5921 #endif /* LIBXML_REGEXP_ENABLED */
5922
5923 /**
5924 * xmlValidateOneElement:
5925 * @ctxt: the validation context
5926 * @doc: a document instance
5927 * @elem: an element instance
5928 *
5929 * Try to validate a single element and it's attributes,
5930 * basically it does the following checks as described by the
5931 * XML-1.0 recommendation:
5932 * - [ VC: Element Valid ]
5933 * - [ VC: Required Attribute ]
5934 * Then call xmlValidateOneAttribute() for each attribute present.
5935 *
5936 * The ID/IDREF checkings are done separately
5937 *
5938 * returns 1 if valid or 0 otherwise
5939 */
5940
5941 int
5942 xmlValidateOneElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5943 xmlNodePtr elem) {
5944 xmlElementPtr elemDecl = NULL;
5945 xmlElementContentPtr cont;
5946 xmlAttributePtr attr;
5947 xmlNodePtr child;
5948 int ret = 1, tmp;
5949 const xmlChar *name;
5950 int extsubset = 0;
5951
5952 CHECK_DTD;
5953
5954 if (elem == NULL) return(0);
5955 switch (elem->type) {
5956 case XML_ATTRIBUTE_NODE:
5957 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5958 "Attribute element not expected\n", NULL, NULL ,NULL);
5959 return(0);
5960 case XML_TEXT_NODE:
5961 if (elem->children != NULL) {
5962 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5963 "Text element has children !\n",
5964 NULL,NULL,NULL);
5965 return(0);
5966 }
5967 if (elem->ns != NULL) {
5968 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5969 "Text element has namespace !\n",
5970 NULL,NULL,NULL);
5971 return(0);
5972 }
5973 if (elem->content == NULL) {
5974 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5975 "Text element has no content !\n",
5976 NULL,NULL,NULL);
5977 return(0);
5978 }
5979 return(1);
5980 case XML_XINCLUDE_START:
5981 case XML_XINCLUDE_END:
5982 return(1);
5983 case XML_CDATA_SECTION_NODE:
5984 case XML_ENTITY_REF_NODE:
5985 case XML_PI_NODE:
5986 case XML_COMMENT_NODE:
5987 return(1);
5988 case XML_ENTITY_NODE:
5989 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5990 "Entity element not expected\n", NULL, NULL ,NULL);
5991 return(0);
5992 case XML_NOTATION_NODE:
5993 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5994 "Notation element not expected\n", NULL, NULL ,NULL);
5995 return(0);
5996 case XML_DOCUMENT_NODE:
5997 case XML_DOCUMENT_TYPE_NODE:
5998 case XML_DOCUMENT_FRAG_NODE:
5999 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6000 "Document element not expected\n", NULL, NULL ,NULL);
6001 return(0);
6002 case XML_HTML_DOCUMENT_NODE:
6003 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6004 "HTML Document not expected\n", NULL, NULL ,NULL);
6005 return(0);
6006 case XML_ELEMENT_NODE:
6007 break;
6008 default:
6009 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6010 "unknown element type\n", NULL, NULL ,NULL);
6011 return(0);
6012 }
6013
6014 /*
6015 * Fetch the declaration
6016 */
6017 elemDecl = xmlValidGetElemDecl(ctxt, doc, elem, &extsubset);
6018 if (elemDecl == NULL)
6019 return(0);
6020
6021 /*
6022 * If vstateNr is not zero that means continuous validation is
6023 * activated, do not try to check the content model at that level.
6024 */
6025 if (ctxt->vstateNr == 0) {
6026 /* Check that the element content matches the definition */
6027 switch (elemDecl->etype) {
6028 case XML_ELEMENT_TYPE_UNDEFINED:
6029 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ELEM,
6030 "No declaration for element %s\n",
6031 elem->name, NULL, NULL);
6032 return(0);
6033 case XML_ELEMENT_TYPE_EMPTY:
6034 if (elem->children != NULL) {
6035 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_EMPTY,
6036 "Element %s was declared EMPTY this one has content\n",
6037 elem->name, NULL, NULL);
6038 ret = 0;
6039 }
6040 break;
6041 case XML_ELEMENT_TYPE_ANY:
6042 /* I don't think anything is required then */
6043 break;
6044 case XML_ELEMENT_TYPE_MIXED:
6045
6046 /* simple case of declared as #PCDATA */
6047 if ((elemDecl->content != NULL) &&
6048 (elemDecl->content->type == XML_ELEMENT_CONTENT_PCDATA)) {
6049 ret = xmlValidateOneCdataElement(ctxt, doc, elem);
6050 if (!ret) {
6051 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_PCDATA,
6052 "Element %s was declared #PCDATA but contains non text nodes\n",
6053 elem->name, NULL, NULL);
6054 }
6055 break;
6056 }
6057 child = elem->children;
6058 /* Hum, this start to get messy */
6059 while (child != NULL) {
6060 if (child->type == XML_ELEMENT_NODE) {
6061 name = child->name;
6062 if ((child->ns != NULL) && (child->ns->prefix != NULL)) {
6063 xmlChar fn[50];
6064 xmlChar *fullname;
6065
6066 fullname = xmlBuildQName(child->name, child->ns->prefix,
6067 fn, 50);
6068 if (fullname == NULL)
6069 return(0);
6070 cont = elemDecl->content;
6071 while (cont != NULL) {
6072 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
6073 if (xmlStrEqual(cont->name, fullname))
6074 break;
6075 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
6076 (cont->c1 != NULL) &&
6077 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
6078 if (xmlStrEqual(cont->c1->name, fullname))
6079 break;
6080 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
6081 (cont->c1 == NULL) ||
6082 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
6083 xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
6084 "Internal: MIXED struct corrupted\n",
6085 NULL);
6086 break;
6087 }
6088 cont = cont->c2;
6089 }
6090 if ((fullname != fn) && (fullname != child->name))
6091 xmlFree(fullname);
6092 if (cont != NULL)
6093 goto child_ok;
6094 }
6095 cont = elemDecl->content;
6096 while (cont != NULL) {
6097 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
6098 if (xmlStrEqual(cont->name, name)) break;
6099 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
6100 (cont->c1 != NULL) &&
6101 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)) {
6102 if (xmlStrEqual(cont->c1->name, name)) break;
6103 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
6104 (cont->c1 == NULL) ||
6105 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)) {
6106 xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
6107 "Internal: MIXED struct corrupted\n",
6108 NULL);
6109 break;
6110 }
6111 cont = cont->c2;
6112 }
6113 if (cont == NULL) {
6114 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_CHILD,
6115 "Element %s is not declared in %s list of possible children\n",
6116 name, elem->name, NULL);
6117 ret = 0;
6118 }
6119 }
6120 child_ok:
6121 child = child->next;
6122 }
6123 break;
6124 case XML_ELEMENT_TYPE_ELEMENT:
6125 if ((doc->standalone == 1) && (extsubset == 1)) {
6126 /*
6127 * VC: Standalone Document Declaration
6128 * - element types with element content, if white space
6129 * occurs directly within any instance of those types.
6130 */
6131 child = elem->children;
6132 while (child != NULL) {
6133 if (child->type == XML_TEXT_NODE) {
6134 const xmlChar *content = child->content;
6135
6136 while (IS_BLANK_CH(*content))
6137 content++;
6138 if (*content == 0) {
6139 xmlErrValidNode(ctxt, elem,
6140 XML_DTD_STANDALONE_WHITE_SPACE,
6141 "standalone: %s declared in the external subset contains white spaces nodes\n",
6142 elem->name, NULL, NULL);
6143 ret = 0;
6144 break;
6145 }
6146 }
6147 child =child->next;
6148 }
6149 }
6150 child = elem->children;
6151 cont = elemDecl->content;
6152 tmp = xmlValidateElementContent(ctxt, child, elemDecl, 1, elem);
6153 if (tmp <= 0)
6154 ret = tmp;
6155 break;
6156 }
6157 } /* not continuous */
6158
6159 /* [ VC: Required Attribute ] */
6160 attr = elemDecl->attributes;
6161 while (attr != NULL) {
6162 if (attr->def == XML_ATTRIBUTE_REQUIRED) {
6163 int qualified = -1;
6164
6165 if ((attr->prefix == NULL) &&
6166 (xmlStrEqual(attr->name, BAD_CAST "xmlns"))) {
6167 xmlNsPtr ns;
6168
6169 ns = elem->nsDef;
6170 while (ns != NULL) {
6171 if (ns->prefix == NULL)
6172 goto found;
6173 ns = ns->next;
6174 }
6175 } else if (xmlStrEqual(attr->prefix, BAD_CAST "xmlns")) {
6176 xmlNsPtr ns;
6177
6178 ns = elem->nsDef;
6179 while (ns != NULL) {
6180 if (xmlStrEqual(attr->name, ns->prefix))
6181 goto found;
6182 ns = ns->next;
6183 }
6184 } else {
6185 xmlAttrPtr attrib;
6186
6187 attrib = elem->properties;
6188 while (attrib != NULL) {
6189 if (xmlStrEqual(attrib->name, attr->name)) {
6190 if (attr->prefix != NULL) {
6191 xmlNsPtr nameSpace = attrib->ns;
6192
6193 if (nameSpace == NULL)
6194 nameSpace = elem->ns;
6195 /*
6196 * qualified names handling is problematic, having a
6197 * different prefix should be possible but DTDs don't
6198 * allow to define the URI instead of the prefix :-(
6199 */
6200 if (nameSpace == NULL) {
6201 if (qualified < 0)
6202 qualified = 0;
6203 } else if (!xmlStrEqual(nameSpace->prefix,
6204 attr->prefix)) {
6205 if (qualified < 1)
6206 qualified = 1;
6207 } else
6208 goto found;
6209 } else {
6210 /*
6211 * We should allow applications to define namespaces
6212 * for their application even if the DTD doesn't
6213 * carry one, otherwise, basically we would always
6214 * break.
6215 */
6216 goto found;
6217 }
6218 }
6219 attrib = attrib->next;
6220 }
6221 }
6222 if (qualified == -1) {
6223 if (attr->prefix == NULL) {
6224 xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
6225 "Element %s does not carry attribute %s\n",
6226 elem->name, attr->name, NULL);
6227 ret = 0;
6228 } else {
6229 xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
6230 "Element %s does not carry attribute %s:%s\n",
6231 elem->name, attr->prefix,attr->name);
6232 ret = 0;
6233 }
6234 } else if (qualified == 0) {
6235 xmlErrValidWarning(ctxt, elem, XML_DTD_NO_PREFIX,
6236 "Element %s required attribute %s:%s has no prefix\n",
6237 elem->name, attr->prefix, attr->name);
6238 } else if (qualified == 1) {
6239 xmlErrValidWarning(ctxt, elem, XML_DTD_DIFFERENT_PREFIX,
6240 "Element %s required attribute %s:%s has different prefix\n",
6241 elem->name, attr->prefix, attr->name);
6242 }
6243 } else if (attr->def == XML_ATTRIBUTE_FIXED) {
6244 /*
6245 * Special tests checking #FIXED namespace declarations
6246 * have the right value since this is not done as an
6247 * attribute checking
6248 */
6249 if ((attr->prefix == NULL) &&
6250 (xmlStrEqual(attr->name, BAD_CAST "xmlns"))) {
6251 xmlNsPtr ns;
6252
6253 ns = elem->nsDef;
6254 while (ns != NULL) {
6255 if (ns->prefix == NULL) {
6256 if (!xmlStrEqual(attr->defaultValue, ns->href)) {
6257 xmlErrValidNode(ctxt, elem,
6258 XML_DTD_ELEM_DEFAULT_NAMESPACE,
6259 "Element %s namespace name for default namespace does not match the DTD\n",
6260 elem->name, NULL, NULL);
6261 ret = 0;
6262 }
6263 goto found;
6264 }
6265 ns = ns->next;
6266 }
6267 } else if (xmlStrEqual(attr->prefix, BAD_CAST "xmlns")) {
6268 xmlNsPtr ns;
6269
6270 ns = elem->nsDef;
6271 while (ns != NULL) {
6272 if (xmlStrEqual(attr->name, ns->prefix)) {
6273 if (!xmlStrEqual(attr->defaultValue, ns->href)) {
6274 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
6275 "Element %s namespace name for %s does not match the DTD\n",
6276 elem->name, ns->prefix, NULL);
6277 ret = 0;
6278 }
6279 goto found;
6280 }
6281 ns = ns->next;
6282 }
6283 }
6284 }
6285 found:
6286 attr = attr->nexth;
6287 }
6288 return(ret);
6289 }
6290
6291 /**
6292 * xmlValidateRoot:
6293 * @ctxt: the validation context
6294 * @doc: a document instance
6295 *
6296 * Try to validate a the root element
6297 * basically it does the following check as described by the
6298 * XML-1.0 recommendation:
6299 * - [ VC: Root Element Type ]
6300 * it doesn't try to recurse or apply other check to the element
6301 *
6302 * returns 1 if valid or 0 otherwise
6303 */
6304
6305 int
6306 xmlValidateRoot(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6307 xmlNodePtr root;
6308 int ret;
6309
6310 if (doc == NULL) return(0);
6311
6312 root = xmlDocGetRootElement(doc);
6313 if ((root == NULL) || (root->name == NULL)) {
6314 xmlErrValid(ctxt, XML_DTD_NO_ROOT,
6315 "no root element\n", NULL);
6316 return(0);
6317 }
6318
6319 /*
6320 * When doing post validation against a separate DTD, those may
6321 * no internal subset has been generated
6322 */
6323 if ((doc->intSubset != NULL) &&
6324 (doc->intSubset->name != NULL)) {
6325 /*
6326 * Check first the document root against the NQName
6327 */
6328 if (!xmlStrEqual(doc->intSubset->name, root->name)) {
6329 if ((root->ns != NULL) && (root->ns->prefix != NULL)) {
6330 xmlChar fn[50];
6331 xmlChar *fullname;
6332
6333 fullname = xmlBuildQName(root->name, root->ns->prefix, fn, 50);
6334 if (fullname == NULL) {
6335 xmlVErrMemory(ctxt, NULL);
6336 return(0);
6337 }
6338 ret = xmlStrEqual(doc->intSubset->name, fullname);
6339 if ((fullname != fn) && (fullname != root->name))
6340 xmlFree(fullname);
6341 if (ret == 1)
6342 goto name_ok;
6343 }
6344 if ((xmlStrEqual(doc->intSubset->name, BAD_CAST "HTML")) &&
6345 (xmlStrEqual(root->name, BAD_CAST "html")))
6346 goto name_ok;
6347 xmlErrValidNode(ctxt, root, XML_DTD_ROOT_NAME,
6348 "root and DTD name do not match '%s' and '%s'\n",
6349 root->name, doc->intSubset->name, NULL);
6350 return(0);
6351 }
6352 }
6353 name_ok:
6354 return(1);
6355 }
6356
6357
6358 /**
6359 * xmlValidateElement:
6360 * @ctxt: the validation context
6361 * @doc: a document instance
6362 * @elem: an element instance
6363 *
6364 * Try to validate the subtree under an element
6365 *
6366 * returns 1 if valid or 0 otherwise
6367 */
6368
6369 int
6370 xmlValidateElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlNodePtr elem) {
6371 xmlNodePtr child;
6372 xmlAttrPtr attr;
6373 xmlNsPtr ns;
6374 const xmlChar *value;
6375 int ret = 1;
6376
6377 if (elem == NULL) return(0);
6378
6379 /*
6380 * XInclude elements were added after parsing in the infoset,
6381 * they don't really mean anything validation wise.
6382 */
6383 if ((elem->type == XML_XINCLUDE_START) ||
6384 (elem->type == XML_XINCLUDE_END) ||
6385 (elem->type == XML_NAMESPACE_DECL))
6386 return(1);
6387
6388 CHECK_DTD;
6389
6390 /*
6391 * Entities references have to be handled separately
6392 */
6393 if (elem->type == XML_ENTITY_REF_NODE) {
6394 return(1);
6395 }
6396
6397 ret &= xmlValidateOneElement(ctxt, doc, elem);
6398 if (elem->type == XML_ELEMENT_NODE) {
6399 attr = elem->properties;
6400 while (attr != NULL) {
6401 value = xmlNodeListGetString(doc, attr->children, 0);
6402 ret &= xmlValidateOneAttribute(ctxt, doc, elem, attr, value);
6403 if (value != NULL)
6404 xmlFree((char *)value);
6405 attr= attr->next;
6406 }
6407 ns = elem->nsDef;
6408 while (ns != NULL) {
6409 if (elem->ns == NULL)
6410 ret &= xmlValidateOneNamespace(ctxt, doc, elem, NULL,
6411 ns, ns->href);
6412 else
6413 ret &= xmlValidateOneNamespace(ctxt, doc, elem,
6414 elem->ns->prefix, ns, ns->href);
6415 ns = ns->next;
6416 }
6417 }
6418 child = elem->children;
6419 while (child != NULL) {
6420 ret &= xmlValidateElement(ctxt, doc, child);
6421 child = child->next;
6422 }
6423
6424 return(ret);
6425 }
6426
6427 /**
6428 * xmlValidateRef:
6429 * @ref: A reference to be validated
6430 * @ctxt: Validation context
6431 * @name: Name of ID we are searching for
6432 *
6433 */
6434 static void
6435 xmlValidateRef(xmlRefPtr ref, xmlValidCtxtPtr ctxt,
6436 const xmlChar *name) {
6437 xmlAttrPtr id;
6438 xmlAttrPtr attr;
6439
6440 if (ref == NULL)
6441 return;
6442 if ((ref->attr == NULL) && (ref->name == NULL))
6443 return;
6444 attr = ref->attr;
6445 if (attr == NULL) {
6446 xmlChar *dup, *str = NULL, *cur, save;
6447
6448 dup = xmlStrdup(name);
6449 if (dup == NULL) {
6450 ctxt->valid = 0;
6451 return;
6452 }
6453 cur = dup;
6454 while (*cur != 0) {
6455 str = cur;
6456 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
6457 save = *cur;
6458 *cur = 0;
6459 id = xmlGetID(ctxt->doc, str);
6460 if (id == NULL) {
6461 xmlErrValidNodeNr(ctxt, NULL, XML_DTD_UNKNOWN_ID,
6462 "attribute %s line %d references an unknown ID \"%s\"\n",
6463 ref->name, ref->lineno, str);
6464 ctxt->valid = 0;
6465 }
6466 if (save == 0)
6467 break;
6468 *cur = save;
6469 while (IS_BLANK_CH(*cur)) cur++;
6470 }
6471 xmlFree(dup);
6472 } else if (attr->atype == XML_ATTRIBUTE_IDREF) {
6473 id = xmlGetID(ctxt->doc, name);
6474 if (id == NULL) {
6475 xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
6476 "IDREF attribute %s references an unknown ID \"%s\"\n",
6477 attr->name, name, NULL);
6478 ctxt->valid = 0;
6479 }
6480 } else if (attr->atype == XML_ATTRIBUTE_IDREFS) {
6481 xmlChar *dup, *str = NULL, *cur, save;
6482
6483 dup = xmlStrdup(name);
6484 if (dup == NULL) {
6485 xmlVErrMemory(ctxt, "IDREFS split");
6486 ctxt->valid = 0;
6487 return;
6488 }
6489 cur = dup;
6490 while (*cur != 0) {
6491 str = cur;
6492 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
6493 save = *cur;
6494 *cur = 0;
6495 id = xmlGetID(ctxt->doc, str);
6496 if (id == NULL) {
6497 xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
6498 "IDREFS attribute %s references an unknown ID \"%s\"\n",
6499 attr->name, str, NULL);
6500 ctxt->valid = 0;
6501 }
6502 if (save == 0)
6503 break;
6504 *cur = save;
6505 while (IS_BLANK_CH(*cur)) cur++;
6506 }
6507 xmlFree(dup);
6508 }
6509 }
6510
6511 /**
6512 * xmlWalkValidateList:
6513 * @data: Contents of current link
6514 * @user: Value supplied by the user
6515 *
6516 * Returns 0 to abort the walk or 1 to continue
6517 */
6518 static int
6519 xmlWalkValidateList(const void *data, const void *user)
6520 {
6521 xmlValidateMemoPtr memo = (xmlValidateMemoPtr)user;
6522 xmlValidateRef((xmlRefPtr)data, memo->ctxt, memo->name);
6523 return 1;
6524 }
6525
6526 /**
6527 * xmlValidateCheckRefCallback:
6528 * @ref_list: List of references
6529 * @ctxt: Validation context
6530 * @name: Name of ID we are searching for
6531 *
6532 */
6533 static void
6534 xmlValidateCheckRefCallback(xmlListPtr ref_list, xmlValidCtxtPtr ctxt,
6535 const xmlChar *name) {
6536 xmlValidateMemo memo;
6537
6538 if (ref_list == NULL)
6539 return;
6540 memo.ctxt = ctxt;
6541 memo.name = name;
6542
6543 xmlListWalk(ref_list, xmlWalkValidateList, &memo);
6544
6545 }
6546
6547 /**
6548 * xmlValidateDocumentFinal:
6549 * @ctxt: the validation context
6550 * @doc: a document instance
6551 *
6552 * Does the final step for the document validation once all the
6553 * incremental validation steps have been completed
6554 *
6555 * basically it does the following checks described by the XML Rec
6556 *
6557 * Check all the IDREF/IDREFS attributes definition for validity
6558 *
6559 * returns 1 if valid or 0 otherwise
6560 */
6561
6562 int
6563 xmlValidateDocumentFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6564 xmlRefTablePtr table;
6565 unsigned int save;
6566
6567 if (ctxt == NULL)
6568 return(0);
6569 if (doc == NULL) {
6570 xmlErrValid(ctxt, XML_DTD_NO_DOC,
6571 "xmlValidateDocumentFinal: doc == NULL\n", NULL);
6572 return(0);
6573 }
6574
6575 /* trick to get correct line id report */
6576 save = ctxt->finishDtd;
6577 ctxt->finishDtd = 0;
6578
6579 /*
6580 * Check all the NOTATION/NOTATIONS attributes
6581 */
6582 /*
6583 * Check all the ENTITY/ENTITIES attributes definition for validity
6584 */
6585 /*
6586 * Check all the IDREF/IDREFS attributes definition for validity
6587 */
6588 table = (xmlRefTablePtr) doc->refs;
6589 ctxt->doc = doc;
6590 ctxt->valid = 1;
6591 xmlHashScan(table, (xmlHashScanner) xmlValidateCheckRefCallback, ctxt);
6592
6593 ctxt->finishDtd = save;
6594 return(ctxt->valid);
6595 }
6596
6597 /**
6598 * xmlValidateDtd:
6599 * @ctxt: the validation context
6600 * @doc: a document instance
6601 * @dtd: a dtd instance
6602 *
6603 * Try to validate the document against the dtd instance
6604 *
6605 * Basically it does check all the definitions in the DtD.
6606 * Note the the internal subset (if present) is de-coupled
6607 * (i.e. not used), which could give problems if ID or IDREF
6608 * is present.
6609 *
6610 * returns 1 if valid or 0 otherwise
6611 */
6612
6613 int
6614 xmlValidateDtd(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlDtdPtr dtd) {
6615 int ret;
6616 xmlDtdPtr oldExt, oldInt;
6617 xmlNodePtr root;
6618
6619 if (dtd == NULL) return(0);
6620 if (doc == NULL) return(0);
6621 oldExt = doc->extSubset;
6622 oldInt = doc->intSubset;
6623 doc->extSubset = dtd;
6624 doc->intSubset = NULL;
6625 ret = xmlValidateRoot(ctxt, doc);
6626 if (ret == 0) {
6627 doc->extSubset = oldExt;
6628 doc->intSubset = oldInt;
6629 return(ret);
6630 }
6631 if (doc->ids != NULL) {
6632 xmlFreeIDTable(doc->ids);
6633 doc->ids = NULL;
6634 }
6635 if (doc->refs != NULL) {
6636 xmlFreeRefTable(doc->refs);
6637 doc->refs = NULL;
6638 }
6639 root = xmlDocGetRootElement(doc);
6640 ret = xmlValidateElement(ctxt, doc, root);
6641 ret &= xmlValidateDocumentFinal(ctxt, doc);
6642 doc->extSubset = oldExt;
6643 doc->intSubset = oldInt;
6644 return(ret);
6645 }
6646
6647 static void
6648 xmlValidateNotationCallback(xmlEntityPtr cur, xmlValidCtxtPtr ctxt,
6649 const xmlChar *name ATTRIBUTE_UNUSED) {
6650 if (cur == NULL)
6651 return;
6652 if (cur->etype == XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
6653 xmlChar *notation = cur->content;
6654
6655 if (notation != NULL) {
6656 int ret;
6657
6658 ret = xmlValidateNotationUse(ctxt, cur->doc, notation);
6659 if (ret != 1) {
6660 ctxt->valid = 0;
6661 }
6662 }
6663 }
6664 }
6665
6666 static void
6667 xmlValidateAttributeCallback(xmlAttributePtr cur, xmlValidCtxtPtr ctxt,
6668 const xmlChar *name ATTRIBUTE_UNUSED) {
6669 int ret;
6670 xmlDocPtr doc;
6671 xmlElementPtr elem = NULL;
6672
6673 if (cur == NULL)
6674 return;
6675 switch (cur->atype) {
6676 case XML_ATTRIBUTE_CDATA:
6677 case XML_ATTRIBUTE_ID:
6678 case XML_ATTRIBUTE_IDREF :
6679 case XML_ATTRIBUTE_IDREFS:
6680 case XML_ATTRIBUTE_NMTOKEN:
6681 case XML_ATTRIBUTE_NMTOKENS:
6682 case XML_ATTRIBUTE_ENUMERATION:
6683 break;
6684 case XML_ATTRIBUTE_ENTITY:
6685 case XML_ATTRIBUTE_ENTITIES:
6686 case XML_ATTRIBUTE_NOTATION:
6687 if (cur->defaultValue != NULL) {
6688
6689 ret = xmlValidateAttributeValue2(ctxt, ctxt->doc, cur->name,
6690 cur->atype, cur->defaultValue);
6691 if ((ret == 0) && (ctxt->valid == 1))
6692 ctxt->valid = 0;
6693 }
6694 if (cur->tree != NULL) {
6695 xmlEnumerationPtr tree = cur->tree;
6696 while (tree != NULL) {
6697 ret = xmlValidateAttributeValue2(ctxt, ctxt->doc,
6698 cur->name, cur->atype, tree->name);
6699 if ((ret == 0) && (ctxt->valid == 1))
6700 ctxt->valid = 0;
6701 tree = tree->next;
6702 }
6703 }
6704 }
6705 if (cur->atype == XML_ATTRIBUTE_NOTATION) {
6706 doc = cur->doc;
6707 if (cur->elem == NULL) {
6708 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
6709 "xmlValidateAttributeCallback(%s): internal error\n",
6710 (const char *) cur->name);
6711 return;
6712 }
6713
6714 if (doc != NULL)
6715 elem = xmlGetDtdElementDesc(doc->intSubset, cur->elem);
6716 if ((elem == NULL) && (doc != NULL))
6717 elem = xmlGetDtdElementDesc(doc->extSubset, cur->elem);
6718 if ((elem == NULL) && (cur->parent != NULL) &&
6719 (cur->parent->type == XML_DTD_NODE))
6720 elem = xmlGetDtdElementDesc((xmlDtdPtr) cur->parent, cur->elem);
6721 if (elem == NULL) {
6722 xmlErrValidNode(ctxt, NULL, XML_DTD_UNKNOWN_ELEM,
6723 "attribute %s: could not find decl for element %s\n",
6724 cur->name, cur->elem, NULL);
6725 return;
6726 }
6727 if (elem->etype == XML_ELEMENT_TYPE_EMPTY) {
6728 xmlErrValidNode(ctxt, NULL, XML_DTD_EMPTY_NOTATION,
6729 "NOTATION attribute %s declared for EMPTY element %s\n",
6730 cur->name, cur->elem, NULL);
6731 ctxt->valid = 0;
6732 }
6733 }
6734 }
6735
6736 /**
6737 * xmlValidateDtdFinal:
6738 * @ctxt: the validation context
6739 * @doc: a document instance
6740 *
6741 * Does the final step for the dtds validation once all the
6742 * subsets have been parsed
6743 *
6744 * basically it does the following checks described by the XML Rec
6745 * - check that ENTITY and ENTITIES type attributes default or
6746 * possible values matches one of the defined entities.
6747 * - check that NOTATION type attributes default or
6748 * possible values matches one of the defined notations.
6749 *
6750 * returns 1 if valid or 0 if invalid and -1 if not well-formed
6751 */
6752
6753 int
6754 xmlValidateDtdFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6755 xmlDtdPtr dtd;
6756 xmlAttributeTablePtr table;
6757 xmlEntitiesTablePtr entities;
6758
6759 if ((doc == NULL) || (ctxt == NULL)) return(0);
6760 if ((doc->intSubset == NULL) && (doc->extSubset == NULL))
6761 return(0);
6762 ctxt->doc = doc;
6763 ctxt->valid = 1;
6764 dtd = doc->intSubset;
6765 if ((dtd != NULL) && (dtd->attributes != NULL)) {
6766 table = (xmlAttributeTablePtr) dtd->attributes;
6767 xmlHashScan(table, (xmlHashScanner) xmlValidateAttributeCallback, ctxt);
6768 }
6769 if ((dtd != NULL) && (dtd->entities != NULL)) {
6770 entities = (xmlEntitiesTablePtr) dtd->entities;
6771 xmlHashScan(entities, (xmlHashScanner) xmlValidateNotationCallback,
6772 ctxt);
6773 }
6774 dtd = doc->extSubset;
6775 if ((dtd != NULL) && (dtd->attributes != NULL)) {
6776 table = (xmlAttributeTablePtr) dtd->attributes;
6777 xmlHashScan(table, (xmlHashScanner) xmlValidateAttributeCallback, ctxt);
6778 }
6779 if ((dtd != NULL) && (dtd->entities != NULL)) {
6780 entities = (xmlEntitiesTablePtr) dtd->entities;
6781 xmlHashScan(entities, (xmlHashScanner) xmlValidateNotationCallback,
6782 ctxt);
6783 }
6784 return(ctxt->valid);
6785 }
6786
6787 /**
6788 * xmlValidateDocument:
6789 * @ctxt: the validation context
6790 * @doc: a document instance
6791 *
6792 * Try to validate the document instance
6793 *
6794 * basically it does the all the checks described by the XML Rec
6795 * i.e. validates the internal and external subset (if present)
6796 * and validate the document tree.
6797 *
6798 * returns 1 if valid or 0 otherwise
6799 */
6800
6801 int
6802 xmlValidateDocument(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6803 int ret;
6804 xmlNodePtr root;
6805
6806 if (doc == NULL)
6807 return(0);
6808 if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
6809 xmlErrValid(ctxt, XML_DTD_NO_DTD,
6810 "no DTD found!\n", NULL);
6811 return(0);
6812 }
6813 if ((doc->intSubset != NULL) && ((doc->intSubset->SystemID != NULL) ||
6814 (doc->intSubset->ExternalID != NULL)) && (doc->extSubset == NULL)) {
6815 xmlChar *sysID;
6816 if (doc->intSubset->SystemID != NULL) {
6817 sysID = xmlBuildURI(doc->intSubset->SystemID,
6818 doc->URL);
6819 if (sysID == NULL) {
6820 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
6821 "Could not build URI for external subset \"%s\"\n",
6822 (const char *) doc->intSubset->SystemID);
6823 return 0;
6824 }
6825 } else
6826 sysID = NULL;
6827 doc->extSubset = xmlParseDTD(doc->intSubset->ExternalID,
6828 (const xmlChar *)sysID);
6829 if (sysID != NULL)
6830 xmlFree(sysID);
6831 if (doc->extSubset == NULL) {
6832 if (doc->intSubset->SystemID != NULL) {
6833 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
6834 "Could not load the external subset \"%s\"\n",
6835 (const char *) doc->intSubset->SystemID);
6836 } else {
6837 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
6838 "Could not load the external subset \"%s\"\n",
6839 (const char *) doc->intSubset->ExternalID);
6840 }
6841 return(0);
6842 }
6843 }
6844
6845 if (doc->ids != NULL) {
6846 xmlFreeIDTable(doc->ids);
6847 doc->ids = NULL;
6848 }
6849 if (doc->refs != NULL) {
6850 xmlFreeRefTable(doc->refs);
6851 doc->refs = NULL;
6852 }
6853 ret = xmlValidateDtdFinal(ctxt, doc);
6854 if (!xmlValidateRoot(ctxt, doc)) return(0);
6855
6856 root = xmlDocGetRootElement(doc);
6857 ret &= xmlValidateElement(ctxt, doc, root);
6858 ret &= xmlValidateDocumentFinal(ctxt, doc);
6859 return(ret);
6860 }
6861
6862 /************************************************************************
6863 * *
6864 * Routines for dynamic validation editing *
6865 * *
6866 ************************************************************************/
6867
6868 /**
6869 * xmlValidGetPotentialChildren:
6870 * @ctree: an element content tree
6871 * @names: an array to store the list of child names
6872 * @len: a pointer to the number of element in the list
6873 * @max: the size of the array
6874 *
6875 * Build/extend a list of potential children allowed by the content tree
6876 *
6877 * returns the number of element in the list, or -1 in case of error.
6878 */
6879
6880 int
6881 xmlValidGetPotentialChildren(xmlElementContent *ctree,
6882 const xmlChar **names,
6883 int *len, int max) {
6884 int i;
6885
6886 if ((ctree == NULL) || (names == NULL) || (len == NULL))
6887 return(-1);
6888 if (*len >= max) return(*len);
6889
6890 switch (ctree->type) {
6891 case XML_ELEMENT_CONTENT_PCDATA:
6892 for (i = 0; i < *len;i++)
6893 if (xmlStrEqual(BAD_CAST "#PCDATA", names[i])) return(*len);
6894 names[(*len)++] = BAD_CAST "#PCDATA";
6895 break;
6896 case XML_ELEMENT_CONTENT_ELEMENT:
6897 for (i = 0; i < *len;i++)
6898 if (xmlStrEqual(ctree->name, names[i])) return(*len);
6899 names[(*len)++] = ctree->name;
6900 break;
6901 case XML_ELEMENT_CONTENT_SEQ:
6902 xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6903 xmlValidGetPotentialChildren(ctree->c2, names, len, max);
6904 break;
6905 case XML_ELEMENT_CONTENT_OR:
6906 xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6907 xmlValidGetPotentialChildren(ctree->c2, names, len, max);
6908 break;
6909 }
6910
6911 return(*len);
6912 }
6913
6914 /*
6915 * Dummy function to suppress messages while we try out valid elements
6916 */
6917 static void XMLCDECL xmlNoValidityErr(void *ctx ATTRIBUTE_UNUSED,
6918 const char *msg ATTRIBUTE_UNUSED, ...) {
6919 return;
6920 }
6921
6922 /**
6923 * xmlValidGetValidElements:
6924 * @prev: an element to insert after
6925 * @next: an element to insert next
6926 * @names: an array to store the list of child names
6927 * @max: the size of the array
6928 *
6929 * This function returns the list of authorized children to insert
6930 * within an existing tree while respecting the validity constraints
6931 * forced by the Dtd. The insertion point is defined using @prev and
6932 * @next in the following ways:
6933 * to insert before 'node': xmlValidGetValidElements(node->prev, node, ...
6934 * to insert next 'node': xmlValidGetValidElements(node, node->next, ...
6935 * to replace 'node': xmlValidGetValidElements(node->prev, node->next, ...
6936 * to prepend a child to 'node': xmlValidGetValidElements(NULL, node->childs,
6937 * to append a child to 'node': xmlValidGetValidElements(node->last, NULL, ...
6938 *
6939 * pointers to the element names are inserted at the beginning of the array
6940 * and do not need to be freed.
6941 *
6942 * returns the number of element in the list, or -1 in case of error. If
6943 * the function returns the value @max the caller is invited to grow the
6944 * receiving array and retry.
6945 */
6946
6947 int
6948 xmlValidGetValidElements(xmlNode *prev, xmlNode *next, const xmlChar **names,
6949 int max) {
6950 xmlValidCtxt vctxt;
6951 int nb_valid_elements = 0;
6952 const xmlChar *elements[256]={0};
6953 int nb_elements = 0, i;
6954 const xmlChar *name;
6955
6956 xmlNode *ref_node;
6957 xmlNode *parent;
6958 xmlNode *test_node;
6959
6960 xmlNode *prev_next;
6961 xmlNode *next_prev;
6962 xmlNode *parent_childs;
6963 xmlNode *parent_last;
6964
6965 xmlElement *element_desc;
6966
6967 if (prev == NULL && next == NULL)
6968 return(-1);
6969
6970 if (names == NULL) return(-1);
6971 if (max <= 0) return(-1);
6972
6973 memset(&vctxt, 0, sizeof (xmlValidCtxt));
6974 vctxt.error = xmlNoValidityErr; /* this suppresses err/warn output */
6975
6976 nb_valid_elements = 0;
6977 ref_node = prev ? prev : next;
6978 parent = ref_node->parent;
6979
6980 /*
6981 * Retrieves the parent element declaration
6982 */
6983 element_desc = xmlGetDtdElementDesc(parent->doc->intSubset,
6984 parent->name);
6985 if ((element_desc == NULL) && (parent->doc->extSubset != NULL))
6986 element_desc = xmlGetDtdElementDesc(parent->doc->extSubset,
6987 parent->name);
6988 if (element_desc == NULL) return(-1);
6989
6990 /*
6991 * Do a backup of the current tree structure
6992 */
6993 prev_next = prev ? prev->next : NULL;
6994 next_prev = next ? next->prev : NULL;
6995 parent_childs = parent->children;
6996 parent_last = parent->last;
6997
6998 /*
6999 * Creates a dummy node and insert it into the tree
7000 */
7001 test_node = xmlNewDocNode (ref_node->doc, NULL, BAD_CAST "<!dummy?>", NULL);
7002 if (test_node == NULL)
7003 return(-1);
7004
7005 test_node->parent = parent;
7006 test_node->prev = prev;
7007 test_node->next = next;
7008 name = test_node->name;
7009
7010 if (prev) prev->next = test_node;
7011 else parent->children = test_node;
7012
7013 if (next) next->prev = test_node;
7014 else parent->last = test_node;
7015
7016 /*
7017 * Insert each potential child node and check if the parent is
7018 * still valid
7019 */
7020 nb_elements = xmlValidGetPotentialChildren(element_desc->content,
7021 elements, &nb_elements, 256);
7022
7023 for (i = 0;i < nb_elements;i++) {
7024 test_node->name = elements[i];
7025 if (xmlValidateOneElement(&vctxt, parent->doc, parent)) {
7026 int j;
7027
7028 for (j = 0; j < nb_valid_elements;j++)
7029 if (xmlStrEqual(elements[i], names[j])) break;
7030 names[nb_valid_elements++] = elements[i];
7031 if (nb_valid_elements >= max) break;
7032 }
7033 }
7034
7035 /*
7036 * Restore the tree structure
7037 */
7038 if (prev) prev->next = prev_next;
7039 if (next) next->prev = next_prev;
7040 parent->children = parent_childs;
7041 parent->last = parent_last;
7042
7043 /*
7044 * Free up the dummy node
7045 */
7046 test_node->name = name;
7047 xmlFreeNode(test_node);
7048
7049 return(nb_valid_elements);
7050 }
7051 #endif /* LIBXML_VALID_ENABLED */
7052
7053 #define bottom_valid
7054 #include "elfgcchack.h"