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