4e402b3972
The 'cursor-mode', which currently is limited to RecordMonitor(), allows the user to either do screen casts where the cursor is hidden, embedded in the framebuffer, or sent as PipeWire stream metadata. The latter allows the user to get cursor updates sent, including the cursor sprite, without requiring a stage paint each frame. Currently this is done by using the cursor sprite texture, and either reading directly from, or drawing to an offscreen framebuffer which is read from instead, in case the texture is scaled. https://gitlab.gnome.org/GNOME/mutter/merge_requests/357
563 lines
17 KiB
C
563 lines
17 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
/*
|
|
* Copyright (C) 2015-2017 Red Hat Inc.
|
|
*
|
|
* 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 "backends/meta-screen-cast-session.h"
|
|
|
|
#include "backends/meta-backend-private.h"
|
|
#include "backends/meta-dbus-session-watcher.h"
|
|
#include "backends/meta-remote-access-controller-private.h"
|
|
#include "backends/meta-screen-cast-monitor-stream.h"
|
|
#include "backends/meta-screen-cast-stream.h"
|
|
#include "backends/meta-screen-cast-window-stream.h"
|
|
#include "core/display-private.h"
|
|
|
|
#define META_SCREEN_CAST_SESSION_DBUS_PATH "/org/gnome/Mutter/ScreenCast/Session"
|
|
|
|
struct _MetaScreenCastSession
|
|
{
|
|
MetaDBusScreenCastSessionSkeleton parent;
|
|
|
|
MetaScreenCast *screen_cast;
|
|
|
|
char *peer_name;
|
|
|
|
MetaScreenCastSessionType session_type;
|
|
char *object_path;
|
|
|
|
GList *streams;
|
|
|
|
MetaScreenCastSessionHandle *handle;
|
|
};
|
|
|
|
static void
|
|
meta_screen_cast_session_init_iface (MetaDBusScreenCastSessionIface *iface);
|
|
|
|
static void
|
|
meta_dbus_session_init_iface (MetaDbusSessionInterface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (MetaScreenCastSession,
|
|
meta_screen_cast_session,
|
|
META_DBUS_TYPE_SCREEN_CAST_SESSION_SKELETON,
|
|
G_IMPLEMENT_INTERFACE (META_DBUS_TYPE_SCREEN_CAST_SESSION,
|
|
meta_screen_cast_session_init_iface)
|
|
G_IMPLEMENT_INTERFACE (META_TYPE_DBUS_SESSION,
|
|
meta_dbus_session_init_iface))
|
|
|
|
struct _MetaScreenCastSessionHandle
|
|
{
|
|
MetaRemoteAccessHandle parent;
|
|
|
|
MetaScreenCastSession *session;
|
|
};
|
|
|
|
G_DEFINE_TYPE (MetaScreenCastSessionHandle,
|
|
meta_screen_cast_session_handle,
|
|
META_TYPE_REMOTE_ACCESS_HANDLE)
|
|
|
|
static MetaScreenCastSessionHandle *
|
|
meta_screen_cast_session_handle_new (MetaScreenCastSession *session);
|
|
|
|
static void
|
|
init_remote_access_handle (MetaScreenCastSession *session)
|
|
{
|
|
MetaBackend *backend = meta_get_backend ();
|
|
MetaRemoteAccessController *remote_access_controller;
|
|
MetaRemoteAccessHandle *remote_access_handle;
|
|
|
|
session->handle = meta_screen_cast_session_handle_new (session);
|
|
|
|
remote_access_controller = meta_backend_get_remote_access_controller (backend);
|
|
remote_access_handle = META_REMOTE_ACCESS_HANDLE (session->handle);
|
|
meta_remote_access_controller_notify_new_handle (remote_access_controller,
|
|
remote_access_handle);
|
|
}
|
|
|
|
gboolean
|
|
meta_screen_cast_session_start (MetaScreenCastSession *session,
|
|
GError **error)
|
|
{
|
|
GList *l;
|
|
|
|
for (l = session->streams; l; l = l->next)
|
|
{
|
|
MetaScreenCastStream *stream = l->data;
|
|
|
|
if (!meta_screen_cast_stream_start (stream, error))
|
|
return FALSE;
|
|
}
|
|
|
|
init_remote_access_handle (session);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
meta_screen_cast_session_close (MetaScreenCastSession *session)
|
|
{
|
|
MetaDBusScreenCastSession *skeleton = META_DBUS_SCREEN_CAST_SESSION (session);
|
|
|
|
g_list_free_full (session->streams, g_object_unref);
|
|
|
|
meta_dbus_session_notify_closed (META_DBUS_SESSION (session));
|
|
|
|
switch (session->session_type)
|
|
{
|
|
case META_SCREEN_CAST_SESSION_TYPE_NORMAL:
|
|
meta_dbus_screen_cast_session_emit_closed (skeleton);
|
|
break;
|
|
case META_SCREEN_CAST_SESSION_TYPE_REMOTE_DESKTOP:
|
|
break;
|
|
}
|
|
|
|
g_dbus_interface_skeleton_unexport (G_DBUS_INTERFACE_SKELETON (session));
|
|
|
|
if (session->handle)
|
|
{
|
|
MetaRemoteAccessHandle *remote_access_handle =
|
|
META_REMOTE_ACCESS_HANDLE (session->handle);
|
|
|
|
meta_remote_access_handle_notify_stopped (remote_access_handle);
|
|
}
|
|
|
|
g_object_unref (session);
|
|
}
|
|
|
|
MetaScreenCastStream *
|
|
meta_screen_cast_session_get_stream (MetaScreenCastSession *session,
|
|
const char *path)
|
|
{
|
|
GList *l;
|
|
|
|
for (l = session->streams; l; l = l->next)
|
|
{
|
|
MetaScreenCastStream *stream = l->data;
|
|
|
|
if (g_strcmp0 (meta_screen_cast_stream_get_object_path (stream),
|
|
path) == 0)
|
|
return stream;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
MetaScreenCast *
|
|
meta_screen_cast_session_get_screen_cast (MetaScreenCastSession *session)
|
|
{
|
|
return session->screen_cast;
|
|
}
|
|
|
|
char *
|
|
meta_screen_cast_session_get_object_path (MetaScreenCastSession *session)
|
|
{
|
|
return session->object_path;
|
|
}
|
|
|
|
static gboolean
|
|
check_permission (MetaScreenCastSession *session,
|
|
GDBusMethodInvocation *invocation)
|
|
{
|
|
return g_strcmp0 (session->peer_name,
|
|
g_dbus_method_invocation_get_sender (invocation)) == 0;
|
|
}
|
|
|
|
static gboolean
|
|
handle_start (MetaDBusScreenCastSession *skeleton,
|
|
GDBusMethodInvocation *invocation)
|
|
{
|
|
MetaScreenCastSession *session = META_SCREEN_CAST_SESSION (skeleton);
|
|
GError *error = NULL;
|
|
|
|
if (!check_permission (session, invocation))
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_ACCESS_DENIED,
|
|
"Permission denied");
|
|
return TRUE;
|
|
}
|
|
|
|
switch (session->session_type)
|
|
{
|
|
case META_SCREEN_CAST_SESSION_TYPE_NORMAL:
|
|
break;
|
|
case META_SCREEN_CAST_SESSION_TYPE_REMOTE_DESKTOP:
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Must be started from remote desktop session");
|
|
return TRUE;
|
|
}
|
|
|
|
if (!meta_screen_cast_session_start (session, &error))
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Failed to start screen cast: %s",
|
|
error->message);
|
|
g_error_free (error);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
meta_dbus_screen_cast_session_complete_start (skeleton, invocation);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
handle_stop (MetaDBusScreenCastSession *skeleton,
|
|
GDBusMethodInvocation *invocation)
|
|
{
|
|
MetaScreenCastSession *session = META_SCREEN_CAST_SESSION (skeleton);
|
|
|
|
if (!check_permission (session, invocation))
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_ACCESS_DENIED,
|
|
"Permission denied");
|
|
return TRUE;
|
|
}
|
|
|
|
switch (session->session_type)
|
|
{
|
|
case META_SCREEN_CAST_SESSION_TYPE_NORMAL:
|
|
break;
|
|
case META_SCREEN_CAST_SESSION_TYPE_REMOTE_DESKTOP:
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Must be stopped from remote desktop session");
|
|
return TRUE;
|
|
}
|
|
|
|
meta_screen_cast_session_close (session);
|
|
|
|
meta_dbus_screen_cast_session_complete_stop (skeleton, invocation);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
on_stream_closed (MetaScreenCastStream *stream,
|
|
MetaScreenCastSession *session)
|
|
{
|
|
meta_screen_cast_session_close (session);
|
|
}
|
|
|
|
static gboolean
|
|
is_valid_cursor_mode (MetaScreenCastCursorMode cursor_mode)
|
|
{
|
|
switch (cursor_mode)
|
|
{
|
|
case META_SCREEN_CAST_CURSOR_MODE_HIDDEN:
|
|
case META_SCREEN_CAST_CURSOR_MODE_EMBEDDED:
|
|
case META_SCREEN_CAST_CURSOR_MODE_METADATA:
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
handle_record_monitor (MetaDBusScreenCastSession *skeleton,
|
|
GDBusMethodInvocation *invocation,
|
|
const char *connector,
|
|
GVariant *properties_variant)
|
|
{
|
|
MetaScreenCastSession *session = META_SCREEN_CAST_SESSION (skeleton);
|
|
GDBusInterfaceSkeleton *interface_skeleton;
|
|
GDBusConnection *connection;
|
|
MetaBackend *backend = meta_get_backend ();
|
|
MetaMonitorManager *monitor_manager =
|
|
meta_backend_get_monitor_manager (backend);
|
|
MetaMonitor *monitor;
|
|
MetaScreenCastCursorMode cursor_mode;
|
|
ClutterStage *stage;
|
|
GError *error = NULL;
|
|
MetaScreenCastMonitorStream *monitor_stream;
|
|
MetaScreenCastStream *stream;
|
|
char *stream_path;
|
|
|
|
if (!check_permission (session, invocation))
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_ACCESS_DENIED,
|
|
"Permission denied");
|
|
return TRUE;
|
|
}
|
|
|
|
interface_skeleton = G_DBUS_INTERFACE_SKELETON (skeleton);
|
|
connection = g_dbus_interface_skeleton_get_connection (interface_skeleton);
|
|
|
|
if (g_str_equal (connector, ""))
|
|
monitor = meta_monitor_manager_get_primary_monitor (monitor_manager);
|
|
else
|
|
monitor = meta_monitor_manager_get_monitor_from_connector (monitor_manager,
|
|
connector);
|
|
|
|
if (!monitor)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Unknown monitor");
|
|
return TRUE;
|
|
}
|
|
|
|
if (!g_variant_lookup (properties_variant, "cursor-mode", "u", &cursor_mode))
|
|
{
|
|
cursor_mode = META_SCREEN_CAST_CURSOR_MODE_HIDDEN;
|
|
}
|
|
else
|
|
{
|
|
if (!is_valid_cursor_mode (cursor_mode))
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Unknown cursor mode");
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
stage = CLUTTER_STAGE (meta_backend_get_stage (backend));
|
|
|
|
monitor_stream = meta_screen_cast_monitor_stream_new (session,
|
|
connection,
|
|
monitor,
|
|
stage,
|
|
cursor_mode,
|
|
&error);
|
|
if (!monitor_stream)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Failed to record monitor: %s",
|
|
error->message);
|
|
g_error_free (error);
|
|
return TRUE;
|
|
}
|
|
|
|
stream = META_SCREEN_CAST_STREAM (monitor_stream);
|
|
stream_path = meta_screen_cast_stream_get_object_path (stream);
|
|
|
|
session->streams = g_list_append (session->streams, stream);
|
|
|
|
g_signal_connect (stream, "closed", G_CALLBACK (on_stream_closed), session);
|
|
|
|
meta_dbus_screen_cast_session_complete_record_monitor (skeleton,
|
|
invocation,
|
|
stream_path);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
handle_record_window (MetaDBusScreenCastSession *skeleton,
|
|
GDBusMethodInvocation *invocation,
|
|
GVariant *properties_variant)
|
|
{
|
|
MetaScreenCastSession *session = META_SCREEN_CAST_SESSION (skeleton);
|
|
GDBusInterfaceSkeleton *interface_skeleton;
|
|
GDBusConnection *connection;
|
|
MetaWindow *window;
|
|
GError *error = NULL;
|
|
MetaDisplay *display;
|
|
GVariant *window_id_variant = NULL;
|
|
MetaScreenCastWindowStream *window_stream;
|
|
MetaScreenCastStream *stream;
|
|
char *stream_path;
|
|
|
|
if (!check_permission (session, invocation))
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_ACCESS_DENIED,
|
|
"Permission denied");
|
|
return TRUE;
|
|
}
|
|
|
|
if (properties_variant)
|
|
window_id_variant = g_variant_lookup_value (properties_variant,
|
|
"window-id",
|
|
G_VARIANT_TYPE ("t"));
|
|
|
|
display = meta_get_display ();
|
|
if (window_id_variant)
|
|
{
|
|
uint64_t window_id;
|
|
|
|
g_variant_get (window_id_variant, "t", &window_id);
|
|
window = meta_display_get_window_from_id (display, window_id);
|
|
}
|
|
else
|
|
{
|
|
window = meta_display_get_focus_window (display);
|
|
}
|
|
|
|
if (!window)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Window not found");
|
|
return TRUE;
|
|
}
|
|
|
|
interface_skeleton = G_DBUS_INTERFACE_SKELETON (skeleton);
|
|
connection = g_dbus_interface_skeleton_get_connection (interface_skeleton);
|
|
|
|
window_stream = meta_screen_cast_window_stream_new (session,
|
|
connection,
|
|
window,
|
|
&error);
|
|
if (!window_stream)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
|
|
G_DBUS_ERROR_FAILED,
|
|
"Failed to record window: %s",
|
|
error->message);
|
|
g_error_free (error);
|
|
return TRUE;
|
|
}
|
|
|
|
stream = META_SCREEN_CAST_STREAM (window_stream);
|
|
stream_path = meta_screen_cast_stream_get_object_path (stream);
|
|
|
|
session->streams = g_list_append (session->streams, stream);
|
|
|
|
g_signal_connect (stream, "closed", G_CALLBACK (on_stream_closed), session);
|
|
|
|
meta_dbus_screen_cast_session_complete_record_window (skeleton,
|
|
invocation,
|
|
stream_path);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_init_iface (MetaDBusScreenCastSessionIface *iface)
|
|
{
|
|
iface->handle_start = handle_start;
|
|
iface->handle_stop = handle_stop;
|
|
iface->handle_record_monitor = handle_record_monitor;
|
|
iface->handle_record_window = handle_record_window;
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_client_vanished (MetaDbusSession *dbus_session)
|
|
{
|
|
meta_screen_cast_session_close (META_SCREEN_CAST_SESSION (dbus_session));
|
|
}
|
|
|
|
static void
|
|
meta_dbus_session_init_iface (MetaDbusSessionInterface *iface)
|
|
{
|
|
iface->client_vanished = meta_screen_cast_session_client_vanished;
|
|
}
|
|
|
|
MetaScreenCastSession *
|
|
meta_screen_cast_session_new (MetaScreenCast *screen_cast,
|
|
MetaScreenCastSessionType session_type,
|
|
const char *peer_name,
|
|
GError **error)
|
|
{
|
|
GDBusInterfaceSkeleton *interface_skeleton;
|
|
MetaScreenCastSession *session;
|
|
GDBusConnection *connection;
|
|
static unsigned int global_session_number = 0;
|
|
|
|
session = g_object_new (META_TYPE_SCREEN_CAST_SESSION, NULL);
|
|
session->screen_cast = screen_cast;
|
|
session->session_type = session_type;
|
|
session->peer_name = g_strdup (peer_name);
|
|
session->object_path =
|
|
g_strdup_printf (META_SCREEN_CAST_SESSION_DBUS_PATH "/u%u",
|
|
++global_session_number);
|
|
|
|
interface_skeleton = G_DBUS_INTERFACE_SKELETON (session);
|
|
connection = meta_screen_cast_get_connection (screen_cast);
|
|
if (!g_dbus_interface_skeleton_export (interface_skeleton,
|
|
connection,
|
|
session->object_path,
|
|
error))
|
|
return NULL;
|
|
|
|
return session;
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_finalize (GObject *object)
|
|
{
|
|
MetaScreenCastSession *session = META_SCREEN_CAST_SESSION (object);
|
|
|
|
g_clear_object (&session->handle);
|
|
g_free (session->peer_name);
|
|
g_free (session->object_path);
|
|
|
|
G_OBJECT_CLASS (meta_screen_cast_session_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_init (MetaScreenCastSession *session)
|
|
{
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_class_init (MetaScreenCastSessionClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = meta_screen_cast_session_finalize;
|
|
}
|
|
|
|
static MetaScreenCastSessionHandle *
|
|
meta_screen_cast_session_handle_new (MetaScreenCastSession *session)
|
|
{
|
|
MetaScreenCastSessionHandle *handle;
|
|
|
|
handle = g_object_new (META_TYPE_SCREEN_CAST_SESSION_HANDLE, NULL);
|
|
handle->session = session;
|
|
|
|
return handle;
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_handle_stop (MetaRemoteAccessHandle *handle)
|
|
{
|
|
MetaScreenCastSession *session;
|
|
|
|
session = META_SCREEN_CAST_SESSION_HANDLE (handle)->session;
|
|
if (!session)
|
|
return;
|
|
|
|
meta_screen_cast_session_close (session);
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_handle_init (MetaScreenCastSessionHandle *handle)
|
|
{
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_session_handle_class_init (MetaScreenCastSessionHandleClass *klass)
|
|
{
|
|
MetaRemoteAccessHandleClass *remote_access_handle_class =
|
|
META_REMOTE_ACCESS_HANDLE_CLASS (klass);
|
|
|
|
remote_access_handle_class->stop = meta_screen_cast_session_handle_stop;
|
|
}
|