/* * Copyright 2016 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ // This is a GPU-backend specific test. #include "Test.h" #if SK_SUPPORT_GPU #include "GrContextPriv.h" #include "GrGpuResourceRef.h" #include "GrProxyProvider.h" #include "GrRenderTargetProxy.h" #include "GrResourceProvider.h" #include "GrSurfaceProxy.h" #include "GrTexture.h" #include "GrTextureProxy.h" int32_t GrIORefProxy::getProxyRefCnt_TestOnly() const { return fRefCnt; } int32_t GrIORefProxy::getBackingRefCnt_TestOnly() const { if (fTarget) { return fTarget->fRefCnt; } return fRefCnt; } int32_t GrIORefProxy::getPendingReadCnt_TestOnly() const { if (fTarget) { return fTarget->fPendingReads; } return fPendingReads; } int32_t GrIORefProxy::getPendingWriteCnt_TestOnly() const { if (fTarget) { return fTarget->fPendingWrites; } return fPendingWrites; } static const int kWidthHeight = 128; static void check_refs(skiatest::Reporter* reporter, GrTextureProxy* proxy, int32_t expectedProxyRefs, int32_t expectedBackingRefs, int32_t expectedNumReads, int32_t expectedNumWrites) { REPORTER_ASSERT(reporter, proxy->getProxyRefCnt_TestOnly() == expectedProxyRefs); REPORTER_ASSERT(reporter, proxy->getBackingRefCnt_TestOnly() == expectedBackingRefs); REPORTER_ASSERT(reporter, proxy->getPendingReadCnt_TestOnly() == expectedNumReads); REPORTER_ASSERT(reporter, proxy->getPendingWriteCnt_TestOnly() == expectedNumWrites); SkASSERT(proxy->getProxyRefCnt_TestOnly() == expectedProxyRefs); SkASSERT(proxy->getBackingRefCnt_TestOnly() == expectedBackingRefs); SkASSERT(proxy->getPendingReadCnt_TestOnly() == expectedNumReads); SkASSERT(proxy->getPendingWriteCnt_TestOnly() == expectedNumWrites); } static sk_sp<GrTextureProxy> make_deferred(GrProxyProvider* proxyProvider) { GrSurfaceDesc desc; desc.fFlags = kRenderTarget_GrSurfaceFlag; desc.fOrigin = kBottomLeft_GrSurfaceOrigin; desc.fWidth = kWidthHeight; desc.fHeight = kWidthHeight; desc.fConfig = kRGBA_8888_GrPixelConfig; return proxyProvider->createProxy(desc, SkBackingFit::kApprox, SkBudgeted::kYes, GrResourceProvider::kNoPendingIO_Flag); } static sk_sp<GrTextureProxy> make_wrapped(GrProxyProvider* proxyProvider) { GrSurfaceDesc desc; desc.fFlags = kRenderTarget_GrSurfaceFlag; desc.fOrigin = kBottomLeft_GrSurfaceOrigin; desc.fWidth = kWidthHeight; desc.fHeight = kWidthHeight; desc.fConfig = kRGBA_8888_GrPixelConfig; return proxyProvider->createInstantiatedProxy(desc, SkBackingFit::kExact, SkBudgeted::kNo); } DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ProxyRefTest, reporter, ctxInfo) { GrProxyProvider* proxyProvider = ctxInfo.grContext()->contextPriv().proxyProvider(); GrResourceProvider* resourceProvider = ctxInfo.grContext()->contextPriv().resourceProvider(); for (auto make : { make_deferred, make_wrapped }) { // A single write { sk_sp<GrTextureProxy> proxy((*make)(proxyProvider)); if (proxy.get()) { GrPendingIOResource<GrSurfaceProxy, kWrite_GrIOType> fWrite(proxy.get()); static const int kExpectedReads = 0; static const int kExpectedWrites = 1; check_refs(reporter, proxy.get(), 1, 1, kExpectedReads, kExpectedWrites); proxy->instantiate(resourceProvider); // In the deferred case, this checks that the refs transfered to the GrSurface check_refs(reporter, proxy.get(), 1, 1, kExpectedReads, kExpectedWrites); } } // A single read { sk_sp<GrTextureProxy> proxy((*make)(proxyProvider)); if (proxy.get()) { GrPendingIOResource<GrSurfaceProxy, kRead_GrIOType> fRead(proxy.get()); static const int kExpectedReads = 1; static const int kExpectedWrites = 0; check_refs(reporter, proxy.get(), 1, 1, kExpectedReads, kExpectedWrites); proxy->instantiate(resourceProvider); // In the deferred case, this checks that the refs transfered to the GrSurface check_refs(reporter, proxy.get(), 1, 1, kExpectedReads, kExpectedWrites); } } // A single read/write pair { sk_sp<GrTextureProxy> proxy((*make)(proxyProvider)); if (proxy.get()) { GrPendingIOResource<GrSurfaceProxy, kRW_GrIOType> fRW(proxy.get()); static const int kExpectedReads = 1; static const int kExpectedWrites = 1; check_refs(reporter, proxy.get(), 1, 1, kExpectedReads, kExpectedWrites); proxy->instantiate(resourceProvider); // In the deferred case, this checks that the refs transferred to the GrSurface check_refs(reporter, proxy.get(), 1, 1, kExpectedReads, kExpectedWrites); } } // Multiple normal refs { sk_sp<GrTextureProxy> proxy((*make)(proxyProvider)); if (proxy.get()) { proxy->ref(); proxy->ref(); static const int kExpectedReads = 0; static const int kExpectedWrites = 0; check_refs(reporter, proxy.get(), 3, 3,kExpectedReads, kExpectedWrites); proxy->instantiate(resourceProvider); // In the deferred case, this checks that the refs transferred to the GrSurface check_refs(reporter, proxy.get(), 3, 3, kExpectedReads, kExpectedWrites); proxy->unref(); proxy->unref(); } } // Continue using (reffing) proxy after instantiation { sk_sp<GrTextureProxy> proxy((*make)(proxyProvider)); if (proxy.get()) { proxy->ref(); GrPendingIOResource<GrSurfaceProxy, kWrite_GrIOType> fWrite(proxy.get()); static const int kExpectedWrites = 1; check_refs(reporter, proxy.get(), 2, 2, 0, kExpectedWrites); proxy->instantiate(resourceProvider); // In the deferred case, this checks that the refs transfered to the GrSurface check_refs(reporter, proxy.get(), 2, 2, 0, kExpectedWrites); proxy->unref(); check_refs(reporter, proxy.get(), 1, 1, 0, kExpectedWrites); GrPendingIOResource<GrSurfaceProxy, kRead_GrIOType> fRead(proxy.get()); check_refs(reporter, proxy.get(), 1, 1, 1, kExpectedWrites); } } } } #endif