gabomdq@7659
|
1 |
/*
|
gabomdq@7659
|
2 |
* Simple DirectMedia Layer
|
slouken@10737
|
3 |
* Copyright (C) 1997-2017 Sam Lantinga <slouken@libsdl.org>
|
gabomdq@7659
|
4 |
*
|
gabomdq@7659
|
5 |
* This software is provided 'as-is', without any express or implied
|
gabomdq@7659
|
6 |
* warranty. In no event will the authors be held liable for any damages
|
gabomdq@7659
|
7 |
* arising from the use of this software.
|
gabomdq@7659
|
8 |
*
|
gabomdq@7659
|
9 |
* Permission is granted to anyone to use this software for any purpose,
|
gabomdq@7659
|
10 |
* including commercial applications, and to alter it and redistribute it
|
gabomdq@7659
|
11 |
* freely, subject to the following restrictions:
|
gabomdq@7659
|
12 |
*
|
gabomdq@7659
|
13 |
* 1. The origin of this software must not be misrepresented; you must not
|
gabomdq@7659
|
14 |
* claim that you wrote the original software. If you use this software
|
gabomdq@7659
|
15 |
* in a product, an acknowledgment in the product documentation would be
|
gabomdq@7659
|
16 |
* appreciated but is not required.
|
gabomdq@7659
|
17 |
* 2. Altered source versions must be plainly marked as such, and must not be
|
gabomdq@7659
|
18 |
* misrepresented as being the original software.
|
gabomdq@7659
|
19 |
* 3. This notice may not be removed or altered from any source distribution.
|
gabomdq@7659
|
20 |
*/
|
icculus@8093
|
21 |
#include "../SDL_internal.h"
|
gabomdq@7659
|
22 |
|
gabomdq@7659
|
23 |
#if SDL_VIDEO_OPENGL_EGL
|
gabomdq@7659
|
24 |
|
slouken@8734
|
25 |
#if SDL_VIDEO_DRIVER_WINDOWS || SDL_VIDEO_DRIVER_WINRT
|
slouken@8734
|
26 |
#include "../core/windows/SDL_windows.h"
|
slouken@8734
|
27 |
#endif
|
slouken@10878
|
28 |
#if SDL_VIDEO_DRIVER_ANDROID
|
slouken@10878
|
29 |
#include <android/native_window.h>
|
slouken@10878
|
30 |
#endif
|
slouken@8734
|
31 |
|
gabomdq@7659
|
32 |
#include "SDL_sysvideo.h"
|
slouken@11175
|
33 |
#include "SDL_log.h"
|
gabomdq@8021
|
34 |
#include "SDL_egl_c.h"
|
gabomdq@8021
|
35 |
#include "SDL_loadso.h"
|
gabomdq@8021
|
36 |
#include "SDL_hints.h"
|
gabomdq@7782
|
37 |
|
slime73@9595
|
38 |
#ifdef EGL_KHR_create_context
|
slime73@9595
|
39 |
/* EGL_OPENGL_ES3_BIT_KHR was added in version 13 of the extension. */
|
slime73@9595
|
40 |
#ifndef EGL_OPENGL_ES3_BIT_KHR
|
slime73@9595
|
41 |
#define EGL_OPENGL_ES3_BIT_KHR 0x00000040
|
slime73@9595
|
42 |
#endif
|
slime73@9595
|
43 |
#endif /* EGL_KHR_create_context */
|
slime73@9595
|
44 |
|
gabomdq@7753
|
45 |
#if SDL_VIDEO_DRIVER_RPI
|
gabomdq@7782
|
46 |
/* Raspbian places the OpenGL ES/EGL binaries in a non standard path */
|
connogriofa@11676
|
47 |
#define DEFAULT_EGL ( vc4 ? "libEGL.so.1" : "/opt/vc/lib/libbrcmEGL.so" )
|
connogriofa@11676
|
48 |
#define DEFAULT_OGL_ES2 ( vc4 ? "libGLESv2.so.2" : "/opt/vc/lib/libbrcmGLESv2.so" )
|
icculus@11462
|
49 |
#define ALT_EGL "/opt/vc/lib/libEGL.so"
|
icculus@11462
|
50 |
#define ALT_OGL_ES2 "/opt/vc/lib/libGLESv2.so"
|
connogriofa@11676
|
51 |
#define DEFAULT_OGL_ES_PVR ( vc4 ? "libGLES_CM.so.1" : "/opt/vc/lib/libbrcmGLESv2.so" )
|
connogriofa@11676
|
52 |
#define DEFAULT_OGL_ES ( vc4 ? "libGLESv1_CM.so.1" : "/opt/vc/lib/libbrcmGLESv2.so" )
|
gabomdq@7782
|
53 |
|
slouken@9164
|
54 |
#elif SDL_VIDEO_DRIVER_ANDROID || SDL_VIDEO_DRIVER_VIVANTE
|
gabomdq@7782
|
55 |
/* Android */
|
gabomdq@7782
|
56 |
#define DEFAULT_EGL "libEGL.so"
|
gabomdq@7782
|
57 |
#define DEFAULT_OGL_ES2 "libGLESv2.so"
|
gabomdq@7782
|
58 |
#define DEFAULT_OGL_ES_PVR "libGLES_CM.so"
|
gabomdq@7782
|
59 |
#define DEFAULT_OGL_ES "libGLESv1_CM.so"
|
gabomdq@7782
|
60 |
|
dludwig@8545
|
61 |
#elif SDL_VIDEO_DRIVER_WINDOWS || SDL_VIDEO_DRIVER_WINRT
|
gabomdq@8021
|
62 |
/* EGL AND OpenGL ES support via ANGLE */
|
gabomdq@8021
|
63 |
#define DEFAULT_EGL "libEGL.dll"
|
gabomdq@8021
|
64 |
#define DEFAULT_OGL_ES2 "libGLESv2.dll"
|
gabomdq@8021
|
65 |
#define DEFAULT_OGL_ES_PVR "libGLES_CM.dll"
|
gabomdq@8021
|
66 |
#define DEFAULT_OGL_ES "libGLESv1_CM.dll"
|
gabomdq@8021
|
67 |
|
gabomdq@7753
|
68 |
#else
|
gabomdq@7782
|
69 |
/* Desktop Linux */
|
gabomdq@8062
|
70 |
#define DEFAULT_OGL "libGL.so.1"
|
gabomdq@7753
|
71 |
#define DEFAULT_EGL "libEGL.so.1"
|
gabomdq@7753
|
72 |
#define DEFAULT_OGL_ES2 "libGLESv2.so.2"
|
gabomdq@7753
|
73 |
#define DEFAULT_OGL_ES_PVR "libGLES_CM.so.1"
|
gabomdq@7753
|
74 |
#define DEFAULT_OGL_ES "libGLESv1_CM.so.1"
|
gabomdq@7753
|
75 |
#endif /* SDL_VIDEO_DRIVER_RPI */
|
gabomdq@7659
|
76 |
|
slouken@11363
|
77 |
#ifdef SDL_VIDEO_STATIC_ANGLE
|
slouken@11363
|
78 |
#define LOAD_FUNC(NAME) \
|
slouken@11363
|
79 |
_this->egl_data->NAME = (void *)NAME;
|
slouken@11363
|
80 |
#else
|
gabomdq@7659
|
81 |
#define LOAD_FUNC(NAME) \
|
slouken@9162
|
82 |
_this->egl_data->NAME = SDL_LoadFunction(_this->egl_data->dll_handle, #NAME); \
|
gabomdq@7659
|
83 |
if (!_this->egl_data->NAME) \
|
gabomdq@7659
|
84 |
{ \
|
gabomdq@7659
|
85 |
return SDL_SetError("Could not retrieve EGL function " #NAME); \
|
gabomdq@7659
|
86 |
}
|
slouken@11363
|
87 |
#endif
|
dludwig@10740
|
88 |
|
dludwig@10740
|
89 |
static const char * SDL_EGL_GetErrorName(EGLint eglErrorCode)
|
dludwig@10740
|
90 |
{
|
dludwig@10740
|
91 |
#define SDL_EGL_ERROR_TRANSLATE(e) case e: return #e;
|
dludwig@10740
|
92 |
switch (eglErrorCode) {
|
dludwig@10740
|
93 |
SDL_EGL_ERROR_TRANSLATE(EGL_SUCCESS);
|
dludwig@10740
|
94 |
SDL_EGL_ERROR_TRANSLATE(EGL_NOT_INITIALIZED);
|
dludwig@10740
|
95 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_ACCESS);
|
dludwig@10740
|
96 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_ALLOC);
|
dludwig@10740
|
97 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_ATTRIBUTE);
|
dludwig@10740
|
98 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_CONTEXT);
|
dludwig@10740
|
99 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_CONFIG);
|
dludwig@10740
|
100 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_CURRENT_SURFACE);
|
dludwig@10740
|
101 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_DISPLAY);
|
dludwig@10740
|
102 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_SURFACE);
|
dludwig@10740
|
103 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_MATCH);
|
dludwig@10740
|
104 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_PARAMETER);
|
dludwig@10740
|
105 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_NATIVE_PIXMAP);
|
dludwig@10740
|
106 |
SDL_EGL_ERROR_TRANSLATE(EGL_BAD_NATIVE_WINDOW);
|
dludwig@10740
|
107 |
SDL_EGL_ERROR_TRANSLATE(EGL_CONTEXT_LOST);
|
dludwig@10740
|
108 |
}
|
dludwig@10740
|
109 |
return "";
|
dludwig@10740
|
110 |
}
|
dludwig@10740
|
111 |
|
dludwig@10740
|
112 |
int SDL_EGL_SetErrorEx(const char * message, const char * eglFunctionName, EGLint eglErrorCode)
|
dludwig@10740
|
113 |
{
|
dludwig@10740
|
114 |
const char * errorText = SDL_EGL_GetErrorName(eglErrorCode);
|
dludwig@10740
|
115 |
char altErrorText[32];
|
dludwig@10740
|
116 |
if (errorText[0] == '\0') {
|
dludwig@10740
|
117 |
/* An unknown-to-SDL error code was reported. Report its hexadecimal value, instead of its name. */
|
dludwig@10740
|
118 |
SDL_snprintf(altErrorText, SDL_arraysize(altErrorText), "0x%x", (unsigned int)eglErrorCode);
|
dludwig@10740
|
119 |
errorText = altErrorText;
|
dludwig@10740
|
120 |
}
|
dludwig@10740
|
121 |
return SDL_SetError("%s (call to %s failed, reporting an error of %s)", message, eglFunctionName, errorText);
|
dludwig@10740
|
122 |
}
|
dludwig@10740
|
123 |
|
gabomdq@7659
|
124 |
/* EGL implementation of SDL OpenGL ES support */
|
slouken@11175
|
125 |
typedef enum {
|
slouken@11175
|
126 |
SDL_EGL_DISPLAY_EXTENSION,
|
slouken@11175
|
127 |
SDL_EGL_CLIENT_EXTENSION
|
slouken@11175
|
128 |
} SDL_EGL_ExtensionType;
|
slouken@11175
|
129 |
|
slouken@11175
|
130 |
static SDL_bool SDL_EGL_HasExtension(_THIS, SDL_EGL_ExtensionType type, const char *ext)
|
knut@8962
|
131 |
{
|
slouken@9821
|
132 |
size_t ext_len;
|
slouken@11175
|
133 |
const char *ext_override;
|
slouken@11175
|
134 |
const char *egl_extstr;
|
slouken@11175
|
135 |
const char *ext_start;
|
slouken@11175
|
136 |
|
slouken@11175
|
137 |
/* Invalid extensions can be rejected early */
|
slouken@11175
|
138 |
if (ext == NULL || *ext == 0 || SDL_strchr(ext, ' ') != NULL) {
|
slouken@11175
|
139 |
/* SDL_LogDebug(SDL_LOG_CATEGORY_VIDEO, "SDL_EGL_HasExtension: Invalid EGL extension"); */
|
slouken@11175
|
140 |
return SDL_FALSE;
|
slouken@11175
|
141 |
}
|
slouken@11175
|
142 |
|
slouken@11175
|
143 |
/* Extensions can be masked with an environment variable.
|
slouken@11175
|
144 |
* Unlike the OpenGL override, this will use the set bits of an integer
|
slouken@11175
|
145 |
* to disable the extension.
|
slouken@11175
|
146 |
* Bit Action
|
slouken@11175
|
147 |
* 0 If set, the display extension is masked and not present to SDL.
|
slouken@11175
|
148 |
* 1 If set, the client extension is masked and not present to SDL.
|
slouken@11175
|
149 |
*/
|
slouken@11175
|
150 |
ext_override = SDL_getenv(ext);
|
slouken@11175
|
151 |
if (ext_override != NULL) {
|
slouken@11175
|
152 |
int disable_ext = SDL_atoi(ext_override);
|
slouken@11175
|
153 |
if (disable_ext & 0x01 && type == SDL_EGL_DISPLAY_EXTENSION) {
|
slouken@11175
|
154 |
return SDL_FALSE;
|
slouken@11175
|
155 |
} else if (disable_ext & 0x02 && type == SDL_EGL_CLIENT_EXTENSION) {
|
slouken@11175
|
156 |
return SDL_FALSE;
|
slouken@11175
|
157 |
}
|
slouken@11175
|
158 |
}
|
knut@8962
|
159 |
|
knut@8962
|
160 |
ext_len = SDL_strlen(ext);
|
slouken@11175
|
161 |
switch (type) {
|
slouken@11175
|
162 |
case SDL_EGL_DISPLAY_EXTENSION:
|
slouken@11175
|
163 |
egl_extstr = _this->egl_data->eglQueryString(_this->egl_data->egl_display, EGL_EXTENSIONS);
|
slouken@11175
|
164 |
break;
|
slouken@11175
|
165 |
case SDL_EGL_CLIENT_EXTENSION:
|
slouken@11175
|
166 |
/* EGL_EXT_client_extensions modifies eglQueryString to return client extensions
|
slouken@11175
|
167 |
* if EGL_NO_DISPLAY is passed. Implementations without it are required to return NULL.
|
slouken@11175
|
168 |
* This behavior is included in EGL 1.5.
|
slouken@11175
|
169 |
*/
|
slouken@11175
|
170 |
egl_extstr = _this->egl_data->eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS);
|
slouken@11175
|
171 |
break;
|
slouken@11175
|
172 |
default:
|
slouken@11175
|
173 |
/* SDL_LogDebug(SDL_LOG_CATEGORY_VIDEO, "SDL_EGL_HasExtension: Invalid extension type"); */
|
slouken@11175
|
174 |
return SDL_FALSE;
|
slouken@11175
|
175 |
}
|
knut@8962
|
176 |
|
slouken@11175
|
177 |
if (egl_extstr != NULL) {
|
slouken@11175
|
178 |
ext_start = egl_extstr;
|
knut@8962
|
179 |
|
slouken@11175
|
180 |
while (*ext_start) {
|
slouken@11175
|
181 |
ext_start = SDL_strstr(ext_start, ext);
|
slouken@11175
|
182 |
if (ext_start == NULL) {
|
slouken@11175
|
183 |
return SDL_FALSE;
|
slouken@11175
|
184 |
}
|
slouken@11175
|
185 |
/* Check if the match is not just a substring of one of the extensions */
|
slouken@11175
|
186 |
if (ext_start == egl_extstr || *(ext_start - 1) == ' ') {
|
slouken@11175
|
187 |
if (ext_start[ext_len] == ' ' || ext_start[ext_len] == 0) {
|
slouken@11175
|
188 |
return SDL_TRUE;
|
knut@8962
|
189 |
}
|
slouken@11175
|
190 |
}
|
slouken@11175
|
191 |
/* If the search stopped in the middle of an extension, skip to the end of it */
|
slouken@11175
|
192 |
ext_start += ext_len;
|
slouken@11175
|
193 |
while (*ext_start != ' ' && *ext_start != 0) {
|
slouken@11175
|
194 |
ext_start++;
|
knut@8962
|
195 |
}
|
knut@8962
|
196 |
}
|
knut@8962
|
197 |
}
|
knut@8962
|
198 |
|
slouken@11175
|
199 |
return SDL_FALSE;
|
knut@8962
|
200 |
}
|
gabomdq@7659
|
201 |
|
gabomdq@7659
|
202 |
void *
|
gabomdq@7659
|
203 |
SDL_EGL_GetProcAddress(_THIS, const char *proc)
|
gabomdq@7659
|
204 |
{
|
gabomdq@7659
|
205 |
static char procname[1024];
|
gabomdq@7659
|
206 |
void *retval;
|
gabomdq@7659
|
207 |
|
gabomdq@7659
|
208 |
/* eglGetProcAddress is busted on Android http://code.google.com/p/android/issues/detail?id=7681 */
|
brandon@11516
|
209 |
#if !defined(SDL_VIDEO_DRIVER_ANDROID)
|
gabomdq@7659
|
210 |
if (_this->egl_data->eglGetProcAddress) {
|
gabomdq@7659
|
211 |
retval = _this->egl_data->eglGetProcAddress(proc);
|
gabomdq@7659
|
212 |
if (retval) {
|
gabomdq@7659
|
213 |
return retval;
|
gabomdq@7659
|
214 |
}
|
gabomdq@7659
|
215 |
}
|
gabomdq@7659
|
216 |
#endif
|
gabomdq@7659
|
217 |
|
gabomdq@8021
|
218 |
retval = SDL_LoadFunction(_this->egl_data->egl_dll_handle, proc);
|
gabomdq@8021
|
219 |
if (!retval && SDL_strlen(proc) <= 1022) {
|
gabomdq@7659
|
220 |
procname[0] = '_';
|
gabomdq@8021
|
221 |
SDL_strlcpy(procname + 1, proc, 1022);
|
gabomdq@8021
|
222 |
retval = SDL_LoadFunction(_this->egl_data->egl_dll_handle, procname);
|
gabomdq@7659
|
223 |
}
|
gabomdq@7659
|
224 |
return retval;
|
gabomdq@7659
|
225 |
}
|
gabomdq@7659
|
226 |
|
gabomdq@7659
|
227 |
void
|
gabomdq@7659
|
228 |
SDL_EGL_UnloadLibrary(_THIS)
|
gabomdq@7659
|
229 |
{
|
slouken@7865
|
230 |
if (_this->egl_data) {
|
slouken@7865
|
231 |
if (_this->egl_data->egl_display) {
|
slouken@7865
|
232 |
_this->egl_data->eglTerminate(_this->egl_data->egl_display);
|
slouken@7865
|
233 |
_this->egl_data->egl_display = NULL;
|
slouken@7865
|
234 |
}
|
slouken@7865
|
235 |
|
gabomdq@8021
|
236 |
if (_this->egl_data->dll_handle) {
|
gabomdq@8021
|
237 |
SDL_UnloadObject(_this->egl_data->dll_handle);
|
gabomdq@8021
|
238 |
_this->egl_data->dll_handle = NULL;
|
slouken@7865
|
239 |
}
|
slouken@7865
|
240 |
if (_this->egl_data->egl_dll_handle) {
|
gabomdq@8021
|
241 |
SDL_UnloadObject(_this->egl_data->egl_dll_handle);
|
slouken@7865
|
242 |
_this->egl_data->egl_dll_handle = NULL;
|
slouken@7865
|
243 |
}
|
gabomdq@7659
|
244 |
|
gabomdq@7659
|
245 |
SDL_free(_this->egl_data);
|
gabomdq@7659
|
246 |
_this->egl_data = NULL;
|
gabomdq@7659
|
247 |
}
|
gabomdq@7659
|
248 |
}
|
gabomdq@7659
|
249 |
|
gabomdq@7659
|
250 |
int
|
slouken@11175
|
251 |
SDL_EGL_LoadLibrary(_THIS, const char *egl_path, NativeDisplayType native_display, EGLenum platform)
|
gabomdq@7659
|
252 |
{
|
gabomdq@8021
|
253 |
void *dll_handle = NULL, *egl_dll_handle = NULL; /* The naming is counter intuitive, but hey, I just work here -- Gabriel */
|
slouken@10461
|
254 |
const char *path = NULL;
|
slouken@11175
|
255 |
int egl_version_major = 0, egl_version_minor = 0;
|
dludwig@8545
|
256 |
#if SDL_VIDEO_DRIVER_WINDOWS || SDL_VIDEO_DRIVER_WINRT
|
gabomdq@8021
|
257 |
const char *d3dcompiler;
|
gabomdq@8021
|
258 |
#endif
|
connogriofa@11676
|
259 |
#if SDL_VIDEO_DRIVER_RPI
|
connogriofa@11676
|
260 |
SDL_bool vc4 = (0 == access("/sys/module/vc4/", F_OK));
|
connogriofa@11676
|
261 |
#endif
|
gabomdq@8021
|
262 |
|
gabomdq@7659
|
263 |
if (_this->egl_data) {
|
slouken@11718
|
264 |
return SDL_SetError("EGL context already created");
|
gabomdq@7659
|
265 |
}
|
slouken@7865
|
266 |
|
slouken@7865
|
267 |
_this->egl_data = (struct SDL_EGL_VideoData *) SDL_calloc(1, sizeof(SDL_EGL_VideoData));
|
slouken@7865
|
268 |
if (!_this->egl_data) {
|
slouken@7865
|
269 |
return SDL_OutOfMemory();
|
slouken@7865
|
270 |
}
|
slouken@7865
|
271 |
|
dludwig@8545
|
272 |
#if SDL_VIDEO_DRIVER_WINDOWS || SDL_VIDEO_DRIVER_WINRT
|
gabomdq@8021
|
273 |
d3dcompiler = SDL_GetHint(SDL_HINT_VIDEO_WIN_D3DCOMPILER);
|
gabomdq@8021
|
274 |
if (!d3dcompiler) {
|
slouken@8733
|
275 |
if (WIN_IsWindowsVistaOrGreater()) {
|
slouken@8733
|
276 |
d3dcompiler = "d3dcompiler_46.dll";
|
slouken@8733
|
277 |
} else {
|
slouken@8740
|
278 |
d3dcompiler = "d3dcompiler_43.dll";
|
slouken@8733
|
279 |
}
|
gabomdq@8021
|
280 |
}
|
gabomdq@8021
|
281 |
if (SDL_strcasecmp(d3dcompiler, "none") != 0) {
|
slouken@11219
|
282 |
if (SDL_LoadObject(d3dcompiler) == NULL) {
|
slouken@11219
|
283 |
SDL_ClearError();
|
slouken@11219
|
284 |
}
|
gabomdq@8021
|
285 |
}
|
slouken@7865
|
286 |
#endif
|
slouken@7865
|
287 |
|
slouken@11363
|
288 |
#ifndef SDL_VIDEO_STATIC_ANGLE
|
gabomdq@7753
|
289 |
/* A funny thing, loading EGL.so first does not work on the Raspberry, so we load libGL* first */
|
gabomdq@8021
|
290 |
path = SDL_getenv("SDL_VIDEO_GL_DRIVER");
|
gabomdq@8021
|
291 |
if (path != NULL) {
|
gabomdq@8021
|
292 |
egl_dll_handle = SDL_LoadObject(path);
|
gabomdq@8021
|
293 |
}
|
gabomdq@8021
|
294 |
|
gabomdq@8021
|
295 |
if (egl_dll_handle == NULL) {
|
slime73@9595
|
296 |
if (_this->gl_config.profile_mask == SDL_GL_CONTEXT_PROFILE_ES) {
|
gabomdq@8062
|
297 |
if (_this->gl_config.major_version > 1) {
|
gabomdq@8062
|
298 |
path = DEFAULT_OGL_ES2;
|
gabomdq@8062
|
299 |
egl_dll_handle = SDL_LoadObject(path);
|
icculus@11462
|
300 |
#ifdef ALT_OGL_ES2
|
connogriofa@11676
|
301 |
if (egl_dll_handle == NULL && !vc4) {
|
icculus@11462
|
302 |
path = ALT_OGL_ES2;
|
icculus@11462
|
303 |
egl_dll_handle = SDL_LoadObject(path);
|
icculus@11462
|
304 |
}
|
icculus@11462
|
305 |
#endif
|
icculus@11462
|
306 |
|
slime73@9595
|
307 |
} else {
|
gabomdq@8062
|
308 |
path = DEFAULT_OGL_ES;
|
gabomdq@8062
|
309 |
egl_dll_handle = SDL_LoadObject(path);
|
gabomdq@8062
|
310 |
if (egl_dll_handle == NULL) {
|
gabomdq@8062
|
311 |
path = DEFAULT_OGL_ES_PVR;
|
gabomdq@8062
|
312 |
egl_dll_handle = SDL_LoadObject(path);
|
gabomdq@8062
|
313 |
}
|
connogriofa@11676
|
314 |
#ifdef ALT_OGL_ES2
|
connogriofa@11676
|
315 |
if (egl_dll_handle == NULL && !vc4) {
|
connogriofa@11676
|
316 |
path = ALT_OGL_ES2;
|
connogriofa@11676
|
317 |
egl_dll_handle = SDL_LoadObject(path);
|
connogriofa@11676
|
318 |
}
|
connogriofa@11676
|
319 |
#endif
|
gabomdq@8062
|
320 |
}
|
gabomdq@8062
|
321 |
}
|
gabomdq@8062
|
322 |
#ifdef DEFAULT_OGL
|
gabomdq@8062
|
323 |
else {
|
gabomdq@8062
|
324 |
path = DEFAULT_OGL;
|
gabomdq@8021
|
325 |
egl_dll_handle = SDL_LoadObject(path);
|
gabomdq@8021
|
326 |
}
|
gabomdq@8062
|
327 |
#endif
|
gabomdq@7753
|
328 |
}
|
slouken@7865
|
329 |
_this->egl_data->egl_dll_handle = egl_dll_handle;
|
gabomdq@7753
|
330 |
|
gabomdq@7753
|
331 |
if (egl_dll_handle == NULL) {
|
gabomdq@8062
|
332 |
return SDL_SetError("Could not initialize OpenGL / GLES library");
|
gabomdq@7753
|
333 |
}
|
gabomdq@8021
|
334 |
|
gabomdq@7753
|
335 |
/* Loading libGL* in the previous step took care of loading libEGL.so, but we future proof by double checking */
|
gabomdq@8021
|
336 |
if (egl_path != NULL) {
|
gabomdq@8021
|
337 |
dll_handle = SDL_LoadObject(egl_path);
|
gabomdq@8021
|
338 |
}
|
gabomdq@8105
|
339 |
/* Try loading a EGL symbol, if it does not work try the default library paths */
|
gabomdq@8872
|
340 |
if (dll_handle == NULL || SDL_LoadFunction(dll_handle, "eglChooseConfig") == NULL) {
|
gabomdq@8021
|
341 |
if (dll_handle != NULL) {
|
gabomdq@8021
|
342 |
SDL_UnloadObject(dll_handle);
|
gabomdq@8021
|
343 |
}
|
gabomdq@8021
|
344 |
path = SDL_getenv("SDL_VIDEO_EGL_DRIVER");
|
gabomdq@7659
|
345 |
if (path == NULL) {
|
gabomdq@7659
|
346 |
path = DEFAULT_EGL;
|
gabomdq@7659
|
347 |
}
|
gabomdq@8021
|
348 |
dll_handle = SDL_LoadObject(path);
|
icculus@11462
|
349 |
|
icculus@11462
|
350 |
#ifdef ALT_EGL
|
connogriofa@11676
|
351 |
if (dll_handle == NULL && !vc4) {
|
icculus@11462
|
352 |
path = ALT_EGL;
|
icculus@11462
|
353 |
dll_handle = SDL_LoadObject(path);
|
icculus@11462
|
354 |
}
|
icculus@11462
|
355 |
#endif
|
icculus@11462
|
356 |
|
gabomdq@8872
|
357 |
if (dll_handle == NULL || SDL_LoadFunction(dll_handle, "eglChooseConfig") == NULL) {
|
gabomdq@8872
|
358 |
if (dll_handle != NULL) {
|
gabomdq@8872
|
359 |
SDL_UnloadObject(dll_handle);
|
gabomdq@8872
|
360 |
}
|
gabomdq@8021
|
361 |
return SDL_SetError("Could not load EGL library");
|
gabomdq@8021
|
362 |
}
|
gabomdq@8763
|
363 |
SDL_ClearError();
|
gabomdq@7753
|
364 |
}
|
slouken@11363
|
365 |
#endif
|
slouken@7865
|
366 |
|
gabomdq@8021
|
367 |
_this->egl_data->dll_handle = dll_handle;
|
gabomdq@7659
|
368 |
|
gabomdq@7659
|
369 |
/* Load new function pointers */
|
gabomdq@7659
|
370 |
LOAD_FUNC(eglGetDisplay);
|
gabomdq@7659
|
371 |
LOAD_FUNC(eglInitialize);
|
gabomdq@7659
|
372 |
LOAD_FUNC(eglTerminate);
|
gabomdq@7659
|
373 |
LOAD_FUNC(eglGetProcAddress);
|
gabomdq@7659
|
374 |
LOAD_FUNC(eglChooseConfig);
|
gabomdq@7659
|
375 |
LOAD_FUNC(eglGetConfigAttrib);
|
gabomdq@7659
|
376 |
LOAD_FUNC(eglCreateContext);
|
gabomdq@7659
|
377 |
LOAD_FUNC(eglDestroyContext);
|
slouken@11718
|
378 |
LOAD_FUNC(eglCreatePBufferSurface);
|
gabomdq@7659
|
379 |
LOAD_FUNC(eglCreateWindowSurface);
|
gabomdq@7659
|
380 |
LOAD_FUNC(eglDestroySurface);
|
gabomdq@7659
|
381 |
LOAD_FUNC(eglMakeCurrent);
|
gabomdq@7659
|
382 |
LOAD_FUNC(eglSwapBuffers);
|
gabomdq@7659
|
383 |
LOAD_FUNC(eglSwapInterval);
|
gabomdq@7659
|
384 |
LOAD_FUNC(eglWaitNative);
|
gabomdq@7659
|
385 |
LOAD_FUNC(eglWaitGL);
|
gabomdq@8062
|
386 |
LOAD_FUNC(eglBindAPI);
|
knut@8962
|
387 |
LOAD_FUNC(eglQueryString);
|
dludwig@10740
|
388 |
LOAD_FUNC(eglGetError);
|
dludwig@10740
|
389 |
|
slouken@11175
|
390 |
if (_this->egl_data->eglQueryString) {
|
slouken@11175
|
391 |
/* EGL 1.5 allows querying for client version */
|
slouken@11175
|
392 |
const char *egl_version = _this->egl_data->eglQueryString(EGL_NO_DISPLAY, EGL_VERSION);
|
slouken@11175
|
393 |
if (egl_version != NULL) {
|
slouken@11175
|
394 |
if (SDL_sscanf(egl_version, "%d.%d", &egl_version_major, &egl_version_minor) != 2) {
|
slouken@11175
|
395 |
egl_version_major = 0;
|
slouken@11175
|
396 |
egl_version_minor = 0;
|
slouken@11175
|
397 |
SDL_LogWarn(SDL_LOG_CATEGORY_VIDEO, "Could not parse EGL version string: %s", egl_version);
|
slouken@11175
|
398 |
}
|
slouken@11175
|
399 |
}
|
slouken@11175
|
400 |
}
|
slouken@11175
|
401 |
|
slouken@11175
|
402 |
if (egl_version_major == 1 && egl_version_minor == 5) {
|
slouken@11175
|
403 |
LOAD_FUNC(eglGetPlatformDisplay);
|
slouken@11175
|
404 |
}
|
slouken@11175
|
405 |
|
slouken@11175
|
406 |
_this->egl_data->egl_display = EGL_NO_DISPLAY;
|
dludwig@8663
|
407 |
#if !defined(__WINRT__)
|
slouken@11175
|
408 |
if (platform) {
|
slouken@11175
|
409 |
if (egl_version_major == 1 && egl_version_minor == 5) {
|
icculus@11418
|
410 |
_this->egl_data->egl_display = _this->egl_data->eglGetPlatformDisplay(platform, (void *)(size_t)native_display, NULL);
|
slouken@11175
|
411 |
} else {
|
slouken@11175
|
412 |
if (SDL_EGL_HasExtension(_this, SDL_EGL_CLIENT_EXTENSION, "EGL_EXT_platform_base")) {
|
slouken@11175
|
413 |
_this->egl_data->eglGetPlatformDisplayEXT = SDL_EGL_GetProcAddress(_this, "eglGetPlatformDisplayEXT");
|
slouken@11175
|
414 |
if (_this->egl_data->eglGetPlatformDisplayEXT) {
|
icculus@11418
|
415 |
_this->egl_data->egl_display = _this->egl_data->eglGetPlatformDisplayEXT(platform, (void *)(size_t)native_display, NULL);
|
slouken@11175
|
416 |
}
|
slouken@11175
|
417 |
}
|
slouken@11175
|
418 |
}
|
slouken@11175
|
419 |
}
|
slouken@11175
|
420 |
/* Try the implementation-specific eglGetDisplay even if eglGetPlatformDisplay fails */
|
slouken@11175
|
421 |
if (_this->egl_data->egl_display == EGL_NO_DISPLAY) {
|
slouken@11175
|
422 |
_this->egl_data->egl_display = _this->egl_data->eglGetDisplay(native_display);
|
slouken@11175
|
423 |
}
|
slouken@11175
|
424 |
if (_this->egl_data->egl_display == EGL_NO_DISPLAY) {
|
gabomdq@7659
|
425 |
return SDL_SetError("Could not get EGL display");
|
gabomdq@7659
|
426 |
}
|
gabomdq@7659
|
427 |
|
gabomdq@7659
|
428 |
if (_this->egl_data->eglInitialize(_this->egl_data->egl_display, NULL, NULL) != EGL_TRUE) {
|
gabomdq@7659
|
429 |
return SDL_SetError("Could not initialize EGL");
|
gabomdq@7659
|
430 |
}
|
dludwig@8663
|
431 |
#endif
|
gabomdq@7659
|
432 |
|
gabomdq@7659
|
433 |
if (path) {
|
gabomdq@8021
|
434 |
SDL_strlcpy(_this->gl_config.driver_path, path, sizeof(_this->gl_config.driver_path) - 1);
|
gabomdq@7659
|
435 |
} else {
|
gabomdq@8021
|
436 |
*_this->gl_config.driver_path = '\0';
|
gabomdq@7659
|
437 |
}
|
gabomdq@7659
|
438 |
|
gabomdq@8041
|
439 |
return 0;
|
gabomdq@7659
|
440 |
}
|
gabomdq@7659
|
441 |
|
gabomdq@7659
|
442 |
int
|
gabomdq@7659
|
443 |
SDL_EGL_ChooseConfig(_THIS)
|
gabomdq@7659
|
444 |
{
|
slouken@11175
|
445 |
/* 64 seems nice. */
|
gabomdq@7659
|
446 |
EGLint attribs[64];
|
gabomdq@8021
|
447 |
EGLint found_configs = 0, value;
|
slouken@11175
|
448 |
#ifdef SDL_VIDEO_DRIVER_KMSDRM
|
slouken@11175
|
449 |
/* Intel EGL on KMS/DRM (al least) returns invalid configs that confuse the bitdiff search used */
|
slouken@11175
|
450 |
/* later in this function, so we simply use the first one when using the KMSDRM driver for now. */
|
slouken@11175
|
451 |
EGLConfig configs[1];
|
slouken@11175
|
452 |
#else
|
gabomdq@8015
|
453 |
/* 128 seems even nicer here */
|
gabomdq@8015
|
454 |
EGLConfig configs[128];
|
slouken@11175
|
455 |
#endif
|
gabomdq@8021
|
456 |
int i, j, best_bitdiff = -1, bitdiff;
|
slouken@11175
|
457 |
|
gabomdq@7659
|
458 |
if (!_this->egl_data) {
|
gabomdq@7659
|
459 |
/* The EGL library wasn't loaded, SDL_GetError() should have info */
|
gabomdq@7659
|
460 |
return -1;
|
gabomdq@7659
|
461 |
}
|
gabomdq@7659
|
462 |
|
gabomdq@7659
|
463 |
/* Get a valid EGL configuration */
|
gabomdq@7659
|
464 |
i = 0;
|
gabomdq@7659
|
465 |
attribs[i++] = EGL_RED_SIZE;
|
gabomdq@7659
|
466 |
attribs[i++] = _this->gl_config.red_size;
|
gabomdq@7659
|
467 |
attribs[i++] = EGL_GREEN_SIZE;
|
gabomdq@7659
|
468 |
attribs[i++] = _this->gl_config.green_size;
|
gabomdq@7659
|
469 |
attribs[i++] = EGL_BLUE_SIZE;
|
gabomdq@7659
|
470 |
attribs[i++] = _this->gl_config.blue_size;
|
gabomdq@7659
|
471 |
|
gabomdq@7659
|
472 |
if (_this->gl_config.alpha_size) {
|
gabomdq@7659
|
473 |
attribs[i++] = EGL_ALPHA_SIZE;
|
gabomdq@7659
|
474 |
attribs[i++] = _this->gl_config.alpha_size;
|
gabomdq@7659
|
475 |
}
|
gabomdq@7659
|
476 |
|
gabomdq@7659
|
477 |
if (_this->gl_config.buffer_size) {
|
gabomdq@7659
|
478 |
attribs[i++] = EGL_BUFFER_SIZE;
|
gabomdq@7659
|
479 |
attribs[i++] = _this->gl_config.buffer_size;
|
gabomdq@7659
|
480 |
}
|
gabomdq@7659
|
481 |
|
gabomdq@7659
|
482 |
attribs[i++] = EGL_DEPTH_SIZE;
|
gabomdq@7659
|
483 |
attribs[i++] = _this->gl_config.depth_size;
|
gabomdq@7659
|
484 |
|
gabomdq@7659
|
485 |
if (_this->gl_config.stencil_size) {
|
gabomdq@7659
|
486 |
attribs[i++] = EGL_STENCIL_SIZE;
|
gabomdq@7659
|
487 |
attribs[i++] = _this->gl_config.stencil_size;
|
gabomdq@7659
|
488 |
}
|
gabomdq@7659
|
489 |
|
gabomdq@7659
|
490 |
if (_this->gl_config.multisamplebuffers) {
|
gabomdq@7659
|
491 |
attribs[i++] = EGL_SAMPLE_BUFFERS;
|
gabomdq@7659
|
492 |
attribs[i++] = _this->gl_config.multisamplebuffers;
|
gabomdq@7659
|
493 |
}
|
gabomdq@7659
|
494 |
|
gabomdq@7659
|
495 |
if (_this->gl_config.multisamplesamples) {
|
gabomdq@7659
|
496 |
attribs[i++] = EGL_SAMPLES;
|
gabomdq@7659
|
497 |
attribs[i++] = _this->gl_config.multisamplesamples;
|
gabomdq@7659
|
498 |
}
|
slime73@9595
|
499 |
|
slime73@9827
|
500 |
if (_this->gl_config.framebuffer_srgb_capable) {
|
slime73@9827
|
501 |
#ifdef EGL_KHR_gl_colorspace
|
slouken@11175
|
502 |
if (SDL_EGL_HasExtension(_this, SDL_EGL_DISPLAY_EXTENSION, "EGL_KHR_gl_colorspace")) {
|
slime73@9827
|
503 |
attribs[i++] = EGL_GL_COLORSPACE_KHR;
|
slime73@9827
|
504 |
attribs[i++] = EGL_GL_COLORSPACE_SRGB_KHR;
|
slime73@9827
|
505 |
} else
|
slime73@9827
|
506 |
#endif
|
slime73@9827
|
507 |
{
|
slime73@9827
|
508 |
return SDL_SetError("EGL implementation does not support sRGB system framebuffers");
|
slime73@9827
|
509 |
}
|
slime73@9827
|
510 |
}
|
slime73@9827
|
511 |
|
gabomdq@7659
|
512 |
attribs[i++] = EGL_RENDERABLE_TYPE;
|
slime73@9595
|
513 |
if (_this->gl_config.profile_mask == SDL_GL_CONTEXT_PROFILE_ES) {
|
slime73@9595
|
514 |
#ifdef EGL_KHR_create_context
|
slime73@9595
|
515 |
if (_this->gl_config.major_version >= 3 &&
|
slouken@11175
|
516 |
SDL_EGL_HasExtension(_this, SDL_EGL_DISPLAY_EXTENSION, "EGL_KHR_create_context")) {
|
slime73@9595
|
517 |
attribs[i++] = EGL_OPENGL_ES3_BIT_KHR;
|
slime73@9595
|
518 |
} else
|
slime73@9595
|
519 |
#endif
|
slime73@9595
|
520 |
if (_this->gl_config.major_version >= 2) {
|
gabomdq@8062
|
521 |
attribs[i++] = EGL_OPENGL_ES2_BIT;
|
gabomdq@8062
|
522 |
} else {
|
gabomdq@8062
|
523 |
attribs[i++] = EGL_OPENGL_ES_BIT;
|
gabomdq@8062
|
524 |
}
|
gabomdq@8062
|
525 |
_this->egl_data->eglBindAPI(EGL_OPENGL_ES_API);
|
slime73@9595
|
526 |
} else {
|
gabomdq@8062
|
527 |
attribs[i++] = EGL_OPENGL_BIT;
|
gabomdq@8062
|
528 |
_this->egl_data->eglBindAPI(EGL_OPENGL_API);
|
gabomdq@7659
|
529 |
}
|
slime73@9827
|
530 |
|
slouken@11677
|
531 |
if (_this->egl_data->egl_surfacetype) {
|
slouken@11677
|
532 |
attribs[i++] = EGL_SURFACE_TYPE;
|
slouken@11677
|
533 |
attribs[i++] = _this->egl_data->egl_surfacetype;
|
slouken@11677
|
534 |
}
|
slouken@11677
|
535 |
|
gabomdq@7659
|
536 |
attribs[i++] = EGL_NONE;
|
slime73@9827
|
537 |
|
gabomdq@7659
|
538 |
if (_this->egl_data->eglChooseConfig(_this->egl_data->egl_display,
|
gabomdq@7659
|
539 |
attribs,
|
gabomdq@8015
|
540 |
configs, SDL_arraysize(configs),
|
gabomdq@7659
|
541 |
&found_configs) == EGL_FALSE ||
|
gabomdq@7659
|
542 |
found_configs == 0) {
|
dludwig@10740
|
543 |
return SDL_EGL_SetError("Couldn't find matching EGL config", "eglChooseConfig");
|
gabomdq@7659
|
544 |
}
|
slime73@9827
|
545 |
|
gabomdq@8015
|
546 |
/* eglChooseConfig returns a number of configurations that match or exceed the requested attribs. */
|
gabomdq@8015
|
547 |
/* From those, we select the one that matches our requirements more closely via a makeshift algorithm */
|
gabomdq@8021
|
548 |
|
slime73@9595
|
549 |
for (i = 0; i < found_configs; i++ ) {
|
gabomdq@8015
|
550 |
bitdiff = 0;
|
gabomdq@8021
|
551 |
for (j = 0; j < SDL_arraysize(attribs) - 1; j += 2) {
|
gabomdq@8015
|
552 |
if (attribs[j] == EGL_NONE) {
|
gabomdq@8015
|
553 |
break;
|
gabomdq@8015
|
554 |
}
|
gabomdq@8015
|
555 |
|
gabomdq@8015
|
556 |
if ( attribs[j+1] != EGL_DONT_CARE && (
|
gabomdq@8015
|
557 |
attribs[j] == EGL_RED_SIZE ||
|
gabomdq@8015
|
558 |
attribs[j] == EGL_GREEN_SIZE ||
|
gabomdq@8015
|
559 |
attribs[j] == EGL_BLUE_SIZE ||
|
gabomdq@8015
|
560 |
attribs[j] == EGL_ALPHA_SIZE ||
|
gabomdq@8015
|
561 |
attribs[j] == EGL_DEPTH_SIZE ||
|
gabomdq@8015
|
562 |
attribs[j] == EGL_STENCIL_SIZE)) {
|
gabomdq@8015
|
563 |
_this->egl_data->eglGetConfigAttrib(_this->egl_data->egl_display, configs[i], attribs[j], &value);
|
gabomdq@8015
|
564 |
bitdiff += value - attribs[j + 1]; /* value is always >= attrib */
|
gabomdq@8015
|
565 |
}
|
gabomdq@8015
|
566 |
}
|
gabomdq@8015
|
567 |
|
gabomdq@8015
|
568 |
if (bitdiff < best_bitdiff || best_bitdiff == -1) {
|
gabomdq@8015
|
569 |
_this->egl_data->egl_config = configs[i];
|
gabomdq@8015
|
570 |
|
gabomdq@8015
|
571 |
best_bitdiff = bitdiff;
|
gabomdq@8015
|
572 |
}
|
slime73@9595
|
573 |
|
slime73@9595
|
574 |
if (bitdiff == 0) {
|
slime73@9595
|
575 |
break; /* we found an exact match! */
|
slime73@9595
|
576 |
}
|
gabomdq@8015
|
577 |
}
|
gabomdq@8015
|
578 |
|
gabomdq@7659
|
579 |
return 0;
|
gabomdq@7659
|
580 |
}
|
gabomdq@7659
|
581 |
|
gabomdq@7659
|
582 |
SDL_GLContext
|
gabomdq@7659
|
583 |
SDL_EGL_CreateContext(_THIS, EGLSurface egl_surface)
|
gabomdq@7659
|
584 |
{
|
slime73@9595
|
585 |
/* max 14 values plus terminator. */
|
slime73@9595
|
586 |
EGLint attribs[15];
|
slime73@9595
|
587 |
int attr = 0;
|
slime73@9595
|
588 |
|
gabomdq@8061
|
589 |
EGLContext egl_context, share_context = EGL_NO_CONTEXT;
|
slime73@9595
|
590 |
EGLint profile_mask = _this->gl_config.profile_mask;
|
slime73@9637
|
591 |
EGLint major_version = _this->gl_config.major_version;
|
slime73@9637
|
592 |
EGLint minor_version = _this->gl_config.minor_version;
|
slime73@9637
|
593 |
SDL_bool profile_es = (profile_mask == SDL_GL_CONTEXT_PROFILE_ES);
|
slime73@9595
|
594 |
|
gabomdq@7659
|
595 |
if (!_this->egl_data) {
|
gabomdq@7659
|
596 |
/* The EGL library wasn't loaded, SDL_GetError() should have info */
|
gabomdq@7659
|
597 |
return NULL;
|
gabomdq@7659
|
598 |
}
|
slime73@9595
|
599 |
|
gabomdq@8061
|
600 |
if (_this->gl_config.share_with_current_context) {
|
gabomdq@8061
|
601 |
share_context = (EGLContext)SDL_GL_GetCurrentContext();
|
gabomdq@7659
|
602 |
}
|
slime73@9595
|
603 |
|
slime73@9595
|
604 |
/* Set the context version and other attributes. */
|
slime73@9637
|
605 |
if ((major_version < 3 || (minor_version == 0 && profile_es)) &&
|
slime73@9637
|
606 |
_this->gl_config.flags == 0 &&
|
slime73@9637
|
607 |
(profile_mask == 0 || profile_es)) {
|
slime73@9637
|
608 |
/* Create a context without using EGL_KHR_create_context attribs.
|
slime73@9637
|
609 |
* When creating a GLES context without EGL_KHR_create_context we can
|
slime73@9637
|
610 |
* only specify the major version. When creating a desktop GL context
|
slime73@9637
|
611 |
* we can't specify any version, so we only try in that case when the
|
slime73@9637
|
612 |
* version is less than 3.0 (matches SDL's GLX/WGL behavior.)
|
slime73@9637
|
613 |
*/
|
slime73@9637
|
614 |
if (profile_es) {
|
slime73@9595
|
615 |
attribs[attr++] = EGL_CONTEXT_CLIENT_VERSION;
|
slime73@9637
|
616 |
attribs[attr++] = SDL_max(major_version, 1);
|
slime73@9595
|
617 |
}
|
slime73@9595
|
618 |
} else {
|
slime73@9595
|
619 |
#ifdef EGL_KHR_create_context
|
slime73@9595
|
620 |
/* The Major/minor version, context profiles, and context flags can
|
slime73@9595
|
621 |
* only be specified when this extension is available.
|
slime73@9595
|
622 |
*/
|
slouken@11175
|
623 |
if (SDL_EGL_HasExtension(_this, SDL_EGL_DISPLAY_EXTENSION, "EGL_KHR_create_context")) {
|
slime73@9595
|
624 |
attribs[attr++] = EGL_CONTEXT_MAJOR_VERSION_KHR;
|
slime73@9637
|
625 |
attribs[attr++] = major_version;
|
slime73@9595
|
626 |
attribs[attr++] = EGL_CONTEXT_MINOR_VERSION_KHR;
|
slime73@9637
|
627 |
attribs[attr++] = minor_version;
|
slime73@9595
|
628 |
|
slime73@9595
|
629 |
/* SDL profile bits match EGL profile bits. */
|
slime73@9595
|
630 |
if (profile_mask != 0 && profile_mask != SDL_GL_CONTEXT_PROFILE_ES) {
|
slime73@9595
|
631 |
attribs[attr++] = EGL_CONTEXT_OPENGL_PROFILE_MASK_KHR;
|
slime73@9595
|
632 |
attribs[attr++] = profile_mask;
|
slime73@9595
|
633 |
}
|
slime73@9595
|
634 |
|
slime73@9595
|
635 |
/* SDL flags match EGL flags. */
|
slime73@9595
|
636 |
if (_this->gl_config.flags != 0) {
|
slime73@9595
|
637 |
attribs[attr++] = EGL_CONTEXT_FLAGS_KHR;
|
slime73@9595
|
638 |
attribs[attr++] = _this->gl_config.flags;
|
slime73@9595
|
639 |
}
|
slime73@9595
|
640 |
} else
|
slime73@9595
|
641 |
#endif /* EGL_KHR_create_context */
|
slime73@9595
|
642 |
{
|
slime73@9595
|
643 |
SDL_SetError("Could not create EGL context (context attributes are not supported)");
|
slime73@9595
|
644 |
return NULL;
|
slime73@9595
|
645 |
}
|
slime73@9595
|
646 |
}
|
slime73@9595
|
647 |
|
flibitijibibo@11347
|
648 |
if (_this->gl_config.no_error) {
|
flibitijibibo@11347
|
649 |
#ifdef EGL_KHR_create_context_no_error
|
flibitijibibo@11347
|
650 |
if (SDL_EGL_HasExtension(_this, SDL_EGL_DISPLAY_EXTENSION, "EGL_KHR_create_context_no_error")) {
|
flibitijibibo@11347
|
651 |
attribs[attr++] = EGL_CONTEXT_OPENGL_NO_ERROR_KHR;
|
flibitijibibo@11347
|
652 |
attribs[attr++] = _this->gl_config.no_error;
|
flibitijibibo@11347
|
653 |
} else
|
flibitijibibo@11347
|
654 |
#endif
|
flibitijibibo@11347
|
655 |
{
|
flibitijibibo@11347
|
656 |
SDL_SetError("EGL implementation does not support no_error contexts");
|
flibitijibibo@11347
|
657 |
return NULL;
|
flibitijibibo@11347
|
658 |
}
|
flibitijibibo@11347
|
659 |
}
|
flibitijibibo@11347
|
660 |
|
slime73@9595
|
661 |
attribs[attr++] = EGL_NONE;
|
slime73@9595
|
662 |
|
gabomdq@8062
|
663 |
/* Bind the API */
|
slime73@9637
|
664 |
if (profile_es) {
|
gabomdq@8062
|
665 |
_this->egl_data->eglBindAPI(EGL_OPENGL_ES_API);
|
slime73@9595
|
666 |
} else {
|
slime73@9595
|
667 |
_this->egl_data->eglBindAPI(EGL_OPENGL_API);
|
slime73@9595
|
668 |
}
|
gabomdq@7659
|
669 |
|
slime73@9595
|
670 |
egl_context = _this->egl_data->eglCreateContext(_this->egl_data->egl_display,
|
slime73@9595
|
671 |
_this->egl_data->egl_config,
|
slime73@9595
|
672 |
share_context, attribs);
|
knut@8962
|
673 |
|
gabomdq@7659
|
674 |
if (egl_context == EGL_NO_CONTEXT) {
|
dludwig@10740
|
675 |
SDL_EGL_SetError("Could not create EGL context", "eglCreateContext");
|
gabomdq@7659
|
676 |
return NULL;
|
gabomdq@7659
|
677 |
}
|
slime73@9595
|
678 |
|
gabomdq@7659
|
679 |
_this->egl_data->egl_swapinterval = 0;
|
slime73@9595
|
680 |
|
gabomdq@7659
|
681 |
if (SDL_EGL_MakeCurrent(_this, egl_surface, egl_context) < 0) {
|
dludwig@10740
|
682 |
/* Save the SDL error set by SDL_EGL_MakeCurrent */
|
dludwig@10740
|
683 |
char errorText[1024];
|
dludwig@10740
|
684 |
SDL_strlcpy(errorText, SDL_GetError(), SDL_arraysize(errorText));
|
dludwig@10740
|
685 |
|
dludwig@10740
|
686 |
/* Delete the context, which may alter the value returned by SDL_GetError() */
|
gabomdq@7659
|
687 |
SDL_EGL_DeleteContext(_this, egl_context);
|
dludwig@10740
|
688 |
|
dludwig@10740
|
689 |
/* Restore the SDL error */
|
dludwig@10740
|
690 |
SDL_SetError("%s", errorText);
|
dludwig@10740
|
691 |
|
gabomdq@7659
|
692 |
return NULL;
|
gabomdq@7659
|
693 |
}
|
slime73@9595
|
694 |
|
gabomdq@7659
|
695 |
return (SDL_GLContext) egl_context;
|
gabomdq@7659
|
696 |
}
|
gabomdq@7659
|
697 |
|
gabomdq@7659
|
698 |
int
|
gabomdq@7659
|
699 |
SDL_EGL_MakeCurrent(_THIS, EGLSurface egl_surface, SDL_GLContext context)
|
gabomdq@7659
|
700 |
{
|
gabomdq@7679
|
701 |
EGLContext egl_context = (EGLContext) context;
|
gabomdq@7679
|
702 |
|
gabomdq@7659
|
703 |
if (!_this->egl_data) {
|
gabomdq@7659
|
704 |
return SDL_SetError("OpenGL not initialized");
|
gabomdq@7659
|
705 |
}
|
gabomdq@7659
|
706 |
|
gabomdq@7659
|
707 |
/* The android emulator crashes badly if you try to eglMakeCurrent
|
gabomdq@7659
|
708 |
* with a valid context and invalid surface, so we have to check for both here.
|
gabomdq@7659
|
709 |
*/
|
gabomdq@7659
|
710 |
if (!egl_context || !egl_surface) {
|
gabomdq@7659
|
711 |
_this->egl_data->eglMakeCurrent(_this->egl_data->egl_display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
slime73@9595
|
712 |
} else {
|
gabomdq@7659
|
713 |
if (!_this->egl_data->eglMakeCurrent(_this->egl_data->egl_display,
|
gabomdq@7659
|
714 |
egl_surface, egl_surface, egl_context)) {
|
dludwig@10740
|
715 |
return SDL_EGL_SetError("Unable to make EGL context current", "eglMakeCurrent");
|
gabomdq@7659
|
716 |
}
|
gabomdq@7659
|
717 |
}
|
gabomdq@7659
|
718 |
|
gabomdq@7659
|
719 |
return 0;
|
gabomdq@7659
|
720 |
}
|
gabomdq@7659
|
721 |
|
gabomdq@7659
|
722 |
int
|
gabomdq@7659
|
723 |
SDL_EGL_SetSwapInterval(_THIS, int interval)
|
gabomdq@7659
|
724 |
{
|
gabomdq@7679
|
725 |
EGLBoolean status;
|
gabomdq@7679
|
726 |
|
gabomdq@7702
|
727 |
if (!_this->egl_data) {
|
gabomdq@7702
|
728 |
return SDL_SetError("EGL not initialized");
|
gabomdq@7659
|
729 |
}
|
gabomdq@7659
|
730 |
|
gabomdq@7659
|
731 |
status = _this->egl_data->eglSwapInterval(_this->egl_data->egl_display, interval);
|
gabomdq@7659
|
732 |
if (status == EGL_TRUE) {
|
gabomdq@7659
|
733 |
_this->egl_data->egl_swapinterval = interval;
|
gabomdq@7659
|
734 |
return 0;
|
gabomdq@7659
|
735 |
}
|
gabomdq@7659
|
736 |
|
dludwig@10740
|
737 |
return SDL_EGL_SetError("Unable to set the EGL swap interval", "eglSwapInterval");
|
gabomdq@7659
|
738 |
}
|
gabomdq@7659
|
739 |
|
gabomdq@7659
|
740 |
int
|
gabomdq@7659
|
741 |
SDL_EGL_GetSwapInterval(_THIS)
|
gabomdq@7659
|
742 |
{
|
gabomdq@7702
|
743 |
if (!_this->egl_data) {
|
philipp@10169
|
744 |
SDL_SetError("EGL not initialized");
|
philipp@10169
|
745 |
return 0;
|
gabomdq@7659
|
746 |
}
|
gabomdq@7659
|
747 |
|
gabomdq@7659
|
748 |
return _this->egl_data->egl_swapinterval;
|
gabomdq@7659
|
749 |
}
|
gabomdq@7659
|
750 |
|
slouken@10693
|
751 |
int
|
gabomdq@7659
|
752 |
SDL_EGL_SwapBuffers(_THIS, EGLSurface egl_surface)
|
gabomdq@7659
|
753 |
{
|
slouken@10693
|
754 |
if (!_this->egl_data->eglSwapBuffers(_this->egl_data->egl_display, egl_surface)) {
|
dludwig@10740
|
755 |
return SDL_EGL_SetError("unable to show color buffer in an OS-native window", "eglSwapBuffers");
|
slouken@10693
|
756 |
}
|
slouken@10693
|
757 |
return 0;
|
gabomdq@7659
|
758 |
}
|
gabomdq@7659
|
759 |
|
gabomdq@7659
|
760 |
void
|
gabomdq@7659
|
761 |
SDL_EGL_DeleteContext(_THIS, SDL_GLContext context)
|
gabomdq@7659
|
762 |
{
|
gabomdq@7679
|
763 |
EGLContext egl_context = (EGLContext) context;
|
gabomdq@7679
|
764 |
|
gabomdq@7659
|
765 |
/* Clean up GLES and EGL */
|
gabomdq@7659
|
766 |
if (!_this->egl_data) {
|
gabomdq@7659
|
767 |
return;
|
gabomdq@7659
|
768 |
}
|
gabomdq@7659
|
769 |
|
gabomdq@7896
|
770 |
if (egl_context != NULL && egl_context != EGL_NO_CONTEXT) {
|
gabomdq@7659
|
771 |
SDL_EGL_MakeCurrent(_this, NULL, NULL);
|
gabomdq@7659
|
772 |
_this->egl_data->eglDestroyContext(_this->egl_data->egl_display, egl_context);
|
gabomdq@7659
|
773 |
}
|
gabomdq@7659
|
774 |
|
gabomdq@7659
|
775 |
}
|
gabomdq@7659
|
776 |
|
gabomdq@7659
|
777 |
EGLSurface *
|
gabomdq@7659
|
778 |
SDL_EGL_CreateSurface(_THIS, NativeWindowType nw)
|
gabomdq@7659
|
779 |
{
|
dludwig@10740
|
780 |
EGLSurface * surface;
|
dludwig@10740
|
781 |
|
gabomdq@8041
|
782 |
if (SDL_EGL_ChooseConfig(_this) != 0) {
|
slouken@8045
|
783 |
return EGL_NO_SURFACE;
|
gabomdq@8041
|
784 |
}
|
gabomdq@8041
|
785 |
|
slouken@10878
|
786 |
#if SDL_VIDEO_DRIVER_ANDROID
|
slouken@8976
|
787 |
{
|
slouken@8976
|
788 |
/* Android docs recommend doing this!
|
slouken@8976
|
789 |
* Ref: http://developer.android.com/reference/android/app/NativeActivity.html
|
slouken@8976
|
790 |
*/
|
slouken@8976
|
791 |
EGLint format;
|
slouken@8976
|
792 |
_this->egl_data->eglGetConfigAttrib(_this->egl_data->egl_display,
|
slouken@8976
|
793 |
_this->egl_data->egl_config,
|
slouken@8976
|
794 |
EGL_NATIVE_VISUAL_ID, &format);
|
gabomdq@8971
|
795 |
|
slouken@8976
|
796 |
ANativeWindow_setBuffersGeometry(nw, 0, 0, format);
|
slouken@8976
|
797 |
}
|
gabomdq@8971
|
798 |
#endif
|
gabomdq@8971
|
799 |
|
dludwig@10740
|
800 |
surface = _this->egl_data->eglCreateWindowSurface(
|
gabomdq@7659
|
801 |
_this->egl_data->egl_display,
|
gabomdq@7659
|
802 |
_this->egl_data->egl_config,
|
gabomdq@7659
|
803 |
nw, NULL);
|
dludwig@10740
|
804 |
if (surface == EGL_NO_SURFACE) {
|
dludwig@10740
|
805 |
SDL_EGL_SetError("unable to create an EGL window surface", "eglCreateWindowSurface");
|
dludwig@10740
|
806 |
}
|
dludwig@10740
|
807 |
return surface;
|
gabomdq@7659
|
808 |
}
|
gabomdq@7659
|
809 |
|
gabomdq@7659
|
810 |
void
|
gabomdq@7659
|
811 |
SDL_EGL_DestroySurface(_THIS, EGLSurface egl_surface)
|
gabomdq@7659
|
812 |
{
|
gabomdq@7659
|
813 |
if (!_this->egl_data) {
|
gabomdq@7659
|
814 |
return;
|
gabomdq@7659
|
815 |
}
|
gabomdq@7659
|
816 |
|
gabomdq@7659
|
817 |
if (egl_surface != EGL_NO_SURFACE) {
|
gabomdq@7659
|
818 |
_this->egl_data->eglDestroySurface(_this->egl_data->egl_display, egl_surface);
|
gabomdq@7659
|
819 |
}
|
gabomdq@7659
|
820 |
}
|
gabomdq@7659
|
821 |
|
gabomdq@7659
|
822 |
#endif /* SDL_VIDEO_OPENGL_EGL */
|
gabomdq@7659
|
823 |
|
gabomdq@7659
|
824 |
/* vi: set ts=4 sw=4 expandtab: */
|
philipp@11192
|
825 |
|