| Index: src/gpu/gl/command_buffer/SkCommandBufferGLContext.cpp
|
| diff --git a/src/gpu/gl/command_buffer/SkCommandBufferGLContext.cpp b/src/gpu/gl/command_buffer/SkCommandBufferGLContext.cpp
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..a5a765b9167856bcfab579072c483cbe02a73c98
|
| --- /dev/null
|
| +++ b/src/gpu/gl/command_buffer/SkCommandBufferGLContext.cpp
|
| @@ -0,0 +1,271 @@
|
| +
|
| +/*
|
| + * Copyright 2015 Google Inc.
|
| + *
|
| + * Use of this source code is governed by a BSD-style license that can be
|
| + * found in the LICENSE file.
|
| + */
|
| +#include <EGL/egl.h>
|
| +
|
| +#include "SkOnce.h"
|
| +#include "gl/GrGLInterface.h"
|
| +#include "gl/GrGLAssembleInterface.h"
|
| +#include "gl/command_buffer/SkCommandBufferGLContext.h"
|
| +
|
| +typedef EGLDisplay (*GetDisplayProc)(EGLNativeDisplayType display_id);
|
| +typedef EGLBoolean (*InitializeProc)(EGLDisplay dpy, EGLint *major, EGLint *minor);
|
| +typedef EGLBoolean (*TerminateProc)(EGLDisplay dpy);
|
| +typedef EGLBoolean (*ChooseConfigProc)(EGLDisplay dpy, const EGLint* attrib_list, EGLConfig* configs, EGLint config_size, EGLint* num_config);
|
| +typedef EGLBoolean (*GetConfigAttrib)(EGLDisplay dpy, EGLConfig config, EGLint attribute, EGLint* value);
|
| +typedef EGLSurface (*CreateWindowSurfaceProc)(EGLDisplay dpy, EGLConfig config, EGLNativeWindowType win, const EGLint* attrib_list);
|
| +typedef EGLSurface (*CreatePbufferSurfaceProc)(EGLDisplay dpy, EGLConfig config, const EGLint* attrib_list);
|
| +typedef EGLBoolean (*DestroySurfaceProc)(EGLDisplay dpy, EGLSurface surface);
|
| +typedef EGLContext (*CreateContextProc)(EGLDisplay dpy, EGLConfig config, EGLContext share_context, const EGLint* attrib_list);
|
| +typedef EGLBoolean (*DestroyContextProc)(EGLDisplay dpy, EGLContext ctx);
|
| +typedef EGLBoolean (*MakeCurrentProc)(EGLDisplay dpy, EGLSurface draw, EGLSurface read, EGLContext ctx);
|
| +typedef EGLBoolean (*SwapBuffersProc)(EGLDisplay dpy, EGLSurface surface);
|
| +typedef __eglMustCastToProperFunctionPointerType (*GetProcAddressProc)(const char* procname);
|
| +
|
| +static GetDisplayProc gfGetDisplay = nullptr;
|
| +static InitializeProc gfInitialize = nullptr;
|
| +static TerminateProc gfTerminate = nullptr;
|
| +static ChooseConfigProc gfChooseConfig = nullptr;
|
| +static GetConfigAttrib gfGetConfigAttrib = nullptr;
|
| +static CreateWindowSurfaceProc gfCreateWindowSurface = nullptr;
|
| +static CreatePbufferSurfaceProc gfCreatePbufferSurface = nullptr;
|
| +static DestroySurfaceProc gfDestroySurface = nullptr;
|
| +static CreateContextProc gfCreateContext = nullptr;
|
| +static DestroyContextProc gfDestroyContext = nullptr;
|
| +static MakeCurrentProc gfMakeCurrent = nullptr;
|
| +static SwapBuffersProc gfSwapBuffers = nullptr;
|
| +static GetProcAddressProc gfGetProcAddress = nullptr;
|
| +
|
| +static HMODULE ghLibrary = nullptr;
|
| +static bool gfFunctionsLoadedSuccessfully = false;
|
| +
|
| +static void load_command_buffer_functions() {
|
| + if (!ghLibrary) {
|
| + ghLibrary = LoadLibrary("command_buffer_gles2.dll");
|
| +
|
| + if (ghLibrary) {
|
| + gfGetDisplay = (GetDisplayProc)::GetProcAddress(ghLibrary, "CommandBuffer_GetDisplay");
|
| + gfInitialize = (InitializeProc)::GetProcAddress(ghLibrary, "CommandBuffer_Initialize");
|
| + gfTerminate = (TerminateProc)::GetProcAddress(ghLibrary, "CommandBuffer_Terminate");
|
| + gfChooseConfig = (ChooseConfigProc)::GetProcAddress(ghLibrary, "CommandBuffer_ChooseConfig");
|
| + gfGetConfigAttrib = (GetConfigAttrib)::GetProcAddress(ghLibrary, "CommandBuffer_GetConfigAttrib");
|
| + gfCreateWindowSurface = (CreateWindowSurfaceProc)::GetProcAddress(ghLibrary, "CommandBuffer_CreateWindowSurface");
|
| + gfCreatePbufferSurface = (CreatePbufferSurfaceProc)::GetProcAddress(ghLibrary, "CommandBuffer_CreatePbufferSurface");
|
| + gfDestroySurface = (DestroySurfaceProc)::GetProcAddress(ghLibrary, "CommandBuffer_DestroySurface");
|
| + gfCreateContext = (CreateContextProc)::GetProcAddress(ghLibrary, "CommandBuffer_CreateContext");
|
| + gfDestroyContext = (DestroyContextProc)::GetProcAddress(ghLibrary, "CommandBuffer_DestroyContext");
|
| + gfMakeCurrent = (MakeCurrentProc)::GetProcAddress(ghLibrary, "CommandBuffer_MakeCurrent");
|
| + gfSwapBuffers = (SwapBuffersProc)::GetProcAddress(ghLibrary, "CommandBuffer_SwapBuffers");
|
| + gfGetProcAddress = (GetProcAddressProc)::GetProcAddress(ghLibrary, "CommandBuffer_GetProcAddress");
|
| +
|
| + gfFunctionsLoadedSuccessfully = gfGetDisplay && gfInitialize && gfTerminate &&
|
| + gfChooseConfig && gfCreateWindowSurface &&
|
| + gfCreatePbufferSurface && gfDestroySurface &&
|
| + gfCreateContext && gfDestroyContext && gfMakeCurrent &&
|
| + gfSwapBuffers && gfGetProcAddress;
|
| +
|
| + }
|
| + }
|
| +}
|
| +
|
| +static GrGLFuncPtr command_buffer_get_gl_proc(void* ctx, const char name[]) {
|
| + GrGLFuncPtr proc = (GrGLFuncPtr) GetProcAddress((HMODULE)ctx, name);
|
| + if (proc) {
|
| + return proc;
|
| + }
|
| + if (!gfFunctionsLoadedSuccessfully) {
|
| + return nullptr;
|
| + }
|
| + return gfGetProcAddress(name);
|
| +}
|
| +
|
| +SK_DECLARE_STATIC_ONCE(loadCommandBufferOnce);
|
| +void LoadCommandBufferOnce() {
|
| + SkOnce(&loadCommandBufferOnce, load_command_buffer_functions);
|
| +}
|
| +
|
| +const GrGLInterface* GrGLCreateCommandBufferInterface() {
|
| + LoadCommandBufferOnce();
|
| + if (!gfFunctionsLoadedSuccessfully) {
|
| + return nullptr;
|
| + }
|
| + return GrGLAssembleGLESInterface(ghLibrary, command_buffer_get_gl_proc);
|
| +}
|
| +
|
| +SkCommandBufferGLContext::SkCommandBufferGLContext()
|
| + : fContext(EGL_NO_CONTEXT)
|
| + , fDisplay(EGL_NO_DISPLAY)
|
| + , fSurface(EGL_NO_SURFACE) {
|
| +
|
| + static const EGLint configAttribs[] = {
|
| + EGL_SURFACE_TYPE, EGL_PBUFFER_BIT,
|
| + EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
| + EGL_RED_SIZE, 8,
|
| + EGL_GREEN_SIZE, 8,
|
| + EGL_BLUE_SIZE, 8,
|
| + EGL_ALPHA_SIZE, 8,
|
| + EGL_NONE
|
| + };
|
| +
|
| + static const EGLint surfaceAttribs[] = {
|
| + EGL_WIDTH, 1,
|
| + EGL_HEIGHT, 1,
|
| + EGL_NONE
|
| + };
|
| +
|
| + initializeGLContext(nullptr, configAttribs, surfaceAttribs);
|
| +}
|
| +
|
| +SkCommandBufferGLContext::SkCommandBufferGLContext(void* nativeWindow, int msaaSampleCount) {
|
| + static const EGLint surfaceAttribs[] = { EGL_NONE };
|
| +
|
| + EGLint configAttribs[] = {
|
| + EGL_RED_SIZE, 8,
|
| + EGL_GREEN_SIZE, 8,
|
| + EGL_BLUE_SIZE, 8,
|
| + EGL_ALPHA_SIZE, 8,
|
| + EGL_DEPTH_SIZE, 8,
|
| + EGL_STENCIL_SIZE, 8,
|
| + EGL_SAMPLE_BUFFERS, 1,
|
| + EGL_SAMPLES, msaaSampleCount,
|
| + EGL_NONE
|
| + };
|
| + if (msaaSampleCount == 0) {
|
| + configAttribs[12] = EGL_NONE;
|
| + }
|
| +
|
| + initializeGLContext(nativeWindow, configAttribs, surfaceAttribs);
|
| +}
|
| +
|
| +void SkCommandBufferGLContext::initializeGLContext(void* nativeWindow, const int* configAttribs,
|
| + const int* surfaceAttribs) {
|
| + LoadCommandBufferOnce();
|
| + if (!gfFunctionsLoadedSuccessfully) {
|
| + return;
|
| + }
|
| +
|
| + fDisplay = gfGetDisplay(static_cast<EGLNativeDisplayType>(EGL_DEFAULT_DISPLAY));
|
| + if (EGL_NO_DISPLAY == fDisplay) {
|
| + SkDebugf("Could not create EGL display!");
|
| + return;
|
| + }
|
| +
|
| + EGLint majorVersion;
|
| + EGLint minorVersion;
|
| + gfInitialize(fDisplay, &majorVersion, &minorVersion);
|
| +
|
| + EGLConfig surfaceConfig = static_cast<EGLConfig>(fConfig);
|
| + EGLint numConfigs;
|
| + gfChooseConfig(fDisplay, configAttribs, &surfaceConfig, 1, &numConfigs);
|
| +
|
| + if (nativeWindow) {
|
| + fSurface = gfCreateWindowSurface(fDisplay, surfaceConfig,
|
| + (EGLNativeWindowType)nativeWindow, surfaceAttribs);
|
| + } else {
|
| + fSurface = gfCreatePbufferSurface(fDisplay, surfaceConfig, surfaceAttribs);
|
| + }
|
| +
|
| + static const EGLint contextAttribs[] = {
|
| + EGL_CONTEXT_CLIENT_VERSION, 2,
|
| + EGL_NONE
|
| + };
|
| + fContext = gfCreateContext(fDisplay, surfaceConfig, nullptr, contextAttribs);
|
| +
|
| + gfMakeCurrent(fDisplay, fSurface, fSurface, fContext);
|
| +
|
| + SkAutoTUnref<const GrGLInterface> gl(GrGLCreateCommandBufferInterface());
|
| + if (nullptr == gl.get()) {
|
| + SkDebugf("Could not create CommandBuffer GL interface!\n");
|
| + this->destroyGLContext();
|
| + return;
|
| + }
|
| + if (!gl->validate()) {
|
| + SkDebugf("Could not validate CommandBuffer GL interface!\n");
|
| + this->destroyGLContext();
|
| + return;
|
| + }
|
| +
|
| + this->init(gl.detach());
|
| +}
|
| +
|
| +SkCommandBufferGLContext::~SkCommandBufferGLContext() {
|
| + this->teardown();
|
| + this->destroyGLContext();
|
| +}
|
| +
|
| +void SkCommandBufferGLContext::destroyGLContext() {
|
| + if (!gfFunctionsLoadedSuccessfully) {
|
| + return;
|
| + }
|
| + if (fDisplay) {
|
| + gfMakeCurrent(fDisplay, 0, 0, 0);
|
| +
|
| + if (fContext) {
|
| + gfDestroyContext(fDisplay, fContext);
|
| + fContext = EGL_NO_CONTEXT;
|
| + }
|
| +
|
| + if (fSurface) {
|
| + gfDestroySurface(fDisplay, fSurface);
|
| + fSurface = EGL_NO_SURFACE;
|
| + }
|
| +
|
| + gfTerminate(fDisplay);
|
| + fDisplay = EGL_NO_DISPLAY;
|
| + }
|
| +}
|
| +
|
| +void SkCommandBufferGLContext::onPlatformMakeCurrent() const {
|
| + if (!gfFunctionsLoadedSuccessfully) {
|
| + return;
|
| + }
|
| + if (!gfMakeCurrent(fDisplay, fSurface, fSurface, fContext)) {
|
| + SkDebugf("Could not set the context.\n");
|
| + }
|
| +}
|
| +
|
| +void SkCommandBufferGLContext::onPlatformSwapBuffers() const {
|
| + if (!gfFunctionsLoadedSuccessfully) {
|
| + return;
|
| + }
|
| + if (!gfSwapBuffers(fDisplay, fSurface)) {
|
| + SkDebugf("Could not complete gfSwapBuffers.\n");
|
| + }
|
| +}
|
| +
|
| +GrGLFuncPtr SkCommandBufferGLContext::onPlatformGetProcAddress(const char* name) const {
|
| + if (!gfFunctionsLoadedSuccessfully) {
|
| + return nullptr;
|
| + }
|
| + return gfGetProcAddress(name);
|
| +}
|
| +
|
| +void SkCommandBufferGLContext::presentCommandBuffer() {
|
| + if (this->gl()) {
|
| + this->gl()->fFunctions.fFlush();
|
| + }
|
| +
|
| + this->onPlatformSwapBuffers();
|
| +}
|
| +
|
| +bool SkCommandBufferGLContext::makeCurrent() {
|
| + return gfMakeCurrent(fDisplay, fSurface, fSurface, fContext) != EGL_FALSE;
|
| +}
|
| +
|
| +int SkCommandBufferGLContext::getStencilBits() {
|
| + EGLint result = 0;
|
| + EGLConfig surfaceConfig = static_cast<EGLConfig>(fConfig);
|
| + gfGetConfigAttrib(fDisplay, surfaceConfig, EGL_STENCIL_SIZE, &result);
|
| + return result;
|
| +}
|
| +
|
| +int SkCommandBufferGLContext::getSampleCount() {
|
| + EGLint result = 0;
|
| + EGLConfig surfaceConfig = static_cast<EGLConfig>(fConfig);
|
| + gfGetConfigAttrib(fDisplay, surfaceConfig, EGL_SAMPLES, &result);
|
| + return result;
|
| +}
|
|
|