Home | History | Annotate | Line # | Download | only in usb
uatp.c revision 1.10.4.10
      1  1.10.4.10     skrll /*	$NetBSD: uatp.c,v 1.10.4.10 2017/08/28 17:52:28 skrll Exp $	*/
      2        1.1  riastrad 
      3        1.1  riastrad /*-
      4        1.7  riastrad  * Copyright (c) 2011-2014 The NetBSD Foundation, Inc.
      5        1.1  riastrad  * All rights reserved.
      6        1.1  riastrad  *
      7        1.7  riastrad  * This code is derived from software contributed to The NetBSD Foundation
      8        1.7  riastrad  * by Taylor R. Campbell.
      9        1.7  riastrad  *
     10        1.1  riastrad  * Redistribution and use in source and binary forms, with or without
     11        1.1  riastrad  * modification, are permitted provided that the following conditions
     12        1.1  riastrad  * are met:
     13        1.1  riastrad  * 1. Redistributions of source code must retain the above copyright
     14        1.1  riastrad  *    notice, this list of conditions and the following disclaimer.
     15        1.1  riastrad  * 2. Redistributions in binary form must reproduce the above copyright
     16        1.1  riastrad  *    notice, this list of conditions and the following disclaimer in the
     17        1.1  riastrad  *    documentation and/or other materials provided with the distribution.
     18        1.1  riastrad  *
     19        1.7  riastrad  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     20        1.7  riastrad  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     21        1.7  riastrad  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     22        1.7  riastrad  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     23        1.7  riastrad  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     24        1.7  riastrad  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     25        1.7  riastrad  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     26        1.7  riastrad  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     27        1.7  riastrad  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     28        1.7  riastrad  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     29        1.7  riastrad  * POSSIBILITY OF SUCH DAMAGE.
     30        1.1  riastrad  */
     31        1.1  riastrad 
     32        1.1  riastrad /*
     33        1.1  riastrad  * uatp(4) - USB Apple Trackpad
     34        1.1  riastrad  *
     35        1.1  riastrad  * The uatp driver talks the protocol of the USB trackpads found in
     36        1.1  riastrad  * Apple laptops since 2005, including PowerBooks, iBooks, MacBooks,
     37        1.1  riastrad  * and MacBook Pros.  Some of these also present generic USB HID mice
     38        1.1  riastrad  * on another USB report id, which the ums(4) driver can handle, but
     39        1.1  riastrad  * Apple's protocol gives more detailed sensor data that lets us detect
     40        1.1  riastrad  * multiple fingers to emulate multi-button mice and scroll wheels.
     41        1.1  riastrad  */
     42   1.10.4.2     skrll 
     43        1.1  riastrad /*
     44        1.1  riastrad  * Protocol
     45        1.1  riastrad  *
     46        1.1  riastrad  * The device has a set of horizontal sensors, each being a column at a
     47        1.1  riastrad  * particular position on the x axis that tells you whether there is
     48        1.1  riastrad  * pressure anywhere on that column, and vertical sensors, each being a
     49        1.1  riastrad  * row at a particular position on the y axis that tells you whether
     50        1.1  riastrad  * there is pressure anywhere on that row.
     51        1.1  riastrad  *
     52        1.1  riastrad  * Whenever the device senses anything, it emits a readout of all of
     53        1.1  riastrad  * the sensors, in some model-dependent order.  (For the order, see
     54        1.1  riastrad  * read_sample_1 and read_sample_2.)  Each sensor datum is an unsigned
     55        1.1  riastrad  * eight-bit quantity representing some measure of pressure.  (Of
     56        1.1  riastrad  * course, it really measures capacitance, not pressure, but we'll call
     57        1.1  riastrad  * it `pressure' here.)
     58        1.1  riastrad  */
     59        1.1  riastrad 
     60        1.1  riastrad /*
     61        1.1  riastrad  * Interpretation
     62        1.1  riastrad  *
     63        1.1  riastrad  * To interpret the finger's position on the trackpad, the driver
     64        1.1  riastrad  * computes a weighted average over all possible positions, weighted by
     65        1.1  riastrad  * the pressure at that position.  The weighted average is computed in
     66        1.1  riastrad  * the dimensions of the screen, rather than the trackpad, in order to
     67        1.1  riastrad  * admit a finer resolution of positions than the trackpad grid.
     68        1.1  riastrad  *
     69        1.1  riastrad  * To update the finger's position smoothly on the trackpad, the driver
     70        1.1  riastrad  * computes a weighted average of the old raw position, the old
     71        1.1  riastrad  * smoothed position, and the new smoothed position.  The weights are
     72        1.1  riastrad  * given by the old_raw_weight, old_smoothed_weight, and new_raw_weight
     73        1.1  riastrad  * sysctl knobs.
     74        1.1  riastrad  *
     75        1.1  riastrad  * Finally, to move the cursor, the driver takes the difference between
     76        1.1  riastrad  * the old and new positions and accelerates it according to some
     77        1.1  riastrad  * heuristic knobs that need to be reworked.
     78        1.1  riastrad  *
     79        1.1  riastrad  * Finally, there are some bells & whistles to detect tapping and to
     80        1.1  riastrad  * emulate a three-button mouse by leaving two or three fingers on the
     81        1.1  riastrad  * trackpad while pressing the button.
     82        1.1  riastrad  */
     83        1.1  riastrad 
     84        1.1  riastrad /*
     85        1.1  riastrad  * Future work
     86        1.1  riastrad  *
     87        1.1  riastrad  * With the raw sensor data available, we could implement fancier bells
     88        1.1  riastrad  * & whistles too, such as pinch-to-zoom.  However, wsmouse supports
     89        1.1  riastrad  * only four-dimensional mice with buttons, and we already use two
     90        1.1  riastrad  * dimensions for mousing and two dimensions for scrolling, so there's
     91        1.1  riastrad  * no straightforward way to report zooming and other gestures to the
     92        1.1  riastrad  * operating system.  Probably a better way to do this would be just to
     93        1.1  riastrad  * attach uhid(4) instead of uatp(4) and to read the raw sensors data
     94        1.1  riastrad  * yourself -- but that requires hairy mode switching for recent models
     95        1.1  riastrad  * (see geyser34_enable_raw_mode).
     96        1.1  riastrad  *
     97        1.1  riastrad  * XXX Rework the acceleration knobs.
     98        1.1  riastrad  * XXX Implement edge scrolling.
     99        1.1  riastrad  * XXX Fix sysctl setup; preserve knobs across suspend/resume.
    100        1.1  riastrad  *     (uatp0 detaches and reattaches across suspend/resume, so as
    101        1.1  riastrad  *     written, the sysctl tree is torn down and rebuilt, losing any
    102        1.1  riastrad  *     state the user may have set.)
    103        1.1  riastrad  * XXX Refactor motion state so I can understand it again.
    104        1.1  riastrad  *     Should make a struct uatp_motion for all that state.
    105        1.1  riastrad  * XXX Add hooks for ignoring trackpad input while typing.
    106        1.1  riastrad  */
    107   1.10.4.2     skrll 
    108        1.1  riastrad /*
    109        1.1  riastrad  * Classifying devices
    110        1.1  riastrad  *
    111        1.1  riastrad  * I have only one MacBook to test this driver, but the driver should
    112        1.1  riastrad  * be applicable to almost every Apple laptop made since the beginning
    113        1.1  riastrad  * of 2005, so the driver reports lots of debugging output to help to
    114        1.1  riastrad  * classify devices.  Boot with `boot -v' (verbose) and check the
    115        1.1  riastrad  * output of `dmesg | grep uatp' to answer the following questions:
    116        1.1  riastrad  *
    117        1.1  riastrad  * - What devices (vendor, product, class, subclass, proto, USB HID
    118        1.1  riastrad  *   report dump) fail to attach when you think they should work?
    119        1.1  riastrad  *     (vendor not apple, class not hid, proto not mouse)
    120        1.1  riastrad  *
    121        1.1  riastrad  * - What devices have an unknown product id?
    122        1.1  riastrad  *     `unknown vendor/product id'
    123        1.1  riastrad  *
    124        1.1  riastrad  * - What devices have the wrong screen-to-trackpad ratios?
    125        1.1  riastrad  *     `... x sensors, scaled by ... for ... points on screen'
    126        1.1  riastrad  *     `... y sensors, scaled by ... for ... points on screen'
    127        1.1  riastrad  *   You can tweak hw.uatp0.x_ratio and hw.uatp0.y_ratio to adjust
    128        1.1  riastrad  *   this, up to a maximum of 384 for each value.
    129        1.1  riastrad  *
    130        1.1  riastrad  * - What devices have the wrong input size?
    131        1.1  riastrad  *     `expected input size ... but got ... for Apple trackpad'
    132        1.1  riastrad  *
    133        1.1  riastrad  * - What devices give wrong-sized packets?
    134        1.1  riastrad  *     `discarding ...-byte input'
    135        1.1  riastrad  *
    136        1.1  riastrad  * - What devices split packets in chunks?
    137        1.1  riastrad  *     `partial packet: ... bytes'
    138        1.1  riastrad  *
    139        1.1  riastrad  * - What devices develop large sensor readouts?
    140        1.1  riastrad  *     `large sensor readout: ...'
    141        1.1  riastrad  *
    142        1.1  riastrad  * - What devices have the wrong number of sensors?  Are there parts of
    143        1.1  riastrad  *   your trackpad that the system doesn't seem to notice?  You can
    144        1.1  riastrad  *   tweak hw.uatp0.x_sensors and hw.uatp0.y_sensors, up to a maximum
    145        1.1  riastrad  *   of 32 for each value.
    146        1.1  riastrad  */
    147   1.10.4.2     skrll 
    148        1.1  riastrad #include <sys/cdefs.h>
    149  1.10.4.10     skrll __KERNEL_RCSID(0, "$NetBSD: uatp.c,v 1.10.4.10 2017/08/28 17:52:28 skrll Exp $");
    150   1.10.4.8     skrll 
    151   1.10.4.8     skrll #ifdef _KERNEL_OPT
    152   1.10.4.8     skrll #include "opt_usb.h"
    153   1.10.4.8     skrll #endif
    154        1.1  riastrad 
    155        1.6  riastrad #include <sys/types.h>
    156        1.6  riastrad #include <sys/param.h>
    157        1.1  riastrad #include <sys/atomic.h>
    158        1.1  riastrad #include <sys/device.h>
    159        1.1  riastrad #include <sys/errno.h>
    160        1.1  riastrad #include <sys/ioctl.h>
    161        1.6  riastrad #include <sys/kernel.h>
    162        1.9  riastrad #include <sys/module.h>
    163        1.1  riastrad #include <sys/sysctl.h>
    164        1.1  riastrad #include <sys/systm.h>
    165        1.1  riastrad #include <sys/time.h>
    166        1.1  riastrad 
    167        1.1  riastrad /* Order is important here...sigh...  */
    168        1.1  riastrad #include <dev/usb/usb.h>
    169        1.1  riastrad #include <dev/usb/usbdi.h>
    170        1.1  riastrad #include <dev/usb/usbdi_util.h>
    171        1.1  riastrad #include <dev/usb/usbdevs.h>
    172        1.1  riastrad #include <dev/usb/uhidev.h>
    173        1.1  riastrad #include <dev/usb/hid.h>
    174        1.1  riastrad #include <dev/usb/usbhid.h>
    175        1.1  riastrad 
    176        1.1  riastrad #include <dev/wscons/wsconsio.h>
    177        1.1  riastrad #include <dev/wscons/wsmousevar.h>
    178        1.1  riastrad 
    179        1.1  riastrad #define CHECK(condition, fail) do {					\
    180        1.1  riastrad 	if (! (condition)) {						\
    181        1.1  riastrad 		aprint_error_dev(uatp_dev(sc), "%s: check failed: %s\n",\
    182        1.1  riastrad 			__func__, #condition);				\
    183        1.1  riastrad 		fail;							\
    184        1.1  riastrad 	}								\
    185        1.1  riastrad } while (0)
    186   1.10.4.2     skrll 
    187        1.1  riastrad #define UATP_DEBUG_ATTACH	(1 << 0)
    188        1.1  riastrad #define UATP_DEBUG_MISC		(1 << 1)
    189        1.1  riastrad #define UATP_DEBUG_WSMOUSE	(1 << 2)
    190        1.1  riastrad #define UATP_DEBUG_IOCTL	(1 << 3)
    191        1.1  riastrad #define UATP_DEBUG_RESET	(1 << 4)
    192        1.1  riastrad #define UATP_DEBUG_INTR		(1 << 5)
    193        1.1  riastrad #define UATP_DEBUG_PARSE	(1 << 6)
    194        1.1  riastrad #define UATP_DEBUG_TAP		(1 << 7)
    195        1.1  riastrad #define UATP_DEBUG_EMUL_BUTTON	(1 << 8)
    196        1.1  riastrad #define UATP_DEBUG_ACCUMULATE	(1 << 9)
    197        1.1  riastrad #define UATP_DEBUG_STATUS	(1 << 10)
    198        1.1  riastrad #define UATP_DEBUG_SPURINTR	(1 << 11)
    199        1.1  riastrad #define UATP_DEBUG_MOVE		(1 << 12)
    200        1.1  riastrad #define UATP_DEBUG_ACCEL	(1 << 13)
    201        1.1  riastrad #define UATP_DEBUG_TRACK_DIST	(1 << 14)
    202        1.1  riastrad #define UATP_DEBUG_PALM		(1 << 15)
    203        1.1  riastrad 
    204        1.1  riastrad #if UATP_DEBUG
    205        1.1  riastrad #  define DPRINTF(sc, flags, format) do {				\
    206        1.1  riastrad 	if ((flags) & (sc)->sc_debug_flags) {				\
    207        1.1  riastrad 		printf("%s: %s: ", device_xname(uatp_dev(sc)), __func__); \
    208        1.1  riastrad 		printf format;						\
    209        1.1  riastrad 	}								\
    210        1.1  riastrad } while (0)
    211        1.1  riastrad #else
    212        1.1  riastrad #  define DPRINTF(sc, flags, format) do {} while (0)
    213        1.1  riastrad #endif
    214        1.1  riastrad 
    215        1.1  riastrad /* Maximum number of bytes in an incoming packet of sensor data.  */
    216        1.1  riastrad #define UATP_MAX_INPUT_SIZE	81
    217        1.1  riastrad 
    218        1.1  riastrad /* Maximum number of sensors in each dimension.  */
    219        1.1  riastrad #define UATP_MAX_X_SENSORS	32
    220        1.1  riastrad #define UATP_MAX_Y_SENSORS	32
    221        1.1  riastrad #define UATP_MAX_SENSORS	32
    222        1.1  riastrad #define UATP_SENSORS		(UATP_MAX_X_SENSORS + UATP_MAX_Y_SENSORS)
    223        1.1  riastrad 
    224        1.1  riastrad /* Maximum accumulated sensor value.  */
    225        1.1  riastrad #define UATP_MAX_ACC		0xff
    226        1.1  riastrad 
    227        1.1  riastrad /* Maximum screen dimension to sensor dimension ratios.  */
    228        1.1  riastrad #define UATP_MAX_X_RATIO	0x180
    229        1.1  riastrad #define UATP_MAX_Y_RATIO	0x180
    230        1.1  riastrad #define UATP_MAX_RATIO		0x180
    231        1.1  riastrad 
    232        1.1  riastrad /* Maximum weight for positions in motion calculation.  */
    233        1.1  riastrad #define UATP_MAX_WEIGHT		0x7f
    234        1.1  riastrad 
    235        1.1  riastrad /* Maximum possible trackpad position in a single dimension.  */
    236        1.1  riastrad #define UATP_MAX_POSITION	(UATP_MAX_SENSORS * UATP_MAX_RATIO)
    237        1.1  riastrad 
    238        1.1  riastrad /* Bounds on acceleration.  */
    239        1.1  riastrad #define UATP_MAX_MOTION_MULTIPLIER	16
    240        1.1  riastrad 
    241        1.1  riastrad /* Status bits transmitted in the last byte of an input packet.  */
    242        1.1  riastrad #define UATP_STATUS_BUTTON	(1 << 0)	/* Button pressed */
    243        1.1  riastrad #define UATP_STATUS_BASE	(1 << 2)	/* Base sensor data */
    244        1.1  riastrad #define UATP_STATUS_POST_RESET	(1 << 4)	/* Post-reset */
    245   1.10.4.2     skrll 
    246        1.1  riastrad /* Forward declarations */
    247        1.1  riastrad 
    248        1.1  riastrad struct uatp_softc;		/* Device driver state.  */
    249        1.1  riastrad struct uatp_descriptor;		/* Descriptor for a particular model.  */
    250        1.1  riastrad struct uatp_parameters;		/* Parameters common to a set of models.  */
    251        1.1  riastrad struct uatp_knobs;		/* User-settable configuration knobs.  */
    252        1.1  riastrad enum uatp_tap_state {
    253        1.1  riastrad 	TAP_STATE_INITIAL,
    254        1.1  riastrad 	TAP_STATE_TAPPING,
    255        1.1  riastrad 	TAP_STATE_TAPPED,
    256        1.1  riastrad 	TAP_STATE_DOUBLE_TAPPING,
    257        1.1  riastrad 	TAP_STATE_DRAGGING_DOWN,
    258        1.1  riastrad 	TAP_STATE_DRAGGING_UP,
    259        1.1  riastrad 	TAP_STATE_TAPPING_IN_DRAG,
    260        1.1  riastrad };
    261        1.1  riastrad 
    262        1.1  riastrad static const struct uatp_descriptor *find_uatp_descriptor
    263        1.1  riastrad     (const struct uhidev_attach_arg *);
    264        1.1  riastrad static device_t uatp_dev(const struct uatp_softc *);
    265        1.1  riastrad static uint8_t *uatp_x_sample(struct uatp_softc *);
    266        1.1  riastrad static uint8_t *uatp_y_sample(struct uatp_softc *);
    267        1.1  riastrad static int *uatp_x_acc(struct uatp_softc *);
    268        1.1  riastrad static int *uatp_y_acc(struct uatp_softc *);
    269        1.1  riastrad static void uatp_clear_position(struct uatp_softc *);
    270        1.1  riastrad static unsigned int uatp_x_sensors(const struct uatp_softc *);
    271        1.1  riastrad static unsigned int uatp_y_sensors(const struct uatp_softc *);
    272        1.1  riastrad static unsigned int uatp_x_ratio(const struct uatp_softc *);
    273        1.1  riastrad static unsigned int uatp_y_ratio(const struct uatp_softc *);
    274        1.1  riastrad static unsigned int uatp_old_raw_weight(const struct uatp_softc *);
    275        1.1  riastrad static unsigned int uatp_old_smoothed_weight(const struct uatp_softc *);
    276        1.1  riastrad static unsigned int uatp_new_raw_weight(const struct uatp_softc *);
    277        1.1  riastrad static int scale_motion(const struct uatp_softc *, int, int *,
    278        1.1  riastrad     const unsigned int *, const unsigned int *);
    279        1.1  riastrad static int uatp_scale_motion(const struct uatp_softc *, int, int *);
    280        1.1  riastrad static int uatp_scale_fast_motion(const struct uatp_softc *, int, int *);
    281        1.1  riastrad static int uatp_match(device_t, cfdata_t, void *);
    282        1.1  riastrad static void uatp_attach(device_t, device_t, void *);
    283        1.1  riastrad static void uatp_setup_sysctl(struct uatp_softc *);
    284        1.1  riastrad static bool uatp_setup_sysctl_knob(struct uatp_softc *, int *, const char *,
    285        1.1  riastrad     const char *);
    286        1.1  riastrad static void uatp_childdet(device_t, device_t);
    287        1.1  riastrad static int uatp_detach(device_t, int);
    288        1.1  riastrad static int uatp_activate(device_t, enum devact);
    289        1.1  riastrad static int uatp_enable(void *);
    290        1.1  riastrad static void uatp_disable(void *);
    291        1.1  riastrad static int uatp_ioctl(void *, unsigned long, void *, int, struct lwp *);
    292        1.1  riastrad static void geyser34_enable_raw_mode(struct uatp_softc *);
    293        1.1  riastrad static void geyser34_initialize(struct uatp_softc *);
    294        1.1  riastrad static int geyser34_finalize(struct uatp_softc *);
    295        1.1  riastrad static void geyser34_deferred_reset(struct uatp_softc *);
    296        1.8  riastrad static void geyser34_reset_task(void *);
    297        1.1  riastrad static void uatp_intr(struct uhidev *, void *, unsigned int);
    298        1.1  riastrad static bool base_sample_softc_flag(const struct uatp_softc *, const uint8_t *);
    299        1.1  riastrad static bool base_sample_input_flag(const struct uatp_softc *, const uint8_t *);
    300        1.1  riastrad static void read_sample_1(uint8_t *, uint8_t *, const uint8_t *);
    301        1.1  riastrad static void read_sample_2(uint8_t *, uint8_t *, const uint8_t *);
    302        1.1  riastrad static void accumulate_sample_1(struct uatp_softc *);
    303        1.1  riastrad static void accumulate_sample_2(struct uatp_softc *);
    304        1.1  riastrad static void uatp_input(struct uatp_softc *, uint32_t, int, int, int, int);
    305        1.1  riastrad static uint32_t uatp_tapped_buttons(struct uatp_softc *);
    306        1.1  riastrad static bool interpret_input(struct uatp_softc *, int *, int *, int *, int *,
    307        1.1  riastrad     uint32_t *);
    308        1.1  riastrad static unsigned int interpret_dimension(struct uatp_softc *, const int *,
    309        1.1  riastrad     unsigned int, unsigned int, unsigned int *, unsigned int *);
    310        1.1  riastrad static void tap_initialize(struct uatp_softc *);
    311        1.1  riastrad static void tap_finalize(struct uatp_softc *);
    312        1.1  riastrad static void tap_enable(struct uatp_softc *);
    313        1.1  riastrad static void tap_disable(struct uatp_softc *);
    314        1.1  riastrad static void tap_transition(struct uatp_softc *, enum uatp_tap_state,
    315        1.1  riastrad     const struct timeval *, unsigned int, unsigned int);
    316        1.1  riastrad static void tap_transition_initial(struct uatp_softc *);
    317        1.1  riastrad static void tap_transition_tapping(struct uatp_softc *, const struct timeval *,
    318        1.1  riastrad     unsigned int);
    319        1.1  riastrad static void tap_transition_double_tapping(struct uatp_softc *,
    320        1.1  riastrad     const struct timeval *, unsigned int);
    321        1.1  riastrad static void tap_transition_dragging_down(struct uatp_softc *);
    322        1.1  riastrad static void tap_transition_tapping_in_drag(struct uatp_softc *,
    323        1.1  riastrad     const struct timeval *, unsigned int);
    324        1.1  riastrad static void tap_transition_tapped(struct uatp_softc *, const struct timeval *);
    325        1.1  riastrad static void tap_transition_dragging_up(struct uatp_softc *);
    326        1.1  riastrad static void tap_reset(struct uatp_softc *);
    327        1.1  riastrad static void tap_reset_wait(struct uatp_softc *);
    328        1.1  riastrad static void tap_touched(struct uatp_softc *, unsigned int);
    329        1.1  riastrad static bool tap_released(struct uatp_softc *);
    330        1.1  riastrad static void schedule_untap(struct uatp_softc *);
    331        1.1  riastrad static void untap_callout(void *);
    332        1.1  riastrad static uint32_t emulated_buttons(struct uatp_softc *, unsigned int);
    333        1.1  riastrad static void update_position(struct uatp_softc *, unsigned int,
    334        1.1  riastrad     unsigned int, unsigned int, int *, int *, int *, int *);
    335        1.1  riastrad static void move_mouse(struct uatp_softc *, unsigned int, unsigned int,
    336        1.1  riastrad     int *, int *);
    337        1.1  riastrad static void scroll_wheel(struct uatp_softc *, unsigned int, unsigned int,
    338        1.1  riastrad     int *, int *);
    339        1.1  riastrad static void move(struct uatp_softc *, const char *, unsigned int, unsigned int,
    340        1.1  riastrad     int *, int *, int *, int *, unsigned int *, unsigned int *, int *, int *);
    341        1.1  riastrad static int smooth(struct uatp_softc *, unsigned int, unsigned int,
    342        1.1  riastrad     unsigned int);
    343        1.1  riastrad static bool motion_below_threshold(struct uatp_softc *, unsigned int,
    344        1.1  riastrad     int, int);
    345        1.1  riastrad static int accelerate(struct uatp_softc *, unsigned int, unsigned int,
    346        1.1  riastrad     unsigned int, unsigned int, bool, int *);
    347   1.10.4.2     skrll 
    348        1.1  riastrad struct uatp_knobs {
    349        1.1  riastrad 	/*
    350        1.1  riastrad 	 * Button emulation.  What do we do when two or three fingers
    351        1.1  riastrad 	 * are on the trackpad when the user presses the button?
    352        1.1  riastrad 	 */
    353        1.1  riastrad 	unsigned int two_finger_buttons;
    354        1.1  riastrad 	unsigned int three_finger_buttons;
    355        1.1  riastrad 
    356        1.1  riastrad #if 0
    357        1.1  riastrad 	/*
    358        1.1  riastrad 	 * Edge scrolling.
    359        1.1  riastrad 	 *
    360        1.1  riastrad 	 * XXX Implement this.  What units should these be in?
    361        1.1  riastrad 	 */
    362        1.1  riastrad 	unsigned int top_edge;
    363        1.1  riastrad 	unsigned int bottom_edge;
    364        1.1  riastrad 	unsigned int left_edge;
    365        1.1  riastrad 	unsigned int right_edge;
    366        1.1  riastrad #endif
    367        1.1  riastrad 
    368        1.1  riastrad 	/*
    369        1.1  riastrad 	 * Multifinger tracking.  What do we do with multiple fingers?
    370        1.1  riastrad 	 * 0. Ignore them.
    371        1.1  riastrad 	 * 1. Try to interpret them as ordinary mousing.
    372        1.1  riastrad 	 * 2. Act like a two-dimensional scroll wheel.
    373        1.1  riastrad 	 */
    374        1.1  riastrad 	unsigned int multifinger_track;
    375        1.1  riastrad 
    376        1.1  riastrad 	/*
    377        1.1  riastrad 	 * Sensor parameters.
    378        1.1  riastrad 	 */
    379        1.1  riastrad 	unsigned int x_sensors;
    380        1.1  riastrad 	unsigned int x_ratio;
    381        1.1  riastrad 	unsigned int y_sensors;
    382        1.1  riastrad 	unsigned int y_ratio;
    383        1.1  riastrad 	unsigned int sensor_threshold;
    384        1.1  riastrad 	unsigned int sensor_normalizer;
    385        1.1  riastrad 	unsigned int palm_width;
    386        1.1  riastrad 	unsigned int old_raw_weight;
    387        1.1  riastrad 	unsigned int old_smoothed_weight;
    388        1.1  riastrad 	unsigned int new_raw_weight;
    389        1.1  riastrad 
    390        1.1  riastrad 	/*
    391        1.1  riastrad 	 * Motion parameters.
    392        1.1  riastrad 	 *
    393        1.1  riastrad 	 * XXX There should be a more principled model of acceleration.
    394        1.1  riastrad 	 */
    395        1.1  riastrad 	unsigned int motion_remainder;
    396        1.1  riastrad 	unsigned int motion_threshold;
    397        1.1  riastrad 	unsigned int motion_multiplier;
    398        1.1  riastrad 	unsigned int motion_divisor;
    399        1.1  riastrad 	unsigned int fast_motion_threshold;
    400        1.1  riastrad 	unsigned int fast_motion_multiplier;
    401        1.1  riastrad 	unsigned int fast_motion_divisor;
    402        1.1  riastrad 	unsigned int fast_per_direction;
    403        1.1  riastrad 	unsigned int motion_delay;
    404        1.1  riastrad 
    405        1.1  riastrad 	/*
    406        1.1  riastrad 	 * Tapping.
    407        1.1  riastrad 	 */
    408        1.1  riastrad 	unsigned int tap_limit_msec;
    409        1.1  riastrad 	unsigned int double_tap_limit_msec;
    410        1.1  riastrad 	unsigned int one_finger_tap_buttons;
    411        1.1  riastrad 	unsigned int two_finger_tap_buttons;
    412        1.1  riastrad 	unsigned int three_finger_tap_buttons;
    413        1.1  riastrad 	unsigned int tap_track_distance_limit;
    414        1.1  riastrad };
    415   1.10.4.2     skrll 
    416        1.1  riastrad static const struct uatp_knobs default_knobs = {
    417        1.1  riastrad 	/*
    418        1.1  riastrad 	 * Button emulation.  Fingers on the trackpad don't change it
    419        1.1  riastrad 	 * by default -- it's still the left button.
    420        1.1  riastrad 	 *
    421        1.1  riastrad 	 * XXX The left button should have a name.
    422        1.1  riastrad 	 */
    423        1.1  riastrad 	 .two_finger_buttons	= 1,
    424        1.1  riastrad 	 .three_finger_buttons	= 1,
    425        1.1  riastrad 
    426        1.1  riastrad #if 0
    427        1.1  riastrad 	/*
    428        1.1  riastrad 	 * Edge scrolling.  Off by default.
    429        1.1  riastrad 	 */
    430        1.1  riastrad 	.top_edge		= 0,
    431        1.1  riastrad 	.bottom_edge		= 0,
    432        1.1  riastrad 	.left_edge		= 0,
    433        1.1  riastrad 	.right_edge		= 0,
    434        1.1  riastrad #endif
    435        1.1  riastrad 
    436        1.1  riastrad 	/*
    437        1.1  riastrad 	 * Multifinger tracking.  Ignore by default.
    438        1.1  riastrad 	 */
    439        1.1  riastrad 	 .multifinger_track	= 0,
    440        1.1  riastrad 
    441        1.1  riastrad 	/*
    442        1.1  riastrad 	 * Sensor parameters.
    443        1.1  riastrad 	 */
    444        1.1  riastrad 	.x_sensors		= 0,	/* default for model */
    445        1.1  riastrad 	.x_ratio		= 0,	/* default for model */
    446        1.1  riastrad 	.y_sensors		= 0,	/* default for model */
    447        1.1  riastrad 	.y_ratio		= 0,	/* default for model */
    448        1.1  riastrad 	.sensor_threshold	= 5,
    449        1.1  riastrad 	.sensor_normalizer	= 5,
    450        1.1  riastrad 	.palm_width		= 0,	/* palm detection disabled */
    451        1.1  riastrad 	.old_raw_weight		= 0,
    452        1.1  riastrad 	.old_smoothed_weight	= 5,
    453        1.1  riastrad 	.new_raw_weight		= 1,
    454        1.1  riastrad 
    455        1.1  riastrad 	/*
    456        1.1  riastrad 	 * Motion parameters.
    457        1.1  riastrad 	 */
    458        1.1  riastrad 	.motion_remainder	= 1,
    459        1.1  riastrad 	.motion_threshold	= 0,
    460        1.1  riastrad 	.motion_multiplier	= 1,
    461        1.1  riastrad 	.motion_divisor		= 1,
    462        1.1  riastrad 	.fast_motion_threshold	= 10,
    463        1.1  riastrad 	.fast_motion_multiplier	= 3,
    464        1.1  riastrad 	.fast_motion_divisor	= 2,
    465        1.1  riastrad 	.fast_per_direction	= 0,
    466        1.1  riastrad 	.motion_delay		= 4,
    467        1.1  riastrad 
    468        1.1  riastrad 	/*
    469        1.1  riastrad 	 * Tapping.  Disabled by default, with a reasonable time set
    470        1.1  riastrad 	 * nevertheless so that you can just set the buttons to enable
    471        1.1  riastrad 	 * it.
    472        1.1  riastrad 	 */
    473        1.1  riastrad 	.tap_limit_msec			= 100,
    474        1.1  riastrad 	.double_tap_limit_msec		= 200,
    475        1.1  riastrad 	.one_finger_tap_buttons		= 0,
    476        1.1  riastrad 	.two_finger_tap_buttons		= 0,
    477        1.1  riastrad 	.three_finger_tap_buttons	= 0,
    478        1.1  riastrad 	.tap_track_distance_limit	= 200,
    479        1.1  riastrad };
    480   1.10.4.2     skrll 
    481        1.1  riastrad struct uatp_softc {
    482        1.1  riastrad 	struct uhidev sc_hdev;		/* USB parent.  */
    483        1.1  riastrad 	device_t sc_wsmousedev;		/* Attached wsmouse device.  */
    484        1.1  riastrad 	const struct uatp_parameters *sc_parameters;
    485        1.1  riastrad 	struct uatp_knobs sc_knobs;
    486        1.1  riastrad 	struct sysctllog *sc_log;	/* Log for sysctl knobs.  */
    487        1.1  riastrad 	const struct sysctlnode *sc_node;	/* Our sysctl node.  */
    488        1.1  riastrad 	unsigned int sc_input_size;	/* Input packet size.  */
    489        1.1  riastrad 	uint8_t sc_input[UATP_MAX_INPUT_SIZE];	/* Buffer for a packet.   */
    490        1.1  riastrad 	unsigned int sc_input_index;	/* Current index into sc_input.  */
    491        1.1  riastrad 	int sc_acc[UATP_SENSORS];	/* Accumulated sensor state.  */
    492        1.1  riastrad 	uint8_t sc_base[UATP_SENSORS];	/* Base sample.  */
    493        1.1  riastrad 	uint8_t sc_sample[UATP_SENSORS];/* Current sample.  */
    494        1.1  riastrad 	unsigned int sc_motion_timer;	/* XXX describe; motion_delay  */
    495        1.1  riastrad 	int sc_x_raw;			/* Raw horiz. mouse position.  */
    496        1.1  riastrad 	int sc_y_raw;			/* Raw vert. mouse position.  */
    497        1.1  riastrad 	int sc_z_raw;			/* Raw horiz. scroll position.  */
    498        1.1  riastrad 	int sc_w_raw;			/* Raw vert. scroll position.  */
    499        1.1  riastrad 	int sc_x_smoothed;		/* Smoothed horiz. mouse position.  */
    500        1.1  riastrad 	int sc_y_smoothed;		/* Smoothed vert. mouse position.  */
    501        1.1  riastrad 	int sc_z_smoothed;		/* Smoothed horiz. scroll position.  */
    502        1.1  riastrad 	int sc_w_smoothed;		/* Smoothed vert. scroll position.  */
    503        1.1  riastrad 	int sc_x_remainder;		/* Remainders from acceleration.  */
    504        1.1  riastrad 	int sc_y_remainder;
    505        1.1  riastrad 	int sc_z_remainder;
    506        1.1  riastrad 	int sc_w_remainder;
    507        1.1  riastrad 	unsigned int sc_track_distance;	/* Distance^2 finger has tracked,
    508        1.1  riastrad 					 * squared to avoid sqrt in kernel.  */
    509        1.1  riastrad 	uint32_t sc_status;		/* Status flags:  */
    510        1.1  riastrad #define UATP_ENABLED	(1 << 0)	/* . Is the wsmouse enabled?  */
    511        1.1  riastrad #define UATP_DYING	(1 << 1)	/* . Have we been deactivated?  */
    512        1.1  riastrad #define UATP_VALID	(1 << 2)	/* . Do we have valid sensor data?  */
    513        1.8  riastrad 	struct usb_task sc_reset_task;	/* Task for resetting device.  */
    514        1.1  riastrad 
    515        1.1  riastrad 	callout_t sc_untap_callout;	/* Releases button after tap.  */
    516        1.1  riastrad 	kmutex_t sc_tap_mutex;		/* Protects the following fields.  */
    517        1.1  riastrad 	enum uatp_tap_state sc_tap_state;	/* Current tap state.  */
    518        1.1  riastrad 	unsigned int sc_tapping_fingers;	/* No. fingers tapping.  */
    519        1.1  riastrad 	unsigned int sc_tapped_fingers;	/* No. fingers of last tap.  */
    520        1.1  riastrad 	struct timeval sc_tap_timer;	/* Timer for tap state transitions.  */
    521        1.1  riastrad 	uint32_t sc_buttons;		/* Physical buttons pressed.  */
    522        1.1  riastrad 	uint32_t sc_all_buttons;	/* Buttons pressed or tapped.  */
    523        1.1  riastrad 
    524        1.1  riastrad #if UATP_DEBUG
    525        1.1  riastrad 	uint32_t sc_debug_flags;	/* Debugging output enabled.  */
    526        1.1  riastrad #endif
    527        1.1  riastrad };
    528   1.10.4.2     skrll 
    529        1.1  riastrad struct uatp_descriptor {
    530        1.1  riastrad 	uint16_t vendor;
    531        1.1  riastrad 	uint16_t product;
    532        1.1  riastrad 	const char *description;
    533        1.1  riastrad 	const struct uatp_parameters *parameters;
    534        1.1  riastrad };
    535        1.1  riastrad 
    536        1.1  riastrad struct uatp_parameters {
    537        1.1  riastrad 	unsigned int x_ratio;		/* Screen width / trackpad width.  */
    538        1.1  riastrad 	unsigned int x_sensors;		/* Number of horizontal sensors.  */
    539        1.1  riastrad 	unsigned int x_sensors_17;	/* XXX Same, on a 17" laptop.  */
    540        1.1  riastrad 	unsigned int y_ratio;		/* Screen height / trackpad height.  */
    541        1.1  riastrad 	unsigned int y_sensors;		/* Number of vertical sensors.  */
    542        1.1  riastrad 	unsigned int input_size;	/* Size in bytes of input packets.  */
    543        1.1  riastrad 
    544        1.1  riastrad 	/* Device-specific initialization routine.  May be null.  */
    545        1.1  riastrad 	void (*initialize)(struct uatp_softc *);
    546        1.1  riastrad 
    547        1.1  riastrad 	/* Device-specific finalization routine.  May be null.  May fail.  */
    548        1.1  riastrad 	int (*finalize)(struct uatp_softc *);
    549        1.1  riastrad 
    550        1.1  riastrad 	/* Tests whether this is a base sample.  Second argument is
    551        1.1  riastrad 	 * input_size bytes long.  */
    552        1.1  riastrad 	bool (*base_sample)(const struct uatp_softc *, const uint8_t *);
    553        1.1  riastrad 
    554        1.1  riastrad 	/* Reads a sensor sample from an input packet.  First argument
    555        1.1  riastrad 	 * is UATP_MAX_X_SENSORS bytes long; second, UATP_MAX_Y_SENSORS
    556        1.1  riastrad 	 * bytes; third, input_size bytes.  */
    557        1.1  riastrad 	void (*read_sample)(uint8_t *, uint8_t *, const uint8_t *);
    558        1.1  riastrad 
    559        1.1  riastrad 	/* Accumulates sensor state in sc->sc_acc.  */
    560        1.1  riastrad 	void (*accumulate)(struct uatp_softc *);
    561        1.1  riastrad 
    562        1.1  riastrad 	/* Called on spurious interrupts to reset.  May be null.  */
    563        1.1  riastrad 	void (*reset)(struct uatp_softc *);
    564        1.1  riastrad };
    565   1.10.4.2     skrll 
    566        1.1  riastrad /* Known device parameters */
    567        1.1  riastrad 
    568        1.1  riastrad static const struct uatp_parameters fountain_parameters = {
    569        1.1  riastrad 	.x_ratio	= 64,	.x_sensors = 16,	.x_sensors_17 = 26,
    570        1.1  riastrad 	.y_ratio	= 43,	.y_sensors = 16,
    571        1.1  riastrad 	.input_size	= 81,
    572        1.1  riastrad 	.initialize	= NULL,
    573        1.1  riastrad 	.finalize	= NULL,
    574        1.1  riastrad 	.base_sample	= base_sample_softc_flag,
    575        1.1  riastrad 	.read_sample	= read_sample_1,
    576        1.1  riastrad 	.accumulate	= accumulate_sample_1,
    577        1.1  riastrad 	.reset		= NULL,
    578        1.1  riastrad };
    579        1.1  riastrad 
    580        1.1  riastrad static const struct uatp_parameters geyser_1_parameters = {
    581        1.1  riastrad 	.x_ratio	= 64,	.x_sensors = 16,	.x_sensors_17 = 26,
    582        1.1  riastrad 	.y_ratio	= 43,	.y_sensors = 16,
    583        1.1  riastrad 	.input_size	= 81,
    584        1.1  riastrad 	.initialize	= NULL,
    585        1.1  riastrad 	.finalize	= NULL,
    586        1.1  riastrad 	.base_sample	= base_sample_softc_flag,
    587        1.1  riastrad 	.read_sample	= read_sample_1,
    588        1.1  riastrad 	.accumulate	= accumulate_sample_1,
    589        1.1  riastrad 	.reset		= NULL,
    590        1.1  riastrad };
    591        1.1  riastrad 
    592        1.1  riastrad static const struct uatp_parameters geyser_2_parameters = {
    593        1.1  riastrad 	.x_ratio	= 64,	.x_sensors = 15,	.x_sensors_17 = 20,
    594        1.1  riastrad 	.y_ratio	= 43,	.y_sensors = 9,
    595        1.1  riastrad 	.input_size	= 64,
    596        1.1  riastrad 	.initialize	= NULL,
    597        1.1  riastrad 	.finalize	= NULL,
    598        1.1  riastrad 	.base_sample	= base_sample_softc_flag,
    599        1.1  riastrad 	.read_sample	= read_sample_2,
    600        1.1  riastrad 	.accumulate	= accumulate_sample_1,
    601        1.1  riastrad 	.reset		= NULL,
    602        1.1  riastrad };
    603        1.1  riastrad 
    604        1.1  riastrad /*
    605        1.1  riastrad  * The Geyser 3 and Geyser 4 share parameters.  They also present
    606        1.1  riastrad  * generic USB HID mice on a different report id, so we have smaller
    607        1.1  riastrad  * packets by one byte (uhidev handles multiplexing report ids) and
    608        1.1  riastrad  * extra initialization work to switch the mode from generic USB HID
    609        1.1  riastrad  * mouse to Apple trackpad.
    610        1.1  riastrad  */
    611        1.1  riastrad 
    612        1.1  riastrad static const struct uatp_parameters geyser_3_4_parameters = {
    613        1.1  riastrad 	.x_ratio	= 64,	.x_sensors = 20, /* XXX */ .x_sensors_17 = 0,
    614        1.1  riastrad 	.y_ratio	= 64,	.y_sensors = 9,
    615        1.1  riastrad 	.input_size	= 63,	/* 64, minus one for the report id.  */
    616        1.1  riastrad 	.initialize	= geyser34_initialize,
    617        1.1  riastrad 	.finalize	= geyser34_finalize,
    618        1.1  riastrad 	.base_sample	= base_sample_input_flag,
    619        1.1  riastrad 	.read_sample	= read_sample_2,
    620        1.1  riastrad 	.accumulate	= accumulate_sample_2,
    621        1.1  riastrad 	.reset		= geyser34_deferred_reset,
    622        1.1  riastrad };
    623   1.10.4.2     skrll 
    624        1.1  riastrad /* Known device models */
    625        1.1  riastrad 
    626        1.1  riastrad #define APPLE_TRACKPAD(PRODUCT, DESCRIPTION, PARAMETERS)		\
    627        1.1  riastrad 	{								\
    628        1.1  riastrad 		.vendor = USB_VENDOR_APPLE,				\
    629        1.1  riastrad 		.product = (PRODUCT),					\
    630        1.1  riastrad 		.description = "Apple " DESCRIPTION " trackpad",	\
    631        1.1  riastrad 		.parameters = (& (PARAMETERS)),				\
    632        1.1  riastrad 	}
    633        1.1  riastrad 
    634        1.1  riastrad #define POWERBOOK_TRACKPAD(PRODUCT, PARAMETERS)				\
    635        1.1  riastrad 	APPLE_TRACKPAD(PRODUCT, "PowerBook/iBook", PARAMETERS)
    636        1.1  riastrad #define MACBOOK_TRACKPAD(PRODUCT, PARAMETERS)				\
    637        1.1  riastrad 	APPLE_TRACKPAD(PRODUCT, "MacBook/MacBook Pro", PARAMETERS)
    638        1.1  riastrad 
    639        1.1  riastrad static const struct uatp_descriptor uatp_descriptors[] =
    640        1.1  riastrad {
    641        1.1  riastrad 	POWERBOOK_TRACKPAD(0x020e, fountain_parameters),
    642        1.1  riastrad 	POWERBOOK_TRACKPAD(0x020f, fountain_parameters),
    643        1.1  riastrad 	POWERBOOK_TRACKPAD(0x030a, fountain_parameters),
    644        1.1  riastrad 
    645        1.1  riastrad 	POWERBOOK_TRACKPAD(0x030b, geyser_1_parameters),
    646        1.1  riastrad 
    647        1.1  riastrad 	POWERBOOK_TRACKPAD(0x0214, geyser_2_parameters),
    648        1.1  riastrad 	POWERBOOK_TRACKPAD(0x0215, geyser_2_parameters),
    649        1.1  riastrad 	POWERBOOK_TRACKPAD(0x0216, geyser_2_parameters),
    650        1.1  riastrad 
    651        1.1  riastrad 	MACBOOK_TRACKPAD(0x0217, geyser_3_4_parameters), /* 3 */
    652        1.1  riastrad 	MACBOOK_TRACKPAD(0x0218, geyser_3_4_parameters), /* 3 */
    653        1.1  riastrad 	MACBOOK_TRACKPAD(0x0219, geyser_3_4_parameters), /* 3 */
    654        1.1  riastrad 
    655        1.1  riastrad 	MACBOOK_TRACKPAD(0x021a, geyser_3_4_parameters), /* 4 */
    656        1.1  riastrad 	MACBOOK_TRACKPAD(0x021b, geyser_3_4_parameters), /* 4 */
    657        1.1  riastrad 	MACBOOK_TRACKPAD(0x021c, geyser_3_4_parameters), /* 4 */
    658        1.1  riastrad 
    659        1.1  riastrad 	MACBOOK_TRACKPAD(0x0229, geyser_3_4_parameters), /* 4 */
    660        1.1  riastrad 	MACBOOK_TRACKPAD(0x022a, geyser_3_4_parameters), /* 4 */
    661        1.1  riastrad 	MACBOOK_TRACKPAD(0x022b, geyser_3_4_parameters), /* 4 */
    662        1.1  riastrad };
    663        1.1  riastrad 
    664        1.1  riastrad #undef MACBOOK_TRACKPAD
    665        1.1  riastrad #undef POWERBOOK_TRACKPAD
    666        1.1  riastrad #undef APPLE_TRACKPAD
    667   1.10.4.2     skrll 
    668        1.1  riastrad /* Miscellaneous utilities */
    669        1.1  riastrad 
    670        1.1  riastrad static const struct uatp_descriptor *
    671        1.1  riastrad find_uatp_descriptor(const struct uhidev_attach_arg *uha)
    672        1.1  riastrad {
    673        1.1  riastrad 	unsigned int i;
    674        1.1  riastrad 
    675        1.1  riastrad 	for (i = 0; i < __arraycount(uatp_descriptors); i++)
    676   1.10.4.3     skrll 		if ((uha->uiaa->uiaa_vendor == uatp_descriptors[i].vendor) &&
    677   1.10.4.3     skrll 		    (uha->uiaa->uiaa_product == uatp_descriptors[i].product))
    678        1.1  riastrad 			return &uatp_descriptors[i];
    679        1.1  riastrad 
    680        1.1  riastrad 	return NULL;
    681        1.1  riastrad }
    682        1.1  riastrad 
    683        1.1  riastrad static device_t
    684        1.1  riastrad uatp_dev(const struct uatp_softc *sc)
    685        1.1  riastrad {
    686        1.1  riastrad 	return sc->sc_hdev.sc_dev;
    687        1.1  riastrad }
    688        1.1  riastrad 
    689        1.1  riastrad static uint8_t *
    690        1.1  riastrad uatp_x_sample(struct uatp_softc *sc)
    691        1.1  riastrad {
    692        1.1  riastrad 	return &sc->sc_sample[0];
    693        1.1  riastrad }
    694        1.1  riastrad 
    695        1.1  riastrad static uint8_t *
    696        1.1  riastrad uatp_y_sample(struct uatp_softc *sc)
    697        1.1  riastrad {
    698        1.1  riastrad 	return &sc->sc_sample[UATP_MAX_X_SENSORS];
    699        1.1  riastrad }
    700        1.1  riastrad 
    701        1.1  riastrad static int *
    702        1.1  riastrad uatp_x_acc(struct uatp_softc *sc)
    703        1.1  riastrad {
    704        1.1  riastrad 	return &sc->sc_acc[0];
    705        1.1  riastrad }
    706        1.1  riastrad 
    707        1.1  riastrad static int *
    708        1.1  riastrad uatp_y_acc(struct uatp_softc *sc)
    709        1.1  riastrad {
    710        1.1  riastrad 	return &sc->sc_acc[UATP_MAX_X_SENSORS];
    711        1.1  riastrad }
    712        1.1  riastrad 
    713        1.1  riastrad static void
    714        1.1  riastrad uatp_clear_position(struct uatp_softc *sc)
    715        1.1  riastrad {
    716        1.1  riastrad 	memset(sc->sc_acc, 0, sizeof(sc->sc_acc));
    717        1.1  riastrad 	sc->sc_motion_timer = 0;
    718        1.1  riastrad 	sc->sc_x_raw = sc->sc_x_smoothed = -1;
    719        1.1  riastrad 	sc->sc_y_raw = sc->sc_y_smoothed = -1;
    720        1.1  riastrad 	sc->sc_z_raw = sc->sc_z_smoothed = -1;
    721        1.1  riastrad 	sc->sc_w_raw = sc->sc_w_smoothed = -1;
    722        1.1  riastrad 	sc->sc_x_remainder = 0;
    723        1.1  riastrad 	sc->sc_y_remainder = 0;
    724        1.1  riastrad 	sc->sc_z_remainder = 0;
    725        1.1  riastrad 	sc->sc_w_remainder = 0;
    726        1.1  riastrad 	sc->sc_track_distance = 0;
    727        1.1  riastrad }
    728   1.10.4.2     skrll 
    729        1.1  riastrad static unsigned int
    730        1.1  riastrad uatp_x_sensors(const struct uatp_softc *sc)
    731        1.1  riastrad {
    732        1.1  riastrad 	if ((0 < sc->sc_knobs.x_sensors) &&
    733        1.1  riastrad 	    (sc->sc_knobs.x_sensors <= UATP_MAX_X_SENSORS))
    734        1.1  riastrad 		return sc->sc_knobs.x_sensors;
    735        1.1  riastrad 	else
    736        1.1  riastrad 		return sc->sc_parameters->x_sensors;
    737        1.1  riastrad }
    738        1.1  riastrad 
    739        1.1  riastrad static unsigned int
    740        1.1  riastrad uatp_y_sensors(const struct uatp_softc *sc)
    741        1.1  riastrad {
    742        1.1  riastrad 	if ((0 < sc->sc_knobs.y_sensors) &&
    743        1.1  riastrad 	    (sc->sc_knobs.y_sensors <= UATP_MAX_Y_SENSORS))
    744        1.1  riastrad 		return sc->sc_knobs.y_sensors;
    745        1.1  riastrad 	else
    746        1.1  riastrad 		return sc->sc_parameters->y_sensors;
    747        1.1  riastrad }
    748        1.1  riastrad 
    749        1.1  riastrad static unsigned int
    750        1.1  riastrad uatp_x_ratio(const struct uatp_softc *sc)
    751        1.1  riastrad {
    752        1.1  riastrad 	/* XXX Reject bogus values in sysctl.  */
    753        1.1  riastrad 	if ((0 < sc->sc_knobs.x_ratio) &&
    754        1.1  riastrad 	    (sc->sc_knobs.x_ratio <= UATP_MAX_X_RATIO))
    755        1.1  riastrad 		return sc->sc_knobs.x_ratio;
    756        1.1  riastrad 	else
    757        1.1  riastrad 		return sc->sc_parameters->x_ratio;
    758        1.1  riastrad }
    759        1.1  riastrad 
    760        1.1  riastrad static unsigned int
    761        1.1  riastrad uatp_y_ratio(const struct uatp_softc *sc)
    762        1.1  riastrad {
    763        1.1  riastrad 	/* XXX Reject bogus values in sysctl.  */
    764        1.1  riastrad 	if ((0 < sc->sc_knobs.y_ratio) &&
    765        1.1  riastrad 	    (sc->sc_knobs.y_ratio <= UATP_MAX_Y_RATIO))
    766        1.1  riastrad 		return sc->sc_knobs.y_ratio;
    767        1.1  riastrad 	else
    768        1.1  riastrad 		return sc->sc_parameters->y_ratio;
    769        1.1  riastrad }
    770   1.10.4.2     skrll 
    771        1.1  riastrad static unsigned int
    772        1.1  riastrad uatp_old_raw_weight(const struct uatp_softc *sc)
    773        1.1  riastrad {
    774        1.1  riastrad 	/* XXX Reject bogus values in sysctl.  */
    775        1.1  riastrad 	if (sc->sc_knobs.old_raw_weight <= UATP_MAX_WEIGHT)
    776        1.1  riastrad 		return sc->sc_knobs.old_raw_weight;
    777        1.1  riastrad 	else
    778        1.1  riastrad 		return 0;
    779        1.1  riastrad }
    780        1.1  riastrad 
    781        1.1  riastrad static unsigned int
    782        1.1  riastrad uatp_old_smoothed_weight(const struct uatp_softc *sc)
    783        1.1  riastrad {
    784        1.1  riastrad 	/* XXX Reject bogus values in sysctl.  */
    785        1.1  riastrad 	if (sc->sc_knobs.old_smoothed_weight <= UATP_MAX_WEIGHT)
    786        1.1  riastrad 		return sc->sc_knobs.old_smoothed_weight;
    787        1.1  riastrad 	else
    788        1.1  riastrad 		return 0;
    789        1.1  riastrad }
    790        1.1  riastrad 
    791        1.1  riastrad static unsigned int
    792        1.1  riastrad uatp_new_raw_weight(const struct uatp_softc *sc)
    793        1.1  riastrad {
    794        1.1  riastrad 	/* XXX Reject bogus values in sysctl.  */
    795        1.1  riastrad 	if ((0 < sc->sc_knobs.new_raw_weight) &&
    796        1.1  riastrad 	    (sc->sc_knobs.new_raw_weight <= UATP_MAX_WEIGHT))
    797        1.1  riastrad 		return sc->sc_knobs.new_raw_weight;
    798        1.1  riastrad 	else
    799        1.1  riastrad 		return 1;
    800        1.1  riastrad }
    801   1.10.4.2     skrll 
    802        1.1  riastrad static int
    803        1.1  riastrad scale_motion(const struct uatp_softc *sc, int delta, int *remainder,
    804        1.1  riastrad     const unsigned int *multiplier, const unsigned int *divisor)
    805        1.1  riastrad {
    806        1.1  riastrad 	int product;
    807        1.1  riastrad 
    808        1.1  riastrad 	/* XXX Limit the divisor?  */
    809        1.1  riastrad 	if (((*multiplier) == 0) ||
    810        1.1  riastrad 	    ((*multiplier) > UATP_MAX_MOTION_MULTIPLIER) ||
    811        1.1  riastrad 	    ((*divisor) == 0))
    812        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_ACCEL,
    813        1.1  riastrad 		    ("bad knobs; %d (+ %d) --> %d, rem 0\n",
    814        1.1  riastrad 			delta, *remainder, (delta + (*remainder))));
    815        1.1  riastrad 	else
    816        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_ACCEL,
    817        1.1  riastrad 		    ("scale %d (+ %d) by %u/%u --> %d, rem %d\n",
    818        1.1  riastrad 			delta, *remainder,
    819        1.1  riastrad 			(*multiplier), (*divisor),
    820        1.1  riastrad 			(((delta + (*remainder)) * ((int) (*multiplier)))
    821        1.1  riastrad 			    / ((int) (*divisor))),
    822        1.1  riastrad 			(((delta + (*remainder)) * ((int) (*multiplier)))
    823        1.1  riastrad 			    % ((int) (*divisor)))));
    824        1.1  riastrad 
    825        1.1  riastrad 	if (sc->sc_knobs.motion_remainder)
    826        1.1  riastrad 		delta += *remainder;
    827        1.1  riastrad 	*remainder = 0;
    828        1.1  riastrad 
    829        1.1  riastrad 	if (((*multiplier) == 0) ||
    830        1.1  riastrad 	    ((*multiplier) > UATP_MAX_MOTION_MULTIPLIER) ||
    831        1.1  riastrad 	    ((*divisor) == 0))
    832        1.1  riastrad 		return delta;
    833        1.1  riastrad 
    834        1.1  riastrad 	product = (delta * ((int) (*multiplier)));
    835        1.1  riastrad 	*remainder = (product % ((int) (*divisor)));
    836   1.10.4.9     skrll 	return product / ((int) (*divisor));
    837        1.1  riastrad }
    838        1.1  riastrad 
    839        1.1  riastrad static int
    840        1.1  riastrad uatp_scale_motion(const struct uatp_softc *sc, int delta, int *remainder)
    841        1.1  riastrad {
    842        1.1  riastrad 	return scale_motion(sc, delta, remainder,
    843        1.1  riastrad 	    &sc->sc_knobs.motion_multiplier,
    844        1.1  riastrad 	    &sc->sc_knobs.motion_divisor);
    845        1.1  riastrad }
    846        1.1  riastrad 
    847        1.1  riastrad static int
    848        1.1  riastrad uatp_scale_fast_motion(const struct uatp_softc *sc, int delta, int *remainder)
    849        1.1  riastrad {
    850        1.1  riastrad 	return scale_motion(sc, delta, remainder,
    851        1.1  riastrad 	    &sc->sc_knobs.fast_motion_multiplier,
    852        1.1  riastrad 	    &sc->sc_knobs.fast_motion_divisor);
    853        1.1  riastrad }
    854   1.10.4.2     skrll 
    855        1.1  riastrad /* Driver goop */
    856        1.1  riastrad 
    857        1.1  riastrad CFATTACH_DECL2_NEW(uatp, sizeof(struct uatp_softc), uatp_match, uatp_attach,
    858        1.1  riastrad     uatp_detach, uatp_activate, NULL, uatp_childdet);
    859        1.1  riastrad 
    860        1.1  riastrad static const struct wsmouse_accessops uatp_accessops = {
    861        1.1  riastrad 	.enable = uatp_enable,
    862        1.1  riastrad 	.disable = uatp_disable,
    863        1.1  riastrad 	.ioctl = uatp_ioctl,
    864        1.1  riastrad };
    865        1.1  riastrad 
    866        1.1  riastrad static int
    867        1.1  riastrad uatp_match(device_t parent, cfdata_t match, void *aux)
    868        1.1  riastrad {
    869        1.1  riastrad 	const struct uhidev_attach_arg *uha = aux;
    870        1.1  riastrad 	void *report_descriptor;
    871        1.1  riastrad 	int report_size, input_size;
    872        1.1  riastrad 	const struct uatp_descriptor *uatp_descriptor;
    873        1.1  riastrad 
    874        1.1  riastrad 	aprint_debug("%s: vendor 0x%04x, product 0x%04x\n", __func__,
    875   1.10.4.3     skrll 	    (unsigned int)uha->uiaa->uiaa_vendor,
    876   1.10.4.3     skrll 	    (unsigned int)uha->uiaa->uiaa_product);
    877        1.1  riastrad 	aprint_debug("%s: class 0x%04x, subclass 0x%04x, proto 0x%04x\n",
    878        1.1  riastrad 	    __func__,
    879   1.10.4.3     skrll 	    (unsigned int)uha->uiaa->uiaa_class,
    880   1.10.4.3     skrll 	    (unsigned int)uha->uiaa->uiaa_subclass,
    881   1.10.4.3     skrll 	    (unsigned int)uha->uiaa->uiaa_proto);
    882        1.1  riastrad 
    883        1.1  riastrad 	uhidev_get_report_desc(uha->parent, &report_descriptor, &report_size);
    884        1.1  riastrad 	input_size = hid_report_size(report_descriptor, report_size,
    885        1.1  riastrad 	    hid_input, uha->reportid);
    886        1.1  riastrad 	aprint_debug("%s: reportid %d, input size %d\n", __func__,
    887        1.1  riastrad 	    (int)uha->reportid, input_size);
    888        1.1  riastrad 
    889        1.1  riastrad 	/*
    890        1.1  riastrad 	 * Keyboards, trackpads, and eject buttons share common vendor
    891        1.1  riastrad 	 * and product ids, but not protocols: only the trackpad
    892        1.1  riastrad 	 * reports a mouse protocol.
    893        1.1  riastrad 	 */
    894   1.10.4.3     skrll 	if (uha->uiaa->uiaa_proto != UIPROTO_MOUSE)
    895        1.1  riastrad 		return UMATCH_NONE;
    896        1.1  riastrad 
    897        1.1  riastrad 	/* Check for a known vendor/product id.  */
    898        1.1  riastrad 	uatp_descriptor = find_uatp_descriptor(uha);
    899        1.1  riastrad 	if (uatp_descriptor == NULL) {
    900        1.1  riastrad 		aprint_debug("%s: unknown vendor/product id\n", __func__);
    901        1.1  riastrad 		return UMATCH_NONE;
    902        1.1  riastrad 	}
    903        1.1  riastrad 
    904        1.1  riastrad 	/* Check for the expected input size.  */
    905        1.1  riastrad 	if ((input_size < 0) ||
    906        1.1  riastrad 	    ((unsigned int)input_size !=
    907        1.1  riastrad 		uatp_descriptor->parameters->input_size)) {
    908        1.1  riastrad 		aprint_debug("%s: expected input size %u\n", __func__,
    909        1.1  riastrad 		    uatp_descriptor->parameters->input_size);
    910        1.1  riastrad 		return UMATCH_NONE;
    911        1.1  riastrad 	}
    912        1.1  riastrad 
    913        1.1  riastrad 	return UMATCH_VENDOR_PRODUCT_CONF_IFACE;
    914        1.1  riastrad }
    915   1.10.4.2     skrll 
    916        1.1  riastrad static void
    917        1.1  riastrad uatp_attach(device_t parent, device_t self, void *aux)
    918        1.1  riastrad {
    919        1.1  riastrad 	struct uatp_softc *sc = device_private(self);
    920        1.1  riastrad 	const struct uhidev_attach_arg *uha = aux;
    921        1.1  riastrad 	const struct uatp_descriptor *uatp_descriptor;
    922        1.1  riastrad 	void *report_descriptor;
    923        1.1  riastrad 	int report_size, input_size;
    924        1.1  riastrad 	struct wsmousedev_attach_args a;
    925        1.1  riastrad 
    926        1.1  riastrad 	/* Set up uhidev state.  (Why doesn't uhidev do most of this?)  */
    927        1.1  riastrad 	sc->sc_hdev.sc_dev = self;
    928        1.1  riastrad 	sc->sc_hdev.sc_intr = uatp_intr;
    929        1.1  riastrad 	sc->sc_hdev.sc_parent = uha->parent;
    930        1.1  riastrad 	sc->sc_hdev.sc_report_id = uha->reportid;
    931        1.1  riastrad 
    932        1.1  riastrad 	/* Identify ourselves to dmesg.  */
    933        1.1  riastrad 	uatp_descriptor = find_uatp_descriptor(uha);
    934        1.1  riastrad 	KASSERT(uatp_descriptor != NULL);
    935        1.1  riastrad 	aprint_normal(": %s\n", uatp_descriptor->description);
    936        1.1  riastrad 	aprint_naive(": %s\n", uatp_descriptor->description);
    937        1.1  riastrad 	aprint_verbose_dev(self,
    938        1.1  riastrad 	    "vendor 0x%04x, product 0x%04x, report id %d\n",
    939   1.10.4.3     skrll 	    (unsigned int)uha->uiaa->uiaa_vendor,
    940   1.10.4.3     skrll 	    (unsigned int)uha->uiaa->uiaa_product,
    941        1.1  riastrad 	    (int)uha->reportid);
    942        1.1  riastrad 
    943        1.1  riastrad 	uhidev_get_report_desc(uha->parent, &report_descriptor, &report_size);
    944        1.1  riastrad 	input_size = hid_report_size(report_descriptor, report_size, hid_input,
    945        1.1  riastrad 	    uha->reportid);
    946        1.1  riastrad 	KASSERT(0 < input_size);
    947        1.1  riastrad 	sc->sc_input_size = input_size;
    948        1.1  riastrad 
    949        1.1  riastrad 	/* Initialize model-specific parameters.  */
    950        1.1  riastrad 	sc->sc_parameters = uatp_descriptor->parameters;
    951        1.4  christos 	KASSERT((int)sc->sc_parameters->input_size == input_size);
    952        1.1  riastrad 	KASSERT(sc->sc_parameters->x_sensors <= UATP_MAX_X_SENSORS);
    953        1.1  riastrad 	KASSERT(sc->sc_parameters->x_ratio <= UATP_MAX_X_RATIO);
    954        1.1  riastrad 	KASSERT(sc->sc_parameters->y_sensors <= UATP_MAX_Y_SENSORS);
    955        1.1  riastrad 	KASSERT(sc->sc_parameters->y_ratio <= UATP_MAX_Y_RATIO);
    956        1.1  riastrad 	aprint_verbose_dev(self,
    957        1.1  riastrad 	    "%u x sensors, scaled by %u for %u points on screen\n",
    958        1.1  riastrad 	    sc->sc_parameters->x_sensors, sc->sc_parameters->x_ratio,
    959        1.1  riastrad 	    sc->sc_parameters->x_sensors * sc->sc_parameters->x_ratio);
    960        1.1  riastrad 	aprint_verbose_dev(self,
    961        1.1  riastrad 	    "%u y sensors, scaled by %u for %u points on screen\n",
    962        1.1  riastrad 	    sc->sc_parameters->y_sensors, sc->sc_parameters->y_ratio,
    963        1.1  riastrad 	    sc->sc_parameters->y_sensors * sc->sc_parameters->y_ratio);
    964        1.1  riastrad 	if (sc->sc_parameters->initialize)
    965        1.1  riastrad 		sc->sc_parameters->initialize(sc);
    966        1.1  riastrad 
    967        1.1  riastrad 	/* Register with pmf.  Nothing special for suspend/resume.  */
    968        1.1  riastrad 	if (!pmf_device_register(self, NULL, NULL))
    969        1.1  riastrad 		aprint_error_dev(self, "couldn't establish power handler\n");
    970        1.1  riastrad 
    971        1.1  riastrad 	/* Initialize knobs and create sysctl subtree to tweak them.  */
    972        1.1  riastrad 	sc->sc_knobs = default_knobs;
    973        1.1  riastrad 	uatp_setup_sysctl(sc);
    974        1.1  riastrad 
    975        1.1  riastrad 	/* Initialize tapping.  */
    976        1.1  riastrad 	tap_initialize(sc);
    977        1.1  riastrad 
    978        1.1  riastrad 	/* Attach wsmouse.  */
    979        1.1  riastrad 	a.accessops = &uatp_accessops;
    980        1.1  riastrad 	a.accesscookie = sc;
    981       1.10  riastrad 	sc->sc_wsmousedev = config_found_ia(self, "wsmousedev", &a,
    982       1.10  riastrad 	    wsmousedevprint);
    983        1.1  riastrad }
    984   1.10.4.2     skrll 
    985        1.1  riastrad /* Sysctl setup */
    986        1.1  riastrad 
    987        1.1  riastrad static void
    988        1.1  riastrad uatp_setup_sysctl(struct uatp_softc *sc)
    989        1.1  riastrad {
    990        1.1  riastrad 	int error;
    991        1.1  riastrad 
    992        1.1  riastrad 	error = sysctl_createv(&sc->sc_log, 0, NULL, &sc->sc_node, 0,
    993        1.1  riastrad 	    CTLTYPE_NODE, device_xname(uatp_dev(sc)),
    994        1.1  riastrad 	    SYSCTL_DESCR("uatp configuration knobs"),
    995        1.1  riastrad 	    NULL, 0, NULL, 0,
    996        1.1  riastrad 	    CTL_HW, CTL_CREATE, CTL_EOL);
    997        1.1  riastrad 	if (error != 0) {
    998        1.1  riastrad 		aprint_error_dev(uatp_dev(sc),
    999        1.1  riastrad 		    "unable to set up sysctl tree hw.%s: %d\n",
   1000        1.1  riastrad 		    device_xname(uatp_dev(sc)), error);
   1001        1.1  riastrad 		goto err;
   1002        1.1  riastrad 	}
   1003        1.1  riastrad 
   1004        1.1  riastrad #if UATP_DEBUG
   1005        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_debug_flags, "debug",
   1006        1.1  riastrad 		"uatp(4) debug flags"))
   1007        1.1  riastrad 		goto err;
   1008        1.1  riastrad #endif
   1009        1.1  riastrad 
   1010        1.1  riastrad 	/*
   1011        1.1  riastrad 	 * Button emulation.
   1012        1.1  riastrad 	 */
   1013        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.two_finger_buttons,
   1014        1.1  riastrad 		"two_finger_buttons",
   1015        1.1  riastrad 		"buttons to emulate with two fingers on trackpad"))
   1016        1.1  riastrad 		goto err;
   1017        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.three_finger_buttons,
   1018        1.1  riastrad 		"three_finger_buttons",
   1019        1.1  riastrad 		"buttons to emulate with three fingers on trackpad"))
   1020        1.1  riastrad 		goto err;
   1021        1.1  riastrad 
   1022        1.1  riastrad #if 0
   1023        1.1  riastrad 	/*
   1024        1.1  riastrad 	 * Edge scrolling.
   1025        1.1  riastrad 	 */
   1026        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.top_edge, "top_edge",
   1027        1.1  riastrad 		"width of top edge for edge scrolling"))
   1028        1.1  riastrad 		goto err;
   1029        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.bottom_edge,
   1030        1.1  riastrad 		"bottom_edge", "width of bottom edge for edge scrolling"))
   1031        1.1  riastrad 		goto err;
   1032        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.left_edge, "left_edge",
   1033        1.1  riastrad 		"width of left edge for edge scrolling"))
   1034        1.1  riastrad 		goto err;
   1035        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.right_edge, "right_edge",
   1036        1.1  riastrad 		"width of right edge for edge scrolling"))
   1037        1.1  riastrad 		goto err;
   1038        1.1  riastrad #endif
   1039   1.10.4.2     skrll 
   1040        1.1  riastrad 	/*
   1041        1.1  riastrad 	 * Multifinger tracking.
   1042        1.1  riastrad 	 */
   1043        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.multifinger_track,
   1044        1.1  riastrad 		"multifinger_track",
   1045        1.1  riastrad 		"0 to ignore multiple fingers, 1 to reset, 2 to scroll"))
   1046        1.1  riastrad 		goto err;
   1047        1.1  riastrad 
   1048        1.1  riastrad 	/*
   1049        1.1  riastrad 	 * Sensor parameters.
   1050        1.1  riastrad 	 */
   1051        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.x_sensors, "x_sensors",
   1052        1.1  riastrad 		"number of x sensors"))
   1053        1.1  riastrad 		goto err;
   1054        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.x_ratio, "x_ratio",
   1055        1.1  riastrad 		"screen width to trackpad width ratio"))
   1056        1.1  riastrad 		goto err;
   1057        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.y_sensors, "y_sensors",
   1058        1.1  riastrad 		"number of y sensors"))
   1059        1.1  riastrad 		goto err;
   1060        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.y_ratio, "y_ratio",
   1061        1.1  riastrad 		"screen height to trackpad height ratio"))
   1062        1.1  riastrad 		goto err;
   1063        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.sensor_threshold,
   1064        1.1  riastrad 		"sensor_threshold", "sensor threshold"))
   1065        1.1  riastrad 		goto err;
   1066        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.sensor_normalizer,
   1067        1.1  riastrad 		"sensor_normalizer", "sensor normalizer"))
   1068        1.1  riastrad 		goto err;
   1069        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.palm_width,
   1070        1.1  riastrad 		"palm_width", "lower bound on width/height of palm"))
   1071        1.1  riastrad 		goto err;
   1072        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.old_raw_weight,
   1073        1.1  riastrad 		"old_raw_weight", "weight of old raw position"))
   1074        1.1  riastrad 		goto err;
   1075        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.old_smoothed_weight,
   1076        1.1  riastrad 		"old_smoothed_weight", "weight of old smoothed position"))
   1077        1.1  riastrad 		goto err;
   1078        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.new_raw_weight,
   1079        1.1  riastrad 		"new_raw_weight", "weight of new raw position"))
   1080        1.1  riastrad 		goto err;
   1081        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.motion_remainder,
   1082        1.1  riastrad 		"motion_remainder", "remember motion division remainder"))
   1083        1.1  riastrad 		goto err;
   1084        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.motion_threshold,
   1085        1.1  riastrad 		"motion_threshold", "threshold before finger moves cursor"))
   1086        1.1  riastrad 		goto err;
   1087        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.motion_multiplier,
   1088        1.1  riastrad 		"motion_multiplier", "numerator of motion scale"))
   1089        1.1  riastrad 		goto err;
   1090        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.motion_divisor,
   1091        1.1  riastrad 		"motion_divisor", "divisor of motion scale"))
   1092        1.1  riastrad 		goto err;
   1093        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.fast_motion_threshold,
   1094        1.1  riastrad 		"fast_motion_threshold", "threshold before fast motion"))
   1095        1.1  riastrad 		goto err;
   1096        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.fast_motion_multiplier,
   1097        1.1  riastrad 		"fast_motion_multiplier", "numerator of fast motion scale"))
   1098        1.1  riastrad 		goto err;
   1099        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.fast_motion_divisor,
   1100        1.1  riastrad 		"fast_motion_divisor", "divisor of fast motion scale"))
   1101        1.1  riastrad 		goto err;
   1102        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.fast_per_direction,
   1103        1.1  riastrad 		"fast_per_direction", "don't frobnitz the veeblefitzer!"))
   1104        1.1  riastrad 		goto err;
   1105        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.motion_delay,
   1106        1.1  riastrad 		"motion_delay", "number of packets before motion kicks in"))
   1107        1.1  riastrad 		goto err;
   1108   1.10.4.2     skrll 
   1109        1.1  riastrad 	/*
   1110        1.1  riastrad 	 * Tapping.
   1111        1.1  riastrad 	 */
   1112        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.tap_limit_msec,
   1113        1.1  riastrad 		"tap_limit_msec", "milliseconds before a touch is not a tap"))
   1114        1.1  riastrad 		goto err;
   1115        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.double_tap_limit_msec,
   1116        1.1  riastrad 		"double_tap_limit_msec",
   1117        1.1  riastrad 		"milliseconds before a second tap keeps the button down"))
   1118        1.1  riastrad 		goto err;
   1119        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.one_finger_tap_buttons,
   1120        1.1  riastrad 		"one_finger_tap_buttons", "buttons for one-finger taps"))
   1121        1.1  riastrad 		goto err;
   1122        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.two_finger_tap_buttons,
   1123        1.1  riastrad 		"two_finger_tap_buttons", "buttons for two-finger taps"))
   1124        1.1  riastrad 		goto err;
   1125        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.three_finger_tap_buttons,
   1126        1.1  riastrad 		"three_finger_tap_buttons", "buttons for three-finger taps"))
   1127        1.1  riastrad 		goto err;
   1128        1.1  riastrad 	if (!uatp_setup_sysctl_knob(sc, &sc->sc_knobs.tap_track_distance_limit,
   1129        1.1  riastrad 		"tap_track_distance_limit",
   1130        1.1  riastrad 		"maximum distance^2 of tracking during tap"))
   1131        1.1  riastrad 		goto err;
   1132        1.1  riastrad 
   1133        1.1  riastrad 	return;
   1134        1.1  riastrad 
   1135        1.1  riastrad err:
   1136        1.1  riastrad 	sysctl_teardown(&sc->sc_log);
   1137        1.1  riastrad 	sc->sc_node = NULL;
   1138        1.1  riastrad }
   1139        1.1  riastrad 
   1140        1.1  riastrad static bool
   1141        1.1  riastrad uatp_setup_sysctl_knob(struct uatp_softc *sc, int *ptr, const char *name,
   1142        1.1  riastrad     const char *description)
   1143        1.1  riastrad {
   1144        1.1  riastrad 	int error;
   1145        1.1  riastrad 
   1146        1.1  riastrad 	error = sysctl_createv(&sc->sc_log, 0, NULL, NULL, CTLFLAG_READWRITE,
   1147        1.1  riastrad 	    CTLTYPE_INT, name, SYSCTL_DESCR(description),
   1148        1.1  riastrad 	    NULL, 0, ptr, 0,
   1149        1.1  riastrad 	    CTL_HW, sc->sc_node->sysctl_num, CTL_CREATE, CTL_EOL);
   1150        1.1  riastrad 	if (error != 0) {
   1151        1.1  riastrad 		aprint_error_dev(uatp_dev(sc),
   1152        1.1  riastrad 		    "unable to setup sysctl node hw.%s.%s: %d\n",
   1153        1.1  riastrad 		    device_xname(uatp_dev(sc)), name, error);
   1154        1.1  riastrad 		return false;
   1155        1.1  riastrad 	}
   1156        1.1  riastrad 
   1157        1.1  riastrad 	return true;
   1158        1.1  riastrad }
   1159   1.10.4.2     skrll 
   1160        1.1  riastrad /* More driver goop */
   1161        1.1  riastrad 
   1162        1.1  riastrad static void
   1163        1.1  riastrad uatp_childdet(device_t self, device_t child)
   1164        1.1  riastrad {
   1165        1.1  riastrad 	struct uatp_softc *sc = device_private(self);
   1166        1.1  riastrad 
   1167        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MISC, ("detaching child %s\n",
   1168        1.1  riastrad 	    device_xname(child)));
   1169        1.1  riastrad 
   1170        1.1  riastrad 	/* Our only child is the wsmouse device.  */
   1171        1.1  riastrad 	if (child == sc->sc_wsmousedev)
   1172        1.1  riastrad 		sc->sc_wsmousedev = NULL;
   1173        1.1  riastrad }
   1174        1.1  riastrad 
   1175        1.1  riastrad static int
   1176        1.1  riastrad uatp_detach(device_t self, int flags)
   1177        1.1  riastrad {
   1178        1.1  riastrad 	struct uatp_softc *sc = device_private(self);
   1179        1.1  riastrad 
   1180        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MISC, ("detaching with flags %d\n", flags));
   1181        1.1  riastrad 
   1182        1.1  riastrad         if (sc->sc_status & UATP_ENABLED) {
   1183        1.1  riastrad 		aprint_error_dev(uatp_dev(sc), "can't detach while enabled\n");
   1184        1.1  riastrad 		return EBUSY;
   1185        1.1  riastrad         }
   1186        1.1  riastrad 
   1187        1.1  riastrad 	if (sc->sc_parameters->finalize) {
   1188        1.1  riastrad 		int error = sc->sc_parameters->finalize(sc);
   1189        1.1  riastrad 		if (error != 0)
   1190        1.1  riastrad 			return error;
   1191        1.1  riastrad 	}
   1192        1.1  riastrad 
   1193        1.1  riastrad 	pmf_device_deregister(self);
   1194        1.1  riastrad 
   1195        1.1  riastrad 	sysctl_teardown(&sc->sc_log);
   1196        1.1  riastrad 	sc->sc_node = NULL;
   1197        1.1  riastrad 
   1198        1.1  riastrad 	tap_finalize(sc);
   1199        1.1  riastrad 
   1200        1.1  riastrad 	return config_detach_children(self, flags);
   1201        1.1  riastrad }
   1202        1.1  riastrad 
   1203        1.1  riastrad static int
   1204        1.1  riastrad uatp_activate(device_t self, enum devact act)
   1205        1.1  riastrad {
   1206        1.1  riastrad 	struct uatp_softc *sc = device_private(self);
   1207        1.1  riastrad 
   1208        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MISC, ("act %d\n", (int)act));
   1209        1.1  riastrad 
   1210        1.1  riastrad 	if (act != DVACT_DEACTIVATE)
   1211        1.1  riastrad 		return EOPNOTSUPP;
   1212        1.1  riastrad 
   1213        1.1  riastrad 	sc->sc_status |= UATP_DYING;
   1214        1.1  riastrad 
   1215        1.1  riastrad 	return 0;
   1216        1.1  riastrad }
   1217   1.10.4.2     skrll 
   1218        1.1  riastrad /* wsmouse routines */
   1219        1.1  riastrad 
   1220        1.1  riastrad static int
   1221        1.1  riastrad uatp_enable(void *v)
   1222        1.1  riastrad {
   1223        1.1  riastrad 	struct uatp_softc *sc = v;
   1224        1.1  riastrad 
   1225        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_WSMOUSE, ("enabling wsmouse\n"));
   1226        1.1  riastrad 
   1227        1.1  riastrad 	/* Refuse to enable if we've been deactivated.  */
   1228        1.1  riastrad 	if (sc->sc_status & UATP_DYING) {
   1229        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_WSMOUSE, ("busy dying\n"));
   1230        1.1  riastrad 		return EIO;
   1231        1.1  riastrad 	}
   1232        1.1  riastrad 
   1233        1.1  riastrad 	/* Refuse to enable if we already are enabled.  */
   1234        1.1  riastrad 	if (sc->sc_status & UATP_ENABLED) {
   1235        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_WSMOUSE, ("already enabled\n"));
   1236        1.1  riastrad 		return EBUSY;
   1237        1.1  riastrad 	}
   1238        1.1  riastrad 
   1239        1.1  riastrad 	sc->sc_status |= UATP_ENABLED;
   1240        1.1  riastrad 	sc->sc_status &=~ UATP_VALID;
   1241        1.1  riastrad 	sc->sc_input_index = 0;
   1242        1.1  riastrad 	tap_enable(sc);
   1243        1.1  riastrad 	uatp_clear_position(sc);
   1244        1.1  riastrad 
   1245        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MISC, ("uhidev_open(%p)\n", &sc->sc_hdev));
   1246        1.1  riastrad 	return uhidev_open(&sc->sc_hdev);
   1247        1.1  riastrad }
   1248        1.1  riastrad 
   1249        1.1  riastrad static void
   1250        1.1  riastrad uatp_disable(void *v)
   1251        1.1  riastrad {
   1252        1.1  riastrad 	struct uatp_softc *sc = v;
   1253        1.1  riastrad 
   1254        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_WSMOUSE, ("disabling wsmouse\n"));
   1255        1.1  riastrad 
   1256        1.1  riastrad 	if (!(sc->sc_status & UATP_ENABLED)) {
   1257        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_WSMOUSE, ("not enabled\n"));
   1258        1.1  riastrad 		return;
   1259        1.1  riastrad 	}
   1260        1.1  riastrad 
   1261        1.1  riastrad 	tap_disable(sc);
   1262        1.1  riastrad 	sc->sc_status &=~ UATP_ENABLED;
   1263        1.1  riastrad 
   1264        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MISC, ("uhidev_close(%p)\n", &sc->sc_hdev));
   1265        1.1  riastrad 	uhidev_close(&sc->sc_hdev);
   1266        1.1  riastrad }
   1267        1.1  riastrad 
   1268        1.1  riastrad static int
   1269        1.1  riastrad uatp_ioctl(void *v, unsigned long cmd, void *data, int flag, struct lwp *p)
   1270        1.1  riastrad {
   1271        1.1  riastrad 
   1272        1.3    martin 	DPRINTF((struct uatp_softc*)v, UATP_DEBUG_IOCTL,
   1273        1.1  riastrad 	    ("cmd %lx, data %p, flag %x, lwp %p\n", cmd, data, flag, p));
   1274        1.1  riastrad 
   1275        1.1  riastrad 	/* XXX Implement any relevant wsmouse(4) ioctls.  */
   1276        1.1  riastrad 	return EPASSTHROUGH;
   1277        1.1  riastrad }
   1278   1.10.4.2     skrll 
   1279        1.1  riastrad /*
   1280        1.1  riastrad  * The Geyser 3 and 4 models talk the generic USB HID mouse protocol by
   1281        1.1  riastrad  * default.  This mode switch makes them give raw sensor data instead
   1282        1.1  riastrad  * so that we can implement tapping, two-finger scrolling, &c.
   1283        1.1  riastrad  */
   1284        1.1  riastrad 
   1285        1.1  riastrad #define GEYSER34_RAW_MODE		0x04
   1286        1.1  riastrad #define GEYSER34_MODE_REPORT_ID		0
   1287        1.1  riastrad #define GEYSER34_MODE_INTERFACE		0
   1288        1.1  riastrad #define GEYSER34_MODE_PACKET_SIZE	8
   1289        1.1  riastrad 
   1290        1.1  riastrad static void
   1291        1.1  riastrad geyser34_enable_raw_mode(struct uatp_softc *sc)
   1292        1.1  riastrad {
   1293   1.10.4.5     skrll 	struct usbd_device *udev = sc->sc_hdev.sc_parent->sc_udev;
   1294        1.1  riastrad 	usb_device_request_t req;
   1295        1.1  riastrad 	usbd_status status;
   1296        1.1  riastrad 	uint8_t report[GEYSER34_MODE_PACKET_SIZE];
   1297        1.1  riastrad 
   1298        1.1  riastrad 	req.bmRequestType = UT_READ_CLASS_INTERFACE;
   1299        1.1  riastrad 	req.bRequest = UR_GET_REPORT;
   1300        1.1  riastrad 	USETW2(req.wValue, UHID_FEATURE_REPORT, GEYSER34_MODE_REPORT_ID);
   1301        1.1  riastrad 	USETW(req.wIndex, GEYSER34_MODE_INTERFACE);
   1302        1.1  riastrad 	USETW(req.wLength, GEYSER34_MODE_PACKET_SIZE);
   1303        1.1  riastrad 
   1304        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_RESET, ("get feature report\n"));
   1305        1.1  riastrad 	status = usbd_do_request(udev, &req, report);
   1306        1.1  riastrad 	if (status != USBD_NORMAL_COMPLETION) {
   1307        1.1  riastrad 		aprint_error_dev(uatp_dev(sc),
   1308        1.1  riastrad 		    "error reading feature report: %s\n", usbd_errstr(status));
   1309        1.1  riastrad 		return;
   1310        1.1  riastrad 	}
   1311   1.10.4.2     skrll 
   1312        1.1  riastrad #if UATP_DEBUG
   1313        1.1  riastrad 	if (sc->sc_debug_flags & UATP_DEBUG_RESET) {
   1314        1.1  riastrad 		unsigned int i;
   1315        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_RESET, ("old feature report:"));
   1316        1.1  riastrad 		for (i = 0; i < GEYSER34_MODE_PACKET_SIZE; i++)
   1317        1.1  riastrad 			printf(" %02x", (unsigned int)report[i]);
   1318        1.1  riastrad 		printf("\n");
   1319        1.1  riastrad 		/* Doing this twice is harmless here and lets this be
   1320        1.1  riastrad 		 * one ifdef.  */
   1321        1.1  riastrad 		report[0] = GEYSER34_RAW_MODE;
   1322        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_RESET, ("new feature report:"));
   1323        1.1  riastrad 		for (i = 0; i < GEYSER34_MODE_PACKET_SIZE; i++)
   1324        1.1  riastrad 			printf(" %02x", (unsigned int)report[i]);
   1325        1.1  riastrad 		printf("\n");
   1326        1.1  riastrad 	}
   1327        1.1  riastrad #endif
   1328        1.1  riastrad 
   1329        1.1  riastrad 	report[0] = GEYSER34_RAW_MODE;
   1330        1.1  riastrad 
   1331        1.1  riastrad 	req.bmRequestType = UT_WRITE_CLASS_INTERFACE;
   1332        1.1  riastrad 	req.bRequest = UR_SET_REPORT;
   1333        1.1  riastrad 	USETW2(req.wValue, UHID_FEATURE_REPORT, GEYSER34_MODE_REPORT_ID);
   1334        1.1  riastrad 	USETW(req.wIndex, GEYSER34_MODE_INTERFACE);
   1335        1.1  riastrad 	USETW(req.wLength, GEYSER34_MODE_PACKET_SIZE);
   1336        1.1  riastrad 
   1337        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_RESET, ("set feature report\n"));
   1338        1.1  riastrad 	status = usbd_do_request(udev, &req, report);
   1339        1.1  riastrad 	if (status != USBD_NORMAL_COMPLETION) {
   1340        1.1  riastrad 		aprint_error_dev(uatp_dev(sc),
   1341        1.1  riastrad 		    "error writing feature report: %s\n", usbd_errstr(status));
   1342        1.1  riastrad 		return;
   1343        1.1  riastrad 	}
   1344        1.1  riastrad }
   1345   1.10.4.2     skrll 
   1346        1.1  riastrad /*
   1347        1.1  riastrad  * The Geyser 3 and 4 need to be reset periodically after we detect a
   1348        1.8  riastrad  * continual flow of spurious interrupts.  We use a USB task for this.
   1349        1.1  riastrad  */
   1350        1.1  riastrad 
   1351        1.1  riastrad static void
   1352        1.1  riastrad geyser34_initialize(struct uatp_softc *sc)
   1353        1.1  riastrad {
   1354        1.8  riastrad 
   1355        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MISC, ("initializing\n"));
   1356        1.1  riastrad 	geyser34_enable_raw_mode(sc);
   1357   1.10.4.4     skrll 	usb_init_task(&sc->sc_reset_task, &geyser34_reset_task, sc, 0);
   1358        1.1  riastrad }
   1359        1.1  riastrad 
   1360        1.1  riastrad static int
   1361        1.1  riastrad geyser34_finalize(struct uatp_softc *sc)
   1362        1.1  riastrad {
   1363        1.8  riastrad 
   1364        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MISC, ("finalizing\n"));
   1365        1.8  riastrad 	usb_rem_task(sc->sc_hdev.sc_parent->sc_udev, &sc->sc_reset_task);
   1366        1.1  riastrad 
   1367        1.1  riastrad 	return 0;
   1368        1.1  riastrad }
   1369        1.1  riastrad 
   1370        1.1  riastrad static void
   1371        1.1  riastrad geyser34_deferred_reset(struct uatp_softc *sc)
   1372        1.1  riastrad {
   1373        1.8  riastrad 
   1374        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_RESET, ("deferring reset\n"));
   1375        1.8  riastrad 	usb_add_task(sc->sc_hdev.sc_parent->sc_udev, &sc->sc_reset_task,
   1376        1.8  riastrad 	    USB_TASKQ_DRIVER);
   1377        1.1  riastrad }
   1378        1.1  riastrad 
   1379        1.1  riastrad static void
   1380        1.8  riastrad geyser34_reset_task(void *arg)
   1381        1.1  riastrad {
   1382        1.1  riastrad 	struct uatp_softc *sc = arg;
   1383        1.1  riastrad 
   1384        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_RESET, ("resetting\n"));
   1385        1.1  riastrad 
   1386        1.1  riastrad 	/* Reset by putting it into raw mode.  Not sure why.  */
   1387        1.1  riastrad 	geyser34_enable_raw_mode(sc);
   1388        1.1  riastrad }
   1389   1.10.4.2     skrll 
   1390        1.1  riastrad /* Interrupt handler */
   1391        1.1  riastrad 
   1392        1.1  riastrad static void
   1393        1.1  riastrad uatp_intr(struct uhidev *addr, void *ibuf, unsigned int len)
   1394        1.1  riastrad {
   1395        1.1  riastrad 	struct uatp_softc *sc = (struct uatp_softc *)addr;
   1396        1.1  riastrad 	uint8_t *input;
   1397        1.1  riastrad 	int dx, dy, dz, dw;
   1398        1.1  riastrad 	uint32_t buttons;
   1399        1.1  riastrad 
   1400        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_INTR, ("softc %p, ibuf %p, len %u\n",
   1401        1.1  riastrad 	    addr, ibuf, len));
   1402        1.1  riastrad 
   1403        1.1  riastrad 	/*
   1404        1.1  riastrad 	 * Some devices break packets up into chunks, so we accumulate
   1405        1.1  riastrad 	 * input up to the expected packet length, or if it would
   1406        1.1  riastrad 	 * overflow, discard the whole packet and start over.
   1407        1.1  riastrad 	 */
   1408        1.1  riastrad 	if (sc->sc_input_size < len) {
   1409        1.1  riastrad 		aprint_error_dev(uatp_dev(sc),
   1410        1.1  riastrad 		    "discarding %u-byte input packet\n", len);
   1411        1.1  riastrad 		sc->sc_input_index = 0;
   1412        1.1  riastrad 		return;
   1413        1.1  riastrad 	} else if (sc->sc_input_size < (sc->sc_input_index + len)) {
   1414        1.1  riastrad 		aprint_error_dev(uatp_dev(sc), "discarding %u-byte input\n",
   1415        1.1  riastrad 		    (sc->sc_input_index + len));
   1416        1.1  riastrad 		sc->sc_input_index = 0;
   1417        1.1  riastrad 		return;
   1418        1.1  riastrad 	}
   1419        1.1  riastrad 
   1420        1.1  riastrad #if UATP_DEBUG
   1421        1.1  riastrad 	if (sc->sc_debug_flags & UATP_DEBUG_INTR) {
   1422        1.1  riastrad 		unsigned int i;
   1423        1.1  riastrad 		uint8_t *bytes = ibuf;
   1424        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_INTR, ("raw"));
   1425        1.1  riastrad 		for (i = 0; i < len; i++)
   1426        1.1  riastrad 			printf(" %02x", (unsigned int)bytes[i]);
   1427        1.1  riastrad 		printf("\n");
   1428        1.1  riastrad 	}
   1429        1.1  riastrad #endif
   1430        1.1  riastrad 
   1431        1.1  riastrad 	memcpy(&sc->sc_input[sc->sc_input_index], ibuf, len);
   1432        1.1  riastrad 	sc->sc_input_index += len;
   1433        1.1  riastrad 	if (sc->sc_input_index != sc->sc_input_size) {
   1434        1.1  riastrad 		/* Wait until packet is complete.  */
   1435        1.1  riastrad 		aprint_verbose_dev(uatp_dev(sc), "partial packet: %u bytes\n",
   1436        1.1  riastrad 		    len);
   1437        1.1  riastrad 		return;
   1438        1.1  riastrad 	}
   1439        1.1  riastrad 
   1440        1.1  riastrad 	/* Clear the buffer and process the now complete packet.  */
   1441        1.1  riastrad 	sc->sc_input_index = 0;
   1442        1.1  riastrad 	input = sc->sc_input;
   1443        1.1  riastrad 
   1444        1.1  riastrad 	/* The last byte's first bit is set iff the button is pressed.
   1445        1.1  riastrad 	 * XXX Left button should have a name.  */
   1446        1.1  riastrad 	buttons = ((input[sc->sc_input_size - 1] & UATP_STATUS_BUTTON)
   1447        1.1  riastrad 	    ? 1 : 0);
   1448        1.1  riastrad 
   1449        1.1  riastrad 	/* Read the sample.  */
   1450        1.1  riastrad 	memset(uatp_x_sample(sc), 0, UATP_MAX_X_SENSORS);
   1451        1.1  riastrad 	memset(uatp_y_sample(sc), 0, UATP_MAX_Y_SENSORS);
   1452        1.1  riastrad 	sc->sc_parameters->read_sample(uatp_x_sample(sc), uatp_y_sample(sc),
   1453        1.1  riastrad 	    input);
   1454   1.10.4.2     skrll 
   1455        1.1  riastrad #if UATP_DEBUG
   1456        1.1  riastrad 	if (sc->sc_debug_flags & UATP_DEBUG_INTR) {
   1457        1.1  riastrad 		unsigned int i;
   1458        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_INTR, ("x sensors"));
   1459        1.1  riastrad 		for (i = 0; i < uatp_x_sensors(sc); i++)
   1460        1.1  riastrad 			printf(" %02x", (unsigned int)uatp_x_sample(sc)[i]);
   1461        1.1  riastrad 		printf("\n");
   1462        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_INTR, ("y sensors"));
   1463        1.1  riastrad 		for (i = 0; i < uatp_y_sensors(sc); i++)
   1464        1.1  riastrad 			printf(" %02x", (unsigned int)uatp_y_sample(sc)[i]);
   1465        1.1  riastrad 		printf("\n");
   1466        1.1  riastrad 	} else if ((sc->sc_debug_flags & UATP_DEBUG_STATUS) &&
   1467        1.1  riastrad 		(input[sc->sc_input_size - 1] &~
   1468        1.1  riastrad 		    (UATP_STATUS_BUTTON | UATP_STATUS_BASE |
   1469        1.1  riastrad 			UATP_STATUS_POST_RESET)))
   1470        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_STATUS, ("status byte: %02x\n",
   1471        1.1  riastrad 		    input[sc->sc_input_size - 1]));
   1472        1.1  riastrad #endif
   1473        1.1  riastrad 
   1474        1.1  riastrad 	/*
   1475        1.1  riastrad 	 * If this is a base sample, initialize the state to interpret
   1476        1.1  riastrad 	 * subsequent samples relative to it, and stop here.
   1477        1.1  riastrad 	 */
   1478        1.1  riastrad 	if (sc->sc_parameters->base_sample(sc, input)) {
   1479        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_PARSE,
   1480        1.1  riastrad 		    ("base sample, buttons %"PRIx32"\n", buttons));
   1481        1.1  riastrad 		/* XXX Should the valid bit ever be reset?  */
   1482        1.1  riastrad 		sc->sc_status |= UATP_VALID;
   1483        1.1  riastrad 		uatp_clear_position(sc);
   1484        1.1  riastrad 		memcpy(sc->sc_base, sc->sc_sample, sizeof(sc->sc_base));
   1485        1.1  riastrad 		/* XXX Perform 17" size detection like Linux?  */
   1486        1.1  riastrad 		return;
   1487        1.1  riastrad 	}
   1488        1.1  riastrad 
   1489        1.1  riastrad 	/* If not, accumulate the change in the sensors.  */
   1490        1.1  riastrad 	sc->sc_parameters->accumulate(sc);
   1491        1.1  riastrad 
   1492        1.1  riastrad #if UATP_DEBUG
   1493        1.1  riastrad 	if (sc->sc_debug_flags & UATP_DEBUG_ACCUMULATE) {
   1494        1.1  riastrad 		unsigned int i;
   1495        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_ACCUMULATE, ("accumulated x state:"));
   1496        1.1  riastrad 		for (i = 0; i < uatp_x_sensors(sc); i++)
   1497        1.1  riastrad 			printf(" %02x", (unsigned int)uatp_x_acc(sc)[i]);
   1498        1.1  riastrad 		printf("\n");
   1499        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_ACCUMULATE, ("accumulated y state:"));
   1500        1.1  riastrad 		for (i = 0; i < uatp_y_sensors(sc); i++)
   1501        1.1  riastrad 			printf(" %02x", (unsigned int)uatp_y_acc(sc)[i]);
   1502        1.1  riastrad 		printf("\n");
   1503        1.1  riastrad 	}
   1504        1.1  riastrad #endif
   1505        1.1  riastrad 
   1506        1.1  riastrad 	/* Compute the change in coordinates and buttons.  */
   1507        1.1  riastrad 	dx = dy = dz = dw = 0;
   1508        1.1  riastrad 	if ((!interpret_input(sc, &dx, &dy, &dz, &dw, &buttons)) &&
   1509        1.1  riastrad 	    /* If there's no input because we're releasing a button,
   1510        1.1  riastrad 	     * then it's not spurious.  XXX Mutex?  */
   1511        1.1  riastrad 	    (sc->sc_buttons == 0)) {
   1512        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_SPURINTR, ("spurious interrupt\n"));
   1513        1.1  riastrad 		if (sc->sc_parameters->reset)
   1514        1.1  riastrad 			sc->sc_parameters->reset(sc);
   1515        1.1  riastrad 		return;
   1516        1.1  riastrad 	}
   1517        1.1  riastrad 
   1518        1.1  riastrad 	/* Report to wsmouse.  */
   1519        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_INTR,
   1520        1.1  riastrad 	    ("buttons %"PRIx32", dx %d, dy %d, dz %d, dw %d\n",
   1521        1.1  riastrad 		buttons, dx, dy, dz, dw));
   1522        1.1  riastrad 	mutex_enter(&sc->sc_tap_mutex);
   1523        1.1  riastrad 	uatp_input(sc, buttons, dx, dy, dz, dw);
   1524        1.1  riastrad 	mutex_exit(&sc->sc_tap_mutex);
   1525        1.1  riastrad }
   1526   1.10.4.2     skrll 
   1527        1.1  riastrad /*
   1528        1.1  riastrad  * Different ways to discern the base sample initializing the state.
   1529        1.1  riastrad  * `base_sample_softc_flag' uses a state flag stored in the softc;
   1530        1.1  riastrad  * `base_sample_input_flag' checks a flag at the end of the input
   1531        1.1  riastrad  * packet.
   1532        1.1  riastrad  */
   1533        1.1  riastrad 
   1534        1.1  riastrad static bool
   1535        1.1  riastrad base_sample_softc_flag(const struct uatp_softc *sc, const uint8_t *input)
   1536        1.1  riastrad {
   1537        1.1  riastrad 	return !(sc->sc_status & UATP_VALID);
   1538        1.1  riastrad }
   1539        1.1  riastrad 
   1540        1.1  riastrad static bool
   1541        1.1  riastrad base_sample_input_flag(const struct uatp_softc *sc, const uint8_t *input)
   1542        1.1  riastrad {
   1543        1.1  riastrad 	/* XXX Should we also check the valid flag?  */
   1544        1.1  riastrad 	return !!(input[sc->sc_input_size - 1] & UATP_STATUS_BASE);
   1545        1.1  riastrad }
   1546        1.1  riastrad 
   1547        1.1  riastrad /*
   1548        1.1  riastrad  * Pick apart the horizontal sensors from the vertical sensors.
   1549        1.1  riastrad  * Different models interleave them in different orders.
   1550        1.1  riastrad  */
   1551        1.1  riastrad 
   1552        1.1  riastrad static void
   1553        1.1  riastrad read_sample_1(uint8_t *x, uint8_t *y, const uint8_t *input)
   1554        1.1  riastrad {
   1555        1.1  riastrad 	unsigned int i;
   1556        1.1  riastrad 
   1557        1.1  riastrad 	for (i = 0; i < 8; i++) {
   1558        1.1  riastrad 		x[i] = input[5 * i + 2];
   1559        1.1  riastrad 		x[i + 8] = input[5 * i + 4];
   1560        1.1  riastrad 		x[i + 16] = input[5 * i + 42];
   1561        1.1  riastrad 		if (i < 2)
   1562        1.1  riastrad 			x[i + 24] = input[5 * i + 44];
   1563        1.1  riastrad 
   1564        1.1  riastrad 		y[i] = input[5 * i + 1];
   1565        1.1  riastrad 		y[i + 8] = input[5 * i + 3];
   1566        1.1  riastrad 	}
   1567        1.1  riastrad }
   1568        1.1  riastrad 
   1569        1.1  riastrad static void
   1570        1.1  riastrad read_sample_2(uint8_t *x, uint8_t *y, const uint8_t *input)
   1571        1.1  riastrad {
   1572        1.1  riastrad 	unsigned int i, j;
   1573        1.1  riastrad 
   1574        1.1  riastrad 	for (i = 0, j = 19; i < 20; i += 2, j += 3) {
   1575        1.1  riastrad 		x[i] = input[j];
   1576        1.1  riastrad 		x[i + 1] = input[j + 1];
   1577        1.1  riastrad 	}
   1578        1.1  riastrad 
   1579        1.1  riastrad 	for (i = 0, j = 1; i < 9; i += 2, j += 3) {
   1580        1.1  riastrad 		y[i] = input[j];
   1581        1.1  riastrad 		y[i + 1] = input[j + 1];
   1582        1.1  riastrad 	}
   1583        1.1  riastrad }
   1584   1.10.4.2     skrll 
   1585        1.1  riastrad static void
   1586        1.1  riastrad accumulate_sample_1(struct uatp_softc *sc)
   1587        1.1  riastrad {
   1588        1.1  riastrad 	unsigned int i;
   1589        1.1  riastrad 
   1590        1.1  riastrad 	for (i = 0; i < UATP_SENSORS; i++) {
   1591        1.1  riastrad 		sc->sc_acc[i] += (int8_t)(sc->sc_sample[i] - sc->sc_base[i]);
   1592        1.1  riastrad 		if (sc->sc_acc[i] < 0) {
   1593        1.1  riastrad 			sc->sc_acc[i] = 0;
   1594        1.1  riastrad 		} else if (UATP_MAX_ACC < sc->sc_acc[i]) {
   1595        1.1  riastrad 			DPRINTF(sc, UATP_DEBUG_ACCUMULATE,
   1596        1.1  riastrad 			    ("overflow %d\n", sc->sc_acc[i]));
   1597        1.1  riastrad 			sc->sc_acc[i] = UATP_MAX_ACC;
   1598        1.1  riastrad 		}
   1599        1.1  riastrad 	}
   1600        1.1  riastrad 
   1601        1.1  riastrad 	memcpy(sc->sc_base, sc->sc_sample, sizeof(sc->sc_base));
   1602        1.1  riastrad }
   1603        1.1  riastrad 
   1604        1.1  riastrad static void
   1605        1.1  riastrad accumulate_sample_2(struct uatp_softc *sc)
   1606        1.1  riastrad {
   1607        1.1  riastrad 	unsigned int i;
   1608        1.1  riastrad 
   1609        1.1  riastrad 	for (i = 0; i < UATP_SENSORS; i++) {
   1610        1.1  riastrad 		sc->sc_acc[i] = (int8_t)(sc->sc_sample[i] - sc->sc_base[i]);
   1611        1.1  riastrad 		if (sc->sc_acc[i] < -0x80) {
   1612        1.1  riastrad 			DPRINTF(sc, UATP_DEBUG_ACCUMULATE,
   1613        1.1  riastrad 			    ("underflow %u - %u = %d\n",
   1614        1.1  riastrad 				(unsigned int)sc->sc_sample[i],
   1615        1.1  riastrad 				(unsigned int)sc->sc_base[i],
   1616        1.1  riastrad 				sc->sc_acc[i]));
   1617        1.1  riastrad 			sc->sc_acc[i] += 0x100;
   1618        1.1  riastrad 		}
   1619        1.1  riastrad 		if (0x7f < sc->sc_acc[i]) {
   1620        1.1  riastrad 			DPRINTF(sc, UATP_DEBUG_ACCUMULATE,
   1621        1.1  riastrad 			    ("overflow %u - %u = %d\n",
   1622        1.1  riastrad 				(unsigned int)sc->sc_sample[i],
   1623        1.1  riastrad 				(unsigned int)sc->sc_base[i],
   1624        1.1  riastrad 				sc->sc_acc[i]));
   1625        1.1  riastrad 			sc->sc_acc[i] -= 0x100;
   1626        1.1  riastrad 		}
   1627        1.1  riastrad 		if (sc->sc_acc[i] < 0)
   1628        1.1  riastrad 			sc->sc_acc[i] = 0;
   1629        1.1  riastrad 	}
   1630        1.1  riastrad }
   1631   1.10.4.2     skrll 
   1632        1.1  riastrad /*
   1633        1.1  riastrad  * Report input to wsmouse, if there is anything interesting to report.
   1634        1.1  riastrad  * We must take into consideration the current tap-and-drag button
   1635        1.1  riastrad  * state.
   1636        1.1  riastrad  */
   1637        1.1  riastrad 
   1638        1.1  riastrad static void
   1639        1.1  riastrad uatp_input(struct uatp_softc *sc, uint32_t buttons,
   1640        1.1  riastrad     int dx, int dy, int dz, int dw)
   1641        1.1  riastrad {
   1642        1.1  riastrad 	uint32_t all_buttons;
   1643        1.1  riastrad 
   1644        1.1  riastrad 	KASSERT(mutex_owned(&sc->sc_tap_mutex));
   1645        1.1  riastrad 	all_buttons = buttons | uatp_tapped_buttons(sc);
   1646        1.1  riastrad 
   1647        1.1  riastrad 	if ((sc->sc_wsmousedev != NULL) &&
   1648        1.1  riastrad 	    ((dx != 0) || (dy != 0) || (dz != 0) || (dw != 0) ||
   1649        1.1  riastrad 		(all_buttons != sc->sc_all_buttons))) {
   1650        1.1  riastrad 		int s = spltty();
   1651        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_WSMOUSE, ("wsmouse input:"
   1652        1.1  riastrad 		    " buttons %"PRIx32", dx %d, dy %d, dz %d, dw %d\n",
   1653        1.1  riastrad 		    all_buttons, dx, -dy, dz, -dw));
   1654        1.1  riastrad 		wsmouse_input(sc->sc_wsmousedev, all_buttons, dx, -dy, dz, -dw,
   1655        1.1  riastrad 		    WSMOUSE_INPUT_DELTA);
   1656        1.1  riastrad 		splx(s);
   1657        1.1  riastrad 	}
   1658        1.1  riastrad 	sc->sc_buttons = buttons;
   1659        1.1  riastrad 	sc->sc_all_buttons = all_buttons;
   1660        1.1  riastrad }
   1661        1.1  riastrad 
   1662        1.1  riastrad /*
   1663        1.1  riastrad  * Interpret the current tap state to decide whether the tap buttons
   1664        1.1  riastrad  * are currently pressed.
   1665        1.1  riastrad  */
   1666        1.1  riastrad 
   1667        1.1  riastrad static uint32_t
   1668        1.1  riastrad uatp_tapped_buttons(struct uatp_softc *sc)
   1669        1.1  riastrad {
   1670        1.1  riastrad 	KASSERT(mutex_owned(&sc->sc_tap_mutex));
   1671        1.1  riastrad 	switch (sc->sc_tap_state) {
   1672        1.1  riastrad 	case TAP_STATE_INITIAL:
   1673        1.1  riastrad 	case TAP_STATE_TAPPING:
   1674        1.1  riastrad 		return 0;
   1675        1.1  riastrad 
   1676        1.1  riastrad 	case TAP_STATE_TAPPED:
   1677        1.1  riastrad 	case TAP_STATE_DOUBLE_TAPPING:
   1678        1.1  riastrad 	case TAP_STATE_DRAGGING_DOWN:
   1679        1.1  riastrad 	case TAP_STATE_DRAGGING_UP:
   1680        1.1  riastrad 	case TAP_STATE_TAPPING_IN_DRAG:
   1681        1.1  riastrad 		CHECK((0 < sc->sc_tapped_fingers), return 0);
   1682        1.1  riastrad 		switch (sc->sc_tapped_fingers) {
   1683        1.1  riastrad 		case 1: return sc->sc_knobs.one_finger_tap_buttons;
   1684        1.1  riastrad 		case 2: return sc->sc_knobs.two_finger_tap_buttons;
   1685        1.1  riastrad 		case 3:
   1686        1.1  riastrad 		default: return sc->sc_knobs.three_finger_tap_buttons;
   1687        1.1  riastrad 		}
   1688        1.1  riastrad 
   1689        1.1  riastrad 	default:
   1690        1.1  riastrad 		aprint_error_dev(uatp_dev(sc), "%s: invalid tap state: %d\n",
   1691        1.1  riastrad 		    __func__, sc->sc_tap_state);
   1692        1.1  riastrad 		return 0;
   1693        1.1  riastrad 	}
   1694        1.1  riastrad }
   1695   1.10.4.2     skrll 
   1696        1.1  riastrad /*
   1697        1.1  riastrad  * Interpret the current input state to find a difference in all the
   1698        1.1  riastrad  * relevant coordinates and buttons to pass on to wsmouse, and update
   1699        1.1  riastrad  * any internal driver state necessary to interpret subsequent input
   1700        1.1  riastrad  * relative to this one.
   1701        1.1  riastrad  */
   1702        1.1  riastrad 
   1703        1.1  riastrad static bool
   1704        1.1  riastrad interpret_input(struct uatp_softc *sc, int *dx, int *dy, int *dz, int *dw,
   1705        1.1  riastrad     uint32_t *buttons)
   1706        1.1  riastrad {
   1707        1.1  riastrad 	unsigned int x_pressure, x_raw, x_fingers;
   1708        1.1  riastrad 	unsigned int y_pressure, y_raw, y_fingers;
   1709        1.1  riastrad 	unsigned int fingers;
   1710        1.1  riastrad 
   1711        1.1  riastrad 	x_pressure = interpret_dimension(sc, uatp_x_acc(sc),
   1712        1.1  riastrad 	    uatp_x_sensors(sc), uatp_x_ratio(sc), &x_raw, &x_fingers);
   1713        1.1  riastrad 	y_pressure = interpret_dimension(sc, uatp_y_acc(sc),
   1714        1.1  riastrad 	    uatp_y_sensors(sc), uatp_y_ratio(sc), &y_raw, &y_fingers);
   1715        1.1  riastrad 
   1716        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_PARSE,
   1717        1.1  riastrad 	    ("x %u @ %u, %uf; y %u @ %u, %uf; buttons %"PRIx32"\n",
   1718        1.1  riastrad 		x_pressure, x_raw, x_fingers,
   1719        1.1  riastrad 		y_pressure, y_raw, y_fingers,
   1720        1.1  riastrad 		*buttons));
   1721        1.1  riastrad 
   1722        1.1  riastrad 	if ((x_pressure == 0) && (y_pressure == 0)) {
   1723        1.1  riastrad 		bool ok;
   1724        1.1  riastrad 		/* No fingers: clear position and maybe report a tap.  */
   1725        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_INTR,
   1726        1.1  riastrad 		    ("no position detected; clearing position\n"));
   1727        1.1  riastrad 		if (*buttons == 0) {
   1728        1.1  riastrad 			ok = tap_released(sc);
   1729        1.1  riastrad 		} else {
   1730        1.1  riastrad 			tap_reset(sc);
   1731        1.1  riastrad 			/* Button pressed: interrupt is not spurious.  */
   1732        1.1  riastrad 			ok = true;
   1733        1.1  riastrad 		}
   1734        1.1  riastrad 		/*
   1735        1.1  riastrad 		 * Don't clear the position until after tap_released,
   1736        1.1  riastrad 		 * which needs to know the track distance.
   1737        1.1  riastrad 		 */
   1738        1.1  riastrad 		uatp_clear_position(sc);
   1739        1.1  riastrad 		return ok;
   1740        1.1  riastrad 	} else if ((x_pressure == 0) || (y_pressure == 0)) {
   1741        1.1  riastrad 		/* XXX What to do here?  */
   1742        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_INTR,
   1743        1.1  riastrad 		    ("pressure in only one dimension; ignoring\n"));
   1744        1.1  riastrad 		return true;
   1745        1.1  riastrad 	} else if ((x_pressure == 1) && (y_pressure == 1)) {
   1746        1.1  riastrad 		fingers = max(x_fingers, y_fingers);
   1747        1.1  riastrad 		CHECK((0 < fingers), return false);
   1748        1.1  riastrad 		if (*buttons == 0)
   1749        1.1  riastrad 			tap_touched(sc, fingers);
   1750        1.1  riastrad 		else if (fingers == 1)
   1751        1.1  riastrad 			tap_reset(sc);
   1752        1.1  riastrad 		else		/* Multiple fingers, button pressed.  */
   1753        1.1  riastrad 			*buttons = emulated_buttons(sc, fingers);
   1754        1.1  riastrad 		update_position(sc, fingers, x_raw, y_raw, dx, dy, dz, dw);
   1755        1.1  riastrad 		return true;
   1756        1.1  riastrad 	} else {
   1757        1.1  riastrad 		/* Palm detected in either or both of the dimensions.  */
   1758        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_INTR, ("palm detected; ignoring\n"));
   1759        1.1  riastrad 		return true;
   1760        1.1  riastrad 	}
   1761        1.1  riastrad }
   1762   1.10.4.2     skrll 
   1763        1.1  riastrad /*
   1764        1.1  riastrad  * Interpret the accumulated sensor state along one dimension to find
   1765        1.1  riastrad  * the number, mean position, and pressure of fingers.  Returns 0 to
   1766        1.1  riastrad  * indicate no pressure, returns 1 and sets *position and *fingers to
   1767        1.1  riastrad  * indicate fingers, and returns 2 to indicate palm.
   1768        1.1  riastrad  *
   1769        1.1  riastrad  * XXX Give symbolic names to the return values.
   1770        1.1  riastrad  */
   1771        1.1  riastrad 
   1772        1.1  riastrad static unsigned int
   1773        1.1  riastrad interpret_dimension(struct uatp_softc *sc, const int *acc,
   1774        1.1  riastrad     unsigned int n_sensors, unsigned int ratio,
   1775        1.1  riastrad     unsigned int *position, unsigned int *fingers)
   1776        1.1  riastrad {
   1777        1.1  riastrad 	unsigned int i, v, n_fingers, sum;
   1778        1.1  riastrad 	unsigned int total[UATP_MAX_SENSORS];
   1779        1.1  riastrad 	unsigned int weighted[UATP_MAX_SENSORS];
   1780        1.1  riastrad 	unsigned int sensor_threshold = sc->sc_knobs.sensor_threshold;
   1781        1.1  riastrad 	unsigned int sensor_normalizer = sc->sc_knobs.sensor_normalizer;
   1782        1.1  riastrad 	unsigned int width = 0;	/* GCC is not smart enough.  */
   1783        1.1  riastrad 	unsigned int palm_width = sc->sc_knobs.palm_width;
   1784        1.1  riastrad 	enum { none, nondecreasing, decreasing } state = none;
   1785        1.1  riastrad 
   1786        1.1  riastrad 	if (sensor_threshold < sensor_normalizer)
   1787        1.1  riastrad 		sensor_normalizer = sensor_threshold;
   1788        1.1  riastrad 	if (palm_width == 0)	/* Effectively disable palm detection.  */
   1789        1.1  riastrad 		palm_width = UATP_MAX_POSITION;
   1790        1.1  riastrad 
   1791        1.1  riastrad #define CHECK_(condition) CHECK(condition, return 0)
   1792        1.1  riastrad 
   1793        1.1  riastrad 	/*
   1794        1.1  riastrad 	 * Arithmetic bounds:
   1795        1.1  riastrad 	 * . n_sensors is at most UATP_MAX_SENSORS,
   1796        1.1  riastrad 	 * . n_fingers is at most UATP_MAX_SENSORS,
   1797        1.1  riastrad 	 * . i is at most UATP_MAX_SENSORS,
   1798        1.1  riastrad 	 * . sc->sc_acc[i] is at most UATP_MAX_ACC,
   1799        1.1  riastrad 	 * . i * sc->sc_acc[i] is at most UATP_MAX_SENSORS * UATP_MAX_ACC,
   1800        1.1  riastrad 	 * . each total[j] is at most UATP_MAX_SENSORS * UATP_MAX_ACC,
   1801        1.1  riastrad 	 * . each weighted[j] is at most UATP_MAX_SENSORS^2 * UATP_MAX_ACC,
   1802        1.1  riastrad 	 * . ratio is at most UATP_MAX_RATIO,
   1803        1.1  riastrad 	 * . each weighted[j] * ratio is at most
   1804        1.1  riastrad 	 *     UATP_MAX_SENSORS^2 * UATP_MAX_ACC * UATP_MAX_RATIO,
   1805        1.1  riastrad 	 *   which is #x5fa0000 with the current values of the constants,
   1806        1.1  riastrad 	 *   and
   1807        1.1  riastrad 	 * . the sum of the positions is at most
   1808        1.1  riastrad 	 *     UATP_MAX_SENSORS * UATP_MAX_POSITION,
   1809        1.1  riastrad 	 *   which is #x60000 with the current values of the constants.
   1810        1.1  riastrad 	 * Hence all of the arithmetic here fits in int (and thus also
   1811        1.1  riastrad 	 * unsigned int).  If you change the constants, though, you
   1812        1.1  riastrad 	 * must update the analysis.
   1813        1.1  riastrad 	 */
   1814        1.1  riastrad 	__CTASSERT(0x5fa0000 == (UATP_MAX_SENSORS * UATP_MAX_SENSORS *
   1815        1.1  riastrad 		UATP_MAX_ACC * UATP_MAX_RATIO));
   1816        1.1  riastrad 	__CTASSERT(0x60000 == (UATP_MAX_SENSORS * UATP_MAX_POSITION));
   1817        1.1  riastrad 	CHECK_(n_sensors <= UATP_MAX_SENSORS);
   1818        1.1  riastrad 	CHECK_(ratio <= UATP_MAX_RATIO);
   1819        1.1  riastrad 
   1820        1.1  riastrad 	/*
   1821        1.1  riastrad 	 * Detect each finger by looking for a consecutive sequence of
   1822        1.1  riastrad 	 * increasing and then decreasing pressures above the sensor
   1823        1.1  riastrad 	 * threshold.  Compute the finger's position as the weighted
   1824        1.1  riastrad 	 * average of positions, weighted by the pressure at that
   1825        1.1  riastrad 	 * position.  Finally, return the average finger position.
   1826        1.1  riastrad 	 */
   1827        1.1  riastrad 
   1828        1.1  riastrad 	n_fingers = 0;
   1829   1.10.4.6     skrll 	memset(weighted, 0, sizeof(weighted));
   1830   1.10.4.6     skrll 	memset(total, 0, sizeof(total));
   1831   1.10.4.2     skrll 
   1832        1.1  riastrad 	for (i = 0; i < n_sensors; i++) {
   1833        1.1  riastrad 		CHECK_(0 <= acc[i]);
   1834        1.1  riastrad 		v = acc[i];
   1835        1.1  riastrad 
   1836        1.1  riastrad 		/* Ignore values outside a sensible interval.  */
   1837        1.1  riastrad 		if (v <= sensor_threshold) {
   1838        1.1  riastrad 			state = none;
   1839        1.1  riastrad 			continue;
   1840        1.1  riastrad 		} else if (UATP_MAX_ACC < v) {
   1841        1.1  riastrad 			aprint_verbose_dev(uatp_dev(sc),
   1842        1.1  riastrad 			    "ignoring large accumulated sensor state: %u\n",
   1843        1.1  riastrad 			    v);
   1844        1.1  riastrad 			continue;
   1845        1.1  riastrad 		}
   1846        1.1  riastrad 
   1847        1.1  riastrad 		switch (state) {
   1848        1.1  riastrad 		case none:
   1849        1.1  riastrad 			n_fingers += 1;
   1850        1.1  riastrad 			CHECK_(n_fingers <= n_sensors);
   1851        1.1  riastrad 			state = nondecreasing;
   1852        1.1  riastrad 			width = 1;
   1853        1.1  riastrad 			break;
   1854        1.1  riastrad 
   1855        1.1  riastrad 		case nondecreasing:
   1856        1.1  riastrad 		case decreasing:
   1857        1.1  riastrad 			CHECK_(0 < i);
   1858        1.1  riastrad 			CHECK_(0 <= acc[i - 1]);
   1859        1.1  riastrad 			width += 1;
   1860        1.1  riastrad 			if (palm_width <= (width * ratio)) {
   1861        1.1  riastrad 				DPRINTF(sc, UATP_DEBUG_PALM,
   1862        1.1  riastrad 				    ("palm detected\n"));
   1863        1.1  riastrad 				return 2;
   1864        1.1  riastrad 			} else if ((state == nondecreasing) &&
   1865        1.1  riastrad 			    ((unsigned int)acc[i - 1] > v)) {
   1866        1.1  riastrad 				state = decreasing;
   1867        1.1  riastrad 			} else if ((state == decreasing) &&
   1868        1.1  riastrad 			    ((unsigned int)acc[i - 1] < v)) {
   1869        1.1  riastrad 				n_fingers += 1;
   1870        1.1  riastrad 				CHECK_(n_fingers <= n_sensors);
   1871        1.1  riastrad 				state = nondecreasing;
   1872        1.1  riastrad 				width = 1;
   1873        1.1  riastrad 			}
   1874        1.1  riastrad 			break;
   1875        1.1  riastrad 
   1876        1.1  riastrad 		default:
   1877        1.1  riastrad 			aprint_error_dev(uatp_dev(sc),
   1878        1.1  riastrad 			    "bad finger detection state: %d", state);
   1879        1.1  riastrad 			return 0;
   1880        1.1  riastrad 		}
   1881        1.1  riastrad 
   1882        1.1  riastrad 		v -= sensor_normalizer;
   1883        1.1  riastrad 		total[n_fingers - 1] += v;
   1884        1.1  riastrad 		weighted[n_fingers - 1] += (i * v);
   1885        1.1  riastrad 		CHECK_(total[n_fingers - 1] <=
   1886        1.1  riastrad 		    (UATP_MAX_SENSORS * UATP_MAX_ACC));
   1887        1.1  riastrad 		CHECK_(weighted[n_fingers - 1] <=
   1888        1.1  riastrad 		    (UATP_MAX_SENSORS * UATP_MAX_SENSORS * UATP_MAX_ACC));
   1889        1.1  riastrad 	}
   1890        1.1  riastrad 
   1891        1.1  riastrad 	if (n_fingers == 0)
   1892        1.1  riastrad 		return 0;
   1893        1.1  riastrad 
   1894        1.1  riastrad 	sum = 0;
   1895        1.1  riastrad 	for (i = 0; i < n_fingers; i++) {
   1896        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_PARSE,
   1897        1.1  riastrad 		    ("finger at %u\n", ((weighted[i] * ratio) / total[i])));
   1898        1.1  riastrad 		sum += ((weighted[i] * ratio) / total[i]);
   1899        1.1  riastrad 		CHECK_(sum <= UATP_MAX_SENSORS * UATP_MAX_POSITION);
   1900        1.1  riastrad 	}
   1901        1.1  riastrad 
   1902        1.1  riastrad 	*fingers = n_fingers;
   1903        1.1  riastrad 	*position = (sum / n_fingers);
   1904        1.1  riastrad 	return 1;
   1905        1.1  riastrad 
   1906        1.1  riastrad #undef CHECK_
   1907        1.1  riastrad }
   1908   1.10.4.2     skrll 
   1909        1.1  riastrad /* Tapping */
   1910        1.1  riastrad 
   1911        1.1  riastrad /*
   1912        1.1  riastrad  * There is a very hairy state machine for detecting taps.  At every
   1913        1.1  riastrad  * touch, we record the maximum number of fingers touched, and don't
   1914        1.1  riastrad  * reset it to zero until the finger is released.
   1915        1.1  riastrad  *
   1916        1.1  riastrad  * INITIAL STATE
   1917        1.1  riastrad  * (no tapping fingers; no tapped fingers)
   1918        1.1  riastrad  * - On touch, go to TAPPING STATE.
   1919        1.1  riastrad  * - On any other input, remain in INITIAL STATE.
   1920        1.1  riastrad  *
   1921        1.1  riastrad  * TAPPING STATE: Finger touched; might be tap.
   1922        1.1  riastrad  * (tapping fingers; no tapped fingers)
   1923        1.1  riastrad  * - On release within the tap limit, go to TAPPED STATE.
   1924        1.1  riastrad  * - On release after the tap limit, go to INITIAL STATE.
   1925        1.1  riastrad  * - On any other input, remain in TAPPING STATE.
   1926        1.1  riastrad  *
   1927        1.1  riastrad  * TAPPED STATE: Finger recently tapped, and might double-tap.
   1928        1.1  riastrad  * (no tapping fingers; tapped fingers)
   1929        1.1  riastrad  * - On touch within the double-tap limit, go to DOUBLE-TAPPING STATE.
   1930        1.1  riastrad  * - On touch after the double-tap limit, go to TAPPING STATE.
   1931        1.1  riastrad  * - On no event after the double-tap limit, go to INITIAL STATE.
   1932        1.1  riastrad  * - On any other input, remain in TAPPED STATE.
   1933        1.1  riastrad  *
   1934        1.1  riastrad  * DOUBLE-TAPPING STATE: Finger touched soon after tap; might be double-tap.
   1935        1.1  riastrad  * (tapping fingers; tapped fingers)
   1936        1.1  riastrad  * - On release within the tap limit, release button and go to TAPPED STATE.
   1937        1.1  riastrad  * - On release after the tap limit, go to DRAGGING UP STATE.
   1938        1.1  riastrad  * - On touch after the tap limit, go to DRAGGING DOWN STATE.
   1939        1.1  riastrad  * - On any other input, remain in DOUBLE-TAPPING STATE.
   1940        1.1  riastrad  *
   1941        1.1  riastrad  * DRAGGING DOWN STATE: Finger has double-tapped and is dragging, not tapping.
   1942        1.1  riastrad  * (no tapping fingers; tapped fingers)
   1943        1.1  riastrad  * - On release, go to DRAGGING UP STATE.
   1944        1.1  riastrad  * - On any other input, remain in DRAGGING DOWN STATE.
   1945        1.1  riastrad  *
   1946        1.1  riastrad  * DRAGGING UP STATE: Finger has double-tapped and is up.
   1947        1.1  riastrad  * (no tapping fingers; tapped fingers)
   1948        1.1  riastrad  * - On touch, go to TAPPING IN DRAG STATE.
   1949        1.1  riastrad  * - On any other input, remain in DRAGGING UP STATE.
   1950        1.1  riastrad  *
   1951        1.1  riastrad  * TAPPING IN DRAG STATE: Tap-dancing while cross-dressed.
   1952        1.1  riastrad  * (tapping fingers; tapped fingers)
   1953        1.1  riastrad  * - On release within the tap limit, go to TAPPED STATE.
   1954        1.1  riastrad  * - On release after the tap limit, go to DRAGGING UP STATE.
   1955        1.1  riastrad  * - On any other input, remain in TAPPING IN DRAG STATE.
   1956        1.1  riastrad  *
   1957        1.1  riastrad  * Warning:  The graph of states is split into two components, those
   1958        1.1  riastrad  * with tapped fingers and those without.  The only path from any state
   1959        1.1  riastrad  * without tapped fingers to a state with tapped fingers must pass
   1960        1.1  riastrad  * through TAPPED STATE.  Also, the only transitions into TAPPED STATE
   1961        1.1  riastrad  * must be from states with tapping fingers, which become the tapped
   1962        1.1  riastrad  * fingers.  If you edit the state machine, you must either preserve
   1963        1.1  riastrad  * these properties, or globally transform the state machine to avoid
   1964        1.1  riastrad  * the bad consequences of violating these properties.
   1965        1.1  riastrad  */
   1966   1.10.4.2     skrll 
   1967        1.1  riastrad static void
   1968        1.1  riastrad uatp_tap_limit(const struct uatp_softc *sc, struct timeval *limit)
   1969        1.1  riastrad {
   1970        1.1  riastrad 	unsigned int msec = sc->sc_knobs.tap_limit_msec;
   1971        1.1  riastrad 	limit->tv_sec = 0;
   1972        1.1  riastrad 	limit->tv_usec = ((msec < 1000) ? (1000 * msec) : 100000);
   1973        1.1  riastrad }
   1974        1.1  riastrad 
   1975        1.1  riastrad #if UATP_DEBUG
   1976        1.1  riastrad 
   1977        1.1  riastrad #  define TAP_DEBUG_PRE(sc)	tap_debug((sc), __func__, "")
   1978        1.1  riastrad #  define TAP_DEBUG_POST(sc)	tap_debug((sc), __func__, " ->")
   1979        1.1  riastrad 
   1980        1.1  riastrad static void
   1981        1.1  riastrad tap_debug(struct uatp_softc *sc, const char *caller, const char *prefix)
   1982        1.1  riastrad {
   1983        1.1  riastrad 	char buffer[128];
   1984        1.1  riastrad 	const char *state;
   1985        1.1  riastrad 
   1986        1.1  riastrad 	KASSERT(mutex_owned(&sc->sc_tap_mutex));
   1987        1.1  riastrad 	switch (sc->sc_tap_state) {
   1988        1.1  riastrad 	case TAP_STATE_INITIAL:		state = "initial";		break;
   1989        1.1  riastrad 	case TAP_STATE_TAPPING:		state = "tapping";		break;
   1990        1.1  riastrad 	case TAP_STATE_TAPPED:		state = "tapped";		break;
   1991        1.1  riastrad 	case TAP_STATE_DOUBLE_TAPPING:	state = "double-tapping";	break;
   1992        1.1  riastrad 	case TAP_STATE_DRAGGING_DOWN:	state = "dragging-down";	break;
   1993        1.1  riastrad 	case TAP_STATE_DRAGGING_UP:	state = "dragging-up";		break;
   1994        1.1  riastrad 	case TAP_STATE_TAPPING_IN_DRAG:	state = "tapping-in-drag";	break;
   1995        1.1  riastrad 	default:
   1996   1.10.4.6     skrll 		snprintf(buffer, sizeof(buffer), "unknown (%d)",
   1997        1.1  riastrad 		    sc->sc_tap_state);
   1998        1.1  riastrad 		state = buffer;
   1999        1.1  riastrad 		break;
   2000        1.1  riastrad 	}
   2001        1.1  riastrad 
   2002        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_TAP,
   2003        1.1  riastrad 	    ("%s:%s state %s, %u tapping, %u tapped\n",
   2004        1.1  riastrad 		caller, prefix, state,
   2005        1.1  riastrad 		sc->sc_tapping_fingers, sc->sc_tapped_fingers));
   2006        1.1  riastrad }
   2007        1.1  riastrad 
   2008        1.1  riastrad #else	/* !UATP_DEBUG */
   2009        1.1  riastrad 
   2010        1.1  riastrad #  define TAP_DEBUG_PRE(sc)	do {} while (0)
   2011        1.1  riastrad #  define TAP_DEBUG_POST(sc)	do {} while (0)
   2012        1.1  riastrad 
   2013        1.1  riastrad #endif
   2014   1.10.4.2     skrll 
   2015        1.1  riastrad static void
   2016        1.1  riastrad tap_initialize(struct uatp_softc *sc)
   2017        1.1  riastrad {
   2018   1.10.4.4     skrll 	callout_init(&sc->sc_untap_callout, 0);
   2019        1.1  riastrad 	callout_setfunc(&sc->sc_untap_callout, untap_callout, sc);
   2020   1.10.4.7     skrll 	mutex_init(&sc->sc_tap_mutex, MUTEX_DEFAULT, IPL_SOFTUSB);
   2021        1.1  riastrad }
   2022        1.1  riastrad 
   2023        1.1  riastrad static void
   2024        1.1  riastrad tap_finalize(struct uatp_softc *sc)
   2025        1.1  riastrad {
   2026        1.1  riastrad 	/* XXX Can the callout still be scheduled here?  */
   2027        1.1  riastrad 	callout_destroy(&sc->sc_untap_callout);
   2028        1.1  riastrad 	mutex_destroy(&sc->sc_tap_mutex);
   2029        1.1  riastrad }
   2030        1.1  riastrad 
   2031        1.1  riastrad static void
   2032        1.1  riastrad tap_enable(struct uatp_softc *sc)
   2033        1.1  riastrad {
   2034        1.1  riastrad 	mutex_enter(&sc->sc_tap_mutex);
   2035        1.1  riastrad 	tap_transition_initial(sc);
   2036        1.1  riastrad 	sc->sc_buttons = 0;	/* XXX Not the right place?  */
   2037        1.1  riastrad 	sc->sc_all_buttons = 0;
   2038        1.1  riastrad 	mutex_exit(&sc->sc_tap_mutex);
   2039        1.1  riastrad }
   2040        1.1  riastrad 
   2041        1.1  riastrad static void
   2042        1.1  riastrad tap_disable(struct uatp_softc *sc)
   2043        1.1  riastrad {
   2044        1.1  riastrad 	/* Reset tapping, and wait for any callouts to complete.  */
   2045        1.1  riastrad 	tap_reset_wait(sc);
   2046        1.1  riastrad }
   2047        1.1  riastrad 
   2048        1.1  riastrad /*
   2049        1.1  riastrad  * Reset tap state.  If the untap callout has just fired, it may signal
   2050        1.1  riastrad  * a harmless button release event before this returns.
   2051        1.1  riastrad  */
   2052        1.1  riastrad 
   2053        1.1  riastrad static void
   2054        1.1  riastrad tap_reset(struct uatp_softc *sc)
   2055        1.1  riastrad {
   2056  1.10.4.10     skrll 
   2057        1.1  riastrad 	callout_stop(&sc->sc_untap_callout);
   2058        1.1  riastrad 	mutex_enter(&sc->sc_tap_mutex);
   2059        1.1  riastrad 	tap_transition_initial(sc);
   2060        1.1  riastrad 	mutex_exit(&sc->sc_tap_mutex);
   2061        1.1  riastrad }
   2062        1.1  riastrad 
   2063        1.1  riastrad /* Reset, but don't return until the callout is done running.  */
   2064        1.1  riastrad 
   2065        1.1  riastrad static void
   2066        1.1  riastrad tap_reset_wait(struct uatp_softc *sc)
   2067        1.1  riastrad {
   2068        1.1  riastrad 
   2069  1.10.4.10     skrll 	callout_halt(&sc->sc_untap_callout, NULL);
   2070        1.1  riastrad 	mutex_enter(&sc->sc_tap_mutex);
   2071        1.1  riastrad 	tap_transition_initial(sc);
   2072        1.1  riastrad 	mutex_exit(&sc->sc_tap_mutex);
   2073        1.1  riastrad }
   2074   1.10.4.2     skrll 
   2075        1.1  riastrad static const struct timeval zero_timeval;
   2076        1.1  riastrad 
   2077        1.1  riastrad static void
   2078        1.1  riastrad tap_transition(struct uatp_softc *sc, enum uatp_tap_state tap_state,
   2079        1.1  riastrad     const struct timeval *start_time,
   2080        1.1  riastrad     unsigned int tapping_fingers, unsigned int tapped_fingers)
   2081        1.1  riastrad {
   2082        1.1  riastrad 	KASSERT(mutex_owned(&sc->sc_tap_mutex));
   2083        1.1  riastrad 	sc->sc_tap_state = tap_state;
   2084        1.1  riastrad 	sc->sc_tap_timer = *start_time;
   2085        1.1  riastrad 	sc->sc_tapping_fingers = tapping_fingers;
   2086        1.1  riastrad 	sc->sc_tapped_fingers = tapped_fingers;
   2087        1.1  riastrad }
   2088        1.1  riastrad 
   2089        1.1  riastrad static void
   2090        1.1  riastrad tap_transition_initial(struct uatp_softc *sc)
   2091        1.1  riastrad {
   2092        1.1  riastrad 	/*
   2093        1.1  riastrad 	 * No checks.  This state is always kosher, and sometimes a
   2094        1.1  riastrad 	 * fallback in case of failure.
   2095        1.1  riastrad 	 */
   2096        1.1  riastrad 	tap_transition(sc, TAP_STATE_INITIAL, &zero_timeval, 0, 0);
   2097        1.1  riastrad }
   2098        1.1  riastrad 
   2099        1.1  riastrad /* Touch transitions */
   2100        1.1  riastrad 
   2101        1.1  riastrad static void
   2102        1.1  riastrad tap_transition_tapping(struct uatp_softc *sc, const struct timeval *start_time,
   2103        1.1  riastrad     unsigned int fingers)
   2104        1.1  riastrad {
   2105        1.1  riastrad 	CHECK((sc->sc_tapping_fingers <= fingers),
   2106        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2107        1.1  riastrad 	tap_transition(sc, TAP_STATE_TAPPING, start_time, fingers, 0);
   2108        1.1  riastrad }
   2109        1.1  riastrad 
   2110        1.1  riastrad static void
   2111        1.1  riastrad tap_transition_double_tapping(struct uatp_softc *sc,
   2112        1.1  riastrad     const struct timeval *start_time, unsigned int fingers)
   2113        1.1  riastrad {
   2114        1.1  riastrad 	CHECK((sc->sc_tapping_fingers <= fingers),
   2115        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2116        1.1  riastrad 	CHECK((0 < sc->sc_tapped_fingers),
   2117        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2118        1.1  riastrad 	tap_transition(sc, TAP_STATE_DOUBLE_TAPPING, start_time, fingers,
   2119        1.1  riastrad 	    sc->sc_tapped_fingers);
   2120        1.1  riastrad }
   2121   1.10.4.2     skrll 
   2122        1.1  riastrad static void
   2123        1.1  riastrad tap_transition_dragging_down(struct uatp_softc *sc)
   2124        1.1  riastrad {
   2125        1.1  riastrad 	CHECK((0 < sc->sc_tapped_fingers),
   2126        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2127        1.1  riastrad 	tap_transition(sc, TAP_STATE_DRAGGING_DOWN, &zero_timeval, 0,
   2128        1.1  riastrad 	    sc->sc_tapped_fingers);
   2129        1.1  riastrad }
   2130        1.1  riastrad 
   2131        1.1  riastrad static void
   2132        1.1  riastrad tap_transition_tapping_in_drag(struct uatp_softc *sc,
   2133        1.1  riastrad     const struct timeval *start_time, unsigned int fingers)
   2134        1.1  riastrad {
   2135        1.1  riastrad 	CHECK((sc->sc_tapping_fingers <= fingers),
   2136        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2137        1.1  riastrad 	CHECK((0 < sc->sc_tapped_fingers),
   2138        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2139        1.1  riastrad 	tap_transition(sc, TAP_STATE_TAPPING_IN_DRAG, start_time, fingers,
   2140        1.1  riastrad 	    sc->sc_tapped_fingers);
   2141        1.1  riastrad }
   2142        1.1  riastrad 
   2143        1.1  riastrad /* Release transitions */
   2144        1.1  riastrad 
   2145        1.1  riastrad static void
   2146        1.1  riastrad tap_transition_tapped(struct uatp_softc *sc, const struct timeval *start_time)
   2147        1.1  riastrad {
   2148        1.1  riastrad 	/*
   2149        1.1  riastrad 	 * The fingers that were tapping -- of which there must have
   2150        1.1  riastrad 	 * been at least one -- are now the fingers that have tapped,
   2151        1.1  riastrad 	 * and there are no longer fingers tapping.
   2152        1.1  riastrad 	 */
   2153        1.1  riastrad 	CHECK((0 < sc->sc_tapping_fingers),
   2154        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2155        1.1  riastrad 	tap_transition(sc, TAP_STATE_TAPPED, start_time, 0,
   2156        1.1  riastrad 	    sc->sc_tapping_fingers);
   2157        1.1  riastrad 	schedule_untap(sc);
   2158        1.1  riastrad }
   2159        1.1  riastrad 
   2160        1.1  riastrad static void
   2161        1.1  riastrad tap_transition_dragging_up(struct uatp_softc *sc)
   2162        1.1  riastrad {
   2163        1.1  riastrad 	CHECK((0 < sc->sc_tapped_fingers),
   2164        1.1  riastrad 	    do { tap_transition_initial(sc); return; } while (0));
   2165        1.1  riastrad 	tap_transition(sc, TAP_STATE_DRAGGING_UP, &zero_timeval, 0,
   2166        1.1  riastrad 	    sc->sc_tapped_fingers);
   2167        1.1  riastrad }
   2168   1.10.4.2     skrll 
   2169        1.1  riastrad static void
   2170        1.1  riastrad tap_touched(struct uatp_softc *sc, unsigned int fingers)
   2171        1.1  riastrad {
   2172        1.1  riastrad 	struct timeval now, diff, limit;
   2173        1.1  riastrad 
   2174        1.1  riastrad 	CHECK((0 < fingers), return);
   2175        1.1  riastrad 	callout_stop(&sc->sc_untap_callout);
   2176        1.1  riastrad 	mutex_enter(&sc->sc_tap_mutex);
   2177        1.1  riastrad 	TAP_DEBUG_PRE(sc);
   2178        1.1  riastrad 	/*
   2179        1.1  riastrad 	 * Guarantee that the number of tapping fingers never decreases
   2180        1.1  riastrad 	 * except when it is reset to zero on release.
   2181        1.1  riastrad 	 */
   2182        1.1  riastrad 	if (fingers < sc->sc_tapping_fingers)
   2183        1.1  riastrad 		fingers = sc->sc_tapping_fingers;
   2184        1.1  riastrad 	switch (sc->sc_tap_state) {
   2185        1.1  riastrad 	case TAP_STATE_INITIAL:
   2186        1.1  riastrad 		getmicrouptime(&now);
   2187        1.1  riastrad 		tap_transition_tapping(sc, &now, fingers);
   2188        1.1  riastrad 		break;
   2189        1.1  riastrad 
   2190        1.1  riastrad 	case TAP_STATE_TAPPING:
   2191        1.1  riastrad 		/*
   2192        1.1  riastrad 		 * Number of fingers may have increased, so transition
   2193        1.1  riastrad 		 * even though we're already in TAPPING.
   2194        1.1  riastrad 		 */
   2195        1.1  riastrad 		tap_transition_tapping(sc, &sc->sc_tap_timer, fingers);
   2196        1.1  riastrad 		break;
   2197        1.1  riastrad 
   2198        1.1  riastrad 	case TAP_STATE_TAPPED:
   2199        1.1  riastrad 		getmicrouptime(&now);
   2200        1.1  riastrad 		/*
   2201        1.1  riastrad 		 * If the double-tap time limit has passed, it's the
   2202        1.1  riastrad 		 * callout's responsibility to handle that event, so we
   2203        1.1  riastrad 		 * assume the limit has not passed yet.
   2204        1.1  riastrad 		 */
   2205        1.1  riastrad 		tap_transition_double_tapping(sc, &now, fingers);
   2206        1.1  riastrad 		break;
   2207        1.1  riastrad 
   2208        1.1  riastrad 	case TAP_STATE_DOUBLE_TAPPING:
   2209        1.1  riastrad 		getmicrouptime(&now);
   2210        1.1  riastrad 		timersub(&now, &sc->sc_tap_timer, &diff);
   2211        1.1  riastrad 		uatp_tap_limit(sc, &limit);
   2212        1.1  riastrad 		if (timercmp(&diff, &limit, >) ||
   2213        1.1  riastrad 		    (sc->sc_track_distance >
   2214        1.1  riastrad 			sc->sc_knobs.tap_track_distance_limit))
   2215        1.1  riastrad 			tap_transition_dragging_down(sc);
   2216        1.1  riastrad 		break;
   2217        1.1  riastrad 
   2218        1.1  riastrad 	case TAP_STATE_DRAGGING_DOWN:
   2219        1.1  riastrad 		break;
   2220        1.1  riastrad 
   2221        1.1  riastrad 	case TAP_STATE_DRAGGING_UP:
   2222        1.1  riastrad 		getmicrouptime(&now);
   2223        1.1  riastrad 		tap_transition_tapping_in_drag(sc, &now, fingers);
   2224        1.1  riastrad 		break;
   2225        1.1  riastrad 
   2226        1.1  riastrad 	case TAP_STATE_TAPPING_IN_DRAG:
   2227        1.1  riastrad 		/*
   2228        1.1  riastrad 		 * Number of fingers may have increased, so transition
   2229        1.1  riastrad 		 * even though we're already in TAPPING IN DRAG.
   2230        1.1  riastrad 		 */
   2231        1.1  riastrad 		tap_transition_tapping_in_drag(sc, &sc->sc_tap_timer, fingers);
   2232        1.1  riastrad 		break;
   2233        1.1  riastrad 
   2234        1.1  riastrad 	default:
   2235        1.1  riastrad 		aprint_error_dev(uatp_dev(sc), "%s: invalid tap state: %d\n",
   2236        1.1  riastrad 		    __func__, sc->sc_tap_state);
   2237        1.1  riastrad 		tap_transition_initial(sc);
   2238        1.1  riastrad 		break;
   2239        1.1  riastrad 	}
   2240        1.1  riastrad 	TAP_DEBUG_POST(sc);
   2241        1.1  riastrad 	mutex_exit(&sc->sc_tap_mutex);
   2242        1.1  riastrad }
   2243   1.10.4.2     skrll 
   2244        1.1  riastrad static bool
   2245        1.1  riastrad tap_released(struct uatp_softc *sc)
   2246        1.1  riastrad {
   2247        1.1  riastrad 	struct timeval now, diff, limit;
   2248        1.1  riastrad 	void (*non_tapped_transition)(struct uatp_softc *);
   2249        1.1  riastrad 	bool ok, temporary_release;
   2250        1.1  riastrad 
   2251        1.1  riastrad 	mutex_enter(&sc->sc_tap_mutex);
   2252        1.1  riastrad 	TAP_DEBUG_PRE(sc);
   2253        1.1  riastrad 	switch (sc->sc_tap_state) {
   2254        1.1  riastrad 	case TAP_STATE_INITIAL:
   2255        1.1  riastrad 	case TAP_STATE_TAPPED:
   2256        1.1  riastrad 	case TAP_STATE_DRAGGING_UP:
   2257        1.1  riastrad 		/* Spurious interrupt: fingers are already off.  */
   2258        1.1  riastrad 		ok = false;
   2259        1.1  riastrad 		break;
   2260        1.1  riastrad 
   2261        1.1  riastrad 	case TAP_STATE_TAPPING:
   2262        1.1  riastrad 		temporary_release = false;
   2263        1.1  riastrad 		non_tapped_transition = &tap_transition_initial;
   2264        1.1  riastrad 		goto maybe_tap;
   2265        1.1  riastrad 
   2266        1.1  riastrad 	case TAP_STATE_DOUBLE_TAPPING:
   2267        1.1  riastrad 		temporary_release = true;
   2268        1.1  riastrad 		non_tapped_transition = &tap_transition_dragging_up;
   2269        1.1  riastrad 		goto maybe_tap;
   2270        1.1  riastrad 
   2271        1.1  riastrad 	case TAP_STATE_TAPPING_IN_DRAG:
   2272        1.1  riastrad 		temporary_release = false;
   2273        1.1  riastrad 		non_tapped_transition = &tap_transition_dragging_up;
   2274        1.1  riastrad 		goto maybe_tap;
   2275        1.1  riastrad 
   2276        1.1  riastrad 	maybe_tap:
   2277        1.1  riastrad 		getmicrouptime(&now);
   2278        1.1  riastrad 		timersub(&now, &sc->sc_tap_timer, &diff);
   2279        1.1  riastrad 		uatp_tap_limit(sc, &limit);
   2280        1.1  riastrad 		if (timercmp(&diff, &limit, <=) &&
   2281        1.1  riastrad 		    (sc->sc_track_distance <=
   2282        1.1  riastrad 			sc->sc_knobs.tap_track_distance_limit)) {
   2283        1.1  riastrad 			if (temporary_release) {
   2284        1.1  riastrad 				/*
   2285        1.1  riastrad 				 * XXX Kludge: Temporarily transition
   2286        1.1  riastrad 				 * to a tap state that uatp_input will
   2287        1.1  riastrad 				 * interpret as `no buttons tapped',
   2288        1.1  riastrad 				 * saving the tapping fingers.  There
   2289        1.1  riastrad 				 * should instead be a separate routine
   2290        1.1  riastrad 				 * uatp_input_untapped.
   2291        1.1  riastrad 				 */
   2292        1.1  riastrad 				unsigned int fingers = sc->sc_tapping_fingers;
   2293        1.1  riastrad 				tap_transition_initial(sc);
   2294        1.1  riastrad 				uatp_input(sc, 0, 0, 0, 0, 0);
   2295        1.1  riastrad 				sc->sc_tapping_fingers = fingers;
   2296        1.1  riastrad 			}
   2297        1.1  riastrad 			tap_transition_tapped(sc, &now);
   2298        1.1  riastrad 		} else {
   2299        1.1  riastrad 			(*non_tapped_transition)(sc);
   2300        1.1  riastrad 		}
   2301        1.1  riastrad 		ok = true;
   2302        1.1  riastrad 		break;
   2303        1.1  riastrad 
   2304        1.1  riastrad 	case TAP_STATE_DRAGGING_DOWN:
   2305        1.1  riastrad 		tap_transition_dragging_up(sc);
   2306        1.1  riastrad 		ok = true;
   2307        1.1  riastrad 		break;
   2308        1.1  riastrad 
   2309        1.1  riastrad 	default:
   2310        1.1  riastrad 		aprint_error_dev(uatp_dev(sc), "%s: invalid tap state: %d\n",
   2311        1.1  riastrad 		    __func__, sc->sc_tap_state);
   2312        1.1  riastrad 		tap_transition_initial(sc);
   2313        1.1  riastrad 		ok = false;
   2314        1.1  riastrad 		break;
   2315        1.1  riastrad 	}
   2316        1.1  riastrad 	TAP_DEBUG_POST(sc);
   2317        1.1  riastrad 	mutex_exit(&sc->sc_tap_mutex);
   2318        1.1  riastrad 	return ok;
   2319        1.1  riastrad }
   2320   1.10.4.2     skrll 
   2321        1.1  riastrad /* Untapping: Releasing the button after a tap */
   2322        1.1  riastrad 
   2323        1.1  riastrad static void
   2324        1.1  riastrad schedule_untap(struct uatp_softc *sc)
   2325        1.1  riastrad {
   2326        1.1  riastrad 	unsigned int ms = sc->sc_knobs.double_tap_limit_msec;
   2327        1.1  riastrad 	if (ms <= 1000)
   2328        1.1  riastrad 		callout_schedule(&sc->sc_untap_callout, mstohz(ms));
   2329        1.1  riastrad 	else			/* XXX Reject bogus values in sysctl.  */
   2330        1.1  riastrad 		aprint_error_dev(uatp_dev(sc),
   2331        1.1  riastrad 		    "double-tap delay too long: %ums\n", ms);
   2332        1.1  riastrad }
   2333        1.1  riastrad 
   2334        1.1  riastrad static void
   2335        1.1  riastrad untap_callout(void *arg)
   2336        1.1  riastrad {
   2337        1.1  riastrad 	struct uatp_softc *sc = arg;
   2338        1.1  riastrad 
   2339        1.1  riastrad 	mutex_enter(&sc->sc_tap_mutex);
   2340        1.1  riastrad 	TAP_DEBUG_PRE(sc);
   2341        1.1  riastrad 	switch (sc->sc_tap_state) {
   2342        1.1  riastrad 	case TAP_STATE_TAPPED:
   2343        1.1  riastrad 		tap_transition_initial(sc);
   2344        1.1  riastrad 		/*
   2345        1.1  riastrad 		 * XXX Kludge: Call uatp_input after the state transition
   2346        1.1  riastrad 		 * to make sure that it will actually release the button.
   2347        1.1  riastrad 		 */
   2348        1.1  riastrad 		uatp_input(sc, 0, 0, 0, 0, 0);
   2349        1.1  riastrad 
   2350        1.1  riastrad 	case TAP_STATE_INITIAL:
   2351        1.1  riastrad 	case TAP_STATE_TAPPING:
   2352        1.1  riastrad 	case TAP_STATE_DOUBLE_TAPPING:
   2353        1.1  riastrad 	case TAP_STATE_DRAGGING_UP:
   2354        1.1  riastrad 	case TAP_STATE_DRAGGING_DOWN:
   2355        1.1  riastrad 	case TAP_STATE_TAPPING_IN_DRAG:
   2356        1.1  riastrad 		/*
   2357        1.1  riastrad 		 * Somebody else got in and changed the state before we
   2358        1.1  riastrad 		 * untapped.  Let them take over; do nothing here.
   2359        1.1  riastrad 		 */
   2360        1.1  riastrad 		break;
   2361        1.1  riastrad 
   2362        1.1  riastrad 	default:
   2363        1.1  riastrad 		aprint_error_dev(uatp_dev(sc), "%s: invalid tap state: %d\n",
   2364        1.1  riastrad 		    __func__, sc->sc_tap_state);
   2365        1.1  riastrad 		tap_transition_initial(sc);
   2366        1.1  riastrad 		/* XXX Just in case...?  */
   2367        1.1  riastrad 		uatp_input(sc, 0, 0, 0, 0, 0);
   2368        1.1  riastrad 		break;
   2369        1.1  riastrad 	}
   2370        1.1  riastrad 	TAP_DEBUG_POST(sc);
   2371        1.1  riastrad 	mutex_exit(&sc->sc_tap_mutex);
   2372        1.1  riastrad }
   2373   1.10.4.2     skrll 
   2374        1.1  riastrad /*
   2375        1.1  riastrad  * Emulate different buttons if the user holds down n fingers while
   2376        1.1  riastrad  * pressing the physical button.  (This is unrelated to tapping.)
   2377        1.1  riastrad  */
   2378        1.1  riastrad 
   2379        1.1  riastrad static uint32_t
   2380        1.1  riastrad emulated_buttons(struct uatp_softc *sc, unsigned int fingers)
   2381        1.1  riastrad {
   2382        1.1  riastrad 	CHECK((1 < fingers), return 0);
   2383        1.1  riastrad 
   2384        1.1  riastrad 	switch (fingers) {
   2385        1.1  riastrad 	case 2:
   2386        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_EMUL_BUTTON,
   2387        1.1  riastrad 		    ("2-finger emulated button: %"PRIx32"\n",
   2388        1.1  riastrad 			sc->sc_knobs.two_finger_buttons));
   2389        1.1  riastrad 		return sc->sc_knobs.two_finger_buttons;
   2390        1.1  riastrad 
   2391        1.1  riastrad 	case 3:
   2392        1.1  riastrad 	default:
   2393        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_EMUL_BUTTON,
   2394        1.1  riastrad 		    ("3-finger emulated button: %"PRIx32"\n",
   2395        1.1  riastrad 			sc->sc_knobs.three_finger_buttons));
   2396        1.1  riastrad 		return sc->sc_knobs.three_finger_buttons;
   2397        1.1  riastrad 	}
   2398        1.1  riastrad }
   2399   1.10.4.2     skrll 
   2400        1.1  riastrad /*
   2401        1.1  riastrad  * Update the position known to the driver based on the position and
   2402        1.1  riastrad  * number of fingers.  dx, dy, dz, and dw are expected to hold zero;
   2403        1.1  riastrad  * update_position may store nonzero changes in position in them.
   2404        1.1  riastrad  */
   2405        1.1  riastrad 
   2406        1.1  riastrad static void
   2407        1.1  riastrad update_position(struct uatp_softc *sc, unsigned int fingers,
   2408        1.1  riastrad     unsigned int x_raw, unsigned int y_raw,
   2409        1.1  riastrad     int *dx, int *dy, int *dz, int *dw)
   2410        1.1  riastrad {
   2411        1.1  riastrad 	CHECK((0 < fingers), return);
   2412        1.1  riastrad 
   2413        1.1  riastrad 	if ((fingers == 1) || (sc->sc_knobs.multifinger_track == 1))
   2414        1.1  riastrad 		move_mouse(sc, x_raw, y_raw, dx, dy);
   2415        1.1  riastrad 	else if (sc->sc_knobs.multifinger_track == 2)
   2416        1.1  riastrad 		scroll_wheel(sc, x_raw, y_raw, dz, dw);
   2417        1.1  riastrad }
   2418        1.1  riastrad 
   2419        1.1  riastrad /*
   2420        1.1  riastrad  * XXX Scrolling needs to use a totally different motion model.
   2421        1.1  riastrad  */
   2422        1.1  riastrad 
   2423        1.1  riastrad static void
   2424        1.1  riastrad move_mouse(struct uatp_softc *sc, unsigned int x_raw, unsigned int y_raw,
   2425        1.1  riastrad     int *dx, int *dy)
   2426        1.1  riastrad {
   2427        1.1  riastrad 	move(sc, "mouse", x_raw, y_raw, &sc->sc_x_raw, &sc->sc_y_raw,
   2428        1.1  riastrad 	    &sc->sc_x_smoothed, &sc->sc_y_smoothed,
   2429        1.1  riastrad 	    &sc->sc_x_remainder, &sc->sc_y_remainder,
   2430        1.1  riastrad 	    dx, dy);
   2431        1.1  riastrad }
   2432        1.1  riastrad 
   2433        1.1  riastrad static void
   2434        1.1  riastrad scroll_wheel(struct uatp_softc *sc, unsigned int x_raw, unsigned int y_raw,
   2435        1.1  riastrad     int *dz, int *dw)
   2436        1.1  riastrad {
   2437        1.1  riastrad 	move(sc, "scroll", x_raw, y_raw, &sc->sc_z_raw, &sc->sc_w_raw,
   2438        1.1  riastrad 	    &sc->sc_z_smoothed, &sc->sc_w_smoothed,
   2439        1.1  riastrad 	    &sc->sc_z_remainder, &sc->sc_w_remainder,
   2440        1.1  riastrad 	    dz, dw);
   2441        1.1  riastrad }
   2442   1.10.4.2     skrll 
   2443        1.1  riastrad static void
   2444        1.1  riastrad move(struct uatp_softc *sc, const char *ctx, unsigned int a, unsigned int b,
   2445        1.1  riastrad     int *a_raw, int *b_raw,
   2446        1.1  riastrad     int *a_smoothed, int *b_smoothed,
   2447        1.1  riastrad     unsigned int *a_remainder, unsigned int *b_remainder,
   2448        1.1  riastrad     int *da, int *db)
   2449        1.1  riastrad {
   2450        1.1  riastrad #define CHECK_(condition) CHECK(condition, return)
   2451        1.1  riastrad 
   2452        1.1  riastrad 	int old_a_raw = *a_raw, old_a_smoothed = *a_smoothed;
   2453        1.1  riastrad 	int old_b_raw = *b_raw, old_b_smoothed = *b_smoothed;
   2454        1.1  riastrad 	unsigned int a_dist, b_dist, dist_squared;
   2455        1.1  riastrad 	bool a_fast, b_fast;
   2456        1.1  riastrad 
   2457        1.1  riastrad 	/*
   2458        1.1  riastrad 	 * Make sure the quadratics in motion_below_threshold and
   2459        1.1  riastrad 	 * tracking distance don't overflow int arithmetic.
   2460        1.1  riastrad 	 */
   2461        1.1  riastrad 	__CTASSERT(0x12000000 == (2 * UATP_MAX_POSITION * UATP_MAX_POSITION));
   2462        1.1  riastrad 
   2463        1.1  riastrad 	CHECK_(a <= UATP_MAX_POSITION);
   2464        1.1  riastrad 	CHECK_(b <= UATP_MAX_POSITION);
   2465        1.1  riastrad 	*a_raw = a;
   2466        1.1  riastrad 	*b_raw = b;
   2467        1.1  riastrad 	if ((old_a_raw < 0) || (old_b_raw < 0)) {
   2468        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_MOVE,
   2469        1.1  riastrad 		    ("initialize %s position (%d, %d) -> (%d, %d)\n", ctx,
   2470        1.1  riastrad 			old_a_raw, old_b_raw, a, b));
   2471        1.1  riastrad 		return;
   2472        1.1  riastrad 	}
   2473        1.1  riastrad 
   2474        1.1  riastrad 	if ((old_a_smoothed < 0) || (old_b_smoothed < 0)) {
   2475        1.1  riastrad 		/* XXX Does this make sense?  */
   2476        1.1  riastrad 		old_a_smoothed = old_a_raw;
   2477        1.1  riastrad 		old_b_smoothed = old_b_raw;
   2478        1.1  riastrad 	}
   2479        1.1  riastrad 
   2480        1.1  riastrad 	CHECK_(0 <= old_a_raw);
   2481        1.1  riastrad 	CHECK_(0 <= old_b_raw);
   2482        1.1  riastrad 	CHECK_(old_a_raw <= UATP_MAX_POSITION);
   2483        1.1  riastrad 	CHECK_(old_b_raw <= UATP_MAX_POSITION);
   2484        1.1  riastrad 	CHECK_(0 <= old_a_smoothed);
   2485        1.1  riastrad 	CHECK_(0 <= old_b_smoothed);
   2486        1.1  riastrad 	CHECK_(old_a_smoothed <= UATP_MAX_POSITION);
   2487        1.1  riastrad 	CHECK_(old_b_smoothed <= UATP_MAX_POSITION);
   2488        1.1  riastrad 	CHECK_(0 <= *a_raw);
   2489        1.1  riastrad 	CHECK_(0 <= *b_raw);
   2490        1.1  riastrad 	CHECK_(*a_raw <= UATP_MAX_POSITION);
   2491        1.1  riastrad 	CHECK_(*b_raw <= UATP_MAX_POSITION);
   2492        1.1  riastrad 	*a_smoothed = smooth(sc, old_a_raw, old_a_smoothed, *a_raw);
   2493        1.1  riastrad 	*b_smoothed = smooth(sc, old_b_raw, old_b_smoothed, *b_raw);
   2494        1.1  riastrad 	CHECK_(0 <= *a_smoothed);
   2495        1.1  riastrad 	CHECK_(0 <= *b_smoothed);
   2496        1.1  riastrad 	CHECK_(*a_smoothed <= UATP_MAX_POSITION);
   2497        1.1  riastrad 	CHECK_(*b_smoothed <= UATP_MAX_POSITION);
   2498   1.10.4.2     skrll 
   2499        1.1  riastrad 	if (sc->sc_motion_timer < sc->sc_knobs.motion_delay) {
   2500        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_MOVE, ("delay motion %u\n",
   2501        1.1  riastrad 			sc->sc_motion_timer));
   2502        1.1  riastrad 		sc->sc_motion_timer += 1;
   2503        1.1  riastrad 		return;
   2504        1.1  riastrad 	}
   2505        1.1  riastrad 
   2506        1.1  riastrad 	/* XXX Use raw distances or smoothed distances?  Acceleration?  */
   2507        1.1  riastrad 	if (*a_smoothed < old_a_smoothed)
   2508        1.1  riastrad 		a_dist = old_a_smoothed - *a_smoothed;
   2509        1.1  riastrad 	else
   2510        1.1  riastrad 		a_dist = *a_smoothed - old_a_smoothed;
   2511        1.1  riastrad 
   2512        1.1  riastrad 	if (*b_smoothed < old_b_smoothed)
   2513        1.1  riastrad 		b_dist = old_b_smoothed - *b_smoothed;
   2514        1.1  riastrad 	else
   2515        1.1  riastrad 		b_dist = *b_smoothed - old_b_smoothed;
   2516        1.1  riastrad 
   2517        1.1  riastrad 	dist_squared = (a_dist * a_dist) + (b_dist * b_dist);
   2518        1.1  riastrad 	if (dist_squared < ((2 * UATP_MAX_POSITION * UATP_MAX_POSITION)
   2519        1.1  riastrad 		- sc->sc_track_distance))
   2520        1.1  riastrad 		sc->sc_track_distance += dist_squared;
   2521        1.1  riastrad 	else
   2522        1.1  riastrad 		sc->sc_track_distance = (2 * UATP_MAX_POSITION *
   2523        1.1  riastrad 		    UATP_MAX_POSITION);
   2524        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_TRACK_DIST, ("finger has tracked %u units^2\n",
   2525        1.1  riastrad 		sc->sc_track_distance));
   2526        1.1  riastrad 
   2527        1.1  riastrad 	/*
   2528        1.1  riastrad 	 * The checks above guarantee that the differences here are at
   2529        1.1  riastrad 	 * most UATP_MAX_POSITION in magnitude, since both minuend and
   2530        1.1  riastrad 	 * subtrahend are nonnegative and at most UATP_MAX_POSITION.
   2531        1.1  riastrad 	 */
   2532        1.1  riastrad 	if (motion_below_threshold(sc, sc->sc_knobs.motion_threshold,
   2533        1.1  riastrad 		(int)(*a_smoothed - old_a_smoothed),
   2534        1.1  riastrad 		(int)(*b_smoothed - old_b_smoothed))) {
   2535        1.1  riastrad 		DPRINTF(sc, UATP_DEBUG_MOVE,
   2536        1.1  riastrad 		    ("%s motion too small: (%d, %d) -> (%d, %d)\n", ctx,
   2537        1.1  riastrad 			old_a_smoothed, old_b_smoothed,
   2538        1.1  riastrad 			*a_smoothed, *b_smoothed));
   2539        1.1  riastrad 		return;
   2540        1.1  riastrad 	}
   2541        1.1  riastrad 	if (sc->sc_knobs.fast_per_direction == 0) {
   2542        1.1  riastrad 		a_fast = b_fast = !motion_below_threshold(sc,
   2543        1.1  riastrad 		    sc->sc_knobs.fast_motion_threshold,
   2544        1.1  riastrad 		    (int)(*a_smoothed - old_a_smoothed),
   2545        1.1  riastrad 		    (int)(*b_smoothed - old_b_smoothed));
   2546        1.1  riastrad 	} else {
   2547        1.1  riastrad 		a_fast = !motion_below_threshold(sc,
   2548        1.1  riastrad 		    sc->sc_knobs.fast_motion_threshold,
   2549        1.1  riastrad 		    (int)(*a_smoothed - old_a_smoothed),
   2550        1.1  riastrad 		    0);
   2551        1.1  riastrad 		b_fast = !motion_below_threshold(sc,
   2552        1.1  riastrad 		    sc->sc_knobs.fast_motion_threshold,
   2553        1.1  riastrad 		    0,
   2554        1.1  riastrad 		    (int)(*b_smoothed - old_b_smoothed));
   2555        1.1  riastrad 	}
   2556        1.1  riastrad 	*da = accelerate(sc, old_a_raw, *a_raw, old_a_smoothed, *a_smoothed,
   2557        1.1  riastrad 	    a_fast, a_remainder);
   2558        1.1  riastrad 	*db = accelerate(sc, old_b_raw, *b_raw, old_b_smoothed, *b_smoothed,
   2559        1.1  riastrad 	    b_fast, b_remainder);
   2560        1.1  riastrad 	DPRINTF(sc, UATP_DEBUG_MOVE,
   2561        1.1  riastrad 	    ("update %s position (%d, %d) -> (%d, %d), move by (%d, %d)\n",
   2562        1.1  riastrad 		ctx, old_a_smoothed, old_b_smoothed, *a_smoothed, *b_smoothed,
   2563        1.1  riastrad 		*da, *db));
   2564        1.1  riastrad 
   2565        1.1  riastrad #undef CHECK_
   2566        1.1  riastrad }
   2567   1.10.4.2     skrll 
   2568        1.1  riastrad static int
   2569        1.1  riastrad smooth(struct uatp_softc *sc, unsigned int old_raw, unsigned int old_smoothed,
   2570        1.1  riastrad     unsigned int raw)
   2571        1.1  riastrad {
   2572        1.1  riastrad #define CHECK_(condition) CHECK(condition, return old_raw)
   2573        1.1  riastrad 
   2574        1.1  riastrad 	/*
   2575        1.1  riastrad 	 * Arithmetic bounds:
   2576        1.1  riastrad 	 * . the weights are at most UATP_MAX_WEIGHT;
   2577        1.1  riastrad 	 * . the positions are at most UATP_MAX_POSITION; and so
   2578        1.1  riastrad 	 * . the numerator of the average is at most
   2579        1.1  riastrad 	 *     3 * UATP_MAX_WEIGHT * UATP_MAX_POSITION,
   2580        1.1  riastrad 	 *   which is #x477000, fitting comfortably in an int.
   2581        1.1  riastrad 	 */
   2582        1.1  riastrad 	__CTASSERT(0x477000 == (3 * UATP_MAX_WEIGHT * UATP_MAX_POSITION));
   2583        1.1  riastrad 	unsigned int old_raw_weight = uatp_old_raw_weight(sc);
   2584        1.1  riastrad 	unsigned int old_smoothed_weight = uatp_old_smoothed_weight(sc);
   2585        1.1  riastrad 	unsigned int new_raw_weight = uatp_new_raw_weight(sc);
   2586        1.1  riastrad 	CHECK_(old_raw_weight <= UATP_MAX_WEIGHT);
   2587        1.1  riastrad 	CHECK_(old_smoothed_weight <= UATP_MAX_WEIGHT);
   2588        1.1  riastrad 	CHECK_(new_raw_weight <= UATP_MAX_WEIGHT);
   2589        1.1  riastrad 	CHECK_(old_raw <= UATP_MAX_POSITION);
   2590        1.1  riastrad 	CHECK_(old_smoothed <= UATP_MAX_POSITION);
   2591        1.1  riastrad 	CHECK_(raw <= UATP_MAX_POSITION);
   2592        1.1  riastrad 	return (((old_raw_weight * old_raw) +
   2593        1.1  riastrad 		(old_smoothed_weight * old_smoothed) +
   2594        1.1  riastrad 		(new_raw_weight * raw))
   2595        1.1  riastrad 	    / (old_raw_weight + old_smoothed_weight + new_raw_weight));
   2596        1.1  riastrad 
   2597        1.1  riastrad #undef CHECK_
   2598        1.1  riastrad }
   2599        1.1  riastrad 
   2600        1.1  riastrad static bool
   2601        1.1  riastrad motion_below_threshold(struct uatp_softc *sc, unsigned int threshold,
   2602        1.1  riastrad     int x, int y)
   2603        1.1  riastrad {
   2604        1.1  riastrad 	unsigned int x_squared, y_squared;
   2605        1.1  riastrad 
   2606        1.1  riastrad 	/* Caller guarantees the multiplication will not overflow.  */
   2607        1.1  riastrad 	KASSERT(-UATP_MAX_POSITION <= x);
   2608        1.1  riastrad 	KASSERT(-UATP_MAX_POSITION <= y);
   2609        1.1  riastrad 	KASSERT(x <= UATP_MAX_POSITION);
   2610        1.1  riastrad 	KASSERT(y <= UATP_MAX_POSITION);
   2611        1.1  riastrad 	__CTASSERT(0x12000000 == (2 * UATP_MAX_POSITION * UATP_MAX_POSITION));
   2612        1.1  riastrad 
   2613        1.1  riastrad 	x_squared = (x * x);
   2614        1.1  riastrad 	y_squared = (y * y);
   2615        1.1  riastrad 
   2616   1.10.4.9     skrll 	return (x_squared + y_squared) < threshold;
   2617        1.1  riastrad }
   2618        1.1  riastrad 
   2619        1.1  riastrad static int
   2620        1.1  riastrad accelerate(struct uatp_softc *sc, unsigned int old_raw, unsigned int raw,
   2621        1.1  riastrad     unsigned int old_smoothed, unsigned int smoothed, bool fast,
   2622        1.1  riastrad     int *remainder)
   2623        1.1  riastrad {
   2624        1.1  riastrad #define CHECK_(condition) CHECK(condition, return 0)
   2625        1.1  riastrad 
   2626        1.1  riastrad 	/* Guarantee that the scaling won't overflow.  */
   2627        1.1  riastrad 	__CTASSERT(0x30000 ==
   2628        1.1  riastrad 	    (UATP_MAX_POSITION * UATP_MAX_MOTION_MULTIPLIER));
   2629        1.1  riastrad 
   2630        1.1  riastrad 	CHECK_(old_raw <= UATP_MAX_POSITION);
   2631        1.1  riastrad 	CHECK_(raw <= UATP_MAX_POSITION);
   2632        1.1  riastrad 	CHECK_(old_smoothed <= UATP_MAX_POSITION);
   2633        1.1  riastrad 	CHECK_(smoothed <= UATP_MAX_POSITION);
   2634        1.1  riastrad 
   2635        1.1  riastrad 	return (fast ? uatp_scale_fast_motion : uatp_scale_motion)
   2636        1.1  riastrad 	    (sc, (((int) smoothed) - ((int) old_smoothed)), remainder);
   2637        1.1  riastrad 
   2638        1.1  riastrad #undef CHECK_
   2639        1.1  riastrad }
   2640   1.10.4.2     skrll 
   2641        1.9  riastrad MODULE(MODULE_CLASS_DRIVER, uatp, NULL);
   2642        1.9  riastrad 
   2643        1.9  riastrad #ifdef _MODULE
   2644        1.9  riastrad #include "ioconf.c"
   2645        1.9  riastrad #endif
   2646        1.9  riastrad 
   2647        1.9  riastrad static int
   2648        1.9  riastrad uatp_modcmd(modcmd_t cmd, void *aux)
   2649        1.9  riastrad {
   2650        1.9  riastrad 	int error = 0;
   2651        1.9  riastrad 
   2652        1.9  riastrad 	switch (cmd) {
   2653        1.9  riastrad 	case MODULE_CMD_INIT:
   2654        1.9  riastrad #ifdef _MODULE
   2655        1.9  riastrad 		error = config_init_component(cfdriver_ioconf_uatp,
   2656        1.9  riastrad 		    cfattach_ioconf_uatp, cfdata_ioconf_uatp);
   2657        1.9  riastrad #endif
   2658        1.9  riastrad 		return error;
   2659        1.9  riastrad 	case MODULE_CMD_FINI:
   2660        1.9  riastrad #ifdef _MODULE
   2661        1.9  riastrad 		error = config_fini_component(cfdriver_ioconf_uatp,
   2662        1.9  riastrad 		    cfattach_ioconf_uatp, cfdata_ioconf_uatp);
   2663        1.9  riastrad #endif
   2664        1.9  riastrad 		return error;
   2665        1.9  riastrad 	default:
   2666        1.9  riastrad 		return ENOTTY;
   2667        1.9  riastrad 	}
   2668        1.9  riastrad }
   2669