John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2014 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 | #include "DeferredLayerUpdater.h" |
| 17 | |
Fedor Kudasov | d501e10 | 2019-06-21 10:22:53 +0100 | [diff] [blame] | 18 | #include <GLES2/gl2.h> |
| 19 | #include <GLES2/gl2ext.h> |
| 20 | |
Stan Iliev | 5af5d30 | 2020-01-13 11:29:18 -0500 | [diff] [blame] | 21 | // TODO: Use public SurfaceTexture APIs once available and include public NDK header file instead. |
| 22 | #include <surfacetexture/surface_texture_platform.h> |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 23 | #include "AutoBackendTextureRelease.h" |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 24 | #include "Properties.h" |
sergeyv | 3e9999b | 2017-01-19 15:37:02 -0800 | [diff] [blame] | 25 | #include "renderstate/RenderState.h" |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 26 | #include "renderthread/EglManager.h" |
| 27 | #include "renderthread/RenderThread.h" |
| 28 | #include "renderthread/VulkanManager.h" |
| 29 | |
| 30 | using namespace android::uirenderer::renderthread; |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 31 | |
| 32 | namespace android { |
| 33 | namespace uirenderer { |
| 34 | |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 35 | DeferredLayerUpdater::DeferredLayerUpdater(RenderState& renderState) |
sergeyv | 3e9999b | 2017-01-19 15:37:02 -0800 | [diff] [blame] | 36 | : mRenderState(renderState) |
| 37 | , mBlend(false) |
Stan Iliev | 5af5d30 | 2020-01-13 11:29:18 -0500 | [diff] [blame] | 38 | , mSurfaceTexture(nullptr, [](ASurfaceTexture*) {}) |
Chris Craik | d41c4d8 | 2015-01-05 15:51:13 -0800 | [diff] [blame] | 39 | , mTransform(nullptr) |
sergeyv | 00eb43d | 2017-02-13 14:34:15 -0800 | [diff] [blame] | 40 | , mGLContextAttached(false) |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 41 | , mUpdateTexImage(false) |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 42 | , mLayer(nullptr) { |
Derek Sollenberger | 28a4d99 | 2018-09-20 13:37:24 -0400 | [diff] [blame] | 43 | renderState.registerContextCallback(this); |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 44 | } |
| 45 | |
| 46 | DeferredLayerUpdater::~DeferredLayerUpdater() { |
Chris Craik | d41c4d8 | 2015-01-05 15:51:13 -0800 | [diff] [blame] | 47 | setTransform(nullptr); |
Derek Sollenberger | 28a4d99 | 2018-09-20 13:37:24 -0400 | [diff] [blame] | 48 | mRenderState.removeContextCallback(this); |
| 49 | destroyLayer(); |
| 50 | } |
| 51 | |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 52 | void DeferredLayerUpdater::setSurfaceTexture(AutoTextureRelease&& consumer) { |
| 53 | mSurfaceTexture = std::move(consumer); |
Fedor Kudasov | d501e10 | 2019-06-21 10:22:53 +0100 | [diff] [blame] | 54 | |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 55 | GLenum target = ASurfaceTexture_getCurrentTextureTarget(mSurfaceTexture.get()); |
| 56 | LOG_ALWAYS_FATAL_IF(target != GL_TEXTURE_2D && target != GL_TEXTURE_EXTERNAL_OES, |
| 57 | "set unsupported SurfaceTexture with target %x", target); |
Fedor Kudasov | d501e10 | 2019-06-21 10:22:53 +0100 | [diff] [blame] | 58 | } |
| 59 | |
Derek Sollenberger | 28a4d99 | 2018-09-20 13:37:24 -0400 | [diff] [blame] | 60 | void DeferredLayerUpdater::onContextDestroyed() { |
sergeyv | 3e9999b | 2017-01-19 15:37:02 -0800 | [diff] [blame] | 61 | destroyLayer(); |
| 62 | } |
| 63 | |
| 64 | void DeferredLayerUpdater::destroyLayer() { |
sergeyv | 00eb43d | 2017-02-13 14:34:15 -0800 | [diff] [blame] | 65 | if (!mLayer) { |
| 66 | return; |
sergeyv | 3e9999b | 2017-01-19 15:37:02 -0800 | [diff] [blame] | 67 | } |
sergeyv | 00eb43d | 2017-02-13 14:34:15 -0800 | [diff] [blame] | 68 | |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 69 | if (mSurfaceTexture.get() && mGLContextAttached) { |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 70 | ASurfaceTexture_releaseConsumerOwnership(mSurfaceTexture.get()); |
sergeyv | 00eb43d | 2017-02-13 14:34:15 -0800 | [diff] [blame] | 71 | mGLContextAttached = false; |
sergeyv | 00eb43d | 2017-02-13 14:34:15 -0800 | [diff] [blame] | 72 | } |
| 73 | |
| 74 | mLayer->postDecStrong(); |
Stan Iliev | 1a025a7 | 2018-09-05 16:35:11 -0400 | [diff] [blame] | 75 | |
sergeyv | 00eb43d | 2017-02-13 14:34:15 -0800 | [diff] [blame] | 76 | mLayer = nullptr; |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 77 | |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 78 | for (auto& [index, slot] : mImageSlots) { |
| 79 | slot.clear(mRenderState.getRenderThread().getGrContext()); |
| 80 | } |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 81 | mImageSlots.clear(); |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 82 | } |
| 83 | |
Derek Sollenberger | 674554f | 2014-02-19 16:47:32 +0000 | [diff] [blame] | 84 | void DeferredLayerUpdater::setPaint(const SkPaint* paint) { |
Chris Craik | bf6f0f2 | 2015-10-01 12:36:07 -0700 | [diff] [blame] | 85 | mAlpha = PaintUtils::getAlphaDirect(paint); |
Mike Reed | 260ab72 | 2016-10-07 15:59:20 -0400 | [diff] [blame] | 86 | mMode = PaintUtils::getBlendModeDirect(paint); |
Derek Sollenberger | be3876c | 2018-04-20 16:13:31 -0400 | [diff] [blame] | 87 | if (paint) { |
| 88 | mColorFilter = paint->refColorFilter(); |
| 89 | } else { |
| 90 | mColorFilter.reset(); |
| 91 | } |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 92 | } |
| 93 | |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 94 | status_t DeferredLayerUpdater::createReleaseFence(bool useFenceSync, EGLSyncKHR* eglFence, |
| 95 | EGLDisplay* display, int* releaseFence, |
| 96 | void* handle) { |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 97 | *display = EGL_NO_DISPLAY; |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 98 | DeferredLayerUpdater* dlu = (DeferredLayerUpdater*)handle; |
| 99 | RenderState& renderState = dlu->mRenderState; |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 100 | status_t err; |
| 101 | if (Properties::getRenderPipelineType() == RenderPipelineType::SkiaGL) { |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 102 | EglManager& eglManager = renderState.getRenderThread().eglManager(); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 103 | *display = eglManager.eglDisplay(); |
| 104 | err = eglManager.createReleaseFence(useFenceSync, eglFence, releaseFence); |
| 105 | } else { |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 106 | int previousSlot = dlu->mCurrentSlot; |
| 107 | if (previousSlot != -1) { |
| 108 | dlu->mImageSlots[previousSlot].releaseQueueOwnership( |
| 109 | renderState.getRenderThread().getGrContext()); |
| 110 | } |
| 111 | err = renderState.getRenderThread().vulkanManager().createReleaseFence( |
| 112 | releaseFence, renderState.getRenderThread().getGrContext()); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 113 | } |
| 114 | return err; |
| 115 | } |
| 116 | |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 117 | status_t DeferredLayerUpdater::fenceWait(int fence, void* handle) { |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 118 | // Wait on the producer fence for the buffer to be ready. |
| 119 | status_t err; |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 120 | DeferredLayerUpdater* dlu = (DeferredLayerUpdater*)handle; |
| 121 | RenderState& renderState = dlu->mRenderState; |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 122 | if (Properties::getRenderPipelineType() == RenderPipelineType::SkiaGL) { |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 123 | err = renderState.getRenderThread().eglManager().fenceWait(fence); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 124 | } else { |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 125 | err = renderState.getRenderThread().vulkanManager().fenceWait( |
| 126 | fence, renderState.getRenderThread().getGrContext()); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 127 | } |
| 128 | return err; |
| 129 | } |
| 130 | |
Chris Craik | d2dfd8f | 2015-12-16 14:27:20 -0800 | [diff] [blame] | 131 | void DeferredLayerUpdater::apply() { |
sergeyv | 3e9999b | 2017-01-19 15:37:02 -0800 | [diff] [blame] | 132 | if (!mLayer) { |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 133 | mLayer = new Layer(mRenderState, mColorFilter, mAlpha, mMode); |
sergeyv | 3e9999b | 2017-01-19 15:37:02 -0800 | [diff] [blame] | 134 | } |
| 135 | |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 136 | mLayer->setColorFilter(mColorFilter); |
| 137 | mLayer->setAlpha(mAlpha, mMode); |
| 138 | |
John Reck | 25fbb3f | 2014-06-12 13:46:45 -0700 | [diff] [blame] | 139 | if (mSurfaceTexture.get()) { |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 140 | if (!mGLContextAttached) { |
| 141 | mGLContextAttached = true; |
| 142 | mUpdateTexImage = true; |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 143 | ASurfaceTexture_takeConsumerOwnership(mSurfaceTexture.get()); |
Stan Iliev | 85f9096 | 2018-08-31 18:35:06 +0000 | [diff] [blame] | 144 | } |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 145 | if (mUpdateTexImage) { |
| 146 | mUpdateTexImage = false; |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 147 | android_dataspace dataspace; |
| 148 | int slot; |
| 149 | bool newContent = false; |
ramindani | ef54c03 | 2021-07-16 22:33:25 +0000 | [diff] [blame^] | 150 | ARect rect; |
| 151 | uint32_t textureTransform; |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 152 | // Note: ASurfaceTexture_dequeueBuffer discards all but the last frame. This |
| 153 | // is necessary if the SurfaceTexture queue is in synchronous mode, and we |
| 154 | // cannot tell which mode it is in. |
| 155 | AHardwareBuffer* hardwareBuffer = ASurfaceTexture_dequeueBuffer( |
ramindani | ef54c03 | 2021-07-16 22:33:25 +0000 | [diff] [blame^] | 156 | mSurfaceTexture.get(), &slot, &dataspace, &newContent, createReleaseFence, |
| 157 | fenceWait, this, &rect, &textureTransform); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 158 | |
| 159 | if (hardwareBuffer) { |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 160 | mCurrentSlot = slot; |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 161 | sk_sp<SkImage> layerImage = mImageSlots[slot].createIfNeeded( |
| 162 | hardwareBuffer, dataspace, newContent, |
| 163 | mRenderState.getRenderThread().getGrContext()); |
Stan Iliev | d125b44 | 2020-07-16 17:03:27 -0400 | [diff] [blame] | 164 | // unref to match the ref added by ASurfaceTexture_dequeueBuffer. eglCreateImageKHR |
| 165 | // (invoked by createIfNeeded) will add a ref to the AHardwareBuffer. |
| 166 | AHardwareBuffer_release(hardwareBuffer); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 167 | if (layerImage.get()) { |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 168 | // force filtration if buffer size != layer size |
| 169 | bool forceFilter = |
| 170 | mWidth != layerImage->width() || mHeight != layerImage->height(); |
ramindani | ef54c03 | 2021-07-16 22:33:25 +0000 | [diff] [blame^] | 171 | SkRect cropRect = |
| 172 | SkRect::MakeLTRB(rect.left, rect.top, rect.right, rect.bottom); |
| 173 | updateLayer(forceFilter, textureTransform, cropRect, layerImage); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 174 | } |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 175 | } |
| 176 | } |
| 177 | |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 178 | if (mTransform) { |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 179 | mLayer->getTransform() = *mTransform; |
Chris Craik | d41c4d8 | 2015-01-05 15:51:13 -0800 | [diff] [blame] | 180 | setTransform(nullptr); |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 181 | } |
| 182 | } |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 183 | } |
| 184 | |
ramindani | ef54c03 | 2021-07-16 22:33:25 +0000 | [diff] [blame^] | 185 | void DeferredLayerUpdater::updateLayer(bool forceFilter, const uint32_t textureTransform, |
| 186 | const SkRect cropRect, const sk_sp<SkImage>& layerImage) { |
Greg Daniel | 45ec62b | 2017-01-04 14:27:00 -0500 | [diff] [blame] | 187 | mLayer->setBlend(mBlend); |
| 188 | mLayer->setForceFilter(forceFilter); |
| 189 | mLayer->setSize(mWidth, mHeight); |
ramindani | ef54c03 | 2021-07-16 22:33:25 +0000 | [diff] [blame^] | 190 | mLayer->setTextureTransform(textureTransform); |
| 191 | mLayer->setCropRect(cropRect); |
Stan Iliev | 564ca3e | 2018-09-04 22:00:00 +0000 | [diff] [blame] | 192 | mLayer->setImage(layerImage); |
Greg Daniel | 45ec62b | 2017-01-04 14:27:00 -0500 | [diff] [blame] | 193 | } |
| 194 | |
John Reck | 918ad52 | 2014-06-27 14:45:25 -0700 | [diff] [blame] | 195 | void DeferredLayerUpdater::detachSurfaceTexture() { |
| 196 | if (mSurfaceTexture.get()) { |
sergeyv | 00eb43d | 2017-02-13 14:34:15 -0800 | [diff] [blame] | 197 | destroyLayer(); |
Chris Craik | d41c4d8 | 2015-01-05 15:51:13 -0800 | [diff] [blame] | 198 | mSurfaceTexture = nullptr; |
John Reck | 918ad52 | 2014-06-27 14:45:25 -0700 | [diff] [blame] | 199 | } |
| 200 | } |
| 201 | |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 202 | sk_sp<SkImage> DeferredLayerUpdater::ImageSlot::createIfNeeded(AHardwareBuffer* buffer, |
| 203 | android_dataspace dataspace, |
| 204 | bool forceCreate, |
Adlai Holler | f8c434e | 2020-07-27 11:42:45 -0400 | [diff] [blame] | 205 | GrDirectContext* context) { |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 206 | if (!mTextureRelease || !mTextureRelease->getImage().get() || dataspace != mDataspace || |
| 207 | forceCreate || mBuffer != buffer) { |
| 208 | if (buffer != mBuffer) { |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 209 | clear(context); |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 210 | } |
| 211 | |
| 212 | if (!buffer) { |
| 213 | return nullptr; |
| 214 | } |
| 215 | |
| 216 | if (!mTextureRelease) { |
| 217 | mTextureRelease = new AutoBackendTextureRelease(context, buffer); |
| 218 | } else { |
| 219 | mTextureRelease->newBufferContent(context); |
| 220 | } |
| 221 | |
| 222 | mDataspace = dataspace; |
| 223 | mBuffer = buffer; |
| 224 | mTextureRelease->makeImage(buffer, dataspace, context); |
| 225 | } |
| 226 | return mTextureRelease ? mTextureRelease->getImage() : nullptr; |
| 227 | } |
| 228 | |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 229 | void DeferredLayerUpdater::ImageSlot::clear(GrDirectContext* context) { |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 230 | if (mTextureRelease) { |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 231 | if (Properties::getRenderPipelineType() == RenderPipelineType::SkiaVulkan) { |
| 232 | this->releaseQueueOwnership(context); |
| 233 | } |
Stan Iliev | aaa9e83 | 2019-09-17 14:07:23 -0400 | [diff] [blame] | 234 | // The following unref counteracts the initial mUsageCount of 1, set by default initializer. |
| 235 | mTextureRelease->unref(true); |
| 236 | mTextureRelease = nullptr; |
| 237 | } |
| 238 | |
| 239 | mBuffer = nullptr; |
| 240 | } |
| 241 | |
Greg Daniel | 27e1fa2 | 2021-05-26 09:24:15 -0400 | [diff] [blame] | 242 | void DeferredLayerUpdater::ImageSlot::releaseQueueOwnership(GrDirectContext* context) { |
| 243 | LOG_ALWAYS_FATAL_IF(Properties::getRenderPipelineType() != RenderPipelineType::SkiaVulkan); |
| 244 | if (mTextureRelease) { |
| 245 | mTextureRelease->releaseQueueOwnership(context); |
| 246 | } |
| 247 | } |
| 248 | |
John Reck | 04fc583 | 2014-02-05 16:38:25 -0800 | [diff] [blame] | 249 | } /* namespace uirenderer */ |
| 250 | } /* namespace android */ |