2017-07-04 16:55:08 -04:00
|
|
|
// SuperTuxKart - a fun racing game with go-kart
|
|
|
|
// Copyright (C) 2016-2017 SuperTuxKart-Team
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License
|
|
|
|
// as published by the Free Software Foundation; either version 3
|
|
|
|
// of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
#include "IrrCompileConfig.h"
|
|
|
|
|
|
|
|
#if defined(_IRR_COMPILE_WITH_EGL_)
|
|
|
|
|
|
|
|
#include <cassert>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstring>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#ifdef _IRR_COMPILE_WITH_ANDROID_DEVICE_
|
2018-07-26 18:05:39 -04:00
|
|
|
#include "stk_android_native_app_glue.h"
|
2017-04-20 17:20:21 -04:00
|
|
|
#endif
|
|
|
|
|
2017-04-05 14:28:38 -04:00
|
|
|
#include "CContextEGL.h"
|
|
|
|
#include "os.h"
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
using namespace irr;
|
2017-04-05 15:46:37 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
ContextManagerEGL::ContextManagerEGL()
|
|
|
|
{
|
|
|
|
m_egl_window = 0;
|
|
|
|
m_egl_display = EGL_NO_DISPLAY;
|
|
|
|
m_egl_surface = EGL_NO_SURFACE;
|
|
|
|
m_egl_context = EGL_NO_CONTEXT;
|
|
|
|
m_egl_config = 0;
|
|
|
|
m_egl_version = 0;
|
|
|
|
m_is_legacy_device = false;
|
|
|
|
m_initialized = false;
|
2018-01-06 20:16:13 -05:00
|
|
|
eglGetPlatformDisplay = NULL;
|
2017-04-21 02:51:54 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
memset(&m_creation_params, 0, sizeof(ContextEGLParams));
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
|
|
|
|
ContextManagerEGL::~ContextManagerEGL()
|
2017-04-05 14:28:38 -04:00
|
|
|
{
|
2017-04-20 17:20:21 -04:00
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ContextManagerEGL::init(const ContextEGLParams& params)
|
2017-04-05 14:28:38 -04:00
|
|
|
{
|
2017-04-20 17:20:21 -04:00
|
|
|
if (m_initialized)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_creation_params = params;
|
|
|
|
m_egl_window = m_creation_params.window;
|
2018-01-06 20:16:13 -05:00
|
|
|
|
|
|
|
bool success = initExtensions();
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: Could not initialize EGL extensions.\n");
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-20 17:20:21 -04:00
|
|
|
|
2018-01-06 20:16:13 -05:00
|
|
|
success = initDisplay();
|
2017-04-20 17:20:21 -04:00
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: Could not initialize EGL display.\n");
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool has_minimum_requirements = false;
|
|
|
|
|
|
|
|
if (m_creation_params.opengl_api == CEGL_API_OPENGL)
|
|
|
|
{
|
|
|
|
if (hasEGLExtension("EGL_KHR_create_context") || m_egl_version >= 150)
|
|
|
|
{
|
|
|
|
has_minimum_requirements = true;
|
|
|
|
eglBindAPI(EGL_OPENGL_API);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (m_creation_params.opengl_api == CEGL_API_OPENGL_ES)
|
|
|
|
{
|
|
|
|
if (m_egl_version >= 130)
|
|
|
|
{
|
|
|
|
has_minimum_requirements = true;
|
|
|
|
eglBindAPI(EGL_OPENGL_ES_API);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_minimum_requirements)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: EGL version is too old.\n");
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
success = chooseConfig();
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: Couldn't get EGL config.\n");
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
success = createSurface();
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: Couldn't create EGL surface.\n");
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
success = createContext();
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: Couldn't create OpenGL context.\n");
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
success = eglMakeCurrent(m_egl_display, m_egl_surface, m_egl_surface,
|
|
|
|
m_egl_context);
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
checkEGLError();
|
|
|
|
os::Printer::log("Error: Couldn't make context current for EGL display.\n");
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
eglSwapInterval(m_egl_display, m_creation_params.vsync_enabled ? 1 : 0);
|
|
|
|
|
|
|
|
m_initialized = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-06 20:16:13 -05:00
|
|
|
bool ContextManagerEGL::initExtensions()
|
|
|
|
{
|
|
|
|
if (hasEGLExtension("EGL_KHR_platform_base"))
|
|
|
|
{
|
|
|
|
eglGetPlatformDisplay = (eglGetPlatformDisplay_t)
|
|
|
|
eglGetProcAddress("eglGetPlatformDisplay");
|
|
|
|
}
|
|
|
|
else if (hasEGLExtension("EGL_EXT_platform_base"))
|
|
|
|
{
|
|
|
|
eglGetPlatformDisplay = (eglGetPlatformDisplay_t)
|
|
|
|
eglGetProcAddress("eglGetPlatformDisplayEXT");
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool ContextManagerEGL::initDisplay()
|
2017-04-05 14:28:38 -04:00
|
|
|
{
|
2017-04-21 02:51:54 -04:00
|
|
|
EGLNativeDisplayType display = m_creation_params.display;
|
2017-04-20 17:20:21 -04:00
|
|
|
|
|
|
|
#ifdef _IRR_COMPILE_WITH_ANDROID_DEVICE_
|
|
|
|
display = EGL_DEFAULT_DISPLAY;
|
2017-04-05 14:28:38 -04:00
|
|
|
#endif
|
|
|
|
|
2018-01-06 17:06:12 -05:00
|
|
|
EGLenum platform = 0;
|
|
|
|
|
|
|
|
switch (m_creation_params.platform)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
2018-01-06 17:06:12 -05:00
|
|
|
case CEGL_PLATFORM_ANDROID:
|
|
|
|
platform = EGL_PLATFORM_ANDROID;
|
|
|
|
break;
|
|
|
|
case CEGL_PLATFORM_GBM:
|
|
|
|
platform = EGL_PLATFORM_GBM;
|
|
|
|
break;
|
|
|
|
case CEGL_PLATFORM_WAYLAND:
|
|
|
|
platform = EGL_PLATFORM_WAYLAND;
|
|
|
|
break;
|
|
|
|
case CEGL_PLATFORM_X11:
|
|
|
|
platform = EGL_PLATFORM_X11;
|
|
|
|
break;
|
|
|
|
case CEGL_PLATFORM_DEFAULT:
|
|
|
|
break;
|
|
|
|
}
|
2018-01-06 20:16:13 -05:00
|
|
|
|
|
|
|
if (m_creation_params.platform != CEGL_PLATFORM_DEFAULT &&
|
|
|
|
eglGetPlatformDisplay != NULL)
|
2018-01-06 17:06:12 -05:00
|
|
|
{
|
2018-01-06 20:16:13 -05:00
|
|
|
m_egl_display = eglGetPlatformDisplay(platform, (void*)display, NULL);
|
2017-04-20 17:20:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_display == EGL_NO_DISPLAY)
|
2018-01-06 17:06:12 -05:00
|
|
|
{
|
|
|
|
m_egl_display = eglGetDisplay(display);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_display == EGL_NO_DISPLAY && display != EGL_DEFAULT_DISPLAY)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
|
|
|
m_egl_display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_display == EGL_NO_DISPLAY)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int egl_version_major = 0;
|
|
|
|
int egl_version_minor = 0;
|
|
|
|
|
|
|
|
bool success = eglInitialize(m_egl_display, &egl_version_major,
|
|
|
|
&egl_version_minor);
|
|
|
|
|
2017-04-21 02:51:54 -04:00
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
m_egl_version = 100 * egl_version_major + 10 * egl_version_minor;
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ContextManagerEGL::chooseConfig()
|
|
|
|
{
|
|
|
|
std::vector<EGLint> config_attribs;
|
|
|
|
config_attribs.push_back(EGL_RED_SIZE);
|
|
|
|
config_attribs.push_back(8);
|
|
|
|
config_attribs.push_back(EGL_GREEN_SIZE);
|
|
|
|
config_attribs.push_back(8);
|
|
|
|
config_attribs.push_back(EGL_BLUE_SIZE);
|
|
|
|
config_attribs.push_back(8);
|
|
|
|
config_attribs.push_back(EGL_ALPHA_SIZE);
|
|
|
|
config_attribs.push_back(m_creation_params.with_alpha_channel ? 8 : 0);
|
|
|
|
config_attribs.push_back(EGL_DEPTH_SIZE);
|
|
|
|
config_attribs.push_back(16);
|
2017-06-24 15:39:58 -04:00
|
|
|
// config_attribs.push_back(EGL_BUFFER_SIZE);
|
|
|
|
// config_attribs.push_back(24);
|
2017-04-20 17:20:21 -04:00
|
|
|
// config_attribs.push_back(EGL_STENCIL_SIZE);
|
|
|
|
// config_attribs.push_back(stencil_buffer);
|
|
|
|
// config_attribs.push_back(EGL_SAMPLE_BUFFERS);
|
|
|
|
// config_attribs.push_back(antialias ? 1 : 0);
|
|
|
|
// config_attribs.push_back(EGL_SAMPLES);
|
|
|
|
// config_attribs.push_back(antialias);
|
|
|
|
|
|
|
|
if (m_creation_params.opengl_api == CEGL_API_OPENGL)
|
|
|
|
{
|
2017-04-21 02:51:54 -04:00
|
|
|
config_attribs.push_back(EGL_RENDERABLE_TYPE);
|
|
|
|
config_attribs.push_back(EGL_OPENGL_BIT);
|
|
|
|
}
|
2017-04-20 17:20:21 -04:00
|
|
|
else if (m_creation_params.opengl_api == CEGL_API_OPENGL_ES)
|
|
|
|
{
|
2017-04-21 02:51:54 -04:00
|
|
|
config_attribs.push_back(EGL_RENDERABLE_TYPE);
|
|
|
|
config_attribs.push_back(EGL_OPENGL_ES2_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_creation_params.surface_type == CEGL_SURFACE_WINDOW)
|
|
|
|
{
|
|
|
|
config_attribs.push_back(EGL_SURFACE_TYPE);
|
|
|
|
config_attribs.push_back(EGL_WINDOW_BIT);
|
|
|
|
}
|
|
|
|
else if (m_creation_params.surface_type == CEGL_SURFACE_PBUFFER)
|
|
|
|
{
|
|
|
|
config_attribs.push_back(EGL_SURFACE_TYPE);
|
|
|
|
config_attribs.push_back(EGL_PBUFFER_BIT);
|
|
|
|
}
|
2017-04-05 14:43:04 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
config_attribs.push_back(EGL_NONE);
|
|
|
|
config_attribs.push_back(0);
|
2017-04-05 14:43:04 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
EGLint num_configs = 0;
|
2017-04-05 14:43:04 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool success = eglChooseConfig(m_egl_display, &config_attribs[0],
|
|
|
|
&m_egl_config, 1, &num_configs);
|
2017-04-05 14:43:04 -04:00
|
|
|
|
2017-06-24 15:39:58 -04:00
|
|
|
if (!success || m_egl_config == NULL || num_configs < 1)
|
|
|
|
{
|
|
|
|
config_attribs[1] = 5; //EGL_RED_SIZE
|
|
|
|
config_attribs[3] = 6; //EGL_GREEN_SIZE
|
|
|
|
config_attribs[5] = 5; //EGL_BLUE_SIZE
|
|
|
|
config_attribs[7] = 0; //EGL_ALPHA_SIZE
|
|
|
|
config_attribs[9] = 1; //EGL_DEPTH_SIZE
|
|
|
|
|
|
|
|
success = eglChooseConfig(m_egl_display, &config_attribs[0],
|
2018-01-06 20:16:13 -05:00
|
|
|
&m_egl_config, 1, &num_configs);
|
2017-06-24 15:39:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!success || m_egl_config == NULL || num_configs < 1)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-05 14:43:04 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
#ifdef _IRR_COMPILE_WITH_ANDROID_DEVICE_
|
|
|
|
EGLint format = 0;
|
|
|
|
eglGetConfigAttrib(m_egl_display, m_egl_config, EGL_NATIVE_VISUAL_ID,
|
|
|
|
&format);
|
|
|
|
ANativeWindow_setBuffersGeometry(m_egl_window, 0, 0, format);
|
|
|
|
#endif
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
return true;
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool ContextManagerEGL::createSurface()
|
|
|
|
{
|
2017-09-17 15:51:07 -04:00
|
|
|
unsigned int colorspace_attr_pos = 0;
|
|
|
|
unsigned int largest_pbuffer_attr_pos = 0;
|
|
|
|
|
|
|
|
std::vector<EGLint> attribs;
|
|
|
|
|
2018-01-06 17:15:22 -05:00
|
|
|
if (m_creation_params.opengl_api == CEGL_API_OPENGL &&
|
|
|
|
m_creation_params.handle_srgb == true)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
2017-09-17 15:51:07 -04:00
|
|
|
if (hasEGLExtension("EGL_KHR_gl_colorspace") || m_egl_version >= 150)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
2017-09-17 15:51:07 -04:00
|
|
|
attribs.push_back(EGL_GL_COLORSPACE);
|
2018-01-06 17:15:22 -05:00
|
|
|
attribs.push_back(EGL_GL_COLORSPACE_SRGB);
|
2017-09-17 15:51:07 -04:00
|
|
|
colorspace_attr_pos = attribs.size() - 1;
|
2017-04-20 17:20:21 -04:00
|
|
|
}
|
2017-09-17 15:51:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (m_creation_params.surface_type == CEGL_SURFACE_PBUFFER)
|
|
|
|
{
|
|
|
|
attribs.push_back(EGL_WIDTH);
|
|
|
|
attribs.push_back(m_creation_params.pbuffer_width);
|
|
|
|
attribs.push_back(EGL_HEIGHT);
|
|
|
|
attribs.push_back(m_creation_params.pbuffer_height);
|
|
|
|
attribs.push_back(EGL_LARGEST_PBUFFER);
|
|
|
|
attribs.push_back(EGL_FALSE);
|
|
|
|
largest_pbuffer_attr_pos = attribs.size() - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
attribs.push_back(EGL_NONE);
|
|
|
|
attribs.push_back(0);
|
|
|
|
|
|
|
|
if (m_egl_surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
if (m_creation_params.surface_type == CEGL_SURFACE_WINDOW)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
|
|
|
m_egl_surface = eglCreateWindowSurface(m_egl_display, m_egl_config,
|
2017-09-17 15:51:07 -04:00
|
|
|
m_egl_window, &attribs[0]);
|
|
|
|
}
|
|
|
|
else if (m_creation_params.surface_type == CEGL_SURFACE_PBUFFER)
|
|
|
|
{
|
|
|
|
m_egl_surface = eglCreatePbufferSurface(m_egl_display, m_egl_config,
|
|
|
|
&attribs[0]);
|
2017-04-20 17:20:21 -04:00
|
|
|
}
|
|
|
|
}
|
2017-09-17 15:51:07 -04:00
|
|
|
|
2018-01-06 17:15:22 -05:00
|
|
|
if (m_egl_surface == EGL_NO_SURFACE && colorspace_attr_pos > 0)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
2017-09-17 15:51:07 -04:00
|
|
|
attribs[colorspace_attr_pos] = EGL_GL_COLORSPACE_LINEAR;
|
|
|
|
|
|
|
|
if (m_creation_params.surface_type == CEGL_SURFACE_WINDOW)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
2017-09-17 15:51:07 -04:00
|
|
|
m_egl_surface = eglCreateWindowSurface(m_egl_display, m_egl_config,
|
|
|
|
m_egl_window, &attribs[0]);
|
2017-04-20 17:20:21 -04:00
|
|
|
}
|
2017-09-17 15:51:07 -04:00
|
|
|
else if (m_creation_params.surface_type == CEGL_SURFACE_PBUFFER)
|
|
|
|
{
|
|
|
|
m_egl_surface = eglCreatePbufferSurface(m_egl_display, m_egl_config,
|
|
|
|
&attribs[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_surface == EGL_NO_SURFACE && largest_pbuffer_attr_pos > 0)
|
|
|
|
{
|
|
|
|
attribs[largest_pbuffer_attr_pos] = EGL_TRUE;
|
|
|
|
|
|
|
|
if (m_creation_params.surface_type == CEGL_SURFACE_PBUFFER)
|
2017-04-20 17:20:21 -04:00
|
|
|
{
|
2017-09-17 15:51:07 -04:00
|
|
|
m_egl_surface = eglCreatePbufferSurface(m_egl_display, m_egl_config,
|
|
|
|
&attribs[0]);
|
2017-04-20 17:20:21 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_egl_surface != EGL_NO_SURFACE;
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-05 15:41:25 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool ContextManagerEGL::createContext()
|
|
|
|
{
|
|
|
|
m_is_legacy_device = false;
|
|
|
|
|
|
|
|
if (m_creation_params.opengl_api == CEGL_API_OPENGL_ES)
|
|
|
|
{
|
|
|
|
if (!m_creation_params.force_legacy_device)
|
|
|
|
{
|
|
|
|
if (m_egl_context == EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
std::vector<EGLint> context_attribs;
|
|
|
|
context_attribs.push_back(EGL_CONTEXT_CLIENT_VERSION);
|
|
|
|
context_attribs.push_back(3);
|
|
|
|
context_attribs.push_back(EGL_NONE);
|
|
|
|
context_attribs.push_back(0);
|
|
|
|
|
|
|
|
m_egl_context = eglCreateContext(m_egl_display,
|
|
|
|
m_egl_config,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
&context_attribs[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_context == EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
m_is_legacy_device = true;
|
|
|
|
|
|
|
|
std::vector<EGLint> context_attribs;
|
|
|
|
context_attribs.push_back(EGL_CONTEXT_CLIENT_VERSION);
|
|
|
|
context_attribs.push_back(2);
|
|
|
|
context_attribs.push_back(EGL_NONE);
|
|
|
|
context_attribs.push_back(0);
|
|
|
|
|
|
|
|
m_egl_context = eglCreateContext(m_egl_display,
|
|
|
|
m_egl_config,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
&context_attribs[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (m_creation_params.opengl_api == CEGL_API_OPENGL)
|
|
|
|
{
|
|
|
|
if (!m_creation_params.force_legacy_device)
|
|
|
|
{
|
|
|
|
if (m_egl_context == EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
std::vector<EGLint> context_attribs;
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MAJOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(4);
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MINOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(3);
|
|
|
|
context_attribs.push_back(EGL_NONE);
|
|
|
|
context_attribs.push_back(0);
|
|
|
|
|
|
|
|
m_egl_context = eglCreateContext(m_egl_display,
|
|
|
|
m_egl_config,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
&context_attribs[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_context == EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
std::vector<EGLint> context_attribs;
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MAJOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(3);
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MINOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(3);
|
|
|
|
context_attribs.push_back(EGL_NONE);
|
|
|
|
context_attribs.push_back(0);
|
|
|
|
|
|
|
|
m_egl_context = eglCreateContext(m_egl_display,
|
|
|
|
m_egl_config,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
&context_attribs[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_context == EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
std::vector<EGLint> context_attribs;
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MAJOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(3);
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MINOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(1);
|
|
|
|
context_attribs.push_back(EGL_NONE);
|
|
|
|
context_attribs.push_back(0);
|
|
|
|
|
|
|
|
m_egl_context = eglCreateContext(m_egl_display,
|
|
|
|
m_egl_config,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
&context_attribs[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_context == EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
m_is_legacy_device = true;
|
|
|
|
|
|
|
|
std::vector<EGLint> context_attribs;
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MAJOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(2);
|
2017-04-21 02:51:54 -04:00
|
|
|
context_attribs.push_back(EGL_CONTEXT_MINOR_VERSION);
|
2017-04-20 17:20:21 -04:00
|
|
|
context_attribs.push_back(1);
|
|
|
|
context_attribs.push_back(EGL_NONE);
|
|
|
|
context_attribs.push_back(0);
|
|
|
|
|
|
|
|
m_egl_context = eglCreateContext(m_egl_display,
|
|
|
|
m_egl_config,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
&context_attribs[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_egl_context != EGL_NO_CONTEXT;
|
2017-04-05 14:28:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
void ContextManagerEGL::close()
|
2017-04-05 14:28:38 -04:00
|
|
|
{
|
2017-04-20 17:20:21 -04:00
|
|
|
eglMakeCurrent(m_egl_display, EGL_NO_SURFACE, EGL_NO_SURFACE,
|
|
|
|
EGL_NO_CONTEXT);
|
|
|
|
|
|
|
|
if (m_egl_context != EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
eglDestroyContext(m_egl_display, m_egl_context);
|
|
|
|
m_egl_context = EGL_NO_CONTEXT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_surface != EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
eglDestroySurface(m_egl_display, m_egl_surface);
|
|
|
|
m_egl_surface = EGL_NO_SURFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_egl_display != EGL_NO_DISPLAY)
|
|
|
|
{
|
|
|
|
eglTerminate(m_egl_display);
|
|
|
|
m_egl_display = EGL_NO_DISPLAY;
|
|
|
|
}
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool ContextManagerEGL::swapBuffers()
|
|
|
|
{
|
|
|
|
bool success = eglSwapBuffers(m_egl_display, m_egl_surface);
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
#ifdef DEBUG
|
2017-04-21 02:51:54 -04:00
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
eglGetError();
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
#endif
|
2017-04-20 17:20:21 -04:00
|
|
|
|
2017-04-21 02:51:54 -04:00
|
|
|
return success;
|
2017-04-05 14:28:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-11 18:57:01 -04:00
|
|
|
bool ContextManagerEGL::makeCurrent()
|
|
|
|
{
|
|
|
|
bool success = eglMakeCurrent(m_egl_display, m_egl_surface, m_egl_surface,
|
|
|
|
m_egl_context);
|
|
|
|
|
2017-05-13 18:05:36 -04:00
|
|
|
return success;
|
2017-05-11 18:57:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
void ContextManagerEGL::reloadEGLSurface(void* window)
|
2017-04-05 14:28:38 -04:00
|
|
|
{
|
2017-06-24 15:39:58 -04:00
|
|
|
if (!m_initialized)
|
|
|
|
return;
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
#ifdef _IRR_COMPILE_WITH_ANDROID_DEVICE_
|
|
|
|
m_egl_window = (ANativeWindow*)window;
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
if (!m_egl_window)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: Invalid EGL window.\n");
|
|
|
|
}
|
|
|
|
#endif
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
eglMakeCurrent(m_egl_display, EGL_NO_SURFACE, EGL_NO_SURFACE,
|
2017-04-21 02:51:54 -04:00
|
|
|
EGL_NO_CONTEXT);
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
eglDestroySurface(m_egl_display, m_egl_surface);
|
2017-04-21 02:51:54 -04:00
|
|
|
m_egl_surface = EGL_NO_SURFACE;
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool success = createSurface();
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
os::Printer::log("Error: Could not create EGL surface.");
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
success = eglMakeCurrent(m_egl_display, m_egl_surface, m_egl_surface,
|
2017-04-21 02:51:54 -04:00
|
|
|
m_egl_context);
|
2017-04-20 17:20:21 -04:00
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
checkEGLError();
|
|
|
|
os::Printer::log("Error: Couldn't make context current for EGL display.\n");
|
|
|
|
}
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool ContextManagerEGL::getSurfaceDimensions(int* width, int* height)
|
|
|
|
{
|
|
|
|
if (!eglQuerySurface(m_egl_display, m_egl_surface, EGL_WIDTH, width))
|
|
|
|
return false;
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
if (!eglQuerySurface(m_egl_display, m_egl_surface, EGL_HEIGHT, height))
|
|
|
|
return false;
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
return true;
|
2017-04-05 14:28:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
bool ContextManagerEGL::hasEGLExtension(const char* extension)
|
2017-04-05 14:28:38 -04:00
|
|
|
{
|
2017-04-20 17:20:21 -04:00
|
|
|
const char* extensions = eglQueryString(m_egl_display, EGL_EXTENSIONS);
|
|
|
|
|
|
|
|
if (extensions && strstr(extensions, extension) != NULL)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2017-04-05 14:28:38 -04:00
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
return false;
|
2017-04-05 14:28:38 -04:00
|
|
|
}
|
|
|
|
|
2017-04-20 17:20:21 -04:00
|
|
|
|
|
|
|
bool ContextManagerEGL::checkEGLError()
|
|
|
|
{
|
|
|
|
bool result = true;
|
|
|
|
|
|
|
|
switch (eglGetError())
|
|
|
|
{
|
|
|
|
case EGL_SUCCESS:
|
|
|
|
result = false;
|
|
|
|
break;
|
|
|
|
case EGL_NOT_INITIALIZED:
|
|
|
|
os::Printer::log("Error: EGL_NOT_INITIALIZED\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_ACCESS:
|
|
|
|
os::Printer::log("Error: EGL_BAD_ACCESS\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_ALLOC:
|
|
|
|
os::Printer::log("Error: EGL_BAD_ALLOC\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_ATTRIBUTE:
|
|
|
|
os::Printer::log("Error: EGL_BAD_ATTRIBUTE\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_CONTEXT:
|
|
|
|
os::Printer::log("Error: EGL_BAD_CONTEXT\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_CONFIG:
|
|
|
|
os::Printer::log("Error: EGL_BAD_CONFIG\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_CURRENT_SURFACE:
|
|
|
|
os::Printer::log("Error: EGL_BAD_CURRENT_SURFACE\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_DISPLAY:
|
|
|
|
os::Printer::log("Error: EGL_BAD_DISPLAY\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_SURFACE:
|
|
|
|
os::Printer::log("Error: EGL_BAD_SURFACE\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_MATCH:
|
|
|
|
os::Printer::log("Error: EGL_BAD_MATCH\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_PARAMETER:
|
|
|
|
os::Printer::log("Error: EGL_BAD_PARAMETER\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_NATIVE_PIXMAP:
|
|
|
|
os::Printer::log("Error: EGL_BAD_NATIVE_PIXMAP\n");
|
|
|
|
break;
|
|
|
|
case EGL_BAD_NATIVE_WINDOW:
|
|
|
|
os::Printer::log("Error: EGL_BAD_NATIVE_WINDOW\n");
|
|
|
|
break;
|
|
|
|
case EGL_CONTEXT_LOST:
|
|
|
|
os::Printer::log("Error: EGL_CONTEXT_LOST\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
os::Printer::log("Error: Unknown EGL error.\n");
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
|
|
|
|
return result;
|
2017-04-05 14:28:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|