| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 1 | #undef LOG_TAG | 
|  | 2 | #define LOG_TAG "SchedulerUnittests" | 
|  | 3 |  | 
|  | 4 | #include <gmock/gmock.h> | 
|  | 5 | #include <gtest/gtest.h> | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 6 | #include <log/log.h> | 
|  | 7 |  | 
| Ana Krulec | e588e31 | 2018-09-18 12:32:24 -0700 | [diff] [blame] | 8 | #include <mutex> | 
|  | 9 |  | 
| Ana Krulec | e588e31 | 2018-09-18 12:32:24 -0700 | [diff] [blame] | 10 | #include "Scheduler/EventControlThread.h" | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 11 | #include "Scheduler/EventThread.h" | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 12 | #include "Scheduler/RefreshRateConfigs.h" | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 13 | #include "TestableScheduler.h" | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 14 | #include "mock/MockEventThread.h" | 
|  | 15 |  | 
|  | 16 | using testing::_; | 
|  | 17 | using testing::Return; | 
|  | 18 |  | 
|  | 19 | namespace android { | 
|  | 20 |  | 
| Dominik Laskowski | dcb38bb | 2019-01-25 02:35:50 -0800 | [diff] [blame] | 21 | constexpr PhysicalDisplayId PHYSICAL_DISPLAY_ID = 999; | 
|  | 22 |  | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 23 | class SchedulerTest : public testing::Test { | 
|  | 24 | protected: | 
| Ana Krulec | 85c39af | 2018-12-26 17:29:57 -0800 | [diff] [blame] | 25 | class MockEventThreadConnection : public android::EventThreadConnection { | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 26 | public: | 
| Ana Krulec | 85c39af | 2018-12-26 17:29:57 -0800 | [diff] [blame] | 27 | explicit MockEventThreadConnection(EventThread* eventThread) | 
| Ady Abraham | 0f4a1b1 | 2019-06-04 16:04:04 -0700 | [diff] [blame] | 28 | : EventThreadConnection(eventThread, ResyncCallback(), | 
|  | 29 | ISurfaceComposer::eConfigChangedSuppress) {} | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 30 | ~MockEventThreadConnection() = default; | 
|  | 31 |  | 
|  | 32 | MOCK_METHOD1(stealReceiveChannel, status_t(gui::BitTube* outChannel)); | 
|  | 33 | MOCK_METHOD1(setVsyncRate, status_t(uint32_t count)); | 
|  | 34 | MOCK_METHOD0(requestNextVsync, void()); | 
|  | 35 | }; | 
|  | 36 |  | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 37 | SchedulerTest(); | 
|  | 38 | ~SchedulerTest() override; | 
|  | 39 |  | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 40 | std::unique_ptr<scheduler::RefreshRateConfigs> mRefreshRateConfigs; | 
|  | 41 | std::unique_ptr<TestableScheduler> mScheduler; | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 42 |  | 
|  | 43 | Scheduler::ConnectionHandle mConnectionHandle; | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 44 | mock::EventThread* mEventThread; | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 45 | sp<MockEventThreadConnection> mEventThreadConnection; | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 46 | }; | 
|  | 47 |  | 
|  | 48 | SchedulerTest::SchedulerTest() { | 
|  | 49 | const ::testing::TestInfo* const test_info = | 
|  | 50 | ::testing::UnitTest::GetInstance()->current_test_info(); | 
|  | 51 | ALOGD("**** Setting up for %s.%s\n", test_info->test_case_name(), test_info->name()); | 
|  | 52 |  | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 53 | std::vector<scheduler::RefreshRateConfigs::InputConfig> configs{{/*hwcId=*/0, 16666667}}; | 
|  | 54 | mRefreshRateConfigs = | 
|  | 55 | std::make_unique<scheduler::RefreshRateConfigs>(/*refreshRateSwitching=*/false, configs, | 
|  | 56 | /*currentConfig=*/0); | 
|  | 57 |  | 
|  | 58 | mScheduler = std::make_unique<TestableScheduler>(*mRefreshRateConfigs); | 
|  | 59 |  | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 60 | auto eventThread = std::make_unique<mock::EventThread>(); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 61 | mEventThread = eventThread.get(); | 
| Ana Krulec | 85c39af | 2018-12-26 17:29:57 -0800 | [diff] [blame] | 62 | EXPECT_CALL(*mEventThread, registerDisplayEventConnection(_)).WillOnce(Return(0)); | 
|  | 63 |  | 
|  | 64 | mEventThreadConnection = new MockEventThreadConnection(mEventThread); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 65 |  | 
|  | 66 | // createConnection call to scheduler makes a createEventConnection call to EventThread. Make | 
|  | 67 | // sure that call gets executed and returns an EventThread::Connection object. | 
| Ady Abraham | 0f4a1b1 | 2019-06-04 16:04:04 -0700 | [diff] [blame] | 68 | EXPECT_CALL(*mEventThread, createEventConnection(_, _)) | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 69 | .WillRepeatedly(Return(mEventThreadConnection)); | 
|  | 70 |  | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 71 | mConnectionHandle = mScheduler->createConnection(std::move(eventThread)); | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 72 | EXPECT_TRUE(mConnectionHandle); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 73 | } | 
|  | 74 |  | 
|  | 75 | SchedulerTest::~SchedulerTest() { | 
|  | 76 | const ::testing::TestInfo* const test_info = | 
|  | 77 | ::testing::UnitTest::GetInstance()->current_test_info(); | 
|  | 78 | ALOGD("**** Tearing down after %s.%s\n", test_info->test_case_name(), test_info->name()); | 
|  | 79 | } | 
|  | 80 |  | 
|  | 81 | namespace { | 
|  | 82 | /* ------------------------------------------------------------------------ | 
|  | 83 | * Test cases | 
|  | 84 | */ | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 85 |  | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 86 | TEST_F(SchedulerTest, invalidConnectionHandle) { | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 87 | Scheduler::ConnectionHandle handle; | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 88 |  | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 89 | sp<IDisplayEventConnection> connection; | 
| Dominik Laskowski | f654d57 | 2018-12-20 11:03:06 -0800 | [diff] [blame] | 90 | ASSERT_NO_FATAL_FAILURE( | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 91 | connection = mScheduler->createDisplayEventConnection(handle, | 
|  | 92 | ISurfaceComposer:: | 
|  | 93 | eConfigChangedSuppress)); | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 94 | EXPECT_FALSE(connection); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 95 | EXPECT_FALSE(mScheduler->getEventConnection(handle)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 96 |  | 
|  | 97 | // The EXPECT_CALLS make sure we don't call the functions on the subsequent event threads. | 
|  | 98 | EXPECT_CALL(*mEventThread, onHotplugReceived(_, _)).Times(0); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 99 | ASSERT_NO_FATAL_FAILURE(mScheduler->onHotplugReceived(handle, PHYSICAL_DISPLAY_ID, false)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 100 |  | 
|  | 101 | EXPECT_CALL(*mEventThread, onScreenAcquired()).Times(0); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 102 | ASSERT_NO_FATAL_FAILURE(mScheduler->onScreenAcquired(handle)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 103 |  | 
|  | 104 | EXPECT_CALL(*mEventThread, onScreenReleased()).Times(0); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 105 | ASSERT_NO_FATAL_FAILURE(mScheduler->onScreenReleased(handle)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 106 |  | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 107 | std::string output; | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 108 | EXPECT_CALL(*mEventThread, dump(_)).Times(0); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 109 | ASSERT_NO_FATAL_FAILURE(mScheduler->dump(handle, output)); | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 110 | EXPECT_TRUE(output.empty()); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 111 |  | 
|  | 112 | EXPECT_CALL(*mEventThread, setPhaseOffset(_)).Times(0); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 113 | ASSERT_NO_FATAL_FAILURE(mScheduler->setPhaseOffset(handle, 10)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 114 | } | 
|  | 115 |  | 
|  | 116 | TEST_F(SchedulerTest, validConnectionHandle) { | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 117 | sp<IDisplayEventConnection> connection; | 
| Dominik Laskowski | f654d57 | 2018-12-20 11:03:06 -0800 | [diff] [blame] | 118 | ASSERT_NO_FATAL_FAILURE( | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 119 | connection = mScheduler->createDisplayEventConnection(mConnectionHandle, | 
|  | 120 | ISurfaceComposer:: | 
|  | 121 | eConfigChangedSuppress)); | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 122 | ASSERT_EQ(mEventThreadConnection, connection); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 123 | EXPECT_TRUE(mScheduler->getEventConnection(mConnectionHandle)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 124 |  | 
| Dominik Laskowski | dcb38bb | 2019-01-25 02:35:50 -0800 | [diff] [blame] | 125 | EXPECT_CALL(*mEventThread, onHotplugReceived(PHYSICAL_DISPLAY_ID, false)).Times(1); | 
|  | 126 | ASSERT_NO_FATAL_FAILURE( | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 127 | mScheduler->onHotplugReceived(mConnectionHandle, PHYSICAL_DISPLAY_ID, false)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 128 |  | 
|  | 129 | EXPECT_CALL(*mEventThread, onScreenAcquired()).Times(1); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 130 | ASSERT_NO_FATAL_FAILURE(mScheduler->onScreenAcquired(mConnectionHandle)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 131 |  | 
|  | 132 | EXPECT_CALL(*mEventThread, onScreenReleased()).Times(1); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 133 | ASSERT_NO_FATAL_FAILURE(mScheduler->onScreenReleased(mConnectionHandle)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 134 |  | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 135 | std::string output("dump"); | 
|  | 136 | EXPECT_CALL(*mEventThread, dump(output)).Times(1); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 137 | ASSERT_NO_FATAL_FAILURE(mScheduler->dump(mConnectionHandle, output)); | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 138 | EXPECT_FALSE(output.empty()); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 139 |  | 
|  | 140 | EXPECT_CALL(*mEventThread, setPhaseOffset(10)).Times(1); | 
| Steven Thomas | 2bbaabe | 2019-08-28 16:08:35 -0700 | [diff] [blame] | 141 | ASSERT_NO_FATAL_FAILURE(mScheduler->setPhaseOffset(mConnectionHandle, 10)); | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 142 | } | 
| Dominik Laskowski | 9804183 | 2019-08-01 18:35:59 -0700 | [diff] [blame] | 143 |  | 
| Ana Krulec | 0c8cd52 | 2018-08-31 12:27:28 -0700 | [diff] [blame] | 144 | } // namespace | 
| Ana Krulec | 3084c05 | 2018-11-21 20:27:17 +0100 | [diff] [blame] | 145 | } // namespace android |