1177 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			1177 lines
		
	
	
		
			38 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.
 | 
						|
 */
 | 
						|
 | 
						|
#include <xwayland-config.h>
 | 
						|
 | 
						|
#include <compositeext.h>
 | 
						|
 | 
						|
#define MESA_EGL_NO_X11_HEADERS
 | 
						|
#define EGL_NO_X11
 | 
						|
#include <glamor_egl.h>
 | 
						|
 | 
						|
#include <glamor.h>
 | 
						|
#include <glamor_context.h>
 | 
						|
#include <glamor_glx_provider.h>
 | 
						|
#ifdef GLXEXT
 | 
						|
#include "glx_extinit.h"
 | 
						|
#endif
 | 
						|
 | 
						|
#include "linux-dmabuf-unstable-v1-client-protocol.h"
 | 
						|
#include "drm-client-protocol.h"
 | 
						|
#include <drm_fourcc.h>
 | 
						|
 | 
						|
#include "xwayland-glamor.h"
 | 
						|
#include "xwayland-screen.h"
 | 
						|
#include "xwayland-window.h"
 | 
						|
#include "xwayland-window-buffers.h"
 | 
						|
 | 
						|
#include <sys/mman.h>
 | 
						|
 | 
						|
static void
 | 
						|
glamor_egl_make_current(struct glamor_context *glamor_ctx)
 | 
						|
{
 | 
						|
    eglMakeCurrent(glamor_ctx->display, EGL_NO_SURFACE,
 | 
						|
                   EGL_NO_SURFACE, EGL_NO_CONTEXT);
 | 
						|
    if (!eglMakeCurrent(glamor_ctx->display,
 | 
						|
                        EGL_NO_SURFACE, EGL_NO_SURFACE,
 | 
						|
                        glamor_ctx->ctx))
 | 
						|
        FatalError("Failed to make EGL context current\n");
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_glamor_egl_make_current(struct xwl_screen *xwl_screen)
 | 
						|
{
 | 
						|
    EGLContext ctx = xwl_screen->glamor_ctx->ctx;
 | 
						|
    
 | 
						|
    if (lastGLContext == ctx)
 | 
						|
        return;
 | 
						|
 | 
						|
    lastGLContext = ctx;
 | 
						|
    xwl_screen->glamor_ctx->make_current(xwl_screen->glamor_ctx);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
glamor_egl_screen_init(ScreenPtr screen, struct glamor_context *glamor_ctx)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
 | 
						|
    glamor_set_glvnd_vendor(screen, xwl_screen->glvnd_vendor);
 | 
						|
    glamor_enable_dri3(screen);
 | 
						|
    glamor_ctx->ctx = xwl_screen->egl_context;
 | 
						|
    glamor_ctx->display = xwl_screen->egl_display;
 | 
						|
 | 
						|
    glamor_ctx->make_current = glamor_egl_make_current;
 | 
						|
 | 
						|
    xwl_screen->glamor_ctx = glamor_ctx;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_check_flip(WindowPtr present_window, PixmapPtr pixmap)
 | 
						|
{
 | 
						|
    ScreenPtr screen = pixmap->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
    PixmapPtr backing_pixmap = screen->GetWindowPixmap(present_window);
 | 
						|
 | 
						|
    if (pixmap->drawable.depth != backing_pixmap->drawable.depth)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    if (xwl_screen->egl_backend->check_flip)
 | 
						|
        return xwl_screen->egl_backend->check_flip(pixmap);
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_glamor_is_modifier_supported_in_formats(struct xwl_format *formats, int num_formats,
 | 
						|
                                            uint32_t format, uint64_t modifier)
 | 
						|
{
 | 
						|
    struct xwl_format *xwl_format = NULL;
 | 
						|
    int i;
 | 
						|
 | 
						|
    for (i = 0; i < num_formats; i++) {
 | 
						|
        if (formats[i].format == format) {
 | 
						|
            xwl_format = &formats[i];
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (xwl_format) {
 | 
						|
        for (i = 0; i < xwl_format->num_modifiers; i++) {
 | 
						|
            if (xwl_format->modifiers[i] == modifier) {
 | 
						|
                return TRUE;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_feedback_is_modifier_supported(struct xwl_dmabuf_feedback *xwl_feedback,
 | 
						|
                                   uint32_t format, uint64_t modifier,
 | 
						|
                                   int supports_scanout)
 | 
						|
{
 | 
						|
    for (int i = 0; i < xwl_feedback->dev_formats_len; i++) {
 | 
						|
        struct xwl_device_formats *dev_formats = &xwl_feedback->dev_formats[i];
 | 
						|
 | 
						|
        if (supports_scanout && !dev_formats->supports_scanout)
 | 
						|
            continue;
 | 
						|
 | 
						|
        if (xwl_glamor_is_modifier_supported_in_formats(dev_formats->formats,
 | 
						|
                                                        dev_formats->num_formats,
 | 
						|
                                                        format, modifier))
 | 
						|
            return TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_is_modifier_supported(struct xwl_screen *xwl_screen,
 | 
						|
                                 uint32_t format, uint64_t modifier)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
 | 
						|
    /*
 | 
						|
     * If we are using dmabuf v4, then we need to check in the main
 | 
						|
     * device and per-window format lists. For older protocol
 | 
						|
     * versions we can just check the list returned by the dmabuf.modifier
 | 
						|
     * events in xwl_screen
 | 
						|
     */
 | 
						|
    if (xwl_screen->dmabuf_protocol_version < 4) {
 | 
						|
        return xwl_glamor_is_modifier_supported_in_formats(xwl_screen->formats,
 | 
						|
                                                           xwl_screen->num_formats,
 | 
						|
                                                           format, modifier);
 | 
						|
    }
 | 
						|
 | 
						|
    if (xwl_feedback_is_modifier_supported(&xwl_screen->default_feedback, format, modifier, FALSE))
 | 
						|
        return TRUE;
 | 
						|
 | 
						|
    xorg_list_for_each_entry(xwl_window, &xwl_screen->window_list, link_window) {
 | 
						|
        if (xwl_feedback_is_modifier_supported(&xwl_window->feedback, format, modifier, FALSE))
 | 
						|
            return TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
uint32_t
 | 
						|
wl_drm_format_for_depth(int depth)
 | 
						|
{
 | 
						|
    switch (depth) {
 | 
						|
    case 15:
 | 
						|
        return WL_DRM_FORMAT_XRGB1555;
 | 
						|
    case 16:
 | 
						|
        return WL_DRM_FORMAT_RGB565;
 | 
						|
    case 24:
 | 
						|
        return WL_DRM_FORMAT_XRGB8888;
 | 
						|
    case 30:
 | 
						|
        return WL_DRM_FORMAT_ARGB2101010;
 | 
						|
    default:
 | 
						|
        ErrorF("unexpected depth: %d\n", depth);
 | 
						|
    case 32:
 | 
						|
        return WL_DRM_FORMAT_ARGB8888;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static drmDevice *
 | 
						|
xwl_screen_get_main_dev(struct xwl_screen *xwl_screen)
 | 
						|
{
 | 
						|
    /*
 | 
						|
     * If we have gbm then get our main device from it. Otherwise use what
 | 
						|
     * the compositor told us.
 | 
						|
     */
 | 
						|
    if (xwl_screen->gbm_backend.is_available)
 | 
						|
        return xwl_screen->gbm_backend.get_main_device(xwl_screen);
 | 
						|
    else
 | 
						|
        return xwl_screen->default_feedback.main_dev;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_get_formats(struct xwl_format *format_array, int format_array_len,
 | 
						|
               CARD32 *num_formats, CARD32 **formats)
 | 
						|
{
 | 
						|
    *num_formats = 0;
 | 
						|
    *formats = NULL;
 | 
						|
 | 
						|
    if (format_array_len == 0)
 | 
						|
       return TRUE;
 | 
						|
 | 
						|
    *formats = calloc(format_array_len, sizeof(CARD32));
 | 
						|
    if (*formats == NULL)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    for (int i = 0; i < format_array_len; i++)
 | 
						|
       (*formats)[i] = format_array[i].format;
 | 
						|
    *num_formats = format_array_len;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_get_formats_for_device(struct xwl_dmabuf_feedback *xwl_feedback, drmDevice *device,
 | 
						|
                           CARD32 *num_formats, CARD32 **formats)
 | 
						|
{
 | 
						|
    CARD32 *ret = NULL;
 | 
						|
    uint32_t count = 0;
 | 
						|
 | 
						|
    /* go through all matching sets of tranches for the window's device */
 | 
						|
    for (int i = 0; i < xwl_feedback->dev_formats_len; i++) {
 | 
						|
        if (drmDevicesEqual(xwl_feedback->dev_formats[i].drm_dev, device)) {
 | 
						|
            struct xwl_device_formats *dev_formats = &xwl_feedback->dev_formats[i];
 | 
						|
 | 
						|
            /* Append the formats from this tranche to the list */
 | 
						|
            ret = xnfreallocarray(ret, count + dev_formats->num_formats, sizeof(CARD32));
 | 
						|
 | 
						|
            for (int j = 0; j < dev_formats->num_formats; j++) {
 | 
						|
                bool found = false;
 | 
						|
 | 
						|
                /* Check if this format is already present in the list */
 | 
						|
                for (int k = 0; k < count; k++) {
 | 
						|
                    if (ret[k] == dev_formats->formats[j].format) {
 | 
						|
                        found = true;
 | 
						|
                        break;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                /* If this format has not yet been added, do so now */
 | 
						|
                if (!found)
 | 
						|
                    ret[count++] = dev_formats->formats[j].format;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    *num_formats = count;
 | 
						|
    *formats = ret;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_get_formats(ScreenPtr screen,
 | 
						|
                       CARD32 *num_formats, CARD32 **formats)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
 | 
						|
    /* Explicitly zero the count as the caller may ignore the return value */
 | 
						|
    *num_formats = 0;
 | 
						|
 | 
						|
    if (!xwl_screen->dmabuf)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    if (xwl_screen->dmabuf_protocol_version >= 4) {
 | 
						|
        drmDevice *main_dev = xwl_screen_get_main_dev(xwl_screen);
 | 
						|
 | 
						|
        return xwl_get_formats_for_device(&xwl_screen->default_feedback, main_dev,
 | 
						|
                                          num_formats, formats);
 | 
						|
    }
 | 
						|
 | 
						|
    return xwl_get_formats(xwl_screen->formats, xwl_screen->num_formats,
 | 
						|
                           num_formats, formats);
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_get_modifiers_for_format(struct xwl_format *format_array, int num_formats,
 | 
						|
                             uint32_t format, uint32_t *num_modifiers, uint64_t **modifiers)
 | 
						|
{
 | 
						|
    struct xwl_format *xwl_format = NULL;
 | 
						|
    int i;
 | 
						|
 | 
						|
    *num_modifiers = 0;
 | 
						|
    *modifiers = NULL;
 | 
						|
 | 
						|
    if (num_formats == 0)
 | 
						|
       return TRUE;
 | 
						|
 | 
						|
    for (i = 0; i < num_formats; i++) {
 | 
						|
       if (format_array[i].format == format) {
 | 
						|
          xwl_format = &format_array[i];
 | 
						|
          break;
 | 
						|
       }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!xwl_format ||
 | 
						|
        (xwl_format->num_modifiers == 1 &&
 | 
						|
         xwl_format->modifiers[0] == DRM_FORMAT_MOD_INVALID))
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    *modifiers = calloc(xwl_format->num_modifiers, sizeof(uint64_t));
 | 
						|
    if (*modifiers == NULL)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    for (i = 0; i < xwl_format->num_modifiers; i++)
 | 
						|
       (*modifiers)[i] = xwl_format->modifiers[i];
 | 
						|
    *num_modifiers = xwl_format->num_modifiers;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_get_modifiers_for_device(struct xwl_dmabuf_feedback *feedback, drmDevice *device,
 | 
						|
                             uint32_t format, uint32_t *num_modifiers,
 | 
						|
                             uint64_t **modifiers,
 | 
						|
                             Bool *supports_scanout)
 | 
						|
{
 | 
						|
    /* Now try to find a matching set of tranches for the window's device */
 | 
						|
    for (int i = 0; i < feedback->dev_formats_len; i++) {
 | 
						|
        struct xwl_device_formats *dev_formats = &feedback->dev_formats[i];
 | 
						|
 | 
						|
        if (drmDevicesEqual(dev_formats->drm_dev, device) &&
 | 
						|
            xwl_get_modifiers_for_format(dev_formats->formats, dev_formats->num_formats,
 | 
						|
                                         format, num_modifiers, modifiers)) {
 | 
						|
            if (supports_scanout)
 | 
						|
                *supports_scanout = !!dev_formats->supports_scanout;
 | 
						|
            return TRUE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_get_modifiers(ScreenPtr screen, uint32_t format,
 | 
						|
                         uint32_t *num_modifiers, uint64_t **modifiers)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
    drmDevice *main_dev;
 | 
						|
 | 
						|
    /* Explicitly zero the count as the caller may ignore the return value */
 | 
						|
    *num_modifiers = 0;
 | 
						|
    *modifiers = NULL;
 | 
						|
 | 
						|
    if (!xwl_screen->dmabuf)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    if (xwl_screen->dmabuf_protocol_version >= 4) {
 | 
						|
        main_dev = xwl_screen_get_main_dev(xwl_screen);
 | 
						|
 | 
						|
        return xwl_get_modifiers_for_device(&xwl_screen->default_feedback, main_dev,
 | 
						|
                                            format, num_modifiers, modifiers,
 | 
						|
                                            NULL);
 | 
						|
    } else {
 | 
						|
        return xwl_get_modifiers_for_format(xwl_screen->formats, xwl_screen->num_formats,
 | 
						|
                                            format, num_modifiers, modifiers);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_get_drawable_modifiers_and_scanout(DrawablePtr drawable,
 | 
						|
                                              uint32_t format,
 | 
						|
                                              uint32_t *num_modifiers,
 | 
						|
                                              uint64_t **modifiers,
 | 
						|
                                              Bool *supports_scanout)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(drawable->pScreen);
 | 
						|
    struct xwl_window *xwl_window;
 | 
						|
    drmDevice *main_dev;
 | 
						|
 | 
						|
    *num_modifiers = 0;
 | 
						|
    *modifiers = NULL;
 | 
						|
    if (supports_scanout)
 | 
						|
        *supports_scanout = FALSE;
 | 
						|
 | 
						|
    /* We can only return per-drawable modifiers if the compositor supports feedback */
 | 
						|
    if (xwl_screen->dmabuf_protocol_version < 4)
 | 
						|
        return TRUE;
 | 
						|
 | 
						|
    if (drawable->type != DRAWABLE_WINDOW || !xwl_screen->dmabuf)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    xwl_window = xwl_window_from_window((WindowPtr)drawable);
 | 
						|
 | 
						|
    /* couldn't find drawable for window */
 | 
						|
    if (!xwl_window)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    main_dev = xwl_screen_get_main_dev(xwl_screen);
 | 
						|
 | 
						|
    return xwl_get_modifiers_for_device(&xwl_window->feedback, main_dev,
 | 
						|
                                        format, num_modifiers, modifiers,
 | 
						|
                                        supports_scanout);
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_get_drawable_modifiers(DrawablePtr drawable, uint32_t format,
 | 
						|
                                  uint32_t *num_modifiers, uint64_t **modifiers)
 | 
						|
{
 | 
						|
    return xwl_glamor_get_drawable_modifiers_and_scanout(drawable,
 | 
						|
                                                         format, num_modifiers,
 | 
						|
                                                         modifiers, NULL);
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_handle_format(void *data, struct zwp_linux_dmabuf_v1 *dmabuf,
 | 
						|
                         uint32_t format)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_add_format_and_mod_to_list(struct xwl_format **formats,
 | 
						|
                               uint32_t *num_formats,
 | 
						|
                               uint32_t format,
 | 
						|
                               uint64_t modifier)
 | 
						|
{
 | 
						|
    struct xwl_format *xwl_format = NULL;
 | 
						|
    int i;
 | 
						|
 | 
						|
    for (i = 0; i < *num_formats; i++) {
 | 
						|
        if ((*formats)[i].format == format) {
 | 
						|
            xwl_format = &(*formats)[i];
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (xwl_format == NULL) {
 | 
						|
        (*num_formats)++;
 | 
						|
        *formats = xnfrealloc(*formats, *num_formats * sizeof(*xwl_format));
 | 
						|
        xwl_format = &(*formats)[*num_formats - 1];
 | 
						|
        xwl_format->format = format;
 | 
						|
        xwl_format->num_modifiers = 0;
 | 
						|
        xwl_format->modifiers = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    for (i = 0; i < xwl_format->num_modifiers; i++) {
 | 
						|
        /* don't add it if the modifier already exists */
 | 
						|
        if (xwl_format->modifiers[i] == modifier)
 | 
						|
            return;
 | 
						|
    }
 | 
						|
 | 
						|
    xwl_format->num_modifiers++;
 | 
						|
    xwl_format->modifiers = xnfrealloc(xwl_format->modifiers,
 | 
						|
                                       xwl_format->num_modifiers * sizeof(uint64_t));
 | 
						|
    xwl_format->modifiers[xwl_format->num_modifiers - 1]  = modifier;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_handle_modifier(void *data, struct zwp_linux_dmabuf_v1 *dmabuf,
 | 
						|
                           uint32_t format, uint32_t modifier_hi,
 | 
						|
                           uint32_t modifier_lo)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = data;
 | 
						|
 | 
						|
    xwl_add_format_and_mod_to_list(&xwl_screen->formats, &xwl_screen->num_formats,
 | 
						|
                                   format,
 | 
						|
                                   ((uint64_t)modifier_hi << 32 | (uint64_t)modifier_lo));
 | 
						|
}
 | 
						|
 | 
						|
static const struct zwp_linux_dmabuf_v1_listener xwl_dmabuf_listener = {
 | 
						|
    .format = xwl_dmabuf_handle_format,
 | 
						|
    .modifier = xwl_dmabuf_handle_modifier
 | 
						|
};
 | 
						|
 | 
						|
/*
 | 
						|
 * We need to check if the compositor is resending all of the tranche
 | 
						|
 * information. Each tranche event will call this method to see
 | 
						|
 * if the existing format info should be cleared before refilling.
 | 
						|
 */
 | 
						|
static void
 | 
						|
xwl_check_reset_tranche_info(struct xwl_dmabuf_feedback *xwl_feedback)
 | 
						|
{
 | 
						|
    if (!xwl_feedback->feedback_done)
 | 
						|
        return;
 | 
						|
 | 
						|
    xwl_feedback->feedback_done = false;
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_clear_dev_formats(xwl_feedback);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_feedback_main_device(void *data,
 | 
						|
                                struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                struct wl_array *dev)
 | 
						|
{
 | 
						|
    struct xwl_dmabuf_feedback *xwl_feedback = data;
 | 
						|
    dev_t devid;
 | 
						|
 | 
						|
    xwl_check_reset_tranche_info(xwl_feedback);
 | 
						|
 | 
						|
    assert(dev->size == sizeof(dev_t));
 | 
						|
    memcpy(&devid, dev->data, sizeof(dev_t));
 | 
						|
 | 
						|
    if (drmGetDeviceFromDevId(devid, 0, &xwl_feedback->main_dev) != 0)
 | 
						|
        ErrorF("linux_dmabuf_feedback.main_device: Failed to fetch DRM device\n");
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_feedback_tranche_target_device(void *data,
 | 
						|
                                          struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                          struct wl_array *dev)
 | 
						|
{
 | 
						|
    struct xwl_dmabuf_feedback *xwl_feedback = data;
 | 
						|
    dev_t devid;
 | 
						|
 | 
						|
    xwl_check_reset_tranche_info(xwl_feedback);
 | 
						|
 | 
						|
    assert(dev->size == sizeof(dev_t));
 | 
						|
    memcpy(&devid, dev->data, sizeof(dev_t));
 | 
						|
 | 
						|
    if (drmGetDeviceFromDevId(devid, 0, &xwl_feedback->tmp_tranche.drm_dev) != 0)
 | 
						|
        ErrorF("linux_dmabuf_feedback.tranche_target_device: Failed to fetch DRM device\n");
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_feedback_tranche_flags(void *data,
 | 
						|
                                  struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                  uint32_t flags)
 | 
						|
{
 | 
						|
    struct xwl_dmabuf_feedback *xwl_feedback = data;
 | 
						|
 | 
						|
    xwl_check_reset_tranche_info(xwl_feedback);
 | 
						|
 | 
						|
    if (flags & ZWP_LINUX_DMABUF_FEEDBACK_V1_TRANCHE_FLAGS_SCANOUT)
 | 
						|
        xwl_feedback->tmp_tranche.supports_scanout = true;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_feedback_tranche_formats(void *data,
 | 
						|
                                    struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                    struct wl_array *indices)
 | 
						|
{
 | 
						|
    struct xwl_dmabuf_feedback *xwl_feedback = data;
 | 
						|
    struct xwl_device_formats *tranche = &xwl_feedback->tmp_tranche;
 | 
						|
    uint16_t *index;
 | 
						|
 | 
						|
    xwl_check_reset_tranche_info(xwl_feedback);
 | 
						|
 | 
						|
    wl_array_for_each(index, indices) {
 | 
						|
        if (*index >= xwl_feedback->format_table.len) {
 | 
						|
            ErrorF("linux_dmabuf_feedback.tranche_formats: Index given to us by the compositor"
 | 
						|
                   " is too large to fit in the format table\n");
 | 
						|
            continue;
 | 
						|
        }
 | 
						|
 | 
						|
        /* Look up this format/mod in the format table */
 | 
						|
        struct xwl_format_table_entry *entry = &xwl_feedback->format_table.entry[*index];
 | 
						|
 | 
						|
        /* Add it to the in-progress tranche */
 | 
						|
        xwl_add_format_and_mod_to_list(&tranche->formats, &tranche->num_formats,
 | 
						|
                                       entry->format,
 | 
						|
                                       entry->modifier);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_append_to_tranche(struct xwl_device_formats *dst, struct xwl_device_formats *src)
 | 
						|
{
 | 
						|
    struct xwl_format *format;
 | 
						|
 | 
						|
    for (int i = 0; i < src->num_formats; i++) {
 | 
						|
        format = &src->formats[i];
 | 
						|
 | 
						|
        for (int j = 0; j < format->num_modifiers; j++)
 | 
						|
            xwl_add_format_and_mod_to_list(&dst->formats, &dst->num_formats,
 | 
						|
                                           format->format,
 | 
						|
                                           format->modifiers[j]);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_feedback_tranche_done(void *data,
 | 
						|
                                 struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback)
 | 
						|
{
 | 
						|
    struct xwl_dmabuf_feedback *xwl_feedback = data;
 | 
						|
    struct xwl_device_formats *tranche;
 | 
						|
    int appended = false;
 | 
						|
 | 
						|
    /*
 | 
						|
     * No need to call xwl_check_reset_tranche_info, the other events should have been
 | 
						|
     * triggered first
 | 
						|
     */
 | 
						|
 | 
						|
    if (xwl_feedback->tmp_tranche.drm_dev == NULL) {
 | 
						|
        xwl_device_formats_destroy(&xwl_feedback->tmp_tranche);
 | 
						|
        goto out;
 | 
						|
    }
 | 
						|
 | 
						|
    /*
 | 
						|
     * First check if there is an existing tranche for this device+flags combo. We
 | 
						|
     * will combine it with this tranche, since we can only send one modifier list
 | 
						|
     * in DRI3 but the compositor may report multiple tranches per device (KDE
 | 
						|
     * does this)
 | 
						|
     */
 | 
						|
    for (int i = 0; i < xwl_feedback->dev_formats_len; i++) {
 | 
						|
        tranche = &xwl_feedback->dev_formats[i];
 | 
						|
        if (tranche->drm_dev == xwl_feedback->tmp_tranche.drm_dev &&
 | 
						|
            tranche->supports_scanout == xwl_feedback->tmp_tranche.supports_scanout) {
 | 
						|
            appended = true;
 | 
						|
 | 
						|
            /* Add all format/mods to this tranche */
 | 
						|
            xwl_append_to_tranche(tranche, &xwl_feedback->tmp_tranche);
 | 
						|
 | 
						|
            /* Now free our temp tranche's allocations */
 | 
						|
            xwl_device_formats_destroy(&xwl_feedback->tmp_tranche);
 | 
						|
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!appended) {
 | 
						|
        xwl_feedback->dev_formats_len++;
 | 
						|
        xwl_feedback->dev_formats = xnfrealloc(xwl_feedback->dev_formats,
 | 
						|
                                               sizeof(struct xwl_device_formats) *
 | 
						|
                                               xwl_feedback->dev_formats_len);
 | 
						|
 | 
						|
        /* copy the temporary tranche into the official array */
 | 
						|
        memcpy(&xwl_feedback->dev_formats[xwl_feedback->dev_formats_len - 1],
 | 
						|
               &xwl_feedback->tmp_tranche,
 | 
						|
               sizeof(struct xwl_device_formats));
 | 
						|
    }
 | 
						|
 | 
						|
out:
 | 
						|
    /* reset the tranche */
 | 
						|
    memset(&xwl_feedback->tmp_tranche, 0, sizeof(struct xwl_device_formats));
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_feedback_done(void *data, struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback)
 | 
						|
{
 | 
						|
    struct xwl_dmabuf_feedback *xwl_feedback = data;
 | 
						|
 | 
						|
    xwl_feedback->feedback_done = true;
 | 
						|
    xwl_feedback->unprocessed_feedback_pending = true;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_dmabuf_feedback_format_table(void *data,
 | 
						|
                                 struct zwp_linux_dmabuf_feedback_v1 *zwp_linux_dmabuf_feedback_v1,
 | 
						|
                                 int32_t fd, uint32_t size)
 | 
						|
{
 | 
						|
    struct xwl_dmabuf_feedback *xwl_feedback = data;
 | 
						|
    /* Unmap the old table */
 | 
						|
    if (xwl_feedback->format_table.entry) {
 | 
						|
        munmap(xwl_feedback->format_table.entry,
 | 
						|
               xwl_feedback->format_table.len * sizeof(struct xwl_format_table_entry));
 | 
						|
    }
 | 
						|
 | 
						|
    assert(size % sizeof(struct xwl_format_table_entry) == 0);
 | 
						|
    xwl_feedback->format_table.len = size / sizeof(struct xwl_format_table_entry);
 | 
						|
    xwl_feedback->format_table.entry = mmap(NULL, size, PROT_READ, MAP_PRIVATE, fd, 0);
 | 
						|
    close(fd);
 | 
						|
 | 
						|
    if (xwl_feedback->format_table.entry == MAP_FAILED) {
 | 
						|
        ErrorF("linux_dmabuf_feedback.format_table: Could not map the format"
 | 
						|
               " table: Compositor bug or out of resources\n");
 | 
						|
        xwl_feedback->format_table.len = 0;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static const struct zwp_linux_dmabuf_feedback_v1_listener xwl_dmabuf_feedback_listener = {
 | 
						|
    .done = xwl_dmabuf_feedback_done,
 | 
						|
    .format_table = xwl_dmabuf_feedback_format_table,
 | 
						|
    .main_device = xwl_dmabuf_feedback_main_device,
 | 
						|
    .tranche_done = xwl_dmabuf_feedback_tranche_done,
 | 
						|
    .tranche_target_device = xwl_dmabuf_feedback_tranche_target_device,
 | 
						|
    .tranche_formats = xwl_dmabuf_feedback_tranche_formats,
 | 
						|
    .tranche_flags = xwl_dmabuf_feedback_tranche_flags,
 | 
						|
};
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_screen_set_dmabuf_interface(struct xwl_screen *xwl_screen,
 | 
						|
                                uint32_t id, uint32_t version)
 | 
						|
{
 | 
						|
    /* We either support versions 3 or 4. 4 is needed for dmabuf feedback */
 | 
						|
    int supported_version = version >= 4 ? 4 : 3;
 | 
						|
 | 
						|
    if (version < 3)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    xwl_screen->dmabuf =
 | 
						|
        wl_registry_bind(xwl_screen->registry, id, &zwp_linux_dmabuf_v1_interface, supported_version);
 | 
						|
    xwl_screen->dmabuf_protocol_version = supported_version;
 | 
						|
    zwp_linux_dmabuf_v1_add_listener(xwl_screen->dmabuf, &xwl_dmabuf_listener, xwl_screen);
 | 
						|
 | 
						|
    /* If the compositor supports it, request the default feedback hints */
 | 
						|
    if (version >= 4) {
 | 
						|
        xwl_screen->default_feedback.dmabuf_feedback =
 | 
						|
            zwp_linux_dmabuf_v1_get_default_feedback(xwl_screen->dmabuf);
 | 
						|
        if (!xwl_screen->default_feedback.dmabuf_feedback)
 | 
						|
            return FALSE;
 | 
						|
 | 
						|
        zwp_linux_dmabuf_feedback_v1_add_listener(xwl_screen->default_feedback.dmabuf_feedback,
 | 
						|
                                                  &xwl_dmabuf_feedback_listener,
 | 
						|
                                                  &xwl_screen->default_feedback);
 | 
						|
    }
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_dmabuf_feedback_main_device(void *data,
 | 
						|
                                       struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                       struct wl_array *dev)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_main_device(&xwl_window->feedback, dmabuf_feedback, dev);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_dmabuf_feedback_tranche_target_device(void *data,
 | 
						|
                                                 struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                                 struct wl_array *dev)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_tranche_target_device(&xwl_window->feedback, dmabuf_feedback, dev);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_dmabuf_feedback_tranche_flags(void *data,
 | 
						|
                                         struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                         uint32_t flags)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_tranche_flags(&xwl_window->feedback, dmabuf_feedback, flags);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_dmabuf_feedback_tranche_formats(void *data,
 | 
						|
                                           struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                           struct wl_array *indices)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_tranche_formats(&xwl_window->feedback, dmabuf_feedback, indices);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_dmabuf_feedback_tranche_done(void *data,
 | 
						|
                                        struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_tranche_done(&xwl_window->feedback, dmabuf_feedback);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_dmabuf_feedback_done(void *data,
 | 
						|
                                struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
    uint32_t format = wl_drm_format_for_depth(xwl_window->window->drawable.depth);
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_done(&xwl_window->feedback, dmabuf_feedback);
 | 
						|
 | 
						|
    xwl_window->has_implicit_scanout_support =
 | 
						|
        xwl_feedback_is_modifier_supported(&xwl_window->feedback, format,
 | 
						|
                                           DRM_FORMAT_MOD_INVALID, TRUE);
 | 
						|
    DebugF("XWAYLAND: Window 0x%x can%s get implicit scanout support\n",
 | 
						|
            xwl_window->window->drawable.id,
 | 
						|
            xwl_window->has_implicit_scanout_support ? "" : "not");
 | 
						|
 | 
						|
    /* If the linux-dmabuf v4 per-surface feedback changed, make sure the
 | 
						|
     * window buffers get re-created with appropriate parameters.
 | 
						|
     */
 | 
						|
    xwl_window_buffers_dispose(xwl_window);
 | 
						|
    xwl_window_recycle_pixmap(xwl_window);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_window_dmabuf_feedback_format_table(void *data,
 | 
						|
                                        struct zwp_linux_dmabuf_feedback_v1 *dmabuf_feedback,
 | 
						|
                                        int32_t fd, uint32_t size)
 | 
						|
{
 | 
						|
    struct xwl_window *xwl_window = data;
 | 
						|
 | 
						|
    xwl_dmabuf_feedback_format_table(&xwl_window->feedback, dmabuf_feedback, fd, size);
 | 
						|
}
 | 
						|
 | 
						|
static const struct zwp_linux_dmabuf_feedback_v1_listener xwl_window_dmabuf_feedback_listener = {
 | 
						|
    .done = xwl_window_dmabuf_feedback_done,
 | 
						|
    .format_table = xwl_window_dmabuf_feedback_format_table,
 | 
						|
    .main_device = xwl_window_dmabuf_feedback_main_device,
 | 
						|
    .tranche_done = xwl_window_dmabuf_feedback_tranche_done,
 | 
						|
    .tranche_target_device = xwl_window_dmabuf_feedback_tranche_target_device,
 | 
						|
    .tranche_formats = xwl_window_dmabuf_feedback_tranche_formats,
 | 
						|
    .tranche_flags = xwl_window_dmabuf_feedback_tranche_flags,
 | 
						|
};
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_dmabuf_setup_feedback_for_window(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    xwl_window->feedback.dmabuf_feedback =
 | 
						|
        zwp_linux_dmabuf_v1_get_surface_feedback(xwl_screen->dmabuf, xwl_window->surface);
 | 
						|
 | 
						|
    if (!xwl_window->feedback.dmabuf_feedback)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    zwp_linux_dmabuf_feedback_v1_add_listener(xwl_window->feedback.dmabuf_feedback,
 | 
						|
                                              &xwl_window_dmabuf_feedback_listener,
 | 
						|
                                              xwl_window);
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_glamor_init_wl_registry(struct xwl_screen *xwl_screen,
 | 
						|
                            struct wl_registry *registry,
 | 
						|
                            uint32_t id, const char *interface,
 | 
						|
                            uint32_t version)
 | 
						|
{
 | 
						|
    if (xwl_screen->gbm_backend.is_available &&
 | 
						|
        xwl_screen->gbm_backend.init_wl_registry(xwl_screen,
 | 
						|
                                                 registry,
 | 
						|
                                                 id,
 | 
						|
                                                 interface,
 | 
						|
                                                 version)) {
 | 
						|
        /* no-op */
 | 
						|
    } else if (xwl_screen->eglstream_backend.is_available &&
 | 
						|
               xwl_screen->eglstream_backend.init_wl_registry(xwl_screen,
 | 
						|
                                                              registry,
 | 
						|
                                                              id,
 | 
						|
                                                              interface,
 | 
						|
                                                              version)) {
 | 
						|
        /* no-op */
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_has_wl_interfaces(struct xwl_screen *xwl_screen,
 | 
						|
                            struct xwl_egl_backend *xwl_egl_backend)
 | 
						|
{
 | 
						|
    return xwl_egl_backend->has_wl_interfaces(xwl_screen);
 | 
						|
}
 | 
						|
 | 
						|
struct wl_buffer *
 | 
						|
xwl_glamor_pixmap_get_wl_buffer(PixmapPtr pixmap)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(pixmap->drawable.pScreen);
 | 
						|
 | 
						|
    if (xwl_screen->egl_backend->get_wl_buffer_for_pixmap)
 | 
						|
        return xwl_screen->egl_backend->get_wl_buffer_for_pixmap(pixmap);
 | 
						|
 | 
						|
    return NULL;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_post_damage(struct xwl_window *xwl_window,
 | 
						|
                       PixmapPtr pixmap, RegionPtr region)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    if (xwl_screen->egl_backend->post_damage)
 | 
						|
        return xwl_screen->egl_backend->post_damage(xwl_window, pixmap, region);
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_allow_commits(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    if (xwl_screen->egl_backend->allow_commits)
 | 
						|
        return xwl_screen->egl_backend->allow_commits(xwl_window);
 | 
						|
    else
 | 
						|
        return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_avoid_implicit_redirect(WindowPtr window)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    WindowOptPtr parent_optional;
 | 
						|
    VisualPtr parent_visual = NULL;
 | 
						|
    VisualPtr window_visual = NULL;
 | 
						|
    DepthPtr depth32 = NULL;
 | 
						|
    int i;
 | 
						|
 | 
						|
    if (!window->optional)
 | 
						|
        return;
 | 
						|
 | 
						|
    parent_optional = FindWindowWithOptional(window)->optional;
 | 
						|
    if (window->optional == parent_optional ||
 | 
						|
        window->optional->visual == parent_optional->visual ||
 | 
						|
        CompositeIsImplicitRedirectException(screen, parent_optional->visual,
 | 
						|
                                             window->optional->visual))
 | 
						|
        return;
 | 
						|
 | 
						|
    for (i = 0; i < screen->numDepths; i++) {
 | 
						|
        if (screen->allowedDepths[i].depth == 32) {
 | 
						|
            depth32 = &screen->allowedDepths[i];
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!depth32)
 | 
						|
        return;
 | 
						|
 | 
						|
    for (i = 0; i < depth32->numVids; i++) {
 | 
						|
        XID argb_vid = depth32->vids[i];
 | 
						|
 | 
						|
        if (argb_vid != parent_optional->visual)
 | 
						|
            continue;
 | 
						|
 | 
						|
        if (!compIsAlternateVisual(screen, argb_vid))
 | 
						|
            break;
 | 
						|
 | 
						|
        for (i = 0; i < screen->numVisuals; i++) {
 | 
						|
            if (screen->visuals[i].vid == argb_vid) {
 | 
						|
                parent_visual = &screen->visuals[i];
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!parent_visual)
 | 
						|
        return;
 | 
						|
 | 
						|
    for (i = 0; i < screen->numVisuals; i++) {
 | 
						|
        if (screen->visuals[i].vid == window->optional->visual) {
 | 
						|
            window_visual = &screen->visuals[i];
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if ((window_visual->class != TrueColor &&
 | 
						|
         window_visual->class != DirectColor) ||
 | 
						|
        window_visual->redMask != parent_visual->redMask ||
 | 
						|
        window_visual->greenMask != parent_visual->greenMask ||
 | 
						|
        window_visual->blueMask != parent_visual->blueMask ||
 | 
						|
        window_visual->offsetRed != parent_visual->offsetRed ||
 | 
						|
        window_visual->offsetGreen != parent_visual->offsetGreen ||
 | 
						|
        window_visual->offsetBlue != parent_visual->offsetBlue)
 | 
						|
        return;
 | 
						|
 | 
						|
    CompositeRegisterImplicitRedirectionException(screen, parent_visual->vid, window_visual->vid);
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_glamor_create_window(WindowPtr window)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
    Bool ret;
 | 
						|
 | 
						|
    if (window->parent)
 | 
						|
        xwl_avoid_implicit_redirect(window);
 | 
						|
 | 
						|
    screen->CreateWindow = xwl_screen->CreateWindow;
 | 
						|
    ret = (*screen->CreateWindow) (window);
 | 
						|
    xwl_screen->CreateWindow = screen->CreateWindow;
 | 
						|
    screen->CreateWindow = xwl_glamor_create_window;
 | 
						|
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
xwl_glamor_reparent_window(WindowPtr window, WindowPtr old_parent)
 | 
						|
{
 | 
						|
    ScreenPtr screen = window->drawable.pScreen;
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
 | 
						|
    xwl_avoid_implicit_redirect(window);
 | 
						|
 | 
						|
    screen->ReparentWindow = xwl_screen->ReparentWindow;
 | 
						|
    (*screen->ReparentWindow) (window, old_parent);
 | 
						|
    xwl_screen->ReparentWindow = screen->ReparentWindow;
 | 
						|
    screen->ReparentWindow = xwl_glamor_reparent_window;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_glamor_create_screen_resources(ScreenPtr screen)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_screen_get(screen);
 | 
						|
    int ret;
 | 
						|
 | 
						|
    screen->CreateScreenResources = xwl_screen->CreateScreenResources;
 | 
						|
    ret = (*screen->CreateScreenResources) (screen);
 | 
						|
    xwl_screen->CreateScreenResources = screen->CreateScreenResources;
 | 
						|
    screen->CreateScreenResources = xwl_glamor_create_screen_resources;
 | 
						|
 | 
						|
    if (!ret)
 | 
						|
        return ret;
 | 
						|
 | 
						|
    xwl_screen->CreateWindow = screen->CreateWindow;
 | 
						|
    screen->CreateWindow = xwl_glamor_create_window;
 | 
						|
    xwl_screen->ReparentWindow = screen->ReparentWindow;
 | 
						|
    screen->ReparentWindow = xwl_glamor_reparent_window;
 | 
						|
 | 
						|
    if (xwl_screen->rootless) {
 | 
						|
        screen->devPrivate =
 | 
						|
            fbCreatePixmap(screen, 0, 0, screen->rootDepth, 0);
 | 
						|
    }
 | 
						|
    else {
 | 
						|
        screen->devPrivate = screen->CreatePixmap(
 | 
						|
            screen, screen->width, screen->height, screen->rootDepth,
 | 
						|
            CREATE_PIXMAP_USAGE_BACKING_PIXMAP);
 | 
						|
    }
 | 
						|
 | 
						|
    SetRootClip(screen, xwl_screen->root_clip_mode);
 | 
						|
 | 
						|
    return screen->devPrivate != NULL;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
glamor_egl_fd_name_from_pixmap(ScreenPtr screen,
 | 
						|
                               PixmapPtr pixmap,
 | 
						|
                               CARD16 *stride, CARD32 *size)
 | 
						|
{
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_needs_buffer_flush(struct xwl_screen *xwl_screen)
 | 
						|
{
 | 
						|
    if (!xwl_screen->glamor || !xwl_screen->egl_backend)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    return (xwl_screen->egl_backend->backend_flags &
 | 
						|
                XWL_EGL_BACKEND_NEEDS_BUFFER_FLUSH);
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_needs_n_buffering(struct xwl_screen *xwl_screen)
 | 
						|
{
 | 
						|
    /* wl_shm benefits from n-buffering */
 | 
						|
    if (!xwl_screen->glamor || !xwl_screen->egl_backend)
 | 
						|
        return TRUE;
 | 
						|
 | 
						|
    return (xwl_screen->egl_backend->backend_flags &
 | 
						|
                XWL_EGL_BACKEND_NEEDS_N_BUFFERING);
 | 
						|
}
 | 
						|
 | 
						|
PixmapPtr
 | 
						|
xwl_glamor_create_pixmap_for_window(struct xwl_window *xwl_window)
 | 
						|
{
 | 
						|
    struct xwl_screen *xwl_screen = xwl_window->xwl_screen;
 | 
						|
 | 
						|
    if (!xwl_screen->glamor || !xwl_screen->egl_backend)
 | 
						|
        return NullPixmap;
 | 
						|
 | 
						|
    if (xwl_screen->egl_backend->create_pixmap_for_window)
 | 
						|
        return xwl_screen->egl_backend->create_pixmap_for_window(xwl_window);
 | 
						|
    else
 | 
						|
        return NullPixmap;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_glamor_init_backends(struct xwl_screen *xwl_screen, Bool use_eglstream)
 | 
						|
{
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    xwl_glamor_init_gbm(xwl_screen);
 | 
						|
    if (!xwl_screen->gbm_backend.is_available && !use_eglstream)
 | 
						|
        ErrorF("Xwayland glamor: GBM backend (default) is not available\n");
 | 
						|
#endif
 | 
						|
#ifdef XWL_HAS_EGLSTREAM
 | 
						|
    xwl_glamor_init_eglstream(xwl_screen);
 | 
						|
    if (!xwl_screen->eglstream_backend.is_available && use_eglstream)
 | 
						|
        ErrorF("Xwayland glamor: EGLStream backend requested but not available\n");
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_glamor_select_gbm_backend(struct xwl_screen *xwl_screen)
 | 
						|
{
 | 
						|
#ifdef GLAMOR_HAS_GBM
 | 
						|
    if (xwl_screen->gbm_backend.is_available &&
 | 
						|
        xwl_glamor_has_wl_interfaces(xwl_screen, &xwl_screen->gbm_backend)) {
 | 
						|
        xwl_screen->egl_backend = &xwl_screen->gbm_backend;
 | 
						|
        LogMessageVerb(X_INFO, 3, "glamor: Using GBM backend\n");
 | 
						|
        return TRUE;
 | 
						|
    }
 | 
						|
    else
 | 
						|
        LogMessageVerb(X_INFO, 3,
 | 
						|
                       "Missing Wayland requirements for glamor GBM backend\n");
 | 
						|
#endif
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
static Bool
 | 
						|
xwl_glamor_select_eglstream_backend(struct xwl_screen *xwl_screen)
 | 
						|
{
 | 
						|
#ifdef XWL_HAS_EGLSTREAM
 | 
						|
    if (xwl_screen->eglstream_backend.is_available &&
 | 
						|
        xwl_glamor_has_wl_interfaces(xwl_screen, &xwl_screen->eglstream_backend)) {
 | 
						|
        xwl_screen->egl_backend = &xwl_screen->eglstream_backend;
 | 
						|
        LogMessageVerb(X_INFO, 3, "glamor: Using EGLStream backend\n");
 | 
						|
        return TRUE;
 | 
						|
    }
 | 
						|
    else
 | 
						|
        LogMessageVerb(X_INFO, 3,
 | 
						|
                       "Missing Wayland requirements for glamor EGLStream backend\n");
 | 
						|
#endif
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
xwl_glamor_select_backend(struct xwl_screen *xwl_screen, Bool use_eglstream)
 | 
						|
{
 | 
						|
    if (!xwl_glamor_select_eglstream_backend(xwl_screen)) {
 | 
						|
        if (!use_eglstream)
 | 
						|
            xwl_glamor_select_gbm_backend(xwl_screen);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
Bool
 | 
						|
xwl_glamor_init(struct xwl_screen *xwl_screen)
 | 
						|
{
 | 
						|
    ScreenPtr screen = xwl_screen->screen;
 | 
						|
    const char *no_glamor_env;
 | 
						|
 | 
						|
    no_glamor_env = getenv("XWAYLAND_NO_GLAMOR");
 | 
						|
    if (no_glamor_env && *no_glamor_env != '0') {
 | 
						|
        ErrorF("Disabling glamor and dri3 support, XWAYLAND_NO_GLAMOR is set\n");
 | 
						|
        return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!xwl_screen->egl_backend->init_egl(xwl_screen)) {
 | 
						|
        ErrorF("EGL setup failed, disabling glamor\n");
 | 
						|
        return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!glamor_init(xwl_screen->screen, GLAMOR_USE_EGL_SCREEN)) {
 | 
						|
        ErrorF("Failed to initialize glamor\n");
 | 
						|
        return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!xwl_screen->egl_backend->init_screen(xwl_screen)) {
 | 
						|
        ErrorF("EGL backend init_screen() failed, disabling glamor\n");
 | 
						|
        return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
    xwl_screen->CreateScreenResources = screen->CreateScreenResources;
 | 
						|
    screen->CreateScreenResources = xwl_glamor_create_screen_resources;
 | 
						|
 | 
						|
#ifdef XV
 | 
						|
    if (!xwl_glamor_xv_init(screen))
 | 
						|
        ErrorF("Failed to initialize glamor Xv extension\n");
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef GLXEXT
 | 
						|
    GlxPushProvider(&glamor_provider);
 | 
						|
#endif
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 |