blob: 6502979f9c4a818cb27e0af7af774e18af2ea89d [file] [log] [blame]
Jim Van Verth8026ccc2018-10-04 13:10:39 -04001/*
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 Kleinc0bd9f92019-04-23 12:05:21 -05008#include "include/core/SkPromiseImageTexture.h"
9#include "include/gpu/GrBackendSurface.h"
10#include "include/gpu/GrContext.h"
11#include "include/gpu/GrTexture.h"
12#include "include/private/GrRecordingContext.h"
13#include "src/core/SkBitmapCache.h"
14#include "src/core/SkTLList.h"
15#include "src/gpu/GrClip.h"
16#include "src/gpu/GrContextPriv.h"
Brian Salomonf2ebdd92019-09-30 12:15:30 -040017#include "src/gpu/GrImageInfo.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050018#include "src/gpu/GrRecordingContextPriv.h"
19#include "src/gpu/GrRenderTargetContext.h"
20#include "src/gpu/GrTextureAdjuster.h"
21#include "src/gpu/effects/GrYUVtoRGBEffect.h"
22#include "src/image/SkImage_Gpu.h"
23#include "src/image/SkImage_GpuBase.h"
24#include "src/image/SkReadPixelsRec.h"
Jim Van Verth8026ccc2018-10-04 13:10:39 -040025
Brian Salomon9f2b86c2019-10-22 10:37:46 -040026SkImage_GpuBase::SkImage_GpuBase(sk_sp<GrContext> context, SkISize size, uint32_t uniqueID,
Brian Salomon5ad6fd32019-03-21 15:30:08 -040027 SkColorType ct, SkAlphaType at, sk_sp<SkColorSpace> cs)
Brian Salomon9f2b86c2019-10-22 10:37:46 -040028 : INHERITED(SkImageInfo::Make(size, ct, at, std::move(cs)), uniqueID)
Brian Salomon5ad6fd32019-03-21 15:30:08 -040029 , fContext(std::move(context)) {}
Jim Van Verth8026ccc2018-10-04 13:10:39 -040030
Jim Van Verth8026ccc2018-10-04 13:10:39 -040031//////////////////////////////////////////////////////////////////////////////////////////////////
32
Robert Phillipsfd0d9702019-02-01 10:19:42 -050033#if GR_TEST_UTILS
34void SkImage_GpuBase::resetContext(sk_sp<GrContext> newContext) {
Robert Phillipsfe0963c2019-02-07 13:25:07 -050035 SkASSERT(fContext->priv().matches(newContext.get()));
Robert Phillipsfd0d9702019-02-01 10:19:42 -050036 fContext = newContext;
37}
38#endif
39
Robert Phillips62221e72019-07-24 15:07:38 -040040bool SkImage_GpuBase::ValidateBackendTexture(const GrCaps* caps, const GrBackendTexture& tex,
41 GrColorType grCT, SkColorType ct, SkAlphaType at,
Jim Van Verth8026ccc2018-10-04 13:10:39 -040042 sk_sp<SkColorSpace> cs) {
43 if (!tex.isValid()) {
44 return false;
45 }
46 // TODO: Create a SkImageColorInfo struct for color, alpha, and color space so we don't need to
47 // create a fake image info here.
48 SkImageInfo info = SkImageInfo::Make(1, 1, ct, at, cs);
49 if (!SkImageInfoIsValid(info)) {
50 return false;
51 }
Brian Salomonf391d0f2018-12-14 09:18:50 -050052 GrBackendFormat backendFormat = tex.getBackendFormat();
53 if (!backendFormat.isValid()) {
54 return false;
55 }
Greg Daniele877dce2019-07-11 10:52:43 -040056
Robert Phillips62221e72019-07-24 15:07:38 -040057 return caps->areColorTypeAndFormatCompatible(grCT, backendFormat);
Jim Van Verth8026ccc2018-10-04 13:10:39 -040058}
59
Robert Phillipsead321b2019-12-19 10:16:32 -050060bool SkImage_GpuBase::ValidateCompressedBackendTexture(const GrCaps* caps,
61 const GrBackendTexture& tex,
62 SkAlphaType at) {
63
64 if (!tex.isValid() || tex.width() <= 0 || tex.height() <= 0) {
65 return false;
66 }
67
68 if (tex.width() > caps->maxTextureSize() || tex.height() > caps->maxTextureSize()) {
69 return false;
70 }
71
72 if (at == kUnknown_SkAlphaType) {
73 return false;
74 }
75
76 GrBackendFormat backendFormat = tex.getBackendFormat();
77 if (!backendFormat.isValid()) {
78 return false;
79 }
80
81 if (!caps->isFormatCompressed(backendFormat)) {
82 return false;
83 }
84
85 return true;
86}
87
Jim Van Verth8026ccc2018-10-04 13:10:39 -040088//////////////////////////////////////////////////////////////////////////////////////////////////
89
Brian Osmane50cdf02018-10-19 13:02:14 -040090bool SkImage_GpuBase::getROPixels(SkBitmap* dst, CachingHint chint) const {
Robert Phillips920d4882019-03-04 15:16:44 -050091 auto direct = fContext->priv().asDirectContext();
92 if (!direct) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -040093 // DDL TODO: buffer up the readback so it occurs when the DDL is drawn?
94 return false;
95 }
96
Jim Van Verth8026ccc2018-10-04 13:10:39 -040097 const auto desc = SkBitmapCacheDesc::Make(this);
98 if (SkBitmapCache::Find(desc, dst)) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -040099 SkASSERT(dst->isImmutable());
100 SkASSERT(dst->getPixels());
101 return true;
102 }
103
104 SkBitmapCache::RecPtr rec = nullptr;
105 SkPixmap pmap;
106 if (kAllow_CachingHint == chint) {
Brian Salomon5ad6fd32019-03-21 15:30:08 -0400107 rec = SkBitmapCache::Alloc(desc, this->imageInfo(), &pmap);
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400108 if (!rec) {
109 return false;
110 }
111 } else {
Brian Salomon5ad6fd32019-03-21 15:30:08 -0400112 if (!dst->tryAllocPixels(this->imageInfo()) || !dst->peekPixels(&pmap)) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400113 return false;
114 }
115 }
116
Greg Daniel37c127f2020-02-05 10:37:27 -0500117 const GrSurfaceProxyView* view = this->view(direct);
118 SkASSERT(view);
Greg Danielba88ab62019-07-26 09:14:01 -0400119 GrColorType grColorType = SkColorTypeAndFormatToGrColorType(fContext->priv().caps(),
120 this->colorType(),
Greg Daniel37c127f2020-02-05 10:37:27 -0500121 view->proxy()->backendFormat());
Greg Danielba88ab62019-07-26 09:14:01 -0400122
Greg Daniel37c127f2020-02-05 10:37:27 -0500123 auto sContext = GrSurfaceContext::Make(direct, *view, grColorType, this->alphaType(),
124 this->refColorSpace());
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400125 if (!sContext) {
126 return false;
127 }
128
Brian Salomon1d435302019-07-01 13:05:28 -0400129 if (!sContext->readPixels(pmap.info(), pmap.writable_addr(), pmap.rowBytes(), {0, 0})) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400130 return false;
131 }
132
133 if (rec) {
134 SkBitmapCache::Add(std::move(rec), dst);
135 this->notifyAddedToRasterCache();
136 }
137 return true;
138}
139
Robert Phillips6603a172019-03-05 12:35:44 -0500140sk_sp<SkImage> SkImage_GpuBase::onMakeSubset(GrRecordingContext* context,
141 const SkIRect& subset) const {
142 if (!context || !fContext->priv().matches(context)) {
143 return nullptr;
144 }
145
Greg Daniel37c127f2020-02-05 10:37:27 -0500146 const GrSurfaceProxyView* view = this->view(context);
147 SkASSERT(view && view->proxy());
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400148
Greg Daniel3155f7f2020-01-16 16:54:26 -0500149 GrColorType grColorType = SkColorTypeToGrColorType(this->colorType());
150
Greg Daniel40903af2020-01-30 14:55:05 -0500151 GrSurfaceProxyView copyView =
Greg Daniel37c127f2020-02-05 10:37:27 -0500152 GrSurfaceProxy::Copy(context, view->proxy(), view->origin(), grColorType,
Greg Daniel40903af2020-01-30 14:55:05 -0500153 GrMipMapped::kNo, subset, SkBackingFit::kExact,
Greg Daniel37c127f2020-02-05 10:37:27 -0500154 view->proxy()->isBudgeted());
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400155
Greg Daniel40903af2020-01-30 14:55:05 -0500156 if (!copyView.proxy()) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400157 return nullptr;
158 }
159
160 // MDB: this call is okay bc we know 'sContext' was kExact
Greg Daniel40903af2020-01-30 14:55:05 -0500161 return sk_make_sp<SkImage_Gpu>(fContext, kNeedNewImageUniqueID, std::move(copyView),
Greg Daniel7c165a42020-01-22 12:22:36 -0500162 this->colorType(), this->alphaType(), this->refColorSpace());
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400163}
164
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400165bool SkImage_GpuBase::onReadPixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRB,
166 int srcX, int srcY, CachingHint) const {
Robert Phillips920d4882019-03-04 15:16:44 -0500167 auto direct = fContext->priv().asDirectContext();
168 if (!direct) {
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400169 // DDL TODO: buffer up the readback so it occurs when the DDL is drawn?
170 return false;
171 }
172
Brian Salomon5ad6fd32019-03-21 15:30:08 -0400173 if (!SkImageInfoValidConversion(dstInfo, this->imageInfo())) {
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400174 return false;
175 }
176
Greg Daniel37c127f2020-02-05 10:37:27 -0500177 const GrSurfaceProxyView* view = this->view(direct);
178 SkASSERT(view);
Greg Danielba88ab62019-07-26 09:14:01 -0400179 GrColorType grColorType = SkColorTypeAndFormatToGrColorType(fContext->priv().caps(),
180 this->colorType(),
Greg Daniel37c127f2020-02-05 10:37:27 -0500181 view->proxy()->backendFormat());
Greg Danielba88ab62019-07-26 09:14:01 -0400182
Greg Daniel37c127f2020-02-05 10:37:27 -0500183 auto sContext = GrSurfaceContext::Make(direct, *view, grColorType, this->alphaType(),
184 this->refColorSpace());
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400185 if (!sContext) {
186 return false;
187 }
188
Brian Salomon1d435302019-07-01 13:05:28 -0400189 return sContext->readPixels(dstInfo, dstPixels, dstRB, {srcX, srcY});
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400190}
191
Greg Danielfebdedf2020-02-05 17:06:27 -0500192GrSurfaceProxyView SkImage_GpuBase::refView(GrRecordingContext* context, GrSamplerState params,
193 SkScalar scaleAdjust[2]) const {
Robert Phillips6603a172019-03-05 12:35:44 -0500194 if (!context || !fContext->priv().matches(context)) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400195 SkASSERT(0);
Greg Danielfebdedf2020-02-05 17:06:27 -0500196 return {};
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400197 }
198
Greg Daniel37c127f2020-02-05 10:37:27 -0500199 GrTextureAdjuster adjuster(fContext.get(), *this->view(context), this->imageInfo().colorInfo(),
200 this->uniqueID());
Greg Danielfebdedf2020-02-05 17:06:27 -0500201 return adjuster.viewForParams(params, scaleAdjust);
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400202}
203
204GrBackendTexture SkImage_GpuBase::onGetBackendTexture(bool flushPendingGrContextIO,
205 GrSurfaceOrigin* origin) const {
Robert Phillips920d4882019-03-04 15:16:44 -0500206 auto direct = fContext->priv().asDirectContext();
207 if (!direct) {
208 // This image was created with a DDL context and cannot be instantiated.
Robert Phillips6603a172019-03-05 12:35:44 -0500209 return GrBackendTexture(); // invalid
Robert Phillips920d4882019-03-04 15:16:44 -0500210 }
211
Greg Danielfebdedf2020-02-05 17:06:27 -0500212 const GrSurfaceProxyView* view = this->view(direct);
213 SkASSERT(view && *view);
214 GrSurfaceProxy* proxy = view->proxy();
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400215
Robert Phillips920d4882019-03-04 15:16:44 -0500216 if (!proxy->isInstantiated()) {
217 auto resourceProvider = direct->priv().resourceProvider();
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400218
Robert Phillips920d4882019-03-04 15:16:44 -0500219 if (!proxy->instantiate(resourceProvider)) {
220 return GrBackendTexture(); // invalid
221 }
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400222 }
223
224 GrTexture* texture = proxy->peekTexture();
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400225 if (texture) {
226 if (flushPendingGrContextIO) {
Greg Danielfebdedf2020-02-05 17:06:27 -0500227 direct->priv().flushSurface(proxy);
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400228 }
229 if (origin) {
230 *origin = proxy->origin();
231 }
232 return texture->getBackendTexture();
233 }
234 return GrBackendTexture(); // invalid
235}
236
237GrTexture* SkImage_GpuBase::onGetTexture() const {
238 GrTextureProxy* proxy = this->peekProxy();
Robert Phillips193c4212019-03-04 12:18:53 -0500239 if (proxy && proxy->isInstantiated()) {
240 return proxy->peekTexture();
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400241 }
242
Robert Phillips193c4212019-03-04 12:18:53 -0500243 auto direct = fContext->priv().asDirectContext();
244 if (!direct) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400245 // This image was created with a DDL context and cannot be instantiated.
246 return nullptr;
247 }
248
Greg Danielfebdedf2020-02-05 17:06:27 -0500249 const GrSurfaceProxyView* view = this->view(direct);
250 SkASSERT(view && *view && !view->proxy()->isInstantiated());
Robert Phillips193c4212019-03-04 12:18:53 -0500251
Greg Danielfebdedf2020-02-05 17:06:27 -0500252 if (!view->proxy()->instantiate(direct->priv().resourceProvider())) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400253 return nullptr;
254 }
255
Greg Danielfebdedf2020-02-05 17:06:27 -0500256 return view->proxy()->peekTexture();
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400257}
258
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400259bool SkImage_GpuBase::onIsValid(GrContext* context) const {
260 // The base class has already checked that context isn't abandoned (if it's not nullptr)
Robert Phillips920d4882019-03-04 15:16:44 -0500261 if (fContext->priv().abandoned()) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400262 return false;
263 }
264
Robert Phillips920d4882019-03-04 15:16:44 -0500265 if (context && !fContext->priv().matches(context)) {
Jim Van Verth8026ccc2018-10-04 13:10:39 -0400266 return false;
267 }
268
269 return true;
270}
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400271
Jim Van Verth0e671942018-11-09 12:03:57 -0500272bool SkImage_GpuBase::MakeTempTextureProxies(GrContext* ctx, const GrBackendTexture yuvaTextures[],
Jim Van Verth53275362018-11-09 15:42:35 -0500273 int numTextures, const SkYUVAIndex yuvaIndices[4],
274 GrSurfaceOrigin imageOrigin,
Jim Van Verth0e671942018-11-09 12:03:57 -0500275 sk_sp<GrTextureProxy> tempTextureProxies[4]) {
Robert Phillips9da87e02019-02-04 13:26:26 -0500276 GrProxyProvider* proxyProvider = ctx->priv().proxyProvider();
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400277 const GrCaps* caps = ctx->priv().caps();
Jim Van Verth0e671942018-11-09 12:03:57 -0500278
Jim Van Verth0e671942018-11-09 12:03:57 -0500279 for (int textureIndex = 0; textureIndex < numTextures; ++textureIndex) {
Robert Phillips62221e72019-07-24 15:07:38 -0400280 GrBackendFormat backendFormat = yuvaTextures[textureIndex].getBackendFormat();
Brian Salomonf391d0f2018-12-14 09:18:50 -0500281 if (!backendFormat.isValid()) {
282 return false;
283 }
Robert Phillips1cd1ed82019-07-23 13:21:01 -0400284
Robert Phillips00c9f0d2019-08-02 17:17:35 -0400285 GrColorType grColorType = caps->getYUVAColorTypeFromBackendFormat(
286 backendFormat,
287 yuvaIndices[3].fIndex == textureIndex);
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400288 if (GrColorType::kUnknown == grColorType) {
289 return false;
290 }
291
Robert Phillips62221e72019-07-24 15:07:38 -0400292 SkASSERT(yuvaTextures[textureIndex].isValid());
Jim Van Verth0e671942018-11-09 12:03:57 -0500293
Brian Salomonaa6ca0a2019-01-24 16:03:07 -0500294 tempTextureProxies[textureIndex] = proxyProvider->wrapBackendTexture(
Robert Phillips62221e72019-07-24 15:07:38 -0400295 yuvaTextures[textureIndex], grColorType, imageOrigin, kBorrow_GrWrapOwnership,
Brian Salomonaa6ca0a2019-01-24 16:03:07 -0500296 GrWrapCacheable::kNo, kRead_GrIOType);
Jim Van Verth0e671942018-11-09 12:03:57 -0500297 if (!tempTextureProxies[textureIndex]) {
298 return false;
299 }
Jim Van Verth53275362018-11-09 15:42:35 -0500300
301 // Check that each texture contains the channel data for the corresponding YUVA index
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400302 auto componentFlags = GrColorTypeComponentFlags(grColorType);
Jim Van Verth53275362018-11-09 15:42:35 -0500303 for (int yuvaIndex = 0; yuvaIndex < SkYUVAIndex::kIndexCount; ++yuvaIndex) {
304 if (yuvaIndices[yuvaIndex].fIndex == textureIndex) {
305 switch (yuvaIndices[yuvaIndex].fChannel) {
306 case SkColorChannel::kR:
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400307 // TODO: Chrome needs to be patched before this can be
308 // enforced.
309// if (!(kRed_SkColorTypeComponentFlag & componentFlags)) {
310// return false;
311// }
312 break;
313 case SkColorChannel::kG:
314 if (!(kGreen_SkColorTypeComponentFlag & componentFlags)) {
Jim Van Verth53275362018-11-09 15:42:35 -0500315 return false;
316 }
317 break;
Jim Van Verth53275362018-11-09 15:42:35 -0500318 case SkColorChannel::kB:
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400319 if (!(kBlue_SkColorTypeComponentFlag & componentFlags)) {
Jim Van Verth53275362018-11-09 15:42:35 -0500320 return false;
321 }
322 break;
323 case SkColorChannel::kA:
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400324 if (!(kAlpha_SkColorTypeComponentFlag & componentFlags)) {
Jim Van Verth53275362018-11-09 15:42:35 -0500325 return false;
326 }
327 break;
328 }
329 }
330 }
Jim Van Verth0e671942018-11-09 12:03:57 -0500331 }
332
333 return true;
334}
335
336bool SkImage_GpuBase::RenderYUVAToRGBA(GrContext* ctx, GrRenderTargetContext* renderTargetContext,
337 const SkRect& rect, SkYUVColorSpace yuvColorSpace,
Brian Osmane9560492019-02-05 17:00:03 -0500338 sk_sp<GrColorSpaceXform> colorSpaceXform,
Jim Van Verth0e671942018-11-09 12:03:57 -0500339 const sk_sp<GrTextureProxy> proxies[4],
340 const SkYUVAIndex yuvaIndices[4]) {
341 SkASSERT(renderTargetContext);
342 if (!renderTargetContext->asSurfaceProxy()) {
343 return false;
344 }
345
346 GrPaint paint;
347 paint.setPorterDuffXPFactory(SkBlendMode::kSrc);
348
Brian Salomonca6b2f42020-01-24 11:31:21 -0500349 const auto& caps = *ctx->priv().caps();
Brian Osmane9560492019-02-05 17:00:03 -0500350 auto fp = GrYUVtoRGBEffect::Make(proxies, yuvaIndices, yuvColorSpace,
Brian Salomonca6b2f42020-01-24 11:31:21 -0500351 GrSamplerState::Filter::kNearest, caps);
Brian Osmane9560492019-02-05 17:00:03 -0500352 if (colorSpaceXform) {
353 fp = GrColorSpaceXformEffect::Make(std::move(fp), std::move(colorSpaceXform));
354 }
355 paint.addColorFragmentProcessor(std::move(fp));
Jim Van Verth0e671942018-11-09 12:03:57 -0500356
357 renderTargetContext->drawRect(GrNoClip(), std::move(paint), GrAA::kNo, SkMatrix::I(), rect);
Jim Van Verth0e671942018-11-09 12:03:57 -0500358 return true;
359}
360
Brian Salomonbe5a0932018-12-10 10:03:26 -0500361sk_sp<GrTextureProxy> SkImage_GpuBase::MakePromiseImageLazyProxy(
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400362 GrContext* context, int width, int height, GrSurfaceOrigin origin, GrColorType colorType,
Brian Salomonbe5a0932018-12-10 10:03:26 -0500363 GrBackendFormat backendFormat, GrMipMapped mipMapped,
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500364 PromiseImageTextureFulfillProc fulfillProc,
365 PromiseImageTextureReleaseProc releaseProc,
366 PromiseImageTextureDoneProc doneProc,
Brian Salomon0cc57542019-03-08 13:28:46 -0500367 PromiseImageTextureContext textureContext,
368 PromiseImageApiVersion version) {
Brian Salomonbe5a0932018-12-10 10:03:26 -0500369 SkASSERT(context);
370 SkASSERT(width > 0 && height > 0);
371 SkASSERT(doneProc);
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400372 SkASSERT(colorType != GrColorType::kUnknown);
Brian Salomonbe5a0932018-12-10 10:03:26 -0500373
374 if (!fulfillProc || !releaseProc) {
375 doneProc(textureContext);
376 return nullptr;
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400377 }
378
Brian Salomonbe5a0932018-12-10 10:03:26 -0500379 if (mipMapped == GrMipMapped::kYes &&
380 GrTextureTypeHasRestrictedSampling(backendFormat.textureType())) {
381 // It is invalid to have a GL_TEXTURE_EXTERNAL or GL_TEXTURE_RECTANGLE and have mips as
382 // well.
383 doneProc(textureContext);
384 return nullptr;
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400385 }
Brian Salomonbe5a0932018-12-10 10:03:26 -0500386
387 /**
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500388 * This class is the lazy instantiation callback for promise images. It manages calling the
389 * client's Fulfill, Release, and Done procs. It attempts to reuse a GrTexture instance in
Brian Salomonb2c5dae2019-03-04 10:25:17 -0500390 * cases where the client provides the same SkPromiseImageTexture as Fulfill results for
391 * multiple SkImages. The created GrTexture is given a key based on a unique ID associated with
392 * the SkPromiseImageTexture.
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500393 *
Brian Salomonb2c5dae2019-03-04 10:25:17 -0500394 * The GrTexutre idle proc mechanism is used to call the Release and Done procs. We use this
395 * instead of the GrSurface release proc because the GrTexture is cached and therefore may
396 * outlive the proxy into which this callback is installed.
397 *
398 * A key invalidation message is installed on the SkPromiseImageTexture so that the GrTexture
399 * is deleted once it can no longer be used to instantiate a proxy.
Brian Salomonbe5a0932018-12-10 10:03:26 -0500400 */
401 class PromiseLazyInstantiateCallback {
402 public:
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500403 PromiseLazyInstantiateCallback(PromiseImageTextureFulfillProc fulfillProc,
404 PromiseImageTextureReleaseProc releaseProc,
405 PromiseImageTextureDoneProc doneProc,
406 PromiseImageTextureContext context,
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400407 GrColorType colorType,
Brian Salomon0cc57542019-03-08 13:28:46 -0500408 PromiseImageApiVersion version)
409 : fFulfillProc(fulfillProc)
410 , fReleaseProc(releaseProc)
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400411 , fColorType(colorType)
Brian Salomon0cc57542019-03-08 13:28:46 -0500412 , fVersion(version) {
Brian Salomone80b8092019-03-08 13:25:19 -0500413 fDoneCallback = sk_make_sp<GrRefCntedCallback>(doneProc, context);
Brian Salomonb2c5dae2019-03-04 10:25:17 -0500414 }
415 PromiseLazyInstantiateCallback(PromiseLazyInstantiateCallback&&) = default;
416 PromiseLazyInstantiateCallback(const PromiseLazyInstantiateCallback&) {
417 // Because we get wrapped in std::function we must be copyable. But we should never
418 // be copied.
419 SkASSERT(false);
420 }
421 PromiseLazyInstantiateCallback& operator=(PromiseLazyInstantiateCallback&&) = default;
422 PromiseLazyInstantiateCallback& operator=(const PromiseLazyInstantiateCallback&) {
423 SkASSERT(false);
424 return *this;
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500425 }
Brian Salomon553610d2019-01-14 17:34:12 -0500426
Brian Salomon88b8d112019-03-07 15:25:34 +0000427 ~PromiseLazyInstantiateCallback() {
Brian Salomon876a0172019-03-08 11:12:14 -0500428 // Our destructor can run on any thread. We trigger the unref of fTexture by message.
Robert Phillipsddc21482019-10-16 14:30:09 -0400429 // This unreffed texture pointer is a real problem! When the context has been
430 // abandoned, the GrTexture pointed to by this pointer is deleted! Due to virtual
431 // inheritance any manipulation of this pointer at that point will cause a crash.
432 // For now we "work around" the problem by just passing it, untouched, into the
433 // message bus but this very fragile.
434 // In the future the GrSurface class hierarchy refactoring should eliminate this
435 // difficulty by removing the virtual inheritance.
Brian Salomon876a0172019-03-08 11:12:14 -0500436 if (fTexture) {
Robert Phillipsddc21482019-10-16 14:30:09 -0400437 SkMessageBus<GrTextureFreedMessage>::Post({fTexture, fTextureContextID});
Brian Salomon876a0172019-03-08 11:12:14 -0500438 }
Brian Salomon88b8d112019-03-07 15:25:34 +0000439 }
440
Brian Salomonbeb7f522019-08-30 16:19:42 -0400441 GrSurfaceProxy::LazyCallbackResult operator()(GrResourceProvider* resourceProvider) {
Brian Salomonb6a3a3b2019-04-01 12:29:34 -0400442 // We use the unique key in a way that is unrelated to the SkImage-based key that the
443 // proxy may receive, hence kUnsynced.
444 static constexpr auto kKeySyncMode =
445 GrSurfaceProxy::LazyInstantiationKeyMode::kUnsynced;
446
Brian Salomonbeb7f522019-08-30 16:19:42 -0400447 // In order to make the SkImage "thread safe" we rely on holding an extra ref to the
448 // texture in the callback and signalling the unref via a message to the resource cache.
449 // We need to extend the callback's lifetime to that of the proxy.
450 static constexpr auto kReleaseCallbackOnInstantiation = false;
451
Brian Salomon876a0172019-03-08 11:12:14 -0500452 // Our proxy is getting instantiated for the second+ time. We are only allowed to call
453 // Fulfill once. So return our cached result.
454 if (fTexture) {
Brian Salomonbeb7f522019-08-30 16:19:42 -0400455 return {sk_ref_sp(fTexture), kReleaseCallbackOnInstantiation, kKeySyncMode};
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400456 } else if (fColorType == GrColorType::kUnknown) {
Brian Salomond538d3d2019-04-04 12:18:17 -0400457 // We've already called fulfill and it failed. Our contract says that we should only
458 // call each callback once.
459 return {};
Brian Salomon876a0172019-03-08 11:12:14 -0500460 }
Brian Salomone80b8092019-03-08 13:25:19 -0500461 SkASSERT(fDoneCallback);
462 PromiseImageTextureContext textureContext = fDoneCallback->context();
Brian Salomonb2c5dae2019-03-04 10:25:17 -0500463 sk_sp<SkPromiseImageTexture> promiseTexture = fFulfillProc(textureContext);
464 // From here on out our contract is that the release proc must be called, even if
465 // the return from fulfill was invalid or we fail for some other reason.
Brian Salomone80b8092019-03-08 13:25:19 -0500466 auto releaseCallback = sk_make_sp<GrRefCntedCallback>(fReleaseProc, textureContext);
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500467 if (!promiseTexture) {
Brian Salomond538d3d2019-04-04 12:18:17 -0400468 // This records that we have failed.
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400469 fColorType = GrColorType::kUnknown;
Brian Salomonb6a3a3b2019-04-01 12:29:34 -0400470 return {};
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500471 }
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500472
Robert Phillips62221e72019-07-24 15:07:38 -0400473 const GrBackendTexture& backendTexture = promiseTexture->backendTexture();
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500474 if (!backendTexture.isValid()) {
Brian Salomonb6a3a3b2019-04-01 12:29:34 -0400475 return {};
Brian Salomon559c6172019-01-10 10:23:44 -0500476 }
477
Brian Salomonf55e8d52019-01-30 17:28:20 -0500478 sk_sp<GrTexture> tex;
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500479 static const GrUniqueKey::Domain kDomain = GrUniqueKey::GenerateDomain();
Brian Salomon7d88f312019-02-28 10:03:03 -0500480 GrUniqueKey key;
481 GrUniqueKey::Builder builder(&key, kDomain, 2, "promise");
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500482 builder[0] = promiseTexture->uniqueID();
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400483 builder[1] = (uint32_t) fColorType;
Brian Salomoncdd8a0a2019-01-10 12:09:52 -0500484 builder.finish();
Brian Salomon0d606762019-01-25 09:58:38 -0500485 // A texture with this key may already exist from a different instance of this lazy
486 // callback. This could happen if the client fulfills a promise image with a texture
487 // that was previously used to fulfill a different promise image.
Brian Salomon7d88f312019-02-28 10:03:03 -0500488 if (auto surf = resourceProvider->findByUniqueKey<GrSurface>(key)) {
Brian Salomon0d606762019-01-25 09:58:38 -0500489 tex = sk_ref_sp(surf->asTexture());
490 SkASSERT(tex);
491 } else {
492 if ((tex = resourceProvider->wrapBackendTexture(
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400493 backendTexture, fColorType, kBorrow_GrWrapOwnership,
494 GrWrapCacheable::kYes, kRead_GrIOType))) {
Brian Salomon7d88f312019-02-28 10:03:03 -0500495 tex->resourcePriv().setUniqueKey(key);
Brian Salomon0d606762019-01-25 09:58:38 -0500496 } else {
Brian Salomonb6a3a3b2019-04-01 12:29:34 -0400497 return {};
Brian Salomon0d606762019-01-25 09:58:38 -0500498 }
499 }
Brian Salomon0cc57542019-03-08 13:28:46 -0500500 auto releaseIdleState = fVersion == PromiseImageApiVersion::kLegacy
501 ? GrTexture::IdleState::kFinished
502 : GrTexture::IdleState::kFlushed;
503 tex->addIdleProc(std::move(releaseCallback), releaseIdleState);
Brian Salomone80b8092019-03-08 13:25:19 -0500504 tex->addIdleProc(std::move(fDoneCallback), GrTexture::IdleState::kFinished);
Brian Salomonb2c5dae2019-03-04 10:25:17 -0500505 promiseTexture->addKeyToInvalidate(tex->getContext()->priv().contextID(), key);
Brian Salomon876a0172019-03-08 11:12:14 -0500506 fTexture = tex.get();
507 // We need to hold on to the GrTexture in case our proxy gets reinstantiated. However,
508 // we can't unref in our destructor because we may be on another thread then. So we
509 // let the cache know it is waiting on an unref message. We will send that message from
510 // our destructor.
511 GrContext* context = fTexture->getContext();
Robert Phillipsddc21482019-10-16 14:30:09 -0400512 context->priv().getResourceCache()->insertDelayedTextureUnref(fTexture);
Brian Salomon876a0172019-03-08 11:12:14 -0500513 fTextureContextID = context->priv().contextID();
Brian Salomonbeb7f522019-08-30 16:19:42 -0400514 return {std::move(tex), kReleaseCallbackOnInstantiation, kKeySyncMode};
Brian Salomonbe5a0932018-12-10 10:03:26 -0500515 }
516
517 private:
Brian Salomone80b8092019-03-08 13:25:19 -0500518 PromiseImageTextureFulfillProc fFulfillProc;
519 PromiseImageTextureReleaseProc fReleaseProc;
520 sk_sp<GrRefCntedCallback> fDoneCallback;
Brian Salomon876a0172019-03-08 11:12:14 -0500521 GrTexture* fTexture = nullptr;
522 uint32_t fTextureContextID = SK_InvalidUniqueID;
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400523 GrColorType fColorType;
Brian Salomon0cc57542019-03-08 13:28:46 -0500524 PromiseImageApiVersion fVersion;
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400525 } callback(fulfillProc, releaseProc, doneProc, textureContext, colorType, version);
Brian Salomonbe5a0932018-12-10 10:03:26 -0500526
Robert Phillips9da87e02019-02-04 13:26:26 -0500527 GrProxyProvider* proxyProvider = context->priv().proxyProvider();
Brian Salomonbe5a0932018-12-10 10:03:26 -0500528
529 GrSurfaceDesc desc;
530 desc.fWidth = width;
531 desc.fHeight = height;
Brian Salomonbe5a0932018-12-10 10:03:26 -0500532
Chris Dalton95d8ceb2019-07-30 11:17:59 -0600533 // Ganesh assumes that, when wrapping a mipmapped backend texture from a client, that its
534 // mipmaps are fully fleshed out.
535 GrMipMapsStatus mipMapsStatus = (GrMipMapped::kYes == mipMapped)
536 ? GrMipMapsStatus::kValid : GrMipMapsStatus::kNotAllocated;
537
Greg Danielce3ddaa2020-01-22 16:58:15 -0500538 GrSwizzle readSwizzle = context->priv().caps()->getReadSwizzle(backendFormat, colorType);
539
Brian Salomonbe5a0932018-12-10 10:03:26 -0500540 // We pass kReadOnly here since we should treat content of the client's texture as immutable.
Brian Salomone8a766b2019-07-19 14:24:36 -0400541 // The promise API provides no way for the client to indicated that the texture is protected.
Brian Salomonf2c2ba92019-07-17 09:59:59 -0400542 return proxyProvider->createLazyProxy(
Greg Danielce3ddaa2020-01-22 16:58:15 -0500543 std::move(callback), backendFormat, desc, readSwizzle, GrRenderable::kNo, 1, origin,
544 mipMapped, mipMapsStatus, GrInternalSurfaceFlags::kReadOnly, SkBackingFit::kExact,
545 SkBudgeted::kNo, GrProtected::kNo, GrSurfaceProxy::UseAllocator::kYes);
Jim Van Verth8bbce0e2018-10-08 14:34:52 -0400546}