2012-08-14 22:02:48 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2011 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2015-08-27 14:41:13 +00:00
|
|
|
#include "SkTypes.h"
|
|
|
|
|
2012-08-14 22:02:48 +00:00
|
|
|
#if SK_SUPPORT_GPU
|
2014-01-10 22:08:27 +00:00
|
|
|
|
2012-08-14 22:02:48 +00:00
|
|
|
#include "GrContextFactory.h"
|
2017-08-23 14:12:00 +00:00
|
|
|
#include "GrContextPriv.h"
|
2015-12-04 07:04:50 +00:00
|
|
|
#include "GrCaps.h"
|
2017-08-23 14:12:00 +00:00
|
|
|
#include "SkExecutor.h"
|
2014-01-10 22:08:27 +00:00
|
|
|
#include "Test.h"
|
2012-08-14 22:02:48 +00:00
|
|
|
|
2016-04-05 19:59:06 +00:00
|
|
|
using namespace sk_gpu_test;
|
2016-03-31 01:56:19 +00:00
|
|
|
|
2017-11-15 20:48:03 +00:00
|
|
|
DEF_GPUTEST(GrContextFactory_NVPRContextOptionHasPathRenderingSupport, reporter, options) {
|
2015-12-04 07:04:50 +00:00
|
|
|
// Test that if NVPR is requested, the context always has path rendering
|
|
|
|
// or the context creation fails.
|
2016-04-05 18:06:27 +00:00
|
|
|
for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) {
|
2017-11-15 20:48:03 +00:00
|
|
|
GrContextFactory testFactory(options);
|
|
|
|
// Test that if NVPR is possible, caps are in sync.
|
2016-04-05 18:06:27 +00:00
|
|
|
GrContextFactory::ContextType ctxType = static_cast<GrContextFactory::ContextType>(i);
|
|
|
|
GrContext* context = testFactory.get(ctxType,
|
2017-02-21 19:36:05 +00:00
|
|
|
GrContextFactory::ContextOverrides::kRequireNVPRSupport);
|
Add config options to run different GPU APIs to dm and nanobench
Add extended config specification form that can be used to run different
gpu backend with different APIs.
The configs can be specified with the form:
gpu(api=string,dit=bool,nvpr=bool,samples=int)
This replaces and removes the --gpuAPI flag.
All existing configs should still work.
Adds following documentation:
out/Debug/dm --help config
Flags:
--config: type: string default: 565 8888 gpu nonrendering
Options: 565 8888 debug gpu gpudebug gpudft gpunull msaa16 msaa4
nonrendering null nullgpu nvprmsaa16 nvprmsaa4 pdf pdf_poppler skp svg
xps or use extended form 'backend(option=value,...)'.
Extended form: 'backend(option=value,...)'
Possible backends and options:
gpu(api=string,dit=bool,nvpr=bool,samples=int) GPU backend
api type: string default: native.
Select graphics API to use with gpu backend.
Options:
native Use platform default OpenGL or OpenGL ES backend.
gl Use OpenGL.
gles Use OpenGL ES.
debug Use debug OpenGL.
null Use null OpenGL.
dit type: bool default: false.
Use device independent text.
nvpr type: bool default: false.
Use NV_path_rendering OpenGL and OpenGL ES extension.
samples type: int default: 0.
Use multisampling with N samples.
Predefined configs:
gpu = gpu()
msaa4 = gpu(samples=4)
msaa16 = gpu(samples=16)
nvprmsaa4 = gpu(nvpr=true,samples=4)
nvprmsaa16 = gpu(nvpr=true,samples=16)
gpudft = gpu(dit=true)
gpudebug = gpu(api=debug)
gpunull = gpu(api=null)
debug = gpu(api=debug)
nullgpu = gpu(api=null)
BUG=skia:2992
Committed: https://skia.googlesource.com/skia/+/e13ca329fca4c28cf4e078561f591ab27b743d23
GOLD_TRYBOT_URL= https://gold.skia.org/search2?unt=true&query=source_type%3Dgm&master=false&issue=1490113005
Committed: https://skia.googlesource.com/skia/+/c8b4336444e7b90382e04e33665fb3b8490b825b
Committed: https://skia.googlesource.com/skia/+/9ebc3f0ee6db215dde461dc4777d85988cf272dd
Review URL: https://codereview.chromium.org/1490113005
2015-12-23 09:33:00 +00:00
|
|
|
if (!context) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-05-11 14:14:21 +00:00
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
context->contextPriv().caps()->shaderCaps()->pathRenderingSupport());
|
2015-12-04 07:04:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-15 20:48:03 +00:00
|
|
|
DEF_GPUTEST(GrContextFactory_NoPathRenderingIfNVPRDisabled, reporter, options) {
|
2017-02-21 19:36:05 +00:00
|
|
|
// Test that if NVPR is explicitly disabled, the context has no path rendering support.
|
2015-12-04 07:04:50 +00:00
|
|
|
|
2016-04-05 18:06:27 +00:00
|
|
|
for (int i = 0; i <= GrContextFactory::kLastContextType; ++i) {
|
2017-11-15 20:48:03 +00:00
|
|
|
GrContextFactory testFactory(options);
|
2016-04-05 18:06:27 +00:00
|
|
|
GrContextFactory::ContextType ctxType = (GrContextFactory::ContextType)i;
|
2017-02-21 19:36:05 +00:00
|
|
|
GrContext* context =
|
|
|
|
testFactory.get(ctxType, GrContextFactory::ContextOverrides::kDisableNVPR);
|
2015-12-04 07:04:50 +00:00
|
|
|
if (context) {
|
2018-05-11 14:14:21 +00:00
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
!context->contextPriv().caps()->shaderCaps()->pathRenderingSupport());
|
2015-12-04 07:04:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-14 22:02:48 +00:00
|
|
|
|
2017-11-15 20:48:03 +00:00
|
|
|
DEF_GPUTEST(GrContextFactory_RequiredSRGBSupport, reporter, options) {
|
2016-03-30 18:19:36 +00:00
|
|
|
// 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.
|
2016-04-05 18:06:27 +00:00
|
|
|
for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) {
|
2017-11-15 20:48:03 +00:00
|
|
|
GrContextFactory testFactory(options);
|
|
|
|
// Test that if sRGB is requested, caps are in sync.
|
2016-04-05 18:06:27 +00:00
|
|
|
GrContextFactory::ContextType ctxType = static_cast<GrContextFactory::ContextType>(i);
|
2016-03-30 18:19:36 +00:00
|
|
|
GrContext* context =
|
2017-02-21 19:36:05 +00:00
|
|
|
testFactory.get(ctxType, GrContextFactory::ContextOverrides::kRequireSRGBSupport);
|
2016-03-30 18:19:36 +00:00
|
|
|
|
|
|
|
if (context) {
|
2018-05-11 14:14:21 +00:00
|
|
|
REPORTER_ASSERT(reporter, context->contextPriv().caps()->srgbSupport());
|
2016-03-30 18:19:36 +00:00
|
|
|
} else {
|
2016-04-05 18:06:27 +00:00
|
|
|
context = testFactory.get(ctxType);
|
2016-03-30 18:19:36 +00:00
|
|
|
if (context) {
|
2018-05-11 14:14:21 +00:00
|
|
|
REPORTER_ASSERT(reporter, !context->contextPriv().caps()->srgbSupport());
|
2016-03-30 18:19:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-15 20:48:03 +00:00
|
|
|
DEF_GPUTEST(GrContextFactory_abandon, reporter, options) {
|
2016-04-05 18:06:27 +00:00
|
|
|
for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) {
|
2017-11-15 20:48:03 +00:00
|
|
|
GrContextFactory testFactory(options);
|
2016-04-05 18:06:27 +00:00
|
|
|
GrContextFactory::ContextType ctxType = (GrContextFactory::ContextType) i;
|
2016-04-05 19:59:06 +00:00
|
|
|
ContextInfo info1 = testFactory.getContextInfo(ctxType);
|
2016-05-11 13:33:06 +00:00
|
|
|
if (!info1.grContext()) {
|
2016-01-07 07:49:30 +00:00
|
|
|
continue;
|
|
|
|
}
|
2016-05-11 17:38:05 +00:00
|
|
|
REPORTER_ASSERT(reporter, info1.testContext());
|
2016-01-07 07:49:30 +00:00
|
|
|
// Ref for comparison. The API does not explicitly say that this stays alive.
|
2016-05-11 13:33:06 +00:00
|
|
|
info1.grContext()->ref();
|
2016-01-07 07:49:30 +00:00
|
|
|
testFactory.abandonContexts();
|
|
|
|
|
|
|
|
// Test that we get different context after abandon.
|
2016-04-05 19:59:06 +00:00
|
|
|
ContextInfo info2 = testFactory.getContextInfo(ctxType);
|
2016-05-11 13:33:06 +00:00
|
|
|
REPORTER_ASSERT(reporter, info2.grContext());
|
2016-05-11 17:38:05 +00:00
|
|
|
REPORTER_ASSERT(reporter, info2.testContext());
|
|
|
|
|
2016-05-11 13:33:06 +00:00
|
|
|
REPORTER_ASSERT(reporter, info1.grContext() != info2.grContext());
|
|
|
|
// The GL context should also change, but it also could get the same address.
|
2016-01-07 07:49:30 +00:00
|
|
|
|
2016-05-11 13:33:06 +00:00
|
|
|
info1.grContext()->unref();
|
2016-01-07 07:49:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-15 20:48:03 +00:00
|
|
|
DEF_GPUTEST(GrContextFactory_sharedContexts, reporter, options) {
|
2017-02-21 21:58:08 +00:00
|
|
|
for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) {
|
2017-11-15 20:48:03 +00:00
|
|
|
GrContextFactory testFactory(options);
|
2017-02-21 21:58:08 +00:00
|
|
|
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.
|
2017-02-24 19:51:44 +00:00
|
|
|
ContextInfo info2 = testFactory.getSharedContextInfo(info1.grContext());
|
2017-02-21 21:58:08 +00:00
|
|
|
REPORTER_ASSERT(reporter, !info2.grContext());
|
|
|
|
info1.grContext()->unref();
|
|
|
|
|
|
|
|
// Create a new base context
|
|
|
|
ContextInfo info3 = testFactory.getContextInfo(ctxType);
|
2017-02-28 15:05:43 +00:00
|
|
|
if (!info3.grContext()) {
|
|
|
|
// Vulkan NexusPlayer bot fails here. Sigh.
|
|
|
|
continue;
|
|
|
|
}
|
2017-02-21 21:58:08 +00:00
|
|
|
|
|
|
|
// Creating a context in a share group may fail, but should never crash.
|
2017-02-24 19:51:44 +00:00
|
|
|
ContextInfo info4 = testFactory.getSharedContextInfo(info3.grContext());
|
2017-02-21 21:58:08 +00:00
|
|
|
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.
|
2017-02-24 19:51:44 +00:00
|
|
|
ContextInfo info5 = testFactory.getSharedContextInfo(info3.grContext(), 1);
|
2017-02-21 21:58:08 +00:00
|
|
|
REPORTER_ASSERT(reporter, info5.grContext());
|
|
|
|
REPORTER_ASSERT(reporter, info5.testContext());
|
|
|
|
REPORTER_ASSERT(reporter, info5.grContext() != info4.grContext());
|
|
|
|
REPORTER_ASSERT(reporter, info5.testContext() != info4.testContext());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-15 20:48:03 +00:00
|
|
|
DEF_GPUTEST(GrContextFactory_executorAndTaskGroup, reporter, options) {
|
|
|
|
for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) {
|
|
|
|
// Verify that contexts have a task group iff we supply an executor with context options
|
|
|
|
GrContextOptions contextOptions = options;
|
|
|
|
contextOptions.fExecutor = nullptr;
|
|
|
|
GrContextFactory serialFactory(contextOptions);
|
2017-08-23 14:12:00 +00:00
|
|
|
|
2017-11-15 20:48:03 +00:00
|
|
|
std::unique_ptr<SkExecutor> threadPool = SkExecutor::MakeFIFOThreadPool(1);
|
|
|
|
contextOptions.fExecutor = threadPool.get();
|
|
|
|
GrContextFactory threadedFactory(contextOptions);
|
2017-08-23 14:12:00 +00:00
|
|
|
|
|
|
|
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());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-10 14:23:25 +00:00
|
|
|
DEF_GPUTEST_FOR_ALL_CONTEXTS(GrContextDump, reporter, ctxInfo) {
|
|
|
|
// Ensure that GrContext::dump doesn't assert (which is possible, if the JSON code is wrong)
|
2018-03-06 13:20:37 +00:00
|
|
|
SkString result = ctxInfo.grContext()->contextPriv().dump();
|
2017-08-10 14:23:25 +00:00
|
|
|
REPORTER_ASSERT(reporter, !result.isEmpty());
|
|
|
|
}
|
|
|
|
|
2012-08-14 22:02:48 +00:00
|
|
|
#endif
|