blob: a08df71eae1eaf5c41ad17d72598638db0beed84 [file] [log] [blame]
Adam Lesinski59e04c62016-02-04 15:59:23 -08001/*
2 * Copyright (C) 2016 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 "Resource.h"
18#include "ResourceTable.h"
19#include "StringPool.h"
20#include "ValueVisitor.h"
21#include "proto/ProtoHelpers.h"
22#include "proto/ProtoSerialize.h"
23#include "util/BigBuffer.h"
24
Adam Lesinskid5083f62017-01-16 15:07:21 -080025#include "android-base/logging.h"
Adam Lesinskice5e56e2016-10-21 17:56:45 -070026
Adam Lesinski08535002017-08-04 16:15:17 -070027using ::google::protobuf::io::CodedInputStream;
28using ::google::protobuf::io::CodedOutputStream;
29using ::google::protobuf::io::ZeroCopyOutputStream;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -070030
Adam Lesinski59e04c62016-02-04 15:59:23 -080031namespace aapt {
32
33namespace {
34
35class PbSerializerVisitor : public RawValueVisitor {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070036 public:
37 using RawValueVisitor::Visit;
Adam Lesinski59e04c62016-02-04 15:59:23 -080038
Adam Lesinski08535002017-08-04 16:15:17 -070039 // Constructor to use when expecting to serialize any value.
40 PbSerializerVisitor(StringPool* source_pool, pb::Value* out_pb_value)
41 : source_pool_(source_pool), out_pb_value_(out_pb_value), out_pb_item_(nullptr) {
42 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070043
Adam Lesinski08535002017-08-04 16:15:17 -070044 // Constructor to use when expecting to serialize an Item.
45 PbSerializerVisitor(StringPool* sourcePool, pb::Item* outPbItem)
46 : source_pool_(sourcePool), out_pb_value_(nullptr), out_pb_item_(outPbItem) {
47 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070048
49 void Visit(Reference* ref) override {
50 SerializeReferenceToPb(*ref, pb_item()->mutable_ref());
51 }
52
53 void Visit(String* str) override {
Adam Lesinski08535002017-08-04 16:15:17 -070054 pb_item()->mutable_str()->set_value(*str->value);
55 }
56
57 void Visit(RawString* str) override {
58 pb_item()->mutable_raw_str()->set_value(*str->value);
Adam Lesinskice5e56e2016-10-21 17:56:45 -070059 }
60
61 void Visit(StyledString* str) override {
Adam Lesinski08535002017-08-04 16:15:17 -070062 pb::StyledString* pb_str = pb_item()->mutable_styled_str();
63 pb_str->set_value(str->value->value);
64
65 for (const StringPool::Span& span : str->value->spans) {
66 pb::StyledString::Span* pb_span = pb_str->add_span();
67 pb_span->set_tag(*span.name);
68 pb_span->set_first_char(span.first_char);
69 pb_span->set_last_char(span.last_char);
70 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070071 }
72
73 void Visit(FileReference* file) override {
Adam Lesinski08535002017-08-04 16:15:17 -070074 pb_item()->mutable_file()->set_path(*file->path);
Adam Lesinskice5e56e2016-10-21 17:56:45 -070075 }
76
Adam Lesinski08535002017-08-04 16:15:17 -070077 void Visit(Id* /*id*/) override {
78 pb_item()->mutable_id();
Adam Lesinskice5e56e2016-10-21 17:56:45 -070079 }
80
81 void Visit(BinaryPrimitive* prim) override {
82 android::Res_value val = {};
83 prim->Flatten(&val);
84
85 pb::Primitive* pb_prim = pb_item()->mutable_prim();
86 pb_prim->set_type(val.dataType);
87 pb_prim->set_data(val.data);
88 }
89
Adam Lesinski060b53d2017-07-28 17:10:35 -070090 void VisitItem(Item* item) override {
91 LOG(FATAL) << "unimplemented item";
92 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070093
94 void Visit(Attribute* attr) override {
95 pb::Attribute* pb_attr = pb_compound_value()->mutable_attr();
96 pb_attr->set_format_flags(attr->type_mask);
97 pb_attr->set_min_int(attr->min_int);
98 pb_attr->set_max_int(attr->max_int);
99
100 for (auto& symbol : attr->symbols) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700101 pb::Attribute_Symbol* pb_symbol = pb_attr->add_symbol();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700102 SerializeItemCommonToPb(symbol.symbol, pb_symbol);
103 SerializeReferenceToPb(symbol.symbol, pb_symbol->mutable_name());
104 pb_symbol->set_value(symbol.value);
105 }
106 }
107
108 void Visit(Style* style) override {
109 pb::Style* pb_style = pb_compound_value()->mutable_style();
110 if (style->parent) {
111 SerializeReferenceToPb(style->parent.value(), pb_style->mutable_parent());
112 SerializeSourceToPb(style->parent.value().GetSource(), source_pool_,
113 pb_style->mutable_parent_source());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800114 }
115
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700116 for (Style::Entry& entry : style->entries) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700117 pb::Style_Entry* pb_entry = pb_style->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700118 SerializeReferenceToPb(entry.key, pb_entry->mutable_key());
119
120 pb::Item* pb_item = pb_entry->mutable_item();
121 SerializeItemCommonToPb(entry.key, pb_entry);
Adam Lesinski08535002017-08-04 16:15:17 -0700122 PbSerializerVisitor sub_visitor(source_pool_, pb_item);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700123 entry.value->Accept(&sub_visitor);
124 }
125 }
126
127 void Visit(Styleable* styleable) override {
128 pb::Styleable* pb_styleable = pb_compound_value()->mutable_styleable();
129 for (Reference& entry : styleable->entries) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700130 pb::Styleable_Entry* pb_entry = pb_styleable->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700131 SerializeItemCommonToPb(entry, pb_entry);
132 SerializeReferenceToPb(entry, pb_entry->mutable_attr());
133 }
134 }
135
136 void Visit(Array* array) override {
137 pb::Array* pb_array = pb_compound_value()->mutable_array();
Adam Lesinski4ffea042017-08-10 15:37:28 -0700138 for (auto& value : array->elements) {
139 pb::Array_Element* pb_element = pb_array->add_element();
140 SerializeItemCommonToPb(*value, pb_element);
141 PbSerializerVisitor sub_visitor(source_pool_, pb_element->mutable_item());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700142 value->Accept(&sub_visitor);
143 }
144 }
145
146 void Visit(Plural* plural) override {
147 pb::Plural* pb_plural = pb_compound_value()->mutable_plural();
148 const size_t count = plural->values.size();
149 for (size_t i = 0; i < count; i++) {
150 if (!plural->values[i]) {
151 // No plural value set here.
152 continue;
153 }
154
Adam Lesinski4ffea042017-08-10 15:37:28 -0700155 pb::Plural_Entry* pb_entry = pb_plural->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700156 pb_entry->set_arity(SerializePluralEnumToPb(i));
157 pb::Item* pb_element = pb_entry->mutable_item();
158 SerializeItemCommonToPb(*plural->values[i], pb_entry);
Adam Lesinski08535002017-08-04 16:15:17 -0700159 PbSerializerVisitor sub_visitor(source_pool_, pb_element);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700160 plural->values[i]->Accept(&sub_visitor);
161 }
162 }
163
164 private:
165 DISALLOW_COPY_AND_ASSIGN(PbSerializerVisitor);
166
167 pb::Item* pb_item() {
168 if (out_pb_value_) {
169 return out_pb_value_->mutable_item();
170 }
171 return out_pb_item_;
172 }
173
174 pb::CompoundValue* pb_compound_value() {
175 CHECK(out_pb_value_ != nullptr);
176 return out_pb_value_->mutable_compound_value();
177 }
178
179 template <typename T>
180 void SerializeItemCommonToPb(const Item& item, T* pb_item) {
Adam Lesinski08535002017-08-04 16:15:17 -0700181 SerializeSourceToPb(item.GetSource(), source_pool_, pb_item->mutable_source());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700182 if (!item.GetComment().empty()) {
183 pb_item->set_comment(item.GetComment());
184 }
185 }
186
187 void SerializeReferenceToPb(const Reference& ref, pb::Reference* pb_ref) {
188 if (ref.id) {
189 pb_ref->set_id(ref.id.value().id);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800190 }
191
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700192 if (ref.name) {
Adam Lesinski08535002017-08-04 16:15:17 -0700193 pb_ref->set_name(ref.name.value().ToString());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800194 }
195
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700196 pb_ref->set_private_(ref.private_reference);
197 pb_ref->set_type(SerializeReferenceTypeToPb(ref.reference_type));
198 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800199
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700200 StringPool* source_pool_;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700201 pb::Value* out_pb_value_;
202 pb::Item* out_pb_item_;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800203};
204
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700205} // namespace
Adam Lesinski59e04c62016-02-04 15:59:23 -0800206
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700207std::unique_ptr<pb::ResourceTable> SerializeTableToPb(ResourceTable* table) {
Adam Lesinski060b53d2017-07-28 17:10:35 -0700208 // We must do this before writing the resources, since the string pool IDs may change.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700209 table->string_pool.Prune();
Adam Lesinski060b53d2017-07-28 17:10:35 -0700210 table->string_pool.Sort([](const StringPool::Context& a, const StringPool::Context& b) -> int {
211 int diff = util::compare(a.priority, b.priority);
212 if (diff == 0) {
213 diff = a.config.compare(b.config);
214 }
215 return diff;
216 });
Adam Lesinski59e04c62016-02-04 15:59:23 -0800217
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700218 auto pb_table = util::make_unique<pb::ResourceTable>();
Adam Lesinski08535002017-08-04 16:15:17 -0700219 StringPool source_pool;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800220
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700221 for (auto& package : table->packages) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700222 pb::Package* pb_package = pb_table->add_package();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700223 if (package->id) {
224 pb_package->set_package_id(package->id.value());
225 }
226 pb_package->set_package_name(package->name);
227
228 for (auto& type : package->types) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700229 pb::Type* pb_type = pb_package->add_type();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700230 if (type->id) {
231 pb_type->set_id(type->id.value());
232 }
Adam Lesinskid5083f62017-01-16 15:07:21 -0800233 pb_type->set_name(ToString(type->type).to_string());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700234
235 for (auto& entry : type->entries) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700236 pb::Entry* pb_entry = pb_type->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700237 if (entry->id) {
238 pb_entry->set_id(entry->id.value());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800239 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700240 pb_entry->set_name(entry->name);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800241
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700242 // Write the SymbolStatus struct.
243 pb::SymbolStatus* pb_status = pb_entry->mutable_symbol_status();
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700244 pb_status->set_visibility(SerializeVisibilityToPb(entry->symbol_status.state));
245 SerializeSourceToPb(entry->symbol_status.source, &source_pool, pb_status->mutable_source());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700246 pb_status->set_comment(entry->symbol_status.comment);
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700247 pb_status->set_allow_new(entry->symbol_status.allow_new);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800248
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700249 for (auto& config_value : entry->values) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700250 pb::ConfigValue* pb_config_value = pb_entry->add_config_value();
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700251 SerializeConfig(config_value->config, pb_config_value->mutable_config());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700252 if (!config_value->product.empty()) {
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700253 pb_config_value->mutable_config()->set_product(config_value->product);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700254 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800255
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700256 pb::Value* pb_value = pb_config_value->mutable_value();
257 SerializeSourceToPb(config_value->value->GetSource(), &source_pool,
258 pb_value->mutable_source());
259 if (!config_value->value->GetComment().empty()) {
260 pb_value->set_comment(config_value->value->GetComment());
261 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800262
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700263 if (config_value->value->IsWeak()) {
264 pb_value->set_weak(true);
265 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800266
Adam Lesinski08535002017-08-04 16:15:17 -0700267 PbSerializerVisitor visitor(&source_pool, pb_value);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700268 config_value->value->Accept(&visitor);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800269 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700270 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800271 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700272 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800273
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700274 SerializeStringPoolToPb(source_pool, pb_table->mutable_source_pool());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700275 return pb_table;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800276}
277
Adam Lesinski4ffea042017-08-10 15:37:28 -0700278std::unique_ptr<pb::internal::CompiledFile> SerializeCompiledFileToPb(const ResourceFile& file) {
279 auto pb_file = util::make_unique<pb::internal::CompiledFile>();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700280 pb_file->set_resource_name(file.name.ToString());
281 pb_file->set_source_path(file.source.path);
282 SerializeConfig(file.config, pb_file->mutable_config());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800283
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700284 for (const SourcedResourceName& exported : file.exported_symbols) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700285 pb::internal::CompiledFile_Symbol* pb_symbol = pb_file->add_exported_symbol();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700286 pb_symbol->set_resource_name(exported.name.ToString());
Adam Lesinski4ffea042017-08-10 15:37:28 -0700287 pb_symbol->mutable_source()->set_line_number(exported.line);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700288 }
289 return pb_file;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800290}
291
Adam Lesinski4ffea042017-08-10 15:37:28 -0700292CompiledFileOutputStream::CompiledFileOutputStream(ZeroCopyOutputStream* out) : out_(out) {
293}
Adam Lesinski59e04c62016-02-04 15:59:23 -0800294
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700295void CompiledFileOutputStream::EnsureAlignedWrite() {
296 const int padding = out_.ByteCount() % 4;
297 if (padding > 0) {
298 uint32_t zero = 0u;
299 out_.WriteRaw(&zero, padding);
300 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800301}
302
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700303void CompiledFileOutputStream::WriteLittleEndian32(uint32_t val) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700304 EnsureAlignedWrite();
305 out_.WriteLittleEndian32(val);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700306}
307
Adam Lesinski4ffea042017-08-10 15:37:28 -0700308void CompiledFileOutputStream::WriteCompiledFile(const pb::internal::CompiledFile* compiled_file) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700309 EnsureAlignedWrite();
310 out_.WriteLittleEndian64(static_cast<uint64_t>(compiled_file->ByteSize()));
311 compiled_file->SerializeWithCachedSizes(&out_);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700312}
313
314void CompiledFileOutputStream::WriteData(const BigBuffer* buffer) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700315 EnsureAlignedWrite();
316 out_.WriteLittleEndian64(static_cast<uint64_t>(buffer->size()));
317 for (const BigBuffer::Block& block : *buffer) {
318 out_.WriteRaw(block.buffer.get(), block.size);
319 }
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700320}
321
322void CompiledFileOutputStream::WriteData(const void* data, size_t len) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700323 EnsureAlignedWrite();
324 out_.WriteLittleEndian64(static_cast<uint64_t>(len));
325 out_.WriteRaw(data, len);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700326}
327
Adam Lesinski4ffea042017-08-10 15:37:28 -0700328bool CompiledFileOutputStream::HadError() {
329 return out_.HadError();
330}
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700331
332CompiledFileInputStream::CompiledFileInputStream(const void* data, size_t size)
333 : in_(static_cast<const uint8_t*>(data), size) {}
334
335void CompiledFileInputStream::EnsureAlignedRead() {
336 const int padding = in_.CurrentPosition() % 4;
337 if (padding > 0) {
338 // Reads are always 4 byte aligned.
339 in_.Skip(padding);
340 }
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700341}
342
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700343bool CompiledFileInputStream::ReadLittleEndian32(uint32_t* out_val) {
344 EnsureAlignedRead();
345 return in_.ReadLittleEndian32(out_val);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700346}
347
Adam Lesinski4ffea042017-08-10 15:37:28 -0700348bool CompiledFileInputStream::ReadCompiledFile(pb::internal::CompiledFile* out_val) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700349 EnsureAlignedRead();
350
Tamas Berghammer383db5eb2016-06-22 15:21:38 +0100351 google::protobuf::uint64 pb_size = 0u;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700352 if (!in_.ReadLittleEndian64(&pb_size)) {
353 return false;
354 }
355
356 CodedInputStream::Limit l = in_.PushLimit(static_cast<int>(pb_size));
357
358 // Check that we haven't tried to read past the end.
359 if (static_cast<uint64_t>(in_.BytesUntilLimit()) != pb_size) {
360 in_.PopLimit(l);
361 in_.PushLimit(0);
362 return false;
363 }
364
365 if (!out_val->ParsePartialFromCodedStream(&in_)) {
366 in_.PopLimit(l);
367 in_.PushLimit(0);
368 return false;
369 }
370
371 in_.PopLimit(l);
372 return true;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700373}
374
Adam Lesinski4ffea042017-08-10 15:37:28 -0700375bool CompiledFileInputStream::ReadDataMetaData(uint64_t* out_offset, uint64_t* out_len) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700376 EnsureAlignedRead();
377
Tamas Berghammer383db5eb2016-06-22 15:21:38 +0100378 google::protobuf::uint64 pb_size = 0u;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700379 if (!in_.ReadLittleEndian64(&pb_size)) {
380 return false;
381 }
382
383 // Check that we aren't trying to read past the end.
384 if (pb_size > static_cast<uint64_t>(in_.BytesUntilLimit())) {
385 in_.PushLimit(0);
386 return false;
387 }
388
389 uint64_t offset = static_cast<uint64_t>(in_.CurrentPosition());
390 if (!in_.Skip(pb_size)) {
391 return false;
392 }
393
394 *out_offset = offset;
395 *out_len = pb_size;
396 return true;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700397}
398
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700399} // namespace aapt