We were serializing the wrong Source when writing to protobuf Change-Id: Ib65e962ebc1244ddf4af0e5666aaaebe5f7864a8
320 lines
12 KiB
C++
320 lines
12 KiB
C++
/*
|
|
* Copyright (C) 2016 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "Resource.h"
|
|
#include "ResourceTable.h"
|
|
#include "StringPool.h"
|
|
#include "ValueVisitor.h"
|
|
#include "proto/ProtoHelpers.h"
|
|
#include "proto/ProtoSerialize.h"
|
|
#include "util/BigBuffer.h"
|
|
|
|
namespace aapt {
|
|
|
|
namespace {
|
|
|
|
class PbSerializerVisitor : public RawValueVisitor {
|
|
public:
|
|
using RawValueVisitor::visit;
|
|
|
|
/**
|
|
* Constructor to use when expecting to serialize any value.
|
|
*/
|
|
PbSerializerVisitor(StringPool* sourcePool, StringPool* symbolPool, pb::Value* outPbValue) :
|
|
mSourcePool(sourcePool), mSymbolPool(symbolPool), mOutPbValue(outPbValue),
|
|
mOutPbItem(nullptr) {
|
|
}
|
|
|
|
/**
|
|
* Constructor to use when expecting to serialize an Item.
|
|
*/
|
|
PbSerializerVisitor(StringPool* sourcePool, StringPool* symbolPool, pb::Item* outPbItem) :
|
|
mSourcePool(sourcePool), mSymbolPool(symbolPool), mOutPbValue(nullptr),
|
|
mOutPbItem(outPbItem) {
|
|
}
|
|
|
|
void visit(Reference* ref) override {
|
|
serializeReferenceToPb(*ref, getPbItem()->mutable_ref());
|
|
}
|
|
|
|
void visit(String* str) override {
|
|
getPbItem()->mutable_str()->set_idx(str->value.getIndex());
|
|
}
|
|
|
|
void visit(StyledString* str) override {
|
|
getPbItem()->mutable_str()->set_idx(str->value.getIndex());
|
|
}
|
|
|
|
void visit(FileReference* file) override {
|
|
getPbItem()->mutable_file()->set_path_idx(file->path.getIndex());
|
|
}
|
|
|
|
void visit(Id* id) override {
|
|
getPbItem()->mutable_id();
|
|
}
|
|
|
|
void visit(RawString* rawStr) override {
|
|
getPbItem()->mutable_raw_str()->set_idx(rawStr->value.getIndex());
|
|
}
|
|
|
|
void visit(BinaryPrimitive* prim) override {
|
|
android::Res_value val = {};
|
|
prim->flatten(&val);
|
|
|
|
pb::Primitive* pbPrim = getPbItem()->mutable_prim();
|
|
pbPrim->set_type(val.dataType);
|
|
pbPrim->set_data(val.data);
|
|
}
|
|
|
|
void visitItem(Item* item) override {
|
|
assert(false && "unimplemented item");
|
|
}
|
|
|
|
void visit(Attribute* attr) override {
|
|
pb::Attribute* pbAttr = getPbCompoundValue()->mutable_attr();
|
|
pbAttr->set_format_flags(attr->typeMask);
|
|
pbAttr->set_min_int(attr->minInt);
|
|
pbAttr->set_max_int(attr->maxInt);
|
|
|
|
for (auto& symbol : attr->symbols) {
|
|
pb::Attribute_Symbol* pbSymbol = pbAttr->add_symbols();
|
|
serializeItemCommonToPb(symbol.symbol, pbSymbol);
|
|
serializeReferenceToPb(symbol.symbol, pbSymbol->mutable_name());
|
|
pbSymbol->set_value(symbol.value);
|
|
}
|
|
}
|
|
|
|
void visit(Style* style) override {
|
|
pb::Style* pbStyle = getPbCompoundValue()->mutable_style();
|
|
if (style->parent) {
|
|
serializeReferenceToPb(style->parent.value(), pbStyle->mutable_parent());
|
|
serializeSourceToPb(style->parent.value().getSource(),
|
|
mSourcePool,
|
|
pbStyle->mutable_parent_source());
|
|
}
|
|
|
|
for (Style::Entry& entry : style->entries) {
|
|
pb::Style_Entry* pbEntry = pbStyle->add_entries();
|
|
serializeReferenceToPb(entry.key, pbEntry->mutable_key());
|
|
|
|
pb::Item* pbItem = pbEntry->mutable_item();
|
|
serializeItemCommonToPb(entry.key, pbEntry);
|
|
PbSerializerVisitor subVisitor(mSourcePool, mSymbolPool, pbItem);
|
|
entry.value->accept(&subVisitor);
|
|
}
|
|
}
|
|
|
|
void visit(Styleable* styleable) override {
|
|
pb::Styleable* pbStyleable = getPbCompoundValue()->mutable_styleable();
|
|
for (Reference& entry : styleable->entries) {
|
|
pb::Styleable_Entry* pbEntry = pbStyleable->add_entries();
|
|
serializeItemCommonToPb(entry, pbEntry);
|
|
serializeReferenceToPb(entry, pbEntry->mutable_attr());
|
|
}
|
|
}
|
|
|
|
void visit(Array* array) override {
|
|
pb::Array* pbArray = getPbCompoundValue()->mutable_array();
|
|
for (auto& value : array->items) {
|
|
pb::Array_Entry* pbEntry = pbArray->add_entries();
|
|
serializeItemCommonToPb(*value, pbEntry);
|
|
PbSerializerVisitor subVisitor(mSourcePool, mSymbolPool, pbEntry->mutable_item());
|
|
value->accept(&subVisitor);
|
|
}
|
|
}
|
|
|
|
void visit(Plural* plural) override {
|
|
pb::Plural* pbPlural = getPbCompoundValue()->mutable_plural();
|
|
const size_t count = plural->values.size();
|
|
for (size_t i = 0; i < count; i++) {
|
|
if (!plural->values[i]) {
|
|
// No plural value set here.
|
|
continue;
|
|
}
|
|
|
|
pb::Plural_Entry* pbEntry = pbPlural->add_entries();
|
|
pbEntry->set_arity(serializePluralEnumToPb(i));
|
|
pb::Item* pbElement = pbEntry->mutable_item();
|
|
serializeItemCommonToPb(*plural->values[i], pbEntry);
|
|
PbSerializerVisitor subVisitor(mSourcePool, mSymbolPool, pbElement);
|
|
plural->values[i]->accept(&subVisitor);
|
|
}
|
|
}
|
|
|
|
private:
|
|
pb::Item* getPbItem() {
|
|
if (mOutPbValue) {
|
|
return mOutPbValue->mutable_item();
|
|
}
|
|
return mOutPbItem;
|
|
}
|
|
|
|
pb::CompoundValue* getPbCompoundValue() {
|
|
assert(mOutPbValue);
|
|
return mOutPbValue->mutable_compound_value();
|
|
}
|
|
|
|
template <typename T>
|
|
void serializeItemCommonToPb(const Item& item, T* pbItem) {
|
|
serializeSourceToPb(item.getSource(), mSourcePool, pbItem->mutable_source());
|
|
if (!item.getComment().empty()) {
|
|
pbItem->set_comment(util::utf16ToUtf8(item.getComment()));
|
|
}
|
|
}
|
|
|
|
void serializeReferenceToPb(const Reference& ref, pb::Reference* pbRef) {
|
|
if (ref.id) {
|
|
pbRef->set_id(ref.id.value().id);
|
|
} else if (ref.name) {
|
|
StringPool::Ref symbolRef = mSymbolPool->makeRef(ref.name.value().toString());
|
|
pbRef->set_symbol_idx(static_cast<uint32_t>(symbolRef.getIndex()));
|
|
}
|
|
pbRef->set_private_(ref.privateReference);
|
|
pbRef->set_type(serializeReferenceTypeToPb(ref.referenceType));
|
|
}
|
|
|
|
StringPool* mSourcePool;
|
|
StringPool* mSymbolPool;
|
|
pb::Value* mOutPbValue;
|
|
pb::Item* mOutPbItem;
|
|
};
|
|
|
|
} // namespace
|
|
|
|
std::unique_ptr<pb::ResourceTable> serializeTableToPb(ResourceTable* table) {
|
|
// We must do this before writing the resources, since the string pool IDs may change.
|
|
table->stringPool.sort([](const StringPool::Entry& a, const StringPool::Entry& b) -> bool {
|
|
int diff = a.context.priority - b.context.priority;
|
|
if (diff < 0) return true;
|
|
if (diff > 0) return false;
|
|
diff = a.context.config.compare(b.context.config);
|
|
if (diff < 0) return true;
|
|
if (diff > 0) return false;
|
|
return a.value < b.value;
|
|
});
|
|
table->stringPool.prune();
|
|
|
|
std::unique_ptr<pb::ResourceTable> pbTable = util::make_unique<pb::ResourceTable>();
|
|
serializeStringPoolToPb(table->stringPool, pbTable->mutable_string_pool());
|
|
|
|
StringPool sourcePool, symbolPool;
|
|
|
|
for (auto& package : table->packages) {
|
|
pb::Package* pbPackage = pbTable->add_packages();
|
|
if (package->id) {
|
|
pbPackage->set_package_id(package->id.value());
|
|
}
|
|
pbPackage->set_package_name(util::utf16ToUtf8(package->name));
|
|
|
|
for (auto& type : package->types) {
|
|
pb::Type* pbType = pbPackage->add_types();
|
|
if (type->id) {
|
|
pbType->set_id(type->id.value());
|
|
}
|
|
pbType->set_name(util::utf16ToUtf8(toString(type->type)));
|
|
|
|
for (auto& entry : type->entries) {
|
|
pb::Entry* pbEntry = pbType->add_entries();
|
|
if (entry->id) {
|
|
pbEntry->set_id(entry->id.value());
|
|
}
|
|
pbEntry->set_name(util::utf16ToUtf8(entry->name));
|
|
|
|
// Write the SymbolStatus struct.
|
|
pb::SymbolStatus* pbStatus = pbEntry->mutable_symbol_status();
|
|
pbStatus->set_visibility(serializeVisibilityToPb(entry->symbolStatus.state));
|
|
serializeSourceToPb(entry->symbolStatus.source, &sourcePool,
|
|
pbStatus->mutable_source());
|
|
pbStatus->set_comment(util::utf16ToUtf8(entry->symbolStatus.comment));
|
|
|
|
for (auto& configValue : entry->values) {
|
|
pb::ConfigValue* pbConfigValue = pbEntry->add_config_values();
|
|
serializeConfig(configValue->config, pbConfigValue->mutable_config());
|
|
if (!configValue->product.empty()) {
|
|
pbConfigValue->mutable_config()->set_product(configValue->product);
|
|
}
|
|
|
|
pb::Value* pbValue = pbConfigValue->mutable_value();
|
|
serializeSourceToPb(configValue->value->getSource(), &sourcePool,
|
|
pbValue->mutable_source());
|
|
if (!configValue->value->getComment().empty()) {
|
|
pbValue->set_comment(util::utf16ToUtf8(configValue->value->getComment()));
|
|
}
|
|
|
|
if (configValue->value->isWeak()) {
|
|
pbValue->set_weak(true);
|
|
}
|
|
|
|
PbSerializerVisitor visitor(&sourcePool, &symbolPool, pbValue);
|
|
configValue->value->accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
serializeStringPoolToPb(sourcePool, pbTable->mutable_source_pool());
|
|
serializeStringPoolToPb(symbolPool, pbTable->mutable_symbol_pool());
|
|
return pbTable;
|
|
}
|
|
|
|
std::unique_ptr<pb::CompiledFile> serializeCompiledFileToPb(const ResourceFile& file) {
|
|
std::unique_ptr<pb::CompiledFile> pbFile = util::make_unique<pb::CompiledFile>();
|
|
pbFile->set_resource_name(util::utf16ToUtf8(file.name.toString()));
|
|
pbFile->set_source_path(file.source.path);
|
|
serializeConfig(file.config, pbFile->mutable_config());
|
|
|
|
for (const SourcedResourceName& exported : file.exportedSymbols) {
|
|
pb::CompiledFile_Symbol* pbSymbol = pbFile->add_exported_symbols();
|
|
pbSymbol->set_resource_name(util::utf16ToUtf8(exported.name.toString()));
|
|
pbSymbol->set_line_no(exported.line);
|
|
}
|
|
return pbFile;
|
|
}
|
|
|
|
CompiledFileOutputStream::CompiledFileOutputStream(google::protobuf::io::ZeroCopyOutputStream* out,
|
|
pb::CompiledFile* pbFile) :
|
|
mOut(out), mPbFile(pbFile) {
|
|
}
|
|
|
|
bool CompiledFileOutputStream::ensureFileWritten() {
|
|
if (mPbFile) {
|
|
const uint64_t pbSize = mPbFile->ByteSize();
|
|
mOut.WriteLittleEndian64(pbSize);
|
|
mPbFile->SerializeWithCachedSizes(&mOut);
|
|
const size_t padding = 4 - (pbSize & 0x03);
|
|
if (padding > 0) {
|
|
uint32_t zero = 0u;
|
|
mOut.WriteRaw(&zero, padding);
|
|
}
|
|
mPbFile = nullptr;
|
|
}
|
|
return !mOut.HadError();
|
|
}
|
|
|
|
bool CompiledFileOutputStream::Write(const void* data, int size) {
|
|
if (!ensureFileWritten()) {
|
|
return false;
|
|
}
|
|
mOut.WriteRaw(data, size);
|
|
return !mOut.HadError();
|
|
}
|
|
|
|
bool CompiledFileOutputStream::Finish() {
|
|
return ensureFileWritten();
|
|
}
|
|
|
|
} // namespace aapt
|