• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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