| /* |
| * Copyright 2011 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SkTypes.h" |
| |
| #if SK_SUPPORT_GPU |
| |
| #include "GrContextFactory.h" |
| #include "GrContextPriv.h" |
| #include "GrCaps.h" |
| #include "SkExecutor.h" |
| #include "Test.h" |
| |
| using namespace sk_gpu_test; |
| |
| DEF_GPUTEST(GrContextFactory_NVPRContextOptionHasPathRenderingSupport, reporter, /*factory*/) { |
| // Test that if NVPR is requested, the context always has path rendering |
| // or the context creation fails. |
| GrContextFactory testFactory; |
| // Test that if NVPR is possible, caps are in sync. |
| for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) { |
| GrContextFactory::ContextType ctxType = static_cast<GrContextFactory::ContextType>(i); |
| GrContext* context = testFactory.get(ctxType, |
| GrContextFactory::ContextOverrides::kRequireNVPRSupport); |
| if (!context) { |
| continue; |
| } |
| REPORTER_ASSERT( |
| reporter, |
| context->caps()->shaderCaps()->pathRenderingSupport()); |
| } |
| } |
| |
| DEF_GPUTEST(GrContextFactory_NoPathRenderingIfNVPRDisabled, reporter, /*factory*/) { |
| // Test that if NVPR is explicitly disabled, the context has no path rendering support. |
| |
| GrContextFactory testFactory; |
| for (int i = 0; i <= GrContextFactory::kLastContextType; ++i) { |
| GrContextFactory::ContextType ctxType = (GrContextFactory::ContextType)i; |
| GrContext* context = |
| testFactory.get(ctxType, GrContextFactory::ContextOverrides::kDisableNVPR); |
| if (context) { |
| REPORTER_ASSERT( |
| reporter, |
| !context->caps()->shaderCaps()->pathRenderingSupport()); |
| } |
| } |
| } |
| |
| DEF_GPUTEST(GrContextFactory_RequiredSRGBSupport, reporter, /*factory*/) { |
| // Test that if sRGB support is requested, the context always has that capability |
| // or the context creation fails. Also test that if the creation fails, a context |
| // created without that flag would not have had sRGB support. |
| GrContextFactory testFactory; |
| // Test that if sRGB is requested, caps are in sync. |
| for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) { |
| GrContextFactory::ContextType ctxType = static_cast<GrContextFactory::ContextType>(i); |
| GrContext* context = |
| testFactory.get(ctxType, GrContextFactory::ContextOverrides::kRequireSRGBSupport); |
| |
| if (context) { |
| REPORTER_ASSERT(reporter, context->caps()->srgbSupport()); |
| } else { |
| context = testFactory.get(ctxType); |
| if (context) { |
| REPORTER_ASSERT(reporter, !context->caps()->srgbSupport()); |
| } |
| } |
| } |
| } |
| |
| DEF_GPUTEST(GrContextFactory_abandon, reporter, /*factory*/) { |
| GrContextFactory testFactory; |
| for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) { |
| GrContextFactory::ContextType ctxType = (GrContextFactory::ContextType) i; |
| ContextInfo info1 = testFactory.getContextInfo(ctxType); |
| if (!info1.grContext()) { |
| continue; |
| } |
| REPORTER_ASSERT(reporter, info1.testContext()); |
| // Ref for comparison. The API does not explicitly say that this stays alive. |
| info1.grContext()->ref(); |
| testFactory.abandonContexts(); |
| |
| // Test that we get different context after abandon. |
| ContextInfo info2 = testFactory.getContextInfo(ctxType); |
| REPORTER_ASSERT(reporter, info2.grContext()); |
| REPORTER_ASSERT(reporter, info2.testContext()); |
| |
| REPORTER_ASSERT(reporter, info1.grContext() != info2.grContext()); |
| // The GL context should also change, but it also could get the same address. |
| |
| info1.grContext()->unref(); |
| } |
| } |
| |
| DEF_GPUTEST(GrContextFactory_sharedContexts, reporter, /*factory*/) { |
| GrContextFactory testFactory; |
| |
| for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) { |
| GrContextFactory::ContextType ctxType = static_cast<GrContextFactory::ContextType>(i); |
| ContextInfo info1 = testFactory.getContextInfo(ctxType); |
| if (!info1.grContext()) { |
| continue; |
| } |
| |
| // Ref for passing in. The API does not explicitly say that this stays alive. |
| info1.grContext()->ref(); |
| testFactory.abandonContexts(); |
| |
| // Test that creating a context in a share group with an abandoned context fails. |
| ContextInfo info2 = testFactory.getSharedContextInfo(info1.grContext()); |
| REPORTER_ASSERT(reporter, !info2.grContext()); |
| info1.grContext()->unref(); |
| |
| // Create a new base context |
| ContextInfo info3 = testFactory.getContextInfo(ctxType); |
| if (!info3.grContext()) { |
| // Vulkan NexusPlayer bot fails here. Sigh. |
| continue; |
| } |
| |
| // Creating a context in a share group may fail, but should never crash. |
| ContextInfo info4 = testFactory.getSharedContextInfo(info3.grContext()); |
| if (!info4.grContext()) { |
| continue; |
| } |
| REPORTER_ASSERT(reporter, info3.grContext() != info4.grContext()); |
| REPORTER_ASSERT(reporter, info3.testContext() != info4.testContext()); |
| |
| // Passing a different index should create a new (unique) context. |
| ContextInfo info5 = testFactory.getSharedContextInfo(info3.grContext(), 1); |
| REPORTER_ASSERT(reporter, info5.grContext()); |
| REPORTER_ASSERT(reporter, info5.testContext()); |
| REPORTER_ASSERT(reporter, info5.grContext() != info4.grContext()); |
| REPORTER_ASSERT(reporter, info5.testContext() != info4.testContext()); |
| } |
| } |
| |
| DEF_GPUTEST(GrContextFactory_executorAndTaskGroup, reporter, /*factory*/) { |
| // Verify that contexts have a task group iff we supply an executor with context options |
| GrContextOptions contextOptions; |
| contextOptions.fExecutor = nullptr; |
| GrContextFactory serialFactory(contextOptions); |
| |
| std::unique_ptr<SkExecutor> threadPool = SkExecutor::MakeThreadPool(1); |
| contextOptions.fExecutor = threadPool.get(); |
| GrContextFactory threadedFactory(contextOptions); |
| |
| for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) { |
| GrContextFactory::ContextType ctxType = static_cast<GrContextFactory::ContextType>(i); |
| ContextInfo serialInfo = serialFactory.getContextInfo(ctxType); |
| if (GrContext* serialContext = serialInfo.grContext()) { |
| REPORTER_ASSERT(reporter, nullptr == serialContext->contextPriv().getTaskGroup()); |
| } |
| |
| ContextInfo threadedInfo = threadedFactory.getContextInfo(ctxType); |
| if (GrContext* threadedContext = threadedInfo.grContext()) { |
| REPORTER_ASSERT(reporter, nullptr != threadedContext->contextPriv().getTaskGroup()); |
| } |
| } |
| } |
| |
| DEF_GPUTEST_FOR_ALL_CONTEXTS(GrContextDump, reporter, ctxInfo) { |
| // Ensure that GrContext::dump doesn't assert (which is possible, if the JSON code is wrong) |
| SkString result = ctxInfo.grContext()->dump(); |
| REPORTER_ASSERT(reporter, !result.isEmpty()); |
| } |
| |
| #endif |