2018-12-15 16:11:41 -05:00
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
// Copyright (c) Yuxuan Shui <yshuiv7@gmail.com>
|
|
|
|
|
|
|
|
/// Function pointers to init VSync modes.
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "log.h"
|
2018-12-15 21:48:02 -05:00
|
|
|
|
|
|
|
#ifdef CONFIG_OPENGL
|
2019-02-08 17:34:53 -05:00
|
|
|
#include "backend/gl/glx.h"
|
2018-12-15 16:11:41 -05:00
|
|
|
#include "opengl.h"
|
2018-12-15 21:48:02 -05:00
|
|
|
#endif
|
2018-12-15 16:11:41 -05:00
|
|
|
|
2018-12-21 17:25:24 -05:00
|
|
|
#ifdef CONFIG_VSYNC_DRM
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <drm.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#endif
|
|
|
|
|
2019-01-18 18:30:44 -05:00
|
|
|
#include "config.h"
|
2018-12-15 16:11:41 -05:00
|
|
|
#include "vsync.h"
|
|
|
|
|
2018-12-21 17:25:24 -05:00
|
|
|
#ifdef CONFIG_VSYNC_DRM
|
|
|
|
/**
|
|
|
|
* Wait for next VSync, DRM method.
|
|
|
|
*
|
|
|
|
* Stolen from: https://github.com/MythTV/mythtv/blob/master/mythtv/libs/libmythtv/vsync.cpp
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
vsync_drm_wait(session_t *ps) {
|
|
|
|
int ret = -1;
|
|
|
|
drm_wait_vblank_t vbl;
|
|
|
|
|
|
|
|
vbl.request.type = _DRM_VBLANK_RELATIVE,
|
|
|
|
vbl.request.sequence = 1;
|
|
|
|
|
|
|
|
do {
|
|
|
|
ret = ioctl(ps->drm_fd, DRM_IOCTL_WAIT_VBLANK, &vbl);
|
|
|
|
vbl.request.type &= ~_DRM_VBLANK_RELATIVE;
|
|
|
|
} while (ret && errno == EINTR);
|
|
|
|
|
|
|
|
if (ret)
|
|
|
|
log_error("VBlank ioctl did not work, unimplemented in this drmver?");
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-12-15 16:11:41 -05:00
|
|
|
/**
|
|
|
|
* Initialize DRM VSync.
|
|
|
|
*
|
|
|
|
* @return true for success, false otherwise
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
vsync_drm_init(session_t *ps) {
|
|
|
|
#ifdef CONFIG_VSYNC_DRM
|
|
|
|
// Should we always open card0?
|
|
|
|
if (ps->drm_fd < 0 && (ps->drm_fd = open("/dev/dri/card0", O_RDWR)) < 0) {
|
2018-12-20 12:19:41 -05:00
|
|
|
log_error("Failed to open device.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vsync_drm_wait(ps))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
#else
|
2018-12-20 12:19:41 -05:00
|
|
|
log_error("compton is not compiled with DRM VSync support.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize OpenGL VSync.
|
|
|
|
*
|
|
|
|
* Stolen from: http://git.tuxfamily.org/?p=ccm/cairocompmgr.git;a=commitdiff;h=efa4ceb97da501e8630ca7f12c99b1dce853c73e
|
|
|
|
* Possible original source: http://www.inb.uni-luebeck.de/~boehme/xvideo_sync.html
|
|
|
|
*
|
|
|
|
* @return true for success, false otherwise
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
vsync_opengl_init(session_t *ps) {
|
|
|
|
#ifdef CONFIG_OPENGL
|
|
|
|
if (!ensure_glx_context(ps))
|
|
|
|
return false;
|
|
|
|
|
2019-02-08 18:57:14 -05:00
|
|
|
return glxext.has_GLX_SGI_video_sync;
|
2018-12-15 16:11:41 -05:00
|
|
|
#else
|
2018-12-20 12:19:41 -05:00
|
|
|
log_error("compton is not compiled with OpenGL VSync support.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
vsync_opengl_oml_init(session_t *ps) {
|
|
|
|
#ifdef CONFIG_OPENGL
|
|
|
|
if (!ensure_glx_context(ps))
|
|
|
|
return false;
|
|
|
|
|
2019-02-08 18:57:14 -05:00
|
|
|
return glxext.has_GLX_OML_sync_control;
|
2018-12-15 16:11:41 -05:00
|
|
|
#else
|
2018-12-20 12:19:41 -05:00
|
|
|
log_error("compton is not compiled with OpenGL VSync support.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_OPENGL
|
2019-02-08 18:57:14 -05:00
|
|
|
static inline bool
|
|
|
|
vsync_opengl_swc_swap_interval(session_t *ps, unsigned int interval) {
|
|
|
|
if (glxext.has_GLX_MESA_swap_control)
|
|
|
|
return glXSwapIntervalMESA(interval) == 0;
|
|
|
|
else if (glxext.has_GLX_SGI_swap_control)
|
|
|
|
return glXSwapIntervalSGI(interval) == 0;
|
|
|
|
else if (glxext.has_GLX_EXT_swap_control) {
|
|
|
|
GLXDrawable d = glXGetCurrentDrawable();
|
|
|
|
if (d == None) {
|
|
|
|
// We don't have a context??
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
}
|
2019-02-08 18:57:14 -05:00
|
|
|
glXSwapIntervalEXT(ps->dpy, glXGetCurrentDrawable(), interval);
|
|
|
|
return true;
|
2018-12-15 16:11:41 -05:00
|
|
|
}
|
2019-02-08 18:57:14 -05:00
|
|
|
return false;
|
2018-12-15 16:11:41 -05:00
|
|
|
}
|
2019-02-08 18:57:14 -05:00
|
|
|
#endif
|
2018-12-15 16:11:41 -05:00
|
|
|
|
|
|
|
static bool
|
|
|
|
vsync_opengl_swc_init(session_t *ps) {
|
|
|
|
#ifdef CONFIG_OPENGL
|
|
|
|
if (!bkend_use_glx(ps)) {
|
2018-12-20 12:19:41 -05:00
|
|
|
log_warn("OpenGL swap control requires the GLX backend.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!vsync_opengl_swc_swap_interval(ps, 1)) {
|
2018-12-20 12:19:41 -05:00
|
|
|
log_error("Failed to load a swap control extension.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
#else
|
2018-12-20 12:19:41 -05:00
|
|
|
log_error("compton is not compiled with OpenGL VSync support.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
vsync_opengl_mswc_init(session_t *ps) {
|
2018-12-20 12:19:41 -05:00
|
|
|
log_warn("opengl-mswc is deprecated, please use opengl-swc instead.");
|
2018-12-15 16:11:41 -05:00
|
|
|
return vsync_opengl_swc_init(ps);
|
|
|
|
}
|
|
|
|
|
2018-12-16 19:50:57 -05:00
|
|
|
bool (*const VSYNC_FUNCS_INIT[NUM_VSYNC])(session_t *ps) = {
|
2018-12-15 16:11:41 -05:00
|
|
|
[VSYNC_DRM ] = vsync_drm_init,
|
|
|
|
[VSYNC_OPENGL ] = vsync_opengl_init,
|
|
|
|
[VSYNC_OPENGL_OML ] = vsync_opengl_oml_init,
|
|
|
|
[VSYNC_OPENGL_SWC ] = vsync_opengl_swc_init,
|
|
|
|
[VSYNC_OPENGL_MSWC ] = vsync_opengl_mswc_init,
|
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef CONFIG_OPENGL
|
|
|
|
/**
|
|
|
|
* Wait for next VSync, OpenGL method.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
vsync_opengl_wait(session_t *ps) {
|
|
|
|
unsigned vblank_count = 0;
|
|
|
|
|
2019-02-08 18:57:14 -05:00
|
|
|
glXGetVideoSyncSGI(&vblank_count);
|
|
|
|
glXWaitVideoSyncSGI(2, (vblank_count + 1) % 2, &vblank_count);
|
2018-12-15 16:11:41 -05:00
|
|
|
// I see some code calling glXSwapIntervalSGI(1) afterwards, is it required?
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Wait for next VSync, OpenGL OML method.
|
|
|
|
*
|
|
|
|
* https://mail.gnome.org/archives/clutter-list/2012-November/msg00031.html
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
vsync_opengl_oml_wait(session_t *ps) {
|
|
|
|
int64_t ust = 0, msc = 0, sbc = 0;
|
|
|
|
|
2019-02-08 18:57:14 -05:00
|
|
|
glXGetSyncValuesOML(ps->dpy, ps->reg_win, &ust, &msc, &sbc);
|
|
|
|
glXWaitForMscOML(ps->dpy, ps->reg_win, 0, 2, (msc + 1) % 2,
|
2018-12-15 16:11:41 -05:00
|
|
|
&ust, &msc, &sbc);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/// Function pointers to wait for VSync.
|
2018-12-16 19:50:57 -05:00
|
|
|
int (*const VSYNC_FUNCS_WAIT[NUM_VSYNC])(session_t *ps) = {
|
2018-12-15 16:11:41 -05:00
|
|
|
#ifdef CONFIG_VSYNC_DRM
|
|
|
|
[VSYNC_DRM ] = vsync_drm_wait,
|
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_OPENGL
|
|
|
|
[VSYNC_OPENGL ] = vsync_opengl_wait,
|
|
|
|
[VSYNC_OPENGL_OML ] = vsync_opengl_oml_wait,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef CONFIG_OPENGL
|
|
|
|
static void
|
|
|
|
vsync_opengl_swc_deinit(session_t *ps) {
|
|
|
|
vsync_opengl_swc_swap_interval(ps, 0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/// Function pointers to deinitialize VSync.
|
2018-12-16 19:50:57 -05:00
|
|
|
void (*const VSYNC_FUNCS_DEINIT[NUM_VSYNC])(session_t *ps) = {
|
2018-12-15 16:11:41 -05:00
|
|
|
#ifdef CONFIG_OPENGL
|
|
|
|
[VSYNC_OPENGL_SWC ] = vsync_opengl_swc_deinit,
|
|
|
|
[VSYNC_OPENGL_MSWC ] = vsync_opengl_swc_deinit,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize current VSync method.
|
|
|
|
*/
|
|
|
|
bool vsync_init(session_t *ps) {
|
|
|
|
// Mesa turns on swap control by default, undo that
|
2019-02-08 19:16:47 -05:00
|
|
|
#ifdef CONFIG_OPENGL
|
2018-12-15 16:11:41 -05:00
|
|
|
if (bkend_use_glx(ps))
|
|
|
|
vsync_opengl_swc_swap_interval(ps, 0);
|
2019-02-08 19:16:47 -05:00
|
|
|
#endif
|
2018-12-15 16:11:41 -05:00
|
|
|
|
|
|
|
if (ps->o.vsync && VSYNC_FUNCS_INIT[ps->o.vsync]
|
|
|
|
&& !VSYNC_FUNCS_INIT[ps->o.vsync](ps)) {
|
|
|
|
ps->o.vsync = VSYNC_NONE;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Wait for next VSync.
|
|
|
|
*/
|
|
|
|
void vsync_wait(session_t *ps) {
|
|
|
|
if (!ps->o.vsync)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (VSYNC_FUNCS_WAIT[ps->o.vsync])
|
|
|
|
VSYNC_FUNCS_WAIT[ps->o.vsync](ps);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Deinitialize current VSync method.
|
|
|
|
*/
|
|
|
|
void vsync_deinit(session_t *ps) {
|
|
|
|
if (ps->o.vsync && VSYNC_FUNCS_DEINIT[ps->o.vsync])
|
|
|
|
VSYNC_FUNCS_DEINIT[ps->o.vsync](ps);
|
|
|
|
}
|