blob: e5eaf2fe62a84a4ed463f620bc5a35ec4021f5b9 [file] [log] [blame]
Adam Lesinski2ae4a872015-11-02 16:10:55 -08001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Adam Lesinski2ae4a872015-11-02 16:10:55 -080017#include "link/ManifestFixer.h"
Adam Lesinskice5e56e2016-10-21 17:56:45 -070018
19#include <unordered_set>
20
21#include "android-base/logging.h"
22
Adam Lesinskicacb28f2016-10-19 12:18:14 -070023#include "ResourceUtils.h"
Adam Lesinski2ae4a872015-11-02 16:10:55 -080024#include "util/Util.h"
Adam Lesinskicc5609d2016-04-05 12:41:07 -070025#include "xml/XmlActionExecutor.h"
Adam Lesinski467f1712015-11-16 17:35:44 -080026#include "xml/XmlDom.h"
Adam Lesinski2ae4a872015-11-02 16:10:55 -080027
Adam Lesinskid5083f62017-01-16 15:07:21 -080028using android::StringPiece;
29
Adam Lesinski2ae4a872015-11-02 16:10:55 -080030namespace aapt {
31
Adam Lesinskicc5609d2016-04-05 12:41:07 -070032/**
Adam Lesinskicacb28f2016-10-19 12:18:14 -070033 * This is how PackageManager builds class names from AndroidManifest.xml
34 * entries.
Adam Lesinskicc5609d2016-04-05 12:41:07 -070035 */
Adam Lesinskice5e56e2016-10-21 17:56:45 -070036static bool NameIsJavaClassName(xml::Element* el, xml::Attribute* attr,
Adam Lesinskicc5609d2016-04-05 12:41:07 -070037 SourcePathDiagnostics* diag) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -070038 // We allow unqualified class names (ie: .HelloActivity)
39 // Since we don't know the package name, we can just make a fake one here and
40 // the test will be identical as long as the real package name is valid too.
Adam Lesinskice5e56e2016-10-21 17:56:45 -070041 Maybe<std::string> fully_qualified_class_name =
42 util::GetFullyQualifiedClassName("a", attr->value);
Adam Lesinskicc5609d2016-04-05 12:41:07 -070043
Adam Lesinskice5e56e2016-10-21 17:56:45 -070044 StringPiece qualified_class_name = fully_qualified_class_name
45 ? fully_qualified_class_name.value()
46 : attr->value;
Adam Lesinskid0f116b2016-07-08 15:00:32 -070047
Adam Lesinskice5e56e2016-10-21 17:56:45 -070048 if (!util::IsJavaClassName(qualified_class_name)) {
49 diag->Error(DiagMessage(el->line_number)
Adam Lesinskicacb28f2016-10-19 12:18:14 -070050 << "attribute 'android:name' in <" << el->name
51 << "> tag must be a valid Java class name");
Adam Lesinski52364f72016-01-11 13:10:24 -080052 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -070053 }
54 return true;
55}
56
Adam Lesinskice5e56e2016-10-21 17:56:45 -070057static bool OptionalNameIsJavaClassName(xml::Element* el,
Adam Lesinskicacb28f2016-10-19 12:18:14 -070058 SourcePathDiagnostics* diag) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070059 if (xml::Attribute* attr = el->FindAttribute(xml::kSchemaAndroid, "name")) {
60 return NameIsJavaClassName(el, attr, diag);
Adam Lesinskicacb28f2016-10-19 12:18:14 -070061 }
62 return true;
63}
64
Adam Lesinskice5e56e2016-10-21 17:56:45 -070065static bool RequiredNameIsJavaClassName(xml::Element* el,
Adam Lesinskicacb28f2016-10-19 12:18:14 -070066 SourcePathDiagnostics* diag) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070067 if (xml::Attribute* attr = el->FindAttribute(xml::kSchemaAndroid, "name")) {
68 return NameIsJavaClassName(el, attr, diag);
Adam Lesinskicacb28f2016-10-19 12:18:14 -070069 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070070 diag->Error(DiagMessage(el->line_number)
Adam Lesinskicacb28f2016-10-19 12:18:14 -070071 << "<" << el->name << "> is missing attribute 'android:name'");
72 return false;
Adam Lesinski52364f72016-01-11 13:10:24 -080073}
74
Adam Lesinskice5e56e2016-10-21 17:56:45 -070075static bool VerifyManifest(xml::Element* el, SourcePathDiagnostics* diag) {
76 xml::Attribute* attr = el->FindAttribute({}, "package");
Adam Lesinskicacb28f2016-10-19 12:18:14 -070077 if (!attr) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070078 diag->Error(DiagMessage(el->line_number)
Adam Lesinskicacb28f2016-10-19 12:18:14 -070079 << "<manifest> tag is missing 'package' attribute");
80 return false;
Adam Lesinskice5e56e2016-10-21 17:56:45 -070081 } else if (ResourceUtils::IsReference(attr->value)) {
82 diag->Error(
83 DiagMessage(el->line_number)
Adam Lesinskicacb28f2016-10-19 12:18:14 -070084 << "attribute 'package' in <manifest> tag must not be a reference");
85 return false;
Adam Lesinskice5e56e2016-10-21 17:56:45 -070086 } else if (!util::IsJavaPackageName(attr->value)) {
87 diag->Error(DiagMessage(el->line_number)
Adam Lesinskicacb28f2016-10-19 12:18:14 -070088 << "attribute 'package' in <manifest> tag is not a valid Java "
89 "package name: '"
90 << attr->value << "'");
91 return false;
92 }
93 return true;
Adam Lesinski52364f72016-01-11 13:10:24 -080094}
95
Adam Lesinski6b17d2c2016-08-10 11:37:06 -070096/**
Adam Lesinskicacb28f2016-10-19 12:18:14 -070097 * The coreApp attribute in <manifest> is not a regular AAPT attribute, so type
Adam Lesinskice5e56e2016-10-21 17:56:45 -070098 * checking on it is manual.
Adam Lesinski6b17d2c2016-08-10 11:37:06 -070099 */
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700100static bool FixCoreAppAttribute(xml::Element* el, SourcePathDiagnostics* diag) {
101 if (xml::Attribute* attr = el->FindAttribute("", "coreApp")) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700102 std::unique_ptr<BinaryPrimitive> result =
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700103 ResourceUtils::TryParseBool(attr->value);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700104 if (!result) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700105 diag->Error(DiagMessage(el->line_number)
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700106 << "attribute coreApp must be a boolean");
107 return false;
Adam Lesinski6b17d2c2016-08-10 11:37:06 -0700108 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700109 attr->compiled_value = std::move(result);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700110 }
111 return true;
Adam Lesinski6b17d2c2016-08-10 11:37:06 -0700112}
113
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700114bool ManifestFixer::BuildRules(xml::XmlActionExecutor* executor,
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700115 IDiagnostics* diag) {
116 // First verify some options.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700117 if (options_.rename_manifest_package) {
118 if (!util::IsJavaPackageName(options_.rename_manifest_package.value())) {
119 diag->Error(DiagMessage() << "invalid manifest package override '"
120 << options_.rename_manifest_package.value()
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700121 << "'");
122 return false;
123 }
124 }
125
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700126 if (options_.rename_instrumentation_target_package) {
127 if (!util::IsJavaPackageName(
128 options_.rename_instrumentation_target_package.value())) {
129 diag->Error(DiagMessage()
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700130 << "invalid instrumentation target package override '"
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700131 << options_.rename_instrumentation_target_package.value()
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700132 << "'");
133 return false;
134 }
135 }
136
137 // Common intent-filter actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700138 xml::XmlNodeAction intent_filter_action;
139 intent_filter_action["action"];
140 intent_filter_action["category"];
141 intent_filter_action["data"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700142
143 // Common meta-data actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700144 xml::XmlNodeAction meta_data_action;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700145
146 // Manifest actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700147 xml::XmlNodeAction& manifest_action = (*executor)["manifest"];
148 manifest_action.Action(VerifyManifest);
149 manifest_action.Action(FixCoreAppAttribute);
150 manifest_action.Action([&](xml::Element* el) -> bool {
151 if (options_.version_name_default) {
152 if (el->FindAttribute(xml::kSchemaAndroid, "versionName") == nullptr) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700153 el->attributes.push_back(
154 xml::Attribute{xml::kSchemaAndroid, "versionName",
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700155 options_.version_name_default.value()});
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700156 }
Adam Lesinskicc5609d2016-04-05 12:41:07 -0700157 }
158
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700159 if (options_.version_code_default) {
160 if (el->FindAttribute(xml::kSchemaAndroid, "versionCode") == nullptr) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700161 el->attributes.push_back(
162 xml::Attribute{xml::kSchemaAndroid, "versionCode",
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700163 options_.version_code_default.value()});
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700164 }
Adam Lesinskicc5609d2016-04-05 12:41:07 -0700165 }
Adam Lesinski2ae4a872015-11-02 16:10:55 -0800166 return true;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700167 });
168
169 // Meta tags.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700170 manifest_action["eat-comment"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700171
172 // Uses-sdk actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700173 manifest_action["uses-sdk"].Action([&](xml::Element* el) -> bool {
174 if (options_.min_sdk_version_default &&
175 el->FindAttribute(xml::kSchemaAndroid, "minSdkVersion") == nullptr) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700176 // There was no minSdkVersion defined and we have a default to assign.
177 el->attributes.push_back(
178 xml::Attribute{xml::kSchemaAndroid, "minSdkVersion",
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700179 options_.min_sdk_version_default.value()});
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700180 }
181
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700182 if (options_.target_sdk_version_default &&
183 el->FindAttribute(xml::kSchemaAndroid, "targetSdkVersion") == nullptr) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700184 // There was no targetSdkVersion defined and we have a default to assign.
185 el->attributes.push_back(
186 xml::Attribute{xml::kSchemaAndroid, "targetSdkVersion",
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700187 options_.target_sdk_version_default.value()});
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700188 }
189 return true;
190 });
191
192 // Instrumentation actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700193 manifest_action["instrumentation"].Action([&](xml::Element* el) -> bool {
194 if (!options_.rename_instrumentation_target_package) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700195 return true;
196 }
197
198 if (xml::Attribute* attr =
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700199 el->FindAttribute(xml::kSchemaAndroid, "targetPackage")) {
200 attr->value = options_.rename_instrumentation_target_package.value();
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700201 }
202 return true;
203 });
204
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700205 manifest_action["original-package"];
206 manifest_action["protected-broadcast"];
207 manifest_action["uses-permission"];
208 manifest_action["permission"];
209 manifest_action["permission-tree"];
210 manifest_action["permission-group"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700211
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700212 manifest_action["uses-configuration"];
213 manifest_action["uses-feature"];
214 manifest_action["supports-screens"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700215
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700216 manifest_action["compatible-screens"];
217 manifest_action["compatible-screens"]["screen"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700218
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700219 manifest_action["supports-gl-texture"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700220
Adam Lesinski5119e512016-12-05 19:48:20 -0800221 manifest_action["meta-data"] = meta_data_action;
222
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700223 // Application actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700224 xml::XmlNodeAction& application_action = manifest_action["application"];
225 application_action.Action(OptionalNameIsJavaClassName);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700226
227 // Uses library actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700228 application_action["uses-library"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700229
230 // Meta-data.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700231 application_action["meta-data"] = meta_data_action;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700232
233 // Activity actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700234 application_action["activity"].Action(RequiredNameIsJavaClassName);
235 application_action["activity"]["intent-filter"] = intent_filter_action;
236 application_action["activity"]["meta-data"] = meta_data_action;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700237
238 // Activity alias actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700239 application_action["activity-alias"]["intent-filter"] = intent_filter_action;
240 application_action["activity-alias"]["meta-data"] = meta_data_action;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700241
242 // Service actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700243 application_action["service"].Action(RequiredNameIsJavaClassName);
244 application_action["service"]["intent-filter"] = intent_filter_action;
245 application_action["service"]["meta-data"] = meta_data_action;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700246
247 // Receiver actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700248 application_action["receiver"].Action(RequiredNameIsJavaClassName);
249 application_action["receiver"]["intent-filter"] = intent_filter_action;
250 application_action["receiver"]["meta-data"] = meta_data_action;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700251
252 // Provider actions.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700253 application_action["provider"].Action(RequiredNameIsJavaClassName);
254 application_action["provider"]["intent-filter"] = intent_filter_action;
255 application_action["provider"]["meta-data"] = meta_data_action;
256 application_action["provider"]["grant-uri-permissions"];
257 application_action["provider"]["path-permissions"];
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700258
259 return true;
Adam Lesinski2ae4a872015-11-02 16:10:55 -0800260}
261
Adam Lesinski52364f72016-01-11 13:10:24 -0800262class FullyQualifiedClassNameVisitor : public xml::Visitor {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700263 public:
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700264 using xml::Visitor::Visit;
Adam Lesinski52364f72016-01-11 13:10:24 -0800265
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700266 explicit FullyQualifiedClassNameVisitor(const StringPiece& package)
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700267 : package_(package) {}
Adam Lesinski52364f72016-01-11 13:10:24 -0800268
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700269 void Visit(xml::Element* el) override {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700270 for (xml::Attribute& attr : el->attributes) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700271 if (attr.namespace_uri == xml::kSchemaAndroid &&
272 class_attributes_.find(attr.name) != class_attributes_.end()) {
273 if (Maybe<std::string> new_value =
274 util::GetFullyQualifiedClassName(package_, attr.value)) {
275 attr.value = std::move(new_value.value());
Adam Lesinski52364f72016-01-11 13:10:24 -0800276 }
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700277 }
Adam Lesinski52364f72016-01-11 13:10:24 -0800278 }
279
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700280 // Super implementation to iterate over the children.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700281 xml::Visitor::Visit(el);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700282 }
283
284 private:
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700285 StringPiece package_;
286 std::unordered_set<StringPiece> class_attributes_ = {"name"};
Adam Lesinski52364f72016-01-11 13:10:24 -0800287};
288
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700289static bool RenameManifestPackage(const StringPiece& package_override,
290 xml::Element* manifest_el) {
291 xml::Attribute* attr = manifest_el->FindAttribute({}, "package");
Adam Lesinski52364f72016-01-11 13:10:24 -0800292
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700293 // We've already verified that the manifest element is present, with a package
294 // name specified.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700295 CHECK(attr != nullptr);
Adam Lesinski52364f72016-01-11 13:10:24 -0800296
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700297 std::string original_package = std::move(attr->value);
Adam Lesinskid5083f62017-01-16 15:07:21 -0800298 attr->value = package_override.to_string();
Adam Lesinski52364f72016-01-11 13:10:24 -0800299
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700300 FullyQualifiedClassNameVisitor visitor(original_package);
301 manifest_el->Accept(&visitor);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700302 return true;
Adam Lesinski52364f72016-01-11 13:10:24 -0800303}
304
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700305bool ManifestFixer::Consume(IAaptContext* context, xml::XmlResource* doc) {
306 xml::Element* root = xml::FindRootElement(doc->root.get());
307 if (!root || !root->namespace_uri.empty() || root->name != "manifest") {
308 context->GetDiagnostics()->Error(DiagMessage(doc->file.source)
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700309 << "root tag must be <manifest>");
310 return false;
311 }
Adam Lesinski2ae4a872015-11-02 16:10:55 -0800312
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700313 if ((options_.min_sdk_version_default ||
314 options_.target_sdk_version_default) &&
315 root->FindChild({}, "uses-sdk") == nullptr) {
Adam Lesinskie343eb12016-10-27 16:31:58 -0700316 // Auto insert a <uses-sdk> element. This must be inserted before the
317 // <application> tag. The device runtime PackageParser will make SDK version
318 // decisions while parsing <application>.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700319 std::unique_ptr<xml::Element> uses_sdk = util::make_unique<xml::Element>();
320 uses_sdk->name = "uses-sdk";
Adam Lesinskie343eb12016-10-27 16:31:58 -0700321 root->InsertChild(0, std::move(uses_sdk));
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700322 }
Adam Lesinski2ae4a872015-11-02 16:10:55 -0800323
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700324 xml::XmlActionExecutor executor;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700325 if (!BuildRules(&executor, context->GetDiagnostics())) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700326 return false;
327 }
Adam Lesinskicc5609d2016-04-05 12:41:07 -0700328
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700329 if (!executor.Execute(xml::XmlActionExecutorPolicy::kWhitelist,
330 context->GetDiagnostics(), doc)) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700331 return false;
332 }
Adam Lesinskicc5609d2016-04-05 12:41:07 -0700333
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700334 if (options_.rename_manifest_package) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700335 // Rename manifest package outside of the XmlActionExecutor.
Adam Lesinskie343eb12016-10-27 16:31:58 -0700336 // We need to extract the old package name and FullyQualify all class
337 // names.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700338 if (!RenameManifestPackage(options_.rename_manifest_package.value(),
339 root)) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700340 return false;
Adam Lesinskicc5609d2016-04-05 12:41:07 -0700341 }
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700342 }
343 return true;
Adam Lesinski2ae4a872015-11-02 16:10:55 -0800344}
345
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700346} // namespace aapt