9ac2ef4469882b4f11834a73698b368f136b3bc8
[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 #ifdef EGL_WL_create_wayland_buffer_from_image
977 { "eglCreateWaylandBufferFromImageWL", (_EGLProc) eglCreateWaylandBufferFromImageWL },
978 #endif
979 { "eglPostSubBufferNV", (_EGLProc) eglPostSubBufferNV },
980 #ifdef EGL_EXT_swap_buffers_with_damage
981 { "eglSwapBuffersWithDamageEXT", (_EGLProc) eglSwapBuffersWithDamageEXT },
982 #endif
983 { NULL, NULL }
984 };
985 EGLint i;
986 _EGLProc ret;
987
988 if (!procname)
989 RETURN_EGL_SUCCESS(NULL, NULL);
990
991 ret = NULL;
992 if (strncmp(procname, "egl", 3) == 0) {
993 for (i = 0; egl_functions[i].name; i++) {
994 if (strcmp(egl_functions[i].name, procname) == 0) {
995 ret = egl_functions[i].function;
996 break;
997 }
998 }
999 }
1000 if (!ret)
1001 ret = _eglGetDriverProc(procname);
1002
1003 RETURN_EGL_SUCCESS(NULL, ret);
1004 }
1005
1006
1007 #ifdef EGL_MESA_screen_surface
1008
1009
1010 /*
1011 * EGL_MESA_screen extension
1012 */
1013
1014 EGLBoolean EGLAPIENTRY
1015 eglChooseModeMESA(EGLDisplay dpy, EGLScreenMESA screen,
1016 const EGLint *attrib_list, EGLModeMESA *modes,
1017 EGLint modes_size, EGLint *num_modes)
1018 {
1019 _EGLDisplay *disp = _eglLockDisplay(dpy);
1020 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1021 _EGLDriver *drv;
1022 EGLBoolean ret;
1023
1024 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1025 ret = drv->API.ChooseModeMESA(drv, disp, scrn, attrib_list,
1026 modes, modes_size, num_modes);
1027
1028 RETURN_EGL_EVAL(disp, ret);
1029 }
1030
1031
1032 EGLBoolean EGLAPIENTRY
1033 eglGetModesMESA(EGLDisplay dpy, EGLScreenMESA screen, EGLModeMESA *modes,
1034 EGLint mode_size, EGLint *num_mode)
1035 {
1036 _EGLDisplay *disp = _eglLockDisplay(dpy);
1037 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1038 _EGLDriver *drv;
1039 EGLBoolean ret;
1040
1041 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1042 ret = drv->API.GetModesMESA(drv, disp, scrn, modes, mode_size, num_mode);
1043
1044 RETURN_EGL_EVAL(disp, ret);
1045 }
1046
1047
1048 EGLBoolean EGLAPIENTRY
1049 eglGetModeAttribMESA(EGLDisplay dpy, EGLModeMESA mode,
1050 EGLint attribute, EGLint *value)
1051 {
1052 _EGLDisplay *disp = _eglLockDisplay(dpy);
1053 _EGLMode *m = _eglLookupMode(mode, disp);
1054 _EGLDriver *drv;
1055 EGLBoolean ret;
1056
1057 _EGL_CHECK_MODE(disp, m, EGL_FALSE, drv);
1058 ret = drv->API.GetModeAttribMESA(drv, disp, m, attribute, value);
1059
1060 RETURN_EGL_EVAL(disp, ret);
1061 }
1062
1063
1064 EGLBoolean EGLAPIENTRY
1065 eglCopyContextMESA(EGLDisplay dpy, EGLContext source, EGLContext dest,
1066 EGLint mask)
1067 {
1068 _EGLDisplay *disp = _eglLockDisplay(dpy);
1069 _EGLContext *source_context = _eglLookupContext(source, disp);
1070 _EGLContext *dest_context = _eglLookupContext(dest, disp);
1071 _EGLDriver *drv;
1072 EGLBoolean ret;
1073
1074 _EGL_CHECK_CONTEXT(disp, source_context, EGL_FALSE, drv);
1075 if (!dest_context)
1076 RETURN_EGL_ERROR(disp, EGL_BAD_CONTEXT, EGL_FALSE);
1077
1078 ret = drv->API.CopyContextMESA(drv, disp,
1079 source_context, dest_context, mask);
1080
1081 RETURN_EGL_EVAL(disp, ret);
1082 }
1083
1084
1085 EGLBoolean EGLAPIENTRY
1086 eglGetScreensMESA(EGLDisplay dpy, EGLScreenMESA *screens,
1087 EGLint max_screens, EGLint *num_screens)
1088 {
1089 _EGLDisplay *disp = _eglLockDisplay(dpy);
1090 _EGLDriver *drv;
1091 EGLBoolean ret;
1092
1093 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1094 ret = drv->API.GetScreensMESA(drv, disp, screens, max_screens, num_screens);
1095
1096 RETURN_EGL_EVAL(disp, ret);
1097 }
1098
1099
1100 EGLSurface EGLAPIENTRY
1101 eglCreateScreenSurfaceMESA(EGLDisplay dpy, EGLConfig config,
1102 const EGLint *attrib_list)
1103 {
1104 _EGLDisplay *disp = _eglLockDisplay(dpy);
1105 _EGLConfig *conf = _eglLookupConfig(config, disp);
1106 _EGLDriver *drv;
1107 _EGLSurface *surf;
1108 EGLSurface ret;
1109
1110 _EGL_CHECK_CONFIG(disp, conf, EGL_NO_SURFACE, drv);
1111
1112 surf = drv->API.CreateScreenSurfaceMESA(drv, disp, conf, attrib_list);
1113 ret = (surf) ? _eglLinkSurface(surf) : EGL_NO_SURFACE;
1114
1115 RETURN_EGL_EVAL(disp, ret);
1116 }
1117
1118
1119 EGLBoolean EGLAPIENTRY
1120 eglShowScreenSurfaceMESA(EGLDisplay dpy, EGLint screen,
1121 EGLSurface surface, EGLModeMESA mode)
1122 {
1123 _EGLDisplay *disp = _eglLockDisplay(dpy);
1124 _EGLScreen *scrn = _eglLookupScreen((EGLScreenMESA) screen, disp);
1125 _EGLSurface *surf = _eglLookupSurface(surface, disp);
1126 _EGLMode *m = _eglLookupMode(mode, disp);
1127 _EGLDriver *drv;
1128 EGLBoolean ret;
1129
1130 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1131 if (!surf && surface != EGL_NO_SURFACE)
1132 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
1133 if (!m && mode != EGL_NO_MODE_MESA)
1134 RETURN_EGL_ERROR(disp, EGL_BAD_MODE_MESA, EGL_FALSE);
1135
1136 ret = drv->API.ShowScreenSurfaceMESA(drv, disp, scrn, surf, m);
1137
1138 RETURN_EGL_EVAL(disp, ret);
1139 }
1140
1141
1142 EGLBoolean EGLAPIENTRY
1143 eglScreenPositionMESA(EGLDisplay dpy, EGLScreenMESA screen, EGLint x, EGLint y)
1144 {
1145 _EGLDisplay *disp = _eglLockDisplay(dpy);
1146 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1147 _EGLDriver *drv;
1148 EGLBoolean ret;
1149
1150 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1151 ret = drv->API.ScreenPositionMESA(drv, disp, scrn, x, y);
1152
1153 RETURN_EGL_EVAL(disp, ret);
1154 }
1155
1156
1157 EGLBoolean EGLAPIENTRY
1158 eglQueryScreenMESA(EGLDisplay dpy, EGLScreenMESA screen,
1159 EGLint attribute, EGLint *value)
1160 {
1161 _EGLDisplay *disp = _eglLockDisplay(dpy);
1162 _EGLScreen *scrn = _eglLookupScreen(screen, disp);
1163 _EGLDriver *drv;
1164 EGLBoolean ret;
1165
1166 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1167 ret = drv->API.QueryScreenMESA(drv, disp, scrn, attribute, value);
1168
1169 RETURN_EGL_EVAL(disp, ret);
1170 }
1171
1172
1173 EGLBoolean EGLAPIENTRY
1174 eglQueryScreenSurfaceMESA(EGLDisplay dpy, EGLScreenMESA screen,
1175 EGLSurface *surface)
1176 {
1177 _EGLDisplay *disp = _eglLockDisplay(dpy);
1178 _EGLScreen *scrn = _eglLookupScreen((EGLScreenMESA) screen, disp);
1179 _EGLDriver *drv;
1180 _EGLSurface *surf;
1181 EGLBoolean ret;
1182
1183 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1184 ret = drv->API.QueryScreenSurfaceMESA(drv, disp, scrn, &surf);
1185 if (ret && surface)
1186 *surface = _eglGetSurfaceHandle(surf);
1187
1188 RETURN_EGL_EVAL(disp, ret);
1189 }
1190
1191
1192 EGLBoolean EGLAPIENTRY
1193 eglQueryScreenModeMESA(EGLDisplay dpy, EGLScreenMESA screen, EGLModeMESA *mode)
1194 {
1195 _EGLDisplay *disp = _eglLockDisplay(dpy);
1196 _EGLScreen *scrn = _eglLookupScreen((EGLScreenMESA) screen, disp);
1197 _EGLDriver *drv;
1198 _EGLMode *m;
1199 EGLBoolean ret;
1200
1201 _EGL_CHECK_SCREEN(disp, scrn, EGL_FALSE, drv);
1202 ret = drv->API.QueryScreenModeMESA(drv, disp, scrn, &m);
1203 if (ret && mode)
1204 *mode = m->Handle;
1205
1206 RETURN_EGL_EVAL(disp, ret);
1207 }
1208
1209
1210 const char * EGLAPIENTRY
1211 eglQueryModeStringMESA(EGLDisplay dpy, EGLModeMESA mode)
1212 {
1213 _EGLDisplay *disp = _eglLockDisplay(dpy);
1214 _EGLMode *m = _eglLookupMode(mode, disp);
1215 _EGLDriver *drv;
1216 const char *ret;
1217
1218 _EGL_CHECK_MODE(disp, m, NULL, drv);
1219 ret = drv->API.QueryModeStringMESA(drv, disp, m);
1220
1221 RETURN_EGL_EVAL(disp, ret);
1222 }
1223
1224
1225 #endif /* EGL_MESA_screen_surface */
1226
1227
1228 #ifdef EGL_MESA_drm_display
1229
1230 EGLDisplay EGLAPIENTRY
1231 eglGetDRMDisplayMESA(int fd)
1232 {
1233 _EGLDisplay *dpy = _eglFindDisplay(_EGL_PLATFORM_DRM, (void *) (intptr_t) fd);
1234 return _eglGetDisplayHandle(dpy);
1235 }
1236
1237 #endif /* EGL_MESA_drm_display */
1238
1239 /**
1240 ** EGL 1.2
1241 **/
1242
1243 /**
1244 * Specify the client API to use for subsequent calls including:
1245 * eglCreateContext()
1246 * eglGetCurrentContext()
1247 * eglGetCurrentDisplay()
1248 * eglGetCurrentSurface()
1249 * eglMakeCurrent(when the ctx parameter is EGL NO CONTEXT)
1250 * eglWaitClient()
1251 * eglWaitNative()
1252 * See section 3.7 "Rendering Context" in the EGL specification for details.
1253 */
1254 EGLBoolean EGLAPIENTRY
1255 eglBindAPI(EGLenum api)
1256 {
1257 _EGLThreadInfo *t = _eglGetCurrentThread();
1258
1259 if (_eglIsCurrentThreadDummy())
1260 RETURN_EGL_ERROR(NULL, EGL_BAD_ALLOC, EGL_FALSE);
1261
1262 if (!_eglIsApiValid(api))
1263 RETURN_EGL_ERROR(NULL, EGL_BAD_PARAMETER, EGL_FALSE);
1264
1265 t->CurrentAPIIndex = _eglConvertApiToIndex(api);
1266
1267 RETURN_EGL_SUCCESS(NULL, EGL_TRUE);
1268 }
1269
1270
1271 /**
1272 * Return the last value set with eglBindAPI().
1273 */
1274 EGLenum EGLAPIENTRY
1275 eglQueryAPI(void)
1276 {
1277 _EGLThreadInfo *t = _eglGetCurrentThread();
1278 EGLenum ret;
1279
1280 /* returns one of EGL_OPENGL_API, EGL_OPENGL_ES_API or EGL_OPENVG_API */
1281 ret = _eglConvertApiFromIndex(t->CurrentAPIIndex);
1282
1283 RETURN_EGL_SUCCESS(NULL, ret);
1284 }
1285
1286
1287 EGLSurface EGLAPIENTRY
1288 eglCreatePbufferFromClientBuffer(EGLDisplay dpy, EGLenum buftype,
1289 EGLClientBuffer buffer, EGLConfig config,
1290 const EGLint *attrib_list)
1291 {
1292 _EGLDisplay *disp = _eglLockDisplay(dpy);
1293 _EGLConfig *conf = _eglLookupConfig(config, disp);
1294 _EGLDriver *drv;
1295 _EGLSurface *surf;
1296 EGLSurface ret;
1297
1298 _EGL_CHECK_CONFIG(disp, conf, EGL_NO_SURFACE, drv);
1299
1300 surf = drv->API.CreatePbufferFromClientBuffer(drv, disp, buftype, buffer,
1301 conf, attrib_list);
1302 ret = (surf) ? _eglLinkSurface(surf) : EGL_NO_SURFACE;
1303
1304 RETURN_EGL_EVAL(disp, ret);
1305 }
1306
1307
1308 EGLBoolean EGLAPIENTRY
1309 eglReleaseThread(void)
1310 {
1311 /* unbind current contexts */
1312 if (!_eglIsCurrentThreadDummy()) {
1313 _EGLThreadInfo *t = _eglGetCurrentThread();
1314 EGLint api_index = t->CurrentAPIIndex;
1315 EGLint i;
1316
1317 for (i = 0; i < _EGL_API_NUM_APIS; i++) {
1318 _EGLContext *ctx = t->CurrentContexts[i];
1319 if (ctx) {
1320 _EGLDisplay *disp = ctx->Resource.Display;
1321 _EGLDriver *drv;
1322
1323 t->CurrentAPIIndex = i;
1324
1325 _eglLockMutex(&disp->Mutex);
1326 drv = disp->Driver;
1327 (void) drv->API.MakeCurrent(drv, disp, NULL, NULL, NULL);
1328 _eglUnlockMutex(&disp->Mutex);
1329 }
1330 }
1331
1332 t->CurrentAPIIndex = api_index;
1333 }
1334
1335 _eglDestroyCurrentThread();
1336
1337 RETURN_EGL_SUCCESS(NULL, EGL_TRUE);
1338 }
1339
1340
1341 EGLImageKHR EGLAPIENTRY
1342 eglCreateImageKHR(EGLDisplay dpy, EGLContext ctx, EGLenum target,
1343 EGLClientBuffer buffer, const EGLint *attr_list)
1344 {
1345 _EGLDisplay *disp = _eglLockDisplay(dpy);
1346 _EGLContext *context = _eglLookupContext(ctx, disp);
1347 _EGLDriver *drv;
1348 _EGLImage *img;
1349 EGLImageKHR ret;
1350
1351 _EGL_CHECK_DISPLAY(disp, EGL_NO_IMAGE_KHR, drv);
1352 if (!disp->Extensions.KHR_image_base)
1353 RETURN_EGL_EVAL(disp, EGL_NO_IMAGE_KHR);
1354 if (!context && ctx != EGL_NO_CONTEXT)
1355 RETURN_EGL_ERROR(disp, EGL_BAD_CONTEXT, EGL_NO_IMAGE_KHR);
1356 /* "If <target> is EGL_LINUX_DMA_BUF_EXT, <dpy> must be a valid display,
1357 * <ctx> must be EGL_NO_CONTEXT..."
1358 */
1359 if (ctx != EGL_NO_CONTEXT && target == EGL_LINUX_DMA_BUF_EXT)
1360 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_NO_IMAGE_KHR);
1361
1362 img = drv->API.CreateImageKHR(drv,
1363 disp, context, target, buffer, attr_list);
1364 ret = (img) ? _eglLinkImage(img) : EGL_NO_IMAGE_KHR;
1365
1366 RETURN_EGL_EVAL(disp, ret);
1367 }
1368
1369
1370 EGLBoolean EGLAPIENTRY
1371 eglDestroyImageKHR(EGLDisplay dpy, EGLImageKHR image)
1372 {
1373 _EGLDisplay *disp = _eglLockDisplay(dpy);
1374 _EGLImage *img = _eglLookupImage(image, disp);
1375 _EGLDriver *drv;
1376 EGLBoolean ret;
1377
1378 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1379 if (!disp->Extensions.KHR_image_base)
1380 RETURN_EGL_EVAL(disp, EGL_FALSE);
1381 if (!img)
1382 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1383
1384 _eglUnlinkImage(img);
1385 ret = drv->API.DestroyImageKHR(drv, disp, img);
1386
1387 RETURN_EGL_EVAL(disp, ret);
1388 }
1389
1390
1391 EGLSyncKHR EGLAPIENTRY
1392 eglCreateSyncKHR(EGLDisplay dpy, EGLenum type, const EGLint *attrib_list)
1393 {
1394 _EGLDisplay *disp = _eglLockDisplay(dpy);
1395 _EGLDriver *drv;
1396 _EGLSync *sync;
1397 EGLSyncKHR ret;
1398
1399 _EGL_CHECK_DISPLAY(disp, EGL_NO_SYNC_KHR, drv);
1400 if (!disp->Extensions.KHR_reusable_sync)
1401 RETURN_EGL_EVAL(disp, EGL_NO_SYNC_KHR);
1402
1403 sync = drv->API.CreateSyncKHR(drv, disp, type, attrib_list);
1404 ret = (sync) ? _eglLinkSync(sync) : EGL_NO_SYNC_KHR;
1405
1406 RETURN_EGL_EVAL(disp, ret);
1407 }
1408
1409
1410 EGLBoolean EGLAPIENTRY
1411 eglDestroySyncKHR(EGLDisplay dpy, EGLSyncKHR sync)
1412 {
1413 _EGLDisplay *disp = _eglLockDisplay(dpy);
1414 _EGLSync *s = _eglLookupSync(sync, disp);
1415 _EGLDriver *drv;
1416 EGLBoolean ret;
1417
1418 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1419 assert(disp->Extensions.KHR_reusable_sync);
1420
1421 _eglUnlinkSync(s);
1422 ret = drv->API.DestroySyncKHR(drv, disp, s);
1423
1424 RETURN_EGL_EVAL(disp, ret);
1425 }
1426
1427
1428 EGLint EGLAPIENTRY
1429 eglClientWaitSyncKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLint flags, EGLTimeKHR timeout)
1430 {
1431 _EGLDisplay *disp = _eglLockDisplay(dpy);
1432 _EGLSync *s = _eglLookupSync(sync, disp);
1433 _EGLDriver *drv;
1434 EGLint ret;
1435
1436 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1437 assert(disp->Extensions.KHR_reusable_sync);
1438 ret = drv->API.ClientWaitSyncKHR(drv, disp, s, flags, timeout);
1439
1440 RETURN_EGL_EVAL(disp, ret);
1441 }
1442
1443
1444 EGLBoolean EGLAPIENTRY
1445 eglSignalSyncKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLenum mode)
1446 {
1447 _EGLDisplay *disp = _eglLockDisplay(dpy);
1448 _EGLSync *s = _eglLookupSync(sync, disp);
1449 _EGLDriver *drv;
1450 EGLBoolean ret;
1451
1452 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1453 assert(disp->Extensions.KHR_reusable_sync);
1454 ret = drv->API.SignalSyncKHR(drv, disp, s, mode);
1455
1456 RETURN_EGL_EVAL(disp, ret);
1457 }
1458
1459
1460 EGLBoolean EGLAPIENTRY
1461 eglGetSyncAttribKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLint attribute, EGLint *value)
1462 {
1463 _EGLDisplay *disp = _eglLockDisplay(dpy);
1464 _EGLSync *s = _eglLookupSync(sync, disp);
1465 _EGLDriver *drv;
1466 EGLBoolean ret;
1467
1468 _EGL_CHECK_SYNC(disp, s, EGL_FALSE, drv);
1469 assert(disp->Extensions.KHR_reusable_sync);
1470 ret = drv->API.GetSyncAttribKHR(drv, disp, s, attribute, value);
1471
1472 RETURN_EGL_EVAL(disp, ret);
1473 }
1474
1475
1476 #ifdef EGL_NOK_swap_region
1477
1478 EGLBoolean EGLAPIENTRY
1479 eglSwapBuffersRegionNOK(EGLDisplay dpy, EGLSurface surface,
1480 EGLint numRects, const EGLint *rects)
1481 {
1482 _EGLContext *ctx = _eglGetCurrentContext();
1483 _EGLDisplay *disp = _eglLockDisplay(dpy);
1484 _EGLSurface *surf = _eglLookupSurface(surface, disp);
1485 _EGLDriver *drv;
1486 EGLBoolean ret;
1487
1488 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
1489
1490 if (!disp->Extensions.NOK_swap_region)
1491 RETURN_EGL_EVAL(disp, EGL_FALSE);
1492
1493 /* surface must be bound to current context in EGL 1.4 */
1494 if (_eglGetContextHandle(ctx) == EGL_NO_CONTEXT ||
1495 surf != ctx->DrawSurface)
1496 RETURN_EGL_ERROR(disp, EGL_BAD_SURFACE, EGL_FALSE);
1497
1498 ret = drv->API.SwapBuffersRegionNOK(drv, disp, surf, numRects, rects);
1499
1500 RETURN_EGL_EVAL(disp, ret);
1501 }
1502
1503 #endif /* EGL_NOK_swap_region */
1504
1505
1506 #ifdef EGL_MESA_drm_image
1507
1508 EGLImageKHR EGLAPIENTRY
1509 eglCreateDRMImageMESA(EGLDisplay dpy, const EGLint *attr_list)
1510 {
1511 _EGLDisplay *disp = _eglLockDisplay(dpy);
1512 _EGLDriver *drv;
1513 _EGLImage *img;
1514 EGLImageKHR ret;
1515
1516 _EGL_CHECK_DISPLAY(disp, EGL_NO_IMAGE_KHR, drv);
1517 if (!disp->Extensions.MESA_drm_image)
1518 RETURN_EGL_EVAL(disp, EGL_NO_IMAGE_KHR);
1519
1520 img = drv->API.CreateDRMImageMESA(drv, disp, attr_list);
1521 ret = (img) ? _eglLinkImage(img) : EGL_NO_IMAGE_KHR;
1522
1523 RETURN_EGL_EVAL(disp, ret);
1524 }
1525
1526 EGLBoolean EGLAPIENTRY
1527 eglExportDRMImageMESA(EGLDisplay dpy, EGLImageKHR image,
1528 EGLint *name, EGLint *handle, EGLint *stride)
1529 {
1530 _EGLDisplay *disp = _eglLockDisplay(dpy);
1531 _EGLImage *img = _eglLookupImage(image, disp);
1532 _EGLDriver *drv;
1533 EGLBoolean ret;
1534
1535 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1536 assert(disp->Extensions.MESA_drm_image);
1537
1538 if (!img)
1539 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1540
1541 ret = drv->API.ExportDRMImageMESA(drv, disp, img, name, handle, stride);
1542
1543 RETURN_EGL_EVAL(disp, ret);
1544 }
1545
1546 #endif
1547
1548 #ifdef EGL_WL_bind_wayland_display
1549 struct wl_display;
1550
1551 EGLBoolean EGLAPIENTRY
1552 eglBindWaylandDisplayWL(EGLDisplay dpy, struct wl_display *display)
1553 {
1554 _EGLDisplay *disp = _eglLockDisplay(dpy);
1555 _EGLDriver *drv;
1556 EGLBoolean ret;
1557
1558 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1559 assert(disp->Extensions.WL_bind_wayland_display);
1560
1561 if (!display)
1562 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1563
1564 ret = drv->API.BindWaylandDisplayWL(drv, disp, display);
1565
1566 RETURN_EGL_EVAL(disp, ret);
1567 }
1568
1569 EGLBoolean EGLAPIENTRY
1570 eglUnbindWaylandDisplayWL(EGLDisplay dpy, struct wl_display *display)
1571 {
1572 _EGLDisplay *disp = _eglLockDisplay(dpy);
1573 _EGLDriver *drv;
1574 EGLBoolean ret;
1575
1576 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1577 assert(disp->Extensions.WL_bind_wayland_display);
1578
1579 if (!display)
1580 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1581
1582 ret = drv->API.UnbindWaylandDisplayWL(drv, disp, display);
1583
1584 RETURN_EGL_EVAL(disp, ret);
1585 }
1586
1587 EGLBoolean EGLAPIENTRY
1588 eglQueryWaylandBufferWL(EGLDisplay dpy, struct wl_resource *buffer,
1589 EGLint attribute, EGLint *value)
1590 {
1591 _EGLDisplay *disp = _eglLockDisplay(dpy);
1592 _EGLDriver *drv;
1593 EGLBoolean ret;
1594
1595 _EGL_CHECK_DISPLAY(disp, EGL_FALSE, drv);
1596 assert(disp->Extensions.WL_bind_wayland_display);
1597
1598 if (!buffer)
1599 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, EGL_FALSE);
1600
1601 ret = drv->API.QueryWaylandBufferWL(drv, disp, buffer, attribute, value);
1602
1603 RETURN_EGL_EVAL(disp, ret);
1604 }
1605 #endif
1606
1607 #ifdef EGL_WL_create_wayland_buffer_from_image
1608 struct wl_buffer * EGLAPIENTRY
1609 eglCreateWaylandBufferFromImageWL(EGLDisplay dpy, EGLImageKHR image)
1610 {
1611 _EGLDisplay *disp = _eglLockDisplay(dpy);
1612 _EGLImage *img;
1613 _EGLDriver *drv;
1614 struct wl_buffer *ret;
1615
1616 _EGL_CHECK_DISPLAY(disp, NULL, drv);
1617 assert(disp->Extensions.WL_create_wayland_buffer_from_image);
1618
1619 img = _eglLookupImage(image, disp);
1620
1621 if (!img)
1622 RETURN_EGL_ERROR(disp, EGL_BAD_PARAMETER, NULL);
1623
1624 ret = drv->API.CreateWaylandBufferFromImageWL(drv, disp, img);
1625
1626 RETURN_EGL_EVAL(disp, ret);
1627 }
1628 #endif
1629
1630 EGLBoolean EGLAPIENTRY
1631 eglPostSubBufferNV(EGLDisplay dpy, EGLSurface surface,
1632 EGLint x, EGLint y, EGLint width, EGLint height)
1633 {
1634 _EGLDisplay *disp = _eglLockDisplay(dpy);
1635 _EGLSurface *surf = _eglLookupSurface(surface, disp);
1636 _EGLDriver *drv;
1637 EGLBoolean ret;
1638
1639 _EGL_CHECK_SURFACE(disp, surf, EGL_FALSE, drv);
1640
1641 if (!disp->Extensions.NV_post_sub_buffer)
1642 RETURN_EGL_EVAL(disp, EGL_FALSE);
1643
1644 ret = drv->API.PostSubBufferNV(drv, disp, surf, x, y, width, height);
1645
1646 RETURN_EGL_EVAL(disp, ret);
1647 }