slouken@1952
|
1 |
/*
|
slouken@1952
|
2 |
SDL - Simple DirectMedia Layer
|
slouken@3697
|
3 |
Copyright (C) 1997-2010 Sam Lantinga
|
slouken@1952
|
4 |
|
slouken@1952
|
5 |
This library is free software; you can redistribute it and/or
|
slouken@1952
|
6 |
modify it under the terms of the GNU Lesser General Public
|
slouken@1952
|
7 |
License as published by the Free Software Foundation; either
|
slouken@1952
|
8 |
version 2.1 of the License, or (at your option) any later version.
|
slouken@1952
|
9 |
|
slouken@1952
|
10 |
This library is distributed in the hope that it will be useful,
|
slouken@1952
|
11 |
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
slouken@1952
|
12 |
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
slouken@1952
|
13 |
Lesser General Public License for more details.
|
slouken@1952
|
14 |
|
slouken@1952
|
15 |
You should have received a copy of the GNU Lesser General Public
|
slouken@3361
|
16 |
License along with this library; if not, write to the Free Software
|
slouken@1952
|
17 |
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
slouken@1952
|
18 |
|
slouken@1952
|
19 |
Sam Lantinga
|
slouken@1952
|
20 |
slouken@libsdl.org
|
slouken@1952
|
21 |
*/
|
slouken@1952
|
22 |
#include "SDL_config.h"
|
slouken@1952
|
23 |
|
slouken@1952
|
24 |
#include "SDL_x11video.h"
|
slouken@1952
|
25 |
|
slouken@1952
|
26 |
/* GLX implementation of SDL OpenGL support */
|
slouken@1952
|
27 |
|
slouken@1952
|
28 |
#if SDL_VIDEO_OPENGL_GLX
|
slouken@1952
|
29 |
#include "SDL_loadso.h"
|
slouken@1952
|
30 |
|
slouken@1952
|
31 |
#if defined(__IRIX__)
|
slouken@1952
|
32 |
/* IRIX doesn't have a GL library versioning system */
|
slouken@1952
|
33 |
#define DEFAULT_OPENGL "libGL.so"
|
slouken@1952
|
34 |
#elif defined(__MACOSX__)
|
slouken@1952
|
35 |
#define DEFAULT_OPENGL "/usr/X11R6/lib/libGL.1.dylib"
|
slouken@1952
|
36 |
#elif defined(__QNXNTO__)
|
slouken@1952
|
37 |
#define DEFAULT_OPENGL "libGL.so.3"
|
slouken@1952
|
38 |
#else
|
slouken@1952
|
39 |
#define DEFAULT_OPENGL "libGL.so.1"
|
slouken@1952
|
40 |
#endif
|
slouken@1952
|
41 |
|
slouken@3241
|
42 |
#ifndef GLX_NONE_EXT
|
slouken@3241
|
43 |
#define GLX_NONE_EXT 0x8000
|
slouken@3241
|
44 |
#endif
|
slouken@3241
|
45 |
|
slouken@1952
|
46 |
#ifndef GLX_ARB_multisample
|
slouken@1952
|
47 |
#define GLX_ARB_multisample
|
slouken@1952
|
48 |
#define GLX_SAMPLE_BUFFERS_ARB 100000
|
slouken@1952
|
49 |
#define GLX_SAMPLES_ARB 100001
|
slouken@1952
|
50 |
#endif
|
slouken@1952
|
51 |
|
slouken@1952
|
52 |
#ifndef GLX_EXT_visual_rating
|
slouken@1952
|
53 |
#define GLX_EXT_visual_rating
|
slouken@1952
|
54 |
#define GLX_VISUAL_CAVEAT_EXT 0x20
|
slouken@1952
|
55 |
#define GLX_NONE_EXT 0x8000
|
slouken@1952
|
56 |
#define GLX_SLOW_VISUAL_EXT 0x8001
|
slouken@1952
|
57 |
#define GLX_NON_CONFORMANT_VISUAL_EXT 0x800D
|
slouken@1952
|
58 |
#endif
|
slouken@1952
|
59 |
|
slouken@3100
|
60 |
#ifndef GLX_ARB_create_context
|
slouken@3100
|
61 |
#define GLX_ARB_create_context
|
slouken@3100
|
62 |
#define GLX_CONTEXT_MAJOR_VERSION_ARB 0x2091
|
slouken@3100
|
63 |
#define GLX_CONTEXT_MINOR_VERSION_ARB 0x2092
|
slouken@3100
|
64 |
#define GLX_CONTEXT_FLAGS_ARB 0x2094
|
slouken@3100
|
65 |
#define GLX_CONTEXT_DEBUG_BIT_ARB 0x0001
|
slouken@3100
|
66 |
#define GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB 0x0002
|
slouken@5021
|
67 |
|
slouken@5021
|
68 |
/* Typedef for the GL 3.0 context creation function */
|
slouken@5021
|
69 |
typedef GLXContext(*PFNGLXCREATECONTEXTATTRIBSARBPROC) (Display * dpy,
|
slouken@5021
|
70 |
GLXFBConfig config,
|
slouken@5021
|
71 |
GLXContext
|
slouken@5021
|
72 |
share_context,
|
slouken@5021
|
73 |
Bool direct,
|
slouken@5021
|
74 |
const int
|
slouken@5021
|
75 |
*attrib_list);
|
slouken@3100
|
76 |
#endif
|
slouken@3100
|
77 |
|
slouken@1952
|
78 |
#define OPENGL_REQUIRS_DLOPEN
|
slouken@1952
|
79 |
#if defined(OPENGL_REQUIRS_DLOPEN) && defined(SDL_LOADSO_DLOPEN)
|
slouken@1952
|
80 |
#include <dlfcn.h>
|
slouken@1952
|
81 |
#define GL_LoadObject(X) dlopen(X, (RTLD_NOW|RTLD_GLOBAL))
|
slouken@1952
|
82 |
#define GL_LoadFunction dlsym
|
slouken@1952
|
83 |
#define GL_UnloadObject dlclose
|
slouken@1952
|
84 |
#else
|
slouken@1952
|
85 |
#define GL_LoadObject SDL_LoadObject
|
slouken@1952
|
86 |
#define GL_LoadFunction SDL_LoadFunction
|
slouken@1952
|
87 |
#define GL_UnloadObject SDL_UnloadObject
|
slouken@1952
|
88 |
#endif
|
slouken@1952
|
89 |
|
slouken@3057
|
90 |
static void X11_GL_InitExtensions(_THIS);
|
bob@2322
|
91 |
|
slouken@3100
|
92 |
|
slouken@1952
|
93 |
int
|
slouken@1952
|
94 |
X11_GL_LoadLibrary(_THIS, const char *path)
|
slouken@1952
|
95 |
{
|
slouken@1952
|
96 |
void *handle;
|
slouken@1952
|
97 |
|
slouken@3057
|
98 |
/* Load the OpenGL library */
|
slouken@3057
|
99 |
if (path == NULL) {
|
slouken@3057
|
100 |
path = SDL_getenv("SDL_OPENGL_LIBRARY");
|
slouken@1952
|
101 |
}
|
slouken@3057
|
102 |
if (path == NULL) {
|
slouken@3057
|
103 |
path = DEFAULT_OPENGL;
|
slouken@3057
|
104 |
}
|
slouken@3057
|
105 |
_this->gl_config.dll_handle = SDL_LoadObject(path);
|
slouken@3057
|
106 |
if (!_this->gl_config.dll_handle) {
|
slouken@3057
|
107 |
return -1;
|
slouken@3057
|
108 |
}
|
slouken@3057
|
109 |
SDL_strlcpy(_this->gl_config.driver_path, path,
|
slouken@3057
|
110 |
SDL_arraysize(_this->gl_config.driver_path));
|
slouken@2244
|
111 |
|
slouken@3057
|
112 |
/* Allocate OpenGL memory */
|
slouken@3057
|
113 |
_this->gl_data =
|
slouken@3057
|
114 |
(struct SDL_GLDriverData *) SDL_calloc(1,
|
slouken@3057
|
115 |
sizeof(struct
|
slouken@3057
|
116 |
SDL_GLDriverData));
|
slouken@3057
|
117 |
if (!_this->gl_data) {
|
slouken@3057
|
118 |
SDL_OutOfMemory();
|
slouken@3057
|
119 |
return -1;
|
slouken@3057
|
120 |
}
|
slouken@3057
|
121 |
|
slouken@3057
|
122 |
/* Load function pointers */
|
slouken@3057
|
123 |
handle = _this->gl_config.dll_handle;
|
slouken@1952
|
124 |
_this->gl_data->glXGetProcAddress =
|
slouken@4550
|
125 |
(void *(*)(const GLubyte *))
|
slouken@4550
|
126 |
GL_LoadFunction(handle, "glXGetProcAddressARB");
|
slouken@1952
|
127 |
_this->gl_data->glXChooseVisual =
|
slouken@4550
|
128 |
(XVisualInfo * (*)(Display *, int, int *))
|
slouken@4550
|
129 |
X11_GL_GetProcAddress(_this, "glXChooseVisual");
|
slouken@1952
|
130 |
_this->gl_data->glXCreateContext =
|
slouken@1952
|
131 |
(GLXContext(*)(Display *, XVisualInfo *, GLXContext, int))
|
slouken@4550
|
132 |
X11_GL_GetProcAddress(_this, "glXCreateContext");
|
slouken@1952
|
133 |
_this->gl_data->glXDestroyContext =
|
slouken@4550
|
134 |
(void (*)(Display *, GLXContext))
|
slouken@4550
|
135 |
X11_GL_GetProcAddress(_this, "glXDestroyContext");
|
slouken@1952
|
136 |
_this->gl_data->glXMakeCurrent =
|
slouken@4550
|
137 |
(int (*)(Display *, GLXDrawable, GLXContext))
|
slouken@4550
|
138 |
X11_GL_GetProcAddress(_this, "glXMakeCurrent");
|
slouken@1952
|
139 |
_this->gl_data->glXSwapBuffers =
|
slouken@4550
|
140 |
(void (*)(Display *, GLXDrawable))
|
slouken@4550
|
141 |
X11_GL_GetProcAddress(_this, "glXSwapBuffers");
|
slouken@1952
|
142 |
|
slouken@1952
|
143 |
if (!_this->gl_data->glXChooseVisual ||
|
slouken@1952
|
144 |
!_this->gl_data->glXCreateContext ||
|
slouken@1952
|
145 |
!_this->gl_data->glXDestroyContext ||
|
slouken@4550
|
146 |
!_this->gl_data->glXMakeCurrent ||
|
slouken@4550
|
147 |
!_this->gl_data->glXSwapBuffers) {
|
slouken@1952
|
148 |
SDL_SetError("Could not retrieve OpenGL functions");
|
slouken@1952
|
149 |
return -1;
|
slouken@1952
|
150 |
}
|
slouken@1952
|
151 |
|
slouken@3057
|
152 |
/* Initialize extensions */
|
slouken@3057
|
153 |
X11_GL_InitExtensions(_this);
|
slouken@3057
|
154 |
|
slouken@1952
|
155 |
return 0;
|
slouken@1952
|
156 |
}
|
slouken@1952
|
157 |
|
slouken@1952
|
158 |
void *
|
slouken@1952
|
159 |
X11_GL_GetProcAddress(_THIS, const char *proc)
|
slouken@1952
|
160 |
{
|
slouken@1952
|
161 |
if (_this->gl_data->glXGetProcAddress) {
|
slouken@1952
|
162 |
return _this->gl_data->glXGetProcAddress((const GLubyte *) proc);
|
slouken@1952
|
163 |
}
|
slouken@4550
|
164 |
return GL_LoadFunction(_this->gl_config.dll_handle, proc);
|
slouken@1952
|
165 |
}
|
slouken@1952
|
166 |
|
slouken@3057
|
167 |
void
|
slouken@1952
|
168 |
X11_GL_UnloadLibrary(_THIS)
|
slouken@1952
|
169 |
{
|
slouken@3057
|
170 |
/* Don't actually unload the library, since it may have registered
|
slouken@3057
|
171 |
* X11 shutdown hooks, per the notes at:
|
slouken@3057
|
172 |
* http://dri.sourceforge.net/doc/DRIuserguide.html
|
slouken@3057
|
173 |
*/
|
slouken@3057
|
174 |
#if 0
|
slouken@3057
|
175 |
GL_UnloadObject(_this->gl_config.dll_handle);
|
slouken@3057
|
176 |
_this->gl_config.dll_handle = NULL;
|
slouken@3057
|
177 |
#endif
|
slouken@3057
|
178 |
|
slouken@3057
|
179 |
/* Free OpenGL memory */
|
slouken@3057
|
180 |
SDL_free(_this->gl_data);
|
slouken@3057
|
181 |
_this->gl_data = NULL;
|
slouken@1952
|
182 |
}
|
slouken@1952
|
183 |
|
slouken@1952
|
184 |
static SDL_bool
|
slouken@1952
|
185 |
HasExtension(const char *extension, const char *extensions)
|
slouken@1952
|
186 |
{
|
slouken@1952
|
187 |
const char *start;
|
slouken@1952
|
188 |
const char *where, *terminator;
|
slouken@1952
|
189 |
|
slouken@1952
|
190 |
/* Extension names should not have spaces. */
|
slouken@1952
|
191 |
where = SDL_strchr(extension, ' ');
|
slouken@1952
|
192 |
if (where || *extension == '\0')
|
slouken@1952
|
193 |
return SDL_FALSE;
|
slouken@1952
|
194 |
|
slouken@1952
|
195 |
if (!extensions)
|
slouken@1952
|
196 |
return SDL_FALSE;
|
slouken@1952
|
197 |
|
slouken@1952
|
198 |
/* It takes a bit of care to be fool-proof about parsing the
|
slouken@1952
|
199 |
* OpenGL extensions string. Don't be fooled by sub-strings,
|
slouken@1952
|
200 |
* etc. */
|
slouken@1952
|
201 |
|
slouken@1952
|
202 |
start = extensions;
|
slouken@1952
|
203 |
|
slouken@1952
|
204 |
for (;;) {
|
slouken@1952
|
205 |
where = SDL_strstr(start, extension);
|
slouken@1952
|
206 |
if (!where)
|
slouken@1952
|
207 |
break;
|
slouken@1952
|
208 |
|
slouken@1952
|
209 |
terminator = where + SDL_strlen(extension);
|
slouken@1952
|
210 |
if (where == start || *(where - 1) == ' ')
|
slouken@1952
|
211 |
if (*terminator == ' ' || *terminator == '\0')
|
slouken@1952
|
212 |
return SDL_TRUE;
|
slouken@1952
|
213 |
|
slouken@1952
|
214 |
start = terminator;
|
slouken@1952
|
215 |
}
|
slouken@1952
|
216 |
return SDL_FALSE;
|
slouken@1952
|
217 |
}
|
slouken@1952
|
218 |
|
slouken@1952
|
219 |
static void
|
slouken@1952
|
220 |
X11_GL_InitExtensions(_THIS)
|
slouken@1952
|
221 |
{
|
slouken@1952
|
222 |
Display *display = ((SDL_VideoData *) _this->driverdata)->display;
|
slouken@5244
|
223 |
int screen = DefaultScreen(display);
|
slouken@1952
|
224 |
XVisualInfo *vinfo;
|
slouken@1952
|
225 |
XSetWindowAttributes xattr;
|
slouken@1952
|
226 |
Window w;
|
slouken@1952
|
227 |
GLXContext context;
|
slouken@1952
|
228 |
const char *(*glXQueryExtensionsStringFunc) (Display *, int);
|
slouken@1952
|
229 |
const char *extensions;
|
slouken@1952
|
230 |
|
slouken@1952
|
231 |
vinfo = X11_GL_GetVisual(_this, display, screen);
|
slouken@1952
|
232 |
if (!vinfo) {
|
slouken@1952
|
233 |
return;
|
slouken@1952
|
234 |
}
|
slouken@1952
|
235 |
xattr.background_pixel = 0;
|
slouken@1952
|
236 |
xattr.border_pixel = 0;
|
slouken@1952
|
237 |
xattr.colormap =
|
slouken@1952
|
238 |
XCreateColormap(display, RootWindow(display, screen), vinfo->visual,
|
slouken@1952
|
239 |
AllocNone);
|
slouken@1952
|
240 |
w = XCreateWindow(display, RootWindow(display, screen), 0, 0, 32, 32, 0,
|
slouken@1952
|
241 |
vinfo->depth, InputOutput, vinfo->visual,
|
slouken@1952
|
242 |
(CWBackPixel | CWBorderPixel | CWColormap), &xattr);
|
slouken@1952
|
243 |
context = _this->gl_data->glXCreateContext(display, vinfo, NULL, True);
|
slouken@1952
|
244 |
if (context) {
|
slouken@1952
|
245 |
_this->gl_data->glXMakeCurrent(display, w, context);
|
slouken@1952
|
246 |
}
|
slouken@1952
|
247 |
XFree(vinfo);
|
slouken@1952
|
248 |
|
slouken@1952
|
249 |
glXQueryExtensionsStringFunc =
|
slouken@1952
|
250 |
(const char *(*)(Display *, int)) X11_GL_GetProcAddress(_this,
|
slouken@1952
|
251 |
"glXQueryExtensionsString");
|
slouken@1952
|
252 |
if (glXQueryExtensionsStringFunc) {
|
slouken@1952
|
253 |
extensions = glXQueryExtensionsStringFunc(display, screen);
|
slouken@1952
|
254 |
} else {
|
slouken@1952
|
255 |
extensions = NULL;
|
slouken@1952
|
256 |
}
|
slouken@1952
|
257 |
|
slouken@1952
|
258 |
/* Check for SGI_swap_control */
|
icculus@2105
|
259 |
if (HasExtension("GLX_SGI_swap_control", extensions)) {
|
slouken@1952
|
260 |
_this->gl_data->glXSwapIntervalSGI =
|
slouken@1952
|
261 |
(int (*)(int)) X11_GL_GetProcAddress(_this, "glXSwapIntervalSGI");
|
slouken@1952
|
262 |
}
|
slouken@1952
|
263 |
|
slouken@1952
|
264 |
/* Check for GLX_MESA_swap_control */
|
slouken@1952
|
265 |
if (HasExtension("GLX_MESA_swap_control", extensions)) {
|
slouken@1952
|
266 |
_this->gl_data->glXSwapIntervalMESA =
|
slouken@1952
|
267 |
(GLint(*)(unsigned)) X11_GL_GetProcAddress(_this,
|
slouken@1952
|
268 |
"glXSwapIntervalMESA");
|
slouken@1952
|
269 |
_this->gl_data->glXGetSwapIntervalMESA =
|
slouken@1952
|
270 |
(GLint(*)(void)) X11_GL_GetProcAddress(_this,
|
slouken@1952
|
271 |
"glXGetSwapIntervalMESA");
|
slouken@1952
|
272 |
}
|
slouken@1952
|
273 |
|
slouken@1952
|
274 |
/* Check for GLX_EXT_visual_rating */
|
slouken@1952
|
275 |
if (HasExtension("GLX_EXT_visual_rating", extensions)) {
|
slouken@1952
|
276 |
_this->gl_data->HAS_GLX_EXT_visual_rating = SDL_TRUE;
|
slouken@1952
|
277 |
}
|
slouken@1952
|
278 |
|
slouken@1952
|
279 |
if (context) {
|
slouken@1952
|
280 |
_this->gl_data->glXMakeCurrent(display, None, NULL);
|
slouken@1952
|
281 |
_this->gl_data->glXDestroyContext(display, context);
|
slouken@1952
|
282 |
}
|
slouken@1952
|
283 |
XDestroyWindow(display, w);
|
bob@2323
|
284 |
X11_PumpEvents(_this);
|
slouken@1952
|
285 |
}
|
slouken@1952
|
286 |
|
slouken@1952
|
287 |
XVisualInfo *
|
slouken@1952
|
288 |
X11_GL_GetVisual(_THIS, Display * display, int screen)
|
slouken@1952
|
289 |
{
|
slouken@1952
|
290 |
XVisualInfo *vinfo;
|
slouken@1952
|
291 |
|
slouken@1952
|
292 |
/* 64 seems nice. */
|
slouken@1952
|
293 |
int attribs[64];
|
slouken@3100
|
294 |
int i = 0;
|
slouken@3139
|
295 |
|
slouken@1952
|
296 |
/* Setup our GLX attributes according to the gl_config. */
|
slouken@1952
|
297 |
attribs[i++] = GLX_RGBA;
|
slouken@1952
|
298 |
attribs[i++] = GLX_RED_SIZE;
|
slouken@1952
|
299 |
attribs[i++] = _this->gl_config.red_size;
|
slouken@1952
|
300 |
attribs[i++] = GLX_GREEN_SIZE;
|
slouken@1952
|
301 |
attribs[i++] = _this->gl_config.green_size;
|
slouken@1952
|
302 |
attribs[i++] = GLX_BLUE_SIZE;
|
slouken@1952
|
303 |
attribs[i++] = _this->gl_config.blue_size;
|
slouken@1952
|
304 |
|
slouken@1952
|
305 |
if (_this->gl_config.alpha_size) {
|
slouken@1952
|
306 |
attribs[i++] = GLX_ALPHA_SIZE;
|
slouken@1952
|
307 |
attribs[i++] = _this->gl_config.alpha_size;
|
slouken@1952
|
308 |
}
|
slouken@1952
|
309 |
|
slouken@1952
|
310 |
if (_this->gl_config.buffer_size) {
|
slouken@1952
|
311 |
attribs[i++] = GLX_BUFFER_SIZE;
|
slouken@1952
|
312 |
attribs[i++] = _this->gl_config.buffer_size;
|
slouken@1952
|
313 |
}
|
slouken@1952
|
314 |
|
slouken@1952
|
315 |
if (_this->gl_config.double_buffer) {
|
slouken@1952
|
316 |
attribs[i++] = GLX_DOUBLEBUFFER;
|
slouken@1952
|
317 |
}
|
slouken@1952
|
318 |
|
slouken@1952
|
319 |
attribs[i++] = GLX_DEPTH_SIZE;
|
slouken@1952
|
320 |
attribs[i++] = _this->gl_config.depth_size;
|
slouken@1952
|
321 |
|
slouken@1952
|
322 |
if (_this->gl_config.stencil_size) {
|
slouken@1952
|
323 |
attribs[i++] = GLX_STENCIL_SIZE;
|
slouken@1952
|
324 |
attribs[i++] = _this->gl_config.stencil_size;
|
slouken@1952
|
325 |
}
|
slouken@1952
|
326 |
|
slouken@1952
|
327 |
if (_this->gl_config.accum_red_size) {
|
slouken@1952
|
328 |
attribs[i++] = GLX_ACCUM_RED_SIZE;
|
slouken@1952
|
329 |
attribs[i++] = _this->gl_config.accum_red_size;
|
slouken@1952
|
330 |
}
|
slouken@1952
|
331 |
|
slouken@1952
|
332 |
if (_this->gl_config.accum_green_size) {
|
slouken@1952
|
333 |
attribs[i++] = GLX_ACCUM_GREEN_SIZE;
|
slouken@1952
|
334 |
attribs[i++] = _this->gl_config.accum_green_size;
|
slouken@1952
|
335 |
}
|
slouken@1952
|
336 |
|
slouken@1952
|
337 |
if (_this->gl_config.accum_blue_size) {
|
slouken@1952
|
338 |
attribs[i++] = GLX_ACCUM_BLUE_SIZE;
|
slouken@1952
|
339 |
attribs[i++] = _this->gl_config.accum_blue_size;
|
slouken@1952
|
340 |
}
|
slouken@1952
|
341 |
|
slouken@1952
|
342 |
if (_this->gl_config.accum_alpha_size) {
|
slouken@1952
|
343 |
attribs[i++] = GLX_ACCUM_ALPHA_SIZE;
|
slouken@1952
|
344 |
attribs[i++] = _this->gl_config.accum_alpha_size;
|
slouken@1952
|
345 |
}
|
slouken@1952
|
346 |
|
slouken@1952
|
347 |
if (_this->gl_config.stereo) {
|
slouken@1952
|
348 |
attribs[i++] = GLX_STEREO;
|
slouken@1952
|
349 |
}
|
slouken@1952
|
350 |
|
slouken@1952
|
351 |
if (_this->gl_config.multisamplebuffers) {
|
slouken@1952
|
352 |
attribs[i++] = GLX_SAMPLE_BUFFERS_ARB;
|
slouken@1952
|
353 |
attribs[i++] = _this->gl_config.multisamplebuffers;
|
slouken@1952
|
354 |
}
|
slouken@1952
|
355 |
|
slouken@1952
|
356 |
if (_this->gl_config.multisamplesamples) {
|
slouken@1952
|
357 |
attribs[i++] = GLX_SAMPLES_ARB;
|
slouken@1952
|
358 |
attribs[i++] = _this->gl_config.multisamplesamples;
|
slouken@1952
|
359 |
}
|
slouken@1952
|
360 |
|
slouken@3571
|
361 |
if (_this->gl_config.accelerated >= 0 &&
|
slouken@3571
|
362 |
_this->gl_data->HAS_GLX_EXT_visual_rating) {
|
slouken@1952
|
363 |
attribs[i++] = GLX_VISUAL_CAVEAT_EXT;
|
slouken@3570
|
364 |
attribs[i++] = _this->gl_config.accelerated ? GLX_NONE_EXT :
|
slouken@3570
|
365 |
GLX_SLOW_VISUAL_EXT;
|
slouken@1952
|
366 |
}
|
slouken@3570
|
367 |
|
slouken@1952
|
368 |
attribs[i++] = None;
|
slouken@1952
|
369 |
|
slouken@1952
|
370 |
vinfo = _this->gl_data->glXChooseVisual(display, screen, attribs);
|
slouken@1952
|
371 |
if (!vinfo) {
|
slouken@1952
|
372 |
SDL_SetError("Couldn't find matching GLX visual");
|
slouken@1952
|
373 |
}
|
slouken@1952
|
374 |
return vinfo;
|
slouken@1952
|
375 |
}
|
slouken@1952
|
376 |
|
slouken@1952
|
377 |
SDL_GLContext
|
slouken@1952
|
378 |
X11_GL_CreateContext(_THIS, SDL_Window * window)
|
slouken@1952
|
379 |
{
|
slouken@1952
|
380 |
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
slouken@1952
|
381 |
Display *display = data->videodata->display;
|
slouken@1952
|
382 |
int screen =
|
slouken@3685
|
383 |
((SDL_DisplayData *) window->display->driverdata)->screen;
|
slouken@1952
|
384 |
XWindowAttributes xattr;
|
slouken@1952
|
385 |
XVisualInfo v, *vinfo;
|
slouken@1952
|
386 |
int n;
|
slouken@1952
|
387 |
GLXContext context = NULL;
|
slouken@1952
|
388 |
|
slouken@1955
|
389 |
/* We do this to create a clean separation between X and GLX errors. */
|
slouken@1952
|
390 |
XSync(display, False);
|
slouken@3685
|
391 |
XGetWindowAttributes(display, data->xwindow, &xattr);
|
slouken@1952
|
392 |
v.screen = screen;
|
slouken@1952
|
393 |
v.visualid = XVisualIDFromVisual(xattr.visual);
|
slouken@1952
|
394 |
vinfo = XGetVisualInfo(display, VisualScreenMask | VisualIDMask, &v, &n);
|
slouken@1952
|
395 |
if (vinfo) {
|
slouken@3139
|
396 |
if (_this->gl_config.major_version < 3) {
|
slouken@3100
|
397 |
context =
|
slouken@3100
|
398 |
_this->gl_data->glXCreateContext(display, vinfo, NULL, True);
|
slouken@3100
|
399 |
} else {
|
slouken@3100
|
400 |
/* If we want a GL 3.0 context or later we need to get a temporary
|
slouken@3100
|
401 |
context to grab the new context creation function */
|
slouken@3139
|
402 |
GLXContext temp_context =
|
slouken@3139
|
403 |
_this->gl_data->glXCreateContext(display, vinfo, NULL, True);
|
slouken@3100
|
404 |
if (!temp_context) {
|
slouken@3139
|
405 |
SDL_SetError("Could not create GL context");
|
slouken@3100
|
406 |
return NULL;
|
slouken@3100
|
407 |
} else {
|
slouken@3100
|
408 |
int attribs[] = {
|
slouken@3139
|
409 |
GLX_CONTEXT_MAJOR_VERSION_ARB,
|
slouken@3139
|
410 |
_this->gl_config.major_version,
|
slouken@3139
|
411 |
GLX_CONTEXT_MINOR_VERSION_ARB,
|
slouken@3139
|
412 |
_this->gl_config.minor_version,
|
slouken@3139
|
413 |
0
|
slouken@3100
|
414 |
};
|
slouken@3139
|
415 |
|
slouken@3100
|
416 |
/* Get a pointer to the context creation function for GL 3.0 */
|
slouken@3139
|
417 |
PFNGLXCREATECONTEXTATTRIBSARBPROC glXCreateContextAttribs =
|
slouken@3139
|
418 |
(PFNGLXCREATECONTEXTATTRIBSARBPROC) _this->gl_data->
|
slouken@3139
|
419 |
glXGetProcAddress((GLubyte *)
|
slouken@3139
|
420 |
"glXCreateContextAttribsARB");
|
slouken@3100
|
421 |
if (!glXCreateContextAttribs) {
|
slouken@3100
|
422 |
SDL_SetError("GL 3.x is not supported");
|
slouken@3100
|
423 |
context = temp_context;
|
slouken@3100
|
424 |
} else {
|
slouken@3105
|
425 |
/* Create a GL 3.x context */
|
slouken@3100
|
426 |
GLXFBConfig *framebuffer_config = NULL;
|
slouken@3139
|
427 |
int fbcount = 0;
|
slouken@3139
|
428 |
GLXFBConfig *(*glXChooseFBConfig) (Display * disp,
|
slouken@3139
|
429 |
int screen,
|
slouken@3139
|
430 |
const int *attrib_list,
|
slouken@3139
|
431 |
int *nelements);
|
slouken@3105
|
432 |
|
slouken@3139
|
433 |
glXChooseFBConfig =
|
slouken@3139
|
434 |
(GLXFBConfig *
|
slouken@3139
|
435 |
(*)(Display *, int, const int *,
|
slouken@3139
|
436 |
int *)) _this->gl_data->
|
slouken@3139
|
437 |
glXGetProcAddress((GLubyte *) "glXChooseFBConfig");
|
slouken@3105
|
438 |
|
slouken@3139
|
439 |
if (!glXChooseFBConfig
|
slouken@3139
|
440 |
|| !(framebuffer_config =
|
slouken@3139
|
441 |
glXChooseFBConfig(display,
|
slouken@3139
|
442 |
DefaultScreen(display), NULL,
|
slouken@3139
|
443 |
&fbcount))) {
|
slouken@3139
|
444 |
SDL_SetError
|
slouken@3139
|
445 |
("No good framebuffers found. GL 3.x disabled");
|
slouken@3100
|
446 |
context = temp_context;
|
slouken@3139
|
447 |
} else {
|
slouken@3139
|
448 |
context =
|
slouken@3139
|
449 |
glXCreateContextAttribs(display,
|
slouken@3139
|
450 |
framebuffer_config[0],
|
slouken@3139
|
451 |
NULL, True, attribs);
|
slouken@3139
|
452 |
_this->gl_data->glXDestroyContext(display,
|
slouken@3139
|
453 |
temp_context);
|
slouken@3100
|
454 |
}
|
slouken@3100
|
455 |
}
|
slouken@3100
|
456 |
}
|
slouken@3100
|
457 |
}
|
slouken@1952
|
458 |
XFree(vinfo);
|
slouken@1952
|
459 |
}
|
slouken@1952
|
460 |
XSync(display, False);
|
slouken@1952
|
461 |
|
slouken@1952
|
462 |
if (!context) {
|
slouken@1952
|
463 |
SDL_SetError("Could not create GL context");
|
slouken@2178
|
464 |
return NULL;
|
slouken@1952
|
465 |
}
|
slouken@2178
|
466 |
|
slouken@2178
|
467 |
if (X11_GL_MakeCurrent(_this, window, context) < 0) {
|
slouken@2178
|
468 |
X11_GL_DeleteContext(_this, context);
|
slouken@2178
|
469 |
return NULL;
|
slouken@2178
|
470 |
}
|
slouken@2178
|
471 |
|
slouken@2178
|
472 |
return context;
|
slouken@1952
|
473 |
}
|
slouken@1952
|
474 |
|
slouken@1952
|
475 |
int
|
slouken@1952
|
476 |
X11_GL_MakeCurrent(_THIS, SDL_Window * window, SDL_GLContext context)
|
slouken@1952
|
477 |
{
|
slouken@1952
|
478 |
Display *display = ((SDL_VideoData *) _this->driverdata)->display;
|
slouken@1952
|
479 |
Window drawable =
|
slouken@3685
|
480 |
(window ? ((SDL_WindowData *) window->driverdata)->xwindow : None);
|
slouken@1952
|
481 |
GLXContext glx_context = (GLXContext) context;
|
slouken@1952
|
482 |
int status;
|
slouken@1952
|
483 |
|
slouken@1952
|
484 |
status = 0;
|
slouken@1952
|
485 |
if (!_this->gl_data->glXMakeCurrent(display, drawable, glx_context)) {
|
slouken@1952
|
486 |
SDL_SetError("Unable to make GL context current");
|
slouken@1952
|
487 |
status = -1;
|
slouken@1952
|
488 |
}
|
slouken@1952
|
489 |
XSync(display, False);
|
slouken@1952
|
490 |
|
slouken@1952
|
491 |
return (status);
|
slouken@1952
|
492 |
}
|
slouken@1952
|
493 |
|
bob@2327
|
494 |
/*
|
bob@2327
|
495 |
0 is a valid argument to glxSwapIntervalMESA and setting it to 0
|
bob@2327
|
496 |
with the MESA version of the extension will undo the effect of a
|
bob@2327
|
497 |
previous call with a value that is greater than zero (or at least
|
bob@2327
|
498 |
that is what the FM says. OTOH, 0 is an invalid argument to
|
bob@2327
|
499 |
glxSwapIntervalSGI and it returns an error if you call it with 0 as
|
bob@2327
|
500 |
an argument.
|
bob@2327
|
501 |
*/
|
bob@2327
|
502 |
|
bob@2327
|
503 |
static int swapinterval = -1;
|
slouken@1952
|
504 |
int
|
slouken@1952
|
505 |
X11_GL_SetSwapInterval(_THIS, int interval)
|
slouken@1952
|
506 |
{
|
slouken@1952
|
507 |
int status;
|
slouken@1952
|
508 |
|
slouken@1952
|
509 |
if (_this->gl_data->glXSwapIntervalMESA) {
|
slouken@1952
|
510 |
status = _this->gl_data->glXSwapIntervalMESA(interval);
|
slouken@1952
|
511 |
if (status != 0) {
|
slouken@1952
|
512 |
SDL_SetError("glxSwapIntervalMESA failed");
|
slouken@1952
|
513 |
status = -1;
|
bob@2327
|
514 |
} else {
|
bob@2327
|
515 |
swapinterval = interval;
|
slouken@1952
|
516 |
}
|
slouken@1952
|
517 |
} else if (_this->gl_data->glXSwapIntervalSGI) {
|
slouken@1952
|
518 |
status = _this->gl_data->glXSwapIntervalSGI(interval);
|
slouken@1952
|
519 |
if (status != 0) {
|
slouken@1952
|
520 |
SDL_SetError("glxSwapIntervalSGI failed");
|
slouken@1952
|
521 |
status = -1;
|
bob@2327
|
522 |
} else {
|
bob@2327
|
523 |
swapinterval = interval;
|
slouken@1952
|
524 |
}
|
slouken@1952
|
525 |
} else {
|
slouken@1952
|
526 |
SDL_Unsupported();
|
slouken@1952
|
527 |
status = -1;
|
slouken@1952
|
528 |
}
|
slouken@1952
|
529 |
return status;
|
slouken@1952
|
530 |
}
|
slouken@1952
|
531 |
|
slouken@1952
|
532 |
int
|
slouken@1952
|
533 |
X11_GL_GetSwapInterval(_THIS)
|
slouken@1952
|
534 |
{
|
slouken@1952
|
535 |
if (_this->gl_data->glXGetSwapIntervalMESA) {
|
slouken@1952
|
536 |
return _this->gl_data->glXGetSwapIntervalMESA();
|
slouken@1952
|
537 |
} else {
|
bob@2327
|
538 |
return swapinterval;
|
slouken@1952
|
539 |
}
|
slouken@1952
|
540 |
}
|
slouken@1952
|
541 |
|
slouken@1952
|
542 |
void
|
slouken@1952
|
543 |
X11_GL_SwapWindow(_THIS, SDL_Window * window)
|
slouken@1952
|
544 |
{
|
slouken@1952
|
545 |
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
slouken@1952
|
546 |
Display *display = data->videodata->display;
|
slouken@1952
|
547 |
|
slouken@3685
|
548 |
_this->gl_data->glXSwapBuffers(display, data->xwindow);
|
slouken@1952
|
549 |
}
|
slouken@1952
|
550 |
|
slouken@1952
|
551 |
void
|
slouken@1952
|
552 |
X11_GL_DeleteContext(_THIS, SDL_GLContext context)
|
slouken@1952
|
553 |
{
|
slouken@1952
|
554 |
Display *display = ((SDL_VideoData *) _this->driverdata)->display;
|
slouken@1952
|
555 |
GLXContext glx_context = (GLXContext) context;
|
slouken@1952
|
556 |
|
slouken@1952
|
557 |
_this->gl_data->glXDestroyContext(display, glx_context);
|
bob@2324
|
558 |
XSync(display, False);
|
slouken@1952
|
559 |
}
|
slouken@1952
|
560 |
|
slouken@1952
|
561 |
#endif /* SDL_VIDEO_OPENGL_GLX */
|
slouken@1952
|
562 |
|
slouken@1952
|
563 |
/* vi: set ts=4 sw=4 expandtab: */
|