/* * 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 "DisplayTransactionTestHelpers.h" #include "FpsOps.h" namespace android { namespace { class CreateDisplayTest : public DisplayTransactionTest { public: void createDisplayWithRequestedRefreshRate(const String8& name, uint64_t displayId, float pacesetterDisplayRefreshRate, float requestedRefreshRate, float expectedAdjustedRefreshRate) { // -------------------------------------------------------------------- // Call Expectations // -------------------------------------------------------------------- // Invocation sp displayToken = mFlinger.createDisplay(name, false, requestedRefreshRate); // -------------------------------------------------------------------- // Postconditions // The display should have been added to the current state ASSERT_TRUE(hasCurrentDisplayState(displayToken)); const auto& display = getCurrentDisplayState(displayToken); EXPECT_TRUE(display.isVirtual()); EXPECT_EQ(display.requestedRefreshRate, Fps::fromValue(requestedRefreshRate)); EXPECT_EQ(name.string(), display.displayName); std::optional vid = DisplayId::fromValue(displayId | DisplayId::FLAG_VIRTUAL); ASSERT_TRUE(vid.has_value()); sp device = mFlinger.createVirtualDisplayDevice(displayToken, *vid, requestedRefreshRate); EXPECT_TRUE(device->isVirtual()); device->adjustRefreshRate(Fps::fromValue(pacesetterDisplayRefreshRate)); // verifying desired value EXPECT_EQ(device->getAdjustedRefreshRate(), Fps::fromValue(expectedAdjustedRefreshRate)); // verifying rounding up if (requestedRefreshRate < pacesetterDisplayRefreshRate) { EXPECT_GE(device->getAdjustedRefreshRate(), Fps::fromValue(requestedRefreshRate)); } else { EXPECT_EQ(device->getAdjustedRefreshRate(), Fps::fromValue(pacesetterDisplayRefreshRate)); } // -------------------------------------------------------------------- // Cleanup conditions } }; TEST_F(CreateDisplayTest, createDisplaySetsCurrentStateForNonsecureDisplay) { const String8 name("virtual.test"); // -------------------------------------------------------------------- // Call Expectations // -------------------------------------------------------------------- // Invocation sp displayToken = mFlinger.createDisplay(name, false); // -------------------------------------------------------------------- // Postconditions // The display should have been added to the current state ASSERT_TRUE(hasCurrentDisplayState(displayToken)); const auto& display = getCurrentDisplayState(displayToken); EXPECT_TRUE(display.isVirtual()); EXPECT_FALSE(display.isSecure); EXPECT_EQ(name.string(), display.displayName); // -------------------------------------------------------------------- // Cleanup conditions // Creating the display commits a display transaction. EXPECT_CALL(*mFlinger.scheduler(), scheduleFrame()).Times(1); } TEST_F(CreateDisplayTest, createDisplaySetsCurrentStateForSecureDisplay) { const String8 name("virtual.test"); // -------------------------------------------------------------------- // Call Expectations // -------------------------------------------------------------------- // Invocation int64_t oldId = IPCThreadState::self()->clearCallingIdentity(); // Set the calling identity to graphics so captureDisplay with secure is allowed. IPCThreadState::self()->restoreCallingIdentity(static_cast(AID_GRAPHICS) << 32 | AID_GRAPHICS); sp displayToken = mFlinger.createDisplay(name, true); IPCThreadState::self()->restoreCallingIdentity(oldId); // -------------------------------------------------------------------- // Postconditions // The display should have been added to the current state ASSERT_TRUE(hasCurrentDisplayState(displayToken)); const auto& display = getCurrentDisplayState(displayToken); EXPECT_TRUE(display.isVirtual()); EXPECT_TRUE(display.isSecure); EXPECT_EQ(name.string(), display.displayName); // -------------------------------------------------------------------- // Cleanup conditions // Creating the display commits a display transaction. EXPECT_CALL(*mFlinger.scheduler(), scheduleFrame()).Times(1); } // Requesting 0 tells SF not to do anything, i.e., default to refresh as physical displays TEST_F(CreateDisplayTest, createDisplayWithRequestedRefreshRate0) { const String8 displayName("virtual.test"); const uint64_t displayId = 123ull; const float kPacesetterDisplayRefreshRate = 60.f; const float kRequestedRefreshRate = 0.f; const float kExpectedAdjustedRefreshRate = 0.f; createDisplayWithRequestedRefreshRate(displayName, displayId, kPacesetterDisplayRefreshRate, kRequestedRefreshRate, kExpectedAdjustedRefreshRate); } // Requesting negative refresh rate, will be ignored, same as requesting 0 TEST_F(CreateDisplayTest, createDisplayWithRequestedRefreshRateNegative) { const String8 displayName("virtual.test"); const uint64_t displayId = 123ull; const float kPacesetterDisplayRefreshRate = 60.f; const float kRequestedRefreshRate = -60.f; const float kExpectedAdjustedRefreshRate = 0.f; createDisplayWithRequestedRefreshRate(displayName, displayId, kPacesetterDisplayRefreshRate, kRequestedRefreshRate, kExpectedAdjustedRefreshRate); } // Requesting a higher refresh rate than the pacesetter TEST_F(CreateDisplayTest, createDisplayWithRequestedRefreshRateHigh) { const String8 displayName("virtual.test"); const uint64_t displayId = 123ull; const float kPacesetterDisplayRefreshRate = 60.f; const float kRequestedRefreshRate = 90.f; const float kExpectedAdjustedRefreshRate = 60.f; createDisplayWithRequestedRefreshRate(displayName, displayId, kPacesetterDisplayRefreshRate, kRequestedRefreshRate, kExpectedAdjustedRefreshRate); } // Requesting the same refresh rate as the pacesetter TEST_F(CreateDisplayTest, createDisplayWithRequestedRefreshRateSame) { const String8 displayName("virtual.test"); const uint64_t displayId = 123ull; const float kPacesetterDisplayRefreshRate = 60.f; const float kRequestedRefreshRate = 60.f; const float kExpectedAdjustedRefreshRate = 60.f; createDisplayWithRequestedRefreshRate(displayName, displayId, kPacesetterDisplayRefreshRate, kRequestedRefreshRate, kExpectedAdjustedRefreshRate); } // Requesting a divisor (30) of the pacesetter (60) should be honored TEST_F(CreateDisplayTest, createDisplayWithRequestedRefreshRateDivisor) { const String8 displayName("virtual.test"); const uint64_t displayId = 123ull; const float kPacesetterDisplayRefreshRate = 60.f; const float kRequestedRefreshRate = 30.f; const float kExpectedAdjustedRefreshRate = 30.f; createDisplayWithRequestedRefreshRate(displayName, displayId, kPacesetterDisplayRefreshRate, kRequestedRefreshRate, kExpectedAdjustedRefreshRate); } // Requesting a non divisor (45) of the pacesetter (120) should round up to a divisor (60) TEST_F(CreateDisplayTest, createDisplayWithRequestedRefreshRateNoneDivisor) { const String8 displayName("virtual.test"); const uint64_t displayId = 123ull; const float kPacesetterDisplayRefreshRate = 120.f; const float kRequestedRefreshRate = 45.f; const float kExpectedAdjustedRefreshRate = 60.f; createDisplayWithRequestedRefreshRate(displayName, displayId, kPacesetterDisplayRefreshRate, kRequestedRefreshRate, kExpectedAdjustedRefreshRate); } // Requesting a non divisor (75) of the pacesetter (120) should round up to pacesetter (120) TEST_F(CreateDisplayTest, createDisplayWithRequestedRefreshRateNoneDivisorMax) { const String8 displayName("virtual.test"); const uint64_t displayId = 123ull; const float kPacesetterDisplayRefreshRate = 120.f; const float kRequestedRefreshRate = 75.f; const float kExpectedAdjustedRefreshRate = 120.f; createDisplayWithRequestedRefreshRate(displayName, displayId, kPacesetterDisplayRefreshRate, kRequestedRefreshRate, kExpectedAdjustedRefreshRate); } } // namespace } // namespace android