Packages without any resources should not expect to have a DynamicRefTable. Bug:16895517 Bug:17056720 Change-Id: Id006f6bdbf08f30505f6ba5982bc9d1b09db0f0a
215 lines
6.7 KiB
C++
215 lines
6.7 KiB
C++
/*
|
|
* Copyright (C) 2014 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 <androidfw/ResourceTypes.h>
|
|
|
|
#include <utils/String8.h>
|
|
#include <utils/String16.h>
|
|
#include "TestHelpers.h"
|
|
#include "data/basic/R.h"
|
|
#include "data/lib/R.h"
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
using namespace android;
|
|
|
|
namespace {
|
|
|
|
/**
|
|
* Include a binary resource table.
|
|
*
|
|
* Package: com.android.test.basic
|
|
*/
|
|
#include "data/basic/basic_arsc.h"
|
|
|
|
#include "data/lib/lib_arsc.h"
|
|
|
|
enum { MAY_NOT_BE_BAG = false };
|
|
|
|
TEST(ResTableTest, shouldLoadSuccessfully) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
}
|
|
|
|
TEST(ResTableTest, simpleTypeIsRetrievedCorrectly) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
|
|
Res_value val;
|
|
ssize_t block = table.getResource(base::R::string::test1, &val, MAY_NOT_BE_BAG);
|
|
|
|
ASSERT_GE(block, 0);
|
|
ASSERT_EQ(Res_value::TYPE_STRING, val.dataType);
|
|
|
|
const ResStringPool* pool = table.getTableStringBlock(block);
|
|
ASSERT_TRUE(NULL != pool);
|
|
ASSERT_EQ(String8("test1"), pool->string8ObjectAt(val.data));
|
|
}
|
|
|
|
TEST(ResTableTest, resourceNameIsResolved) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
|
|
String16 defPackage("com.android.test.basic");
|
|
String16 testName("@string/test1");
|
|
uint32_t resID = table.identifierForName(testName.string(), testName.size(),
|
|
0, 0,
|
|
defPackage.string(), defPackage.size());
|
|
ASSERT_NE(uint32_t(0x00000000), resID);
|
|
ASSERT_EQ(base::R::string::test1, resID);
|
|
}
|
|
|
|
TEST(ResTableTest, noParentThemeIsAppliedCorrectly) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
|
|
ResTable::Theme theme(table);
|
|
ASSERT_EQ(NO_ERROR, theme.applyStyle(base::R::style::Theme1));
|
|
|
|
Res_value val;
|
|
uint32_t specFlags = 0;
|
|
ssize_t index = theme.getAttribute(base::R::attr::attr1, &val, &specFlags);
|
|
ASSERT_GE(index, 0);
|
|
ASSERT_EQ(Res_value::TYPE_INT_DEC, val.dataType);
|
|
ASSERT_EQ(uint32_t(100), val.data);
|
|
|
|
index = theme.getAttribute(base::R::attr::attr2, &val, &specFlags);
|
|
ASSERT_GE(index, 0);
|
|
ASSERT_EQ(Res_value::TYPE_REFERENCE, val.dataType);
|
|
ASSERT_EQ(base::R::integer::number1, val.data);
|
|
}
|
|
|
|
TEST(ResTableTest, parentThemeIsAppliedCorrectly) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
|
|
ResTable::Theme theme(table);
|
|
ASSERT_EQ(NO_ERROR, theme.applyStyle(base::R::style::Theme2));
|
|
|
|
Res_value val;
|
|
uint32_t specFlags = 0;
|
|
ssize_t index = theme.getAttribute(base::R::attr::attr1, &val, &specFlags);
|
|
ASSERT_GE(index, 0);
|
|
ASSERT_EQ(Res_value::TYPE_INT_DEC, val.dataType);
|
|
ASSERT_EQ(uint32_t(300), val.data);
|
|
|
|
index = theme.getAttribute(base::R::attr::attr2, &val, &specFlags);
|
|
ASSERT_GE(index, 0);
|
|
ASSERT_EQ(Res_value::TYPE_REFERENCE, val.dataType);
|
|
ASSERT_EQ(base::R::integer::number1, val.data);
|
|
}
|
|
|
|
TEST(ResTableTest, libraryThemeIsAppliedCorrectly) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(lib_arsc, lib_arsc_len));
|
|
|
|
ResTable::Theme theme(table);
|
|
ASSERT_EQ(NO_ERROR, theme.applyStyle(lib::R::style::Theme));
|
|
|
|
Res_value val;
|
|
uint32_t specFlags = 0;
|
|
ssize_t index = theme.getAttribute(lib::R::attr::attr1, &val, &specFlags);
|
|
ASSERT_GE(index, 0);
|
|
ASSERT_EQ(Res_value::TYPE_INT_DEC, val.dataType);
|
|
ASSERT_EQ(uint32_t(700), val.data);
|
|
}
|
|
|
|
TEST(ResTableTest, referenceToBagIsNotResolved) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
|
|
Res_value val;
|
|
ssize_t block = table.getResource(base::R::integer::number2, &val, MAY_NOT_BE_BAG);
|
|
ASSERT_GE(block, 0);
|
|
ASSERT_EQ(Res_value::TYPE_REFERENCE, val.dataType);
|
|
ASSERT_EQ(base::R::array::integerArray1, val.data);
|
|
|
|
ssize_t newBlock = table.resolveReference(&val, block);
|
|
EXPECT_EQ(block, newBlock);
|
|
EXPECT_EQ(Res_value::TYPE_REFERENCE, val.dataType);
|
|
EXPECT_EQ(base::R::array::integerArray1, val.data);
|
|
}
|
|
|
|
TEST(ResTableTest, resourcesStillAccessibleAfterParameterChange) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
|
|
Res_value val;
|
|
ssize_t block = table.getResource(base::R::integer::number1, &val, MAY_NOT_BE_BAG);
|
|
ASSERT_GE(block, 0);
|
|
ASSERT_EQ(Res_value::TYPE_INT_DEC, val.dataType);
|
|
|
|
const ResTable::bag_entry* entry;
|
|
ssize_t count = table.lockBag(base::R::array::integerArray1, &entry);
|
|
ASSERT_GE(count, 0);
|
|
table.unlockBag(entry);
|
|
|
|
ResTable_config param;
|
|
memset(¶m, 0, sizeof(param));
|
|
param.density = 320;
|
|
table.setParameters(¶m);
|
|
|
|
block = table.getResource(base::R::integer::number1, &val, MAY_NOT_BE_BAG);
|
|
ASSERT_GE(block, 0);
|
|
ASSERT_EQ(Res_value::TYPE_INT_DEC, val.dataType);
|
|
|
|
count = table.lockBag(base::R::array::integerArray1, &entry);
|
|
ASSERT_GE(count, 0);
|
|
table.unlockBag(entry);
|
|
}
|
|
|
|
TEST(ResTableTest, resourceIsOverridenWithBetterConfig) {
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.add(basic_arsc, basic_arsc_len));
|
|
|
|
Res_value val;
|
|
ssize_t block = table.getResource(base::R::integer::number1, &val, MAY_NOT_BE_BAG);
|
|
ASSERT_GE(block, 0);
|
|
ASSERT_EQ(Res_value::TYPE_INT_DEC, val.dataType);
|
|
ASSERT_EQ(uint32_t(200), val.data);
|
|
|
|
ResTable_config param;
|
|
memset(¶m, 0, sizeof(param));
|
|
param.language[0] = 's';
|
|
param.language[1] = 'v';
|
|
param.country[0] = 'S';
|
|
param.country[1] = 'E';
|
|
table.setParameters(¶m);
|
|
|
|
block = table.getResource(base::R::integer::number1, &val, MAY_NOT_BE_BAG);
|
|
ASSERT_GE(block, 0);
|
|
ASSERT_EQ(Res_value::TYPE_INT_DEC, val.dataType);
|
|
ASSERT_EQ(uint32_t(400), val.data);
|
|
}
|
|
|
|
TEST(ResTableTest, emptyTableHasSensibleDefaults) {
|
|
const int32_t assetCookie = 1;
|
|
|
|
ResTable table;
|
|
ASSERT_EQ(NO_ERROR, table.addEmpty(assetCookie));
|
|
|
|
// Adding an empty table gives us one table!
|
|
ASSERT_EQ(uint32_t(1), table.getTableCount());
|
|
|
|
// Adding an empty table doesn't mean we get packages.
|
|
ASSERT_EQ(uint32_t(0), table.getBasePackageCount());
|
|
|
|
Res_value val;
|
|
ASSERT_LT(table.getResource(base::R::integer::number1, &val, MAY_NOT_BE_BAG), 0);
|
|
}
|
|
|
|
}
|