2015-06-21 17:33:46 +02:00
|
|
|
/*
|
|
|
|
Simple DirectMedia Layer
|
2018-01-03 19:03:25 +01:00
|
|
|
Copyright (C) 1997-2018 Sam Lantinga <slouken@libsdl.org>
|
2015-06-21 17:33:46 +02:00
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any damages
|
|
|
|
arising from the use of this software.
|
|
|
|
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it
|
|
|
|
freely, subject to the following restrictions:
|
|
|
|
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
|
|
claim that you wrote the original software. If you use this software
|
|
|
|
in a product, an acknowledgment in the product documentation would be
|
|
|
|
appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "../../SDL_internal.h"
|
|
|
|
|
|
|
|
#if SDL_VIDEO_DRIVER_WAYLAND
|
|
|
|
|
|
|
|
#include "SDL_video.h"
|
|
|
|
#include "SDL_mouse.h"
|
|
|
|
#include "SDL_stdinc.h"
|
|
|
|
#include "../../events/SDL_events_c.h"
|
|
|
|
|
|
|
|
#include "SDL_waylandvideo.h"
|
|
|
|
#include "SDL_waylandevents_c.h"
|
|
|
|
#include "SDL_waylandwindow.h"
|
|
|
|
#include "SDL_waylandopengles.h"
|
|
|
|
#include "SDL_waylandmouse.h"
|
|
|
|
#include "SDL_waylandtouch.h"
|
2016-11-06 17:34:27 +01:00
|
|
|
#include "SDL_waylandclipboard.h"
|
2017-08-28 04:15:57 +02:00
|
|
|
#include "SDL_waylandvulkan.h"
|
2015-06-21 17:33:46 +02:00
|
|
|
|
2016-09-01 10:22:58 +02:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
2015-06-21 17:33:46 +02:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <xkbcommon/xkbcommon.h>
|
|
|
|
|
|
|
|
#include "SDL_waylanddyn.h"
|
|
|
|
#include <wayland-util.h>
|
|
|
|
|
2018-02-07 19:13:55 +01:00
|
|
|
#include "xdg-shell-unstable-v6-client-protocol.h"
|
|
|
|
|
2015-06-21 17:33:46 +02:00
|
|
|
#define WAYLANDVID_DRIVER_NAME "wayland"
|
|
|
|
|
|
|
|
/* Initialization/Query functions */
|
|
|
|
static int
|
|
|
|
Wayland_VideoInit(_THIS);
|
|
|
|
|
|
|
|
static void
|
|
|
|
Wayland_GetDisplayModes(_THIS, SDL_VideoDisplay *sdl_display);
|
|
|
|
static int
|
|
|
|
Wayland_SetDisplayMode(_THIS, SDL_VideoDisplay *display, SDL_DisplayMode *mode);
|
|
|
|
|
|
|
|
static void
|
|
|
|
Wayland_VideoQuit(_THIS);
|
|
|
|
|
2016-09-01 10:22:58 +02:00
|
|
|
/* Find out what class name we should use
|
|
|
|
* Based on src/video/x11/SDL_x11video.c */
|
|
|
|
static char *
|
|
|
|
get_classname()
|
|
|
|
{
|
2018-02-07 19:13:55 +01:00
|
|
|
/* !!! FIXME: this is probably wrong, albeit harmless in many common cases. From protocol spec:
|
|
|
|
"The surface class identifies the general class of applications
|
|
|
|
to which the surface belongs. A common convention is to use the
|
|
|
|
file name (or the full path if it is a non-standard location) of
|
|
|
|
the application's .desktop file as the class." */
|
|
|
|
|
2016-09-01 10:22:58 +02:00
|
|
|
char *spot;
|
|
|
|
#if defined(__LINUX__) || defined(__FREEBSD__)
|
|
|
|
char procfile[1024];
|
|
|
|
char linkfile[1024];
|
|
|
|
int linksize;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* First allow environment variable override */
|
|
|
|
spot = SDL_getenv("SDL_VIDEO_WAYLAND_WMCLASS");
|
|
|
|
if (spot) {
|
|
|
|
return SDL_strdup(spot);
|
|
|
|
} else {
|
|
|
|
/* Fallback to the "old" envvar */
|
|
|
|
spot = SDL_getenv("SDL_VIDEO_X11_WMCLASS");
|
|
|
|
if (spot) {
|
|
|
|
return SDL_strdup(spot);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Next look at the application's executable name */
|
|
|
|
#if defined(__LINUX__) || defined(__FREEBSD__)
|
|
|
|
#if defined(__LINUX__)
|
|
|
|
SDL_snprintf(procfile, SDL_arraysize(procfile), "/proc/%d/exe", getpid());
|
|
|
|
#elif defined(__FREEBSD__)
|
|
|
|
SDL_snprintf(procfile, SDL_arraysize(procfile), "/proc/%d/file",
|
|
|
|
getpid());
|
|
|
|
#else
|
|
|
|
#error Where can we find the executable name?
|
|
|
|
#endif
|
|
|
|
linksize = readlink(procfile, linkfile, sizeof(linkfile) - 1);
|
|
|
|
if (linksize > 0) {
|
|
|
|
linkfile[linksize] = '\0';
|
|
|
|
spot = SDL_strrchr(linkfile, '/');
|
|
|
|
if (spot) {
|
|
|
|
return SDL_strdup(spot + 1);
|
|
|
|
} else {
|
|
|
|
return SDL_strdup(linkfile);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* __LINUX__ || __FREEBSD__ */
|
|
|
|
|
|
|
|
/* Finally use the default we've used forever */
|
|
|
|
return SDL_strdup("SDL_App");
|
|
|
|
}
|
|
|
|
|
2015-06-21 17:33:46 +02:00
|
|
|
/* Wayland driver bootstrap functions */
|
|
|
|
static int
|
|
|
|
Wayland_Available(void)
|
|
|
|
{
|
|
|
|
struct wl_display *display = NULL;
|
|
|
|
if (SDL_WAYLAND_LoadSymbols()) {
|
|
|
|
display = WAYLAND_wl_display_connect(NULL);
|
|
|
|
if (display != NULL) {
|
|
|
|
WAYLAND_wl_display_disconnect(display);
|
|
|
|
}
|
|
|
|
SDL_WAYLAND_UnloadSymbols();
|
|
|
|
}
|
|
|
|
|
|
|
|
return (display != NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
Wayland_DeleteDevice(SDL_VideoDevice *device)
|
|
|
|
{
|
|
|
|
SDL_free(device);
|
|
|
|
SDL_WAYLAND_UnloadSymbols();
|
|
|
|
}
|
|
|
|
|
|
|
|
static SDL_VideoDevice *
|
|
|
|
Wayland_CreateDevice(int devindex)
|
|
|
|
{
|
|
|
|
SDL_VideoDevice *device;
|
|
|
|
|
|
|
|
if (!SDL_WAYLAND_LoadSymbols()) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize all variables that we clean on shutdown */
|
|
|
|
device = SDL_calloc(1, sizeof(SDL_VideoDevice));
|
|
|
|
if (!device) {
|
|
|
|
SDL_WAYLAND_UnloadSymbols();
|
|
|
|
SDL_OutOfMemory();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the function pointers */
|
|
|
|
device->VideoInit = Wayland_VideoInit;
|
|
|
|
device->VideoQuit = Wayland_VideoQuit;
|
|
|
|
device->SetDisplayMode = Wayland_SetDisplayMode;
|
|
|
|
device->GetDisplayModes = Wayland_GetDisplayModes;
|
|
|
|
device->GetWindowWMInfo = Wayland_GetWindowWMInfo;
|
|
|
|
|
|
|
|
device->PumpEvents = Wayland_PumpEvents;
|
|
|
|
|
|
|
|
device->GL_SwapWindow = Wayland_GLES_SwapWindow;
|
|
|
|
device->GL_GetSwapInterval = Wayland_GLES_GetSwapInterval;
|
|
|
|
device->GL_SetSwapInterval = Wayland_GLES_SetSwapInterval;
|
|
|
|
device->GL_MakeCurrent = Wayland_GLES_MakeCurrent;
|
|
|
|
device->GL_CreateContext = Wayland_GLES_CreateContext;
|
|
|
|
device->GL_LoadLibrary = Wayland_GLES_LoadLibrary;
|
|
|
|
device->GL_UnloadLibrary = Wayland_GLES_UnloadLibrary;
|
|
|
|
device->GL_GetProcAddress = Wayland_GLES_GetProcAddress;
|
|
|
|
device->GL_DeleteContext = Wayland_GLES_DeleteContext;
|
|
|
|
|
2017-08-28 09:43:14 +02:00
|
|
|
device->CreateSDLWindow = Wayland_CreateWindow;
|
2015-06-21 17:33:46 +02:00
|
|
|
device->ShowWindow = Wayland_ShowWindow;
|
|
|
|
device->SetWindowFullscreen = Wayland_SetWindowFullscreen;
|
2016-10-08 03:11:03 +02:00
|
|
|
device->MaximizeWindow = Wayland_MaximizeWindow;
|
|
|
|
device->RestoreWindow = Wayland_RestoreWindow;
|
2015-06-21 17:33:46 +02:00
|
|
|
device->SetWindowSize = Wayland_SetWindowSize;
|
2016-10-08 03:11:03 +02:00
|
|
|
device->SetWindowTitle = Wayland_SetWindowTitle;
|
2015-06-21 17:33:46 +02:00
|
|
|
device->DestroyWindow = Wayland_DestroyWindow;
|
|
|
|
device->SetWindowHitTest = Wayland_SetWindowHitTest;
|
|
|
|
|
2016-11-06 17:34:27 +01:00
|
|
|
device->SetClipboardText = Wayland_SetClipboardText;
|
|
|
|
device->GetClipboardText = Wayland_GetClipboardText;
|
|
|
|
device->HasClipboardText = Wayland_HasClipboardText;
|
|
|
|
|
2017-08-28 09:11:38 +02:00
|
|
|
#if SDL_VIDEO_VULKAN
|
2017-08-28 04:15:57 +02:00
|
|
|
device->Vulkan_LoadLibrary = Wayland_Vulkan_LoadLibrary;
|
|
|
|
device->Vulkan_UnloadLibrary = Wayland_Vulkan_UnloadLibrary;
|
|
|
|
device->Vulkan_GetInstanceExtensions = Wayland_Vulkan_GetInstanceExtensions;
|
|
|
|
device->Vulkan_CreateSurface = Wayland_Vulkan_CreateSurface;
|
|
|
|
#endif
|
|
|
|
|
2015-06-21 17:33:46 +02:00
|
|
|
device->free = Wayland_DeleteDevice;
|
|
|
|
|
|
|
|
return device;
|
|
|
|
}
|
|
|
|
|
|
|
|
VideoBootStrap Wayland_bootstrap = {
|
|
|
|
WAYLANDVID_DRIVER_NAME, "SDL Wayland video driver",
|
|
|
|
Wayland_Available, Wayland_CreateDevice
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
display_handle_geometry(void *data,
|
|
|
|
struct wl_output *output,
|
|
|
|
int x, int y,
|
|
|
|
int physical_width,
|
|
|
|
int physical_height,
|
|
|
|
int subpixel,
|
|
|
|
const char *make,
|
|
|
|
const char *model,
|
|
|
|
int transform)
|
|
|
|
|
|
|
|
{
|
|
|
|
SDL_VideoDisplay *display = data;
|
|
|
|
|
2016-06-28 21:08:23 +02:00
|
|
|
display->name = SDL_strdup(model);
|
2015-06-21 17:33:46 +02:00
|
|
|
display->driverdata = output;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
display_handle_mode(void *data,
|
|
|
|
struct wl_output *output,
|
|
|
|
uint32_t flags,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
int refresh)
|
|
|
|
{
|
|
|
|
SDL_VideoDisplay *display = data;
|
|
|
|
SDL_DisplayMode mode;
|
|
|
|
|
|
|
|
SDL_zero(mode);
|
2017-01-03 09:44:05 +01:00
|
|
|
mode.format = SDL_PIXELFORMAT_RGB888;
|
2015-06-21 17:33:46 +02:00
|
|
|
mode.w = width;
|
|
|
|
mode.h = height;
|
|
|
|
mode.refresh_rate = refresh / 1000; // mHz to Hz
|
2018-01-03 19:49:26 +01:00
|
|
|
mode.driverdata = display->driverdata;
|
2015-06-21 17:33:46 +02:00
|
|
|
SDL_AddDisplayMode(display, &mode);
|
|
|
|
|
|
|
|
if (flags & WL_OUTPUT_MODE_CURRENT) {
|
|
|
|
display->current_mode = mode;
|
|
|
|
display->desktop_mode = mode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
display_handle_done(void *data,
|
|
|
|
struct wl_output *output)
|
|
|
|
{
|
|
|
|
SDL_VideoDisplay *display = data;
|
|
|
|
SDL_AddVideoDisplay(display);
|
|
|
|
SDL_free(display->name);
|
|
|
|
SDL_free(display);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
display_handle_scale(void *data,
|
|
|
|
struct wl_output *output,
|
|
|
|
int32_t factor)
|
|
|
|
{
|
|
|
|
// TODO: do HiDPI stuff.
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_output_listener output_listener = {
|
|
|
|
display_handle_geometry,
|
|
|
|
display_handle_mode,
|
|
|
|
display_handle_done,
|
|
|
|
display_handle_scale
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
Wayland_add_display(SDL_VideoData *d, uint32_t id)
|
|
|
|
{
|
|
|
|
struct wl_output *output;
|
|
|
|
SDL_VideoDisplay *display = SDL_malloc(sizeof *display);
|
|
|
|
if (!display) {
|
|
|
|
SDL_OutOfMemory();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
SDL_zero(*display);
|
|
|
|
|
|
|
|
output = wl_registry_bind(d->registry, id, &wl_output_interface, 2);
|
|
|
|
if (!output) {
|
|
|
|
SDL_SetError("Failed to retrieve output.");
|
2016-11-16 22:09:40 +01:00
|
|
|
SDL_free(display);
|
2015-06-21 17:33:46 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl_output_add_listener(output, &output_listener, display);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SDL_VIDEO_DRIVER_WAYLAND_QT_TOUCH
|
|
|
|
static void
|
|
|
|
windowmanager_hints(void *data, struct qt_windowmanager *qt_windowmanager,
|
|
|
|
int32_t show_is_fullscreen)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
windowmanager_quit(void *data, struct qt_windowmanager *qt_windowmanager)
|
|
|
|
{
|
|
|
|
SDL_SendQuit();
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct qt_windowmanager_listener windowmanager_listener = {
|
|
|
|
windowmanager_hints,
|
|
|
|
windowmanager_quit,
|
|
|
|
};
|
|
|
|
#endif /* SDL_VIDEO_DRIVER_WAYLAND_QT_TOUCH */
|
|
|
|
|
2018-02-07 19:13:55 +01:00
|
|
|
|
|
|
|
static void
|
|
|
|
handle_ping_zxdg_shell(void *data, struct zxdg_shell_v6 *zxdg, uint32_t serial)
|
|
|
|
{
|
|
|
|
zxdg_shell_v6_pong(zxdg, serial);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct zxdg_shell_v6_listener shell_listener_zxdg = {
|
|
|
|
handle_ping_zxdg_shell
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-06-21 17:33:46 +02:00
|
|
|
static void
|
|
|
|
display_handle_global(void *data, struct wl_registry *registry, uint32_t id,
|
|
|
|
const char *interface, uint32_t version)
|
|
|
|
{
|
|
|
|
SDL_VideoData *d = data;
|
|
|
|
|
|
|
|
if (strcmp(interface, "wl_compositor") == 0) {
|
|
|
|
d->compositor = wl_registry_bind(d->registry, id, &wl_compositor_interface, 1);
|
|
|
|
} else if (strcmp(interface, "wl_output") == 0) {
|
|
|
|
Wayland_add_display(d, id);
|
|
|
|
} else if (strcmp(interface, "wl_seat") == 0) {
|
|
|
|
Wayland_display_add_input(d, id);
|
2018-02-07 19:13:55 +01:00
|
|
|
} else if (strcmp(interface, "zxdg_shell_v6") == 0) {
|
|
|
|
d->shell.zxdg = wl_registry_bind(d->registry, id, &zxdg_shell_v6_interface, 1);
|
|
|
|
zxdg_shell_v6_add_listener(d->shell.zxdg, &shell_listener_zxdg, NULL);
|
2015-06-21 17:33:46 +02:00
|
|
|
} else if (strcmp(interface, "wl_shell") == 0) {
|
2018-02-07 19:13:55 +01:00
|
|
|
d->shell.wl = wl_registry_bind(d->registry, id, &wl_shell_interface, 1);
|
2015-06-21 17:33:46 +02:00
|
|
|
} else if (strcmp(interface, "wl_shm") == 0) {
|
|
|
|
d->shm = wl_registry_bind(registry, id, &wl_shm_interface, 1);
|
|
|
|
d->cursor_theme = WAYLAND_wl_cursor_theme_load(NULL, 32, d->shm);
|
2016-09-01 10:26:56 +02:00
|
|
|
} else if (strcmp(interface, "zwp_relative_pointer_manager_v1") == 0) {
|
|
|
|
Wayland_display_add_relative_pointer_manager(d, id);
|
|
|
|
} else if (strcmp(interface, "zwp_pointer_constraints_v1") == 0) {
|
|
|
|
Wayland_display_add_pointer_constraints(d, id);
|
2016-11-06 17:34:27 +01:00
|
|
|
} else if (strcmp(interface, "wl_data_device_manager") == 0) {
|
|
|
|
d->data_device_manager = wl_registry_bind(d->registry, id, &wl_data_device_manager_interface, 3);
|
2018-02-07 19:13:55 +01:00
|
|
|
|
2015-06-21 17:33:46 +02:00
|
|
|
#ifdef SDL_VIDEO_DRIVER_WAYLAND_QT_TOUCH
|
|
|
|
} else if (strcmp(interface, "qt_touch_extension") == 0) {
|
|
|
|
Wayland_touch_create(d, id);
|
|
|
|
} else if (strcmp(interface, "qt_surface_extension") == 0) {
|
|
|
|
d->surface_extension = wl_registry_bind(registry, id,
|
|
|
|
&qt_surface_extension_interface, 1);
|
|
|
|
} else if (strcmp(interface, "qt_windowmanager") == 0) {
|
|
|
|
d->windowmanager = wl_registry_bind(registry, id,
|
|
|
|
&qt_windowmanager_interface, 1);
|
|
|
|
qt_windowmanager_add_listener(d->windowmanager, &windowmanager_listener, d);
|
|
|
|
#endif /* SDL_VIDEO_DRIVER_WAYLAND_QT_TOUCH */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_registry_listener registry_listener = {
|
2017-09-29 19:07:37 +02:00
|
|
|
display_handle_global,
|
|
|
|
NULL, /* global_remove */
|
2015-06-21 17:33:46 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
int
|
|
|
|
Wayland_VideoInit(_THIS)
|
|
|
|
{
|
|
|
|
SDL_VideoData *data = SDL_malloc(sizeof *data);
|
|
|
|
if (data == NULL)
|
|
|
|
return SDL_OutOfMemory();
|
|
|
|
memset(data, 0, sizeof *data);
|
|
|
|
|
|
|
|
_this->driverdata = data;
|
|
|
|
|
Fixed bug 3328 - Race condition in Wayland_VideoInit
Robert Folland
When running this little test program with SDL2 on Wayland it often crashes in SDL_Init.
From a backtrace it is apparent that there is a race condition in creating a xkb_context_ref. Sometimes it is 0x0.
By moving the relevant lines higher up in Wayland_VideoInit (in SDL2-2.0.4/src/video/wayland/SDL_waylandvideo.c:302) this seems to get fixed.
I moved the call to WAYLAND_xkb_context_new() up to before the call to WAYLAND_wl_display_connect().
Here is the test program (just a loop of init and quit), and a backtrace from gdb:
#include <cstdio>
#include <stdlib.h>
#include <SDL2/SDL.h>
#include <unistd.h>
#include <iostream>
int main(int argc, char **argv)
{
int count = atoi(argv[1]);
for (int i = 0; i < count; i++) {
std::cout << "Init " << i << std::endl;
if (SDL_Init(SDL_INIT_VIDEO) < 0) {
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
"Couldn't initialize SDL: %s\n",
SDL_GetError());
return 1;
}
std::cout << "Quit" << std::endl;
SDL_Quit();
}
return 0;
}
Init 12
Quit
Init 13
Program received signal SIGSEGV, Segmentation fault.
xkb_context_ref (ctx=ctx@entry=0x0) at src/context.c:156
156 ctx->refcnt++;
(gdb) bt
#0 xkb_context_ref (ctx=ctx@entry=0x0) at src/context.c:156
#1 0x00007ffff5e1cd4c in xkb_keymap_new (ctx=0x0, format=XKB_KEYMAP_FORMAT_TEXT_V1, flags=flags@entry=XKB_KEYMAP_COMPILE_NO_FLAGS) at src/keymap-priv.c:65
#2 0x00007ffff5e1c6cc in xkb_keymap_new_from_buffer (ctx=<optimized out>,
buffer=0x7ffff7fd5000 "xkb_keymap {\nxkb_keycodes \"(unnamed)\" {\n\tminimum = 8;\n\tmaximum = 255;\n\t<ESC>", ' ' <repeats 16 times>, "= 9;\n\t<AE01>", ' ' <re
peats 15 times>, "= 10;\n\t<AE02>", ' ' <repeats 15 times>, "= 11;\n\t<AE03>", ' ' <repeats 15 times>, "= 12;\n\t<AE04>", ' ' <repeats 12 times>..., length=48090,
format=<optimized out>, flags=<optimized out>) at src/keymap.c:191
#3 0x00007ffff7b8ea4e in keyboard_handle_keymap (data=0x6169b0, keyboard=<optimized out>, format=<optimized out>, fd=5, size=48091)
at /home/vlab/abs/sdl2/src/SDL2-2.0.4/src/video/wayland/SDL_waylandevents.c:269
#4 0x00007ffff64501f0 in ffi_call_unix64 () from /usr/lib/libffi.so.6
#5 0x00007ffff644fc58 in ffi_call () from /usr/lib/libffi.so.6
#6 0x00007ffff665be3e in wl_closure_invoke (closure=closure@entry=0x61f000, flags=flags@entry=1, target=<optimized out>, target@entry=0x616d20,
opcode=opcode@entry=0, data=<optimized out>) at src/connection.c:949
#7 0x00007ffff6658be0 in dispatch_event (display=<optimized out>, queue=<optimized out>) at src/wayland-client.c:1274
#8 0x00007ffff6659db4 in dispatch_queue (queue=0x617398, display=0x6172d0) at src/wayland-client.c:1420
#9 wl_display_dispatch_queue_pending (display=0x6172d0, queue=0x617398) at src/wayland-client.c:1662
#10 0x00007ffff665a0cf in wl_display_roundtrip_queue (display=0x6172d0, queue=0x617398) at src/wayland-client.c:1085
#11 0x00007ffff7b8faa0 in Wayland_VideoInit (_this=<optimized out>) at /home/vlab/abs/sdl2/src/SDL2-2.0.4/src/video/wayland/SDL_waylandvideo.c:302
#12 0x00007ffff7b7aed6 in SDL_VideoInit_REAL (driver_name=<optimized out>, driver_name@entry=0x0) at /home/vlab/abs/sdl2/src/SDL2-2.0.4/src/video/SDL_video.c:513
#13 0x00007ffff7ae0ee7 in SDL_InitSubSystem_REAL (flags=16416) at /home/vlab/abs/sdl2/src/SDL2-2.0.4/src/SDL.c:173
#14 0x0000000000400b24 in main (argc=2, argv=0x7fffffffebb8) at vplay-init.cpp:13
(gdb)
2016-10-13 13:54:43 +02:00
|
|
|
data->xkb_context = WAYLAND_xkb_context_new(0);
|
|
|
|
if (!data->xkb_context) {
|
|
|
|
return SDL_SetError("Failed to create XKB context");
|
|
|
|
}
|
|
|
|
|
2015-06-21 17:33:46 +02:00
|
|
|
data->display = WAYLAND_wl_display_connect(NULL);
|
|
|
|
if (data->display == NULL) {
|
|
|
|
return SDL_SetError("Failed to connect to a Wayland display");
|
|
|
|
}
|
|
|
|
|
|
|
|
data->registry = wl_display_get_registry(data->display);
|
|
|
|
if (data->registry == NULL) {
|
|
|
|
return SDL_SetError("Failed to get the Wayland registry");
|
|
|
|
}
|
|
|
|
|
|
|
|
wl_registry_add_listener(data->registry, ®istry_listener, data);
|
|
|
|
|
|
|
|
// First roundtrip to receive all registry objects.
|
|
|
|
WAYLAND_wl_display_roundtrip(data->display);
|
|
|
|
|
|
|
|
// Second roundtrip to receive all output events.
|
|
|
|
WAYLAND_wl_display_roundtrip(data->display);
|
|
|
|
|
|
|
|
Wayland_InitMouse();
|
|
|
|
|
2016-09-01 10:22:58 +02:00
|
|
|
/* Get the surface class name, usually the name of the application */
|
|
|
|
data->classname = get_classname();
|
|
|
|
|
2015-06-21 17:33:46 +02:00
|
|
|
WAYLAND_wl_display_flush(data->display);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
Wayland_GetDisplayModes(_THIS, SDL_VideoDisplay *sdl_display)
|
|
|
|
{
|
|
|
|
// Nothing to do here, everything was already done in the wl_output
|
|
|
|
// callbacks.
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
Wayland_SetDisplayMode(_THIS, SDL_VideoDisplay *display, SDL_DisplayMode *mode)
|
|
|
|
{
|
|
|
|
return SDL_Unsupported();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Wayland_VideoQuit(_THIS)
|
|
|
|
{
|
|
|
|
SDL_VideoData *data = _this->driverdata;
|
2018-01-03 19:49:26 +01:00
|
|
|
int i, j;
|
2015-06-21 17:33:46 +02:00
|
|
|
|
|
|
|
Wayland_FiniMouse ();
|
|
|
|
|
|
|
|
for (i = 0; i < _this->num_displays; ++i) {
|
|
|
|
SDL_VideoDisplay *display = &_this->displays[i];
|
|
|
|
wl_output_destroy(display->driverdata);
|
|
|
|
display->driverdata = NULL;
|
2018-01-03 19:49:26 +01:00
|
|
|
|
|
|
|
for (j = display->num_display_modes; j--;) {
|
|
|
|
display->display_modes[j].driverdata = NULL;
|
|
|
|
}
|
|
|
|
display->desktop_mode.driverdata = NULL;
|
2015-06-21 17:33:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Wayland_display_destroy_input(data);
|
2016-09-01 10:26:56 +02:00
|
|
|
Wayland_display_destroy_pointer_constraints(data);
|
|
|
|
Wayland_display_destroy_relative_pointer_manager(data);
|
2015-06-21 17:33:46 +02:00
|
|
|
|
|
|
|
if (data->xkb_context) {
|
|
|
|
WAYLAND_xkb_context_unref(data->xkb_context);
|
|
|
|
data->xkb_context = NULL;
|
|
|
|
}
|
|
|
|
#ifdef SDL_VIDEO_DRIVER_WAYLAND_QT_TOUCH
|
|
|
|
if (data->windowmanager)
|
|
|
|
qt_windowmanager_destroy(data->windowmanager);
|
|
|
|
|
|
|
|
if (data->surface_extension)
|
|
|
|
qt_surface_extension_destroy(data->surface_extension);
|
|
|
|
|
|
|
|
Wayland_touch_destroy(data);
|
|
|
|
#endif /* SDL_VIDEO_DRIVER_WAYLAND_QT_TOUCH */
|
|
|
|
|
|
|
|
if (data->shm)
|
|
|
|
wl_shm_destroy(data->shm);
|
|
|
|
|
|
|
|
if (data->cursor_theme)
|
|
|
|
WAYLAND_wl_cursor_theme_destroy(data->cursor_theme);
|
|
|
|
|
2018-02-07 19:13:55 +01:00
|
|
|
if (data->shell.wl)
|
|
|
|
wl_shell_destroy(data->shell.wl);
|
|
|
|
|
|
|
|
if (data->shell.zxdg)
|
|
|
|
zxdg_shell_v6_destroy(data->shell.zxdg);
|
2015-06-21 17:33:46 +02:00
|
|
|
|
|
|
|
if (data->compositor)
|
|
|
|
wl_compositor_destroy(data->compositor);
|
|
|
|
|
|
|
|
if (data->registry)
|
|
|
|
wl_registry_destroy(data->registry);
|
|
|
|
|
|
|
|
if (data->display) {
|
|
|
|
WAYLAND_wl_display_flush(data->display);
|
|
|
|
WAYLAND_wl_display_disconnect(data->display);
|
|
|
|
}
|
|
|
|
|
2016-09-01 10:22:58 +02:00
|
|
|
SDL_free(data->classname);
|
2017-05-25 23:00:58 +02:00
|
|
|
SDL_free(data);
|
2015-06-21 17:33:46 +02:00
|
|
|
_this->driverdata = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* SDL_VIDEO_DRIVER_WAYLAND */
|
|
|
|
|
|
|
|
/* vi: set ts=4 sw=4 expandtab: */
|