2013-01-13 21:39:25 -05:00
|
|
|
/*
|
2014-02-11 01:07:04 -05:00
|
|
|
* Copyright 2011-2014 Branimir Karadzic. All rights reserved.
|
2013-01-13 21:39:25 -05:00
|
|
|
* License: http://www.opensource.org/licenses/BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "bgfx_p.h"
|
|
|
|
|
2014-08-24 20:41:41 -04:00
|
|
|
#if (BX_PLATFORM_FREEBSD || BX_PLATFORM_LINUX) && (BGFX_CONFIG_RENDERER_OPENGLES || BGFX_CONFIG_RENDERER_OPENGL)
|
2013-01-13 21:39:25 -05:00
|
|
|
# include "renderer_gl.h"
|
2013-04-27 02:43:31 -04:00
|
|
|
# define GLX_GLXEXT_PROTOTYPES
|
|
|
|
# include <glx/glxext.h>
|
2013-01-13 21:39:25 -05:00
|
|
|
|
|
|
|
namespace bgfx
|
|
|
|
{
|
2013-04-28 01:39:15 -04:00
|
|
|
typedef int (*PFNGLXSWAPINTERVALMESAPROC)(uint32_t _interval);
|
|
|
|
|
2013-04-27 18:16:05 -04:00
|
|
|
PFNGLXCREATECONTEXTATTRIBSARBPROC glXCreateContextAttribsARB;
|
|
|
|
PFNGLXSWAPINTERVALEXTPROC glXSwapIntervalEXT;
|
2013-04-28 01:39:15 -04:00
|
|
|
PFNGLXSWAPINTERVALMESAPROC glXSwapIntervalMESA;
|
2013-04-27 18:16:05 -04:00
|
|
|
PFNGLXSWAPINTERVALSGIPROC glXSwapIntervalSGI;
|
|
|
|
|
2013-12-08 01:01:32 -05:00
|
|
|
# define GL_IMPORT(_optional, _proto, _func, _import) _proto _func
|
2014-02-19 02:36:13 -05:00
|
|
|
# include "glimports.h"
|
2013-01-13 21:39:25 -05:00
|
|
|
|
|
|
|
static ::Display* s_display;
|
|
|
|
static ::Window s_window;
|
|
|
|
|
|
|
|
void x11SetDisplayWindow(::Display* _display, ::Window _window)
|
|
|
|
{
|
|
|
|
s_display = _display;
|
|
|
|
s_window = _window;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GlContext::create(uint32_t _width, uint32_t _height)
|
|
|
|
{
|
2013-12-07 13:26:17 -05:00
|
|
|
BX_UNUSED(_width, _height);
|
2013-01-13 21:39:25 -05:00
|
|
|
XLockDisplay(s_display);
|
|
|
|
|
|
|
|
int major, minor;
|
|
|
|
bool version = glXQueryVersion(s_display, &major, &minor);
|
|
|
|
BGFX_FATAL(version, Fatal::UnableToInitialize, "Failed to query GLX version");
|
2013-07-25 01:22:38 -04:00
|
|
|
BGFX_FATAL( (major == 1 && minor >= 2) || major > 1
|
2013-01-13 21:39:25 -05:00
|
|
|
, Fatal::UnableToInitialize
|
2013-07-25 01:22:38 -04:00
|
|
|
, "GLX version is not >=1.2 (%d.%d)."
|
2013-01-13 21:39:25 -05:00
|
|
|
, major
|
|
|
|
, minor
|
|
|
|
);
|
|
|
|
|
2013-04-27 02:43:31 -04:00
|
|
|
int32_t screen = DefaultScreen(s_display);
|
|
|
|
|
|
|
|
const char* extensions = glXQueryExtensionsString(s_display, screen);
|
|
|
|
BX_TRACE("GLX extensions:");
|
|
|
|
dumpExtensions(extensions);
|
|
|
|
|
2013-01-13 21:39:25 -05:00
|
|
|
const int attrsGlx[] =
|
|
|
|
{
|
|
|
|
GLX_RENDER_TYPE, GLX_RGBA_BIT,
|
|
|
|
GLX_DRAWABLE_TYPE, GLX_WINDOW_BIT,
|
|
|
|
GLX_DOUBLEBUFFER, true,
|
|
|
|
GLX_RED_SIZE, 8,
|
|
|
|
GLX_BLUE_SIZE, 8,
|
|
|
|
GLX_GREEN_SIZE, 8,
|
2013-03-11 02:18:38 -04:00
|
|
|
// GLX_ALPHA_SIZE, 8,
|
2013-01-13 21:39:25 -05:00
|
|
|
GLX_DEPTH_SIZE, 24,
|
|
|
|
GLX_STENCIL_SIZE, 8,
|
2013-03-11 02:18:38 -04:00
|
|
|
0,
|
2013-01-13 21:39:25 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
// Find suitable config
|
|
|
|
GLXFBConfig bestConfig = NULL;
|
|
|
|
|
|
|
|
int numConfigs;
|
2013-04-27 02:43:31 -04:00
|
|
|
GLXFBConfig* configs = glXChooseFBConfig(s_display, screen, attrsGlx, &numConfigs);
|
2013-01-13 21:39:25 -05:00
|
|
|
|
|
|
|
BX_TRACE("glX num configs %d", numConfigs);
|
|
|
|
|
2013-10-19 18:36:26 -04:00
|
|
|
XVisualInfo* visualInfo = NULL;
|
2013-01-13 21:39:25 -05:00
|
|
|
for (int ii = 0; ii < numConfigs; ++ii)
|
|
|
|
{
|
|
|
|
visualInfo = glXGetVisualFromFBConfig(s_display, configs[ii]);
|
|
|
|
if (NULL != visualInfo)
|
|
|
|
{
|
|
|
|
BX_TRACE("---");
|
|
|
|
bool valid = true;
|
2013-08-07 00:04:28 -04:00
|
|
|
for (uint32_t attr = 6; attr < BX_COUNTOF(attrsGlx)-1 && attrsGlx[attr] != None; attr += 2)
|
2013-01-13 21:39:25 -05:00
|
|
|
{
|
|
|
|
int value;
|
|
|
|
glXGetFBConfigAttrib(s_display, configs[ii], attrsGlx[attr], &value);
|
|
|
|
BX_TRACE("glX %d/%d %2d: %4x, %8x (%8x%s)"
|
|
|
|
, ii
|
|
|
|
, numConfigs
|
|
|
|
, attr/2
|
|
|
|
, attrsGlx[attr]
|
|
|
|
, value
|
|
|
|
, attrsGlx[attr + 1]
|
|
|
|
, value < attrsGlx[attr + 1] ? " *" : ""
|
|
|
|
);
|
|
|
|
|
|
|
|
if (value < attrsGlx[attr + 1])
|
|
|
|
{
|
|
|
|
valid = false;
|
|
|
|
#if !BGFX_CONFIG_DEBUG
|
|
|
|
break;
|
|
|
|
#endif // BGFX_CONFIG_DEBUG
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (valid)
|
|
|
|
{
|
|
|
|
bestConfig = configs[ii];
|
2013-10-19 18:36:26 -04:00
|
|
|
BX_TRACE("Best config %d.", ii);
|
2013-01-13 21:39:25 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
XFree(visualInfo);
|
2013-10-19 18:36:26 -04:00
|
|
|
visualInfo = NULL;
|
2013-01-13 21:39:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
XFree(configs);
|
|
|
|
BGFX_FATAL(visualInfo, Fatal::UnableToInitialize, "Failed to find a suitable X11 display configuration.");
|
|
|
|
|
|
|
|
BX_TRACE("Create GL 2.1 context.");
|
|
|
|
m_context = glXCreateContext(s_display, visualInfo, 0, GL_TRUE);
|
|
|
|
BGFX_FATAL(NULL != m_context, Fatal::UnableToInitialize, "Failed to create GL 2.1 context.");
|
|
|
|
|
|
|
|
XFree(visualInfo);
|
|
|
|
|
2013-03-09 18:56:26 -05:00
|
|
|
#if BGFX_CONFIG_RENDERER_OPENGL >= 31
|
2013-04-27 18:16:05 -04:00
|
|
|
glXCreateContextAttribsARB = (PFNGLXCREATECONTEXTATTRIBSARBPROC)glXGetProcAddress( (const GLubyte*)"glXCreateContextAttribsARB");
|
2013-05-17 00:31:59 -04:00
|
|
|
|
2013-01-13 21:39:25 -05:00
|
|
|
if (NULL != glXCreateContextAttribsARB)
|
|
|
|
{
|
2013-03-04 23:24:02 -05:00
|
|
|
BX_TRACE("Create GL 3.1 context.");
|
2013-01-13 21:39:25 -05:00
|
|
|
const int contextAttrs[] =
|
|
|
|
{
|
|
|
|
GLX_CONTEXT_MAJOR_VERSION_ARB, 3,
|
2013-03-04 23:24:02 -05:00
|
|
|
GLX_CONTEXT_MINOR_VERSION_ARB, 1,
|
|
|
|
GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB,
|
|
|
|
0,
|
2013-01-13 21:39:25 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
GLXContext context = glXCreateContextAttribsARB(s_display, bestConfig, 0, true, contextAttrs);
|
|
|
|
|
|
|
|
if (NULL != context)
|
|
|
|
{
|
|
|
|
glXDestroyContext(s_display, m_context);
|
|
|
|
m_context = context;
|
|
|
|
}
|
|
|
|
}
|
2013-03-09 18:56:26 -05:00
|
|
|
#else
|
|
|
|
BX_UNUSED(bestConfig);
|
|
|
|
#endif // BGFX_CONFIG_RENDERER_OPENGL >= 31
|
2013-01-13 21:39:25 -05:00
|
|
|
|
|
|
|
XUnlockDisplay(s_display);
|
|
|
|
|
|
|
|
import();
|
|
|
|
|
2013-05-17 00:31:59 -04:00
|
|
|
glXMakeCurrent(s_display, s_window, m_context);
|
|
|
|
|
2013-04-27 18:16:05 -04:00
|
|
|
glXSwapIntervalEXT = (PFNGLXSWAPINTERVALEXTPROC)glXGetProcAddress( (const GLubyte*)"glXSwapIntervalEXT");
|
2013-04-27 02:43:31 -04:00
|
|
|
if (NULL != glXSwapIntervalEXT)
|
|
|
|
{
|
|
|
|
BX_TRACE("Using glXSwapIntervalEXT.");
|
2013-04-27 18:26:23 -04:00
|
|
|
glXSwapIntervalEXT(s_display, s_window, 0);
|
2013-04-27 02:43:31 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-04-28 01:39:15 -04:00
|
|
|
glXSwapIntervalMESA = (PFNGLXSWAPINTERVALMESAPROC)glXGetProcAddress( (const GLubyte*)"glXSwapIntervalMESA");
|
|
|
|
if (NULL != glXSwapIntervalMESA)
|
2013-04-27 02:43:31 -04:00
|
|
|
{
|
2013-04-28 01:39:15 -04:00
|
|
|
BX_TRACE("Using glXSwapIntervalMESA.");
|
|
|
|
glXSwapIntervalMESA(0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
glXSwapIntervalSGI = (PFNGLXSWAPINTERVALSGIPROC)glXGetProcAddress( (const GLubyte*)"glXSwapIntervalSGI");
|
|
|
|
if (NULL != glXSwapIntervalSGI)
|
|
|
|
{
|
|
|
|
BX_TRACE("Using glXSwapIntervalSGI.");
|
|
|
|
glXSwapIntervalSGI(0);
|
|
|
|
}
|
2013-04-27 02:43:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-13 21:39:25 -05:00
|
|
|
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
glXSwapBuffers(s_display, s_window);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GlContext::destroy()
|
|
|
|
{
|
2013-01-18 01:38:13 -05:00
|
|
|
glXMakeCurrent(s_display, 0, 0);
|
|
|
|
glXDestroyContext(s_display, m_context);
|
2013-01-13 21:39:25 -05:00
|
|
|
}
|
|
|
|
|
2013-04-27 18:16:05 -04:00
|
|
|
void GlContext::resize(uint32_t /*_width*/, uint32_t /*_height*/, bool _vsync)
|
2013-01-13 21:39:25 -05:00
|
|
|
{
|
2013-04-27 18:16:05 -04:00
|
|
|
int32_t interval = _vsync ? 1 : 0;
|
|
|
|
|
|
|
|
if (NULL != glXSwapIntervalEXT)
|
|
|
|
{
|
2013-04-27 18:26:23 -04:00
|
|
|
glXSwapIntervalEXT(s_display, s_window, interval);
|
2013-04-27 18:16:05 -04:00
|
|
|
}
|
2013-04-28 01:39:15 -04:00
|
|
|
else if (NULL != glXSwapIntervalMESA)
|
|
|
|
{
|
|
|
|
glXSwapIntervalMESA(interval);
|
|
|
|
}
|
2013-04-27 18:16:05 -04:00
|
|
|
else if (NULL != glXSwapIntervalSGI)
|
|
|
|
{
|
|
|
|
glXSwapIntervalSGI(interval);
|
|
|
|
}
|
2013-01-13 21:39:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void GlContext::swap()
|
|
|
|
{
|
|
|
|
glXSwapBuffers(s_display, s_window);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GlContext::import()
|
|
|
|
{
|
2014-02-19 02:36:13 -05:00
|
|
|
# define GL_EXTENSION(_optional, _proto, _func, _import) \
|
|
|
|
{ \
|
|
|
|
if (NULL == _func) \
|
|
|
|
{ \
|
2014-02-19 23:53:06 -05:00
|
|
|
_func = (_proto)glXGetProcAddress( (const GLubyte*)#_import); \
|
|
|
|
BX_TRACE("%p " #_func " (" #_import ")", _func); \
|
2014-02-19 02:36:13 -05:00
|
|
|
BGFX_FATAL(_optional || NULL != _func, Fatal::UnableToInitialize, "Failed to create OpenGL context. glXGetProcAddress %s", #_import); \
|
|
|
|
} \
|
|
|
|
}
|
2013-01-13 21:39:25 -05:00
|
|
|
# include "glimports.h"
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace bgfx
|
|
|
|
|
2014-08-24 20:41:41 -04:00
|
|
|
#endif // (BX_PLATFORM_FREEBSD || BX_PLATFORM_LINUX) && (BGFX_CONFIG_RENDERER_OPENGLES || BGFX_CONFIG_RENDERER_OPENGL)
|