2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2012 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2014-02-17 05:18:15 -05:00
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2015-10-05 22:45:55 -05:00
|
|
|
#include <array>
|
|
|
|
#include <cstdlib>
|
2016-02-05 10:50:08 -06:00
|
|
|
#include <sstream>
|
|
|
|
#include <vector>
|
2015-10-05 22:45:55 -05:00
|
|
|
|
2015-09-19 04:40:00 +12:00
|
|
|
#include "Common/GL/GLInterface/EGL.h"
|
2015-09-26 17:13:07 -04:00
|
|
|
#include "Common/Logging/Log.h"
|
2017-07-01 10:42:42 -07:00
|
|
|
#include "Core/Config/GraphicsSettings.h"
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2016-01-23 06:22:14 -06:00
|
|
|
#ifndef EGL_KHR_create_context
|
|
|
|
#define EGL_KHR_create_context 1
|
2016-06-24 10:43:46 +02:00
|
|
|
#define EGL_CONTEXT_MAJOR_VERSION_KHR 0x3098
|
|
|
|
#define EGL_CONTEXT_MINOR_VERSION_KHR 0x30FB
|
|
|
|
#define EGL_CONTEXT_FLAGS_KHR 0x30FC
|
2016-01-23 06:22:14 -06:00
|
|
|
#define EGL_CONTEXT_OPENGL_PROFILE_MASK_KHR 0x30FD
|
|
|
|
#define EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_KHR 0x31BD
|
2016-06-24 10:43:46 +02:00
|
|
|
#define EGL_NO_RESET_NOTIFICATION_KHR 0x31BE
|
|
|
|
#define EGL_LOSE_CONTEXT_ON_RESET_KHR 0x31BF
|
|
|
|
#define EGL_CONTEXT_OPENGL_DEBUG_BIT_KHR 0x00000001
|
2016-01-23 06:22:14 -06:00
|
|
|
#define EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR 0x00000002
|
|
|
|
#define EGL_CONTEXT_OPENGL_ROBUST_ACCESS_BIT_KHR 0x00000004
|
|
|
|
#define EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT_KHR 0x00000001
|
|
|
|
#define EGL_CONTEXT_OPENGL_COMPATIBILITY_PROFILE_BIT_KHR 0x00000002
|
2016-06-24 10:43:46 +02:00
|
|
|
#define EGL_OPENGL_ES3_BIT_KHR 0x00000040
|
2016-01-23 06:22:14 -06:00
|
|
|
#endif /* EGL_KHR_create_context */
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
GLContextEGL::~GLContextEGL() = default;
|
|
|
|
|
|
|
|
bool GLContextEGL::IsHeadless() const
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
return m_host_window == nullptr;
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
2018-10-03 23:02:45 +10:00
|
|
|
|
|
|
|
void GLContextEGL::Swap()
|
2013-03-24 21:06:34 -05:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
if (m_egl_surface != EGL_NO_SURFACE)
|
|
|
|
eglSwapBuffers(m_egl_display, m_egl_surface);
|
|
|
|
}
|
|
|
|
void GLContextEGL::SwapInterval(int interval)
|
|
|
|
{
|
|
|
|
eglSwapInterval(m_egl_display, interval);
|
2013-03-24 21:06:34 -05:00
|
|
|
}
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
void* GLContextEGL::GetFuncAddress(const std::string& name)
|
2013-12-30 07:22:50 -06:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
return (void*)eglGetProcAddress(name.c_str());
|
2013-12-30 07:22:50 -06:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
void GLContextEGL::DetectMode(bool has_handle)
|
2014-01-18 04:11:59 +00:00
|
|
|
{
|
2017-07-01 10:42:42 -07:00
|
|
|
bool preferGLES = Config::Get(Config::GFX_PREFER_GLES);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
EGLint num_configs;
|
2018-10-03 23:02:45 +10:00
|
|
|
bool supportsGL = false, supportsGLES3 = false;
|
|
|
|
std::array<int, 3> renderable_types{{EGL_OPENGL_BIT, EGL_OPENGL_ES3_BIT_KHR}};
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
for (auto renderable_type : renderable_types)
|
|
|
|
{
|
|
|
|
// attributes for a visual in RGBA format with at least
|
|
|
|
// 8 bits per color
|
2017-12-29 23:20:16 -05:00
|
|
|
int attribs[] = {EGL_RED_SIZE,
|
|
|
|
8,
|
|
|
|
EGL_GREEN_SIZE,
|
|
|
|
8,
|
|
|
|
EGL_BLUE_SIZE,
|
|
|
|
8,
|
|
|
|
EGL_RENDERABLE_TYPE,
|
|
|
|
renderable_type,
|
|
|
|
EGL_SURFACE_TYPE,
|
2018-10-03 23:02:45 +10:00
|
|
|
has_handle ? EGL_WINDOW_BIT : 0,
|
2016-06-24 10:43:46 +02:00
|
|
|
EGL_NONE};
|
|
|
|
|
|
|
|
// Get how many configs there are
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!eglChooseConfig(m_egl_display, attribs, nullptr, 0, &num_configs))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: couldn't get an EGL visual config");
|
2016-06-24 10:43:46 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
EGLConfig* config = new EGLConfig[num_configs];
|
|
|
|
|
|
|
|
// Get all the configurations
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!eglChooseConfig(m_egl_display, attribs, config, num_configs, &num_configs))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: couldn't get an EGL visual config");
|
2016-06-24 10:43:46 +02:00
|
|
|
delete[] config;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < num_configs; ++i)
|
|
|
|
{
|
|
|
|
EGLint attribVal;
|
|
|
|
bool ret;
|
2018-10-03 23:02:45 +10:00
|
|
|
ret = eglGetConfigAttrib(m_egl_display, config[i], EGL_RENDERABLE_TYPE, &attribVal);
|
2016-06-24 10:43:46 +02:00
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
if (attribVal & EGL_OPENGL_BIT)
|
|
|
|
supportsGL = true;
|
2018-10-03 23:02:45 +10:00
|
|
|
if (attribVal & EGL_OPENGL_ES3_BIT_KHR)
|
2016-06-24 10:43:46 +02:00
|
|
|
supportsGLES3 = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete[] config;
|
|
|
|
}
|
|
|
|
|
2017-07-01 10:42:42 -07:00
|
|
|
if (preferGLES)
|
|
|
|
{
|
|
|
|
if (supportsGLES3)
|
2018-10-03 23:02:45 +10:00
|
|
|
m_opengl_mode = Mode::OpenGLES;
|
2017-07-01 10:42:42 -07:00
|
|
|
else if (supportsGL)
|
2018-10-03 23:02:45 +10:00
|
|
|
m_opengl_mode = Mode::OpenGL;
|
2017-07-01 10:42:42 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (supportsGL)
|
2018-10-03 23:02:45 +10:00
|
|
|
m_opengl_mode = Mode::OpenGL;
|
2017-07-01 10:42:42 -07:00
|
|
|
else if (supportsGLES3)
|
2018-10-03 23:02:45 +10:00
|
|
|
m_opengl_mode = Mode::OpenGLES;
|
2017-07-01 10:42:42 -07:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (m_opengl_mode == Mode::OpenGL)
|
2017-07-01 10:42:42 -07:00
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Using OpenGL");
|
|
|
|
}
|
2018-10-03 23:02:45 +10:00
|
|
|
else if (m_opengl_mode == Mode::OpenGLES)
|
2017-07-01 10:42:42 -07:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
INFO_LOG(VIDEO, "Using OpenGL|ES");
|
2017-07-01 10:42:42 -07:00
|
|
|
}
|
2018-10-03 23:02:45 +10:00
|
|
|
else if (m_opengl_mode == Mode::Detect)
|
2017-07-01 10:42:42 -07:00
|
|
|
{
|
|
|
|
// Errored before we found a mode
|
|
|
|
ERROR_LOG(VIDEO, "Error: Failed to detect OpenGL flavour, falling back to OpenGL");
|
|
|
|
// This will fail to create a context, as it'll try to use the same attribs we just failed to
|
|
|
|
// find a matching config with
|
2018-10-03 23:02:45 +10:00
|
|
|
m_opengl_mode = Mode::OpenGL;
|
2017-07-01 10:42:42 -07:00
|
|
|
}
|
2014-01-18 04:11:59 +00:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
EGLDisplay GLContextEGL::OpenEGLDisplay()
|
|
|
|
{
|
|
|
|
return eglGetDisplay(EGL_DEFAULT_DISPLAY);
|
|
|
|
}
|
|
|
|
|
|
|
|
EGLNativeWindowType GLContextEGL::GetEGLNativeWindow(EGLConfig config)
|
|
|
|
{
|
|
|
|
return reinterpret_cast<EGLNativeWindowType>(EGL_DEFAULT_DISPLAY);
|
|
|
|
}
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
// Create rendering window.
|
2014-02-16 23:51:41 -05:00
|
|
|
// Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize()
|
2018-10-03 23:03:19 +10:00
|
|
|
bool GLContextEGL::Initialize(void* display_handle, void* window_handle, bool stereo, bool core)
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
const bool has_handle = !!window_handle;
|
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
EGLint egl_major, egl_minor;
|
|
|
|
bool supports_core_profile = false;
|
|
|
|
|
2018-10-03 23:03:19 +10:00
|
|
|
m_host_display = display_handle;
|
2018-10-03 23:02:45 +10:00
|
|
|
m_host_window = window_handle;
|
2018-10-03 23:03:19 +10:00
|
|
|
m_egl_display = OpenEGLDisplay();
|
2018-10-03 23:02:45 +10:00
|
|
|
m_is_core_context = core;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!m_egl_display)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: eglGetDisplay() failed");
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!eglInitialize(m_egl_display, &egl_major, &egl_minor))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: eglInitialize() failed");
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Detection code */
|
|
|
|
EGLint num_configs;
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (m_opengl_mode == Mode::Detect)
|
|
|
|
DetectMode(has_handle);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
// attributes for a visual in RGBA format with at least
|
|
|
|
// 8 bits per color
|
|
|
|
int attribs[] = {EGL_RENDERABLE_TYPE,
|
2018-10-03 23:02:45 +10:00
|
|
|
0,
|
2016-06-24 10:43:46 +02:00
|
|
|
EGL_RED_SIZE,
|
|
|
|
8,
|
|
|
|
EGL_GREEN_SIZE,
|
|
|
|
8,
|
|
|
|
EGL_BLUE_SIZE,
|
|
|
|
8,
|
2017-12-29 23:20:16 -05:00
|
|
|
EGL_SURFACE_TYPE,
|
2018-10-03 23:02:45 +10:00
|
|
|
has_handle ? EGL_WINDOW_BIT : 0,
|
2016-06-24 10:43:46 +02:00
|
|
|
EGL_NONE};
|
|
|
|
|
|
|
|
std::vector<EGLint> ctx_attribs;
|
2018-10-03 23:02:45 +10:00
|
|
|
switch (m_opengl_mode)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
case Mode::OpenGL:
|
2016-06-24 10:43:46 +02:00
|
|
|
attribs[1] = EGL_OPENGL_BIT;
|
|
|
|
ctx_attribs = {EGL_NONE};
|
|
|
|
break;
|
2018-10-03 23:02:45 +10:00
|
|
|
case Mode::OpenGLES:
|
|
|
|
attribs[1] = EGL_OPENGL_ES3_BIT_KHR;
|
2016-06-24 10:43:46 +02:00
|
|
|
ctx_attribs = {EGL_CONTEXT_CLIENT_VERSION, 3, EGL_NONE};
|
|
|
|
break;
|
|
|
|
default:
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(VIDEO, "Unknown opengl mode set");
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!eglChooseConfig(m_egl_display, attribs, &m_config, 1, &num_configs))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: couldn't get an EGL visual config");
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (m_opengl_mode == Mode::OpenGL)
|
2016-06-24 10:43:46 +02:00
|
|
|
eglBindAPI(EGL_OPENGL_API);
|
|
|
|
else
|
|
|
|
eglBindAPI(EGL_OPENGL_ES_API);
|
|
|
|
|
|
|
|
std::string tmp;
|
2018-10-03 23:02:45 +10:00
|
|
|
std::istringstream buffer(eglQueryString(m_egl_display, EGL_EXTENSIONS));
|
2016-06-24 10:43:46 +02:00
|
|
|
while (buffer >> tmp)
|
|
|
|
{
|
|
|
|
if (tmp == "EGL_KHR_surfaceless_context")
|
|
|
|
m_supports_surfaceless = true;
|
|
|
|
else if (tmp == "EGL_KHR_create_context")
|
|
|
|
supports_core_profile = true;
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (supports_core_profile && core && m_opengl_mode == Mode::OpenGL)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
|
|
|
std::array<std::pair<int, int>, 7> versions_to_try = {{
|
2018-04-12 14:18:04 +02:00
|
|
|
{4, 5},
|
|
|
|
{4, 4},
|
|
|
|
{4, 3},
|
|
|
|
{4, 2},
|
|
|
|
{4, 1},
|
|
|
|
{4, 0},
|
|
|
|
{3, 3},
|
2016-06-24 10:43:46 +02:00
|
|
|
}};
|
|
|
|
|
|
|
|
for (const auto& version : versions_to_try)
|
|
|
|
{
|
|
|
|
std::vector<EGLint> core_attribs = {EGL_CONTEXT_MAJOR_VERSION_KHR,
|
|
|
|
version.first,
|
|
|
|
EGL_CONTEXT_MINOR_VERSION_KHR,
|
|
|
|
version.second,
|
|
|
|
EGL_CONTEXT_OPENGL_PROFILE_MASK_KHR,
|
|
|
|
EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT_KHR,
|
|
|
|
EGL_CONTEXT_FLAGS_KHR,
|
|
|
|
EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR,
|
|
|
|
EGL_NONE};
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
m_egl_context = eglCreateContext(m_egl_display, m_config, EGL_NO_CONTEXT, &core_attribs[0]);
|
|
|
|
if (m_egl_context)
|
2017-07-27 11:57:32 +10:00
|
|
|
{
|
|
|
|
m_attribs = std::move(core_attribs);
|
2016-06-24 10:43:46 +02:00
|
|
|
break;
|
2017-07-27 11:57:32 +10:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!m_egl_context)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
m_is_core_context = false;
|
|
|
|
m_egl_context = eglCreateContext(m_egl_display, m_config, EGL_NO_CONTEXT, &ctx_attribs[0]);
|
2017-07-27 11:57:32 +10:00
|
|
|
m_attribs = std::move(ctx_attribs);
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!m_egl_context)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: eglCreateContext failed");
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CreateWindowSurface())
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(VIDEO, "Error: CreateWindowSurface failed 0x%04x", eglGetError());
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2016-01-21 19:03:58 -06:00
|
|
|
}
|
2016-01-10 12:28:05 -06:00
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
std::unique_ptr<GLContext> GLContextEGL::CreateSharedContext()
|
2016-01-21 19:03:58 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
std::unique_ptr<GLContextEGL> context = std::make_unique<GLContextEGL>();
|
|
|
|
if (!context->Initialize(this))
|
2016-06-24 10:43:46 +02:00
|
|
|
return nullptr;
|
|
|
|
return context;
|
2016-01-21 19:03:58 -06:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
bool GLContextEGL::Initialize(GLContext* main_context)
|
2016-01-21 19:03:58 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
GLContextEGL* egl_context = static_cast<GLContextEGL*>(main_context);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
m_opengl_mode = egl_context->m_opengl_mode;
|
2018-10-03 23:03:19 +10:00
|
|
|
m_host_display = egl_context->m_host_display;
|
2018-10-03 23:02:45 +10:00
|
|
|
m_egl_display = egl_context->m_egl_display;
|
|
|
|
m_is_core_context = egl_context->m_is_core_context;
|
2016-06-24 10:43:46 +02:00
|
|
|
m_config = egl_context->m_config;
|
|
|
|
m_supports_surfaceless = egl_context->m_supports_surfaceless;
|
|
|
|
m_is_shared = true;
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
if (egl_context->GetMode() == Mode::OpenGL)
|
2016-06-24 10:43:46 +02:00
|
|
|
eglBindAPI(EGL_OPENGL_API);
|
|
|
|
else
|
|
|
|
eglBindAPI(EGL_OPENGL_ES_API);
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
m_egl_context = eglCreateContext(m_egl_display, m_config, egl_context->m_egl_context,
|
|
|
|
egl_context->m_attribs.data());
|
|
|
|
if (!m_egl_context)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: eglCreateContext failed 0x%04x", eglGetError());
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CreateWindowSurface())
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(VIDEO, "Error: CreateWindowSurface failed 0x%04x", eglGetError());
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
bool GLContextEGL::CreateWindowSurface()
|
2016-01-10 12:28:05 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
if (m_host_window)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
EGLNativeWindowType native_window = GetEGLNativeWindow(m_config);
|
|
|
|
m_egl_surface = eglCreateWindowSurface(m_egl_display, m_config, native_window, nullptr);
|
|
|
|
if (!m_egl_surface)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: eglCreateWindowSurface failed");
|
2016-06-24 10:43:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!m_supports_surfaceless)
|
|
|
|
{
|
|
|
|
EGLint attrib_list[] = {
|
|
|
|
EGL_NONE,
|
|
|
|
};
|
2018-10-03 23:02:45 +10:00
|
|
|
m_egl_surface = eglCreatePbufferSurface(m_egl_display, m_config, attrib_list);
|
|
|
|
if (!m_egl_surface)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Error: eglCreatePbufferSurface failed");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
m_egl_surface = EGL_NO_SURFACE;
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
return true;
|
2016-01-10 12:28:05 -06:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
void GLContextEGL::DestroyWindowSurface()
|
2016-01-10 12:28:05 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
if (m_egl_surface != EGL_NO_SURFACE && !eglDestroySurface(m_egl_display, m_egl_surface))
|
2016-06-24 10:43:46 +02:00
|
|
|
NOTICE_LOG(VIDEO, "Could not destroy window surface.");
|
2018-10-03 23:02:45 +10:00
|
|
|
m_egl_surface = EGL_NO_SURFACE;
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
2015-09-04 19:58:58 -05:00
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
bool GLContextEGL::MakeCurrent()
|
2016-01-10 12:28:05 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
return eglMakeCurrent(m_egl_display, m_egl_surface, m_egl_surface, m_egl_context);
|
2016-01-10 12:28:05 -06:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
void GLContextEGL::UpdateSurface(void* window_handle)
|
2016-01-10 12:28:05 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
m_host_window = window_handle;
|
2016-06-24 10:43:46 +02:00
|
|
|
ClearCurrent();
|
|
|
|
DestroyWindowSurface();
|
|
|
|
CreateWindowSurface();
|
|
|
|
MakeCurrent();
|
2016-01-10 12:28:05 -06:00
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
bool GLContextEGL::ClearCurrent()
|
2015-09-04 19:58:58 -05:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
return eglMakeCurrent(m_egl_display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
2015-09-04 19:58:58 -05:00
|
|
|
}
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
// Close backend
|
2018-10-03 23:02:45 +10:00
|
|
|
void GLContextEGL::Shutdown()
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
if (m_egl_context)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!eglMakeCurrent(m_egl_display, m_egl_surface, m_egl_surface, m_egl_context))
|
2016-06-24 10:43:46 +02:00
|
|
|
NOTICE_LOG(VIDEO, "Could not release drawing context.");
|
2018-10-03 23:02:45 +10:00
|
|
|
eglMakeCurrent(m_egl_display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
|
|
|
if (!eglDestroyContext(m_egl_display, m_egl_context))
|
2016-06-24 10:43:46 +02:00
|
|
|
NOTICE_LOG(VIDEO, "Could not destroy drawing context.");
|
|
|
|
DestroyWindowSurface();
|
2018-10-03 23:02:45 +10:00
|
|
|
if (!m_is_shared && !eglTerminate(m_egl_display))
|
2016-06-24 10:43:46 +02:00
|
|
|
NOTICE_LOG(VIDEO, "Could not destroy display connection.");
|
2018-10-03 23:02:45 +10:00
|
|
|
m_egl_context = nullptr;
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|