1 //
2 // Copyright 2020 The ANGLE Project Authors. All rights reserved.
3 // Use of this source code is governed by a BSD-style license that can be
4 // found in the LICENSE file.
5 //
6 // egl_stubs.cpp: Stubs for EGL entry points.
7 //
8
9 #include "libGLESv2/egl_stubs_autogen.h"
10
11 #include "common/angle_version_info.h"
12 #include "libANGLE/Context.h"
13 #include "libANGLE/Display.h"
14 #include "libANGLE/EGLSync.h"
15 #include "libANGLE/Surface.h"
16 #include "libANGLE/Thread.h"
17 #include "libANGLE/queryutils.h"
18 #include "libANGLE/validationEGL.h"
19 #include "libGLESv2/global_state.h"
20 #include "libGLESv2/proc_table_egl.h"
21
22 namespace egl
23 {
24 namespace
25 {
26
CompareProc(const ProcEntry & a,const char * b)27 bool CompareProc(const ProcEntry &a, const char *b)
28 {
29 return strcmp(a.first, b) < 0;
30 }
31
ClipConfigs(const std::vector<const Config * > & filteredConfigs,EGLConfig * outputConfigs,EGLint configSize,EGLint * numConfigs)32 void ClipConfigs(const std::vector<const Config *> &filteredConfigs,
33 EGLConfig *outputConfigs,
34 EGLint configSize,
35 EGLint *numConfigs)
36 {
37 EGLint resultSize = static_cast<EGLint>(filteredConfigs.size());
38 if (outputConfigs)
39 {
40 resultSize = std::max(std::min(resultSize, configSize), 0);
41 for (EGLint i = 0; i < resultSize; i++)
42 {
43 outputConfigs[i] = const_cast<Config *>(filteredConfigs[i]);
44 }
45 }
46 *numConfigs = resultSize;
47 }
48 } // anonymous namespace
49
BindAPI(Thread * thread,EGLenum api)50 EGLBoolean BindAPI(Thread *thread, EGLenum api)
51 {
52 thread->setAPI(api);
53
54 thread->setSuccess();
55 return EGL_TRUE;
56 }
57
BindTexImage(Thread * thread,Display * display,Surface * eglSurface,EGLint buffer)58 EGLBoolean BindTexImage(Thread *thread, Display *display, Surface *eglSurface, EGLint buffer)
59 {
60 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglBindTexImage",
61 GetDisplayIfValid(display), EGL_FALSE);
62
63 gl::Context *context = thread->getContext();
64 if (context)
65 {
66 gl::TextureType type =
67 egl_gl::EGLTextureTargetToTextureType(eglSurface->getTextureTarget());
68 gl::Texture *textureObject = context->getTextureByType(type);
69 ANGLE_EGL_TRY_RETURN(thread, eglSurface->bindTexImage(context, textureObject, buffer),
70 "eglBindTexImage", GetSurfaceIfValid(display, eglSurface), EGL_FALSE);
71 }
72
73 thread->setSuccess();
74 return EGL_TRUE;
75 }
76
ChooseConfig(Thread * thread,Display * display,const AttributeMap & attribMap,EGLConfig * configs,EGLint config_size,EGLint * num_config)77 EGLBoolean ChooseConfig(Thread *thread,
78 Display *display,
79 const AttributeMap &attribMap,
80 EGLConfig *configs,
81 EGLint config_size,
82 EGLint *num_config)
83 {
84 ClipConfigs(display->chooseConfig(attribMap), configs, config_size, num_config);
85
86 thread->setSuccess();
87 return EGL_TRUE;
88 }
89
ClientWaitSync(Thread * thread,Display * display,Sync * syncObject,EGLint flags,EGLTime timeout)90 EGLint ClientWaitSync(Thread *thread,
91 Display *display,
92 Sync *syncObject,
93 EGLint flags,
94 EGLTime timeout)
95 {
96 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglClientWaitSync",
97 GetDisplayIfValid(display), EGL_FALSE);
98 gl::Context *currentContext = thread->getContext();
99 EGLint syncStatus = EGL_FALSE;
100 ANGLE_EGL_TRY_RETURN(
101 thread, syncObject->clientWait(display, currentContext, flags, timeout, &syncStatus),
102 "eglClientWaitSync", GetSyncIfValid(display, syncObject), EGL_FALSE);
103
104 thread->setSuccess();
105 return syncStatus;
106 }
107
CopyBuffers(Thread * thread,Display * display,Surface * eglSurface,EGLNativePixmapType target)108 EGLBoolean CopyBuffers(Thread *thread,
109 Display *display,
110 Surface *eglSurface,
111 EGLNativePixmapType target)
112 {
113 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCopyBuffers",
114 GetDisplayIfValid(display), EGL_FALSE);
115 UNIMPLEMENTED(); // FIXME
116
117 thread->setSuccess();
118 return 0;
119 }
120
CreateContext(Thread * thread,Display * display,Config * configuration,gl::Context * sharedGLContext,const AttributeMap & attributes)121 EGLContext CreateContext(Thread *thread,
122 Display *display,
123 Config *configuration,
124 gl::Context *sharedGLContext,
125 const AttributeMap &attributes)
126 {
127 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreateContext",
128 GetDisplayIfValid(display), EGL_NO_CONTEXT);
129 gl::Context *context = nullptr;
130 ANGLE_EGL_TRY_RETURN(thread,
131 display->createContext(configuration, sharedGLContext, thread->getAPI(),
132 attributes, &context),
133 "eglCreateContext", GetDisplayIfValid(display), EGL_NO_CONTEXT);
134
135 thread->setSuccess();
136 return static_cast<EGLContext>(context);
137 }
138
CreateImage(Thread * thread,Display * display,gl::Context * context,EGLenum target,EGLClientBuffer buffer,const AttributeMap & attributes)139 EGLImage CreateImage(Thread *thread,
140 Display *display,
141 gl::Context *context,
142 EGLenum target,
143 EGLClientBuffer buffer,
144 const AttributeMap &attributes)
145 {
146 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreateImage",
147 GetDisplayIfValid(display), EGL_FALSE);
148
149 Image *image = nullptr;
150 Error error = display->createImage(context, target, buffer, attributes, &image);
151 if (error.isError())
152 {
153 thread->setError(error, "eglCreateImage", GetDisplayIfValid(display));
154 return EGL_NO_IMAGE;
155 }
156
157 thread->setSuccess();
158 return static_cast<EGLImage>(image);
159 }
160
CreatePbufferFromClientBuffer(Thread * thread,Display * display,EGLenum buftype,EGLClientBuffer buffer,Config * configuration,const AttributeMap & attributes)161 EGLSurface CreatePbufferFromClientBuffer(Thread *thread,
162 Display *display,
163 EGLenum buftype,
164 EGLClientBuffer buffer,
165 Config *configuration,
166 const AttributeMap &attributes)
167 {
168 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreatePbufferFromClientBuffer",
169 GetDisplayIfValid(display), EGL_NO_SURFACE);
170 Surface *surface = nullptr;
171 ANGLE_EGL_TRY_RETURN(thread,
172 display->createPbufferFromClientBuffer(configuration, buftype, buffer,
173 attributes, &surface),
174 "eglCreatePbufferFromClientBuffer", GetDisplayIfValid(display),
175 EGL_NO_SURFACE);
176
177 return static_cast<EGLSurface>(surface);
178 }
179
CreatePbufferSurface(Thread * thread,Display * display,Config * configuration,const AttributeMap & attributes)180 EGLSurface CreatePbufferSurface(Thread *thread,
181 Display *display,
182 Config *configuration,
183 const AttributeMap &attributes)
184 {
185 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreatePbufferSurface",
186 GetDisplayIfValid(display), EGL_NO_SURFACE);
187 Surface *surface = nullptr;
188 ANGLE_EGL_TRY_RETURN(thread, display->createPbufferSurface(configuration, attributes, &surface),
189 "eglCreatePbufferSurface", GetDisplayIfValid(display), EGL_NO_SURFACE);
190
191 return static_cast<EGLSurface>(surface);
192 }
193
CreatePixmapSurface(Thread * thread,Display * display,Config * configuration,EGLNativePixmapType pixmap,const AttributeMap & attributes)194 EGLSurface CreatePixmapSurface(Thread *thread,
195 Display *display,
196 Config *configuration,
197 EGLNativePixmapType pixmap,
198 const AttributeMap &attributes)
199 {
200 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreatePixmapSurface",
201 GetDisplayIfValid(display), EGL_NO_SURFACE);
202 Surface *surface = nullptr;
203 ANGLE_EGL_TRY_RETURN(thread,
204 display->createPixmapSurface(configuration, pixmap, attributes, &surface),
205 "eglCreatePixmapSurface", GetDisplayIfValid(display), EGL_NO_SURFACE);
206
207 thread->setSuccess();
208 return static_cast<EGLSurface>(surface);
209 }
210
CreatePlatformPixmapSurface(Thread * thread,Display * display,Config * configuration,void * pixmap,const AttributeMap & attributes)211 EGLSurface CreatePlatformPixmapSurface(Thread *thread,
212 Display *display,
213 Config *configuration,
214 void *pixmap,
215 const AttributeMap &attributes)
216 {
217 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreatePlatformPixmapSurface",
218 GetDisplayIfValid(display), EGL_NO_SURFACE);
219 Surface *surface = nullptr;
220 EGLNativePixmapType nativePixmap = reinterpret_cast<EGLNativePixmapType>(pixmap);
221 ANGLE_EGL_TRY_RETURN(
222 thread, display->createPixmapSurface(configuration, nativePixmap, attributes, &surface),
223 "eglCreatePlatformPixmapSurface", GetDisplayIfValid(display), EGL_NO_SURFACE);
224
225 thread->setSuccess();
226 return static_cast<EGLSurface>(surface);
227 }
228
CreatePlatformWindowSurface(Thread * thread,Display * display,Config * configuration,void * win,const AttributeMap & attributes)229 EGLSurface CreatePlatformWindowSurface(Thread *thread,
230 Display *display,
231 Config *configuration,
232 void *win,
233 const AttributeMap &attributes)
234 {
235 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreatePlatformWindowSurface",
236 GetDisplayIfValid(display), EGL_NO_SURFACE);
237 Surface *surface = nullptr;
238 EGLNativeWindowType nativeWindow = reinterpret_cast<EGLNativeWindowType>(win);
239 ANGLE_EGL_TRY_RETURN(
240 thread, display->createWindowSurface(configuration, nativeWindow, attributes, &surface),
241 "eglPlatformCreateWindowSurface", GetDisplayIfValid(display), EGL_NO_SURFACE);
242
243 return static_cast<EGLSurface>(surface);
244 }
245
CreateSync(Thread * thread,Display * display,EGLenum type,const AttributeMap & attributes)246 EGLSync CreateSync(Thread *thread, Display *display, EGLenum type, const AttributeMap &attributes)
247 {
248 gl::Context *currentContext = thread->getContext();
249
250 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreateSync",
251 GetDisplayIfValid(display), EGL_FALSE);
252 Sync *syncObject = nullptr;
253 ANGLE_EGL_TRY_RETURN(thread, display->createSync(currentContext, type, attributes, &syncObject),
254 "eglCreateSync", GetDisplayIfValid(display), EGL_NO_SYNC);
255
256 thread->setSuccess();
257 return static_cast<EGLSync>(syncObject);
258 }
259
CreateWindowSurface(Thread * thread,Display * display,Config * configuration,EGLNativeWindowType win,const AttributeMap & attributes)260 EGLSurface CreateWindowSurface(Thread *thread,
261 Display *display,
262 Config *configuration,
263 EGLNativeWindowType win,
264 const AttributeMap &attributes)
265 {
266 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglCreateWindowSurface",
267 GetDisplayIfValid(display), EGL_NO_SURFACE);
268
269 Surface *surface = nullptr;
270 ANGLE_EGL_TRY_RETURN(thread,
271 display->createWindowSurface(configuration, win, attributes, &surface),
272 "eglCreateWindowSurface", GetDisplayIfValid(display), EGL_NO_SURFACE);
273
274 return static_cast<EGLSurface>(surface);
275 }
276
DestroyContext(Thread * thread,Display * display,gl::Context * context)277 EGLBoolean DestroyContext(Thread *thread, Display *display, gl::Context *context)
278 {
279 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglDestroyContext",
280 GetDisplayIfValid(display), EGL_FALSE);
281
282 ScopedSyncCurrentContextFromThread scopedSyncCurrent(thread);
283
284 ANGLE_EGL_TRY_RETURN(thread, display->destroyContext(thread, context), "eglDestroyContext",
285 GetContextIfValid(display, context), EGL_FALSE);
286 thread->setSuccess();
287 return EGL_TRUE;
288 }
289
DestroyImage(Thread * thread,Display * display,Image * img)290 EGLBoolean DestroyImage(Thread *thread, Display *display, Image *img)
291 {
292 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglDestroyImage",
293 GetDisplayIfValid(display), EGL_FALSE);
294 display->destroyImage(img);
295
296 thread->setSuccess();
297 return EGL_TRUE;
298 }
299
DestroySurface(Thread * thread,Display * display,Surface * eglSurface)300 EGLBoolean DestroySurface(Thread *thread, Display *display, Surface *eglSurface)
301 {
302 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglDestroySurface",
303 GetDisplayIfValid(display), EGL_FALSE);
304 ANGLE_EGL_TRY_RETURN(thread, display->destroySurface(eglSurface), "eglDestroySurface",
305 GetSurfaceIfValid(display, eglSurface), EGL_FALSE);
306
307 thread->setSuccess();
308 return EGL_TRUE;
309 }
310
DestroySync(Thread * thread,Display * display,Sync * syncObject)311 EGLBoolean DestroySync(Thread *thread, Display *display, Sync *syncObject)
312 {
313 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglDestroySync",
314 GetDisplayIfValid(display), EGL_FALSE);
315 display->destroySync(syncObject);
316
317 thread->setSuccess();
318 return EGL_TRUE;
319 }
320
GetConfigAttrib(Thread * thread,Display * display,Config * configuration,EGLint attribute,EGLint * value)321 EGLBoolean GetConfigAttrib(Thread *thread,
322 Display *display,
323 Config *configuration,
324 EGLint attribute,
325 EGLint *value)
326 {
327 QueryConfigAttrib(configuration, attribute, value);
328
329 thread->setSuccess();
330 return EGL_TRUE;
331 }
332
GetConfigs(Thread * thread,Display * display,EGLConfig * configs,EGLint config_size,EGLint * num_config)333 EGLBoolean GetConfigs(Thread *thread,
334 Display *display,
335 EGLConfig *configs,
336 EGLint config_size,
337 EGLint *num_config)
338 {
339 ClipConfigs(display->getConfigs(AttributeMap()), configs, config_size, num_config);
340
341 thread->setSuccess();
342 return EGL_TRUE;
343 }
344
GetCurrentContext(Thread * thread)345 EGLContext GetCurrentContext(Thread *thread)
346 {
347 gl::Context *context = thread->getContext();
348
349 thread->setSuccess();
350 return static_cast<EGLContext>(context);
351 }
352
GetCurrentDisplay(Thread * thread)353 EGLDisplay GetCurrentDisplay(Thread *thread)
354 {
355 thread->setSuccess();
356 if (thread->getContext() != nullptr)
357 {
358 return thread->getContext()->getDisplay();
359 }
360 return EGL_NO_DISPLAY;
361 }
362
GetCurrentSurface(Thread * thread,EGLint readdraw)363 EGLSurface GetCurrentSurface(Thread *thread, EGLint readdraw)
364 {
365 if (readdraw == EGL_READ)
366 {
367 thread->setSuccess();
368 return thread->getCurrentReadSurface();
369 }
370 else if (readdraw == EGL_DRAW)
371 {
372 thread->setSuccess();
373 return thread->getCurrentDrawSurface();
374 }
375 else
376 {
377 thread->setError(EglBadParameter(), "eglGetCurrentSurface", nullptr);
378 return EGL_NO_SURFACE;
379 }
380 }
381
GetDisplay(Thread * thread,EGLNativeDisplayType display_id)382 EGLDisplay GetDisplay(Thread *thread, EGLNativeDisplayType display_id)
383 {
384 return Display::GetDisplayFromNativeDisplay(display_id, AttributeMap());
385 }
386
GetError(Thread * thread)387 EGLint GetError(Thread *thread)
388 {
389 EGLint error = thread->getError();
390 thread->setSuccess();
391 return error;
392 }
393
GetPlatformDisplay(Thread * thread,EGLenum platform,void * native_display,const AttributeMap & attribMap)394 EGLDisplay GetPlatformDisplay(Thread *thread,
395 EGLenum platform,
396 void *native_display,
397 const AttributeMap &attribMap)
398 {
399 if (platform == EGL_PLATFORM_ANGLE_ANGLE)
400 {
401 return Display::GetDisplayFromNativeDisplay(
402 gl::bitCast<EGLNativeDisplayType>(native_display), attribMap);
403 }
404 else if (platform == EGL_PLATFORM_DEVICE_EXT)
405 {
406 Device *eglDevice = static_cast<Device *>(native_display);
407 return Display::GetDisplayFromDevice(eglDevice, attribMap);
408 }
409 else
410 {
411 UNREACHABLE();
412 return EGL_NO_DISPLAY;
413 }
414 }
415
GetProcAddress(Thread * thread,const char * procname)416 __eglMustCastToProperFunctionPointerType GetProcAddress(Thread *thread, const char *procname)
417 {
418 const ProcEntry *entry =
419 std::lower_bound(&g_procTable[0], &g_procTable[g_numProcs], procname, CompareProc);
420
421 thread->setSuccess();
422
423 if (entry == &g_procTable[g_numProcs] || strcmp(entry->first, procname) != 0)
424 {
425 return nullptr;
426 }
427
428 return entry->second;
429 }
430
GetSyncAttrib(Thread * thread,Display * display,Sync * syncObject,EGLint attribute,EGLAttrib * value)431 EGLBoolean GetSyncAttrib(Thread *thread,
432 Display *display,
433 Sync *syncObject,
434 EGLint attribute,
435 EGLAttrib *value)
436 {
437 EGLint valueExt;
438 ANGLE_EGL_TRY_RETURN(thread, GetSyncAttrib(display, syncObject, attribute, &valueExt),
439 "eglGetSyncAttrib", GetSyncIfValid(display, syncObject), EGL_FALSE);
440 *value = valueExt;
441
442 thread->setSuccess();
443 return EGL_TRUE;
444 }
445
Initialize(Thread * thread,Display * display,EGLint * major,EGLint * minor)446 EGLBoolean Initialize(Thread *thread, Display *display, EGLint *major, EGLint *minor)
447 {
448 ANGLE_EGL_TRY_RETURN(thread, display->initialize(), "eglInitialize", GetDisplayIfValid(display),
449 EGL_FALSE);
450
451 if (major)
452 *major = 1;
453 if (minor)
454 *minor = 5;
455
456 thread->setSuccess();
457 return EGL_TRUE;
458 }
459
MakeCurrent(Thread * thread,Display * display,Surface * drawSurface,Surface * readSurface,gl::Context * context)460 EGLBoolean MakeCurrent(Thread *thread,
461 Display *display,
462 Surface *drawSurface,
463 Surface *readSurface,
464 gl::Context *context)
465 {
466 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglMakeCurrent",
467 GetDisplayIfValid(display), EGL_FALSE);
468 ScopedSyncCurrentContextFromThread scopedSyncCurrent(thread);
469
470 Surface *previousDraw = thread->getCurrentDrawSurface();
471 Surface *previousRead = thread->getCurrentReadSurface();
472 gl::Context *previousContext = thread->getContext();
473
474 // Only call makeCurrent if the context or surfaces have changed.
475 if (previousDraw != drawSurface || previousRead != readSurface || previousContext != context)
476 {
477 ANGLE_EGL_TRY_RETURN(
478 thread,
479 display->makeCurrent(thread, previousContext, drawSurface, readSurface, context),
480 "eglMakeCurrent", GetContextIfValid(display, context), EGL_FALSE);
481 }
482
483 thread->setSuccess();
484 return EGL_TRUE;
485 }
486
QueryAPI(Thread * thread)487 EGLenum QueryAPI(Thread *thread)
488 {
489 EGLenum API = thread->getAPI();
490
491 thread->setSuccess();
492 return API;
493 }
494
QueryContext(Thread * thread,Display * display,gl::Context * context,EGLint attribute,EGLint * value)495 EGLBoolean QueryContext(Thread *thread,
496 Display *display,
497 gl::Context *context,
498 EGLint attribute,
499 EGLint *value)
500 {
501 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglQueryContext",
502 GetDisplayIfValid(display), EGL_FALSE);
503 QueryContextAttrib(context, attribute, value);
504
505 thread->setSuccess();
506 return EGL_TRUE;
507 }
508
QueryString(Thread * thread,Display * display,EGLint name)509 const char *QueryString(Thread *thread, Display *display, EGLint name)
510 {
511 if (display)
512 {
513 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglQueryString",
514 GetDisplayIfValid(display), nullptr);
515 }
516
517 const char *result = nullptr;
518 switch (name)
519 {
520 case EGL_CLIENT_APIS:
521 result = "OpenGL_ES";
522 break;
523 case EGL_EXTENSIONS:
524 if (display == EGL_NO_DISPLAY)
525 {
526 result = Display::GetClientExtensionString().c_str();
527 }
528 else
529 {
530 result = display->getExtensionString().c_str();
531 }
532 break;
533 case EGL_VENDOR:
534 result = display->getVendorString().c_str();
535 break;
536 case EGL_VERSION:
537 {
538 static const char *sVersionString =
539 MakeStaticString(std::string("1.5 (ANGLE ") + angle::GetANGLEVersionString() + ")");
540 result = sVersionString;
541 break;
542 }
543 default:
544 UNREACHABLE();
545 break;
546 }
547
548 thread->setSuccess();
549 return result;
550 }
551
QuerySurface(Thread * thread,Display * display,Surface * eglSurface,EGLint attribute,EGLint * value)552 EGLBoolean QuerySurface(Thread *thread,
553 Display *display,
554 Surface *eglSurface,
555 EGLint attribute,
556 EGLint *value)
557 {
558 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglQuerySurface",
559 GetDisplayIfValid(display), EGL_FALSE);
560 ANGLE_EGL_TRY_RETURN(
561 thread, QuerySurfaceAttrib(display, thread->getContext(), eglSurface, attribute, value),
562 "eglQuerySurface", GetSurfaceIfValid(display, eglSurface), EGL_FALSE);
563
564 thread->setSuccess();
565 return EGL_TRUE;
566 }
567
ReleaseTexImage(Thread * thread,Display * display,Surface * eglSurface,EGLint buffer)568 EGLBoolean ReleaseTexImage(Thread *thread, Display *display, Surface *eglSurface, EGLint buffer)
569 {
570 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglReleaseTexImage",
571 GetDisplayIfValid(display), EGL_FALSE);
572 gl::Texture *texture = eglSurface->getBoundTexture();
573
574 if (texture)
575 {
576 ANGLE_EGL_TRY_RETURN(thread, eglSurface->releaseTexImage(thread->getContext(), buffer),
577 "eglReleaseTexImage", GetSurfaceIfValid(display, eglSurface),
578 EGL_FALSE);
579 }
580
581 thread->setSuccess();
582 return EGL_TRUE;
583 }
584
ReleaseThread(Thread * thread)585 EGLBoolean ReleaseThread(Thread *thread)
586 {
587 ScopedSyncCurrentContextFromThread scopedSyncCurrent(thread);
588
589 Surface *previousDraw = thread->getCurrentDrawSurface();
590 Surface *previousRead = thread->getCurrentReadSurface();
591 gl::Context *previousContext = thread->getContext();
592 Display *previousDisplay = thread->getDisplay();
593
594 if (previousDisplay != EGL_NO_DISPLAY)
595 {
596 ANGLE_EGL_TRY_RETURN(thread, previousDisplay->prepareForCall(), "eglReleaseThread",
597 GetDisplayIfValid(previousDisplay), EGL_FALSE);
598 // Only call makeCurrent if the context or surfaces have changed.
599 if (previousDraw != EGL_NO_SURFACE || previousRead != EGL_NO_SURFACE ||
600 previousContext != EGL_NO_CONTEXT)
601 {
602 ANGLE_EGL_TRY_RETURN(
603 thread,
604 previousDisplay->makeCurrent(thread, previousContext, nullptr, nullptr, nullptr),
605 "eglReleaseThread", nullptr, EGL_FALSE);
606 }
607 ANGLE_EGL_TRY_RETURN(thread, previousDisplay->releaseThread(), "eglReleaseThread",
608 GetDisplayIfValid(previousDisplay), EGL_FALSE);
609 }
610
611 thread->setSuccess();
612 return EGL_TRUE;
613 }
614
SurfaceAttrib(Thread * thread,Display * display,Surface * eglSurface,EGLint attribute,EGLint value)615 EGLBoolean SurfaceAttrib(Thread *thread,
616 Display *display,
617 Surface *eglSurface,
618 EGLint attribute,
619 EGLint value)
620 {
621 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglSurfaceAttrib",
622 GetDisplayIfValid(display), EGL_FALSE);
623
624 ANGLE_EGL_TRY_RETURN(thread, SetSurfaceAttrib(eglSurface, attribute, value), "eglSurfaceAttrib",
625 GetDisplayIfValid(display), EGL_FALSE);
626
627 thread->setSuccess();
628 return EGL_TRUE;
629 }
630
SwapBuffers(Thread * thread,Display * display,Surface * eglSurface)631 EGLBoolean SwapBuffers(Thread *thread, Display *display, Surface *eglSurface)
632 {
633 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglSwapBuffers",
634 GetDisplayIfValid(display), EGL_FALSE);
635
636 ANGLE_EGL_TRY_RETURN(thread, eglSurface->swap(thread->getContext()), "eglSwapBuffers",
637 GetSurfaceIfValid(display, eglSurface), EGL_FALSE);
638
639 thread->setSuccess();
640 return EGL_TRUE;
641 }
642
SwapInterval(Thread * thread,Display * display,EGLint interval)643 EGLBoolean SwapInterval(Thread *thread, Display *display, EGLint interval)
644 {
645 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglSwapInterval",
646 GetDisplayIfValid(display), EGL_FALSE);
647
648 Surface *drawSurface = static_cast<Surface *>(thread->getCurrentDrawSurface());
649 const Config *surfaceConfig = drawSurface->getConfig();
650 EGLint clampedInterval = std::min(std::max(interval, surfaceConfig->minSwapInterval),
651 surfaceConfig->maxSwapInterval);
652
653 drawSurface->setSwapInterval(clampedInterval);
654
655 thread->setSuccess();
656 return EGL_TRUE;
657 }
658
Terminate(Thread * thread,Display * display)659 EGLBoolean Terminate(Thread *thread, Display *display)
660 {
661 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglTerminate",
662 GetDisplayIfValid(display), EGL_FALSE);
663
664 ScopedSyncCurrentContextFromThread scopedSyncCurrent(thread);
665
666 ANGLE_EGL_TRY_RETURN(thread, display->terminate(thread, Display::TerminateReason::Api),
667 "eglTerminate", GetDisplayIfValid(display), EGL_FALSE);
668
669 thread->setSuccess();
670 return EGL_TRUE;
671 }
672
WaitClient(Thread * thread)673 EGLBoolean WaitClient(Thread *thread)
674 {
675 Display *display = thread->getDisplay();
676 if (display == nullptr)
677 {
678 // EGL spec says this about eglWaitClient -
679 // If there is no current context for the current rendering API,
680 // the function has no effect but still returns EGL_TRUE.
681 return EGL_TRUE;
682 }
683
684 gl::Context *context = thread->getContext();
685
686 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglWaitClient",
687 GetDisplayIfValid(display), EGL_FALSE);
688 ANGLE_EGL_TRY_RETURN(thread, display->waitClient(context), "eglWaitClient",
689 GetContextIfValid(display, context), EGL_FALSE);
690
691 thread->setSuccess();
692 return EGL_TRUE;
693 }
694
WaitGL(Thread * thread)695 EGLBoolean WaitGL(Thread *thread)
696 {
697 Display *display = thread->getDisplay();
698 if (display == nullptr)
699 {
700 // EGL spec says this about eglWaitGL -
701 // eglWaitGL is ignored if there is no current EGL rendering context for OpenGL ES.
702 return EGL_TRUE;
703 }
704
705 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglWaitGL", GetDisplayIfValid(display),
706 EGL_FALSE);
707
708 // eglWaitGL like calling eglWaitClient with the OpenGL ES API bound. Since we only implement
709 // OpenGL ES we can do the call directly.
710 ANGLE_EGL_TRY_RETURN(thread, display->waitClient(thread->getContext()), "eglWaitGL",
711 GetDisplayIfValid(display), EGL_FALSE);
712
713 thread->setSuccess();
714 return EGL_TRUE;
715 }
716
WaitNative(Thread * thread,EGLint engine)717 EGLBoolean WaitNative(Thread *thread, EGLint engine)
718 {
719 Display *display = thread->getDisplay();
720 if (display == nullptr)
721 {
722 // EGL spec says this about eglWaitNative -
723 // eglWaitNative is ignored if there is no current EGL rendering context.
724 return EGL_TRUE;
725 }
726
727 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglWaitNative",
728 GetDisplayIfValid(display), EGL_FALSE);
729 ANGLE_EGL_TRY_RETURN(thread, display->waitNative(thread->getContext(), engine), "eglWaitNative",
730 GetThreadIfValid(thread), EGL_FALSE);
731
732 thread->setSuccess();
733 return EGL_TRUE;
734 }
735
WaitSync(Thread * thread,Display * display,Sync * syncObject,EGLint flags)736 EGLBoolean WaitSync(Thread *thread, Display *display, Sync *syncObject, EGLint flags)
737 {
738 ANGLE_EGL_TRY_RETURN(thread, display->prepareForCall(), "eglWaitSync",
739 GetDisplayIfValid(display), EGL_FALSE);
740 gl::Context *currentContext = thread->getContext();
741 ANGLE_EGL_TRY_RETURN(thread, syncObject->serverWait(display, currentContext, flags),
742 "eglWaitSync", GetSyncIfValid(display, syncObject), EGL_FALSE);
743
744 thread->setSuccess();
745 return EGL_TRUE;
746 }
747 } // namespace egl
748