Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2019 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 | #ifndef ANDROID_TRANSACTION_TEST_HARNESSES |
| 17 | #define ANDROID_TRANSACTION_TEST_HARNESSES |
| 18 | |
Melody Hsu | 793f836 | 2024-01-08 20:00:35 +0000 | [diff] [blame] | 19 | #include <common/FlagManager.h> |
Dominik Laskowski | 3cb3d4e | 2019-11-21 11:14:45 -0800 | [diff] [blame] | 20 | #include <ui/DisplayState.h> |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 21 | |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 22 | #include "LayerTransactionTest.h" |
Melody Hsu | 793f836 | 2024-01-08 20:00:35 +0000 | [diff] [blame] | 23 | #include "ui/LayerStack.h" |
Dominik Laskowski | 3cb3d4e | 2019-11-21 11:14:45 -0800 | [diff] [blame] | 24 | |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 25 | namespace android { |
| 26 | |
| 27 | using android::hardware::graphics::common::V1_1::BufferUsage; |
| 28 | |
| 29 | class LayerRenderPathTestHarness { |
| 30 | public: |
| 31 | LayerRenderPathTestHarness(LayerTransactionTest* delegate, RenderPath renderPath) |
| 32 | : mDelegate(delegate), mRenderPath(renderPath) {} |
| 33 | |
| 34 | std::unique_ptr<ScreenCapture> getScreenCapture() { |
| 35 | switch (mRenderPath) { |
| 36 | case RenderPath::SCREENSHOT: |
| 37 | return mDelegate->screenshot(); |
| 38 | case RenderPath::VIRTUAL_DISPLAY: |
| 39 | |
Huihong Luo | 31b5ac2 | 2022-08-15 20:38:10 -0700 | [diff] [blame] | 40 | const auto ids = SurfaceComposerClient::getPhysicalDisplayIds(); |
Melody Hsu | 793f836 | 2024-01-08 20:00:35 +0000 | [diff] [blame] | 41 | const PhysicalDisplayId displayId = ids.front(); |
Huihong Luo | 31b5ac2 | 2022-08-15 20:38:10 -0700 | [diff] [blame] | 42 | const auto displayToken = ids.empty() |
| 43 | ? nullptr |
Melody Hsu | 793f836 | 2024-01-08 20:00:35 +0000 | [diff] [blame] | 44 | : SurfaceComposerClient::getPhysicalDisplayToken(displayId); |
Dominik Laskowski | 3cb3d4e | 2019-11-21 11:14:45 -0800 | [diff] [blame] | 45 | |
| 46 | ui::DisplayState displayState; |
| 47 | SurfaceComposerClient::getDisplayState(displayToken, &displayState); |
| 48 | |
Marin Shalamanov | a7fe304 | 2021-01-29 21:02:08 +0100 | [diff] [blame] | 49 | ui::DisplayMode displayMode; |
| 50 | SurfaceComposerClient::getActiveDisplayMode(displayToken, &displayMode); |
| 51 | const ui::Size& resolution = displayMode.resolution; |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 52 | |
| 53 | sp<IBinder> vDisplay; |
| 54 | sp<IGraphicBufferProducer> producer; |
| 55 | sp<IGraphicBufferConsumer> consumer; |
| 56 | sp<BufferItemConsumer> itemConsumer; |
| 57 | BufferQueue::createBufferQueue(&producer, &consumer); |
| 58 | |
| 59 | consumer->setConsumerName(String8("Virtual disp consumer")); |
Dominik Laskowski | 3cb3d4e | 2019-11-21 11:14:45 -0800 | [diff] [blame] | 60 | consumer->setDefaultBufferSize(resolution.getWidth(), resolution.getHeight()); |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 61 | |
Ady Abraham | d11bade | 2022-08-01 16:18:03 -0700 | [diff] [blame] | 62 | itemConsumer = sp<BufferItemConsumer>::make(consumer, |
| 63 | // Sample usage bits from screenrecord |
| 64 | GRALLOC_USAGE_HW_VIDEO_ENCODER | |
| 65 | GRALLOC_USAGE_SW_READ_OFTEN); |
| 66 | sp<BufferListener> listener = sp<BufferListener>::make(this); |
Arthur Hung | 5814427 | 2021-01-16 03:43:53 +0000 | [diff] [blame] | 67 | itemConsumer->setFrameAvailableListener(listener); |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 68 | |
Alan Ding | d53801c | 2024-05-08 16:45:29 -0700 | [diff] [blame] | 69 | static const std::string kDisplayName("VirtualDisplay"); |
| 70 | vDisplay = SurfaceComposerClient::createVirtualDisplay(kDisplayName, |
| 71 | false /*isSecure*/); |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 72 | |
Melody Hsu | 793f836 | 2024-01-08 20:00:35 +0000 | [diff] [blame] | 73 | constexpr ui::LayerStack layerStack{ |
| 74 | 848472}; // ASCII for TTH (TransactionTestHarnesses) |
| 75 | sp<SurfaceControl> mirrorSc = |
| 76 | SurfaceComposerClient::getDefault()->mirrorDisplay(displayId); |
| 77 | |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 78 | SurfaceComposerClient::Transaction t; |
| 79 | t.setDisplaySurface(vDisplay, producer); |
Dominik Laskowski | 3cb3d4e | 2019-11-21 11:14:45 -0800 | [diff] [blame] | 80 | t.setDisplayProjection(vDisplay, displayState.orientation, |
Marin Shalamanov | 6ad317c | 2020-07-29 23:34:07 +0200 | [diff] [blame] | 81 | Rect(displayState.layerStackSpaceRect), Rect(resolution)); |
Melody Hsu | 793f836 | 2024-01-08 20:00:35 +0000 | [diff] [blame] | 82 | if (FlagManager::getInstance().ce_fence_promise()) { |
| 83 | t.setDisplayLayerStack(vDisplay, layerStack); |
| 84 | t.setLayerStack(mirrorSc, layerStack); |
| 85 | } else { |
| 86 | t.setDisplayLayerStack(vDisplay, ui::DEFAULT_LAYER_STACK); |
| 87 | } |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 88 | t.apply(); |
| 89 | SurfaceComposerClient::Transaction().apply(true); |
Arthur Hung | 5814427 | 2021-01-16 03:43:53 +0000 | [diff] [blame] | 90 | |
| 91 | std::unique_lock lock(mMutex); |
| 92 | mAvailable = false; |
| 93 | // Wait for frame buffer ready. |
| 94 | mCondition.wait_for(lock, std::chrono::seconds(2), |
| 95 | [this]() NO_THREAD_SAFETY_ANALYSIS { return mAvailable; }); |
| 96 | |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 97 | BufferItem item; |
| 98 | itemConsumer->acquireBuffer(&item, 0, true); |
Alec Mouri | 14d5b86 | 2022-04-27 21:20:04 +0000 | [diff] [blame] | 99 | constexpr bool kContainsHdr = false; |
| 100 | auto sc = std::make_unique<ScreenCapture>(item.mGraphicBuffer, kContainsHdr); |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 101 | itemConsumer->releaseBuffer(item); |
Melody Hsu | 793f836 | 2024-01-08 20:00:35 +0000 | [diff] [blame] | 102 | |
| 103 | // Possible race condition with destroying virtual displays, in which |
| 104 | // CompositionEngine::present may attempt to be called on the same |
| 105 | // display multiple times. The layerStack is set to invalid here so |
| 106 | // that the display is ignored if that scenario occurs. |
| 107 | if (FlagManager::getInstance().ce_fence_promise()) { |
| 108 | t.setLayerStack(mirrorSc, ui::INVALID_LAYER_STACK); |
| 109 | t.apply(true); |
| 110 | } |
Alan Ding | d53801c | 2024-05-08 16:45:29 -0700 | [diff] [blame] | 111 | SurfaceComposerClient::destroyVirtualDisplay(vDisplay); |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 112 | return sc; |
| 113 | } |
| 114 | } |
| 115 | |
| 116 | protected: |
| 117 | LayerTransactionTest* mDelegate; |
| 118 | RenderPath mRenderPath; |
Arthur Hung | 5814427 | 2021-01-16 03:43:53 +0000 | [diff] [blame] | 119 | std::mutex mMutex; |
| 120 | std::condition_variable mCondition; |
| 121 | bool mAvailable = false; |
| 122 | |
| 123 | void onFrameAvailable() { |
| 124 | std::unique_lock lock(mMutex); |
| 125 | mAvailable = true; |
| 126 | mCondition.notify_all(); |
| 127 | } |
| 128 | |
| 129 | class BufferListener : public ConsumerBase::FrameAvailableListener { |
| 130 | public: |
| 131 | BufferListener(LayerRenderPathTestHarness* owner) : mOwner(owner) {} |
| 132 | LayerRenderPathTestHarness* mOwner; |
| 133 | |
| 134 | void onFrameAvailable(const BufferItem& /*item*/) { mOwner->onFrameAvailable(); } |
| 135 | }; |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 136 | }; |
| 137 | |
| 138 | class LayerTypeTransactionHarness : public LayerTransactionTest { |
| 139 | public: |
| 140 | LayerTypeTransactionHarness(uint32_t layerType) : mLayerType(layerType) {} |
| 141 | |
| 142 | sp<SurfaceControl> createLayer(const char* name, uint32_t width, uint32_t height, |
Valerie Hau | 1acd696 | 2019-10-28 16:35:48 -0700 | [diff] [blame] | 143 | uint32_t flags = 0, SurfaceControl* parent = nullptr, |
chaviw | debadb8 | 2020-03-26 14:57:24 -0700 | [diff] [blame] | 144 | uint32_t* outTransformHint = nullptr, |
| 145 | PixelFormat format = PIXEL_FORMAT_RGBA_8888) { |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 146 | // if the flags already have a layer type specified, return an error |
| 147 | if (flags & ISurfaceComposerClient::eFXSurfaceMask) { |
| 148 | return nullptr; |
| 149 | } |
Valerie Hau | 1acd696 | 2019-10-28 16:35:48 -0700 | [diff] [blame] | 150 | return LayerTransactionTest::createLayer(name, width, height, flags | mLayerType, parent, |
chaviw | debadb8 | 2020-03-26 14:57:24 -0700 | [diff] [blame] | 151 | outTransformHint, format); |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 152 | } |
| 153 | |
Marin Shalamanov | 4608442 | 2020-10-13 12:33:42 +0200 | [diff] [blame] | 154 | void fillLayerColor(const sp<SurfaceControl>& layer, const Color& color, uint32_t bufferWidth, |
| 155 | uint32_t bufferHeight) { |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 156 | ASSERT_NO_FATAL_FAILURE(LayerTransactionTest::fillLayerColor(mLayerType, layer, color, |
| 157 | bufferWidth, bufferHeight)); |
| 158 | } |
| 159 | |
Marin Shalamanov | 4608442 | 2020-10-13 12:33:42 +0200 | [diff] [blame] | 160 | void fillLayerQuadrant(const sp<SurfaceControl>& layer, uint32_t bufferWidth, |
| 161 | uint32_t bufferHeight, const Color& topLeft, const Color& topRight, |
Valerie Hau | 9cfc6d8 | 2019-09-23 13:54:07 -0700 | [diff] [blame] | 162 | const Color& bottomLeft, const Color& bottomRight) { |
| 163 | ASSERT_NO_FATAL_FAILURE(LayerTransactionTest::fillLayerQuadrant(mLayerType, layer, |
| 164 | bufferWidth, bufferHeight, |
| 165 | topLeft, topRight, |
| 166 | bottomLeft, bottomRight)); |
| 167 | } |
| 168 | |
| 169 | protected: |
| 170 | uint32_t mLayerType; |
| 171 | }; |
| 172 | } // namespace android |
| 173 | #endif |