2014-02-06 15:05:36 +10:00
|
|
|
/*
|
2015-05-28 08:23:59 +10:00
|
|
|
* Copyright © 2014-2015 Red Hat, Inc.
|
2014-02-06 15:05:36 +10:00
|
|
|
*
|
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software and
|
|
|
|
|
* its documentation for any purpose is hereby granted without fee, provided
|
|
|
|
|
* that the above copyright notice appear in all copies and that both that
|
|
|
|
|
* copyright notice and this permission notice appear in supporting
|
|
|
|
|
* documentation, and that the name of the copyright holders not be used in
|
|
|
|
|
* advertising or publicity pertaining to distribution of the software
|
|
|
|
|
* without specific, written prior permission. The copyright holders make
|
|
|
|
|
* no representations about the suitability of this software for any
|
|
|
|
|
* purpose. It is provided "as is" without express or implied warranty.
|
|
|
|
|
*
|
|
|
|
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
|
|
|
|
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
|
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
|
|
|
* SPECIAL, 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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
#include <assert.h>
|
2014-02-06 18:57:10 +10:00
|
|
|
#include <math.h>
|
2014-02-06 15:32:32 +10:00
|
|
|
#include <stdbool.h>
|
2014-07-15 14:01:00 +10:00
|
|
|
#include <limits.h>
|
2014-02-06 15:32:32 +10:00
|
|
|
|
2014-02-07 13:39:27 +10:00
|
|
|
#include "evdev-mt-touchpad.h"
|
2014-02-06 15:05:36 +10:00
|
|
|
|
2015-03-13 13:56:14 +10:00
|
|
|
/* Number found by trial-and error, seems to be 1200, divided by the
|
|
|
|
|
* TP_MAGIC_SLOWDOWN in filter.c */
|
|
|
|
|
#define DEFAULT_ACCEL_NUMERATOR 3000.0
|
2014-02-06 18:57:10 +10:00
|
|
|
#define DEFAULT_HYSTERESIS_MARGIN_DENOMINATOR 700.0
|
2014-09-28 13:21:08 +02:00
|
|
|
#define DEFAULT_TRACKPOINT_ACTIVITY_TIMEOUT 500 /* ms */
|
2015-04-20 16:20:00 +10:00
|
|
|
#define DEFAULT_KEYBOARD_ACTIVITY_TIMEOUT_1 200 /* ms */
|
|
|
|
|
#define DEFAULT_KEYBOARD_ACTIVITY_TIMEOUT_2 500 /* ms */
|
2015-04-09 14:23:16 +10:00
|
|
|
#define FAKE_FINGER_OVERFLOW (1 << 7)
|
2014-02-06 15:05:36 +10:00
|
|
|
|
2014-02-06 18:57:10 +10:00
|
|
|
static inline int
|
|
|
|
|
tp_hysteresis(int in, int center, int margin)
|
|
|
|
|
{
|
|
|
|
|
int diff = in - center;
|
|
|
|
|
if (abs(diff) <= margin)
|
|
|
|
|
return center;
|
|
|
|
|
|
|
|
|
|
if (diff > margin)
|
|
|
|
|
return center + diff - margin;
|
2014-06-30 14:27:18 +02:00
|
|
|
else
|
2014-02-06 18:57:10 +10:00
|
|
|
return center + diff + margin;
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-11 09:10:13 +10:00
|
|
|
static inline struct device_coords *
|
2014-02-06 15:32:32 +10:00
|
|
|
tp_motion_history_offset(struct tp_touch *t, int offset)
|
|
|
|
|
{
|
|
|
|
|
int offset_index =
|
|
|
|
|
(t->history.index - offset + TOUCHPAD_HISTORY_LENGTH) %
|
|
|
|
|
TOUCHPAD_HISTORY_LENGTH;
|
|
|
|
|
|
|
|
|
|
return &t->history.samples[offset_index];
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-24 16:43:24 +01:00
|
|
|
struct normalized_coords
|
2014-02-06 19:43:48 +10:00
|
|
|
tp_filter_motion(struct tp_dispatch *tp,
|
2015-03-24 16:43:24 +01:00
|
|
|
const struct normalized_coords *unaccelerated,
|
2014-12-04 11:44:09 +08:00
|
|
|
uint64_t time)
|
2014-02-06 19:43:48 +10:00
|
|
|
{
|
2015-03-24 16:43:24 +01:00
|
|
|
if (normalized_is_zero(*unaccelerated))
|
|
|
|
|
return *unaccelerated;
|
2014-02-06 19:43:48 +10:00
|
|
|
|
2015-03-24 16:43:24 +01:00
|
|
|
return filter_dispatch(tp->device->pointer.filter,
|
|
|
|
|
unaccelerated, tp, time);
|
2014-02-06 19:43:48 +10:00
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static inline void
|
|
|
|
|
tp_motion_history_push(struct tp_touch *t)
|
|
|
|
|
{
|
|
|
|
|
int motion_index = (t->history.index + 1) % TOUCHPAD_HISTORY_LENGTH;
|
|
|
|
|
|
|
|
|
|
if (t->history.count < TOUCHPAD_HISTORY_LENGTH)
|
|
|
|
|
t->history.count++;
|
|
|
|
|
|
2015-03-11 09:24:52 +10:00
|
|
|
t->history.samples[motion_index] = t->point;
|
2014-02-06 15:32:32 +10:00
|
|
|
t->history.index = motion_index;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 18:57:10 +10:00
|
|
|
static inline void
|
|
|
|
|
tp_motion_hysteresis(struct tp_dispatch *tp,
|
|
|
|
|
struct tp_touch *t)
|
|
|
|
|
{
|
2015-03-11 09:24:52 +10:00
|
|
|
int x = t->point.x,
|
|
|
|
|
y = t->point.y;
|
2014-02-06 18:57:10 +10:00
|
|
|
|
|
|
|
|
if (t->history.count == 0) {
|
2015-03-11 09:24:52 +10:00
|
|
|
t->hysteresis_center = t->point;
|
2014-02-06 18:57:10 +10:00
|
|
|
} else {
|
|
|
|
|
x = tp_hysteresis(x,
|
2015-03-11 09:24:52 +10:00
|
|
|
t->hysteresis_center.x,
|
|
|
|
|
tp->hysteresis_margin.x);
|
2014-02-06 18:57:10 +10:00
|
|
|
y = tp_hysteresis(y,
|
2015-03-11 09:24:52 +10:00
|
|
|
t->hysteresis_center.y,
|
|
|
|
|
tp->hysteresis_margin.y);
|
|
|
|
|
t->hysteresis_center.x = x;
|
|
|
|
|
t->hysteresis_center.y = y;
|
|
|
|
|
t->point.x = x;
|
|
|
|
|
t->point.y = y;
|
2014-02-06 18:57:10 +10:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static inline void
|
|
|
|
|
tp_motion_history_reset(struct tp_touch *t)
|
|
|
|
|
{
|
|
|
|
|
t->history.count = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline struct tp_touch *
|
|
|
|
|
tp_current_touch(struct tp_dispatch *tp)
|
|
|
|
|
{
|
2014-06-18 14:22:24 +02:00
|
|
|
return &tp->touches[min(tp->slot, tp->ntouches - 1)];
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
|
|
|
|
|
2014-02-14 15:12:22 +10:00
|
|
|
static inline struct tp_touch *
|
|
|
|
|
tp_get_touch(struct tp_dispatch *tp, unsigned int slot)
|
|
|
|
|
{
|
|
|
|
|
assert(slot < tp->ntouches);
|
|
|
|
|
return &tp->touches[slot];
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-12 10:31:12 +10:00
|
|
|
static inline unsigned int
|
|
|
|
|
tp_fake_finger_count(struct tp_dispatch *tp)
|
|
|
|
|
{
|
2015-04-09 14:23:16 +10:00
|
|
|
if (tp->fake_touches & FAKE_FINGER_OVERFLOW)
|
|
|
|
|
return FAKE_FINGER_OVERFLOW;
|
|
|
|
|
else /* don't count BTN_TOUCH */
|
|
|
|
|
return ffs(tp->fake_touches >> 1);
|
2014-12-12 10:31:12 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
|
tp_fake_finger_is_touching(struct tp_dispatch *tp)
|
|
|
|
|
{
|
|
|
|
|
return tp->fake_touches & 0x1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
|
tp_fake_finger_set(struct tp_dispatch *tp,
|
|
|
|
|
unsigned int code,
|
|
|
|
|
bool is_press)
|
|
|
|
|
{
|
|
|
|
|
unsigned int shift;
|
|
|
|
|
|
|
|
|
|
switch (code) {
|
|
|
|
|
case BTN_TOUCH:
|
2015-04-09 14:23:16 +10:00
|
|
|
if (!is_press)
|
|
|
|
|
tp->fake_touches &= ~FAKE_FINGER_OVERFLOW;
|
2014-12-12 10:31:12 +10:00
|
|
|
shift = 0;
|
|
|
|
|
break;
|
|
|
|
|
case BTN_TOOL_FINGER:
|
|
|
|
|
shift = 1;
|
|
|
|
|
break;
|
|
|
|
|
case BTN_TOOL_DOUBLETAP:
|
|
|
|
|
case BTN_TOOL_TRIPLETAP:
|
|
|
|
|
case BTN_TOOL_QUADTAP:
|
|
|
|
|
shift = code - BTN_TOOL_DOUBLETAP + 2;
|
|
|
|
|
break;
|
2015-04-09 14:23:16 +10:00
|
|
|
/* when QUINTTAP is released we're either switching to 6 fingers
|
|
|
|
|
(flag stays in place until BTN_TOUCH is released) or
|
|
|
|
|
one of DOUBLE/TRIPLE/QUADTAP (will clear the flag on press) */
|
|
|
|
|
case BTN_TOOL_QUINTTAP:
|
|
|
|
|
if (is_press)
|
|
|
|
|
tp->fake_touches |= FAKE_FINGER_OVERFLOW;
|
|
|
|
|
return;
|
2014-12-12 10:31:12 +10:00
|
|
|
default:
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-09 14:23:16 +10:00
|
|
|
if (is_press) {
|
|
|
|
|
tp->fake_touches &= ~FAKE_FINGER_OVERFLOW;
|
2014-12-12 10:31:12 +10:00
|
|
|
tp->fake_touches |= 1 << shift;
|
2015-04-09 14:23:16 +10:00
|
|
|
|
|
|
|
|
} else {
|
2014-12-12 10:31:12 +10:00
|
|
|
tp->fake_touches &= ~(0x1 << shift);
|
2015-04-09 14:23:16 +10:00
|
|
|
}
|
2014-12-12 10:31:12 +10:00
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static inline void
|
2014-12-11 16:39:04 +10:00
|
|
|
tp_new_touch(struct tp_dispatch *tp, struct tp_touch *t, uint64_t time)
|
2014-02-06 15:32:32 +10:00
|
|
|
{
|
2014-12-11 16:39:04 +10:00
|
|
|
if (t->state == TOUCH_BEGIN ||
|
|
|
|
|
t->state == TOUCH_UPDATE ||
|
|
|
|
|
t->state == TOUCH_HOVERING)
|
2014-07-18 11:06:36 +02:00
|
|
|
return;
|
|
|
|
|
|
2014-12-11 16:39:04 +10:00
|
|
|
/* we begin the touch as hovering because until BTN_TOUCH happens we
|
|
|
|
|
* don't know if it's a touch down or not. And BTN_TOUCH may happen
|
|
|
|
|
* after ABS_MT_TRACKING_ID */
|
2014-07-18 11:06:36 +02:00
|
|
|
tp_motion_history_reset(t);
|
|
|
|
|
t->dirty = true;
|
2014-12-11 16:39:04 +10:00
|
|
|
t->has_ended = false;
|
|
|
|
|
t->state = TOUCH_HOVERING;
|
2014-07-18 11:06:36 +02:00
|
|
|
t->pinned.is_pinned = false;
|
|
|
|
|
t->millis = time;
|
2014-12-11 16:39:04 +10:00
|
|
|
tp->queued |= TOUCHPAD_EVENT_MOTION;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
|
tp_begin_touch(struct tp_dispatch *tp, struct tp_touch *t, uint64_t time)
|
|
|
|
|
{
|
|
|
|
|
t->dirty = true;
|
|
|
|
|
t->state = TOUCH_BEGIN;
|
|
|
|
|
t->millis = time;
|
2014-07-18 11:06:36 +02:00
|
|
|
tp->nfingers_down++;
|
|
|
|
|
assert(tp->nfingers_down >= 1);
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
|
|
|
|
|
2014-12-11 16:39:04 +10:00
|
|
|
/**
|
|
|
|
|
* End a touch, even if the touch sequence is still active.
|
|
|
|
|
*/
|
2014-02-06 15:32:32 +10:00
|
|
|
static inline void
|
2014-07-18 11:06:36 +02:00
|
|
|
tp_end_touch(struct tp_dispatch *tp, struct tp_touch *t, uint64_t time)
|
2014-02-06 15:32:32 +10:00
|
|
|
{
|
2014-12-11 16:39:04 +10:00
|
|
|
switch (t->state) {
|
|
|
|
|
case TOUCH_HOVERING:
|
|
|
|
|
t->state = TOUCH_NONE;
|
|
|
|
|
/* fallthough */
|
|
|
|
|
case TOUCH_NONE:
|
|
|
|
|
case TOUCH_END:
|
2014-02-06 15:32:32 +10:00
|
|
|
return;
|
2014-12-11 16:39:04 +10:00
|
|
|
case TOUCH_BEGIN:
|
|
|
|
|
case TOUCH_UPDATE:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
}
|
2014-02-06 15:32:32 +10:00
|
|
|
|
|
|
|
|
t->dirty = true;
|
2015-04-20 16:09:31 +10:00
|
|
|
t->palm.state = PALM_NONE;
|
2014-02-06 15:32:32 +10:00
|
|
|
t->state = TOUCH_END;
|
2014-04-15 14:27:59 +02:00
|
|
|
t->pinned.is_pinned = false;
|
2014-07-18 11:06:36 +02:00
|
|
|
t->millis = time;
|
2015-05-21 16:32:42 +10:00
|
|
|
t->palm.time = 0;
|
2014-02-06 15:32:32 +10:00
|
|
|
assert(tp->nfingers_down >= 1);
|
|
|
|
|
tp->nfingers_down--;
|
2014-02-07 13:48:06 +10:00
|
|
|
tp->queued |= TOUCHPAD_EVENT_MOTION;
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
|
|
|
|
|
2014-12-11 16:39:04 +10:00
|
|
|
/**
|
|
|
|
|
* End the touch sequence on ABS_MT_TRACKING_ID -1 or when the BTN_TOOL_* 0 is received.
|
|
|
|
|
*/
|
|
|
|
|
static inline void
|
|
|
|
|
tp_end_sequence(struct tp_dispatch *tp, struct tp_touch *t, uint64_t time)
|
|
|
|
|
{
|
|
|
|
|
t->has_ended = true;
|
|
|
|
|
tp_end_touch(tp, t, time);
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static double
|
|
|
|
|
tp_estimate_delta(int x0, int x1, int x2, int x3)
|
|
|
|
|
{
|
2014-07-18 11:06:39 +02:00
|
|
|
return (x0 + x1 - x2 - x3) / 4.0;
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
|
|
|
|
|
2015-03-11 09:58:11 +10:00
|
|
|
struct normalized_coords
|
|
|
|
|
tp_get_delta(struct tp_touch *t)
|
2014-02-06 15:05:36 +10:00
|
|
|
{
|
2015-03-25 11:01:04 +01:00
|
|
|
struct device_float_coords delta;
|
2015-03-24 13:14:18 +01:00
|
|
|
const struct normalized_coords zero = { 0.0, 0.0 };
|
2015-03-11 09:58:11 +10:00
|
|
|
|
|
|
|
|
if (t->history.count < TOUCHPAD_MIN_SAMPLES)
|
2015-03-24 13:14:18 +01:00
|
|
|
return zero;
|
|
|
|
|
|
2015-03-25 11:01:04 +01:00
|
|
|
delta.x = tp_estimate_delta(tp_motion_history_offset(t, 0)->x,
|
|
|
|
|
tp_motion_history_offset(t, 1)->x,
|
|
|
|
|
tp_motion_history_offset(t, 2)->x,
|
|
|
|
|
tp_motion_history_offset(t, 3)->x);
|
|
|
|
|
delta.y = tp_estimate_delta(tp_motion_history_offset(t, 0)->y,
|
|
|
|
|
tp_motion_history_offset(t, 1)->y,
|
|
|
|
|
tp_motion_history_offset(t, 2)->y,
|
|
|
|
|
tp_motion_history_offset(t, 3)->y);
|
2015-03-24 13:14:18 +01:00
|
|
|
|
|
|
|
|
return tp_normalize_delta(t->tp, delta);
|
2014-02-06 15:05:36 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2014-02-06 15:32:32 +10:00
|
|
|
tp_process_absolute(struct tp_dispatch *tp,
|
|
|
|
|
const struct input_event *e,
|
2014-04-08 12:29:45 +02:00
|
|
|
uint64_t time)
|
2014-02-06 15:05:36 +10:00
|
|
|
{
|
2014-02-06 15:32:32 +10:00
|
|
|
struct tp_touch *t = tp_current_touch(tp);
|
|
|
|
|
|
|
|
|
|
switch(e->code) {
|
|
|
|
|
case ABS_MT_POSITION_X:
|
2015-03-11 09:24:52 +10:00
|
|
|
t->point.x = e->value;
|
2014-02-06 15:32:32 +10:00
|
|
|
t->millis = time;
|
|
|
|
|
t->dirty = true;
|
2014-02-07 13:48:06 +10:00
|
|
|
tp->queued |= TOUCHPAD_EVENT_MOTION;
|
2014-02-06 15:32:32 +10:00
|
|
|
break;
|
|
|
|
|
case ABS_MT_POSITION_Y:
|
2015-03-11 09:24:52 +10:00
|
|
|
t->point.y = e->value;
|
2014-02-06 15:32:32 +10:00
|
|
|
t->millis = time;
|
|
|
|
|
t->dirty = true;
|
2014-02-07 13:48:06 +10:00
|
|
|
tp->queued |= TOUCHPAD_EVENT_MOTION;
|
2014-02-06 15:32:32 +10:00
|
|
|
break;
|
|
|
|
|
case ABS_MT_SLOT:
|
|
|
|
|
tp->slot = e->value;
|
|
|
|
|
break;
|
2015-05-06 19:41:25 -04:00
|
|
|
case ABS_MT_DISTANCE:
|
|
|
|
|
t->distance = e->value;
|
|
|
|
|
break;
|
2014-02-06 15:32:32 +10:00
|
|
|
case ABS_MT_TRACKING_ID:
|
|
|
|
|
if (e->value != -1)
|
2014-12-11 16:39:04 +10:00
|
|
|
tp_new_touch(tp, t, time);
|
2014-02-06 15:32:32 +10:00
|
|
|
else
|
2014-12-11 16:39:04 +10:00
|
|
|
tp_end_sequence(tp, t, time);
|
2015-05-27 18:29:26 +10:00
|
|
|
break;
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
2014-02-06 15:05:36 +10:00
|
|
|
}
|
|
|
|
|
|
2014-02-14 14:18:27 +10:00
|
|
|
static void
|
|
|
|
|
tp_process_absolute_st(struct tp_dispatch *tp,
|
|
|
|
|
const struct input_event *e,
|
2014-04-08 12:29:45 +02:00
|
|
|
uint64_t time)
|
2014-02-14 14:18:27 +10:00
|
|
|
{
|
|
|
|
|
struct tp_touch *t = tp_current_touch(tp);
|
|
|
|
|
|
|
|
|
|
switch(e->code) {
|
|
|
|
|
case ABS_X:
|
2015-03-11 09:24:52 +10:00
|
|
|
t->point.x = e->value;
|
2014-02-14 14:18:27 +10:00
|
|
|
t->millis = time;
|
|
|
|
|
t->dirty = true;
|
2014-03-25 11:30:42 +10:00
|
|
|
tp->queued |= TOUCHPAD_EVENT_MOTION;
|
2014-02-14 14:18:27 +10:00
|
|
|
break;
|
|
|
|
|
case ABS_Y:
|
2015-03-11 09:24:52 +10:00
|
|
|
t->point.y = e->value;
|
2014-02-14 14:18:27 +10:00
|
|
|
t->millis = time;
|
|
|
|
|
t->dirty = true;
|
|
|
|
|
tp->queued |= TOUCHPAD_EVENT_MOTION;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-14 15:12:22 +10:00
|
|
|
static void
|
2015-04-08 17:00:50 +10:00
|
|
|
tp_process_fake_touches(struct tp_dispatch *tp,
|
|
|
|
|
uint64_t time)
|
2014-02-14 15:12:22 +10:00
|
|
|
{
|
|
|
|
|
struct tp_touch *t;
|
|
|
|
|
unsigned int nfake_touches;
|
2014-07-18 11:06:38 +02:00
|
|
|
unsigned int i, start;
|
2014-02-14 15:12:22 +10:00
|
|
|
|
2014-12-12 10:31:12 +10:00
|
|
|
nfake_touches = tp_fake_finger_count(tp);
|
2015-04-09 14:23:16 +10:00
|
|
|
if (nfake_touches == FAKE_FINGER_OVERFLOW)
|
|
|
|
|
return;
|
2014-02-14 15:12:22 +10:00
|
|
|
|
2015-04-09 15:27:34 +10:00
|
|
|
start = tp->has_mt ? tp->num_slots : 0;
|
2014-07-18 11:06:38 +02:00
|
|
|
for (i = start; i < tp->ntouches; i++) {
|
2014-02-14 15:12:22 +10:00
|
|
|
t = tp_get_touch(tp, i);
|
2014-07-18 11:06:38 +02:00
|
|
|
if (i < nfake_touches)
|
2014-12-11 16:39:04 +10:00
|
|
|
tp_new_touch(tp, t, time);
|
2014-07-18 11:06:38 +02:00
|
|
|
else
|
2014-12-11 16:39:04 +10:00
|
|
|
tp_end_sequence(tp, t, time);
|
2014-02-14 15:12:22 +10:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-28 15:38:00 +10:00
|
|
|
static void
|
|
|
|
|
tp_process_trackpoint_button(struct tp_dispatch *tp,
|
|
|
|
|
const struct input_event *e,
|
|
|
|
|
uint64_t time)
|
|
|
|
|
{
|
|
|
|
|
struct evdev_dispatch *dispatch;
|
|
|
|
|
struct input_event event;
|
|
|
|
|
|
|
|
|
|
if (!tp->buttons.trackpoint ||
|
|
|
|
|
(tp->device->tags & EVDEV_TAG_TOUCHPAD_TRACKPOINT) == 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
dispatch = tp->buttons.trackpoint->dispatch;
|
|
|
|
|
|
|
|
|
|
event = *e;
|
|
|
|
|
|
|
|
|
|
switch (event.code) {
|
|
|
|
|
case BTN_0:
|
|
|
|
|
event.code = BTN_LEFT;
|
|
|
|
|
break;
|
|
|
|
|
case BTN_1:
|
|
|
|
|
event.code = BTN_RIGHT;
|
|
|
|
|
break;
|
|
|
|
|
case BTN_2:
|
|
|
|
|
event.code = BTN_MIDDLE;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dispatch->interface->process(dispatch,
|
|
|
|
|
tp->buttons.trackpoint,
|
|
|
|
|
&event, time);
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static void
|
|
|
|
|
tp_process_key(struct tp_dispatch *tp,
|
|
|
|
|
const struct input_event *e,
|
2014-04-08 12:29:45 +02:00
|
|
|
uint64_t time)
|
2014-02-06 15:32:32 +10:00
|
|
|
{
|
|
|
|
|
switch (e->code) {
|
|
|
|
|
case BTN_LEFT:
|
|
|
|
|
case BTN_MIDDLE:
|
|
|
|
|
case BTN_RIGHT:
|
2014-03-27 13:09:06 +10:00
|
|
|
tp_process_button(tp, e, time);
|
2014-02-06 15:32:32 +10:00
|
|
|
break;
|
2014-02-14 14:18:27 +10:00
|
|
|
case BTN_TOUCH:
|
2014-12-11 16:39:04 +10:00
|
|
|
case BTN_TOOL_FINGER:
|
2014-02-14 15:12:22 +10:00
|
|
|
case BTN_TOOL_DOUBLETAP:
|
|
|
|
|
case BTN_TOOL_TRIPLETAP:
|
|
|
|
|
case BTN_TOOL_QUADTAP:
|
2015-04-09 14:23:16 +10:00
|
|
|
case BTN_TOOL_QUINTTAP:
|
2015-04-08 17:00:50 +10:00
|
|
|
tp_fake_finger_set(tp, e->code, !!e->value);
|
2014-02-14 14:18:27 +10:00
|
|
|
break;
|
2015-01-28 15:38:00 +10:00
|
|
|
case BTN_0:
|
|
|
|
|
case BTN_1:
|
|
|
|
|
case BTN_2:
|
|
|
|
|
tp_process_trackpoint_button(tp, e, time);
|
|
|
|
|
break;
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-17 14:24:20 +10:00
|
|
|
static void
|
2014-04-15 14:27:59 +02:00
|
|
|
tp_unpin_finger(struct tp_dispatch *tp, struct tp_touch *t)
|
2014-02-17 14:24:20 +10:00
|
|
|
{
|
2014-04-15 14:27:59 +02:00
|
|
|
unsigned int xdist, ydist;
|
2014-02-17 14:24:20 +10:00
|
|
|
|
2014-04-15 14:27:59 +02:00
|
|
|
if (!t->pinned.is_pinned)
|
|
|
|
|
return;
|
|
|
|
|
|
2015-03-11 09:24:52 +10:00
|
|
|
xdist = abs(t->point.x - t->pinned.center.x);
|
|
|
|
|
ydist = abs(t->point.y - t->pinned.center.y);
|
2014-02-17 14:24:20 +10:00
|
|
|
|
2014-04-04 17:22:02 +02:00
|
|
|
if (xdist * xdist + ydist * ydist >=
|
|
|
|
|
tp->buttons.motion_dist * tp->buttons.motion_dist) {
|
|
|
|
|
t->pinned.is_pinned = false;
|
2015-01-14 10:17:06 +01:00
|
|
|
return;
|
2014-02-17 14:24:20 +10:00
|
|
|
}
|
2015-01-14 10:17:06 +01:00
|
|
|
|
|
|
|
|
/* The finger may slowly drift, adjust the center */
|
2015-03-11 09:24:52 +10:00
|
|
|
t->pinned.center.x = t->point.x + t->pinned.center.x / 2;
|
|
|
|
|
t->pinned.center.y = t->point.y + t->pinned.center.y / 2;
|
2014-02-17 14:24:20 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2014-04-15 14:27:59 +02:00
|
|
|
tp_pin_fingers(struct tp_dispatch *tp)
|
2014-02-17 14:24:20 +10:00
|
|
|
{
|
2014-04-15 14:27:59 +02:00
|
|
|
struct tp_touch *t;
|
2014-02-17 14:24:20 +10:00
|
|
|
|
|
|
|
|
tp_for_each_touch(tp, t) {
|
2014-04-15 14:27:59 +02:00
|
|
|
t->pinned.is_pinned = true;
|
2015-03-11 09:24:52 +10:00
|
|
|
t->pinned.center = t->point;
|
2014-02-17 14:24:20 +10:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-16 13:30:24 +01:00
|
|
|
int
|
2014-04-04 17:22:02 +02:00
|
|
|
tp_touch_active(struct tp_dispatch *tp, struct tp_touch *t)
|
|
|
|
|
{
|
|
|
|
|
return (t->state == TOUCH_BEGIN || t->state == TOUCH_UPDATE) &&
|
2015-04-20 16:09:31 +10:00
|
|
|
t->palm.state == PALM_NONE &&
|
2014-11-24 12:16:06 +01:00
|
|
|
!t->pinned.is_pinned &&
|
|
|
|
|
tp_button_touch_active(tp, t) &&
|
|
|
|
|
tp_edge_scroll_touch_active(tp, t);
|
2014-04-04 17:22:02 +02:00
|
|
|
}
|
2014-02-17 14:24:20 +10:00
|
|
|
|
2015-04-15 15:21:08 +10:00
|
|
|
bool
|
|
|
|
|
tp_palm_tap_is_palm(struct tp_dispatch *tp, struct tp_touch *t)
|
|
|
|
|
{
|
|
|
|
|
if (t->state != TOUCH_BEGIN)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
if (t->point.x > tp->palm.left_edge &&
|
|
|
|
|
t->point.x < tp->palm.right_edge)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
/* We're inside the left/right palm edge and in the northern half of
|
|
|
|
|
* the touchpad - this tap is a palm */
|
2015-05-22 15:14:04 +10:00
|
|
|
if (t->point.y < tp->palm.vert_center) {
|
|
|
|
|
log_debug(tp_libinput_context(tp),
|
|
|
|
|
"palm: palm-tap detected\n");
|
2015-04-15 15:21:08 +10:00
|
|
|
return true;
|
2015-05-22 15:14:04 +10:00
|
|
|
}
|
2015-04-15 15:21:08 +10:00
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-21 13:30:24 +10:00
|
|
|
static int
|
|
|
|
|
tp_palm_detect_dwt(struct tp_dispatch *tp, struct tp_touch *t, uint64_t time)
|
|
|
|
|
{
|
2015-05-21 16:32:42 +10:00
|
|
|
if (tp->dwt.keyboard_active &&
|
|
|
|
|
t->state == TOUCH_BEGIN) {
|
2015-05-21 13:30:24 +10:00
|
|
|
t->palm.state = PALM_TYPING;
|
|
|
|
|
t->palm.time = time;
|
|
|
|
|
t->palm.first = t->point;
|
|
|
|
|
return 1;
|
2015-05-21 16:32:42 +10:00
|
|
|
} else if (!tp->dwt.keyboard_active &&
|
|
|
|
|
t->state == TOUCH_UPDATE &&
|
|
|
|
|
t->palm.state == PALM_TYPING)
|
|
|
|
|
{
|
|
|
|
|
/* If a touch has started before the first or after the last
|
|
|
|
|
key press, release it on timeout. Benefit: a palm rested
|
|
|
|
|
while typing on the touchpad will be ignored, but a touch
|
|
|
|
|
started once we stop typing will be able to control the
|
|
|
|
|
pointer (alas not tap, etc.).
|
|
|
|
|
*/
|
|
|
|
|
if (t->palm.time == 0 ||
|
|
|
|
|
t->palm.time > tp->dwt.keyboard_last_press_time) {
|
|
|
|
|
t->palm.state = PALM_NONE;
|
|
|
|
|
log_debug(tp_libinput_context(tp),
|
|
|
|
|
"palm: touch released, timeout after typing\n");
|
|
|
|
|
}
|
2015-05-21 13:30:24 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-10 17:34:08 +10:00
|
|
|
static void
|
2014-07-14 16:06:51 +10:00
|
|
|
tp_palm_detect(struct tp_dispatch *tp, struct tp_touch *t, uint64_t time)
|
2014-07-10 17:34:08 +10:00
|
|
|
{
|
2014-07-14 16:06:51 +10:00
|
|
|
const int PALM_TIMEOUT = 200; /* ms */
|
2014-07-14 16:38:46 +10:00
|
|
|
const int DIRECTIONS = NE|E|SE|SW|W|NW;
|
2015-03-25 15:05:19 +01:00
|
|
|
struct device_float_coords delta;
|
|
|
|
|
int dirs;
|
2014-07-14 16:06:51 +10:00
|
|
|
|
2015-05-21 13:30:24 +10:00
|
|
|
if (tp_palm_detect_dwt(tp, t, time))
|
2015-05-22 15:14:04 +10:00
|
|
|
goto out;
|
2015-04-20 15:51:20 +10:00
|
|
|
|
2014-07-14 16:06:51 +10:00
|
|
|
/* If labelled a touch as palm, we unlabel as palm when
|
2014-07-14 16:38:46 +10:00
|
|
|
we move out of the palm edge zone within the timeout, provided
|
|
|
|
|
the direction is within 45 degrees of the horizontal.
|
2014-07-14 16:06:51 +10:00
|
|
|
*/
|
2015-04-20 16:09:31 +10:00
|
|
|
if (t->palm.state == PALM_EDGE) {
|
2014-07-14 16:06:51 +10:00
|
|
|
if (time < t->palm.time + PALM_TIMEOUT &&
|
2015-03-11 09:24:52 +10:00
|
|
|
(t->point.x > tp->palm.left_edge && t->point.x < tp->palm.right_edge)) {
|
2015-03-25 15:05:19 +01:00
|
|
|
delta = device_delta(t->point, t->palm.first);
|
|
|
|
|
dirs = normalized_get_direction(
|
|
|
|
|
tp_normalize_delta(tp, delta));
|
2014-07-14 16:38:46 +10:00
|
|
|
if ((dirs & DIRECTIONS) && !(dirs & ~DIRECTIONS)) {
|
2015-04-20 16:09:31 +10:00
|
|
|
t->palm.state = PALM_NONE;
|
2015-05-22 15:14:04 +10:00
|
|
|
log_debug(tp_libinput_context(tp),
|
|
|
|
|
"palm: touch released, out of edge zone\n");
|
2014-07-14 16:38:46 +10:00
|
|
|
}
|
2014-07-14 16:06:51 +10:00
|
|
|
}
|
2014-07-10 17:34:08 +10:00
|
|
|
return;
|
2014-07-14 16:06:51 +10:00
|
|
|
}
|
2014-07-10 17:34:08 +10:00
|
|
|
|
|
|
|
|
/* palm must start in exclusion zone, it's ok to move into
|
|
|
|
|
the zone without being a palm */
|
|
|
|
|
if (t->state != TOUCH_BEGIN ||
|
2015-03-11 09:24:52 +10:00
|
|
|
(t->point.x > tp->palm.left_edge && t->point.x < tp->palm.right_edge))
|
2014-07-10 17:34:08 +10:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* don't detect palm in software button areas, it's
|
|
|
|
|
likely that legitimate touches start in the area
|
|
|
|
|
covered by the exclusion zone */
|
|
|
|
|
if (tp->buttons.is_clickpad &&
|
|
|
|
|
tp_button_is_inside_softbutton_area(tp, t))
|
|
|
|
|
return;
|
|
|
|
|
|
2015-04-20 16:09:31 +10:00
|
|
|
t->palm.state = PALM_EDGE;
|
2014-07-14 16:06:51 +10:00
|
|
|
t->palm.time = time;
|
2015-03-11 09:24:52 +10:00
|
|
|
t->palm.first = t->point;
|
2015-05-22 15:14:04 +10:00
|
|
|
|
|
|
|
|
out:
|
|
|
|
|
log_debug(tp_libinput_context(tp),
|
|
|
|
|
"palm: palm detected (%s)\n",
|
|
|
|
|
t->palm.state == PALM_EDGE ? "edge" : "typing");
|
2014-07-10 17:34:08 +10:00
|
|
|
}
|
|
|
|
|
|
2014-12-11 16:39:04 +10:00
|
|
|
static void
|
2015-05-06 19:41:25 -04:00
|
|
|
tp_unhover_abs_distance(struct tp_dispatch *tp, uint64_t time)
|
|
|
|
|
{
|
|
|
|
|
struct tp_touch *t;
|
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < tp->ntouches; i++) {
|
|
|
|
|
t = tp_get_touch(tp, i);
|
|
|
|
|
|
|
|
|
|
if (t->state == TOUCH_HOVERING) {
|
|
|
|
|
if (t->distance == 0) {
|
|
|
|
|
/* avoid jumps when landing a finger */
|
|
|
|
|
tp_motion_history_reset(t);
|
|
|
|
|
tp_begin_touch(tp, t, time);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (t->distance > 0)
|
|
|
|
|
tp_end_touch(tp, t, time);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
tp_unhover_fake_touches(struct tp_dispatch *tp, uint64_t time)
|
2014-12-11 16:39:04 +10:00
|
|
|
{
|
|
|
|
|
struct tp_touch *t;
|
|
|
|
|
unsigned int nfake_touches;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
if (!tp->fake_touches && !tp->nfingers_down)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
nfake_touches = tp_fake_finger_count(tp);
|
2015-04-09 14:23:16 +10:00
|
|
|
if (nfake_touches == FAKE_FINGER_OVERFLOW)
|
|
|
|
|
return;
|
|
|
|
|
|
2014-12-11 16:39:04 +10:00
|
|
|
if (tp->nfingers_down == nfake_touches &&
|
|
|
|
|
((tp->nfingers_down == 0 && !tp_fake_finger_is_touching(tp)) ||
|
|
|
|
|
(tp->nfingers_down > 0 && tp_fake_finger_is_touching(tp))))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* if BTN_TOUCH is set and we have less fingers down than fake
|
|
|
|
|
* touches, switch each hovering touch to BEGIN
|
|
|
|
|
* until nfingers_down matches nfake_touches
|
|
|
|
|
*/
|
|
|
|
|
if (tp_fake_finger_is_touching(tp) &&
|
|
|
|
|
tp->nfingers_down < nfake_touches) {
|
|
|
|
|
for (i = 0; i < (int)tp->ntouches; i++) {
|
|
|
|
|
t = tp_get_touch(tp, i);
|
|
|
|
|
|
|
|
|
|
if (t->state == TOUCH_HOVERING) {
|
|
|
|
|
tp_begin_touch(tp, t, time);
|
|
|
|
|
|
|
|
|
|
if (tp->nfingers_down >= nfake_touches)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* if BTN_TOUCH is unset end all touches, we're hovering now. If we
|
|
|
|
|
* have too many touches also end some of them. This is done in
|
|
|
|
|
* reverse order.
|
|
|
|
|
*/
|
|
|
|
|
if (tp->nfingers_down > nfake_touches ||
|
|
|
|
|
!tp_fake_finger_is_touching(tp)) {
|
|
|
|
|
for (i = tp->ntouches - 1; i >= 0; i--) {
|
|
|
|
|
t = tp_get_touch(tp, i);
|
|
|
|
|
|
2015-04-09 15:27:33 +10:00
|
|
|
if (t->state == TOUCH_HOVERING ||
|
|
|
|
|
t->state == TOUCH_NONE)
|
2014-12-11 16:39:04 +10:00
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
tp_end_touch(tp, t, time);
|
|
|
|
|
|
|
|
|
|
if (tp_fake_finger_is_touching(tp) &&
|
|
|
|
|
tp->nfingers_down == nfake_touches)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-06 19:41:25 -04:00
|
|
|
static void
|
|
|
|
|
tp_unhover_touches(struct tp_dispatch *tp, uint64_t time)
|
|
|
|
|
{
|
|
|
|
|
if (tp->reports_distance)
|
|
|
|
|
tp_unhover_abs_distance(tp, time);
|
|
|
|
|
else
|
|
|
|
|
tp_unhover_fake_touches(tp, time);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static void
|
2014-04-08 12:29:45 +02:00
|
|
|
tp_process_state(struct tp_dispatch *tp, uint64_t time)
|
2014-02-06 15:32:32 +10:00
|
|
|
{
|
|
|
|
|
struct tp_touch *t;
|
2014-02-14 15:12:22 +10:00
|
|
|
struct tp_touch *first = tp_get_touch(tp, 0);
|
2014-07-18 11:06:38 +02:00
|
|
|
unsigned int i;
|
2015-06-10 09:54:06 +10:00
|
|
|
bool restart_filter = false;
|
2014-02-06 15:32:32 +10:00
|
|
|
|
2015-04-08 17:00:50 +10:00
|
|
|
tp_process_fake_touches(tp, time);
|
2014-12-11 16:39:04 +10:00
|
|
|
tp_unhover_touches(tp, time);
|
|
|
|
|
|
2014-07-18 11:06:38 +02:00
|
|
|
for (i = 0; i < tp->ntouches; i++) {
|
|
|
|
|
t = tp_get_touch(tp, i);
|
2014-07-21 15:25:47 +02:00
|
|
|
|
|
|
|
|
/* semi-mt finger postions may "jump" when nfingers changes */
|
|
|
|
|
if (tp->semi_mt && tp->nfingers_down != tp->old_nfingers_down)
|
|
|
|
|
tp_motion_history_reset(t);
|
|
|
|
|
|
2015-04-09 15:27:34 +10:00
|
|
|
if (i >= tp->num_slots && t->state != TOUCH_NONE) {
|
2015-03-11 09:24:52 +10:00
|
|
|
t->point = first->point;
|
2014-02-14 15:12:22 +10:00
|
|
|
if (!t->dirty)
|
|
|
|
|
t->dirty = first->dirty;
|
2014-07-10 17:17:53 +10:00
|
|
|
}
|
2014-02-06 15:32:32 +10:00
|
|
|
|
2014-07-18 11:06:37 +02:00
|
|
|
if (!t->dirty)
|
|
|
|
|
continue;
|
|
|
|
|
|
2014-07-14 16:06:51 +10:00
|
|
|
tp_palm_detect(tp, t, time);
|
2014-07-10 17:34:08 +10:00
|
|
|
|
2014-02-06 18:57:10 +10:00
|
|
|
tp_motion_hysteresis(tp, t);
|
2014-02-06 15:32:32 +10:00
|
|
|
tp_motion_history_push(t);
|
2014-04-15 14:27:59 +02:00
|
|
|
|
|
|
|
|
tp_unpin_finger(tp, t);
|
2015-06-10 09:54:06 +10:00
|
|
|
|
|
|
|
|
if (t->state == TOUCH_BEGIN)
|
|
|
|
|
restart_filter = true;
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
2014-02-17 14:24:20 +10:00
|
|
|
|
2015-06-10 09:54:06 +10:00
|
|
|
if (restart_filter)
|
|
|
|
|
filter_restart(tp->device->pointer.filter, tp, time);
|
|
|
|
|
|
touchpad: Add clickpad-style software buttons
Almost all non Apple touchpads have visible markings for software button areas,
so limit clickfinger behavior to Apple clickpads, and implement software button
areas for others.
This is a slightly fancier implementation than the simplest model and ported
over from libtouchpad. It implements a state machine for the software buttons
with left and right buttons currently implemented. Buttons are oriented
left-to-right, in a horizontal bar. No random button placement allowed.
In general, the procedure is:
- if a finger sets down in the left button area, a click is a left click
- if a finger sets down in the right button area, a click is a right click
- if a finger leaves the button area, a click is a left click
- if a finger starts outside the button area, a click is a left click
Two timeouts are used to handle buttons more smoothly:
- if a finger sets down in a button area but "immediately" moves over
to a different area, that area takes effect on a click.
- if a finger leaves a button area and "immediately" clicks or moves back into
the area, the button still takes effect on a click.
- if a finger changes between areas and stays there for a timeout, that area
takes effect on a click.
Note the button area states are named BOTTOM_foo to make it easier to later
add support for a top button area such as can be found on the Thinkpad [2-5]40
series.
Co-authored-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Peter Hutterer <peter.hutterer@who-t.net>
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Jonas Ådahl <jadahl@gmail.com>
Reviewed-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Peter Hutterer <peter.hutterer@who-t.net>
2014-03-28 09:44:11 +10:00
|
|
|
tp_button_handle_state(tp, time);
|
2015-02-16 10:16:30 +01:00
|
|
|
tp_edge_scroll_handle_state(tp, time);
|
touchpad: Add clickpad-style software buttons
Almost all non Apple touchpads have visible markings for software button areas,
so limit clickfinger behavior to Apple clickpads, and implement software button
areas for others.
This is a slightly fancier implementation than the simplest model and ported
over from libtouchpad. It implements a state machine for the software buttons
with left and right buttons currently implemented. Buttons are oriented
left-to-right, in a horizontal bar. No random button placement allowed.
In general, the procedure is:
- if a finger sets down in the left button area, a click is a left click
- if a finger sets down in the right button area, a click is a right click
- if a finger leaves the button area, a click is a left click
- if a finger starts outside the button area, a click is a left click
Two timeouts are used to handle buttons more smoothly:
- if a finger sets down in a button area but "immediately" moves over
to a different area, that area takes effect on a click.
- if a finger leaves a button area and "immediately" clicks or moves back into
the area, the button still takes effect on a click.
- if a finger changes between areas and stays there for a timeout, that area
takes effect on a click.
Note the button area states are named BOTTOM_foo to make it easier to later
add support for a top button area such as can be found on the Thinkpad [2-5]40
series.
Co-authored-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Peter Hutterer <peter.hutterer@who-t.net>
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Jonas Ådahl <jadahl@gmail.com>
Reviewed-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Peter Hutterer <peter.hutterer@who-t.net>
2014-03-28 09:44:11 +10:00
|
|
|
|
2014-04-15 14:27:59 +02:00
|
|
|
/*
|
|
|
|
|
* We have a physical button down event on a clickpad. To avoid
|
|
|
|
|
* spurious pointer moves by the clicking finger we pin all fingers.
|
|
|
|
|
* We unpin fingers when they move more then a certain threshold to
|
|
|
|
|
* to allow drag and drop.
|
2014-02-17 14:24:20 +10:00
|
|
|
*/
|
|
|
|
|
if ((tp->queued & TOUCHPAD_EVENT_BUTTON_PRESS) &&
|
2014-04-07 13:44:07 +02:00
|
|
|
tp->buttons.is_clickpad)
|
2014-04-15 14:27:59 +02:00
|
|
|
tp_pin_fingers(tp);
|
2015-02-16 13:30:24 +01:00
|
|
|
|
|
|
|
|
tp_gesture_handle_state(tp, time);
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2014-04-08 12:29:45 +02:00
|
|
|
tp_post_process_state(struct tp_dispatch *tp, uint64_t time)
|
2014-02-06 15:32:32 +10:00
|
|
|
{
|
|
|
|
|
struct tp_touch *t;
|
|
|
|
|
|
|
|
|
|
tp_for_each_touch(tp, t) {
|
2014-12-11 16:39:04 +10:00
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
if (!t->dirty)
|
|
|
|
|
continue;
|
|
|
|
|
|
2014-12-11 16:39:04 +10:00
|
|
|
if (t->state == TOUCH_END) {
|
|
|
|
|
if (t->has_ended)
|
|
|
|
|
t->state = TOUCH_NONE;
|
|
|
|
|
else
|
|
|
|
|
t->state = TOUCH_HOVERING;
|
|
|
|
|
} else if (t->state == TOUCH_BEGIN) {
|
2014-02-06 15:32:32 +10:00
|
|
|
t->state = TOUCH_UPDATE;
|
2014-12-11 16:39:04 +10:00
|
|
|
}
|
2014-02-06 15:32:32 +10:00
|
|
|
|
|
|
|
|
t->dirty = false;
|
|
|
|
|
}
|
2014-02-07 13:48:06 +10:00
|
|
|
|
2014-07-21 15:25:47 +02:00
|
|
|
tp->old_nfingers_down = tp->nfingers_down;
|
2014-02-07 13:59:38 +10:00
|
|
|
tp->buttons.old_state = tp->buttons.state;
|
|
|
|
|
|
2014-02-07 13:48:06 +10:00
|
|
|
tp->queued = TOUCHPAD_EVENT_NONE;
|
2014-02-06 15:32:32 +10:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 12:47:09 +01:00
|
|
|
static void
|
|
|
|
|
tp_post_events(struct tp_dispatch *tp, uint64_t time)
|
|
|
|
|
{
|
|
|
|
|
int filter_motion = 0;
|
|
|
|
|
|
|
|
|
|
/* Only post (top) button events while suspended */
|
|
|
|
|
if (tp->device->suspended) {
|
|
|
|
|
tp_post_button_events(tp, time);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
filter_motion |= tp_tap_handle_state(tp, time);
|
|
|
|
|
filter_motion |= tp_post_button_events(tp, time);
|
|
|
|
|
|
2015-04-20 15:51:20 +10:00
|
|
|
if (filter_motion ||
|
|
|
|
|
tp->sendevents.trackpoint_active ||
|
2015-05-25 10:07:51 +10:00
|
|
|
tp->dwt.keyboard_active) {
|
2015-02-16 10:16:30 +01:00
|
|
|
tp_edge_scroll_stop_events(tp, time);
|
2015-02-16 13:30:24 +01:00
|
|
|
tp_gesture_stop(tp, time);
|
2014-12-09 12:47:09 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-16 10:16:30 +01:00
|
|
|
if (tp_edge_scroll_post_events(tp, time) != 0)
|
|
|
|
|
return;
|
2014-12-09 12:47:09 +01:00
|
|
|
|
2015-02-16 13:30:24 +01:00
|
|
|
tp_gesture_post_events(tp, time);
|
2014-12-09 12:47:09 +01:00
|
|
|
}
|
|
|
|
|
|
2014-09-01 17:05:49 +10:00
|
|
|
static void
|
|
|
|
|
tp_handle_state(struct tp_dispatch *tp,
|
|
|
|
|
uint64_t time)
|
|
|
|
|
{
|
|
|
|
|
tp_process_state(tp, time);
|
|
|
|
|
tp_post_events(tp, time);
|
|
|
|
|
tp_post_process_state(tp, time);
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static void
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_process(struct evdev_dispatch *dispatch,
|
|
|
|
|
struct evdev_device *device,
|
|
|
|
|
struct input_event *e,
|
|
|
|
|
uint64_t time)
|
2014-02-06 15:32:32 +10:00
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp =
|
|
|
|
|
(struct tp_dispatch *)dispatch;
|
|
|
|
|
|
|
|
|
|
switch (e->type) {
|
|
|
|
|
case EV_ABS:
|
2014-02-14 14:18:27 +10:00
|
|
|
if (tp->has_mt)
|
|
|
|
|
tp_process_absolute(tp, e, time);
|
|
|
|
|
else
|
|
|
|
|
tp_process_absolute_st(tp, e, time);
|
2014-02-06 15:32:32 +10:00
|
|
|
break;
|
|
|
|
|
case EV_KEY:
|
|
|
|
|
tp_process_key(tp, e, time);
|
|
|
|
|
break;
|
|
|
|
|
case EV_SYN:
|
2014-09-01 17:05:49 +10:00
|
|
|
tp_handle_state(tp, time);
|
2014-02-06 15:32:32 +10:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-28 13:21:08 +02:00
|
|
|
static void
|
2014-12-05 12:50:39 +01:00
|
|
|
tp_remove_sendevents(struct tp_dispatch *tp)
|
2014-09-28 13:21:08 +02:00
|
|
|
{
|
|
|
|
|
libinput_timer_cancel(&tp->sendevents.trackpoint_timer);
|
2015-05-25 10:07:51 +10:00
|
|
|
libinput_timer_cancel(&tp->dwt.keyboard_timer);
|
2014-09-28 13:21:08 +02:00
|
|
|
|
|
|
|
|
if (tp->buttons.trackpoint)
|
|
|
|
|
libinput_device_remove_event_listener(
|
|
|
|
|
&tp->sendevents.trackpoint_listener);
|
2015-04-20 15:51:20 +10:00
|
|
|
|
2015-05-25 10:07:51 +10:00
|
|
|
if (tp->dwt.keyboard)
|
2015-04-20 15:51:20 +10:00
|
|
|
libinput_device_remove_event_listener(
|
2015-05-25 10:07:51 +10:00
|
|
|
&tp->dwt.keyboard_listener);
|
2014-09-28 13:21:08 +02:00
|
|
|
}
|
|
|
|
|
|
2014-12-05 12:50:39 +01:00
|
|
|
static void
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_remove(struct evdev_dispatch *dispatch)
|
2014-12-05 12:50:39 +01:00
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp =
|
|
|
|
|
(struct tp_dispatch*)dispatch;
|
|
|
|
|
|
|
|
|
|
tp_remove_tap(tp);
|
|
|
|
|
tp_remove_buttons(tp);
|
|
|
|
|
tp_remove_sendevents(tp);
|
2015-02-16 10:16:30 +01:00
|
|
|
tp_remove_edge_scroll(tp);
|
2015-02-16 13:30:24 +01:00
|
|
|
tp_remove_gesture(tp);
|
2014-12-05 12:50:39 +01:00
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static void
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_destroy(struct evdev_dispatch *dispatch)
|
2014-02-06 15:32:32 +10:00
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp =
|
|
|
|
|
(struct tp_dispatch*)dispatch;
|
|
|
|
|
|
|
|
|
|
free(tp->touches);
|
|
|
|
|
free(tp);
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-30 14:48:42 +10:00
|
|
|
static void
|
|
|
|
|
tp_release_fake_touches(struct tp_dispatch *tp)
|
|
|
|
|
{
|
|
|
|
|
tp->fake_touches = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-01 17:17:18 +10:00
|
|
|
static void
|
2014-12-04 11:08:01 +10:00
|
|
|
tp_clear_state(struct tp_dispatch *tp)
|
2014-09-01 17:17:18 +10:00
|
|
|
{
|
2015-05-22 15:16:31 +10:00
|
|
|
uint64_t now = libinput_now(tp_libinput_context(tp));
|
2014-09-01 17:17:18 +10:00
|
|
|
struct tp_touch *t;
|
|
|
|
|
|
|
|
|
|
/* Unroll the touchpad state.
|
|
|
|
|
* Release buttons first. If tp is a clickpad, the button event
|
|
|
|
|
* must come before the touch up. If it isn't, the order doesn't
|
|
|
|
|
* matter anyway
|
|
|
|
|
*
|
|
|
|
|
* Then cancel all timeouts on the taps, triggering the last set
|
|
|
|
|
* of events.
|
|
|
|
|
*
|
|
|
|
|
* Then lift all touches so the touchpad is in a neutral state.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
tp_release_all_buttons(tp, now);
|
|
|
|
|
tp_release_all_taps(tp, now);
|
|
|
|
|
|
|
|
|
|
tp_for_each_touch(tp, t) {
|
2014-12-11 16:39:04 +10:00
|
|
|
tp_end_sequence(tp, t, now);
|
2014-09-01 17:17:18 +10:00
|
|
|
}
|
2015-04-30 14:48:42 +10:00
|
|
|
tp_release_fake_touches(tp);
|
2014-09-01 17:17:18 +10:00
|
|
|
|
|
|
|
|
tp_handle_state(tp, now);
|
2014-09-16 16:22:40 +02:00
|
|
|
}
|
2014-09-01 17:17:18 +10:00
|
|
|
|
2014-09-16 16:22:40 +02:00
|
|
|
static void
|
|
|
|
|
tp_suspend(struct tp_dispatch *tp, struct evdev_device *device)
|
|
|
|
|
{
|
2014-12-04 11:08:01 +10:00
|
|
|
tp_clear_state(tp);
|
2014-09-16 16:22:41 +02:00
|
|
|
|
|
|
|
|
/* On devices with top softwarebuttons we don't actually suspend the
|
|
|
|
|
* device, to keep the "trackpoint" buttons working. tp_post_events()
|
|
|
|
|
* will only send events for the trackpoint while suspended.
|
|
|
|
|
*/
|
|
|
|
|
if (tp->buttons.has_topbuttons) {
|
|
|
|
|
evdev_notify_suspended_device(device);
|
2014-09-17 15:35:30 +02:00
|
|
|
/* Enlarge topbutton area while suspended */
|
2015-01-14 14:54:43 +01:00
|
|
|
tp_init_top_softbuttons(tp, device, 1.5);
|
2014-09-16 16:22:41 +02:00
|
|
|
} else {
|
|
|
|
|
evdev_device_suspend(device);
|
|
|
|
|
}
|
2014-09-01 17:17:18 +10:00
|
|
|
}
|
|
|
|
|
|
2015-05-19 15:57:11 +10:00
|
|
|
static void
|
|
|
|
|
tp_interface_suspend(struct evdev_dispatch *dispatch,
|
|
|
|
|
struct evdev_device *device)
|
|
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch *)dispatch;
|
|
|
|
|
|
|
|
|
|
tp_clear_state(tp);
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-01 17:17:18 +10:00
|
|
|
static void
|
|
|
|
|
tp_resume(struct tp_dispatch *tp, struct evdev_device *device)
|
|
|
|
|
{
|
2014-09-16 16:22:41 +02:00
|
|
|
if (tp->buttons.has_topbuttons) {
|
|
|
|
|
/* tap state-machine is offline while suspended, reset state */
|
2014-12-04 11:08:01 +10:00
|
|
|
tp_clear_state(tp);
|
2014-09-17 15:35:30 +02:00
|
|
|
/* restore original topbutton area size */
|
2015-01-14 14:54:43 +01:00
|
|
|
tp_init_top_softbuttons(tp, device, 1.0);
|
2014-09-16 16:22:41 +02:00
|
|
|
evdev_notify_resumed_device(device);
|
|
|
|
|
} else {
|
|
|
|
|
evdev_device_resume(device);
|
|
|
|
|
}
|
2014-09-01 17:17:18 +10:00
|
|
|
}
|
|
|
|
|
|
2014-09-28 13:21:08 +02:00
|
|
|
static void
|
|
|
|
|
tp_trackpoint_timeout(uint64_t now, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = data;
|
|
|
|
|
|
|
|
|
|
tp_tap_resume(tp, now);
|
|
|
|
|
tp->sendevents.trackpoint_active = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
tp_trackpoint_event(uint64_t time, struct libinput_event *event, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = data;
|
|
|
|
|
|
|
|
|
|
/* Buttons do not count as trackpad activity, as people may use
|
|
|
|
|
the trackpoint buttons in combination with the touchpad. */
|
|
|
|
|
if (event->type == LIBINPUT_EVENT_POINTER_BUTTON)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (!tp->sendevents.trackpoint_active) {
|
2015-02-16 10:24:28 +01:00
|
|
|
tp_edge_scroll_stop_events(tp, time);
|
2015-02-16 13:30:24 +01:00
|
|
|
tp_gesture_stop(tp, time);
|
2014-09-28 13:21:08 +02:00
|
|
|
tp_tap_suspend(tp, time);
|
|
|
|
|
tp->sendevents.trackpoint_active = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
libinput_timer_set(&tp->sendevents.trackpoint_timer,
|
|
|
|
|
time + DEFAULT_TRACKPOINT_ACTIVITY_TIMEOUT);
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-20 15:51:20 +10:00
|
|
|
static void
|
|
|
|
|
tp_keyboard_timeout(uint64_t now, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = data;
|
|
|
|
|
|
|
|
|
|
tp_tap_resume(tp, now);
|
2015-05-22 15:14:04 +10:00
|
|
|
|
2015-05-25 10:07:51 +10:00
|
|
|
tp->dwt.keyboard_active = false;
|
2015-05-22 15:14:04 +10:00
|
|
|
|
|
|
|
|
log_debug(tp_libinput_context(tp), "palm: keyboard timeout\n");
|
2015-04-20 15:51:20 +10:00
|
|
|
}
|
|
|
|
|
|
2015-05-25 08:48:25 +10:00
|
|
|
static inline bool
|
|
|
|
|
tp_key_ignore_for_dwt(unsigned int keycode)
|
|
|
|
|
{
|
|
|
|
|
switch (keycode) {
|
|
|
|
|
/* Ignore modifiers to be responsive to ctrl-click, alt-tab, etc. */
|
|
|
|
|
case KEY_LEFTCTRL:
|
|
|
|
|
case KEY_RIGHTCTRL:
|
|
|
|
|
case KEY_LEFTALT:
|
|
|
|
|
case KEY_RIGHTALT:
|
|
|
|
|
case KEY_LEFTSHIFT:
|
|
|
|
|
case KEY_RIGHTSHIFT:
|
|
|
|
|
case KEY_FN:
|
|
|
|
|
case KEY_CAPSLOCK:
|
|
|
|
|
case KEY_TAB:
|
|
|
|
|
case KEY_COMPOSE:
|
|
|
|
|
case KEY_RIGHTMETA:
|
|
|
|
|
case KEY_LEFTMETA:
|
|
|
|
|
return true;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Ignore keys not part of the "typewriter set", i.e. F-keys,
|
|
|
|
|
* multimedia keys, numpad, etc.
|
|
|
|
|
*/
|
|
|
|
|
if (keycode >= KEY_F1)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-20 15:51:20 +10:00
|
|
|
static void
|
|
|
|
|
tp_keyboard_event(uint64_t time, struct libinput_event *event, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = data;
|
|
|
|
|
struct libinput_event_keyboard *kbdev;
|
2015-04-20 16:20:00 +10:00
|
|
|
unsigned int timeout;
|
2015-04-20 15:51:20 +10:00
|
|
|
|
|
|
|
|
if (event->type != LIBINPUT_EVENT_KEYBOARD_KEY)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
kbdev = libinput_event_get_keyboard_event(event);
|
2015-04-20 16:20:00 +10:00
|
|
|
|
|
|
|
|
/* Only trigger the timer on key down. */
|
|
|
|
|
if (libinput_event_keyboard_get_key_state(kbdev) !=
|
|
|
|
|
LIBINPUT_KEY_STATE_PRESSED)
|
|
|
|
|
return;
|
|
|
|
|
|
2015-04-20 15:51:20 +10:00
|
|
|
/* modifier keys don't trigger disable-while-typing so things like
|
|
|
|
|
* ctrl+zoom or ctrl+click are possible */
|
2015-05-25 08:48:25 +10:00
|
|
|
if (tp_key_ignore_for_dwt(libinput_event_keyboard_get_key(kbdev)))
|
|
|
|
|
return;
|
2015-04-20 15:51:20 +10:00
|
|
|
|
2015-05-25 10:07:51 +10:00
|
|
|
if (!tp->dwt.keyboard_active) {
|
2015-04-20 15:51:20 +10:00
|
|
|
tp_edge_scroll_stop_events(tp, time);
|
|
|
|
|
tp_gesture_stop(tp, time);
|
|
|
|
|
tp_tap_suspend(tp, time);
|
2015-05-25 10:07:51 +10:00
|
|
|
tp->dwt.keyboard_active = true;
|
2015-04-20 16:20:00 +10:00
|
|
|
timeout = DEFAULT_KEYBOARD_ACTIVITY_TIMEOUT_1;
|
|
|
|
|
} else {
|
|
|
|
|
timeout = DEFAULT_KEYBOARD_ACTIVITY_TIMEOUT_2;
|
2015-04-20 15:51:20 +10:00
|
|
|
}
|
|
|
|
|
|
2015-05-21 16:32:42 +10:00
|
|
|
tp->dwt.keyboard_last_press_time = time;
|
2015-05-25 10:07:51 +10:00
|
|
|
libinput_timer_set(&tp->dwt.keyboard_timer,
|
2015-04-20 16:20:00 +10:00
|
|
|
time + timeout);
|
2015-04-20 15:51:20 +10:00
|
|
|
}
|
|
|
|
|
|
2015-05-22 16:07:10 +10:00
|
|
|
static bool
|
|
|
|
|
tp_want_dwt(struct evdev_device *touchpad,
|
|
|
|
|
struct evdev_device *keyboard)
|
|
|
|
|
{
|
|
|
|
|
unsigned int bus_tp = libevdev_get_id_bustype(touchpad->evdev),
|
|
|
|
|
bus_kbd = libevdev_get_id_bustype(keyboard->evdev);
|
|
|
|
|
|
|
|
|
|
if (bus_tp == BUS_BLUETOOTH || bus_kbd == BUS_BLUETOOTH)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
/* evemu will set the right bus type */
|
|
|
|
|
if (bus_tp == BUS_VIRTUAL || bus_kbd == BUS_VIRTUAL)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
/* If the touchpad is on serio, the keyboard is too, so ignore any
|
|
|
|
|
other devices */
|
|
|
|
|
if (bus_tp == BUS_I8042 && bus_kbd != bus_tp)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
/* Wacom makes touchpads, but not internal ones */
|
|
|
|
|
if (libevdev_get_id_vendor(touchpad->evdev) == VENDOR_ID_WACOM)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
/* everything else we don't really know, so we have to assume
|
|
|
|
|
they go together */
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-03 15:45:57 +10:00
|
|
|
static void
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_device_added(struct evdev_device *device,
|
|
|
|
|
struct evdev_device *added_device)
|
2014-09-03 15:45:57 +10:00
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch*)device->dispatch;
|
2015-04-14 17:04:45 +10:00
|
|
|
unsigned int bus_tp = libevdev_get_id_bustype(device->evdev),
|
2015-05-22 16:07:10 +10:00
|
|
|
bus_trp = libevdev_get_id_bustype(added_device->evdev);
|
|
|
|
|
bool tp_is_internal, trp_is_internal;
|
2015-04-14 17:04:45 +10:00
|
|
|
|
|
|
|
|
tp_is_internal = bus_tp != BUS_USB && bus_tp != BUS_BLUETOOTH;
|
|
|
|
|
trp_is_internal = bus_trp != BUS_USB && bus_trp != BUS_BLUETOOTH;
|
2014-09-03 15:45:57 +10:00
|
|
|
|
2014-09-16 16:22:38 +02:00
|
|
|
if (tp->buttons.trackpoint == NULL &&
|
2015-04-14 17:04:45 +10:00
|
|
|
(added_device->tags & EVDEV_TAG_TRACKPOINT) &&
|
|
|
|
|
tp_is_internal && trp_is_internal) {
|
2014-09-16 16:22:39 +02:00
|
|
|
/* Don't send any pending releases to the new trackpoint */
|
|
|
|
|
tp->buttons.active_is_topbutton = false;
|
2014-09-16 16:22:38 +02:00
|
|
|
tp->buttons.trackpoint = added_device;
|
2014-09-28 13:21:08 +02:00
|
|
|
libinput_device_add_event_listener(&added_device->base,
|
|
|
|
|
&tp->sendevents.trackpoint_listener,
|
|
|
|
|
tp_trackpoint_event, tp);
|
2014-09-16 16:22:39 +02:00
|
|
|
}
|
2014-09-16 16:22:38 +02:00
|
|
|
|
2015-05-22 16:07:10 +10:00
|
|
|
if (added_device->tags & EVDEV_TAG_KEYBOARD &&
|
|
|
|
|
tp->dwt.keyboard == NULL &&
|
|
|
|
|
tp_want_dwt(device, added_device)) {
|
2015-05-22 15:14:04 +10:00
|
|
|
log_debug(tp_libinput_context(tp),
|
|
|
|
|
"palm: dwt activated with %s<->%s\n",
|
|
|
|
|
device->devname,
|
|
|
|
|
added_device->devname);
|
|
|
|
|
|
2015-05-22 16:07:10 +10:00
|
|
|
libinput_device_add_event_listener(&added_device->base,
|
|
|
|
|
&tp->dwt.keyboard_listener,
|
|
|
|
|
tp_keyboard_event, tp);
|
|
|
|
|
tp->dwt.keyboard = added_device;
|
|
|
|
|
tp->dwt.keyboard_active = false;
|
2015-04-20 15:51:20 +10:00
|
|
|
}
|
|
|
|
|
|
2014-09-03 15:45:57 +10:00
|
|
|
if (tp->sendevents.current_mode !=
|
|
|
|
|
LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (added_device->tags & EVDEV_TAG_EXTERNAL_MOUSE)
|
|
|
|
|
tp_suspend(tp, device);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_device_removed(struct evdev_device *device,
|
|
|
|
|
struct evdev_device *removed_device)
|
2014-09-03 15:45:57 +10:00
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch*)device->dispatch;
|
|
|
|
|
struct libinput_device *dev;
|
|
|
|
|
|
2014-09-16 16:22:39 +02:00
|
|
|
if (removed_device == tp->buttons.trackpoint) {
|
|
|
|
|
/* Clear any pending releases for the trackpoint */
|
|
|
|
|
if (tp->buttons.active && tp->buttons.active_is_topbutton) {
|
|
|
|
|
tp->buttons.active = 0;
|
|
|
|
|
tp->buttons.active_is_topbutton = false;
|
|
|
|
|
}
|
2014-09-28 13:21:08 +02:00
|
|
|
libinput_device_remove_event_listener(
|
|
|
|
|
&tp->sendevents.trackpoint_listener);
|
2014-09-16 16:22:38 +02:00
|
|
|
tp->buttons.trackpoint = NULL;
|
2014-09-16 16:22:39 +02:00
|
|
|
}
|
2014-09-16 16:22:38 +02:00
|
|
|
|
2015-05-25 10:07:51 +10:00
|
|
|
if (removed_device == tp->dwt.keyboard) {
|
2015-04-20 15:51:20 +10:00
|
|
|
libinput_device_remove_event_listener(
|
2015-05-25 10:07:51 +10:00
|
|
|
&tp->dwt.keyboard_listener);
|
|
|
|
|
tp->dwt.keyboard = NULL;
|
2015-04-20 15:51:20 +10:00
|
|
|
}
|
|
|
|
|
|
2014-09-03 15:45:57 +10:00
|
|
|
if (tp->sendevents.current_mode !=
|
|
|
|
|
LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
list_for_each(dev, &device->base.seat->devices_list, link) {
|
|
|
|
|
struct evdev_device *d = (struct evdev_device*)dev;
|
|
|
|
|
if (d != removed_device &&
|
|
|
|
|
(d->tags & EVDEV_TAG_EXTERNAL_MOUSE)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tp_resume(tp, device);
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-02 17:22:41 -04:00
|
|
|
void
|
|
|
|
|
evdev_tag_touchpad(struct evdev_device *device,
|
|
|
|
|
struct udev_device *udev_device)
|
2014-09-03 15:45:57 +10:00
|
|
|
{
|
|
|
|
|
int bustype;
|
|
|
|
|
|
|
|
|
|
/* simple approach: touchpads on USB or Bluetooth are considered
|
|
|
|
|
* external, anything else is internal. Exception is Apple -
|
|
|
|
|
* internal touchpads are connected over USB and it doesn't have
|
|
|
|
|
* external USB touchpads anyway.
|
|
|
|
|
*/
|
|
|
|
|
bustype = libevdev_get_id_bustype(device->evdev);
|
|
|
|
|
if (bustype == BUS_USB) {
|
2015-06-05 10:13:56 +10:00
|
|
|
if (device->model == EVDEV_MODEL_APPLE_TOUCHPAD)
|
2014-09-03 15:45:57 +10:00
|
|
|
device->tags |= EVDEV_TAG_INTERNAL_TOUCHPAD;
|
|
|
|
|
} else if (bustype != BUS_BLUETOOTH)
|
|
|
|
|
device->tags |= EVDEV_TAG_INTERNAL_TOUCHPAD;
|
2015-01-28 15:38:00 +10:00
|
|
|
|
|
|
|
|
if (udev_device_get_property_value(udev_device,
|
|
|
|
|
"TOUCHPAD_HAS_TRACKPOINT_BUTTONS"))
|
|
|
|
|
device->tags |= EVDEV_TAG_TOUCHPAD_TRACKPOINT;
|
2014-09-03 15:45:57 +10:00
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static struct evdev_dispatch_interface tp_interface = {
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_process,
|
2015-05-19 15:57:11 +10:00
|
|
|
tp_interface_suspend,
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_remove,
|
|
|
|
|
tp_interface_destroy,
|
|
|
|
|
tp_interface_device_added,
|
|
|
|
|
tp_interface_device_removed,
|
|
|
|
|
tp_interface_device_removed, /* device_suspended, treat as remove */
|
|
|
|
|
tp_interface_device_added, /* device_resumed, treat as add */
|
2014-02-06 15:05:36 +10:00
|
|
|
};
|
|
|
|
|
|
touchpad: Add clickpad-style software buttons
Almost all non Apple touchpads have visible markings for software button areas,
so limit clickfinger behavior to Apple clickpads, and implement software button
areas for others.
This is a slightly fancier implementation than the simplest model and ported
over from libtouchpad. It implements a state machine for the software buttons
with left and right buttons currently implemented. Buttons are oriented
left-to-right, in a horizontal bar. No random button placement allowed.
In general, the procedure is:
- if a finger sets down in the left button area, a click is a left click
- if a finger sets down in the right button area, a click is a right click
- if a finger leaves the button area, a click is a left click
- if a finger starts outside the button area, a click is a left click
Two timeouts are used to handle buttons more smoothly:
- if a finger sets down in a button area but "immediately" moves over
to a different area, that area takes effect on a click.
- if a finger leaves a button area and "immediately" clicks or moves back into
the area, the button still takes effect on a click.
- if a finger changes between areas and stays there for a timeout, that area
takes effect on a click.
Note the button area states are named BOTTOM_foo to make it easier to later
add support for a top button area such as can be found on the Thinkpad [2-5]40
series.
Co-authored-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Peter Hutterer <peter.hutterer@who-t.net>
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Jonas Ådahl <jadahl@gmail.com>
Reviewed-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Peter Hutterer <peter.hutterer@who-t.net>
2014-03-28 09:44:11 +10:00
|
|
|
static void
|
|
|
|
|
tp_init_touch(struct tp_dispatch *tp,
|
|
|
|
|
struct tp_touch *t)
|
|
|
|
|
{
|
2014-06-06 17:01:06 +02:00
|
|
|
t->tp = tp;
|
2014-12-11 16:39:04 +10:00
|
|
|
t->has_ended = true;
|
touchpad: Add clickpad-style software buttons
Almost all non Apple touchpads have visible markings for software button areas,
so limit clickfinger behavior to Apple clickpads, and implement software button
areas for others.
This is a slightly fancier implementation than the simplest model and ported
over from libtouchpad. It implements a state machine for the software buttons
with left and right buttons currently implemented. Buttons are oriented
left-to-right, in a horizontal bar. No random button placement allowed.
In general, the procedure is:
- if a finger sets down in the left button area, a click is a left click
- if a finger sets down in the right button area, a click is a right click
- if a finger leaves the button area, a click is a left click
- if a finger starts outside the button area, a click is a left click
Two timeouts are used to handle buttons more smoothly:
- if a finger sets down in a button area but "immediately" moves over
to a different area, that area takes effect on a click.
- if a finger leaves a button area and "immediately" clicks or moves back into
the area, the button still takes effect on a click.
- if a finger changes between areas and stays there for a timeout, that area
takes effect on a click.
Note the button area states are named BOTTOM_foo to make it easier to later
add support for a top button area such as can be found on the Thinkpad [2-5]40
series.
Co-authored-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Peter Hutterer <peter.hutterer@who-t.net>
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Jonas Ådahl <jadahl@gmail.com>
Reviewed-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Peter Hutterer <peter.hutterer@who-t.net>
2014-03-28 09:44:11 +10:00
|
|
|
}
|
|
|
|
|
|
2015-05-08 13:30:13 +10:00
|
|
|
static void
|
|
|
|
|
tp_sync_touch(struct tp_dispatch *tp,
|
|
|
|
|
struct evdev_device *device,
|
|
|
|
|
struct tp_touch *t,
|
|
|
|
|
int slot)
|
|
|
|
|
{
|
|
|
|
|
struct libevdev *evdev = device->evdev;
|
|
|
|
|
|
|
|
|
|
if (!libevdev_fetch_slot_value(evdev,
|
|
|
|
|
slot,
|
|
|
|
|
ABS_MT_POSITION_X,
|
|
|
|
|
&t->point.x))
|
|
|
|
|
t->point.x = libevdev_get_event_value(evdev, EV_ABS, ABS_X);
|
|
|
|
|
if (!libevdev_fetch_slot_value(evdev,
|
|
|
|
|
slot,
|
|
|
|
|
ABS_MT_POSITION_Y,
|
|
|
|
|
&t->point.y))
|
|
|
|
|
t->point.y = libevdev_get_event_value(evdev, EV_ABS, ABS_Y);
|
|
|
|
|
|
|
|
|
|
libevdev_fetch_slot_value(evdev, slot, ABS_MT_DISTANCE, &t->distance);
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:05:36 +10:00
|
|
|
static int
|
2014-02-06 15:32:32 +10:00
|
|
|
tp_init_slots(struct tp_dispatch *tp,
|
2014-02-06 15:05:36 +10:00
|
|
|
struct evdev_device *device)
|
|
|
|
|
{
|
2014-02-14 14:18:27 +10:00
|
|
|
const struct input_absinfo *absinfo;
|
2014-07-18 11:06:38 +02:00
|
|
|
struct map {
|
|
|
|
|
unsigned int code;
|
|
|
|
|
int ntouches;
|
|
|
|
|
} max_touches[] = {
|
|
|
|
|
{ BTN_TOOL_QUINTTAP, 5 },
|
|
|
|
|
{ BTN_TOOL_QUADTAP, 4 },
|
|
|
|
|
{ BTN_TOOL_TRIPLETAP, 3 },
|
|
|
|
|
{ BTN_TOOL_DOUBLETAP, 2 },
|
|
|
|
|
};
|
|
|
|
|
struct map *m;
|
|
|
|
|
unsigned int i, n_btn_tool_touches = 1;
|
2014-02-06 15:32:32 +10:00
|
|
|
|
2014-02-14 14:18:27 +10:00
|
|
|
absinfo = libevdev_get_abs_info(device->evdev, ABS_MT_SLOT);
|
|
|
|
|
if (absinfo) {
|
2015-04-09 15:27:34 +10:00
|
|
|
tp->num_slots = absinfo->maximum + 1;
|
2014-02-14 14:18:27 +10:00
|
|
|
tp->slot = absinfo->value;
|
|
|
|
|
tp->has_mt = true;
|
|
|
|
|
} else {
|
2015-04-09 15:27:34 +10:00
|
|
|
tp->num_slots = 1;
|
2014-02-14 14:18:27 +10:00
|
|
|
tp->slot = 0;
|
|
|
|
|
tp->has_mt = false;
|
2014-07-18 11:06:38 +02:00
|
|
|
}
|
2014-04-15 14:27:58 +02:00
|
|
|
|
2014-07-21 15:25:47 +02:00
|
|
|
tp->semi_mt = libevdev_has_property(device->evdev, INPUT_PROP_SEMI_MT);
|
|
|
|
|
|
2014-07-18 11:06:38 +02:00
|
|
|
ARRAY_FOR_EACH(max_touches, m) {
|
|
|
|
|
if (libevdev_has_event_code(device->evdev,
|
|
|
|
|
EV_KEY,
|
|
|
|
|
m->code)) {
|
|
|
|
|
n_btn_tool_touches = m->ntouches;
|
|
|
|
|
break;
|
2014-04-15 14:27:58 +02:00
|
|
|
}
|
2014-02-14 14:18:27 +10:00
|
|
|
}
|
2014-07-18 11:06:38 +02:00
|
|
|
|
2015-04-09 15:27:34 +10:00
|
|
|
tp->ntouches = max(tp->num_slots, n_btn_tool_touches);
|
2014-07-18 11:06:38 +02:00
|
|
|
tp->touches = calloc(tp->ntouches, sizeof(struct tp_touch));
|
2014-04-29 01:26:28 +02:00
|
|
|
if (!tp->touches)
|
|
|
|
|
return -1;
|
2014-02-06 15:32:32 +10:00
|
|
|
|
touchpad: Add clickpad-style software buttons
Almost all non Apple touchpads have visible markings for software button areas,
so limit clickfinger behavior to Apple clickpads, and implement software button
areas for others.
This is a slightly fancier implementation than the simplest model and ported
over from libtouchpad. It implements a state machine for the software buttons
with left and right buttons currently implemented. Buttons are oriented
left-to-right, in a horizontal bar. No random button placement allowed.
In general, the procedure is:
- if a finger sets down in the left button area, a click is a left click
- if a finger sets down in the right button area, a click is a right click
- if a finger leaves the button area, a click is a left click
- if a finger starts outside the button area, a click is a left click
Two timeouts are used to handle buttons more smoothly:
- if a finger sets down in a button area but "immediately" moves over
to a different area, that area takes effect on a click.
- if a finger leaves a button area and "immediately" clicks or moves back into
the area, the button still takes effect on a click.
- if a finger changes between areas and stays there for a timeout, that area
takes effect on a click.
Note the button area states are named BOTTOM_foo to make it easier to later
add support for a top button area such as can be found on the Thinkpad [2-5]40
series.
Co-authored-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Peter Hutterer <peter.hutterer@who-t.net>
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Jonas Ådahl <jadahl@gmail.com>
Reviewed-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Peter Hutterer <peter.hutterer@who-t.net>
2014-03-28 09:44:11 +10:00
|
|
|
for (i = 0; i < tp->ntouches; i++)
|
|
|
|
|
tp_init_touch(tp, &tp->touches[i]);
|
|
|
|
|
|
2015-05-08 13:30:13 +10:00
|
|
|
/* Always sync the first touch so we get ABS_X/Y synced on
|
|
|
|
|
* single-touch touchpads */
|
|
|
|
|
tp_sync_touch(tp, device, &tp->touches[0], 0);
|
|
|
|
|
for (i = 1; i < tp->num_slots; i++)
|
|
|
|
|
tp_sync_touch(tp, device, &tp->touches[i], i);
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-01 14:53:18 +02:00
|
|
|
static int
|
|
|
|
|
tp_init_accel(struct tp_dispatch *tp, double diagonal)
|
2014-05-24 16:53:45 +02:00
|
|
|
{
|
|
|
|
|
int res_x, res_y;
|
2015-04-23 14:32:40 -04:00
|
|
|
accel_profile_func_t profile;
|
2014-05-24 16:53:45 +02:00
|
|
|
|
2015-03-16 13:49:52 +10:00
|
|
|
res_x = tp->device->abs.absinfo_x->resolution;
|
|
|
|
|
res_y = tp->device->abs.absinfo_y->resolution;
|
2014-05-24 16:53:45 +02:00
|
|
|
|
2014-07-01 14:53:18 +02:00
|
|
|
/*
|
|
|
|
|
* Not all touchpads report the same amount of units/mm (resolution).
|
2014-11-28 10:08:02 +10:00
|
|
|
* Normalize motion events to the default mouse DPI as base
|
|
|
|
|
* (unaccelerated) speed. This also evens out any differences in x
|
|
|
|
|
* and y resolution, so that a circle on the
|
2014-07-08 13:43:45 +10:00
|
|
|
* touchpad does not turn into an elipse on the screen.
|
2014-07-01 14:53:18 +02:00
|
|
|
*/
|
2015-06-03 11:30:11 +10:00
|
|
|
if (!tp->device->abs.fake_resolution) {
|
2014-11-28 10:08:02 +10:00
|
|
|
tp->accel.x_scale_coeff = (DEFAULT_MOUSE_DPI/25.4) / res_x;
|
|
|
|
|
tp->accel.y_scale_coeff = (DEFAULT_MOUSE_DPI/25.4) / res_y;
|
2014-05-24 16:53:45 +02:00
|
|
|
} else {
|
2014-07-01 14:53:18 +02:00
|
|
|
/*
|
|
|
|
|
* For touchpads where the driver does not provide resolution, fall
|
|
|
|
|
* back to scaling motion events based on the diagonal size in units.
|
|
|
|
|
*/
|
|
|
|
|
tp->accel.x_scale_coeff = DEFAULT_ACCEL_NUMERATOR / diagonal;
|
|
|
|
|
tp->accel.y_scale_coeff = DEFAULT_ACCEL_NUMERATOR / diagonal;
|
2014-05-24 16:53:45 +02:00
|
|
|
}
|
2014-02-06 19:43:48 +10:00
|
|
|
|
2015-04-23 14:32:40 -04:00
|
|
|
switch (tp->device->model) {
|
|
|
|
|
case EVDEV_MODEL_LENOVO_X230:
|
|
|
|
|
profile = touchpad_lenovo_x230_accel_profile;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
profile = touchpad_accel_profile_linear;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (evdev_device_init_pointer_acceleration(tp->device, profile) == -1)
|
2014-02-06 19:43:48 +10:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2014-02-06 15:32:32 +10:00
|
|
|
|
2014-11-04 10:43:39 +01:00
|
|
|
static uint32_t
|
2014-11-19 12:16:28 +10:00
|
|
|
tp_scroll_config_scroll_method_get_methods(struct libinput_device *device)
|
2014-11-04 10:43:39 +01:00
|
|
|
{
|
|
|
|
|
struct evdev_device *evdev = (struct evdev_device*)device;
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch*)evdev->dispatch;
|
2014-11-19 12:16:28 +10:00
|
|
|
uint32_t methods = LIBINPUT_CONFIG_SCROLL_NO_SCROLL;
|
2014-11-04 10:43:39 +01:00
|
|
|
|
|
|
|
|
if (tp->ntouches >= 2)
|
2014-11-19 12:16:28 +10:00
|
|
|
methods |= LIBINPUT_CONFIG_SCROLL_2FG;
|
2014-11-04 10:43:39 +01:00
|
|
|
|
2014-11-24 12:16:06 +01:00
|
|
|
if (!tp->buttons.is_clickpad)
|
|
|
|
|
methods |= LIBINPUT_CONFIG_SCROLL_EDGE;
|
|
|
|
|
|
2014-11-19 12:16:28 +10:00
|
|
|
return methods;
|
2014-11-04 10:43:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static enum libinput_config_status
|
2014-11-19 12:16:28 +10:00
|
|
|
tp_scroll_config_scroll_method_set_method(struct libinput_device *device,
|
|
|
|
|
enum libinput_config_scroll_method method)
|
2014-11-04 10:43:39 +01:00
|
|
|
{
|
|
|
|
|
struct evdev_device *evdev = (struct evdev_device*)device;
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch*)evdev->dispatch;
|
2015-05-22 15:16:31 +10:00
|
|
|
uint64_t time = libinput_now(tp_libinput_context(tp));
|
2014-11-04 10:43:39 +01:00
|
|
|
|
2014-11-19 12:16:28 +10:00
|
|
|
if (method == tp->scroll.method)
|
2014-11-04 10:43:39 +01:00
|
|
|
return LIBINPUT_CONFIG_STATUS_SUCCESS;
|
|
|
|
|
|
2015-02-16 10:16:30 +01:00
|
|
|
tp_edge_scroll_stop_events(tp, time);
|
2015-02-16 13:30:24 +01:00
|
|
|
tp_gesture_stop_twofinger_scroll(tp, time);
|
2015-02-16 10:16:30 +01:00
|
|
|
|
2014-11-19 12:16:28 +10:00
|
|
|
tp->scroll.method = method;
|
2014-11-04 10:43:39 +01:00
|
|
|
|
|
|
|
|
return LIBINPUT_CONFIG_STATUS_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 12:16:28 +10:00
|
|
|
static enum libinput_config_scroll_method
|
|
|
|
|
tp_scroll_config_scroll_method_get_method(struct libinput_device *device)
|
2014-11-04 10:43:39 +01:00
|
|
|
{
|
|
|
|
|
struct evdev_device *evdev = (struct evdev_device*)device;
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch*)evdev->dispatch;
|
|
|
|
|
|
2014-11-19 12:16:28 +10:00
|
|
|
return tp->scroll.method;
|
2014-11-04 10:43:39 +01:00
|
|
|
}
|
|
|
|
|
|
2014-11-24 12:16:06 +01:00
|
|
|
static enum libinput_config_scroll_method
|
|
|
|
|
tp_scroll_get_default_method(struct tp_dispatch *tp)
|
|
|
|
|
{
|
|
|
|
|
if (tp->ntouches >= 2)
|
|
|
|
|
return LIBINPUT_CONFIG_SCROLL_2FG;
|
|
|
|
|
else
|
|
|
|
|
return LIBINPUT_CONFIG_SCROLL_EDGE;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 12:16:28 +10:00
|
|
|
static enum libinput_config_scroll_method
|
|
|
|
|
tp_scroll_config_scroll_method_get_default_method(struct libinput_device *device)
|
2014-11-04 10:43:39 +01:00
|
|
|
{
|
2014-11-24 12:16:06 +01:00
|
|
|
struct evdev_device *evdev = (struct evdev_device*)device;
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch*)evdev->dispatch;
|
|
|
|
|
|
|
|
|
|
return tp_scroll_get_default_method(tp);
|
2014-11-04 10:43:39 +01:00
|
|
|
}
|
|
|
|
|
|
2014-09-18 15:51:53 +10:00
|
|
|
static int
|
2014-11-18 11:01:10 +10:00
|
|
|
tp_init_scroll(struct tp_dispatch *tp, struct evdev_device *device)
|
2014-09-18 15:51:53 +10:00
|
|
|
{
|
2014-11-24 12:16:06 +01:00
|
|
|
if (tp_edge_scroll_init(tp, device) != 0)
|
|
|
|
|
return -1;
|
2014-09-18 15:51:53 +10:00
|
|
|
|
2014-11-18 11:01:10 +10:00
|
|
|
evdev_init_natural_scroll(device);
|
2014-09-18 15:51:53 +10:00
|
|
|
|
2014-11-19 12:16:28 +10:00
|
|
|
tp->scroll.config_method.get_methods = tp_scroll_config_scroll_method_get_methods;
|
|
|
|
|
tp->scroll.config_method.set_method = tp_scroll_config_scroll_method_set_method;
|
|
|
|
|
tp->scroll.config_method.get_method = tp_scroll_config_scroll_method_get_method;
|
|
|
|
|
tp->scroll.config_method.get_default_method = tp_scroll_config_scroll_method_get_default_method;
|
2014-11-24 12:16:06 +01:00
|
|
|
tp->scroll.method = tp_scroll_get_default_method(tp);
|
2014-11-19 12:16:28 +10:00
|
|
|
tp->device->base.config.scroll_method = &tp->scroll.config_method;
|
2014-11-04 10:43:39 +01:00
|
|
|
|
2014-11-10 09:24:37 +10:00
|
|
|
/* In mm for touchpads with valid resolution, see tp_init_accel() */
|
|
|
|
|
tp->device->scroll.threshold = 5.0;
|
|
|
|
|
|
2014-09-18 15:51:53 +10:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-10 17:34:08 +10:00
|
|
|
static int
|
|
|
|
|
tp_init_palmdetect(struct tp_dispatch *tp,
|
|
|
|
|
struct evdev_device *device)
|
|
|
|
|
{
|
2015-04-15 15:21:08 +10:00
|
|
|
int width, height;
|
2015-05-25 16:02:56 +10:00
|
|
|
unsigned int vendor_id;
|
2014-07-10 17:34:08 +10:00
|
|
|
|
2014-07-21 11:27:21 +10:00
|
|
|
tp->palm.right_edge = INT_MAX;
|
|
|
|
|
tp->palm.left_edge = INT_MIN;
|
2015-04-15 15:21:08 +10:00
|
|
|
tp->palm.vert_center = INT_MIN;
|
2014-07-21 11:27:21 +10:00
|
|
|
|
2014-07-10 17:34:08 +10:00
|
|
|
width = abs(device->abs.absinfo_x->maximum -
|
|
|
|
|
device->abs.absinfo_x->minimum);
|
2015-04-15 15:21:08 +10:00
|
|
|
height = abs(device->abs.absinfo_y->maximum -
|
|
|
|
|
device->abs.absinfo_y->minimum);
|
2014-07-10 17:34:08 +10:00
|
|
|
|
2015-05-25 16:02:56 +10:00
|
|
|
vendor_id = evdev_device_get_id_vendor(device);
|
|
|
|
|
|
|
|
|
|
/* Wacom doesn't have internal touchpads,
|
|
|
|
|
* Apple touchpads are always big enough to warrant palm detection */
|
|
|
|
|
if (vendor_id == VENDOR_ID_WACOM) {
|
|
|
|
|
return 0;
|
2015-06-05 10:13:56 +10:00
|
|
|
} else if (device->model != EVDEV_MODEL_APPLE_TOUCHPAD) {
|
2014-07-21 14:20:35 +10:00
|
|
|
/* We don't know how big the touchpad is */
|
|
|
|
|
if (device->abs.absinfo_x->resolution == 1)
|
|
|
|
|
return 0;
|
|
|
|
|
|
2015-04-14 15:18:19 +10:00
|
|
|
/* Enable palm detection on touchpads >= 70 mm. Anything smaller
|
2014-07-21 14:20:35 +10:00
|
|
|
probably won't need it, until we find out it does */
|
2015-04-16 16:13:01 +10:00
|
|
|
if (width/device->abs.absinfo_x->resolution < 70)
|
2014-07-21 14:20:35 +10:00
|
|
|
return 0;
|
|
|
|
|
}
|
2014-07-15 14:01:00 +10:00
|
|
|
|
2014-07-10 17:34:08 +10:00
|
|
|
/* palm edges are 5% of the width on each side */
|
|
|
|
|
tp->palm.right_edge = device->abs.absinfo_x->maximum - width * 0.05;
|
|
|
|
|
tp->palm.left_edge = device->abs.absinfo_x->minimum + width * 0.05;
|
2015-04-15 15:21:08 +10:00
|
|
|
tp->palm.vert_center = device->abs.absinfo_y->minimum + height/2;
|
2014-07-10 17:34:08 +10:00
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-28 13:21:08 +02:00
|
|
|
static int
|
|
|
|
|
tp_init_sendevents(struct tp_dispatch *tp,
|
|
|
|
|
struct evdev_device *device)
|
|
|
|
|
{
|
|
|
|
|
libinput_timer_init(&tp->sendevents.trackpoint_timer,
|
2015-05-22 15:16:31 +10:00
|
|
|
tp_libinput_context(tp),
|
2014-09-28 13:21:08 +02:00
|
|
|
tp_trackpoint_timeout, tp);
|
2015-04-20 15:51:20 +10:00
|
|
|
|
2015-05-25 10:07:51 +10:00
|
|
|
libinput_timer_init(&tp->dwt.keyboard_timer,
|
2015-05-22 15:16:31 +10:00
|
|
|
tp_libinput_context(tp),
|
2015-04-20 15:51:20 +10:00
|
|
|
tp_keyboard_timeout, tp);
|
2014-09-28 13:21:08 +02:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-28 13:41:58 +10:00
|
|
|
static int
|
|
|
|
|
tp_sanity_check(struct tp_dispatch *tp,
|
|
|
|
|
struct evdev_device *device)
|
|
|
|
|
{
|
|
|
|
|
struct libevdev *evdev = device->evdev;
|
|
|
|
|
struct libinput *libinput = tp_libinput_context(tp);
|
|
|
|
|
|
|
|
|
|
if (!libevdev_has_event_code(evdev, EV_ABS, ABS_X))
|
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
|
|
if (!libevdev_has_event_code(evdev, EV_KEY, BTN_TOUCH))
|
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
|
|
if (!libevdev_has_event_code(evdev, EV_KEY, BTN_TOOL_FINGER))
|
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
error:
|
|
|
|
|
log_bug_kernel(libinput,
|
|
|
|
|
"device %s failed touchpad sanity checks\n");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
static int
|
|
|
|
|
tp_init(struct tp_dispatch *tp,
|
|
|
|
|
struct evdev_device *device)
|
|
|
|
|
{
|
2014-02-06 18:57:10 +10:00
|
|
|
int width, height;
|
|
|
|
|
double diagonal;
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
tp->base.interface = &tp_interface;
|
|
|
|
|
tp->device = device;
|
|
|
|
|
|
2015-05-28 13:41:58 +10:00
|
|
|
if (tp_sanity_check(tp, device) != 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
if (tp_init_slots(tp, device) != 0)
|
|
|
|
|
return -1;
|
2014-02-06 15:05:36 +10:00
|
|
|
|
2014-06-19 11:11:36 +10:00
|
|
|
width = abs(device->abs.absinfo_x->maximum -
|
|
|
|
|
device->abs.absinfo_x->minimum);
|
|
|
|
|
height = abs(device->abs.absinfo_y->maximum -
|
|
|
|
|
device->abs.absinfo_y->minimum);
|
2014-02-06 18:57:10 +10:00
|
|
|
diagonal = sqrt(width*width + height*height);
|
|
|
|
|
|
2015-05-06 19:41:25 -04:00
|
|
|
tp->reports_distance = libevdev_has_event_code(device->evdev,
|
|
|
|
|
EV_ABS,
|
|
|
|
|
ABS_MT_DISTANCE);
|
|
|
|
|
|
2015-03-11 09:24:52 +10:00
|
|
|
tp->hysteresis_margin.x =
|
2014-02-06 18:57:10 +10:00
|
|
|
diagonal / DEFAULT_HYSTERESIS_MARGIN_DENOMINATOR;
|
2015-03-11 09:24:52 +10:00
|
|
|
tp->hysteresis_margin.y =
|
2014-02-06 18:57:10 +10:00
|
|
|
diagonal / DEFAULT_HYSTERESIS_MARGIN_DENOMINATOR;
|
|
|
|
|
|
2014-02-06 19:43:48 +10:00
|
|
|
if (tp_init_accel(tp, diagonal) != 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2014-02-07 15:18:17 +10:00
|
|
|
if (tp_init_tap(tp) != 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2014-03-27 13:09:06 +10:00
|
|
|
if (tp_init_buttons(tp, device) != 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2014-07-10 17:34:08 +10:00
|
|
|
if (tp_init_palmdetect(tp, device) != 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2014-09-28 13:21:08 +02:00
|
|
|
if (tp_init_sendevents(tp, device) != 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2014-11-18 11:01:10 +10:00
|
|
|
if (tp_init_scroll(tp, device) != 0)
|
2014-11-04 10:43:39 +01:00
|
|
|
return -1;
|
|
|
|
|
|
2015-02-16 13:30:24 +01:00
|
|
|
if (tp_init_gesture(tp) != 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2014-07-26 21:34:47 +02:00
|
|
|
device->seat_caps |= EVDEV_DEVICE_POINTER;
|
|
|
|
|
|
2014-02-06 15:05:36 +10:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-01 17:17:18 +10:00
|
|
|
static uint32_t
|
|
|
|
|
tp_sendevents_get_modes(struct libinput_device *device)
|
|
|
|
|
{
|
2014-09-03 15:45:57 +10:00
|
|
|
struct evdev_device *evdev = (struct evdev_device*)device;
|
2014-10-30 15:36:52 +10:00
|
|
|
uint32_t modes = LIBINPUT_CONFIG_SEND_EVENTS_DISABLED;
|
2014-09-03 15:45:57 +10:00
|
|
|
|
|
|
|
|
if (evdev->tags & EVDEV_TAG_INTERNAL_TOUCHPAD)
|
|
|
|
|
modes |= LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE;
|
|
|
|
|
|
|
|
|
|
return modes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
tp_suspend_conditional(struct tp_dispatch *tp,
|
|
|
|
|
struct evdev_device *device)
|
|
|
|
|
{
|
|
|
|
|
struct libinput_device *dev;
|
|
|
|
|
|
|
|
|
|
list_for_each(dev, &device->base.seat->devices_list, link) {
|
|
|
|
|
struct evdev_device *d = (struct evdev_device*)dev;
|
|
|
|
|
if (d->tags & EVDEV_TAG_EXTERNAL_MOUSE) {
|
|
|
|
|
tp_suspend(tp, device);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-09-01 17:17:18 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static enum libinput_config_status
|
|
|
|
|
tp_sendevents_set_mode(struct libinput_device *device,
|
|
|
|
|
enum libinput_config_send_events_mode mode)
|
|
|
|
|
{
|
|
|
|
|
struct evdev_device *evdev = (struct evdev_device*)device;
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch*)evdev->dispatch;
|
|
|
|
|
|
2014-10-30 15:36:52 +10:00
|
|
|
/* DISABLED overrides any DISABLED_ON_ */
|
|
|
|
|
if ((mode & LIBINPUT_CONFIG_SEND_EVENTS_DISABLED) &&
|
|
|
|
|
(mode & LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE))
|
|
|
|
|
mode &= ~LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE;
|
|
|
|
|
|
2014-09-01 17:17:18 +10:00
|
|
|
if (mode == tp->sendevents.current_mode)
|
|
|
|
|
return LIBINPUT_CONFIG_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
|
|
switch(mode) {
|
|
|
|
|
case LIBINPUT_CONFIG_SEND_EVENTS_ENABLED:
|
|
|
|
|
tp_resume(tp, evdev);
|
|
|
|
|
break;
|
|
|
|
|
case LIBINPUT_CONFIG_SEND_EVENTS_DISABLED:
|
|
|
|
|
tp_suspend(tp, evdev);
|
|
|
|
|
break;
|
2014-09-03 15:45:57 +10:00
|
|
|
case LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE:
|
|
|
|
|
tp_suspend_conditional(tp, evdev);
|
|
|
|
|
break;
|
2014-09-01 17:17:18 +10:00
|
|
|
default:
|
|
|
|
|
return LIBINPUT_CONFIG_STATUS_UNSUPPORTED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tp->sendevents.current_mode = mode;
|
|
|
|
|
|
|
|
|
|
return LIBINPUT_CONFIG_STATUS_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static enum libinput_config_send_events_mode
|
|
|
|
|
tp_sendevents_get_mode(struct libinput_device *device)
|
|
|
|
|
{
|
|
|
|
|
struct evdev_device *evdev = (struct evdev_device*)device;
|
|
|
|
|
struct tp_dispatch *dispatch = (struct tp_dispatch*)evdev->dispatch;
|
|
|
|
|
|
|
|
|
|
return dispatch->sendevents.current_mode;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static enum libinput_config_send_events_mode
|
|
|
|
|
tp_sendevents_get_default_mode(struct libinput_device *device)
|
|
|
|
|
{
|
|
|
|
|
return LIBINPUT_CONFIG_SEND_EVENTS_ENABLED;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-23 14:35:42 +10:00
|
|
|
static void
|
|
|
|
|
tp_change_to_left_handed(struct evdev_device *device)
|
|
|
|
|
{
|
|
|
|
|
struct tp_dispatch *tp = (struct tp_dispatch *)device->dispatch;
|
|
|
|
|
|
2015-01-06 21:20:22 -05:00
|
|
|
if (device->left_handed.want_enabled == device->left_handed.enabled)
|
2014-09-23 14:35:42 +10:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (tp->buttons.state & 0x3) /* BTN_LEFT|BTN_RIGHT */
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* tapping and clickfinger aren't affected by left-handed config,
|
|
|
|
|
* so checking physical buttons is enough */
|
|
|
|
|
|
2015-01-06 21:20:22 -05:00
|
|
|
device->left_handed.enabled = device->left_handed.want_enabled;
|
2014-09-23 14:35:42 +10:00
|
|
|
}
|
|
|
|
|
|
2014-11-24 12:16:05 +01:00
|
|
|
struct model_lookup_t {
|
|
|
|
|
uint16_t vendor;
|
|
|
|
|
uint16_t product_start;
|
|
|
|
|
uint16_t product_end;
|
|
|
|
|
enum touchpad_model model;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static struct model_lookup_t model_lookup_table[] = {
|
|
|
|
|
{ 0x0002, 0x0007, 0x0007, MODEL_SYNAPTICS },
|
|
|
|
|
{ 0x0002, 0x0008, 0x0008, MODEL_ALPS },
|
|
|
|
|
{ 0x0002, 0x000e, 0x000e, MODEL_ELANTECH },
|
|
|
|
|
{ 0x05ac, 0, 0x0222, MODEL_APPLETOUCH },
|
|
|
|
|
{ 0x05ac, 0x0223, 0x0228, MODEL_UNIBODY_MACBOOK },
|
|
|
|
|
{ 0x05ac, 0x0229, 0x022b, MODEL_APPLETOUCH },
|
|
|
|
|
{ 0x05ac, 0x022c, 0xffff, MODEL_UNIBODY_MACBOOK },
|
|
|
|
|
{ 0, 0, 0, 0 }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static enum touchpad_model
|
|
|
|
|
tp_get_model(struct evdev_device *device)
|
|
|
|
|
{
|
|
|
|
|
struct model_lookup_t *lookup;
|
|
|
|
|
uint16_t vendor = libevdev_get_id_vendor(device->evdev);
|
|
|
|
|
uint16_t product = libevdev_get_id_product(device->evdev);
|
|
|
|
|
|
|
|
|
|
for (lookup = model_lookup_table; lookup->vendor; lookup++) {
|
|
|
|
|
if (lookup->vendor == vendor &&
|
|
|
|
|
lookup->product_start <= product &&
|
|
|
|
|
product <= lookup->product_end)
|
|
|
|
|
return lookup->model;
|
|
|
|
|
}
|
|
|
|
|
return MODEL_UNKNOWN;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-06 15:05:36 +10:00
|
|
|
struct evdev_dispatch *
|
|
|
|
|
evdev_mt_touchpad_create(struct evdev_device *device)
|
|
|
|
|
{
|
2014-02-06 15:32:32 +10:00
|
|
|
struct tp_dispatch *tp;
|
2014-02-06 15:05:36 +10:00
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
tp = zalloc(sizeof *tp);
|
|
|
|
|
if (!tp)
|
2014-02-06 15:05:36 +10:00
|
|
|
return NULL;
|
|
|
|
|
|
2014-11-24 12:16:05 +01:00
|
|
|
tp->model = tp_get_model(device);
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
if (tp_init(tp, device) != 0) {
|
2015-05-19 15:52:17 +10:00
|
|
|
tp_interface_destroy(&tp->base);
|
2014-02-06 15:05:36 +10:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-01 17:17:18 +10:00
|
|
|
device->base.config.sendevents = &tp->sendevents.config;
|
|
|
|
|
|
|
|
|
|
tp->sendevents.current_mode = LIBINPUT_CONFIG_SEND_EVENTS_ENABLED;
|
|
|
|
|
tp->sendevents.config.get_modes = tp_sendevents_get_modes;
|
|
|
|
|
tp->sendevents.config.set_mode = tp_sendevents_set_mode;
|
|
|
|
|
tp->sendevents.config.get_mode = tp_sendevents_get_mode;
|
|
|
|
|
tp->sendevents.config.get_default_mode = tp_sendevents_get_default_mode;
|
|
|
|
|
|
2014-09-23 14:35:42 +10:00
|
|
|
evdev_init_left_handed(device, tp_change_to_left_handed);
|
|
|
|
|
|
2014-02-06 15:32:32 +10:00
|
|
|
return &tp->base;
|
2014-02-06 15:05:36 +10:00
|
|
|
}
|