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