/* $NetBSD: mouse.c,v 1.11 2021/09/28 06:20:09 nia Exp $ */ /*- * Copyright (c) 1998, 2006, 2012 The NetBSD Foundation, Inc. * All rights reserved. * * This code is derived from software contributed to The NetBSD Foundation * by Juergen Hannken-Illjes and Julio M. Merino Vidal. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #include #include "wsconsctl.h" static int reverse_scrolling; static int horiz_scroll_dist; static int vert_scroll_dist; static int mstype; static int resolution; static int samplerate; static struct wsmouse_calibcoords calibration; static char *calibration_samples; static struct wsmouse_repeat repeat; static void mouse_get_parameters(int); static void mouse_put_parameters(int); static void mouse_get_calibration(int); static void mouse_put_calibration(int); static void mouse_get_repeat(int); static void mouse_put_repeat(int); struct field mouse_field_tab[] = { { "resolution", &resolution, FMT_UINT, FLG_WRONLY }, { "samplerate", &samplerate, FMT_UINT, FLG_WRONLY }, { "type", &mstype, FMT_MSTYPE, FLG_RDONLY }, { "scroll.reverse", &reverse_scrolling, FMT_INT, FLG_MODIFY }, { "scroll.distance.x", &horiz_scroll_dist, FMT_INT, FLG_MODIFY }, { "scroll.distance.y", &vert_scroll_dist, FMT_INT, FLG_MODIFY }, { "calibration.minx", &calibration.minx, FMT_INT, FLG_MODIFY }, { "calibration.miny", &calibration.miny, FMT_INT, FLG_MODIFY }, { "calibration.maxx", &calibration.maxx, FMT_INT, FLG_MODIFY }, { "calibration.maxy", &calibration.maxy, FMT_INT, FLG_MODIFY }, { "calibration.samples", &calibration_samples, FMT_STRING, FLG_MODIFY }, { "repeat.buttons", &repeat.wr_buttons, FMT_BITFIELD, FLG_MODIFY }, { "repeat.delay.first", &repeat.wr_delay_first, FMT_UINT, FLG_MODIFY }, { "repeat.delay.decrement", &repeat.wr_delay_decrement, FMT_UINT, FLG_MODIFY }, { "repeat.delay.minimum", &repeat.wr_delay_minimum, FMT_UINT, FLG_MODIFY }, }; int mouse_field_tab_len = sizeof(mouse_field_tab)/ sizeof(mouse_field_tab[0]); void mouse_get_values(int fd) { if (field_by_value(&mstype)->flags & FLG_GET) if (ioctl(fd, WSMOUSEIO_GTYPE, &mstype) < 0) err(EXIT_FAILURE, "WSMOUSEIO_GTYPE"); if (field_by_value(&calibration.minx)->flags & FLG_GET || field_by_value(&calibration.miny)->flags & FLG_GET || field_by_value(&calibration.maxx)->flags & FLG_GET || field_by_value(&calibration.maxy)->flags & FLG_GET || field_by_value(&calibration_samples)->flags & FLG_GET) mouse_get_calibration(fd); if (field_by_value(&repeat.wr_buttons)->flags & FLG_GET || field_by_value(&repeat.wr_delay_first)->flags & FLG_GET || field_by_value(&repeat.wr_delay_decrement)->flags & FLG_GET || field_by_value(&repeat.wr_delay_minimum)->flags & FLG_GET) mouse_get_repeat(fd); if (field_by_value(&horiz_scroll_dist)->flags & FLG_GET || field_by_value(&vert_scroll_dist)->flags & FLG_GET || field_by_value(&reverse_scrolling)->flags & FLG_GET) mouse_get_parameters(fd); } static void mouse_get_parameters(int fd) { struct wsmouse_param params[WSMOUSECFG_MAX]; struct wsmouse_parameters pl; unsigned int i; pl.nparams = 0; pl.params = params; if (field_by_value(&reverse_scrolling)->flags & FLG_GET) params[pl.nparams++].key = WSMOUSECFG_REVERSE_SCROLLING; if (field_by_value(&horiz_scroll_dist)->flags & FLG_GET) params[pl.nparams++].key = WSMOUSECFG_HORIZSCROLLDIST; if (field_by_value(&vert_scroll_dist)->flags & FLG_GET) params[pl.nparams++].key = WSMOUSECFG_VERTSCROLLDIST; if (ioctl(fd, WSMOUSEIO_GETPARAMS, &pl) < 0) { if (field_by_value(&horiz_scroll_dist)->flags & FLG_GET) field_disable_by_value(&horiz_scroll_dist); if (field_by_value(&vert_scroll_dist)->flags & FLG_GET) field_disable_by_value(&vert_scroll_dist); if (field_by_value(&reverse_scrolling)->flags & FLG_GET) field_disable_by_value(&reverse_scrolling); return; } for (i = 0; i < pl.nparams; ++i) { switch (params[i].key) { case WSMOUSECFG_REVERSE_SCROLLING: reverse_scrolling = params[i].value; break; case WSMOUSECFG_HORIZSCROLLDIST: horiz_scroll_dist = params[i].value; break; case WSMOUSECFG_VERTSCROLLDIST: vert_scroll_dist = params[i].value; break; } } } static void mouse_get_calibration(int fd) { struct wsmouse_calibcoords tmp; char *samples; char buf[48]; int i; if (ioctl(fd, WSMOUSEIO_GCALIBCOORDS, &tmp) < 0) { field_disable_by_value(&calibration.minx); field_disable_by_value(&calibration.miny); field_disable_by_value(&calibration.maxx); field_disable_by_value(&calibration.maxy); field_disable_by_value(&calibration_samples); return; } if (field_by_value(&calibration.minx)->flags & FLG_GET) calibration.minx = tmp.minx; if (field_by_value(&calibration.miny)->flags & FLG_GET) calibration.miny = tmp.miny; if (field_by_value(&calibration.maxx)->flags & FLG_GET) calibration.maxx = tmp.maxx; if (field_by_value(&calibration.maxy)->flags & FLG_GET) calibration.maxy = tmp.maxy; if (field_by_value(&calibration_samples)->flags & FLG_GET) { free(calibration_samples); if (tmp.samplelen <= 0) { calibration_samples = strdup(""); if (calibration_samples == NULL) err(EXIT_FAILURE, "could not list calibration" " samples"); } else { samples = malloc(tmp.samplelen * sizeof(buf)); if (samples == NULL) err(EXIT_FAILURE, "could not list calibration" " samples"); samples[0] = '\0'; for (i = 0; i < tmp.samplelen; i++) { snprintf(buf, sizeof(buf), "%s%d,%d,%d,%d", (i == 0) ? "" : ":", tmp.samples[i].rawx, tmp.samples[i].rawy, tmp.samples[i].x, tmp.samples[i].y); strcat(samples, buf); } calibration_samples = samples; } } } static void mouse_get_repeat(int fd) { struct wsmouse_repeat tmp; if (ioctl(fd, WSMOUSEIO_GETREPEAT, &tmp) < 0) err(EXIT_FAILURE, "WSMOUSEIO_GETREPEAT"); if (field_by_value(&repeat.wr_buttons)->flags & FLG_GET) repeat.wr_buttons = tmp.wr_buttons; if (field_by_value(&repeat.wr_delay_first)->flags & FLG_GET) repeat.wr_delay_first = tmp.wr_delay_first; if (field_by_value(&repeat.wr_delay_decrement)->flags & FLG_GET) repeat.wr_delay_decrement = tmp.wr_delay_decrement; if (field_by_value(&repeat.wr_delay_minimum)->flags & FLG_GET) repeat.wr_delay_minimum = tmp.wr_delay_minimum; } void mouse_put_values(int fd) { int tmp; if (field_by_value(&resolution)->flags & FLG_SET) { tmp = resolution; if (ioctl(fd, WSMOUSEIO_SRES, &tmp) < 0) err(EXIT_FAILURE, "WSMOUSEIO_SRES"); pr_field(field_by_value(&resolution), " -> "); } if (field_by_value(&samplerate)->flags & FLG_SET) { tmp = samplerate; if (ioctl(fd, WSMOUSEIO_SRATE, &tmp) < 0) err(EXIT_FAILURE, "WSMOUSEIO_SRATE"); pr_field(field_by_value(&samplerate), " -> "); } if (field_by_value(&calibration.minx)->flags & FLG_SET || field_by_value(&calibration.miny)->flags & FLG_SET || field_by_value(&calibration.maxx)->flags & FLG_SET || field_by_value(&calibration.maxy)->flags & FLG_SET || field_by_value(&calibration_samples)->flags & FLG_SET) mouse_put_calibration(fd); if (field_by_value(&repeat.wr_buttons)->flags & FLG_SET || field_by_value(&repeat.wr_delay_first)->flags & FLG_SET || field_by_value(&repeat.wr_delay_decrement)->flags & FLG_SET || field_by_value(&repeat.wr_delay_minimum)->flags & FLG_SET) mouse_put_repeat(fd); if (field_by_value(&horiz_scroll_dist)->flags & FLG_SET || field_by_value(&vert_scroll_dist)->flags & FLG_SET || field_by_value(&reverse_scrolling)->flags & FLG_SET) mouse_put_parameters(fd); } static void mouse_put_parameters(int fd) { struct wsmouse_param params[WSMOUSECFG_MAX]; struct wsmouse_parameters pl; pl.nparams = 0; pl.params = params; if (field_by_value(&reverse_scrolling)->flags & FLG_SET) { params[pl.nparams].key = WSMOUSECFG_REVERSE_SCROLLING; params[pl.nparams++].value = reverse_scrolling; } if (field_by_value(&horiz_scroll_dist)->flags & FLG_SET) { params[pl.nparams].key = WSMOUSECFG_HORIZSCROLLDIST; params[pl.nparams++].value = horiz_scroll_dist; } if (field_by_value(&vert_scroll_dist)->flags & FLG_SET) { params[pl.nparams].key = WSMOUSECFG_VERTSCROLLDIST; params[pl.nparams++].value = vert_scroll_dist; } if (ioctl(fd, WSMOUSEIO_SETPARAMS, &pl) < 0) { if (field_by_value(&horiz_scroll_dist)->flags & FLG_SET) field_disable_by_value(&horiz_scroll_dist); if (field_by_value(&vert_scroll_dist)->flags & FLG_SET) field_disable_by_value(&vert_scroll_dist); if (field_by_value(&vert_scroll_dist)->flags & FLG_SET) field_disable_by_value(&reverse_scrolling); return; } } static void mouse_put_calibration(int fd) { struct wsmouse_calibcoords tmp; int i; const char *p; char *q; /* Fetch current values into the temporary structure. */ if (ioctl(fd, WSMOUSEIO_GCALIBCOORDS, &tmp) < 0) err(EXIT_FAILURE, "WSMOUSEIO_GCALIBCOORDS"); /* Overwrite the desired values in the temporary structure. */ if (field_by_value(&calibration.minx)->flags & FLG_SET) tmp.minx = calibration.minx; if (field_by_value(&calibration.miny)->flags & FLG_SET) tmp.miny = calibration.miny; if (field_by_value(&calibration.maxx)->flags & FLG_SET) tmp.maxx = calibration.maxx; if (field_by_value(&calibration.maxy)->flags & FLG_SET) tmp.maxy = calibration.maxy; if (field_by_value(&calibration_samples)->flags & FLG_SET) { p = calibration_samples; for (i = 0; p[0] != '\0' && i < WSMOUSE_CALIBCOORDS_MAX; i++) { tmp.samples[i].rawx = strtol(p, &q, 0); if (*q != ',') break; p = q + 1; tmp.samples[i].rawy = strtol(p, &q, 0); if (*q != ',') break; p = q + 1; tmp.samples[i].x = strtol(p, &q, 0); if (*q != ',') break; p = q + 1; tmp.samples[i].y = strtol(p, &q, 0); p = q + 1; if (*q != '\0' && *q != ':') break; } if (p[0] != '\0') errx(EXIT_FAILURE, "%s: invalid calibration data", calibration_samples); tmp.samplelen = i; } /* Set new values for calibrating events. */ if (ioctl(fd, WSMOUSEIO_SCALIBCOORDS, &tmp) < 0) err(EXIT_FAILURE, "WSMOUSEIO_SCALIBCOORDS"); /* Now print what changed. */ if (field_by_value(&calibration.minx)->flags & FLG_SET) pr_field(field_by_value(&calibration.minx), " -> "); if (field_by_value(&calibration.miny)->flags & FLG_SET) pr_field(field_by_value(&calibration.miny), " -> "); if (field_by_value(&calibration.maxx)->flags & FLG_SET) pr_field(field_by_value(&calibration.maxx), " -> "); if (field_by_value(&calibration.maxy)->flags & FLG_SET) pr_field(field_by_value(&calibration.maxy), " -> "); if (field_by_value(&calibration_samples)->flags & FLG_SET) pr_field(field_by_value(&calibration_samples), " -> "); } static void mouse_put_repeat(int fd) { struct wsmouse_repeat tmp; /* Fetch current values into the temporary structure. */ if (ioctl(fd, WSMOUSEIO_GETREPEAT, &tmp) < 0) err(EXIT_FAILURE, "WSMOUSEIO_GETREPEAT"); /* Overwrite the desired values in the temporary structure. */ if (field_by_value(&repeat.wr_buttons)->flags & FLG_SET) tmp.wr_buttons = repeat.wr_buttons; if (field_by_value(&repeat.wr_delay_first)->flags & FLG_SET) tmp.wr_delay_first = repeat.wr_delay_first; if (field_by_value(&repeat.wr_delay_decrement)->flags & FLG_SET) tmp.wr_delay_decrement = repeat.wr_delay_decrement; if (field_by_value(&repeat.wr_delay_minimum)->flags & FLG_SET) tmp.wr_delay_minimum = repeat.wr_delay_minimum; /* Set new values for repeating events. */ if (ioctl(fd, WSMOUSEIO_SETREPEAT, &tmp) < 0) err(EXIT_FAILURE, "WSMOUSEIO_SETREPEAT"); /* Now print what changed. */ if (field_by_value(&repeat.wr_buttons)->flags & FLG_SET) pr_field(field_by_value(&repeat.wr_buttons), " -> "); if (field_by_value(&repeat.wr_delay_first)->flags & FLG_SET) pr_field(field_by_value(&repeat.wr_delay_first), " -> "); if (field_by_value(&repeat.wr_delay_decrement)->flags & FLG_SET) pr_field(field_by_value(&repeat.wr_delay_decrement), " -> "); if (field_by_value(&repeat.wr_delay_minimum)->flags & FLG_SET) pr_field(field_by_value(&repeat.wr_delay_minimum), " -> "); }