/* * Copyright 2020 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. */ #undef LOG_TAG #define LOG_TAG "LibSurfaceFlingerUnittests" #include #include #include // TODO(b/129481165): remove the #pragma below and fix conversion issues #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wconversion" #include "BufferQueueLayer.h" #include "BufferStateLayer.h" #include "EffectLayer.h" #include "Layer.h" // TODO(b/129481165): remove the #pragma below and fix conversion issues #pragma clang diagnostic pop // ignored "-Wconversion" #include "TestableSurfaceFlinger.h" #include "mock/DisplayHardware/MockComposer.h" #include "mock/MockDispSync.h" #include "mock/MockEventControlThread.h" #include "mock/MockEventThread.h" #include "mock/MockMessageQueue.h" namespace android { using testing::_; using testing::DoAll; using testing::Mock; using testing::Return; using testing::SetArgPointee; using android::Hwc2::IComposer; using android::Hwc2::IComposerClient; using FakeHwcDisplayInjector = TestableSurfaceFlinger::FakeHwcDisplayInjector; using FrameRate = Layer::FrameRate; using FrameRateCompatibility = Layer::FrameRateCompatibility; class LayerFactory { public: virtual ~LayerFactory() = default; virtual std::string name() = 0; virtual sp createLayer(TestableSurfaceFlinger& flinger) = 0; protected: static constexpr uint32_t WIDTH = 100; static constexpr uint32_t HEIGHT = 100; static constexpr uint32_t LAYER_FLAGS = 0; }; class BufferQueueLayerFactory : public LayerFactory { public: std::string name() override { return "BufferQueueLayer"; } sp createLayer(TestableSurfaceFlinger& flinger) override { sp client; LayerCreationArgs args(flinger.flinger(), client, "buffer-queue-layer", WIDTH, HEIGHT, LAYER_FLAGS, LayerMetadata()); return new BufferQueueLayer(args); } }; class BufferStateLayerFactory : public LayerFactory { public: std::string name() override { return "BufferStateLayer"; } sp createLayer(TestableSurfaceFlinger& flinger) override { sp client; LayerCreationArgs args(flinger.flinger(), client, "buffer-queue-layer", WIDTH, HEIGHT, LAYER_FLAGS, LayerMetadata()); return new BufferStateLayer(args); } }; class EffectLayerFactory : public LayerFactory { public: std::string name() override { return "EffectLayer"; } sp createLayer(TestableSurfaceFlinger& flinger) override { sp client; LayerCreationArgs args(flinger.flinger(), client, "color-layer", WIDTH, HEIGHT, LAYER_FLAGS, LayerMetadata()); return new EffectLayer(args); } }; std::string PrintToStringParamName( const ::testing::TestParamInfo>& info) { return info.param->name(); } /** * This class tests the behaviour of Layer::SetFrameRate and Layer::GetFrameRate */ class SetFrameRateTest : public ::testing::TestWithParam> { protected: const FrameRate FRAME_RATE_VOTE1 = FrameRate(67.f, FrameRateCompatibility::Default); const FrameRate FRAME_RATE_VOTE2 = FrameRate(14.f, FrameRateCompatibility::ExactOrMultiple); const FrameRate FRAME_RATE_VOTE3 = FrameRate(99.f, FrameRateCompatibility::NoVote); const FrameRate FRAME_RATE_TREE = FrameRate(0, FrameRateCompatibility::NoVote); const FrameRate FRAME_RATE_NO_VOTE = FrameRate(0, FrameRateCompatibility::Default); SetFrameRateTest(); void setupScheduler(); void setupComposer(uint32_t virtualDisplayCount); void addChild(sp layer, sp child); void removeChild(sp layer, sp child); void reparentChildren(sp layer, sp child); void commitTransaction(); TestableSurfaceFlinger mFlinger; Hwc2::mock::Composer* mComposer = nullptr; mock::MessageQueue* mMessageQueue = new mock::MessageQueue(); std::vector> mLayers; }; SetFrameRateTest::SetFrameRateTest() { const ::testing::TestInfo* const test_info = ::testing::UnitTest::GetInstance()->current_test_info(); ALOGD("**** Setting up for %s.%s\n", test_info->test_case_name(), test_info->name()); mFlinger.mutableUseFrameRateApi() = true; setupScheduler(); setupComposer(0); mFlinger.mutableEventQueue().reset(mMessageQueue); } void SetFrameRateTest::addChild(sp layer, sp child) { layer.get()->addChild(child.get()); } void SetFrameRateTest::removeChild(sp layer, sp child) { layer.get()->removeChild(child.get()); } void SetFrameRateTest::reparentChildren(sp parent, sp newParent) { parent.get()->reparentChildren(newParent); } void SetFrameRateTest::commitTransaction() { for (auto layer : mLayers) { layer.get()->commitTransaction(layer.get()->getCurrentState()); } } void SetFrameRateTest::setupScheduler() { auto eventThread = std::make_unique(); auto sfEventThread = std::make_unique(); EXPECT_CALL(*eventThread, registerDisplayEventConnection(_)); EXPECT_CALL(*eventThread, createEventConnection(_, _)) .WillOnce(Return(new EventThreadConnection(eventThread.get(), ResyncCallback(), ISurfaceComposer::eConfigChangedSuppress))); EXPECT_CALL(*sfEventThread, registerDisplayEventConnection(_)); EXPECT_CALL(*sfEventThread, createEventConnection(_, _)) .WillOnce(Return(new EventThreadConnection(sfEventThread.get(), ResyncCallback(), ISurfaceComposer::eConfigChangedSuppress))); auto primaryDispSync = std::make_unique(); EXPECT_CALL(*primaryDispSync, computeNextRefresh(0, _)).WillRepeatedly(Return(0)); EXPECT_CALL(*primaryDispSync, getPeriod()) .WillRepeatedly(Return(FakeHwcDisplayInjector::DEFAULT_REFRESH_RATE)); EXPECT_CALL(*primaryDispSync, expectedPresentTime(_)).WillRepeatedly(Return(0)); mFlinger.setupScheduler(std::move(primaryDispSync), std::make_unique(), std::move(eventThread), std::move(sfEventThread)); } void SetFrameRateTest::setupComposer(uint32_t virtualDisplayCount) { mComposer = new Hwc2::mock::Composer(); EXPECT_CALL(*mComposer, getMaxVirtualDisplayCount()).WillOnce(Return(virtualDisplayCount)); mFlinger.setupComposer(std::unique_ptr(mComposer)); Mock::VerifyAndClear(mComposer); } namespace { /* ------------------------------------------------------------------------ * Test cases */ TEST_P(SetFrameRateTest, SetAndGet) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto layer = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); layer->setFrameRate(FRAME_RATE_VOTE1); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE1, layer->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetParent) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); addChild(child1, child2); child2->setFrameRate(FRAME_RATE_VOTE1); commitTransaction(); EXPECT_EQ(FRAME_RATE_TREE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); child2->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetParentAllVote) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); addChild(child1, child2); child2->setFrameRate(FRAME_RATE_VOTE1); child1->setFrameRate(FRAME_RATE_VOTE2); parent->setFrameRate(FRAME_RATE_VOTE3); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE3, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE2, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); child2->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE3, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE2, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child2->getFrameRateForLayerTree()); child1->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE3, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child2->getFrameRateForLayerTree()); parent->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetChild) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); addChild(child1, child2); parent->setFrameRate(FRAME_RATE_VOTE1); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE1, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child2->getFrameRateForLayerTree()); parent->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetChildAllVote) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); addChild(child1, child2); child2->setFrameRate(FRAME_RATE_VOTE1); child1->setFrameRate(FRAME_RATE_VOTE2); parent->setFrameRate(FRAME_RATE_VOTE3); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE3, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE2, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); parent->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_TREE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE2, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); child1->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_TREE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); child2->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetChildAddAfterVote) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); parent->setFrameRate(FRAME_RATE_VOTE1); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE1, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); addChild(child1, child2); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE1, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child2->getFrameRateForLayerTree()); parent->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetChildRemoveAfterVote) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); addChild(child1, child2); parent->setFrameRate(FRAME_RATE_VOTE1); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE1, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child2->getFrameRateForLayerTree()); removeChild(child1, child2); commitTransaction(); EXPECT_EQ(FRAME_RATE_VOTE1, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); parent->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetParentNotInTree) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2_1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); addChild(child1, child2); addChild(child1, child2_1); child2->setFrameRate(FRAME_RATE_VOTE1); commitTransaction(); EXPECT_EQ(FRAME_RATE_TREE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2_1->getFrameRateForLayerTree()); child2->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2_1->getFrameRateForLayerTree()); } TEST_P(SetFrameRateTest, SetAndGetRearentChildren) { EXPECT_CALL(*mMessageQueue, invalidate()).Times(1); const auto& layerFactory = GetParam(); auto parent = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto parent2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child1 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); auto child2 = mLayers.emplace_back(layerFactory->createLayer(mFlinger)); addChild(parent, child1); addChild(child1, child2); child2->setFrameRate(FRAME_RATE_VOTE1); commitTransaction(); EXPECT_EQ(FRAME_RATE_TREE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent2->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); reparentChildren(parent, parent2); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, parent2->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_TREE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_VOTE1, child2->getFrameRateForLayerTree()); child2->setFrameRate(FRAME_RATE_NO_VOTE); commitTransaction(); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, parent2->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child1->getFrameRateForLayerTree()); EXPECT_EQ(FRAME_RATE_NO_VOTE, child2->getFrameRateForLayerTree()); } INSTANTIATE_TEST_SUITE_P(PerLayerType, SetFrameRateTest, testing::Values(std::make_shared(), std::make_shared(), std::make_shared()), PrintToStringParamName); } // namespace } // namespace android