/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* * The contents of this file are subject to the Netscape Public * License Version 1.1 (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.mozilla.org/NPL/ * * Software distributed under the License is distributed on an "AS * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or * implied. See the License for the specific language governing * rights and limitations under the License. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is Netscape * Communications Corporation. Portions created by Netscape are * Copyright (C) 1998 Netscape Communications Corporation. All * Rights Reserved. * * Contributor(s): * Pierre Phaneuf */ #include "nsDTDUtils.h" #include "CNavDTD.h" #include "nsIParserNode.h" #include "nsParserNode.h" #include "nsIObserverService.h" #include "nsIServiceManager.h" MOZ_DECL_CTOR_COUNTER(nsEntryStack); MOZ_DECL_CTOR_COUNTER(nsDTDContext); MOZ_DECL_CTOR_COUNTER(CTokenRecycler); MOZ_DECL_CTOR_COUNTER(CNodeRecycler); MOZ_DECL_CTOR_COUNTER(CObserverService); /************************************************************************************** A few notes about how residual style handling is performed: 1. The style stack contains nsTagEntry elements. 2. Every tag on the containment stack can have it's own residual style stack. 3. When a style leaks, it's mParent member is set to the level on the stack where it originated. A node with an mParent of 0 is not opened on tag stack, but is open on stylestack. 4. An easy way to tell that a container on the element stack is a residual style tag is that it's use count is >1. **************************************************************************************/ /** * Default constructor * @update harishd 04/04/99 * @update gess 04/22/99 */ nsEntryStack::nsEntryStack() { MOZ_COUNT_CTOR(nsEntryStack); mCapacity=0; mCount=0; mEntries=0; } /** * Default destructor * @update harishd 04/04/99 * @update gess 04/22/99 */ nsEntryStack::~nsEntryStack() { MOZ_COUNT_DTOR(nsEntryStack); if(mEntries) { //add code here to recycle the node if you have one... delete [] mEntries; mEntries=0; } mCount=mCapacity=0; } /** * Resets state of stack to be empty. * @update harishd 04/04/99 */ void nsEntryStack::Empty(void) { mCount=0; } /** * * @update gess 04/22/99 */ void nsEntryStack::EnsureCapacityFor(PRInt32 aNewMax,PRInt32 aShiftOffset) { if(mCapacitymUseCount++; ((nsCParserNode*)aNode)->mToken->mUseCount++; mEntries[mCount].mTag=(eHTMLTags)aNode->GetNodeType(); mEntries[mCount].mNode=(nsIParserNode*)aNode; mEntries[mCount].mParent=aStyleStack; mEntries[mCount++].mStyles=0; } } /** * This method inserts the given node onto the front of this stack * * @update gess 11/10/99 */ void nsEntryStack::PushFront(const nsIParserNode* aNode,nsEntryStack* aStyleStack) { if(aNode) { if(mCount0;index--) { mEntries[index]=mEntries[index-1]; } } else EnsureCapacityFor(mCount+1,1); ((nsCParserNode*)aNode)->mUseCount++; ((nsCParserNode*)aNode)->mToken->mUseCount++; mEntries[0].mTag=(eHTMLTags)aNode->GetNodeType(); mEntries[0].mNode=(nsIParserNode*)aNode; mEntries[0].mParent=aStyleStack; mEntries[0].mStyles=0; mCount++; } } /** * * @update gess 11/10/99 */ void nsEntryStack::Append(nsEntryStack *aStack) { if(aStack) { PRInt32 theCount=aStack->mCount; EnsureCapacityFor(mCount+aStack->mCount,0); PRInt32 theIndex=0; for(theIndex=0;theIndexmEntries[theIndex]; mEntries[mCount++].mParent=0; } } } /** * This method removes the node for the given tag * from anywhere within this entry stack, and shifts * other entries down. * * NOTE: It's odd to be removing an element from the middle * of a stack, but it's necessary because of how MALFORMED * html can be. * * anIndex: the index within the stack of the tag to be removed * aTag: the id of the tag to be removed * @update gess 02/25/00 */ nsIParserNode* nsEntryStack::Remove(PRInt32 anIndex,eHTMLTags aTag) { nsIParserNode* result=0; if((0mUseCount--; ((nsCParserNode*)result)->mToken->mUseCount--; PRInt32 theIndex=0; mCount-=1; for(theIndex=anIndex;theIndexmCount; PRUint32 sindex=0; nsTagEntry *theStyleEntry=theStyleStack->mEntries; for(sindex=scount-1;sindex>0;sindex--){ if(theStyleEntry->mTag==aTag) { theStyleEntry->mParent=0; //this tells us that the style is not open at any level break; } theStyleEntry++; } //for } } return result; } /** * * @update harishd 04/04/99 * @update gess 04/21/99 */ nsIParserNode* nsEntryStack::Pop(void) { nsIParserNode* result=0; if(0mUseCount--; ((nsCParserNode*)result)->mToken->mUseCount--; mEntries[mCount].mNode=0; mEntries[mCount].mStyles=0; nsEntryStack* theStyleStack=mEntries[mCount].mParent; if(theStyleStack) { //now we have to tell the residual style stack where this tag //originated that it's no longer in use. PRUint32 scount=theStyleStack->mCount; PRUint32 sindex=0; nsTagEntry *theStyleEntry=theStyleStack->mEntries; for(sindex=scount-1;sindex>0;sindex--){ if(theStyleEntry->mTag==mEntries[mCount].mTag) { theStyleEntry->mParent=0; //this tells us that the style is not open at any level break; } theStyleEntry++; } //for } } return result; } /** * * @update harishd 04/04/99 * @update gess 04/21/99 */ eHTMLTags nsEntryStack::First() const { eHTMLTags result=eHTMLTag_unknown; if(0mPrevious; delete theState; } } /** * * @update gess7/9/98 */ PRBool nsDTDContext::HasOpenContainer(eHTMLTags aTag) const { PRInt32 theIndex=mStack.LastOf(aTag); return PRBool(-1GetNodeType(); int size=mStack.mCount; if(size< eMaxTags) mXTags[size]=theTag; #endif mStack.Push((nsIParserNode*)aNode,aStyleStack); } } /** * @update gess 11/11/99, * harishd 04/04/99 */ nsIParserNode* nsDTDContext::Pop(nsEntryStack *&aChildStyleStack) { PRInt32 theSize=mStack.mCount; nsIParserNode* result=0; if(00) && (theSize <= eMaxTags)) mXTags[theSize-1]=eHTMLTag_unknown; #endif nsTagEntry* theEntry=mStack.EntryAt(mStack.mCount-1); if(theEntry) { aChildStyleStack=theEntry->mStyles; } result=mStack.Pop(); theEntry->mParent=0; } return result; } /** * * @update harishd 04/07/00 */ nsIParserNode* nsDTDContext::Pop() { nsEntryStack *theTempStyleStack=0; // This has no use here... return Pop(theTempStyleStack); } /** * * @update gess7/9/98 */ eHTMLTags nsDTDContext::First(void) const { return mStack.First(); } /** * * @update gess7/9/98 */ eHTMLTags nsDTDContext::TagAt(PRInt32 anIndex) const { return mStack.TagAt(anIndex); } /** * * @update gess7/9/98 */ nsTagEntry* nsDTDContext::LastEntry(void) const { return mStack.EntryAt(mStack.mCount-1); } /** * * @update gess7/9/98 */ eHTMLTags nsDTDContext::Last() const { return mStack.Last(); } /** * * @update gess7/9/98 */ nsEntryStack* nsDTDContext::GetStylesAt(PRInt32 anIndex) const { nsEntryStack* result=0; if(anIndexmStyles; } } return result; } /** * * @update gess 04/28/99 */ void nsDTDContext::PushStyle(const nsIParserNode* aNode){ nsTagEntry* theEntry=mStack.EntryAt(mStack.mCount-1); if(theEntry ) { nsEntryStack* theStack=theEntry->mStyles; if(!theStack) { theStack=theEntry->mStyles=new nsEntryStack(); } if(theStack) { theStack->Push(aNode); mResidualStyleCount++; } } //if } /** * Call this when you have an EntryStack full of styles * that you want to push at this level. * * @update gess 04/28/99 */ void nsDTDContext::PushStyles(nsEntryStack *aStyles){ if(aStyles) { nsTagEntry* theEntry=mStack.EntryAt(mStack.mCount-1); if(theEntry ) { nsEntryStack* theStyles=theEntry->mStyles; if(!theStyles) { theEntry->mStyles=aStyles; PRUint32 scount=aStyles->mCount; PRUint32 sindex=0; theEntry=aStyles->mEntries; for(sindex=0;sindexmParent=0; //this tells us that the style is not open at any level theEntry++; mResidualStyleCount++; } //for } else { theStyles->Append(aStyles); // Delete aStyles since it has been copied to theStyles... delete aStyles; aStyles=0; } } //if(theEntry ) else if(mStack.mCount==0) { // If you're here it means that we have hit the rock bottom // ,of the stack, and there's no need to handle anymore styles. // Fix for bug 29048 delete aStyles; aStyles=0; } }//if(aStyles) } /** * * @update gess 04/28/99 */ nsIParserNode* nsDTDContext::PopStyle(void){ nsIParserNode *result=0; nsTagEntry *theEntry=mStack.EntryAt(mStack.mCount-1); if(theEntry && (theEntry->mNode)) { nsEntryStack* theStyleStack=theEntry->mParent; if(theStyleStack){ result=theStyleStack->Pop(); mResidualStyleCount--; } } //if return result; } /** * * @update gess 04/28/99 */ nsIParserNode* nsDTDContext::PopStyle(eHTMLTags aTag){ PRInt32 theLevel=0; nsIParserNode* result=0; for(theLevel=mStack.mCount-1;theLevel>0;theLevel--) { nsEntryStack *theStack=mStack.mEntries[theLevel].mStyles; if(theStack) { if(aTag==theStack->Last()) { result=theStack->Pop(); mResidualStyleCount--; } else { // NS_ERROR("bad residual style entry"); } } } return result; } /** * * This is similar to popstyle, except that it removes the * style tag given from anywhere in the style stack, and * not just at the top. * * @update gess 01/26/00 */ nsIParserNode* nsDTDContext::RemoveStyle(eHTMLTags aTag){ PRInt32 theLevel=0; nsIParserNode* result=0; for(theLevel=mStack.mCount-1;theLevel>0;theLevel--) { nsEntryStack *theStack=mStack.mEntries[theLevel].mStyles; if(theStack) { if(aTag==theStack->Last()) { result=theStack->Pop(); mResidualStyleCount--; } else { // NS_ERROR("bad residual style entry"); } } } return result; } /** * * @update harishd 04/10/00 */ nsresult nsDTDContext::GetNodeRecycler(CNodeRecycler*& aNodeRecycler){ nsresult result=NS_OK; if(!mNodeRecycler) { mNodeRecycler=new CNodeRecycler(); if(mNodeRecycler==0) result=NS_ERROR_OUT_OF_MEMORY; } aNodeRecycler=mNodeRecycler; return result; } /** * * @update hairshd 04/10/00 */ void nsDTDContext::FreeNodeRecycler(){ if(mNodeRecycler) { delete mNodeRecycler; } } /************************************************************** Now define the tokenrecycler class... **************************************************************/ /** * * @update gess7/25/98 * @param */ CTokenRecycler::CTokenRecycler() : nsITokenRecycler() { MOZ_COUNT_CTOR(CTokenRecycler); int i=0; for(i=0;iForEach(theDeallocator); delete mTokenCache[i]; mTokenCache[i]=0; } } } class CTokenFinder: public nsDequeFunctor{ public: CTokenFinder(CToken* aToken) {mToken=aToken;} virtual void* operator()(void* anObject) { if(anObject==mToken) { return anObject; } return 0; } CToken* mToken; }; /** * This method gets called when someone wants to recycle a token * @update gess7/24/98 * @param aToken -- token to be recycled. * @return nada */ void CTokenRecycler::RecycleToken(CToken* aToken) { if(aToken) { PRInt32 theType=aToken->GetTokenType(); #if 0 //This should be disabled since it's only debug code. CTokenFinder finder(aToken); CToken* theMatch; theMatch=(CToken*)mTokenCache[theType-1]->FirstThat(finder); if(theMatch) { printf("dup token: %p\n",theMatch); } #endif aToken->mUseCount=1; mTokenCache[theType-1]->Push(aToken); } } /** * * @update vidur 11/12/98 * @param * @return */ CToken* CTokenRecycler::CreateTokenOfType(eHTMLTokenTypes aType,eHTMLTags aTag, const nsString& aString) { CToken* result=(CToken*)mTokenCache[aType-1]->Pop(); if(result) { result->Reinitialize(aTag,aString); } else { #ifdef NS_DEBUG mTotals[aType-1]++; #endif switch(aType){ case eToken_start: result=new CStartToken(aTag); break; case eToken_end: result=new CEndToken(aTag); break; case eToken_comment: result=new CCommentToken(); break; case eToken_entity: result=new CEntityToken(aString); break; case eToken_whitespace: result=new CWhitespaceToken(); break; case eToken_newline: result=new CNewlineToken(); break; case eToken_text: result=new CTextToken(aString); break; case eToken_attribute: result=new CAttributeToken(); break; case eToken_script: result=new CScriptToken(); break; case eToken_style: result=new CStyleToken(); break; case eToken_skippedcontent: result=new CSkippedContentToken(aString); break; case eToken_instruction: result=new CInstructionToken(); break; case eToken_cdatasection: result=new CCDATASectionToken(); break; case eToken_error: result=new CErrorToken(); break; case eToken_doctypeDecl: result=new CDoctypeDeclToken(); break; default: break; } } return result; } /** * * @update vidur 11/12/98 * @param * @return */ CToken* CTokenRecycler::CreateTokenOfType(eHTMLTokenTypes aType,eHTMLTags aTag) { CToken* result=(CToken*)mTokenCache[aType-1]->Pop(); if(result) { result->Reinitialize(aTag,mEmpty); } else { #ifdef NS_DEBUG mTotals[aType-1]++; #endif switch(aType){ case eToken_start: result=new CStartToken(aTag); break; case eToken_end: result=new CEndToken(aTag); break; case eToken_comment: result=new CCommentToken(); break; case eToken_attribute: result=new CAttributeToken(); break; case eToken_entity: result=new CEntityToken(); break; case eToken_whitespace: result=new CWhitespaceToken(); break; case eToken_newline: result=new CNewlineToken(); break; case eToken_text: result=new CTextToken(mEmpty); break; case eToken_script: result=new CScriptToken(); break; case eToken_style: result=new CStyleToken(); break; case eToken_skippedcontent: result=new CSkippedContentToken(mEmpty); break; case eToken_instruction: result=new CInstructionToken(); break; case eToken_cdatasection: result=new CCDATASectionToken(); break; case eToken_error: result=new CErrorToken(); break; case eToken_doctypeDecl: result=new CDoctypeDeclToken(aTag); break; default: break; } } return result; } CNodeRecycler::CNodeRecycler(): mSharedNodes(0) { MOZ_COUNT_CTOR(CTokenRecycler); #ifdef NS_DEBUG gNodeCount=0; #endif } CNodeRecycler::~CNodeRecycler() { MOZ_COUNT_DTOR(CTokenRecycler); nsCParserNode* theNode=0; #ifdef NS_DEBUG #if 0 PRInt32 count=gNodeCount-mSharedNodes.GetSize(); if(count) { printf("%i of %i nodes leaked!\n",count,gNodeCount); } #endif #endif while((theNode=(nsCParserNode*)mSharedNodes.Pop())){ delete theNode; } } void CNodeRecycler::RecycleNode(nsCParserNode* aNode,nsITokenRecycler* aTokenRecycler) { if(aNode && (!aNode->mUseCount)) { // If the node contains tokens there better me a token recycler.. if(aTokenRecycler) { if(aNode->mToken) { if(!aNode->mToken->mUseCount) { aTokenRecycler->RecycleToken(aNode->mToken); } } CToken* theToken=0; while((theToken=(CToken*)aNode->PopAttributeToken())){ if(!theToken->mUseCount) { aTokenRecycler->RecycleToken(theToken); } } } mSharedNodes.Push(aNode); } } nsCParserNode* CNodeRecycler::CreateNode(void) { nsCParserNode* result=0; if(0 const char* prefix=" "; fstream out(aFilename,ios::out); out << "==================================================" << endl; out << aTitle << endl; out << "=================================================="; int i,j=0; int written; for(i=1;i" << endl; out << prefix; written=0; if(theDTD.IsContainer(i)) { for(j=1;j> 24) ^ *data_blk_ptr++ ) & 0xff; crc_accum = ( crc_accum << 8 ) ^ crc_table[i]; } return crc_accum; } /************************************************************** Define the nsIElementObserver release class... **************************************************************/ class nsObserverReleaser: public nsDequeFunctor{ public: virtual void* operator()(void* anObject) { nsIElementObserver* theObserver= (nsIElementObserver*)anObject; NS_RELEASE(theObserver); return 0; } }; /************************************************************** This defines the topic object used by the observer service. The observerService uses a list of these, 1 per topic when registering tags. **************************************************************/ nsObserverTopic::nsObserverTopic(const nsString& aTopic) : mTopic(aTopic), mKeys(0), mValues(0) { nsCRT::zero(mObservers,sizeof(mObservers)); mCharsetKey.AssignWithConversion("charset"); mSourceKey.AssignWithConversion("charsetSource"); mDTDKey.AssignWithConversion("X_COMMAND"); } nsObserverTopic::~nsObserverTopic() { nsObserverReleaser theReleaser; PRInt32 theIndex=0; for(theIndex=0;theIndex<=NS_HTML_TAG_MAX;theIndex++){ if(mObservers[theIndex]){ mObservers[theIndex]->ForEach(theReleaser); delete mObservers[theIndex]; mObservers[theIndex]=0; } } } PRBool nsObserverTopic::Matches(const nsString& aString) { PRBool result=aString.Equals(mTopic); return result; } nsDeque* nsObserverTopic::GetObserversForTag(eHTMLTags aTag) { if(aTag <= NS_HTML_TAG_MAX) { return mObservers[aTag]; } return 0; } void nsObserverTopic::RegisterObserverForTag(nsIElementObserver *anObserver,eHTMLTags aTag) { if(anObserver) { if(mObservers[aTag] == nsnull) { mObservers[aTag] = new nsDeque(0); } NS_ADDREF(anObserver); mObservers[aTag]->Push(anObserver); } } /** * This method will notify observers registered for specific tags. * * @update harishd 08/29/99 * @param aTag - The tag for which observers could be waiting for. * @param aNode - * @param aUniqueID - The document ID. * @param aDTD - The current DTD. * @param aCharsetValue - * @param aCharsetSource - * @return if SUCCESS return NS_OK else return ERROR code. */ nsresult nsObserverTopic::Notify(eHTMLTags aTag,nsIParserNode& aNode,void* aUniqueID,nsIParser* aParser) { nsresult result=NS_OK; nsDeque* theDeque=GetObserversForTag(aTag); if(theDeque){ nsAutoString theCharsetValue; nsCharsetSource theCharsetSource; aParser->GetDocumentCharset(theCharsetValue,theCharsetSource); PRInt32 theAttrCount =aNode.GetAttributeCount(); PRUint32 theDequeSize=theDeque->GetSize(); if(0FirstThat(theNotifier); result=theNotifier.mResult; }//if } return result; } /****************************************************************************** This class is used to store ref's to tag observers during the parse phase. Note that for simplicity, this is a singleton that is constructed in the CNavDTD and shared for the duration of the application session. Later on it might be nice to use a more dynamic approach that would permit observers to come and go on a document basis. I changed the observerservice to store topics so that we can distinguish observers by topic. Up till now, they've all just be thrown into the same observer list, which was wrong. This fixes bug #28825. ******************************************************************************/ CObserverService::CObserverService() : mTopics(0) { MOZ_COUNT_CTOR(CObserverService); nsAutoString theHTMLTopic; theHTMLTopic.AssignWithConversion(kHTMLTextContentType); RegisterObservers(theHTMLTopic); nsAutoString theXMLTopic; theXMLTopic.AssignWithConversion(kXMLTextContentType); //use the mimetype for the topic RegisterObservers(theXMLTopic); } CObserverService::~CObserverService() { MOZ_COUNT_DTOR(CObserverService); nsObserverTopic *theTopic=(nsObserverTopic*)mTopics.Pop(); while(theTopic) { delete theTopic; theTopic=(nsObserverTopic*)mTopics.Pop(); } } nsObserverTopic* CObserverService::GetTopic(const nsString& aTopic) { PRInt32 theIndex=0; nsObserverTopic *theTopic=(nsObserverTopic*)mTopics.ObjectAt(theIndex++); while(theTopic) { if(theTopic->Matches(aTopic)) return theTopic; theTopic=(nsObserverTopic*)mTopics.ObjectAt(theIndex++); } return 0; } nsObserverTopic* CObserverService::CreateTopic(const nsString& aTopic) { nsObserverTopic* theTopic=new nsObserverTopic(aTopic); mTopics.Push(theTopic); return theTopic; } /** * This method will maintain lists of observers registered for specific tags. * * @update rickg 03.23.2000 * @param aTopic - The topic under which observers register for. * @return if SUCCESS return NS_OK else return ERROR code. */ void CObserverService::RegisterObservers(const nsString& aTopic) { nsresult result = NS_OK; nsIObserverService* theObserverService = nsnull; result = nsServiceManager::GetService(NS_OBSERVERSERVICE_PROGID, NS_GET_IID(nsIObserverService), (nsISupports**) &theObserverService, nsnull); if(result == NS_OK){ nsIEnumerator* theEnum = nsnull; result = theObserverService->EnumerateObserverList(aTopic.GetUnicode(), &theEnum); nsServiceManager::ReleaseService(NS_OBSERVERSERVICE_PROGID, theObserverService); if(result == NS_OK) { nsCOMPtr theElementObserver; nsISupports *inst = nsnull; nsObserverTopic *theTopic=0; for (theEnum->First(); theEnum->IsDone() != NS_OK; theEnum->Next()) { result = theEnum->CurrentItem(&inst); if (NS_SUCCEEDED(result)) { theElementObserver = do_QueryInterface(inst, &result); NS_RELEASE(inst); } if (result == NS_OK) { const char* theTagStr = nsnull; PRUint32 theTagIndex = 0; theTagStr = theElementObserver->GetTagNameAt(theTagIndex); while (theTagStr != nsnull) { eHTMLTags theTag = nsHTMLTags::LookupTag(nsCAutoString(theTagStr)); if((eHTMLTag_userdefined!=theTag) && (theTag <= NS_HTML_TAG_MAX)){ theTopic=GetTopic(aTopic); if(!theTopic) theTopic=CreateTopic(aTopic); if(theTopic) { theTopic->RegisterObserverForTag(theElementObserver,theTag); } } theTagIndex++; theTagStr = theElementObserver->GetTagNameAt(theTagIndex); } } } } NS_IF_RELEASE(theEnum); } } /** * This method will maintain lists of observers registered for specific tags. * * @update rickg 03.23.2000 * @param aTopic - The topic under which observers register for. * @return if SUCCESS return NS_OK else return ERROR code. */ void CObserverService::UnregisterObservers(const nsString& aTopic) { } /** * This method will notify observers registered for specific tags. * * @update rickg 03.23.2000 * @param aTag - The tag for which observers could be waiting for. * @param aNode - * @param aUniqueID - The document ID. * @param aDTD - The current DTD. * @param aCharsetValue - * @param aCharsetSource - * @return if SUCCESS return NS_OK else return ERROR code. */ nsresult CObserverService::Notify( eHTMLTags aTag, nsIParserNode& aNode, void* aUniqueID, const nsString& aTopic, nsIParser* aParser) { nsresult result=NS_OK; nsObserverTopic *theTopic=GetTopic(aTopic); if(theTopic) { result=theTopic->Notify(aTag,aNode,aUniqueID,aParser); } return result; } /** * This method will look for the list of observers registered for * a specific tag. * * @update rickg 03.23.2000 * @param aTag - The tag for which observers could be waiting for. * @return if FOUND return "observer list" else return nsnull; * */ nsDeque* CObserverService::GetObserversForTagInTopic(eHTMLTags aTag,const nsString& aTopic) { nsObserverTopic *theTopic=GetTopic(aTopic); if(theTopic) { return theTopic->GetObserversForTag(aTag); } return 0; }