| 1 | /* |
| 2 | Simple DirectMedia Layer |
| 3 | Copyright (C) 1997-2025 Sam Lantinga <slouken@libsdl.org> |
| 4 | |
| 5 | This software is provided 'as-is', without any express or implied |
| 6 | warranty. In no event will the authors be held liable for any damages |
| 7 | arising from the use of this software. |
| 8 | |
| 9 | Permission is granted to anyone to use this software for any purpose, |
| 10 | including commercial applications, and to alter it and redistribute it |
| 11 | freely, subject to the following restrictions: |
| 12 | |
| 13 | 1. The origin of this software must not be misrepresented; you must not |
| 14 | claim that you wrote the original software. If you use this software |
| 15 | in a product, an acknowledgment in the product documentation would be |
| 16 | appreciated but is not required. |
| 17 | 2. Altered source versions must be plainly marked as such, and must not be |
| 18 | misrepresented as being the original software. |
| 19 | 3. This notice may not be removed or altered from any source distribution. |
| 20 | */ |
| 21 | #include "SDL_internal.h" |
| 22 | |
| 23 | #if defined(SDL_VIDEO_DRIVER_X11) && defined(SDL_VIDEO_OPENGL_EGL) |
| 24 | |
| 25 | #include "SDL_x11video.h" |
| 26 | #include "SDL_x11opengles.h" |
| 27 | #include "SDL_x11opengl.h" |
| 28 | #include "SDL_x11xsync.h" |
| 29 | |
| 30 | // EGL implementation of SDL OpenGL support |
| 31 | |
| 32 | bool X11_GLES_LoadLibrary(SDL_VideoDevice *_this, const char *path) |
| 33 | { |
| 34 | SDL_VideoData *data = _this->internal; |
| 35 | |
| 36 | // If the profile requested is not GL ES, switch over to X11_GL functions |
| 37 | if ((_this->gl_config.profile_mask != SDL_GL_CONTEXT_PROFILE_ES) && |
| 38 | !SDL_GetHintBoolean(SDL_HINT_VIDEO_FORCE_EGL, false)) { |
| 39 | #ifdef SDL_VIDEO_OPENGL_GLX |
| 40 | X11_GLES_UnloadLibrary(_this); |
| 41 | _this->GL_LoadLibrary = X11_GL_LoadLibrary; |
| 42 | _this->GL_GetProcAddress = X11_GL_GetProcAddress; |
| 43 | _this->GL_UnloadLibrary = X11_GL_UnloadLibrary; |
| 44 | _this->GL_CreateContext = X11_GL_CreateContext; |
| 45 | _this->GL_MakeCurrent = X11_GL_MakeCurrent; |
| 46 | _this->GL_SetSwapInterval = X11_GL_SetSwapInterval; |
| 47 | _this->GL_GetSwapInterval = X11_GL_GetSwapInterval; |
| 48 | _this->GL_SwapWindow = X11_GL_SwapWindow; |
| 49 | _this->GL_DestroyContext = X11_GL_DestroyContext; |
| 50 | return X11_GL_LoadLibrary(_this, path); |
| 51 | #else |
| 52 | return SDL_SetError("SDL not configured with OpenGL/GLX support" ); |
| 53 | #endif |
| 54 | } |
| 55 | |
| 56 | return SDL_EGL_LoadLibrary(_this, path, (NativeDisplayType)data->display, _this->gl_config.egl_platform); |
| 57 | } |
| 58 | |
| 59 | XVisualInfo *X11_GLES_GetVisual(SDL_VideoDevice *_this, Display *display, int screen, bool transparent) |
| 60 | { |
| 61 | |
| 62 | XVisualInfo *egl_visualinfo = NULL; |
| 63 | EGLint visual_id = 0; |
| 64 | XVisualInfo vi_in; |
| 65 | int out_count = 0; |
| 66 | |
| 67 | if (!_this->egl_data) { |
| 68 | // The EGL library wasn't loaded, SDL_GetError() should have info |
| 69 | return NULL; |
| 70 | } |
| 71 | |
| 72 | if (_this->egl_data->eglGetConfigAttrib(_this->egl_data->egl_display, |
| 73 | _this->egl_data->egl_config, |
| 74 | EGL_NATIVE_VISUAL_ID, |
| 75 | &visual_id) == EGL_FALSE) { |
| 76 | visual_id = 0; |
| 77 | } |
| 78 | if (visual_id != 0) { |
| 79 | vi_in.screen = screen; |
| 80 | vi_in.visualid = visual_id; |
| 81 | egl_visualinfo = X11_XGetVisualInfo(display, VisualScreenMask | VisualIDMask, &vi_in, &out_count); |
| 82 | if (transparent && egl_visualinfo) { |
| 83 | Uint32 format = X11_GetPixelFormatFromVisualInfo(display, egl_visualinfo); |
| 84 | if (!SDL_ISPIXELFORMAT_ALPHA(format)) { |
| 85 | // not transparent! |
| 86 | X11_XFree(egl_visualinfo); |
| 87 | egl_visualinfo = NULL; |
| 88 | } |
| 89 | } |
| 90 | } |
| 91 | |
| 92 | if(!egl_visualinfo) { |
| 93 | // Use the default visual when all else fails |
| 94 | vi_in.screen = screen; |
| 95 | egl_visualinfo = X11_XGetVisualInfo(display, |
| 96 | VisualScreenMask, |
| 97 | &vi_in, &out_count); |
| 98 | |
| 99 | // Return the first transparent Visual |
| 100 | if (transparent) { |
| 101 | int i; |
| 102 | for (i = 0; i < out_count; i++) { |
| 103 | XVisualInfo *v = &egl_visualinfo[i]; |
| 104 | Uint32 format = X11_GetPixelFormatFromVisualInfo(display, v); |
| 105 | if (SDL_ISPIXELFORMAT_ALPHA(format)) { // found! |
| 106 | // re-request it to have a copy that can be X11_XFree'ed later |
| 107 | vi_in.screen = screen; |
| 108 | vi_in.visualid = v->visualid; |
| 109 | X11_XFree(egl_visualinfo); |
| 110 | egl_visualinfo = X11_XGetVisualInfo(display, VisualScreenMask | VisualIDMask, &vi_in, &out_count); |
| 111 | return egl_visualinfo; |
| 112 | } |
| 113 | } |
| 114 | } |
| 115 | } |
| 116 | return egl_visualinfo; |
| 117 | } |
| 118 | |
| 119 | SDL_GLContext X11_GLES_CreateContext(SDL_VideoDevice *_this, SDL_Window *window) |
| 120 | { |
| 121 | SDL_GLContext context; |
| 122 | SDL_WindowData *data = window->internal; |
| 123 | Display *display = data->videodata->display; |
| 124 | |
| 125 | X11_XSync(display, False); |
| 126 | context = SDL_EGL_CreateContext(_this, data->egl_surface); |
| 127 | X11_XSync(display, False); |
| 128 | |
| 129 | return context; |
| 130 | } |
| 131 | |
| 132 | SDL_EGLSurface X11_GLES_GetEGLSurface(SDL_VideoDevice *_this, SDL_Window *window) |
| 133 | { |
| 134 | SDL_WindowData *data = window->internal; |
| 135 | return data->egl_surface; |
| 136 | } |
| 137 | |
| 138 | bool X11_GLES_SwapWindow(SDL_VideoDevice *_this, SDL_Window *window) |
| 139 | { |
| 140 | const bool ret = SDL_EGL_SwapBuffers(_this, window->internal->egl_surface); \ |
| 141 | |
| 142 | #ifdef SDL_VIDEO_DRIVER_X11_XSYNC |
| 143 | X11_HandlePresent(window); |
| 144 | #endif /* SDL_VIDEO_DRIVER_X11_XSYNC */ |
| 145 | |
| 146 | return ret; |
| 147 | } |
| 148 | |
| 149 | SDL_EGL_MakeCurrent_impl(X11) |
| 150 | |
| 151 | #endif // SDL_VIDEO_DRIVER_X11 && SDL_VIDEO_OPENGL_EGL |
| 152 | |
| 153 | |