blob: e15a0adbda9b27b59a02a1c87d4fb5be8556f503 [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
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070017#include "proto/ProtoSerialize.h"
18
19#include "android-base/logging.h"
20
Adam Lesinski59e04c62016-02-04 15:59:23 -080021#include "Resource.h"
22#include "ResourceTable.h"
23#include "StringPool.h"
24#include "ValueVisitor.h"
25#include "proto/ProtoHelpers.h"
Adam Lesinski59e04c62016-02-04 15:59:23 -080026#include "util/BigBuffer.h"
27
Adam Lesinski08535002017-08-04 16:15:17 -070028using ::google::protobuf::io::CodedInputStream;
29using ::google::protobuf::io::CodedOutputStream;
30using ::google::protobuf::io::ZeroCopyOutputStream;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -070031
Adam Lesinski59e04c62016-02-04 15:59:23 -080032namespace aapt {
33
34namespace {
35
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070036class PbSerializerVisitor : public ConstValueVisitor {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070037 public:
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070038 using ConstValueVisitor::Visit;
Adam Lesinski59e04c62016-02-04 15:59:23 -080039
Adam Lesinski08535002017-08-04 16:15:17 -070040 // Constructor to use when expecting to serialize any value.
41 PbSerializerVisitor(StringPool* source_pool, pb::Value* out_pb_value)
42 : source_pool_(source_pool), out_pb_value_(out_pb_value), out_pb_item_(nullptr) {
43 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070044
Adam Lesinski08535002017-08-04 16:15:17 -070045 // Constructor to use when expecting to serialize an Item.
46 PbSerializerVisitor(StringPool* sourcePool, pb::Item* outPbItem)
47 : source_pool_(sourcePool), out_pb_value_(nullptr), out_pb_item_(outPbItem) {
48 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070049
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070050 void Visit(const Reference* ref) override {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070051 SerializeReferenceToPb(*ref, pb_item()->mutable_ref());
52 }
53
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070054 void Visit(const String* str) override {
Adam Lesinski08535002017-08-04 16:15:17 -070055 pb_item()->mutable_str()->set_value(*str->value);
56 }
57
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070058 void Visit(const RawString* str) override {
Adam Lesinski08535002017-08-04 16:15:17 -070059 pb_item()->mutable_raw_str()->set_value(*str->value);
Adam Lesinskice5e56e2016-10-21 17:56:45 -070060 }
61
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070062 void Visit(const StyledString* str) override {
Adam Lesinski08535002017-08-04 16:15:17 -070063 pb::StyledString* pb_str = pb_item()->mutable_styled_str();
64 pb_str->set_value(str->value->value);
65
66 for (const StringPool::Span& span : str->value->spans) {
67 pb::StyledString::Span* pb_span = pb_str->add_span();
68 pb_span->set_tag(*span.name);
69 pb_span->set_first_char(span.first_char);
70 pb_span->set_last_char(span.last_char);
71 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070072 }
73
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070074 void Visit(const FileReference* file) override {
Adam Lesinski08535002017-08-04 16:15:17 -070075 pb_item()->mutable_file()->set_path(*file->path);
Adam Lesinskice5e56e2016-10-21 17:56:45 -070076 }
77
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070078 void Visit(const Id* /*id*/) override {
Adam Lesinski08535002017-08-04 16:15:17 -070079 pb_item()->mutable_id();
Adam Lesinskice5e56e2016-10-21 17:56:45 -070080 }
81
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070082 void Visit(const BinaryPrimitive* prim) override {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070083 android::Res_value val = {};
84 prim->Flatten(&val);
85
86 pb::Primitive* pb_prim = pb_item()->mutable_prim();
87 pb_prim->set_type(val.dataType);
88 pb_prim->set_data(val.data);
89 }
90
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070091 void VisitItem(const Item* item) override {
Adam Lesinski060b53d2017-07-28 17:10:35 -070092 LOG(FATAL) << "unimplemented item";
93 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070094
Adam Lesinskid3ffa8442017-09-28 13:34:35 -070095 void Visit(const Attribute* attr) override {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070096 pb::Attribute* pb_attr = pb_compound_value()->mutable_attr();
97 pb_attr->set_format_flags(attr->type_mask);
98 pb_attr->set_min_int(attr->min_int);
99 pb_attr->set_max_int(attr->max_int);
100
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700101 for (const auto& symbol : attr->symbols) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700102 pb::Attribute_Symbol* pb_symbol = pb_attr->add_symbol();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700103 SerializeItemCommonToPb(symbol.symbol, pb_symbol);
104 SerializeReferenceToPb(symbol.symbol, pb_symbol->mutable_name());
105 pb_symbol->set_value(symbol.value);
106 }
107 }
108
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700109 void Visit(const Style* style) override {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700110 pb::Style* pb_style = pb_compound_value()->mutable_style();
111 if (style->parent) {
112 SerializeReferenceToPb(style->parent.value(), pb_style->mutable_parent());
113 SerializeSourceToPb(style->parent.value().GetSource(), source_pool_,
114 pb_style->mutable_parent_source());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800115 }
116
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700117 for (const Style::Entry& entry : style->entries) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700118 pb::Style_Entry* pb_entry = pb_style->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700119 SerializeReferenceToPb(entry.key, pb_entry->mutable_key());
120
121 pb::Item* pb_item = pb_entry->mutable_item();
122 SerializeItemCommonToPb(entry.key, pb_entry);
Adam Lesinski08535002017-08-04 16:15:17 -0700123 PbSerializerVisitor sub_visitor(source_pool_, pb_item);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700124 entry.value->Accept(&sub_visitor);
125 }
126 }
127
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700128 void Visit(const Styleable* styleable) override {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700129 pb::Styleable* pb_styleable = pb_compound_value()->mutable_styleable();
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700130 for (const Reference& entry : styleable->entries) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700131 pb::Styleable_Entry* pb_entry = pb_styleable->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700132 SerializeItemCommonToPb(entry, pb_entry);
133 SerializeReferenceToPb(entry, pb_entry->mutable_attr());
134 }
135 }
136
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700137 void Visit(const Array* array) override {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700138 pb::Array* pb_array = pb_compound_value()->mutable_array();
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700139 for (const auto& value : array->elements) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700140 pb::Array_Element* pb_element = pb_array->add_element();
141 SerializeItemCommonToPb(*value, pb_element);
142 PbSerializerVisitor sub_visitor(source_pool_, pb_element->mutable_item());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700143 value->Accept(&sub_visitor);
144 }
145 }
146
Adam Lesinskid3ffa8442017-09-28 13:34:35 -0700147 void Visit(const Plural* plural) override {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700148 pb::Plural* pb_plural = pb_compound_value()->mutable_plural();
149 const size_t count = plural->values.size();
150 for (size_t i = 0; i < count; i++) {
151 if (!plural->values[i]) {
152 // No plural value set here.
153 continue;
154 }
155
Adam Lesinski4ffea042017-08-10 15:37:28 -0700156 pb::Plural_Entry* pb_entry = pb_plural->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700157 pb_entry->set_arity(SerializePluralEnumToPb(i));
158 pb::Item* pb_element = pb_entry->mutable_item();
159 SerializeItemCommonToPb(*plural->values[i], pb_entry);
Adam Lesinski08535002017-08-04 16:15:17 -0700160 PbSerializerVisitor sub_visitor(source_pool_, pb_element);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700161 plural->values[i]->Accept(&sub_visitor);
162 }
163 }
164
165 private:
166 DISALLOW_COPY_AND_ASSIGN(PbSerializerVisitor);
167
168 pb::Item* pb_item() {
169 if (out_pb_value_) {
170 return out_pb_value_->mutable_item();
171 }
172 return out_pb_item_;
173 }
174
175 pb::CompoundValue* pb_compound_value() {
176 CHECK(out_pb_value_ != nullptr);
177 return out_pb_value_->mutable_compound_value();
178 }
179
180 template <typename T>
181 void SerializeItemCommonToPb(const Item& item, T* pb_item) {
Adam Lesinski08535002017-08-04 16:15:17 -0700182 SerializeSourceToPb(item.GetSource(), source_pool_, pb_item->mutable_source());
Adam Lesinski5bd44232017-09-07 09:39:07 -0700183 pb_item->set_comment(item.GetComment());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700184 }
185
186 void SerializeReferenceToPb(const Reference& ref, pb::Reference* pb_ref) {
Adam Lesinski5bd44232017-09-07 09:39:07 -0700187 pb_ref->set_id(ref.id.value_or_default(ResourceId(0x0)).id);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800188
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700189 if (ref.name) {
Adam Lesinski08535002017-08-04 16:15:17 -0700190 pb_ref->set_name(ref.name.value().ToString());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800191 }
192
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700193 pb_ref->set_private_(ref.private_reference);
194 pb_ref->set_type(SerializeReferenceTypeToPb(ref.reference_type));
195 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800196
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700197 StringPool* source_pool_;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700198 pb::Value* out_pb_value_;
199 pb::Item* out_pb_item_;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800200};
201
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700202} // namespace
Adam Lesinski59e04c62016-02-04 15:59:23 -0800203
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700204std::unique_ptr<pb::ResourceTable> SerializeTableToPb(ResourceTable* table) {
Adam Lesinski060b53d2017-07-28 17:10:35 -0700205 // We must do this before writing the resources, since the string pool IDs may change.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700206 table->string_pool.Prune();
Adam Lesinski060b53d2017-07-28 17:10:35 -0700207 table->string_pool.Sort([](const StringPool::Context& a, const StringPool::Context& b) -> int {
208 int diff = util::compare(a.priority, b.priority);
209 if (diff == 0) {
210 diff = a.config.compare(b.config);
211 }
212 return diff;
213 });
Adam Lesinski59e04c62016-02-04 15:59:23 -0800214
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700215 auto pb_table = util::make_unique<pb::ResourceTable>();
Adam Lesinski08535002017-08-04 16:15:17 -0700216 StringPool source_pool;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800217
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700218 for (auto& package : table->packages) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700219 pb::Package* pb_package = pb_table->add_package();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700220 if (package->id) {
Adam Lesinski5bd44232017-09-07 09:39:07 -0700221 pb_package->mutable_package_id()->set_id(package->id.value());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700222 }
223 pb_package->set_package_name(package->name);
224
225 for (auto& type : package->types) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700226 pb::Type* pb_type = pb_package->add_type();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700227 if (type->id) {
Adam Lesinski5bd44232017-09-07 09:39:07 -0700228 pb_type->mutable_type_id()->set_id(type->id.value());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700229 }
Adam Lesinskid5083f62017-01-16 15:07:21 -0800230 pb_type->set_name(ToString(type->type).to_string());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700231
232 for (auto& entry : type->entries) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700233 pb::Entry* pb_entry = pb_type->add_entry();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700234 if (entry->id) {
Adam Lesinski5bd44232017-09-07 09:39:07 -0700235 pb_entry->mutable_entry_id()->set_id(entry->id.value());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800236 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700237 pb_entry->set_name(entry->name);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800238
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700239 // Write the SymbolStatus struct.
240 pb::SymbolStatus* pb_status = pb_entry->mutable_symbol_status();
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700241 pb_status->set_visibility(SerializeVisibilityToPb(entry->symbol_status.state));
242 SerializeSourceToPb(entry->symbol_status.source, &source_pool, pb_status->mutable_source());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700243 pb_status->set_comment(entry->symbol_status.comment);
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700244 pb_status->set_allow_new(entry->symbol_status.allow_new);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800245
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700246 for (auto& config_value : entry->values) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700247 pb::ConfigValue* pb_config_value = pb_entry->add_config_value();
Adam Lesinski4488f1c2017-05-26 17:33:38 -0700248 SerializeConfig(config_value->config, pb_config_value->mutable_config());
Adam Lesinski5bd44232017-09-07 09:39:07 -0700249 pb_config_value->mutable_config()->set_product(config_value->product);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800250
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700251 pb::Value* pb_value = pb_config_value->mutable_value();
252 SerializeSourceToPb(config_value->value->GetSource(), &source_pool,
253 pb_value->mutable_source());
Adam Lesinski5bd44232017-09-07 09:39:07 -0700254 pb_value->set_comment(config_value->value->GetComment());
255 pb_value->set_weak(config_value->value->IsWeak());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800256
Adam Lesinski08535002017-08-04 16:15:17 -0700257 PbSerializerVisitor visitor(&source_pool, pb_value);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700258 config_value->value->Accept(&visitor);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800259 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700260 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800261 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700262 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800263
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700264 SerializeStringPoolToPb(source_pool, pb_table->mutable_source_pool());
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700265 return pb_table;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800266}
267
Adam Lesinski4ffea042017-08-10 15:37:28 -0700268std::unique_ptr<pb::internal::CompiledFile> SerializeCompiledFileToPb(const ResourceFile& file) {
269 auto pb_file = util::make_unique<pb::internal::CompiledFile>();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700270 pb_file->set_resource_name(file.name.ToString());
271 pb_file->set_source_path(file.source.path);
272 SerializeConfig(file.config, pb_file->mutable_config());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800273
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700274 for (const SourcedResourceName& exported : file.exported_symbols) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700275 pb::internal::CompiledFile_Symbol* pb_symbol = pb_file->add_exported_symbol();
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700276 pb_symbol->set_resource_name(exported.name.ToString());
Adam Lesinski4ffea042017-08-10 15:37:28 -0700277 pb_symbol->mutable_source()->set_line_number(exported.line);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700278 }
279 return pb_file;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800280}
281
Adam Lesinski4ffea042017-08-10 15:37:28 -0700282CompiledFileOutputStream::CompiledFileOutputStream(ZeroCopyOutputStream* out) : out_(out) {
283}
Adam Lesinski59e04c62016-02-04 15:59:23 -0800284
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700285void CompiledFileOutputStream::EnsureAlignedWrite() {
Adam Lesinskib58c3ef2017-09-12 17:39:52 -0700286 const int overflow = out_.ByteCount() % 4;
287 if (overflow > 0) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700288 uint32_t zero = 0u;
Adam Lesinskib58c3ef2017-09-12 17:39:52 -0700289 out_.WriteRaw(&zero, 4 - overflow);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700290 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800291}
292
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700293void CompiledFileOutputStream::WriteLittleEndian32(uint32_t val) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700294 EnsureAlignedWrite();
295 out_.WriteLittleEndian32(val);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700296}
297
Adam Lesinski4ffea042017-08-10 15:37:28 -0700298void CompiledFileOutputStream::WriteCompiledFile(const pb::internal::CompiledFile* compiled_file) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700299 EnsureAlignedWrite();
300 out_.WriteLittleEndian64(static_cast<uint64_t>(compiled_file->ByteSize()));
301 compiled_file->SerializeWithCachedSizes(&out_);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700302}
303
304void CompiledFileOutputStream::WriteData(const BigBuffer* buffer) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700305 EnsureAlignedWrite();
306 out_.WriteLittleEndian64(static_cast<uint64_t>(buffer->size()));
307 for (const BigBuffer::Block& block : *buffer) {
308 out_.WriteRaw(block.buffer.get(), block.size);
309 }
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700310}
311
312void CompiledFileOutputStream::WriteData(const void* data, size_t len) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700313 EnsureAlignedWrite();
314 out_.WriteLittleEndian64(static_cast<uint64_t>(len));
315 out_.WriteRaw(data, len);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700316}
317
Adam Lesinski4ffea042017-08-10 15:37:28 -0700318bool CompiledFileOutputStream::HadError() {
319 return out_.HadError();
320}
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700321
322CompiledFileInputStream::CompiledFileInputStream(const void* data, size_t size)
323 : in_(static_cast<const uint8_t*>(data), size) {}
324
325void CompiledFileInputStream::EnsureAlignedRead() {
Adam Lesinskib58c3ef2017-09-12 17:39:52 -0700326 const int overflow = in_.CurrentPosition() % 4;
327 if (overflow > 0) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700328 // Reads are always 4 byte aligned.
Adam Lesinskib58c3ef2017-09-12 17:39:52 -0700329 in_.Skip(4 - overflow);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700330 }
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700331}
332
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700333bool CompiledFileInputStream::ReadLittleEndian32(uint32_t* out_val) {
334 EnsureAlignedRead();
335 return in_.ReadLittleEndian32(out_val);
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700336}
337
Adam Lesinski4ffea042017-08-10 15:37:28 -0700338bool CompiledFileInputStream::ReadCompiledFile(pb::internal::CompiledFile* out_val) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700339 EnsureAlignedRead();
340
Tamas Berghammer383db5eb2016-06-22 15:21:38 +0100341 google::protobuf::uint64 pb_size = 0u;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700342 if (!in_.ReadLittleEndian64(&pb_size)) {
343 return false;
344 }
345
346 CodedInputStream::Limit l = in_.PushLimit(static_cast<int>(pb_size));
347
348 // Check that we haven't tried to read past the end.
349 if (static_cast<uint64_t>(in_.BytesUntilLimit()) != pb_size) {
350 in_.PopLimit(l);
351 in_.PushLimit(0);
352 return false;
353 }
354
355 if (!out_val->ParsePartialFromCodedStream(&in_)) {
356 in_.PopLimit(l);
357 in_.PushLimit(0);
358 return false;
359 }
360
361 in_.PopLimit(l);
362 return true;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700363}
364
Adam Lesinski4ffea042017-08-10 15:37:28 -0700365bool CompiledFileInputStream::ReadDataMetaData(uint64_t* out_offset, uint64_t* out_len) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700366 EnsureAlignedRead();
367
Tamas Berghammer383db5eb2016-06-22 15:21:38 +0100368 google::protobuf::uint64 pb_size = 0u;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700369 if (!in_.ReadLittleEndian64(&pb_size)) {
370 return false;
371 }
372
373 // Check that we aren't trying to read past the end.
374 if (pb_size > static_cast<uint64_t>(in_.BytesUntilLimit())) {
375 in_.PushLimit(0);
376 return false;
377 }
378
379 uint64_t offset = static_cast<uint64_t>(in_.CurrentPosition());
380 if (!in_.Skip(pb_size)) {
381 return false;
382 }
383
384 *out_offset = offset;
385 *out_len = pb_size;
386 return true;
Adam Lesinski5eeaadd2016-08-25 12:26:56 -0700387}
388
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700389} // namespace aapt