2017-06-21 06:23:44 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 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-monitor-stream-src.h"
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
#include <spa/buffer/meta.h>
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
#include "backends/meta-backend-private.h"
|
2018-12-14 16:47:57 +00:00
|
|
|
#include "backends/meta-cursor-tracker-private.h"
|
2017-06-21 06:23:44 +00:00
|
|
|
#include "backends/meta-logical-monitor.h"
|
|
|
|
#include "backends/meta-monitor.h"
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "backends/meta-screen-cast-monitor-stream.h"
|
2018-12-14 16:47:57 +00:00
|
|
|
#include "backends/meta-screen-cast-session.h"
|
2019-06-18 00:49:45 +00:00
|
|
|
#include "backends/meta-stage-private.h"
|
2017-06-21 06:23:44 +00:00
|
|
|
#include "clutter/clutter.h"
|
|
|
|
#include "clutter/clutter-mutter.h"
|
2018-12-14 16:47:57 +00:00
|
|
|
#include "core/boxes-private.h"
|
2017-06-21 06:23:44 +00:00
|
|
|
|
|
|
|
struct _MetaScreenCastMonitorStreamSrc
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc parent;
|
|
|
|
|
2019-01-03 15:51:08 +00:00
|
|
|
gboolean cursor_bitmap_invalid;
|
2020-01-17 20:08:02 +00:00
|
|
|
gboolean hw_cursor_inhibited;
|
2019-01-03 15:51:08 +00:00
|
|
|
|
2020-01-17 20:08:02 +00:00
|
|
|
GList *watches;
|
2019-06-18 00:49:45 +00:00
|
|
|
|
2020-07-13 12:09:44 +00:00
|
|
|
gulong position_invalidated_handler_id;
|
2018-12-14 16:47:57 +00:00
|
|
|
gulong cursor_changed_handler_id;
|
2022-04-29 14:05:50 +00:00
|
|
|
gulong stage_prepare_frame_handler_id;
|
2021-06-25 16:33:28 +00:00
|
|
|
|
|
|
|
guint maybe_record_idle_id;
|
2017-06-21 06:23:44 +00:00
|
|
|
};
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static void
|
|
|
|
hw_cursor_inhibitor_iface_init (MetaHwCursorInhibitorInterface *iface);
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (MetaScreenCastMonitorStreamSrc,
|
|
|
|
meta_screen_cast_monitor_stream_src,
|
|
|
|
META_TYPE_SCREEN_CAST_STREAM_SRC,
|
|
|
|
G_IMPLEMENT_INTERFACE (META_TYPE_HW_CURSOR_INHIBITOR,
|
|
|
|
hw_cursor_inhibitor_iface_init))
|
2017-06-21 06:23:44 +00:00
|
|
|
|
2022-05-27 18:30:37 +00:00
|
|
|
static MetaBackend *
|
|
|
|
get_backend (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
MetaScreenCastStream *stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
MetaScreenCastSession *session = meta_screen_cast_stream_get_session (stream);
|
|
|
|
MetaScreenCast *screen_cast =
|
|
|
|
meta_screen_cast_session_get_screen_cast (session);
|
|
|
|
|
|
|
|
return meta_screen_cast_get_backend (screen_cast);
|
|
|
|
}
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
static ClutterStage *
|
|
|
|
get_stage (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src;
|
|
|
|
MetaScreenCastStream *stream;
|
|
|
|
MetaScreenCastMonitorStream *monitor_stream;
|
|
|
|
|
|
|
|
src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
monitor_stream = META_SCREEN_CAST_MONITOR_STREAM (stream);
|
|
|
|
|
|
|
|
return meta_screen_cast_monitor_stream_get_stage (monitor_stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaMonitor *
|
|
|
|
get_monitor (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src;
|
|
|
|
MetaScreenCastStream *stream;
|
|
|
|
MetaScreenCastMonitorStream *monitor_stream;
|
|
|
|
|
|
|
|
src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
monitor_stream = META_SCREEN_CAST_MONITOR_STREAM (stream);
|
|
|
|
|
|
|
|
return meta_screen_cast_monitor_stream_get_monitor (monitor_stream);
|
|
|
|
}
|
|
|
|
|
2021-02-01 09:41:57 +00:00
|
|
|
static gboolean
|
2017-06-21 06:23:44 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_get_specs (MetaScreenCastStreamSrc *src,
|
|
|
|
int *width,
|
|
|
|
int *height,
|
|
|
|
float *frame_rate)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
2022-05-27 18:30:37 +00:00
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
2017-06-21 06:23:44 +00:00
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
float scale;
|
|
|
|
MetaMonitorMode *mode;
|
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
mode = meta_monitor_get_current_mode (monitor);
|
|
|
|
|
2022-05-27 18:30:37 +00:00
|
|
|
if (meta_backend_is_stage_views_scaled (backend))
|
2019-01-25 21:06:39 +00:00
|
|
|
scale = logical_monitor->scale;
|
|
|
|
else
|
|
|
|
scale = 1.0;
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
*width = (int) roundf (logical_monitor->rect.width * scale);
|
|
|
|
*height = (int) roundf (logical_monitor->rect.height * scale);
|
|
|
|
*frame_rate = meta_monitor_mode_get_refresh_rate (mode);
|
2021-02-01 09:41:57 +00:00
|
|
|
|
|
|
|
return TRUE;
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|
|
|
|
|
2021-06-25 16:33:28 +00:00
|
|
|
static gboolean
|
|
|
|
maybe_record_frame_on_idle (gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (user_data);
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
MetaScreenCastRecordFlag flags;
|
|
|
|
|
|
|
|
monitor_src->maybe_record_idle_id = 0;
|
|
|
|
|
|
|
|
flags = META_SCREEN_CAST_RECORD_FLAG_NONE;
|
|
|
|
meta_screen_cast_stream_src_maybe_record_frame (src, flags);
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
}
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
static void
|
2020-04-20 20:24:55 +00:00
|
|
|
stage_painted (MetaStage *stage,
|
|
|
|
ClutterStageView *view,
|
|
|
|
ClutterPaintContext *paint_context,
|
|
|
|
gpointer user_data)
|
2017-06-21 06:23:44 +00:00
|
|
|
{
|
2021-06-25 16:33:28 +00:00
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (user_data);
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
2017-06-21 06:23:44 +00:00
|
|
|
|
2021-06-25 16:33:28 +00:00
|
|
|
if (monitor_src->maybe_record_idle_id)
|
|
|
|
return;
|
|
|
|
|
|
|
|
monitor_src->maybe_record_idle_id = g_idle_add (maybe_record_frame_on_idle,
|
|
|
|
src);
|
|
|
|
g_source_set_name_by_id (monitor_src->maybe_record_idle_id,
|
|
|
|
"[mutter] maybe_record_frame_on_idle [monitor-src]");
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 18:41:57 +00:00
|
|
|
static void
|
|
|
|
before_stage_painted (MetaStage *stage,
|
|
|
|
ClutterStageView *view,
|
|
|
|
ClutterPaintContext *paint_context,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
2021-06-25 16:33:28 +00:00
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (user_data);
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
screen-cast/monitor-stream: Immediately record scanout frames
When the before-paint function is executed, it's only purpose
is to check if there's any scanout queue, and immediately
record it if any.
However, since [1], we regressed in this specific case with the
introduction of an idle callback in the before-paint function.
The regression only happens when the PipeWire stream is using
DMA-BUF buffers, and it would operate as follows:
1. In before-paint, when there's a scanout available, we queue
an idle callback to capture the monitor. The idle callback
(almost always) executes after the scanout is pulled from
the stage view
2. meta_screen_cast_stream_src_maybe_record_frame() is called
by the idle callback. In the DMA-BUF case, it then runs
meta_screen_cast_monitor_stream_src_record_to_framebuffer()
3. In meta_screen_cast_monitor_stream_src_record_to_framebuffer(),
because the stage view doesn't have a scanout anymore, it
ends up calling cogl_blit_framebuffer() with the stage view
framebuffer. This is the regression bug.
This regression presents itself in the form of the screencast
stream showing the desktop when there's an unredirected fullscreen
application window running.
Revert before-paint - and only that - back to immediately capturing
any available scanout. Only record these frames when the target
buffer is a DMA-BUF handle. Nothing is captured on before-paint if
the stream is not using DMA-BUF, since the regular paint routine
will handle these frames regularly post-paint.
[1] https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1914
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2186>
2021-12-27 18:32:27 +00:00
|
|
|
MetaScreenCastRecordFlag flags;
|
2020-08-29 18:41:57 +00:00
|
|
|
|
2021-06-25 16:33:28 +00:00
|
|
|
if (monitor_src->maybe_record_idle_id)
|
|
|
|
return;
|
2020-08-29 18:41:57 +00:00
|
|
|
|
2021-06-25 16:33:28 +00:00
|
|
|
if (!clutter_stage_view_peek_scanout (view))
|
|
|
|
return;
|
|
|
|
|
screen-cast/monitor-stream: Immediately record scanout frames
When the before-paint function is executed, it's only purpose
is to check if there's any scanout queue, and immediately
record it if any.
However, since [1], we regressed in this specific case with the
introduction of an idle callback in the before-paint function.
The regression only happens when the PipeWire stream is using
DMA-BUF buffers, and it would operate as follows:
1. In before-paint, when there's a scanout available, we queue
an idle callback to capture the monitor. The idle callback
(almost always) executes after the scanout is pulled from
the stage view
2. meta_screen_cast_stream_src_maybe_record_frame() is called
by the idle callback. In the DMA-BUF case, it then runs
meta_screen_cast_monitor_stream_src_record_to_framebuffer()
3. In meta_screen_cast_monitor_stream_src_record_to_framebuffer(),
because the stage view doesn't have a scanout anymore, it
ends up calling cogl_blit_framebuffer() with the stage view
framebuffer. This is the regression bug.
This regression presents itself in the form of the screencast
stream showing the desktop when there's an unredirected fullscreen
application window running.
Revert before-paint - and only that - back to immediately capturing
any available scanout. Only record these frames when the target
buffer is a DMA-BUF handle. Nothing is captured on before-paint if
the stream is not using DMA-BUF, since the regular paint routine
will handle these frames regularly post-paint.
[1] https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1914
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2186>
2021-12-27 18:32:27 +00:00
|
|
|
flags = META_SCREEN_CAST_RECORD_FLAG_DMABUF_ONLY;
|
|
|
|
meta_screen_cast_stream_src_maybe_record_frame (src, flags);
|
2020-08-29 18:41:57 +00:00
|
|
|
}
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static gboolean
|
|
|
|
is_cursor_in_stream (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaCursorRenderer *cursor_renderer =
|
|
|
|
meta_backend_get_cursor_renderer (backend);
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
2019-02-20 15:23:04 +00:00
|
|
|
graphene_rect_t logical_monitor_rect;
|
2018-12-14 16:47:57 +00:00
|
|
|
MetaCursorSprite *cursor_sprite;
|
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
logical_monitor_layout = meta_logical_monitor_get_layout (logical_monitor);
|
|
|
|
logical_monitor_rect =
|
2019-02-20 15:23:04 +00:00
|
|
|
meta_rectangle_to_graphene_rect (&logical_monitor_layout);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
|
|
|
cursor_sprite = meta_cursor_renderer_get_cursor (cursor_renderer);
|
|
|
|
if (cursor_sprite)
|
|
|
|
{
|
2019-02-20 15:23:04 +00:00
|
|
|
graphene_rect_t cursor_rect;
|
2018-12-14 16:47:57 +00:00
|
|
|
|
|
|
|
cursor_rect = meta_cursor_renderer_calculate_rect (cursor_renderer,
|
|
|
|
cursor_sprite);
|
2019-02-20 15:23:04 +00:00
|
|
|
return graphene_rect_intersection (&cursor_rect,
|
|
|
|
&logical_monitor_rect,
|
|
|
|
NULL);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-13 10:28:46 +00:00
|
|
|
MetaCursorTracker *cursor_tracker =
|
|
|
|
meta_backend_get_cursor_tracker (backend);
|
2019-02-20 14:53:44 +00:00
|
|
|
graphene_point_t cursor_position;
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2020-07-13 10:28:46 +00:00
|
|
|
meta_cursor_tracker_get_pointer (cursor_tracker, &cursor_position, NULL);
|
2019-02-20 15:23:04 +00:00
|
|
|
return graphene_rect_contains_point (&logical_monitor_rect,
|
|
|
|
&cursor_position);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-01 09:06:13 +00:00
|
|
|
static gboolean
|
|
|
|
is_redraw_queued (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaRenderer *renderer = meta_backend_get_renderer (backend);
|
|
|
|
ClutterStage *stage = get_stage (monitor_src);
|
|
|
|
MetaMonitor *monitor = get_monitor (monitor_src);
|
|
|
|
g_autoptr (GList) views = NULL;
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
views = meta_renderer_get_views_for_monitor (renderer, monitor);
|
|
|
|
for (l = views; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = l->data;
|
|
|
|
|
|
|
|
if (clutter_stage_is_redraw_queued_on_view (stage, CLUTTER_STAGE_VIEW (view)))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static void
|
|
|
|
sync_cursor_state (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
2020-07-03 14:48:52 +00:00
|
|
|
MetaScreenCastRecordFlag flags;
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2020-04-01 09:06:13 +00:00
|
|
|
if (is_redraw_queued (monitor_src))
|
2018-12-14 16:47:57 +00:00
|
|
|
return;
|
|
|
|
|
2020-07-03 21:57:31 +00:00
|
|
|
if (meta_screen_cast_stream_src_pending_follow_up_frame (src))
|
|
|
|
return;
|
|
|
|
|
2020-07-03 14:52:43 +00:00
|
|
|
flags = META_SCREEN_CAST_RECORD_FLAG_CURSOR_ONLY;
|
2020-07-03 14:48:52 +00:00
|
|
|
meta_screen_cast_stream_src_maybe_record_frame (src, flags);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-07-13 12:09:44 +00:00
|
|
|
pointer_position_invalidated (MetaCursorTracker *cursor_tracker,
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src)
|
2018-12-14 16:47:57 +00:00
|
|
|
{
|
2022-04-29 14:05:50 +00:00
|
|
|
ClutterStage *stage = get_stage (monitor_src);
|
|
|
|
|
|
|
|
clutter_stage_schedule_update (stage);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_changed (MetaCursorTracker *cursor_tracker,
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
2019-01-03 15:51:08 +00:00
|
|
|
monitor_src->cursor_bitmap_invalid = TRUE;
|
2018-12-14 16:47:57 +00:00
|
|
|
sync_cursor_state (monitor_src);
|
|
|
|
}
|
|
|
|
|
2022-04-29 14:05:50 +00:00
|
|
|
static void
|
|
|
|
on_prepare_frame (ClutterStage *stage,
|
|
|
|
ClutterStageView *stage_view,
|
2022-10-13 15:50:34 +00:00
|
|
|
ClutterFrame *frame,
|
2022-04-29 14:05:50 +00:00
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
sync_cursor_state (monitor_src);
|
|
|
|
}
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static void
|
|
|
|
inhibit_hw_cursor (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaHwCursorInhibitor *inhibitor;
|
2020-07-13 15:45:41 +00:00
|
|
|
MetaBackend *backend;
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2020-01-17 20:08:02 +00:00
|
|
|
g_return_if_fail (!monitor_src->hw_cursor_inhibited);
|
|
|
|
|
2020-07-13 15:45:41 +00:00
|
|
|
backend = get_backend (monitor_src);
|
2018-12-14 16:47:57 +00:00
|
|
|
inhibitor = META_HW_CURSOR_INHIBITOR (monitor_src);
|
2020-07-13 15:45:41 +00:00
|
|
|
meta_backend_add_hw_cursor_inhibitor (backend, inhibitor);
|
2020-01-17 20:08:02 +00:00
|
|
|
|
|
|
|
monitor_src->hw_cursor_inhibited = TRUE;
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
uninhibit_hw_cursor (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaHwCursorInhibitor *inhibitor;
|
2020-07-13 15:45:41 +00:00
|
|
|
MetaBackend *backend;
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2020-01-17 20:08:02 +00:00
|
|
|
g_return_if_fail (monitor_src->hw_cursor_inhibited);
|
|
|
|
|
2020-07-13 15:45:41 +00:00
|
|
|
backend = get_backend (monitor_src);
|
2018-12-14 16:47:57 +00:00
|
|
|
inhibitor = META_HW_CURSOR_INHIBITOR (monitor_src);
|
2020-07-13 15:45:41 +00:00
|
|
|
meta_backend_remove_hw_cursor_inhibitor (backend, inhibitor);
|
2020-01-17 20:08:02 +00:00
|
|
|
|
|
|
|
monitor_src->hw_cursor_inhibited = FALSE;
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
static void
|
2020-08-29 18:37:38 +00:00
|
|
|
add_view_watches (MetaScreenCastMonitorStreamSrc *monitor_src,
|
|
|
|
MetaStageWatchPhase watch_phase,
|
|
|
|
MetaStageWatchFunc callback)
|
2017-06-21 06:23:44 +00:00
|
|
|
{
|
2018-12-14 16:47:57 +00:00
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
2019-06-18 00:49:45 +00:00
|
|
|
MetaRenderer *renderer = meta_backend_get_renderer (backend);
|
2020-01-17 20:08:02 +00:00
|
|
|
ClutterStage *stage;
|
|
|
|
MetaStage *meta_stage;
|
2019-06-18 00:49:45 +00:00
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
2020-01-17 20:08:02 +00:00
|
|
|
MetaRectangle logical_monitor_layout;
|
|
|
|
GList *l;
|
2017-06-21 06:23:44 +00:00
|
|
|
|
|
|
|
stage = get_stage (monitor_src);
|
2019-06-18 00:49:45 +00:00
|
|
|
meta_stage = META_STAGE (stage);
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
2020-01-17 20:08:02 +00:00
|
|
|
logical_monitor_layout = meta_logical_monitor_get_layout (logical_monitor);
|
2019-06-18 00:49:45 +00:00
|
|
|
|
2020-01-17 20:08:02 +00:00
|
|
|
for (l = meta_renderer_get_views (renderer); l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = l->data;
|
|
|
|
MetaRectangle view_layout;
|
|
|
|
|
|
|
|
clutter_stage_view_get_layout (CLUTTER_STAGE_VIEW (view), &view_layout);
|
|
|
|
if (meta_rectangle_overlap (&logical_monitor_layout, &view_layout))
|
|
|
|
{
|
|
|
|
MetaStageWatch *watch;
|
|
|
|
|
|
|
|
watch = meta_stage_watch_view (meta_stage,
|
|
|
|
CLUTTER_STAGE_VIEW (view),
|
|
|
|
watch_phase,
|
2020-08-29 18:37:38 +00:00
|
|
|
callback,
|
2020-01-17 20:08:02 +00:00
|
|
|
monitor_src);
|
|
|
|
|
|
|
|
monitor_src->watches = g_list_prepend (monitor_src->watches, watch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-27 22:25:51 +00:00
|
|
|
static void
|
|
|
|
reattach_watches (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
MetaScreenCastStream *stream;
|
|
|
|
ClutterStage *stage;
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
stage = get_stage (monitor_src);
|
|
|
|
|
|
|
|
for (l = monitor_src->watches; l; l = l->next)
|
|
|
|
meta_stage_remove_watch (META_STAGE (stage), l->data);
|
|
|
|
g_clear_pointer (&monitor_src->watches, g_list_free);
|
|
|
|
|
2021-06-25 16:29:06 +00:00
|
|
|
add_view_watches (monitor_src,
|
|
|
|
META_STAGE_WATCH_BEFORE_PAINT,
|
|
|
|
before_stage_painted);
|
|
|
|
|
2021-01-27 22:25:51 +00:00
|
|
|
switch (meta_screen_cast_stream_get_cursor_mode (stream))
|
|
|
|
{
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_METADATA:
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_HIDDEN:
|
|
|
|
add_view_watches (monitor_src,
|
|
|
|
META_STAGE_WATCH_AFTER_ACTOR_PAINT,
|
|
|
|
stage_painted);
|
|
|
|
break;
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_EMBEDDED:
|
|
|
|
add_view_watches (monitor_src,
|
|
|
|
META_STAGE_WATCH_AFTER_PAINT,
|
|
|
|
stage_painted);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_monitors_changed (MetaMonitorManager *monitor_manager,
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
reattach_watches (monitor_src);
|
|
|
|
}
|
|
|
|
|
2020-01-17 20:08:02 +00:00
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_enable (MetaScreenCastStreamSrc *src)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
2021-01-27 22:25:51 +00:00
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
2020-01-17 20:08:02 +00:00
|
|
|
MetaCursorTracker *cursor_tracker = meta_backend_get_cursor_tracker (backend);
|
2022-04-29 14:05:50 +00:00
|
|
|
ClutterStage *stage = get_stage (monitor_src);
|
2020-01-17 20:08:02 +00:00
|
|
|
MetaScreenCastStream *stream;
|
|
|
|
|
|
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
|
|
|
switch (meta_screen_cast_stream_get_cursor_mode (stream))
|
|
|
|
{
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_METADATA:
|
2020-07-13 12:09:44 +00:00
|
|
|
monitor_src->position_invalidated_handler_id =
|
|
|
|
g_signal_connect_after (cursor_tracker, "position-invalidated",
|
|
|
|
G_CALLBACK (pointer_position_invalidated),
|
2018-12-14 16:47:57 +00:00
|
|
|
monitor_src);
|
|
|
|
monitor_src->cursor_changed_handler_id =
|
|
|
|
g_signal_connect_after (cursor_tracker, "cursor-changed",
|
|
|
|
G_CALLBACK (cursor_changed),
|
|
|
|
monitor_src);
|
2022-04-29 14:05:50 +00:00
|
|
|
monitor_src->stage_prepare_frame_handler_id =
|
|
|
|
g_signal_connect_after (stage, "prepare-frame",
|
|
|
|
G_CALLBACK (on_prepare_frame),
|
|
|
|
monitor_src);
|
2020-07-29 09:43:44 +00:00
|
|
|
meta_cursor_tracker_track_position (cursor_tracker);
|
2021-01-27 22:25:51 +00:00
|
|
|
break;
|
2018-12-14 16:47:57 +00:00
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_HIDDEN:
|
|
|
|
break;
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_EMBEDDED:
|
|
|
|
inhibit_hw_cursor (monitor_src);
|
2020-07-29 09:43:44 +00:00
|
|
|
meta_cursor_tracker_track_position (cursor_tracker);
|
2018-12-14 16:47:57 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-01-27 22:25:51 +00:00
|
|
|
reattach_watches (monitor_src);
|
|
|
|
g_signal_connect_object (monitor_manager, "monitors-changed-internal",
|
|
|
|
G_CALLBACK (on_monitors_changed),
|
|
|
|
monitor_src, 0);
|
|
|
|
|
|
|
|
clutter_actor_queue_redraw (CLUTTER_ACTOR (get_stage (monitor_src)));
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2017-06-26 04:55:48 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_disable (MetaScreenCastStreamSrc *src)
|
2017-06-21 06:23:44 +00:00
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
2017-06-26 04:55:48 +00:00
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
2020-07-29 09:43:44 +00:00
|
|
|
MetaScreenCastStream *stream = meta_screen_cast_stream_src_get_stream (src);
|
2018-12-14 16:47:57 +00:00
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaCursorTracker *cursor_tracker = meta_backend_get_cursor_tracker (backend);
|
2017-06-21 06:23:44 +00:00
|
|
|
ClutterStage *stage;
|
2019-06-18 00:49:45 +00:00
|
|
|
MetaStage *meta_stage;
|
2020-01-17 20:08:02 +00:00
|
|
|
GList *l;
|
2017-06-21 06:23:44 +00:00
|
|
|
|
|
|
|
stage = get_stage (monitor_src);
|
2019-06-18 00:49:45 +00:00
|
|
|
meta_stage = META_STAGE (stage);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2020-01-17 20:08:02 +00:00
|
|
|
for (l = monitor_src->watches; l; l = l->next)
|
2018-12-14 16:47:57 +00:00
|
|
|
{
|
2020-01-17 20:08:02 +00:00
|
|
|
MetaStageWatch *watch = l->data;
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2020-01-17 20:08:02 +00:00
|
|
|
meta_stage_remove_watch (meta_stage, watch);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
2020-01-17 20:08:02 +00:00
|
|
|
g_clear_pointer (&monitor_src->watches, g_list_free);
|
|
|
|
|
|
|
|
if (monitor_src->hw_cursor_inhibited)
|
|
|
|
uninhibit_hw_cursor (monitor_src);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2020-07-13 12:09:44 +00:00
|
|
|
g_clear_signal_handler (&monitor_src->position_invalidated_handler_id,
|
2019-11-16 00:25:52 +00:00
|
|
|
cursor_tracker);
|
|
|
|
g_clear_signal_handler (&monitor_src->cursor_changed_handler_id,
|
|
|
|
cursor_tracker);
|
2022-04-29 14:05:50 +00:00
|
|
|
g_clear_signal_handler (&monitor_src->stage_prepare_frame_handler_id,
|
|
|
|
stage);
|
2020-07-29 09:43:44 +00:00
|
|
|
|
2021-06-25 16:33:28 +00:00
|
|
|
g_clear_handle_id (&monitor_src->maybe_record_idle_id, g_source_remove);
|
|
|
|
|
2020-07-29 09:43:44 +00:00
|
|
|
switch (meta_screen_cast_stream_get_cursor_mode (stream))
|
|
|
|
{
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_METADATA:
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_EMBEDDED:
|
|
|
|
meta_cursor_tracker_untrack_position (cursor_tracker);
|
|
|
|
break;
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_HIDDEN:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static gboolean
|
2020-07-03 14:57:01 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_record_to_buffer (MetaScreenCastStreamSrc *src,
|
2021-02-01 08:17:15 +00:00
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
int stride,
|
2020-07-03 14:57:01 +00:00
|
|
|
uint8_t *data,
|
|
|
|
GError **error)
|
2017-06-26 04:55:48 +00:00
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
2020-07-29 09:43:44 +00:00
|
|
|
MetaScreenCastStream *stream = meta_screen_cast_stream_src_get_stream (src);
|
2022-05-27 18:30:37 +00:00
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
2017-06-26 04:55:48 +00:00
|
|
|
ClutterStage *stage;
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
2021-05-08 11:22:27 +00:00
|
|
|
float scale;
|
2021-06-25 16:33:28 +00:00
|
|
|
ClutterPaintFlag paint_flags = CLUTTER_PAINT_FLAG_CLEAR;
|
2017-06-26 04:55:48 +00:00
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
2020-04-01 09:06:13 +00:00
|
|
|
stage = get_stage (monitor_src);
|
2021-05-08 11:22:27 +00:00
|
|
|
|
2022-05-27 18:30:37 +00:00
|
|
|
if (meta_backend_is_stage_views_scaled (backend))
|
2021-05-08 11:22:27 +00:00
|
|
|
scale = meta_logical_monitor_get_scale (logical_monitor);
|
|
|
|
else
|
|
|
|
scale = 1.0;
|
|
|
|
|
2020-07-29 09:43:44 +00:00
|
|
|
switch (meta_screen_cast_stream_get_cursor_mode (stream))
|
|
|
|
{
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_METADATA:
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_HIDDEN:
|
2021-06-25 16:33:28 +00:00
|
|
|
paint_flags |= CLUTTER_PAINT_FLAG_NO_CURSORS;
|
|
|
|
break;
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_EMBEDDED:
|
|
|
|
paint_flags |= CLUTTER_PAINT_FLAG_FORCE_CURSORS;
|
2020-07-29 09:43:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-06-25 16:33:28 +00:00
|
|
|
if (!clutter_stage_paint_to_buffer (stage, &logical_monitor->rect, scale,
|
|
|
|
data,
|
|
|
|
stride,
|
|
|
|
CLUTTER_CAIRO_FORMAT_ARGB32,
|
|
|
|
paint_flags,
|
|
|
|
error))
|
|
|
|
return FALSE;
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2019-12-09 13:11:51 +00:00
|
|
|
static gboolean
|
2020-07-03 14:57:01 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_record_to_framebuffer (MetaScreenCastStreamSrc *src,
|
|
|
|
CoglFramebuffer *framebuffer,
|
|
|
|
GError **error)
|
2019-12-09 13:11:51 +00:00
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
screen-cast/monitor: Always use clutter_stage_paint_to_framebuffer()
'screen-cast/monitor-src: Use clutter_stage_paint_to_buffer'
(6c818cd8d5d29a21288d64fcbb13eae6a69d65df) made the non-dma-buf path use
clutter_stage_paint_to_buffer() to avoid running into direct scanout
issues. At a glance, the dma-buf paths didn't have the same issue since
it explicitly handled dma-bufs by blitting them.
What it also did was move the recording to an idle callback, to avoid
paint reentry issues. A side effect of this, however, is that it also
broke the dma-buf paths, as they rely on the back buffer existing, and
the stage view direct scanout already being setup, which it isn't in an
idle callback.
Fix this by using the dma-buf variant of
clutter_stage_paint_to_buffer(): clutter_stage_paint_to_framebuffer().
This has some negative performance impact, but we can't use
cogl_blit_framebuffer() when using an idle callback for recording.
Potential performance improvements to make things work more as they did
before is to enhance 'cogl_blit_framebuffer()' a bit, making it a vfunc
that could be implemented by MetaOnscreenNative. A flag to say whether
to look at the back or front buffer would let MetaOnscreenNative know
whether to use the already committed-to-KMS buffer, or the current back
buffer.
Fixes: 6c818cd8d5d29a21288d64fcbb13eae6a69d65df
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2282
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2462>
2022-06-13 16:41:17 +00:00
|
|
|
MetaScreenCastStream *stream = meta_screen_cast_stream_src_get_stream (src);
|
2022-05-27 18:30:37 +00:00
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
screen-cast/monitor: Always use clutter_stage_paint_to_framebuffer()
'screen-cast/monitor-src: Use clutter_stage_paint_to_buffer'
(6c818cd8d5d29a21288d64fcbb13eae6a69d65df) made the non-dma-buf path use
clutter_stage_paint_to_buffer() to avoid running into direct scanout
issues. At a glance, the dma-buf paths didn't have the same issue since
it explicitly handled dma-bufs by blitting them.
What it also did was move the recording to an idle callback, to avoid
paint reentry issues. A side effect of this, however, is that it also
broke the dma-buf paths, as they rely on the back buffer existing, and
the stage view direct scanout already being setup, which it isn't in an
idle callback.
Fix this by using the dma-buf variant of
clutter_stage_paint_to_buffer(): clutter_stage_paint_to_framebuffer().
This has some negative performance impact, but we can't use
cogl_blit_framebuffer() when using an idle callback for recording.
Potential performance improvements to make things work more as they did
before is to enhance 'cogl_blit_framebuffer()' a bit, making it a vfunc
that could be implemented by MetaOnscreenNative. A flag to say whether
to look at the back or front buffer would let MetaOnscreenNative know
whether to use the already committed-to-KMS buffer, or the current back
buffer.
Fixes: 6c818cd8d5d29a21288d64fcbb13eae6a69d65df
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2282
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2462>
2022-06-13 16:41:17 +00:00
|
|
|
ClutterStage *stage = get_stage (monitor_src);
|
2019-12-09 13:11:51 +00:00
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
|
|
|
float view_scale;
|
screen-cast/monitor: Always use clutter_stage_paint_to_framebuffer()
'screen-cast/monitor-src: Use clutter_stage_paint_to_buffer'
(6c818cd8d5d29a21288d64fcbb13eae6a69d65df) made the non-dma-buf path use
clutter_stage_paint_to_buffer() to avoid running into direct scanout
issues. At a glance, the dma-buf paths didn't have the same issue since
it explicitly handled dma-bufs by blitting them.
What it also did was move the recording to an idle callback, to avoid
paint reentry issues. A side effect of this, however, is that it also
broke the dma-buf paths, as they rely on the back buffer existing, and
the stage view direct scanout already being setup, which it isn't in an
idle callback.
Fix this by using the dma-buf variant of
clutter_stage_paint_to_buffer(): clutter_stage_paint_to_framebuffer().
This has some negative performance impact, but we can't use
cogl_blit_framebuffer() when using an idle callback for recording.
Potential performance improvements to make things work more as they did
before is to enhance 'cogl_blit_framebuffer()' a bit, making it a vfunc
that could be implemented by MetaOnscreenNative. A flag to say whether
to look at the back or front buffer would let MetaOnscreenNative know
whether to use the already committed-to-KMS buffer, or the current back
buffer.
Fixes: 6c818cd8d5d29a21288d64fcbb13eae6a69d65df
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2282
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2462>
2022-06-13 16:41:17 +00:00
|
|
|
ClutterPaintFlag paint_flags = CLUTTER_PAINT_FLAG_CLEAR;
|
2019-12-09 13:11:51 +00:00
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
logical_monitor_layout = meta_logical_monitor_get_layout (logical_monitor);
|
|
|
|
|
2022-05-27 18:30:37 +00:00
|
|
|
if (meta_backend_is_stage_views_scaled (backend))
|
2019-12-09 13:11:51 +00:00
|
|
|
view_scale = meta_logical_monitor_get_scale (logical_monitor);
|
|
|
|
else
|
|
|
|
view_scale = 1.0;
|
|
|
|
|
screen-cast/monitor: Always use clutter_stage_paint_to_framebuffer()
'screen-cast/monitor-src: Use clutter_stage_paint_to_buffer'
(6c818cd8d5d29a21288d64fcbb13eae6a69d65df) made the non-dma-buf path use
clutter_stage_paint_to_buffer() to avoid running into direct scanout
issues. At a glance, the dma-buf paths didn't have the same issue since
it explicitly handled dma-bufs by blitting them.
What it also did was move the recording to an idle callback, to avoid
paint reentry issues. A side effect of this, however, is that it also
broke the dma-buf paths, as they rely on the back buffer existing, and
the stage view direct scanout already being setup, which it isn't in an
idle callback.
Fix this by using the dma-buf variant of
clutter_stage_paint_to_buffer(): clutter_stage_paint_to_framebuffer().
This has some negative performance impact, but we can't use
cogl_blit_framebuffer() when using an idle callback for recording.
Potential performance improvements to make things work more as they did
before is to enhance 'cogl_blit_framebuffer()' a bit, making it a vfunc
that could be implemented by MetaOnscreenNative. A flag to say whether
to look at the back or front buffer would let MetaOnscreenNative know
whether to use the already committed-to-KMS buffer, or the current back
buffer.
Fixes: 6c818cd8d5d29a21288d64fcbb13eae6a69d65df
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2282
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2462>
2022-06-13 16:41:17 +00:00
|
|
|
switch (meta_screen_cast_stream_get_cursor_mode (stream))
|
2019-12-09 13:11:51 +00:00
|
|
|
{
|
screen-cast/monitor: Always use clutter_stage_paint_to_framebuffer()
'screen-cast/monitor-src: Use clutter_stage_paint_to_buffer'
(6c818cd8d5d29a21288d64fcbb13eae6a69d65df) made the non-dma-buf path use
clutter_stage_paint_to_buffer() to avoid running into direct scanout
issues. At a glance, the dma-buf paths didn't have the same issue since
it explicitly handled dma-bufs by blitting them.
What it also did was move the recording to an idle callback, to avoid
paint reentry issues. A side effect of this, however, is that it also
broke the dma-buf paths, as they rely on the back buffer existing, and
the stage view direct scanout already being setup, which it isn't in an
idle callback.
Fix this by using the dma-buf variant of
clutter_stage_paint_to_buffer(): clutter_stage_paint_to_framebuffer().
This has some negative performance impact, but we can't use
cogl_blit_framebuffer() when using an idle callback for recording.
Potential performance improvements to make things work more as they did
before is to enhance 'cogl_blit_framebuffer()' a bit, making it a vfunc
that could be implemented by MetaOnscreenNative. A flag to say whether
to look at the back or front buffer would let MetaOnscreenNative know
whether to use the already committed-to-KMS buffer, or the current back
buffer.
Fixes: 6c818cd8d5d29a21288d64fcbb13eae6a69d65df
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2282
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2462>
2022-06-13 16:41:17 +00:00
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_METADATA:
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_HIDDEN:
|
|
|
|
paint_flags |= CLUTTER_PAINT_FLAG_NO_CURSORS;
|
|
|
|
break;
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_EMBEDDED:
|
|
|
|
paint_flags |= CLUTTER_PAINT_FLAG_FORCE_CURSORS;
|
|
|
|
break;
|
|
|
|
}
|
2019-12-09 13:11:51 +00:00
|
|
|
|
screen-cast/monitor: Always use clutter_stage_paint_to_framebuffer()
'screen-cast/monitor-src: Use clutter_stage_paint_to_buffer'
(6c818cd8d5d29a21288d64fcbb13eae6a69d65df) made the non-dma-buf path use
clutter_stage_paint_to_buffer() to avoid running into direct scanout
issues. At a glance, the dma-buf paths didn't have the same issue since
it explicitly handled dma-bufs by blitting them.
What it also did was move the recording to an idle callback, to avoid
paint reentry issues. A side effect of this, however, is that it also
broke the dma-buf paths, as they rely on the back buffer existing, and
the stage view direct scanout already being setup, which it isn't in an
idle callback.
Fix this by using the dma-buf variant of
clutter_stage_paint_to_buffer(): clutter_stage_paint_to_framebuffer().
This has some negative performance impact, but we can't use
cogl_blit_framebuffer() when using an idle callback for recording.
Potential performance improvements to make things work more as they did
before is to enhance 'cogl_blit_framebuffer()' a bit, making it a vfunc
that could be implemented by MetaOnscreenNative. A flag to say whether
to look at the back or front buffer would let MetaOnscreenNative know
whether to use the already committed-to-KMS buffer, or the current back
buffer.
Fixes: 6c818cd8d5d29a21288d64fcbb13eae6a69d65df
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2282
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2462>
2022-06-13 16:41:17 +00:00
|
|
|
clutter_stage_paint_to_framebuffer (stage,
|
2020-08-29 02:01:11 +00:00
|
|
|
framebuffer,
|
screen-cast/monitor: Always use clutter_stage_paint_to_framebuffer()
'screen-cast/monitor-src: Use clutter_stage_paint_to_buffer'
(6c818cd8d5d29a21288d64fcbb13eae6a69d65df) made the non-dma-buf path use
clutter_stage_paint_to_buffer() to avoid running into direct scanout
issues. At a glance, the dma-buf paths didn't have the same issue since
it explicitly handled dma-bufs by blitting them.
What it also did was move the recording to an idle callback, to avoid
paint reentry issues. A side effect of this, however, is that it also
broke the dma-buf paths, as they rely on the back buffer existing, and
the stage view direct scanout already being setup, which it isn't in an
idle callback.
Fix this by using the dma-buf variant of
clutter_stage_paint_to_buffer(): clutter_stage_paint_to_framebuffer().
This has some negative performance impact, but we can't use
cogl_blit_framebuffer() when using an idle callback for recording.
Potential performance improvements to make things work more as they did
before is to enhance 'cogl_blit_framebuffer()' a bit, making it a vfunc
that could be implemented by MetaOnscreenNative. A flag to say whether
to look at the back or front buffer would let MetaOnscreenNative know
whether to use the already committed-to-KMS buffer, or the current back
buffer.
Fixes: 6c818cd8d5d29a21288d64fcbb13eae6a69d65df
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2282
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2462>
2022-06-13 16:41:17 +00:00
|
|
|
&logical_monitor_layout,
|
|
|
|
view_scale,
|
|
|
|
paint_flags);
|
2019-12-09 13:11:51 +00:00
|
|
|
|
2021-01-27 11:50:46 +00:00
|
|
|
cogl_framebuffer_flush (framebuffer);
|
2019-12-09 13:11:51 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-07-03 21:57:31 +00:00
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_record_follow_up (MetaScreenCastStreamSrc *src)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaRenderer *renderer = meta_backend_get_renderer (backend);
|
|
|
|
ClutterStage *stage = get_stage (monitor_src);
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
|
|
|
GList *l;
|
|
|
|
|
2021-06-25 16:33:28 +00:00
|
|
|
g_clear_handle_id (&monitor_src->maybe_record_idle_id, g_source_remove);
|
|
|
|
|
2020-07-03 21:57:31 +00:00
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
logical_monitor_layout = meta_logical_monitor_get_layout (logical_monitor);
|
|
|
|
|
|
|
|
for (l = meta_renderer_get_views (renderer); l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = l->data;
|
|
|
|
MetaRectangle view_layout;
|
|
|
|
MetaRectangle damage;
|
|
|
|
|
|
|
|
clutter_stage_view_get_layout (CLUTTER_STAGE_VIEW (view), &view_layout);
|
|
|
|
|
|
|
|
if (!meta_rectangle_overlap (&logical_monitor_layout, &view_layout))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
damage = (cairo_rectangle_int_t) {
|
|
|
|
.x = view_layout.x,
|
|
|
|
.y = view_layout.y,
|
|
|
|
.width = 1,
|
|
|
|
.height = 1,
|
|
|
|
};
|
|
|
|
clutter_actor_queue_redraw_with_clip (CLUTTER_ACTOR (stage), &damage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_set_cursor_metadata (MetaScreenCastStreamSrc *src,
|
|
|
|
struct spa_meta_cursor *spa_meta_cursor)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaCursorRenderer *cursor_renderer =
|
|
|
|
meta_backend_get_cursor_renderer (backend);
|
2020-09-01 13:25:02 +00:00
|
|
|
MetaCursorTracker *cursor_tracker =
|
|
|
|
meta_backend_get_cursor_tracker (backend);
|
2018-12-14 16:47:57 +00:00
|
|
|
MetaCursorSprite *cursor_sprite;
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
2019-02-20 15:23:04 +00:00
|
|
|
graphene_rect_t logical_monitor_rect;
|
2018-12-14 16:47:57 +00:00
|
|
|
float view_scale;
|
2019-02-20 14:53:44 +00:00
|
|
|
graphene_point_t cursor_position;
|
2019-01-21 13:32:05 +00:00
|
|
|
int x, y;
|
2018-12-14 16:47:57 +00:00
|
|
|
|
|
|
|
cursor_sprite = meta_cursor_renderer_get_cursor (cursor_renderer);
|
|
|
|
|
2020-09-01 13:25:02 +00:00
|
|
|
if (!meta_cursor_tracker_get_pointer_visible (cursor_tracker) ||
|
|
|
|
!is_cursor_in_stream (monitor_src))
|
2018-12-14 16:47:57 +00:00
|
|
|
{
|
2019-01-21 13:32:05 +00:00
|
|
|
meta_screen_cast_stream_src_unset_cursor_metadata (src,
|
|
|
|
spa_meta_cursor);
|
2018-12-14 16:47:57 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
logical_monitor_layout = meta_logical_monitor_get_layout (logical_monitor);
|
|
|
|
logical_monitor_rect =
|
2019-02-20 15:23:04 +00:00
|
|
|
meta_rectangle_to_graphene_rect (&logical_monitor_layout);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
2022-05-27 18:30:37 +00:00
|
|
|
if (meta_backend_is_stage_views_scaled (backend))
|
2020-01-17 20:08:02 +00:00
|
|
|
view_scale = meta_logical_monitor_get_scale (logical_monitor);
|
2018-12-14 16:47:57 +00:00
|
|
|
else
|
|
|
|
view_scale = 1.0;
|
|
|
|
|
2020-07-13 10:28:46 +00:00
|
|
|
meta_cursor_tracker_get_pointer (cursor_tracker, &cursor_position, NULL);
|
2018-12-14 16:47:57 +00:00
|
|
|
cursor_position.x -= logical_monitor_rect.origin.x;
|
|
|
|
cursor_position.y -= logical_monitor_rect.origin.y;
|
|
|
|
cursor_position.x *= view_scale;
|
|
|
|
cursor_position.y *= view_scale;
|
|
|
|
|
2019-01-21 13:32:05 +00:00
|
|
|
x = (int) roundf (cursor_position.x);
|
|
|
|
y = (int) roundf (cursor_position.y);
|
2019-01-03 15:51:08 +00:00
|
|
|
|
2019-01-21 13:32:05 +00:00
|
|
|
if (monitor_src->cursor_bitmap_invalid)
|
2019-01-03 15:51:08 +00:00
|
|
|
{
|
2019-01-21 13:32:05 +00:00
|
|
|
if (cursor_sprite)
|
2018-12-14 16:47:57 +00:00
|
|
|
{
|
2019-01-21 13:32:05 +00:00
|
|
|
float cursor_scale;
|
|
|
|
float scale;
|
2022-07-02 16:50:45 +00:00
|
|
|
MetaMonitorTransform transform;
|
2019-01-21 13:32:05 +00:00
|
|
|
|
|
|
|
cursor_scale = meta_cursor_sprite_get_texture_scale (cursor_sprite);
|
|
|
|
scale = view_scale * cursor_scale;
|
2022-07-02 16:50:45 +00:00
|
|
|
transform = meta_cursor_sprite_get_texture_transform (cursor_sprite);
|
|
|
|
|
2019-01-21 13:32:05 +00:00
|
|
|
meta_screen_cast_stream_src_set_cursor_sprite_metadata (src,
|
|
|
|
spa_meta_cursor,
|
|
|
|
cursor_sprite,
|
|
|
|
x, y,
|
2022-07-02 16:50:45 +00:00
|
|
|
scale,
|
|
|
|
transform);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-01-21 13:32:05 +00:00
|
|
|
meta_screen_cast_stream_src_set_empty_cursor_sprite_metadata (src,
|
|
|
|
spa_meta_cursor,
|
|
|
|
x, y);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
2019-01-21 13:32:05 +00:00
|
|
|
|
|
|
|
monitor_src->cursor_bitmap_invalid = FALSE;
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-01-21 13:32:05 +00:00
|
|
|
meta_screen_cast_stream_src_set_cursor_position_metadata (src,
|
|
|
|
spa_meta_cursor,
|
|
|
|
x, y);
|
2018-12-14 16:47:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2020-07-13 15:29:51 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_is_cursor_inhibited (MetaHwCursorInhibitor *inhibitor)
|
2018-12-14 16:47:57 +00:00
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (inhibitor);
|
|
|
|
|
|
|
|
return is_cursor_in_stream (monitor_src);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
hw_cursor_inhibitor_iface_init (MetaHwCursorInhibitorInterface *iface)
|
|
|
|
{
|
2020-07-13 15:29:51 +00:00
|
|
|
iface->is_cursor_inhibited =
|
|
|
|
meta_screen_cast_monitor_stream_src_is_cursor_inhibited;
|
2017-06-26 04:55:48 +00:00
|
|
|
}
|
2017-06-21 06:23:44 +00:00
|
|
|
|
2017-06-26 04:55:48 +00:00
|
|
|
MetaScreenCastMonitorStreamSrc *
|
|
|
|
meta_screen_cast_monitor_stream_src_new (MetaScreenCastMonitorStream *monitor_stream,
|
|
|
|
GError **error)
|
|
|
|
{
|
2017-06-28 07:17:32 +00:00
|
|
|
return g_initable_new (META_TYPE_SCREEN_CAST_MONITOR_STREAM_SRC, NULL, error,
|
2017-06-26 04:55:48 +00:00
|
|
|
"stream", monitor_stream,
|
|
|
|
NULL);
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_init (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
2019-01-03 15:51:08 +00:00
|
|
|
monitor_src->cursor_bitmap_invalid = TRUE;
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_class_init (MetaScreenCastMonitorStreamSrcClass *klass)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrcClass *src_class =
|
|
|
|
META_SCREEN_CAST_STREAM_SRC_CLASS (klass);
|
|
|
|
|
|
|
|
src_class->get_specs = meta_screen_cast_monitor_stream_src_get_specs;
|
2017-06-26 04:55:48 +00:00
|
|
|
src_class->enable = meta_screen_cast_monitor_stream_src_enable;
|
|
|
|
src_class->disable = meta_screen_cast_monitor_stream_src_disable;
|
2020-07-03 14:42:45 +00:00
|
|
|
src_class->record_to_buffer =
|
|
|
|
meta_screen_cast_monitor_stream_src_record_to_buffer;
|
|
|
|
src_class->record_to_framebuffer =
|
|
|
|
meta_screen_cast_monitor_stream_src_record_to_framebuffer;
|
2020-07-03 21:57:31 +00:00
|
|
|
src_class->record_follow_up =
|
|
|
|
meta_screen_cast_monitor_stream_record_follow_up;
|
2018-12-14 16:47:57 +00:00
|
|
|
src_class->set_cursor_metadata =
|
|
|
|
meta_screen_cast_monitor_stream_src_set_cursor_metadata;
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|