|  | /* | 
|  | ** Copyright 2018, The Android Open Source Project | 
|  | ** | 
|  | ** Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | ** you may not use this file except in compliance with the License. | 
|  | ** You may obtain a copy of the License at | 
|  | ** | 
|  | **     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | ** | 
|  | ** Unless required by applicable law or agreed to in writing, software | 
|  | ** distributed under the License is distributed on an "AS IS" BASIS, | 
|  | ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | ** See the License for the specific language governing permissions and | 
|  | ** limitations under the License. | 
|  | */ | 
|  |  | 
|  | #define ATRACE_TAG ATRACE_TAG_GRAPHICS | 
|  |  | 
|  | #include <EGL/egl.h> | 
|  | #include <EGL/eglext.h> | 
|  |  | 
|  | #include "../egl_impl.h" | 
|  | #include "egl_layers.h" | 
|  | #include "egl_platform_entries.h" | 
|  | #include "egl_tls.h" | 
|  | #include "egl_trace.h" | 
|  |  | 
|  | using namespace android; | 
|  |  | 
|  | namespace android { | 
|  |  | 
|  | extern EGLBoolean egl_init_drivers(); | 
|  |  | 
|  | } // namespace android | 
|  |  | 
|  | static inline void clearError() { | 
|  | egl_tls_t::clearError(); | 
|  | } | 
|  |  | 
|  | EGLDisplay eglGetDisplay(EGLNativeDisplayType display) { | 
|  | ATRACE_CALL(); | 
|  |  | 
|  | if (egl_init_drivers() == EGL_FALSE) { | 
|  | return setError(EGL_BAD_PARAMETER, EGL_NO_DISPLAY); | 
|  | } | 
|  |  | 
|  | // Call down the chain, which usually points directly to the impl | 
|  | // but may also be routed through layers | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetDisplay(display); | 
|  | } | 
|  |  | 
|  | EGLDisplay eglGetPlatformDisplay(EGLenum platform, EGLNativeDisplayType display, | 
|  | const EGLAttrib* attrib_list) { | 
|  | ATRACE_CALL(); | 
|  |  | 
|  | if (egl_init_drivers() == EGL_FALSE) { | 
|  | return setError(EGL_BAD_PARAMETER, EGL_NO_DISPLAY); | 
|  | } | 
|  |  | 
|  | // Call down the chain, which usually points directly to the impl | 
|  | // but may also be routed through layers | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetPlatformDisplay(platform, display, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglInitialize(EGLDisplay dpy, EGLint* major, EGLint* minor) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglInitialize(dpy, major, minor); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglTerminate(EGLDisplay dpy) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglTerminate(dpy); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetConfigs(EGLDisplay dpy, EGLConfig* configs, EGLint config_size, | 
|  | EGLint* num_config) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetConfigs(dpy, configs, config_size, num_config); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglChooseConfig(EGLDisplay dpy, const EGLint* attrib_list, EGLConfig* configs, | 
|  | EGLint config_size, EGLint* num_config) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglChooseConfig(dpy, attrib_list, configs, config_size, num_config); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetConfigAttrib(EGLDisplay dpy, EGLConfig config, EGLint attribute, EGLint* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetConfigAttrib(dpy, config, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLSurface eglCreateWindowSurface(EGLDisplay dpy, EGLConfig config, NativeWindowType window, | 
|  | const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateWindowSurface(dpy, config, window, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLSurface eglCreatePlatformWindowSurface(EGLDisplay dpy, EGLConfig config, void* native_window, | 
|  | const EGLAttrib* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreatePlatformWindowSurface(dpy, config, native_window, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLSurface eglCreatePixmapSurface(EGLDisplay dpy, EGLConfig config, NativePixmapType pixmap, | 
|  | const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreatePixmapSurface(dpy, config, pixmap, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLSurface eglCreatePlatformPixmapSurface(EGLDisplay dpy, EGLConfig config, void* native_pixmap, | 
|  | const EGLAttrib* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreatePlatformPixmapSurface(dpy, config, native_pixmap, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLSurface eglCreatePbufferSurface(EGLDisplay dpy, EGLConfig config, const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreatePbufferSurface(dpy, config, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglDestroySurface(EGLDisplay dpy, EGLSurface surface) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDestroySurface(dpy, surface); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglQuerySurface(EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLint* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQuerySurface(dpy, surface, attribute, value); | 
|  | } | 
|  |  | 
|  | void EGLAPI eglBeginFrame(EGLDisplay dpy, EGLSurface surface) { | 
|  | ATRACE_CALL(); | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | cnx->platform.eglBeginFrame(dpy, surface); | 
|  | } | 
|  |  | 
|  | EGLContext eglCreateContext(EGLDisplay dpy, EGLConfig config, EGLContext share_list, | 
|  | const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateContext(dpy, config, share_list, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglDestroyContext(EGLDisplay dpy, EGLContext ctx) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDestroyContext(dpy, ctx); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglMakeCurrent(EGLDisplay dpy, EGLSurface draw, EGLSurface read, EGLContext ctx) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglMakeCurrent(dpy, draw, read, ctx); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglQueryContext(EGLDisplay dpy, EGLContext ctx, EGLint attribute, EGLint* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQueryContext(dpy, ctx, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLContext eglGetCurrentContext(void) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetCurrentContext(); | 
|  | } | 
|  |  | 
|  | EGLSurface eglGetCurrentSurface(EGLint readdraw) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetCurrentSurface(readdraw); | 
|  | } | 
|  |  | 
|  | EGLDisplay eglGetCurrentDisplay(void) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetCurrentDisplay(); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglWaitGL(void) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglWaitGL(); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglWaitNative(EGLint engine) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglWaitNative(engine); | 
|  | } | 
|  |  | 
|  | EGLint eglGetError(void) { | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetError(); | 
|  | } | 
|  |  | 
|  | __eglMustCastToProperFunctionPointerType eglGetProcAddress(const char* procname) { | 
|  | // eglGetProcAddress() could be the very first function called | 
|  | // in which case we must make sure we've initialized ourselves, this | 
|  | // happens the first time egl_get_display() is called. | 
|  |  | 
|  | if (egl_init_drivers() == EGL_FALSE) { | 
|  | setError(EGL_BAD_PARAMETER, NULL); | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetProcAddress(procname); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglSwapBuffersWithDamageKHR(EGLDisplay dpy, EGLSurface draw, EGLint* rects, | 
|  | EGLint n_rects) { | 
|  | ATRACE_CALL(); | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglSwapBuffersWithDamageKHR(dpy, draw, rects, n_rects); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglSwapBuffers(EGLDisplay dpy, EGLSurface surface) { | 
|  | ATRACE_CALL(); | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglSwapBuffers(dpy, surface); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglCopyBuffers(EGLDisplay dpy, EGLSurface surface, NativePixmapType target) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCopyBuffers(dpy, surface, target); | 
|  | } | 
|  |  | 
|  | const char* eglQueryString(EGLDisplay dpy, EGLint name) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQueryString(dpy, name); | 
|  | } | 
|  |  | 
|  | extern "C" EGLAPI const char* eglQueryStringImplementationANDROID(EGLDisplay dpy, EGLint name) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQueryStringImplementationANDROID(dpy, name); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglSurfaceAttrib(EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLint value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglSurfaceAttrib(dpy, surface, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglBindTexImage(EGLDisplay dpy, EGLSurface surface, EGLint buffer) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglBindTexImage(dpy, surface, buffer); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglReleaseTexImage(EGLDisplay dpy, EGLSurface surface, EGLint buffer) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglReleaseTexImage(dpy, surface, buffer); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglSwapInterval(EGLDisplay dpy, EGLint interval) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglSwapInterval(dpy, interval); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglWaitClient(void) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglWaitClient(); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglBindAPI(EGLenum api) { | 
|  | if (egl_init_drivers() == EGL_FALSE) { | 
|  | return setError(EGL_BAD_PARAMETER, (EGLBoolean)EGL_FALSE); | 
|  | } | 
|  |  | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglBindAPI(api); | 
|  | } | 
|  |  | 
|  | EGLenum eglQueryAPI(void) { | 
|  | if (egl_init_drivers() == EGL_FALSE) { | 
|  | return setError(EGL_BAD_PARAMETER, (EGLBoolean)EGL_FALSE); | 
|  | } | 
|  |  | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQueryAPI(); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglReleaseThread(void) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglReleaseThread(); | 
|  | } | 
|  |  | 
|  | EGLSurface eglCreatePbufferFromClientBuffer(EGLDisplay dpy, EGLenum buftype, EGLClientBuffer buffer, | 
|  | EGLConfig config, const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreatePbufferFromClientBuffer(dpy, buftype, buffer, config, | 
|  | attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglLockSurfaceKHR(EGLDisplay dpy, EGLSurface surface, const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglLockSurfaceKHR(dpy, surface, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglUnlockSurfaceKHR(EGLDisplay dpy, EGLSurface surface) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglUnlockSurfaceKHR(dpy, surface); | 
|  | } | 
|  |  | 
|  | EGLImageKHR eglCreateImageKHR(EGLDisplay dpy, EGLContext ctx, EGLenum target, | 
|  | EGLClientBuffer buffer, const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateImageKHR(dpy, ctx, target, buffer, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLImage eglCreateImage(EGLDisplay dpy, EGLContext ctx, EGLenum target, EGLClientBuffer buffer, | 
|  | const EGLAttrib* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateImage(dpy, ctx, target, buffer, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglDestroyImageKHR(EGLDisplay dpy, EGLImageKHR img) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDestroyImageKHR(dpy, img); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglDestroyImage(EGLDisplay dpy, EGLImageKHR img) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDestroyImage(dpy, img); | 
|  | } | 
|  |  | 
|  | // ---------------------------------------------------------------------------- | 
|  | // EGL_EGLEXT_VERSION 5 | 
|  | // ---------------------------------------------------------------------------- | 
|  |  | 
|  | EGLSyncKHR eglCreateSync(EGLDisplay dpy, EGLenum type, const EGLAttrib* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateSync(dpy, type, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLSyncKHR eglCreateSyncKHR(EGLDisplay dpy, EGLenum type, const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateSyncKHR(dpy, type, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglDestroySync(EGLDisplay dpy, EGLSyncKHR sync) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDestroySync(dpy, sync); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglDestroySyncKHR(EGLDisplay dpy, EGLSyncKHR sync) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDestroySyncKHR(dpy, sync); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglSignalSyncKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLenum mode) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglSignalSyncKHR(dpy, sync, mode); | 
|  | } | 
|  |  | 
|  | EGLint eglClientWaitSync(EGLDisplay dpy, EGLSync sync, EGLint flags, EGLTimeKHR timeout) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglClientWaitSyncKHR(dpy, sync, flags, timeout); | 
|  | } | 
|  |  | 
|  | EGLint eglClientWaitSyncKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLint flags, EGLTimeKHR timeout) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglClientWaitSyncKHR(dpy, sync, flags, timeout); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetSyncAttrib(EGLDisplay dpy, EGLSync sync, EGLint attribute, EGLAttrib* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetSyncAttrib(dpy, sync, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetSyncAttribKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLint attribute, EGLint* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetSyncAttribKHR(dpy, sync, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLStreamKHR eglCreateStreamKHR(EGLDisplay dpy, const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateStreamKHR(dpy, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglDestroyStreamKHR(EGLDisplay dpy, EGLStreamKHR stream) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDestroyStreamKHR(dpy, stream); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglStreamAttribKHR(EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, | 
|  | EGLint value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglStreamAttribKHR(dpy, stream, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglQueryStreamKHR(EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, | 
|  | EGLint* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQueryStreamKHR(dpy, stream, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglQueryStreamu64KHR(EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, | 
|  | EGLuint64KHR* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQueryStreamu64KHR(dpy, stream, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglQueryStreamTimeKHR(EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, | 
|  | EGLTimeKHR* value) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglQueryStreamTimeKHR(dpy, stream, attribute, value); | 
|  | } | 
|  |  | 
|  | EGLSurface eglCreateStreamProducerSurfaceKHR(EGLDisplay dpy, EGLConfig config, EGLStreamKHR stream, | 
|  | const EGLint* attrib_list) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateStreamProducerSurfaceKHR(dpy, config, stream, attrib_list); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglStreamConsumerGLTextureExternalKHR(EGLDisplay dpy, EGLStreamKHR stream) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglStreamConsumerGLTextureExternalKHR(dpy, stream); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglStreamConsumerAcquireKHR(EGLDisplay dpy, EGLStreamKHR stream) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglStreamConsumerAcquireKHR(dpy, stream); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglStreamConsumerReleaseKHR(EGLDisplay dpy, EGLStreamKHR stream) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglStreamConsumerReleaseKHR(dpy, stream); | 
|  | } | 
|  |  | 
|  | EGLNativeFileDescriptorKHR eglGetStreamFileDescriptorKHR(EGLDisplay dpy, EGLStreamKHR stream) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetStreamFileDescriptorKHR(dpy, stream); | 
|  | } | 
|  |  | 
|  | EGLStreamKHR eglCreateStreamFromFileDescriptorKHR(EGLDisplay dpy, | 
|  | EGLNativeFileDescriptorKHR file_descriptor) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglCreateStreamFromFileDescriptorKHR(dpy, file_descriptor); | 
|  | } | 
|  |  | 
|  | EGLint eglWaitSyncKHR(EGLDisplay dpy, EGLSyncKHR sync, EGLint flags) { | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglWaitSyncKHR(dpy, sync, flags); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglWaitSync(EGLDisplay dpy, EGLSync sync, EGLint flags) { | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglWaitSync(dpy, sync, flags); | 
|  | } | 
|  |  | 
|  | EGLint eglDupNativeFenceFDANDROID(EGLDisplay dpy, EGLSyncKHR sync) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglDupNativeFenceFDANDROID(dpy, sync); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglPresentationTimeANDROID(EGLDisplay dpy, EGLSurface surface, EGLnsecsANDROID time) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglPresentationTimeANDROID(dpy, surface, time); | 
|  | } | 
|  |  | 
|  | EGLClientBuffer eglGetNativeClientBufferANDROID(const AHardwareBuffer* buffer) { | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetNativeClientBufferANDROID(buffer); | 
|  | } | 
|  |  | 
|  | EGLuint64NV eglGetSystemTimeFrequencyNV() { | 
|  | if (egl_init_drivers() == EGL_FALSE) { | 
|  | return setError(EGL_BAD_PARAMETER, (EGLuint64NV)EGL_FALSE); | 
|  | } | 
|  |  | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetSystemTimeFrequencyNV(); | 
|  | } | 
|  |  | 
|  | EGLuint64NV eglGetSystemTimeNV() { | 
|  | if (egl_init_drivers() == EGL_FALSE) { | 
|  | return setError(EGL_BAD_PARAMETER, (EGLuint64NV)EGL_FALSE); | 
|  | } | 
|  |  | 
|  | clearError(); | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetSystemTimeNV(); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglSetDamageRegionKHR(EGLDisplay dpy, EGLSurface surface, EGLint* rects, | 
|  | EGLint n_rects) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglSetDamageRegionKHR(dpy, surface, rects, n_rects); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetNextFrameIdANDROID(EGLDisplay dpy, EGLSurface surface, EGLuint64KHR* frameId) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetNextFrameIdANDROID(dpy, surface, frameId); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetCompositorTimingANDROID(EGLDisplay dpy, EGLSurface surface, EGLint numTimestamps, | 
|  | const EGLint* names, EGLnsecsANDROID* values) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetCompositorTimingANDROID(dpy, surface, numTimestamps, names, values); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetCompositorTimingSupportedANDROID(EGLDisplay dpy, EGLSurface surface, EGLint name) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetCompositorTimingSupportedANDROID(dpy, surface, name); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetFrameTimestampsANDROID(EGLDisplay dpy, EGLSurface surface, EGLuint64KHR frameId, | 
|  | EGLint numTimestamps, const EGLint* timestamps, | 
|  | EGLnsecsANDROID* values) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetFrameTimestampsANDROID(dpy, surface, frameId, numTimestamps, | 
|  | timestamps, values); | 
|  | } | 
|  |  | 
|  | EGLBoolean eglGetFrameTimestampSupportedANDROID(EGLDisplay dpy, EGLSurface surface, | 
|  | EGLint timestamp) { | 
|  | clearError(); | 
|  |  | 
|  | egl_connection_t* const cnx = &gEGLImpl; | 
|  | return cnx->platform.eglGetFrameTimestampSupportedANDROID(dpy, surface, timestamp); | 
|  | } |