2021-01-29 00:13:53 +01:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <wayland-client.h>
|
|
|
|
#include <wlr/util/log.h>
|
|
|
|
|
|
|
|
#include <EGL/egl.h>
|
|
|
|
#include <EGL/eglext.h>
|
|
|
|
|
|
|
|
#include "egl_common.h"
|
|
|
|
|
|
|
|
EGLDisplay egl_display;
|
|
|
|
EGLConfig egl_config;
|
|
|
|
EGLContext egl_context;
|
|
|
|
|
|
|
|
PFNEGLGETPLATFORMDISPLAYEXTPROC eglGetPlatformDisplayEXT;
|
|
|
|
PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC eglCreatePlatformWindowSurfaceEXT;
|
|
|
|
|
|
|
|
EGLint config_attribs[] = {
|
|
|
|
EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
|
|
|
|
EGL_RED_SIZE, 1,
|
|
|
|
EGL_GREEN_SIZE, 1,
|
|
|
|
EGL_BLUE_SIZE, 1,
|
|
|
|
EGL_ALPHA_SIZE, 1,
|
|
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
|
|
|
EGL_NONE,
|
|
|
|
};
|
|
|
|
|
|
|
|
EGLint context_attribs[] = {
|
|
|
|
EGL_CONTEXT_CLIENT_VERSION, 2,
|
|
|
|
EGL_NONE,
|
|
|
|
};
|
|
|
|
|
|
|
|
bool egl_init(struct wl_display *display) {
|
|
|
|
const char *client_exts_str =
|
|
|
|
eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS);
|
|
|
|
if (client_exts_str == NULL) {
|
|
|
|
if (eglGetError() == EGL_BAD_DISPLAY) {
|
|
|
|
wlr_log(WLR_ERROR,
|
|
|
|
"EGL_EXT_client_extensions not supported");
|
|
|
|
} else {
|
|
|
|
wlr_log(WLR_ERROR,
|
|
|
|
"Failed to query EGL client extensions");
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strstr(client_exts_str, "EGL_EXT_platform_base")) {
|
|
|
|
wlr_log(WLR_ERROR, "EGL_EXT_platform_base not supported");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strstr(client_exts_str, "EGL_EXT_platform_wayland")) {
|
|
|
|
wlr_log(WLR_ERROR, "EGL_EXT_platform_wayland not supported");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
eglGetPlatformDisplayEXT =
|
|
|
|
(void *)eglGetProcAddress("eglGetPlatformDisplayEXT");
|
|
|
|
if (eglGetPlatformDisplayEXT == NULL) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to get eglGetPlatformDisplayEXT");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
eglCreatePlatformWindowSurfaceEXT =
|
|
|
|
(void *)eglGetProcAddress("eglCreatePlatformWindowSurfaceEXT");
|
|
|
|
if (eglCreatePlatformWindowSurfaceEXT == NULL) {
|
|
|
|
wlr_log(WLR_ERROR,
|
|
|
|
"Failed to get eglCreatePlatformWindowSurfaceEXT");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
egl_display =
|
|
|
|
eglGetPlatformDisplayEXT(EGL_PLATFORM_WAYLAND_EXT,
|
|
|
|
display, NULL);
|
|
|
|
if (egl_display == EGL_NO_DISPLAY) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to create EGL display");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2021-01-30 00:00:58 +01:00
|
|
|
if (eglInitialize(egl_display, NULL, NULL) == EGL_FALSE) {
|
2021-01-29 00:13:53 +01:00
|
|
|
wlr_log(WLR_ERROR, "Failed to initialize EGL");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
EGLint matched = 0;
|
|
|
|
if (!eglChooseConfig(egl_display, config_attribs,
|
|
|
|
&egl_config, 1, &matched)) {
|
|
|
|
wlr_log(WLR_ERROR, "eglChooseConfig failed");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if (matched == 0) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to match an EGL config");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
egl_context =
|
|
|
|
eglCreateContext(egl_display, egl_config,
|
|
|
|
EGL_NO_CONTEXT, context_attribs);
|
|
|
|
if (egl_context == EGL_NO_CONTEXT) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to create EGL context");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
error:
|
|
|
|
eglMakeCurrent(EGL_NO_DISPLAY, EGL_NO_SURFACE,
|
|
|
|
EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
|
|
|
if (egl_display) {
|
|
|
|
eglTerminate(egl_display);
|
|
|
|
}
|
|
|
|
eglReleaseThread();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void egl_finish(void) {
|
|
|
|
eglMakeCurrent(egl_display, EGL_NO_SURFACE,
|
|
|
|
EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
|
|
|
eglDestroyContext(egl_display, egl_context);
|
|
|
|
eglTerminate(egl_display);
|
|
|
|
eglReleaseThread();
|
|
|
|
}
|