/* * Copyright 2016 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "GrTextureProxy.h" #include "GrTextureProxyPriv.h" #include "GrContext.h" #include "GrContextPriv.h" #include "GrDeferredProxyUploader.h" #include "GrProxyProvider.h" #include "GrSurfacePriv.h" #include "GrTexturePriv.h" // Deferred version - with data GrTextureProxy::GrTextureProxy(const GrSurfaceDesc& srcDesc, GrMipMapped mipMapped, GrTextureType textureType, SkBackingFit fit, SkBudgeted budgeted, const void* srcData, size_t /*rowBytes*/, GrInternalSurfaceFlags surfaceFlags) : INHERITED(srcDesc, kTopLeft_GrSurfaceOrigin, fit, budgeted, surfaceFlags) , fMipMapped(mipMapped) , fTextureType(textureType) , fProxyProvider(nullptr) , fDeferredUploader(nullptr) { SkASSERT(!srcData); // currently handled in Make() } // Deferred version - no data GrTextureProxy::GrTextureProxy(const GrSurfaceDesc& srcDesc, GrSurfaceOrigin origin, GrMipMapped mipMapped, GrTextureType textureType, SkBackingFit fit, SkBudgeted budgeted, GrInternalSurfaceFlags surfaceFlags) : INHERITED(srcDesc, origin, fit, budgeted, surfaceFlags) , fMipMapped(mipMapped) , fTextureType(textureType) , fProxyProvider(nullptr) , fDeferredUploader(nullptr) {} // Lazy-callback version GrTextureProxy::GrTextureProxy(LazyInstantiateCallback&& callback, LazyInstantiationType lazyType, const GrSurfaceDesc& desc, GrSurfaceOrigin origin, GrMipMapped mipMapped, GrTextureType textureType, SkBackingFit fit, SkBudgeted budgeted, GrInternalSurfaceFlags surfaceFlags) : INHERITED(std::move(callback), lazyType, desc, origin, fit, budgeted, surfaceFlags) , fMipMapped(mipMapped) , fTextureType(textureType) , fProxyProvider(nullptr) , fDeferredUploader(nullptr) {} // Wrapped version GrTextureProxy::GrTextureProxy(sk_sp surf, GrSurfaceOrigin origin) : INHERITED(std::move(surf), origin, SkBackingFit::kExact) , fMipMapped(fTarget->asTexture()->texturePriv().mipMapped()) , fTextureType(fTarget->asTexture()->texturePriv().textureType()) , fProxyProvider(nullptr) , fDeferredUploader(nullptr) { if (fTarget->getUniqueKey().isValid()) { fProxyProvider = fTarget->asTexture()->getContext()->contextPriv().proxyProvider(); fProxyProvider->adoptUniqueKeyFromSurface(this, fTarget); } } GrTextureProxy::~GrTextureProxy() { // Due to the order of cleanup the GrSurface this proxy may have wrapped may have gone away // at this point. Zero out the pointer so the cache invalidation code doesn't try to use it. fTarget = nullptr; if (fUniqueKey.isValid()) { fProxyProvider->processInvalidProxyUniqueKey(fUniqueKey, this, false); } else { SkASSERT(!fProxyProvider); } } bool GrTextureProxy::instantiate(GrResourceProvider* resourceProvider) { if (LazyState::kNot != this->lazyInstantiationState()) { return false; } if (!this->instantiateImpl(resourceProvider, 1, /* needsStencil = */ false, kNone_GrSurfaceFlags, fMipMapped, fUniqueKey.isValid() ? &fUniqueKey : nullptr)) { return false; } SkASSERT(!fTarget->asRenderTarget()); SkASSERT(fTarget->asTexture()); return true; } sk_sp GrTextureProxy::createSurface(GrResourceProvider* resourceProvider) const { sk_sp surface= this->createSurfaceImpl(resourceProvider, 1, /* needsStencil = */ false, kNone_GrSurfaceFlags, fMipMapped); if (!surface) { return nullptr; } SkASSERT(!surface->asRenderTarget()); SkASSERT(surface->asTexture()); return surface; } void GrTextureProxyPriv::setDeferredUploader(std::unique_ptr uploader) { SkASSERT(!fTextureProxy->fDeferredUploader); fTextureProxy->fDeferredUploader = std::move(uploader); } void GrTextureProxyPriv::scheduleUpload(GrOpFlushState* flushState) { // The texture proxy's contents may already have been uploaded or instantiation may have failed if (fTextureProxy->fDeferredUploader && fTextureProxy->fTarget) { fTextureProxy->fDeferredUploader->scheduleUpload(flushState, fTextureProxy); } } void GrTextureProxyPriv::resetDeferredUploader() { SkASSERT(fTextureProxy->fDeferredUploader); fTextureProxy->fDeferredUploader.reset(); } // This method parallels the highest_filter_mode functions in GrGLTexture & GrVkTexture. GrSamplerState::Filter GrTextureProxy::highestFilterMode() const { if (fTarget) { return fTarget->asTexture()->texturePriv().highestFilterMode(); } // In OpenGL, GR_GL_TEXTURE_RECTANGLE and GR_GL_TEXTURE_EXTERNAL (which have a highest filter // mode of bilerp) can only be created via wrapping. return GrSamplerState::Filter::kMipMap; } GrMipMapped GrTextureProxy::mipMapped() const { if (this->priv().isInstantiated()) { return this->priv().peekTexture()->texturePriv().mipMapped(); } return fMipMapped; } size_t GrTextureProxy::onUninstantiatedGpuMemorySize() const { return GrSurface::ComputeSize(this->config(), this->width(), this->height(), 1, this->texPriv().proxyMipMapped(), !this->priv().isExact()); } void GrTextureProxy::setUniqueKey(GrProxyProvider* proxyProvider, const GrUniqueKey& key) { SkASSERT(key.isValid()); SkASSERT(!fUniqueKey.isValid()); // proxies can only ever get one uniqueKey if (fTarget) { if (!fTarget->getUniqueKey().isValid()) { fTarget->resourcePriv().setUniqueKey(key); } SkASSERT(fTarget->getUniqueKey() == key); } fUniqueKey = key; fProxyProvider = proxyProvider; } void GrTextureProxy::clearUniqueKey() { fUniqueKey.reset(); fProxyProvider = nullptr; } #ifdef SK_DEBUG void GrTextureProxy::onValidateSurface(const GrSurface* surface) { SkASSERT(!surface->asRenderTarget()); // Anything that is checked here should be duplicated in GrTextureRenderTargetProxy's version SkASSERT(surface->asTexture()); SkASSERT(GrMipMapped::kNo == this->texPriv().proxyMipMapped() || GrMipMapped::kYes == surface->asTexture()->texturePriv().mipMapped()); SkASSERT(surface->asTexture()->texturePriv().textureType() == fTextureType); } #endif