blob: 1cc9ba40bdca4beaed7ed30b70eeda6e28c7105b [file] [log] [blame]
/*
* 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 <scheduler/Fps.h>
#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<IBinder> 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<VirtualDisplayId> vid =
DisplayId::fromValue<VirtualDisplayId>(displayId | DisplayId::FLAG_VIRTUAL);
ASSERT_TRUE(vid.has_value());
sp<DisplayDevice> 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<IBinder> 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<int64_t>(AID_GRAPHICS) << 32 |
AID_GRAPHICS);
sp<IBinder> 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