blob: 6b5d3f01435a1a893bbc8fb5255e43c52a2b8794 [file] [log] [blame]
robertphillips76948d42016-05-04 12:47:41 -07001/*
2 * Copyright 2016 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
Greg Danielf91aeb22019-06-18 09:58:02 -04008#include "src/gpu/GrTextureProxy.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -05009#include "src/gpu/GrTextureProxyPriv.h"
robertphillips76948d42016-05-04 12:47:41 -070010
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/gpu/GrContext.h"
12#include "src/gpu/GrContextPriv.h"
13#include "src/gpu/GrDeferredProxyUploader.h"
14#include "src/gpu/GrProxyProvider.h"
15#include "src/gpu/GrSurfacePriv.h"
16#include "src/gpu/GrTexturePriv.h"
robertphillips76948d42016-05-04 12:47:41 -070017
Brian Salomon58389b92018-03-07 13:01:25 -050018// Deferred version - no data
Brian Salomonbeb7f522019-08-30 16:19:42 -040019GrTextureProxy::GrTextureProxy(const GrBackendFormat& format,
20 const GrSurfaceDesc& srcDesc,
21 GrSurfaceOrigin origin,
22 GrMipMapped mipMapped,
23 GrMipMapsStatus mipMapsStatus,
24 const GrSwizzle& textureSwizzle,
25 SkBackingFit fit,
26 SkBudgeted budgeted,
27 GrProtected isProtected,
28 GrInternalSurfaceFlags surfaceFlags,
29 UseAllocator useAllocator)
Brian Salomonf2c2ba92019-07-17 09:59:59 -040030 : INHERITED(format, srcDesc, GrRenderable::kNo, origin, textureSwizzle, fit, budgeted,
Brian Salomonbeb7f522019-08-30 16:19:42 -040031 isProtected, surfaceFlags, useAllocator)
Brian Salomon58389b92018-03-07 13:01:25 -050032 , fMipMapped(mipMapped)
Brian Salomonbeb7f522019-08-30 16:19:42 -040033 , fMipMapsStatus(mipMapsStatus) SkDEBUGCODE(, fInitialMipMapsStatus(fMipMapsStatus))
Brian Salomon58389b92018-03-07 13:01:25 -050034 , fProxyProvider(nullptr)
Stephen White3c0a50f2020-01-16 18:19:54 -050035 , fDeferredUploader(nullptr) {
36 SkASSERT(!(fSurfaceFlags & GrInternalSurfaceFlags::kFramebufferOnly));
37}
Brian Salomon58389b92018-03-07 13:01:25 -050038
Chris Dalton706a6ff2017-11-29 22:01:06 -070039// Lazy-callback version
Brian Salomonbeb7f522019-08-30 16:19:42 -040040GrTextureProxy::GrTextureProxy(LazyInstantiateCallback&& callback,
41 const GrBackendFormat& format,
42 const GrSurfaceDesc& desc,
43 GrSurfaceOrigin origin,
44 GrMipMapped mipMapped,
45 GrMipMapsStatus mipMapsStatus,
46 const GrSwizzle& texSwizzle,
47 SkBackingFit fit,
48 SkBudgeted budgeted,
49 GrProtected isProtected,
50 GrInternalSurfaceFlags surfaceFlags,
51 UseAllocator useAllocator)
52 : INHERITED(std::move(callback), format, desc, GrRenderable::kNo, origin, texSwizzle, fit,
53 budgeted, isProtected, surfaceFlags, useAllocator)
Greg Daniel65fa8ca2018-01-10 17:06:31 -050054 , fMipMapped(mipMapped)
Brian Salomonbeb7f522019-08-30 16:19:42 -040055 , fMipMapsStatus(mipMapsStatus) SkDEBUGCODE(, fInitialMipMapsStatus(fMipMapsStatus))
Robert Phillips1afd4cd2018-01-08 13:40:32 -050056 , fProxyProvider(nullptr)
Stephen White3c0a50f2020-01-16 18:19:54 -050057 , fDeferredUploader(nullptr) {
58 SkASSERT(!(fSurfaceFlags & GrInternalSurfaceFlags::kFramebufferOnly));
59}
Chris Dalton706a6ff2017-11-29 22:01:06 -070060
61// Wrapped version
Brian Salomonbeb7f522019-08-30 16:19:42 -040062GrTextureProxy::GrTextureProxy(sk_sp<GrSurface> surf,
63 GrSurfaceOrigin origin,
64 const GrSwizzle& textureSwizzle,
65 UseAllocator useAllocator)
66 : INHERITED(std::move(surf), origin, textureSwizzle, SkBackingFit::kExact, useAllocator)
Greg Daniele252f082017-10-23 16:05:23 -040067 , fMipMapped(fTarget->asTexture()->texturePriv().mipMapped())
Chris Dalton95d8ceb2019-07-30 11:17:59 -060068 , fMipMapsStatus(fTarget->asTexture()->texturePriv().mipMapsStatus())
Brian Salomonbeb7f522019-08-30 16:19:42 -040069 SkDEBUGCODE(, fInitialMipMapsStatus(fMipMapsStatus))
Robert Phillips1afd4cd2018-01-08 13:40:32 -050070 , fProxyProvider(nullptr)
Brian Osman099fa0f2017-10-02 16:38:32 -040071 , fDeferredUploader(nullptr) {
Robert Phillipsae7d3f32017-09-21 08:26:08 -040072 if (fTarget->getUniqueKey().isValid()) {
Robert Phillips9da87e02019-02-04 13:26:26 -050073 fProxyProvider = fTarget->asTexture()->getContext()->priv().proxyProvider();
Robert Phillipse5f73282019-06-18 17:15:04 -040074 fProxyProvider->adoptUniqueKeyFromSurface(this, fTarget.get());
Robert Phillipsae7d3f32017-09-21 08:26:08 -040075 }
76}
77
78GrTextureProxy::~GrTextureProxy() {
79 // Due to the order of cleanup the GrSurface this proxy may have wrapped may have gone away
80 // at this point. Zero out the pointer so the cache invalidation code doesn't try to use it.
81 fTarget = nullptr;
Robert Phillips0790f8a2018-09-18 13:11:03 -040082
83 // In DDL-mode, uniquely keyed proxies keep their key even after their originating
84 // proxy provider has gone away. In that case there is noone to send the invalid key
85 // message to (Note: in this case we don't want to remove its cached resource).
86 if (fUniqueKey.isValid() && fProxyProvider) {
Robert Phillips427966a2018-12-20 17:20:43 -050087 fProxyProvider->processInvalidUniqueKey(fUniqueKey, this,
88 GrProxyProvider::InvalidateGPUResource::kNo);
Robert Phillipsae7d3f32017-09-21 08:26:08 -040089 } else {
Robert Phillips1afd4cd2018-01-08 13:40:32 -050090 SkASSERT(!fProxyProvider);
Robert Phillipsae7d3f32017-09-21 08:26:08 -040091 }
Robert Phillipseee4d6e2017-06-05 09:26:07 -040092}
robertphillips76948d42016-05-04 12:47:41 -070093
Robert Phillips10d17212019-04-24 14:09:10 -040094bool GrTextureProxy::instantiate(GrResourceProvider* resourceProvider) {
Brian Salomonbeb7f522019-08-30 16:19:42 -040095 if (this->isLazy()) {
Greg Daniel0a375db2018-02-01 12:21:39 -050096 return false;
97 }
Chris Dalton0b68dda2019-11-07 21:08:03 -070098 if (!this->instantiateImpl(resourceProvider, 1, GrRenderable::kNo, fMipMapped,
99 fUniqueKey.isValid() ? &fUniqueKey : nullptr)) {
Robert Phillipseee4d6e2017-06-05 09:26:07 -0400100 return false;
robertphillips76948d42016-05-04 12:47:41 -0700101 }
Robert Phillipseee4d6e2017-06-05 09:26:07 -0400102
Robert Phillipsb5204762019-06-19 14:12:13 -0400103 SkASSERT(!this->peekRenderTarget());
104 SkASSERT(this->peekTexture());
Robert Phillipseee4d6e2017-06-05 09:26:07 -0400105 return true;
robertphillips76948d42016-05-04 12:47:41 -0700106}
107
Robert Phillips5af44de2017-07-18 14:49:38 -0400108sk_sp<GrSurface> GrTextureProxy::createSurface(GrResourceProvider* resourceProvider) const {
Chris Dalton0b68dda2019-11-07 21:08:03 -0700109 sk_sp<GrSurface> surface = this->createSurfaceImpl(resourceProvider, 1, GrRenderable::kNo,
110 fMipMapped);
Robert Phillips5af44de2017-07-18 14:49:38 -0400111 if (!surface) {
112 return nullptr;
113 }
114
Greg Daniele252f082017-10-23 16:05:23 -0400115 SkASSERT(!surface->asRenderTarget());
Robert Phillips5af44de2017-07-18 14:49:38 -0400116 SkASSERT(surface->asTexture());
117 return surface;
118}
119
Brian Osman099fa0f2017-10-02 16:38:32 -0400120void GrTextureProxyPriv::setDeferredUploader(std::unique_ptr<GrDeferredProxyUploader> uploader) {
121 SkASSERT(!fTextureProxy->fDeferredUploader);
122 fTextureProxy->fDeferredUploader = std::move(uploader);
123}
124
125void GrTextureProxyPriv::scheduleUpload(GrOpFlushState* flushState) {
Robert Phillipsa3f70262018-02-08 10:59:38 -0500126 // The texture proxy's contents may already have been uploaded or instantiation may have failed
Robert Phillipsb5204762019-06-19 14:12:13 -0400127 if (fTextureProxy->fDeferredUploader && fTextureProxy->isInstantiated()) {
Brian Osman099fa0f2017-10-02 16:38:32 -0400128 fTextureProxy->fDeferredUploader->scheduleUpload(flushState, fTextureProxy);
129 }
130}
131
132void GrTextureProxyPriv::resetDeferredUploader() {
133 SkASSERT(fTextureProxy->fDeferredUploader);
134 fTextureProxy->fDeferredUploader.reset();
135}
136
Greg Daniel3b2ebbb2018-02-09 10:49:23 -0500137GrMipMapped GrTextureProxy::mipMapped() const {
Brian Salomonfd98c2c2018-07-31 17:25:29 -0400138 if (this->isInstantiated()) {
139 return this->peekTexture()->texturePriv().mipMapped();
Greg Daniel3b2ebbb2018-02-09 10:49:23 -0500140 }
141 return fMipMapped;
142}
143
Greg Daniel7fd7a8a2019-10-10 16:10:31 -0400144size_t GrTextureProxy::onUninstantiatedGpuMemorySize(const GrCaps& caps) const {
Brian Salomon9f2b86c2019-10-22 10:37:46 -0400145 return GrSurface::ComputeSize(caps, this->backendFormat(), this->dimensions(), 1,
146 this->proxyMipMapped(), !this->priv().isExact());
Robert Phillips8bc06d02016-11-01 17:28:40 -0400147}
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400148
Michael Ludwig8fa469d2019-11-25 16:08:44 -0500149GrSamplerState::Filter GrTextureProxy::HighestFilterMode(GrTextureType textureType) {
150 return GrTextureTypeHasRestrictedSampling(textureType) ? GrSamplerState::Filter::kBilerp
151 : GrSamplerState::Filter::kMipMap;
152}
153
Michael Ludwigfcdd0612019-11-25 08:34:31 -0500154bool GrTextureProxy::ProxiesAreCompatibleAsDynamicState(const GrSurfaceProxy* first,
155 const GrSurfaceProxy* second) {
Michael Ludwig08bd1f72019-12-02 14:27:21 -0500156 // In order to be compatible, the proxies should also have the same texture type. This is
157 // checked explicitly since the GrBackendFormat == operator does not compare texture type
Greg Danielc71c7962020-01-14 16:44:18 -0500158 return first->backendFormat().textureType() == second->backendFormat().textureType() &&
Greg Daniel45723ac2018-11-30 10:12:43 -0500159 first->backendFormat() == second->backendFormat();
160}
161
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500162void GrTextureProxy::setUniqueKey(GrProxyProvider* proxyProvider, const GrUniqueKey& key) {
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400163 SkASSERT(key.isValid());
164 SkASSERT(!fUniqueKey.isValid()); // proxies can only ever get one uniqueKey
165
Brian Salomonb6a3a3b2019-04-01 12:29:34 -0400166 if (fTarget && fSyncTargetKey) {
Robert Phillipsadbe1322018-01-17 13:35:46 -0500167 if (!fTarget->getUniqueKey().isValid()) {
168 fTarget->resourcePriv().setUniqueKey(key);
169 }
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400170 SkASSERT(fTarget->getUniqueKey() == key);
171 }
172
173 fUniqueKey = key;
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500174 fProxyProvider = proxyProvider;
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400175}
176
177void GrTextureProxy::clearUniqueKey() {
178 fUniqueKey.reset();
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500179 fProxyProvider = nullptr;
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400180}
181
Chris Dalton706a6ff2017-11-29 22:01:06 -0700182#ifdef SK_DEBUG
Greg Daniel849dce12018-04-24 14:32:53 -0400183void GrTextureProxy::onValidateSurface(const GrSurface* surface) {
Robert Phillipse8fabb22018-02-04 14:33:21 -0500184 SkASSERT(!surface->asRenderTarget());
185
186 // Anything that is checked here should be duplicated in GrTextureRenderTargetProxy's version
187 SkASSERT(surface->asTexture());
Chris Dalton95d8ceb2019-07-30 11:17:59 -0600188 // It is possible to fulfill a non-mipmapped proxy with a mipmapped texture.
Brian Salomonfd98c2c2018-07-31 17:25:29 -0400189 SkASSERT(GrMipMapped::kNo == this->proxyMipMapped() ||
Robert Phillips6ba15ec2018-02-08 16:26:47 -0500190 GrMipMapped::kYes == surface->asTexture()->texturePriv().mipMapped());
Brian Salomonc67c31c2018-12-06 10:00:03 -0500191
Greg Daniel4065d452018-11-16 15:43:41 -0500192 SkASSERT(surface->asTexture()->texturePriv().textureType() == this->textureType());
Brian Salomonc67c31c2018-12-06 10:00:03 -0500193
194 GrInternalSurfaceFlags proxyFlags = fSurfaceFlags;
195 GrInternalSurfaceFlags surfaceFlags = surface->surfacePriv().flags();
Chris Dalton3f7932e2019-08-19 00:39:13 -0600196 SkASSERT(((int)proxyFlags & kGrInternalTextureFlagsMask) ==
197 ((int)surfaceFlags & kGrInternalTextureFlagsMask));
Chris Dalton706a6ff2017-11-29 22:01:06 -0700198}
Brian Salomonc67c31c2018-12-06 10:00:03 -0500199
Chris Dalton706a6ff2017-11-29 22:01:06 -0700200#endif
201