[mutter] Split out the code that renders the code into a MetaCursorRenderer
- From: Jasper St. Pierre <jstpierre src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [mutter] Split out the code that renders the code into a MetaCursorRenderer
- Date: Tue, 22 Apr 2014 00:25:45 +0000 (UTC)
commit d189ddcc86f6aeb0ed25f00209a87d5734c528db
Author: Jasper St. Pierre <jstpierre mecheye net>
Date: Mon Apr 21 18:05:59 2014 -0400
Split out the code that renders the code into a MetaCursorRenderer
Right now, this is the same exact same mess it's always been, but
it will be fixed up soon with backend-specific renderers.
src/Makefile.am | 2 +
src/backends/meta-cursor-renderer.c | 393 ++++++++++++++++++++++++++++
src/backends/meta-cursor-renderer.h | 71 +++++
src/backends/meta-cursor-tracker-private.h | 13 +-
src/backends/meta-cursor-tracker.c | 332 +++--------------------
src/wayland/meta-wayland-stage.c | 2 +-
6 files changed, 511 insertions(+), 302 deletions(-)
---
diff --git a/src/Makefile.am b/src/Makefile.am
index 78a18be..09721c0 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -54,6 +54,8 @@ libmutter_la_SOURCES = \
backends/meta-cursor-private.h \
backends/meta-cursor-tracker.c \
backends/meta-cursor-tracker-private.h \
+ backends/meta-cursor-renderer.c \
+ backends/meta-cursor-renderer.h \
backends/meta-display-config-shared.h \
backends/meta-idle-monitor.c \
backends/meta-idle-monitor-private.h \
diff --git a/src/backends/meta-cursor-renderer.c b/src/backends/meta-cursor-renderer.c
new file mode 100644
index 0000000..d5aa09d
--- /dev/null
+++ b/src/backends/meta-cursor-renderer.c
@@ -0,0 +1,393 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
+
+/*
+ * Copyright (C) 2014 Red Hat
+ *
+ * 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.
+ *
+ * Written by:
+ * Jasper St. Pierre <jstpierre mecheye net>
+ */
+
+#include "config.h"
+
+#include "meta-cursor-renderer.h"
+#include "meta-cursor-private.h"
+
+#include <cogl/cogl.h>
+#include <cogl/cogl-wayland-server.h>
+#include <clutter/clutter.h>
+#include <gbm.h>
+
+#include "meta-monitor-manager.h"
+
+#include "wayland/meta-wayland-private.h"
+
+struct _MetaCursorRendererPrivate
+{
+ MetaScreen *screen;
+
+ gboolean has_hw_cursor;
+
+ int current_x, current_y;
+ MetaRectangle current_rect;
+ MetaRectangle previous_rect;
+ gboolean previous_is_valid;
+
+ CoglPipeline *pipeline;
+ int drm_fd;
+ struct gbm_device *gbm;
+
+ MetaCursorReference *displayed_cursor;
+};
+typedef struct _MetaCursorRendererPrivate MetaCursorRendererPrivate;
+
+G_DEFINE_TYPE_WITH_PRIVATE (MetaCursorRenderer, meta_cursor_renderer, G_TYPE_OBJECT);
+
+static void
+set_crtc_has_hw_cursor (MetaCursorRenderer *renderer,
+ MetaCRTC *crtc,
+ gboolean has)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ if (has)
+ {
+ MetaCursorReference *displayed_cursor = priv->displayed_cursor;
+ struct gbm_bo *bo;
+ union gbm_bo_handle handle;
+ int width, height;
+ int hot_x, hot_y;
+
+ bo = meta_cursor_reference_get_gbm_bo (displayed_cursor, &hot_x, &hot_y);
+
+ handle = gbm_bo_get_handle (bo);
+ width = gbm_bo_get_width (bo);
+ height = gbm_bo_get_height (bo);
+
+ drmModeSetCursor2 (priv->drm_fd, crtc->crtc_id, handle.u32,
+ width, height, hot_x, hot_y);
+ crtc->has_hw_cursor = TRUE;
+ }
+ else
+ {
+ drmModeSetCursor2 (priv->drm_fd, crtc->crtc_id, 0, 0, 0, 0, 0);
+ crtc->has_hw_cursor = FALSE;
+ }
+}
+
+static void
+on_monitors_changed (MetaMonitorManager *monitors,
+ MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+ MetaCRTC *crtcs;
+ unsigned int i, n_crtcs;
+
+ if (!priv->has_hw_cursor)
+ return;
+
+ /* Go through the new list of monitors, find out where the cursor is */
+ meta_monitor_manager_get_resources (monitors, NULL, NULL, &crtcs, &n_crtcs, NULL, NULL);
+
+ for (i = 0; i < n_crtcs; i++)
+ {
+ MetaRectangle *rect = &crtcs[i].rect;
+ gboolean has;
+
+ has = meta_rectangle_overlap (&priv->current_rect, rect);
+
+ /* Need to do it unconditionally here, our tracking is
+ wrong because we reloaded the CRTCs */
+ set_crtc_has_hw_cursor (renderer, &crtcs[i], has);
+ }
+}
+
+static void
+meta_cursor_renderer_finalize (GObject *object)
+{
+ MetaCursorRenderer *renderer = META_CURSOR_RENDERER (object);
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ if (priv->pipeline)
+ cogl_object_unref (priv->pipeline);
+ if (priv->gbm)
+ gbm_device_destroy (priv->gbm);
+
+ G_OBJECT_CLASS (meta_cursor_renderer_parent_class)->finalize (object);
+}
+
+static void
+meta_cursor_renderer_class_init (MetaCursorRendererClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->finalize = meta_cursor_renderer_finalize;
+}
+
+static void
+meta_cursor_renderer_init (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+ CoglContext *ctx = clutter_backend_get_cogl_context (clutter_get_default_backend ());
+ MetaMonitorManager *monitors;
+
+ monitors = meta_monitor_manager_get ();
+ g_signal_connect_object (monitors, "monitors-changed",
+ G_CALLBACK (on_monitors_changed), renderer, 0);
+
+ priv->pipeline = cogl_pipeline_new (ctx);
+
+#if defined(CLUTTER_WINDOWING_EGL)
+ if (clutter_check_windowing_backend (CLUTTER_WINDOWING_EGL))
+ {
+ CoglRenderer *cogl_renderer = cogl_display_get_renderer (cogl_context_get_display (ctx));
+ priv->drm_fd = cogl_kms_renderer_get_kms_fd (cogl_renderer);
+ priv->gbm = gbm_create_device (priv->drm_fd);
+ }
+#endif
+}
+
+static gboolean
+should_have_hw_cursor (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ if (priv->displayed_cursor)
+ return (meta_cursor_reference_get_gbm_bo (priv->displayed_cursor, NULL, NULL) != NULL);
+ else
+ return FALSE;
+}
+
+static void
+update_hw_cursor (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+ MetaMonitorManager *monitors;
+ MetaCRTC *crtcs;
+ unsigned int i, n_crtcs;
+ gboolean enabled;
+
+ enabled = should_have_hw_cursor (renderer);
+ priv->has_hw_cursor = enabled;
+
+ monitors = meta_monitor_manager_get ();
+ meta_monitor_manager_get_resources (monitors, NULL, NULL, &crtcs, &n_crtcs, NULL, NULL);
+
+ for (i = 0; i < n_crtcs; i++)
+ {
+ MetaRectangle *rect = &crtcs[i].rect;
+ gboolean has;
+
+ has = enabled && meta_rectangle_overlap (&priv->current_rect, rect);
+
+ if (has || crtcs[i].has_hw_cursor)
+ set_crtc_has_hw_cursor (renderer, &crtcs[i], has);
+ }
+}
+
+static void
+move_hw_cursor (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+ MetaMonitorManager *monitors;
+ MetaCRTC *crtcs;
+ unsigned int i, n_crtcs;
+
+ monitors = meta_monitor_manager_get ();
+ meta_monitor_manager_get_resources (monitors, NULL, NULL, &crtcs, &n_crtcs, NULL, NULL);
+
+ g_assert (priv->has_hw_cursor);
+
+ for (i = 0; i < n_crtcs; i++)
+ {
+ MetaRectangle *rect = &crtcs[i].rect;
+ gboolean has;
+
+ has = meta_rectangle_overlap (&priv->current_rect, rect);
+
+ if (has != crtcs[i].has_hw_cursor)
+ set_crtc_has_hw_cursor (renderer, &crtcs[i], has);
+ if (has)
+ drmModeMoveCursor (priv->drm_fd, crtcs[i].crtc_id,
+ priv->current_rect.x - rect->x,
+ priv->current_rect.y - rect->y);
+ }
+}
+
+static void
+queue_redraw (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+ MetaWaylandCompositor *compositor = meta_wayland_compositor_get_default ();
+ ClutterActor *stage = compositor->stage;
+ cairo_rectangle_int_t clip;
+
+ g_assert (meta_is_wayland_compositor ());
+
+ /* Clear the location the cursor was at before, if we need to. */
+ if (priv->previous_is_valid)
+ {
+ clip.x = priv->previous_rect.x;
+ clip.y = priv->previous_rect.y;
+ clip.width = priv->previous_rect.width;
+ clip.height = priv->previous_rect.height;
+ clutter_actor_queue_redraw_with_clip (stage, &clip);
+ priv->previous_is_valid = FALSE;
+ }
+
+ if (priv->has_hw_cursor || !priv->displayed_cursor)
+ return;
+
+ clip.x = priv->current_rect.x;
+ clip.y = priv->current_rect.y;
+ clip.width = priv->current_rect.width;
+ clip.height = priv->current_rect.height;
+ clutter_actor_queue_redraw_with_clip (stage, &clip);
+}
+
+static void
+update_cursor (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ if (priv->displayed_cursor)
+ {
+ CoglTexture *texture;
+ int hot_x, hot_y;
+
+ texture = meta_cursor_reference_get_cogl_texture (priv->displayed_cursor, &hot_x, &hot_y);
+
+ priv->current_rect.x = priv->current_x - hot_x;
+ priv->current_rect.y = priv->current_y - hot_y;
+ priv->current_rect.width = cogl_texture_get_width (COGL_TEXTURE (texture));
+ priv->current_rect.height = cogl_texture_get_height (COGL_TEXTURE (texture));
+ }
+ else
+ {
+ priv->current_rect.x = 0;
+ priv->current_rect.y = 0;
+ priv->current_rect.width = 0;
+ priv->current_rect.height = 0;
+ }
+
+ if (meta_is_wayland_compositor ())
+ {
+ if (priv->has_hw_cursor)
+ move_hw_cursor (renderer);
+ else
+ queue_redraw (renderer);
+ }
+}
+
+static void
+update_pipeline (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ if (meta_is_wayland_compositor ())
+ {
+ if (priv->displayed_cursor)
+ {
+ CoglTexture *texture = meta_cursor_reference_get_cogl_texture (priv->displayed_cursor, NULL, NULL);
+ cogl_pipeline_set_layer_texture (priv->pipeline, 0, texture);
+ }
+ else
+ cogl_pipeline_set_layer_texture (priv->pipeline, 0, NULL);
+
+ update_hw_cursor (renderer);
+ }
+}
+
+MetaCursorRenderer *
+meta_cursor_renderer_new (MetaScreen *screen)
+{
+ MetaCursorRenderer *renderer;
+ MetaCursorRendererPrivate *priv;
+
+ renderer = META_CURSOR_RENDERER (g_object_new (META_TYPE_CURSOR_RENDERER, NULL));
+ priv = meta_cursor_renderer_get_instance_private (renderer);
+ priv->screen = screen;
+
+ return renderer;
+}
+
+void
+meta_cursor_renderer_set_cursor (MetaCursorRenderer *renderer,
+ MetaCursorReference *cursor)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ if (priv->displayed_cursor == cursor)
+ return;
+
+ priv->displayed_cursor = cursor;
+ update_pipeline (renderer);
+ update_cursor (renderer);
+}
+
+void
+meta_cursor_renderer_set_position (MetaCursorRenderer *renderer,
+ int x, int y)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ g_assert (meta_is_wayland_compositor ());
+
+ priv->current_x = x;
+ priv->current_y = y;
+
+ update_cursor (renderer);
+}
+
+void
+meta_cursor_renderer_paint (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ g_assert (meta_is_wayland_compositor ());
+
+ if (priv->has_hw_cursor || !priv->displayed_cursor)
+ return;
+
+ cogl_framebuffer_draw_rectangle (cogl_get_draw_framebuffer (),
+ priv->pipeline,
+ priv->current_rect.x,
+ priv->current_rect.y,
+ priv->current_rect.x +
+ priv->current_rect.width,
+ priv->current_rect.y +
+ priv->current_rect.height);
+
+ priv->previous_rect = priv->current_rect;
+ priv->previous_is_valid = TRUE;
+}
+
+void
+meta_cursor_renderer_force_update (MetaCursorRenderer *renderer)
+{
+ g_assert (meta_is_wayland_compositor ());
+
+ update_hw_cursor (renderer);
+}
+
+struct gbm_device *
+meta_cursor_renderer_get_gbm_device (MetaCursorRenderer *renderer)
+{
+ MetaCursorRendererPrivate *priv = meta_cursor_renderer_get_instance_private (renderer);
+
+ return priv->gbm;
+}
diff --git a/src/backends/meta-cursor-renderer.h b/src/backends/meta-cursor-renderer.h
new file mode 100644
index 0000000..82e24fb
--- /dev/null
+++ b/src/backends/meta-cursor-renderer.h
@@ -0,0 +1,71 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
+
+/*
+ * Copyright (C) 2014 Red Hat
+ *
+ * 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.
+ *
+ * Written by:
+ * Jasper St. Pierre <jstpierre mecheye net>
+ */
+
+#ifndef META_CURSOR_RENDERER_H
+#define META_CURSOR_RENDERER_H
+
+#include <glib-object.h>
+
+#include <meta/screen.h>
+#include "meta-cursor.h"
+
+#include <gbm.h>
+
+#define META_TYPE_CURSOR_RENDERER (meta_cursor_renderer_get_type ())
+#define META_CURSOR_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), META_TYPE_CURSOR_RENDERER,
MetaCursorRenderer))
+#define META_CURSOR_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), META_TYPE_CURSOR_RENDERER,
MetaCursorRendererClass))
+#define META_IS_CURSOR_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), META_TYPE_CURSOR_RENDERER))
+#define META_IS_CURSOR_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), META_TYPE_CURSOR_RENDERER))
+#define META_CURSOR_RENDERER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), META_TYPE_CURSOR_RENDERER,
MetaCursorRendererClass))
+
+typedef struct _MetaCursorRenderer MetaCursorRenderer;
+typedef struct _MetaCursorRendererClass MetaCursorRendererClass;
+
+struct _MetaCursorRenderer
+{
+ GObject parent;
+};
+
+struct _MetaCursorRendererClass
+{
+ GObjectClass parent_class;
+};
+
+GType meta_cursor_renderer_get_type (void) G_GNUC_CONST;
+
+MetaCursorRenderer * meta_cursor_renderer_new (MetaScreen *screen);
+
+void meta_cursor_renderer_set_cursor (MetaCursorRenderer *renderer,
+ MetaCursorReference *cursor);
+
+void meta_cursor_renderer_set_position (MetaCursorRenderer *renderer,
+ int x, int y);
+
+void meta_cursor_renderer_paint (MetaCursorRenderer *renderer);
+
+void meta_cursor_renderer_force_update (MetaCursorRenderer *renderer);
+
+struct gbm_device * meta_cursor_renderer_get_gbm_device (MetaCursorRenderer *renderer);
+
+#endif /* META_CURSOR_RENDERER_H */
diff --git a/src/backends/meta-cursor-tracker-private.h b/src/backends/meta-cursor-tracker-private.h
index 93661f8..fa743c6 100644
--- a/src/backends/meta-cursor-tracker-private.h
+++ b/src/backends/meta-cursor-tracker-private.h
@@ -27,14 +27,15 @@
#include <gbm.h>
#include "meta-cursor.h"
+#include "meta-cursor-renderer.h"
struct _MetaCursorTracker {
GObject parent_instance;
MetaScreen *screen;
+ MetaCursorRenderer *renderer;
gboolean is_showing;
- gboolean has_hw_cursor;
/* The cursor tracker stores the cursor for the current grab
* operation, the cursor for the window with pointer focus, and
@@ -61,15 +62,6 @@ struct _MetaCursorTracker {
MetaCursorReference *window_cursor;
MetaCursorReference *root_cursor;
-
- int current_x, current_y;
- MetaRectangle current_rect;
- MetaRectangle previous_rect;
- gboolean previous_is_valid;
-
- CoglPipeline *pipeline;
- int drm_fd;
- struct gbm_device *gbm;
};
struct _MetaCursorTrackerClass {
@@ -90,7 +82,6 @@ void meta_cursor_tracker_set_root_cursor (MetaCursorTracker *tracker,
void meta_cursor_tracker_update_position (MetaCursorTracker *tracker,
int new_x,
int new_y);
-void meta_cursor_tracker_paint (MetaCursorTracker *tracker);
void meta_cursor_tracker_force_update (MetaCursorTracker *tracker);
diff --git a/src/backends/meta-cursor-tracker.c b/src/backends/meta-cursor-tracker.c
index 65f9dfa..87e46f0 100644
--- a/src/backends/meta-cursor-tracker.c
+++ b/src/backends/meta-cursor-tracker.c
@@ -43,7 +43,6 @@
#include "meta-cursor-private.h"
#include "meta-cursor-tracker-private.h"
#include "screen-private.h"
-#include "meta-monitor-manager.h"
#include "wayland/meta-wayland-private.h"
@@ -56,10 +55,42 @@ enum {
static guint signals[LAST_SIGNAL];
-static void meta_cursor_tracker_set_crtc_has_hw_cursor (MetaCursorTracker *tracker,
- MetaCRTC *crtc,
- gboolean has_hw_cursor);
-static void sync_cursor (MetaCursorTracker *tracker);
+static MetaCursorReference *
+get_displayed_cursor (MetaCursorTracker *tracker)
+{
+ if (!tracker->is_showing)
+ return NULL;
+
+ if (tracker->grab_cursor)
+ return tracker->grab_cursor;
+
+ if (tracker->has_window_cursor)
+ return tracker->window_cursor;
+
+ return tracker->root_cursor;
+}
+
+static void
+update_displayed_cursor (MetaCursorTracker *tracker)
+{
+ meta_cursor_renderer_set_cursor (tracker->renderer, tracker->displayed_cursor);
+}
+
+static void
+sync_cursor (MetaCursorTracker *tracker)
+{
+ MetaCursorReference *displayed_cursor = get_displayed_cursor (tracker);
+
+ if (tracker->displayed_cursor == displayed_cursor)
+ return;
+
+ g_clear_pointer (&tracker->displayed_cursor, meta_cursor_reference_unref);
+ if (displayed_cursor)
+ tracker->displayed_cursor = meta_cursor_reference_ref (displayed_cursor);
+
+ update_displayed_cursor (tracker);
+ g_signal_emit (tracker, signals[CURSOR_CHANGED], 0);
+}
static void
meta_cursor_tracker_init (MetaCursorTracker *self)
@@ -82,11 +113,6 @@ meta_cursor_tracker_finalize (GObject *object)
if (self->root_cursor)
meta_cursor_reference_unref (self->root_cursor);
- if (self->pipeline)
- cogl_object_unref (self->pipeline);
- if (self->gbm)
- gbm_device_destroy (self->gbm);
-
G_OBJECT_CLASS (meta_cursor_tracker_parent_class)->finalize (object);
}
@@ -105,63 +131,20 @@ meta_cursor_tracker_class_init (MetaCursorTrackerClass *klass)
G_TYPE_NONE, 0);
}
-static void
-on_monitors_changed (MetaMonitorManager *monitors,
- MetaCursorTracker *tracker)
-{
- MetaCRTC *crtcs;
- unsigned int i, n_crtcs;
-
- if (!tracker->has_hw_cursor)
- return;
-
- /* Go through the new list of monitors, find out where the cursor is */
- meta_monitor_manager_get_resources (monitors, NULL, NULL, &crtcs, &n_crtcs, NULL, NULL);
-
- for (i = 0; i < n_crtcs; i++)
- {
- MetaRectangle *rect = &crtcs[i].rect;
- gboolean has;
-
- has = meta_rectangle_overlap (&tracker->current_rect, rect);
-
- /* Need to do it unconditionally here, our tracking is
- wrong because we reloaded the CRTCs */
- meta_cursor_tracker_set_crtc_has_hw_cursor (tracker, &crtcs[i], has);
- }
-}
-
static MetaCursorTracker *
make_wayland_cursor_tracker (MetaScreen *screen)
{
MetaWaylandCompositor *compositor;
- CoglContext *ctx;
- MetaMonitorManager *monitors;
MetaCursorTracker *self;
self = g_object_new (META_TYPE_CURSOR_TRACKER, NULL);
self->screen = screen;
-
- ctx = clutter_backend_get_cogl_context (clutter_get_default_backend ());
- self->pipeline = cogl_pipeline_new (ctx);
+ self->renderer = meta_cursor_renderer_new (screen);
compositor = meta_wayland_compositor_get_default ();
compositor->seat->pointer.cursor_tracker = self;
meta_cursor_tracker_update_position (self, 0, 0);
-#if defined(CLUTTER_WINDOWING_EGL)
- if (clutter_check_windowing_backend (CLUTTER_WINDOWING_EGL))
- {
- CoglRenderer *cogl_renderer = cogl_display_get_renderer (cogl_context_get_display (ctx));
- self->drm_fd = cogl_kms_renderer_get_kms_fd (cogl_renderer);
- self->gbm = gbm_create_device (self->drm_fd);
- }
-#endif
-
- monitors = meta_monitor_manager_get ();
- g_signal_connect_object (monitors, "monitors-changed",
- G_CALLBACK (on_monitors_changed), self, 0);
-
return self;
}
@@ -169,7 +152,9 @@ static MetaCursorTracker *
make_x11_cursor_tracker (MetaScreen *screen)
{
MetaCursorTracker *self = g_object_new (META_TYPE_CURSOR_TRACKER, NULL);
+
self->screen = screen;
+ self->renderer = meta_cursor_renderer_new (screen);
XFixesSelectCursorInput (screen->display->xdisplay,
screen->xroot,
@@ -398,184 +383,6 @@ meta_cursor_tracker_set_root_cursor (MetaCursorTracker *tracker,
sync_cursor (tracker);
}
-static gboolean
-should_have_hw_cursor (MetaCursorTracker *tracker)
-{
- if (tracker->displayed_cursor)
- return (meta_cursor_reference_get_gbm_bo (tracker->displayed_cursor, NULL, NULL) != NULL);
- else
- return FALSE;
-}
-
-static void
-update_hw_cursor (MetaCursorTracker *tracker)
-{
- MetaMonitorManager *monitors;
- MetaCRTC *crtcs;
- unsigned int i, n_crtcs;
- gboolean enabled;
-
- enabled = should_have_hw_cursor (tracker);
- tracker->has_hw_cursor = enabled;
-
- monitors = meta_monitor_manager_get ();
- meta_monitor_manager_get_resources (monitors, NULL, NULL, &crtcs, &n_crtcs, NULL, NULL);
-
- for (i = 0; i < n_crtcs; i++)
- {
- MetaRectangle *rect = &crtcs[i].rect;
- gboolean has;
-
- has = enabled && meta_rectangle_overlap (&tracker->current_rect, rect);
-
- if (has || crtcs[i].has_hw_cursor)
- meta_cursor_tracker_set_crtc_has_hw_cursor (tracker, &crtcs[i], has);
- }
-}
-
-static void
-move_hw_cursor (MetaCursorTracker *tracker)
-{
- MetaMonitorManager *monitors;
- MetaCRTC *crtcs;
- unsigned int i, n_crtcs;
-
- monitors = meta_monitor_manager_get ();
- meta_monitor_manager_get_resources (monitors, NULL, NULL, &crtcs, &n_crtcs, NULL, NULL);
-
- g_assert (tracker->has_hw_cursor);
-
- for (i = 0; i < n_crtcs; i++)
- {
- MetaRectangle *rect = &crtcs[i].rect;
- gboolean has;
-
- has = meta_rectangle_overlap (&tracker->current_rect, rect);
-
- if (has != crtcs[i].has_hw_cursor)
- meta_cursor_tracker_set_crtc_has_hw_cursor (tracker, &crtcs[i], has);
- if (has)
- drmModeMoveCursor (tracker->drm_fd, crtcs[i].crtc_id,
- tracker->current_rect.x - rect->x,
- tracker->current_rect.y - rect->y);
- }
-}
-
-static MetaCursorReference *
-get_displayed_cursor (MetaCursorTracker *tracker)
-{
- if (!tracker->is_showing)
- return NULL;
-
- if (tracker->grab_cursor)
- return tracker->grab_cursor;
-
- if (tracker->has_window_cursor)
- return tracker->window_cursor;
-
- return tracker->root_cursor;
-}
-
-static void
-update_displayed_cursor (MetaCursorTracker *tracker)
-{
- if (meta_is_wayland_compositor ())
- {
- if (tracker->displayed_cursor)
- {
- CoglTexture *texture = meta_cursor_reference_get_cogl_texture (tracker->displayed_cursor, NULL,
NULL);
- cogl_pipeline_set_layer_texture (tracker->pipeline, 0, texture);
- }
- else
- cogl_pipeline_set_layer_texture (tracker->pipeline, 0, NULL);
-
- update_hw_cursor (tracker);
- }
-}
-
-static void
-sync_displayed_cursor (MetaCursorTracker *tracker)
-{
- MetaCursorReference *displayed_cursor = get_displayed_cursor (tracker);
-
- if (tracker->displayed_cursor == displayed_cursor)
- return;
-
- g_clear_pointer (&tracker->displayed_cursor, meta_cursor_reference_unref);
- if (displayed_cursor)
- tracker->displayed_cursor = meta_cursor_reference_ref (displayed_cursor);
-
- update_displayed_cursor (tracker);
- g_signal_emit (tracker, signals[CURSOR_CHANGED], 0);
-}
-
-static void
-meta_cursor_tracker_queue_redraw (MetaCursorTracker *tracker)
-{
- MetaWaylandCompositor *compositor = meta_wayland_compositor_get_default ();
- ClutterActor *stage = compositor->stage;
- cairo_rectangle_int_t clip;
-
- g_assert (meta_is_wayland_compositor ());
-
- /* Clear the location the cursor was at before, if we need to. */
- if (tracker->previous_is_valid)
- {
- clip.x = tracker->previous_rect.x;
- clip.y = tracker->previous_rect.y;
- clip.width = tracker->previous_rect.width;
- clip.height = tracker->previous_rect.height;
- clutter_actor_queue_redraw_with_clip (stage, &clip);
- tracker->previous_is_valid = FALSE;
- }
-
- if (tracker->has_hw_cursor || !tracker->displayed_cursor)
- return;
-
- clip.x = tracker->current_rect.x;
- clip.y = tracker->current_rect.y;
- clip.width = tracker->current_rect.width;
- clip.height = tracker->current_rect.height;
- clutter_actor_queue_redraw_with_clip (stage, &clip);
-}
-
-static void
-sync_cursor (MetaCursorTracker *tracker)
-{
- MetaCursorReference *displayed_cursor;
-
- sync_displayed_cursor (tracker);
- displayed_cursor = tracker->displayed_cursor;
-
- if (displayed_cursor)
- {
- CoglTexture *texture;
- int hot_x, hot_y;
-
- texture = meta_cursor_reference_get_cogl_texture (displayed_cursor, &hot_x, &hot_y);
-
- tracker->current_rect.x = tracker->current_x - hot_x;
- tracker->current_rect.y = tracker->current_y - hot_y;
- tracker->current_rect.width = cogl_texture_get_width (COGL_TEXTURE (texture));
- tracker->current_rect.height = cogl_texture_get_height (COGL_TEXTURE (texture));
- }
- else
- {
- tracker->current_rect.x = 0;
- tracker->current_rect.y = 0;
- tracker->current_rect.width = 0;
- tracker->current_rect.height = 0;
- }
-
- if (meta_is_wayland_compositor ())
- {
- if (tracker->has_hw_cursor)
- move_hw_cursor (tracker);
- else
- meta_cursor_tracker_queue_redraw (tracker);
- }
-}
-
void
meta_cursor_tracker_update_position (MetaCursorTracker *tracker,
int new_x,
@@ -583,61 +390,7 @@ meta_cursor_tracker_update_position (MetaCursorTracker *tracker,
{
g_assert (meta_is_wayland_compositor ());
- tracker->current_x = new_x;
- tracker->current_y = new_y;
-
- sync_cursor (tracker);
-}
-
-void
-meta_cursor_tracker_paint (MetaCursorTracker *tracker)
-{
- g_assert (meta_is_wayland_compositor ());
-
- if (tracker->has_hw_cursor || !tracker->displayed_cursor)
- return;
-
- cogl_framebuffer_draw_rectangle (cogl_get_draw_framebuffer (),
- tracker->pipeline,
- tracker->current_rect.x,
- tracker->current_rect.y,
- tracker->current_rect.x +
- tracker->current_rect.width,
- tracker->current_rect.y +
- tracker->current_rect.height);
-
- tracker->previous_rect = tracker->current_rect;
- tracker->previous_is_valid = TRUE;
-}
-
-static void
-meta_cursor_tracker_set_crtc_has_hw_cursor (MetaCursorTracker *tracker,
- MetaCRTC *crtc,
- gboolean has)
-{
- if (has)
- {
- MetaCursorReference *displayed_cursor = tracker->displayed_cursor;
- struct gbm_bo *bo;
- union gbm_bo_handle handle;
- int width, height;
- int hot_x, hot_y;
-
- bo = meta_cursor_reference_get_gbm_bo (displayed_cursor, &hot_x, &hot_y);
-
- handle = gbm_bo_get_handle (bo);
- width = gbm_bo_get_width (bo);
- height = gbm_bo_get_height (bo);
-
- drmModeSetCursor2 (tracker->drm_fd, crtc->crtc_id, handle.u32,
- width, height, hot_x, hot_y);
- crtc->has_hw_cursor = TRUE;
- }
- else
- {
- drmModeSetCursor2 (tracker->drm_fd, crtc->crtc_id, 0, 0, 0, 0, 0);
- crtc->has_hw_cursor = FALSE;
- }
+ meta_cursor_renderer_set_position (tracker->renderer, new_x, new_y);
}
static void
@@ -725,12 +478,11 @@ meta_cursor_tracker_force_update (MetaCursorTracker *tracker)
{
g_assert (meta_is_wayland_compositor ());
- update_hw_cursor (tracker);
- sync_cursor (tracker);
+ meta_cursor_renderer_force_update (tracker->renderer);
}
struct gbm_device *
meta_cursor_tracker_get_gbm_device (MetaCursorTracker *tracker)
{
- return tracker->gbm;
+ return meta_cursor_renderer_get_gbm_device (tracker->renderer);
}
diff --git a/src/wayland/meta-wayland-stage.c b/src/wayland/meta-wayland-stage.c
index 1a63f33..2595b08 100644
--- a/src/wayland/meta-wayland-stage.c
+++ b/src/wayland/meta-wayland-stage.c
@@ -42,7 +42,7 @@ meta_wayland_stage_paint (ClutterActor *actor)
compositor = meta_wayland_compositor_get_default ();
if (compositor->seat->pointer.cursor_tracker)
- meta_cursor_tracker_paint (compositor->seat->pointer.cursor_tracker);
+ meta_cursor_renderer_paint (compositor->seat->pointer.cursor_tracker->renderer);
}
static void
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]