blob: 57a1a9ee285206846d7677ab1560608add343d0b [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 */
David Kilzer4472c3a2016-05-13 15:13:17 +080096static void LIBXML_ATTR_FORMAT(3,0)
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 */
David Kilzer4472c3a2016-05-13 15:13:17 +0800140static void LIBXML_ATTR_FORMAT(4,0)
Daniel Veillardf54cd532004-02-25 11:52:31 +0000141xmlErrValidNode(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,
Nick Wellnhoferc709f3f2017-09-07 19:52:39 +0200166 (const char *) str2,
Daniel Veillardf54cd532004-02-25 11:52:31 +0000167 (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 */
David Kilzer4472c3a2016-05-13 15:13:17 +0800183static void LIBXML_ATTR_FORMAT(4,0)
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 */
David Kilzer4472c3a2016-05-13 15:13:17 +0800224static void LIBXML_ATTR_FORMAT(4,0)
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,
Nick Wellnhoferc709f3f2017-09-07 19:52:39 +0200250 (const char *) str2,
Daniel Veillardbb5abab2003-10-03 22:21:51 +0000251 (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:
Daniel Veillard94691dc2017-06-07 16:47:36 +02001175 if ((content->c1 != NULL) &&
1176 ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1177 (content->c1->type == XML_ELEMENT_CONTENT_SEQ)))
Owen Taylor3473f882001-02-23 17:55:21 +00001178 xmlDumpElementContent(buf, content->c1, 1);
1179 else
1180 xmlDumpElementContent(buf, content->c1, 0);
1181 xmlBufferWriteChar(buf, " , ");
Daniel Veillard94691dc2017-06-07 16:47:36 +02001182 if ((content->c2 != NULL) &&
1183 ((content->c2->type == XML_ELEMENT_CONTENT_OR) ||
1184 ((content->c2->type == XML_ELEMENT_CONTENT_SEQ) &&
1185 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE))))
Owen Taylor3473f882001-02-23 17:55:21 +00001186 xmlDumpElementContent(buf, content->c2, 1);
1187 else
1188 xmlDumpElementContent(buf, content->c2, 0);
1189 break;
1190 case XML_ELEMENT_CONTENT_OR:
Daniel Veillard94691dc2017-06-07 16:47:36 +02001191 if ((content->c1 != NULL) &&
1192 ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1193 (content->c1->type == XML_ELEMENT_CONTENT_SEQ)))
Owen Taylor3473f882001-02-23 17:55:21 +00001194 xmlDumpElementContent(buf, content->c1, 1);
1195 else
1196 xmlDumpElementContent(buf, content->c1, 0);
1197 xmlBufferWriteChar(buf, " | ");
Daniel Veillard94691dc2017-06-07 16:47:36 +02001198 if ((content->c2 != NULL) &&
1199 ((content->c2->type == XML_ELEMENT_CONTENT_SEQ) ||
1200 ((content->c2->type == XML_ELEMENT_CONTENT_OR) &&
1201 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE))))
Owen Taylor3473f882001-02-23 17:55:21 +00001202 xmlDumpElementContent(buf, content->c2, 1);
1203 else
1204 xmlDumpElementContent(buf, content->c2, 0);
1205 break;
1206 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001207 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001208 "Internal: ELEMENT content corrupted invalid type\n",
1209 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001210 }
1211 if (glob)
1212 xmlBufferWriteChar(buf, ")");
1213 switch (content->ocur) {
1214 case XML_ELEMENT_CONTENT_ONCE:
1215 break;
1216 case XML_ELEMENT_CONTENT_OPT:
1217 xmlBufferWriteChar(buf, "?");
1218 break;
1219 case XML_ELEMENT_CONTENT_MULT:
1220 xmlBufferWriteChar(buf, "*");
1221 break;
1222 case XML_ELEMENT_CONTENT_PLUS:
1223 xmlBufferWriteChar(buf, "+");
1224 break;
1225 }
1226}
1227
1228/**
1229 * xmlSprintfElementContent:
1230 * @buf: an output buffer
1231 * @content: An element table
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001232 * @englob: 1 if one must print the englobing parenthesis, 0 otherwise
Owen Taylor3473f882001-02-23 17:55:21 +00001233 *
Daniel Veillardd3d06722001-08-15 12:06:36 +00001234 * Deprecated, unsafe, use xmlSnprintfElementContent
1235 */
1236void
1237xmlSprintfElementContent(char *buf ATTRIBUTE_UNUSED,
1238 xmlElementContentPtr content ATTRIBUTE_UNUSED,
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001239 int englob ATTRIBUTE_UNUSED) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00001240}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001241#endif /* LIBXML_OUTPUT_ENABLED */
Daniel Veillardd3d06722001-08-15 12:06:36 +00001242
1243/**
1244 * xmlSnprintfElementContent:
1245 * @buf: an output buffer
1246 * @size: the buffer size
1247 * @content: An element table
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001248 * @englob: 1 if one must print the englobing parenthesis, 0 otherwise
Daniel Veillardd3d06722001-08-15 12:06:36 +00001249 *
Owen Taylor3473f882001-02-23 17:55:21 +00001250 * This will dump the content of the element content definition
1251 * Intended just for the debug routine
1252 */
1253void
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001254xmlSnprintfElementContent(char *buf, int size, xmlElementContentPtr content, int englob) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00001255 int len;
1256
Owen Taylor3473f882001-02-23 17:55:21 +00001257 if (content == NULL) return;
Daniel Veillardd3d06722001-08-15 12:06:36 +00001258 len = strlen(buf);
1259 if (size - len < 50) {
1260 if ((size - len > 4) && (buf[len - 1] != '.'))
1261 strcat(buf, " ...");
1262 return;
1263 }
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001264 if (englob) strcat(buf, "(");
Owen Taylor3473f882001-02-23 17:55:21 +00001265 switch (content->type) {
1266 case XML_ELEMENT_CONTENT_PCDATA:
1267 strcat(buf, "#PCDATA");
1268 break;
Nick Wellnhofer932cc982017-06-03 02:01:29 +02001269 case XML_ELEMENT_CONTENT_ELEMENT: {
1270 int qnameLen = xmlStrlen(content->name);
1271
1272 if (content->prefix != NULL)
1273 qnameLen += xmlStrlen(content->prefix) + 1;
1274 if (size - len < qnameLen + 10) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00001275 strcat(buf, " ...");
1276 return;
1277 }
Nick Wellnhofer932cc982017-06-03 02:01:29 +02001278 if (content->prefix != NULL) {
1279 strcat(buf, (char *) content->prefix);
1280 strcat(buf, ":");
1281 }
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001282 if (content->name != NULL)
1283 strcat(buf, (char *) content->name);
Owen Taylor3473f882001-02-23 17:55:21 +00001284 break;
Nick Wellnhofer932cc982017-06-03 02:01:29 +02001285 }
Owen Taylor3473f882001-02-23 17:55:21 +00001286 case XML_ELEMENT_CONTENT_SEQ:
1287 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1288 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001289 xmlSnprintfElementContent(buf, size, content->c1, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001290 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001291 xmlSnprintfElementContent(buf, size, content->c1, 0);
1292 len = strlen(buf);
1293 if (size - len < 50) {
1294 if ((size - len > 4) && (buf[len - 1] != '.'))
1295 strcat(buf, " ...");
1296 return;
1297 }
Owen Taylor3473f882001-02-23 17:55:21 +00001298 strcat(buf, " , ");
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00001299 if (((content->c2->type == XML_ELEMENT_CONTENT_OR) ||
1300 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)) &&
1301 (content->c2->type != XML_ELEMENT_CONTENT_ELEMENT))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001302 xmlSnprintfElementContent(buf, size, content->c2, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001303 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001304 xmlSnprintfElementContent(buf, size, content->c2, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00001305 break;
1306 case XML_ELEMENT_CONTENT_OR:
1307 if ((content->c1->type == XML_ELEMENT_CONTENT_OR) ||
1308 (content->c1->type == XML_ELEMENT_CONTENT_SEQ))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001309 xmlSnprintfElementContent(buf, size, content->c1, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001310 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001311 xmlSnprintfElementContent(buf, size, content->c1, 0);
1312 len = strlen(buf);
1313 if (size - len < 50) {
1314 if ((size - len > 4) && (buf[len - 1] != '.'))
1315 strcat(buf, " ...");
1316 return;
1317 }
Owen Taylor3473f882001-02-23 17:55:21 +00001318 strcat(buf, " | ");
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00001319 if (((content->c2->type == XML_ELEMENT_CONTENT_SEQ) ||
1320 (content->c2->ocur != XML_ELEMENT_CONTENT_ONCE)) &&
1321 (content->c2->type != XML_ELEMENT_CONTENT_ELEMENT))
Daniel Veillardd3d06722001-08-15 12:06:36 +00001322 xmlSnprintfElementContent(buf, size, content->c2, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00001323 else
Daniel Veillardd3d06722001-08-15 12:06:36 +00001324 xmlSnprintfElementContent(buf, size, content->c2, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00001325 break;
1326 }
Nick Wellnhofer932cc982017-06-03 02:01:29 +02001327 if (size - strlen(buf) <= 2) return;
Daniel Veillard1b75c3b2005-06-26 21:49:08 +00001328 if (englob)
Owen Taylor3473f882001-02-23 17:55:21 +00001329 strcat(buf, ")");
1330 switch (content->ocur) {
1331 case XML_ELEMENT_CONTENT_ONCE:
1332 break;
1333 case XML_ELEMENT_CONTENT_OPT:
1334 strcat(buf, "?");
1335 break;
1336 case XML_ELEMENT_CONTENT_MULT:
1337 strcat(buf, "*");
1338 break;
1339 case XML_ELEMENT_CONTENT_PLUS:
1340 strcat(buf, "+");
1341 break;
1342 }
1343}
1344
1345/****************************************************************
1346 * *
1347 * Registration of DTD declarations *
1348 * *
1349 ****************************************************************/
1350
1351/**
Owen Taylor3473f882001-02-23 17:55:21 +00001352 * xmlFreeElement:
1353 * @elem: An element
1354 *
1355 * Deallocate the memory used by an element definition
1356 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001357static void
Owen Taylor3473f882001-02-23 17:55:21 +00001358xmlFreeElement(xmlElementPtr elem) {
1359 if (elem == NULL) return;
1360 xmlUnlinkNode((xmlNodePtr) elem);
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001361 xmlFreeDocElementContent(elem->doc, elem->content);
Owen Taylor3473f882001-02-23 17:55:21 +00001362 if (elem->name != NULL)
1363 xmlFree((xmlChar *) elem->name);
1364 if (elem->prefix != NULL)
1365 xmlFree((xmlChar *) elem->prefix);
Daniel Veillard84d70a42002-09-16 10:51:38 +00001366#ifdef LIBXML_REGEXP_ENABLED
1367 if (elem->contModel != NULL)
1368 xmlRegFreeRegexp(elem->contModel);
1369#endif
Owen Taylor3473f882001-02-23 17:55:21 +00001370 xmlFree(elem);
1371}
1372
1373
1374/**
1375 * xmlAddElementDecl:
1376 * @ctxt: the validation context
1377 * @dtd: pointer to the DTD
1378 * @name: the entity name
1379 * @type: the element type
1380 * @content: the element content tree or NULL
1381 *
1382 * Register a new element declaration
1383 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001384 * Returns NULL if not, otherwise the entity
Owen Taylor3473f882001-02-23 17:55:21 +00001385 */
1386xmlElementPtr
William M. Brackedb65a72004-02-06 07:36:04 +00001387xmlAddElementDecl(xmlValidCtxtPtr ctxt,
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001388 xmlDtdPtr dtd, const xmlChar *name,
Owen Taylor3473f882001-02-23 17:55:21 +00001389 xmlElementTypeVal type,
1390 xmlElementContentPtr content) {
1391 xmlElementPtr ret;
1392 xmlElementTablePtr table;
Daniel Veillarda10efa82001-04-18 13:09:01 +00001393 xmlAttributePtr oldAttributes = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00001394 xmlChar *ns, *uqname;
1395
1396 if (dtd == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001397 return(NULL);
1398 }
1399 if (name == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001400 return(NULL);
1401 }
Daniel Veillard316a5c32005-01-23 22:56:39 +00001402
Owen Taylor3473f882001-02-23 17:55:21 +00001403 switch (type) {
1404 case XML_ELEMENT_TYPE_EMPTY:
1405 if (content != NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001406 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001407 "xmlAddElementDecl: content != NULL for EMPTY\n",
1408 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001409 return(NULL);
1410 }
1411 break;
1412 case XML_ELEMENT_TYPE_ANY:
1413 if (content != NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001414 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001415 "xmlAddElementDecl: content != NULL for ANY\n",
1416 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001417 return(NULL);
1418 }
1419 break;
1420 case XML_ELEMENT_TYPE_MIXED:
1421 if (content == NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001422 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001423 "xmlAddElementDecl: content == NULL for MIXED\n",
1424 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001425 return(NULL);
1426 }
1427 break;
1428 case XML_ELEMENT_TYPE_ELEMENT:
1429 if (content == NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001430 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001431 "xmlAddElementDecl: content == NULL for ELEMENT\n",
1432 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001433 return(NULL);
1434 }
1435 break;
1436 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001437 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001438 "Internal: ELEMENT decl corrupted invalid type\n",
1439 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001440 return(NULL);
1441 }
1442
1443 /*
1444 * check if name is a QName
1445 */
1446 uqname = xmlSplitQName2(name, &ns);
1447 if (uqname != NULL)
1448 name = uqname;
1449
1450 /*
1451 * Create the Element table if needed.
1452 */
1453 table = (xmlElementTablePtr) dtd->elements;
1454 if (table == NULL) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00001455 xmlDictPtr dict = NULL;
1456
1457 if (dtd->doc != NULL)
1458 dict = dtd->doc->dict;
1459 table = xmlHashCreateDict(0, dict);
Owen Taylor3473f882001-02-23 17:55:21 +00001460 dtd->elements = (void *) table;
1461 }
1462 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001463 xmlVErrMemory(ctxt,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001464 "xmlAddElementDecl: Table creation failed!\n");
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001465 if (uqname != NULL)
1466 xmlFree(uqname);
1467 if (ns != NULL)
1468 xmlFree(ns);
Owen Taylor3473f882001-02-23 17:55:21 +00001469 return(NULL);
1470 }
1471
Daniel Veillarda10efa82001-04-18 13:09:01 +00001472 /*
1473 * lookup old attributes inserted on an undefined element in the
1474 * internal subset.
1475 */
1476 if ((dtd->doc != NULL) && (dtd->doc->intSubset != NULL)) {
1477 ret = xmlHashLookup2(dtd->doc->intSubset->elements, name, ns);
1478 if ((ret != NULL) && (ret->etype == XML_ELEMENT_TYPE_UNDEFINED)) {
1479 oldAttributes = ret->attributes;
1480 ret->attributes = NULL;
1481 xmlHashRemoveEntry2(dtd->doc->intSubset->elements, name, ns, NULL);
1482 xmlFreeElement(ret);
1483 }
Owen Taylor3473f882001-02-23 17:55:21 +00001484 }
Owen Taylor3473f882001-02-23 17:55:21 +00001485
1486 /*
Daniel Veillarda10efa82001-04-18 13:09:01 +00001487 * The element may already be present if one of its attribute
1488 * was registered first
1489 */
1490 ret = xmlHashLookup2(table, name, ns);
1491 if (ret != NULL) {
1492 if (ret->etype != XML_ELEMENT_TYPE_UNDEFINED) {
Daniel Veillard4432df22003-09-28 18:58:27 +00001493#ifdef LIBXML_VALID_ENABLED
Daniel Veillarda10efa82001-04-18 13:09:01 +00001494 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001495 * The element is already defined in this DTD.
Daniel Veillarda10efa82001-04-18 13:09:01 +00001496 */
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001497 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ELEM_REDEFINED,
1498 "Redefinition of element %s\n",
1499 name, NULL, NULL);
Daniel Veillard4432df22003-09-28 18:58:27 +00001500#endif /* LIBXML_VALID_ENABLED */
Daniel Veillarda10efa82001-04-18 13:09:01 +00001501 if (uqname != NULL)
1502 xmlFree(uqname);
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001503 if (ns != NULL)
1504 xmlFree(ns);
Daniel Veillarda10efa82001-04-18 13:09:01 +00001505 return(NULL);
1506 }
William M. Brackd6e347e2005-04-15 01:34:41 +00001507 if (ns != NULL) {
1508 xmlFree(ns);
1509 ns = NULL;
1510 }
Daniel Veillarda10efa82001-04-18 13:09:01 +00001511 } else {
1512 ret = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
1513 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001514 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001515 if (uqname != NULL)
1516 xmlFree(uqname);
1517 if (ns != NULL)
1518 xmlFree(ns);
Daniel Veillarda10efa82001-04-18 13:09:01 +00001519 return(NULL);
1520 }
1521 memset(ret, 0, sizeof(xmlElement));
1522 ret->type = XML_ELEMENT_DECL;
1523
1524 /*
1525 * fill the structure.
1526 */
1527 ret->name = xmlStrdup(name);
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001528 if (ret->name == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001529 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00001530 if (uqname != NULL)
1531 xmlFree(uqname);
1532 if (ns != NULL)
1533 xmlFree(ns);
1534 xmlFree(ret);
1535 return(NULL);
1536 }
Daniel Veillarda10efa82001-04-18 13:09:01 +00001537 ret->prefix = ns;
1538
1539 /*
1540 * Validity Check:
1541 * Insertion must not fail
1542 */
1543 if (xmlHashAddEntry2(table, name, ns, ret)) {
Daniel Veillard4432df22003-09-28 18:58:27 +00001544#ifdef LIBXML_VALID_ENABLED
Daniel Veillarda10efa82001-04-18 13:09:01 +00001545 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001546 * The element is already defined in this DTD.
Daniel Veillarda10efa82001-04-18 13:09:01 +00001547 */
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001548 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ELEM_REDEFINED,
1549 "Redefinition of element %s\n",
1550 name, NULL, NULL);
Daniel Veillard4432df22003-09-28 18:58:27 +00001551#endif /* LIBXML_VALID_ENABLED */
Daniel Veillarda10efa82001-04-18 13:09:01 +00001552 xmlFreeElement(ret);
1553 if (uqname != NULL)
1554 xmlFree(uqname);
1555 return(NULL);
1556 }
William M. Brack4e52f2f2003-09-14 18:07:39 +00001557 /*
1558 * For new element, may have attributes from earlier
1559 * definition in internal subset
1560 */
1561 ret->attributes = oldAttributes;
Daniel Veillarda10efa82001-04-18 13:09:01 +00001562 }
1563
1564 /*
1565 * Finish to fill the structure.
Owen Taylor3473f882001-02-23 17:55:21 +00001566 */
1567 ret->etype = type;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001568 /*
1569 * Avoid a stupid copy when called by the parser
1570 * and flag it by setting a special parent value
1571 * so the parser doesn't unallocate it.
1572 */
Daniel Veillardc394f732005-01-26 00:04:52 +00001573 if ((ctxt != NULL) &&
1574 ((ctxt->finishDtd == XML_CTXT_FINISH_DTD_0) ||
1575 (ctxt->finishDtd == XML_CTXT_FINISH_DTD_1))) {
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001576 ret->content = content;
1577 if (content != NULL)
1578 content->parent = (xmlElementContentPtr) 1;
1579 } else {
1580 ret->content = xmlCopyDocElementContent(dtd->doc, content);
1581 }
Owen Taylor3473f882001-02-23 17:55:21 +00001582
1583 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001584 * Link it to the DTD
Owen Taylor3473f882001-02-23 17:55:21 +00001585 */
1586 ret->parent = dtd;
1587 ret->doc = dtd->doc;
1588 if (dtd->last == NULL) {
1589 dtd->children = dtd->last = (xmlNodePtr) ret;
1590 } else {
1591 dtd->last->next = (xmlNodePtr) ret;
1592 ret->prev = dtd->last;
1593 dtd->last = (xmlNodePtr) ret;
1594 }
1595 if (uqname != NULL)
1596 xmlFree(uqname);
1597 return(ret);
1598}
1599
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01001600static void
1601xmlFreeElementTableEntry(void *elem, const xmlChar *name ATTRIBUTE_UNUSED) {
1602 xmlFreeElement((xmlElementPtr) elem);
1603}
1604
Owen Taylor3473f882001-02-23 17:55:21 +00001605/**
1606 * xmlFreeElementTable:
1607 * @table: An element table
1608 *
1609 * Deallocate the memory used by an element hash table.
1610 */
1611void
1612xmlFreeElementTable(xmlElementTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01001613 xmlHashFree(table, xmlFreeElementTableEntry);
Owen Taylor3473f882001-02-23 17:55:21 +00001614}
1615
Daniel Veillard652327a2003-09-29 18:02:38 +00001616#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001617/**
1618 * xmlCopyElement:
1619 * @elem: An element
1620 *
1621 * Build a copy of an element.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001622 *
Owen Taylor3473f882001-02-23 17:55:21 +00001623 * Returns the new xmlElementPtr or NULL in case of error.
1624 */
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01001625static void *
1626xmlCopyElement(void *payload, const xmlChar *name ATTRIBUTE_UNUSED) {
1627 xmlElementPtr elem = (xmlElementPtr) payload;
Owen Taylor3473f882001-02-23 17:55:21 +00001628 xmlElementPtr cur;
1629
1630 cur = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
1631 if (cur == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001632 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00001633 return(NULL);
1634 }
1635 memset(cur, 0, sizeof(xmlElement));
1636 cur->type = XML_ELEMENT_DECL;
1637 cur->etype = elem->etype;
1638 if (elem->name != NULL)
1639 cur->name = xmlStrdup(elem->name);
1640 else
1641 cur->name = NULL;
1642 if (elem->prefix != NULL)
1643 cur->prefix = xmlStrdup(elem->prefix);
1644 else
1645 cur->prefix = NULL;
1646 cur->content = xmlCopyElementContent(elem->content);
1647 /* TODO : rebuild the attribute list on the copy */
1648 cur->attributes = NULL;
1649 return(cur);
1650}
1651
1652/**
1653 * xmlCopyElementTable:
1654 * @table: An element table
1655 *
1656 * Build a copy of an element table.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001657 *
Owen Taylor3473f882001-02-23 17:55:21 +00001658 * Returns the new xmlElementTablePtr or NULL in case of error.
1659 */
1660xmlElementTablePtr
1661xmlCopyElementTable(xmlElementTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01001662 return((xmlElementTablePtr) xmlHashCopy(table, xmlCopyElement));
Owen Taylor3473f882001-02-23 17:55:21 +00001663}
Daniel Veillard652327a2003-09-29 18:02:38 +00001664#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001665
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001666#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001667/**
1668 * xmlDumpElementDecl:
1669 * @buf: the XML buffer output
1670 * @elem: An element table
1671 *
1672 * This will dump the content of the element declaration as an XML
1673 * DTD definition
1674 */
1675void
1676xmlDumpElementDecl(xmlBufferPtr buf, xmlElementPtr elem) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00001677 if ((buf == NULL) || (elem == NULL))
1678 return;
Owen Taylor3473f882001-02-23 17:55:21 +00001679 switch (elem->etype) {
1680 case XML_ELEMENT_TYPE_EMPTY:
1681 xmlBufferWriteChar(buf, "<!ELEMENT ");
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001682 if (elem->prefix != NULL) {
1683 xmlBufferWriteCHAR(buf, elem->prefix);
1684 xmlBufferWriteChar(buf, ":");
1685 }
Owen Taylor3473f882001-02-23 17:55:21 +00001686 xmlBufferWriteCHAR(buf, elem->name);
1687 xmlBufferWriteChar(buf, " EMPTY>\n");
1688 break;
1689 case XML_ELEMENT_TYPE_ANY:
1690 xmlBufferWriteChar(buf, "<!ELEMENT ");
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001691 if (elem->prefix != NULL) {
1692 xmlBufferWriteCHAR(buf, elem->prefix);
1693 xmlBufferWriteChar(buf, ":");
1694 }
Owen Taylor3473f882001-02-23 17:55:21 +00001695 xmlBufferWriteCHAR(buf, elem->name);
1696 xmlBufferWriteChar(buf, " ANY>\n");
1697 break;
1698 case XML_ELEMENT_TYPE_MIXED:
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 case XML_ELEMENT_TYPE_ELEMENT:
1710 xmlBufferWriteChar(buf, "<!ELEMENT ");
Daniel Veillardbe480fb2001-11-08 23:36:42 +00001711 if (elem->prefix != NULL) {
1712 xmlBufferWriteCHAR(buf, elem->prefix);
1713 xmlBufferWriteChar(buf, ":");
1714 }
Owen Taylor3473f882001-02-23 17:55:21 +00001715 xmlBufferWriteCHAR(buf, elem->name);
1716 xmlBufferWriteChar(buf, " ");
1717 xmlDumpElementContent(buf, elem->content, 1);
1718 xmlBufferWriteChar(buf, ">\n");
1719 break;
1720 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001721 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001722 "Internal: ELEMENT struct corrupted invalid type\n",
1723 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001724 }
1725}
1726
1727/**
William M. Brack9e660592003-10-20 14:56:06 +00001728 * xmlDumpElementDeclScan:
1729 * @elem: An element table
1730 * @buf: the XML buffer output
1731 *
1732 * This routine is used by the hash scan function. It just reverses
1733 * the arguments.
1734 */
1735static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01001736xmlDumpElementDeclScan(void *elem, void *buf,
1737 const xmlChar *name ATTRIBUTE_UNUSED) {
1738 xmlDumpElementDecl((xmlBufferPtr) buf, (xmlElementPtr) elem);
William M. Brack9e660592003-10-20 14:56:06 +00001739}
1740
1741/**
Owen Taylor3473f882001-02-23 17:55:21 +00001742 * xmlDumpElementTable:
1743 * @buf: the XML buffer output
1744 * @table: An element table
1745 *
1746 * This will dump the content of the element table as an XML DTD definition
1747 */
1748void
1749xmlDumpElementTable(xmlBufferPtr buf, xmlElementTablePtr table) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00001750 if ((buf == NULL) || (table == NULL))
1751 return;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01001752 xmlHashScan(table, xmlDumpElementDeclScan, buf);
Owen Taylor3473f882001-02-23 17:55:21 +00001753}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001754#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001755
1756/**
1757 * xmlCreateEnumeration:
1758 * @name: the enumeration name or NULL
1759 *
1760 * create and initialize an enumeration attribute node.
1761 *
1762 * Returns the xmlEnumerationPtr just created or NULL in case
1763 * of error.
1764 */
1765xmlEnumerationPtr
Daniel Veillard2fdbd322003-08-18 12:15:38 +00001766xmlCreateEnumeration(const xmlChar *name) {
Owen Taylor3473f882001-02-23 17:55:21 +00001767 xmlEnumerationPtr ret;
1768
1769 ret = (xmlEnumerationPtr) xmlMalloc(sizeof(xmlEnumeration));
1770 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00001771 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00001772 return(NULL);
1773 }
1774 memset(ret, 0, sizeof(xmlEnumeration));
1775
1776 if (name != NULL)
1777 ret->name = xmlStrdup(name);
1778 return(ret);
1779}
1780
1781/**
1782 * xmlFreeEnumeration:
1783 * @cur: the tree to free.
1784 *
1785 * free an enumeration attribute node (recursive).
1786 */
1787void
1788xmlFreeEnumeration(xmlEnumerationPtr cur) {
1789 if (cur == NULL) return;
1790
1791 if (cur->next != NULL) xmlFreeEnumeration(cur->next);
1792
1793 if (cur->name != NULL) xmlFree((xmlChar *) cur->name);
Owen Taylor3473f882001-02-23 17:55:21 +00001794 xmlFree(cur);
1795}
1796
Daniel Veillard652327a2003-09-29 18:02:38 +00001797#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001798/**
1799 * xmlCopyEnumeration:
1800 * @cur: the tree to copy.
1801 *
1802 * Copy an enumeration attribute node (recursive).
1803 *
1804 * Returns the xmlEnumerationPtr just created or NULL in case
1805 * of error.
1806 */
1807xmlEnumerationPtr
1808xmlCopyEnumeration(xmlEnumerationPtr cur) {
1809 xmlEnumerationPtr ret;
1810
1811 if (cur == NULL) return(NULL);
1812 ret = xmlCreateEnumeration((xmlChar *) cur->name);
Gaurav Gupta658b86c2014-08-07 11:19:03 +08001813 if (ret == NULL) return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001814
1815 if (cur->next != NULL) ret->next = xmlCopyEnumeration(cur->next);
1816 else ret->next = NULL;
1817
1818 return(ret);
1819}
Daniel Veillard652327a2003-09-29 18:02:38 +00001820#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001821
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001822#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001823/**
1824 * xmlDumpEnumeration:
1825 * @buf: the XML buffer output
1826 * @enum: An enumeration
1827 *
1828 * This will dump the content of the enumeration
1829 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001830static void
Owen Taylor3473f882001-02-23 17:55:21 +00001831xmlDumpEnumeration(xmlBufferPtr buf, xmlEnumerationPtr cur) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00001832 if ((buf == NULL) || (cur == NULL))
1833 return;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001834
Owen Taylor3473f882001-02-23 17:55:21 +00001835 xmlBufferWriteCHAR(buf, cur->name);
1836 if (cur->next == NULL)
1837 xmlBufferWriteChar(buf, ")");
1838 else {
1839 xmlBufferWriteChar(buf, " | ");
1840 xmlDumpEnumeration(buf, cur->next);
1841 }
1842}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001843#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001844
Daniel Veillard4432df22003-09-28 18:58:27 +00001845#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001846/**
Owen Taylor3473f882001-02-23 17:55:21 +00001847 * xmlScanIDAttributeDecl:
1848 * @ctxt: the validation context
1849 * @elem: the element name
Daniel Veillarddbee0f12005-06-27 13:42:57 +00001850 * @err: whether to raise errors here
Owen Taylor3473f882001-02-23 17:55:21 +00001851 *
1852 * Verify that the element don't have too many ID attributes
1853 * declared.
1854 *
1855 * Returns the number of ID attributes found.
1856 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001857static int
Daniel Veillarddbee0f12005-06-27 13:42:57 +00001858xmlScanIDAttributeDecl(xmlValidCtxtPtr ctxt, xmlElementPtr elem, int err) {
Owen Taylor3473f882001-02-23 17:55:21 +00001859 xmlAttributePtr cur;
1860 int ret = 0;
1861
1862 if (elem == NULL) return(0);
1863 cur = elem->attributes;
1864 while (cur != NULL) {
1865 if (cur->atype == XML_ATTRIBUTE_ID) {
1866 ret ++;
Daniel Veillarddbee0f12005-06-27 13:42:57 +00001867 if ((ret > 1) && (err))
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001868 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_MULTIPLE_ID,
Daniel Veillarda10efa82001-04-18 13:09:01 +00001869 "Element %s has too many ID attributes defined : %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001870 elem->name, cur->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001871 }
1872 cur = cur->nexth;
1873 }
1874 return(ret);
1875}
Daniel Veillard4432df22003-09-28 18:58:27 +00001876#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00001877
1878/**
1879 * xmlFreeAttribute:
1880 * @elem: An attribute
1881 *
1882 * Deallocate the memory used by an attribute definition
1883 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00001884static void
Owen Taylor3473f882001-02-23 17:55:21 +00001885xmlFreeAttribute(xmlAttributePtr attr) {
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001886 xmlDictPtr dict;
1887
Owen Taylor3473f882001-02-23 17:55:21 +00001888 if (attr == NULL) return;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001889 if (attr->doc != NULL)
1890 dict = attr->doc->dict;
1891 else
1892 dict = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00001893 xmlUnlinkNode((xmlNodePtr) attr);
1894 if (attr->tree != NULL)
1895 xmlFreeEnumeration(attr->tree);
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001896 if (dict) {
1897 if ((attr->elem != NULL) && (!xmlDictOwns(dict, attr->elem)))
1898 xmlFree((xmlChar *) attr->elem);
1899 if ((attr->name != NULL) && (!xmlDictOwns(dict, attr->name)))
1900 xmlFree((xmlChar *) attr->name);
1901 if ((attr->prefix != NULL) && (!xmlDictOwns(dict, attr->prefix)))
1902 xmlFree((xmlChar *) attr->prefix);
1903 if ((attr->defaultValue != NULL) &&
1904 (!xmlDictOwns(dict, attr->defaultValue)))
1905 xmlFree((xmlChar *) attr->defaultValue);
1906 } else {
1907 if (attr->elem != NULL)
1908 xmlFree((xmlChar *) attr->elem);
1909 if (attr->name != NULL)
1910 xmlFree((xmlChar *) attr->name);
1911 if (attr->defaultValue != NULL)
1912 xmlFree((xmlChar *) attr->defaultValue);
1913 if (attr->prefix != NULL)
1914 xmlFree((xmlChar *) attr->prefix);
1915 }
Owen Taylor3473f882001-02-23 17:55:21 +00001916 xmlFree(attr);
1917}
1918
1919
1920/**
1921 * xmlAddAttributeDecl:
1922 * @ctxt: the validation context
1923 * @dtd: pointer to the DTD
1924 * @elem: the element name
1925 * @name: the attribute name
1926 * @ns: the attribute namespace prefix
1927 * @type: the attribute type
1928 * @def: the attribute default type
1929 * @defaultValue: the attribute default value
1930 * @tree: if it's an enumeration, the associated list
1931 *
1932 * Register a new attribute declaration
1933 * Note that @tree becomes the ownership of the DTD
1934 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00001935 * Returns NULL if not new, otherwise the attribute decl
Owen Taylor3473f882001-02-23 17:55:21 +00001936 */
1937xmlAttributePtr
William M. Brackedb65a72004-02-06 07:36:04 +00001938xmlAddAttributeDecl(xmlValidCtxtPtr ctxt,
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00001939 xmlDtdPtr dtd, const xmlChar *elem,
Owen Taylor3473f882001-02-23 17:55:21 +00001940 const xmlChar *name, const xmlChar *ns,
1941 xmlAttributeType type, xmlAttributeDefault def,
1942 const xmlChar *defaultValue, xmlEnumerationPtr tree) {
1943 xmlAttributePtr ret;
1944 xmlAttributeTablePtr table;
1945 xmlElementPtr elemDef;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001946 xmlDictPtr dict = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00001947
1948 if (dtd == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001949 xmlFreeEnumeration(tree);
1950 return(NULL);
1951 }
1952 if (name == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001953 xmlFreeEnumeration(tree);
1954 return(NULL);
1955 }
1956 if (elem == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00001957 xmlFreeEnumeration(tree);
1958 return(NULL);
1959 }
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00001960 if (dtd->doc != NULL)
1961 dict = dtd->doc->dict;
Daniel Veillardd85f4f42002-03-25 10:48:46 +00001962
Daniel Veillard4432df22003-09-28 18:58:27 +00001963#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00001964 /*
1965 * Check the type and possibly the default value.
1966 */
1967 switch (type) {
1968 case XML_ATTRIBUTE_CDATA:
1969 break;
1970 case XML_ATTRIBUTE_ID:
1971 break;
1972 case XML_ATTRIBUTE_IDREF:
1973 break;
1974 case XML_ATTRIBUTE_IDREFS:
1975 break;
1976 case XML_ATTRIBUTE_ENTITY:
1977 break;
1978 case XML_ATTRIBUTE_ENTITIES:
1979 break;
1980 case XML_ATTRIBUTE_NMTOKEN:
1981 break;
1982 case XML_ATTRIBUTE_NMTOKENS:
1983 break;
1984 case XML_ATTRIBUTE_ENUMERATION:
1985 break;
1986 case XML_ATTRIBUTE_NOTATION:
1987 break;
1988 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001989 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00001990 "Internal: ATTRIBUTE struct corrupted invalid type\n",
1991 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00001992 xmlFreeEnumeration(tree);
1993 return(NULL);
1994 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001995 if ((defaultValue != NULL) &&
Daniel Veillardae0765b2008-07-31 19:54:59 +00001996 (!xmlValidateAttributeValueInternal(dtd->doc, type, defaultValue))) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00001997 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_ATTRIBUTE_DEFAULT,
1998 "Attribute %s of %s: invalid default value\n",
1999 elem, name, defaultValue);
Owen Taylor3473f882001-02-23 17:55:21 +00002000 defaultValue = NULL;
Daniel Veillard42595322004-11-08 10:52:06 +00002001 if (ctxt != NULL)
2002 ctxt->valid = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00002003 }
Daniel Veillard4432df22003-09-28 18:58:27 +00002004#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002005
2006 /*
Daniel Veillardd85f4f42002-03-25 10:48:46 +00002007 * Check first that an attribute defined in the external subset wasn't
2008 * already defined in the internal subset
2009 */
2010 if ((dtd->doc != NULL) && (dtd->doc->extSubset == dtd) &&
2011 (dtd->doc->intSubset != NULL) &&
2012 (dtd->doc->intSubset->attributes != NULL)) {
2013 ret = xmlHashLookup3(dtd->doc->intSubset->attributes, name, ns, elem);
Daniel Veillardae0765b2008-07-31 19:54:59 +00002014 if (ret != NULL) {
2015 xmlFreeEnumeration(tree);
Daniel Veillardd85f4f42002-03-25 10:48:46 +00002016 return(NULL);
Daniel Veillardae0765b2008-07-31 19:54:59 +00002017 }
Daniel Veillardd85f4f42002-03-25 10:48:46 +00002018 }
2019
2020 /*
Owen Taylor3473f882001-02-23 17:55:21 +00002021 * Create the Attribute table if needed.
2022 */
2023 table = (xmlAttributeTablePtr) dtd->attributes;
2024 if (table == NULL) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00002025 table = xmlHashCreateDict(0, dict);
Owen Taylor3473f882001-02-23 17:55:21 +00002026 dtd->attributes = (void *) table;
2027 }
2028 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002029 xmlVErrMemory(ctxt,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002030 "xmlAddAttributeDecl: Table creation failed!\n");
Daniel Veillardae0765b2008-07-31 19:54:59 +00002031 xmlFreeEnumeration(tree);
Owen Taylor3473f882001-02-23 17:55:21 +00002032 return(NULL);
2033 }
2034
2035
2036 ret = (xmlAttributePtr) xmlMalloc(sizeof(xmlAttribute));
2037 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002038 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillardae0765b2008-07-31 19:54:59 +00002039 xmlFreeEnumeration(tree);
Owen Taylor3473f882001-02-23 17:55:21 +00002040 return(NULL);
2041 }
2042 memset(ret, 0, sizeof(xmlAttribute));
2043 ret->type = XML_ATTRIBUTE_DECL;
2044
2045 /*
2046 * fill the structure.
2047 */
2048 ret->atype = type;
William M. Brackf810de02005-07-06 22:48:41 +00002049 /*
2050 * doc must be set before possible error causes call
2051 * to xmlFreeAttribute (because it's used to check on
2052 * dict use)
2053 */
2054 ret->doc = dtd->doc;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00002055 if (dict) {
2056 ret->name = xmlDictLookup(dict, name, -1);
2057 ret->prefix = xmlDictLookup(dict, ns, -1);
2058 ret->elem = xmlDictLookup(dict, elem, -1);
2059 } else {
2060 ret->name = xmlStrdup(name);
2061 ret->prefix = xmlStrdup(ns);
2062 ret->elem = xmlStrdup(elem);
2063 }
Owen Taylor3473f882001-02-23 17:55:21 +00002064 ret->def = def;
2065 ret->tree = tree;
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00002066 if (defaultValue != NULL) {
2067 if (dict)
2068 ret->defaultValue = xmlDictLookup(dict, defaultValue, -1);
2069 else
2070 ret->defaultValue = xmlStrdup(defaultValue);
2071 }
Owen Taylor3473f882001-02-23 17:55:21 +00002072
2073 /*
2074 * Validity Check:
2075 * Search the DTD for previous declarations of the ATTLIST
2076 */
Daniel Veillardcee2b3a2005-01-25 00:22:52 +00002077 if (xmlHashAddEntry3(table, ret->name, ret->prefix, ret->elem, ret) < 0) {
Daniel Veillard4432df22003-09-28 18:58:27 +00002078#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002079 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002080 * The attribute is already defined in this DTD.
Owen Taylor3473f882001-02-23 17:55:21 +00002081 */
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002082 xmlErrValidWarning(ctxt, (xmlNodePtr) dtd, XML_DTD_ATTRIBUTE_REDEFINED,
Daniel Veillard58e44c92002-08-02 22:19:49 +00002083 "Attribute %s of element %s: already defined\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002084 name, elem, NULL);
Daniel Veillard4432df22003-09-28 18:58:27 +00002085#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002086 xmlFreeAttribute(ret);
2087 return(NULL);
2088 }
2089
2090 /*
2091 * Validity Check:
2092 * Multiple ID per element
2093 */
Daniel Veillarda10efa82001-04-18 13:09:01 +00002094 elemDef = xmlGetDtdElementDesc2(dtd, elem, 1);
Owen Taylor3473f882001-02-23 17:55:21 +00002095 if (elemDef != NULL) {
Daniel Veillard48da9102001-08-07 01:10:10 +00002096
Daniel Veillard4432df22003-09-28 18:58:27 +00002097#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002098 if ((type == XML_ATTRIBUTE_ID) &&
Daniel Veillarddbee0f12005-06-27 13:42:57 +00002099 (xmlScanIDAttributeDecl(NULL, elemDef, 1) != 0)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002100 xmlErrValidNode(ctxt, (xmlNodePtr) dtd, XML_DTD_MULTIPLE_ID,
Owen Taylor3473f882001-02-23 17:55:21 +00002101 "Element %s has too may ID attributes defined : %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00002102 elem, name, NULL);
Daniel Veillard42595322004-11-08 10:52:06 +00002103 if (ctxt != NULL)
2104 ctxt->valid = 0;
Daniel Veillardc7612992002-02-17 22:47:37 +00002105 }
Daniel Veillard4432df22003-09-28 18:58:27 +00002106#endif /* LIBXML_VALID_ENABLED */
Daniel Veillardc7612992002-02-17 22:47:37 +00002107
Daniel Veillard48da9102001-08-07 01:10:10 +00002108 /*
2109 * Insert namespace default def first they need to be
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002110 * processed first.
Daniel Veillard48da9102001-08-07 01:10:10 +00002111 */
2112 if ((xmlStrEqual(ret->name, BAD_CAST "xmlns")) ||
2113 ((ret->prefix != NULL &&
2114 (xmlStrEqual(ret->prefix, BAD_CAST "xmlns"))))) {
2115 ret->nexth = elemDef->attributes;
2116 elemDef->attributes = ret;
2117 } else {
2118 xmlAttributePtr tmp = elemDef->attributes;
2119
2120 while ((tmp != NULL) &&
2121 ((xmlStrEqual(tmp->name, BAD_CAST "xmlns")) ||
2122 ((ret->prefix != NULL &&
2123 (xmlStrEqual(ret->prefix, BAD_CAST "xmlns")))))) {
2124 if (tmp->nexth == NULL)
2125 break;
2126 tmp = tmp->nexth;
2127 }
2128 if (tmp != NULL) {
2129 ret->nexth = tmp->nexth;
2130 tmp->nexth = ret;
2131 } else {
2132 ret->nexth = elemDef->attributes;
2133 elemDef->attributes = ret;
2134 }
2135 }
Owen Taylor3473f882001-02-23 17:55:21 +00002136 }
2137
2138 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002139 * Link it to the DTD
Owen Taylor3473f882001-02-23 17:55:21 +00002140 */
2141 ret->parent = dtd;
Owen Taylor3473f882001-02-23 17:55:21 +00002142 if (dtd->last == NULL) {
2143 dtd->children = dtd->last = (xmlNodePtr) ret;
2144 } else {
2145 dtd->last->next = (xmlNodePtr) ret;
2146 ret->prev = dtd->last;
2147 dtd->last = (xmlNodePtr) ret;
2148 }
2149 return(ret);
2150}
2151
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002152static void
2153xmlFreeAttributeTableEntry(void *attr, const xmlChar *name ATTRIBUTE_UNUSED) {
2154 xmlFreeAttribute((xmlAttributePtr) attr);
2155}
2156
Owen Taylor3473f882001-02-23 17:55:21 +00002157/**
2158 * xmlFreeAttributeTable:
2159 * @table: An attribute table
2160 *
2161 * Deallocate the memory used by an entities hash table.
2162 */
2163void
2164xmlFreeAttributeTable(xmlAttributeTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002165 xmlHashFree(table, xmlFreeAttributeTableEntry);
Owen Taylor3473f882001-02-23 17:55:21 +00002166}
2167
Daniel Veillard652327a2003-09-29 18:02:38 +00002168#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002169/**
2170 * xmlCopyAttribute:
2171 * @attr: An attribute
2172 *
2173 * Build a copy of an attribute.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002174 *
Owen Taylor3473f882001-02-23 17:55:21 +00002175 * Returns the new xmlAttributePtr or NULL in case of error.
2176 */
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002177static void *
2178xmlCopyAttribute(void *payload, const xmlChar *name ATTRIBUTE_UNUSED) {
2179 xmlAttributePtr attr = (xmlAttributePtr) payload;
Owen Taylor3473f882001-02-23 17:55:21 +00002180 xmlAttributePtr cur;
2181
2182 cur = (xmlAttributePtr) xmlMalloc(sizeof(xmlAttribute));
2183 if (cur == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002184 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002185 return(NULL);
2186 }
2187 memset(cur, 0, sizeof(xmlAttribute));
Daniel Veillard36065812002-01-24 15:02:46 +00002188 cur->type = XML_ATTRIBUTE_DECL;
Owen Taylor3473f882001-02-23 17:55:21 +00002189 cur->atype = attr->atype;
2190 cur->def = attr->def;
2191 cur->tree = xmlCopyEnumeration(attr->tree);
2192 if (attr->elem != NULL)
2193 cur->elem = xmlStrdup(attr->elem);
2194 if (attr->name != NULL)
2195 cur->name = xmlStrdup(attr->name);
Daniel Veillard36065812002-01-24 15:02:46 +00002196 if (attr->prefix != NULL)
2197 cur->prefix = xmlStrdup(attr->prefix);
Owen Taylor3473f882001-02-23 17:55:21 +00002198 if (attr->defaultValue != NULL)
2199 cur->defaultValue = xmlStrdup(attr->defaultValue);
2200 return(cur);
2201}
2202
2203/**
2204 * xmlCopyAttributeTable:
2205 * @table: An attribute table
2206 *
2207 * Build a copy of an attribute table.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002208 *
Owen Taylor3473f882001-02-23 17:55:21 +00002209 * Returns the new xmlAttributeTablePtr or NULL in case of error.
2210 */
2211xmlAttributeTablePtr
2212xmlCopyAttributeTable(xmlAttributeTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002213 return((xmlAttributeTablePtr) xmlHashCopy(table, xmlCopyAttribute));
Owen Taylor3473f882001-02-23 17:55:21 +00002214}
Daniel Veillard652327a2003-09-29 18:02:38 +00002215#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002216
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002217#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002218/**
2219 * xmlDumpAttributeDecl:
2220 * @buf: the XML buffer output
2221 * @attr: An attribute declaration
2222 *
2223 * This will dump the content of the attribute declaration as an XML
2224 * DTD definition
2225 */
2226void
2227xmlDumpAttributeDecl(xmlBufferPtr buf, xmlAttributePtr attr) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002228 if ((buf == NULL) || (attr == NULL))
2229 return;
Owen Taylor3473f882001-02-23 17:55:21 +00002230 xmlBufferWriteChar(buf, "<!ATTLIST ");
2231 xmlBufferWriteCHAR(buf, attr->elem);
2232 xmlBufferWriteChar(buf, " ");
2233 if (attr->prefix != NULL) {
2234 xmlBufferWriteCHAR(buf, attr->prefix);
2235 xmlBufferWriteChar(buf, ":");
2236 }
2237 xmlBufferWriteCHAR(buf, attr->name);
2238 switch (attr->atype) {
2239 case XML_ATTRIBUTE_CDATA:
2240 xmlBufferWriteChar(buf, " CDATA");
2241 break;
2242 case XML_ATTRIBUTE_ID:
2243 xmlBufferWriteChar(buf, " ID");
2244 break;
2245 case XML_ATTRIBUTE_IDREF:
2246 xmlBufferWriteChar(buf, " IDREF");
2247 break;
2248 case XML_ATTRIBUTE_IDREFS:
2249 xmlBufferWriteChar(buf, " IDREFS");
2250 break;
2251 case XML_ATTRIBUTE_ENTITY:
2252 xmlBufferWriteChar(buf, " ENTITY");
2253 break;
2254 case XML_ATTRIBUTE_ENTITIES:
2255 xmlBufferWriteChar(buf, " ENTITIES");
2256 break;
2257 case XML_ATTRIBUTE_NMTOKEN:
2258 xmlBufferWriteChar(buf, " NMTOKEN");
2259 break;
2260 case XML_ATTRIBUTE_NMTOKENS:
2261 xmlBufferWriteChar(buf, " NMTOKENS");
2262 break;
2263 case XML_ATTRIBUTE_ENUMERATION:
2264 xmlBufferWriteChar(buf, " (");
2265 xmlDumpEnumeration(buf, attr->tree);
2266 break;
2267 case XML_ATTRIBUTE_NOTATION:
2268 xmlBufferWriteChar(buf, " NOTATION (");
2269 xmlDumpEnumeration(buf, attr->tree);
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 type\n",
2274 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00002275 }
2276 switch (attr->def) {
2277 case XML_ATTRIBUTE_NONE:
2278 break;
2279 case XML_ATTRIBUTE_REQUIRED:
2280 xmlBufferWriteChar(buf, " #REQUIRED");
2281 break;
2282 case XML_ATTRIBUTE_IMPLIED:
2283 xmlBufferWriteChar(buf, " #IMPLIED");
2284 break;
2285 case XML_ATTRIBUTE_FIXED:
2286 xmlBufferWriteChar(buf, " #FIXED");
2287 break;
2288 default:
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002289 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002290 "Internal: ATTRIBUTE struct corrupted invalid def\n",
2291 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00002292 }
2293 if (attr->defaultValue != NULL) {
2294 xmlBufferWriteChar(buf, " ");
2295 xmlBufferWriteQuotedString(buf, attr->defaultValue);
2296 }
2297 xmlBufferWriteChar(buf, ">\n");
2298}
2299
2300/**
William M. Brack9e660592003-10-20 14:56:06 +00002301 * xmlDumpAttributeDeclScan:
2302 * @attr: An attribute declaration
2303 * @buf: the XML buffer output
2304 *
2305 * This is used with the hash scan function - just reverses arguments
2306 */
2307static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002308xmlDumpAttributeDeclScan(void *attr, void *buf,
2309 const xmlChar *name ATTRIBUTE_UNUSED) {
2310 xmlDumpAttributeDecl((xmlBufferPtr) buf, (xmlAttributePtr) attr);
William M. Brack9e660592003-10-20 14:56:06 +00002311}
2312
2313/**
Owen Taylor3473f882001-02-23 17:55:21 +00002314 * xmlDumpAttributeTable:
2315 * @buf: the XML buffer output
2316 * @table: An attribute table
2317 *
2318 * This will dump the content of the attribute table as an XML DTD definition
2319 */
2320void
2321xmlDumpAttributeTable(xmlBufferPtr buf, xmlAttributeTablePtr table) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002322 if ((buf == NULL) || (table == NULL))
2323 return;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002324 xmlHashScan(table, xmlDumpAttributeDeclScan, buf);
Owen Taylor3473f882001-02-23 17:55:21 +00002325}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002326#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002327
2328/************************************************************************
2329 * *
2330 * NOTATIONs *
2331 * *
2332 ************************************************************************/
2333/**
Owen Taylor3473f882001-02-23 17:55:21 +00002334 * xmlFreeNotation:
2335 * @not: A notation
2336 *
2337 * Deallocate the memory used by an notation definition
2338 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002339static void
Owen Taylor3473f882001-02-23 17:55:21 +00002340xmlFreeNotation(xmlNotationPtr nota) {
2341 if (nota == NULL) return;
2342 if (nota->name != NULL)
2343 xmlFree((xmlChar *) nota->name);
2344 if (nota->PublicID != NULL)
2345 xmlFree((xmlChar *) nota->PublicID);
2346 if (nota->SystemID != NULL)
2347 xmlFree((xmlChar *) nota->SystemID);
Owen Taylor3473f882001-02-23 17:55:21 +00002348 xmlFree(nota);
2349}
2350
2351
2352/**
2353 * xmlAddNotationDecl:
2354 * @dtd: pointer to the DTD
2355 * @ctxt: the validation context
2356 * @name: the entity name
2357 * @PublicID: the public identifier or NULL
2358 * @SystemID: the system identifier or NULL
2359 *
2360 * Register a new notation declaration
2361 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002362 * Returns NULL if not, otherwise the entity
Owen Taylor3473f882001-02-23 17:55:21 +00002363 */
2364xmlNotationPtr
William M. Brackedb65a72004-02-06 07:36:04 +00002365xmlAddNotationDecl(xmlValidCtxtPtr ctxt, xmlDtdPtr dtd,
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002366 const xmlChar *name,
Owen Taylor3473f882001-02-23 17:55:21 +00002367 const xmlChar *PublicID, const xmlChar *SystemID) {
2368 xmlNotationPtr ret;
2369 xmlNotationTablePtr table;
2370
2371 if (dtd == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002372 return(NULL);
2373 }
2374 if (name == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002375 return(NULL);
2376 }
2377 if ((PublicID == NULL) && (SystemID == NULL)) {
Daniel Veillard7aea52d2002-02-17 23:07:47 +00002378 return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00002379 }
2380
2381 /*
2382 * Create the Notation table if needed.
2383 */
2384 table = (xmlNotationTablePtr) dtd->notations;
Daniel Veillard316a5c32005-01-23 22:56:39 +00002385 if (table == NULL) {
2386 xmlDictPtr dict = NULL;
2387 if (dtd->doc != NULL)
2388 dict = dtd->doc->dict;
2389
2390 dtd->notations = table = xmlHashCreateDict(0, dict);
2391 }
Owen Taylor3473f882001-02-23 17:55:21 +00002392 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002393 xmlVErrMemory(ctxt,
Owen Taylor3473f882001-02-23 17:55:21 +00002394 "xmlAddNotationDecl: Table creation failed!\n");
2395 return(NULL);
2396 }
2397
2398 ret = (xmlNotationPtr) xmlMalloc(sizeof(xmlNotation));
2399 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002400 xmlVErrMemory(ctxt, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002401 return(NULL);
2402 }
2403 memset(ret, 0, sizeof(xmlNotation));
2404
2405 /*
2406 * fill the structure.
2407 */
2408 ret->name = xmlStrdup(name);
2409 if (SystemID != NULL)
2410 ret->SystemID = xmlStrdup(SystemID);
2411 if (PublicID != NULL)
2412 ret->PublicID = xmlStrdup(PublicID);
2413
2414 /*
2415 * Validity Check:
2416 * Check the DTD for previous declarations of the ATTLIST
2417 */
2418 if (xmlHashAddEntry(table, name, ret)) {
Daniel Veillard4432df22003-09-28 18:58:27 +00002419#ifdef LIBXML_VALID_ENABLED
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002420 xmlErrValid(NULL, XML_DTD_NOTATION_REDEFINED,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002421 "xmlAddNotationDecl: %s already defined\n",
2422 (const char *) name);
Daniel Veillard4432df22003-09-28 18:58:27 +00002423#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002424 xmlFreeNotation(ret);
2425 return(NULL);
2426 }
2427 return(ret);
2428}
2429
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002430static void
2431xmlFreeNotationTableEntry(void *nota, const xmlChar *name ATTRIBUTE_UNUSED) {
2432 xmlFreeNotation((xmlNotationPtr) nota);
2433}
2434
Owen Taylor3473f882001-02-23 17:55:21 +00002435/**
2436 * xmlFreeNotationTable:
2437 * @table: An notation table
2438 *
2439 * Deallocate the memory used by an entities hash table.
2440 */
2441void
2442xmlFreeNotationTable(xmlNotationTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002443 xmlHashFree(table, xmlFreeNotationTableEntry);
Owen Taylor3473f882001-02-23 17:55:21 +00002444}
2445
Daniel Veillard652327a2003-09-29 18:02:38 +00002446#ifdef LIBXML_TREE_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002447/**
2448 * xmlCopyNotation:
2449 * @nota: A notation
2450 *
2451 * Build a copy of a notation.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002452 *
Owen Taylor3473f882001-02-23 17:55:21 +00002453 * Returns the new xmlNotationPtr or NULL in case of error.
2454 */
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002455static void *
2456xmlCopyNotation(void *payload, const xmlChar *name ATTRIBUTE_UNUSED) {
2457 xmlNotationPtr nota = (xmlNotationPtr) payload;
Owen Taylor3473f882001-02-23 17:55:21 +00002458 xmlNotationPtr cur;
2459
2460 cur = (xmlNotationPtr) xmlMalloc(sizeof(xmlNotation));
2461 if (cur == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002462 xmlVErrMemory(NULL, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002463 return(NULL);
2464 }
2465 if (nota->name != NULL)
2466 cur->name = xmlStrdup(nota->name);
2467 else
2468 cur->name = NULL;
2469 if (nota->PublicID != NULL)
2470 cur->PublicID = xmlStrdup(nota->PublicID);
2471 else
2472 cur->PublicID = NULL;
2473 if (nota->SystemID != NULL)
2474 cur->SystemID = xmlStrdup(nota->SystemID);
2475 else
2476 cur->SystemID = NULL;
2477 return(cur);
2478}
2479
2480/**
2481 * xmlCopyNotationTable:
2482 * @table: A notation table
2483 *
2484 * Build a copy of a notation table.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002485 *
Owen Taylor3473f882001-02-23 17:55:21 +00002486 * Returns the new xmlNotationTablePtr or NULL in case of error.
2487 */
2488xmlNotationTablePtr
2489xmlCopyNotationTable(xmlNotationTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002490 return((xmlNotationTablePtr) xmlHashCopy(table, xmlCopyNotation));
Owen Taylor3473f882001-02-23 17:55:21 +00002491}
Daniel Veillard652327a2003-09-29 18:02:38 +00002492#endif /* LIBXML_TREE_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002493
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002494#ifdef LIBXML_OUTPUT_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002495/**
2496 * xmlDumpNotationDecl:
2497 * @buf: the XML buffer output
2498 * @nota: A notation declaration
2499 *
2500 * This will dump the content the notation declaration as an XML DTD definition
2501 */
2502void
2503xmlDumpNotationDecl(xmlBufferPtr buf, xmlNotationPtr nota) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002504 if ((buf == NULL) || (nota == NULL))
2505 return;
Owen Taylor3473f882001-02-23 17:55:21 +00002506 xmlBufferWriteChar(buf, "<!NOTATION ");
2507 xmlBufferWriteCHAR(buf, nota->name);
2508 if (nota->PublicID != NULL) {
2509 xmlBufferWriteChar(buf, " PUBLIC ");
2510 xmlBufferWriteQuotedString(buf, nota->PublicID);
2511 if (nota->SystemID != NULL) {
2512 xmlBufferWriteChar(buf, " ");
Daniel Veillard41c4a752004-09-08 20:55:38 +00002513 xmlBufferWriteQuotedString(buf, nota->SystemID);
Owen Taylor3473f882001-02-23 17:55:21 +00002514 }
2515 } else {
2516 xmlBufferWriteChar(buf, " SYSTEM ");
Daniel Veillard41c4a752004-09-08 20:55:38 +00002517 xmlBufferWriteQuotedString(buf, nota->SystemID);
Owen Taylor3473f882001-02-23 17:55:21 +00002518 }
2519 xmlBufferWriteChar(buf, " >\n");
2520}
2521
2522/**
William M. Brack9e660592003-10-20 14:56:06 +00002523 * xmlDumpNotationDeclScan:
2524 * @nota: A notation declaration
2525 * @buf: the XML buffer output
2526 *
2527 * This is called with the hash scan function, and just reverses args
2528 */
2529static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002530xmlDumpNotationDeclScan(void *nota, void *buf,
2531 const xmlChar *name ATTRIBUTE_UNUSED) {
2532 xmlDumpNotationDecl((xmlBufferPtr) buf, (xmlNotationPtr) nota);
William M. Brack9e660592003-10-20 14:56:06 +00002533}
2534
2535/**
Owen Taylor3473f882001-02-23 17:55:21 +00002536 * xmlDumpNotationTable:
2537 * @buf: the XML buffer output
2538 * @table: A notation table
2539 *
2540 * This will dump the content of the notation table as an XML DTD definition
2541 */
2542void
2543xmlDumpNotationTable(xmlBufferPtr buf, xmlNotationTablePtr table) {
Daniel Veillardce682bc2004-11-05 17:22:25 +00002544 if ((buf == NULL) || (table == NULL))
2545 return;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002546 xmlHashScan(table, xmlDumpNotationDeclScan, buf);
Owen Taylor3473f882001-02-23 17:55:21 +00002547}
Daniel Veillarda9cce9c2003-09-29 13:20:24 +00002548#endif /* LIBXML_OUTPUT_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002549
2550/************************************************************************
2551 * *
2552 * IDs *
2553 * *
2554 ************************************************************************/
2555/**
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002556 * DICT_FREE:
2557 * @str: a string
2558 *
Jan Pokornýbb654fe2016-04-13 16:56:07 +02002559 * Free a string if it is not owned by the "dict" dictionary in the
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002560 * current scope
2561 */
2562#define DICT_FREE(str) \
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002563 if ((str) && ((!dict) || \
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002564 (xmlDictOwns(dict, (const xmlChar *)(str)) == 0))) \
2565 xmlFree((char *)(str));
2566
2567/**
Owen Taylor3473f882001-02-23 17:55:21 +00002568 * xmlFreeID:
2569 * @not: A id
2570 *
2571 * Deallocate the memory used by an id definition
2572 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00002573static void
Owen Taylor3473f882001-02-23 17:55:21 +00002574xmlFreeID(xmlIDPtr id) {
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002575 xmlDictPtr dict = NULL;
2576
Owen Taylor3473f882001-02-23 17:55:21 +00002577 if (id == NULL) return;
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002578
2579 if (id->doc != NULL)
2580 dict = id->doc->dict;
2581
Owen Taylor3473f882001-02-23 17:55:21 +00002582 if (id->value != NULL)
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002583 DICT_FREE(id->value)
Daniel Veillardea7751d2002-12-20 00:16:24 +00002584 if (id->name != NULL)
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002585 DICT_FREE(id->name)
Owen Taylor3473f882001-02-23 17:55:21 +00002586 xmlFree(id);
2587}
2588
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002589
Owen Taylor3473f882001-02-23 17:55:21 +00002590/**
2591 * xmlAddID:
2592 * @ctxt: the validation context
2593 * @doc: pointer to the document
2594 * @value: the value name
2595 * @attr: the attribute holding the ID
2596 *
2597 * Register a new id declaration
2598 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002599 * Returns NULL if not, otherwise the new xmlIDPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002600 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002601xmlIDPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002602xmlAddID(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value,
2603 xmlAttrPtr attr) {
2604 xmlIDPtr ret;
2605 xmlIDTablePtr table;
2606
2607 if (doc == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002608 return(NULL);
2609 }
2610 if (value == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002611 return(NULL);
2612 }
2613 if (attr == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002614 return(NULL);
2615 }
2616
2617 /*
2618 * Create the ID table if needed.
2619 */
2620 table = (xmlIDTablePtr) doc->ids;
Daniel Veillard316a5c32005-01-23 22:56:39 +00002621 if (table == NULL) {
2622 doc->ids = table = xmlHashCreateDict(0, doc->dict);
2623 }
Owen Taylor3473f882001-02-23 17:55:21 +00002624 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002625 xmlVErrMemory(ctxt,
Owen Taylor3473f882001-02-23 17:55:21 +00002626 "xmlAddID: Table creation failed!\n");
2627 return(NULL);
2628 }
2629
2630 ret = (xmlIDPtr) xmlMalloc(sizeof(xmlID));
2631 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002632 xmlVErrMemory(ctxt, "malloc failed");
Owen Taylor3473f882001-02-23 17:55:21 +00002633 return(NULL);
2634 }
2635
2636 /*
2637 * fill the structure.
2638 */
2639 ret->value = xmlStrdup(value);
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002640 ret->doc = doc;
Daniel Veillardea7751d2002-12-20 00:16:24 +00002641 if ((ctxt != NULL) && (ctxt->vstateNr != 0)) {
2642 /*
2643 * Operating in streaming mode, attr is gonna disapear
2644 */
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002645 if (doc->dict != NULL)
2646 ret->name = xmlDictLookup(doc->dict, attr->name, -1);
2647 else
2648 ret->name = xmlStrdup(attr->name);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002649 ret->attr = NULL;
2650 } else {
2651 ret->attr = attr;
2652 ret->name = NULL;
2653 }
2654 ret->lineno = xmlGetLineNo(attr->parent);
Owen Taylor3473f882001-02-23 17:55:21 +00002655
2656 if (xmlHashAddEntry(table, value, ret) < 0) {
Daniel Veillard4432df22003-09-28 18:58:27 +00002657#ifdef LIBXML_VALID_ENABLED
Owen Taylor3473f882001-02-23 17:55:21 +00002658 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002659 * The id is already defined in this DTD.
Owen Taylor3473f882001-02-23 17:55:21 +00002660 */
Daniel Veillardef709ce2015-09-10 19:41:41 +08002661 if (ctxt != NULL) {
2662 xmlErrValidNode(ctxt, attr->parent, XML_DTD_ID_REDEFINED,
2663 "ID %s already defined\n", value, NULL, NULL);
2664 }
Daniel Veillard4432df22003-09-28 18:58:27 +00002665#endif /* LIBXML_VALID_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00002666 xmlFreeID(ret);
2667 return(NULL);
2668 }
Daniel Veillard249d7bb2003-03-19 21:02:29 +00002669 if (attr != NULL)
2670 attr->atype = XML_ATTRIBUTE_ID;
Owen Taylor3473f882001-02-23 17:55:21 +00002671 return(ret);
2672}
2673
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002674static void
2675xmlFreeIDTableEntry(void *id, const xmlChar *name ATTRIBUTE_UNUSED) {
2676 xmlFreeID((xmlIDPtr) id);
2677}
2678
Owen Taylor3473f882001-02-23 17:55:21 +00002679/**
2680 * xmlFreeIDTable:
2681 * @table: An id table
2682 *
2683 * Deallocate the memory used by an ID hash table.
2684 */
2685void
2686xmlFreeIDTable(xmlIDTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002687 xmlHashFree(table, xmlFreeIDTableEntry);
Owen Taylor3473f882001-02-23 17:55:21 +00002688}
2689
2690/**
2691 * xmlIsID:
2692 * @doc: the document
2693 * @elem: the element carrying the attribute
2694 * @attr: the attribute
2695 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002696 * Determine whether an attribute is of type ID. In case we have DTD(s)
Daniel Veillard9dc1cf12002-10-08 08:26:11 +00002697 * then this is done if DTD loading has been requested. In the case
2698 * of HTML documents parsed with the HTML parser, then ID detection is
2699 * done systematically.
Owen Taylor3473f882001-02-23 17:55:21 +00002700 *
2701 * Returns 0 or 1 depending on the lookup result
2702 */
2703int
2704xmlIsID(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) {
Daniel Veillard54f9a4f2005-09-03 13:28:24 +00002705 if ((attr == NULL) || (attr->name == NULL)) return(0);
2706 if ((attr->ns != NULL) && (attr->ns->prefix != NULL) &&
2707 (!strcmp((char *) attr->name, "id")) &&
2708 (!strcmp((char *) attr->ns->prefix, "xml")))
2709 return(1);
Owen Taylor3473f882001-02-23 17:55:21 +00002710 if (doc == NULL) return(0);
Daniel Veillardf3c06692009-10-16 16:47:58 +02002711 if ((doc->intSubset == NULL) && (doc->extSubset == NULL) &&
2712 (doc->type != XML_HTML_DOCUMENT_NODE)) {
Owen Taylor3473f882001-02-23 17:55:21 +00002713 return(0);
2714 } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
Rob Richards04bffc02006-03-03 16:44:37 +00002715 if ((xmlStrEqual(BAD_CAST "id", attr->name)) ||
2716 ((xmlStrEqual(BAD_CAST "name", attr->name)) &&
Daniel Veillard38431c32007-06-12 16:20:09 +00002717 ((elem == NULL) || (xmlStrEqual(elem->name, BAD_CAST "a")))))
Owen Taylor3473f882001-02-23 17:55:21 +00002718 return(1);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002719 return(0);
Daniel Veillard379a3b72005-08-12 10:18:14 +00002720 } else if (elem == NULL) {
2721 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00002722 } else {
Daniel Veillard465a0002005-08-22 12:07:04 +00002723 xmlAttributePtr attrDecl = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00002724
Daniel Veillard379a3b72005-08-12 10:18:14 +00002725 xmlChar felem[50], fattr[50];
2726 xmlChar *fullelemname, *fullattrname;
2727
2728 fullelemname = (elem->ns != NULL && elem->ns->prefix != NULL) ?
2729 xmlBuildQName(elem->name, elem->ns->prefix, felem, 50) :
2730 (xmlChar *)elem->name;
2731
2732 fullattrname = (attr->ns != NULL && attr->ns->prefix != NULL) ?
2733 xmlBuildQName(attr->name, attr->ns->prefix, fattr, 50) :
2734 (xmlChar *)attr->name;
2735
2736 if (fullelemname != NULL && fullattrname != NULL) {
2737 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullelemname,
2738 fullattrname);
Daniel Veillard37f961d2002-07-06 17:53:56 +00002739 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillard379a3b72005-08-12 10:18:14 +00002740 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullelemname,
2741 fullattrname);
Daniel Veillard37f961d2002-07-06 17:53:56 +00002742 }
Owen Taylor3473f882001-02-23 17:55:21 +00002743
Daniel Veillard379a3b72005-08-12 10:18:14 +00002744 if ((fullattrname != fattr) && (fullattrname != attr->name))
2745 xmlFree(fullattrname);
2746 if ((fullelemname != felem) && (fullelemname != elem->name))
2747 xmlFree(fullelemname);
2748
Owen Taylor3473f882001-02-23 17:55:21 +00002749 if ((attrDecl != NULL) && (attrDecl->atype == XML_ATTRIBUTE_ID))
2750 return(1);
2751 }
2752 return(0);
2753}
2754
2755/**
Daniel Veillard01c13b52002-12-10 15:19:08 +00002756 * xmlRemoveID:
Owen Taylor3473f882001-02-23 17:55:21 +00002757 * @doc: the document
2758 * @attr: the attribute
2759 *
2760 * Remove the given attribute from the ID table maintained internally.
2761 *
2762 * Returns -1 if the lookup failed and 0 otherwise
2763 */
2764int
2765xmlRemoveID(xmlDocPtr doc, xmlAttrPtr attr) {
Owen Taylor3473f882001-02-23 17:55:21 +00002766 xmlIDTablePtr table;
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002767 xmlIDPtr id;
Owen Taylor3473f882001-02-23 17:55:21 +00002768 xmlChar *ID;
2769
2770 if (doc == NULL) return(-1);
2771 if (attr == NULL) return(-1);
Denis Pauk01461792013-08-06 09:55:55 +03002772
Owen Taylor3473f882001-02-23 17:55:21 +00002773 table = (xmlIDTablePtr) doc->ids;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002774 if (table == NULL)
Owen Taylor3473f882001-02-23 17:55:21 +00002775 return(-1);
2776
Owen Taylor3473f882001-02-23 17:55:21 +00002777 ID = xmlNodeListGetString(doc, attr->children, 1);
2778 if (ID == NULL)
Denis Pauk01461792013-08-06 09:55:55 +03002779 return(-1);
2780
Daniel Veillard8d7b5c72003-11-15 18:24:36 +00002781 id = xmlHashLookup(table, ID);
2782 if (id == NULL || id->attr != attr) {
Denis Pauk01461792013-08-06 09:55:55 +03002783 xmlFree(ID);
2784 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00002785 }
Denis Pauk01461792013-08-06 09:55:55 +03002786
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002787 xmlHashRemoveEntry(table, ID, xmlFreeIDTableEntry);
Owen Taylor3473f882001-02-23 17:55:21 +00002788 xmlFree(ID);
Denis Pauk01461792013-08-06 09:55:55 +03002789 attr->atype = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00002790 return(0);
2791}
2792
2793/**
2794 * xmlGetID:
2795 * @doc: pointer to the document
2796 * @ID: the ID value
2797 *
2798 * Search the attribute declaring the given ID
2799 *
2800 * Returns NULL if not found, otherwise the xmlAttrPtr defining the ID
2801 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002802xmlAttrPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002803xmlGetID(xmlDocPtr doc, const xmlChar *ID) {
2804 xmlIDTablePtr table;
2805 xmlIDPtr id;
2806
2807 if (doc == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002808 return(NULL);
2809 }
2810
2811 if (ID == NULL) {
Owen Taylor3473f882001-02-23 17:55:21 +00002812 return(NULL);
2813 }
2814
2815 table = (xmlIDTablePtr) doc->ids;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002816 if (table == NULL)
Owen Taylor3473f882001-02-23 17:55:21 +00002817 return(NULL);
2818
2819 id = xmlHashLookup(table, ID);
2820 if (id == NULL)
2821 return(NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002822 if (id->attr == NULL) {
2823 /*
2824 * We are operating on a stream, return a well known reference
2825 * since the attribute node doesn't exist anymore
2826 */
2827 return((xmlAttrPtr) doc);
2828 }
Owen Taylor3473f882001-02-23 17:55:21 +00002829 return(id->attr);
2830}
2831
2832/************************************************************************
2833 * *
2834 * Refs *
2835 * *
2836 ************************************************************************/
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002837typedef struct xmlRemoveMemo_t
Owen Taylor3473f882001-02-23 17:55:21 +00002838{
2839 xmlListPtr l;
2840 xmlAttrPtr ap;
Daniel Veillard8730c562001-02-26 10:49:57 +00002841} xmlRemoveMemo;
2842
2843typedef xmlRemoveMemo *xmlRemoveMemoPtr;
2844
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002845typedef struct xmlValidateMemo_t
Daniel Veillard8730c562001-02-26 10:49:57 +00002846{
2847 xmlValidCtxtPtr ctxt;
2848 const xmlChar *name;
2849} xmlValidateMemo;
2850
2851typedef xmlValidateMemo *xmlValidateMemoPtr;
Owen Taylor3473f882001-02-23 17:55:21 +00002852
2853/**
Owen Taylor3473f882001-02-23 17:55:21 +00002854 * xmlFreeRef:
2855 * @lk: A list link
2856 *
2857 * Deallocate the memory used by a ref definition
2858 */
2859static void
2860xmlFreeRef(xmlLinkPtr lk) {
Daniel Veillard37721922001-05-04 15:21:12 +00002861 xmlRefPtr ref = (xmlRefPtr)xmlLinkGetData(lk);
2862 if (ref == NULL) return;
2863 if (ref->value != NULL)
2864 xmlFree((xmlChar *)ref->value);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002865 if (ref->name != NULL)
2866 xmlFree((xmlChar *)ref->name);
Daniel Veillard37721922001-05-04 15:21:12 +00002867 xmlFree(ref);
Owen Taylor3473f882001-02-23 17:55:21 +00002868}
2869
2870/**
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002871 * xmlFreeRefTableEntry:
Owen Taylor3473f882001-02-23 17:55:21 +00002872 * @list_ref: A list of references.
2873 *
2874 * Deallocate the memory used by a list of references
2875 */
2876static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01002877xmlFreeRefTableEntry(void *payload, const xmlChar *name ATTRIBUTE_UNUSED) {
2878 xmlListPtr list_ref = (xmlListPtr) payload;
Daniel Veillard37721922001-05-04 15:21:12 +00002879 if (list_ref == NULL) return;
2880 xmlListDelete(list_ref);
Owen Taylor3473f882001-02-23 17:55:21 +00002881}
2882
2883/**
2884 * xmlWalkRemoveRef:
2885 * @data: Contents of current link
2886 * @user: Value supplied by the user
2887 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002888 * Returns 0 to abort the walk or 1 to continue
Owen Taylor3473f882001-02-23 17:55:21 +00002889 */
2890static int
Nick Wellnhofer4dd6d7a2017-11-09 17:28:00 +01002891xmlWalkRemoveRef(const void *data, void *user)
Owen Taylor3473f882001-02-23 17:55:21 +00002892{
Daniel Veillard37721922001-05-04 15:21:12 +00002893 xmlAttrPtr attr0 = ((xmlRefPtr)data)->attr;
2894 xmlAttrPtr attr1 = ((xmlRemoveMemoPtr)user)->ap;
2895 xmlListPtr ref_list = ((xmlRemoveMemoPtr)user)->l;
Owen Taylor3473f882001-02-23 17:55:21 +00002896
Daniel Veillard37721922001-05-04 15:21:12 +00002897 if (attr0 == attr1) { /* Matched: remove and terminate walk */
2898 xmlListRemoveFirst(ref_list, (void *)data);
2899 return 0;
2900 }
2901 return 1;
Owen Taylor3473f882001-02-23 17:55:21 +00002902}
2903
2904/**
Daniel Veillard770075b2004-02-25 10:44:30 +00002905 * xmlDummyCompare
2906 * @data0: Value supplied by the user
2907 * @data1: Value supplied by the user
2908 *
2909 * Do nothing, return 0. Used to create unordered lists.
2910 */
2911static int
Daniel Veillardf54cd532004-02-25 11:52:31 +00002912xmlDummyCompare(const void *data0 ATTRIBUTE_UNUSED,
2913 const void *data1 ATTRIBUTE_UNUSED)
Daniel Veillard770075b2004-02-25 10:44:30 +00002914{
2915 return (0);
2916}
2917
2918/**
Owen Taylor3473f882001-02-23 17:55:21 +00002919 * xmlAddRef:
2920 * @ctxt: the validation context
2921 * @doc: pointer to the document
2922 * @value: the value name
2923 * @attr: the attribute holding the Ref
2924 *
2925 * Register a new ref declaration
2926 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00002927 * Returns NULL if not, otherwise the new xmlRefPtr
Owen Taylor3473f882001-02-23 17:55:21 +00002928 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002929xmlRefPtr
William M. Brackedb65a72004-02-06 07:36:04 +00002930xmlAddRef(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value,
Owen Taylor3473f882001-02-23 17:55:21 +00002931 xmlAttrPtr attr) {
Daniel Veillard37721922001-05-04 15:21:12 +00002932 xmlRefPtr ret;
2933 xmlRefTablePtr table;
2934 xmlListPtr ref_list;
Owen Taylor3473f882001-02-23 17:55:21 +00002935
Daniel Veillard37721922001-05-04 15:21:12 +00002936 if (doc == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00002937 return(NULL);
2938 }
2939 if (value == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00002940 return(NULL);
2941 }
2942 if (attr == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00002943 return(NULL);
2944 }
Owen Taylor3473f882001-02-23 17:55:21 +00002945
Daniel Veillard37721922001-05-04 15:21:12 +00002946 /*
Owen Taylor3473f882001-02-23 17:55:21 +00002947 * Create the Ref table if needed.
2948 */
Daniel Veillard37721922001-05-04 15:21:12 +00002949 table = (xmlRefTablePtr) doc->refs;
Daniel Veillard316a5c32005-01-23 22:56:39 +00002950 if (table == NULL) {
2951 doc->refs = table = xmlHashCreateDict(0, doc->dict);
2952 }
Daniel Veillard37721922001-05-04 15:21:12 +00002953 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002954 xmlVErrMemory(ctxt,
Daniel Veillard37721922001-05-04 15:21:12 +00002955 "xmlAddRef: Table creation failed!\n");
2956 return(NULL);
2957 }
Owen Taylor3473f882001-02-23 17:55:21 +00002958
Daniel Veillard37721922001-05-04 15:21:12 +00002959 ret = (xmlRefPtr) xmlMalloc(sizeof(xmlRef));
2960 if (ret == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00002961 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard37721922001-05-04 15:21:12 +00002962 return(NULL);
2963 }
Owen Taylor3473f882001-02-23 17:55:21 +00002964
Daniel Veillard37721922001-05-04 15:21:12 +00002965 /*
Owen Taylor3473f882001-02-23 17:55:21 +00002966 * fill the structure.
2967 */
Daniel Veillard37721922001-05-04 15:21:12 +00002968 ret->value = xmlStrdup(value);
Daniel Veillardea7751d2002-12-20 00:16:24 +00002969 if ((ctxt != NULL) && (ctxt->vstateNr != 0)) {
2970 /*
2971 * Operating in streaming mode, attr is gonna disapear
2972 */
2973 ret->name = xmlStrdup(attr->name);
2974 ret->attr = NULL;
2975 } else {
2976 ret->name = NULL;
2977 ret->attr = attr;
2978 }
2979 ret->lineno = xmlGetLineNo(attr->parent);
Owen Taylor3473f882001-02-23 17:55:21 +00002980
Daniel Veillard37721922001-05-04 15:21:12 +00002981 /* To add a reference :-
2982 * References are maintained as a list of references,
2983 * Lookup the entry, if no entry create new nodelist
2984 * Add the owning node to the NodeList
2985 * Return the ref
2986 */
Owen Taylor3473f882001-02-23 17:55:21 +00002987
Daniel Veillard37721922001-05-04 15:21:12 +00002988 if (NULL == (ref_list = xmlHashLookup(table, value))) {
Daniel Veillard770075b2004-02-25 10:44:30 +00002989 if (NULL == (ref_list = xmlListCreate(xmlFreeRef, xmlDummyCompare))) {
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002990 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2991 "xmlAddRef: Reference list creation failed!\n",
2992 NULL);
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00002993 goto failed;
Daniel Veillard37721922001-05-04 15:21:12 +00002994 }
2995 if (xmlHashAddEntry(table, value, ref_list) < 0) {
2996 xmlListDelete(ref_list);
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00002997 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
2998 "xmlAddRef: Reference list insertion failed!\n",
2999 NULL);
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00003000 goto failed;
Daniel Veillard37721922001-05-04 15:21:12 +00003001 }
3002 }
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00003003 if (xmlListAppend(ref_list, ret) != 0) {
3004 xmlErrValid(NULL, XML_ERR_INTERNAL_ERROR,
3005 "xmlAddRef: Reference list insertion failed!\n",
3006 NULL);
3007 goto failed;
3008 }
Daniel Veillard37721922001-05-04 15:21:12 +00003009 return(ret);
Daniel Veillardf6cf57a2007-05-09 23:53:30 +00003010failed:
3011 if (ret != NULL) {
3012 if (ret->value != NULL)
3013 xmlFree((char *)ret->value);
3014 if (ret->name != NULL)
3015 xmlFree((char *)ret->name);
3016 xmlFree(ret);
3017 }
3018 return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003019}
3020
3021/**
3022 * xmlFreeRefTable:
3023 * @table: An ref table
3024 *
3025 * Deallocate the memory used by an Ref hash table.
3026 */
3027void
3028xmlFreeRefTable(xmlRefTablePtr table) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01003029 xmlHashFree(table, xmlFreeRefTableEntry);
Owen Taylor3473f882001-02-23 17:55:21 +00003030}
3031
3032/**
3033 * xmlIsRef:
3034 * @doc: the document
3035 * @elem: the element carrying the attribute
3036 * @attr: the attribute
3037 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003038 * Determine whether an attribute is of type Ref. In case we have DTD(s)
Owen Taylor3473f882001-02-23 17:55:21 +00003039 * then this is simple, otherwise we use an heuristic: name Ref (upper
3040 * or lowercase).
3041 *
3042 * Returns 0 or 1 depending on the lookup result
3043 */
3044int
3045xmlIsRef(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) {
Daniel Veillardce244ad2004-11-05 10:03:46 +00003046 if (attr == NULL)
3047 return(0);
3048 if (doc == NULL) {
3049 doc = attr->doc;
3050 if (doc == NULL) return(0);
3051 }
3052
Daniel Veillard37721922001-05-04 15:21:12 +00003053 if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
3054 return(0);
3055 } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
3056 /* TODO @@@ */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003057 return(0);
Daniel Veillard37721922001-05-04 15:21:12 +00003058 } else {
3059 xmlAttributePtr attrDecl;
Owen Taylor3473f882001-02-23 17:55:21 +00003060
Daniel Veillardce244ad2004-11-05 10:03:46 +00003061 if (elem == NULL) return(0);
Daniel Veillard37721922001-05-04 15:21:12 +00003062 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, attr->name);
3063 if ((attrDecl == NULL) && (doc->extSubset != NULL))
3064 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
3065 elem->name, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00003066
Daniel Veillard37721922001-05-04 15:21:12 +00003067 if ((attrDecl != NULL) &&
3068 (attrDecl->atype == XML_ATTRIBUTE_IDREF ||
3069 attrDecl->atype == XML_ATTRIBUTE_IDREFS))
3070 return(1);
3071 }
3072 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003073}
3074
3075/**
Daniel Veillard01c13b52002-12-10 15:19:08 +00003076 * xmlRemoveRef:
Owen Taylor3473f882001-02-23 17:55:21 +00003077 * @doc: the document
3078 * @attr: the attribute
3079 *
3080 * Remove the given attribute from the Ref table maintained internally.
3081 *
3082 * Returns -1 if the lookup failed and 0 otherwise
3083 */
3084int
3085xmlRemoveRef(xmlDocPtr doc, xmlAttrPtr attr) {
Daniel Veillard37721922001-05-04 15:21:12 +00003086 xmlListPtr ref_list;
3087 xmlRefTablePtr table;
3088 xmlChar *ID;
3089 xmlRemoveMemo target;
Owen Taylor3473f882001-02-23 17:55:21 +00003090
Daniel Veillard37721922001-05-04 15:21:12 +00003091 if (doc == NULL) return(-1);
3092 if (attr == NULL) return(-1);
Denis Pauk01461792013-08-06 09:55:55 +03003093
Daniel Veillard37721922001-05-04 15:21:12 +00003094 table = (xmlRefTablePtr) doc->refs;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003095 if (table == NULL)
Daniel Veillard37721922001-05-04 15:21:12 +00003096 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003097
Daniel Veillard37721922001-05-04 15:21:12 +00003098 ID = xmlNodeListGetString(doc, attr->children, 1);
3099 if (ID == NULL)
3100 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003101
Denis Pauk01461792013-08-06 09:55:55 +03003102 ref_list = xmlHashLookup(table, ID);
Daniel Veillard37721922001-05-04 15:21:12 +00003103 if(ref_list == NULL) {
3104 xmlFree(ID);
3105 return (-1);
3106 }
Denis Pauk01461792013-08-06 09:55:55 +03003107
Daniel Veillard37721922001-05-04 15:21:12 +00003108 /* At this point, ref_list refers to a list of references which
3109 * have the same key as the supplied attr. Our list of references
3110 * is ordered by reference address and we don't have that information
3111 * here to use when removing. We'll have to walk the list and
3112 * check for a matching attribute, when we find one stop the walk
3113 * and remove the entry.
3114 * The list is ordered by reference, so that means we don't have the
3115 * key. Passing the list and the reference to the walker means we
3116 * will have enough data to be able to remove the entry.
3117 */
3118 target.l = ref_list;
3119 target.ap = attr;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003120
Daniel Veillard37721922001-05-04 15:21:12 +00003121 /* Remove the supplied attr from our list */
3122 xmlListWalk(ref_list, xmlWalkRemoveRef, &target);
Owen Taylor3473f882001-02-23 17:55:21 +00003123
Daniel Veillard37721922001-05-04 15:21:12 +00003124 /*If the list is empty then remove the list entry in the hash */
3125 if (xmlListEmpty(ref_list))
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01003126 xmlHashUpdateEntry(table, ID, NULL, xmlFreeRefTableEntry);
Daniel Veillard37721922001-05-04 15:21:12 +00003127 xmlFree(ID);
3128 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003129}
3130
3131/**
3132 * xmlGetRefs:
3133 * @doc: pointer to the document
3134 * @ID: the ID value
3135 *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003136 * Find the set of references for the supplied ID.
Owen Taylor3473f882001-02-23 17:55:21 +00003137 *
3138 * Returns NULL if not found, otherwise node set for the ID.
3139 */
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003140xmlListPtr
Owen Taylor3473f882001-02-23 17:55:21 +00003141xmlGetRefs(xmlDocPtr doc, const xmlChar *ID) {
Daniel Veillard37721922001-05-04 15:21:12 +00003142 xmlRefTablePtr table;
Owen Taylor3473f882001-02-23 17:55:21 +00003143
Daniel Veillard37721922001-05-04 15:21:12 +00003144 if (doc == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00003145 return(NULL);
3146 }
Owen Taylor3473f882001-02-23 17:55:21 +00003147
Daniel Veillard37721922001-05-04 15:21:12 +00003148 if (ID == NULL) {
Daniel Veillard37721922001-05-04 15:21:12 +00003149 return(NULL);
3150 }
Owen Taylor3473f882001-02-23 17:55:21 +00003151
Daniel Veillard37721922001-05-04 15:21:12 +00003152 table = (xmlRefTablePtr) doc->refs;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003153 if (table == NULL)
Daniel Veillard37721922001-05-04 15:21:12 +00003154 return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003155
Daniel Veillard37721922001-05-04 15:21:12 +00003156 return (xmlHashLookup(table, ID));
Owen Taylor3473f882001-02-23 17:55:21 +00003157}
3158
3159/************************************************************************
3160 * *
3161 * Routines for validity checking *
3162 * *
3163 ************************************************************************/
3164
3165/**
3166 * xmlGetDtdElementDesc:
3167 * @dtd: a pointer to the DtD to search
3168 * @name: the element name
3169 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003170 * Search the DTD for the description of this element
Owen Taylor3473f882001-02-23 17:55:21 +00003171 *
3172 * returns the xmlElementPtr if found or NULL
3173 */
3174
3175xmlElementPtr
3176xmlGetDtdElementDesc(xmlDtdPtr dtd, const xmlChar *name) {
3177 xmlElementTablePtr table;
3178 xmlElementPtr cur;
3179 xmlChar *uqname = NULL, *prefix = NULL;
3180
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00003181 if ((dtd == NULL) || (name == NULL)) return(NULL);
Daniel Veillarda10efa82001-04-18 13:09:01 +00003182 if (dtd->elements == NULL)
3183 return(NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003184 table = (xmlElementTablePtr) dtd->elements;
3185
3186 uqname = xmlSplitQName2(name, &prefix);
Daniel Veillarda10efa82001-04-18 13:09:01 +00003187 if (uqname != NULL)
3188 name = uqname;
3189 cur = xmlHashLookup2(table, name, prefix);
3190 if (prefix != NULL) xmlFree(prefix);
3191 if (uqname != NULL) xmlFree(uqname);
3192 return(cur);
3193}
3194/**
3195 * xmlGetDtdElementDesc2:
3196 * @dtd: a pointer to the DtD to search
3197 * @name: the element name
3198 * @create: create an empty description if not found
3199 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003200 * Search the DTD for the description of this element
Daniel Veillarda10efa82001-04-18 13:09:01 +00003201 *
3202 * returns the xmlElementPtr if found or NULL
3203 */
3204
Daniel Veillard86fd5a72001-12-13 14:55:21 +00003205static xmlElementPtr
Daniel Veillarda10efa82001-04-18 13:09:01 +00003206xmlGetDtdElementDesc2(xmlDtdPtr dtd, const xmlChar *name, int create) {
3207 xmlElementTablePtr table;
3208 xmlElementPtr cur;
3209 xmlChar *uqname = NULL, *prefix = NULL;
3210
3211 if (dtd == NULL) return(NULL);
3212 if (dtd->elements == NULL) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00003213 xmlDictPtr dict = NULL;
3214
3215 if (dtd->doc != NULL)
3216 dict = dtd->doc->dict;
3217
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003218 if (!create)
Daniel Veillarda10efa82001-04-18 13:09:01 +00003219 return(NULL);
3220 /*
3221 * Create the Element table if needed.
3222 */
3223 table = (xmlElementTablePtr) dtd->elements;
3224 if (table == NULL) {
Daniel Veillard316a5c32005-01-23 22:56:39 +00003225 table = xmlHashCreateDict(0, dict);
Daniel Veillarda10efa82001-04-18 13:09:01 +00003226 dtd->elements = (void *) table;
3227 }
3228 if (table == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00003229 xmlVErrMemory(NULL, "element table allocation failed");
Daniel Veillarda10efa82001-04-18 13:09:01 +00003230 return(NULL);
3231 }
3232 }
3233 table = (xmlElementTablePtr) dtd->elements;
3234
3235 uqname = xmlSplitQName2(name, &prefix);
3236 if (uqname != NULL)
3237 name = uqname;
3238 cur = xmlHashLookup2(table, name, prefix);
3239 if ((cur == NULL) && (create)) {
3240 cur = (xmlElementPtr) xmlMalloc(sizeof(xmlElement));
3241 if (cur == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00003242 xmlVErrMemory(NULL, "malloc failed");
Daniel Veillarda10efa82001-04-18 13:09:01 +00003243 return(NULL);
3244 }
3245 memset(cur, 0, sizeof(xmlElement));
3246 cur->type = XML_ELEMENT_DECL;
3247
3248 /*
3249 * fill the structure.
3250 */
3251 cur->name = xmlStrdup(name);
3252 cur->prefix = xmlStrdup(prefix);
3253 cur->etype = XML_ELEMENT_TYPE_UNDEFINED;
3254
3255 xmlHashAddEntry2(table, name, prefix, cur);
3256 }
3257 if (prefix != NULL) xmlFree(prefix);
3258 if (uqname != NULL) xmlFree(uqname);
Owen Taylor3473f882001-02-23 17:55:21 +00003259 return(cur);
3260}
3261
3262/**
3263 * xmlGetDtdQElementDesc:
3264 * @dtd: a pointer to the DtD to search
3265 * @name: the element name
3266 * @prefix: the element namespace prefix
3267 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003268 * Search the DTD for the description of this element
Owen Taylor3473f882001-02-23 17:55:21 +00003269 *
3270 * returns the xmlElementPtr if found or NULL
3271 */
3272
Daniel Veillard48da9102001-08-07 01:10:10 +00003273xmlElementPtr
Owen Taylor3473f882001-02-23 17:55:21 +00003274xmlGetDtdQElementDesc(xmlDtdPtr dtd, const xmlChar *name,
3275 const xmlChar *prefix) {
3276 xmlElementTablePtr table;
3277
3278 if (dtd == NULL) return(NULL);
3279 if (dtd->elements == NULL) return(NULL);
3280 table = (xmlElementTablePtr) dtd->elements;
3281
3282 return(xmlHashLookup2(table, name, prefix));
3283}
3284
3285/**
3286 * xmlGetDtdAttrDesc:
3287 * @dtd: a pointer to the DtD to search
3288 * @elem: the element name
3289 * @name: the attribute name
3290 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003291 * Search the DTD for the description of this attribute on
Owen Taylor3473f882001-02-23 17:55:21 +00003292 * this element.
3293 *
3294 * returns the xmlAttributePtr if found or NULL
3295 */
3296
3297xmlAttributePtr
3298xmlGetDtdAttrDesc(xmlDtdPtr dtd, const xmlChar *elem, const xmlChar *name) {
3299 xmlAttributeTablePtr table;
3300 xmlAttributePtr cur;
3301 xmlChar *uqname = NULL, *prefix = NULL;
3302
3303 if (dtd == NULL) return(NULL);
3304 if (dtd->attributes == NULL) return(NULL);
3305
3306 table = (xmlAttributeTablePtr) dtd->attributes;
3307 if (table == NULL)
3308 return(NULL);
3309
3310 uqname = xmlSplitQName2(name, &prefix);
3311
3312 if (uqname != NULL) {
3313 cur = xmlHashLookup3(table, uqname, prefix, elem);
3314 if (prefix != NULL) xmlFree(prefix);
3315 if (uqname != NULL) xmlFree(uqname);
3316 } else
3317 cur = xmlHashLookup3(table, name, NULL, elem);
3318 return(cur);
3319}
3320
3321/**
3322 * xmlGetDtdQAttrDesc:
3323 * @dtd: a pointer to the DtD to search
3324 * @elem: the element name
3325 * @name: the attribute name
3326 * @prefix: the attribute namespace prefix
3327 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003328 * Search the DTD for the description of this qualified attribute on
Owen Taylor3473f882001-02-23 17:55:21 +00003329 * this element.
3330 *
3331 * returns the xmlAttributePtr if found or NULL
3332 */
3333
Daniel Veillard48da9102001-08-07 01:10:10 +00003334xmlAttributePtr
Owen Taylor3473f882001-02-23 17:55:21 +00003335xmlGetDtdQAttrDesc(xmlDtdPtr dtd, const xmlChar *elem, const xmlChar *name,
3336 const xmlChar *prefix) {
3337 xmlAttributeTablePtr table;
3338
3339 if (dtd == NULL) return(NULL);
3340 if (dtd->attributes == NULL) return(NULL);
3341 table = (xmlAttributeTablePtr) dtd->attributes;
3342
3343 return(xmlHashLookup3(table, name, prefix, elem));
3344}
3345
3346/**
3347 * xmlGetDtdNotationDesc:
3348 * @dtd: a pointer to the DtD to search
3349 * @name: the notation name
3350 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003351 * Search the DTD for the description of this notation
Owen Taylor3473f882001-02-23 17:55:21 +00003352 *
3353 * returns the xmlNotationPtr if found or NULL
3354 */
3355
3356xmlNotationPtr
3357xmlGetDtdNotationDesc(xmlDtdPtr dtd, const xmlChar *name) {
3358 xmlNotationTablePtr table;
3359
3360 if (dtd == NULL) return(NULL);
3361 if (dtd->notations == NULL) return(NULL);
3362 table = (xmlNotationTablePtr) dtd->notations;
3363
3364 return(xmlHashLookup(table, name));
3365}
3366
Daniel Veillardf54cd532004-02-25 11:52:31 +00003367#if defined(LIBXML_VALID_ENABLED) || defined(LIBXML_SCHEMAS_ENABLED)
Owen Taylor3473f882001-02-23 17:55:21 +00003368/**
3369 * xmlValidateNotationUse:
3370 * @ctxt: the validation context
3371 * @doc: the document
3372 * @notationName: the notation name to check
3373 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003374 * Validate that the given name match a notation declaration.
Owen Taylor3473f882001-02-23 17:55:21 +00003375 * - [ VC: Notation Declared ]
3376 *
3377 * returns 1 if valid or 0 otherwise
3378 */
3379
3380int
3381xmlValidateNotationUse(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3382 const xmlChar *notationName) {
3383 xmlNotationPtr notaDecl;
Daniel Veillardeab3ac92009-08-12 10:39:29 +02003384 if ((doc == NULL) || (doc->intSubset == NULL) ||
3385 (notationName == NULL)) return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003386
3387 notaDecl = xmlGetDtdNotationDesc(doc->intSubset, notationName);
3388 if ((notaDecl == NULL) && (doc->extSubset != NULL))
3389 notaDecl = xmlGetDtdNotationDesc(doc->extSubset, notationName);
3390
Daniel Veillarde637c4a2003-03-30 21:10:09 +00003391 if ((notaDecl == NULL) && (ctxt != NULL)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003392 xmlErrValidNode(ctxt, (xmlNodePtr) doc, XML_DTD_UNKNOWN_NOTATION,
3393 "NOTATION %s is not declared\n",
3394 notationName, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003395 return(0);
3396 }
3397 return(1);
3398}
Daniel Veillardf54cd532004-02-25 11:52:31 +00003399#endif /* LIBXML_VALID_ENABLED or LIBXML_SCHEMAS_ENABLED */
Owen Taylor3473f882001-02-23 17:55:21 +00003400
3401/**
Daniel Veillard01c13b52002-12-10 15:19:08 +00003402 * xmlIsMixedElement:
Owen Taylor3473f882001-02-23 17:55:21 +00003403 * @doc: the document
3404 * @name: the element name
3405 *
3406 * Search in the DtDs whether an element accept Mixed content (or ANY)
3407 * basically if it is supposed to accept text childs
3408 *
3409 * returns 0 if no, 1 if yes, and -1 if no element description is available
3410 */
3411
3412int
3413xmlIsMixedElement(xmlDocPtr doc, const xmlChar *name) {
3414 xmlElementPtr elemDecl;
3415
3416 if ((doc == NULL) || (doc->intSubset == NULL)) return(-1);
3417
3418 elemDecl = xmlGetDtdElementDesc(doc->intSubset, name);
3419 if ((elemDecl == NULL) && (doc->extSubset != NULL))
3420 elemDecl = xmlGetDtdElementDesc(doc->extSubset, name);
3421 if (elemDecl == NULL) return(-1);
3422 switch (elemDecl->etype) {
Daniel Veillarda10efa82001-04-18 13:09:01 +00003423 case XML_ELEMENT_TYPE_UNDEFINED:
3424 return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00003425 case XML_ELEMENT_TYPE_ELEMENT:
3426 return(0);
3427 case XML_ELEMENT_TYPE_EMPTY:
3428 /*
3429 * return 1 for EMPTY since we want VC error to pop up
3430 * on <empty> </empty> for example
3431 */
3432 case XML_ELEMENT_TYPE_ANY:
3433 case XML_ELEMENT_TYPE_MIXED:
3434 return(1);
3435 }
3436 return(1);
3437}
3438
Daniel Veillard4432df22003-09-28 18:58:27 +00003439#ifdef LIBXML_VALID_ENABLED
Daniel Veillardae0765b2008-07-31 19:54:59 +00003440
3441static int
3442xmlIsDocNameStartChar(xmlDocPtr doc, int c) {
3443 if ((doc == NULL) || (doc->properties & XML_DOC_OLD10) == 0) {
3444 /*
3445 * Use the new checks of production [4] [4a] amd [5] of the
3446 * Update 5 of XML-1.0
3447 */
3448 if (((c >= 'a') && (c <= 'z')) ||
3449 ((c >= 'A') && (c <= 'Z')) ||
3450 (c == '_') || (c == ':') ||
3451 ((c >= 0xC0) && (c <= 0xD6)) ||
3452 ((c >= 0xD8) && (c <= 0xF6)) ||
3453 ((c >= 0xF8) && (c <= 0x2FF)) ||
3454 ((c >= 0x370) && (c <= 0x37D)) ||
3455 ((c >= 0x37F) && (c <= 0x1FFF)) ||
3456 ((c >= 0x200C) && (c <= 0x200D)) ||
3457 ((c >= 0x2070) && (c <= 0x218F)) ||
3458 ((c >= 0x2C00) && (c <= 0x2FEF)) ||
3459 ((c >= 0x3001) && (c <= 0xD7FF)) ||
3460 ((c >= 0xF900) && (c <= 0xFDCF)) ||
3461 ((c >= 0xFDF0) && (c <= 0xFFFD)) ||
3462 ((c >= 0x10000) && (c <= 0xEFFFF)))
3463 return(1);
3464 } else {
3465 if (IS_LETTER(c) || (c == '_') || (c == ':'))
3466 return(1);
3467 }
3468 return(0);
3469}
3470
3471static int
3472xmlIsDocNameChar(xmlDocPtr doc, int c) {
3473 if ((doc == NULL) || (doc->properties & XML_DOC_OLD10) == 0) {
3474 /*
3475 * Use the new checks of production [4] [4a] amd [5] of the
3476 * Update 5 of XML-1.0
3477 */
3478 if (((c >= 'a') && (c <= 'z')) ||
3479 ((c >= 'A') && (c <= 'Z')) ||
3480 ((c >= '0') && (c <= '9')) || /* !start */
3481 (c == '_') || (c == ':') ||
3482 (c == '-') || (c == '.') || (c == 0xB7) || /* !start */
3483 ((c >= 0xC0) && (c <= 0xD6)) ||
3484 ((c >= 0xD8) && (c <= 0xF6)) ||
3485 ((c >= 0xF8) && (c <= 0x2FF)) ||
3486 ((c >= 0x300) && (c <= 0x36F)) || /* !start */
3487 ((c >= 0x370) && (c <= 0x37D)) ||
3488 ((c >= 0x37F) && (c <= 0x1FFF)) ||
3489 ((c >= 0x200C) && (c <= 0x200D)) ||
3490 ((c >= 0x203F) && (c <= 0x2040)) || /* !start */
3491 ((c >= 0x2070) && (c <= 0x218F)) ||
3492 ((c >= 0x2C00) && (c <= 0x2FEF)) ||
3493 ((c >= 0x3001) && (c <= 0xD7FF)) ||
3494 ((c >= 0xF900) && (c <= 0xFDCF)) ||
3495 ((c >= 0xFDF0) && (c <= 0xFFFD)) ||
3496 ((c >= 0x10000) && (c <= 0xEFFFF)))
3497 return(1);
3498 } else {
3499 if ((IS_LETTER(c)) || (IS_DIGIT(c)) ||
3500 (c == '.') || (c == '-') ||
3501 (c == '_') || (c == ':') ||
3502 (IS_COMBINING(c)) ||
3503 (IS_EXTENDER(c)))
3504 return(1);
3505 }
3506 return(0);
3507}
3508
3509/**
3510 * xmlValidateNameValue:
3511 * @doc: pointer to the document or NULL
3512 * @value: an Name value
3513 *
3514 * Validate that the given value match Name production
3515 *
3516 * returns 1 if valid or 0 otherwise
3517 */
3518
3519static int
3520xmlValidateNameValueInternal(xmlDocPtr doc, const xmlChar *value) {
3521 const xmlChar *cur;
3522 int val, len;
3523
3524 if (value == NULL) return(0);
3525 cur = value;
3526 val = xmlStringCurrentChar(NULL, cur, &len);
3527 cur += len;
3528 if (!xmlIsDocNameStartChar(doc, val))
3529 return(0);
3530
3531 val = xmlStringCurrentChar(NULL, cur, &len);
3532 cur += len;
3533 while (xmlIsDocNameChar(doc, val)) {
3534 val = xmlStringCurrentChar(NULL, cur, &len);
3535 cur += len;
3536 }
3537
3538 if (val != 0) return(0);
3539
3540 return(1);
3541}
3542
Owen Taylor3473f882001-02-23 17:55:21 +00003543/**
3544 * xmlValidateNameValue:
3545 * @value: an Name value
3546 *
3547 * Validate that the given value match Name production
3548 *
3549 * returns 1 if valid or 0 otherwise
3550 */
3551
Daniel Veillard9b731d72002-04-14 12:56:08 +00003552int
Owen Taylor3473f882001-02-23 17:55:21 +00003553xmlValidateNameValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003554 return(xmlValidateNameValueInternal(NULL, value));
3555}
3556
3557/**
3558 * xmlValidateNamesValueInternal:
3559 * @doc: pointer to the document or NULL
3560 * @value: an Names value
3561 *
3562 * Validate that the given value match Names production
3563 *
3564 * returns 1 if valid or 0 otherwise
3565 */
3566
3567static int
3568xmlValidateNamesValueInternal(xmlDocPtr doc, const xmlChar *value) {
Owen Taylor3473f882001-02-23 17:55:21 +00003569 const xmlChar *cur;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003570 int val, len;
Owen Taylor3473f882001-02-23 17:55:21 +00003571
3572 if (value == NULL) return(0);
3573 cur = value;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003574 val = xmlStringCurrentChar(NULL, cur, &len);
3575 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003576
3577 if (!xmlIsDocNameStartChar(doc, val))
Owen Taylor3473f882001-02-23 17:55:21 +00003578 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003579
Daniel Veillardd8224e02002-01-13 15:43:22 +00003580 val = xmlStringCurrentChar(NULL, cur, &len);
3581 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003582 while (xmlIsDocNameChar(doc, val)) {
Daniel Veillardd8224e02002-01-13 15:43:22 +00003583 val = xmlStringCurrentChar(NULL, cur, &len);
3584 cur += len;
3585 }
Owen Taylor3473f882001-02-23 17:55:21 +00003586
Daniel Veillardae0765b2008-07-31 19:54:59 +00003587 /* Should not test IS_BLANK(val) here -- see erratum E20*/
3588 while (val == 0x20) {
3589 while (val == 0x20) {
3590 val = xmlStringCurrentChar(NULL, cur, &len);
3591 cur += len;
3592 }
3593
3594 if (!xmlIsDocNameStartChar(doc, val))
3595 return(0);
3596
3597 val = xmlStringCurrentChar(NULL, cur, &len);
3598 cur += len;
3599
3600 while (xmlIsDocNameChar(doc, val)) {
3601 val = xmlStringCurrentChar(NULL, cur, &len);
3602 cur += len;
3603 }
3604 }
3605
Daniel Veillardd8224e02002-01-13 15:43:22 +00003606 if (val != 0) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003607
3608 return(1);
3609}
3610
3611/**
3612 * xmlValidateNamesValue:
3613 * @value: an Names value
3614 *
3615 * Validate that the given value match Names production
3616 *
3617 * returns 1 if valid or 0 otherwise
3618 */
3619
Daniel Veillard9b731d72002-04-14 12:56:08 +00003620int
Owen Taylor3473f882001-02-23 17:55:21 +00003621xmlValidateNamesValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003622 return(xmlValidateNamesValueInternal(NULL, value));
3623}
3624
3625/**
3626 * xmlValidateNmtokenValueInternal:
3627 * @doc: pointer to the document or NULL
3628 * @value: an Nmtoken value
3629 *
3630 * Validate that the given value match Nmtoken production
3631 *
3632 * [ VC: Name Token ]
3633 *
3634 * returns 1 if valid or 0 otherwise
3635 */
3636
3637static int
3638xmlValidateNmtokenValueInternal(xmlDocPtr doc, const xmlChar *value) {
Owen Taylor3473f882001-02-23 17:55:21 +00003639 const xmlChar *cur;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003640 int val, len;
Owen Taylor3473f882001-02-23 17:55:21 +00003641
3642 if (value == NULL) return(0);
3643 cur = value;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003644 val = xmlStringCurrentChar(NULL, cur, &len);
3645 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003646
3647 if (!xmlIsDocNameChar(doc, val))
Owen Taylor3473f882001-02-23 17:55:21 +00003648 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003649
Daniel Veillardd8224e02002-01-13 15:43:22 +00003650 val = xmlStringCurrentChar(NULL, cur, &len);
3651 cur += len;
Daniel Veillardae0765b2008-07-31 19:54:59 +00003652 while (xmlIsDocNameChar(doc, val)) {
Daniel Veillardd8224e02002-01-13 15:43:22 +00003653 val = xmlStringCurrentChar(NULL, cur, &len);
3654 cur += len;
Owen Taylor3473f882001-02-23 17:55:21 +00003655 }
3656
Daniel Veillardd8224e02002-01-13 15:43:22 +00003657 if (val != 0) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003658
3659 return(1);
3660}
3661
3662/**
3663 * xmlValidateNmtokenValue:
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003664 * @value: an Nmtoken value
Owen Taylor3473f882001-02-23 17:55:21 +00003665 *
3666 * Validate that the given value match Nmtoken production
3667 *
3668 * [ VC: Name Token ]
Daniel Veillardae0765b2008-07-31 19:54:59 +00003669 *
Owen Taylor3473f882001-02-23 17:55:21 +00003670 * returns 1 if valid or 0 otherwise
3671 */
3672
Daniel Veillard9b731d72002-04-14 12:56:08 +00003673int
Owen Taylor3473f882001-02-23 17:55:21 +00003674xmlValidateNmtokenValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003675 return(xmlValidateNmtokenValueInternal(NULL, value));
3676}
3677
3678/**
3679 * xmlValidateNmtokensValueInternal:
3680 * @doc: pointer to the document or NULL
3681 * @value: an Nmtokens value
3682 *
3683 * Validate that the given value match Nmtokens production
3684 *
3685 * [ VC: Name Token ]
3686 *
3687 * returns 1 if valid or 0 otherwise
3688 */
3689
3690static int
3691xmlValidateNmtokensValueInternal(xmlDocPtr doc, const xmlChar *value) {
Owen Taylor3473f882001-02-23 17:55:21 +00003692 const xmlChar *cur;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003693 int val, len;
Owen Taylor3473f882001-02-23 17:55:21 +00003694
3695 if (value == NULL) return(0);
3696 cur = value;
Daniel Veillardd8224e02002-01-13 15:43:22 +00003697 val = xmlStringCurrentChar(NULL, cur, &len);
3698 cur += len;
Owen Taylor3473f882001-02-23 17:55:21 +00003699
Daniel Veillardae0765b2008-07-31 19:54:59 +00003700 while (IS_BLANK(val)) {
Daniel Veillardd8224e02002-01-13 15:43:22 +00003701 val = xmlStringCurrentChar(NULL, cur, &len);
3702 cur += len;
3703 }
Owen Taylor3473f882001-02-23 17:55:21 +00003704
Daniel Veillardae0765b2008-07-31 19:54:59 +00003705 if (!xmlIsDocNameChar(doc, val))
3706 return(0);
3707
3708 while (xmlIsDocNameChar(doc, val)) {
3709 val = xmlStringCurrentChar(NULL, cur, &len);
3710 cur += len;
3711 }
3712
3713 /* Should not test IS_BLANK(val) here -- see erratum E20*/
3714 while (val == 0x20) {
3715 while (val == 0x20) {
3716 val = xmlStringCurrentChar(NULL, cur, &len);
3717 cur += len;
3718 }
3719 if (val == 0) return(1);
3720
3721 if (!xmlIsDocNameChar(doc, val))
3722 return(0);
3723
3724 val = xmlStringCurrentChar(NULL, cur, &len);
3725 cur += len;
3726
3727 while (xmlIsDocNameChar(doc, val)) {
3728 val = xmlStringCurrentChar(NULL, cur, &len);
3729 cur += len;
3730 }
3731 }
3732
Daniel Veillardd8224e02002-01-13 15:43:22 +00003733 if (val != 0) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00003734
3735 return(1);
3736}
3737
3738/**
3739 * xmlValidateNmtokensValue:
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003740 * @value: an Nmtokens value
Owen Taylor3473f882001-02-23 17:55:21 +00003741 *
3742 * Validate that the given value match Nmtokens production
3743 *
3744 * [ VC: Name Token ]
Daniel Veillardae0765b2008-07-31 19:54:59 +00003745 *
Owen Taylor3473f882001-02-23 17:55:21 +00003746 * returns 1 if valid or 0 otherwise
3747 */
3748
Daniel Veillard9b731d72002-04-14 12:56:08 +00003749int
Owen Taylor3473f882001-02-23 17:55:21 +00003750xmlValidateNmtokensValue(const xmlChar *value) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00003751 return(xmlValidateNmtokensValueInternal(NULL, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003752}
3753
3754/**
3755 * xmlValidateNotationDecl:
3756 * @ctxt: the validation context
3757 * @doc: a document instance
3758 * @nota: a notation definition
3759 *
3760 * Try to validate a single notation definition
3761 * basically it does the following checks as described by the
3762 * XML-1.0 recommendation:
Daniel Veillardcbaf3992001-12-31 16:16:02 +00003763 * - it seems that no validity constraint exists on notation declarations
Owen Taylor3473f882001-02-23 17:55:21 +00003764 * But this function get called anyway ...
3765 *
3766 * returns 1 if valid or 0 otherwise
3767 */
3768
3769int
Daniel Veillardc86a4fa2001-03-26 16:28:29 +00003770xmlValidateNotationDecl(xmlValidCtxtPtr ctxt ATTRIBUTE_UNUSED, xmlDocPtr doc ATTRIBUTE_UNUSED,
3771 xmlNotationPtr nota ATTRIBUTE_UNUSED) {
Owen Taylor3473f882001-02-23 17:55:21 +00003772 int ret = 1;
3773
3774 return(ret);
3775}
3776
3777/**
Daniel Veillardbe2bd6a2008-11-27 15:26:28 +00003778 * xmlValidateAttributeValueInternal:
3779 * @doc: the document
Owen Taylor3473f882001-02-23 17:55:21 +00003780 * @type: an attribute type
3781 * @value: an attribute value
3782 *
3783 * Validate that the given attribute value match the proper production
3784 *
Owen Taylor3473f882001-02-23 17:55:21 +00003785 * returns 1 if valid or 0 otherwise
3786 */
3787
Daniel Veillardae0765b2008-07-31 19:54:59 +00003788static int
3789xmlValidateAttributeValueInternal(xmlDocPtr doc, xmlAttributeType type,
3790 const xmlChar *value) {
Owen Taylor3473f882001-02-23 17:55:21 +00003791 switch (type) {
3792 case XML_ATTRIBUTE_ENTITIES:
3793 case XML_ATTRIBUTE_IDREFS:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003794 return(xmlValidateNamesValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003795 case XML_ATTRIBUTE_ENTITY:
3796 case XML_ATTRIBUTE_IDREF:
3797 case XML_ATTRIBUTE_ID:
3798 case XML_ATTRIBUTE_NOTATION:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003799 return(xmlValidateNameValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003800 case XML_ATTRIBUTE_NMTOKENS:
3801 case XML_ATTRIBUTE_ENUMERATION:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003802 return(xmlValidateNmtokensValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003803 case XML_ATTRIBUTE_NMTOKEN:
Daniel Veillardae0765b2008-07-31 19:54:59 +00003804 return(xmlValidateNmtokenValueInternal(doc, value));
Owen Taylor3473f882001-02-23 17:55:21 +00003805 case XML_ATTRIBUTE_CDATA:
3806 break;
3807 }
3808 return(1);
3809}
3810
Daniel Veillardbe2bd6a2008-11-27 15:26:28 +00003811/**
3812 * xmlValidateAttributeValue:
3813 * @type: an attribute type
3814 * @value: an attribute value
3815 *
3816 * Validate that the given attribute value match the proper production
3817 *
3818 * [ VC: ID ]
3819 * Values of type ID must match the Name production....
3820 *
3821 * [ VC: IDREF ]
3822 * Values of type IDREF must match the Name production, and values
3823 * of type IDREFS must match Names ...
3824 *
3825 * [ VC: Entity Name ]
3826 * Values of type ENTITY must match the Name production, values
3827 * of type ENTITIES must match Names ...
3828 *
3829 * [ VC: Name Token ]
3830 * Values of type NMTOKEN must match the Nmtoken production; values
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003831 * of type NMTOKENS must match Nmtokens.
Daniel Veillardbe2bd6a2008-11-27 15:26:28 +00003832 *
3833 * returns 1 if valid or 0 otherwise
3834 */
Daniel Veillardae0765b2008-07-31 19:54:59 +00003835int
3836xmlValidateAttributeValue(xmlAttributeType type, const xmlChar *value) {
3837 return(xmlValidateAttributeValueInternal(NULL, type, value));
3838}
3839
Owen Taylor3473f882001-02-23 17:55:21 +00003840/**
3841 * xmlValidateAttributeValue2:
3842 * @ctxt: the validation context
3843 * @doc: the document
3844 * @name: the attribute name (used for error reporting only)
3845 * @type: the attribute type
3846 * @value: the attribute value
3847 *
3848 * Validate that the given attribute value match a given type.
3849 * This typically cannot be done before having finished parsing
3850 * the subsets.
3851 *
3852 * [ VC: IDREF ]
3853 * Values of type IDREF must match one of the declared IDs
3854 * Values of type IDREFS must match a sequence of the declared IDs
3855 * each Name must match the value of an ID attribute on some element
3856 * in the XML document; i.e. IDREF values must match the value of
3857 * some ID attribute
3858 *
3859 * [ VC: Entity Name ]
3860 * Values of type ENTITY must match one declared entity
3861 * Values of type ENTITIES must match a sequence of declared entities
3862 *
3863 * [ VC: Notation Attributes ]
3864 * all notation names in the declaration must be declared.
3865 *
3866 * returns 1 if valid or 0 otherwise
3867 */
3868
Daniel Veillard56a4cb82001-03-24 17:00:36 +00003869static int
Owen Taylor3473f882001-02-23 17:55:21 +00003870xmlValidateAttributeValue2(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3871 const xmlChar *name, xmlAttributeType type, const xmlChar *value) {
3872 int ret = 1;
3873 switch (type) {
3874 case XML_ATTRIBUTE_IDREFS:
3875 case XML_ATTRIBUTE_IDREF:
3876 case XML_ATTRIBUTE_ID:
3877 case XML_ATTRIBUTE_NMTOKENS:
3878 case XML_ATTRIBUTE_ENUMERATION:
3879 case XML_ATTRIBUTE_NMTOKEN:
3880 case XML_ATTRIBUTE_CDATA:
3881 break;
3882 case XML_ATTRIBUTE_ENTITY: {
3883 xmlEntityPtr ent;
3884
3885 ent = xmlGetDocEntity(doc, value);
Daniel Veillard62998c02003-09-15 12:56:36 +00003886 /* yeah it's a bit messy... */
Daniel Veillard878eab02002-02-19 13:46:09 +00003887 if ((ent == NULL) && (doc->standalone == 1)) {
3888 doc->standalone = 0;
3889 ent = xmlGetDocEntity(doc, value);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08003890 }
Owen Taylor3473f882001-02-23 17:55:21 +00003891 if (ent == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003892 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3893 XML_DTD_UNKNOWN_ENTITY,
Owen Taylor3473f882001-02-23 17:55:21 +00003894 "ENTITY attribute %s reference an unknown entity \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003895 name, value, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003896 ret = 0;
3897 } else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003898 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3899 XML_DTD_ENTITY_TYPE,
Owen Taylor3473f882001-02-23 17:55:21 +00003900 "ENTITY attribute %s reference an entity \"%s\" of wrong type\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003901 name, value, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003902 ret = 0;
3903 }
3904 break;
3905 }
3906 case XML_ATTRIBUTE_ENTITIES: {
3907 xmlChar *dup, *nam = NULL, *cur, save;
3908 xmlEntityPtr ent;
3909
3910 dup = xmlStrdup(value);
3911 if (dup == NULL)
3912 return(0);
3913 cur = dup;
3914 while (*cur != 0) {
3915 nam = cur;
William M. Brack76e95df2003-10-18 16:20:14 +00003916 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00003917 save = *cur;
3918 *cur = 0;
3919 ent = xmlGetDocEntity(doc, nam);
3920 if (ent == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003921 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3922 XML_DTD_UNKNOWN_ENTITY,
Owen Taylor3473f882001-02-23 17:55:21 +00003923 "ENTITIES attribute %s reference an unknown entity \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003924 name, nam, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003925 ret = 0;
3926 } else if (ent->etype != XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003927 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3928 XML_DTD_ENTITY_TYPE,
Owen Taylor3473f882001-02-23 17:55:21 +00003929 "ENTITIES attribute %s reference an entity \"%s\" of wrong type\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003930 name, nam, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003931 ret = 0;
3932 }
3933 if (save == 0)
3934 break;
3935 *cur = save;
William M. Brack76e95df2003-10-18 16:20:14 +00003936 while (IS_BLANK_CH(*cur)) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00003937 }
3938 xmlFree(dup);
3939 break;
3940 }
3941 case XML_ATTRIBUTE_NOTATION: {
3942 xmlNotationPtr nota;
3943
3944 nota = xmlGetDtdNotationDesc(doc->intSubset, value);
3945 if ((nota == NULL) && (doc->extSubset != NULL))
3946 nota = xmlGetDtdNotationDesc(doc->extSubset, value);
3947
3948 if (nota == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003949 xmlErrValidNode(ctxt, (xmlNodePtr) doc,
3950 XML_DTD_UNKNOWN_NOTATION,
Owen Taylor3473f882001-02-23 17:55:21 +00003951 "NOTATION attribute %s reference an unknown notation \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00003952 name, value, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00003953 ret = 0;
3954 }
3955 break;
3956 }
3957 }
3958 return(ret);
3959}
3960
3961/**
Daniel Veillard8dc16a62002-02-19 21:08:48 +00003962 * xmlValidCtxtNormalizeAttributeValue:
3963 * @ctxt: the validation context
3964 * @doc: the document
3965 * @elem: the parent
3966 * @name: the attribute name
3967 * @value: the attribute value
3968 * @ctxt: the validation context or NULL
3969 *
3970 * Does the validation related extra step of the normalization of attribute
3971 * values:
3972 *
3973 * If the declared value is not CDATA, then the XML processor must further
3974 * process the normalized attribute value by discarding any leading and
3975 * trailing space (#x20) characters, and by replacing sequences of space
3976 * (#x20) characters by single space (#x20) character.
3977 *
3978 * Also check VC: Standalone Document Declaration in P32, and update
3979 * ctxt->valid accordingly
3980 *
3981 * returns a new normalized string if normalization is needed, NULL otherwise
3982 * the caller must free the returned value.
3983 */
3984
3985xmlChar *
3986xmlValidCtxtNormalizeAttributeValue(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
3987 xmlNodePtr elem, const xmlChar *name, const xmlChar *value) {
3988 xmlChar *ret, *dst;
3989 const xmlChar *src;
3990 xmlAttributePtr attrDecl = NULL;
3991 int extsubset = 0;
3992
3993 if (doc == NULL) return(NULL);
3994 if (elem == NULL) return(NULL);
3995 if (name == NULL) return(NULL);
3996 if (value == NULL) return(NULL);
3997
3998 if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00003999 xmlChar fn[50];
4000 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004001
Daniel Veillardc00cda82003-04-07 10:22:39 +00004002 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4003 if (fullname == NULL)
Daniel Veillard24505b02005-07-28 23:49:35 +00004004 return(NULL);
Daniel Veillardc00cda82003-04-07 10:22:39 +00004005 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, name);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00004006 if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004007 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname, name);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00004008 if (attrDecl != NULL)
4009 extsubset = 1;
4010 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00004011 if ((fullname != fn) && (fullname != elem->name))
4012 xmlFree(fullname);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00004013 }
4014 if ((attrDecl == NULL) && (doc->intSubset != NULL))
4015 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, name);
4016 if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
4017 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, elem->name, name);
4018 if (attrDecl != NULL)
4019 extsubset = 1;
4020 }
4021
4022 if (attrDecl == NULL)
4023 return(NULL);
4024 if (attrDecl->atype == XML_ATTRIBUTE_CDATA)
4025 return(NULL);
4026
4027 ret = xmlStrdup(value);
4028 if (ret == NULL)
4029 return(NULL);
4030 src = value;
4031 dst = ret;
4032 while (*src == 0x20) src++;
4033 while (*src != 0) {
4034 if (*src == 0x20) {
4035 while (*src == 0x20) src++;
4036 if (*src != 0)
4037 *dst++ = 0x20;
4038 } else {
4039 *dst++ = *src++;
4040 }
4041 }
4042 *dst = 0;
4043 if ((doc->standalone) && (extsubset == 1) && (!xmlStrEqual(value, ret))) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004044 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_STANDALONE,
Daniel Veillard8dc16a62002-02-19 21:08:48 +00004045"standalone: %s on %s value had to be normalized based on external subset declaration\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004046 name, elem->name, NULL);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00004047 ctxt->valid = 0;
4048 }
4049 return(ret);
4050}
4051
4052/**
Owen Taylor3473f882001-02-23 17:55:21 +00004053 * xmlValidNormalizeAttributeValue:
4054 * @doc: the document
4055 * @elem: the parent
4056 * @name: the attribute name
4057 * @value: the attribute value
4058 *
4059 * Does the validation related extra step of the normalization of attribute
4060 * values:
4061 *
4062 * If the declared value is not CDATA, then the XML processor must further
4063 * process the normalized attribute value by discarding any leading and
4064 * trailing space (#x20) characters, and by replacing sequences of space
4065 * (#x20) characters by single space (#x20) character.
4066 *
Daniel Veillard652327a2003-09-29 18:02:38 +00004067 * Returns a new normalized string if normalization is needed, NULL otherwise
Owen Taylor3473f882001-02-23 17:55:21 +00004068 * the caller must free the returned value.
4069 */
4070
4071xmlChar *
4072xmlValidNormalizeAttributeValue(xmlDocPtr doc, xmlNodePtr elem,
4073 const xmlChar *name, const xmlChar *value) {
4074 xmlChar *ret, *dst;
4075 const xmlChar *src;
4076 xmlAttributePtr attrDecl = NULL;
4077
4078 if (doc == NULL) return(NULL);
4079 if (elem == NULL) return(NULL);
4080 if (name == NULL) return(NULL);
4081 if (value == NULL) return(NULL);
4082
4083 if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004084 xmlChar fn[50];
4085 xmlChar *fullname;
Daniel Veillard594e5df2009-09-07 14:58:47 +02004086
Daniel Veillardc00cda82003-04-07 10:22:39 +00004087 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4088 if (fullname == NULL)
Daniel Veillard24505b02005-07-28 23:49:35 +00004089 return(NULL);
Daniel Veillardc00cda82003-04-07 10:22:39 +00004090 if ((fullname != fn) && (fullname != elem->name))
4091 xmlFree(fullname);
Owen Taylor3473f882001-02-23 17:55:21 +00004092 }
4093 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, elem->name, name);
4094 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4095 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, elem->name, name);
4096
4097 if (attrDecl == NULL)
4098 return(NULL);
4099 if (attrDecl->atype == XML_ATTRIBUTE_CDATA)
4100 return(NULL);
4101
4102 ret = xmlStrdup(value);
4103 if (ret == NULL)
4104 return(NULL);
4105 src = value;
4106 dst = ret;
4107 while (*src == 0x20) src++;
4108 while (*src != 0) {
4109 if (*src == 0x20) {
4110 while (*src == 0x20) src++;
4111 if (*src != 0)
4112 *dst++ = 0x20;
4113 } else {
4114 *dst++ = *src++;
4115 }
4116 }
4117 *dst = 0;
4118 return(ret);
4119}
4120
Daniel Veillard56a4cb82001-03-24 17:00:36 +00004121static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01004122xmlValidateAttributeIdCallback(void *payload, void *data,
4123 const xmlChar *name ATTRIBUTE_UNUSED) {
4124 xmlAttributePtr attr = (xmlAttributePtr) payload;
4125 int *count = (int *) data;
Owen Taylor3473f882001-02-23 17:55:21 +00004126 if (attr->atype == XML_ATTRIBUTE_ID) (*count)++;
4127}
4128
4129/**
4130 * xmlValidateAttributeDecl:
4131 * @ctxt: the validation context
4132 * @doc: a document instance
4133 * @attr: an attribute definition
4134 *
4135 * Try to validate a single attribute definition
4136 * basically it does the following checks as described by the
4137 * XML-1.0 recommendation:
4138 * - [ VC: Attribute Default Legal ]
4139 * - [ VC: Enumeration ]
4140 * - [ VC: ID Attribute Default ]
4141 *
4142 * The ID/IDREF uniqueness and matching are done separately
4143 *
4144 * returns 1 if valid or 0 otherwise
4145 */
4146
4147int
4148xmlValidateAttributeDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4149 xmlAttributePtr attr) {
4150 int ret = 1;
4151 int val;
4152 CHECK_DTD;
4153 if(attr == NULL) return(1);
Daniel Veillardae0765b2008-07-31 19:54:59 +00004154
Owen Taylor3473f882001-02-23 17:55:21 +00004155 /* Attribute Default Legal */
4156 /* Enumeration */
4157 if (attr->defaultValue != NULL) {
Daniel Veillardae0765b2008-07-31 19:54:59 +00004158 val = xmlValidateAttributeValueInternal(doc, attr->atype,
4159 attr->defaultValue);
Owen Taylor3473f882001-02-23 17:55:21 +00004160 if (val == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004161 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004162 "Syntax of default value for attribute %s of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004163 attr->name, attr->elem, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004164 }
4165 ret &= val;
4166 }
4167
4168 /* ID Attribute Default */
4169 if ((attr->atype == XML_ATTRIBUTE_ID)&&
4170 (attr->def != XML_ATTRIBUTE_IMPLIED) &&
4171 (attr->def != XML_ATTRIBUTE_REQUIRED)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004172 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_FIXED,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004173 "ID attribute %s of %s is not valid must be #IMPLIED or #REQUIRED\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004174 attr->name, attr->elem, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004175 ret = 0;
4176 }
4177
4178 /* One ID per Element Type */
4179 if (attr->atype == XML_ATTRIBUTE_ID) {
4180 int nbId;
4181
Daniel Veillardcbaf3992001-12-31 16:16:02 +00004182 /* the trick is that we parse DtD as their own internal subset */
Owen Taylor3473f882001-02-23 17:55:21 +00004183 xmlElementPtr elem = xmlGetDtdElementDesc(doc->intSubset,
4184 attr->elem);
4185 if (elem != NULL) {
Daniel Veillarddbee0f12005-06-27 13:42:57 +00004186 nbId = xmlScanIDAttributeDecl(NULL, elem, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00004187 } else {
4188 xmlAttributeTablePtr table;
4189
4190 /*
4191 * The attribute may be declared in the internal subset and the
4192 * element in the external subset.
4193 */
4194 nbId = 0;
Daniel Veillard11ce4002006-03-10 00:36:23 +00004195 if (doc->intSubset != NULL) {
4196 table = (xmlAttributeTablePtr) doc->intSubset->attributes;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01004197 xmlHashScan3(table, NULL, NULL, attr->elem,
Daniel Veillard11ce4002006-03-10 00:36:23 +00004198 xmlValidateAttributeIdCallback, &nbId);
4199 }
Owen Taylor3473f882001-02-23 17:55:21 +00004200 }
4201 if (nbId > 1) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004202
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004203 xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
Owen Taylor3473f882001-02-23 17:55:21 +00004204 "Element %s has %d ID attribute defined in the internal subset : %s\n",
4205 attr->elem, nbId, attr->name);
4206 } else if (doc->extSubset != NULL) {
4207 int extId = 0;
4208 elem = xmlGetDtdElementDesc(doc->extSubset, attr->elem);
4209 if (elem != NULL) {
Daniel Veillarddbee0f12005-06-27 13:42:57 +00004210 extId = xmlScanIDAttributeDecl(NULL, elem, 0);
Owen Taylor3473f882001-02-23 17:55:21 +00004211 }
4212 if (extId > 1) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004213 xmlErrValidNodeNr(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
Owen Taylor3473f882001-02-23 17:55:21 +00004214 "Element %s has %d ID attribute defined in the external subset : %s\n",
4215 attr->elem, extId, attr->name);
4216 } else if (extId + nbId > 1) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004217 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ID_SUBSET,
Owen Taylor3473f882001-02-23 17:55:21 +00004218"Element %s has ID attributes defined in the internal and external subset : %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004219 attr->elem, attr->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004220 }
4221 }
4222 }
4223
4224 /* Validity Constraint: Enumeration */
4225 if ((attr->defaultValue != NULL) && (attr->tree != NULL)) {
4226 xmlEnumerationPtr tree = attr->tree;
4227 while (tree != NULL) {
4228 if (xmlStrEqual(tree->name, attr->defaultValue)) break;
4229 tree = tree->next;
4230 }
4231 if (tree == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004232 xmlErrValidNode(ctxt, (xmlNodePtr) attr, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004233"Default value \"%s\" for attribute %s of %s is not among the enumerated set\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004234 attr->defaultValue, attr->name, attr->elem);
4235 ret = 0;
4236 }
4237 }
4238
4239 return(ret);
4240}
4241
4242/**
4243 * xmlValidateElementDecl:
4244 * @ctxt: the validation context
4245 * @doc: a document instance
4246 * @elem: an element definition
4247 *
4248 * Try to validate a single element definition
4249 * basically it does the following checks as described by the
4250 * XML-1.0 recommendation:
4251 * - [ VC: One ID per Element Type ]
4252 * - [ VC: No Duplicate Types ]
4253 * - [ VC: Unique Element Type Declaration ]
4254 *
4255 * returns 1 if valid or 0 otherwise
4256 */
4257
4258int
4259xmlValidateElementDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4260 xmlElementPtr elem) {
4261 int ret = 1;
4262 xmlElementPtr tst;
4263
4264 CHECK_DTD;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004265
Owen Taylor3473f882001-02-23 17:55:21 +00004266 if (elem == NULL) return(1);
4267
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00004268#if 0
Daniel Veillard84d70a42002-09-16 10:51:38 +00004269#ifdef LIBXML_REGEXP_ENABLED
4270 /* Build the regexp associated to the content model */
4271 ret = xmlValidBuildContentModel(ctxt, elem);
4272#endif
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00004273#endif
Daniel Veillard84d70a42002-09-16 10:51:38 +00004274
Owen Taylor3473f882001-02-23 17:55:21 +00004275 /* No Duplicate Types */
4276 if (elem->etype == XML_ELEMENT_TYPE_MIXED) {
4277 xmlElementContentPtr cur, next;
4278 const xmlChar *name;
4279
4280 cur = elem->content;
4281 while (cur != NULL) {
4282 if (cur->type != XML_ELEMENT_CONTENT_OR) break;
4283 if (cur->c1 == NULL) break;
4284 if (cur->c1->type == XML_ELEMENT_CONTENT_ELEMENT) {
4285 name = cur->c1->name;
4286 next = cur->c2;
4287 while (next != NULL) {
4288 if (next->type == XML_ELEMENT_CONTENT_ELEMENT) {
Daniel Veillard7b68df92003-08-03 22:58:54 +00004289 if ((xmlStrEqual(next->name, name)) &&
Daniel Veillarda7216122009-08-21 18:22:58 +02004290 (xmlStrEqual(next->prefix, cur->c1->prefix))) {
4291 if (cur->c1->prefix == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004292 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Owen Taylor3473f882001-02-23 17:55:21 +00004293 "Definition of %s has duplicate references of %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004294 elem->name, name, NULL);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004295 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004296 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Daniel Veillard7b68df92003-08-03 22:58:54 +00004297 "Definition of %s has duplicate references of %s:%s\n",
Daniel Veillarda7216122009-08-21 18:22:58 +02004298 elem->name, cur->c1->prefix, name);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004299 }
Owen Taylor3473f882001-02-23 17:55:21 +00004300 ret = 0;
4301 }
4302 break;
4303 }
4304 if (next->c1 == NULL) break;
4305 if (next->c1->type != XML_ELEMENT_CONTENT_ELEMENT) break;
Daniel Veillard7b68df92003-08-03 22:58:54 +00004306 if ((xmlStrEqual(next->c1->name, name)) &&
Daniel Veillarda7216122009-08-21 18:22:58 +02004307 (xmlStrEqual(next->c1->prefix, cur->c1->prefix))) {
4308 if (cur->c1->prefix == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004309 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Daniel Veillard7b68df92003-08-03 22:58:54 +00004310 "Definition of %s has duplicate references to %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004311 elem->name, name, NULL);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004312 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004313 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_CONTENT_ERROR,
Daniel Veillard7b68df92003-08-03 22:58:54 +00004314 "Definition of %s has duplicate references to %s:%s\n",
Daniel Veillarda7216122009-08-21 18:22:58 +02004315 elem->name, cur->c1->prefix, name);
Daniel Veillard7b68df92003-08-03 22:58:54 +00004316 }
Owen Taylor3473f882001-02-23 17:55:21 +00004317 ret = 0;
4318 }
4319 next = next->c2;
4320 }
4321 }
4322 cur = cur->c2;
4323 }
4324 }
4325
4326 /* VC: Unique Element Type Declaration */
4327 tst = xmlGetDtdElementDesc(doc->intSubset, elem->name);
Daniel Veillarda10efa82001-04-18 13:09:01 +00004328 if ((tst != NULL ) && (tst != elem) &&
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004329 ((tst->prefix == elem->prefix) ||
4330 (xmlStrEqual(tst->prefix, elem->prefix))) &&
Daniel Veillarda10efa82001-04-18 13:09:01 +00004331 (tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004332 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4333 "Redefinition of element %s\n",
4334 elem->name, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004335 ret = 0;
4336 }
4337 tst = xmlGetDtdElementDesc(doc->extSubset, elem->name);
Daniel Veillarda10efa82001-04-18 13:09:01 +00004338 if ((tst != NULL ) && (tst != elem) &&
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004339 ((tst->prefix == elem->prefix) ||
4340 (xmlStrEqual(tst->prefix, elem->prefix))) &&
Daniel Veillarda10efa82001-04-18 13:09:01 +00004341 (tst->etype != XML_ELEMENT_TYPE_UNDEFINED)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004342 xmlErrValidNode(ctxt, (xmlNodePtr) elem, XML_DTD_ELEM_REDEFINED,
4343 "Redefinition of element %s\n",
4344 elem->name, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004345 ret = 0;
4346 }
Daniel Veillarda10efa82001-04-18 13:09:01 +00004347 /* One ID per Element Type
4348 * already done when registering the attribute
Owen Taylor3473f882001-02-23 17:55:21 +00004349 if (xmlScanIDAttributeDecl(ctxt, elem) > 1) {
4350 ret = 0;
Daniel Veillarda10efa82001-04-18 13:09:01 +00004351 } */
Owen Taylor3473f882001-02-23 17:55:21 +00004352 return(ret);
4353}
4354
4355/**
4356 * xmlValidateOneAttribute:
4357 * @ctxt: the validation context
4358 * @doc: a document instance
4359 * @elem: an element instance
4360 * @attr: an attribute instance
4361 * @value: the attribute value (without entities processing)
4362 *
4363 * Try to validate a single attribute for an element
4364 * basically it does the following checks as described by the
4365 * XML-1.0 recommendation:
4366 * - [ VC: Attribute Value Type ]
4367 * - [ VC: Fixed Attribute Default ]
4368 * - [ VC: Entity Name ]
4369 * - [ VC: Name Token ]
4370 * - [ VC: ID ]
4371 * - [ VC: IDREF ]
4372 * - [ VC: Entity Name ]
4373 * - [ VC: Notation Attributes ]
4374 *
4375 * The ID/IDREF uniqueness and matching are done separately
4376 *
4377 * returns 1 if valid or 0 otherwise
4378 */
4379
4380int
4381xmlValidateOneAttribute(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004382 xmlNodePtr elem, xmlAttrPtr attr, const xmlChar *value)
Daniel Veillard07cb8222003-09-10 10:51:05 +00004383{
Owen Taylor3473f882001-02-23 17:55:21 +00004384 xmlAttributePtr attrDecl = NULL;
4385 int val;
4386 int ret = 1;
4387
4388 CHECK_DTD;
4389 if ((elem == NULL) || (elem->name == NULL)) return(0);
4390 if ((attr == NULL) || (attr->name == NULL)) return(0);
4391
4392 if ((elem->ns != NULL) && (elem->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004393 xmlChar fn[50];
4394 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004395
Daniel Veillardc00cda82003-04-07 10:22:39 +00004396 fullname = xmlBuildQName(elem->name, elem->ns->prefix, fn, 50);
4397 if (fullname == NULL)
4398 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00004399 if (attr->ns != NULL) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004400 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
Owen Taylor3473f882001-02-23 17:55:21 +00004401 attr->name, attr->ns->prefix);
4402 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillardc00cda82003-04-07 10:22:39 +00004403 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
Owen Taylor3473f882001-02-23 17:55:21 +00004404 attr->name, attr->ns->prefix);
4405 } else {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004406 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00004407 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4408 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
Daniel Veillardc00cda82003-04-07 10:22:39 +00004409 fullname, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00004410 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00004411 if ((fullname != fn) && (fullname != elem->name))
4412 xmlFree(fullname);
Owen Taylor3473f882001-02-23 17:55:21 +00004413 }
4414 if (attrDecl == NULL) {
4415 if (attr->ns != NULL) {
4416 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, elem->name,
4417 attr->name, attr->ns->prefix);
4418 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4419 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, elem->name,
4420 attr->name, attr->ns->prefix);
4421 } else {
4422 attrDecl = xmlGetDtdAttrDesc(doc->intSubset,
4423 elem->name, attr->name);
4424 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4425 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4426 elem->name, attr->name);
4427 }
4428 }
4429
4430
4431 /* Validity Constraint: Attribute Value Type */
4432 if (attrDecl == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004433 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004434 "No declaration for attribute %s of element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004435 attr->name, elem->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004436 return(0);
4437 }
4438 attr->atype = attrDecl->atype;
4439
Daniel Veillardae0765b2008-07-31 19:54:59 +00004440 val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
Owen Taylor3473f882001-02-23 17:55:21 +00004441 if (val == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004442 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004443 "Syntax of value for attribute %s of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004444 attr->name, elem->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00004445 ret = 0;
4446 }
4447
4448 /* Validity constraint: Fixed Attribute Default */
4449 if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4450 if (!xmlStrEqual(value, attrDecl->defaultValue)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004451 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004452 "Value for attribute %s of %s is different from default \"%s\"\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004453 attr->name, elem->name, attrDecl->defaultValue);
4454 ret = 0;
4455 }
4456 }
4457
4458 /* Validity Constraint: ID uniqueness */
4459 if (attrDecl->atype == XML_ATTRIBUTE_ID) {
4460 if (xmlAddID(ctxt, doc, value, attr) == NULL)
4461 ret = 0;
4462 }
4463
4464 if ((attrDecl->atype == XML_ATTRIBUTE_IDREF) ||
4465 (attrDecl->atype == XML_ATTRIBUTE_IDREFS)) {
4466 if (xmlAddRef(ctxt, doc, value, attr) == NULL)
4467 ret = 0;
4468 }
4469
4470 /* Validity Constraint: Notation Attributes */
4471 if (attrDecl->atype == XML_ATTRIBUTE_NOTATION) {
4472 xmlEnumerationPtr tree = attrDecl->tree;
4473 xmlNotationPtr nota;
4474
4475 /* First check that the given NOTATION was declared */
4476 nota = xmlGetDtdNotationDesc(doc->intSubset, value);
4477 if (nota == NULL)
4478 nota = xmlGetDtdNotationDesc(doc->extSubset, value);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004479
Owen Taylor3473f882001-02-23 17:55:21 +00004480 if (nota == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004481 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004482 "Value \"%s\" for attribute %s of %s is not a declared Notation\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004483 value, attr->name, elem->name);
4484 ret = 0;
4485 }
4486
4487 /* Second, verify that it's among the list */
4488 while (tree != NULL) {
4489 if (xmlStrEqual(tree->name, value)) break;
4490 tree = tree->next;
4491 }
4492 if (tree == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004493 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004494"Value \"%s\" for attribute %s of %s is not among the enumerated notations\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004495 value, attr->name, elem->name);
4496 ret = 0;
4497 }
4498 }
4499
4500 /* Validity Constraint: Enumeration */
4501 if (attrDecl->atype == XML_ATTRIBUTE_ENUMERATION) {
4502 xmlEnumerationPtr tree = attrDecl->tree;
4503 while (tree != NULL) {
4504 if (xmlStrEqual(tree->name, value)) break;
4505 tree = tree->next;
4506 }
4507 if (tree == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004508 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004509 "Value \"%s\" for attribute %s of %s is not among the enumerated set\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004510 value, attr->name, elem->name);
4511 ret = 0;
4512 }
4513 }
4514
4515 /* Fixed Attribute Default */
4516 if ((attrDecl->def == XML_ATTRIBUTE_FIXED) &&
4517 (!xmlStrEqual(attrDecl->defaultValue, value))) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004518 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00004519 "Value for attribute %s of %s must be \"%s\"\n",
Owen Taylor3473f882001-02-23 17:55:21 +00004520 attr->name, elem->name, attrDecl->defaultValue);
4521 ret = 0;
4522 }
4523
4524 /* Extra check for the attribute value */
4525 ret &= xmlValidateAttributeValue2(ctxt, doc, attr->name,
4526 attrDecl->atype, value);
4527
4528 return(ret);
4529}
4530
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004531/**
4532 * xmlValidateOneNamespace:
4533 * @ctxt: the validation context
4534 * @doc: a document instance
4535 * @elem: an element instance
Daniel Veillarda9b66d02002-12-11 14:23:49 +00004536 * @prefix: the namespace prefix
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004537 * @ns: an namespace declaration instance
4538 * @value: the attribute value (without entities processing)
4539 *
4540 * Try to validate a single namespace declaration for an element
4541 * basically it does the following checks as described by the
4542 * XML-1.0 recommendation:
4543 * - [ VC: Attribute Value Type ]
4544 * - [ VC: Fixed Attribute Default ]
4545 * - [ VC: Entity Name ]
4546 * - [ VC: Name Token ]
4547 * - [ VC: ID ]
4548 * - [ VC: IDREF ]
4549 * - [ VC: Entity Name ]
4550 * - [ VC: Notation Attributes ]
4551 *
4552 * The ID/IDREF uniqueness and matching are done separately
4553 *
4554 * returns 1 if valid or 0 otherwise
4555 */
4556
4557int
4558xmlValidateOneNamespace(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
4559xmlNodePtr elem, const xmlChar *prefix, xmlNsPtr ns, const xmlChar *value) {
4560 /* xmlElementPtr elemDecl; */
4561 xmlAttributePtr attrDecl = NULL;
4562 int val;
4563 int ret = 1;
4564
4565 CHECK_DTD;
4566 if ((elem == NULL) || (elem->name == NULL)) return(0);
4567 if ((ns == NULL) || (ns->href == NULL)) return(0);
4568
4569 if (prefix != NULL) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004570 xmlChar fn[50];
4571 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004572
Daniel Veillardc00cda82003-04-07 10:22:39 +00004573 fullname = xmlBuildQName(elem->name, prefix, fn, 50);
4574 if (fullname == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00004575 xmlVErrMemory(ctxt, "Validating namespace");
Daniel Veillardc00cda82003-04-07 10:22:39 +00004576 return(0);
4577 }
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004578 if (ns->prefix != NULL) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004579 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004580 ns->prefix, BAD_CAST "xmlns");
4581 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillardc00cda82003-04-07 10:22:39 +00004582 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004583 ns->prefix, BAD_CAST "xmlns");
4584 } else {
Daniel Veillardc00cda82003-04-07 10:22:39 +00004585 attrDecl = xmlGetDtdAttrDesc(doc->intSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004586 BAD_CAST "xmlns");
4587 if ((attrDecl == NULL) && (doc->extSubset != NULL))
Daniel Veillardc00cda82003-04-07 10:22:39 +00004588 attrDecl = xmlGetDtdAttrDesc(doc->extSubset, fullname,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004589 BAD_CAST "xmlns");
4590 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00004591 if ((fullname != fn) && (fullname != elem->name))
4592 xmlFree(fullname);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004593 }
4594 if (attrDecl == NULL) {
4595 if (ns->prefix != NULL) {
4596 attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, elem->name,
4597 ns->prefix, BAD_CAST "xmlns");
4598 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4599 attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, elem->name,
4600 ns->prefix, BAD_CAST "xmlns");
4601 } else {
4602 attrDecl = xmlGetDtdAttrDesc(doc->intSubset,
4603 elem->name, BAD_CAST "xmlns");
4604 if ((attrDecl == NULL) && (doc->extSubset != NULL))
4605 attrDecl = xmlGetDtdAttrDesc(doc->extSubset,
4606 elem->name, BAD_CAST "xmlns");
4607 }
4608 }
4609
4610
4611 /* Validity Constraint: Attribute Value Type */
4612 if (attrDecl == NULL) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004613 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004614 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004615 "No declaration for attribute xmlns:%s of element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004616 ns->prefix, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004617 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004618 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ATTRIBUTE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004619 "No declaration for attribute xmlns of element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004620 elem->name, NULL, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004621 }
4622 return(0);
4623 }
4624
Daniel Veillardae0765b2008-07-31 19:54:59 +00004625 val = xmlValidateAttributeValueInternal(doc, attrDecl->atype, value);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004626 if (val == 0) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004627 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004628 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004629 "Syntax of value for attribute xmlns:%s of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004630 ns->prefix, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004631 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004632 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004633 "Syntax of value for attribute xmlns of %s is not valid\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004634 elem->name, NULL, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004635 }
4636 ret = 0;
4637 }
4638
4639 /* Validity constraint: Fixed Attribute Default */
4640 if (attrDecl->def == XML_ATTRIBUTE_FIXED) {
4641 if (!xmlStrEqual(value, attrDecl->defaultValue)) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004642 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004643 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004644 "Value for attribute xmlns:%s of %s is different from default \"%s\"\n",
4645 ns->prefix, elem->name, attrDecl->defaultValue);
4646 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004647 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_DEFAULT,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004648 "Value for attribute xmlns of %s is different from default \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004649 elem->name, attrDecl->defaultValue, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004650 }
4651 ret = 0;
4652 }
4653 }
4654
Nick Wellnhofer92b9e8c2017-06-06 12:56:28 +02004655 /*
4656 * Casting ns to xmlAttrPtr is wrong. We'd need separate functions
4657 * xmlAddID and xmlAddRef for namespace declarations, but it makes
4658 * no practical sense to use ID types anyway.
4659 */
4660#if 0
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004661 /* Validity Constraint: ID uniqueness */
4662 if (attrDecl->atype == XML_ATTRIBUTE_ID) {
Brian C. Youngd24a6c42017-04-06 17:02:29 -07004663 if (xmlAddID(ctxt, doc, value, (xmlAttrPtr) attrDecl) == NULL)
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004664 ret = 0;
4665 }
4666
4667 if ((attrDecl->atype == XML_ATTRIBUTE_IDREF) ||
4668 (attrDecl->atype == XML_ATTRIBUTE_IDREFS)) {
Brian C. Youngd24a6c42017-04-06 17:02:29 -07004669 if (xmlAddRef(ctxt, doc, value, (xmlAttrPtr) attrDecl) == NULL)
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004670 ret = 0;
4671 }
Nick Wellnhofer92b9e8c2017-06-06 12:56:28 +02004672#endif
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004673
4674 /* Validity Constraint: Notation Attributes */
4675 if (attrDecl->atype == XML_ATTRIBUTE_NOTATION) {
4676 xmlEnumerationPtr tree = attrDecl->tree;
4677 xmlNotationPtr nota;
4678
4679 /* First check that the given NOTATION was declared */
4680 nota = xmlGetDtdNotationDesc(doc->intSubset, value);
4681 if (nota == NULL)
4682 nota = xmlGetDtdNotationDesc(doc->extSubset, value);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004683
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004684 if (nota == NULL) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004685 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004686 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004687 "Value \"%s\" for attribute xmlns:%s of %s is not a declared Notation\n",
4688 value, ns->prefix, elem->name);
4689 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004690 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_NOTATION,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004691 "Value \"%s\" for attribute xmlns of %s is not a declared Notation\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004692 value, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004693 }
4694 ret = 0;
4695 }
4696
4697 /* Second, verify that it's among the list */
4698 while (tree != NULL) {
4699 if (xmlStrEqual(tree->name, value)) break;
4700 tree = tree->next;
4701 }
4702 if (tree == NULL) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004703 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004704 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004705"Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated notations\n",
4706 value, ns->prefix, elem->name);
4707 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004708 xmlErrValidNode(ctxt, elem, XML_DTD_NOTATION_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004709"Value \"%s\" for attribute xmlns of %s is not among the enumerated notations\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004710 value, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004711 }
4712 ret = 0;
4713 }
4714 }
4715
4716 /* Validity Constraint: Enumeration */
4717 if (attrDecl->atype == XML_ATTRIBUTE_ENUMERATION) {
4718 xmlEnumerationPtr tree = attrDecl->tree;
4719 while (tree != NULL) {
4720 if (xmlStrEqual(tree->name, value)) break;
4721 tree = tree->next;
4722 }
4723 if (tree == NULL) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004724 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004725 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004726"Value \"%s\" for attribute xmlns:%s of %s is not among the enumerated set\n",
4727 value, ns->prefix, elem->name);
4728 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004729 xmlErrValidNode(ctxt, elem, XML_DTD_ATTRIBUTE_VALUE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004730"Value \"%s\" for attribute xmlns of %s is not among the enumerated set\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004731 value, elem->name, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004732 }
4733 ret = 0;
4734 }
4735 }
4736
4737 /* Fixed Attribute Default */
4738 if ((attrDecl->def == XML_ATTRIBUTE_FIXED) &&
4739 (!xmlStrEqual(attrDecl->defaultValue, value))) {
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004740 if (ns->prefix != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004741 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004742 "Value for attribute xmlns:%s of %s must be \"%s\"\n",
4743 ns->prefix, elem->name, attrDecl->defaultValue);
4744 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004745 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004746 "Value for attribute xmlns of %s must be \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00004747 elem->name, attrDecl->defaultValue, NULL);
Daniel Veillard90d68fb2002-09-26 16:10:21 +00004748 }
4749 ret = 0;
4750 }
4751
4752 /* Extra check for the attribute value */
4753 if (ns->prefix != NULL) {
4754 ret &= xmlValidateAttributeValue2(ctxt, doc, ns->prefix,
4755 attrDecl->atype, value);
4756 } else {
4757 ret &= xmlValidateAttributeValue2(ctxt, doc, BAD_CAST "xmlns",
4758 attrDecl->atype, value);
4759 }
4760
4761 return(ret);
4762}
4763
Daniel Veillard118aed72002-09-24 14:13:13 +00004764#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004765/**
4766 * xmlValidateSkipIgnorable:
4767 * @ctxt: the validation context
4768 * @child: the child list
4769 *
4770 * Skip ignorable elements w.r.t. the validation process
4771 *
4772 * returns the first element to consider for validation of the content model
4773 */
4774
4775static xmlNodePtr
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004776xmlValidateSkipIgnorable(xmlNodePtr child) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004777 while (child != NULL) {
4778 switch (child->type) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004779 /* These things are ignored (skipped) during validation. */
4780 case XML_PI_NODE:
4781 case XML_COMMENT_NODE:
4782 case XML_XINCLUDE_START:
4783 case XML_XINCLUDE_END:
4784 child = child->next;
4785 break;
4786 case XML_TEXT_NODE:
4787 if (xmlIsBlankNode(child))
4788 child = child->next;
4789 else
4790 return(child);
4791 break;
4792 /* keep current node */
4793 default:
4794 return(child);
4795 }
4796 }
4797 return(child);
4798}
4799
4800/**
4801 * xmlValidateElementType:
4802 * @ctxt: the validation context
4803 *
4804 * Try to validate the content model of an element internal function
4805 *
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004806 * returns 1 if valid or 0 ,-1 in case of error, -2 if an entity
4807 * reference is found and -3 if the validation succeeded but
4808 * the content model is not determinist.
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004809 */
4810
4811static int
4812xmlValidateElementType(xmlValidCtxtPtr ctxt) {
Daniel Veillardb4545fd2001-11-20 09:37:09 +00004813 int ret = -1;
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00004814 int determinist = 1;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004815
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004816 NODE = xmlValidateSkipIgnorable(NODE);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004817 if ((NODE == NULL) && (CONT == NULL))
4818 return(1);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08004819 if ((NODE == NULL) &&
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004820 ((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
4821 (CONT->ocur == XML_ELEMENT_CONTENT_OPT))) {
4822 return(1);
4823 }
4824 if (CONT == NULL) return(-1);
Daniel Veillard7533cc82001-04-24 15:52:00 +00004825 if ((NODE != NULL) && (NODE->type == XML_ENTITY_REF_NODE))
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004826 return(-2);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004827
4828 /*
4829 * We arrive here when more states need to be examined
4830 */
4831cont:
4832
4833 /*
4834 * We just recovered from a rollback generated by a possible
4835 * epsilon transition, go directly to the analysis phase
4836 */
4837 if (STATE == ROLLBACK_PARENT) {
Daniel Veillardcbaf3992001-12-31 16:16:02 +00004838 DEBUG_VALID_MSG("restored parent branch");
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004839 DEBUG_VALID_STATE(NODE, CONT)
4840 ret = 1;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004841 goto analyze;
4842 }
4843
4844 DEBUG_VALID_STATE(NODE, CONT)
4845 /*
4846 * we may have to save a backup state here. This is the equivalent
4847 * of handling epsilon transition in NFAs.
4848 */
Daniel Veillarde62d36c2001-05-15 08:53:16 +00004849 if ((CONT != NULL) &&
Daniel Veillardce2c2f02001-10-18 14:57:24 +00004850 ((CONT->parent == NULL) ||
4851 (CONT->parent->type != XML_ELEMENT_CONTENT_OR)) &&
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004852 ((CONT->ocur == XML_ELEMENT_CONTENT_MULT) ||
Daniel Veillardca1f1722001-04-20 15:47:35 +00004853 (CONT->ocur == XML_ELEMENT_CONTENT_OPT) ||
Daniel Veillard5344c602001-12-31 16:37:34 +00004854 ((CONT->ocur == XML_ELEMENT_CONTENT_PLUS) && (OCCURRENCE)))) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004855 DEBUG_VALID_MSG("saving parent branch");
Daniel Veillard940492d2002-04-15 10:15:25 +00004856 if (vstateVPush(ctxt, CONT, NODE, DEPTH, OCCURS, ROLLBACK_PARENT) < 0)
4857 return(0);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004858 }
4859
4860
4861 /*
4862 * Check first if the content matches
4863 */
4864 switch (CONT->type) {
4865 case XML_ELEMENT_CONTENT_PCDATA:
4866 if (NODE == NULL) {
4867 DEBUG_VALID_MSG("pcdata failed no node");
4868 ret = 0;
4869 break;
4870 }
4871 if (NODE->type == XML_TEXT_NODE) {
4872 DEBUG_VALID_MSG("pcdata found, skip to next");
4873 /*
4874 * go to next element in the content model
4875 * skipping ignorable elems
4876 */
4877 do {
4878 NODE = NODE->next;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004879 NODE = xmlValidateSkipIgnorable(NODE);
4880 if ((NODE != NULL) &&
4881 (NODE->type == XML_ENTITY_REF_NODE))
4882 return(-2);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004883 } while ((NODE != NULL) &&
4884 ((NODE->type != XML_ELEMENT_NODE) &&
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00004885 (NODE->type != XML_TEXT_NODE) &&
4886 (NODE->type != XML_CDATA_SECTION_NODE)));
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004887 ret = 1;
4888 break;
4889 } else {
4890 DEBUG_VALID_MSG("pcdata failed");
4891 ret = 0;
4892 break;
4893 }
4894 break;
4895 case XML_ELEMENT_CONTENT_ELEMENT:
4896 if (NODE == NULL) {
4897 DEBUG_VALID_MSG("element failed no node");
4898 ret = 0;
4899 break;
4900 }
Daniel Veillard8bdd2202001-06-11 12:47:59 +00004901 ret = ((NODE->type == XML_ELEMENT_NODE) &&
4902 (xmlStrEqual(NODE->name, CONT->name)));
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004903 if (ret == 1) {
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004904 if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4905 ret = (CONT->prefix == NULL);
4906 } else if (CONT->prefix == NULL) {
4907 ret = 0;
4908 } else {
4909 ret = xmlStrEqual(NODE->ns->prefix, CONT->prefix);
4910 }
4911 }
4912 if (ret == 1) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004913 DEBUG_VALID_MSG("element found, skip to next");
4914 /*
4915 * go to next element in the content model
4916 * skipping ignorable elems
4917 */
4918 do {
4919 NODE = NODE->next;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00004920 NODE = xmlValidateSkipIgnorable(NODE);
4921 if ((NODE != NULL) &&
4922 (NODE->type == XML_ENTITY_REF_NODE))
4923 return(-2);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004924 } while ((NODE != NULL) &&
4925 ((NODE->type != XML_ELEMENT_NODE) &&
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00004926 (NODE->type != XML_TEXT_NODE) &&
4927 (NODE->type != XML_CDATA_SECTION_NODE)));
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004928 } else {
4929 DEBUG_VALID_MSG("element failed");
4930 ret = 0;
4931 break;
4932 }
4933 break;
4934 case XML_ELEMENT_CONTENT_OR:
4935 /*
Daniel Veillard85349052001-04-20 13:48:21 +00004936 * Small optimization.
4937 */
4938 if (CONT->c1->type == XML_ELEMENT_CONTENT_ELEMENT) {
4939 if ((NODE == NULL) ||
4940 (!xmlStrEqual(NODE->name, CONT->c1->name))) {
4941 DEPTH++;
4942 CONT = CONT->c2;
4943 goto cont;
4944 }
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004945 if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4946 ret = (CONT->c1->prefix == NULL);
4947 } else if (CONT->c1->prefix == NULL) {
4948 ret = 0;
4949 } else {
4950 ret = xmlStrEqual(NODE->ns->prefix, CONT->c1->prefix);
4951 }
4952 if (ret == 0) {
4953 DEPTH++;
4954 CONT = CONT->c2;
4955 goto cont;
4956 }
Daniel Veillard85349052001-04-20 13:48:21 +00004957 }
4958
4959 /*
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004960 * save the second branch 'or' branch
4961 */
4962 DEBUG_VALID_MSG("saving 'or' branch");
Daniel Veillard940492d2002-04-15 10:15:25 +00004963 if (vstateVPush(ctxt, CONT->c2, NODE, (unsigned char)(DEPTH + 1),
4964 OCCURS, ROLLBACK_OR) < 0)
4965 return(-1);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004966 DEPTH++;
4967 CONT = CONT->c1;
4968 goto cont;
4969 case XML_ELEMENT_CONTENT_SEQ:
Daniel Veillard1d047672001-06-09 16:41:01 +00004970 /*
4971 * Small optimization.
4972 */
4973 if ((CONT->c1->type == XML_ELEMENT_CONTENT_ELEMENT) &&
4974 ((CONT->c1->ocur == XML_ELEMENT_CONTENT_OPT) ||
4975 (CONT->c1->ocur == XML_ELEMENT_CONTENT_MULT))) {
4976 if ((NODE == NULL) ||
4977 (!xmlStrEqual(NODE->name, CONT->c1->name))) {
4978 DEPTH++;
4979 CONT = CONT->c2;
4980 goto cont;
4981 }
Daniel Veillardbe480fb2001-11-08 23:36:42 +00004982 if ((NODE->ns == NULL) || (NODE->ns->prefix == NULL)) {
4983 ret = (CONT->c1->prefix == NULL);
4984 } else if (CONT->c1->prefix == NULL) {
4985 ret = 0;
4986 } else {
4987 ret = xmlStrEqual(NODE->ns->prefix, CONT->c1->prefix);
4988 }
4989 if (ret == 0) {
4990 DEPTH++;
4991 CONT = CONT->c2;
4992 goto cont;
4993 }
Daniel Veillard1d047672001-06-09 16:41:01 +00004994 }
Daniel Veillarddab4cb32001-04-20 13:03:48 +00004995 DEPTH++;
4996 CONT = CONT->c1;
4997 goto cont;
4998 }
4999
5000 /*
5001 * At this point handle going up in the tree
5002 */
5003 if (ret == -1) {
5004 DEBUG_VALID_MSG("error found returning");
5005 return(ret);
5006 }
5007analyze:
5008 while (CONT != NULL) {
5009 /*
Daniel Veillardcbaf3992001-12-31 16:16:02 +00005010 * First do the analysis depending on the occurrence model at
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005011 * this level.
5012 */
5013 if (ret == 0) {
5014 switch (CONT->ocur) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005015 xmlNodePtr cur;
5016
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005017 case XML_ELEMENT_CONTENT_ONCE:
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005018 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005019 DEBUG_VALID_MSG("Once branch failed, rollback");
5020 if (vstateVPop(ctxt) < 0 ) {
5021 DEBUG_VALID_MSG("exhaustion, failed");
5022 return(0);
5023 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005024 if (cur != ctxt->vstate->node)
5025 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005026 goto cont;
5027 case XML_ELEMENT_CONTENT_PLUS:
Daniel Veillard5344c602001-12-31 16:37:34 +00005028 if (OCCURRENCE == 0) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005029 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005030 DEBUG_VALID_MSG("Plus branch failed, rollback");
5031 if (vstateVPop(ctxt) < 0 ) {
5032 DEBUG_VALID_MSG("exhaustion, failed");
5033 return(0);
5034 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005035 if (cur != ctxt->vstate->node)
5036 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005037 goto cont;
5038 }
5039 DEBUG_VALID_MSG("Plus branch found");
5040 ret = 1;
5041 break;
5042 case XML_ELEMENT_CONTENT_MULT:
5043#ifdef DEBUG_VALID_ALGO
Daniel Veillard5344c602001-12-31 16:37:34 +00005044 if (OCCURRENCE == 0) {
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005045 DEBUG_VALID_MSG("Mult branch failed");
5046 } else {
5047 DEBUG_VALID_MSG("Mult branch found");
5048 }
5049#endif
5050 ret = 1;
5051 break;
5052 case XML_ELEMENT_CONTENT_OPT:
5053 DEBUG_VALID_MSG("Option branch failed");
5054 ret = 1;
5055 break;
5056 }
5057 } else {
5058 switch (CONT->ocur) {
5059 case XML_ELEMENT_CONTENT_OPT:
5060 DEBUG_VALID_MSG("Option branch succeeded");
5061 ret = 1;
5062 break;
5063 case XML_ELEMENT_CONTENT_ONCE:
5064 DEBUG_VALID_MSG("Once branch succeeded");
5065 ret = 1;
5066 break;
5067 case XML_ELEMENT_CONTENT_PLUS:
5068 if (STATE == ROLLBACK_PARENT) {
5069 DEBUG_VALID_MSG("Plus branch rollback");
5070 ret = 1;
5071 break;
5072 }
5073 if (NODE == NULL) {
5074 DEBUG_VALID_MSG("Plus branch exhausted");
5075 ret = 1;
5076 break;
5077 }
5078 DEBUG_VALID_MSG("Plus branch succeeded, continuing");
Daniel Veillard5344c602001-12-31 16:37:34 +00005079 SET_OCCURRENCE;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005080 goto cont;
5081 case XML_ELEMENT_CONTENT_MULT:
5082 if (STATE == ROLLBACK_PARENT) {
5083 DEBUG_VALID_MSG("Mult branch rollback");
5084 ret = 1;
5085 break;
5086 }
5087 if (NODE == NULL) {
5088 DEBUG_VALID_MSG("Mult branch exhausted");
5089 ret = 1;
5090 break;
5091 }
5092 DEBUG_VALID_MSG("Mult branch succeeded, continuing");
Daniel Veillard5344c602001-12-31 16:37:34 +00005093 /* SET_OCCURRENCE; */
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005094 goto cont;
5095 }
5096 }
5097 STATE = 0;
5098
5099 /*
5100 * Then act accordingly at the parent level
5101 */
Daniel Veillard5344c602001-12-31 16:37:34 +00005102 RESET_OCCURRENCE;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005103 if (CONT->parent == NULL)
5104 break;
5105
5106 switch (CONT->parent->type) {
5107 case XML_ELEMENT_CONTENT_PCDATA:
5108 DEBUG_VALID_MSG("Error: parent pcdata");
5109 return(-1);
5110 case XML_ELEMENT_CONTENT_ELEMENT:
5111 DEBUG_VALID_MSG("Error: parent element");
5112 return(-1);
5113 case XML_ELEMENT_CONTENT_OR:
5114 if (ret == 1) {
5115 DEBUG_VALID_MSG("Or succeeded");
5116 CONT = CONT->parent;
5117 DEPTH--;
5118 } else {
5119 DEBUG_VALID_MSG("Or failed");
5120 CONT = CONT->parent;
5121 DEPTH--;
5122 }
5123 break;
5124 case XML_ELEMENT_CONTENT_SEQ:
5125 if (ret == 0) {
5126 DEBUG_VALID_MSG("Sequence failed");
5127 CONT = CONT->parent;
5128 DEPTH--;
5129 } else if (CONT == CONT->parent->c1) {
5130 DEBUG_VALID_MSG("Sequence testing 2nd branch");
5131 CONT = CONT->parent->c2;
5132 goto cont;
5133 } else {
5134 DEBUG_VALID_MSG("Sequence succeeded");
5135 CONT = CONT->parent;
5136 DEPTH--;
5137 }
5138 }
5139 }
5140 if (NODE != NULL) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005141 xmlNodePtr cur;
5142
5143 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005144 DEBUG_VALID_MSG("Failed, remaining input, rollback");
5145 if (vstateVPop(ctxt) < 0 ) {
5146 DEBUG_VALID_MSG("exhaustion, failed");
5147 return(0);
5148 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005149 if (cur != ctxt->vstate->node)
5150 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005151 goto cont;
5152 }
5153 if (ret == 0) {
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005154 xmlNodePtr cur;
5155
5156 cur = ctxt->vstate->node;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005157 DEBUG_VALID_MSG("Failure, rollback");
5158 if (vstateVPop(ctxt) < 0 ) {
5159 DEBUG_VALID_MSG("exhaustion, failed");
5160 return(0);
5161 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005162 if (cur != ctxt->vstate->node)
5163 determinist = -3;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005164 goto cont;
5165 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005166 return(determinist);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005167}
Daniel Veillard23e73572002-09-19 19:56:43 +00005168#endif
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005169
5170/**
Daniel Veillardd3d06722001-08-15 12:06:36 +00005171 * xmlSnprintfElements:
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005172 * @buf: an output buffer
Daniel Veillardd3d06722001-08-15 12:06:36 +00005173 * @size: the size of the buffer
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005174 * @content: An element
5175 * @glob: 1 if one must print the englobing parenthesis, 0 otherwise
5176 *
5177 * This will dump the list of elements to the buffer
5178 * Intended just for the debug routine
5179 */
5180static void
Daniel Veillardd3d06722001-08-15 12:06:36 +00005181xmlSnprintfElements(char *buf, int size, xmlNodePtr node, int glob) {
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005182 xmlNodePtr cur;
Daniel Veillardd3d06722001-08-15 12:06:36 +00005183 int len;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005184
5185 if (node == NULL) return;
5186 if (glob) strcat(buf, "(");
5187 cur = node;
5188 while (cur != NULL) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00005189 len = strlen(buf);
5190 if (size - len < 50) {
5191 if ((size - len > 4) && (buf[len - 1] != '.'))
5192 strcat(buf, " ...");
5193 return;
5194 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005195 switch (cur->type) {
5196 case XML_ELEMENT_NODE:
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005197 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
Daniel Veillard4b3a84f2002-03-19 14:36:46 +00005198 if (size - len < xmlStrlen(cur->ns->prefix) + 10) {
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005199 if ((size - len > 4) && (buf[len - 1] != '.'))
5200 strcat(buf, " ...");
5201 return;
5202 }
5203 strcat(buf, (char *) cur->ns->prefix);
5204 strcat(buf, ":");
5205 }
Daniel Veillard4b3a84f2002-03-19 14:36:46 +00005206 if (size - len < xmlStrlen(cur->name) + 10) {
Daniel Veillardd3d06722001-08-15 12:06:36 +00005207 if ((size - len > 4) && (buf[len - 1] != '.'))
5208 strcat(buf, " ...");
5209 return;
5210 }
5211 strcat(buf, (char *) cur->name);
5212 if (cur->next != NULL)
5213 strcat(buf, " ");
5214 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005215 case XML_TEXT_NODE:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005216 if (xmlIsBlankNode(cur))
5217 break;
J. Peter Mugaasd2c329a2017-10-21 13:49:31 +02005218 /* Falls through. */
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005219 case XML_CDATA_SECTION_NODE:
5220 case XML_ENTITY_REF_NODE:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005221 strcat(buf, "CDATA");
5222 if (cur->next != NULL)
5223 strcat(buf, " ");
5224 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005225 case XML_ATTRIBUTE_NODE:
5226 case XML_DOCUMENT_NODE:
Daniel Veillardeae522a2001-04-23 13:41:34 +00005227#ifdef LIBXML_DOCB_ENABLED
5228 case XML_DOCB_DOCUMENT_NODE:
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005229#endif
5230 case XML_HTML_DOCUMENT_NODE:
5231 case XML_DOCUMENT_TYPE_NODE:
5232 case XML_DOCUMENT_FRAG_NODE:
5233 case XML_NOTATION_NODE:
5234 case XML_NAMESPACE_DECL:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005235 strcat(buf, "???");
5236 if (cur->next != NULL)
5237 strcat(buf, " ");
5238 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005239 case XML_ENTITY_NODE:
5240 case XML_PI_NODE:
5241 case XML_DTD_NODE:
5242 case XML_COMMENT_NODE:
5243 case XML_ELEMENT_DECL:
5244 case XML_ATTRIBUTE_DECL:
5245 case XML_ENTITY_DECL:
5246 case XML_XINCLUDE_START:
5247 case XML_XINCLUDE_END:
Daniel Veillardd3d06722001-08-15 12:06:36 +00005248 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005249 }
5250 cur = cur->next;
5251 }
5252 if (glob) strcat(buf, ")");
5253}
5254
5255/**
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005256 * xmlValidateElementContent:
5257 * @ctxt: the validation context
5258 * @child: the child list
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005259 * @elemDecl: pointer to the element declaration
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005260 * @warn: emit the error message
Daniel Veillardb9cd8b42002-09-05 10:58:49 +00005261 * @parent: the parent element (for error reporting)
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005262 *
5263 * Try to validate the content model of an element
5264 *
5265 * returns 1 if valid or 0 if not and -1 in case of error
5266 */
5267
5268static int
5269xmlValidateElementContent(xmlValidCtxtPtr ctxt, xmlNodePtr child,
Daniel Veillardb9cd8b42002-09-05 10:58:49 +00005270 xmlElementPtr elemDecl, int warn, xmlNodePtr parent) {
Daniel Veillard5acfd6b2002-09-18 16:29:02 +00005271 int ret = 1;
Daniel Veillard23e73572002-09-19 19:56:43 +00005272#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillard01992e02002-10-09 10:20:30 +00005273 xmlNodePtr repl = NULL, last = NULL, tmp;
Daniel Veillard23e73572002-09-19 19:56:43 +00005274#endif
Daniel Veillard01992e02002-10-09 10:20:30 +00005275 xmlNodePtr cur;
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005276 xmlElementContentPtr cont;
5277 const xmlChar *name;
5278
Gauravc570b372013-09-30 10:43:47 +08005279 if ((elemDecl == NULL) || (parent == NULL) || (ctxt == NULL))
Daniel Veillardbe480fb2001-11-08 23:36:42 +00005280 return(-1);
5281 cont = elemDecl->content;
5282 name = elemDecl->name;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005283
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005284#ifdef LIBXML_REGEXP_ENABLED
5285 /* Build the regexp associated to the content model */
5286 if (elemDecl->contModel == NULL)
5287 ret = xmlValidBuildContentModel(ctxt, elemDecl);
5288 if (elemDecl->contModel == NULL) {
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005289 return(-1);
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005290 } else {
5291 xmlRegExecCtxtPtr exec;
5292
Daniel Veillardec498e12003-02-05 11:01:50 +00005293 if (!xmlRegexpIsDeterminist(elemDecl->contModel)) {
5294 return(-1);
5295 }
Daniel Veillard01992e02002-10-09 10:20:30 +00005296 ctxt->nodeMax = 0;
5297 ctxt->nodeNr = 0;
5298 ctxt->nodeTab = NULL;
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005299 exec = xmlRegNewExecCtxt(elemDecl->contModel, NULL, NULL);
5300 if (exec != NULL) {
5301 cur = child;
5302 while (cur != NULL) {
5303 switch (cur->type) {
5304 case XML_ENTITY_REF_NODE:
5305 /*
5306 * Push the current node to be able to roll back
5307 * and process within the entity
5308 */
5309 if ((cur->children != NULL) &&
5310 (cur->children->children != NULL)) {
5311 nodeVPush(ctxt, cur);
5312 cur = cur->children->children;
5313 continue;
5314 }
5315 break;
5316 case XML_TEXT_NODE:
5317 if (xmlIsBlankNode(cur))
5318 break;
5319 ret = 0;
5320 goto fail;
5321 case XML_CDATA_SECTION_NODE:
Daniel Veillardea7751d2002-12-20 00:16:24 +00005322 /* TODO */
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005323 ret = 0;
5324 goto fail;
5325 case XML_ELEMENT_NODE:
5326 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00005327 xmlChar fn[50];
5328 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005329
Daniel Veillardc00cda82003-04-07 10:22:39 +00005330 fullname = xmlBuildQName(cur->name,
5331 cur->ns->prefix, fn, 50);
5332 if (fullname == NULL) {
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005333 ret = -1;
5334 goto fail;
5335 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00005336 ret = xmlRegExecPushString(exec, fullname, NULL);
5337 if ((fullname != fn) && (fullname != cur->name))
5338 xmlFree(fullname);
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005339 } else {
5340 ret = xmlRegExecPushString(exec, cur->name, NULL);
5341 }
5342 break;
5343 default:
5344 break;
5345 }
5346 /*
5347 * Switch to next element
5348 */
5349 cur = cur->next;
5350 while (cur == NULL) {
5351 cur = nodeVPop(ctxt);
5352 if (cur == NULL)
5353 break;
5354 cur = cur->next;
5355 }
5356 }
5357 ret = xmlRegExecPushString(exec, NULL, NULL);
5358fail:
5359 xmlRegFreeExecCtxt(exec);
5360 }
5361 }
5362#else /* LIBXML_REGEXP_ENABLED */
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005363 /*
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005364 * Allocate the stack
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005365 */
5366 ctxt->vstateMax = 8;
5367 ctxt->vstateTab = (xmlValidState *) xmlMalloc(
5368 ctxt->vstateMax * sizeof(ctxt->vstateTab[0]));
5369 if (ctxt->vstateTab == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00005370 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005371 return(-1);
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005372 }
5373 /*
5374 * The first entry in the stack is reserved to the current state
5375 */
Daniel Veillarda9142e72001-06-19 11:07:54 +00005376 ctxt->nodeMax = 0;
5377 ctxt->nodeNr = 0;
Daniel Veillard61b33d52001-04-24 13:55:12 +00005378 ctxt->nodeTab = NULL;
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005379 ctxt->vstate = &ctxt->vstateTab[0];
5380 ctxt->vstateNr = 1;
5381 CONT = cont;
5382 NODE = child;
5383 DEPTH = 0;
5384 OCCURS = 0;
5385 STATE = 0;
5386 ret = xmlValidateElementType(ctxt);
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005387 if ((ret == -3) && (warn)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005388 xmlErrValidWarning(ctxt, child, XML_DTD_CONTENT_NOT_DETERMINIST,
5389 "Content model for Element %s is ambiguous\n",
Daniel Veillard0cc72772003-10-13 14:00:21 +00005390 name, NULL, NULL);
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005391 } else if (ret == -2) {
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005392 /*
5393 * An entities reference appeared at this level.
5394 * Buid a minimal representation of this node content
5395 * sufficient to run the validation process on it
5396 */
5397 DEBUG_VALID_MSG("Found an entity reference, linearizing");
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005398 cur = child;
5399 while (cur != NULL) {
5400 switch (cur->type) {
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005401 case XML_ENTITY_REF_NODE:
5402 /*
5403 * Push the current node to be able to roll back
5404 * and process within the entity
5405 */
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005406 if ((cur->children != NULL) &&
5407 (cur->children->children != NULL)) {
5408 nodeVPush(ctxt, cur);
5409 cur = cur->children->children;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005410 continue;
5411 }
Daniel Veillard64b98c02001-06-17 17:20:21 +00005412 break;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005413 case XML_TEXT_NODE:
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005414 if (xmlIsBlankNode(cur))
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005415 break;
Daniel Veillard04e2dae2001-07-09 20:07:25 +00005416 /* no break on purpose */
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00005417 case XML_CDATA_SECTION_NODE:
5418 /* no break on purpose */
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005419 case XML_ELEMENT_NODE:
5420 /*
5421 * Allocate a new node and minimally fills in
5422 * what's required
5423 */
5424 tmp = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
5425 if (tmp == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00005426 xmlVErrMemory(ctxt, "malloc failed");
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005427 xmlFreeNodeList(repl);
5428 ret = -1;
5429 goto done;
5430 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005431 tmp->type = cur->type;
5432 tmp->name = cur->name;
5433 tmp->ns = cur->ns;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005434 tmp->next = NULL;
Daniel Veillarded472f32001-12-13 08:48:14 +00005435 tmp->content = NULL;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005436 if (repl == NULL)
5437 repl = last = tmp;
5438 else {
5439 last->next = tmp;
5440 last = tmp;
5441 }
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00005442 if (cur->type == XML_CDATA_SECTION_NODE) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005443 /*
Daniel Veillardd6dc4cb2002-02-19 14:18:08 +00005444 * E59 spaces in CDATA does not match the
5445 * nonterminal S
5446 */
5447 tmp->content = xmlStrdup(BAD_CAST "CDATA");
5448 }
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005449 break;
5450 default:
5451 break;
5452 }
5453 /*
5454 * Switch to next element
5455 */
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005456 cur = cur->next;
5457 while (cur == NULL) {
5458 cur = nodeVPop(ctxt);
5459 if (cur == NULL)
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005460 break;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005461 cur = cur->next;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005462 }
5463 }
5464
5465 /*
5466 * Relaunch the validation
5467 */
5468 ctxt->vstate = &ctxt->vstateTab[0];
5469 ctxt->vstateNr = 1;
5470 CONT = cont;
5471 NODE = repl;
5472 DEPTH = 0;
5473 OCCURS = 0;
5474 STATE = 0;
5475 ret = xmlValidateElementType(ctxt);
5476 }
Daniel Veillarda646cfd2002-09-17 21:50:03 +00005477#endif /* LIBXML_REGEXP_ENABLED */
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005478 if ((warn) && ((ret != 1) && (ret != -3))) {
Daniel Veillard6ad5c4a2006-10-11 16:43:06 +00005479 if (ctxt != NULL) {
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005480 char expr[5000];
5481 char list[5000];
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005482
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005483 expr[0] = 0;
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005484 xmlSnprintfElementContent(&expr[0], 5000, cont, 1);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005485 list[0] = 0;
Daniel Veillard01992e02002-10-09 10:20:30 +00005486#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005487 if (repl != NULL)
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005488 xmlSnprintfElements(&list[0], 5000, repl, 1);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005489 else
Daniel Veillard01992e02002-10-09 10:20:30 +00005490#endif /* LIBXML_REGEXP_ENABLED */
Daniel Veillarda76fe5c2003-04-24 16:06:47 +00005491 xmlSnprintfElements(&list[0], 5000, child, 1);
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005492
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005493 if (name != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005494 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5495 "Element %s content does not follow the DTD, expecting %s, got %s\n",
5496 name, BAD_CAST expr, BAD_CAST list);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005497 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005498 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5499 "Element content does not follow the DTD, expecting %s, got %s\n",
5500 BAD_CAST expr, BAD_CAST list, NULL);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005501 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005502 } else {
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005503 if (name != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005504 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
Daniel Veillard58e44c92002-08-02 22:19:49 +00005505 "Element %s content does not follow the DTD\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005506 name, NULL, NULL);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005507 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005508 xmlErrValidNode(ctxt, parent, XML_DTD_CONTENT_MODEL,
5509 "Element content does not follow the DTD\n",
5510 NULL, NULL, NULL);
Daniel Veillardb4545fd2001-11-20 09:37:09 +00005511 }
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005512 }
5513 ret = 0;
5514 }
Daniel Veillard4de4d3b2001-05-07 20:50:47 +00005515 if (ret == -3)
5516 ret = 1;
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005517
Daniel Veillard23e73572002-09-19 19:56:43 +00005518#ifndef LIBXML_REGEXP_ENABLED
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005519done:
5520 /*
5521 * Deallocate the copy if done, and free up the validation stack
5522 */
5523 while (repl != NULL) {
5524 tmp = repl->next;
5525 xmlFree(repl);
5526 repl = tmp;
5527 }
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005528 ctxt->vstateMax = 0;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005529 if (ctxt->vstateTab != NULL) {
5530 xmlFree(ctxt->vstateTab);
5531 ctxt->vstateTab = NULL;
5532 }
Daniel Veillard01992e02002-10-09 10:20:30 +00005533#endif
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005534 ctxt->nodeMax = 0;
Daniel Veillarda9142e72001-06-19 11:07:54 +00005535 ctxt->nodeNr = 0;
Daniel Veillard34b1b3a2001-04-21 14:16:10 +00005536 if (ctxt->nodeTab != NULL) {
5537 xmlFree(ctxt->nodeTab);
5538 ctxt->nodeTab = NULL;
5539 }
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005540 return(ret);
Daniel Veillard1c14b8d2001-04-21 10:28:59 +00005541
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005542}
Daniel Veillarddab4cb32001-04-20 13:03:48 +00005543
Owen Taylor3473f882001-02-23 17:55:21 +00005544/**
Daniel Veillard04e2dae2001-07-09 20:07:25 +00005545 * xmlValidateCdataElement:
5546 * @ctxt: the validation context
5547 * @doc: a document instance
5548 * @elem: an element instance
5549 *
5550 * Check that an element follows #CDATA
5551 *
5552 * returns 1 if valid or 0 otherwise
5553 */
5554static int
5555xmlValidateOneCdataElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5556 xmlNodePtr elem) {
5557 int ret = 1;
5558 xmlNodePtr cur, child;
5559
Daniel Veillard3e62adb2012-08-09 14:24:02 +08005560 if ((ctxt == NULL) || (doc == NULL) || (elem == NULL) ||
5561 (elem->type != XML_ELEMENT_NODE))
Daniel Veillard04e2dae2001-07-09 20:07:25 +00005562 return(0);
5563
5564 child = elem->children;
5565
5566 cur = child;
5567 while (cur != NULL) {
5568 switch (cur->type) {
5569 case XML_ENTITY_REF_NODE:
5570 /*
5571 * Push the current node to be able to roll back
5572 * and process within the entity
5573 */
5574 if ((cur->children != NULL) &&
5575 (cur->children->children != NULL)) {
5576 nodeVPush(ctxt, cur);
5577 cur = cur->children->children;
5578 continue;
5579 }
5580 break;
5581 case XML_COMMENT_NODE:
5582 case XML_PI_NODE:
5583 case XML_TEXT_NODE:
5584 case XML_CDATA_SECTION_NODE:
5585 break;
5586 default:
5587 ret = 0;
5588 goto done;
5589 }
5590 /*
5591 * Switch to next element
5592 */
5593 cur = cur->next;
5594 while (cur == NULL) {
5595 cur = nodeVPop(ctxt);
5596 if (cur == NULL)
5597 break;
5598 cur = cur->next;
5599 }
5600 }
5601done:
5602 ctxt->nodeMax = 0;
5603 ctxt->nodeNr = 0;
5604 if (ctxt->nodeTab != NULL) {
5605 xmlFree(ctxt->nodeTab);
5606 ctxt->nodeTab = NULL;
5607 }
5608 return(ret);
5609}
5610
5611/**
Daniel Veillardea7751d2002-12-20 00:16:24 +00005612 * xmlValidateCheckMixed:
5613 * @ctxt: the validation context
5614 * @cont: the mixed content model
5615 * @qname: the qualified name as appearing in the serialization
5616 *
5617 * Check if the given node is part of the content model.
5618 *
5619 * Returns 1 if yes, 0 if no, -1 in case of error
5620 */
5621static int
William M. Brackedb65a72004-02-06 07:36:04 +00005622xmlValidateCheckMixed(xmlValidCtxtPtr ctxt,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005623 xmlElementContentPtr cont, const xmlChar *qname) {
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005624 const xmlChar *name;
5625 int plen;
5626 name = xmlSplitQName3(qname, &plen);
5627
5628 if (name == NULL) {
5629 while (cont != NULL) {
5630 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
5631 if ((cont->prefix == NULL) && (xmlStrEqual(cont->name, qname)))
5632 return(1);
5633 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
5634 (cont->c1 != NULL) &&
5635 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
5636 if ((cont->c1->prefix == NULL) &&
5637 (xmlStrEqual(cont->c1->name, qname)))
5638 return(1);
5639 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
5640 (cont->c1 == NULL) ||
5641 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005642 xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00005643 "Internal: MIXED struct corrupted\n",
5644 NULL);
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005645 break;
5646 }
5647 cont = cont->c2;
Daniel Veillardea7751d2002-12-20 00:16:24 +00005648 }
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005649 } else {
5650 while (cont != NULL) {
5651 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
5652 if ((cont->prefix != NULL) &&
5653 (xmlStrncmp(cont->prefix, qname, plen) == 0) &&
5654 (xmlStrEqual(cont->name, name)))
5655 return(1);
5656 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
5657 (cont->c1 != NULL) &&
5658 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
5659 if ((cont->c1->prefix != NULL) &&
5660 (xmlStrncmp(cont->c1->prefix, qname, plen) == 0) &&
5661 (xmlStrEqual(cont->c1->name, name)))
5662 return(1);
5663 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
5664 (cont->c1 == NULL) ||
5665 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005666 xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00005667 "Internal: MIXED struct corrupted\n",
5668 NULL);
Daniel Veillard8d73bcb2003-08-04 01:06:15 +00005669 break;
5670 }
5671 cont = cont->c2;
5672 }
Daniel Veillardea7751d2002-12-20 00:16:24 +00005673 }
5674 return(0);
5675}
5676
5677/**
5678 * xmlValidGetElemDecl:
5679 * @ctxt: the validation context
5680 * @doc: a document instance
5681 * @elem: an element instance
5682 * @extsubset: pointer, (out) indicate if the declaration was found
5683 * in the external subset.
5684 *
5685 * Finds a declaration associated to an element in the document.
5686 *
5687 * returns the pointer to the declaration or NULL if not found.
5688 */
5689static xmlElementPtr
5690xmlValidGetElemDecl(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5691 xmlNodePtr elem, int *extsubset) {
5692 xmlElementPtr elemDecl = NULL;
5693 const xmlChar *prefix = NULL;
5694
Daniel Veillardf8e3db02012-09-11 13:26:36 +08005695 if ((ctxt == NULL) || (doc == NULL) ||
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005696 (elem == NULL) || (elem->name == NULL))
5697 return(NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005698 if (extsubset != NULL)
5699 *extsubset = 0;
5700
5701 /*
5702 * Fetch the declaration for the qualified name
5703 */
5704 if ((elem->ns != NULL) && (elem->ns->prefix != NULL))
5705 prefix = elem->ns->prefix;
5706
5707 if (prefix != NULL) {
5708 elemDecl = xmlGetDtdQElementDesc(doc->intSubset,
5709 elem->name, prefix);
5710 if ((elemDecl == NULL) && (doc->extSubset != NULL)) {
5711 elemDecl = xmlGetDtdQElementDesc(doc->extSubset,
5712 elem->name, prefix);
5713 if ((elemDecl != NULL) && (extsubset != NULL))
5714 *extsubset = 1;
5715 }
5716 }
5717
5718 /*
5719 * Fetch the declaration for the non qualified name
5720 * This is "non-strict" validation should be done on the
5721 * full QName but in that case being flexible makes sense.
5722 */
5723 if (elemDecl == NULL) {
5724 elemDecl = xmlGetDtdElementDesc(doc->intSubset, elem->name);
5725 if ((elemDecl == NULL) && (doc->extSubset != NULL)) {
5726 elemDecl = xmlGetDtdElementDesc(doc->extSubset, elem->name);
5727 if ((elemDecl != NULL) && (extsubset != NULL))
5728 *extsubset = 1;
5729 }
5730 }
5731 if (elemDecl == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005732 xmlErrValidNode(ctxt, elem,
5733 XML_DTD_UNKNOWN_ELEM,
5734 "No declaration for element %s\n",
5735 elem->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005736 }
5737 return(elemDecl);
5738}
5739
Daniel Veillard0e298ad2003-02-04 16:14:33 +00005740#ifdef LIBXML_REGEXP_ENABLED
Daniel Veillardea7751d2002-12-20 00:16:24 +00005741/**
5742 * xmlValidatePushElement:
5743 * @ctxt: the validation context
5744 * @doc: a document instance
5745 * @elem: an element instance
5746 * @qname: the qualified name as appearing in the serialization
5747 *
5748 * Push a new element start on the validation stack.
5749 *
5750 * returns 1 if no validation problem was found or 0 otherwise
5751 */
5752int
5753xmlValidatePushElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5754 xmlNodePtr elem, const xmlChar *qname) {
5755 int ret = 1;
5756 xmlElementPtr eDecl;
5757 int extsubset = 0;
5758
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005759 if (ctxt == NULL)
5760 return(0);
Daniel Veillardef8dd7b2003-03-23 12:02:56 +00005761/* printf("PushElem %s\n", qname); */
Daniel Veillardea7751d2002-12-20 00:16:24 +00005762 if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5763 xmlValidStatePtr state = ctxt->vstate;
5764 xmlElementPtr elemDecl;
5765
5766 /*
Nick Wellnhofer8bbe4502017-06-17 16:15:09 +02005767 * Check the new element against the content model of the new elem.
Daniel Veillardea7751d2002-12-20 00:16:24 +00005768 */
5769 if (state->elemDecl != NULL) {
5770 elemDecl = state->elemDecl;
5771
5772 switch(elemDecl->etype) {
5773 case XML_ELEMENT_TYPE_UNDEFINED:
5774 ret = 0;
5775 break;
5776 case XML_ELEMENT_TYPE_EMPTY:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005777 xmlErrValidNode(ctxt, state->node,
5778 XML_DTD_NOT_EMPTY,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005779 "Element %s was declared EMPTY this one has content\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005780 state->node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005781 ret = 0;
5782 break;
5783 case XML_ELEMENT_TYPE_ANY:
5784 /* I don't think anything is required then */
5785 break;
5786 case XML_ELEMENT_TYPE_MIXED:
5787 /* simple case of declared as #PCDATA */
5788 if ((elemDecl->content != NULL) &&
5789 (elemDecl->content->type ==
5790 XML_ELEMENT_CONTENT_PCDATA)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005791 xmlErrValidNode(ctxt, state->node,
5792 XML_DTD_NOT_PCDATA,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005793 "Element %s was declared #PCDATA but contains non text nodes\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005794 state->node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005795 ret = 0;
5796 } else {
5797 ret = xmlValidateCheckMixed(ctxt, elemDecl->content,
5798 qname);
5799 if (ret != 1) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005800 xmlErrValidNode(ctxt, state->node,
5801 XML_DTD_INVALID_CHILD,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005802 "Element %s is not declared in %s list of possible children\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005803 qname, state->node->name, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005804 }
5805 }
5806 break;
5807 case XML_ELEMENT_TYPE_ELEMENT:
5808 /*
5809 * TODO:
5810 * VC: Standalone Document Declaration
5811 * - element types with element content, if white space
5812 * occurs directly within any instance of those types.
5813 */
5814 if (state->exec != NULL) {
5815 ret = xmlRegExecPushString(state->exec, qname, NULL);
5816 if (ret < 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005817 xmlErrValidNode(ctxt, state->node,
5818 XML_DTD_CONTENT_MODEL,
5819 "Element %s content does not follow the DTD, Misplaced %s\n",
5820 state->node->name, qname, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005821 ret = 0;
5822 } else {
5823 ret = 1;
5824 }
5825 }
5826 break;
5827 }
5828 }
5829 }
5830 eDecl = xmlValidGetElemDecl(ctxt, doc, elem, &extsubset);
5831 vstateVPush(ctxt, eDecl, elem);
5832 return(ret);
5833}
5834
5835/**
5836 * xmlValidatePushCData:
5837 * @ctxt: the validation context
5838 * @data: some character data read
Michael Woodfb27e2c2012-09-28 08:59:33 +02005839 * @len: the length of the data
Daniel Veillardea7751d2002-12-20 00:16:24 +00005840 *
5841 * check the CData parsed for validation in the current stack
5842 *
5843 * returns 1 if no validation problem was found or 0 otherwise
5844 */
5845int
5846xmlValidatePushCData(xmlValidCtxtPtr ctxt, const xmlChar *data, int len) {
5847 int ret = 1;
5848
Daniel Veillardef8dd7b2003-03-23 12:02:56 +00005849/* printf("CDATA %s %d\n", data, len); */
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005850 if (ctxt == NULL)
5851 return(0);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005852 if (len <= 0)
5853 return(ret);
5854 if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5855 xmlValidStatePtr state = ctxt->vstate;
5856 xmlElementPtr elemDecl;
5857
5858 /*
Nick Wellnhofer8bbe4502017-06-17 16:15:09 +02005859 * Check the new element against the content model of the new elem.
Daniel Veillardea7751d2002-12-20 00:16:24 +00005860 */
5861 if (state->elemDecl != NULL) {
5862 elemDecl = state->elemDecl;
5863
5864 switch(elemDecl->etype) {
5865 case XML_ELEMENT_TYPE_UNDEFINED:
5866 ret = 0;
5867 break;
5868 case XML_ELEMENT_TYPE_EMPTY:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005869 xmlErrValidNode(ctxt, state->node,
5870 XML_DTD_NOT_EMPTY,
Daniel Veillardea7751d2002-12-20 00:16:24 +00005871 "Element %s was declared EMPTY this one has content\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005872 state->node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005873 ret = 0;
5874 break;
5875 case XML_ELEMENT_TYPE_ANY:
5876 break;
5877 case XML_ELEMENT_TYPE_MIXED:
5878 break;
5879 case XML_ELEMENT_TYPE_ELEMENT:
5880 if (len > 0) {
5881 int i;
5882
5883 for (i = 0;i < len;i++) {
William M. Brack76e95df2003-10-18 16:20:14 +00005884 if (!IS_BLANK_CH(data[i])) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005885 xmlErrValidNode(ctxt, state->node,
5886 XML_DTD_CONTENT_MODEL,
5887 "Element %s content does not follow the DTD, Text not allowed\n",
5888 state->node->name, NULL, NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005889 ret = 0;
5890 goto done;
5891 }
5892 }
5893 /*
5894 * TODO:
5895 * VC: Standalone Document Declaration
5896 * element types with element content, if white space
5897 * occurs directly within any instance of those types.
5898 */
5899 }
5900 break;
5901 }
5902 }
5903 }
5904done:
5905 return(ret);
5906}
5907
5908/**
5909 * xmlValidatePopElement:
5910 * @ctxt: the validation context
5911 * @doc: a document instance
5912 * @elem: an element instance
5913 * @qname: the qualified name as appearing in the serialization
5914 *
5915 * Pop the element end from the validation stack.
5916 *
5917 * returns 1 if no validation problem was found or 0 otherwise
5918 */
5919int
5920xmlValidatePopElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc ATTRIBUTE_UNUSED,
Daniel Veillard580ced82003-03-21 21:22:48 +00005921 xmlNodePtr elem ATTRIBUTE_UNUSED,
5922 const xmlChar *qname ATTRIBUTE_UNUSED) {
Daniel Veillardea7751d2002-12-20 00:16:24 +00005923 int ret = 1;
5924
Daniel Veillardc0be74b2004-11-03 19:16:55 +00005925 if (ctxt == NULL)
5926 return(0);
Daniel Veillardef8dd7b2003-03-23 12:02:56 +00005927/* printf("PopElem %s\n", qname); */
Daniel Veillardea7751d2002-12-20 00:16:24 +00005928 if ((ctxt->vstateNr > 0) && (ctxt->vstate != NULL)) {
5929 xmlValidStatePtr state = ctxt->vstate;
5930 xmlElementPtr elemDecl;
5931
5932 /*
Nick Wellnhofer8bbe4502017-06-17 16:15:09 +02005933 * Check the new element against the content model of the new elem.
Daniel Veillardea7751d2002-12-20 00:16:24 +00005934 */
5935 if (state->elemDecl != NULL) {
5936 elemDecl = state->elemDecl;
5937
5938 if (elemDecl->etype == XML_ELEMENT_TYPE_ELEMENT) {
5939 if (state->exec != NULL) {
5940 ret = xmlRegExecPushString(state->exec, NULL, NULL);
5941 if (ret == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005942 xmlErrValidNode(ctxt, state->node,
5943 XML_DTD_CONTENT_MODEL,
5944 "Element %s content does not follow the DTD, Expecting more child\n",
5945 state->node->name, NULL,NULL);
Daniel Veillardea7751d2002-12-20 00:16:24 +00005946 } else {
5947 /*
5948 * previous validation errors should not generate
5949 * a new one here
5950 */
5951 ret = 1;
5952 }
5953 }
5954 }
5955 }
5956 vstateVPop(ctxt);
5957 }
5958 return(ret);
5959}
Daniel Veillard0e298ad2003-02-04 16:14:33 +00005960#endif /* LIBXML_REGEXP_ENABLED */
Daniel Veillardea7751d2002-12-20 00:16:24 +00005961
5962/**
Owen Taylor3473f882001-02-23 17:55:21 +00005963 * xmlValidateOneElement:
5964 * @ctxt: the validation context
5965 * @doc: a document instance
5966 * @elem: an element instance
5967 *
5968 * Try to validate a single element and it's attributes,
5969 * basically it does the following checks as described by the
5970 * XML-1.0 recommendation:
5971 * - [ VC: Element Valid ]
5972 * - [ VC: Required Attribute ]
5973 * Then call xmlValidateOneAttribute() for each attribute present.
5974 *
5975 * The ID/IDREF checkings are done separately
5976 *
5977 * returns 1 if valid or 0 otherwise
5978 */
5979
5980int
5981xmlValidateOneElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc,
5982 xmlNodePtr elem) {
5983 xmlElementPtr elemDecl = NULL;
5984 xmlElementContentPtr cont;
5985 xmlAttributePtr attr;
5986 xmlNodePtr child;
Daniel Veillard8dc16a62002-02-19 21:08:48 +00005987 int ret = 1, tmp;
Owen Taylor3473f882001-02-23 17:55:21 +00005988 const xmlChar *name;
Daniel Veillard8dc16a62002-02-19 21:08:48 +00005989 int extsubset = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00005990
5991 CHECK_DTD;
5992
5993 if (elem == NULL) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00005994 switch (elem->type) {
5995 case XML_ATTRIBUTE_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00005996 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
5997 "Attribute element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00005998 return(0);
5999 case XML_TEXT_NODE:
6000 if (elem->children != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006001 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6002 "Text element has children !\n",
6003 NULL,NULL,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006004 return(0);
6005 }
Owen Taylor3473f882001-02-23 17:55:21 +00006006 if (elem->ns != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006007 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6008 "Text element has namespace !\n",
6009 NULL,NULL,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006010 return(0);
6011 }
Owen Taylor3473f882001-02-23 17:55:21 +00006012 if (elem->content == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006013 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6014 "Text element has no content !\n",
6015 NULL,NULL,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006016 return(0);
6017 }
6018 return(1);
6019 case XML_XINCLUDE_START:
6020 case XML_XINCLUDE_END:
6021 return(1);
6022 case XML_CDATA_SECTION_NODE:
6023 case XML_ENTITY_REF_NODE:
6024 case XML_PI_NODE:
6025 case XML_COMMENT_NODE:
6026 return(1);
6027 case XML_ENTITY_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006028 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6029 "Entity element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006030 return(0);
6031 case XML_NOTATION_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006032 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6033 "Notation element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006034 return(0);
6035 case XML_DOCUMENT_NODE:
6036 case XML_DOCUMENT_TYPE_NODE:
6037 case XML_DOCUMENT_FRAG_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006038 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6039 "Document element not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006040 return(0);
6041 case XML_HTML_DOCUMENT_NODE:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006042 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6043 "HTML Document not expected\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006044 return(0);
6045 case XML_ELEMENT_NODE:
6046 break;
6047 default:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006048 xmlErrValidNode(ctxt, elem, XML_ERR_INTERNAL_ERROR,
6049 "unknown element type\n", NULL, NULL ,NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006050 return(0);
6051 }
Owen Taylor3473f882001-02-23 17:55:21 +00006052
6053 /*
Daniel Veillardea7751d2002-12-20 00:16:24 +00006054 * Fetch the declaration
Owen Taylor3473f882001-02-23 17:55:21 +00006055 */
Daniel Veillardea7751d2002-12-20 00:16:24 +00006056 elemDecl = xmlValidGetElemDecl(ctxt, doc, elem, &extsubset);
6057 if (elemDecl == NULL)
Owen Taylor3473f882001-02-23 17:55:21 +00006058 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006059
Daniel Veillardea7751d2002-12-20 00:16:24 +00006060 /*
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006061 * If vstateNr is not zero that means continuous validation is
Daniel Veillardea7751d2002-12-20 00:16:24 +00006062 * activated, do not try to check the content model at that level.
6063 */
6064 if (ctxt->vstateNr == 0) {
Daniel Veillardcbaf3992001-12-31 16:16:02 +00006065 /* Check that the element content matches the definition */
Owen Taylor3473f882001-02-23 17:55:21 +00006066 switch (elemDecl->etype) {
Daniel Veillarda10efa82001-04-18 13:09:01 +00006067 case XML_ELEMENT_TYPE_UNDEFINED:
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006068 xmlErrValidNode(ctxt, elem, XML_DTD_UNKNOWN_ELEM,
6069 "No declaration for element %s\n",
6070 elem->name, NULL, NULL);
Daniel Veillarda10efa82001-04-18 13:09:01 +00006071 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006072 case XML_ELEMENT_TYPE_EMPTY:
6073 if (elem->children != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006074 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_EMPTY,
Owen Taylor3473f882001-02-23 17:55:21 +00006075 "Element %s was declared EMPTY this one has content\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006076 elem->name, NULL, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006077 ret = 0;
6078 }
6079 break;
6080 case XML_ELEMENT_TYPE_ANY:
6081 /* I don't think anything is required then */
6082 break;
6083 case XML_ELEMENT_TYPE_MIXED:
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006084
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006085 /* simple case of declared as #PCDATA */
6086 if ((elemDecl->content != NULL) &&
6087 (elemDecl->content->type == XML_ELEMENT_CONTENT_PCDATA)) {
6088 ret = xmlValidateOneCdataElement(ctxt, doc, elem);
6089 if (!ret) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006090 xmlErrValidNode(ctxt, elem, XML_DTD_NOT_PCDATA,
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006091 "Element %s was declared #PCDATA but contains non text nodes\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006092 elem->name, NULL, NULL);
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006093 }
6094 break;
6095 }
Owen Taylor3473f882001-02-23 17:55:21 +00006096 child = elem->children;
Daniel Veillard04e2dae2001-07-09 20:07:25 +00006097 /* Hum, this start to get messy */
Owen Taylor3473f882001-02-23 17:55:21 +00006098 while (child != NULL) {
6099 if (child->type == XML_ELEMENT_NODE) {
6100 name = child->name;
6101 if ((child->ns != NULL) && (child->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00006102 xmlChar fn[50];
6103 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006104
Daniel Veillardc00cda82003-04-07 10:22:39 +00006105 fullname = xmlBuildQName(child->name, child->ns->prefix,
6106 fn, 50);
6107 if (fullname == NULL)
6108 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006109 cont = elemDecl->content;
6110 while (cont != NULL) {
6111 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00006112 if (xmlStrEqual(cont->name, fullname))
6113 break;
Owen Taylor3473f882001-02-23 17:55:21 +00006114 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
6115 (cont->c1 != NULL) &&
6116 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)){
Daniel Veillardc00cda82003-04-07 10:22:39 +00006117 if (xmlStrEqual(cont->c1->name, fullname))
6118 break;
Owen Taylor3473f882001-02-23 17:55:21 +00006119 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
6120 (cont->c1 == NULL) ||
6121 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)){
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006122 xmlErrValid(NULL, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00006123 "Internal: MIXED struct corrupted\n",
6124 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006125 break;
6126 }
6127 cont = cont->c2;
6128 }
Daniel Veillardc00cda82003-04-07 10:22:39 +00006129 if ((fullname != fn) && (fullname != child->name))
6130 xmlFree(fullname);
Owen Taylor3473f882001-02-23 17:55:21 +00006131 if (cont != NULL)
6132 goto child_ok;
6133 }
6134 cont = elemDecl->content;
6135 while (cont != NULL) {
6136 if (cont->type == XML_ELEMENT_CONTENT_ELEMENT) {
6137 if (xmlStrEqual(cont->name, name)) break;
6138 } else if ((cont->type == XML_ELEMENT_CONTENT_OR) &&
6139 (cont->c1 != NULL) &&
6140 (cont->c1->type == XML_ELEMENT_CONTENT_ELEMENT)) {
6141 if (xmlStrEqual(cont->c1->name, name)) break;
6142 } else if ((cont->type != XML_ELEMENT_CONTENT_OR) ||
6143 (cont->c1 == NULL) ||
6144 (cont->c1->type != XML_ELEMENT_CONTENT_PCDATA)) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006145 xmlErrValid(ctxt, XML_DTD_MIXED_CORRUPT,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00006146 "Internal: MIXED struct corrupted\n",
6147 NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006148 break;
6149 }
6150 cont = cont->c2;
6151 }
6152 if (cont == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006153 xmlErrValidNode(ctxt, elem, XML_DTD_INVALID_CHILD,
Daniel Veillard5e5c2d02002-02-09 18:03:01 +00006154 "Element %s is not declared in %s list of possible children\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006155 name, elem->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006156 ret = 0;
6157 }
6158 }
6159child_ok:
6160 child = child->next;
6161 }
6162 break;
6163 case XML_ELEMENT_TYPE_ELEMENT:
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006164 if ((doc->standalone == 1) && (extsubset == 1)) {
6165 /*
6166 * VC: Standalone Document Declaration
6167 * - element types with element content, if white space
6168 * occurs directly within any instance of those types.
6169 */
6170 child = elem->children;
6171 while (child != NULL) {
6172 if (child->type == XML_TEXT_NODE) {
6173 const xmlChar *content = child->content;
6174
William M. Brack76e95df2003-10-18 16:20:14 +00006175 while (IS_BLANK_CH(*content))
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006176 content++;
6177 if (*content == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006178 xmlErrValidNode(ctxt, elem,
6179 XML_DTD_STANDALONE_WHITE_SPACE,
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006180"standalone: %s declared in the external subset contains white spaces nodes\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006181 elem->name, NULL, NULL);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006182 ret = 0;
6183 break;
6184 }
6185 }
6186 child =child->next;
6187 }
6188 }
Owen Taylor3473f882001-02-23 17:55:21 +00006189 child = elem->children;
6190 cont = elemDecl->content;
Daniel Veillardb9cd8b42002-09-05 10:58:49 +00006191 tmp = xmlValidateElementContent(ctxt, child, elemDecl, 1, elem);
Daniel Veillard8dc16a62002-02-19 21:08:48 +00006192 if (tmp <= 0)
6193 ret = tmp;
Owen Taylor3473f882001-02-23 17:55:21 +00006194 break;
6195 }
Daniel Veillardea7751d2002-12-20 00:16:24 +00006196 } /* not continuous */
Owen Taylor3473f882001-02-23 17:55:21 +00006197
6198 /* [ VC: Required Attribute ] */
6199 attr = elemDecl->attributes;
6200 while (attr != NULL) {
6201 if (attr->def == XML_ATTRIBUTE_REQUIRED) {
Owen Taylor3473f882001-02-23 17:55:21 +00006202 int qualified = -1;
Owen Taylor3473f882001-02-23 17:55:21 +00006203
Daniel Veillarde4301c82002-02-13 13:32:35 +00006204 if ((attr->prefix == NULL) &&
6205 (xmlStrEqual(attr->name, BAD_CAST "xmlns"))) {
6206 xmlNsPtr ns;
6207
6208 ns = elem->nsDef;
6209 while (ns != NULL) {
6210 if (ns->prefix == NULL)
Owen Taylor3473f882001-02-23 17:55:21 +00006211 goto found;
Daniel Veillarde4301c82002-02-13 13:32:35 +00006212 ns = ns->next;
Owen Taylor3473f882001-02-23 17:55:21 +00006213 }
Daniel Veillarde4301c82002-02-13 13:32:35 +00006214 } else if (xmlStrEqual(attr->prefix, BAD_CAST "xmlns")) {
6215 xmlNsPtr ns;
6216
6217 ns = elem->nsDef;
6218 while (ns != NULL) {
6219 if (xmlStrEqual(attr->name, ns->prefix))
6220 goto found;
6221 ns = ns->next;
6222 }
6223 } else {
6224 xmlAttrPtr attrib;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006225
Daniel Veillarde4301c82002-02-13 13:32:35 +00006226 attrib = elem->properties;
6227 while (attrib != NULL) {
6228 if (xmlStrEqual(attrib->name, attr->name)) {
6229 if (attr->prefix != NULL) {
6230 xmlNsPtr nameSpace = attrib->ns;
6231
6232 if (nameSpace == NULL)
6233 nameSpace = elem->ns;
6234 /*
6235 * qualified names handling is problematic, having a
6236 * different prefix should be possible but DTDs don't
6237 * allow to define the URI instead of the prefix :-(
6238 */
6239 if (nameSpace == NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006240 if (qualified < 0)
Daniel Veillarde4301c82002-02-13 13:32:35 +00006241 qualified = 0;
6242 } else if (!xmlStrEqual(nameSpace->prefix,
6243 attr->prefix)) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006244 if (qualified < 1)
Daniel Veillarde4301c82002-02-13 13:32:35 +00006245 qualified = 1;
6246 } else
6247 goto found;
6248 } else {
6249 /*
6250 * We should allow applications to define namespaces
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006251 * for their application even if the DTD doesn't
Daniel Veillarde4301c82002-02-13 13:32:35 +00006252 * carry one, otherwise, basically we would always
6253 * break.
6254 */
6255 goto found;
6256 }
6257 }
6258 attrib = attrib->next;
6259 }
Owen Taylor3473f882001-02-23 17:55:21 +00006260 }
6261 if (qualified == -1) {
6262 if (attr->prefix == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006263 xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006264 "Element %s does not carry attribute %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006265 elem->name, attr->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006266 ret = 0;
6267 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006268 xmlErrValidNode(ctxt, elem, XML_DTD_MISSING_ATTRIBUTE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006269 "Element %s does not carry attribute %s:%s\n",
Owen Taylor3473f882001-02-23 17:55:21 +00006270 elem->name, attr->prefix,attr->name);
6271 ret = 0;
6272 }
6273 } else if (qualified == 0) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006274 xmlErrValidWarning(ctxt, elem, XML_DTD_NO_PREFIX,
Owen Taylor3473f882001-02-23 17:55:21 +00006275 "Element %s required attribute %s:%s has no prefix\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006276 elem->name, attr->prefix, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00006277 } else if (qualified == 1) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006278 xmlErrValidWarning(ctxt, elem, XML_DTD_DIFFERENT_PREFIX,
Owen Taylor3473f882001-02-23 17:55:21 +00006279 "Element %s required attribute %s:%s has different prefix\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006280 elem->name, attr->prefix, attr->name);
Owen Taylor3473f882001-02-23 17:55:21 +00006281 }
Daniel Veillarde4301c82002-02-13 13:32:35 +00006282 } else if (attr->def == XML_ATTRIBUTE_FIXED) {
6283 /*
6284 * Special tests checking #FIXED namespace declarations
6285 * have the right value since this is not done as an
6286 * attribute checking
6287 */
6288 if ((attr->prefix == NULL) &&
6289 (xmlStrEqual(attr->name, BAD_CAST "xmlns"))) {
6290 xmlNsPtr ns;
6291
6292 ns = elem->nsDef;
6293 while (ns != NULL) {
6294 if (ns->prefix == NULL) {
6295 if (!xmlStrEqual(attr->defaultValue, ns->href)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006296 xmlErrValidNode(ctxt, elem,
6297 XML_DTD_ELEM_DEFAULT_NAMESPACE,
Daniel Veillarde4301c82002-02-13 13:32:35 +00006298 "Element %s namespace name for default namespace does not match the DTD\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006299 elem->name, NULL, NULL);
Daniel Veillardc7612992002-02-17 22:47:37 +00006300 ret = 0;
Daniel Veillarde4301c82002-02-13 13:32:35 +00006301 }
6302 goto found;
6303 }
6304 ns = ns->next;
6305 }
6306 } else if (xmlStrEqual(attr->prefix, BAD_CAST "xmlns")) {
6307 xmlNsPtr ns;
6308
6309 ns = elem->nsDef;
6310 while (ns != NULL) {
6311 if (xmlStrEqual(attr->name, ns->prefix)) {
6312 if (!xmlStrEqual(attr->defaultValue, ns->href)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006313 xmlErrValidNode(ctxt, elem, XML_DTD_ELEM_NAMESPACE,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006314 "Element %s namespace name for %s does not match the DTD\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006315 elem->name, ns->prefix, NULL);
Daniel Veillardc7612992002-02-17 22:47:37 +00006316 ret = 0;
Daniel Veillarde4301c82002-02-13 13:32:35 +00006317 }
6318 goto found;
6319 }
6320 ns = ns->next;
6321 }
6322 }
Owen Taylor3473f882001-02-23 17:55:21 +00006323 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006324found:
Owen Taylor3473f882001-02-23 17:55:21 +00006325 attr = attr->nexth;
6326 }
6327 return(ret);
6328}
6329
6330/**
6331 * xmlValidateRoot:
6332 * @ctxt: the validation context
6333 * @doc: a document instance
6334 *
6335 * Try to validate a the root element
6336 * basically it does the following check as described by the
6337 * XML-1.0 recommendation:
6338 * - [ VC: Root Element Type ]
6339 * it doesn't try to recurse or apply other check to the element
6340 *
6341 * returns 1 if valid or 0 otherwise
6342 */
6343
6344int
6345xmlValidateRoot(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6346 xmlNodePtr root;
Daniel Veillardc00cda82003-04-07 10:22:39 +00006347 int ret;
6348
Owen Taylor3473f882001-02-23 17:55:21 +00006349 if (doc == NULL) return(0);
6350
6351 root = xmlDocGetRootElement(doc);
6352 if ((root == NULL) || (root->name == NULL)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006353 xmlErrValid(ctxt, XML_DTD_NO_ROOT,
6354 "no root element\n", NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006355 return(0);
6356 }
6357
6358 /*
6359 * When doing post validation against a separate DTD, those may
6360 * no internal subset has been generated
6361 */
6362 if ((doc->intSubset != NULL) &&
6363 (doc->intSubset->name != NULL)) {
6364 /*
6365 * Check first the document root against the NQName
6366 */
6367 if (!xmlStrEqual(doc->intSubset->name, root->name)) {
6368 if ((root->ns != NULL) && (root->ns->prefix != NULL)) {
Daniel Veillardc00cda82003-04-07 10:22:39 +00006369 xmlChar fn[50];
6370 xmlChar *fullname;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006371
Daniel Veillardc00cda82003-04-07 10:22:39 +00006372 fullname = xmlBuildQName(root->name, root->ns->prefix, fn, 50);
6373 if (fullname == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00006374 xmlVErrMemory(ctxt, NULL);
Daniel Veillardc00cda82003-04-07 10:22:39 +00006375 return(0);
6376 }
6377 ret = xmlStrEqual(doc->intSubset->name, fullname);
6378 if ((fullname != fn) && (fullname != root->name))
6379 xmlFree(fullname);
6380 if (ret == 1)
Owen Taylor3473f882001-02-23 17:55:21 +00006381 goto name_ok;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006382 }
Owen Taylor3473f882001-02-23 17:55:21 +00006383 if ((xmlStrEqual(doc->intSubset->name, BAD_CAST "HTML")) &&
6384 (xmlStrEqual(root->name, BAD_CAST "html")))
6385 goto name_ok;
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006386 xmlErrValidNode(ctxt, root, XML_DTD_ROOT_NAME,
6387 "root and DTD name do not match '%s' and '%s'\n",
6388 root->name, doc->intSubset->name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006389 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006390 }
6391 }
6392name_ok:
6393 return(1);
6394}
6395
6396
6397/**
6398 * xmlValidateElement:
6399 * @ctxt: the validation context
6400 * @doc: a document instance
6401 * @elem: an element instance
6402 *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006403 * Try to validate the subtree under an element
Owen Taylor3473f882001-02-23 17:55:21 +00006404 *
6405 * returns 1 if valid or 0 otherwise
6406 */
6407
6408int
6409xmlValidateElement(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlNodePtr elem) {
6410 xmlNodePtr child;
6411 xmlAttrPtr attr;
Daniel Veillarde133dd82003-10-30 10:42:20 +00006412 xmlNsPtr ns;
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006413 const xmlChar *value;
Owen Taylor3473f882001-02-23 17:55:21 +00006414 int ret = 1;
6415
6416 if (elem == NULL) return(0);
6417
6418 /*
6419 * XInclude elements were added after parsing in the infoset,
6420 * they don't really mean anything validation wise.
6421 */
6422 if ((elem->type == XML_XINCLUDE_START) ||
Daniel Veillard3e62adb2012-08-09 14:24:02 +08006423 (elem->type == XML_XINCLUDE_END) ||
6424 (elem->type == XML_NAMESPACE_DECL))
Owen Taylor3473f882001-02-23 17:55:21 +00006425 return(1);
6426
6427 CHECK_DTD;
6428
Daniel Veillard10ea86c2001-06-20 13:55:33 +00006429 /*
6430 * Entities references have to be handled separately
6431 */
6432 if (elem->type == XML_ENTITY_REF_NODE) {
6433 return(1);
6434 }
6435
Owen Taylor3473f882001-02-23 17:55:21 +00006436 ret &= xmlValidateOneElement(ctxt, doc, elem);
Daniel Veillard8874b942005-08-25 13:19:21 +00006437 if (elem->type == XML_ELEMENT_NODE) {
6438 attr = elem->properties;
6439 while (attr != NULL) {
6440 value = xmlNodeListGetString(doc, attr->children, 0);
6441 ret &= xmlValidateOneAttribute(ctxt, doc, elem, attr, value);
6442 if (value != NULL)
6443 xmlFree((char *)value);
6444 attr= attr->next;
6445 }
6446 ns = elem->nsDef;
6447 while (ns != NULL) {
6448 if (elem->ns == NULL)
6449 ret &= xmlValidateOneNamespace(ctxt, doc, elem, NULL,
6450 ns, ns->href);
6451 else
6452 ret &= xmlValidateOneNamespace(ctxt, doc, elem,
6453 elem->ns->prefix, ns, ns->href);
6454 ns = ns->next;
6455 }
Daniel Veillarde133dd82003-10-30 10:42:20 +00006456 }
Owen Taylor3473f882001-02-23 17:55:21 +00006457 child = elem->children;
6458 while (child != NULL) {
6459 ret &= xmlValidateElement(ctxt, doc, child);
6460 child = child->next;
6461 }
6462
6463 return(ret);
6464}
6465
Daniel Veillard8730c562001-02-26 10:49:57 +00006466/**
6467 * xmlValidateRef:
6468 * @ref: A reference to be validated
6469 * @ctxt: Validation context
6470 * @name: Name of ID we are searching for
6471 *
6472 */
Daniel Veillard56a4cb82001-03-24 17:00:36 +00006473static void
Daniel Veillard8730c562001-02-26 10:49:57 +00006474xmlValidateRef(xmlRefPtr ref, xmlValidCtxtPtr ctxt,
Owen Taylor3473f882001-02-23 17:55:21 +00006475 const xmlChar *name) {
6476 xmlAttrPtr id;
6477 xmlAttrPtr attr;
6478
6479 if (ref == NULL)
6480 return;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006481 if ((ref->attr == NULL) && (ref->name == NULL))
Owen Taylor3473f882001-02-23 17:55:21 +00006482 return;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006483 attr = ref->attr;
6484 if (attr == NULL) {
6485 xmlChar *dup, *str = NULL, *cur, save;
6486
6487 dup = xmlStrdup(name);
6488 if (dup == NULL) {
6489 ctxt->valid = 0;
6490 return;
6491 }
6492 cur = dup;
6493 while (*cur != 0) {
6494 str = cur;
William M. Brack76e95df2003-10-18 16:20:14 +00006495 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006496 save = *cur;
6497 *cur = 0;
6498 id = xmlGetID(ctxt->doc, str);
6499 if (id == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006500 xmlErrValidNodeNr(ctxt, NULL, XML_DTD_UNKNOWN_ID,
Daniel Veillardea7751d2002-12-20 00:16:24 +00006501 "attribute %s line %d references an unknown ID \"%s\"\n",
6502 ref->name, ref->lineno, str);
6503 ctxt->valid = 0;
6504 }
6505 if (save == 0)
6506 break;
6507 *cur = save;
William M. Brack76e95df2003-10-18 16:20:14 +00006508 while (IS_BLANK_CH(*cur)) cur++;
Daniel Veillardea7751d2002-12-20 00:16:24 +00006509 }
6510 xmlFree(dup);
6511 } else if (attr->atype == XML_ATTRIBUTE_IDREF) {
Owen Taylor3473f882001-02-23 17:55:21 +00006512 id = xmlGetID(ctxt->doc, name);
6513 if (id == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006514 xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
Daniel Veillardea7751d2002-12-20 00:16:24 +00006515 "IDREF attribute %s references an unknown ID \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006516 attr->name, name, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006517 ctxt->valid = 0;
6518 }
6519 } else if (attr->atype == XML_ATTRIBUTE_IDREFS) {
6520 xmlChar *dup, *str = NULL, *cur, save;
6521
6522 dup = xmlStrdup(name);
6523 if (dup == NULL) {
Daniel Veillardce9457f2003-10-05 21:33:18 +00006524 xmlVErrMemory(ctxt, "IDREFS split");
Owen Taylor3473f882001-02-23 17:55:21 +00006525 ctxt->valid = 0;
6526 return;
6527 }
6528 cur = dup;
6529 while (*cur != 0) {
6530 str = cur;
William M. Brack76e95df2003-10-18 16:20:14 +00006531 while ((*cur != 0) && (!IS_BLANK_CH(*cur))) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00006532 save = *cur;
6533 *cur = 0;
6534 id = xmlGetID(ctxt->doc, str);
6535 if (id == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006536 xmlErrValidNode(ctxt, attr->parent, XML_DTD_UNKNOWN_ID,
Daniel Veillardea7751d2002-12-20 00:16:24 +00006537 "IDREFS attribute %s references an unknown ID \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006538 attr->name, str, NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006539 ctxt->valid = 0;
6540 }
6541 if (save == 0)
6542 break;
6543 *cur = save;
William M. Brack76e95df2003-10-18 16:20:14 +00006544 while (IS_BLANK_CH(*cur)) cur++;
Owen Taylor3473f882001-02-23 17:55:21 +00006545 }
6546 xmlFree(dup);
6547 }
6548}
6549
6550/**
Daniel Veillard8730c562001-02-26 10:49:57 +00006551 * xmlWalkValidateList:
6552 * @data: Contents of current link
6553 * @user: Value supplied by the user
6554 *
Daniel Veillardcbaf3992001-12-31 16:16:02 +00006555 * Returns 0 to abort the walk or 1 to continue
Daniel Veillard8730c562001-02-26 10:49:57 +00006556 */
6557static int
Nick Wellnhofer4dd6d7a2017-11-09 17:28:00 +01006558xmlWalkValidateList(const void *data, void *user)
Daniel Veillard8730c562001-02-26 10:49:57 +00006559{
6560 xmlValidateMemoPtr memo = (xmlValidateMemoPtr)user;
6561 xmlValidateRef((xmlRefPtr)data, memo->ctxt, memo->name);
6562 return 1;
6563}
6564
6565/**
6566 * xmlValidateCheckRefCallback:
6567 * @ref_list: List of references
6568 * @ctxt: Validation context
6569 * @name: Name of ID we are searching for
6570 *
6571 */
6572static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006573xmlValidateCheckRefCallback(void *payload, void *data, const xmlChar *name) {
6574 xmlListPtr ref_list = (xmlListPtr) payload;
6575 xmlValidCtxtPtr ctxt = (xmlValidCtxtPtr) data;
Daniel Veillard8730c562001-02-26 10:49:57 +00006576 xmlValidateMemo memo;
6577
6578 if (ref_list == NULL)
6579 return;
6580 memo.ctxt = ctxt;
6581 memo.name = name;
6582
6583 xmlListWalk(ref_list, xmlWalkValidateList, &memo);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006584
Daniel Veillard8730c562001-02-26 10:49:57 +00006585}
6586
6587/**
Owen Taylor3473f882001-02-23 17:55:21 +00006588 * xmlValidateDocumentFinal:
6589 * @ctxt: the validation context
6590 * @doc: a document instance
6591 *
6592 * Does the final step for the document validation once all the
6593 * incremental validation steps have been completed
6594 *
6595 * basically it does the following checks described by the XML Rec
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006596 *
Daniel Veillardc3da18a2003-03-18 00:31:04 +00006597 * Check all the IDREF/IDREFS attributes definition for validity
Owen Taylor3473f882001-02-23 17:55:21 +00006598 *
6599 * returns 1 if valid or 0 otherwise
6600 */
6601
6602int
6603xmlValidateDocumentFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6604 xmlRefTablePtr table;
Daniel Veillardcb3549e2011-11-11 11:25:07 +08006605 unsigned int save;
Owen Taylor3473f882001-02-23 17:55:21 +00006606
Daniel Veillardc0be74b2004-11-03 19:16:55 +00006607 if (ctxt == NULL)
6608 return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006609 if (doc == NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006610 xmlErrValid(ctxt, XML_DTD_NO_DOC,
Daniel Veillard2b8c4a12003-10-02 22:28:19 +00006611 "xmlValidateDocumentFinal: doc == NULL\n", NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006612 return(0);
6613 }
6614
Daniel Veillardcb3549e2011-11-11 11:25:07 +08006615 /* trick to get correct line id report */
6616 save = ctxt->finishDtd;
6617 ctxt->finishDtd = 0;
6618
Owen Taylor3473f882001-02-23 17:55:21 +00006619 /*
6620 * Check all the NOTATION/NOTATIONS attributes
6621 */
6622 /*
6623 * Check all the ENTITY/ENTITIES attributes definition for validity
6624 */
6625 /*
6626 * Check all the IDREF/IDREFS attributes definition for validity
6627 */
6628 table = (xmlRefTablePtr) doc->refs;
6629 ctxt->doc = doc;
6630 ctxt->valid = 1;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006631 xmlHashScan(table, xmlValidateCheckRefCallback, ctxt);
Daniel Veillardcb3549e2011-11-11 11:25:07 +08006632
6633 ctxt->finishDtd = save;
Owen Taylor3473f882001-02-23 17:55:21 +00006634 return(ctxt->valid);
6635}
6636
6637/**
6638 * xmlValidateDtd:
6639 * @ctxt: the validation context
6640 * @doc: a document instance
6641 * @dtd: a dtd instance
6642 *
6643 * Try to validate the document against the dtd instance
6644 *
William M. Brack367df6e2004-10-23 18:14:36 +00006645 * Basically it does check all the definitions in the DtD.
6646 * Note the the internal subset (if present) is de-coupled
6647 * (i.e. not used), which could give problems if ID or IDREF
6648 * is present.
Owen Taylor3473f882001-02-23 17:55:21 +00006649 *
6650 * returns 1 if valid or 0 otherwise
6651 */
6652
6653int
6654xmlValidateDtd(xmlValidCtxtPtr ctxt, xmlDocPtr doc, xmlDtdPtr dtd) {
6655 int ret;
William M. Brack367df6e2004-10-23 18:14:36 +00006656 xmlDtdPtr oldExt, oldInt;
Owen Taylor3473f882001-02-23 17:55:21 +00006657 xmlNodePtr root;
6658
6659 if (dtd == NULL) return(0);
6660 if (doc == NULL) return(0);
6661 oldExt = doc->extSubset;
William M. Brack367df6e2004-10-23 18:14:36 +00006662 oldInt = doc->intSubset;
Owen Taylor3473f882001-02-23 17:55:21 +00006663 doc->extSubset = dtd;
William M. Brack367df6e2004-10-23 18:14:36 +00006664 doc->intSubset = NULL;
Owen Taylor3473f882001-02-23 17:55:21 +00006665 ret = xmlValidateRoot(ctxt, doc);
6666 if (ret == 0) {
6667 doc->extSubset = oldExt;
William M. Brack367df6e2004-10-23 18:14:36 +00006668 doc->intSubset = oldInt;
Owen Taylor3473f882001-02-23 17:55:21 +00006669 return(ret);
6670 }
6671 if (doc->ids != NULL) {
6672 xmlFreeIDTable(doc->ids);
6673 doc->ids = NULL;
6674 }
6675 if (doc->refs != NULL) {
6676 xmlFreeRefTable(doc->refs);
6677 doc->refs = NULL;
6678 }
6679 root = xmlDocGetRootElement(doc);
6680 ret = xmlValidateElement(ctxt, doc, root);
6681 ret &= xmlValidateDocumentFinal(ctxt, doc);
6682 doc->extSubset = oldExt;
William M. Brack367df6e2004-10-23 18:14:36 +00006683 doc->intSubset = oldInt;
Owen Taylor3473f882001-02-23 17:55:21 +00006684 return(ret);
6685}
6686
Daniel Veillard56a4cb82001-03-24 17:00:36 +00006687static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006688xmlValidateNotationCallback(void *payload, void *data,
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006689 const xmlChar *name ATTRIBUTE_UNUSED) {
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006690 xmlEntityPtr cur = (xmlEntityPtr) payload;
6691 xmlValidCtxtPtr ctxt = (xmlValidCtxtPtr) data;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006692 if (cur == NULL)
6693 return;
6694 if (cur->etype == XML_EXTERNAL_GENERAL_UNPARSED_ENTITY) {
6695 xmlChar *notation = cur->content;
6696
Daniel Veillard878eab02002-02-19 13:46:09 +00006697 if (notation != NULL) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006698 int ret;
6699
6700 ret = xmlValidateNotationUse(ctxt, cur->doc, notation);
6701 if (ret != 1) {
Daniel Veillard878eab02002-02-19 13:46:09 +00006702 ctxt->valid = 0;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006703 }
6704 }
6705 }
6706}
6707
6708static void
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006709xmlValidateAttributeCallback(void *payload, void *data,
6710 const xmlChar *name ATTRIBUTE_UNUSED) {
6711 xmlAttributePtr cur = (xmlAttributePtr) payload;
6712 xmlValidCtxtPtr ctxt = (xmlValidCtxtPtr) data;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006713 int ret;
Daniel Veillard878eab02002-02-19 13:46:09 +00006714 xmlDocPtr doc;
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006715 xmlElementPtr elem = NULL;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006716
Owen Taylor3473f882001-02-23 17:55:21 +00006717 if (cur == NULL)
6718 return;
6719 switch (cur->atype) {
6720 case XML_ATTRIBUTE_CDATA:
6721 case XML_ATTRIBUTE_ID:
6722 case XML_ATTRIBUTE_IDREF :
6723 case XML_ATTRIBUTE_IDREFS:
6724 case XML_ATTRIBUTE_NMTOKEN:
6725 case XML_ATTRIBUTE_NMTOKENS:
6726 case XML_ATTRIBUTE_ENUMERATION:
6727 break;
6728 case XML_ATTRIBUTE_ENTITY:
6729 case XML_ATTRIBUTE_ENTITIES:
6730 case XML_ATTRIBUTE_NOTATION:
6731 if (cur->defaultValue != NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006732
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006733 ret = xmlValidateAttributeValue2(ctxt, ctxt->doc, cur->name,
6734 cur->atype, cur->defaultValue);
6735 if ((ret == 0) && (ctxt->valid == 1))
6736 ctxt->valid = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00006737 }
6738 if (cur->tree != NULL) {
6739 xmlEnumerationPtr tree = cur->tree;
6740 while (tree != NULL) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006741 ret = xmlValidateAttributeValue2(ctxt, ctxt->doc,
Owen Taylor3473f882001-02-23 17:55:21 +00006742 cur->name, cur->atype, tree->name);
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006743 if ((ret == 0) && (ctxt->valid == 1))
6744 ctxt->valid = 0;
Owen Taylor3473f882001-02-23 17:55:21 +00006745 tree = tree->next;
6746 }
6747 }
6748 }
Daniel Veillard878eab02002-02-19 13:46:09 +00006749 if (cur->atype == XML_ATTRIBUTE_NOTATION) {
6750 doc = cur->doc;
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006751 if (cur->elem == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006752 xmlErrValid(ctxt, XML_ERR_INTERNAL_ERROR,
Daniel Veillard878eab02002-02-19 13:46:09 +00006753 "xmlValidateAttributeCallback(%s): internal error\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006754 (const char *) cur->name);
Daniel Veillard878eab02002-02-19 13:46:09 +00006755 return;
6756 }
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006757
6758 if (doc != NULL)
6759 elem = xmlGetDtdElementDesc(doc->intSubset, cur->elem);
6760 if ((elem == NULL) && (doc != NULL))
Daniel Veillard878eab02002-02-19 13:46:09 +00006761 elem = xmlGetDtdElementDesc(doc->extSubset, cur->elem);
Daniel Veillarda8ff65d2003-11-03 16:20:10 +00006762 if ((elem == NULL) && (cur->parent != NULL) &&
6763 (cur->parent->type == XML_DTD_NODE))
6764 elem = xmlGetDtdElementDesc((xmlDtdPtr) cur->parent, cur->elem);
Daniel Veillard878eab02002-02-19 13:46:09 +00006765 if (elem == NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006766 xmlErrValidNode(ctxt, NULL, XML_DTD_UNKNOWN_ELEM,
Daniel Veillard878eab02002-02-19 13:46:09 +00006767 "attribute %s: could not find decl for element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006768 cur->name, cur->elem, NULL);
Daniel Veillard878eab02002-02-19 13:46:09 +00006769 return;
6770 }
6771 if (elem->etype == XML_ELEMENT_TYPE_EMPTY) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006772 xmlErrValidNode(ctxt, NULL, XML_DTD_EMPTY_NOTATION,
Daniel Veillard58e44c92002-08-02 22:19:49 +00006773 "NOTATION attribute %s declared for EMPTY element %s\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006774 cur->name, cur->elem, NULL);
Daniel Veillard878eab02002-02-19 13:46:09 +00006775 ctxt->valid = 0;
6776 }
6777 }
Owen Taylor3473f882001-02-23 17:55:21 +00006778}
6779
6780/**
6781 * xmlValidateDtdFinal:
6782 * @ctxt: the validation context
6783 * @doc: a document instance
6784 *
6785 * Does the final step for the dtds validation once all the
6786 * subsets have been parsed
6787 *
6788 * basically it does the following checks described by the XML Rec
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006789 * - check that ENTITY and ENTITIES type attributes default or
Owen Taylor3473f882001-02-23 17:55:21 +00006790 * possible values matches one of the defined entities.
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006791 * - check that NOTATION type attributes default or
Owen Taylor3473f882001-02-23 17:55:21 +00006792 * possible values matches one of the defined notations.
6793 *
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006794 * returns 1 if valid or 0 if invalid and -1 if not well-formed
Owen Taylor3473f882001-02-23 17:55:21 +00006795 */
6796
6797int
6798xmlValidateDtdFinal(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
Owen Taylor3473f882001-02-23 17:55:21 +00006799 xmlDtdPtr dtd;
6800 xmlAttributeTablePtr table;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006801 xmlEntitiesTablePtr entities;
Owen Taylor3473f882001-02-23 17:55:21 +00006802
Daniel Veillard2cba4152008-08-27 11:45:41 +00006803 if ((doc == NULL) || (ctxt == NULL)) return(0);
Owen Taylor3473f882001-02-23 17:55:21 +00006804 if ((doc->intSubset == NULL) && (doc->extSubset == NULL))
6805 return(0);
6806 ctxt->doc = doc;
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006807 ctxt->valid = 1;
Owen Taylor3473f882001-02-23 17:55:21 +00006808 dtd = doc->intSubset;
6809 if ((dtd != NULL) && (dtd->attributes != NULL)) {
6810 table = (xmlAttributeTablePtr) dtd->attributes;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006811 xmlHashScan(table, xmlValidateAttributeCallback, ctxt);
Daniel Veillard878eab02002-02-19 13:46:09 +00006812 }
6813 if ((dtd != NULL) && (dtd->entities != NULL)) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006814 entities = (xmlEntitiesTablePtr) dtd->entities;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006815 xmlHashScan(entities, xmlValidateNotationCallback, ctxt);
Owen Taylor3473f882001-02-23 17:55:21 +00006816 }
6817 dtd = doc->extSubset;
6818 if ((dtd != NULL) && (dtd->attributes != NULL)) {
6819 table = (xmlAttributeTablePtr) dtd->attributes;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006820 xmlHashScan(table, xmlValidateAttributeCallback, ctxt);
Daniel Veillard878eab02002-02-19 13:46:09 +00006821 }
6822 if ((dtd != NULL) && (dtd->entities != NULL)) {
Daniel Veillard8ab0f582002-02-18 18:31:38 +00006823 entities = (xmlEntitiesTablePtr) dtd->entities;
Nick Wellnhofere03f0a12017-11-09 16:42:47 +01006824 xmlHashScan(entities, xmlValidateNotationCallback, ctxt);
Owen Taylor3473f882001-02-23 17:55:21 +00006825 }
6826 return(ctxt->valid);
6827}
6828
6829/**
6830 * xmlValidateDocument:
6831 * @ctxt: the validation context
6832 * @doc: a document instance
6833 *
6834 * Try to validate the document instance
6835 *
6836 * basically it does the all the checks described by the XML Rec
6837 * i.e. validates the internal and external subset (if present)
6838 * and validate the document tree.
6839 *
6840 * returns 1 if valid or 0 otherwise
6841 */
6842
6843int
6844xmlValidateDocument(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
6845 int ret;
6846 xmlNodePtr root;
6847
Daniel Veillardc0be74b2004-11-03 19:16:55 +00006848 if (doc == NULL)
6849 return(0);
Daniel Veillard2fd85422002-10-16 14:32:41 +00006850 if ((doc->intSubset == NULL) && (doc->extSubset == NULL)) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006851 xmlErrValid(ctxt, XML_DTD_NO_DTD,
6852 "no DTD found!\n", NULL);
Owen Taylor3473f882001-02-23 17:55:21 +00006853 return(0);
Daniel Veillard2fd85422002-10-16 14:32:41 +00006854 }
Owen Taylor3473f882001-02-23 17:55:21 +00006855 if ((doc->intSubset != NULL) && ((doc->intSubset->SystemID != NULL) ||
6856 (doc->intSubset->ExternalID != NULL)) && (doc->extSubset == NULL)) {
William M. Brack8c22f9f2004-08-06 16:23:27 +00006857 xmlChar *sysID;
6858 if (doc->intSubset->SystemID != NULL) {
William M. Brackbebe7302004-08-05 06:46:47 +00006859 sysID = xmlBuildURI(doc->intSubset->SystemID,
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006860 doc->URL);
William M. Brack8c22f9f2004-08-06 16:23:27 +00006861 if (sysID == NULL) {
6862 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
6863 "Could not build URI for external subset \"%s\"\n",
6864 (const char *) doc->intSubset->SystemID);
6865 return 0;
6866 }
6867 } else
William M. Brackbebe7302004-08-05 06:46:47 +00006868 sysID = NULL;
William M. Brack8c22f9f2004-08-06 16:23:27 +00006869 doc->extSubset = xmlParseDTD(doc->intSubset->ExternalID,
William M. Brackbebe7302004-08-05 06:46:47 +00006870 (const xmlChar *)sysID);
William M. Brackbebe7302004-08-05 06:46:47 +00006871 if (sysID != NULL)
6872 xmlFree(sysID);
Owen Taylor3473f882001-02-23 17:55:21 +00006873 if (doc->extSubset == NULL) {
6874 if (doc->intSubset->SystemID != NULL) {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006875 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
Owen Taylor3473f882001-02-23 17:55:21 +00006876 "Could not load the external subset \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006877 (const char *) doc->intSubset->SystemID);
Owen Taylor3473f882001-02-23 17:55:21 +00006878 } else {
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006879 xmlErrValid(ctxt, XML_DTD_LOAD_ERROR,
Owen Taylor3473f882001-02-23 17:55:21 +00006880 "Could not load the external subset \"%s\"\n",
Daniel Veillardbb5abab2003-10-03 22:21:51 +00006881 (const char *) doc->intSubset->ExternalID);
Owen Taylor3473f882001-02-23 17:55:21 +00006882 }
6883 return(0);
6884 }
6885 }
6886
6887 if (doc->ids != NULL) {
6888 xmlFreeIDTable(doc->ids);
6889 doc->ids = NULL;
6890 }
6891 if (doc->refs != NULL) {
6892 xmlFreeRefTable(doc->refs);
6893 doc->refs = NULL;
6894 }
6895 ret = xmlValidateDtdFinal(ctxt, doc);
6896 if (!xmlValidateRoot(ctxt, doc)) return(0);
6897
6898 root = xmlDocGetRootElement(doc);
6899 ret &= xmlValidateElement(ctxt, doc, root);
6900 ret &= xmlValidateDocumentFinal(ctxt, doc);
6901 return(ret);
6902}
6903
Owen Taylor3473f882001-02-23 17:55:21 +00006904/************************************************************************
6905 * *
6906 * Routines for dynamic validation editing *
6907 * *
6908 ************************************************************************/
6909
6910/**
6911 * xmlValidGetPotentialChildren:
6912 * @ctree: an element content tree
Daniel Veillard7802ba52005-10-27 11:56:20 +00006913 * @names: an array to store the list of child names
Owen Taylor3473f882001-02-23 17:55:21 +00006914 * @len: a pointer to the number of element in the list
6915 * @max: the size of the array
6916 *
6917 * Build/extend a list of potential children allowed by the content tree
6918 *
6919 * returns the number of element in the list, or -1 in case of error.
6920 */
6921
6922int
Daniel Veillard7802ba52005-10-27 11:56:20 +00006923xmlValidGetPotentialChildren(xmlElementContent *ctree,
6924 const xmlChar **names,
Owen Taylor3473f882001-02-23 17:55:21 +00006925 int *len, int max) {
6926 int i;
6927
Daniel Veillard7802ba52005-10-27 11:56:20 +00006928 if ((ctree == NULL) || (names == NULL) || (len == NULL))
Owen Taylor3473f882001-02-23 17:55:21 +00006929 return(-1);
6930 if (*len >= max) return(*len);
6931
6932 switch (ctree->type) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006933 case XML_ELEMENT_CONTENT_PCDATA:
Owen Taylor3473f882001-02-23 17:55:21 +00006934 for (i = 0; i < *len;i++)
Daniel Veillard7802ba52005-10-27 11:56:20 +00006935 if (xmlStrEqual(BAD_CAST "#PCDATA", names[i])) return(*len);
6936 names[(*len)++] = BAD_CAST "#PCDATA";
Owen Taylor3473f882001-02-23 17:55:21 +00006937 break;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006938 case XML_ELEMENT_CONTENT_ELEMENT:
Owen Taylor3473f882001-02-23 17:55:21 +00006939 for (i = 0; i < *len;i++)
Daniel Veillard7802ba52005-10-27 11:56:20 +00006940 if (xmlStrEqual(ctree->name, names[i])) return(*len);
6941 names[(*len)++] = ctree->name;
Owen Taylor3473f882001-02-23 17:55:21 +00006942 break;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006943 case XML_ELEMENT_CONTENT_SEQ:
Daniel Veillard7802ba52005-10-27 11:56:20 +00006944 xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6945 xmlValidGetPotentialChildren(ctree->c2, names, len, max);
Owen Taylor3473f882001-02-23 17:55:21 +00006946 break;
6947 case XML_ELEMENT_CONTENT_OR:
Daniel Veillard7802ba52005-10-27 11:56:20 +00006948 xmlValidGetPotentialChildren(ctree->c1, names, len, max);
6949 xmlValidGetPotentialChildren(ctree->c2, names, len, max);
Owen Taylor3473f882001-02-23 17:55:21 +00006950 break;
6951 }
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006952
Owen Taylor3473f882001-02-23 17:55:21 +00006953 return(*len);
6954}
6955
William M. Brack9333cc22004-06-24 08:33:40 +00006956/*
6957 * Dummy function to suppress messages while we try out valid elements
6958 */
Daniel Veillardffa3c742005-07-21 13:24:09 +00006959static void XMLCDECL xmlNoValidityErr(void *ctx ATTRIBUTE_UNUSED,
William M. Brack9333cc22004-06-24 08:33:40 +00006960 const char *msg ATTRIBUTE_UNUSED, ...) {
6961 return;
6962}
6963
Owen Taylor3473f882001-02-23 17:55:21 +00006964/**
6965 * xmlValidGetValidElements:
6966 * @prev: an element to insert after
6967 * @next: an element to insert next
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00006968 * @names: an array to store the list of child names
Owen Taylor3473f882001-02-23 17:55:21 +00006969 * @max: the size of the array
6970 *
6971 * This function returns the list of authorized children to insert
6972 * within an existing tree while respecting the validity constraints
6973 * forced by the Dtd. The insertion point is defined using @prev and
6974 * @next in the following ways:
6975 * to insert before 'node': xmlValidGetValidElements(node->prev, node, ...
6976 * to insert next 'node': xmlValidGetValidElements(node, node->next, ...
6977 * to replace 'node': xmlValidGetValidElements(node->prev, node->next, ...
6978 * to prepend a child to 'node': xmlValidGetValidElements(NULL, node->childs,
6979 * to append a child to 'node': xmlValidGetValidElements(node->last, NULL, ...
6980 *
6981 * pointers to the element names are inserted at the beginning of the array
6982 * and do not need to be freed.
6983 *
6984 * returns the number of element in the list, or -1 in case of error. If
6985 * the function returns the value @max the caller is invited to grow the
6986 * receiving array and retry.
6987 */
6988
6989int
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00006990xmlValidGetValidElements(xmlNode *prev, xmlNode *next, const xmlChar **names,
Owen Taylor3473f882001-02-23 17:55:21 +00006991 int max) {
Daniel Veillardf69bb4b2001-05-19 13:24:56 +00006992 xmlValidCtxt vctxt;
Owen Taylor3473f882001-02-23 17:55:21 +00006993 int nb_valid_elements = 0;
Daniel Veillarde18bce02014-02-06 10:47:20 +01006994 const xmlChar *elements[256]={0};
Owen Taylor3473f882001-02-23 17:55:21 +00006995 int nb_elements = 0, i;
Daniel Veillard5e5c2d02002-02-09 18:03:01 +00006996 const xmlChar *name;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08006997
Owen Taylor3473f882001-02-23 17:55:21 +00006998 xmlNode *ref_node;
6999 xmlNode *parent;
7000 xmlNode *test_node;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007001
Owen Taylor3473f882001-02-23 17:55:21 +00007002 xmlNode *prev_next;
7003 xmlNode *next_prev;
7004 xmlNode *parent_childs;
7005 xmlNode *parent_last;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007006
Owen Taylor3473f882001-02-23 17:55:21 +00007007 xmlElement *element_desc;
7008
7009 if (prev == NULL && next == NULL)
7010 return(-1);
7011
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00007012 if (names == NULL) return(-1);
Owen Taylor3473f882001-02-23 17:55:21 +00007013 if (max <= 0) return(-1);
7014
William M. Brack9333cc22004-06-24 08:33:40 +00007015 memset(&vctxt, 0, sizeof (xmlValidCtxt));
7016 vctxt.error = xmlNoValidityErr; /* this suppresses err/warn output */
7017
Owen Taylor3473f882001-02-23 17:55:21 +00007018 nb_valid_elements = 0;
7019 ref_node = prev ? prev : next;
7020 parent = ref_node->parent;
7021
7022 /*
7023 * Retrieves the parent element declaration
7024 */
7025 element_desc = xmlGetDtdElementDesc(parent->doc->intSubset,
7026 parent->name);
7027 if ((element_desc == NULL) && (parent->doc->extSubset != NULL))
7028 element_desc = xmlGetDtdElementDesc(parent->doc->extSubset,
7029 parent->name);
7030 if (element_desc == NULL) return(-1);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007031
Owen Taylor3473f882001-02-23 17:55:21 +00007032 /*
7033 * Do a backup of the current tree structure
7034 */
7035 prev_next = prev ? prev->next : NULL;
7036 next_prev = next ? next->prev : NULL;
7037 parent_childs = parent->children;
7038 parent_last = parent->last;
7039
7040 /*
7041 * Creates a dummy node and insert it into the tree
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007042 */
Daniel Veillard95ddcd32004-10-26 21:53:55 +00007043 test_node = xmlNewDocNode (ref_node->doc, NULL, BAD_CAST "<!dummy?>", NULL);
Gaurav Gupta658b86c2014-08-07 11:19:03 +08007044 if (test_node == NULL)
7045 return(-1);
7046
Owen Taylor3473f882001-02-23 17:55:21 +00007047 test_node->parent = parent;
7048 test_node->prev = prev;
7049 test_node->next = next;
Daniel Veillardd455d792002-02-08 13:37:46 +00007050 name = test_node->name;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007051
Owen Taylor3473f882001-02-23 17:55:21 +00007052 if (prev) prev->next = test_node;
7053 else parent->children = test_node;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007054
Owen Taylor3473f882001-02-23 17:55:21 +00007055 if (next) next->prev = test_node;
7056 else parent->last = test_node;
7057
7058 /*
7059 * Insert each potential child node and check if the parent is
7060 * still valid
7061 */
7062 nb_elements = xmlValidGetPotentialChildren(element_desc->content,
7063 elements, &nb_elements, 256);
Daniel Veillardf8e3db02012-09-11 13:26:36 +08007064
Owen Taylor3473f882001-02-23 17:55:21 +00007065 for (i = 0;i < nb_elements;i++) {
7066 test_node->name = elements[i];
Daniel Veillardf69bb4b2001-05-19 13:24:56 +00007067 if (xmlValidateOneElement(&vctxt, parent->doc, parent)) {
Owen Taylor3473f882001-02-23 17:55:21 +00007068 int j;
7069
7070 for (j = 0; j < nb_valid_elements;j++)
Daniel Veillardaecc0dc2004-05-08 02:32:07 +00007071 if (xmlStrEqual(elements[i], names[j])) break;
7072 names[nb_valid_elements++] = elements[i];
Owen Taylor3473f882001-02-23 17:55:21 +00007073 if (nb_valid_elements >= max) break;
7074 }
7075 }
7076
7077 /*
7078 * Restore the tree structure
7079 */
7080 if (prev) prev->next = prev_next;
7081 if (next) next->prev = next_prev;
7082 parent->children = parent_childs;
7083 parent->last = parent_last;
Daniel Veillardd455d792002-02-08 13:37:46 +00007084
7085 /*
7086 * Free up the dummy node
7087 */
7088 test_node->name = name;
7089 xmlFreeNode(test_node);
7090
Owen Taylor3473f882001-02-23 17:55:21 +00007091 return(nb_valid_elements);
7092}
Daniel Veillard4432df22003-09-28 18:58:27 +00007093#endif /* LIBXML_VALID_ENABLED */
7094
Daniel Veillard5d4644e2005-04-01 13:11:58 +00007095#define bottom_valid
7096#include "elfgcchack.h"