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
29static 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
40static int
41xmlValidateAttributeValueInternal(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 */
57static void
58xmlVErrMemory(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 */
96static void
97xmlErrValid(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 */
140static void
141xmlErrValidNode(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 */
183static void
184xmlErrValidNodeNr(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 */
224static void
225xmlErrValidWarning(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
266typedef 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
273static int
274vstateVPush(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
317static int
318vstateVPop(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
351typedef 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
373static int
374vstateVPush(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
421static int
422vstateVPop(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
436static int
437nodeVPush(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}
465static xmlNodePtr
466nodeVPop(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
483static void
484xmlValidPrintNode(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
558static void
559xmlValidPrintNodeList(xmlNodePtr cur) {
560    if (cur == NULL)
561	xmlGenericError(xmlGenericErrorContext, "null ");
562    while (cur != NULL) {
563	xmlValidPrintNode(cur);
564	cur = cur->next;
565    }
566}
567
568static void
569xmlValidDebug(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
580static void
581xmlValidDebugState(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
606static void
607xmlValidStateDebug(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 */
658static int
659xmlValidBuildAContentModel(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 */
812int
813xmlValidBuildContentModel(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 */
880xmlValidCtxtPtr 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 */
899void
900xmlFreeValidCtxt(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 */
920xmlElementContentPtr
921xmlNewDocElementContent(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 */
993xmlElementContentPtr
994xmlNewElementContent(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 */
1007xmlElementContentPtr
1008xmlCopyDocElementContent(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 */
1088xmlElementContentPtr
1089xmlCopyElementContent(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 */
1100void
1101xmlFreeDocElementContent(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 */
1144void
1145xmlFreeElementContent(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 */
1158static void
1159xmlDumpElementContent(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 */
1232void
1233xmlSprintfElementContent(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 */
1249void
1250xmlSnprintfElementContent(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 */
1351static void
1352xmlFreeElement(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 */
1380xmlElementPtr
1381xmlAddElementDecl(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 */
1600void
1601xmlFreeElementTable(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 */
1614static xmlElementPtr
1615xmlCopyElement(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 */
1648xmlElementTablePtr
1649xmlCopyElementTable(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 */
1664void
1665xmlDumpElementDecl(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 */
1724static void
1725xmlDumpElementDeclScan(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 */
1736void
1737xmlDumpElementTable(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 */
1753xmlEnumerationPtr
1754xmlCreateEnumeration(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 */
1775void
1776xmlFreeEnumeration(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 */
1795xmlEnumerationPtr
1796xmlCopyEnumeration(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 */
1818static void
1819xmlDumpEnumeration(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 */
1845static int
1846xmlScanIDAttributeDecl(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 */
1872static void
1873xmlFreeAttribute(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 */
1925xmlAttributePtr
1926xmlAddAttributeDecl(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 */
2146void
2147xmlFreeAttributeTable(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 */
2160static xmlAttributePtr
2161xmlCopyAttribute(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 */
2193xmlAttributeTablePtr
2194xmlCopyAttributeTable(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 */
2209void
2210xmlDumpAttributeDecl(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 */
2290static void
2291xmlDumpAttributeDeclScan(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 */
2302void
2303xmlDumpAttributeTable(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 */
2321static void
2322xmlFreeNotation(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 */
2346xmlNotationPtr
2347xmlAddNotationDecl(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 */
2418void
2419xmlFreeNotationTable(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 */
2432static xmlNotationPtr
2433xmlCopyNotation(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 */
2464xmlNotationTablePtr
2465xmlCopyNotationTable(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 */
2479void
2480xmlDumpNotationDecl(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 */
2506static void
2507xmlDumpNotationDeclScan(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 */
2518void
2519xmlDumpNotationTable(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 */
2549static void
2550xmlFreeID(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 */
2577xmlIDPtr
2578xmlAddID(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	xmlErrValidNode(ctxt, attr->parent, XML_DTD_ID_REDEFINED,
2638			"ID %s already defined\n", value, NULL, NULL);
2639#endif /* LIBXML_VALID_ENABLED */
2640	xmlFreeID(ret);
2641	return(NULL);
2642    }
2643    if (attr != NULL)
2644	attr->atype = XML_ATTRIBUTE_ID;
2645    return(ret);
2646}
2647
2648/**
2649 * xmlFreeIDTable:
2650 * @table:  An id table
2651 *
2652 * Deallocate the memory used by an ID hash table.
2653 */
2654void
2655xmlFreeIDTable(xmlIDTablePtr table) {
2656    xmlHashFree(table, (xmlHashDeallocator) xmlFreeID);
2657}
2658
2659/**
2660 * xmlIsID:
2661 * @doc:  the document
2662 * @elem:  the element carrying the attribute
2663 * @attr:  the attribute
2664 *
2665 * Determine whether an attribute is of type ID. In case we have DTD(s)
2666 * then this is done if DTD loading has been requested. In the case
2667 * of HTML documents parsed with the HTML parser, then ID detection is
2668 * done systematically.
2669 *
2670 * Returns 0 or 1 depending on the lookup result
2671 */
2672int
2673xmlIsID(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) {
2674    if ((attr == NULL) || (attr->name == NULL)) return(0);
2675    if ((attr->ns != NULL) && (attr->ns->prefix != NULL) &&
2676        (!strcmp((char *) attr->name, "id")) &&
2677        (!strcmp((char *) attr->ns->prefix, "xml")))
2678	return(1);
2679    if (doc == NULL) return(0);
2680    if ((doc->intSubset == NULL) && (doc->extSubset == NULL) &&
2681        (doc->type != XML_HTML_DOCUMENT_NODE)) {
2682	return(0);
2683    } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
2684        if ((xmlStrEqual(BAD_CAST "id", attr->name)) ||
2685	    ((xmlStrEqual(BAD_CAST "name", attr->name)) &&
2686	    ((elem == NULL) || (xmlStrEqual(elem->name, BAD_CAST "a")))))
2687	    return(1);
2688	return(0);
2689    } else if (elem == NULL) {
2690	return(0);
2691    } else {
2692	xmlAttributePtr attrDecl = NULL;
2693
2694	xmlChar felem[50], fattr[50];
2695	xmlChar *fullelemname, *fullattrname;
2696
2697	fullelemname = (elem->ns != NULL && elem->ns->prefix != NULL) ?
2698	    xmlBuildQName(elem->name, elem->ns->prefix, felem, 50) :
2699	    (xmlChar *)elem->name;
2700
2701	fullattrname = (attr->ns != NULL && attr->ns->prefix != NULL) ?
2702	    xmlBuildQName(attr->name, attr->ns->prefix, fattr, 50) :
2703	    (xmlChar *)attr->name;
2704
2705	if (fullelemname != NULL && fullattrname != NULL) {
2706	    attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullelemname,
2707		                         fullattrname);
2708	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
2709		attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullelemname,
2710					     fullattrname);
2711	}
2712
2713	if ((fullattrname != fattr) && (fullattrname != attr->name))
2714	    xmlFree(fullattrname);
2715	if ((fullelemname != felem) && (fullelemname != elem->name))
2716	    xmlFree(fullelemname);
2717
2718        if ((attrDecl != NULL) && (attrDecl->atype == XML_ATTRIBUTE_ID))
2719	    return(1);
2720    }
2721    return(0);
2722}
2723
2724/**
2725 * xmlRemoveID:
2726 * @doc:  the document
2727 * @attr:  the attribute
2728 *
2729 * Remove the given attribute from the ID table maintained internally.
2730 *
2731 * Returns -1 if the lookup failed and 0 otherwise
2732 */
2733int
2734xmlRemoveID(xmlDocPtr doc, xmlAttrPtr attr) {
2735    xmlIDTablePtr table;
2736    xmlIDPtr id;
2737    xmlChar *ID;
2738
2739    if (doc == NULL) return(-1);
2740    if (attr == NULL) return(-1);
2741
2742    table = (xmlIDTablePtr) doc->ids;
2743    if (table == NULL)
2744        return(-1);
2745
2746    ID = xmlNodeListGetString(doc, attr->children, 1);
2747    if (ID == NULL)
2748        return(-1);
2749
2750    id = xmlHashLookup(table, ID);
2751    if (id == NULL || id->attr != attr) {
2752        xmlFree(ID);
2753        return(-1);
2754    }
2755
2756    xmlHashRemoveEntry(table, ID, (xmlHashDeallocator) xmlFreeID);
2757    xmlFree(ID);
2758    attr->atype = 0;
2759    return(0);
2760}
2761
2762/**
2763 * xmlGetID:
2764 * @doc:  pointer to the document
2765 * @ID:  the ID value
2766 *
2767 * Search the attribute declaring the given ID
2768 *
2769 * Returns NULL if not found, otherwise the xmlAttrPtr defining the ID
2770 */
2771xmlAttrPtr
2772xmlGetID(xmlDocPtr doc, const xmlChar *ID) {
2773    xmlIDTablePtr table;
2774    xmlIDPtr id;
2775
2776    if (doc == NULL) {
2777	return(NULL);
2778    }
2779
2780    if (ID == NULL) {
2781	return(NULL);
2782    }
2783
2784    table = (xmlIDTablePtr) doc->ids;
2785    if (table == NULL)
2786        return(NULL);
2787
2788    id = xmlHashLookup(table, ID);
2789    if (id == NULL)
2790	return(NULL);
2791    if (id->attr == NULL) {
2792	/*
2793	 * We are operating on a stream, return a well known reference
2794	 * since the attribute node doesn't exist anymore
2795	 */
2796	return((xmlAttrPtr) doc);
2797    }
2798    return(id->attr);
2799}
2800
2801/************************************************************************
2802 *									*
2803 *				Refs					*
2804 *									*
2805 ************************************************************************/
2806typedef struct xmlRemoveMemo_t
2807{
2808	xmlListPtr l;
2809	xmlAttrPtr ap;
2810} xmlRemoveMemo;
2811
2812typedef xmlRemoveMemo *xmlRemoveMemoPtr;
2813
2814typedef struct xmlValidateMemo_t
2815{
2816    xmlValidCtxtPtr ctxt;
2817    const xmlChar *name;
2818} xmlValidateMemo;
2819
2820typedef xmlValidateMemo *xmlValidateMemoPtr;
2821
2822/**
2823 * xmlFreeRef:
2824 * @lk:  A list link
2825 *
2826 * Deallocate the memory used by a ref definition
2827 */
2828static void
2829xmlFreeRef(xmlLinkPtr lk) {
2830    xmlRefPtr ref = (xmlRefPtr)xmlLinkGetData(lk);
2831    if (ref == NULL) return;
2832    if (ref->value != NULL)
2833        xmlFree((xmlChar *)ref->value);
2834    if (ref->name != NULL)
2835        xmlFree((xmlChar *)ref->name);
2836    xmlFree(ref);
2837}
2838
2839/**
2840 * xmlFreeRefList:
2841 * @list_ref:  A list of references.
2842 *
2843 * Deallocate the memory used by a list of references
2844 */
2845static void
2846xmlFreeRefList(xmlListPtr list_ref) {
2847    if (list_ref == NULL) return;
2848    xmlListDelete(list_ref);
2849}
2850
2851/**
2852 * xmlWalkRemoveRef:
2853 * @data:  Contents of current link
2854 * @user:  Value supplied by the user
2855 *
2856 * Returns 0 to abort the walk or 1 to continue
2857 */
2858static int
2859xmlWalkRemoveRef(const void *data, const void *user)
2860{
2861    xmlAttrPtr attr0 = ((xmlRefPtr)data)->attr;
2862    xmlAttrPtr attr1 = ((xmlRemoveMemoPtr)user)->ap;
2863    xmlListPtr ref_list = ((xmlRemoveMemoPtr)user)->l;
2864
2865    if (attr0 == attr1) { /* Matched: remove and terminate walk */
2866        xmlListRemoveFirst(ref_list, (void *)data);
2867        return 0;
2868    }
2869    return 1;
2870}
2871
2872/**
2873 * xmlDummyCompare
2874 * @data0:  Value supplied by the user
2875 * @data1:  Value supplied by the user
2876 *
2877 * Do nothing, return 0. Used to create unordered lists.
2878 */
2879static int
2880xmlDummyCompare(const void *data0 ATTRIBUTE_UNUSED,
2881                const void *data1 ATTRIBUTE_UNUSED)
2882{
2883    return (0);
2884}
2885
2886/**
2887 * xmlAddRef:
2888 * @ctxt:  the validation context
2889 * @doc:  pointer to the document
2890 * @value:  the value name
2891 * @attr:  the attribute holding the Ref
2892 *
2893 * Register a new ref declaration
2894 *
2895 * Returns NULL if not, otherwise the new xmlRefPtr
2896 */
2897xmlRefPtr
2898xmlAddRef(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value,
2899    xmlAttrPtr attr) {
2900    xmlRefPtr ret;
2901    xmlRefTablePtr table;
2902    xmlListPtr ref_list;
2903
2904    if (doc == NULL) {
2905        return(NULL);
2906    }
2907    if (value == NULL) {
2908        return(NULL);
2909    }
2910    if (attr == NULL) {
2911        return(NULL);
2912    }
2913
2914    /*
2915     * Create the Ref table if needed.
2916     */
2917    table = (xmlRefTablePtr) doc->refs;
2918    if (table == NULL) {
2919        doc->refs = table = xmlHashCreateDict(0, doc->dict);
2920    }
2921    if (table == NULL) {
2922	xmlVErrMemory(ctxt,
2923            "xmlAddRef: Table creation failed!\n");
2924        return(NULL);
2925    }
2926
2927    ret = (xmlRefPtr) xmlMalloc(sizeof(xmlRef));
2928    if (ret == NULL) {
2929	xmlVErrMemory(ctxt, "malloc failed");
2930        return(NULL);
2931    }
2932
2933    /*
2934     * fill the structure.
2935     */
2936    ret->value = xmlStrdup(value);
2937    if ((ctxt != NULL) && (ctxt->vstateNr != 0)) {
2938	/*
2939	 * Operating in streaming mode, attr is gonna disapear
2940	 */
2941	ret->name = xmlStrdup(attr->name);
2942	ret->attr = NULL;
2943    } else {
2944	ret->name = NULL;
2945	ret->attr = attr;
2946    }
2947    ret->lineno = xmlGetLineNo(attr->parent);
2948
2949    /* To add a reference :-
2950     * References are maintained as a list of references,
2951     * Lookup the entry, if no entry create new nodelist
2952     * Add the owning node to the NodeList
2953     * Return the ref
2954     */
2955
2956    if (NULL == (ref_list = xmlHashLookup(table, value))) {
2957        if (NULL == (ref_list = xmlListCreate(xmlFreeRef, xmlDummyCompare))) {
2958	    xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2959		    "xmlAddRef: Reference list creation failed!\n",
2960		    NULL);
2961	    goto failed;
2962        }
2963        if (xmlHashAddEntry(table, value, ref_list) < 0) {
2964            xmlListDelete(ref_list);
2965	    xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2966		    "xmlAddRef: Reference list insertion failed!\n",
2967		    NULL);
2968	    goto failed;
2969        }
2970    }
2971    if (xmlListAppend(ref_list, ret) != 0) {
2972	xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2973		    "xmlAddRef: Reference list insertion failed!\n",
2974		    NULL);
2975        goto failed;
2976    }
2977    return(ret);
2978failed:
2979    if (ret != NULL) {
2980        if (ret->value != NULL)
2981	    xmlFree((char *)ret->value);
2982        if (ret->name != NULL)
2983	    xmlFree((char *)ret->name);
2984        xmlFree(ret);
2985    }
2986    return(NULL);
2987}
2988
2989/**
2990 * xmlFreeRefTable:
2991 * @table:  An ref table
2992 *
2993 * Deallocate the memory used by an Ref hash table.
2994 */
2995void
2996xmlFreeRefTable(xmlRefTablePtr table) {
2997    xmlHashFree(table, (xmlHashDeallocator) xmlFreeRefList);
2998}
2999
3000/**
3001 * xmlIsRef:
3002 * @doc:  the document
3003 * @elem:  the element carrying the attribute
3004 * @attr:  the attribute
3005 *
3006 * Determine whether an attribute is of type Ref. In case we have DTD(s)
3007 * then this is simple, otherwise we use an heuristic: name Ref (upper
3008 * or lowercase).
3009 *
3010 * Returns 0 or 1 depending on the lookup result
3011 */
3012int
3013xmlIsRef(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) {
3014    if (attr == NULL)
3015        return(0);
3016    if (doc == NULL) {
3017        doc = attr->doc;
3018	if (doc == NULL) return(0);
3019    }
3020
3021    if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
3022        return(0);
3023    } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
3024        /* TODO @@@ */
3025        return(0);
3026    } else {
3027        xmlAttributePtr attrDecl;
3028
3029        if (elem == NULL) return(0);
3030        attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, attr->name);
3031        if ((attrDecl == NULL) && (doc->extSubset != NULL))
3032            attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
3033		                         elem->name, attr->name);
3034
3035	if ((attrDecl != NULL) &&
3036	    (attrDecl->atype == XML_ATTRIBUTE_IDREF ||
3037	     attrDecl->atype == XML_ATTRIBUTE_IDREFS))
3038	return(1);
3039    }
3040    return(0);
3041}
3042
3043/**
3044 * xmlRemoveRef:
3045 * @doc:  the document
3046 * @attr:  the attribute
3047 *
3048 * Remove the given attribute from the Ref table maintained internally.
3049 *
3050 * Returns -1 if the lookup failed and 0 otherwise
3051 */
3052int
3053xmlRemoveRef(xmlDocPtr doc, xmlAttrPtr attr) {
3054    xmlListPtr ref_list;
3055    xmlRefTablePtr table;
3056    xmlChar *ID;
3057    xmlRemoveMemo target;
3058
3059    if (doc == NULL) return(-1);
3060    if (attr == NULL) return(-1);
3061
3062    table = (xmlRefTablePtr) doc->refs;
3063    if (table == NULL)
3064        return(-1);
3065
3066    ID = xmlNodeListGetString(doc, attr->children, 1);
3067    if (ID == NULL)
3068        return(-1);
3069
3070    ref_list = xmlHashLookup(table, ID);
3071    if(ref_list == NULL) {
3072        xmlFree(ID);
3073        return (-1);
3074    }
3075
3076    /* At this point, ref_list refers to a list of references which
3077     * have the same key as the supplied attr. Our list of references
3078     * is ordered by reference address and we don't have that information
3079     * here to use when removing. We'll have to walk the list and
3080     * check for a matching attribute, when we find one stop the walk
3081     * and remove the entry.
3082     * The list is ordered by reference, so that means we don't have the
3083     * key. Passing the list and the reference to the walker means we
3084     * will have enough data to be able to remove the entry.
3085     */
3086    target.l = ref_list;
3087    target.ap = attr;
3088
3089    /* Remove the supplied attr from our list */
3090    xmlListWalk(ref_list, xmlWalkRemoveRef, &target);
3091
3092    /*If the list is empty then remove the list entry in the hash */
3093    if (xmlListEmpty(ref_list))
3094        xmlHashUpdateEntry(table, ID, NULL, (xmlHashDeallocator)
3095        xmlFreeRefList);
3096    xmlFree(ID);
3097    return(0);
3098}
3099
3100/**
3101 * xmlGetRefs:
3102 * @doc:  pointer to the document
3103 * @ID:  the ID value
3104 *
3105 * Find the set of references for the supplied ID.
3106 *
3107 * Returns NULL if not found, otherwise node set for the ID.
3108 */
3109xmlListPtr
3110xmlGetRefs(xmlDocPtr doc, const xmlChar *ID) {
3111    xmlRefTablePtr table;
3112
3113    if (doc == NULL) {
3114        return(NULL);
3115    }
3116
3117    if (ID == NULL) {
3118        return(NULL);
3119    }
3120
3121    table = (xmlRefTablePtr) doc->refs;
3122    if (table == NULL)
3123        return(NULL);
3124
3125    return (xmlHashLookup(table, ID));
3126}
3127
3128/************************************************************************
3129 *									*
3130 *		Routines for validity checking				*
3131 *									*
3132 ************************************************************************/
3133
3134/**
3135 * xmlGetDtdElementDesc:
3136 * @dtd:  a pointer to the DtD to search
3137 * @name:  the element name
3138 *
3139 * Search the DTD for the description of this element
3140 *
3141 * returns the xmlElementPtr if found or NULL
3142 */
3143
3144xmlElementPtr
3145xmlGetDtdElementDesc(xmlDtdPtr dtd, const xmlChar *name) {
3146    xmlElementTablePtr table;
3147    xmlElementPtr cur;
3148    xmlChar *uqname = NULL, *prefix = NULL;
3149
3150    if ((dtd == NULL) || (name == NULL)) return(NULL);
3151    if (dtd->elements == NULL)
3152	return(NULL);
3153    table = (xmlElementTablePtr) dtd->elements;
3154
3155    uqname = xmlSplitQName2(name, &prefix);
3156    if (uqname != NULL)
3157        name = uqname;
3158    cur = xmlHashLookup2(table, name, prefix);
3159    if (prefix != NULL) xmlFree(prefix);
3160    if (uqname != NULL) xmlFree(uqname);
3161    return(cur);
3162}
3163/**
3164 * xmlGetDtdElementDesc2:
3165 * @dtd:  a pointer to the DtD to search
3166 * @name:  the element name
3167 * @create:  create an empty description if not found
3168 *
3169 * Search the DTD for the description of this element
3170 *
3171 * returns the xmlElementPtr if found or NULL
3172 */
3173
3174static xmlElementPtr
3175xmlGetDtdElementDesc2(xmlDtdPtr dtd, const xmlChar *name, int create) {
3176    xmlElementTablePtr table;
3177    xmlElementPtr cur;
3178    xmlChar *uqname = NULL, *prefix = NULL;
3179
3180    if (dtd == NULL) return(NULL);
3181    if (dtd->elements == NULL) {
3182	xmlDictPtr dict = NULL;
3183
3184	if (dtd->doc != NULL)
3185	    dict = dtd->doc->dict;
3186
3187	if (!create)
3188	    return(NULL);
3189	/*
3190	 * Create the Element table if needed.
3191	 */
3192	table = (xmlElementTablePtr) dtd->elements;
3193	if (table == NULL) {
3194	    table = xmlHashCreateDict(0, dict);
3195	    dtd->elements = (void *) table;
3196	}
3197	if (table == NULL) {
3198	    xmlVErrMemory(NULL, "element table allocation failed");
3199	    return(NULL);
3200	}
3201    }
3202    table = (xmlElementTablePtr) dtd->elements;
3203
3204    uqname = xmlSplitQName2(name, &prefix);
3205    if (uqname != NULL)
3206        name = uqname;
3207    cur = xmlHashLookup2(table, name, prefix);
3208    if ((cur == NULL) && (create)) {
3209	cur = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
3210	if (cur == NULL) {
3211	    xmlVErrMemory(NULL, "malloc failed");
3212	    return(NULL);
3213	}
3214	memset(cur, 0, sizeof(xmlElement));
3215	cur->type = XML_ELEMENT_DECL;
3216
3217	/*
3218	 * fill the structure.
3219	 */
3220	cur->name = xmlStrdup(name);
3221	cur->prefix = xmlStrdup(prefix);
3222	cur->etype = XML_ELEMENT_TYPE_UNDEFINED;
3223
3224	xmlHashAddEntry2(table, name, prefix, cur);
3225    }
3226    if (prefix != NULL) xmlFree(prefix);
3227    if (uqname != NULL) xmlFree(uqname);
3228    return(cur);
3229}
3230
3231/**
3232 * xmlGetDtdQElementDesc:
3233 * @dtd:  a pointer to the DtD to search
3234 * @name:  the element name
3235 * @prefix:  the element namespace prefix
3236 *
3237 * Search the DTD for the description of this element
3238 *
3239 * returns the xmlElementPtr if found or NULL
3240 */
3241
3242xmlElementPtr
3243xmlGetDtdQElementDesc(xmlDtdPtr dtd, const xmlChar *name,
3244	              const xmlChar *prefix) {
3245    xmlElementTablePtr table;
3246
3247    if (dtd == NULL) return(NULL);
3248    if (dtd->elements == NULL) return(NULL);
3249    table = (xmlElementTablePtr) dtd->elements;
3250
3251    return(xmlHashLookup2(table, name, prefix));
3252}
3253
3254/**
3255 * xmlGetDtdAttrDesc:
3256 * @dtd:  a pointer to the DtD to search
3257 * @elem:  the element name
3258 * @name:  the attribute name
3259 *
3260 * Search the DTD for the description of this attribute on
3261 * this element.
3262 *
3263 * returns the xmlAttributePtr if found or NULL
3264 */
3265
3266xmlAttributePtr
3267xmlGetDtdAttrDesc(xmlDtdPtr dtd, const xmlChar *elem, const xmlChar *name) {
3268    xmlAttributeTablePtr table;
3269    xmlAttributePtr cur;
3270    xmlChar *uqname = NULL, *prefix = NULL;
3271
3272    if (dtd == NULL) return(NULL);
3273    if (dtd->attributes == NULL) return(NULL);
3274
3275    table = (xmlAttributeTablePtr) dtd->attributes;
3276    if (table == NULL)
3277	return(NULL);
3278
3279    uqname = xmlSplitQName2(name, &prefix);
3280
3281    if (uqname != NULL) {
3282	cur = xmlHashLookup3(table, uqname, prefix, elem);
3283	if (prefix != NULL) xmlFree(prefix);
3284	if (uqname != NULL) xmlFree(uqname);
3285    } else
3286	cur = xmlHashLookup3(table, name, NULL, elem);
3287    return(cur);
3288}
3289
3290/**
3291 * xmlGetDtdQAttrDesc:
3292 * @dtd:  a pointer to the DtD to search
3293 * @elem:  the element name
3294 * @name:  the attribute name
3295 * @prefix:  the attribute namespace prefix
3296 *
3297 * Search the DTD for the description of this qualified attribute on
3298 * this element.
3299 *
3300 * returns the xmlAttributePtr if found or NULL
3301 */
3302
3303xmlAttributePtr
3304xmlGetDtdQAttrDesc(xmlDtdPtr dtd, const xmlChar *elem, const xmlChar *name,
3305	          const xmlChar *prefix) {
3306    xmlAttributeTablePtr table;
3307
3308    if (dtd == NULL) return(NULL);
3309    if (dtd->attributes == NULL) return(NULL);
3310    table = (xmlAttributeTablePtr) dtd->attributes;
3311
3312    return(xmlHashLookup3(table, name, prefix, elem));
3313}
3314
3315/**
3316 * xmlGetDtdNotationDesc:
3317 * @dtd:  a pointer to the DtD to search
3318 * @name:  the notation name
3319 *
3320 * Search the DTD for the description of this notation
3321 *
3322 * returns the xmlNotationPtr if found or NULL
3323 */
3324
3325xmlNotationPtr
3326xmlGetDtdNotationDesc(xmlDtdPtr dtd, const xmlChar *name) {
3327    xmlNotationTablePtr table;
3328
3329    if (dtd == NULL) return(NULL);
3330    if (dtd->notations == NULL) return(NULL);
3331    table = (xmlNotationTablePtr) dtd->notations;
3332
3333    return(xmlHashLookup(table, name));
3334}
3335
3336#if defined(LIBXML_VALID_ENABLED) || defined(LIBXML_SCHEMAS_ENABLED)
3337/**
3338 * xmlValidateNotationUse:
3339 * @ctxt:  the validation context
3340 * @doc:  the document
3341 * @notationName:  the notation name to check
3342 *
3343 * Validate that the given name match a notation declaration.
3344 * - [ VC: Notation Declared ]
3345 *
3346 * returns 1 if valid or 0 otherwise
3347 */
3348
3349int
3350xmlValidateNotationUse(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3351                       const xmlChar *notationName) {
3352    xmlNotationPtr notaDecl;
3353    if ((doc == NULL) || (doc->intSubset == NULL) ||
3354        (notationName == NULL)) return(-1);
3355
3356    notaDecl = xmlGetDtdNotationDesc(doc->intSubset, notationName);
3357    if ((notaDecl == NULL) && (doc->extSubset != NULL))
3358	notaDecl = xmlGetDtdNotationDesc(doc->extSubset, notationName);
3359
3360    if ((notaDecl == NULL) && (ctxt != NULL)) {
3361	xmlErrValidNode(ctxt, (xmlNodePtr) doc, XML_DTD_UNKNOWN_NOTATION,
3362	                "NOTATION %s is not declared\n",
3363		        notationName, NULL, NULL);
3364	return(0);
3365    }
3366    return(1);
3367}
3368#endif /* LIBXML_VALID_ENABLED or LIBXML_SCHEMAS_ENABLED */
3369
3370/**
3371 * xmlIsMixedElement:
3372 * @doc:  the document
3373 * @name:  the element name
3374 *
3375 * Search in the DtDs whether an element accept Mixed content (or ANY)
3376 * basically if it is supposed to accept text childs
3377 *
3378 * returns 0 if no, 1 if yes, and -1 if no element description is available
3379 */
3380
3381int
3382xmlIsMixedElement(xmlDocPtr doc, const xmlChar *name) {
3383    xmlElementPtr elemDecl;
3384
3385    if ((doc == NULL) || (doc->intSubset == NULL)) return(-1);
3386
3387    elemDecl = xmlGetDtdElementDesc(doc->intSubset, name);
3388    if ((elemDecl == NULL) && (doc->extSubset != NULL))
3389	elemDecl = xmlGetDtdElementDesc(doc->extSubset, name);
3390    if (elemDecl == NULL) return(-1);
3391    switch (elemDecl->etype) {
3392	case XML_ELEMENT_TYPE_UNDEFINED:
3393	    return(-1);
3394	case XML_ELEMENT_TYPE_ELEMENT:
3395	    return(0);
3396        case XML_ELEMENT_TYPE_EMPTY:
3397	    /*
3398	     * return 1 for EMPTY since we want VC error to pop up
3399	     * on <empty>     </empty> for example
3400	     */
3401	case XML_ELEMENT_TYPE_ANY:
3402	case XML_ELEMENT_TYPE_MIXED:
3403	    return(1);
3404    }
3405    return(1);
3406}
3407
3408#ifdef LIBXML_VALID_ENABLED
3409
3410static int
3411xmlIsDocNameStartChar(xmlDocPtr doc, int c) {
3412    if ((doc == NULL) || (doc->properties & XML_DOC_OLD10) == 0) {
3413        /*
3414	 * Use the new checks of production [4] [4a] amd [5] of the
3415	 * Update 5 of XML-1.0
3416	 */
3417	if (((c >= 'a') && (c <= 'z')) ||
3418	    ((c >= 'A') && (c <= 'Z')) ||
3419	    (c == '_') || (c == ':') ||
3420	    ((c >= 0xC0) && (c <= 0xD6)) ||
3421	    ((c >= 0xD8) && (c <= 0xF6)) ||
3422	    ((c >= 0xF8) && (c <= 0x2FF)) ||
3423	    ((c >= 0x370) && (c <= 0x37D)) ||
3424	    ((c >= 0x37F) && (c <= 0x1FFF)) ||
3425	    ((c >= 0x200C) && (c <= 0x200D)) ||
3426	    ((c >= 0x2070) && (c <= 0x218F)) ||
3427	    ((c >= 0x2C00) && (c <= 0x2FEF)) ||
3428	    ((c >= 0x3001) && (c <= 0xD7FF)) ||
3429	    ((c >= 0xF900) && (c <= 0xFDCF)) ||
3430	    ((c >= 0xFDF0) && (c <= 0xFFFD)) ||
3431	    ((c >= 0x10000) && (c <= 0xEFFFF)))
3432	    return(1);
3433    } else {
3434        if (IS_LETTER(c) || (c == '_') || (c == ':'))
3435	    return(1);
3436    }
3437    return(0);
3438}
3439
3440static int
3441xmlIsDocNameChar(xmlDocPtr doc, int c) {
3442    if ((doc == NULL) || (doc->properties & XML_DOC_OLD10) == 0) {
3443        /*
3444	 * Use the new checks of production [4] [4a] amd [5] of the
3445	 * Update 5 of XML-1.0
3446	 */
3447	if (((c >= 'a') && (c <= 'z')) ||
3448	    ((c >= 'A') && (c <= 'Z')) ||
3449	    ((c >= '0') && (c <= '9')) || /* !start */
3450	    (c == '_') || (c == ':') ||
3451	    (c == '-') || (c == '.') || (c == 0xB7) || /* !start */
3452	    ((c >= 0xC0) && (c <= 0xD6)) ||
3453	    ((c >= 0xD8) && (c <= 0xF6)) ||
3454	    ((c >= 0xF8) && (c <= 0x2FF)) ||
3455	    ((c >= 0x300) && (c <= 0x36F)) || /* !start */
3456	    ((c >= 0x370) && (c <= 0x37D)) ||
3457	    ((c >= 0x37F) && (c <= 0x1FFF)) ||
3458	    ((c >= 0x200C) && (c <= 0x200D)) ||
3459	    ((c >= 0x203F) && (c <= 0x2040)) || /* !start */
3460	    ((c >= 0x2070) && (c <= 0x218F)) ||
3461	    ((c >= 0x2C00) && (c <= 0x2FEF)) ||
3462	    ((c >= 0x3001) && (c <= 0xD7FF)) ||
3463	    ((c >= 0xF900) && (c <= 0xFDCF)) ||
3464	    ((c >= 0xFDF0) && (c <= 0xFFFD)) ||
3465	    ((c >= 0x10000) && (c <= 0xEFFFF)))
3466	     return(1);
3467    } else {
3468        if ((IS_LETTER(c)) || (IS_DIGIT(c)) ||
3469            (c == '.') || (c == '-') ||
3470	    (c == '_') || (c == ':') ||
3471	    (IS_COMBINING(c)) ||
3472	    (IS_EXTENDER(c)))
3473	    return(1);
3474    }
3475    return(0);
3476}
3477
3478/**
3479 * xmlValidateNameValue:
3480 * @doc:  pointer to the document or NULL
3481 * @value:  an Name value
3482 *
3483 * Validate that the given value match Name production
3484 *
3485 * returns 1 if valid or 0 otherwise
3486 */
3487
3488static int
3489xmlValidateNameValueInternal(xmlDocPtr doc, const xmlChar *value) {
3490    const xmlChar *cur;
3491    int val, len;
3492
3493    if (value == NULL) return(0);
3494    cur = value;
3495    val = xmlStringCurrentChar(NULL, cur, &len);
3496    cur += len;
3497    if (!xmlIsDocNameStartChar(doc, val))
3498	return(0);
3499
3500    val = xmlStringCurrentChar(NULL, cur, &len);
3501    cur += len;
3502    while (xmlIsDocNameChar(doc, val)) {
3503	val = xmlStringCurrentChar(NULL, cur, &len);
3504	cur += len;
3505    }
3506
3507    if (val != 0) return(0);
3508
3509    return(1);
3510}
3511
3512/**
3513 * xmlValidateNameValue:
3514 * @value:  an Name value
3515 *
3516 * Validate that the given value match Name production
3517 *
3518 * returns 1 if valid or 0 otherwise
3519 */
3520
3521int
3522xmlValidateNameValue(const xmlChar *value) {
3523    return(xmlValidateNameValueInternal(NULL, value));
3524}
3525
3526/**
3527 * xmlValidateNamesValueInternal:
3528 * @doc:  pointer to the document or NULL
3529 * @value:  an Names value
3530 *
3531 * Validate that the given value match Names production
3532 *
3533 * returns 1 if valid or 0 otherwise
3534 */
3535
3536static int
3537xmlValidateNamesValueInternal(xmlDocPtr doc, const xmlChar *value) {
3538    const xmlChar *cur;
3539    int val, len;
3540
3541    if (value == NULL) return(0);
3542    cur = value;
3543    val = xmlStringCurrentChar(NULL, cur, &len);
3544    cur += len;
3545
3546    if (!xmlIsDocNameStartChar(doc, val))
3547	return(0);
3548
3549    val = xmlStringCurrentChar(NULL, cur, &len);
3550    cur += len;
3551    while (xmlIsDocNameChar(doc, val)) {
3552	val = xmlStringCurrentChar(NULL, cur, &len);
3553	cur += len;
3554    }
3555
3556    /* Should not test IS_BLANK(val) here -- see erratum E20*/
3557    while (val == 0x20) {
3558	while (val == 0x20) {
3559	    val = xmlStringCurrentChar(NULL, cur, &len);
3560	    cur += len;
3561	}
3562
3563	if (!xmlIsDocNameStartChar(doc, val))
3564	    return(0);
3565
3566	val = xmlStringCurrentChar(NULL, cur, &len);
3567	cur += len;
3568
3569	while (xmlIsDocNameChar(doc, val)) {
3570	    val = xmlStringCurrentChar(NULL, cur, &len);
3571	    cur += len;
3572	}
3573    }
3574
3575    if (val != 0) return(0);
3576
3577    return(1);
3578}
3579
3580/**
3581 * xmlValidateNamesValue:
3582 * @value:  an Names value
3583 *
3584 * Validate that the given value match Names production
3585 *
3586 * returns 1 if valid or 0 otherwise
3587 */
3588
3589int
3590xmlValidateNamesValue(const xmlChar *value) {
3591    return(xmlValidateNamesValueInternal(NULL, value));
3592}
3593
3594/**
3595 * xmlValidateNmtokenValueInternal:
3596 * @doc:  pointer to the document or NULL
3597 * @value:  an Nmtoken value
3598 *
3599 * Validate that the given value match Nmtoken production
3600 *
3601 * [ VC: Name Token ]
3602 *
3603 * returns 1 if valid or 0 otherwise
3604 */
3605
3606static int
3607xmlValidateNmtokenValueInternal(xmlDocPtr doc, const xmlChar *value) {
3608    const xmlChar *cur;
3609    int val, len;
3610
3611    if (value == NULL) return(0);
3612    cur = value;
3613    val = xmlStringCurrentChar(NULL, cur, &len);
3614    cur += len;
3615
3616    if (!xmlIsDocNameChar(doc, val))
3617	return(0);
3618
3619    val = xmlStringCurrentChar(NULL, cur, &len);
3620    cur += len;
3621    while (xmlIsDocNameChar(doc, val)) {
3622	val = xmlStringCurrentChar(NULL, cur, &len);
3623	cur += len;
3624    }
3625
3626    if (val != 0) return(0);
3627
3628    return(1);
3629}
3630
3631/**
3632 * xmlValidateNmtokenValue:
3633 * @value:  an Nmtoken value
3634 *
3635 * Validate that the given value match Nmtoken production
3636 *
3637 * [ VC: Name Token ]
3638 *
3639 * returns 1 if valid or 0 otherwise
3640 */
3641
3642int
3643xmlValidateNmtokenValue(const xmlChar *value) {
3644    return(xmlValidateNmtokenValueInternal(NULL, value));
3645}
3646
3647/**
3648 * xmlValidateNmtokensValueInternal:
3649 * @doc:  pointer to the document or NULL
3650 * @value:  an Nmtokens value
3651 *
3652 * Validate that the given value match Nmtokens production
3653 *
3654 * [ VC: Name Token ]
3655 *
3656 * returns 1 if valid or 0 otherwise
3657 */
3658
3659static int
3660xmlValidateNmtokensValueInternal(xmlDocPtr doc, const xmlChar *value) {
3661    const xmlChar *cur;
3662    int val, len;
3663
3664    if (value == NULL) return(0);
3665    cur = value;
3666    val = xmlStringCurrentChar(NULL, cur, &len);
3667    cur += len;
3668
3669    while (IS_BLANK(val)) {
3670	val = xmlStringCurrentChar(NULL, cur, &len);
3671	cur += len;
3672    }
3673
3674    if (!xmlIsDocNameChar(doc, val))
3675	return(0);
3676
3677    while (xmlIsDocNameChar(doc, val)) {
3678	val = xmlStringCurrentChar(NULL, cur, &len);
3679	cur += len;
3680    }
3681
3682    /* Should not test IS_BLANK(val) here -- see erratum E20*/
3683    while (val == 0x20) {
3684	while (val == 0x20) {
3685	    val = xmlStringCurrentChar(NULL, cur, &len);
3686	    cur += len;
3687	}
3688	if (val == 0) return(1);
3689
3690	if (!xmlIsDocNameChar(doc, val))
3691	    return(0);
3692
3693	val = xmlStringCurrentChar(NULL, cur, &len);
3694	cur += len;
3695
3696	while (xmlIsDocNameChar(doc, val)) {
3697	    val = xmlStringCurrentChar(NULL, cur, &len);
3698	    cur += len;
3699	}
3700    }
3701
3702    if (val != 0) return(0);
3703
3704    return(1);
3705}
3706
3707/**
3708 * xmlValidateNmtokensValue:
3709 * @value:  an Nmtokens value
3710 *
3711 * Validate that the given value match Nmtokens production
3712 *
3713 * [ VC: Name Token ]
3714 *
3715 * returns 1 if valid or 0 otherwise
3716 */
3717
3718int
3719xmlValidateNmtokensValue(const xmlChar *value) {
3720    return(xmlValidateNmtokensValueInternal(NULL, value));
3721}
3722
3723/**
3724 * xmlValidateNotationDecl:
3725 * @ctxt:  the validation context
3726 * @doc:  a document instance
3727 * @nota:  a notation definition
3728 *
3729 * Try to validate a single notation definition
3730 * basically it does the following checks as described by the
3731 * XML-1.0 recommendation:
3732 *  - it seems that no validity constraint exists on notation declarations
3733 * But this function get called anyway ...
3734 *
3735 * returns 1 if valid or 0 otherwise
3736 */
3737
3738int
3739xmlValidateNotationDecl(xmlValidCtxtPtr ctxt ATTRIBUTE_UNUSED, xmlDocPtr doc ATTRIBUTE_UNUSED,
3740                         xmlNotationPtr nota ATTRIBUTE_UNUSED) {
3741    int ret = 1;
3742
3743    return(ret);
3744}
3745
3746/**
3747 * xmlValidateAttributeValueInternal:
3748 * @doc: the document
3749 * @type:  an attribute type
3750 * @value:  an attribute value
3751 *
3752 * Validate that the given attribute value match  the proper production
3753 *
3754 * returns 1 if valid or 0 otherwise
3755 */
3756
3757static int
3758xmlValidateAttributeValueInternal(xmlDocPtr doc, xmlAttributeType type,
3759                                  const xmlChar *value) {
3760    switch (type) {
3761	case XML_ATTRIBUTE_ENTITIES:
3762	case XML_ATTRIBUTE_IDREFS:
3763	    return(xmlValidateNamesValueInternal(doc, value));
3764	case XML_ATTRIBUTE_ENTITY:
3765	case XML_ATTRIBUTE_IDREF:
3766	case XML_ATTRIBUTE_ID:
3767	case XML_ATTRIBUTE_NOTATION:
3768	    return(xmlValidateNameValueInternal(doc, value));
3769	case XML_ATTRIBUTE_NMTOKENS:
3770	case XML_ATTRIBUTE_ENUMERATION:
3771	    return(xmlValidateNmtokensValueInternal(doc, value));
3772	case XML_ATTRIBUTE_NMTOKEN:
3773	    return(xmlValidateNmtokenValueInternal(doc, value));
3774        case XML_ATTRIBUTE_CDATA:
3775	    break;
3776    }
3777    return(1);
3778}
3779
3780/**
3781 * xmlValidateAttributeValue:
3782 * @type:  an attribute type
3783 * @value:  an attribute value
3784 *
3785 * Validate that the given attribute value match  the proper production
3786 *
3787 * [ VC: ID ]
3788 * Values of type ID must match the Name production....
3789 *
3790 * [ VC: IDREF ]
3791 * Values of type IDREF must match the Name production, and values
3792 * of type IDREFS must match Names ...
3793 *
3794 * [ VC: Entity Name ]
3795 * Values of type ENTITY must match the Name production, values
3796 * of type ENTITIES must match Names ...
3797 *
3798 * [ VC: Name Token ]
3799 * Values of type NMTOKEN must match the Nmtoken production; values
3800 * of type NMTOKENS must match Nmtokens.
3801 *
3802 * returns 1 if valid or 0 otherwise
3803 */
3804int
3805xmlValidateAttributeValue(xmlAttributeType type, const xmlChar *value) {
3806    return(xmlValidateAttributeValueInternal(NULL, type, value));
3807}
3808
3809/**
3810 * xmlValidateAttributeValue2:
3811 * @ctxt:  the validation context
3812 * @doc:  the document
3813 * @name:  the attribute name (used for error reporting only)
3814 * @type:  the attribute type
3815 * @value:  the attribute value
3816 *
3817 * Validate that the given attribute value match a given type.
3818 * This typically cannot be done before having finished parsing
3819 * the subsets.
3820 *
3821 * [ VC: IDREF ]
3822 * Values of type IDREF must match one of the declared IDs
3823 * Values of type IDREFS must match a sequence of the declared IDs
3824 * each Name must match the value of an ID attribute on some element
3825 * in the XML document; i.e. IDREF values must match the value of
3826 * some ID attribute
3827 *
3828 * [ VC: Entity Name ]
3829 * Values of type ENTITY must match one declared entity
3830 * Values of type ENTITIES must match a sequence of declared entities
3831 *
3832 * [ VC: Notation Attributes ]
3833 * all notation names in the declaration must be declared.
3834 *
3835 * returns 1 if valid or 0 otherwise
3836 */
3837
3838static int
3839xmlValidateAttributeValue2(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3840      const xmlChar *name, xmlAttributeType type, const xmlChar *value) {
3841    int ret = 1;
3842    switch (type) {
3843	case XML_ATTRIBUTE_IDREFS:
3844	case XML_ATTRIBUTE_IDREF:
3845	case XML_ATTRIBUTE_ID:
3846	case XML_ATTRIBUTE_NMTOKENS:
3847	case XML_ATTRIBUTE_ENUMERATION:
3848	case XML_ATTRIBUTE_NMTOKEN:
3849        case XML_ATTRIBUTE_CDATA:
3850	    break;
3851	case XML_ATTRIBUTE_ENTITY: {
3852	    xmlEntityPtr ent;
3853
3854	    ent = xmlGetDocEntity(doc, value);
3855	    /* yeah it's a bit messy... */
3856	    if ((ent == NULL) && (doc->standalone == 1)) {
3857		doc->standalone = 0;
3858		ent = xmlGetDocEntity(doc, value);
3859	    }
3860	    if (ent == NULL) {
3861		xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3862				XML_DTD_UNKNOWN_ENTITY,
3863   "ENTITY attribute %s reference an unknown entity \"%s\"\n",
3864		       name, value, NULL);
3865		ret = 0;
3866	    } else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
3867		xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3868				XML_DTD_ENTITY_TYPE,
3869   "ENTITY attribute %s reference an entity \"%s\" of wrong type\n",
3870		       name, value, NULL);
3871		ret = 0;
3872	    }
3873	    break;
3874        }
3875	case XML_ATTRIBUTE_ENTITIES: {
3876	    xmlChar *dup, *nam = NULL, *cur, save;
3877	    xmlEntityPtr ent;
3878
3879	    dup = xmlStrdup(value);
3880	    if (dup == NULL)
3881		return(0);
3882	    cur = dup;
3883	    while (*cur != 0) {
3884		nam = cur;
3885		while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
3886		save = *cur;
3887		*cur = 0;
3888		ent = xmlGetDocEntity(doc, nam);
3889		if (ent == NULL) {
3890		    xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3891				    XML_DTD_UNKNOWN_ENTITY,
3892       "ENTITIES attribute %s reference an unknown entity \"%s\"\n",
3893			   name, nam, NULL);
3894		    ret = 0;
3895		} else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
3896		    xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3897				    XML_DTD_ENTITY_TYPE,
3898       "ENTITIES attribute %s reference an entity \"%s\" of wrong type\n",
3899			   name, nam, NULL);
3900		    ret = 0;
3901		}
3902		if (save == 0)
3903		    break;
3904		*cur = save;
3905		while (IS_BLANK_CH(*cur)) cur++;
3906	    }
3907	    xmlFree(dup);
3908	    break;
3909	}
3910	case XML_ATTRIBUTE_NOTATION: {
3911	    xmlNotationPtr nota;
3912
3913	    nota = xmlGetDtdNotationDesc(doc->intSubset, value);
3914	    if ((nota == NULL) && (doc->extSubset != NULL))
3915		nota = xmlGetDtdNotationDesc(doc->extSubset, value);
3916
3917	    if (nota == NULL) {
3918		xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3919		                XML_DTD_UNKNOWN_NOTATION,
3920       "NOTATION attribute %s reference an unknown notation \"%s\"\n",
3921		       name, value, NULL);
3922		ret = 0;
3923	    }
3924	    break;
3925        }
3926    }
3927    return(ret);
3928}
3929
3930/**
3931 * xmlValidCtxtNormalizeAttributeValue:
3932 * @ctxt: the validation context
3933 * @doc:  the document
3934 * @elem:  the parent
3935 * @name:  the attribute name
3936 * @value:  the attribute value
3937 * @ctxt:  the validation context or NULL
3938 *
3939 * Does the validation related extra step of the normalization of attribute
3940 * values:
3941 *
3942 * If the declared value is not CDATA, then the XML processor must further
3943 * process the normalized attribute value by discarding any leading and
3944 * trailing space (#x20) characters, and by replacing sequences of space
3945 * (#x20) characters by single space (#x20) character.
3946 *
3947 * Also  check VC: Standalone Document Declaration in P32, and update
3948 *  ctxt->valid accordingly
3949 *
3950 * returns a new normalized string if normalization is needed, NULL otherwise
3951 *      the caller must free the returned value.
3952 */
3953
3954xmlChar *
3955xmlValidCtxtNormalizeAttributeValue(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3956	     xmlNodePtr elem, const xmlChar *name, const xmlChar *value) {
3957    xmlChar *ret, *dst;
3958    const xmlChar *src;
3959    xmlAttributePtr attrDecl = NULL;
3960    int extsubset = 0;
3961
3962    if (doc == NULL) return(NULL);
3963    if (elem == NULL) return(NULL);
3964    if (name == NULL) return(NULL);
3965    if (value == NULL) return(NULL);
3966
3967    if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
3968	xmlChar fn[50];
3969	xmlChar *fullname;
3970
3971	fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
3972	if (fullname == NULL)
3973	    return(NULL);
3974	attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, name);
3975	if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
3976	    attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname, name);
3977	    if (attrDecl != NULL)
3978		extsubset = 1;
3979	}
3980	if ((fullname != fn) && (fullname != elem->name))
3981	    xmlFree(fullname);
3982    }
3983    if ((attrDecl == NULL) && (doc->intSubset != NULL))
3984	attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, name);
3985    if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
3986	attrDecl = xmlGetDtdAttrDesc(doc->extSubset, elem->name, name);
3987	if (attrDecl != NULL)
3988	    extsubset = 1;
3989    }
3990
3991    if (attrDecl == NULL)
3992	return(NULL);
3993    if (attrDecl->atype == XML_ATTRIBUTE_CDATA)
3994	return(NULL);
3995
3996    ret = xmlStrdup(value);
3997    if (ret == NULL)
3998	return(NULL);
3999    src = value;
4000    dst = ret;
4001    while (*src == 0x20) src++;
4002    while (*src != 0) {
4003	if (*src == 0x20) {
4004	    while (*src == 0x20) src++;
4005	    if (*src != 0)
4006		*dst++ = 0x20;
4007	} else {
4008	    *dst++ = *src++;
4009	}
4010    }
4011    *dst = 0;
4012    if ((doc->standalone) && (extsubset == 1) && (!xmlStrEqual(value, ret))) {
4013	xmlErrValidNode(ctxt, elem, XML_DTD_NOT_STANDALONE,
4014"standalone: %s on %s value had to be normalized based on external subset declaration\n",
4015	       name, elem->name, NULL);
4016	ctxt->valid = 0;
4017    }
4018    return(ret);
4019}
4020
4021/**
4022 * xmlValidNormalizeAttributeValue:
4023 * @doc:  the document
4024 * @elem:  the parent
4025 * @name:  the attribute name
4026 * @value:  the attribute value
4027 *
4028 * Does the validation related extra step of the normalization of attribute
4029 * values:
4030 *
4031 * If the declared value is not CDATA, then the XML processor must further
4032 * process the normalized attribute value by discarding any leading and
4033 * trailing space (#x20) characters, and by replacing sequences of space
4034 * (#x20) characters by single space (#x20) character.
4035 *
4036 * Returns a new normalized string if normalization is needed, NULL otherwise
4037 *      the caller must free the returned value.
4038 */
4039
4040xmlChar *
4041xmlValidNormalizeAttributeValue(xmlDocPtr doc, xmlNodePtr elem,
4042			        const xmlChar *name, const xmlChar *value) {
4043    xmlChar *ret, *dst;
4044    const xmlChar *src;
4045    xmlAttributePtr attrDecl = NULL;
4046
4047    if (doc == NULL) return(NULL);
4048    if (elem == NULL) return(NULL);
4049    if (name == NULL) return(NULL);
4050    if (value == NULL) return(NULL);
4051
4052    if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
4053	xmlChar fn[50];
4054	xmlChar *fullname;
4055
4056	fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4057	if (fullname == NULL)
4058	    return(NULL);
4059	if ((fullname != fn) && (fullname != elem->name))
4060	    xmlFree(fullname);
4061    }
4062    attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, name);
4063    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4064	attrDecl = xmlGetDtdAttrDesc(doc->extSubset, elem->name, name);
4065
4066    if (attrDecl == NULL)
4067	return(NULL);
4068    if (attrDecl->atype == XML_ATTRIBUTE_CDATA)
4069	return(NULL);
4070
4071    ret = xmlStrdup(value);
4072    if (ret == NULL)
4073	return(NULL);
4074    src = value;
4075    dst = ret;
4076    while (*src == 0x20) src++;
4077    while (*src != 0) {
4078	if (*src == 0x20) {
4079	    while (*src == 0x20) src++;
4080	    if (*src != 0)
4081		*dst++ = 0x20;
4082	} else {
4083	    *dst++ = *src++;
4084	}
4085    }
4086    *dst = 0;
4087    return(ret);
4088}
4089
4090static void
4091xmlValidateAttributeIdCallback(xmlAttributePtr attr, int *count,
4092	                       const xmlChar* name ATTRIBUTE_UNUSED) {
4093    if (attr->atype == XML_ATTRIBUTE_ID) (*count)++;
4094}
4095
4096/**
4097 * xmlValidateAttributeDecl:
4098 * @ctxt:  the validation context
4099 * @doc:  a document instance
4100 * @attr:  an attribute definition
4101 *
4102 * Try to validate a single attribute definition
4103 * basically it does the following checks as described by the
4104 * XML-1.0 recommendation:
4105 *  - [ VC: Attribute Default Legal ]
4106 *  - [ VC: Enumeration ]
4107 *  - [ VC: ID Attribute Default ]
4108 *
4109 * The ID/IDREF uniqueness and matching are done separately
4110 *
4111 * returns 1 if valid or 0 otherwise
4112 */
4113
4114int
4115xmlValidateAttributeDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4116                         xmlAttributePtr attr) {
4117    int ret = 1;
4118    int val;
4119    CHECK_DTD;
4120    if(attr == NULL) return(1);
4121
4122    /* Attribute Default Legal */
4123    /* Enumeration */
4124    if (attr->defaultValue != NULL) {
4125	val = xmlValidateAttributeValueInternal(doc, attr->atype,
4126	                                        attr->defaultValue);
4127	if (val == 0) {
4128	    xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_DEFAULT,
4129	       "Syntax of default value for attribute %s of %s is not valid\n",
4130	           attr->name, attr->elem, NULL);
4131	}
4132        ret &= val;
4133    }
4134
4135    /* ID Attribute Default */
4136    if ((attr->atype == XML_ATTRIBUTE_ID)&&
4137        (attr->def != XML_ATTRIBUTE_IMPLIED) &&
4138	(attr->def != XML_ATTRIBUTE_REQUIRED)) {
4139	xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_FIXED,
4140          "ID attribute %s of %s is not valid must be #IMPLIED or #REQUIRED\n",
4141	       attr->name, attr->elem, NULL);
4142	ret = 0;
4143    }
4144
4145    /* One ID per Element Type */
4146    if (attr->atype == XML_ATTRIBUTE_ID) {
4147        int nbId;
4148
4149	/* the trick is that we parse DtD as their own internal subset */
4150        xmlElementPtr elem = xmlGetDtdElementDesc(doc->intSubset,
4151	                                          attr->elem);
4152	if (elem != NULL) {
4153	    nbId = xmlScanIDAttributeDecl(NULL, elem, 0);
4154	} else {
4155	    xmlAttributeTablePtr table;
4156
4157	    /*
4158	     * The attribute may be declared in the internal subset and the
4159	     * element in the external subset.
4160	     */
4161	    nbId = 0;
4162	    if (doc->intSubset != NULL) {
4163		table = (xmlAttributeTablePtr) doc->intSubset->attributes;
4164		xmlHashScan3(table, NULL, NULL, attr->elem, (xmlHashScanner)
4165			     xmlValidateAttributeIdCallback, &nbId);
4166	    }
4167	}
4168	if (nbId > 1) {
4169
4170	    xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
4171       "Element %s has %d ID attribute defined in the internal subset : %s\n",
4172		   attr->elem, nbId, attr->name);
4173	} else if (doc->extSubset != NULL) {
4174	    int extId = 0;
4175	    elem = xmlGetDtdElementDesc(doc->extSubset, attr->elem);
4176	    if (elem != NULL) {
4177		extId = xmlScanIDAttributeDecl(NULL, elem, 0);
4178	    }
4179	    if (extId > 1) {
4180		xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
4181       "Element %s has %d ID attribute defined in the external subset : %s\n",
4182		       attr->elem, extId, attr->name);
4183	    } else if (extId + nbId > 1) {
4184		xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
4185"Element %s has ID attributes defined in the internal and external subset : %s\n",
4186		       attr->elem, attr->name, NULL);
4187	    }
4188	}
4189    }
4190
4191    /* Validity Constraint: Enumeration */
4192    if ((attr->defaultValue != NULL) && (attr->tree != NULL)) {
4193        xmlEnumerationPtr tree = attr->tree;
4194	while (tree != NULL) {
4195	    if (xmlStrEqual(tree->name, attr->defaultValue)) break;
4196	    tree = tree->next;
4197	}
4198	if (tree == NULL) {
4199	    xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_VALUE,
4200"Default value \"%s\" for attribute %s of %s is not among the enumerated set\n",
4201		   attr->defaultValue, attr->name, attr->elem);
4202	    ret = 0;
4203	}
4204    }
4205
4206    return(ret);
4207}
4208
4209/**
4210 * xmlValidateElementDecl:
4211 * @ctxt:  the validation context
4212 * @doc:  a document instance
4213 * @elem:  an element definition
4214 *
4215 * Try to validate a single element definition
4216 * basically it does the following checks as described by the
4217 * XML-1.0 recommendation:
4218 *  - [ VC: One ID per Element Type ]
4219 *  - [ VC: No Duplicate Types ]
4220 *  - [ VC: Unique Element Type Declaration ]
4221 *
4222 * returns 1 if valid or 0 otherwise
4223 */
4224
4225int
4226xmlValidateElementDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4227                       xmlElementPtr elem) {
4228    int ret = 1;
4229    xmlElementPtr tst;
4230
4231    CHECK_DTD;
4232
4233    if (elem == NULL) return(1);
4234
4235#if 0
4236#ifdef LIBXML_REGEXP_ENABLED
4237    /* Build the regexp associated to the content model */
4238    ret = xmlValidBuildContentModel(ctxt, elem);
4239#endif
4240#endif
4241
4242    /* No Duplicate Types */
4243    if (elem->etype == XML_ELEMENT_TYPE_MIXED) {
4244	xmlElementContentPtr cur, next;
4245        const xmlChar *name;
4246
4247	cur = elem->content;
4248	while (cur != NULL) {
4249	    if (cur->type != XML_ELEMENT_CONTENT_OR) break;
4250	    if (cur->c1 == NULL) break;
4251	    if (cur->c1->type == XML_ELEMENT_CONTENT_ELEMENT) {
4252		name = cur->c1->name;
4253		next = cur->c2;
4254		while (next != NULL) {
4255		    if (next->type == XML_ELEMENT_CONTENT_ELEMENT) {
4256		        if ((xmlStrEqual(next->name, name)) &&
4257			    (xmlStrEqual(next->prefix, cur->c1->prefix))) {
4258			    if (cur->c1->prefix == NULL) {
4259				xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4260		   "Definition of %s has duplicate references of %s\n",
4261				       elem->name, name, NULL);
4262			    } else {
4263				xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4264		   "Definition of %s has duplicate references of %s:%s\n",
4265				       elem->name, cur->c1->prefix, name);
4266			    }
4267			    ret = 0;
4268			}
4269			break;
4270		    }
4271		    if (next->c1 == NULL) break;
4272		    if (next->c1->type != XML_ELEMENT_CONTENT_ELEMENT) break;
4273		    if ((xmlStrEqual(next->c1->name, name)) &&
4274		        (xmlStrEqual(next->c1->prefix, cur->c1->prefix))) {
4275			if (cur->c1->prefix == NULL) {
4276			    xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4277	       "Definition of %s has duplicate references to %s\n",
4278				   elem->name, name, NULL);
4279			} else {
4280			    xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
4281	       "Definition of %s has duplicate references to %s:%s\n",
4282				   elem->name, cur->c1->prefix, name);
4283			}
4284			ret = 0;
4285		    }
4286		    next = next->c2;
4287		}
4288	    }
4289	    cur = cur->c2;
4290	}
4291    }
4292
4293    /* VC: Unique Element Type Declaration */
4294    tst = xmlGetDtdElementDesc(doc->intSubset, elem->name);
4295    if ((tst != NULL ) && (tst != elem) &&
4296	((tst->prefix == elem->prefix) ||
4297	 (xmlStrEqual(tst->prefix, elem->prefix))) &&
4298	(tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
4299	xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4300	                "Redefinition of element %s\n",
4301		       elem->name, NULL, NULL);
4302	ret = 0;
4303    }
4304    tst = xmlGetDtdElementDesc(doc->extSubset, elem->name);
4305    if ((tst != NULL ) && (tst != elem) &&
4306	((tst->prefix == elem->prefix) ||
4307	 (xmlStrEqual(tst->prefix, elem->prefix))) &&
4308	(tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
4309	xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4310	                "Redefinition of element %s\n",
4311		       elem->name, NULL, NULL);
4312	ret = 0;
4313    }
4314    /* One ID per Element Type
4315     * already done when registering the attribute
4316    if (xmlScanIDAttributeDecl(ctxt, elem) > 1) {
4317	ret = 0;
4318    } */
4319    return(ret);
4320}
4321
4322/**
4323 * xmlValidateOneAttribute:
4324 * @ctxt:  the validation context
4325 * @doc:  a document instance
4326 * @elem:  an element instance
4327 * @attr:  an attribute instance
4328 * @value:  the attribute value (without entities processing)
4329 *
4330 * Try to validate a single attribute for an element
4331 * basically it does the following checks as described by the
4332 * XML-1.0 recommendation:
4333 *  - [ VC: Attribute Value Type ]
4334 *  - [ VC: Fixed Attribute Default ]
4335 *  - [ VC: Entity Name ]
4336 *  - [ VC: Name Token ]
4337 *  - [ VC: ID ]
4338 *  - [ VC: IDREF ]
4339 *  - [ VC: Entity Name ]
4340 *  - [ VC: Notation Attributes ]
4341 *
4342 * The ID/IDREF uniqueness and matching are done separately
4343 *
4344 * returns 1 if valid or 0 otherwise
4345 */
4346
4347int
4348xmlValidateOneAttribute(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4349                        xmlNodePtr elem, xmlAttrPtr attr, const xmlChar *value)
4350{
4351    xmlAttributePtr attrDecl =  NULL;
4352    int val;
4353    int ret = 1;
4354
4355    CHECK_DTD;
4356    if ((elem == NULL) || (elem->name == NULL)) return(0);
4357    if ((attr == NULL) || (attr->name == NULL)) return(0);
4358
4359    if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
4360	xmlChar fn[50];
4361	xmlChar *fullname;
4362
4363	fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4364	if (fullname == NULL)
4365	    return(0);
4366	if (attr->ns != NULL) {
4367	    attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
4368		                          attr->name, attr->ns->prefix);
4369	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4370		attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
4371					      attr->name, attr->ns->prefix);
4372	} else {
4373	    attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, attr->name);
4374	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4375		attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4376					     fullname, attr->name);
4377	}
4378	if ((fullname != fn) && (fullname != elem->name))
4379	    xmlFree(fullname);
4380    }
4381    if (attrDecl == NULL) {
4382	if (attr->ns != NULL) {
4383	    attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, elem->name,
4384		                          attr->name, attr->ns->prefix);
4385	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4386		attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, elem->name,
4387					      attr->name, attr->ns->prefix);
4388	} else {
4389	    attrDecl = xmlGetDtdAttrDesc(doc->intSubset,
4390		                         elem->name, attr->name);
4391	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4392		attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4393					     elem->name, attr->name);
4394	}
4395    }
4396
4397
4398    /* Validity Constraint: Attribute Value Type */
4399    if (attrDecl == NULL) {
4400	xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
4401	       "No declaration for attribute %s of element %s\n",
4402	       attr->name, elem->name, NULL);
4403	return(0);
4404    }
4405    attr->atype = attrDecl->atype;
4406
4407    val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
4408    if (val == 0) {
4409	    xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4410	   "Syntax of value for attribute %s of %s is not valid\n",
4411	       attr->name, elem->name, NULL);
4412        ret = 0;
4413    }
4414
4415    /* Validity constraint: Fixed Attribute Default */
4416    if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4417	if (!xmlStrEqual(value, attrDecl->defaultValue)) {
4418	    xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
4419	   "Value for attribute %s of %s is different from default \"%s\"\n",
4420		   attr->name, elem->name, attrDecl->defaultValue);
4421	    ret = 0;
4422	}
4423    }
4424
4425    /* Validity Constraint: ID uniqueness */
4426    if (attrDecl->atype == XML_ATTRIBUTE_ID) {
4427        if (xmlAddID(ctxt, doc, value, attr) == NULL)
4428	    ret = 0;
4429    }
4430
4431    if ((attrDecl->atype == XML_ATTRIBUTE_IDREF) ||
4432	(attrDecl->atype == XML_ATTRIBUTE_IDREFS)) {
4433        if (xmlAddRef(ctxt, doc, value, attr) == NULL)
4434	    ret = 0;
4435    }
4436
4437    /* Validity Constraint: Notation Attributes */
4438    if (attrDecl->atype == XML_ATTRIBUTE_NOTATION) {
4439        xmlEnumerationPtr tree = attrDecl->tree;
4440        xmlNotationPtr nota;
4441
4442        /* First check that the given NOTATION was declared */
4443	nota = xmlGetDtdNotationDesc(doc->intSubset, value);
4444	if (nota == NULL)
4445	    nota = xmlGetDtdNotationDesc(doc->extSubset, value);
4446
4447	if (nota == NULL) {
4448	    xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
4449       "Value \"%s\" for attribute %s of %s is not a declared Notation\n",
4450		   value, attr->name, elem->name);
4451	    ret = 0;
4452        }
4453
4454	/* Second, verify that it's among the list */
4455	while (tree != NULL) {
4456	    if (xmlStrEqual(tree->name, value)) break;
4457	    tree = tree->next;
4458	}
4459	if (tree == NULL) {
4460	    xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
4461"Value \"%s\" for attribute %s of %s is not among the enumerated notations\n",
4462		   value, attr->name, elem->name);
4463	    ret = 0;
4464	}
4465    }
4466
4467    /* Validity Constraint: Enumeration */
4468    if (attrDecl->atype == XML_ATTRIBUTE_ENUMERATION) {
4469        xmlEnumerationPtr tree = attrDecl->tree;
4470	while (tree != NULL) {
4471	    if (xmlStrEqual(tree->name, value)) break;
4472	    tree = tree->next;
4473	}
4474	if (tree == NULL) {
4475	    xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4476       "Value \"%s\" for attribute %s of %s is not among the enumerated set\n",
4477		   value, attr->name, elem->name);
4478	    ret = 0;
4479	}
4480    }
4481
4482    /* Fixed Attribute Default */
4483    if ((attrDecl->def == XML_ATTRIBUTE_FIXED) &&
4484        (!xmlStrEqual(attrDecl->defaultValue, value))) {
4485	xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4486	   "Value for attribute %s of %s must be \"%s\"\n",
4487	       attr->name, elem->name, attrDecl->defaultValue);
4488        ret = 0;
4489    }
4490
4491    /* Extra check for the attribute value */
4492    ret &= xmlValidateAttributeValue2(ctxt, doc, attr->name,
4493				      attrDecl->atype, value);
4494
4495    return(ret);
4496}
4497
4498/**
4499 * xmlValidateOneNamespace:
4500 * @ctxt:  the validation context
4501 * @doc:  a document instance
4502 * @elem:  an element instance
4503 * @prefix:  the namespace prefix
4504 * @ns:  an namespace declaration instance
4505 * @value:  the attribute value (without entities processing)
4506 *
4507 * Try to validate a single namespace declaration for an element
4508 * basically it does the following checks as described by the
4509 * XML-1.0 recommendation:
4510 *  - [ VC: Attribute Value Type ]
4511 *  - [ VC: Fixed Attribute Default ]
4512 *  - [ VC: Entity Name ]
4513 *  - [ VC: Name Token ]
4514 *  - [ VC: ID ]
4515 *  - [ VC: IDREF ]
4516 *  - [ VC: Entity Name ]
4517 *  - [ VC: Notation Attributes ]
4518 *
4519 * The ID/IDREF uniqueness and matching are done separately
4520 *
4521 * returns 1 if valid or 0 otherwise
4522 */
4523
4524int
4525xmlValidateOneNamespace(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4526xmlNodePtr elem, const xmlChar *prefix, xmlNsPtr ns, const xmlChar *value) {
4527    /* xmlElementPtr elemDecl; */
4528    xmlAttributePtr attrDecl =  NULL;
4529    int val;
4530    int ret = 1;
4531
4532    CHECK_DTD;
4533    if ((elem == NULL) || (elem->name == NULL)) return(0);
4534    if ((ns == NULL) || (ns->href == NULL)) return(0);
4535
4536    if (prefix != NULL) {
4537	xmlChar fn[50];
4538	xmlChar *fullname;
4539
4540	fullname = xmlBuildQName(elem->name, prefix, fn, 50);
4541	if (fullname == NULL) {
4542	    xmlVErrMemory(ctxt, "Validating namespace");
4543	    return(0);
4544	}
4545	if (ns->prefix != NULL) {
4546	    attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
4547		                          ns->prefix, BAD_CAST "xmlns");
4548	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4549		attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
4550					  ns->prefix, BAD_CAST "xmlns");
4551	} else {
4552	    attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname,
4553		                         BAD_CAST "xmlns");
4554	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4555		attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname,
4556			                 BAD_CAST "xmlns");
4557	}
4558	if ((fullname != fn) && (fullname != elem->name))
4559	    xmlFree(fullname);
4560    }
4561    if (attrDecl == NULL) {
4562	if (ns->prefix != NULL) {
4563	    attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, elem->name,
4564		                          ns->prefix, BAD_CAST "xmlns");
4565	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4566		attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, elem->name,
4567					      ns->prefix, BAD_CAST "xmlns");
4568	} else {
4569	    attrDecl = xmlGetDtdAttrDesc(doc->intSubset,
4570		                         elem->name, BAD_CAST "xmlns");
4571	    if ((attrDecl == NULL) && (doc->extSubset != NULL))
4572		attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4573					     elem->name, BAD_CAST "xmlns");
4574	}
4575    }
4576
4577
4578    /* Validity Constraint: Attribute Value Type */
4579    if (attrDecl == NULL) {
4580	if (ns->prefix != NULL) {
4581	    xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
4582		   "No declaration for attribute xmlns:%s of element %s\n",
4583		   ns->prefix, elem->name, NULL);
4584	} else {
4585	    xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
4586		   "No declaration for attribute xmlns of element %s\n",
4587		   elem->name, NULL, NULL);
4588	}
4589	return(0);
4590    }
4591
4592    val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
4593    if (val == 0) {
4594	if (ns->prefix != NULL) {
4595	    xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
4596	       "Syntax of value for attribute xmlns:%s of %s is not valid\n",
4597		   ns->prefix, elem->name, NULL);
4598	} else {
4599	    xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
4600	       "Syntax of value for attribute xmlns of %s is not valid\n",
4601		   elem->name, NULL, NULL);
4602	}
4603        ret = 0;
4604    }
4605
4606    /* Validity constraint: Fixed Attribute Default */
4607    if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4608	if (!xmlStrEqual(value, attrDecl->defaultValue)) {
4609	    if (ns->prefix != NULL) {
4610		xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
4611       "Value for attribute xmlns:%s of %s is different from default \"%s\"\n",
4612		       ns->prefix, elem->name, attrDecl->defaultValue);
4613	    } else {
4614		xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
4615       "Value for attribute xmlns of %s is different from default \"%s\"\n",
4616		       elem->name, attrDecl->defaultValue, NULL);
4617	    }
4618	    ret = 0;
4619	}
4620    }
4621
4622    /* Validity Constraint: ID uniqueness */
4623    if (attrDecl->atype == XML_ATTRIBUTE_ID) {
4624        if (xmlAddID(ctxt, doc, value, (xmlAttrPtr) ns) == NULL)
4625	    ret = 0;
4626    }
4627
4628    if ((attrDecl->atype == XML_ATTRIBUTE_IDREF) ||
4629	(attrDecl->atype == XML_ATTRIBUTE_IDREFS)) {
4630        if (xmlAddRef(ctxt, doc, value, (xmlAttrPtr) ns) == NULL)
4631	    ret = 0;
4632    }
4633
4634    /* Validity Constraint: Notation Attributes */
4635    if (attrDecl->atype == XML_ATTRIBUTE_NOTATION) {
4636        xmlEnumerationPtr tree = attrDecl->tree;
4637        xmlNotationPtr nota;
4638
4639        /* First check that the given NOTATION was declared */
4640	nota = xmlGetDtdNotationDesc(doc->intSubset, value);
4641	if (nota == NULL)
4642	    nota = xmlGetDtdNotationDesc(doc->extSubset, value);
4643
4644	if (nota == NULL) {
4645	    if (ns->prefix != NULL) {
4646		xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
4647       "Value \"%s\" for attribute xmlns:%s of %s is not a declared Notation\n",
4648		       value, ns->prefix, elem->name);
4649	    } else {
4650		xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
4651       "Value \"%s\" for attribute xmlns of %s is not a declared Notation\n",
4652		       value, elem->name, NULL);
4653	    }
4654	    ret = 0;
4655        }
4656
4657	/* Second, verify that it's among the list */
4658	while (tree != NULL) {
4659	    if (xmlStrEqual(tree->name, value)) break;
4660	    tree = tree->next;
4661	}
4662	if (tree == NULL) {
4663	    if (ns->prefix != NULL) {
4664		xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
4665"Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated notations\n",
4666		       value, ns->prefix, elem->name);
4667	    } else {
4668		xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
4669"Value \"%s\" for attribute xmlns of %s is not among the enumerated notations\n",
4670		       value, elem->name, NULL);
4671	    }
4672	    ret = 0;
4673	}
4674    }
4675
4676    /* Validity Constraint: Enumeration */
4677    if (attrDecl->atype == XML_ATTRIBUTE_ENUMERATION) {
4678        xmlEnumerationPtr tree = attrDecl->tree;
4679	while (tree != NULL) {
4680	    if (xmlStrEqual(tree->name, value)) break;
4681	    tree = tree->next;
4682	}
4683	if (tree == NULL) {
4684	    if (ns->prefix != NULL) {
4685		xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4686"Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated set\n",
4687		       value, ns->prefix, elem->name);
4688	    } else {
4689		xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
4690"Value \"%s\" for attribute xmlns of %s is not among the enumerated set\n",
4691		       value, elem->name, NULL);
4692	    }
4693	    ret = 0;
4694	}
4695    }
4696
4697    /* Fixed Attribute Default */
4698    if ((attrDecl->def == XML_ATTRIBUTE_FIXED) &&
4699        (!xmlStrEqual(attrDecl->defaultValue, value))) {
4700	if (ns->prefix != NULL) {
4701	    xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
4702		   "Value for attribute xmlns:%s of %s must be \"%s\"\n",
4703		   ns->prefix, elem->name, attrDecl->defaultValue);
4704	} else {
4705	    xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
4706		   "Value for attribute xmlns of %s must be \"%s\"\n",
4707		   elem->name, attrDecl->defaultValue, NULL);
4708	}
4709        ret = 0;
4710    }
4711
4712    /* Extra check for the attribute value */
4713    if (ns->prefix != NULL) {
4714	ret &= xmlValidateAttributeValue2(ctxt, doc, ns->prefix,
4715					  attrDecl->atype, value);
4716    } else {
4717	ret &= xmlValidateAttributeValue2(ctxt, doc, BAD_CAST "xmlns",
4718					  attrDecl->atype, value);
4719    }
4720
4721    return(ret);
4722}
4723
4724#ifndef  LIBXML_REGEXP_ENABLED
4725/**
4726 * xmlValidateSkipIgnorable:
4727 * @ctxt:  the validation context
4728 * @child:  the child list
4729 *
4730 * Skip ignorable elements w.r.t. the validation process
4731 *
4732 * returns the first element to consider for validation of the content model
4733 */
4734
4735static xmlNodePtr
4736xmlValidateSkipIgnorable(xmlNodePtr child) {
4737    while (child != NULL) {
4738	switch (child->type) {
4739	    /* These things are ignored (skipped) during validation.  */
4740	    case XML_PI_NODE:
4741	    case XML_COMMENT_NODE:
4742	    case XML_XINCLUDE_START:
4743	    case XML_XINCLUDE_END:
4744		child = child->next;
4745		break;
4746	    case XML_TEXT_NODE:
4747		if (xmlIsBlankNode(child))
4748		    child = child->next;
4749		else
4750		    return(child);
4751		break;
4752	    /* keep current node */
4753	    default:
4754		return(child);
4755	}
4756    }
4757    return(child);
4758}
4759
4760/**
4761 * xmlValidateElementType:
4762 * @ctxt:  the validation context
4763 *
4764 * Try to validate the content model of an element internal function
4765 *
4766 * returns 1 if valid or 0 ,-1 in case of error, -2 if an entity
4767 *           reference is found and -3 if the validation succeeded but
4768 *           the content model is not determinist.
4769 */
4770
4771static int
4772xmlValidateElementType(xmlValidCtxtPtr ctxt) {
4773    int ret = -1;
4774    int determinist = 1;
4775
4776    NODE = xmlValidateSkipIgnorable(NODE);
4777    if ((NODE == NULL) && (CONT == NULL))
4778	return(1);
4779    if ((NODE == NULL) &&
4780	((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
4781	 (CONT->ocur == XML_ELEMENT_CONTENT_OPT))) {
4782	return(1);
4783    }
4784    if (CONT == NULL) return(-1);
4785    if ((NODE != NULL) && (NODE->type == XML_ENTITY_REF_NODE))
4786	return(-2);
4787
4788    /*
4789     * We arrive here when more states need to be examined
4790     */
4791cont:
4792
4793    /*
4794     * We just recovered from a rollback generated by a possible
4795     * epsilon transition, go directly to the analysis phase
4796     */
4797    if (STATE == ROLLBACK_PARENT) {
4798	DEBUG_VALID_MSG("restored parent branch");
4799	DEBUG_VALID_STATE(NODE, CONT)
4800	ret = 1;
4801	goto analyze;
4802    }
4803
4804    DEBUG_VALID_STATE(NODE, CONT)
4805    /*
4806     * we may have to save a backup state here. This is the equivalent
4807     * of handling epsilon transition in NFAs.
4808     */
4809    if ((CONT != NULL) &&
4810	((CONT->parent == NULL) ||
4811	 (CONT->parent->type != XML_ELEMENT_CONTENT_OR)) &&
4812	((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
4813	 (CONT->ocur == XML_ELEMENT_CONTENT_OPT) ||
4814	 ((CONT->ocur == XML_ELEMENT_CONTENT_PLUS) && (OCCURRENCE)))) {
4815	DEBUG_VALID_MSG("saving parent branch");
4816	if (vstateVPush(ctxt, CONT, NODE, DEPTH, OCCURS, ROLLBACK_PARENT) < 0)
4817	    return(0);
4818    }
4819
4820
4821    /*
4822     * Check first if the content matches
4823     */
4824    switch (CONT->type) {
4825	case XML_ELEMENT_CONTENT_PCDATA:
4826	    if (NODE == NULL) {
4827		DEBUG_VALID_MSG("pcdata failed no node");
4828		ret = 0;
4829		break;
4830	    }
4831	    if (NODE->type == XML_TEXT_NODE) {
4832		DEBUG_VALID_MSG("pcdata found, skip to next");
4833		/*
4834		 * go to next element in the content model
4835		 * skipping ignorable elems
4836		 */
4837		do {
4838		    NODE = NODE->next;
4839		    NODE = xmlValidateSkipIgnorable(NODE);
4840		    if ((NODE != NULL) &&
4841			(NODE->type == XML_ENTITY_REF_NODE))
4842			return(-2);
4843		} while ((NODE != NULL) &&
4844			 ((NODE->type != XML_ELEMENT_NODE) &&
4845			  (NODE->type != XML_TEXT_NODE) &&
4846			  (NODE->type != XML_CDATA_SECTION_NODE)));
4847                ret = 1;
4848		break;
4849	    } else {
4850		DEBUG_VALID_MSG("pcdata failed");
4851		ret = 0;
4852		break;
4853	    }
4854	    break;
4855	case XML_ELEMENT_CONTENT_ELEMENT:
4856	    if (NODE == NULL) {
4857		DEBUG_VALID_MSG("element failed no node");
4858		ret = 0;
4859		break;
4860	    }
4861	    ret = ((NODE->type == XML_ELEMENT_NODE) &&
4862		   (xmlStrEqual(NODE->name, CONT->name)));
4863	    if (ret == 1) {
4864		if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4865		    ret = (CONT->prefix == NULL);
4866		} else if (CONT->prefix == NULL) {
4867		    ret = 0;
4868		} else {
4869		    ret = xmlStrEqual(NODE->ns->prefix, CONT->prefix);
4870		}
4871	    }
4872	    if (ret == 1) {
4873		DEBUG_VALID_MSG("element found, skip to next");
4874		/*
4875		 * go to next element in the content model
4876		 * skipping ignorable elems
4877		 */
4878		do {
4879		    NODE = NODE->next;
4880		    NODE = xmlValidateSkipIgnorable(NODE);
4881		    if ((NODE != NULL) &&
4882			(NODE->type == XML_ENTITY_REF_NODE))
4883			return(-2);
4884		} while ((NODE != NULL) &&
4885			 ((NODE->type != XML_ELEMENT_NODE) &&
4886			  (NODE->type != XML_TEXT_NODE) &&
4887			  (NODE->type != XML_CDATA_SECTION_NODE)));
4888	    } else {
4889		DEBUG_VALID_MSG("element failed");
4890		ret = 0;
4891		break;
4892	    }
4893	    break;
4894	case XML_ELEMENT_CONTENT_OR:
4895	    /*
4896	     * Small optimization.
4897	     */
4898	    if (CONT->c1->type == XML_ELEMENT_CONTENT_ELEMENT) {
4899		if ((NODE == NULL) ||
4900		    (!xmlStrEqual(NODE->name, CONT->c1->name))) {
4901		    DEPTH++;
4902		    CONT = CONT->c2;
4903		    goto cont;
4904		}
4905		if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4906		    ret = (CONT->c1->prefix == NULL);
4907		} else if (CONT->c1->prefix == NULL) {
4908		    ret = 0;
4909		} else {
4910		    ret = xmlStrEqual(NODE->ns->prefix, CONT->c1->prefix);
4911		}
4912		if (ret == 0) {
4913		    DEPTH++;
4914		    CONT = CONT->c2;
4915		    goto cont;
4916		}
4917	    }
4918
4919	    /*
4920	     * save the second branch 'or' branch
4921	     */
4922	    DEBUG_VALID_MSG("saving 'or' branch");
4923	    if (vstateVPush(ctxt, CONT->c2, NODE, (unsigned char)(DEPTH + 1),
4924			    OCCURS, ROLLBACK_OR) < 0)
4925		return(-1);
4926	    DEPTH++;
4927	    CONT = CONT->c1;
4928	    goto cont;
4929	case XML_ELEMENT_CONTENT_SEQ:
4930	    /*
4931	     * Small optimization.
4932	     */
4933	    if ((CONT->c1->type == XML_ELEMENT_CONTENT_ELEMENT) &&
4934		((CONT->c1->ocur == XML_ELEMENT_CONTENT_OPT) ||
4935		 (CONT->c1->ocur == XML_ELEMENT_CONTENT_MULT))) {
4936		if ((NODE == NULL) ||
4937		    (!xmlStrEqual(NODE->name, CONT->c1->name))) {
4938		    DEPTH++;
4939		    CONT = CONT->c2;
4940		    goto cont;
4941		}
4942		if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4943		    ret = (CONT->c1->prefix == NULL);
4944		} else if (CONT->c1->prefix == NULL) {
4945		    ret = 0;
4946		} else {
4947		    ret = xmlStrEqual(NODE->ns->prefix, CONT->c1->prefix);
4948		}
4949		if (ret == 0) {
4950		    DEPTH++;
4951		    CONT = CONT->c2;
4952		    goto cont;
4953		}
4954	    }
4955	    DEPTH++;
4956	    CONT = CONT->c1;
4957	    goto cont;
4958    }
4959
4960    /*
4961     * At this point handle going up in the tree
4962     */
4963    if (ret == -1) {
4964	DEBUG_VALID_MSG("error found returning");
4965	return(ret);
4966    }
4967analyze:
4968    while (CONT != NULL) {
4969	/*
4970	 * First do the analysis depending on the occurrence model at
4971	 * this level.
4972	 */
4973	if (ret == 0) {
4974	    switch (CONT->ocur) {
4975		xmlNodePtr cur;
4976
4977		case XML_ELEMENT_CONTENT_ONCE:
4978		    cur = ctxt->vstate->node;
4979		    DEBUG_VALID_MSG("Once branch failed, rollback");
4980		    if (vstateVPop(ctxt) < 0 ) {
4981			DEBUG_VALID_MSG("exhaustion, failed");
4982			return(0);
4983		    }
4984		    if (cur != ctxt->vstate->node)
4985			determinist = -3;
4986		    goto cont;
4987		case XML_ELEMENT_CONTENT_PLUS:
4988		    if (OCCURRENCE == 0) {
4989			cur = ctxt->vstate->node;
4990			DEBUG_VALID_MSG("Plus branch failed, rollback");
4991			if (vstateVPop(ctxt) < 0 ) {
4992			    DEBUG_VALID_MSG("exhaustion, failed");
4993			    return(0);
4994			}
4995			if (cur != ctxt->vstate->node)
4996			    determinist = -3;
4997			goto cont;
4998		    }
4999		    DEBUG_VALID_MSG("Plus branch found");
5000		    ret = 1;
5001		    break;
5002		case XML_ELEMENT_CONTENT_MULT:
5003#ifdef DEBUG_VALID_ALGO
5004		    if (OCCURRENCE == 0) {
5005			DEBUG_VALID_MSG("Mult branch failed");
5006		    } else {
5007			DEBUG_VALID_MSG("Mult branch found");
5008		    }
5009#endif
5010		    ret = 1;
5011		    break;
5012		case XML_ELEMENT_CONTENT_OPT:
5013		    DEBUG_VALID_MSG("Option branch failed");
5014		    ret = 1;
5015		    break;
5016	    }
5017	} else {
5018	    switch (CONT->ocur) {
5019		case XML_ELEMENT_CONTENT_OPT:
5020		    DEBUG_VALID_MSG("Option branch succeeded");
5021		    ret = 1;
5022		    break;
5023		case XML_ELEMENT_CONTENT_ONCE:
5024		    DEBUG_VALID_MSG("Once branch succeeded");
5025		    ret = 1;
5026		    break;
5027		case XML_ELEMENT_CONTENT_PLUS:
5028		    if (STATE == ROLLBACK_PARENT) {
5029			DEBUG_VALID_MSG("Plus branch rollback");
5030			ret = 1;
5031			break;
5032		    }
5033		    if (NODE == NULL) {
5034			DEBUG_VALID_MSG("Plus branch exhausted");
5035			ret = 1;
5036			break;
5037		    }
5038		    DEBUG_VALID_MSG("Plus branch succeeded, continuing");
5039		    SET_OCCURRENCE;
5040		    goto cont;
5041		case XML_ELEMENT_CONTENT_MULT:
5042		    if (STATE == ROLLBACK_PARENT) {
5043			DEBUG_VALID_MSG("Mult branch rollback");
5044			ret = 1;
5045			break;
5046		    }
5047		    if (NODE == NULL) {
5048			DEBUG_VALID_MSG("Mult branch exhausted");
5049			ret = 1;
5050			break;
5051		    }
5052		    DEBUG_VALID_MSG("Mult branch succeeded, continuing");
5053		    /* SET_OCCURRENCE; */
5054		    goto cont;
5055	    }
5056	}
5057	STATE = 0;
5058
5059	/*
5060	 * Then act accordingly at the parent level
5061	 */
5062	RESET_OCCURRENCE;
5063	if (CONT->parent == NULL)
5064	    break;
5065
5066	switch (CONT->parent->type) {
5067	    case XML_ELEMENT_CONTENT_PCDATA:
5068		DEBUG_VALID_MSG("Error: parent pcdata");
5069		return(-1);
5070	    case XML_ELEMENT_CONTENT_ELEMENT:
5071		DEBUG_VALID_MSG("Error: parent element");
5072		return(-1);
5073	    case XML_ELEMENT_CONTENT_OR:
5074		if (ret == 1) {
5075		    DEBUG_VALID_MSG("Or succeeded");
5076		    CONT = CONT->parent;
5077		    DEPTH--;
5078		} else {
5079		    DEBUG_VALID_MSG("Or failed");
5080		    CONT = CONT->parent;
5081		    DEPTH--;
5082		}
5083		break;
5084	    case XML_ELEMENT_CONTENT_SEQ:
5085		if (ret == 0) {
5086		    DEBUG_VALID_MSG("Sequence failed");
5087		    CONT = CONT->parent;
5088		    DEPTH--;
5089		} else if (CONT == CONT->parent->c1) {
5090		    DEBUG_VALID_MSG("Sequence testing 2nd branch");
5091		    CONT = CONT->parent->c2;
5092		    goto cont;
5093		} else {
5094		    DEBUG_VALID_MSG("Sequence succeeded");
5095		    CONT = CONT->parent;
5096		    DEPTH--;
5097		}
5098	}
5099    }
5100    if (NODE != NULL) {
5101	xmlNodePtr cur;
5102
5103	cur = ctxt->vstate->node;
5104	DEBUG_VALID_MSG("Failed, remaining input, rollback");
5105	if (vstateVPop(ctxt) < 0 ) {
5106	    DEBUG_VALID_MSG("exhaustion, failed");
5107	    return(0);
5108	}
5109	if (cur != ctxt->vstate->node)
5110	    determinist = -3;
5111	goto cont;
5112    }
5113    if (ret == 0) {
5114	xmlNodePtr cur;
5115
5116	cur = ctxt->vstate->node;
5117	DEBUG_VALID_MSG("Failure, rollback");
5118	if (vstateVPop(ctxt) < 0 ) {
5119	    DEBUG_VALID_MSG("exhaustion, failed");
5120	    return(0);
5121	}
5122	if (cur != ctxt->vstate->node)
5123	    determinist = -3;
5124	goto cont;
5125    }
5126    return(determinist);
5127}
5128#endif
5129
5130/**
5131 * xmlSnprintfElements:
5132 * @buf:  an output buffer
5133 * @size:  the size of the buffer
5134 * @content:  An element
5135 * @glob: 1 if one must print the englobing parenthesis, 0 otherwise
5136 *
5137 * This will dump the list of elements to the buffer
5138 * Intended just for the debug routine
5139 */
5140static void
5141xmlSnprintfElements(char *buf, int size, xmlNodePtr node, int glob) {
5142    xmlNodePtr cur;
5143    int len;
5144
5145    if (node == NULL) return;
5146    if (glob) strcat(buf, "(");
5147    cur = node;
5148    while (cur != NULL) {
5149	len = strlen(buf);
5150	if (size - len < 50) {
5151	    if ((size - len > 4) && (buf[len - 1] != '.'))
5152		strcat(buf, " ...");
5153	    return;
5154	}
5155        switch (cur->type) {
5156            case XML_ELEMENT_NODE:
5157		if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
5158		    if (size - len < xmlStrlen(cur->ns->prefix) + 10) {
5159			if ((size - len > 4) && (buf[len - 1] != '.'))
5160			    strcat(buf, " ...");
5161			return;
5162		    }
5163		    strcat(buf, (char *) cur->ns->prefix);
5164		    strcat(buf, ":");
5165		}
5166                if (size - len < xmlStrlen(cur->name) + 10) {
5167		    if ((size - len > 4) && (buf[len - 1] != '.'))
5168			strcat(buf, " ...");
5169		    return;
5170		}
5171	        strcat(buf, (char *) cur->name);
5172		if (cur->next != NULL)
5173		    strcat(buf, " ");
5174		break;
5175            case XML_TEXT_NODE:
5176		if (xmlIsBlankNode(cur))
5177		    break;
5178            case XML_CDATA_SECTION_NODE:
5179            case XML_ENTITY_REF_NODE:
5180	        strcat(buf, "CDATA");
5181		if (cur->next != NULL)
5182		    strcat(buf, " ");
5183		break;
5184            case XML_ATTRIBUTE_NODE:
5185            case XML_DOCUMENT_NODE:
5186#ifdef LIBXML_DOCB_ENABLED
5187	    case XML_DOCB_DOCUMENT_NODE:
5188#endif
5189	    case XML_HTML_DOCUMENT_NODE:
5190            case XML_DOCUMENT_TYPE_NODE:
5191            case XML_DOCUMENT_FRAG_NODE:
5192            case XML_NOTATION_NODE:
5193	    case XML_NAMESPACE_DECL:
5194	        strcat(buf, "???");
5195		if (cur->next != NULL)
5196		    strcat(buf, " ");
5197		break;
5198            case XML_ENTITY_NODE:
5199            case XML_PI_NODE:
5200            case XML_DTD_NODE:
5201            case XML_COMMENT_NODE:
5202	    case XML_ELEMENT_DECL:
5203	    case XML_ATTRIBUTE_DECL:
5204	    case XML_ENTITY_DECL:
5205	    case XML_XINCLUDE_START:
5206	    case XML_XINCLUDE_END:
5207		break;
5208	}
5209	cur = cur->next;
5210    }
5211    if (glob) strcat(buf, ")");
5212}
5213
5214/**
5215 * xmlValidateElementContent:
5216 * @ctxt:  the validation context
5217 * @child:  the child list
5218 * @elemDecl:  pointer to the element declaration
5219 * @warn:  emit the error message
5220 * @parent: the parent element (for error reporting)
5221 *
5222 * Try to validate the content model of an element
5223 *
5224 * returns 1 if valid or 0 if not and -1 in case of error
5225 */
5226
5227static int
5228xmlValidateElementContent(xmlValidCtxtPtr ctxt, xmlNodePtr child,
5229       xmlElementPtr elemDecl, int warn, xmlNodePtr parent) {
5230    int ret = 1;
5231#ifndef  LIBXML_REGEXP_ENABLED
5232    xmlNodePtr repl = NULL, last = NULL, tmp;
5233#endif
5234    xmlNodePtr cur;
5235    xmlElementContentPtr cont;
5236    const xmlChar *name;
5237
5238    if ((elemDecl == NULL) || (parent == NULL) || (ctxt == NULL))
5239	return(-1);
5240    cont = elemDecl->content;
5241    name = elemDecl->name;
5242
5243#ifdef LIBXML_REGEXP_ENABLED
5244    /* Build the regexp associated to the content model */
5245    if (elemDecl->contModel == NULL)
5246	ret = xmlValidBuildContentModel(ctxt, elemDecl);
5247    if (elemDecl->contModel == NULL) {
5248	return(-1);
5249    } else {
5250	xmlRegExecCtxtPtr exec;
5251
5252	if (!xmlRegexpIsDeterminist(elemDecl->contModel)) {
5253	    return(-1);
5254	}
5255	ctxt->nodeMax = 0;
5256	ctxt->nodeNr = 0;
5257	ctxt->nodeTab = NULL;
5258	exec = xmlRegNewExecCtxt(elemDecl->contModel, NULL, NULL);
5259	if (exec != NULL) {
5260	    cur = child;
5261	    while (cur != NULL) {
5262		switch (cur->type) {
5263		    case XML_ENTITY_REF_NODE:
5264			/*
5265			 * Push the current node to be able to roll back
5266			 * and process within the entity
5267			 */
5268			if ((cur->children != NULL) &&
5269			    (cur->children->children != NULL)) {
5270			    nodeVPush(ctxt, cur);
5271			    cur = cur->children->children;
5272			    continue;
5273			}
5274			break;
5275		    case XML_TEXT_NODE:
5276			if (xmlIsBlankNode(cur))
5277			    break;
5278			ret = 0;
5279			goto fail;
5280		    case XML_CDATA_SECTION_NODE:
5281			/* TODO */
5282			ret = 0;
5283			goto fail;
5284		    case XML_ELEMENT_NODE:
5285			if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
5286			    xmlChar fn[50];
5287			    xmlChar *fullname;
5288
5289			    fullname = xmlBuildQName(cur->name,
5290				                     cur->ns->prefix, fn, 50);
5291			    if (fullname == NULL) {
5292				ret = -1;
5293				goto fail;
5294			    }
5295                            ret = xmlRegExecPushString(exec, fullname, NULL);
5296			    if ((fullname != fn) && (fullname != cur->name))
5297				xmlFree(fullname);
5298			} else {
5299			    ret = xmlRegExecPushString(exec, cur->name, NULL);
5300			}
5301			break;
5302		    default:
5303			break;
5304		}
5305		/*
5306		 * Switch to next element
5307		 */
5308		cur = cur->next;
5309		while (cur == NULL) {
5310		    cur = nodeVPop(ctxt);
5311		    if (cur == NULL)
5312			break;
5313		    cur = cur->next;
5314		}
5315	    }
5316	    ret = xmlRegExecPushString(exec, NULL, NULL);
5317fail:
5318	    xmlRegFreeExecCtxt(exec);
5319	}
5320    }
5321#else  /* LIBXML_REGEXP_ENABLED */
5322    /*
5323     * Allocate the stack
5324     */
5325    ctxt->vstateMax = 8;
5326    ctxt->vstateTab = (xmlValidState *) xmlMalloc(
5327		 ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
5328    if (ctxt->vstateTab == NULL) {
5329	xmlVErrMemory(ctxt, "malloc failed");
5330	return(-1);
5331    }
5332    /*
5333     * The first entry in the stack is reserved to the current state
5334     */
5335    ctxt->nodeMax = 0;
5336    ctxt->nodeNr = 0;
5337    ctxt->nodeTab = NULL;
5338    ctxt->vstate = &ctxt->vstateTab[0];
5339    ctxt->vstateNr = 1;
5340    CONT = cont;
5341    NODE = child;
5342    DEPTH = 0;
5343    OCCURS = 0;
5344    STATE = 0;
5345    ret = xmlValidateElementType(ctxt);
5346    if ((ret == -3) && (warn)) {
5347	xmlErrValidWarning(ctxt, child, XML_DTD_CONTENT_NOT_DETERMINIST,
5348	       "Content model for Element %s is ambiguous\n",
5349	                   name, NULL, NULL);
5350    } else if (ret == -2) {
5351	/*
5352	 * An entities reference appeared at this level.
5353	 * Buid a minimal representation of this node content
5354	 * sufficient to run the validation process on it
5355	 */
5356	DEBUG_VALID_MSG("Found an entity reference, linearizing");
5357	cur = child;
5358	while (cur != NULL) {
5359	    switch (cur->type) {
5360		case XML_ENTITY_REF_NODE:
5361		    /*
5362		     * Push the current node to be able to roll back
5363		     * and process within the entity
5364		     */
5365		    if ((cur->children != NULL) &&
5366			(cur->children->children != NULL)) {
5367			nodeVPush(ctxt, cur);
5368			cur = cur->children->children;
5369			continue;
5370		    }
5371		    break;
5372		case XML_TEXT_NODE:
5373		    if (xmlIsBlankNode(cur))
5374			break;
5375		    /* no break on purpose */
5376		case XML_CDATA_SECTION_NODE:
5377		    /* no break on purpose */
5378		case XML_ELEMENT_NODE:
5379		    /*
5380		     * Allocate a new node and minimally fills in
5381		     * what's required
5382		     */
5383		    tmp = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
5384		    if (tmp == NULL) {
5385			xmlVErrMemory(ctxt, "malloc failed");
5386			xmlFreeNodeList(repl);
5387			ret = -1;
5388			goto done;
5389		    }
5390		    tmp->type = cur->type;
5391		    tmp->name = cur->name;
5392		    tmp->ns = cur->ns;
5393		    tmp->next = NULL;
5394		    tmp->content = NULL;
5395		    if (repl == NULL)
5396			repl = last = tmp;
5397		    else {
5398			last->next = tmp;
5399			last = tmp;
5400		    }
5401		    if (cur->type == XML_CDATA_SECTION_NODE) {
5402			/*
5403			 * E59 spaces in CDATA does not match the
5404			 * nonterminal S
5405			 */
5406			tmp->content = xmlStrdup(BAD_CAST "CDATA");
5407		    }
5408		    break;
5409		default:
5410		    break;
5411	    }
5412	    /*
5413	     * Switch to next element
5414	     */
5415	    cur = cur->next;
5416	    while (cur == NULL) {
5417		cur = nodeVPop(ctxt);
5418		if (cur == NULL)
5419		    break;
5420		cur = cur->next;
5421	    }
5422	}
5423
5424	/*
5425	 * Relaunch the validation
5426	 */
5427	ctxt->vstate = &ctxt->vstateTab[0];
5428	ctxt->vstateNr = 1;
5429	CONT = cont;
5430	NODE = repl;
5431	DEPTH = 0;
5432	OCCURS = 0;
5433	STATE = 0;
5434	ret = xmlValidateElementType(ctxt);
5435    }
5436#endif /* LIBXML_REGEXP_ENABLED */
5437    if ((warn) && ((ret != 1) && (ret != -3))) {
5438	if (ctxt != NULL) {
5439	    char expr[5000];
5440	    char list[5000];
5441
5442	    expr[0] = 0;
5443	    xmlSnprintfElementContent(&expr[0], 5000, cont, 1);
5444	    list[0] = 0;
5445#ifndef LIBXML_REGEXP_ENABLED
5446	    if (repl != NULL)
5447		xmlSnprintfElements(&list[0], 5000, repl, 1);
5448	    else
5449#endif /* LIBXML_REGEXP_ENABLED */
5450		xmlSnprintfElements(&list[0], 5000, child, 1);
5451
5452	    if (name != NULL) {
5453		xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5454	   "Element %s content does not follow the DTD, expecting %s, got %s\n",
5455		       name, BAD_CAST expr, BAD_CAST list);
5456	    } else {
5457		xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5458	   "Element content does not follow the DTD, expecting %s, got %s\n",
5459		       BAD_CAST expr, BAD_CAST list, NULL);
5460	    }
5461	} else {
5462	    if (name != NULL) {
5463		xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5464		       "Element %s content does not follow the DTD\n",
5465		       name, NULL, NULL);
5466	    } else {
5467		xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5468		       "Element content does not follow the DTD\n",
5469		                NULL, NULL, NULL);
5470	    }
5471	}
5472	ret = 0;
5473    }
5474    if (ret == -3)
5475	ret = 1;
5476
5477#ifndef  LIBXML_REGEXP_ENABLED
5478done:
5479    /*
5480     * Deallocate the copy if done, and free up the validation stack
5481     */
5482    while (repl != NULL) {
5483	tmp = repl->next;
5484	xmlFree(repl);
5485	repl = tmp;
5486    }
5487    ctxt->vstateMax = 0;
5488    if (ctxt->vstateTab != NULL) {
5489	xmlFree(ctxt->vstateTab);
5490	ctxt->vstateTab = NULL;
5491    }
5492#endif
5493    ctxt->nodeMax = 0;
5494    ctxt->nodeNr = 0;
5495    if (ctxt->nodeTab != NULL) {
5496	xmlFree(ctxt->nodeTab);
5497	ctxt->nodeTab = NULL;
5498    }
5499    return(ret);
5500
5501}
5502
5503/**
5504 * xmlValidateCdataElement:
5505 * @ctxt:  the validation context
5506 * @doc:  a document instance
5507 * @elem:  an element instance
5508 *
5509 * Check that an element follows #CDATA
5510 *
5511 * returns 1 if valid or 0 otherwise
5512 */
5513static int
5514xmlValidateOneCdataElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5515                           xmlNodePtr elem) {
5516    int ret = 1;
5517    xmlNodePtr cur, child;
5518
5519    if ((ctxt == NULL) || (doc == NULL) || (elem == NULL) ||
5520        (elem->type != XML_ELEMENT_NODE))
5521	return(0);
5522
5523    child = elem->children;
5524
5525    cur = child;
5526    while (cur != NULL) {
5527	switch (cur->type) {
5528	    case XML_ENTITY_REF_NODE:
5529		/*
5530		 * Push the current node to be able to roll back
5531		 * and process within the entity
5532		 */
5533		if ((cur->children != NULL) &&
5534		    (cur->children->children != NULL)) {
5535		    nodeVPush(ctxt, cur);
5536		    cur = cur->children->children;
5537		    continue;
5538		}
5539		break;
5540	    case XML_COMMENT_NODE:
5541	    case XML_PI_NODE:
5542	    case XML_TEXT_NODE:
5543	    case XML_CDATA_SECTION_NODE:
5544		break;
5545	    default:
5546		ret = 0;
5547		goto done;
5548	}
5549	/*
5550	 * Switch to next element
5551	 */
5552	cur = cur->next;
5553	while (cur == NULL) {
5554	    cur = nodeVPop(ctxt);
5555	    if (cur == NULL)
5556		break;
5557	    cur = cur->next;
5558	}
5559    }
5560done:
5561    ctxt->nodeMax = 0;
5562    ctxt->nodeNr = 0;
5563    if (ctxt->nodeTab != NULL) {
5564	xmlFree(ctxt->nodeTab);
5565	ctxt->nodeTab = NULL;
5566    }
5567    return(ret);
5568}
5569
5570/**
5571 * xmlValidateCheckMixed:
5572 * @ctxt:  the validation context
5573 * @cont:  the mixed content model
5574 * @qname:  the qualified name as appearing in the serialization
5575 *
5576 * Check if the given node is part of the content model.
5577 *
5578 * Returns 1 if yes, 0 if no, -1 in case of error
5579 */
5580static int
5581xmlValidateCheckMixed(xmlValidCtxtPtr ctxt,
5582	              xmlElementContentPtr cont, const xmlChar *qname) {
5583    const xmlChar *name;
5584    int plen;
5585    name = xmlSplitQName3(qname, &plen);
5586
5587    if (name == NULL) {
5588	while (cont != NULL) {
5589	    if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
5590		if ((cont->prefix == NULL) && (xmlStrEqual(cont->name, qname)))
5591		    return(1);
5592	    } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
5593	       (cont->c1 != NULL) &&
5594	       (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
5595		if ((cont->c1->prefix == NULL) &&
5596		    (xmlStrEqual(cont->c1->name, qname)))
5597		    return(1);
5598	    } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
5599		(cont->c1 == NULL) ||
5600		(cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
5601		xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
5602			"Internal: MIXED struct corrupted\n",
5603			NULL);
5604		break;
5605	    }
5606	    cont = cont->c2;
5607	}
5608    } else {
5609	while (cont != NULL) {
5610	    if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
5611		if ((cont->prefix != NULL) &&
5612		    (xmlStrncmp(cont->prefix, qname, plen) == 0) &&
5613		    (xmlStrEqual(cont->name, name)))
5614		    return(1);
5615	    } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
5616	       (cont->c1 != NULL) &&
5617	       (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
5618		if ((cont->c1->prefix != NULL) &&
5619		    (xmlStrncmp(cont->c1->prefix, qname, plen) == 0) &&
5620		    (xmlStrEqual(cont->c1->name, name)))
5621		    return(1);
5622	    } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
5623		(cont->c1 == NULL) ||
5624		(cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
5625		xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
5626			"Internal: MIXED struct corrupted\n",
5627			NULL);
5628		break;
5629	    }
5630	    cont = cont->c2;
5631	}
5632    }
5633    return(0);
5634}
5635
5636/**
5637 * xmlValidGetElemDecl:
5638 * @ctxt:  the validation context
5639 * @doc:  a document instance
5640 * @elem:  an element instance
5641 * @extsubset:  pointer, (out) indicate if the declaration was found
5642 *              in the external subset.
5643 *
5644 * Finds a declaration associated to an element in the document.
5645 *
5646 * returns the pointer to the declaration or NULL if not found.
5647 */
5648static xmlElementPtr
5649xmlValidGetElemDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5650	            xmlNodePtr elem, int *extsubset) {
5651    xmlElementPtr elemDecl = NULL;
5652    const xmlChar *prefix = NULL;
5653
5654    if ((ctxt == NULL) || (doc == NULL) ||
5655        (elem == NULL) || (elem->name == NULL))
5656        return(NULL);
5657    if (extsubset != NULL)
5658	*extsubset = 0;
5659
5660    /*
5661     * Fetch the declaration for the qualified name
5662     */
5663    if ((elem->ns != NULL) && (elem->ns->prefix != NULL))
5664	prefix = elem->ns->prefix;
5665
5666    if (prefix != NULL) {
5667	elemDecl = xmlGetDtdQElementDesc(doc->intSubset,
5668		                         elem->name, prefix);
5669	if ((elemDecl == NULL) && (doc->extSubset != NULL)) {
5670	    elemDecl = xmlGetDtdQElementDesc(doc->extSubset,
5671		                             elem->name, prefix);
5672	    if ((elemDecl != NULL) && (extsubset != NULL))
5673		*extsubset = 1;
5674	}
5675    }
5676
5677    /*
5678     * Fetch the declaration for the non qualified name
5679     * This is "non-strict" validation should be done on the
5680     * full QName but in that case being flexible makes sense.
5681     */
5682    if (elemDecl == NULL) {
5683	elemDecl = xmlGetDtdElementDesc(doc->intSubset, elem->name);
5684	if ((elemDecl == NULL) && (doc->extSubset != NULL)) {
5685	    elemDecl = xmlGetDtdElementDesc(doc->extSubset, elem->name);
5686	    if ((elemDecl != NULL) && (extsubset != NULL))
5687		*extsubset = 1;
5688	}
5689    }
5690    if (elemDecl == NULL) {
5691	xmlErrValidNode(ctxt, elem,
5692			XML_DTD_UNKNOWN_ELEM,
5693	       "No declaration for element %s\n",
5694	       elem->name, NULL, NULL);
5695    }
5696    return(elemDecl);
5697}
5698
5699#ifdef LIBXML_REGEXP_ENABLED
5700/**
5701 * xmlValidatePushElement:
5702 * @ctxt:  the validation context
5703 * @doc:  a document instance
5704 * @elem:  an element instance
5705 * @qname:  the qualified name as appearing in the serialization
5706 *
5707 * Push a new element start on the validation stack.
5708 *
5709 * returns 1 if no validation problem was found or 0 otherwise
5710 */
5711int
5712xmlValidatePushElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5713                       xmlNodePtr elem, const xmlChar *qname) {
5714    int ret = 1;
5715    xmlElementPtr eDecl;
5716    int extsubset = 0;
5717
5718    if (ctxt == NULL)
5719        return(0);
5720/* printf("PushElem %s\n", qname); */
5721    if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5722	xmlValidStatePtr state = ctxt->vstate;
5723	xmlElementPtr elemDecl;
5724
5725	/*
5726	 * Check the new element agaisnt the content model of the new elem.
5727	 */
5728	if (state->elemDecl != NULL) {
5729	    elemDecl = state->elemDecl;
5730
5731	    switch(elemDecl->etype) {
5732		case XML_ELEMENT_TYPE_UNDEFINED:
5733		    ret = 0;
5734		    break;
5735		case XML_ELEMENT_TYPE_EMPTY:
5736		    xmlErrValidNode(ctxt, state->node,
5737				    XML_DTD_NOT_EMPTY,
5738	       "Element %s was declared EMPTY this one has content\n",
5739			   state->node->name, NULL, NULL);
5740		    ret = 0;
5741		    break;
5742		case XML_ELEMENT_TYPE_ANY:
5743		    /* I don't think anything is required then */
5744		    break;
5745		case XML_ELEMENT_TYPE_MIXED:
5746		    /* simple case of declared as #PCDATA */
5747		    if ((elemDecl->content != NULL) &&
5748			(elemDecl->content->type ==
5749			 XML_ELEMENT_CONTENT_PCDATA)) {
5750			xmlErrValidNode(ctxt, state->node,
5751					XML_DTD_NOT_PCDATA,
5752	       "Element %s was declared #PCDATA but contains non text nodes\n",
5753				state->node->name, NULL, NULL);
5754			ret = 0;
5755		    } else {
5756			ret = xmlValidateCheckMixed(ctxt, elemDecl->content,
5757				                    qname);
5758			if (ret != 1) {
5759			    xmlErrValidNode(ctxt, state->node,
5760					    XML_DTD_INVALID_CHILD,
5761	       "Element %s is not declared in %s list of possible children\n",
5762				    qname, state->node->name, NULL);
5763			}
5764		    }
5765		    break;
5766		case XML_ELEMENT_TYPE_ELEMENT:
5767		    /*
5768		     * TODO:
5769		     * VC: Standalone Document Declaration
5770		     *     - element types with element content, if white space
5771		     *       occurs directly within any instance of those types.
5772		     */
5773		    if (state->exec != NULL) {
5774			ret = xmlRegExecPushString(state->exec, qname, NULL);
5775			if (ret < 0) {
5776			    xmlErrValidNode(ctxt, state->node,
5777					    XML_DTD_CONTENT_MODEL,
5778	       "Element %s content does not follow the DTD, Misplaced %s\n",
5779				   state->node->name, qname, NULL);
5780			    ret = 0;
5781			} else {
5782			    ret = 1;
5783			}
5784		    }
5785		    break;
5786	    }
5787	}
5788    }
5789    eDecl = xmlValidGetElemDecl(ctxt, doc, elem, &extsubset);
5790    vstateVPush(ctxt, eDecl, elem);
5791    return(ret);
5792}
5793
5794/**
5795 * xmlValidatePushCData:
5796 * @ctxt:  the validation context
5797 * @data:  some character data read
5798 * @len:  the length of the data
5799 *
5800 * check the CData parsed for validation in the current stack
5801 *
5802 * returns 1 if no validation problem was found or 0 otherwise
5803 */
5804int
5805xmlValidatePushCData(xmlValidCtxtPtr ctxt, const xmlChar *data, int len) {
5806    int ret = 1;
5807
5808/* printf("CDATA %s %d\n", data, len); */
5809    if (ctxt == NULL)
5810        return(0);
5811    if (len <= 0)
5812	return(ret);
5813    if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5814	xmlValidStatePtr state = ctxt->vstate;
5815	xmlElementPtr elemDecl;
5816
5817	/*
5818	 * Check the new element agaisnt the content model of the new elem.
5819	 */
5820	if (state->elemDecl != NULL) {
5821	    elemDecl = state->elemDecl;
5822
5823	    switch(elemDecl->etype) {
5824		case XML_ELEMENT_TYPE_UNDEFINED:
5825		    ret = 0;
5826		    break;
5827		case XML_ELEMENT_TYPE_EMPTY:
5828		    xmlErrValidNode(ctxt, state->node,
5829				    XML_DTD_NOT_EMPTY,
5830	       "Element %s was declared EMPTY this one has content\n",
5831			   state->node->name, NULL, NULL);
5832		    ret = 0;
5833		    break;
5834		case XML_ELEMENT_TYPE_ANY:
5835		    break;
5836		case XML_ELEMENT_TYPE_MIXED:
5837		    break;
5838		case XML_ELEMENT_TYPE_ELEMENT:
5839		    if (len > 0) {
5840			int i;
5841
5842			for (i = 0;i < len;i++) {
5843			    if (!IS_BLANK_CH(data[i])) {
5844				xmlErrValidNode(ctxt, state->node,
5845						XML_DTD_CONTENT_MODEL,
5846	   "Element %s content does not follow the DTD, Text not allowed\n",
5847				       state->node->name, NULL, NULL);
5848				ret = 0;
5849				goto done;
5850			    }
5851			}
5852			/*
5853			 * TODO:
5854			 * VC: Standalone Document Declaration
5855			 *  element types with element content, if white space
5856			 *  occurs directly within any instance of those types.
5857			 */
5858		    }
5859		    break;
5860	    }
5861	}
5862    }
5863done:
5864    return(ret);
5865}
5866
5867/**
5868 * xmlValidatePopElement:
5869 * @ctxt:  the validation context
5870 * @doc:  a document instance
5871 * @elem:  an element instance
5872 * @qname:  the qualified name as appearing in the serialization
5873 *
5874 * Pop the element end from the validation stack.
5875 *
5876 * returns 1 if no validation problem was found or 0 otherwise
5877 */
5878int
5879xmlValidatePopElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc ATTRIBUTE_UNUSED,
5880                      xmlNodePtr elem ATTRIBUTE_UNUSED,
5881		      const xmlChar *qname ATTRIBUTE_UNUSED) {
5882    int ret = 1;
5883
5884    if (ctxt == NULL)
5885        return(0);
5886/* printf("PopElem %s\n", qname); */
5887    if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5888	xmlValidStatePtr state = ctxt->vstate;
5889	xmlElementPtr elemDecl;
5890
5891	/*
5892	 * Check the new element agaisnt the content model of the new elem.
5893	 */
5894	if (state->elemDecl != NULL) {
5895	    elemDecl = state->elemDecl;
5896
5897	    if (elemDecl->etype == XML_ELEMENT_TYPE_ELEMENT) {
5898		if (state->exec != NULL) {
5899		    ret = xmlRegExecPushString(state->exec, NULL, NULL);
5900		    if (ret == 0) {
5901			xmlErrValidNode(ctxt, state->node,
5902			                XML_DTD_CONTENT_MODEL,
5903	   "Element %s content does not follow the DTD, Expecting more child\n",
5904			       state->node->name, NULL,NULL);
5905		    } else {
5906			/*
5907			 * previous validation errors should not generate
5908			 * a new one here
5909			 */
5910			ret = 1;
5911		    }
5912		}
5913	    }
5914	}
5915	vstateVPop(ctxt);
5916    }
5917    return(ret);
5918}
5919#endif /* LIBXML_REGEXP_ENABLED */
5920
5921/**
5922 * xmlValidateOneElement:
5923 * @ctxt:  the validation context
5924 * @doc:  a document instance
5925 * @elem:  an element instance
5926 *
5927 * Try to validate a single element and it's attributes,
5928 * basically it does the following checks as described by the
5929 * XML-1.0 recommendation:
5930 *  - [ VC: Element Valid ]
5931 *  - [ VC: Required Attribute ]
5932 * Then call xmlValidateOneAttribute() for each attribute present.
5933 *
5934 * The ID/IDREF checkings are done separately
5935 *
5936 * returns 1 if valid or 0 otherwise
5937 */
5938
5939int
5940xmlValidateOneElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5941                      xmlNodePtr elem) {
5942    xmlElementPtr elemDecl = NULL;
5943    xmlElementContentPtr cont;
5944    xmlAttributePtr attr;
5945    xmlNodePtr child;
5946    int ret = 1, tmp;
5947    const xmlChar *name;
5948    int extsubset = 0;
5949
5950    CHECK_DTD;
5951
5952    if (elem == NULL) return(0);
5953    switch (elem->type) {
5954        case XML_ATTRIBUTE_NODE:
5955	    xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5956		   "Attribute element not expected\n", NULL, NULL ,NULL);
5957	    return(0);
5958        case XML_TEXT_NODE:
5959	    if (elem->children != NULL) {
5960		xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5961		                "Text element has children !\n",
5962				NULL,NULL,NULL);
5963		return(0);
5964	    }
5965	    if (elem->ns != NULL) {
5966		xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5967		                "Text element has namespace !\n",
5968				NULL,NULL,NULL);
5969		return(0);
5970	    }
5971	    if (elem->content == NULL) {
5972		xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5973		                "Text element has no content !\n",
5974				NULL,NULL,NULL);
5975		return(0);
5976	    }
5977	    return(1);
5978        case XML_XINCLUDE_START:
5979        case XML_XINCLUDE_END:
5980            return(1);
5981        case XML_CDATA_SECTION_NODE:
5982        case XML_ENTITY_REF_NODE:
5983        case XML_PI_NODE:
5984        case XML_COMMENT_NODE:
5985	    return(1);
5986        case XML_ENTITY_NODE:
5987	    xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5988		   "Entity element not expected\n", NULL, NULL ,NULL);
5989	    return(0);
5990        case XML_NOTATION_NODE:
5991	    xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5992		   "Notation element not expected\n", NULL, NULL ,NULL);
5993	    return(0);
5994        case XML_DOCUMENT_NODE:
5995        case XML_DOCUMENT_TYPE_NODE:
5996        case XML_DOCUMENT_FRAG_NODE:
5997	    xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5998		   "Document element not expected\n", NULL, NULL ,NULL);
5999	    return(0);
6000        case XML_HTML_DOCUMENT_NODE:
6001	    xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6002		   "HTML Document not expected\n", NULL, NULL ,NULL);
6003	    return(0);
6004        case XML_ELEMENT_NODE:
6005	    break;
6006	default:
6007	    xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6008		   "unknown element type\n", NULL, NULL ,NULL);
6009	    return(0);
6010    }
6011
6012    /*
6013     * Fetch the declaration
6014     */
6015    elemDecl = xmlValidGetElemDecl(ctxt, doc, elem, &extsubset);
6016    if (elemDecl == NULL)
6017	return(0);
6018
6019    /*
6020     * If vstateNr is not zero that means continuous validation is
6021     * activated, do not try to check the content model at that level.
6022     */
6023    if (ctxt->vstateNr == 0) {
6024    /* Check that the element content matches the definition */
6025    switch (elemDecl->etype) {
6026        case XML_ELEMENT_TYPE_UNDEFINED:
6027	    xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ELEM,
6028	                    "No declaration for element %s\n",
6029		   elem->name, NULL, NULL);
6030	    return(0);
6031        case XML_ELEMENT_TYPE_EMPTY:
6032	    if (elem->children != NULL) {
6033		xmlErrValidNode(ctxt, elem, XML_DTD_NOT_EMPTY,
6034	       "Element %s was declared EMPTY this one has content\n",
6035	               elem->name, NULL, NULL);
6036		ret = 0;
6037	    }
6038	    break;
6039        case XML_ELEMENT_TYPE_ANY:
6040	    /* I don't think anything is required then */
6041	    break;
6042        case XML_ELEMENT_TYPE_MIXED:
6043
6044	    /* simple case of declared as #PCDATA */
6045	    if ((elemDecl->content != NULL) &&
6046		(elemDecl->content->type == XML_ELEMENT_CONTENT_PCDATA)) {
6047		ret = xmlValidateOneCdataElement(ctxt, doc, elem);
6048		if (!ret) {
6049		    xmlErrValidNode(ctxt, elem, XML_DTD_NOT_PCDATA,
6050	       "Element %s was declared #PCDATA but contains non text nodes\n",
6051			   elem->name, NULL, NULL);
6052		}
6053		break;
6054	    }
6055	    child = elem->children;
6056	    /* Hum, this start to get messy */
6057	    while (child != NULL) {
6058	        if (child->type == XML_ELEMENT_NODE) {
6059		    name = child->name;
6060		    if ((child->ns != NULL) && (child->ns->prefix != NULL)) {
6061			xmlChar fn[50];
6062			xmlChar *fullname;
6063
6064			fullname = xmlBuildQName(child->name, child->ns->prefix,
6065				                 fn, 50);
6066			if (fullname == NULL)
6067			    return(0);
6068			cont = elemDecl->content;
6069			while (cont != NULL) {
6070			    if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
6071				if (xmlStrEqual(cont->name, fullname))
6072				    break;
6073			    } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
6074			       (cont->c1 != NULL) &&
6075			       (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
6076				if (xmlStrEqual(cont->c1->name, fullname))
6077				    break;
6078			    } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
6079				(cont->c1 == NULL) ||
6080				(cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
6081				xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
6082					"Internal: MIXED struct corrupted\n",
6083					NULL);
6084				break;
6085			    }
6086			    cont = cont->c2;
6087			}
6088			if ((fullname != fn) && (fullname != child->name))
6089			    xmlFree(fullname);
6090			if (cont != NULL)
6091			    goto child_ok;
6092		    }
6093		    cont = elemDecl->content;
6094		    while (cont != NULL) {
6095		        if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
6096			    if (xmlStrEqual(cont->name, name)) break;
6097			} else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
6098			   (cont->c1 != NULL) &&
6099			   (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)) {
6100			    if (xmlStrEqual(cont->c1->name, name)) break;
6101			} else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
6102			    (cont->c1 == NULL) ||
6103			    (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)) {
6104			    xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
6105				    "Internal: MIXED struct corrupted\n",
6106				    NULL);
6107			    break;
6108			}
6109			cont = cont->c2;
6110		    }
6111		    if (cont == NULL) {
6112			xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_CHILD,
6113	       "Element %s is not declared in %s list of possible children\n",
6114			       name, elem->name, NULL);
6115			ret = 0;
6116		    }
6117		}
6118child_ok:
6119	        child = child->next;
6120	    }
6121	    break;
6122        case XML_ELEMENT_TYPE_ELEMENT:
6123	    if ((doc->standalone == 1) && (extsubset == 1)) {
6124		/*
6125		 * VC: Standalone Document Declaration
6126		 *     - element types with element content, if white space
6127		 *       occurs directly within any instance of those types.
6128		 */
6129		child = elem->children;
6130		while (child != NULL) {
6131		    if (child->type == XML_TEXT_NODE) {
6132			const xmlChar *content = child->content;
6133
6134			while (IS_BLANK_CH(*content))
6135			    content++;
6136			if (*content == 0) {
6137			    xmlErrValidNode(ctxt, elem,
6138			                    XML_DTD_STANDALONE_WHITE_SPACE,
6139"standalone: %s declared in the external subset contains white spaces nodes\n",
6140				   elem->name, NULL, NULL);
6141			    ret = 0;
6142			    break;
6143			}
6144		    }
6145		    child =child->next;
6146		}
6147	    }
6148	    child = elem->children;
6149	    cont = elemDecl->content;
6150	    tmp = xmlValidateElementContent(ctxt, child, elemDecl, 1, elem);
6151	    if (tmp <= 0)
6152		ret = tmp;
6153	    break;
6154    }
6155    } /* not continuous */
6156
6157    /* [ VC: Required Attribute ] */
6158    attr = elemDecl->attributes;
6159    while (attr != NULL) {
6160	if (attr->def == XML_ATTRIBUTE_REQUIRED) {
6161	    int qualified = -1;
6162
6163	    if ((attr->prefix == NULL) &&
6164		(xmlStrEqual(attr->name, BAD_CAST "xmlns"))) {
6165		xmlNsPtr ns;
6166
6167		ns = elem->nsDef;
6168		while (ns != NULL) {
6169		    if (ns->prefix == NULL)
6170			goto found;
6171		    ns = ns->next;
6172		}
6173	    } else if (xmlStrEqual(attr->prefix, BAD_CAST "xmlns")) {
6174		xmlNsPtr ns;
6175
6176		ns = elem->nsDef;
6177		while (ns != NULL) {
6178		    if (xmlStrEqual(attr->name, ns->prefix))
6179			goto found;
6180		    ns = ns->next;
6181		}
6182	    } else {
6183		xmlAttrPtr attrib;
6184
6185		attrib = elem->properties;
6186		while (attrib != NULL) {
6187		    if (xmlStrEqual(attrib->name, attr->name)) {
6188			if (attr->prefix != NULL) {
6189			    xmlNsPtr nameSpace = attrib->ns;
6190
6191			    if (nameSpace == NULL)
6192				nameSpace = elem->ns;
6193			    /*
6194			     * qualified names handling is problematic, having a
6195			     * different prefix should be possible but DTDs don't
6196			     * allow to define the URI instead of the prefix :-(
6197			     */
6198			    if (nameSpace == NULL) {
6199				if (qualified < 0)
6200				    qualified = 0;
6201			    } else if (!xmlStrEqual(nameSpace->prefix,
6202						    attr->prefix)) {
6203				if (qualified < 1)
6204				    qualified = 1;
6205			    } else
6206				goto found;
6207			} else {
6208			    /*
6209			     * We should allow applications to define namespaces
6210			     * for their application even if the DTD doesn't
6211			     * carry one, otherwise, basically we would always
6212			     * break.
6213			     */
6214			    goto found;
6215			}
6216		    }
6217		    attrib = attrib->next;
6218		}
6219	    }
6220	    if (qualified == -1) {
6221		if (attr->prefix == NULL) {
6222		    xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
6223		       "Element %s does not carry attribute %s\n",
6224			   elem->name, attr->name, NULL);
6225		    ret = 0;
6226	        } else {
6227		    xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
6228		       "Element %s does not carry attribute %s:%s\n",
6229			   elem->name, attr->prefix,attr->name);
6230		    ret = 0;
6231		}
6232	    } else if (qualified == 0) {
6233		xmlErrValidWarning(ctxt, elem, XML_DTD_NO_PREFIX,
6234		   "Element %s required attribute %s:%s has no prefix\n",
6235		       elem->name, attr->prefix, attr->name);
6236	    } else if (qualified == 1) {
6237		xmlErrValidWarning(ctxt, elem, XML_DTD_DIFFERENT_PREFIX,
6238		   "Element %s required attribute %s:%s has different prefix\n",
6239		       elem->name, attr->prefix, attr->name);
6240	    }
6241	} else if (attr->def == XML_ATTRIBUTE_FIXED) {
6242	    /*
6243	     * Special tests checking #FIXED namespace declarations
6244	     * have the right value since this is not done as an
6245	     * attribute checking
6246	     */
6247	    if ((attr->prefix == NULL) &&
6248		(xmlStrEqual(attr->name, BAD_CAST "xmlns"))) {
6249		xmlNsPtr ns;
6250
6251		ns = elem->nsDef;
6252		while (ns != NULL) {
6253		    if (ns->prefix == NULL) {
6254			if (!xmlStrEqual(attr->defaultValue, ns->href)) {
6255			    xmlErrValidNode(ctxt, elem,
6256			           XML_DTD_ELEM_DEFAULT_NAMESPACE,
6257   "Element %s namespace name for default namespace does not match the DTD\n",
6258				   elem->name, NULL, NULL);
6259			    ret = 0;
6260			}
6261			goto found;
6262		    }
6263		    ns = ns->next;
6264		}
6265	    } else if (xmlStrEqual(attr->prefix, BAD_CAST "xmlns")) {
6266		xmlNsPtr ns;
6267
6268		ns = elem->nsDef;
6269		while (ns != NULL) {
6270		    if (xmlStrEqual(attr->name, ns->prefix)) {
6271			if (!xmlStrEqual(attr->defaultValue, ns->href)) {
6272			    xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
6273		   "Element %s namespace name for %s does not match the DTD\n",
6274				   elem->name, ns->prefix, NULL);
6275			    ret = 0;
6276			}
6277			goto found;
6278		    }
6279		    ns = ns->next;
6280		}
6281	    }
6282	}
6283found:
6284        attr = attr->nexth;
6285    }
6286    return(ret);
6287}
6288
6289/**
6290 * xmlValidateRoot:
6291 * @ctxt:  the validation context
6292 * @doc:  a document instance
6293 *
6294 * Try to validate a the root element
6295 * basically it does the following check as described by the
6296 * XML-1.0 recommendation:
6297 *  - [ VC: Root Element Type ]
6298 * it doesn't try to recurse or apply other check to the element
6299 *
6300 * returns 1 if valid or 0 otherwise
6301 */
6302
6303int
6304xmlValidateRoot(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6305    xmlNodePtr root;
6306    int ret;
6307
6308    if (doc == NULL) return(0);
6309
6310    root = xmlDocGetRootElement(doc);
6311    if ((root == NULL) || (root->name == NULL)) {
6312	xmlErrValid(ctxt, XML_DTD_NO_ROOT,
6313	            "no root element\n", NULL);
6314        return(0);
6315    }
6316
6317    /*
6318     * When doing post validation against a separate DTD, those may
6319     * no internal subset has been generated
6320     */
6321    if ((doc->intSubset != NULL) &&
6322	(doc->intSubset->name != NULL)) {
6323	/*
6324	 * Check first the document root against the NQName
6325	 */
6326	if (!xmlStrEqual(doc->intSubset->name, root->name)) {
6327	    if ((root->ns != NULL) && (root->ns->prefix != NULL)) {
6328		xmlChar fn[50];
6329		xmlChar *fullname;
6330
6331		fullname = xmlBuildQName(root->name, root->ns->prefix, fn, 50);
6332		if (fullname == NULL) {
6333		    xmlVErrMemory(ctxt, NULL);
6334		    return(0);
6335		}
6336		ret = xmlStrEqual(doc->intSubset->name, fullname);
6337		if ((fullname != fn) && (fullname != root->name))
6338		    xmlFree(fullname);
6339		if (ret == 1)
6340		    goto name_ok;
6341	    }
6342	    if ((xmlStrEqual(doc->intSubset->name, BAD_CAST "HTML")) &&
6343		(xmlStrEqual(root->name, BAD_CAST "html")))
6344		goto name_ok;
6345	    xmlErrValidNode(ctxt, root, XML_DTD_ROOT_NAME,
6346		   "root and DTD name do not match '%s' and '%s'\n",
6347		   root->name, doc->intSubset->name, NULL);
6348	    return(0);
6349	}
6350    }
6351name_ok:
6352    return(1);
6353}
6354
6355
6356/**
6357 * xmlValidateElement:
6358 * @ctxt:  the validation context
6359 * @doc:  a document instance
6360 * @elem:  an element instance
6361 *
6362 * Try to validate the subtree under an element
6363 *
6364 * returns 1 if valid or 0 otherwise
6365 */
6366
6367int
6368xmlValidateElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlNodePtr elem) {
6369    xmlNodePtr child;
6370    xmlAttrPtr attr;
6371    xmlNsPtr ns;
6372    const xmlChar *value;
6373    int ret = 1;
6374
6375    if (elem == NULL) return(0);
6376
6377    /*
6378     * XInclude elements were added after parsing in the infoset,
6379     * they don't really mean anything validation wise.
6380     */
6381    if ((elem->type == XML_XINCLUDE_START) ||
6382	(elem->type == XML_XINCLUDE_END) ||
6383	(elem->type == XML_NAMESPACE_DECL))
6384	return(1);
6385
6386    CHECK_DTD;
6387
6388    /*
6389     * Entities references have to be handled separately
6390     */
6391    if (elem->type == XML_ENTITY_REF_NODE) {
6392	return(1);
6393    }
6394
6395    ret &= xmlValidateOneElement(ctxt, doc, elem);
6396    if (elem->type == XML_ELEMENT_NODE) {
6397	attr = elem->properties;
6398	while (attr != NULL) {
6399	    value = xmlNodeListGetString(doc, attr->children, 0);
6400	    ret &= xmlValidateOneAttribute(ctxt, doc, elem, attr, value);
6401	    if (value != NULL)
6402		xmlFree((char *)value);
6403	    attr= attr->next;
6404	}
6405	ns = elem->nsDef;
6406	while (ns != NULL) {
6407	    if (elem->ns == NULL)
6408		ret &= xmlValidateOneNamespace(ctxt, doc, elem, NULL,
6409					       ns, ns->href);
6410	    else
6411		ret &= xmlValidateOneNamespace(ctxt, doc, elem,
6412		                               elem->ns->prefix, ns, ns->href);
6413	    ns = ns->next;
6414	}
6415    }
6416    child = elem->children;
6417    while (child != NULL) {
6418        ret &= xmlValidateElement(ctxt, doc, child);
6419        child = child->next;
6420    }
6421
6422    return(ret);
6423}
6424
6425/**
6426 * xmlValidateRef:
6427 * @ref:   A reference to be validated
6428 * @ctxt:  Validation context
6429 * @name:  Name of ID we are searching for
6430 *
6431 */
6432static void
6433xmlValidateRef(xmlRefPtr ref, xmlValidCtxtPtr ctxt,
6434	                   const xmlChar *name) {
6435    xmlAttrPtr id;
6436    xmlAttrPtr attr;
6437
6438    if (ref == NULL)
6439	return;
6440    if ((ref->attr == NULL) && (ref->name == NULL))
6441	return;
6442    attr = ref->attr;
6443    if (attr == NULL) {
6444	xmlChar *dup, *str = NULL, *cur, save;
6445
6446	dup = xmlStrdup(name);
6447	if (dup == NULL) {
6448	    ctxt->valid = 0;
6449	    return;
6450	}
6451	cur = dup;
6452	while (*cur != 0) {
6453	    str = cur;
6454	    while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
6455	    save = *cur;
6456	    *cur = 0;
6457	    id = xmlGetID(ctxt->doc, str);
6458	    if (id == NULL) {
6459		xmlErrValidNodeNr(ctxt, NULL, XML_DTD_UNKNOWN_ID,
6460	   "attribute %s line %d references an unknown ID \"%s\"\n",
6461		       ref->name, ref->lineno, str);
6462		ctxt->valid = 0;
6463	    }
6464	    if (save == 0)
6465		break;
6466	    *cur = save;
6467	    while (IS_BLANK_CH(*cur)) cur++;
6468	}
6469	xmlFree(dup);
6470    } else if (attr->atype == XML_ATTRIBUTE_IDREF) {
6471	id = xmlGetID(ctxt->doc, name);
6472	if (id == NULL) {
6473	    xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
6474	   "IDREF attribute %s references an unknown ID \"%s\"\n",
6475		   attr->name, name, NULL);
6476	    ctxt->valid = 0;
6477	}
6478    } else if (attr->atype == XML_ATTRIBUTE_IDREFS) {
6479	xmlChar *dup, *str = NULL, *cur, save;
6480
6481	dup = xmlStrdup(name);
6482	if (dup == NULL) {
6483	    xmlVErrMemory(ctxt, "IDREFS split");
6484	    ctxt->valid = 0;
6485	    return;
6486	}
6487	cur = dup;
6488	while (*cur != 0) {
6489	    str = cur;
6490	    while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
6491	    save = *cur;
6492	    *cur = 0;
6493	    id = xmlGetID(ctxt->doc, str);
6494	    if (id == NULL) {
6495		xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
6496	   "IDREFS attribute %s references an unknown ID \"%s\"\n",
6497			     attr->name, str, NULL);
6498		ctxt->valid = 0;
6499	    }
6500	    if (save == 0)
6501		break;
6502	    *cur = save;
6503	    while (IS_BLANK_CH(*cur)) cur++;
6504	}
6505	xmlFree(dup);
6506    }
6507}
6508
6509/**
6510 * xmlWalkValidateList:
6511 * @data:  Contents of current link
6512 * @user:  Value supplied by the user
6513 *
6514 * Returns 0 to abort the walk or 1 to continue
6515 */
6516static int
6517xmlWalkValidateList(const void *data, const void *user)
6518{
6519	xmlValidateMemoPtr memo = (xmlValidateMemoPtr)user;
6520	xmlValidateRef((xmlRefPtr)data, memo->ctxt, memo->name);
6521	return 1;
6522}
6523
6524/**
6525 * xmlValidateCheckRefCallback:
6526 * @ref_list:  List of references
6527 * @ctxt:  Validation context
6528 * @name:  Name of ID we are searching for
6529 *
6530 */
6531static void
6532xmlValidateCheckRefCallback(xmlListPtr ref_list, xmlValidCtxtPtr ctxt,
6533	                   const xmlChar *name) {
6534    xmlValidateMemo memo;
6535
6536    if (ref_list == NULL)
6537	return;
6538    memo.ctxt = ctxt;
6539    memo.name = name;
6540
6541    xmlListWalk(ref_list, xmlWalkValidateList, &memo);
6542
6543}
6544
6545/**
6546 * xmlValidateDocumentFinal:
6547 * @ctxt:  the validation context
6548 * @doc:  a document instance
6549 *
6550 * Does the final step for the document validation once all the
6551 * incremental validation steps have been completed
6552 *
6553 * basically it does the following checks described by the XML Rec
6554 *
6555 * Check all the IDREF/IDREFS attributes definition for validity
6556 *
6557 * returns 1 if valid or 0 otherwise
6558 */
6559
6560int
6561xmlValidateDocumentFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6562    xmlRefTablePtr table;
6563    unsigned int save;
6564
6565    if (ctxt == NULL)
6566        return(0);
6567    if (doc == NULL) {
6568        xmlErrValid(ctxt, XML_DTD_NO_DOC,
6569		"xmlValidateDocumentFinal: doc == NULL\n", NULL);
6570	return(0);
6571    }
6572
6573    /* trick to get correct line id report */
6574    save = ctxt->finishDtd;
6575    ctxt->finishDtd = 0;
6576
6577    /*
6578     * Check all the NOTATION/NOTATIONS attributes
6579     */
6580    /*
6581     * Check all the ENTITY/ENTITIES attributes definition for validity
6582     */
6583    /*
6584     * Check all the IDREF/IDREFS attributes definition for validity
6585     */
6586    table = (xmlRefTablePtr) doc->refs;
6587    ctxt->doc = doc;
6588    ctxt->valid = 1;
6589    xmlHashScan(table, (xmlHashScanner) xmlValidateCheckRefCallback, ctxt);
6590
6591    ctxt->finishDtd = save;
6592    return(ctxt->valid);
6593}
6594
6595/**
6596 * xmlValidateDtd:
6597 * @ctxt:  the validation context
6598 * @doc:  a document instance
6599 * @dtd:  a dtd instance
6600 *
6601 * Try to validate the document against the dtd instance
6602 *
6603 * Basically it does check all the definitions in the DtD.
6604 * Note the the internal subset (if present) is de-coupled
6605 * (i.e. not used), which could give problems if ID or IDREF
6606 * is present.
6607 *
6608 * returns 1 if valid or 0 otherwise
6609 */
6610
6611int
6612xmlValidateDtd(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlDtdPtr dtd) {
6613    int ret;
6614    xmlDtdPtr oldExt, oldInt;
6615    xmlNodePtr root;
6616
6617    if (dtd == NULL) return(0);
6618    if (doc == NULL) return(0);
6619    oldExt = doc->extSubset;
6620    oldInt = doc->intSubset;
6621    doc->extSubset = dtd;
6622    doc->intSubset = NULL;
6623    ret = xmlValidateRoot(ctxt, doc);
6624    if (ret == 0) {
6625	doc->extSubset = oldExt;
6626	doc->intSubset = oldInt;
6627	return(ret);
6628    }
6629    if (doc->ids != NULL) {
6630          xmlFreeIDTable(doc->ids);
6631          doc->ids = NULL;
6632    }
6633    if (doc->refs != NULL) {
6634          xmlFreeRefTable(doc->refs);
6635          doc->refs = NULL;
6636    }
6637    root = xmlDocGetRootElement(doc);
6638    ret = xmlValidateElement(ctxt, doc, root);
6639    ret &= xmlValidateDocumentFinal(ctxt, doc);
6640    doc->extSubset = oldExt;
6641    doc->intSubset = oldInt;
6642    return(ret);
6643}
6644
6645static void
6646xmlValidateNotationCallback(xmlEntityPtr cur, xmlValidCtxtPtr ctxt,
6647	                    const xmlChar *name ATTRIBUTE_UNUSED) {
6648    if (cur == NULL)
6649	return;
6650    if (cur->etype == XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
6651	xmlChar *notation = cur->content;
6652
6653	if (notation != NULL) {
6654	    int ret;
6655
6656	    ret = xmlValidateNotationUse(ctxt, cur->doc, notation);
6657	    if (ret != 1) {
6658		ctxt->valid = 0;
6659	    }
6660	}
6661    }
6662}
6663
6664static void
6665xmlValidateAttributeCallback(xmlAttributePtr cur, xmlValidCtxtPtr ctxt,
6666	                    const xmlChar *name ATTRIBUTE_UNUSED) {
6667    int ret;
6668    xmlDocPtr doc;
6669    xmlElementPtr elem = NULL;
6670
6671    if (cur == NULL)
6672	return;
6673    switch (cur->atype) {
6674	case XML_ATTRIBUTE_CDATA:
6675	case XML_ATTRIBUTE_ID:
6676	case XML_ATTRIBUTE_IDREF	:
6677	case XML_ATTRIBUTE_IDREFS:
6678	case XML_ATTRIBUTE_NMTOKEN:
6679	case XML_ATTRIBUTE_NMTOKENS:
6680	case XML_ATTRIBUTE_ENUMERATION:
6681	    break;
6682	case XML_ATTRIBUTE_ENTITY:
6683	case XML_ATTRIBUTE_ENTITIES:
6684	case XML_ATTRIBUTE_NOTATION:
6685	    if (cur->defaultValue != NULL) {
6686
6687		ret = xmlValidateAttributeValue2(ctxt, ctxt->doc, cur->name,
6688			                         cur->atype, cur->defaultValue);
6689		if ((ret == 0) && (ctxt->valid == 1))
6690		    ctxt->valid = 0;
6691	    }
6692	    if (cur->tree != NULL) {
6693		xmlEnumerationPtr tree = cur->tree;
6694		while (tree != NULL) {
6695		    ret = xmlValidateAttributeValue2(ctxt, ctxt->doc,
6696				    cur->name, cur->atype, tree->name);
6697		    if ((ret == 0) && (ctxt->valid == 1))
6698			ctxt->valid = 0;
6699		    tree = tree->next;
6700		}
6701	    }
6702    }
6703    if (cur->atype == XML_ATTRIBUTE_NOTATION) {
6704	doc = cur->doc;
6705	if (cur->elem == NULL) {
6706	    xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
6707		   "xmlValidateAttributeCallback(%s): internal error\n",
6708		   (const char *) cur->name);
6709	    return;
6710	}
6711
6712	if (doc != NULL)
6713	    elem = xmlGetDtdElementDesc(doc->intSubset, cur->elem);
6714	if ((elem == NULL) && (doc != NULL))
6715	    elem = xmlGetDtdElementDesc(doc->extSubset, cur->elem);
6716	if ((elem == NULL) && (cur->parent != NULL) &&
6717	    (cur->parent->type == XML_DTD_NODE))
6718	    elem = xmlGetDtdElementDesc((xmlDtdPtr) cur->parent, cur->elem);
6719	if (elem == NULL) {
6720	    xmlErrValidNode(ctxt, NULL, XML_DTD_UNKNOWN_ELEM,
6721		   "attribute %s: could not find decl for element %s\n",
6722		   cur->name, cur->elem, NULL);
6723	    return;
6724	}
6725	if (elem->etype == XML_ELEMENT_TYPE_EMPTY) {
6726	    xmlErrValidNode(ctxt, NULL, XML_DTD_EMPTY_NOTATION,
6727		   "NOTATION attribute %s declared for EMPTY element %s\n",
6728		   cur->name, cur->elem, NULL);
6729	    ctxt->valid = 0;
6730	}
6731    }
6732}
6733
6734/**
6735 * xmlValidateDtdFinal:
6736 * @ctxt:  the validation context
6737 * @doc:  a document instance
6738 *
6739 * Does the final step for the dtds validation once all the
6740 * subsets have been parsed
6741 *
6742 * basically it does the following checks described by the XML Rec
6743 * - check that ENTITY and ENTITIES type attributes default or
6744 *   possible values matches one of the defined entities.
6745 * - check that NOTATION type attributes default or
6746 *   possible values matches one of the defined notations.
6747 *
6748 * returns 1 if valid or 0 if invalid and -1 if not well-formed
6749 */
6750
6751int
6752xmlValidateDtdFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6753    xmlDtdPtr dtd;
6754    xmlAttributeTablePtr table;
6755    xmlEntitiesTablePtr entities;
6756
6757    if ((doc == NULL) || (ctxt == NULL)) return(0);
6758    if ((doc->intSubset == NULL) && (doc->extSubset == NULL))
6759	return(0);
6760    ctxt->doc = doc;
6761    ctxt->valid = 1;
6762    dtd = doc->intSubset;
6763    if ((dtd != NULL) && (dtd->attributes != NULL)) {
6764	table = (xmlAttributeTablePtr) dtd->attributes;
6765	xmlHashScan(table, (xmlHashScanner) xmlValidateAttributeCallback, ctxt);
6766    }
6767    if ((dtd != NULL) && (dtd->entities != NULL)) {
6768	entities = (xmlEntitiesTablePtr) dtd->entities;
6769	xmlHashScan(entities, (xmlHashScanner) xmlValidateNotationCallback,
6770		    ctxt);
6771    }
6772    dtd = doc->extSubset;
6773    if ((dtd != NULL) && (dtd->attributes != NULL)) {
6774	table = (xmlAttributeTablePtr) dtd->attributes;
6775	xmlHashScan(table, (xmlHashScanner) xmlValidateAttributeCallback, ctxt);
6776    }
6777    if ((dtd != NULL) && (dtd->entities != NULL)) {
6778	entities = (xmlEntitiesTablePtr) dtd->entities;
6779	xmlHashScan(entities, (xmlHashScanner) xmlValidateNotationCallback,
6780		    ctxt);
6781    }
6782    return(ctxt->valid);
6783}
6784
6785/**
6786 * xmlValidateDocument:
6787 * @ctxt:  the validation context
6788 * @doc:  a document instance
6789 *
6790 * Try to validate the document instance
6791 *
6792 * basically it does the all the checks described by the XML Rec
6793 * i.e. validates the internal and external subset (if present)
6794 * and validate the document tree.
6795 *
6796 * returns 1 if valid or 0 otherwise
6797 */
6798
6799int
6800xmlValidateDocument(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6801    int ret;
6802    xmlNodePtr root;
6803
6804    if (doc == NULL)
6805        return(0);
6806    if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
6807        xmlErrValid(ctxt, XML_DTD_NO_DTD,
6808	            "no DTD found!\n", NULL);
6809	return(0);
6810    }
6811    if ((doc->intSubset != NULL) && ((doc->intSubset->SystemID != NULL) ||
6812	(doc->intSubset->ExternalID != NULL)) && (doc->extSubset == NULL)) {
6813	xmlChar *sysID;
6814	if (doc->intSubset->SystemID != NULL) {
6815	    sysID = xmlBuildURI(doc->intSubset->SystemID,
6816			doc->URL);
6817	    if (sysID == NULL) {
6818	        xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
6819			"Could not build URI for external subset \"%s\"\n",
6820			(const char *) doc->intSubset->SystemID);
6821		return 0;
6822	    }
6823	} else
6824	    sysID = NULL;
6825        doc->extSubset = xmlParseDTD(doc->intSubset->ExternalID,
6826			(const xmlChar *)sysID);
6827	if (sysID != NULL)
6828	    xmlFree(sysID);
6829        if (doc->extSubset == NULL) {
6830	    if (doc->intSubset->SystemID != NULL) {
6831		xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
6832		       "Could not load the external subset \"%s\"\n",
6833		       (const char *) doc->intSubset->SystemID);
6834	    } else {
6835		xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
6836		       "Could not load the external subset \"%s\"\n",
6837		       (const char *) doc->intSubset->ExternalID);
6838	    }
6839	    return(0);
6840	}
6841    }
6842
6843    if (doc->ids != NULL) {
6844          xmlFreeIDTable(doc->ids);
6845          doc->ids = NULL;
6846    }
6847    if (doc->refs != NULL) {
6848          xmlFreeRefTable(doc->refs);
6849          doc->refs = NULL;
6850    }
6851    ret = xmlValidateDtdFinal(ctxt, doc);
6852    if (!xmlValidateRoot(ctxt, doc)) return(0);
6853
6854    root = xmlDocGetRootElement(doc);
6855    ret &= xmlValidateElement(ctxt, doc, root);
6856    ret &= xmlValidateDocumentFinal(ctxt, doc);
6857    return(ret);
6858}
6859
6860/************************************************************************
6861 *									*
6862 *		Routines for dynamic validation editing			*
6863 *									*
6864 ************************************************************************/
6865
6866/**
6867 * xmlValidGetPotentialChildren:
6868 * @ctree:  an element content tree
6869 * @names:  an array to store the list of child names
6870 * @len:  a pointer to the number of element in the list
6871 * @max:  the size of the array
6872 *
6873 * Build/extend a list of  potential children allowed by the content tree
6874 *
6875 * returns the number of element in the list, or -1 in case of error.
6876 */
6877
6878int
6879xmlValidGetPotentialChildren(xmlElementContent *ctree,
6880                             const xmlChar **names,
6881                             int *len, int max) {
6882    int i;
6883
6884    if ((ctree == NULL) || (names == NULL) || (len == NULL))
6885        return(-1);
6886    if (*len >= max) return(*len);
6887
6888    switch (ctree->type) {
6889	case XML_ELEMENT_CONTENT_PCDATA:
6890	    for (i = 0; i < *len;i++)
6891		if (xmlStrEqual(BAD_CAST "#PCDATA", names[i])) return(*len);
6892	    names[(*len)++] = BAD_CAST "#PCDATA";
6893	    break;
6894	case XML_ELEMENT_CONTENT_ELEMENT:
6895	    for (i = 0; i < *len;i++)
6896		if (xmlStrEqual(ctree->name, names[i])) return(*len);
6897	    names[(*len)++] = ctree->name;
6898	    break;
6899	case XML_ELEMENT_CONTENT_SEQ:
6900	    xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6901	    xmlValidGetPotentialChildren(ctree->c2, names, len, max);
6902	    break;
6903	case XML_ELEMENT_CONTENT_OR:
6904	    xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6905	    xmlValidGetPotentialChildren(ctree->c2, names, len, max);
6906	    break;
6907   }
6908
6909   return(*len);
6910}
6911
6912/*
6913 * Dummy function to suppress messages while we try out valid elements
6914 */
6915static void XMLCDECL xmlNoValidityErr(void *ctx ATTRIBUTE_UNUSED,
6916                                const char *msg ATTRIBUTE_UNUSED, ...) {
6917    return;
6918}
6919
6920/**
6921 * xmlValidGetValidElements:
6922 * @prev:  an element to insert after
6923 * @next:  an element to insert next
6924 * @names:  an array to store the list of child names
6925 * @max:  the size of the array
6926 *
6927 * This function returns the list of authorized children to insert
6928 * within an existing tree while respecting the validity constraints
6929 * forced by the Dtd. The insertion point is defined using @prev and
6930 * @next in the following ways:
6931 *  to insert before 'node': xmlValidGetValidElements(node->prev, node, ...
6932 *  to insert next 'node': xmlValidGetValidElements(node, node->next, ...
6933 *  to replace 'node': xmlValidGetValidElements(node->prev, node->next, ...
6934 *  to prepend a child to 'node': xmlValidGetValidElements(NULL, node->childs,
6935 *  to append a child to 'node': xmlValidGetValidElements(node->last, NULL, ...
6936 *
6937 * pointers to the element names are inserted at the beginning of the array
6938 * and do not need to be freed.
6939 *
6940 * returns the number of element in the list, or -1 in case of error. If
6941 *    the function returns the value @max the caller is invited to grow the
6942 *    receiving array and retry.
6943 */
6944
6945int
6946xmlValidGetValidElements(xmlNode *prev, xmlNode *next, const xmlChar **names,
6947                         int max) {
6948    xmlValidCtxt vctxt;
6949    int nb_valid_elements = 0;
6950    const xmlChar *elements[256]={0};
6951    int nb_elements = 0, i;
6952    const xmlChar *name;
6953
6954    xmlNode *ref_node;
6955    xmlNode *parent;
6956    xmlNode *test_node;
6957
6958    xmlNode *prev_next;
6959    xmlNode *next_prev;
6960    xmlNode *parent_childs;
6961    xmlNode *parent_last;
6962
6963    xmlElement *element_desc;
6964
6965    if (prev == NULL && next == NULL)
6966        return(-1);
6967
6968    if (names == NULL) return(-1);
6969    if (max <= 0) return(-1);
6970
6971    memset(&vctxt, 0, sizeof (xmlValidCtxt));
6972    vctxt.error = xmlNoValidityErr;	/* this suppresses err/warn output */
6973
6974    nb_valid_elements = 0;
6975    ref_node = prev ? prev : next;
6976    parent = ref_node->parent;
6977
6978    /*
6979     * Retrieves the parent element declaration
6980     */
6981    element_desc = xmlGetDtdElementDesc(parent->doc->intSubset,
6982                                         parent->name);
6983    if ((element_desc == NULL) && (parent->doc->extSubset != NULL))
6984        element_desc = xmlGetDtdElementDesc(parent->doc->extSubset,
6985                                             parent->name);
6986    if (element_desc == NULL) return(-1);
6987
6988    /*
6989     * Do a backup of the current tree structure
6990     */
6991    prev_next = prev ? prev->next : NULL;
6992    next_prev = next ? next->prev : NULL;
6993    parent_childs = parent->children;
6994    parent_last = parent->last;
6995
6996    /*
6997     * Creates a dummy node and insert it into the tree
6998     */
6999    test_node = xmlNewDocNode (ref_node->doc, NULL, BAD_CAST "<!dummy?>", NULL);
7000    if (test_node == NULL)
7001        return(-1);
7002
7003    test_node->parent = parent;
7004    test_node->prev = prev;
7005    test_node->next = next;
7006    name = test_node->name;
7007
7008    if (prev) prev->next = test_node;
7009    else parent->children = test_node;
7010
7011    if (next) next->prev = test_node;
7012    else parent->last = test_node;
7013
7014    /*
7015     * Insert each potential child node and check if the parent is
7016     * still valid
7017     */
7018    nb_elements = xmlValidGetPotentialChildren(element_desc->content,
7019		       elements, &nb_elements, 256);
7020
7021    for (i = 0;i < nb_elements;i++) {
7022	test_node->name = elements[i];
7023	if (xmlValidateOneElement(&vctxt, parent->doc, parent)) {
7024	    int j;
7025
7026	    for (j = 0; j < nb_valid_elements;j++)
7027		if (xmlStrEqual(elements[i], names[j])) break;
7028	    names[nb_valid_elements++] = elements[i];
7029	    if (nb_valid_elements >= max) break;
7030	}
7031    }
7032
7033    /*
7034     * Restore the tree structure
7035     */
7036    if (prev) prev->next = prev_next;
7037    if (next) next->prev = next_prev;
7038    parent->children = parent_childs;
7039    parent->last = parent_last;
7040
7041    /*
7042     * Free up the dummy node
7043     */
7044    test_node->name = name;
7045    xmlFreeNode(test_node);
7046
7047    return(nb_valid_elements);
7048}
7049#endif /* LIBXML_VALID_ENABLED */
7050
7051#define bottom_valid
7052#include "elfgcchack.h"
7053