[mutter/wip/carlosg/abs-input-mapper: 216/220] backends: Add MetaInputMapper
- From: Carlos Garnacho <carlosg src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [mutter/wip/carlosg/abs-input-mapper: 216/220] backends: Add MetaInputMapper
- Date: Fri, 9 Nov 2018 16:01:28 +0000 (UTC)
commit 58ea36690b88f0ce67869d4a55deb1a720d86a80
Author: Carlos Garnacho <carlosg gnome org>
Date: Fri Apr 20 16:34:32 2018 +0200
backends: Add MetaInputMapper
This object takes care of mapping absolute devices to monitors,
to do so it uses 3 heuristics, in this order of preference:
- If a device is known to be builtin, it's assigned to the
builtin monitor.
- If input device and monitor match sizes (with an error margin
of 5%)
- If input device name and monitor vendor/product in EDID match
somehow (from "full", through "partial", to just "vendor")
The most favorable outputs are then assigned to each device, making
sure not to assign two devices of the same kind to the same output.
This object replaces (and is mostly 1:1 with) GsdDeviceMapper in
g-s-d. That object would perform these same heuristics, and let
mutter indirectly know through settings changes. This object allows
doing the same in-process.
src/Makefile.am | 2 +
src/backends/meta-input-mapper-private.h | 40 ++
src/backends/meta-input-mapper.c | 617 +++++++++++++++++++++++++++++++
src/meson.build | 2 +
4 files changed, 661 insertions(+)
---
diff --git a/src/Makefile.am b/src/Makefile.am
index b4f22de0d..4341be397 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -139,6 +139,8 @@ libmutter_@LIBMUTTER_API_VERSION@_la_SOURCES = \
backends/meta-idle-monitor-private.h \
backends/meta-idle-monitor-dbus.c \
backends/meta-idle-monitor-dbus.h \
+ backends/meta-input-mapper.c \
+ backends/meta-input-mapper-private.h \
backends/meta-input-settings.c \
backends/meta-input-settings-private.h \
backends/meta-logical-monitor.c \
diff --git a/src/backends/meta-input-mapper-private.h b/src/backends/meta-input-mapper-private.h
new file mode 100644
index 000000000..53bc66b06
--- /dev/null
+++ b/src/backends/meta-input-mapper-private.h
@@ -0,0 +1,40 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
+
+/*
+ * Copyright 2018 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, see <http://www.gnu.org/licenses/>.
+ *
+ * Author: Carlos Garnacho <carlosg gnome org>
+ */
+
+#ifndef META_INPUT_MAPPER_H
+#define META_INPUT_MAPPER_H
+
+#include <clutter/clutter.h>
+
+#define META_TYPE_INPUT_MAPPER (meta_input_mapper_get_type ())
+
+G_DECLARE_FINAL_TYPE (MetaInputMapper, meta_input_mapper,
+ META, INPUT_MAPPER, GObject)
+
+MetaInputMapper * meta_input_mapper_new (void);
+
+void meta_input_mapper_add_device (MetaInputMapper *mapper,
+ ClutterInputDevice *device,
+ gboolean builtin);
+void meta_input_mapper_remove_device (MetaInputMapper *mapper,
+ ClutterInputDevice *device);
+
+#endif /* META_INPUT_MAPPER_H */
diff --git a/src/backends/meta-input-mapper.c b/src/backends/meta-input-mapper.c
new file mode 100644
index 000000000..026dacb58
--- /dev/null
+++ b/src/backends/meta-input-mapper.c
@@ -0,0 +1,617 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
+/*
+ * Copyright 2018 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, see <http://www.gnu.org/licenses/>.
+ *
+ * Author: Carlos Garnacho <carlosg gnome org>
+ */
+
+#include "config.h"
+
+#include "meta-input-mapper-private.h"
+#include "meta-monitor-manager-private.h"
+#include "meta-logical-monitor.h"
+#include "meta-backend-private.h"
+
+#define MAX_SIZE_MATCH_DIFF 0.05
+
+typedef struct _MetaMapperInputInfo MetaMapperInputInfo;
+typedef struct _MetaMapperOutputInfo MetaMapperOutputInfo;
+typedef struct _MappingHelper MappingHelper;
+typedef struct _DeviceCandidates DeviceCandidates;
+
+struct _MetaInputMapper
+{
+ GObject parent_instance;
+ MetaMonitorManager *monitor_manager;
+ ClutterDeviceManager *input_device_manager;
+ GHashTable *input_devices; /* ClutterInputDevice -> MetaMapperInputInfo */
+ GHashTable *output_devices; /* MetaLogicalMonitor -> MetaMapperOutputInfo */
+};
+
+typedef enum {
+ META_INPUT_CAP_TOUCH = 1 << 0, /* touch device, either touchscreen or tablet */
+ META_INPUT_CAP_STYLUS = 1 << 1, /* tablet pen */
+ META_INPUT_CAP_ERASER = 1 << 2, /* tablet eraser */
+ META_INPUT_CAP_PAD = 1 << 3, /* pad device, most usually in tablets */
+ META_INPUT_CAP_CURSOR = 1 << 4 /* pointer-like device in tablets */
+} MetaInputCapabilityFlags;
+
+typedef enum {
+ META_MATCH_IS_BUILTIN, /* Output is builtin, applies mainly to system-integrated devices */
+ META_MATCH_SIZE, /* Size from input device and output match */
+ META_MATCH_EDID_FULL, /* Full EDID model match, eg. "Cintiq 12WX" */
+ META_MATCH_EDID_PARTIAL, /* Partial EDID model match, eg. "Cintiq" */
+ META_MATCH_EDID_VENDOR, /* EDID vendor match, eg. "WAC" for Wacom */
+ N_OUTPUT_MATCHES
+} MetaOutputMatchType;
+
+struct _MetaMapperInputInfo
+{
+ ClutterInputDevice *device;
+ MetaInputMapper *mapper;
+ MetaMapperOutputInfo *output;
+ guint builtin : 1;
+};
+
+struct _MetaMapperOutputInfo
+{
+ MetaLogicalMonitor *logical_monitor;
+ GList *input_devices;
+ MetaInputCapabilityFlags attached_caps;
+};
+
+struct _MappingHelper
+{
+ GArray *device_maps;
+};
+
+struct _DeviceCandidates
+{
+ MetaMapperInputInfo *input;
+
+ MetaMonitor *candidates[N_OUTPUT_MATCHES];
+
+ MetaOutputMatchType best;
+};
+
+enum {
+ DEVICE_MAPPED,
+ N_SIGNALS
+};
+
+static guint signals[N_SIGNALS] = { 0, };
+
+G_DEFINE_TYPE (MetaInputMapper, meta_input_mapper, G_TYPE_OBJECT)
+
+static MetaMapperInputInfo *
+mapper_input_info_new (ClutterInputDevice *device,
+ MetaInputMapper *mapper,
+ gboolean builtin)
+{
+ MetaMapperInputInfo *info;
+
+ info = g_new0 (MetaMapperInputInfo, 1);
+ info->mapper = mapper;
+ info->device = device;
+ info->builtin = builtin;
+
+ return info;
+}
+
+static void
+mapper_input_info_free (MetaMapperInputInfo *info)
+{
+ g_free (info);
+}
+
+static MetaMapperOutputInfo *
+mapper_output_info_new (MetaLogicalMonitor *logical_monitor)
+{
+ MetaMapperOutputInfo *info;
+
+ info = g_new0 (MetaMapperOutputInfo, 1);
+ info->logical_monitor = logical_monitor;
+
+ return info;
+}
+
+static void
+mapper_output_info_free (MetaMapperOutputInfo *info)
+{
+ g_free (info);
+}
+
+static MetaInputCapabilityFlags
+mapper_input_info_get_caps (MetaMapperInputInfo *info)
+{
+ ClutterInputDeviceType type;
+
+ type = clutter_input_device_get_device_type (info->device);
+
+ switch (type)
+ {
+ case CLUTTER_TOUCHSCREEN_DEVICE:
+ return META_INPUT_CAP_TOUCH;
+ case CLUTTER_TABLET_DEVICE:
+ case CLUTTER_PEN_DEVICE:
+ return META_INPUT_CAP_STYLUS;
+ case CLUTTER_ERASER_DEVICE:
+ return META_INPUT_CAP_ERASER;
+ case CLUTTER_CURSOR_DEVICE:
+ return META_INPUT_CAP_CURSOR;
+ case CLUTTER_PAD_DEVICE:
+ return META_INPUT_CAP_PAD;
+ default:
+ return 0;
+ }
+}
+
+static void
+mapper_input_info_set_output (MetaMapperInputInfo *input,
+ MetaMapperOutputInfo *output,
+ MetaMonitor *monitor)
+{
+ if (input->output == output)
+ return;
+
+ input->output = output;
+ g_signal_emit (input->mapper, signals[DEVICE_MAPPED], 0,
+ input->device,
+ output ? output->logical_monitor : NULL, monitor);
+}
+
+static void
+mapper_output_info_add_input (MetaMapperOutputInfo *output,
+ MetaMapperInputInfo *input,
+ MetaMonitor *monitor)
+{
+ g_assert (input->output == NULL);
+
+ output->input_devices = g_list_prepend (output->input_devices, input);
+ output->attached_caps |= mapper_input_info_get_caps (input);
+
+ mapper_input_info_set_output (input, output, monitor);
+}
+
+static void
+mapper_output_info_remove_input (MetaMapperOutputInfo *output,
+ MetaMapperInputInfo *input)
+{
+ GList *l;
+
+ g_assert (input->output == output);
+
+ output->input_devices = g_list_remove (output->input_devices, input);
+ output->attached_caps = 0;
+
+ for (l = output->input_devices; l; l = l->next)
+ output->attached_caps |= mapper_input_info_get_caps (l->data);
+
+ mapper_input_info_set_output (input, NULL, NULL);
+}
+
+static void
+mapper_output_info_clear_inputs (MetaMapperOutputInfo *output)
+{
+ while (output->input_devices)
+ {
+ MetaMapperInputInfo *input = output->input_devices->data;
+
+ mapper_input_info_set_output (input, NULL, NULL);
+ output->input_devices = g_list_remove (output->input_devices, input);
+ }
+
+ output->attached_caps = 0;
+}
+
+static void
+mapping_helper_init (MappingHelper *helper)
+{
+ helper->device_maps = g_array_new (FALSE, FALSE, sizeof (DeviceCandidates));
+}
+
+static void
+mapping_helper_release (MappingHelper *helper)
+{
+ g_array_unref (helper->device_maps);
+}
+
+static gboolean
+match_edid (MetaMapperInputInfo *input,
+ MetaMonitor *monitor,
+ MetaOutputMatchType *match_type)
+{
+ const gchar *dev_name;
+
+ dev_name = clutter_input_device_get_device_name (input->device);
+
+ if (strcasestr (dev_name, meta_monitor_get_vendor (monitor)) == NULL)
+ return FALSE;
+
+ *match_type = META_MATCH_EDID_VENDOR;
+
+ if (strcasestr (dev_name, meta_monitor_get_product (monitor)) != NULL)
+ {
+ *match_type = META_MATCH_EDID_FULL;
+ }
+ else
+ {
+ char **split;
+ int i = 0;
+
+ split = g_strsplit (meta_monitor_get_product (monitor), " ", -1);
+
+ while (split[i])
+ {
+ if (strcasestr (dev_name, split[i]) != NULL)
+ {
+ *match_type = META_MATCH_EDID_PARTIAL;
+ break;
+ }
+ }
+
+ g_strfreev (split);
+ }
+
+ return TRUE;
+}
+
+static gboolean
+find_size_match (MetaMapperInputInfo *input,
+ GList *monitors,
+ MetaMonitor **matched_monitor)
+{
+ double min_w_diff, min_h_diff;
+ double i_width, i_height;
+ gboolean found = FALSE;
+ GList *l;
+
+ min_w_diff = min_h_diff = MAX_SIZE_MATCH_DIFF;
+
+ if (!clutter_input_device_get_physical_size (input->device, &i_width, &i_height))
+ return FALSE;
+
+ for (l = monitors; l; l = l->next)
+ {
+ MetaMonitor *monitor = l->data;
+ double w_diff, h_diff;
+ int o_width, o_height;
+
+ meta_monitor_get_physical_dimensions (monitor, &o_width, &o_height);
+ w_diff = ABS (1 - ((double) o_width / i_width));
+ h_diff = ABS (1 - ((double) o_height / i_height));
+
+ if (w_diff >= min_w_diff || h_diff >= min_h_diff)
+ continue;
+
+ *matched_monitor = monitor;
+ min_w_diff = w_diff;
+ min_h_diff = h_diff;
+ found = TRUE;
+ }
+
+ return found;
+}
+
+static gboolean
+find_builtin_output (MetaInputMapper *mapper,
+ MetaMonitor **matched_monitor)
+{
+ MetaMonitor *panel;
+
+ panel = meta_monitor_manager_get_laptop_panel (mapper->monitor_manager);
+ *matched_monitor = panel;
+ return panel != NULL;
+}
+
+static gboolean
+guess_candidates (MetaInputMapper *mapper,
+ MetaMapperInputInfo *input,
+ DeviceCandidates *info)
+{
+ MetaOutputMatchType best = N_OUTPUT_MATCHES;
+ GList *monitors, *l;
+ MetaMonitor *matched_monitor = NULL;
+
+ monitors = meta_monitor_manager_get_monitors (mapper->monitor_manager);
+
+ for (l = monitors; l; l = l->next)
+ {
+ MetaOutputMatchType edid_match;
+
+ if (match_edid (input, l->data, &edid_match))
+ {
+ best = MIN (best, edid_match);
+ info->candidates[edid_match] = l->data;
+ }
+ }
+
+ if (find_size_match (input, monitors, &matched_monitor))
+ {
+ best = MIN (best, META_MATCH_SIZE);
+ info->candidates[META_MATCH_SIZE] = matched_monitor;
+ }
+
+ if (input->builtin)
+ {
+ best = MIN (best, META_MATCH_IS_BUILTIN);
+ find_builtin_output (mapper, &info->candidates[META_MATCH_IS_BUILTIN]);
+ }
+
+ if (best < N_OUTPUT_MATCHES)
+ {
+ info->best = best;
+ return TRUE;
+ }
+ else
+ {
+ return FALSE;
+ }
+}
+
+static void
+mapping_helper_add (MappingHelper *helper,
+ MetaMapperInputInfo *input,
+ MetaInputMapper *mapper)
+{
+ DeviceCandidates info = { 0, };
+ guint i, pos = 0;
+
+ info.input = input;
+
+ if (!guess_candidates (mapper, input, &info))
+ return;
+
+ for (i = 0; i < helper->device_maps->len; i++)
+ {
+ DeviceCandidates *elem;
+
+ elem = &g_array_index (helper->device_maps, DeviceCandidates, i);
+
+ if (elem->best < info.best)
+ pos = i;
+ }
+
+ if (pos >= helper->device_maps->len)
+ g_array_append_val (helper->device_maps, info);
+ else
+ g_array_insert_val (helper->device_maps, pos, info);
+}
+
+static void
+mapping_helper_apply (MappingHelper *helper,
+ MetaInputMapper *mapper)
+{
+ guint i;
+
+ /* Now, decide which input claims which output */
+ for (i = 0; i < helper->device_maps->len; i++)
+ {
+ MetaMapperOutputInfo *output;
+ DeviceCandidates *info;
+ MetaOutputMatchType j;
+
+ info = &g_array_index (helper->device_maps, DeviceCandidates, i);
+
+ for (j = 0; j < N_OUTPUT_MATCHES; j++)
+ {
+ MetaLogicalMonitor *logical_monitor;
+
+ if (!info->candidates[j])
+ continue;
+
+ logical_monitor =
+ meta_monitor_get_logical_monitor (info->candidates[j]);
+ output = g_hash_table_lookup (mapper->output_devices,
+ logical_monitor);
+
+ if (!output)
+ continue;
+
+ if (output->attached_caps & mapper_input_info_get_caps (info->input))
+ continue;
+
+ mapper_output_info_add_input (output, info->input,
+ info->candidates[j]);
+ break;
+ }
+ }
+}
+
+static void
+mapper_recalculate_candidates (MetaInputMapper *mapper)
+{
+ MetaMapperInputInfo *input;
+ MappingHelper helper;
+ GHashTableIter iter;
+
+ mapping_helper_init (&helper);
+ g_hash_table_iter_init (&iter, mapper->input_devices);
+
+ while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &input))
+ mapping_helper_add (&helper, input, mapper);
+
+ mapping_helper_apply (&helper, mapper);
+ mapping_helper_release (&helper);
+}
+
+static void
+mapper_recalculate_input (MetaInputMapper *mapper,
+ MetaMapperInputInfo *input)
+{
+ MappingHelper helper;
+
+ mapping_helper_init (&helper);
+ mapping_helper_add (&helper, input, mapper);
+ mapping_helper_apply (&helper, mapper);
+ mapping_helper_release (&helper);
+}
+
+static void
+mapper_update_outputs (MetaInputMapper *mapper)
+{
+ MetaMapperOutputInfo *output;
+ GList *logical_monitors, *l;
+ GHashTableIter iter;
+
+ g_hash_table_iter_init (&iter, mapper->output_devices);
+
+ while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &output))
+ {
+ mapper_output_info_clear_inputs (output);
+ g_hash_table_iter_remove (&iter);
+ }
+
+ logical_monitors =
+ meta_monitor_manager_get_logical_monitors (mapper->monitor_manager);
+
+ for (l = logical_monitors; l; l = l->next)
+ {
+ MetaLogicalMonitor *logical_monitor = l->data;
+ MetaMapperOutputInfo *info;
+
+ info = mapper_output_info_new (logical_monitor);
+ g_hash_table_insert (mapper->output_devices, logical_monitor, info);
+ }
+
+ mapper_recalculate_candidates (mapper);
+}
+
+static void
+input_mapper_monitors_changed_cb (MetaMonitorManager *monitor_manager,
+ MetaInputMapper *mapper)
+{
+ mapper_update_outputs (mapper);
+}
+
+static void
+input_mapper_device_removed_cb (ClutterDeviceManager *device_manager,
+ ClutterInputDevice *device,
+ MetaInputMapper *mapper)
+{
+ meta_input_mapper_remove_device (mapper, device);
+}
+
+static void
+meta_input_mapper_finalize (GObject *object)
+{
+ MetaInputMapper *mapper = META_INPUT_MAPPER (object);
+
+ g_signal_handlers_disconnect_by_func (mapper->monitor_manager,
+ input_mapper_monitors_changed_cb,
+ mapper);
+ g_signal_handlers_disconnect_by_func (mapper->input_device_manager,
+ input_mapper_device_removed_cb,
+ mapper);
+
+ g_hash_table_unref (mapper->input_devices);
+ g_hash_table_unref (mapper->output_devices);
+
+ G_OBJECT_CLASS (meta_input_mapper_parent_class)->finalize (object);
+}
+
+static void
+meta_input_mapper_constructed (GObject *object)
+{
+ MetaInputMapper *mapper = META_INPUT_MAPPER (object);
+ MetaBackend *backend;
+
+ G_OBJECT_CLASS (meta_input_mapper_parent_class)->constructed (object);
+
+ mapper->input_device_manager = clutter_device_manager_get_default ();
+ g_signal_connect (mapper->input_device_manager, "device-removed",
+ G_CALLBACK (input_mapper_device_removed_cb), mapper);
+
+ backend = meta_get_backend ();
+ mapper->monitor_manager = meta_backend_get_monitor_manager (backend);
+ g_signal_connect (mapper->monitor_manager, "monitors-changed-internal",
+ G_CALLBACK (input_mapper_monitors_changed_cb), mapper);
+
+ mapper_update_outputs (mapper);
+}
+
+static void
+meta_input_mapper_class_init (MetaInputMapperClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->constructed = meta_input_mapper_constructed;
+ object_class->finalize = meta_input_mapper_finalize;
+
+ signals[DEVICE_MAPPED] =
+ g_signal_new ("device-mapped",
+ G_TYPE_FROM_CLASS (object_class),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL, NULL,
+ G_TYPE_NONE, 3,
+ CLUTTER_TYPE_INPUT_DEVICE,
+ G_TYPE_POINTER, G_TYPE_POINTER);
+}
+
+static void
+meta_input_mapper_init (MetaInputMapper *mapper)
+{
+ mapper->input_devices =
+ g_hash_table_new_full (NULL, NULL, NULL,
+ (GDestroyNotify) mapper_input_info_free);
+ mapper->output_devices =
+ g_hash_table_new_full (NULL, NULL, NULL,
+ (GDestroyNotify) mapper_output_info_free);
+}
+
+MetaInputMapper *
+meta_input_mapper_new (void)
+{
+ return g_object_new (META_TYPE_INPUT_MAPPER, NULL);
+}
+
+void
+meta_input_mapper_add_device (MetaInputMapper *mapper,
+ ClutterInputDevice *device,
+ gboolean builtin)
+{
+ MetaMapperInputInfo *info;
+
+ g_return_if_fail (mapper != NULL);
+ g_return_if_fail (device != NULL);
+
+ if (g_hash_table_contains (mapper->input_devices, device))
+ return;
+
+ info = mapper_input_info_new (device, mapper, builtin);
+ g_hash_table_insert (mapper->input_devices, device, info);
+ mapper_recalculate_input (mapper, info);
+}
+
+void
+meta_input_mapper_remove_device (MetaInputMapper *mapper,
+ ClutterInputDevice *device)
+{
+ MetaMapperInputInfo *input;
+
+ g_return_if_fail (mapper != NULL);
+ g_return_if_fail (device != NULL);
+
+ input = g_hash_table_lookup (mapper->input_devices, device);
+
+ if (input)
+ {
+ if (input->output)
+ mapper_output_info_remove_input (input->output, input);
+ g_hash_table_remove (mapper->input_devices, device);
+ }
+}
diff --git a/src/meson.build b/src/meson.build
index 4ea30b06d..b3ff85f37 100644
--- a/src/meson.build
+++ b/src/meson.build
@@ -184,6 +184,8 @@ mutter_sources = [
'backends/meta-idle-monitor-dbus.c',
'backends/meta-idle-monitor-dbus.h',
'backends/meta-idle-monitor-private.h',
+ 'backends/meta-input-mapper.c',
+ 'backends/meta-input-mapper-private.h',
'backends/meta-input-settings.c',
'backends/meta-input-settings-private.h',
'backends/meta-logical-monitor.c',
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]