2021-06-06 01:26:18 +00:00
|
|
|
/**
|
|
|
|
* Looking Glass
|
2021-08-04 09:48:32 +00:00
|
|
|
* Copyright © 2017-2021 The Looking Glass Authors
|
2021-06-06 01:26:18 +00:00
|
|
|
* https://looking-glass.io
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License as published by the Free
|
|
|
|
* Software Foundation; either version 2 of the License, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc., 59
|
|
|
|
* Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2019-03-28 00:02:36 +00:00
|
|
|
#include "interface/renderer.h"
|
2018-11-19 22:50:09 +00:00
|
|
|
|
2019-04-11 01:12:59 +00:00
|
|
|
#include "common/debug.h"
|
2021-07-11 01:20:08 +00:00
|
|
|
#include "common/KVMFR.h"
|
2019-05-21 05:03:59 +00:00
|
|
|
#include "common/option.h"
|
2019-05-23 06:56:13 +00:00
|
|
|
#include "common/sysinfo.h"
|
2021-08-11 06:42:31 +00:00
|
|
|
#include "common/rects.h"
|
2020-01-17 03:35:08 +00:00
|
|
|
#include "common/time.h"
|
2020-05-21 01:44:56 +00:00
|
|
|
#include "common/locking.h"
|
2021-07-08 01:45:54 +00:00
|
|
|
#include "app.h"
|
2021-01-25 08:58:36 +00:00
|
|
|
#include "util.h"
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2021-01-26 10:46:30 +00:00
|
|
|
#include <EGL/egl.h>
|
2021-07-18 09:39:28 +00:00
|
|
|
#include <GLES3/gl32.h>
|
2019-03-26 06:29:49 +00:00
|
|
|
|
2021-07-08 01:45:54 +00:00
|
|
|
#include "cimgui.h"
|
|
|
|
#include "generator/output/cimgui_impl.h"
|
|
|
|
|
2021-02-21 06:25:54 +00:00
|
|
|
#include <math.h>
|
2021-01-26 10:46:30 +00:00
|
|
|
#include <string.h>
|
2021-01-19 15:06:49 +00:00
|
|
|
|
2021-01-15 09:30:03 +00:00
|
|
|
#include "app.h"
|
2021-01-26 10:46:30 +00:00
|
|
|
#include "egl_dynprocs.h"
|
2019-03-28 00:02:36 +00:00
|
|
|
#include "model.h"
|
|
|
|
#include "shader.h"
|
2021-07-15 10:04:25 +00:00
|
|
|
#include "damage.h"
|
2019-03-28 00:02:36 +00:00
|
|
|
#include "desktop.h"
|
|
|
|
#include "cursor.h"
|
|
|
|
#include "splash.h"
|
2021-08-11 08:53:36 +00:00
|
|
|
#include "postprocess.h"
|
2021-08-03 01:37:58 +00:00
|
|
|
#include "util.h"
|
2018-12-15 13:54:37 +00:00
|
|
|
|
2018-12-15 23:57:01 +00:00
|
|
|
#define SPLASH_FADE_TIME 1000000
|
2021-08-03 01:37:58 +00:00
|
|
|
#define MAX_BUFFER_AGE 3
|
|
|
|
#define DESKTOP_DAMAGE_COUNT 4
|
|
|
|
#define MAX_ACCUMULATED_DAMAGE ((KVMFR_MAX_DAMAGE_RECTS + MAX_OVERLAY_RECTS + 2) * MAX_BUFFER_AGE)
|
|
|
|
#define IDX_AGO(counter, i, total) ((counter) + (total) - i) % (total)
|
2018-09-22 06:26:10 +00:00
|
|
|
|
|
|
|
struct Options
|
|
|
|
{
|
2018-09-22 08:00:52 +00:00
|
|
|
bool vsync;
|
2020-12-31 04:31:24 +00:00
|
|
|
bool doubleBuffer;
|
2018-09-22 06:26:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct Inst
|
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
LG_Renderer base;
|
|
|
|
|
2020-10-29 15:32:25 +00:00
|
|
|
bool dmaSupport;
|
2018-09-22 06:26:10 +00:00
|
|
|
LG_RendererParams params;
|
|
|
|
struct Options opt;
|
|
|
|
|
2019-01-01 13:04:40 +00:00
|
|
|
EGLNativeWindowType nativeWind;
|
|
|
|
EGLDisplay display;
|
|
|
|
EGLConfig configs;
|
|
|
|
EGLSurface surface;
|
2020-05-21 01:44:56 +00:00
|
|
|
EGLContext context, frameContext;
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2018-12-12 10:41:51 +00:00
|
|
|
EGL_Desktop * desktop; // the desktop
|
|
|
|
EGL_Cursor * cursor; // the mouse cursor
|
2018-12-15 13:54:37 +00:00
|
|
|
EGL_Splash * splash; // the splash screen
|
2021-07-15 10:04:25 +00:00
|
|
|
EGL_Damage * damage; // the damage display
|
2021-07-08 01:45:54 +00:00
|
|
|
bool imgui; // if imgui was initialized
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2018-09-23 10:45:20 +00:00
|
|
|
LG_RendererFormat format;
|
2021-01-19 09:36:43 +00:00
|
|
|
bool formatValid;
|
2020-08-11 04:45:08 +00:00
|
|
|
bool start;
|
2018-12-15 13:54:37 +00:00
|
|
|
uint64_t waitFadeTime;
|
|
|
|
bool waitDone;
|
2018-09-24 09:48:11 +00:00
|
|
|
|
2021-01-18 15:44:56 +00:00
|
|
|
int width, height;
|
2021-02-21 06:25:54 +00:00
|
|
|
float uiScale;
|
2021-08-03 12:27:46 +00:00
|
|
|
struct DoubleRect destRect;
|
2021-01-18 15:44:56 +00:00
|
|
|
LG_RendererRotate rotate; //client side rotation
|
2018-11-19 22:50:09 +00:00
|
|
|
|
2018-12-15 23:57:01 +00:00
|
|
|
float translateX , translateY;
|
|
|
|
float scaleX , scaleY;
|
|
|
|
float splashRatio;
|
|
|
|
float screenScaleX, screenScaleY;
|
2021-02-21 00:50:59 +00:00
|
|
|
|
|
|
|
int viewportWidth, viewportHeight;
|
|
|
|
enum EGL_DesktopScaleType scaleType;
|
2018-10-03 16:31:37 +00:00
|
|
|
|
2021-07-23 10:03:20 +00:00
|
|
|
bool cursorVisible;
|
|
|
|
int cursorX , cursorY;
|
|
|
|
float mouseWidth , mouseHeight;
|
|
|
|
float mouseScaleX, mouseScaleY;
|
|
|
|
bool showDamage;
|
2018-11-19 22:50:09 +00:00
|
|
|
|
2021-05-11 20:49:39 +00:00
|
|
|
struct CursorState cursorLast;
|
2021-07-11 01:20:08 +00:00
|
|
|
|
2021-08-01 00:37:23 +00:00
|
|
|
bool hadOverlay;
|
|
|
|
struct DesktopDamage desktopDamage[DESKTOP_DAMAGE_COUNT];
|
2021-08-01 08:24:55 +00:00
|
|
|
unsigned int desktopDamageIdx;
|
|
|
|
LG_Lock desktopDamageLock;
|
2021-07-28 05:06:06 +00:00
|
|
|
|
2021-08-03 01:37:58 +00:00
|
|
|
bool hasBufferAge;
|
|
|
|
struct Rect overlayHistory[DESKTOP_DAMAGE_COUNT][MAX_OVERLAY_RECTS + 1];
|
|
|
|
int overlayHistoryCount[DESKTOP_DAMAGE_COUNT];
|
|
|
|
unsigned int overlayHistoryIdx;
|
|
|
|
|
2021-07-28 05:06:06 +00:00
|
|
|
RingBuffer importTimings;
|
|
|
|
GraphHandle importGraph;
|
2018-09-22 06:26:10 +00:00
|
|
|
};
|
|
|
|
|
2019-05-21 05:03:59 +00:00
|
|
|
static struct Option egl_options[] =
|
|
|
|
{
|
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "vsync",
|
|
|
|
.description = "Enable vsync",
|
|
|
|
.type = OPTION_TYPE_BOOL,
|
[client] fix hang in eglSwapBuffers when exiting while not visible
eglSwapBuffers is allowed to block when called with a nonzero interval
parameter. On Wayland, Mesa will block until a frame callback arrives.
If an application is not visible, a compositor is free to not schedule
frame callbacks (in order to save CPU time rendering something that is
entirely invisible).
Currently, starting Looking Glass from a terminal, hiding it
entirely, and sending ^C will cause Looking Glass to hang joining the
render thread until the window is made visible again.
Calling eglDestroySurface is insufficient to unblock eglSwapBuffers, as
it attempts to grab the same underlying mutex.
Instead, this commit makes it so that we pass a 0 interval to
eglSwapBuffers when running on Wayland, such that we don't block waiting
for a frame callback. This is not entirely ideal as it *does* mean
Looking Glass submits buffers while hidden, but it seems better than
hanging on exit.
It also forces opengl:vsync and egl:vsync flags to off when running on
Wayland, as they are meaningless there.
2021-01-04 02:09:55 +00:00
|
|
|
.value.x_bool = false,
|
2019-05-21 05:03:59 +00:00
|
|
|
},
|
2019-05-27 08:39:21 +00:00
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "doubleBuffer",
|
|
|
|
.description = "Enable double buffering",
|
|
|
|
.type = OPTION_TYPE_BOOL,
|
2020-05-19 07:34:24 +00:00
|
|
|
.value.x_bool = false
|
2019-05-27 08:39:21 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "multisample",
|
|
|
|
.description = "Enable Multisampling",
|
|
|
|
.type = OPTION_TYPE_BOOL,
|
|
|
|
.value.x_bool = true
|
|
|
|
},
|
2019-05-23 06:32:18 +00:00
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "nvGainMax",
|
|
|
|
.description = "The maximum night vision gain",
|
|
|
|
.type = OPTION_TYPE_INT,
|
|
|
|
.value.x_int = 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "nvGain",
|
|
|
|
.description = "The initial night vision gain at startup",
|
|
|
|
.type = OPTION_TYPE_INT,
|
|
|
|
.value.x_int = 0
|
|
|
|
},
|
2020-11-08 19:59:54 +00:00
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "cbMode",
|
2020-11-08 20:42:59 +00:00
|
|
|
.description = "Color Blind Mode (0 = Off, 1 = Protanope, 2 = Deuteranope, 3 = Tritanope)",
|
2020-11-08 19:59:54 +00:00
|
|
|
.type = OPTION_TYPE_INT,
|
|
|
|
.value.x_int = 0
|
|
|
|
},
|
2021-02-22 05:03:50 +00:00
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "scale",
|
|
|
|
.description = "Set the scale algorithm (0 = auto, 1 = nearest, 2 = linear)",
|
|
|
|
.type = OPTION_TYPE_INT,
|
2021-08-08 07:16:10 +00:00
|
|
|
.validator = egl_desktopScaleValidate,
|
2021-02-22 05:03:50 +00:00
|
|
|
.value.x_int = 0
|
|
|
|
},
|
2021-07-18 04:14:31 +00:00
|
|
|
{
|
|
|
|
.module = "egl",
|
|
|
|
.name = "debug",
|
|
|
|
.description = "Enable debug output",
|
|
|
|
.type = OPTION_TYPE_BOOL,
|
|
|
|
.value.x_bool = false
|
|
|
|
},
|
2021-08-09 15:10:08 +00:00
|
|
|
|
2019-05-21 05:03:59 +00:00
|
|
|
{0}
|
|
|
|
};
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static const char * egl_getName(void)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
|
|
|
return "EGL";
|
|
|
|
}
|
|
|
|
|
2021-07-29 06:31:34 +00:00
|
|
|
static void egl_setup(void)
|
2019-05-21 05:03:59 +00:00
|
|
|
{
|
|
|
|
option_register(egl_options);
|
2021-08-11 08:53:36 +00:00
|
|
|
egl_postProcessEarlyInit();
|
2019-05-21 05:03:59 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 05:32:01 +00:00
|
|
|
static bool egl_create(LG_Renderer ** renderer, const LG_RendererParams params,
|
|
|
|
bool * needsOpenGL)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2020-11-29 10:43:28 +00:00
|
|
|
// check if EGL is even available
|
2021-08-15 22:16:09 +00:00
|
|
|
if (!eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS))
|
|
|
|
{
|
|
|
|
DEBUG_WARN("EGL is not available");
|
2020-11-29 10:43:28 +00:00
|
|
|
return false;
|
2021-08-15 22:16:09 +00:00
|
|
|
}
|
2020-11-29 10:43:28 +00:00
|
|
|
|
2018-09-22 06:26:10 +00:00
|
|
|
// create our local storage
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = calloc(1, sizeof(*this));
|
|
|
|
if (!this)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
DEBUG_INFO("Failed to allocate %lu bytes", sizeof(*this));
|
2018-09-22 06:26:10 +00:00
|
|
|
return false;
|
|
|
|
}
|
2021-08-08 05:32:01 +00:00
|
|
|
*renderer = &this->base;
|
2018-09-22 06:26:10 +00:00
|
|
|
|
|
|
|
// safe off parameteres and init our default option values
|
2019-05-21 05:03:59 +00:00
|
|
|
memcpy(&this->params, ¶ms, sizeof(LG_RendererParams));
|
|
|
|
|
2020-12-31 04:31:24 +00:00
|
|
|
this->opt.vsync = option_get_bool("egl", "vsync");
|
|
|
|
this->opt.doubleBuffer = option_get_bool("egl", "doubleBuffer");
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2018-12-15 23:57:01 +00:00
|
|
|
this->translateX = 0;
|
|
|
|
this->translateY = 0;
|
|
|
|
this->scaleX = 1.0f;
|
|
|
|
this->scaleY = 1.0f;
|
|
|
|
this->screenScaleX = 1.0f;
|
|
|
|
this->screenScaleY = 1.0f;
|
2021-02-21 06:25:54 +00:00
|
|
|
this->uiScale = 1.0;
|
2018-09-24 09:48:11 +00:00
|
|
|
|
2021-08-01 08:24:55 +00:00
|
|
|
LG_LOCK_INIT(this->desktopDamageLock);
|
2021-08-01 00:37:23 +00:00
|
|
|
this->desktopDamage[0].count = -1;
|
2021-07-11 01:20:08 +00:00
|
|
|
|
2021-07-28 05:06:06 +00:00
|
|
|
this->importTimings = ringbuffer_new(256, sizeof(float));
|
2021-07-29 01:56:50 +00:00
|
|
|
this->importGraph = app_registerGraph("IMPORT", this->importTimings, 0.0f, 5.0f);
|
2021-07-28 05:06:06 +00:00
|
|
|
|
2021-01-27 08:38:34 +00:00
|
|
|
*needsOpenGL = false;
|
2018-09-22 06:26:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:32:01 +00:00
|
|
|
static bool egl_initialize(LG_Renderer * renderer)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2020-12-31 04:31:24 +00:00
|
|
|
DEBUG_INFO("Double buffering is %s", this->opt.doubleBuffer ? "on" : "off");
|
2018-09-22 06:26:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:32:01 +00:00
|
|
|
static void egl_deinitialize(LG_Renderer * renderer)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2021-07-08 01:45:54 +00:00
|
|
|
if (this->imgui)
|
|
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
|
|
|
2021-07-28 05:06:06 +00:00
|
|
|
app_unregisterGraph(this->importGraph);
|
|
|
|
ringbuffer_free(&this->importTimings);
|
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_desktopFree(&this->desktop);
|
|
|
|
egl_cursorFree (&this->cursor);
|
|
|
|
egl_splashFree (&this->splash);
|
|
|
|
egl_damageFree (&this->damage);
|
2018-09-24 09:48:11 +00:00
|
|
|
|
2020-05-21 01:44:56 +00:00
|
|
|
LG_LOCK_FREE(this->lock);
|
2021-08-01 08:24:55 +00:00
|
|
|
LG_LOCK_FREE(this->desktopDamageLock);
|
2020-05-21 01:44:56 +00:00
|
|
|
|
2021-01-24 08:39:00 +00:00
|
|
|
eglMakeCurrent(this->display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
|
|
|
|
|
|
|
if (this->frameContext)
|
|
|
|
eglDestroyContext(this->display, this->frameContext);
|
|
|
|
|
|
|
|
if (this->context)
|
|
|
|
eglDestroyContext(this->display, this->context);
|
|
|
|
|
|
|
|
eglTerminate(this->display);
|
|
|
|
|
2018-09-22 06:26:10 +00:00
|
|
|
free(this);
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:32:01 +00:00
|
|
|
static bool egl_supports(LG_Renderer * renderer, LG_RendererSupport flag)
|
2020-10-29 15:32:25 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2020-10-29 15:32:25 +00:00
|
|
|
|
|
|
|
switch(flag)
|
|
|
|
{
|
|
|
|
case LG_SUPPORTS_DMABUF:
|
|
|
|
return this->dmaSupport;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static void egl_onRestart(LG_Renderer * renderer)
|
2020-08-11 04:30:44 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2020-08-11 04:30:44 +00:00
|
|
|
|
|
|
|
eglDestroyContext(this->display, this->frameContext);
|
|
|
|
this->frameContext = NULL;
|
2020-08-11 04:45:08 +00:00
|
|
|
this->start = false;
|
2020-08-11 04:30:44 +00:00
|
|
|
}
|
|
|
|
|
2021-01-18 15:44:56 +00:00
|
|
|
static void egl_calc_mouse_size(struct Inst * this)
|
|
|
|
{
|
2021-01-19 09:36:43 +00:00
|
|
|
if (!this->formatValid)
|
|
|
|
return;
|
|
|
|
|
2021-08-13 23:52:30 +00:00
|
|
|
int w, h;
|
2021-02-23 09:25:30 +00:00
|
|
|
|
2021-01-18 15:44:56 +00:00
|
|
|
switch(this->format.rotate)
|
|
|
|
{
|
|
|
|
case LG_ROTATE_0:
|
|
|
|
case LG_ROTATE_180:
|
|
|
|
this->mouseScaleX = 2.0f / this->format.width;
|
|
|
|
this->mouseScaleY = 2.0f / this->format.height;
|
|
|
|
w = this->format.width;
|
|
|
|
h = this->format.height;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_90:
|
|
|
|
case LG_ROTATE_270:
|
|
|
|
this->mouseScaleX = 2.0f / this->format.height;
|
|
|
|
this->mouseScaleY = 2.0f / this->format.width;
|
|
|
|
w = this->format.height;
|
|
|
|
h = this->format.width;
|
|
|
|
break;
|
2021-01-19 15:06:49 +00:00
|
|
|
|
|
|
|
default:
|
2021-08-13 23:43:51 +00:00
|
|
|
DEBUG_UNREACHABLE();
|
2021-01-18 15:44:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch((this->format.rotate + this->rotate) % LG_ROTATE_MAX)
|
|
|
|
{
|
|
|
|
case LG_ROTATE_0:
|
|
|
|
case LG_ROTATE_180:
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_cursorSetSize(this->cursor,
|
2021-01-18 15:44:56 +00:00
|
|
|
(this->mouseWidth * (1.0f / w)) * this->scaleX,
|
|
|
|
(this->mouseHeight * (1.0f / h)) * this->scaleY
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_90:
|
|
|
|
case LG_ROTATE_270:
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_cursorSetSize(this->cursor,
|
2021-01-18 15:44:56 +00:00
|
|
|
(this->mouseWidth * (1.0f / w)) * this->scaleY,
|
|
|
|
(this->mouseHeight * (1.0f / h)) * this->scaleX
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void egl_calc_mouse_state(struct Inst * this)
|
|
|
|
{
|
2021-01-19 09:36:43 +00:00
|
|
|
if (!this->formatValid)
|
|
|
|
return;
|
|
|
|
|
2021-01-18 15:44:56 +00:00
|
|
|
switch((this->format.rotate + this->rotate) % LG_ROTATE_MAX)
|
|
|
|
{
|
|
|
|
case LG_ROTATE_0:
|
|
|
|
case LG_ROTATE_180:
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_cursorSetState(
|
2021-01-18 15:44:56 +00:00
|
|
|
this->cursor,
|
|
|
|
this->cursorVisible,
|
|
|
|
(((float)this->cursorX * this->mouseScaleX) - 1.0f) * this->scaleX,
|
|
|
|
(((float)this->cursorY * this->mouseScaleY) - 1.0f) * this->scaleY
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_90:
|
|
|
|
case LG_ROTATE_270:
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_cursorSetState(
|
2021-01-18 15:44:56 +00:00
|
|
|
this->cursor,
|
|
|
|
this->cursorVisible,
|
|
|
|
(((float)this->cursorX * this->mouseScaleX) - 1.0f) * this->scaleY,
|
|
|
|
(((float)this->cursorY * this->mouseScaleY) - 1.0f) * this->scaleX
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-21 00:50:59 +00:00
|
|
|
static void egl_update_scale_type(struct Inst * this)
|
|
|
|
{
|
2021-02-21 14:38:26 +00:00
|
|
|
int width = 0, height = 0;
|
2021-02-21 00:50:59 +00:00
|
|
|
|
|
|
|
switch (this->rotate)
|
|
|
|
{
|
|
|
|
case LG_ROTATE_0:
|
|
|
|
case LG_ROTATE_180:
|
|
|
|
width = this->format.width;
|
|
|
|
height = this->format.height;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_90:
|
|
|
|
case LG_ROTATE_270:
|
|
|
|
width = this->format.height;
|
|
|
|
height = this->format.width;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (width == this->viewportWidth || height == this->viewportHeight)
|
|
|
|
this->scaleType = EGL_DESKTOP_NOSCALE;
|
|
|
|
else if (width > this->viewportWidth || height > this->viewportHeight)
|
|
|
|
this->scaleType = EGL_DESKTOP_DOWNSCALE;
|
|
|
|
else
|
|
|
|
this->scaleType = EGL_DESKTOP_UPSCALE;
|
|
|
|
}
|
|
|
|
|
2021-08-09 04:08:10 +00:00
|
|
|
void egl_resetViewport(EGL * this)
|
|
|
|
{
|
|
|
|
glViewport(0, 0, this->width, this->height);
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static void egl_onResize(LG_Renderer * renderer, const int width, const int height, const double scale,
|
2021-01-18 15:44:56 +00:00
|
|
|
const LG_RendererRect destRect, LG_RendererRotate rotate)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2018-09-24 09:48:11 +00:00
|
|
|
|
2021-02-21 06:25:54 +00:00
|
|
|
this->width = width * scale;
|
|
|
|
this->height = height * scale;
|
|
|
|
this->uiScale = (float) scale;
|
|
|
|
this->rotate = rotate;
|
2021-01-18 15:44:56 +00:00
|
|
|
|
2021-02-21 05:47:38 +00:00
|
|
|
this->destRect.x = destRect.x * scale;
|
|
|
|
this->destRect.y = destRect.y * scale;
|
|
|
|
this->destRect.w = destRect.w * scale;
|
|
|
|
this->destRect.h = destRect.h * scale;
|
2018-11-19 22:50:09 +00:00
|
|
|
|
2021-02-21 05:47:38 +00:00
|
|
|
glViewport(0, 0, this->width, this->height);
|
2018-09-24 09:48:11 +00:00
|
|
|
|
2018-10-03 16:31:37 +00:00
|
|
|
if (destRect.valid)
|
|
|
|
{
|
2021-02-21 05:47:38 +00:00
|
|
|
this->translateX = 1.0f - (((this->destRect.w / 2) + this->destRect.x) * 2) / (float)this->width;
|
|
|
|
this->translateY = 1.0f - (((this->destRect.h / 2) + this->destRect.y) * 2) / (float)this->height;
|
|
|
|
this->scaleX = (float)this->destRect.w / (float)this->width;
|
|
|
|
this->scaleY = (float)this->destRect.h / (float)this->height;
|
|
|
|
this->viewportWidth = this->destRect.w;
|
|
|
|
this->viewportHeight = this->destRect.h;
|
2018-10-03 16:31:37 +00:00
|
|
|
}
|
|
|
|
|
2021-02-21 00:50:59 +00:00
|
|
|
egl_update_scale_type(this);
|
2021-01-18 15:44:56 +00:00
|
|
|
egl_calc_mouse_size(this);
|
2018-12-15 13:54:37 +00:00
|
|
|
|
2018-12-15 23:57:01 +00:00
|
|
|
this->splashRatio = (float)width / (float)height;
|
2021-02-21 05:47:38 +00:00
|
|
|
this->screenScaleX = 1.0f / this->width;
|
|
|
|
this->screenScaleY = 1.0f / this->height;
|
2019-05-28 04:06:15 +00:00
|
|
|
|
2021-01-18 15:44:56 +00:00
|
|
|
egl_calc_mouse_state(this);
|
2021-05-11 20:49:39 +00:00
|
|
|
|
2021-08-01 08:24:55 +00:00
|
|
|
INTERLOCKED_SECTION(this->desktopDamageLock, {
|
|
|
|
this->desktopDamage[this->desktopDamageIdx].count = -1;
|
|
|
|
});
|
2021-07-15 10:04:25 +00:00
|
|
|
|
2021-07-23 04:01:10 +00:00
|
|
|
// this is needed to refresh the font atlas texture
|
|
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
|
|
ImGui_ImplOpenGL3_NewFrame();
|
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_damageResize(this->damage, this->translateX, this->translateY, this->scaleX, this->scaleY);
|
2021-08-10 03:08:54 +00:00
|
|
|
egl_desktopResize(this->desktop, this->width, this->height);
|
2018-09-22 06:26:10 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static bool egl_onMouseShape(LG_Renderer * renderer, const LG_RendererCursor cursor,
|
2021-01-18 15:44:56 +00:00
|
|
|
const int width, const int height,
|
2021-01-15 01:40:59 +00:00
|
|
|
const int pitch, const uint8_t * data)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2021-01-18 15:44:56 +00:00
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
if (!egl_cursorSetShape(this->cursor, cursor, width, height, pitch, data))
|
2018-09-24 09:48:11 +00:00
|
|
|
{
|
2018-12-12 07:53:55 +00:00
|
|
|
DEBUG_ERROR("Failed to update the cursor shape");
|
|
|
|
return false;
|
2018-09-24 09:48:11 +00:00
|
|
|
}
|
|
|
|
|
2018-12-12 07:53:55 +00:00
|
|
|
this->mouseWidth = width;
|
|
|
|
this->mouseHeight = height;
|
2021-01-18 15:44:56 +00:00
|
|
|
egl_calc_mouse_size(this);
|
2018-09-24 09:48:11 +00:00
|
|
|
|
2018-09-22 06:26:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static bool egl_onMouseEvent(LG_Renderer * renderer, const bool visible, const int x, const int y)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2019-05-28 04:06:15 +00:00
|
|
|
this->cursorVisible = visible;
|
|
|
|
this->cursorX = x;
|
|
|
|
this->cursorY = y;
|
2021-01-18 15:44:56 +00:00
|
|
|
egl_calc_mouse_state(this);
|
2018-09-22 06:26:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static bool egl_onFrameFormat(LG_Renderer * renderer, const LG_RendererFormat format)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2020-10-12 08:43:29 +00:00
|
|
|
memcpy(&this->format, &format, sizeof(LG_RendererFormat));
|
2021-01-19 09:36:43 +00:00
|
|
|
this->formatValid = true;
|
2019-04-19 01:23:51 +00:00
|
|
|
|
2020-05-22 07:47:19 +00:00
|
|
|
/* this event runs in a second thread so we need to init it here */
|
|
|
|
if (!this->frameContext)
|
|
|
|
{
|
|
|
|
static EGLint attrs[] = {
|
|
|
|
EGL_CONTEXT_CLIENT_VERSION, 2,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!(this->frameContext = eglCreateContext(this->display, this->configs, this->context, attrs)))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to create the frame context");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!eglMakeCurrent(this->display, EGL_NO_SURFACE, EGL_NO_SURFACE, this->frameContext))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to make the frame context current");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-21 00:50:59 +00:00
|
|
|
egl_update_scale_type(this);
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_damageSetup(this->damage, format.width, format.height);
|
2021-07-15 10:04:25 +00:00
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
return egl_desktopSetup(this->desktop, format);
|
2020-10-12 08:43:29 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static bool egl_onFrame(LG_Renderer * renderer, const FrameBuffer * frame, int dmaFd,
|
2021-07-11 01:20:08 +00:00
|
|
|
const FrameDamageRect * damageRects, int damageRectsCount)
|
2020-10-12 08:43:29 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2020-10-12 08:43:29 +00:00
|
|
|
|
2021-07-28 05:06:06 +00:00
|
|
|
uint64_t start = nanotime();
|
2021-08-10 03:10:51 +00:00
|
|
|
if (!egl_desktopUpdate(this->desktop, frame, dmaFd, damageRects, damageRectsCount))
|
2018-12-12 10:41:51 +00:00
|
|
|
{
|
2020-05-21 01:44:56 +00:00
|
|
|
DEBUG_INFO("Failed to to update the desktop");
|
2018-12-12 10:41:51 +00:00
|
|
|
return false;
|
2018-09-23 05:48:44 +00:00
|
|
|
}
|
2021-07-28 05:06:06 +00:00
|
|
|
ringbuffer_push(this->importTimings, &(float){ (nanotime() - start) * 1e-6f });
|
2018-09-23 05:48:44 +00:00
|
|
|
|
2020-08-11 04:45:08 +00:00
|
|
|
this->start = true;
|
2021-07-11 01:20:08 +00:00
|
|
|
|
2021-08-01 08:24:55 +00:00
|
|
|
INTERLOCKED_SECTION(this->desktopDamageLock, {
|
|
|
|
struct DesktopDamage * damage = this->desktopDamage + this->desktopDamageIdx;
|
2021-08-02 01:01:24 +00:00
|
|
|
if (damage->count == -1 || damageRectsCount == 0 ||
|
|
|
|
damage->count + damageRectsCount >= KVMFR_MAX_DAMAGE_RECTS)
|
2021-08-01 08:24:55 +00:00
|
|
|
damage->count = -1;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
memcpy(damage->rects + damage->count, damageRects, damageRectsCount * sizeof(FrameDamageRect));
|
|
|
|
damage->count += damageRectsCount;
|
|
|
|
}
|
|
|
|
});
|
2021-07-11 01:20:08 +00:00
|
|
|
|
2018-09-22 06:26:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-07-18 03:40:03 +00:00
|
|
|
static void debugCallback(GLenum source, GLenum type, GLuint id,
|
|
|
|
GLenum severity, GLsizei length, const GLchar * message,
|
|
|
|
const void * userParam)
|
|
|
|
{
|
|
|
|
enum DebugLevel level = DEBUG_LEVEL_FIXME;
|
|
|
|
switch (severity)
|
|
|
|
{
|
|
|
|
case GL_DEBUG_SEVERITY_HIGH:
|
|
|
|
level = DEBUG_LEVEL_ERROR;
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_SEVERITY_MEDIUM:
|
|
|
|
level = DEBUG_LEVEL_WARN;
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_SEVERITY_LOW:
|
2021-07-18 04:14:31 +00:00
|
|
|
case GL_DEBUG_SEVERITY_NOTIFICATION:
|
2021-07-18 03:40:03 +00:00
|
|
|
level = DEBUG_LEVEL_INFO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char * sourceName = "unknown";
|
|
|
|
switch (source)
|
|
|
|
{
|
|
|
|
case GL_DEBUG_SOURCE_API:
|
|
|
|
sourceName = "OpenGL API";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_SOURCE_WINDOW_SYSTEM:
|
|
|
|
sourceName = "window system";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_SOURCE_SHADER_COMPILER:
|
|
|
|
sourceName = "shader compiler";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_SOURCE_THIRD_PARTY:
|
|
|
|
sourceName = "third party";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_SOURCE_APPLICATION:
|
|
|
|
sourceName = "application";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_SOURCE_OTHER:
|
|
|
|
sourceName = "other";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char * typeName = "unknown";
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case GL_DEBUG_TYPE_ERROR:
|
|
|
|
typeName = "error";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR:
|
|
|
|
typeName = "deprecated behaviour";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR:
|
|
|
|
typeName = "undefined behaviour";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_PORTABILITY:
|
|
|
|
typeName = "portability";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_PERFORMANCE:
|
|
|
|
typeName = "performance";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_MARKER:
|
|
|
|
typeName = "marker";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_PUSH_GROUP:
|
|
|
|
typeName = "group pushing";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_POP_GROUP:
|
|
|
|
typeName = "group popping";
|
|
|
|
break;
|
|
|
|
case GL_DEBUG_TYPE_OTHER:
|
|
|
|
typeName = "other";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_PRINT(level, "GL message (source: %s, type: %s): %s", sourceName, typeName, message);
|
|
|
|
}
|
|
|
|
|
2021-08-11 23:04:45 +00:00
|
|
|
static void egl_configUI(void * opaque, int * id)
|
2021-08-06 07:23:49 +00:00
|
|
|
{
|
|
|
|
struct Inst * this = opaque;
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_damageConfigUI(this->damage);
|
2021-08-06 07:48:07 +00:00
|
|
|
igSeparator();
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_desktopConfigUI(this->desktop);
|
2021-08-06 07:23:49 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static bool egl_renderStartup(LG_Renderer * renderer, bool useDMA)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2021-01-26 10:46:30 +00:00
|
|
|
this->nativeWind = app_getEGLNativeWindow();
|
|
|
|
if (!this->nativeWind)
|
2021-08-14 02:04:54 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to get EGL native window");
|
2018-09-22 06:26:10 +00:00
|
|
|
return false;
|
2021-08-14 02:04:54 +00:00
|
|
|
}
|
2021-01-25 05:04:33 +00:00
|
|
|
|
2021-01-26 10:46:30 +00:00
|
|
|
this->display = app_getEGLDisplay();
|
2018-09-22 06:26:10 +00:00
|
|
|
if (this->display == EGL_NO_DISPLAY)
|
2021-08-14 02:04:54 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to get EGL display");
|
2018-09-22 06:26:10 +00:00
|
|
|
return false;
|
2021-08-14 02:04:54 +00:00
|
|
|
}
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2020-10-29 15:32:25 +00:00
|
|
|
int maj, min;
|
|
|
|
if (!eglInitialize(this->display, &maj, &min))
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Unable to initialize EGL");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-01-15 09:30:03 +00:00
|
|
|
int maxSamples = 1;
|
|
|
|
if (option_get_bool("egl", "multisample"))
|
|
|
|
{
|
|
|
|
if (app_getProp(LG_DS_MAX_MULTISAMPLE, &maxSamples) && maxSamples > 1)
|
|
|
|
{
|
|
|
|
if (maxSamples > 4)
|
|
|
|
maxSamples = 4;
|
|
|
|
|
|
|
|
DEBUG_INFO("Multisampling enabled, max samples: %d", maxSamples);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-22 06:26:10 +00:00
|
|
|
EGLint attr[] =
|
|
|
|
{
|
2021-01-31 04:56:12 +00:00
|
|
|
EGL_BUFFER_SIZE , 24,
|
2018-09-25 13:04:29 +00:00
|
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
2021-01-15 09:30:03 +00:00
|
|
|
EGL_SAMPLE_BUFFERS , maxSamples > 0 ? 1 : 0,
|
|
|
|
EGL_SAMPLES , maxSamples,
|
2018-09-22 06:26:10 +00:00
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
EGLint num_config;
|
|
|
|
if (!eglChooseConfig(this->display, attr, &this->configs, 1, &num_config))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to choose config (eglError: 0x%x)", eglGetError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-12-31 04:31:24 +00:00
|
|
|
const EGLint surfattr[] =
|
|
|
|
{
|
|
|
|
EGL_RENDER_BUFFER, this->opt.doubleBuffer ? EGL_BACK_BUFFER : EGL_SINGLE_BUFFER,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
this->surface = eglCreateWindowSurface(this->display, this->configs, this->nativeWind, surfattr);
|
2018-09-22 06:26:10 +00:00
|
|
|
if (this->surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to create EGL surface (eglError: 0x%x)", eglGetError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-31 23:18:21 +00:00
|
|
|
const char * client_exts = eglQueryString(this->display, EGL_EXTENSIONS);
|
2021-07-18 04:14:31 +00:00
|
|
|
bool debugContext = option_get_bool("egl", "debug");
|
2021-08-19 11:18:11 +00:00
|
|
|
EGLint ctxattr[5];
|
|
|
|
int ctxidx = 0;
|
|
|
|
|
|
|
|
ctxattr[ctxidx++] = EGL_CONTEXT_CLIENT_VERSION;
|
|
|
|
ctxattr[ctxidx++] = 2;
|
|
|
|
|
|
|
|
if (maj > 1 || (maj == 1 && min >= 5))
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-19 11:18:11 +00:00
|
|
|
ctxattr[ctxidx++] = EGL_CONTEXT_OPENGL_DEBUG;
|
|
|
|
ctxattr[ctxidx++] = debugContext ? EGL_TRUE : EGL_FALSE;
|
|
|
|
}
|
2021-08-31 23:18:21 +00:00
|
|
|
else if (util_hasGLExt(client_exts, "EGL_KHR_create_context"))
|
|
|
|
{
|
|
|
|
ctxattr[ctxidx++] = EGL_CONTEXT_FLAGS_KHR;
|
|
|
|
ctxattr[ctxidx++] = debugContext ? EGL_CONTEXT_OPENGL_DEBUG_BIT_KHR : 0;
|
|
|
|
}
|
2021-08-19 11:18:11 +00:00
|
|
|
else if (debugContext)
|
2021-08-31 23:18:21 +00:00
|
|
|
DEBUG_WARN("Cannot create debug contexts before EGL 1.5 without EGL_KHR_create_context");
|
2021-08-19 11:18:11 +00:00
|
|
|
|
|
|
|
ctxattr[ctxidx++] = EGL_NONE;
|
2018-09-22 06:26:10 +00:00
|
|
|
|
|
|
|
this->context = eglCreateContext(this->display, this->configs, EGL_NO_CONTEXT, ctxattr);
|
|
|
|
if (this->context == EGL_NO_CONTEXT)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to create EGL context (eglError: 0x%x)", eglGetError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-12-31 04:31:24 +00:00
|
|
|
EGLint rb = 0;
|
|
|
|
eglQuerySurface(this->display, this->surface, EGL_RENDER_BUFFER, &rb);
|
|
|
|
switch(rb)
|
|
|
|
{
|
|
|
|
case EGL_SINGLE_BUFFER:
|
|
|
|
DEBUG_INFO("Single buffer mode");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EGL_BACK_BUFFER:
|
|
|
|
DEBUG_INFO("Back buffer mode");
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
DEBUG_WARN("Unknown render buffer mode: %d", rb);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-09-22 06:26:10 +00:00
|
|
|
eglMakeCurrent(this->display, this->surface, this->surface, this->context);
|
2021-08-31 23:18:21 +00:00
|
|
|
const char * gl_exts = (const char *)glGetString(GL_EXTENSIONS);
|
|
|
|
const char * vendor = (const char *)glGetString(GL_VENDOR);
|
2020-10-29 15:32:25 +00:00
|
|
|
|
2021-07-18 03:40:03 +00:00
|
|
|
DEBUG_INFO("EGL : %d.%d", maj, min);
|
|
|
|
DEBUG_INFO("Vendor : %s", vendor);
|
|
|
|
DEBUG_INFO("Renderer: %s", glGetString(GL_RENDERER));
|
|
|
|
DEBUG_INFO("Version : %s", glGetString(GL_VERSION ));
|
|
|
|
DEBUG_INFO("EGL APIs: %s", eglQueryString(this->display, EGL_CLIENT_APIS));
|
|
|
|
DEBUG_INFO("EGL Exts: %s", client_exts);
|
|
|
|
DEBUG_INFO("GL Exts : %s", gl_exts);
|
|
|
|
|
|
|
|
GLint esMaj, esMin;
|
|
|
|
glGetIntegerv(GL_MAJOR_VERSION, &esMaj);
|
|
|
|
glGetIntegerv(GL_MINOR_VERSION, &esMin);
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2021-07-18 09:39:28 +00:00
|
|
|
if (!util_hasGLExt(gl_exts, "GL_EXT_buffer_storage"))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("GL_EXT_buffer_storage is needed to use EGL backend");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!util_hasGLExt(gl_exts, "GL_EXT_texture_format_BGRA8888"))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("GL_EXT_texture_format_BGRA8888 is needed to use EGL backend");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-03 01:37:58 +00:00
|
|
|
this->hasBufferAge = util_hasGLExt(client_exts, "EGL_EXT_buffer_age");
|
2021-08-03 01:43:33 +00:00
|
|
|
if (!this->hasBufferAge)
|
|
|
|
DEBUG_WARN("GL_EXT_buffer_age is not supported, will not perform as well.");
|
2021-08-03 01:37:58 +00:00
|
|
|
|
2021-01-26 10:46:30 +00:00
|
|
|
if (g_egl_dynProcs.glEGLImageTargetTexture2DOES)
|
2021-01-24 01:05:18 +00:00
|
|
|
{
|
2021-05-13 20:36:01 +00:00
|
|
|
if (util_hasGLExt(client_exts, "EGL_EXT_image_dma_buf_import"))
|
2021-08-07 23:49:55 +00:00
|
|
|
this->dmaSupport = true;
|
2021-07-29 06:21:34 +00:00
|
|
|
else
|
|
|
|
DEBUG_INFO("EGL_EXT_image_dma_buf_import unavailable, DMA support disabled");
|
2021-01-24 01:05:18 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
DEBUG_INFO("glEGLImageTargetTexture2DOES unavilable, DMA support disabled");
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2021-08-02 17:58:30 +00:00
|
|
|
if (!this->dmaSupport)
|
|
|
|
useDMA = false;
|
|
|
|
|
2021-07-18 04:14:31 +00:00
|
|
|
if (debugContext)
|
2021-07-18 03:40:03 +00:00
|
|
|
{
|
2021-07-18 04:14:31 +00:00
|
|
|
if ((esMaj > 3 || (esMaj == 3 && esMin >= 2)) && g_egl_dynProcs.glDebugMessageCallback)
|
|
|
|
{
|
|
|
|
g_egl_dynProcs.glDebugMessageCallback(debugCallback, NULL);
|
|
|
|
DEBUG_INFO("Using debug message callback from OpenGL ES 3.2+");
|
|
|
|
}
|
|
|
|
else if (util_hasGLExt(gl_exts, "GL_KHR_debug") && g_egl_dynProcs.glDebugMessageCallbackKHR)
|
|
|
|
{
|
|
|
|
g_egl_dynProcs.glDebugMessageCallbackKHR(debugCallback, NULL);
|
|
|
|
DEBUG_INFO("Using debug message callback from GL_KHR_debug");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
DEBUG_INFO("Debug message callback not supported");
|
2021-07-18 03:40:03 +00:00
|
|
|
}
|
|
|
|
else
|
2021-07-18 04:14:31 +00:00
|
|
|
DEBUG_INFO("Debug messages disabled, enable with egl:debug=true");
|
2021-07-18 03:40:03 +00:00
|
|
|
|
2018-12-12 10:41:51 +00:00
|
|
|
eglSwapInterval(this->display, this->opt.vsync ? 1 : 0);
|
2018-09-23 05:48:44 +00:00
|
|
|
|
2021-08-09 04:08:10 +00:00
|
|
|
if (!egl_desktopInit(this, &this->desktop, this->display, useDMA, MAX_ACCUMULATED_DAMAGE))
|
2018-12-12 10:41:51 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to initialize the desktop");
|
2018-09-22 08:00:52 +00:00
|
|
|
return false;
|
2018-12-12 10:41:51 +00:00
|
|
|
}
|
2018-09-24 09:48:11 +00:00
|
|
|
|
2021-08-11 08:53:36 +00:00
|
|
|
if (!egl_cursorInit(&this->cursor))
|
2018-12-12 07:53:55 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to initialize the cursor");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
if (!egl_splashInit(&this->splash))
|
2018-12-15 13:54:37 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to initialize the splash screen");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
if (!egl_damageInit(&this->damage))
|
2021-07-15 10:04:25 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to initialize the damage display");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-07-18 06:38:37 +00:00
|
|
|
if (!ImGui_ImplOpenGL3_Init("#version 300 es"))
|
2021-07-08 01:45:54 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to initialize ImGui");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-11 23:04:45 +00:00
|
|
|
app_overlayConfigRegister("EGL", egl_configUI, this);
|
2021-08-06 07:23:49 +00:00
|
|
|
|
2021-07-08 01:45:54 +00:00
|
|
|
this->imgui = true;
|
2018-09-22 06:26:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:43:42 +00:00
|
|
|
static bool egl_needsRender(LG_Renderer * renderer)
|
2021-08-01 08:18:08 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2021-08-01 08:18:08 +00:00
|
|
|
return !this->waitDone;
|
|
|
|
}
|
|
|
|
|
2021-08-03 01:37:58 +00:00
|
|
|
inline static EGLint egl_buffer_age(struct Inst * this)
|
|
|
|
{
|
|
|
|
if (!this->hasBufferAge)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
EGLint result;
|
|
|
|
eglQuerySurface(this->display, this->surface, EGL_BUFFER_AGE_EXT, &result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline static void renderLetterBox(struct Inst * this)
|
|
|
|
{
|
|
|
|
bool hLB = this->destRect.x > 0;
|
|
|
|
bool vLB = this->destRect.y > 0;
|
|
|
|
|
|
|
|
if (hLB || vLB)
|
|
|
|
{
|
|
|
|
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
|
|
|
|
glEnable(GL_SCISSOR_TEST);
|
|
|
|
|
|
|
|
if (hLB)
|
|
|
|
{
|
2021-08-06 17:28:52 +00:00
|
|
|
glScissor(0.0f, 0.0f, this->destRect.x + 0.5f, this->height);
|
2021-08-03 01:37:58 +00:00
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
|
|
|
|
float x2 = this->destRect.x + this->destRect.w;
|
2021-08-06 18:00:18 +00:00
|
|
|
glScissor(x2 - 0.5, 0.0f, this->width - x2, this->height + 1.0f);
|
2021-08-03 01:37:58 +00:00
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vLB)
|
|
|
|
{
|
2021-08-06 17:28:52 +00:00
|
|
|
glScissor(0.0f, 0.0f, this->width, this->destRect.y + 0.5f);
|
2021-08-03 01:37:58 +00:00
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
|
|
|
|
float y2 = this->destRect.y + this->destRect.h;
|
2021-08-06 18:00:18 +00:00
|
|
|
glScissor(0.0f, y2 - 0.5, this->width, this->height - y2 + 1.0f);
|
2021-08-03 01:37:58 +00:00
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
glDisable(GL_SCISSOR_TEST);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-08 05:32:01 +00:00
|
|
|
static bool egl_render(LG_Renderer * renderer, LG_RendererRotate rotate,
|
|
|
|
const bool newFrame, const bool invalidateWindow,
|
|
|
|
void (*preSwap)(void * udata), void * udata)
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:32:01 +00:00
|
|
|
struct Inst * this = UPCAST(struct Inst, renderer);
|
2021-08-03 01:37:58 +00:00
|
|
|
EGLint bufferAge = egl_buffer_age(this);
|
|
|
|
bool renderAll = invalidateWindow || !this->start || this->hadOverlay ||
|
|
|
|
bufferAge <= 0 || bufferAge > MAX_BUFFER_AGE;
|
2018-09-22 06:26:10 +00:00
|
|
|
|
2021-07-11 07:52:41 +00:00
|
|
|
bool hasOverlay = false;
|
2021-07-19 22:36:22 +00:00
|
|
|
struct CursorState cursorState = { .visible = false };
|
2021-08-01 08:24:55 +00:00
|
|
|
struct DesktopDamage * desktopDamage;
|
|
|
|
|
2021-08-03 01:37:58 +00:00
|
|
|
char accumulated_[
|
|
|
|
sizeof(struct DamageRects) +
|
|
|
|
MAX_ACCUMULATED_DAMAGE * sizeof(struct FrameDamageRect)
|
|
|
|
];
|
|
|
|
struct DamageRects * accumulated = (struct DamageRects *) accumulated_;
|
|
|
|
accumulated->count = 0;
|
|
|
|
|
2021-08-01 08:24:55 +00:00
|
|
|
INTERLOCKED_SECTION(this->desktopDamageLock, {
|
2021-08-03 01:37:58 +00:00
|
|
|
if (!renderAll)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < bufferAge; ++i)
|
|
|
|
{
|
|
|
|
struct DesktopDamage * damage = this->desktopDamage +
|
|
|
|
IDX_AGO(this->desktopDamageIdx, i, DESKTOP_DAMAGE_COUNT);
|
|
|
|
|
|
|
|
if (damage->count < 0)
|
|
|
|
{
|
|
|
|
renderAll = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-08-04 21:56:32 +00:00
|
|
|
for (int j = 0; j < damage->count; ++j)
|
|
|
|
{
|
|
|
|
struct FrameDamageRect * rect = damage->rects + j;
|
|
|
|
int x = rect->x > 0 ? rect->x - 1 : 0;
|
|
|
|
int y = rect->y > 0 ? rect->y - 1 : 0;
|
|
|
|
accumulated->rects[accumulated->count++] = (struct FrameDamageRect) {
|
|
|
|
.x = x, .y = y,
|
|
|
|
.width = min(this->format.width - x, rect->width + 2),
|
|
|
|
.height = min(this->format.height - y, rect->height + 2),
|
|
|
|
};
|
|
|
|
}
|
2021-08-03 01:37:58 +00:00
|
|
|
}
|
|
|
|
}
|
2021-08-01 08:24:55 +00:00
|
|
|
desktopDamage = this->desktopDamage + this->desktopDamageIdx;
|
|
|
|
this->desktopDamageIdx = (this->desktopDamageIdx + 1) % DESKTOP_DAMAGE_COUNT;
|
2021-08-03 01:37:58 +00:00
|
|
|
this->desktopDamage[this->desktopDamageIdx].count = 0;
|
2021-08-01 08:24:55 +00:00
|
|
|
});
|
2021-05-11 20:49:39 +00:00
|
|
|
|
2021-08-03 01:37:58 +00:00
|
|
|
if (!renderAll)
|
|
|
|
{
|
|
|
|
double matrix[6];
|
|
|
|
egl_screenToDesktopMatrix(matrix, this->format.width, this->format.height,
|
|
|
|
this->translateX, this->translateY, this->scaleX, this->scaleY, rotate,
|
|
|
|
this->width, this->height);
|
|
|
|
|
|
|
|
for (int i = 0; i < bufferAge; ++i)
|
|
|
|
{
|
|
|
|
int idx = IDX_AGO(this->overlayHistoryIdx, i, DESKTOP_DAMAGE_COUNT);
|
|
|
|
int count = this->overlayHistoryCount[idx];
|
|
|
|
struct Rect * damage = this->overlayHistory[idx];
|
|
|
|
|
|
|
|
if (count < 0)
|
|
|
|
{
|
|
|
|
renderAll = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int j = 0; j < count; ++j)
|
|
|
|
accumulated->count += egl_screenToDesktop(
|
|
|
|
accumulated->rects + accumulated->count, matrix, damage + j,
|
|
|
|
this->format.width, this->format.height
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-08-11 06:42:31 +00:00
|
|
|
accumulated->count = rectsMergeOverlapping(accumulated->rects, accumulated->count);
|
2021-08-03 01:37:58 +00:00
|
|
|
}
|
2021-08-10 08:54:13 +00:00
|
|
|
++this->overlayHistoryIdx;
|
2021-08-03 01:37:58 +00:00
|
|
|
|
2021-07-11 01:20:08 +00:00
|
|
|
if (this->start)
|
|
|
|
{
|
2021-08-08 07:16:10 +00:00
|
|
|
if (egl_desktopRender(this->desktop,
|
2021-08-11 08:53:36 +00:00
|
|
|
this->destRect.w, this->destRect.h,
|
2020-08-11 04:45:08 +00:00
|
|
|
this->translateX, this->translateY,
|
|
|
|
this->scaleX , this->scaleY ,
|
2021-08-03 01:37:58 +00:00
|
|
|
this->scaleType , rotate, renderAll ? NULL : accumulated))
|
2020-07-19 04:29:29 +00:00
|
|
|
{
|
2021-07-11 01:20:08 +00:00
|
|
|
if (!this->waitFadeTime)
|
|
|
|
{
|
|
|
|
if (!this->params.quickSplash)
|
|
|
|
this->waitFadeTime = microtime() + SPLASH_FADE_TIME;
|
|
|
|
else
|
|
|
|
this->waitDone = true;
|
|
|
|
}
|
2021-01-18 15:44:56 +00:00
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
cursorState = egl_cursorRender(this->cursor,
|
2021-07-19 22:36:22 +00:00
|
|
|
(this->format.rotate + rotate) % LG_ROTATE_MAX,
|
|
|
|
this->width, this->height);
|
2021-07-11 01:20:08 +00:00
|
|
|
}
|
|
|
|
else
|
2021-08-03 01:37:58 +00:00
|
|
|
hasOverlay = true;
|
2019-06-12 12:36:00 +00:00
|
|
|
}
|
2018-11-19 22:50:09 +00:00
|
|
|
|
2021-08-03 12:29:04 +00:00
|
|
|
renderLetterBox(this);
|
|
|
|
|
2018-12-15 13:54:37 +00:00
|
|
|
if (!this->waitDone)
|
|
|
|
{
|
2018-12-16 01:17:12 +00:00
|
|
|
float a = 1.0f;
|
2018-12-15 13:54:37 +00:00
|
|
|
if (!this->waitFadeTime)
|
|
|
|
a = 1.0f;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uint64_t t = microtime();
|
|
|
|
if (t > this->waitFadeTime)
|
|
|
|
this->waitDone = true;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uint64_t delta = this->waitFadeTime - t;
|
2018-12-15 23:57:01 +00:00
|
|
|
a = 1.0f / SPLASH_FADE_TIME * delta;
|
2018-12-15 13:54:37 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-01 23:36:17 +00:00
|
|
|
|
|
|
|
if (!this->waitDone)
|
2021-07-11 07:52:41 +00:00
|
|
|
{
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_splashRender(this->splash, a, this->splashRatio);
|
2021-07-11 07:52:41 +00:00
|
|
|
hasOverlay = true;
|
|
|
|
}
|
2018-12-15 23:57:01 +00:00
|
|
|
}
|
2021-07-11 07:52:41 +00:00
|
|
|
else if (!this->start)
|
2020-08-11 04:54:48 +00:00
|
|
|
{
|
2021-08-08 07:16:10 +00:00
|
|
|
egl_splashRender(this->splash, 1.0f, this->splashRatio);
|
2021-07-11 07:52:41 +00:00
|
|
|
hasOverlay = true;
|
2020-08-11 04:54:48 +00:00
|
|
|
}
|
2018-12-15 23:57:01 +00:00
|
|
|
|
2021-08-08 07:16:10 +00:00
|
|
|
hasOverlay |= egl_damageRender(this->damage, rotate, newFrame ? desktopDamage : NULL);
|
2021-07-25 05:29:29 +00:00
|
|
|
hasOverlay |= invalidateWindow;
|
2021-07-11 07:52:41 +00:00
|
|
|
|
2021-07-22 09:43:16 +00:00
|
|
|
struct Rect damage[KVMFR_MAX_DAMAGE_RECTS + MAX_OVERLAY_RECTS + 2];
|
|
|
|
int damageIdx = app_renderOverlay(damage, MAX_OVERLAY_RECTS);
|
2021-07-22 07:27:30 +00:00
|
|
|
|
2021-07-22 08:53:21 +00:00
|
|
|
switch (damageIdx)
|
2021-07-22 07:27:30 +00:00
|
|
|
{
|
2021-07-22 08:53:21 +00:00
|
|
|
case 0: // no overlay
|
|
|
|
break;
|
|
|
|
case -1: // full damage
|
2021-07-22 07:27:30 +00:00
|
|
|
hasOverlay = true;
|
2021-07-22 08:53:21 +00:00
|
|
|
// fallthrough
|
|
|
|
default:
|
|
|
|
ImGui_ImplOpenGL3_NewFrame();
|
|
|
|
ImGui_ImplOpenGL3_RenderDrawData(igGetDrawData());
|
|
|
|
|
|
|
|
for (int i = 0; i < damageIdx; ++i)
|
|
|
|
damage[i].y = this->height - damage[i].y - damage[i].h;
|
2021-07-22 07:27:30 +00:00
|
|
|
}
|
2021-05-11 20:49:39 +00:00
|
|
|
|
2021-08-03 01:37:58 +00:00
|
|
|
if (damageIdx >= 0 && cursorState.visible)
|
|
|
|
damage[damageIdx++] = cursorState.rect;
|
|
|
|
|
|
|
|
int overlayHistoryIdx = this->overlayHistoryIdx % DESKTOP_DAMAGE_COUNT;
|
|
|
|
if (hasOverlay)
|
|
|
|
this->overlayHistoryCount[overlayHistoryIdx] = -1;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (damageIdx > 0)
|
|
|
|
memcpy(this->overlayHistory[overlayHistoryIdx], damage, damageIdx * sizeof(struct Rect));
|
|
|
|
this->overlayHistoryCount[overlayHistoryIdx] = damageIdx;
|
|
|
|
}
|
|
|
|
|
2021-07-11 07:52:41 +00:00
|
|
|
if (!hasOverlay && !this->hadOverlay)
|
2021-05-11 20:49:39 +00:00
|
|
|
{
|
2021-07-19 22:36:22 +00:00
|
|
|
if (this->cursorLast.visible)
|
|
|
|
damage[damageIdx++] = this->cursorLast.rect;
|
2021-05-11 20:49:39 +00:00
|
|
|
|
2021-08-01 00:37:23 +00:00
|
|
|
if (desktopDamage->count == -1)
|
|
|
|
// -1 damage count means invalidating entire window.
|
|
|
|
damageIdx = 0;
|
|
|
|
else
|
2021-07-11 01:20:08 +00:00
|
|
|
{
|
2021-08-02 22:40:19 +00:00
|
|
|
double matrix[6];
|
|
|
|
egl_desktopToScreenMatrix(matrix, this->format.width, this->format.height,
|
|
|
|
this->translateX, this->translateY, this->scaleX, this->scaleY, rotate,
|
|
|
|
this->width, this->height);
|
|
|
|
|
2021-08-01 00:37:23 +00:00
|
|
|
for (int i = 0; i < desktopDamage->count; ++i)
|
2021-08-02 22:40:19 +00:00
|
|
|
damage[damageIdx++] = egl_desktopToScreen(matrix, desktopDamage->rects + i);
|
2021-05-11 20:49:39 +00:00
|
|
|
}
|
|
|
|
}
|
2021-07-22 08:58:32 +00:00
|
|
|
else
|
|
|
|
damageIdx = 0;
|
2021-08-01 00:37:23 +00:00
|
|
|
|
2021-07-11 07:52:41 +00:00
|
|
|
this->hadOverlay = hasOverlay;
|
2021-08-01 05:47:59 +00:00
|
|
|
this->cursorLast = cursorState;
|
2021-07-08 01:45:54 +00:00
|
|
|
|
2021-08-06 15:44:36 +00:00
|
|
|
preSwap(udata);
|
2021-05-11 20:49:39 +00:00
|
|
|
app_eglSwapBuffers(this->display, this->surface, damage, damageIdx);
|
2018-09-22 06:26:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-08 04:43:04 +00:00
|
|
|
struct LG_RendererOps LGR_EGL =
|
2018-09-22 06:26:10 +00:00
|
|
|
{
|
2021-08-08 05:43:42 +00:00
|
|
|
.getName = egl_getName,
|
|
|
|
.setup = egl_setup,
|
|
|
|
.create = egl_create,
|
|
|
|
.initialize = egl_initialize,
|
|
|
|
.deinitialize = egl_deinitialize,
|
|
|
|
.supports = egl_supports,
|
|
|
|
.onRestart = egl_onRestart,
|
|
|
|
.onResize = egl_onResize,
|
|
|
|
.onMouseShape = egl_onMouseShape,
|
|
|
|
.onMouseEvent = egl_onMouseEvent,
|
|
|
|
.onFrameFormat = egl_onFrameFormat,
|
|
|
|
.onFrame = egl_onFrame,
|
|
|
|
.renderStartup = egl_renderStartup,
|
|
|
|
.needsRender = egl_needsRender,
|
|
|
|
.render = egl_render
|
2020-01-17 00:50:00 +00:00
|
|
|
};
|