wayland: Implement stub presentation-time

The presentation-time protocol allows surfaces to get accurate
timestamps of when their contents were shown on screen.

This commit implements a stub version of the protocol which correctly
discards all presentation feedback objects (as if the surface contents
are never shown on screen). Subsequent commits will implement sending
the presented events to surfaces shown on screen.

Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1484>
This commit is contained in:
Ivan Molodetskikh 2020-10-07 12:02:41 +03:00 committed by Marge Bot
parent 2f4027200f
commit dccc60ec3e
7 changed files with 239 additions and 0 deletions

View File

@ -562,6 +562,8 @@ if have_wayland
'wayland/meta-wayland-pointer.h', 'wayland/meta-wayland-pointer.h',
'wayland/meta-wayland-popup.c', 'wayland/meta-wayland-popup.c',
'wayland/meta-wayland-popup.h', 'wayland/meta-wayland-popup.h',
'wayland/meta-wayland-presentation-time.c',
'wayland/meta-wayland-presentation-time-private.h',
'wayland/meta-wayland-private.h', 'wayland/meta-wayland-private.h',
'wayland/meta-wayland-region.c', 'wayland/meta-wayland-region.c',
'wayland/meta-wayland-region.h', 'wayland/meta-wayland-region.h',

View File

@ -0,0 +1,42 @@
/*
* presentation-time protocol
*
* Copyright (C) 2020 Ivan Molodetskikh <yalterz@gmail.com>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
*
*/
#ifndef META_WAYLAND_PRESENTATION_TIME_PRIVATE_H
#define META_WAYLAND_PRESENTATION_TIME_PRIVATE_H
#include <wayland-server.h>
#include "wayland/meta-wayland-types.h"
typedef struct _MetaWaylandPresentationFeedback
{
struct wl_list link;
struct wl_resource *resource;
MetaWaylandSurface *surface;
} MetaWaylandPresentationFeedback;
void meta_wayland_init_presentation_time (MetaWaylandCompositor *compositor);
void meta_wayland_presentation_feedback_discard (MetaWaylandPresentationFeedback *feedback);
#endif /* META_WAYLAND_PRESENTATION_TIME_PRIVATE_H */

View File

@ -0,0 +1,129 @@
/*
* presentation-time protocol
*
* Copyright (C) 2020 Ivan Molodetskikh <yalterz@gmail.com>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
*
*/
#include "config.h"
#include "meta-wayland-presentation-time-private.h"
#include <glib.h>
#include "wayland/meta-wayland-private.h"
#include "wayland/meta-wayland-surface.h"
#include "wayland/meta-wayland-versions.h"
#include "presentation-time-server-protocol.h"
static void
wp_presentation_feedback_destructor (struct wl_resource *resource)
{
MetaWaylandPresentationFeedback *feedback =
wl_resource_get_user_data (resource);
wl_list_remove (&feedback->link);
g_free (feedback);
}
static void
wp_presentation_destroy (struct wl_client *client,
struct wl_resource *resource)
{
wl_resource_destroy (resource);
}
static void
wp_presentation_feedback (struct wl_client *client,
struct wl_resource *resource,
struct wl_resource *surface_resource,
uint32_t callback_id)
{
MetaWaylandSurface *surface = wl_resource_get_user_data (surface_resource);
MetaWaylandSurfaceState *pending;
MetaWaylandPresentationFeedback *feedback;
feedback = g_new0 (MetaWaylandPresentationFeedback, 1);
wl_list_init (&feedback->link);
feedback->resource = wl_resource_create (client,
&wp_presentation_feedback_interface,
wl_resource_get_version (resource),
callback_id);
wl_resource_set_implementation (feedback->resource,
NULL,
feedback,
wp_presentation_feedback_destructor);
if (surface == NULL)
{
g_warn_if_reached ();
meta_wayland_presentation_feedback_discard (feedback);
return;
}
pending = meta_wayland_surface_get_pending_state (surface);
wl_list_insert (&pending->presentation_feedback_list, &feedback->link);
feedback->surface = surface;
}
static const struct wp_presentation_interface
meta_wayland_presentation_interface = {
wp_presentation_destroy,
wp_presentation_feedback,
};
static void
wp_presentation_bind (struct wl_client *client,
void *data,
uint32_t version,
uint32_t id)
{
struct wl_resource *resource;
resource = wl_resource_create (client,
&wp_presentation_interface,
version,
id);
wl_resource_set_implementation (resource,
&meta_wayland_presentation_interface,
NULL,
NULL);
/* Presentation timestamps in Mutter are guaranteed to be CLOCK_MONOTONIC. */
wp_presentation_send_clock_id (resource, CLOCK_MONOTONIC);
}
void
meta_wayland_init_presentation_time (MetaWaylandCompositor *compositor)
{
if (wl_global_create (compositor->wayland_display,
&wp_presentation_interface,
META_WP_PRESENTATION_VERSION,
NULL,
wp_presentation_bind) == NULL)
g_error ("Failed to register a global wp_presentation object");
}
void
meta_wayland_presentation_feedback_discard (MetaWaylandPresentationFeedback *feedback)
{
wp_presentation_feedback_send_discarded (feedback->resource);
wl_resource_destroy (feedback->resource);
}

View File

@ -46,6 +46,7 @@
#include "wayland/meta-wayland-legacy-xdg-shell.h" #include "wayland/meta-wayland-legacy-xdg-shell.h"
#include "wayland/meta-wayland-outputs.h" #include "wayland/meta-wayland-outputs.h"
#include "wayland/meta-wayland-pointer.h" #include "wayland/meta-wayland-pointer.h"
#include "wayland/meta-wayland-presentation-time-private.h"
#include "wayland/meta-wayland-private.h" #include "wayland/meta-wayland-private.h"
#include "wayland/meta-wayland-region.h" #include "wayland/meta-wayland-region.h"
#include "wayland/meta-wayland-seat.h" #include "wayland/meta-wayland-seat.h"
@ -466,6 +467,20 @@ meta_wayland_surface_state_set_default (MetaWaylandSurfaceState *state)
state->has_new_buffer_transform = FALSE; state->has_new_buffer_transform = FALSE;
state->has_new_viewport_src_rect = FALSE; state->has_new_viewport_src_rect = FALSE;
state->has_new_viewport_dst_size = FALSE; state->has_new_viewport_dst_size = FALSE;
wl_list_init (&state->presentation_feedback_list);
}
static void
meta_wayland_surface_state_discard_presentation_feedback (MetaWaylandSurfaceState *state)
{
while (!wl_list_empty (&state->presentation_feedback_list))
{
MetaWaylandPresentationFeedback *feedback =
wl_container_of (state->presentation_feedback_list.next, feedback, link);
meta_wayland_presentation_feedback_discard (feedback);
}
} }
static void static void
@ -483,6 +498,8 @@ meta_wayland_surface_state_clear (MetaWaylandSurfaceState *state)
wl_list_for_each_safe (cb, next, &state->frame_callback_list, link) wl_list_for_each_safe (cb, next, &state->frame_callback_list, link)
wl_resource_destroy (cb->resource); wl_resource_destroy (cb->resource);
meta_wayland_surface_state_discard_presentation_feedback (state);
} }
static void static void
@ -592,6 +609,10 @@ meta_wayland_surface_state_merge_into (MetaWaylandSurfaceState *from,
to); to);
} }
wl_list_insert_list (&to->presentation_feedback_list,
&from->presentation_feedback_list);
wl_list_init (&from->presentation_feedback_list);
meta_wayland_surface_state_reset (from); meta_wayland_surface_state_reset (from);
} }
@ -627,6 +648,19 @@ meta_wayland_surface_state_class_init (MetaWaylandSurfaceStateClass *klass)
G_TYPE_NONE, 0); G_TYPE_NONE, 0);
} }
static void
meta_wayland_surface_discard_presentation_feedback (MetaWaylandSurface *surface)
{
while (!wl_list_empty (&surface->presentation_time.feedback_list))
{
MetaWaylandPresentationFeedback *feedback =
wl_container_of (surface->presentation_time.feedback_list.next,
feedback, link);
meta_wayland_presentation_feedback_discard (feedback);
}
}
static void static void
meta_wayland_surface_apply_state (MetaWaylandSurface *surface, meta_wayland_surface_apply_state (MetaWaylandSurface *surface,
MetaWaylandSurfaceState *state) MetaWaylandSurfaceState *state)
@ -755,6 +789,16 @@ meta_wayland_surface_apply_state (MetaWaylandSurface *surface,
surface->input_region = NULL; surface->input_region = NULL;
} }
/*
* A new commit indicates a new content update, so any previous
* content update did not go on screen and needs to be discarded.
*/
meta_wayland_surface_discard_presentation_feedback (surface);
wl_list_insert_list (&surface->presentation_time.feedback_list,
&state->presentation_feedback_list);
wl_list_init (&state->presentation_feedback_list);
if (surface->role) if (surface->role)
{ {
meta_wayland_surface_role_apply_state (surface->role, state); meta_wayland_surface_role_apply_state (surface->role, state);
@ -870,6 +914,13 @@ meta_wayland_surface_commit (MetaWaylandSurface *surface)
MetaWaylandSurfaceState *cached_state; MetaWaylandSurfaceState *cached_state;
cached_state = meta_wayland_surface_ensure_cached_state (surface); cached_state = meta_wayland_surface_ensure_cached_state (surface);
/*
* A new commit indicates a new content update, so any previous
* cached content update did not go on screen and needs to be discarded.
*/
meta_wayland_surface_state_discard_presentation_feedback (cached_state);
meta_wayland_surface_state_merge_into (pending, cached_state); meta_wayland_surface_state_merge_into (pending, cached_state);
} }
else else
@ -1350,6 +1401,8 @@ wl_surface_destructor (struct wl_resource *resource)
link) link)
wl_resource_destroy (cb->resource); wl_resource_destroy (cb->resource);
meta_wayland_surface_discard_presentation_feedback (surface);
if (surface->resource) if (surface->resource)
wl_resource_set_user_data (surface->resource, NULL); wl_resource_set_user_data (surface->resource, NULL);
@ -1397,6 +1450,8 @@ meta_wayland_surface_create (MetaWaylandCompositor *compositor,
surface->outputs = g_hash_table_new (NULL, NULL); surface->outputs = g_hash_table_new (NULL, NULL);
surface->shortcut_inhibited_seats = g_hash_table_new (NULL, NULL); surface->shortcut_inhibited_seats = g_hash_table_new (NULL, NULL);
wl_list_init (&surface->presentation_time.feedback_list);
meta_wayland_compositor_notify_surface_id (compositor, id, surface); meta_wayland_compositor_notify_surface_id (compositor, id, surface);
return surface; return surface;

View File

@ -120,6 +120,9 @@ struct _MetaWaylandSurfaceState
gboolean has_new_viewport_dst_size; gboolean has_new_viewport_dst_size;
int viewport_dst_width; int viewport_dst_width;
int viewport_dst_height; int viewport_dst_height;
/* presentation-time */
struct wl_list presentation_feedback_list;
}; };
struct _MetaWaylandDragDestFuncs struct _MetaWaylandDragDestFuncs
@ -227,6 +230,11 @@ struct _MetaWaylandSurface
/* table of seats for which shortcuts are inhibited */ /* table of seats for which shortcuts are inhibited */
GHashTable *shortcut_inhibited_seats; GHashTable *shortcut_inhibited_seats;
/* presentation-time */
struct {
struct wl_list feedback_list;
} presentation_time;
}; };
void meta_wayland_shell_init (MetaWaylandCompositor *compositor); void meta_wayland_shell_init (MetaWaylandCompositor *compositor);

View File

@ -57,5 +57,6 @@
#define META_WP_VIEWPORTER_VERSION 1 #define META_WP_VIEWPORTER_VERSION 1
#define META_GTK_PRIMARY_SELECTION_VERSION 1 #define META_GTK_PRIMARY_SELECTION_VERSION 1
#define META_ZWP_PRIMARY_SELECTION_V1_VERSION 1 #define META_ZWP_PRIMARY_SELECTION_V1_VERSION 1
#define META_WP_PRESENTATION_VERSION 1
#endif #endif

View File

@ -39,6 +39,7 @@
#include "wayland/meta-wayland-inhibit-shortcuts-dialog.h" #include "wayland/meta-wayland-inhibit-shortcuts-dialog.h"
#include "wayland/meta-wayland-inhibit-shortcuts.h" #include "wayland/meta-wayland-inhibit-shortcuts.h"
#include "wayland/meta-wayland-outputs.h" #include "wayland/meta-wayland-outputs.h"
#include "wayland/meta-wayland-presentation-time-private.h"
#include "wayland/meta-wayland-private.h" #include "wayland/meta-wayland-private.h"
#include "wayland/meta-wayland-region.h" #include "wayland/meta-wayland-region.h"
#include "wayland/meta-wayland-seat.h" #include "wayland/meta-wayland-seat.h"
@ -444,6 +445,7 @@ meta_wayland_compositor_setup (MetaWaylandCompositor *compositor)
meta_wayland_surface_inhibit_shortcuts_dialog_init (); meta_wayland_surface_inhibit_shortcuts_dialog_init ();
meta_wayland_text_input_init (compositor); meta_wayland_text_input_init (compositor);
meta_wayland_gtk_text_input_init (compositor); meta_wayland_gtk_text_input_init (compositor);
meta_wayland_init_presentation_time (compositor);
/* Xwayland specific protocol, needs to be filtered out for all other clients */ /* Xwayland specific protocol, needs to be filtered out for all other clients */
if (meta_xwayland_grab_keyboard_init (compositor)) if (meta_xwayland_grab_keyboard_init (compositor))