mirror of
git://source.winehq.org/git/wine.git
synced 2024-11-01 08:13:18 +00:00
de66ea9df6
If the compositor doesn't provide the logical dimensions (or sane ones), use the physical pixel dimensions as the logical dimensions.
392 lines
13 KiB
C
392 lines
13 KiB
C
/*
|
|
* Wayland output handling
|
|
*
|
|
* Copyright (c) 2020 Alexandros Frantzis for Collabora Ltd
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#if 0
|
|
#pragma makedep unix
|
|
#endif
|
|
|
|
#include "config.h"
|
|
|
|
#include "waylanddrv.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(waylanddrv);
|
|
|
|
static const int32_t default_refresh = 60000;
|
|
static uint32_t next_output_id = 0;
|
|
|
|
#define WAYLAND_OUTPUT_CHANGED_MODES 0x01
|
|
#define WAYLAND_OUTPUT_CHANGED_NAME 0x02
|
|
#define WAYLAND_OUTPUT_CHANGED_LOGICAL_XY 0x04
|
|
#define WAYLAND_OUTPUT_CHANGED_LOGICAL_WH 0x08
|
|
|
|
/**********************************************************************
|
|
* Output handling
|
|
*/
|
|
|
|
/* Compare a mode rb_tree key with the provided mode rb_entry and return -1 if
|
|
* the key compares less than the entry, 0 if the key compares equal to the
|
|
* entry, and 1 if the key compares greater than the entry.
|
|
*
|
|
* The comparison is based on comparing the width, height and refresh in that
|
|
* order. */
|
|
static int wayland_output_mode_cmp_rb(const void *key,
|
|
const struct rb_entry *entry)
|
|
{
|
|
const struct wayland_output_mode *key_mode = key;
|
|
const struct wayland_output_mode *entry_mode =
|
|
RB_ENTRY_VALUE(entry, const struct wayland_output_mode, entry);
|
|
|
|
if (key_mode->width < entry_mode->width) return -1;
|
|
if (key_mode->width > entry_mode->width) return 1;
|
|
if (key_mode->height < entry_mode->height) return -1;
|
|
if (key_mode->height > entry_mode->height) return 1;
|
|
if (key_mode->refresh < entry_mode->refresh) return -1;
|
|
if (key_mode->refresh > entry_mode->refresh) return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void wayland_output_state_add_mode(struct wayland_output_state *state,
|
|
int32_t width, int32_t height,
|
|
int32_t refresh, BOOL current)
|
|
{
|
|
struct rb_entry *mode_entry;
|
|
struct wayland_output_mode *mode;
|
|
struct wayland_output_mode key =
|
|
{
|
|
.width = width,
|
|
.height = height,
|
|
.refresh = refresh,
|
|
};
|
|
|
|
mode_entry = rb_get(&state->modes, &key);
|
|
if (mode_entry)
|
|
{
|
|
mode = RB_ENTRY_VALUE(mode_entry, struct wayland_output_mode, entry);
|
|
}
|
|
else
|
|
{
|
|
mode = calloc(1, sizeof(*mode));
|
|
if (!mode)
|
|
{
|
|
ERR("Failed to allocate space for wayland_output_mode\n");
|
|
return;
|
|
}
|
|
mode->width = width;
|
|
mode->height = height;
|
|
mode->refresh = refresh;
|
|
rb_put(&state->modes, mode, &mode->entry);
|
|
}
|
|
|
|
if (current) state->current_mode = mode;
|
|
}
|
|
|
|
static void maybe_init_display_devices(void)
|
|
{
|
|
DWORD desktop_pid = 0;
|
|
HWND desktop_hwnd;
|
|
|
|
/* Right after process init we initialize all the display devices, so there
|
|
* is no need to react to each individual event at that time. This check
|
|
* also helps us avoid calling NtUserGetDesktopWindow() (see below) at
|
|
* process init time, since it may not be safe. */
|
|
if (!process_wayland.initialized) return;
|
|
|
|
desktop_hwnd = NtUserGetDesktopWindow();
|
|
NtUserGetWindowThread(desktop_hwnd, &desktop_pid);
|
|
|
|
/* We only update the display devices from the desktop process. */
|
|
if (GetCurrentProcessId() != desktop_pid) return;
|
|
|
|
NtUserPostMessage(desktop_hwnd, WM_WAYLAND_INIT_DISPLAY_DEVICES, 0, 0);
|
|
}
|
|
|
|
static void wayland_output_mode_free_rb(struct rb_entry *entry, void *ctx)
|
|
{
|
|
free(RB_ENTRY_VALUE(entry, struct wayland_output_mode, entry));
|
|
}
|
|
|
|
static void wayland_output_done(struct wayland_output *output)
|
|
{
|
|
struct wayland_output_mode *mode;
|
|
|
|
/* Update current state from pending state. */
|
|
pthread_mutex_lock(&process_wayland.output_mutex);
|
|
|
|
if (output->pending_flags & WAYLAND_OUTPUT_CHANGED_MODES)
|
|
{
|
|
RB_FOR_EACH_ENTRY(mode, &output->pending.modes, struct wayland_output_mode, entry)
|
|
{
|
|
wayland_output_state_add_mode(&output->current,
|
|
mode->width, mode->height, mode->refresh,
|
|
mode == output->pending.current_mode);
|
|
}
|
|
rb_destroy(&output->pending.modes, wayland_output_mode_free_rb, NULL);
|
|
rb_init(&output->pending.modes, wayland_output_mode_cmp_rb);
|
|
}
|
|
|
|
if (output->pending_flags & WAYLAND_OUTPUT_CHANGED_NAME)
|
|
{
|
|
free(output->current.name);
|
|
output->current.name = output->pending.name;
|
|
output->pending.name = NULL;
|
|
}
|
|
|
|
if (output->pending_flags & WAYLAND_OUTPUT_CHANGED_LOGICAL_XY)
|
|
{
|
|
output->current.logical_x = output->pending.logical_x;
|
|
output->current.logical_y = output->pending.logical_y;
|
|
}
|
|
|
|
if (output->pending_flags & WAYLAND_OUTPUT_CHANGED_LOGICAL_WH)
|
|
{
|
|
output->current.logical_w = output->pending.logical_w;
|
|
output->current.logical_h = output->pending.logical_h;
|
|
}
|
|
|
|
output->pending_flags = 0;
|
|
|
|
/* Ensure the logical dimensions have sane values. */
|
|
if ((!output->current.logical_w || !output->current.logical_h) &&
|
|
output->current.current_mode)
|
|
{
|
|
output->current.logical_w = output->current.current_mode->width;
|
|
output->current.logical_h = output->current.current_mode->height;
|
|
}
|
|
|
|
pthread_mutex_unlock(&process_wayland.output_mutex);
|
|
|
|
TRACE("name=%s logical=%d,%d+%dx%d\n",
|
|
output->current.name, output->current.logical_x, output->current.logical_y,
|
|
output->current.logical_w, output->current.logical_h);
|
|
|
|
RB_FOR_EACH_ENTRY(mode, &output->current.modes, struct wayland_output_mode, entry)
|
|
{
|
|
TRACE("mode %dx%d @ %d %s\n",
|
|
mode->width, mode->height, mode->refresh,
|
|
output->current.current_mode == mode ? "*" : "");
|
|
}
|
|
|
|
maybe_init_display_devices();
|
|
}
|
|
|
|
static void output_handle_geometry(void *data, struct wl_output *wl_output,
|
|
int32_t x, int32_t y,
|
|
int32_t physical_width, int32_t physical_height,
|
|
int32_t subpixel,
|
|
const char *make, const char *model,
|
|
int32_t output_transform)
|
|
{
|
|
}
|
|
|
|
static void output_handle_mode(void *data, struct wl_output *wl_output,
|
|
uint32_t flags, int32_t width, int32_t height,
|
|
int32_t refresh)
|
|
{
|
|
struct wayland_output *output = data;
|
|
|
|
/* Windows apps don't expect a zero refresh rate, so use a default value. */
|
|
if (refresh == 0) refresh = default_refresh;
|
|
|
|
wayland_output_state_add_mode(&output->pending, width, height, refresh,
|
|
(flags & WL_OUTPUT_MODE_CURRENT));
|
|
|
|
output->pending_flags |= WAYLAND_OUTPUT_CHANGED_MODES;
|
|
}
|
|
|
|
static void output_handle_done(void *data, struct wl_output *wl_output)
|
|
{
|
|
struct wayland_output *output = data;
|
|
|
|
if (!output->zxdg_output_v1 ||
|
|
zxdg_output_v1_get_version(output->zxdg_output_v1) >= 3)
|
|
{
|
|
wayland_output_done(output);
|
|
}
|
|
}
|
|
|
|
static void output_handle_scale(void *data, struct wl_output *wl_output,
|
|
int32_t scale)
|
|
{
|
|
}
|
|
|
|
static const struct wl_output_listener output_listener = {
|
|
output_handle_geometry,
|
|
output_handle_mode,
|
|
output_handle_done,
|
|
output_handle_scale
|
|
};
|
|
|
|
static void zxdg_output_v1_handle_logical_position(void *data,
|
|
struct zxdg_output_v1 *zxdg_output_v1,
|
|
int32_t x,
|
|
int32_t y)
|
|
{
|
|
struct wayland_output *output = data;
|
|
TRACE("logical_x=%d logical_y=%d\n", x, y);
|
|
output->pending.logical_x = x;
|
|
output->pending.logical_y = y;
|
|
output->pending_flags |= WAYLAND_OUTPUT_CHANGED_LOGICAL_XY;
|
|
}
|
|
|
|
static void zxdg_output_v1_handle_logical_size(void *data,
|
|
struct zxdg_output_v1 *zxdg_output_v1,
|
|
int32_t width,
|
|
int32_t height)
|
|
{
|
|
struct wayland_output *output = data;
|
|
TRACE("logical_w=%d logical_h=%d\n", width, height);
|
|
output->pending.logical_w = width;
|
|
output->pending.logical_h = height;
|
|
output->pending_flags |= WAYLAND_OUTPUT_CHANGED_LOGICAL_WH;
|
|
}
|
|
|
|
static void zxdg_output_v1_handle_done(void *data,
|
|
struct zxdg_output_v1 *zxdg_output_v1)
|
|
{
|
|
if (zxdg_output_v1_get_version(zxdg_output_v1) < 3)
|
|
{
|
|
struct wayland_output *output = data;
|
|
wayland_output_done(output);
|
|
}
|
|
}
|
|
|
|
static void zxdg_output_v1_handle_name(void *data,
|
|
struct zxdg_output_v1 *zxdg_output_v1,
|
|
const char *name)
|
|
{
|
|
struct wayland_output *output = data;
|
|
|
|
free(output->pending.name);
|
|
output->pending.name = strdup(name);
|
|
output->pending_flags |= WAYLAND_OUTPUT_CHANGED_NAME;
|
|
}
|
|
|
|
static void zxdg_output_v1_handle_description(void *data,
|
|
struct zxdg_output_v1 *zxdg_output_v1,
|
|
const char *description)
|
|
{
|
|
}
|
|
|
|
static const struct zxdg_output_v1_listener zxdg_output_v1_listener = {
|
|
zxdg_output_v1_handle_logical_position,
|
|
zxdg_output_v1_handle_logical_size,
|
|
zxdg_output_v1_handle_done,
|
|
zxdg_output_v1_handle_name,
|
|
zxdg_output_v1_handle_description,
|
|
};
|
|
|
|
/**********************************************************************
|
|
* wayland_output_create
|
|
*
|
|
* Creates a wayland_output and adds it to the output list.
|
|
*/
|
|
BOOL wayland_output_create(uint32_t id, uint32_t version)
|
|
{
|
|
struct wayland_output *output = calloc(1, sizeof(*output));
|
|
int name_len;
|
|
|
|
if (!output)
|
|
{
|
|
ERR("Failed to allocate space for wayland_output\n");
|
|
goto err;
|
|
}
|
|
|
|
output->wl_output = wl_registry_bind(process_wayland.wl_registry, id,
|
|
&wl_output_interface,
|
|
version < 2 ? version : 2);
|
|
output->global_id = id;
|
|
wl_output_add_listener(output->wl_output, &output_listener, output);
|
|
|
|
wl_list_init(&output->link);
|
|
rb_init(&output->pending.modes, wayland_output_mode_cmp_rb);
|
|
rb_init(&output->current.modes, wayland_output_mode_cmp_rb);
|
|
|
|
/* Have a fallback while we don't have compositor given name. */
|
|
name_len = snprintf(NULL, 0, "WaylandOutput%d", next_output_id);
|
|
output->current.name = malloc(name_len + 1);
|
|
if (output->current.name)
|
|
{
|
|
snprintf(output->current.name, name_len + 1, "WaylandOutput%d", next_output_id++);
|
|
}
|
|
else
|
|
{
|
|
ERR("Couldn't allocate space for output name\n");
|
|
goto err;
|
|
}
|
|
|
|
if (process_wayland.zxdg_output_manager_v1)
|
|
wayland_output_use_xdg_extension(output);
|
|
|
|
pthread_mutex_lock(&process_wayland.output_mutex);
|
|
wl_list_insert(process_wayland.output_list.prev, &output->link);
|
|
pthread_mutex_unlock(&process_wayland.output_mutex);
|
|
|
|
return TRUE;
|
|
|
|
err:
|
|
if (output) wayland_output_destroy(output);
|
|
return FALSE;
|
|
}
|
|
|
|
static void wayland_output_state_deinit(struct wayland_output_state *state)
|
|
{
|
|
rb_destroy(&state->modes, wayland_output_mode_free_rb, NULL);
|
|
free(state->name);
|
|
}
|
|
|
|
/**********************************************************************
|
|
* wayland_output_destroy
|
|
*
|
|
* Destroys a wayland_output.
|
|
*/
|
|
void wayland_output_destroy(struct wayland_output *output)
|
|
{
|
|
pthread_mutex_lock(&process_wayland.output_mutex);
|
|
wl_list_remove(&output->link);
|
|
pthread_mutex_unlock(&process_wayland.output_mutex);
|
|
|
|
wayland_output_state_deinit(&output->pending);
|
|
wayland_output_state_deinit(&output->current);
|
|
if (output->zxdg_output_v1)
|
|
zxdg_output_v1_destroy(output->zxdg_output_v1);
|
|
wl_output_destroy(output->wl_output);
|
|
free(output);
|
|
|
|
maybe_init_display_devices();
|
|
}
|
|
|
|
/**********************************************************************
|
|
* wayland_output_use_xdg_extension
|
|
*
|
|
* Use the zxdg_output_v1 extension to get output information.
|
|
*/
|
|
void wayland_output_use_xdg_extension(struct wayland_output *output)
|
|
{
|
|
output->zxdg_output_v1 =
|
|
zxdg_output_manager_v1_get_xdg_output(process_wayland.zxdg_output_manager_v1,
|
|
output->wl_output);
|
|
zxdg_output_v1_add_listener(output->zxdg_output_v1, &zxdg_output_v1_listener,
|
|
output);
|
|
}
|