|
|
@@ -1,393 +0,0 @@
|
|
|
-/*
|
|
|
- * Copyright 2011-2023 Branimir Karadzic. All rights reserved.
|
|
|
- * License: https://github.com/bkaradzic/bgfx/blob/master/LICENSE
|
|
|
- */
|
|
|
-
|
|
|
-#include "bgfx_p.h"
|
|
|
-
|
|
|
-#if (BX_PLATFORM_BSD || BX_PLATFORM_LINUX) && (BGFX_CONFIG_RENDERER_OPENGLES || BGFX_CONFIG_RENDERER_OPENGL)
|
|
|
-# include "renderer_gl.h"
|
|
|
-
|
|
|
-# if BGFX_USE_GLX
|
|
|
-# define GLX_GLXEXT_PROTOTYPES
|
|
|
-# include <glx/glxext.h>
|
|
|
-
|
|
|
-// glxext will include X11 which #defines None, Status, etc.
|
|
|
-#undef None
|
|
|
-#undef Status
|
|
|
-
|
|
|
-namespace bgfx { namespace gl
|
|
|
-{
|
|
|
- typedef int (*PFNGLXSWAPINTERVALMESAPROC)(uint32_t _interval);
|
|
|
-
|
|
|
- PFNGLXCREATECONTEXTATTRIBSARBPROC glXCreateContextAttribsARB;
|
|
|
- PFNGLXSWAPINTERVALEXTPROC glXSwapIntervalEXT;
|
|
|
- PFNGLXSWAPINTERVALMESAPROC glXSwapIntervalMESA;
|
|
|
- PFNGLXSWAPINTERVALSGIPROC glXSwapIntervalSGI;
|
|
|
-
|
|
|
-# define GL_IMPORT(_optional, _proto, _func, _import) _proto _func
|
|
|
-# include "glimports.h"
|
|
|
-
|
|
|
- struct SwapChainGL
|
|
|
- {
|
|
|
- SwapChainGL(::Display* display, ::Window _window, XVisualInfo* _visualInfo, GLXContext _context)
|
|
|
- : m_display(display)
|
|
|
- , m_window(_window)
|
|
|
- {
|
|
|
- m_context = glXCreateContext(m_display, _visualInfo, _context, GL_TRUE);
|
|
|
- }
|
|
|
-
|
|
|
- ~SwapChainGL()
|
|
|
- {
|
|
|
- glXMakeCurrent(m_display, 0, 0);
|
|
|
- glXDestroyContext(m_display, m_context);
|
|
|
- }
|
|
|
-
|
|
|
- void makeCurrent()
|
|
|
- {
|
|
|
- glXMakeCurrent(m_display, m_window, m_context);
|
|
|
- }
|
|
|
-
|
|
|
- void swapBuffers()
|
|
|
- {
|
|
|
- glXSwapBuffers(m_display, m_window);
|
|
|
- }
|
|
|
-
|
|
|
- ::Display* m_display;
|
|
|
- Window m_window;
|
|
|
- GLXContext m_context;
|
|
|
- };
|
|
|
-
|
|
|
- static bool haveGlxExtension(const char* _ext, const char* _extList)
|
|
|
- {
|
|
|
- // _extList is assumed to be a space-separated, null-terminated list of
|
|
|
- // extension names, and no extension name ever contains a space.
|
|
|
- const char* end = _extList + bx::strLen(_extList);
|
|
|
- const char* searchStart = _extList;
|
|
|
- for(;;)
|
|
|
- {
|
|
|
- bx::StringView found = bx::strFind(searchStart, _ext);
|
|
|
- if (found.isEmpty() )
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- // We found the substring, but need an exact match, with a word
|
|
|
- // boundary at both the front and back of the found spot.
|
|
|
- if ((found.getPtr() == _extList || *(found.getPtr() - 1) == ' ')
|
|
|
- && (found.getTerm() == end || *found.getTerm() == ' ') )
|
|
|
- {
|
|
|
- return true;
|
|
|
- }
|
|
|
- // else, keep searching
|
|
|
- searchStart = found.getTerm();
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- template<typename ProtoT>
|
|
|
- static ProtoT glXGetProcAddress(const char* _name)
|
|
|
- {
|
|
|
- return reinterpret_cast<ProtoT>( (void*)::glXGetProcAddress( (const GLubyte*)_name) );
|
|
|
- }
|
|
|
-
|
|
|
- void GlContext::create(uint32_t _width, uint32_t _height, uint32_t /*_flags*/)
|
|
|
- {
|
|
|
- BX_UNUSED(_width, _height);
|
|
|
-
|
|
|
- m_context = (GLXContext)g_platformData.context;
|
|
|
- m_display = (::Display*)g_platformData.ndt;
|
|
|
-
|
|
|
- // It's possible the user has provided the window handle, but not
|
|
|
- // the display handle. If this is the case, try opening the default
|
|
|
- // display
|
|
|
- if (NULL == m_display)
|
|
|
- {
|
|
|
- m_display = XOpenDisplay(NULL);
|
|
|
- BGFX_FATAL(m_display, Fatal::UnableToInitialize, "XOpenDisplay(NULL) : Failed to open default display");
|
|
|
- }
|
|
|
-
|
|
|
- if (NULL == g_platformData.context)
|
|
|
- {
|
|
|
- XLockDisplay(m_display);
|
|
|
-
|
|
|
- int major, minor;
|
|
|
- bool version = glXQueryVersion(m_display, &major, &minor);
|
|
|
- BGFX_FATAL(version, Fatal::UnableToInitialize, "Failed to query GLX version");
|
|
|
- BGFX_FATAL( (major == 1 && minor >= 2) || major > 1
|
|
|
- , Fatal::UnableToInitialize
|
|
|
- , "GLX version is not >=1.2 (%d.%d)."
|
|
|
- , major
|
|
|
- , minor
|
|
|
- );
|
|
|
-
|
|
|
- int32_t screen = DefaultScreen(m_display);
|
|
|
-
|
|
|
- const char* extensions = glXQueryExtensionsString(m_display, screen);
|
|
|
- BX_TRACE("GLX extensions:");
|
|
|
- dumpExtensions(extensions);
|
|
|
-
|
|
|
- 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,
|
|
|
- // GLX_ALPHA_SIZE, 8,
|
|
|
- GLX_DEPTH_SIZE, 24,
|
|
|
- GLX_STENCIL_SIZE, 8,
|
|
|
- 0,
|
|
|
- };
|
|
|
-
|
|
|
- // Find suitable config
|
|
|
- GLXFBConfig bestConfig = NULL;
|
|
|
-
|
|
|
- int numConfigs;
|
|
|
- GLXFBConfig* configs = glXChooseFBConfig(m_display, screen, attrsGlx, &numConfigs);
|
|
|
-
|
|
|
- BX_TRACE("glX num configs %d", numConfigs);
|
|
|
-
|
|
|
- for (int ii = 0; ii < numConfigs; ++ii)
|
|
|
- {
|
|
|
- m_visualInfo = glXGetVisualFromFBConfig(m_display, configs[ii]);
|
|
|
- if (NULL != m_visualInfo)
|
|
|
- {
|
|
|
- BX_TRACE("---");
|
|
|
- bool valid = true;
|
|
|
- for (uint32_t attr = 6; attr < BX_COUNTOF(attrsGlx)-1 && attrsGlx[attr] != 0; attr += 2)
|
|
|
- {
|
|
|
- int value;
|
|
|
- glXGetFBConfigAttrib(m_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];
|
|
|
- BX_TRACE("Best config %d.", ii);
|
|
|
- break;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- XFree(m_visualInfo);
|
|
|
- m_visualInfo = NULL;
|
|
|
- }
|
|
|
-
|
|
|
- XFree(configs);
|
|
|
- BGFX_FATAL(m_visualInfo, Fatal::UnableToInitialize, "Failed to find a suitable X11 display configuration.");
|
|
|
-
|
|
|
- BX_TRACE("Create GL 2.1 context.");
|
|
|
- m_context = glXCreateContext(m_display, m_visualInfo, 0, GL_TRUE);
|
|
|
- BGFX_FATAL(NULL != m_context, Fatal::UnableToInitialize, "Failed to create GL 2.1 context.");
|
|
|
-
|
|
|
- glXCreateContextAttribsARB = glXGetProcAddress<PFNGLXCREATECONTEXTATTRIBSARBPROC>("glXCreateContextAttribsARB");
|
|
|
-
|
|
|
- if (NULL != glXCreateContextAttribsARB)
|
|
|
- {
|
|
|
- BX_TRACE("Create GL %d.%d context.", BGFX_CONFIG_RENDERER_OPENGL / 10, BGFX_CONFIG_RENDERER_OPENGL % 10);
|
|
|
- int32_t flags = BGFX_CONFIG_DEBUG ? GLX_CONTEXT_DEBUG_BIT_ARB : 0;
|
|
|
- const int contextAttrs[] =
|
|
|
- {
|
|
|
- GLX_CONTEXT_MAJOR_VERSION_ARB, BGFX_CONFIG_RENDERER_OPENGL / 10,
|
|
|
- GLX_CONTEXT_MINOR_VERSION_ARB, BGFX_CONFIG_RENDERER_OPENGL % 10,
|
|
|
- GLX_CONTEXT_FLAGS_ARB, flags,
|
|
|
- GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB,
|
|
|
- 0,
|
|
|
- };
|
|
|
-
|
|
|
- GLXContext context = glXCreateContextAttribsARB(m_display, bestConfig, 0, true, contextAttrs);
|
|
|
-
|
|
|
- if (NULL != context)
|
|
|
- {
|
|
|
- glXDestroyContext(m_display, m_context);
|
|
|
- m_context = context;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- XUnlockDisplay(m_display);
|
|
|
- }
|
|
|
-
|
|
|
- import();
|
|
|
-
|
|
|
- glXMakeCurrent(m_display, (::Window)g_platformData.nwh, m_context);
|
|
|
- m_current = NULL;
|
|
|
-
|
|
|
- const char* extensions = glXQueryExtensionsString(m_display, DefaultScreen(m_display) );
|
|
|
-
|
|
|
- if (NULL != extensions)
|
|
|
- {
|
|
|
- bool foundSwapControl = false;
|
|
|
-
|
|
|
- if (haveGlxExtension("GLX_EXT_swap_control", extensions) )
|
|
|
- {
|
|
|
- glXSwapIntervalEXT = glXGetProcAddress<PFNGLXSWAPINTERVALEXTPROC>("glXSwapIntervalEXT");
|
|
|
-
|
|
|
- if (NULL != glXSwapIntervalEXT)
|
|
|
- {
|
|
|
- BX_TRACE("Using glXSwapIntervalEXT.");
|
|
|
- glXSwapIntervalEXT(m_display, (::Window)g_platformData.nwh, 0);
|
|
|
- foundSwapControl = true;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if (!foundSwapControl
|
|
|
- && haveGlxExtension("GLX_MESA_swap_control", extensions) )
|
|
|
- {
|
|
|
- glXSwapIntervalMESA = glXGetProcAddress<PFNGLXSWAPINTERVALMESAPROC>("glXSwapIntervalMESA");
|
|
|
-
|
|
|
- if (NULL != glXSwapIntervalMESA)
|
|
|
- {
|
|
|
- BX_TRACE("Using glXSwapIntervalMESA.");
|
|
|
- glXSwapIntervalMESA(0);
|
|
|
- foundSwapControl = true;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if (!foundSwapControl
|
|
|
- && haveGlxExtension("GLX_SGI_swap_control", extensions) )
|
|
|
- {
|
|
|
- glXSwapIntervalSGI = glXGetProcAddress<PFNGLXSWAPINTERVALSGIPROC>("glXSwapIntervalSGI");
|
|
|
-
|
|
|
- if (NULL != glXSwapIntervalSGI)
|
|
|
- {
|
|
|
- BX_TRACE("Using glXSwapIntervalSGI.");
|
|
|
- glXSwapIntervalSGI(0);
|
|
|
- foundSwapControl = true;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
|
|
|
- glClear(GL_COLOR_BUFFER_BIT);
|
|
|
- glXSwapBuffers(m_display, (::Window)g_platformData.nwh);
|
|
|
-
|
|
|
- g_internalData.context = m_context;
|
|
|
- }
|
|
|
-
|
|
|
- void GlContext::destroy()
|
|
|
- {
|
|
|
- glXMakeCurrent(m_display, 0, 0);
|
|
|
- if (NULL == g_platformData.context)
|
|
|
- {
|
|
|
- glXDestroyContext(m_display, m_context);
|
|
|
- XFree(m_visualInfo);
|
|
|
- }
|
|
|
- // If we opened the display, have to close it
|
|
|
- if (NULL == g_platformData.ndt)
|
|
|
- {
|
|
|
- XCloseDisplay(m_display);
|
|
|
- }
|
|
|
- m_context = NULL;
|
|
|
- m_visualInfo = NULL;
|
|
|
- m_display = NULL;
|
|
|
- }
|
|
|
-
|
|
|
- void GlContext::resize(uint32_t /*_width*/, uint32_t /*_height*/, uint32_t _flags)
|
|
|
- {
|
|
|
- bool vsync = !!(_flags&BGFX_RESET_VSYNC);
|
|
|
- int32_t interval = vsync ? 1 : 0;
|
|
|
-
|
|
|
- if (NULL != glXSwapIntervalEXT)
|
|
|
- {
|
|
|
- glXSwapIntervalEXT(m_display, (::Window)g_platformData.nwh, interval);
|
|
|
- }
|
|
|
- else if (NULL != glXSwapIntervalMESA)
|
|
|
- {
|
|
|
- glXSwapIntervalMESA(interval);
|
|
|
- }
|
|
|
- else if (NULL != glXSwapIntervalSGI)
|
|
|
- {
|
|
|
- glXSwapIntervalSGI(interval);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- uint64_t GlContext::getCaps() const
|
|
|
- {
|
|
|
- return BGFX_CAPS_SWAP_CHAIN;
|
|
|
- }
|
|
|
-
|
|
|
- SwapChainGL* GlContext::createSwapChain(void* _nwh)
|
|
|
- {
|
|
|
- return BX_NEW(g_allocator, SwapChainGL)(m_display, (::Window)_nwh, m_visualInfo, m_context);
|
|
|
- }
|
|
|
-
|
|
|
- void GlContext::destroySwapChain(SwapChainGL* _swapChain)
|
|
|
- {
|
|
|
- bx::deleteObject(g_allocator, _swapChain);
|
|
|
- glXMakeCurrent(m_display, (::Window)g_platformData.nwh, m_context);
|
|
|
- }
|
|
|
-
|
|
|
- void GlContext::swap(SwapChainGL* _swapChain)
|
|
|
- {
|
|
|
- makeCurrent(_swapChain);
|
|
|
-
|
|
|
- if (NULL == _swapChain)
|
|
|
- {
|
|
|
- glXSwapBuffers(m_display, (::Window)g_platformData.nwh);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- _swapChain->swapBuffers();
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- void GlContext::makeCurrent(SwapChainGL* _swapChain)
|
|
|
- {
|
|
|
- if (m_current != _swapChain)
|
|
|
- {
|
|
|
- m_current = _swapChain;
|
|
|
-
|
|
|
- if (NULL == _swapChain)
|
|
|
- {
|
|
|
- glXMakeCurrent(m_display, (::Window)g_platformData.nwh, m_context);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- _swapChain->makeCurrent();
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- void GlContext::import()
|
|
|
- {
|
|
|
- BX_TRACE("Import:");
|
|
|
-
|
|
|
-# define GL_EXTENSION(_optional, _proto, _func, _import) \
|
|
|
- { \
|
|
|
- if (NULL == _func) \
|
|
|
- { \
|
|
|
- _func = glXGetProcAddress<_proto>(#_import); \
|
|
|
- BX_TRACE("%p " #_func " (" #_import ")", _func); \
|
|
|
- BGFX_FATAL(_optional || NULL != _func \
|
|
|
- , Fatal::UnableToInitialize \
|
|
|
- , "Failed to create OpenGL context. glXGetProcAddress %s", #_import); \
|
|
|
- } \
|
|
|
- }
|
|
|
-
|
|
|
-# include "glimports.h"
|
|
|
-
|
|
|
-# undef GL_EXTENSION
|
|
|
- }
|
|
|
-
|
|
|
-} /* namespace gl */ } // namespace bgfx
|
|
|
-
|
|
|
-# endif // BGFX_USE_GLX
|
|
|
-
|
|
|
-#endif // (BX_PLATFORM_BSD || BX_PLATFORM_LINUX) && (BGFX_CONFIG_RENDERER_OPENGLES || BGFX_CONFIG_RENDERER_OPENGL)
|