blob: 4a2273182eb4eefc0b1282a27044ac9f7461ade8 [file] [log] [blame]
Lloyd Pique4603f3c2020-02-11 12:06:56 -08001/*
2 * Copyright 2020 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17// TODO(b/129481165): remove the #pragma below and fix conversion issues
18#pragma clang diagnostic push
19#pragma clang diagnostic ignored "-Wconversion"
20
21#undef LOG_TAG
22#define LOG_TAG "LibSurfaceFlingerUnittests"
23
24#include <vector>
25
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070026// StrictMock<T> derives from T and is not marked final, so the destructor of T is expected to be
27// virtual in case StrictMock<T> is used as a polymorphic base class. That is not the case here.
28#pragma clang diagnostic push
29#pragma clang diagnostic ignored "-Wnon-virtual-dtor"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080030#include <gmock/gmock.h>
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070031#pragma clang diagnostic pop
32
Alec Mouri9b133ca2023-11-14 19:00:01 +000033#include <common/FlagManager.h>
Lloyd Pique4603f3c2020-02-11 12:06:56 -080034#include <gui/LayerMetadata.h>
35#include <log/log.h>
ramindanib90711b2023-10-02 15:13:19 -070036#include <chrono>
Lloyd Pique4603f3c2020-02-11 12:06:56 -080037
Ady Abraham3db8a3c2023-11-20 17:53:47 -080038#include <common/test/FlagUtils.h>
Marin Shalamanov23c44202020-12-22 19:09:20 +010039#include "DisplayHardware/DisplayMode.h"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080040#include "DisplayHardware/HWComposer.h"
Marin Shalamanov12c9e5a2021-01-07 00:25:35 +010041#include "DisplayHardware/Hal.h"
Alec Mouriff793872022-01-13 17:45:06 -080042#include "DisplayIdentificationTestHelpers.h"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080043#include "mock/DisplayHardware/MockComposer.h"
Lloyd Piquea516c002021-05-07 14:36:58 -070044#include "mock/DisplayHardware/MockHWC2.h"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080045
Sally Qid57eb0d2023-11-07 16:46:15 -080046#include <com_android_graphics_surfaceflinger_flags.h>
47
Lloyd Pique4603f3c2020-02-11 12:06:56 -080048// TODO(b/129481165): remove the #pragma below and fix conversion issues
49#pragma clang diagnostic pop // ignored "-Wconversion"
50
51namespace android {
Lloyd Pique4603f3c2020-02-11 12:06:56 -080052
Marin Shalamanov6e840172020-12-14 22:13:28 +010053namespace V2_1 = hardware::graphics::composer::V2_1;
54namespace V2_4 = hardware::graphics::composer::V2_4;
Ady Abrahamde549d42022-01-26 19:19:17 -080055namespace aidl = aidl::android::hardware::graphics::composer3;
ramindanib90711b2023-10-02 15:13:19 -070056using namespace std::chrono_literals;
Marin Shalamanov6e840172020-12-14 22:13:28 +010057
58using Hwc2::Config;
Peiyong Line9d809e2020-04-14 13:10:48 -070059
ramindani12bfe6b2023-02-03 13:29:19 -080060using ::aidl::android::hardware::graphics::composer3::RefreshRateChangedDebugData;
ramindani0cd1d8d2023-06-13 13:43:23 -070061using hal::IComposerClient;
Lloyd Pique4603f3c2020-02-11 12:06:56 -080062using ::testing::_;
63using ::testing::DoAll;
Lloyd Pique4603f3c2020-02-11 12:06:56 -080064using ::testing::Return;
65using ::testing::SetArgPointee;
66using ::testing::StrictMock;
67
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070068struct HWComposerTest : testing::Test {
69 using HalError = hardware::graphics::composer::V2_1::Error;
Marin Shalamanov8b196592021-08-09 16:24:42 +020070
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070071 Hwc2::mock::Composer* const mHal = new StrictMock<Hwc2::mock::Composer>();
72 impl::HWComposer mHwc{std::unique_ptr<Hwc2::Composer>(mHal)};
Marin Shalamanov8b196592021-08-09 16:24:42 +020073
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070074 void expectHotplugConnect(hal::HWDisplayId hwcDisplayId) {
75 constexpr uint8_t kPort = 255;
76 EXPECT_CALL(*mHal, getDisplayIdentificationData(hwcDisplayId, _, _))
77 .WillOnce(DoAll(SetArgPointee<1>(kPort),
78 SetArgPointee<2>(getExternalEdid()), Return(HalError::NONE)));
Marin Shalamanov8b196592021-08-09 16:24:42 +020079
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070080 EXPECT_CALL(*mHal, setClientTargetSlotCount(_));
81 EXPECT_CALL(*mHal, setVsyncEnabled(hwcDisplayId, Hwc2::IComposerClient::Vsync::DISABLE));
Leon Scroggins IIIe24d78f2022-09-20 16:38:19 -040082 EXPECT_CALL(*mHal, onHotplugConnect(hwcDisplayId));
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070083 }
ramindanib90711b2023-10-02 15:13:19 -070084
ramindanif8c0f102023-10-09 12:42:57 -070085 void setDisplayData(HalDisplayId displayId, TimePoint lastExpectedPresentTimestamp,
86 Fps lastFrameInterval) {
ramindanib90711b2023-10-02 15:13:19 -070087 ASSERT_TRUE(mHwc.mDisplayData.find(displayId) != mHwc.mDisplayData.end());
88 auto& displayData = mHwc.mDisplayData.at(displayId);
ramindanif8c0f102023-10-09 12:42:57 -070089 std::scoped_lock lock{displayData.expectedPresentLock};
ramindanib90711b2023-10-02 15:13:19 -070090 displayData.lastExpectedPresentTimestamp = lastExpectedPresentTimestamp;
ramindanif8c0f102023-10-09 12:42:57 -070091 displayData.lastFrameInterval = lastFrameInterval;
ramindanib90711b2023-10-02 15:13:19 -070092 }
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070093};
Marin Shalamanov8b196592021-08-09 16:24:42 +020094
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070095TEST_F(HWComposerTest, isHeadless) {
96 ASSERT_TRUE(mHwc.isHeadless());
97
98 constexpr hal::HWDisplayId kHwcDisplayId = 1;
99 expectHotplugConnect(kHwcDisplayId);
100
101 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
Marin Shalamanov8b196592021-08-09 16:24:42 +0200102 ASSERT_TRUE(info);
Marin Shalamanov8b196592021-08-09 16:24:42 +0200103
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700104 ASSERT_FALSE(mHwc.isHeadless());
105
106 mHwc.disconnectDisplay(info->id);
107 ASSERT_TRUE(mHwc.isHeadless());
108}
109
110TEST_F(HWComposerTest, getActiveMode) {
111 // Unknown display.
112 EXPECT_EQ(mHwc.getActiveMode(PhysicalDisplayId::fromPort(0)), std::nullopt);
113
114 constexpr hal::HWDisplayId kHwcDisplayId = 2;
115 expectHotplugConnect(kHwcDisplayId);
116
117 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
118 ASSERT_TRUE(info);
119
120 {
121 // Display is known to SF but not HWC, e.g. the hotplug disconnect is pending.
122 EXPECT_CALL(*mHal, getActiveConfig(kHwcDisplayId, _))
123 .WillOnce(Return(HalError::BAD_DISPLAY));
124
125 EXPECT_EQ(mHwc.getActiveMode(info->id), std::nullopt);
126 }
127 {
128 constexpr hal::HWConfigId kConfigId = 42;
129 EXPECT_CALL(*mHal, getActiveConfig(kHwcDisplayId, _))
130 .WillOnce(DoAll(SetArgPointee<1>(kConfigId), Return(HalError::NONE)));
131
132 EXPECT_EQ(mHwc.getActiveMode(info->id), kConfigId);
133 }
Marin Shalamanov8b196592021-08-09 16:24:42 +0200134}
135
ramindani0cd1d8d2023-06-13 13:43:23 -0700136TEST_F(HWComposerTest, getModesWithLegacyDisplayConfigs) {
137 constexpr hal::HWDisplayId kHwcDisplayId = 2;
138 constexpr hal::HWConfigId kConfigId = 42;
ramindani263a3f12023-07-18 20:44:49 -0700139 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
ramindani0cd1d8d2023-06-13 13:43:23 -0700140
141 expectHotplugConnect(kHwcDisplayId);
142 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
143 ASSERT_TRUE(info);
144
145 EXPECT_CALL(*mHal, getDisplayConfigurationsSupported()).WillRepeatedly(Return(false));
146
147 {
148 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
149 .WillOnce(Return(HalError::BAD_DISPLAY));
ramindani263a3f12023-07-18 20:44:49 -0700150 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
ramindani0cd1d8d2023-06-13 13:43:23 -0700151 }
152 {
153 constexpr int32_t kWidth = 480;
154 constexpr int32_t kHeight = 720;
155 constexpr int32_t kConfigGroup = 1;
156 constexpr int32_t kVsyncPeriod = 16666667;
157
158 EXPECT_CALL(*mHal,
159 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::WIDTH,
160 _))
161 .WillRepeatedly(DoAll(SetArgPointee<3>(kWidth), Return(HalError::NONE)));
162 EXPECT_CALL(*mHal,
163 getDisplayAttribute(kHwcDisplayId, kConfigId,
164 IComposerClient::Attribute::HEIGHT, _))
165 .WillRepeatedly(DoAll(SetArgPointee<3>(kHeight), Return(HalError::NONE)));
166 EXPECT_CALL(*mHal,
167 getDisplayAttribute(kHwcDisplayId, kConfigId,
168 IComposerClient::Attribute::CONFIG_GROUP, _))
169 .WillRepeatedly(DoAll(SetArgPointee<3>(kConfigGroup), Return(HalError::NONE)));
170 EXPECT_CALL(*mHal,
171 getDisplayAttribute(kHwcDisplayId, kConfigId,
172 IComposerClient::Attribute::VSYNC_PERIOD, _))
173 .WillRepeatedly(DoAll(SetArgPointee<3>(kVsyncPeriod), Return(HalError::NONE)));
174
175 // Optional Parameters UNSUPPORTED
176 EXPECT_CALL(*mHal,
177 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
178 _))
179 .WillOnce(Return(HalError::UNSUPPORTED));
180 EXPECT_CALL(*mHal,
181 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
182 _))
183 .WillOnce(Return(HalError::UNSUPPORTED));
184
185 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
186 .WillRepeatedly(DoAll(SetArgPointee<1>(std::vector<hal::HWConfigId>{kConfigId}),
187 Return(HalError::NONE)));
188
ramindani263a3f12023-07-18 20:44:49 -0700189 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700190 EXPECT_EQ(modes.size(), size_t{1});
191 EXPECT_EQ(modes.front().hwcId, kConfigId);
192 EXPECT_EQ(modes.front().width, kWidth);
193 EXPECT_EQ(modes.front().height, kHeight);
194 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
195 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
196 EXPECT_EQ(modes.front().dpiX, -1);
197 EXPECT_EQ(modes.front().dpiY, -1);
198
199 // Optional parameters are supported
200 constexpr int32_t kDpi = 320;
201 EXPECT_CALL(*mHal,
202 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
203 _))
204 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
205 EXPECT_CALL(*mHal,
206 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
207 _))
208 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
209
ramindani263a3f12023-07-18 20:44:49 -0700210 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700211 EXPECT_EQ(modes.size(), size_t{1});
212 EXPECT_EQ(modes.front().hwcId, kConfigId);
213 EXPECT_EQ(modes.front().width, kWidth);
214 EXPECT_EQ(modes.front().height, kHeight);
215 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
216 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
217 // DPI values are scaled by 1000 in the legacy implementation.
218 EXPECT_EQ(modes.front().dpiX, kDpi / 1000.f);
219 EXPECT_EQ(modes.front().dpiY, kDpi / 1000.f);
220 }
221}
222
Sally Qid57eb0d2023-11-07 16:46:15 -0800223TEST_F(HWComposerTest, getModesWithDisplayConfigurations_VRR_OFF) {
224 // if vrr_config is off, getDisplayConfigurationsSupported() is off as well
225 // then getModesWithLegacyDisplayConfigs should be called instead
226 SET_FLAG_FOR_TEST(com::android::graphics::surfaceflinger::flags::vrr_config, false);
227 ASSERT_FALSE(FlagManager::getInstance().vrr_config());
228
229 constexpr hal::HWDisplayId kHwcDisplayId = 2;
230 constexpr hal::HWConfigId kConfigId = 42;
231 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
232
233 expectHotplugConnect(kHwcDisplayId);
234 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
235 ASSERT_TRUE(info);
236
237 EXPECT_CALL(*mHal, getDisplayConfigurationsSupported()).WillRepeatedly(Return(false));
238
239 {
240 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
241 .WillOnce(Return(HalError::BAD_DISPLAY));
242 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
243 }
244 {
245 constexpr int32_t kWidth = 480;
246 constexpr int32_t kHeight = 720;
247 constexpr int32_t kConfigGroup = 1;
248 constexpr int32_t kVsyncPeriod = 16666667;
249
250 EXPECT_CALL(*mHal,
251 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::WIDTH,
252 _))
253 .WillRepeatedly(DoAll(SetArgPointee<3>(kWidth), Return(HalError::NONE)));
254 EXPECT_CALL(*mHal,
255 getDisplayAttribute(kHwcDisplayId, kConfigId,
256 IComposerClient::Attribute::HEIGHT, _))
257 .WillRepeatedly(DoAll(SetArgPointee<3>(kHeight), Return(HalError::NONE)));
258 EXPECT_CALL(*mHal,
259 getDisplayAttribute(kHwcDisplayId, kConfigId,
260 IComposerClient::Attribute::CONFIG_GROUP, _))
261 .WillRepeatedly(DoAll(SetArgPointee<3>(kConfigGroup), Return(HalError::NONE)));
262 EXPECT_CALL(*mHal,
263 getDisplayAttribute(kHwcDisplayId, kConfigId,
264 IComposerClient::Attribute::VSYNC_PERIOD, _))
265 .WillRepeatedly(DoAll(SetArgPointee<3>(kVsyncPeriod), Return(HalError::NONE)));
266
267 // Optional Parameters UNSUPPORTED
268 EXPECT_CALL(*mHal,
269 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
270 _))
271 .WillOnce(Return(HalError::UNSUPPORTED));
272 EXPECT_CALL(*mHal,
273 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
274 _))
275 .WillOnce(Return(HalError::UNSUPPORTED));
276
277 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
278 .WillRepeatedly(DoAll(SetArgPointee<1>(std::vector<hal::HWConfigId>{kConfigId}),
279 Return(HalError::NONE)));
280
281 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
282 EXPECT_EQ(modes.size(), size_t{1});
283 EXPECT_EQ(modes.front().hwcId, kConfigId);
284 EXPECT_EQ(modes.front().width, kWidth);
285 EXPECT_EQ(modes.front().height, kHeight);
286 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
287 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
288 EXPECT_EQ(modes.front().dpiX, -1);
289 EXPECT_EQ(modes.front().dpiY, -1);
290
291 // Optional parameters are supported
292 constexpr int32_t kDpi = 320;
293 EXPECT_CALL(*mHal,
294 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
295 _))
296 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
297 EXPECT_CALL(*mHal,
298 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
299 _))
300 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
301
302 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
303 EXPECT_EQ(modes.size(), size_t{1});
304 EXPECT_EQ(modes.front().hwcId, kConfigId);
305 EXPECT_EQ(modes.front().width, kWidth);
306 EXPECT_EQ(modes.front().height, kHeight);
307 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
308 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
309 // DPI values are scaled by 1000 in the legacy implementation.
310 EXPECT_EQ(modes.front().dpiX, kDpi / 1000.f);
311 EXPECT_EQ(modes.front().dpiY, kDpi / 1000.f);
312 }
313}
314
315TEST_F(HWComposerTest, getModesWithDisplayConfigurations_VRR_ON) {
316 SET_FLAG_FOR_TEST(com::android::graphics::surfaceflinger::flags::vrr_config, true);
317 ASSERT_TRUE(FlagManager::getInstance().vrr_config());
318
ramindani0cd1d8d2023-06-13 13:43:23 -0700319 constexpr hal::HWDisplayId kHwcDisplayId = 2;
320 constexpr hal::HWConfigId kConfigId = 42;
ramindani263a3f12023-07-18 20:44:49 -0700321 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
ramindani0cd1d8d2023-06-13 13:43:23 -0700322 expectHotplugConnect(kHwcDisplayId);
323 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
324 ASSERT_TRUE(info);
325
326 EXPECT_CALL(*mHal, getDisplayConfigurationsSupported()).WillRepeatedly(Return(true));
327
328 {
ramindani263a3f12023-07-18 20:44:49 -0700329 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
ramindani0cd1d8d2023-06-13 13:43:23 -0700330 .WillOnce(Return(HalError::BAD_DISPLAY));
ramindani263a3f12023-07-18 20:44:49 -0700331 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
ramindani0cd1d8d2023-06-13 13:43:23 -0700332 }
333 {
334 constexpr int32_t kWidth = 480;
335 constexpr int32_t kHeight = 720;
336 constexpr int32_t kConfigGroup = 1;
337 constexpr int32_t kVsyncPeriod = 16666667;
ramindanib90711b2023-10-02 15:13:19 -0700338 const hal::VrrConfig vrrConfig =
339 hal::VrrConfig{.minFrameIntervalNs = static_cast<Fps>(120_Hz).getPeriodNsecs(),
340 .notifyExpectedPresentConfig = hal::VrrConfig::
341 NotifyExpectedPresentConfig{.notifyExpectedPresentHeadsUpNs =
342 ms2ns(30),
343 .notifyExpectedPresentTimeoutNs =
344 ms2ns(30)}};
345 hal::DisplayConfiguration displayConfiguration{.configId = kConfigId,
346 .width = kWidth,
347 .height = kHeight,
348 .configGroup = kConfigGroup,
349 .vsyncPeriod = kVsyncPeriod,
350 .vrrConfig = vrrConfig};
ramindani0cd1d8d2023-06-13 13:43:23 -0700351
ramindani263a3f12023-07-18 20:44:49 -0700352 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
353 .WillOnce(DoAll(SetArgPointee<2>(std::vector<hal::DisplayConfiguration>{
ramindani0cd1d8d2023-06-13 13:43:23 -0700354 displayConfiguration}),
355 Return(HalError::NONE)));
356
357 // Optional dpi not supported
ramindani263a3f12023-07-18 20:44:49 -0700358 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700359 EXPECT_EQ(modes.size(), size_t{1});
360 EXPECT_EQ(modes.front().hwcId, kConfigId);
361 EXPECT_EQ(modes.front().width, kWidth);
362 EXPECT_EQ(modes.front().height, kHeight);
363 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
364 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
ramindanib90711b2023-10-02 15:13:19 -0700365 EXPECT_EQ(modes.front().vrrConfig, vrrConfig);
ramindani0cd1d8d2023-06-13 13:43:23 -0700366 EXPECT_EQ(modes.front().dpiX, -1);
367 EXPECT_EQ(modes.front().dpiY, -1);
368
369 // Supports optional dpi parameter
370 constexpr int32_t kDpi = 320;
371 displayConfiguration.dpi = {kDpi, kDpi};
372
ramindani263a3f12023-07-18 20:44:49 -0700373 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
374 .WillOnce(DoAll(SetArgPointee<2>(std::vector<hal::DisplayConfiguration>{
ramindani0cd1d8d2023-06-13 13:43:23 -0700375 displayConfiguration}),
376 Return(HalError::NONE)));
377
ramindani263a3f12023-07-18 20:44:49 -0700378 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700379 EXPECT_EQ(modes.size(), size_t{1});
380 EXPECT_EQ(modes.front().hwcId, kConfigId);
381 EXPECT_EQ(modes.front().width, kWidth);
382 EXPECT_EQ(modes.front().height, kHeight);
383 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
384 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
ramindanib90711b2023-10-02 15:13:19 -0700385 EXPECT_EQ(modes.front().vrrConfig, vrrConfig);
ramindani0cd1d8d2023-06-13 13:43:23 -0700386 EXPECT_EQ(modes.front().dpiX, kDpi);
387 EXPECT_EQ(modes.front().dpiY, kDpi);
388 }
389}
390
Leon Scroggins III959a7ff2023-02-07 11:24:25 -0500391TEST_F(HWComposerTest, onVsync) {
392 constexpr hal::HWDisplayId kHwcDisplayId = 1;
393 expectHotplugConnect(kHwcDisplayId);
394
395 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
396 ASSERT_TRUE(info);
397
398 const auto physicalDisplayId = info->id;
399
400 // Deliberately chosen not to match DisplayData.lastPresentTimestamp's
401 // initial value.
402 constexpr nsecs_t kTimestamp = 1;
403 auto displayIdOpt = mHwc.onVsync(kHwcDisplayId, kTimestamp);
404 ASSERT_TRUE(displayIdOpt);
405 EXPECT_EQ(physicalDisplayId, displayIdOpt);
406
407 // Attempt to send the same time stamp again.
408 displayIdOpt = mHwc.onVsync(kHwcDisplayId, kTimestamp);
409 EXPECT_FALSE(displayIdOpt);
410}
411
412TEST_F(HWComposerTest, onVsyncInvalid) {
413 constexpr hal::HWDisplayId kInvalidHwcDisplayId = 2;
414 constexpr nsecs_t kTimestamp = 1;
415 const auto displayIdOpt = mHwc.onVsync(kInvalidHwcDisplayId, kTimestamp);
416 EXPECT_FALSE(displayIdOpt);
417}
418
ramindanib90711b2023-10-02 15:13:19 -0700419TEST_F(HWComposerTest, notifyExpectedPresentTimeout) {
420 constexpr hal::HWDisplayId kHwcDisplayId = 2;
421 expectHotplugConnect(kHwcDisplayId);
422 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
423 ASSERT_TRUE(info);
424
425 auto expectedPresentTime = systemTime() + ms2ns(10);
ramindanif8c0f102023-10-09 12:42:57 -0700426 static constexpr Fps Fps60Hz = 60_Hz;
427 static constexpr int32_t kFrameInterval5HzNs = static_cast<Fps>(5_Hz).getPeriodNsecs();
428 static constexpr int32_t kFrameInterval60HzNs = Fps60Hz.getPeriodNsecs();
429 static constexpr int32_t kFrameInterval120HzNs = static_cast<Fps>(120_Hz).getPeriodNsecs();
430 static constexpr Period kVsyncPeriod =
431 Period::fromNs(static_cast<Fps>(240_Hz).getPeriodNsecs());
432 static constexpr Period kTimeoutNs = Period::fromNs(kFrameInterval5HzNs);
433 static constexpr auto kLastExpectedPresentTimestamp = TimePoint::fromNs(0);
ramindanib90711b2023-10-02 15:13:19 -0700434
ramindanif8c0f102023-10-09 12:42:57 -0700435 ASSERT_NO_FATAL_FAILURE(setDisplayData(info->id, kLastExpectedPresentTimestamp, Fps60Hz));
ramindanib90711b2023-10-02 15:13:19 -0700436
437 {
438 // Very first ExpectedPresent after idle, no previous timestamp
439 EXPECT_CALL(*mHal,
ramindanif8c0f102023-10-09 12:42:57 -0700440 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
ramindanib90711b2023-10-02 15:13:19 -0700441 .WillOnce(Return(HalError::NONE));
ramindanif8c0f102023-10-09 12:42:57 -0700442 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
443 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700444 kTimeoutNs);
445 }
446 {
ramindanif8c0f102023-10-09 12:42:57 -0700447 // Absent timeoutNs
448 expectedPresentTime += 2 * kFrameInterval5HzNs;
449 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
450 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
451 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
452 /*timeoutOpt*/ std::nullopt);
453 }
454 {
455 // Timeout is 0
456 expectedPresentTime += kFrameInterval60HzNs;
ramindanib90711b2023-10-02 15:13:19 -0700457 EXPECT_CALL(*mHal,
ramindanif8c0f102023-10-09 12:42:57 -0700458 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
ramindanib90711b2023-10-02 15:13:19 -0700459 .WillOnce(Return(HalError::NONE));
ramindanif8c0f102023-10-09 12:42:57 -0700460 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
461 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
462 Period::fromNs(0));
463 }
464 {
465 // ExpectedPresent is after the timeoutNs
466 expectedPresentTime += 2 * kFrameInterval5HzNs;
467 EXPECT_CALL(*mHal,
468 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
469 .WillOnce(Return(HalError::NONE));
470 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
471 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
472 kTimeoutNs);
473 }
474 {
475 // ExpectedPresent has not changed
476 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
477 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
478 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700479 kTimeoutNs);
480 }
481 {
482 // ExpectedPresent is after the last reported ExpectedPresent.
ramindanif8c0f102023-10-09 12:42:57 -0700483 expectedPresentTime += kFrameInterval60HzNs;
ramindanib90711b2023-10-02 15:13:19 -0700484 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
ramindanif8c0f102023-10-09 12:42:57 -0700485 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
486 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700487 kTimeoutNs);
488 }
489 {
490 // ExpectedPresent is before the last reported ExpectedPresent but after the timeoutNs,
491 // representing we changed our decision and want to present earlier than previously
492 // reported.
ramindanif8c0f102023-10-09 12:42:57 -0700493 expectedPresentTime -= kFrameInterval120HzNs;
ramindanib90711b2023-10-02 15:13:19 -0700494 EXPECT_CALL(*mHal,
ramindanif8c0f102023-10-09 12:42:57 -0700495 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
ramindanib90711b2023-10-02 15:13:19 -0700496 .WillOnce(Return(HalError::NONE));
ramindanif8c0f102023-10-09 12:42:57 -0700497 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
498 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700499 kTimeoutNs);
500 }
501}
502
ramindanif8c0f102023-10-09 12:42:57 -0700503TEST_F(HWComposerTest, notifyExpectedPresentRenderRateChanged) {
504 constexpr hal::HWDisplayId kHwcDisplayId = 2;
505 expectHotplugConnect(kHwcDisplayId);
506 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
507 ASSERT_TRUE(info);
508
509 const auto now = systemTime();
510 auto expectedPresentTime = now;
511 static constexpr Period kTimeoutNs = Period::fromNs(static_cast<Fps>(1_Hz).getPeriodNsecs());
512
513 ASSERT_NO_FATAL_FAILURE(setDisplayData(info->id, TimePoint::fromNs(now), Fps::fromValue(0)));
514 static constexpr int32_t kFrameIntervalNs120Hz = static_cast<Fps>(120_Hz).getPeriodNsecs();
515 static constexpr int32_t kFrameIntervalNs96Hz = static_cast<Fps>(96_Hz).getPeriodNsecs();
516 static constexpr int32_t kFrameIntervalNs80Hz = static_cast<Fps>(80_Hz).getPeriodNsecs();
517 static constexpr int32_t kFrameIntervalNs60Hz = static_cast<Fps>(60_Hz).getPeriodNsecs();
518 static constexpr int32_t kFrameIntervalNs40Hz = static_cast<Fps>(40_Hz).getPeriodNsecs();
519 static constexpr int32_t kFrameIntervalNs30Hz = static_cast<Fps>(30_Hz).getPeriodNsecs();
520 static constexpr int32_t kFrameIntervalNs24Hz = static_cast<Fps>(24_Hz).getPeriodNsecs();
521 static constexpr int32_t kFrameIntervalNs20Hz = static_cast<Fps>(20_Hz).getPeriodNsecs();
522 static constexpr Period kVsyncPeriod =
523 Period::fromNs(static_cast<Fps>(240_Hz).getPeriodNsecs());
524
525 struct FrameRateIntervalTestData {
526 int32_t frameIntervalNs;
527 bool callExpectedPresent;
528 };
529 const std::vector<FrameRateIntervalTestData> frameIntervals = {
530 {kFrameIntervalNs60Hz, true}, {kFrameIntervalNs96Hz, true},
531 {kFrameIntervalNs80Hz, true}, {kFrameIntervalNs120Hz, true},
532 {kFrameIntervalNs80Hz, true}, {kFrameIntervalNs60Hz, true},
533 {kFrameIntervalNs60Hz, false}, {kFrameIntervalNs30Hz, false},
534 {kFrameIntervalNs24Hz, true}, {kFrameIntervalNs40Hz, true},
535 {kFrameIntervalNs20Hz, false}, {kFrameIntervalNs60Hz, true},
536 {kFrameIntervalNs20Hz, false}, {kFrameIntervalNs120Hz, true},
537 };
538
539 for (const auto& [frameIntervalNs, callExpectedPresent] : frameIntervals) {
540 {
541 expectedPresentTime += frameIntervalNs;
542 if (callExpectedPresent) {
543 EXPECT_CALL(*mHal,
544 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime,
545 frameIntervalNs))
546 .WillOnce(Return(HalError::NONE));
547 } else {
548 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
549 }
550 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
551 TimePoint::fromNs(expectedPresentTime),
552 Fps::fromPeriodNsecs(frameIntervalNs), kTimeoutNs);
553 }
554 }
555}
556
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700557struct MockHWC2ComposerCallback final : StrictMock<HWC2::ComposerCallback> {
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700558 MOCK_METHOD2(onComposerHalHotplug, void(hal::HWDisplayId, hal::Connection));
559 MOCK_METHOD1(onComposerHalRefresh, void(hal::HWDisplayId));
560 MOCK_METHOD3(onComposerHalVsync,
561 void(hal::HWDisplayId, int64_t timestamp, std::optional<hal::VsyncPeriodNanos>));
562 MOCK_METHOD2(onComposerHalVsyncPeriodTimingChanged,
563 void(hal::HWDisplayId, const hal::VsyncPeriodChangeTimeline&));
564 MOCK_METHOD1(onComposerHalSeamlessPossible, void(hal::HWDisplayId));
Yichi Chen1a417af2022-01-21 15:29:52 +0800565 MOCK_METHOD1(onComposerHalVsyncIdle, void(hal::HWDisplayId));
ramindani12bfe6b2023-02-03 13:29:19 -0800566 MOCK_METHOD(void, onRefreshRateChangedDebug, (const RefreshRateChangedDebugData&), (override));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800567};
568
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700569struct HWComposerSetCallbackTest : HWComposerTest {
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700570 MockHWC2ComposerCallback mCallback;
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800571};
572
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700573TEST_F(HWComposerSetCallbackTest, loadsLayerMetadataSupport) {
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800574 const std::string kMetadata1Name = "com.example.metadata.1";
575 constexpr bool kMetadata1Mandatory = false;
576 const std::string kMetadata2Name = "com.example.metadata.2";
577 constexpr bool kMetadata2Mandatory = true;
578
Ady Abrahamde549d42022-01-26 19:19:17 -0800579 EXPECT_CALL(*mHal, getCapabilities()).WillOnce(Return(std::vector<aidl::Capability>{}));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800580 EXPECT_CALL(*mHal, getLayerGenericMetadataKeys(_))
Peiyong Line9d809e2020-04-14 13:10:48 -0700581 .WillOnce(DoAll(SetArgPointee<0>(std::vector<hal::LayerGenericMetadataKey>{
582 {kMetadata1Name, kMetadata1Mandatory},
583 {kMetadata2Name, kMetadata2Mandatory},
584 }),
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800585 Return(hardware::graphics::composer::V2_4::Error::NONE)));
Sally Qibb866c12022-10-17 11:31:20 -0700586 EXPECT_CALL(*mHal, getOverlaySupport(_)).WillOnce(Return(HalError::NONE));
Kriti Dang674b9372022-11-18 10:58:44 +0100587 EXPECT_CALL(*mHal, getHdrConversionCapabilities(_)).WillOnce(Return(HalError::NONE));
Sally Qibb866c12022-10-17 11:31:20 -0700588
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800589 EXPECT_CALL(*mHal, registerCallback(_));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800590
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700591 mHwc.setCallback(mCallback);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800592
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700593 const auto& supported = mHwc.getSupportedLayerGenericMetadata();
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800594 EXPECT_EQ(2u, supported.size());
595 EXPECT_EQ(1u, supported.count(kMetadata1Name));
596 EXPECT_EQ(kMetadata1Mandatory, supported.find(kMetadata1Name)->second);
597 EXPECT_EQ(1u, supported.count(kMetadata2Name));
598 EXPECT_EQ(kMetadata2Mandatory, supported.find(kMetadata2Name)->second);
599}
600
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700601TEST_F(HWComposerSetCallbackTest, handlesUnsupportedCallToGetLayerGenericMetadataKeys) {
Ady Abrahamde549d42022-01-26 19:19:17 -0800602 EXPECT_CALL(*mHal, getCapabilities()).WillOnce(Return(std::vector<aidl::Capability>{}));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800603 EXPECT_CALL(*mHal, getLayerGenericMetadataKeys(_))
604 .WillOnce(Return(hardware::graphics::composer::V2_4::Error::UNSUPPORTED));
Sally Qibb866c12022-10-17 11:31:20 -0700605 EXPECT_CALL(*mHal, getOverlaySupport(_)).WillOnce(Return(HalError::UNSUPPORTED));
Kriti Dang674b9372022-11-18 10:58:44 +0100606 EXPECT_CALL(*mHal, getHdrConversionCapabilities(_)).WillOnce(Return(HalError::UNSUPPORTED));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800607 EXPECT_CALL(*mHal, registerCallback(_));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800608
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700609 mHwc.setCallback(mCallback);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800610
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700611 const auto& supported = mHwc.getSupportedLayerGenericMetadata();
612 EXPECT_TRUE(supported.empty());
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800613}
614
615struct HWComposerLayerTest : public testing::Test {
Peiyong Line9d809e2020-04-14 13:10:48 -0700616 static constexpr hal::HWDisplayId kDisplayId = static_cast<hal::HWDisplayId>(1001);
617 static constexpr hal::HWLayerId kLayerId = static_cast<hal::HWLayerId>(1002);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800618
Ady Abrahamde549d42022-01-26 19:19:17 -0800619 HWComposerLayerTest(const std::unordered_set<aidl::Capability>& capabilities)
Lloyd Piquea516c002021-05-07 14:36:58 -0700620 : mCapabilies(capabilities) {
621 EXPECT_CALL(mDisplay, getId()).WillRepeatedly(Return(kDisplayId));
622 }
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800623
Lloyd Piquea516c002021-05-07 14:36:58 -0700624 ~HWComposerLayerTest() override {
625 EXPECT_CALL(mDisplay, onLayerDestroyed(kLayerId));
626 EXPECT_CALL(*mHal, destroyLayer(kDisplayId, kLayerId));
627 }
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800628
629 std::unique_ptr<Hwc2::mock::Composer> mHal{new StrictMock<Hwc2::mock::Composer>()};
Ady Abrahamde549d42022-01-26 19:19:17 -0800630 const std::unordered_set<aidl::Capability> mCapabilies;
Lloyd Piquea516c002021-05-07 14:36:58 -0700631 StrictMock<HWC2::mock::Display> mDisplay;
632 HWC2::impl::Layer mLayer{*mHal, mCapabilies, mDisplay, kLayerId};
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800633};
634
635struct HWComposerLayerGenericMetadataTest : public HWComposerLayerTest {
636 static const std::string kLayerGenericMetadata1Name;
637 static constexpr bool kLayerGenericMetadata1Mandatory = false;
638 static const std::vector<uint8_t> kLayerGenericMetadata1Value;
639 static const std::string kLayerGenericMetadata2Name;
640 static constexpr bool kLayerGenericMetadata2Mandatory = true;
641 static const std::vector<uint8_t> kLayerGenericMetadata2Value;
642
643 HWComposerLayerGenericMetadataTest() : HWComposerLayerTest({}) {}
644};
645
646const std::string HWComposerLayerGenericMetadataTest::kLayerGenericMetadata1Name =
647 "com.example.metadata.1";
648
649const std::vector<uint8_t> HWComposerLayerGenericMetadataTest::kLayerGenericMetadata1Value = {1u,
650 2u,
651 3u};
652
653const std::string HWComposerLayerGenericMetadataTest::kLayerGenericMetadata2Name =
654 "com.example.metadata.2";
655
656const std::vector<uint8_t> HWComposerLayerGenericMetadataTest::kLayerGenericMetadata2Value = {45u,
657 67u};
658
659TEST_F(HWComposerLayerGenericMetadataTest, forwardsSupportedMetadata) {
660 EXPECT_CALL(*mHal,
661 setLayerGenericMetadata(kDisplayId, kLayerId, kLayerGenericMetadata1Name,
662 kLayerGenericMetadata1Mandatory,
663 kLayerGenericMetadata1Value))
664 .WillOnce(Return(hardware::graphics::composer::V2_4::Error::NONE));
665 auto result = mLayer.setLayerGenericMetadata(kLayerGenericMetadata1Name,
666 kLayerGenericMetadata1Mandatory,
667 kLayerGenericMetadata1Value);
Peiyong Line9d809e2020-04-14 13:10:48 -0700668 EXPECT_EQ(hal::Error::NONE, result);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800669
670 EXPECT_CALL(*mHal,
671 setLayerGenericMetadata(kDisplayId, kLayerId, kLayerGenericMetadata2Name,
672 kLayerGenericMetadata2Mandatory,
673 kLayerGenericMetadata2Value))
674 .WillOnce(Return(hardware::graphics::composer::V2_4::Error::UNSUPPORTED));
675 result = mLayer.setLayerGenericMetadata(kLayerGenericMetadata2Name,
676 kLayerGenericMetadata2Mandatory,
677 kLayerGenericMetadata2Value);
Peiyong Line9d809e2020-04-14 13:10:48 -0700678 EXPECT_EQ(hal::Error::UNSUPPORTED, result);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800679}
680
Dominik Laskowski3dce4f42021-03-08 20:48:28 -0800681} // namespace android