xref: /aosp_15_r20/frameworks/base/tools/aapt/AaptXml.cpp (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker  * Copyright (C) 2014 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 #include <androidfw/ResourceTypes.h>
18*d57664e9SAndroid Build Coastguard Worker #include <utils/String8.h>
19*d57664e9SAndroid Build Coastguard Worker 
20*d57664e9SAndroid Build Coastguard Worker #include "AaptXml.h"
21*d57664e9SAndroid Build Coastguard Worker 
22*d57664e9SAndroid Build Coastguard Worker using namespace android;
23*d57664e9SAndroid Build Coastguard Worker 
24*d57664e9SAndroid Build Coastguard Worker namespace AaptXml {
25*d57664e9SAndroid Build Coastguard Worker 
getStringAttributeAtIndex(const ResXMLTree & tree,ssize_t attrIndex,String8 * outError)26*d57664e9SAndroid Build Coastguard Worker static String8 getStringAttributeAtIndex(const ResXMLTree& tree, ssize_t attrIndex,
27*d57664e9SAndroid Build Coastguard Worker         String8* outError) {
28*d57664e9SAndroid Build Coastguard Worker     Res_value value;
29*d57664e9SAndroid Build Coastguard Worker     if (tree.getAttributeValue(attrIndex, &value) < 0) {
30*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
31*d57664e9SAndroid Build Coastguard Worker             *outError = "could not find attribute at index";
32*d57664e9SAndroid Build Coastguard Worker         }
33*d57664e9SAndroid Build Coastguard Worker         return String8();
34*d57664e9SAndroid Build Coastguard Worker     }
35*d57664e9SAndroid Build Coastguard Worker 
36*d57664e9SAndroid Build Coastguard Worker     if (value.dataType != Res_value::TYPE_STRING) {
37*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
38*d57664e9SAndroid Build Coastguard Worker             *outError = "attribute is not a string value";
39*d57664e9SAndroid Build Coastguard Worker         }
40*d57664e9SAndroid Build Coastguard Worker         return String8();
41*d57664e9SAndroid Build Coastguard Worker     }
42*d57664e9SAndroid Build Coastguard Worker 
43*d57664e9SAndroid Build Coastguard Worker     size_t len;
44*d57664e9SAndroid Build Coastguard Worker     const char16_t* str = tree.getAttributeStringValue(attrIndex, &len);
45*d57664e9SAndroid Build Coastguard Worker     return str ? String8(str, len) : String8();
46*d57664e9SAndroid Build Coastguard Worker }
47*d57664e9SAndroid Build Coastguard Worker 
getIntegerAttributeAtIndex(const ResXMLTree & tree,ssize_t attrIndex,int32_t defValue,String8 * outError)48*d57664e9SAndroid Build Coastguard Worker static int32_t getIntegerAttributeAtIndex(const ResXMLTree& tree, ssize_t attrIndex,
49*d57664e9SAndroid Build Coastguard Worker     int32_t defValue, String8* outError) {
50*d57664e9SAndroid Build Coastguard Worker     Res_value value;
51*d57664e9SAndroid Build Coastguard Worker     if (tree.getAttributeValue(attrIndex, &value) < 0) {
52*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
53*d57664e9SAndroid Build Coastguard Worker             *outError = "could not find attribute at index";
54*d57664e9SAndroid Build Coastguard Worker         }
55*d57664e9SAndroid Build Coastguard Worker         return defValue;
56*d57664e9SAndroid Build Coastguard Worker     }
57*d57664e9SAndroid Build Coastguard Worker 
58*d57664e9SAndroid Build Coastguard Worker     if (value.dataType < Res_value::TYPE_FIRST_INT
59*d57664e9SAndroid Build Coastguard Worker             || value.dataType > Res_value::TYPE_LAST_INT) {
60*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
61*d57664e9SAndroid Build Coastguard Worker             *outError = "attribute is not an integer value";
62*d57664e9SAndroid Build Coastguard Worker         }
63*d57664e9SAndroid Build Coastguard Worker         return defValue;
64*d57664e9SAndroid Build Coastguard Worker     }
65*d57664e9SAndroid Build Coastguard Worker     return value.data;
66*d57664e9SAndroid Build Coastguard Worker }
67*d57664e9SAndroid Build Coastguard Worker 
68*d57664e9SAndroid Build Coastguard Worker 
indexOfAttribute(const ResXMLTree & tree,uint32_t attrRes)69*d57664e9SAndroid Build Coastguard Worker ssize_t indexOfAttribute(const ResXMLTree& tree, uint32_t attrRes) {
70*d57664e9SAndroid Build Coastguard Worker     size_t attrCount = tree.getAttributeCount();
71*d57664e9SAndroid Build Coastguard Worker     for (size_t i = 0; i < attrCount; i++) {
72*d57664e9SAndroid Build Coastguard Worker         if (tree.getAttributeNameResID(i) == attrRes) {
73*d57664e9SAndroid Build Coastguard Worker             return (ssize_t)i;
74*d57664e9SAndroid Build Coastguard Worker         }
75*d57664e9SAndroid Build Coastguard Worker     }
76*d57664e9SAndroid Build Coastguard Worker     return -1;
77*d57664e9SAndroid Build Coastguard Worker }
78*d57664e9SAndroid Build Coastguard Worker 
getAttribute(const ResXMLTree & tree,const char * ns,const char * attr,String8 * outError)79*d57664e9SAndroid Build Coastguard Worker String8 getAttribute(const ResXMLTree& tree, const char* ns,
80*d57664e9SAndroid Build Coastguard Worker         const char* attr, String8* outError) {
81*d57664e9SAndroid Build Coastguard Worker     ssize_t idx = tree.indexOfAttribute(ns, attr);
82*d57664e9SAndroid Build Coastguard Worker     if (idx < 0) {
83*d57664e9SAndroid Build Coastguard Worker         return String8();
84*d57664e9SAndroid Build Coastguard Worker     }
85*d57664e9SAndroid Build Coastguard Worker     return getStringAttributeAtIndex(tree, idx, outError);
86*d57664e9SAndroid Build Coastguard Worker }
87*d57664e9SAndroid Build Coastguard Worker 
getAttribute(const ResXMLTree & tree,uint32_t attrRes,String8 * outError)88*d57664e9SAndroid Build Coastguard Worker String8 getAttribute(const ResXMLTree& tree, uint32_t attrRes, String8* outError) {
89*d57664e9SAndroid Build Coastguard Worker     ssize_t idx = indexOfAttribute(tree, attrRes);
90*d57664e9SAndroid Build Coastguard Worker     if (idx < 0) {
91*d57664e9SAndroid Build Coastguard Worker         return String8();
92*d57664e9SAndroid Build Coastguard Worker     }
93*d57664e9SAndroid Build Coastguard Worker     return getStringAttributeAtIndex(tree, idx, outError);
94*d57664e9SAndroid Build Coastguard Worker }
95*d57664e9SAndroid Build Coastguard Worker 
getResolvedAttribute(const ResTable & resTable,const ResXMLTree & tree,uint32_t attrRes,String8 * outError)96*d57664e9SAndroid Build Coastguard Worker String8 getResolvedAttribute(const ResTable& resTable, const ResXMLTree& tree,
97*d57664e9SAndroid Build Coastguard Worker         uint32_t attrRes, String8* outError) {
98*d57664e9SAndroid Build Coastguard Worker     ssize_t idx = indexOfAttribute(tree, attrRes);
99*d57664e9SAndroid Build Coastguard Worker     if (idx < 0) {
100*d57664e9SAndroid Build Coastguard Worker         return String8();
101*d57664e9SAndroid Build Coastguard Worker     }
102*d57664e9SAndroid Build Coastguard Worker 
103*d57664e9SAndroid Build Coastguard Worker     Res_value value;
104*d57664e9SAndroid Build Coastguard Worker     if (tree.getAttributeValue(idx, &value) == BAD_TYPE) {
105*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
106*d57664e9SAndroid Build Coastguard Worker             *outError = "attribute value is corrupt";
107*d57664e9SAndroid Build Coastguard Worker         }
108*d57664e9SAndroid Build Coastguard Worker         return String8();
109*d57664e9SAndroid Build Coastguard Worker     }
110*d57664e9SAndroid Build Coastguard Worker 
111*d57664e9SAndroid Build Coastguard Worker     // Check if the string is inline in the XML.
112*d57664e9SAndroid Build Coastguard Worker     if (value.dataType == Res_value::TYPE_STRING) {
113*d57664e9SAndroid Build Coastguard Worker         size_t len;
114*d57664e9SAndroid Build Coastguard Worker         const char16_t* str = tree.getAttributeStringValue(idx, &len);
115*d57664e9SAndroid Build Coastguard Worker         return str ? String8(str, len) : String8();
116*d57664e9SAndroid Build Coastguard Worker     }
117*d57664e9SAndroid Build Coastguard Worker 
118*d57664e9SAndroid Build Coastguard Worker     // Resolve the reference if there is one.
119*d57664e9SAndroid Build Coastguard Worker     ssize_t block = resTable.resolveReference(&value, 0);
120*d57664e9SAndroid Build Coastguard Worker     if (block < 0) {
121*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
122*d57664e9SAndroid Build Coastguard Worker             *outError = "attribute value reference does not exist";
123*d57664e9SAndroid Build Coastguard Worker         }
124*d57664e9SAndroid Build Coastguard Worker         return String8();
125*d57664e9SAndroid Build Coastguard Worker     }
126*d57664e9SAndroid Build Coastguard Worker 
127*d57664e9SAndroid Build Coastguard Worker     if (value.dataType != Res_value::TYPE_STRING) {
128*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
129*d57664e9SAndroid Build Coastguard Worker             *outError = "attribute is not a string value";
130*d57664e9SAndroid Build Coastguard Worker         }
131*d57664e9SAndroid Build Coastguard Worker         return String8();
132*d57664e9SAndroid Build Coastguard Worker     }
133*d57664e9SAndroid Build Coastguard Worker 
134*d57664e9SAndroid Build Coastguard Worker     size_t len;
135*d57664e9SAndroid Build Coastguard Worker     const char16_t* str = resTable.valueToString(&value, static_cast<size_t>(block), NULL, &len);
136*d57664e9SAndroid Build Coastguard Worker     return str ? String8(str, len) : String8();
137*d57664e9SAndroid Build Coastguard Worker }
138*d57664e9SAndroid Build Coastguard Worker 
getIntegerAttribute(const ResXMLTree & tree,const char * ns,const char * attr,int32_t defValue,String8 * outError)139*d57664e9SAndroid Build Coastguard Worker int32_t getIntegerAttribute(const ResXMLTree& tree, const char* ns,
140*d57664e9SAndroid Build Coastguard Worker         const char* attr, int32_t defValue, String8* outError) {
141*d57664e9SAndroid Build Coastguard Worker     ssize_t idx = tree.indexOfAttribute(ns, attr);
142*d57664e9SAndroid Build Coastguard Worker     if (idx < 0) {
143*d57664e9SAndroid Build Coastguard Worker         return defValue;
144*d57664e9SAndroid Build Coastguard Worker     }
145*d57664e9SAndroid Build Coastguard Worker     return getIntegerAttributeAtIndex(tree, idx, defValue, outError);
146*d57664e9SAndroid Build Coastguard Worker }
147*d57664e9SAndroid Build Coastguard Worker 
getIntegerAttribute(const ResXMLTree & tree,uint32_t attrRes,int32_t defValue,String8 * outError)148*d57664e9SAndroid Build Coastguard Worker int32_t getIntegerAttribute(const ResXMLTree& tree, uint32_t attrRes, int32_t defValue,
149*d57664e9SAndroid Build Coastguard Worker         String8* outError) {
150*d57664e9SAndroid Build Coastguard Worker     ssize_t idx = indexOfAttribute(tree, attrRes);
151*d57664e9SAndroid Build Coastguard Worker     if (idx < 0) {
152*d57664e9SAndroid Build Coastguard Worker         return defValue;
153*d57664e9SAndroid Build Coastguard Worker     }
154*d57664e9SAndroid Build Coastguard Worker     return getIntegerAttributeAtIndex(tree, idx, defValue, outError);
155*d57664e9SAndroid Build Coastguard Worker }
156*d57664e9SAndroid Build Coastguard Worker 
getResolvedIntegerAttribute(const ResTable & resTable,const ResXMLTree & tree,uint32_t attrRes,int32_t defValue,String8 * outError)157*d57664e9SAndroid Build Coastguard Worker int32_t getResolvedIntegerAttribute(const ResTable& resTable, const ResXMLTree& tree,
158*d57664e9SAndroid Build Coastguard Worker         uint32_t attrRes, int32_t defValue, String8* outError) {
159*d57664e9SAndroid Build Coastguard Worker     ssize_t idx = indexOfAttribute(tree, attrRes);
160*d57664e9SAndroid Build Coastguard Worker     if (idx < 0) {
161*d57664e9SAndroid Build Coastguard Worker         return defValue;
162*d57664e9SAndroid Build Coastguard Worker     }
163*d57664e9SAndroid Build Coastguard Worker     Res_value value;
164*d57664e9SAndroid Build Coastguard Worker     if (tree.getAttributeValue(idx, &value) != NO_ERROR) {
165*d57664e9SAndroid Build Coastguard Worker         if (value.dataType == Res_value::TYPE_REFERENCE) {
166*d57664e9SAndroid Build Coastguard Worker             resTable.resolveReference(&value, 0);
167*d57664e9SAndroid Build Coastguard Worker         }
168*d57664e9SAndroid Build Coastguard Worker         if (value.dataType < Res_value::TYPE_FIRST_INT
169*d57664e9SAndroid Build Coastguard Worker                 || value.dataType > Res_value::TYPE_LAST_INT) {
170*d57664e9SAndroid Build Coastguard Worker             if (outError != NULL) {
171*d57664e9SAndroid Build Coastguard Worker                 *outError = "attribute is not an integer value";
172*d57664e9SAndroid Build Coastguard Worker             }
173*d57664e9SAndroid Build Coastguard Worker             return defValue;
174*d57664e9SAndroid Build Coastguard Worker         }
175*d57664e9SAndroid Build Coastguard Worker     }
176*d57664e9SAndroid Build Coastguard Worker     return value.data;
177*d57664e9SAndroid Build Coastguard Worker }
178*d57664e9SAndroid Build Coastguard Worker 
getResolvedResourceAttribute(const ResTable & resTable,const ResXMLTree & tree,uint32_t attrRes,Res_value * outValue,String8 * outError)179*d57664e9SAndroid Build Coastguard Worker void getResolvedResourceAttribute(const ResTable& resTable, const ResXMLTree& tree,
180*d57664e9SAndroid Build Coastguard Worker         uint32_t attrRes, Res_value* outValue, String8* outError) {
181*d57664e9SAndroid Build Coastguard Worker     ssize_t idx = indexOfAttribute(tree, attrRes);
182*d57664e9SAndroid Build Coastguard Worker     if (idx < 0) {
183*d57664e9SAndroid Build Coastguard Worker         if (outError != NULL) {
184*d57664e9SAndroid Build Coastguard Worker             *outError = "attribute could not be found";
185*d57664e9SAndroid Build Coastguard Worker         }
186*d57664e9SAndroid Build Coastguard Worker         return;
187*d57664e9SAndroid Build Coastguard Worker     }
188*d57664e9SAndroid Build Coastguard Worker     if (tree.getAttributeValue(idx, outValue) != NO_ERROR) {
189*d57664e9SAndroid Build Coastguard Worker         if (outValue->dataType == Res_value::TYPE_REFERENCE) {
190*d57664e9SAndroid Build Coastguard Worker             resTable.resolveReference(outValue, 0);
191*d57664e9SAndroid Build Coastguard Worker         }
192*d57664e9SAndroid Build Coastguard Worker         // The attribute was found and was resolved if need be.
193*d57664e9SAndroid Build Coastguard Worker         return;
194*d57664e9SAndroid Build Coastguard Worker     }
195*d57664e9SAndroid Build Coastguard Worker     if (outError != NULL) {
196*d57664e9SAndroid Build Coastguard Worker         *outError = "error getting resolved resource attribute";
197*d57664e9SAndroid Build Coastguard Worker     }
198*d57664e9SAndroid Build Coastguard Worker }
199*d57664e9SAndroid Build Coastguard Worker 
200*d57664e9SAndroid Build Coastguard Worker } // namespace AaptXml
201