blob: ba2d3e28adf3d000d5e52dbac63e18ff3c942fbc [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
ramindani355fca82023-12-18 12:07:34 -080024#include <optional>
Lloyd Pique4603f3c2020-02-11 12:06:56 -080025#include <vector>
26
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070027// StrictMock<T> derives from T and is not marked final, so the destructor of T is expected to be
28// virtual in case StrictMock<T> is used as a polymorphic base class. That is not the case here.
29#pragma clang diagnostic push
30#pragma clang diagnostic ignored "-Wnon-virtual-dtor"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080031#include <gmock/gmock.h>
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070032#pragma clang diagnostic pop
33
Alec Mouri9b133ca2023-11-14 19:00:01 +000034#include <common/FlagManager.h>
Lloyd Pique4603f3c2020-02-11 12:06:56 -080035#include <gui/LayerMetadata.h>
36#include <log/log.h>
ramindanib90711b2023-10-02 15:13:19 -070037#include <chrono>
Lloyd Pique4603f3c2020-02-11 12:06:56 -080038
Ady Abraham3db8a3c2023-11-20 17:53:47 -080039#include <common/test/FlagUtils.h>
Marin Shalamanov23c44202020-12-22 19:09:20 +010040#include "DisplayHardware/DisplayMode.h"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080041#include "DisplayHardware/HWComposer.h"
Marin Shalamanov12c9e5a2021-01-07 00:25:35 +010042#include "DisplayHardware/Hal.h"
Alec Mouriff793872022-01-13 17:45:06 -080043#include "DisplayIdentificationTestHelpers.h"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080044#include "mock/DisplayHardware/MockComposer.h"
Lloyd Piquea516c002021-05-07 14:36:58 -070045#include "mock/DisplayHardware/MockHWC2.h"
Lloyd Pique4603f3c2020-02-11 12:06:56 -080046
Sally Qid57eb0d2023-11-07 16:46:15 -080047#include <com_android_graphics_surfaceflinger_flags.h>
48
Lloyd Pique4603f3c2020-02-11 12:06:56 -080049// TODO(b/129481165): remove the #pragma below and fix conversion issues
50#pragma clang diagnostic pop // ignored "-Wconversion"
51
52namespace android {
Lloyd Pique4603f3c2020-02-11 12:06:56 -080053
Marin Shalamanov6e840172020-12-14 22:13:28 +010054namespace V2_1 = hardware::graphics::composer::V2_1;
55namespace V2_4 = hardware::graphics::composer::V2_4;
Ady Abrahamde549d42022-01-26 19:19:17 -080056namespace aidl = aidl::android::hardware::graphics::composer3;
ramindanib90711b2023-10-02 15:13:19 -070057using namespace std::chrono_literals;
Marin Shalamanov6e840172020-12-14 22:13:28 +010058
59using Hwc2::Config;
Peiyong Line9d809e2020-04-14 13:10:48 -070060
Huihong Luo817ebb82023-12-13 12:54:24 -080061using ::aidl::android::hardware::drm::HdcpLevels;
Huihong Luo791bef92023-09-19 21:18:45 -070062using ::aidl::android::hardware::graphics::common::DisplayHotplugEvent;
ramindani12bfe6b2023-02-03 13:29:19 -080063using ::aidl::android::hardware::graphics::composer3::RefreshRateChangedDebugData;
ramindani0cd1d8d2023-06-13 13:43:23 -070064using hal::IComposerClient;
Lloyd Pique4603f3c2020-02-11 12:06:56 -080065using ::testing::_;
66using ::testing::DoAll;
Lloyd Pique4603f3c2020-02-11 12:06:56 -080067using ::testing::Return;
68using ::testing::SetArgPointee;
69using ::testing::StrictMock;
70
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070071struct HWComposerTest : testing::Test {
Manasi Navarefc2a4a72024-11-12 23:59:21 +000072 using HalError = hal::Error;
Marin Shalamanov8b196592021-08-09 16:24:42 +020073
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070074 Hwc2::mock::Composer* const mHal = new StrictMock<Hwc2::mock::Composer>();
75 impl::HWComposer mHwc{std::unique_ptr<Hwc2::Composer>(mHal)};
Marin Shalamanov8b196592021-08-09 16:24:42 +020076
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070077 void expectHotplugConnect(hal::HWDisplayId hwcDisplayId) {
78 constexpr uint8_t kPort = 255;
79 EXPECT_CALL(*mHal, getDisplayIdentificationData(hwcDisplayId, _, _))
80 .WillOnce(DoAll(SetArgPointee<1>(kPort),
81 SetArgPointee<2>(getExternalEdid()), Return(HalError::NONE)));
Marin Shalamanov8b196592021-08-09 16:24:42 +020082
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070083 EXPECT_CALL(*mHal, setClientTargetSlotCount(_));
84 EXPECT_CALL(*mHal, setVsyncEnabled(hwcDisplayId, Hwc2::IComposerClient::Vsync::DISABLE));
Leon Scroggins IIIe24d78f2022-09-20 16:38:19 -040085 EXPECT_CALL(*mHal, onHotplugConnect(hwcDisplayId));
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070086 }
ramindani355fca82023-12-18 12:07:34 -080087
88 void setVrrTimeoutHint(bool status) { mHwc.mEnableVrrTimeout = status; }
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070089};
Marin Shalamanov8b196592021-08-09 16:24:42 +020090
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -070091TEST_F(HWComposerTest, isHeadless) {
92 ASSERT_TRUE(mHwc.isHeadless());
93
94 constexpr hal::HWDisplayId kHwcDisplayId = 1;
95 expectHotplugConnect(kHwcDisplayId);
96
97 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
Marin Shalamanov8b196592021-08-09 16:24:42 +020098 ASSERT_TRUE(info);
Marin Shalamanov8b196592021-08-09 16:24:42 +020099
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700100 ASSERT_FALSE(mHwc.isHeadless());
101
102 mHwc.disconnectDisplay(info->id);
103 ASSERT_TRUE(mHwc.isHeadless());
104}
105
Dominik Laskowski969cdcb2024-02-08 16:35:29 -0500106TEST_F(HWComposerTest, getDisplayConnectionType) {
107 // Unknown display.
108 EXPECT_EQ(mHwc.getDisplayConnectionType(PhysicalDisplayId::fromPort(0)),
109 ui::DisplayConnectionType::Internal);
110
111 constexpr hal::HWDisplayId kHwcDisplayId = 1;
112 expectHotplugConnect(kHwcDisplayId);
113
114 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
115 ASSERT_TRUE(info);
116
117 EXPECT_CALL(*mHal, getDisplayConnectionType(kHwcDisplayId, _))
118 .WillOnce(DoAll(SetArgPointee<1>(IComposerClient::DisplayConnectionType::EXTERNAL),
119 Return(V2_4::Error::NONE)));
120
121 // The first call caches the connection type.
122 EXPECT_EQ(mHwc.getDisplayConnectionType(info->id), ui::DisplayConnectionType::External);
123
124 // Subsequent calls return the cached connection type.
125 EXPECT_EQ(mHwc.getDisplayConnectionType(info->id), ui::DisplayConnectionType::External);
126 EXPECT_EQ(mHwc.getDisplayConnectionType(info->id), ui::DisplayConnectionType::External);
127}
128
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700129TEST_F(HWComposerTest, getActiveMode) {
130 // Unknown display.
Dominik Laskowskid940a012024-01-28 13:25:44 -0500131 EXPECT_EQ(mHwc.getActiveMode(PhysicalDisplayId::fromPort(0)), ftl::Unexpected(BAD_INDEX));
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700132
133 constexpr hal::HWDisplayId kHwcDisplayId = 2;
134 expectHotplugConnect(kHwcDisplayId);
135
136 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
137 ASSERT_TRUE(info);
138
139 {
140 // Display is known to SF but not HWC, e.g. the hotplug disconnect is pending.
141 EXPECT_CALL(*mHal, getActiveConfig(kHwcDisplayId, _))
142 .WillOnce(Return(HalError::BAD_DISPLAY));
143
Dominik Laskowskid940a012024-01-28 13:25:44 -0500144 EXPECT_EQ(mHwc.getActiveMode(info->id), ftl::Unexpected(UNKNOWN_ERROR));
145 }
146 {
147 EXPECT_CALL(*mHal, getActiveConfig(kHwcDisplayId, _))
148 .WillOnce(Return(HalError::BAD_CONFIG));
149
150 EXPECT_EQ(mHwc.getActiveMode(info->id), ftl::Unexpected(NO_INIT));
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700151 }
152 {
153 constexpr hal::HWConfigId kConfigId = 42;
154 EXPECT_CALL(*mHal, getActiveConfig(kHwcDisplayId, _))
155 .WillOnce(DoAll(SetArgPointee<1>(kConfigId), Return(HalError::NONE)));
156
Dominik Laskowskid940a012024-01-28 13:25:44 -0500157 EXPECT_EQ(mHwc.getActiveMode(info->id).value_opt(), kConfigId);
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700158 }
Marin Shalamanov8b196592021-08-09 16:24:42 +0200159}
160
ramindani0cd1d8d2023-06-13 13:43:23 -0700161TEST_F(HWComposerTest, getModesWithLegacyDisplayConfigs) {
162 constexpr hal::HWDisplayId kHwcDisplayId = 2;
163 constexpr hal::HWConfigId kConfigId = 42;
ramindani263a3f12023-07-18 20:44:49 -0700164 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
ramindani0cd1d8d2023-06-13 13:43:23 -0700165
166 expectHotplugConnect(kHwcDisplayId);
167 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
168 ASSERT_TRUE(info);
Lucas Berthou8d0a0c42024-08-27 14:32:31 +0000169 ASSERT_TRUE(info->preferredDetailedTimingDescriptor.has_value());
ramindani0cd1d8d2023-06-13 13:43:23 -0700170
ramindani19919ff2023-12-07 11:27:06 -0800171 EXPECT_CALL(*mHal, isVrrSupported()).WillRepeatedly(Return(false));
ramindani0cd1d8d2023-06-13 13:43:23 -0700172
173 {
174 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
175 .WillOnce(Return(HalError::BAD_DISPLAY));
ramindani263a3f12023-07-18 20:44:49 -0700176 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
ramindani0cd1d8d2023-06-13 13:43:23 -0700177 }
178 {
179 constexpr int32_t kWidth = 480;
180 constexpr int32_t kHeight = 720;
181 constexpr int32_t kConfigGroup = 1;
182 constexpr int32_t kVsyncPeriod = 16666667;
Lucas Berthou8d0a0c42024-08-27 14:32:31 +0000183 constexpr float kMmPerInch = 25.4f;
184 const ui::Size size = info->preferredDetailedTimingDescriptor->physicalSizeInMm;
185 const float expectedDpiX = (kWidth * kMmPerInch / size.width);
186 const float expectedDpiY = (kHeight * kMmPerInch / size.height);
ramindani0cd1d8d2023-06-13 13:43:23 -0700187
188 EXPECT_CALL(*mHal,
189 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::WIDTH,
190 _))
191 .WillRepeatedly(DoAll(SetArgPointee<3>(kWidth), Return(HalError::NONE)));
192 EXPECT_CALL(*mHal,
193 getDisplayAttribute(kHwcDisplayId, kConfigId,
194 IComposerClient::Attribute::HEIGHT, _))
195 .WillRepeatedly(DoAll(SetArgPointee<3>(kHeight), Return(HalError::NONE)));
196 EXPECT_CALL(*mHal,
197 getDisplayAttribute(kHwcDisplayId, kConfigId,
198 IComposerClient::Attribute::CONFIG_GROUP, _))
199 .WillRepeatedly(DoAll(SetArgPointee<3>(kConfigGroup), Return(HalError::NONE)));
200 EXPECT_CALL(*mHal,
201 getDisplayAttribute(kHwcDisplayId, kConfigId,
202 IComposerClient::Attribute::VSYNC_PERIOD, _))
203 .WillRepeatedly(DoAll(SetArgPointee<3>(kVsyncPeriod), Return(HalError::NONE)));
204
205 // Optional Parameters UNSUPPORTED
206 EXPECT_CALL(*mHal,
207 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
208 _))
209 .WillOnce(Return(HalError::UNSUPPORTED));
210 EXPECT_CALL(*mHal,
211 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
212 _))
213 .WillOnce(Return(HalError::UNSUPPORTED));
214
215 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
216 .WillRepeatedly(DoAll(SetArgPointee<1>(std::vector<hal::HWConfigId>{kConfigId}),
217 Return(HalError::NONE)));
218
ramindani263a3f12023-07-18 20:44:49 -0700219 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700220 EXPECT_EQ(modes.size(), size_t{1});
221 EXPECT_EQ(modes.front().hwcId, kConfigId);
222 EXPECT_EQ(modes.front().width, kWidth);
223 EXPECT_EQ(modes.front().height, kHeight);
224 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
225 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
Lucas Berthou8d0a0c42024-08-27 14:32:31 +0000226 if (!FlagManager::getInstance().correct_dpi_with_display_size()) {
227 EXPECT_EQ(modes.front().dpiX, -1);
228 EXPECT_EQ(modes.front().dpiY, -1);
229 } else {
230 EXPECT_EQ(modes.front().dpiX, expectedDpiX);
231 EXPECT_EQ(modes.front().dpiY, expectedDpiY);
232 }
ramindani0cd1d8d2023-06-13 13:43:23 -0700233
234 // Optional parameters are supported
235 constexpr int32_t kDpi = 320;
236 EXPECT_CALL(*mHal,
237 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
238 _))
239 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
240 EXPECT_CALL(*mHal,
241 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
242 _))
243 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
244
ramindani263a3f12023-07-18 20:44:49 -0700245 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700246 EXPECT_EQ(modes.size(), size_t{1});
247 EXPECT_EQ(modes.front().hwcId, kConfigId);
248 EXPECT_EQ(modes.front().width, kWidth);
249 EXPECT_EQ(modes.front().height, kHeight);
250 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
251 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
252 // DPI values are scaled by 1000 in the legacy implementation.
253 EXPECT_EQ(modes.front().dpiX, kDpi / 1000.f);
254 EXPECT_EQ(modes.front().dpiY, kDpi / 1000.f);
255 }
256}
257
Sally Qid57eb0d2023-11-07 16:46:15 -0800258TEST_F(HWComposerTest, getModesWithDisplayConfigurations_VRR_OFF) {
259 // if vrr_config is off, getDisplayConfigurationsSupported() is off as well
260 // then getModesWithLegacyDisplayConfigs should be called instead
261 SET_FLAG_FOR_TEST(com::android::graphics::surfaceflinger::flags::vrr_config, false);
262 ASSERT_FALSE(FlagManager::getInstance().vrr_config());
263
264 constexpr hal::HWDisplayId kHwcDisplayId = 2;
265 constexpr hal::HWConfigId kConfigId = 42;
266 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
267
268 expectHotplugConnect(kHwcDisplayId);
269 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
270 ASSERT_TRUE(info);
271
ramindani19919ff2023-12-07 11:27:06 -0800272 EXPECT_CALL(*mHal, isVrrSupported()).WillRepeatedly(Return(false));
Sally Qid57eb0d2023-11-07 16:46:15 -0800273
274 {
275 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
276 .WillOnce(Return(HalError::BAD_DISPLAY));
277 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
278 }
279 {
280 constexpr int32_t kWidth = 480;
281 constexpr int32_t kHeight = 720;
282 constexpr int32_t kConfigGroup = 1;
283 constexpr int32_t kVsyncPeriod = 16666667;
Lucas Berthou8d0a0c42024-08-27 14:32:31 +0000284 constexpr float kMmPerInch = 25.4f;
285 const ui::Size size = info->preferredDetailedTimingDescriptor->physicalSizeInMm;
286 const float expectedDpiX = (kWidth * kMmPerInch / size.width);
287 const float expectedDpiY = (kHeight * kMmPerInch / size.height);
Sally Qid57eb0d2023-11-07 16:46:15 -0800288
289 EXPECT_CALL(*mHal,
290 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::WIDTH,
291 _))
292 .WillRepeatedly(DoAll(SetArgPointee<3>(kWidth), Return(HalError::NONE)));
293 EXPECT_CALL(*mHal,
294 getDisplayAttribute(kHwcDisplayId, kConfigId,
295 IComposerClient::Attribute::HEIGHT, _))
296 .WillRepeatedly(DoAll(SetArgPointee<3>(kHeight), Return(HalError::NONE)));
297 EXPECT_CALL(*mHal,
298 getDisplayAttribute(kHwcDisplayId, kConfigId,
299 IComposerClient::Attribute::CONFIG_GROUP, _))
300 .WillRepeatedly(DoAll(SetArgPointee<3>(kConfigGroup), Return(HalError::NONE)));
301 EXPECT_CALL(*mHal,
302 getDisplayAttribute(kHwcDisplayId, kConfigId,
303 IComposerClient::Attribute::VSYNC_PERIOD, _))
304 .WillRepeatedly(DoAll(SetArgPointee<3>(kVsyncPeriod), Return(HalError::NONE)));
305
306 // Optional Parameters UNSUPPORTED
307 EXPECT_CALL(*mHal,
308 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
309 _))
310 .WillOnce(Return(HalError::UNSUPPORTED));
311 EXPECT_CALL(*mHal,
312 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
313 _))
314 .WillOnce(Return(HalError::UNSUPPORTED));
315
316 EXPECT_CALL(*mHal, getDisplayConfigs(kHwcDisplayId, _))
317 .WillRepeatedly(DoAll(SetArgPointee<1>(std::vector<hal::HWConfigId>{kConfigId}),
318 Return(HalError::NONE)));
319
320 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
321 EXPECT_EQ(modes.size(), size_t{1});
322 EXPECT_EQ(modes.front().hwcId, kConfigId);
323 EXPECT_EQ(modes.front().width, kWidth);
324 EXPECT_EQ(modes.front().height, kHeight);
325 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
326 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
Lucas Berthou8d0a0c42024-08-27 14:32:31 +0000327 if (!FlagManager::getInstance().correct_dpi_with_display_size()) {
328 EXPECT_EQ(modes.front().dpiX, -1);
329 EXPECT_EQ(modes.front().dpiY, -1);
330 } else {
331 EXPECT_EQ(modes.front().dpiX, expectedDpiX);
332 EXPECT_EQ(modes.front().dpiY, expectedDpiY);
333 }
Sally Qid57eb0d2023-11-07 16:46:15 -0800334
335 // Optional parameters are supported
336 constexpr int32_t kDpi = 320;
337 EXPECT_CALL(*mHal,
338 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_X,
339 _))
340 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
341 EXPECT_CALL(*mHal,
342 getDisplayAttribute(kHwcDisplayId, kConfigId, IComposerClient::Attribute::DPI_Y,
343 _))
344 .WillOnce(DoAll(SetArgPointee<3>(kDpi), Return(HalError::NONE)));
345
346 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
347 EXPECT_EQ(modes.size(), size_t{1});
348 EXPECT_EQ(modes.front().hwcId, kConfigId);
349 EXPECT_EQ(modes.front().width, kWidth);
350 EXPECT_EQ(modes.front().height, kHeight);
351 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
352 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
353 // DPI values are scaled by 1000 in the legacy implementation.
354 EXPECT_EQ(modes.front().dpiX, kDpi / 1000.f);
355 EXPECT_EQ(modes.front().dpiY, kDpi / 1000.f);
356 }
357}
358
359TEST_F(HWComposerTest, getModesWithDisplayConfigurations_VRR_ON) {
360 SET_FLAG_FOR_TEST(com::android::graphics::surfaceflinger::flags::vrr_config, true);
361 ASSERT_TRUE(FlagManager::getInstance().vrr_config());
362
ramindani0cd1d8d2023-06-13 13:43:23 -0700363 constexpr hal::HWDisplayId kHwcDisplayId = 2;
364 constexpr hal::HWConfigId kConfigId = 42;
ramindani263a3f12023-07-18 20:44:49 -0700365 constexpr int32_t kMaxFrameIntervalNs = 50000000; // 20Fps
ramindani0cd1d8d2023-06-13 13:43:23 -0700366 expectHotplugConnect(kHwcDisplayId);
367 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
368 ASSERT_TRUE(info);
369
ramindani19919ff2023-12-07 11:27:06 -0800370 EXPECT_CALL(*mHal, isVrrSupported()).WillRepeatedly(Return(true));
ramindani0cd1d8d2023-06-13 13:43:23 -0700371
372 {
ramindani263a3f12023-07-18 20:44:49 -0700373 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
ramindani0cd1d8d2023-06-13 13:43:23 -0700374 .WillOnce(Return(HalError::BAD_DISPLAY));
ramindani263a3f12023-07-18 20:44:49 -0700375 EXPECT_TRUE(mHwc.getModes(info->id, kMaxFrameIntervalNs).empty());
ramindani0cd1d8d2023-06-13 13:43:23 -0700376 }
377 {
ramindani355fca82023-12-18 12:07:34 -0800378 setVrrTimeoutHint(true);
ramindani0cd1d8d2023-06-13 13:43:23 -0700379 constexpr int32_t kWidth = 480;
380 constexpr int32_t kHeight = 720;
381 constexpr int32_t kConfigGroup = 1;
382 constexpr int32_t kVsyncPeriod = 16666667;
Lucas Berthou8d0a0c42024-08-27 14:32:31 +0000383 constexpr float kMmPerInch = 25.4f;
384 const ui::Size size = info->preferredDetailedTimingDescriptor->physicalSizeInMm;
385 const float expectedDpiX = (kWidth * kMmPerInch / size.width);
386 const float expectedDpiY = (kHeight * kMmPerInch / size.height);
Sasha McIntoshf08ed642024-10-24 15:43:07 -0400387 const OutputType hdrOutputType = OutputType::SYSTEM;
ramindanib90711b2023-10-02 15:13:19 -0700388 const hal::VrrConfig vrrConfig =
389 hal::VrrConfig{.minFrameIntervalNs = static_cast<Fps>(120_Hz).getPeriodNsecs(),
390 .notifyExpectedPresentConfig = hal::VrrConfig::
ramindanicbd7a6d2023-12-19 16:00:30 -0800391 NotifyExpectedPresentConfig{.headsUpNs = ms2ns(30),
392 .timeoutNs = ms2ns(30)}};
ramindanib90711b2023-10-02 15:13:19 -0700393 hal::DisplayConfiguration displayConfiguration{.configId = kConfigId,
394 .width = kWidth,
395 .height = kHeight,
396 .configGroup = kConfigGroup,
397 .vsyncPeriod = kVsyncPeriod,
Sasha McIntoshf08ed642024-10-24 15:43:07 -0400398 .vrrConfig = vrrConfig,
399 .hdrOutputType = hdrOutputType};
ramindani0cd1d8d2023-06-13 13:43:23 -0700400
ramindani263a3f12023-07-18 20:44:49 -0700401 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
402 .WillOnce(DoAll(SetArgPointee<2>(std::vector<hal::DisplayConfiguration>{
ramindani0cd1d8d2023-06-13 13:43:23 -0700403 displayConfiguration}),
404 Return(HalError::NONE)));
405
406 // Optional dpi not supported
ramindani263a3f12023-07-18 20:44:49 -0700407 auto modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700408 EXPECT_EQ(modes.size(), size_t{1});
409 EXPECT_EQ(modes.front().hwcId, kConfigId);
410 EXPECT_EQ(modes.front().width, kWidth);
411 EXPECT_EQ(modes.front().height, kHeight);
412 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
413 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
ramindanib90711b2023-10-02 15:13:19 -0700414 EXPECT_EQ(modes.front().vrrConfig, vrrConfig);
Sasha McIntoshf08ed642024-10-24 15:43:07 -0400415 EXPECT_EQ(modes.front().hdrOutputType, hdrOutputType);
Lucas Berthou8d0a0c42024-08-27 14:32:31 +0000416 if (!FlagManager::getInstance().correct_dpi_with_display_size()) {
417 EXPECT_EQ(modes.front().dpiX, -1);
418 EXPECT_EQ(modes.front().dpiY, -1);
419 } else {
420 EXPECT_EQ(modes.front().dpiX, expectedDpiX);
421 EXPECT_EQ(modes.front().dpiY, expectedDpiY);
422 }
ramindani0cd1d8d2023-06-13 13:43:23 -0700423
424 // Supports optional dpi parameter
425 constexpr int32_t kDpi = 320;
426 displayConfiguration.dpi = {kDpi, kDpi};
427
ramindani263a3f12023-07-18 20:44:49 -0700428 EXPECT_CALL(*mHal, getDisplayConfigurations(kHwcDisplayId, _, _))
ramindani355fca82023-12-18 12:07:34 -0800429 .WillRepeatedly(DoAll(SetArgPointee<2>(std::vector<hal::DisplayConfiguration>{
430 displayConfiguration}),
431 Return(HalError::NONE)));
ramindani0cd1d8d2023-06-13 13:43:23 -0700432
ramindani263a3f12023-07-18 20:44:49 -0700433 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
ramindani0cd1d8d2023-06-13 13:43:23 -0700434 EXPECT_EQ(modes.size(), size_t{1});
435 EXPECT_EQ(modes.front().hwcId, kConfigId);
436 EXPECT_EQ(modes.front().width, kWidth);
437 EXPECT_EQ(modes.front().height, kHeight);
438 EXPECT_EQ(modes.front().configGroup, kConfigGroup);
439 EXPECT_EQ(modes.front().vsyncPeriod, kVsyncPeriod);
ramindanib90711b2023-10-02 15:13:19 -0700440 EXPECT_EQ(modes.front().vrrConfig, vrrConfig);
Sasha McIntoshf08ed642024-10-24 15:43:07 -0400441 EXPECT_EQ(modes.front().hdrOutputType, hdrOutputType);
ramindani0cd1d8d2023-06-13 13:43:23 -0700442 EXPECT_EQ(modes.front().dpiX, kDpi);
443 EXPECT_EQ(modes.front().dpiY, kDpi);
ramindani355fca82023-12-18 12:07:34 -0800444
445 setVrrTimeoutHint(false);
446 modes = mHwc.getModes(info->id, kMaxFrameIntervalNs);
447 EXPECT_EQ(modes.front().vrrConfig->notifyExpectedPresentConfig, std::nullopt);
ramindani0cd1d8d2023-06-13 13:43:23 -0700448 }
449}
450
Leon Scroggins III959a7ff2023-02-07 11:24:25 -0500451TEST_F(HWComposerTest, onVsync) {
452 constexpr hal::HWDisplayId kHwcDisplayId = 1;
453 expectHotplugConnect(kHwcDisplayId);
454
455 const auto info = mHwc.onHotplug(kHwcDisplayId, hal::Connection::CONNECTED);
456 ASSERT_TRUE(info);
457
458 const auto physicalDisplayId = info->id;
459
460 // Deliberately chosen not to match DisplayData.lastPresentTimestamp's
461 // initial value.
462 constexpr nsecs_t kTimestamp = 1;
463 auto displayIdOpt = mHwc.onVsync(kHwcDisplayId, kTimestamp);
464 ASSERT_TRUE(displayIdOpt);
465 EXPECT_EQ(physicalDisplayId, displayIdOpt);
466
467 // Attempt to send the same time stamp again.
468 displayIdOpt = mHwc.onVsync(kHwcDisplayId, kTimestamp);
469 EXPECT_FALSE(displayIdOpt);
470}
471
472TEST_F(HWComposerTest, onVsyncInvalid) {
473 constexpr hal::HWDisplayId kInvalidHwcDisplayId = 2;
474 constexpr nsecs_t kTimestamp = 1;
475 const auto displayIdOpt = mHwc.onVsync(kInvalidHwcDisplayId, kTimestamp);
476 EXPECT_FALSE(displayIdOpt);
477}
478
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700479struct MockHWC2ComposerCallback final : StrictMock<HWC2::ComposerCallback> {
Huihong Luo791bef92023-09-19 21:18:45 -0700480 MOCK_METHOD(void, onComposerHalHotplugEvent, (hal::HWDisplayId, DisplayHotplugEvent),
481 (override));
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700482 MOCK_METHOD1(onComposerHalRefresh, void(hal::HWDisplayId));
483 MOCK_METHOD3(onComposerHalVsync,
484 void(hal::HWDisplayId, int64_t timestamp, std::optional<hal::VsyncPeriodNanos>));
485 MOCK_METHOD2(onComposerHalVsyncPeriodTimingChanged,
486 void(hal::HWDisplayId, const hal::VsyncPeriodChangeTimeline&));
487 MOCK_METHOD1(onComposerHalSeamlessPossible, void(hal::HWDisplayId));
Yichi Chen1a417af2022-01-21 15:29:52 +0800488 MOCK_METHOD1(onComposerHalVsyncIdle, void(hal::HWDisplayId));
ramindani12bfe6b2023-02-03 13:29:19 -0800489 MOCK_METHOD(void, onRefreshRateChangedDebug, (const RefreshRateChangedDebugData&), (override));
Huihong Luo817ebb82023-12-13 12:54:24 -0800490 MOCK_METHOD(void, onComposerHalHdcpLevelsChanged, (hal::HWDisplayId, const HdcpLevels&),
491 (override));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800492};
493
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700494struct HWComposerSetCallbackTest : HWComposerTest {
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700495 MockHWC2ComposerCallback mCallback;
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800496};
497
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700498TEST_F(HWComposerSetCallbackTest, loadsLayerMetadataSupport) {
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800499 const std::string kMetadata1Name = "com.example.metadata.1";
500 constexpr bool kMetadata1Mandatory = false;
501 const std::string kMetadata2Name = "com.example.metadata.2";
502 constexpr bool kMetadata2Mandatory = true;
503
Ady Abrahamde549d42022-01-26 19:19:17 -0800504 EXPECT_CALL(*mHal, getCapabilities()).WillOnce(Return(std::vector<aidl::Capability>{}));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800505 EXPECT_CALL(*mHal, getLayerGenericMetadataKeys(_))
Peiyong Line9d809e2020-04-14 13:10:48 -0700506 .WillOnce(DoAll(SetArgPointee<0>(std::vector<hal::LayerGenericMetadataKey>{
507 {kMetadata1Name, kMetadata1Mandatory},
508 {kMetadata2Name, kMetadata2Mandatory},
509 }),
Dominik Laskowski969cdcb2024-02-08 16:35:29 -0500510 Return(V2_4::Error::NONE)));
Sally Qibb866c12022-10-17 11:31:20 -0700511 EXPECT_CALL(*mHal, getOverlaySupport(_)).WillOnce(Return(HalError::NONE));
Kriti Dang674b9372022-11-18 10:58:44 +0100512 EXPECT_CALL(*mHal, getHdrConversionCapabilities(_)).WillOnce(Return(HalError::NONE));
Sally Qibb866c12022-10-17 11:31:20 -0700513
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800514 EXPECT_CALL(*mHal, registerCallback(_));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800515
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700516 mHwc.setCallback(mCallback);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800517
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700518 const auto& supported = mHwc.getSupportedLayerGenericMetadata();
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800519 EXPECT_EQ(2u, supported.size());
520 EXPECT_EQ(1u, supported.count(kMetadata1Name));
521 EXPECT_EQ(kMetadata1Mandatory, supported.find(kMetadata1Name)->second);
522 EXPECT_EQ(1u, supported.count(kMetadata2Name));
523 EXPECT_EQ(kMetadata2Mandatory, supported.find(kMetadata2Name)->second);
524}
525
Dominik Laskowski0deb06e2021-04-16 23:18:31 -0700526TEST_F(HWComposerSetCallbackTest, handlesUnsupportedCallToGetLayerGenericMetadataKeys) {
Ady Abrahamde549d42022-01-26 19:19:17 -0800527 EXPECT_CALL(*mHal, getCapabilities()).WillOnce(Return(std::vector<aidl::Capability>{}));
Dominik Laskowski969cdcb2024-02-08 16:35:29 -0500528 EXPECT_CALL(*mHal, getLayerGenericMetadataKeys(_)).WillOnce(Return(V2_4::Error::UNSUPPORTED));
Sally Qibb866c12022-10-17 11:31:20 -0700529 EXPECT_CALL(*mHal, getOverlaySupport(_)).WillOnce(Return(HalError::UNSUPPORTED));
Kriti Dang674b9372022-11-18 10:58:44 +0100530 EXPECT_CALL(*mHal, getHdrConversionCapabilities(_)).WillOnce(Return(HalError::UNSUPPORTED));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800531 EXPECT_CALL(*mHal, registerCallback(_));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800532
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700533 mHwc.setCallback(mCallback);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800534
Dominik Laskowskie2c5b0a2022-08-10 14:53:53 -0700535 const auto& supported = mHwc.getSupportedLayerGenericMetadata();
536 EXPECT_TRUE(supported.empty());
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800537}
538
539struct HWComposerLayerTest : public testing::Test {
Peiyong Line9d809e2020-04-14 13:10:48 -0700540 static constexpr hal::HWDisplayId kDisplayId = static_cast<hal::HWDisplayId>(1001);
541 static constexpr hal::HWLayerId kLayerId = static_cast<hal::HWLayerId>(1002);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800542
Ady Abrahamde549d42022-01-26 19:19:17 -0800543 HWComposerLayerTest(const std::unordered_set<aidl::Capability>& capabilities)
Lloyd Piquea516c002021-05-07 14:36:58 -0700544 : mCapabilies(capabilities) {
545 EXPECT_CALL(mDisplay, getId()).WillRepeatedly(Return(kDisplayId));
546 }
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800547
Lloyd Piquea516c002021-05-07 14:36:58 -0700548 ~HWComposerLayerTest() override {
549 EXPECT_CALL(mDisplay, onLayerDestroyed(kLayerId));
550 EXPECT_CALL(*mHal, destroyLayer(kDisplayId, kLayerId));
551 }
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800552
553 std::unique_ptr<Hwc2::mock::Composer> mHal{new StrictMock<Hwc2::mock::Composer>()};
Ady Abrahamde549d42022-01-26 19:19:17 -0800554 const std::unordered_set<aidl::Capability> mCapabilies;
Lloyd Piquea516c002021-05-07 14:36:58 -0700555 StrictMock<HWC2::mock::Display> mDisplay;
556 HWC2::impl::Layer mLayer{*mHal, mCapabilies, mDisplay, kLayerId};
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800557};
558
559struct HWComposerLayerGenericMetadataTest : public HWComposerLayerTest {
560 static const std::string kLayerGenericMetadata1Name;
561 static constexpr bool kLayerGenericMetadata1Mandatory = false;
562 static const std::vector<uint8_t> kLayerGenericMetadata1Value;
563 static const std::string kLayerGenericMetadata2Name;
564 static constexpr bool kLayerGenericMetadata2Mandatory = true;
565 static const std::vector<uint8_t> kLayerGenericMetadata2Value;
566
567 HWComposerLayerGenericMetadataTest() : HWComposerLayerTest({}) {}
568};
569
570const std::string HWComposerLayerGenericMetadataTest::kLayerGenericMetadata1Name =
571 "com.example.metadata.1";
572
573const std::vector<uint8_t> HWComposerLayerGenericMetadataTest::kLayerGenericMetadata1Value = {1u,
574 2u,
575 3u};
576
577const std::string HWComposerLayerGenericMetadataTest::kLayerGenericMetadata2Name =
578 "com.example.metadata.2";
579
580const std::vector<uint8_t> HWComposerLayerGenericMetadataTest::kLayerGenericMetadata2Value = {45u,
581 67u};
582
583TEST_F(HWComposerLayerGenericMetadataTest, forwardsSupportedMetadata) {
584 EXPECT_CALL(*mHal,
585 setLayerGenericMetadata(kDisplayId, kLayerId, kLayerGenericMetadata1Name,
586 kLayerGenericMetadata1Mandatory,
587 kLayerGenericMetadata1Value))
Dominik Laskowski969cdcb2024-02-08 16:35:29 -0500588 .WillOnce(Return(V2_4::Error::NONE));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800589 auto result = mLayer.setLayerGenericMetadata(kLayerGenericMetadata1Name,
590 kLayerGenericMetadata1Mandatory,
591 kLayerGenericMetadata1Value);
Peiyong Line9d809e2020-04-14 13:10:48 -0700592 EXPECT_EQ(hal::Error::NONE, result);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800593
594 EXPECT_CALL(*mHal,
595 setLayerGenericMetadata(kDisplayId, kLayerId, kLayerGenericMetadata2Name,
596 kLayerGenericMetadata2Mandatory,
597 kLayerGenericMetadata2Value))
Dominik Laskowski969cdcb2024-02-08 16:35:29 -0500598 .WillOnce(Return(V2_4::Error::UNSUPPORTED));
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800599 result = mLayer.setLayerGenericMetadata(kLayerGenericMetadata2Name,
600 kLayerGenericMetadata2Mandatory,
601 kLayerGenericMetadata2Value);
Peiyong Line9d809e2020-04-14 13:10:48 -0700602 EXPECT_EQ(hal::Error::UNSUPPORTED, result);
Lloyd Pique4603f3c2020-02-11 12:06:56 -0800603}
604
Dominik Laskowski3dce4f42021-03-08 20:48:28 -0800605} // namespace android