blob: c5e179fb4cd1a3c2a032b18024d0f1feaa585078 [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
Huihong Luo791bef92023-09-19 21:18:45 -070060using ::aidl::android::hardware::graphics::common::DisplayHotplugEvent;
ramindani12bfe6b2023-02-03 13:29:19 -080061using ::aidl::android::hardware::graphics::composer3::RefreshRateChangedDebugData;
ramindani0cd1d8d2023-06-13 13:43:23 -070062using hal::IComposerClient;
Lloyd Pique4603f3c2020-02-11 12:06:56 -080063using ::testing::_;
64using ::testing::DoAll;
Lloyd Pique4603f3c2020-02-11 12:06:56 -080065using ::testing::Return;
66using ::testing::SetArgPointee;
67using ::testing::StrictMock;
68
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070069struct HWComposerTest : testing::Test {
70 using HalError = hardware::graphics::composer::V2_1::Error;
Marin Shalamanov8b196592021-08-09 16:24:42 +020071
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070072 Hwc2::mock::Composer* const mHal = new StrictMock<Hwc2::mock::Composer>();
73 impl::HWComposer mHwc{std::unique_ptr<Hwc2::Composer>(mHal)};
Marin Shalamanov8b196592021-08-09 16:24:42 +020074
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070075 void expectHotplugConnect(hal::HWDisplayId hwcDisplayId) {
76 constexpr uint8_t kPort = 255;
77 EXPECT_CALL(*mHal, getDisplayIdentificationData(hwcDisplayId, _, _))
78 .WillOnce(DoAll(SetArgPointee<1>(kPort),
79 SetArgPointee<2>(getExternalEdid()), Return(HalError::NONE)));
Marin Shalamanov8b196592021-08-09 16:24:42 +020080
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070081 EXPECT_CALL(*mHal, setClientTargetSlotCount(_));
82 EXPECT_CALL(*mHal, setVsyncEnabled(hwcDisplayId, Hwc2::IComposerClient::Vsync::DISABLE));
Leon Scroggins IIIe24d78f2022-09-20 16:38:19 -040083 EXPECT_CALL(*mHal, onHotplugConnect(hwcDisplayId));
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070084 }
ramindanib90711b2023-10-02 15:13:19 -070085
ramindanif8c0f102023-10-09 12:42:57 -070086 void setDisplayData(HalDisplayId displayId, TimePoint lastExpectedPresentTimestamp,
87 Fps lastFrameInterval) {
ramindanib90711b2023-10-02 15:13:19 -070088 ASSERT_TRUE(mHwc.mDisplayData.find(displayId) != mHwc.mDisplayData.end());
89 auto& displayData = mHwc.mDisplayData.at(displayId);
ramindanif8c0f102023-10-09 12:42:57 -070090 std::scoped_lock lock{displayData.expectedPresentLock};
ramindanib90711b2023-10-02 15:13:19 -070091 displayData.lastExpectedPresentTimestamp = lastExpectedPresentTimestamp;
ramindanif8c0f102023-10-09 12:42:57 -070092 displayData.lastFrameInterval = lastFrameInterval;
ramindanib90711b2023-10-02 15:13:19 -070093 }
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070094};
Marin Shalamanov8b196592021-08-09 16:24:42 +020095
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070096TEST_F(HWComposerTest, isHeadless) {
97 ASSERT_TRUE(mHwc.isHeadless());
98
99 constexpr hal::HWDisplayId kHwcDisplayId = 1;
100 expectHotplugConnect(kHwcDisplayId);
101
102 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
Marin Shalamanov8b196592021-08-09 16:24:42 +0200103 ASSERT_TRUE(info);
Marin Shalamanov8b196592021-08-09 16:24:42 +0200104
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700105 ASSERT_FALSE(mHwc.isHeadless());
106
107 mHwc.disconnectDisplay(info->id);
108 ASSERT_TRUE(mHwc.isHeadless());
109}
110
111TEST_F(HWComposerTest, getActiveMode) {
112 // Unknown display.
113 EXPECT_EQ(mHwc.getActiveMode(PhysicalDisplayId::fromPort(0)), std::nullopt);
114
115 constexpr hal::HWDisplayId kHwcDisplayId = 2;
116 expectHotplugConnect(kHwcDisplayId);
117
118 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
119 ASSERT_TRUE(info);
120
121 {
122 // Display is known to SF but not HWC, e.g. the hotplug disconnect is pending.
123 EXPECT_CALL(*mHal, getActiveConfig(kHwcDisplayId, _))
124 .WillOnce(Return(HalError::BAD_DISPLAY));
125
126 EXPECT_EQ(mHwc.getActiveMode(info->id), std::nullopt);
127 }
128 {
129 constexpr hal::HWConfigId kConfigId = 42;
130 EXPECT_CALL(*mHal, getActiveConfig(kHwcDisplayId, _))
131 .WillOnce(DoAll(SetArgPointee<1>(kConfigId), Return(HalError::NONE)));
132
133 EXPECT_EQ(mHwc.getActiveMode(info->id), kConfigId);
134 }
Marin Shalamanov8b196592021-08-09 16:24:42 +0200135}
136
ramindani0cd1d8d2023-06-13 13:43:23 -0700137TEST_F(HWComposerTest, getModesWithLegacyDisplayConfigs) {
138 constexpr hal::HWDisplayId kHwcDisplayId = 2;
139 constexpr hal::HWConfigId kConfigId = 42;
ramindani263a3f12023-07-18 20:44:49 -0700140 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
ramindani0cd1d8d2023-06-13 13:43:23 -0700141
142 expectHotplugConnect(kHwcDisplayId);
143 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
144 ASSERT_TRUE(info);
145
ramindani19919ff2023-12-07 11:27:06 -0800146 EXPECT_CALL(*mHal, isVrrSupported()).WillRepeatedly(Return(false));
ramindani0cd1d8d2023-06-13 13:43:23 -0700147
148 {
149 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
150 .WillOnce(Return(HalError::BAD_DISPLAY));
ramindani263a3f12023-07-18 20:44:49 -0700151 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
ramindani0cd1d8d2023-06-13 13:43:23 -0700152 }
153 {
154 constexpr int32_t kWidth = 480;
155 constexpr int32_t kHeight = 720;
156 constexpr int32_t kConfigGroup = 1;
157 constexpr int32_t kVsyncPeriod = 16666667;
158
159 EXPECT_CALL(*mHal,
160 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::WIDTH,
161 _))
162 .WillRepeatedly(DoAll(SetArgPointee<3>(kWidth), Return(HalError::NONE)));
163 EXPECT_CALL(*mHal,
164 getDisplayAttribute(kHwcDisplayId, kConfigId,
165 IComposerClient::Attribute::HEIGHT, _))
166 .WillRepeatedly(DoAll(SetArgPointee<3>(kHeight), Return(HalError::NONE)));
167 EXPECT_CALL(*mHal,
168 getDisplayAttribute(kHwcDisplayId, kConfigId,
169 IComposerClient::Attribute::CONFIG_GROUP, _))
170 .WillRepeatedly(DoAll(SetArgPointee<3>(kConfigGroup), Return(HalError::NONE)));
171 EXPECT_CALL(*mHal,
172 getDisplayAttribute(kHwcDisplayId, kConfigId,
173 IComposerClient::Attribute::VSYNC_PERIOD, _))
174 .WillRepeatedly(DoAll(SetArgPointee<3>(kVsyncPeriod), Return(HalError::NONE)));
175
176 // Optional Parameters UNSUPPORTED
177 EXPECT_CALL(*mHal,
178 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
179 _))
180 .WillOnce(Return(HalError::UNSUPPORTED));
181 EXPECT_CALL(*mHal,
182 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
183 _))
184 .WillOnce(Return(HalError::UNSUPPORTED));
185
186 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
187 .WillRepeatedly(DoAll(SetArgPointee<1>(std::vector<hal::HWConfigId>{kConfigId}),
188 Return(HalError::NONE)));
189
ramindani263a3f12023-07-18 20:44:49 -0700190 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700191 EXPECT_EQ(modes.size(), size_t{1});
192 EXPECT_EQ(modes.front().hwcId, kConfigId);
193 EXPECT_EQ(modes.front().width, kWidth);
194 EXPECT_EQ(modes.front().height, kHeight);
195 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
196 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
197 EXPECT_EQ(modes.front().dpiX, -1);
198 EXPECT_EQ(modes.front().dpiY, -1);
199
200 // Optional parameters are supported
201 constexpr int32_t kDpi = 320;
202 EXPECT_CALL(*mHal,
203 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
204 _))
205 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
206 EXPECT_CALL(*mHal,
207 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
208 _))
209 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
210
ramindani263a3f12023-07-18 20:44:49 -0700211 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700212 EXPECT_EQ(modes.size(), size_t{1});
213 EXPECT_EQ(modes.front().hwcId, kConfigId);
214 EXPECT_EQ(modes.front().width, kWidth);
215 EXPECT_EQ(modes.front().height, kHeight);
216 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
217 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
218 // DPI values are scaled by 1000 in the legacy implementation.
219 EXPECT_EQ(modes.front().dpiX, kDpi / 1000.f);
220 EXPECT_EQ(modes.front().dpiY, kDpi / 1000.f);
221 }
222}
223
Sally Qid57eb0d2023-11-07 16:46:15 -0800224TEST_F(HWComposerTest, getModesWithDisplayConfigurations_VRR_OFF) {
225 // if vrr_config is off, getDisplayConfigurationsSupported() is off as well
226 // then getModesWithLegacyDisplayConfigs should be called instead
227 SET_FLAG_FOR_TEST(com::android::graphics::surfaceflinger::flags::vrr_config, false);
228 ASSERT_FALSE(FlagManager::getInstance().vrr_config());
229
230 constexpr hal::HWDisplayId kHwcDisplayId = 2;
231 constexpr hal::HWConfigId kConfigId = 42;
232 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
233
234 expectHotplugConnect(kHwcDisplayId);
235 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
236 ASSERT_TRUE(info);
237
ramindani19919ff2023-12-07 11:27:06 -0800238 EXPECT_CALL(*mHal, isVrrSupported()).WillRepeatedly(Return(false));
Sally Qid57eb0d2023-11-07 16:46:15 -0800239
240 {
241 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
242 .WillOnce(Return(HalError::BAD_DISPLAY));
243 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
244 }
245 {
246 constexpr int32_t kWidth = 480;
247 constexpr int32_t kHeight = 720;
248 constexpr int32_t kConfigGroup = 1;
249 constexpr int32_t kVsyncPeriod = 16666667;
250
251 EXPECT_CALL(*mHal,
252 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::WIDTH,
253 _))
254 .WillRepeatedly(DoAll(SetArgPointee<3>(kWidth), Return(HalError::NONE)));
255 EXPECT_CALL(*mHal,
256 getDisplayAttribute(kHwcDisplayId, kConfigId,
257 IComposerClient::Attribute::HEIGHT, _))
258 .WillRepeatedly(DoAll(SetArgPointee<3>(kHeight), Return(HalError::NONE)));
259 EXPECT_CALL(*mHal,
260 getDisplayAttribute(kHwcDisplayId, kConfigId,
261 IComposerClient::Attribute::CONFIG_GROUP, _))
262 .WillRepeatedly(DoAll(SetArgPointee<3>(kConfigGroup), Return(HalError::NONE)));
263 EXPECT_CALL(*mHal,
264 getDisplayAttribute(kHwcDisplayId, kConfigId,
265 IComposerClient::Attribute::VSYNC_PERIOD, _))
266 .WillRepeatedly(DoAll(SetArgPointee<3>(kVsyncPeriod), Return(HalError::NONE)));
267
268 // Optional Parameters UNSUPPORTED
269 EXPECT_CALL(*mHal,
270 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
271 _))
272 .WillOnce(Return(HalError::UNSUPPORTED));
273 EXPECT_CALL(*mHal,
274 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
275 _))
276 .WillOnce(Return(HalError::UNSUPPORTED));
277
278 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
279 .WillRepeatedly(DoAll(SetArgPointee<1>(std::vector<hal::HWConfigId>{kConfigId}),
280 Return(HalError::NONE)));
281
282 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
283 EXPECT_EQ(modes.size(), size_t{1});
284 EXPECT_EQ(modes.front().hwcId, kConfigId);
285 EXPECT_EQ(modes.front().width, kWidth);
286 EXPECT_EQ(modes.front().height, kHeight);
287 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
288 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
289 EXPECT_EQ(modes.front().dpiX, -1);
290 EXPECT_EQ(modes.front().dpiY, -1);
291
292 // Optional parameters are supported
293 constexpr int32_t kDpi = 320;
294 EXPECT_CALL(*mHal,
295 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
296 _))
297 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
298 EXPECT_CALL(*mHal,
299 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
300 _))
301 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
302
303 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
304 EXPECT_EQ(modes.size(), size_t{1});
305 EXPECT_EQ(modes.front().hwcId, kConfigId);
306 EXPECT_EQ(modes.front().width, kWidth);
307 EXPECT_EQ(modes.front().height, kHeight);
308 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
309 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
310 // DPI values are scaled by 1000 in the legacy implementation.
311 EXPECT_EQ(modes.front().dpiX, kDpi / 1000.f);
312 EXPECT_EQ(modes.front().dpiY, kDpi / 1000.f);
313 }
314}
315
316TEST_F(HWComposerTest, getModesWithDisplayConfigurations_VRR_ON) {
317 SET_FLAG_FOR_TEST(com::android::graphics::surfaceflinger::flags::vrr_config, true);
318 ASSERT_TRUE(FlagManager::getInstance().vrr_config());
319
ramindani0cd1d8d2023-06-13 13:43:23 -0700320 constexpr hal::HWDisplayId kHwcDisplayId = 2;
321 constexpr hal::HWConfigId kConfigId = 42;
ramindani263a3f12023-07-18 20:44:49 -0700322 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
ramindani0cd1d8d2023-06-13 13:43:23 -0700323 expectHotplugConnect(kHwcDisplayId);
324 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
325 ASSERT_TRUE(info);
326
ramindani19919ff2023-12-07 11:27:06 -0800327 EXPECT_CALL(*mHal, isVrrSupported()).WillRepeatedly(Return(true));
ramindani0cd1d8d2023-06-13 13:43:23 -0700328
329 {
ramindani263a3f12023-07-18 20:44:49 -0700330 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
ramindani0cd1d8d2023-06-13 13:43:23 -0700331 .WillOnce(Return(HalError::BAD_DISPLAY));
ramindani263a3f12023-07-18 20:44:49 -0700332 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
ramindani0cd1d8d2023-06-13 13:43:23 -0700333 }
334 {
335 constexpr int32_t kWidth = 480;
336 constexpr int32_t kHeight = 720;
337 constexpr int32_t kConfigGroup = 1;
338 constexpr int32_t kVsyncPeriod = 16666667;
ramindanib90711b2023-10-02 15:13:19 -0700339 const hal::VrrConfig vrrConfig =
340 hal::VrrConfig{.minFrameIntervalNs = static_cast<Fps>(120_Hz).getPeriodNsecs(),
341 .notifyExpectedPresentConfig = hal::VrrConfig::
342 NotifyExpectedPresentConfig{.notifyExpectedPresentHeadsUpNs =
343 ms2ns(30),
344 .notifyExpectedPresentTimeoutNs =
345 ms2ns(30)}};
346 hal::DisplayConfiguration displayConfiguration{.configId = kConfigId,
347 .width = kWidth,
348 .height = kHeight,
349 .configGroup = kConfigGroup,
350 .vsyncPeriod = kVsyncPeriod,
351 .vrrConfig = vrrConfig};
ramindani0cd1d8d2023-06-13 13:43:23 -0700352
ramindani263a3f12023-07-18 20:44:49 -0700353 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
354 .WillOnce(DoAll(SetArgPointee<2>(std::vector<hal::DisplayConfiguration>{
ramindani0cd1d8d2023-06-13 13:43:23 -0700355 displayConfiguration}),
356 Return(HalError::NONE)));
357
358 // Optional dpi not supported
ramindani263a3f12023-07-18 20:44:49 -0700359 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700360 EXPECT_EQ(modes.size(), size_t{1});
361 EXPECT_EQ(modes.front().hwcId, kConfigId);
362 EXPECT_EQ(modes.front().width, kWidth);
363 EXPECT_EQ(modes.front().height, kHeight);
364 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
365 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
ramindanib90711b2023-10-02 15:13:19 -0700366 EXPECT_EQ(modes.front().vrrConfig, vrrConfig);
ramindani0cd1d8d2023-06-13 13:43:23 -0700367 EXPECT_EQ(modes.front().dpiX, -1);
368 EXPECT_EQ(modes.front().dpiY, -1);
369
370 // Supports optional dpi parameter
371 constexpr int32_t kDpi = 320;
372 displayConfiguration.dpi = {kDpi, kDpi};
373
ramindani263a3f12023-07-18 20:44:49 -0700374 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
375 .WillOnce(DoAll(SetArgPointee<2>(std::vector<hal::DisplayConfiguration>{
ramindani0cd1d8d2023-06-13 13:43:23 -0700376 displayConfiguration}),
377 Return(HalError::NONE)));
378
ramindani263a3f12023-07-18 20:44:49 -0700379 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700380 EXPECT_EQ(modes.size(), size_t{1});
381 EXPECT_EQ(modes.front().hwcId, kConfigId);
382 EXPECT_EQ(modes.front().width, kWidth);
383 EXPECT_EQ(modes.front().height, kHeight);
384 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
385 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
ramindanib90711b2023-10-02 15:13:19 -0700386 EXPECT_EQ(modes.front().vrrConfig, vrrConfig);
ramindani0cd1d8d2023-06-13 13:43:23 -0700387 EXPECT_EQ(modes.front().dpiX, kDpi);
388 EXPECT_EQ(modes.front().dpiY, kDpi);
389 }
390}
391
Leon Scroggins III959a7ff2023-02-07 11:24:25 -0500392TEST_F(HWComposerTest, onVsync) {
393 constexpr hal::HWDisplayId kHwcDisplayId = 1;
394 expectHotplugConnect(kHwcDisplayId);
395
396 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
397 ASSERT_TRUE(info);
398
399 const auto physicalDisplayId = info->id;
400
401 // Deliberately chosen not to match DisplayData.lastPresentTimestamp's
402 // initial value.
403 constexpr nsecs_t kTimestamp = 1;
404 auto displayIdOpt = mHwc.onVsync(kHwcDisplayId, kTimestamp);
405 ASSERT_TRUE(displayIdOpt);
406 EXPECT_EQ(physicalDisplayId, displayIdOpt);
407
408 // Attempt to send the same time stamp again.
409 displayIdOpt = mHwc.onVsync(kHwcDisplayId, kTimestamp);
410 EXPECT_FALSE(displayIdOpt);
411}
412
413TEST_F(HWComposerTest, onVsyncInvalid) {
414 constexpr hal::HWDisplayId kInvalidHwcDisplayId = 2;
415 constexpr nsecs_t kTimestamp = 1;
416 const auto displayIdOpt = mHwc.onVsync(kInvalidHwcDisplayId, kTimestamp);
417 EXPECT_FALSE(displayIdOpt);
418}
419
ramindanib90711b2023-10-02 15:13:19 -0700420TEST_F(HWComposerTest, notifyExpectedPresentTimeout) {
421 constexpr hal::HWDisplayId kHwcDisplayId = 2;
422 expectHotplugConnect(kHwcDisplayId);
423 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
424 ASSERT_TRUE(info);
425
426 auto expectedPresentTime = systemTime() + ms2ns(10);
ramindanif8c0f102023-10-09 12:42:57 -0700427 static constexpr Fps Fps60Hz = 60_Hz;
428 static constexpr int32_t kFrameInterval5HzNs = static_cast<Fps>(5_Hz).getPeriodNsecs();
429 static constexpr int32_t kFrameInterval60HzNs = Fps60Hz.getPeriodNsecs();
430 static constexpr int32_t kFrameInterval120HzNs = static_cast<Fps>(120_Hz).getPeriodNsecs();
431 static constexpr Period kVsyncPeriod =
432 Period::fromNs(static_cast<Fps>(240_Hz).getPeriodNsecs());
433 static constexpr Period kTimeoutNs = Period::fromNs(kFrameInterval5HzNs);
434 static constexpr auto kLastExpectedPresentTimestamp = TimePoint::fromNs(0);
ramindanib90711b2023-10-02 15:13:19 -0700435
ramindanif8c0f102023-10-09 12:42:57 -0700436 ASSERT_NO_FATAL_FAILURE(setDisplayData(info->id, kLastExpectedPresentTimestamp, Fps60Hz));
ramindanib90711b2023-10-02 15:13:19 -0700437
438 {
439 // Very first ExpectedPresent after idle, no previous timestamp
440 EXPECT_CALL(*mHal,
ramindanif8c0f102023-10-09 12:42:57 -0700441 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
ramindanib90711b2023-10-02 15:13:19 -0700442 .WillOnce(Return(HalError::NONE));
ramindanif8c0f102023-10-09 12:42:57 -0700443 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
444 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700445 kTimeoutNs);
446 }
447 {
ramindanif8c0f102023-10-09 12:42:57 -0700448 // Absent timeoutNs
449 expectedPresentTime += 2 * kFrameInterval5HzNs;
450 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
451 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
452 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
453 /*timeoutOpt*/ std::nullopt);
454 }
455 {
456 // Timeout is 0
457 expectedPresentTime += kFrameInterval60HzNs;
ramindanib90711b2023-10-02 15:13:19 -0700458 EXPECT_CALL(*mHal,
ramindanif8c0f102023-10-09 12:42:57 -0700459 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
ramindanib90711b2023-10-02 15:13:19 -0700460 .WillOnce(Return(HalError::NONE));
ramindanif8c0f102023-10-09 12:42:57 -0700461 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
462 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
463 Period::fromNs(0));
464 }
465 {
466 // ExpectedPresent is after the timeoutNs
467 expectedPresentTime += 2 * kFrameInterval5HzNs;
468 EXPECT_CALL(*mHal,
469 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
470 .WillOnce(Return(HalError::NONE));
471 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
472 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
473 kTimeoutNs);
474 }
475 {
476 // ExpectedPresent has not changed
477 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
478 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
479 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700480 kTimeoutNs);
481 }
482 {
483 // ExpectedPresent is after the last reported ExpectedPresent.
ramindanif8c0f102023-10-09 12:42:57 -0700484 expectedPresentTime += kFrameInterval60HzNs;
ramindanib90711b2023-10-02 15:13:19 -0700485 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
ramindanif8c0f102023-10-09 12:42:57 -0700486 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
487 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700488 kTimeoutNs);
489 }
490 {
491 // ExpectedPresent is before the last reported ExpectedPresent but after the timeoutNs,
492 // representing we changed our decision and want to present earlier than previously
493 // reported.
ramindanif8c0f102023-10-09 12:42:57 -0700494 expectedPresentTime -= kFrameInterval120HzNs;
ramindanib90711b2023-10-02 15:13:19 -0700495 EXPECT_CALL(*mHal,
ramindanif8c0f102023-10-09 12:42:57 -0700496 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime, kFrameInterval60HzNs))
ramindanib90711b2023-10-02 15:13:19 -0700497 .WillOnce(Return(HalError::NONE));
ramindanif8c0f102023-10-09 12:42:57 -0700498 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
499 TimePoint::fromNs(expectedPresentTime), Fps60Hz,
ramindanib90711b2023-10-02 15:13:19 -0700500 kTimeoutNs);
501 }
502}
503
ramindanif8c0f102023-10-09 12:42:57 -0700504TEST_F(HWComposerTest, notifyExpectedPresentRenderRateChanged) {
505 constexpr hal::HWDisplayId kHwcDisplayId = 2;
506 expectHotplugConnect(kHwcDisplayId);
507 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
508 ASSERT_TRUE(info);
509
510 const auto now = systemTime();
511 auto expectedPresentTime = now;
512 static constexpr Period kTimeoutNs = Period::fromNs(static_cast<Fps>(1_Hz).getPeriodNsecs());
513
514 ASSERT_NO_FATAL_FAILURE(setDisplayData(info->id, TimePoint::fromNs(now), Fps::fromValue(0)));
515 static constexpr int32_t kFrameIntervalNs120Hz = static_cast<Fps>(120_Hz).getPeriodNsecs();
516 static constexpr int32_t kFrameIntervalNs96Hz = static_cast<Fps>(96_Hz).getPeriodNsecs();
517 static constexpr int32_t kFrameIntervalNs80Hz = static_cast<Fps>(80_Hz).getPeriodNsecs();
518 static constexpr int32_t kFrameIntervalNs60Hz = static_cast<Fps>(60_Hz).getPeriodNsecs();
519 static constexpr int32_t kFrameIntervalNs40Hz = static_cast<Fps>(40_Hz).getPeriodNsecs();
520 static constexpr int32_t kFrameIntervalNs30Hz = static_cast<Fps>(30_Hz).getPeriodNsecs();
521 static constexpr int32_t kFrameIntervalNs24Hz = static_cast<Fps>(24_Hz).getPeriodNsecs();
522 static constexpr int32_t kFrameIntervalNs20Hz = static_cast<Fps>(20_Hz).getPeriodNsecs();
523 static constexpr Period kVsyncPeriod =
524 Period::fromNs(static_cast<Fps>(240_Hz).getPeriodNsecs());
525
526 struct FrameRateIntervalTestData {
527 int32_t frameIntervalNs;
528 bool callExpectedPresent;
529 };
530 const std::vector<FrameRateIntervalTestData> frameIntervals = {
531 {kFrameIntervalNs60Hz, true}, {kFrameIntervalNs96Hz, true},
532 {kFrameIntervalNs80Hz, true}, {kFrameIntervalNs120Hz, true},
533 {kFrameIntervalNs80Hz, true}, {kFrameIntervalNs60Hz, true},
534 {kFrameIntervalNs60Hz, false}, {kFrameIntervalNs30Hz, false},
535 {kFrameIntervalNs24Hz, true}, {kFrameIntervalNs40Hz, true},
536 {kFrameIntervalNs20Hz, false}, {kFrameIntervalNs60Hz, true},
537 {kFrameIntervalNs20Hz, false}, {kFrameIntervalNs120Hz, true},
538 };
539
540 for (const auto& [frameIntervalNs, callExpectedPresent] : frameIntervals) {
541 {
542 expectedPresentTime += frameIntervalNs;
543 if (callExpectedPresent) {
544 EXPECT_CALL(*mHal,
545 notifyExpectedPresent(kHwcDisplayId, expectedPresentTime,
546 frameIntervalNs))
547 .WillOnce(Return(HalError::NONE));
548 } else {
549 EXPECT_CALL(*mHal, notifyExpectedPresent(kHwcDisplayId, _, _)).Times(0);
550 }
551 mHwc.notifyExpectedPresentIfRequired(info->id, kVsyncPeriod,
552 TimePoint::fromNs(expectedPresentTime),
553 Fps::fromPeriodNsecs(frameIntervalNs), kTimeoutNs);
554 }
555 }
556}
557
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700558struct MockHWC2ComposerCallback final : StrictMock<HWC2::ComposerCallback> {
Huihong Luo791bef92023-09-19 21:18:45 -0700559 MOCK_METHOD(void, onComposerHalHotplugEvent, (hal::HWDisplayId, DisplayHotplugEvent),
560 (override));
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700561 MOCK_METHOD1(onComposerHalRefresh, void(hal::HWDisplayId));
562 MOCK_METHOD3(onComposerHalVsync,
563 void(hal::HWDisplayId, int64_t timestamp, std::optional<hal::VsyncPeriodNanos>));
564 MOCK_METHOD2(onComposerHalVsyncPeriodTimingChanged,
565 void(hal::HWDisplayId, const hal::VsyncPeriodChangeTimeline&));
566 MOCK_METHOD1(onComposerHalSeamlessPossible, void(hal::HWDisplayId));
Yichi Chen1a417af2022-01-21 15:29:52 +0800567 MOCK_METHOD1(onComposerHalVsyncIdle, void(hal::HWDisplayId));
ramindani12bfe6b2023-02-03 13:29:19 -0800568 MOCK_METHOD(void, onRefreshRateChangedDebug, (const RefreshRateChangedDebugData&), (override));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800569};
570
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700571struct HWComposerSetCallbackTest : HWComposerTest {
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700572 MockHWC2ComposerCallback mCallback;
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800573};
574
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700575TEST_F(HWComposerSetCallbackTest, loadsLayerMetadataSupport) {
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800576 const std::string kMetadata1Name = "com.example.metadata.1";
577 constexpr bool kMetadata1Mandatory = false;
578 const std::string kMetadata2Name = "com.example.metadata.2";
579 constexpr bool kMetadata2Mandatory = true;
580
Ady Abrahamde549d42022-01-26 19:19:17 -0800581 EXPECT_CALL(*mHal, getCapabilities()).WillOnce(Return(std::vector<aidl::Capability>{}));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800582 EXPECT_CALL(*mHal, getLayerGenericMetadataKeys(_))
Peiyong Line9d809e2020-04-14 13:10:48 -0700583 .WillOnce(DoAll(SetArgPointee<0>(std::vector<hal::LayerGenericMetadataKey>{
584 {kMetadata1Name, kMetadata1Mandatory},
585 {kMetadata2Name, kMetadata2Mandatory},
586 }),
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800587 Return(hardware::graphics::composer::V2_4::Error::NONE)));
Sally Qibb866c12022-10-17 11:31:20 -0700588 EXPECT_CALL(*mHal, getOverlaySupport(_)).WillOnce(Return(HalError::NONE));
Kriti Dang674b9372022-11-18 10:58:44 +0100589 EXPECT_CALL(*mHal, getHdrConversionCapabilities(_)).WillOnce(Return(HalError::NONE));
Sally Qibb866c12022-10-17 11:31:20 -0700590
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800591 EXPECT_CALL(*mHal, registerCallback(_));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800592
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700593 mHwc.setCallback(mCallback);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800594
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700595 const auto& supported = mHwc.getSupportedLayerGenericMetadata();
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800596 EXPECT_EQ(2u, supported.size());
597 EXPECT_EQ(1u, supported.count(kMetadata1Name));
598 EXPECT_EQ(kMetadata1Mandatory, supported.find(kMetadata1Name)->second);
599 EXPECT_EQ(1u, supported.count(kMetadata2Name));
600 EXPECT_EQ(kMetadata2Mandatory, supported.find(kMetadata2Name)->second);
601}
602
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700603TEST_F(HWComposerSetCallbackTest, handlesUnsupportedCallToGetLayerGenericMetadataKeys) {
Ady Abrahamde549d42022-01-26 19:19:17 -0800604 EXPECT_CALL(*mHal, getCapabilities()).WillOnce(Return(std::vector<aidl::Capability>{}));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800605 EXPECT_CALL(*mHal, getLayerGenericMetadataKeys(_))
606 .WillOnce(Return(hardware::graphics::composer::V2_4::Error::UNSUPPORTED));
Sally Qibb866c12022-10-17 11:31:20 -0700607 EXPECT_CALL(*mHal, getOverlaySupport(_)).WillOnce(Return(HalError::UNSUPPORTED));
Kriti Dang674b9372022-11-18 10:58:44 +0100608 EXPECT_CALL(*mHal, getHdrConversionCapabilities(_)).WillOnce(Return(HalError::UNSUPPORTED));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800609 EXPECT_CALL(*mHal, registerCallback(_));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800610
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700611 mHwc.setCallback(mCallback);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800612
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700613 const auto& supported = mHwc.getSupportedLayerGenericMetadata();
614 EXPECT_TRUE(supported.empty());
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800615}
616
617struct HWComposerLayerTest : public testing::Test {
Peiyong Line9d809e2020-04-14 13:10:48 -0700618 static constexpr hal::HWDisplayId kDisplayId = static_cast<hal::HWDisplayId>(1001);
619 static constexpr hal::HWLayerId kLayerId = static_cast<hal::HWLayerId>(1002);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800620
Ady Abrahamde549d42022-01-26 19:19:17 -0800621 HWComposerLayerTest(const std::unordered_set<aidl::Capability>& capabilities)
Lloyd Piquea516c002021-05-07 14:36:58 -0700622 : mCapabilies(capabilities) {
623 EXPECT_CALL(mDisplay, getId()).WillRepeatedly(Return(kDisplayId));
624 }
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800625
Lloyd Piquea516c002021-05-07 14:36:58 -0700626 ~HWComposerLayerTest() override {
627 EXPECT_CALL(mDisplay, onLayerDestroyed(kLayerId));
628 EXPECT_CALL(*mHal, destroyLayer(kDisplayId, kLayerId));
629 }
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800630
631 std::unique_ptr<Hwc2::mock::Composer> mHal{new StrictMock<Hwc2::mock::Composer>()};
Ady Abrahamde549d42022-01-26 19:19:17 -0800632 const std::unordered_set<aidl::Capability> mCapabilies;
Lloyd Piquea516c002021-05-07 14:36:58 -0700633 StrictMock<HWC2::mock::Display> mDisplay;
634 HWC2::impl::Layer mLayer{*mHal, mCapabilies, mDisplay, kLayerId};
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800635};
636
637struct HWComposerLayerGenericMetadataTest : public HWComposerLayerTest {
638 static const std::string kLayerGenericMetadata1Name;
639 static constexpr bool kLayerGenericMetadata1Mandatory = false;
640 static const std::vector<uint8_t> kLayerGenericMetadata1Value;
641 static const std::string kLayerGenericMetadata2Name;
642 static constexpr bool kLayerGenericMetadata2Mandatory = true;
643 static const std::vector<uint8_t> kLayerGenericMetadata2Value;
644
645 HWComposerLayerGenericMetadataTest() : HWComposerLayerTest({}) {}
646};
647
648const std::string HWComposerLayerGenericMetadataTest::kLayerGenericMetadata1Name =
649 "com.example.metadata.1";
650
651const std::vector<uint8_t> HWComposerLayerGenericMetadataTest::kLayerGenericMetadata1Value = {1u,
652 2u,
653 3u};
654
655const std::string HWComposerLayerGenericMetadataTest::kLayerGenericMetadata2Name =
656 "com.example.metadata.2";
657
658const std::vector<uint8_t> HWComposerLayerGenericMetadataTest::kLayerGenericMetadata2Value = {45u,
659 67u};
660
661TEST_F(HWComposerLayerGenericMetadataTest, forwardsSupportedMetadata) {
662 EXPECT_CALL(*mHal,
663 setLayerGenericMetadata(kDisplayId, kLayerId, kLayerGenericMetadata1Name,
664 kLayerGenericMetadata1Mandatory,
665 kLayerGenericMetadata1Value))
666 .WillOnce(Return(hardware::graphics::composer::V2_4::Error::NONE));
667 auto result = mLayer.setLayerGenericMetadata(kLayerGenericMetadata1Name,
668 kLayerGenericMetadata1Mandatory,
669 kLayerGenericMetadata1Value);
Peiyong Line9d809e2020-04-14 13:10:48 -0700670 EXPECT_EQ(hal::Error::NONE, result);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800671
672 EXPECT_CALL(*mHal,
673 setLayerGenericMetadata(kDisplayId, kLayerId, kLayerGenericMetadata2Name,
674 kLayerGenericMetadata2Mandatory,
675 kLayerGenericMetadata2Value))
676 .WillOnce(Return(hardware::graphics::composer::V2_4::Error::UNSUPPORTED));
677 result = mLayer.setLayerGenericMetadata(kLayerGenericMetadata2Name,
678 kLayerGenericMetadata2Mandatory,
679 kLayerGenericMetadata2Value);
Peiyong Line9d809e2020-04-14 13:10:48 -0700680 EXPECT_EQ(hal::Error::UNSUPPORTED, result);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800681}
682
Dominik Laskowski3dce4f42021-03-08 20:48:28 -0800683} // namespace android