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>
|
|
|
|
|
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"
|
2012-12-17 15:01:52 -06:00
|
|
|
|
|
|
|
// Show the current FPS
|
2012-12-26 00:34:09 -06:00
|
|
|
void cInterfaceEGL::Swap()
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2016-01-10 12:28:05 -06:00
|
|
|
if (egl_surf != EGL_NO_SURFACE)
|
|
|
|
eglSwapBuffers(egl_dpy, egl_surf);
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
2013-03-24 21:06:34 -05:00
|
|
|
void cInterfaceEGL::SwapInterval(int Interval)
|
|
|
|
{
|
2015-09-19 06:12:20 +12:00
|
|
|
eglSwapInterval(egl_dpy, Interval);
|
2013-03-24 21:06:34 -05:00
|
|
|
}
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-03-12 15:33:41 -04:00
|
|
|
void* cInterfaceEGL::GetFuncAddress(const std::string& name)
|
2013-12-30 07:22:50 -06:00
|
|
|
{
|
|
|
|
return (void*)eglGetProcAddress(name.c_str());
|
|
|
|
}
|
|
|
|
|
2014-01-18 04:11:59 +00:00
|
|
|
void cInterfaceEGL::DetectMode()
|
|
|
|
{
|
2016-01-02 14:19:28 -05:00
|
|
|
if (s_opengl_mode != GLInterfaceMode::MODE_DETECT)
|
2014-01-18 04:11:59 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
EGLint num_configs;
|
|
|
|
bool supportsGL = false, supportsGLES2 = false, supportsGLES3 = false;
|
2015-07-21 11:29:15 -05:00
|
|
|
std::array<int, 3> renderable_types = {
|
|
|
|
EGL_OPENGL_BIT,
|
|
|
|
(1 << 6), /* EGL_OPENGL_ES3_BIT_KHR */
|
|
|
|
EGL_OPENGL_ES2_BIT,
|
|
|
|
};
|
2014-01-18 04:11:59 +00:00
|
|
|
|
2015-07-21 11:29:15 -05:00
|
|
|
for (auto renderable_type : renderable_types)
|
2014-02-16 23:51:41 -05:00
|
|
|
{
|
2015-07-21 11:29:15 -05:00
|
|
|
// attributes for a visual in RGBA format with at least
|
|
|
|
// 8 bits per color
|
|
|
|
int attribs[] = {
|
|
|
|
EGL_RED_SIZE, 8,
|
|
|
|
EGL_GREEN_SIZE, 8,
|
|
|
|
EGL_BLUE_SIZE, 8,
|
|
|
|
EGL_RENDERABLE_TYPE, renderable_type,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
2014-01-18 04:11:59 +00:00
|
|
|
|
2015-07-21 11:29:15 -05:00
|
|
|
// Get how many configs there are
|
|
|
|
if (!eglChooseConfig( egl_dpy, attribs, nullptr, 0, &num_configs))
|
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Error: couldn't get an EGL visual config\n");
|
2015-08-17 18:52:17 -05:00
|
|
|
continue;
|
2015-07-21 11:29:15 -05:00
|
|
|
}
|
2014-01-18 04:11:59 +00:00
|
|
|
|
2015-08-17 18:52:17 -05:00
|
|
|
EGLConfig* config = new EGLConfig[num_configs];
|
2014-01-18 04:11:59 +00:00
|
|
|
|
2015-07-21 11:29:15 -05:00
|
|
|
// Get all the configurations
|
|
|
|
if (!eglChooseConfig(egl_dpy, attribs, config, num_configs, &num_configs))
|
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Error: couldn't get an EGL visual config\n");
|
2015-08-17 18:52:17 -05:00
|
|
|
delete[] config;
|
|
|
|
continue;
|
2015-07-21 11:29:15 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < num_configs; ++i)
|
2014-01-18 04:11:59 +00:00
|
|
|
{
|
2015-07-21 11:29:15 -05:00
|
|
|
EGLint attribVal;
|
|
|
|
bool ret;
|
|
|
|
ret = eglGetConfigAttrib(egl_dpy, config[i], EGL_RENDERABLE_TYPE, &attribVal);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
if (attribVal & EGL_OPENGL_BIT)
|
|
|
|
supportsGL = true;
|
|
|
|
if (attribVal & (1 << 6)) /* EGL_OPENGL_ES3_BIT_KHR */
|
|
|
|
supportsGLES3 = true;
|
|
|
|
if (attribVal & EGL_OPENGL_ES2_BIT)
|
|
|
|
supportsGLES2 = true;
|
|
|
|
}
|
2014-01-18 04:11:59 +00:00
|
|
|
}
|
2015-08-17 18:52:17 -05:00
|
|
|
delete[] config;
|
2014-01-18 04:11:59 +00:00
|
|
|
}
|
2015-07-21 11:29:15 -05:00
|
|
|
|
2014-01-18 04:11:59 +00:00
|
|
|
if (supportsGL)
|
|
|
|
s_opengl_mode = GLInterfaceMode::MODE_OPENGL;
|
|
|
|
else if (supportsGLES3)
|
|
|
|
s_opengl_mode = GLInterfaceMode::MODE_OPENGLES3;
|
|
|
|
else if (supportsGLES2)
|
|
|
|
s_opengl_mode = GLInterfaceMode::MODE_OPENGLES2;
|
2015-07-21 11:29:15 -05:00
|
|
|
|
2014-01-18 04:11:59 +00:00
|
|
|
if (s_opengl_mode == GLInterfaceMode::MODE_DETECT) // Errored before we found a mode
|
|
|
|
s_opengl_mode = GLInterfaceMode::MODE_OPENGL; // Fall back to OpenGL
|
|
|
|
}
|
|
|
|
|
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()
|
2015-09-17 17:48:25 +02:00
|
|
|
bool cInterfaceEGL::Create(void *window_handle, bool core)
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2013-01-24 10:39:38 -06:00
|
|
|
EGLint egl_major, egl_minor;
|
2014-01-18 04:11:59 +00:00
|
|
|
|
2014-08-09 10:31:27 -04:00
|
|
|
egl_dpy = OpenDisplay();
|
2016-01-10 12:28:05 -06:00
|
|
|
m_host_window = (EGLNativeWindowType) window_handle;
|
|
|
|
m_has_handle = !!window_handle;
|
2016-01-21 19:03:58 -06:00
|
|
|
m_core = false;
|
2014-01-18 04:11:59 +00:00
|
|
|
|
2014-08-09 09:49:45 -04:00
|
|
|
if (!egl_dpy)
|
2014-02-16 23:51:41 -05:00
|
|
|
{
|
2014-01-18 04:11:59 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: eglGetDisplay() failed\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-08-09 09:49:45 -04:00
|
|
|
if (!eglInitialize(egl_dpy, &egl_major, &egl_minor))
|
2014-02-16 23:51:41 -05:00
|
|
|
{
|
2014-01-18 04:11:59 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: eglInitialize() failed\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Detection code */
|
2013-03-24 21:06:34 -05:00
|
|
|
EGLint num_configs;
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-01-18 04:11:59 +00:00
|
|
|
DetectMode();
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
// attributes for a visual in RGBA format with at least
|
2013-10-29 19:19:56 +01:00
|
|
|
// 8 bits per color
|
2012-12-17 15:01:52 -06:00
|
|
|
int attribs[] = {
|
2013-12-12 12:43:49 -06:00
|
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
2012-12-17 15:01:52 -06:00
|
|
|
EGL_RED_SIZE, 8,
|
|
|
|
EGL_GREEN_SIZE, 8,
|
|
|
|
EGL_BLUE_SIZE, 8,
|
|
|
|
EGL_NONE };
|
|
|
|
|
2013-12-12 12:43:49 -06:00
|
|
|
EGLint ctx_attribs[] = {
|
2012-12-17 15:01:52 -06:00
|
|
|
EGL_CONTEXT_CLIENT_VERSION, 2,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
2014-03-11 00:30:55 +13:00
|
|
|
switch (s_opengl_mode)
|
2013-12-12 12:43:49 -06:00
|
|
|
{
|
2016-01-21 19:03:58 -06:00
|
|
|
case GLInterfaceMode::MODE_OPENGL:
|
|
|
|
attribs[1] = EGL_OPENGL_BIT;
|
|
|
|
ctx_attribs[0] = EGL_NONE;
|
|
|
|
break;
|
|
|
|
case GLInterfaceMode::MODE_OPENGLES2:
|
|
|
|
attribs[1] = EGL_OPENGL_ES2_BIT;
|
|
|
|
ctx_attribs[1] = 2;
|
|
|
|
break;
|
|
|
|
case GLInterfaceMode::MODE_OPENGLES3:
|
|
|
|
attribs[1] = (1 << 6); /* EGL_OPENGL_ES3_BIT_KHR */
|
|
|
|
ctx_attribs[1] = 3;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ERROR_LOG(VIDEO, "Unknown opengl mode set\n");
|
|
|
|
return false;
|
|
|
|
break;
|
2013-12-12 12:43:49 -06:00
|
|
|
}
|
2013-01-24 10:39:38 -06:00
|
|
|
|
2016-01-10 12:28:05 -06:00
|
|
|
if (!eglChooseConfig( egl_dpy, attribs, &m_config, 1, &num_configs))
|
2014-02-16 23:51:41 -05:00
|
|
|
{
|
2014-01-18 04:11:59 +00:00
|
|
|
INFO_LOG(VIDEO, "Error: couldn't get an EGL visual config\n");
|
2016-01-21 19:03:58 -06:00
|
|
|
return false;
|
2013-01-24 10:39:38 -06:00
|
|
|
}
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2016-01-02 14:19:28 -05:00
|
|
|
if (s_opengl_mode == GLInterfaceMode::MODE_OPENGL)
|
2013-12-12 22:08:54 +00:00
|
|
|
eglBindAPI(EGL_OPENGL_API);
|
|
|
|
else
|
|
|
|
eglBindAPI(EGL_OPENGL_ES_API);
|
2013-03-24 21:06:34 -05:00
|
|
|
|
2016-01-10 12:28:05 -06:00
|
|
|
egl_ctx = eglCreateContext(egl_dpy, m_config, EGL_NO_CONTEXT, ctx_attribs );
|
2014-08-09 09:49:45 -04:00
|
|
|
if (!egl_ctx)
|
2014-02-16 23:51:41 -05:00
|
|
|
{
|
2013-04-23 14:17:16 -05:00
|
|
|
INFO_LOG(VIDEO, "Error: eglCreateContext failed\n");
|
2016-01-21 19:03:58 -06:00
|
|
|
return false;
|
2013-01-24 10:39:38 -06:00
|
|
|
}
|
|
|
|
|
2016-01-10 12:28:05 -06:00
|
|
|
std::string tmp;
|
|
|
|
std::istringstream buffer(eglQueryString(egl_dpy, EGL_EXTENSIONS));
|
|
|
|
while (buffer >> tmp)
|
2014-02-16 23:51:41 -05:00
|
|
|
{
|
2016-01-10 12:28:05 -06:00
|
|
|
if (tmp == "EGL_KHR_surfaceless_context")
|
|
|
|
{
|
|
|
|
m_supports_surfaceless = true;
|
|
|
|
break;
|
|
|
|
}
|
2013-01-24 10:39:38 -06:00
|
|
|
}
|
|
|
|
|
2016-01-21 19:03:58 -06:00
|
|
|
if (!CreateWindowSurface())
|
|
|
|
{
|
|
|
|
ERROR_LOG(VIDEO, "Error: CreateWindowSurface failed 0x%04x\n", eglGetError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2016-01-10 12:28:05 -06:00
|
|
|
|
2016-01-21 19:03:58 -06:00
|
|
|
std::unique_ptr<cInterfaceBase> cInterfaceEGL::CreateSharedContext()
|
|
|
|
{
|
|
|
|
std::unique_ptr<cInterfaceBase> context = std::make_unique<cInterfaceEGL>();
|
|
|
|
if (!context->Create(this))
|
|
|
|
return nullptr;
|
|
|
|
return context;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cInterfaceEGL::Create(cInterfaceBase* main_context)
|
|
|
|
{
|
|
|
|
cInterfaceEGL* egl_context = static_cast<cInterfaceEGL*>(main_context);
|
|
|
|
|
|
|
|
egl_dpy = egl_context->egl_dpy;
|
|
|
|
m_core = egl_context->m_core;
|
|
|
|
m_config = egl_context->m_config;
|
|
|
|
m_supports_surfaceless = egl_context->m_supports_surfaceless;
|
|
|
|
m_is_shared = true;
|
|
|
|
m_has_handle = false;
|
|
|
|
|
|
|
|
EGLint ctx_attribs[] = {
|
|
|
|
EGL_CONTEXT_CLIENT_VERSION, 2,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (egl_context->GetMode())
|
|
|
|
{
|
|
|
|
case GLInterfaceMode::MODE_OPENGL:
|
|
|
|
ctx_attribs[0] = EGL_NONE;
|
|
|
|
break;
|
|
|
|
case GLInterfaceMode::MODE_OPENGLES2:
|
|
|
|
ctx_attribs[1] = 2;
|
|
|
|
break;
|
|
|
|
case GLInterfaceMode::MODE_OPENGLES3:
|
|
|
|
ctx_attribs[1] = 3;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
INFO_LOG(VIDEO, "Unknown opengl mode set\n");
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (egl_context->GetMode() == GLInterfaceMode::MODE_OPENGL)
|
|
|
|
eglBindAPI(EGL_OPENGL_API);
|
|
|
|
else
|
|
|
|
eglBindAPI(EGL_OPENGL_ES_API);
|
|
|
|
|
|
|
|
egl_ctx = eglCreateContext(egl_dpy, m_config, egl_context->egl_ctx, ctx_attribs );
|
|
|
|
if (!egl_ctx)
|
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Error: eglCreateContext failed 0x%04x\n", eglGetError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CreateWindowSurface())
|
|
|
|
{
|
|
|
|
ERROR_LOG(VIDEO, "Error: CreateWindowSurface failed 0x%04x\n", eglGetError());
|
|
|
|
return false;
|
|
|
|
}
|
2012-12-17 15:01:52 -06:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-01-21 19:03:58 -06:00
|
|
|
bool cInterfaceEGL::CreateWindowSurface()
|
2016-01-10 12:28:05 -06:00
|
|
|
{
|
|
|
|
if (m_has_handle)
|
|
|
|
{
|
|
|
|
EGLNativeWindowType native_window = InitializePlatform(m_host_window, m_config);
|
|
|
|
egl_surf = eglCreateWindowSurface(egl_dpy, m_config, native_window, nullptr);
|
|
|
|
if (!egl_surf)
|
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Error: eglCreateWindowSurface failed\n");
|
2016-01-21 19:03:58 -06:00
|
|
|
return false;
|
2016-01-10 12:28:05 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!m_supports_surfaceless)
|
|
|
|
{
|
|
|
|
EGLint attrib_list[] =
|
|
|
|
{
|
|
|
|
EGL_NONE,
|
|
|
|
};
|
|
|
|
egl_surf = eglCreatePbufferSurface(egl_dpy, m_config, attrib_list);
|
|
|
|
if (!egl_surf)
|
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Error: eglCreatePbufferSurface failed");
|
2016-01-21 19:03:58 -06:00
|
|
|
return false;
|
2016-01-10 12:28:05 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
egl_surf = EGL_NO_SURFACE;
|
|
|
|
}
|
2016-01-21 19:03:58 -06:00
|
|
|
return true;
|
2016-01-10 12:28:05 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void cInterfaceEGL::DestroyWindowSurface()
|
|
|
|
{
|
|
|
|
if (egl_surf != EGL_NO_SURFACE && !eglDestroySurface(egl_dpy, egl_surf))
|
|
|
|
NOTICE_LOG(VIDEO, "Could not destroy window surface.");
|
|
|
|
egl_surf = EGL_NO_SURFACE;
|
|
|
|
}
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
bool cInterfaceEGL::MakeCurrent()
|
|
|
|
{
|
2014-08-09 09:49:45 -04:00
|
|
|
return eglMakeCurrent(egl_dpy, egl_surf, egl_surf, egl_ctx);
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
2015-09-04 19:58:58 -05:00
|
|
|
|
2016-01-10 12:28:05 -06:00
|
|
|
void cInterfaceEGL::UpdateHandle(void* window_handle)
|
|
|
|
{
|
|
|
|
m_host_window = (EGLNativeWindowType)window_handle;
|
|
|
|
m_has_handle = !!window_handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cInterfaceEGL::UpdateSurface()
|
|
|
|
{
|
|
|
|
ClearCurrent();
|
|
|
|
DestroyWindowSurface();
|
|
|
|
CreateWindowSurface();
|
|
|
|
MakeCurrent();
|
|
|
|
}
|
|
|
|
|
2015-09-04 19:58:58 -05:00
|
|
|
bool cInterfaceEGL::ClearCurrent()
|
|
|
|
{
|
|
|
|
return eglMakeCurrent(egl_dpy, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
|
|
|
}
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
// Close backend
|
|
|
|
void cInterfaceEGL::Shutdown()
|
|
|
|
{
|
2014-08-09 10:31:27 -04:00
|
|
|
ShutdownPlatform();
|
2014-08-09 09:49:45 -04:00
|
|
|
if (egl_ctx)
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2016-01-21 19:03:58 -06:00
|
|
|
if (!eglMakeCurrent(egl_dpy, egl_surf, egl_surf, egl_ctx))
|
|
|
|
NOTICE_LOG(VIDEO, "Could not release drawing context.");
|
2014-08-09 09:49:45 -04:00
|
|
|
eglMakeCurrent(egl_dpy, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
|
|
|
if (!eglDestroyContext(egl_dpy, egl_ctx))
|
2013-04-23 14:17:16 -05:00
|
|
|
NOTICE_LOG(VIDEO, "Could not destroy drawing context.");
|
2016-01-10 12:28:05 -06:00
|
|
|
DestroyWindowSurface();
|
2016-01-21 19:03:58 -06:00
|
|
|
if (!m_is_shared && !eglTerminate(egl_dpy))
|
2013-04-23 14:17:16 -05:00
|
|
|
NOTICE_LOG(VIDEO, "Could not destroy display connection.");
|
2014-08-09 09:49:45 -04:00
|
|
|
egl_ctx = nullptr;
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
|
|
|
}
|