egl_context.c 23.2 KB
Newer Older
Cloudef's avatar
Cloudef committed
1
//========================================================================
Camilla Berglund's avatar
Camilla Berglund committed
2
// GLFW 3.3 EGL - www.glfw.org
Cloudef's avatar
Cloudef committed
3
4
//------------------------------------------------------------------------
// Copyright (c) 2002-2006 Marcus Geelnard
Camilla Löwy's avatar
Camilla Löwy committed
5
// Copyright (c) 2006-2016 Camilla Löwy <elmindreda@glfw.org>
Cloudef's avatar
Cloudef committed
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
//
// This software is provided 'as-is', without any express or implied
// warranty. In no event will the authors be held liable for any damages
// arising from the use of this software.
//
// Permission is granted to anyone to use this software for any purpose,
// including commercial applications, and to alter it and redistribute it
// freely, subject to the following restrictions:
//
// 1. The origin of this software must not be misrepresented; you must not
//    claim that you wrote the original software. If you use this software
//    in a product, an acknowledgment in the product documentation would
//    be appreciated but is not required.
//
// 2. Altered source versions must be plainly marked as such, and must not
//    be misrepresented as being the original software.
//
// 3. This notice may not be removed or altered from any source
//    distribution.
//
//========================================================================

#include "internal.h"

#include <stdio.h>
Camilla Berglund's avatar
Camilla Berglund committed
31
#include <string.h>
Cloudef's avatar
Cloudef committed
32
#include <stdlib.h>
33
#include <assert.h>
Cloudef's avatar
Cloudef committed
34

35

36
// Return a description of the specified EGL error
37
//
38
static const char* getEGLErrorString(EGLint error)
39
40
41
42
43
44
45
46
47
48
49
50
{
    switch (error)
    {
        case EGL_SUCCESS:
            return "Success";
        case EGL_NOT_INITIALIZED:
            return "EGL is not or could not be initialized";
        case EGL_BAD_ACCESS:
            return "EGL cannot access a requested resource";
        case EGL_BAD_ALLOC:
            return "EGL failed to allocate resources for the requested operation";
        case EGL_BAD_ATTRIBUTE:
Camilla Berglund's avatar
Camilla Berglund committed
51
            return "An unrecognized attribute or attribute value was passed in the attribute list";
52
        case EGL_BAD_CONTEXT:
Camilla Berglund's avatar
Camilla Berglund committed
53
            return "An EGLContext argument does not name a valid EGL rendering context";
54
        case EGL_BAD_CONFIG:
Camilla Berglund's avatar
Camilla Berglund committed
55
            return "An EGLConfig argument does not name a valid EGL frame buffer configuration";
56
        case EGL_BAD_CURRENT_SURFACE:
Camilla Berglund's avatar
Camilla Berglund committed
57
            return "The current surface of the calling thread is a window, pixel buffer or pixmap that is no longer valid";
58
        case EGL_BAD_DISPLAY:
Camilla Berglund's avatar
Camilla Berglund committed
59
            return "An EGLDisplay argument does not name a valid EGL display connection";
60
        case EGL_BAD_SURFACE:
Camilla Berglund's avatar
Camilla Berglund committed
61
            return "An EGLSurface argument does not name a valid surface configured for GL rendering";
62
63
64
65
66
        case EGL_BAD_MATCH:
            return "Arguments are inconsistent";
        case EGL_BAD_PARAMETER:
            return "One or more argument values are invalid";
        case EGL_BAD_NATIVE_PIXMAP:
Camilla Berglund's avatar
Camilla Berglund committed
67
            return "A NativePixmapType argument does not refer to a valid native pixmap";
68
        case EGL_BAD_NATIVE_WINDOW:
Camilla Berglund's avatar
Camilla Berglund committed
69
            return "A NativeWindowType argument does not refer to a valid native window";
70
71
        case EGL_CONTEXT_LOST:
            return "The application must destroy all contexts and reinitialise";
Camilla Berglund's avatar
Cleanup    
Camilla Berglund committed
72
73
        default:
            return "ERROR: UNKNOWN EGL ERROR";
74
75
76
    }
}

77
78
// Returns the specified attribute of the specified EGLConfig
//
79
static int getEGLConfigAttrib(EGLConfig config, int attrib)
80
81
{
    int value;
82
    eglGetConfigAttrib(_glfw.egl.display, config, attrib, &value);
83
84
85
    return value;
}

86
// Return the EGLConfig most closely matching the specified hints
87
//
88
static GLFWbool chooseEGLConfig(const _GLFWctxconfig* ctxconfig,
89
90
                                const _GLFWfbconfig* desired,
                                EGLConfig* result)
91
92
93
94
95
96
{
    EGLConfig* nativeConfigs;
    _GLFWfbconfig* usableConfigs;
    const _GLFWfbconfig* closest;
    int i, nativeCount, usableCount;

97
    eglGetConfigs(_glfw.egl.display, NULL, 0, &nativeCount);
98
99
100
    if (!nativeCount)
    {
        _glfwInputError(GLFW_API_UNAVAILABLE, "EGL: No EGLConfigs returned");
101
        return GLFW_FALSE;
102
103
    }

104
    nativeConfigs = calloc(nativeCount, sizeof(EGLConfig));
105
    eglGetConfigs(_glfw.egl.display, nativeConfigs, nativeCount, &nativeCount);
106

107
    usableConfigs = calloc(nativeCount, sizeof(_GLFWfbconfig));
108
109
110
111
112
113
114
    usableCount = 0;

    for (i = 0;  i < nativeCount;  i++)
    {
        const EGLConfig n = nativeConfigs[i];
        _GLFWfbconfig* u = usableConfigs + usableCount;

Camilla Berglund's avatar
Camilla Berglund committed
115
        // Only consider RGB(A) EGLConfigs
116
        if (getEGLConfigAttrib(n, EGL_COLOR_BUFFER_TYPE) != EGL_RGB_BUFFER)
117
118
            continue;

Camilla Berglund's avatar
Camilla Berglund committed
119
        // Only consider window EGLConfigs
120
        if (!(getEGLConfigAttrib(n, EGL_SURFACE_TYPE) & EGL_WINDOW_BIT))
121
122
            continue;

123
124
#if defined(_GLFW_X11)
        // Only consider EGLConfigs with associated Visuals
125
        if (!getEGLConfigAttrib(n, EGL_NATIVE_VISUAL_ID))
126
127
128
            continue;
#endif // _GLFW_X11

129
        if (ctxconfig->client == GLFW_OPENGL_ES_API)
130
        {
131
            if (ctxconfig->major == 1)
132
            {
133
                if (!(getEGLConfigAttrib(n, EGL_RENDERABLE_TYPE) & EGL_OPENGL_ES_BIT))
134
135
136
137
                    continue;
            }
            else
            {
138
                if (!(getEGLConfigAttrib(n, EGL_RENDERABLE_TYPE) & EGL_OPENGL_ES2_BIT))
139
140
141
                    continue;
            }
        }
142
        else if (ctxconfig->client == GLFW_OPENGL_API)
143
        {
144
            if (!(getEGLConfigAttrib(n, EGL_RENDERABLE_TYPE) & EGL_OPENGL_BIT))
145
146
147
                continue;
        }

148
149
150
        u->redBits = getEGLConfigAttrib(n, EGL_RED_SIZE);
        u->greenBits = getEGLConfigAttrib(n, EGL_GREEN_SIZE);
        u->blueBits = getEGLConfigAttrib(n, EGL_BLUE_SIZE);
151

152
153
154
        u->alphaBits = getEGLConfigAttrib(n, EGL_ALPHA_SIZE);
        u->depthBits = getEGLConfigAttrib(n, EGL_DEPTH_SIZE);
        u->stencilBits = getEGLConfigAttrib(n, EGL_STENCIL_SIZE);
155

156
        u->samples = getEGLConfigAttrib(n, EGL_SAMPLES);
157
        u->doublebuffer = GLFW_TRUE;
158

Camilla Berglund's avatar
Cleanup    
Camilla Berglund committed
159
        u->handle = (uintptr_t) n;
160
161
162
163
164
        usableCount++;
    }

    closest = _glfwChooseFBConfig(desired, usableConfigs, usableCount);
    if (closest)
Camilla Berglund's avatar
Cleanup    
Camilla Berglund committed
165
        *result = (EGLConfig) closest->handle;
166
167
168
169

    free(nativeConfigs);
    free(usableConfigs);

Camilla Berglund's avatar
Cleanup    
Camilla Berglund committed
170
    return closest != NULL;
171
172
}

173
static void makeContextCurrentEGL(_GLFWwindow* window)
174
175
176
177
178
179
180
181
182
183
{
    if (window)
    {
        if (!eglMakeCurrent(_glfw.egl.display,
                            window->context.egl.surface,
                            window->context.egl.surface,
                            window->context.egl.handle))
        {
            _glfwInputError(GLFW_PLATFORM_ERROR,
                            "EGL: Failed to make context current: %s",
184
                            getEGLErrorString(eglGetError()));
185
186
187
188
189
190
191
192
193
194
195
196
            return;
        }
    }
    else
    {
        if (!eglMakeCurrent(_glfw.egl.display,
                            EGL_NO_SURFACE,
                            EGL_NO_SURFACE,
                            EGL_NO_CONTEXT))
        {
            _glfwInputError(GLFW_PLATFORM_ERROR,
                            "EGL: Failed to clear current context: %s",
197
                            getEGLErrorString(eglGetError()));
198
199
200
201
            return;
        }
    }

202
    _glfwPlatformSetTls(&_glfw.contextSlot, window);
203
204
}

205
static void swapBuffersEGL(_GLFWwindow* window)
206
{
207
    if (window != _glfwPlatformGetTls(&_glfw.contextSlot))
208
209
210
211
212
213
214
215
216
    {
        _glfwInputError(GLFW_PLATFORM_ERROR,
                        "EGL: The context must be current on the calling thread when swapping buffers");
        return;
    }

    eglSwapBuffers(_glfw.egl.display, window->context.egl.surface);
}

217
static void swapIntervalEGL(int interval)
218
219
220
221
{
    eglSwapInterval(_glfw.egl.display, interval);
}

222
static int extensionSupportedEGL(const char* extension)
223
224
225
226
227
228
229
230
231
232
233
{
    const char* extensions = eglQueryString(_glfw.egl.display, EGL_EXTENSIONS);
    if (extensions)
    {
        if (_glfwStringInExtensionString(extension, extensions))
            return GLFW_TRUE;
    }

    return GLFW_FALSE;
}

234
static GLFWglproc getProcAddressEGL(const char* procname)
235
{
236
    _GLFWwindow* window = _glfwPlatformGetTls(&_glfw.contextSlot);
237
238
239
240
241
242
243
244
245
246
247
248

    if (window->context.egl.client)
    {
        GLFWglproc proc = (GLFWglproc) _glfw_dlsym(window->context.egl.client,
                                                   procname);
        if (proc)
            return proc;
    }

    return eglGetProcAddress(procname);
}

249
static void destroyContextEGL(_GLFWwindow* window)
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
{
#if defined(_GLFW_X11)
    // NOTE: Do not unload libGL.so.1 while the X11 display is still open,
    //       as it will make XCloseDisplay segfault
    if (window->context.client != GLFW_OPENGL_API)
#endif // _GLFW_X11
    {
        if (window->context.egl.client)
        {
            _glfw_dlclose(window->context.egl.client);
            window->context.egl.client = NULL;
        }
    }

    if (window->context.egl.surface)
    {
        eglDestroySurface(_glfw.egl.display, window->context.egl.surface);
        window->context.egl.surface = EGL_NO_SURFACE;
    }

    if (window->context.egl.handle)
    {
        eglDestroyContext(_glfw.egl.display, window->context.egl.handle);
        window->context.egl.handle = EGL_NO_CONTEXT;
    }
}

277

278
279
280
//////////////////////////////////////////////////////////////////////////
//////                       GLFW internal API                      //////
//////////////////////////////////////////////////////////////////////////
Cloudef's avatar
Cloudef committed
281

282
// Initialize EGL
283
//
Camilla Berglund's avatar
Camilla Berglund committed
284
GLFWbool _glfwInitEGL(void)
Cloudef's avatar
Cloudef committed
285
{
286
287
288
289
290
291
292
293
    int i;
    const char* sonames[] =
    {
#if defined(_GLFW_WIN32)
        "libEGL.dll",
        "EGL.dll",
#elif defined(_GLFW_COCOA)
        "libEGL.dylib",
Camilla Löwy's avatar
Camilla Löwy committed
294
295
#elif defined(__CYGWIN__)
        "libEGL-1.so",
296
297
298
299
300
301
#else
        "libEGL.so.1",
#endif
        NULL
    };

302
303
304
    if (_glfw.egl.handle)
        return GLFW_TRUE;

305
306
307
308
309
310
311
312
    for (i = 0;  sonames[i];  i++)
    {
        _glfw.egl.handle = _glfw_dlopen(sonames[i]);
        if (_glfw.egl.handle)
            break;
    }

    if (!_glfw.egl.handle)
313
314
    {
        _glfwInputError(GLFW_API_UNAVAILABLE, "EGL: Library not found");
315
        return GLFW_FALSE;
316
    }
317

318
319
    _glfw.egl.prefix = (strncmp(sonames[i], "lib", 3) == 0);

320
    _glfw.egl.GetConfigAttrib = (PFN_eglGetConfigAttrib)
321
        _glfw_dlsym(_glfw.egl.handle, "eglGetConfigAttrib");
322
    _glfw.egl.GetConfigs = (PFN_eglGetConfigs)
323
        _glfw_dlsym(_glfw.egl.handle, "eglGetConfigs");
324
    _glfw.egl.GetDisplay = (PFN_eglGetDisplay)
325
        _glfw_dlsym(_glfw.egl.handle, "eglGetDisplay");
326
    _glfw.egl.GetError = (PFN_eglGetError)
327
        _glfw_dlsym(_glfw.egl.handle, "eglGetError");
328
    _glfw.egl.Initialize = (PFN_eglInitialize)
329
        _glfw_dlsym(_glfw.egl.handle, "eglInitialize");
330
    _glfw.egl.Terminate = (PFN_eglTerminate)
331
        _glfw_dlsym(_glfw.egl.handle, "eglTerminate");
332
    _glfw.egl.BindAPI = (PFN_eglBindAPI)
333
        _glfw_dlsym(_glfw.egl.handle, "eglBindAPI");
334
    _glfw.egl.CreateContext = (PFN_eglCreateContext)
335
        _glfw_dlsym(_glfw.egl.handle, "eglCreateContext");
336
    _glfw.egl.DestroySurface = (PFN_eglDestroySurface)
337
        _glfw_dlsym(_glfw.egl.handle, "eglDestroySurface");
338
    _glfw.egl.DestroyContext = (PFN_eglDestroyContext)
339
        _glfw_dlsym(_glfw.egl.handle, "eglDestroyContext");
340
    _glfw.egl.CreateWindowSurface = (PFN_eglCreateWindowSurface)
341
        _glfw_dlsym(_glfw.egl.handle, "eglCreateWindowSurface");
342
    _glfw.egl.MakeCurrent = (PFN_eglMakeCurrent)
343
        _glfw_dlsym(_glfw.egl.handle, "eglMakeCurrent");
344
    _glfw.egl.SwapBuffers = (PFN_eglSwapBuffers)
345
        _glfw_dlsym(_glfw.egl.handle, "eglSwapBuffers");
346
    _glfw.egl.SwapInterval = (PFN_eglSwapInterval)
347
        _glfw_dlsym(_glfw.egl.handle, "eglSwapInterval");
348
    _glfw.egl.QueryString = (PFN_eglQueryString)
349
        _glfw_dlsym(_glfw.egl.handle, "eglQueryString");
350
    _glfw.egl.GetProcAddress = (PFN_eglGetProcAddress)
351
352
        _glfw_dlsym(_glfw.egl.handle, "eglGetProcAddress");

353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
    if (!_glfw.egl.GetConfigAttrib ||
        !_glfw.egl.GetConfigs ||
        !_glfw.egl.GetDisplay ||
        !_glfw.egl.GetError ||
        !_glfw.egl.Initialize ||
        !_glfw.egl.Terminate ||
        !_glfw.egl.BindAPI ||
        !_glfw.egl.CreateContext ||
        !_glfw.egl.DestroySurface ||
        !_glfw.egl.DestroyContext ||
        !_glfw.egl.CreateWindowSurface ||
        !_glfw.egl.MakeCurrent ||
        !_glfw.egl.SwapBuffers ||
        !_glfw.egl.SwapInterval ||
        !_glfw.egl.QueryString ||
        !_glfw.egl.GetProcAddress)
    {
        _glfwInputError(GLFW_PLATFORM_ERROR,
                        "EGL: Failed to load required entry points");
372
373

        _glfwTerminateEGL();
374
375
376
        return GLFW_FALSE;
    }

377
    _glfw.egl.display = eglGetDisplay(_GLFW_EGL_NATIVE_DISPLAY);
378
    if (_glfw.egl.display == EGL_NO_DISPLAY)
Cloudef's avatar
Cloudef committed
379
    {
380
381
        _glfwInputError(GLFW_API_UNAVAILABLE,
                        "EGL: Failed to get EGL display: %s",
382
                        getEGLErrorString(eglGetError()));
383
384

        _glfwTerminateEGL();
385
        return GLFW_FALSE;
Cloudef's avatar
Cloudef committed
386
387
    }

388
    if (!eglInitialize(_glfw.egl.display, &_glfw.egl.major, &_glfw.egl.minor))
Cloudef's avatar
Cloudef committed
389
    {
390
391
        _glfwInputError(GLFW_API_UNAVAILABLE,
                        "EGL: Failed to initialize EGL: %s",
392
                        getEGLErrorString(eglGetError()));
393
394

        _glfwTerminateEGL();
395
        return GLFW_FALSE;
Cloudef's avatar
Cloudef committed
396
397
    }

Camilla Berglund's avatar
Camilla Berglund committed
398
    _glfw.egl.KHR_create_context =
399
        extensionSupportedEGL("EGL_KHR_create_context");
400
    _glfw.egl.KHR_create_context_no_error =
401
        extensionSupportedEGL("EGL_KHR_create_context_no_error");
402
    _glfw.egl.KHR_gl_colorspace =
403
        extensionSupportedEGL("EGL_KHR_gl_colorspace");
404
405
    _glfw.egl.KHR_get_all_proc_addresses =
        extensionSupportedEGL("EGL_KHR_get_all_proc_addresses");
406
407
    _glfw.egl.KHR_context_flush_control =
        extensionSupportedEGL("EGL_KHR_context_flush_control");
Cloudef's avatar
Cloudef committed
408

409
    return GLFW_TRUE;
410
}
411

412
// Terminate EGL
413
//
414
void _glfwTerminateEGL(void)
415
{
416
    if (_glfw.egl.display)
417
    {
418
        eglTerminate(_glfw.egl.display);
419
420
421
422
423
        _glfw.egl.display = EGL_NO_DISPLAY;
    }

    if (_glfw.egl.handle)
    {
424
425
426
        _glfw_dlclose(_glfw.egl.handle);
        _glfw.egl.handle = NULL;
    }
Cloudef's avatar
Cloudef committed
427
428
}

Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
429
#define setAttrib(a, v) \
430
{ \
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
431
432
433
    assert((size_t) (index + 1) < sizeof(attribs) / sizeof(attribs[0])); \
    attribs[index++] = a; \
    attribs[index++] = v; \
434
}
Cloudef's avatar
Cloudef committed
435

436
// Create the OpenGL or OpenGL ES context
437
//
Camilla Berglund's avatar
Camilla Berglund committed
438
439
440
GLFWbool _glfwCreateContextEGL(_GLFWwindow* window,
                               const _GLFWctxconfig* ctxconfig,
                               const _GLFWfbconfig* fbconfig)
Cloudef's avatar
Cloudef committed
441
{
442
    EGLint attribs[40];
443
    EGLConfig config;
Cloudef's avatar
Cloudef committed
444
    EGLContext share = NULL;
445
    int index = 0;
Cloudef's avatar
Cloudef committed
446

447
448
449
450
451
452
    if (!_glfw.egl.display)
    {
        _glfwInputError(GLFW_API_UNAVAILABLE, "EGL: API not available");
        return GLFW_FALSE;
    }

453
    if (ctxconfig->share)
454
        share = ctxconfig->share->context.egl.handle;
Cloudef's avatar
Cloudef committed
455

456
    if (!chooseEGLConfig(ctxconfig, fbconfig, &config))
Cloudef's avatar
Cloudef committed
457
    {
458
        _glfwInputError(GLFW_FORMAT_UNAVAILABLE,
459
                        "EGL: Failed to find a suitable EGLConfig");
460
        return GLFW_FALSE;
Cloudef's avatar
Cloudef committed
461
462
    }

463
    if (ctxconfig->client == GLFW_OPENGL_ES_API)
464
    {
465
        if (!eglBindAPI(EGL_OPENGL_ES_API))
466
        {
Camilla Berglund's avatar
Camilla Berglund committed
467
            _glfwInputError(GLFW_API_UNAVAILABLE,
468
                            "EGL: Failed to bind OpenGL ES: %s",
469
                            getEGLErrorString(eglGetError()));
470
            return GLFW_FALSE;
471
472
        }
    }
Cloudef's avatar
Cloudef committed
473
    else
474
    {
475
        if (!eglBindAPI(EGL_OPENGL_API))
476
        {
Camilla Berglund's avatar
Camilla Berglund committed
477
            _glfwInputError(GLFW_API_UNAVAILABLE,
478
                            "EGL: Failed to bind OpenGL: %s",
479
                            getEGLErrorString(eglGetError()));
480
            return GLFW_FALSE;
481
482
        }
    }
Cloudef's avatar
Cloudef committed
483

484
    if (_glfw.egl.KHR_create_context)
485
    {
486
        int mask = 0, flags = 0;
487

488
        if (ctxconfig->client == GLFW_OPENGL_API)
489
        {
490
491
492
            if (ctxconfig->forward)
                flags |= EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR;

493
            if (ctxconfig->profile == GLFW_OPENGL_CORE_PROFILE)
494
                mask |= EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT_KHR;
495
            else if (ctxconfig->profile == GLFW_OPENGL_COMPAT_PROFILE)
496
                mask |= EGL_CONTEXT_OPENGL_COMPATIBILITY_PROFILE_BIT_KHR;
497
        }
498

499
500
501
        if (ctxconfig->debug)
            flags |= EGL_CONTEXT_OPENGL_DEBUG_BIT_KHR;

502
        if (ctxconfig->robustness)
503
        {
504
            if (ctxconfig->robustness == GLFW_NO_RESET_NOTIFICATION)
505
            {
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
506
507
                setAttrib(EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_KHR,
                          EGL_NO_RESET_NOTIFICATION_KHR);
508
            }
509
            else if (ctxconfig->robustness == GLFW_LOSE_CONTEXT_ON_RESET)
510
            {
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
511
512
                setAttrib(EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_KHR,
                          EGL_LOSE_CONTEXT_ON_RESET_KHR);
513
            }
514

515
516
517
            flags |= EGL_CONTEXT_OPENGL_ROBUST_ACCESS_BIT_KHR;
        }

518
519
520
        if (ctxconfig->noerror)
        {
            if (_glfw.egl.KHR_create_context_no_error)
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
521
                setAttrib(EGL_CONTEXT_OPENGL_NO_ERROR_KHR, GLFW_TRUE);
522
523
        }

524
        if (ctxconfig->major != 1 || ctxconfig->minor != 0)
525
        {
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
526
527
            setAttrib(EGL_CONTEXT_MAJOR_VERSION_KHR, ctxconfig->major);
            setAttrib(EGL_CONTEXT_MINOR_VERSION_KHR, ctxconfig->minor);
528
        }
529
530

        if (mask)
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
531
            setAttrib(EGL_CONTEXT_OPENGL_PROFILE_MASK_KHR, mask);
532
533

        if (flags)
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
534
            setAttrib(EGL_CONTEXT_FLAGS_KHR, flags);
535
    }
536
537
    else
    {
538
        if (ctxconfig->client == GLFW_OPENGL_ES_API)
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
539
            setAttrib(EGL_CONTEXT_CLIENT_VERSION, ctxconfig->major);
540
    }
541

542
543
544
545
    if (_glfw.egl.KHR_context_flush_control)
    {
        if (ctxconfig->release == GLFW_RELEASE_BEHAVIOR_NONE)
        {
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
546
547
            setAttrib(EGL_CONTEXT_RELEASE_BEHAVIOR_KHR,
                      EGL_CONTEXT_RELEASE_BEHAVIOR_NONE_KHR);
548
549
550
        }
        else if (ctxconfig->release == GLFW_RELEASE_BEHAVIOR_FLUSH)
        {
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
551
552
            setAttrib(EGL_CONTEXT_RELEASE_BEHAVIOR_KHR,
                      EGL_CONTEXT_RELEASE_BEHAVIOR_FLUSH_KHR);
553
        }
554
    }
Cloudef's avatar
Cloudef committed
555

Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
556
    setAttrib(EGL_NONE, EGL_NONE);
557

558
559
    window->context.egl.handle = eglCreateContext(_glfw.egl.display,
                                                  config, share, attribs);
560

561
    if (window->context.egl.handle == EGL_NO_CONTEXT)
Cloudef's avatar
Cloudef committed
562
    {
Camilla Berglund's avatar
Camilla Berglund committed
563
        _glfwInputError(GLFW_VERSION_UNAVAILABLE,
564
                        "EGL: Failed to create context: %s",
565
                        getEGLErrorString(eglGetError()));
566
        return GLFW_FALSE;
Cloudef's avatar
Cloudef committed
567
568
    }

569
570
571
572
573
574
575
    // Set up attributes for surface creation
    {
        int index = 0;

        if (fbconfig->sRGB)
        {
            if (_glfw.egl.KHR_gl_colorspace)
Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
576
                setAttrib(EGL_GL_COLORSPACE_KHR, EGL_GL_COLORSPACE_SRGB_KHR);
577
578
        }

Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
579
        setAttrib(EGL_NONE, EGL_NONE);
580
581
    }

582
    window->context.egl.surface =
583
584
585
        eglCreateWindowSurface(_glfw.egl.display,
                               config,
                               _GLFW_EGL_NATIVE_WINDOW,
586
                               attribs);
587
    if (window->context.egl.surface == EGL_NO_SURFACE)
588
589
590
    {
        _glfwInputError(GLFW_PLATFORM_ERROR,
                        "EGL: Failed to create window surface: %s",
591
                        getEGLErrorString(eglGetError()));
592
593
594
        return GLFW_FALSE;
    }

595
    window->context.egl.config = config;
Cloudef's avatar
Cloudef committed
596

597
    // Load the appropriate client library
598
    if (!_glfw.egl.KHR_get_all_proc_addresses)
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
    {
        int i;
        const char** sonames;
        const char* es1sonames[] =
        {
#if defined(_GLFW_WIN32)
            "GLESv1_CM.dll",
            "libGLES_CM.dll",
#elif defined(_GLFW_COCOA)
            "libGLESv1_CM.dylib",
#else
            "libGLESv1_CM.so.1",
            "libGLES_CM.so.1",
#endif
            NULL
        };
        const char* es2sonames[] =
        {
#if defined(_GLFW_WIN32)
            "GLESv2.dll",
            "libGLESv2.dll",
#elif defined(_GLFW_COCOA)
            "libGLESv2.dylib",
Camilla Löwy's avatar
Camilla Löwy committed
622
623
#elif defined(__CYGWIN__)
            "libGLESv2-2.so",
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
#else
            "libGLESv2.so.2",
#endif
            NULL
        };
        const char* glsonames[] =
        {
#if defined(_GLFW_WIN32)
#elif defined(_GLFW_COCOA)
#else
            "libGL.so.1",
#endif
            NULL
        };

639
        if (ctxconfig->client == GLFW_OPENGL_ES_API)
640
641
642
643
644
645
646
647
648
649
650
        {
            if (ctxconfig->major == 1)
                sonames = es1sonames;
            else
                sonames = es2sonames;
        }
        else
            sonames = glsonames;

        for (i = 0;  sonames[i];  i++)
        {
651
652
653
654
655
            // HACK: Match presence of lib prefix to increase chance of finding
            //       a matching pair in the jungle that is Win32 EGL/GLES
            if (_glfw.egl.prefix != (strncmp(sonames[i], "lib", 3) == 0))
                continue;

656
657
            window->context.egl.client = _glfw_dlopen(sonames[i]);
            if (window->context.egl.client)
658
659
660
                break;
        }

661
        if (!window->context.egl.client)
662
663
664
        {
            _glfwInputError(GLFW_API_UNAVAILABLE,
                            "EGL: Failed to load client library");
665
            return GLFW_FALSE;
666
667
668
        }
    }

669
670
671
672
673
674
    window->context.makeCurrent = makeContextCurrentEGL;
    window->context.swapBuffers = swapBuffersEGL;
    window->context.swapInterval = swapIntervalEGL;
    window->context.extensionSupported = extensionSupportedEGL;
    window->context.getProcAddress = getProcAddressEGL;
    window->context.destroy = destroyContextEGL;
675

676
    return GLFW_TRUE;
Cloudef's avatar
Cloudef committed
677
678
}

Camilla Löwy's avatar
Cleanup    
Camilla Löwy committed
679
#undef setAttrib
Cloudef's avatar
Cloudef committed
680

681
682
683
// Returns the Visual and depth of the chosen EGLConfig
//
#if defined(_GLFW_X11)
684
685
686
GLFWbool _glfwChooseVisualEGL(const _GLFWctxconfig* ctxconfig,
                              const _GLFWfbconfig* fbconfig,
                              Visual** visual, int* depth)
687
688
689
690
691
692
693
{
    XVisualInfo* result;
    XVisualInfo desired;
    EGLConfig native;
    EGLint visualID = 0, count = 0;
    const long vimask = VisualScreenMask | VisualIDMask;

694
    if (!chooseEGLConfig(ctxconfig, fbconfig, &native))
695
696
697
698
699
700
    {
        _glfwInputError(GLFW_FORMAT_UNAVAILABLE,
                        "EGL: Failed to find a suitable EGLConfig");
        return GLFW_FALSE;
    }

701
702
    eglGetConfigAttrib(_glfw.egl.display, native,
                       EGL_NATIVE_VISUAL_ID, &visualID);
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721

    desired.screen = _glfw.x11.screen;
    desired.visualid = visualID;

    result = XGetVisualInfo(_glfw.x11.display, vimask, &desired, &count);
    if (!result)
    {
        _glfwInputError(GLFW_PLATFORM_ERROR,
                        "EGL: Failed to retrieve Visual for EGLConfig");
        return GLFW_FALSE;
    }

    *visual = result->visual;
    *depth = result->depth;

    XFree(result);
    return GLFW_TRUE;
}
#endif // _GLFW_X11
Camilla Berglund's avatar
Camilla Berglund committed
722
723


724
725
726
727
728
729
//////////////////////////////////////////////////////////////////////////
//////                        GLFW native API                       //////
//////////////////////////////////////////////////////////////////////////

GLFWAPI EGLDisplay glfwGetEGLDisplay(void)
{
730
    _GLFW_REQUIRE_INIT_OR_RETURN(EGL_NO_DISPLAY);
731
732
733
734
735
736
    return _glfw.egl.display;
}

GLFWAPI EGLContext glfwGetEGLContext(GLFWwindow* handle)
{
    _GLFWwindow* window = (_GLFWwindow*) handle;
737
    _GLFW_REQUIRE_INIT_OR_RETURN(EGL_NO_CONTEXT);
738

739
    if (window->context.client == GLFW_NO_API)
740
741
742
743
744
    {
        _glfwInputError(GLFW_NO_WINDOW_CONTEXT, NULL);
        return EGL_NO_CONTEXT;
    }

745
    return window->context.egl.handle;
746
747
748
749
750
}

GLFWAPI EGLSurface glfwGetEGLSurface(GLFWwindow* handle)
{
    _GLFWwindow* window = (_GLFWwindow*) handle;
751
    _GLFW_REQUIRE_INIT_OR_RETURN(EGL_NO_SURFACE);
752

753
    if (window->context.client == GLFW_NO_API)
754
755
756
757
758
    {
        _glfwInputError(GLFW_NO_WINDOW_CONTEXT, NULL);
        return EGL_NO_SURFACE;
    }

759
    return window->context.egl.surface;
760
761
}