blob: 47daf23c63815a64bec401d255a68ff3d8adfa93 [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
Jeremy Meyera761f332022-10-21 17:42:14 +000019#include <sys/stat.h> // umask
20#include <sys/types.h> // umask
21
22#include <android-base/file.h>
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080023#include <androidfw/ResourceUtils.h>
Jeremy Meyer65871022022-07-12 22:45:08 +000024#include <androidfw/StringPool.h>
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080025#include <google/protobuf/io/coded_stream.h>
Jeremy Meyer65871022022-07-12 22:45:08 +000026#include <google/protobuf/io/zero_copy_stream_impl.h>
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080027#include <utils/ByteOrder.h>
28#include <zlib.h>
29
30#include <fstream>
Mårten Kongstad1195a6b2021-05-11 12:57:01 +000031#include <map>
32#include <memory>
33#include <string>
34#include <utility>
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080035
36namespace android::idmap2 {
37
Jeremy Meyer65871022022-07-12 22:45:08 +000038constexpr auto kBufferSize = 1024;
39
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080040namespace {
41bool Read32(std::istream& stream, uint32_t* out) {
42 uint32_t value;
43 if (stream.read(reinterpret_cast<char*>(&value), sizeof(uint32_t))) {
44 *out = dtohl(value);
45 return true;
46 }
47 return false;
48}
49
50void Write32(std::ostream& stream, uint32_t value) {
51 uint32_t x = htodl(value);
52 stream.write(reinterpret_cast<char*>(&x), sizeof(uint32_t));
53}
54} // namespace
55
56FabricatedOverlay::FabricatedOverlay(pb::FabricatedOverlay&& overlay,
Jeremy Meyer65871022022-07-12 22:45:08 +000057 std::string&& string_pool_data,
Jeremy Meyerc04bcb32023-07-27 22:49:14 +000058 std::vector<FabricatedOverlay::BinaryData> binary_files,
Jeremy Meyera761f332022-10-21 17:42:14 +000059 off_t total_binary_bytes,
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080060 std::optional<uint32_t> crc_from_disk)
Jeremy Meyer65871022022-07-12 22:45:08 +000061 : overlay_pb_(std::forward<pb::FabricatedOverlay>(overlay)),
62 string_pool_data_(std::move(string_pool_data)),
Jeremy Meyera761f332022-10-21 17:42:14 +000063 binary_files_(std::move(binary_files)),
64 total_binary_bytes_(total_binary_bytes),
Jeremy Meyer65871022022-07-12 22:45:08 +000065 crc_from_disk_(crc_from_disk) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -080066}
67
68FabricatedOverlay::Builder::Builder(const std::string& package_name, const std::string& name,
69 const std::string& target_package_name) {
70 package_name_ = package_name;
71 name_ = name;
72 target_package_name_ = target_package_name;
73}
74
75FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetOverlayable(const std::string& name) {
76 target_overlayable_ = name;
77 return *this;
78}
79
80FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue(
Jeremy Meyer77c8a932022-08-18 22:06:55 +000081 const std::string& resource_name, uint8_t data_type, uint32_t data_value,
82 const std::string& configuration) {
Jeremy Meyera761f332022-10-21 17:42:14 +000083 entries_.emplace_back(
Jeremy Meyerc04bcb32023-07-27 22:49:14 +000084 Entry{resource_name, data_type, data_value, "", std::nullopt, 0, 0, configuration});
Jeremy Meyer05466592022-06-02 21:31:15 +000085 return *this;
86}
87
88FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue(
Jeremy Meyer77c8a932022-08-18 22:06:55 +000089 const std::string& resource_name, uint8_t data_type, const std::string& data_string_value,
90 const std::string& configuration) {
Jeremy Meyera761f332022-10-21 17:42:14 +000091 entries_.emplace_back(
Jeremy Meyerc04bcb32023-07-27 22:49:14 +000092 Entry{resource_name, data_type, 0, data_string_value, std::nullopt, 0, 0, configuration});
Jeremy Meyera761f332022-10-21 17:42:14 +000093 return *this;
94}
95
96FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue(
97 const std::string& resource_name, std::optional<android::base::borrowed_fd>&& binary_value,
Jeremy Meyerc04bcb32023-07-27 22:49:14 +000098 off64_t data_binary_offset, size_t data_binary_size, const std::string& configuration) {
99 entries_.emplace_back(Entry{resource_name, 0, 0, "", binary_value,
100 data_binary_offset, data_binary_size, configuration});
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800101 return *this;
102}
103
104Result<FabricatedOverlay> FabricatedOverlay::Builder::Build() {
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700105 using ConfigMap = std::map<std::string, TargetValue, std::less<>>;
106 using EntryMap = std::map<std::string, ConfigMap, std::less<>>;
107 using TypeMap = std::map<std::string, EntryMap, std::less<>>;
108 using PackageMap = std::map<std::string, TypeMap, std::less<>>;
Jeremy Meyer65871022022-07-12 22:45:08 +0000109 PackageMap package_map;
110 android::StringPool string_pool;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800111 for (const auto& res_entry : entries_) {
112 StringPiece package_substr;
113 StringPiece type_name;
114 StringPiece entry_name;
115 if (!android::ExtractResourceName(StringPiece(res_entry.resource_name), &package_substr,
116 &type_name, &entry_name)) {
117 return Error("failed to parse resource name '%s'", res_entry.resource_name.c_str());
118 }
119
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700120 std::string_view package_name = package_substr.empty() ? target_package_name_ : package_substr;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800121 if (type_name.empty()) {
122 return Error("resource name '%s' missing type name", res_entry.resource_name.c_str());
123 }
124
125 if (entry_name.empty()) {
126 return Error("resource name '%s' missing entry name", res_entry.resource_name.c_str());
127 }
128
Jeremy Meyer65871022022-07-12 22:45:08 +0000129 auto package = package_map.find(package_name);
130 if (package == package_map.end()) {
131 package = package_map
132 .insert(std::make_pair(package_name, TypeMap()))
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800133 .first;
134 }
135
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700136 auto type = package->second.find(type_name);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800137 if (type == package->second.end()) {
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700138 type = package->second.insert(std::make_pair(type_name, EntryMap())).first;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800139 }
140
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700141 auto entry = type->second.find(entry_name);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800142 if (entry == type->second.end()) {
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700143 entry = type->second.insert(std::make_pair(entry_name, ConfigMap())).first;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800144 }
145
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000146 auto value = entry->second.find(res_entry.configuration);
147 if (value == entry->second.end()) {
148 value = entry->second.insert(std::make_pair(res_entry.configuration, TargetValue())).first;
149 }
150
151 value->second = TargetValue{res_entry.data_type, res_entry.data_value,
Jeremy Meyerc04bcb32023-07-27 22:49:14 +0000152 res_entry.data_string_value, res_entry.data_binary_value,
153 res_entry.data_binary_offset, res_entry.data_binary_size};
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800154 }
155
156 pb::FabricatedOverlay overlay_pb;
157 overlay_pb.set_package_name(package_name_);
158 overlay_pb.set_name(name_);
159 overlay_pb.set_target_package_name(target_package_name_);
160 overlay_pb.set_target_overlayable(target_overlayable_);
161
Jeremy Meyerc04bcb32023-07-27 22:49:14 +0000162 std::vector<FabricatedOverlay::BinaryData> binary_files;
Jeremy Meyera761f332022-10-21 17:42:14 +0000163 size_t total_binary_bytes = 0;
164 // 16 for the number of bytes in the frro file before the binary data
165 const size_t FRRO_HEADER_SIZE = 16;
166
Jeremy Meyer65871022022-07-12 22:45:08 +0000167 for (auto& package : package_map) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800168 auto package_pb = overlay_pb.add_packages();
169 package_pb->set_name(package.first);
170
Jeremy Meyer65871022022-07-12 22:45:08 +0000171 for (auto& type : package.second) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800172 auto type_pb = package_pb->add_types();
173 type_pb->set_name(type.first);
174
Jeremy Meyer65871022022-07-12 22:45:08 +0000175 for (auto& entry : type.second) {
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000176 for (const auto& value: entry.second) {
177 auto entry_pb = type_pb->add_entries();
178 entry_pb->set_name(entry.first);
179 entry_pb->set_configuration(value.first);
180 pb::ResourceValue* pb_value = entry_pb->mutable_res_value();
181 pb_value->set_data_type(value.second.data_type);
182 if (value.second.data_type == Res_value::TYPE_STRING) {
183 auto ref = string_pool.MakeRef(value.second.data_string_value);
184 pb_value->set_data_value(ref.index());
Jeremy Meyera761f332022-10-21 17:42:14 +0000185 } else if (value.second.data_binary_value.has_value()) {
186 pb_value->set_data_type(Res_value::TYPE_STRING);
Jeremy Meyera761f332022-10-21 17:42:14 +0000187 std::string uri
188 = StringPrintf("frro:/%s?offset=%d&size=%d", frro_path_.c_str(),
189 static_cast<int> (FRRO_HEADER_SIZE + total_binary_bytes),
Jeremy Meyerc04bcb32023-07-27 22:49:14 +0000190 static_cast<int> (value.second.data_binary_size));
191 total_binary_bytes += value.second.data_binary_size;
192 binary_files.emplace_back(FabricatedOverlay::BinaryData{
193 value.second.data_binary_value->get(),
194 value.second.data_binary_offset,
195 value.second.data_binary_size});
Jeremy Meyera761f332022-10-21 17:42:14 +0000196 auto ref = string_pool.MakeRef(std::move(uri));
197 pb_value->set_data_value(ref.index());
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000198 } else {
199 pb_value->set_data_value(value.second.data_value);
200 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000201 }
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800202 }
203 }
204 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000205 android::BigBuffer string_buffer(kBufferSize);
206 android::StringPool::FlattenUtf8(&string_buffer, string_pool, nullptr);
Jeremy Meyera761f332022-10-21 17:42:14 +0000207 return FabricatedOverlay(std::move(overlay_pb), string_buffer.to_string(),
208 std::move(binary_files), total_binary_bytes);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800209}
210
211Result<FabricatedOverlay> FabricatedOverlay::FromBinaryStream(std::istream& stream) {
212 uint32_t magic;
213 if (!Read32(stream, &magic)) {
214 return Error("Failed to read fabricated overlay magic.");
215 }
216
217 if (magic != kFabricatedOverlayMagic) {
218 return Error("Not a fabricated overlay file.");
219 }
220
221 uint32_t version;
222 if (!Read32(stream, &version)) {
223 return Error("Failed to read fabricated overlay version.");
224 }
225
Jeremy Meyera761f332022-10-21 17:42:14 +0000226 if (version < 1 || version > 3) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800227 return Error("Invalid fabricated overlay version '%u'.", version);
228 }
229
230 uint32_t crc;
231 if (!Read32(stream, &crc)) {
Jeremy Meyer65871022022-07-12 22:45:08 +0000232 return Error("Failed to read fabricated overlay crc.");
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800233 }
234
235 pb::FabricatedOverlay overlay{};
Jeremy Meyer65871022022-07-12 22:45:08 +0000236 std::string sp_data;
Jeremy Meyera761f332022-10-21 17:42:14 +0000237 uint32_t total_binary_bytes;
238 if (version == 3) {
239 if (!Read32(stream, &total_binary_bytes)) {
240 return Error("Failed read total binary bytes.");
241 }
242 stream.seekg(total_binary_bytes, std::istream::cur);
243 }
244 if (version >= 2) {
Jeremy Meyer65871022022-07-12 22:45:08 +0000245 uint32_t sp_size;
246 if (!Read32(stream, &sp_size)) {
247 return Error("Failed read string pool size.");
248 }
249 std::string buf(sp_size, '\0');
250 if (!stream.read(buf.data(), sp_size)) {
251 return Error("Failed to read string pool.");
252 }
253 sp_data = buf;
Jeremy Meyera761f332022-10-21 17:42:14 +0000254 }
255 if (!overlay.ParseFromIstream(&stream)) {
256 return Error("Failed read fabricated overlay proto.");
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800257 }
258
259 // If the proto version is the latest version, then the contents of the proto must be the same
260 // when the proto is re-serialized; otherwise, the crc must be calculated because migrating the
261 // proto to the latest version will likely change the contents of the fabricated overlay.
Jeremy Meyera761f332022-10-21 17:42:14 +0000262 return FabricatedOverlay(std::move(overlay), std::move(sp_data), {}, total_binary_bytes,
Jeremy Meyer65871022022-07-12 22:45:08 +0000263 version == kFabricatedOverlayCurrentVersion
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800264 ? std::optional<uint32_t>(crc)
265 : std::nullopt);
266}
267
268Result<FabricatedOverlay::SerializedData*> FabricatedOverlay::InitializeData() const {
269 if (!data_.has_value()) {
Jeremy Meyer65871022022-07-12 22:45:08 +0000270 auto pb_size = overlay_pb_.ByteSizeLong();
271 auto pb_data = std::unique_ptr<uint8_t[]>(new uint8_t[pb_size]);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800272
273 // Ensure serialization is deterministic
Jeremy Meyer65871022022-07-12 22:45:08 +0000274 google::protobuf::io::ArrayOutputStream array_stream(pb_data.get(), pb_size);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800275 google::protobuf::io::CodedOutputStream output_stream(&array_stream);
276 output_stream.SetSerializationDeterministic(true);
277 overlay_pb_.SerializeWithCachedSizes(&output_stream);
Jeremy Meyer65871022022-07-12 22:45:08 +0000278 if (output_stream.HadError() || pb_size != output_stream.ByteCount()) {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800279 return Error("Failed to serialize fabricated overlay.");
280 }
281
282 // Calculate the crc using the proto data and the version.
Jeremy Meyer65871022022-07-12 22:45:08 +0000283 uint32_t pb_crc = crc32(0L, Z_NULL, 0);
284 pb_crc = crc32(pb_crc, reinterpret_cast<const uint8_t*>(&kFabricatedOverlayCurrentVersion),
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800285 sizeof(uint32_t));
Jeremy Meyer65871022022-07-12 22:45:08 +0000286 pb_crc = crc32(pb_crc, pb_data.get(), pb_size);
287
288 data_ = SerializedData{std::move(pb_data), pb_size, pb_crc, string_pool_data_};
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800289 }
290 return &(*data_);
291}
292Result<uint32_t> FabricatedOverlay::GetCrc() const {
293 if (crc_from_disk_.has_value()) {
294 return *crc_from_disk_;
295 }
296 auto data = InitializeData();
297 if (!data) {
298 return data.GetError();
299 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000300 return (*data)->pb_crc;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800301}
302
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800303Result<Unit> FabricatedOverlay::ToBinaryStream(std::ostream& stream) const {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800304 auto data = InitializeData();
305 if (!data) {
306 return data.GetError();
307 }
308
309 Write32(stream, kFabricatedOverlayMagic);
310 Write32(stream, kFabricatedOverlayCurrentVersion);
Jeremy Meyer65871022022-07-12 22:45:08 +0000311 Write32(stream, (*data)->pb_crc);
Jeremy Meyera761f332022-10-21 17:42:14 +0000312 Write32(stream, total_binary_bytes_);
313 std::string file_contents;
Jeremy Meyerc04bcb32023-07-27 22:49:14 +0000314 for (const FabricatedOverlay::BinaryData fd : binary_files_) {
315 file_contents.resize(fd.size);
316 if (!ReadFullyAtOffset(fd.file_descriptor, file_contents.data(), fd.size, fd.offset)) {
Jeremy Meyera761f332022-10-21 17:42:14 +0000317 return Error("Failed to read binary file data.");
318 }
319 stream.write(file_contents.data(), file_contents.length());
320 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000321 Write32(stream, (*data)->sp_data.length());
322 stream.write((*data)->sp_data.data(), (*data)->sp_data.length());
323 if (stream.bad()) {
324 return Error("Failed to write string pool data.");
325 }
326 stream.write(reinterpret_cast<const char*>((*data)->pb_data.get()), (*data)->pb_data_size);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800327 if (stream.bad()) {
328 return Error("Failed to write serialized fabricated overlay.");
329 }
330
331 return Unit{};
332}
333
334using FabContainer = FabricatedOverlayContainer;
335FabContainer::FabricatedOverlayContainer(FabricatedOverlay&& overlay, std::string&& path)
336 : overlay_(std::forward<FabricatedOverlay>(overlay)), path_(std::forward<std::string>(path)) {
337}
338
339FabContainer::~FabricatedOverlayContainer() = default;
340
341Result<std::unique_ptr<FabContainer>> FabContainer::FromPath(std::string path) {
342 std::fstream fin(path);
343 auto overlay = FabricatedOverlay::FromBinaryStream(fin);
344 if (!overlay) {
345 return overlay.GetError();
346 }
347 return std::unique_ptr<FabContainer>(
348 new FabricatedOverlayContainer(std::move(*overlay), std::move(path)));
349}
350
351std::unique_ptr<FabricatedOverlayContainer> FabContainer::FromOverlay(FabricatedOverlay&& overlay) {
352 return std::unique_ptr<FabContainer>(
353 new FabricatedOverlayContainer(std::move(overlay), {} /* path */));
354}
355
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800356OverlayManifestInfo FabContainer::GetManifestInfo() const {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800357 const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800358 return OverlayManifestInfo{
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800359 .package_name = overlay_pb.package_name(),
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800360 .name = overlay_pb.name(),
361 .target_package = overlay_pb.target_package_name(),
362 .target_name = overlay_pb.target_overlayable(),
363 };
364}
365
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800366Result<OverlayManifestInfo> FabContainer::FindOverlayInfo(const std::string& name) const {
367 const OverlayManifestInfo info = GetManifestInfo();
368 if (name != info.name) {
369 return Error("Failed to find name '%s' in fabricated overlay", name.c_str());
370 }
371 return info;
372}
373
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800374Result<OverlayData> FabContainer::GetOverlayData(const OverlayManifestInfo& info) const {
375 const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_;
376 if (info.name != overlay_pb.name()) {
377 return Error("Failed to find name '%s' in fabricated overlay", info.name.c_str());
378 }
379
380 OverlayData result{};
381 for (const auto& package : overlay_pb.packages()) {
382 for (const auto& type : package.types()) {
383 for (const auto& entry : type.entries()) {
384 auto name = base::StringPrintf("%s:%s/%s", package.name().c_str(), type.name().c_str(),
385 entry.name().c_str());
386 const auto& res_value = entry.res_value();
387 result.pairs.emplace_back(OverlayData::Value{
Jeremy Meyer77c8a932022-08-18 22:06:55 +0000388 name, TargetValueWithConfig{.config = entry.configuration(), .value = TargetValue{
389 .data_type = static_cast<uint8_t>(res_value.data_type()),
390 .data_value = res_value.data_value()}}});
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800391 }
392 }
393 }
Jeremy Meyer65871022022-07-12 22:45:08 +0000394 const uint32_t string_pool_data_length = overlay_.string_pool_data_.length();
395 result.string_pool_data = OverlayData::InlineStringPoolData{
396 .data = std::unique_ptr<uint8_t[]>(new uint8_t[string_pool_data_length]),
397 .data_length = string_pool_data_length,
398 .string_pool_offset = 0,
399 };
400 memcpy(result.string_pool_data->data.get(), overlay_.string_pool_data_.data(),
401 string_pool_data_length);
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800402 return result;
403}
404
405Result<uint32_t> FabContainer::GetCrc() const {
406 return overlay_.GetCrc();
407}
408
409const std::string& FabContainer::GetPath() const {
410 return path_;
411}
412
413Result<std::string> FabContainer::GetResourceName(ResourceId /* id */) const {
414 return Error("Fabricated overlay does not contain resources.");
415}
416
Mårten Kongstad1195a6b2021-05-11 12:57:01 +0000417} // namespace android::idmap2