GLInterface: Support shared contexts in GLX

This commit is contained in:
Ryan Houdek 2016-01-23 04:21:53 -06:00 committed by Stenzek
parent 621287e7eb
commit 01ae02482c
2 changed files with 149 additions and 44 deletions

View File

@ -2,7 +2,8 @@
// Licensed under GPLv2+ // Licensed under GPLv2+
// Refer to the license.txt file included. // Refer to the license.txt file included.
#include <string> #include <array>
#include <sstream>
#include "Common/GL/GLInterface/GLX.h" #include "Common/GL/GLInterface/GLX.h"
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
@ -17,6 +18,9 @@ typedef int (*PFNGLXSWAPINTERVALSGIPROC)(int interval);
static PFNGLXCREATECONTEXTATTRIBSPROC glXCreateContextAttribs = nullptr; static PFNGLXCREATECONTEXTATTRIBSPROC glXCreateContextAttribs = nullptr;
static PFNGLXSWAPINTERVALSGIPROC glXSwapIntervalSGI = nullptr; static PFNGLXSWAPINTERVALSGIPROC glXSwapIntervalSGI = nullptr;
static PFNGLXCREATEGLXPBUFFERSGIXPROC glXCreateGLXPbufferSGIX = nullptr;
static PFNGLXDESTROYGLXPBUFFERSGIXPROC glXDestroyGLXPbufferSGIX = nullptr;
static bool s_glxError; static bool s_glxError;
static int ctxErrorHandler(Display* dpy, XErrorEvent* ev) static int ctxErrorHandler(Display* dpy, XErrorEvent* ev)
{ {
@ -26,7 +30,7 @@ static int ctxErrorHandler(Display* dpy, XErrorEvent* ev)
void cInterfaceGLX::SwapInterval(int Interval) void cInterfaceGLX::SwapInterval(int Interval)
{ {
if (glXSwapIntervalSGI) if (glXSwapIntervalSGI && m_has_handle)
glXSwapIntervalSGI(Interval); glXSwapIntervalSGI(Interval);
else else
ERROR_LOG(VIDEO, "No support for SwapInterval (framerate clamped to monitor refresh rate)."); ERROR_LOG(VIDEO, "No support for SwapInterval (framerate clamped to monitor refresh rate).");
@ -45,6 +49,9 @@ void cInterfaceGLX::Swap()
// Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize() // Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize()
bool cInterfaceGLX::Create(void* window_handle, bool stereo, bool core) bool cInterfaceGLX::Create(void* window_handle, bool stereo, bool core)
{ {
m_has_handle = !!window_handle;
m_host_window = (Window)window_handle;
dpy = XOpenDisplay(nullptr); dpy = XOpenDisplay(nullptr);
int screen = DefaultScreen(dpy); int screen = DefaultScreen(dpy);
@ -100,51 +107,43 @@ bool cInterfaceGLX::Create(void* window_handle, bool stereo, bool core)
fbconfig = *fbc; fbconfig = *fbc;
XFree(fbc); XFree(fbc);
// Get an appropriate visual
XVisualInfo* vi = glXGetVisualFromFBConfig(dpy, fbconfig);
s_glxError = false; s_glxError = false;
XErrorHandler oldHandler = XSetErrorHandler(&ctxErrorHandler); XErrorHandler oldHandler = XSetErrorHandler(&ctxErrorHandler);
// Create a GLX context. // Create a GLX context.
// We try to get a 4.0 core profile, else we try 3.3, else try it with anything we get. // We try to get a 4.0 core profile, else we try 3.3, else try it with anything we get.
int context_attribs[] = {GLX_CONTEXT_MAJOR_VERSION_ARB, std::array<int, 9> context_attribs = {
4, {GLX_CONTEXT_MAJOR_VERSION_ARB, 4, GLX_CONTEXT_MINOR_VERSION_ARB, 0,
GLX_CONTEXT_MINOR_VERSION_ARB, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, GLX_CONTEXT_FLAGS_ARB,
0, GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB, None}};
GLX_CONTEXT_PROFILE_MASK_ARB,
GLX_CONTEXT_CORE_PROFILE_BIT_ARB,
GLX_CONTEXT_FLAGS_ARB,
GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB,
None};
ctx = nullptr; ctx = nullptr;
if (core) if (core)
{ {
ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, context_attribs); ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, &context_attribs[0]);
XSync(dpy, False); XSync(dpy, False);
m_attribs.insert(m_attribs.end(), context_attribs.begin(), context_attribs.end());
} }
if (core && (!ctx || s_glxError)) if (core && (!ctx || s_glxError))
{ {
int context_attribs_33[] = {GLX_CONTEXT_MAJOR_VERSION_ARB, std::array<int, 9> context_attribs_33 = {
3, {GLX_CONTEXT_MAJOR_VERSION_ARB, 3, GLX_CONTEXT_MINOR_VERSION_ARB, 3,
GLX_CONTEXT_MINOR_VERSION_ARB, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, GLX_CONTEXT_FLAGS_ARB,
3, GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB, None}};
GLX_CONTEXT_PROFILE_MASK_ARB,
GLX_CONTEXT_CORE_PROFILE_BIT_ARB,
GLX_CONTEXT_FLAGS_ARB,
GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB,
None};
s_glxError = false; s_glxError = false;
ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, context_attribs_33); ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, &context_attribs_33[0]);
XSync(dpy, False); XSync(dpy, False);
m_attribs.clear();
m_attribs.insert(m_attribs.end(), context_attribs_33.begin(), context_attribs_33.end());
} }
if (!ctx || s_glxError) if (!ctx || s_glxError)
{ {
int context_attribs_legacy[] = {GLX_CONTEXT_MAJOR_VERSION_ARB, 1, GLX_CONTEXT_MINOR_VERSION_ARB, std::array<int, 5> context_attribs_legacy = {
0, None}; {GLX_CONTEXT_MAJOR_VERSION_ARB, 1, GLX_CONTEXT_MINOR_VERSION_ARB, 0, None}};
s_glxError = false; s_glxError = false;
ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, context_attribs_legacy); ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, &context_attribs_legacy[0]);
XSync(dpy, False); XSync(dpy, False);
m_attribs.clear();
m_attribs.insert(m_attribs.end(), context_attribs_legacy.begin(), context_attribs_legacy.end());
} }
if (!ctx || s_glxError) if (!ctx || s_glxError)
{ {
@ -153,30 +152,116 @@ bool cInterfaceGLX::Create(void* window_handle, bool stereo, bool core)
} }
XSetErrorHandler(oldHandler); XSetErrorHandler(oldHandler);
XWindow.Initialize(dpy); std::string tmp;
std::istringstream buffer(glXQueryExtensionsString(dpy, screen));
Window parent = (Window)window_handle; while (buffer >> tmp)
XWindowAttributes attribs;
if (!XGetWindowAttributes(dpy, parent, &attribs))
{ {
ERROR_LOG(VIDEO, "Window attribute retrieval failed"); if (tmp == "GLX_SGIX_pbuffer")
return false; m_supports_pbuffer = true;
} }
s_backbuffer_width = attribs.width; if (m_supports_pbuffer)
s_backbuffer_height = attribs.height; {
// Get the function pointers we require
glXCreateGLXPbufferSGIX =
(PFNGLXCREATEGLXPBUFFERSGIXPROC)GetFuncAddress("glXCreateGLXPbufferSGIX");
glXDestroyGLXPbufferSGIX =
(PFNGLXDESTROYGLXPBUFFERSGIXPROC)GetFuncAddress("glXDestroyGLXPbufferSGIX");
}
win = XWindow.CreateXWindow(parent, vi); if (!CreateWindowSurface())
XFree(vi); {
ERROR_LOG(VIDEO, "Error: CreateWindowSurface failed\n");
return false;
}
return true;
}
bool cInterfaceGLX::Create(cInterfaceBase* main_context)
{
cInterfaceGLX* glx_context = static_cast<cInterfaceGLX*>(main_context);
m_has_handle = false;
dpy = glx_context->dpy;
fbconfig = glx_context->fbconfig;
s_glxError = false;
XErrorHandler oldHandler = XSetErrorHandler(&ctxErrorHandler);
ctx = glXCreateContextAttribs(dpy, fbconfig, glx_context->ctx, True, &glx_context->m_attribs[0]);
XSync(dpy, False);
if (!ctx || s_glxError)
{
ERROR_LOG(VIDEO, "Unable to create GL context.");
return false;
}
XSetErrorHandler(oldHandler);
if (!CreateWindowSurface())
{
ERROR_LOG(VIDEO, "Error: CreateWindowSurface failed\n");
return false;
}
return true;
}
std::unique_ptr<cInterfaceBase> cInterfaceGLX::CreateSharedContext()
{
std::unique_ptr<cInterfaceBase> context = std::make_unique<cInterfaceGLX>();
if (!context->Create(this))
return nullptr;
return context;
}
bool cInterfaceGLX::CreateWindowSurface()
{
if (m_has_handle)
{
// Get an appropriate visual
XVisualInfo* vi = glXGetVisualFromFBConfig(dpy, fbconfig);
XWindow.Initialize(dpy);
XWindowAttributes attribs;
if (!XGetWindowAttributes(dpy, m_host_window, &attribs))
{
ERROR_LOG(VIDEO, "Window attribute retrieval failed");
return false;
}
s_backbuffer_width = attribs.width;
s_backbuffer_height = attribs.height;
win = XWindow.CreateXWindow(m_host_window, vi);
XFree(vi);
}
else
{
win = m_pbuffer = glXCreateGLXPbufferSGIX(dpy, fbconfig, 1, 1, nullptr);
if (!m_pbuffer)
return false;
}
return true; return true;
} }
void cInterfaceGLX::DestroyWindowSurface()
{
if (!m_pbuffer)
{
XWindow.DestroyXWindow();
}
else
{
glXDestroyGLXPbufferSGIX(dpy, m_pbuffer);
m_pbuffer = 0;
}
}
bool cInterfaceGLX::MakeCurrent() bool cInterfaceGLX::MakeCurrent()
{ {
bool success = glXMakeCurrent(dpy, win, ctx); bool success = glXMakeCurrent(dpy, win, ctx);
if (success) if (success && !glXSwapIntervalSGI)
{ {
// load this function based on the current bound context // load this function based on the current bound context
glXSwapIntervalSGI = glXSwapIntervalSGI =
@ -193,11 +278,18 @@ bool cInterfaceGLX::ClearCurrent()
// Close backend // Close backend
void cInterfaceGLX::Shutdown() void cInterfaceGLX::Shutdown()
{ {
XWindow.DestroyXWindow(); DestroyWindowSurface();
if (ctx) if (ctx)
{ {
glXDestroyContext(dpy, ctx); glXDestroyContext(dpy, ctx);
XCloseDisplay(dpy);
ctx = nullptr; // Don't close the display connection if we are a shared context.
// Saves doing reference counting on this object, and the main context will always
// be shut down last anyway.
if (m_has_handle)
{
XCloseDisplay(dpy);
ctx = nullptr;
}
} }
} }

View File

@ -5,7 +5,10 @@
#pragma once #pragma once
#include <GL/glx.h> #include <GL/glx.h>
#include <GL/glxext.h>
#include <memory>
#include <string> #include <string>
#include <vector>
#include "Common/GL/GLInterface/X11_Util.h" #include "Common/GL/GLInterface/X11_Util.h"
#include "Common/GL/GLInterfaceBase.h" #include "Common/GL/GLInterfaceBase.h"
@ -13,11 +16,19 @@
class cInterfaceGLX : public cInterfaceBase class cInterfaceGLX : public cInterfaceBase
{ {
private: private:
Window m_host_window;
cX11Window XWindow; cX11Window XWindow;
Display* dpy; Display* dpy;
Window win; Window win;
GLXContext ctx; GLXContext ctx;
GLXFBConfig fbconfig; GLXFBConfig fbconfig;
bool m_has_handle;
bool m_supports_pbuffer = false;
GLXPbufferSGIX m_pbuffer = 0;
std::vector<int> m_attribs;
bool CreateWindowSurface();
void DestroyWindowSurface();
public: public:
friend class cX11Window; friend class cX11Window;
@ -25,7 +36,9 @@ public:
void Swap() override; void Swap() override;
void* GetFuncAddress(const std::string& name) override; void* GetFuncAddress(const std::string& name) override;
bool Create(void* window_handle, bool stereo, bool core) override; bool Create(void* window_handle, bool stereo, bool core) override;
bool Create(cInterfaceBase* main_context) override;
bool MakeCurrent() override; bool MakeCurrent() override;
bool ClearCurrent() override; bool ClearCurrent() override;
void Shutdown() override; void Shutdown() override;
std::unique_ptr<cInterfaceBase> CreateSharedContext() override;
}; };