/* * 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 "shared.h" enum options { OPT_DEVICE, OPT_UDEV, OPT_GRAB, OPT_HELP, OPT_VERBOSE, OPT_TAP_ENABLE, OPT_TAP_DISABLE, OPT_TAP_MAP, OPT_DRAG_ENABLE, OPT_DRAG_DISABLE, OPT_DRAG_LOCK_ENABLE, OPT_DRAG_LOCK_DISABLE, OPT_NATURAL_SCROLL_ENABLE, OPT_NATURAL_SCROLL_DISABLE, OPT_LEFT_HANDED_ENABLE, OPT_LEFT_HANDED_DISABLE, OPT_MIDDLEBUTTON_ENABLE, OPT_MIDDLEBUTTON_DISABLE, OPT_DWT_ENABLE, OPT_DWT_DISABLE, OPT_CLICK_METHOD, OPT_SCROLL_METHOD, OPT_SCROLL_BUTTON, OPT_SPEED, OPT_PROFILE, OPT_SHOW_KEYCODES, OPT_QUIET, }; 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_usage(const char *command) { printf("Usage: libinput %s [options] [--udev []|--device /dev/input/event0]\n", command); printf("--udev .... Use udev device discovery (default).\n" " Specifying a seat ID is optional.\n" "--device /path/to/device .... open the given device only\n" "\n" "Features:\n" "--enable-tap\n" "--disable-tap.... enable/disable tapping\n" "--enable-drag\n" "--disable-drag.... enable/disable tap-n-drag\n" "--enable-drag-lock\n" "--disable-drag-lock.... enable/disable tapping drag lock\n" "--enable-natural-scrolling\n" "--disable-natural-scrolling.... enable/disable natural scrolling\n" "--enable-left-handed\n" "--disable-left-handed.... enable/disable left-handed button configuration\n" "--enable-middlebutton\n" "--disable-middlebutton.... enable/disable middle button emulation\n" "--enable-dwt\n" "--disable-dwt..... enable/disable disable-while-typing\n" "--set-click-method=[none|clickfinger|buttonareas] .... set the desired click method\n" "--set-scroll-method=[none|twofinger|edge|button] ... set the desired scroll method\n" "--set-scroll-button=BTN_MIDDLE ... set the button to the given button code\n" "--set-profile=[adaptive|flat].... set pointer acceleration profile\n" "--set-speed=.... set pointer acceleration speed (allowed range [-1, 1]) \n" "--set-tap-map=[lrm|lmr] ... set button mapping for tapping\n" "--show-keycodes.... show all key codes while typing\n" "\n" "These options apply to all applicable devices, if a feature\n" "is not explicitly specified it is left at each device's default.\n" "\n" "Other options:\n" "--grab .......... Exclusively grab all openend devices\n" "--help .......... Print this help.\n" "--verbose ....... Print debugging output.\n" "--quiet ......... Only print libinput messages, useful in combination with --verbose.\n"); } void tools_init_context(struct tools_context *context) { struct tools_options *options = &context->options; context->user_data = NULL; 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->backend = BACKEND_UDEV; options->seat = "seat0"; options->speed = 0.0; options->profile = LIBINPUT_CONFIG_ACCEL_PROFILE_NONE; options->show_keycodes = false; } int tools_parse_args(const char *command, int argc, char **argv, struct tools_context *context) { struct tools_options *options = &context->options; while (1) { int c; int option_index = 0; static struct option opts[] = { { "device", required_argument, 0, OPT_DEVICE }, { "udev", no_argument, 0, OPT_UDEV }, { "grab", no_argument, 0, OPT_GRAB }, { "help", no_argument, 0, OPT_HELP }, { "verbose", no_argument, 0, OPT_VERBOSE }, { "quiet", no_argument, 0, OPT_QUIET }, { "enable-tap", no_argument, 0, OPT_TAP_ENABLE }, { "disable-tap", no_argument, 0, OPT_TAP_DISABLE }, { "enable-drag", no_argument, 0, OPT_DRAG_ENABLE }, { "disable-drag", no_argument, 0, OPT_DRAG_DISABLE }, { "enable-drag-lock", no_argument, 0, OPT_DRAG_LOCK_ENABLE }, { "disable-drag-lock", no_argument, 0, OPT_DRAG_LOCK_DISABLE }, { "enable-natural-scrolling", no_argument, 0, OPT_NATURAL_SCROLL_ENABLE }, { "disable-natural-scrolling", no_argument, 0, OPT_NATURAL_SCROLL_DISABLE }, { "enable-left-handed", no_argument, 0, OPT_LEFT_HANDED_ENABLE }, { "disable-left-handed", no_argument, 0, OPT_LEFT_HANDED_DISABLE }, { "enable-middlebutton", no_argument, 0, OPT_MIDDLEBUTTON_ENABLE }, { "disable-middlebutton", no_argument, 0, OPT_MIDDLEBUTTON_DISABLE }, { "enable-dwt", no_argument, 0, OPT_DWT_ENABLE }, { "disable-dwt", no_argument, 0, OPT_DWT_DISABLE }, { "set-click-method", required_argument, 0, OPT_CLICK_METHOD }, { "set-scroll-method", required_argument, 0, OPT_SCROLL_METHOD }, { "set-scroll-button", required_argument, 0, OPT_SCROLL_BUTTON }, { "set-profile", required_argument, 0, OPT_PROFILE }, { "set-tap-map", required_argument, 0, OPT_TAP_MAP }, { "set-speed", required_argument, 0, OPT_SPEED }, { "show-keycodes", no_argument, 0, OPT_SHOW_KEYCODES }, { 0, 0, 0, 0} }; c = getopt_long(argc, argv, "h", opts, &option_index); if (c == -1) break; switch(c) { case 'h': case OPT_HELP: tools_usage(command); exit(0); case OPT_DEVICE: options->backend = BACKEND_DEVICE; if (!optarg) { tools_usage(command); return 1; } options->device = optarg; break; case OPT_UDEV: options->backend = BACKEND_UDEV; if (optarg) options->seat = optarg; break; case OPT_GRAB: options->grab = 1; break; case OPT_VERBOSE: options->verbose = 1; break; case OPT_TAP_ENABLE: options->tapping = 1; break; case OPT_TAP_DISABLE: options->tapping = 0; break; case OPT_TAP_MAP: if (!optarg) { tools_usage(command); 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 { tools_usage(command); 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) { tools_usage(command); 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 { tools_usage(command); return 1; } break; case OPT_SCROLL_METHOD: if (!optarg) { tools_usage(command); 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 { tools_usage(command); return 1; } break; case OPT_SCROLL_BUTTON: if (!optarg) { tools_usage(command); 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) { tools_usage(command); return 1; } options->speed = atof(optarg); break; case OPT_PROFILE: if (!optarg) { tools_usage(command); 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 { tools_usage(command); return 1; } break; case OPT_SHOW_KEYCODES: options->show_keycodes = true; break; case OPT_QUIET: options->quiet = true; break; default: tools_usage(command); return 1; } } if (optind < argc) { tools_usage(command); return 1; } return 0; } static struct libinput * open_udev(const struct libinput_interface *interface, void *userdata, const char *seat, int verbose) { 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, userdata, udev); if (!li) { fprintf(stderr, "Failed to initialize context from udev\n"); goto out; } if (verbose) { libinput_log_set_handler(li, log_handler); 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 * open_device(const struct libinput_interface *interface, void *userdata, const char *path, int verbose) { struct libinput_device *device; struct libinput *li; li = libinput_path_create_context(interface, userdata); 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 int open_restricted(const char *path, int flags, void *user_data) { const struct tools_context *context = user_data; int fd = open(path, flags); if (fd < 0) fprintf(stderr, "Failed to open %s (%s)\n", path, strerror(errno)); else if (context->options.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, }; struct libinput * tools_open_backend(struct tools_context *context) { struct libinput *li = NULL; struct tools_options *options = &context->options; if (options->backend == BACKEND_UDEV) { li = open_udev(&interface, context, options->seat, options->verbose); } else if (options->backend == BACKEND_DEVICE) { li = open_device(&interface, context, options->device, options->verbose); } else { 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); } } static char* find_device(const char *udev_tag) { struct udev *udev; struct udev_enumerate *e; struct udev_list_entry *entry; 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 = 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; } static inline void setup_path(void) { const char *path = getenv("PATH"); char new_path[PATH_MAX]; snprintf(new_path, sizeof(new_path), "%s:%s", LIBINPUT_TOOL_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); fprintf(stderr, "Failed to execute '%s' (%s)\n", command, strerror(errno)); return EXIT_FAILURE; }