2014-12-18 14:42:42 +10:00
|
|
|
/*
|
|
|
|
|
* Copyright © 2014 Red Hat, Inc.
|
|
|
|
|
*
|
2015-06-11 12:09:18 +10:00
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
|
|
|
* to deal in the Software without restriction, including without limitation
|
|
|
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
|
* Software is furnished to do so, subject to the following conditions:
|
2014-12-18 14:42:42 +10:00
|
|
|
*
|
2015-06-11 12:09:18 +10:00
|
|
|
* The above copyright notice and this permission notice (including the next
|
|
|
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
|
|
|
* Software.
|
|
|
|
|
*
|
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
|
* DEALINGS IN THE SOFTWARE.
|
2014-12-18 14:42:42 +10:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
|
|
#include <errno.h>
|
2015-06-24 14:52:48 +10:00
|
|
|
#include <fcntl.h>
|
2017-09-05 14:12:09 +10:00
|
|
|
#include <fnmatch.h>
|
2014-12-18 14:42:42 +10:00
|
|
|
#include <getopt.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
2017-05-23 15:07:31 +10:00
|
|
|
#include <sys/stat.h>
|
2014-12-18 15:02:45 +10:00
|
|
|
#include <libudev.h>
|
2014-12-18 14:42:42 +10:00
|
|
|
|
2015-04-21 18:24:39 +10:00
|
|
|
#include <libevdev/libevdev.h>
|
2015-05-26 08:46:05 +10:00
|
|
|
#include <libinput-util.h>
|
2015-04-21 18:24:39 +10:00
|
|
|
|
2014-12-18 14:42:42 +10:00
|
|
|
#include "shared.h"
|
|
|
|
|
|
2016-10-24 11:06:23 +10:00
|
|
|
LIBINPUT_ATTRIBUTE_PRINTF(3, 0)
|
2014-12-18 15:02:45 +10:00
|
|
|
static void
|
|
|
|
|
log_handler(struct libinput *li,
|
|
|
|
|
enum libinput_log_priority priority,
|
|
|
|
|
const char *format,
|
|
|
|
|
va_list args)
|
|
|
|
|
{
|
2017-02-13 16:28:28 +10:00
|
|
|
static int is_tty = -1;
|
|
|
|
|
|
|
|
|
|
if (is_tty == -1)
|
|
|
|
|
is_tty = isatty(STDOUT_FILENO);
|
|
|
|
|
|
|
|
|
|
if (is_tty) {
|
|
|
|
|
if (priority >= LIBINPUT_LOG_PRIORITY_ERROR)
|
|
|
|
|
printf(ANSI_RED);
|
|
|
|
|
else if (priority >= LIBINPUT_LOG_PRIORITY_INFO)
|
|
|
|
|
printf(ANSI_HIGHLIGHT);
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-18 15:02:45 +10:00
|
|
|
vprintf(format, args);
|
2017-02-13 16:28:28 +10:00
|
|
|
|
|
|
|
|
if (is_tty && priority >= LIBINPUT_LOG_PRIORITY_INFO)
|
|
|
|
|
printf(ANSI_NORMAL);
|
2014-12-18 15:02:45 +10:00
|
|
|
}
|
|
|
|
|
|
2014-12-18 14:42:42 +10:00
|
|
|
void
|
2017-06-19 18:38:33 +10:00
|
|
|
tools_init_options(struct tools_options *options)
|
2014-12-18 14:42:42 +10:00
|
|
|
{
|
|
|
|
|
memset(options, 0, sizeof(*options));
|
|
|
|
|
options->tapping = -1;
|
2016-07-21 11:46:05 +10:00
|
|
|
options->tap_map = -1;
|
2016-01-22 17:59:19 +10:00
|
|
|
options->drag = -1;
|
2015-06-22 11:07:31 +10:00
|
|
|
options->drag_lock = -1;
|
2014-12-18 15:10:59 +10:00
|
|
|
options->natural_scroll = -1;
|
2014-12-18 15:14:09 +10:00
|
|
|
options->left_handed = -1;
|
2015-04-14 12:08:33 +10:00
|
|
|
options->middlebutton = -1;
|
2015-07-08 15:03:06 +10:00
|
|
|
options->dwt = -1;
|
2015-01-16 10:16:47 +10:00
|
|
|
options->click_method = -1;
|
2015-04-21 11:57:23 +10:00
|
|
|
options->scroll_method = -1;
|
2015-04-21 18:24:39 +10:00
|
|
|
options->scroll_button = -1;
|
2015-04-10 15:45:59 +10:00
|
|
|
options->speed = 0.0;
|
2015-08-27 13:13:47 +10:00
|
|
|
options->profile = LIBINPUT_CONFIG_ACCEL_PROFILE_NONE;
|
2014-12-18 14:42:42 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
2017-06-19 18:38:33 +10:00
|
|
|
tools_parse_option(int option,
|
|
|
|
|
const char *optarg,
|
|
|
|
|
struct tools_options *options)
|
2014-12-18 14:42:42 +10:00
|
|
|
{
|
2017-06-19 18:38:33 +10:00
|
|
|
switch(option) {
|
2018-04-12 11:56:43 +10:00
|
|
|
case OPT_TAP_ENABLE:
|
|
|
|
|
options->tapping = 1;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_TAP_DISABLE:
|
|
|
|
|
options->tapping = 0;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_TAP_MAP:
|
|
|
|
|
if (!optarg)
|
|
|
|
|
return 1;
|
2017-06-19 18:38:33 +10:00
|
|
|
|
2018-04-12 11:56:43 +10:00
|
|
|
if (streq(optarg, "lrm")) {
|
|
|
|
|
options->tap_map = LIBINPUT_CONFIG_TAP_MAP_LRM;
|
|
|
|
|
} else if (streq(optarg, "lmr")) {
|
|
|
|
|
options->tap_map = LIBINPUT_CONFIG_TAP_MAP_LMR;
|
|
|
|
|
} else {
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case OPT_DRAG_ENABLE:
|
|
|
|
|
options->drag = 1;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_DRAG_DISABLE:
|
|
|
|
|
options->drag = 0;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_DRAG_LOCK_ENABLE:
|
|
|
|
|
options->drag_lock = 1;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_DRAG_LOCK_DISABLE:
|
|
|
|
|
options->drag_lock = 0;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_NATURAL_SCROLL_ENABLE:
|
|
|
|
|
options->natural_scroll = 1;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_NATURAL_SCROLL_DISABLE:
|
|
|
|
|
options->natural_scroll = 0;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_LEFT_HANDED_ENABLE:
|
|
|
|
|
options->left_handed = 1;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_LEFT_HANDED_DISABLE:
|
|
|
|
|
options->left_handed = 0;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_MIDDLEBUTTON_ENABLE:
|
|
|
|
|
options->middlebutton = 1;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_MIDDLEBUTTON_DISABLE:
|
|
|
|
|
options->middlebutton = 0;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_DWT_ENABLE:
|
|
|
|
|
options->dwt = LIBINPUT_CONFIG_DWT_ENABLED;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_DWT_DISABLE:
|
|
|
|
|
options->dwt = LIBINPUT_CONFIG_DWT_DISABLED;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_CLICK_METHOD:
|
|
|
|
|
if (!optarg)
|
|
|
|
|
return 1;
|
2017-06-19 18:38:33 +10:00
|
|
|
|
2018-04-12 11:56:43 +10:00
|
|
|
if (streq(optarg, "none")) {
|
|
|
|
|
options->click_method =
|
|
|
|
|
LIBINPUT_CONFIG_CLICK_METHOD_NONE;
|
|
|
|
|
} else if (streq(optarg, "clickfinger")) {
|
|
|
|
|
options->click_method =
|
|
|
|
|
LIBINPUT_CONFIG_CLICK_METHOD_CLICKFINGER;
|
|
|
|
|
} else if (streq(optarg, "buttonareas")) {
|
|
|
|
|
options->click_method =
|
|
|
|
|
LIBINPUT_CONFIG_CLICK_METHOD_BUTTON_AREAS;
|
|
|
|
|
} else {
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case OPT_SCROLL_METHOD:
|
|
|
|
|
if (!optarg)
|
|
|
|
|
return 1;
|
2017-06-19 18:38:33 +10:00
|
|
|
|
2018-04-12 11:56:43 +10:00
|
|
|
if (streq(optarg, "none")) {
|
|
|
|
|
options->scroll_method =
|
|
|
|
|
LIBINPUT_CONFIG_SCROLL_NO_SCROLL;
|
|
|
|
|
} else if (streq(optarg, "twofinger")) {
|
|
|
|
|
options->scroll_method =
|
|
|
|
|
LIBINPUT_CONFIG_SCROLL_2FG;
|
|
|
|
|
} else if (streq(optarg, "edge")) {
|
|
|
|
|
options->scroll_method =
|
|
|
|
|
LIBINPUT_CONFIG_SCROLL_EDGE;
|
|
|
|
|
} else if (streq(optarg, "button")) {
|
|
|
|
|
options->scroll_method =
|
|
|
|
|
LIBINPUT_CONFIG_SCROLL_ON_BUTTON_DOWN;
|
|
|
|
|
} else {
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case OPT_SCROLL_BUTTON:
|
|
|
|
|
if (!optarg) {
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
options->scroll_button =
|
|
|
|
|
libevdev_event_code_from_name(EV_KEY,
|
|
|
|
|
optarg);
|
|
|
|
|
if (options->scroll_button == -1) {
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"Invalid button %s\n",
|
|
|
|
|
optarg);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case OPT_SPEED:
|
|
|
|
|
if (!optarg)
|
|
|
|
|
return 1;
|
|
|
|
|
options->speed = atof(optarg);
|
|
|
|
|
break;
|
|
|
|
|
case OPT_PROFILE:
|
|
|
|
|
if (!optarg)
|
|
|
|
|
return 1;
|
2017-06-19 18:38:33 +10:00
|
|
|
|
2018-04-12 11:56:43 +10:00
|
|
|
if (streq(optarg, "adaptive"))
|
|
|
|
|
options->profile = LIBINPUT_CONFIG_ACCEL_PROFILE_ADAPTIVE;
|
|
|
|
|
else if (streq(optarg, "flat"))
|
|
|
|
|
options->profile = LIBINPUT_CONFIG_ACCEL_PROFILE_FLAT;
|
|
|
|
|
else
|
|
|
|
|
return 1;
|
|
|
|
|
break;
|
|
|
|
|
case OPT_DISABLE_SENDEVENTS:
|
|
|
|
|
if (!optarg)
|
|
|
|
|
return 1;
|
2017-09-05 14:12:09 +10:00
|
|
|
|
2018-04-12 11:56:43 +10:00
|
|
|
snprintf(options->disable_pattern,
|
|
|
|
|
sizeof(options->disable_pattern),
|
|
|
|
|
"%s",
|
|
|
|
|
optarg);
|
|
|
|
|
break;
|
2014-12-18 14:42:42 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2014-12-18 15:02:45 +10:00
|
|
|
|
2017-06-19 18:38:33 +10:00
|
|
|
static int
|
|
|
|
|
open_restricted(const char *path, int flags, void *user_data)
|
|
|
|
|
{
|
|
|
|
|
bool *grab = user_data;
|
|
|
|
|
int fd = open(path, flags);
|
|
|
|
|
|
|
|
|
|
if (fd < 0)
|
|
|
|
|
fprintf(stderr, "Failed to open %s (%s)\n",
|
|
|
|
|
path, strerror(errno));
|
2018-06-20 07:56:05 +10:00
|
|
|
else if (grab && *grab && ioctl(fd, EVIOCGRAB, (void*)1) == -1)
|
2017-06-19 18:38:33 +10:00
|
|
|
fprintf(stderr, "Grab requested, but failed for %s (%s)\n",
|
|
|
|
|
path, strerror(errno));
|
|
|
|
|
|
|
|
|
|
return fd < 0 ? -errno : fd;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
close_restricted(int fd, void *user_data)
|
|
|
|
|
{
|
|
|
|
|
close(fd);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const struct libinput_interface interface = {
|
|
|
|
|
.open_restricted = open_restricted,
|
|
|
|
|
.close_restricted = close_restricted,
|
|
|
|
|
};
|
|
|
|
|
|
2014-12-18 15:02:45 +10:00
|
|
|
static struct libinput *
|
2018-06-13 13:38:09 +10:00
|
|
|
tools_open_udev(const char *seat, bool verbose, bool *grab)
|
2014-12-18 15:02:45 +10:00
|
|
|
{
|
|
|
|
|
struct libinput *li;
|
|
|
|
|
struct udev *udev = udev_new();
|
|
|
|
|
|
|
|
|
|
if (!udev) {
|
|
|
|
|
fprintf(stderr, "Failed to initialize udev\n");
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-13 13:38:09 +10:00
|
|
|
li = libinput_udev_create_context(&interface, grab, udev);
|
2014-12-18 15:02:45 +10:00
|
|
|
if (!li) {
|
|
|
|
|
fprintf(stderr, "Failed to initialize context from udev\n");
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-19 14:10:32 +10:00
|
|
|
libinput_log_set_handler(li, log_handler);
|
|
|
|
|
if (verbose)
|
2014-12-18 15:02:45 +10:00
|
|
|
libinput_log_set_priority(li, LIBINPUT_LOG_PRIORITY_DEBUG);
|
|
|
|
|
|
|
|
|
|
if (libinput_udev_assign_seat(li, seat)) {
|
|
|
|
|
fprintf(stderr, "Failed to set seat\n");
|
|
|
|
|
libinput_unref(li);
|
|
|
|
|
li = NULL;
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
out:
|
|
|
|
|
udev_unref(udev);
|
|
|
|
|
return li;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct libinput *
|
2018-06-13 13:38:09 +10:00
|
|
|
tools_open_device(const char *path, bool verbose, bool *grab)
|
2014-12-18 15:02:45 +10:00
|
|
|
{
|
|
|
|
|
struct libinput_device *device;
|
|
|
|
|
struct libinput *li;
|
|
|
|
|
|
2018-06-13 13:38:09 +10:00
|
|
|
li = libinput_path_create_context(&interface, grab);
|
2014-12-18 15:02:45 +10:00
|
|
|
if (!li) {
|
|
|
|
|
fprintf(stderr, "Failed to initialize context from %s\n", path);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (verbose) {
|
|
|
|
|
libinput_log_set_handler(li, log_handler);
|
|
|
|
|
libinput_log_set_priority(li, LIBINPUT_LOG_PRIORITY_DEBUG);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
device = libinput_path_add_device(li, path);
|
|
|
|
|
if (!device) {
|
|
|
|
|
fprintf(stderr, "Failed to initialized device %s\n", path);
|
|
|
|
|
libinput_unref(li);
|
|
|
|
|
li = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return li;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-17 10:02:49 +10:00
|
|
|
static void
|
|
|
|
|
tools_setenv_quirks_dir(void)
|
|
|
|
|
{
|
|
|
|
|
if (tools_execdir_is_builddir(NULL, 0))
|
|
|
|
|
setenv("LIBINPUT_QUIRKS_DIR", LIBINPUT_QUIRKS_SRCDIR, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-18 15:02:45 +10:00
|
|
|
struct libinput *
|
2017-06-19 18:38:33 +10:00
|
|
|
tools_open_backend(enum tools_backend which,
|
|
|
|
|
const char *seat_or_device,
|
|
|
|
|
bool verbose,
|
2018-06-13 13:38:09 +10:00
|
|
|
bool *grab)
|
2014-12-18 15:02:45 +10:00
|
|
|
{
|
2017-06-19 18:38:33 +10:00
|
|
|
struct libinput *li;
|
|
|
|
|
|
2018-07-17 10:02:49 +10:00
|
|
|
tools_setenv_quirks_dir();
|
|
|
|
|
|
2017-06-19 18:38:33 +10:00
|
|
|
switch (which) {
|
|
|
|
|
case BACKEND_UDEV:
|
|
|
|
|
li = tools_open_udev(seat_or_device, verbose, grab);
|
|
|
|
|
break;
|
|
|
|
|
case BACKEND_DEVICE:
|
|
|
|
|
li = tools_open_device(seat_or_device, verbose, grab);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2014-12-18 15:02:45 +10:00
|
|
|
abort();
|
2017-05-25 15:34:30 +10:00
|
|
|
}
|
2014-12-18 15:02:45 +10:00
|
|
|
|
|
|
|
|
return li;
|
|
|
|
|
}
|
2014-12-18 15:10:09 +10:00
|
|
|
|
|
|
|
|
void
|
|
|
|
|
tools_device_apply_config(struct libinput_device *device,
|
|
|
|
|
struct tools_options *options)
|
|
|
|
|
{
|
|
|
|
|
if (options->tapping != -1)
|
|
|
|
|
libinput_device_config_tap_set_enabled(device, options->tapping);
|
2016-12-05 20:31:31 +10:00
|
|
|
if (options->tap_map != (enum libinput_config_tap_button_map)-1)
|
2016-07-21 11:46:05 +10:00
|
|
|
libinput_device_config_tap_set_button_map(device,
|
|
|
|
|
options->tap_map);
|
2016-01-22 17:59:19 +10:00
|
|
|
if (options->drag != -1)
|
|
|
|
|
libinput_device_config_tap_set_drag_enabled(device,
|
|
|
|
|
options->drag);
|
2015-06-22 11:07:31 +10:00
|
|
|
if (options->drag_lock != -1)
|
|
|
|
|
libinput_device_config_tap_set_drag_lock_enabled(device,
|
|
|
|
|
options->drag_lock);
|
2014-12-18 15:10:59 +10:00
|
|
|
if (options->natural_scroll != -1)
|
|
|
|
|
libinput_device_config_scroll_set_natural_scroll_enabled(device,
|
|
|
|
|
options->natural_scroll);
|
2014-12-18 15:14:09 +10:00
|
|
|
if (options->left_handed != -1)
|
2015-01-06 21:20:22 -05:00
|
|
|
libinput_device_config_left_handed_set(device, options->left_handed);
|
2015-04-14 12:08:33 +10:00
|
|
|
if (options->middlebutton != -1)
|
|
|
|
|
libinput_device_config_middle_emulation_set_enabled(device,
|
|
|
|
|
options->middlebutton);
|
2015-01-16 10:16:47 +10:00
|
|
|
|
2015-07-08 15:03:06 +10:00
|
|
|
if (options->dwt != -1)
|
|
|
|
|
libinput_device_config_dwt_set_enabled(device, options->dwt);
|
|
|
|
|
|
2016-12-05 20:31:31 +10:00
|
|
|
if (options->click_method != (enum libinput_config_click_method)-1)
|
2015-01-16 10:16:47 +10:00
|
|
|
libinput_device_config_click_set_method(device, options->click_method);
|
2015-04-10 15:45:59 +10:00
|
|
|
|
2016-12-05 20:31:31 +10:00
|
|
|
if (options->scroll_method != (enum libinput_config_scroll_method)-1)
|
2015-04-21 11:57:23 +10:00
|
|
|
libinput_device_config_scroll_set_method(device,
|
|
|
|
|
options->scroll_method);
|
2015-04-21 18:24:39 +10:00
|
|
|
if (options->scroll_button != -1)
|
|
|
|
|
libinput_device_config_scroll_set_button(device,
|
|
|
|
|
options->scroll_button);
|
2015-04-21 11:57:23 +10:00
|
|
|
|
2015-08-27 13:13:47 +10:00
|
|
|
if (libinput_device_config_accel_is_available(device)) {
|
2015-04-10 15:45:59 +10:00
|
|
|
libinput_device_config_accel_set_speed(device,
|
|
|
|
|
options->speed);
|
2015-08-27 13:13:47 +10:00
|
|
|
if (options->profile != LIBINPUT_CONFIG_ACCEL_PROFILE_NONE)
|
|
|
|
|
libinput_device_config_accel_set_profile(device,
|
|
|
|
|
options->profile);
|
|
|
|
|
}
|
2017-09-05 14:12:09 +10:00
|
|
|
|
|
|
|
|
if (libinput_device_config_send_events_get_modes(device) &
|
|
|
|
|
LIBINPUT_CONFIG_SEND_EVENTS_DISABLED &&
|
|
|
|
|
fnmatch(options->disable_pattern,
|
|
|
|
|
libinput_device_get_name(device),
|
|
|
|
|
0) != FNM_NOMATCH) {
|
|
|
|
|
libinput_device_config_send_events_set_mode(device,
|
|
|
|
|
LIBINPUT_CONFIG_SEND_EVENTS_DISABLED);
|
|
|
|
|
}
|
2014-12-18 15:10:09 +10:00
|
|
|
}
|
2017-05-23 15:07:31 +10:00
|
|
|
|
|
|
|
|
static char*
|
|
|
|
|
find_device(const char *udev_tag)
|
|
|
|
|
{
|
|
|
|
|
struct udev *udev;
|
|
|
|
|
struct udev_enumerate *e;
|
2018-02-14 15:26:27 +10:00
|
|
|
struct udev_list_entry *entry = NULL;
|
2017-05-23 15:07:31 +10:00
|
|
|
struct udev_device *device;
|
|
|
|
|
const char *path, *sysname;
|
|
|
|
|
char *device_node = NULL;
|
|
|
|
|
|
|
|
|
|
udev = udev_new();
|
|
|
|
|
e = udev_enumerate_new(udev);
|
|
|
|
|
udev_enumerate_add_match_subsystem(e, "input");
|
|
|
|
|
udev_enumerate_scan_devices(e);
|
|
|
|
|
|
|
|
|
|
udev_list_entry_foreach(entry, udev_enumerate_get_list_entry(e)) {
|
|
|
|
|
path = udev_list_entry_get_name(entry);
|
|
|
|
|
device = udev_device_new_from_syspath(udev, path);
|
|
|
|
|
if (!device)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
sysname = udev_device_get_sysname(device);
|
|
|
|
|
if (strncmp("event", sysname, 5) != 0) {
|
|
|
|
|
udev_device_unref(device);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (udev_device_get_property_value(device, udev_tag))
|
2017-07-07 09:47:06 +10:00
|
|
|
device_node = safe_strdup(udev_device_get_devnode(device));
|
2017-05-23 15:07:31 +10:00
|
|
|
|
|
|
|
|
udev_device_unref(device);
|
|
|
|
|
|
|
|
|
|
if (device_node)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
udev_enumerate_unref(e);
|
|
|
|
|
udev_unref(udev);
|
|
|
|
|
|
|
|
|
|
return device_node;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
find_touchpad_device(char *path, size_t path_len)
|
|
|
|
|
{
|
|
|
|
|
char *devnode = find_device("ID_INPUT_TOUCHPAD");
|
|
|
|
|
|
|
|
|
|
if (devnode) {
|
|
|
|
|
snprintf(path, path_len, "%s", devnode);
|
|
|
|
|
free(devnode);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return devnode != NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
is_touchpad_device(const char *devnode)
|
|
|
|
|
{
|
|
|
|
|
struct udev *udev;
|
|
|
|
|
struct udev_device *dev = NULL;
|
|
|
|
|
struct stat st;
|
|
|
|
|
bool is_touchpad = false;
|
|
|
|
|
|
|
|
|
|
if (stat(devnode, &st) < 0)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
udev = udev_new();
|
|
|
|
|
dev = udev_device_new_from_devnum(udev, 'c', st.st_rdev);
|
|
|
|
|
if (!dev)
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
is_touchpad = udev_device_get_property_value(dev, "ID_INPUT_TOUCHPAD");
|
|
|
|
|
out:
|
|
|
|
|
if (dev)
|
|
|
|
|
udev_device_unref(dev);
|
|
|
|
|
udev_unref(udev);
|
|
|
|
|
|
|
|
|
|
return is_touchpad;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-17 10:12:16 +10:00
|
|
|
/**
|
|
|
|
|
* Try to read the directory we're executing from and if it matches the
|
|
|
|
|
* builddir, return it.
|
|
|
|
|
*
|
|
|
|
|
* @param execdir_out If not NULL, set to the exec directory
|
|
|
|
|
* @param sz Size of execdir_out
|
|
|
|
|
*
|
|
|
|
|
* @return true if the execdir is the builddir, false otherwise.
|
|
|
|
|
*
|
|
|
|
|
* If execdir_out is NULL and szt is 0, it merely returns true/false.
|
2018-06-27 13:26:51 +10:00
|
|
|
*/
|
2018-07-17 10:12:16 +10:00
|
|
|
bool
|
|
|
|
|
tools_execdir_is_builddir(char *execdir_out, size_t sz)
|
2018-06-27 13:26:51 +10:00
|
|
|
{
|
|
|
|
|
char execdir[PATH_MAX] = {0};
|
|
|
|
|
char *pathsep;
|
2018-07-17 10:12:16 +10:00
|
|
|
ssize_t nread;
|
2018-06-27 13:26:51 +10:00
|
|
|
|
2018-06-27 14:24:54 +10:00
|
|
|
/* In the case of release builds, the builddir is
|
|
|
|
|
the empty string */
|
|
|
|
|
if (streq(MESON_BUILD_ROOT, ""))
|
2018-07-17 10:12:16 +10:00
|
|
|
return false;
|
2018-06-27 14:24:54 +10:00
|
|
|
|
2018-07-17 10:12:16 +10:00
|
|
|
nread = readlink("/proc/self/exe", execdir, sizeof(execdir) - 1);
|
|
|
|
|
if (nread <= 0 || nread == sizeof(execdir) - 1)
|
|
|
|
|
return false;
|
2018-06-27 13:26:51 +10:00
|
|
|
|
|
|
|
|
/* readlink doesn't terminate the string and readlink says
|
|
|
|
|
anything past sz is undefined */
|
2018-07-17 10:12:16 +10:00
|
|
|
execdir[++nread] = '\0';
|
2018-06-27 13:26:51 +10:00
|
|
|
|
|
|
|
|
pathsep = strrchr(execdir, '/');
|
|
|
|
|
if (!pathsep)
|
2018-07-17 10:12:16 +10:00
|
|
|
return false;
|
2018-06-27 13:26:51 +10:00
|
|
|
|
|
|
|
|
*pathsep = '\0';
|
|
|
|
|
if (!streq(execdir, MESON_BUILD_ROOT))
|
2018-07-17 10:12:16 +10:00
|
|
|
return false;
|
2018-06-27 13:26:51 +10:00
|
|
|
|
2018-07-17 10:12:16 +10:00
|
|
|
if (sz > 0) {
|
|
|
|
|
assert(execdir_out != NULL);
|
|
|
|
|
assert(sz >= (size_t)nread);
|
|
|
|
|
snprintf(execdir_out, nread, "%s", execdir);
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2018-06-27 13:26:51 +10:00
|
|
|
}
|
|
|
|
|
|
2017-05-23 15:07:31 +10:00
|
|
|
static inline void
|
|
|
|
|
setup_path(void)
|
|
|
|
|
{
|
|
|
|
|
const char *path = getenv("PATH");
|
|
|
|
|
char new_path[PATH_MAX];
|
2018-07-17 10:12:16 +10:00
|
|
|
char builddir[PATH_MAX];
|
|
|
|
|
const char *extra_path = LIBINPUT_TOOL_PATH;
|
2018-06-27 13:26:51 +10:00
|
|
|
|
2018-07-17 10:12:16 +10:00
|
|
|
if (tools_execdir_is_builddir(builddir, sizeof(builddir)))
|
|
|
|
|
extra_path = builddir;
|
2017-05-23 15:07:31 +10:00
|
|
|
|
|
|
|
|
snprintf(new_path,
|
|
|
|
|
sizeof(new_path),
|
|
|
|
|
"%s:%s",
|
2018-07-17 10:12:16 +10:00
|
|
|
extra_path,
|
2017-05-23 15:07:31 +10:00
|
|
|
path ? path : "");
|
|
|
|
|
setenv("PATH", new_path, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
tools_exec_command(const char *prefix, int real_argc, char **real_argv)
|
|
|
|
|
{
|
|
|
|
|
char *argv[64] = {NULL};
|
|
|
|
|
char executable[128];
|
|
|
|
|
const char *command;
|
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
|
|
assert((size_t)real_argc < ARRAY_LENGTH(argv));
|
|
|
|
|
|
|
|
|
|
command = real_argv[0];
|
|
|
|
|
|
|
|
|
|
rc = snprintf(executable,
|
|
|
|
|
sizeof(executable),
|
|
|
|
|
"%s-%s",
|
|
|
|
|
prefix,
|
|
|
|
|
command);
|
|
|
|
|
if (rc >= (int)sizeof(executable)) {
|
|
|
|
|
fprintf(stderr, "Failed to assemble command.\n");
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
argv[0] = executable;
|
|
|
|
|
for (int i = 1; i < real_argc; i++)
|
|
|
|
|
argv[i] = real_argv[i];
|
|
|
|
|
|
|
|
|
|
setup_path();
|
|
|
|
|
|
|
|
|
|
rc = execvp(executable, argv);
|
2017-11-09 15:17:37 +10:00
|
|
|
if (rc) {
|
|
|
|
|
if (errno == ENOENT) {
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"libinput: %s is not a libinput command or not installed. "
|
|
|
|
|
"See 'libinput --help'\n",
|
|
|
|
|
command);
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"Failed to execute '%s' (%s)\n",
|
|
|
|
|
command,
|
|
|
|
|
strerror(errno));
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-23 15:07:31 +10:00
|
|
|
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
2018-06-22 16:37:16 +10:00
|
|
|
|
|
|
|
|
void
|
|
|
|
|
tools_list_device_quirks(struct quirks_context *ctx,
|
|
|
|
|
struct udev_device *device,
|
|
|
|
|
void (*callback)(void *data, const char *str),
|
|
|
|
|
void *userdata)
|
|
|
|
|
{
|
|
|
|
|
char buf[256];
|
|
|
|
|
|
|
|
|
|
struct quirks *quirks;
|
|
|
|
|
enum quirk qlist[] = {
|
|
|
|
|
QUIRK_MODEL_ALPS_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_APPLE_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_APPLE_MAGICMOUSE,
|
|
|
|
|
QUIRK_MODEL_TABLET_NO_TILT,
|
|
|
|
|
QUIRK_MODEL_APPLE_TOUCHPAD_ONEBUTTON,
|
|
|
|
|
QUIRK_MODEL_TOUCHPAD_VISIBLE_MARKER,
|
|
|
|
|
QUIRK_MODEL_CYBORG_RAT,
|
|
|
|
|
QUIRK_MODEL_CHROMEBOOK,
|
|
|
|
|
QUIRK_MODEL_HP6910_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_HP8510_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_HP_PAVILION_DM4_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_HP_STREAM11_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_HP_ZBOOK_STUDIO_G3,
|
|
|
|
|
QUIRK_MODEL_TABLET_NO_PROXIMITY_OUT,
|
|
|
|
|
QUIRK_MODEL_LENOVO_SCROLLPOINT,
|
|
|
|
|
QUIRK_MODEL_LENOVO_X230,
|
|
|
|
|
QUIRK_MODEL_LENOVO_T450_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_TABLET_MODE_NO_SUSPEND,
|
|
|
|
|
QUIRK_MODEL_LENOVO_CARBON_X1_6TH,
|
|
|
|
|
QUIRK_MODEL_TRACKBALL,
|
|
|
|
|
QUIRK_MODEL_LOGITECH_MARBLE_MOUSE,
|
|
|
|
|
QUIRK_MODEL_BOUNCING_KEYS,
|
|
|
|
|
QUIRK_MODEL_SYNAPTICS_SERIAL_TOUCHPAD,
|
|
|
|
|
QUIRK_MODEL_SYSTEM76_BONOBO,
|
|
|
|
|
QUIRK_MODEL_CLEVO_W740SU,
|
|
|
|
|
QUIRK_MODEL_SYSTEM76_GALAGO,
|
|
|
|
|
QUIRK_MODEL_SYSTEM76_KUDU,
|
|
|
|
|
QUIRK_MODEL_WACOM_TOUCHPAD,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
QUIRK_ATTR_SIZE_HINT,
|
|
|
|
|
QUIRK_ATTR_TOUCH_SIZE_RANGE,
|
|
|
|
|
QUIRK_ATTR_PALM_SIZE_THRESHOLD,
|
|
|
|
|
QUIRK_ATTR_LID_SWITCH_RELIABILITY,
|
|
|
|
|
QUIRK_ATTR_KEYBOARD_INTEGRATION,
|
|
|
|
|
QUIRK_ATTR_TPKBCOMBO_LAYOUT,
|
|
|
|
|
QUIRK_ATTR_PRESSURE_RANGE,
|
|
|
|
|
QUIRK_ATTR_PALM_PRESSURE_THRESHOLD,
|
|
|
|
|
QUIRK_ATTR_RESOLUTION_HINT,
|
2018-07-09 16:04:50 +10:00
|
|
|
QUIRK_ATTR_TRACKPOINT_MULTIPLIER,
|
2018-06-22 16:37:16 +10:00
|
|
|
QUIRK_ATTR_THUMB_PRESSURE_THRESHOLD,
|
2018-08-17 15:12:58 +02:00
|
|
|
QUIRK_ATTR_USE_VELOCITY_AVERAGING,
|
2018-07-25 14:46:14 +10:00
|
|
|
QUIRK_ATTR_THUMB_SIZE_THRESHOLD,
|
2018-08-22 14:13:33 +10:00
|
|
|
QUIRK_ATTR_MSC_TIMESTAMP,
|
2018-06-22 16:37:16 +10:00
|
|
|
};
|
|
|
|
|
enum quirk *q;
|
|
|
|
|
|
|
|
|
|
quirks = quirks_fetch_for_device(ctx, device);
|
|
|
|
|
if (!quirks)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
ARRAY_FOR_EACH(qlist, q) {
|
|
|
|
|
const char *name;
|
|
|
|
|
struct quirk_dimensions dim;
|
|
|
|
|
struct quirk_range r;
|
|
|
|
|
uint32_t v;
|
|
|
|
|
char *s;
|
2018-07-09 16:00:45 +10:00
|
|
|
double d;
|
2018-06-22 16:37:16 +10:00
|
|
|
|
|
|
|
|
if (!quirks_has_quirk(quirks, *q))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
name = quirk_get_name(*q);
|
|
|
|
|
|
|
|
|
|
switch (*q) {
|
|
|
|
|
case QUIRK_MODEL_ALPS_TOUCHPAD:
|
|
|
|
|
case QUIRK_MODEL_APPLE_MAGICMOUSE:
|
2018-09-18 09:31:29 +10:00
|
|
|
case QUIRK_MODEL_APPLE_TOUCHPAD:
|
2018-06-22 16:37:16 +10:00
|
|
|
case QUIRK_MODEL_APPLE_TOUCHPAD_ONEBUTTON:
|
2018-09-18 09:31:29 +10:00
|
|
|
case QUIRK_MODEL_BOUNCING_KEYS:
|
2018-06-22 16:37:16 +10:00
|
|
|
case QUIRK_MODEL_CHROMEBOOK:
|
2018-09-18 09:31:29 +10:00
|
|
|
case QUIRK_MODEL_CLEVO_W740SU:
|
|
|
|
|
case QUIRK_MODEL_CYBORG_RAT:
|
2018-06-22 16:37:16 +10:00
|
|
|
case QUIRK_MODEL_HP6910_TOUCHPAD:
|
|
|
|
|
case QUIRK_MODEL_HP8510_TOUCHPAD:
|
|
|
|
|
case QUIRK_MODEL_HP_PAVILION_DM4_TOUCHPAD:
|
|
|
|
|
case QUIRK_MODEL_HP_STREAM11_TOUCHPAD:
|
|
|
|
|
case QUIRK_MODEL_HP_ZBOOK_STUDIO_G3:
|
2018-09-18 09:31:29 +10:00
|
|
|
case QUIRK_MODEL_LENOVO_CARBON_X1_6TH:
|
2018-06-22 16:37:16 +10:00
|
|
|
case QUIRK_MODEL_LENOVO_SCROLLPOINT:
|
|
|
|
|
case QUIRK_MODEL_LENOVO_T450_TOUCHPAD:
|
2018-09-18 09:31:29 +10:00
|
|
|
case QUIRK_MODEL_LENOVO_X230:
|
2018-06-22 16:37:16 +10:00
|
|
|
case QUIRK_MODEL_LOGITECH_MARBLE_MOUSE:
|
|
|
|
|
case QUIRK_MODEL_SYNAPTICS_SERIAL_TOUCHPAD:
|
|
|
|
|
case QUIRK_MODEL_SYSTEM76_BONOBO:
|
|
|
|
|
case QUIRK_MODEL_SYSTEM76_GALAGO:
|
|
|
|
|
case QUIRK_MODEL_SYSTEM76_KUDU:
|
2018-09-18 09:31:29 +10:00
|
|
|
case QUIRK_MODEL_TABLET_MODE_NO_SUSPEND:
|
|
|
|
|
case QUIRK_MODEL_TABLET_NO_PROXIMITY_OUT:
|
|
|
|
|
case QUIRK_MODEL_TABLET_NO_TILT:
|
|
|
|
|
case QUIRK_MODEL_TOUCHPAD_VISIBLE_MARKER:
|
|
|
|
|
case QUIRK_MODEL_TRACKBALL:
|
2018-06-22 16:37:16 +10:00
|
|
|
case QUIRK_MODEL_WACOM_TOUCHPAD:
|
|
|
|
|
snprintf(buf, sizeof(buf), "%s=1", name);
|
|
|
|
|
callback(userdata, buf);
|
|
|
|
|
break;
|
|
|
|
|
case QUIRK_ATTR_SIZE_HINT:
|
|
|
|
|
case QUIRK_ATTR_RESOLUTION_HINT:
|
|
|
|
|
quirks_get_dimensions(quirks, *q, &dim);
|
2018-09-12 13:28:16 +10:00
|
|
|
snprintf(buf, sizeof(buf), "%s=%zdx%zd", name, dim.x, dim.y);
|
2018-06-22 16:37:16 +10:00
|
|
|
callback(userdata, buf);
|
|
|
|
|
break;
|
|
|
|
|
case QUIRK_ATTR_TOUCH_SIZE_RANGE:
|
|
|
|
|
case QUIRK_ATTR_PRESSURE_RANGE:
|
|
|
|
|
quirks_get_range(quirks, *q, &r);
|
|
|
|
|
snprintf(buf, sizeof(buf), "%s=%d:%d", name, r.upper, r.lower);
|
|
|
|
|
callback(userdata, buf);
|
|
|
|
|
break;
|
|
|
|
|
case QUIRK_ATTR_PALM_SIZE_THRESHOLD:
|
|
|
|
|
case QUIRK_ATTR_PALM_PRESSURE_THRESHOLD:
|
|
|
|
|
case QUIRK_ATTR_THUMB_PRESSURE_THRESHOLD:
|
2018-07-25 14:46:14 +10:00
|
|
|
case QUIRK_ATTR_THUMB_SIZE_THRESHOLD:
|
2018-06-22 16:37:16 +10:00
|
|
|
quirks_get_uint32(quirks, *q, &v);
|
|
|
|
|
snprintf(buf, sizeof(buf), "%s=%u", name, v);
|
|
|
|
|
callback(userdata, buf);
|
|
|
|
|
break;
|
|
|
|
|
case QUIRK_ATTR_LID_SWITCH_RELIABILITY:
|
|
|
|
|
case QUIRK_ATTR_KEYBOARD_INTEGRATION:
|
|
|
|
|
case QUIRK_ATTR_TPKBCOMBO_LAYOUT:
|
2018-08-22 14:13:33 +10:00
|
|
|
case QUIRK_ATTR_MSC_TIMESTAMP:
|
2018-06-22 16:37:16 +10:00
|
|
|
quirks_get_string(quirks, *q, &s);
|
|
|
|
|
snprintf(buf, sizeof(buf), "%s=%s", name, s);
|
|
|
|
|
callback(userdata, buf);
|
|
|
|
|
break;
|
2018-07-09 16:04:50 +10:00
|
|
|
case QUIRK_ATTR_TRACKPOINT_MULTIPLIER:
|
|
|
|
|
quirks_get_double(quirks, *q, &d);
|
|
|
|
|
snprintf(buf, sizeof(buf), "%s=%0.2f\n", name, d);
|
|
|
|
|
callback(userdata, buf);
|
|
|
|
|
break;
|
2018-08-17 15:12:58 +02:00
|
|
|
case QUIRK_ATTR_USE_VELOCITY_AVERAGING:
|
|
|
|
|
snprintf(buf, sizeof(buf), "%s=1", name);
|
|
|
|
|
callback(userdata, buf);
|
|
|
|
|
break;
|
2018-06-22 16:37:16 +10:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
quirks_unref(quirks);
|
|
|
|
|
}
|