| /* |
| * Copyright 2019 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. |
| */ |
| |
| #include <cstdarg> |
| #include <cstdint> |
| |
| #include <compositionengine/RenderSurfaceCreationArgs.h> |
| #include <compositionengine/impl/RenderSurface.h> |
| #include <compositionengine/mock/CompositionEngine.h> |
| #include <compositionengine/mock/Display.h> |
| #include <compositionengine/mock/DisplaySurface.h> |
| #include <compositionengine/mock/OutputLayer.h> |
| #include <gtest/gtest.h> |
| #include <renderengine/mock/RenderEngine.h> |
| #include <system/window.h> |
| #include <ui/ANativeObjectBase.h> |
| |
| #include "MockHWComposer.h" |
| |
| namespace android::compositionengine { |
| namespace { |
| |
| /* ------------------------------------------------------------------------ |
| * MockNativeWindow |
| * |
| * An intentionally simplified Mock which implements a minimal subset of the full |
| * ANativeWindow interface. |
| */ |
| |
| class MockNativeWindow : public ANativeObjectBase<ANativeWindow, MockNativeWindow, RefBase> { |
| public: |
| MockNativeWindow() { |
| ANativeWindow::setSwapInterval = &forwardSetSwapInterval; |
| ANativeWindow::dequeueBuffer = &forwardDequeueBuffer; |
| ANativeWindow::cancelBuffer = &forwardCancelBuffer; |
| ANativeWindow::queueBuffer = &forwardQueueBuffer; |
| ANativeWindow::query = &forwardQuery; |
| ANativeWindow::perform = &forwardPerform; |
| |
| ANativeWindow::dequeueBuffer_DEPRECATED = &forwardDequeueBufferDeprecated; |
| ANativeWindow::cancelBuffer_DEPRECATED = &forwardCancelBufferDeprecated; |
| ANativeWindow::lockBuffer_DEPRECATED = &forwardLockBufferDeprecated; |
| ANativeWindow::queueBuffer_DEPRECATED = &forwardQueueBufferDeprecated; |
| } |
| |
| MOCK_METHOD1(setSwapInterval, int(int)); |
| MOCK_METHOD2(dequeueBuffer, int(struct ANativeWindowBuffer**, int*)); |
| MOCK_METHOD2(cancelBuffer, int(struct ANativeWindowBuffer*, int)); |
| MOCK_METHOD2(queueBuffer, int(struct ANativeWindowBuffer*, int)); |
| MOCK_CONST_METHOD2(query, int(int, int*)); |
| MOCK_METHOD1(connect, int(int)); |
| MOCK_METHOD1(lockBuffer_DEPRECATED, int(struct ANativeWindowBuffer*)); |
| MOCK_METHOD1(setBuffersFormat, int(PixelFormat)); |
| MOCK_METHOD1(setBuffersDataSpace, int(ui::Dataspace)); |
| MOCK_METHOD1(setUsage, int(uint64_t)); |
| |
| static void unexpectedCall(...) { LOG_ALWAYS_FATAL("Unexpected ANativeWindow API call"); } |
| |
| static int forwardSetSwapInterval(ANativeWindow* window, int interval) { |
| return getSelf(window)->setSwapInterval(interval); |
| } |
| |
| static int forwardDequeueBuffer(ANativeWindow* window, ANativeWindowBuffer** buffer, |
| int* fenceFd) { |
| return getSelf(window)->dequeueBuffer(buffer, fenceFd); |
| } |
| |
| static int forwardCancelBuffer(ANativeWindow* window, ANativeWindowBuffer* buffer, |
| int fenceFd) { |
| return getSelf(window)->cancelBuffer(buffer, fenceFd); |
| } |
| |
| static int forwardQueueBuffer(ANativeWindow* window, ANativeWindowBuffer* buffer, int fenceFd) { |
| return getSelf(window)->queueBuffer(buffer, fenceFd); |
| } |
| |
| static int forwardQuery(const ANativeWindow* window, int what, int* value) { |
| return getSelf(window)->query(what, value); |
| } |
| |
| static int forwardPerform(ANativeWindow* window, int operation, ...) { |
| va_list args; |
| va_start(args, operation); |
| int result = NO_ERROR; |
| switch (operation) { |
| case NATIVE_WINDOW_API_CONNECT: { |
| int api = va_arg(args, int); |
| result = getSelf(window)->connect(api); |
| break; |
| } |
| case NATIVE_WINDOW_SET_BUFFERS_FORMAT: { |
| PixelFormat format = va_arg(args, PixelFormat); |
| result = getSelf(window)->setBuffersFormat(format); |
| break; |
| } |
| case NATIVE_WINDOW_SET_BUFFERS_DATASPACE: { |
| ui::Dataspace dataspace = static_cast<ui::Dataspace>(va_arg(args, int)); |
| result = getSelf(window)->setBuffersDataSpace(dataspace); |
| break; |
| } |
| case NATIVE_WINDOW_SET_USAGE: { |
| // Note: Intentionally widens usage from 32 to 64 bits so we |
| // just have one implementation. |
| uint64_t usage = va_arg(args, uint32_t); |
| result = getSelf(window)->setUsage(usage); |
| break; |
| } |
| case NATIVE_WINDOW_SET_USAGE64: { |
| uint64_t usage = va_arg(args, uint64_t); |
| result = getSelf(window)->setUsage(usage); |
| break; |
| } |
| default: |
| LOG_ALWAYS_FATAL("Unexpected operation %d", operation); |
| break; |
| } |
| |
| va_end(args); |
| return result; |
| } |
| |
| static int forwardDequeueBufferDeprecated(ANativeWindow* window, ANativeWindowBuffer** buffer) { |
| int ignoredFenceFd = -1; |
| return getSelf(window)->dequeueBuffer(buffer, &ignoredFenceFd); |
| } |
| |
| static int forwardCancelBufferDeprecated(ANativeWindow* window, ANativeWindowBuffer* buffer) { |
| return getSelf(window)->cancelBuffer(buffer, -1); |
| } |
| |
| static int forwardLockBufferDeprecated(ANativeWindow* window, ANativeWindowBuffer* buffer) { |
| return getSelf(window)->lockBuffer_DEPRECATED(buffer); |
| } |
| |
| static int forwardQueueBufferDeprecated(ANativeWindow* window, ANativeWindowBuffer* buffer) { |
| return getSelf(window)->queueBuffer(buffer, -1); |
| } |
| }; |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurfaceTest |
| */ |
| |
| constexpr int32_t DEFAULT_DISPLAY_WIDTH = 1920; |
| constexpr int32_t DEFAULT_DISPLAY_HEIGHT = 1080; |
| constexpr std::optional<DisplayId> DEFAULT_DISPLAY_ID = std::make_optional(DisplayId{123u}); |
| const std::string DEFAULT_DISPLAY_NAME = "Mock Display"; |
| |
| using testing::_; |
| using testing::ByMove; |
| using testing::DoAll; |
| using testing::Ref; |
| using testing::Return; |
| using testing::ReturnRef; |
| using testing::SetArgPointee; |
| using testing::StrictMock; |
| |
| class RenderSurfaceTest : public testing::Test { |
| public: |
| RenderSurfaceTest() { |
| EXPECT_CALL(mDisplay, getId()).WillRepeatedly(ReturnRef(DEFAULT_DISPLAY_ID)); |
| EXPECT_CALL(mDisplay, getName()).WillRepeatedly(ReturnRef(DEFAULT_DISPLAY_NAME)); |
| EXPECT_CALL(mCompositionEngine, getHwComposer).WillRepeatedly(ReturnRef(mHwComposer)); |
| EXPECT_CALL(mCompositionEngine, getRenderEngine).WillRepeatedly(ReturnRef(mRenderEngine)); |
| } |
| ~RenderSurfaceTest() override = default; |
| |
| StrictMock<android::mock::HWComposer> mHwComposer; |
| StrictMock<renderengine::mock::RenderEngine> mRenderEngine; |
| StrictMock<mock::CompositionEngine> mCompositionEngine; |
| StrictMock<mock::Display> mDisplay; |
| sp<MockNativeWindow> mNativeWindow = new StrictMock<MockNativeWindow>(); |
| sp<mock::DisplaySurface> mDisplaySurface = new StrictMock<mock::DisplaySurface>(); |
| impl::RenderSurface mSurface{mCompositionEngine, mDisplay, |
| RenderSurfaceCreationArgs{DEFAULT_DISPLAY_WIDTH, |
| DEFAULT_DISPLAY_HEIGHT, mNativeWindow, |
| mDisplaySurface}}; |
| }; |
| |
| /* ------------------------------------------------------------------------ |
| * Basic construction |
| */ |
| |
| TEST_F(RenderSurfaceTest, canInstantiate) { |
| EXPECT_TRUE(mSurface.isValid()); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::initialize() |
| */ |
| |
| TEST_F(RenderSurfaceTest, initializeConfiguresNativeWindow) { |
| EXPECT_CALL(*mNativeWindow, connect(NATIVE_WINDOW_API_EGL)).WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(*mNativeWindow, setBuffersFormat(HAL_PIXEL_FORMAT_RGBA_8888)) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(*mNativeWindow, setUsage(GRALLOC_USAGE_HW_RENDER)).WillOnce(Return(NO_ERROR)); |
| |
| mSurface.initialize(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::getSize() |
| */ |
| |
| TEST_F(RenderSurfaceTest, sizeReturnsConstructedSize) { |
| const ui::Size expected{DEFAULT_DISPLAY_WIDTH, DEFAULT_DISPLAY_HEIGHT}; |
| |
| EXPECT_EQ(expected, mSurface.getSize()); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::getClientTargetAcquireFence() |
| */ |
| |
| TEST_F(RenderSurfaceTest, getClientTargetAcquireFenceForwardsCall) { |
| sp<Fence> fence = new Fence(); |
| |
| EXPECT_CALL(*mDisplaySurface, getClientTargetAcquireFence()).WillOnce(ReturnRef(fence)); |
| |
| EXPECT_EQ(fence.get(), mSurface.getClientTargetAcquireFence().get()); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::setDisplaySize() |
| */ |
| |
| TEST_F(RenderSurfaceTest, setDisplaySizeAppliesChange) { |
| EXPECT_CALL(*mDisplaySurface, resizeBuffers(640, 480)).Times(1); |
| |
| mSurface.setDisplaySize(ui::Size(640, 480)); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::setBufferDataspace() |
| */ |
| |
| TEST_F(RenderSurfaceTest, setBufferDataspaceAppliesChange) { |
| EXPECT_CALL(*mNativeWindow, setBuffersDataSpace(ui::Dataspace::DISPLAY_P3)) |
| .WillOnce(Return(NO_ERROR)); |
| |
| mSurface.setBufferDataspace(ui::Dataspace::DISPLAY_P3); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::setProtected() |
| */ |
| |
| TEST_F(RenderSurfaceTest, setProtectedTrueEnablesProtection) { |
| EXPECT_CALL(*mNativeWindow, setUsage(GRALLOC_USAGE_HW_RENDER | GRALLOC_USAGE_PROTECTED)) |
| .WillOnce(Return(NO_ERROR)); |
| |
| mSurface.setProtected(true); |
| } |
| |
| TEST_F(RenderSurfaceTest, setProtectedFalseDisablesProtection) { |
| EXPECT_CALL(*mNativeWindow, setUsage(GRALLOC_USAGE_HW_RENDER)).WillOnce(Return(NO_ERROR)); |
| |
| mSurface.setProtected(false); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::beginFrame() |
| */ |
| |
| TEST_F(RenderSurfaceTest, beginFrameAppliesChange) { |
| EXPECT_CALL(*mDisplaySurface, beginFrame(true)).WillOnce(Return(NO_ERROR)); |
| |
| EXPECT_EQ(NO_ERROR, mSurface.beginFrame(true)); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::prepareFrame() |
| */ |
| |
| TEST_F(RenderSurfaceTest, prepareFramePassesOutputLayersToHwc) { |
| EXPECT_CALL(mHwComposer, prepare(*DEFAULT_DISPLAY_ID, Ref(mDisplay))) |
| .WillOnce(Return(INVALID_OPERATION)); |
| |
| EXPECT_EQ(INVALID_OPERATION, mSurface.prepareFrame()); |
| } |
| |
| TEST_F(RenderSurfaceTest, prepareFrameTakesEarlyOutOnHwcError) { |
| EXPECT_CALL(mHwComposer, prepare(*DEFAULT_DISPLAY_ID, Ref(mDisplay))) |
| .WillOnce(Return(INVALID_OPERATION)); |
| |
| EXPECT_EQ(INVALID_OPERATION, mSurface.prepareFrame()); |
| } |
| |
| TEST_F(RenderSurfaceTest, prepareFrameHandlesMixedComposition) { |
| EXPECT_CALL(mHwComposer, prepare(*DEFAULT_DISPLAY_ID, Ref(mDisplay))) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| EXPECT_CALL(mHwComposer, hasDeviceComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| |
| EXPECT_CALL(*mDisplaySurface, prepareFrame(DisplaySurface::COMPOSITION_MIXED)) |
| .WillOnce(Return(INVALID_OPERATION)); |
| |
| EXPECT_EQ(INVALID_OPERATION, mSurface.prepareFrame()); |
| } |
| |
| TEST_F(RenderSurfaceTest, prepareFrameHandlesOnlyGlesComposition) { |
| EXPECT_CALL(mHwComposer, prepare(*DEFAULT_DISPLAY_ID, Ref(mDisplay))) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| EXPECT_CALL(mHwComposer, hasDeviceComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(false)); |
| |
| EXPECT_CALL(*mDisplaySurface, prepareFrame(DisplaySurface::COMPOSITION_GLES)) |
| .WillOnce(Return(NO_ERROR)); |
| |
| EXPECT_EQ(NO_ERROR, mSurface.prepareFrame()); |
| } |
| |
| TEST_F(RenderSurfaceTest, prepareFrameHandlesOnlyHwcComposition) { |
| EXPECT_CALL(mHwComposer, prepare(*DEFAULT_DISPLAY_ID, Ref(mDisplay))) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(false)); |
| EXPECT_CALL(mHwComposer, hasDeviceComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| |
| EXPECT_CALL(*mDisplaySurface, prepareFrame(DisplaySurface::COMPOSITION_HWC)) |
| .WillOnce(Return(NO_ERROR)); |
| |
| EXPECT_EQ(NO_ERROR, mSurface.prepareFrame()); |
| } |
| |
| TEST_F(RenderSurfaceTest, prepareFrameHandlesNoComposition) { |
| EXPECT_CALL(mHwComposer, prepare(*DEFAULT_DISPLAY_ID, Ref(mDisplay))) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(false)); |
| EXPECT_CALL(mHwComposer, hasDeviceComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(false)); |
| |
| EXPECT_CALL(*mDisplaySurface, prepareFrame(DisplaySurface::COMPOSITION_HWC)) |
| .WillOnce(Return(NO_ERROR)); |
| |
| EXPECT_EQ(NO_ERROR, mSurface.prepareFrame()); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::dequeueBuffer() |
| */ |
| |
| TEST_F(RenderSurfaceTest, dequeueBufferObtainsABuffer) { |
| sp<GraphicBuffer> buffer = new GraphicBuffer(); |
| |
| EXPECT_CALL(*mNativeWindow, dequeueBuffer(_, _)) |
| .WillOnce( |
| DoAll(SetArgPointee<0>(buffer.get()), SetArgPointee<1>(-1), Return(NO_ERROR))); |
| |
| EXPECT_EQ(buffer.get(), mSurface.dequeueBuffer().get()); |
| |
| EXPECT_EQ(buffer.get(), mSurface.mutableGraphicBufferForTest().get()); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::queueBuffer() |
| */ |
| |
| TEST_F(RenderSurfaceTest, queueBufferHandlesNoClientComposition) { |
| sp<GraphicBuffer> buffer = new GraphicBuffer(); |
| mSurface.mutableGraphicBufferForTest() = buffer; |
| |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(false)); |
| EXPECT_CALL(mHwComposer, hasFlipClientTargetRequest(DEFAULT_DISPLAY_ID)) |
| .WillOnce(Return(false)); |
| EXPECT_CALL(*mDisplaySurface, advanceFrame()).Times(1); |
| |
| mSurface.queueBuffer(base::unique_fd()); |
| |
| EXPECT_EQ(buffer.get(), mSurface.mutableGraphicBufferForTest().get()); |
| } |
| |
| TEST_F(RenderSurfaceTest, queueBufferHandlesClientComposition) { |
| sp<GraphicBuffer> buffer = new GraphicBuffer(); |
| mSurface.mutableGraphicBufferForTest() = buffer; |
| |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| EXPECT_CALL(*mNativeWindow, queueBuffer(buffer->getNativeBuffer(), -1)) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(*mDisplaySurface, advanceFrame()).Times(1); |
| |
| mSurface.queueBuffer(base::unique_fd()); |
| |
| EXPECT_EQ(nullptr, mSurface.mutableGraphicBufferForTest().get()); |
| } |
| |
| TEST_F(RenderSurfaceTest, queueBufferHandlesFlipClientTargetRequest) { |
| sp<GraphicBuffer> buffer = new GraphicBuffer(); |
| mSurface.mutableGraphicBufferForTest() = buffer; |
| |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(false)); |
| EXPECT_CALL(mHwComposer, hasFlipClientTargetRequest(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| EXPECT_CALL(*mNativeWindow, queueBuffer(buffer->getNativeBuffer(), -1)) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(*mDisplaySurface, advanceFrame()).Times(1); |
| |
| mSurface.queueBuffer(base::unique_fd()); |
| |
| EXPECT_EQ(nullptr, mSurface.mutableGraphicBufferForTest().get()); |
| } |
| |
| TEST_F(RenderSurfaceTest, queueBufferHandlesFlipClientTargetRequestWithNoBufferYetDequeued) { |
| sp<GraphicBuffer> buffer = new GraphicBuffer(); |
| |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(false)); |
| EXPECT_CALL(mHwComposer, hasFlipClientTargetRequest(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| EXPECT_CALL(*mNativeWindow, dequeueBuffer(_, _)) |
| .WillOnce( |
| DoAll(SetArgPointee<0>(buffer.get()), SetArgPointee<1>(-1), Return(NO_ERROR))); |
| EXPECT_CALL(*mNativeWindow, queueBuffer(buffer->getNativeBuffer(), -1)) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(*mDisplaySurface, advanceFrame()).Times(1); |
| |
| mSurface.queueBuffer(base::unique_fd()); |
| |
| EXPECT_EQ(nullptr, mSurface.mutableGraphicBufferForTest().get()); |
| } |
| |
| TEST_F(RenderSurfaceTest, queueBufferHandlesNativeWindowQueueBufferFailureOnVirtualDisplay) { |
| sp<GraphicBuffer> buffer = new GraphicBuffer(); |
| mSurface.mutableGraphicBufferForTest() = buffer; |
| |
| EXPECT_CALL(mHwComposer, hasClientComposition(DEFAULT_DISPLAY_ID)).WillOnce(Return(true)); |
| EXPECT_CALL(*mNativeWindow, queueBuffer(buffer->getNativeBuffer(), -1)) |
| .WillOnce(Return(INVALID_OPERATION)); |
| EXPECT_CALL(mDisplay, isVirtual()).WillOnce(Return(true)); |
| EXPECT_CALL(*mNativeWindow, cancelBuffer(buffer->getNativeBuffer(), -1)) |
| .WillOnce(Return(NO_ERROR)); |
| EXPECT_CALL(*mDisplaySurface, advanceFrame()).Times(1); |
| |
| mSurface.queueBuffer(base::unique_fd()); |
| |
| EXPECT_EQ(nullptr, mSurface.mutableGraphicBufferForTest().get()); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::onPresentDisplayCompleted() |
| */ |
| |
| TEST_F(RenderSurfaceTest, onPresentDisplayCompletedForwardsSignal) { |
| EXPECT_CALL(*mDisplaySurface, onFrameCommitted()).Times(1); |
| |
| mSurface.onPresentDisplayCompleted(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::setViewportAndProjection() |
| */ |
| |
| TEST_F(RenderSurfaceTest, setViewportAndProjectionAppliesChang) { |
| mSurface.setSizeForTest(ui::Size(100, 200)); |
| |
| EXPECT_CALL(mRenderEngine, |
| setViewportAndProjection(100, 200, Rect(100, 200), ui::Transform::ROT_0)) |
| .Times(1); |
| |
| mSurface.setViewportAndProjection(); |
| } |
| |
| /* ------------------------------------------------------------------------ |
| * RenderSurface::flip() |
| */ |
| |
| TEST_F(RenderSurfaceTest, flipForwardsSignal) { |
| mSurface.setPageFlipCountForTest(500); |
| |
| mSurface.flip(); |
| |
| EXPECT_EQ(501, mSurface.getPageFlipCount()); |
| } |
| |
| } // namespace |
| } // namespace android::compositionengine |