1461 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			1461 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			C
		
	
	
	
/*
 | 
						|
 * Copyright © 2011-2014 Intel Corporation
 | 
						|
 *
 | 
						|
 * Permission to use, copy, modify, distribute, and sell this software
 | 
						|
 * and its documentation for any purpose is hereby granted without
 | 
						|
 * fee, provided that the above copyright notice appear in all copies
 | 
						|
 * and that both that copyright notice and this permission notice
 | 
						|
 * appear in supporting documentation, and that the name of the
 | 
						|
 * copyright holders not be used in advertising or publicity
 | 
						|
 * pertaining to distribution of the software without specific,
 | 
						|
 * written prior permission.  The copyright holders make no
 | 
						|
 * representations about the suitability of this software for any
 | 
						|
 * purpose.  It is provided "as is" without express or implied
 | 
						|
 * warranty.
 | 
						|
 *
 | 
						|
 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
 | 
						|
 * SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
 | 
						|
 * FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
 | 
						|
 * SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | 
						|
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN
 | 
						|
 * AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
 | 
						|
 * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
 | 
						|
 * SOFTWARE.
 | 
						|
 */
 | 
						|
 | 
						|
#ifdef HAVE_DIX_CONFIG_H
 | 
						|
#include <dix-config.h>
 | 
						|
#endif
 | 
						|
 | 
						|
#include <sys/mman.h>
 | 
						|
 | 
						|
#include <X11/X.h>
 | 
						|
#include <X11/Xatom.h>
 | 
						|
 | 
						|
#include "compositeext.h"
 | 
						|
#include "compint.h"
 | 
						|
#include "inputstr.h"
 | 
						|
#include "propertyst.h"
 | 
						|
 | 
						|
#include "xwayland-types.h"
 | 
						|
#include "xwayland-input.h"
 | 
						|
#include "xwayland-present.h"
 | 
						|
#include "xwayland-screen.h"
 | 
						|
#include "xwayland-window.h"
 | 
						|
#include "xwayland-window-buffers.h"
 | 
						|
#include "xwayland-shm.h"
 | 
						|
 | 
						|
#include "linux-dmabuf-unstable-v1-client-protocol.h"
 | 
						|
#include "tearing-control-v1-client-protocol.h"
 | 
						|
#include "viewporter-client-protocol.h"
 | 
						|
#include "xdg-shell-client-protocol.h"
 | 
						|
#include "xwayland-shell-v1-client-protocol.h"
 | 
						|
 | 
						|
#define DELAYED_WL_SURFACE_DESTROY 1000 /* ms */
 | 
						|
 | 
						|
#define MAX_ROOTFUL_WIDTH 32767
 | 
						|
#define MAX_ROOTFUL_HEIGHT 32767
 | 
						|
#define MIN_ROOTFUL_WIDTH 320
 | 
						|
#define MIN_ROOTFUL_HEIGHT 200
 | 
						|
 | 
						|
static DevPrivateKeyRec xwl_window_private_key;
 | 
						|
static DevPrivateKeyRec xwl_damage_private_key;
 | 
						|
static const char *xwl_surface_tag = "xwl-surface";
 | 
						|
 | 
						|
static Bool xwl_window_attach_buffer(struct xwl_window *);
 | 
						|
 | 
						|
struct xwl_window *
 | 
						|
xwl_window_get(WindowPtr window)
 | 
						|
{
 | 
						|
    return dixLookupPrivate(&window->devPrivates, &xwl_window_private_key);
 | 
						|
}
 | 
						|
 | 
						|
static DamagePtr
 | 
						|
window_get_damage(WindowPtr window)
 | 
						|
{
 | 
						|
    return dixLookupPrivate(&window->devPrivates, &xwl_damage_private_key);
 | 
						|
}
 | 
						|
 | 
						|
struct xwl_window *
 | 
						|
xwl_window_from_window(WindowPtr window)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
 | 
						|
    while (window) {
 | 
						|
        xwl_window = xwl_window_get(window);
 | 
						|
        if (xwl_window)
 | 
						|
            return xwl_window;
 | 
						|
 | 
						|
        window = window->parent;
 | 
						|
    }
 | 
						|
 | 
						|
    return NULL;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_set_xwayland_tag(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    wl_proxy_set_tag((struct wl_proxy *)xwl_window->surface, &xwl_surface_tag);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_clear_xwayland_tag(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    wl_proxy_set_tag((struct wl_proxy *)xwl_window->surface, NULL);
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
is_surface_from_xwl_window(struct wl_surface *surface)
 | 
						|
{
 | 
						|
    return wl_proxy_get_tag((struct wl_proxy *) surface) == &xwl_surface_tag;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_set_allow_commits(struct xwl_window *xwl_window, Bool allow,
 | 
						|
                             const char *debug_msg)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    DamagePtr damage;
 | 
						|
 | 
						|
    xwl_window->allow_commits = allow;
 | 
						|
    DebugF("XWAYLAND: win %d allow_commits = %d (%s)\n",
 | 
						|
           xwl_window->window->drawable.id, allow, debug_msg);
 | 
						|
 | 
						|
    damage = window_get_damage(xwl_window->window);
 | 
						|
    if (allow &&
 | 
						|
        xorg_list_is_empty(&xwl_window->link_damage) &&
 | 
						|
        damage &&
 | 
						|
        RegionNotEmpty(DamageRegion(damage))) {
 | 
						|
        xorg_list_add(&xwl_window->link_damage,
 | 
						|
                      &xwl_screen->damage_window_list);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_set_allow_commits_from_property(struct xwl_window *xwl_window,
 | 
						|
                                           PropertyPtr prop)
 | 
						|
{
 | 
						|
    static Bool warned = FALSE;
 | 
						|
    CARD32 *propdata;
 | 
						|
 | 
						|
    if (prop->propertyName != xwl_window->xwl_screen->allow_commits_prop)
 | 
						|
        FatalError("Xwayland internal error: prop mismatch in %s.\n", __func__);
 | 
						|
 | 
						|
    if (prop->type != XA_CARDINAL || prop->format != 32 || prop->size != 1) {
 | 
						|
        /* Not properly set, so fall back to safe and glitchy */
 | 
						|
        xwl_window_set_allow_commits(xwl_window, TRUE, "WM fault");
 | 
						|
 | 
						|
        if (!warned) {
 | 
						|
            LogMessageVerb(X_WARNING, 0, "Window manager is misusing property %s.\n",
 | 
						|
                           NameForAtom(prop->propertyName));
 | 
						|
            warned = TRUE;
 | 
						|
        }
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    propdata = prop->data;
 | 
						|
    xwl_window_set_allow_commits(xwl_window, !!propdata[0], "from property");
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_update_property(struct xwl_window *xwl_window,
 | 
						|
                           PropertyStateRec *propstate)
 | 
						|
{
 | 
						|
    switch (propstate->state) {
 | 
						|
    case PropertyNewValue:
 | 
						|
        xwl_window_set_allow_commits_from_property(xwl_window, propstate->prop);
 | 
						|
        break;
 | 
						|
 | 
						|
    case PropertyDelete:
 | 
						|
        xwl_window_set_allow_commits(xwl_window, TRUE, "property deleted");
 | 
						|
        break;
 | 
						|
 | 
						|
    default:
 | 
						|
        break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
damage_report(DamagePtr pDamage, RegionPtr pRegion, void *data)
 | 
						|
{
 | 
						|
    WindowPtr window = data;
 | 
						|
    struct xwl_window *xwl_window = xwl_window_get(window);
 | 
						|
    struct xwl_screen *xwl_screen;
 | 
						|
 | 
						|
    if (!xwl_window)
 | 
						|
        return;
 | 
						|
 | 
						|
    xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    if (xwl_window->present_flipped) {
 | 
						|
        /* This damage is from a Present flip, which already committed a new
 | 
						|
         * buffer for the surface, so we don't need to do anything in response
 | 
						|
         */
 | 
						|
        RegionEmpty(DamageRegion(pDamage));
 | 
						|
        xorg_list_del(&xwl_window->link_damage);
 | 
						|
        xwl_window->present_flipped = FALSE;
 | 
						|
        return;
 | 
						|
    }
 | 
						|
#endif
 | 
						|
 | 
						|
    if (xorg_list_is_empty(&xwl_window->link_damage))
 | 
						|
        xorg_list_add(&xwl_window->link_damage, &xwl_screen->damage_window_list);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
damage_destroy(DamagePtr pDamage, void *data)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
register_damage(WindowPtr window)
 | 
						|
{
 | 
						|
    DamagePtr damage;
 | 
						|
 | 
						|
    damage = DamageCreate(damage_report, damage_destroy, DamageReportNonEmpty,
 | 
						|
                          FALSE, window->drawable.pScreen, window);
 | 
						|
    if (damage == NULL) {
 | 
						|
        ErrorF("Failed creating damage\n");
 | 
						|
        return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
    DamageRegister(&window->drawable, damage);
 | 
						|
    DamageSetReportAfterOp(damage, TRUE);
 | 
						|
 | 
						|
    dixSetPrivate(&window->devPrivates, &xwl_damage_private_key, damage);
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
unregister_damage(WindowPtr window)
 | 
						|
{
 | 
						|
    DamagePtr damage;
 | 
						|
 | 
						|
    damage = dixLookupPrivate(&window->devPrivates, &xwl_damage_private_key);
 | 
						|
    if (!damage)
 | 
						|
        return;
 | 
						|
 | 
						|
    DamageUnregister(damage);
 | 
						|
    DamageDestroy(damage);
 | 
						|
 | 
						|
    dixSetPrivate(&window->devPrivates, &xwl_damage_private_key, NULL);
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_window_has_viewport_enabled(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    return (xwl_window->viewport != NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_disable_viewport(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    assert (xwl_window->viewport);
 | 
						|
 | 
						|
    DebugF("XWAYLAND: disabling viewport\n");
 | 
						|
    wp_viewport_destroy(xwl_window->viewport);
 | 
						|
    xwl_window->viewport = NULL;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_enable_viewport(struct xwl_window *xwl_window,
 | 
						|
                           struct xwl_output *xwl_output,
 | 
						|
                           struct xwl_emulated_mode *emulated_mode)
 | 
						|
{
 | 
						|
    if (!xwl_window_has_viewport_enabled(xwl_window)) {
 | 
						|
        DebugF("XWAYLAND: enabling viewport %dx%d -> %dx%d\n",
 | 
						|
               emulated_mode->width, emulated_mode->height,
 | 
						|
               xwl_output->width, xwl_output->height);
 | 
						|
        xwl_window->viewport = wp_viewporter_get_viewport(xwl_window->xwl_screen->viewporter,
 | 
						|
                                                          xwl_window->surface);
 | 
						|
    }
 | 
						|
 | 
						|
    wp_viewport_set_source(xwl_window->viewport,
 | 
						|
                           wl_fixed_from_int(0),
 | 
						|
                           wl_fixed_from_int(0),
 | 
						|
                           wl_fixed_from_int(emulated_mode->width),
 | 
						|
                           wl_fixed_from_int(emulated_mode->height));
 | 
						|
    wp_viewport_set_destination(xwl_window->viewport,
 | 
						|
                                xwl_output->width,
 | 
						|
                                xwl_output->height);
 | 
						|
 | 
						|
    xwl_window->scale_x = (float)emulated_mode->width  / xwl_output->width;
 | 
						|
    xwl_window->scale_y = (float)emulated_mode->height / xwl_output->height;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
window_is_wm_window(WindowPtr window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(window->drawable.pScreen);
 | 
						|
 | 
						|
    return CLIENT_ID(window->drawable.id) == xwl_screen->wm_client_id;
 | 
						|
}
 | 
						|
 | 
						|
static WindowPtr
 | 
						|
window_get_client_toplevel(WindowPtr window)
 | 
						|
{
 | 
						|
    assert(window);
 | 
						|
 | 
						|
    /* If the toplevel window is owned by the window-manager, then the
 | 
						|
     * actual client toplevel window has been reparented to some window-manager
 | 
						|
     * decoration/wrapper windows. In that case recurse by checking the client
 | 
						|
     * of the first *and only* child of the decoration/wrapper window.
 | 
						|
     */
 | 
						|
    while (window_is_wm_window(window)) {
 | 
						|
        if (!window->firstChild || window->firstChild != window->lastChild)
 | 
						|
            return NULL; /* Should never happen, skip resolution emulation */
 | 
						|
 | 
						|
        window = window->firstChild;
 | 
						|
    }
 | 
						|
 | 
						|
    return window;
 | 
						|
}
 | 
						|
 | 
						|
static struct xwl_output *
 | 
						|
xwl_window_get_output(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    struct xwl_output *xwl_output;
 | 
						|
 | 
						|
    xwl_output = xwl_output_from_wl_output(xwl_screen, xwl_window->wl_output);
 | 
						|
 | 
						|
    if (xwl_output)
 | 
						|
        return xwl_output;
 | 
						|
 | 
						|
    return xwl_screen_get_first_output(xwl_screen);
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_window_should_enable_viewport_fullscreen(struct xwl_window *xwl_window,
 | 
						|
                                             struct xwl_output **xwl_output_ret,
 | 
						|
                                             struct xwl_emulated_mode *emulated_mode_ret)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    struct xwl_output *xwl_output;
 | 
						|
 | 
						|
    xwl_output = xwl_window_get_output(xwl_window);
 | 
						|
    if (!xwl_output)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    *xwl_output_ret = xwl_output;
 | 
						|
    emulated_mode_ret->server_output_id = 0;
 | 
						|
    emulated_mode_ret->width = xwl_screen->width;
 | 
						|
    emulated_mode_ret->height = xwl_screen->height;
 | 
						|
    emulated_mode_ret->from_vidmode = FALSE;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_window_should_enable_viewport(struct xwl_window *xwl_window,
 | 
						|
                                  struct xwl_output **xwl_output_ret,
 | 
						|
                                  struct xwl_emulated_mode *emulated_mode_ret)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    struct xwl_emulated_mode *emulated_mode;
 | 
						|
    struct xwl_output *xwl_output;
 | 
						|
    ClientPtr owner;
 | 
						|
    WindowPtr window;
 | 
						|
    DrawablePtr drawable;
 | 
						|
 | 
						|
    if (!xwl_screen_has_viewport_support(xwl_screen))
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    if (xwl_screen->fullscreen)
 | 
						|
        return xwl_window_should_enable_viewport_fullscreen(xwl_window,
 | 
						|
                                                            xwl_output_ret,
 | 
						|
                                                            emulated_mode_ret);
 | 
						|
 | 
						|
    if (!xwl_screen->rootless)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    window = window_get_client_toplevel(xwl_window->window);
 | 
						|
    if (!window)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    owner = wClient(window);
 | 
						|
    drawable = &window->drawable;
 | 
						|
 | 
						|
    /* 1. Test if the window matches the emulated mode on one of the outputs
 | 
						|
     * This path gets hit by most games / libs (e.g. SDL, SFML, OGRE)
 | 
						|
     */
 | 
						|
    xorg_list_for_each_entry(xwl_output, &xwl_screen->output_list, link) {
 | 
						|
        emulated_mode = xwl_output_get_emulated_mode_for_client(xwl_output, owner);
 | 
						|
        if (!emulated_mode)
 | 
						|
            continue;
 | 
						|
 | 
						|
        if (drawable->x == xwl_output->x &&
 | 
						|
            drawable->y == xwl_output->y &&
 | 
						|
            drawable->width  == emulated_mode->width &&
 | 
						|
            drawable->height == emulated_mode->height) {
 | 
						|
 | 
						|
            memcpy(emulated_mode_ret, emulated_mode, sizeof(struct xwl_emulated_mode));
 | 
						|
            *xwl_output_ret = xwl_output;
 | 
						|
            return TRUE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /* 2. Test if the window uses override-redirect + vidmode
 | 
						|
     * and matches (fully covers) the entire screen.
 | 
						|
     * This path gets hit by: allegro4, ClanLib-1.0.
 | 
						|
     */
 | 
						|
    xwl_output = xwl_screen_get_first_output(xwl_screen);
 | 
						|
    emulated_mode = xwl_output_get_emulated_mode_for_client(xwl_output, owner);
 | 
						|
    if (xwl_output && xwl_window->window->overrideRedirect &&
 | 
						|
        emulated_mode && emulated_mode->from_vidmode &&
 | 
						|
        drawable->x == 0 && drawable->y == 0 &&
 | 
						|
        drawable->width  == xwl_screen->width &&
 | 
						|
        drawable->height == xwl_screen->height) {
 | 
						|
 | 
						|
        memcpy(emulated_mode_ret, emulated_mode, sizeof(struct xwl_emulated_mode));
 | 
						|
        *xwl_output_ret = xwl_output;
 | 
						|
        return TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_check_resolution_change_emulation(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_emulated_mode emulated_mode;
 | 
						|
    struct xwl_output *xwl_output;
 | 
						|
 | 
						|
    if (xwl_window_should_enable_viewport(xwl_window, &xwl_output, &emulated_mode))
 | 
						|
        xwl_window_enable_viewport(xwl_window, xwl_output, &emulated_mode);
 | 
						|
    else if (xwl_window_has_viewport_enabled(xwl_window))
 | 
						|
        xwl_window_disable_viewport(xwl_window);
 | 
						|
}
 | 
						|
 | 
						|
/* This checks if the passed in Window is a toplevel client window, note this
 | 
						|
 * returns false for window-manager decoration windows and returns true for
 | 
						|
 * the actual client top-level window even if it has been reparented to
 | 
						|
 * a window-manager decoration window.
 | 
						|
 */
 | 
						|
Bool
 | 
						|
xwl_window_is_toplevel(WindowPtr window)
 | 
						|
{
 | 
						|
    if (window_is_wm_window(window))
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    /* CSD and override-redirect toplevel windows */
 | 
						|
    if (window_get_damage(window))
 | 
						|
        return TRUE;
 | 
						|
 | 
						|
    /* Normal toplevel client windows, reparented to a window-manager window */
 | 
						|
    return window->parent && window_is_wm_window(window->parent);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_init_allow_commits(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    PropertyPtr prop = NULL;
 | 
						|
    int ret;
 | 
						|
 | 
						|
    ret = dixLookupProperty(&prop, xwl_window->window,
 | 
						|
                            xwl_window->xwl_screen->allow_commits_prop,
 | 
						|
                            serverClient, DixReadAccess);
 | 
						|
    if (ret == Success && prop)
 | 
						|
        xwl_window_set_allow_commits_from_property(xwl_window, prop);
 | 
						|
    else
 | 
						|
        xwl_window_set_allow_commits(xwl_window, TRUE, "no property");
 | 
						|
}
 | 
						|
 | 
						|
static uint32_t
 | 
						|
serial_lo(uint64_t value)
 | 
						|
{
 | 
						|
    return value & 0xFFFFFFFFu;
 | 
						|
}
 | 
						|
 | 
						|
static uint32_t
 | 
						|
serial_hi(uint64_t value)
 | 
						|
{
 | 
						|
    return value >> 32u;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
send_window_client_message(struct xwl_window *xwl_window, Atom type_atom, uint64_t value)
 | 
						|
{
 | 
						|
    DeviceIntPtr dev;
 | 
						|
    xEvent e;
 | 
						|
 | 
						|
    e.u.u.type = ClientMessage;
 | 
						|
    e.u.u.detail = 32;
 | 
						|
    e.u.clientMessage.window = xwl_window->window->drawable.id;
 | 
						|
    e.u.clientMessage.u.l.type = type_atom;
 | 
						|
    e.u.clientMessage.u.l.longs0 = serial_lo(value);
 | 
						|
    e.u.clientMessage.u.l.longs1 = serial_hi(value);
 | 
						|
    e.u.clientMessage.u.l.longs2 = 0;
 | 
						|
    e.u.clientMessage.u.l.longs3 = 0;
 | 
						|
    e.u.clientMessage.u.l.longs4 = 0;
 | 
						|
 | 
						|
    dev = PickPointer(serverClient);
 | 
						|
    DeliverEventsToWindow(dev, xwl_window->xwl_screen->screen->root,
 | 
						|
                          &e, 1, SubstructureRedirectMask, NullGrab);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
send_surface_id_event_serial(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    static const char atom_name[] = "WL_SURFACE_SERIAL";
 | 
						|
    static Atom type_atom;
 | 
						|
    uint64_t serial;
 | 
						|
 | 
						|
    if (type_atom == None)
 | 
						|
        type_atom = MakeAtom(atom_name, strlen(atom_name), TRUE);
 | 
						|
 | 
						|
    serial = ++xwl_window->xwl_screen->surface_association_serial;
 | 
						|
 | 
						|
    send_window_client_message(xwl_window, type_atom, serial);
 | 
						|
    xwayland_surface_v1_set_serial(xwl_window->xwayland_surface,
 | 
						|
        serial_lo(serial), serial_hi(serial));
 | 
						|
    wl_surface_commit(xwl_window->surface);
 | 
						|
 | 
						|
    /* Flush wayland display *after* commit in the new path. */
 | 
						|
    wl_display_flush(xwl_window->xwl_screen->display);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
send_surface_id_event_legacy(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    static const char atom_name[] = "WL_SURFACE_ID";
 | 
						|
    static Atom type_atom;
 | 
						|
    uint32_t surface_id;
 | 
						|
 | 
						|
    if (type_atom == None)
 | 
						|
        type_atom = MakeAtom(atom_name, strlen(atom_name), TRUE);
 | 
						|
 | 
						|
    surface_id = wl_proxy_get_id((struct wl_proxy *) xwl_window->surface);
 | 
						|
 | 
						|
    /* Flush wayland display *before* setting the atom in the legacy path */
 | 
						|
    wl_display_flush(xwl_window->xwl_screen->display);
 | 
						|
 | 
						|
    send_window_client_message(xwl_window, type_atom, (uint64_t)surface_id);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
send_surface_id_event(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    return xwl_window->xwayland_surface
 | 
						|
        ? send_surface_id_event_serial(xwl_window)
 | 
						|
        : send_surface_id_event_legacy(xwl_window);
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_window_set_fullscreen(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_output *xwl_output;
 | 
						|
    struct wl_output *wl_output = NULL;
 | 
						|
 | 
						|
    if (!xwl_window->xdg_toplevel)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    xwl_output = xwl_window_get_output(xwl_window);
 | 
						|
    if (xwl_output)
 | 
						|
        wl_output = xwl_output->output;
 | 
						|
 | 
						|
    if (wl_output && xwl_window->wl_output_fullscreen == wl_output)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    xdg_toplevel_set_fullscreen(xwl_window->xdg_toplevel, wl_output);
 | 
						|
    xwl_window_check_resolution_change_emulation(xwl_window);
 | 
						|
    wl_surface_commit(xwl_window->surface);
 | 
						|
 | 
						|
    xwl_window->wl_output_fullscreen = wl_output;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_rootful_update_fullscreen(struct xwl_window *xwl_window,
 | 
						|
                                     struct xwl_output *xwl_output)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    if (!xwl_screen->fullscreen)
 | 
						|
        return;
 | 
						|
 | 
						|
    if (xwl_window->window != xwl_screen->screen->root)
 | 
						|
        return;
 | 
						|
 | 
						|
    if (xwl_window->wl_output_fullscreen != xwl_output->output)
 | 
						|
        return;
 | 
						|
 | 
						|
    /* The size and position of the output may have changed, clear our
 | 
						|
     * output to make sure the next call to xwl_window_set_fullscreen()
 | 
						|
     * recomputes the size and updates the viewport as needed.
 | 
						|
     */
 | 
						|
    xwl_window->wl_output_fullscreen = NULL;
 | 
						|
    xwl_window_set_fullscreen(xwl_window);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_rootful_update_title(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    char title[128];
 | 
						|
    const char *grab_message = "";
 | 
						|
 | 
						|
    if (xwl_screen->host_grab) {
 | 
						|
        if (xwl_screen->has_grab)
 | 
						|
            grab_message = " - ([ctrl]+[shift] releases mouse and keyboard)";
 | 
						|
        else
 | 
						|
            grab_message = " - ([ctrl]+[shift] grabs mouse and keyboard)";
 | 
						|
    }
 | 
						|
 | 
						|
    snprintf(title, sizeof(title), "Xwayland on :%s%s", display, grab_message);
 | 
						|
 | 
						|
#ifdef XWL_HAS_LIBDECOR
 | 
						|
    if (xwl_window->libdecor_frame)
 | 
						|
        libdecor_frame_set_title(xwl_window->libdecor_frame, title);
 | 
						|
    else
 | 
						|
#endif
 | 
						|
    if (xwl_window->xdg_toplevel)
 | 
						|
        xdg_toplevel_set_title(xwl_window->xdg_toplevel, title);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_rootful_set_app_id(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    const char *app_id = "org.freedesktop.Xwayland";
 | 
						|
 | 
						|
#ifdef XWL_HAS_LIBDECOR
 | 
						|
    if (xwl_window->libdecor_frame)
 | 
						|
        libdecor_frame_set_app_id(xwl_window->libdecor_frame, app_id);
 | 
						|
    else
 | 
						|
#endif
 | 
						|
    if (xwl_window->xdg_toplevel)
 | 
						|
        xdg_toplevel_set_app_id(xwl_window->xdg_toplevel, app_id);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_maybe_resize(struct xwl_window *xwl_window, int width, int height)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    struct xwl_output *xwl_output;
 | 
						|
    RRModePtr mode;
 | 
						|
 | 
						|
    /* Clamp the size */
 | 
						|
    width = min(max(width, MIN_ROOTFUL_WIDTH), MAX_ROOTFUL_WIDTH);
 | 
						|
    height = min(max(height, MIN_ROOTFUL_HEIGHT), MAX_ROOTFUL_HEIGHT);
 | 
						|
 | 
						|
    if (width == xwl_screen->width && height == xwl_screen->height)
 | 
						|
        return;
 | 
						|
 | 
						|
    xwl_output = xwl_screen_get_fixed_or_first_output(xwl_screen);
 | 
						|
    if (!xwl_randr_add_modes_fixed(xwl_output, width, height))
 | 
						|
        return;
 | 
						|
 | 
						|
    mode = xwl_output_find_mode(xwl_output, width, height);
 | 
						|
    xwl_output_set_mode_fixed(xwl_output, mode);
 | 
						|
 | 
						|
    xwl_window_attach_buffer(xwl_window);
 | 
						|
}
 | 
						|
 | 
						|
#ifdef XWL_HAS_LIBDECOR
 | 
						|
static void
 | 
						|
xwl_window_libdecor_set_size_limits(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    libdecor_frame_set_min_content_size(xwl_window->libdecor_frame,
 | 
						|
                                        MIN_ROOTFUL_WIDTH, MIN_ROOTFUL_HEIGHT);
 | 
						|
    libdecor_frame_set_max_content_size(xwl_window->libdecor_frame,
 | 
						|
                                        MAX_ROOTFUL_WIDTH, MAX_ROOTFUL_HEIGHT);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_update_libdecor_size(struct xwl_window *xwl_window,
 | 
						|
                                struct libdecor_configuration *configuration /* nullable */,
 | 
						|
                                int width, int height)
 | 
						|
{
 | 
						|
    struct libdecor_state *state;
 | 
						|
 | 
						|
    if (xwl_window->libdecor_frame) {
 | 
						|
	state = libdecor_state_new(width, height);
 | 
						|
	libdecor_frame_commit(xwl_window->libdecor_frame, state, configuration);
 | 
						|
	libdecor_state_free(state);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
handle_libdecor_configure(struct libdecor_frame *frame,
 | 
						|
                          struct libdecor_configuration *configuration,
 | 
						|
                          void *data)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    int width, height;
 | 
						|
 | 
						|
    if (!libdecor_configuration_get_content_size(configuration, frame, &width, &height)) {
 | 
						|
        width = xwl_screen->width;
 | 
						|
        height = xwl_screen->height;
 | 
						|
    }
 | 
						|
 | 
						|
    xwl_window_maybe_resize(xwl_window, width, height);
 | 
						|
    xwl_window_update_libdecor_size(xwl_window, configuration,
 | 
						|
                                    xwl_screen->width, xwl_screen->height);
 | 
						|
    wl_surface_commit(xwl_window->surface);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
handle_libdecor_close(struct libdecor_frame *frame,
 | 
						|
                      void *data)
 | 
						|
{
 | 
						|
    DebugF("Terminating on compositor request");
 | 
						|
    GiveUp(0);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
handle_libdecor_commit(struct libdecor_frame *frame,
 | 
						|
                       void *data)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
    wl_surface_commit(xwl_window->surface);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
handle_libdecor_dismiss_popup(struct libdecor_frame *frame,
 | 
						|
                              const char *seat_name,
 | 
						|
                              void *data)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static struct libdecor_frame_interface libdecor_frame_iface = {
 | 
						|
    handle_libdecor_configure,
 | 
						|
    handle_libdecor_close,
 | 
						|
    handle_libdecor_commit,
 | 
						|
    handle_libdecor_dismiss_popup,
 | 
						|
};
 | 
						|
#endif
 | 
						|
 | 
						|
static void
 | 
						|
xdg_surface_handle_configure(void *data,
 | 
						|
                             struct xdg_surface *xdg_surface,
 | 
						|
                             uint32_t serial)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    if (xwl_screen->fullscreen)
 | 
						|
        xwl_window_set_fullscreen(xwl_window);
 | 
						|
 | 
						|
    xdg_surface_ack_configure(xdg_surface, serial);
 | 
						|
    wl_surface_commit(xwl_window->surface);
 | 
						|
}
 | 
						|
 | 
						|
static const struct xdg_surface_listener xdg_surface_listener = {
 | 
						|
    xdg_surface_handle_configure,
 | 
						|
};
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_surface_enter(void *data,
 | 
						|
                         struct wl_surface *wl_surface,
 | 
						|
                         struct wl_output *wl_output)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    if (xwl_window->wl_output != wl_output) {
 | 
						|
        xwl_window->wl_output = wl_output;
 | 
						|
 | 
						|
        if (xwl_screen->fullscreen)
 | 
						|
            xwl_window_set_fullscreen(xwl_window);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_surface_leave(void *data,
 | 
						|
                         struct wl_surface *wl_surface,
 | 
						|
                         struct wl_output *wl_output)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    if (xwl_window->wl_output == wl_output)
 | 
						|
        xwl_window->wl_output = NULL;
 | 
						|
}
 | 
						|
 | 
						|
static const struct wl_surface_listener surface_listener = {
 | 
						|
    xwl_window_surface_enter,
 | 
						|
    xwl_window_surface_leave
 | 
						|
};
 | 
						|
 | 
						|
static void
 | 
						|
xdg_toplevel_handle_configure(void *data,
 | 
						|
                              struct xdg_toplevel *xdg_toplevel,
 | 
						|
                              int32_t width,
 | 
						|
                              int32_t height,
 | 
						|
                              struct wl_array *states)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    /* Maintain our current size if no dimensions are requested */
 | 
						|
    if (width == 0 && height == 0)
 | 
						|
        return;
 | 
						|
 | 
						|
    if (!xwl_screen->fullscreen) {
 | 
						|
        /* This will be committed by the xdg_surface.configure handler */
 | 
						|
        xwl_window_maybe_resize(xwl_window, width, height);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xdg_toplevel_handle_close(void *data,
 | 
						|
                          struct xdg_toplevel *xdg_toplevel)
 | 
						|
{
 | 
						|
    DebugF("Terminating on compositor request");
 | 
						|
    GiveUp(0);
 | 
						|
}
 | 
						|
 | 
						|
static const struct xdg_toplevel_listener xdg_toplevel_listener = {
 | 
						|
    xdg_toplevel_handle_configure,
 | 
						|
    xdg_toplevel_handle_close,
 | 
						|
};
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_create_root_surface(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    WindowPtr window = xwl_window->window;
 | 
						|
    struct wl_region *region;
 | 
						|
 | 
						|
 | 
						|
#ifdef XWL_HAS_LIBDECOR
 | 
						|
    if (xwl_screen->decorate) {
 | 
						|
        xwl_window->libdecor_frame =
 | 
						|
            libdecor_decorate(xwl_screen->libdecor_context,
 | 
						|
                              xwl_window->surface,
 | 
						|
                              &libdecor_frame_iface,
 | 
						|
                              xwl_window);
 | 
						|
        xwl_window_libdecor_set_size_limits(xwl_window);
 | 
						|
        libdecor_frame_map(xwl_window->libdecor_frame);
 | 
						|
    }
 | 
						|
    else
 | 
						|
#endif
 | 
						|
    {
 | 
						|
        xwl_window->xdg_surface =
 | 
						|
            xdg_wm_base_get_xdg_surface(xwl_screen->xdg_wm_base, xwl_window->surface);
 | 
						|
        if (xwl_window->xdg_surface == NULL) {
 | 
						|
            ErrorF("Failed creating xdg_wm_base xdg_surface\n");
 | 
						|
            goto err_surf;
 | 
						|
        }
 | 
						|
 | 
						|
        xwl_window->xdg_toplevel =
 | 
						|
            xdg_surface_get_toplevel(xwl_window->xdg_surface);
 | 
						|
        if (xwl_window->xdg_surface == NULL) {
 | 
						|
            ErrorF("Failed creating xdg_toplevel\n");
 | 
						|
            goto err_surf;
 | 
						|
        }
 | 
						|
 | 
						|
        wl_surface_add_listener(xwl_window->surface,
 | 
						|
                                &surface_listener, xwl_window);
 | 
						|
 | 
						|
        xdg_surface_add_listener(xwl_window->xdg_surface,
 | 
						|
                                 &xdg_surface_listener, xwl_window);
 | 
						|
 | 
						|
        xdg_toplevel_add_listener(xwl_window->xdg_toplevel,
 | 
						|
                                  &xdg_toplevel_listener,
 | 
						|
                                  xwl_window);
 | 
						|
    }
 | 
						|
 | 
						|
    xwl_window_rootful_update_title(xwl_window);
 | 
						|
    xwl_window_rootful_set_app_id(xwl_window);
 | 
						|
    wl_surface_commit(xwl_window->surface);
 | 
						|
 | 
						|
    region = wl_compositor_create_region(xwl_screen->compositor);
 | 
						|
    if (region == NULL) {
 | 
						|
        ErrorF("Failed creating region\n");
 | 
						|
        goto err_surf;
 | 
						|
    }
 | 
						|
 | 
						|
    wl_region_add(region, 0, 0,
 | 
						|
                  window->drawable.width, window->drawable.height);
 | 
						|
    wl_surface_set_opaque_region(xwl_window->surface, region);
 | 
						|
    wl_region_destroy(region);
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
 | 
						|
err_surf:
 | 
						|
    if (xwl_window->xdg_toplevel)
 | 
						|
        xdg_toplevel_destroy(xwl_window->xdg_toplevel);
 | 
						|
    if (xwl_window->xdg_surface)
 | 
						|
        xdg_surface_destroy(xwl_window->xdg_surface);
 | 
						|
    wl_surface_destroy(xwl_window->surface);
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
ensure_surface_for_window(WindowPtr window)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen;
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
    WindowPtr toplevel;
 | 
						|
 | 
						|
    if (xwl_window_from_window(window))
 | 
						|
        return TRUE;
 | 
						|
 | 
						|
    xwl_screen = xwl_screen_get(screen);
 | 
						|
 | 
						|
    if (xwl_screen->rootless) {
 | 
						|
        if (window->redirectDraw != RedirectDrawManual)
 | 
						|
            return TRUE;
 | 
						|
    }
 | 
						|
    else {
 | 
						|
        if (window->parent)
 | 
						|
            return TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
    xwl_window = calloc(1, sizeof *xwl_window);
 | 
						|
    if (xwl_window == NULL)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    xwl_window->xwl_screen = xwl_screen;
 | 
						|
    xwl_window->window = window;
 | 
						|
    xwl_window->surface = wl_compositor_create_surface(xwl_screen->compositor);
 | 
						|
    if (xwl_window->surface == NULL) {
 | 
						|
        ErrorF("wl_display_create_surface failed\n");
 | 
						|
        goto err;
 | 
						|
    }
 | 
						|
 | 
						|
    if (xwl_screen->xwayland_shell) {
 | 
						|
        xwl_window->xwayland_surface = xwayland_shell_v1_get_xwayland_surface(
 | 
						|
            xwl_screen->xwayland_shell, xwl_window->surface);
 | 
						|
    }
 | 
						|
 | 
						|
    if (!xwl_screen->rootless && !xwl_create_root_surface(xwl_window))
 | 
						|
        goto err;
 | 
						|
 | 
						|
#ifdef XWL_HAS_GLAMOR
 | 
						|
    if (xwl_screen->dmabuf_protocol_version >= 4)
 | 
						|
        xwl_dmabuf_setup_feedback_for_window(xwl_window);
 | 
						|
#endif
 | 
						|
 | 
						|
    wl_display_flush(xwl_screen->display);
 | 
						|
 | 
						|
    send_surface_id_event(xwl_window);
 | 
						|
 | 
						|
    wl_surface_set_user_data(xwl_window->surface, xwl_window);
 | 
						|
    xwl_window_set_xwayland_tag(xwl_window);
 | 
						|
 | 
						|
    compRedirectWindow(serverClient, window, CompositeRedirectManual);
 | 
						|
 | 
						|
    dixSetPrivate(&window->devPrivates, &xwl_window_private_key, xwl_window);
 | 
						|
    xorg_list_init(&xwl_window->link_damage);
 | 
						|
    xorg_list_add(&xwl_window->link_window, &xwl_screen->window_list);
 | 
						|
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    xorg_list_init(&xwl_window->frame_callback_list);
 | 
						|
#endif
 | 
						|
 | 
						|
    xwl_window_buffers_init(xwl_window);
 | 
						|
 | 
						|
    xwl_window_init_allow_commits(xwl_window);
 | 
						|
 | 
						|
    /* When a new window-manager window is realized, then the randr emulation
 | 
						|
     * props may have not been set on the managed client window yet.
 | 
						|
     */
 | 
						|
    if (!xwl_screen->fullscreen && window_is_wm_window(window)) {
 | 
						|
        toplevel = window_get_client_toplevel(window);
 | 
						|
        if (toplevel)
 | 
						|
            xwl_output_set_window_randr_emu_props(xwl_screen, toplevel);
 | 
						|
    } else {
 | 
						|
        /* CSD or O-R toplevel window, check viewport on creation */
 | 
						|
        xwl_window_check_resolution_change_emulation(xwl_window);
 | 
						|
    }
 | 
						|
 | 
						|
    if (xwl_screen->tearing_control_manager) {
 | 
						|
        xwl_window->tearing_control = wp_tearing_control_manager_v1_get_tearing_control(
 | 
						|
            xwl_screen->tearing_control_manager, xwl_window->surface);
 | 
						|
    }
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
 | 
						|
err:
 | 
						|
    free(xwl_window);
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_realize_window(WindowPtr window)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    CompScreenPtr comp_screen = GetCompScreen(screen);
 | 
						|
    struct xwl_screen *xwl_screen;
 | 
						|
    Bool ret;
 | 
						|
 | 
						|
    xwl_screen = xwl_screen_get(screen);
 | 
						|
 | 
						|
    screen->RealizeWindow = xwl_screen->RealizeWindow;
 | 
						|
    ret = (*screen->RealizeWindow) (window);
 | 
						|
    xwl_screen->RealizeWindow = screen->RealizeWindow;
 | 
						|
    screen->RealizeWindow = xwl_realize_window;
 | 
						|
 | 
						|
    if (!ret)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    if (xwl_screen->rootless) {
 | 
						|
        /* We do not want the COW to be mapped when rootless in Xwayland */
 | 
						|
        if (window == comp_screen->pOverlayWin) {
 | 
						|
            window->mapped = FALSE;
 | 
						|
            return TRUE;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!window->parent) {
 | 
						|
            BoxRec box = { 0, 0, xwl_screen->width, xwl_screen->height };
 | 
						|
 | 
						|
            RegionReset(&window->winSize, &box);
 | 
						|
            RegionNull(&window->clipList);
 | 
						|
            RegionNull(&window->borderClip);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (xwl_screen->rootless ?
 | 
						|
        (window->drawable.class == InputOutput &&
 | 
						|
         window->parent == window->drawable.pScreen->root) :
 | 
						|
        !window->parent) {
 | 
						|
        if (!register_damage(window))
 | 
						|
            return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
    return ensure_surface_for_window(window);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_surface_destroy_free_timer(struct xwl_wl_surface *xwl_wl_surface)
 | 
						|
{
 | 
						|
    if (xwl_wl_surface->wl_surface_destroy_timer) {
 | 
						|
        TimerFree(xwl_wl_surface->wl_surface_destroy_timer);
 | 
						|
        xwl_wl_surface->wl_surface_destroy_timer = NULL;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_surface_do_destroy(struct xwl_wl_surface *xwl_wl_surface)
 | 
						|
{
 | 
						|
    wl_surface_destroy(xwl_wl_surface->wl_surface);
 | 
						|
    xorg_list_del(&xwl_wl_surface->link);
 | 
						|
    xwl_surface_destroy_free_timer(xwl_wl_surface);
 | 
						|
    free(xwl_wl_surface);
 | 
						|
}
 | 
						|
 | 
						|
static CARD32
 | 
						|
xwl_surface_destroy_callback(OsTimerPtr timer, CARD32 now, void *arg)
 | 
						|
{
 | 
						|
    struct xwl_wl_surface *xwl_wl_surface = arg;
 | 
						|
 | 
						|
    xwl_window_surface_do_destroy(xwl_wl_surface);
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
release_wl_surface_for_window_legacy_delay(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_wl_surface *xwl_wl_surface;
 | 
						|
 | 
						|
    /* If the Xserver is terminating, destroy the surface immediately */
 | 
						|
    if ((dispatchException & DE_TERMINATE) == DE_TERMINATE) {
 | 
						|
        wl_surface_destroy(xwl_window->surface);
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    /* Break the wl_surface / xwl_window relationship */
 | 
						|
    wl_surface_set_user_data(xwl_window->surface, NULL);
 | 
						|
    xwl_window_clear_xwayland_tag(xwl_window);
 | 
						|
 | 
						|
    /* Schedule the destruction later, to mitigate the race between X11
 | 
						|
     * and Wayland processing so that the compositor has the time to
 | 
						|
     * establish the association before the wl_surface is destroyed.
 | 
						|
     */
 | 
						|
    xwl_wl_surface = xnfcalloc(1, sizeof *xwl_wl_surface);
 | 
						|
    xwl_wl_surface->wl_surface = xwl_window->surface;
 | 
						|
    xorg_list_add(&xwl_wl_surface->link,
 | 
						|
                  &xwl_window->xwl_screen->pending_wl_surface_destroy);
 | 
						|
    xwl_wl_surface->wl_surface_destroy_timer =
 | 
						|
        TimerSet(NULL, 0, DELAYED_WL_SURFACE_DESTROY,
 | 
						|
                 xwl_surface_destroy_callback, xwl_wl_surface);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
release_wl_surface_for_window_shell(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    xwayland_surface_v1_destroy(xwl_window->xwayland_surface);
 | 
						|
    wl_surface_destroy(xwl_window->surface);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
release_wl_surface_for_window(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    if (xwl_window->xwayland_surface)
 | 
						|
        release_wl_surface_for_window_shell(xwl_window);
 | 
						|
    else
 | 
						|
        release_wl_surface_for_window_legacy_delay(xwl_window);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_device_formats_destroy(struct xwl_device_formats *dev_formats)
 | 
						|
{
 | 
						|
    for (int j = 0; j < dev_formats->num_formats; j++)
 | 
						|
        free(dev_formats->formats[j].modifiers);
 | 
						|
    free(dev_formats->formats);
 | 
						|
    drmFreeDevice(&dev_formats->drm_dev);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_dmabuf_feedback_clear_dev_formats(struct xwl_dmabuf_feedback *xwl_feedback)
 | 
						|
{
 | 
						|
    if (xwl_feedback->dev_formats_len == 0)
 | 
						|
        return;
 | 
						|
 | 
						|
    for (int i = 0; i < xwl_feedback->dev_formats_len; i++) {
 | 
						|
        struct xwl_device_formats *dev_format = &xwl_feedback->dev_formats[i];
 | 
						|
        xwl_device_formats_destroy(dev_format);
 | 
						|
    }
 | 
						|
    free(xwl_feedback->dev_formats);
 | 
						|
    xwl_feedback->dev_formats = NULL;
 | 
						|
    xwl_feedback->dev_formats_len = 0;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_dmabuf_feedback_destroy(struct xwl_dmabuf_feedback *xwl_feedback)
 | 
						|
{
 | 
						|
    munmap(xwl_feedback->format_table.entry,
 | 
						|
           xwl_feedback->format_table.len * sizeof(struct xwl_format_table_entry));
 | 
						|
    xwl_dmabuf_feedback_clear_dev_formats(xwl_feedback);
 | 
						|
 | 
						|
    if (xwl_feedback->dmabuf_feedback)
 | 
						|
        zwp_linux_dmabuf_feedback_v1_destroy(xwl_feedback->dmabuf_feedback);
 | 
						|
 | 
						|
    xwl_feedback->dmabuf_feedback = NULL;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_unrealize_window(WindowPtr window)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen;
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
    struct xwl_seat *xwl_seat;
 | 
						|
    Bool ret;
 | 
						|
 | 
						|
    xwl_screen = xwl_screen_get(screen);
 | 
						|
 | 
						|
    xorg_list_for_each_entry(xwl_seat, &xwl_screen->seat_list, link) {
 | 
						|
        if (xwl_seat->focus_window && xwl_seat->focus_window->window == window)
 | 
						|
            xwl_seat->focus_window = NULL;
 | 
						|
        if (xwl_seat->tablet_focus_window && xwl_seat->tablet_focus_window->window == window)
 | 
						|
            xwl_seat->tablet_focus_window = NULL;
 | 
						|
        if (xwl_seat->last_xwindow == window)
 | 
						|
            xwl_seat->last_xwindow = NullWindow;
 | 
						|
        if (xwl_seat->cursor_confinement_window &&
 | 
						|
            xwl_seat->cursor_confinement_window->window == window)
 | 
						|
            xwl_seat_unconfine_pointer(xwl_seat);
 | 
						|
        if (xwl_seat->pointer_warp_emulator &&
 | 
						|
            xwl_seat->pointer_warp_emulator->locked_window &&
 | 
						|
            xwl_seat->pointer_warp_emulator->locked_window->window == window)
 | 
						|
            xwl_seat_destroy_pointer_warp_emulator(xwl_seat);
 | 
						|
        xwl_seat_clear_touch(xwl_seat, window);
 | 
						|
    }
 | 
						|
 | 
						|
    compUnredirectWindow(serverClient, window, CompositeRedirectManual);
 | 
						|
 | 
						|
    screen->UnrealizeWindow = xwl_screen->UnrealizeWindow;
 | 
						|
    ret = (*screen->UnrealizeWindow) (window);
 | 
						|
    xwl_screen->UnrealizeWindow = screen->UnrealizeWindow;
 | 
						|
    screen->UnrealizeWindow = xwl_unrealize_window;
 | 
						|
 | 
						|
    xwl_window = xwl_window_get(window);
 | 
						|
    if (!xwl_window)
 | 
						|
        return ret;
 | 
						|
 | 
						|
    if (xwl_window_has_viewport_enabled(xwl_window))
 | 
						|
        xwl_window_disable_viewport(xwl_window);
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_destroy(&xwl_window->feedback);
 | 
						|
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    if (xwl_window->xwl_screen->present)
 | 
						|
        xwl_present_for_each_frame_callback(xwl_window, xwl_present_unrealize_window);
 | 
						|
#endif
 | 
						|
 | 
						|
    if (xwl_window->tearing_control)
 | 
						|
        wp_tearing_control_v1_destroy(xwl_window->tearing_control);
 | 
						|
 | 
						|
    release_wl_surface_for_window(xwl_window);
 | 
						|
    xorg_list_del(&xwl_window->link_damage);
 | 
						|
    xorg_list_del(&xwl_window->link_window);
 | 
						|
    unregister_damage(window);
 | 
						|
 | 
						|
    xwl_window_buffers_dispose(xwl_window);
 | 
						|
 | 
						|
    if (xwl_window->frame_callback)
 | 
						|
        wl_callback_destroy(xwl_window->frame_callback);
 | 
						|
 | 
						|
    free(xwl_window);
 | 
						|
    dixSetPrivate(&window->devPrivates, &xwl_window_private_key, NULL);
 | 
						|
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_set_window_pixmap(WindowPtr window,
 | 
						|
                             PixmapPtr pixmap)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen;
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
    PixmapPtr old_pixmap;
 | 
						|
 | 
						|
    old_pixmap = (*screen->GetWindowPixmap) (window);
 | 
						|
    xwl_screen = xwl_screen_get(screen);
 | 
						|
 | 
						|
    screen->SetWindowPixmap = xwl_screen->SetWindowPixmap;
 | 
						|
    (*screen->SetWindowPixmap) (window, pixmap);
 | 
						|
    xwl_screen->SetWindowPixmap = screen->SetWindowPixmap;
 | 
						|
    screen->SetWindowPixmap = xwl_window_set_window_pixmap;
 | 
						|
 | 
						|
    if (!RegionNotEmpty(&window->winSize))
 | 
						|
        return;
 | 
						|
 | 
						|
    ensure_surface_for_window(window);
 | 
						|
 | 
						|
    if (old_pixmap->drawable.width == pixmap->drawable.width &&
 | 
						|
        old_pixmap->drawable.height == pixmap->drawable.height)
 | 
						|
       return;
 | 
						|
 | 
						|
    xwl_window = xwl_window_get(window);
 | 
						|
    if (xwl_window)
 | 
						|
            xwl_window_buffers_recycle(xwl_window);
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_change_window_attributes(WindowPtr window, unsigned long mask)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
    OtherClients *others;
 | 
						|
    Bool ret;
 | 
						|
 | 
						|
    screen->ChangeWindowAttributes = xwl_screen->ChangeWindowAttributes;
 | 
						|
    ret = (*screen->ChangeWindowAttributes) (window, mask);
 | 
						|
    xwl_screen->ChangeWindowAttributes = screen->ChangeWindowAttributes;
 | 
						|
    screen->ChangeWindowAttributes = xwl_change_window_attributes;
 | 
						|
 | 
						|
    if (window != screen->root || !(mask & CWEventMask))
 | 
						|
        return ret;
 | 
						|
 | 
						|
    for (others = wOtherClients(window); others; others = others->next) {
 | 
						|
        if (others->mask & (SubstructureRedirectMask | ResizeRedirectMask))
 | 
						|
            xwl_screen->wm_client_id = CLIENT_ID(others->resource);
 | 
						|
    }
 | 
						|
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_resize_window(WindowPtr window,
 | 
						|
                  int x, int y,
 | 
						|
                  unsigned int width, unsigned int height,
 | 
						|
                  WindowPtr sib)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen;
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
 | 
						|
    xwl_screen = xwl_screen_get(screen);
 | 
						|
    xwl_window = xwl_window_from_window(window);
 | 
						|
 | 
						|
    screen->ResizeWindow = xwl_screen->ResizeWindow;
 | 
						|
    (*screen->ResizeWindow) (window, x, y, width, height, sib);
 | 
						|
    xwl_screen->ResizeWindow = screen->ResizeWindow;
 | 
						|
    screen->ResizeWindow = xwl_resize_window;
 | 
						|
 | 
						|
    if (xwl_window) {
 | 
						|
        if (xwl_window_get(window) || xwl_window_is_toplevel(window))
 | 
						|
            xwl_window_check_resolution_change_emulation(xwl_window);
 | 
						|
#ifdef XWL_HAS_LIBDECOR
 | 
						|
        if (window == screen->root)
 | 
						|
            xwl_window_update_libdecor_size(xwl_window, NULL, width, height);
 | 
						|
#endif
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_move_window(WindowPtr window,
 | 
						|
                int x, int y,
 | 
						|
                WindowPtr next_sib,
 | 
						|
                VTKind kind)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen;
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
 | 
						|
    xwl_screen = xwl_screen_get(screen);
 | 
						|
    xwl_window = xwl_window_from_window(window);
 | 
						|
 | 
						|
    screen->MoveWindow = xwl_screen->MoveWindow;
 | 
						|
    (*screen->MoveWindow) (window, x, y, next_sib, kind);
 | 
						|
    xwl_screen->MoveWindow = screen->MoveWindow;
 | 
						|
    screen->MoveWindow = xwl_move_window;
 | 
						|
 | 
						|
    if (xwl_window && (xwl_window_get(window) || xwl_window_is_toplevel(window)))
 | 
						|
        xwl_window_check_resolution_change_emulation(xwl_window);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
frame_callback(void *data,
 | 
						|
               struct wl_callback *callback,
 | 
						|
               uint32_t time)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    wl_callback_destroy (xwl_window->frame_callback);
 | 
						|
    xwl_window->frame_callback = NULL;
 | 
						|
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    if (xwl_window->xwl_screen->present) {
 | 
						|
        xwl_present_for_each_frame_callback(xwl_window, xwl_present_frame_callback);
 | 
						|
 | 
						|
        /* If xwl_window_create_frame_callback was called from
 | 
						|
         * xwl_present_frame_callback, need to make sure all fallback timers
 | 
						|
         * are adjusted correspondingly.
 | 
						|
         */
 | 
						|
        if (xwl_window->frame_callback)
 | 
						|
            xwl_present_for_each_frame_callback(xwl_window, xwl_present_reset_timer);
 | 
						|
    }
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static const struct wl_callback_listener frame_listener = {
 | 
						|
    frame_callback
 | 
						|
};
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_create_frame_callback(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    xwl_window->frame_callback = wl_surface_frame(xwl_window->surface);
 | 
						|
    wl_callback_add_listener(xwl_window->frame_callback, &frame_listener,
 | 
						|
                             xwl_window);
 | 
						|
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    /* If we get called from frame_callback, it will take care of calling
 | 
						|
     * xwl_present_reset_timer.
 | 
						|
     */
 | 
						|
    if (xwl_window->xwl_screen->present &&
 | 
						|
        !xwl_present_entered_for_each_frame_callback())
 | 
						|
        xwl_present_for_each_frame_callback(xwl_window, xwl_present_reset_timer);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_destroy_window(WindowPtr window)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
    Bool ret;
 | 
						|
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    if (xwl_screen->present)
 | 
						|
        xwl_present_cleanup(window);
 | 
						|
#endif
 | 
						|
 | 
						|
    screen->DestroyWindow = xwl_screen->DestroyWindow;
 | 
						|
 | 
						|
    if (screen->DestroyWindow)
 | 
						|
        ret = screen->DestroyWindow (window);
 | 
						|
    else
 | 
						|
        ret = TRUE;
 | 
						|
 | 
						|
    xwl_screen->DestroyWindow = screen->DestroyWindow;
 | 
						|
    screen->DestroyWindow = xwl_destroy_window;
 | 
						|
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_window_attach_buffer(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
    RegionPtr region;
 | 
						|
    BoxPtr box;
 | 
						|
    struct wl_buffer *buffer;
 | 
						|
    PixmapPtr pixmap;
 | 
						|
    int i;
 | 
						|
 | 
						|
    region = DamageRegion(window_get_damage(xwl_window->window));
 | 
						|
    pixmap = xwl_window_buffers_get_pixmap(xwl_window, region);
 | 
						|
 | 
						|
#ifdef XWL_HAS_GLAMOR
 | 
						|
    if (xwl_screen->glamor)
 | 
						|
        buffer = xwl_glamor_pixmap_get_wl_buffer(pixmap);
 | 
						|
    else
 | 
						|
#endif
 | 
						|
        buffer = xwl_shm_pixmap_get_wl_buffer(pixmap);
 | 
						|
 | 
						|
    if (!buffer) {
 | 
						|
        ErrorF("Error getting buffer\n");
 | 
						|
        return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
#ifdef XWL_HAS_GLAMOR
 | 
						|
    if (xwl_screen->glamor) {
 | 
						|
        if (!xwl_glamor_post_damage(xwl_window, pixmap, region)) {
 | 
						|
            ErrorF("glamor: Failed to post damage\n");
 | 
						|
            return FALSE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
#endif
 | 
						|
 | 
						|
    wl_surface_attach(xwl_window->surface, buffer, 0, 0);
 | 
						|
 | 
						|
    /* Arbitrary limit to try to avoid flooding the Wayland
 | 
						|
     * connection. If we flood it too much anyway, this could
 | 
						|
     * abort in libwayland-client.
 | 
						|
     */
 | 
						|
    if (RegionNumRects(region) > 256) {
 | 
						|
        box = RegionExtents(region);
 | 
						|
        xwl_surface_damage(xwl_screen, xwl_window->surface,
 | 
						|
                           box->x1 + xwl_window->window->borderWidth,
 | 
						|
                           box->y1 + xwl_window->window->borderWidth,
 | 
						|
                           box->x2 - box->x1, box->y2 - box->y1);
 | 
						|
    } else {
 | 
						|
        box = RegionRects(region);
 | 
						|
        for (i = 0; i < RegionNumRects(region); i++, box++) {
 | 
						|
            xwl_surface_damage(xwl_screen, xwl_window->surface,
 | 
						|
                               box->x1 + xwl_window->window->borderWidth,
 | 
						|
                               box->y1 + xwl_window->window->borderWidth,
 | 
						|
                               box->x2 - box->x1, box->y2 - box->y1);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_window_post_damage(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    assert(!xwl_window->frame_callback);
 | 
						|
 | 
						|
    if (!xwl_window_attach_buffer(xwl_window))
 | 
						|
        return;
 | 
						|
 | 
						|
    xwl_window_create_frame_callback(xwl_window);
 | 
						|
    DamageEmpty(window_get_damage(xwl_window->window));
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_window_init(void)
 | 
						|
{
 | 
						|
    if (!dixRegisterPrivateKey(&xwl_window_private_key, PRIVATE_WINDOW, 0))
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    if (!dixRegisterPrivateKey(&xwl_damage_private_key, PRIVATE_WINDOW, 0))
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 |