/* * Copyright © 2014 Red Hat, Inc. * * 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: * * 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. */ #include #include #include #include #include #include #include #include #include #include #include #include #include "shared.h" LIBINPUT_ATTRIBUTE_PRINTF(3, 0) static void log_handler(struct libinput *li, enum libinput_log_priority priority, const char *format, va_list args) { 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); } vprintf(format, args); if (is_tty && priority >= LIBINPUT_LOG_PRIORITY_INFO) printf(ANSI_NORMAL); } void tools_init_options(struct tools_options *options) { memset(options, 0, sizeof(*options)); options->tapping = -1; options->tap_map = -1; options->drag = -1; options->drag_lock = -1; options->natural_scroll = -1; options->left_handed = -1; options->middlebutton = -1; options->dwt = -1; options->click_method = -1; options->scroll_method = -1; options->scroll_button = -1; options->speed = 0.0; options->profile = LIBINPUT_CONFIG_ACCEL_PROFILE_NONE; } int tools_parse_option(int option, const char *optarg, struct tools_options *options) { switch(option) { case OPT_TAP_ENABLE: options->tapping = 1; break; case OPT_TAP_DISABLE: options->tapping = 0; break; case OPT_TAP_MAP: if (!optarg) return 1; 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; 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; 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; 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; snprintf(options->disable_pattern, sizeof(options->disable_pattern), "%s", optarg); break; } return 0; } 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)); else if (grab && *grab && ioctl(fd, EVIOCGRAB, (void*)1) == -1) 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, }; static struct libinput * tools_open_udev(const char *seat, bool verbose, bool *grab) { struct libinput *li; struct udev *udev = udev_new(); if (!udev) { fprintf(stderr, "Failed to initialize udev\n"); return NULL; } li = libinput_udev_create_context(&interface, grab, udev); if (!li) { fprintf(stderr, "Failed to initialize context from udev\n"); goto out; } libinput_log_set_handler(li, log_handler); if (verbose) 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 * tools_open_device(const char *path, bool verbose, bool *grab) { struct libinput_device *device; struct libinput *li; li = libinput_path_create_context(&interface, grab); 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; } static void tools_setenv_quirks_dir(void) { if (tools_execdir_is_builddir(NULL, 0)) setenv("LIBINPUT_QUIRKS_DIR", LIBINPUT_QUIRKS_SRCDIR, 0); } struct libinput * tools_open_backend(enum tools_backend which, const char *seat_or_device, bool verbose, bool *grab) { struct libinput *li; tools_setenv_quirks_dir(); 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: abort(); } return li; } 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); if (options->tap_map != (enum libinput_config_tap_button_map)-1) libinput_device_config_tap_set_button_map(device, options->tap_map); if (options->drag != -1) libinput_device_config_tap_set_drag_enabled(device, options->drag); if (options->drag_lock != -1) libinput_device_config_tap_set_drag_lock_enabled(device, options->drag_lock); if (options->natural_scroll != -1) libinput_device_config_scroll_set_natural_scroll_enabled(device, options->natural_scroll); if (options->left_handed != -1) libinput_device_config_left_handed_set(device, options->left_handed); if (options->middlebutton != -1) libinput_device_config_middle_emulation_set_enabled(device, options->middlebutton); if (options->dwt != -1) libinput_device_config_dwt_set_enabled(device, options->dwt); if (options->click_method != (enum libinput_config_click_method)-1) libinput_device_config_click_set_method(device, options->click_method); if (options->scroll_method != (enum libinput_config_scroll_method)-1) libinput_device_config_scroll_set_method(device, options->scroll_method); if (options->scroll_button != -1) libinput_device_config_scroll_set_button(device, options->scroll_button); if (libinput_device_config_accel_is_available(device)) { libinput_device_config_accel_set_speed(device, options->speed); if (options->profile != LIBINPUT_CONFIG_ACCEL_PROFILE_NONE) libinput_device_config_accel_set_profile(device, options->profile); } 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); } } static char* find_device(const char *udev_tag) { struct udev *udev; struct udev_enumerate *e; struct udev_list_entry *entry = NULL; 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)) device_node = safe_strdup(udev_device_get_devnode(device)); 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; } /** * 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. */ bool tools_execdir_is_builddir(char *execdir_out, size_t sz) { char execdir[PATH_MAX] = {0}; char *pathsep; ssize_t nread; /* In the case of release builds, the builddir is the empty string */ if (streq(MESON_BUILD_ROOT, "")) return false; nread = readlink("/proc/self/exe", execdir, sizeof(execdir) - 1); if (nread <= 0 || nread == sizeof(execdir) - 1) return false; /* readlink doesn't terminate the string and readlink says anything past sz is undefined */ execdir[++nread] = '\0'; pathsep = strrchr(execdir, '/'); if (!pathsep) return false; *pathsep = '\0'; if (!streq(execdir, MESON_BUILD_ROOT)) return false; if (sz > 0) { assert(execdir_out != NULL); assert(sz >= (size_t)nread); snprintf(execdir_out, nread, "%s", execdir); } return true; } static inline void setup_path(void) { const char *path = getenv("PATH"); char new_path[PATH_MAX]; char builddir[PATH_MAX]; const char *extra_path = LIBINPUT_TOOL_PATH; if (tools_execdir_is_builddir(builddir, sizeof(builddir))) extra_path = builddir; snprintf(new_path, sizeof(new_path), "%s:%s", extra_path, 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); 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)); } } return EXIT_FAILURE; } static void sprintf_event_codes(char *buf, size_t sz, struct quirks *quirks) { const struct quirk_tuples *t; size_t off = 0; int printed; const char *name; quirks_get_tuples(quirks, QUIRK_ATTR_EVENT_CODE_DISABLE, &t); name = quirk_get_name(QUIRK_ATTR_EVENT_CODE_DISABLE); printed = snprintf(buf, sz, "%s=", name); assert(printed != -1); off += printed; for (size_t i = 0; off < sz && i < t->ntuples; i++) { const char *name = libevdev_event_code_get_name( t->tuples[i].first, t->tuples[i].second); printed = snprintf(buf + off, sz - off, "%s;", name); assert(printed != -1); off += printed; } } 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 q; quirks = quirks_fetch_for_device(ctx, device); if (!quirks) return; q = QUIRK_MODEL_ALPS_TOUCHPAD; do { if (quirks_has_quirk(quirks, q)) { const char *name; name = quirk_get_name(q); snprintf(buf, sizeof(buf), "%s=1", name); callback(userdata, buf); } } while(++q < _QUIRK_LAST_MODEL_QUIRK_); q = QUIRK_ATTR_SIZE_HINT; do { if (quirks_has_quirk(quirks, q)) { const char *name; struct quirk_dimensions dim; struct quirk_range r; uint32_t v; char *s; double d; name = quirk_get_name(q); switch (q) { case QUIRK_ATTR_SIZE_HINT: case QUIRK_ATTR_RESOLUTION_HINT: quirks_get_dimensions(quirks, q, &dim); snprintf(buf, sizeof(buf), "%s=%zdx%zd", name, dim.x, dim.y); 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: case QUIRK_ATTR_THUMB_SIZE_THRESHOLD: 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: case QUIRK_ATTR_MSC_TIMESTAMP: quirks_get_string(quirks, q, &s); snprintf(buf, sizeof(buf), "%s=%s", name, s); callback(userdata, buf); break; 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; case QUIRK_ATTR_USE_VELOCITY_AVERAGING: snprintf(buf, sizeof(buf), "%s=1", name); callback(userdata, buf); break; case QUIRK_ATTR_EVENT_CODE_DISABLE: sprintf_event_codes(buf, sizeof(buf), quirks); callback(userdata, buf); break; default: abort(); break; } } } while(++q < _QUIRK_LAST_ATTR_QUIRK_); quirks_unref(quirks); }