This function try to open a connection to the indicated resource
+via HTTP using the given method, adding the given extra headers
+and the input buffer for the request content.
\ No newline at end of file
diff --git a/doc/html/gnome-xml-xmlmemory.html b/doc/html/gnome-xml-xmlmemory.html
new file mode 100644
index 00000000..44ecc56b
--- /dev/null
+++ b/doc/html/gnome-xml-xmlmemory.html
@@ -0,0 +1,1277 @@
+xmlmemory
\ No newline at end of file
diff --git a/entities.c b/entities.c
index 78f7ec2c..78571e02 100644
--- a/entities.c
+++ b/entities.c
@@ -401,6 +401,8 @@ xmlEncodeEntities(xmlDocPtr doc, const xmlChar *input) {
const xmlChar *cur = input;
xmlChar *out = buffer;
static int warning = 1;
+ int html = 0;
+
if (warning) {
fprintf(stderr, "Deprecated API xmlEncodeEntities() used\n");
@@ -409,6 +411,9 @@ xmlEncodeEntities(xmlDocPtr doc, const xmlChar *input) {
}
if (input == NULL) return(NULL);
+ if (doc != NULL)
+ html = (doc->type == XML_HTML_DOCUMENT_NODE);
+
if (buffer == NULL) {
buffer_size = 1000;
buffer = (xmlChar *) xmlMalloc(buffer_size * sizeof(xmlChar));
@@ -452,7 +457,7 @@ xmlEncodeEntities(xmlDocPtr doc, const xmlChar *input) {
*out++ = 'o';
*out++ = 't';
*out++ = ';';
- } else if (*cur == '\'') {
+ } else if ((*cur == '\'') && (!html)) {
*out++ = '&';
*out++ = 'a';
*out++ = 'p';
@@ -536,8 +541,11 @@ xmlEncodeEntitiesReentrant(xmlDocPtr doc, const xmlChar *input) {
xmlChar *buffer = NULL;
xmlChar *out = NULL;
int buffer_size = 0;
+ int html = 0;
if (input == NULL) return(NULL);
+ if (doc != NULL)
+ html = (doc->type == XML_HTML_DOCUMENT_NODE);
/*
* allocate an translation buffer.
@@ -584,7 +592,7 @@ xmlEncodeEntitiesReentrant(xmlDocPtr doc, const xmlChar *input) {
*out++ = 'o';
*out++ = 't';
*out++ = ';';
- } else if (*cur == '\'') {
+ } else if ((*cur == '\'') && (!html)) {
*out++ = '&';
*out++ = 'a';
*out++ = 'p';
diff --git a/include/libxml/tree.h b/include/libxml/tree.h
index 6c7e795a..6c8ab102 100644
--- a/include/libxml/tree.h
+++ b/include/libxml/tree.h
@@ -35,7 +35,8 @@ typedef enum {
XML_DOCUMENT_NODE= 9,
XML_DOCUMENT_TYPE_NODE= 10,
XML_DOCUMENT_FRAG_NODE= 11,
- XML_NOTATION_NODE= 12
+ XML_NOTATION_NODE= 12,
+ XML_HTML_DOCUMENT_NODE= 13
} xmlElementType;
/*
diff --git a/include/libxml/valid.h b/include/libxml/valid.h
index 73a2a543..8c016a93 100644
--- a/include/libxml/valid.h
+++ b/include/libxml/valid.h
@@ -218,6 +218,14 @@ xmlNotationPtr xmlGetDtdNotationDesc (xmlDtdPtr dtd,
xmlElementPtr xmlGetDtdElementDesc (xmlDtdPtr dtd,
const xmlChar *name);
+int xmlValidGetValidElements(xmlNode *prev,
+ xmlNode *next,
+ const xmlChar **list,
+ int max);
+int xmlValidGetPotentialChildren(xmlElementContent *ctree,
+ const xmlChar **list,
+ int *len,
+ int max);
#ifdef __cplusplus
}
#endif
diff --git a/result/HTML/Down.html.err b/result/HTML/Down.html.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/HTML/fp40.htm b/result/HTML/fp40.htm
index 6497272a..4e70b0c7 100644
--- a/result/HTML/fp40.htm
+++ b/result/HTML/fp40.htm
@@ -143,10 +143,10 @@ Extensions, see the Server Extensions Resource Kit Update at: Microsoft Knowledge Base
For further technical information on FrontPage, please consult Support Online. Use Support
-Online to easily search Microsoft Product Support Services' collection of resources including
-technical articles from Microsoft's extensive Knowledge Base, FAQs, troubleshooters to find
+Online to easily search Microsoft Product Support Services' collection of resources including
+technical articles from Microsoft's extensive Knowledge Base, FAQs, troubleshooters to find
fast, accurate answers. You can also customize the site to control your search using either
-keywords or the site's natural language search engine, which uses normal everyday language for
+keywords or the site's natural language search engine, which uses normal everyday language for
answering inquiries, so you can write your question in your own words. To begin, go to
http://support.microsoft.com/support/.
diff --git a/result/HTML/fp40.htm.err b/result/HTML/fp40.htm.err
new file mode 100644
index 00000000..ab847035
--- /dev/null
+++ b/result/HTML/fp40.htm.err
@@ -0,0 +1,3 @@
+./test/HTML/fp40.htm:153: error: htmlParseEntityRef: no name
+technical articles from Microsoft's extensive Knowledge Base, FAQs, & troublesh
+ ^
diff --git a/result/HTML/liclose.html b/result/HTML/liclose.html
new file mode 100644
index 00000000..edc8eb1e
--- /dev/null
+++ b/result/HTML/liclose.html
@@ -0,0 +1,14 @@
+
+
+
+
+
+
+
+
First item
+
+
Second item, closes the first one
+
+
+
+
diff --git a/result/HTML/liclose.html.err b/result/HTML/liclose.html.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/HTML/reg1.html b/result/HTML/reg1.html
new file mode 100644
index 00000000..0a5d6549
--- /dev/null
+++ b/result/HTML/reg1.html
@@ -0,0 +1,12 @@
+
+
+
+Regression test 1
+
+
+
Regression test 1
+
+Ok file no problem
+
+
+
diff --git a/result/HTML/reg1.html.err b/result/HTML/reg1.html.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/HTML/reg2.html b/result/HTML/reg2.html
new file mode 100644
index 00000000..198ee044
--- /dev/null
+++ b/result/HTML/reg2.html
@@ -0,0 +1,15 @@
+
+
+
+Regression test 2
+
+
+
Regression test 2
+
+Autoclose of tag P
+
+
+Ok file no problem
+
+
+
diff --git a/result/HTML/reg2.html.err b/result/HTML/reg2.html.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/HTML/reg3.html b/result/HTML/reg3.html
new file mode 100644
index 00000000..65866f91
--- /dev/null
+++ b/result/HTML/reg3.html
@@ -0,0 +1,16 @@
+
+
+
+Regression test 3
+
+
+
Regression test 3
+
+Autoclose of tag P
+
+
+
+Ok file no problem
+
+
+
diff --git a/result/HTML/reg3.html.err b/result/HTML/reg3.html.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/HTML/reg4.html b/result/HTML/reg4.html
new file mode 100644
index 00000000..afae62c2
--- /dev/null
+++ b/result/HTML/reg4.html
@@ -0,0 +1,13 @@
+
+
+
+Regression test 4
+
+
+
Regression test 4
+
+Wrong close of tag P
+
+
+
+
diff --git a/result/HTML/reg4.html.err b/result/HTML/reg4.html.err
new file mode 100644
index 00000000..d11f77c5
--- /dev/null
+++ b/result/HTML/reg4.html.err
@@ -0,0 +1,3 @@
+./test/HTML/reg4.html:10: error: Unexpected end tag : P
+
+ ^
diff --git a/result/HTML/test2.html.err b/result/HTML/test2.html.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/HTML/test3.html.err b/result/HTML/test3.html.err
new file mode 100644
index 00000000..82d84a13
--- /dev/null
+++ b/result/HTML/test3.html.err
@@ -0,0 +1,12 @@
+./test/HTML/test3.html:6: error: Unexpected end tag : P
+
+ ^
+./test/HTML/test3.html:13: error: Unexpected end tag : P
+
+ ^
+./test/HTML/test3.html:27: error: Opening and ending tag mismatch: H4 and B
+
Links
+ ^
+./test/HTML/test3.html:27: error: Unexpected end tag : B
+
Links
+ ^
diff --git a/result/XPath/expr/compare b/result/XPath/expr/compare
new file mode 100644
index 00000000..daae1a2e
--- /dev/null
+++ b/result/XPath/expr/compare
@@ -0,0 +1,24 @@
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
diff --git a/result/XPath/expr/equality b/result/XPath/expr/equality
new file mode 100644
index 00000000..92d6d1c7
--- /dev/null
+++ b/result/XPath/expr/equality
@@ -0,0 +1,24 @@
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : true
+Object is a Boolean : false
+Object is a Boolean : false
+Object is a Boolean : true
diff --git a/result/comment.xml b/result/comment.xml
new file mode 100644
index 00000000..567160aa
--- /dev/null
+++ b/result/comment.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/comment2.xml b/result/comment2.xml
new file mode 100644
index 00000000..26242381
--- /dev/null
+++ b/result/comment2.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/noent/comment.xml b/result/noent/comment.xml
new file mode 100644
index 00000000..567160aa
--- /dev/null
+++ b/result/noent/comment.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/noent/comment2.xml b/result/noent/comment2.xml
new file mode 100644
index 00000000..26242381
--- /dev/null
+++ b/result/noent/comment2.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/noent/ns b/result/noent/ns
new file mode 100644
index 00000000..94b927e5
--- /dev/null
+++ b/result/noent/ns
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/result/noent/ns2 b/result/noent/ns2
new file mode 100644
index 00000000..b69ad82f
--- /dev/null
+++ b/result/noent/ns2
@@ -0,0 +1,2 @@
+
+
diff --git a/result/noent/ns3 b/result/noent/ns3
new file mode 100644
index 00000000..b69ad82f
--- /dev/null
+++ b/result/noent/ns3
@@ -0,0 +1,2 @@
+
+
diff --git a/result/noent/ns4 b/result/noent/ns4
new file mode 100644
index 00000000..fb7bc3e7
--- /dev/null
+++ b/result/noent/ns4
@@ -0,0 +1,2 @@
+
+
diff --git a/result/noent/pi.xml b/result/noent/pi.xml
new file mode 100644
index 00000000..27bed5b9
--- /dev/null
+++ b/result/noent/pi.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/noent/pi2.xml b/result/noent/pi2.xml
new file mode 100644
index 00000000..acf76f95
--- /dev/null
+++ b/result/noent/pi2.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/ns b/result/ns
new file mode 100644
index 00000000..94b927e5
--- /dev/null
+++ b/result/ns
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/result/ns2 b/result/ns2
new file mode 100644
index 00000000..b69ad82f
--- /dev/null
+++ b/result/ns2
@@ -0,0 +1,2 @@
+
+
diff --git a/result/ns3 b/result/ns3
new file mode 100644
index 00000000..b69ad82f
--- /dev/null
+++ b/result/ns3
@@ -0,0 +1,2 @@
+
+
diff --git a/result/ns4 b/result/ns4
new file mode 100644
index 00000000..fb7bc3e7
--- /dev/null
+++ b/result/ns4
@@ -0,0 +1,2 @@
+
+
diff --git a/result/pi.xml b/result/pi.xml
new file mode 100644
index 00000000..27bed5b9
--- /dev/null
+++ b/result/pi.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/pi2.xml b/result/pi2.xml
new file mode 100644
index 00000000..acf76f95
--- /dev/null
+++ b/result/pi2.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/result/valid/REC-xml-19980210.xml.err b/result/valid/REC-xml-19980210.xml.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/valid/dia.xml.err b/result/valid/dia.xml.err
new file mode 100644
index 00000000..e69de29b
diff --git a/result/valid/xlink.xml.err b/result/valid/xlink.xml.err
new file mode 100644
index 00000000..6e8beefd
--- /dev/null
+++ b/result/valid/xlink.xml.err
@@ -0,0 +1,6 @@
+./test/valid/xlink.xml:450: validity error: ID dt-arc already defined
+
An arc is contained within an
+ ^
+./test/valid/xlink.xml:530: validity error: IDREF attribute def reference an unknown ID 'dt-xlg'
+
+^
diff --git a/test/HTML/liclose.html b/test/HTML/liclose.html
new file mode 100644
index 00000000..73640344
--- /dev/null
+++ b/test/HTML/liclose.html
@@ -0,0 +1,13 @@
+
+
+
+
+
+
+
+
First item
+
Second item, closes the first one
+
+
+
diff --git a/test/HTML/reg1.html b/test/HTML/reg1.html
new file mode 100644
index 00000000..ecdd007f
--- /dev/null
+++ b/test/HTML/reg1.html
@@ -0,0 +1,10 @@
+
+
+Regression test 1
+
+
+
Regression test 1
+
+Ok file no problem
+
+
diff --git a/test/HTML/reg2.html b/test/HTML/reg2.html
new file mode 100644
index 00000000..7145c194
--- /dev/null
+++ b/test/HTML/reg2.html
@@ -0,0 +1,12 @@
+
+
+Regression test 2
+
+
+
Regression test 2
+
+Autoclose of tag P
+
+Ok file no problem
+
+
diff --git a/test/HTML/reg3.html b/test/HTML/reg3.html
new file mode 100644
index 00000000..014483ba
--- /dev/null
+++ b/test/HTML/reg3.html
@@ -0,0 +1,13 @@
+
+
+Regression test 3
+
+
+
Regression test 3
+
+Autoclose of tag P
+
+
+Ok file no problem
+
+
diff --git a/test/HTML/reg4.html b/test/HTML/reg4.html
new file mode 100644
index 00000000..7d04ca23
--- /dev/null
+++ b/test/HTML/reg4.html
@@ -0,0 +1,12 @@
+
+
+Regression test 4
+
+
+
Regression test 4
+
+Wrong close of tag P
+
+
+
+
diff --git a/test/XPath/docs/id b/test/XPath/docs/id
new file mode 100644
index 00000000..4b6659ff
--- /dev/null
+++ b/test/XPath/docs/id
@@ -0,0 +1,28 @@
+
+
+
+ Welcome to Gnome
+
+
+ The Linux adventure
+
bla bla bla ...
+
+
...
+
+
+ Chapter 2
+
this is chapter 2 ...
+
+
+ Chapter 3
+
this is chapter 3 ...
+
+
+ Chapter 4
+
this is chapter 4 ...
+
+
+ Chapter 5
+
this is chapter 5 ...
+
+
diff --git a/test/XPath/expr/compare b/test/XPath/expr/compare
new file mode 100644
index 00000000..81b26322
--- /dev/null
+++ b/test/XPath/expr/compare
@@ -0,0 +1,24 @@
+0<1
+0<=1
+0>1
+0>=1
+1<0
+1<=0
+1>0
+1>=0
+1<1
+1<=1
+1>1
+1>=1
+'0'<1
+'0'<=1
+'0'>1
+'0'>=1
+0<'1.2'
+0<='1.2'
+0>'1.2'
+0>='1.2'
+false()<1
+false()<=1
+0>true()
+0>=true()
diff --git a/test/XPath/expr/equality b/test/XPath/expr/equality
new file mode 100644
index 00000000..2a8c84fc
--- /dev/null
+++ b/test/XPath/expr/equality
@@ -0,0 +1,25 @@
+1=1
+1!=1
+1=0
+1!=0
+true()=true()
+true()!=true()
+true()=false()
+false()!=true()
+'test'='test'
+'test'!='test'
+'test2'='test'
+'test2'!='test'
+false()=0
+false()!=0
+false()=1
+false()!=1
+0=true()
+0!=true()
+1=true()
+1!=true()
+true()='test'
+false()='test'
+'test'!=true()
+'test'!=false()
+
diff --git a/test/comment.xml b/test/comment.xml
new file mode 100644
index 00000000..98c5effd
--- /dev/null
+++ b/test/comment.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/test/comment2.xml b/test/comment2.xml
new file mode 100644
index 00000000..9e122ecf
--- /dev/null
+++ b/test/comment2.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/test/ns b/test/ns
new file mode 100644
index 00000000..94b927e5
--- /dev/null
+++ b/test/ns
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/test/ns2 b/test/ns2
new file mode 100644
index 00000000..80aaf945
--- /dev/null
+++ b/test/ns2
@@ -0,0 +1,3 @@
+
+
diff --git a/test/ns3 b/test/ns3
new file mode 100644
index 00000000..76bb20d4
--- /dev/null
+++ b/test/ns3
@@ -0,0 +1,3 @@
+
+
diff --git a/test/ns4 b/test/ns4
new file mode 100644
index 00000000..136bf923
--- /dev/null
+++ b/test/ns4
@@ -0,0 +1,2 @@
+
+
diff --git a/test/pi.xml b/test/pi.xml
new file mode 100644
index 00000000..48c7ff04
--- /dev/null
+++ b/test/pi.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/test/pi2.xml b/test/pi2.xml
new file mode 100644
index 00000000..710d51c9
--- /dev/null
+++ b/test/pi2.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
diff --git a/testHTML.c b/testHTML.c
index 9415a398..8bced6cb 100644
--- a/testHTML.c
+++ b/testHTML.c
@@ -15,6 +15,8 @@
#include
#include
+#include
+
#ifdef HAVE_SYS_TYPES_H
#include
@@ -32,12 +34,16 @@
#include
#endif
+#include "xmlmemory.h"
#include "HTMLparser.h"
#include "HTMLtree.h"
#include "debugXML.h"
static int debug = 0;
static int copy = 0;
+static int sax = 0;
+static int repeat = 0;
+static int noout = 0;
/*
* Note: this is perfectly clean HTML, i.e. not a useful test.
@@ -59,12 +65,544 @@ We are doing our best to get it back on-line,\n\
";
*/
+xmlSAXHandler emptySAXHandlerStruct = {
+ NULL, /* internalSubset */
+ NULL, /* isStandalone */
+ NULL, /* hasInternalSubset */
+ NULL, /* hasExternalSubset */
+ NULL, /* resolveEntity */
+ NULL, /* getEntity */
+ NULL, /* entityDecl */
+ NULL, /* notationDecl */
+ NULL, /* attributeDecl */
+ NULL, /* elementDecl */
+ NULL, /* unparsedEntityDecl */
+ NULL, /* setDocumentLocator */
+ NULL, /* startDocument */
+ NULL, /* endDocument */
+ NULL, /* startElement */
+ NULL, /* endElement */
+ NULL, /* reference */
+ NULL, /* characters */
+ NULL, /* ignorableWhitespace */
+ NULL, /* processingInstruction */
+ NULL, /* comment */
+ NULL, /* xmlParserWarning */
+ NULL, /* xmlParserError */
+ NULL, /* xmlParserError */
+ NULL, /* getParameterEntity */
+};
+
+xmlSAXHandlerPtr emptySAXHandler = &emptySAXHandlerStruct;
+extern xmlSAXHandlerPtr debugSAXHandler;
+
+/************************************************************************
+ * *
+ * Debug Handlers *
+ * *
+ ************************************************************************/
+
+/**
+ * isStandaloneDebug:
+ * @ctxt: An XML parser context
+ *
+ * Is this document tagged standalone ?
+ *
+ * Returns 1 if true
+ */
+int
+isStandaloneDebug(void *ctx)
+{
+ fprintf(stdout, "SAX.isStandalone()\n");
+ return(0);
+}
+
+/**
+ * hasInternalSubsetDebug:
+ * @ctxt: An XML parser context
+ *
+ * Does this document has an internal subset
+ *
+ * Returns 1 if true
+ */
+int
+hasInternalSubsetDebug(void *ctx)
+{
+ fprintf(stdout, "SAX.hasInternalSubset()\n");
+ return(0);
+}
+
+/**
+ * hasExternalSubsetDebug:
+ * @ctxt: An XML parser context
+ *
+ * Does this document has an external subset
+ *
+ * Returns 1 if true
+ */
+int
+hasExternalSubsetDebug(void *ctx)
+{
+ fprintf(stdout, "SAX.hasExternalSubset()\n");
+ return(0);
+}
+
+/**
+ * hasInternalSubsetDebug:
+ * @ctxt: An XML parser context
+ *
+ * Does this document has an internal subset
+ */
+void
+internalSubsetDebug(void *ctx, const xmlChar *name,
+ const xmlChar *ExternalID, const xmlChar *SystemID)
+{
+ /* xmlDtdPtr externalSubset; */
+
+ fprintf(stdout, "SAX.internalSubset(%s, %s, %s)\n",
+ name, ExternalID, SystemID);
+
+/***********
+ if ((ExternalID != NULL) || (SystemID != NULL)) {
+ externalSubset = xmlParseDTD(ExternalID, SystemID);
+ if (externalSubset != NULL) {
+ xmlFreeDtd(externalSubset);
+ }
+ }
+ ***********/
+}
+
+/**
+ * resolveEntityDebug:
+ * @ctxt: An XML parser context
+ * @publicId: The public ID of the entity
+ * @systemId: The system ID of the entity
+ *
+ * Special entity resolver, better left to the parser, it has
+ * more context than the application layer.
+ * The default behaviour is to NOT resolve the entities, in that case
+ * the ENTITY_REF nodes are built in the structure (and the parameter
+ * values).
+ *
+ * Returns the xmlParserInputPtr if inlined or NULL for DOM behaviour.
+ */
+xmlParserInputPtr
+resolveEntityDebug(void *ctx, const xmlChar *publicId, const xmlChar *systemId)
+{
+ /* xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx; */
+
+
+ fprintf(stdout, "SAX.resolveEntity(");
+ if (publicId != NULL)
+ fprintf(stdout, "%s", (char *)publicId);
+ else
+ fprintf(stdout, " ");
+ if (systemId != NULL)
+ fprintf(stdout, ", %s)\n", (char *)systemId);
+ else
+ fprintf(stdout, ", )\n");
+/*********
+ if (systemId != NULL) {
+ return(xmlNewInputFromFile(ctxt, (char *) systemId));
+ }
+ *********/
+ return(NULL);
+}
+
+/**
+ * getEntityDebug:
+ * @ctxt: An XML parser context
+ * @name: The entity name
+ *
+ * Get an entity by name
+ *
+ * Returns the xmlParserInputPtr if inlined or NULL for DOM behaviour.
+ */
+xmlEntityPtr
+getEntityDebug(void *ctx, const xmlChar *name)
+{
+ fprintf(stdout, "SAX.getEntity(%s)\n", name);
+ return(NULL);
+}
+
+/**
+ * getParameterEntityDebug:
+ * @ctxt: An XML parser context
+ * @name: The entity name
+ *
+ * Get a parameter entity by name
+ *
+ * Returns the xmlParserInputPtr
+ */
+xmlEntityPtr
+getParameterEntityDebug(void *ctx, const xmlChar *name)
+{
+ fprintf(stdout, "SAX.getParameterEntity(%s)\n", name);
+ return(NULL);
+}
+
+
+/**
+ * entityDeclDebug:
+ * @ctxt: An XML parser context
+ * @name: the entity name
+ * @type: the entity type
+ * @publicId: The public ID of the entity
+ * @systemId: The system ID of the entity
+ * @content: the entity value (without processing).
+ *
+ * An entity definition has been parsed
+ */
+void
+entityDeclDebug(void *ctx, const xmlChar *name, int type,
+ const xmlChar *publicId, const xmlChar *systemId, xmlChar *content)
+{
+ fprintf(stdout, "SAX.entityDecl(%s, %d, %s, %s, %s)\n",
+ name, type, publicId, systemId, content);
+}
+
+/**
+ * attributeDeclDebug:
+ * @ctxt: An XML parser context
+ * @name: the attribute name
+ * @type: the attribute type
+ *
+ * An attribute definition has been parsed
+ */
+void
+attributeDeclDebug(void *ctx, const xmlChar *elem, const xmlChar *name,
+ int type, int def, const xmlChar *defaultValue,
+ xmlEnumerationPtr tree)
+{
+ fprintf(stdout, "SAX.attributeDecl(%s, %s, %d, %d, %s, ...)\n",
+ elem, name, type, def, defaultValue);
+}
+
+/**
+ * elementDeclDebug:
+ * @ctxt: An XML parser context
+ * @name: the element name
+ * @type: the element type
+ * @content: the element value (without processing).
+ *
+ * An element definition has been parsed
+ */
+void
+elementDeclDebug(void *ctx, const xmlChar *name, int type,
+ xmlElementContentPtr content)
+{
+ fprintf(stdout, "SAX.elementDecl(%s, %d, ...)\n",
+ name, type);
+}
+
+/**
+ * notationDeclDebug:
+ * @ctxt: An XML parser context
+ * @name: The name of the notation
+ * @publicId: The public ID of the entity
+ * @systemId: The system ID of the entity
+ *
+ * What to do when a notation declaration has been parsed.
+ */
+void
+notationDeclDebug(void *ctx, const xmlChar *name,
+ const xmlChar *publicId, const xmlChar *systemId)
+{
+ fprintf(stdout, "SAX.notationDecl(%s, %s, %s)\n",
+ (char *) name, (char *) publicId, (char *) systemId);
+}
+
+/**
+ * unparsedEntityDeclDebug:
+ * @ctxt: An XML parser context
+ * @name: The name of the entity
+ * @publicId: The public ID of the entity
+ * @systemId: The system ID of the entity
+ * @notationName: the name of the notation
+ *
+ * What to do when an unparsed entity declaration is parsed
+ */
+void
+unparsedEntityDeclDebug(void *ctx, const xmlChar *name,
+ const xmlChar *publicId, const xmlChar *systemId,
+ const xmlChar *notationName)
+{
+ fprintf(stdout, "SAX.unparsedEntityDecl(%s, %s, %s, %s)\n",
+ (char *) name, (char *) publicId, (char *) systemId,
+ (char *) notationName);
+}
+
+/**
+ * setDocumentLocatorDebug:
+ * @ctxt: An XML parser context
+ * @loc: A SAX Locator
+ *
+ * Receive the document locator at startup, actually xmlDefaultSAXLocator
+ * Everything is available on the context, so this is useless in our case.
+ */
+void
+setDocumentLocatorDebug(void *ctx, xmlSAXLocatorPtr loc)
+{
+ fprintf(stdout, "SAX.setDocumentLocator()\n");
+}
+
+/**
+ * startDocumentDebug:
+ * @ctxt: An XML parser context
+ *
+ * called when the document start being processed.
+ */
+void
+startDocumentDebug(void *ctx)
+{
+ fprintf(stdout, "SAX.startDocument()\n");
+}
+
+/**
+ * endDocumentDebug:
+ * @ctxt: An XML parser context
+ *
+ * called when the document end has been detected.
+ */
+void
+endDocumentDebug(void *ctx)
+{
+ fprintf(stdout, "SAX.endDocument()\n");
+}
+
+/**
+ * startElementDebug:
+ * @ctxt: An XML parser context
+ * @name: The element name
+ *
+ * called when an opening tag has been processed.
+ */
+void
+startElementDebug(void *ctx, const xmlChar *name, const xmlChar **atts)
+{
+ int i;
+
+ fprintf(stdout, "SAX.startElement(%s", (char *) name);
+ if (atts != NULL) {
+ for (i = 0;(atts[i] != NULL);i++) {
+ fprintf(stdout, ", %s='", atts[i++]);
+ fprintf(stdout, "%s'", atts[i]);
+ }
+ }
+ fprintf(stdout, ")\n");
+}
+
+/**
+ * endElementDebug:
+ * @ctxt: An XML parser context
+ * @name: The element name
+ *
+ * called when the end of an element has been detected.
+ */
+void
+endElementDebug(void *ctx, const xmlChar *name)
+{
+ fprintf(stdout, "SAX.endElement(%s)\n", (char *) name);
+}
+
+/**
+ * charactersDebug:
+ * @ctxt: An XML parser context
+ * @ch: a xmlChar string
+ * @len: the number of xmlChar
+ *
+ * receiving some chars from the parser.
+ * Question: how much at a time ???
+ */
+void
+charactersDebug(void *ctx, const xmlChar *ch, int len)
+{
+ int i;
+
+ fprintf(stdout, "SAX.characters(");
+ for (i = 0;(i < len) && (i < 30);i++)
+ fprintf(stdout, "%c", ch[i]);
+ fprintf(stdout, ", %d)\n", len);
+}
+
+/**
+ * referenceDebug:
+ * @ctxt: An XML parser context
+ * @name: The entity name
+ *
+ * called when an entity reference is detected.
+ */
+void
+referenceDebug(void *ctx, const xmlChar *name)
+{
+ fprintf(stdout, "SAX.reference(%s)\n", name);
+}
+
+/**
+ * ignorableWhitespaceDebug:
+ * @ctxt: An XML parser context
+ * @ch: a xmlChar string
+ * @start: the first char in the string
+ * @len: the number of xmlChar
+ *
+ * receiving some ignorable whitespaces from the parser.
+ * Question: how much at a time ???
+ */
+void
+ignorableWhitespaceDebug(void *ctx, const xmlChar *ch, int len)
+{
+ fprintf(stdout, "SAX.ignorableWhitespace(%.30s, %d)\n",
+ (char *) ch, len);
+}
+
+/**
+ * processingInstructionDebug:
+ * @ctxt: An XML parser context
+ * @target: the target name
+ * @data: the PI data's
+ * @len: the number of xmlChar
+ *
+ * A processing instruction has been parsed.
+ */
+void
+processingInstructionDebug(void *ctx, const xmlChar *target,
+ const xmlChar *data)
+{
+ fprintf(stdout, "SAX.processingInstruction(%s, %s)\n",
+ (char *) target, (char *) data);
+}
+
+/**
+ * commentDebug:
+ * @ctxt: An XML parser context
+ * @value: the comment content
+ *
+ * A comment has been parsed.
+ */
+void
+commentDebug(void *ctx, const xmlChar *value)
+{
+ fprintf(stdout, "SAX.comment(%s)\n", value);
+}
+
+/**
+ * warningDebug:
+ * @ctxt: An XML parser context
+ * @msg: the message to display/transmit
+ * @...: extra parameters for the message display
+ *
+ * Display and format a warning messages, gives file, line, position and
+ * extra parameters.
+ */
+void
+warningDebug(void *ctx, const char *msg, ...)
+{
+ va_list args;
+
+ va_start(args, msg);
+ fprintf(stdout, "SAX.warning: ");
+ vfprintf(stdout, msg, args);
+ va_end(args);
+}
+
+/**
+ * errorDebug:
+ * @ctxt: An XML parser context
+ * @msg: the message to display/transmit
+ * @...: extra parameters for the message display
+ *
+ * Display and format a error messages, gives file, line, position and
+ * extra parameters.
+ */
+void
+errorDebug(void *ctx, const char *msg, ...)
+{
+ va_list args;
+
+ va_start(args, msg);
+ fprintf(stdout, "SAX.error: ");
+ vfprintf(stdout, msg, args);
+ va_end(args);
+}
+
+/**
+ * fatalErrorDebug:
+ * @ctxt: An XML parser context
+ * @msg: the message to display/transmit
+ * @...: extra parameters for the message display
+ *
+ * Display and format a fatalError messages, gives file, line, position and
+ * extra parameters.
+ */
+void
+fatalErrorDebug(void *ctx, const char *msg, ...)
+{
+ va_list args;
+
+ va_start(args, msg);
+ fprintf(stdout, "SAX.fatalError: ");
+ vfprintf(stdout, msg, args);
+ va_end(args);
+}
+
+xmlSAXHandler debugSAXHandlerStruct = {
+ internalSubsetDebug,
+ isStandaloneDebug,
+ hasInternalSubsetDebug,
+ hasExternalSubsetDebug,
+ resolveEntityDebug,
+ getEntityDebug,
+ entityDeclDebug,
+ notationDeclDebug,
+ attributeDeclDebug,
+ elementDeclDebug,
+ unparsedEntityDeclDebug,
+ setDocumentLocatorDebug,
+ startDocumentDebug,
+ endDocumentDebug,
+ startElementDebug,
+ endElementDebug,
+ referenceDebug,
+ charactersDebug,
+ ignorableWhitespaceDebug,
+ processingInstructionDebug,
+ commentDebug,
+ warningDebug,
+ errorDebug,
+ fatalErrorDebug,
+ getParameterEntityDebug,
+};
+
+xmlSAXHandlerPtr debugSAXHandler = &debugSAXHandlerStruct;
/************************************************************************
* *
* Debug *
* *
************************************************************************/
+void parseSAXFile(char *filename) {
+ htmlDocPtr doc;
+ /*
+ * Empty callbacks for checking
+ */
+ doc = htmlSAXParseFile(filename, NULL, emptySAXHandler, NULL);
+ if (doc != NULL) {
+ fprintf(stdout, "htmlSAXParseFile returned non-NULL\n");
+ xmlFreeDoc(doc);
+ }
+
+ if (!noout) {
+ /*
+ * Debug callback
+ */
+ doc = htmlSAXParseFile(filename, NULL, debugSAXHandler, NULL);
+ if (doc != NULL) {
+ fprintf(stdout, "htmlSAXParseFile returned non-NULL\n");
+ xmlFreeDoc(doc);
+ }
+ }
+}
+
void parseAndPrintFile(char *filename) {
htmlDocPtr doc, tmp;
@@ -85,10 +623,12 @@ void parseAndPrintFile(char *filename) {
/*
* print it.
*/
- if (!debug)
- htmlDocDump(stdout, doc);
- else
- xmlDebugDumpDocument(stdout, doc);
+ if (!noout) {
+ if (!debug)
+ htmlDocDump(stdout, doc);
+ else
+ xmlDebugDumpDocument(stdout, doc);
+ }
/*
* free it.
@@ -128,7 +668,7 @@ void parseAndPrintBuffer(xmlChar *buf) {
}
int main(int argc, char **argv) {
- int i;
+ int i, count;
int files = 0;
for (i = 1; i < argc ; i++) {
@@ -136,20 +676,43 @@ int main(int argc, char **argv) {
debug++;
else if ((!strcmp(argv[i], "-copy")) || (!strcmp(argv[i], "--copy")))
copy++;
+ else if ((!strcmp(argv[i], "-sax")) || (!strcmp(argv[i], "--sax")))
+ sax++;
+ else if ((!strcmp(argv[i], "-noout")) || (!strcmp(argv[i], "--noout")))
+ noout++;
+ else if ((!strcmp(argv[i], "-repeat")) ||
+ (!strcmp(argv[i], "--repeat")))
+ repeat++;
}
for (i = 1; i < argc ; i++) {
if (argv[i][0] != '-') {
- parseAndPrintFile(argv[i]);
+ if (repeat) {
+ for (count = 0;count < 100 * repeat;count++) {
+ if (sax)
+ parseSAXFile(argv[i]);
+ else
+ parseAndPrintFile(argv[i]);
+ }
+ } else {
+ if (sax)
+ parseSAXFile(argv[i]);
+ else
+ parseAndPrintFile(argv[i]);
+ }
files ++;
}
}
if (files == 0) {
- printf("Usage : %s [--debug] [--copy] HTMLfiles ...\n",
+ printf("Usage : %s [--debug] [--copy] [--copy] HTMLfiles ...\n",
argv[0]);
printf("\tParse the HTML files and output the result of the parsing\n");
printf("\t--debug : dump a debug tree of the in-memory document\n");
printf("\t--copy : used to test the internal copy implementation\n");
+ printf("\t--sax : debug the sequence of SAX callbacks\n");
+ printf("\t--repeat : parse the file 100 times, for timing or profiling\n");
+ printf("\t--noout : do not print the result\n");
}
+ xmlMemoryDump();
return(0);
}
diff --git a/testXPath.c b/testXPath.c
index 236f5062..2c298493 100644
--- a/testXPath.c
+++ b/testXPath.c
@@ -90,7 +90,8 @@ void xmlXPAthDebugDumpNodeSet(FILE *output, xmlNodeSetPtr cur) {
fprintf(output, "%d", i + 1);
if (cur->nodeTab[i] == NULL)
fprintf(output, " NULL\n");
- else if (cur->nodeTab[i]->type == XML_DOCUMENT_NODE)
+ else if ((cur->nodeTab[i]->type == XML_DOCUMENT_NODE) ||
+ (cur->nodeTab[i]->type == XML_HTML_DOCUMENT_NODE))
fprintf(output, " /\n");
else if (cur->nodeTab[i]->type == XML_ATTRIBUTE_NODE)
xmlDebugDumpAttr(output, (xmlAttrPtr)cur->nodeTab[i], 2);
diff --git a/tester.c b/tester.c
index cff492c8..7002614b 100644
--- a/tester.c
+++ b/tester.c
@@ -44,6 +44,7 @@ static int noent = 0;
static int noout = 0;
static int valid = 0;
static int repeat = 0;
+static int insert = 0;
extern int xmlDoValidityCheckingDefaultValue;
@@ -140,10 +141,33 @@ void parseAndPrintFile(char *filename) {
xmlFreeDoc(tmp);
}
- /*
- * print it.
- */
- if (noout == 0) {
+ if (insert) {
+ const xmlChar* list[256];
+ int nb, i;
+ xmlNodePtr node;
+
+ if (doc->root != NULL) {
+ node = doc->root;
+ while ((node != NULL) && (node->last == NULL)) node = node->next;
+ if (node != NULL) {
+ nb = xmlValidGetValidElements(node->last, NULL, list, 256);
+ if (nb < 0) {
+ printf("could not get valid list of elements\n");
+ } else if (nb == 0) {
+ printf("No element can be indersted under root\n");
+ } else {
+ printf("%d element types can be indersted under root:\n",
+ nb);
+ for (i = 0;i < nb;i++) {
+ printf("%s\n", list[i]);
+ }
+ }
+ }
+ }
+ }else if (noout == 0) {
+ /*
+ * print it.
+ */
if (!debug)
xmlDocDump(stdout, doc);
else
@@ -211,6 +235,9 @@ int main(int argc, char **argv) {
else if ((!strcmp(argv[i], "-valid")) ||
(!strcmp(argv[i], "--valid")))
valid++;
+ else if ((!strcmp(argv[i], "-insert")) ||
+ (!strcmp(argv[i], "--insert")))
+ insert++;
else if ((!strcmp(argv[i], "-repeat")) ||
(!strcmp(argv[i], "--repeat")))
repeat++;
@@ -238,6 +265,7 @@ int main(int argc, char **argv) {
printf("\t--noout : don't output the result\n");
printf("\t--valid : validate the document in addition to std well-formed check\n");
printf("\t--repeat : parse the file 100 times, for timing or profiling\n");
+ printf("\t--insert : test for valid insertions\n");
}
xmlMemoryDump();
diff --git a/tree.c b/tree.c
index d1caae9e..3a06dfa9 100644
--- a/tree.c
+++ b/tree.c
@@ -2013,6 +2013,7 @@ xmlNodeGetContent(xmlNodePtr cur) {
case XML_ENTITY_NODE:
case XML_COMMENT_NODE:
case XML_DOCUMENT_NODE:
+ case XML_HTML_DOCUMENT_NODE:
case XML_DOCUMENT_TYPE_NODE:
case XML_NOTATION_NODE:
return(NULL);
@@ -2066,6 +2067,7 @@ xmlNodeSetContent(xmlNodePtr cur, const xmlChar *content) {
cur->content = NULL;
break;
case XML_DOCUMENT_NODE:
+ case XML_HTML_DOCUMENT_NODE:
case XML_DOCUMENT_TYPE_NODE:
break;
case XML_NOTATION_NODE:
@@ -2115,6 +2117,7 @@ xmlNodeSetContentLen(xmlNodePtr cur, const xmlChar *content, int len) {
cur->content = NULL;
break;
case XML_DOCUMENT_NODE:
+ case XML_HTML_DOCUMENT_NODE:
case XML_DOCUMENT_TYPE_NODE:
break;
case XML_NOTATION_NODE:
@@ -2180,6 +2183,7 @@ xmlNodeAddContentLen(xmlNodePtr cur, const xmlChar *content, int len) {
if (content != NULL)
cur->content = xmlStrncat(cur->content, content, len);
case XML_DOCUMENT_NODE:
+ case XML_HTML_DOCUMENT_NODE:
case XML_DOCUMENT_TYPE_NODE:
break;
case XML_NOTATION_NODE:
diff --git a/tree.h b/tree.h
index 6c7e795a..6c8ab102 100644
--- a/tree.h
+++ b/tree.h
@@ -35,7 +35,8 @@ typedef enum {
XML_DOCUMENT_NODE= 9,
XML_DOCUMENT_TYPE_NODE= 10,
XML_DOCUMENT_FRAG_NODE= 11,
- XML_NOTATION_NODE= 12
+ XML_NOTATION_NODE= 12,
+ XML_HTML_DOCUMENT_NODE= 13
} xmlElementType;
/*
diff --git a/valid.c b/valid.c
index 123a93c3..901b030a 100644
--- a/valid.c
+++ b/valid.c
@@ -2670,6 +2670,7 @@ xmlSprintfElementChilds(char *buf, xmlNodePtr node, int glob) {
break;
case XML_ATTRIBUTE_NODE:
case XML_DOCUMENT_NODE:
+ case XML_HTML_DOCUMENT_NODE:
case XML_DOCUMENT_TYPE_NODE:
case XML_DOCUMENT_FRAG_NODE:
case XML_NOTATION_NODE:
@@ -2962,3 +2963,170 @@ xmlValidateDocument(xmlValidCtxtPtr ctxt, xmlDocPtr doc) {
return(ret);
}
+
+/************************************************************************
+ * *
+ * Routines for dynamic validation editing *
+ * *
+ ************************************************************************/
+
+/**
+ * xmlValidGetPotentialChildren:
+ * @ctree: an element content tree
+ * @list: an array to store the list of child names
+ * @len: a pointer to the number of element in the list
+ * @max: the size of the array
+ *
+ * Build/extend a list of potential children allowed by the content tree
+ *
+ * returns the number of element in the list, or -1 in case of error.
+ */
+
+int
+xmlValidGetPotentialChildren(xmlElementContent *ctree, const xmlChar **list,
+ int *len, int max) {
+ int i;
+
+ if ((ctree == NULL) || (list == NULL) || (len == NULL))
+ return(-1);
+ if (*len >= max) return(*len);
+
+ switch (ctree->type) {
+ case XML_ELEMENT_CONTENT_PCDATA:
+ for (i = 0; i < *len;i++)
+ if (!xmlStrcmp("#PCDATA", list[i])) return(*len);
+ list[(*len)++] = "#PCDATA";
+ break;
+ case XML_ELEMENT_CONTENT_ELEMENT:
+ for (i = 0; i < *len;i++)
+ if (!xmlStrcmp(ctree->name, list[i])) return(*len);
+ list[(*len)++] = ctree->name;
+ break;
+ case XML_ELEMENT_CONTENT_SEQ:
+ xmlValidGetPotentialChildren(ctree->c1, list, len, max);
+ xmlValidGetPotentialChildren(ctree->c2, list, len, max);
+ break;
+ case XML_ELEMENT_CONTENT_OR:
+ xmlValidGetPotentialChildren(ctree->c1, list, len, max);
+ xmlValidGetPotentialChildren(ctree->c2, list, len, max);
+ break;
+ }
+
+ return(*len);
+}
+
+/**
+ * xmlValidGetValidElements:
+ * @prev: an element to insert after
+ * @next: an element to insert next
+ * @list: an array to store the list of child names
+ * @max: the size of the array
+ *
+ * This function returns the list of authorized children to insert
+ * within an existing tree while respecting the validity constraints
+ * forced by the Dtd. The insertion point is defined using @prev and
+ * @next in the following ways:
+ * to insert before 'node': xmlValidGetValidElements(node->prev, node, ...
+ * to insert next 'node': xmlValidGetValidElements(node, node->next, ...
+ * to replace 'node': xmlValidGetValidElements(node->prev, node->next, ...
+ * to prepend a child to 'node': xmlValidGetValidElements(NULL, node->childs,
+ * to append a child to 'node': xmlValidGetValidElements(node->last, NULL, ...
+ *
+ * pointers to the element names are inserted at the beginning of the array
+ * and do not need to be freed.
+ *
+ * returns the number of element in the list, or -1 in case of error. If
+ * the function returns the value @max the caller is invited to grow the
+ * receiving array and retry.
+ */
+
+int
+xmlValidGetValidElements(xmlNode *prev, xmlNode *next, const xmlChar **list,
+ int max) {
+ int nb_valid_elements = 0;
+ const xmlChar *elements[256];
+ int nb_elements = 0, i;
+
+ xmlNode *ref_node;
+ xmlNode *parent;
+ xmlNode *test_node;
+
+ xmlNode *prev_next;
+ xmlNode *next_prev;
+ xmlNode *parent_childs;
+ xmlNode *parent_last;
+
+ xmlElement *element_desc;
+
+ if (prev == NULL && next == NULL)
+ return(-1);
+
+ if (list == NULL) return(-1);
+ if (max <= 0) return(-1);
+
+ nb_valid_elements = 0;
+ ref_node = prev ? prev : next;
+ parent = ref_node->parent;
+
+ /*
+ * Retrieves the parent element declaration
+ */
+ element_desc = xmlGetDtdElementDesc(parent->doc->intSubset,
+ parent->name);
+ if ((element_desc == NULL) && (parent->doc->extSubset != NULL))
+ element_desc = xmlGetDtdElementDesc(parent->doc->extSubset,
+ parent->name);
+ if (element_desc == NULL) return(-1);
+
+ /*
+ * Do a backup of the current tree structure
+ */
+ prev_next = prev ? prev->next : NULL;
+ next_prev = next ? next->prev : NULL;
+ parent_childs = parent->childs;
+ parent_last = parent->last;
+
+ /*
+ * Creates a dummy node and insert it into the tree
+ */
+ test_node = xmlNewNode (NULL, "");
+ test_node->doc = ref_node->doc;
+ test_node->parent = parent;
+ test_node->prev = prev;
+ test_node->next = next;
+
+ if (prev) prev->next = test_node;
+ else parent->childs = test_node;
+
+ if (next) next->prev = test_node;
+ else parent->last = test_node;
+
+ /*
+ * Insert each potential child node and check if the parent is
+ * still valid
+ */
+ nb_elements = xmlValidGetPotentialChildren(element_desc->content,
+ elements, &nb_elements, 256);
+
+ for (i = 0;i < nb_elements;i++) {
+ test_node->name = elements[i];
+ if (xmlValidateOneElement(NULL, parent->doc, parent)) {
+ int j;
+
+ for (j = 0; j < nb_valid_elements;j++)
+ if (!xmlStrcmp(elements[i], list[j])) break;
+ list[nb_valid_elements++] = elements[i];
+ if (nb_valid_elements >= max) break;
+ }
+ }
+
+ /*
+ * Restore the tree structure
+ */
+ if (prev) prev->next = prev_next;
+ if (next) next->prev = next_prev;
+ parent->childs = parent_childs;
+ parent->last = parent_last;
+
+ return(nb_valid_elements);
+}
diff --git a/valid.h b/valid.h
index 73a2a543..8c016a93 100644
--- a/valid.h
+++ b/valid.h
@@ -218,6 +218,14 @@ xmlNotationPtr xmlGetDtdNotationDesc (xmlDtdPtr dtd,
xmlElementPtr xmlGetDtdElementDesc (xmlDtdPtr dtd,
const xmlChar *name);
+int xmlValidGetValidElements(xmlNode *prev,
+ xmlNode *next,
+ const xmlChar **list,
+ int max);
+int xmlValidGetPotentialChildren(xmlElementContent *ctree,
+ const xmlChar **list,
+ int *len,
+ int max);
#ifdef __cplusplus
}
#endif
diff --git a/xmlmemory.c b/xmlmemory.c
index ca9e9c01..d88fbb10 100644
--- a/xmlmemory.c
+++ b/xmlmemory.c
@@ -23,6 +23,10 @@
#ifdef HAVE_MALLOC_H
#include
#endif
+#ifdef HAVE_STDLIB_H
+#include
+#endif
+
#include "xmlmemory.h"
@@ -78,7 +82,10 @@ typedef struct memnod {
static unsigned long debugMemSize = 0;
+static unsigned long debugMaxMemSize = 0;
static int block=0;
+int xmlMemStopAtBlock = 0;
+int xmlMemInitialized = 0;
#ifdef MEM_LIST
static MEMHDR *memlist = NULL;
#endif
@@ -94,11 +101,24 @@ void debugmem_list_delete(MEMHDR *);
#define TEST_POINT
#endif
+/**
+ * xmlMallocBreakpoint:
+ *
+ * Breakpoint to use in conjunction with xmlMemStopAtBlock. When the block
+ * number reaches the specified value this function is called. One need to add a breakpoint
+ * to it to get the context in which the given block is allocated.
+ */
+
+void
+xmlMallocBreakpoint(void) {
+ fprintf(stderr, "xmlMallocBreakpoint reached on block %d\n", xmlMemStopAtBlock);
+}
+
/**
* xmlMallocLoc:
* @size: an int specifying the size in byte to allocate.
* @file: the file name or NULL
- * @file: the line number
+ @file: the line number
*
* a malloc() equivalent, with logging of the allocation info.
*
@@ -110,6 +130,7 @@ xmlMallocLoc(int size, const char * file, int line)
{
MEMHDR *p;
+ if (!xmlMemInitialized) xmlInitMemory();
#ifdef DEBUG_MEMORY
fprintf(stderr, "Malloc(%d)\n",size);
#endif
@@ -129,6 +150,7 @@ xmlMallocLoc(int size, const char * file, int line)
p->mh_file = file;
p->mh_line = line;
debugMemSize += size;
+ if (debugMemSize > debugMaxMemSize) debugMaxMemSize = debugMemSize;
#ifdef MEM_LIST
debugmem_list_add(p);
#endif
@@ -137,6 +159,7 @@ xmlMallocLoc(int size, const char * file, int line)
fprintf(stderr, "Malloc(%d) Ok\n",size);
#endif
+ if (xmlMemStopAtBlock == block) xmlMallocBreakpoint();
TEST_POINT
@@ -176,6 +199,7 @@ xmlReallocLoc(void *ptr,int size, const char * file, int line)
MEMHDR *p;
unsigned long number;
+ if (!xmlMemInitialized) xmlInitMemory();
TEST_POINT
p = CLIENT_2_HDR(ptr);
@@ -201,6 +225,7 @@ xmlReallocLoc(void *ptr,int size, const char * file, int line)
p->mh_file = file;
p->mh_line = line;
debugMemSize += size;
+ if (debugMemSize > debugMaxMemSize) debugMaxMemSize = debugMemSize;
#ifdef MEM_LIST
debugmem_list_add(p);
#endif
@@ -281,6 +306,7 @@ xmlMemStrdupLoc(const char *str, const char *file, int line)
size_t size = strlen(str) + 1;
MEMHDR *p;
+ if (!xmlMemInitialized) xmlInitMemory();
TEST_POINT
p = (MEMHDR *) malloc(RESERVE_SIZE+size);
@@ -294,11 +320,14 @@ xmlMemStrdupLoc(const char *str, const char *file, int line)
p->mh_file = file;
p->mh_line = line;
debugMemSize += size;
+ if (debugMemSize > debugMaxMemSize) debugMaxMemSize = debugMemSize;
#ifdef MEM_LIST
debugmem_list_add(p);
#endif
s = HDR_2_CLIENT(p);
+ if (xmlMemStopAtBlock == block) xmlMallocBreakpoint();
+
if (s != NULL)
strcpy(s,str);
else
@@ -365,7 +394,8 @@ xmlMemDisplay(FILE *fp)
#endif
- fprintf(fp," MEMORY ALLOCATED : %lu\n",debugMemSize);
+ fprintf(fp," MEMORY ALLOCATED : %lu, MAX was %lu\n",
+ debugMemSize, debugMaxMemSize);
fprintf(fp,"BLOCK NUMBER SIZE TYPE\n");
idx = 0;
p = memlist;
@@ -473,6 +503,15 @@ int
xmlInitMemory(void)
{
int ret;
+
+#ifdef HAVE_STDLIB_H
+ char *breakpoint;
+
+ breakpoint = getenv("XML_MEM_BREAKPOINT");
+ if (breakpoint != NULL) {
+ sscanf(breakpoint, "%d", &xmlMemStopAtBlock);
+ }
+#endif
#ifdef DEBUG_MEMORY
fprintf(stderr, "xmlInitMemory() Ok\n");
diff --git a/xpath.c b/xpath.c
index 3a703bb7..f646e9f7 100644
--- a/xpath.c
+++ b/xpath.c
@@ -562,7 +562,8 @@ xmlXPathDebugNodeSet(FILE *output, xmlNodeSetPtr obj) {
fprintf(output, " NULL !\n");
return;
}
- if (obj->nodeTab[i]->type == XML_DOCUMENT_NODE)
+ if ((obj->nodeTab[i]->type == XML_DOCUMENT_NODE) ||
+ (obj->nodeTab[i]->type == XML_HTML_DOCUMENT_NODE))
fprintf(output, " /");
else if (obj->nodeTab[i]->name == NULL)
fprintf(output, " noname!");
@@ -1471,7 +1472,8 @@ xmlXPathNextChild(xmlXPathParserContextPtr ctxt, xmlNodePtr cur) {
return(ctxt->context->doc->root);
return(ctxt->context->node->childs);
}
- if (ctxt->context->node->type == XML_DOCUMENT_NODE)
+ if ((ctxt->context->node->type == XML_DOCUMENT_NODE) ||
+ (ctxt->context->node->type == XML_HTML_DOCUMENT_NODE))
return(NULL);
return(cur->next);
}