blob: bde9b0be43615fb5e015f1c053c47745d984c3fa [file] [log] [blame]
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -08001/*
2 * Copyright (C) 2021 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
17#include "idmap2/FabricatedOverlay.h"
18
19#include <androidfw/ResourceUtils.h>
Jeremy Meyer65871022022-07-12 22:45:08 +000020#include <androidfw/StringPool.h>
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080021#include <google/protobuf/io/coded_stream.h>
Jeremy Meyer65871022022-07-12 22:45:08 +000022#include <google/protobuf/io/zero_copy_stream_impl.h>
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080023#include <utils/ByteOrder.h>
24#include <zlib.h>
25
26#include <fstream>
Mårten Kongstad1195a6b2021-05-11 12:57:01 +000027#include <map>
28#include <memory>
29#include <string>
30#include <utility>
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080031
32namespace android::idmap2 {
33
Jeremy Meyer65871022022-07-12 22:45:08 +000034constexpr auto kBufferSize = 1024;
35
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080036namespace {
37bool Read32(std::istream& stream, uint32_t* out) {
38 uint32_t value;
39 if (stream.read(reinterpret_cast<char*>(&value), sizeof(uint32_t))) {
40 *out = dtohl(value);
41 return true;
42 }
43 return false;
44}
45
46void Write32(std::ostream& stream, uint32_t value) {
47 uint32_t x = htodl(value);
48 stream.write(reinterpret_cast<char*>(&x), sizeof(uint32_t));
49}
50} // namespace
51
52FabricatedOverlay::FabricatedOverlay(pb::FabricatedOverlay&& overlay,
Jeremy Meyer65871022022-07-12 22:45:08 +000053 std::string&& string_pool_data,
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080054 std::optional<uint32_t> crc_from_disk)
Jeremy Meyer65871022022-07-12 22:45:08 +000055 : overlay_pb_(std::forward<pb::FabricatedOverlay>(overlay)),
56 string_pool_data_(std::move(string_pool_data)),
57 crc_from_disk_(crc_from_disk) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080058}
59
60FabricatedOverlay::Builder::Builder(const std::string& package_name, const std::string& name,
61 const std::string& target_package_name) {
62 package_name_ = package_name;
63 name_ = name;
64 target_package_name_ = target_package_name;
65}
66
67FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetOverlayable(const std::string& name) {
68 target_overlayable_ = name;
69 return *this;
70}
71
72FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue(
Jeremy Meyer77c8a932022-08-18 22:06:55 +000073 const std::string& resource_name, uint8_t data_type, uint32_t data_value,
74 const std::string& configuration) {
75 entries_.emplace_back(Entry{resource_name, data_type, data_value, "", configuration});
Jeremy Meyer05466592022-06-02 21:31:15 +000076 return *this;
77}
78
79FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue(
Jeremy Meyer77c8a932022-08-18 22:06:55 +000080 const std::string& resource_name, uint8_t data_type, const std::string& data_string_value,
81 const std::string& configuration) {
82 entries_.emplace_back(Entry{resource_name, data_type, 0, data_string_value, configuration});
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080083 return *this;
84}
85
86Result<FabricatedOverlay> FabricatedOverlay::Builder::Build() {
Jeremy Meyer77c8a932022-08-18 22:06:55 +000087 using ConfigMap = std::map<std::string, TargetValue>;
88 using EntryMap = std::map<std::string, ConfigMap>;
Jeremy Meyer65871022022-07-12 22:45:08 +000089 using TypeMap = std::map<std::string, EntryMap>;
90 using PackageMap = std::map<std::string, TypeMap>;
91 PackageMap package_map;
92 android::StringPool string_pool;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080093 for (const auto& res_entry : entries_) {
94 StringPiece package_substr;
95 StringPiece type_name;
96 StringPiece entry_name;
97 if (!android::ExtractResourceName(StringPiece(res_entry.resource_name), &package_substr,
98 &type_name, &entry_name)) {
99 return Error("failed to parse resource name '%s'", res_entry.resource_name.c_str());
100 }
101
102 std::string package_name =
103 package_substr.empty() ? target_package_name_ : package_substr.to_string();
104 if (type_name.empty()) {
105 return Error("resource name '%s' missing type name", res_entry.resource_name.c_str());
106 }
107
108 if (entry_name.empty()) {
109 return Error("resource name '%s' missing entry name", res_entry.resource_name.c_str());
110 }
111
Jeremy Meyer65871022022-07-12 22:45:08 +0000112 auto package = package_map.find(package_name);
113 if (package == package_map.end()) {
114 package = package_map
115 .insert(std::make_pair(package_name, TypeMap()))
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800116 .first;
117 }
118
119 auto type = package->second.find(type_name.to_string());
120 if (type == package->second.end()) {
121 type =
122 package->second
Jeremy Meyer65871022022-07-12 22:45:08 +0000123 .insert(std::make_pair(type_name.to_string(), EntryMap()))
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800124 .first;
125 }
126
127 auto entry = type->second.find(entry_name.to_string());
128 if (entry == type->second.end()) {
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000129 entry = type->second.insert(std::make_pair(entry_name.to_string(), ConfigMap())).first;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800130 }
131
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000132 auto value = entry->second.find(res_entry.configuration);
133 if (value == entry->second.end()) {
134 value = entry->second.insert(std::make_pair(res_entry.configuration, TargetValue())).first;
135 }
136
137 value->second = TargetValue{res_entry.data_type, res_entry.data_value,
138 res_entry.data_string_value};
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800139 }
140
141 pb::FabricatedOverlay overlay_pb;
142 overlay_pb.set_package_name(package_name_);
143 overlay_pb.set_name(name_);
144 overlay_pb.set_target_package_name(target_package_name_);
145 overlay_pb.set_target_overlayable(target_overlayable_);
146
Jeremy Meyer65871022022-07-12 22:45:08 +0000147 for (auto& package : package_map) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800148 auto package_pb = overlay_pb.add_packages();
149 package_pb->set_name(package.first);
150
Jeremy Meyer65871022022-07-12 22:45:08 +0000151 for (auto& type : package.second) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800152 auto type_pb = package_pb->add_types();
153 type_pb->set_name(type.first);
154
Jeremy Meyer65871022022-07-12 22:45:08 +0000155 for (auto& entry : type.second) {
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000156 for (const auto& value: entry.second) {
157 auto entry_pb = type_pb->add_entries();
158 entry_pb->set_name(entry.first);
159 entry_pb->set_configuration(value.first);
160 pb::ResourceValue* pb_value = entry_pb->mutable_res_value();
161 pb_value->set_data_type(value.second.data_type);
162 if (value.second.data_type == Res_value::TYPE_STRING) {
163 auto ref = string_pool.MakeRef(value.second.data_string_value);
164 pb_value->set_data_value(ref.index());
165 } else {
166 pb_value->set_data_value(value.second.data_value);
167 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000168 }
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800169 }
170 }
171 }
172
Jeremy Meyer65871022022-07-12 22:45:08 +0000173 android::BigBuffer string_buffer(kBufferSize);
174 android::StringPool::FlattenUtf8(&string_buffer, string_pool, nullptr);
175 return FabricatedOverlay(std::move(overlay_pb), string_buffer.to_string());
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800176}
177
178Result<FabricatedOverlay> FabricatedOverlay::FromBinaryStream(std::istream& stream) {
179 uint32_t magic;
180 if (!Read32(stream, &magic)) {
181 return Error("Failed to read fabricated overlay magic.");
182 }
183
184 if (magic != kFabricatedOverlayMagic) {
185 return Error("Not a fabricated overlay file.");
186 }
187
188 uint32_t version;
189 if (!Read32(stream, &version)) {
190 return Error("Failed to read fabricated overlay version.");
191 }
192
Jeremy Meyer65871022022-07-12 22:45:08 +0000193 if (version != 1 && version != 2) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800194 return Error("Invalid fabricated overlay version '%u'.", version);
195 }
196
197 uint32_t crc;
198 if (!Read32(stream, &crc)) {
Jeremy Meyer65871022022-07-12 22:45:08 +0000199 return Error("Failed to read fabricated overlay crc.");
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800200 }
201
202 pb::FabricatedOverlay overlay{};
Jeremy Meyer65871022022-07-12 22:45:08 +0000203 std::string sp_data;
204 if (version == 2) {
205 uint32_t sp_size;
206 if (!Read32(stream, &sp_size)) {
207 return Error("Failed read string pool size.");
208 }
209 std::string buf(sp_size, '\0');
210 if (!stream.read(buf.data(), sp_size)) {
211 return Error("Failed to read string pool.");
212 }
213 sp_data = buf;
214
215 if (!overlay.ParseFromIstream(&stream)) {
216 return Error("Failed read fabricated overlay proto.");
217 }
218 } else {
219 if (!overlay.ParseFromIstream(&stream)) {
220 return Error("Failed read fabricated overlay proto.");
221 }
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800222 }
223
224 // If the proto version is the latest version, then the contents of the proto must be the same
225 // when the proto is re-serialized; otherwise, the crc must be calculated because migrating the
226 // proto to the latest version will likely change the contents of the fabricated overlay.
Jeremy Meyer65871022022-07-12 22:45:08 +0000227 return FabricatedOverlay(std::move(overlay), std::move(sp_data),
228 version == kFabricatedOverlayCurrentVersion
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800229 ? std::optional<uint32_t>(crc)
230 : std::nullopt);
231}
232
233Result<FabricatedOverlay::SerializedData*> FabricatedOverlay::InitializeData() const {
234 if (!data_.has_value()) {
Jeremy Meyer65871022022-07-12 22:45:08 +0000235 auto pb_size = overlay_pb_.ByteSizeLong();
236 auto pb_data = std::unique_ptr<uint8_t[]>(new uint8_t[pb_size]);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800237
238 // Ensure serialization is deterministic
Jeremy Meyer65871022022-07-12 22:45:08 +0000239 google::protobuf::io::ArrayOutputStream array_stream(pb_data.get(), pb_size);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800240 google::protobuf::io::CodedOutputStream output_stream(&array_stream);
241 output_stream.SetSerializationDeterministic(true);
242 overlay_pb_.SerializeWithCachedSizes(&output_stream);
Jeremy Meyer65871022022-07-12 22:45:08 +0000243 if (output_stream.HadError() || pb_size != output_stream.ByteCount()) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800244 return Error("Failed to serialize fabricated overlay.");
245 }
246
247 // Calculate the crc using the proto data and the version.
Jeremy Meyer65871022022-07-12 22:45:08 +0000248 uint32_t pb_crc = crc32(0L, Z_NULL, 0);
249 pb_crc = crc32(pb_crc, reinterpret_cast<const uint8_t*>(&kFabricatedOverlayCurrentVersion),
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800250 sizeof(uint32_t));
Jeremy Meyer65871022022-07-12 22:45:08 +0000251 pb_crc = crc32(pb_crc, pb_data.get(), pb_size);
252
253 data_ = SerializedData{std::move(pb_data), pb_size, pb_crc, string_pool_data_};
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800254 }
255 return &(*data_);
256}
257Result<uint32_t> FabricatedOverlay::GetCrc() const {
258 if (crc_from_disk_.has_value()) {
259 return *crc_from_disk_;
260 }
261 auto data = InitializeData();
262 if (!data) {
263 return data.GetError();
264 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000265 return (*data)->pb_crc;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800266}
267
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800268Result<Unit> FabricatedOverlay::ToBinaryStream(std::ostream& stream) const {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800269 auto data = InitializeData();
270 if (!data) {
271 return data.GetError();
272 }
273
274 Write32(stream, kFabricatedOverlayMagic);
275 Write32(stream, kFabricatedOverlayCurrentVersion);
Jeremy Meyer65871022022-07-12 22:45:08 +0000276 Write32(stream, (*data)->pb_crc);
277 Write32(stream, (*data)->sp_data.length());
278 stream.write((*data)->sp_data.data(), (*data)->sp_data.length());
279 if (stream.bad()) {
280 return Error("Failed to write string pool data.");
281 }
282 stream.write(reinterpret_cast<const char*>((*data)->pb_data.get()), (*data)->pb_data_size);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800283 if (stream.bad()) {
284 return Error("Failed to write serialized fabricated overlay.");
285 }
286
287 return Unit{};
288}
289
290using FabContainer = FabricatedOverlayContainer;
291FabContainer::FabricatedOverlayContainer(FabricatedOverlay&& overlay, std::string&& path)
292 : overlay_(std::forward<FabricatedOverlay>(overlay)), path_(std::forward<std::string>(path)) {
293}
294
295FabContainer::~FabricatedOverlayContainer() = default;
296
297Result<std::unique_ptr<FabContainer>> FabContainer::FromPath(std::string path) {
298 std::fstream fin(path);
299 auto overlay = FabricatedOverlay::FromBinaryStream(fin);
300 if (!overlay) {
301 return overlay.GetError();
302 }
303 return std::unique_ptr<FabContainer>(
304 new FabricatedOverlayContainer(std::move(*overlay), std::move(path)));
305}
306
307std::unique_ptr<FabricatedOverlayContainer> FabContainer::FromOverlay(FabricatedOverlay&& overlay) {
308 return std::unique_ptr<FabContainer>(
309 new FabricatedOverlayContainer(std::move(overlay), {} /* path */));
310}
311
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800312OverlayManifestInfo FabContainer::GetManifestInfo() const {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800313 const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800314 return OverlayManifestInfo{
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800315 .package_name = overlay_pb.package_name(),
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800316 .name = overlay_pb.name(),
317 .target_package = overlay_pb.target_package_name(),
318 .target_name = overlay_pb.target_overlayable(),
319 };
320}
321
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800322Result<OverlayManifestInfo> FabContainer::FindOverlayInfo(const std::string& name) const {
323 const OverlayManifestInfo info = GetManifestInfo();
324 if (name != info.name) {
325 return Error("Failed to find name '%s' in fabricated overlay", name.c_str());
326 }
327 return info;
328}
329
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800330Result<OverlayData> FabContainer::GetOverlayData(const OverlayManifestInfo& info) const {
331 const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_;
332 if (info.name != overlay_pb.name()) {
333 return Error("Failed to find name '%s' in fabricated overlay", info.name.c_str());
334 }
335
336 OverlayData result{};
337 for (const auto& package : overlay_pb.packages()) {
338 for (const auto& type : package.types()) {
339 for (const auto& entry : type.entries()) {
340 auto name = base::StringPrintf("%s:%s/%s", package.name().c_str(), type.name().c_str(),
341 entry.name().c_str());
342 const auto& res_value = entry.res_value();
343 result.pairs.emplace_back(OverlayData::Value{
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000344 name, TargetValueWithConfig{.config = entry.configuration(), .value = TargetValue{
345 .data_type = static_cast<uint8_t>(res_value.data_type()),
346 .data_value = res_value.data_value()}}});
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800347 }
348 }
349 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000350 const uint32_t string_pool_data_length = overlay_.string_pool_data_.length();
351 result.string_pool_data = OverlayData::InlineStringPoolData{
352 .data = std::unique_ptr<uint8_t[]>(new uint8_t[string_pool_data_length]),
353 .data_length = string_pool_data_length,
354 .string_pool_offset = 0,
355 };
356 memcpy(result.string_pool_data->data.get(), overlay_.string_pool_data_.data(),
357 string_pool_data_length);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800358 return result;
359}
360
361Result<uint32_t> FabContainer::GetCrc() const {
362 return overlay_.GetCrc();
363}
364
365const std::string& FabContainer::GetPath() const {
366 return path_;
367}
368
369Result<std::string> FabContainer::GetResourceName(ResourceId /* id */) const {
370 return Error("Fabricated overlay does not contain resources.");
371}
372
Mårten Kongstad1195a6b2021-05-11 12:57:01 +0000373} // namespace android::idmap2