1 /* 2 * Copyright 2017 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 8 #include "Test.h" 9 10 #if SK_SUPPORT_GPU 11 12 #include "GrClip.h" 13 #include "GrContextPriv.h" 14 #include "GrProxyProvider.h" 15 #include "GrOnFlushResourceProvider.h" 16 #include "GrRenderTargetContext.h" 17 #include "GrRenderTargetContextPriv.h" 18 #include "GrSurfaceProxy.h" 19 #include "GrSurfaceProxyPriv.h" 20 #include "GrTexture.h" 21 #include "GrTextureProxy.h" 22 #include "GrTextureProxyPriv.h" 23 #include "SkMakeUnique.h" 24 #include "SkRectPriv.h" 25 #include "mock/GrMockTypes.h" 26 27 // This test verifies that lazy proxy callbacks get invoked during flush, after onFlush callbacks, 28 // but before Ops are executed. It also ensures that lazy proxy callbacks are invoked both for 29 // regular Ops and for clips. 30 class LazyProxyTest final : public GrOnFlushCallbackObject { 31 public: LazyProxyTest(skiatest::Reporter * reporter)32 LazyProxyTest(skiatest::Reporter* reporter) 33 : fReporter(reporter) 34 , fHasOpTexture(false) 35 , fHasClipTexture(false) { 36 } 37 ~LazyProxyTest()38 ~LazyProxyTest() override { 39 REPORTER_ASSERT(fReporter, fHasOpTexture); 40 REPORTER_ASSERT(fReporter, fHasClipTexture); 41 } 42 preFlush(GrOnFlushResourceProvider *,const uint32_t *,int,SkTArray<sk_sp<GrRenderTargetContext>> *)43 void preFlush(GrOnFlushResourceProvider*, const uint32_t*, int, 44 SkTArray<sk_sp<GrRenderTargetContext>>*) override { 45 REPORTER_ASSERT(fReporter, !fHasOpTexture); 46 REPORTER_ASSERT(fReporter, !fHasClipTexture); 47 } 48 postFlush(GrDeferredUploadToken,const uint32_t * opListIDs,int numOpListIDs)49 void postFlush(GrDeferredUploadToken, const uint32_t* opListIDs, int numOpListIDs) override { 50 REPORTER_ASSERT(fReporter, fHasOpTexture); 51 REPORTER_ASSERT(fReporter, fHasClipTexture); 52 } 53 54 class Op final : public GrDrawOp { 55 public: 56 DEFINE_OP_CLASS_ID 57 Op(GrProxyProvider * proxyProvider,LazyProxyTest * test,bool nullTexture)58 Op(GrProxyProvider* proxyProvider, LazyProxyTest* test, bool nullTexture) 59 : GrDrawOp(ClassID()), fTest(test) { 60 fProxy = proxyProvider->createFullyLazyProxy([this, nullTexture]( 61 GrResourceProvider* rp, GrSurfaceOrigin* origin) { 62 if (!rp) { 63 return sk_sp<GrTexture>(); 64 } 65 REPORTER_ASSERT(fTest->fReporter, !fTest->fHasOpTexture); 66 fTest->fHasOpTexture = true; 67 *origin = kTopLeft_GrSurfaceOrigin; 68 if (nullTexture) { 69 return sk_sp<GrTexture>(); 70 } else { 71 GrSurfaceDesc desc; 72 desc.fWidth = 1234; 73 desc.fHeight = 567; 74 desc.fOrigin = kTopLeft_GrSurfaceOrigin; 75 desc.fConfig = kRGB_565_GrPixelConfig; 76 sk_sp<GrTexture> texture = rp->createTexture(desc, SkBudgeted::kYes); 77 REPORTER_ASSERT(fTest->fReporter, texture); 78 return texture; 79 } 80 }, GrProxyProvider::Renderable::kNo, kRGB_565_GrPixelConfig); 81 this->setBounds(SkRectPriv::MakeLargest(), GrOp::HasAABloat::kNo, GrOp::IsZeroArea::kNo); 82 } 83 visitProxies(const VisitProxyFunc & func) const84 void visitProxies(const VisitProxyFunc& func) const override { 85 func(fProxy.get()); 86 } 87 onExecute(GrOpFlushState *)88 void onExecute(GrOpFlushState*) override { 89 REPORTER_ASSERT(fTest->fReporter, fTest->fHasOpTexture); 90 REPORTER_ASSERT(fTest->fReporter, fTest->fHasClipTexture); 91 } 92 93 private: name() const94 const char* name() const override { return "LazyProxyTest::Op"; } fixedFunctionFlags() const95 FixedFunctionFlags fixedFunctionFlags() const override { return FixedFunctionFlags::kNone; } finalize(const GrCaps &,const GrAppliedClip *,GrPixelConfigIsClamped)96 RequiresDstTexture finalize(const GrCaps&, const GrAppliedClip*, 97 GrPixelConfigIsClamped) override { 98 return RequiresDstTexture::kNo; 99 } wasRecorded(GrRenderTargetOpList *)100 void wasRecorded(GrRenderTargetOpList*) override {} onCombineIfPossible(GrOp * other,const GrCaps & caps)101 bool onCombineIfPossible(GrOp* other, const GrCaps& caps) override { return false; } onPrepare(GrOpFlushState *)102 void onPrepare(GrOpFlushState*) override {} 103 104 LazyProxyTest* const fTest; 105 sk_sp<GrTextureProxy> fProxy; 106 }; 107 108 class ClipFP : public GrFragmentProcessor { 109 public: ClipFP(GrProxyProvider * proxyProvider,LazyProxyTest * test,GrTextureProxy * atlas)110 ClipFP(GrProxyProvider* proxyProvider, LazyProxyTest* test, GrTextureProxy* atlas) 111 : GrFragmentProcessor(kTestFP_ClassID, kNone_OptimizationFlags) 112 , fProxyProvider(proxyProvider) 113 , fTest(test) 114 , fAtlas(atlas) { 115 fLazyProxy = proxyProvider->createFullyLazyProxy([this](GrResourceProvider* rp, 116 GrSurfaceOrigin* origin) { 117 if (!rp) { 118 return sk_sp<GrTexture>(); 119 } 120 REPORTER_ASSERT(fTest->fReporter, !fTest->fHasClipTexture); 121 fTest->fHasClipTexture = true; 122 *origin = kBottomLeft_GrSurfaceOrigin; 123 fAtlas->instantiate(rp); 124 return sk_ref_sp(fAtlas->priv().peekTexture()); 125 }, GrProxyProvider::Renderable::kYes, kAlpha_half_GrPixelConfig); 126 fAccess.reset(fLazyProxy, GrSamplerState::Filter::kNearest, 127 GrSamplerState::WrapMode::kClamp, kFragment_GrShaderFlag); 128 this->addTextureSampler(&fAccess); 129 } 130 131 private: name() const132 const char* name() const override { return "LazyProxyTest::ClipFP"; } clone() const133 std::unique_ptr<GrFragmentProcessor> clone() const override { 134 return skstd::make_unique<ClipFP>(fProxyProvider, fTest, fAtlas); 135 } onCreateGLSLInstance() const136 GrGLSLFragmentProcessor* onCreateGLSLInstance() const override { return nullptr; } onGetGLSLProcessorKey(const GrShaderCaps &,GrProcessorKeyBuilder *) const137 void onGetGLSLProcessorKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override {} onIsEqual(const GrFragmentProcessor &) const138 bool onIsEqual(const GrFragmentProcessor&) const override { return false; } 139 140 GrProxyProvider* const fProxyProvider; 141 LazyProxyTest* const fTest; 142 GrTextureProxy* const fAtlas; 143 sk_sp<GrTextureProxy> fLazyProxy; 144 TextureSampler fAccess; 145 }; 146 147 148 class Clip : public GrClip { 149 public: Clip(LazyProxyTest * test,GrTextureProxy * atlas)150 Clip(LazyProxyTest* test, GrTextureProxy* atlas) 151 : fTest(test) 152 , fAtlas(atlas) {} 153 154 private: apply(GrContext * context,GrRenderTargetContext *,bool,bool,GrAppliedClip * out,SkRect * bounds) const155 bool apply(GrContext* context, GrRenderTargetContext*, bool, bool, GrAppliedClip* out, 156 SkRect* bounds) const override { 157 GrProxyProvider* proxyProvider = context->contextPriv().proxyProvider(); 158 out->addCoverageFP(skstd::make_unique<ClipFP>(proxyProvider, fTest, fAtlas)); 159 return true; 160 } quickContains(const SkRect &) const161 bool quickContains(const SkRect&) const final { return false; } isRRect(const SkRect & rtBounds,SkRRect * rr,GrAA *) const162 bool isRRect(const SkRect& rtBounds, SkRRect* rr, GrAA*) const final { return false; } getConservativeBounds(int width,int height,SkIRect * rect,bool * iior) const163 void getConservativeBounds(int width, int height, SkIRect* rect, bool* iior) const final { 164 rect->set(0, 0, width, height); 165 if (iior) { 166 *iior = false; 167 } 168 } 169 170 LazyProxyTest* const fTest; 171 GrTextureProxy* fAtlas; 172 }; 173 174 private: 175 skiatest::Reporter* fReporter; 176 bool fHasOpTexture; 177 bool fHasClipTexture; 178 }; 179 180 DEF_GPUTEST(LazyProxyTest, reporter, /* options */) { 181 GrMockOptions mockOptions; 182 mockOptions.fConfigOptions[kAlpha_half_GrPixelConfig].fRenderable[0] = true; 183 mockOptions.fConfigOptions[kAlpha_half_GrPixelConfig].fTexturable = true; 184 sk_sp<GrContext> ctx = GrContext::MakeMock(&mockOptions, GrContextOptions()); 185 GrProxyProvider* proxyProvider = ctx->contextPriv().proxyProvider(); 186 for (bool nullTexture : {false, true}) { 187 LazyProxyTest test(reporter); 188 ctx->contextPriv().addOnFlushCallbackObject(&test); 189 sk_sp<GrRenderTargetContext> rtc = 190 ctx->makeDeferredRenderTargetContext(SkBackingFit::kExact, 100, 100, 191 kRGBA_8888_GrPixelConfig, nullptr); 192 REPORTER_ASSERT(reporter, rtc); 193 sk_sp<GrRenderTargetContext> mockAtlas = 194 ctx->makeDeferredRenderTargetContext(SkBackingFit::kExact, 10, 10, 195 kAlpha_half_GrPixelConfig, nullptr); 196 REPORTER_ASSERT(reporter, mockAtlas); 197 rtc->priv().testingOnly_addDrawOp(LazyProxyTest::Clip(&test, mockAtlas->asTextureProxy()), 198 skstd::make_unique<LazyProxyTest::Op>(proxyProvider, &test, nullTexture)); 199 ctx->contextPriv().testingOnly_flushAndRemoveOnFlushCallbackObject(&test); 200 } 201 } 202 203 static const int kSize = 16; 204 205 DEF_GPUTEST(LazyProxyReleaseTest, reporter, /* options */) { 206 GrMockOptions mockOptions; 207 sk_sp<GrContext> ctx = GrContext::MakeMock(&mockOptions, GrContextOptions()); 208 auto proxyProvider = ctx->contextPriv().proxyProvider(); 209 210 GrSurfaceDesc desc; 211 desc.fWidth = kSize; 212 desc.fHeight = kSize; 213 desc.fConfig = kRGBA_8888_GrPixelConfig; 214 215 for (bool doInstantiate : {true, false}) { 216 int testCount = 0; 217 int* testCountPtr = &testCount; 218 sk_sp<GrTextureProxy> proxy = proxyProvider->createLazyProxy( __anon3769f5d10302(GrResourceProvider* resourceProvider, GrSurfaceOrigin* outOrigin) 219 [testCountPtr](GrResourceProvider* resourceProvider, GrSurfaceOrigin* outOrigin) { 220 if (!resourceProvider) { 221 *testCountPtr = -1; 222 return sk_sp<GrTexture>(); 223 } 224 *testCountPtr = 1; 225 return sk_sp<GrTexture>(); 226 }, desc, GrMipMapped::kNo, SkBackingFit::kExact, SkBudgeted::kNo); 227 228 REPORTER_ASSERT(reporter, 0 == testCount); 229 230 if (doInstantiate) { 231 proxy->priv().doLazyInstantiation(ctx->contextPriv().resourceProvider()); 232 REPORTER_ASSERT(reporter, 1 == testCount); 233 proxy.reset(); 234 REPORTER_ASSERT(reporter, -1 == testCount); 235 } else { 236 proxy.reset(); 237 REPORTER_ASSERT(reporter, -1 == testCount); 238 } 239 } 240 } 241 242 class LazyFailedInstantiationTestOp : public GrDrawOp { 243 public: 244 DEFINE_OP_CLASS_ID 245 LazyFailedInstantiationTestOp(GrProxyProvider * proxyProvider,int * testExecuteValue,bool shouldFailInstantiation)246 LazyFailedInstantiationTestOp(GrProxyProvider* proxyProvider, int* testExecuteValue, 247 bool shouldFailInstantiation) 248 : INHERITED(ClassID()) 249 , fTestExecuteValue(testExecuteValue) { 250 GrSurfaceDesc desc; 251 desc.fWidth = kSize; 252 desc.fHeight = kSize; 253 desc.fConfig = kRGBA_8888_GrPixelConfig; 254 255 fLazyProxy = proxyProvider->createLazyProxy( 256 [testExecuteValue, shouldFailInstantiation, desc] ( 257 GrResourceProvider* rp, GrSurfaceOrigin* /*origin*/) { 258 if (!rp) { 259 return sk_sp<GrTexture>(); 260 } 261 if (shouldFailInstantiation) { 262 *testExecuteValue = 1; 263 return sk_sp<GrTexture>(); 264 } 265 return rp->createTexture(desc, SkBudgeted::kNo); 266 }, desc, GrMipMapped::kNo, SkBackingFit::kExact, SkBudgeted::kNo); 267 268 this->setBounds(SkRect::MakeIWH(kSize, kSize), 269 HasAABloat::kNo, IsZeroArea::kNo); 270 } 271 visitProxies(const VisitProxyFunc & func) const272 void visitProxies(const VisitProxyFunc& func) const override { 273 func(fLazyProxy.get()); 274 } 275 276 private: name() const277 const char* name() const override { return "LazyFailedInstantiationTestOp"; } fixedFunctionFlags() const278 FixedFunctionFlags fixedFunctionFlags() const override { return FixedFunctionFlags::kNone; } finalize(const GrCaps &,const GrAppliedClip *,GrPixelConfigIsClamped)279 RequiresDstTexture finalize(const GrCaps&, const GrAppliedClip*, 280 GrPixelConfigIsClamped) override { 281 return RequiresDstTexture::kNo; 282 } onCombineIfPossible(GrOp * other,const GrCaps & caps)283 bool onCombineIfPossible(GrOp* other, const GrCaps& caps) override { return false; } onPrepare(GrOpFlushState *)284 void onPrepare(GrOpFlushState*) override {} onExecute(GrOpFlushState * state)285 void onExecute(GrOpFlushState* state) override { 286 *fTestExecuteValue = 2; 287 } 288 289 int* fTestExecuteValue; 290 sk_sp<GrSurfaceProxy> fLazyProxy; 291 292 typedef GrDrawOp INHERITED; 293 }; 294 295 // Test that when a lazy proxy fails to instantiate during flush that we drop the Op that it was 296 // associated with. 297 DEF_GPUTEST(LazyProxyFailedInstantiationTest, reporter, /* options */) { 298 GrMockOptions mockOptions; 299 sk_sp<GrContext> ctx = GrContext::MakeMock(&mockOptions, GrContextOptions()); 300 GrProxyProvider* proxyProvider = ctx->contextPriv().proxyProvider(); 301 for (bool failInstantiation : {false, true}) { 302 sk_sp<GrRenderTargetContext> rtc = 303 ctx->makeDeferredRenderTargetContext(SkBackingFit::kExact, 100, 100, 304 kRGBA_8888_GrPixelConfig, nullptr); 305 REPORTER_ASSERT(reporter, rtc); 306 307 rtc->clear(nullptr, 0xbaaaaaad, GrRenderTargetContext::CanClearFullscreen::kYes); 308 309 int executeTestValue = 0; 310 rtc->priv().testingOnly_addDrawOp( 311 skstd::make_unique<LazyFailedInstantiationTestOp>(proxyProvider, &executeTestValue, 312 failInstantiation)); 313 ctx->flush(); 314 315 if (failInstantiation) { 316 #ifdef SK_DISABLE_EXPLICIT_GPU_RESOURCE_ALLOCATION 317 // When we disable explicit gpu resource allocation we don't throw away ops that have 318 // uninstantiated proxies. 319 REPORTER_ASSERT(reporter, 2 == executeTestValue); 320 #else 321 REPORTER_ASSERT(reporter, 1 == executeTestValue); 322 #endif 323 } else { 324 REPORTER_ASSERT(reporter, 2 == executeTestValue); 325 } 326 } 327 328 } 329 330 #endif 331