blob: a5df746ca73304701cb31c8a5c1d6b2abf5a9451 [file] [log] [blame]
Mårten Kongstad02751232018-04-27 13:16:32 +02001/*
2 * Copyright (C) 2018 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
Ryan Mitchell52e1f7a2019-04-12 12:31:42 -070017#include "idmap2/ResourceUtils.h"
18
Ryan Mitchella3628462019-01-14 12:19:40 -080019#include <memory>
Mårten Kongstad02751232018-04-27 13:16:32 +020020#include <string>
Mårten Kongstad02751232018-04-27 13:16:32 +020021
22#include "androidfw/StringPiece.h"
23#include "androidfw/Util.h"
Mårten Kongstad0f763112018-11-19 14:14:37 +010024#include "idmap2/Result.h"
Ryan Mitchellcd965a32019-09-18 14:52:45 -070025#include "idmap2/XmlParser.h"
Ryan Mitchella3628462019-01-14 12:19:40 -080026#include "idmap2/ZipFile.h"
Mårten Kongstad02751232018-04-27 13:16:32 +020027
28using android::StringPiece16;
Ryan Mitchella3628462019-01-14 12:19:40 -080029using android::idmap2::Result;
Ryan Mitchellcd965a32019-09-18 14:52:45 -070030using android::idmap2::XmlParser;
Ryan Mitchella3628462019-01-14 12:19:40 -080031using android::idmap2::ZipFile;
Mårten Kongstad02751232018-04-27 13:16:32 +020032using android::util::Utf16ToUtf8;
33
Mårten Kongstad0eba72a2018-11-29 08:23:14 +010034namespace android::idmap2::utils {
Mårten Kongstad02751232018-04-27 13:16:32 +020035
Ryan Mitchelle753ffe2019-09-23 09:47:02 -070036StringPiece DataTypeToString(uint8_t data_type) {
37 switch (data_type) {
38 case Res_value::TYPE_NULL:
39 return "null";
40 case Res_value::TYPE_REFERENCE:
41 return "reference";
42 case Res_value::TYPE_ATTRIBUTE:
43 return "attribute";
44 case Res_value::TYPE_STRING:
45 return "string";
46 case Res_value::TYPE_FLOAT:
47 return "float";
48 case Res_value::TYPE_DIMENSION:
49 return "dimension";
50 case Res_value::TYPE_FRACTION:
51 return "fraction";
52 case Res_value::TYPE_DYNAMIC_REFERENCE:
53 return "reference (dynamic)";
54 case Res_value::TYPE_DYNAMIC_ATTRIBUTE:
55 return "attribute (dynamic)";
56 case Res_value::TYPE_INT_DEC:
57 case Res_value::TYPE_INT_HEX:
58 return "integer";
59 case Res_value::TYPE_INT_BOOLEAN:
60 return "boolean";
61 case Res_value::TYPE_INT_COLOR_ARGB8:
62 case Res_value::TYPE_INT_COLOR_RGB8:
63 case Res_value::TYPE_INT_COLOR_RGB4:
64 return "color";
65 default:
66 return "unknown";
67 }
68}
69
Ryan Mitchellcd965a32019-09-18 14:52:45 -070070Result<std::string> ResToTypeEntryName(const AssetManager2& am, uint32_t resid) {
Mårten Kongstad02751232018-04-27 13:16:32 +020071 AssetManager2::ResourceName name;
72 if (!am.GetResourceName(resid, &name)) {
Mårten Kongstad49d835d2019-01-31 10:50:48 +010073 return Error("no resource 0x%08x in asset manager", resid);
Mårten Kongstad02751232018-04-27 13:16:32 +020074 }
75 std::string out;
76 if (name.type != nullptr) {
77 out.append(name.type, name.type_len);
78 } else {
79 out += Utf16ToUtf8(StringPiece16(name.type16, name.type_len));
80 }
81 out.append("/");
82 if (name.entry != nullptr) {
83 out.append(name.entry, name.entry_len);
84 } else {
85 out += Utf16ToUtf8(StringPiece16(name.entry16, name.entry_len));
86 }
Mårten Kongstad49d835d2019-01-31 10:50:48 +010087 return out;
Mårten Kongstad02751232018-04-27 13:16:32 +020088}
89
Ryan Mitchella3628462019-01-14 12:19:40 -080090Result<OverlayManifestInfo> ExtractOverlayManifestInfo(const std::string& path,
Ryan Mitchella3628462019-01-14 12:19:40 -080091 bool assert_overlay) {
92 std::unique_ptr<const ZipFile> zip = ZipFile::Open(path);
93 if (!zip) {
Mårten Kongstad49d835d2019-01-31 10:50:48 +010094 return Error("failed to open %s as a zip file", path.c_str());
Ryan Mitchella3628462019-01-14 12:19:40 -080095 }
96
97 std::unique_ptr<const MemoryChunk> entry = zip->Uncompress("AndroidManifest.xml");
98 if (!entry) {
Mårten Kongstad49d835d2019-01-31 10:50:48 +010099 return Error("failed to uncompress AndroidManifest.xml from %s", path.c_str());
Ryan Mitchella3628462019-01-14 12:19:40 -0800100 }
101
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700102 Result<std::unique_ptr<const XmlParser>> xml = XmlParser::Create(entry->buf, entry->size);
Ryan Mitchella3628462019-01-14 12:19:40 -0800103 if (!xml) {
Mårten Kongstad49d835d2019-01-31 10:50:48 +0100104 return Error("failed to parse AndroidManifest.xml from %s", path.c_str());
Ryan Mitchella3628462019-01-14 12:19:40 -0800105 }
106
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700107 auto manifest_it = (*xml)->tree_iterator();
108 if (manifest_it->event() != XmlParser::Event::START_TAG || manifest_it->name() != "manifest") {
109 return Error("root element tag is not <manifest> in AndroidManifest.xml of %s", path.c_str());
110 }
111
112 auto overlay_it = std::find_if(manifest_it.begin(), manifest_it.end(), [](const auto& it) {
113 return it.event() == XmlParser::Event::START_TAG && it.name() == "overlay";
114 });
115
Ryan Mitchella3628462019-01-14 12:19:40 -0800116 OverlayManifestInfo info{};
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700117 if (overlay_it == manifest_it.end()) {
118 if (!assert_overlay) {
119 return info;
Ryan Mitchella3628462019-01-14 12:19:40 -0800120 }
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700121 return Error("<overlay> missing from AndroidManifest.xml of %s", path.c_str());
Ryan Mitchella3628462019-01-14 12:19:40 -0800122 }
123
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700124 if (auto result_str = overlay_it->GetAttributeStringValue("targetPackage")) {
125 info.target_package = *result_str;
Ryan Mitchella3628462019-01-14 12:19:40 -0800126 } else {
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700127 return Error("android:targetPackage missing from <overlay> of %s: %s", path.c_str(),
128 result_str.GetErrorMessage().c_str());
Ryan Mitchella3628462019-01-14 12:19:40 -0800129 }
130
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700131 if (auto result_str = overlay_it->GetAttributeStringValue("targetName")) {
132 info.target_name = *result_str;
Ryan Mitchella3628462019-01-14 12:19:40 -0800133 }
134
Ryan Mitchell9e4f52b2019-09-19 12:15:52 -0700135 if (auto result_value = overlay_it->GetAttributeValue("resourcesMap")) {
136 if ((*result_value).dataType == Res_value::TYPE_REFERENCE) {
137 info.resource_mapping = (*result_value).data;
138 } else {
139 return Error("android:resourcesMap is not a reference in AndroidManifest.xml of %s",
140 path.c_str());
141 }
142 }
143
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700144 if (auto result_value = overlay_it->GetAttributeValue("isStatic")) {
145 if ((*result_value).dataType >= Res_value::TYPE_FIRST_INT &&
146 (*result_value).dataType <= Res_value::TYPE_LAST_INT) {
147 info.is_static = (*result_value).data != 0U;
148 } else {
149 return Error("android:isStatic is not a boolean in AndroidManifest.xml of %s", path.c_str());
150 }
Ryan Mitchella3628462019-01-14 12:19:40 -0800151 }
152
Ryan Mitchellcd965a32019-09-18 14:52:45 -0700153 if (auto result_value = overlay_it->GetAttributeValue("priority")) {
154 if ((*result_value).dataType >= Res_value::TYPE_FIRST_INT &&
155 (*result_value).dataType <= Res_value::TYPE_LAST_INT) {
156 info.priority = (*result_value).data;
157 } else {
158 return Error("android:priority is not an integer in AndroidManifest.xml of %s", path.c_str());
159 }
Ryan Mitchella3628462019-01-14 12:19:40 -0800160 }
161
Ryan Mitchell9e4f52b2019-09-19 12:15:52 -0700162 if (auto result_str = overlay_it->GetAttributeStringValue("requiredSystemPropertyName")) {
163 info.requiredSystemPropertyName = *result_str;
Gabriel Siqueirad7fc4f72019-09-17 18:42:07 -0300164 }
165
Ryan Mitchelle753ffe2019-09-23 09:47:02 -0700166 if (auto result_str = overlay_it->GetAttributeStringValue("requiredSystemPropertyValue")) {
Ryan Mitchell9e4f52b2019-09-19 12:15:52 -0700167 info.requiredSystemPropertyValue = *result_str;
Gabriel Siqueirad7fc4f72019-09-17 18:42:07 -0300168 }
169
Ryan Mitchella3628462019-01-14 12:19:40 -0800170 return info;
171}
172
Mårten Kongstad0eba72a2018-11-29 08:23:14 +0100173} // namespace android::idmap2::utils