backends/x11: Fetch the tablet serial prop on device add

For tablet device, the tool was created when the "Wacom Serial IDs" prop
changed values. This property does not exist on the xf86-input-libinput
driver but v1.5.0 of that driver has a different property for the serial.

The serial is constant (the driver creates one X device per serial), so
we can fetch it after device creation and set it then. For earlier
versions of the driver we assign the random serial 0xffffffaa - good
enough to have at least a tool.

This fixes the crash in #3120 - clutter_event_motion_new()
overrides event->device to the tool's device (if any). Without a tool
motion events use the Virtual Core Pointer instead and our source device
is never added to the stage's priv->pointer_devices.

When we generate an crossing event (which uses the source device) we
fall afoul of an assert in clutter_stage_update_device() that expects
our source device to be in priv->pointer_devices.

Fixes https://gitlab.gnome.org/GNOME/mutter/-/issues/3120

Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3383>
This commit is contained in:
Peter Hutterer 2023-11-10 09:35:39 +10:00 committed by Marge Bot
parent b1dd6973df
commit b5e9279ea0

View File

@ -422,6 +422,68 @@ get_pad_features (XIDeviceInfo *info,
*n_strips = strips;
}
static gboolean
guess_serial_from_libinput_prop (MetaSeatX11 *seat_x11,
ClutterInputDevice *device,
unsigned int *serial_out)
{
Display *xdisplay = xdisplay_from_seat (seat_x11);
int device_id = meta_input_device_x11_get_device_id (device);
gulong nitems, bytes_after;
uint32_t *data = NULL;
int rc, format;
Atom type;
Atom prop;
prop = XInternAtom (xdisplay, "libinput Tablet Tool Serial", True);
if (prop != None)
{
mtk_x11_error_trap_push (xdisplay);
rc = XIGetProperty (xdisplay,
device_id,
prop,
0, 1, False, XA_CARDINAL, &type, &format, &nitems, &bytes_after,
(guchar **) &data);
mtk_x11_error_trap_pop (xdisplay);
if (rc != Success || type != XA_CARDINAL || format != 32 || nitems != 1)
{
XFree (data);
return FALSE;
}
*serial_out = *data;
XFree (data);
return TRUE;
}
/* xf86-input-libinput < 1.5.0 doesn't have the serial prop, let's check
* for a generic property that tells us if it's libinput handling this device.
* If that's the case we give the tool a "random" fixed serial to pass through
* the code that assumes 0 == no tool.
*/
prop = XInternAtom (xdisplay, "libinput Send Events Mode Enabled", True);
if (prop == None)
return FALSE;
mtk_x11_error_trap_push (xdisplay);
rc = XIGetProperty (xdisplay,
device_id,
prop,
0, 1, False, XA_INTEGER, &type, &format, &nitems, &bytes_after,
(guchar **) &data);
mtk_x11_error_trap_pop (xdisplay);
XFree (data);
if (rc != Success)
return FALSE;
*serial_out = 0xffffffaa;
return TRUE;
}
/* The Wacom driver exports the tool type as property. Use that over
guessing based on the device name */
static gboolean
@ -807,6 +869,35 @@ remove_device (MetaSeatX11 *seat_x11,
}
}
static void
update_tool (MetaSeatX11 *seat_x11,
ClutterInputDevice *device,
int serial_id)
{
ClutterInputDeviceTool *tool = NULL;
ClutterInputDeviceToolType type;
MetaInputSettings *input_settings;
if (serial_id != 0)
{
tool = g_hash_table_lookup (seat_x11->tools_by_serial,
GUINT_TO_POINTER (serial_id));
if (!tool)
{
type = clutter_input_device_get_device_type (device) == CLUTTER_ERASER_DEVICE ?
CLUTTER_INPUT_DEVICE_TOOL_ERASER : CLUTTER_INPUT_DEVICE_TOOL_PEN;
tool = meta_input_device_tool_x11_new (serial_id, type);
g_hash_table_insert (seat_x11->tools_by_serial,
GUINT_TO_POINTER (serial_id),
tool);
}
}
meta_input_device_x11_update_tool (device, tool);
input_settings = meta_backend_get_input_settings (seat_x11->backend);
meta_input_settings_notify_tool_change (input_settings, device, tool);
}
static gboolean
meta_seat_x11_handle_event_post (ClutterSeat *seat,
const ClutterEvent *event)
@ -815,7 +906,9 @@ meta_seat_x11_handle_event_post (ClutterSeat *seat,
ClutterInputDevice *device;
MetaInputSettings *input_settings;
ClutterEventType event_type;
gboolean is_touch;
gboolean is_touch, is_tablet_tool;
unsigned int serial;
ClutterInputDeviceType type;
event_type = clutter_event_type (event);
@ -824,8 +917,9 @@ meta_seat_x11_handle_event_post (ClutterSeat *seat,
return TRUE;
device = clutter_event_get_device (event);
is_touch =
clutter_input_device_get_device_type (device) == CLUTTER_TOUCHSCREEN_DEVICE;
type = clutter_input_device_get_device_type (device);
is_touch = type == CLUTTER_TOUCHSCREEN_DEVICE;
is_tablet_tool = type == CLUTTER_PEN_DEVICE || type == CLUTTER_ERASER_DEVICE;
input_settings = meta_backend_get_input_settings (seat_x11->backend);
switch (event_type)
@ -833,6 +927,8 @@ meta_seat_x11_handle_event_post (ClutterSeat *seat,
case CLUTTER_DEVICE_ADDED:
meta_input_settings_add_device (input_settings, device);
seat_x11->has_touchscreens |= is_touch;
if (is_tablet_tool && guess_serial_from_libinput_prop (seat_x11, device, &serial))
update_tool(seat_x11, device, serial);
break;
case CLUTTER_DEVICE_REMOVED:
if (is_touch)
@ -971,31 +1067,10 @@ translate_property_event (MetaSeatX11 *seat_x11,
if (xev->property == serial_ids_prop)
{
ClutterInputDeviceTool *tool = NULL;
ClutterInputDeviceToolType type;
MetaInputSettings *input_settings;
int serial_id;
serial_id = device_get_tool_serial (seat_x11, device);
if (serial_id != 0)
{
tool = g_hash_table_lookup (seat_x11->tools_by_serial,
GUINT_TO_POINTER (serial_id));
if (!tool)
{
type = clutter_input_device_get_device_type (device) == CLUTTER_ERASER_DEVICE ?
CLUTTER_INPUT_DEVICE_TOOL_ERASER : CLUTTER_INPUT_DEVICE_TOOL_PEN;
tool = meta_input_device_tool_x11_new (serial_id, type);
g_hash_table_insert (seat_x11->tools_by_serial,
GUINT_TO_POINTER (serial_id),
tool);
}
}
meta_input_device_x11_update_tool (device, tool);
input_settings = meta_backend_get_input_settings (seat_x11->backend);
meta_input_settings_notify_tool_change (input_settings, device, tool);
update_tool(seat_x11, device, serial_id);
}
}