ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 1 | /** |
| 2 | * Copyright (c) 2022, 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 | |
ramindani | 458e53e | 2022-02-23 17:30:16 +0000 | [diff] [blame] | 17 | #include "VtsComposerClient.h" |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 18 | #include <aidlcommonsupport/NativeHandle.h> |
| 19 | #include <android-base/logging.h> |
| 20 | #include <log/log_main.h> |
| 21 | |
| 22 | #undef LOG_TAG |
| 23 | #define LOG_TAG "VtsComposerClient" |
| 24 | |
| 25 | using namespace std::chrono_literals; |
| 26 | |
| 27 | namespace aidl::android::hardware::graphics::composer3::vts { |
| 28 | |
| 29 | VtsComposerClient::VtsComposerClient(const std::string& name) { |
| 30 | SpAIBinder binder(AServiceManager_waitForService(name.c_str())); |
| 31 | ALOGE_IF(binder == nullptr, "Could not initialize the service binder"); |
| 32 | if (binder != nullptr) { |
| 33 | mComposer = IComposer::fromBinder(binder); |
| 34 | ALOGE_IF(mComposer == nullptr, "Failed to acquire the composer from the binder"); |
| 35 | } |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 36 | |
| 37 | const auto& [status, capabilities] = getCapabilities(); |
| 38 | EXPECT_TRUE(status.isOk()); |
| 39 | if (std::any_of(capabilities.begin(), capabilities.end(), [&](const Capability& cap) { |
| 40 | return cap == Capability::LAYER_LIFECYCLE_BATCH_COMMAND; |
| 41 | })) { |
| 42 | mSupportsBatchedCreateLayer = true; |
| 43 | } |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 44 | } |
| 45 | |
| 46 | ScopedAStatus VtsComposerClient::createClient() { |
| 47 | if (mComposer == nullptr) { |
| 48 | ALOGE("IComposer not initialized"); |
| 49 | return ScopedAStatus::fromServiceSpecificError(IComposerClient::INVALID_CONFIGURATION); |
| 50 | } |
| 51 | auto status = mComposer->createClient(&mComposerClient); |
| 52 | if (!status.isOk() || mComposerClient == nullptr) { |
| 53 | ALOGE("Failed to create client for IComposerClient with %s", |
| 54 | status.getDescription().c_str()); |
| 55 | return status; |
| 56 | } |
| 57 | mComposerCallback = SharedRefBase::make<GraphicsComposerCallback>(); |
| 58 | if (mComposerCallback == nullptr) { |
| 59 | ALOGE("Unable to create ComposerCallback"); |
| 60 | return ScopedAStatus::fromServiceSpecificError(IComposerClient::INVALID_CONFIGURATION); |
| 61 | } |
| 62 | return mComposerClient->registerCallback(mComposerCallback); |
| 63 | } |
| 64 | |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 65 | bool VtsComposerClient::tearDown(ComposerClientWriter* writer) { |
| 66 | return verifyComposerCallbackParams() && destroyAllLayers(writer); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 67 | } |
| 68 | |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 69 | std::pair<ScopedAStatus, int32_t> VtsComposerClient::getInterfaceVersion() const { |
Brian Lindahl | 78ff2d6 | 2022-12-18 11:21:41 -0700 | [diff] [blame] | 70 | int32_t version = 1; |
| 71 | auto status = mComposerClient->getInterfaceVersion(&version); |
| 72 | return {std::move(status), version}; |
| 73 | } |
| 74 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 75 | std::pair<ScopedAStatus, VirtualDisplay> VtsComposerClient::createVirtualDisplay( |
| 76 | int32_t width, int32_t height, PixelFormat pixelFormat, int32_t bufferSlotCount) { |
| 77 | VirtualDisplay outVirtualDisplay; |
| 78 | auto status = mComposerClient->createVirtualDisplay(width, height, pixelFormat, bufferSlotCount, |
| 79 | &outVirtualDisplay); |
| 80 | if (!status.isOk()) { |
| 81 | return {std::move(status), outVirtualDisplay}; |
| 82 | } |
| 83 | return {addDisplayToDisplayResources(outVirtualDisplay.display, /*isVirtual*/ true), |
| 84 | outVirtualDisplay}; |
| 85 | } |
| 86 | |
| 87 | ScopedAStatus VtsComposerClient::destroyVirtualDisplay(int64_t display) { |
| 88 | auto status = mComposerClient->destroyVirtualDisplay(display); |
| 89 | if (!status.isOk()) { |
| 90 | return status; |
| 91 | } |
| 92 | mDisplayResources.erase(display); |
| 93 | return status; |
| 94 | } |
| 95 | |
| 96 | std::pair<ScopedAStatus, int64_t> VtsComposerClient::createLayer(int64_t display, |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 97 | int32_t bufferSlotCount, |
| 98 | ComposerClientWriter* writer) { |
| 99 | if (mSupportsBatchedCreateLayer) { |
| 100 | int64_t layer = mNextLayerHandle++; |
| 101 | writer->setLayerLifecycleBatchCommandType(display, layer, |
| 102 | LayerLifecycleBatchCommandType::CREATE); |
| 103 | writer->setNewBufferSlotCount(display, layer, bufferSlotCount); |
| 104 | return {addLayerToDisplayResources(display, layer), layer}; |
| 105 | } |
| 106 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 107 | int64_t outLayer; |
| 108 | auto status = mComposerClient->createLayer(display, bufferSlotCount, &outLayer); |
| 109 | |
| 110 | if (!status.isOk()) { |
| 111 | return {std::move(status), outLayer}; |
| 112 | } |
| 113 | return {addLayerToDisplayResources(display, outLayer), outLayer}; |
| 114 | } |
| 115 | |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 116 | ScopedAStatus VtsComposerClient::destroyLayer(int64_t display, int64_t layer, |
| 117 | ComposerClientWriter* writer) { |
| 118 | if (mSupportsBatchedCreateLayer) { |
| 119 | writer->setLayerLifecycleBatchCommandType(display, layer, |
| 120 | LayerLifecycleBatchCommandType::DESTROY); |
| 121 | } else { |
| 122 | auto status = mComposerClient->destroyLayer(display, layer); |
| 123 | if (!status.isOk()) { |
| 124 | return status; |
| 125 | } |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 126 | } |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 127 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 128 | removeLayerFromDisplayResources(display, layer); |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 129 | return ScopedAStatus::ok(); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 130 | } |
| 131 | |
| 132 | std::pair<ScopedAStatus, int32_t> VtsComposerClient::getActiveConfig(int64_t display) { |
| 133 | int32_t outConfig; |
| 134 | return {mComposerClient->getActiveConfig(display, &outConfig), outConfig}; |
| 135 | } |
| 136 | |
| 137 | ScopedAStatus VtsComposerClient::setActiveConfig(VtsDisplay* vtsDisplay, int32_t config) { |
| 138 | auto status = mComposerClient->setActiveConfig(vtsDisplay->getDisplayId(), config); |
| 139 | if (!status.isOk()) { |
| 140 | return status; |
| 141 | } |
| 142 | return updateDisplayProperties(vtsDisplay, config); |
| 143 | } |
| 144 | |
ramindani | 9dd8c48 | 2023-02-21 18:07:56 -0800 | [diff] [blame] | 145 | ScopedAStatus VtsComposerClient::setPeakRefreshRateConfig(VtsDisplay* vtsDisplay) { |
| 146 | const auto displayId = vtsDisplay->getDisplayId(); |
| 147 | auto [activeStatus, activeConfig] = getActiveConfig(displayId); |
| 148 | EXPECT_TRUE(activeStatus.isOk()); |
| 149 | auto peakDisplayConfig = vtsDisplay->getDisplayConfig(activeConfig); |
| 150 | auto peakConfig = activeConfig; |
| 151 | |
| 152 | const auto displayConfigs = vtsDisplay->getDisplayConfigs(); |
| 153 | for (const auto [config, displayConfig] : displayConfigs) { |
| 154 | if (displayConfig.configGroup == peakDisplayConfig.configGroup && |
| 155 | displayConfig.vsyncPeriod < peakDisplayConfig.vsyncPeriod) { |
| 156 | peakDisplayConfig = displayConfig; |
| 157 | peakConfig = config; |
| 158 | } |
| 159 | } |
| 160 | return setActiveConfig(vtsDisplay, peakConfig); |
| 161 | } |
| 162 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 163 | std::pair<ScopedAStatus, int32_t> VtsComposerClient::getDisplayAttribute( |
| 164 | int64_t display, int32_t config, DisplayAttribute displayAttribute) { |
| 165 | int32_t outDisplayAttribute; |
| 166 | return {mComposerClient->getDisplayAttribute(display, config, displayAttribute, |
| 167 | &outDisplayAttribute), |
| 168 | outDisplayAttribute}; |
| 169 | } |
| 170 | |
| 171 | ScopedAStatus VtsComposerClient::setPowerMode(int64_t display, PowerMode powerMode) { |
| 172 | return mComposerClient->setPowerMode(display, powerMode); |
| 173 | } |
| 174 | |
| 175 | ScopedAStatus VtsComposerClient::setVsync(int64_t display, bool enable) { |
| 176 | return mComposerClient->setVsyncEnabled(display, enable); |
| 177 | } |
| 178 | |
| 179 | void VtsComposerClient::setVsyncAllowed(bool isAllowed) { |
| 180 | mComposerCallback->setVsyncAllowed(isAllowed); |
| 181 | } |
| 182 | |
| 183 | std::pair<ScopedAStatus, std::vector<float>> VtsComposerClient::getDataspaceSaturationMatrix( |
| 184 | Dataspace dataspace) { |
| 185 | std::vector<float> outMatrix; |
| 186 | return {mComposerClient->getDataspaceSaturationMatrix(dataspace, &outMatrix), outMatrix}; |
| 187 | } |
| 188 | |
| 189 | std::pair<ScopedAStatus, std::vector<CommandResultPayload>> VtsComposerClient::executeCommands( |
| 190 | const std::vector<DisplayCommand>& commands) { |
| 191 | std::vector<CommandResultPayload> outResultPayload; |
| 192 | return {mComposerClient->executeCommands(commands, &outResultPayload), |
| 193 | std::move(outResultPayload)}; |
| 194 | } |
| 195 | |
| 196 | std::optional<VsyncPeriodChangeTimeline> VtsComposerClient::takeLastVsyncPeriodChangeTimeline() { |
| 197 | return mComposerCallback->takeLastVsyncPeriodChangeTimeline(); |
| 198 | } |
| 199 | |
| 200 | ScopedAStatus VtsComposerClient::setContentType(int64_t display, ContentType contentType) { |
| 201 | return mComposerClient->setContentType(display, contentType); |
| 202 | } |
| 203 | |
| 204 | std::pair<ScopedAStatus, VsyncPeriodChangeTimeline> |
| 205 | VtsComposerClient::setActiveConfigWithConstraints(VtsDisplay* vtsDisplay, int32_t config, |
| 206 | const VsyncPeriodChangeConstraints& constraints) { |
| 207 | VsyncPeriodChangeTimeline outTimeline; |
| 208 | auto status = mComposerClient->setActiveConfigWithConstraints( |
| 209 | vtsDisplay->getDisplayId(), config, constraints, &outTimeline); |
| 210 | if (!status.isOk()) { |
| 211 | return {std::move(status), outTimeline}; |
| 212 | } |
| 213 | return {updateDisplayProperties(vtsDisplay, config), outTimeline}; |
| 214 | } |
| 215 | |
| 216 | std::pair<ScopedAStatus, std::vector<DisplayCapability>> VtsComposerClient::getDisplayCapabilities( |
| 217 | int64_t display) { |
| 218 | std::vector<DisplayCapability> outCapabilities; |
| 219 | return {mComposerClient->getDisplayCapabilities(display, &outCapabilities), outCapabilities}; |
| 220 | } |
| 221 | |
| 222 | ScopedAStatus VtsComposerClient::dumpDebugInfo() { |
Ady Abraham | 89d7270 | 2022-02-18 17:04:55 -0800 | [diff] [blame] | 223 | int pipefds[2]; |
| 224 | if (pipe(pipefds) < 0) { |
| 225 | return ScopedAStatus::fromServiceSpecificError(IComposer::EX_NO_RESOURCES); |
| 226 | } |
| 227 | |
| 228 | const auto status = mComposer->dump(pipefds[1], /*args*/ nullptr, /*numArgs*/ 0); |
| 229 | close(pipefds[0]); |
| 230 | close(pipefds[1]); |
| 231 | return ScopedAStatus::fromStatus(status); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 232 | } |
| 233 | |
| 234 | std::pair<ScopedAStatus, DisplayIdentification> VtsComposerClient::getDisplayIdentificationData( |
| 235 | int64_t display) { |
| 236 | DisplayIdentification outDisplayIdentification; |
| 237 | return {mComposerClient->getDisplayIdentificationData(display, &outDisplayIdentification), |
| 238 | outDisplayIdentification}; |
| 239 | } |
| 240 | |
| 241 | std::pair<ScopedAStatus, HdrCapabilities> VtsComposerClient::getHdrCapabilities(int64_t display) { |
| 242 | HdrCapabilities outHdrCapabilities; |
| 243 | return {mComposerClient->getHdrCapabilities(display, &outHdrCapabilities), outHdrCapabilities}; |
| 244 | } |
| 245 | |
| 246 | std::pair<ScopedAStatus, std::vector<PerFrameMetadataKey>> |
| 247 | VtsComposerClient::getPerFrameMetadataKeys(int64_t display) { |
| 248 | std::vector<PerFrameMetadataKey> outPerFrameMetadataKeys; |
| 249 | return {mComposerClient->getPerFrameMetadataKeys(display, &outPerFrameMetadataKeys), |
| 250 | outPerFrameMetadataKeys}; |
| 251 | } |
| 252 | |
| 253 | std::pair<ScopedAStatus, ReadbackBufferAttributes> VtsComposerClient::getReadbackBufferAttributes( |
| 254 | int64_t display) { |
| 255 | ReadbackBufferAttributes outReadbackBufferAttributes; |
| 256 | return {mComposerClient->getReadbackBufferAttributes(display, &outReadbackBufferAttributes), |
| 257 | outReadbackBufferAttributes}; |
| 258 | } |
| 259 | |
| 260 | ScopedAStatus VtsComposerClient::setReadbackBuffer(int64_t display, const native_handle_t* buffer, |
| 261 | const ScopedFileDescriptor& releaseFence) { |
| 262 | return mComposerClient->setReadbackBuffer(display, ::android::dupToAidl(buffer), releaseFence); |
| 263 | } |
| 264 | |
| 265 | std::pair<ScopedAStatus, ScopedFileDescriptor> VtsComposerClient::getReadbackBufferFence( |
| 266 | int64_t display) { |
| 267 | ScopedFileDescriptor outReleaseFence; |
| 268 | return {mComposerClient->getReadbackBufferFence(display, &outReleaseFence), |
| 269 | std::move(outReleaseFence)}; |
| 270 | } |
| 271 | |
| 272 | std::pair<ScopedAStatus, std::vector<ColorMode>> VtsComposerClient::getColorModes(int64_t display) { |
| 273 | std::vector<ColorMode> outColorModes; |
| 274 | return {mComposerClient->getColorModes(display, &outColorModes), outColorModes}; |
| 275 | } |
| 276 | |
| 277 | std::pair<ScopedAStatus, std::vector<RenderIntent>> VtsComposerClient::getRenderIntents( |
| 278 | int64_t display, ColorMode colorMode) { |
| 279 | std::vector<RenderIntent> outRenderIntents; |
| 280 | return {mComposerClient->getRenderIntents(display, colorMode, &outRenderIntents), |
| 281 | outRenderIntents}; |
| 282 | } |
| 283 | |
| 284 | ScopedAStatus VtsComposerClient::setColorMode(int64_t display, ColorMode colorMode, |
| 285 | RenderIntent renderIntent) { |
| 286 | return mComposerClient->setColorMode(display, colorMode, renderIntent); |
| 287 | } |
| 288 | |
| 289 | std::pair<ScopedAStatus, DisplayContentSamplingAttributes> |
| 290 | VtsComposerClient::getDisplayedContentSamplingAttributes(int64_t display) { |
| 291 | DisplayContentSamplingAttributes outAttributes; |
| 292 | return {mComposerClient->getDisplayedContentSamplingAttributes(display, &outAttributes), |
| 293 | outAttributes}; |
| 294 | } |
| 295 | |
| 296 | ScopedAStatus VtsComposerClient::setDisplayedContentSamplingEnabled( |
| 297 | int64_t display, bool isEnabled, FormatColorComponent formatColorComponent, |
| 298 | int64_t maxFrames) { |
| 299 | return mComposerClient->setDisplayedContentSamplingEnabled(display, isEnabled, |
| 300 | formatColorComponent, maxFrames); |
| 301 | } |
| 302 | |
| 303 | std::pair<ScopedAStatus, DisplayContentSample> VtsComposerClient::getDisplayedContentSample( |
| 304 | int64_t display, int64_t maxFrames, int64_t timestamp) { |
| 305 | DisplayContentSample outDisplayContentSample; |
| 306 | return {mComposerClient->getDisplayedContentSample(display, maxFrames, timestamp, |
| 307 | &outDisplayContentSample), |
| 308 | outDisplayContentSample}; |
| 309 | } |
| 310 | |
| 311 | std::pair<ScopedAStatus, DisplayConnectionType> VtsComposerClient::getDisplayConnectionType( |
| 312 | int64_t display) { |
| 313 | DisplayConnectionType outDisplayConnectionType; |
| 314 | return {mComposerClient->getDisplayConnectionType(display, &outDisplayConnectionType), |
| 315 | outDisplayConnectionType}; |
| 316 | } |
| 317 | |
| 318 | std::pair<ScopedAStatus, std::vector<int32_t>> VtsComposerClient::getDisplayConfigs( |
| 319 | int64_t display) { |
| 320 | std::vector<int32_t> outConfigs; |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 321 | if (!getDisplayConfigurationSupported()) { |
| 322 | return {mComposerClient->getDisplayConfigs(display, &outConfigs), outConfigs}; |
| 323 | } |
| 324 | |
| 325 | auto [status, configs] = getDisplayConfigurations(display); |
| 326 | if (!status.isOk()) { |
| 327 | return {std::move(status), outConfigs}; |
| 328 | } |
| 329 | for (const auto& config : configs) { |
| 330 | outConfigs.emplace_back(config.configId); |
| 331 | } |
| 332 | return {std::move(status), outConfigs}; |
| 333 | } |
| 334 | |
| 335 | std::pair<ScopedAStatus, std::vector<DisplayConfiguration>> |
| 336 | VtsComposerClient::getDisplayConfigurations(int64_t display) { |
| 337 | std::vector<DisplayConfiguration> outConfigs; |
ramindani | 2704c5f | 2023-07-18 20:53:24 -0700 | [diff] [blame] | 338 | return {mComposerClient->getDisplayConfigurations(display, kMaxFrameIntervalNs, &outConfigs), |
| 339 | outConfigs}; |
| 340 | } |
| 341 | |
| 342 | ScopedAStatus VtsComposerClient::notifyExpectedPresent(int64_t display, |
| 343 | ClockMonotonicTimestamp expectedPresentTime, |
| 344 | int frameIntervalNs) { |
| 345 | return mComposerClient->notifyExpectedPresent(display, expectedPresentTime, frameIntervalNs); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 346 | } |
| 347 | |
| 348 | std::pair<ScopedAStatus, int32_t> VtsComposerClient::getDisplayVsyncPeriod(int64_t display) { |
| 349 | int32_t outVsyncPeriodNanos; |
| 350 | return {mComposerClient->getDisplayVsyncPeriod(display, &outVsyncPeriodNanos), |
| 351 | outVsyncPeriodNanos}; |
| 352 | } |
| 353 | |
| 354 | ScopedAStatus VtsComposerClient::setAutoLowLatencyMode(int64_t display, bool isEnabled) { |
| 355 | return mComposerClient->setAutoLowLatencyMode(display, isEnabled); |
| 356 | } |
| 357 | |
| 358 | std::pair<ScopedAStatus, std::vector<ContentType>> VtsComposerClient::getSupportedContentTypes( |
| 359 | int64_t display) { |
| 360 | std::vector<ContentType> outContentTypes; |
| 361 | return {mComposerClient->getSupportedContentTypes(display, &outContentTypes), outContentTypes}; |
| 362 | } |
| 363 | |
Leon Scroggins III | de05758 | 2022-01-13 12:26:00 -0500 | [diff] [blame] | 364 | std::pair<ScopedAStatus, std::optional<DisplayDecorationSupport>> |
| 365 | VtsComposerClient::getDisplayDecorationSupport(int64_t display) { |
| 366 | std::optional<DisplayDecorationSupport> outSupport; |
| 367 | return {mComposerClient->getDisplayDecorationSupport(display, &outSupport), outSupport}; |
| 368 | } |
| 369 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 370 | std::pair<ScopedAStatus, int32_t> VtsComposerClient::getMaxVirtualDisplayCount() { |
| 371 | int32_t outMaxVirtualDisplayCount; |
| 372 | return {mComposerClient->getMaxVirtualDisplayCount(&outMaxVirtualDisplayCount), |
| 373 | outMaxVirtualDisplayCount}; |
| 374 | } |
| 375 | |
| 376 | std::pair<ScopedAStatus, std::string> VtsComposerClient::getDisplayName(int64_t display) { |
| 377 | std::string outDisplayName; |
| 378 | return {mComposerClient->getDisplayName(display, &outDisplayName), outDisplayName}; |
| 379 | } |
| 380 | |
| 381 | ScopedAStatus VtsComposerClient::setClientTargetSlotCount(int64_t display, |
| 382 | int32_t bufferSlotCount) { |
| 383 | return mComposerClient->setClientTargetSlotCount(display, bufferSlotCount); |
| 384 | } |
| 385 | |
| 386 | std::pair<ScopedAStatus, std::vector<Capability>> VtsComposerClient::getCapabilities() { |
| 387 | std::vector<Capability> outCapabilities; |
| 388 | return {mComposer->getCapabilities(&outCapabilities), outCapabilities}; |
| 389 | } |
| 390 | |
| 391 | ScopedAStatus VtsComposerClient::setBootDisplayConfig(int64_t display, int32_t config) { |
| 392 | return mComposerClient->setBootDisplayConfig(display, config); |
| 393 | } |
| 394 | |
| 395 | ScopedAStatus VtsComposerClient::clearBootDisplayConfig(int64_t display) { |
| 396 | return mComposerClient->clearBootDisplayConfig(display); |
| 397 | } |
| 398 | |
| 399 | std::pair<ScopedAStatus, int32_t> VtsComposerClient::getPreferredBootDisplayConfig( |
| 400 | int64_t display) { |
| 401 | int32_t outConfig; |
| 402 | return {mComposerClient->getPreferredBootDisplayConfig(display, &outConfig), outConfig}; |
| 403 | } |
| 404 | |
Kriti Dang | 3793ebd | 2022-12-05 13:03:49 +0100 | [diff] [blame] | 405 | std::pair<ScopedAStatus, std::vector<common::HdrConversionCapability>> |
| 406 | VtsComposerClient::getHdrConversionCapabilities() { |
| 407 | std::vector<common::HdrConversionCapability> hdrConversionCapability; |
| 408 | return {mComposerClient->getHdrConversionCapabilities(&hdrConversionCapability), |
| 409 | hdrConversionCapability}; |
| 410 | } |
| 411 | |
Kriti Dang | 3762276 | 2023-02-09 18:15:37 +0100 | [diff] [blame] | 412 | std::pair<ScopedAStatus, common::Hdr> VtsComposerClient::setHdrConversionStrategy( |
Kriti Dang | 3793ebd | 2022-12-05 13:03:49 +0100 | [diff] [blame] | 413 | const common::HdrConversionStrategy& conversionStrategy) { |
Kriti Dang | 3762276 | 2023-02-09 18:15:37 +0100 | [diff] [blame] | 414 | common::Hdr preferredHdrOutputType; |
| 415 | return {mComposerClient->setHdrConversionStrategy(conversionStrategy, &preferredHdrOutputType), |
| 416 | preferredHdrOutputType}; |
Kriti Dang | 3793ebd | 2022-12-05 13:03:49 +0100 | [diff] [blame] | 417 | } |
| 418 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 419 | std::pair<ScopedAStatus, common::Transform> VtsComposerClient::getDisplayPhysicalOrientation( |
| 420 | int64_t display) { |
| 421 | common::Transform outDisplayOrientation; |
| 422 | return {mComposerClient->getDisplayPhysicalOrientation(display, &outDisplayOrientation), |
| 423 | outDisplayOrientation}; |
| 424 | } |
| 425 | |
Sally Qi | 2600d34 | 2022-08-16 12:46:17 -0700 | [diff] [blame] | 426 | std::pair<ScopedAStatus, composer3::OverlayProperties> VtsComposerClient::getOverlaySupport() { |
| 427 | OverlayProperties properties; |
| 428 | return {mComposerClient->getOverlaySupport(&properties), properties}; |
| 429 | } |
| 430 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 431 | ScopedAStatus VtsComposerClient::setIdleTimerEnabled(int64_t display, int32_t timeoutMs) { |
| 432 | return mComposerClient->setIdleTimerEnabled(display, timeoutMs); |
| 433 | } |
| 434 | |
| 435 | int32_t VtsComposerClient::getVsyncIdleCount() { |
| 436 | return mComposerCallback->getVsyncIdleCount(); |
| 437 | } |
| 438 | |
| 439 | int64_t VtsComposerClient::getVsyncIdleTime() { |
| 440 | return mComposerCallback->getVsyncIdleTime(); |
| 441 | } |
| 442 | |
ramindani | 9dd8c48 | 2023-02-21 18:07:56 -0800 | [diff] [blame] | 443 | ndk::ScopedAStatus VtsComposerClient::setRefreshRateChangedCallbackDebugEnabled(int64_t display, |
| 444 | bool enabled) { |
| 445 | mComposerCallback->setRefreshRateChangedDebugDataEnabledCallbackAllowed(enabled); |
| 446 | return mComposerClient->setRefreshRateChangedCallbackDebugEnabled(display, enabled); |
| 447 | } |
| 448 | |
| 449 | std::vector<RefreshRateChangedDebugData> |
| 450 | VtsComposerClient::takeListOfRefreshRateChangedDebugData() { |
| 451 | return mComposerCallback->takeListOfRefreshRateChangedDebugData(); |
ramindani | 8345034 | 2023-02-03 12:52:08 -0800 | [diff] [blame] | 452 | } |
| 453 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 454 | int64_t VtsComposerClient::getInvalidDisplayId() { |
| 455 | // returns an invalid display id (one that has not been registered to a |
| 456 | // display. Currently assuming that a device will never have close to |
| 457 | // std::numeric_limit<uint64_t>::max() displays registered while running tests |
| 458 | int64_t id = std::numeric_limits<int64_t>::max(); |
| 459 | std::vector<int64_t> displays = mComposerCallback->getDisplays(); |
| 460 | while (id > 0) { |
| 461 | if (std::none_of(displays.begin(), displays.end(), |
| 462 | [id](const auto& display) { return id == display; })) { |
| 463 | return id; |
| 464 | } |
| 465 | id--; |
| 466 | } |
| 467 | |
| 468 | // Although 0 could be an invalid display, a return value of 0 |
| 469 | // from getInvalidDisplayId means all other ids are in use, a condition which |
| 470 | // we are assuming a device will never have |
| 471 | EXPECT_NE(0, id); |
| 472 | return id; |
| 473 | } |
| 474 | |
| 475 | std::pair<ScopedAStatus, std::vector<VtsDisplay>> VtsComposerClient::getDisplays() { |
| 476 | while (true) { |
| 477 | // Sleep for a small period of time to allow all built-in displays |
| 478 | // to post hotplug events |
| 479 | std::this_thread::sleep_for(5ms); |
| 480 | std::vector<int64_t> displays = mComposerCallback->getDisplays(); |
| 481 | if (displays.empty()) { |
| 482 | continue; |
| 483 | } |
| 484 | |
| 485 | std::vector<VtsDisplay> vtsDisplays; |
| 486 | vtsDisplays.reserve(displays.size()); |
| 487 | for (int64_t display : displays) { |
| 488 | auto vtsDisplay = VtsDisplay{display}; |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 489 | if (getDisplayConfigurationSupported()) { |
| 490 | auto [status, configs] = getDisplayConfigurations(display); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 491 | if (!status.isOk()) { |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 492 | ALOGE("Unable to get the displays for test, failed to get the DisplayConfigs " |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 493 | "for display %" PRId64, |
| 494 | display); |
| 495 | return {std::move(status), vtsDisplays}; |
| 496 | } |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 497 | addDisplayConfigs(&vtsDisplay, configs); |
| 498 | } else { |
| 499 | auto [status, configs] = getDisplayConfigs(display); |
| 500 | if (!status.isOk()) { |
| 501 | ALOGE("Unable to get the displays for test, failed to get the configs " |
| 502 | "for display %" PRId64, |
| 503 | display); |
| 504 | return {std::move(status), vtsDisplays}; |
| 505 | } |
| 506 | for (int config : configs) { |
| 507 | status = addDisplayConfigLegacy(&vtsDisplay, config); |
| 508 | if (!status.isOk()) { |
| 509 | ALOGE("Unable to get the displays for test, failed to add config " |
| 510 | "for display %" PRId64, |
| 511 | display); |
| 512 | return {std::move(status), vtsDisplays}; |
| 513 | } |
| 514 | } |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 515 | } |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 516 | auto activeConfig = getActiveConfig(display); |
| 517 | if (!activeConfig.first.isOk()) { |
HyunKyoung | 4775bdc | 2022-11-09 20:43:34 +0900 | [diff] [blame] | 518 | ALOGE("Unable to get the displays for test, failed to get active config " |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 519 | "for display %" PRId64, |
| 520 | display); |
| 521 | return {std::move(activeConfig.first), vtsDisplays}; |
HyunKyoung | 4775bdc | 2022-11-09 20:43:34 +0900 | [diff] [blame] | 522 | } |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 523 | auto status = updateDisplayProperties(&vtsDisplay, activeConfig.second); |
HyunKyoung | 4775bdc | 2022-11-09 20:43:34 +0900 | [diff] [blame] | 524 | if (!status.isOk()) { |
| 525 | ALOGE("Unable to get the displays for test, " |
| 526 | "failed to update the properties " |
| 527 | "for display %" PRId64, |
| 528 | display); |
| 529 | return {std::move(status), vtsDisplays}; |
| 530 | } |
| 531 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 532 | vtsDisplays.emplace_back(vtsDisplay); |
| 533 | addDisplayToDisplayResources(display, /*isVirtual*/ false); |
| 534 | } |
| 535 | |
| 536 | return {ScopedAStatus::ok(), vtsDisplays}; |
| 537 | } |
| 538 | } |
| 539 | |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 540 | void VtsComposerClient::addDisplayConfigs(VtsDisplay* vtsDisplay, |
| 541 | const std::vector<DisplayConfiguration>& configs) { |
| 542 | for (const auto& config : configs) { |
ramindani | a2a6dea | 2023-12-11 13:42:36 -0800 | [diff] [blame] | 543 | vtsDisplay->addDisplayConfig(config.configId, |
| 544 | {config.vsyncPeriod, config.configGroup, config.vrrConfig}); |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 545 | } |
| 546 | } |
| 547 | |
| 548 | ScopedAStatus VtsComposerClient::addDisplayConfigLegacy(VtsDisplay* vtsDisplay, int32_t config) { |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 549 | const auto vsyncPeriod = |
| 550 | getDisplayAttribute(vtsDisplay->getDisplayId(), config, DisplayAttribute::VSYNC_PERIOD); |
| 551 | const auto configGroup = |
| 552 | getDisplayAttribute(vtsDisplay->getDisplayId(), config, DisplayAttribute::CONFIG_GROUP); |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 553 | if (vsyncPeriod.first.isOk() && configGroup.first.isOk()) { |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 554 | vtsDisplay->addDisplayConfig(config, {vsyncPeriod.second, configGroup.second}); |
| 555 | return ScopedAStatus::ok(); |
| 556 | } |
| 557 | |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 558 | LOG(ERROR) << "Failed to update display property vsync: " << vsyncPeriod.first.isOk() |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 559 | << ", config: " << configGroup.first.isOk(); |
| 560 | return ScopedAStatus::fromServiceSpecificError(IComposerClient::EX_BAD_CONFIG); |
| 561 | } |
| 562 | |
HyunKyoung | 4775bdc | 2022-11-09 20:43:34 +0900 | [diff] [blame] | 563 | ScopedAStatus VtsComposerClient::updateDisplayProperties(VtsDisplay* vtsDisplay, int32_t config) { |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 564 | if (getDisplayConfigurationSupported()) { |
| 565 | auto [status, configs] = getDisplayConfigurations(vtsDisplay->getDisplayId()); |
| 566 | if (status.isOk()) { |
| 567 | for (const auto& displayConfig : configs) { |
| 568 | if (displayConfig.configId == config) { |
| 569 | vtsDisplay->setDimensions(displayConfig.width, displayConfig.height); |
| 570 | return ScopedAStatus::ok(); |
| 571 | } |
| 572 | } |
| 573 | } |
| 574 | LOG(ERROR) << "Failed to update display property with DisplayConfig"; |
| 575 | } else { |
| 576 | const auto width = |
| 577 | getDisplayAttribute(vtsDisplay->getDisplayId(), config, DisplayAttribute::WIDTH); |
| 578 | const auto height = |
| 579 | getDisplayAttribute(vtsDisplay->getDisplayId(), config, DisplayAttribute::HEIGHT); |
| 580 | if (width.first.isOk() && height.first.isOk()) { |
| 581 | vtsDisplay->setDimensions(width.second, height.second); |
| 582 | return ScopedAStatus::ok(); |
| 583 | } |
HyunKyoung | 4775bdc | 2022-11-09 20:43:34 +0900 | [diff] [blame] | 584 | |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 585 | LOG(ERROR) << "Failed to update display property for width: " << width.first.isOk() |
| 586 | << ", height: " << height.first.isOk(); |
| 587 | } |
HyunKyoung | 4775bdc | 2022-11-09 20:43:34 +0900 | [diff] [blame] | 588 | return ScopedAStatus::fromServiceSpecificError(IComposerClient::EX_BAD_CONFIG); |
| 589 | } |
| 590 | |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 591 | ScopedAStatus VtsComposerClient::addDisplayToDisplayResources(int64_t display, bool isVirtual) { |
| 592 | if (mDisplayResources.insert({display, DisplayResource(isVirtual)}).second) { |
| 593 | return ScopedAStatus::ok(); |
| 594 | } |
| 595 | |
| 596 | ALOGE("Duplicate display id %" PRId64, display); |
| 597 | return ScopedAStatus::fromServiceSpecificError(IComposerClient::EX_BAD_DISPLAY); |
| 598 | } |
| 599 | |
| 600 | ScopedAStatus VtsComposerClient::addLayerToDisplayResources(int64_t display, int64_t layer) { |
| 601 | auto resource = mDisplayResources.find(display); |
| 602 | if (resource == mDisplayResources.end()) { |
| 603 | resource = mDisplayResources.insert({display, DisplayResource(false)}).first; |
| 604 | } |
| 605 | |
| 606 | if (!resource->second.layers.insert(layer).second) { |
| 607 | ALOGE("Duplicate layer id %" PRId64, layer); |
| 608 | return ScopedAStatus::fromServiceSpecificError(IComposerClient::EX_BAD_LAYER); |
| 609 | } |
| 610 | return ScopedAStatus::ok(); |
| 611 | } |
| 612 | |
| 613 | void VtsComposerClient::removeLayerFromDisplayResources(int64_t display, int64_t layer) { |
| 614 | auto resource = mDisplayResources.find(display); |
| 615 | if (resource != mDisplayResources.end()) { |
| 616 | resource->second.layers.erase(layer); |
| 617 | } |
| 618 | } |
| 619 | |
| 620 | bool VtsComposerClient::verifyComposerCallbackParams() { |
| 621 | bool isValid = true; |
| 622 | if (mComposerCallback != nullptr) { |
| 623 | if (mComposerCallback->getInvalidHotplugCount() != 0) { |
| 624 | ALOGE("Invalid hotplug count"); |
| 625 | isValid = false; |
| 626 | } |
| 627 | if (mComposerCallback->getInvalidRefreshCount() != 0) { |
| 628 | ALOGE("Invalid refresh count"); |
| 629 | isValid = false; |
| 630 | } |
| 631 | if (mComposerCallback->getInvalidVsyncCount() != 0) { |
| 632 | ALOGE("Invalid vsync count"); |
| 633 | isValid = false; |
| 634 | } |
| 635 | if (mComposerCallback->getInvalidVsyncPeriodChangeCount() != 0) { |
| 636 | ALOGE("Invalid vsync period change count"); |
| 637 | isValid = false; |
| 638 | } |
| 639 | if (mComposerCallback->getInvalidSeamlessPossibleCount() != 0) { |
| 640 | ALOGE("Invalid seamless possible count"); |
| 641 | isValid = false; |
| 642 | } |
ramindani | 9dd8c48 | 2023-02-21 18:07:56 -0800 | [diff] [blame] | 643 | if (mComposerCallback->getInvalidRefreshRateDebugEnabledCallbackCount() != 0) { |
| 644 | ALOGE("Invalid refresh rate debug enabled callback count"); |
| 645 | isValid = false; |
| 646 | } |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 647 | } |
| 648 | return isValid; |
| 649 | } |
| 650 | |
ramindani | 02fe477 | 2023-06-30 18:04:41 -0700 | [diff] [blame] | 651 | bool VtsComposerClient::getDisplayConfigurationSupported() const { |
| 652 | auto [status, interfaceVersion] = getInterfaceVersion(); |
| 653 | EXPECT_TRUE(status.isOk()); |
| 654 | // getDisplayConfigurations api is supported starting interface version 3 |
| 655 | return interfaceVersion >= 3; |
| 656 | } |
| 657 | |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 658 | bool VtsComposerClient::destroyAllLayers(ComposerClientWriter* writer) { |
JihCheng Chiu | d0dbe4e | 2022-06-11 01:53:11 +0800 | [diff] [blame] | 659 | std::unordered_map<int64_t, DisplayResource> physicalDisplays; |
| 660 | while (!mDisplayResources.empty()) { |
| 661 | const auto& it = mDisplayResources.begin(); |
| 662 | const auto& [display, resource] = *it; |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 663 | |
JihCheng Chiu | d0dbe4e | 2022-06-11 01:53:11 +0800 | [diff] [blame] | 664 | while (!resource.layers.empty()) { |
| 665 | auto layer = *resource.layers.begin(); |
Ady Abraham | a00d246 | 2023-12-26 14:21:20 -0800 | [diff] [blame^] | 666 | const auto status = destroyLayer(display, layer, writer); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 667 | if (!status.isOk()) { |
| 668 | ALOGE("Unable to destroy all the layers, failed at layer %" PRId64 " with error %s", |
| 669 | layer, status.getDescription().c_str()); |
| 670 | return false; |
| 671 | } |
| 672 | } |
| 673 | |
| 674 | if (resource.isVirtual) { |
| 675 | const auto status = destroyVirtualDisplay(display); |
| 676 | if (!status.isOk()) { |
| 677 | ALOGE("Unable to destroy the display %" PRId64 " failed with error %s", display, |
| 678 | status.getDescription().c_str()); |
| 679 | return false; |
| 680 | } |
JihCheng Chiu | d0dbe4e | 2022-06-11 01:53:11 +0800 | [diff] [blame] | 681 | } else { |
| 682 | auto extractIter = mDisplayResources.extract(it); |
| 683 | physicalDisplays.insert(std::move(extractIter)); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 684 | } |
| 685 | } |
JihCheng Chiu | d0dbe4e | 2022-06-11 01:53:11 +0800 | [diff] [blame] | 686 | mDisplayResources.swap(physicalDisplays); |
ramindani | edf3ef9 | 2022-01-07 00:04:23 +0000 | [diff] [blame] | 687 | mDisplayResources.clear(); |
| 688 | return true; |
| 689 | } |
Leon Scroggins III | de05758 | 2022-01-13 12:26:00 -0500 | [diff] [blame] | 690 | } // namespace aidl::android::hardware::graphics::composer3::vts |