2016-03-31 01:56:19 +00:00
|
|
|
|
2014-11-13 19:12:41 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2014 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "GrContextFactory.h"
|
2016-03-31 17:59:06 +00:00
|
|
|
#include "gl/GLTestContext.h"
|
2014-11-13 19:12:41 +00:00
|
|
|
|
2016-11-14 14:53:13 +00:00
|
|
|
#if SK_ANGLE
|
|
|
|
#include "gl/angle/GLTestContext_angle.h"
|
|
|
|
#endif
|
2016-09-21 21:01:32 +00:00
|
|
|
#include "gl/command_buffer/GLTestContext_command_buffer.h"
|
2016-03-31 17:59:06 +00:00
|
|
|
#include "gl/debug/DebugGLTestContext.h"
|
2014-11-13 19:12:41 +00:00
|
|
|
#if SK_MESA
|
2016-03-31 17:59:06 +00:00
|
|
|
#include "gl/mesa/GLTestContext_mesa.h"
|
2016-03-23 18:01:22 +00:00
|
|
|
#endif
|
2016-06-09 20:07:13 +00:00
|
|
|
#ifdef SK_VULKAN
|
2016-05-11 17:09:18 +00:00
|
|
|
#include "vk/VkTestContext.h"
|
2016-04-01 14:42:05 +00:00
|
|
|
#endif
|
2016-03-31 17:59:06 +00:00
|
|
|
#include "gl/null/NullGLTestContext.h"
|
2015-07-01 09:58:50 +00:00
|
|
|
#include "gl/GrGLGpu.h"
|
|
|
|
#include "GrCaps.h"
|
2014-11-13 19:12:41 +00:00
|
|
|
|
2016-12-28 16:19:22 +00:00
|
|
|
#if defined(SK_BUILD_FOR_WIN32) && defined(SK_ENABLE_DISCRETE_GPU)
|
|
|
|
extern "C" {
|
|
|
|
// NVIDIA documents that the presence and value of this symbol programmatically enable the high
|
|
|
|
// performance GPU in laptops with switchable graphics.
|
|
|
|
// https://docs.nvidia.com/gameworks/content/technologies/desktop/optimus.htm
|
|
|
|
// From testing, including this symbol, even if it is set to 0, we still get the NVIDIA GPU.
|
|
|
|
_declspec(dllexport) unsigned long NvOptimusEnablement = 0x00000001;
|
|
|
|
|
|
|
|
// AMD has a similar mechanism, although I don't have an AMD laptop, so this is untested.
|
|
|
|
// https://community.amd.com/thread/169965
|
|
|
|
__declspec(dllexport) int AmdPowerXpressRequestHighPerformance = 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-03-31 01:56:19 +00:00
|
|
|
namespace sk_gpu_test {
|
2016-01-07 07:49:30 +00:00
|
|
|
GrContextFactory::GrContextFactory() { }
|
|
|
|
|
|
|
|
GrContextFactory::GrContextFactory(const GrContextOptions& opts)
|
|
|
|
: fGlobalOptions(opts) {
|
|
|
|
}
|
|
|
|
|
|
|
|
GrContextFactory::~GrContextFactory() {
|
|
|
|
this->destroyContexts();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GrContextFactory::destroyContexts() {
|
|
|
|
for (Context& context : fContexts) {
|
2016-05-11 17:09:18 +00:00
|
|
|
if (context.fTestContext) {
|
|
|
|
context.fTestContext->makeCurrent();
|
2016-01-07 07:49:30 +00:00
|
|
|
}
|
|
|
|
if (!context.fGrContext->unique()) {
|
2016-04-11 21:21:33 +00:00
|
|
|
context.fGrContext->releaseResourcesAndAbandonContext();
|
|
|
|
context.fAbandoned = true;
|
2016-01-07 07:49:30 +00:00
|
|
|
}
|
|
|
|
context.fGrContext->unref();
|
2016-05-11 17:09:18 +00:00
|
|
|
delete context.fTestContext;
|
2016-01-07 07:49:30 +00:00
|
|
|
}
|
|
|
|
fContexts.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GrContextFactory::abandonContexts() {
|
|
|
|
for (Context& context : fContexts) {
|
2016-04-11 21:21:33 +00:00
|
|
|
if (!context.fAbandoned) {
|
2016-05-11 17:09:18 +00:00
|
|
|
if (context.fTestContext) {
|
|
|
|
context.fTestContext->makeCurrent();
|
|
|
|
context.fTestContext->testAbandon();
|
|
|
|
delete(context.fTestContext);
|
|
|
|
context.fTestContext = nullptr;
|
2016-04-11 21:21:33 +00:00
|
|
|
}
|
|
|
|
context.fGrContext->abandonContext();
|
|
|
|
context.fAbandoned = true;
|
2016-01-07 07:49:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-01 18:54:31 +00:00
|
|
|
void GrContextFactory::releaseResourcesAndAbandonContexts() {
|
|
|
|
for (Context& context : fContexts) {
|
2016-04-11 21:21:33 +00:00
|
|
|
if (!context.fAbandoned) {
|
2016-05-11 17:09:18 +00:00
|
|
|
if (context.fTestContext) {
|
|
|
|
context.fTestContext->makeCurrent();
|
2016-04-11 21:21:33 +00:00
|
|
|
}
|
2016-04-01 18:54:31 +00:00
|
|
|
context.fGrContext->releaseResourcesAndAbandonContext();
|
2016-04-11 21:21:33 +00:00
|
|
|
context.fAbandoned = true;
|
2016-05-11 17:09:18 +00:00
|
|
|
if (context.fTestContext) {
|
|
|
|
delete context.fTestContext;
|
|
|
|
context.fTestContext = nullptr;
|
2016-04-11 21:21:33 +00:00
|
|
|
}
|
2016-04-01 18:54:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-24 19:51:44 +00:00
|
|
|
ContextInfo GrContextFactory::getContextInfoInternal(ContextType type, ContextOverrides overrides,
|
|
|
|
GrContext* shareContext, uint32_t shareIndex) {
|
2017-02-21 21:58:08 +00:00
|
|
|
// (shareIndex != 0) -> (shareContext != nullptr)
|
|
|
|
SkASSERT((shareIndex == 0) || (shareContext != nullptr));
|
|
|
|
|
2014-11-13 19:12:41 +00:00
|
|
|
for (int i = 0; i < fContexts.count(); ++i) {
|
2016-01-07 07:49:30 +00:00
|
|
|
Context& context = fContexts[i];
|
|
|
|
if (context.fType == type &&
|
2017-02-21 19:36:05 +00:00
|
|
|
context.fOverrides == overrides &&
|
2017-02-21 21:58:08 +00:00
|
|
|
context.fShareContext == shareContext &&
|
|
|
|
context.fShareIndex == shareIndex &&
|
2016-04-11 21:21:33 +00:00
|
|
|
!context.fAbandoned) {
|
2016-05-11 17:09:18 +00:00
|
|
|
context.fTestContext->makeCurrent();
|
|
|
|
return ContextInfo(context.fBackend, context.fTestContext, context.fGrContext);
|
2014-11-13 19:12:41 +00:00
|
|
|
}
|
|
|
|
}
|
2017-02-21 21:58:08 +00:00
|
|
|
|
|
|
|
// If we're trying to create a context in a share group, find the master context
|
|
|
|
Context* masterContext = nullptr;
|
|
|
|
if (shareContext) {
|
|
|
|
for (int i = 0; i < fContexts.count(); ++i) {
|
|
|
|
if (!fContexts[i].fAbandoned && fContexts[i].fGrContext == shareContext) {
|
|
|
|
masterContext = &fContexts[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-02-24 19:51:44 +00:00
|
|
|
SkASSERT(masterContext && masterContext->fType == type);
|
2017-02-21 21:58:08 +00:00
|
|
|
}
|
|
|
|
|
2016-11-03 18:40:50 +00:00
|
|
|
std::unique_ptr<TestContext> testCtx;
|
2016-04-11 21:21:33 +00:00
|
|
|
GrBackendContext backendContext = 0;
|
|
|
|
sk_sp<const GrGLInterface> glInterface;
|
|
|
|
GrBackend backend = ContextTypeBackend(type);
|
|
|
|
switch (backend) {
|
2016-05-11 17:09:18 +00:00
|
|
|
case kOpenGL_GrBackend: {
|
2017-02-21 21:58:08 +00:00
|
|
|
GLTestContext* glShareContext = masterContext
|
|
|
|
? static_cast<GLTestContext*>(masterContext->fTestContext) : nullptr;
|
2016-05-11 17:09:18 +00:00
|
|
|
GLTestContext* glCtx;
|
2016-04-11 21:21:33 +00:00
|
|
|
switch (type) {
|
|
|
|
case kGL_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = CreatePlatformGLTestContext(kGL_GrGLStandard, glShareContext);
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
|
|
|
case kGLES_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = CreatePlatformGLTestContext(kGLES_GrGLStandard, glShareContext);
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
2016-11-14 14:53:13 +00:00
|
|
|
#if SK_ANGLE
|
2016-10-14 15:13:48 +00:00
|
|
|
case kANGLE_D3D9_ES2_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = MakeANGLETestContext(ANGLEBackend::kD3D9, ANGLEContextVersion::kES2,
|
|
|
|
glShareContext).release();
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
2016-10-14 15:13:48 +00:00
|
|
|
case kANGLE_D3D11_ES2_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = MakeANGLETestContext(ANGLEBackend::kD3D11, ANGLEContextVersion::kES2,
|
|
|
|
glShareContext).release();
|
2016-10-14 15:13:48 +00:00
|
|
|
break;
|
|
|
|
case kANGLE_D3D11_ES3_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = MakeANGLETestContext(ANGLEBackend::kD3D11, ANGLEContextVersion::kES3,
|
|
|
|
glShareContext).release();
|
2016-10-14 15:13:48 +00:00
|
|
|
break;
|
|
|
|
case kANGLE_GL_ES2_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = MakeANGLETestContext(ANGLEBackend::kOpenGL, ANGLEContextVersion::kES2,
|
|
|
|
glShareContext).release();
|
2016-10-14 15:13:48 +00:00
|
|
|
break;
|
|
|
|
case kANGLE_GL_ES3_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = MakeANGLETestContext(ANGLEBackend::kOpenGL, ANGLEContextVersion::kES3,
|
|
|
|
glShareContext).release();
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
2016-11-14 14:53:13 +00:00
|
|
|
#endif
|
2016-04-11 21:21:33 +00:00
|
|
|
case kCommandBuffer_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = CommandBufferGLTestContext::Create(glShareContext);
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
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 SK_MESA
|
2016-04-11 21:21:33 +00:00
|
|
|
case kMESA_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = CreateMesaGLTestContext(glShareContext);
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
2014-11-13 19:12:41 +00:00
|
|
|
#endif
|
2016-04-11 21:21:33 +00:00
|
|
|
case kNullGL_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = CreateNullGLTestContext(
|
|
|
|
ContextOverrides::kRequireNVPRSupport & overrides, glShareContext);
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
|
|
|
case kDebugGL_ContextType:
|
2017-02-21 21:58:08 +00:00
|
|
|
glCtx = CreateDebugGLTestContext(glShareContext);
|
2016-04-11 21:21:33 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return ContextInfo();
|
|
|
|
}
|
2016-05-11 17:09:18 +00:00
|
|
|
if (!glCtx) {
|
2016-04-11 21:21:33 +00:00
|
|
|
return ContextInfo();
|
|
|
|
}
|
2016-05-11 17:09:18 +00:00
|
|
|
testCtx.reset(glCtx);
|
2016-04-11 21:21:33 +00:00
|
|
|
glInterface.reset(SkRef(glCtx->gl()));
|
|
|
|
backendContext = reinterpret_cast<GrBackendContext>(glInterface.get());
|
2014-11-13 19:12:41 +00:00
|
|
|
break;
|
2016-05-11 17:09:18 +00:00
|
|
|
}
|
2016-04-11 21:21:33 +00:00
|
|
|
#ifdef SK_VULKAN
|
2017-05-15 17:50:35 +00:00
|
|
|
case kVulkan_GrBackend: {
|
|
|
|
VkTestContext* vkSharedContext = masterContext
|
|
|
|
? static_cast<VkTestContext*>(masterContext->fTestContext) : nullptr;
|
2016-04-11 21:21:33 +00:00
|
|
|
SkASSERT(kVulkan_ContextType == type);
|
2017-02-21 19:36:05 +00:00
|
|
|
if (ContextOverrides::kRequireNVPRSupport & overrides) {
|
2016-04-11 21:21:33 +00:00
|
|
|
return ContextInfo();
|
|
|
|
}
|
2017-05-15 17:50:35 +00:00
|
|
|
testCtx.reset(CreatePlatformVkTestContext(vkSharedContext));
|
2016-05-11 17:09:18 +00:00
|
|
|
if (!testCtx) {
|
2016-04-11 21:21:33 +00:00
|
|
|
return ContextInfo();
|
|
|
|
}
|
2016-05-11 17:09:18 +00:00
|
|
|
|
2017-01-24 22:22:05 +00:00
|
|
|
// There is some bug (either in Skia or the NV Vulkan driver) where VkDevice
|
|
|
|
// destruction will hang occaisonally. For some reason having an existing GL
|
|
|
|
// context fixes this.
|
|
|
|
if (!fSentinelGLContext) {
|
|
|
|
fSentinelGLContext.reset(CreatePlatformGLTestContext(kGL_GrGLStandard));
|
|
|
|
if (!fSentinelGLContext) {
|
|
|
|
fSentinelGLContext.reset(CreatePlatformGLTestContext(kGLES_GrGLStandard));
|
|
|
|
}
|
|
|
|
}
|
2016-05-11 17:09:18 +00:00
|
|
|
backendContext = testCtx->backendContext();
|
2014-11-13 19:12:41 +00:00
|
|
|
break;
|
2017-05-15 17:50:35 +00:00
|
|
|
}
|
2016-04-11 21:21:33 +00:00
|
|
|
#endif
|
|
|
|
default:
|
2016-01-07 07:49:30 +00:00
|
|
|
return ContextInfo();
|
2014-11-13 19:12:41 +00:00
|
|
|
}
|
2016-05-11 17:09:18 +00:00
|
|
|
testCtx->makeCurrent();
|
|
|
|
SkASSERT(testCtx && testCtx->backend() == backend);
|
2016-07-29 15:14:20 +00:00
|
|
|
GrContextOptions grOptions = fGlobalOptions;
|
2017-02-22 19:00:42 +00:00
|
|
|
if (ContextOverrides::kDisableNVPR & overrides) {
|
|
|
|
grOptions.fSuppressPathRendering = true;
|
|
|
|
}
|
2017-02-21 19:36:05 +00:00
|
|
|
if (ContextOverrides::kUseInstanced & overrides) {
|
2016-07-29 15:14:20 +00:00
|
|
|
grOptions.fEnableInstancedRendering = true;
|
|
|
|
}
|
2017-02-21 19:36:05 +00:00
|
|
|
if (ContextOverrides::kAllowSRGBWithoutDecodeControl & overrides) {
|
|
|
|
grOptions.fRequireDecodeDisableForSRGB = false;
|
|
|
|
}
|
2017-05-08 19:02:07 +00:00
|
|
|
if (ContextOverrides::kAvoidStencilBuffers & overrides) {
|
|
|
|
grOptions.fAvoidStencilBuffers = true;
|
|
|
|
}
|
2017-02-21 21:58:08 +00:00
|
|
|
sk_sp<GrContext> grCtx(GrContext::Create(backend, backendContext, grOptions));
|
2014-11-13 19:12:41 +00:00
|
|
|
if (!grCtx.get()) {
|
2016-01-07 07:49:30 +00:00
|
|
|
return ContextInfo();
|
2014-11-13 19:12:41 +00:00
|
|
|
}
|
2017-02-21 19:36:05 +00:00
|
|
|
if (ContextOverrides::kRequireNVPRSupport & overrides) {
|
2015-07-01 09:58:50 +00:00
|
|
|
if (!grCtx->caps()->shaderCaps()->pathRenderingSupport()) {
|
2016-01-07 07:49:30 +00:00
|
|
|
return ContextInfo();
|
2015-07-01 09:58:50 +00:00
|
|
|
}
|
|
|
|
}
|
2017-02-21 19:36:05 +00:00
|
|
|
if (ContextOverrides::kUseInstanced & overrides) {
|
2016-07-29 15:14:20 +00:00
|
|
|
if (GrCaps::InstancedSupport::kNone == grCtx->caps()->instancedSupport()) {
|
|
|
|
return ContextInfo();
|
|
|
|
}
|
|
|
|
}
|
2017-02-21 19:36:05 +00:00
|
|
|
if (ContextOverrides::kRequireSRGBSupport & overrides) {
|
2016-03-30 18:19:36 +00:00
|
|
|
if (!grCtx->caps()->srgbSupport()) {
|
|
|
|
return ContextInfo();
|
|
|
|
}
|
|
|
|
}
|
2015-07-01 09:58:50 +00:00
|
|
|
|
2016-01-07 07:49:30 +00:00
|
|
|
Context& context = fContexts.push_back();
|
2016-05-11 17:09:18 +00:00
|
|
|
context.fBackend = backend;
|
|
|
|
context.fTestContext = testCtx.release();
|
2016-01-07 07:49:30 +00:00
|
|
|
context.fGrContext = SkRef(grCtx.get());
|
|
|
|
context.fType = type;
|
2017-02-21 19:36:05 +00:00
|
|
|
context.fOverrides = overrides;
|
2016-04-11 21:21:33 +00:00
|
|
|
context.fAbandoned = false;
|
2017-02-21 21:58:08 +00:00
|
|
|
context.fShareContext = shareContext;
|
|
|
|
context.fShareIndex = shareIndex;
|
2016-05-11 17:09:18 +00:00
|
|
|
return ContextInfo(context.fBackend, context.fTestContext, context.fGrContext);
|
2014-11-13 19:12:41 +00:00
|
|
|
}
|
2017-02-21 21:58:08 +00:00
|
|
|
|
2017-02-24 19:51:44 +00:00
|
|
|
ContextInfo GrContextFactory::getContextInfo(ContextType type, ContextOverrides overrides) {
|
|
|
|
return this->getContextInfoInternal(type, overrides, nullptr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ContextInfo GrContextFactory::getSharedContextInfo(GrContext* shareContext, uint32_t shareIndex) {
|
|
|
|
SkASSERT(shareContext);
|
|
|
|
for (int i = 0; i < fContexts.count(); ++i) {
|
|
|
|
if (!fContexts[i].fAbandoned && fContexts[i].fGrContext == shareContext) {
|
|
|
|
return this->getContextInfoInternal(fContexts[i].fType, fContexts[i].fOverrides,
|
|
|
|
shareContext, shareIndex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ContextInfo();
|
|
|
|
}
|
|
|
|
|
2016-03-31 01:56:19 +00:00
|
|
|
} // namespace sk_gpu_test
|