blob: 409aa81dad69e2353ee5698799de0b62e0b4c5ca [file] [log] [blame]
Owen Taylor3473f882001-02-23 17:55:21 +00001/*
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 *
Daniel Veillardc5d64342001-06-24 12:13:24 +00007 * daniel@veillard.com
Owen Taylor3473f882001-02-23 17:55:21 +00008 */
9
Daniel Veillard34ce8be2002-03-18 19:37:11 +000010#define IN_LIBXML
Bjorn Reese70a9da52001-04-21 16:57:29 +000011#include "libxml.h"
Owen Taylor3473f882001-02-23 17:55:21 +000012
Owen Taylor3473f882001-02-23 17:55:21 +000013#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>
Daniel Veillard29b17482004-08-16 00:39:03 +000021#include <libxml/uri.h>
Owen Taylor3473f882001-02-23 17:55:21 +000022#include <libxml/valid.h>
23#include <libxml/parser.h>
24#include <libxml/parserInternals.h>
25#include <libxml/xmlerror.h>
26#include <libxml/list.h>
Daniel Veillard3c01b1d2001-10-17 15:58:35 +000027#include <libxml/globals.h>
Owen Taylor3473f882001-02-23 17:55:21 +000028
Daniel Veillard4432df22003-09-28 18:58:27 +000029static xmlElementPtr xmlGetDtdElementDesc2(xmlDtdPtr dtd, const xmlChar *name,
30 int create);
Daniel Veillarde62d36c2001-05-15 08:53:16 +000031/* #define DEBUG_VALID_ALGO */
Daniel Veillard5acfd6b2002-09-18 16:29:02 +000032/* #define DEBUG_REGEXP_ALGO */
Daniel Veillarde62d36c2001-05-15 08:53:16 +000033
Daniel Veillardf8e3db02012-09-11 13:26:36 +080034#define TODO \
Daniel Veillarda646cfd2002-09-17 21:50:03 +000035 xmlGenericError(xmlGenericErrorContext, \
36 "Unimplemented block at %s:%d\n", \
37 __FILE__, __LINE__);
Owen Taylor3473f882001-02-23 17:55:21 +000038
Daniel Veillardae0765b2008-07-31 19:54:59 +000039#ifdef LIBXML_VALID_ENABLED
40static int
41xmlValidateAttributeValueInternal(xmlDocPtr doc, xmlAttributeType type,
42 const xmlChar *value);
43#endif
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000044/************************************************************************
45 * *
46 * Error handling routines *
47 * *
48 ************************************************************************/
49
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000050/**
Daniel Veillardce9457f2003-10-05 21:33:18 +000051 * xmlVErrMemory:
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000052 * @ctxt: an XML validation parser context
53 * @extra: extra informations
54 *
55 * Handle an out of memory error
56 */
57static void
Daniel Veillardce9457f2003-10-05 21:33:18 +000058xmlVErrMemory(xmlValidCtxtPtr ctxt, const char *extra)
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000059{
Daniel Veillardbb5abab2003-10-03 22:21:51 +000060 xmlGenericErrorFunc channel = NULL;
61 xmlParserCtxtPtr pctxt = NULL;
62 void *data = NULL;
63
64 if (ctxt != NULL) {
65 channel = ctxt->error;
66 data = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +000067 /* 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)) {
Daniel Veillardb2dc5672006-08-22 14:45:40 +000071 long delta = (char *) ctxt - (char *) ctxt->userData;
72 if ((delta > 0) && (delta < 250))
73 pctxt = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +000074 }
Daniel Veillardbb5abab2003-10-03 22:21:51 +000075 }
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000076 if (extra)
Daniel Veillard73000572003-10-11 11:26:42 +000077 __xmlRaiseError(NULL, channel, data,
Daniel Veillard72b9e292003-10-28 15:44:17 +000078 pctxt, NULL, XML_FROM_VALID, XML_ERR_NO_MEMORY,
Daniel Veillardbb5abab2003-10-03 22:21:51 +000079 XML_ERR_FATAL, NULL, 0, extra, NULL, NULL, 0, 0,
80 "Memory allocation failed : %s\n", extra);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000081 else
Daniel Veillard73000572003-10-11 11:26:42 +000082 __xmlRaiseError(NULL, channel, data,
Daniel Veillard72b9e292003-10-28 15:44:17 +000083 pctxt, NULL, XML_FROM_VALID, XML_ERR_NO_MEMORY,
Daniel Veillardbb5abab2003-10-03 22:21:51 +000084 XML_ERR_FATAL, NULL, 0, NULL, NULL, NULL, 0, 0,
85 "Memory allocation failed\n");
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000086}
87
88/**
89 * xmlErrValid:
90 * @ctxt: an XML validation parser context
Daniel Veillardbb5abab2003-10-03 22:21:51 +000091 * @error: the error number
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000092 * @extra: extra informations
93 *
94 * Handle a validation error
95 */
96static void
Daniel Veillardf88d8cf2003-12-08 10:25:02 +000097xmlErrValid(xmlValidCtxtPtr ctxt, xmlParserErrors error,
Daniel Veillardbb5abab2003-10-03 22:21:51 +000098 const char *msg, const char *extra)
Daniel Veillard2b8c4a12003-10-02 22:28:19 +000099{
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000100 xmlGenericErrorFunc channel = NULL;
101 xmlParserCtxtPtr pctxt = NULL;
102 void *data = NULL;
103
104 if (ctxt != NULL) {
105 channel = ctxt->error;
106 data = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000107 /* 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)) {
Daniel Veillardb2dc5672006-08-22 14:45:40 +0000111 long delta = (char *) ctxt - (char *) ctxt->userData;
112 if ((delta > 0) && (delta < 250))
113 pctxt = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000114 }
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000115 }
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000116 if (extra)
Daniel Veillard73000572003-10-11 11:26:42 +0000117 __xmlRaiseError(NULL, channel, data,
Daniel Veillard72b9e292003-10-28 15:44:17 +0000118 pctxt, NULL, XML_FROM_VALID, error,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000119 XML_ERR_ERROR, NULL, 0, extra, NULL, NULL, 0, 0,
120 msg, extra);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000121 else
Daniel Veillard73000572003-10-11 11:26:42 +0000122 __xmlRaiseError(NULL, channel, data,
Daniel Veillard72b9e292003-10-28 15:44:17 +0000123 pctxt, NULL, XML_FROM_VALID, error,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000124 XML_ERR_ERROR, NULL, 0, NULL, NULL, NULL, 0, 0,
Daniel Veillardbccae2d2009-06-04 11:22:45 +0200125 "%s", msg);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000126}
127
Daniel Veillardf54cd532004-02-25 11:52:31 +0000128#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;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000154 /* 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)) {
Daniel Veillardb2dc5672006-08-22 14:45:40 +0000158 long delta = (char *) ctxt - (char *) ctxt->userData;
159 if ((delta > 0) && (delta < 250))
160 pctxt = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000161 }
Daniel Veillardf54cd532004-02-25 11:52:31 +0000162 }
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
Daniel Veillardd61e8fb2003-10-19 21:59:17 +0000171#ifdef LIBXML_VALID_ENABLED
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000172/**
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
Daniel Veillard72b9e292003-10-28 15:44:17 +0000184xmlErrValidNodeNr(xmlValidCtxtPtr ctxt,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000185 xmlNodePtr node, xmlParserErrors error,
186 const char *msg, const xmlChar * str1,
187 int int2, const xmlChar * str3)
188{
Daniel Veillard659e71e2003-10-10 14:10:40 +0000189 xmlStructuredErrorFunc schannel = NULL;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000190 xmlGenericErrorFunc channel = NULL;
191 xmlParserCtxtPtr pctxt = NULL;
192 void *data = NULL;
193
194 if (ctxt != NULL) {
195 channel = ctxt->error;
196 data = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000197 /* 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)) {
Daniel Veillardb2dc5672006-08-22 14:45:40 +0000201 long delta = (char *) ctxt - (char *) ctxt->userData;
202 if ((delta > 0) && (delta < 250))
203 pctxt = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000204 }
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000205 }
Daniel Veillard72b9e292003-10-28 15:44:17 +0000206 __xmlRaiseError(schannel, channel, data, pctxt, node, XML_FROM_VALID, error,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000207 XML_ERR_ERROR, NULL, 0,
208 (const char *) str1,
209 (const char *) str3,
210 NULL, int2, 0, msg, str1, int2, str3);
211}
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000212
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000213/**
214 * xmlErrValidWarning:
215 * @ctxt: an XML validation parser context
216 * @node: the node raising the error
217 * @error: the error number
William M. Brackedb65a72004-02-06 07:36:04 +0000218 * @str1: extra information
219 * @str2: extra information
220 * @str3: extra information
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000221 *
William M. Brackedb65a72004-02-06 07:36:04 +0000222 * Handle a validation error, provide contextual information
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000223 */
224static void
William M. Brackedb65a72004-02-06 07:36:04 +0000225xmlErrValidWarning(xmlValidCtxtPtr ctxt,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000226 xmlNodePtr node, xmlParserErrors error,
227 const char *msg, const xmlChar * str1,
228 const xmlChar * str2, const xmlChar * str3)
229{
Daniel Veillard659e71e2003-10-10 14:10:40 +0000230 xmlStructuredErrorFunc schannel = NULL;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000231 xmlGenericErrorFunc channel = NULL;
232 xmlParserCtxtPtr pctxt = NULL;
233 void *data = NULL;
234
235 if (ctxt != NULL) {
William M. Bracke4d526f2004-12-18 00:01:21 +0000236 channel = ctxt->warning;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000237 data = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000238 /* 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)) {
Daniel Veillardb2dc5672006-08-22 14:45:40 +0000242 long delta = (char *) ctxt - (char *) ctxt->userData;
243 if ((delta > 0) && (delta < 250))
244 pctxt = ctxt->userData;
Daniel Veillardeff45a92004-10-29 12:10:55 +0000245 }
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000246 }
Daniel Veillard72b9e292003-10-28 15:44:17 +0000247 __xmlRaiseError(schannel, channel, data, pctxt, node, XML_FROM_VALID, error,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000248 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
Daniel Veillardea7751d2002-12-20 00:16:24 +0000255
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) {
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000275 if ((ctxt->vstateMax == 0) || (ctxt->vstateTab == NULL)) {
Daniel Veillardea7751d2002-12-20 00:16:24 +0000276 ctxt->vstateMax = 10;
277 ctxt->vstateTab = (xmlValidState *) xmlMalloc(ctxt->vstateMax *
278 sizeof(ctxt->vstateTab[0]));
279 if (ctxt->vstateTab == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000280 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillardea7751d2002-12-20 00:16:24 +0000281 return(-1);
282 }
283 }
284
285 if (ctxt->vstateNr >= ctxt->vstateMax) {
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000286 xmlValidState *tmp;
287
288 tmp = (xmlValidState *) xmlRealloc(ctxt->vstateTab,
289 2 * ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
290 if (tmp == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000291 xmlVErrMemory(ctxt, "realloc failed");
Daniel Veillardea7751d2002-12-20 00:16:24 +0000292 return(-1);
293 }
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000294 ctxt->vstateMax *= 2;
295 ctxt->vstateTab = tmp;
Daniel Veillardea7751d2002-12-20 00:16:24 +0000296 }
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) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800304 ctxt->vstateTab[ctxt->vstateNr].exec =
Daniel Veillardea7751d2002-12-20 00:16:24 +0000305 xmlRegNewExecCtxt(elemDecl->contModel, NULL, NULL);
306 } else {
307 ctxt->vstateTab[ctxt->vstateNr].exec = NULL;
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000308 xmlErrValidNode(ctxt, (xmlNodePtr) elemDecl,
309 XML_ERR_INTERNAL_ERROR,
310 "Failed to build content model regexp for %s\n",
311 node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +0000312 }
313 }
314 return(ctxt->vstateNr++);
315}
316
317static int
318vstateVPop(xmlValidCtxtPtr ctxt) {
319 xmlElementPtr elemDecl;
320
Daniel Veillard336fc7d2002-12-27 19:37:04 +0000321 if (ctxt->vstateNr < 1) return(-1);
Daniel Veillardea7751d2002-12-20 00:16:24 +0000322 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 */
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000338/*
Daniel Veillard1c732d22002-11-30 11:22:59 +0000339 * If regexp are not enabled, it uses a home made algorithm less
340 * complex and easier to
Daniel Veillardcbaf3992001-12-31 16:16:02 +0000341 * debug/maintain than a generic NFA -> DFA state based algo. The
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000342 * 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
Daniel Veillardb44025c2001-10-11 22:55:55 +0000351typedef struct _xmlValidState {
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000352 xmlElementContentPtr cont; /* pointer to the content model subtree */
353 xmlNodePtr node; /* pointer to the current node in the list */
Daniel Veillardcbaf3992001-12-31 16:16:02 +0000354 long occurs;/* bitfield for multiple occurrences */
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000355 unsigned char depth; /* current depth in the overall tree */
356 unsigned char state; /* ROLLBACK_XXX */
357} _xmlValidState;
358
Daniel Veillardfc57b412002-04-29 15:50:14 +0000359#define MAX_RECURSE 25000
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000360#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
Daniel Veillard5344c602001-12-31 16:37:34 +0000367#define OCCURRENCE (ctxt->vstate->occurs & (1 << DEPTH))
368#define PARENT_OCCURRENCE (ctxt->vstate->occurs & ((1 << DEPTH) - 1))
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000369
Daniel Veillard5344c602001-12-31 16:37:34 +0000370#define SET_OCCURRENCE ctxt->vstate->occurs |= (1 << DEPTH)
371#define RESET_OCCURRENCE ctxt->vstate->occurs &= ((1 << DEPTH) - 1)
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000372
373static int
374vstateVPush(xmlValidCtxtPtr ctxt, xmlElementContentPtr cont,
375 xmlNodePtr node, unsigned char depth, long occurs,
376 unsigned char state) {
Daniel Veillardbed7b052001-05-19 14:59:49 +0000377 int i = ctxt->vstateNr - 1;
378
Daniel Veillard940492d2002-04-15 10:15:25 +0000379 if (ctxt->vstateNr > MAX_RECURSE) {
380 return(-1);
381 }
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000382 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) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000387 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000388 return(-1);
389 }
390 }
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000391 if (ctxt->vstateNr >= ctxt->vstateMax) {
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000392 xmlValidState *tmp;
393
394 tmp = (xmlValidState *) xmlRealloc(ctxt->vstateTab,
395 2 * ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
396 if (tmp == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000397 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard940492d2002-04-15 10:15:25 +0000398 return(-1);
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000399 }
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000400 ctxt->vstateMax *= 2;
401 ctxt->vstateTab = tmp;
Daniel Veillard06803992001-04-22 10:35:56 +0000402 ctxt->vstate = &ctxt->vstateTab[0];
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000403 }
Daniel Veillardbed7b052001-05-19 14:59:49 +0000404 /*
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);
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000413 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
Daniel Veillard118aed72002-09-24 14:13:13 +0000434#endif /* LIBXML_REGEXP_ENABLED */
435
Daniel Veillard1c732d22002-11-30 11:22:59 +0000436static 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) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000445 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard1c732d22002-11-30 11:22:59 +0000446 ctxt->nodeMax = 0;
447 return (0);
448 }
449 }
450 if (ctxt->nodeNr >= ctxt->nodeMax) {
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000451 xmlNodePtr *tmp;
452 tmp = (xmlNodePtr *) xmlRealloc(ctxt->nodeTab,
453 ctxt->nodeMax * 2 * sizeof(ctxt->nodeTab[0]));
454 if (tmp == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000455 xmlVErrMemory(ctxt, "realloc failed");
Daniel Veillard1c732d22002-11-30 11:22:59 +0000456 return (0);
457 }
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000458 ctxt->nodeMax *= 2;
459 ctxt->nodeTab = tmp;
Daniel Veillard1c732d22002-11-30 11:22:59 +0000460 }
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)
Daniel Veillard24505b02005-07-28 23:49:35 +0000471 return (NULL);
Daniel Veillard1c732d22002-11-30 11:22:59 +0000472 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];
Daniel Veillard24505b02005-07-28 23:49:35 +0000478 ctxt->nodeTab[ctxt->nodeNr] = NULL;
Daniel Veillard1c732d22002-11-30 11:22:59 +0000479 return (ret);
480}
Owen Taylor3473f882001-02-23 17:55:21 +0000481
Owen Taylor3473f882001-02-23 17:55:21 +0000482#ifdef DEBUG_VALID_ALGO
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000483static 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;
Daniel Veillardce2c2f02001-10-18 14:57:24 +0000529#ifdef LIBXML_DOCB_ENABLED
530 case XML_DOCB_DOCUMENT_NODE:
531 xmlGenericError(xmlGenericErrorContext, "?docb? ");
532 break;
533#endif
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000534 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) {
Owen Taylor3473f882001-02-23 17:55:21 +0000560 if (cur == NULL)
561 xmlGenericError(xmlGenericErrorContext, "null ");
562 while (cur != NULL) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000563 xmlValidPrintNode(cur);
Owen Taylor3473f882001-02-23 17:55:21 +0000564 cur = cur->next;
565 }
566}
567
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000568static void
569xmlValidDebug(xmlNodePtr cur, xmlElementContentPtr cont) {
Daniel Veillard83391282003-03-06 21:37:30 +0000570 char expr[5000];
Owen Taylor3473f882001-02-23 17:55:21 +0000571
572 expr[0] = 0;
573 xmlGenericError(xmlGenericErrorContext, "valid: ");
574 xmlValidPrintNodeList(cur);
575 xmlGenericError(xmlGenericErrorContext, "against ");
Daniel Veillardd3d06722001-08-15 12:06:36 +0000576 xmlSnprintfElementContent(expr, 5000, cont, 1);
Owen Taylor3473f882001-02-23 17:55:21 +0000577 xmlGenericError(xmlGenericErrorContext, "%s\n", expr);
578}
579
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000580static 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/*****
Owen Taylor3473f882001-02-23 17:55:21 +0000620#define DEBUG_VALID_STATE(n,c) xmlValidDebug(n,c);
Daniel Veillarddab4cb32001-04-20 13:03:48 +0000621 *****/
622
623#define DEBUG_VALID_STATE(n,c) xmlValidStateDebug(ctxt);
Daniel Veillarde356c282001-03-10 12:32:04 +0000624#define DEBUG_VALID_MSG(m) \
625 xmlGenericError(xmlGenericErrorContext, "%s\n", m);
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800626
Owen Taylor3473f882001-02-23 17:55:21 +0000627#else
628#define DEBUG_VALID_STATE(n,c)
Daniel Veillarde356c282001-03-10 12:32:04 +0000629#define DEBUG_VALID_MSG(m)
Owen Taylor3473f882001-02-23 17:55:21 +0000630#endif
631
Daniel Veillardcbaf3992001-12-31 16:16:02 +0000632/* TODO: use hash table for accesses to elem and attribute definitions */
Owen Taylor3473f882001-02-23 17:55:21 +0000633
Daniel Veillardb9cd8b42002-09-05 10:58:49 +0000634
Owen Taylor3473f882001-02-23 17:55:21 +0000635#define CHECK_DTD \
636 if (doc == NULL) return(0); \
637 else if ((doc->intSubset == NULL) && \
638 (doc->extSubset == NULL)) return(0)
639
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000640#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 *
Daniel Veillard84d70a42002-09-16 10:51:38 +0000656 * Returns 1 if successful or 0 in case of error.
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000657 */
658static int
659xmlValidBuildAContentModel(xmlElementContentPtr content,
660 xmlValidCtxtPtr ctxt,
661 const xmlChar *name) {
662 if (content == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000663 xmlErrValidNode(ctxt, NULL, XML_ERR_INTERNAL_ERROR,
664 "Found NULL content in content model of %s\n",
665 name, NULL, NULL);
Daniel Veillard84d70a42002-09-16 10:51:38 +0000666 return(0);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000667 }
668 switch (content->type) {
669 case XML_ELEMENT_CONTENT_PCDATA:
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000670 xmlErrValidNode(ctxt, NULL, XML_ERR_INTERNAL_ERROR,
671 "Found PCDATA in content model of %s\n",
672 name, NULL, NULL);
Daniel Veillard84d70a42002-09-16 10:51:38 +0000673 return(0);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000674 break;
675 case XML_ELEMENT_CONTENT_ELEMENT: {
676 xmlAutomataStatePtr oldstate = ctxt->state;
Daniel Veillardc00cda82003-04-07 10:22:39 +0000677 xmlChar fn[50];
678 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800679
Daniel Veillardc00cda82003-04-07 10:22:39 +0000680 fullname = xmlBuildQName(content->name, content->prefix, fn, 50);
681 if (fullname == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000682 xmlVErrMemory(ctxt, "Building content model");
Daniel Veillardc00cda82003-04-07 10:22:39 +0000683 return(0);
Daniel Veillarda646cfd2002-09-17 21:50:03 +0000684 }
685
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000686 switch (content->ocur) {
687 case XML_ELEMENT_CONTENT_ONCE:
688 ctxt->state = xmlAutomataNewTransition(ctxt->am,
Daniel Veillardc00cda82003-04-07 10:22:39 +0000689 ctxt->state, NULL, fullname, NULL);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000690 break;
691 case XML_ELEMENT_CONTENT_OPT:
692 ctxt->state = xmlAutomataNewTransition(ctxt->am,
Daniel Veillardc00cda82003-04-07 10:22:39 +0000693 ctxt->state, NULL, fullname, NULL);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000694 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
695 break;
696 case XML_ELEMENT_CONTENT_PLUS:
697 ctxt->state = xmlAutomataNewTransition(ctxt->am,
Daniel Veillardc00cda82003-04-07 10:22:39 +0000698 ctxt->state, NULL, fullname, NULL);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000699 xmlAutomataNewTransition(ctxt->am, ctxt->state,
Daniel Veillardc00cda82003-04-07 10:22:39 +0000700 ctxt->state, fullname, NULL);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000701 break;
702 case XML_ELEMENT_CONTENT_MULT:
William M. Brack8b0cbb02004-04-17 13:31:06 +0000703 ctxt->state = xmlAutomataNewEpsilon(ctxt->am,
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800704 ctxt->state, NULL);
William M. Brack8b0cbb02004-04-17 13:31:06 +0000705 xmlAutomataNewTransition(ctxt->am,
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800706 ctxt->state, ctxt->state, fullname, NULL);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000707 break;
708 }
Daniel Veillardc00cda82003-04-07 10:22:39 +0000709 if ((fullname != fn) && (fullname != content->name))
710 xmlFree(fullname);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000711 break;
712 }
713 case XML_ELEMENT_CONTENT_SEQ: {
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000714 xmlAutomataStatePtr oldstate, oldend;
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000715 xmlElementContentOccur ocur;
716
717 /*
718 * Simply iterate over the content
719 */
720 oldstate = ctxt->state;
721 ocur = content->ocur;
Daniel Veillardf4be0182003-02-24 19:54:33 +0000722 if (ocur != XML_ELEMENT_CONTENT_ONCE) {
723 ctxt->state = xmlAutomataNewEpsilon(ctxt->am, oldstate, NULL);
724 oldstate = ctxt->state;
725 }
Daniel Veillarda646cfd2002-09-17 21:50:03 +0000726 do {
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000727 xmlValidBuildAContentModel(content->c1, ctxt, name);
728 content = content->c2;
Daniel Veillarda646cfd2002-09-17 21:50:03 +0000729 } while ((content->type == XML_ELEMENT_CONTENT_SEQ) &&
730 (content->ocur == XML_ELEMENT_CONTENT_ONCE));
731 xmlValidBuildAContentModel(content, ctxt, name);
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000732 oldend = ctxt->state;
733 ctxt->state = xmlAutomataNewEpsilon(ctxt->am, oldend, NULL);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000734 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);
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000742 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000743 break;
744 case XML_ELEMENT_CONTENT_PLUS:
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000745 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000746 break;
747 }
748 break;
749 }
750 case XML_ELEMENT_CONTENT_OR: {
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000751 xmlAutomataStatePtr oldstate, oldend;
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000752 xmlElementContentOccur ocur;
753
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000754 ocur = content->ocur;
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800755 if ((ocur == XML_ELEMENT_CONTENT_PLUS) ||
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000756 (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);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000762
763 /*
764 * iterate over the subtypes and remerge the end with an
765 * epsilon transition
766 */
Daniel Veillarda646cfd2002-09-17 21:50:03 +0000767 do {
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000768 ctxt->state = oldstate;
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000769 xmlValidBuildAContentModel(content->c1, ctxt, name);
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000770 xmlAutomataNewEpsilon(ctxt->am, ctxt->state, oldend);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000771 content = content->c2;
Daniel Veillarda646cfd2002-09-17 21:50:03 +0000772 } while ((content->type == XML_ELEMENT_CONTENT_OR) &&
773 (content->ocur == XML_ELEMENT_CONTENT_ONCE));
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000774 ctxt->state = oldstate;
Daniel Veillarda646cfd2002-09-17 21:50:03 +0000775 xmlValidBuildAContentModel(content, ctxt, name);
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000776 xmlAutomataNewEpsilon(ctxt->am, ctxt->state, oldend);
777 ctxt->state = xmlAutomataNewEpsilon(ctxt->am, oldend, NULL);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000778 switch (ocur) {
779 case XML_ELEMENT_CONTENT_ONCE:
780 break;
781 case XML_ELEMENT_CONTENT_OPT:
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000782 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000783 break;
784 case XML_ELEMENT_CONTENT_MULT:
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000785 xmlAutomataNewEpsilon(ctxt->am, oldstate, ctxt->state);
786 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000787 break;
788 case XML_ELEMENT_CONTENT_PLUS:
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000789 xmlAutomataNewEpsilon(ctxt->am, oldend, oldstate);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000790 break;
791 }
792 break;
793 }
794 default:
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000795 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
796 "ContentModel broken for element %s\n",
797 (const char *) name);
Daniel Veillard84d70a42002-09-16 10:51:38 +0000798 return(0);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000799 }
Daniel Veillard84d70a42002-09-16 10:51:38 +0000800 return(1);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000801}
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 *
Daniel Veillard84d70a42002-09-16 10:51:38 +0000810 * Returns 1 in case of success, 0 in case of error
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000811 */
812int
813xmlValidBuildContentModel(xmlValidCtxtPtr ctxt, xmlElementPtr elem) {
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000814
815 if ((ctxt == NULL) || (elem == NULL))
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000816 return(0);
Daniel Veillard84d70a42002-09-16 10:51:38 +0000817 if (elem->type != XML_ELEMENT_DECL)
818 return(0);
819 if (elem->etype != XML_ELEMENT_TYPE_ELEMENT)
820 return(1);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000821 /* TODO: should we rebuild in this case ? */
Daniel Veillardec498e12003-02-05 11:01:50 +0000822 if (elem->contModel != NULL) {
823 if (!xmlRegexpIsDeterminist(elem->contModel)) {
824 ctxt->valid = 0;
825 return(0);
826 }
Daniel Veillard84d70a42002-09-16 10:51:38 +0000827 return(1);
Daniel Veillardec498e12003-02-05 11:01:50 +0000828 }
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000829
830 ctxt->am = xmlNewAutomata();
831 if (ctxt->am == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000832 xmlErrValidNode(ctxt, (xmlNodePtr) elem,
833 XML_ERR_INTERNAL_ERROR,
834 "Cannot create automata for element %s\n",
835 elem->name, NULL, NULL);
Daniel Veillard84d70a42002-09-16 10:51:38 +0000836 return(0);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000837 }
William M. Brack78637da2003-07-31 14:47:38 +0000838 ctxt->state = xmlAutomataGetInitState(ctxt->am);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000839 xmlValidBuildAContentModel(elem->content, ctxt, elem->name);
840 xmlAutomataSetFinalState(ctxt->am, ctxt->state);
Daniel Veillard84d70a42002-09-16 10:51:38 +0000841 elem->contModel = xmlAutomataCompile(ctxt->am);
Daniel Veillarda76fe5c2003-04-24 16:06:47 +0000842 if (xmlRegexpIsDeterminist(elem->contModel) != 1) {
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000843 char expr[5000];
844 expr[0] = 0;
845 xmlSnprintfElementContent(expr, 5000, elem->content, 1);
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000846 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);
Daniel Veillard5acfd6b2002-09-18 16:29:02 +0000850#ifdef DEBUG_REGEXP_ALGO
851 xmlRegexpPrint(stderr, elem->contModel);
852#endif
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000853 ctxt->valid = 0;
Daniel Veillardec498e12003-02-05 11:01:50 +0000854 ctxt->state = NULL;
855 xmlFreeAutomata(ctxt->am);
856 ctxt->am = NULL;
857 return(0);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000858 }
859 ctxt->state = NULL;
860 xmlFreeAutomata(ctxt->am);
861 ctxt->am = NULL;
Daniel Veillard84d70a42002-09-16 10:51:38 +0000862 return(1);
Daniel Veillardaeb258a2002-09-13 14:48:12 +0000863}
864
865#endif /* LIBXML_REGEXP_ENABLED */
866
Owen Taylor3473f882001-02-23 17:55:21 +0000867/****************************************************************
868 * *
869 * Util functions for data allocation/deallocation *
870 * *
871 ****************************************************************/
872
873/**
Daniel Veillarda37aab82003-06-09 09:10:36 +0000874 * xmlNewValidCtxt:
875 *
876 * Allocate a validation context structure.
877 *
878 * Returns NULL if not, otherwise the new validation context structure
879 */
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000880xmlValidCtxtPtr xmlNewValidCtxt(void) {
Daniel Veillarda37aab82003-06-09 09:10:36 +0000881 xmlValidCtxtPtr ret;
882
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000883 if ((ret = xmlMalloc(sizeof (xmlValidCtxt))) == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000884 xmlVErrMemory(NULL, "malloc failed");
Daniel Veillarda37aab82003-06-09 09:10:36 +0000885 return (NULL);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000886 }
Daniel Veillarda37aab82003-06-09 09:10:36 +0000887
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) {
Daniel Veillardc0be74b2004-11-03 19:16:55 +0000901 if (cur->vstateTab != NULL)
902 xmlFree(cur->vstateTab);
903 if (cur->nodeTab != NULL)
904 xmlFree(cur->nodeTab);
Daniel Veillarda37aab82003-06-09 09:10:36 +0000905 xmlFree(cur);
906}
907
Daniel Veillard4432df22003-09-28 18:58:27 +0000908#endif /* LIBXML_VALID_ENABLED */
909
Daniel Veillarda37aab82003-06-09 09:10:36 +0000910/**
Daniel Veillardcee2b3a2005-01-25 00:22:52 +0000911 * xmlNewDocElementContent:
912 * @doc: the document
Owen Taylor3473f882001-02-23 17:55:21 +0000913 * @name: the subelement name or NULL
914 * @type: the type of element content decl
915 *
Daniel Veillardcee2b3a2005-01-25 00:22:52 +0000916 * Allocate an element content structure for the document.
Owen Taylor3473f882001-02-23 17:55:21 +0000917 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +0000918 * Returns NULL if not, otherwise the new element content structure
Owen Taylor3473f882001-02-23 17:55:21 +0000919 */
920xmlElementContentPtr
Daniel Veillardcee2b3a2005-01-25 00:22:52 +0000921xmlNewDocElementContent(xmlDocPtr doc, const xmlChar *name,
922 xmlElementContentType type) {
Owen Taylor3473f882001-02-23 17:55:21 +0000923 xmlElementContentPtr ret;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +0000924 xmlDictPtr dict = NULL;
925
926 if (doc != NULL)
927 dict = doc->dict;
Owen Taylor3473f882001-02-23 17:55:21 +0000928
929 switch(type) {
930 case XML_ELEMENT_CONTENT_ELEMENT:
931 if (name == NULL) {
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000932 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
933 "xmlNewElementContent : name == NULL !\n",
934 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +0000935 }
936 break;
937 case XML_ELEMENT_CONTENT_PCDATA:
938 case XML_ELEMENT_CONTENT_SEQ:
939 case XML_ELEMENT_CONTENT_OR:
940 if (name != NULL) {
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000941 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
942 "xmlNewElementContent : name != NULL !\n",
943 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +0000944 }
945 break;
946 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800947 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +0000948 "Internal: ELEMENT content corrupted invalid type\n",
949 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +0000950 return(NULL);
951 }
952 ret = (xmlElementContentPtr) xmlMalloc(sizeof(xmlElementContent));
953 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +0000954 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +0000955 return(NULL);
956 }
Daniel Veillardd54fa3e2002-02-20 16:48:52 +0000957 memset(ret, 0, sizeof(xmlElementContent));
Owen Taylor3473f882001-02-23 17:55:21 +0000958 ret->type = type;
959 ret->ocur = XML_ELEMENT_CONTENT_ONCE;
Daniel Veillardbe480fb2001-11-08 23:36:42 +0000960 if (name != NULL) {
Daniel Veillardcee2b3a2005-01-25 00:22:52 +0000961 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 }
Daniel Veillardbe480fb2001-11-08 23:36:42 +0000979 }
Daniel Veillardcee2b3a2005-01-25 00:22:52 +0000980 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.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001004 *
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001005 * 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 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001031
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001032 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 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001061
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001062 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 }
Owen Taylor3473f882001-02-23 17:55:21 +00001076 return(ret);
1077}
1078
1079/**
1080 * xmlCopyElementContent:
Daniel Veillard01c13b52002-12-10 15:19:08 +00001081 * @cur: An element content pointer.
Owen Taylor3473f882001-02-23 17:55:21 +00001082 *
1083 * Build a copy of an element content description.
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001084 * Deprecated, use xmlCopyDocElementContent instead
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001085 *
Owen Taylor3473f882001-02-23 17:55:21 +00001086 * Returns the new xmlElementContentPtr or NULL in case of error.
1087 */
1088xmlElementContentPtr
1089xmlCopyElementContent(xmlElementContentPtr cur) {
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001090 return(xmlCopyDocElementContent(NULL, cur));
1091}
Owen Taylor3473f882001-02-23 17:55:21 +00001092
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001093/**
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:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001117 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001118 "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;
Owen Taylor3473f882001-02-23 17:55:21 +00001134 }
Owen Taylor3473f882001-02-23 17:55:21 +00001135}
1136
1137/**
1138 * xmlFreeElementContent:
1139 * @cur: the element content tree to free
1140 *
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001141 * Free an element content structure. The whole subtree is removed.
1142 * Deprecated, use xmlFreeDocElementContent instead
Owen Taylor3473f882001-02-23 17:55:21 +00001143 */
1144void
1145xmlFreeElementContent(xmlElementContentPtr cur) {
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001146 xmlFreeDocElementContent(NULL, cur);
Owen Taylor3473f882001-02-23 17:55:21 +00001147}
1148
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001149#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001150/**
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 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001158static void
Owen Taylor3473f882001-02-23 17:55:21 +00001159xmlDumpElementContent(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:
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001168 if (content->prefix != NULL) {
1169 xmlBufferWriteCHAR(buf, content->prefix);
1170 xmlBufferWriteChar(buf, ":");
1171 }
Owen Taylor3473f882001-02-23 17:55:21 +00001172 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, " , ");
William M. Brack4119d1c2004-06-24 02:24:44 +00001181 if ((content->c2->type == XML_ELEMENT_CONTENT_OR) ||
1182 ((content->c2->type == XML_ELEMENT_CONTENT_SEQ) &&
1183 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)))
Owen Taylor3473f882001-02-23 17:55:21 +00001184 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, " | ");
William M. Brack4119d1c2004-06-24 02:24:44 +00001195 if ((content->c2->type == XML_ELEMENT_CONTENT_SEQ) ||
1196 ((content->c2->type == XML_ELEMENT_CONTENT_OR) &&
1197 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)))
Owen Taylor3473f882001-02-23 17:55:21 +00001198 xmlDumpElementContent(buf, content->c2, 1);
1199 else
1200 xmlDumpElementContent(buf, content->c2, 0);
1201 break;
1202 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001203 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001204 "Internal: ELEMENT content corrupted invalid type\n",
1205 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001206 }
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
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001228 * @englob: 1 if one must print the englobing parenthesis, 0 otherwise
Owen Taylor3473f882001-02-23 17:55:21 +00001229 *
Daniel Veillardd3d06722001-08-15 12:06:36 +00001230 * Deprecated, unsafe, use xmlSnprintfElementContent
1231 */
1232void
1233xmlSprintfElementContent(char *buf ATTRIBUTE_UNUSED,
1234 xmlElementContentPtr content ATTRIBUTE_UNUSED,
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001235 int englob ATTRIBUTE_UNUSED) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00001236}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001237#endif /* LIBXML_OUTPUT_ENABLED */
Daniel Veillardd3d06722001-08-15 12:06:36 +00001238
1239/**
1240 * xmlSnprintfElementContent:
1241 * @buf: an output buffer
1242 * @size: the buffer size
1243 * @content: An element table
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001244 * @englob: 1 if one must print the englobing parenthesis, 0 otherwise
Daniel Veillardd3d06722001-08-15 12:06:36 +00001245 *
Owen Taylor3473f882001-02-23 17:55:21 +00001246 * This will dump the content of the element content definition
1247 * Intended just for the debug routine
1248 */
1249void
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001250xmlSnprintfElementContent(char *buf, int size, xmlElementContentPtr content, int englob) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00001251 int len;
1252
Owen Taylor3473f882001-02-23 17:55:21 +00001253 if (content == NULL) return;
Daniel Veillardd3d06722001-08-15 12:06:36 +00001254 len = strlen(buf);
1255 if (size - len < 50) {
1256 if ((size - len > 4) && (buf[len - 1] != '.'))
1257 strcat(buf, " ...");
1258 return;
1259 }
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001260 if (englob) strcat(buf, "(");
Owen Taylor3473f882001-02-23 17:55:21 +00001261 switch (content->type) {
1262 case XML_ELEMENT_CONTENT_PCDATA:
1263 strcat(buf, "#PCDATA");
1264 break;
1265 case XML_ELEMENT_CONTENT_ELEMENT:
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001266 if (content->prefix != NULL) {
Daniel Veillard4b3a84f2002-03-19 14:36:46 +00001267 if (size - len < xmlStrlen(content->prefix) + 10) {
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001268 strcat(buf, " ...");
1269 return;
1270 }
1271 strcat(buf, (char *) content->prefix);
1272 strcat(buf, ":");
1273 }
Daniel Veillard4b3a84f2002-03-19 14:36:46 +00001274 if (size - len < xmlStrlen(content->name) + 10) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00001275 strcat(buf, " ...");
1276 return;
1277 }
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001278 if (content->name != NULL)
1279 strcat(buf, (char *) content->name);
Owen Taylor3473f882001-02-23 17:55:21 +00001280 break;
1281 case XML_ELEMENT_CONTENT_SEQ:
1282 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1283 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001284 xmlSnprintfElementContent(buf, size, content->c1, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001285 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001286 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 }
Owen Taylor3473f882001-02-23 17:55:21 +00001293 strcat(buf, " , ");
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00001294 if (((content->c2->type == XML_ELEMENT_CONTENT_OR) ||
1295 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)) &&
1296 (content->c2->type != XML_ELEMENT_CONTENT_ELEMENT))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001297 xmlSnprintfElementContent(buf, size, content->c2, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001298 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001299 xmlSnprintfElementContent(buf, size, content->c2, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00001300 break;
1301 case XML_ELEMENT_CONTENT_OR:
1302 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1303 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001304 xmlSnprintfElementContent(buf, size, content->c1, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001305 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001306 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 }
Owen Taylor3473f882001-02-23 17:55:21 +00001313 strcat(buf, " | ");
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00001314 if (((content->c2->type == XML_ELEMENT_CONTENT_SEQ) ||
1315 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)) &&
1316 (content->c2->type != XML_ELEMENT_CONTENT_ELEMENT))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001317 xmlSnprintfElementContent(buf, size, content->c2, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001318 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001319 xmlSnprintfElementContent(buf, size, content->c2, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00001320 break;
1321 }
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001322 if (englob)
Owen Taylor3473f882001-02-23 17:55:21 +00001323 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/**
Owen Taylor3473f882001-02-23 17:55:21 +00001346 * xmlFreeElement:
1347 * @elem: An element
1348 *
1349 * Deallocate the memory used by an element definition
1350 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001351static void
Owen Taylor3473f882001-02-23 17:55:21 +00001352xmlFreeElement(xmlElementPtr elem) {
1353 if (elem == NULL) return;
1354 xmlUnlinkNode((xmlNodePtr) elem);
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001355 xmlFreeDocElementContent(elem->doc, elem->content);
Owen Taylor3473f882001-02-23 17:55:21 +00001356 if (elem->name != NULL)
1357 xmlFree((xmlChar *) elem->name);
1358 if (elem->prefix != NULL)
1359 xmlFree((xmlChar *) elem->prefix);
Daniel Veillard84d70a42002-09-16 10:51:38 +00001360#ifdef LIBXML_REGEXP_ENABLED
1361 if (elem->contModel != NULL)
1362 xmlRegFreeRegexp(elem->contModel);
1363#endif
Owen Taylor3473f882001-02-23 17:55:21 +00001364 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001378 * Returns NULL if not, otherwise the entity
Owen Taylor3473f882001-02-23 17:55:21 +00001379 */
1380xmlElementPtr
William M. Brackedb65a72004-02-06 07:36:04 +00001381xmlAddElementDecl(xmlValidCtxtPtr ctxt,
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001382 xmlDtdPtr dtd, const xmlChar *name,
Owen Taylor3473f882001-02-23 17:55:21 +00001383 xmlElementTypeVal type,
1384 xmlElementContentPtr content) {
1385 xmlElementPtr ret;
1386 xmlElementTablePtr table;
Daniel Veillarda10efa82001-04-18 13:09:01 +00001387 xmlAttributePtr oldAttributes = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00001388 xmlChar *ns, *uqname;
1389
1390 if (dtd == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001391 return(NULL);
1392 }
1393 if (name == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001394 return(NULL);
1395 }
Daniel Veillard316a5c32005-01-23 22:56:39 +00001396
Owen Taylor3473f882001-02-23 17:55:21 +00001397 switch (type) {
1398 case XML_ELEMENT_TYPE_EMPTY:
1399 if (content != NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001400 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001401 "xmlAddElementDecl: content != NULL for EMPTY\n",
1402 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001403 return(NULL);
1404 }
1405 break;
1406 case XML_ELEMENT_TYPE_ANY:
1407 if (content != NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001408 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001409 "xmlAddElementDecl: content != NULL for ANY\n",
1410 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001411 return(NULL);
1412 }
1413 break;
1414 case XML_ELEMENT_TYPE_MIXED:
1415 if (content == NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001416 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001417 "xmlAddElementDecl: content == NULL for MIXED\n",
1418 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001419 return(NULL);
1420 }
1421 break;
1422 case XML_ELEMENT_TYPE_ELEMENT:
1423 if (content == NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001424 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001425 "xmlAddElementDecl: content == NULL for ELEMENT\n",
1426 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001427 return(NULL);
1428 }
1429 break;
1430 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001431 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001432 "Internal: ELEMENT decl corrupted invalid type\n",
1433 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001434 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) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00001449 xmlDictPtr dict = NULL;
1450
1451 if (dtd->doc != NULL)
1452 dict = dtd->doc->dict;
1453 table = xmlHashCreateDict(0, dict);
Owen Taylor3473f882001-02-23 17:55:21 +00001454 dtd->elements = (void *) table;
1455 }
1456 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001457 xmlVErrMemory(ctxt,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001458 "xmlAddElementDecl: Table creation failed!\n");
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001459 if (uqname != NULL)
1460 xmlFree(uqname);
1461 if (ns != NULL)
1462 xmlFree(ns);
Owen Taylor3473f882001-02-23 17:55:21 +00001463 return(NULL);
1464 }
1465
Daniel Veillarda10efa82001-04-18 13:09:01 +00001466 /*
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 }
Owen Taylor3473f882001-02-23 17:55:21 +00001478 }
Owen Taylor3473f882001-02-23 17:55:21 +00001479
1480 /*
Daniel Veillarda10efa82001-04-18 13:09:01 +00001481 * 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) {
Daniel Veillard4432df22003-09-28 18:58:27 +00001487#ifdef LIBXML_VALID_ENABLED
Daniel Veillarda10efa82001-04-18 13:09:01 +00001488 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001489 * The element is already defined in this DTD.
Daniel Veillarda10efa82001-04-18 13:09:01 +00001490 */
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001491 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ELEM_REDEFINED,
1492 "Redefinition of element %s\n",
1493 name, NULL, NULL);
Daniel Veillard4432df22003-09-28 18:58:27 +00001494#endif /* LIBXML_VALID_ENABLED */
Daniel Veillarda10efa82001-04-18 13:09:01 +00001495 if (uqname != NULL)
1496 xmlFree(uqname);
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001497 if (ns != NULL)
1498 xmlFree(ns);
Daniel Veillarda10efa82001-04-18 13:09:01 +00001499 return(NULL);
1500 }
William M. Brackd6e347e2005-04-15 01:34:41 +00001501 if (ns != NULL) {
1502 xmlFree(ns);
1503 ns = NULL;
1504 }
Daniel Veillarda10efa82001-04-18 13:09:01 +00001505 } else {
1506 ret = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
1507 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001508 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001509 if (uqname != NULL)
1510 xmlFree(uqname);
1511 if (ns != NULL)
1512 xmlFree(ns);
Daniel Veillarda10efa82001-04-18 13:09:01 +00001513 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);
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001522 if (ret->name == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001523 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001524 if (uqname != NULL)
1525 xmlFree(uqname);
1526 if (ns != NULL)
1527 xmlFree(ns);
1528 xmlFree(ret);
1529 return(NULL);
1530 }
Daniel Veillarda10efa82001-04-18 13:09:01 +00001531 ret->prefix = ns;
1532
1533 /*
1534 * Validity Check:
1535 * Insertion must not fail
1536 */
1537 if (xmlHashAddEntry2(table, name, ns, ret)) {
Daniel Veillard4432df22003-09-28 18:58:27 +00001538#ifdef LIBXML_VALID_ENABLED
Daniel Veillarda10efa82001-04-18 13:09:01 +00001539 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001540 * The element is already defined in this DTD.
Daniel Veillarda10efa82001-04-18 13:09:01 +00001541 */
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001542 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ELEM_REDEFINED,
1543 "Redefinition of element %s\n",
1544 name, NULL, NULL);
Daniel Veillard4432df22003-09-28 18:58:27 +00001545#endif /* LIBXML_VALID_ENABLED */
Daniel Veillarda10efa82001-04-18 13:09:01 +00001546 xmlFreeElement(ret);
1547 if (uqname != NULL)
1548 xmlFree(uqname);
1549 return(NULL);
1550 }
William M. Brack4e52f2f2003-09-14 18:07:39 +00001551 /*
1552 * For new element, may have attributes from earlier
1553 * definition in internal subset
1554 */
1555 ret->attributes = oldAttributes;
Daniel Veillarda10efa82001-04-18 13:09:01 +00001556 }
1557
1558 /*
1559 * Finish to fill the structure.
Owen Taylor3473f882001-02-23 17:55:21 +00001560 */
1561 ret->etype = type;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001562 /*
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 */
Daniel Veillardc394f732005-01-26 00:04:52 +00001567 if ((ctxt != NULL) &&
1568 ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
1569 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1))) {
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001570 ret->content = content;
1571 if (content != NULL)
1572 content->parent = (xmlElementContentPtr) 1;
1573 } else {
1574 ret->content = xmlCopyDocElementContent(dtd->doc, content);
1575 }
Owen Taylor3473f882001-02-23 17:55:21 +00001576
1577 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001578 * Link it to the DTD
Owen Taylor3473f882001-02-23 17:55:21 +00001579 */
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
Daniel Veillard652327a2003-09-29 18:02:38 +00001605#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001606/**
1607 * xmlCopyElement:
1608 * @elem: An element
1609 *
1610 * Build a copy of an element.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001611 *
Owen Taylor3473f882001-02-23 17:55:21 +00001612 * Returns the new xmlElementPtr or NULL in case of error.
1613 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001614static xmlElementPtr
Owen Taylor3473f882001-02-23 17:55:21 +00001615xmlCopyElement(xmlElementPtr elem) {
1616 xmlElementPtr cur;
1617
1618 cur = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
1619 if (cur == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001620 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00001621 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.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001645 *
Owen Taylor3473f882001-02-23 17:55:21 +00001646 * Returns the new xmlElementTablePtr or NULL in case of error.
1647 */
1648xmlElementTablePtr
1649xmlCopyElementTable(xmlElementTablePtr table) {
1650 return((xmlElementTablePtr) xmlHashCopy(table,
1651 (xmlHashCopier) xmlCopyElement));
1652}
Daniel Veillard652327a2003-09-29 18:02:38 +00001653#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001654
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001655#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001656/**
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) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00001666 if ((buf == NULL) || (elem == NULL))
1667 return;
Owen Taylor3473f882001-02-23 17:55:21 +00001668 switch (elem->etype) {
1669 case XML_ELEMENT_TYPE_EMPTY:
1670 xmlBufferWriteChar(buf, "<!ELEMENT ");
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001671 if (elem->prefix != NULL) {
1672 xmlBufferWriteCHAR(buf, elem->prefix);
1673 xmlBufferWriteChar(buf, ":");
1674 }
Owen Taylor3473f882001-02-23 17:55:21 +00001675 xmlBufferWriteCHAR(buf, elem->name);
1676 xmlBufferWriteChar(buf, " EMPTY>\n");
1677 break;
1678 case XML_ELEMENT_TYPE_ANY:
1679 xmlBufferWriteChar(buf, "<!ELEMENT ");
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001680 if (elem->prefix != NULL) {
1681 xmlBufferWriteCHAR(buf, elem->prefix);
1682 xmlBufferWriteChar(buf, ":");
1683 }
Owen Taylor3473f882001-02-23 17:55:21 +00001684 xmlBufferWriteCHAR(buf, elem->name);
1685 xmlBufferWriteChar(buf, " ANY>\n");
1686 break;
1687 case XML_ELEMENT_TYPE_MIXED:
1688 xmlBufferWriteChar(buf, "<!ELEMENT ");
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001689 if (elem->prefix != NULL) {
1690 xmlBufferWriteCHAR(buf, elem->prefix);
1691 xmlBufferWriteChar(buf, ":");
1692 }
Owen Taylor3473f882001-02-23 17:55:21 +00001693 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 ");
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001700 if (elem->prefix != NULL) {
1701 xmlBufferWriteCHAR(buf, elem->prefix);
1702 xmlBufferWriteChar(buf, ":");
1703 }
Owen Taylor3473f882001-02-23 17:55:21 +00001704 xmlBufferWriteCHAR(buf, elem->name);
1705 xmlBufferWriteChar(buf, " ");
1706 xmlDumpElementContent(buf, elem->content, 1);
1707 xmlBufferWriteChar(buf, ">\n");
1708 break;
1709 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001710 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001711 "Internal: ELEMENT struct corrupted invalid type\n",
1712 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001713 }
1714}
1715
1716/**
William M. Brack9e660592003-10-20 14:56:06 +00001717 * 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/**
Owen Taylor3473f882001-02-23 17:55:21 +00001730 * 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) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00001738 if ((buf == NULL) || (table == NULL))
1739 return;
William M. Brack9e660592003-10-20 14:56:06 +00001740 xmlHashScan(table, (xmlHashScanner) xmlDumpElementDeclScan, buf);
Owen Taylor3473f882001-02-23 17:55:21 +00001741}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001742#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001743
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
Daniel Veillard2fdbd322003-08-18 12:15:38 +00001754xmlCreateEnumeration(const xmlChar *name) {
Owen Taylor3473f882001-02-23 17:55:21 +00001755 xmlEnumerationPtr ret;
1756
1757 ret = (xmlEnumerationPtr) xmlMalloc(sizeof(xmlEnumeration));
1758 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001759 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00001760 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);
Owen Taylor3473f882001-02-23 17:55:21 +00001782 xmlFree(cur);
1783}
1784
Daniel Veillard652327a2003-09-29 18:02:38 +00001785#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001786/**
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);
Gaurav Gupta658b86c2014-08-07 11:19:03 +08001801 if (ret == NULL) return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001802
1803 if (cur->next != NULL) ret->next = xmlCopyEnumeration(cur->next);
1804 else ret->next = NULL;
1805
1806 return(ret);
1807}
Daniel Veillard652327a2003-09-29 18:02:38 +00001808#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001809
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001810#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001811/**
1812 * xmlDumpEnumeration:
1813 * @buf: the XML buffer output
1814 * @enum: An enumeration
1815 *
1816 * This will dump the content of the enumeration
1817 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001818static void
Owen Taylor3473f882001-02-23 17:55:21 +00001819xmlDumpEnumeration(xmlBufferPtr buf, xmlEnumerationPtr cur) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00001820 if ((buf == NULL) || (cur == NULL))
1821 return;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001822
Owen Taylor3473f882001-02-23 17:55:21 +00001823 xmlBufferWriteCHAR(buf, cur->name);
1824 if (cur->next == NULL)
1825 xmlBufferWriteChar(buf, ")");
1826 else {
1827 xmlBufferWriteChar(buf, " | ");
1828 xmlDumpEnumeration(buf, cur->next);
1829 }
1830}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001831#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001832
Daniel Veillard4432df22003-09-28 18:58:27 +00001833#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001834/**
Owen Taylor3473f882001-02-23 17:55:21 +00001835 * xmlScanIDAttributeDecl:
1836 * @ctxt: the validation context
1837 * @elem: the element name
Daniel Veillarddbee0f12005-06-27 13:42:57 +00001838 * @err: whether to raise errors here
Owen Taylor3473f882001-02-23 17:55:21 +00001839 *
1840 * Verify that the element don't have too many ID attributes
1841 * declared.
1842 *
1843 * Returns the number of ID attributes found.
1844 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001845static int
Daniel Veillarddbee0f12005-06-27 13:42:57 +00001846xmlScanIDAttributeDecl(xmlValidCtxtPtr ctxt, xmlElementPtr elem, int err) {
Owen Taylor3473f882001-02-23 17:55:21 +00001847 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 ++;
Daniel Veillarddbee0f12005-06-27 13:42:57 +00001855 if ((ret > 1) && (err))
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001856 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_MULTIPLE_ID,
Daniel Veillarda10efa82001-04-18 13:09:01 +00001857 "Element %s has too many ID attributes defined : %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001858 elem->name, cur->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001859 }
1860 cur = cur->nexth;
1861 }
1862 return(ret);
1863}
Daniel Veillard4432df22003-09-28 18:58:27 +00001864#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001865
1866/**
1867 * xmlFreeAttribute:
1868 * @elem: An attribute
1869 *
1870 * Deallocate the memory used by an attribute definition
1871 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001872static void
Owen Taylor3473f882001-02-23 17:55:21 +00001873xmlFreeAttribute(xmlAttributePtr attr) {
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001874 xmlDictPtr dict;
1875
Owen Taylor3473f882001-02-23 17:55:21 +00001876 if (attr == NULL) return;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001877 if (attr->doc != NULL)
1878 dict = attr->doc->dict;
1879 else
1880 dict = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00001881 xmlUnlinkNode((xmlNodePtr) attr);
1882 if (attr->tree != NULL)
1883 xmlFreeEnumeration(attr->tree);
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001884 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 }
Owen Taylor3473f882001-02-23 17:55:21 +00001904 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001923 * Returns NULL if not new, otherwise the attribute decl
Owen Taylor3473f882001-02-23 17:55:21 +00001924 */
1925xmlAttributePtr
William M. Brackedb65a72004-02-06 07:36:04 +00001926xmlAddAttributeDecl(xmlValidCtxtPtr ctxt,
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001927 xmlDtdPtr dtd, const xmlChar *elem,
Owen Taylor3473f882001-02-23 17:55:21 +00001928 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;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001934 xmlDictPtr dict = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00001935
1936 if (dtd == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001937 xmlFreeEnumeration(tree);
1938 return(NULL);
1939 }
1940 if (name == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001941 xmlFreeEnumeration(tree);
1942 return(NULL);
1943 }
1944 if (elem == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001945 xmlFreeEnumeration(tree);
1946 return(NULL);
1947 }
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001948 if (dtd->doc != NULL)
1949 dict = dtd->doc->dict;
Daniel Veillardd85f4f42002-03-25 10:48:46 +00001950
Daniel Veillard4432df22003-09-28 18:58:27 +00001951#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001952 /*
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:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001977 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001978 "Internal: ATTRIBUTE struct corrupted invalid type\n",
1979 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001980 xmlFreeEnumeration(tree);
1981 return(NULL);
1982 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001983 if ((defaultValue != NULL) &&
Daniel Veillardae0765b2008-07-31 19:54:59 +00001984 (!xmlValidateAttributeValueInternal(dtd->doc, type, defaultValue))) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001985 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ATTRIBUTE_DEFAULT,
1986 "Attribute %s of %s: invalid default value\n",
1987 elem, name, defaultValue);
Owen Taylor3473f882001-02-23 17:55:21 +00001988 defaultValue = NULL;
Daniel Veillard42595322004-11-08 10:52:06 +00001989 if (ctxt != NULL)
1990 ctxt->valid = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00001991 }
Daniel Veillard4432df22003-09-28 18:58:27 +00001992#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001993
1994 /*
Daniel Veillardd85f4f42002-03-25 10:48:46 +00001995 * 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);
Daniel Veillardae0765b2008-07-31 19:54:59 +00002002 if (ret != NULL) {
2003 xmlFreeEnumeration(tree);
Daniel Veillardd85f4f42002-03-25 10:48:46 +00002004 return(NULL);
Daniel Veillardae0765b2008-07-31 19:54:59 +00002005 }
Daniel Veillardd85f4f42002-03-25 10:48:46 +00002006 }
2007
2008 /*
Owen Taylor3473f882001-02-23 17:55:21 +00002009 * Create the Attribute table if needed.
2010 */
2011 table = (xmlAttributeTablePtr) dtd->attributes;
2012 if (table == NULL) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00002013 table = xmlHashCreateDict(0, dict);
Owen Taylor3473f882001-02-23 17:55:21 +00002014 dtd->attributes = (void *) table;
2015 }
2016 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002017 xmlVErrMemory(ctxt,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002018 "xmlAddAttributeDecl: Table creation failed!\n");
Daniel Veillardae0765b2008-07-31 19:54:59 +00002019 xmlFreeEnumeration(tree);
Owen Taylor3473f882001-02-23 17:55:21 +00002020 return(NULL);
2021 }
2022
2023
2024 ret = (xmlAttributePtr) xmlMalloc(sizeof(xmlAttribute));
2025 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002026 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillardae0765b2008-07-31 19:54:59 +00002027 xmlFreeEnumeration(tree);
Owen Taylor3473f882001-02-23 17:55:21 +00002028 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;
William M. Brackf810de02005-07-06 22:48:41 +00002037 /*
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;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00002043 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 }
Owen Taylor3473f882001-02-23 17:55:21 +00002052 ret->def = def;
2053 ret->tree = tree;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00002054 if (defaultValue != NULL) {
2055 if (dict)
2056 ret->defaultValue = xmlDictLookup(dict, defaultValue, -1);
2057 else
2058 ret->defaultValue = xmlStrdup(defaultValue);
2059 }
Owen Taylor3473f882001-02-23 17:55:21 +00002060
2061 /*
2062 * Validity Check:
2063 * Search the DTD for previous declarations of the ATTLIST
2064 */
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00002065 if (xmlHashAddEntry3(table, ret->name, ret->prefix, ret->elem, ret) < 0) {
Daniel Veillard4432df22003-09-28 18:58:27 +00002066#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002067 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002068 * The attribute is already defined in this DTD.
Owen Taylor3473f882001-02-23 17:55:21 +00002069 */
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002070 xmlErrValidWarning(ctxt, (xmlNodePtr) dtd, XML_DTD_ATTRIBUTE_REDEFINED,
Daniel Veillard58e44c92002-08-02 22:19:49 +00002071 "Attribute %s of element %s: already defined\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002072 name, elem, NULL);
Daniel Veillard4432df22003-09-28 18:58:27 +00002073#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002074 xmlFreeAttribute(ret);
2075 return(NULL);
2076 }
2077
2078 /*
2079 * Validity Check:
2080 * Multiple ID per element
2081 */
Daniel Veillarda10efa82001-04-18 13:09:01 +00002082 elemDef = xmlGetDtdElementDesc2(dtd, elem, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00002083 if (elemDef != NULL) {
Daniel Veillard48da9102001-08-07 01:10:10 +00002084
Daniel Veillard4432df22003-09-28 18:58:27 +00002085#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002086 if ((type == XML_ATTRIBUTE_ID) &&
Daniel Veillarddbee0f12005-06-27 13:42:57 +00002087 (xmlScanIDAttributeDecl(NULL, elemDef, 1) != 0)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002088 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_MULTIPLE_ID,
Owen Taylor3473f882001-02-23 17:55:21 +00002089 "Element %s has too may ID attributes defined : %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002090 elem, name, NULL);
Daniel Veillard42595322004-11-08 10:52:06 +00002091 if (ctxt != NULL)
2092 ctxt->valid = 0;
Daniel Veillardc7612992002-02-17 22:47:37 +00002093 }
Daniel Veillard4432df22003-09-28 18:58:27 +00002094#endif /* LIBXML_VALID_ENABLED */
Daniel Veillardc7612992002-02-17 22:47:37 +00002095
Daniel Veillard48da9102001-08-07 01:10:10 +00002096 /*
2097 * Insert namespace default def first they need to be
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002098 * processed first.
Daniel Veillard48da9102001-08-07 01:10:10 +00002099 */
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 }
Owen Taylor3473f882001-02-23 17:55:21 +00002124 }
2125
2126 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002127 * Link it to the DTD
Owen Taylor3473f882001-02-23 17:55:21 +00002128 */
2129 ret->parent = dtd;
Owen Taylor3473f882001-02-23 17:55:21 +00002130 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
Daniel Veillard652327a2003-09-29 18:02:38 +00002151#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002152/**
2153 * xmlCopyAttribute:
2154 * @attr: An attribute
2155 *
2156 * Build a copy of an attribute.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002157 *
Owen Taylor3473f882001-02-23 17:55:21 +00002158 * Returns the new xmlAttributePtr or NULL in case of error.
2159 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002160static xmlAttributePtr
Owen Taylor3473f882001-02-23 17:55:21 +00002161xmlCopyAttribute(xmlAttributePtr attr) {
2162 xmlAttributePtr cur;
2163
2164 cur = (xmlAttributePtr) xmlMalloc(sizeof(xmlAttribute));
2165 if (cur == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002166 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002167 return(NULL);
2168 }
2169 memset(cur, 0, sizeof(xmlAttribute));
Daniel Veillard36065812002-01-24 15:02:46 +00002170 cur->type = XML_ATTRIBUTE_DECL;
Owen Taylor3473f882001-02-23 17:55:21 +00002171 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);
Daniel Veillard36065812002-01-24 15:02:46 +00002178 if (attr->prefix != NULL)
2179 cur->prefix = xmlStrdup(attr->prefix);
Owen Taylor3473f882001-02-23 17:55:21 +00002180 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.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002190 *
Owen Taylor3473f882001-02-23 17:55:21 +00002191 * Returns the new xmlAttributeTablePtr or NULL in case of error.
2192 */
2193xmlAttributeTablePtr
2194xmlCopyAttributeTable(xmlAttributeTablePtr table) {
2195 return((xmlAttributeTablePtr) xmlHashCopy(table,
2196 (xmlHashCopier) xmlCopyAttribute));
2197}
Daniel Veillard652327a2003-09-29 18:02:38 +00002198#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002199
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002200#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002201/**
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) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002211 if ((buf == NULL) || (attr == NULL))
2212 return;
Owen Taylor3473f882001-02-23 17:55:21 +00002213 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:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002255 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002256 "Internal: ATTRIBUTE struct corrupted invalid type\n",
2257 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00002258 }
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:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002272 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002273 "Internal: ATTRIBUTE struct corrupted invalid def\n",
2274 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00002275 }
2276 if (attr->defaultValue != NULL) {
2277 xmlBufferWriteChar(buf, " ");
2278 xmlBufferWriteQuotedString(buf, attr->defaultValue);
2279 }
2280 xmlBufferWriteChar(buf, ">\n");
2281}
2282
2283/**
William M. Brack9e660592003-10-20 14:56:06 +00002284 * 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/**
Owen Taylor3473f882001-02-23 17:55:21 +00002296 * 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) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002304 if ((buf == NULL) || (table == NULL))
2305 return;
William M. Brack9e660592003-10-20 14:56:06 +00002306 xmlHashScan(table, (xmlHashScanner) xmlDumpAttributeDeclScan, buf);
Owen Taylor3473f882001-02-23 17:55:21 +00002307}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002308#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002309
2310/************************************************************************
2311 * *
2312 * NOTATIONs *
2313 * *
2314 ************************************************************************/
2315/**
Owen Taylor3473f882001-02-23 17:55:21 +00002316 * xmlFreeNotation:
2317 * @not: A notation
2318 *
2319 * Deallocate the memory used by an notation definition
2320 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002321static void
Owen Taylor3473f882001-02-23 17:55:21 +00002322xmlFreeNotation(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);
Owen Taylor3473f882001-02-23 17:55:21 +00002330 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002344 * Returns NULL if not, otherwise the entity
Owen Taylor3473f882001-02-23 17:55:21 +00002345 */
2346xmlNotationPtr
William M. Brackedb65a72004-02-06 07:36:04 +00002347xmlAddNotationDecl(xmlValidCtxtPtr ctxt, xmlDtdPtr dtd,
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002348 const xmlChar *name,
Owen Taylor3473f882001-02-23 17:55:21 +00002349 const xmlChar *PublicID, const xmlChar *SystemID) {
2350 xmlNotationPtr ret;
2351 xmlNotationTablePtr table;
2352
2353 if (dtd == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002354 return(NULL);
2355 }
2356 if (name == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002357 return(NULL);
2358 }
2359 if ((PublicID == NULL) && (SystemID == NULL)) {
Daniel Veillard7aea52d2002-02-17 23:07:47 +00002360 return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00002361 }
2362
2363 /*
2364 * Create the Notation table if needed.
2365 */
2366 table = (xmlNotationTablePtr) dtd->notations;
Daniel Veillard316a5c32005-01-23 22:56:39 +00002367 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 }
Owen Taylor3473f882001-02-23 17:55:21 +00002374 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002375 xmlVErrMemory(ctxt,
Owen Taylor3473f882001-02-23 17:55:21 +00002376 "xmlAddNotationDecl: Table creation failed!\n");
2377 return(NULL);
2378 }
2379
2380 ret = (xmlNotationPtr) xmlMalloc(sizeof(xmlNotation));
2381 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002382 xmlVErrMemory(ctxt, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002383 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)) {
Daniel Veillard4432df22003-09-28 18:58:27 +00002401#ifdef LIBXML_VALID_ENABLED
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002402 xmlErrValid(NULL, XML_DTD_NOTATION_REDEFINED,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002403 "xmlAddNotationDecl: %s already defined\n",
2404 (const char *) name);
Daniel Veillard4432df22003-09-28 18:58:27 +00002405#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002406 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
Daniel Veillard652327a2003-09-29 18:02:38 +00002423#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002424/**
2425 * xmlCopyNotation:
2426 * @nota: A notation
2427 *
2428 * Build a copy of a notation.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002429 *
Owen Taylor3473f882001-02-23 17:55:21 +00002430 * Returns the new xmlNotationPtr or NULL in case of error.
2431 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002432static xmlNotationPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002433xmlCopyNotation(xmlNotationPtr nota) {
2434 xmlNotationPtr cur;
2435
2436 cur = (xmlNotationPtr) xmlMalloc(sizeof(xmlNotation));
2437 if (cur == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002438 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002439 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.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002461 *
Owen Taylor3473f882001-02-23 17:55:21 +00002462 * Returns the new xmlNotationTablePtr or NULL in case of error.
2463 */
2464xmlNotationTablePtr
2465xmlCopyNotationTable(xmlNotationTablePtr table) {
2466 return((xmlNotationTablePtr) xmlHashCopy(table,
2467 (xmlHashCopier) xmlCopyNotation));
2468}
Daniel Veillard652327a2003-09-29 18:02:38 +00002469#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002470
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002471#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002472/**
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) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002481 if ((buf == NULL) || (nota == NULL))
2482 return;
Owen Taylor3473f882001-02-23 17:55:21 +00002483 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, " ");
Daniel Veillard41c4a752004-09-08 20:55:38 +00002490 xmlBufferWriteQuotedString(buf, nota->SystemID);
Owen Taylor3473f882001-02-23 17:55:21 +00002491 }
2492 } else {
2493 xmlBufferWriteChar(buf, " SYSTEM ");
Daniel Veillard41c4a752004-09-08 20:55:38 +00002494 xmlBufferWriteQuotedString(buf, nota->SystemID);
Owen Taylor3473f882001-02-23 17:55:21 +00002495 }
2496 xmlBufferWriteChar(buf, " >\n");
2497}
2498
2499/**
William M. Brack9e660592003-10-20 14:56:06 +00002500 * 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/**
Owen Taylor3473f882001-02-23 17:55:21 +00002512 * 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) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002520 if ((buf == NULL) || (table == NULL))
2521 return;
William M. Brack9e660592003-10-20 14:56:06 +00002522 xmlHashScan(table, (xmlHashScanner) xmlDumpNotationDeclScan, buf);
Owen Taylor3473f882001-02-23 17:55:21 +00002523}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002524#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002525
2526/************************************************************************
2527 * *
2528 * IDs *
2529 * *
2530 ************************************************************************/
2531/**
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002532 * 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) \
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002539 if ((str) && ((!dict) || \
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002540 (xmlDictOwns(dict, (const xmlChar *)(str)) == 0))) \
2541 xmlFree((char *)(str));
2542
2543/**
Owen Taylor3473f882001-02-23 17:55:21 +00002544 * xmlFreeID:
2545 * @not: A id
2546 *
2547 * Deallocate the memory used by an id definition
2548 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002549static void
Owen Taylor3473f882001-02-23 17:55:21 +00002550xmlFreeID(xmlIDPtr id) {
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002551 xmlDictPtr dict = NULL;
2552
Owen Taylor3473f882001-02-23 17:55:21 +00002553 if (id == NULL) return;
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002554
2555 if (id->doc != NULL)
2556 dict = id->doc->dict;
2557
Owen Taylor3473f882001-02-23 17:55:21 +00002558 if (id->value != NULL)
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002559 DICT_FREE(id->value)
Daniel Veillardea7751d2002-12-20 00:16:24 +00002560 if (id->name != NULL)
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002561 DICT_FREE(id->name)
Owen Taylor3473f882001-02-23 17:55:21 +00002562 xmlFree(id);
2563}
2564
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002565
Owen Taylor3473f882001-02-23 17:55:21 +00002566/**
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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002575 * Returns NULL if not, otherwise the new xmlIDPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002576 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002577xmlIDPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002578xmlAddID(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value,
2579 xmlAttrPtr attr) {
2580 xmlIDPtr ret;
2581 xmlIDTablePtr table;
2582
2583 if (doc == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002584 return(NULL);
2585 }
2586 if (value == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002587 return(NULL);
2588 }
2589 if (attr == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002590 return(NULL);
2591 }
2592
2593 /*
2594 * Create the ID table if needed.
2595 */
2596 table = (xmlIDTablePtr) doc->ids;
Daniel Veillard316a5c32005-01-23 22:56:39 +00002597 if (table == NULL) {
2598 doc->ids = table = xmlHashCreateDict(0, doc->dict);
2599 }
Owen Taylor3473f882001-02-23 17:55:21 +00002600 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002601 xmlVErrMemory(ctxt,
Owen Taylor3473f882001-02-23 17:55:21 +00002602 "xmlAddID: Table creation failed!\n");
2603 return(NULL);
2604 }
2605
2606 ret = (xmlIDPtr) xmlMalloc(sizeof(xmlID));
2607 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002608 xmlVErrMemory(ctxt, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002609 return(NULL);
2610 }
2611
2612 /*
2613 * fill the structure.
2614 */
2615 ret->value = xmlStrdup(value);
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002616 ret->doc = doc;
Daniel Veillardea7751d2002-12-20 00:16:24 +00002617 if ((ctxt != NULL) && (ctxt->vstateNr != 0)) {
2618 /*
2619 * Operating in streaming mode, attr is gonna disapear
2620 */
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002621 if (doc->dict != NULL)
2622 ret->name = xmlDictLookup(doc->dict, attr->name, -1);
2623 else
2624 ret->name = xmlStrdup(attr->name);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002625 ret->attr = NULL;
2626 } else {
2627 ret->attr = attr;
2628 ret->name = NULL;
2629 }
2630 ret->lineno = xmlGetLineNo(attr->parent);
Owen Taylor3473f882001-02-23 17:55:21 +00002631
2632 if (xmlHashAddEntry(table, value, ret) < 0) {
Daniel Veillard4432df22003-09-28 18:58:27 +00002633#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002634 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002635 * The id is already defined in this DTD.
Owen Taylor3473f882001-02-23 17:55:21 +00002636 */
Daniel Veillarda16eb962014-06-10 16:06:14 +08002637 xmlErrValidNode(ctxt, attr->parent, XML_DTD_ID_REDEFINED,
2638 "ID %s already defined\n", value, NULL, NULL);
Daniel Veillard4432df22003-09-28 18:58:27 +00002639#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002640 xmlFreeID(ret);
2641 return(NULL);
2642 }
Daniel Veillard249d7bb2003-03-19 21:02:29 +00002643 if (attr != NULL)
2644 attr->atype = XML_ATTRIBUTE_ID;
Owen Taylor3473f882001-02-23 17:55:21 +00002645 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002665 * Determine whether an attribute is of type ID. In case we have DTD(s)
Daniel Veillard9dc1cf12002-10-08 08:26:11 +00002666 * 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.
Owen Taylor3473f882001-02-23 17:55:21 +00002669 *
2670 * Returns 0 or 1 depending on the lookup result
2671 */
2672int
2673xmlIsID(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) {
Daniel Veillard54f9a4f2005-09-03 13:28:24 +00002674 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);
Owen Taylor3473f882001-02-23 17:55:21 +00002679 if (doc == NULL) return(0);
Daniel Veillardf3c06692009-10-16 16:47:58 +02002680 if ((doc->intSubset == NULL) && (doc->extSubset == NULL) &&
2681 (doc->type != XML_HTML_DOCUMENT_NODE)) {
Owen Taylor3473f882001-02-23 17:55:21 +00002682 return(0);
2683 } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
Rob Richards04bffc02006-03-03 16:44:37 +00002684 if ((xmlStrEqual(BAD_CAST "id", attr->name)) ||
2685 ((xmlStrEqual(BAD_CAST "name", attr->name)) &&
Daniel Veillard38431c32007-06-12 16:20:09 +00002686 ((elem == NULL) || (xmlStrEqual(elem->name, BAD_CAST "a")))))
Owen Taylor3473f882001-02-23 17:55:21 +00002687 return(1);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002688 return(0);
Daniel Veillard379a3b72005-08-12 10:18:14 +00002689 } else if (elem == NULL) {
2690 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00002691 } else {
Daniel Veillard465a0002005-08-22 12:07:04 +00002692 xmlAttributePtr attrDecl = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00002693
Daniel Veillard379a3b72005-08-12 10:18:14 +00002694 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);
Daniel Veillard37f961d2002-07-06 17:53:56 +00002708 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillard379a3b72005-08-12 10:18:14 +00002709 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullelemname,
2710 fullattrname);
Daniel Veillard37f961d2002-07-06 17:53:56 +00002711 }
Owen Taylor3473f882001-02-23 17:55:21 +00002712
Daniel Veillard379a3b72005-08-12 10:18:14 +00002713 if ((fullattrname != fattr) && (fullattrname != attr->name))
2714 xmlFree(fullattrname);
2715 if ((fullelemname != felem) && (fullelemname != elem->name))
2716 xmlFree(fullelemname);
2717
Owen Taylor3473f882001-02-23 17:55:21 +00002718 if ((attrDecl != NULL) && (attrDecl->atype == XML_ATTRIBUTE_ID))
2719 return(1);
2720 }
2721 return(0);
2722}
2723
2724/**
Daniel Veillard01c13b52002-12-10 15:19:08 +00002725 * xmlRemoveID:
Owen Taylor3473f882001-02-23 17:55:21 +00002726 * @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) {
Owen Taylor3473f882001-02-23 17:55:21 +00002735 xmlIDTablePtr table;
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002736 xmlIDPtr id;
Owen Taylor3473f882001-02-23 17:55:21 +00002737 xmlChar *ID;
2738
2739 if (doc == NULL) return(-1);
2740 if (attr == NULL) return(-1);
Denis Pauk01461792013-08-06 09:55:55 +03002741
Owen Taylor3473f882001-02-23 17:55:21 +00002742 table = (xmlIDTablePtr) doc->ids;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002743 if (table == NULL)
Owen Taylor3473f882001-02-23 17:55:21 +00002744 return(-1);
2745
Owen Taylor3473f882001-02-23 17:55:21 +00002746 ID = xmlNodeListGetString(doc, attr->children, 1);
2747 if (ID == NULL)
Denis Pauk01461792013-08-06 09:55:55 +03002748 return(-1);
2749
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002750 id = xmlHashLookup(table, ID);
2751 if (id == NULL || id->attr != attr) {
Denis Pauk01461792013-08-06 09:55:55 +03002752 xmlFree(ID);
2753 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00002754 }
Denis Pauk01461792013-08-06 09:55:55 +03002755
Daniel Veillard91b955c2004-12-10 10:26:42 +00002756 xmlHashRemoveEntry(table, ID, (xmlHashDeallocator) xmlFreeID);
Owen Taylor3473f882001-02-23 17:55:21 +00002757 xmlFree(ID);
Denis Pauk01461792013-08-06 09:55:55 +03002758 attr->atype = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00002759 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 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002771xmlAttrPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002772xmlGetID(xmlDocPtr doc, const xmlChar *ID) {
2773 xmlIDTablePtr table;
2774 xmlIDPtr id;
2775
2776 if (doc == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002777 return(NULL);
2778 }
2779
2780 if (ID == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002781 return(NULL);
2782 }
2783
2784 table = (xmlIDTablePtr) doc->ids;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002785 if (table == NULL)
Owen Taylor3473f882001-02-23 17:55:21 +00002786 return(NULL);
2787
2788 id = xmlHashLookup(table, ID);
2789 if (id == NULL)
2790 return(NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002791 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 }
Owen Taylor3473f882001-02-23 17:55:21 +00002798 return(id->attr);
2799}
2800
2801/************************************************************************
2802 * *
2803 * Refs *
2804 * *
2805 ************************************************************************/
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002806typedef struct xmlRemoveMemo_t
Owen Taylor3473f882001-02-23 17:55:21 +00002807{
2808 xmlListPtr l;
2809 xmlAttrPtr ap;
Daniel Veillard8730c562001-02-26 10:49:57 +00002810} xmlRemoveMemo;
2811
2812typedef xmlRemoveMemo *xmlRemoveMemoPtr;
2813
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002814typedef struct xmlValidateMemo_t
Daniel Veillard8730c562001-02-26 10:49:57 +00002815{
2816 xmlValidCtxtPtr ctxt;
2817 const xmlChar *name;
2818} xmlValidateMemo;
2819
2820typedef xmlValidateMemo *xmlValidateMemoPtr;
Owen Taylor3473f882001-02-23 17:55:21 +00002821
2822/**
Owen Taylor3473f882001-02-23 17:55:21 +00002823 * xmlFreeRef:
2824 * @lk: A list link
2825 *
2826 * Deallocate the memory used by a ref definition
2827 */
2828static void
2829xmlFreeRef(xmlLinkPtr lk) {
Daniel Veillard37721922001-05-04 15:21:12 +00002830 xmlRefPtr ref = (xmlRefPtr)xmlLinkGetData(lk);
2831 if (ref == NULL) return;
2832 if (ref->value != NULL)
2833 xmlFree((xmlChar *)ref->value);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002834 if (ref->name != NULL)
2835 xmlFree((xmlChar *)ref->name);
Daniel Veillard37721922001-05-04 15:21:12 +00002836 xmlFree(ref);
Owen Taylor3473f882001-02-23 17:55:21 +00002837}
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) {
Daniel Veillard37721922001-05-04 15:21:12 +00002847 if (list_ref == NULL) return;
2848 xmlListDelete(list_ref);
Owen Taylor3473f882001-02-23 17:55:21 +00002849}
2850
2851/**
2852 * xmlWalkRemoveRef:
2853 * @data: Contents of current link
2854 * @user: Value supplied by the user
2855 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002856 * Returns 0 to abort the walk or 1 to continue
Owen Taylor3473f882001-02-23 17:55:21 +00002857 */
2858static int
2859xmlWalkRemoveRef(const void *data, const void *user)
2860{
Daniel Veillard37721922001-05-04 15:21:12 +00002861 xmlAttrPtr attr0 = ((xmlRefPtr)data)->attr;
2862 xmlAttrPtr attr1 = ((xmlRemoveMemoPtr)user)->ap;
2863 xmlListPtr ref_list = ((xmlRemoveMemoPtr)user)->l;
Owen Taylor3473f882001-02-23 17:55:21 +00002864
Daniel Veillard37721922001-05-04 15:21:12 +00002865 if (attr0 == attr1) { /* Matched: remove and terminate walk */
2866 xmlListRemoveFirst(ref_list, (void *)data);
2867 return 0;
2868 }
2869 return 1;
Owen Taylor3473f882001-02-23 17:55:21 +00002870}
2871
2872/**
Daniel Veillard770075b2004-02-25 10:44:30 +00002873 * 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
Daniel Veillardf54cd532004-02-25 11:52:31 +00002880xmlDummyCompare(const void *data0 ATTRIBUTE_UNUSED,
2881 const void *data1 ATTRIBUTE_UNUSED)
Daniel Veillard770075b2004-02-25 10:44:30 +00002882{
2883 return (0);
2884}
2885
2886/**
Owen Taylor3473f882001-02-23 17:55:21 +00002887 * 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002895 * Returns NULL if not, otherwise the new xmlRefPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002896 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002897xmlRefPtr
William M. Brackedb65a72004-02-06 07:36:04 +00002898xmlAddRef(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value,
Owen Taylor3473f882001-02-23 17:55:21 +00002899 xmlAttrPtr attr) {
Daniel Veillard37721922001-05-04 15:21:12 +00002900 xmlRefPtr ret;
2901 xmlRefTablePtr table;
2902 xmlListPtr ref_list;
Owen Taylor3473f882001-02-23 17:55:21 +00002903
Daniel Veillard37721922001-05-04 15:21:12 +00002904 if (doc == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00002905 return(NULL);
2906 }
2907 if (value == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00002908 return(NULL);
2909 }
2910 if (attr == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00002911 return(NULL);
2912 }
Owen Taylor3473f882001-02-23 17:55:21 +00002913
Daniel Veillard37721922001-05-04 15:21:12 +00002914 /*
Owen Taylor3473f882001-02-23 17:55:21 +00002915 * Create the Ref table if needed.
2916 */
Daniel Veillard37721922001-05-04 15:21:12 +00002917 table = (xmlRefTablePtr) doc->refs;
Daniel Veillard316a5c32005-01-23 22:56:39 +00002918 if (table == NULL) {
2919 doc->refs = table = xmlHashCreateDict(0, doc->dict);
2920 }
Daniel Veillard37721922001-05-04 15:21:12 +00002921 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002922 xmlVErrMemory(ctxt,
Daniel Veillard37721922001-05-04 15:21:12 +00002923 "xmlAddRef: Table creation failed!\n");
2924 return(NULL);
2925 }
Owen Taylor3473f882001-02-23 17:55:21 +00002926
Daniel Veillard37721922001-05-04 15:21:12 +00002927 ret = (xmlRefPtr) xmlMalloc(sizeof(xmlRef));
2928 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002929 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard37721922001-05-04 15:21:12 +00002930 return(NULL);
2931 }
Owen Taylor3473f882001-02-23 17:55:21 +00002932
Daniel Veillard37721922001-05-04 15:21:12 +00002933 /*
Owen Taylor3473f882001-02-23 17:55:21 +00002934 * fill the structure.
2935 */
Daniel Veillard37721922001-05-04 15:21:12 +00002936 ret->value = xmlStrdup(value);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002937 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);
Owen Taylor3473f882001-02-23 17:55:21 +00002948
Daniel Veillard37721922001-05-04 15:21:12 +00002949 /* 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 */
Owen Taylor3473f882001-02-23 17:55:21 +00002955
Daniel Veillard37721922001-05-04 15:21:12 +00002956 if (NULL == (ref_list = xmlHashLookup(table, value))) {
Daniel Veillard770075b2004-02-25 10:44:30 +00002957 if (NULL == (ref_list = xmlListCreate(xmlFreeRef, xmlDummyCompare))) {
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002958 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2959 "xmlAddRef: Reference list creation failed!\n",
2960 NULL);
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00002961 goto failed;
Daniel Veillard37721922001-05-04 15:21:12 +00002962 }
2963 if (xmlHashAddEntry(table, value, ref_list) < 0) {
2964 xmlListDelete(ref_list);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002965 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2966 "xmlAddRef: Reference list insertion failed!\n",
2967 NULL);
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00002968 goto failed;
Daniel Veillard37721922001-05-04 15:21:12 +00002969 }
2970 }
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00002971 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 }
Daniel Veillard37721922001-05-04 15:21:12 +00002977 return(ret);
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00002978failed:
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);
Owen Taylor3473f882001-02-23 17:55:21 +00002987}
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) {
Daniel Veillard37721922001-05-04 15:21:12 +00002997 xmlHashFree(table, (xmlHashDeallocator) xmlFreeRefList);
Owen Taylor3473f882001-02-23 17:55:21 +00002998}
2999
3000/**
3001 * xmlIsRef:
3002 * @doc: the document
3003 * @elem: the element carrying the attribute
3004 * @attr: the attribute
3005 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003006 * Determine whether an attribute is of type Ref. In case we have DTD(s)
Owen Taylor3473f882001-02-23 17:55:21 +00003007 * 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) {
Daniel Veillardce244ad2004-11-05 10:03:46 +00003014 if (attr == NULL)
3015 return(0);
3016 if (doc == NULL) {
3017 doc = attr->doc;
3018 if (doc == NULL) return(0);
3019 }
3020
Daniel Veillard37721922001-05-04 15:21:12 +00003021 if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
3022 return(0);
3023 } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
3024 /* TODO @@@ */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003025 return(0);
Daniel Veillard37721922001-05-04 15:21:12 +00003026 } else {
3027 xmlAttributePtr attrDecl;
Owen Taylor3473f882001-02-23 17:55:21 +00003028
Daniel Veillardce244ad2004-11-05 10:03:46 +00003029 if (elem == NULL) return(0);
Daniel Veillard37721922001-05-04 15:21:12 +00003030 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);
Owen Taylor3473f882001-02-23 17:55:21 +00003034
Daniel Veillard37721922001-05-04 15:21:12 +00003035 if ((attrDecl != NULL) &&
3036 (attrDecl->atype == XML_ATTRIBUTE_IDREF ||
3037 attrDecl->atype == XML_ATTRIBUTE_IDREFS))
3038 return(1);
3039 }
3040 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003041}
3042
3043/**
Daniel Veillard01c13b52002-12-10 15:19:08 +00003044 * xmlRemoveRef:
Owen Taylor3473f882001-02-23 17:55:21 +00003045 * @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) {
Daniel Veillard37721922001-05-04 15:21:12 +00003054 xmlListPtr ref_list;
3055 xmlRefTablePtr table;
3056 xmlChar *ID;
3057 xmlRemoveMemo target;
Owen Taylor3473f882001-02-23 17:55:21 +00003058
Daniel Veillard37721922001-05-04 15:21:12 +00003059 if (doc == NULL) return(-1);
3060 if (attr == NULL) return(-1);
Denis Pauk01461792013-08-06 09:55:55 +03003061
Daniel Veillard37721922001-05-04 15:21:12 +00003062 table = (xmlRefTablePtr) doc->refs;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003063 if (table == NULL)
Daniel Veillard37721922001-05-04 15:21:12 +00003064 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003065
Daniel Veillard37721922001-05-04 15:21:12 +00003066 ID = xmlNodeListGetString(doc, attr->children, 1);
3067 if (ID == NULL)
3068 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003069
Denis Pauk01461792013-08-06 09:55:55 +03003070 ref_list = xmlHashLookup(table, ID);
Daniel Veillard37721922001-05-04 15:21:12 +00003071 if(ref_list == NULL) {
3072 xmlFree(ID);
3073 return (-1);
3074 }
Denis Pauk01461792013-08-06 09:55:55 +03003075
Daniel Veillard37721922001-05-04 15:21:12 +00003076 /* 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;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003088
Daniel Veillard37721922001-05-04 15:21:12 +00003089 /* Remove the supplied attr from our list */
3090 xmlListWalk(ref_list, xmlWalkRemoveRef, &target);
Owen Taylor3473f882001-02-23 17:55:21 +00003091
Daniel Veillard37721922001-05-04 15:21:12 +00003092 /*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);
Owen Taylor3473f882001-02-23 17:55:21 +00003098}
3099
3100/**
3101 * xmlGetRefs:
3102 * @doc: pointer to the document
3103 * @ID: the ID value
3104 *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003105 * Find the set of references for the supplied ID.
Owen Taylor3473f882001-02-23 17:55:21 +00003106 *
3107 * Returns NULL if not found, otherwise node set for the ID.
3108 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003109xmlListPtr
Owen Taylor3473f882001-02-23 17:55:21 +00003110xmlGetRefs(xmlDocPtr doc, const xmlChar *ID) {
Daniel Veillard37721922001-05-04 15:21:12 +00003111 xmlRefTablePtr table;
Owen Taylor3473f882001-02-23 17:55:21 +00003112
Daniel Veillard37721922001-05-04 15:21:12 +00003113 if (doc == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00003114 return(NULL);
3115 }
Owen Taylor3473f882001-02-23 17:55:21 +00003116
Daniel Veillard37721922001-05-04 15:21:12 +00003117 if (ID == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00003118 return(NULL);
3119 }
Owen Taylor3473f882001-02-23 17:55:21 +00003120
Daniel Veillard37721922001-05-04 15:21:12 +00003121 table = (xmlRefTablePtr) doc->refs;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003122 if (table == NULL)
Daniel Veillard37721922001-05-04 15:21:12 +00003123 return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003124
Daniel Veillard37721922001-05-04 15:21:12 +00003125 return (xmlHashLookup(table, ID));
Owen Taylor3473f882001-02-23 17:55:21 +00003126}
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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003139 * Search the DTD for the description of this element
Owen Taylor3473f882001-02-23 17:55:21 +00003140 *
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
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00003150 if ((dtd == NULL) || (name == NULL)) return(NULL);
Daniel Veillarda10efa82001-04-18 13:09:01 +00003151 if (dtd->elements == NULL)
3152 return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003153 table = (xmlElementTablePtr) dtd->elements;
3154
3155 uqname = xmlSplitQName2(name, &prefix);
Daniel Veillarda10efa82001-04-18 13:09:01 +00003156 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003169 * Search the DTD for the description of this element
Daniel Veillarda10efa82001-04-18 13:09:01 +00003170 *
3171 * returns the xmlElementPtr if found or NULL
3172 */
3173
Daniel Veillard86fd5a72001-12-13 14:55:21 +00003174static xmlElementPtr
Daniel Veillarda10efa82001-04-18 13:09:01 +00003175xmlGetDtdElementDesc2(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) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00003182 xmlDictPtr dict = NULL;
3183
3184 if (dtd->doc != NULL)
3185 dict = dtd->doc->dict;
3186
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003187 if (!create)
Daniel Veillarda10efa82001-04-18 13:09:01 +00003188 return(NULL);
3189 /*
3190 * Create the Element table if needed.
3191 */
3192 table = (xmlElementTablePtr) dtd->elements;
3193 if (table == NULL) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00003194 table = xmlHashCreateDict(0, dict);
Daniel Veillarda10efa82001-04-18 13:09:01 +00003195 dtd->elements = (void *) table;
3196 }
3197 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00003198 xmlVErrMemory(NULL, "element table allocation failed");
Daniel Veillarda10efa82001-04-18 13:09:01 +00003199 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) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00003211 xmlVErrMemory(NULL, "malloc failed");
Daniel Veillarda10efa82001-04-18 13:09:01 +00003212 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);
Owen Taylor3473f882001-02-23 17:55:21 +00003228 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003237 * Search the DTD for the description of this element
Owen Taylor3473f882001-02-23 17:55:21 +00003238 *
3239 * returns the xmlElementPtr if found or NULL
3240 */
3241
Daniel Veillard48da9102001-08-07 01:10:10 +00003242xmlElementPtr
Owen Taylor3473f882001-02-23 17:55:21 +00003243xmlGetDtdQElementDesc(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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003260 * Search the DTD for the description of this attribute on
Owen Taylor3473f882001-02-23 17:55:21 +00003261 * 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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003297 * Search the DTD for the description of this qualified attribute on
Owen Taylor3473f882001-02-23 17:55:21 +00003298 * this element.
3299 *
3300 * returns the xmlAttributePtr if found or NULL
3301 */
3302
Daniel Veillard48da9102001-08-07 01:10:10 +00003303xmlAttributePtr
Owen Taylor3473f882001-02-23 17:55:21 +00003304xmlGetDtdQAttrDesc(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 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003320 * Search the DTD for the description of this notation
Owen Taylor3473f882001-02-23 17:55:21 +00003321 *
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
Daniel Veillardf54cd532004-02-25 11:52:31 +00003336#if defined(LIBXML_VALID_ENABLED) || defined(LIBXML_SCHEMAS_ENABLED)
Owen Taylor3473f882001-02-23 17:55:21 +00003337/**
3338 * xmlValidateNotationUse:
3339 * @ctxt: the validation context
3340 * @doc: the document
3341 * @notationName: the notation name to check
3342 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003343 * Validate that the given name match a notation declaration.
Owen Taylor3473f882001-02-23 17:55:21 +00003344 * - [ 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;
Daniel Veillardeab3ac92009-08-12 10:39:29 +02003353 if ((doc == NULL) || (doc->intSubset == NULL) ||
3354 (notationName == NULL)) return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003355
3356 notaDecl = xmlGetDtdNotationDesc(doc->intSubset, notationName);
3357 if ((notaDecl == NULL) && (doc->extSubset != NULL))
3358 notaDecl = xmlGetDtdNotationDesc(doc->extSubset, notationName);
3359
Daniel Veillarde637c4a2003-03-30 21:10:09 +00003360 if ((notaDecl == NULL) && (ctxt != NULL)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003361 xmlErrValidNode(ctxt, (xmlNodePtr) doc, XML_DTD_UNKNOWN_NOTATION,
3362 "NOTATION %s is not declared\n",
3363 notationName, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003364 return(0);
3365 }
3366 return(1);
3367}
Daniel Veillardf54cd532004-02-25 11:52:31 +00003368#endif /* LIBXML_VALID_ENABLED or LIBXML_SCHEMAS_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00003369
3370/**
Daniel Veillard01c13b52002-12-10 15:19:08 +00003371 * xmlIsMixedElement:
Owen Taylor3473f882001-02-23 17:55:21 +00003372 * @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) {
Daniel Veillarda10efa82001-04-18 13:09:01 +00003392 case XML_ELEMENT_TYPE_UNDEFINED:
3393 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003394 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
Daniel Veillard4432df22003-09-28 18:58:27 +00003408#ifdef LIBXML_VALID_ENABLED
Daniel Veillardae0765b2008-07-31 19:54:59 +00003409
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
Owen Taylor3473f882001-02-23 17:55:21 +00003512/**
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
Daniel Veillard9b731d72002-04-14 12:56:08 +00003521int
Owen Taylor3473f882001-02-23 17:55:21 +00003522xmlValidateNameValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003523 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) {
Owen Taylor3473f882001-02-23 17:55:21 +00003538 const xmlChar *cur;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003539 int val, len;
Owen Taylor3473f882001-02-23 17:55:21 +00003540
3541 if (value == NULL) return(0);
3542 cur = value;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003543 val = xmlStringCurrentChar(NULL, cur, &len);
3544 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003545
3546 if (!xmlIsDocNameStartChar(doc, val))
Owen Taylor3473f882001-02-23 17:55:21 +00003547 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003548
Daniel Veillardd8224e02002-01-13 15:43:22 +00003549 val = xmlStringCurrentChar(NULL, cur, &len);
3550 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003551 while (xmlIsDocNameChar(doc, val)) {
Daniel Veillardd8224e02002-01-13 15:43:22 +00003552 val = xmlStringCurrentChar(NULL, cur, &len);
3553 cur += len;
3554 }
Owen Taylor3473f882001-02-23 17:55:21 +00003555
Daniel Veillardae0765b2008-07-31 19:54:59 +00003556 /* 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
Daniel Veillardd8224e02002-01-13 15:43:22 +00003575 if (val != 0) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003576
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
Daniel Veillard9b731d72002-04-14 12:56:08 +00003589int
Owen Taylor3473f882001-02-23 17:55:21 +00003590xmlValidateNamesValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003591 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) {
Owen Taylor3473f882001-02-23 17:55:21 +00003608 const xmlChar *cur;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003609 int val, len;
Owen Taylor3473f882001-02-23 17:55:21 +00003610
3611 if (value == NULL) return(0);
3612 cur = value;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003613 val = xmlStringCurrentChar(NULL, cur, &len);
3614 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003615
3616 if (!xmlIsDocNameChar(doc, val))
Owen Taylor3473f882001-02-23 17:55:21 +00003617 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003618
Daniel Veillardd8224e02002-01-13 15:43:22 +00003619 val = xmlStringCurrentChar(NULL, cur, &len);
3620 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003621 while (xmlIsDocNameChar(doc, val)) {
Daniel Veillardd8224e02002-01-13 15:43:22 +00003622 val = xmlStringCurrentChar(NULL, cur, &len);
3623 cur += len;
Owen Taylor3473f882001-02-23 17:55:21 +00003624 }
3625
Daniel Veillardd8224e02002-01-13 15:43:22 +00003626 if (val != 0) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003627
3628 return(1);
3629}
3630
3631/**
3632 * xmlValidateNmtokenValue:
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003633 * @value: an Nmtoken value
Owen Taylor3473f882001-02-23 17:55:21 +00003634 *
3635 * Validate that the given value match Nmtoken production
3636 *
3637 * [ VC: Name Token ]
Daniel Veillardae0765b2008-07-31 19:54:59 +00003638 *
Owen Taylor3473f882001-02-23 17:55:21 +00003639 * returns 1 if valid or 0 otherwise
3640 */
3641
Daniel Veillard9b731d72002-04-14 12:56:08 +00003642int
Owen Taylor3473f882001-02-23 17:55:21 +00003643xmlValidateNmtokenValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003644 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) {
Owen Taylor3473f882001-02-23 17:55:21 +00003661 const xmlChar *cur;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003662 int val, len;
Owen Taylor3473f882001-02-23 17:55:21 +00003663
3664 if (value == NULL) return(0);
3665 cur = value;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003666 val = xmlStringCurrentChar(NULL, cur, &len);
3667 cur += len;
Owen Taylor3473f882001-02-23 17:55:21 +00003668
Daniel Veillardae0765b2008-07-31 19:54:59 +00003669 while (IS_BLANK(val)) {
Daniel Veillardd8224e02002-01-13 15:43:22 +00003670 val = xmlStringCurrentChar(NULL, cur, &len);
3671 cur += len;
3672 }
Owen Taylor3473f882001-02-23 17:55:21 +00003673
Daniel Veillardae0765b2008-07-31 19:54:59 +00003674 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
Daniel Veillardd8224e02002-01-13 15:43:22 +00003702 if (val != 0) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003703
3704 return(1);
3705}
3706
3707/**
3708 * xmlValidateNmtokensValue:
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003709 * @value: an Nmtokens value
Owen Taylor3473f882001-02-23 17:55:21 +00003710 *
3711 * Validate that the given value match Nmtokens production
3712 *
3713 * [ VC: Name Token ]
Daniel Veillardae0765b2008-07-31 19:54:59 +00003714 *
Owen Taylor3473f882001-02-23 17:55:21 +00003715 * returns 1 if valid or 0 otherwise
3716 */
3717
Daniel Veillard9b731d72002-04-14 12:56:08 +00003718int
Owen Taylor3473f882001-02-23 17:55:21 +00003719xmlValidateNmtokensValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003720 return(xmlValidateNmtokensValueInternal(NULL, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003721}
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:
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003732 * - it seems that no validity constraint exists on notation declarations
Owen Taylor3473f882001-02-23 17:55:21 +00003733 * But this function get called anyway ...
3734 *
3735 * returns 1 if valid or 0 otherwise
3736 */
3737
3738int
Daniel Veillardc86a4fa2001-03-26 16:28:29 +00003739xmlValidateNotationDecl(xmlValidCtxtPtr ctxt ATTRIBUTE_UNUSED, xmlDocPtr doc ATTRIBUTE_UNUSED,
3740 xmlNotationPtr nota ATTRIBUTE_UNUSED) {
Owen Taylor3473f882001-02-23 17:55:21 +00003741 int ret = 1;
3742
3743 return(ret);
3744}
3745
3746/**
Daniel Veillardbe2bd6a2008-11-27 15:26:28 +00003747 * xmlValidateAttributeValueInternal:
3748 * @doc: the document
Owen Taylor3473f882001-02-23 17:55:21 +00003749 * @type: an attribute type
3750 * @value: an attribute value
3751 *
3752 * Validate that the given attribute value match the proper production
3753 *
Owen Taylor3473f882001-02-23 17:55:21 +00003754 * returns 1 if valid or 0 otherwise
3755 */
3756
Daniel Veillardae0765b2008-07-31 19:54:59 +00003757static int
3758xmlValidateAttributeValueInternal(xmlDocPtr doc, xmlAttributeType type,
3759 const xmlChar *value) {
Owen Taylor3473f882001-02-23 17:55:21 +00003760 switch (type) {
3761 case XML_ATTRIBUTE_ENTITIES:
3762 case XML_ATTRIBUTE_IDREFS:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003763 return(xmlValidateNamesValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003764 case XML_ATTRIBUTE_ENTITY:
3765 case XML_ATTRIBUTE_IDREF:
3766 case XML_ATTRIBUTE_ID:
3767 case XML_ATTRIBUTE_NOTATION:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003768 return(xmlValidateNameValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003769 case XML_ATTRIBUTE_NMTOKENS:
3770 case XML_ATTRIBUTE_ENUMERATION:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003771 return(xmlValidateNmtokensValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003772 case XML_ATTRIBUTE_NMTOKEN:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003773 return(xmlValidateNmtokenValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003774 case XML_ATTRIBUTE_CDATA:
3775 break;
3776 }
3777 return(1);
3778}
3779
Daniel Veillardbe2bd6a2008-11-27 15:26:28 +00003780/**
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
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003800 * of type NMTOKENS must match Nmtokens.
Daniel Veillardbe2bd6a2008-11-27 15:26:28 +00003801 *
3802 * returns 1 if valid or 0 otherwise
3803 */
Daniel Veillardae0765b2008-07-31 19:54:59 +00003804int
3805xmlValidateAttributeValue(xmlAttributeType type, const xmlChar *value) {
3806 return(xmlValidateAttributeValueInternal(NULL, type, value));
3807}
3808
Owen Taylor3473f882001-02-23 17:55:21 +00003809/**
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
Daniel Veillard56a4cb82001-03-24 17:00:36 +00003838static int
Owen Taylor3473f882001-02-23 17:55:21 +00003839xmlValidateAttributeValue2(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);
Daniel Veillard62998c02003-09-15 12:56:36 +00003855 /* yeah it's a bit messy... */
Daniel Veillard878eab02002-02-19 13:46:09 +00003856 if ((ent == NULL) && (doc->standalone == 1)) {
3857 doc->standalone = 0;
3858 ent = xmlGetDocEntity(doc, value);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003859 }
Owen Taylor3473f882001-02-23 17:55:21 +00003860 if (ent == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003861 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3862 XML_DTD_UNKNOWN_ENTITY,
Owen Taylor3473f882001-02-23 17:55:21 +00003863 "ENTITY attribute %s reference an unknown entity \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003864 name, value, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003865 ret = 0;
3866 } else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003867 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3868 XML_DTD_ENTITY_TYPE,
Owen Taylor3473f882001-02-23 17:55:21 +00003869 "ENTITY attribute %s reference an entity \"%s\" of wrong type\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003870 name, value, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003871 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;
William M. Brack76e95df2003-10-18 16:20:14 +00003885 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00003886 save = *cur;
3887 *cur = 0;
3888 ent = xmlGetDocEntity(doc, nam);
3889 if (ent == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003890 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3891 XML_DTD_UNKNOWN_ENTITY,
Owen Taylor3473f882001-02-23 17:55:21 +00003892 "ENTITIES attribute %s reference an unknown entity \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003893 name, nam, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003894 ret = 0;
3895 } else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003896 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3897 XML_DTD_ENTITY_TYPE,
Owen Taylor3473f882001-02-23 17:55:21 +00003898 "ENTITIES attribute %s reference an entity \"%s\" of wrong type\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003899 name, nam, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003900 ret = 0;
3901 }
3902 if (save == 0)
3903 break;
3904 *cur = save;
William M. Brack76e95df2003-10-18 16:20:14 +00003905 while (IS_BLANK_CH(*cur)) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00003906 }
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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003918 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3919 XML_DTD_UNKNOWN_NOTATION,
Owen Taylor3473f882001-02-23 17:55:21 +00003920 "NOTATION attribute %s reference an unknown notation \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003921 name, value, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003922 ret = 0;
3923 }
3924 break;
3925 }
3926 }
3927 return(ret);
3928}
3929
3930/**
Daniel Veillard8dc16a62002-02-19 21:08:48 +00003931 * 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)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00003968 xmlChar fn[50];
3969 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003970
Daniel Veillardc00cda82003-04-07 10:22:39 +00003971 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
3972 if (fullname == NULL)
Daniel Veillard24505b02005-07-28 23:49:35 +00003973 return(NULL);
Daniel Veillardc00cda82003-04-07 10:22:39 +00003974 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, name);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00003975 if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00003976 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname, name);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00003977 if (attrDecl != NULL)
3978 extsubset = 1;
3979 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00003980 if ((fullname != fn) && (fullname != elem->name))
3981 xmlFree(fullname);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00003982 }
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))) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004013 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_STANDALONE,
Daniel Veillard8dc16a62002-02-19 21:08:48 +00004014"standalone: %s on %s value had to be normalized based on external subset declaration\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004015 name, elem->name, NULL);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00004016 ctxt->valid = 0;
4017 }
4018 return(ret);
4019}
4020
4021/**
Owen Taylor3473f882001-02-23 17:55:21 +00004022 * 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 *
Daniel Veillard652327a2003-09-29 18:02:38 +00004036 * Returns a new normalized string if normalization is needed, NULL otherwise
Owen Taylor3473f882001-02-23 17:55:21 +00004037 * 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)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004053 xmlChar fn[50];
4054 xmlChar *fullname;
Daniel Veillard594e5df2009-09-07 14:58:47 +02004055
Daniel Veillardc00cda82003-04-07 10:22:39 +00004056 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4057 if (fullname == NULL)
Daniel Veillard24505b02005-07-28 23:49:35 +00004058 return(NULL);
Daniel Veillardc00cda82003-04-07 10:22:39 +00004059 if ((fullname != fn) && (fullname != elem->name))
4060 xmlFree(fullname);
Owen Taylor3473f882001-02-23 17:55:21 +00004061 }
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
Daniel Veillard56a4cb82001-03-24 17:00:36 +00004090static void
Owen Taylor3473f882001-02-23 17:55:21 +00004091xmlValidateAttributeIdCallback(xmlAttributePtr attr, int *count,
Daniel Veillardc86a4fa2001-03-26 16:28:29 +00004092 const xmlChar* name ATTRIBUTE_UNUSED) {
Owen Taylor3473f882001-02-23 17:55:21 +00004093 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);
Daniel Veillardae0765b2008-07-31 19:54:59 +00004121
Owen Taylor3473f882001-02-23 17:55:21 +00004122 /* Attribute Default Legal */
4123 /* Enumeration */
4124 if (attr->defaultValue != NULL) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00004125 val = xmlValidateAttributeValueInternal(doc, attr->atype,
4126 attr->defaultValue);
Owen Taylor3473f882001-02-23 17:55:21 +00004127 if (val == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004128 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004129 "Syntax of default value for attribute %s of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004130 attr->name, attr->elem, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004131 }
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)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004139 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_FIXED,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004140 "ID attribute %s of %s is not valid must be #IMPLIED or #REQUIRED\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004141 attr->name, attr->elem, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004142 ret = 0;
4143 }
4144
4145 /* One ID per Element Type */
4146 if (attr->atype == XML_ATTRIBUTE_ID) {
4147 int nbId;
4148
Daniel Veillardcbaf3992001-12-31 16:16:02 +00004149 /* the trick is that we parse DtD as their own internal subset */
Owen Taylor3473f882001-02-23 17:55:21 +00004150 xmlElementPtr elem = xmlGetDtdElementDesc(doc->intSubset,
4151 attr->elem);
4152 if (elem != NULL) {
Daniel Veillarddbee0f12005-06-27 13:42:57 +00004153 nbId = xmlScanIDAttributeDecl(NULL, elem, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00004154 } 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;
Daniel Veillard11ce4002006-03-10 00:36:23 +00004162 if (doc->intSubset != NULL) {
4163 table = (xmlAttributeTablePtr) doc->intSubset->attributes;
4164 xmlHashScan3(table, NULL, NULL, attr->elem, (xmlHashScanner)
4165 xmlValidateAttributeIdCallback, &nbId);
4166 }
Owen Taylor3473f882001-02-23 17:55:21 +00004167 }
4168 if (nbId > 1) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004169
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004170 xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
Owen Taylor3473f882001-02-23 17:55:21 +00004171 "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) {
Daniel Veillarddbee0f12005-06-27 13:42:57 +00004177 extId = xmlScanIDAttributeDecl(NULL, elem, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00004178 }
4179 if (extId > 1) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004180 xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
Owen Taylor3473f882001-02-23 17:55:21 +00004181 "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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004184 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
Owen Taylor3473f882001-02-23 17:55:21 +00004185"Element %s has ID attributes defined in the internal and external subset : %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004186 attr->elem, attr->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004187 }
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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004199 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004200"Default value \"%s\" for attribute %s of %s is not among the enumerated set\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004201 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;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004232
Owen Taylor3473f882001-02-23 17:55:21 +00004233 if (elem == NULL) return(1);
4234
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00004235#if 0
Daniel Veillard84d70a42002-09-16 10:51:38 +00004236#ifdef LIBXML_REGEXP_ENABLED
4237 /* Build the regexp associated to the content model */
4238 ret = xmlValidBuildContentModel(ctxt, elem);
4239#endif
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00004240#endif
Daniel Veillard84d70a42002-09-16 10:51:38 +00004241
Owen Taylor3473f882001-02-23 17:55:21 +00004242 /* 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) {
Daniel Veillard7b68df92003-08-03 22:58:54 +00004256 if ((xmlStrEqual(next->name, name)) &&
Daniel Veillarda7216122009-08-21 18:22:58 +02004257 (xmlStrEqual(next->prefix, cur->c1->prefix))) {
4258 if (cur->c1->prefix == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004259 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Owen Taylor3473f882001-02-23 17:55:21 +00004260 "Definition of %s has duplicate references of %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004261 elem->name, name, NULL);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004262 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004263 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Daniel Veillard7b68df92003-08-03 22:58:54 +00004264 "Definition of %s has duplicate references of %s:%s\n",
Daniel Veillarda7216122009-08-21 18:22:58 +02004265 elem->name, cur->c1->prefix, name);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004266 }
Owen Taylor3473f882001-02-23 17:55:21 +00004267 ret = 0;
4268 }
4269 break;
4270 }
4271 if (next->c1 == NULL) break;
4272 if (next->c1->type != XML_ELEMENT_CONTENT_ELEMENT) break;
Daniel Veillard7b68df92003-08-03 22:58:54 +00004273 if ((xmlStrEqual(next->c1->name, name)) &&
Daniel Veillarda7216122009-08-21 18:22:58 +02004274 (xmlStrEqual(next->c1->prefix, cur->c1->prefix))) {
4275 if (cur->c1->prefix == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004276 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Daniel Veillard7b68df92003-08-03 22:58:54 +00004277 "Definition of %s has duplicate references to %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004278 elem->name, name, NULL);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004279 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004280 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Daniel Veillard7b68df92003-08-03 22:58:54 +00004281 "Definition of %s has duplicate references to %s:%s\n",
Daniel Veillarda7216122009-08-21 18:22:58 +02004282 elem->name, cur->c1->prefix, name);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004283 }
Owen Taylor3473f882001-02-23 17:55:21 +00004284 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);
Daniel Veillarda10efa82001-04-18 13:09:01 +00004295 if ((tst != NULL ) && (tst != elem) &&
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004296 ((tst->prefix == elem->prefix) ||
4297 (xmlStrEqual(tst->prefix, elem->prefix))) &&
Daniel Veillarda10efa82001-04-18 13:09:01 +00004298 (tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004299 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4300 "Redefinition of element %s\n",
4301 elem->name, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004302 ret = 0;
4303 }
4304 tst = xmlGetDtdElementDesc(doc->extSubset, elem->name);
Daniel Veillarda10efa82001-04-18 13:09:01 +00004305 if ((tst != NULL ) && (tst != elem) &&
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004306 ((tst->prefix == elem->prefix) ||
4307 (xmlStrEqual(tst->prefix, elem->prefix))) &&
Daniel Veillarda10efa82001-04-18 13:09:01 +00004308 (tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004309 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4310 "Redefinition of element %s\n",
4311 elem->name, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004312 ret = 0;
4313 }
Daniel Veillarda10efa82001-04-18 13:09:01 +00004314 /* One ID per Element Type
4315 * already done when registering the attribute
Owen Taylor3473f882001-02-23 17:55:21 +00004316 if (xmlScanIDAttributeDecl(ctxt, elem) > 1) {
4317 ret = 0;
Daniel Veillarda10efa82001-04-18 13:09:01 +00004318 } */
Owen Taylor3473f882001-02-23 17:55:21 +00004319 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,
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004349 xmlNodePtr elem, xmlAttrPtr attr, const xmlChar *value)
Daniel Veillard07cb8222003-09-10 10:51:05 +00004350{
Owen Taylor3473f882001-02-23 17:55:21 +00004351 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)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004360 xmlChar fn[50];
4361 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004362
Daniel Veillardc00cda82003-04-07 10:22:39 +00004363 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4364 if (fullname == NULL)
4365 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00004366 if (attr->ns != NULL) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004367 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
Owen Taylor3473f882001-02-23 17:55:21 +00004368 attr->name, attr->ns->prefix);
4369 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillardc00cda82003-04-07 10:22:39 +00004370 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
Owen Taylor3473f882001-02-23 17:55:21 +00004371 attr->name, attr->ns->prefix);
4372 } else {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004373 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00004374 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4375 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
Daniel Veillardc00cda82003-04-07 10:22:39 +00004376 fullname, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00004377 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00004378 if ((fullname != fn) && (fullname != elem->name))
4379 xmlFree(fullname);
Owen Taylor3473f882001-02-23 17:55:21 +00004380 }
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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004400 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004401 "No declaration for attribute %s of element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004402 attr->name, elem->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004403 return(0);
4404 }
4405 attr->atype = attrDecl->atype;
4406
Daniel Veillardae0765b2008-07-31 19:54:59 +00004407 val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
Owen Taylor3473f882001-02-23 17:55:21 +00004408 if (val == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004409 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004410 "Syntax of value for attribute %s of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004411 attr->name, elem->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004412 ret = 0;
4413 }
4414
4415 /* Validity constraint: Fixed Attribute Default */
4416 if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4417 if (!xmlStrEqual(value, attrDecl->defaultValue)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004418 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004419 "Value for attribute %s of %s is different from default \"%s\"\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004420 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);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004446
Owen Taylor3473f882001-02-23 17:55:21 +00004447 if (nota == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004448 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004449 "Value \"%s\" for attribute %s of %s is not a declared Notation\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004450 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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004460 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004461"Value \"%s\" for attribute %s of %s is not among the enumerated notations\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004462 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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004475 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004476 "Value \"%s\" for attribute %s of %s is not among the enumerated set\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004477 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))) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004485 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004486 "Value for attribute %s of %s must be \"%s\"\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004487 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
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004498/**
4499 * xmlValidateOneNamespace:
4500 * @ctxt: the validation context
4501 * @doc: a document instance
4502 * @elem: an element instance
Daniel Veillarda9b66d02002-12-11 14:23:49 +00004503 * @prefix: the namespace prefix
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004504 * @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) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004537 xmlChar fn[50];
4538 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004539
Daniel Veillardc00cda82003-04-07 10:22:39 +00004540 fullname = xmlBuildQName(elem->name, prefix, fn, 50);
4541 if (fullname == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00004542 xmlVErrMemory(ctxt, "Validating namespace");
Daniel Veillardc00cda82003-04-07 10:22:39 +00004543 return(0);
4544 }
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004545 if (ns->prefix != NULL) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004546 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004547 ns->prefix, BAD_CAST "xmlns");
4548 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillardc00cda82003-04-07 10:22:39 +00004549 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004550 ns->prefix, BAD_CAST "xmlns");
4551 } else {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004552 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004553 BAD_CAST "xmlns");
4554 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillardc00cda82003-04-07 10:22:39 +00004555 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004556 BAD_CAST "xmlns");
4557 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00004558 if ((fullname != fn) && (fullname != elem->name))
4559 xmlFree(fullname);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004560 }
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) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004580 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004581 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004582 "No declaration for attribute xmlns:%s of element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004583 ns->prefix, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004584 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004585 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004586 "No declaration for attribute xmlns of element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004587 elem->name, NULL, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004588 }
4589 return(0);
4590 }
4591
Daniel Veillardae0765b2008-07-31 19:54:59 +00004592 val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004593 if (val == 0) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004594 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004595 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004596 "Syntax of value for attribute xmlns:%s of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004597 ns->prefix, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004598 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004599 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004600 "Syntax of value for attribute xmlns of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004601 elem->name, NULL, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004602 }
4603 ret = 0;
4604 }
4605
4606 /* Validity constraint: Fixed Attribute Default */
4607 if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4608 if (!xmlStrEqual(value, attrDecl->defaultValue)) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004609 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004610 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004611 "Value for attribute xmlns:%s of %s is different from default \"%s\"\n",
4612 ns->prefix, elem->name, attrDecl->defaultValue);
4613 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004614 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004615 "Value for attribute xmlns of %s is different from default \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004616 elem->name, attrDecl->defaultValue, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004617 }
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);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004643
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004644 if (nota == NULL) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004645 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004646 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004647 "Value \"%s\" for attribute xmlns:%s of %s is not a declared Notation\n",
4648 value, ns->prefix, elem->name);
4649 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004650 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004651 "Value \"%s\" for attribute xmlns of %s is not a declared Notation\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004652 value, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004653 }
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) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004663 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004664 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004665"Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated notations\n",
4666 value, ns->prefix, elem->name);
4667 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004668 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004669"Value \"%s\" for attribute xmlns of %s is not among the enumerated notations\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004670 value, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004671 }
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) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004684 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004685 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004686"Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated set\n",
4687 value, ns->prefix, elem->name);
4688 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004689 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004690"Value \"%s\" for attribute xmlns of %s is not among the enumerated set\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004691 value, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004692 }
4693 ret = 0;
4694 }
4695 }
4696
4697 /* Fixed Attribute Default */
4698 if ((attrDecl->def == XML_ATTRIBUTE_FIXED) &&
4699 (!xmlStrEqual(attrDecl->defaultValue, value))) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004700 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004701 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004702 "Value for attribute xmlns:%s of %s must be \"%s\"\n",
4703 ns->prefix, elem->name, attrDecl->defaultValue);
4704 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004705 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004706 "Value for attribute xmlns of %s must be \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004707 elem->name, attrDecl->defaultValue, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004708 }
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
Daniel Veillard118aed72002-09-24 14:13:13 +00004724#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004725/**
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
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004736xmlValidateSkipIgnorable(xmlNodePtr child) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004737 while (child != NULL) {
4738 switch (child->type) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004739 /* 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 *
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004766 * 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.
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004769 */
4770
4771static int
4772xmlValidateElementType(xmlValidCtxtPtr ctxt) {
Daniel Veillardb4545fd2001-11-20 09:37:09 +00004773 int ret = -1;
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004774 int determinist = 1;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004775
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004776 NODE = xmlValidateSkipIgnorable(NODE);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004777 if ((NODE == NULL) && (CONT == NULL))
4778 return(1);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004779 if ((NODE == NULL) &&
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004780 ((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
4781 (CONT->ocur == XML_ELEMENT_CONTENT_OPT))) {
4782 return(1);
4783 }
4784 if (CONT == NULL) return(-1);
Daniel Veillard7533cc82001-04-24 15:52:00 +00004785 if ((NODE != NULL) && (NODE->type == XML_ENTITY_REF_NODE))
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004786 return(-2);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004787
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) {
Daniel Veillardcbaf3992001-12-31 16:16:02 +00004798 DEBUG_VALID_MSG("restored parent branch");
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004799 DEBUG_VALID_STATE(NODE, CONT)
4800 ret = 1;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004801 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 */
Daniel Veillarde62d36c2001-05-15 08:53:16 +00004809 if ((CONT != NULL) &&
Daniel Veillardce2c2f02001-10-18 14:57:24 +00004810 ((CONT->parent == NULL) ||
4811 (CONT->parent->type != XML_ELEMENT_CONTENT_OR)) &&
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004812 ((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
Daniel Veillardca1f1722001-04-20 15:47:35 +00004813 (CONT->ocur == XML_ELEMENT_CONTENT_OPT) ||
Daniel Veillard5344c602001-12-31 16:37:34 +00004814 ((CONT->ocur == XML_ELEMENT_CONTENT_PLUS) && (OCCURRENCE)))) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004815 DEBUG_VALID_MSG("saving parent branch");
Daniel Veillard940492d2002-04-15 10:15:25 +00004816 if (vstateVPush(ctxt, CONT, NODE, DEPTH, OCCURS, ROLLBACK_PARENT) < 0)
4817 return(0);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004818 }
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;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004839 NODE = xmlValidateSkipIgnorable(NODE);
4840 if ((NODE != NULL) &&
4841 (NODE->type == XML_ENTITY_REF_NODE))
4842 return(-2);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004843 } while ((NODE != NULL) &&
4844 ((NODE->type != XML_ELEMENT_NODE) &&
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00004845 (NODE->type != XML_TEXT_NODE) &&
4846 (NODE->type != XML_CDATA_SECTION_NODE)));
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004847 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 }
Daniel Veillard8bdd2202001-06-11 12:47:59 +00004861 ret = ((NODE->type == XML_ELEMENT_NODE) &&
4862 (xmlStrEqual(NODE->name, CONT->name)));
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004863 if (ret == 1) {
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004864 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) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004873 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;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004880 NODE = xmlValidateSkipIgnorable(NODE);
4881 if ((NODE != NULL) &&
4882 (NODE->type == XML_ENTITY_REF_NODE))
4883 return(-2);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004884 } while ((NODE != NULL) &&
4885 ((NODE->type != XML_ELEMENT_NODE) &&
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00004886 (NODE->type != XML_TEXT_NODE) &&
4887 (NODE->type != XML_CDATA_SECTION_NODE)));
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004888 } else {
4889 DEBUG_VALID_MSG("element failed");
4890 ret = 0;
4891 break;
4892 }
4893 break;
4894 case XML_ELEMENT_CONTENT_OR:
4895 /*
Daniel Veillard85349052001-04-20 13:48:21 +00004896 * 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 }
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004905 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 }
Daniel Veillard85349052001-04-20 13:48:21 +00004917 }
4918
4919 /*
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004920 * save the second branch 'or' branch
4921 */
4922 DEBUG_VALID_MSG("saving 'or' branch");
Daniel Veillard940492d2002-04-15 10:15:25 +00004923 if (vstateVPush(ctxt, CONT->c2, NODE, (unsigned char)(DEPTH + 1),
4924 OCCURS, ROLLBACK_OR) < 0)
4925 return(-1);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004926 DEPTH++;
4927 CONT = CONT->c1;
4928 goto cont;
4929 case XML_ELEMENT_CONTENT_SEQ:
Daniel Veillard1d047672001-06-09 16:41:01 +00004930 /*
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 }
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004942 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 }
Daniel Veillard1d047672001-06-09 16:41:01 +00004954 }
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004955 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 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00004970 * First do the analysis depending on the occurrence model at
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004971 * this level.
4972 */
4973 if (ret == 0) {
4974 switch (CONT->ocur) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004975 xmlNodePtr cur;
4976
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004977 case XML_ELEMENT_CONTENT_ONCE:
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004978 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004979 DEBUG_VALID_MSG("Once branch failed, rollback");
4980 if (vstateVPop(ctxt) < 0 ) {
4981 DEBUG_VALID_MSG("exhaustion, failed");
4982 return(0);
4983 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004984 if (cur != ctxt->vstate->node)
4985 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004986 goto cont;
4987 case XML_ELEMENT_CONTENT_PLUS:
Daniel Veillard5344c602001-12-31 16:37:34 +00004988 if (OCCURRENCE == 0) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004989 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004990 DEBUG_VALID_MSG("Plus branch failed, rollback");
4991 if (vstateVPop(ctxt) < 0 ) {
4992 DEBUG_VALID_MSG("exhaustion, failed");
4993 return(0);
4994 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004995 if (cur != ctxt->vstate->node)
4996 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004997 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
Daniel Veillard5344c602001-12-31 16:37:34 +00005004 if (OCCURRENCE == 0) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005005 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");
Daniel Veillard5344c602001-12-31 16:37:34 +00005039 SET_OCCURRENCE;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005040 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");
Daniel Veillard5344c602001-12-31 16:37:34 +00005053 /* SET_OCCURRENCE; */
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005054 goto cont;
5055 }
5056 }
5057 STATE = 0;
5058
5059 /*
5060 * Then act accordingly at the parent level
5061 */
Daniel Veillard5344c602001-12-31 16:37:34 +00005062 RESET_OCCURRENCE;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005063 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) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005101 xmlNodePtr cur;
5102
5103 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005104 DEBUG_VALID_MSG("Failed, remaining input, rollback");
5105 if (vstateVPop(ctxt) < 0 ) {
5106 DEBUG_VALID_MSG("exhaustion, failed");
5107 return(0);
5108 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005109 if (cur != ctxt->vstate->node)
5110 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005111 goto cont;
5112 }
5113 if (ret == 0) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005114 xmlNodePtr cur;
5115
5116 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005117 DEBUG_VALID_MSG("Failure, rollback");
5118 if (vstateVPop(ctxt) < 0 ) {
5119 DEBUG_VALID_MSG("exhaustion, failed");
5120 return(0);
5121 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005122 if (cur != ctxt->vstate->node)
5123 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005124 goto cont;
5125 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005126 return(determinist);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005127}
Daniel Veillard23e73572002-09-19 19:56:43 +00005128#endif
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005129
5130/**
Daniel Veillardd3d06722001-08-15 12:06:36 +00005131 * xmlSnprintfElements:
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005132 * @buf: an output buffer
Daniel Veillardd3d06722001-08-15 12:06:36 +00005133 * @size: the size of the buffer
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005134 * @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
Daniel Veillardd3d06722001-08-15 12:06:36 +00005141xmlSnprintfElements(char *buf, int size, xmlNodePtr node, int glob) {
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005142 xmlNodePtr cur;
Daniel Veillardd3d06722001-08-15 12:06:36 +00005143 int len;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005144
5145 if (node == NULL) return;
5146 if (glob) strcat(buf, "(");
5147 cur = node;
5148 while (cur != NULL) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00005149 len = strlen(buf);
5150 if (size - len < 50) {
5151 if ((size - len > 4) && (buf[len - 1] != '.'))
5152 strcat(buf, " ...");
5153 return;
5154 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005155 switch (cur->type) {
5156 case XML_ELEMENT_NODE:
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005157 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
Daniel Veillard4b3a84f2002-03-19 14:36:46 +00005158 if (size - len < xmlStrlen(cur->ns->prefix) + 10) {
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005159 if ((size - len > 4) && (buf[len - 1] != '.'))
5160 strcat(buf, " ...");
5161 return;
5162 }
5163 strcat(buf, (char *) cur->ns->prefix);
5164 strcat(buf, ":");
5165 }
Daniel Veillard4b3a84f2002-03-19 14:36:46 +00005166 if (size - len < xmlStrlen(cur->name) + 10) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00005167 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;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005175 case XML_TEXT_NODE:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005176 if (xmlIsBlankNode(cur))
5177 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005178 case XML_CDATA_SECTION_NODE:
5179 case XML_ENTITY_REF_NODE:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005180 strcat(buf, "CDATA");
5181 if (cur->next != NULL)
5182 strcat(buf, " ");
5183 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005184 case XML_ATTRIBUTE_NODE:
5185 case XML_DOCUMENT_NODE:
Daniel Veillardeae522a2001-04-23 13:41:34 +00005186#ifdef LIBXML_DOCB_ENABLED
5187 case XML_DOCB_DOCUMENT_NODE:
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005188#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:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005194 strcat(buf, "???");
5195 if (cur->next != NULL)
5196 strcat(buf, " ");
5197 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005198 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:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005207 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005208 }
5209 cur = cur->next;
5210 }
5211 if (glob) strcat(buf, ")");
5212}
5213
5214/**
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005215 * xmlValidateElementContent:
5216 * @ctxt: the validation context
5217 * @child: the child list
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005218 * @elemDecl: pointer to the element declaration
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005219 * @warn: emit the error message
Daniel Veillardb9cd8b42002-09-05 10:58:49 +00005220 * @parent: the parent element (for error reporting)
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005221 *
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,
Daniel Veillardb9cd8b42002-09-05 10:58:49 +00005229 xmlElementPtr elemDecl, int warn, xmlNodePtr parent) {
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00005230 int ret = 1;
Daniel Veillard23e73572002-09-19 19:56:43 +00005231#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillard01992e02002-10-09 10:20:30 +00005232 xmlNodePtr repl = NULL, last = NULL, tmp;
Daniel Veillard23e73572002-09-19 19:56:43 +00005233#endif
Daniel Veillard01992e02002-10-09 10:20:30 +00005234 xmlNodePtr cur;
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005235 xmlElementContentPtr cont;
5236 const xmlChar *name;
5237
Gauravc570b372013-09-30 10:43:47 +08005238 if ((elemDecl == NULL) || (parent == NULL) || (ctxt == NULL))
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005239 return(-1);
5240 cont = elemDecl->content;
5241 name = elemDecl->name;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005242
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005243#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) {
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005248 return(-1);
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005249 } else {
5250 xmlRegExecCtxtPtr exec;
5251
Daniel Veillardec498e12003-02-05 11:01:50 +00005252 if (!xmlRegexpIsDeterminist(elemDecl->contModel)) {
5253 return(-1);
5254 }
Daniel Veillard01992e02002-10-09 10:20:30 +00005255 ctxt->nodeMax = 0;
5256 ctxt->nodeNr = 0;
5257 ctxt->nodeTab = NULL;
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005258 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:
Daniel Veillardea7751d2002-12-20 00:16:24 +00005281 /* TODO */
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005282 ret = 0;
5283 goto fail;
5284 case XML_ELEMENT_NODE:
5285 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00005286 xmlChar fn[50];
5287 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005288
Daniel Veillardc00cda82003-04-07 10:22:39 +00005289 fullname = xmlBuildQName(cur->name,
5290 cur->ns->prefix, fn, 50);
5291 if (fullname == NULL) {
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005292 ret = -1;
5293 goto fail;
5294 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00005295 ret = xmlRegExecPushString(exec, fullname, NULL);
5296 if ((fullname != fn) && (fullname != cur->name))
5297 xmlFree(fullname);
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005298 } 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 */
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005322 /*
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005323 * Allocate the stack
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005324 */
5325 ctxt->vstateMax = 8;
5326 ctxt->vstateTab = (xmlValidState *) xmlMalloc(
5327 ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
5328 if (ctxt->vstateTab == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00005329 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005330 return(-1);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005331 }
5332 /*
5333 * The first entry in the stack is reserved to the current state
5334 */
Daniel Veillarda9142e72001-06-19 11:07:54 +00005335 ctxt->nodeMax = 0;
5336 ctxt->nodeNr = 0;
Daniel Veillard61b33d52001-04-24 13:55:12 +00005337 ctxt->nodeTab = NULL;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005338 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);
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005346 if ((ret == -3) && (warn)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005347 xmlErrValidWarning(ctxt, child, XML_DTD_CONTENT_NOT_DETERMINIST,
5348 "Content model for Element %s is ambiguous\n",
Daniel Veillard0cc72772003-10-13 14:00:21 +00005349 name, NULL, NULL);
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005350 } else if (ret == -2) {
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005351 /*
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");
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005357 cur = child;
5358 while (cur != NULL) {
5359 switch (cur->type) {
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005360 case XML_ENTITY_REF_NODE:
5361 /*
5362 * Push the current node to be able to roll back
5363 * and process within the entity
5364 */
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005365 if ((cur->children != NULL) &&
5366 (cur->children->children != NULL)) {
5367 nodeVPush(ctxt, cur);
5368 cur = cur->children->children;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005369 continue;
5370 }
Daniel Veillard64b98c02001-06-17 17:20:21 +00005371 break;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005372 case XML_TEXT_NODE:
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005373 if (xmlIsBlankNode(cur))
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005374 break;
Daniel Veillard04e2dae2001-07-09 20:07:25 +00005375 /* no break on purpose */
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00005376 case XML_CDATA_SECTION_NODE:
5377 /* no break on purpose */
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005378 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) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00005385 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005386 xmlFreeNodeList(repl);
5387 ret = -1;
5388 goto done;
5389 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005390 tmp->type = cur->type;
5391 tmp->name = cur->name;
5392 tmp->ns = cur->ns;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005393 tmp->next = NULL;
Daniel Veillarded472f32001-12-13 08:48:14 +00005394 tmp->content = NULL;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005395 if (repl == NULL)
5396 repl = last = tmp;
5397 else {
5398 last->next = tmp;
5399 last = tmp;
5400 }
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00005401 if (cur->type == XML_CDATA_SECTION_NODE) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005402 /*
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00005403 * E59 spaces in CDATA does not match the
5404 * nonterminal S
5405 */
5406 tmp->content = xmlStrdup(BAD_CAST "CDATA");
5407 }
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005408 break;
5409 default:
5410 break;
5411 }
5412 /*
5413 * Switch to next element
5414 */
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005415 cur = cur->next;
5416 while (cur == NULL) {
5417 cur = nodeVPop(ctxt);
5418 if (cur == NULL)
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005419 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005420 cur = cur->next;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005421 }
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 }
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005436#endif /* LIBXML_REGEXP_ENABLED */
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005437 if ((warn) && ((ret != 1) && (ret != -3))) {
Daniel Veillard6ad5c4a2006-10-11 16:43:06 +00005438 if (ctxt != NULL) {
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005439 char expr[5000];
5440 char list[5000];
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005441
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005442 expr[0] = 0;
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005443 xmlSnprintfElementContent(&expr[0], 5000, cont, 1);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005444 list[0] = 0;
Daniel Veillard01992e02002-10-09 10:20:30 +00005445#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005446 if (repl != NULL)
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005447 xmlSnprintfElements(&list[0], 5000, repl, 1);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005448 else
Daniel Veillard01992e02002-10-09 10:20:30 +00005449#endif /* LIBXML_REGEXP_ENABLED */
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005450 xmlSnprintfElements(&list[0], 5000, child, 1);
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005451
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005452 if (name != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005453 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);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005456 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005457 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);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005460 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005461 } else {
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005462 if (name != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005463 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
Daniel Veillard58e44c92002-08-02 22:19:49 +00005464 "Element %s content does not follow the DTD\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005465 name, NULL, NULL);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005466 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005467 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5468 "Element content does not follow the DTD\n",
5469 NULL, NULL, NULL);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005470 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005471 }
5472 ret = 0;
5473 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005474 if (ret == -3)
5475 ret = 1;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005476
Daniel Veillard23e73572002-09-19 19:56:43 +00005477#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005478done:
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 }
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005487 ctxt->vstateMax = 0;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005488 if (ctxt->vstateTab != NULL) {
5489 xmlFree(ctxt->vstateTab);
5490 ctxt->vstateTab = NULL;
5491 }
Daniel Veillard01992e02002-10-09 10:20:30 +00005492#endif
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005493 ctxt->nodeMax = 0;
Daniel Veillarda9142e72001-06-19 11:07:54 +00005494 ctxt->nodeNr = 0;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005495 if (ctxt->nodeTab != NULL) {
5496 xmlFree(ctxt->nodeTab);
5497 ctxt->nodeTab = NULL;
5498 }
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005499 return(ret);
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005500
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005501}
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005502
Owen Taylor3473f882001-02-23 17:55:21 +00005503/**
Daniel Veillard04e2dae2001-07-09 20:07:25 +00005504 * 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
Daniel Veillard3e62adb2012-08-09 14:24:02 +08005519 if ((ctxt == NULL) || (doc == NULL) || (elem == NULL) ||
5520 (elem->type != XML_ELEMENT_NODE))
Daniel Veillard04e2dae2001-07-09 20:07:25 +00005521 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/**
Daniel Veillardea7751d2002-12-20 00:16:24 +00005571 * 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
William M. Brackedb65a72004-02-06 07:36:04 +00005581xmlValidateCheckMixed(xmlValidCtxtPtr ctxt,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005582 xmlElementContentPtr cont, const xmlChar *qname) {
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005583 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)){
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005601 xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00005602 "Internal: MIXED struct corrupted\n",
5603 NULL);
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005604 break;
5605 }
5606 cont = cont->c2;
Daniel Veillardea7751d2002-12-20 00:16:24 +00005607 }
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005608 } 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)){
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005625 xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00005626 "Internal: MIXED struct corrupted\n",
5627 NULL);
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005628 break;
5629 }
5630 cont = cont->c2;
5631 }
Daniel Veillardea7751d2002-12-20 00:16:24 +00005632 }
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
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005654 if ((ctxt == NULL) || (doc == NULL) ||
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005655 (elem == NULL) || (elem->name == NULL))
5656 return(NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005657 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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005691 xmlErrValidNode(ctxt, elem,
5692 XML_DTD_UNKNOWN_ELEM,
5693 "No declaration for element %s\n",
5694 elem->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005695 }
5696 return(elemDecl);
5697}
5698
Daniel Veillard0e298ad2003-02-04 16:14:33 +00005699#ifdef LIBXML_REGEXP_ENABLED
Daniel Veillardea7751d2002-12-20 00:16:24 +00005700/**
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
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005718 if (ctxt == NULL)
5719 return(0);
Daniel Veillardef8dd7b2003-03-23 12:02:56 +00005720/* printf("PushElem %s\n", qname); */
Daniel Veillardea7751d2002-12-20 00:16:24 +00005721 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:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005736 xmlErrValidNode(ctxt, state->node,
5737 XML_DTD_NOT_EMPTY,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005738 "Element %s was declared EMPTY this one has content\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005739 state->node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005740 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)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005750 xmlErrValidNode(ctxt, state->node,
5751 XML_DTD_NOT_PCDATA,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005752 "Element %s was declared #PCDATA but contains non text nodes\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005753 state->node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005754 ret = 0;
5755 } else {
5756 ret = xmlValidateCheckMixed(ctxt, elemDecl->content,
5757 qname);
5758 if (ret != 1) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005759 xmlErrValidNode(ctxt, state->node,
5760 XML_DTD_INVALID_CHILD,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005761 "Element %s is not declared in %s list of possible children\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005762 qname, state->node->name, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005763 }
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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005776 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);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005780 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
Michael Woodfb27e2c2012-09-28 08:59:33 +02005798 * @len: the length of the data
Daniel Veillardea7751d2002-12-20 00:16:24 +00005799 *
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
Daniel Veillardef8dd7b2003-03-23 12:02:56 +00005808/* printf("CDATA %s %d\n", data, len); */
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005809 if (ctxt == NULL)
5810 return(0);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005811 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:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005828 xmlErrValidNode(ctxt, state->node,
5829 XML_DTD_NOT_EMPTY,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005830 "Element %s was declared EMPTY this one has content\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005831 state->node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005832 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++) {
William M. Brack76e95df2003-10-18 16:20:14 +00005843 if (!IS_BLANK_CH(data[i])) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005844 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);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005848 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,
Daniel Veillard580ced82003-03-21 21:22:48 +00005880 xmlNodePtr elem ATTRIBUTE_UNUSED,
5881 const xmlChar *qname ATTRIBUTE_UNUSED) {
Daniel Veillardea7751d2002-12-20 00:16:24 +00005882 int ret = 1;
5883
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005884 if (ctxt == NULL)
5885 return(0);
Daniel Veillardef8dd7b2003-03-23 12:02:56 +00005886/* printf("PopElem %s\n", qname); */
Daniel Veillardea7751d2002-12-20 00:16:24 +00005887 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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005901 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);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005905 } 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}
Daniel Veillard0e298ad2003-02-04 16:14:33 +00005919#endif /* LIBXML_REGEXP_ENABLED */
Daniel Veillardea7751d2002-12-20 00:16:24 +00005920
5921/**
Owen Taylor3473f882001-02-23 17:55:21 +00005922 * 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;
Daniel Veillard8dc16a62002-02-19 21:08:48 +00005946 int ret = 1, tmp;
Owen Taylor3473f882001-02-23 17:55:21 +00005947 const xmlChar *name;
Daniel Veillard8dc16a62002-02-19 21:08:48 +00005948 int extsubset = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00005949
5950 CHECK_DTD;
5951
5952 if (elem == NULL) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00005953 switch (elem->type) {
5954 case XML_ATTRIBUTE_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005955 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5956 "Attribute element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005957 return(0);
5958 case XML_TEXT_NODE:
5959 if (elem->children != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005960 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5961 "Text element has children !\n",
5962 NULL,NULL,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005963 return(0);
5964 }
Owen Taylor3473f882001-02-23 17:55:21 +00005965 if (elem->ns != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005966 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5967 "Text element has namespace !\n",
5968 NULL,NULL,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005969 return(0);
5970 }
Owen Taylor3473f882001-02-23 17:55:21 +00005971 if (elem->content == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005972 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5973 "Text element has no content !\n",
5974 NULL,NULL,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005975 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:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005987 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5988 "Entity element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005989 return(0);
5990 case XML_NOTATION_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005991 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5992 "Notation element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005993 return(0);
5994 case XML_DOCUMENT_NODE:
5995 case XML_DOCUMENT_TYPE_NODE:
5996 case XML_DOCUMENT_FRAG_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005997 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5998 "Document element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005999 return(0);
6000 case XML_HTML_DOCUMENT_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006001 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6002 "HTML Document not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006003 return(0);
6004 case XML_ELEMENT_NODE:
6005 break;
6006 default:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006007 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6008 "unknown element type\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006009 return(0);
6010 }
Owen Taylor3473f882001-02-23 17:55:21 +00006011
6012 /*
Daniel Veillardea7751d2002-12-20 00:16:24 +00006013 * Fetch the declaration
Owen Taylor3473f882001-02-23 17:55:21 +00006014 */
Daniel Veillardea7751d2002-12-20 00:16:24 +00006015 elemDecl = xmlValidGetElemDecl(ctxt, doc, elem, &extsubset);
6016 if (elemDecl == NULL)
Owen Taylor3473f882001-02-23 17:55:21 +00006017 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006018
Daniel Veillardea7751d2002-12-20 00:16:24 +00006019 /*
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006020 * If vstateNr is not zero that means continuous validation is
Daniel Veillardea7751d2002-12-20 00:16:24 +00006021 * activated, do not try to check the content model at that level.
6022 */
6023 if (ctxt->vstateNr == 0) {
Daniel Veillardcbaf3992001-12-31 16:16:02 +00006024 /* Check that the element content matches the definition */
Owen Taylor3473f882001-02-23 17:55:21 +00006025 switch (elemDecl->etype) {
Daniel Veillarda10efa82001-04-18 13:09:01 +00006026 case XML_ELEMENT_TYPE_UNDEFINED:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006027 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ELEM,
6028 "No declaration for element %s\n",
6029 elem->name, NULL, NULL);
Daniel Veillarda10efa82001-04-18 13:09:01 +00006030 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006031 case XML_ELEMENT_TYPE_EMPTY:
6032 if (elem->children != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006033 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_EMPTY,
Owen Taylor3473f882001-02-23 17:55:21 +00006034 "Element %s was declared EMPTY this one has content\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006035 elem->name, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006036 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:
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006043
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006044 /* 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) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006049 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_PCDATA,
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006050 "Element %s was declared #PCDATA but contains non text nodes\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006051 elem->name, NULL, NULL);
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006052 }
6053 break;
6054 }
Owen Taylor3473f882001-02-23 17:55:21 +00006055 child = elem->children;
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006056 /* Hum, this start to get messy */
Owen Taylor3473f882001-02-23 17:55:21 +00006057 while (child != NULL) {
6058 if (child->type == XML_ELEMENT_NODE) {
6059 name = child->name;
6060 if ((child->ns != NULL) && (child->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00006061 xmlChar fn[50];
6062 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006063
Daniel Veillardc00cda82003-04-07 10:22:39 +00006064 fullname = xmlBuildQName(child->name, child->ns->prefix,
6065 fn, 50);
6066 if (fullname == NULL)
6067 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006068 cont = elemDecl->content;
6069 while (cont != NULL) {
6070 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00006071 if (xmlStrEqual(cont->name, fullname))
6072 break;
Owen Taylor3473f882001-02-23 17:55:21 +00006073 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
6074 (cont->c1 != NULL) &&
6075 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
Daniel Veillardc00cda82003-04-07 10:22:39 +00006076 if (xmlStrEqual(cont->c1->name, fullname))
6077 break;
Owen Taylor3473f882001-02-23 17:55:21 +00006078 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
6079 (cont->c1 == NULL) ||
6080 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006081 xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00006082 "Internal: MIXED struct corrupted\n",
6083 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006084 break;
6085 }
6086 cont = cont->c2;
6087 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00006088 if ((fullname != fn) && (fullname != child->name))
6089 xmlFree(fullname);
Owen Taylor3473f882001-02-23 17:55:21 +00006090 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)) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006104 xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00006105 "Internal: MIXED struct corrupted\n",
6106 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006107 break;
6108 }
6109 cont = cont->c2;
6110 }
6111 if (cont == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006112 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_CHILD,
Daniel Veillard5e5c2d02002-02-09 18:03:01 +00006113 "Element %s is not declared in %s list of possible children\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006114 name, elem->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006115 ret = 0;
6116 }
6117 }
6118child_ok:
6119 child = child->next;
6120 }
6121 break;
6122 case XML_ELEMENT_TYPE_ELEMENT:
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006123 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
William M. Brack76e95df2003-10-18 16:20:14 +00006134 while (IS_BLANK_CH(*content))
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006135 content++;
6136 if (*content == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006137 xmlErrValidNode(ctxt, elem,
6138 XML_DTD_STANDALONE_WHITE_SPACE,
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006139"standalone: %s declared in the external subset contains white spaces nodes\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006140 elem->name, NULL, NULL);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006141 ret = 0;
6142 break;
6143 }
6144 }
6145 child =child->next;
6146 }
6147 }
Owen Taylor3473f882001-02-23 17:55:21 +00006148 child = elem->children;
6149 cont = elemDecl->content;
Daniel Veillardb9cd8b42002-09-05 10:58:49 +00006150 tmp = xmlValidateElementContent(ctxt, child, elemDecl, 1, elem);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006151 if (tmp <= 0)
6152 ret = tmp;
Owen Taylor3473f882001-02-23 17:55:21 +00006153 break;
6154 }
Daniel Veillardea7751d2002-12-20 00:16:24 +00006155 } /* not continuous */
Owen Taylor3473f882001-02-23 17:55:21 +00006156
6157 /* [ VC: Required Attribute ] */
6158 attr = elemDecl->attributes;
6159 while (attr != NULL) {
6160 if (attr->def == XML_ATTRIBUTE_REQUIRED) {
Owen Taylor3473f882001-02-23 17:55:21 +00006161 int qualified = -1;
Owen Taylor3473f882001-02-23 17:55:21 +00006162
Daniel Veillarde4301c82002-02-13 13:32:35 +00006163 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)
Owen Taylor3473f882001-02-23 17:55:21 +00006170 goto found;
Daniel Veillarde4301c82002-02-13 13:32:35 +00006171 ns = ns->next;
Owen Taylor3473f882001-02-23 17:55:21 +00006172 }
Daniel Veillarde4301c82002-02-13 13:32:35 +00006173 } 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;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006184
Daniel Veillarde4301c82002-02-13 13:32:35 +00006185 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) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006199 if (qualified < 0)
Daniel Veillarde4301c82002-02-13 13:32:35 +00006200 qualified = 0;
6201 } else if (!xmlStrEqual(nameSpace->prefix,
6202 attr->prefix)) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006203 if (qualified < 1)
Daniel Veillarde4301c82002-02-13 13:32:35 +00006204 qualified = 1;
6205 } else
6206 goto found;
6207 } else {
6208 /*
6209 * We should allow applications to define namespaces
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006210 * for their application even if the DTD doesn't
Daniel Veillarde4301c82002-02-13 13:32:35 +00006211 * carry one, otherwise, basically we would always
6212 * break.
6213 */
6214 goto found;
6215 }
6216 }
6217 attrib = attrib->next;
6218 }
Owen Taylor3473f882001-02-23 17:55:21 +00006219 }
6220 if (qualified == -1) {
6221 if (attr->prefix == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006222 xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006223 "Element %s does not carry attribute %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006224 elem->name, attr->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006225 ret = 0;
6226 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006227 xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006228 "Element %s does not carry attribute %s:%s\n",
Owen Taylor3473f882001-02-23 17:55:21 +00006229 elem->name, attr->prefix,attr->name);
6230 ret = 0;
6231 }
6232 } else if (qualified == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006233 xmlErrValidWarning(ctxt, elem, XML_DTD_NO_PREFIX,
Owen Taylor3473f882001-02-23 17:55:21 +00006234 "Element %s required attribute %s:%s has no prefix\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006235 elem->name, attr->prefix, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00006236 } else if (qualified == 1) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006237 xmlErrValidWarning(ctxt, elem, XML_DTD_DIFFERENT_PREFIX,
Owen Taylor3473f882001-02-23 17:55:21 +00006238 "Element %s required attribute %s:%s has different prefix\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006239 elem->name, attr->prefix, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00006240 }
Daniel Veillarde4301c82002-02-13 13:32:35 +00006241 } 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)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006255 xmlErrValidNode(ctxt, elem,
6256 XML_DTD_ELEM_DEFAULT_NAMESPACE,
Daniel Veillarde4301c82002-02-13 13:32:35 +00006257 "Element %s namespace name for default namespace does not match the DTD\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006258 elem->name, NULL, NULL);
Daniel Veillardc7612992002-02-17 22:47:37 +00006259 ret = 0;
Daniel Veillarde4301c82002-02-13 13:32:35 +00006260 }
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)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006272 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006273 "Element %s namespace name for %s does not match the DTD\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006274 elem->name, ns->prefix, NULL);
Daniel Veillardc7612992002-02-17 22:47:37 +00006275 ret = 0;
Daniel Veillarde4301c82002-02-13 13:32:35 +00006276 }
6277 goto found;
6278 }
6279 ns = ns->next;
6280 }
6281 }
Owen Taylor3473f882001-02-23 17:55:21 +00006282 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006283found:
Owen Taylor3473f882001-02-23 17:55:21 +00006284 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;
Daniel Veillardc00cda82003-04-07 10:22:39 +00006306 int ret;
6307
Owen Taylor3473f882001-02-23 17:55:21 +00006308 if (doc == NULL) return(0);
6309
6310 root = xmlDocGetRootElement(doc);
6311 if ((root == NULL) || (root->name == NULL)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006312 xmlErrValid(ctxt, XML_DTD_NO_ROOT,
6313 "no root element\n", NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006314 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)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00006328 xmlChar fn[50];
6329 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006330
Daniel Veillardc00cda82003-04-07 10:22:39 +00006331 fullname = xmlBuildQName(root->name, root->ns->prefix, fn, 50);
6332 if (fullname == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00006333 xmlVErrMemory(ctxt, NULL);
Daniel Veillardc00cda82003-04-07 10:22:39 +00006334 return(0);
6335 }
6336 ret = xmlStrEqual(doc->intSubset->name, fullname);
6337 if ((fullname != fn) && (fullname != root->name))
6338 xmlFree(fullname);
6339 if (ret == 1)
Owen Taylor3473f882001-02-23 17:55:21 +00006340 goto name_ok;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006341 }
Owen Taylor3473f882001-02-23 17:55:21 +00006342 if ((xmlStrEqual(doc->intSubset->name, BAD_CAST "HTML")) &&
6343 (xmlStrEqual(root->name, BAD_CAST "html")))
6344 goto name_ok;
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006345 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);
Owen Taylor3473f882001-02-23 17:55:21 +00006348 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006349 }
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 *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006362 * Try to validate the subtree under an element
Owen Taylor3473f882001-02-23 17:55:21 +00006363 *
6364 * returns 1 if valid or 0 otherwise
6365 */
6366
6367int
6368xmlValidateElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlNodePtr elem) {
6369 xmlNodePtr child;
6370 xmlAttrPtr attr;
Daniel Veillarde133dd82003-10-30 10:42:20 +00006371 xmlNsPtr ns;
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006372 const xmlChar *value;
Owen Taylor3473f882001-02-23 17:55:21 +00006373 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) ||
Daniel Veillard3e62adb2012-08-09 14:24:02 +08006382 (elem->type == XML_XINCLUDE_END) ||
6383 (elem->type == XML_NAMESPACE_DECL))
Owen Taylor3473f882001-02-23 17:55:21 +00006384 return(1);
6385
6386 CHECK_DTD;
6387
Daniel Veillard10ea86c2001-06-20 13:55:33 +00006388 /*
6389 * Entities references have to be handled separately
6390 */
6391 if (elem->type == XML_ENTITY_REF_NODE) {
6392 return(1);
6393 }
6394
Owen Taylor3473f882001-02-23 17:55:21 +00006395 ret &= xmlValidateOneElement(ctxt, doc, elem);
Daniel Veillard8874b942005-08-25 13:19:21 +00006396 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 }
Daniel Veillarde133dd82003-10-30 10:42:20 +00006415 }
Owen Taylor3473f882001-02-23 17:55:21 +00006416 child = elem->children;
6417 while (child != NULL) {
6418 ret &= xmlValidateElement(ctxt, doc, child);
6419 child = child->next;
6420 }
6421
6422 return(ret);
6423}
6424
Daniel Veillard8730c562001-02-26 10:49:57 +00006425/**
6426 * xmlValidateRef:
6427 * @ref: A reference to be validated
6428 * @ctxt: Validation context
6429 * @name: Name of ID we are searching for
6430 *
6431 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00006432static void
Daniel Veillard8730c562001-02-26 10:49:57 +00006433xmlValidateRef(xmlRefPtr ref, xmlValidCtxtPtr ctxt,
Owen Taylor3473f882001-02-23 17:55:21 +00006434 const xmlChar *name) {
6435 xmlAttrPtr id;
6436 xmlAttrPtr attr;
6437
6438 if (ref == NULL)
6439 return;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006440 if ((ref->attr == NULL) && (ref->name == NULL))
Owen Taylor3473f882001-02-23 17:55:21 +00006441 return;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006442 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;
William M. Brack76e95df2003-10-18 16:20:14 +00006454 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006455 save = *cur;
6456 *cur = 0;
6457 id = xmlGetID(ctxt->doc, str);
6458 if (id == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006459 xmlErrValidNodeNr(ctxt, NULL, XML_DTD_UNKNOWN_ID,
Daniel Veillardea7751d2002-12-20 00:16:24 +00006460 "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;
William M. Brack76e95df2003-10-18 16:20:14 +00006467 while (IS_BLANK_CH(*cur)) cur++;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006468 }
6469 xmlFree(dup);
6470 } else if (attr->atype == XML_ATTRIBUTE_IDREF) {
Owen Taylor3473f882001-02-23 17:55:21 +00006471 id = xmlGetID(ctxt->doc, name);
6472 if (id == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006473 xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
Daniel Veillardea7751d2002-12-20 00:16:24 +00006474 "IDREF attribute %s references an unknown ID \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006475 attr->name, name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006476 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) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00006483 xmlVErrMemory(ctxt, "IDREFS split");
Owen Taylor3473f882001-02-23 17:55:21 +00006484 ctxt->valid = 0;
6485 return;
6486 }
6487 cur = dup;
6488 while (*cur != 0) {
6489 str = cur;
William M. Brack76e95df2003-10-18 16:20:14 +00006490 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00006491 save = *cur;
6492 *cur = 0;
6493 id = xmlGetID(ctxt->doc, str);
6494 if (id == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006495 xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
Daniel Veillardea7751d2002-12-20 00:16:24 +00006496 "IDREFS attribute %s references an unknown ID \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006497 attr->name, str, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006498 ctxt->valid = 0;
6499 }
6500 if (save == 0)
6501 break;
6502 *cur = save;
William M. Brack76e95df2003-10-18 16:20:14 +00006503 while (IS_BLANK_CH(*cur)) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00006504 }
6505 xmlFree(dup);
6506 }
6507}
6508
6509/**
Daniel Veillard8730c562001-02-26 10:49:57 +00006510 * xmlWalkValidateList:
6511 * @data: Contents of current link
6512 * @user: Value supplied by the user
6513 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00006514 * Returns 0 to abort the walk or 1 to continue
Daniel Veillard8730c562001-02-26 10:49:57 +00006515 */
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);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006542
Daniel Veillard8730c562001-02-26 10:49:57 +00006543}
6544
6545/**
Owen Taylor3473f882001-02-23 17:55:21 +00006546 * 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
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006554 *
Daniel Veillardc3da18a2003-03-18 00:31:04 +00006555 * Check all the IDREF/IDREFS attributes definition for validity
Owen Taylor3473f882001-02-23 17:55:21 +00006556 *
6557 * returns 1 if valid or 0 otherwise
6558 */
6559
6560int
6561xmlValidateDocumentFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6562 xmlRefTablePtr table;
Daniel Veillardcb3549e2011-11-11 11:25:07 +08006563 unsigned int save;
Owen Taylor3473f882001-02-23 17:55:21 +00006564
Daniel Veillardc0be74b2004-11-03 19:16:55 +00006565 if (ctxt == NULL)
6566 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006567 if (doc == NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006568 xmlErrValid(ctxt, XML_DTD_NO_DOC,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00006569 "xmlValidateDocumentFinal: doc == NULL\n", NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006570 return(0);
6571 }
6572
Daniel Veillardcb3549e2011-11-11 11:25:07 +08006573 /* trick to get correct line id report */
6574 save = ctxt->finishDtd;
6575 ctxt->finishDtd = 0;
6576
Owen Taylor3473f882001-02-23 17:55:21 +00006577 /*
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);
Daniel Veillardcb3549e2011-11-11 11:25:07 +08006590
6591 ctxt->finishDtd = save;
Owen Taylor3473f882001-02-23 17:55:21 +00006592 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 *
William M. Brack367df6e2004-10-23 18:14:36 +00006603 * 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.
Owen Taylor3473f882001-02-23 17:55:21 +00006607 *
6608 * returns 1 if valid or 0 otherwise
6609 */
6610
6611int
6612xmlValidateDtd(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlDtdPtr dtd) {
6613 int ret;
William M. Brack367df6e2004-10-23 18:14:36 +00006614 xmlDtdPtr oldExt, oldInt;
Owen Taylor3473f882001-02-23 17:55:21 +00006615 xmlNodePtr root;
6616
6617 if (dtd == NULL) return(0);
6618 if (doc == NULL) return(0);
6619 oldExt = doc->extSubset;
William M. Brack367df6e2004-10-23 18:14:36 +00006620 oldInt = doc->intSubset;
Owen Taylor3473f882001-02-23 17:55:21 +00006621 doc->extSubset = dtd;
William M. Brack367df6e2004-10-23 18:14:36 +00006622 doc->intSubset = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00006623 ret = xmlValidateRoot(ctxt, doc);
6624 if (ret == 0) {
6625 doc->extSubset = oldExt;
William M. Brack367df6e2004-10-23 18:14:36 +00006626 doc->intSubset = oldInt;
Owen Taylor3473f882001-02-23 17:55:21 +00006627 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;
William M. Brack367df6e2004-10-23 18:14:36 +00006641 doc->intSubset = oldInt;
Owen Taylor3473f882001-02-23 17:55:21 +00006642 return(ret);
6643}
6644
Daniel Veillard56a4cb82001-03-24 17:00:36 +00006645static void
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006646xmlValidateNotationCallback(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
Daniel Veillard878eab02002-02-19 13:46:09 +00006653 if (notation != NULL) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006654 int ret;
6655
6656 ret = xmlValidateNotationUse(ctxt, cur->doc, notation);
6657 if (ret != 1) {
Daniel Veillard878eab02002-02-19 13:46:09 +00006658 ctxt->valid = 0;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006659 }
6660 }
6661 }
6662}
6663
6664static void
Owen Taylor3473f882001-02-23 17:55:21 +00006665xmlValidateAttributeCallback(xmlAttributePtr cur, xmlValidCtxtPtr ctxt,
Daniel Veillardc86a4fa2001-03-26 16:28:29 +00006666 const xmlChar *name ATTRIBUTE_UNUSED) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006667 int ret;
Daniel Veillard878eab02002-02-19 13:46:09 +00006668 xmlDocPtr doc;
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006669 xmlElementPtr elem = NULL;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006670
Owen Taylor3473f882001-02-23 17:55:21 +00006671 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) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006686
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006687 ret = xmlValidateAttributeValue2(ctxt, ctxt->doc, cur->name,
6688 cur->atype, cur->defaultValue);
6689 if ((ret == 0) && (ctxt->valid == 1))
6690 ctxt->valid = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00006691 }
6692 if (cur->tree != NULL) {
6693 xmlEnumerationPtr tree = cur->tree;
6694 while (tree != NULL) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006695 ret = xmlValidateAttributeValue2(ctxt, ctxt->doc,
Owen Taylor3473f882001-02-23 17:55:21 +00006696 cur->name, cur->atype, tree->name);
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006697 if ((ret == 0) && (ctxt->valid == 1))
6698 ctxt->valid = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00006699 tree = tree->next;
6700 }
6701 }
6702 }
Daniel Veillard878eab02002-02-19 13:46:09 +00006703 if (cur->atype == XML_ATTRIBUTE_NOTATION) {
6704 doc = cur->doc;
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006705 if (cur->elem == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006706 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard878eab02002-02-19 13:46:09 +00006707 "xmlValidateAttributeCallback(%s): internal error\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006708 (const char *) cur->name);
Daniel Veillard878eab02002-02-19 13:46:09 +00006709 return;
6710 }
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006711
6712 if (doc != NULL)
6713 elem = xmlGetDtdElementDesc(doc->intSubset, cur->elem);
6714 if ((elem == NULL) && (doc != NULL))
Daniel Veillard878eab02002-02-19 13:46:09 +00006715 elem = xmlGetDtdElementDesc(doc->extSubset, cur->elem);
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006716 if ((elem == NULL) && (cur->parent != NULL) &&
6717 (cur->parent->type == XML_DTD_NODE))
6718 elem = xmlGetDtdElementDesc((xmlDtdPtr) cur->parent, cur->elem);
Daniel Veillard878eab02002-02-19 13:46:09 +00006719 if (elem == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006720 xmlErrValidNode(ctxt, NULL, XML_DTD_UNKNOWN_ELEM,
Daniel Veillard878eab02002-02-19 13:46:09 +00006721 "attribute %s: could not find decl for element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006722 cur->name, cur->elem, NULL);
Daniel Veillard878eab02002-02-19 13:46:09 +00006723 return;
6724 }
6725 if (elem->etype == XML_ELEMENT_TYPE_EMPTY) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006726 xmlErrValidNode(ctxt, NULL, XML_DTD_EMPTY_NOTATION,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006727 "NOTATION attribute %s declared for EMPTY element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006728 cur->name, cur->elem, NULL);
Daniel Veillard878eab02002-02-19 13:46:09 +00006729 ctxt->valid = 0;
6730 }
6731 }
Owen Taylor3473f882001-02-23 17:55:21 +00006732}
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
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006743 * - check that ENTITY and ENTITIES type attributes default or
Owen Taylor3473f882001-02-23 17:55:21 +00006744 * possible values matches one of the defined entities.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006745 * - check that NOTATION type attributes default or
Owen Taylor3473f882001-02-23 17:55:21 +00006746 * possible values matches one of the defined notations.
6747 *
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006748 * returns 1 if valid or 0 if invalid and -1 if not well-formed
Owen Taylor3473f882001-02-23 17:55:21 +00006749 */
6750
6751int
6752xmlValidateDtdFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
Owen Taylor3473f882001-02-23 17:55:21 +00006753 xmlDtdPtr dtd;
6754 xmlAttributeTablePtr table;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006755 xmlEntitiesTablePtr entities;
Owen Taylor3473f882001-02-23 17:55:21 +00006756
Daniel Veillard2cba4152008-08-27 11:45:41 +00006757 if ((doc == NULL) || (ctxt == NULL)) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006758 if ((doc->intSubset == NULL) && (doc->extSubset == NULL))
6759 return(0);
6760 ctxt->doc = doc;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006761 ctxt->valid = 1;
Owen Taylor3473f882001-02-23 17:55:21 +00006762 dtd = doc->intSubset;
6763 if ((dtd != NULL) && (dtd->attributes != NULL)) {
6764 table = (xmlAttributeTablePtr) dtd->attributes;
6765 xmlHashScan(table, (xmlHashScanner) xmlValidateAttributeCallback, ctxt);
Daniel Veillard878eab02002-02-19 13:46:09 +00006766 }
6767 if ((dtd != NULL) && (dtd->entities != NULL)) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006768 entities = (xmlEntitiesTablePtr) dtd->entities;
6769 xmlHashScan(entities, (xmlHashScanner) xmlValidateNotationCallback,
6770 ctxt);
Owen Taylor3473f882001-02-23 17:55:21 +00006771 }
6772 dtd = doc->extSubset;
6773 if ((dtd != NULL) && (dtd->attributes != NULL)) {
6774 table = (xmlAttributeTablePtr) dtd->attributes;
6775 xmlHashScan(table, (xmlHashScanner) xmlValidateAttributeCallback, ctxt);
Daniel Veillard878eab02002-02-19 13:46:09 +00006776 }
6777 if ((dtd != NULL) && (dtd->entities != NULL)) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006778 entities = (xmlEntitiesTablePtr) dtd->entities;
6779 xmlHashScan(entities, (xmlHashScanner) xmlValidateNotationCallback,
6780 ctxt);
Owen Taylor3473f882001-02-23 17:55:21 +00006781 }
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
Daniel Veillardc0be74b2004-11-03 19:16:55 +00006804 if (doc == NULL)
6805 return(0);
Daniel Veillard2fd85422002-10-16 14:32:41 +00006806 if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006807 xmlErrValid(ctxt, XML_DTD_NO_DTD,
6808 "no DTD found!\n", NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006809 return(0);
Daniel Veillard2fd85422002-10-16 14:32:41 +00006810 }
Owen Taylor3473f882001-02-23 17:55:21 +00006811 if ((doc->intSubset != NULL) && ((doc->intSubset->SystemID != NULL) ||
6812 (doc->intSubset->ExternalID != NULL)) && (doc->extSubset == NULL)) {
William M. Brack8c22f9f2004-08-06 16:23:27 +00006813 xmlChar *sysID;
6814 if (doc->intSubset->SystemID != NULL) {
William M. Brackbebe7302004-08-05 06:46:47 +00006815 sysID = xmlBuildURI(doc->intSubset->SystemID,
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006816 doc->URL);
William M. Brack8c22f9f2004-08-06 16:23:27 +00006817 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
William M. Brackbebe7302004-08-05 06:46:47 +00006824 sysID = NULL;
William M. Brack8c22f9f2004-08-06 16:23:27 +00006825 doc->extSubset = xmlParseDTD(doc->intSubset->ExternalID,
William M. Brackbebe7302004-08-05 06:46:47 +00006826 (const xmlChar *)sysID);
William M. Brackbebe7302004-08-05 06:46:47 +00006827 if (sysID != NULL)
6828 xmlFree(sysID);
Owen Taylor3473f882001-02-23 17:55:21 +00006829 if (doc->extSubset == NULL) {
6830 if (doc->intSubset->SystemID != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006831 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
Owen Taylor3473f882001-02-23 17:55:21 +00006832 "Could not load the external subset \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006833 (const char *) doc->intSubset->SystemID);
Owen Taylor3473f882001-02-23 17:55:21 +00006834 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006835 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
Owen Taylor3473f882001-02-23 17:55:21 +00006836 "Could not load the external subset \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006837 (const char *) doc->intSubset->ExternalID);
Owen Taylor3473f882001-02-23 17:55:21 +00006838 }
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
Owen Taylor3473f882001-02-23 17:55:21 +00006860/************************************************************************
6861 * *
6862 * Routines for dynamic validation editing *
6863 * *
6864 ************************************************************************/
6865
6866/**
6867 * xmlValidGetPotentialChildren:
6868 * @ctree: an element content tree
Daniel Veillard7802ba52005-10-27 11:56:20 +00006869 * @names: an array to store the list of child names
Owen Taylor3473f882001-02-23 17:55:21 +00006870 * @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
Daniel Veillard7802ba52005-10-27 11:56:20 +00006879xmlValidGetPotentialChildren(xmlElementContent *ctree,
6880 const xmlChar **names,
Owen Taylor3473f882001-02-23 17:55:21 +00006881 int *len, int max) {
6882 int i;
6883
Daniel Veillard7802ba52005-10-27 11:56:20 +00006884 if ((ctree == NULL) || (names == NULL) || (len == NULL))
Owen Taylor3473f882001-02-23 17:55:21 +00006885 return(-1);
6886 if (*len >= max) return(*len);
6887
6888 switch (ctree->type) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006889 case XML_ELEMENT_CONTENT_PCDATA:
Owen Taylor3473f882001-02-23 17:55:21 +00006890 for (i = 0; i < *len;i++)
Daniel Veillard7802ba52005-10-27 11:56:20 +00006891 if (xmlStrEqual(BAD_CAST "#PCDATA", names[i])) return(*len);
6892 names[(*len)++] = BAD_CAST "#PCDATA";
Owen Taylor3473f882001-02-23 17:55:21 +00006893 break;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006894 case XML_ELEMENT_CONTENT_ELEMENT:
Owen Taylor3473f882001-02-23 17:55:21 +00006895 for (i = 0; i < *len;i++)
Daniel Veillard7802ba52005-10-27 11:56:20 +00006896 if (xmlStrEqual(ctree->name, names[i])) return(*len);
6897 names[(*len)++] = ctree->name;
Owen Taylor3473f882001-02-23 17:55:21 +00006898 break;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006899 case XML_ELEMENT_CONTENT_SEQ:
Daniel Veillard7802ba52005-10-27 11:56:20 +00006900 xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6901 xmlValidGetPotentialChildren(ctree->c2, names, len, max);
Owen Taylor3473f882001-02-23 17:55:21 +00006902 break;
6903 case XML_ELEMENT_CONTENT_OR:
Daniel Veillard7802ba52005-10-27 11:56:20 +00006904 xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6905 xmlValidGetPotentialChildren(ctree->c2, names, len, max);
Owen Taylor3473f882001-02-23 17:55:21 +00006906 break;
6907 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006908
Owen Taylor3473f882001-02-23 17:55:21 +00006909 return(*len);
6910}
6911
William M. Brack9333cc22004-06-24 08:33:40 +00006912/*
6913 * Dummy function to suppress messages while we try out valid elements
6914 */
Daniel Veillardffa3c742005-07-21 13:24:09 +00006915static void XMLCDECL xmlNoValidityErr(void *ctx ATTRIBUTE_UNUSED,
William M. Brack9333cc22004-06-24 08:33:40 +00006916 const char *msg ATTRIBUTE_UNUSED, ...) {
6917 return;
6918}
6919
Owen Taylor3473f882001-02-23 17:55:21 +00006920/**
6921 * xmlValidGetValidElements:
6922 * @prev: an element to insert after
6923 * @next: an element to insert next
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00006924 * @names: an array to store the list of child names
Owen Taylor3473f882001-02-23 17:55:21 +00006925 * @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
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00006946xmlValidGetValidElements(xmlNode *prev, xmlNode *next, const xmlChar **names,
Owen Taylor3473f882001-02-23 17:55:21 +00006947 int max) {
Daniel Veillardf69bb4b2001-05-19 13:24:56 +00006948 xmlValidCtxt vctxt;
Owen Taylor3473f882001-02-23 17:55:21 +00006949 int nb_valid_elements = 0;
Daniel Veillarde18bce02014-02-06 10:47:20 +01006950 const xmlChar *elements[256]={0};
Owen Taylor3473f882001-02-23 17:55:21 +00006951 int nb_elements = 0, i;
Daniel Veillard5e5c2d02002-02-09 18:03:01 +00006952 const xmlChar *name;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006953
Owen Taylor3473f882001-02-23 17:55:21 +00006954 xmlNode *ref_node;
6955 xmlNode *parent;
6956 xmlNode *test_node;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006957
Owen Taylor3473f882001-02-23 17:55:21 +00006958 xmlNode *prev_next;
6959 xmlNode *next_prev;
6960 xmlNode *parent_childs;
6961 xmlNode *parent_last;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006962
Owen Taylor3473f882001-02-23 17:55:21 +00006963 xmlElement *element_desc;
6964
6965 if (prev == NULL && next == NULL)
6966 return(-1);
6967
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00006968 if (names == NULL) return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00006969 if (max <= 0) return(-1);
6970
William M. Brack9333cc22004-06-24 08:33:40 +00006971 memset(&vctxt, 0, sizeof (xmlValidCtxt));
6972 vctxt.error = xmlNoValidityErr; /* this suppresses err/warn output */
6973
Owen Taylor3473f882001-02-23 17:55:21 +00006974 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);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006987
Owen Taylor3473f882001-02-23 17:55:21 +00006988 /*
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
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006998 */
Daniel Veillard95ddcd32004-10-26 21:53:55 +00006999 test_node = xmlNewDocNode (ref_node->doc, NULL, BAD_CAST "<!dummy?>", NULL);
Gaurav Gupta658b86c2014-08-07 11:19:03 +08007000 if (test_node == NULL)
7001 return(-1);
7002
Owen Taylor3473f882001-02-23 17:55:21 +00007003 test_node->parent = parent;
7004 test_node->prev = prev;
7005 test_node->next = next;
Daniel Veillardd455d792002-02-08 13:37:46 +00007006 name = test_node->name;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007007
Owen Taylor3473f882001-02-23 17:55:21 +00007008 if (prev) prev->next = test_node;
7009 else parent->children = test_node;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007010
Owen Taylor3473f882001-02-23 17:55:21 +00007011 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);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007020
Owen Taylor3473f882001-02-23 17:55:21 +00007021 for (i = 0;i < nb_elements;i++) {
7022 test_node->name = elements[i];
Daniel Veillardf69bb4b2001-05-19 13:24:56 +00007023 if (xmlValidateOneElement(&vctxt, parent->doc, parent)) {
Owen Taylor3473f882001-02-23 17:55:21 +00007024 int j;
7025
7026 for (j = 0; j < nb_valid_elements;j++)
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00007027 if (xmlStrEqual(elements[i], names[j])) break;
7028 names[nb_valid_elements++] = elements[i];
Owen Taylor3473f882001-02-23 17:55:21 +00007029 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;
Daniel Veillardd455d792002-02-08 13:37:46 +00007040
7041 /*
7042 * Free up the dummy node
7043 */
7044 test_node->name = name;
7045 xmlFreeNode(test_node);
7046
Owen Taylor3473f882001-02-23 17:55:21 +00007047 return(nb_valid_elements);
7048}
Daniel Veillard4432df22003-09-28 18:58:27 +00007049#endif /* LIBXML_VALID_ENABLED */
7050
Daniel Veillard5d4644e2005-04-01 13:11:58 +00007051#define bottom_valid
7052#include "elfgcchack.h"