egl: Enable EGL_EXT_client_extensions
[mesa.git] / src / egl / main / eglapi.c
1 /**************************************************************************
2 *
3 * Copyright 2008 Tungsten Graphics, Inc., Cedar Park, Texas.
4 * Copyright 2009-2010 Chia-I Wu <olvaffe@gmail.com>
5 * Copyright 2010-2011 LunarG, Inc.
6 * All Rights Reserved.
7 *
8 * Permission is hereby granted, free of charge, to any person obtaining a
9 * copy of this software and associated documentation files (the
10 * "Software"), to deal in the Software without restriction, including
11 * without limitation the rights to use, copy, modify, merge, publish,
12 * distribute, sub license, and/or sell copies of the Software, and to
13 * permit persons to whom the Software is furnished to do so, subject to
14 * the following conditions:
15 *
16 * The above copyright notice and this permission notice (including the
17 * next paragraph) shall be included in all copies or substantial portions
18 * of the Software.
19 *
20 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
23 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
24 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
25 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
26 * DEALINGS IN THE SOFTWARE.
27 *
28 **************************************************************************/
29
30
31 /**
32 * Public EGL API entrypoints
33 *
34 * Generally, we use the EGLDisplay parameter as a key to lookup the
35 * appropriate device driver handle, then jump though the driver's
36 * dispatch table to handle the function.
37 *
38 * That allows us the option of supporting multiple, simultaneous,
39 * heterogeneous hardware devices in the future.
40 *
41 * The EGLDisplay, EGLConfig, EGLContext and EGLSurface types are
42 * opaque handles. Internal objects are linked to a display to
43 * create the handles.
44 *
45 * For each public API entry point, the opaque handles are looked up
46 * before being dispatched to the drivers. When it fails to look up
47 * a handle, one of
48 *
49 * EGL_BAD_DISPLAY
50 * EGL_BAD_CONFIG
51 * EGL_BAD_CONTEXT
52 * EGL_BAD_SURFACE
53 * EGL_BAD_SCREEN_MESA
54 * EGL_BAD_MODE_MESA
55 *
56 * is generated and the driver function is not called. An
57 * uninitialized EGLDisplay has no driver associated with it. When
58 * such display is detected,
59 *
60 * EGL_NOT_INITIALIZED
61 *
62 * is generated.
63 *
64 * Some of the entry points use current display, context, or surface
65 * implicitly. For such entry points, the implicit objects are also
66 * checked before calling the driver function. Other than the
67 * errors listed above,
68 *
69 * EGL_BAD_CURRENT_SURFACE
70 *
71 * may also be generated.
72 *
73 * Notes on naming conventions:
74 *
75 * eglFooBar - public EGL function
76 * EGL_FOO_BAR - public EGL token
77 * EGLDatatype - public EGL datatype
78 *
79 * _eglFooBar - private EGL function
80 * _EGLDatatype - private EGL datatype, typedef'd struct
81 * _egl_struct - private EGL struct, non-typedef'd
82 *
83 */
84
85
86 #include <stdio.h>
87 #include <stdlib.h>
88 #include <string.h>
89
90 #include "eglglobals.h"
91 #include "eglcontext.h"
92 #include "egldisplay.h"
93 #include "egltypedefs.h"
94 #include "eglcurrent.h"
95 #include "egldriver.h"
96 #include "eglsurface.h"
97 #include "eglconfig.h"
98 #include "eglscreen.h"
99 #include "eglmode.h"
100 #include "eglimage.h"
101 #include "eglsync.h"
102
103
104 /**
105 * Macros to help return an API entrypoint.
106 *
107 * These macros will unlock the display and record the error code.
108 */
109 #define RETURN_EGL_ERROR(disp, err, ret) \
110 do { \
111 if (disp) \
112 _eglUnlockDisplay(disp); \
113 /* EGL error codes are non-zero */ \
114 if (err) \
115 _eglError(err, __FUNCTION__); \
116 return ret; \
117 } while (0)
118
119 #define RETURN_EGL_SUCCESS(disp, ret) \
120 RETURN_EGL_ERROR(disp, EGL_SUCCESS, ret)
121
122 /* record EGL_SUCCESS only when ret evaluates to true */
123 #define RETURN_EGL_EVAL(disp, ret) \
124 RETURN_EGL_ERROR(disp, (ret) ? EGL_SUCCESS : 0, ret)
125
126
127 /*
128 * A bunch of macros and checks to simplify error checking.
129 */
130
131 #define _EGL_CHECK_DISPLAY(disp, ret, drv) \
132 do { \
133 drv = _eglCheckDisplay(disp, __FUNCTION__); \
134 if (!drv) \
135 RETURN_EGL_ERROR(disp, 0, ret); \
136 } while (0)
137
138 #define _EGL_CHECK_OBJECT(disp, type, obj, ret, drv) \
139 do { \
140 drv = _eglCheck ## type(disp, obj, __FUNCTION__); \
141 if (!drv) \
142 RETURN_EGL_ERROR(disp, 0, ret); \
143 } while (0)
144
145 #define _EGL_CHECK_SURFACE(disp, surf, ret, drv) \
146 _EGL_CHECK_OBJECT(disp, Surface, surf, ret, drv)
147
148 #define _EGL_CHECK_CONTEXT(disp, context, ret, drv) \
149 _EGL_CHECK_OBJECT(disp, Context, context, ret, drv)
150
151 #define _EGL_CHECK_CONFIG(disp, conf, ret, drv) \
152 _EGL_CHECK_OBJECT(disp, Config, conf, ret, drv)
153
154 #define _EGL_CHECK_SCREEN(disp, scrn, ret, drv) \
155 _EGL_CHECK_OBJECT(disp, Screen, scrn, ret, drv)
156
157 #define _EGL_CHECK_MODE(disp, m, ret, drv) \
158 _EGL_CHECK_OBJECT(disp, Mode, m, ret, drv)
159
160 #define _EGL_CHECK_SYNC(disp, s, ret, drv) \
161 _EGL_CHECK_OBJECT(disp, Sync, s, ret, drv)
162
163
164 static INLINE _EGLDriver *
165 _eglCheckDisplay(_EGLDisplay *disp, const char *msg)
166 {
167 if (!disp) {
168 _eglError(EGL_BAD_DISPLAY, msg);
169 return NULL;
170 }
171 if (!disp->Initialized) {
172 _eglError(EGL_NOT_INITIALIZED, msg);
173 return NULL;
174 }
175 return disp->Driver;
176 }
177
178
179 static INLINE _EGLDriver *
180 _eglCheckSurface(_EGLDisplay *disp, _EGLSurface *surf, const char *msg)
181 {
182 _EGLDriver *drv = _eglCheckDisplay(disp, msg);
183 if (!drv)
184 return NULL;
185 if (!surf) {
186 _eglError(EGL_BAD_SURFACE, msg);
187 return NULL;
188 }
189 return drv;
190 }
191
192
193 static INLINE _EGLDriver *
194 _eglCheckContext(_EGLDisplay *disp, _EGLContext *context, const char *msg)
195 {
196 _EGLDriver *drv = _eglCheckDisplay(disp, msg);
197 if (!drv)
198 return NULL;
199 if (!context) {
200 _eglError(EGL_BAD_CONTEXT, msg);
201 return NULL;
202 }
203 return drv;
204 }
205
206
207 static INLINE _EGLDriver *
208 _eglCheckConfig(_EGLDisplay *disp, _EGLConfig *conf, const char *msg)
209 {
210 _EGLDriver *drv = _eglCheckDisplay(disp, msg);
211 if (!drv)
212 return NULL;
213 if (!conf) {
214 _eglError(EGL_BAD_CONFIG, msg);
215 return NULL;
216 }
217 return drv;
218 }
219
220
221 static INLINE _EGLDriver *
222 _eglCheckSync(_EGLDisplay *disp, _EGLSync *s, const char *msg)
223 {
224 _EGLDriver *drv = _eglCheckDisplay(disp, msg);
225 if (!drv)
226 return NULL;
227 if (!s) {
228 _eglError(EGL_BAD_PARAMETER, msg);
229 return NULL;
230 }
231 return drv;
232 }
233
234
235 #ifdef EGL_MESA_screen_surface
236
237
238 static INLINE _EGLDriver *
239 _eglCheckScreen(_EGLDisplay *disp, _EGLScreen *scrn, const char *msg)
240 {
241 _EGLDriver *drv = _eglCheckDisplay(disp, msg);
242 if (!drv)
243 return NULL;
244 if (!scrn) {
245 _eglError(EGL_BAD_SCREEN_MESA, msg);
246 return NULL;
247 }
248 return drv;
249 }
250
251
252 static INLINE _EGLDriver *
253 _eglCheckMode(_EGLDisplay *disp, _EGLMode *m, const char *msg)
254 {
255 _EGLDriver *drv = _eglCheckDisplay(disp, msg);
256 if (!drv)
257 return NULL;
258 if (!m) {
259 _eglError(EGL_BAD_MODE_MESA, msg);
260 return NULL;
261 }
262 return drv;
263 }
264
265
266 #endif /* EGL_MESA_screen_surface */
267
268
269 /**
270 * Lookup and lock a display.
271 */
272 static INLINE _EGLDisplay *
273 _eglLockDisplay(EGLDisplay display)
274 {
275 _EGLDisplay *dpy = _eglLookupDisplay(display);
276 if (dpy)
277 _eglLockMutex(&dpy->Mutex);
278 return dpy;
279 }
280
281
282 /**
283 * Unlock a display.
284 */
285 static INLINE void
286 _eglUnlockDisplay(_EGLDisplay *dpy)
287 {
288 _eglUnlockMutex(&dpy->Mutex);
289 }
290
291
292 /**
293 * This is typically the first EGL function that an application calls.
294 * It associates a private _EGLDisplay object to the native display.
295 */
296 EGLDisplay EGLAPIENTRY
297 eglGetDisplay(EGLNativeDisplayType nativeDisplay)
298 {
299 _EGLPlatformType plat = _eglGetNativePlatform(nativeDisplay);
300 _EGLDisplay *dpy = _eglFindDisplay(plat, (void *) nativeDisplay);
301 return _eglGetDisplayHandle(dpy);
302 }
303
304
305 /**
306 * This is typically the second EGL function that an application calls.
307 * Here we load/initialize the actual hardware driver.
308 */
309 EGLBoolean EGLAPIENTRY
310 eglInitialize(EGLDisplay dpy, EGLint *major, EGLint *minor)
311 {
312 _EGLDisplay *disp = _eglLockDisplay(dpy);
313
314 if (!disp)
315 RETURN_EGL_ERROR(NULL, EGL_BAD_DISPLAY, EGL_FALSE);
316
317 if (!disp->Initialized) {
318 if (!_eglMatchDriver(disp, EGL_FALSE))
319 RETURN_EGL_ERROR(disp, EGL_NOT_INITIALIZED, EGL_FALSE);
320
321 /* limit to APIs supported by core */
322 disp->ClientAPIs &= _EGL_API_ALL_BITS;
323 }
324
325 /* Update applications version of major and minor if not NULL */
326 if ((major != NULL) && (minor != NULL)) {
327 *major = disp->VersionMajor;
328 *minor = disp->VersionMinor;
329 }
330
331 RETURN_EGL_SUCCESS(disp, EGL_TRUE);
332 }
333
334
335 EGLBoolean EGLAPIENTRY
336 eglTerminate(EGLDisplay dpy)
337 {
338 _EGLDisplay *disp = _eglLockDisplay(dpy);
339
340 if (!disp)
341 RETURN_EGL_ERROR(NULL, EGL_BAD_DISPLAY, EGL_FALSE);
342
343 if (disp->Initialized) {
344 _EGLDriver *drv = disp->Driver;
345
346 drv->API.Terminate(drv, disp);
347 /* do not reset disp->Driver */
348 disp->Initialized = EGL_FALSE;
349 }
350
351 RETURN_EGL_SUCCESS(disp, EGL_TRUE);
352 }
353
354
355 const char * EGLAPIENTRY
356 eglQueryString(EGLDisplay dpy, EGLint name)
357 {
358 _EGLDisplay *disp;
359 _EGLDriver *drv;
360 const char *ret;
361
362 if (dpy == EGL_NO_DISPLAY && name == EGL_EXTENSIONS) {
363 RETURN_EGL_SUCCESS(NULL, _eglGlobal.ClientExtensionString);
364 }
365
366 disp = _eglLockDisplay(dpy);
367 _EGL_CHECK_DISPLAY(disp, NULL, drv);
368 ret = drv->API.QueryString(drv, disp, name);
369
370 RETURN_EGL_EVAL(disp, ret);
371 }
372
373
374 EGLBoolean EGLAPIENTRY
375 eglGetConfigs(EGLDisplay dpy, EGLConfig *configs,
376 EGLint config_size, EGLint *num_config)
377 {
378 _EGLDisplay *disp = _eglLockDisplay(dpy);
379 _EGLDriver *drv;
380 EGLBoolean ret;
381
382 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
383 ret = drv->API.GetConfigs(drv, disp, configs, config_size, num_config);
384
385 RETURN_EGL_EVAL(disp, ret);
386 }
387
388
389 EGLBoolean EGLAPIENTRY
390 eglChooseConfig(EGLDisplay dpy, const EGLint *attrib_list, EGLConfig *configs,
391 EGLint config_size, EGLint *num_config)
392 {
393 _EGLDisplay *disp = _eglLockDisplay(dpy);
394 _EGLDriver *drv;
395 EGLBoolean ret;
396
397 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
398 ret = drv->API.ChooseConfig(drv, disp, attrib_list, configs,
399 config_size, num_config);
400
401 RETURN_EGL_EVAL(disp, ret);
402 }
403
404
405 EGLBoolean EGLAPIENTRY
406 eglGetConfigAttrib(EGLDisplay dpy, EGLConfig config,
407 EGLint attribute, EGLint *value)
408 {
409 _EGLDisplay *disp = _eglLockDisplay(dpy);
410 _EGLConfig *conf = _eglLookupConfig(config, disp);
411 _EGLDriver *drv;
412 EGLBoolean ret;
413
414 _EGL_CHECK_CONFIG(disp, conf, EGL_FALSE, drv);
415 ret = drv->API.GetConfigAttrib(drv, disp, conf, attribute, value);
416
417 RETURN_EGL_EVAL(disp, ret);
418 }
419
420
421 EGLContext EGLAPIENTRY
422 eglCreateContext(EGLDisplay dpy, EGLConfig config, EGLContext share_list,
423 const EGLint *attrib_list)
424 {
425 _EGLDisplay *disp = _eglLockDisplay(dpy);
426 _EGLConfig *conf = _eglLookupConfig(config, disp);
427 _EGLContext *share = _eglLookupContext(share_list, disp);
428 _EGLDriver *drv;
429 _EGLContext *context;
430 EGLContext ret;
431
432 _EGL_CHECK_DISPLAY(disp, EGL_NO_CONTEXT, drv);
433
434 if (!config) {
435 /* config may be NULL if surfaceless */
436 if (!disp->Extensions.KHR_surfaceless_context)
437 RETURN_EGL_ERROR(disp, EGL_BAD_CONFIG, EGL_NO_CONTEXT);
438 }
439
440 if (!share && share_list != EGL_NO_CONTEXT)
441 RETURN_EGL_ERROR(disp, EGL_BAD_CONTEXT, EGL_NO_CONTEXT);
442
443 context = drv->API.CreateContext(drv, disp, conf, share, attrib_list);
444 ret = (context) ? _eglLinkContext(context) : EGL_NO_CONTEXT;
445
446 RETURN_EGL_EVAL(disp, ret);
447 }
448
449
450 EGLBoolean EGLAPIENTRY
451 eglDestroyContext(EGLDisplay dpy, EGLContext ctx)
452 {
453 _EGLDisplay *disp = _eglLockDisplay(dpy);
454 _EGLContext *context = _eglLookupContext(ctx, disp);
455 _EGLDriver *drv;
456 EGLBoolean ret;
457
458 _EGL_CHECK_CONTEXT(disp, context, EGL_FALSE, drv);
459 _eglUnlinkContext(context);
460 ret = drv->API.DestroyContext(drv, disp, context);
461
462 RETURN_EGL_EVAL(disp, ret);
463 }
464
465
466 EGLBoolean EGLAPIENTRY
467 eglMakeCurrent(EGLDisplay dpy, EGLSurface draw, EGLSurface read,
468 EGLContext ctx)
469 {
470 _EGLDisplay *disp = _eglLockDisplay(dpy);
471 _EGLContext *context = _eglLookupContext(ctx, disp);
472 _EGLSurface *draw_surf = _eglLookupSurface(draw, disp);
473 _EGLSurface *read_surf = _eglLookupSurface(read, disp);
474 _EGLDriver *drv;
475 EGLBoolean ret;
476
477 if (!disp)
478 RETURN_EGL_ERROR(disp, EGL_BAD_DISPLAY, EGL_FALSE);
479 drv = disp->Driver;
480
481 /* display is allowed to be uninitialized under certain condition */
482 if (!disp->Initialized) {
483 if (draw != EGL_NO_SURFACE || read != EGL_NO_SURFACE ||
484 ctx != EGL_NO_CONTEXT)
485 RETURN_EGL_ERROR(disp, EGL_BAD_DISPLAY, EGL_FALSE);
486 }
487 if (!drv)
488 RETURN_EGL_SUCCESS(disp, EGL_TRUE);
489
490 if (!context && ctx != EGL_NO_CONTEXT)
491 RETURN_EGL_ERROR(disp, EGL_BAD_CONTEXT, EGL_FALSE);
492 if (!draw_surf || !read_surf) {
493 /* surfaces may be NULL if surfaceless */
494 if (!disp->Extensions.KHR_surfaceless_context)
495 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
496
497 if ((!draw_surf && draw != EGL_NO_SURFACE) ||
498 (!read_surf && read != EGL_NO_SURFACE))
499 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
500 if (draw_surf || read_surf)
501 RETURN_EGL_ERROR(disp, EGL_BAD_MATCH, EGL_FALSE);
502 }
503
504 ret = drv->API.MakeCurrent(drv, disp, draw_surf, read_surf, context);
505
506 RETURN_EGL_EVAL(disp, ret);
507 }
508
509
510 EGLBoolean EGLAPIENTRY
511 eglQueryContext(EGLDisplay dpy, EGLContext ctx,
512 EGLint attribute, EGLint *value)
513 {
514 _EGLDisplay *disp = _eglLockDisplay(dpy);
515 _EGLContext *context = _eglLookupContext(ctx, disp);
516 _EGLDriver *drv;
517 EGLBoolean ret;
518
519 _EGL_CHECK_CONTEXT(disp, context, EGL_FALSE, drv);
520 ret = drv->API.QueryContext(drv, disp, context, attribute, value);
521
522 RETURN_EGL_EVAL(disp, ret);
523 }
524
525
526 EGLSurface EGLAPIENTRY
527 eglCreateWindowSurface(EGLDisplay dpy, EGLConfig config,
528 EGLNativeWindowType window, const EGLint *attrib_list)
529 {
530 _EGLDisplay *disp = _eglLockDisplay(dpy);
531 _EGLConfig *conf = _eglLookupConfig(config, disp);
532 _EGLDriver *drv;
533 _EGLSurface *surf;
534 EGLSurface ret;
535
536 _EGL_CHECK_CONFIG(disp, conf, EGL_NO_SURFACE, drv);
537 if (disp->Platform != _eglGetNativePlatform(disp->PlatformDisplay))
538 RETURN_EGL_ERROR(disp, EGL_BAD_NATIVE_WINDOW, EGL_NO_SURFACE);
539
540 surf = drv->API.CreateWindowSurface(drv, disp, conf, window, attrib_list);
541 ret = (surf) ? _eglLinkSurface(surf) : EGL_NO_SURFACE;
542
543 RETURN_EGL_EVAL(disp, ret);
544 }
545
546
547 EGLSurface EGLAPIENTRY
548 eglCreatePixmapSurface(EGLDisplay dpy, EGLConfig config,
549 EGLNativePixmapType pixmap, const EGLint *attrib_list)
550 {
551 _EGLDisplay *disp = _eglLockDisplay(dpy);
552 _EGLConfig *conf = _eglLookupConfig(config, disp);
553 _EGLDriver *drv;
554 _EGLSurface *surf;
555 EGLSurface ret;
556
557 _EGL_CHECK_CONFIG(disp, conf, EGL_NO_SURFACE, drv);
558 if (disp->Platform != _eglGetNativePlatform(disp->PlatformDisplay))
559 RETURN_EGL_ERROR(disp, EGL_BAD_NATIVE_PIXMAP, EGL_NO_SURFACE);
560
561 surf = drv->API.CreatePixmapSurface(drv, disp, conf, pixmap, attrib_list);
562 ret = (surf) ? _eglLinkSurface(surf) : EGL_NO_SURFACE;
563
564 RETURN_EGL_EVAL(disp, ret);
565 }
566
567
568 EGLSurface EGLAPIENTRY
569 eglCreatePbufferSurface(EGLDisplay dpy, EGLConfig config,
570 const EGLint *attrib_list)
571 {
572 _EGLDisplay *disp = _eglLockDisplay(dpy);
573 _EGLConfig *conf = _eglLookupConfig(config, disp);
574 _EGLDriver *drv;
575 _EGLSurface *surf;
576 EGLSurface ret;
577
578 _EGL_CHECK_CONFIG(disp, conf, EGL_NO_SURFACE, drv);
579
580 surf = drv->API.CreatePbufferSurface(drv, disp, conf, attrib_list);
581 ret = (surf) ? _eglLinkSurface(surf) : EGL_NO_SURFACE;
582
583 RETURN_EGL_EVAL(disp, ret);
584 }
585
586
587 EGLBoolean EGLAPIENTRY
588 eglDestroySurface(EGLDisplay dpy, EGLSurface surface)
589 {
590 _EGLDisplay *disp = _eglLockDisplay(dpy);
591 _EGLSurface *surf = _eglLookupSurface(surface, disp);
592 _EGLDriver *drv;
593 EGLBoolean ret;
594
595 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
596 _eglUnlinkSurface(surf);
597 ret = drv->API.DestroySurface(drv, disp, surf);
598
599 RETURN_EGL_EVAL(disp, ret);
600 }
601
602 EGLBoolean EGLAPIENTRY
603 eglQuerySurface(EGLDisplay dpy, EGLSurface surface,
604 EGLint attribute, EGLint *value)
605 {
606 _EGLDisplay *disp = _eglLockDisplay(dpy);
607 _EGLSurface *surf = _eglLookupSurface(surface, disp);
608 _EGLDriver *drv;
609 EGLBoolean ret;
610
611 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
612 ret = drv->API.QuerySurface(drv, disp, surf, attribute, value);
613
614 RETURN_EGL_EVAL(disp, ret);
615 }
616
617 EGLBoolean EGLAPIENTRY
618 eglSurfaceAttrib(EGLDisplay dpy, EGLSurface surface,
619 EGLint attribute, EGLint value)
620 {
621 _EGLDisplay *disp = _eglLockDisplay(dpy);
622 _EGLSurface *surf = _eglLookupSurface(surface, disp);
623 _EGLDriver *drv;
624 EGLBoolean ret;
625
626 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
627 ret = drv->API.SurfaceAttrib(drv, disp, surf, attribute, value);
628
629 RETURN_EGL_EVAL(disp, ret);
630 }
631
632
633 EGLBoolean EGLAPIENTRY
634 eglBindTexImage(EGLDisplay dpy, EGLSurface surface, EGLint buffer)
635 {
636 _EGLDisplay *disp = _eglLockDisplay(dpy);
637 _EGLSurface *surf = _eglLookupSurface(surface, disp);
638 _EGLDriver *drv;
639 EGLBoolean ret;
640
641 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
642 ret = drv->API.BindTexImage(drv, disp, surf, buffer);
643
644 RETURN_EGL_EVAL(disp, ret);
645 }
646
647
648 EGLBoolean EGLAPIENTRY
649 eglReleaseTexImage(EGLDisplay dpy, EGLSurface surface, EGLint buffer)
650 {
651 _EGLDisplay *disp = _eglLockDisplay(dpy);
652 _EGLSurface *surf = _eglLookupSurface(surface, disp);
653 _EGLDriver *drv;
654 EGLBoolean ret;
655
656 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
657 ret = drv->API.ReleaseTexImage(drv, disp, surf, buffer);
658
659 RETURN_EGL_EVAL(disp, ret);
660 }
661
662
663 EGLBoolean EGLAPIENTRY
664 eglSwapInterval(EGLDisplay dpy, EGLint interval)
665 {
666 _EGLDisplay *disp = _eglLockDisplay(dpy);
667 _EGLContext *ctx = _eglGetCurrentContext();
668 _EGLSurface *surf;
669 _EGLDriver *drv;
670 EGLBoolean ret;
671
672 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
673
674 if (_eglGetContextHandle(ctx) == EGL_NO_CONTEXT ||
675 ctx->Resource.Display != disp)
676 RETURN_EGL_ERROR(disp, EGL_BAD_CONTEXT, EGL_FALSE);
677
678 surf = ctx->DrawSurface;
679 if (_eglGetSurfaceHandle(surf) == EGL_NO_SURFACE)
680 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
681
682 ret = drv->API.SwapInterval(drv, disp, surf, interval);
683
684 RETURN_EGL_EVAL(disp, ret);
685 }
686
687
688 EGLBoolean EGLAPIENTRY
689 eglSwapBuffers(EGLDisplay dpy, EGLSurface surface)
690 {
691 _EGLContext *ctx = _eglGetCurrentContext();
692 _EGLDisplay *disp = _eglLockDisplay(dpy);
693 _EGLSurface *surf = _eglLookupSurface(surface, disp);
694 _EGLDriver *drv;
695 EGLBoolean ret;
696
697 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
698
699 /* surface must be bound to current context in EGL 1.4 */
700 if (_eglGetContextHandle(ctx) == EGL_NO_CONTEXT ||
701 surf != ctx->DrawSurface)
702 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
703
704 ret = drv->API.SwapBuffers(drv, disp, surf);
705
706 RETURN_EGL_EVAL(disp, ret);
707 }
708
709
710 #ifdef EGL_EXT_swap_buffers_with_damage
711
712 EGLBoolean EGLAPIENTRY
713 eglSwapBuffersWithDamageEXT(EGLDisplay dpy, EGLSurface surface,
714 EGLint *rects, EGLint n_rects)
715 {
716 _EGLContext *ctx = _eglGetCurrentContext();
717 _EGLDisplay *disp = _eglLockDisplay(dpy);
718 _EGLSurface *surf = _eglLookupSurface(surface, disp);
719 _EGLDriver *drv;
720 EGLBoolean ret;
721
722 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
723
724 /* surface must be bound to current context in EGL 1.4 */
725 if (_eglGetContextHandle(ctx) == EGL_NO_CONTEXT ||
726 surf != ctx->DrawSurface)
727 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
728
729 if ((n_rects > 0 && rects == NULL) || n_rects < 0)
730 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
731
732 ret = drv->API.SwapBuffersWithDamageEXT(drv, disp, surf, rects, n_rects);
733
734 RETURN_EGL_EVAL(disp, ret);
735 }
736
737 #endif /* EGL_EXT_swap_buffers_with_damage */
738
739 EGLBoolean EGLAPIENTRY
740 eglCopyBuffers(EGLDisplay dpy, EGLSurface surface, EGLNativePixmapType target)
741 {
742 _EGLDisplay *disp = _eglLockDisplay(dpy);
743 _EGLSurface *surf = _eglLookupSurface(surface, disp);
744 _EGLDriver *drv;
745 EGLBoolean ret;
746
747 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
748 if (disp->Platform != _eglGetNativePlatform(disp->PlatformDisplay))
749 RETURN_EGL_ERROR(disp, EGL_BAD_NATIVE_PIXMAP, EGL_FALSE);
750 ret = drv->API.CopyBuffers(drv, disp, surf, target);
751
752 RETURN_EGL_EVAL(disp, ret);
753 }
754
755
756 EGLBoolean EGLAPIENTRY
757 eglWaitClient(void)
758 {
759 _EGLContext *ctx = _eglGetCurrentContext();
760 _EGLDisplay *disp;
761 _EGLDriver *drv;
762 EGLBoolean ret;
763
764 if (!ctx)
765 RETURN_EGL_SUCCESS(NULL, EGL_TRUE);
766
767 disp = ctx->Resource.Display;
768 _eglLockMutex(&disp->Mutex);
769
770 /* let bad current context imply bad current surface */
771 if (_eglGetContextHandle(ctx) == EGL_NO_CONTEXT ||
772 _eglGetSurfaceHandle(ctx->DrawSurface) == EGL_NO_SURFACE)
773 RETURN_EGL_ERROR(disp, EGL_BAD_CURRENT_SURFACE, EGL_FALSE);
774
775 /* a valid current context implies an initialized current display */
776 assert(disp->Initialized);
777 drv = disp->Driver;
778 ret = drv->API.WaitClient(drv, disp, ctx);
779
780 RETURN_EGL_EVAL(disp, ret);
781 }
782
783
784 EGLBoolean EGLAPIENTRY
785 eglWaitGL(void)
786 {
787 _EGLThreadInfo *t = _eglGetCurrentThread();
788 EGLint api_index = t->CurrentAPIIndex;
789 EGLint es_index = _eglConvertApiToIndex(EGL_OPENGL_ES_API);
790 EGLBoolean ret;
791
792 if (api_index != es_index && _eglIsCurrentThreadDummy())
793 RETURN_EGL_ERROR(NULL, EGL_BAD_ALLOC, EGL_FALSE);
794
795 t->CurrentAPIIndex = es_index;
796 ret = eglWaitClient();
797 t->CurrentAPIIndex = api_index;
798 return ret;
799 }
800
801
802 EGLBoolean EGLAPIENTRY
803 eglWaitNative(EGLint engine)
804 {
805 _EGLContext *ctx = _eglGetCurrentContext();
806 _EGLDisplay *disp;
807 _EGLDriver *drv;
808 EGLBoolean ret;
809
810 if (!ctx)
811 RETURN_EGL_SUCCESS(NULL, EGL_TRUE);
812
813 disp = ctx->Resource.Display;
814 _eglLockMutex(&disp->Mutex);
815
816 /* let bad current context imply bad current surface */
817 if (_eglGetContextHandle(ctx) == EGL_NO_CONTEXT ||
818 _eglGetSurfaceHandle(ctx->DrawSurface) == EGL_NO_SURFACE)
819 RETURN_EGL_ERROR(disp, EGL_BAD_CURRENT_SURFACE, EGL_FALSE);
820
821 /* a valid current context implies an initialized current display */
822 assert(disp->Initialized);
823 drv = disp->Driver;
824 ret = drv->API.WaitNative(drv, disp, engine);
825
826 RETURN_EGL_EVAL(disp, ret);
827 }
828
829
830 EGLDisplay EGLAPIENTRY
831 eglGetCurrentDisplay(void)
832 {
833 _EGLContext *ctx = _eglGetCurrentContext();
834 EGLDisplay ret;
835
836 ret = (ctx) ? _eglGetDisplayHandle(ctx->Resource.Display) : EGL_NO_DISPLAY;
837
838 RETURN_EGL_SUCCESS(NULL, ret);
839 }
840
841
842 EGLContext EGLAPIENTRY
843 eglGetCurrentContext(void)
844 {
845 _EGLContext *ctx = _eglGetCurrentContext();
846 EGLContext ret;
847
848 ret = _eglGetContextHandle(ctx);
849
850 RETURN_EGL_SUCCESS(NULL, ret);
851 }
852
853
854 EGLSurface EGLAPIENTRY
855 eglGetCurrentSurface(EGLint readdraw)
856 {
857 _EGLContext *ctx = _eglGetCurrentContext();
858 EGLint err = EGL_SUCCESS;
859 _EGLSurface *surf;
860 EGLSurface ret;
861
862 if (!ctx)
863 RETURN_EGL_SUCCESS(NULL, EGL_NO_SURFACE);
864
865 switch (readdraw) {
866 case EGL_DRAW:
867 surf = ctx->DrawSurface;
868 break;
869 case EGL_READ:
870 surf = ctx->ReadSurface;
871 break;
872 default:
873 surf = NULL;
874 err = EGL_BAD_PARAMETER;
875 break;
876 }
877
878 ret = _eglGetSurfaceHandle(surf);
879
880 RETURN_EGL_ERROR(NULL, err, ret);
881 }
882
883
884 EGLint EGLAPIENTRY
885 eglGetError(void)
886 {
887 _EGLThreadInfo *t = _eglGetCurrentThread();
888 EGLint e = t->LastError;
889 if (!_eglIsCurrentThreadDummy())
890 t->LastError = EGL_SUCCESS;
891 return e;
892 }
893
894
895 __eglMustCastToProperFunctionPointerType EGLAPIENTRY
896 eglGetProcAddress(const char *procname)
897 {
898 static const struct {
899 const char *name;
900 _EGLProc function;
901 } egl_functions[] = {
902 /* core functions should not be queryable, but, well... */
903 #ifdef _EGL_GET_CORE_ADDRESSES
904 /* alphabetical order */
905 { "eglBindAPI", (_EGLProc) eglBindAPI },
906 { "eglBindTexImage", (_EGLProc) eglBindTexImage },
907 { "eglChooseConfig", (_EGLProc) eglChooseConfig },
908 { "eglCopyBuffers", (_EGLProc) eglCopyBuffers },
909 { "eglCreateContext", (_EGLProc) eglCreateContext },
910 { "eglCreatePbufferFromClientBuffer", (_EGLProc) eglCreatePbufferFromClientBuffer },
911 { "eglCreatePbufferSurface", (_EGLProc) eglCreatePbufferSurface },
912 { "eglCreatePixmapSurface", (_EGLProc) eglCreatePixmapSurface },
913 { "eglCreateWindowSurface", (_EGLProc) eglCreateWindowSurface },
914 { "eglDestroyContext", (_EGLProc) eglDestroyContext },
915 { "eglDestroySurface", (_EGLProc) eglDestroySurface },
916 { "eglGetConfigAttrib", (_EGLProc) eglGetConfigAttrib },
917 { "eglGetConfigs", (_EGLProc) eglGetConfigs },
918 { "eglGetCurrentContext", (_EGLProc) eglGetCurrentContext },
919 { "eglGetCurrentDisplay", (_EGLProc) eglGetCurrentDisplay },
920 { "eglGetCurrentSurface", (_EGLProc) eglGetCurrentSurface },
921 { "eglGetDisplay", (_EGLProc) eglGetDisplay },
922 { "eglGetError", (_EGLProc) eglGetError },
923 { "eglGetProcAddress", (_EGLProc) eglGetProcAddress },
924 { "eglInitialize", (_EGLProc) eglInitialize },
925 { "eglMakeCurrent", (_EGLProc) eglMakeCurrent },
926 { "eglQueryAPI", (_EGLProc) eglQueryAPI },
927 { "eglQueryContext", (_EGLProc) eglQueryContext },
928 { "eglQueryString", (_EGLProc) eglQueryString },
929 { "eglQuerySurface", (_EGLProc) eglQuerySurface },
930 { "eglReleaseTexImage", (_EGLProc) eglReleaseTexImage },
931 { "eglReleaseThread", (_EGLProc) eglReleaseThread },
932 { "eglSurfaceAttrib", (_EGLProc) eglSurfaceAttrib },
933 { "eglSwapBuffers", (_EGLProc) eglSwapBuffers },
934 { "eglSwapInterval", (_EGLProc) eglSwapInterval },
935 { "eglTerminate", (_EGLProc) eglTerminate },
936 { "eglWaitClient", (_EGLProc) eglWaitClient },
937 { "eglWaitGL", (_EGLProc) eglWaitGL },
938 { "eglWaitNative", (_EGLProc) eglWaitNative },
939 #endif /* _EGL_GET_CORE_ADDRESSES */
940 #ifdef EGL_MESA_screen_surface
941 { "eglChooseModeMESA", (_EGLProc) eglChooseModeMESA },
942 { "eglGetModesMESA", (_EGLProc) eglGetModesMESA },
943 { "eglGetModeAttribMESA", (_EGLProc) eglGetModeAttribMESA },
944 { "eglCopyContextMESA", (_EGLProc) eglCopyContextMESA },
945 { "eglGetScreensMESA", (_EGLProc) eglGetScreensMESA },
946 { "eglCreateScreenSurfaceMESA", (_EGLProc) eglCreateScreenSurfaceMESA },
947 { "eglShowScreenSurfaceMESA", (_EGLProc) eglShowScreenSurfaceMESA },
948 { "eglScreenPositionMESA", (_EGLProc) eglScreenPositionMESA },
949 { "eglQueryScreenMESA", (_EGLProc) eglQueryScreenMESA },
950 { "eglQueryScreenSurfaceMESA", (_EGLProc) eglQueryScreenSurfaceMESA },
951 { "eglQueryScreenModeMESA", (_EGLProc) eglQueryScreenModeMESA },
952 { "eglQueryModeStringMESA", (_EGLProc) eglQueryModeStringMESA },
953 #endif /* EGL_MESA_screen_surface */
954 #ifdef EGL_MESA_drm_display
955 { "eglGetDRMDisplayMESA", (_EGLProc) eglGetDRMDisplayMESA },
956 #endif
957 { "eglCreateImageKHR", (_EGLProc) eglCreateImageKHR },
958 { "eglDestroyImageKHR", (_EGLProc) eglDestroyImageKHR },
959 { "eglCreateSyncKHR", (_EGLProc) eglCreateSyncKHR },
960 { "eglDestroySyncKHR", (_EGLProc) eglDestroySyncKHR },
961 { "eglClientWaitSyncKHR", (_EGLProc) eglClientWaitSyncKHR },
962 { "eglSignalSyncKHR", (_EGLProc) eglSignalSyncKHR },
963 { "eglGetSyncAttribKHR", (_EGLProc) eglGetSyncAttribKHR },
964 #ifdef EGL_NOK_swap_region
965 { "eglSwapBuffersRegionNOK", (_EGLProc) eglSwapBuffersRegionNOK },
966 #endif
967 #ifdef EGL_MESA_drm_image
968 { "eglCreateDRMImageMESA", (_EGLProc) eglCreateDRMImageMESA },
969 { "eglExportDRMImageMESA", (_EGLProc) eglExportDRMImageMESA },
970 #endif
971 #ifdef EGL_WL_bind_wayland_display
972 { "eglBindWaylandDisplayWL", (_EGLProc) eglBindWaylandDisplayWL },
973 { "eglUnbindWaylandDisplayWL", (_EGLProc) eglUnbindWaylandDisplayWL },
974 { "eglQueryWaylandBufferWL", (_EGLProc) eglQueryWaylandBufferWL },
975 #endif
976 { "eglPostSubBufferNV", (_EGLProc) eglPostSubBufferNV },
977 #ifdef EGL_EXT_swap_buffers_with_damage
978 { "eglSwapBuffersWithDamageEXT", (_EGLProc) eglSwapBuffersWithDamageEXT },
979 #endif
980 { NULL, NULL }
981 };
982 EGLint i;
983 _EGLProc ret;
984
985 if (!procname)
986 RETURN_EGL_SUCCESS(NULL, NULL);
987
988 ret = NULL;
989 if (strncmp(procname, "egl", 3) == 0) {
990 for (i = 0; egl_functions[i].name; i++) {
991 if (strcmp(egl_functions[i].name, procname) == 0) {
992 ret = egl_functions[i].function;
993 break;
994 }
995 }
996 }
997 if (!ret)
998 ret = _eglGetDriverProc(procname);
999
1000 RETURN_EGL_SUCCESS(NULL, ret);
1001 }
1002
1003
1004 #ifdef EGL_MESA_screen_surface
1005
1006
1007 /*
1008 * EGL_MESA_screen extension
1009 */
1010
1011 EGLBoolean EGLAPIENTRY
1012 eglChooseModeMESA(EGLDisplay dpy, EGLScreenMESA screen,
1013 const EGLint *attrib_list, EGLModeMESA *modes,
1014 EGLint modes_size, EGLint *num_modes)
1015 {
1016 _EGLDisplay *disp = _eglLockDisplay(dpy);
1017 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1018 _EGLDriver *drv;
1019 EGLBoolean ret;
1020
1021 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1022 ret = drv->API.ChooseModeMESA(drv, disp, scrn, attrib_list,
1023 modes, modes_size, num_modes);
1024
1025 RETURN_EGL_EVAL(disp, ret);
1026 }
1027
1028
1029 EGLBoolean EGLAPIENTRY
1030 eglGetModesMESA(EGLDisplay dpy, EGLScreenMESA screen, EGLModeMESA *modes,
1031 EGLint mode_size, EGLint *num_mode)
1032 {
1033 _EGLDisplay *disp = _eglLockDisplay(dpy);
1034 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1035 _EGLDriver *drv;
1036 EGLBoolean ret;
1037
1038 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1039 ret = drv->API.GetModesMESA(drv, disp, scrn, modes, mode_size, num_mode);
1040
1041 RETURN_EGL_EVAL(disp, ret);
1042 }
1043
1044
1045 EGLBoolean EGLAPIENTRY
1046 eglGetModeAttribMESA(EGLDisplay dpy, EGLModeMESA mode,
1047 EGLint attribute, EGLint *value)
1048 {
1049 _EGLDisplay *disp = _eglLockDisplay(dpy);
1050 _EGLMode *m = _eglLookupMode(mode, disp);
1051 _EGLDriver *drv;
1052 EGLBoolean ret;
1053
1054 _EGL_CHECK_MODE(disp, m, EGL_FALSE, drv);
1055 ret = drv->API.GetModeAttribMESA(drv, disp, m, attribute, value);
1056
1057 RETURN_EGL_EVAL(disp, ret);
1058 }
1059
1060
1061 EGLBoolean EGLAPIENTRY
1062 eglCopyContextMESA(EGLDisplay dpy, EGLContext source, EGLContext dest,
1063 EGLint mask)
1064 {
1065 _EGLDisplay *disp = _eglLockDisplay(dpy);
1066 _EGLContext *source_context = _eglLookupContext(source, disp);
1067 _EGLContext *dest_context = _eglLookupContext(dest, disp);
1068 _EGLDriver *drv;
1069 EGLBoolean ret;
1070
1071 _EGL_CHECK_CONTEXT(disp, source_context, EGL_FALSE, drv);
1072 if (!dest_context)
1073 RETURN_EGL_ERROR(disp, EGL_BAD_CONTEXT, EGL_FALSE);
1074
1075 ret = drv->API.CopyContextMESA(drv, disp,
1076 source_context, dest_context, mask);
1077
1078 RETURN_EGL_EVAL(disp, ret);
1079 }
1080
1081
1082 EGLBoolean EGLAPIENTRY
1083 eglGetScreensMESA(EGLDisplay dpy, EGLScreenMESA *screens,
1084 EGLint max_screens, EGLint *num_screens)
1085 {
1086 _EGLDisplay *disp = _eglLockDisplay(dpy);
1087 _EGLDriver *drv;
1088 EGLBoolean ret;
1089
1090 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1091 ret = drv->API.GetScreensMESA(drv, disp, screens, max_screens, num_screens);
1092
1093 RETURN_EGL_EVAL(disp, ret);
1094 }
1095
1096
1097 EGLSurface EGLAPIENTRY
1098 eglCreateScreenSurfaceMESA(EGLDisplay dpy, EGLConfig config,
1099 const EGLint *attrib_list)
1100 {
1101 _EGLDisplay *disp = _eglLockDisplay(dpy);
1102 _EGLConfig *conf = _eglLookupConfig(config, disp);
1103 _EGLDriver *drv;
1104 _EGLSurface *surf;
1105 EGLSurface ret;
1106
1107 _EGL_CHECK_CONFIG(disp, conf, EGL_NO_SURFACE, drv);
1108
1109 surf = drv->API.CreateScreenSurfaceMESA(drv, disp, conf, attrib_list);
1110 ret = (surf) ? _eglLinkSurface(surf) : EGL_NO_SURFACE;
1111
1112 RETURN_EGL_EVAL(disp, ret);
1113 }
1114
1115
1116 EGLBoolean EGLAPIENTRY
1117 eglShowScreenSurfaceMESA(EGLDisplay dpy, EGLint screen,
1118 EGLSurface surface, EGLModeMESA mode)
1119 {
1120 _EGLDisplay *disp = _eglLockDisplay(dpy);
1121 _EGLScreen *scrn = _eglLookupScreen((EGLScreenMESA) screen, disp);
1122 _EGLSurface *surf = _eglLookupSurface(surface, disp);
1123 _EGLMode *m = _eglLookupMode(mode, disp);
1124 _EGLDriver *drv;
1125 EGLBoolean ret;
1126
1127 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1128 if (!surf && surface != EGL_NO_SURFACE)
1129 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
1130 if (!m && mode != EGL_NO_MODE_MESA)
1131 RETURN_EGL_ERROR(disp, EGL_BAD_MODE_MESA, EGL_FALSE);
1132
1133 ret = drv->API.ShowScreenSurfaceMESA(drv, disp, scrn, surf, m);
1134
1135 RETURN_EGL_EVAL(disp, ret);
1136 }
1137
1138
1139 EGLBoolean EGLAPIENTRY
1140 eglScreenPositionMESA(EGLDisplay dpy, EGLScreenMESA screen, EGLint x, EGLint y)
1141 {
1142 _EGLDisplay *disp = _eglLockDisplay(dpy);
1143 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1144 _EGLDriver *drv;
1145 EGLBoolean ret;
1146
1147 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1148 ret = drv->API.ScreenPositionMESA(drv, disp, scrn, x, y);
1149
1150 RETURN_EGL_EVAL(disp, ret);
1151 }
1152
1153
1154 EGLBoolean EGLAPIENTRY
1155 eglQueryScreenMESA(EGLDisplay dpy, EGLScreenMESA screen,
1156 EGLint attribute, EGLint *value)
1157 {
1158 _EGLDisplay *disp = _eglLockDisplay(dpy);
1159 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1160 _EGLDriver *drv;
1161 EGLBoolean ret;
1162
1163 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1164 ret = drv->API.QueryScreenMESA(drv, disp, scrn, attribute, value);
1165
1166 RETURN_EGL_EVAL(disp, ret);
1167 }
1168
1169
1170 EGLBoolean EGLAPIENTRY
1171 eglQueryScreenSurfaceMESA(EGLDisplay dpy, EGLScreenMESA screen,
1172 EGLSurface *surface)
1173 {
1174 _EGLDisplay *disp = _eglLockDisplay(dpy);
1175 _EGLScreen *scrn = _eglLookupScreen((EGLScreenMESA) screen, disp);
1176 _EGLDriver *drv;
1177 _EGLSurface *surf;
1178 EGLBoolean ret;
1179
1180 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1181 ret = drv->API.QueryScreenSurfaceMESA(drv, disp, scrn, &surf);
1182 if (ret && surface)
1183 *surface = _eglGetSurfaceHandle(surf);
1184
1185 RETURN_EGL_EVAL(disp, ret);
1186 }
1187
1188
1189 EGLBoolean EGLAPIENTRY
1190 eglQueryScreenModeMESA(EGLDisplay dpy, EGLScreenMESA screen, EGLModeMESA *mode)
1191 {
1192 _EGLDisplay *disp = _eglLockDisplay(dpy);
1193 _EGLScreen *scrn = _eglLookupScreen((EGLScreenMESA) screen, disp);
1194 _EGLDriver *drv;
1195 _EGLMode *m;
1196 EGLBoolean ret;
1197
1198 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1199 ret = drv->API.QueryScreenModeMESA(drv, disp, scrn, &m);
1200 if (ret && mode)
1201 *mode = m->Handle;
1202
1203 RETURN_EGL_EVAL(disp, ret);
1204 }
1205
1206
1207 const char * EGLAPIENTRY
1208 eglQueryModeStringMESA(EGLDisplay dpy, EGLModeMESA mode)
1209 {
1210 _EGLDisplay *disp = _eglLockDisplay(dpy);
1211 _EGLMode *m = _eglLookupMode(mode, disp);
1212 _EGLDriver *drv;
1213 const char *ret;
1214
1215 _EGL_CHECK_MODE(disp, m, NULL, drv);
1216 ret = drv->API.QueryModeStringMESA(drv, disp, m);
1217
1218 RETURN_EGL_EVAL(disp, ret);
1219 }
1220
1221
1222 #endif /* EGL_MESA_screen_surface */
1223
1224
1225 #ifdef EGL_MESA_drm_display
1226
1227 EGLDisplay EGLAPIENTRY
1228 eglGetDRMDisplayMESA(int fd)
1229 {
1230 _EGLDisplay *dpy = _eglFindDisplay(_EGL_PLATFORM_DRM, (void *) (intptr_t) fd);
1231 return _eglGetDisplayHandle(dpy);
1232 }
1233
1234 #endif /* EGL_MESA_drm_display */
1235
1236 /**
1237 ** EGL 1.2
1238 **/
1239
1240 /**
1241 * Specify the client API to use for subsequent calls including:
1242 * eglCreateContext()
1243 * eglGetCurrentContext()
1244 * eglGetCurrentDisplay()
1245 * eglGetCurrentSurface()
1246 * eglMakeCurrent(when the ctx parameter is EGL NO CONTEXT)
1247 * eglWaitClient()
1248 * eglWaitNative()
1249 * See section 3.7 "Rendering Context" in the EGL specification for details.
1250 */
1251 EGLBoolean EGLAPIENTRY
1252 eglBindAPI(EGLenum api)
1253 {
1254 _EGLThreadInfo *t = _eglGetCurrentThread();
1255
1256 if (_eglIsCurrentThreadDummy())
1257 RETURN_EGL_ERROR(NULL, EGL_BAD_ALLOC, EGL_FALSE);
1258
1259 if (!_eglIsApiValid(api))
1260 RETURN_EGL_ERROR(NULL, EGL_BAD_PARAMETER, EGL_FALSE);
1261
1262 t->CurrentAPIIndex = _eglConvertApiToIndex(api);
1263
1264 RETURN_EGL_SUCCESS(NULL, EGL_TRUE);
1265 }
1266
1267
1268 /**
1269 * Return the last value set with eglBindAPI().
1270 */
1271 EGLenum EGLAPIENTRY
1272 eglQueryAPI(void)
1273 {
1274 _EGLThreadInfo *t = _eglGetCurrentThread();
1275 EGLenum ret;
1276
1277 /* returns one of EGL_OPENGL_API, EGL_OPENGL_ES_API or EGL_OPENVG_API */
1278 ret = _eglConvertApiFromIndex(t->CurrentAPIIndex);
1279
1280 RETURN_EGL_SUCCESS(NULL, ret);
1281 }
1282
1283
1284 EGLSurface EGLAPIENTRY
1285 eglCreatePbufferFromClientBuffer(EGLDisplay dpy, EGLenum buftype,
1286 EGLClientBuffer buffer, EGLConfig config,
1287 const EGLint *attrib_list)
1288 {
1289 _EGLDisplay *disp = _eglLockDisplay(dpy);
1290 _EGLConfig *conf = _eglLookupConfig(config, disp);
1291 _EGLDriver *drv;
1292 _EGLSurface *surf;
1293 EGLSurface ret;
1294
1295 _EGL_CHECK_CONFIG(disp, conf, EGL_NO_SURFACE, drv);
1296
1297 surf = drv->API.CreatePbufferFromClientBuffer(drv, disp, buftype, buffer,
1298 conf, attrib_list);
1299 ret = (surf) ? _eglLinkSurface(surf) : EGL_NO_SURFACE;
1300
1301 RETURN_EGL_EVAL(disp, ret);
1302 }
1303
1304
1305 EGLBoolean EGLAPIENTRY
1306 eglReleaseThread(void)
1307 {
1308 /* unbind current contexts */
1309 if (!_eglIsCurrentThreadDummy()) {
1310 _EGLThreadInfo *t = _eglGetCurrentThread();
1311 EGLint api_index = t->CurrentAPIIndex;
1312 EGLint i;
1313
1314 for (i = 0; i < _EGL_API_NUM_APIS; i++) {
1315 _EGLContext *ctx = t->CurrentContexts[i];
1316 if (ctx) {
1317 _EGLDisplay *disp = ctx->Resource.Display;
1318 _EGLDriver *drv;
1319
1320 t->CurrentAPIIndex = i;
1321
1322 _eglLockMutex(&disp->Mutex);
1323 drv = disp->Driver;
1324 (void) drv->API.MakeCurrent(drv, disp, NULL, NULL, NULL);
1325 _eglUnlockMutex(&disp->Mutex);
1326 }
1327 }
1328
1329 t->CurrentAPIIndex = api_index;
1330 }
1331
1332 _eglDestroyCurrentThread();
1333
1334 RETURN_EGL_SUCCESS(NULL, EGL_TRUE);
1335 }
1336
1337
1338 EGLImageKHR EGLAPIENTRY
1339 eglCreateImageKHR(EGLDisplay dpy, EGLContext ctx, EGLenum target,
1340 EGLClientBuffer buffer, const EGLint *attr_list)
1341 {
1342 _EGLDisplay *disp = _eglLockDisplay(dpy);
1343 _EGLContext *context = _eglLookupContext(ctx, disp);
1344 _EGLDriver *drv;
1345 _EGLImage *img;
1346 EGLImageKHR ret;
1347
1348 _EGL_CHECK_DISPLAY(disp, EGL_NO_IMAGE_KHR, drv);
1349 if (!disp->Extensions.KHR_image_base)
1350 RETURN_EGL_EVAL(disp, EGL_NO_IMAGE_KHR);
1351 if (!context && ctx != EGL_NO_CONTEXT)
1352 RETURN_EGL_ERROR(disp, EGL_BAD_CONTEXT, EGL_NO_IMAGE_KHR);
1353 /* "If <target> is EGL_LINUX_DMA_BUF_EXT, <dpy> must be a valid display,
1354 * <ctx> must be EGL_NO_CONTEXT..."
1355 */
1356 if (ctx != EGL_NO_CONTEXT && target == EGL_LINUX_DMA_BUF_EXT)
1357 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_NO_IMAGE_KHR);
1358
1359 img = drv->API.CreateImageKHR(drv,
1360 disp, context, target, buffer, attr_list);
1361 ret = (img) ? _eglLinkImage(img) : EGL_NO_IMAGE_KHR;
1362
1363 RETURN_EGL_EVAL(disp, ret);
1364 }
1365
1366
1367 EGLBoolean EGLAPIENTRY
1368 eglDestroyImageKHR(EGLDisplay dpy, EGLImageKHR image)
1369 {
1370 _EGLDisplay *disp = _eglLockDisplay(dpy);
1371 _EGLImage *img = _eglLookupImage(image, disp);
1372 _EGLDriver *drv;
1373 EGLBoolean ret;
1374
1375 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1376 if (!disp->Extensions.KHR_image_base)
1377 RETURN_EGL_EVAL(disp, EGL_FALSE);
1378 if (!img)
1379 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1380
1381 _eglUnlinkImage(img);
1382 ret = drv->API.DestroyImageKHR(drv, disp, img);
1383
1384 RETURN_EGL_EVAL(disp, ret);
1385 }
1386
1387
1388 EGLSyncKHR EGLAPIENTRY
1389 eglCreateSyncKHR(EGLDisplay dpy, EGLenum type, const EGLint *attrib_list)
1390 {
1391 _EGLDisplay *disp = _eglLockDisplay(dpy);
1392 _EGLDriver *drv;
1393 _EGLSync *sync;
1394 EGLSyncKHR ret;
1395
1396 _EGL_CHECK_DISPLAY(disp, EGL_NO_SYNC_KHR, drv);
1397 if (!disp->Extensions.KHR_reusable_sync)
1398 RETURN_EGL_EVAL(disp, EGL_NO_SYNC_KHR);
1399
1400 sync = drv->API.CreateSyncKHR(drv, disp, type, attrib_list);
1401 ret = (sync) ? _eglLinkSync(sync) : EGL_NO_SYNC_KHR;
1402
1403 RETURN_EGL_EVAL(disp, ret);
1404 }
1405
1406
1407 EGLBoolean EGLAPIENTRY
1408 eglDestroySyncKHR(EGLDisplay dpy, EGLSyncKHR sync)
1409 {
1410 _EGLDisplay *disp = _eglLockDisplay(dpy);
1411 _EGLSync *s = _eglLookupSync(sync, disp);
1412 _EGLDriver *drv;
1413 EGLBoolean ret;
1414
1415 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1416 assert(disp->Extensions.KHR_reusable_sync);
1417
1418 _eglUnlinkSync(s);
1419 ret = drv->API.DestroySyncKHR(drv, disp, s);
1420
1421 RETURN_EGL_EVAL(disp, ret);
1422 }
1423
1424
1425 EGLint EGLAPIENTRY
1426 eglClientWaitSyncKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLint flags, EGLTimeKHR timeout)
1427 {
1428 _EGLDisplay *disp = _eglLockDisplay(dpy);
1429 _EGLSync *s = _eglLookupSync(sync, disp);
1430 _EGLDriver *drv;
1431 EGLint ret;
1432
1433 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1434 assert(disp->Extensions.KHR_reusable_sync);
1435 ret = drv->API.ClientWaitSyncKHR(drv, disp, s, flags, timeout);
1436
1437 RETURN_EGL_EVAL(disp, ret);
1438 }
1439
1440
1441 EGLBoolean EGLAPIENTRY
1442 eglSignalSyncKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLenum mode)
1443 {
1444 _EGLDisplay *disp = _eglLockDisplay(dpy);
1445 _EGLSync *s = _eglLookupSync(sync, disp);
1446 _EGLDriver *drv;
1447 EGLBoolean ret;
1448
1449 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1450 assert(disp->Extensions.KHR_reusable_sync);
1451 ret = drv->API.SignalSyncKHR(drv, disp, s, mode);
1452
1453 RETURN_EGL_EVAL(disp, ret);
1454 }
1455
1456
1457 EGLBoolean EGLAPIENTRY
1458 eglGetSyncAttribKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLint attribute, EGLint *value)
1459 {
1460 _EGLDisplay *disp = _eglLockDisplay(dpy);
1461 _EGLSync *s = _eglLookupSync(sync, disp);
1462 _EGLDriver *drv;
1463 EGLBoolean ret;
1464
1465 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1466 assert(disp->Extensions.KHR_reusable_sync);
1467 ret = drv->API.GetSyncAttribKHR(drv, disp, s, attribute, value);
1468
1469 RETURN_EGL_EVAL(disp, ret);
1470 }
1471
1472
1473 #ifdef EGL_NOK_swap_region
1474
1475 EGLBoolean EGLAPIENTRY
1476 eglSwapBuffersRegionNOK(EGLDisplay dpy, EGLSurface surface,
1477 EGLint numRects, const EGLint *rects)
1478 {
1479 _EGLContext *ctx = _eglGetCurrentContext();
1480 _EGLDisplay *disp = _eglLockDisplay(dpy);
1481 _EGLSurface *surf = _eglLookupSurface(surface, disp);
1482 _EGLDriver *drv;
1483 EGLBoolean ret;
1484
1485 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
1486
1487 if (!disp->Extensions.NOK_swap_region)
1488 RETURN_EGL_EVAL(disp, EGL_FALSE);
1489
1490 /* surface must be bound to current context in EGL 1.4 */
1491 if (_eglGetContextHandle(ctx) == EGL_NO_CONTEXT ||
1492 surf != ctx->DrawSurface)
1493 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
1494
1495 ret = drv->API.SwapBuffersRegionNOK(drv, disp, surf, numRects, rects);
1496
1497 RETURN_EGL_EVAL(disp, ret);
1498 }
1499
1500 #endif /* EGL_NOK_swap_region */
1501
1502
1503 #ifdef EGL_MESA_drm_image
1504
1505 EGLImageKHR EGLAPIENTRY
1506 eglCreateDRMImageMESA(EGLDisplay dpy, const EGLint *attr_list)
1507 {
1508 _EGLDisplay *disp = _eglLockDisplay(dpy);
1509 _EGLDriver *drv;
1510 _EGLImage *img;
1511 EGLImageKHR ret;
1512
1513 _EGL_CHECK_DISPLAY(disp, EGL_NO_IMAGE_KHR, drv);
1514 if (!disp->Extensions.MESA_drm_image)
1515 RETURN_EGL_EVAL(disp, EGL_NO_IMAGE_KHR);
1516
1517 img = drv->API.CreateDRMImageMESA(drv, disp, attr_list);
1518 ret = (img) ? _eglLinkImage(img) : EGL_NO_IMAGE_KHR;
1519
1520 RETURN_EGL_EVAL(disp, ret);
1521 }
1522
1523 EGLBoolean EGLAPIENTRY
1524 eglExportDRMImageMESA(EGLDisplay dpy, EGLImageKHR image,
1525 EGLint *name, EGLint *handle, EGLint *stride)
1526 {
1527 _EGLDisplay *disp = _eglLockDisplay(dpy);
1528 _EGLImage *img = _eglLookupImage(image, disp);
1529 _EGLDriver *drv;
1530 EGLBoolean ret;
1531
1532 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1533 assert(disp->Extensions.MESA_drm_image);
1534
1535 if (!img)
1536 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1537
1538 ret = drv->API.ExportDRMImageMESA(drv, disp, img, name, handle, stride);
1539
1540 RETURN_EGL_EVAL(disp, ret);
1541 }
1542
1543 #endif
1544
1545 #ifdef EGL_WL_bind_wayland_display
1546 struct wl_display;
1547
1548 EGLBoolean EGLAPIENTRY
1549 eglBindWaylandDisplayWL(EGLDisplay dpy, struct wl_display *display)
1550 {
1551 _EGLDisplay *disp = _eglLockDisplay(dpy);
1552 _EGLDriver *drv;
1553 EGLBoolean ret;
1554
1555 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1556 assert(disp->Extensions.WL_bind_wayland_display);
1557
1558 if (!display)
1559 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1560
1561 ret = drv->API.BindWaylandDisplayWL(drv, disp, display);
1562
1563 RETURN_EGL_EVAL(disp, ret);
1564 }
1565
1566 EGLBoolean EGLAPIENTRY
1567 eglUnbindWaylandDisplayWL(EGLDisplay dpy, struct wl_display *display)
1568 {
1569 _EGLDisplay *disp = _eglLockDisplay(dpy);
1570 _EGLDriver *drv;
1571 EGLBoolean ret;
1572
1573 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1574 assert(disp->Extensions.WL_bind_wayland_display);
1575
1576 if (!display)
1577 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1578
1579 ret = drv->API.UnbindWaylandDisplayWL(drv, disp, display);
1580
1581 RETURN_EGL_EVAL(disp, ret);
1582 }
1583
1584 EGLBoolean EGLAPIENTRY
1585 eglQueryWaylandBufferWL(EGLDisplay dpy, struct wl_resource *buffer,
1586 EGLint attribute, EGLint *value)
1587 {
1588 _EGLDisplay *disp = _eglLockDisplay(dpy);
1589 _EGLDriver *drv;
1590 EGLBoolean ret;
1591
1592 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1593 assert(disp->Extensions.WL_bind_wayland_display);
1594
1595 if (!buffer)
1596 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1597
1598 ret = drv->API.QueryWaylandBufferWL(drv, disp, buffer, attribute, value);
1599
1600 RETURN_EGL_EVAL(disp, ret);
1601 }
1602 #endif
1603
1604
1605 EGLBoolean EGLAPIENTRY
1606 eglPostSubBufferNV(EGLDisplay dpy, EGLSurface surface,
1607 EGLint x, EGLint y, EGLint width, EGLint height)
1608 {
1609 _EGLDisplay *disp = _eglLockDisplay(dpy);
1610 _EGLSurface *surf = _eglLookupSurface(surface, disp);
1611 _EGLDriver *drv;
1612 EGLBoolean ret;
1613
1614 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
1615
1616 if (!disp->Extensions.NV_post_sub_buffer)
1617 RETURN_EGL_EVAL(disp, EGL_FALSE);
1618
1619 ret = drv->API.PostSubBufferNV(drv, disp, surf, x, y, width, height);
1620
1621 RETURN_EGL_EVAL(disp, ret);
1622 }