|  | /* | 
|  | * Copyright (C) 2021 The Android Open Source Project | 
|  | * | 
|  | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | * you may not use this file except in compliance with the License. | 
|  | * You may obtain a copy of the License at | 
|  | * | 
|  | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include "idmap2/FabricatedOverlay.h" | 
|  |  | 
|  | #include <sys/stat.h>   // umask | 
|  | #include <sys/types.h>  // umask | 
|  |  | 
|  | #include <android-base/file.h> | 
|  | #include <android-base/strings.h> | 
|  | #include <androidfw/BigBuffer.h> | 
|  | #include <androidfw/BigBufferStream.h> | 
|  | #include <androidfw/FileStream.h> | 
|  | #include <androidfw/Image.h> | 
|  | #include <androidfw/Png.h> | 
|  | #include <androidfw/ResourceUtils.h> | 
|  | #include <androidfw/StringPiece.h> | 
|  | #include <androidfw/StringPool.h> | 
|  | #include <androidfw/Streams.h> | 
|  | #include <google/protobuf/io/coded_stream.h> | 
|  | #include <google/protobuf/io/zero_copy_stream_impl.h> | 
|  | #include <utils/ByteOrder.h> | 
|  | #include <zlib.h> | 
|  |  | 
|  | #include <fstream> | 
|  | #include <map> | 
|  | #include <memory> | 
|  | #include <string> | 
|  | #include <utility> | 
|  | #include <sys/utsname.h> | 
|  |  | 
|  | namespace android::idmap2 { | 
|  | constexpr auto kBufferSize = 1024; | 
|  |  | 
|  | namespace { | 
|  | bool Read32(std::istream& stream, uint32_t* out) { | 
|  | uint32_t value; | 
|  | if (stream.read(reinterpret_cast<char*>(&value), sizeof(uint32_t))) { | 
|  | *out = dtohl(value); | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void Write32(std::ostream& stream, uint32_t value) { | 
|  | uint32_t x = htodl(value); | 
|  | stream.write(reinterpret_cast<char*>(&x), sizeof(uint32_t)); | 
|  | } | 
|  | }  // namespace | 
|  |  | 
|  | FabricatedOverlay::FabricatedOverlay(pb::FabricatedOverlay&& overlay, | 
|  | std::string&& string_pool_data, | 
|  | std::vector<FabricatedOverlay::BinaryData> binary_files, | 
|  | off_t total_binary_bytes, | 
|  | std::optional<uint32_t> crc_from_disk) | 
|  | : overlay_pb_(std::forward<pb::FabricatedOverlay>(overlay)), | 
|  | string_pool_data_(std::move(string_pool_data)), | 
|  | binary_files_(std::move(binary_files)), | 
|  | total_binary_bytes_(total_binary_bytes), | 
|  | crc_from_disk_(crc_from_disk) { | 
|  | } | 
|  |  | 
|  | FabricatedOverlay::Builder::Builder(const std::string& package_name, const std::string& name, | 
|  | const std::string& target_package_name) { | 
|  | package_name_ = package_name; | 
|  | name_ = name; | 
|  | target_package_name_ = target_package_name; | 
|  | } | 
|  |  | 
|  | FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetOverlayable(const std::string& name) { | 
|  | target_overlayable_ = name; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue( | 
|  | const std::string& resource_name, uint8_t data_type, uint32_t data_value, | 
|  | const std::string& configuration) { | 
|  | entries_.emplace_back( | 
|  | Entry{resource_name, data_type, data_value, "", std::nullopt, 0, 0, configuration, false}); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue( | 
|  | const std::string& resource_name, uint8_t data_type, const std::string& data_string_value, | 
|  | const std::string& configuration) { | 
|  | entries_.emplace_back( | 
|  | Entry{resource_name, | 
|  | data_type, | 
|  | 0, | 
|  | data_string_value, | 
|  | std::nullopt, | 
|  | 0, | 
|  | 0, | 
|  | configuration, | 
|  | false}); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue( | 
|  | const std::string& resource_name, std::optional<android::base::borrowed_fd>&& binary_value, | 
|  | off64_t data_binary_offset, size_t data_binary_size, const std::string& configuration, | 
|  | bool nine_patch) { | 
|  | entries_.emplace_back(Entry{resource_name, 0, 0, "", binary_value, | 
|  | data_binary_offset, data_binary_size, configuration, nine_patch}); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | static Result<FabricatedOverlay::BinaryData> buildBinaryData( | 
|  | pb::ResourceValue* pb_value, const TargetValue &value) { | 
|  | pb_value->set_data_type(Res_value::TYPE_STRING); | 
|  | size_t binary_size; | 
|  | off64_t binary_offset; | 
|  | std::unique_ptr<android::InputStream> binary_stream; | 
|  |  | 
|  | if (value.nine_patch) { | 
|  | std::string file_contents; | 
|  | file_contents.resize(value.data_binary_size); | 
|  | if (!base::ReadFullyAtOffset(value.data_binary_value->get(), file_contents.data(), | 
|  | value.data_binary_size, value.data_binary_offset)) { | 
|  | return Error("Failed to read binary file data."); | 
|  | } | 
|  | const StringPiece content(file_contents.c_str(), file_contents.size()); | 
|  | android::PngChunkFilter png_chunk_filter(content); | 
|  | android::AndroidLogDiagnostics diag; | 
|  | auto png = android::ReadPng(&png_chunk_filter, &diag); | 
|  | if (!png) { | 
|  | return Error("Error opening file as png"); | 
|  | } | 
|  |  | 
|  | std::string err; | 
|  | std::unique_ptr<NinePatch> nine_patch = NinePatch::Create(png->rows.get(), | 
|  | png->width, png->height, | 
|  | &err); | 
|  | if (!nine_patch) { | 
|  | return Error("%s", err.c_str()); | 
|  | } | 
|  |  | 
|  | png->width -= 2; | 
|  | png->height -= 2; | 
|  | memmove(png->rows.get(), png->rows.get() + 1, png->height * sizeof(uint8_t**)); | 
|  | for (int32_t h = 0; h < png->height; h++) { | 
|  | memmove(png->rows[h], png->rows[h] + 4, png->width * 4); | 
|  | } | 
|  |  | 
|  | android::BigBuffer buffer(value.data_binary_size); | 
|  | android::BigBufferOutputStream buffer_output_stream(&buffer); | 
|  | if (!android::WritePng(png.get(), nine_patch.get(), &buffer_output_stream, {}, | 
|  | &diag, false)) { | 
|  | return Error("Error writing frro png"); | 
|  | } | 
|  |  | 
|  | binary_size = buffer.size(); | 
|  | binary_offset = 0; | 
|  | android::BigBufferInputStream *buffer_input_stream | 
|  | = new android::BigBufferInputStream(std::move(buffer)); | 
|  | binary_stream.reset(buffer_input_stream); | 
|  | } else { | 
|  | binary_size = value.data_binary_size; | 
|  | binary_offset = value.data_binary_offset; | 
|  | android::FileInputStream *fis | 
|  | = new android::FileInputStream(value.data_binary_value.value()); | 
|  | binary_stream.reset(fis); | 
|  | } | 
|  |  | 
|  | return FabricatedOverlay::BinaryData{ | 
|  | std::move(binary_stream), | 
|  | binary_offset, | 
|  | binary_size}; | 
|  | } | 
|  |  | 
|  | Result<FabricatedOverlay> FabricatedOverlay::Builder::Build() { | 
|  | using ConfigMap = std::map<std::string, TargetValue, std::less<>>; | 
|  | using EntryMap = std::map<std::string, ConfigMap, std::less<>>; | 
|  | using TypeMap = std::map<std::string, EntryMap, std::less<>>; | 
|  | using PackageMap = std::map<std::string, TypeMap, std::less<>>; | 
|  | PackageMap package_map; | 
|  | android::StringPool string_pool; | 
|  | for (const auto& res_entry : entries_) { | 
|  | StringPiece package_substr; | 
|  | StringPiece type_name; | 
|  | StringPiece entry_name; | 
|  | if (!android::ExtractResourceName(StringPiece(res_entry.resource_name), &package_substr, | 
|  | &type_name, &entry_name)) { | 
|  | return Error("failed to parse resource name '%s'", res_entry.resource_name.c_str()); | 
|  | } | 
|  |  | 
|  | std::string_view package_name = package_substr.empty() ? target_package_name_ : package_substr; | 
|  | if (type_name.empty()) { | 
|  | return Error("resource name '%s' missing type name", res_entry.resource_name.c_str()); | 
|  | } | 
|  |  | 
|  | if (entry_name.empty()) { | 
|  | return Error("resource name '%s' missing entry name", res_entry.resource_name.c_str()); | 
|  | } | 
|  |  | 
|  | auto package = package_map.find(package_name); | 
|  | if (package == package_map.end()) { | 
|  | package = package_map | 
|  | .insert(std::make_pair(package_name, TypeMap())) | 
|  | .first; | 
|  | } | 
|  |  | 
|  | auto type = package->second.find(type_name); | 
|  | if (type == package->second.end()) { | 
|  | type = package->second.insert(std::make_pair(type_name, EntryMap())).first; | 
|  | } | 
|  |  | 
|  | auto entry = type->second.find(entry_name); | 
|  | if (entry == type->second.end()) { | 
|  | entry = type->second.insert(std::make_pair(entry_name, ConfigMap())).first; | 
|  | } | 
|  |  | 
|  | auto value = entry->second.find(res_entry.configuration); | 
|  | if (value == entry->second.end()) { | 
|  | value = entry->second.insert(std::make_pair(res_entry.configuration, TargetValue())).first; | 
|  | } | 
|  |  | 
|  | value->second = TargetValue{res_entry.data_type, res_entry.data_value, | 
|  | res_entry.data_string_value, res_entry.data_binary_value, | 
|  | res_entry.data_binary_offset, res_entry.data_binary_size, | 
|  | res_entry.nine_patch}; | 
|  | } | 
|  |  | 
|  | pb::FabricatedOverlay overlay_pb; | 
|  | overlay_pb.set_package_name(package_name_); | 
|  | overlay_pb.set_name(name_); | 
|  | overlay_pb.set_target_package_name(target_package_name_); | 
|  | overlay_pb.set_target_overlayable(target_overlayable_); | 
|  |  | 
|  | std::vector<FabricatedOverlay::BinaryData> binary_files; | 
|  | size_t total_binary_bytes = 0; | 
|  | // 16 for the number of bytes in the frro file before the binary data | 
|  | const size_t FRRO_HEADER_SIZE = 16; | 
|  |  | 
|  | for (auto& package : package_map) { | 
|  | auto package_pb = overlay_pb.add_packages(); | 
|  | package_pb->set_name(package.first); | 
|  |  | 
|  | for (auto& type : package.second) { | 
|  | auto type_pb = package_pb->add_types(); | 
|  | type_pb->set_name(type.first); | 
|  |  | 
|  | for (auto& entry : type.second) { | 
|  | for (const auto& value: entry.second) { | 
|  | auto entry_pb = type_pb->add_entries(); | 
|  | entry_pb->set_name(entry.first); | 
|  | entry_pb->set_configuration(value.first); | 
|  | pb::ResourceValue* pb_value = entry_pb->mutable_res_value(); | 
|  | pb_value->set_data_type(value.second.data_type); | 
|  | if (value.second.data_type == Res_value::TYPE_STRING) { | 
|  | auto ref = string_pool.MakeRef(value.second.data_string_value); | 
|  | pb_value->set_data_value(ref.index()); | 
|  | } else if (value.second.data_binary_value.has_value()) { | 
|  | auto binary_data = buildBinaryData(pb_value, value.second); | 
|  | if (!binary_data) { | 
|  | return binary_data.GetError(); | 
|  | } | 
|  | pb_value->set_data_type(Res_value::TYPE_STRING); | 
|  |  | 
|  | std::string uri | 
|  | = StringPrintf("frro:/%s?offset=%d&size=%d", frro_path_.c_str(), | 
|  | static_cast<int> (FRRO_HEADER_SIZE + total_binary_bytes), | 
|  | static_cast<int> (binary_data->size)); | 
|  | total_binary_bytes += binary_data->size; | 
|  | binary_files.emplace_back(std::move(*binary_data)); | 
|  | auto ref = string_pool.MakeRef(std::move(uri)); | 
|  | pb_value->set_data_value(ref.index()); | 
|  | } else { | 
|  | pb_value->set_data_value(value.second.data_value); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | android::BigBuffer string_buffer(kBufferSize); | 
|  | android::StringPool::FlattenUtf8(&string_buffer, string_pool, nullptr); | 
|  | return FabricatedOverlay(std::move(overlay_pb), string_buffer.to_string(), | 
|  | std::move(binary_files), total_binary_bytes); | 
|  | } | 
|  |  | 
|  | Result<FabricatedOverlay> FabricatedOverlay::FromBinaryStream(std::istream& stream) { | 
|  | uint32_t magic; | 
|  | if (!Read32(stream, &magic)) { | 
|  | return Error("Failed to read fabricated overlay magic."); | 
|  | } | 
|  |  | 
|  | if (magic != kFabricatedOverlayMagic) { | 
|  | return Error("Not a fabricated overlay file."); | 
|  | } | 
|  |  | 
|  | uint32_t version; | 
|  | if (!Read32(stream, &version)) { | 
|  | return Error("Failed to read fabricated overlay version."); | 
|  | } | 
|  |  | 
|  | if (version < 1 || version > 3) { | 
|  | return Error("Invalid fabricated overlay version '%u'.", version); | 
|  | } | 
|  |  | 
|  | uint32_t crc; | 
|  | if (!Read32(stream, &crc)) { | 
|  | return Error("Failed to read fabricated overlay crc."); | 
|  | } | 
|  |  | 
|  | pb::FabricatedOverlay overlay{}; | 
|  | std::string sp_data; | 
|  | uint32_t total_binary_bytes; | 
|  | if (version == 3) { | 
|  | if (!Read32(stream, &total_binary_bytes)) { | 
|  | return Error("Failed read total binary bytes."); | 
|  | } | 
|  | stream.seekg(total_binary_bytes, std::istream::cur); | 
|  | } | 
|  | if (version >= 2) { | 
|  | uint32_t sp_size; | 
|  | if (!Read32(stream, &sp_size)) { | 
|  | return Error("Failed read string pool size."); | 
|  | } | 
|  | std::string buf(sp_size, '\0'); | 
|  | if (!stream.read(buf.data(), sp_size)) { | 
|  | return Error("Failed to read string pool."); | 
|  | } | 
|  | sp_data = buf; | 
|  | } | 
|  | if (!overlay.ParseFromIstream(&stream)) { | 
|  | return Error("Failed read fabricated overlay proto."); | 
|  | } | 
|  |  | 
|  | // If the proto version is the latest version, then the contents of the proto must be the same | 
|  | // when the proto is re-serialized; otherwise, the crc must be calculated because migrating the | 
|  | // proto to the latest version will likely change the contents of the fabricated overlay. | 
|  | return FabricatedOverlay(std::move(overlay), std::move(sp_data), {}, total_binary_bytes, | 
|  | version == kFabricatedOverlayCurrentVersion | 
|  | ? std::optional<uint32_t>(crc) | 
|  | : std::nullopt); | 
|  | } | 
|  |  | 
|  | Result<FabricatedOverlay::SerializedData*> FabricatedOverlay::InitializeData() const { | 
|  | if (!data_.has_value()) { | 
|  | auto pb_size = overlay_pb_.ByteSizeLong(); | 
|  | auto pb_data = std::unique_ptr<uint8_t[]>(new uint8_t[pb_size]); | 
|  |  | 
|  | // Ensure serialization is deterministic | 
|  | google::protobuf::io::ArrayOutputStream array_stream(pb_data.get(), pb_size); | 
|  | google::protobuf::io::CodedOutputStream output_stream(&array_stream); | 
|  | output_stream.SetSerializationDeterministic(true); | 
|  | overlay_pb_.SerializeWithCachedSizes(&output_stream); | 
|  | if (output_stream.HadError() || pb_size != output_stream.ByteCount()) { | 
|  | return Error("Failed to serialize fabricated overlay."); | 
|  | } | 
|  |  | 
|  | // Calculate the crc using the proto data and the version. | 
|  | uint32_t pb_crc = crc32(0L, Z_NULL, 0); | 
|  | pb_crc = crc32(pb_crc, reinterpret_cast<const uint8_t*>(&kFabricatedOverlayCurrentVersion), | 
|  | sizeof(uint32_t)); | 
|  | pb_crc = crc32(pb_crc, pb_data.get(), pb_size); | 
|  |  | 
|  | data_ = SerializedData{std::move(pb_data), pb_size, pb_crc, string_pool_data_}; | 
|  | } | 
|  | return &(*data_); | 
|  | } | 
|  | Result<uint32_t> FabricatedOverlay::GetCrc() const { | 
|  | if (crc_from_disk_.has_value()) { | 
|  | return *crc_from_disk_; | 
|  | } | 
|  | auto data = InitializeData(); | 
|  | if (!data) { | 
|  | return data.GetError(); | 
|  | } | 
|  | return (*data)->pb_crc; | 
|  | } | 
|  |  | 
|  | Result<Unit> FabricatedOverlay::ToBinaryStream(std::ostream& stream) const { | 
|  | auto data = InitializeData(); | 
|  | if (!data) { | 
|  | return data.GetError(); | 
|  | } | 
|  |  | 
|  | Write32(stream, kFabricatedOverlayMagic); | 
|  | Write32(stream, kFabricatedOverlayCurrentVersion); | 
|  | Write32(stream, (*data)->pb_crc); | 
|  | Write32(stream, total_binary_bytes_); | 
|  | std::string file_contents; | 
|  | for (const FabricatedOverlay::BinaryData& bd : binary_files_) { | 
|  | file_contents.resize(bd.size); | 
|  | if (!bd.input_stream->ReadFullyAtOffset(file_contents.data(), bd.size, bd.offset)) { | 
|  | return Error("Failed to read binary file data."); | 
|  | } | 
|  | stream.write(file_contents.data(), file_contents.length()); | 
|  | } | 
|  | Write32(stream, (*data)->sp_data.length()); | 
|  | stream.write((*data)->sp_data.data(), (*data)->sp_data.length()); | 
|  | if (stream.bad()) { | 
|  | return Error("Failed to write string pool data."); | 
|  | } | 
|  | stream.write(reinterpret_cast<const char*>((*data)->pb_data.get()), (*data)->pb_data_size); | 
|  | if (stream.bad()) { | 
|  | return Error("Failed to write serialized fabricated overlay."); | 
|  | } | 
|  |  | 
|  | return Unit{}; | 
|  | } | 
|  |  | 
|  | using FabContainer = FabricatedOverlayContainer; | 
|  | FabContainer::FabricatedOverlayContainer(FabricatedOverlay&& overlay, std::string&& path) | 
|  | : overlay_(std::forward<FabricatedOverlay>(overlay)), path_(std::forward<std::string>(path)) { | 
|  | } | 
|  |  | 
|  | FabContainer::~FabricatedOverlayContainer() = default; | 
|  |  | 
|  | Result<std::unique_ptr<FabContainer>> FabContainer::FromPath(std::string path) { | 
|  | std::fstream fin(path); | 
|  | auto overlay = FabricatedOverlay::FromBinaryStream(fin); | 
|  | if (!overlay) { | 
|  | return overlay.GetError(); | 
|  | } | 
|  | return std::unique_ptr<FabContainer>( | 
|  | new FabricatedOverlayContainer(std::move(*overlay), std::move(path))); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<FabricatedOverlayContainer> FabContainer::FromOverlay(FabricatedOverlay&& overlay) { | 
|  | return std::unique_ptr<FabContainer>( | 
|  | new FabricatedOverlayContainer(std::move(overlay), {} /* path */)); | 
|  | } | 
|  |  | 
|  | OverlayManifestInfo FabContainer::GetManifestInfo() const { | 
|  | const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_; | 
|  | return OverlayManifestInfo{ | 
|  | .package_name = overlay_pb.package_name(), | 
|  | .name = overlay_pb.name(), | 
|  | .target_package = overlay_pb.target_package_name(), | 
|  | .target_name = overlay_pb.target_overlayable(), | 
|  | }; | 
|  | } | 
|  |  | 
|  | Result<OverlayManifestInfo> FabContainer::FindOverlayInfo(const std::string& name) const { | 
|  | const OverlayManifestInfo info = GetManifestInfo(); | 
|  | if (name != info.name) { | 
|  | return Error("Failed to find name '%s' in fabricated overlay", name.c_str()); | 
|  | } | 
|  | return info; | 
|  | } | 
|  |  | 
|  | Result<OverlayData> FabContainer::GetOverlayData(const OverlayManifestInfo& info) const { | 
|  | const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_; | 
|  | if (info.name != overlay_pb.name()) { | 
|  | return Error("Failed to find name '%s' in fabricated overlay", info.name.c_str()); | 
|  | } | 
|  |  | 
|  | OverlayData result{}; | 
|  | for (const auto& package : overlay_pb.packages()) { | 
|  | for (const auto& type : package.types()) { | 
|  | for (const auto& entry : type.entries()) { | 
|  | auto name = base::StringPrintf("%s:%s/%s", package.name().c_str(), type.name().c_str(), | 
|  | entry.name().c_str()); | 
|  | const auto& res_value = entry.res_value(); | 
|  | result.pairs.emplace_back(OverlayData::Value{ | 
|  | name, TargetValueWithConfig{.config = entry.configuration(), .value = TargetValue{ | 
|  | .data_type = static_cast<uint8_t>(res_value.data_type()), | 
|  | .data_value = res_value.data_value()}}}); | 
|  | } | 
|  | } | 
|  | } | 
|  | const uint32_t string_pool_data_length = overlay_.string_pool_data_.length(); | 
|  | result.string_pool_data = OverlayData::InlineStringPoolData{ | 
|  | .data = std::unique_ptr<uint8_t[]>(new uint8_t[string_pool_data_length]), | 
|  | .data_length = string_pool_data_length, | 
|  | .string_pool_offset = 0, | 
|  | }; | 
|  | memcpy(result.string_pool_data->data.get(), overlay_.string_pool_data_.data(), | 
|  | string_pool_data_length); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | Result<uint32_t> FabContainer::GetCrc() const { | 
|  | return overlay_.GetCrc(); | 
|  | } | 
|  |  | 
|  | const std::string& FabContainer::GetPath() const { | 
|  | return path_; | 
|  | } | 
|  |  | 
|  | Result<std::string> FabContainer::GetResourceName(ResourceId /* id */) const { | 
|  | return Error("Fabricated overlay does not contain resources."); | 
|  | } | 
|  |  | 
|  | }  // namespace android::idmap2 |