2022-12-21 18:54:17 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2021 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
|
2023-08-07 11:50:23 +02:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2022-12-21 18:54:17 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "backends/native/meta-kms-cursor-manager.h"
|
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
|
|
|
|
#include "backends/native/meta-kms-crtc.h"
|
|
|
|
#include "backends/native/meta-kms-impl.h"
|
|
|
|
#include "backends/native/meta-kms-device-private.h"
|
|
|
|
#include "backends/native/meta-kms-update-private.h"
|
|
|
|
#include "backends/native/meta-thread.h"
|
|
|
|
|
|
|
|
typedef struct _MetaKmsCursorManagerImpl
|
|
|
|
{
|
|
|
|
MetaKmsImpl *impl;
|
|
|
|
GPtrArray *crtc_states;
|
|
|
|
|
|
|
|
MetaKmsCursorQueryInImpl cursor_query_in_impl_func;
|
|
|
|
gpointer cursor_query_in_impl_func_user_data;
|
|
|
|
|
|
|
|
MetaKmsUpdateFilter *update_filter;
|
|
|
|
} MetaKmsCursorManagerImpl;
|
|
|
|
|
|
|
|
typedef struct _CrtcStateImpl
|
|
|
|
{
|
|
|
|
gatomicrefcount ref_count;
|
|
|
|
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl;
|
|
|
|
|
|
|
|
MetaKmsCrtc *crtc;
|
|
|
|
graphene_rect_t layout;
|
|
|
|
float scale;
|
2023-08-14 18:05:41 +08:00
|
|
|
MetaMonitorTransform transform;
|
2022-12-21 18:54:17 +01:00
|
|
|
MetaDrmBuffer *buffer;
|
|
|
|
graphene_point_t hotspot;
|
|
|
|
|
|
|
|
gboolean cursor_invalidated;
|
|
|
|
gboolean force_update;
|
|
|
|
gboolean has_cursor;
|
|
|
|
|
|
|
|
graphene_point_t pending_hotspot;
|
|
|
|
MetaDrmBuffer *pending_buffer;
|
|
|
|
MetaDrmBuffer *active_buffer;
|
|
|
|
MetaDrmBuffer *presenting_buffer;
|
|
|
|
} CrtcStateImpl;
|
|
|
|
|
|
|
|
struct _MetaKmsCursorManager
|
|
|
|
{
|
|
|
|
GObject parent;
|
|
|
|
|
|
|
|
MetaKms *kms;
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (MetaKmsCursorManager, meta_kms_cursor_manager,
|
|
|
|
G_TYPE_OBJECT)
|
|
|
|
|
|
|
|
static GQuark quark_cursor_manager_impl;
|
|
|
|
|
|
|
|
static CrtcStateImpl *
|
|
|
|
find_crtc_state (MetaKmsCursorManagerImpl *cursor_manager_impl,
|
|
|
|
MetaKmsCrtc *crtc)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!cursor_manager_impl->crtc_states)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < cursor_manager_impl->crtc_states->len; i++)
|
|
|
|
{
|
|
|
|
CrtcStateImpl *crtc_state_impl =
|
|
|
|
g_ptr_array_index (cursor_manager_impl->crtc_states, i);
|
|
|
|
|
|
|
|
if (crtc_state_impl->crtc == crtc)
|
|
|
|
return crtc_state_impl;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CrtcStateImpl *
|
|
|
|
crtc_state_impl_new (MetaKmsCursorManagerImpl *cursor_manager_impl,
|
|
|
|
MetaKmsCrtc *crtc,
|
|
|
|
graphene_rect_t layout,
|
|
|
|
float scale,
|
|
|
|
MetaDrmBuffer *buffer)
|
|
|
|
{
|
|
|
|
CrtcStateImpl *crtc_state_impl;
|
|
|
|
|
|
|
|
crtc_state_impl = g_new0 (CrtcStateImpl, 1);
|
|
|
|
g_atomic_ref_count_init (&crtc_state_impl->ref_count);
|
|
|
|
crtc_state_impl->cursor_manager_impl = cursor_manager_impl;
|
|
|
|
crtc_state_impl->crtc = crtc;
|
|
|
|
crtc_state_impl->layout = layout;
|
|
|
|
crtc_state_impl->scale = scale;
|
|
|
|
crtc_state_impl->buffer = buffer;
|
|
|
|
|
|
|
|
return crtc_state_impl;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CrtcStateImpl *
|
|
|
|
crtc_state_impl_ref (CrtcStateImpl *crtc_state_impl)
|
|
|
|
{
|
|
|
|
g_atomic_ref_count_inc (&crtc_state_impl->ref_count);
|
|
|
|
return crtc_state_impl;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
crtc_state_impl_unref (CrtcStateImpl *crtc_state_impl)
|
|
|
|
{
|
|
|
|
if (g_atomic_ref_count_dec (&crtc_state_impl->ref_count))
|
|
|
|
{
|
|
|
|
g_warn_if_fail (!crtc_state_impl->buffer);
|
|
|
|
g_warn_if_fail (!crtc_state_impl->pending_buffer);
|
|
|
|
g_warn_if_fail (!crtc_state_impl->active_buffer);
|
|
|
|
g_warn_if_fail (!crtc_state_impl->presenting_buffer);
|
|
|
|
g_free (crtc_state_impl);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
crtc_state_impl_swap_buffer (CrtcStateImpl *crtc_state_impl,
|
|
|
|
MetaDrmBuffer **from_buffer_ref,
|
|
|
|
MetaDrmBuffer **to_buffer_ref)
|
|
|
|
{
|
|
|
|
MetaDrmBuffer *buffer_to_release;
|
|
|
|
|
|
|
|
if (*from_buffer_ref == *to_buffer_ref)
|
|
|
|
{
|
|
|
|
buffer_to_release = g_steal_pointer (from_buffer_ref);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
buffer_to_release = g_steal_pointer (to_buffer_ref);
|
|
|
|
*to_buffer_ref = g_steal_pointer (from_buffer_ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buffer_to_release)
|
|
|
|
{
|
|
|
|
MetaKmsDevice *device = meta_kms_crtc_get_device (crtc_state_impl->crtc);
|
|
|
|
MetaKms *kms = meta_kms_device_get_kms (device);
|
|
|
|
|
|
|
|
meta_thread_queue_callback (META_THREAD (kms),
|
|
|
|
NULL, NULL,
|
|
|
|
buffer_to_release,
|
|
|
|
g_object_unref);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
crtc_state_impl_notify_presented (CrtcStateImpl *crtc_state_impl)
|
|
|
|
{
|
|
|
|
crtc_state_impl_swap_buffer (crtc_state_impl,
|
|
|
|
&crtc_state_impl->active_buffer,
|
|
|
|
&crtc_state_impl->presenting_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_page_flip_feedback_flipped (MetaKmsCrtc *crtc,
|
|
|
|
unsigned int sequence,
|
|
|
|
unsigned int tv_sec,
|
|
|
|
unsigned int tv_usec,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
CrtcStateImpl *crtc_state_impl = user_data;
|
|
|
|
|
|
|
|
crtc_state_impl_notify_presented (crtc_state_impl);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_page_flip_feedback_ready (MetaKmsCrtc *crtc,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_page_flip_feedback_mode_set_fallback (MetaKmsCrtc *crtc,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
CrtcStateImpl *crtc_state_impl = user_data;
|
|
|
|
|
|
|
|
crtc_state_impl_notify_presented (crtc_state_impl);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_page_flip_feedback_discarded (MetaKmsCrtc *crtc,
|
|
|
|
gpointer user_data,
|
|
|
|
const GError *error)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static const MetaKmsPageFlipListenerVtable cursor_page_flip_listener_vtable = {
|
|
|
|
.flipped = cursor_page_flip_feedback_flipped,
|
|
|
|
.ready = cursor_page_flip_feedback_ready,
|
|
|
|
.mode_set_fallback = cursor_page_flip_feedback_mode_set_fallback,
|
|
|
|
.discarded = cursor_page_flip_feedback_discarded,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_result_feedback (const MetaKmsFeedback *feedback,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
CrtcStateImpl *crtc_state_impl = user_data;
|
|
|
|
|
|
|
|
switch (meta_kms_feedback_get_result (feedback))
|
|
|
|
{
|
|
|
|
case META_KMS_FEEDBACK_PASSED:
|
|
|
|
break;
|
|
|
|
case META_KMS_FEEDBACK_FAILED:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
crtc_state_impl->cursor_invalidated = FALSE;
|
|
|
|
|
|
|
|
crtc_state_impl_swap_buffer (crtc_state_impl,
|
|
|
|
&crtc_state_impl->pending_buffer,
|
|
|
|
&crtc_state_impl->active_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const MetaKmsResultListenerVtable cursor_result_listener_vtable = {
|
|
|
|
.feedback = cursor_result_feedback,
|
|
|
|
};
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
get_current_cursor_position (MetaKmsCursorManagerImpl *cursor_manager_impl,
|
|
|
|
float *x,
|
|
|
|
float *y)
|
|
|
|
{
|
|
|
|
gpointer user_data;
|
|
|
|
|
|
|
|
if (!cursor_manager_impl->cursor_query_in_impl_func)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
user_data = cursor_manager_impl->cursor_query_in_impl_func_user_data;
|
|
|
|
cursor_manager_impl->cursor_query_in_impl_func (x, y, user_data);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
calculate_cursor_rect (CrtcStateImpl *crtc_state_impl,
|
|
|
|
MetaDrmBuffer *buffer,
|
|
|
|
const graphene_point_t *hotspot,
|
|
|
|
float x,
|
|
|
|
float y,
|
|
|
|
graphene_rect_t *out_cursor_rect)
|
|
|
|
{
|
2023-08-14 18:21:44 +08:00
|
|
|
int crtc_x, crtc_y, crtc_width, crtc_height;
|
2022-12-21 18:54:17 +01:00
|
|
|
int buffer_width, buffer_height;
|
|
|
|
graphene_rect_t cursor_rect;
|
|
|
|
|
|
|
|
crtc_x = (x - crtc_state_impl->layout.origin.x) * crtc_state_impl->scale;
|
|
|
|
crtc_y = (y - crtc_state_impl->layout.origin.y) * crtc_state_impl->scale;
|
2023-08-14 18:21:44 +08:00
|
|
|
crtc_width = roundf (crtc_state_impl->layout.size.width *
|
|
|
|
crtc_state_impl->scale);
|
|
|
|
crtc_height = roundf (crtc_state_impl->layout.size.height *
|
|
|
|
crtc_state_impl->scale);
|
|
|
|
|
|
|
|
meta_monitor_transform_transform_point (crtc_state_impl->transform,
|
|
|
|
&crtc_width, &crtc_height,
|
|
|
|
&crtc_x, &crtc_y);
|
2022-12-21 18:54:17 +01:00
|
|
|
|
|
|
|
buffer_width = meta_drm_buffer_get_width (buffer);
|
|
|
|
buffer_height = meta_drm_buffer_get_height (buffer);
|
|
|
|
|
|
|
|
cursor_rect = (graphene_rect_t) {
|
|
|
|
.origin = {
|
|
|
|
.x = crtc_x - hotspot->x,
|
|
|
|
.y = crtc_y - hotspot->y,
|
|
|
|
},
|
|
|
|
.size = {
|
|
|
|
.width = buffer_width,
|
|
|
|
.height = buffer_height,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
if (cursor_rect.origin.x + cursor_rect.size.width > 0.0 &&
|
|
|
|
cursor_rect.origin.y + cursor_rect.size.height > 0.0 &&
|
2023-08-14 18:21:44 +08:00
|
|
|
cursor_rect.origin.x < crtc_width &&
|
|
|
|
cursor_rect.origin.y < crtc_height)
|
2022-12-21 18:54:17 +01:00
|
|
|
{
|
|
|
|
if (out_cursor_rect)
|
|
|
|
*out_cursor_rect = cursor_rect;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaKmsUpdate *
|
|
|
|
maybe_update_cursor_plane (MetaKmsCursorManagerImpl *cursor_manager_impl,
|
|
|
|
MetaKmsCrtc *crtc,
|
|
|
|
MetaKmsUpdate *update,
|
|
|
|
MetaDrmBuffer **old_buffer)
|
|
|
|
{
|
|
|
|
MetaKmsImpl *impl = cursor_manager_impl->impl;
|
|
|
|
MetaThreadImpl *thread_impl = META_THREAD_IMPL (impl);
|
|
|
|
MetaKmsDevice *device;
|
|
|
|
CrtcStateImpl *crtc_state_impl;
|
|
|
|
float x, y;
|
|
|
|
MetaDrmBuffer *buffer;
|
|
|
|
const graphene_point_t *hotspot;
|
|
|
|
gboolean should_have_cursor;
|
|
|
|
gboolean did_have_cursor;
|
|
|
|
graphene_rect_t cursor_rect;
|
|
|
|
MetaKmsPlane *cursor_plane;
|
|
|
|
|
|
|
|
g_assert (old_buffer && !*old_buffer);
|
|
|
|
|
|
|
|
if (!get_current_cursor_position (cursor_manager_impl, &x, &y))
|
|
|
|
return update;
|
|
|
|
|
|
|
|
crtc_state_impl = find_crtc_state (cursor_manager_impl, crtc);
|
|
|
|
g_return_val_if_fail (crtc_state_impl, update);
|
|
|
|
|
|
|
|
if (!crtc_state_impl->cursor_invalidated)
|
|
|
|
return update;
|
|
|
|
|
|
|
|
device = meta_kms_crtc_get_device (crtc_state_impl->crtc);
|
|
|
|
buffer = crtc_state_impl->buffer;
|
|
|
|
hotspot = &crtc_state_impl->hotspot;
|
|
|
|
|
|
|
|
if (buffer)
|
|
|
|
{
|
|
|
|
should_have_cursor = calculate_cursor_rect (crtc_state_impl,
|
|
|
|
buffer,
|
|
|
|
hotspot,
|
|
|
|
x, y,
|
|
|
|
&cursor_rect);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
should_have_cursor = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
did_have_cursor = crtc_state_impl->has_cursor;
|
|
|
|
crtc_state_impl->has_cursor = should_have_cursor;
|
|
|
|
|
|
|
|
if (!should_have_cursor && !did_have_cursor)
|
|
|
|
return update;
|
|
|
|
|
|
|
|
if (!update)
|
|
|
|
{
|
|
|
|
MetaKmsImplDevice *impl_device =
|
|
|
|
meta_kms_device_get_impl_device (device);
|
|
|
|
|
|
|
|
update = meta_kms_update_new (device);
|
|
|
|
meta_kms_update_realize (update, impl_device);
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor_plane = meta_kms_device_get_cursor_plane_for (device, crtc);
|
|
|
|
|
|
|
|
if (should_have_cursor)
|
|
|
|
{
|
|
|
|
int width, height;
|
|
|
|
MetaFixed16Rectangle src_rect;
|
|
|
|
MetaRectangle dst_rect;
|
|
|
|
MetaKmsAssignPlaneFlag assign_plane_flags =
|
|
|
|
META_KMS_ASSIGN_PLANE_FLAG_NONE;
|
2023-08-04 10:17:17 +02:00
|
|
|
MetaKmsPlaneAssignment *plane_assignment;
|
2022-12-21 18:54:17 +01:00
|
|
|
|
|
|
|
if (crtc_state_impl->pending_buffer != crtc_state_impl->buffer)
|
|
|
|
{
|
|
|
|
*old_buffer = g_steal_pointer (&crtc_state_impl->pending_buffer);
|
|
|
|
crtc_state_impl->pending_buffer = g_object_ref (buffer);
|
|
|
|
}
|
2023-08-02 16:15:11 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
assign_plane_flags |= META_KMS_ASSIGN_PLANE_FLAG_FB_UNCHANGED;
|
|
|
|
}
|
2022-12-21 18:54:17 +01:00
|
|
|
|
|
|
|
width = meta_drm_buffer_get_width (buffer);
|
|
|
|
height = meta_drm_buffer_get_height (buffer);
|
|
|
|
|
|
|
|
src_rect = (MetaFixed16Rectangle) {
|
|
|
|
.x = meta_fixed_16_from_int (0),
|
|
|
|
.y = meta_fixed_16_from_int (0),
|
|
|
|
.width = meta_fixed_16_from_int (width),
|
|
|
|
.height = meta_fixed_16_from_int (height),
|
|
|
|
};
|
|
|
|
dst_rect = (MetaRectangle) {
|
|
|
|
.x = round (cursor_rect.origin.x),
|
|
|
|
.y = round (cursor_rect.origin.y),
|
|
|
|
.width = round (cursor_rect.size.width),
|
|
|
|
.height = round (cursor_rect.size.height),
|
|
|
|
};
|
|
|
|
|
2023-08-04 10:17:17 +02:00
|
|
|
plane_assignment = meta_kms_update_assign_plane (update,
|
|
|
|
crtc, cursor_plane,
|
|
|
|
buffer,
|
|
|
|
src_rect, dst_rect,
|
|
|
|
assign_plane_flags);
|
|
|
|
meta_kms_plane_assignment_set_cursor_hotspot (plane_assignment,
|
|
|
|
(int) roundf (hotspot->x),
|
|
|
|
(int) roundf (hotspot->y));
|
2022-12-21 18:54:17 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*old_buffer = g_steal_pointer (&crtc_state_impl->pending_buffer);
|
|
|
|
meta_kms_update_unassign_plane (update, crtc, cursor_plane);
|
|
|
|
}
|
|
|
|
|
|
|
|
meta_kms_update_add_page_flip_listener (update,
|
|
|
|
crtc,
|
|
|
|
&cursor_page_flip_listener_vtable,
|
|
|
|
META_KMS_PAGE_FLIP_LISTENER_FLAG_NONE,
|
|
|
|
meta_thread_impl_get_main_context (thread_impl),
|
|
|
|
crtc_state_impl_ref (crtc_state_impl),
|
|
|
|
(GDestroyNotify) crtc_state_impl_unref);
|
|
|
|
meta_kms_update_add_result_listener (update,
|
|
|
|
&cursor_result_listener_vtable,
|
|
|
|
meta_thread_impl_get_main_context (thread_impl),
|
|
|
|
crtc_state_impl_ref (crtc_state_impl),
|
|
|
|
(GDestroyNotify) crtc_state_impl_unref);
|
|
|
|
|
|
|
|
return update;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_old_buffers (gpointer retval,
|
|
|
|
const GError *error,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GList *old_buffers = retval;
|
|
|
|
|
|
|
|
g_list_free_full (old_buffers, g_object_unref);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaKmsUpdate *
|
|
|
|
update_filter_cb (MetaKmsImpl *impl,
|
|
|
|
MetaKmsCrtc *crtc,
|
|
|
|
MetaKmsUpdate *update,
|
|
|
|
MetaKmsUpdateFlag flags,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaThreadImpl *thread_impl = META_THREAD_IMPL (impl);
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl = user_data;
|
|
|
|
|
|
|
|
if (flags & META_KMS_UPDATE_FLAG_TEST_ONLY)
|
|
|
|
return update;
|
|
|
|
|
|
|
|
if (flags & META_KMS_UPDATE_FLAG_MODE_SET)
|
|
|
|
{
|
|
|
|
GPtrArray *crtc_states = cursor_manager_impl->crtc_states;
|
|
|
|
GList *old_buffers = NULL;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
g_return_val_if_fail (crtc_states, update);
|
|
|
|
|
|
|
|
for (i = 0; i < crtc_states->len; i++)
|
|
|
|
{
|
|
|
|
CrtcStateImpl *crtc_state_impl = g_ptr_array_index (crtc_states, i);
|
|
|
|
MetaKmsCrtc *crtc = crtc_state_impl->crtc;
|
|
|
|
MetaDrmBuffer *old_buffer = NULL;
|
|
|
|
|
2023-07-20 11:29:49 +02:00
|
|
|
if (meta_kms_crtc_get_device (crtc) !=
|
|
|
|
meta_kms_update_get_device (update))
|
|
|
|
continue;
|
|
|
|
|
2022-12-21 18:54:17 +01:00
|
|
|
update = maybe_update_cursor_plane (cursor_manager_impl,
|
|
|
|
crtc, update, &old_buffer);
|
|
|
|
if (old_buffer)
|
|
|
|
old_buffers = g_list_prepend (old_buffers, old_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_buffers)
|
|
|
|
{
|
|
|
|
meta_thread_queue_callback (meta_thread_impl_get_thread (thread_impl),
|
|
|
|
g_main_context_default (),
|
|
|
|
NULL,
|
|
|
|
old_buffers,
|
|
|
|
(GDestroyNotify) free_old_buffers);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MetaDrmBuffer *old_buffer = NULL;
|
|
|
|
|
|
|
|
update = maybe_update_cursor_plane (cursor_manager_impl,
|
|
|
|
crtc, update, &old_buffer);
|
|
|
|
|
|
|
|
if (old_buffer)
|
|
|
|
{
|
|
|
|
meta_thread_queue_callback (meta_thread_impl_get_thread (thread_impl),
|
|
|
|
g_main_context_default (),
|
|
|
|
NULL,
|
|
|
|
old_buffer, g_object_unref);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return update;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_kms_cursor_manager_impl_free (MetaKmsCursorManagerImpl *cursor_manager_impl)
|
|
|
|
{
|
|
|
|
g_warn_if_fail (!cursor_manager_impl->crtc_states);
|
2023-08-16 21:19:42 +02:00
|
|
|
|
|
|
|
g_free (cursor_manager_impl);
|
2022-12-21 18:54:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static MetaKmsCursorManagerImpl *
|
|
|
|
ensure_cursor_manager_impl (MetaKmsImpl *impl)
|
|
|
|
{
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl;
|
|
|
|
|
|
|
|
cursor_manager_impl = g_object_get_qdata (G_OBJECT (impl),
|
|
|
|
quark_cursor_manager_impl);
|
|
|
|
if (!cursor_manager_impl)
|
|
|
|
{
|
|
|
|
cursor_manager_impl = g_new0 (MetaKmsCursorManagerImpl, 1);
|
|
|
|
cursor_manager_impl->impl = impl;
|
|
|
|
|
|
|
|
g_object_set_qdata (G_OBJECT (impl),
|
|
|
|
quark_cursor_manager_impl,
|
|
|
|
cursor_manager_impl);
|
|
|
|
|
|
|
|
cursor_manager_impl->update_filter =
|
|
|
|
meta_kms_impl_add_update_filter (impl, update_filter_cb,
|
|
|
|
cursor_manager_impl);
|
|
|
|
}
|
|
|
|
return cursor_manager_impl;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gpointer
|
|
|
|
finalize_in_impl (MetaThreadImpl *thread_impl,
|
|
|
|
gpointer user_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaKmsImpl *impl = META_KMS_IMPL (thread_impl);
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl;
|
|
|
|
|
|
|
|
cursor_manager_impl = g_object_steal_qdata (G_OBJECT (impl),
|
|
|
|
quark_cursor_manager_impl);
|
|
|
|
if (cursor_manager_impl)
|
|
|
|
{
|
|
|
|
GPtrArray *crtc_states;
|
|
|
|
|
|
|
|
meta_kms_impl_remove_update_filter (impl,
|
|
|
|
cursor_manager_impl->update_filter);
|
|
|
|
crtc_states = g_steal_pointer (&cursor_manager_impl->crtc_states);
|
|
|
|
meta_kms_cursor_manager_impl_free (cursor_manager_impl);
|
|
|
|
return crtc_states;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_kms_cursor_manager_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
MetaKmsCursorManager *cursor_manager = META_KMS_CURSOR_MANAGER (object);
|
|
|
|
GPtrArray *crtc_states;
|
|
|
|
|
|
|
|
crtc_states =
|
|
|
|
meta_thread_run_impl_task_sync (META_THREAD (cursor_manager->kms),
|
|
|
|
finalize_in_impl, NULL, NULL);
|
|
|
|
g_clear_pointer (&crtc_states, g_ptr_array_unref);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_kms_cursor_manager_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_kms_cursor_manager_class_init (MetaKmsCursorManagerClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->finalize = meta_kms_cursor_manager_finalize;
|
|
|
|
|
|
|
|
quark_cursor_manager_impl =
|
|
|
|
g_quark_from_static_string ("-meta-kms-cursor-manager-quark");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_kms_cursor_manager_init (MetaKmsCursorManager *cursor_manager)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
MetaKmsCursorManager *
|
|
|
|
meta_kms_cursor_manager_new (MetaKms *kms)
|
|
|
|
{
|
|
|
|
MetaKmsCursorManager *cursor_manager;
|
|
|
|
|
|
|
|
cursor_manager = g_object_new (META_TYPE_KMS_CURSOR_MANAGER, NULL);
|
|
|
|
cursor_manager->kms = kms;
|
|
|
|
|
|
|
|
return cursor_manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
MetaKmsCursorQueryInImpl func;
|
|
|
|
gpointer user_data;
|
|
|
|
} SetQueryFuncData;
|
|
|
|
|
|
|
|
static gpointer
|
|
|
|
set_query_func_in_impl (MetaThreadImpl *thread_impl,
|
|
|
|
gpointer user_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
SetQueryFuncData *data = user_data;
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl =
|
|
|
|
ensure_cursor_manager_impl (META_KMS_IMPL (thread_impl));
|
|
|
|
|
|
|
|
cursor_manager_impl->cursor_query_in_impl_func = data->func;
|
|
|
|
cursor_manager_impl->cursor_query_in_impl_func_user_data = data->user_data;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_kms_cursor_manager_set_query_func (MetaKmsCursorManager *cursor_manager,
|
|
|
|
MetaKmsCursorQueryInImpl func,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
SetQueryFuncData *data;
|
|
|
|
|
|
|
|
data = g_new0 (SetQueryFuncData, 1);
|
|
|
|
data->func = func;
|
|
|
|
data->user_data = user_data;
|
|
|
|
|
|
|
|
meta_thread_post_impl_task (META_THREAD (cursor_manager->kms),
|
|
|
|
set_query_func_in_impl,
|
|
|
|
data, g_free,
|
|
|
|
NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gpointer
|
|
|
|
position_changed_in_impl (MetaThreadImpl *thread_impl,
|
|
|
|
gpointer user_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl =
|
|
|
|
ensure_cursor_manager_impl (META_KMS_IMPL (thread_impl));
|
|
|
|
const graphene_point_t *position = user_data;
|
|
|
|
GPtrArray *crtc_states;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
crtc_states = cursor_manager_impl->crtc_states;
|
|
|
|
g_return_val_if_fail (crtc_states, NULL);
|
|
|
|
|
|
|
|
for (i = 0; i < crtc_states->len; i++)
|
|
|
|
{
|
|
|
|
CrtcStateImpl *crtc_state_impl = g_ptr_array_index (crtc_states, i);
|
|
|
|
MetaDrmBuffer *buffer;
|
|
|
|
const graphene_point_t *hotspot;
|
|
|
|
gboolean did_have_cursor;
|
|
|
|
gboolean should_have_cursor;
|
|
|
|
|
|
|
|
buffer = crtc_state_impl->buffer;
|
|
|
|
hotspot = &crtc_state_impl->hotspot;
|
|
|
|
|
|
|
|
if (buffer)
|
|
|
|
{
|
|
|
|
should_have_cursor = calculate_cursor_rect (crtc_state_impl,
|
|
|
|
buffer,
|
|
|
|
hotspot,
|
|
|
|
position->x,
|
|
|
|
position->y,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
should_have_cursor = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
did_have_cursor = crtc_state_impl->has_cursor;
|
|
|
|
|
|
|
|
if (did_have_cursor != should_have_cursor ||
|
|
|
|
should_have_cursor)
|
|
|
|
{
|
|
|
|
MetaKmsCrtc *crtc = crtc_state_impl->crtc;
|
|
|
|
MetaKmsDevice *device = meta_kms_crtc_get_device (crtc);
|
|
|
|
MetaKmsImplDevice *impl_device =
|
|
|
|
meta_kms_device_get_impl_device (device);
|
|
|
|
|
|
|
|
crtc_state_impl->cursor_invalidated = TRUE;
|
|
|
|
|
|
|
|
meta_kms_impl_device_schedule_process (impl_device,
|
|
|
|
crtc_state_impl->crtc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static graphene_point_t *
|
|
|
|
copy_point (const graphene_point_t *point)
|
|
|
|
{
|
|
|
|
return graphene_point_init_from_point (graphene_point_alloc (), point);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_kms_cursor_manager_position_changed_in_input_impl (MetaKmsCursorManager *cursor_manager,
|
|
|
|
const graphene_point_t *position)
|
|
|
|
{
|
|
|
|
meta_thread_post_impl_task (META_THREAD (cursor_manager->kms),
|
|
|
|
position_changed_in_impl,
|
|
|
|
copy_point (position),
|
|
|
|
(GDestroyNotify) graphene_point_free,
|
|
|
|
NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
MetaKmsCrtc *crtc;
|
|
|
|
MetaDrmBuffer *buffer;
|
2023-08-14 18:05:41 +08:00
|
|
|
MetaMonitorTransform transform;
|
2022-12-21 18:54:17 +01:00
|
|
|
graphene_point_t hotspot;
|
|
|
|
} UpdateSpriteData;
|
|
|
|
|
|
|
|
static gpointer
|
|
|
|
update_sprite_in_impl (MetaThreadImpl *thread_impl,
|
|
|
|
gpointer user_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl =
|
|
|
|
ensure_cursor_manager_impl (META_KMS_IMPL (thread_impl));
|
|
|
|
UpdateSpriteData *data = user_data;
|
|
|
|
MetaKmsCrtc *crtc = data->crtc;
|
|
|
|
MetaKmsDevice *device = meta_kms_crtc_get_device (crtc);
|
|
|
|
MetaKmsImplDevice *impl_device =
|
|
|
|
meta_kms_device_get_impl_device (device);
|
|
|
|
CrtcStateImpl *crtc_state_impl;
|
|
|
|
MetaDrmBuffer *old_buffer;
|
|
|
|
|
|
|
|
crtc_state_impl = find_crtc_state (cursor_manager_impl, crtc);
|
|
|
|
g_return_val_if_fail (crtc_state_impl, NULL);
|
|
|
|
|
|
|
|
old_buffer = g_steal_pointer (&crtc_state_impl->buffer);
|
|
|
|
crtc_state_impl->buffer = g_steal_pointer (&data->buffer);
|
2023-08-14 18:05:41 +08:00
|
|
|
crtc_state_impl->transform = data->transform;
|
2022-12-21 18:54:17 +01:00
|
|
|
crtc_state_impl->hotspot = data->hotspot;
|
|
|
|
crtc_state_impl->cursor_invalidated = TRUE;
|
|
|
|
|
|
|
|
meta_kms_impl_device_schedule_process (impl_device,
|
|
|
|
crtc_state_impl->crtc);
|
|
|
|
|
|
|
|
if (old_buffer)
|
|
|
|
{
|
|
|
|
MetaThread *thread = meta_thread_impl_get_thread (thread_impl);
|
|
|
|
|
|
|
|
meta_thread_queue_callback (thread,
|
|
|
|
NULL, NULL,
|
|
|
|
old_buffer, g_object_unref);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_kms_cursor_manager_update_sprite (MetaKmsCursorManager *cursor_manager,
|
|
|
|
MetaKmsCrtc *crtc,
|
|
|
|
MetaDrmBuffer *buffer,
|
2023-08-14 18:05:41 +08:00
|
|
|
MetaMonitorTransform transform,
|
2022-12-21 18:54:17 +01:00
|
|
|
const graphene_point_t *hotspot)
|
|
|
|
{
|
|
|
|
UpdateSpriteData *data;
|
|
|
|
|
|
|
|
data = g_new0 (UpdateSpriteData, 1);
|
|
|
|
data->crtc = crtc;
|
|
|
|
data->buffer = buffer ? g_object_ref (buffer) : NULL;
|
2023-08-14 18:05:41 +08:00
|
|
|
data->transform = transform;
|
2022-12-21 18:54:17 +01:00
|
|
|
if (hotspot)
|
|
|
|
data->hotspot = *hotspot;
|
|
|
|
|
|
|
|
meta_thread_post_impl_task (META_THREAD (cursor_manager->kms),
|
|
|
|
update_sprite_in_impl,
|
|
|
|
data, g_free,
|
|
|
|
NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
crtc_state_impl_clear_in_main (CrtcStateImpl *crtc_state_impl)
|
|
|
|
{
|
|
|
|
g_clear_object (&crtc_state_impl->buffer);
|
|
|
|
g_clear_object (&crtc_state_impl->pending_buffer);
|
|
|
|
g_clear_object (&crtc_state_impl->active_buffer);
|
|
|
|
g_clear_object (&crtc_state_impl->presenting_buffer);
|
|
|
|
crtc_state_impl_unref (crtc_state_impl);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
clear_crtc_states_in_impl (MetaThreadImpl *thread_impl,
|
|
|
|
GPtrArray *crtc_states)
|
|
|
|
{
|
|
|
|
MetaThread *thread = meta_thread_impl_get_thread (thread_impl);
|
|
|
|
|
|
|
|
meta_thread_queue_callback (thread,
|
|
|
|
NULL, NULL,
|
|
|
|
crtc_states,
|
|
|
|
(GDestroyNotify) g_ptr_array_unref);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gpointer
|
|
|
|
update_viewports_in_impl (MetaThreadImpl *thread_impl,
|
|
|
|
gpointer user_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaKmsCursorManagerImpl *cursor_manager_impl =
|
|
|
|
ensure_cursor_manager_impl (META_KMS_IMPL (thread_impl));
|
|
|
|
GArray *layouts = user_data;
|
|
|
|
GPtrArray *crtc_states;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
crtc_states =
|
|
|
|
g_ptr_array_new_full (layouts->len,
|
|
|
|
(GDestroyNotify) crtc_state_impl_clear_in_main);
|
|
|
|
|
|
|
|
for (i = 0; i < layouts->len; i++)
|
|
|
|
{
|
|
|
|
MetaKmsCrtcLayout *crtc_layout =
|
|
|
|
&g_array_index (layouts, MetaKmsCrtcLayout, i);
|
|
|
|
CrtcStateImpl *crtc_state_impl;
|
|
|
|
CrtcStateImpl *old_crtc_state;
|
|
|
|
|
|
|
|
old_crtc_state = find_crtc_state (cursor_manager_impl, crtc_layout->crtc);
|
|
|
|
if (old_crtc_state)
|
|
|
|
{
|
|
|
|
crtc_state_impl =
|
|
|
|
crtc_state_impl_new (cursor_manager_impl,
|
|
|
|
crtc_layout->crtc,
|
|
|
|
crtc_layout->layout,
|
|
|
|
crtc_layout->scale,
|
|
|
|
g_steal_pointer (&old_crtc_state->buffer));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
crtc_state_impl =
|
|
|
|
crtc_state_impl_new (cursor_manager_impl,
|
|
|
|
crtc_layout->crtc,
|
|
|
|
crtc_layout->layout,
|
|
|
|
crtc_layout->scale,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
crtc_state_impl->cursor_invalidated = TRUE;
|
|
|
|
g_ptr_array_add (crtc_states, crtc_state_impl);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cursor_manager_impl->crtc_states)
|
|
|
|
clear_crtc_states_in_impl (thread_impl, cursor_manager_impl->crtc_states);
|
|
|
|
cursor_manager_impl->crtc_states = crtc_states;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_kms_cursor_manager_update_crtc_layout (MetaKmsCursorManager *cursor_manager,
|
|
|
|
GArray *layouts)
|
|
|
|
{
|
|
|
|
meta_thread_post_impl_task (META_THREAD (cursor_manager->kms),
|
|
|
|
update_viewports_in_impl,
|
|
|
|
g_array_ref (layouts),
|
|
|
|
(GDestroyNotify) g_array_unref,
|
|
|
|
NULL, NULL);
|
|
|
|
}
|