| /* |
| * Copyright 2018 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "src/gpu/ganesh/GrProxyProvider.h" |
| |
| #include "include/core/SkBitmap.h" |
| #include "include/core/SkImage.h" |
| #include "include/core/SkTextureCompressionType.h" |
| #include "include/gpu/GrDirectContext.h" |
| #include "include/private/base/SingleOwner.h" |
| #include "include/private/gpu/ganesh/GrImageContext.h" |
| #include "src/core/SkAutoPixmapStorage.h" |
| #include "src/core/SkCompressedDataUtils.h" |
| #include "src/core/SkImageInfoPriv.h" |
| #include "src/core/SkImagePriv.h" |
| #include "src/core/SkMipmap.h" |
| #include "src/core/SkTraceEvent.h" |
| #include "src/gpu/ganesh/GrCaps.h" |
| #include "src/gpu/ganesh/GrContextThreadSafeProxyPriv.h" |
| #include "src/gpu/ganesh/GrDirectContextPriv.h" |
| #include "src/gpu/ganesh/GrImageContextPriv.h" |
| #include "src/gpu/ganesh/GrRenderTarget.h" |
| #include "src/gpu/ganesh/GrResourceProvider.h" |
| #include "src/gpu/ganesh/GrSurfaceProxy.h" |
| #include "src/gpu/ganesh/GrSurfaceProxyPriv.h" |
| #include "src/gpu/ganesh/GrTexture.h" |
| #include "src/gpu/ganesh/GrTextureProxyCacheAccess.h" |
| #include "src/gpu/ganesh/GrTextureRenderTargetProxy.h" |
| #include "src/gpu/ganesh/SkGr.h" |
| #include "src/image/SkImage_Base.h" |
| |
| #define ASSERT_SINGLE_OWNER SKGPU_ASSERT_SINGLE_OWNER(fImageContext->priv().singleOwner()) |
| |
| GrProxyProvider::GrProxyProvider(GrImageContext* imageContext) : fImageContext(imageContext) {} |
| |
| GrProxyProvider::~GrProxyProvider() { |
| if (this->renderingDirectly()) { |
| // In DDL-mode a proxy provider can still have extant uniquely keyed proxies (since |
| // they need their unique keys to, potentially, find a cached resource when the |
| // DDL is played) but, in non-DDL-mode they should all have been cleaned up by this point. |
| SkASSERT(!fUniquelyKeyedProxies.count()); |
| } |
| } |
| |
| bool GrProxyProvider::assignUniqueKeyToProxy(const skgpu::UniqueKey& key, GrTextureProxy* proxy) { |
| ASSERT_SINGLE_OWNER |
| SkASSERT(key.isValid()); |
| if (this->isAbandoned() || !proxy) { |
| return false; |
| } |
| |
| // Only the proxyProvider that created a proxy should be assigning unique keys to it. |
| SkASSERT(this->isDDLProvider() == proxy->creatingProvider()); |
| |
| #ifdef SK_DEBUG |
| { |
| auto direct = fImageContext->asDirectContext(); |
| if (direct) { |
| GrResourceCache* resourceCache = direct->priv().getResourceCache(); |
| // If there is already a GrResource with this key then the caller has violated the |
| // normal usage pattern of uniquely keyed resources (e.g., they have created one w/o |
| // first seeing if it already existed in the cache). |
| SkASSERT(!resourceCache->findAndRefUniqueResource(key)); |
| } |
| } |
| #endif |
| |
| SkASSERT(!fUniquelyKeyedProxies.find(key)); // multiple proxies can't get the same key |
| |
| proxy->cacheAccess().setUniqueKey(this, key); |
| SkASSERT(proxy->getUniqueKey() == key); |
| fUniquelyKeyedProxies.add(proxy); |
| return true; |
| } |
| |
| void GrProxyProvider::adoptUniqueKeyFromSurface(GrTextureProxy* proxy, const GrSurface* surf) { |
| SkASSERT(surf->getUniqueKey().isValid()); |
| proxy->cacheAccess().setUniqueKey(this, surf->getUniqueKey()); |
| SkASSERT(proxy->getUniqueKey() == surf->getUniqueKey()); |
| // multiple proxies can't get the same key |
| SkASSERT(!fUniquelyKeyedProxies.find(surf->getUniqueKey())); |
| fUniquelyKeyedProxies.add(proxy); |
| } |
| |
| void GrProxyProvider::removeUniqueKeyFromProxy(GrTextureProxy* proxy) { |
| ASSERT_SINGLE_OWNER |
| SkASSERT(proxy); |
| SkASSERT(proxy->getUniqueKey().isValid()); |
| |
| if (this->isAbandoned()) { |
| return; |
| } |
| |
| this->processInvalidUniqueKey(proxy->getUniqueKey(), proxy, InvalidateGPUResource::kYes); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::findProxyByUniqueKey(const skgpu::UniqueKey& key) { |
| ASSERT_SINGLE_OWNER |
| |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| GrTextureProxy* proxy = fUniquelyKeyedProxies.find(key); |
| if (proxy) { |
| return sk_ref_sp(proxy); |
| } |
| return nullptr; |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| #if defined(GR_TEST_UTILS) |
| sk_sp<GrTextureProxy> GrProxyProvider::testingOnly_createInstantiatedProxy( |
| SkISize dimensions, |
| const GrBackendFormat& format, |
| GrRenderable renderable, |
| int renderTargetSampleCnt, |
| SkBackingFit fit, |
| skgpu::Budgeted budgeted, |
| GrProtected isProtected) { |
| ASSERT_SINGLE_OWNER |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| auto direct = fImageContext->asDirectContext(); |
| if (!direct) { |
| return nullptr; |
| } |
| |
| if (this->caps()->isFormatCompressed(format)) { |
| // TODO: Allow this to go to GrResourceProvider::createCompressedTexture() once we no longer |
| // rely on GrColorType to get a swizzle for the proxy. |
| return nullptr; |
| } |
| |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| sk_sp<GrTexture> tex; |
| |
| if (SkBackingFit::kApprox == fit) { |
| tex = resourceProvider->createApproxTexture( |
| dimensions, |
| format, |
| format.textureType(), |
| renderable, |
| renderTargetSampleCnt, |
| isProtected, |
| /*label=*/"InstantiatedProxyViaApproxTexture_Test"); |
| } else { |
| tex = resourceProvider->createTexture(dimensions, |
| format, |
| format.textureType(), |
| renderable, |
| renderTargetSampleCnt, |
| skgpu::Mipmapped::kNo, |
| budgeted, |
| isProtected, |
| /*label=*/"InstantiatedProxyViaTexture_Test"); |
| } |
| if (!tex) { |
| return nullptr; |
| } |
| |
| return this->createWrapped(std::move(tex), UseAllocator::kYes); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::testingOnly_createInstantiatedProxy( |
| SkISize dimensions, |
| GrColorType colorType, |
| GrRenderable renderable, |
| int renderTargetSampleCnt, |
| SkBackingFit fit, |
| skgpu::Budgeted budgeted, |
| GrProtected isProtected) { |
| ASSERT_SINGLE_OWNER |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| auto format = this->caps()->getDefaultBackendFormat(colorType, renderable); |
| return this->testingOnly_createInstantiatedProxy(dimensions, |
| format, |
| renderable, |
| renderTargetSampleCnt, |
| fit, |
| budgeted, |
| isProtected); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::testingOnly_createWrapped(sk_sp<GrTexture> tex) { |
| return this->createWrapped(std::move(tex), UseAllocator::kYes); |
| } |
| #endif |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::createWrapped(sk_sp<GrTexture> tex, |
| UseAllocator useAllocator) { |
| #ifdef SK_DEBUG |
| if (tex->getUniqueKey().isValid()) { |
| SkASSERT(!this->findProxyByUniqueKey(tex->getUniqueKey())); |
| } |
| #endif |
| |
| if (tex->asRenderTarget()) { |
| return sk_sp<GrTextureProxy>(new GrTextureRenderTargetProxy( |
| std::move(tex), useAllocator, this->isDDLProvider())); |
| } else { |
| return sk_sp<GrTextureProxy>( |
| new GrTextureProxy(std::move(tex), useAllocator, this->isDDLProvider())); |
| } |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::findOrCreateProxyByUniqueKey(const skgpu::UniqueKey& key, |
| UseAllocator useAllocator) { |
| ASSERT_SINGLE_OWNER |
| |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| sk_sp<GrTextureProxy> result = this->findProxyByUniqueKey(key); |
| if (result) { |
| return result; |
| } |
| |
| auto direct = fImageContext->asDirectContext(); |
| if (!direct) { |
| return nullptr; |
| } |
| |
| GrResourceCache* resourceCache = direct->priv().getResourceCache(); |
| |
| GrGpuResource* resource = resourceCache->findAndRefUniqueResource(key); |
| if (!resource) { |
| return nullptr; |
| } |
| |
| sk_sp<GrTexture> texture(static_cast<GrSurface*>(resource)->asTexture()); |
| SkASSERT(texture); |
| |
| result = this->createWrapped(std::move(texture), useAllocator); |
| SkASSERT(result->getUniqueKey() == key); |
| // createWrapped should've added this for us |
| SkASSERT(fUniquelyKeyedProxies.find(key)); |
| return result; |
| } |
| |
| GrSurfaceProxyView GrProxyProvider::findCachedProxyWithColorTypeFallback( |
| const skgpu::UniqueKey& key, |
| GrSurfaceOrigin origin, |
| GrColorType ct, |
| int sampleCnt) { |
| auto proxy = this->findOrCreateProxyByUniqueKey(key); |
| if (!proxy) { |
| return {}; |
| } |
| const GrCaps* caps = fImageContext->priv().caps(); |
| |
| // Assume that we used a fallback color type if and only if the proxy is renderable. |
| if (proxy->asRenderTargetProxy()) { |
| GrBackendFormat expectedFormat; |
| std::tie(ct, expectedFormat) = caps->getFallbackColorTypeAndFormat(ct, sampleCnt); |
| SkASSERT(expectedFormat == proxy->backendFormat()); |
| } |
| skgpu::Swizzle swizzle = caps->getReadSwizzle(proxy->backendFormat(), ct); |
| return {std::move(proxy), origin, swizzle}; |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::createProxyFromBitmap(const SkBitmap& bitmap, |
| skgpu::Mipmapped mipmapped, |
| SkBackingFit fit, |
| skgpu::Budgeted budgeted) { |
| ASSERT_SINGLE_OWNER |
| SkASSERT(fit == SkBackingFit::kExact || mipmapped == skgpu::Mipmapped::kNo); |
| |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| if (!SkImageInfoIsValid(bitmap.info())) { |
| return nullptr; |
| } |
| |
| ATRACE_ANDROID_FRAMEWORK("Upload %sTexture [%dx%d]", |
| skgpu::Mipmapped::kYes == mipmapped ? "MipMap " : "", |
| bitmap.width(), |
| bitmap.height()); |
| |
| // In non-ddl we will always instantiate right away. Thus we never want to copy the SkBitmap |
| // even if its mutable. In ddl, if the bitmap is mutable then we must make a copy since the |
| // upload of the data to the gpu can happen at anytime and the bitmap may change by then. |
| SkBitmap copyBitmap = bitmap; |
| if (!this->renderingDirectly() && !bitmap.isImmutable()) { |
| copyBitmap.allocPixels(); |
| if (!bitmap.readPixels(copyBitmap.pixmap())) { |
| return nullptr; |
| } |
| if (mipmapped == skgpu::Mipmapped::kYes && bitmap.fMips) { |
| copyBitmap.fMips = sk_sp<SkMipmap>(SkMipmap::Build(copyBitmap.pixmap(), |
| /* factoryProc= */ nullptr, |
| /* computeContents= */ false)); |
| if (copyBitmap.fMips) { |
| for (int i = 0; i < copyBitmap.fMips->countLevels(); ++i) { |
| SkMipmap::Level src, dst; |
| bitmap.fMips->getLevel(i, &src); |
| copyBitmap.fMips->getLevel(i, &dst); |
| src.fPixmap.readPixels(dst.fPixmap); |
| } |
| } |
| } |
| copyBitmap.setImmutable(); |
| } |
| |
| sk_sp<GrTextureProxy> proxy; |
| if (mipmapped == skgpu::Mipmapped::kNo || |
| !SkMipmap::ComputeLevelCount(copyBitmap.dimensions())) { |
| proxy = this->createNonMippedProxyFromBitmap(copyBitmap, fit, budgeted); |
| } else { |
| proxy = this->createMippedProxyFromBitmap(copyBitmap, budgeted); |
| } |
| |
| if (!proxy) { |
| return nullptr; |
| } |
| |
| auto direct = fImageContext->asDirectContext(); |
| if (direct) { |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| |
| // In order to reuse code we always create a lazy proxy. When we aren't in DDL mode however |
| // we're better off instantiating the proxy immediately here. |
| if (!proxy->priv().doLazyInstantiation(resourceProvider)) { |
| return nullptr; |
| } |
| } |
| return proxy; |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::createNonMippedProxyFromBitmap(const SkBitmap& bitmap, |
| SkBackingFit fit, |
| skgpu::Budgeted budgeted) { |
| auto dims = bitmap.dimensions(); |
| |
| auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
| GrBackendFormat format = this->caps()->getDefaultBackendFormat(colorType, GrRenderable::kNo); |
| if (!format.isValid()) { |
| return nullptr; |
| } |
| |
| sk_sp<GrTextureProxy> proxy = this->createLazyProxy( |
| [bitmap](GrResourceProvider* resourceProvider, const LazySurfaceDesc& desc) { |
| SkASSERT(desc.fMipmapped == skgpu::Mipmapped::kNo); |
| GrMipLevel mipLevel = {bitmap.getPixels(), bitmap.rowBytes(), nullptr}; |
| auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
| return LazyCallbackResult(resourceProvider->createTexture(desc.fDimensions, |
| desc.fFormat, |
| desc.fTextureType, |
| colorType, |
| desc.fRenderable, |
| desc.fSampleCnt, |
| desc.fBudgeted, |
| desc.fFit, |
| desc.fProtected, |
| mipLevel, |
| desc.fLabel)); |
| }, |
| format, |
| dims, |
| skgpu::Mipmapped::kNo, |
| GrMipmapStatus::kNotAllocated, |
| GrInternalSurfaceFlags::kNone, |
| fit, |
| budgeted, |
| GrProtected::kNo, |
| UseAllocator::kYes, |
| "ProxyProvider_CreateNonMippedProxyFromBitmap"); |
| |
| if (!proxy) { |
| return nullptr; |
| } |
| SkASSERT(proxy->dimensions() == bitmap.dimensions()); |
| return proxy; |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::createMippedProxyFromBitmap(const SkBitmap& bitmap, |
| skgpu::Budgeted budgeted) { |
| SkASSERT(this->caps()->mipmapSupport()); |
| |
| auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
| GrBackendFormat format = this->caps()->getDefaultBackendFormat(colorType, GrRenderable::kNo); |
| if (!format.isValid()) { |
| return nullptr; |
| } |
| |
| sk_sp<SkMipmap> mipmaps = bitmap.fMips; |
| if (!mipmaps) { |
| mipmaps.reset(SkMipmap::Build(bitmap.pixmap(), /* factoryProc= */ nullptr)); |
| if (!mipmaps) { |
| return nullptr; |
| } |
| } |
| |
| auto dims = bitmap.dimensions(); |
| |
| sk_sp<GrTextureProxy> proxy = this->createLazyProxy( |
| [bitmap, mipmaps](GrResourceProvider* resourceProvider, const LazySurfaceDesc& desc) { |
| const int mipLevelCount = mipmaps->countLevels() + 1; |
| std::unique_ptr<GrMipLevel[]> texels(new GrMipLevel[mipLevelCount]); |
| auto colorType = SkColorTypeToGrColorType(bitmap.colorType()); |
| |
| texels[0].fPixels = bitmap.getPixels(); |
| texels[0].fRowBytes = bitmap.rowBytes(); |
| |
| for (int i = 1; i < mipLevelCount; ++i) { |
| SkMipmap::Level generatedMipLevel; |
| mipmaps->getLevel(i - 1, &generatedMipLevel); |
| texels[i].fPixels = generatedMipLevel.fPixmap.addr(); |
| texels[i].fRowBytes = generatedMipLevel.fPixmap.rowBytes(); |
| SkASSERT(texels[i].fPixels); |
| SkASSERT(generatedMipLevel.fPixmap.colorType() == bitmap.colorType()); |
| } |
| return LazyCallbackResult(resourceProvider->createTexture(desc.fDimensions, |
| desc.fFormat, |
| desc.fTextureType, |
| colorType, |
| GrRenderable::kNo, |
| 1, |
| desc.fBudgeted, |
| skgpu::Mipmapped::kYes, |
| GrProtected::kNo, |
| texels.get(), |
| desc.fLabel)); |
| }, |
| format, |
| dims, |
| skgpu::Mipmapped::kYes, |
| GrMipmapStatus::kValid, |
| GrInternalSurfaceFlags::kNone, |
| SkBackingFit::kExact, |
| budgeted, |
| GrProtected::kNo, |
| UseAllocator::kYes, |
| "ProxyProvider_CreateMippedProxyFromBitmap"); |
| |
| if (!proxy) { |
| return nullptr; |
| } |
| |
| SkASSERT(proxy->dimensions() == bitmap.dimensions()); |
| |
| return proxy; |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::createProxy(const GrBackendFormat& format, |
| SkISize dimensions, |
| GrRenderable renderable, |
| int renderTargetSampleCnt, |
| skgpu::Mipmapped mipmapped, |
| SkBackingFit fit, |
| skgpu::Budgeted budgeted, |
| GrProtected isProtected, |
| std::string_view label, |
| GrInternalSurfaceFlags surfaceFlags, |
| GrSurfaceProxy::UseAllocator useAllocator) { |
| ASSERT_SINGLE_OWNER |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| const GrCaps* caps = this->caps(); |
| |
| if (caps->isFormatCompressed(format)) { |
| // Deferred proxies for compressed textures are not supported. |
| return nullptr; |
| } |
| |
| if (skgpu::Mipmapped::kYes == mipmapped) { |
| // SkMipmap doesn't include the base level in the level count so we have to add 1 |
| int mipCount = SkMipmap::ComputeLevelCount(dimensions.fWidth, dimensions.fHeight) + 1; |
| if (1 == mipCount) { |
| mipmapped = skgpu::Mipmapped::kNo; |
| } |
| } |
| |
| if (!caps->validateSurfaceParams(dimensions, |
| format, |
| renderable, |
| renderTargetSampleCnt, |
| mipmapped, |
| GrTextureType::k2D)) { |
| return nullptr; |
| } |
| GrMipmapStatus mipmapStatus = (skgpu::Mipmapped::kYes == mipmapped) |
| ? GrMipmapStatus::kDirty |
| : GrMipmapStatus::kNotAllocated; |
| if (renderable == GrRenderable::kYes) { |
| renderTargetSampleCnt = caps->getRenderTargetSampleCount(renderTargetSampleCnt, format); |
| SkASSERT(renderTargetSampleCnt); |
| GrInternalSurfaceFlags extraFlags = caps->getExtraSurfaceFlagsForDeferredRT(); |
| // We know anything we instantiate later from this deferred path will be |
| // both texturable and renderable |
| return sk_sp<GrTextureProxy>(new GrTextureRenderTargetProxy(*caps, |
| format, |
| dimensions, |
| renderTargetSampleCnt, |
| mipmapped, |
| mipmapStatus, |
| fit, |
| budgeted, |
| isProtected, |
| surfaceFlags | extraFlags, |
| useAllocator, |
| this->isDDLProvider(), |
| label)); |
| } |
| |
| return sk_sp<GrTextureProxy>(new GrTextureProxy(format, |
| dimensions, |
| mipmapped, |
| mipmapStatus, |
| fit, |
| budgeted, |
| isProtected, |
| surfaceFlags, |
| useAllocator, |
| this->isDDLProvider(), |
| label)); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::createCompressedTextureProxy( |
| SkISize dimensions, |
| skgpu::Budgeted budgeted, |
| skgpu::Mipmapped mipmapped, |
| GrProtected isProtected, |
| SkTextureCompressionType compressionType, |
| sk_sp<SkData> data) { |
| ASSERT_SINGLE_OWNER |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| GrBackendFormat format = this->caps()->getBackendFormatFromCompressionType(compressionType); |
| |
| if (!this->caps()->isFormatTexturable(format, GrTextureType::k2D)) { |
| return nullptr; |
| } |
| |
| GrMipmapStatus mipmapStatus = (skgpu::Mipmapped::kYes == mipmapped) |
| ? GrMipmapStatus::kValid |
| : GrMipmapStatus::kNotAllocated; |
| |
| sk_sp<GrTextureProxy> proxy = this->createLazyProxy( |
| [data](GrResourceProvider* resourceProvider, const LazySurfaceDesc& desc) { |
| return LazyCallbackResult( |
| resourceProvider->createCompressedTexture(desc.fDimensions, |
| desc.fFormat, |
| desc.fBudgeted, |
| desc.fMipmapped, |
| desc.fProtected, |
| data.get(), |
| desc.fLabel)); |
| }, |
| format, |
| dimensions, |
| mipmapped, |
| mipmapStatus, |
| GrInternalSurfaceFlags::kReadOnly, |
| SkBackingFit::kExact, |
| skgpu::Budgeted::kYes, |
| GrProtected::kNo, |
| UseAllocator::kYes, |
| "ProxyProvider_CreateCompressedTextureProxy"); |
| |
| if (!proxy) { |
| return nullptr; |
| } |
| |
| auto direct = fImageContext->asDirectContext(); |
| if (direct) { |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| // In order to reuse code we always create a lazy proxy. When we aren't in DDL mode however |
| // we're better off instantiating the proxy immediately here. |
| if (!proxy->priv().doLazyInstantiation(resourceProvider)) { |
| return nullptr; |
| } |
| } |
| return proxy; |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::wrapBackendTexture( |
| const GrBackendTexture& backendTex, |
| GrWrapOwnership ownership, |
| GrWrapCacheable cacheable, |
| GrIOType ioType, |
| sk_sp<skgpu::RefCntedCallback> releaseHelper) { |
| SkASSERT(ioType != kWrite_GrIOType); |
| |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| // This is only supported on a direct GrContext. |
| auto direct = fImageContext->asDirectContext(); |
| if (!direct) { |
| return nullptr; |
| } |
| |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| |
| sk_sp<GrTexture> tex = |
| resourceProvider->wrapBackendTexture(backendTex, ownership, cacheable, ioType); |
| if (!tex) { |
| return nullptr; |
| } |
| |
| if (releaseHelper) { |
| tex->setRelease(std::move(releaseHelper)); |
| } |
| |
| SkASSERT(!tex->asRenderTarget()); // Strictly a GrTexture |
| // Make sure we match how we created the proxy with skgpu::Budgeted::kNo |
| SkASSERT(GrBudgetedType::kBudgeted != tex->resourcePriv().budgetedType()); |
| |
| return sk_sp<GrTextureProxy>( |
| new GrTextureProxy(std::move(tex), UseAllocator::kNo, this->isDDLProvider())); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::wrapCompressedBackendTexture( |
| const GrBackendTexture& beTex, |
| GrWrapOwnership ownership, |
| GrWrapCacheable cacheable, |
| sk_sp<skgpu::RefCntedCallback> releaseHelper) { |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| // This is only supported on a direct GrContext. |
| auto direct = fImageContext->asDirectContext(); |
| if (!direct) { |
| return nullptr; |
| } |
| |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| |
| sk_sp<GrTexture> tex = resourceProvider->wrapCompressedBackendTexture(beTex, ownership, |
| cacheable); |
| if (!tex) { |
| return nullptr; |
| } |
| |
| if (releaseHelper) { |
| tex->setRelease(std::move(releaseHelper)); |
| } |
| |
| SkASSERT(!tex->asRenderTarget()); // Strictly a GrTexture |
| // Make sure we match how we created the proxy with skgpu::Budgeted::kNo |
| SkASSERT(GrBudgetedType::kBudgeted != tex->resourcePriv().budgetedType()); |
| |
| return sk_sp<GrTextureProxy>( |
| new GrTextureProxy(std::move(tex), UseAllocator::kNo, this->isDDLProvider())); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::wrapRenderableBackendTexture( |
| const GrBackendTexture& backendTex, |
| int sampleCnt, |
| GrWrapOwnership ownership, |
| GrWrapCacheable cacheable, |
| sk_sp<skgpu::RefCntedCallback> releaseHelper) { |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| // This is only supported on a direct GrContext. |
| auto direct = fImageContext->asDirectContext(); |
| if (!direct) { |
| return nullptr; |
| } |
| |
| const GrCaps* caps = this->caps(); |
| |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| |
| sampleCnt = caps->getRenderTargetSampleCount(sampleCnt, backendTex.getBackendFormat()); |
| SkASSERT(sampleCnt); |
| |
| sk_sp<GrTexture> tex = resourceProvider->wrapRenderableBackendTexture( |
| backendTex, sampleCnt, ownership, cacheable); |
| if (!tex) { |
| return nullptr; |
| } |
| |
| if (releaseHelper) { |
| tex->setRelease(std::move(releaseHelper)); |
| } |
| |
| SkASSERT(tex->asRenderTarget()); // A GrTextureRenderTarget |
| // Make sure we match how we created the proxy with skgpu::Budgeted::kNo |
| SkASSERT(GrBudgetedType::kBudgeted != tex->resourcePriv().budgetedType()); |
| |
| return sk_sp<GrTextureProxy>(new GrTextureRenderTargetProxy( |
| std::move(tex), UseAllocator::kNo, this->isDDLProvider())); |
| } |
| |
| GrResourceProvider* GrProxyProvider::resourceProvider() const { |
| GrDirectContext* direct = fImageContext->asDirectContext(); |
| if (!direct) { |
| return nullptr; |
| } |
| |
| return direct->priv().resourceProvider(); |
| } |
| |
| sk_sp<GrSurfaceProxy> GrProxyProvider::wrapBackendRenderTarget( |
| const GrBackendRenderTarget& backendRT, |
| sk_sp<skgpu::RefCntedCallback> releaseHelper) { |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| |
| // This is only supported on a direct GrContext. |
| auto direct = fImageContext->asDirectContext(); |
| if (!direct) { |
| return nullptr; |
| } |
| |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| |
| sk_sp<GrRenderTarget> rt = resourceProvider->wrapBackendRenderTarget(backendRT); |
| if (!rt) { |
| return nullptr; |
| } |
| |
| if (releaseHelper) { |
| rt->setRelease(std::move(releaseHelper)); |
| } |
| |
| SkASSERT(!rt->asTexture()); // A GrRenderTarget that's not textureable |
| SkASSERT(!rt->getUniqueKey().isValid()); |
| // Make sure we match how we created the proxy with skgpu::Budgeted::kNo |
| SkASSERT(GrBudgetedType::kBudgeted != rt->resourcePriv().budgetedType()); |
| |
| return sk_sp<GrRenderTargetProxy>( |
| new GrRenderTargetProxy(std::move(rt), UseAllocator::kNo, {})); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::CreatePromiseProxy(GrContextThreadSafeProxy* threadSafeProxy, |
| LazyInstantiateCallback&& callback, |
| const GrBackendFormat& format, |
| SkISize dimensions, |
| skgpu::Mipmapped mipmapped) { |
| if (threadSafeProxy->priv().abandoned()) { |
| return nullptr; |
| } |
| SkASSERT((dimensions.fWidth <= 0 && dimensions.fHeight <= 0) || |
| (dimensions.fWidth > 0 && dimensions.fHeight > 0)); |
| |
| if (dimensions.fWidth > threadSafeProxy->priv().caps()->maxTextureSize() || |
| dimensions.fHeight > threadSafeProxy->priv().caps()->maxTextureSize()) { |
| return nullptr; |
| } |
| if (!threadSafeProxy->priv().caps()->isFormatTexturable(format, format.textureType())) { |
| return nullptr; |
| } |
| // Ganesh assumes that, when wrapping a mipmapped backend texture from a client, that its |
| // mipmaps are fully fleshed out. |
| GrMipmapStatus mipmapStatus = (skgpu::Mipmapped::kYes == mipmapped) |
| ? GrMipmapStatus::kValid |
| : GrMipmapStatus::kNotAllocated; |
| |
| // We pass kReadOnly here since we should treat content of the client's texture as immutable. |
| // The promise API provides no way for the client to indicate that the texture is protected. |
| auto proxy = sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(callback), |
| format, |
| dimensions, |
| mipmapped, |
| mipmapStatus, |
| SkBackingFit::kExact, |
| skgpu::Budgeted::kNo, |
| GrProtected::kNo, |
| GrInternalSurfaceFlags::kReadOnly, |
| GrSurfaceProxy::UseAllocator::kYes, |
| GrDDLProvider::kYes, |
| /*label=*/"PromiseProxy")); |
| proxy->priv().setIsPromiseProxy(); |
| return proxy; |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::createLazyProxy(LazyInstantiateCallback&& callback, |
| const GrBackendFormat& format, |
| SkISize dimensions, |
| skgpu::Mipmapped mipmapped, |
| GrMipmapStatus mipmapStatus, |
| GrInternalSurfaceFlags surfaceFlags, |
| SkBackingFit fit, |
| skgpu::Budgeted budgeted, |
| GrProtected isProtected, |
| GrSurfaceProxy::UseAllocator useAllocator, |
| std::string_view label) { |
| ASSERT_SINGLE_OWNER |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| SkASSERT((dimensions.fWidth <= 0 && dimensions.fHeight <= 0) || |
| (dimensions.fWidth > 0 && dimensions.fHeight > 0)); |
| |
| if (!format.isValid() || format.backend() != fImageContext->backend()) { |
| return nullptr; |
| } |
| |
| if (dimensions.fWidth > this->caps()->maxTextureSize() || |
| dimensions.fHeight > this->caps()->maxTextureSize()) { |
| return nullptr; |
| } |
| |
| return sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(callback), |
| format, |
| dimensions, |
| mipmapped, |
| mipmapStatus, |
| fit, |
| budgeted, |
| isProtected, |
| surfaceFlags, |
| useAllocator, |
| this->isDDLProvider(), |
| label)); |
| } |
| |
| sk_sp<GrRenderTargetProxy> GrProxyProvider::createLazyRenderTargetProxy( |
| LazyInstantiateCallback&& callback, |
| const GrBackendFormat& format, |
| SkISize dimensions, |
| int sampleCnt, |
| GrInternalSurfaceFlags surfaceFlags, |
| const TextureInfo* textureInfo, |
| GrMipmapStatus mipmapStatus, |
| SkBackingFit fit, |
| skgpu::Budgeted budgeted, |
| GrProtected isProtected, |
| bool wrapsVkSecondaryCB, |
| UseAllocator useAllocator) { |
| ASSERT_SINGLE_OWNER |
| if (this->isAbandoned()) { |
| return nullptr; |
| } |
| SkASSERT((dimensions.fWidth <= 0 && dimensions.fHeight <= 0) || |
| (dimensions.fWidth > 0 && dimensions.fHeight > 0)); |
| |
| if (dimensions.fWidth > this->caps()->maxRenderTargetSize() || |
| dimensions.fHeight > this->caps()->maxRenderTargetSize()) { |
| return nullptr; |
| } |
| |
| if (textureInfo) { |
| // Wrapped vulkan secondary command buffers don't support texturing since we won't have an |
| // actual VkImage to texture from. |
| SkASSERT(!wrapsVkSecondaryCB); |
| return sk_sp<GrRenderTargetProxy>(new GrTextureRenderTargetProxy( |
| *this->caps(), |
| std::move(callback), |
| format, |
| dimensions, |
| sampleCnt, |
| textureInfo->fMipmapped, |
| mipmapStatus, |
| fit, |
| budgeted, |
| isProtected, |
| surfaceFlags, |
| useAllocator, |
| this->isDDLProvider(), |
| /*label=*/"TextureRenderTarget_LazyRenderTargetProxy")); |
| } |
| |
| GrRenderTargetProxy::WrapsVkSecondaryCB vkSCB = |
| wrapsVkSecondaryCB ? GrRenderTargetProxy::WrapsVkSecondaryCB::kYes |
| : GrRenderTargetProxy::WrapsVkSecondaryCB::kNo; |
| |
| return sk_sp<GrRenderTargetProxy>( |
| new GrRenderTargetProxy(std::move(callback), |
| format, |
| dimensions, |
| sampleCnt, |
| fit, |
| budgeted, |
| isProtected, |
| surfaceFlags, |
| useAllocator, |
| vkSCB, |
| /*label=*/"RenderTargetProxy_LazyRenderTargetProxy")); |
| } |
| |
| sk_sp<GrTextureProxy> GrProxyProvider::MakeFullyLazyProxy(LazyInstantiateCallback&& callback, |
| const GrBackendFormat& format, |
| GrRenderable renderable, |
| int renderTargetSampleCnt, |
| GrProtected isProtected, |
| const GrCaps& caps, |
| UseAllocator useAllocator) { |
| if (!format.isValid()) { |
| return nullptr; |
| } |
| |
| SkASSERT(renderTargetSampleCnt == 1 || renderable == GrRenderable::kYes); |
| // TODO: If we ever have callers requesting specific surface flags then we shouldn't use the |
| // extra deferred flags here. Instead those callers should all pass in exactly what they want. |
| // However, as of today all uses of this essentially create a deferred proxy in the end. |
| GrInternalSurfaceFlags surfaceFlags = caps.getExtraSurfaceFlagsForDeferredRT(); |
| |
| // MakeFullyLazyProxy is only called at flush time so we know these texture proxies are |
| // not being created by a DDL provider. |
| static constexpr SkISize kLazyDims = {-1, -1}; |
| if (GrRenderable::kYes == renderable) { |
| return sk_sp<GrTextureProxy>( |
| new GrTextureRenderTargetProxy(caps, |
| std::move(callback), |
| format, |
| kLazyDims, |
| renderTargetSampleCnt, |
| skgpu::Mipmapped::kNo, |
| GrMipmapStatus::kNotAllocated, |
| SkBackingFit::kApprox, |
| skgpu::Budgeted::kYes, |
| isProtected, |
| surfaceFlags, |
| useAllocator, |
| GrDDLProvider::kNo, |
| /*label=*/"TextureRenderTarget_FullyLazyProxy")); |
| } else { |
| return sk_sp<GrTextureProxy>(new GrTextureProxy(std::move(callback), |
| format, |
| kLazyDims, |
| skgpu::Mipmapped::kNo, |
| GrMipmapStatus::kNotAllocated, |
| SkBackingFit::kApprox, |
| skgpu::Budgeted::kYes, |
| isProtected, |
| surfaceFlags, |
| useAllocator, |
| GrDDLProvider::kNo, |
| /*label=*/"Texture_FullyLazyProxy")); |
| } |
| } |
| |
| void GrProxyProvider::processInvalidUniqueKey(const skgpu::UniqueKey& key, |
| GrTextureProxy* proxy, |
| InvalidateGPUResource invalidateGPUResource) { |
| this->processInvalidUniqueKeyImpl(key, proxy, invalidateGPUResource, RemoveTableEntry::kYes); |
| } |
| |
| void GrProxyProvider::processInvalidUniqueKeyImpl(const skgpu::UniqueKey& key, |
| GrTextureProxy* proxy, |
| InvalidateGPUResource invalidateGPUResource, |
| RemoveTableEntry removeTableEntry) { |
| SkASSERT(key.isValid()); |
| |
| if (!proxy) { |
| proxy = fUniquelyKeyedProxies.find(key); |
| } |
| SkASSERT(!proxy || proxy->getUniqueKey() == key); |
| |
| // Locate the corresponding GrGpuResource (if it needs to be invalidated) before clearing the |
| // proxy's unique key. We must do it in this order because 'key' may alias the proxy's key. |
| sk_sp<GrGpuResource> invalidGpuResource; |
| if (InvalidateGPUResource::kYes == invalidateGPUResource) { |
| auto direct = fImageContext->asDirectContext(); |
| if (direct) { |
| GrResourceProvider* resourceProvider = direct->priv().resourceProvider(); |
| invalidGpuResource = resourceProvider->findByUniqueKey<GrGpuResource>(key); |
| } |
| SkASSERT(!invalidGpuResource || invalidGpuResource->getUniqueKey() == key); |
| } |
| |
| // Note: this method is called for the whole variety of GrGpuResources so often 'key' |
| // will not be in 'fUniquelyKeyedProxies'. |
| if (proxy) { |
| if (removeTableEntry == RemoveTableEntry::kYes) { |
| fUniquelyKeyedProxies.remove(key); |
| } |
| proxy->cacheAccess().clearUniqueKey(); |
| } |
| |
| if (invalidGpuResource) { |
| invalidGpuResource->resourcePriv().removeUniqueKey(); |
| } |
| } |
| |
| GrDDLProvider GrProxyProvider::isDDLProvider() const { |
| return fImageContext->asDirectContext() ? GrDDLProvider::kNo : GrDDLProvider::kYes; |
| } |
| |
| uint32_t GrProxyProvider::contextID() const { |
| return fImageContext->priv().contextID(); |
| } |
| |
| const GrCaps* GrProxyProvider::caps() const { |
| return fImageContext->priv().caps(); |
| } |
| |
| sk_sp<const GrCaps> GrProxyProvider::refCaps() const { |
| return fImageContext->priv().refCaps(); |
| } |
| |
| bool GrProxyProvider::isAbandoned() const { |
| return fImageContext->priv().abandoned(); |
| } |
| |
| void GrProxyProvider::orphanAllUniqueKeys() { |
| fUniquelyKeyedProxies.foreach([&](GrTextureProxy* proxy){ |
| proxy->fProxyProvider = nullptr; |
| }); |
| } |
| |
| void GrProxyProvider::removeAllUniqueKeys() { |
| fUniquelyKeyedProxies.foreach([&](GrTextureProxy* proxy){ |
| // It's not safe to remove table entries while iterating with foreach(), |
| // but since we're going to remove them all anyway, simply save that for the end. |
| this->processInvalidUniqueKeyImpl(proxy->getUniqueKey(), proxy, |
| InvalidateGPUResource::kNo, |
| RemoveTableEntry::kNo); |
| }); |
| // Removing all those table entries is safe now. |
| fUniquelyKeyedProxies.reset(); |
| } |
| |
| bool GrProxyProvider::renderingDirectly() const { |
| return fImageContext->asDirectContext(); |
| } |