2015-05-23 22:55:12 -06:00
|
|
|
// Copyright 2012 Dolphin Emulator Project
|
2015-05-17 17:08:10 -06:00
|
|
|
// Licensed under GPLv2+
|
2014-02-10 11:54:46 -07:00
|
|
|
// Refer to the license.txt file included.
|
2012-12-17 14:01:52 -07:00
|
|
|
|
2014-03-12 13:33:41 -06:00
|
|
|
#include <string>
|
|
|
|
|
2014-11-01 08:33:02 -06:00
|
|
|
#include "VideoBackends/OGL/GLInterface/GLX.h"
|
2014-02-17 03:18:15 -07:00
|
|
|
|
|
|
|
#include "VideoCommon/RenderBase.h"
|
|
|
|
#include "VideoCommon/VideoConfig.h"
|
2012-12-17 14:01:52 -07:00
|
|
|
|
2014-12-20 02:57:34 -07:00
|
|
|
#define GLX_CONTEXT_MAJOR_VERSION_ARB 0x2091
|
|
|
|
#define GLX_CONTEXT_MINOR_VERSION_ARB 0x2092
|
|
|
|
|
|
|
|
typedef GLXContext (*PFNGLXCREATECONTEXTATTRIBSPROC)(Display*, GLXFBConfig, GLXContext, Bool, const int*);
|
2013-12-30 06:22:50 -07:00
|
|
|
typedef int ( * PFNGLXSWAPINTERVALSGIPROC) (int interval);
|
2014-12-20 02:57:34 -07:00
|
|
|
|
|
|
|
static PFNGLXCREATECONTEXTATTRIBSPROC glXCreateContextAttribs = nullptr;
|
2014-07-08 07:58:25 -06:00
|
|
|
static PFNGLXSWAPINTERVALSGIPROC glXSwapIntervalSGI = nullptr;
|
2013-12-30 06:22:50 -07:00
|
|
|
|
2014-12-20 02:57:34 -07:00
|
|
|
static bool s_glxError;
|
|
|
|
static int ctxErrorHandler(Display *dpy, XErrorEvent *ev)
|
|
|
|
{
|
2015-01-07 13:48:59 -07:00
|
|
|
s_glxError = true;
|
|
|
|
return 0;
|
2014-12-20 02:57:34 -07:00
|
|
|
}
|
|
|
|
|
2013-01-24 09:31:08 -07:00
|
|
|
void cInterfaceGLX::SwapInterval(int Interval)
|
|
|
|
{
|
|
|
|
if (glXSwapIntervalSGI)
|
|
|
|
glXSwapIntervalSGI(Interval);
|
|
|
|
else
|
|
|
|
ERROR_LOG(VIDEO, "No support for SwapInterval (framerate clamped to monitor refresh rate).");
|
|
|
|
}
|
2014-03-17 17:17:12 -06:00
|
|
|
void* cInterfaceGLX::GetFuncAddress(const std::string& name)
|
2013-12-30 06:22:50 -07:00
|
|
|
{
|
|
|
|
return (void*)glXGetProcAddress((const GLubyte*)name.c_str());
|
|
|
|
}
|
2013-01-24 09:31:08 -07:00
|
|
|
|
2012-12-25 23:34:09 -07:00
|
|
|
void cInterfaceGLX::Swap()
|
2012-12-17 14:01:52 -07:00
|
|
|
{
|
2014-08-09 07:49:45 -06:00
|
|
|
glXSwapBuffers(dpy, win);
|
2012-12-17 14:01:52 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create rendering window.
|
2014-02-16 21:51:41 -07:00
|
|
|
// Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize()
|
2014-08-05 22:44:21 -06:00
|
|
|
bool cInterfaceGLX::Create(void *window_handle)
|
2012-12-17 14:01:52 -07:00
|
|
|
{
|
2014-08-09 07:49:45 -06:00
|
|
|
dpy = XOpenDisplay(nullptr);
|
|
|
|
int screen = DefaultScreen(dpy);
|
2012-12-17 14:01:52 -07:00
|
|
|
|
2014-12-20 02:57:34 -07:00
|
|
|
// checking glx version
|
|
|
|
int glxMajorVersion, glxMinorVersion;
|
2014-08-09 07:49:45 -06:00
|
|
|
glXQueryVersion(dpy, &glxMajorVersion, &glxMinorVersion);
|
2014-12-20 02:57:34 -07:00
|
|
|
if (glxMajorVersion < 1 || (glxMajorVersion == 1 && glxMinorVersion < 4))
|
|
|
|
{
|
|
|
|
ERROR_LOG(VIDEO, "glX-Version %d.%d detected, but need at least 1.4",
|
|
|
|
glxMajorVersion, glxMinorVersion);
|
|
|
|
return false;
|
|
|
|
}
|
2012-12-17 14:01:52 -07:00
|
|
|
|
2014-12-20 02:57:34 -07:00
|
|
|
// loading core context creation function
|
|
|
|
glXCreateContextAttribs = (PFNGLXCREATECONTEXTATTRIBSPROC)GetFuncAddress("glXCreateContextAttribsARB");
|
|
|
|
if (!glXCreateContextAttribs)
|
2012-12-17 14:01:52 -07:00
|
|
|
{
|
2014-12-20 02:57:34 -07:00
|
|
|
ERROR_LOG(VIDEO, "glXCreateContextAttribsARB not found, do you support GLX_ARB_create_context?");
|
|
|
|
return false;
|
2012-12-17 14:01:52 -07:00
|
|
|
}
|
2014-12-20 02:57:34 -07:00
|
|
|
|
|
|
|
// choosing framebuffer
|
|
|
|
int visual_attribs[] =
|
|
|
|
{
|
|
|
|
GLX_X_RENDERABLE , True,
|
|
|
|
GLX_DRAWABLE_TYPE , GLX_WINDOW_BIT,
|
|
|
|
GLX_X_VISUAL_TYPE , GLX_TRUE_COLOR,
|
|
|
|
GLX_RED_SIZE , 8,
|
|
|
|
GLX_GREEN_SIZE , 8,
|
|
|
|
GLX_BLUE_SIZE , 8,
|
|
|
|
GLX_DEPTH_SIZE , 0,
|
|
|
|
GLX_STENCIL_SIZE , 0,
|
|
|
|
GLX_DOUBLEBUFFER , True,
|
|
|
|
None
|
|
|
|
};
|
|
|
|
int fbcount = 0;
|
|
|
|
GLXFBConfig* fbc = glXChooseFBConfig(dpy, screen, visual_attribs, &fbcount);
|
|
|
|
if (!fbc || !fbcount)
|
2014-08-30 15:01:19 -06:00
|
|
|
{
|
2014-12-20 02:57:34 -07:00
|
|
|
ERROR_LOG(VIDEO, "Failed to retrieve a framebuffer config");
|
|
|
|
return false;
|
2014-08-30 15:01:19 -06:00
|
|
|
}
|
2014-12-20 02:57:34 -07:00
|
|
|
fbconfig = *fbc;
|
|
|
|
XFree(fbc);
|
|
|
|
|
|
|
|
// Get an appropriate visual
|
2015-03-08 10:42:37 -06:00
|
|
|
XVisualInfo* vi = glXGetVisualFromFBConfig(dpy, fbconfig);
|
2012-12-17 14:01:52 -07:00
|
|
|
|
|
|
|
// Create a GLX context.
|
2014-12-20 02:57:34 -07:00
|
|
|
// We try to get a 3.3 core profile, else we try it with anything we get.
|
|
|
|
int context_attribs[] =
|
2012-12-17 14:01:52 -07:00
|
|
|
{
|
2014-12-20 02:57:34 -07:00
|
|
|
GLX_CONTEXT_MAJOR_VERSION_ARB, 3,
|
|
|
|
GLX_CONTEXT_MINOR_VERSION_ARB, 3,
|
|
|
|
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;
|
|
|
|
XErrorHandler oldHandler = XSetErrorHandler(&ctxErrorHandler);
|
|
|
|
ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, context_attribs);
|
|
|
|
XSync(dpy, False);
|
|
|
|
if (!ctx || s_glxError)
|
|
|
|
{
|
|
|
|
int context_attribs_legacy[] =
|
|
|
|
{
|
|
|
|
GLX_CONTEXT_MAJOR_VERSION_ARB, 1,
|
|
|
|
GLX_CONTEXT_MINOR_VERSION_ARB, 0,
|
|
|
|
None
|
|
|
|
};
|
|
|
|
s_glxError = false;
|
|
|
|
ctx = glXCreateContextAttribs(dpy, fbconfig, 0, True, context_attribs_legacy);
|
|
|
|
XSync(dpy, False);
|
|
|
|
if (!ctx || s_glxError)
|
|
|
|
{
|
|
|
|
ERROR_LOG(VIDEO, "Unable to create GL context.");
|
|
|
|
return false;
|
|
|
|
}
|
2012-12-17 14:01:52 -07:00
|
|
|
}
|
2014-12-20 02:57:34 -07:00
|
|
|
XSetErrorHandler(oldHandler);
|
2012-12-17 14:01:52 -07:00
|
|
|
|
2014-08-09 07:49:45 -06:00
|
|
|
XWindow.Initialize(dpy);
|
|
|
|
|
|
|
|
Window parent = (Window)window_handle;
|
2014-08-21 13:39:03 -06:00
|
|
|
|
|
|
|
XWindowAttributes attribs;
|
|
|
|
if (!XGetWindowAttributes(dpy, parent, &attribs))
|
|
|
|
{
|
|
|
|
ERROR_LOG(VIDEO, "Window attribute retrieval failed");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
s_backbuffer_width = attribs.width;
|
|
|
|
s_backbuffer_height = attribs.height;
|
|
|
|
|
2014-08-09 07:49:45 -06:00
|
|
|
win = XWindow.CreateXWindow(parent, vi);
|
2015-03-08 10:42:37 -06:00
|
|
|
XFree(vi);
|
|
|
|
|
2012-12-17 14:01:52 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cInterfaceGLX::MakeCurrent()
|
|
|
|
{
|
2014-08-09 07:49:45 -06:00
|
|
|
bool success = glXMakeCurrent(dpy, win, ctx);
|
2014-02-24 04:45:02 -07:00
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
// load this function based on the current bound context
|
|
|
|
glXSwapIntervalSGI = (PFNGLXSWAPINTERVALSGIPROC)GLInterface->GetFuncAddress("glXSwapIntervalSGI");
|
|
|
|
}
|
|
|
|
return success;
|
2012-12-17 14:01:52 -07:00
|
|
|
}
|
2013-04-10 19:32:07 -06:00
|
|
|
|
|
|
|
bool cInterfaceGLX::ClearCurrent()
|
|
|
|
{
|
2014-08-09 07:49:45 -06:00
|
|
|
return glXMakeCurrent(dpy, None, nullptr);
|
2013-04-10 19:32:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-17 14:01:52 -07:00
|
|
|
// Close backend
|
|
|
|
void cInterfaceGLX::Shutdown()
|
|
|
|
{
|
2012-12-26 11:12:26 -07:00
|
|
|
XWindow.DestroyXWindow();
|
2014-08-09 07:49:45 -06:00
|
|
|
if (ctx)
|
2012-12-17 14:01:52 -07:00
|
|
|
{
|
2014-08-09 07:49:45 -06:00
|
|
|
glXDestroyContext(dpy, ctx);
|
|
|
|
XCloseDisplay(dpy);
|
|
|
|
ctx = nullptr;
|
2012-12-17 14:01:52 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|