Bumping manifests a=b2g-bump
[gecko.git] / dom / base / NodeInfo.cpp
bloba87b8f65067cf9a45a7c0552da0d3fcb81542920
1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim: set ts=8 sts=2 et sw=2 tw=80: */
3 /* This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 /*
8 * Class that represents a prefix/namespace/localName triple; a single
9 * nodeinfo is shared by all elements in a document that have that
10 * prefix, namespace, and localName.
13 #include "mozilla/dom/NodeInfo.h"
14 #include "mozilla/dom/NodeInfoInlines.h"
16 #include "mozilla/ArrayUtils.h"
17 #include "mozilla/Likely.h"
19 #include "nsNodeInfoManager.h"
20 #include "nsCOMPtr.h"
21 #include "nsString.h"
22 #include "nsIAtom.h"
23 #include "nsDOMString.h"
24 #include "nsCRT.h"
25 #include "nsContentUtils.h"
26 #include "nsReadableUtils.h"
27 #include "nsAutoPtr.h"
28 #include "prprf.h"
29 #include "nsIDocument.h"
30 #include "nsGkAtoms.h"
31 #include "nsCCUncollectableMarker.h"
32 #include "nsNameSpaceManager.h"
34 using namespace mozilla;
35 using mozilla::dom::NodeInfo;
37 NodeInfo::~NodeInfo()
39 mOwnerManager->RemoveNodeInfo(this);
41 NS_RELEASE(mInner.mName);
42 NS_IF_RELEASE(mInner.mPrefix);
43 NS_IF_RELEASE(mInner.mExtraName);
46 NodeInfo::NodeInfo(nsIAtom *aName, nsIAtom *aPrefix, int32_t aNamespaceID,
47 uint16_t aNodeType, nsIAtom* aExtraName,
48 nsNodeInfoManager *aOwnerManager)
50 CheckValidNodeInfo(aNodeType, aName, aNamespaceID, aExtraName);
51 NS_ABORT_IF_FALSE(aOwnerManager, "Invalid aOwnerManager");
53 // Initialize mInner
54 NS_ADDREF(mInner.mName = aName);
55 NS_IF_ADDREF(mInner.mPrefix = aPrefix);
56 mInner.mNamespaceID = aNamespaceID;
57 mInner.mNodeType = aNodeType;
58 mOwnerManager = aOwnerManager;
59 NS_IF_ADDREF(mInner.mExtraName = aExtraName);
61 mDocument = aOwnerManager->GetDocument();
63 // Now compute our cached members.
65 // Qualified name. If we have no prefix, use ToString on
66 // mInner.mName so that we get to share its buffer.
67 if (aPrefix) {
68 mQualifiedName = nsDependentAtomString(mInner.mPrefix) +
69 NS_LITERAL_STRING(":") +
70 nsDependentAtomString(mInner.mName);
71 } else {
72 mInner.mName->ToString(mQualifiedName);
75 MOZ_ASSERT_IF(aNodeType != nsIDOMNode::ELEMENT_NODE &&
76 aNodeType != nsIDOMNode::ATTRIBUTE_NODE &&
77 aNodeType != UINT16_MAX,
78 aNamespaceID == kNameSpaceID_None && !aPrefix);
80 switch (aNodeType) {
81 case nsIDOMNode::ELEMENT_NODE:
82 case nsIDOMNode::ATTRIBUTE_NODE:
83 // Correct the case for HTML
84 if (aNodeType == nsIDOMNode::ELEMENT_NODE &&
85 aNamespaceID == kNameSpaceID_XHTML && GetDocument() &&
86 GetDocument()->IsHTML()) {
87 nsContentUtils::ASCIIToUpper(mQualifiedName, mNodeName);
88 } else {
89 mNodeName = mQualifiedName;
91 mInner.mName->ToString(mLocalName);
92 break;
93 case nsIDOMNode::TEXT_NODE:
94 case nsIDOMNode::CDATA_SECTION_NODE:
95 case nsIDOMNode::COMMENT_NODE:
96 case nsIDOMNode::DOCUMENT_NODE:
97 case nsIDOMNode::DOCUMENT_FRAGMENT_NODE:
98 mInner.mName->ToString(mNodeName);
99 SetDOMStringToNull(mLocalName);
100 break;
101 case nsIDOMNode::PROCESSING_INSTRUCTION_NODE:
102 case nsIDOMNode::DOCUMENT_TYPE_NODE:
103 mInner.mExtraName->ToString(mNodeName);
104 SetDOMStringToNull(mLocalName);
105 break;
106 default:
107 NS_ABORT_IF_FALSE(aNodeType == UINT16_MAX,
108 "Unknown node type");
113 // nsISupports
115 NS_IMPL_CYCLE_COLLECTION_CLASS(NodeInfo)
117 NS_IMPL_CYCLE_COLLECTION_UNLINK_0(NodeInfo)
119 static const char* kNSURIs[] = {
120 " ([none])",
121 " (xmlns)",
122 " (xml)",
123 " (xhtml)",
124 " (XLink)",
125 " (XSLT)",
126 " (XBL)",
127 " (MathML)",
128 " (RDF)",
129 " (XUL)"
132 NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INTERNAL(NodeInfo)
133 if (MOZ_UNLIKELY(cb.WantDebugInfo())) {
134 char name[72];
135 uint32_t nsid = tmp->NamespaceID();
136 nsAtomCString localName(tmp->NameAtom());
137 if (nsid < ArrayLength(kNSURIs)) {
138 PR_snprintf(name, sizeof(name), "NodeInfo%s %s", kNSURIs[nsid],
139 localName.get());
141 else {
142 PR_snprintf(name, sizeof(name), "NodeInfo %s", localName.get());
145 cb.DescribeRefCountedNode(tmp->mRefCnt.get(), name);
147 else {
148 NS_IMPL_CYCLE_COLLECTION_DESCRIBE(NodeInfo, tmp->mRefCnt.get())
151 NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mOwnerManager)
152 NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
154 NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_BEGIN(NodeInfo)
155 return nsCCUncollectableMarker::sGeneration && tmp->CanSkip();
156 NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_END
158 NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_BEGIN(NodeInfo)
159 return nsCCUncollectableMarker::sGeneration && tmp->CanSkip();
160 NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_END
162 NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_BEGIN(NodeInfo)
163 return nsCCUncollectableMarker::sGeneration && tmp->CanSkip();
164 NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_END
167 NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(NodeInfo, AddRef)
168 NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(NodeInfo, Release)
170 void
171 NodeInfo::GetName(nsAString& aName) const
173 mInner.mName->ToString(aName);
176 void
177 NodeInfo::GetPrefix(nsAString& aPrefix) const
179 if (mInner.mPrefix) {
180 mInner.mPrefix->ToString(aPrefix);
181 } else {
182 SetDOMStringToNull(aPrefix);
186 void
187 NodeInfo::GetNamespaceURI(nsAString& aNameSpaceURI) const
189 if (mInner.mNamespaceID > 0) {
190 nsresult rv =
191 nsContentUtils::NameSpaceManager()->GetNameSpaceURI(mInner.mNamespaceID,
192 aNameSpaceURI);
193 // How can we possibly end up with a bogus namespace ID here?
194 if (NS_FAILED(rv)) {
195 MOZ_CRASH();
197 } else {
198 SetDOMStringToNull(aNameSpaceURI);
202 bool
203 NodeInfo::NamespaceEquals(const nsAString& aNamespaceURI) const
205 int32_t nsid =
206 nsContentUtils::NameSpaceManager()->GetNameSpaceID(aNamespaceURI);
208 return mozilla::dom::NodeInfo::NamespaceEquals(nsid);
211 void
212 NodeInfo::DeleteCycleCollectable()
214 nsRefPtr<nsNodeInfoManager> kungFuDeathGrip = mOwnerManager;
215 delete this;
218 bool
219 NodeInfo::CanSkip()
221 return mDocument &&
222 nsCCUncollectableMarker::InGeneration(mDocument->GetMarkedCCGeneration());