| /* |
| * 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 |