android_frameworks_base/tools/aapt2/flatten/TableFlattener_test.cpp
Adam Lesinski 59e04c6f92 AAPT2: Switch to protobuf for intermediate format
Without needing to conform to the runtime data format,
it is much easier to add new features such as debugging symbols
and carrying over product data to link time.

This also simplifies the runtime format parser and serializer,
which will change much less frequently than the protobuf intermediate
format.

Change-Id: I209787bbf087db0a58a534cb8511c51d21133e00
2016-02-09 19:59:17 +00:00

232 lines
9.6 KiB
C++

/*
* Copyright (C) 2015 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 "flatten/TableFlattener.h"
#include "test/Builders.h"
#include "test/Context.h"
#include "unflatten/BinaryResourceParser.h"
#include "util/Util.h"
#include <gtest/gtest.h>
using namespace android;
namespace aapt {
class TableFlattenerTest : public ::testing::Test {
public:
void SetUp() override {
mContext = test::ContextBuilder()
.setCompilationPackage(u"com.app.test")
.setPackageId(0x7f)
.build();
}
::testing::AssertionResult flatten(ResourceTable* table, ResTable* outTable) {
BigBuffer buffer(1024);
TableFlattener flattener(&buffer);
if (!flattener.consume(mContext.get(), table)) {
return ::testing::AssertionFailure() << "failed to flatten ResourceTable";
}
std::unique_ptr<uint8_t[]> data = util::copy(buffer);
if (outTable->add(data.get(), buffer.size(), -1, true) != NO_ERROR) {
return ::testing::AssertionFailure() << "flattened ResTable is corrupt";
}
return ::testing::AssertionSuccess();
}
::testing::AssertionResult flatten(ResourceTable* table, ResourceTable* outTable) {
BigBuffer buffer(1024);
TableFlattener flattener(&buffer);
if (!flattener.consume(mContext.get(), table)) {
return ::testing::AssertionFailure() << "failed to flatten ResourceTable";
}
std::unique_ptr<uint8_t[]> data = util::copy(buffer);
BinaryResourceParser parser(mContext.get(), outTable, {}, data.get(), buffer.size());
if (!parser.parse()) {
return ::testing::AssertionFailure() << "flattened ResTable is corrupt";
}
return ::testing::AssertionSuccess();
}
::testing::AssertionResult exists(ResTable* table,
const StringPiece16& expectedName,
const ResourceId expectedId,
const ConfigDescription& expectedConfig,
const uint8_t expectedDataType, const uint32_t expectedData,
const uint32_t expectedSpecFlags) {
const ResourceName expectedResName = test::parseNameOrDie(expectedName);
table->setParameters(&expectedConfig);
ResTable_config config;
Res_value val;
uint32_t specFlags;
if (table->getResource(expectedId.id, &val, false, 0, &specFlags, &config) < 0) {
return ::testing::AssertionFailure() << "could not find resource with";
}
if (expectedDataType != val.dataType) {
return ::testing::AssertionFailure()
<< "expected data type "
<< std::hex << (int) expectedDataType << " but got data type "
<< (int) val.dataType << std::dec << " instead";
}
if (expectedData != val.data) {
return ::testing::AssertionFailure()
<< "expected data "
<< std::hex << expectedData << " but got data "
<< val.data << std::dec << " instead";
}
if (expectedSpecFlags != specFlags) {
return ::testing::AssertionFailure()
<< "expected specFlags "
<< std::hex << expectedSpecFlags << " but got specFlags "
<< specFlags << std::dec << " instead";
}
ResTable::resource_name actualName;
if (!table->getResourceName(expectedId.id, false, &actualName)) {
return ::testing::AssertionFailure() << "failed to find resource name";
}
StringPiece16 package16(actualName.package, actualName.packageLen);
if (package16 != expectedResName.package) {
return ::testing::AssertionFailure()
<< "expected package '" << expectedResName.package << "' but got '"
<< package16 << "'";
}
StringPiece16 type16(actualName.type, actualName.typeLen);
if (type16 != toString(expectedResName.type)) {
return ::testing::AssertionFailure()
<< "expected type '" << expectedResName.type
<< "' but got '" << type16 << "'";
}
StringPiece16 name16(actualName.name, actualName.nameLen);
if (name16 != expectedResName.entry) {
return ::testing::AssertionFailure()
<< "expected name '" << expectedResName.entry
<< "' but got '" << name16 << "'";
}
if (expectedConfig != config) {
return ::testing::AssertionFailure()
<< "expected config '" << expectedConfig << "' but got '"
<< ConfigDescription(config) << "'";
}
return ::testing::AssertionSuccess();
}
private:
std::unique_ptr<IAaptContext> mContext;
};
TEST_F(TableFlattenerTest, FlattenFullyLinkedTable) {
std::unique_ptr<ResourceTable> table = test::ResourceTableBuilder()
.setPackageId(u"com.app.test", 0x7f)
.addSimple(u"@com.app.test:id/one", ResourceId(0x7f020000))
.addSimple(u"@com.app.test:id/two", ResourceId(0x7f020001))
.addValue(u"@com.app.test:id/three", ResourceId(0x7f020002),
test::buildReference(u"@com.app.test:id/one", ResourceId(0x7f020000)))
.addValue(u"@com.app.test:integer/one", ResourceId(0x7f030000),
util::make_unique<BinaryPrimitive>(uint8_t(Res_value::TYPE_INT_DEC), 1u))
.addValue(u"@com.app.test:integer/one", ResourceId(0x7f030000),
test::parseConfigOrDie("v1"),
util::make_unique<BinaryPrimitive>(uint8_t(Res_value::TYPE_INT_DEC), 2u))
.addString(u"@com.app.test:string/test", ResourceId(0x7f040000), u"foo")
.addString(u"@com.app.test:layout/bar", ResourceId(0x7f050000), u"res/layout/bar.xml")
.build();
ResTable resTable;
ASSERT_TRUE(flatten(table.get(), &resTable));
EXPECT_TRUE(exists(&resTable, u"@com.app.test:id/one", ResourceId(0x7f020000), {},
Res_value::TYPE_INT_BOOLEAN, 0u, 0u));
EXPECT_TRUE(exists(&resTable, u"@com.app.test:id/two", ResourceId(0x7f020001), {},
Res_value::TYPE_INT_BOOLEAN, 0u, 0u));
EXPECT_TRUE(exists(&resTable, u"@com.app.test:id/three", ResourceId(0x7f020002), {},
Res_value::TYPE_REFERENCE, 0x7f020000u, 0u));
EXPECT_TRUE(exists(&resTable, u"@com.app.test:integer/one", ResourceId(0x7f030000),
{}, Res_value::TYPE_INT_DEC, 1u,
ResTable_config::CONFIG_VERSION));
EXPECT_TRUE(exists(&resTable, u"@com.app.test:integer/one", ResourceId(0x7f030000),
test::parseConfigOrDie("v1"), Res_value::TYPE_INT_DEC, 2u,
ResTable_config::CONFIG_VERSION));
StringPiece16 fooStr = u"foo";
ssize_t idx = resTable.getTableStringBlock(0)->indexOfString(fooStr.data(), fooStr.size());
ASSERT_GE(idx, 0);
EXPECT_TRUE(exists(&resTable, u"@com.app.test:string/test", ResourceId(0x7f040000),
{}, Res_value::TYPE_STRING, (uint32_t) idx, 0u));
StringPiece16 barPath = u"res/layout/bar.xml";
idx = resTable.getTableStringBlock(0)->indexOfString(barPath.data(), barPath.size());
ASSERT_GE(idx, 0);
EXPECT_TRUE(exists(&resTable, u"@com.app.test:layout/bar", ResourceId(0x7f050000), {},
Res_value::TYPE_STRING, (uint32_t) idx, 0u));
}
TEST_F(TableFlattenerTest, FlattenEntriesWithGapsInIds) {
std::unique_ptr<ResourceTable> table = test::ResourceTableBuilder()
.setPackageId(u"com.app.test", 0x7f)
.addSimple(u"@com.app.test:id/one", ResourceId(0x7f020001))
.addSimple(u"@com.app.test:id/three", ResourceId(0x7f020003))
.build();
ResTable resTable;
ASSERT_TRUE(flatten(table.get(), &resTable));
EXPECT_TRUE(exists(&resTable, u"@com.app.test:id/one", ResourceId(0x7f020001), {},
Res_value::TYPE_INT_BOOLEAN, 0u, 0u));
EXPECT_TRUE(exists(&resTable, u"@com.app.test:id/three", ResourceId(0x7f020003), {},
Res_value::TYPE_INT_BOOLEAN, 0u, 0u));
}
TEST_F(TableFlattenerTest, FlattenMinMaxAttributes) {
Attribute attr(false);
attr.typeMask = android::ResTable_map::TYPE_INTEGER;
attr.minInt = 10;
attr.maxInt = 23;
std::unique_ptr<ResourceTable> table = test::ResourceTableBuilder()
.setPackageId(u"android", 0x01)
.addValue(u"@android:attr/foo", ResourceId(0x01010000),
util::make_unique<Attribute>(attr))
.build();
ResourceTable result;
ASSERT_TRUE(flatten(table.get(), &result));
Attribute* actualAttr = test::getValue<Attribute>(&result, u"@android:attr/foo");
ASSERT_NE(nullptr, actualAttr);
EXPECT_EQ(attr.isWeak(), actualAttr->isWeak());
EXPECT_EQ(attr.typeMask, actualAttr->typeMask);
EXPECT_EQ(attr.minInt, actualAttr->minInt);
EXPECT_EQ(attr.maxInt, actualAttr->maxInt);
}
} // namespace aapt