/* * 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 "androidfw/AttributeResolution.h" #include <array> #include "android-base/file.h" #include "android-base/logging.h" #include "android-base/macros.h" #include "androidfw/AssetManager2.h" #include "androidfw/ResourceUtils.h" #include "TestHelpers.h" #include "data/styles/R.h" using com::android::app::R; namespace android { class AttributeResolutionTest : public ::testing::Test { public: virtual void SetUp() override { styles_assets_ = ApkAssets::Load(GetTestDataPath() + "/styles/styles.apk"); ASSERT_NE(nullptr, styles_assets_); assetmanager_.SetApkAssets({styles_assets_.get()}); } protected: std::unique_ptr<const ApkAssets> styles_assets_; AssetManager2 assetmanager_; }; class AttributeResolutionXmlTest : public AttributeResolutionTest { public: virtual void SetUp() override { AttributeResolutionTest::SetUp(); std::unique_ptr<Asset> asset = assetmanager_.OpenNonAsset("res/layout/layout.xml", Asset::ACCESS_BUFFER); ASSERT_NE(nullptr, asset); ASSERT_EQ(NO_ERROR, xml_parser_.setTo(asset->getBuffer(true), asset->getLength(), true /*copyData*/)); // Skip to the first tag. while (xml_parser_.next() != ResXMLParser::START_TAG) { } } protected: ResXMLTree xml_parser_; }; TEST(AttributeResolutionLibraryTest, ApplyStyleWithDefaultStyleResId) { AssetManager2 assetmanager; auto apk_assets = ApkAssets::LoadAsSharedLibrary(GetTestDataPath() + "/styles/styles.apk"); ASSERT_NE(nullptr, apk_assets); assetmanager.SetApkAssets({apk_assets.get()}); std::unique_ptr<Theme> theme = assetmanager.NewTheme(); std::array<uint32_t, 2> attrs{ {fix_package_id(R::attr::attr_one, 0x02), fix_package_id(R::attr::attr_two, 0x02)}}; std::array<uint32_t, attrs.size() * STYLE_NUM_ENTRIES> values; std::array<uint32_t, attrs.size() + 1> indices; ApplyStyle(theme.get(), nullptr /*xml_parser*/, 0u /*def_style_attr*/, fix_package_id(R::style::StyleOne, 0x02), attrs.data(), attrs.size(), values.data(), indices.data()); const uint32_t public_flag = ResTable_typeSpec::SPEC_PUBLIC; const uint32_t* values_cursor = values.data(); EXPECT_EQ(Res_value::TYPE_INT_DEC, values_cursor[STYLE_TYPE]); EXPECT_EQ(1u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_INT_DEC, values_cursor[STYLE_TYPE]); EXPECT_EQ(2u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); } TEST_F(AttributeResolutionTest, Theme) { std::unique_ptr<Theme> theme = assetmanager_.NewTheme(); ASSERT_TRUE(theme->ApplyStyle(R::style::StyleTwo)); std::array<uint32_t, 5> attrs{{R::attr::attr_one, R::attr::attr_two, R::attr::attr_three, R::attr::attr_four, R::attr::attr_empty}}; std::array<uint32_t, attrs.size() * STYLE_NUM_ENTRIES> values; ASSERT_TRUE(ResolveAttrs(theme.get(), 0u /*def_style_attr*/, 0u /*def_style_res*/, nullptr /*src_values*/, 0 /*src_values_length*/, attrs.data(), attrs.size(), values.data(), nullptr /*out_indices*/)); const uint32_t public_flag = ResTable_typeSpec::SPEC_PUBLIC; const uint32_t* values_cursor = values.data(); EXPECT_EQ(Res_value::TYPE_INT_DEC, values_cursor[STYLE_TYPE]); EXPECT_EQ(1u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_STRING, values_cursor[STYLE_TYPE]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_INT_DEC, values_cursor[STYLE_TYPE]); EXPECT_EQ(3u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_NULL, values_cursor[STYLE_TYPE]); EXPECT_EQ(Res_value::DATA_NULL_UNDEFINED, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); // @empty comes from the theme, so it has the same asset cookie and changing configurations flags // as the theme. values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_NULL, values_cursor[STYLE_TYPE]); EXPECT_EQ(Res_value::DATA_NULL_EMPTY, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); } TEST_F(AttributeResolutionXmlTest, XmlParser) { std::array<uint32_t, 5> attrs{{R::attr::attr_one, R::attr::attr_two, R::attr::attr_three, R::attr::attr_four, R::attr::attr_empty}}; std::array<uint32_t, attrs.size() * STYLE_NUM_ENTRIES> values; ASSERT_TRUE(RetrieveAttributes(&assetmanager_, &xml_parser_, attrs.data(), attrs.size(), values.data(), nullptr /*out_indices*/)); uint32_t* values_cursor = values.data(); EXPECT_EQ(Res_value::TYPE_NULL, values_cursor[STYLE_TYPE]); EXPECT_EQ(Res_value::DATA_NULL_EMPTY, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_NULL, values_cursor[STYLE_TYPE]); EXPECT_EQ(0u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_INT_DEC, values_cursor[STYLE_TYPE]); EXPECT_EQ(10u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_ATTRIBUTE, values_cursor[STYLE_TYPE]); EXPECT_EQ(R::attr::attr_indirect, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_NULL, values_cursor[STYLE_TYPE]); EXPECT_EQ(Res_value::DATA_NULL_UNDEFINED, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); } TEST_F(AttributeResolutionXmlTest, ThemeAndXmlParser) { std::unique_ptr<Theme> theme = assetmanager_.NewTheme(); ASSERT_TRUE(theme->ApplyStyle(R::style::StyleTwo)); std::array<uint32_t, 6> attrs{{R::attr::attr_one, R::attr::attr_two, R::attr::attr_three, R::attr::attr_four, R::attr::attr_five, R::attr::attr_empty}}; std::array<uint32_t, attrs.size() * STYLE_NUM_ENTRIES> values; std::array<uint32_t, attrs.size() + 1> indices; ApplyStyle(theme.get(), &xml_parser_, 0u /*def_style_attr*/, 0u /*def_style_res*/, attrs.data(), attrs.size(), values.data(), indices.data()); const uint32_t public_flag = ResTable_typeSpec::SPEC_PUBLIC; uint32_t* values_cursor = values.data(); EXPECT_EQ(Res_value::TYPE_NULL, values_cursor[STYLE_TYPE]); EXPECT_EQ(Res_value::DATA_NULL_EMPTY, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_STRING, values_cursor[STYLE_TYPE]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_INT_DEC, values_cursor[STYLE_TYPE]); EXPECT_EQ(10u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(uint32_t(-1), values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(0u, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_INT_DEC, values_cursor[STYLE_TYPE]); EXPECT_EQ(3u, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_STRING, values_cursor[STYLE_TYPE]); EXPECT_EQ(R::string::string_one, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); // @empty comes from the theme, so it has the same asset cookie and changing configurations flags // as the theme. values_cursor += STYLE_NUM_ENTRIES; EXPECT_EQ(Res_value::TYPE_NULL, values_cursor[STYLE_TYPE]); EXPECT_EQ(Res_value::DATA_NULL_EMPTY, values_cursor[STYLE_DATA]); EXPECT_EQ(0u, values_cursor[STYLE_RESOURCE_ID]); EXPECT_EQ(1u, values_cursor[STYLE_ASSET_COOKIE]); EXPECT_EQ(0u, values_cursor[STYLE_DENSITY]); EXPECT_EQ(public_flag, values_cursor[STYLE_CHANGING_CONFIGURATIONS]); // The first element of indices contains the number of indices. std::array<uint32_t, 7> expected_indices = {{6u, 0u, 1u, 2u, 3u, 4u, 5u}}; EXPECT_EQ(expected_indices, indices); } } // namespace android