blob: de10bb80bf7e6b0971fa54f830654e6e4c2693b7 [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 Lesinski5eeaadd2016-08-25 12:26:56 -070027using google::protobuf::io::CodedOutputStream;
28using google::protobuf::io::CodedInputStream;
29using google::protobuf::io::ZeroCopyOutputStream;
30
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 Lesinskice5e56e2016-10-21 17:56:45 -070039 /**
40 * Constructor to use when expecting to serialize any value.
41 */
42 PbSerializerVisitor(StringPool* source_pool, StringPool* symbol_pool,
43 pb::Value* out_pb_value)
44 : source_pool_(source_pool),
45 symbol_pool_(symbol_pool),
46 out_pb_value_(out_pb_value),
47 out_pb_item_(nullptr) {}
48
49 /**
50 * Constructor to use when expecting to serialize an Item.
51 */
52 PbSerializerVisitor(StringPool* sourcePool, StringPool* symbolPool,
53 pb::Item* outPbItem)
54 : source_pool_(sourcePool),
55 symbol_pool_(symbolPool),
56 out_pb_value_(nullptr),
57 out_pb_item_(outPbItem) {}
58
59 void Visit(Reference* ref) override {
60 SerializeReferenceToPb(*ref, pb_item()->mutable_ref());
61 }
62
63 void Visit(String* str) override {
64 pb_item()->mutable_str()->set_idx(str->value.index());
65 }
66
67 void Visit(StyledString* str) override {
68 pb_item()->mutable_str()->set_idx(str->value.index());
69 }
70
71 void Visit(FileReference* file) override {
72 pb_item()->mutable_file()->set_path_idx(file->path.index());
73 }
74
75 void Visit(Id* id) override { pb_item()->mutable_id(); }
76
77 void Visit(RawString* raw_str) override {
78 pb_item()->mutable_raw_str()->set_idx(raw_str->value.index());
79 }
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
90 void VisitItem(Item* item) override { LOG(FATAL) << "unimplemented item"; }
91
92 void Visit(Attribute* attr) override {
93 pb::Attribute* pb_attr = pb_compound_value()->mutable_attr();
94 pb_attr->set_format_flags(attr->type_mask);
95 pb_attr->set_min_int(attr->min_int);
96 pb_attr->set_max_int(attr->max_int);
97
98 for (auto& symbol : attr->symbols) {
99 pb::Attribute_Symbol* pb_symbol = pb_attr->add_symbols();
100 SerializeItemCommonToPb(symbol.symbol, pb_symbol);
101 SerializeReferenceToPb(symbol.symbol, pb_symbol->mutable_name());
102 pb_symbol->set_value(symbol.value);
103 }
104 }
105
106 void Visit(Style* style) override {
107 pb::Style* pb_style = pb_compound_value()->mutable_style();
108 if (style->parent) {
109 SerializeReferenceToPb(style->parent.value(), pb_style->mutable_parent());
110 SerializeSourceToPb(style->parent.value().GetSource(), source_pool_,
111 pb_style->mutable_parent_source());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800112 }
113
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700114 for (Style::Entry& entry : style->entries) {
115 pb::Style_Entry* pb_entry = pb_style->add_entries();
116 SerializeReferenceToPb(entry.key, pb_entry->mutable_key());
117
118 pb::Item* pb_item = pb_entry->mutable_item();
119 SerializeItemCommonToPb(entry.key, pb_entry);
120 PbSerializerVisitor sub_visitor(source_pool_, symbol_pool_, pb_item);
121 entry.value->Accept(&sub_visitor);
122 }
123 }
124
125 void Visit(Styleable* styleable) override {
126 pb::Styleable* pb_styleable = pb_compound_value()->mutable_styleable();
127 for (Reference& entry : styleable->entries) {
128 pb::Styleable_Entry* pb_entry = pb_styleable->add_entries();
129 SerializeItemCommonToPb(entry, pb_entry);
130 SerializeReferenceToPb(entry, pb_entry->mutable_attr());
131 }
132 }
133
134 void Visit(Array* array) override {
135 pb::Array* pb_array = pb_compound_value()->mutable_array();
136 for (auto& value : array->items) {
137 pb::Array_Entry* pb_entry = pb_array->add_entries();
138 SerializeItemCommonToPb(*value, pb_entry);
139 PbSerializerVisitor sub_visitor(source_pool_, symbol_pool_,
140 pb_entry->mutable_item());
141 value->Accept(&sub_visitor);
142 }
143 }
144
145 void Visit(Plural* plural) override {
146 pb::Plural* pb_plural = pb_compound_value()->mutable_plural();
147 const size_t count = plural->values.size();
148 for (size_t i = 0; i < count; i++) {
149 if (!plural->values[i]) {
150 // No plural value set here.
151 continue;
152 }
153
154 pb::Plural_Entry* pb_entry = pb_plural->add_entries();
155 pb_entry->set_arity(SerializePluralEnumToPb(i));
156 pb::Item* pb_element = pb_entry->mutable_item();
157 SerializeItemCommonToPb(*plural->values[i], pb_entry);
158 PbSerializerVisitor sub_visitor(source_pool_, symbol_pool_, pb_element);
159 plural->values[i]->Accept(&sub_visitor);
160 }
161 }
162
163 private:
164 DISALLOW_COPY_AND_ASSIGN(PbSerializerVisitor);
165
166 pb::Item* pb_item() {
167 if (out_pb_value_) {
168 return out_pb_value_->mutable_item();
169 }
170 return out_pb_item_;
171 }
172
173 pb::CompoundValue* pb_compound_value() {
174 CHECK(out_pb_value_ != nullptr);
175 return out_pb_value_->mutable_compound_value();
176 }
177
178 template <typename T>
179 void SerializeItemCommonToPb(const Item& item, T* pb_item) {
180 SerializeSourceToPb(item.GetSource(), source_pool_,
181 pb_item->mutable_source());
182 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) {
193 StringPool::Ref symbol_ref =
194 symbol_pool_->MakeRef(ref.name.value().ToString());
195 pb_ref->set_symbol_idx(static_cast<uint32_t>(symbol_ref.index()));
Adam Lesinski59e04c62016-02-04 15:59:23 -0800196 }
197
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700198 pb_ref->set_private_(ref.private_reference);
199 pb_ref->set_type(SerializeReferenceTypeToPb(ref.reference_type));
200 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800201
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700202 StringPool* source_pool_;
203 StringPool* symbol_pool_;
204 pb::Value* out_pb_value_;
205 pb::Item* out_pb_item_;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800206};
207
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700208} // namespace
Adam Lesinski59e04c62016-02-04 15:59:23 -0800209
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700210std::unique_ptr<pb::ResourceTable> SerializeTableToPb(ResourceTable* table) {
211 // We must do this before writing the resources, since the string pool IDs may
212 // change.
213 table->string_pool.Sort(
214 [](const StringPool::Entry& a, const StringPool::Entry& b) -> bool {
Adam Lesinski59e04c62016-02-04 15:59:23 -0800215 int diff = a.context.priority - b.context.priority;
216 if (diff < 0) return true;
217 if (diff > 0) return false;
218 diff = a.context.config.compare(b.context.config);
219 if (diff < 0) return true;
220 if (diff > 0) return false;
221 return a.value < b.value;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700222 });
223 table->string_pool.Prune();
Adam Lesinski59e04c62016-02-04 15:59:23 -0800224
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700225 auto pb_table = util::make_unique<pb::ResourceTable>();
226 SerializeStringPoolToPb(table->string_pool, pb_table->mutable_string_pool());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800227
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700228 StringPool source_pool, symbol_pool;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800229
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700230 for (auto& package : table->packages) {
231 pb::Package* pb_package = pb_table->add_packages();
232 if (package->id) {
233 pb_package->set_package_id(package->id.value());
234 }
235 pb_package->set_package_name(package->name);
236
237 for (auto& type : package->types) {
238 pb::Type* pb_type = pb_package->add_types();
239 if (type->id) {
240 pb_type->set_id(type->id.value());
241 }
Adam Lesinskid5083f62017-01-16 15:07:21 -0800242 pb_type->set_name(ToString(type->type).to_string());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700243
244 for (auto& entry : type->entries) {
245 pb::Entry* pb_entry = pb_type->add_entries();
246 if (entry->id) {
247 pb_entry->set_id(entry->id.value());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800248 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700249 pb_entry->set_name(entry->name);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800250
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700251 // Write the SymbolStatus struct.
252 pb::SymbolStatus* pb_status = pb_entry->mutable_symbol_status();
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700253 pb_status->set_visibility(SerializeVisibilityToPb(entry->symbol_status.state));
254 SerializeSourceToPb(entry->symbol_status.source, &source_pool, pb_status->mutable_source());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700255 pb_status->set_comment(entry->symbol_status.comment);
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700256 pb_status->set_allow_new(entry->symbol_status.allow_new);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800257
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700258 for (auto& config_value : entry->values) {
259 pb::ConfigValue* pb_config_value = pb_entry->add_config_values();
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700260 SerializeConfig(config_value->config, pb_config_value->mutable_config());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700261 if (!config_value->product.empty()) {
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700262 pb_config_value->mutable_config()->set_product(config_value->product);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700263 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800264
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700265 pb::Value* pb_value = pb_config_value->mutable_value();
266 SerializeSourceToPb(config_value->value->GetSource(), &source_pool,
267 pb_value->mutable_source());
268 if (!config_value->value->GetComment().empty()) {
269 pb_value->set_comment(config_value->value->GetComment());
270 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800271
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700272 if (config_value->value->IsWeak()) {
273 pb_value->set_weak(true);
274 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800275
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700276 PbSerializerVisitor visitor(&source_pool, &symbol_pool, pb_value);
277 config_value->value->Accept(&visitor);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800278 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700279 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800280 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700281 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800282
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700283 SerializeStringPoolToPb(source_pool, pb_table->mutable_source_pool());
284 SerializeStringPoolToPb(symbol_pool, pb_table->mutable_symbol_pool());
285 return pb_table;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800286}
287
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700288std::unique_ptr<pb::CompiledFile> SerializeCompiledFileToPb(
289 const ResourceFile& file) {
290 auto pb_file = util::make_unique<pb::CompiledFile>();
291 pb_file->set_resource_name(file.name.ToString());
292 pb_file->set_source_path(file.source.path);
293 SerializeConfig(file.config, pb_file->mutable_config());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800294
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700295 for (const SourcedResourceName& exported : file.exported_symbols) {
296 pb::CompiledFile_Symbol* pb_symbol = pb_file->add_exported_symbols();
297 pb_symbol->set_resource_name(exported.name.ToString());
298 pb_symbol->set_line_no(exported.line);
299 }
300 return pb_file;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800301}
302
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700303CompiledFileOutputStream::CompiledFileOutputStream(ZeroCopyOutputStream* out)
304 : out_(out) {}
Adam Lesinski59e04c62016-02-04 15:59:23 -0800305
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700306void CompiledFileOutputStream::EnsureAlignedWrite() {
307 const int padding = out_.ByteCount() % 4;
308 if (padding > 0) {
309 uint32_t zero = 0u;
310 out_.WriteRaw(&zero, padding);
311 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800312}
313
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700314void CompiledFileOutputStream::WriteLittleEndian32(uint32_t val) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700315 EnsureAlignedWrite();
316 out_.WriteLittleEndian32(val);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700317}
318
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700319void CompiledFileOutputStream::WriteCompiledFile(
320 const pb::CompiledFile* compiled_file) {
321 EnsureAlignedWrite();
322 out_.WriteLittleEndian64(static_cast<uint64_t>(compiled_file->ByteSize()));
323 compiled_file->SerializeWithCachedSizes(&out_);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700324}
325
326void CompiledFileOutputStream::WriteData(const BigBuffer* buffer) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700327 EnsureAlignedWrite();
328 out_.WriteLittleEndian64(static_cast<uint64_t>(buffer->size()));
329 for (const BigBuffer::Block& block : *buffer) {
330 out_.WriteRaw(block.buffer.get(), block.size);
331 }
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700332}
333
334void CompiledFileOutputStream::WriteData(const void* data, size_t len) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700335 EnsureAlignedWrite();
336 out_.WriteLittleEndian64(static_cast<uint64_t>(len));
337 out_.WriteRaw(data, len);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700338}
339
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700340bool CompiledFileOutputStream::HadError() { return out_.HadError(); }
341
342CompiledFileInputStream::CompiledFileInputStream(const void* data, size_t size)
343 : in_(static_cast<const uint8_t*>(data), size) {}
344
345void CompiledFileInputStream::EnsureAlignedRead() {
346 const int padding = in_.CurrentPosition() % 4;
347 if (padding > 0) {
348 // Reads are always 4 byte aligned.
349 in_.Skip(padding);
350 }
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700351}
352
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700353bool CompiledFileInputStream::ReadLittleEndian32(uint32_t* out_val) {
354 EnsureAlignedRead();
355 return in_.ReadLittleEndian32(out_val);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700356}
357
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700358bool CompiledFileInputStream::ReadCompiledFile(pb::CompiledFile* out_val) {
359 EnsureAlignedRead();
360
Tamas Berghammer383db5eb2016-06-22 15:21:38 +0100361 google::protobuf::uint64 pb_size = 0u;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700362 if (!in_.ReadLittleEndian64(&pb_size)) {
363 return false;
364 }
365
366 CodedInputStream::Limit l = in_.PushLimit(static_cast<int>(pb_size));
367
368 // Check that we haven't tried to read past the end.
369 if (static_cast<uint64_t>(in_.BytesUntilLimit()) != pb_size) {
370 in_.PopLimit(l);
371 in_.PushLimit(0);
372 return false;
373 }
374
375 if (!out_val->ParsePartialFromCodedStream(&in_)) {
376 in_.PopLimit(l);
377 in_.PushLimit(0);
378 return false;
379 }
380
381 in_.PopLimit(l);
382 return true;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700383}
384
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700385bool CompiledFileInputStream::ReadDataMetaData(uint64_t* out_offset,
386 uint64_t* out_len) {
387 EnsureAlignedRead();
388
Tamas Berghammer383db5eb2016-06-22 15:21:38 +0100389 google::protobuf::uint64 pb_size = 0u;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700390 if (!in_.ReadLittleEndian64(&pb_size)) {
391 return false;
392 }
393
394 // Check that we aren't trying to read past the end.
395 if (pb_size > static_cast<uint64_t>(in_.BytesUntilLimit())) {
396 in_.PushLimit(0);
397 return false;
398 }
399
400 uint64_t offset = static_cast<uint64_t>(in_.CurrentPosition());
401 if (!in_.Skip(pb_size)) {
402 return false;
403 }
404
405 *out_offset = offset;
406 *out_len = pb_size;
407 return true;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700408}
409
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700410} // namespace aapt