443 lines
15 KiB
C++
443 lines
15 KiB
C++
|
/*
|
||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||
|
* contributor license agreements. See the NOTICE file distributed with
|
||
|
* this work for additional information regarding copyright ownership.
|
||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||
|
* (the "License"); you may not use this file except in compliance with
|
||
|
* the License. You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* $Id: ReaderMgr.hpp 696218 2008-09-17 09:31:41Z borisk $
|
||
|
*/
|
||
|
|
||
|
#if !defined(XERCESC_INCLUDE_GUARD_READERMGR_HPP)
|
||
|
#define XERCESC_INCLUDE_GUARD_READERMGR_HPP
|
||
|
|
||
|
#include <xercesc/internal/XMLReader.hpp>
|
||
|
#include <xercesc/util/PlatformUtils.hpp>
|
||
|
#include <xercesc/util/RefStackOf.hpp>
|
||
|
#include <xercesc/sax/Locator.hpp>
|
||
|
#include <xercesc/framework/XMLBuffer.hpp>
|
||
|
|
||
|
XERCES_CPP_NAMESPACE_BEGIN
|
||
|
|
||
|
class XMLEntityDecl;
|
||
|
class XMLEntityHandler;
|
||
|
class XMLDocumentHandler;
|
||
|
class XMLScanner;
|
||
|
|
||
|
|
||
|
// ---------------------------------------------------------------------------
|
||
|
// This class is used by the scanner. The scanner must deal with expansion
|
||
|
// of entities, some of which are totally different files (external parsed
|
||
|
// entities.) It does so by pushing readers onto a stack. The top reader is
|
||
|
// the one it wants to read out of, but that one must be popped when it is
|
||
|
// empty. To keep that logic from being all over the place, the scanner
|
||
|
// talks to the reader manager, which handles the stack and popping off
|
||
|
// used up readers.
|
||
|
// ---------------------------------------------------------------------------
|
||
|
class XMLPARSER_EXPORT ReaderMgr : public XMemory
|
||
|
, public Locator
|
||
|
{
|
||
|
public :
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Class specific types
|
||
|
// -----------------------------------------------------------------------
|
||
|
struct LastExtEntityInfo : public XMemory
|
||
|
{
|
||
|
const XMLCh* systemId;
|
||
|
const XMLCh* publicId;
|
||
|
XMLFileLoc lineNumber;
|
||
|
XMLFileLoc colNumber;
|
||
|
};
|
||
|
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Constructors and Destructor
|
||
|
// -----------------------------------------------------------------------
|
||
|
ReaderMgr(MemoryManager* const manager = XMLPlatformUtils::fgMemoryManager);
|
||
|
~ReaderMgr();
|
||
|
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Convenience scanning methods
|
||
|
//
|
||
|
// This are all convenience methods that work in terms of the core
|
||
|
// character spooling methods.
|
||
|
// -----------------------------------------------------------------------
|
||
|
bool atEOF() const;
|
||
|
bool getName(XMLBuffer& toFill);
|
||
|
bool getQName(XMLBuffer& toFill, int* colonPosition);
|
||
|
bool getNameToken(XMLBuffer& toFill);
|
||
|
XMLCh getNextChar();
|
||
|
bool getNextCharIfNot(const XMLCh chNotToGet, XMLCh& chGotten);
|
||
|
void movePlainContentChars(XMLBuffer &dest);
|
||
|
void getSpaces(XMLBuffer& toFill);
|
||
|
void getUpToCharOrWS(XMLBuffer& toFill, const XMLCh toCheck);
|
||
|
bool isEmpty() const;
|
||
|
bool lookingAtChar(const XMLCh toCheck);
|
||
|
bool lookingAtSpace();
|
||
|
XMLCh peekNextChar();
|
||
|
bool skipIfQuote(XMLCh& chGotten);
|
||
|
void skipPastChar(const XMLCh toSkip);
|
||
|
bool skipPastSpaces(bool inDecl = false);
|
||
|
void skipToChar(const XMLCh toSkipTo);
|
||
|
bool skippedChar(const XMLCh toSkip);
|
||
|
bool skippedSpace();
|
||
|
bool skippedString(const XMLCh* const toSkip);
|
||
|
bool skippedStringLong(const XMLCh* const toSkip);
|
||
|
void skipQuotedString(const XMLCh quoteCh);
|
||
|
XMLCh skipUntilIn(const XMLCh* const listToSkip);
|
||
|
XMLCh skipUntilInOrWS(const XMLCh* const listToSkip);
|
||
|
bool peekString(const XMLCh* const toPeek);
|
||
|
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Control methods
|
||
|
// -----------------------------------------------------------------------
|
||
|
void cleanStackBackTo(const XMLSize_t readerNum);
|
||
|
XMLReader* createReader
|
||
|
(
|
||
|
const InputSource& src
|
||
|
, const bool xmlDecl
|
||
|
, const XMLReader::RefFrom refFrom
|
||
|
, const XMLReader::Types type
|
||
|
, const XMLReader::Sources source
|
||
|
, const bool calcSrsOfs = true
|
||
|
);
|
||
|
XMLReader* createReader
|
||
|
(
|
||
|
const XMLCh* const sysId
|
||
|
, const XMLCh* const pubId
|
||
|
, const bool xmlDecl
|
||
|
, const XMLReader::RefFrom refFrom
|
||
|
, const XMLReader::Types type
|
||
|
, const XMLReader::Sources source
|
||
|
, InputSource*& srcToFill
|
||
|
, const bool calcSrcOfs = true
|
||
|
, const bool disableDefaultEntityResolution = false
|
||
|
);
|
||
|
XMLReader* createReader
|
||
|
(
|
||
|
const XMLCh* const baseURI
|
||
|
, const XMLCh* const sysId
|
||
|
, const XMLCh* const pubId
|
||
|
, const bool xmlDecl
|
||
|
, const XMLReader::RefFrom refFrom
|
||
|
, const XMLReader::Types type
|
||
|
, const XMLReader::Sources source
|
||
|
, InputSource*& srcToFill
|
||
|
, const bool calcSrcOfs = true
|
||
|
, const bool disableDefaultEntityResolution = false
|
||
|
);
|
||
|
XMLReader* createIntEntReader
|
||
|
(
|
||
|
const XMLCh* const sysId
|
||
|
, const XMLReader::RefFrom refFrom
|
||
|
, const XMLReader::Types type
|
||
|
, const XMLCh* const dataBuf
|
||
|
, const XMLSize_t dataLen
|
||
|
, const bool copyBuf
|
||
|
, const bool calcSrcOfs = true
|
||
|
);
|
||
|
bool isScanningPERefOutOfLiteral() const;
|
||
|
bool pushReader
|
||
|
(
|
||
|
XMLReader* const reader
|
||
|
, XMLEntityDecl* const entity
|
||
|
);
|
||
|
void reset();
|
||
|
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Getter methods
|
||
|
// -----------------------------------------------------------------------
|
||
|
const XMLCh* getCurrentEncodingStr() const;
|
||
|
const XMLEntityDecl* getCurrentEntity() const;
|
||
|
XMLEntityDecl* getCurrentEntity();
|
||
|
const XMLReader* getCurrentReader() const;
|
||
|
XMLReader* getCurrentReader();
|
||
|
XMLSize_t getCurrentReaderNum() const;
|
||
|
XMLSize_t getReaderDepth() const;
|
||
|
void getLastExtEntityInfo(LastExtEntityInfo& lastInfo) const;
|
||
|
XMLFilePos getSrcOffset() const;
|
||
|
bool getThrowEOE() const;
|
||
|
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Setter methods
|
||
|
// -----------------------------------------------------------------------
|
||
|
void setEntityHandler(XMLEntityHandler* const newHandler);
|
||
|
void setThrowEOE(const bool newValue);
|
||
|
void setXMLVersion(const XMLReader::XMLVersion version);
|
||
|
void setStandardUriConformant(const bool newValue);
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Implement the SAX Locator interface
|
||
|
// -----------------------------------------------------------------------
|
||
|
virtual const XMLCh* getPublicId() const;
|
||
|
virtual const XMLCh* getSystemId() const;
|
||
|
virtual XMLFileLoc getLineNumber() const;
|
||
|
virtual XMLFileLoc getColumnNumber() const;
|
||
|
|
||
|
|
||
|
private :
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Private helper methods
|
||
|
// -----------------------------------------------------------------------
|
||
|
const XMLReader* getLastExtEntity(const XMLEntityDecl*& itsEntity) const;
|
||
|
bool popReader();
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Unimplemented constructors and operators
|
||
|
// -----------------------------------------------------------------------
|
||
|
ReaderMgr(const ReaderMgr&);
|
||
|
ReaderMgr& operator=(const ReaderMgr&);
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Private data members
|
||
|
//
|
||
|
// fCurEntity
|
||
|
// This is the current top of stack entity. We pull it off the stack
|
||
|
// and store it here for efficiency.
|
||
|
//
|
||
|
// fCurReader
|
||
|
// This is the current top of stack reader. We pull it off the
|
||
|
// stack and store it here for efficiency.
|
||
|
//
|
||
|
// fEntityHandler
|
||
|
// This is the installed entity handler. Its installed via the
|
||
|
// scanner but he passes it on to us since we need it the most, in
|
||
|
// process of creating external entity readers.
|
||
|
//
|
||
|
// fEntityStack
|
||
|
// We need to keep up with which of the pushed readers are pushed
|
||
|
// entity values that are being spooled. This is done to avoid the
|
||
|
// problem of recursive definitions. This stack consists of refs to
|
||
|
// EntityDecl objects for the pushed entities.
|
||
|
//
|
||
|
// fNextReaderNum
|
||
|
// This is the reader serial number value. Each new reader that is
|
||
|
// created from this reader is given a successive number. This lets
|
||
|
// us catch things like partial markup errors and such.
|
||
|
//
|
||
|
// fReaderStack
|
||
|
// This is the stack of reader references. We own all the readers
|
||
|
// and destroy them when they are used up.
|
||
|
//
|
||
|
// fThrowEOE
|
||
|
// This flag controls whether we throw an exception when we hit an
|
||
|
// end of entity. The scanner doesn't really need to know about ends
|
||
|
// of entities in the int/ext subsets, so it will turn this flag off
|
||
|
// until it gets into the content usually.
|
||
|
//
|
||
|
// fXMLVersion
|
||
|
// Enum to indicate if each Reader should be created as XML 1.1 or
|
||
|
// XML 1.0 conformant
|
||
|
//
|
||
|
// fStandardUriConformant
|
||
|
// This flag controls whether we force conformant URI
|
||
|
// -----------------------------------------------------------------------
|
||
|
XMLEntityDecl* fCurEntity;
|
||
|
XMLReader* fCurReader;
|
||
|
XMLEntityHandler* fEntityHandler;
|
||
|
RefStackOf<XMLEntityDecl>* fEntityStack;
|
||
|
unsigned int fNextReaderNum;
|
||
|
RefStackOf<XMLReader>* fReaderStack;
|
||
|
bool fThrowEOE;
|
||
|
XMLReader::XMLVersion fXMLVersion;
|
||
|
bool fStandardUriConformant;
|
||
|
MemoryManager* fMemoryManager;
|
||
|
};
|
||
|
|
||
|
|
||
|
|
||
|
// ---------------------------------------------------------------------------
|
||
|
// ReaderMgr: Inlined methods
|
||
|
//
|
||
|
// NOTE: We cannot put these in alphabetical and type order as we usually
|
||
|
// do because some of the compilers we have to support are too stupid to
|
||
|
// understand out of order inlines!
|
||
|
// ---------------------------------------------------------------------------
|
||
|
inline XMLSize_t ReaderMgr::getCurrentReaderNum() const
|
||
|
{
|
||
|
return fCurReader->getReaderNum();
|
||
|
}
|
||
|
|
||
|
inline const XMLReader* ReaderMgr::getCurrentReader() const
|
||
|
{
|
||
|
return fCurReader;
|
||
|
}
|
||
|
|
||
|
inline XMLReader* ReaderMgr::getCurrentReader()
|
||
|
{
|
||
|
return fCurReader;
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::getName(XMLBuffer& toFill)
|
||
|
{
|
||
|
toFill.reset();
|
||
|
return fCurReader->getName(toFill, false);
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::getQName(XMLBuffer& toFill, int *colonPosition)
|
||
|
{
|
||
|
toFill.reset();
|
||
|
return fCurReader->getQName(toFill, colonPosition);
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::getNameToken(XMLBuffer& toFill)
|
||
|
{
|
||
|
toFill.reset();
|
||
|
return fCurReader->getName(toFill, true);
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::getNextCharIfNot(const XMLCh chNotToGet, XMLCh& chGotten)
|
||
|
{
|
||
|
return fCurReader->getNextCharIfNot(chNotToGet, chGotten);
|
||
|
}
|
||
|
|
||
|
inline void ReaderMgr::movePlainContentChars(XMLBuffer &dest)
|
||
|
{
|
||
|
fCurReader->movePlainContentChars(dest);
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::getThrowEOE() const
|
||
|
{
|
||
|
return fThrowEOE;
|
||
|
}
|
||
|
|
||
|
inline XMLFilePos ReaderMgr::getSrcOffset() const
|
||
|
{
|
||
|
return fCurReader? fCurReader->getSrcOffset() : 0;
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::lookingAtChar(const XMLCh chToCheck)
|
||
|
{
|
||
|
return (chToCheck == peekNextChar());
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::lookingAtSpace()
|
||
|
{
|
||
|
XMLCh c = peekNextChar();
|
||
|
return fCurReader->isWhitespace(c);
|
||
|
}
|
||
|
|
||
|
inline void ReaderMgr::setThrowEOE(const bool newValue)
|
||
|
{
|
||
|
fThrowEOE = newValue;
|
||
|
}
|
||
|
|
||
|
inline void ReaderMgr::setStandardUriConformant(const bool newValue)
|
||
|
{
|
||
|
fStandardUriConformant = newValue;
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::skippedString(const XMLCh* const toSkip)
|
||
|
{
|
||
|
return fCurReader->skippedString(toSkip);
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::skippedStringLong(const XMLCh* const toSkip)
|
||
|
{
|
||
|
return fCurReader->skippedStringLong(toSkip);
|
||
|
}
|
||
|
|
||
|
inline void ReaderMgr::skipToChar(const XMLCh toSkipTo)
|
||
|
{
|
||
|
XMLCh nextCh = 0;
|
||
|
do
|
||
|
{
|
||
|
// Get chars until we find the one to skip
|
||
|
nextCh = getNextChar();
|
||
|
}
|
||
|
// Break out at end of input or the char to skip
|
||
|
while((nextCh != toSkipTo) && nextCh!=0);
|
||
|
}
|
||
|
|
||
|
inline void ReaderMgr::skipPastChar(const XMLCh toSkipPast)
|
||
|
{
|
||
|
XMLCh nextCh = 0;
|
||
|
do
|
||
|
{
|
||
|
// Get chars until we find the one to skip
|
||
|
nextCh = getNextChar();
|
||
|
}
|
||
|
while((nextCh != toSkipPast) && nextCh!=0);
|
||
|
}
|
||
|
|
||
|
inline bool ReaderMgr::peekString(const XMLCh* const toPeek)
|
||
|
{
|
||
|
return fCurReader->peekString(toPeek);
|
||
|
}
|
||
|
|
||
|
inline void ReaderMgr::setEntityHandler(XMLEntityHandler* const newHandler)
|
||
|
{
|
||
|
fEntityHandler = newHandler;
|
||
|
}
|
||
|
|
||
|
inline void ReaderMgr::setXMLVersion(const XMLReader::XMLVersion version)
|
||
|
{
|
||
|
fXMLVersion = version;
|
||
|
fCurReader->setXMLVersion(version);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// This is a simple class to temporarily change the 'throw at end of entity'
|
||
|
// flag of the reader manager. There are some places where we need to
|
||
|
// turn this on and off on a scoped basis.
|
||
|
//
|
||
|
class XMLPARSER_EXPORT ThrowEOEJanitor
|
||
|
{
|
||
|
public :
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Constructors and destructor
|
||
|
// -----------------------------------------------------------------------
|
||
|
ThrowEOEJanitor(ReaderMgr* mgrTarget, const bool newValue) :
|
||
|
|
||
|
fOld(mgrTarget->getThrowEOE())
|
||
|
, fMgr(mgrTarget)
|
||
|
{
|
||
|
mgrTarget->setThrowEOE(newValue);
|
||
|
}
|
||
|
|
||
|
~ThrowEOEJanitor()
|
||
|
{
|
||
|
fMgr->setThrowEOE(fOld);
|
||
|
};
|
||
|
|
||
|
private :
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Unimplemented constructors and operators
|
||
|
// -----------------------------------------------------------------------
|
||
|
ThrowEOEJanitor(const ThrowEOEJanitor&);
|
||
|
ThrowEOEJanitor& operator=(const ThrowEOEJanitor&);
|
||
|
|
||
|
// -----------------------------------------------------------------------
|
||
|
// Private data members
|
||
|
//
|
||
|
// fOld
|
||
|
// The previous value of the flag, which we replaced during ctor,
|
||
|
// and will replace during dtor.
|
||
|
//
|
||
|
// fMgr
|
||
|
// A pointer to the reader manager we are going to set/reset the
|
||
|
// flag on.
|
||
|
// -----------------------------------------------------------------------
|
||
|
bool fOld;
|
||
|
ReaderMgr* fMgr;
|
||
|
};
|
||
|
|
||
|
XERCES_CPP_NAMESPACE_END
|
||
|
|
||
|
#endif
|