summaryrefslogtreecommitdiff
path: root/textproc/xerces-c
diff options
context:
space:
mode:
authorhasso <hasso>2009-08-28 05:24:34 +0000
committerhasso <hasso>2009-08-28 05:24:34 +0000
commitdcf0c1df57427ab46cf63a2eb6c85c8c701bc8a3 (patch)
treebb38b044fa10c29902d66be89aa1852a314c1d0c /textproc/xerces-c
parent39aa442e5ba5f7231d4979cc38a206ae62e28bf2 (diff)
downloadpkgsrc-dcf0c1df57427ab46cf63a2eb6c85c8c701bc8a3.tar.gz
Add the fix for CVE-2009-1885 - nested DTD structure XML parsing remote denial
of service vulnerability. While there fix MASTER_SITES as recommended by pkglint. Bump PKGREVISION.
Diffstat (limited to 'textproc/xerces-c')
-rw-r--r--textproc/xerces-c/Makefile9
-rw-r--r--textproc/xerces-c/distinfo3
-rw-r--r--textproc/xerces-c/patches/patch-ba653
3 files changed, 658 insertions, 7 deletions
diff --git a/textproc/xerces-c/Makefile b/textproc/xerces-c/Makefile
index 4df2c65af8c..e040a2e534c 100644
--- a/textproc/xerces-c/Makefile
+++ b/textproc/xerces-c/Makefile
@@ -1,14 +1,11 @@
-# $NetBSD: Makefile,v 1.37 2009/05/22 17:59:41 zafer Exp $
+# $NetBSD: Makefile,v 1.38 2009/08/28 05:24:34 hasso Exp $
#
DISTNAME= xerces-c-src_2_8_0
+PKGREVISION= 1
PKGNAME= ${DISTNAME:S/-src_/-/:S/_/./g}
CATEGORIES= textproc
-MASTER_SITES= http://www.apache.org/dist/xerces/c/2/sources/ \
- http://apache.speedbone.de/xerces/c/2/sources/ \
- ftp://ftp.riken.jp/net/apache/xerces/c/2/sources/
-
-
+MASTER_SITES= ${MASTER_SITE_APACHE:=xerces/c/2/sources/}
MAINTAINER= pkgsrc-users@NetBSD.org
HOMEPAGE= http://xerces.apache.org/xerces-c/
diff --git a/textproc/xerces-c/distinfo b/textproc/xerces-c/distinfo
index f9a86852858..fbacf07b875 100644
--- a/textproc/xerces-c/distinfo
+++ b/textproc/xerces-c/distinfo
@@ -1,4 +1,4 @@
-$NetBSD: distinfo,v 1.12 2009/05/10 11:24:26 hasso Exp $
+$NetBSD: distinfo,v 1.13 2009/08/28 05:24:34 hasso Exp $
SHA1 (xerces-c-src_2_8_0.tar.gz) = f0803b1330daec3f44b17dee64c3c99de6b3cd3e
RMD160 (xerces-c-src_2_8_0.tar.gz) = d576df7870b043a338358834a0e1180dca39e838
@@ -22,3 +22,4 @@ SHA1 (patch-aq) = 9e25a396655fe11a068ddb4a48c23902214273e6
SHA1 (patch-ar) = 8b8055056afdb0c3d8a556211bf44aec496727a1
SHA1 (patch-as) = 3eaec415b91dcd129eaa7c5c6e0b5b417ad651bc
SHA1 (patch-at) = 31682536fccb69cbf29872793b2eb798e15e3bda
+SHA1 (patch-ba) = 5bf2099aa80dfa3c91d5297fd7238a5a049131d9
diff --git a/textproc/xerces-c/patches/patch-ba b/textproc/xerces-c/patches/patch-ba
new file mode 100644
index 00000000000..5fee6e1e3bb
--- /dev/null
+++ b/textproc/xerces-c/patches/patch-ba
@@ -0,0 +1,653 @@
+$NetBSD: patch-ba,v 1.1 2009/08/28 05:24:34 hasso Exp $
+
+Fix for CVE-2009-1885 from upstream subversion repo.
+
+--- ../../src/xercesc/validators/DTD/DTDScanner.cpp 2008/05/21 16:42:53 658750
++++ ../../src/xercesc/validators/DTD/DTDScanner.cpp 2009/08/24 14:03:57 807224
+@@ -27,7 +27,9 @@
+ #include <xercesc/util/FlagJanitor.hpp>
+ #include <xercesc/util/Janitor.hpp>
+ #include <xercesc/util/XMLUniDefs.hpp>
++#include <xercesc/util/ValueStackOf.hpp>
+ #include <xercesc/util/UnexpectedEOFException.hpp>
++#include <xercesc/util/OutOfMemoryException.hpp>
+ #include <xercesc/sax/InputSource.hpp>
+ #include <xercesc/framework/XMLDocumentHandler.hpp>
+ #include <xercesc/framework/XMLEntityHandler.hpp>
+@@ -39,7 +41,6 @@
+ #include <xercesc/validators/DTD/DTDEntityDecl.hpp>
+ #include <xercesc/validators/DTD/DocTypeHandler.hpp>
+ #include <xercesc/validators/DTD/DTDScanner.hpp>
+-#include <xercesc/util/OutOfMemoryException.hpp>
+
+ XERCES_CPP_NAMESPACE_BEGIN
+
+@@ -1046,338 +1047,354 @@
+ // Check for a PE ref here, but don't require spaces
+ checkForPERef(false, true);
+
+- // We have to check entity nesting here
+- unsigned int curReader;
+-
++ ValueStackOf<XMLSize_t>* arrNestedDecl=NULL;
+ //
+ // We know that the caller just saw an opening parenthesis, so we need
+- // to parse until we hit the end of it, recursing for other nested
+- // parentheses we see.
++ // to parse until we hit the end of it; if we find several parenthesis,
++ // store them in an array to be processed later.
+ //
+ // We have to check for one up front, since it could be something like
+ // (((a)*)) etc...
+ //
+ ContentSpecNode* curNode = 0;
+- if (fReaderMgr->skippedChar(chOpenParen))
++ while(fReaderMgr->skippedChar(chOpenParen))
+ {
+- curReader = fReaderMgr->getCurrentReaderNum();
++ // to check entity nesting
++ const unsigned int curReader = fReaderMgr->getCurrentReaderNum();
++ if(arrNestedDecl==NULL)
++ arrNestedDecl=new (fMemoryManager) ValueStackOf<XMLSize_t>(5, fMemoryManager);
++ arrNestedDecl->push(curReader);
+
+- // Lets call ourself and get back the resulting node
+- curNode = scanChildren(elemDecl, bufToUse);
++ // Check for a PE ref here, but don't require spaces
++ checkForPERef(false, true);
++ }
+
+- // If that failed, no need to go further, return failure
+- if (!curNode)
+- return 0;
++ // We must find a leaf node here, either standalone or nested in the parenthesis
++ if (!fReaderMgr->getName(bufToUse))
++ {
++ fScanner->emitError(XMLErrs::ExpectedElementName);
++ return 0;
++ }
+
+- if (curReader != fReaderMgr->getCurrentReaderNum() && fScanner->getDoValidation())
+- fScanner->getValidator()->emitError(XMLValid::PartialMarkupInPE);
++ //
++ // Create a leaf node for it. If we can find the element id for
++ // this element, then use it. Else, we have to fault in an element
++ // decl, marked as created because of being in a content model.
++ //
++ XMLElementDecl* decl = fDTDGrammar->getElemDecl(fEmptyNamespaceId, 0, bufToUse.getRawBuffer(), Grammar::TOP_LEVEL_SCOPE);
++ if (!decl)
++ {
++ decl = new (fGrammarPoolMemoryManager) DTDElementDecl
++ (
++ bufToUse.getRawBuffer()
++ , fEmptyNamespaceId
++ , DTDElementDecl::Any
++ , fGrammarPoolMemoryManager
++ );
++ decl->setCreateReason(XMLElementDecl::InContentModel);
++ decl->setExternalElemDeclaration(isReadingExternalEntity());
++ fDTDGrammar->putElemDecl(decl);
+ }
+- else
++ curNode = new (fGrammarPoolMemoryManager) ContentSpecNode
++ (
++ decl->getElementName()
++ , fGrammarPoolMemoryManager
++ );
++
++ // Check for a PE ref here, but don't require spaces
++ const bool gotSpaces = checkForPERef(false, true);
++
++ // Check for a repetition character after the leaf
++ XMLCh repCh = fReaderMgr->peekNextChar();
++ ContentSpecNode* tmpNode = makeRepNode(repCh, curNode, fGrammarPoolMemoryManager);
++ if (tmpNode != curNode)
+ {
+- // Not a nested paren, so it must be a leaf node
+- if (!fReaderMgr->getName(bufToUse))
++ if (gotSpaces)
+ {
+- fScanner->emitError(XMLErrs::ExpectedElementName);
+- return 0;
++ if (fScanner->emitErrorWillThrowException(XMLErrs::UnexpectedWhitespace))
++ {
++ delete tmpNode;
++ }
++ fScanner->emitError(XMLErrs::UnexpectedWhitespace);
+ }
++ fReaderMgr->getNextChar();
++ curNode = tmpNode;
++ }
+
++ while(arrNestedDecl==NULL || !arrNestedDecl->empty())
++ {
++ // Check for a PE ref here, but don't require spaces
++ checkForPERef(false, true);
++
+ //
+- // Create a leaf node for it. If we can find the element id for
+- // this element, then use it. Else, we have to fault in an element
+- // decl, marked as created because of being in a content model.
++ // Ok, the next character tells us what kind of content this particular
++ // model this particular parentesized section is. Its either a choice if
++ // we see ',', a sequence if we see '|', or a single leaf node if we see
++ // a closing paren.
++ //
++ const XMLCh opCh = fReaderMgr->peekNextChar();
++
++ if ((opCh != chComma)
++ && (opCh != chPipe)
++ && (opCh != chCloseParen))
++ {
++ // Not a legal char, so delete our node and return failure
++ delete curNode;
++ fScanner->emitError(XMLErrs::ExpectedSeqChoiceLeaf);
++ return 0;
++ }
++
+ //
+- XMLElementDecl* decl = fDTDGrammar->getElemDecl(fEmptyNamespaceId, 0, bufToUse.getRawBuffer(), Grammar::TOP_LEVEL_SCOPE);
+- if (!decl)
++ // Create the head node of the correct type. We need this to remember
++ // the top of the local tree. If it was a single subexpr, then just
++ // set the head node to the current node. For the others, we'll build
++ // the tree off the second child as we move across.
++ //
++ ContentSpecNode* headNode = 0;
++ ContentSpecNode::NodeTypes curType = ContentSpecNode::UnknownType;
++ if (opCh == chComma)
+ {
+- decl = new (fGrammarPoolMemoryManager) DTDElementDecl
++ curType = ContentSpecNode::Sequence;
++ headNode = new (fGrammarPoolMemoryManager) ContentSpecNode
+ (
+- bufToUse.getRawBuffer()
+- , fEmptyNamespaceId
+- , DTDElementDecl::Any
++ curType
++ , curNode
++ , 0
++ , true
++ , true
+ , fGrammarPoolMemoryManager
+ );
+- decl->setCreateReason(XMLElementDecl::InContentModel);
+- decl->setExternalElemDeclaration(isReadingExternalEntity());
+- fDTDGrammar->putElemDecl(decl);
++ curNode = headNode;
+ }
+- curNode = new (fGrammarPoolMemoryManager) ContentSpecNode
+- (
+- decl->getElementName()
+- , fGrammarPoolMemoryManager
+- );
+-
+- // Check for a PE ref here, but don't require spaces
+- const bool gotSpaces = checkForPERef(false, true);
+-
+- // Check for a repetition character after the leaf
+- const XMLCh repCh = fReaderMgr->peekNextChar();
+- ContentSpecNode* tmpNode = makeRepNode(repCh, curNode, fGrammarPoolMemoryManager);
+- if (tmpNode != curNode)
++ else if (opCh == chPipe)
+ {
+- if (gotSpaces)
+- {
+- if (fScanner->emitErrorWillThrowException(XMLErrs::UnexpectedWhitespace))
+- {
+- delete tmpNode;
+- }
+- fScanner->emitError(XMLErrs::UnexpectedWhitespace);
+- }
++ curType = ContentSpecNode::Choice;
++ headNode = new (fGrammarPoolMemoryManager) ContentSpecNode
++ (
++ curType
++ , curNode
++ , 0
++ , true
++ , true
++ , fGrammarPoolMemoryManager
++ );
++ curNode = headNode;
++ }
++ else
++ {
++ headNode = curNode;
+ fReaderMgr->getNextChar();
+- curNode = tmpNode;
+ }
+- }
+-
+- // Check for a PE ref here, but don't require spaces
+- checkForPERef(false, true);
+
+- //
+- // Ok, the next character tells us what kind of content this particular
+- // model this particular parentesized section is. Its either a choice if
+- // we see ',', a sequence if we see '|', or a single leaf node if we see
+- // a closing paren.
+- //
+- const XMLCh opCh = fReaderMgr->peekNextChar();
+-
+- if ((opCh != chComma)
+- && (opCh != chPipe)
+- && (opCh != chCloseParen))
+- {
+- // Not a legal char, so delete our node and return failure
+- delete curNode;
+- fScanner->emitError(XMLErrs::ExpectedSeqChoiceLeaf);
+- return 0;
+- }
+-
+- //
+- // Create the head node of the correct type. We need this to remember
+- // the top of the local tree. If it was a single subexpr, then just
+- // set the head node to the current node. For the others, we'll build
+- // the tree off the second child as we move across.
+- //
+- ContentSpecNode* headNode = 0;
+- ContentSpecNode::NodeTypes curType = ContentSpecNode::UnknownType;
+- if (opCh == chComma)
+- {
+- curType = ContentSpecNode::Sequence;
+- headNode = new (fGrammarPoolMemoryManager) ContentSpecNode
+- (
+- curType
+- , curNode
+- , 0
+- , true
+- , true
+- , fGrammarPoolMemoryManager
+- );
+- curNode = headNode;
+- }
+- else if (opCh == chPipe)
+- {
+- curType = ContentSpecNode::Choice;
+- headNode = new (fGrammarPoolMemoryManager) ContentSpecNode
+- (
+- curType
+- , curNode
+- , 0
+- , true
+- , true
+- , fGrammarPoolMemoryManager
+- );
+- curNode = headNode;
+- }
+- else
+- {
+- headNode = curNode;
+- fReaderMgr->getNextChar();
+- }
+-
+- //
+- // If it was a sequence or choice, we just loop until we get to the
+- // end of our section, adding each new leaf or sub expression to the
+- // right child of the current node, and making that new node the current
+- // node.
+- //
+- if ((opCh == chComma) || (opCh == chPipe))
+- {
+- ContentSpecNode* lastNode = 0;
+- while (true)
++ //
++ // If it was a sequence or choice, we just loop until we get to the
++ // end of our section, adding each new leaf or sub expression to the
++ // right child of the current node, and making that new node the current
++ // node.
++ //
++ if ((opCh == chComma) || (opCh == chPipe))
+ {
+- //
+- // The next thing must either be another | or , character followed
+- // by another leaf or subexpression, or a closing parenthesis, or a
+- // PE ref.
+- //
+- if (fReaderMgr->lookingAtChar(chPercent))
+- {
+- checkForPERef(false, true);
+- }
+- else if (fReaderMgr->skippedSpace())
+- {
+- // Just skip whitespace
+- fReaderMgr->skipPastSpaces();
+- }
+- else if (fReaderMgr->skippedChar(chCloseParen))
++ ContentSpecNode* lastNode = 0;
++ while (true)
+ {
+ //
+- // We've hit the end of this section, so break out. But, we
+- // need to see if we left a partial sequence of choice node
+- // without a second node. If so, we have to undo that and
+- // put its left child into the right node of the previous
+- // node.
++ // The next thing must either be another | or , character followed
++ // by another leaf or subexpression, or a closing parenthesis, or a
++ // PE ref.
+ //
+- if ((curNode->getType() == ContentSpecNode::Choice)
+- || (curNode->getType() == ContentSpecNode::Sequence))
++ if (fReaderMgr->lookingAtChar(chPercent))
++ {
++ checkForPERef(false, true);
++ }
++ else if (fReaderMgr->skippedSpace())
+ {
+- if (!curNode->getSecond())
++ // Just skip whitespace
++ fReaderMgr->skipPastSpaces();
++ }
++ else if (fReaderMgr->skippedChar(chCloseParen))
++ {
++ //
++ // We've hit the end of this section, so break out. But, we
++ // need to see if we left a partial sequence of choice node
++ // without a second node. If so, we have to undo that and
++ // put its left child into the right node of the previous
++ // node.
++ //
++ if ((curNode->getType() == ContentSpecNode::Choice)
++ || (curNode->getType() == ContentSpecNode::Sequence))
+ {
+- ContentSpecNode* saveFirst = curNode->orphanFirst();
+- lastNode->setSecond(saveFirst);
+- curNode = lastNode;
++ if (!curNode->getSecond())
++ {
++ ContentSpecNode* saveFirst = curNode->orphanFirst();
++ lastNode->setSecond(saveFirst);
++ curNode = lastNode;
++ }
+ }
++ break;
+ }
+- break;
+- }
+- else if (fReaderMgr->skippedChar(opCh))
+- {
+- // Check for a PE ref here, but don't require spaces
+- checkForPERef(false, true);
+-
+- if (fReaderMgr->skippedChar(chOpenParen))
++ else if (fReaderMgr->skippedChar(opCh))
+ {
+- curReader = fReaderMgr->getCurrentReaderNum();
++ // Check for a PE ref here, but don't require spaces
++ checkForPERef(false, true);
+
+- // Recurse to handle this new guy
+- ContentSpecNode* subNode;
+- try {
+- subNode = scanChildren(elemDecl, bufToUse);
+- }
+- catch (const XMLErrs::Codes)
++ if (fReaderMgr->skippedChar(chOpenParen))
+ {
+- delete headNode;
+- throw;
+- }
++ const unsigned int curReader = fReaderMgr->getCurrentReaderNum();
+
+- // If it failed, we are done, clean up here and return failure
+- if (!subNode)
+- {
+- delete headNode;
+- return 0;
++ // Recurse to handle this new guy
++ ContentSpecNode* subNode;
++ try {
++ subNode = scanChildren(elemDecl, bufToUse);
++ }
++ catch (const XMLErrs::Codes)
++ {
++ delete headNode;
++ throw;
++ }
++
++ // If it failed, we are done, clean up here and return failure
++ if (!subNode)
++ {
++ delete headNode;
++ return 0;
++ }
++
++ if (curReader != fReaderMgr->getCurrentReaderNum() && fScanner->getDoValidation())
++ fScanner->getValidator()->emitError(XMLValid::PartialMarkupInPE);
++
++ // Else patch it in and make it the new current
++ ContentSpecNode* newCur = new (fGrammarPoolMemoryManager) ContentSpecNode
++ (
++ curType
++ , subNode
++ , 0
++ , true
++ , true
++ , fGrammarPoolMemoryManager
++ );
++ curNode->setSecond(newCur);
++ lastNode = curNode;
++ curNode = newCur;
+ }
++ else
++ {
++ //
++ // Got to be a leaf node, so get a name. If we cannot get
++ // one, then clean up and get outa here.
++ //
++ if (!fReaderMgr->getName(bufToUse))
++ {
++ delete headNode;
++ fScanner->emitError(XMLErrs::ExpectedElementName);
++ return 0;
++ }
++
++ //
++ // Create a leaf node for it. If we can find the element
++ // id for this element, then use it. Else, we have to
++ // fault in an element decl, marked as created because
++ // of being in a content model.
++ //
++ XMLElementDecl* decl = fDTDGrammar->getElemDecl(fEmptyNamespaceId, 0, bufToUse.getRawBuffer(), Grammar::TOP_LEVEL_SCOPE);
++ if (!decl)
++ {
++ decl = new (fGrammarPoolMemoryManager) DTDElementDecl
++ (
++ bufToUse.getRawBuffer()
++ , fEmptyNamespaceId
++ , DTDElementDecl::Any
++ , fGrammarPoolMemoryManager
++ );
++ decl->setCreateReason(XMLElementDecl::InContentModel);
++ decl->setExternalElemDeclaration(isReadingExternalEntity());
++ fDTDGrammar->putElemDecl(decl);
++ }
+
+- if (curReader != fReaderMgr->getCurrentReaderNum() && fScanner->getDoValidation())
+- fScanner->getValidator()->emitError(XMLValid::PartialMarkupInPE);
++ ContentSpecNode* tmpLeaf = new (fGrammarPoolMemoryManager) ContentSpecNode
++ (
++ decl->getElementName()
++ , fGrammarPoolMemoryManager
++ );
+
+- // Else patch it in and make it the new current
+- ContentSpecNode* newCur = new (fGrammarPoolMemoryManager) ContentSpecNode
+- (
+- curType
+- , subNode
+- , 0
+- , true
+- , true
+- , fGrammarPoolMemoryManager
+- );
+- curNode->setSecond(newCur);
+- lastNode = curNode;
+- curNode = newCur;
++ // Check for a repetition character after the leaf
++ const XMLCh repCh = fReaderMgr->peekNextChar();
++ ContentSpecNode* tmpLeaf2 = makeRepNode(repCh, tmpLeaf, fGrammarPoolMemoryManager);
++ if (tmpLeaf != tmpLeaf2)
++ fReaderMgr->getNextChar();
++
++ //
++ // Create a new sequence or choice node, with the leaf
++ // (or rep surrounding it) we just got as its first node.
++ // Make the new node the second node of the current node,
++ // and then make it the current node.
++ //
++ ContentSpecNode* newCur = new (fGrammarPoolMemoryManager) ContentSpecNode
++ (
++ curType
++ , tmpLeaf2
++ , 0
++ , true
++ , true
++ , fGrammarPoolMemoryManager
++ );
++ curNode->setSecond(newCur);
++ lastNode = curNode;
++ curNode = newCur;
++ }
+ }
+ else
+ {
+- //
+- // Got to be a leaf node, so get a name. If we cannot get
+- // one, then clean up and get outa here.
+- //
+- if (!fReaderMgr->getName(bufToUse))
++ // Cannot be valid
++ delete headNode; // emitError may do a throw so need to clean-up first
++ if (opCh == chComma)
+ {
+- delete headNode;
+- fScanner->emitError(XMLErrs::ExpectedElementName);
+- return 0;
++ fScanner->emitError(XMLErrs::ExpectedChoiceOrCloseParen);
+ }
+-
+- //
+- // Create a leaf node for it. If we can find the element
+- // id for this element, then use it. Else, we have to
+- // fault in an element decl, marked as created because
+- // of being in a content model.
+- //
+- XMLElementDecl* decl = fDTDGrammar->getElemDecl(fEmptyNamespaceId, 0, bufToUse.getRawBuffer(), Grammar::TOP_LEVEL_SCOPE);
+- if (!decl)
++ else
+ {
+- decl = new (fGrammarPoolMemoryManager) DTDElementDecl
++ fScanner->emitError
+ (
+- bufToUse.getRawBuffer()
+- , fEmptyNamespaceId
+- , DTDElementDecl::Any
+- , fGrammarPoolMemoryManager
++ XMLErrs::ExpectedSeqOrCloseParen
++ , elemDecl.getFullName()
+ );
+- decl->setCreateReason(XMLElementDecl::InContentModel);
+- decl->setExternalElemDeclaration(isReadingExternalEntity());
+- fDTDGrammar->putElemDecl(decl);
+- }
++ }
++ return 0;
++ }
++ }
++ }
+
+- ContentSpecNode* tmpLeaf = new (fGrammarPoolMemoryManager) ContentSpecNode
+- (
+- decl->getElementName()
+- , fGrammarPoolMemoryManager
+- );
++ //
++ // We saw the terminating parenthesis so lets check for any repetition
++ // character, and create a node for that, making the head node the child
++ // of it.
++ //
++ const XMLCh repCh = fReaderMgr->peekNextChar();
++ curNode = makeRepNode(repCh, headNode, fGrammarPoolMemoryManager);
++ if (curNode != headNode)
++ fReaderMgr->getNextChar();
++
++ // prepare for recursion
++ if(arrNestedDecl==NULL)
++ break;
++ else
++ {
++ // If that failed, no need to go further, return failure
++ if (!curNode)
++ return 0;
+
+- // Check for a repetition character after the leaf
+- const XMLCh repCh = fReaderMgr->peekNextChar();
+- ContentSpecNode* tmpLeaf2 = makeRepNode(repCh, tmpLeaf, fGrammarPoolMemoryManager);
+- if (tmpLeaf != tmpLeaf2)
+- fReaderMgr->getNextChar();
++ const unsigned int curReader = arrNestedDecl->pop();
++ if (curReader != fReaderMgr->getCurrentReaderNum() && fScanner->getValidationScheme() == XMLScanner::Val_Always)
++ fScanner->getValidator()->emitError(XMLValid::PartialMarkupInPE);
+
+- //
+- // Create a new sequence or choice node, with the leaf
+- // (or rep surrounding it) we just got as its first node.
+- // Make the new node the second node of the current node,
+- // and then make it the current node.
+- //
+- ContentSpecNode* newCur = new (fGrammarPoolMemoryManager) ContentSpecNode
+- (
+- curType
+- , tmpLeaf2
+- , 0
+- , true
+- , true
+- , fGrammarPoolMemoryManager
+- );
+- curNode->setSecond(newCur);
+- lastNode = curNode;
+- curNode = newCur;
+- }
+- }
+- else
++ if(arrNestedDecl->empty())
+ {
+- // Cannot be valid
+- delete headNode; // emitError may do a throw so need to clean-up first
+- if (opCh == chComma)
+- {
+- fScanner->emitError(XMLErrs::ExpectedChoiceOrCloseParen);
+- }
+- else
+- {
+- fScanner->emitError
+- (
+- XMLErrs::ExpectedSeqOrCloseParen
+- , elemDecl.getFullName()
+- );
+- }
+- return 0;
++ delete arrNestedDecl;
++ arrNestedDecl=NULL;
+ }
+ }
+ }
+
+- //
+- // We saw the terminating parenthesis so lets check for any repetition
+- // character, and create a node for that, making the head node the child
+- // of it.
+- //
+- XMLCh repCh = fReaderMgr->peekNextChar();
+- ContentSpecNode* retNode = makeRepNode(repCh, headNode, fGrammarPoolMemoryManager);
+- if (retNode != headNode)
+- fReaderMgr->getNextChar();
+-
+- return retNode;
++ return curNode;
+ }
+
+