2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 04:12:37 -07:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Class that represents a prefix/namespace/localName triple; a single
|
|
|
|
* nodeinfo is shared by all elements in a document that have that
|
|
|
|
* prefix, namespace, and localName.
|
|
|
|
*/
|
|
|
|
|
2011-10-10 22:50:08 -07:00
|
|
|
#include "mozilla/Util.h"
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "nscore.h"
|
|
|
|
#include "nsNodeInfo.h"
|
|
|
|
#include "nsNodeInfoManager.h"
|
|
|
|
#include "nsCOMPtr.h"
|
|
|
|
#include "nsString.h"
|
|
|
|
#include "nsIAtom.h"
|
|
|
|
#include "nsDOMString.h"
|
|
|
|
#include "nsCRT.h"
|
|
|
|
#include "nsContentUtils.h"
|
|
|
|
#include "nsReadableUtils.h"
|
|
|
|
#include "nsAutoPtr.h"
|
|
|
|
#include NEW_H
|
|
|
|
#include "nsFixedSizeAllocator.h"
|
2010-09-14 21:46:40 -07:00
|
|
|
#include "prprf.h"
|
2011-07-20 12:18:54 -07:00
|
|
|
#include "nsIDocument.h"
|
|
|
|
#include "nsGkAtoms.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-10-10 22:50:08 -07:00
|
|
|
using namespace mozilla;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
static const size_t kNodeInfoPoolSizes[] = {
|
|
|
|
sizeof(nsNodeInfo)
|
|
|
|
};
|
|
|
|
|
2012-04-18 21:15:23 -07:00
|
|
|
static const PRInt32 kNodeInfoPoolInitialSize = sizeof(nsNodeInfo) * 64;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// static
|
2012-07-30 07:20:58 -07:00
|
|
|
nsFixedSizeAllocator* nsNodeInfo::sNodeInfoPool = nullptr;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// static
|
|
|
|
nsNodeInfo*
|
2011-05-05 09:26:32 -07:00
|
|
|
nsNodeInfo::Create(nsIAtom *aName, nsIAtom *aPrefix, PRInt32 aNamespaceID,
|
2011-06-14 00:56:49 -07:00
|
|
|
PRUint16 aNodeType, nsIAtom *aExtraName,
|
2011-05-05 09:26:32 -07:00
|
|
|
nsNodeInfoManager *aOwnerManager)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
if (!sNodeInfoPool) {
|
|
|
|
sNodeInfoPool = new nsFixedSizeAllocator();
|
|
|
|
if (!sNodeInfoPool)
|
2012-07-30 07:20:58 -07:00
|
|
|
return nullptr;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsresult rv = sNodeInfoPool->Init("NodeInfo Pool", kNodeInfoPoolSizes,
|
|
|
|
1, kNodeInfoPoolInitialSize);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
delete sNodeInfoPool;
|
2012-07-30 07:20:58 -07:00
|
|
|
sNodeInfoPool = nullptr;
|
|
|
|
return nullptr;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new one
|
|
|
|
void* place = sNodeInfoPool->Alloc(sizeof(nsNodeInfo));
|
2011-05-05 09:26:32 -07:00
|
|
|
return place ?
|
2011-06-14 00:56:49 -07:00
|
|
|
new (place) nsNodeInfo(aName, aPrefix, aNamespaceID, aNodeType, aExtraName,
|
|
|
|
aOwnerManager) :
|
2012-07-30 07:20:58 -07:00
|
|
|
nullptr;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nsNodeInfo::~nsNodeInfo()
|
|
|
|
{
|
2011-05-05 09:26:32 -07:00
|
|
|
mOwnerManager->RemoveNodeInfo(this);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-05-05 09:26:32 -07:00
|
|
|
NS_RELEASE(mInner.mName);
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_IF_RELEASE(mInner.mPrefix);
|
2011-06-14 00:56:49 -07:00
|
|
|
NS_IF_RELEASE(mInner.mExtraName);
|
2011-07-26 04:11:14 -07:00
|
|
|
NS_RELEASE(mOwnerManager);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-05 09:26:32 -07:00
|
|
|
nsNodeInfo::nsNodeInfo(nsIAtom *aName, nsIAtom *aPrefix, PRInt32 aNamespaceID,
|
2011-06-14 00:56:49 -07:00
|
|
|
PRUint16 aNodeType, nsIAtom* aExtraName,
|
2011-05-05 09:26:32 -07:00
|
|
|
nsNodeInfoManager *aOwnerManager)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2012-08-16 08:07:55 -07:00
|
|
|
CheckValidNodeInfo(aNodeType, aName, aNamespaceID, aExtraName);
|
2011-06-14 00:56:49 -07:00
|
|
|
NS_ABORT_IF_FALSE(aOwnerManager, "Invalid aOwnerManager");
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-06-14 00:56:49 -07:00
|
|
|
// Initialize mInner
|
|
|
|
NS_ADDREF(mInner.mName = aName);
|
|
|
|
NS_IF_ADDREF(mInner.mPrefix = aPrefix);
|
2007-03-22 10:30:00 -07:00
|
|
|
mInner.mNamespaceID = aNamespaceID;
|
2011-06-14 00:56:49 -07:00
|
|
|
mInner.mNodeType = aNodeType;
|
|
|
|
NS_ADDREF(mOwnerManager = aOwnerManager);
|
|
|
|
NS_IF_ADDREF(mInner.mExtraName = aExtraName);
|
2011-05-05 09:26:32 -07:00
|
|
|
|
2011-06-14 17:58:57 -07:00
|
|
|
mDocument = aOwnerManager->GetDocument();
|
|
|
|
|
2011-05-05 09:26:32 -07:00
|
|
|
// Now compute our cached members.
|
|
|
|
|
|
|
|
// Qualified name. If we have no prefix, use ToString on
|
|
|
|
// mInner.mName so that we get to share its buffer.
|
|
|
|
if (aPrefix) {
|
2011-06-14 00:56:49 -07:00
|
|
|
mQualifiedName = nsDependentAtomString(mInner.mPrefix) +
|
|
|
|
NS_LITERAL_STRING(":") +
|
|
|
|
nsDependentAtomString(mInner.mName);
|
2011-05-05 09:26:32 -07:00
|
|
|
} else {
|
|
|
|
mInner.mName->ToString(mQualifiedName);
|
|
|
|
}
|
|
|
|
|
2011-06-14 00:56:49 -07:00
|
|
|
switch (aNodeType) {
|
|
|
|
case nsIDOMNode::ELEMENT_NODE:
|
|
|
|
case nsIDOMNode::ATTRIBUTE_NODE:
|
2011-06-14 00:56:49 -07:00
|
|
|
// Correct the case for HTML
|
|
|
|
if (aNodeType == nsIDOMNode::ELEMENT_NODE &&
|
|
|
|
aNamespaceID == kNameSpaceID_XHTML && GetDocument() &&
|
|
|
|
GetDocument()->IsHTML()) {
|
|
|
|
nsContentUtils::ASCIIToUpper(mQualifiedName, mNodeName);
|
|
|
|
} else {
|
|
|
|
mNodeName = mQualifiedName;
|
|
|
|
}
|
2011-06-14 00:56:49 -07:00
|
|
|
mInner.mName->ToString(mLocalName);
|
|
|
|
break;
|
|
|
|
case nsIDOMNode::TEXT_NODE:
|
|
|
|
case nsIDOMNode::CDATA_SECTION_NODE:
|
|
|
|
case nsIDOMNode::COMMENT_NODE:
|
|
|
|
case nsIDOMNode::DOCUMENT_NODE:
|
|
|
|
case nsIDOMNode::DOCUMENT_FRAGMENT_NODE:
|
|
|
|
mInner.mName->ToString(mNodeName);
|
|
|
|
SetDOMStringToNull(mLocalName);
|
|
|
|
break;
|
|
|
|
case nsIDOMNode::PROCESSING_INSTRUCTION_NODE:
|
|
|
|
case nsIDOMNode::DOCUMENT_TYPE_NODE:
|
|
|
|
mInner.mExtraName->ToString(mNodeName);
|
|
|
|
SetDOMStringToNull(mLocalName);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
NS_ABORT_IF_FALSE(aNodeType == PR_UINT16_MAX,
|
|
|
|
"Unknown node type");
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// nsISupports
|
|
|
|
|
2008-03-28 07:09:00 -07:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(nsNodeInfo)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_0(nsNodeInfo)
|
2009-02-27 06:48:26 -08:00
|
|
|
|
|
|
|
static const char* kNSURIs[] = {
|
|
|
|
" ([none])",
|
|
|
|
" (xmlns)",
|
|
|
|
" (xml)",
|
|
|
|
" (xhtml)",
|
|
|
|
" (XLink)",
|
|
|
|
" (XSLT)",
|
|
|
|
" (XBL)",
|
|
|
|
" (MathML)",
|
|
|
|
" (RDF)",
|
|
|
|
" (XUL)"
|
|
|
|
};
|
|
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INTERNAL(nsNodeInfo)
|
|
|
|
if (NS_UNLIKELY(cb.WantDebugInfo())) {
|
|
|
|
char name[72];
|
|
|
|
PRUint32 nsid = tmp->NamespaceID();
|
|
|
|
nsAtomCString localName(tmp->NameAtom());
|
2011-10-10 22:50:08 -07:00
|
|
|
if (nsid < ArrayLength(kNSURIs)) {
|
2009-02-27 06:48:26 -08:00
|
|
|
PR_snprintf(name, sizeof(name), "nsNodeInfo%s %s", kNSURIs[nsid],
|
|
|
|
localName.get());
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
PR_snprintf(name, sizeof(name), "nsNodeInfo %s", localName.get());
|
|
|
|
}
|
|
|
|
|
2011-06-23 14:10:52 -07:00
|
|
|
cb.DescribeRefCountedNode(tmp->mRefCnt.get(), sizeof(nsNodeInfo), name);
|
2009-02-27 06:48:26 -08:00
|
|
|
}
|
|
|
|
else {
|
2011-06-23 14:10:52 -07:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_DESCRIBE(nsNodeInfo, tmp->mRefCnt.get())
|
2009-02-27 06:48:26 -08:00
|
|
|
}
|
|
|
|
|
2012-06-26 13:35:29 -07:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_RAWPTR(mOwnerManager)
|
2008-03-28 07:09:00 -07:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(nsNodeInfo)
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE_WITH_DESTROY(nsNodeInfo, LastRelease())
|
|
|
|
NS_INTERFACE_TABLE_HEAD(nsNodeInfo)
|
|
|
|
NS_INTERFACE_TABLE1(nsNodeInfo, nsINodeInfo)
|
|
|
|
NS_INTERFACE_TABLE_TO_MAP_SEGUE_CYCLE_COLLECTION(nsNodeInfo)
|
|
|
|
NS_INTERFACE_MAP_END
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// nsINodeInfo
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsNodeInfo::GetNamespaceURI(nsAString& aNameSpaceURI) const
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
if (mInner.mNamespaceID > 0) {
|
|
|
|
rv = nsContentUtils::NameSpaceManager()->GetNameSpaceURI(mInner.mNamespaceID,
|
|
|
|
aNameSpaceURI);
|
|
|
|
} else {
|
|
|
|
SetDOMStringToNull(aNameSpaceURI);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool
|
2007-03-22 10:30:00 -07:00
|
|
|
nsNodeInfo::NamespaceEquals(const nsAString& aNamespaceURI) const
|
|
|
|
{
|
|
|
|
PRInt32 nsid =
|
|
|
|
nsContentUtils::NameSpaceManager()->GetNameSpaceID(aNamespaceURI);
|
|
|
|
|
|
|
|
return nsINodeInfo::NamespaceEquals(nsid);
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
void
|
|
|
|
nsNodeInfo::ClearCache()
|
|
|
|
{
|
|
|
|
// Clear our cache.
|
|
|
|
delete sNodeInfoPool;
|
2012-07-30 07:20:58 -07:00
|
|
|
sNodeInfoPool = nullptr;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsNodeInfo::LastRelease()
|
|
|
|
{
|
|
|
|
nsRefPtr<nsNodeInfoManager> kungFuDeathGrip = mOwnerManager;
|
|
|
|
this->~nsNodeInfo();
|
|
|
|
|
|
|
|
// The refcount balancing and destructor re-entrancy protection
|
|
|
|
// code in Release() sets mRefCnt to 1 so we have to set it to 0
|
|
|
|
// here to prevent leaks
|
|
|
|
mRefCnt = 0;
|
|
|
|
|
|
|
|
NS_ASSERTION(sNodeInfoPool, "No NodeInfoPool when deleting NodeInfo!!!");
|
|
|
|
sNodeInfoPool->Free(this, sizeof(nsNodeInfo));
|
|
|
|
}
|