1867 lines
45 KiB
C
1867 lines
45 KiB
C
/* $OpenBSD: wstpad.c,v 1.31 2022/06/09 22:17:18 bru Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2015, 2016 Ulf Brosziewski
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
/*
|
|
* touchpad input processing
|
|
*/
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/malloc.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/signalvar.h>
|
|
#include <sys/timeout.h>
|
|
|
|
#include <dev/wscons/wsconsio.h>
|
|
#include <dev/wscons/wsmousevar.h>
|
|
#include <dev/wscons/wseventvar.h>
|
|
#include <dev/wscons/wsmouseinput.h>
|
|
|
|
#define BTNMASK(n) ((n) > 0 && (n) <= 32 ? 1 << ((n) - 1) : 0)
|
|
|
|
#define LEFTBTN BTNMASK(1)
|
|
#define MIDDLEBTN BTNMASK(2)
|
|
#define RIGHTBTN BTNMASK(3)
|
|
|
|
#define PRIMARYBTN LEFTBTN
|
|
|
|
#define PRIMARYBTN_CLICKED(tp) ((tp)->btns_sync & PRIMARYBTN & (tp)->btns)
|
|
#define PRIMARYBTN_RELEASED(tp) ((tp)->btns_sync & PRIMARYBTN & ~(tp)->btns)
|
|
|
|
#define IS_MT(tp) ((tp)->features & WSTPAD_MT)
|
|
#define DISABLE(tp) ((tp)->features & WSTPAD_DISABLE)
|
|
|
|
/*
|
|
* Ratios to the height or width of the touchpad surface, in
|
|
* [*.12] fixed-point format:
|
|
*/
|
|
#define V_EDGE_RATIO_DEFAULT 205
|
|
#define B_EDGE_RATIO_DEFAULT 410
|
|
#define T_EDGE_RATIO_DEFAULT 512
|
|
#define CENTER_RATIO_DEFAULT 512
|
|
|
|
#define TAP_MAXTIME_DEFAULT 180
|
|
#define TAP_CLICKTIME_DEFAULT 180
|
|
#define TAP_LOCKTIME_DEFAULT 0
|
|
#define TAP_BTNMAP_SIZE 3
|
|
|
|
#define CLICKDELAY_MS 20
|
|
#define FREEZE_MS 100
|
|
#define MATCHINTERVAL_MS 45
|
|
#define STOPINTERVAL_MS 55
|
|
|
|
#define MAG_LOW (10 << 12)
|
|
#define MAG_MEDIUM (18 << 12)
|
|
|
|
enum tpad_handlers {
|
|
SOFTBUTTON_HDLR,
|
|
TOPBUTTON_HDLR,
|
|
TAP_HDLR,
|
|
F2SCROLL_HDLR,
|
|
EDGESCROLL_HDLR,
|
|
CLICK_HDLR,
|
|
};
|
|
|
|
enum tap_state {
|
|
TAP_DETECT,
|
|
TAP_IGNORE,
|
|
TAP_LIFTED,
|
|
TAP_LOCKED,
|
|
TAP_LOCKED_DRAG,
|
|
};
|
|
|
|
enum tpad_cmd {
|
|
CLEAR_MOTION_DELTAS,
|
|
SOFTBUTTON_DOWN,
|
|
SOFTBUTTON_UP,
|
|
TAPBUTTON_SYNC,
|
|
TAPBUTTON_DOWN,
|
|
TAPBUTTON_UP,
|
|
VSCROLL,
|
|
HSCROLL,
|
|
};
|
|
|
|
/*
|
|
* tpad_touch.flags:
|
|
*/
|
|
#define L_EDGE (1 << 0)
|
|
#define R_EDGE (1 << 1)
|
|
#define T_EDGE (1 << 2)
|
|
#define B_EDGE (1 << 3)
|
|
#define THUMB (1 << 4)
|
|
|
|
#define EDGES (L_EDGE | R_EDGE | T_EDGE | B_EDGE)
|
|
|
|
/*
|
|
* A touch is "centered" if it does not start and remain at the top
|
|
* edge or one of the vertical edges. Two-finger scrolling and tapping
|
|
* require that at least one touch is centered.
|
|
*/
|
|
#define CENTERED(t) (((t)->flags & (L_EDGE | R_EDGE | T_EDGE)) == 0)
|
|
|
|
enum touchstates {
|
|
TOUCH_NONE,
|
|
TOUCH_BEGIN,
|
|
TOUCH_UPDATE,
|
|
TOUCH_END,
|
|
};
|
|
|
|
struct tpad_touch {
|
|
u_int flags;
|
|
enum touchstates state;
|
|
int x;
|
|
int y;
|
|
int dir;
|
|
struct timespec start;
|
|
struct timespec match;
|
|
struct position *pos;
|
|
struct {
|
|
int x;
|
|
int y;
|
|
struct timespec time;
|
|
} orig;
|
|
};
|
|
|
|
/*
|
|
* wstpad.features
|
|
*/
|
|
#define WSTPAD_SOFTBUTTONS (1 << 0)
|
|
#define WSTPAD_SOFTMBTN (1 << 1)
|
|
#define WSTPAD_TOPBUTTONS (1 << 2)
|
|
#define WSTPAD_TWOFINGERSCROLL (1 << 3)
|
|
#define WSTPAD_EDGESCROLL (1 << 4)
|
|
#define WSTPAD_HORIZSCROLL (1 << 5)
|
|
#define WSTPAD_SWAPSIDES (1 << 6)
|
|
#define WSTPAD_DISABLE (1 << 7)
|
|
|
|
#define WSTPAD_MT (1 << 31)
|
|
|
|
|
|
struct wstpad {
|
|
u_int features;
|
|
u_int handlers;
|
|
|
|
/*
|
|
* t always points into the tpad_touches array, which has at
|
|
* least one element. If there is more than one, t selects
|
|
* the pointer-controlling touch.
|
|
*/
|
|
struct tpad_touch *t;
|
|
struct tpad_touch *tpad_touches;
|
|
|
|
u_int mtcycle;
|
|
u_int ignore;
|
|
|
|
int contacts;
|
|
int prev_contacts;
|
|
u_int btns;
|
|
u_int btns_sync;
|
|
int ratio;
|
|
|
|
struct timespec time;
|
|
|
|
u_int freeze;
|
|
struct timespec freeze_ts;
|
|
|
|
/* edge coordinates */
|
|
struct {
|
|
int left;
|
|
int right;
|
|
int top;
|
|
int bottom;
|
|
int center;
|
|
int center_left;
|
|
int center_right;
|
|
int low;
|
|
} edge;
|
|
|
|
struct {
|
|
/* ratios to the surface width or height */
|
|
int left_edge;
|
|
int right_edge;
|
|
int top_edge;
|
|
int bottom_edge;
|
|
int center_width;
|
|
/* two-finger contacts */
|
|
int f2pressure;
|
|
int f2width;
|
|
} params;
|
|
|
|
/* handler state and configuration: */
|
|
|
|
u_int softbutton;
|
|
u_int sbtnswap;
|
|
|
|
struct {
|
|
enum tap_state state;
|
|
int contacts;
|
|
int valid;
|
|
u_int pending;
|
|
u_int button;
|
|
int masked;
|
|
int maxdist;
|
|
struct timeout to;
|
|
/* parameters: */
|
|
struct timespec maxtime;
|
|
int clicktime;
|
|
int locktime;
|
|
u_int btnmap[TAP_BTNMAP_SIZE];
|
|
} tap;
|
|
|
|
struct {
|
|
int dz;
|
|
int dw;
|
|
int hdist;
|
|
int vdist;
|
|
int mag;
|
|
} scroll;
|
|
};
|
|
|
|
static const struct timespec match_interval =
|
|
{ .tv_sec = 0, .tv_nsec = MATCHINTERVAL_MS * 1000000 };
|
|
|
|
static const struct timespec stop_interval =
|
|
{ .tv_sec = 0, .tv_nsec = STOPINTERVAL_MS * 1000000 };
|
|
|
|
/*
|
|
* Coordinates in the wstpad struct are "normalized" device coordinates,
|
|
* the orientation is left-to-right and upward.
|
|
*/
|
|
static inline int
|
|
normalize_abs(struct axis_filter *filter, int val)
|
|
{
|
|
return (filter->inv ? filter->inv - val : val);
|
|
}
|
|
|
|
static inline int
|
|
normalize_rel(struct axis_filter *filter, int val)
|
|
{
|
|
return (filter->inv ? -val : val);
|
|
}
|
|
|
|
/*
|
|
* Directions of motion are represented by numbers in the range 0 - 11,
|
|
* corresponding to clockwise counted circle sectors:
|
|
*
|
|
* 11 | 0
|
|
* 10 | 1
|
|
* 9 | 2
|
|
* -------+-------
|
|
* 8 | 3
|
|
* 7 | 4
|
|
* 6 | 5
|
|
*
|
|
*/
|
|
/* Tangent constants in [*.12] fixed-point format: */
|
|
#define TAN_DEG_60 7094
|
|
#define TAN_DEG_30 2365
|
|
|
|
#define NORTH(d) ((d) == 0 || (d) == 11)
|
|
#define SOUTH(d) ((d) == 5 || (d) == 6)
|
|
#define EAST(d) ((d) == 2 || (d) == 3)
|
|
#define WEST(d) ((d) == 8 || (d) == 9)
|
|
|
|
static inline int
|
|
direction(int dx, int dy, int ratio)
|
|
{
|
|
int rdy, dir = -1;
|
|
|
|
if (dx || dy) {
|
|
rdy = abs(dy) * ratio;
|
|
if (abs(dx) * TAN_DEG_60 < rdy)
|
|
dir = 0;
|
|
else if (abs(dx) * TAN_DEG_30 < rdy)
|
|
dir = 1;
|
|
else
|
|
dir = 2;
|
|
if ((dx < 0) != (dy < 0))
|
|
dir = 5 - dir;
|
|
if (dx < 0)
|
|
dir += 6;
|
|
}
|
|
return dir;
|
|
}
|
|
|
|
static inline int
|
|
dircmp(int dir1, int dir2)
|
|
{
|
|
int diff = abs(dir1 - dir2);
|
|
return (diff <= 6 ? diff : 12 - diff);
|
|
}
|
|
|
|
/*
|
|
* Update direction and timespec attributes for a touch. They are used to
|
|
* determine whether it is moving - or resting - stably.
|
|
*
|
|
* The callers pass touches from the current frame and the touches that are
|
|
* no longer present in the update cycle to this function. Even though this
|
|
* ensures that pairs of zero deltas do not result from stale coordinates,
|
|
* zero deltas do not reset the state immediately. A short time span - the
|
|
* "stop interval" - must pass before the state is cleared, which is
|
|
* necessary because some touchpads report intermediate stops when a touch
|
|
* is moving very slowly.
|
|
*/
|
|
void
|
|
wstpad_set_direction(struct wstpad *tp, struct tpad_touch *t, int dx, int dy)
|
|
{
|
|
int dir;
|
|
struct timespec ts;
|
|
|
|
if (t->state != TOUCH_UPDATE) {
|
|
t->dir = -1;
|
|
memcpy(&t->start, &tp->time, sizeof(struct timespec));
|
|
return;
|
|
}
|
|
|
|
dir = direction(dx, dy, tp->ratio);
|
|
if (dir >= 0) {
|
|
if (t->dir < 0 || dircmp(dir, t->dir) > 1) {
|
|
memcpy(&t->start, &tp->time, sizeof(struct timespec));
|
|
}
|
|
t->dir = dir;
|
|
memcpy(&t->match, &tp->time, sizeof(struct timespec));
|
|
} else if (t->dir >= 0) {
|
|
timespecsub(&tp->time, &t->match, &ts);
|
|
if (timespeccmp(&ts, &stop_interval, >=)) {
|
|
t->dir = -1;
|
|
memcpy(&t->start, &t->match, sizeof(struct timespec));
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Make a rough, but quick estimation of the speed of a touch. Its
|
|
* distance to the previous position is scaled by factors derived
|
|
* from the average update rate and the deceleration parameter
|
|
* (filter.dclr). The unit of the result is:
|
|
* (filter.dclr / 100) device units per millisecond
|
|
*
|
|
* Magnitudes are returned in [*.12] fixed-point format. For purposes
|
|
* of filtering, they are divided into medium and high speeds
|
|
* (> MAG_MEDIUM), low speeds, and very low speeds (< MAG_LOW).
|
|
*
|
|
* The scale factors are not affected if deceleration is turned off.
|
|
*/
|
|
static inline int
|
|
magnitude(struct wsmouseinput *input, int dx, int dy)
|
|
{
|
|
int h, v;
|
|
|
|
h = abs(dx) * input->filter.h.mag_scale;
|
|
v = abs(dy) * input->filter.v.mag_scale;
|
|
/* Return an "alpha-max-plus-beta-min" approximation: */
|
|
return (h >= v ? h + 3 * v / 8 : v + 3 * h / 8);
|
|
}
|
|
|
|
/*
|
|
* Treat a touch as stable if it is moving at a medium or high speed,
|
|
* if it is moving continuously, or if it has stopped for a certain
|
|
* time span.
|
|
*/
|
|
int
|
|
wstpad_is_stable(struct wsmouseinput *input, struct tpad_touch *t)
|
|
{
|
|
struct timespec ts;
|
|
|
|
if (t->dir >= 0) {
|
|
if (magnitude(input, t->pos->dx, t->pos->dy) > MAG_MEDIUM)
|
|
return (1);
|
|
timespecsub(&t->match, &t->start, &ts);
|
|
} else {
|
|
timespecsub(&input->tp->time, &t->start, &ts);
|
|
}
|
|
|
|
return (timespeccmp(&ts, &match_interval, >=));
|
|
}
|
|
|
|
/*
|
|
* If a touch starts in an edge area, pointer movement will be
|
|
* suppressed as long as it stays in that area.
|
|
*/
|
|
static inline u_int
|
|
edge_flags(struct wstpad *tp, int x, int y)
|
|
{
|
|
u_int flags = 0;
|
|
|
|
if (x < tp->edge.left)
|
|
flags |= L_EDGE;
|
|
else if (x >= tp->edge.right)
|
|
flags |= R_EDGE;
|
|
if (y < tp->edge.bottom)
|
|
flags |= B_EDGE;
|
|
else if (y >= tp->edge.top)
|
|
flags |= T_EDGE;
|
|
|
|
return (flags);
|
|
}
|
|
|
|
static inline struct tpad_touch *
|
|
get_2nd_touch(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
int slot;
|
|
|
|
if (IS_MT(tp)) {
|
|
slot = ffs(input->mt.touches & ~(input->mt.ptr | tp->ignore));
|
|
if (slot)
|
|
return &tp->tpad_touches[--slot];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* Suppress pointer motion for a short period of time. */
|
|
static inline void
|
|
set_freeze_ts(struct wstpad *tp, int sec, int ms)
|
|
{
|
|
tp->freeze_ts.tv_sec = sec;
|
|
tp->freeze_ts.tv_nsec = ms * 1000000;
|
|
timespecadd(&tp->time, &tp->freeze_ts, &tp->freeze_ts);
|
|
}
|
|
|
|
|
|
/* Return TRUE if two-finger- or edge-scrolling would be valid. */
|
|
int
|
|
wstpad_scroll_coords(struct wsmouseinput *input, int *dx, int *dy)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
|
|
if (tp->contacts != tp->prev_contacts || tp->btns || tp->btns_sync) {
|
|
tp->scroll.dz = 0;
|
|
tp->scroll.dw = 0;
|
|
return (0);
|
|
}
|
|
if ((input->motion.sync & SYNC_POSITION) == 0)
|
|
return (0);
|
|
/*
|
|
* Try to exclude accidental scroll events by checking whether the
|
|
* pointer-controlling touch is stable. The check, which may cause
|
|
* a short delay, is only applied initially, a touch that stops and
|
|
* resumes scrolling is not affected.
|
|
*/
|
|
if (tp->scroll.dz || tp->scroll.dw || wstpad_is_stable(input, tp->t)) {
|
|
*dx = normalize_rel(&input->filter.h, input->motion.pos.dx);
|
|
*dy = normalize_rel(&input->filter.v, input->motion.pos.dy);
|
|
return (*dx || *dy);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
void
|
|
wstpad_scroll(struct wstpad *tp, int dx, int dy, int mag, u_int *cmds)
|
|
{
|
|
int dz, dw, n = 1;
|
|
|
|
/*
|
|
* The function applies strong deceleration, but only to input with
|
|
* very low speeds. A higher threshold might make applications
|
|
* without support for precision scrolling appear unresponsive.
|
|
*/
|
|
mag = tp->scroll.mag = imin(MAG_MEDIUM,
|
|
(mag + 3 * tp->scroll.mag) / 4);
|
|
if (mag < MAG_LOW)
|
|
n = (MAG_LOW - mag) / 4096 + 1;
|
|
|
|
if (dy && tp->scroll.vdist) {
|
|
if (tp->scroll.dw) {
|
|
/*
|
|
* Before switching the axis, wstpad_scroll_coords()
|
|
* should check again whether the movement is stable.
|
|
*/
|
|
tp->scroll.dw = 0;
|
|
return;
|
|
}
|
|
dz = -dy * 4096 / (tp->scroll.vdist * n);
|
|
if (tp->scroll.dz) {
|
|
if ((dy < 0) != (tp->scroll.dz > 0))
|
|
tp->scroll.dz = -tp->scroll.dz;
|
|
dz = (dz + 3 * tp->scroll.dz) / 4;
|
|
}
|
|
if (dz) {
|
|
tp->scroll.dz = dz;
|
|
*cmds |= 1 << VSCROLL;
|
|
}
|
|
|
|
} else if (dx && tp->scroll.hdist) {
|
|
if (tp->scroll.dz) {
|
|
tp->scroll.dz = 0;
|
|
return;
|
|
}
|
|
dw = dx * 4096 / (tp->scroll.hdist * n);
|
|
if (tp->scroll.dw) {
|
|
if ((dx > 0) != (tp->scroll.dw > 0))
|
|
tp->scroll.dw = -tp->scroll.dw;
|
|
dw = (dw + 3 * tp->scroll.dw) / 4;
|
|
}
|
|
if (dw) {
|
|
tp->scroll.dw = dw;
|
|
*cmds |= 1 << HSCROLL;
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
wstpad_f2scroll(struct wsmouseinput *input, u_int *cmds)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *t2;
|
|
int dir, dx, dy, centered;
|
|
|
|
if (tp->ignore == 0) {
|
|
if (tp->contacts != 2)
|
|
return;
|
|
} else if (tp->contacts != 3 || (tp->ignore == input->mt.ptr)) {
|
|
return;
|
|
}
|
|
|
|
if (!wstpad_scroll_coords(input, &dx, &dy))
|
|
return;
|
|
|
|
dir = tp->t->dir;
|
|
if (!(NORTH(dir) || SOUTH(dir)))
|
|
dy = 0;
|
|
if (!(EAST(dir) || WEST(dir)))
|
|
dx = 0;
|
|
|
|
if (dx || dy) {
|
|
centered = CENTERED(tp->t);
|
|
if (IS_MT(tp)) {
|
|
t2 = get_2nd_touch(input);
|
|
if (t2 == NULL)
|
|
return;
|
|
dir = t2->dir;
|
|
if ((dy > 0 && !NORTH(dir)) || (dy < 0 && !SOUTH(dir)))
|
|
return;
|
|
if ((dx > 0 && !EAST(dir)) || (dx < 0 && !WEST(dir)))
|
|
return;
|
|
if (!wstpad_is_stable(input, t2) &&
|
|
!(tp->scroll.dz || tp->scroll.dw))
|
|
return;
|
|
centered |= CENTERED(t2);
|
|
}
|
|
if (centered) {
|
|
wstpad_scroll(tp, dx, dy,
|
|
magnitude(input, dx, dy), cmds);
|
|
set_freeze_ts(tp, 0, FREEZE_MS);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
wstpad_edgescroll(struct wsmouseinput *input, u_int *cmds)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *t = tp->t;
|
|
u_int v_edge, b_edge;
|
|
int dx, dy;
|
|
|
|
if (!wstpad_scroll_coords(input, &dx, &dy) || tp->contacts != 1)
|
|
return;
|
|
|
|
v_edge = (tp->features & WSTPAD_SWAPSIDES) ? L_EDGE : R_EDGE;
|
|
b_edge = (tp->features & WSTPAD_HORIZSCROLL) ? B_EDGE : 0;
|
|
|
|
if ((t->flags & v_edge) == 0)
|
|
dy = 0;
|
|
if ((t->flags & b_edge) == 0)
|
|
dx = 0;
|
|
|
|
if (dx || dy)
|
|
wstpad_scroll(tp, dx, dy, magnitude(input, dx, dy), cmds);
|
|
}
|
|
|
|
static inline u_int
|
|
sbtn(struct wstpad *tp, int x, int y)
|
|
{
|
|
if (y >= tp->edge.bottom)
|
|
return (0);
|
|
if ((tp->features & WSTPAD_SOFTMBTN)
|
|
&& x >= tp->edge.center_left
|
|
&& x < tp->edge.center_right)
|
|
return (MIDDLEBTN);
|
|
return ((x < tp->edge.center ? LEFTBTN : RIGHTBTN) ^ tp->sbtnswap);
|
|
}
|
|
|
|
static inline u_int
|
|
top_sbtn(struct wstpad *tp, int x, int y)
|
|
{
|
|
if (y < tp->edge.top)
|
|
return (0);
|
|
if (x < tp->edge.center_left)
|
|
return (LEFTBTN ^ tp->sbtnswap);
|
|
return (x > tp->edge.center_right
|
|
? (RIGHTBTN ^ tp->sbtnswap) : MIDDLEBTN);
|
|
}
|
|
|
|
u_int
|
|
wstpad_get_sbtn(struct wsmouseinput *input, int top)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *t = tp->t;
|
|
u_int btn;
|
|
|
|
btn = 0;
|
|
if (tp->contacts) {
|
|
btn = top ? top_sbtn(tp, t->x, t->y) : sbtn(tp, t->x, t->y);
|
|
/*
|
|
* If there is no middle-button area, but contacts in both
|
|
* halves of the edge zone, generate a middle-button event:
|
|
*/
|
|
if (btn && IS_MT(tp) && tp->contacts == 2
|
|
&& !top && !(tp->features & WSTPAD_SOFTMBTN)) {
|
|
if ((t = get_2nd_touch(input)) != NULL)
|
|
btn |= sbtn(tp, t->x, t->y);
|
|
if (btn == (LEFTBTN | RIGHTBTN))
|
|
btn = MIDDLEBTN;
|
|
}
|
|
}
|
|
return (btn != PRIMARYBTN ? btn : 0);
|
|
}
|
|
|
|
void
|
|
wstpad_softbuttons(struct wsmouseinput *input, u_int *cmds, int hdlr)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
int top = (hdlr == TOPBUTTON_HDLR);
|
|
|
|
if (tp->softbutton && PRIMARYBTN_RELEASED(tp)) {
|
|
*cmds |= 1 << SOFTBUTTON_UP;
|
|
return;
|
|
}
|
|
|
|
if (tp->softbutton == 0 && PRIMARYBTN_CLICKED(tp)) {
|
|
tp->softbutton = wstpad_get_sbtn(input, top);
|
|
if (tp->softbutton)
|
|
*cmds |= 1 << SOFTBUTTON_DOWN;
|
|
}
|
|
}
|
|
|
|
/* Check whether the duration of t is within the tap limit. */
|
|
int
|
|
wstpad_is_tap(struct wstpad *tp, struct tpad_touch *t)
|
|
{
|
|
struct timespec ts;
|
|
|
|
timespecsub(&tp->time, &t->orig.time, &ts);
|
|
return (timespeccmp(&ts, &tp->tap.maxtime, <));
|
|
}
|
|
|
|
/*
|
|
* At least one MT touch must remain close to its origin and end
|
|
* in the main area. The same conditions apply to one-finger taps
|
|
* on single-touch devices.
|
|
*/
|
|
void
|
|
wstpad_tap_filter(struct wstpad *tp, struct tpad_touch *t)
|
|
{
|
|
int dx, dy, dist = 0;
|
|
|
|
if (IS_MT(tp) || tp->tap.contacts == 1) {
|
|
dx = abs(t->x - t->orig.x) << 12;
|
|
dy = abs(t->y - t->orig.y) * tp->ratio;
|
|
dist = (dx >= dy ? dx + 3 * dy / 8 : dy + 3 * dx / 8);
|
|
}
|
|
tp->tap.valid = (CENTERED(t) && dist <= (tp->tap.maxdist << 12));
|
|
}
|
|
|
|
|
|
/*
|
|
* Return the oldest touch in the TOUCH_END state, or NULL.
|
|
*/
|
|
struct tpad_touch *
|
|
wstpad_tap_touch(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *s, *t = NULL;
|
|
u_int lifted;
|
|
int slot;
|
|
|
|
if (IS_MT(tp)) {
|
|
lifted = (input->mt.sync[MTS_TOUCH] & ~input->mt.touches);
|
|
FOREACHBIT(lifted, slot) {
|
|
s = &tp->tpad_touches[slot];
|
|
if (tp->tap.state == TAP_DETECT && !tp->tap.valid)
|
|
wstpad_tap_filter(tp, s);
|
|
if (t == NULL || timespeccmp(&t->orig.time,
|
|
&s->orig.time, >))
|
|
t = s;
|
|
}
|
|
} else {
|
|
if (tp->t->state == TOUCH_END) {
|
|
t = tp->t;
|
|
if (tp->tap.state == TAP_DETECT && !tp->tap.valid)
|
|
wstpad_tap_filter(tp, t);
|
|
}
|
|
}
|
|
|
|
return (t);
|
|
}
|
|
|
|
/* Determine the "tap button", keep track of whether a touch is masked. */
|
|
u_int
|
|
wstpad_tap_button(struct wstpad *tp)
|
|
{
|
|
int n = tp->tap.contacts - tp->contacts - 1;
|
|
|
|
tp->tap.masked = tp->contacts;
|
|
|
|
return (n >= 0 && n < TAP_BTNMAP_SIZE ? tp->tap.btnmap[n] : 0);
|
|
}
|
|
|
|
/*
|
|
* In the hold/drag state, do not mask touches if no masking was involved
|
|
* in the preceding tap gesture.
|
|
*/
|
|
static inline int
|
|
tap_unmask(struct wstpad *tp)
|
|
{
|
|
return ((tp->tap.button || tp->tap.pending) && tp->tap.masked == 0);
|
|
}
|
|
|
|
/*
|
|
* In the default configuration, this handler maps one-, two-, and
|
|
* three-finger taps to left-button, right-button, and middle-button
|
|
* events, respectively. Setting the LOCKTIME parameter enables
|
|
* "locked drags", which are finished by a timeout or a tap-to-end
|
|
* gesture.
|
|
*/
|
|
void
|
|
wstpad_tap(struct wsmouseinput *input, u_int *cmds)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *t;
|
|
int contacts, is_tap, slot, err = 0;
|
|
|
|
/* Synchronize the button states, if necessary. */
|
|
if (input->btn.sync)
|
|
*cmds |= 1 << TAPBUTTON_SYNC;
|
|
|
|
/*
|
|
* It is possible to produce a click within the tap timeout.
|
|
* Wait for a new touch before generating new button events.
|
|
*/
|
|
if (PRIMARYBTN_RELEASED(tp))
|
|
tp->tap.contacts = 0;
|
|
|
|
/* Reset the detection state whenever a new touch starts. */
|
|
if (tp->contacts > tp->prev_contacts || (IS_MT(tp) &&
|
|
(input->mt.touches & input->mt.sync[MTS_TOUCH]))) {
|
|
tp->tap.contacts = tp->contacts;
|
|
tp->tap.valid = 0;
|
|
}
|
|
|
|
/*
|
|
* The filtered number of active touches excludes a masked
|
|
* touch if its duration exceeds the tap limit.
|
|
*/
|
|
contacts = tp->contacts;
|
|
if ((slot = ffs(input->mt.ptr_mask) - 1) >= 0
|
|
&& !wstpad_is_tap(tp, &tp->tpad_touches[slot])
|
|
&& !tap_unmask(tp)) {
|
|
contacts--;
|
|
}
|
|
|
|
switch (tp->tap.state) {
|
|
case TAP_DETECT:
|
|
/* Find the oldest touch in the TOUCH_END state. */
|
|
t = wstpad_tap_touch(input);
|
|
if (t) {
|
|
is_tap = wstpad_is_tap(tp, t);
|
|
if (is_tap && contacts == 0) {
|
|
if (tp->tap.button)
|
|
*cmds |= 1 << TAPBUTTON_UP;
|
|
tp->tap.pending = (tp->tap.valid
|
|
? wstpad_tap_button(tp) : 0);
|
|
if (tp->tap.pending) {
|
|
tp->tap.state = TAP_LIFTED;
|
|
err = !timeout_add_msec(&tp->tap.to,
|
|
CLICKDELAY_MS);
|
|
}
|
|
} else if (!is_tap && tp->tap.locktime == 0) {
|
|
if (contacts == 0 && tp->tap.button)
|
|
*cmds |= 1 << TAPBUTTON_UP;
|
|
else if (contacts)
|
|
tp->tap.state = TAP_IGNORE;
|
|
} else if (!is_tap && tp->tap.button) {
|
|
if (contacts == 0) {
|
|
tp->tap.state = TAP_LOCKED;
|
|
err = !timeout_add_msec(&tp->tap.to,
|
|
tp->tap.locktime);
|
|
} else {
|
|
tp->tap.state = TAP_LOCKED_DRAG;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case TAP_IGNORE:
|
|
if (contacts == 0) {
|
|
tp->tap.state = TAP_DETECT;
|
|
if (tp->tap.button)
|
|
*cmds |= 1 << TAPBUTTON_UP;
|
|
}
|
|
break;
|
|
case TAP_LIFTED:
|
|
if (contacts) {
|
|
timeout_del(&tp->tap.to);
|
|
tp->tap.state = TAP_DETECT;
|
|
if (tp->tap.pending)
|
|
*cmds |= 1 << TAPBUTTON_DOWN;
|
|
}
|
|
break;
|
|
case TAP_LOCKED:
|
|
if (contacts) {
|
|
timeout_del(&tp->tap.to);
|
|
tp->tap.state = TAP_LOCKED_DRAG;
|
|
}
|
|
break;
|
|
case TAP_LOCKED_DRAG:
|
|
if (contacts == 0) {
|
|
t = wstpad_tap_touch(input);
|
|
if (t && wstpad_is_tap(tp, t)) {
|
|
/* "tap-to-end" */
|
|
*cmds |= 1 << TAPBUTTON_UP;
|
|
tp->tap.state = TAP_DETECT;
|
|
} else {
|
|
tp->tap.state = TAP_LOCKED;
|
|
err = !timeout_add_msec(&tp->tap.to,
|
|
tp->tap.locktime);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (err) { /* Did timeout_add fail? */
|
|
input->sbtn.buttons &= ~tp->tap.button;
|
|
input->sbtn.sync |= tp->tap.button;
|
|
tp->tap.pending = 0;
|
|
tp->tap.button = 0;
|
|
tp->tap.state = TAP_DETECT;
|
|
}
|
|
}
|
|
|
|
int
|
|
wstpad_tap_sync(struct wsmouseinput *input) {
|
|
struct wstpad *tp = input->tp;
|
|
|
|
return ((tp->tap.button & (input->btn.buttons | tp->softbutton)) == 0
|
|
|| (tp->tap.button == PRIMARYBTN && tp->softbutton));
|
|
}
|
|
|
|
void
|
|
wstpad_tap_timeout(void *p)
|
|
{
|
|
struct wsmouseinput *input = p;
|
|
struct wstpad *tp = input->tp;
|
|
struct evq_access evq;
|
|
u_int btn;
|
|
int s, ev;
|
|
|
|
s = spltty();
|
|
evq.evar = *input->evar;
|
|
if (evq.evar != NULL && tp != NULL) {
|
|
ev = 0;
|
|
if (tp->tap.pending) {
|
|
tp->tap.button = tp->tap.pending;
|
|
tp->tap.pending = 0;
|
|
input->sbtn.buttons |= tp->tap.button;
|
|
timeout_add_msec(&tp->tap.to, tp->tap.clicktime);
|
|
if (wstpad_tap_sync(input)) {
|
|
ev = BTN_DOWN_EV;
|
|
btn = ffs(tp->tap.button) - 1;
|
|
}
|
|
} else {
|
|
if (wstpad_tap_sync(input)) {
|
|
ev = BTN_UP_EV;
|
|
btn = ffs(tp->tap.button) - 1;
|
|
}
|
|
if (tp->tap.button != tp->softbutton)
|
|
input->sbtn.buttons &= ~tp->tap.button;
|
|
tp->tap.button = 0;
|
|
tp->tap.state = TAP_DETECT;
|
|
}
|
|
if (ev) {
|
|
evq.put = evq.evar->put;
|
|
evq.result = EVQ_RESULT_NONE;
|
|
getnanotime(&evq.ts);
|
|
wsmouse_evq_put(&evq, ev, btn);
|
|
wsmouse_evq_put(&evq, SYNC_EV, 0);
|
|
if (evq.result == EVQ_RESULT_SUCCESS) {
|
|
if (input->flags & LOG_EVENTS) {
|
|
wsmouse_log_events(input, &evq);
|
|
}
|
|
evq.evar->put = evq.put;
|
|
WSEVENT_WAKEUP(evq.evar);
|
|
} else {
|
|
input->sbtn.sync |= tp->tap.button;
|
|
}
|
|
}
|
|
}
|
|
splx(s);
|
|
}
|
|
|
|
/*
|
|
* Suppress accidental pointer movements after a click on a clickpad.
|
|
*/
|
|
void
|
|
wstpad_click(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
|
|
if (tp->contacts == 1 &&
|
|
(PRIMARYBTN_CLICKED(tp) || PRIMARYBTN_RELEASED(tp)))
|
|
set_freeze_ts(tp, 0, FREEZE_MS);
|
|
}
|
|
|
|
/* Translate the "command" bits into the sync-state of wsmouse. */
|
|
void
|
|
wstpad_cmds(struct wsmouseinput *input, u_int cmds)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
int n;
|
|
|
|
FOREACHBIT(cmds, n) {
|
|
switch (n) {
|
|
case CLEAR_MOTION_DELTAS:
|
|
input->motion.dx = input->motion.dy = 0;
|
|
if (input->motion.dz == 0 && input->motion.dw == 0)
|
|
input->motion.sync &= ~SYNC_DELTAS;
|
|
continue;
|
|
case SOFTBUTTON_DOWN:
|
|
input->btn.sync &= ~PRIMARYBTN;
|
|
input->sbtn.buttons |= tp->softbutton;
|
|
if (tp->softbutton != tp->tap.button)
|
|
input->sbtn.sync |= tp->softbutton;
|
|
continue;
|
|
case SOFTBUTTON_UP:
|
|
input->btn.sync &= ~PRIMARYBTN;
|
|
if (tp->softbutton != tp->tap.button) {
|
|
input->sbtn.buttons &= ~tp->softbutton;
|
|
input->sbtn.sync |= tp->softbutton;
|
|
}
|
|
tp->softbutton = 0;
|
|
continue;
|
|
case TAPBUTTON_SYNC:
|
|
if (tp->tap.button)
|
|
input->btn.sync &= ~tp->tap.button;
|
|
continue;
|
|
case TAPBUTTON_DOWN:
|
|
tp->tap.button = tp->tap.pending;
|
|
tp->tap.pending = 0;
|
|
input->sbtn.buttons |= tp->tap.button;
|
|
if (wstpad_tap_sync(input))
|
|
input->sbtn.sync |= tp->tap.button;
|
|
continue;
|
|
case TAPBUTTON_UP:
|
|
if (tp->tap.button != tp->softbutton)
|
|
input->sbtn.buttons &= ~tp->tap.button;
|
|
if (wstpad_tap_sync(input))
|
|
input->sbtn.sync |= tp->tap.button;
|
|
tp->tap.button = 0;
|
|
continue;
|
|
case HSCROLL:
|
|
input->motion.dw = tp->scroll.dw;
|
|
input->motion.sync |= SYNC_DELTAS;
|
|
continue;
|
|
case VSCROLL:
|
|
input->motion.dz = tp->scroll.dz;
|
|
input->motion.sync |= SYNC_DELTAS;
|
|
continue;
|
|
default:
|
|
printf("[wstpad] invalid cmd %d\n", n);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Set the state of touches that have ended. TOUCH_END is a transitional
|
|
* state and will be changed to TOUCH_NONE before process_input() returns.
|
|
*/
|
|
static inline void
|
|
clear_touchstates(struct wsmouseinput *input, enum touchstates state)
|
|
{
|
|
u_int touches;
|
|
int slot;
|
|
|
|
touches = input->mt.sync[MTS_TOUCH] & ~input->mt.touches;
|
|
FOREACHBIT(touches, slot)
|
|
input->tp->tpad_touches[slot].state = state;
|
|
}
|
|
|
|
void
|
|
wstpad_mt_inputs(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *t;
|
|
int slot, dx, dy;
|
|
u_int touches, inactive;
|
|
|
|
/* TOUCH_BEGIN */
|
|
touches = input->mt.touches & input->mt.sync[MTS_TOUCH];
|
|
FOREACHBIT(touches, slot) {
|
|
t = &tp->tpad_touches[slot];
|
|
t->state = TOUCH_BEGIN;
|
|
t->x = normalize_abs(&input->filter.h, t->pos->x);
|
|
t->y = normalize_abs(&input->filter.v, t->pos->y);
|
|
t->orig.x = t->x;
|
|
t->orig.y = t->y;
|
|
memcpy(&t->orig.time, &tp->time, sizeof(struct timespec));
|
|
t->flags = edge_flags(tp, t->x, t->y);
|
|
wstpad_set_direction(tp, t, 0, 0);
|
|
}
|
|
|
|
/* TOUCH_UPDATE */
|
|
touches = input->mt.touches & input->mt.frame;
|
|
if (touches & tp->mtcycle) {
|
|
/*
|
|
* Slot data may be synchronized separately, in any order,
|
|
* or not at all if there is no delta. Identify the touches
|
|
* without deltas.
|
|
*/
|
|
inactive = input->mt.touches & ~tp->mtcycle;
|
|
tp->mtcycle = touches;
|
|
} else {
|
|
inactive = 0;
|
|
tp->mtcycle |= touches;
|
|
}
|
|
touches = input->mt.touches & ~input->mt.sync[MTS_TOUCH];
|
|
FOREACHBIT(touches, slot) {
|
|
t = &tp->tpad_touches[slot];
|
|
t->state = TOUCH_UPDATE;
|
|
if ((1 << slot) & input->mt.frame) {
|
|
dx = normalize_abs(&input->filter.h, t->pos->x) - t->x;
|
|
t->x += dx;
|
|
dy = normalize_abs(&input->filter.v, t->pos->y) - t->y;
|
|
t->y += dy;
|
|
t->flags &= (~EDGES | edge_flags(tp, t->x, t->y));
|
|
if (wsmouse_hysteresis(input, t->pos))
|
|
dx = dy = 0;
|
|
wstpad_set_direction(tp, t, dx, dy);
|
|
} else if ((1 << slot) & inactive) {
|
|
wstpad_set_direction(tp, t, 0, 0);
|
|
}
|
|
}
|
|
|
|
clear_touchstates(input, TOUCH_END);
|
|
}
|
|
|
|
/*
|
|
* Identify "thumb" contacts in the bottom area. The identification
|
|
* has three stages:
|
|
* 1. If exactly one of two or more touches is in the bottom area, it
|
|
* is masked, which means it does not receive pointer control as long
|
|
* as there are alternatives. Once set, the mask will only be cleared
|
|
* when the touch is released.
|
|
* Tap detection ignores a masked touch if it does not participate in
|
|
* a tap gesture.
|
|
* 2. If the pointer-controlling touch is moving stably while a masked
|
|
* touch in the bottom area is resting, or only moving minimally, the
|
|
* pointer mask is copied to tp->ignore. In this stage, the masked
|
|
* touch does not block pointer movement, and it is ignored by
|
|
* wstpad_f2scroll().
|
|
* Decisions are made more or less immediately, there may be errors
|
|
* in edge cases. If a fast or long upward movement is detected,
|
|
* tp->ignore is cleared. There is no other transition from stage 2
|
|
* to scrolling, or vice versa, for a pair of touches.
|
|
* 3. If tp->ignore is set and the touch is resting, it is marked as
|
|
* thumb, and it will be ignored until it ends.
|
|
*/
|
|
void
|
|
wstpad_mt_masks(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *t;
|
|
struct position *pos;
|
|
u_int mask;
|
|
int slot;
|
|
|
|
tp->ignore &= input->mt.touches;
|
|
|
|
if (tp->contacts < 2)
|
|
return;
|
|
|
|
if (tp->ignore) {
|
|
slot = ffs(tp->ignore) - 1;
|
|
t = &tp->tpad_touches[slot];
|
|
if (t->flags & THUMB)
|
|
return;
|
|
if (t->dir < 0 && wstpad_is_stable(input, t)) {
|
|
t->flags |= THUMB;
|
|
return;
|
|
}
|
|
/* The edge.low area is a bit larger than the bottom area. */
|
|
if (t->y >= tp->edge.low || (NORTH(t->dir) &&
|
|
magnitude(input, t->pos->dx, t->pos->dy) >= MAG_MEDIUM))
|
|
tp->ignore = 0;
|
|
return;
|
|
}
|
|
|
|
if (input->mt.ptr_mask == 0) {
|
|
mask = ~0;
|
|
FOREACHBIT(input->mt.touches, slot) {
|
|
t = &tp->tpad_touches[slot];
|
|
if (t->flags & B_EDGE) {
|
|
mask &= (1 << slot);
|
|
input->mt.ptr_mask = mask;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((input->mt.ptr_mask & ~input->mt.ptr)
|
|
&& !(tp->scroll.dz || tp->scroll.dw)
|
|
&& tp->t->dir >= 0
|
|
&& wstpad_is_stable(input, tp->t)) {
|
|
|
|
slot = ffs(input->mt.ptr_mask) - 1;
|
|
t = &tp->tpad_touches[slot];
|
|
|
|
if (t->y >= tp->edge.low)
|
|
return;
|
|
|
|
if (!wstpad_is_stable(input, t))
|
|
return;
|
|
|
|
/* Default hysteresis limits are low. Make a strict check. */
|
|
pos = tp->t->pos;
|
|
if (abs(pos->acc_dx) < 3 * input->filter.h.hysteresis
|
|
&& abs(pos->acc_dy) < 3 * input->filter.v.hysteresis)
|
|
return;
|
|
|
|
if (t->dir >= 0) {
|
|
/* Treat t as thumb if it is slow while tp->t is fast. */
|
|
if (magnitude(input, t->pos->dx, t->pos->dy) > MAG_LOW
|
|
|| magnitude(input, pos->dx, pos->dy) < MAG_MEDIUM)
|
|
return;
|
|
}
|
|
|
|
tp->ignore = input->mt.ptr_mask;
|
|
}
|
|
}
|
|
|
|
void
|
|
wstpad_touch_inputs(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
struct tpad_touch *t;
|
|
int slot, x, y, dx, dy;
|
|
|
|
tp->btns = input->btn.buttons;
|
|
tp->btns_sync = input->btn.sync;
|
|
|
|
tp->prev_contacts = tp->contacts;
|
|
tp->contacts = input->touch.contacts;
|
|
|
|
if (tp->contacts == 1 &&
|
|
((tp->params.f2width &&
|
|
input->touch.width >= tp->params.f2width)
|
|
|| (tp->params.f2pressure &&
|
|
input->touch.pressure >= tp->params.f2pressure)))
|
|
tp->contacts = 2;
|
|
|
|
if (IS_MT(tp)) {
|
|
wstpad_mt_inputs(input);
|
|
if (input->mt.ptr) {
|
|
slot = ffs(input->mt.ptr) - 1;
|
|
tp->t = &tp->tpad_touches[slot];
|
|
}
|
|
wstpad_mt_masks(input);
|
|
} else {
|
|
t = tp->t;
|
|
if (tp->contacts)
|
|
t->state = (tp->prev_contacts ?
|
|
TOUCH_UPDATE : TOUCH_BEGIN);
|
|
else
|
|
t->state = (tp->prev_contacts ?
|
|
TOUCH_END : TOUCH_NONE);
|
|
|
|
dx = dy = 0;
|
|
x = normalize_abs(&input->filter.h, t->pos->x);
|
|
y = normalize_abs(&input->filter.v, t->pos->y);
|
|
if (t->state == TOUCH_BEGIN) {
|
|
t->x = t->orig.x = x;
|
|
t->y = t->orig.y = y;
|
|
memcpy(&t->orig.time, &tp->time,
|
|
sizeof(struct timespec));
|
|
t->flags = edge_flags(tp, x, y);
|
|
} else if (input->motion.sync & SYNC_POSITION) {
|
|
if (!wsmouse_hysteresis(input, t->pos)) {
|
|
dx = x - t->x;
|
|
dy = y - t->y;
|
|
}
|
|
t->x = x;
|
|
t->y = y;
|
|
t->flags &= (~EDGES | edge_flags(tp, x, y));
|
|
}
|
|
wstpad_set_direction(tp, t, dx, dy);
|
|
}
|
|
}
|
|
|
|
static inline int
|
|
t2_ignore(struct wsmouseinput *input)
|
|
{
|
|
/*
|
|
* If there are two touches, do not block pointer movement if they
|
|
* perform a click-and-drag action, or if the second touch is
|
|
* resting in the bottom area.
|
|
*/
|
|
return (input->tp->contacts == 2 && ((input->tp->btns & PRIMARYBTN)
|
|
|| (input->tp->ignore & ~input->mt.ptr)));
|
|
}
|
|
|
|
void
|
|
wstpad_process_input(struct wsmouseinput *input, struct evq_access *evq)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
u_int handlers, hdlr, cmds;
|
|
|
|
memcpy(&tp->time, &evq->ts, sizeof(struct timespec));
|
|
wstpad_touch_inputs(input);
|
|
|
|
cmds = 0;
|
|
handlers = tp->handlers;
|
|
if (DISABLE(tp))
|
|
handlers &= ((1 << TOPBUTTON_HDLR) | (1 << SOFTBUTTON_HDLR));
|
|
|
|
FOREACHBIT(handlers, hdlr) {
|
|
switch (hdlr) {
|
|
case SOFTBUTTON_HDLR:
|
|
case TOPBUTTON_HDLR:
|
|
wstpad_softbuttons(input, &cmds, hdlr);
|
|
continue;
|
|
case TAP_HDLR:
|
|
wstpad_tap(input, &cmds);
|
|
continue;
|
|
case F2SCROLL_HDLR:
|
|
wstpad_f2scroll(input, &cmds);
|
|
continue;
|
|
case EDGESCROLL_HDLR:
|
|
wstpad_edgescroll(input, &cmds);
|
|
continue;
|
|
case CLICK_HDLR:
|
|
wstpad_click(input);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* Check whether pointer movement should be blocked. */
|
|
if (input->motion.dx || input->motion.dy) {
|
|
if (DISABLE(tp)
|
|
|| (tp->t->flags & tp->freeze)
|
|
|| timespeccmp(&tp->time, &tp->freeze_ts, <)
|
|
|| (tp->contacts > 1 && !t2_ignore(input))) {
|
|
|
|
cmds |= 1 << CLEAR_MOTION_DELTAS;
|
|
}
|
|
}
|
|
|
|
wstpad_cmds(input, cmds);
|
|
|
|
if (IS_MT(tp))
|
|
clear_touchstates(input, TOUCH_NONE);
|
|
}
|
|
|
|
/*
|
|
* Try to determine the average interval between two updates. Various
|
|
* conditions are checked in order to ensure that only valid samples enter
|
|
* into the calculation. Above all, it is restricted to motion events
|
|
* occurring when there is only one contact. MT devices may need more than
|
|
* one packet to transmit their state if there are multiple touches, and
|
|
* the update frequency may be higher in this case.
|
|
*/
|
|
void
|
|
wstpad_track_interval(struct wsmouseinput *input, struct timespec *time)
|
|
{
|
|
static const struct timespec limit = { 0, 30 * 1000000L };
|
|
struct timespec ts;
|
|
int samples;
|
|
|
|
if (input->motion.sync == 0
|
|
|| (input->touch.sync & SYNC_CONTACTS)
|
|
|| (input->touch.contacts > 1)) {
|
|
input->intv.track = 0;
|
|
return;
|
|
}
|
|
if (input->intv.track) {
|
|
timespecsub(time, &input->intv.ts, &ts);
|
|
if (timespeccmp(&ts, &limit, <)) {
|
|
/* The unit of the sum is 4096 nanoseconds. */
|
|
input->intv.sum += ts.tv_nsec >> 12;
|
|
samples = ++input->intv.samples;
|
|
/*
|
|
* Make the first calculation quickly and later
|
|
* a more reliable one:
|
|
*/
|
|
if (samples == 8) {
|
|
input->intv.avg = input->intv.sum << 9;
|
|
wstpad_init_deceleration(input);
|
|
} else if (samples == 128) {
|
|
input->intv.avg = input->intv.sum << 5;
|
|
wstpad_init_deceleration(input);
|
|
input->intv.samples = 0;
|
|
input->intv.sum = 0;
|
|
input->flags &= ~TRACK_INTERVAL;
|
|
}
|
|
}
|
|
}
|
|
memcpy(&input->intv.ts, time, sizeof(struct timespec));
|
|
input->intv.track = 1;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* The default acceleration options of X don't work convincingly with
|
|
* touchpads (the synaptics driver installs its own "acceleration
|
|
* profile" and callback function). As a preliminary workaround, this
|
|
* filter applies a simple deceleration scheme to small deltas, based
|
|
* on the "magnitude" of the delta pair. A magnitude of 8 corresponds,
|
|
* roughly, to a speed of (filter.dclr / 12.5) device units per milli-
|
|
* second. If its magnitude is smaller than 7 a delta will be downscaled
|
|
* by the factor 2/8, deltas with magnitudes from 7 to 11 by factors
|
|
* ranging from 3/8 to 7/8.
|
|
*/
|
|
int
|
|
wstpad_decelerate(struct wsmouseinput *input, int *dx, int *dy)
|
|
{
|
|
int mag, n, h, v;
|
|
|
|
mag = magnitude(input, *dx, *dy);
|
|
|
|
/* Don't change deceleration levels abruptly. */
|
|
mag = (mag + 7 * input->filter.mag) / 8;
|
|
/* Don't use arbitrarily high values. */
|
|
input->filter.mag = imin(mag, 24 << 12);
|
|
|
|
n = imax((mag >> 12) - 4, 2);
|
|
if (n < 8) {
|
|
/* Scale by (n / 8). */
|
|
h = *dx * n + input->filter.h.dclr_rmdr;
|
|
v = *dy * n + input->filter.v.dclr_rmdr;
|
|
input->filter.h.dclr_rmdr = (h >= 0 ? h & 7 : -(-h & 7));
|
|
input->filter.v.dclr_rmdr = (v >= 0 ? v & 7 : -(-v & 7));
|
|
*dx = h / 8;
|
|
*dy = v / 8;
|
|
return (1);
|
|
}
|
|
return (0);
|
|
}
|
|
|
|
void
|
|
wstpad_filter(struct wsmouseinput *input)
|
|
{
|
|
struct axis_filter *h = &input->filter.h;
|
|
struct axis_filter *v = &input->filter.v;
|
|
struct position *pos = &input->motion.pos;
|
|
int strength = input->filter.mode & 7;
|
|
int dx, dy;
|
|
|
|
if (!(input->motion.sync & SYNC_POSITION)
|
|
|| (h->dmax && (abs(pos->dx) > h->dmax))
|
|
|| (v->dmax && (abs(pos->dy) > v->dmax))) {
|
|
dx = dy = 0;
|
|
} else {
|
|
dx = pos->dx;
|
|
dy = pos->dy;
|
|
}
|
|
|
|
if (wsmouse_hysteresis(input, pos))
|
|
dx = dy = 0;
|
|
|
|
if (input->filter.dclr && wstpad_decelerate(input, &dx, &dy))
|
|
/* Strong smoothing may hamper the precision at low speeds. */
|
|
strength = imin(strength, 2);
|
|
|
|
if (strength) {
|
|
if ((input->touch.sync & SYNC_CONTACTS)
|
|
|| input->mt.ptr != input->mt.prev_ptr) {
|
|
h->avg = v->avg = 0;
|
|
}
|
|
/* Use a weighted decaying average for smoothing. */
|
|
dx = dx * (8 - strength) + h->avg * strength + h->avg_rmdr;
|
|
dy = dy * (8 - strength) + v->avg * strength + v->avg_rmdr;
|
|
h->avg_rmdr = (dx >= 0 ? dx & 7 : -(-dx & 7));
|
|
v->avg_rmdr = (dy >= 0 ? dy & 7 : -(-dy & 7));
|
|
dx = h->avg = dx / 8;
|
|
dy = v->avg = dy / 8;
|
|
}
|
|
|
|
input->motion.dx = dx;
|
|
input->motion.dy = dy;
|
|
}
|
|
|
|
|
|
/*
|
|
* Compatibility-mode conversions. wstpad_filter transforms and filters
|
|
* the coordinate inputs, extended functionality is provided by
|
|
* wstpad_process_input.
|
|
*/
|
|
void
|
|
wstpad_compat_convert(struct wsmouseinput *input, struct evq_access *evq)
|
|
{
|
|
if (input->flags & TRACK_INTERVAL)
|
|
wstpad_track_interval(input, &evq->ts);
|
|
|
|
wstpad_filter(input);
|
|
|
|
if ((input->motion.dx || input->motion.dy)
|
|
&& !(input->motion.sync & SYNC_DELTAS)) {
|
|
input->motion.dz = input->motion.dw = 0;
|
|
input->motion.sync |= SYNC_DELTAS;
|
|
}
|
|
|
|
if (input->tp != NULL)
|
|
wstpad_process_input(input, evq);
|
|
|
|
input->motion.sync &= ~SYNC_POSITION;
|
|
input->touch.sync = 0;
|
|
}
|
|
|
|
int
|
|
wstpad_init(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
int i, slots;
|
|
|
|
if (tp != NULL)
|
|
return (0);
|
|
|
|
input->tp = tp = malloc(sizeof(struct wstpad),
|
|
M_DEVBUF, M_WAITOK | M_ZERO);
|
|
if (tp == NULL)
|
|
return (-1);
|
|
|
|
slots = imax(input->mt.num_slots, 1);
|
|
tp->tpad_touches = malloc(slots * sizeof(struct tpad_touch),
|
|
M_DEVBUF, M_WAITOK | M_ZERO);
|
|
if (tp->tpad_touches == NULL) {
|
|
free(tp, M_DEVBUF, sizeof(struct wstpad));
|
|
return (-1);
|
|
}
|
|
|
|
tp->t = &tp->tpad_touches[0];
|
|
if (input->mt.num_slots) {
|
|
tp->features |= WSTPAD_MT;
|
|
for (i = 0; i < input->mt.num_slots; i++)
|
|
tp->tpad_touches[i].pos = &input->mt.slots[i].pos;
|
|
} else {
|
|
tp->t->pos = &input->motion.pos;
|
|
}
|
|
|
|
timeout_set(&tp->tap.to, wstpad_tap_timeout, input);
|
|
|
|
tp->ratio = input->filter.ratio;
|
|
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* Integer square root (Halleck's method)
|
|
*
|
|
* An adaption of code from John B. Halleck (from
|
|
* http://www.cc.utah.edu/~nahaj/factoring/code.html). This version is
|
|
* used and published under the OpenBSD license terms with his permission.
|
|
*
|
|
* Cf. also Martin Guy's "Square root by abacus" method.
|
|
*/
|
|
static inline u_int
|
|
isqrt(u_int n)
|
|
{
|
|
u_int root, sqbit;
|
|
|
|
root = 0;
|
|
sqbit = 1 << (sizeof(u_int) * 8 - 2);
|
|
while (sqbit) {
|
|
if (n >= (sqbit | root)) {
|
|
n -= (sqbit | root);
|
|
root = (root >> 1) | sqbit;
|
|
} else {
|
|
root >>= 1;
|
|
}
|
|
sqbit >>= 2;
|
|
}
|
|
return (root);
|
|
}
|
|
|
|
void
|
|
wstpad_init_deceleration(struct wsmouseinput *input)
|
|
{
|
|
int n, dclr;
|
|
|
|
if ((dclr = input->filter.dclr) == 0)
|
|
return;
|
|
|
|
dclr = imax(dclr, 4);
|
|
|
|
/*
|
|
* For a standard update rate of about 80Hz, (dclr) units
|
|
* will be mapped to a magnitude of 8. If the average rate
|
|
* is significantly higher or lower, adjust the coefficient
|
|
* accordingly:
|
|
*/
|
|
if (input->intv.avg == 0) {
|
|
n = 8;
|
|
} else {
|
|
n = 8 * 13000000 / input->intv.avg;
|
|
n = imax(imin(n, 32), 4);
|
|
}
|
|
input->filter.h.mag_scale = (n << 12) / dclr;
|
|
input->filter.v.mag_scale = (input->filter.ratio ?
|
|
n * input->filter.ratio : n << 12) / dclr;
|
|
input->filter.h.dclr_rmdr = 0;
|
|
input->filter.v.dclr_rmdr = 0;
|
|
input->flags |= TRACK_INTERVAL;
|
|
}
|
|
|
|
int
|
|
wstpad_configure(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp;
|
|
int width, height, diag, offset, h_res, v_res, h_unit, v_unit, i;
|
|
|
|
width = abs(input->hw.x_max - input->hw.x_min);
|
|
height = abs(input->hw.y_max - input->hw.y_min);
|
|
if (width == 0 || height == 0)
|
|
return (-1); /* We can't do anything. */
|
|
|
|
if (input->tp == NULL && wstpad_init(input))
|
|
return (-1);
|
|
tp = input->tp;
|
|
|
|
if (!(input->flags & CONFIGURED)) {
|
|
/*
|
|
* The filter parameters are derived from the length of the
|
|
* diagonal in device units, with some magic constants which
|
|
* are partly adapted from libinput or synaptics code, or are
|
|
* based on tests and guess work. The absolute resolution
|
|
* values might not be reliable, but if they are present the
|
|
* settings are adapted to the ratio.
|
|
*/
|
|
h_res = input->hw.h_res;
|
|
v_res = input->hw.v_res;
|
|
if (h_res == 0 || v_res == 0)
|
|
h_res = v_res = 1;
|
|
diag = isqrt(width * width + height * height);
|
|
input->filter.h.scale = (imin(920, diag) << 12) / diag;
|
|
input->filter.v.scale = input->filter.h.scale * h_res / v_res;
|
|
h_unit = imax(diag / 280, 3);
|
|
v_unit = imax((h_unit * v_res + h_res / 2) / h_res, 3);
|
|
input->filter.h.hysteresis = h_unit;
|
|
input->filter.v.hysteresis = v_unit;
|
|
input->filter.mode = FILTER_MODE_DEFAULT;
|
|
input->filter.dclr = h_unit - h_unit / 5;
|
|
wstpad_init_deceleration(input);
|
|
|
|
tp->features &= (WSTPAD_MT | WSTPAD_DISABLE);
|
|
|
|
if (input->hw.contacts_max != 1)
|
|
tp->features |= WSTPAD_TWOFINGERSCROLL;
|
|
else
|
|
tp->features |= WSTPAD_EDGESCROLL;
|
|
|
|
if (input->hw.hw_type == WSMOUSEHW_CLICKPAD) {
|
|
if (input->hw.type == WSMOUSE_TYPE_SYNAP_SBTN) {
|
|
tp->features |= WSTPAD_TOPBUTTONS;
|
|
} else {
|
|
tp->features |= WSTPAD_SOFTBUTTONS;
|
|
tp->features |= WSTPAD_SOFTMBTN;
|
|
}
|
|
}
|
|
|
|
tp->params.left_edge = V_EDGE_RATIO_DEFAULT;
|
|
tp->params.right_edge = V_EDGE_RATIO_DEFAULT;
|
|
tp->params.bottom_edge = ((tp->features & WSTPAD_SOFTBUTTONS)
|
|
? B_EDGE_RATIO_DEFAULT : 0);
|
|
tp->params.top_edge = ((tp->features & WSTPAD_TOPBUTTONS)
|
|
? T_EDGE_RATIO_DEFAULT : 0);
|
|
tp->params.center_width = CENTER_RATIO_DEFAULT;
|
|
|
|
tp->tap.maxtime.tv_nsec = TAP_MAXTIME_DEFAULT * 1000000;
|
|
tp->tap.clicktime = TAP_CLICKTIME_DEFAULT;
|
|
tp->tap.locktime = TAP_LOCKTIME_DEFAULT;
|
|
|
|
tp->scroll.hdist = 4 * h_unit;
|
|
tp->scroll.vdist = 4 * v_unit;
|
|
tp->tap.maxdist = 4 * h_unit;
|
|
}
|
|
|
|
/* A touch with a flag set in this mask does not move the pointer. */
|
|
tp->freeze = EDGES;
|
|
|
|
offset = width * tp->params.left_edge / 4096;
|
|
tp->edge.left = (offset ? input->hw.x_min + offset : INT_MIN);
|
|
offset = width * tp->params.right_edge / 4096;
|
|
tp->edge.right = (offset ? input->hw.x_max - offset : INT_MAX);
|
|
offset = height * tp->params.bottom_edge / 4096;
|
|
tp->edge.bottom = (offset ? input->hw.y_min + offset : INT_MIN);
|
|
tp->edge.low = tp->edge.bottom + offset / 2;
|
|
offset = height * tp->params.top_edge / 4096;
|
|
tp->edge.top = (offset ? input->hw.y_max - offset : INT_MAX);
|
|
|
|
offset = width * abs(tp->params.center_width) / 8192;
|
|
tp->edge.center = input->hw.x_min + width / 2;
|
|
tp->edge.center_left = tp->edge.center - offset;
|
|
tp->edge.center_right = tp->edge.center + offset;
|
|
|
|
tp->handlers = 0;
|
|
|
|
if (tp->features & WSTPAD_SOFTBUTTONS)
|
|
tp->handlers |= 1 << SOFTBUTTON_HDLR;
|
|
if (tp->features & WSTPAD_TOPBUTTONS)
|
|
tp->handlers |= 1 << TOPBUTTON_HDLR;
|
|
|
|
if (tp->features & WSTPAD_TWOFINGERSCROLL)
|
|
tp->handlers |= 1 << F2SCROLL_HDLR;
|
|
else if (tp->features & WSTPAD_EDGESCROLL)
|
|
tp->handlers |= 1 << EDGESCROLL_HDLR;
|
|
|
|
for (i = 0; i < TAP_BTNMAP_SIZE; i++) {
|
|
if (tp->tap.btnmap[i] == 0)
|
|
continue;
|
|
|
|
tp->tap.clicktime = imin(imax(tp->tap.clicktime, 80), 350);
|
|
if (tp->tap.locktime)
|
|
tp->tap.locktime =
|
|
imin(imax(tp->tap.locktime, 150), 5000);
|
|
tp->handlers |= 1 << TAP_HDLR;
|
|
break;
|
|
}
|
|
|
|
if (input->hw.hw_type == WSMOUSEHW_CLICKPAD)
|
|
tp->handlers |= 1 << CLICK_HDLR;
|
|
|
|
tp->sbtnswap = ((tp->features & WSTPAD_SWAPSIDES)
|
|
? (LEFTBTN | RIGHTBTN) : 0);
|
|
|
|
return (0);
|
|
}
|
|
|
|
void
|
|
wstpad_reset(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
|
|
if (tp != NULL) {
|
|
timeout_del(&tp->tap.to);
|
|
tp->tap.state = TAP_DETECT;
|
|
}
|
|
|
|
if (input->sbtn.buttons) {
|
|
input->sbtn.sync = input->sbtn.buttons;
|
|
input->sbtn.buttons = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
wstpad_cleanup(struct wsmouseinput *input)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
int slots;
|
|
|
|
timeout_del(&tp->tap.to);
|
|
slots = imax(input->mt.num_slots, 1);
|
|
free(tp->tpad_touches, M_DEVBUF, slots * sizeof(struct tpad_touch));
|
|
free(tp, M_DEVBUF, sizeof(struct wstpad));
|
|
input->tp = NULL;
|
|
}
|
|
|
|
int
|
|
wstpad_set_param(struct wsmouseinput *input, int key, int val)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
u_int flag;
|
|
|
|
if (tp == NULL)
|
|
return (EINVAL);
|
|
|
|
switch (key) {
|
|
case WSMOUSECFG_SOFTBUTTONS ... WSMOUSECFG_DISABLE:
|
|
switch (key) {
|
|
case WSMOUSECFG_SOFTBUTTONS:
|
|
flag = WSTPAD_SOFTBUTTONS;
|
|
break;
|
|
case WSMOUSECFG_SOFTMBTN:
|
|
flag = WSTPAD_SOFTMBTN;
|
|
break;
|
|
case WSMOUSECFG_TOPBUTTONS:
|
|
flag = WSTPAD_TOPBUTTONS;
|
|
break;
|
|
case WSMOUSECFG_TWOFINGERSCROLL:
|
|
flag = WSTPAD_TWOFINGERSCROLL;
|
|
break;
|
|
case WSMOUSECFG_EDGESCROLL:
|
|
flag = WSTPAD_EDGESCROLL;
|
|
break;
|
|
case WSMOUSECFG_HORIZSCROLL:
|
|
flag = WSTPAD_HORIZSCROLL;
|
|
break;
|
|
case WSMOUSECFG_SWAPSIDES:
|
|
flag = WSTPAD_SWAPSIDES;
|
|
break;
|
|
case WSMOUSECFG_DISABLE:
|
|
flag = WSTPAD_DISABLE;
|
|
break;
|
|
}
|
|
if (val)
|
|
tp->features |= flag;
|
|
else
|
|
tp->features &= ~flag;
|
|
break;
|
|
case WSMOUSECFG_LEFT_EDGE:
|
|
tp->params.left_edge = val;
|
|
break;
|
|
case WSMOUSECFG_RIGHT_EDGE:
|
|
tp->params.right_edge = val;
|
|
break;
|
|
case WSMOUSECFG_TOP_EDGE:
|
|
tp->params.top_edge = val;
|
|
break;
|
|
case WSMOUSECFG_BOTTOM_EDGE:
|
|
tp->params.bottom_edge = val;
|
|
break;
|
|
case WSMOUSECFG_CENTERWIDTH:
|
|
tp->params.center_width = val;
|
|
break;
|
|
case WSMOUSECFG_HORIZSCROLLDIST:
|
|
tp->scroll.hdist = val;
|
|
break;
|
|
case WSMOUSECFG_VERTSCROLLDIST:
|
|
tp->scroll.vdist = val;
|
|
break;
|
|
case WSMOUSECFG_F2WIDTH:
|
|
tp->params.f2width = val;
|
|
break;
|
|
case WSMOUSECFG_F2PRESSURE:
|
|
tp->params.f2pressure = val;
|
|
break;
|
|
case WSMOUSECFG_TAP_MAXTIME:
|
|
tp->tap.maxtime.tv_nsec = imin(val, 999) * 1000000;
|
|
break;
|
|
case WSMOUSECFG_TAP_CLICKTIME:
|
|
tp->tap.clicktime = val;
|
|
break;
|
|
case WSMOUSECFG_TAP_LOCKTIME:
|
|
tp->tap.locktime = val;
|
|
break;
|
|
case WSMOUSECFG_TAP_ONE_BTNMAP:
|
|
tp->tap.btnmap[0] = BTNMASK(val);
|
|
break;
|
|
case WSMOUSECFG_TAP_TWO_BTNMAP:
|
|
tp->tap.btnmap[1] = BTNMASK(val);
|
|
break;
|
|
case WSMOUSECFG_TAP_THREE_BTNMAP:
|
|
tp->tap.btnmap[2] = BTNMASK(val);
|
|
break;
|
|
default:
|
|
return (ENOTSUP);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
wstpad_get_param(struct wsmouseinput *input, int key, int *pval)
|
|
{
|
|
struct wstpad *tp = input->tp;
|
|
u_int flag;
|
|
|
|
if (tp == NULL)
|
|
return (EINVAL);
|
|
|
|
switch (key) {
|
|
case WSMOUSECFG_SOFTBUTTONS ... WSMOUSECFG_DISABLE:
|
|
switch (key) {
|
|
case WSMOUSECFG_SOFTBUTTONS:
|
|
flag = WSTPAD_SOFTBUTTONS;
|
|
break;
|
|
case WSMOUSECFG_SOFTMBTN:
|
|
flag = WSTPAD_SOFTMBTN;
|
|
break;
|
|
case WSMOUSECFG_TOPBUTTONS:
|
|
flag = WSTPAD_TOPBUTTONS;
|
|
break;
|
|
case WSMOUSECFG_TWOFINGERSCROLL:
|
|
flag = WSTPAD_TWOFINGERSCROLL;
|
|
break;
|
|
case WSMOUSECFG_EDGESCROLL:
|
|
flag = WSTPAD_EDGESCROLL;
|
|
break;
|
|
case WSMOUSECFG_HORIZSCROLL:
|
|
flag = WSTPAD_HORIZSCROLL;
|
|
break;
|
|
case WSMOUSECFG_SWAPSIDES:
|
|
flag = WSTPAD_SWAPSIDES;
|
|
break;
|
|
case WSMOUSECFG_DISABLE:
|
|
flag = WSTPAD_DISABLE;
|
|
break;
|
|
}
|
|
*pval = !!(tp->features & flag);
|
|
break;
|
|
case WSMOUSECFG_LEFT_EDGE:
|
|
*pval = tp->params.left_edge;
|
|
break;
|
|
case WSMOUSECFG_RIGHT_EDGE:
|
|
*pval = tp->params.right_edge;
|
|
break;
|
|
case WSMOUSECFG_TOP_EDGE:
|
|
*pval = tp->params.top_edge;
|
|
break;
|
|
case WSMOUSECFG_BOTTOM_EDGE:
|
|
*pval = tp->params.bottom_edge;
|
|
break;
|
|
case WSMOUSECFG_CENTERWIDTH:
|
|
*pval = tp->params.center_width;
|
|
break;
|
|
case WSMOUSECFG_HORIZSCROLLDIST:
|
|
*pval = tp->scroll.hdist;
|
|
break;
|
|
case WSMOUSECFG_VERTSCROLLDIST:
|
|
*pval = tp->scroll.vdist;
|
|
break;
|
|
case WSMOUSECFG_F2WIDTH:
|
|
*pval = tp->params.f2width;
|
|
break;
|
|
case WSMOUSECFG_F2PRESSURE:
|
|
*pval = tp->params.f2pressure;
|
|
break;
|
|
case WSMOUSECFG_TAP_MAXTIME:
|
|
*pval = tp->tap.maxtime.tv_nsec / 1000000;
|
|
break;
|
|
case WSMOUSECFG_TAP_CLICKTIME:
|
|
*pval = tp->tap.clicktime;
|
|
break;
|
|
case WSMOUSECFG_TAP_LOCKTIME:
|
|
*pval = tp->tap.locktime;
|
|
break;
|
|
case WSMOUSECFG_TAP_ONE_BTNMAP:
|
|
*pval = ffs(tp->tap.btnmap[0]);
|
|
break;
|
|
case WSMOUSECFG_TAP_TWO_BTNMAP:
|
|
*pval = ffs(tp->tap.btnmap[1]);
|
|
break;
|
|
case WSMOUSECFG_TAP_THREE_BTNMAP:
|
|
*pval = ffs(tp->tap.btnmap[2]);
|
|
break;
|
|
default:
|
|
return (ENOTSUP);
|
|
}
|
|
|
|
return (0);
|
|
}
|