• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
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 
25 #include "android-base/logging.h"
26 
27 using ::google::protobuf::io::CodedInputStream;
28 using ::google::protobuf::io::CodedOutputStream;
29 using ::google::protobuf::io::ZeroCopyOutputStream;
30 
31 namespace aapt {
32 
33 namespace {
34 
35 class PbSerializerVisitor : public RawValueVisitor {
36  public:
37   using RawValueVisitor::Visit;
38 
39   // Constructor to use when expecting to serialize any value.
PbSerializerVisitor(StringPool * source_pool,pb::Value * out_pb_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   }
43 
44   // Constructor to use when expecting to serialize an Item.
PbSerializerVisitor(StringPool * sourcePool,pb::Item * outPbItem)45   PbSerializerVisitor(StringPool* sourcePool, pb::Item* outPbItem)
46       : source_pool_(sourcePool), out_pb_value_(nullptr), out_pb_item_(outPbItem) {
47   }
48 
Visit(Reference * ref)49   void Visit(Reference* ref) override {
50     SerializeReferenceToPb(*ref, pb_item()->mutable_ref());
51   }
52 
Visit(String * str)53   void Visit(String* str) override {
54     pb_item()->mutable_str()->set_value(*str->value);
55   }
56 
Visit(RawString * str)57   void Visit(RawString* str) override {
58     pb_item()->mutable_raw_str()->set_value(*str->value);
59   }
60 
Visit(StyledString * str)61   void Visit(StyledString* str) override {
62     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     }
71   }
72 
Visit(FileReference * file)73   void Visit(FileReference* file) override {
74     pb_item()->mutable_file()->set_path(*file->path);
75   }
76 
Visit(Id *)77   void Visit(Id* /*id*/) override {
78     pb_item()->mutable_id();
79   }
80 
Visit(BinaryPrimitive * prim)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 
VisitItem(Item * item)90   void VisitItem(Item* item) override {
91     LOG(FATAL) << "unimplemented item";
92   }
93 
Visit(Attribute * attr)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) {
101       pb::Attribute_Symbol* pb_symbol = pb_attr->add_symbol();
102       SerializeItemCommonToPb(symbol.symbol, pb_symbol);
103       SerializeReferenceToPb(symbol.symbol, pb_symbol->mutable_name());
104       pb_symbol->set_value(symbol.value);
105     }
106   }
107 
Visit(Style * style)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());
114     }
115 
116     for (Style::Entry& entry : style->entries) {
117       pb::Style_Entry* pb_entry = pb_style->add_entry();
118       SerializeReferenceToPb(entry.key, pb_entry->mutable_key());
119 
120       pb::Item* pb_item = pb_entry->mutable_item();
121       SerializeItemCommonToPb(entry.key, pb_entry);
122       PbSerializerVisitor sub_visitor(source_pool_, pb_item);
123       entry.value->Accept(&sub_visitor);
124     }
125   }
126 
Visit(Styleable * styleable)127   void Visit(Styleable* styleable) override {
128     pb::Styleable* pb_styleable = pb_compound_value()->mutable_styleable();
129     for (Reference& entry : styleable->entries) {
130       pb::Styleable_Entry* pb_entry = pb_styleable->add_entry();
131       SerializeItemCommonToPb(entry, pb_entry);
132       SerializeReferenceToPb(entry, pb_entry->mutable_attr());
133     }
134   }
135 
Visit(Array * array)136   void Visit(Array* array) override {
137     pb::Array* pb_array = pb_compound_value()->mutable_array();
138     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());
142       value->Accept(&sub_visitor);
143     }
144   }
145 
Visit(Plural * plural)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 
155       pb::Plural_Entry* pb_entry = pb_plural->add_entry();
156       pb_entry->set_arity(SerializePluralEnumToPb(i));
157       pb::Item* pb_element = pb_entry->mutable_item();
158       SerializeItemCommonToPb(*plural->values[i], pb_entry);
159       PbSerializerVisitor sub_visitor(source_pool_, pb_element);
160       plural->values[i]->Accept(&sub_visitor);
161     }
162   }
163 
164  private:
165   DISALLOW_COPY_AND_ASSIGN(PbSerializerVisitor);
166 
pb_item()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 
pb_compound_value()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>
SerializeItemCommonToPb(const Item & item,T * pb_item)180   void SerializeItemCommonToPb(const Item& item, T* pb_item) {
181     SerializeSourceToPb(item.GetSource(), source_pool_, pb_item->mutable_source());
182     if (!item.GetComment().empty()) {
183       pb_item->set_comment(item.GetComment());
184     }
185   }
186 
SerializeReferenceToPb(const Reference & ref,pb::Reference * pb_ref)187   void SerializeReferenceToPb(const Reference& ref, pb::Reference* pb_ref) {
188     if (ref.id) {
189       pb_ref->set_id(ref.id.value().id);
190     }
191 
192     if (ref.name) {
193       pb_ref->set_name(ref.name.value().ToString());
194     }
195 
196     pb_ref->set_private_(ref.private_reference);
197     pb_ref->set_type(SerializeReferenceTypeToPb(ref.reference_type));
198   }
199 
200   StringPool* source_pool_;
201   pb::Value* out_pb_value_;
202   pb::Item* out_pb_item_;
203 };
204 
205 }  // namespace
206 
SerializeTableToPb(ResourceTable * table)207 std::unique_ptr<pb::ResourceTable> SerializeTableToPb(ResourceTable* table) {
208   // We must do this before writing the resources, since the string pool IDs may change.
209   table->string_pool.Prune();
210   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   });
217 
218   auto pb_table = util::make_unique<pb::ResourceTable>();
219   StringPool source_pool;
220 
221   for (auto& package : table->packages) {
222     pb::Package* pb_package = pb_table->add_package();
223     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) {
229       pb::Type* pb_type = pb_package->add_type();
230       if (type->id) {
231         pb_type->set_id(type->id.value());
232       }
233       pb_type->set_name(ToString(type->type).to_string());
234 
235       for (auto& entry : type->entries) {
236         pb::Entry* pb_entry = pb_type->add_entry();
237         if (entry->id) {
238           pb_entry->set_id(entry->id.value());
239         }
240         pb_entry->set_name(entry->name);
241 
242         // Write the SymbolStatus struct.
243         pb::SymbolStatus* pb_status = pb_entry->mutable_symbol_status();
244         pb_status->set_visibility(SerializeVisibilityToPb(entry->symbol_status.state));
245         SerializeSourceToPb(entry->symbol_status.source, &source_pool, pb_status->mutable_source());
246         pb_status->set_comment(entry->symbol_status.comment);
247         pb_status->set_allow_new(entry->symbol_status.allow_new);
248 
249         for (auto& config_value : entry->values) {
250           pb::ConfigValue* pb_config_value = pb_entry->add_config_value();
251           SerializeConfig(config_value->config, pb_config_value->mutable_config());
252           if (!config_value->product.empty()) {
253             pb_config_value->mutable_config()->set_product(config_value->product);
254           }
255 
256           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           }
262 
263           if (config_value->value->IsWeak()) {
264             pb_value->set_weak(true);
265           }
266 
267           PbSerializerVisitor visitor(&source_pool, pb_value);
268           config_value->value->Accept(&visitor);
269         }
270       }
271     }
272   }
273 
274   SerializeStringPoolToPb(source_pool, pb_table->mutable_source_pool());
275   return pb_table;
276 }
277 
SerializeCompiledFileToPb(const ResourceFile & file)278 std::unique_ptr<pb::internal::CompiledFile> SerializeCompiledFileToPb(const ResourceFile& file) {
279   auto pb_file = util::make_unique<pb::internal::CompiledFile>();
280   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());
283 
284   for (const SourcedResourceName& exported : file.exported_symbols) {
285     pb::internal::CompiledFile_Symbol* pb_symbol = pb_file->add_exported_symbol();
286     pb_symbol->set_resource_name(exported.name.ToString());
287     pb_symbol->mutable_source()->set_line_number(exported.line);
288   }
289   return pb_file;
290 }
291 
CompiledFileOutputStream(ZeroCopyOutputStream * out)292 CompiledFileOutputStream::CompiledFileOutputStream(ZeroCopyOutputStream* out) : out_(out) {
293 }
294 
EnsureAlignedWrite()295 void CompiledFileOutputStream::EnsureAlignedWrite() {
296   const int padding = out_.ByteCount() % 4;
297   if (padding > 0) {
298     uint32_t zero = 0u;
299     out_.WriteRaw(&zero, padding);
300   }
301 }
302 
WriteLittleEndian32(uint32_t val)303 void CompiledFileOutputStream::WriteLittleEndian32(uint32_t val) {
304   EnsureAlignedWrite();
305   out_.WriteLittleEndian32(val);
306 }
307 
WriteCompiledFile(const pb::internal::CompiledFile * compiled_file)308 void CompiledFileOutputStream::WriteCompiledFile(const pb::internal::CompiledFile* compiled_file) {
309   EnsureAlignedWrite();
310   out_.WriteLittleEndian64(static_cast<uint64_t>(compiled_file->ByteSize()));
311   compiled_file->SerializeWithCachedSizes(&out_);
312 }
313 
WriteData(const BigBuffer * buffer)314 void CompiledFileOutputStream::WriteData(const BigBuffer* buffer) {
315   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   }
320 }
321 
WriteData(const void * data,size_t len)322 void CompiledFileOutputStream::WriteData(const void* data, size_t len) {
323   EnsureAlignedWrite();
324   out_.WriteLittleEndian64(static_cast<uint64_t>(len));
325   out_.WriteRaw(data, len);
326 }
327 
HadError()328 bool CompiledFileOutputStream::HadError() {
329   return out_.HadError();
330 }
331 
CompiledFileInputStream(const void * data,size_t size)332 CompiledFileInputStream::CompiledFileInputStream(const void* data, size_t size)
333     : in_(static_cast<const uint8_t*>(data), size) {}
334 
EnsureAlignedRead()335 void 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   }
341 }
342 
ReadLittleEndian32(uint32_t * out_val)343 bool CompiledFileInputStream::ReadLittleEndian32(uint32_t* out_val) {
344   EnsureAlignedRead();
345   return in_.ReadLittleEndian32(out_val);
346 }
347 
ReadCompiledFile(pb::internal::CompiledFile * out_val)348 bool CompiledFileInputStream::ReadCompiledFile(pb::internal::CompiledFile* out_val) {
349   EnsureAlignedRead();
350 
351   google::protobuf::uint64 pb_size = 0u;
352   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;
373 }
374 
ReadDataMetaData(uint64_t * out_offset,uint64_t * out_len)375 bool CompiledFileInputStream::ReadDataMetaData(uint64_t* out_offset, uint64_t* out_len) {
376   EnsureAlignedRead();
377 
378   google::protobuf::uint64 pb_size = 0u;
379   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;
397 }
398 
399 }  // namespace aapt
400