Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2018 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "src/gpu/GrProxyProvider.h" |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 9 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 10 | #include "include/core/SkBitmap.h" |
| 11 | #include "include/core/SkImage.h" |
Robert Phillips | b7bfbc2 | 2020-07-01 12:55:01 -0400 | [diff] [blame] | 12 | #include "include/gpu/GrDirectContext.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 13 | #include "include/private/GrImageContext.h" |
| 14 | #include "include/private/GrResourceKey.h" |
| 15 | #include "include/private/GrSingleOwner.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 16 | #include "include/private/SkImageInfoPriv.h" |
| 17 | #include "src/core/SkAutoPixmapStorage.h" |
Robert Phillips | 99dead9 | 2020-01-27 16:11:57 -0500 | [diff] [blame] | 18 | #include "src/core/SkCompressedDataUtils.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 19 | #include "src/core/SkImagePriv.h" |
Mike Reed | 13711eb | 2020-07-14 17:16:32 -0400 | [diff] [blame] | 20 | #include "src/core/SkMipmap.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 21 | #include "src/core/SkTraceEvent.h" |
| 22 | #include "src/gpu/GrCaps.h" |
Adlai Holler | dced31f | 2021-02-19 12:33:46 -0500 | [diff] [blame] | 23 | #include "src/gpu/GrContextThreadSafeProxyPriv.h" |
Adlai Holler | a069304 | 2020-10-14 11:23:11 -0400 | [diff] [blame] | 24 | #include "src/gpu/GrDirectContextPriv.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 25 | #include "src/gpu/GrImageContextPriv.h" |
Brian Salomon | 201cdbb | 2019-08-14 17:00:30 -0400 | [diff] [blame] | 26 | #include "src/gpu/GrRenderTarget.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 27 | #include "src/gpu/GrResourceProvider.h" |
Greg Daniel | f91aeb2 | 2019-06-18 09:58:02 -0400 | [diff] [blame] | 28 | #include "src/gpu/GrSurfaceProxy.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 29 | #include "src/gpu/GrSurfaceProxyPriv.h" |
Greg Daniel | 456f9b5 | 2020-03-05 19:14:18 +0000 | [diff] [blame] | 30 | #include "src/gpu/GrTexture.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 31 | #include "src/gpu/GrTextureProxyCacheAccess.h" |
| 32 | #include "src/gpu/GrTextureRenderTargetProxy.h" |
| 33 | #include "src/gpu/SkGr.h" |
| 34 | #include "src/image/SkImage_Base.h" |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 35 | |
Greg Daniel | 8426165 | 2021-09-19 17:53:40 -0400 | [diff] [blame] | 36 | #ifdef SK_VULKAN |
| 37 | #include "include/gpu/vk/GrVkTypes.h" |
| 38 | #endif |
| 39 | |
Adlai Holler | 33dbd65 | 2020-06-01 12:35:42 -0400 | [diff] [blame] | 40 | #define ASSERT_SINGLE_OWNER GR_ASSERT_SINGLE_OWNER(fImageContext->priv().singleOwner()) |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 41 | |
Robert Phillips | a9162df | 2019-02-11 14:12:03 -0500 | [diff] [blame] | 42 | GrProxyProvider::GrProxyProvider(GrImageContext* imageContext) : fImageContext(imageContext) {} |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 43 | |
| 44 | GrProxyProvider::~GrProxyProvider() { |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 45 | if (this->renderingDirectly()) { |
Robert Phillips | 0790f8a | 2018-09-18 13:11:03 -0400 | [diff] [blame] | 46 | // In DDL-mode a proxy provider can still have extant uniquely keyed proxies (since |
| 47 | // they need their unique keys to, potentially, find a cached resource when the |
| 48 | // DDL is played) but, in non-DDL-mode they should all have been cleaned up by this point. |
| 49 | SkASSERT(!fUniquelyKeyedProxies.count()); |
| 50 | } |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 51 | } |
| 52 | |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 53 | bool GrProxyProvider::assignUniqueKeyToProxy(const GrUniqueKey& key, GrTextureProxy* proxy) { |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 54 | ASSERT_SINGLE_OWNER |
| 55 | SkASSERT(key.isValid()); |
| 56 | if (this->isAbandoned() || !proxy) { |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 57 | return false; |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 58 | } |
| 59 | |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 60 | // Only the proxyProvider that created a proxy should be assigning unique keys to it. |
| 61 | SkASSERT(this->isDDLProvider() == proxy->creatingProvider()); |
| 62 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 63 | #ifdef SK_DEBUG |
| 64 | { |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 65 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 66 | if (direct) { |
| 67 | GrResourceCache* resourceCache = direct->priv().getResourceCache(); |
| 68 | // If there is already a GrResource with this key then the caller has violated the |
| 69 | // normal usage pattern of uniquely keyed resources (e.g., they have created one w/o |
| 70 | // first seeing if it already existed in the cache). |
| 71 | SkASSERT(!resourceCache->findAndRefUniqueResource(key)); |
| 72 | } |
| 73 | } |
| 74 | #endif |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 75 | |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 76 | SkASSERT(!fUniquelyKeyedProxies.find(key)); // multiple proxies can't get the same key |
| 77 | |
| 78 | proxy->cacheAccess().setUniqueKey(this, key); |
| 79 | SkASSERT(proxy->getUniqueKey() == key); |
| 80 | fUniquelyKeyedProxies.add(proxy); |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 81 | return true; |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 82 | } |
| 83 | |
| 84 | void GrProxyProvider::adoptUniqueKeyFromSurface(GrTextureProxy* proxy, const GrSurface* surf) { |
| 85 | SkASSERT(surf->getUniqueKey().isValid()); |
| 86 | proxy->cacheAccess().setUniqueKey(this, surf->getUniqueKey()); |
| 87 | SkASSERT(proxy->getUniqueKey() == surf->getUniqueKey()); |
| 88 | // multiple proxies can't get the same key |
| 89 | SkASSERT(!fUniquelyKeyedProxies.find(surf->getUniqueKey())); |
| 90 | fUniquelyKeyedProxies.add(proxy); |
| 91 | } |
| 92 | |
Chris Dalton | 2de13dd | 2019-01-03 15:11:59 -0700 | [diff] [blame] | 93 | void GrProxyProvider::removeUniqueKeyFromProxy(GrTextureProxy* proxy) { |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 94 | ASSERT_SINGLE_OWNER |
Chris Dalton | 2de13dd | 2019-01-03 15:11:59 -0700 | [diff] [blame] | 95 | SkASSERT(proxy); |
| 96 | SkASSERT(proxy->getUniqueKey().isValid()); |
| 97 | |
| 98 | if (this->isAbandoned()) { |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 99 | return; |
| 100 | } |
Robert Phillips | 0790f8a | 2018-09-18 13:11:03 -0400 | [diff] [blame] | 101 | |
Chris Dalton | 2de13dd | 2019-01-03 15:11:59 -0700 | [diff] [blame] | 102 | this->processInvalidUniqueKey(proxy->getUniqueKey(), proxy, InvalidateGPUResource::kYes); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 103 | } |
| 104 | |
Greg Daniel | 3a36511 | 2020-02-14 10:47:18 -0500 | [diff] [blame] | 105 | sk_sp<GrTextureProxy> GrProxyProvider::findProxyByUniqueKey(const GrUniqueKey& key) { |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 106 | ASSERT_SINGLE_OWNER |
| 107 | |
| 108 | if (this->isAbandoned()) { |
| 109 | return nullptr; |
| 110 | } |
| 111 | |
Brian Salomon | 01ceae9 | 2019-04-02 11:49:54 -0400 | [diff] [blame] | 112 | GrTextureProxy* proxy = fUniquelyKeyedProxies.find(key); |
Brian Salomon | 01ceae9 | 2019-04-02 11:49:54 -0400 | [diff] [blame] | 113 | if (proxy) { |
Robert Phillips | e5f7328 | 2019-06-18 17:15:04 -0400 | [diff] [blame] | 114 | return sk_ref_sp(proxy); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 115 | } |
Robert Phillips | e5f7328 | 2019-06-18 17:15:04 -0400 | [diff] [blame] | 116 | return nullptr; |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 117 | } |
| 118 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 119 | /////////////////////////////////////////////////////////////////////////////// |
| 120 | |
| 121 | #if GR_TEST_UTILS |
| 122 | sk_sp<GrTextureProxy> GrProxyProvider::testingOnly_createInstantiatedProxy( |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 123 | SkISize dimensions, |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 124 | const GrBackendFormat& format, |
| 125 | GrRenderable renderable, |
| 126 | int renderTargetSampleCnt, |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 127 | SkBackingFit fit, |
| 128 | SkBudgeted budgeted, |
| 129 | GrProtected isProtected) { |
Greg Daniel | e20fcad | 2020-01-08 11:52:34 -0500 | [diff] [blame] | 130 | ASSERT_SINGLE_OWNER |
| 131 | if (this->isAbandoned()) { |
| 132 | return nullptr; |
| 133 | } |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 134 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 135 | if (!direct) { |
| 136 | return nullptr; |
| 137 | } |
| 138 | |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 139 | if (this->caps()->isFormatCompressed(format)) { |
| 140 | // TODO: Allow this to go to GrResourceProvider::createCompressedTexture() once we no longer |
Greg Daniel | 4cb2933 | 2020-01-23 10:07:02 -0500 | [diff] [blame] | 141 | // rely on GrColorType to get a swizzle for the proxy. |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 142 | return nullptr; |
| 143 | } |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 144 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 145 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| 146 | sk_sp<GrTexture> tex; |
| 147 | |
| 148 | if (SkBackingFit::kApprox == fit) { |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 149 | tex = resourceProvider->createApproxTexture(dimensions, |
| 150 | format, |
| 151 | format.textureType(), |
| 152 | renderable, |
| 153 | renderTargetSampleCnt, |
| 154 | isProtected); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 155 | } else { |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 156 | tex = resourceProvider->createTexture(dimensions, |
| 157 | format, |
| 158 | format.textureType(), |
| 159 | renderable, |
| 160 | renderTargetSampleCnt, |
| 161 | GrMipmapped::kNo, |
| 162 | budgeted, |
| 163 | isProtected); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 164 | } |
| 165 | if (!tex) { |
| 166 | return nullptr; |
| 167 | } |
| 168 | |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 169 | return this->createWrapped(std::move(tex), UseAllocator::kYes); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 170 | } |
| 171 | |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 172 | sk_sp<GrTextureProxy> GrProxyProvider::testingOnly_createInstantiatedProxy( |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 173 | SkISize dimensions, |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 174 | GrColorType colorType, |
| 175 | GrRenderable renderable, |
| 176 | int renderTargetSampleCnt, |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 177 | SkBackingFit fit, |
| 178 | SkBudgeted budgeted, |
| 179 | GrProtected isProtected) { |
Greg Daniel | e20fcad | 2020-01-08 11:52:34 -0500 | [diff] [blame] | 180 | ASSERT_SINGLE_OWNER |
| 181 | if (this->isAbandoned()) { |
| 182 | return nullptr; |
| 183 | } |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 184 | auto format = this->caps()->getDefaultBackendFormat(colorType, renderable); |
Brian Salomon | 9f2b86c | 2019-10-22 10:37:46 -0400 | [diff] [blame] | 185 | return this->testingOnly_createInstantiatedProxy(dimensions, |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 186 | format, |
| 187 | renderable, |
| 188 | renderTargetSampleCnt, |
Brian Salomon | 4eb38b7 | 2019-08-05 12:58:39 -0400 | [diff] [blame] | 189 | fit, |
| 190 | budgeted, |
| 191 | isProtected); |
| 192 | } |
| 193 | |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 194 | sk_sp<GrTextureProxy> GrProxyProvider::testingOnly_createWrapped(sk_sp<GrTexture> tex) { |
| 195 | return this->createWrapped(std::move(tex), UseAllocator::kYes); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 196 | } |
| 197 | #endif |
| 198 | |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 199 | sk_sp<GrTextureProxy> GrProxyProvider::createWrapped(sk_sp<GrTexture> tex, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 200 | UseAllocator useAllocator) { |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 201 | #ifdef SK_DEBUG |
| 202 | if (tex->getUniqueKey().isValid()) { |
Greg Daniel | 3a36511 | 2020-02-14 10:47:18 -0500 | [diff] [blame] | 203 | SkASSERT(!this->findProxyByUniqueKey(tex->getUniqueKey())); |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 204 | } |
| 205 | #endif |
| 206 | |
| 207 | if (tex->asRenderTarget()) { |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 208 | return sk_sp<GrTextureProxy>(new GrTextureRenderTargetProxy(std::move(tex), useAllocator, |
| 209 | this->isDDLProvider())); |
Greg Daniel | 3a36511 | 2020-02-14 10:47:18 -0500 | [diff] [blame] | 210 | } else { |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 211 | return sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(tex), useAllocator, |
| 212 | this->isDDLProvider())); |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 213 | } |
| 214 | } |
| 215 | |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 216 | sk_sp<GrTextureProxy> GrProxyProvider::findOrCreateProxyByUniqueKey(const GrUniqueKey& key, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 217 | UseAllocator useAllocator) { |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 218 | ASSERT_SINGLE_OWNER |
| 219 | |
| 220 | if (this->isAbandoned()) { |
| 221 | return nullptr; |
| 222 | } |
| 223 | |
Greg Daniel | 3a36511 | 2020-02-14 10:47:18 -0500 | [diff] [blame] | 224 | sk_sp<GrTextureProxy> result = this->findProxyByUniqueKey(key); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 225 | if (result) { |
| 226 | return result; |
| 227 | } |
| 228 | |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 229 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 230 | if (!direct) { |
Robert Phillips | d5f9cdd | 2018-01-31 09:29:48 -0500 | [diff] [blame] | 231 | return nullptr; |
| 232 | } |
| 233 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 234 | GrResourceCache* resourceCache = direct->priv().getResourceCache(); |
| 235 | |
| 236 | GrGpuResource* resource = resourceCache->findAndRefUniqueResource(key); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 237 | if (!resource) { |
| 238 | return nullptr; |
| 239 | } |
| 240 | |
| 241 | sk_sp<GrTexture> texture(static_cast<GrSurface*>(resource)->asTexture()); |
| 242 | SkASSERT(texture); |
| 243 | |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 244 | result = this->createWrapped(std::move(texture), useAllocator); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 245 | SkASSERT(result->getUniqueKey() == key); |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 246 | // createWrapped should've added this for us |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 247 | SkASSERT(fUniquelyKeyedProxies.find(key)); |
| 248 | return result; |
| 249 | } |
| 250 | |
Brian Salomon | d005b69 | 2020-04-01 15:47:05 -0400 | [diff] [blame] | 251 | GrSurfaceProxyView GrProxyProvider::findCachedProxyWithColorTypeFallback(const GrUniqueKey& key, |
| 252 | GrSurfaceOrigin origin, |
Brian Salomon | 0029db0 | 2020-04-03 10:41:24 -0400 | [diff] [blame] | 253 | GrColorType ct, |
| 254 | int sampleCnt) { |
Brian Salomon | d005b69 | 2020-04-01 15:47:05 -0400 | [diff] [blame] | 255 | auto proxy = this->findOrCreateProxyByUniqueKey(key); |
| 256 | if (!proxy) { |
| 257 | return {}; |
| 258 | } |
Robert Phillips | 87b57fa | 2021-06-08 13:10:04 -0400 | [diff] [blame] | 259 | const GrCaps* caps = fImageContext->priv().caps(); |
| 260 | |
Brian Salomon | d005b69 | 2020-04-01 15:47:05 -0400 | [diff] [blame] | 261 | // Assume that we used a fallback color type if and only if the proxy is renderable. |
| 262 | if (proxy->asRenderTargetProxy()) { |
| 263 | GrBackendFormat expectedFormat; |
Robert Phillips | 87b57fa | 2021-06-08 13:10:04 -0400 | [diff] [blame] | 264 | std::tie(ct, expectedFormat) = caps->getFallbackColorTypeAndFormat(ct, sampleCnt); |
Brian Salomon | d005b69 | 2020-04-01 15:47:05 -0400 | [diff] [blame] | 265 | SkASSERT(expectedFormat == proxy->backendFormat()); |
| 266 | } |
Robert Phillips | 87b57fa | 2021-06-08 13:10:04 -0400 | [diff] [blame] | 267 | GrSwizzle swizzle = caps->getReadSwizzle(proxy->backendFormat(), ct); |
Brian Salomon | d005b69 | 2020-04-01 15:47:05 -0400 | [diff] [blame] | 268 | return {std::move(proxy), origin, swizzle}; |
| 269 | } |
| 270 | |
Brian Osman | de49665 | 2019-03-22 13:42:33 -0400 | [diff] [blame] | 271 | sk_sp<GrTextureProxy> GrProxyProvider::createProxyFromBitmap(const SkBitmap& bitmap, |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 272 | GrMipmapped mipMapped, |
Brian Salomon | bc074a6 | 2020-03-18 10:06:13 -0400 | [diff] [blame] | 273 | SkBackingFit fit, |
| 274 | SkBudgeted budgeted) { |
Brian Osman | 412674f | 2019-02-07 15:34:58 -0500 | [diff] [blame] | 275 | ASSERT_SINGLE_OWNER |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 276 | SkASSERT(fit == SkBackingFit::kExact || mipMapped == GrMipmapped::kNo); |
Brian Osman | 412674f | 2019-02-07 15:34:58 -0500 | [diff] [blame] | 277 | |
| 278 | if (this->isAbandoned()) { |
| 279 | return nullptr; |
| 280 | } |
| 281 | |
Brian Osman | 2b23c4b | 2018-06-01 12:25:08 -0400 | [diff] [blame] | 282 | if (!SkImageInfoIsValid(bitmap.info())) { |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 283 | return nullptr; |
| 284 | } |
| 285 | |
Brian Osman | de49665 | 2019-03-22 13:42:33 -0400 | [diff] [blame] | 286 | ATRACE_ANDROID_FRAMEWORK("Upload %sTexture [%ux%u]", |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 287 | GrMipmapped::kYes == mipMapped ? "MipMap " : "", |
Brian Osman | de49665 | 2019-03-22 13:42:33 -0400 | [diff] [blame] | 288 | bitmap.width(), bitmap.height()); |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 289 | |
| 290 | // In non-ddl we will always instantiate right away. Thus we never want to copy the SkBitmap |
| 291 | // even if its mutable. In ddl, if the bitmap is mutable then we must make a copy since the |
| 292 | // upload of the data to the gpu can happen at anytime and the bitmap may change by then. |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 293 | SkBitmap copyBitmap = bitmap; |
| 294 | if (!this->renderingDirectly() && !bitmap.isImmutable()) { |
| 295 | copyBitmap.allocPixels(); |
| 296 | if (!bitmap.readPixels(copyBitmap.pixmap())) { |
| 297 | return nullptr; |
| 298 | } |
| 299 | copyBitmap.setImmutable(); |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 300 | } |
| 301 | |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 302 | sk_sp<GrTextureProxy> proxy; |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 303 | if (mipMapped == GrMipmapped::kNo || |
Mike Reed | 13711eb | 2020-07-14 17:16:32 -0400 | [diff] [blame] | 304 | 0 == SkMipmap::ComputeLevelCount(copyBitmap.width(), copyBitmap.height())) { |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 305 | proxy = this->createNonMippedProxyFromBitmap(copyBitmap, fit, budgeted); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 306 | } else { |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 307 | proxy = this->createMippedProxyFromBitmap(copyBitmap, budgeted); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 308 | } |
| 309 | |
| 310 | if (!proxy) { |
| 311 | return nullptr; |
| 312 | } |
| 313 | |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 314 | auto direct = fImageContext->asDirectContext(); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 315 | if (direct) { |
| 316 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| 317 | |
| 318 | // In order to reuse code we always create a lazy proxy. When we aren't in DDL mode however |
| 319 | // we're better off instantiating the proxy immediately here. |
| 320 | if (!proxy->priv().doLazyInstantiation(resourceProvider)) { |
| 321 | return nullptr; |
| 322 | } |
| 323 | } |
| 324 | return proxy; |
| 325 | } |
| 326 | |
| 327 | sk_sp<GrTextureProxy> GrProxyProvider::createNonMippedProxyFromBitmap(const SkBitmap& bitmap, |
| 328 | SkBackingFit fit, |
Brian Salomon | bc074a6 | 2020-03-18 10:06:13 -0400 | [diff] [blame] | 329 | SkBudgeted budgeted) { |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 330 | auto dims = bitmap.dimensions(); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 331 | |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 332 | auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
| 333 | GrBackendFormat format = this->caps()->getDefaultBackendFormat(colorType, GrRenderable::kNo); |
| 334 | if (!format.isValid()) { |
| 335 | return nullptr; |
| 336 | } |
| 337 | |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 338 | sk_sp<GrTextureProxy> proxy = this->createLazyProxy( |
Brian Salomon | 63410e9 | 2020-03-23 18:32:50 -0400 | [diff] [blame] | 339 | [bitmap](GrResourceProvider* resourceProvider, const LazySurfaceDesc& desc) { |
Brian Salomon | 40a4062 | 2020-07-21 10:32:07 -0400 | [diff] [blame] | 340 | SkASSERT(desc.fMipmapped == GrMipmapped::kNo); |
Brian Salomon | 2c67340 | 2021-04-02 14:36:58 -0400 | [diff] [blame] | 341 | GrMipLevel mipLevel = {bitmap.getPixels(), bitmap.rowBytes(), nullptr}; |
Brian Salomon | 63410e9 | 2020-03-23 18:32:50 -0400 | [diff] [blame] | 342 | auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 343 | return LazyCallbackResult(resourceProvider->createTexture( |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 344 | desc.fDimensions, |
| 345 | desc.fFormat, |
Greg Daniel | e56d31f | 2021-08-16 09:29:56 -0400 | [diff] [blame] | 346 | desc.fTextureType, |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 347 | colorType, |
| 348 | desc.fRenderable, |
| 349 | desc.fSampleCnt, |
| 350 | desc.fBudgeted, |
| 351 | desc.fFit, |
| 352 | desc.fProtected, |
| 353 | mipLevel)); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 354 | }, |
Greg Daniel | c113f00 | 2020-08-26 13:28:22 -0400 | [diff] [blame] | 355 | format, dims, GrMipmapped::kNo, GrMipmapStatus::kNotAllocated, |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 356 | GrInternalSurfaceFlags::kNone, fit, budgeted, GrProtected::kNo, UseAllocator::kYes); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 357 | |
| 358 | if (!proxy) { |
| 359 | return nullptr; |
| 360 | } |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 361 | SkASSERT(proxy->dimensions() == bitmap.dimensions()); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 362 | return proxy; |
| 363 | } |
| 364 | |
| 365 | sk_sp<GrTextureProxy> GrProxyProvider::createMippedProxyFromBitmap(const SkBitmap& bitmap, |
Brian Salomon | bc074a6 | 2020-03-18 10:06:13 -0400 | [diff] [blame] | 366 | SkBudgeted budgeted) { |
Brian Salomon | 69100f0 | 2020-07-21 10:49:25 -0400 | [diff] [blame] | 367 | SkASSERT(this->caps()->mipmapSupport()); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 368 | |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 369 | auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
| 370 | GrBackendFormat format = this->caps()->getDefaultBackendFormat(colorType, GrRenderable::kNo); |
| 371 | if (!format.isValid()) { |
| 372 | return nullptr; |
| 373 | } |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 374 | |
Mike Reed | 13711eb | 2020-07-14 17:16:32 -0400 | [diff] [blame] | 375 | sk_sp<SkMipmap> mipmaps(SkMipmap::Build(bitmap.pixmap(), nullptr)); |
Brian Osman | bc6b9cb | 2018-09-13 13:43:25 -0400 | [diff] [blame] | 376 | if (!mipmaps) { |
| 377 | return nullptr; |
| 378 | } |
| 379 | |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 380 | auto dims = bitmap.dimensions(); |
Greg Daniel | ce3ddaa | 2020-01-22 16:58:15 -0500 | [diff] [blame] | 381 | |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 382 | sk_sp<GrTextureProxy> proxy = this->createLazyProxy( |
Brian Salomon | 63410e9 | 2020-03-23 18:32:50 -0400 | [diff] [blame] | 383 | [bitmap, mipmaps](GrResourceProvider* resourceProvider, const LazySurfaceDesc& desc) { |
Brian Osman | 1b97f13 | 2018-09-13 17:33:48 +0000 | [diff] [blame] | 384 | const int mipLevelCount = mipmaps->countLevels() + 1; |
| 385 | std::unique_ptr<GrMipLevel[]> texels(new GrMipLevel[mipLevelCount]); |
Brian Salomon | 63410e9 | 2020-03-23 18:32:50 -0400 | [diff] [blame] | 386 | auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
Brian Osman | 1b97f13 | 2018-09-13 17:33:48 +0000 | [diff] [blame] | 387 | |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 388 | texels[0].fPixels = bitmap.getPixels(); |
| 389 | texels[0].fRowBytes = bitmap.rowBytes(); |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 390 | |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 391 | for (int i = 1; i < mipLevelCount; ++i) { |
Mike Reed | 13711eb | 2020-07-14 17:16:32 -0400 | [diff] [blame] | 392 | SkMipmap::Level generatedMipLevel; |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 393 | mipmaps->getLevel(i - 1, &generatedMipLevel); |
| 394 | texels[i].fPixels = generatedMipLevel.fPixmap.addr(); |
| 395 | texels[i].fRowBytes = generatedMipLevel.fPixmap.rowBytes(); |
| 396 | SkASSERT(texels[i].fPixels); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 397 | SkASSERT(generatedMipLevel.fPixmap.colorType() == bitmap.colorType()); |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 398 | } |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 399 | return LazyCallbackResult(resourceProvider->createTexture( |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 400 | desc.fDimensions, |
| 401 | desc.fFormat, |
Greg Daniel | e56d31f | 2021-08-16 09:29:56 -0400 | [diff] [blame] | 402 | desc.fTextureType, |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 403 | colorType, |
| 404 | GrRenderable::kNo, |
| 405 | 1, |
| 406 | desc.fBudgeted, |
| 407 | GrMipMapped::kYes, |
| 408 | GrProtected::kNo, |
| 409 | texels.get())); |
Brian Salomon | 2a4f983 | 2018-03-03 22:43:43 -0500 | [diff] [blame] | 410 | }, |
Greg Daniel | c113f00 | 2020-08-26 13:28:22 -0400 | [diff] [blame] | 411 | format, dims, GrMipmapped::kYes, GrMipmapStatus::kValid, GrInternalSurfaceFlags::kNone, |
| 412 | SkBackingFit::kExact, budgeted, GrProtected::kNo, UseAllocator::kYes); |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 413 | |
Greg Daniel | 92cbf3f | 2018-04-12 16:50:17 -0400 | [diff] [blame] | 414 | if (!proxy) { |
| 415 | return nullptr; |
| 416 | } |
| 417 | |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 418 | SkASSERT(proxy->dimensions() == bitmap.dimensions()); |
Greg Daniel | 6f5441a | 2020-01-28 17:02:49 -0500 | [diff] [blame] | 419 | |
Greg Daniel | a4ead65 | 2018-02-07 10:21:48 -0500 | [diff] [blame] | 420 | return proxy; |
| 421 | } |
| 422 | |
Greg Daniel | 4065d45 | 2018-11-16 15:43:41 -0500 | [diff] [blame] | 423 | sk_sp<GrTextureProxy> GrProxyProvider::createProxy(const GrBackendFormat& format, |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 424 | SkISize dimensions, |
Brian Salomon | f2c2ba9 | 2019-07-17 09:59:59 -0400 | [diff] [blame] | 425 | GrRenderable renderable, |
Brian Salomon | 27b4d8d | 2019-07-22 14:23:45 -0400 | [diff] [blame] | 426 | int renderTargetSampleCnt, |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 427 | GrMipmapped mipMapped, |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 428 | SkBackingFit fit, |
| 429 | SkBudgeted budgeted, |
Brian Salomon | e8a766b | 2019-07-19 14:24:36 -0400 | [diff] [blame] | 430 | GrProtected isProtected, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 431 | GrInternalSurfaceFlags surfaceFlags, |
| 432 | GrSurfaceProxy::UseAllocator useAllocator) { |
Greg Daniel | e20fcad | 2020-01-08 11:52:34 -0500 | [diff] [blame] | 433 | ASSERT_SINGLE_OWNER |
| 434 | if (this->isAbandoned()) { |
| 435 | return nullptr; |
| 436 | } |
| 437 | |
Robert Phillips | 8ff8bcc | 2019-07-29 17:03:35 -0400 | [diff] [blame] | 438 | const GrCaps* caps = this->caps(); |
| 439 | |
| 440 | if (caps->isFormatCompressed(format)) { |
Brian Salomon | bb8dde8 | 2019-06-27 10:52:13 -0400 | [diff] [blame] | 441 | // Deferred proxies for compressed textures are not supported. |
| 442 | return nullptr; |
| 443 | } |
Robert Phillips | 0902c98 | 2019-07-16 07:47:56 -0400 | [diff] [blame] | 444 | |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 445 | if (GrMipmapped::kYes == mipMapped) { |
Mike Reed | 13711eb | 2020-07-14 17:16:32 -0400 | [diff] [blame] | 446 | // SkMipmap doesn't include the base level in the level count so we have to add 1 |
| 447 | int mipCount = SkMipmap::ComputeLevelCount(dimensions.fWidth, dimensions.fHeight) + 1; |
Greg Daniel | f6f7b67 | 2018-02-15 13:06:26 -0500 | [diff] [blame] | 448 | if (1 == mipCount) { |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 449 | mipMapped = GrMipmapped::kNo; |
Greg Daniel | f6f7b67 | 2018-02-15 13:06:26 -0500 | [diff] [blame] | 450 | } |
| 451 | } |
| 452 | |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 453 | if (!caps->validateSurfaceParams(dimensions, |
| 454 | format, |
| 455 | renderable, |
| 456 | renderTargetSampleCnt, |
| 457 | mipMapped, |
| 458 | GrTextureType::k2D)) { |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 459 | return nullptr; |
| 460 | } |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 461 | GrMipmapStatus mipmapStatus = (GrMipmapped::kYes == mipMapped) |
| 462 | ? GrMipmapStatus::kDirty |
| 463 | : GrMipmapStatus::kNotAllocated; |
Brian Salomon | f2c2ba9 | 2019-07-17 09:59:59 -0400 | [diff] [blame] | 464 | if (renderable == GrRenderable::kYes) { |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 465 | renderTargetSampleCnt = caps->getRenderTargetSampleCount(renderTargetSampleCnt, format); |
Greg Daniel | 6fa62e2 | 2019-08-07 15:52:37 -0400 | [diff] [blame] | 466 | SkASSERT(renderTargetSampleCnt); |
Greg Daniel | 638b2e8 | 2020-08-27 14:29:00 -0400 | [diff] [blame] | 467 | GrInternalSurfaceFlags extraFlags = caps->getExtraSurfaceFlagsForDeferredRT(); |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 468 | // We know anything we instantiate later from this deferred path will be |
| 469 | // both texturable and renderable |
Brian Salomon | 27b4d8d | 2019-07-22 14:23:45 -0400 | [diff] [blame] | 470 | return sk_sp<GrTextureProxy>(new GrTextureRenderTargetProxy( |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 471 | *caps, format, dimensions, renderTargetSampleCnt, mipMapped, mipmapStatus, fit, |
Greg Daniel | 638b2e8 | 2020-08-27 14:29:00 -0400 | [diff] [blame] | 472 | budgeted, isProtected, surfaceFlags | extraFlags, useAllocator, |
| 473 | this->isDDLProvider())); |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 474 | } |
| 475 | |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 476 | return sk_sp<GrTextureProxy>(new GrTextureProxy(format, dimensions, mipMapped, mipmapStatus, |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 477 | fit, budgeted, isProtected, surfaceFlags, |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 478 | useAllocator, this->isDDLProvider())); |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 479 | } |
| 480 | |
Brian Salomon | bb8dde8 | 2019-06-27 10:52:13 -0400 | [diff] [blame] | 481 | sk_sp<GrTextureProxy> GrProxyProvider::createCompressedTextureProxy( |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 482 | SkISize dimensions, SkBudgeted budgeted, GrMipmapped mipMapped, GrProtected isProtected, |
Robert Phillips | e4720c6 | 2020-01-14 14:33:24 -0500 | [diff] [blame] | 483 | SkImage::CompressionType compressionType, sk_sp<SkData> data) { |
Greg Daniel | e20fcad | 2020-01-08 11:52:34 -0500 | [diff] [blame] | 484 | ASSERT_SINGLE_OWNER |
| 485 | if (this->isAbandoned()) { |
| 486 | return nullptr; |
| 487 | } |
Brian Salomon | bb8dde8 | 2019-06-27 10:52:13 -0400 | [diff] [blame] | 488 | |
Robert Phillips | 8ff8bcc | 2019-07-29 17:03:35 -0400 | [diff] [blame] | 489 | GrBackendFormat format = this->caps()->getBackendFormatFromCompressionType(compressionType); |
| 490 | |
Greg Daniel | 0e9d34d | 2021-08-13 16:20:18 -0400 | [diff] [blame] | 491 | if (!this->caps()->isFormatTexturable(format, GrTextureType::k2D)) { |
Jim Van Verth | ee06b33 | 2019-01-18 10:36:32 -0500 | [diff] [blame] | 492 | return nullptr; |
| 493 | } |
| 494 | |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 495 | GrMipmapStatus mipmapStatus = (GrMipmapped::kYes == mipMapped) ? GrMipmapStatus::kValid |
| 496 | : GrMipmapStatus::kNotAllocated; |
Robert Phillips | e4720c6 | 2020-01-14 14:33:24 -0500 | [diff] [blame] | 497 | |
Jim Van Verth | ee06b33 | 2019-01-18 10:36:32 -0500 | [diff] [blame] | 498 | sk_sp<GrTextureProxy> proxy = this->createLazyProxy( |
Brian Salomon | 63410e9 | 2020-03-23 18:32:50 -0400 | [diff] [blame] | 499 | [data](GrResourceProvider* resourceProvider, const LazySurfaceDesc& desc) { |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 500 | return LazyCallbackResult(resourceProvider->createCompressedTexture( |
Brian Salomon | 40a4062 | 2020-07-21 10:32:07 -0400 | [diff] [blame] | 501 | desc.fDimensions, desc.fFormat, desc.fBudgeted, desc.fMipmapped, |
Brian Salomon | 63410e9 | 2020-03-23 18:32:50 -0400 | [diff] [blame] | 502 | desc.fProtected, data.get())); |
Brian Salomon | bb8dde8 | 2019-06-27 10:52:13 -0400 | [diff] [blame] | 503 | }, |
Greg Daniel | c113f00 | 2020-08-26 13:28:22 -0400 | [diff] [blame] | 504 | format, dimensions, mipMapped, mipmapStatus,GrInternalSurfaceFlags::kReadOnly, |
| 505 | SkBackingFit::kExact, SkBudgeted::kYes, GrProtected::kNo, UseAllocator::kYes); |
Jim Van Verth | ee06b33 | 2019-01-18 10:36:32 -0500 | [diff] [blame] | 506 | |
| 507 | if (!proxy) { |
| 508 | return nullptr; |
| 509 | } |
| 510 | |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 511 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 512 | if (direct) { |
| 513 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
Jim Van Verth | ee06b33 | 2019-01-18 10:36:32 -0500 | [diff] [blame] | 514 | // In order to reuse code we always create a lazy proxy. When we aren't in DDL mode however |
| 515 | // we're better off instantiating the proxy immediately here. |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 516 | if (!proxy->priv().doLazyInstantiation(resourceProvider)) { |
Jim Van Verth | ee06b33 | 2019-01-18 10:36:32 -0500 | [diff] [blame] | 517 | return nullptr; |
| 518 | } |
| 519 | } |
| 520 | return proxy; |
| 521 | } |
| 522 | |
Brian Salomon | 7578f3e | 2018-03-07 14:39:54 -0500 | [diff] [blame] | 523 | sk_sp<GrTextureProxy> GrProxyProvider::wrapBackendTexture(const GrBackendTexture& backendTex, |
Brian Salomon | 7578f3e | 2018-03-07 14:39:54 -0500 | [diff] [blame] | 524 | GrWrapOwnership ownership, |
Brian Salomon | aa6ca0a | 2019-01-24 16:03:07 -0500 | [diff] [blame] | 525 | GrWrapCacheable cacheable, |
Brian Salomon | c67c31c | 2018-12-06 10:00:03 -0500 | [diff] [blame] | 526 | GrIOType ioType, |
Robert Phillips | e22c5ca | 2020-06-19 12:29:57 -0400 | [diff] [blame] | 527 | sk_sp<GrRefCntedCallback> releaseHelper) { |
Brian Salomon | c67c31c | 2018-12-06 10:00:03 -0500 | [diff] [blame] | 528 | SkASSERT(ioType != kWrite_GrIOType); |
Robert Phillips | e22c5ca | 2020-06-19 12:29:57 -0400 | [diff] [blame] | 529 | |
Robert Phillips | f9bec20 | 2018-01-16 09:21:01 -0500 | [diff] [blame] | 530 | if (this->isAbandoned()) { |
| 531 | return nullptr; |
| 532 | } |
| 533 | |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 534 | // This is only supported on a direct GrContext. |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 535 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 536 | if (!direct) { |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 537 | return nullptr; |
| 538 | } |
| 539 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 540 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| 541 | |
Brian Salomon | aa6ca0a | 2019-01-24 16:03:07 -0500 | [diff] [blame] | 542 | sk_sp<GrTexture> tex = |
Brian Salomon | 8a78e9c | 2020-03-27 10:42:15 -0400 | [diff] [blame] | 543 | resourceProvider->wrapBackendTexture(backendTex, ownership, cacheable, ioType); |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 544 | if (!tex) { |
| 545 | return nullptr; |
| 546 | } |
Robert Phillips | adbe132 | 2018-01-17 13:35:46 -0500 | [diff] [blame] | 547 | |
Robert Phillips | e22c5ca | 2020-06-19 12:29:57 -0400 | [diff] [blame] | 548 | if (releaseHelper) { |
| 549 | tex->setRelease(std::move(releaseHelper)); |
Greg Daniel | 6a0176b | 2018-01-30 09:28:44 -0500 | [diff] [blame] | 550 | } |
| 551 | |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 552 | SkASSERT(!tex->asRenderTarget()); // Strictly a GrTexture |
| 553 | // Make sure we match how we created the proxy with SkBudgeted::kNo |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 554 | SkASSERT(GrBudgetedType::kBudgeted != tex->resourcePriv().budgetedType()); |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 555 | |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 556 | return sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(tex), UseAllocator::kNo, |
| 557 | this->isDDLProvider())); |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 558 | } |
| 559 | |
Robert Phillips | a112133 | 2020-06-29 13:05:29 -0400 | [diff] [blame] | 560 | sk_sp<GrTextureProxy> GrProxyProvider::wrapCompressedBackendTexture( |
| 561 | const GrBackendTexture& beTex, |
| 562 | GrWrapOwnership ownership, |
| 563 | GrWrapCacheable cacheable, |
| 564 | sk_sp<GrRefCntedCallback> releaseHelper) { |
Robert Phillips | ead321b | 2019-12-19 10:16:32 -0500 | [diff] [blame] | 565 | if (this->isAbandoned()) { |
| 566 | return nullptr; |
| 567 | } |
| 568 | |
| 569 | // This is only supported on a direct GrContext. |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 570 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | ead321b | 2019-12-19 10:16:32 -0500 | [diff] [blame] | 571 | if (!direct) { |
| 572 | return nullptr; |
| 573 | } |
| 574 | |
| 575 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| 576 | |
| 577 | sk_sp<GrTexture> tex = resourceProvider->wrapCompressedBackendTexture(beTex, ownership, |
| 578 | cacheable); |
| 579 | if (!tex) { |
| 580 | return nullptr; |
| 581 | } |
| 582 | |
Robert Phillips | a112133 | 2020-06-29 13:05:29 -0400 | [diff] [blame] | 583 | if (releaseHelper) { |
| 584 | tex->setRelease(std::move(releaseHelper)); |
Robert Phillips | ead321b | 2019-12-19 10:16:32 -0500 | [diff] [blame] | 585 | } |
| 586 | |
| 587 | SkASSERT(!tex->asRenderTarget()); // Strictly a GrTexture |
| 588 | // Make sure we match how we created the proxy with SkBudgeted::kNo |
| 589 | SkASSERT(GrBudgetedType::kBudgeted != tex->resourcePriv().budgetedType()); |
| 590 | |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 591 | return sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(tex), UseAllocator::kNo, |
| 592 | this->isDDLProvider())); |
Robert Phillips | ead321b | 2019-12-19 10:16:32 -0500 | [diff] [blame] | 593 | } |
| 594 | |
Brian Salomon | 7578f3e | 2018-03-07 14:39:54 -0500 | [diff] [blame] | 595 | sk_sp<GrTextureProxy> GrProxyProvider::wrapRenderableBackendTexture( |
Brian Salomon | 8a78e9c | 2020-03-27 10:42:15 -0400 | [diff] [blame] | 596 | const GrBackendTexture& backendTex, |
| 597 | int sampleCnt, |
| 598 | GrWrapOwnership ownership, |
| 599 | GrWrapCacheable cacheable, |
Robert Phillips | a112133 | 2020-06-29 13:05:29 -0400 | [diff] [blame] | 600 | sk_sp<GrRefCntedCallback> releaseHelper) { |
Robert Phillips | f9bec20 | 2018-01-16 09:21:01 -0500 | [diff] [blame] | 601 | if (this->isAbandoned()) { |
| 602 | return nullptr; |
| 603 | } |
| 604 | |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 605 | // This is only supported on a direct GrContext. |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 606 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 607 | if (!direct) { |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 608 | return nullptr; |
| 609 | } |
| 610 | |
Robert Phillips | 0902c98 | 2019-07-16 07:47:56 -0400 | [diff] [blame] | 611 | const GrCaps* caps = this->caps(); |
| 612 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 613 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| 614 | |
Greg Daniel | 6fa62e2 | 2019-08-07 15:52:37 -0400 | [diff] [blame] | 615 | sampleCnt = caps->getRenderTargetSampleCount(sampleCnt, backendTex.getBackendFormat()); |
| 616 | SkASSERT(sampleCnt); |
| 617 | |
Brian Salomon | 8a78e9c | 2020-03-27 10:42:15 -0400 | [diff] [blame] | 618 | sk_sp<GrTexture> tex = resourceProvider->wrapRenderableBackendTexture( |
| 619 | backendTex, sampleCnt, ownership, cacheable); |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 620 | if (!tex) { |
| 621 | return nullptr; |
Greg Daniel | 2a30390 | 2018-02-20 10:25:54 -0500 | [diff] [blame] | 622 | } |
| 623 | |
Robert Phillips | a112133 | 2020-06-29 13:05:29 -0400 | [diff] [blame] | 624 | if (releaseHelper) { |
| 625 | tex->setRelease(std::move(releaseHelper)); |
Greg Daniel | 8ce7991 | 2019-02-05 10:08:43 -0500 | [diff] [blame] | 626 | } |
| 627 | |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 628 | SkASSERT(tex->asRenderTarget()); // A GrTextureRenderTarget |
| 629 | // Make sure we match how we created the proxy with SkBudgeted::kNo |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 630 | SkASSERT(GrBudgetedType::kBudgeted != tex->resourcePriv().budgetedType()); |
Greg Daniel | 6abda43 | 2018-02-15 14:55:00 -0500 | [diff] [blame] | 631 | |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 632 | return sk_sp<GrTextureProxy>(new GrTextureRenderTargetProxy(std::move(tex), UseAllocator::kNo, |
| 633 | this->isDDLProvider())); |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 634 | } |
| 635 | |
Brian Salomon | 7578f3e | 2018-03-07 14:39:54 -0500 | [diff] [blame] | 636 | sk_sp<GrSurfaceProxy> GrProxyProvider::wrapBackendRenderTarget( |
Brian Salomon | 8a78e9c | 2020-03-27 10:42:15 -0400 | [diff] [blame] | 637 | const GrBackendRenderTarget& backendRT, |
Robert Phillips | a112133 | 2020-06-29 13:05:29 -0400 | [diff] [blame] | 638 | sk_sp<GrRefCntedCallback> releaseHelper) { |
Robert Phillips | f9bec20 | 2018-01-16 09:21:01 -0500 | [diff] [blame] | 639 | if (this->isAbandoned()) { |
| 640 | return nullptr; |
| 641 | } |
| 642 | |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 643 | // This is only supported on a direct GrContext. |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 644 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 645 | if (!direct) { |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 646 | return nullptr; |
Greg Daniel | 2a30390 | 2018-02-20 10:25:54 -0500 | [diff] [blame] | 647 | } |
| 648 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 649 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| 650 | |
Brian Salomon | 8a78e9c | 2020-03-27 10:42:15 -0400 | [diff] [blame] | 651 | sk_sp<GrRenderTarget> rt = resourceProvider->wrapBackendRenderTarget(backendRT); |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 652 | if (!rt) { |
| 653 | return nullptr; |
Greg Daniel | 2a30390 | 2018-02-20 10:25:54 -0500 | [diff] [blame] | 654 | } |
Greg Daniel | 8ce7991 | 2019-02-05 10:08:43 -0500 | [diff] [blame] | 655 | |
Robert Phillips | a112133 | 2020-06-29 13:05:29 -0400 | [diff] [blame] | 656 | if (releaseHelper) { |
| 657 | rt->setRelease(std::move(releaseHelper)); |
Greg Daniel | 8ce7991 | 2019-02-05 10:08:43 -0500 | [diff] [blame] | 658 | } |
| 659 | |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 660 | SkASSERT(!rt->asTexture()); // A GrRenderTarget that's not textureable |
| 661 | SkASSERT(!rt->getUniqueKey().isValid()); |
| 662 | // Make sure we match how we created the proxy with SkBudgeted::kNo |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 663 | SkASSERT(GrBudgetedType::kBudgeted != rt->resourcePriv().budgetedType()); |
Brian Salomon | f777897 | 2018-03-08 10:13:17 -0500 | [diff] [blame] | 664 | |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 665 | return sk_sp<GrRenderTargetProxy>(new GrRenderTargetProxy(std::move(rt), UseAllocator::kNo)); |
Robert Phillips | 0bd24dc | 2018-01-16 08:06:32 -0500 | [diff] [blame] | 666 | } |
| 667 | |
Greg Daniel | 8426165 | 2021-09-19 17:53:40 -0400 | [diff] [blame] | 668 | #ifdef SK_VULKAN |
Greg Daniel | b46add8 | 2019-01-02 14:51:29 -0500 | [diff] [blame] | 669 | sk_sp<GrRenderTargetProxy> GrProxyProvider::wrapVulkanSecondaryCBAsRenderTarget( |
| 670 | const SkImageInfo& imageInfo, const GrVkDrawableInfo& vkInfo) { |
| 671 | if (this->isAbandoned()) { |
| 672 | return nullptr; |
| 673 | } |
| 674 | |
| 675 | // This is only supported on a direct GrContext. |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 676 | auto direct = fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 677 | if (!direct) { |
Greg Daniel | b46add8 | 2019-01-02 14:51:29 -0500 | [diff] [blame] | 678 | return nullptr; |
| 679 | } |
| 680 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 681 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
Greg Daniel | b46add8 | 2019-01-02 14:51:29 -0500 | [diff] [blame] | 682 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 683 | sk_sp<GrRenderTarget> rt = resourceProvider->wrapVulkanSecondaryCBAsRenderTarget(imageInfo, |
| 684 | vkInfo); |
Greg Daniel | b46add8 | 2019-01-02 14:51:29 -0500 | [diff] [blame] | 685 | if (!rt) { |
| 686 | return nullptr; |
| 687 | } |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 688 | |
Greg Daniel | b46add8 | 2019-01-02 14:51:29 -0500 | [diff] [blame] | 689 | SkASSERT(!rt->asTexture()); // A GrRenderTarget that's not textureable |
| 690 | SkASSERT(!rt->getUniqueKey().isValid()); |
| 691 | // This proxy should be unbudgeted because we're just wrapping an external resource |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 692 | SkASSERT(GrBudgetedType::kBudgeted != rt->resourcePriv().budgetedType()); |
Greg Daniel | b46add8 | 2019-01-02 14:51:29 -0500 | [diff] [blame] | 693 | |
Brian Salomon | 1c53a9f | 2019-08-12 14:10:12 -0400 | [diff] [blame] | 694 | GrColorType colorType = SkColorTypeToGrColorType(imageInfo.colorType()); |
Greg Daniel | 2c19e7f | 2019-06-18 13:29:21 -0400 | [diff] [blame] | 695 | |
Greg Daniel | de4bbdb | 2021-04-13 14:23:23 -0400 | [diff] [blame] | 696 | if (!this->caps()->isFormatAsColorTypeRenderable( |
| 697 | colorType, GrBackendFormat::MakeVk(vkInfo.fFormat), /*sampleCount=*/1)) { |
Brian Salomon | 1c53a9f | 2019-08-12 14:10:12 -0400 | [diff] [blame] | 698 | return nullptr; |
| 699 | } |
| 700 | |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 701 | return sk_sp<GrRenderTargetProxy>(new GrRenderTargetProxy( |
| 702 | std::move(rt), UseAllocator::kNo, GrRenderTargetProxy::WrapsVkSecondaryCB::kYes)); |
Brian Salomon | e8a766b | 2019-07-19 14:24:36 -0400 | [diff] [blame] | 703 | } |
Greg Daniel | 8426165 | 2021-09-19 17:53:40 -0400 | [diff] [blame] | 704 | #else |
| 705 | sk_sp<GrRenderTargetProxy> GrProxyProvider::wrapVulkanSecondaryCBAsRenderTarget( |
| 706 | const SkImageInfo&, const GrVkDrawableInfo&) { |
| 707 | return nullptr; |
| 708 | } |
| 709 | #endif |
Brian Salomon | e8a766b | 2019-07-19 14:24:36 -0400 | [diff] [blame] | 710 | |
Adlai Holler | dced31f | 2021-02-19 12:33:46 -0500 | [diff] [blame] | 711 | sk_sp<GrTextureProxy> GrProxyProvider::CreatePromiseProxy(GrContextThreadSafeProxy* threadSafeProxy, |
| 712 | LazyInstantiateCallback&& callback, |
| 713 | const GrBackendFormat& format, |
| 714 | SkISize dimensions, |
| 715 | GrMipmapped mipMapped) { |
| 716 | if (threadSafeProxy->priv().abandoned()) { |
| 717 | return nullptr; |
| 718 | } |
| 719 | SkASSERT((dimensions.fWidth <= 0 && dimensions.fHeight <= 0) || |
| 720 | (dimensions.fWidth > 0 && dimensions.fHeight > 0)); |
| 721 | |
| 722 | if (dimensions.fWidth > threadSafeProxy->priv().caps()->maxTextureSize() || |
| 723 | dimensions.fHeight > threadSafeProxy->priv().caps()->maxTextureSize()) { |
| 724 | return nullptr; |
| 725 | } |
| 726 | // Ganesh assumes that, when wrapping a mipmapped backend texture from a client, that its |
| 727 | // mipmaps are fully fleshed out. |
| 728 | GrMipmapStatus mipmapStatus = (GrMipmapped::kYes == mipMapped) ? GrMipmapStatus::kValid |
| 729 | : GrMipmapStatus::kNotAllocated; |
| 730 | |
| 731 | // We pass kReadOnly here since we should treat content of the client's texture as immutable. |
| 732 | // The promise API provides no way for the client to indicate that the texture is protected. |
Adlai Holler | 55aaefe | 2021-03-03 16:12:56 -0700 | [diff] [blame] | 733 | auto proxy = sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(callback), |
| 734 | format, |
| 735 | dimensions, |
| 736 | mipMapped, |
| 737 | mipmapStatus, |
| 738 | SkBackingFit::kExact, |
| 739 | SkBudgeted::kNo, |
| 740 | GrProtected::kNo, |
| 741 | GrInternalSurfaceFlags::kReadOnly, |
| 742 | GrSurfaceProxy::UseAllocator::kYes, |
| 743 | GrDDLProvider::kYes)); |
| 744 | proxy->priv().setIsPromiseProxy(); |
| 745 | return proxy; |
Adlai Holler | dced31f | 2021-02-19 12:33:46 -0500 | [diff] [blame] | 746 | } |
| 747 | |
Brian Salomon | e8a766b | 2019-07-19 14:24:36 -0400 | [diff] [blame] | 748 | sk_sp<GrTextureProxy> GrProxyProvider::createLazyProxy(LazyInstantiateCallback&& callback, |
| 749 | const GrBackendFormat& format, |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 750 | SkISize dimensions, |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 751 | GrMipmapped mipMapped, |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 752 | GrMipmapStatus mipmapStatus, |
Brian Salomon | e8a766b | 2019-07-19 14:24:36 -0400 | [diff] [blame] | 753 | GrInternalSurfaceFlags surfaceFlags, |
| 754 | SkBackingFit fit, |
| 755 | SkBudgeted budgeted, |
| 756 | GrProtected isProtected, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 757 | GrSurfaceProxy::UseAllocator useAllocator) { |
Greg Daniel | e20fcad | 2020-01-08 11:52:34 -0500 | [diff] [blame] | 758 | ASSERT_SINGLE_OWNER |
| 759 | if (this->isAbandoned()) { |
| 760 | return nullptr; |
| 761 | } |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 762 | SkASSERT((dimensions.fWidth <= 0 && dimensions.fHeight <= 0) || |
| 763 | (dimensions.fWidth > 0 && dimensions.fHeight > 0)); |
Greg Daniel | 92cbf3f | 2018-04-12 16:50:17 -0400 | [diff] [blame] | 764 | |
Brian Salomon | bd3792d | 2020-11-10 14:17:58 -0500 | [diff] [blame] | 765 | if (!format.isValid() || format.backend() != fImageContext->backend()) { |
Robert Phillips | 0a15cc6 | 2019-07-30 12:49:10 -0400 | [diff] [blame] | 766 | return nullptr; |
| 767 | } |
| 768 | |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 769 | if (dimensions.fWidth > this->caps()->maxTextureSize() || |
| 770 | dimensions.fHeight > this->caps()->maxTextureSize()) { |
Greg Daniel | 92cbf3f | 2018-04-12 16:50:17 -0400 | [diff] [blame] | 771 | return nullptr; |
| 772 | } |
| 773 | |
Greg Daniel | c113f00 | 2020-08-26 13:28:22 -0400 | [diff] [blame] | 774 | return sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(callback), |
| 775 | format, |
| 776 | dimensions, |
| 777 | mipMapped, |
| 778 | mipmapStatus, |
| 779 | fit, |
| 780 | budgeted, |
| 781 | isProtected, |
| 782 | surfaceFlags, |
| 783 | useAllocator, |
| 784 | this->isDDLProvider())); |
Robert Phillips | 777707b | 2018-01-17 11:40:14 -0500 | [diff] [blame] | 785 | } |
| 786 | |
Robert Phillips | e8fabb2 | 2018-02-04 14:33:21 -0500 | [diff] [blame] | 787 | sk_sp<GrRenderTargetProxy> GrProxyProvider::createLazyRenderTargetProxy( |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 788 | LazyInstantiateCallback&& callback, |
| 789 | const GrBackendFormat& format, |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 790 | SkISize dimensions, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 791 | int sampleCnt, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 792 | GrInternalSurfaceFlags surfaceFlags, |
| 793 | const TextureInfo* textureInfo, |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 794 | GrMipmapStatus mipmapStatus, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 795 | SkBackingFit fit, |
| 796 | SkBudgeted budgeted, |
| 797 | GrProtected isProtected, |
| 798 | bool wrapsVkSecondaryCB, |
| 799 | UseAllocator useAllocator) { |
Greg Daniel | e20fcad | 2020-01-08 11:52:34 -0500 | [diff] [blame] | 800 | ASSERT_SINGLE_OWNER |
| 801 | if (this->isAbandoned()) { |
| 802 | return nullptr; |
| 803 | } |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 804 | SkASSERT((dimensions.fWidth <= 0 && dimensions.fHeight <= 0) || |
| 805 | (dimensions.fWidth > 0 && dimensions.fHeight > 0)); |
Greg Daniel | 92cbf3f | 2018-04-12 16:50:17 -0400 | [diff] [blame] | 806 | |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 807 | if (dimensions.fWidth > this->caps()->maxRenderTargetSize() || |
| 808 | dimensions.fHeight > this->caps()->maxRenderTargetSize()) { |
Greg Daniel | 92cbf3f | 2018-04-12 16:50:17 -0400 | [diff] [blame] | 809 | return nullptr; |
| 810 | } |
| 811 | |
Brian Salomon | 7226c23 | 2018-07-30 13:13:17 -0400 | [diff] [blame] | 812 | if (textureInfo) { |
Greg Daniel | b085fa9 | 2019-03-05 16:55:12 -0500 | [diff] [blame] | 813 | // Wrapped vulkan secondary command buffers don't support texturing since we won't have an |
| 814 | // actual VkImage to texture from. |
| 815 | SkASSERT(!wrapsVkSecondaryCB); |
Brian Salomon | 7226c23 | 2018-07-30 13:13:17 -0400 | [diff] [blame] | 816 | return sk_sp<GrRenderTargetProxy>(new GrTextureRenderTargetProxy( |
Greg Daniel | 3a36511 | 2020-02-14 10:47:18 -0500 | [diff] [blame] | 817 | *this->caps(), std::move(callback), format, dimensions, sampleCnt, |
Brian Salomon | 40a4062 | 2020-07-21 10:32:07 -0400 | [diff] [blame] | 818 | textureInfo->fMipmapped, mipmapStatus, fit, budgeted, isProtected, surfaceFlags, |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 819 | useAllocator, this->isDDLProvider())); |
Robert Phillips | e8fabb2 | 2018-02-04 14:33:21 -0500 | [diff] [blame] | 820 | } |
| 821 | |
Greg Daniel | b085fa9 | 2019-03-05 16:55:12 -0500 | [diff] [blame] | 822 | GrRenderTargetProxy::WrapsVkSecondaryCB vkSCB = |
| 823 | wrapsVkSecondaryCB ? GrRenderTargetProxy::WrapsVkSecondaryCB::kYes |
| 824 | : GrRenderTargetProxy::WrapsVkSecondaryCB::kNo; |
| 825 | |
Greg Daniel | 3a36511 | 2020-02-14 10:47:18 -0500 | [diff] [blame] | 826 | return sk_sp<GrRenderTargetProxy>( |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 827 | new GrRenderTargetProxy(std::move(callback), format, dimensions, sampleCnt, fit, |
| 828 | budgeted, isProtected, surfaceFlags, useAllocator, vkSCB)); |
Robert Phillips | e8fabb2 | 2018-02-04 14:33:21 -0500 | [diff] [blame] | 829 | } |
| 830 | |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 831 | sk_sp<GrTextureProxy> GrProxyProvider::MakeFullyLazyProxy(LazyInstantiateCallback&& callback, |
| 832 | const GrBackendFormat& format, |
| 833 | GrRenderable renderable, |
| 834 | int renderTargetSampleCnt, |
| 835 | GrProtected isProtected, |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 836 | const GrCaps& caps, |
| 837 | UseAllocator useAllocator) { |
Robert Phillips | 0a15cc6 | 2019-07-30 12:49:10 -0400 | [diff] [blame] | 838 | if (!format.isValid()) { |
| 839 | return nullptr; |
| 840 | } |
| 841 | |
Brian Salomon | 27b4d8d | 2019-07-22 14:23:45 -0400 | [diff] [blame] | 842 | SkASSERT(renderTargetSampleCnt == 1 || renderable == GrRenderable::kYes); |
Greg Daniel | 638b2e8 | 2020-08-27 14:29:00 -0400 | [diff] [blame] | 843 | // TODO: If we ever have callers requesting specific surface flags then we shouldn't use the |
| 844 | // extra deferred flags here. Instead those callers should all pass in exactly what they want. |
| 845 | // However, as of today all uses of this essentially create a deferred proxy in the end. |
| 846 | GrInternalSurfaceFlags surfaceFlags = caps.getExtraSurfaceFlagsForDeferredRT(); |
Robert Phillips | 777707b | 2018-01-17 11:40:14 -0500 | [diff] [blame] | 847 | |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 848 | // MakeFullyLazyProxy is only called at flush time so we know these texture proxies are |
| 849 | // not being created by a DDL provider. |
Brian Salomon | a56a746 | 2020-02-07 14:17:25 -0500 | [diff] [blame] | 850 | static constexpr SkISize kLazyDims = {-1, -1}; |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 851 | if (GrRenderable::kYes == renderable) { |
| 852 | return sk_sp<GrTextureProxy>(new GrTextureRenderTargetProxy( |
Greg Daniel | 3a36511 | 2020-02-14 10:47:18 -0500 | [diff] [blame] | 853 | caps, std::move(callback), format, kLazyDims, renderTargetSampleCnt, |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 854 | GrMipmapped::kNo, GrMipmapStatus::kNotAllocated, SkBackingFit::kApprox, |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 855 | SkBudgeted::kYes, isProtected, surfaceFlags, useAllocator, GrDDLProvider::kNo)); |
Brian Salomon | df1bd6d | 2020-03-26 20:37:01 -0400 | [diff] [blame] | 856 | } else { |
| 857 | return sk_sp<GrTextureProxy>( |
Brian Salomon | 7e67dca | 2020-07-21 09:27:25 -0400 | [diff] [blame] | 858 | new GrTextureProxy(std::move(callback), format, kLazyDims, GrMipmapped::kNo, |
Brian Salomon | a6db510 | 2020-07-21 09:56:23 -0400 | [diff] [blame] | 859 | GrMipmapStatus::kNotAllocated, SkBackingFit::kApprox, |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 860 | SkBudgeted::kYes, isProtected, surfaceFlags, useAllocator, |
| 861 | GrDDLProvider::kNo)); |
Brian Salomon | beb7f52 | 2019-08-30 16:19:42 -0400 | [diff] [blame] | 862 | } |
Robert Phillips | 777707b | 2018-01-17 11:40:14 -0500 | [diff] [blame] | 863 | } |
| 864 | |
Robert Phillips | 427966a | 2018-12-20 17:20:43 -0500 | [diff] [blame] | 865 | void GrProxyProvider::processInvalidUniqueKey(const GrUniqueKey& key, GrTextureProxy* proxy, |
| 866 | InvalidateGPUResource invalidateGPUResource) { |
Mike Klein | 04ef810 | 2020-03-16 12:44:23 -0500 | [diff] [blame] | 867 | this->processInvalidUniqueKeyImpl(key, proxy, invalidateGPUResource, RemoveTableEntry::kYes); |
| 868 | } |
| 869 | |
| 870 | void GrProxyProvider::processInvalidUniqueKeyImpl(const GrUniqueKey& key, GrTextureProxy* proxy, |
| 871 | InvalidateGPUResource invalidateGPUResource, |
| 872 | RemoveTableEntry removeTableEntry) { |
Chris Dalton | 2de13dd | 2019-01-03 15:11:59 -0700 | [diff] [blame] | 873 | SkASSERT(key.isValid()); |
| 874 | |
Robert Phillips | 427966a | 2018-12-20 17:20:43 -0500 | [diff] [blame] | 875 | if (!proxy) { |
| 876 | proxy = fUniquelyKeyedProxies.find(key); |
| 877 | } |
Chris Dalton | 2de13dd | 2019-01-03 15:11:59 -0700 | [diff] [blame] | 878 | SkASSERT(!proxy || proxy->getUniqueKey() == key); |
| 879 | |
| 880 | // Locate the corresponding GrGpuResource (if it needs to be invalidated) before clearing the |
| 881 | // proxy's unique key. We must do it in this order because 'key' may alias the proxy's key. |
| 882 | sk_sp<GrGpuResource> invalidGpuResource; |
| 883 | if (InvalidateGPUResource::kYes == invalidateGPUResource) { |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 884 | auto direct = fImageContext->asDirectContext(); |
Brian Salomon | 01ceae9 | 2019-04-02 11:49:54 -0400 | [diff] [blame] | 885 | if (direct) { |
| 886 | GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| 887 | invalidGpuResource = resourceProvider->findByUniqueKey<GrGpuResource>(key); |
Chris Dalton | 2de13dd | 2019-01-03 15:11:59 -0700 | [diff] [blame] | 888 | } |
| 889 | SkASSERT(!invalidGpuResource || invalidGpuResource->getUniqueKey() == key); |
| 890 | } |
Robert Phillips | 427966a | 2018-12-20 17:20:43 -0500 | [diff] [blame] | 891 | |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 892 | // Note: this method is called for the whole variety of GrGpuResources so often 'key' |
| 893 | // will not be in 'fUniquelyKeyedProxies'. |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 894 | if (proxy) { |
Mike Klein | 04ef810 | 2020-03-16 12:44:23 -0500 | [diff] [blame] | 895 | if (removeTableEntry == RemoveTableEntry::kYes) { |
| 896 | fUniquelyKeyedProxies.remove(key); |
| 897 | } |
Robert Phillips | 427966a | 2018-12-20 17:20:43 -0500 | [diff] [blame] | 898 | proxy->cacheAccess().clearUniqueKey(); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 899 | } |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 900 | |
Chris Dalton | 2de13dd | 2019-01-03 15:11:59 -0700 | [diff] [blame] | 901 | if (invalidGpuResource) { |
| 902 | invalidGpuResource->resourcePriv().removeUniqueKey(); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 903 | } |
| 904 | } |
| 905 | |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 906 | GrDDLProvider GrProxyProvider::isDDLProvider() const { |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 907 | return fImageContext->asDirectContext() ? GrDDLProvider::kNo : GrDDLProvider::kYes; |
Robert Phillips | f10b2a5 | 2020-05-15 10:20:04 -0400 | [diff] [blame] | 908 | } |
| 909 | |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 910 | uint32_t GrProxyProvider::contextID() const { |
| 911 | return fImageContext->priv().contextID(); |
| 912 | } |
| 913 | |
| 914 | const GrCaps* GrProxyProvider::caps() const { |
| 915 | return fImageContext->priv().caps(); |
| 916 | } |
| 917 | |
| 918 | sk_sp<const GrCaps> GrProxyProvider::refCaps() const { |
| 919 | return fImageContext->priv().refCaps(); |
| 920 | } |
| 921 | |
Robert Phillips | a9162df | 2019-02-11 14:12:03 -0500 | [diff] [blame] | 922 | bool GrProxyProvider::isAbandoned() const { |
| 923 | return fImageContext->priv().abandoned(); |
| 924 | } |
| 925 | |
Robert Phillips | 0790f8a | 2018-09-18 13:11:03 -0400 | [diff] [blame] | 926 | void GrProxyProvider::orphanAllUniqueKeys() { |
Mike Klein | cff6396 | 2020-03-14 16:22:45 -0500 | [diff] [blame] | 927 | fUniquelyKeyedProxies.foreach([&](GrTextureProxy* proxy){ |
| 928 | proxy->fProxyProvider = nullptr; |
| 929 | }); |
Robert Phillips | 0790f8a | 2018-09-18 13:11:03 -0400 | [diff] [blame] | 930 | } |
| 931 | |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 932 | void GrProxyProvider::removeAllUniqueKeys() { |
Mike Klein | 04ef810 | 2020-03-16 12:44:23 -0500 | [diff] [blame] | 933 | fUniquelyKeyedProxies.foreach([&](GrTextureProxy* proxy){ |
| 934 | // It's not safe to remove table entries while iterating with foreach(), |
| 935 | // but since we're going to remove them all anyway, simply save that for the end. |
| 936 | this->processInvalidUniqueKeyImpl(proxy->getUniqueKey(), proxy, |
| 937 | InvalidateGPUResource::kNo, |
| 938 | RemoveTableEntry::kNo); |
| 939 | }); |
| 940 | // Removing all those table entries is safe now. |
| 941 | fUniquelyKeyedProxies.reset(); |
Robert Phillips | 1afd4cd | 2018-01-08 13:40:32 -0500 | [diff] [blame] | 942 | } |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 943 | |
| 944 | bool GrProxyProvider::renderingDirectly() const { |
Robert Phillips | f8f45d9 | 2020-07-01 11:11:18 -0400 | [diff] [blame] | 945 | return fImageContext->asDirectContext(); |
Robert Phillips | a41c685 | 2019-02-07 10:44:10 -0500 | [diff] [blame] | 946 | } |