| /* |
| * Copyright 2018 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| // TODO(b/129481165): remove the #pragma below and fix conversion issues |
| #include "renderengine/ExternalTexture.h" |
| #pragma clang diagnostic push |
| #pragma clang diagnostic ignored "-Wconversion" |
| #pragma clang diagnostic ignored "-Wextra" |
| |
| #undef LOG_TAG |
| #define LOG_TAG "CompositionTest" |
| |
| #include <compositionengine/Display.h> |
| #include <compositionengine/mock/DisplaySurface.h> |
| #include <ftl/future.h> |
| #include <gmock/gmock.h> |
| #include <gtest/gtest.h> |
| #include <gui/IProducerListener.h> |
| #include <gui/LayerMetadata.h> |
| #include <log/log.h> |
| #include <renderengine/mock/FakeExternalTexture.h> |
| #include <renderengine/mock/RenderEngine.h> |
| #include <system/window.h> |
| #include <utils/String8.h> |
| |
| #include "DisplayRenderArea.h" |
| #include "Layer.h" |
| #include "TestableSurfaceFlinger.h" |
| #include "mock/DisplayHardware/MockComposer.h" |
| #include "mock/DisplayHardware/MockPowerAdvisor.h" |
| #include "mock/MockEventThread.h" |
| #include "mock/MockTimeStats.h" |
| #include "mock/MockVsyncController.h" |
| #include "mock/system/window/MockNativeWindow.h" |
| |
| namespace android { |
| namespace { |
| |
| namespace hal = android::hardware::graphics::composer::hal; |
| |
| using hal::Error; |
| using hal::IComposer; |
| using hal::IComposerClient; |
| using hal::PowerMode; |
| using hal::Transform; |
| |
| using aidl::android::hardware::graphics::composer3::Capability; |
| |
| using testing::_; |
| using testing::AtLeast; |
| using testing::DoAll; |
| using testing::IsNull; |
| using testing::Mock; |
| using testing::Return; |
| using testing::ReturnRef; |
| using testing::SetArgPointee; |
| |
| using FakeHwcDisplayInjector = TestableSurfaceFlinger::FakeHwcDisplayInjector; |
| using FakeDisplayDeviceInjector = TestableSurfaceFlinger::FakeDisplayDeviceInjector; |
| |
| constexpr hal::HWDisplayId HWC_DISPLAY = FakeHwcDisplayInjector::DEFAULT_HWC_DISPLAY_ID; |
| constexpr hal::HWLayerId HWC_LAYER = 5000; |
| constexpr Transform DEFAULT_TRANSFORM = static_cast<Transform>(0); |
| |
| constexpr PhysicalDisplayId DEFAULT_DISPLAY_ID = PhysicalDisplayId::fromPort(42u); |
| constexpr int DEFAULT_DISPLAY_WIDTH = 1920; |
| constexpr int DEFAULT_DISPLAY_HEIGHT = 1024; |
| |
| constexpr int DEFAULT_TEXTURE_ID = 6000; |
| constexpr ui::LayerStack LAYER_STACK{7000u}; |
| |
| constexpr int DEFAULT_DISPLAY_MAX_LUMINANCE = 500; |
| |
| constexpr int DEFAULT_SIDEBAND_STREAM = 51; |
| |
| MATCHER(IsIdentityMatrix, "") { |
| constexpr auto kIdentity = mat4(); |
| return (mat4(arg) == kIdentity); |
| } |
| |
| class CompositionTest : public testing::Test { |
| public: |
| CompositionTest() { |
| const ::testing::TestInfo* const test_info = |
| ::testing::UnitTest::GetInstance()->current_test_info(); |
| ALOGD("**** Setting up for %s.%s\n", test_info->test_case_name(), test_info->name()); |
| |
| mFlinger.setupMockScheduler({.displayId = DEFAULT_DISPLAY_ID}); |
| |
| EXPECT_CALL(*mNativeWindow, query(NATIVE_WINDOW_WIDTH, _)) |
| .WillRepeatedly(DoAll(SetArgPointee<1>(DEFAULT_DISPLAY_WIDTH), Return(0))); |
| EXPECT_CALL(*mNativeWindow, query(NATIVE_WINDOW_HEIGHT, _)) |
| .WillRepeatedly(DoAll(SetArgPointee<1>(DEFAULT_DISPLAY_HEIGHT), Return(0))); |
| |
| mFlinger.setupRenderEngine(std::unique_ptr<renderengine::RenderEngine>(mRenderEngine)); |
| mFlinger.setupTimeStats(std::shared_ptr<TimeStats>(mTimeStats)); |
| |
| mComposer = new Hwc2::mock::Composer(); |
| mPowerAdvisor = new Hwc2::mock::PowerAdvisor(); |
| mFlinger.setupComposer(std::unique_ptr<Hwc2::Composer>(mComposer)); |
| mFlinger.setupPowerAdvisor(std::unique_ptr<Hwc2::PowerAdvisor>(mPowerAdvisor)); |
| mFlinger.mutableMaxRenderTargetSize() = 16384; |
| } |
| |
| ~CompositionTest() { |
| const ::testing::TestInfo* const test_info = |
| ::testing::UnitTest::GetInstance()->current_test_info(); |
| ALOGD("**** Tearing down after %s.%s\n", test_info->test_case_name(), test_info->name()); |
| } |
| |
| void setupForceGeometryDirty() { |
| // TODO: This requires the visible region and other related |
| // state to be set, and is problematic for BufferLayers since they are |
| // not visible without a buffer (and setting up a buffer looks like a |
| // pain) |
| // mFlinger.mutableVisibleRegionsDirty() = true; |
| |
| mFlinger.mutableGeometryDirty() = true; |
| } |
| |
| template <typename Case> |
| void displayRefreshCompositionDirtyGeometry(); |
| |
| template <typename Case> |
| void displayRefreshCompositionDirtyFrame(); |
| |
| template <typename Case> |
| void captureScreenComposition(); |
| |
| std::unordered_set<Capability> mDefaultCapabilities = {Capability::SIDEBAND_STREAM}; |
| |
| bool mDisplayOff = false; |
| TestableSurfaceFlinger mFlinger; |
| sp<DisplayDevice> mDisplay; |
| sp<compositionengine::mock::DisplaySurface> mDisplaySurface = |
| sp<compositionengine::mock::DisplaySurface>::make(); |
| sp<mock::NativeWindow> mNativeWindow = sp<mock::NativeWindow>::make(); |
| std::vector<sp<Layer>> mAuxiliaryLayers; |
| |
| sp<GraphicBuffer> mBuffer = |
| sp<GraphicBuffer>::make(1u, 1u, PIXEL_FORMAT_RGBA_8888, |
| GRALLOC_USAGE_SW_WRITE_OFTEN | GRALLOC_USAGE_SW_READ_OFTEN); |
| ANativeWindowBuffer* mNativeWindowBuffer = mBuffer->getNativeBuffer(); |
| |
| Hwc2::mock::Composer* mComposer = nullptr; |
| renderengine::mock::RenderEngine* mRenderEngine = new renderengine::mock::RenderEngine(); |
| mock::TimeStats* mTimeStats = new mock::TimeStats(); |
| Hwc2::mock::PowerAdvisor* mPowerAdvisor = nullptr; |
| |
| sp<Fence> mClientTargetAcquireFence = Fence::NO_FENCE; |
| |
| std::shared_ptr<renderengine::ExternalTexture> mCaptureScreenBuffer; |
| }; |
| |
| template <typename LayerCase> |
| void CompositionTest::displayRefreshCompositionDirtyGeometry() { |
| setupForceGeometryDirty(); |
| LayerCase::setupForDirtyGeometry(this); |
| |
| // -------------------------------------------------------------------- |
| // Invocation |
| |
| mFlinger.commitAndComposite(); |
| |
| LayerCase::cleanup(this); |
| } |
| |
| template <typename LayerCase> |
| void CompositionTest::displayRefreshCompositionDirtyFrame() { |
| LayerCase::setupForDirtyFrame(this); |
| |
| // -------------------------------------------------------------------- |
| // Invocation |
| |
| mFlinger.commitAndComposite(); |
| |
| LayerCase::cleanup(this); |
| } |
| |
| template <typename LayerCase> |
| void CompositionTest::captureScreenComposition() { |
| LayerCase::setupForScreenCapture(this); |
| |
| const Rect sourceCrop(0, 0, DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT); |
| constexpr bool regionSampling = false; |
| |
| auto renderArea = DisplayRenderArea::create(mDisplay, sourceCrop, sourceCrop.getSize(), |
| ui::Dataspace::V0_SRGB, true, true); |
| |
| auto traverseLayers = [this](const LayerVector::Visitor& visitor) { |
| return mFlinger.traverseLayersInLayerStack(mDisplay->getLayerStack(), |
| CaptureArgs::UNSET_UID, {}, visitor); |
| }; |
| |
| auto getLayerSnapshots = RenderArea::fromTraverseLayersLambda(traverseLayers); |
| |
| const uint32_t usage = GRALLOC_USAGE_SW_READ_OFTEN | GRALLOC_USAGE_SW_WRITE_OFTEN | |
| GRALLOC_USAGE_HW_RENDER | GRALLOC_USAGE_HW_TEXTURE; |
| mCaptureScreenBuffer = |
| std::make_shared<renderengine::mock::FakeExternalTexture>(renderArea->getReqWidth(), |
| renderArea->getReqHeight(), |
| HAL_PIXEL_FORMAT_RGBA_8888, 1, |
| usage); |
| |
| auto future = mFlinger.renderScreenImpl(std::move(renderArea), getLayerSnapshots, |
| mCaptureScreenBuffer, regionSampling); |
| ASSERT_TRUE(future.valid()); |
| const auto fenceResult = future.get(); |
| |
| EXPECT_EQ(NO_ERROR, fenceStatus(fenceResult)); |
| if (fenceResult.ok()) { |
| fenceResult.value()->waitForever(LOG_TAG); |
| } |
| |
| LayerCase::cleanup(this); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Variants for each display configuration which can be tested |
| */ |
| |
| template <typename Derived> |
| struct BaseDisplayVariant { |
| static constexpr bool IS_SECURE = true; |
| static constexpr hal::PowerMode INIT_POWER_MODE = hal::PowerMode::ON; |
| |
| static void setupPreconditions(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, setPowerMode(HWC_DISPLAY, Derived::INIT_POWER_MODE)) |
| .WillOnce(Return(Error::NONE)); |
| |
| FakeHwcDisplayInjector(DEFAULT_DISPLAY_ID, hal::DisplayType::PHYSICAL, true /* isPrimary */) |
| .setCapabilities(&test->mDefaultCapabilities) |
| .setPowerMode(Derived::INIT_POWER_MODE) |
| .inject(&test->mFlinger, test->mComposer); |
| Mock::VerifyAndClear(test->mComposer); |
| |
| EXPECT_CALL(*test->mNativeWindow, query(NATIVE_WINDOW_WIDTH, _)) |
| .WillRepeatedly(DoAll(SetArgPointee<1>(DEFAULT_DISPLAY_WIDTH), Return(0))); |
| EXPECT_CALL(*test->mNativeWindow, query(NATIVE_WINDOW_HEIGHT, _)) |
| .WillRepeatedly(DoAll(SetArgPointee<1>(DEFAULT_DISPLAY_HEIGHT), Return(0))); |
| EXPECT_CALL(*test->mNativeWindow, perform(NATIVE_WINDOW_SET_BUFFERS_FORMAT)).Times(1); |
| EXPECT_CALL(*test->mNativeWindow, perform(NATIVE_WINDOW_API_CONNECT)).Times(1); |
| EXPECT_CALL(*test->mNativeWindow, perform(NATIVE_WINDOW_SET_USAGE64)).Times(1); |
| |
| const ::testing::TestInfo* const test_info = |
| ::testing::UnitTest::GetInstance()->current_test_info(); |
| |
| auto ceDisplayArgs = compositionengine::DisplayCreationArgsBuilder() |
| .setId(DEFAULT_DISPLAY_ID) |
| .setPixels({DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT}) |
| .setIsSecure(Derived::IS_SECURE) |
| .setPowerAdvisor(test->mPowerAdvisor) |
| .setName(std::string("Injected display for ") + |
| test_info->test_case_name() + "." + test_info->name()) |
| .build(); |
| |
| auto compositionDisplay = |
| compositionengine::impl::createDisplay(test->mFlinger.getCompositionEngine(), |
| ceDisplayArgs); |
| |
| constexpr auto kDisplayConnectionType = ui::DisplayConnectionType::Internal; |
| constexpr bool kIsPrimary = true; |
| |
| test->mDisplay = |
| FakeDisplayDeviceInjector(test->mFlinger, compositionDisplay, |
| kDisplayConnectionType, HWC_DISPLAY, kIsPrimary) |
| .setDisplaySurface(test->mDisplaySurface) |
| .setNativeWindow(test->mNativeWindow) |
| .setSecure(Derived::IS_SECURE) |
| .setPowerMode(Derived::INIT_POWER_MODE) |
| .setRefreshRateSelector(test->mFlinger.scheduler()->refreshRateSelector()) |
| .skipRegisterDisplay() |
| .inject(); |
| Mock::VerifyAndClear(test->mNativeWindow.get()); |
| |
| constexpr bool kIsInternal = kDisplayConnectionType == ui::DisplayConnectionType::Internal; |
| test->mDisplay->setLayerFilter({LAYER_STACK, kIsInternal}); |
| } |
| |
| template <typename Case> |
| static void setupPreconditionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, getDisplayCapabilities(HWC_DISPLAY, _)) |
| .WillOnce(DoAll(SetArgPointee<1>( |
| std::vector<aidl::android::hardware::graphics::composer3:: |
| DisplayCapability>({})), |
| Return(Error::NONE))); |
| } |
| |
| template <typename Case> |
| static void setupCommonCompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, setColorTransform(HWC_DISPLAY, IsIdentityMatrix())).Times(1); |
| EXPECT_CALL(*test->mComposer, getDisplayRequests(HWC_DISPLAY, _, _, _)).Times(1); |
| EXPECT_CALL(*test->mComposer, acceptDisplayChanges(HWC_DISPLAY)).Times(1); |
| EXPECT_CALL(*test->mComposer, presentDisplay(HWC_DISPLAY, _)).Times(1); |
| EXPECT_CALL(*test->mComposer, getReleaseFences(HWC_DISPLAY, _, _)).Times(1); |
| |
| EXPECT_CALL(*test->mDisplaySurface, onFrameCommitted()).Times(1); |
| EXPECT_CALL(*test->mDisplaySurface, advanceFrame(_)).Times(1); |
| |
| Case::CompositionType::setupHwcSetCallExpectations(test); |
| Case::CompositionType::setupHwcGetCallExpectations(test); |
| } |
| |
| template <typename Case> |
| static void setupCommonScreensCaptureCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mRenderEngine, drawLayers) |
| .WillRepeatedly([&](const renderengine::DisplaySettings& displaySettings, |
| const std::vector<renderengine::LayerSettings>&, |
| const std::shared_ptr<renderengine::ExternalTexture>&, |
| base::unique_fd&&) -> ftl::Future<FenceResult> { |
| EXPECT_EQ(DEFAULT_DISPLAY_MAX_LUMINANCE, displaySettings.maxLuminance); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.physicalDisplay); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.clip); |
| return ftl::yield<FenceResult>(Fence::NO_FENCE); |
| }); |
| } |
| |
| static void setupNonEmptyFrameCompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mDisplaySurface, beginFrame(true)).Times(1); |
| } |
| |
| static void setupEmptyFrameCompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mDisplaySurface, beginFrame(false)).Times(1); |
| } |
| |
| static void setupHwcCompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, presentOrValidateDisplay(HWC_DISPLAY, _, _, _, _, _, _)) |
| .Times(1); |
| |
| EXPECT_CALL(*test->mDisplaySurface, |
| prepareFrame(compositionengine::DisplaySurface::CompositionType::Hwc)) |
| .Times(1); |
| } |
| |
| static void setupHwcClientCompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, presentOrValidateDisplay(HWC_DISPLAY, _, _, _, _, _, _)) |
| .Times(1); |
| } |
| |
| static void setupHwcForcedClientCompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, validateDisplay(HWC_DISPLAY, _, _, _, _)).Times(1); |
| } |
| |
| static void setupRECompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mDisplaySurface, |
| prepareFrame(compositionengine::DisplaySurface::CompositionType::Gpu)) |
| .Times(1); |
| EXPECT_CALL(*test->mDisplaySurface, getClientTargetAcquireFence()) |
| .WillRepeatedly(ReturnRef(test->mClientTargetAcquireFence)); |
| |
| EXPECT_CALL(*test->mNativeWindow, queueBuffer(_, _)).WillOnce(Return(0)); |
| EXPECT_CALL(*test->mNativeWindow, dequeueBuffer(_, _)) |
| .WillOnce(DoAll(SetArgPointee<0>(test->mNativeWindowBuffer), SetArgPointee<1>(-1), |
| Return(0))); |
| EXPECT_CALL(*test->mRenderEngine, drawLayers) |
| .WillRepeatedly([&](const renderengine::DisplaySettings& displaySettings, |
| const std::vector<renderengine::LayerSettings>&, |
| const std::shared_ptr<renderengine::ExternalTexture>&, |
| base::unique_fd&&) -> ftl::Future<FenceResult> { |
| EXPECT_EQ(DEFAULT_DISPLAY_MAX_LUMINANCE, displaySettings.maxLuminance); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.physicalDisplay); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.clip); |
| EXPECT_EQ(ui::Dataspace::UNKNOWN, displaySettings.outputDataspace); |
| return ftl::yield<FenceResult>(Fence::NO_FENCE); |
| }); |
| } |
| |
| template <typename Case> |
| static void setupRELayerCompositionCallExpectations(CompositionTest* test) { |
| Case::Layer::setupRECompositionCallExpectations(test); |
| } |
| |
| template <typename Case> |
| static void setupRELayerScreenshotCompositionCallExpectations(CompositionTest* test) { |
| Case::Layer::setupREScreenshotCompositionCallExpectations(test); |
| } |
| }; |
| |
| struct DefaultDisplaySetupVariant : public BaseDisplayVariant<DefaultDisplaySetupVariant> {}; |
| |
| struct InsecureDisplaySetupVariant : public BaseDisplayVariant<InsecureDisplaySetupVariant> { |
| static constexpr bool IS_SECURE = false; |
| |
| template <typename Case> |
| static void setupRELayerCompositionCallExpectations(CompositionTest* test) { |
| Case::Layer::setupInsecureRECompositionCallExpectations(test); |
| } |
| |
| template <typename Case> |
| static void setupRELayerScreenshotCompositionCallExpectations(CompositionTest* test) { |
| Case::Layer::setupInsecureREScreenshotCompositionCallExpectations(test); |
| } |
| }; |
| |
| struct PoweredOffDisplaySetupVariant : public BaseDisplayVariant<PoweredOffDisplaySetupVariant> { |
| static constexpr hal::PowerMode INIT_POWER_MODE = hal::PowerMode::OFF; |
| |
| template <typename Case> |
| static void setupPreconditionCallExpectations(CompositionTest*) {} |
| |
| template <typename Case> |
| static void setupCommonCompositionCallExpectations(CompositionTest* test) { |
| // TODO: This seems like an unnecessary call if display is powered off. |
| EXPECT_CALL(*test->mComposer, setColorTransform(HWC_DISPLAY, IsIdentityMatrix())).Times(1); |
| |
| // TODO: This seems like an unnecessary call if display is powered off. |
| Case::CompositionType::setupHwcSetCallExpectations(test); |
| } |
| |
| static void setupHwcCompositionCallExpectations(CompositionTest*) {} |
| static void setupHwcClientCompositionCallExpectations(CompositionTest*) {} |
| static void setupHwcForcedClientCompositionCallExpectations(CompositionTest*) {} |
| |
| static void setupRECompositionCallExpectations(CompositionTest* test) { |
| // TODO: This seems like an unnecessary call if display is powered off. |
| EXPECT_CALL(*test->mDisplaySurface, getClientTargetAcquireFence()) |
| .WillRepeatedly(ReturnRef(test->mClientTargetAcquireFence)); |
| } |
| |
| template <typename Case> |
| static void setupRELayerCompositionCallExpectations(CompositionTest*) {} |
| }; |
| |
| /* ------------------------------------------------------------------------ |
| * Variants for each layer configuration which can be tested |
| */ |
| |
| template <typename LayerProperties> |
| struct BaseLayerProperties { |
| static constexpr uint32_t WIDTH = 100; |
| static constexpr uint32_t HEIGHT = 100; |
| static constexpr PixelFormat FORMAT = PIXEL_FORMAT_RGBA_8888; |
| static constexpr uint64_t USAGE = |
| GraphicBuffer::USAGE_SW_READ_NEVER | GraphicBuffer::USAGE_SW_WRITE_NEVER; |
| static constexpr android_dataspace DATASPACE = HAL_DATASPACE_UNKNOWN; |
| static constexpr uint32_t SCALING_MODE = 0; |
| static constexpr uint32_t TRANSFORM = 0; |
| static constexpr uint32_t LAYER_FLAGS = 0; |
| static constexpr float COLOR[] = {1.f, 1.f, 1.f, 1.f}; |
| static constexpr IComposerClient::BlendMode BLENDMODE = |
| IComposerClient::BlendMode::PREMULTIPLIED; |
| |
| static void setupLatchedBuffer(CompositionTest* test, sp<Layer> layer) { |
| Mock::VerifyAndClear(test->mRenderEngine); |
| |
| const auto buffer = std::make_shared< |
| renderengine::mock::FakeExternalTexture>(LayerProperties::WIDTH, |
| LayerProperties::HEIGHT, |
| DEFAULT_TEXTURE_ID, |
| LayerProperties::FORMAT, |
| LayerProperties::USAGE | |
| GraphicBuffer::USAGE_HW_TEXTURE); |
| |
| auto& layerDrawingState = test->mFlinger.mutableLayerDrawingState(layer); |
| layerDrawingState.crop = Rect(0, 0, LayerProperties::HEIGHT, LayerProperties::WIDTH); |
| layerDrawingState.buffer = buffer; |
| layerDrawingState.acquireFence = Fence::NO_FENCE; |
| layerDrawingState.dataspace = ui::Dataspace::UNKNOWN; |
| layer->setSurfaceDamageRegion( |
| Region(Rect(LayerProperties::HEIGHT, LayerProperties::WIDTH))); |
| |
| bool ignoredRecomputeVisibleRegions; |
| layer->latchBuffer(ignoredRecomputeVisibleRegions, 0); |
| Mock::VerifyAndClear(test->mRenderEngine); |
| } |
| |
| static void setupLayerState(CompositionTest* test, sp<Layer> layer) { |
| setupLatchedBuffer(test, layer); |
| } |
| |
| static void setupHwcSetGeometryCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| // TODO: Coverage of other values |
| EXPECT_CALL(*test->mComposer, |
| setLayerBlendMode(HWC_DISPLAY, HWC_LAYER, LayerProperties::BLENDMODE)) |
| .Times(1); |
| // TODO: Coverage of other values for origin |
| EXPECT_CALL(*test->mComposer, |
| setLayerDisplayFrame(HWC_DISPLAY, HWC_LAYER, |
| IComposerClient::Rect({0, 0, LayerProperties::WIDTH, |
| LayerProperties::HEIGHT}))) |
| .Times(1); |
| EXPECT_CALL(*test->mComposer, |
| setLayerPlaneAlpha(HWC_DISPLAY, HWC_LAYER, LayerProperties::COLOR[3])) |
| .Times(1); |
| // TODO: Coverage of other values |
| EXPECT_CALL(*test->mComposer, setLayerZOrder(HWC_DISPLAY, HWC_LAYER, 0u)).Times(1); |
| |
| // These expectations retire on saturation as the code path these |
| // expectations are for appears to make an extra call to them. |
| // TODO: Investigate this extra call |
| EXPECT_CALL(*test->mComposer, |
| setLayerTransform(HWC_DISPLAY, HWC_LAYER, DEFAULT_TRANSFORM)) |
| .Times(AtLeast(1)) |
| .RetiresOnSaturation(); |
| } |
| } |
| |
| static void setupHwcSetSourceCropBufferCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| EXPECT_CALL(*test->mComposer, |
| setLayerSourceCrop(HWC_DISPLAY, HWC_LAYER, |
| IComposerClient::FRect({0.f, 0.f, LayerProperties::WIDTH, |
| LayerProperties::HEIGHT}))) |
| .Times(1); |
| } |
| } |
| |
| static void setupHwcSetSourceCropColorCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| EXPECT_CALL(*test->mComposer, |
| setLayerSourceCrop(HWC_DISPLAY, HWC_LAYER, |
| IComposerClient::FRect({0.f, 0.f, 0.f, 0.f}))) |
| .Times(1); |
| } |
| } |
| |
| static void setupHwcSetPerFrameCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| EXPECT_CALL(*test->mComposer, |
| setLayerVisibleRegion(HWC_DISPLAY, HWC_LAYER, |
| std::vector<IComposerClient::Rect>( |
| {IComposerClient::Rect( |
| {0, 0, LayerProperties::WIDTH, |
| LayerProperties::HEIGHT})}))) |
| .Times(1); |
| } |
| } |
| |
| static void setupHwcSetPerFrameColorCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| EXPECT_CALL(*test->mComposer, setLayerSurfaceDamage(HWC_DISPLAY, HWC_LAYER, _)) |
| .Times(1); |
| |
| // TODO: use COLOR |
| EXPECT_CALL(*test->mComposer, |
| setLayerColor(HWC_DISPLAY, HWC_LAYER, |
| aidl::android::hardware::graphics::composer3::Color( |
| {1.0f, 1.0f, 1.0f, 1.0f}))) |
| .Times(1); |
| } |
| } |
| |
| static void setupHwcSetPerFrameBufferCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| EXPECT_CALL(*test->mComposer, setLayerSurfaceDamage(HWC_DISPLAY, HWC_LAYER, _)) |
| .Times(1); |
| EXPECT_CALL(*test->mComposer, setLayerBuffer(HWC_DISPLAY, HWC_LAYER, _, _, _)).Times(1); |
| } |
| } |
| |
| static void setupREBufferCompositionCommonCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mRenderEngine, drawLayers) |
| .WillOnce([&](const renderengine::DisplaySettings& displaySettings, |
| const std::vector<renderengine::LayerSettings>& layerSettings, |
| const std::shared_ptr<renderengine::ExternalTexture>&, |
| base::unique_fd&&) -> ftl::Future<FenceResult> { |
| EXPECT_EQ(DEFAULT_DISPLAY_MAX_LUMINANCE, displaySettings.maxLuminance); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.physicalDisplay); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.clip); |
| // screen capture adds an additional color layer as an alpha |
| // prefill, so gtet the back layer. |
| ftl::Future<FenceResult> resultFuture = |
| ftl::yield<FenceResult>(Fence::NO_FENCE); |
| if (layerSettings.empty()) { |
| ADD_FAILURE() << "layerSettings was not expected to be empty in " |
| "setupREBufferCompositionCommonCallExpectations " |
| "verification lambda"; |
| return resultFuture; |
| } |
| const renderengine::LayerSettings layer = layerSettings.back(); |
| EXPECT_THAT(layer.source.buffer.buffer, Not(IsNull())); |
| EXPECT_THAT(layer.source.buffer.fence, Not(IsNull())); |
| EXPECT_EQ(true, layer.source.buffer.usePremultipliedAlpha); |
| EXPECT_EQ(false, layer.source.buffer.isOpaque); |
| EXPECT_EQ(0.0, layer.geometry.roundedCornersRadius.x); |
| EXPECT_EQ(0.0, layer.geometry.roundedCornersRadius.y); |
| EXPECT_EQ(ui::Dataspace::V0_SRGB, layer.sourceDataspace); |
| EXPECT_EQ(LayerProperties::COLOR[3], layer.alpha); |
| return resultFuture; |
| }); |
| } |
| |
| static void setupREBufferCompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupREBufferCompositionCommonCallExpectations(test); |
| } |
| |
| static void setupInsecureREBufferCompositionCallExpectations(CompositionTest* test) { |
| setupREBufferCompositionCallExpectations(test); |
| } |
| |
| static void setupREBufferScreenshotCompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupREBufferCompositionCommonCallExpectations(test); |
| } |
| |
| static void setupInsecureREBufferScreenshotCompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupREBufferCompositionCommonCallExpectations(test); |
| } |
| |
| static void setupREColorCompositionCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mRenderEngine, drawLayers) |
| .WillOnce([&](const renderengine::DisplaySettings& displaySettings, |
| const std::vector<renderengine::LayerSettings>& layerSettings, |
| const std::shared_ptr<renderengine::ExternalTexture>&, |
| base::unique_fd&&) -> ftl::Future<FenceResult> { |
| EXPECT_EQ(DEFAULT_DISPLAY_MAX_LUMINANCE, displaySettings.maxLuminance); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.physicalDisplay); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.clip); |
| // screen capture adds an additional color layer as an alpha |
| // prefill, so get the back layer. |
| ftl::Future<FenceResult> resultFuture = |
| ftl::yield<FenceResult>(Fence::NO_FENCE); |
| if (layerSettings.empty()) { |
| ADD_FAILURE() |
| << "layerSettings was not expected to be empty in " |
| "setupREColorCompositionCallExpectations verification lambda"; |
| return resultFuture; |
| } |
| const renderengine::LayerSettings layer = layerSettings.back(); |
| EXPECT_THAT(layer.source.buffer.buffer, IsNull()); |
| EXPECT_EQ(half3(LayerProperties::COLOR[0], LayerProperties::COLOR[1], |
| LayerProperties::COLOR[2]), |
| layer.source.solidColor); |
| EXPECT_EQ(0.0, layer.geometry.roundedCornersRadius.x); |
| EXPECT_EQ(0.0, layer.geometry.roundedCornersRadius.y); |
| EXPECT_EQ(ui::Dataspace::V0_SRGB, layer.sourceDataspace); |
| EXPECT_EQ(LayerProperties::COLOR[3], layer.alpha); |
| return resultFuture; |
| }); |
| } |
| |
| static void setupREColorScreenshotCompositionCallExpectations(CompositionTest* test) { |
| setupREColorCompositionCallExpectations(test); |
| } |
| }; |
| |
| struct DefaultLayerProperties : public BaseLayerProperties<DefaultLayerProperties> {}; |
| |
| struct EffectLayerProperties : public BaseLayerProperties<EffectLayerProperties> { |
| static constexpr IComposerClient::BlendMode BLENDMODE = IComposerClient::BlendMode::NONE; |
| }; |
| |
| struct SidebandLayerProperties : public BaseLayerProperties<SidebandLayerProperties> { |
| using Base = BaseLayerProperties<SidebandLayerProperties>; |
| static constexpr IComposerClient::BlendMode BLENDMODE = IComposerClient::BlendMode::NONE; |
| |
| static void setupLayerState(CompositionTest* test, sp<Layer> layer) { |
| sp<NativeHandle> stream = |
| NativeHandle::create(reinterpret_cast<native_handle_t*>(DEFAULT_SIDEBAND_STREAM), |
| false); |
| test->mFlinger.setLayerSidebandStream(layer, stream); |
| auto& layerDrawingState = test->mFlinger.mutableLayerDrawingState(layer); |
| layerDrawingState.crop = |
| Rect(0, 0, SidebandLayerProperties::HEIGHT, SidebandLayerProperties::WIDTH); |
| } |
| |
| static void setupHwcSetSourceCropBufferCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, |
| setLayerSourceCrop(HWC_DISPLAY, HWC_LAYER, |
| IComposerClient::FRect({0.f, 0.f, -1.f, -1.f}))) |
| .Times(1); |
| } |
| |
| static void setupHwcSetPerFrameBufferCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, |
| setLayerSidebandStream(HWC_DISPLAY, HWC_LAYER, |
| reinterpret_cast<native_handle_t*>( |
| DEFAULT_SIDEBAND_STREAM))) |
| .WillOnce(Return(Error::NONE)); |
| |
| EXPECT_CALL(*test->mComposer, setLayerSurfaceDamage(HWC_DISPLAY, HWC_LAYER, _)).Times(1); |
| } |
| |
| static void setupREBufferCompositionCommonCallExpectations(CompositionTest* /*test*/) {} |
| }; |
| |
| template <typename LayerProperties> |
| struct CommonSecureLayerProperties : public BaseLayerProperties<LayerProperties> { |
| using Base = BaseLayerProperties<LayerProperties>; |
| |
| static void setupInsecureREBufferCompositionCommonCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mRenderEngine, drawLayers) |
| .WillOnce([&](const renderengine::DisplaySettings& displaySettings, |
| const std::vector<renderengine::LayerSettings>& layerSettings, |
| const std::shared_ptr<renderengine::ExternalTexture>&, |
| base::unique_fd&&) -> ftl::Future<FenceResult> { |
| EXPECT_EQ(DEFAULT_DISPLAY_MAX_LUMINANCE, displaySettings.maxLuminance); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.physicalDisplay); |
| EXPECT_EQ(Rect(DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT), |
| displaySettings.clip); |
| // screen capture adds an additional color layer as an alpha |
| // prefill, so get the back layer. |
| ftl::Future<FenceResult> resultFuture = |
| ftl::yield<FenceResult>(Fence::NO_FENCE); |
| if (layerSettings.empty()) { |
| ADD_FAILURE() << "layerSettings was not expected to be empty in " |
| "setupInsecureREBufferCompositionCommonCallExpectations " |
| "verification lambda"; |
| return resultFuture; |
| } |
| const renderengine::LayerSettings layer = layerSettings.back(); |
| EXPECT_THAT(layer.source.buffer.buffer, IsNull()); |
| EXPECT_EQ(half3(0.0f, 0.0f, 0.0f), layer.source.solidColor); |
| EXPECT_EQ(0.0, layer.geometry.roundedCornersRadius.x); |
| EXPECT_EQ(0.0, layer.geometry.roundedCornersRadius.y); |
| EXPECT_EQ(ui::Dataspace::V0_SRGB, layer.sourceDataspace); |
| EXPECT_EQ(1.0f, layer.alpha); |
| return resultFuture; |
| }); |
| } |
| |
| static void setupInsecureREBufferCompositionCallExpectations(CompositionTest* test) { |
| setupInsecureREBufferCompositionCommonCallExpectations(test); |
| } |
| |
| static void setupInsecureREBufferScreenshotCompositionCallExpectations(CompositionTest* test) { |
| setupInsecureREBufferCompositionCommonCallExpectations(test); |
| } |
| }; |
| |
| struct ParentSecureLayerProperties |
| : public CommonSecureLayerProperties<ParentSecureLayerProperties> {}; |
| |
| struct SecureLayerProperties : public CommonSecureLayerProperties<SecureLayerProperties> { |
| static constexpr uint32_t LAYER_FLAGS = ISurfaceComposerClient::eSecure; |
| }; |
| |
| struct CursorLayerProperties : public BaseLayerProperties<CursorLayerProperties> { |
| using Base = BaseLayerProperties<CursorLayerProperties>; |
| |
| static void setupLayerState(CompositionTest* test, sp<Layer> layer) { |
| Base::setupLayerState(test, layer); |
| test->mFlinger.setLayerPotentialCursor(layer, true); |
| } |
| }; |
| |
| struct NoLayerVariant { |
| using FlingerLayerType = sp<Layer>; |
| |
| static FlingerLayerType createLayer(CompositionTest*) { return FlingerLayerType(); } |
| static void injectLayer(CompositionTest*, FlingerLayerType) {} |
| static void cleanupInjectedLayers(CompositionTest*) {} |
| |
| static void setupCallExpectationsForDirtyGeometry(CompositionTest*) {} |
| static void setupCallExpectationsForDirtyFrame(CompositionTest*) {} |
| }; |
| |
| template <typename LayerProperties> |
| struct BaseLayerVariant { |
| template <typename L, typename F> |
| static sp<L> createLayerWithFactory(CompositionTest* test, F factory) { |
| EXPECT_CALL(*test->mFlinger.scheduler(), postMessage(_)).Times(0); |
| |
| sp<L> layer = factory(); |
| |
| // Layer should be registered with scheduler. |
| EXPECT_EQ(1u, test->mFlinger.scheduler()->layerHistorySize()); |
| |
| Mock::VerifyAndClear(test->mComposer); |
| Mock::VerifyAndClear(test->mRenderEngine); |
| Mock::VerifyAndClearExpectations(test->mFlinger.scheduler()); |
| |
| initLayerDrawingStateAndComputeBounds(test, layer); |
| |
| return layer; |
| } |
| |
| template <typename L> |
| static void initLayerDrawingStateAndComputeBounds(CompositionTest* test, sp<L> layer) { |
| auto& layerDrawingState = test->mFlinger.mutableLayerDrawingState(layer); |
| layerDrawingState.layerStack = LAYER_STACK; |
| layerDrawingState.color = half4(LayerProperties::COLOR[0], LayerProperties::COLOR[1], |
| LayerProperties::COLOR[2], LayerProperties::COLOR[3]); |
| layer->computeBounds(FloatRect(0, 0, 100, 100), ui::Transform(), 0.f /* shadowRadius */); |
| } |
| |
| static void injectLayer(CompositionTest* test, sp<Layer> layer) { |
| EXPECT_CALL(*test->mComposer, createLayer(HWC_DISPLAY, _)) |
| .WillOnce(DoAll(SetArgPointee<1>(HWC_LAYER), Return(Error::NONE))); |
| |
| auto outputLayer = test->mDisplay->getCompositionDisplay()->injectOutputLayerForTest( |
| layer->getCompositionEngineLayerFE()); |
| outputLayer->editState().visibleRegion = Region(Rect(0, 0, 100, 100)); |
| outputLayer->editState().outputSpaceVisibleRegion = Region(Rect(0, 0, 100, 100)); |
| |
| Mock::VerifyAndClear(test->mComposer); |
| |
| test->mFlinger.mutableDrawingState().layersSortedByZ.add(layer); |
| test->mFlinger.mutableVisibleRegionsDirty() = true; |
| } |
| |
| static void cleanupInjectedLayers(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, destroyLayer(HWC_DISPLAY, HWC_LAYER)) |
| .WillOnce(Return(Error::NONE)); |
| |
| test->mDisplay->getCompositionDisplay()->clearOutputLayers(); |
| test->mFlinger.mutableDrawingState().layersSortedByZ.clear(); |
| test->mFlinger.mutablePreviouslyComposedLayers().clear(); |
| |
| // Layer should be unregistered with scheduler. |
| test->mFlinger.commit(); |
| EXPECT_EQ(0u, test->mFlinger.scheduler()->layerHistorySize()); |
| } |
| }; |
| |
| template <typename LayerProperties> |
| struct EffectLayerVariant : public BaseLayerVariant<LayerProperties> { |
| using Base = BaseLayerVariant<LayerProperties>; |
| using FlingerLayerType = sp<Layer>; |
| |
| static FlingerLayerType createLayer(CompositionTest* test) { |
| FlingerLayerType layer = Base::template createLayerWithFactory<Layer>(test, [test]() { |
| return sp<Layer>::make(LayerCreationArgs(test->mFlinger.flinger(), sp<Client>(), |
| "test-layer", LayerProperties::LAYER_FLAGS, |
| LayerMetadata())); |
| }); |
| |
| auto& layerDrawingState = test->mFlinger.mutableLayerDrawingState(layer); |
| layerDrawingState.crop = Rect(0, 0, LayerProperties::HEIGHT, LayerProperties::WIDTH); |
| return layer; |
| } |
| |
| static void setupRECompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupREColorCompositionCallExpectations(test); |
| } |
| |
| static void setupREScreenshotCompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupREColorScreenshotCompositionCallExpectations(test); |
| } |
| |
| static void setupCallExpectationsForDirtyGeometry(CompositionTest* test) { |
| LayerProperties::setupHwcSetGeometryCallExpectations(test); |
| LayerProperties::setupHwcSetSourceCropColorCallExpectations(test); |
| } |
| |
| static void setupCallExpectationsForDirtyFrame(CompositionTest* test) { |
| LayerProperties::setupHwcSetPerFrameCallExpectations(test); |
| LayerProperties::setupHwcSetPerFrameColorCallExpectations(test); |
| } |
| }; |
| |
| template <typename LayerProperties> |
| struct BufferLayerVariant : public BaseLayerVariant<LayerProperties> { |
| using Base = BaseLayerVariant<LayerProperties>; |
| using FlingerLayerType = sp<Layer>; |
| |
| static FlingerLayerType createLayer(CompositionTest* test) { |
| FlingerLayerType layer = Base::template createLayerWithFactory<Layer>(test, [test]() { |
| LayerCreationArgs args(test->mFlinger.flinger(), sp<Client>(), "test-layer", |
| LayerProperties::LAYER_FLAGS, LayerMetadata()); |
| return sp<Layer>::make(args); |
| }); |
| |
| LayerProperties::setupLayerState(test, layer); |
| |
| return layer; |
| } |
| |
| static void cleanupInjectedLayers(CompositionTest* test) { |
| Base::cleanupInjectedLayers(test); |
| } |
| |
| static void setupCallExpectationsForDirtyGeometry(CompositionTest* test) { |
| LayerProperties::setupHwcSetGeometryCallExpectations(test); |
| LayerProperties::setupHwcSetSourceCropBufferCallExpectations(test); |
| } |
| |
| static void setupCallExpectationsForDirtyFrame(CompositionTest* test) { |
| LayerProperties::setupHwcSetPerFrameCallExpectations(test); |
| LayerProperties::setupHwcSetPerFrameBufferCallExpectations(test); |
| } |
| |
| static void setupRECompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupREBufferCompositionCallExpectations(test); |
| } |
| |
| static void setupInsecureRECompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupInsecureREBufferCompositionCallExpectations(test); |
| } |
| |
| static void setupREScreenshotCompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupREBufferScreenshotCompositionCallExpectations(test); |
| } |
| |
| static void setupInsecureREScreenshotCompositionCallExpectations(CompositionTest* test) { |
| LayerProperties::setupInsecureREBufferScreenshotCompositionCallExpectations(test); |
| } |
| }; |
| |
| template <typename LayerProperties> |
| struct ContainerLayerVariant : public BaseLayerVariant<LayerProperties> { |
| using Base = BaseLayerVariant<LayerProperties>; |
| using FlingerLayerType = sp<Layer>; |
| |
| static FlingerLayerType createLayer(CompositionTest* test) { |
| LayerCreationArgs args(test->mFlinger.flinger(), sp<Client>(), "test-container-layer", |
| LayerProperties::LAYER_FLAGS, LayerMetadata()); |
| FlingerLayerType layer = sp<Layer>::make(args); |
| Base::template initLayerDrawingStateAndComputeBounds(test, layer); |
| return layer; |
| } |
| }; |
| |
| template <typename LayerVariant, typename ParentLayerVariant> |
| struct ChildLayerVariant : public LayerVariant { |
| using Base = LayerVariant; |
| using FlingerLayerType = typename LayerVariant::FlingerLayerType; |
| using ParentBase = ParentLayerVariant; |
| |
| static FlingerLayerType createLayer(CompositionTest* test) { |
| // Need to create child layer first. Otherwise layer history size will be 2. |
| FlingerLayerType layer = Base::createLayer(test); |
| |
| typename ParentBase::FlingerLayerType parentLayer = ParentBase::createLayer(test); |
| parentLayer->addChild(layer); |
| test->mFlinger.setLayerDrawingParent(layer, parentLayer); |
| |
| test->mAuxiliaryLayers.push_back(parentLayer); |
| |
| return layer; |
| } |
| |
| static void cleanupInjectedLayers(CompositionTest* test) { |
| // Clear auxiliary layers first so that child layer can be successfully destroyed in the |
| // following call. |
| test->mAuxiliaryLayers.clear(); |
| |
| Base::cleanupInjectedLayers(test); |
| } |
| }; |
| |
| /* ------------------------------------------------------------------------ |
| * Variants to control how the composition type is changed |
| */ |
| |
| struct NoCompositionTypeVariant { |
| static void setupHwcSetCallExpectations(CompositionTest*) {} |
| |
| static void setupHwcGetCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, getChangedCompositionTypes(HWC_DISPLAY, _, _)).Times(1); |
| } |
| }; |
| |
| template <aidl::android::hardware::graphics::composer3::Composition CompositionType> |
| struct KeepCompositionTypeVariant { |
| static constexpr aidl::android::hardware::graphics::composer3::Composition TYPE = |
| CompositionType; |
| |
| static void setupHwcSetCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| EXPECT_CALL(*test->mComposer, |
| setLayerCompositionType(HWC_DISPLAY, HWC_LAYER, CompositionType)) |
| .Times(1); |
| } |
| } |
| |
| static void setupHwcGetCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, getChangedCompositionTypes(HWC_DISPLAY, _, _)).Times(1); |
| } |
| }; |
| |
| template <aidl::android::hardware::graphics::composer3::Composition InitialCompositionType, |
| aidl::android::hardware::graphics::composer3::Composition FinalCompositionType> |
| struct ChangeCompositionTypeVariant { |
| static constexpr aidl::android::hardware::graphics::composer3::Composition TYPE = |
| FinalCompositionType; |
| |
| static void setupHwcSetCallExpectations(CompositionTest* test) { |
| if (!test->mDisplayOff) { |
| EXPECT_CALL(*test->mComposer, |
| setLayerCompositionType(HWC_DISPLAY, HWC_LAYER, InitialCompositionType)) |
| .Times(1); |
| } |
| } |
| |
| static void setupHwcGetCallExpectations(CompositionTest* test) { |
| EXPECT_CALL(*test->mComposer, getChangedCompositionTypes(HWC_DISPLAY, _, _)) |
| .WillOnce(DoAll(SetArgPointee<1>(std::vector<Hwc2::Layer>{ |
| static_cast<Hwc2::Layer>(HWC_LAYER)}), |
| SetArgPointee<2>( |
| std::vector<aidl::android::hardware::graphics::composer3:: |
| Composition>{FinalCompositionType}), |
| Return(Error::NONE))); |
| } |
| }; |
| |
| /* ------------------------------------------------------------------------ |
| * Variants to select how the composition is expected to be handled |
| */ |
| |
| struct CompositionResultBaseVariant { |
| static void setupLayerState(CompositionTest*, sp<Layer>) {} |
| |
| template <typename Case> |
| static void setupCallExpectationsForDirtyGeometry(CompositionTest* test) { |
| Case::Layer::setupCallExpectationsForDirtyGeometry(test); |
| } |
| |
| template <typename Case> |
| static void setupCallExpectationsForDirtyFrame(CompositionTest* test) { |
| Case::Layer::setupCallExpectationsForDirtyFrame(test); |
| } |
| }; |
| |
| struct NoCompositionResultVariant : public CompositionResultBaseVariant { |
| template <typename Case> |
| static void setupCallExpectations(CompositionTest* test) { |
| Case::Display::setupEmptyFrameCompositionCallExpectations(test); |
| Case::Display::setupHwcCompositionCallExpectations(test); |
| } |
| }; |
| |
| struct HwcCompositionResultVariant : public CompositionResultBaseVariant { |
| template <typename Case> |
| static void setupCallExpectations(CompositionTest* test) { |
| Case::Display::setupNonEmptyFrameCompositionCallExpectations(test); |
| Case::Display::setupHwcCompositionCallExpectations(test); |
| } |
| }; |
| |
| struct RECompositionResultVariant : public CompositionResultBaseVariant { |
| template <typename Case> |
| static void setupCallExpectations(CompositionTest* test) { |
| Case::Display::setupNonEmptyFrameCompositionCallExpectations(test); |
| Case::Display::setupHwcClientCompositionCallExpectations(test); |
| Case::Display::setupRECompositionCallExpectations(test); |
| Case::Display::template setupRELayerCompositionCallExpectations<Case>(test); |
| } |
| }; |
| |
| struct ForcedClientCompositionResultVariant : public CompositionResultBaseVariant { |
| static void setupLayerState(CompositionTest* test, sp<Layer> layer) { |
| const auto outputLayer = |
| TestableSurfaceFlinger::findOutputLayerForDisplay(layer, test->mDisplay); |
| LOG_FATAL_IF(!outputLayer); |
| outputLayer->editState().forceClientComposition = true; |
| } |
| |
| template <typename Case> |
| static void setupCallExpectations(CompositionTest* test) { |
| Case::Display::setupNonEmptyFrameCompositionCallExpectations(test); |
| Case::Display::setupHwcForcedClientCompositionCallExpectations(test); |
| Case::Display::setupRECompositionCallExpectations(test); |
| Case::Display::template setupRELayerCompositionCallExpectations<Case>(test); |
| } |
| |
| template <typename Case> |
| static void setupCallExpectationsForDirtyGeometry(CompositionTest*) {} |
| |
| template <typename Case> |
| static void setupCallExpectationsForDirtyFrame(CompositionTest*) {} |
| }; |
| |
| struct ForcedClientCompositionViaDebugOptionResultVariant : public CompositionResultBaseVariant { |
| static void setupLayerState(CompositionTest* test, sp<Layer>) { |
| test->mFlinger.mutableDebugDisableHWC() = true; |
| } |
| |
| template <typename Case> |
| static void setupCallExpectations(CompositionTest* test) { |
| Case::Display::setupNonEmptyFrameCompositionCallExpectations(test); |
| Case::Display::setupHwcForcedClientCompositionCallExpectations(test); |
| Case::Display::setupRECompositionCallExpectations(test); |
| Case::Display::template setupRELayerCompositionCallExpectations<Case>(test); |
| } |
| |
| template <typename Case> |
| static void setupCallExpectationsForDirtyGeometry(CompositionTest*) {} |
| |
| template <typename Case> |
| static void setupCallExpectationsForDirtyFrame(CompositionTest*) {} |
| }; |
| |
| struct EmptyScreenshotResultVariant { |
| static void setupLayerState(CompositionTest*, sp<Layer>) {} |
| |
| template <typename Case> |
| static void setupCallExpectations(CompositionTest*) {} |
| }; |
| |
| struct REScreenshotResultVariant : public EmptyScreenshotResultVariant { |
| using Base = EmptyScreenshotResultVariant; |
| |
| template <typename Case> |
| static void setupCallExpectations(CompositionTest* test) { |
| Base::template setupCallExpectations<Case>(test); |
| Case::Display::template setupRELayerScreenshotCompositionCallExpectations<Case>(test); |
| } |
| }; |
| |
| /* ------------------------------------------------------------------------ |
| * Composition test case, containing all the variants being tested |
| */ |
| |
| template <typename DisplayCase, typename LayerCase, typename CompositionTypeCase, |
| typename CompositionResultCase> |
| struct CompositionCase { |
| using ThisCase = |
| CompositionCase<DisplayCase, LayerCase, CompositionTypeCase, CompositionResultCase>; |
| using Display = DisplayCase; |
| using Layer = LayerCase; |
| using CompositionType = CompositionTypeCase; |
| using CompositionResult = CompositionResultCase; |
| |
| static void setupCommon(CompositionTest* test) { |
| Display::template setupPreconditionCallExpectations<ThisCase>(test); |
| Display::setupPreconditions(test); |
| |
| auto layer = Layer::createLayer(test); |
| Layer::injectLayer(test, layer); |
| CompositionResult::setupLayerState(test, layer); |
| } |
| |
| static void setupForDirtyGeometry(CompositionTest* test) { |
| setupCommon(test); |
| |
| Display::template setupCommonCompositionCallExpectations<ThisCase>(test); |
| CompositionResult::template setupCallExpectationsForDirtyGeometry<ThisCase>(test); |
| CompositionResult::template setupCallExpectationsForDirtyFrame<ThisCase>(test); |
| CompositionResult::template setupCallExpectations<ThisCase>(test); |
| } |
| |
| static void setupForDirtyFrame(CompositionTest* test) { |
| setupCommon(test); |
| |
| Display::template setupCommonCompositionCallExpectations<ThisCase>(test); |
| CompositionResult::template setupCallExpectationsForDirtyFrame<ThisCase>(test); |
| CompositionResult::template setupCallExpectations<ThisCase>(test); |
| } |
| |
| static void setupForScreenCapture(CompositionTest* test) { |
| setupCommon(test); |
| |
| Display::template setupCommonScreensCaptureCallExpectations<ThisCase>(test); |
| CompositionResult::template setupCallExpectations<ThisCase>(test); |
| } |
| |
| static void cleanup(CompositionTest* test) { |
| Layer::cleanupInjectedLayers(test); |
| |
| for (auto& displayData : test->mFlinger.mutableHwcDisplayData()) { |
| static_cast<TestableSurfaceFlinger::HWC2Display*>(displayData.second.hwcDisplay.get()) |
| ->mutableLayers() |
| .clear(); |
| } |
| } |
| }; |
| |
| /* ------------------------------------------------------------------------ |
| * Composition cases to test |
| */ |
| |
| TEST_F(CompositionTest, noLayersDoesMinimalWorkWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry< |
| CompositionCase<DefaultDisplaySetupVariant, NoLayerVariant, NoCompositionTypeVariant, |
| NoCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, noLayersDoesMinimalWorkWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame< |
| CompositionCase<DefaultDisplaySetupVariant, NoLayerVariant, NoCompositionTypeVariant, |
| NoCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, noLayersDoesMinimalWorkToCaptureScreen) { |
| captureScreenComposition< |
| CompositionCase<DefaultDisplaySetupVariant, NoLayerVariant, NoCompositionTypeVariant, |
| EmptyScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Simple buffer layers |
| */ |
| |
| TEST_F(CompositionTest, HWCComposedNormalBufferLayerWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, HWCComposedNormalBufferLayerWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, REComposedNormalBufferLayer) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| ChangeCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE, |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| RECompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenNormalBufferLayer) { |
| captureScreenComposition< |
| CompositionCase<DefaultDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Single-color layers |
| */ |
| |
| TEST_F(CompositionTest, HWCComposedEffectLayerWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| DefaultDisplaySetupVariant, EffectLayerVariant<EffectLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::SOLID_COLOR>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, HWCComposedEffectLayerWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, EffectLayerVariant<EffectLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::SOLID_COLOR>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, REComposedEffectLayer) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, EffectLayerVariant<EffectLayerProperties>, |
| ChangeCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::SOLID_COLOR, |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| RECompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenEffectLayer) { |
| captureScreenComposition< |
| CompositionCase<DefaultDisplaySetupVariant, EffectLayerVariant<EffectLayerProperties>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Layers with sideband buffers |
| */ |
| |
| TEST_F(CompositionTest, HWCComposedSidebandBufferLayerWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<SidebandLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::SIDEBAND>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, HWCComposedSidebandBufferLayerWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<SidebandLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::SIDEBAND>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, REComposedSidebandBufferLayer) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<SidebandLayerProperties>, |
| ChangeCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::SIDEBAND, |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| RECompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenSidebandBufferLayer) { |
| captureScreenComposition< |
| CompositionCase<DefaultDisplaySetupVariant, BufferLayerVariant<SidebandLayerProperties>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Layers with ISurfaceComposerClient::eSecure, on a secure display |
| */ |
| |
| TEST_F(CompositionTest, HWCComposedSecureBufferLayerWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<SecureLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, HWCComposedSecureBufferLayerWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<SecureLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, REComposedSecureBufferLayer) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<SecureLayerProperties>, |
| ChangeCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE, |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| RECompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenSecureBufferLayerOnSecureDisplay) { |
| captureScreenComposition< |
| CompositionCase<DefaultDisplaySetupVariant, BufferLayerVariant<SecureLayerProperties>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Layers with ISurfaceComposerClient::eSecure, on a non-secure display |
| */ |
| |
| TEST_F(CompositionTest, HWCComposedSecureBufferLayerOnInsecureDisplayWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| InsecureDisplaySetupVariant, BufferLayerVariant<SecureLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| ForcedClientCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, HWCComposedSecureBufferLayerOnInsecureDisplayWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| InsecureDisplaySetupVariant, BufferLayerVariant<SecureLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| ForcedClientCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenSecureBufferLayerOnInsecureDisplay) { |
| captureScreenComposition< |
| CompositionCase<InsecureDisplaySetupVariant, BufferLayerVariant<SecureLayerProperties>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Layers with a parent layer with ISurfaceComposerClient::eSecure, on a non-secure display |
| */ |
| |
| TEST_F(CompositionTest, |
| HWCComposedBufferLayerWithSecureParentLayerOnInsecureDisplayWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| InsecureDisplaySetupVariant, |
| ChildLayerVariant<BufferLayerVariant<ParentSecureLayerProperties>, |
| ContainerLayerVariant<SecureLayerProperties>>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| ForcedClientCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, |
| HWCComposedBufferLayerWithSecureParentLayerOnInsecureDisplayWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| InsecureDisplaySetupVariant, |
| ChildLayerVariant<BufferLayerVariant<ParentSecureLayerProperties>, |
| ContainerLayerVariant<SecureLayerProperties>>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| ForcedClientCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenBufferLayerWithSecureParentLayerOnInsecureDisplay) { |
| captureScreenComposition< |
| CompositionCase<InsecureDisplaySetupVariant, |
| ChildLayerVariant<BufferLayerVariant<ParentSecureLayerProperties>, |
| ContainerLayerVariant<SecureLayerProperties>>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Cursor layers |
| */ |
| |
| TEST_F(CompositionTest, HWCComposedCursorLayerWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<CursorLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CURSOR>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, HWCComposedCursorLayerWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<CursorLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CURSOR>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, REComposedCursorLayer) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<CursorLayerProperties>, |
| ChangeCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CURSOR, |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| RECompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenCursorLayer) { |
| captureScreenComposition< |
| CompositionCase<DefaultDisplaySetupVariant, BufferLayerVariant<CursorLayerProperties>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Simple buffer layer on a display which is powered off. |
| */ |
| |
| TEST_F(CompositionTest, displayOffHWCComposedNormalBufferLayerWithDirtyGeometry) { |
| mDisplayOff = true; |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| PoweredOffDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, displayOffHWCComposedNormalBufferLayerWithDirtyFrame) { |
| mDisplayOff = true; |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| PoweredOffDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE>, |
| HwcCompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, displayOffREComposedNormalBufferLayer) { |
| mDisplayOff = true; |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| PoweredOffDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| ChangeCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::DEVICE, |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| RECompositionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, captureScreenNormalBufferLayerOnPoweredOffDisplay) { |
| captureScreenComposition<CompositionCase< |
| PoweredOffDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| NoCompositionTypeVariant, REScreenshotResultVariant>>(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * Client composition forced through debug/developer settings |
| */ |
| |
| TEST_F(CompositionTest, DebugOptionForcingClientCompositionOfBufferLayerWithDirtyGeometry) { |
| displayRefreshCompositionDirtyGeometry<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| ForcedClientCompositionViaDebugOptionResultVariant>>(); |
| } |
| |
| TEST_F(CompositionTest, DebugOptionForcingClientCompositionOfBufferLayerWithDirtyFrame) { |
| displayRefreshCompositionDirtyFrame<CompositionCase< |
| DefaultDisplaySetupVariant, BufferLayerVariant<DefaultLayerProperties>, |
| KeepCompositionTypeVariant< |
| aidl::android::hardware::graphics::composer3::Composition::CLIENT>, |
| ForcedClientCompositionViaDebugOptionResultVariant>>(); |
| } |
| |
| } // namespace |
| } // namespace android |
| |
| // TODO(b/129481165): remove the #pragma below and fix conversion issues |
| #pragma clang diagnostic pop // ignored "-Wconversion -Wextra" |