xref: /aosp_15_r20/frameworks/base/tools/aapt2/xml/XmlPullParser.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_PULL_PARSER_H
18*d57664e9SAndroid Build Coastguard Worker #define AAPT_XML_PULL_PARSER_H
19*d57664e9SAndroid Build Coastguard Worker 
20*d57664e9SAndroid Build Coastguard Worker #include <expat.h>
21*d57664e9SAndroid Build Coastguard Worker 
22*d57664e9SAndroid Build Coastguard Worker #include <optional>
23*d57664e9SAndroid Build Coastguard Worker #include <ostream>
24*d57664e9SAndroid Build Coastguard Worker #include <queue>
25*d57664e9SAndroid Build Coastguard Worker #include <stack>
26*d57664e9SAndroid Build Coastguard Worker #include <string>
27*d57664e9SAndroid Build Coastguard Worker #include <vector>
28*d57664e9SAndroid Build Coastguard Worker 
29*d57664e9SAndroid Build Coastguard Worker #include "Resource.h"
30*d57664e9SAndroid Build Coastguard Worker #include "android-base/macros.h"
31*d57664e9SAndroid Build Coastguard Worker #include "androidfw/Streams.h"
32*d57664e9SAndroid Build Coastguard Worker #include "androidfw/StringPiece.h"
33*d57664e9SAndroid Build Coastguard Worker #include "process/IResourceTableConsumer.h"
34*d57664e9SAndroid Build Coastguard Worker #include "xml/XmlUtil.h"
35*d57664e9SAndroid Build Coastguard Worker 
36*d57664e9SAndroid Build Coastguard Worker namespace aapt {
37*d57664e9SAndroid Build Coastguard Worker namespace xml {
38*d57664e9SAndroid Build Coastguard Worker 
39*d57664e9SAndroid Build Coastguard Worker class XmlPullParser : public IPackageDeclStack {
40*d57664e9SAndroid Build Coastguard Worker  public:
41*d57664e9SAndroid Build Coastguard Worker   enum class Event {
42*d57664e9SAndroid Build Coastguard Worker     kBadDocument,
43*d57664e9SAndroid Build Coastguard Worker     kStartDocument,
44*d57664e9SAndroid Build Coastguard Worker     kEndDocument,
45*d57664e9SAndroid Build Coastguard Worker 
46*d57664e9SAndroid Build Coastguard Worker     kStartNamespace,
47*d57664e9SAndroid Build Coastguard Worker     kEndNamespace,
48*d57664e9SAndroid Build Coastguard Worker     kStartElement,
49*d57664e9SAndroid Build Coastguard Worker     kEndElement,
50*d57664e9SAndroid Build Coastguard Worker     kText,
51*d57664e9SAndroid Build Coastguard Worker     kComment,
52*d57664e9SAndroid Build Coastguard Worker     kCdataStart,
53*d57664e9SAndroid Build Coastguard Worker     kCdataEnd,
54*d57664e9SAndroid Build Coastguard Worker   };
55*d57664e9SAndroid Build Coastguard Worker 
56*d57664e9SAndroid Build Coastguard Worker   /**
57*d57664e9SAndroid Build Coastguard Worker    * Skips to the next direct descendant node of the given start_depth,
58*d57664e9SAndroid Build Coastguard Worker    * skipping namespace nodes.
59*d57664e9SAndroid Build Coastguard Worker    *
60*d57664e9SAndroid Build Coastguard Worker    * When NextChildNode() returns true, you can expect Comments, Text, and
61*d57664e9SAndroid Build Coastguard Worker    * StartElement events.
62*d57664e9SAndroid Build Coastguard Worker    */
63*d57664e9SAndroid Build Coastguard Worker   static bool NextChildNode(XmlPullParser* parser, size_t start_depth);
64*d57664e9SAndroid Build Coastguard Worker   static bool SkipCurrentElement(XmlPullParser* parser);
65*d57664e9SAndroid Build Coastguard Worker   static bool IsGoodEvent(Event event);
66*d57664e9SAndroid Build Coastguard Worker 
67*d57664e9SAndroid Build Coastguard Worker   explicit XmlPullParser(android::InputStream* in);
68*d57664e9SAndroid Build Coastguard Worker   ~XmlPullParser();
69*d57664e9SAndroid Build Coastguard Worker 
70*d57664e9SAndroid Build Coastguard Worker   /**
71*d57664e9SAndroid Build Coastguard Worker    * Returns the current event that is being processed.
72*d57664e9SAndroid Build Coastguard Worker    */
73*d57664e9SAndroid Build Coastguard Worker   Event event() const;
74*d57664e9SAndroid Build Coastguard Worker 
75*d57664e9SAndroid Build Coastguard Worker   const std::string& error() const;
76*d57664e9SAndroid Build Coastguard Worker 
77*d57664e9SAndroid Build Coastguard Worker   /**
78*d57664e9SAndroid Build Coastguard Worker    * Note, unlike XmlPullParser, the first call to next() will return
79*d57664e9SAndroid Build Coastguard Worker    * StartElement of the first element.
80*d57664e9SAndroid Build Coastguard Worker    */
81*d57664e9SAndroid Build Coastguard Worker   Event Next();
82*d57664e9SAndroid Build Coastguard Worker 
83*d57664e9SAndroid Build Coastguard Worker   //
84*d57664e9SAndroid Build Coastguard Worker   // These are available for all nodes.
85*d57664e9SAndroid Build Coastguard Worker   //
86*d57664e9SAndroid Build Coastguard Worker 
87*d57664e9SAndroid Build Coastguard Worker   const std::string& comment() const;
88*d57664e9SAndroid Build Coastguard Worker   size_t line_number() const;
89*d57664e9SAndroid Build Coastguard Worker   size_t depth() const;
90*d57664e9SAndroid Build Coastguard Worker 
91*d57664e9SAndroid Build Coastguard Worker   /**
92*d57664e9SAndroid Build Coastguard Worker    * Returns the character data for a Text event.
93*d57664e9SAndroid Build Coastguard Worker    */
94*d57664e9SAndroid Build Coastguard Worker   const std::string& text() const;
95*d57664e9SAndroid Build Coastguard Worker 
96*d57664e9SAndroid Build Coastguard Worker   //
97*d57664e9SAndroid Build Coastguard Worker   // Namespace prefix and URI are available for StartNamespace and EndNamespace.
98*d57664e9SAndroid Build Coastguard Worker   //
99*d57664e9SAndroid Build Coastguard Worker 
100*d57664e9SAndroid Build Coastguard Worker   const std::string& namespace_prefix() const;
101*d57664e9SAndroid Build Coastguard Worker   const std::string& namespace_uri() const;
102*d57664e9SAndroid Build Coastguard Worker 
103*d57664e9SAndroid Build Coastguard Worker   //
104*d57664e9SAndroid Build Coastguard Worker   // These are available for StartElement and EndElement.
105*d57664e9SAndroid Build Coastguard Worker   //
106*d57664e9SAndroid Build Coastguard Worker 
107*d57664e9SAndroid Build Coastguard Worker   const std::string& element_namespace() const;
108*d57664e9SAndroid Build Coastguard Worker   const std::string& element_name() const;
109*d57664e9SAndroid Build Coastguard Worker 
110*d57664e9SAndroid Build Coastguard Worker   /*
111*d57664e9SAndroid Build Coastguard Worker    * Uses the current stack of namespaces to resolve the package. Eg:
112*d57664e9SAndroid Build Coastguard Worker    * xmlns:app = "http://schemas.android.com/apk/res/com.android.app"
113*d57664e9SAndroid Build Coastguard Worker    * ...
114*d57664e9SAndroid Build Coastguard Worker    * android:text="@app:string/message"
115*d57664e9SAndroid Build Coastguard Worker    *
116*d57664e9SAndroid Build Coastguard Worker    * In this case, 'app' will be converted to 'com.android.app'.
117*d57664e9SAndroid Build Coastguard Worker    *
118*d57664e9SAndroid Build Coastguard Worker    * If xmlns:app="http://schemas.android.com/apk/res-auto", then
119*d57664e9SAndroid Build Coastguard Worker    * 'package' will be set to 'defaultPackage'.
120*d57664e9SAndroid Build Coastguard Worker    */
121*d57664e9SAndroid Build Coastguard Worker   std::optional<ExtractedPackage> TransformPackageAlias(android::StringPiece alias) const override;
122*d57664e9SAndroid Build Coastguard Worker 
123*d57664e9SAndroid Build Coastguard Worker   struct PackageDecl {
124*d57664e9SAndroid Build Coastguard Worker     std::string prefix;
125*d57664e9SAndroid Build Coastguard Worker     ExtractedPackage package;
126*d57664e9SAndroid Build Coastguard Worker   };
127*d57664e9SAndroid Build Coastguard Worker 
128*d57664e9SAndroid Build Coastguard Worker   const std::vector<PackageDecl>& package_decls() const;
129*d57664e9SAndroid Build Coastguard Worker 
130*d57664e9SAndroid Build Coastguard Worker   //
131*d57664e9SAndroid Build Coastguard Worker   // Remaining methods are for retrieving information about attributes
132*d57664e9SAndroid Build Coastguard Worker   // associated with a StartElement.
133*d57664e9SAndroid Build Coastguard Worker   //
134*d57664e9SAndroid Build Coastguard Worker   // Attributes must be in sorted order (according to the less than operator
135*d57664e9SAndroid Build Coastguard Worker   // of struct Attribute).
136*d57664e9SAndroid Build Coastguard Worker   //
137*d57664e9SAndroid Build Coastguard Worker 
138*d57664e9SAndroid Build Coastguard Worker   struct Attribute {
139*d57664e9SAndroid Build Coastguard Worker     std::string namespace_uri;
140*d57664e9SAndroid Build Coastguard Worker     std::string name;
141*d57664e9SAndroid Build Coastguard Worker     std::string value;
142*d57664e9SAndroid Build Coastguard Worker 
143*d57664e9SAndroid Build Coastguard Worker     int compare(const Attribute& rhs) const;
144*d57664e9SAndroid Build Coastguard Worker     bool operator<(const Attribute& rhs) const;
145*d57664e9SAndroid Build Coastguard Worker     bool operator==(const Attribute& rhs) const;
146*d57664e9SAndroid Build Coastguard Worker     bool operator!=(const Attribute& rhs) const;
147*d57664e9SAndroid Build Coastguard Worker   };
148*d57664e9SAndroid Build Coastguard Worker 
149*d57664e9SAndroid Build Coastguard Worker   using const_iterator = std::vector<Attribute>::const_iterator;
150*d57664e9SAndroid Build Coastguard Worker 
151*d57664e9SAndroid Build Coastguard Worker   const_iterator begin_attributes() const;
152*d57664e9SAndroid Build Coastguard Worker   const_iterator end_attributes() const;
153*d57664e9SAndroid Build Coastguard Worker   size_t attribute_count() const;
154*d57664e9SAndroid Build Coastguard Worker   const_iterator FindAttribute(android::StringPiece namespace_uri, android::StringPiece name) const;
155*d57664e9SAndroid Build Coastguard Worker 
156*d57664e9SAndroid Build Coastguard Worker  private:
157*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(XmlPullParser);
158*d57664e9SAndroid Build Coastguard Worker 
159*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL StartNamespaceHandler(void* user_data, const char* prefix,
160*d57664e9SAndroid Build Coastguard Worker                                             const char* uri);
161*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL StartElementHandler(void* user_data, const char* name,
162*d57664e9SAndroid Build Coastguard Worker                                           const char** attrs);
163*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL CharacterDataHandler(void* user_data, const char* s,
164*d57664e9SAndroid Build Coastguard Worker                                            int len);
165*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL EndElementHandler(void* user_data, const char* name);
166*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL EndNamespaceHandler(void* user_data, const char* prefix);
167*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL CommentDataHandler(void* user_data, const char* comment);
168*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL StartCdataSectionHandler(void* user_data);
169*d57664e9SAndroid Build Coastguard Worker   static void XMLCALL EndCdataSectionHandler(void* user_data);
170*d57664e9SAndroid Build Coastguard Worker 
171*d57664e9SAndroid Build Coastguard Worker   struct EventData {
172*d57664e9SAndroid Build Coastguard Worker     Event event;
173*d57664e9SAndroid Build Coastguard Worker     size_t line_number;
174*d57664e9SAndroid Build Coastguard Worker     size_t depth;
175*d57664e9SAndroid Build Coastguard Worker     std::string data1;
176*d57664e9SAndroid Build Coastguard Worker     std::string data2;
177*d57664e9SAndroid Build Coastguard Worker     std::vector<Attribute> attributes;
178*d57664e9SAndroid Build Coastguard Worker   };
179*d57664e9SAndroid Build Coastguard Worker 
180*d57664e9SAndroid Build Coastguard Worker   android::InputStream* in_;
181*d57664e9SAndroid Build Coastguard Worker   XML_Parser parser_;
182*d57664e9SAndroid Build Coastguard Worker   std::queue<EventData> event_queue_;
183*d57664e9SAndroid Build Coastguard Worker   std::string error_;
184*d57664e9SAndroid Build Coastguard Worker   const std::string empty_;
185*d57664e9SAndroid Build Coastguard Worker   size_t depth_;
186*d57664e9SAndroid Build Coastguard Worker   std::stack<std::string> namespace_uris_;
187*d57664e9SAndroid Build Coastguard Worker   std::vector<PackageDecl> package_aliases_;
188*d57664e9SAndroid Build Coastguard Worker };
189*d57664e9SAndroid Build Coastguard Worker 
190*d57664e9SAndroid Build Coastguard Worker /**
191*d57664e9SAndroid Build Coastguard Worker  * Finds the attribute in the current element within the global namespace.
192*d57664e9SAndroid Build Coastguard Worker  */
193*d57664e9SAndroid Build Coastguard Worker std::optional<android::StringPiece> FindAttribute(const XmlPullParser* parser,
194*d57664e9SAndroid Build Coastguard Worker                                                   android::StringPiece name);
195*d57664e9SAndroid Build Coastguard Worker 
196*d57664e9SAndroid Build Coastguard Worker /**
197*d57664e9SAndroid Build Coastguard Worker  * Finds the attribute in the current element within the given namespace.
198*d57664e9SAndroid Build Coastguard Worker  */
199*d57664e9SAndroid Build Coastguard Worker std::optional<android::StringPiece> FindAttribute(const XmlPullParser* parser,
200*d57664e9SAndroid Build Coastguard Worker                                                   android::StringPiece namespace_uri,
201*d57664e9SAndroid Build Coastguard Worker                                                   android::StringPiece name);
202*d57664e9SAndroid Build Coastguard Worker 
203*d57664e9SAndroid Build Coastguard Worker /**
204*d57664e9SAndroid Build Coastguard Worker  * Finds the attribute in the current element within the global namespace. The
205*d57664e9SAndroid Build Coastguard Worker  * attribute's value
206*d57664e9SAndroid Build Coastguard Worker  * must not be the empty string.
207*d57664e9SAndroid Build Coastguard Worker  */
208*d57664e9SAndroid Build Coastguard Worker std::optional<android::StringPiece> FindNonEmptyAttribute(const XmlPullParser* parser,
209*d57664e9SAndroid Build Coastguard Worker                                                           android::StringPiece name);
210*d57664e9SAndroid Build Coastguard Worker 
211*d57664e9SAndroid Build Coastguard Worker //
212*d57664e9SAndroid Build Coastguard Worker // Implementation
213*d57664e9SAndroid Build Coastguard Worker //
214*d57664e9SAndroid Build Coastguard Worker 
215*d57664e9SAndroid Build Coastguard Worker inline ::std::ostream& operator<<(::std::ostream& out,
216*d57664e9SAndroid Build Coastguard Worker                                   XmlPullParser::Event event) {
217*d57664e9SAndroid Build Coastguard Worker   switch (event) {
218*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kBadDocument:
219*d57664e9SAndroid Build Coastguard Worker       return out << "BadDocument";
220*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kStartDocument:
221*d57664e9SAndroid Build Coastguard Worker       return out << "StartDocument";
222*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kEndDocument:
223*d57664e9SAndroid Build Coastguard Worker       return out << "EndDocument";
224*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kStartNamespace:
225*d57664e9SAndroid Build Coastguard Worker       return out << "StartNamespace";
226*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kEndNamespace:
227*d57664e9SAndroid Build Coastguard Worker       return out << "EndNamespace";
228*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kStartElement:
229*d57664e9SAndroid Build Coastguard Worker       return out << "StartElement";
230*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kEndElement:
231*d57664e9SAndroid Build Coastguard Worker       return out << "EndElement";
232*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kText:
233*d57664e9SAndroid Build Coastguard Worker       return out << "Text";
234*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kComment:
235*d57664e9SAndroid Build Coastguard Worker       return out << "Comment";
236*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kCdataStart:
237*d57664e9SAndroid Build Coastguard Worker       return out << "CdataStart";
238*d57664e9SAndroid Build Coastguard Worker     case XmlPullParser::Event::kCdataEnd:
239*d57664e9SAndroid Build Coastguard Worker       return out << "CdataEnd";
240*d57664e9SAndroid Build Coastguard Worker   }
241*d57664e9SAndroid Build Coastguard Worker   return out;
242*d57664e9SAndroid Build Coastguard Worker }
243*d57664e9SAndroid Build Coastguard Worker 
NextChildNode(XmlPullParser * parser,size_t start_depth)244*d57664e9SAndroid Build Coastguard Worker inline bool XmlPullParser::NextChildNode(XmlPullParser* parser, size_t start_depth) {
245*d57664e9SAndroid Build Coastguard Worker   Event event;
246*d57664e9SAndroid Build Coastguard Worker 
247*d57664e9SAndroid Build Coastguard Worker   // First get back to the start depth.
248*d57664e9SAndroid Build Coastguard Worker   while (IsGoodEvent(event = parser->Next()) && parser->depth() > start_depth + 1) {
249*d57664e9SAndroid Build Coastguard Worker   }
250*d57664e9SAndroid Build Coastguard Worker 
251*d57664e9SAndroid Build Coastguard Worker   // Now look for the first good node.
252*d57664e9SAndroid Build Coastguard Worker   while ((event != Event::kEndElement || parser->depth() > start_depth) && IsGoodEvent(event)) {
253*d57664e9SAndroid Build Coastguard Worker     switch (event) {
254*d57664e9SAndroid Build Coastguard Worker       case Event::kText:
255*d57664e9SAndroid Build Coastguard Worker       case Event::kComment:
256*d57664e9SAndroid Build Coastguard Worker       case Event::kStartElement:
257*d57664e9SAndroid Build Coastguard Worker       case Event::kCdataStart:
258*d57664e9SAndroid Build Coastguard Worker       case Event::kCdataEnd:
259*d57664e9SAndroid Build Coastguard Worker         return true;
260*d57664e9SAndroid Build Coastguard Worker       default:
261*d57664e9SAndroid Build Coastguard Worker         break;
262*d57664e9SAndroid Build Coastguard Worker     }
263*d57664e9SAndroid Build Coastguard Worker     event = parser->Next();
264*d57664e9SAndroid Build Coastguard Worker   }
265*d57664e9SAndroid Build Coastguard Worker   return false;
266*d57664e9SAndroid Build Coastguard Worker }
267*d57664e9SAndroid Build Coastguard Worker 
SkipCurrentElement(XmlPullParser * parser)268*d57664e9SAndroid Build Coastguard Worker inline bool XmlPullParser::SkipCurrentElement(XmlPullParser* parser) {
269*d57664e9SAndroid Build Coastguard Worker   int depth = 1;
270*d57664e9SAndroid Build Coastguard Worker   while (depth > 0) {
271*d57664e9SAndroid Build Coastguard Worker     switch (parser->Next()) {
272*d57664e9SAndroid Build Coastguard Worker       case Event::kEndDocument:
273*d57664e9SAndroid Build Coastguard Worker         return true;
274*d57664e9SAndroid Build Coastguard Worker       case Event::kBadDocument:
275*d57664e9SAndroid Build Coastguard Worker         return false;
276*d57664e9SAndroid Build Coastguard Worker       case Event::kStartElement:
277*d57664e9SAndroid Build Coastguard Worker         depth++;
278*d57664e9SAndroid Build Coastguard Worker         break;
279*d57664e9SAndroid Build Coastguard Worker       case Event::kEndElement:
280*d57664e9SAndroid Build Coastguard Worker         depth--;
281*d57664e9SAndroid Build Coastguard Worker         break;
282*d57664e9SAndroid Build Coastguard Worker       default:
283*d57664e9SAndroid Build Coastguard Worker         break;
284*d57664e9SAndroid Build Coastguard Worker     }
285*d57664e9SAndroid Build Coastguard Worker   }
286*d57664e9SAndroid Build Coastguard Worker   return true;
287*d57664e9SAndroid Build Coastguard Worker }
288*d57664e9SAndroid Build Coastguard Worker 
IsGoodEvent(XmlPullParser::Event event)289*d57664e9SAndroid Build Coastguard Worker inline bool XmlPullParser::IsGoodEvent(XmlPullParser::Event event) {
290*d57664e9SAndroid Build Coastguard Worker   return event != Event::kBadDocument && event != Event::kEndDocument;
291*d57664e9SAndroid Build Coastguard Worker }
292*d57664e9SAndroid Build Coastguard Worker 
compare(const Attribute & rhs)293*d57664e9SAndroid Build Coastguard Worker inline int XmlPullParser::Attribute::compare(const Attribute& rhs) const {
294*d57664e9SAndroid Build Coastguard Worker   int cmp = namespace_uri.compare(rhs.namespace_uri);
295*d57664e9SAndroid Build Coastguard Worker   if (cmp != 0) return cmp;
296*d57664e9SAndroid Build Coastguard Worker   return name.compare(rhs.name);
297*d57664e9SAndroid Build Coastguard Worker }
298*d57664e9SAndroid Build Coastguard Worker 
299*d57664e9SAndroid Build Coastguard Worker inline bool XmlPullParser::Attribute::operator<(const Attribute& rhs) const {
300*d57664e9SAndroid Build Coastguard Worker   return compare(rhs) < 0;
301*d57664e9SAndroid Build Coastguard Worker }
302*d57664e9SAndroid Build Coastguard Worker 
303*d57664e9SAndroid Build Coastguard Worker inline bool XmlPullParser::Attribute::operator==(const Attribute& rhs) const {
304*d57664e9SAndroid Build Coastguard Worker   return compare(rhs) == 0;
305*d57664e9SAndroid Build Coastguard Worker }
306*d57664e9SAndroid Build Coastguard Worker 
307*d57664e9SAndroid Build Coastguard Worker inline bool XmlPullParser::Attribute::operator!=(const Attribute& rhs) const {
308*d57664e9SAndroid Build Coastguard Worker   return compare(rhs) != 0;
309*d57664e9SAndroid Build Coastguard Worker }
310*d57664e9SAndroid Build Coastguard Worker 
311*d57664e9SAndroid Build Coastguard Worker }  // namespace xml
312*d57664e9SAndroid Build Coastguard Worker }  // namespace aapt
313*d57664e9SAndroid Build Coastguard Worker 
314*d57664e9SAndroid Build Coastguard Worker #endif  // AAPT_XML_PULL_PARSER_H
315