blob: 4f61801aa594de711e5e6a082387cab1187251ac [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>
20#include <google/protobuf/io/coded_stream.h>
21#include <google/protobuf/io/zero_copy_stream_impl_lite.h>
22#include <utils/ByteOrder.h>
23#include <zlib.h>
24
25#include <fstream>
26
27namespace android::idmap2 {
28
29namespace {
30bool Read32(std::istream& stream, uint32_t* out) {
31 uint32_t value;
32 if (stream.read(reinterpret_cast<char*>(&value), sizeof(uint32_t))) {
33 *out = dtohl(value);
34 return true;
35 }
36 return false;
37}
38
39void Write32(std::ostream& stream, uint32_t value) {
40 uint32_t x = htodl(value);
41 stream.write(reinterpret_cast<char*>(&x), sizeof(uint32_t));
42}
43} // namespace
44
45FabricatedOverlay::FabricatedOverlay(pb::FabricatedOverlay&& overlay,
46 std::optional<uint32_t> crc_from_disk)
47 : overlay_pb_(std::forward<pb::FabricatedOverlay>(overlay)), crc_from_disk_(crc_from_disk) {
48}
49
50FabricatedOverlay::Builder::Builder(const std::string& package_name, const std::string& name,
51 const std::string& target_package_name) {
52 package_name_ = package_name;
53 name_ = name;
54 target_package_name_ = target_package_name;
55}
56
57FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetOverlayable(const std::string& name) {
58 target_overlayable_ = name;
59 return *this;
60}
61
62FabricatedOverlay::Builder& FabricatedOverlay::Builder::SetResourceValue(
63 const std::string& resource_name, uint8_t data_type, uint32_t data_value) {
64 entries_.emplace_back(Entry{resource_name, data_type, data_value});
65 return *this;
66}
67
68Result<FabricatedOverlay> FabricatedOverlay::Builder::Build() {
69 std::map<std::string, std::map<std::string, std::map<std::string, TargetValue>>> entries;
70 for (const auto& res_entry : entries_) {
71 StringPiece package_substr;
72 StringPiece type_name;
73 StringPiece entry_name;
74 if (!android::ExtractResourceName(StringPiece(res_entry.resource_name), &package_substr,
75 &type_name, &entry_name)) {
76 return Error("failed to parse resource name '%s'", res_entry.resource_name.c_str());
77 }
78
79 std::string package_name =
80 package_substr.empty() ? target_package_name_ : package_substr.to_string();
81 if (type_name.empty()) {
82 return Error("resource name '%s' missing type name", res_entry.resource_name.c_str());
83 }
84
85 if (entry_name.empty()) {
86 return Error("resource name '%s' missing entry name", res_entry.resource_name.c_str());
87 }
88
89 auto package = entries.find(package_name);
90 if (package == entries.end()) {
91 package = entries
92 .insert(std::make_pair<>(
93 package_name, std::map<std::string, std::map<std::string, TargetValue>>()))
94 .first;
95 }
96
97 auto type = package->second.find(type_name.to_string());
98 if (type == package->second.end()) {
99 type =
100 package->second
101 .insert(std::make_pair<>(type_name.to_string(), std::map<std::string, TargetValue>()))
102 .first;
103 }
104
105 auto entry = type->second.find(entry_name.to_string());
106 if (entry == type->second.end()) {
107 entry = type->second.insert(std::make_pair<>(entry_name.to_string(), TargetValue())).first;
108 }
109
110 entry->second = TargetValue{res_entry.data_type, res_entry.data_value};
111 }
112
113 pb::FabricatedOverlay overlay_pb;
114 overlay_pb.set_package_name(package_name_);
115 overlay_pb.set_name(name_);
116 overlay_pb.set_target_package_name(target_package_name_);
117 overlay_pb.set_target_overlayable(target_overlayable_);
118
119 for (const auto& package : entries) {
120 auto package_pb = overlay_pb.add_packages();
121 package_pb->set_name(package.first);
122
123 for (const auto& type : package.second) {
124 auto type_pb = package_pb->add_types();
125 type_pb->set_name(type.first);
126
127 for (const auto& entry : type.second) {
128 auto entry_pb = type_pb->add_entries();
129 entry_pb->set_name(entry.first);
130 pb::ResourceValue* value = entry_pb->mutable_res_value();
131 value->set_data_type(entry.second.data_type);
132 value->set_data_value(entry.second.data_value);
133 }
134 }
135 }
136
137 return FabricatedOverlay(std::move(overlay_pb));
138}
139
140Result<FabricatedOverlay> FabricatedOverlay::FromBinaryStream(std::istream& stream) {
141 uint32_t magic;
142 if (!Read32(stream, &magic)) {
143 return Error("Failed to read fabricated overlay magic.");
144 }
145
146 if (magic != kFabricatedOverlayMagic) {
147 return Error("Not a fabricated overlay file.");
148 }
149
150 uint32_t version;
151 if (!Read32(stream, &version)) {
152 return Error("Failed to read fabricated overlay version.");
153 }
154
155 if (version != 1) {
156 return Error("Invalid fabricated overlay version '%u'.", version);
157 }
158
159 uint32_t crc;
160 if (!Read32(stream, &crc)) {
161 return Error("Failed to read fabricated overlay version.");
162 }
163
164 pb::FabricatedOverlay overlay{};
165 if (!overlay.ParseFromIstream(&stream)) {
166 return Error("Failed read fabricated overlay proto.");
167 }
168
169 // If the proto version is the latest version, then the contents of the proto must be the same
170 // when the proto is re-serialized; otherwise, the crc must be calculated because migrating the
171 // proto to the latest version will likely change the contents of the fabricated overlay.
172 return FabricatedOverlay(std::move(overlay), version == kFabricatedOverlayCurrentVersion
173 ? std::optional<uint32_t>(crc)
174 : std::nullopt);
175}
176
177Result<FabricatedOverlay::SerializedData*> FabricatedOverlay::InitializeData() const {
178 if (!data_.has_value()) {
179 auto size = overlay_pb_.ByteSizeLong();
180 auto data = std::unique_ptr<uint8_t[]>(new uint8_t[size]);
181
182 // Ensure serialization is deterministic
183 google::protobuf::io::ArrayOutputStream array_stream(data.get(), size);
184 google::protobuf::io::CodedOutputStream output_stream(&array_stream);
185 output_stream.SetSerializationDeterministic(true);
186 overlay_pb_.SerializeWithCachedSizes(&output_stream);
187 if (output_stream.HadError() || size != output_stream.ByteCount()) {
188 return Error("Failed to serialize fabricated overlay.");
189 }
190
191 // Calculate the crc using the proto data and the version.
192 uint32_t crc = crc32(0L, Z_NULL, 0);
193 crc = crc32(crc, reinterpret_cast<const uint8_t*>(&kFabricatedOverlayCurrentVersion),
194 sizeof(uint32_t));
195 crc = crc32(crc, data.get(), size);
196 data_ = SerializedData{std::move(data), size, crc};
197 }
198 return &(*data_);
199}
200Result<uint32_t> FabricatedOverlay::GetCrc() const {
201 if (crc_from_disk_.has_value()) {
202 return *crc_from_disk_;
203 }
204 auto data = InitializeData();
205 if (!data) {
206 return data.GetError();
207 }
208 return (*data)->crc;
209}
210
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800211Result<Unit> FabricatedOverlay::ToBinaryStream(std::ostream& stream) const {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800212 auto data = InitializeData();
213 if (!data) {
214 return data.GetError();
215 }
216
217 Write32(stream, kFabricatedOverlayMagic);
218 Write32(stream, kFabricatedOverlayCurrentVersion);
219 Write32(stream, (*data)->crc);
220 stream.write(reinterpret_cast<const char*>((*data)->data.get()), (*data)->data_size);
221 if (stream.bad()) {
222 return Error("Failed to write serialized fabricated overlay.");
223 }
224
225 return Unit{};
226}
227
228using FabContainer = FabricatedOverlayContainer;
229FabContainer::FabricatedOverlayContainer(FabricatedOverlay&& overlay, std::string&& path)
230 : overlay_(std::forward<FabricatedOverlay>(overlay)), path_(std::forward<std::string>(path)) {
231}
232
233FabContainer::~FabricatedOverlayContainer() = default;
234
235Result<std::unique_ptr<FabContainer>> FabContainer::FromPath(std::string path) {
236 std::fstream fin(path);
237 auto overlay = FabricatedOverlay::FromBinaryStream(fin);
238 if (!overlay) {
239 return overlay.GetError();
240 }
241 return std::unique_ptr<FabContainer>(
242 new FabricatedOverlayContainer(std::move(*overlay), std::move(path)));
243}
244
245std::unique_ptr<FabricatedOverlayContainer> FabContainer::FromOverlay(FabricatedOverlay&& overlay) {
246 return std::unique_ptr<FabContainer>(
247 new FabricatedOverlayContainer(std::move(overlay), {} /* path */));
248}
249
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800250OverlayManifestInfo FabContainer::GetManifestInfo() const {
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800251 const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_;
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800252 return OverlayManifestInfo{
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800253 .package_name = overlay_pb.package_name(),
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800254 .name = overlay_pb.name(),
255 .target_package = overlay_pb.target_package_name(),
256 .target_name = overlay_pb.target_overlayable(),
257 };
258}
259
Ryan Mitchell6a2ca782021-01-19 13:51:15 -0800260Result<OverlayManifestInfo> FabContainer::FindOverlayInfo(const std::string& name) const {
261 const OverlayManifestInfo info = GetManifestInfo();
262 if (name != info.name) {
263 return Error("Failed to find name '%s' in fabricated overlay", name.c_str());
264 }
265 return info;
266}
267
Ryan Mitchell2ed8bfa2021-01-08 13:34:28 -0800268Result<OverlayData> FabContainer::GetOverlayData(const OverlayManifestInfo& info) const {
269 const pb::FabricatedOverlay& overlay_pb = overlay_.overlay_pb_;
270 if (info.name != overlay_pb.name()) {
271 return Error("Failed to find name '%s' in fabricated overlay", info.name.c_str());
272 }
273
274 OverlayData result{};
275 for (const auto& package : overlay_pb.packages()) {
276 for (const auto& type : package.types()) {
277 for (const auto& entry : type.entries()) {
278 auto name = base::StringPrintf("%s:%s/%s", package.name().c_str(), type.name().c_str(),
279 entry.name().c_str());
280 const auto& res_value = entry.res_value();
281 result.pairs.emplace_back(OverlayData::Value{
282 name, TargetValue{.data_type = static_cast<uint8_t>(res_value.data_type()),
283 .data_value = res_value.data_value()}});
284 }
285 }
286 }
287 return result;
288}
289
290Result<uint32_t> FabContainer::GetCrc() const {
291 return overlay_.GetCrc();
292}
293
294const std::string& FabContainer::GetPath() const {
295 return path_;
296}
297
298Result<std::string> FabContainer::GetResourceName(ResourceId /* id */) const {
299 return Error("Fabricated overlay does not contain resources.");
300}
301
302} // namespace android::idmap2