2014-11-05 00:46:40 +00:00
|
|
|
#include <base/detect.h>
|
|
|
|
|
|
|
|
#if defined(CONF_FAMILY_WINDOWS)
|
2020-09-26 19:41:58 +00:00
|
|
|
// For FlashWindowEx, FLASHWINFO, FLASHW_TRAY
|
|
|
|
#define _WIN32_WINNT 0x0501
|
|
|
|
#define WINVER 0x0501
|
2014-11-05 00:46:40 +00:00
|
|
|
#endif
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
#ifndef CONF_BACKEND_OPENGL_ES
|
2018-02-12 22:14:14 +00:00
|
|
|
#include <GL/glew.h>
|
2021-04-30 22:42:37 +00:00
|
|
|
#endif
|
|
|
|
|
2017-12-02 21:19:57 +00:00
|
|
|
#include <engine/storage.h>
|
2017-10-20 07:08:49 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
#include "SDL.h"
|
2021-04-30 22:42:37 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
#include "SDL_syswm.h"
|
2015-08-24 20:46:28 +00:00
|
|
|
#include <base/detect.h>
|
2016-04-29 22:34:12 +00:00
|
|
|
#include <base/math.h>
|
2018-03-13 20:44:58 +00:00
|
|
|
#include <cmath>
|
2020-09-26 19:41:58 +00:00
|
|
|
#include <stdlib.h>
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2021-01-31 20:54:04 +00:00
|
|
|
#include "SDL_hints.h"
|
|
|
|
|
2014-10-26 00:25:36 +00:00
|
|
|
#if defined(SDL_VIDEO_DRIVER_X11)
|
2020-09-26 19:41:58 +00:00
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
2014-10-26 00:25:36 +00:00
|
|
|
#endif
|
|
|
|
|
2014-10-28 00:12:11 +00:00
|
|
|
#include <engine/shared/config.h>
|
2016-08-30 23:39:59 +00:00
|
|
|
|
2012-02-05 12:22:39 +00:00
|
|
|
#include <base/tl/threading.h>
|
|
|
|
|
2016-08-27 19:10:27 +00:00
|
|
|
#if defined(CONF_VIDEORECORDER)
|
2020-09-26 19:41:58 +00:00
|
|
|
#include "video.h"
|
2016-08-27 19:10:27 +00:00
|
|
|
#endif
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
#include "backend_sdl.h"
|
|
|
|
|
2021-05-02 00:52:13 +00:00
|
|
|
#if !defined(CONF_BACKEND_OPENGL_ES)
|
|
|
|
#include "backend/opengl/backend_opengl3.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(CONF_BACKEND_OPENGL_ES3) || defined(CONF_BACKEND_OPENGL_ES)
|
|
|
|
#include "backend/opengles/backend_opengles3.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "graphics_threaded.h"
|
2017-09-02 13:24:07 +00:00
|
|
|
|
2020-11-18 06:36:19 +00:00
|
|
|
#include <engine/shared/image_manipulation.h>
|
|
|
|
|
2017-10-17 14:38:40 +00:00
|
|
|
#ifdef __MINGW32__
|
2020-09-26 19:41:58 +00:00
|
|
|
extern "C" {
|
|
|
|
int putenv(const char *);
|
2017-10-17 14:38:40 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-08-09 15:56:47 +00:00
|
|
|
/*
|
|
|
|
sync_barrier - creates a full hardware fence
|
|
|
|
*/
|
|
|
|
#if defined(__GNUC__)
|
2020-09-26 19:41:58 +00:00
|
|
|
inline void sync_barrier()
|
|
|
|
{
|
|
|
|
__sync_synchronize();
|
|
|
|
}
|
2020-08-09 15:56:47 +00:00
|
|
|
#elif defined(_MSC_VER)
|
2020-09-26 19:41:58 +00:00
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <windows.h>
|
|
|
|
inline void sync_barrier()
|
|
|
|
{
|
|
|
|
MemoryBarrier();
|
|
|
|
}
|
2020-08-09 15:56:47 +00:00
|
|
|
#else
|
2020-09-26 19:41:58 +00:00
|
|
|
#error missing atomic implementation for this compiler
|
2020-08-09 15:56:47 +00:00
|
|
|
#endif
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
// ------------ CGraphicsBackend_Threaded
|
|
|
|
|
|
|
|
void CGraphicsBackend_Threaded::ThreadFunc(void *pUser)
|
|
|
|
{
|
|
|
|
CGraphicsBackend_Threaded *pThis = (CGraphicsBackend_Threaded *)pUser;
|
|
|
|
|
|
|
|
while(!pThis->m_Shutdown)
|
|
|
|
{
|
2020-11-29 16:53:54 +00:00
|
|
|
pThis->m_Activity.Wait();
|
2012-01-03 20:39:10 +00:00
|
|
|
if(pThis->m_pBuffer)
|
|
|
|
{
|
2021-02-12 12:40:29 +00:00
|
|
|
#ifdef CONF_PLATFORM_MACOS
|
2020-09-26 19:41:58 +00:00
|
|
|
CAutoreleasePool AutoreleasePool;
|
|
|
|
#endif
|
2012-01-03 20:39:10 +00:00
|
|
|
pThis->m_pProcessor->RunBuffer(pThis->m_pBuffer);
|
2016-08-30 23:39:59 +00:00
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
sync_barrier();
|
|
|
|
pThis->m_pBuffer = 0x0;
|
2020-11-29 16:53:54 +00:00
|
|
|
pThis->m_BufferDone.Signal();
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
2020-09-26 19:41:58 +00:00
|
|
|
#if defined(CONF_VIDEORECORDER)
|
|
|
|
if(IVideo::Current())
|
|
|
|
IVideo::Current()->NextVideoFrameThread();
|
|
|
|
#endif
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CGraphicsBackend_Threaded::CGraphicsBackend_Threaded()
|
|
|
|
{
|
|
|
|
m_pBuffer = 0x0;
|
|
|
|
m_pProcessor = 0x0;
|
|
|
|
m_pThread = 0x0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGraphicsBackend_Threaded::StartProcessor(ICommandProcessor *pProcessor)
|
|
|
|
{
|
|
|
|
m_Shutdown = false;
|
|
|
|
m_pProcessor = pProcessor;
|
2019-03-19 10:44:16 +00:00
|
|
|
m_pThread = thread_init(ThreadFunc, this, "CGraphicsBackend_Threaded");
|
2020-11-29 16:53:54 +00:00
|
|
|
m_BufferDone.Signal();
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CGraphicsBackend_Threaded::StopProcessor()
|
|
|
|
{
|
|
|
|
m_Shutdown = true;
|
2020-11-29 16:53:54 +00:00
|
|
|
m_Activity.Signal();
|
2019-03-19 10:44:16 +00:00
|
|
|
if(m_pThread)
|
|
|
|
thread_wait(m_pThread);
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CGraphicsBackend_Threaded::RunBuffer(CCommandBuffer *pBuffer)
|
|
|
|
{
|
|
|
|
WaitForIdle();
|
|
|
|
m_pBuffer = pBuffer;
|
2020-11-29 16:53:54 +00:00
|
|
|
m_Activity.Signal();
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CGraphicsBackend_Threaded::IsIdle() const
|
|
|
|
{
|
|
|
|
return m_pBuffer == 0x0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGraphicsBackend_Threaded::WaitForIdle()
|
|
|
|
{
|
|
|
|
while(m_pBuffer != 0x0)
|
2020-11-29 16:53:54 +00:00
|
|
|
m_BufferDone.Wait();
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ------------ CCommandProcessorFragment_General
|
|
|
|
|
|
|
|
void CCommandProcessorFragment_General::Cmd_Signal(const CCommandBuffer::SCommand_Signal *pCommand)
|
|
|
|
{
|
2020-11-29 16:53:54 +00:00
|
|
|
pCommand->m_pSemaphore->Signal();
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
bool CCommandProcessorFragment_General::RunCommand(const CCommandBuffer::SCommand *pBaseCommand)
|
2012-01-03 20:39:10 +00:00
|
|
|
{
|
|
|
|
switch(pBaseCommand->m_Cmd)
|
|
|
|
{
|
|
|
|
case CCommandBuffer::CMD_NOP: break;
|
|
|
|
case CCommandBuffer::CMD_SIGNAL: Cmd_Signal(static_cast<const CCommandBuffer::SCommand_Signal *>(pBaseCommand)); break;
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2021-04-30 22:42:37 +00:00
|
|
|
}
|
2020-08-29 10:10:38 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
// ------------ CCommandProcessorFragment_SDL
|
|
|
|
void CCommandProcessorFragment_SDL::Cmd_Init(const SCommand_Init *pCommand)
|
|
|
|
{
|
|
|
|
m_GLContext = pCommand->m_GLContext;
|
|
|
|
m_pWindow = pCommand->m_pWindow;
|
|
|
|
SDL_GL_MakeCurrent(m_pWindow, m_GLContext);
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CCommandProcessorFragment_SDL::Cmd_Shutdown(const SCommand_Shutdown *pCommand)
|
|
|
|
{
|
2015-08-24 20:46:28 +00:00
|
|
|
SDL_GL_MakeCurrent(NULL, NULL);
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CCommandProcessorFragment_SDL::Cmd_Swap(const CCommandBuffer::SCommand_Swap *pCommand)
|
|
|
|
{
|
2015-08-24 20:46:28 +00:00
|
|
|
SDL_GL_SwapWindow(m_pWindow);
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
2016-04-29 22:34:12 +00:00
|
|
|
void CCommandProcessorFragment_SDL::Cmd_VSync(const CCommandBuffer::SCommand_VSync *pCommand)
|
|
|
|
{
|
|
|
|
*pCommand->m_pRetOk = SDL_GL_SetSwapInterval(pCommand->m_VSync) == 0;
|
|
|
|
}
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
void CCommandProcessorFragment_SDL::Cmd_VideoModes(const CCommandBuffer::SCommand_VideoModes *pCommand)
|
|
|
|
{
|
2015-08-24 20:46:28 +00:00
|
|
|
SDL_DisplayMode mode;
|
|
|
|
int maxModes = SDL_GetNumDisplayModes(pCommand->m_Screen),
|
2020-09-26 19:41:58 +00:00
|
|
|
numModes = 0;
|
2015-08-24 20:46:28 +00:00
|
|
|
|
|
|
|
for(int i = 0; i < maxModes; i++)
|
2012-01-03 20:39:10 +00:00
|
|
|
{
|
2015-08-24 20:46:28 +00:00
|
|
|
if(SDL_GetDisplayMode(pCommand->m_Screen, i, &mode) < 0)
|
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to get display mode: %s", SDL_GetError());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-03-21 10:24:44 +00:00
|
|
|
bool AlreadyFound = false;
|
2015-08-24 20:46:28 +00:00
|
|
|
for(int j = 0; j < numModes; j++)
|
2012-01-03 20:39:10 +00:00
|
|
|
{
|
2015-08-24 20:46:28 +00:00
|
|
|
if(pCommand->m_pModes[j].m_Width == mode.w && pCommand->m_pModes[j].m_Height == mode.h)
|
|
|
|
{
|
2017-03-21 10:24:44 +00:00
|
|
|
AlreadyFound = true;
|
2012-01-03 20:39:10 +00:00
|
|
|
break;
|
2015-08-24 20:46:28 +00:00
|
|
|
}
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
2017-03-21 10:24:44 +00:00
|
|
|
if(AlreadyFound)
|
2015-08-24 20:46:28 +00:00
|
|
|
continue;
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2015-08-24 20:46:28 +00:00
|
|
|
pCommand->m_pModes[numModes].m_Width = mode.w;
|
|
|
|
pCommand->m_pModes[numModes].m_Height = mode.h;
|
|
|
|
pCommand->m_pModes[numModes].m_Red = 8;
|
|
|
|
pCommand->m_pModes[numModes].m_Green = 8;
|
|
|
|
pCommand->m_pModes[numModes].m_Blue = 8;
|
|
|
|
numModes++;
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
2015-08-24 20:46:28 +00:00
|
|
|
*pCommand->m_pNumModes = numModes;
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CCommandProcessorFragment_SDL::CCommandProcessorFragment_SDL()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CCommandProcessorFragment_SDL::RunCommand(const CCommandBuffer::SCommand *pBaseCommand)
|
|
|
|
{
|
|
|
|
switch(pBaseCommand->m_Cmd)
|
|
|
|
{
|
|
|
|
case CCommandBuffer::CMD_SWAP: Cmd_Swap(static_cast<const CCommandBuffer::SCommand_Swap *>(pBaseCommand)); break;
|
2016-04-29 22:34:12 +00:00
|
|
|
case CCommandBuffer::CMD_VSYNC: Cmd_VSync(static_cast<const CCommandBuffer::SCommand_VSync *>(pBaseCommand)); break;
|
2012-01-03 20:39:10 +00:00
|
|
|
case CCommandBuffer::CMD_VIDEOMODES: Cmd_VideoModes(static_cast<const CCommandBuffer::SCommand_VideoModes *>(pBaseCommand)); break;
|
|
|
|
case CMD_INIT: Cmd_Init(static_cast<const SCommand_Init *>(pBaseCommand)); break;
|
|
|
|
case CMD_SHUTDOWN: Cmd_Shutdown(static_cast<const SCommand_Shutdown *>(pBaseCommand)); break;
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------ CCommandProcessor_SDL_OpenGL
|
|
|
|
|
|
|
|
void CCommandProcessor_SDL_OpenGL::RunBuffer(CCommandBuffer *pBuffer)
|
|
|
|
{
|
2020-10-11 15:08:04 +00:00
|
|
|
for(CCommandBuffer::SCommand *pCommand = pBuffer->Head(); pCommand; pCommand = pCommand->m_pNext)
|
2012-01-03 20:39:10 +00:00
|
|
|
{
|
2020-10-11 15:08:04 +00:00
|
|
|
if(m_pOpenGL->RunCommand(pCommand))
|
2020-09-26 19:41:58 +00:00
|
|
|
continue;
|
2020-08-29 10:10:38 +00:00
|
|
|
|
2020-10-11 15:08:04 +00:00
|
|
|
if(m_SDL.RunCommand(pCommand))
|
2012-01-03 20:39:10 +00:00
|
|
|
continue;
|
|
|
|
|
2020-10-11 15:08:04 +00:00
|
|
|
if(m_General.RunCommand(pCommand))
|
2012-01-03 20:39:10 +00:00
|
|
|
continue;
|
2015-07-09 00:08:14 +00:00
|
|
|
|
2020-10-26 08:57:41 +00:00
|
|
|
dbg_msg("gfx", "unknown command %d", pCommand->m_Cmd);
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
CCommandProcessor_SDL_OpenGL::CCommandProcessor_SDL_OpenGL(EBackendType BackendType, int OpenGLMajor, int OpenGLMinor, int OpenGLPatch)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
m_BackendType = BackendType;
|
|
|
|
|
|
|
|
if(BackendType == BACKEND_TYPE_OPENGL_ES)
|
2020-09-26 19:41:58 +00:00
|
|
|
{
|
2021-05-02 00:52:13 +00:00
|
|
|
#if defined(CONF_BACKEND_OPENGL_ES) || defined(CONF_BACKEND_OPENGL_ES3)
|
2021-04-30 22:42:37 +00:00
|
|
|
if(OpenGLMajor < 3)
|
|
|
|
{
|
2021-05-02 00:52:13 +00:00
|
|
|
m_pOpenGL = new CCommandProcessorFragment_OpenGLES();
|
2021-04-30 22:42:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-05-02 00:52:13 +00:00
|
|
|
m_pOpenGL = new CCommandProcessorFragment_OpenGLES3();
|
2021-04-30 22:42:37 +00:00
|
|
|
}
|
2021-05-02 00:52:13 +00:00
|
|
|
#endif
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(BackendType == BACKEND_TYPE_OPENGL)
|
2020-09-26 19:41:58 +00:00
|
|
|
{
|
2021-05-02 00:52:13 +00:00
|
|
|
#if !defined(CONF_BACKEND_OPENGL_ES)
|
2021-04-30 22:42:37 +00:00
|
|
|
if(OpenGLMajor < 2)
|
|
|
|
{
|
|
|
|
m_pOpenGL = new CCommandProcessorFragment_OpenGL();
|
|
|
|
}
|
|
|
|
if(OpenGLMajor == 2)
|
|
|
|
{
|
|
|
|
m_pOpenGL = new CCommandProcessorFragment_OpenGL2();
|
|
|
|
}
|
|
|
|
if(OpenGLMajor == 3 && OpenGLMinor == 0)
|
|
|
|
{
|
|
|
|
m_pOpenGL = new CCommandProcessorFragment_OpenGL3();
|
|
|
|
}
|
|
|
|
else if((OpenGLMajor == 3 && OpenGLMinor == 3) || OpenGLMajor >= 4)
|
|
|
|
{
|
|
|
|
m_pOpenGL = new CCommandProcessorFragment_OpenGL3_3();
|
|
|
|
}
|
2021-05-02 00:52:13 +00:00
|
|
|
#endif
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-30 21:51:33 +00:00
|
|
|
CCommandProcessor_SDL_OpenGL::~CCommandProcessor_SDL_OpenGL()
|
|
|
|
{
|
|
|
|
delete m_pOpenGL;
|
|
|
|
}
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
// ------------ CGraphicsBackend_SDL_OpenGL
|
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
static bool BackendInitGlew(EBackendType BackendType, int &GlewMajor, int &GlewMinor, int &GlewPatch)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
if(BackendType == BACKEND_TYPE_OPENGL)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
#ifndef CONF_BACKEND_OPENGL_ES
|
|
|
|
//support graphic cards that are pretty old(and linux)
|
|
|
|
glewExperimental = GL_TRUE;
|
|
|
|
if(GLEW_OK != glewInit())
|
|
|
|
return false;
|
2020-08-29 10:10:38 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
#ifdef GLEW_VERSION_4_6
|
|
|
|
if(GLEW_VERSION_4_6)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
GlewMajor = 4;
|
|
|
|
GlewMinor = 6;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
#endif
|
|
|
|
if(GLEW_VERSION_4_5)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
GlewMajor = 4;
|
|
|
|
GlewMinor = 5;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
if(GLEW_VERSION_4_4)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
GlewMajor = 4;
|
|
|
|
GlewMinor = 4;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
if(GLEW_VERSION_4_3)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
GlewMajor = 4;
|
|
|
|
GlewMinor = 3;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
if(GLEW_VERSION_4_2)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
GlewMajor = 4;
|
|
|
|
GlewMinor = 2;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
if(GLEW_VERSION_4_1)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
GlewMajor = 4;
|
|
|
|
GlewMinor = 1;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_4_0)
|
|
|
|
{
|
|
|
|
GlewMajor = 4;
|
|
|
|
GlewMinor = 0;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_3_3)
|
|
|
|
{
|
|
|
|
GlewMajor = 3;
|
|
|
|
GlewMinor = 3;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_3_0)
|
|
|
|
{
|
|
|
|
GlewMajor = 3;
|
|
|
|
GlewMinor = 0;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_2_1)
|
|
|
|
{
|
|
|
|
GlewMajor = 2;
|
|
|
|
GlewMinor = 1;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_2_0)
|
|
|
|
{
|
|
|
|
GlewMajor = 2;
|
|
|
|
GlewMinor = 0;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_1_5)
|
|
|
|
{
|
|
|
|
GlewMajor = 1;
|
|
|
|
GlewMinor = 5;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_1_4)
|
|
|
|
{
|
|
|
|
GlewMajor = 1;
|
|
|
|
GlewMinor = 4;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_1_3)
|
|
|
|
{
|
|
|
|
GlewMajor = 1;
|
|
|
|
GlewMinor = 3;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_1_2_1)
|
|
|
|
{
|
|
|
|
GlewMajor = 1;
|
|
|
|
GlewMinor = 2;
|
|
|
|
GlewPatch = 1;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_1_2)
|
|
|
|
{
|
|
|
|
GlewMajor = 1;
|
|
|
|
GlewMinor = 2;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(GLEW_VERSION_1_1)
|
|
|
|
{
|
|
|
|
GlewMajor = 1;
|
|
|
|
GlewMinor = 1;
|
|
|
|
GlewPatch = 0;
|
|
|
|
return true;
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
#endif
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(BackendType == BACKEND_TYPE_OPENGL_ES)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
// just assume the version we need
|
|
|
|
GlewMajor = 3;
|
|
|
|
GlewMinor = 0;
|
|
|
|
GlewPatch = 0;
|
|
|
|
|
|
|
|
return true;
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int IsVersionSupportedGlew(EBackendType BackendType, int VersionMajor, int VersionMinor, int VersionPatch, int GlewMajor, int GlewMinor, int GlewPatch)
|
|
|
|
{
|
|
|
|
int InitError = 0;
|
|
|
|
|
|
|
|
if(BackendType == BACKEND_TYPE_OPENGL)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
if(VersionMajor >= 4 && GlewMajor < 4)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(VersionMajor >= 3 && GlewMajor < 3)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(VersionMajor == 3 && GlewMajor == 3)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
if(VersionMinor >= 3 && GlewMinor < 3)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 2 && GlewMinor < 2)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 1 && GlewMinor < 1)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 0 && GlewMinor < 0)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(VersionMajor >= 2 && GlewMajor < 2)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(VersionMajor == 2 && GlewMajor == 2)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
if(VersionMinor >= 1 && GlewMinor < 1)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
if(VersionMinor >= 0 && GlewMinor < 0)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(VersionMajor >= 1 && GlewMajor < 1)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(VersionMajor == 1 && GlewMajor == 1)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
if(VersionMinor >= 5 && GlewMinor < 5)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 4 && GlewMinor < 4)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 3 && GlewMinor < 3)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 2 && GlewMinor < 2)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
else if(VersionMinor == 2 && GlewMinor == 2)
|
|
|
|
{
|
|
|
|
if(VersionPatch >= 1 && GlewPatch < 1)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionPatch >= 0 && GlewPatch < 0)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 1 && GlewMinor < 1)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
|
|
|
if(VersionMinor >= 0 && GlewMinor < 0)
|
|
|
|
{
|
|
|
|
InitError = -1;
|
|
|
|
}
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return InitError;
|
|
|
|
}
|
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
EBackendType CGraphicsBackend_SDL_OpenGL::DetectBackend()
|
|
|
|
{
|
|
|
|
#ifndef CONF_BACKEND_OPENGL_ES
|
2021-05-02 00:52:13 +00:00
|
|
|
#ifdef CONF_BACKEND_OPENGL_ES3
|
|
|
|
const char *pEnvDriver = getenv("DDNET_DRIVER");
|
|
|
|
if(pEnvDriver && str_comp(pEnvDriver, "GLES") == 0)
|
|
|
|
return BACKEND_TYPE_OPENGL_ES;
|
|
|
|
else
|
|
|
|
return BACKEND_TYPE_OPENGL;
|
|
|
|
#else
|
2021-04-30 22:42:37 +00:00
|
|
|
return BACKEND_TYPE_OPENGL;
|
2021-05-02 00:52:13 +00:00
|
|
|
#endif
|
2021-04-30 22:42:37 +00:00
|
|
|
#else
|
|
|
|
return BACKEND_TYPE_OPENGL_ES;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGraphicsBackend_SDL_OpenGL::ClampDriverVersion(EBackendType BackendType)
|
|
|
|
{
|
|
|
|
if(BackendType == BACKEND_TYPE_OPENGL)
|
|
|
|
{
|
|
|
|
//clamp the versions to existing versions(only for OpenGL major <= 3)
|
|
|
|
if(g_Config.m_GfxOpenGLMajor == 1)
|
|
|
|
{
|
|
|
|
g_Config.m_GfxOpenGLMinor = clamp(g_Config.m_GfxOpenGLMinor, 1, 5);
|
|
|
|
if(g_Config.m_GfxOpenGLMinor == 2)
|
|
|
|
g_Config.m_GfxOpenGLPatch = clamp(g_Config.m_GfxOpenGLPatch, 0, 1);
|
|
|
|
else
|
|
|
|
g_Config.m_GfxOpenGLPatch = 0;
|
|
|
|
}
|
|
|
|
else if(g_Config.m_GfxOpenGLMajor == 2)
|
|
|
|
{
|
|
|
|
g_Config.m_GfxOpenGLMinor = clamp(g_Config.m_GfxOpenGLMinor, 0, 1);
|
|
|
|
g_Config.m_GfxOpenGLPatch = 0;
|
|
|
|
}
|
|
|
|
else if(g_Config.m_GfxOpenGLMajor == 3)
|
|
|
|
{
|
|
|
|
g_Config.m_GfxOpenGLMinor = clamp(g_Config.m_GfxOpenGLMinor, 0, 3);
|
|
|
|
if(g_Config.m_GfxOpenGLMinor < 3)
|
|
|
|
g_Config.m_GfxOpenGLMinor = 0;
|
|
|
|
g_Config.m_GfxOpenGLPatch = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(BackendType == BACKEND_TYPE_OPENGL_ES)
|
|
|
|
{
|
2021-05-02 00:52:13 +00:00
|
|
|
#if !defined(CONF_BACKEND_OPENGL_ES3)
|
2021-04-30 22:42:37 +00:00
|
|
|
// Make sure GLES is set to 1.0 (which is equivalent to OpenGL 1.3), if its not set to >= 3.0(which is equivalent to OpenGL 3.3)
|
|
|
|
if(g_Config.m_GfxOpenGLMajor < 3)
|
|
|
|
{
|
|
|
|
g_Config.m_GfxOpenGLMajor = 1;
|
|
|
|
g_Config.m_GfxOpenGLMinor = 0;
|
|
|
|
g_Config.m_GfxOpenGLPatch = 0;
|
|
|
|
|
|
|
|
// GLES also doesnt know GL_QUAD
|
|
|
|
g_Config.m_GfxQuadAsTriangle = 1;
|
|
|
|
}
|
2021-05-01 00:39:06 +00:00
|
|
|
#else
|
|
|
|
g_Config.m_GfxOpenGLMajor = 3;
|
|
|
|
g_Config.m_GfxOpenGLMinor = 0;
|
|
|
|
g_Config.m_GfxOpenGLPatch = 0;
|
|
|
|
#endif
|
2021-04-30 22:42:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CGraphicsBackend_SDL_OpenGL::IsModernAPI(EBackendType BackendType)
|
|
|
|
{
|
|
|
|
if(BackendType == BACKEND_TYPE_OPENGL)
|
|
|
|
return (g_Config.m_GfxOpenGLMajor == 3 && g_Config.m_GfxOpenGLMinor == 3) || g_Config.m_GfxOpenGLMajor >= 4;
|
|
|
|
else if(BackendType == BACKEND_TYPE_OPENGL_ES)
|
|
|
|
return g_Config.m_GfxOpenGLMajor >= 3;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-02 21:19:57 +00:00
|
|
|
int CGraphicsBackend_SDL_OpenGL::Init(const char *pName, int *Screen, int *pWidth, int *pHeight, int FsaaSamples, int Flags, int *pDesktopWidth, int *pDesktopHeight, int *pCurrentWidth, int *pCurrentHeight, IStorage *pStorage)
|
2018-04-13 19:34:12 +00:00
|
|
|
{
|
2020-03-20 12:48:45 +00:00
|
|
|
// print sdl version
|
|
|
|
{
|
|
|
|
SDL_version Compiled;
|
|
|
|
SDL_version Linked;
|
|
|
|
|
|
|
|
SDL_VERSION(&Compiled);
|
|
|
|
SDL_GetVersion(&Linked);
|
2020-03-23 11:50:19 +00:00
|
|
|
dbg_msg("sdl", "SDL version %d.%d.%d (compiled = %d.%d.%d)", Linked.major, Linked.minor, Linked.patch,
|
|
|
|
Compiled.major, Compiled.minor, Compiled.patch);
|
2020-03-20 12:48:45 +00:00
|
|
|
}
|
2020-03-23 11:50:19 +00:00
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
if(!SDL_WasInit(SDL_INIT_VIDEO))
|
|
|
|
{
|
|
|
|
if(SDL_InitSubSystem(SDL_INIT_VIDEO) < 0)
|
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to init SDL video: %s", SDL_GetError());
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_SDL_INIT_FAILED;
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
EBackendType BackendType = DetectBackend();
|
2018-08-02 18:52:51 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
ClampDriverVersion(BackendType);
|
2020-08-29 10:10:38 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
m_UseNewOpenGL = IsModernAPI(BackendType);
|
2018-08-02 18:52:51 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, g_Config.m_GfxOpenGLMajor);
|
|
|
|
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, g_Config.m_GfxOpenGLMinor);
|
|
|
|
dbg_msg("gfx", "Created OpenGL %zu.%zu context.", (size_t)g_Config.m_GfxOpenGLMajor, (size_t)g_Config.m_GfxOpenGLMinor);
|
2018-08-02 18:52:51 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
if(BackendType == BACKEND_TYPE_OPENGL)
|
2017-09-13 18:33:58 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
if(g_Config.m_GfxOpenGLMajor == 3 && g_Config.m_GfxOpenGLMinor == 0)
|
2018-04-13 19:34:12 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_COMPATIBILITY);
|
2017-09-02 13:24:07 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(m_UseNewOpenGL)
|
2018-04-13 19:34:12 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_CORE);
|
2018-04-13 19:34:12 +00:00
|
|
|
}
|
2017-09-27 10:16:34 +00:00
|
|
|
}
|
2021-04-30 22:42:37 +00:00
|
|
|
else if(BackendType == BACKEND_TYPE_OPENGL_ES)
|
2020-08-29 10:10:38 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_ES);
|
2020-08-29 10:10:38 +00:00
|
|
|
}
|
2017-09-02 13:24:07 +00:00
|
|
|
|
2016-04-29 22:34:12 +00:00
|
|
|
// set screen
|
|
|
|
SDL_Rect ScreenPos;
|
|
|
|
m_NumScreens = SDL_GetNumVideoDisplays();
|
|
|
|
if(m_NumScreens > 0)
|
|
|
|
{
|
2016-05-07 21:19:46 +00:00
|
|
|
if(*Screen < 0 || *Screen >= m_NumScreens)
|
|
|
|
*Screen = 0;
|
2016-04-29 22:34:12 +00:00
|
|
|
if(SDL_GetDisplayBounds(*Screen, &ScreenPos) != 0)
|
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to retrieve screen information: %s", SDL_GetError());
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_SDL_SCREEN_INFO_REQUEST_FAILED;
|
2016-04-29 22:34:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2015-08-24 20:46:28 +00:00
|
|
|
{
|
2016-04-29 22:34:12 +00:00
|
|
|
dbg_msg("gfx", "unable to retrieve number of screens: %s", SDL_GetError());
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_SDL_SCREEN_REQUEST_FAILED;
|
2015-08-24 20:46:28 +00:00
|
|
|
}
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2016-04-29 22:34:12 +00:00
|
|
|
// store desktop resolution for settings reset button
|
|
|
|
SDL_DisplayMode DisplayMode;
|
|
|
|
if(SDL_GetDesktopDisplayMode(*Screen, &DisplayMode))
|
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to get desktop resolution: %s", SDL_GetError());
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_SDL_SCREEN_RESOLUTION_REQUEST_FAILED;
|
2016-04-29 22:34:12 +00:00
|
|
|
}
|
|
|
|
*pDesktopWidth = DisplayMode.w;
|
|
|
|
*pDesktopHeight = DisplayMode.h;
|
|
|
|
|
|
|
|
// use desktop resolution as default resolution
|
2017-07-11 12:08:03 +00:00
|
|
|
if(*pWidth == 0 || *pHeight == 0)
|
|
|
|
{
|
|
|
|
*pWidth = *pDesktopWidth;
|
|
|
|
*pHeight = *pDesktopHeight;
|
|
|
|
}
|
2012-03-04 11:46:55 +00:00
|
|
|
|
2015-08-24 20:46:28 +00:00
|
|
|
// set flags
|
2020-08-26 13:15:06 +00:00
|
|
|
int SdlFlags = SDL_WINDOW_OPENGL | SDL_WINDOW_INPUT_GRABBED | SDL_WINDOW_INPUT_FOCUS | SDL_WINDOW_MOUSE_FOCUS;
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Flags & IGraphicsBackend::INITFLAG_HIGHDPI)
|
2020-04-07 20:37:46 +00:00
|
|
|
SdlFlags |= SDL_WINDOW_ALLOW_HIGHDPI;
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Flags & IGraphicsBackend::INITFLAG_RESIZABLE)
|
2015-08-24 20:46:28 +00:00
|
|
|
SdlFlags |= SDL_WINDOW_RESIZABLE;
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Flags & IGraphicsBackend::INITFLAG_BORDERLESS)
|
2015-08-24 20:46:28 +00:00
|
|
|
SdlFlags |= SDL_WINDOW_BORDERLESS;
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Flags & IGraphicsBackend::INITFLAG_FULLSCREEN)
|
2016-04-30 21:10:09 +00:00
|
|
|
{
|
2018-12-28 16:29:41 +00:00
|
|
|
// when we are at fullscreen, we really shouldn't allow window sizes, that aren't supported by the driver
|
2017-10-23 16:02:18 +00:00
|
|
|
bool SupportedResolution = false;
|
|
|
|
SDL_DisplayMode mode;
|
2017-10-23 16:31:44 +00:00
|
|
|
int maxModes = SDL_GetNumDisplayModes(g_Config.m_GfxScreen);
|
2017-10-23 16:02:18 +00:00
|
|
|
|
2018-03-13 20:44:58 +00:00
|
|
|
for(int i = 0; i < maxModes; i++)
|
2017-10-23 16:02:18 +00:00
|
|
|
{
|
2018-03-13 20:44:58 +00:00
|
|
|
if(SDL_GetDisplayMode(g_Config.m_GfxScreen, i, &mode) < 0)
|
2017-10-23 16:02:18 +00:00
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to get display mode: %s", SDL_GetError());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-03-13 20:44:58 +00:00
|
|
|
if(*pWidth == mode.w && *pHeight == mode.h)
|
2017-10-23 16:02:18 +00:00
|
|
|
{
|
|
|
|
SupportedResolution = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(SupportedResolution)
|
|
|
|
SdlFlags |= SDL_WINDOW_FULLSCREEN;
|
|
|
|
else
|
|
|
|
SdlFlags |= SDL_WINDOW_FULLSCREEN_DESKTOP;
|
2016-04-30 21:10:09 +00:00
|
|
|
}
|
2021-04-02 20:45:49 +00:00
|
|
|
else if(Flags & (IGraphicsBackend::INITFLAG_DESKTOP_FULLSCREEN))
|
2021-01-31 20:54:04 +00:00
|
|
|
{
|
|
|
|
SdlFlags |= SDL_WINDOW_FULLSCREEN_DESKTOP;
|
|
|
|
}
|
2012-01-03 20:39:10 +00:00
|
|
|
|
|
|
|
// set gl attributes
|
2015-08-24 20:46:28 +00:00
|
|
|
SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1);
|
2012-01-03 20:39:10 +00:00
|
|
|
if(FsaaSamples)
|
|
|
|
{
|
|
|
|
SDL_GL_SetAttribute(SDL_GL_MULTISAMPLEBUFFERS, 1);
|
|
|
|
SDL_GL_SetAttribute(SDL_GL_MULTISAMPLESAMPLES, FsaaSamples);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SDL_GL_SetAttribute(SDL_GL_MULTISAMPLEBUFFERS, 0);
|
|
|
|
SDL_GL_SetAttribute(SDL_GL_MULTISAMPLESAMPLES, 0);
|
|
|
|
}
|
|
|
|
|
2016-05-07 13:59:13 +00:00
|
|
|
if(g_Config.m_InpMouseOld)
|
|
|
|
SDL_SetHint(SDL_HINT_MOUSE_RELATIVE_MODE_WARP, "1");
|
|
|
|
|
2015-08-24 20:46:28 +00:00
|
|
|
m_pWindow = SDL_CreateWindow(
|
|
|
|
pName,
|
2020-12-12 11:58:59 +00:00
|
|
|
SDL_WINDOWPOS_CENTERED,
|
|
|
|
SDL_WINDOWPOS_CENTERED,
|
2015-08-24 20:46:28 +00:00
|
|
|
*pWidth,
|
|
|
|
*pHeight,
|
|
|
|
SdlFlags);
|
2012-01-03 20:39:10 +00:00
|
|
|
|
|
|
|
// set caption
|
2015-08-24 20:46:28 +00:00
|
|
|
if(m_pWindow == NULL)
|
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to create window: %s", SDL_GetError());
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_SDL_WINDOW_CREATE_FAILED;
|
2015-08-24 20:46:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m_GLContext = SDL_GL_CreateContext(m_pWindow);
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2015-08-24 20:46:28 +00:00
|
|
|
if(m_GLContext == NULL)
|
2012-01-03 20:39:10 +00:00
|
|
|
{
|
2021-04-30 22:42:37 +00:00
|
|
|
SDL_DestroyWindow(m_pWindow);
|
2015-08-24 20:46:28 +00:00
|
|
|
dbg_msg("gfx", "unable to create OpenGL context: %s", SDL_GetError());
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_OPENGL_CONTEXT_FAILED;
|
2015-07-09 00:08:14 +00:00
|
|
|
}
|
2017-10-20 07:08:49 +00:00
|
|
|
|
2020-08-29 10:10:38 +00:00
|
|
|
int GlewMajor = 0;
|
|
|
|
int GlewMinor = 0;
|
|
|
|
int GlewPatch = 0;
|
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
if(!BackendInitGlew(BackendType, GlewMajor, GlewMinor, GlewPatch))
|
|
|
|
{
|
|
|
|
SDL_GL_DeleteContext(m_GLContext);
|
|
|
|
SDL_DestroyWindow(m_pWindow);
|
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_UNKNOWN;
|
|
|
|
}
|
2020-08-29 10:10:38 +00:00
|
|
|
|
|
|
|
int InitError = 0;
|
2020-11-05 19:38:37 +00:00
|
|
|
const char *pErrorStr = NULL;
|
2020-08-29 10:10:38 +00:00
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
InitError = IsVersionSupportedGlew(BackendType, g_Config.m_GfxOpenGLMajor, g_Config.m_GfxOpenGLMinor, g_Config.m_GfxOpenGLPatch, GlewMajor, GlewMinor, GlewPatch);
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2018-07-17 07:46:17 +00:00
|
|
|
SDL_GL_GetDrawableSize(m_pWindow, pCurrentWidth, pCurrentHeight);
|
2020-09-26 19:41:58 +00:00
|
|
|
SDL_GL_SetSwapInterval(Flags & IGraphicsBackend::INITFLAG_VSYNC ? 1 : 0);
|
2015-08-24 20:46:28 +00:00
|
|
|
SDL_GL_MakeCurrent(NULL, NULL);
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2020-08-29 10:10:38 +00:00
|
|
|
if(InitError != 0)
|
|
|
|
{
|
|
|
|
SDL_GL_DeleteContext(m_GLContext);
|
|
|
|
SDL_DestroyWindow(m_pWindow);
|
|
|
|
|
|
|
|
// try setting to glew supported version
|
|
|
|
g_Config.m_GfxOpenGLMajor = GlewMajor;
|
|
|
|
g_Config.m_GfxOpenGLMinor = GlewMinor;
|
|
|
|
g_Config.m_GfxOpenGLPatch = GlewPatch;
|
2020-09-26 19:41:58 +00:00
|
|
|
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_OPENGL_VERSION_FAILED;
|
|
|
|
}
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
// start the command processor
|
2021-04-30 22:42:37 +00:00
|
|
|
m_pProcessor = new CCommandProcessor_SDL_OpenGL(BackendType, g_Config.m_GfxOpenGLMajor, g_Config.m_GfxOpenGLMinor, g_Config.m_GfxOpenGLPatch);
|
2012-01-03 20:39:10 +00:00
|
|
|
StartProcessor(m_pProcessor);
|
2020-09-26 19:41:58 +00:00
|
|
|
|
2020-11-05 19:38:37 +00:00
|
|
|
mem_zero(m_aErrorString, sizeof(m_aErrorString) / sizeof(m_aErrorString[0]));
|
|
|
|
|
2012-10-06 21:31:02 +00:00
|
|
|
// issue init commands for OpenGL and SDL
|
2012-01-03 20:39:10 +00:00
|
|
|
CCommandBuffer CmdBuffer(1024, 512);
|
2020-08-29 10:10:38 +00:00
|
|
|
//run sdl first to have the context in the thread
|
|
|
|
CCommandProcessorFragment_SDL::SCommand_Init CmdSDL;
|
|
|
|
CmdSDL.m_pWindow = m_pWindow;
|
|
|
|
CmdSDL.m_GLContext = m_GLContext;
|
2021-04-07 17:03:07 +00:00
|
|
|
CmdBuffer.AddCommandUnsafe(CmdSDL);
|
2020-08-29 10:10:38 +00:00
|
|
|
RunBuffer(&CmdBuffer);
|
|
|
|
WaitForIdle();
|
|
|
|
CmdBuffer.Reset();
|
|
|
|
|
2020-08-29 15:44:23 +00:00
|
|
|
if(InitError == 0)
|
|
|
|
{
|
2021-05-02 00:52:13 +00:00
|
|
|
CCommandProcessorFragment_OpenGLBase::SCommand_Init CmdOpenGL;
|
2020-08-29 15:44:23 +00:00
|
|
|
CmdOpenGL.m_pTextureMemoryUsage = &m_TextureMemoryUsage;
|
|
|
|
CmdOpenGL.m_pStorage = pStorage;
|
|
|
|
CmdOpenGL.m_pCapabilities = &m_Capabilites;
|
|
|
|
CmdOpenGL.m_pInitError = &InitError;
|
2021-04-30 22:42:37 +00:00
|
|
|
CmdOpenGL.m_pCapabilities = &m_Capabilites;
|
|
|
|
CmdOpenGL.m_RequestedMajor = g_Config.m_GfxOpenGLMajor;
|
|
|
|
CmdOpenGL.m_RequestedMinor = g_Config.m_GfxOpenGLMinor;
|
|
|
|
CmdOpenGL.m_RequestedPatch = g_Config.m_GfxOpenGLPatch;
|
|
|
|
CmdOpenGL.m_GlewMajor = GlewMajor;
|
|
|
|
CmdOpenGL.m_GlewMinor = GlewMinor;
|
|
|
|
CmdOpenGL.m_GlewPatch = GlewPatch;
|
|
|
|
CmdOpenGL.m_pInitError = &InitError;
|
|
|
|
CmdOpenGL.m_pErrStringPtr = &pErrorStr;
|
|
|
|
CmdOpenGL.m_pVendorString = m_aVendorString;
|
|
|
|
CmdOpenGL.m_pVersionString = m_aVersionString;
|
|
|
|
CmdOpenGL.m_pRendererString = m_aRendererString;
|
|
|
|
CmdOpenGL.m_RequestedBackend = BackendType;
|
2021-04-07 17:03:07 +00:00
|
|
|
CmdBuffer.AddCommandUnsafe(CmdOpenGL);
|
2021-04-30 22:42:37 +00:00
|
|
|
|
2020-08-29 15:44:23 +00:00
|
|
|
RunBuffer(&CmdBuffer);
|
|
|
|
WaitForIdle();
|
|
|
|
CmdBuffer.Reset();
|
|
|
|
|
|
|
|
if(InitError == -2)
|
|
|
|
{
|
2021-05-02 00:52:13 +00:00
|
|
|
CCommandProcessorFragment_OpenGLBase::SCommand_Shutdown CmdGL;
|
2021-04-07 17:03:07 +00:00
|
|
|
CmdBuffer.AddCommandUnsafe(CmdGL);
|
2020-09-03 09:43:32 +00:00
|
|
|
RunBuffer(&CmdBuffer);
|
|
|
|
WaitForIdle();
|
|
|
|
CmdBuffer.Reset();
|
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
g_Config.m_GfxOpenGLMajor = m_Capabilites.m_ContextMajor;
|
|
|
|
g_Config.m_GfxOpenGLMinor = m_Capabilites.m_ContextMinor;
|
|
|
|
g_Config.m_GfxOpenGLPatch = m_Capabilites.m_ContextPatch;
|
2020-08-29 15:44:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-29 10:10:38 +00:00
|
|
|
if(InitError != 0)
|
2020-08-26 18:01:32 +00:00
|
|
|
{
|
2020-08-29 10:10:38 +00:00
|
|
|
CCommandProcessorFragment_SDL::SCommand_Shutdown Cmd;
|
2021-04-07 17:03:07 +00:00
|
|
|
CmdBuffer.AddCommandUnsafe(Cmd);
|
2017-09-02 13:24:07 +00:00
|
|
|
RunBuffer(&CmdBuffer);
|
|
|
|
WaitForIdle();
|
2020-08-29 10:10:38 +00:00
|
|
|
CmdBuffer.Reset();
|
|
|
|
|
|
|
|
// stop and delete the processor
|
|
|
|
StopProcessor();
|
|
|
|
delete m_pProcessor;
|
|
|
|
m_pProcessor = 0;
|
|
|
|
|
|
|
|
SDL_GL_DeleteContext(m_GLContext);
|
|
|
|
SDL_DestroyWindow(m_pWindow);
|
|
|
|
|
|
|
|
// try setting to version string's supported version
|
|
|
|
if(InitError == -2)
|
|
|
|
{
|
|
|
|
g_Config.m_GfxOpenGLMajor = m_Capabilites.m_ContextMajor;
|
|
|
|
g_Config.m_GfxOpenGLMinor = m_Capabilites.m_ContextMinor;
|
|
|
|
g_Config.m_GfxOpenGLPatch = m_Capabilites.m_ContextPatch;
|
|
|
|
}
|
|
|
|
|
2020-11-05 19:38:37 +00:00
|
|
|
if(pErrorStr != NULL)
|
|
|
|
{
|
|
|
|
str_copy(m_aErrorString, pErrorStr, sizeof(m_aErrorString) / sizeof(m_aErrorString[0]));
|
|
|
|
}
|
|
|
|
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_OPENGL_VERSION_FAILED;
|
2017-09-02 13:24:07 +00:00
|
|
|
}
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2018-03-13 20:44:58 +00:00
|
|
|
if(SetWindowScreen(g_Config.m_GfxScreen))
|
2017-10-23 16:02:18 +00:00
|
|
|
{
|
|
|
|
// query the current displaymode, when running in fullscreen
|
|
|
|
// this is required if DPI scaling is active
|
2020-09-26 19:41:58 +00:00
|
|
|
if(SdlFlags & SDL_WINDOW_FULLSCREEN)
|
2017-10-23 16:02:18 +00:00
|
|
|
{
|
|
|
|
SDL_DisplayMode CurrentDisplayMode;
|
|
|
|
SDL_GetCurrentDisplayMode(g_Config.m_GfxScreen, &CurrentDisplayMode);
|
|
|
|
|
|
|
|
*pCurrentWidth = CurrentDisplayMode.w;
|
|
|
|
*pCurrentHeight = CurrentDisplayMode.h;
|
|
|
|
|
|
|
|
// since the window is centered, calculate how much the viewport has to be fixed
|
|
|
|
//int XOverflow = (*pWidth > *pCurrentWidth ? (*pWidth - *pCurrentWidth) : 0);
|
|
|
|
//int YOverflow = (*pHeight > *pCurrentHeight ? (*pHeight - *pCurrentHeight) : 0);
|
|
|
|
//TODO: current problem is, that the opengl driver knows about the scaled display,
|
|
|
|
//so the viewport cannot be adjusted for resolutions, that are higher than allowed by the display driver
|
2018-04-13 19:34:12 +00:00
|
|
|
|
2021-05-01 21:33:42 +00:00
|
|
|
CCommandBuffer::SCommand_Update_Viewport CmdSDL;
|
2017-10-23 16:02:18 +00:00
|
|
|
CmdSDL.m_X = 0;
|
|
|
|
CmdSDL.m_Y = 0;
|
|
|
|
|
|
|
|
CmdSDL.m_Width = CurrentDisplayMode.w;
|
|
|
|
CmdSDL.m_Height = CurrentDisplayMode.h;
|
2021-04-07 17:03:07 +00:00
|
|
|
CmdBuffer.AddCommandUnsafe(CmdSDL);
|
2017-10-23 16:02:18 +00:00
|
|
|
RunBuffer(&CmdBuffer);
|
2018-04-13 19:34:12 +00:00
|
|
|
WaitForIdle();
|
2020-08-29 10:10:38 +00:00
|
|
|
CmdBuffer.Reset();
|
2017-10-23 16:02:18 +00:00
|
|
|
}
|
|
|
|
}
|
2016-05-07 19:16:31 +00:00
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
// return
|
2020-08-29 10:10:38 +00:00
|
|
|
return EGraphicsBackendErrorCodes::GRAPHICS_BACKEND_ERROR_CODE_NONE;
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CGraphicsBackend_SDL_OpenGL::Shutdown()
|
|
|
|
{
|
|
|
|
// issue a shutdown command
|
|
|
|
CCommandBuffer CmdBuffer(1024, 512);
|
2021-05-02 00:52:13 +00:00
|
|
|
CCommandProcessorFragment_OpenGLBase::SCommand_Shutdown CmdGL;
|
2021-04-07 17:03:07 +00:00
|
|
|
CmdBuffer.AddCommandUnsafe(CmdGL);
|
2020-09-03 09:43:32 +00:00
|
|
|
RunBuffer(&CmdBuffer);
|
|
|
|
WaitForIdle();
|
|
|
|
CmdBuffer.Reset();
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
CCommandProcessorFragment_SDL::SCommand_Shutdown Cmd;
|
2021-04-07 17:03:07 +00:00
|
|
|
CmdBuffer.AddCommandUnsafe(Cmd);
|
2012-01-03 20:39:10 +00:00
|
|
|
RunBuffer(&CmdBuffer);
|
|
|
|
WaitForIdle();
|
2020-08-29 10:10:38 +00:00
|
|
|
CmdBuffer.Reset();
|
2015-07-09 00:08:14 +00:00
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
// stop and delete the processor
|
|
|
|
StopProcessor();
|
|
|
|
delete m_pProcessor;
|
|
|
|
m_pProcessor = 0;
|
|
|
|
|
2015-08-24 20:46:28 +00:00
|
|
|
SDL_GL_DeleteContext(m_GLContext);
|
|
|
|
SDL_DestroyWindow(m_pWindow);
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
SDL_QuitSubSystem(SDL_INIT_VIDEO);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-10-06 21:31:02 +00:00
|
|
|
int CGraphicsBackend_SDL_OpenGL::MemoryUsage() const
|
|
|
|
{
|
|
|
|
return m_TextureMemoryUsage;
|
|
|
|
}
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
void CGraphicsBackend_SDL_OpenGL::Minimize()
|
|
|
|
{
|
2015-08-24 20:46:28 +00:00
|
|
|
SDL_MinimizeWindow(m_pWindow);
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CGraphicsBackend_SDL_OpenGL::Maximize()
|
|
|
|
{
|
|
|
|
// TODO: SDL
|
|
|
|
}
|
|
|
|
|
2021-01-31 20:54:04 +00:00
|
|
|
void CGraphicsBackend_SDL_OpenGL::SetWindowParams(int FullscreenMode, bool IsBorderless)
|
2016-04-29 19:07:10 +00:00
|
|
|
{
|
2021-01-31 20:54:04 +00:00
|
|
|
if(FullscreenMode > 0)
|
|
|
|
{
|
|
|
|
if(FullscreenMode == 1)
|
|
|
|
{
|
2021-04-17 15:49:41 +00:00
|
|
|
#if defined(CONF_PLATFORM_MACOS) || defined(CONF_PLATFORM_HAIKU) // Todo SDL: remove this when fixed (game freezes when losing focus in fullscreen)
|
2021-01-31 20:54:04 +00:00
|
|
|
SDL_SetWindowFullscreen(m_pWindow, SDL_WINDOW_FULLSCREEN_DESKTOP);
|
2016-04-29 22:34:12 +00:00
|
|
|
#else
|
2021-01-31 20:54:04 +00:00
|
|
|
SDL_SetWindowFullscreen(m_pWindow, SDL_WINDOW_FULLSCREEN);
|
2016-04-29 22:34:12 +00:00
|
|
|
#endif
|
2021-01-31 20:54:04 +00:00
|
|
|
}
|
|
|
|
else if(FullscreenMode == 2)
|
|
|
|
{
|
|
|
|
SDL_SetWindowFullscreen(m_pWindow, SDL_WINDOW_FULLSCREEN_DESKTOP);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SDL_SetWindowFullscreen(m_pWindow, 0);
|
|
|
|
SDL_SetWindowBordered(m_pWindow, SDL_bool(!IsBorderless));
|
|
|
|
}
|
2016-04-29 22:34:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CGraphicsBackend_SDL_OpenGL::SetWindowScreen(int Index)
|
|
|
|
{
|
|
|
|
if(Index >= 0 && Index < m_NumScreens)
|
|
|
|
{
|
|
|
|
SDL_Rect ScreenPos;
|
|
|
|
if(SDL_GetDisplayBounds(Index, &ScreenPos) == 0)
|
|
|
|
{
|
|
|
|
SDL_SetWindowPosition(m_pWindow,
|
2016-05-07 14:35:31 +00:00
|
|
|
SDL_WINDOWPOS_CENTERED_DISPLAY(Index),
|
|
|
|
SDL_WINDOWPOS_CENTERED_DISPLAY(Index));
|
2016-04-29 22:34:12 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CGraphicsBackend_SDL_OpenGL::GetWindowScreen()
|
|
|
|
{
|
|
|
|
return SDL_GetWindowDisplayIndex(m_pWindow);
|
2016-04-29 19:07:10 +00:00
|
|
|
}
|
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
int CGraphicsBackend_SDL_OpenGL::WindowActive()
|
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
return SDL_GetWindowFlags(m_pWindow) & SDL_WINDOW_INPUT_FOCUS;
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CGraphicsBackend_SDL_OpenGL::WindowOpen()
|
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
return SDL_GetWindowFlags(m_pWindow) & SDL_WINDOW_SHOWN;
|
2014-10-18 14:17:36 +00:00
|
|
|
}
|
2012-01-03 20:39:10 +00:00
|
|
|
|
2015-08-24 23:01:38 +00:00
|
|
|
void CGraphicsBackend_SDL_OpenGL::SetWindowGrab(bool Grab)
|
|
|
|
{
|
|
|
|
SDL_SetWindowGrab(m_pWindow, Grab ? SDL_TRUE : SDL_FALSE);
|
|
|
|
}
|
|
|
|
|
2020-12-12 11:58:59 +00:00
|
|
|
void CGraphicsBackend_SDL_OpenGL::ResizeWindow(int w, int h)
|
|
|
|
{
|
|
|
|
SDL_SetWindowSize(m_pWindow, w, h);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGraphicsBackend_SDL_OpenGL::GetViewportSize(int &w, int &h)
|
|
|
|
{
|
|
|
|
SDL_GL_GetDrawableSize(m_pWindow, &w, &h);
|
|
|
|
}
|
|
|
|
|
2014-10-18 14:17:36 +00:00
|
|
|
void CGraphicsBackend_SDL_OpenGL::NotifyWindow()
|
|
|
|
{
|
2015-08-25 00:39:48 +00:00
|
|
|
// get window handle
|
|
|
|
SDL_SysWMinfo info;
|
|
|
|
SDL_VERSION(&info.version);
|
|
|
|
if(!SDL_GetWindowWMInfo(m_pWindow, &info))
|
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to obtain window handle");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
#if defined(CONF_FAMILY_WINDOWS)
|
|
|
|
FLASHWINFO desc;
|
|
|
|
desc.cbSize = sizeof(desc);
|
|
|
|
desc.hwnd = info.info.win.window;
|
|
|
|
desc.dwFlags = FLASHW_TRAY;
|
|
|
|
desc.uCount = 3; // flash 3 times
|
|
|
|
desc.dwTimeout = 0;
|
|
|
|
|
|
|
|
FlashWindowEx(&desc);
|
2021-02-12 12:40:29 +00:00
|
|
|
#elif defined(SDL_VIDEO_DRIVER_X11) && !defined(CONF_PLATFORM_MACOS)
|
2020-09-26 19:41:58 +00:00
|
|
|
Display *dpy = info.info.x11.display;
|
|
|
|
Window win = info.info.x11.window;
|
|
|
|
|
|
|
|
// Old hints
|
|
|
|
XWMHints *wmhints;
|
|
|
|
wmhints = XAllocWMHints();
|
|
|
|
wmhints->flags = XUrgencyHint;
|
|
|
|
XSetWMHints(dpy, win, wmhints);
|
|
|
|
XFree(wmhints);
|
|
|
|
|
|
|
|
// More modern way of notifying
|
|
|
|
static Atom demandsAttention = XInternAtom(dpy, "_NET_WM_STATE_DEMANDS_ATTENTION", true);
|
|
|
|
static Atom wmState = XInternAtom(dpy, "_NET_WM_STATE", true);
|
|
|
|
XChangeProperty(dpy, win, wmState, XA_ATOM, 32, PropModeReplace,
|
|
|
|
(unsigned char *)&demandsAttention, 1);
|
|
|
|
#endif
|
2012-01-03 20:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
IGraphicsBackend *CreateGraphicsBackend() { return new CGraphicsBackend_SDL_OpenGL; }
|