ead09bf6cc
If the compiler cannot figure out that the condition for setting the dev variable is the same as the condition for accessing it, it will complain about potential uninitialized use.
573 lines
17 KiB
C
573 lines
17 KiB
C
/*
|
|
* Wayland Support
|
|
*
|
|
* Copyright (C) 2015 Red Hat
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*
|
|
* Author: Carlos Garnacho <carlosg@gnome.org>
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include "config.h"
|
|
|
|
#include <glib.h>
|
|
|
|
#include <wayland-server.h>
|
|
#include "tablet-unstable-v2-server-protocol.h"
|
|
|
|
#include "meta-wayland-private.h"
|
|
#include "meta-wayland-tablet-seat.h"
|
|
#include "meta-wayland-tablet.h"
|
|
#include "meta-wayland-tablet-tool.h"
|
|
#include "meta-wayland-tablet-pad.h"
|
|
|
|
#ifdef HAVE_NATIVE_BACKEND
|
|
#include <clutter/evdev/clutter-evdev.h>
|
|
#include "backends/native/meta-backend-native.h"
|
|
#endif
|
|
|
|
static void
|
|
unbind_resource (struct wl_resource *resource)
|
|
{
|
|
wl_list_remove (wl_resource_get_link (resource));
|
|
}
|
|
|
|
static void
|
|
notify_tool_added (MetaWaylandTabletSeat *tablet_seat,
|
|
struct wl_resource *client_resource,
|
|
MetaWaylandTabletTool *tool)
|
|
{
|
|
struct wl_resource *tool_resource;
|
|
struct wl_client *client;
|
|
|
|
client = wl_resource_get_client (client_resource);
|
|
tool_resource = meta_wayland_tablet_tool_lookup_resource (tool, client);
|
|
|
|
if (!tool_resource)
|
|
return;
|
|
|
|
zwp_tablet_seat_v2_send_tool_added (client_resource, tool_resource);
|
|
}
|
|
|
|
static void
|
|
notify_tablet_added (MetaWaylandTabletSeat *tablet_seat,
|
|
struct wl_resource *client_resource,
|
|
ClutterInputDevice *device)
|
|
{
|
|
struct wl_resource *resource;
|
|
MetaWaylandTablet *tablet;
|
|
struct wl_client *client;
|
|
|
|
tablet = g_hash_table_lookup (tablet_seat->tablets, device);
|
|
|
|
if (!tablet)
|
|
return;
|
|
|
|
client = wl_resource_get_client (client_resource);
|
|
|
|
if (meta_wayland_tablet_lookup_resource (tablet, client))
|
|
return;
|
|
|
|
resource = meta_wayland_tablet_create_new_resource (tablet, client,
|
|
client_resource, 0);
|
|
if (!resource)
|
|
return;
|
|
|
|
zwp_tablet_seat_v2_send_tablet_added (client_resource, resource);
|
|
meta_wayland_tablet_notify (tablet, resource);
|
|
}
|
|
|
|
static void
|
|
broadcast_tablet_added (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDevice *device)
|
|
{
|
|
struct wl_resource *resource;
|
|
|
|
wl_resource_for_each (resource, &tablet_seat->resource_list)
|
|
{
|
|
notify_tablet_added (tablet_seat, resource, device);
|
|
}
|
|
}
|
|
|
|
static void
|
|
notify_tablets (MetaWaylandTabletSeat *tablet_seat,
|
|
struct wl_resource *client_resource)
|
|
{
|
|
ClutterInputDevice *device;
|
|
GHashTableIter iter;
|
|
|
|
g_hash_table_iter_init (&iter, tablet_seat->tablets);
|
|
|
|
while (g_hash_table_iter_next (&iter, (gpointer *) &device, NULL))
|
|
notify_tablet_added (tablet_seat, client_resource, device);
|
|
}
|
|
|
|
static void
|
|
notify_pad_added (MetaWaylandTabletSeat *tablet_seat,
|
|
struct wl_resource *tablet_seat_resource,
|
|
ClutterInputDevice *device)
|
|
{
|
|
struct wl_resource *resource;
|
|
MetaWaylandTabletPad *pad;
|
|
struct wl_client *client;
|
|
|
|
pad = g_hash_table_lookup (tablet_seat->pads, device);
|
|
|
|
if (!pad)
|
|
return;
|
|
|
|
client = wl_resource_get_client (tablet_seat_resource);
|
|
|
|
if (meta_wayland_tablet_pad_lookup_resource (pad, client))
|
|
return;
|
|
|
|
resource = meta_wayland_tablet_pad_create_new_resource (pad, client,
|
|
tablet_seat_resource,
|
|
0);
|
|
if (!resource)
|
|
return;
|
|
|
|
zwp_tablet_seat_v2_send_pad_added (tablet_seat_resource, resource);
|
|
meta_wayland_tablet_pad_notify (pad, resource);
|
|
}
|
|
|
|
static void
|
|
broadcast_pad_added (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDevice *device)
|
|
{
|
|
struct wl_resource *resource;
|
|
|
|
wl_resource_for_each (resource, &tablet_seat->resource_list)
|
|
{
|
|
notify_pad_added (tablet_seat, resource, device);
|
|
}
|
|
}
|
|
|
|
static void
|
|
notify_pads (MetaWaylandTabletSeat *tablet_seat,
|
|
struct wl_resource *tablet_seat_resource)
|
|
{
|
|
ClutterInputDevice *device;
|
|
GHashTableIter iter;
|
|
|
|
g_hash_table_iter_init (&iter, tablet_seat->pads);
|
|
|
|
while (g_hash_table_iter_next (&iter, (gpointer *) &device, NULL))
|
|
notify_pad_added (tablet_seat, tablet_seat_resource, device);
|
|
}
|
|
|
|
static gboolean
|
|
is_tablet_device (ClutterInputDevice *device)
|
|
{
|
|
ClutterInputDeviceType device_type;
|
|
|
|
if (clutter_input_device_get_device_mode (device) == CLUTTER_INPUT_MODE_MASTER)
|
|
return FALSE;
|
|
|
|
device_type = clutter_input_device_get_device_type (device);
|
|
|
|
return (device_type == CLUTTER_TABLET_DEVICE ||
|
|
device_type == CLUTTER_PEN_DEVICE ||
|
|
device_type == CLUTTER_ERASER_DEVICE ||
|
|
device_type == CLUTTER_CURSOR_DEVICE);
|
|
}
|
|
|
|
static gboolean
|
|
is_pad_device (ClutterInputDevice *device)
|
|
{
|
|
ClutterInputDeviceType device_type;
|
|
|
|
if (clutter_input_device_get_device_mode (device) == CLUTTER_INPUT_MODE_MASTER)
|
|
return FALSE;
|
|
|
|
device_type = clutter_input_device_get_device_type (device);
|
|
|
|
return device_type == CLUTTER_PAD_DEVICE;
|
|
}
|
|
|
|
static void
|
|
meta_wayland_tablet_seat_device_added (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDevice *device)
|
|
{
|
|
MetaWaylandSurface *pad_focus = tablet_seat->seat->keyboard.focus_surface;
|
|
|
|
if (is_tablet_device (device))
|
|
{
|
|
MetaWaylandTablet *tablet;
|
|
GList *pads, *l;
|
|
|
|
tablet = meta_wayland_tablet_new (device, tablet_seat);
|
|
g_hash_table_insert (tablet_seat->tablets, device, tablet);
|
|
broadcast_tablet_added (tablet_seat, device);
|
|
|
|
/* Because the insertion order is undefined, there might be already
|
|
* pads that are logically paired to this tablet. Look those up and
|
|
* refocus them.
|
|
*/
|
|
pads = meta_wayland_tablet_seat_lookup_paired_pads (tablet_seat,
|
|
tablet);
|
|
|
|
for (l = pads; l; l = l->next)
|
|
meta_wayland_tablet_pad_set_focus (l->data, pad_focus);
|
|
|
|
g_list_free (pads);
|
|
}
|
|
else if (is_pad_device (device))
|
|
{
|
|
MetaWaylandTabletPad *pad;
|
|
|
|
pad = meta_wayland_tablet_pad_new (device, tablet_seat);
|
|
g_hash_table_insert (tablet_seat->pads, device, pad);
|
|
broadcast_pad_added (tablet_seat, device);
|
|
|
|
meta_wayland_tablet_pad_set_focus (pad, pad_focus);
|
|
}
|
|
}
|
|
|
|
static void
|
|
meta_wayland_tablet_seat_device_removed (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDevice *device)
|
|
{
|
|
g_hash_table_remove (tablet_seat->tablets, device);
|
|
g_hash_table_remove (tablet_seat->pads, device);
|
|
}
|
|
|
|
static void
|
|
tablet_seat_destroy (struct wl_client *client,
|
|
struct wl_resource *resource)
|
|
{
|
|
wl_resource_destroy (resource);
|
|
}
|
|
|
|
static const struct zwp_tablet_seat_v2_interface tablet_seat_interface = {
|
|
tablet_seat_destroy
|
|
};
|
|
|
|
MetaWaylandTabletSeat *
|
|
meta_wayland_tablet_seat_new (MetaWaylandTabletManager *manager,
|
|
MetaWaylandSeat *seat)
|
|
{
|
|
MetaWaylandTabletSeat *tablet_seat;
|
|
const GSList *devices, *l;
|
|
|
|
tablet_seat = g_slice_new0 (MetaWaylandTabletSeat);
|
|
tablet_seat->manager = manager;
|
|
tablet_seat->seat = seat;
|
|
tablet_seat->device_manager = clutter_device_manager_get_default ();
|
|
tablet_seat->tablets = g_hash_table_new_full (NULL, NULL, NULL,
|
|
(GDestroyNotify) meta_wayland_tablet_free);
|
|
tablet_seat->tools = g_hash_table_new_full (NULL, NULL, NULL,
|
|
(GDestroyNotify) meta_wayland_tablet_tool_free);
|
|
tablet_seat->pads = g_hash_table_new_full (NULL, NULL, NULL,
|
|
(GDestroyNotify) meta_wayland_tablet_pad_free);
|
|
wl_list_init (&tablet_seat->resource_list);
|
|
|
|
g_signal_connect_swapped (tablet_seat->device_manager, "device-added",
|
|
G_CALLBACK (meta_wayland_tablet_seat_device_added),
|
|
tablet_seat);
|
|
g_signal_connect_swapped (tablet_seat->device_manager, "device-removed",
|
|
G_CALLBACK (meta_wayland_tablet_seat_device_removed),
|
|
tablet_seat);
|
|
|
|
devices = clutter_device_manager_peek_devices (tablet_seat->device_manager);
|
|
|
|
for (l = devices; l; l = l->next)
|
|
meta_wayland_tablet_seat_device_added (tablet_seat, l->data);
|
|
|
|
return tablet_seat;
|
|
}
|
|
|
|
void
|
|
meta_wayland_tablet_seat_free (MetaWaylandTabletSeat *tablet_seat)
|
|
{
|
|
struct wl_resource *resource, *next;
|
|
|
|
wl_resource_for_each_safe (resource, next, &tablet_seat->resource_list)
|
|
{
|
|
wl_list_remove (wl_resource_get_link (resource));
|
|
wl_list_init (wl_resource_get_link (resource));
|
|
}
|
|
|
|
g_signal_handlers_disconnect_by_data (tablet_seat->device_manager,
|
|
tablet_seat);
|
|
g_hash_table_destroy (tablet_seat->tablets);
|
|
g_hash_table_destroy (tablet_seat->tools);
|
|
g_hash_table_destroy (tablet_seat->pads);
|
|
g_slice_free (MetaWaylandTabletSeat, tablet_seat);
|
|
}
|
|
|
|
struct wl_resource *
|
|
meta_wayland_tablet_seat_create_new_resource (MetaWaylandTabletSeat *tablet_seat,
|
|
struct wl_client *client,
|
|
struct wl_resource *manager_resource,
|
|
uint32_t id)
|
|
{
|
|
struct wl_resource *resource;
|
|
|
|
resource = wl_resource_create (client, &zwp_tablet_seat_v2_interface,
|
|
wl_resource_get_version (manager_resource),
|
|
id);
|
|
wl_resource_set_implementation (resource, &tablet_seat_interface,
|
|
tablet_seat, unbind_resource);
|
|
wl_resource_set_user_data (resource, tablet_seat);
|
|
wl_list_insert (&tablet_seat->resource_list, wl_resource_get_link (resource));
|
|
|
|
/* Notify client of all available tablets/pads */
|
|
notify_tablets (tablet_seat, resource);
|
|
notify_pads (tablet_seat, resource);
|
|
|
|
return resource;
|
|
}
|
|
|
|
struct wl_resource *
|
|
meta_wayland_tablet_seat_lookup_resource (MetaWaylandTabletSeat *tablet_seat,
|
|
struct wl_client *client)
|
|
{
|
|
return wl_resource_find_for_client (&tablet_seat->resource_list, client);
|
|
}
|
|
|
|
MetaWaylandTablet *
|
|
meta_wayland_tablet_seat_lookup_tablet (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDevice *device)
|
|
{
|
|
return g_hash_table_lookup (tablet_seat->tablets, device);
|
|
}
|
|
|
|
MetaWaylandTabletTool *
|
|
meta_wayland_tablet_seat_lookup_tool (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDeviceTool *tool)
|
|
{
|
|
return g_hash_table_lookup (tablet_seat->tools, tool);
|
|
}
|
|
|
|
MetaWaylandTabletPad *
|
|
meta_wayland_tablet_seat_lookup_pad (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDevice *device)
|
|
{
|
|
return g_hash_table_lookup (tablet_seat->pads, device);
|
|
}
|
|
|
|
static MetaWaylandTabletTool *
|
|
meta_wayland_tablet_seat_ensure_tool (MetaWaylandTabletSeat *tablet_seat,
|
|
ClutterInputDevice *device,
|
|
ClutterInputDeviceTool *device_tool)
|
|
{
|
|
MetaWaylandTabletTool *tool;
|
|
|
|
tool = g_hash_table_lookup (tablet_seat->tools, device_tool);
|
|
|
|
if (!tool)
|
|
{
|
|
tool = meta_wayland_tablet_tool_new (tablet_seat, device, device_tool);
|
|
g_hash_table_insert (tablet_seat->tools, device_tool, tool);
|
|
}
|
|
|
|
return tool;
|
|
}
|
|
|
|
void
|
|
meta_wayland_tablet_seat_update (MetaWaylandTabletSeat *tablet_seat,
|
|
const ClutterEvent *event)
|
|
{
|
|
ClutterInputDevice *device;
|
|
ClutterInputDeviceTool *device_tool;
|
|
MetaWaylandTabletTool *tool = NULL;
|
|
MetaWaylandTabletPad *pad = NULL;
|
|
|
|
device = clutter_event_get_source_device (event);
|
|
|
|
switch (event->type)
|
|
{
|
|
case CLUTTER_PROXIMITY_IN:
|
|
case CLUTTER_PROXIMITY_OUT:
|
|
case CLUTTER_BUTTON_PRESS:
|
|
case CLUTTER_BUTTON_RELEASE:
|
|
case CLUTTER_MOTION:
|
|
device_tool = clutter_event_get_device_tool (event);
|
|
|
|
if (device && device_tool)
|
|
tool = meta_wayland_tablet_seat_ensure_tool (tablet_seat, device, device_tool);
|
|
|
|
if (!tool)
|
|
return;
|
|
|
|
meta_wayland_tablet_tool_update (tool, event);
|
|
break;
|
|
case CLUTTER_PAD_BUTTON_PRESS:
|
|
case CLUTTER_PAD_BUTTON_RELEASE:
|
|
case CLUTTER_PAD_RING:
|
|
case CLUTTER_PAD_STRIP:
|
|
pad = g_hash_table_lookup (tablet_seat->pads, device);
|
|
if (!pad)
|
|
return;
|
|
|
|
return meta_wayland_tablet_pad_update (pad, event);
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
meta_wayland_tablet_seat_handle_event (MetaWaylandTabletSeat *tablet_seat,
|
|
const ClutterEvent *event)
|
|
{
|
|
ClutterInputDeviceTool *device_tool;
|
|
MetaWaylandTabletTool *tool = NULL;
|
|
MetaWaylandTabletPad *pad = NULL;
|
|
|
|
switch (event->type)
|
|
{
|
|
case CLUTTER_PROXIMITY_IN:
|
|
case CLUTTER_PROXIMITY_OUT:
|
|
case CLUTTER_BUTTON_PRESS:
|
|
case CLUTTER_BUTTON_RELEASE:
|
|
case CLUTTER_MOTION:
|
|
device_tool = clutter_event_get_device_tool (event);
|
|
|
|
if (device_tool)
|
|
tool = g_hash_table_lookup (tablet_seat->tools, device_tool);
|
|
|
|
if (!tool)
|
|
return CLUTTER_EVENT_PROPAGATE;
|
|
|
|
meta_wayland_tablet_tool_handle_event (tool, event);
|
|
return CLUTTER_EVENT_PROPAGATE;
|
|
case CLUTTER_PAD_BUTTON_PRESS:
|
|
case CLUTTER_PAD_BUTTON_RELEASE:
|
|
case CLUTTER_PAD_RING:
|
|
case CLUTTER_PAD_STRIP:
|
|
pad = g_hash_table_lookup (tablet_seat->pads,
|
|
clutter_event_get_source_device (event));
|
|
if (!pad)
|
|
return CLUTTER_EVENT_PROPAGATE;
|
|
|
|
return meta_wayland_tablet_pad_handle_event (pad, event);
|
|
default:
|
|
return CLUTTER_EVENT_STOP;
|
|
}
|
|
}
|
|
|
|
void
|
|
meta_wayland_tablet_seat_notify_tool (MetaWaylandTabletSeat *tablet_seat,
|
|
MetaWaylandTabletTool *tool,
|
|
struct wl_client *client)
|
|
{
|
|
struct wl_resource *resource;
|
|
|
|
resource = wl_resource_find_for_client (&tablet_seat->resource_list, client);
|
|
|
|
if (resource)
|
|
notify_tool_added (tablet_seat, resource, tool);
|
|
}
|
|
|
|
static GList *
|
|
lookup_grouped_devices (ClutterInputDevice *device,
|
|
ClutterInputDeviceType type)
|
|
{
|
|
ClutterDeviceManager *device_manager;
|
|
const GSList *devices, *l;
|
|
GList *group = NULL;
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
#ifdef HAVE_NATIVE_BACKEND
|
|
struct libinput_device *dev = NULL, *libinput_device;
|
|
|
|
if (META_IS_BACKEND_NATIVE (backend))
|
|
dev = clutter_evdev_input_device_get_libinput_device (device);
|
|
#endif
|
|
device_manager = clutter_device_manager_get_default ();
|
|
devices = clutter_device_manager_peek_devices (device_manager);
|
|
|
|
for (l = devices; l; l = l->next)
|
|
{
|
|
if (l->data == device)
|
|
continue;
|
|
if (clutter_input_device_get_device_type (l->data) != type)
|
|
continue;
|
|
|
|
#ifdef HAVE_NATIVE_BACKEND
|
|
if (META_IS_BACKEND_NATIVE (backend))
|
|
{
|
|
libinput_device = clutter_evdev_input_device_get_libinput_device (l->data);
|
|
|
|
if (libinput_device_get_device_group (dev) !=
|
|
libinput_device_get_device_group (libinput_device))
|
|
continue;
|
|
}
|
|
#endif
|
|
|
|
group = g_list_prepend (group, l->data);
|
|
}
|
|
|
|
return group;
|
|
}
|
|
|
|
MetaWaylandTablet *
|
|
meta_wayland_tablet_seat_lookup_paired_tablet (MetaWaylandTabletSeat *tablet_seat,
|
|
MetaWaylandTabletPad *pad)
|
|
{
|
|
MetaWaylandTablet *tablet;
|
|
GList *devices;
|
|
|
|
devices = lookup_grouped_devices (pad->device, CLUTTER_TABLET_DEVICE);
|
|
|
|
if (!devices)
|
|
return NULL;
|
|
|
|
/* We only accept one device here */
|
|
g_warn_if_fail (!devices->next);
|
|
|
|
tablet = meta_wayland_tablet_seat_lookup_tablet (pad->tablet_seat,
|
|
devices->data);
|
|
g_list_free (devices);
|
|
|
|
return tablet;
|
|
}
|
|
|
|
GList *
|
|
meta_wayland_tablet_seat_lookup_paired_pads (MetaWaylandTabletSeat *tablet_seat,
|
|
MetaWaylandTablet *tablet)
|
|
{
|
|
GList *l, *devices, *pads = NULL;
|
|
MetaWaylandTabletPad *pad;
|
|
|
|
devices = lookup_grouped_devices (tablet->device, CLUTTER_PAD_DEVICE);
|
|
|
|
for (l = devices; l; l = l->next)
|
|
{
|
|
pad = meta_wayland_tablet_seat_lookup_pad (tablet_seat, l->data);
|
|
if (pad)
|
|
pads = g_list_prepend (pads, pad);
|
|
}
|
|
|
|
return pads;
|
|
}
|
|
|
|
void
|
|
meta_wayland_tablet_seat_set_pad_focus (MetaWaylandTabletSeat *tablet_seat,
|
|
MetaWaylandSurface *surface)
|
|
{
|
|
MetaWaylandTabletPad *pad;
|
|
GHashTableIter iter;
|
|
|
|
g_hash_table_iter_init (&iter, tablet_seat->pads);
|
|
|
|
while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &pad))
|
|
meta_wayland_tablet_pad_set_focus (pad, surface);
|
|
}
|