xref: /aosp_15_r20/frameworks/base/tools/aapt2/xml/XmlDom.h (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker  * Copyright (C) 2015 The Android Open Source Project
3*d57664e9SAndroid Build Coastguard Worker  *
4*d57664e9SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*d57664e9SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*d57664e9SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*d57664e9SAndroid Build Coastguard Worker  *
8*d57664e9SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*d57664e9SAndroid Build Coastguard Worker  *
10*d57664e9SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*d57664e9SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*d57664e9SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*d57664e9SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*d57664e9SAndroid Build Coastguard Worker  * limitations under the License.
15*d57664e9SAndroid Build Coastguard Worker  */
16*d57664e9SAndroid Build Coastguard Worker 
17*d57664e9SAndroid Build Coastguard Worker #ifndef AAPT_XML_DOM_H
18*d57664e9SAndroid Build Coastguard Worker #define AAPT_XML_DOM_H
19*d57664e9SAndroid Build Coastguard Worker 
20*d57664e9SAndroid Build Coastguard Worker #include <memory>
21*d57664e9SAndroid Build Coastguard Worker #include <string>
22*d57664e9SAndroid Build Coastguard Worker #include <vector>
23*d57664e9SAndroid Build Coastguard Worker 
24*d57664e9SAndroid Build Coastguard Worker #include "Resource.h"
25*d57664e9SAndroid Build Coastguard Worker #include "ResourceValues.h"
26*d57664e9SAndroid Build Coastguard Worker #include "androidfw/IDiagnostics.h"
27*d57664e9SAndroid Build Coastguard Worker #include "androidfw/Streams.h"
28*d57664e9SAndroid Build Coastguard Worker #include "androidfw/StringPiece.h"
29*d57664e9SAndroid Build Coastguard Worker #include "util/Util.h"
30*d57664e9SAndroid Build Coastguard Worker #include "xml/XmlUtil.h"
31*d57664e9SAndroid Build Coastguard Worker 
32*d57664e9SAndroid Build Coastguard Worker namespace aapt {
33*d57664e9SAndroid Build Coastguard Worker namespace xml {
34*d57664e9SAndroid Build Coastguard Worker 
35*d57664e9SAndroid Build Coastguard Worker class Element;
36*d57664e9SAndroid Build Coastguard Worker class Visitor;
37*d57664e9SAndroid Build Coastguard Worker class ConstVisitor;
38*d57664e9SAndroid Build Coastguard Worker 
39*d57664e9SAndroid Build Coastguard Worker // Base class for all XML nodes.
40*d57664e9SAndroid Build Coastguard Worker class Node {
41*d57664e9SAndroid Build Coastguard Worker  public:
42*d57664e9SAndroid Build Coastguard Worker   virtual ~Node() = default;
43*d57664e9SAndroid Build Coastguard Worker 
44*d57664e9SAndroid Build Coastguard Worker   Element* parent = nullptr;
45*d57664e9SAndroid Build Coastguard Worker   size_t line_number = 0u;
46*d57664e9SAndroid Build Coastguard Worker   size_t column_number = 0u;
47*d57664e9SAndroid Build Coastguard Worker   std::string comment;
48*d57664e9SAndroid Build Coastguard Worker 
49*d57664e9SAndroid Build Coastguard Worker   virtual void Accept(Visitor* visitor) = 0;
50*d57664e9SAndroid Build Coastguard Worker   virtual void Accept(ConstVisitor* visitor) const = 0;
51*d57664e9SAndroid Build Coastguard Worker 
52*d57664e9SAndroid Build Coastguard Worker   using ElementCloneFunc = std::function<void(const Element&, Element*)>;
53*d57664e9SAndroid Build Coastguard Worker 
54*d57664e9SAndroid Build Coastguard Worker   // Clones the Node subtree, using the given function to decide how to clone an Element.
55*d57664e9SAndroid Build Coastguard Worker   virtual std::unique_ptr<Node> Clone(const ElementCloneFunc& el_cloner) const = 0;
56*d57664e9SAndroid Build Coastguard Worker };
57*d57664e9SAndroid Build Coastguard Worker 
58*d57664e9SAndroid Build Coastguard Worker // A namespace declaration (xmlns:prefix="uri").
59*d57664e9SAndroid Build Coastguard Worker struct NamespaceDecl {
60*d57664e9SAndroid Build Coastguard Worker   std::string prefix;
61*d57664e9SAndroid Build Coastguard Worker   std::string uri;
62*d57664e9SAndroid Build Coastguard Worker   size_t line_number = 0u;
63*d57664e9SAndroid Build Coastguard Worker   size_t column_number = 0u;
64*d57664e9SAndroid Build Coastguard Worker };
65*d57664e9SAndroid Build Coastguard Worker 
66*d57664e9SAndroid Build Coastguard Worker struct AaptAttribute {
67*d57664e9SAndroid Build Coastguard Worker   explicit AaptAttribute(const ::aapt::Attribute& attr, const std::optional<ResourceId>& resid = {})
attributeAaptAttribute68*d57664e9SAndroid Build Coastguard Worker       : attribute(attr), id(resid) {
69*d57664e9SAndroid Build Coastguard Worker   }
70*d57664e9SAndroid Build Coastguard Worker 
71*d57664e9SAndroid Build Coastguard Worker   aapt::Attribute attribute;
72*d57664e9SAndroid Build Coastguard Worker   std::optional<ResourceId> id;
73*d57664e9SAndroid Build Coastguard Worker };
74*d57664e9SAndroid Build Coastguard Worker 
75*d57664e9SAndroid Build Coastguard Worker // An XML attribute.
76*d57664e9SAndroid Build Coastguard Worker struct Attribute {
77*d57664e9SAndroid Build Coastguard Worker   std::string namespace_uri;
78*d57664e9SAndroid Build Coastguard Worker   std::string name;
79*d57664e9SAndroid Build Coastguard Worker   std::string value;
80*d57664e9SAndroid Build Coastguard Worker 
81*d57664e9SAndroid Build Coastguard Worker   std::optional<AaptAttribute> compiled_attribute;
82*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<Item> compiled_value;
83*d57664e9SAndroid Build Coastguard Worker };
84*d57664e9SAndroid Build Coastguard Worker 
85*d57664e9SAndroid Build Coastguard Worker // An Element XML node.
86*d57664e9SAndroid Build Coastguard Worker class Element : public Node {
87*d57664e9SAndroid Build Coastguard Worker  public:
88*d57664e9SAndroid Build Coastguard Worker   // Ordered namespace prefix declarations.
89*d57664e9SAndroid Build Coastguard Worker   std::vector<NamespaceDecl> namespace_decls;
90*d57664e9SAndroid Build Coastguard Worker 
91*d57664e9SAndroid Build Coastguard Worker   std::string namespace_uri;
92*d57664e9SAndroid Build Coastguard Worker   std::string name;
93*d57664e9SAndroid Build Coastguard Worker   std::vector<Attribute> attributes;
94*d57664e9SAndroid Build Coastguard Worker   std::vector<std::unique_ptr<Node>> children;
95*d57664e9SAndroid Build Coastguard Worker 
96*d57664e9SAndroid Build Coastguard Worker   void AppendChild(std::unique_ptr<Node> child);
97*d57664e9SAndroid Build Coastguard Worker   void InsertChild(size_t index, std::unique_ptr<Node> child);
98*d57664e9SAndroid Build Coastguard Worker 
99*d57664e9SAndroid Build Coastguard Worker   Attribute* FindAttribute(android::StringPiece ns, android::StringPiece name);
100*d57664e9SAndroid Build Coastguard Worker   const Attribute* FindAttribute(android::StringPiece ns, android::StringPiece name) const;
101*d57664e9SAndroid Build Coastguard Worker   Attribute* FindOrCreateAttribute(android::StringPiece ns, android::StringPiece name);
102*d57664e9SAndroid Build Coastguard Worker   void RemoveAttribute(android::StringPiece ns, android::StringPiece name);
103*d57664e9SAndroid Build Coastguard Worker 
104*d57664e9SAndroid Build Coastguard Worker   Element* FindChild(android::StringPiece ns, android::StringPiece name);
105*d57664e9SAndroid Build Coastguard Worker   const Element* FindChild(android::StringPiece ns, android::StringPiece name) const;
106*d57664e9SAndroid Build Coastguard Worker 
107*d57664e9SAndroid Build Coastguard Worker   Element* FindChildWithAttribute(android::StringPiece ns, android::StringPiece name,
108*d57664e9SAndroid Build Coastguard Worker                                   android::StringPiece attr_ns, android::StringPiece attr_name,
109*d57664e9SAndroid Build Coastguard Worker                                   android::StringPiece attr_value);
110*d57664e9SAndroid Build Coastguard Worker 
111*d57664e9SAndroid Build Coastguard Worker   const Element* FindChildWithAttribute(android::StringPiece ns, android::StringPiece name,
112*d57664e9SAndroid Build Coastguard Worker                                         android::StringPiece attr_ns,
113*d57664e9SAndroid Build Coastguard Worker                                         android::StringPiece attr_name,
114*d57664e9SAndroid Build Coastguard Worker                                         android::StringPiece attr_value) const;
115*d57664e9SAndroid Build Coastguard Worker 
116*d57664e9SAndroid Build Coastguard Worker   std::vector<Element*> GetChildElements();
117*d57664e9SAndroid Build Coastguard Worker 
118*d57664e9SAndroid Build Coastguard Worker   // Due to overriding of subtypes not working with unique_ptr, define a convenience Clone method
119*d57664e9SAndroid Build Coastguard Worker   // that knows cloning an element returns an element.
120*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<Element> CloneElement(const ElementCloneFunc& el_cloner) const;
121*d57664e9SAndroid Build Coastguard Worker 
122*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<Node> Clone(const ElementCloneFunc& el_cloner) const override;
123*d57664e9SAndroid Build Coastguard Worker 
124*d57664e9SAndroid Build Coastguard Worker   void Accept(Visitor* visitor) override;
125*d57664e9SAndroid Build Coastguard Worker   void Accept(ConstVisitor* visitor) const override;
126*d57664e9SAndroid Build Coastguard Worker };
127*d57664e9SAndroid Build Coastguard Worker 
128*d57664e9SAndroid Build Coastguard Worker // A Text (CDATA) XML node. Can not have any children.
129*d57664e9SAndroid Build Coastguard Worker class Text : public Node {
130*d57664e9SAndroid Build Coastguard Worker  public:
131*d57664e9SAndroid Build Coastguard Worker   std::string text;
132*d57664e9SAndroid Build Coastguard Worker 
133*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<Node> Clone(const ElementCloneFunc& el_cloner) const override;
134*d57664e9SAndroid Build Coastguard Worker 
135*d57664e9SAndroid Build Coastguard Worker   void Accept(Visitor* visitor) override;
136*d57664e9SAndroid Build Coastguard Worker   void Accept(ConstVisitor* visitor) const override;
137*d57664e9SAndroid Build Coastguard Worker };
138*d57664e9SAndroid Build Coastguard Worker 
139*d57664e9SAndroid Build Coastguard Worker // An XML resource with a source, name, and XML tree.
140*d57664e9SAndroid Build Coastguard Worker class XmlResource {
141*d57664e9SAndroid Build Coastguard Worker  public:
142*d57664e9SAndroid Build Coastguard Worker   ResourceFile file;
143*d57664e9SAndroid Build Coastguard Worker 
144*d57664e9SAndroid Build Coastguard Worker   // StringPool must come before the xml::Node. Destructors are called in reverse order, and
145*d57664e9SAndroid Build Coastguard Worker   // the xml::Node may have StringPool references that need to be destroyed before the StringPool
146*d57664e9SAndroid Build Coastguard Worker   // is destroyed.
147*d57664e9SAndroid Build Coastguard Worker   android::StringPool string_pool;
148*d57664e9SAndroid Build Coastguard Worker 
149*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<xml::Element> root;
150*d57664e9SAndroid Build Coastguard Worker 
151*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<XmlResource> Clone() const;
152*d57664e9SAndroid Build Coastguard Worker };
153*d57664e9SAndroid Build Coastguard Worker 
154*d57664e9SAndroid Build Coastguard Worker // Inflates an XML DOM from an InputStream, logging errors to the logger.
155*d57664e9SAndroid Build Coastguard Worker std::unique_ptr<XmlResource> Inflate(android::InputStream* in, android::IDiagnostics* diag,
156*d57664e9SAndroid Build Coastguard Worker                                      const android::Source& source);
157*d57664e9SAndroid Build Coastguard Worker 
158*d57664e9SAndroid Build Coastguard Worker // Inflates an XML DOM from a binary ResXMLTree.
159*d57664e9SAndroid Build Coastguard Worker std::unique_ptr<XmlResource> Inflate(const void* data, size_t len,
160*d57664e9SAndroid Build Coastguard Worker                                      std::string* out_error = nullptr);
161*d57664e9SAndroid Build Coastguard Worker 
162*d57664e9SAndroid Build Coastguard Worker Element* FindRootElement(Node* node);
163*d57664e9SAndroid Build Coastguard Worker 
164*d57664e9SAndroid Build Coastguard Worker // Visitor whose default implementation visits the children nodes of any node.
165*d57664e9SAndroid Build Coastguard Worker class Visitor {
166*d57664e9SAndroid Build Coastguard Worker  public:
167*d57664e9SAndroid Build Coastguard Worker   virtual ~Visitor() = default;
168*d57664e9SAndroid Build Coastguard Worker 
Visit(Element * el)169*d57664e9SAndroid Build Coastguard Worker   virtual void Visit(Element* el) {
170*d57664e9SAndroid Build Coastguard Worker     VisitChildren(el);
171*d57664e9SAndroid Build Coastguard Worker   }
172*d57664e9SAndroid Build Coastguard Worker 
Visit(Text * text)173*d57664e9SAndroid Build Coastguard Worker   virtual void Visit(Text* text) {
174*d57664e9SAndroid Build Coastguard Worker   }
175*d57664e9SAndroid Build Coastguard Worker 
176*d57664e9SAndroid Build Coastguard Worker  protected:
177*d57664e9SAndroid Build Coastguard Worker   Visitor() = default;
178*d57664e9SAndroid Build Coastguard Worker 
VisitChildren(Element * el)179*d57664e9SAndroid Build Coastguard Worker   void VisitChildren(Element* el) {
180*d57664e9SAndroid Build Coastguard Worker     for (auto& child : el->children) {
181*d57664e9SAndroid Build Coastguard Worker       child->Accept(this);
182*d57664e9SAndroid Build Coastguard Worker     }
183*d57664e9SAndroid Build Coastguard Worker   }
184*d57664e9SAndroid Build Coastguard Worker 
BeforeVisitElement(Element * el)185*d57664e9SAndroid Build Coastguard Worker   virtual void BeforeVisitElement(Element* el) {
186*d57664e9SAndroid Build Coastguard Worker   }
AfterVisitElement(Element * el)187*d57664e9SAndroid Build Coastguard Worker   virtual void AfterVisitElement(Element* el) {
188*d57664e9SAndroid Build Coastguard Worker   }
189*d57664e9SAndroid Build Coastguard Worker 
190*d57664e9SAndroid Build Coastguard Worker  private:
191*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(Visitor);
192*d57664e9SAndroid Build Coastguard Worker 
193*d57664e9SAndroid Build Coastguard Worker   friend class Element;
194*d57664e9SAndroid Build Coastguard Worker };
195*d57664e9SAndroid Build Coastguard Worker 
196*d57664e9SAndroid Build Coastguard Worker class ConstVisitor {
197*d57664e9SAndroid Build Coastguard Worker  public:
198*d57664e9SAndroid Build Coastguard Worker   virtual ~ConstVisitor() = default;
199*d57664e9SAndroid Build Coastguard Worker 
Visit(const Element * el)200*d57664e9SAndroid Build Coastguard Worker   virtual void Visit(const Element* el) {
201*d57664e9SAndroid Build Coastguard Worker     VisitChildren(el);
202*d57664e9SAndroid Build Coastguard Worker   }
203*d57664e9SAndroid Build Coastguard Worker 
Visit(const Text * text)204*d57664e9SAndroid Build Coastguard Worker   virtual void Visit(const Text* text) {
205*d57664e9SAndroid Build Coastguard Worker   }
206*d57664e9SAndroid Build Coastguard Worker 
207*d57664e9SAndroid Build Coastguard Worker  protected:
208*d57664e9SAndroid Build Coastguard Worker   ConstVisitor() = default;
209*d57664e9SAndroid Build Coastguard Worker 
VisitChildren(const Element * el)210*d57664e9SAndroid Build Coastguard Worker   void VisitChildren(const Element* el) {
211*d57664e9SAndroid Build Coastguard Worker     for (const auto& child : el->children) {
212*d57664e9SAndroid Build Coastguard Worker       child->Accept(this);
213*d57664e9SAndroid Build Coastguard Worker     }
214*d57664e9SAndroid Build Coastguard Worker   }
215*d57664e9SAndroid Build Coastguard Worker 
BeforeVisitElement(const Element * el)216*d57664e9SAndroid Build Coastguard Worker   virtual void BeforeVisitElement(const Element* el) {
217*d57664e9SAndroid Build Coastguard Worker   }
218*d57664e9SAndroid Build Coastguard Worker 
AfterVisitElement(const Element * el)219*d57664e9SAndroid Build Coastguard Worker   virtual void AfterVisitElement(const Element* el) {
220*d57664e9SAndroid Build Coastguard Worker   }
221*d57664e9SAndroid Build Coastguard Worker 
222*d57664e9SAndroid Build Coastguard Worker  private:
223*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(ConstVisitor);
224*d57664e9SAndroid Build Coastguard Worker 
225*d57664e9SAndroid Build Coastguard Worker   friend class Element;
226*d57664e9SAndroid Build Coastguard Worker };
227*d57664e9SAndroid Build Coastguard Worker 
228*d57664e9SAndroid Build Coastguard Worker // An XML DOM visitor that will record the package name for a namespace prefix.
229*d57664e9SAndroid Build Coastguard Worker class PackageAwareVisitor : public Visitor, public IPackageDeclStack {
230*d57664e9SAndroid Build Coastguard Worker  public:
231*d57664e9SAndroid Build Coastguard Worker   using Visitor::Visit;
232*d57664e9SAndroid Build Coastguard Worker 
233*d57664e9SAndroid Build Coastguard Worker   std::optional<ExtractedPackage> TransformPackageAlias(android::StringPiece alias) const override;
234*d57664e9SAndroid Build Coastguard Worker 
235*d57664e9SAndroid Build Coastguard Worker  protected:
236*d57664e9SAndroid Build Coastguard Worker   PackageAwareVisitor() = default;
237*d57664e9SAndroid Build Coastguard Worker 
238*d57664e9SAndroid Build Coastguard Worker   void BeforeVisitElement(Element* el) override;
239*d57664e9SAndroid Build Coastguard Worker   void AfterVisitElement(Element* el) override;
240*d57664e9SAndroid Build Coastguard Worker 
241*d57664e9SAndroid Build Coastguard Worker  private:
242*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(PackageAwareVisitor);
243*d57664e9SAndroid Build Coastguard Worker 
244*d57664e9SAndroid Build Coastguard Worker   struct PackageDecl {
245*d57664e9SAndroid Build Coastguard Worker     std::string prefix;
246*d57664e9SAndroid Build Coastguard Worker     ExtractedPackage package;
247*d57664e9SAndroid Build Coastguard Worker   };
248*d57664e9SAndroid Build Coastguard Worker 
249*d57664e9SAndroid Build Coastguard Worker   std::vector<std::vector<PackageDecl>> package_decls_;
250*d57664e9SAndroid Build Coastguard Worker };
251*d57664e9SAndroid Build Coastguard Worker 
252*d57664e9SAndroid Build Coastguard Worker namespace internal {
253*d57664e9SAndroid Build Coastguard Worker 
254*d57664e9SAndroid Build Coastguard Worker // Base class that overrides the default behaviour and does not descend into child nodes.
255*d57664e9SAndroid Build Coastguard Worker class NodeCastBase : public ConstVisitor {
256*d57664e9SAndroid Build Coastguard Worker  public:
Visit(const Element * el)257*d57664e9SAndroid Build Coastguard Worker   void Visit(const Element* el) override {
258*d57664e9SAndroid Build Coastguard Worker   }
Visit(const Text * el)259*d57664e9SAndroid Build Coastguard Worker   void Visit(const Text* el) override {
260*d57664e9SAndroid Build Coastguard Worker   }
261*d57664e9SAndroid Build Coastguard Worker 
262*d57664e9SAndroid Build Coastguard Worker  protected:
263*d57664e9SAndroid Build Coastguard Worker   NodeCastBase() = default;
264*d57664e9SAndroid Build Coastguard Worker 
BeforeVisitElement(const Element * el)265*d57664e9SAndroid Build Coastguard Worker   void BeforeVisitElement(const Element* el) override {
266*d57664e9SAndroid Build Coastguard Worker   }
AfterVisitElement(const Element * el)267*d57664e9SAndroid Build Coastguard Worker   void AfterVisitElement(const Element* el) override {
268*d57664e9SAndroid Build Coastguard Worker   }
269*d57664e9SAndroid Build Coastguard Worker 
270*d57664e9SAndroid Build Coastguard Worker  private:
271*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(NodeCastBase);
272*d57664e9SAndroid Build Coastguard Worker };
273*d57664e9SAndroid Build Coastguard Worker 
274*d57664e9SAndroid Build Coastguard Worker template <typename T>
275*d57664e9SAndroid Build Coastguard Worker class NodeCastImpl : public NodeCastBase {
276*d57664e9SAndroid Build Coastguard Worker  public:
277*d57664e9SAndroid Build Coastguard Worker   using NodeCastBase::Visit;
278*d57664e9SAndroid Build Coastguard Worker 
279*d57664e9SAndroid Build Coastguard Worker   NodeCastImpl() = default;
280*d57664e9SAndroid Build Coastguard Worker 
281*d57664e9SAndroid Build Coastguard Worker   const T* value = nullptr;
282*d57664e9SAndroid Build Coastguard Worker 
Visit(const T * v)283*d57664e9SAndroid Build Coastguard Worker   void Visit(const T* v) override {
284*d57664e9SAndroid Build Coastguard Worker     value = v;
285*d57664e9SAndroid Build Coastguard Worker   }
286*d57664e9SAndroid Build Coastguard Worker 
287*d57664e9SAndroid Build Coastguard Worker  private:
288*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(NodeCastImpl);
289*d57664e9SAndroid Build Coastguard Worker };
290*d57664e9SAndroid Build Coastguard Worker 
291*d57664e9SAndroid Build Coastguard Worker }  // namespace internal
292*d57664e9SAndroid Build Coastguard Worker 
293*d57664e9SAndroid Build Coastguard Worker template <typename T>
NodeCast(const Node * node)294*d57664e9SAndroid Build Coastguard Worker const T* NodeCast(const Node* node) {
295*d57664e9SAndroid Build Coastguard Worker   internal::NodeCastImpl<T> visitor;
296*d57664e9SAndroid Build Coastguard Worker   node->Accept(&visitor);
297*d57664e9SAndroid Build Coastguard Worker   return visitor.value;
298*d57664e9SAndroid Build Coastguard Worker }
299*d57664e9SAndroid Build Coastguard Worker 
300*d57664e9SAndroid Build Coastguard Worker template <typename T>
NodeCast(Node * node)301*d57664e9SAndroid Build Coastguard Worker T* NodeCast(Node* node) {
302*d57664e9SAndroid Build Coastguard Worker   return const_cast<T*>(NodeCast<T>(static_cast<const T*>(node)));
303*d57664e9SAndroid Build Coastguard Worker }
304*d57664e9SAndroid Build Coastguard Worker 
305*d57664e9SAndroid Build Coastguard Worker }  // namespace xml
306*d57664e9SAndroid Build Coastguard Worker }  // namespace aapt
307*d57664e9SAndroid Build Coastguard Worker 
308*d57664e9SAndroid Build Coastguard Worker #endif  // AAPT_XML_DOM_H
309