2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* ALPS touchpad PS/2 mouse driver
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003 Neil Brown <neilb@cse.unsw.edu.au>
|
2005-07-11 14:08:04 +08:00
|
|
|
* Copyright (c) 2003-2005 Peter Osterlund <petero2@telia.com>
|
2005-04-17 06:20:36 +08:00
|
|
|
* Copyright (c) 2004 Dmitry Torokhov <dtor@mail.ru>
|
|
|
|
* Copyright (c) 2005 Vojtech Pavlik <vojtech@suse.cz>
|
2009-12-16 00:39:50 +08:00
|
|
|
* Copyright (c) 2009 Sebastian Kapfer <sebastian_kapfer@gmx.net>
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
* ALPS detection, tap switching and status querying info is taken from
|
|
|
|
* tpconfig utility (by C. Scott Ananian and Bruce Kall).
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 as published by
|
|
|
|
* the Free Software Foundation.
|
|
|
|
*/
|
|
|
|
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <linux/input.h>
|
2011-11-08 11:54:13 +08:00
|
|
|
#include <linux/input/mt.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <linux/serio.h>
|
|
|
|
#include <linux/libps2.h>
|
|
|
|
|
|
|
|
#include "psmouse.h"
|
|
|
|
#include "alps.h"
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
/*
|
|
|
|
* Definitions for ALPS version 3 and 4 command mode protocol
|
|
|
|
*/
|
|
|
|
#define ALPS_CMD_NIBBLE_10 0x01f2
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
#define ALPS_REG_BASE_RUSHMORE 0xc2c0
|
|
|
|
#define ALPS_REG_BASE_PINNACLE 0x0000
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
static const struct alps_nibble_commands alps_v3_nibble_commands[] = {
|
|
|
|
{ PSMOUSE_CMD_SETPOLL, 0x00 }, /* 0 */
|
|
|
|
{ PSMOUSE_CMD_RESET_DIS, 0x00 }, /* 1 */
|
|
|
|
{ PSMOUSE_CMD_SETSCALE21, 0x00 }, /* 2 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x0a }, /* 3 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x14 }, /* 4 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x28 }, /* 5 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x3c }, /* 6 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x50 }, /* 7 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x64 }, /* 8 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0xc8 }, /* 9 */
|
|
|
|
{ ALPS_CMD_NIBBLE_10, 0x00 }, /* a */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x00 }, /* b */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x01 }, /* c */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x02 }, /* d */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x03 }, /* e */
|
|
|
|
{ PSMOUSE_CMD_SETSCALE11, 0x00 }, /* f */
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct alps_nibble_commands alps_v4_nibble_commands[] = {
|
|
|
|
{ PSMOUSE_CMD_ENABLE, 0x00 }, /* 0 */
|
|
|
|
{ PSMOUSE_CMD_RESET_DIS, 0x00 }, /* 1 */
|
|
|
|
{ PSMOUSE_CMD_SETSCALE21, 0x00 }, /* 2 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x0a }, /* 3 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x14 }, /* 4 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x28 }, /* 5 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x3c }, /* 6 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x50 }, /* 7 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0x64 }, /* 8 */
|
|
|
|
{ PSMOUSE_CMD_SETRATE, 0xc8 }, /* 9 */
|
|
|
|
{ ALPS_CMD_NIBBLE_10, 0x00 }, /* a */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x00 }, /* b */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x01 }, /* c */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x02 }, /* d */
|
|
|
|
{ PSMOUSE_CMD_SETRES, 0x03 }, /* e */
|
|
|
|
{ PSMOUSE_CMD_SETSCALE11, 0x00 }, /* f */
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-11-17 14:12:22 +08:00
|
|
|
#define ALPS_DUALPOINT 0x02 /* touchpad has trackstick */
|
|
|
|
#define ALPS_PASS 0x04 /* device has a pass-through port */
|
|
|
|
|
|
|
|
#define ALPS_WHEEL 0x08 /* hardware wheel present */
|
|
|
|
#define ALPS_FW_BK_1 0x10 /* front & back buttons present */
|
|
|
|
#define ALPS_FW_BK_2 0x20 /* front & back buttons present */
|
|
|
|
#define ALPS_FOUR_BUTTONS 0x40 /* 4 direction button present */
|
2009-12-16 00:39:50 +08:00
|
|
|
#define ALPS_PS2_INTERLEAVED 0x80 /* 3-byte PS/2 packet interleaved with
|
|
|
|
6-byte ALPS packet */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-09-11 09:54:39 +08:00
|
|
|
static const struct alps_model_info alps_model_data[] = {
|
2011-11-08 11:53:36 +08:00
|
|
|
{ { 0x32, 0x02, 0x14 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, ALPS_PASS | ALPS_DUALPOINT }, /* Toshiba Salellite Pro M10 */
|
|
|
|
{ { 0x33, 0x02, 0x0a }, 0x00, ALPS_PROTO_V1, 0x88, 0xf8, 0 }, /* UMAX-530T */
|
|
|
|
{ { 0x53, 0x02, 0x0a }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, 0 },
|
|
|
|
{ { 0x53, 0x02, 0x14 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, 0 },
|
|
|
|
{ { 0x60, 0x03, 0xc8 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, 0 }, /* HP ze1115 */
|
|
|
|
{ { 0x63, 0x02, 0x0a }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, 0 },
|
|
|
|
{ { 0x63, 0x02, 0x14 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, 0 },
|
|
|
|
{ { 0x63, 0x02, 0x28 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, ALPS_FW_BK_2 }, /* Fujitsu Siemens S6010 */
|
|
|
|
{ { 0x63, 0x02, 0x3c }, 0x00, ALPS_PROTO_V2, 0x8f, 0x8f, ALPS_WHEEL }, /* Toshiba Satellite S2400-103 */
|
|
|
|
{ { 0x63, 0x02, 0x50 }, 0x00, ALPS_PROTO_V2, 0xef, 0xef, ALPS_FW_BK_1 }, /* NEC Versa L320 */
|
|
|
|
{ { 0x63, 0x02, 0x64 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, 0 },
|
|
|
|
{ { 0x63, 0x03, 0xc8 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, ALPS_PASS | ALPS_DUALPOINT }, /* Dell Latitude D800 */
|
|
|
|
{ { 0x73, 0x00, 0x0a }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, ALPS_DUALPOINT }, /* ThinkPad R61 8918-5QG */
|
|
|
|
{ { 0x73, 0x02, 0x0a }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, 0 },
|
|
|
|
{ { 0x73, 0x02, 0x14 }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, ALPS_FW_BK_2 }, /* Ahtec Laptop */
|
|
|
|
{ { 0x20, 0x02, 0x0e }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, ALPS_PASS | ALPS_DUALPOINT }, /* XXX */
|
|
|
|
{ { 0x22, 0x02, 0x0a }, 0x00, ALPS_PROTO_V2, 0xf8, 0xf8, ALPS_PASS | ALPS_DUALPOINT },
|
|
|
|
{ { 0x22, 0x02, 0x14 }, 0x00, ALPS_PROTO_V2, 0xff, 0xff, ALPS_PASS | ALPS_DUALPOINT }, /* Dell Latitude D600 */
|
2009-12-16 00:39:50 +08:00
|
|
|
/* Dell Latitude E5500, E6400, E6500, Precision M4400 */
|
2011-11-08 11:53:36 +08:00
|
|
|
{ { 0x62, 0x02, 0x14 }, 0x00, ALPS_PROTO_V2, 0xcf, 0xcf,
|
2009-12-16 00:39:50 +08:00
|
|
|
ALPS_PASS | ALPS_DUALPOINT | ALPS_PS2_INTERLEAVED },
|
2013-10-31 15:55:58 +08:00
|
|
|
{ { 0x73, 0x00, 0x14 }, 0x00, ALPS_PROTO_V2, 0xcf, 0xcf, ALPS_DUALPOINT }, /* Dell XT2 */
|
2011-11-08 11:53:36 +08:00
|
|
|
{ { 0x73, 0x02, 0x50 }, 0x00, ALPS_PROTO_V2, 0xcf, 0xcf, ALPS_FOUR_BUTTONS }, /* Dell Vostro 1400 */
|
|
|
|
{ { 0x52, 0x01, 0x14 }, 0x00, ALPS_PROTO_V2, 0xff, 0xff,
|
|
|
|
ALPS_PASS | ALPS_DUALPOINT | ALPS_PS2_INTERLEAVED }, /* Toshiba Tecra A11-11L */
|
|
|
|
{ { 0x73, 0x02, 0x64 }, 0x8a, ALPS_PROTO_V4, 0x8f, 0x8f, 0 },
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2013-02-14 14:22:08 +08:00
|
|
|
static void alps_set_abs_params_st(struct alps_data *priv,
|
|
|
|
struct input_dev *dev1);
|
|
|
|
static void alps_set_abs_params_mt(struct alps_data *priv,
|
|
|
|
struct input_dev *dev1);
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* XXX - this entry is suspicious. First byte has zero lower nibble,
|
|
|
|
* which is what a normal mouse would report. Also, the value 0x0e
|
|
|
|
* isn't valid per PS/2 spec.
|
|
|
|
*/
|
|
|
|
|
2011-11-08 11:53:15 +08:00
|
|
|
/* Packet formats are described in Documentation/input/alps.txt */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
static bool alps_is_valid_first_byte(struct alps_data *priv,
|
2009-12-16 00:39:50 +08:00
|
|
|
unsigned char data)
|
|
|
|
{
|
2013-02-14 12:56:33 +08:00
|
|
|
return (data & priv->mask0) == priv->byte0;
|
2009-12-16 00:39:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_report_buttons(struct psmouse *psmouse,
|
|
|
|
struct input_dev *dev1, struct input_dev *dev2,
|
|
|
|
int left, int right, int middle)
|
|
|
|
{
|
2010-03-14 14:23:58 +08:00
|
|
|
struct input_dev *dev;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If shared button has already been reported on the
|
|
|
|
* other device (dev2) then this event should be also
|
|
|
|
* sent through that device.
|
|
|
|
*/
|
|
|
|
dev = test_bit(BTN_LEFT, dev2->key) ? dev2 : dev1;
|
|
|
|
input_report_key(dev, BTN_LEFT, left);
|
|
|
|
|
|
|
|
dev = test_bit(BTN_RIGHT, dev2->key) ? dev2 : dev1;
|
|
|
|
input_report_key(dev, BTN_RIGHT, right);
|
|
|
|
|
|
|
|
dev = test_bit(BTN_MIDDLE, dev2->key) ? dev2 : dev1;
|
|
|
|
input_report_key(dev, BTN_MIDDLE, middle);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Sync the _other_ device now, we'll do the first
|
|
|
|
* device later once we report the rest of the events.
|
|
|
|
*/
|
|
|
|
input_sync(dev2);
|
2009-12-16 00:39:50 +08:00
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
static void alps_process_packet_v1_v2(struct psmouse *psmouse)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
unsigned char *packet = psmouse->packet;
|
2005-09-15 15:01:44 +08:00
|
|
|
struct input_dev *dev = psmouse->dev;
|
|
|
|
struct input_dev *dev2 = priv->dev2;
|
2005-04-17 06:20:36 +08:00
|
|
|
int x, y, z, ges, fin, left, right, middle;
|
2005-06-01 15:39:18 +08:00
|
|
|
int back = 0, forward = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->proto_version == ALPS_PROTO_V1) {
|
2006-05-30 11:30:36 +08:00
|
|
|
left = packet[2] & 0x10;
|
|
|
|
right = packet[2] & 0x08;
|
2005-04-17 06:20:36 +08:00
|
|
|
middle = 0;
|
|
|
|
x = packet[1] | ((packet[0] & 0x07) << 7);
|
|
|
|
y = packet[4] | ((packet[3] & 0x07) << 7);
|
|
|
|
z = packet[5];
|
|
|
|
} else {
|
|
|
|
left = packet[3] & 1;
|
|
|
|
right = packet[3] & 2;
|
|
|
|
middle = packet[3] & 4;
|
|
|
|
x = packet[1] | ((packet[2] & 0x78) << (7 - 3));
|
|
|
|
y = packet[4] | ((packet[3] & 0x70) << (7 - 4));
|
|
|
|
z = packet[5];
|
|
|
|
}
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_FW_BK_1) {
|
2008-03-18 12:39:55 +08:00
|
|
|
back = packet[0] & 0x10;
|
|
|
|
forward = packet[2] & 4;
|
2005-06-01 15:39:18 +08:00
|
|
|
}
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_FW_BK_2) {
|
2005-06-01 15:39:18 +08:00
|
|
|
back = packet[3] & 4;
|
|
|
|
forward = packet[2] & 4;
|
|
|
|
if ((middle = forward && back))
|
|
|
|
forward = back = 0;
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
ges = packet[2] & 1;
|
|
|
|
fin = packet[2] & 2;
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if ((priv->flags & ALPS_DUALPOINT) && z == 127) {
|
2005-04-17 06:20:36 +08:00
|
|
|
input_report_rel(dev2, REL_X, (x > 383 ? (x - 768) : x));
|
|
|
|
input_report_rel(dev2, REL_Y, -(y > 255 ? (y - 512) : y));
|
2009-06-11 15:15:09 +08:00
|
|
|
|
2009-12-16 00:39:50 +08:00
|
|
|
alps_report_buttons(psmouse, dev2, dev, left, right, middle);
|
2009-06-11 15:15:09 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
input_sync(dev2);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-12-16 00:39:50 +08:00
|
|
|
alps_report_buttons(psmouse, dev, dev2, left, right, middle);
|
2009-06-11 15:15:09 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Convert hardware tap to a reasonable Z value */
|
2009-11-17 14:12:22 +08:00
|
|
|
if (ges && !fin)
|
|
|
|
z = 40;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* A "tap and drag" operation is reported by the hardware as a transition
|
|
|
|
* from (!fin && ges) to (fin && ges). This should be translated to the
|
|
|
|
* sequence Z>0, Z==0, Z>0, so the Z==0 event has to be generated manually.
|
|
|
|
*/
|
|
|
|
if (ges && fin && !priv->prev_fin) {
|
|
|
|
input_report_abs(dev, ABS_X, x);
|
|
|
|
input_report_abs(dev, ABS_Y, y);
|
|
|
|
input_report_abs(dev, ABS_PRESSURE, 0);
|
|
|
|
input_report_key(dev, BTN_TOOL_FINGER, 0);
|
|
|
|
input_sync(dev);
|
|
|
|
}
|
|
|
|
priv->prev_fin = fin;
|
|
|
|
|
2009-11-17 14:12:22 +08:00
|
|
|
if (z > 30)
|
|
|
|
input_report_key(dev, BTN_TOUCH, 1);
|
|
|
|
if (z < 25)
|
|
|
|
input_report_key(dev, BTN_TOUCH, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (z > 0) {
|
|
|
|
input_report_abs(dev, ABS_X, x);
|
|
|
|
input_report_abs(dev, ABS_Y, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
input_report_abs(dev, ABS_PRESSURE, z);
|
|
|
|
input_report_key(dev, BTN_TOOL_FINGER, z > 0);
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_WHEEL)
|
2005-09-04 14:40:43 +08:00
|
|
|
input_report_rel(dev, REL_WHEEL, ((packet[2] << 1) & 0x08) - ((packet[0] >> 4) & 0x07));
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & (ALPS_FW_BK_1 | ALPS_FW_BK_2)) {
|
2005-06-01 15:39:18 +08:00
|
|
|
input_report_key(dev, BTN_FORWARD, forward);
|
|
|
|
input_report_key(dev, BTN_BACK, back);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_FOUR_BUTTONS) {
|
2009-11-17 14:12:22 +08:00
|
|
|
input_report_key(dev, BTN_0, packet[2] & 4);
|
|
|
|
input_report_key(dev, BTN_1, packet[0] & 0x10);
|
|
|
|
input_report_key(dev, BTN_2, packet[3] & 4);
|
|
|
|
input_report_key(dev, BTN_3, packet[0] & 0x20);
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
input_sync(dev);
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:54:13 +08:00
|
|
|
/*
|
|
|
|
* Process bitmap data from v3 and v4 protocols. Returns the number of
|
|
|
|
* fingers detected. A return value of 0 means at least one of the
|
|
|
|
* bitmaps was empty.
|
|
|
|
*
|
|
|
|
* The bitmaps don't have enough data to track fingers, so this function
|
|
|
|
* only generates points representing a bounding box of all contacts.
|
|
|
|
* These points are returned in x1, y1, x2, and y2 when the return value
|
|
|
|
* is greater than 0.
|
|
|
|
*/
|
2013-02-14 14:24:55 +08:00
|
|
|
static int alps_process_bitmap(struct alps_data *priv,
|
|
|
|
unsigned int x_map, unsigned int y_map,
|
2011-11-08 11:54:13 +08:00
|
|
|
int *x1, int *y1, int *x2, int *y2)
|
|
|
|
{
|
|
|
|
struct alps_bitmap_point {
|
|
|
|
int start_bit;
|
|
|
|
int num_bits;
|
|
|
|
};
|
|
|
|
|
|
|
|
int fingers_x = 0, fingers_y = 0, fingers;
|
|
|
|
int i, bit, prev_bit;
|
|
|
|
struct alps_bitmap_point x_low = {0,}, x_high = {0,};
|
|
|
|
struct alps_bitmap_point y_low = {0,}, y_high = {0,};
|
|
|
|
struct alps_bitmap_point *point;
|
|
|
|
|
|
|
|
if (!x_map || !y_map)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
*x1 = *y1 = *x2 = *y2 = 0;
|
|
|
|
|
|
|
|
prev_bit = 0;
|
|
|
|
point = &x_low;
|
|
|
|
for (i = 0; x_map != 0; i++, x_map >>= 1) {
|
|
|
|
bit = x_map & 1;
|
|
|
|
if (bit) {
|
|
|
|
if (!prev_bit) {
|
|
|
|
point->start_bit = i;
|
|
|
|
fingers_x++;
|
|
|
|
}
|
|
|
|
point->num_bits++;
|
|
|
|
} else {
|
|
|
|
if (prev_bit)
|
|
|
|
point = &x_high;
|
|
|
|
else
|
|
|
|
point->num_bits = 0;
|
|
|
|
}
|
|
|
|
prev_bit = bit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* y bitmap is reversed for what we need (lower positions are in
|
|
|
|
* higher bits), so we process from the top end.
|
|
|
|
*/
|
2013-02-14 14:24:55 +08:00
|
|
|
y_map = y_map << (sizeof(y_map) * BITS_PER_BYTE - priv->y_bits);
|
2011-11-08 11:54:13 +08:00
|
|
|
prev_bit = 0;
|
|
|
|
point = &y_low;
|
|
|
|
for (i = 0; y_map != 0; i++, y_map <<= 1) {
|
|
|
|
bit = y_map & (1 << (sizeof(y_map) * BITS_PER_BYTE - 1));
|
|
|
|
if (bit) {
|
|
|
|
if (!prev_bit) {
|
|
|
|
point->start_bit = i;
|
|
|
|
fingers_y++;
|
|
|
|
}
|
|
|
|
point->num_bits++;
|
|
|
|
} else {
|
|
|
|
if (prev_bit)
|
|
|
|
point = &y_high;
|
|
|
|
else
|
|
|
|
point->num_bits = 0;
|
|
|
|
}
|
|
|
|
prev_bit = bit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fingers can overlap, so we use the maximum count of fingers
|
|
|
|
* on either axis as the finger count.
|
|
|
|
*/
|
|
|
|
fingers = max(fingers_x, fingers_y);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If total fingers is > 1 but either axis reports only a single
|
|
|
|
* contact, we have overlapping or adjacent fingers. For the
|
|
|
|
* purposes of creating a bounding box, divide the single contact
|
|
|
|
* (roughly) equally between the two points.
|
|
|
|
*/
|
|
|
|
if (fingers > 1) {
|
|
|
|
if (fingers_x == 1) {
|
|
|
|
i = x_low.num_bits / 2;
|
|
|
|
x_low.num_bits = x_low.num_bits - i;
|
|
|
|
x_high.start_bit = x_low.start_bit + i;
|
|
|
|
x_high.num_bits = max(i, 1);
|
|
|
|
} else if (fingers_y == 1) {
|
|
|
|
i = y_low.num_bits / 2;
|
|
|
|
y_low.num_bits = y_low.num_bits - i;
|
|
|
|
y_high.start_bit = y_low.start_bit + i;
|
|
|
|
y_high.num_bits = max(i, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:24:55 +08:00
|
|
|
*x1 = (priv->x_max * (2 * x_low.start_bit + x_low.num_bits - 1)) /
|
|
|
|
(2 * (priv->x_bits - 1));
|
|
|
|
*y1 = (priv->y_max * (2 * y_low.start_bit + y_low.num_bits - 1)) /
|
|
|
|
(2 * (priv->y_bits - 1));
|
2011-11-08 11:54:13 +08:00
|
|
|
|
|
|
|
if (fingers > 1) {
|
2013-02-14 14:24:55 +08:00
|
|
|
*x2 = (priv->x_max *
|
|
|
|
(2 * x_high.start_bit + x_high.num_bits - 1)) /
|
|
|
|
(2 * (priv->x_bits - 1));
|
|
|
|
*y2 = (priv->y_max *
|
|
|
|
(2 * y_high.start_bit + y_high.num_bits - 1)) /
|
|
|
|
(2 * (priv->y_bits - 1));
|
2011-11-08 11:54:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return fingers;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_set_slot(struct input_dev *dev, int slot, bool active,
|
|
|
|
int x, int y)
|
|
|
|
{
|
|
|
|
input_mt_slot(dev, slot);
|
|
|
|
input_mt_report_slot_state(dev, MT_TOOL_FINGER, active);
|
|
|
|
if (active) {
|
|
|
|
input_report_abs(dev, ABS_MT_POSITION_X, x);
|
|
|
|
input_report_abs(dev, ABS_MT_POSITION_Y, y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_report_semi_mt_data(struct input_dev *dev, int num_fingers,
|
|
|
|
int x1, int y1, int x2, int y2)
|
|
|
|
{
|
|
|
|
alps_set_slot(dev, 0, num_fingers != 0, x1, y1);
|
|
|
|
alps_set_slot(dev, 1, num_fingers == 2, x2, y2);
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
static void alps_process_trackstick_packet_v3(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
unsigned char *packet = psmouse->packet;
|
|
|
|
struct input_dev *dev = priv->dev2;
|
|
|
|
int x, y, z, left, right, middle;
|
|
|
|
|
|
|
|
/* Sanity check packet */
|
|
|
|
if (!(packet[0] & 0x40)) {
|
|
|
|
psmouse_dbg(psmouse, "Bad trackstick packet, discarding\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* There's a special packet that seems to indicate the end
|
|
|
|
* of a stream of trackstick data. Filter these out.
|
|
|
|
*/
|
|
|
|
if (packet[1] == 0x7f && packet[2] == 0x7f && packet[4] == 0x7f)
|
|
|
|
return;
|
|
|
|
|
|
|
|
x = (s8)(((packet[0] & 0x20) << 2) | (packet[1] & 0x7f));
|
|
|
|
y = (s8)(((packet[0] & 0x10) << 3) | (packet[2] & 0x7f));
|
|
|
|
z = (packet[4] & 0x7c) >> 2;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The x and y values tend to be quite large, and when used
|
|
|
|
* alone the trackstick is difficult to use. Scale them down
|
|
|
|
* to compensate.
|
|
|
|
*/
|
|
|
|
x /= 8;
|
|
|
|
y /= 8;
|
|
|
|
|
|
|
|
input_report_rel(dev, REL_X, x);
|
|
|
|
input_report_rel(dev, REL_Y, -y);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Most ALPS models report the trackstick buttons in the touchpad
|
|
|
|
* packets, but a few report them here. No reliable way has been
|
|
|
|
* found to differentiate between the models upfront, so we enable
|
|
|
|
* the quirk in response to seeing a button press in the trackstick
|
|
|
|
* packet.
|
|
|
|
*/
|
|
|
|
left = packet[3] & 0x01;
|
|
|
|
right = packet[3] & 0x02;
|
|
|
|
middle = packet[3] & 0x04;
|
|
|
|
|
|
|
|
if (!(priv->quirks & ALPS_QUIRK_TRACKSTICK_BUTTONS) &&
|
|
|
|
(left || right || middle))
|
|
|
|
priv->quirks |= ALPS_QUIRK_TRACKSTICK_BUTTONS;
|
|
|
|
|
|
|
|
if (priv->quirks & ALPS_QUIRK_TRACKSTICK_BUTTONS) {
|
|
|
|
input_report_key(dev, BTN_LEFT, left);
|
|
|
|
input_report_key(dev, BTN_RIGHT, right);
|
|
|
|
input_report_key(dev, BTN_MIDDLE, middle);
|
|
|
|
}
|
|
|
|
|
|
|
|
input_sync(dev);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:26:11 +08:00
|
|
|
static void alps_decode_buttons_v3(struct alps_fields *f, unsigned char *p)
|
|
|
|
{
|
|
|
|
f->left = !!(p[3] & 0x01);
|
|
|
|
f->right = !!(p[3] & 0x02);
|
|
|
|
f->middle = !!(p[3] & 0x04);
|
|
|
|
|
|
|
|
f->ts_left = !!(p[3] & 0x10);
|
|
|
|
f->ts_right = !!(p[3] & 0x20);
|
|
|
|
f->ts_middle = !!(p[3] & 0x40);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_decode_pinnacle(struct alps_fields *f, unsigned char *p)
|
|
|
|
{
|
|
|
|
f->first_mp = !!(p[4] & 0x40);
|
|
|
|
f->is_mp = !!(p[0] & 0x40);
|
|
|
|
|
|
|
|
f->fingers = (p[5] & 0x3) + 1;
|
|
|
|
f->x_map = ((p[4] & 0x7e) << 8) |
|
|
|
|
((p[1] & 0x7f) << 2) |
|
|
|
|
((p[0] & 0x30) >> 4);
|
|
|
|
f->y_map = ((p[3] & 0x70) << 4) |
|
|
|
|
((p[2] & 0x7f) << 1) |
|
|
|
|
(p[4] & 0x01);
|
|
|
|
|
|
|
|
f->x = ((p[1] & 0x7f) << 4) | ((p[4] & 0x30) >> 2) |
|
|
|
|
((p[0] & 0x30) >> 4);
|
|
|
|
f->y = ((p[2] & 0x7f) << 4) | (p[4] & 0x0f);
|
|
|
|
f->z = p[5] & 0x7f;
|
|
|
|
|
|
|
|
alps_decode_buttons_v3(f, p);
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:27:08 +08:00
|
|
|
static void alps_decode_rushmore(struct alps_fields *f, unsigned char *p)
|
|
|
|
{
|
|
|
|
alps_decode_pinnacle(f, p);
|
|
|
|
|
|
|
|
f->x_map |= (p[5] & 0x10) << 11;
|
|
|
|
f->y_map |= (p[5] & 0x20) << 6;
|
|
|
|
}
|
|
|
|
|
2013-02-22 14:58:28 +08:00
|
|
|
static void alps_decode_dolphin(struct alps_fields *f, unsigned char *p)
|
|
|
|
{
|
|
|
|
f->first_mp = !!(p[0] & 0x02);
|
|
|
|
f->is_mp = !!(p[0] & 0x20);
|
|
|
|
|
|
|
|
f->fingers = ((p[0] & 0x6) >> 1 |
|
|
|
|
(p[0] & 0x10) >> 2);
|
|
|
|
f->x_map = ((p[2] & 0x60) >> 5) |
|
|
|
|
((p[4] & 0x7f) << 2) |
|
|
|
|
((p[5] & 0x7f) << 9) |
|
|
|
|
((p[3] & 0x07) << 16) |
|
|
|
|
((p[3] & 0x70) << 15) |
|
|
|
|
((p[0] & 0x01) << 22);
|
|
|
|
f->y_map = (p[1] & 0x7f) |
|
|
|
|
((p[2] & 0x1f) << 7);
|
|
|
|
|
|
|
|
f->x = ((p[1] & 0x7f) | ((p[4] & 0x0f) << 7));
|
|
|
|
f->y = ((p[2] & 0x7f) | ((p[4] & 0xf0) << 3));
|
|
|
|
f->z = (p[0] & 4) ? 0 : p[5] & 0x7f;
|
|
|
|
|
|
|
|
alps_decode_buttons_v3(f, p);
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
static void alps_process_touchpad_packet_v3(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
unsigned char *packet = psmouse->packet;
|
|
|
|
struct input_dev *dev = psmouse->dev;
|
|
|
|
struct input_dev *dev2 = priv->dev2;
|
2011-11-08 11:54:13 +08:00
|
|
|
int x1 = 0, y1 = 0, x2 = 0, y2 = 0;
|
|
|
|
int fingers = 0, bmap_fingers;
|
2013-02-14 14:26:11 +08:00
|
|
|
struct alps_fields f;
|
|
|
|
|
|
|
|
priv->decode_fields(&f, packet);
|
2011-11-08 11:53:36 +08:00
|
|
|
|
|
|
|
/*
|
2011-11-08 11:54:13 +08:00
|
|
|
* There's no single feature of touchpad position and bitmap packets
|
|
|
|
* that can be used to distinguish between them. We rely on the fact
|
|
|
|
* that a bitmap packet should always follow a position packet with
|
|
|
|
* bit 6 of packet[4] set.
|
2011-11-08 11:53:36 +08:00
|
|
|
*/
|
|
|
|
if (priv->multi_packet) {
|
|
|
|
/*
|
|
|
|
* Sometimes a position packet will indicate a multi-packet
|
|
|
|
* sequence, but then what follows is another position
|
|
|
|
* packet. Check for this, and when it happens process the
|
|
|
|
* position packet as usual.
|
|
|
|
*/
|
2013-02-14 14:26:11 +08:00
|
|
|
if (f.is_mp) {
|
|
|
|
fingers = f.fingers;
|
2013-02-14 14:24:55 +08:00
|
|
|
bmap_fingers = alps_process_bitmap(priv,
|
2013-02-14 14:26:11 +08:00
|
|
|
f.x_map, f.y_map,
|
2011-11-08 11:54:13 +08:00
|
|
|
&x1, &y1, &x2, &y2);
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
/*
|
2011-11-08 11:54:13 +08:00
|
|
|
* We shouldn't report more than one finger if
|
|
|
|
* we don't have two coordinates.
|
2011-11-08 11:53:36 +08:00
|
|
|
*/
|
2011-11-08 11:54:13 +08:00
|
|
|
if (fingers > 1 && bmap_fingers < 2)
|
|
|
|
fingers = bmap_fingers;
|
|
|
|
|
|
|
|
/* Now process position packet */
|
2013-02-14 14:26:11 +08:00
|
|
|
priv->decode_fields(&f, priv->multi_data);
|
2011-11-08 11:54:13 +08:00
|
|
|
} else {
|
|
|
|
priv->multi_packet = 0;
|
2011-11-08 11:53:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:54:13 +08:00
|
|
|
/*
|
|
|
|
* Bit 6 of byte 0 is not usually set in position packets. The only
|
|
|
|
* times it seems to be set is in situations where the data is
|
|
|
|
* suspect anyway, e.g. a palm resting flat on the touchpad. Given
|
|
|
|
* this combined with the fact that this bit is useful for filtering
|
|
|
|
* out misidentified bitmap packets, we reject anything with this
|
|
|
|
* bit set.
|
|
|
|
*/
|
2013-02-14 14:26:11 +08:00
|
|
|
if (f.is_mp)
|
2011-11-08 11:54:13 +08:00
|
|
|
return;
|
|
|
|
|
2013-02-14 14:26:11 +08:00
|
|
|
if (!priv->multi_packet && f.first_mp) {
|
2011-11-08 11:53:36 +08:00
|
|
|
priv->multi_packet = 1;
|
2011-11-08 11:54:13 +08:00
|
|
|
memcpy(priv->multi_data, packet, sizeof(priv->multi_data));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
priv->multi_packet = 0;
|
2011-11-08 11:53:36 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Sometimes the hardware sends a single packet with z = 0
|
|
|
|
* in the middle of a stream. Real releases generate packets
|
|
|
|
* with x, y, and z all zero, so these seem to be flukes.
|
|
|
|
* Ignore them.
|
|
|
|
*/
|
2013-02-14 14:26:11 +08:00
|
|
|
if (f.x && f.y && !f.z)
|
2011-11-08 11:53:36 +08:00
|
|
|
return;
|
|
|
|
|
2011-11-08 11:54:13 +08:00
|
|
|
/*
|
|
|
|
* If we don't have MT data or the bitmaps were empty, we have
|
|
|
|
* to rely on ST data.
|
|
|
|
*/
|
|
|
|
if (!fingers) {
|
2013-02-14 14:26:11 +08:00
|
|
|
x1 = f.x;
|
|
|
|
y1 = f.y;
|
|
|
|
fingers = f.z > 0 ? 1 : 0;
|
2011-11-08 11:54:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-14 14:26:11 +08:00
|
|
|
if (f.z >= 64)
|
2011-11-08 11:53:36 +08:00
|
|
|
input_report_key(dev, BTN_TOUCH, 1);
|
|
|
|
else
|
|
|
|
input_report_key(dev, BTN_TOUCH, 0);
|
|
|
|
|
2011-11-08 11:54:13 +08:00
|
|
|
alps_report_semi_mt_data(dev, fingers, x1, y1, x2, y2);
|
|
|
|
|
2012-05-11 13:31:59 +08:00
|
|
|
input_mt_report_finger_count(dev, fingers);
|
2011-11-08 11:54:13 +08:00
|
|
|
|
2013-02-14 14:26:11 +08:00
|
|
|
input_report_key(dev, BTN_LEFT, f.left);
|
|
|
|
input_report_key(dev, BTN_RIGHT, f.right);
|
|
|
|
input_report_key(dev, BTN_MIDDLE, f.middle);
|
2011-11-08 11:54:13 +08:00
|
|
|
|
2013-02-14 14:26:11 +08:00
|
|
|
if (f.z > 0) {
|
|
|
|
input_report_abs(dev, ABS_X, f.x);
|
|
|
|
input_report_abs(dev, ABS_Y, f.y);
|
2011-11-08 11:53:36 +08:00
|
|
|
}
|
2013-02-14 14:26:11 +08:00
|
|
|
input_report_abs(dev, ABS_PRESSURE, f.z);
|
2011-11-08 11:53:36 +08:00
|
|
|
|
|
|
|
input_sync(dev);
|
|
|
|
|
|
|
|
if (!(priv->quirks & ALPS_QUIRK_TRACKSTICK_BUTTONS)) {
|
2013-02-14 14:26:11 +08:00
|
|
|
input_report_key(dev2, BTN_LEFT, f.ts_left);
|
|
|
|
input_report_key(dev2, BTN_RIGHT, f.ts_right);
|
|
|
|
input_report_key(dev2, BTN_MIDDLE, f.ts_middle);
|
2011-11-08 11:53:36 +08:00
|
|
|
input_sync(dev2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_process_packet_v3(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
unsigned char *packet = psmouse->packet;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* v3 protocol packets come in three types, two representing
|
|
|
|
* touchpad data and one representing trackstick data.
|
|
|
|
* Trackstick packets seem to be distinguished by always
|
|
|
|
* having 0x3f in the last byte. This value has never been
|
|
|
|
* observed in the last byte of either of the other types
|
|
|
|
* of packets.
|
|
|
|
*/
|
|
|
|
if (packet[5] == 0x3f) {
|
|
|
|
alps_process_trackstick_packet_v3(psmouse);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
alps_process_touchpad_packet_v3(psmouse);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_process_packet_v4(struct psmouse *psmouse)
|
|
|
|
{
|
2012-05-11 13:31:59 +08:00
|
|
|
struct alps_data *priv = psmouse->private;
|
2011-11-08 11:53:36 +08:00
|
|
|
unsigned char *packet = psmouse->packet;
|
|
|
|
struct input_dev *dev = psmouse->dev;
|
2012-05-11 13:31:59 +08:00
|
|
|
int offset;
|
2011-11-08 11:53:36 +08:00
|
|
|
int x, y, z;
|
|
|
|
int left, right;
|
2012-05-11 13:31:59 +08:00
|
|
|
int x1, y1, x2, y2;
|
|
|
|
int fingers = 0;
|
|
|
|
unsigned int x_bitmap, y_bitmap;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* v4 has a 6-byte encoding for bitmap data, but this data is
|
|
|
|
* broken up between 3 normal packets. Use priv->multi_packet to
|
|
|
|
* track our position in the bitmap packet.
|
|
|
|
*/
|
|
|
|
if (packet[6] & 0x40) {
|
|
|
|
/* sync, reset position */
|
|
|
|
priv->multi_packet = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (WARN_ON_ONCE(priv->multi_packet > 2))
|
|
|
|
return;
|
|
|
|
|
|
|
|
offset = 2 * priv->multi_packet;
|
|
|
|
priv->multi_data[offset] = packet[6];
|
|
|
|
priv->multi_data[offset + 1] = packet[7];
|
|
|
|
|
|
|
|
if (++priv->multi_packet > 2) {
|
|
|
|
priv->multi_packet = 0;
|
|
|
|
|
|
|
|
x_bitmap = ((priv->multi_data[2] & 0x1f) << 10) |
|
|
|
|
((priv->multi_data[3] & 0x60) << 3) |
|
|
|
|
((priv->multi_data[0] & 0x3f) << 2) |
|
|
|
|
((priv->multi_data[1] & 0x60) >> 5);
|
|
|
|
y_bitmap = ((priv->multi_data[5] & 0x01) << 10) |
|
|
|
|
((priv->multi_data[3] & 0x1f) << 5) |
|
|
|
|
(priv->multi_data[1] & 0x1f);
|
|
|
|
|
2013-02-14 14:24:55 +08:00
|
|
|
fingers = alps_process_bitmap(priv, x_bitmap, y_bitmap,
|
2012-05-11 13:31:59 +08:00
|
|
|
&x1, &y1, &x2, &y2);
|
|
|
|
|
|
|
|
/* Store MT data.*/
|
|
|
|
priv->fingers = fingers;
|
|
|
|
priv->x1 = x1;
|
|
|
|
priv->x2 = x2;
|
|
|
|
priv->y1 = y1;
|
|
|
|
priv->y2 = y2;
|
|
|
|
}
|
2011-11-08 11:53:36 +08:00
|
|
|
|
|
|
|
left = packet[4] & 0x01;
|
|
|
|
right = packet[4] & 0x02;
|
|
|
|
|
|
|
|
x = ((packet[1] & 0x7f) << 4) | ((packet[3] & 0x30) >> 2) |
|
|
|
|
((packet[0] & 0x30) >> 4);
|
|
|
|
y = ((packet[2] & 0x7f) << 4) | (packet[3] & 0x0f);
|
|
|
|
z = packet[5] & 0x7f;
|
|
|
|
|
2012-05-11 13:31:59 +08:00
|
|
|
/*
|
|
|
|
* If there were no contacts in the bitmap, use ST
|
|
|
|
* points in MT reports.
|
|
|
|
* If there were two contacts or more, report MT data.
|
|
|
|
*/
|
|
|
|
if (priv->fingers < 2) {
|
|
|
|
x1 = x;
|
|
|
|
y1 = y;
|
|
|
|
fingers = z > 0 ? 1 : 0;
|
|
|
|
} else {
|
|
|
|
fingers = priv->fingers;
|
|
|
|
x1 = priv->x1;
|
|
|
|
x2 = priv->x2;
|
|
|
|
y1 = priv->y1;
|
|
|
|
y2 = priv->y2;
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
if (z >= 64)
|
|
|
|
input_report_key(dev, BTN_TOUCH, 1);
|
|
|
|
else
|
|
|
|
input_report_key(dev, BTN_TOUCH, 0);
|
|
|
|
|
2012-05-11 13:31:59 +08:00
|
|
|
alps_report_semi_mt_data(dev, fingers, x1, y1, x2, y2);
|
|
|
|
|
2012-05-11 13:31:59 +08:00
|
|
|
input_mt_report_finger_count(dev, fingers);
|
2012-05-11 13:31:59 +08:00
|
|
|
|
|
|
|
input_report_key(dev, BTN_LEFT, left);
|
|
|
|
input_report_key(dev, BTN_RIGHT, right);
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
if (z > 0) {
|
|
|
|
input_report_abs(dev, ABS_X, x);
|
|
|
|
input_report_abs(dev, ABS_Y, y);
|
|
|
|
}
|
|
|
|
input_report_abs(dev, ABS_PRESSURE, z);
|
|
|
|
|
|
|
|
input_sync(dev);
|
|
|
|
}
|
|
|
|
|
2009-12-16 00:39:50 +08:00
|
|
|
static void alps_report_bare_ps2_packet(struct psmouse *psmouse,
|
|
|
|
unsigned char packet[],
|
|
|
|
bool report_buttons)
|
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
struct input_dev *dev2 = priv->dev2;
|
|
|
|
|
|
|
|
if (report_buttons)
|
|
|
|
alps_report_buttons(psmouse, dev2, psmouse->dev,
|
|
|
|
packet[0] & 1, packet[0] & 2, packet[0] & 4);
|
|
|
|
|
|
|
|
input_report_rel(dev2, REL_X,
|
|
|
|
packet[1] ? packet[1] - ((packet[0] << 4) & 0x100) : 0);
|
|
|
|
input_report_rel(dev2, REL_Y,
|
|
|
|
packet[2] ? ((packet[0] << 3) & 0x100) - packet[2] : 0);
|
|
|
|
|
|
|
|
input_sync(dev2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static psmouse_ret_t alps_handle_interleaved_ps2(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
|
|
|
|
if (psmouse->pktcnt < 6)
|
|
|
|
return PSMOUSE_GOOD_DATA;
|
|
|
|
|
|
|
|
if (psmouse->pktcnt == 6) {
|
|
|
|
/*
|
|
|
|
* Start a timer to flush the packet if it ends up last
|
|
|
|
* 6-byte packet in the stream. Timer needs to fire
|
|
|
|
* psmouse core times out itself. 20 ms should be enough
|
|
|
|
* to decide if we are getting more data or not.
|
|
|
|
*/
|
|
|
|
mod_timer(&priv->timer, jiffies + msecs_to_jiffies(20));
|
|
|
|
return PSMOUSE_GOOD_DATA;
|
|
|
|
}
|
|
|
|
|
|
|
|
del_timer(&priv->timer);
|
|
|
|
|
|
|
|
if (psmouse->packet[6] & 0x80) {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Highest bit is set - that means we either had
|
|
|
|
* complete ALPS packet and this is start of the
|
|
|
|
* next packet or we got garbage.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (((psmouse->packet[3] |
|
|
|
|
psmouse->packet[4] |
|
|
|
|
psmouse->packet[5]) & 0x80) ||
|
2013-02-14 12:56:33 +08:00
|
|
|
(!alps_is_valid_first_byte(priv, psmouse->packet[6]))) {
|
2011-10-11 09:27:03 +08:00
|
|
|
psmouse_dbg(psmouse,
|
2012-10-30 15:24:41 +08:00
|
|
|
"refusing packet %4ph (suspected interleaved ps/2)\n",
|
|
|
|
psmouse->packet + 3);
|
2009-12-16 00:39:50 +08:00
|
|
|
return PSMOUSE_BAD_DATA;
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:22:08 +08:00
|
|
|
priv->process_packet(psmouse);
|
2009-12-16 00:39:50 +08:00
|
|
|
|
|
|
|
/* Continue with the next packet */
|
|
|
|
psmouse->packet[0] = psmouse->packet[6];
|
|
|
|
psmouse->pktcnt = 1;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* High bit is 0 - that means that we indeed got a PS/2
|
|
|
|
* packet in the middle of ALPS packet.
|
|
|
|
*
|
|
|
|
* There is also possibility that we got 6-byte ALPS
|
|
|
|
* packet followed by 3-byte packet from trackpoint. We
|
|
|
|
* can not distinguish between these 2 scenarios but
|
2011-10-11 09:27:03 +08:00
|
|
|
* because the latter is unlikely to happen in course of
|
2009-12-16 00:39:50 +08:00
|
|
|
* normal operation (user would need to press all
|
|
|
|
* buttons on the pad and start moving trackpoint
|
|
|
|
* without touching the pad surface) we assume former.
|
|
|
|
* Even if we are wrong the wost thing that would happen
|
|
|
|
* the cursor would jump but we should not get protocol
|
2011-10-11 09:27:03 +08:00
|
|
|
* de-synchronization.
|
2009-12-16 00:39:50 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
alps_report_bare_ps2_packet(psmouse, &psmouse->packet[3],
|
|
|
|
false);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Continue with the standard ALPS protocol handling,
|
|
|
|
* but make sure we won't process it as an interleaved
|
|
|
|
* packet again, which may happen if all buttons are
|
|
|
|
* pressed. To avoid this let's reset the 4th bit which
|
|
|
|
* is normally 1.
|
|
|
|
*/
|
|
|
|
psmouse->packet[3] = psmouse->packet[6] & 0xf7;
|
|
|
|
psmouse->pktcnt = 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PSMOUSE_GOOD_DATA;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_flush_packet(unsigned long data)
|
|
|
|
{
|
|
|
|
struct psmouse *psmouse = (struct psmouse *)data;
|
2013-02-14 14:22:08 +08:00
|
|
|
struct alps_data *priv = psmouse->private;
|
2009-12-16 00:39:50 +08:00
|
|
|
|
|
|
|
serio_pause_rx(psmouse->ps2dev.serio);
|
|
|
|
|
2011-11-08 11:53:30 +08:00
|
|
|
if (psmouse->pktcnt == psmouse->pktsize) {
|
2009-12-16 00:39:50 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We did not any more data in reasonable amount of time.
|
|
|
|
* Validate the last 3 bytes and process as a standard
|
|
|
|
* ALPS packet.
|
|
|
|
*/
|
|
|
|
if ((psmouse->packet[3] |
|
|
|
|
psmouse->packet[4] |
|
|
|
|
psmouse->packet[5]) & 0x80) {
|
2011-10-11 09:27:03 +08:00
|
|
|
psmouse_dbg(psmouse,
|
2012-10-30 15:24:41 +08:00
|
|
|
"refusing packet %3ph (suspected interleaved ps/2)\n",
|
|
|
|
psmouse->packet + 3);
|
2009-12-16 00:39:50 +08:00
|
|
|
} else {
|
2013-02-14 14:22:08 +08:00
|
|
|
priv->process_packet(psmouse);
|
2009-12-16 00:39:50 +08:00
|
|
|
}
|
|
|
|
psmouse->pktcnt = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
serio_continue_rx(psmouse->ps2dev.serio);
|
|
|
|
}
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 21:55:46 +08:00
|
|
|
static psmouse_ret_t alps_process_byte(struct psmouse *psmouse)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
|
|
|
|
if ((psmouse->packet[0] & 0xc8) == 0x08) { /* PS/2 packet */
|
|
|
|
if (psmouse->pktcnt == 3) {
|
2009-12-16 00:39:50 +08:00
|
|
|
alps_report_bare_ps2_packet(psmouse, psmouse->packet,
|
|
|
|
true);
|
2005-04-17 06:20:36 +08:00
|
|
|
return PSMOUSE_FULL_PACKET;
|
|
|
|
}
|
|
|
|
return PSMOUSE_GOOD_DATA;
|
|
|
|
}
|
|
|
|
|
2009-12-16 00:39:50 +08:00
|
|
|
/* Check for PS/2 packet stuffed in the middle of ALPS packet. */
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if ((priv->flags & ALPS_PS2_INTERLEAVED) &&
|
2009-12-16 00:39:50 +08:00
|
|
|
psmouse->pktcnt >= 4 && (psmouse->packet[3] & 0x0f) == 0x0f) {
|
|
|
|
return alps_handle_interleaved_ps2(psmouse);
|
|
|
|
}
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (!alps_is_valid_first_byte(priv, psmouse->packet[0])) {
|
2011-10-11 09:27:03 +08:00
|
|
|
psmouse_dbg(psmouse,
|
|
|
|
"refusing packet[0] = %x (mask0 = %x, byte0 = %x)\n",
|
2013-02-14 12:56:33 +08:00
|
|
|
psmouse->packet[0], priv->mask0, priv->byte0);
|
2005-04-17 06:20:36 +08:00
|
|
|
return PSMOUSE_BAD_DATA;
|
2009-12-16 00:39:50 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2011-11-08 11:53:30 +08:00
|
|
|
/* Bytes 2 - pktsize should have 0 in the highest bit */
|
2013-02-22 14:58:28 +08:00
|
|
|
if (priv->proto_version != ALPS_PROTO_V5 &&
|
|
|
|
psmouse->pktcnt >= 2 && psmouse->pktcnt <= psmouse->pktsize &&
|
2009-12-16 00:39:50 +08:00
|
|
|
(psmouse->packet[psmouse->pktcnt - 1] & 0x80)) {
|
2011-10-11 09:27:03 +08:00
|
|
|
psmouse_dbg(psmouse, "refusing packet[%i] = %x\n",
|
|
|
|
psmouse->pktcnt - 1,
|
|
|
|
psmouse->packet[psmouse->pktcnt - 1]);
|
2005-04-17 06:20:36 +08:00
|
|
|
return PSMOUSE_BAD_DATA;
|
2009-12-16 00:39:50 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2011-11-08 11:53:30 +08:00
|
|
|
if (psmouse->pktcnt == psmouse->pktsize) {
|
2013-02-14 14:22:08 +08:00
|
|
|
priv->process_packet(psmouse);
|
2005-04-17 06:20:36 +08:00
|
|
|
return PSMOUSE_FULL_PACKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PSMOUSE_GOOD_DATA;
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
static int alps_command_mode_send_nibble(struct psmouse *psmouse, int nibble)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
int command;
|
|
|
|
unsigned char *param;
|
|
|
|
unsigned char dummy[4];
|
|
|
|
|
|
|
|
BUG_ON(nibble > 0xf);
|
|
|
|
|
|
|
|
command = priv->nibble_commands[nibble].command;
|
|
|
|
param = (command & 0x0f00) ?
|
|
|
|
dummy : (unsigned char *)&priv->nibble_commands[nibble].data;
|
|
|
|
|
|
|
|
if (ps2_command(ps2dev, param, command))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_command_mode_set_addr(struct psmouse *psmouse, int addr)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
struct alps_data *priv = psmouse->private;
|
|
|
|
int i, nibble;
|
|
|
|
|
|
|
|
if (ps2_command(ps2dev, NULL, priv->addr_command))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (i = 12; i >= 0; i -= 4) {
|
|
|
|
nibble = (addr >> i) & 0xf;
|
|
|
|
if (alps_command_mode_send_nibble(psmouse, nibble))
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __alps_command_mode_read_reg(struct psmouse *psmouse, int addr)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
unsigned char param[4];
|
|
|
|
|
|
|
|
if (ps2_command(ps2dev, param, PSMOUSE_CMD_GETINFO))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The address being read is returned in the first two bytes
|
|
|
|
* of the result. Check that this address matches the expected
|
|
|
|
* address.
|
|
|
|
*/
|
|
|
|
if (addr != ((param[0] << 8) | param[1]))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return param[2];
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_command_mode_read_reg(struct psmouse *psmouse, int addr)
|
|
|
|
{
|
|
|
|
if (alps_command_mode_set_addr(psmouse, addr))
|
|
|
|
return -1;
|
|
|
|
return __alps_command_mode_read_reg(psmouse, addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __alps_command_mode_write_reg(struct psmouse *psmouse, u8 value)
|
|
|
|
{
|
|
|
|
if (alps_command_mode_send_nibble(psmouse, (value >> 4) & 0xf))
|
|
|
|
return -1;
|
|
|
|
if (alps_command_mode_send_nibble(psmouse, value & 0xf))
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_command_mode_write_reg(struct psmouse *psmouse, int addr,
|
|
|
|
u8 value)
|
|
|
|
{
|
|
|
|
if (alps_command_mode_set_addr(psmouse, addr))
|
|
|
|
return -1;
|
|
|
|
return __alps_command_mode_write_reg(psmouse, value);
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:19:01 +08:00
|
|
|
static int alps_rpt_cmd(struct psmouse *psmouse, int init_command,
|
|
|
|
int repeated_command, unsigned char *param)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
|
|
|
|
param[0] = 0;
|
|
|
|
if (init_command && ps2_command(ps2dev, param, init_command))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
if (ps2_command(ps2dev, NULL, repeated_command) ||
|
|
|
|
ps2_command(ps2dev, NULL, repeated_command) ||
|
|
|
|
ps2_command(ps2dev, NULL, repeated_command))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
param[0] = param[1] = param[2] = 0xff;
|
|
|
|
if (ps2_command(ps2dev, param, PSMOUSE_CMD_GETINFO))
|
|
|
|
return -EIO;
|
|
|
|
|
2013-02-15 01:04:24 +08:00
|
|
|
psmouse_dbg(psmouse, "%2.2X report: %3ph\n",
|
|
|
|
repeated_command, param);
|
2013-02-14 14:19:01 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-02-22 14:58:20 +08:00
|
|
|
static int alps_enter_command_mode(struct psmouse *psmouse)
|
2011-11-08 11:53:36 +08:00
|
|
|
{
|
|
|
|
unsigned char param[4];
|
|
|
|
|
2013-02-14 14:19:01 +08:00
|
|
|
if (alps_rpt_cmd(psmouse, 0, PSMOUSE_CMD_RESET_WRAP, param)) {
|
2011-11-08 11:53:36 +08:00
|
|
|
psmouse_err(psmouse, "failed to enter command mode\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-02-22 14:58:28 +08:00
|
|
|
if ((param[0] != 0x88 || (param[1] != 0x07 && param[1] != 0x08)) &&
|
|
|
|
param[0] != 0x73) {
|
2011-11-08 11:53:36 +08:00
|
|
|
psmouse_dbg(psmouse,
|
2013-02-14 14:19:01 +08:00
|
|
|
"unknown response while entering command mode\n");
|
2011-11-08 11:53:36 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int alps_exit_command_mode(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
if (ps2_command(ps2dev, NULL, PSMOUSE_CMD_SETSTREAM))
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* For DualPoint devices select the device that should respond to
|
|
|
|
* subsequent commands. It looks like glidepad is behind stickpointer,
|
|
|
|
* I'd thought it would be other way around...
|
|
|
|
*/
|
2011-11-08 11:53:36 +08:00
|
|
|
static int alps_passthrough_mode_v2(struct psmouse *psmouse, bool enable)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
int cmd = enable ? PSMOUSE_CMD_SETSCALE21 : PSMOUSE_CMD_SETSCALE11;
|
|
|
|
|
|
|
|
if (ps2_command(ps2dev, NULL, cmd) ||
|
|
|
|
ps2_command(ps2dev, NULL, cmd) ||
|
|
|
|
ps2_command(ps2dev, NULL, cmd) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_DISABLE))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* we may get 3 more bytes, just ignore them */
|
2005-06-01 15:39:51 +08:00
|
|
|
ps2_drain(ps2dev, 3, 100);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
static int alps_absolute_mode_v1_v2(struct psmouse *psmouse)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
|
|
|
|
/* Try ALPS magic knock - 4 disable before enable */
|
|
|
|
if (ps2_command(ps2dev, NULL, PSMOUSE_CMD_DISABLE) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_DISABLE) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_DISABLE) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_DISABLE) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_ENABLE))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Switch mouse to poll (remote) mode so motion data will not
|
|
|
|
* get in our way
|
|
|
|
*/
|
|
|
|
return ps2_command(&psmouse->ps2dev, NULL, PSMOUSE_CMD_SETPOLL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_get_status(struct psmouse *psmouse, char *param)
|
|
|
|
{
|
|
|
|
/* Get status: 0xF5 0xF5 0xF5 0xE9 */
|
2013-02-14 14:19:01 +08:00
|
|
|
if (alps_rpt_cmd(psmouse, 0, PSMOUSE_CMD_DISABLE, param))
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Turn touchpad tapping on or off. The sequences are:
|
|
|
|
* 0xE9 0xF5 0xF5 0xF3 0x0A to enable,
|
|
|
|
* 0xE9 0xF5 0xF5 0xE8 0x00 to disable.
|
|
|
|
* My guess that 0xE9 (GetInfo) is here as a sync point.
|
|
|
|
* For models that also have stickpointer (DualPoints) its tapping
|
|
|
|
* is controlled separately (0xE6 0xE6 0xE6 0xF3 0x14|0x0A) but
|
|
|
|
* we don't fiddle with it.
|
|
|
|
*/
|
|
|
|
static int alps_tap_mode(struct psmouse *psmouse, int enable)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
int cmd = enable ? PSMOUSE_CMD_SETRATE : PSMOUSE_CMD_SETRES;
|
|
|
|
unsigned char tap_arg = enable ? 0x0A : 0x00;
|
|
|
|
unsigned char param[4];
|
|
|
|
|
|
|
|
if (ps2_command(ps2dev, param, PSMOUSE_CMD_GETINFO) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_DISABLE) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_DISABLE) ||
|
|
|
|
ps2_command(ps2dev, &tap_arg, cmd))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (alps_get_status(psmouse, param))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-01-14 13:27:37 +08:00
|
|
|
/*
|
|
|
|
* alps_poll() - poll the touchpad for current motion packet.
|
|
|
|
* Used in resync.
|
|
|
|
*/
|
|
|
|
static int alps_poll(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
2011-11-08 11:53:30 +08:00
|
|
|
unsigned char buf[sizeof(psmouse->packet)];
|
2009-09-10 10:13:20 +08:00
|
|
|
bool poll_failed;
|
2006-01-14 13:27:37 +08:00
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_PASS)
|
2011-11-08 11:53:36 +08:00
|
|
|
alps_passthrough_mode_v2(psmouse, true);
|
2006-01-14 13:27:37 +08:00
|
|
|
|
|
|
|
poll_failed = ps2_command(&psmouse->ps2dev, buf,
|
|
|
|
PSMOUSE_CMD_POLL | (psmouse->pktsize << 8)) < 0;
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_PASS)
|
2011-11-08 11:53:36 +08:00
|
|
|
alps_passthrough_mode_v2(psmouse, false);
|
2006-01-14 13:27:37 +08:00
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (poll_failed || (buf[0] & priv->mask0) != priv->byte0)
|
2006-01-14 13:27:37 +08:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if ((psmouse->badbyte & 0xc8) == 0x08) {
|
|
|
|
/*
|
|
|
|
* Poll the track stick ...
|
|
|
|
*/
|
|
|
|
if (ps2_command(&psmouse->ps2dev, buf, PSMOUSE_CMD_POLL | (3 << 8)))
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(psmouse->packet, buf, sizeof(buf));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
static int alps_hw_init_v1_v2(struct psmouse *psmouse)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
2005-05-17 12:53:06 +08:00
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if ((priv->flags & ALPS_PASS) &&
|
2011-11-08 11:53:36 +08:00
|
|
|
alps_passthrough_mode_v2(psmouse, true)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
2009-09-10 10:13:20 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-09-10 10:13:20 +08:00
|
|
|
if (alps_tap_mode(psmouse, true)) {
|
2011-10-11 09:27:03 +08:00
|
|
|
psmouse_warn(psmouse, "Failed to enable hardware tapping\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
2005-07-11 14:08:04 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
if (alps_absolute_mode_v1_v2(psmouse)) {
|
2011-10-11 09:27:03 +08:00
|
|
|
psmouse_err(psmouse, "Failed to enable absolute mode\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if ((priv->flags & ALPS_PASS) &&
|
2011-11-08 11:53:36 +08:00
|
|
|
alps_passthrough_mode_v2(psmouse, false)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
2009-09-10 10:13:20 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-05-15 11:51:54 +08:00
|
|
|
/* ALPS needs stream mode, otherwise it won't report any data */
|
|
|
|
if (ps2_command(&psmouse->ps2dev, NULL, PSMOUSE_CMD_SETSTREAM)) {
|
2011-10-11 09:27:03 +08:00
|
|
|
psmouse_err(psmouse, "Failed to enable stream mode\n");
|
2007-05-15 11:51:54 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
/*
|
2013-02-14 14:28:07 +08:00
|
|
|
* Enable or disable passthrough mode to the trackstick.
|
2011-11-08 11:53:36 +08:00
|
|
|
*/
|
2013-02-14 14:28:07 +08:00
|
|
|
static int alps_passthrough_mode_v3(struct psmouse *psmouse,
|
|
|
|
int reg_base, bool enable)
|
2011-11-08 11:53:36 +08:00
|
|
|
{
|
2013-02-14 14:28:07 +08:00
|
|
|
int reg_val, ret = -1;
|
2011-11-08 11:53:36 +08:00
|
|
|
|
2013-02-22 14:58:20 +08:00
|
|
|
if (alps_enter_command_mode(psmouse))
|
2011-11-08 11:53:36 +08:00
|
|
|
return -1;
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, reg_base + 0x0008);
|
|
|
|
if (reg_val == -1)
|
|
|
|
goto error;
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
if (enable)
|
|
|
|
reg_val |= 0x01;
|
|
|
|
else
|
|
|
|
reg_val &= ~0x01;
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
ret = __alps_command_mode_write_reg(psmouse, reg_val);
|
2011-11-08 11:53:36 +08:00
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
error:
|
|
|
|
if (alps_exit_command_mode(psmouse))
|
|
|
|
ret = -1;
|
|
|
|
return ret;
|
2011-11-08 11:53:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Must be in command mode when calling this function */
|
|
|
|
static int alps_absolute_mode_v3(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
int reg_val;
|
|
|
|
|
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, 0x0004);
|
|
|
|
if (reg_val == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
reg_val |= 0x06;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, reg_val))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
static int alps_probe_trackstick_v3(struct psmouse *psmouse, int reg_base)
|
2011-11-08 11:53:36 +08:00
|
|
|
{
|
2013-02-14 14:28:07 +08:00
|
|
|
int ret = -EIO, reg_val;
|
2011-11-08 11:53:36 +08:00
|
|
|
|
2013-02-22 14:58:20 +08:00
|
|
|
if (alps_enter_command_mode(psmouse))
|
2011-11-08 11:53:36 +08:00
|
|
|
goto error;
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, reg_base + 0x08);
|
2011-11-08 11:53:36 +08:00
|
|
|
if (reg_val == -1)
|
|
|
|
goto error;
|
2013-02-14 14:28:07 +08:00
|
|
|
|
|
|
|
/* bit 7: trackstick is present */
|
|
|
|
ret = reg_val & 0x80 ? 0 : -ENODEV;
|
|
|
|
|
|
|
|
error:
|
|
|
|
alps_exit_command_mode(psmouse);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_setup_trackstick_v3(struct psmouse *psmouse, int reg_base)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
int ret = 0;
|
|
|
|
unsigned char param[4];
|
|
|
|
|
|
|
|
if (alps_passthrough_mode_v3(psmouse, reg_base, true))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* E7 report for the trackstick
|
|
|
|
*
|
|
|
|
* There have been reports of failures to seem to trace back
|
|
|
|
* to the above trackstick check failing. When these occur
|
|
|
|
* this E7 report fails, so when that happens we continue
|
|
|
|
* with the assumption that there isn't a trackstick after
|
|
|
|
* all.
|
|
|
|
*/
|
|
|
|
if (alps_rpt_cmd(psmouse, 0, PSMOUSE_CMD_SETSCALE21, param)) {
|
|
|
|
psmouse_warn(psmouse, "trackstick E7 report failed\n");
|
|
|
|
ret = -ENODEV;
|
|
|
|
} else {
|
2013-02-15 01:04:24 +08:00
|
|
|
psmouse_dbg(psmouse, "trackstick E7 report: %3ph\n", param);
|
2011-11-08 11:53:36 +08:00
|
|
|
|
|
|
|
/*
|
2013-02-14 14:28:07 +08:00
|
|
|
* Not sure what this does, but it is absolutely
|
|
|
|
* essential. Without it, the touchpad does not
|
|
|
|
* work at all and the trackstick just emits normal
|
|
|
|
* PS/2 packets.
|
2011-11-08 11:53:36 +08:00
|
|
|
*/
|
2013-02-14 14:28:07 +08:00
|
|
|
if (ps2_command(ps2dev, NULL, PSMOUSE_CMD_SETSCALE11) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_SETSCALE11) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_SETSCALE11) ||
|
|
|
|
alps_command_mode_send_nibble(psmouse, 0x9) ||
|
|
|
|
alps_command_mode_send_nibble(psmouse, 0x4)) {
|
|
|
|
psmouse_err(psmouse,
|
|
|
|
"Error sending magic E6 sequence\n");
|
|
|
|
ret = -EIO;
|
|
|
|
goto error;
|
2011-11-08 11:53:36 +08:00
|
|
|
}
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
/*
|
|
|
|
* This ensures the trackstick packets are in the format
|
|
|
|
* supported by this driver. If bit 1 isn't set the packet
|
|
|
|
* format is different.
|
|
|
|
*/
|
2013-02-22 14:58:20 +08:00
|
|
|
if (alps_enter_command_mode(psmouse) ||
|
2013-02-14 14:28:07 +08:00
|
|
|
alps_command_mode_write_reg(psmouse,
|
|
|
|
reg_base + 0x08, 0x82) ||
|
|
|
|
alps_exit_command_mode(psmouse))
|
|
|
|
ret = -EIO;
|
2011-11-08 11:53:36 +08:00
|
|
|
}
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
error:
|
|
|
|
if (alps_passthrough_mode_v3(psmouse, reg_base, false))
|
|
|
|
ret = -EIO;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_hw_init_v3(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
int reg_val;
|
|
|
|
unsigned char param[4];
|
|
|
|
|
|
|
|
reg_val = alps_probe_trackstick_v3(psmouse, ALPS_REG_BASE_PINNACLE);
|
|
|
|
if (reg_val == -EIO)
|
|
|
|
goto error;
|
2013-02-15 01:04:24 +08:00
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
if (reg_val == 0 &&
|
|
|
|
alps_setup_trackstick_v3(psmouse, ALPS_REG_BASE_PINNACLE) == -EIO)
|
|
|
|
goto error;
|
|
|
|
|
2013-02-22 14:58:20 +08:00
|
|
|
if (alps_enter_command_mode(psmouse) ||
|
2013-02-14 14:28:07 +08:00
|
|
|
alps_absolute_mode_v3(psmouse)) {
|
2011-11-08 11:53:36 +08:00
|
|
|
psmouse_err(psmouse, "Failed to enter absolute mode\n");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, 0x0006);
|
|
|
|
if (reg_val == -1)
|
|
|
|
goto error;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, reg_val | 0x01))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, 0x0007);
|
|
|
|
if (reg_val == -1)
|
|
|
|
goto error;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, reg_val | 0x01))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_read_reg(psmouse, 0x0144) == -1)
|
|
|
|
goto error;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, 0x04))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_read_reg(psmouse, 0x0159) == -1)
|
|
|
|
goto error;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, 0x03))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_read_reg(psmouse, 0x0163) == -1)
|
|
|
|
goto error;
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0163, 0x03))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_read_reg(psmouse, 0x0162) == -1)
|
|
|
|
goto error;
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0162, 0x04))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
alps_exit_command_mode(psmouse);
|
|
|
|
|
|
|
|
/* Set rate and enable data reporting */
|
|
|
|
param[0] = 0x64;
|
|
|
|
if (ps2_command(ps2dev, param, PSMOUSE_CMD_SETRATE) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_ENABLE)) {
|
|
|
|
psmouse_err(psmouse, "Failed to enable data reporting\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
/*
|
|
|
|
* Leaving the touchpad in command mode will essentially render
|
|
|
|
* it unusable until the machine reboots, so exit it here just
|
|
|
|
* to be safe
|
|
|
|
*/
|
|
|
|
alps_exit_command_mode(psmouse);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:27:08 +08:00
|
|
|
static int alps_hw_init_rushmore_v3(struct psmouse *psmouse)
|
|
|
|
{
|
2013-02-14 14:28:07 +08:00
|
|
|
struct alps_data *priv = psmouse->private;
|
2013-02-14 14:27:08 +08:00
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
int reg_val, ret = -1;
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
if (priv->flags & ALPS_DUALPOINT) {
|
|
|
|
reg_val = alps_setup_trackstick_v3(psmouse,
|
|
|
|
ALPS_REG_BASE_RUSHMORE);
|
|
|
|
if (reg_val == -EIO)
|
|
|
|
goto error;
|
|
|
|
if (reg_val == -ENODEV)
|
|
|
|
priv->flags &= ~ALPS_DUALPOINT;
|
|
|
|
}
|
|
|
|
|
2013-02-22 14:58:20 +08:00
|
|
|
if (alps_enter_command_mode(psmouse) ||
|
2013-02-14 14:27:08 +08:00
|
|
|
alps_command_mode_read_reg(psmouse, 0xc2d9) == -1 ||
|
|
|
|
alps_command_mode_write_reg(psmouse, 0xc2cb, 0x00))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, 0xc2c6);
|
|
|
|
if (reg_val == -1)
|
|
|
|
goto error;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, reg_val & 0xfd))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0xc2c9, 0x64))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
/* enter absolute mode */
|
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, 0xc2c4);
|
|
|
|
if (reg_val == -1)
|
|
|
|
goto error;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, reg_val | 0x02))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
alps_exit_command_mode(psmouse);
|
|
|
|
return ps2_command(ps2dev, NULL, PSMOUSE_CMD_ENABLE);
|
|
|
|
|
|
|
|
error:
|
|
|
|
alps_exit_command_mode(psmouse);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
/* Must be in command mode when calling this function */
|
|
|
|
static int alps_absolute_mode_v4(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
int reg_val;
|
|
|
|
|
|
|
|
reg_val = alps_command_mode_read_reg(psmouse, 0x0004);
|
|
|
|
if (reg_val == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
reg_val |= 0x02;
|
|
|
|
if (__alps_command_mode_write_reg(psmouse, reg_val))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_hw_init_v4(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
unsigned char param[4];
|
|
|
|
|
2013-02-22 14:58:20 +08:00
|
|
|
if (alps_enter_command_mode(psmouse))
|
2011-11-08 11:53:36 +08:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_absolute_mode_v4(psmouse)) {
|
|
|
|
psmouse_err(psmouse, "Failed to enter absolute mode\n");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0007, 0x8c))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0149, 0x03))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0160, 0x03))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x017f, 0x15))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0151, 0x01))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0168, 0x03))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x014a, 0x03))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (alps_command_mode_write_reg(psmouse, 0x0161, 0x03))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
alps_exit_command_mode(psmouse);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This sequence changes the output from a 9-byte to an
|
|
|
|
* 8-byte format. All the same data seems to be present,
|
|
|
|
* just in a more compact format.
|
|
|
|
*/
|
|
|
|
param[0] = 0xc8;
|
|
|
|
param[1] = 0x64;
|
|
|
|
param[2] = 0x50;
|
|
|
|
if (ps2_command(ps2dev, ¶m[0], PSMOUSE_CMD_SETRATE) ||
|
|
|
|
ps2_command(ps2dev, ¶m[1], PSMOUSE_CMD_SETRATE) ||
|
|
|
|
ps2_command(ps2dev, ¶m[2], PSMOUSE_CMD_SETRATE) ||
|
|
|
|
ps2_command(ps2dev, param, PSMOUSE_CMD_GETID))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* Set rate and enable data reporting */
|
|
|
|
param[0] = 0x64;
|
|
|
|
if (ps2_command(ps2dev, param, PSMOUSE_CMD_SETRATE) ||
|
|
|
|
ps2_command(ps2dev, NULL, PSMOUSE_CMD_ENABLE)) {
|
|
|
|
psmouse_err(psmouse, "Failed to enable data reporting\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
/*
|
|
|
|
* Leaving the touchpad in command mode will essentially render
|
|
|
|
* it unusable until the machine reboots, so exit it here just
|
|
|
|
* to be safe
|
|
|
|
*/
|
|
|
|
alps_exit_command_mode(psmouse);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-02-22 14:58:28 +08:00
|
|
|
static int alps_hw_init_dolphin_v1(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct ps2dev *ps2dev = &psmouse->ps2dev;
|
|
|
|
unsigned char param[2];
|
|
|
|
|
|
|
|
/* This is dolphin "v1" as empirically defined by florin9doi */
|
|
|
|
param[0] = 0x64;
|
|
|
|
param[1] = 0x28;
|
|
|
|
|
|
|
|
if (ps2_command(ps2dev, NULL, PSMOUSE_CMD_SETSTREAM) ||
|
|
|
|
ps2_command(ps2dev, ¶m[0], PSMOUSE_CMD_SETRATE) ||
|
|
|
|
ps2_command(ps2dev, ¶m[1], PSMOUSE_CMD_SETRATE))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:22:08 +08:00
|
|
|
static void alps_set_defaults(struct alps_data *priv)
|
2011-11-08 11:53:36 +08:00
|
|
|
{
|
2013-02-14 14:23:34 +08:00
|
|
|
priv->byte0 = 0x8f;
|
|
|
|
priv->mask0 = 0x8f;
|
|
|
|
priv->flags = ALPS_DUALPOINT;
|
|
|
|
|
2013-02-14 14:24:55 +08:00
|
|
|
priv->x_max = 2000;
|
|
|
|
priv->y_max = 1400;
|
|
|
|
priv->x_bits = 15;
|
|
|
|
priv->y_bits = 11;
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
switch (priv->proto_version) {
|
2011-11-08 11:53:36 +08:00
|
|
|
case ALPS_PROTO_V1:
|
|
|
|
case ALPS_PROTO_V2:
|
2013-02-14 14:22:08 +08:00
|
|
|
priv->hw_init = alps_hw_init_v1_v2;
|
|
|
|
priv->process_packet = alps_process_packet_v1_v2;
|
|
|
|
priv->set_abs_params = alps_set_abs_params_st;
|
2011-11-08 11:53:36 +08:00
|
|
|
break;
|
|
|
|
case ALPS_PROTO_V3:
|
2013-02-14 14:22:08 +08:00
|
|
|
priv->hw_init = alps_hw_init_v3;
|
|
|
|
priv->process_packet = alps_process_packet_v3;
|
|
|
|
priv->set_abs_params = alps_set_abs_params_mt;
|
2013-02-14 14:26:11 +08:00
|
|
|
priv->decode_fields = alps_decode_pinnacle;
|
2013-02-14 14:23:04 +08:00
|
|
|
priv->nibble_commands = alps_v3_nibble_commands;
|
|
|
|
priv->addr_command = PSMOUSE_CMD_RESET_WRAP;
|
2011-11-08 11:53:36 +08:00
|
|
|
break;
|
|
|
|
case ALPS_PROTO_V4:
|
2013-02-14 14:22:08 +08:00
|
|
|
priv->hw_init = alps_hw_init_v4;
|
|
|
|
priv->process_packet = alps_process_packet_v4;
|
|
|
|
priv->set_abs_params = alps_set_abs_params_mt;
|
2013-02-14 14:23:04 +08:00
|
|
|
priv->nibble_commands = alps_v4_nibble_commands;
|
|
|
|
priv->addr_command = PSMOUSE_CMD_DISABLE;
|
2011-11-08 11:53:36 +08:00
|
|
|
break;
|
2013-02-22 14:58:28 +08:00
|
|
|
case ALPS_PROTO_V5:
|
|
|
|
priv->hw_init = alps_hw_init_dolphin_v1;
|
|
|
|
priv->process_packet = alps_process_packet_v3;
|
|
|
|
priv->decode_fields = alps_decode_dolphin;
|
|
|
|
priv->set_abs_params = alps_set_abs_params_mt;
|
|
|
|
priv->nibble_commands = alps_v3_nibble_commands;
|
|
|
|
priv->addr_command = PSMOUSE_CMD_RESET_WRAP;
|
|
|
|
priv->byte0 = 0xc8;
|
|
|
|
priv->mask0 = 0xc8;
|
|
|
|
priv->flags = 0;
|
|
|
|
priv->x_max = 1360;
|
|
|
|
priv->y_max = 660;
|
|
|
|
priv->x_bits = 23;
|
|
|
|
priv->y_bits = 12;
|
|
|
|
break;
|
2011-11-08 11:53:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
static int alps_match_table(struct psmouse *psmouse, struct alps_data *priv,
|
|
|
|
unsigned char *e7, unsigned char *ec)
|
2013-02-14 12:57:04 +08:00
|
|
|
{
|
2013-02-14 14:19:59 +08:00
|
|
|
const struct alps_model_info *model;
|
2013-02-14 12:57:04 +08:00
|
|
|
int i;
|
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
for (i = 0; i < ARRAY_SIZE(alps_model_data); i++) {
|
|
|
|
model = &alps_model_data[i];
|
|
|
|
|
|
|
|
if (!memcmp(e7, model->signature, sizeof(model->signature)) &&
|
|
|
|
(!model->command_mode_resp ||
|
|
|
|
model->command_mode_resp == ec[2])) {
|
|
|
|
|
|
|
|
priv->proto_version = model->proto_version;
|
2013-02-14 14:22:08 +08:00
|
|
|
alps_set_defaults(priv);
|
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
priv->flags = model->flags;
|
|
|
|
priv->byte0 = model->byte0;
|
|
|
|
priv->mask0 = model->mask0;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alps_identify(struct psmouse *psmouse, struct alps_data *priv)
|
|
|
|
{
|
|
|
|
unsigned char e6[4], e7[4], ec[4];
|
|
|
|
|
2013-02-14 12:57:04 +08:00
|
|
|
/*
|
|
|
|
* First try "E6 report".
|
|
|
|
* ALPS should return 0,0,10 or 0,0,100 if no buttons are pressed.
|
|
|
|
* The bits 0-2 of the first byte will be 1s if some buttons are
|
|
|
|
* pressed.
|
|
|
|
*/
|
2013-02-14 14:19:59 +08:00
|
|
|
if (alps_rpt_cmd(psmouse, PSMOUSE_CMD_SETRES,
|
|
|
|
PSMOUSE_CMD_SETSCALE11, e6))
|
|
|
|
return -EIO;
|
2013-02-14 12:57:04 +08:00
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
if ((e6[0] & 0xf8) != 0 || e6[1] != 0 || (e6[2] != 10 && e6[2] != 100))
|
|
|
|
return -EINVAL;
|
2013-02-14 12:57:04 +08:00
|
|
|
|
|
|
|
/*
|
2013-02-14 14:19:59 +08:00
|
|
|
* Now get the "E7" and "EC" reports. These will uniquely identify
|
|
|
|
* most ALPS touchpads.
|
2013-02-14 12:57:04 +08:00
|
|
|
*/
|
2013-02-14 14:19:59 +08:00
|
|
|
if (alps_rpt_cmd(psmouse, PSMOUSE_CMD_SETRES,
|
|
|
|
PSMOUSE_CMD_SETSCALE21, e7) ||
|
|
|
|
alps_rpt_cmd(psmouse, PSMOUSE_CMD_SETRES,
|
|
|
|
PSMOUSE_CMD_RESET_WRAP, ec) ||
|
|
|
|
alps_exit_command_mode(psmouse))
|
|
|
|
return -EIO;
|
2013-02-14 12:57:04 +08:00
|
|
|
|
2013-02-14 14:23:34 +08:00
|
|
|
if (alps_match_table(psmouse, priv, e7, ec) == 0) {
|
2013-02-22 14:58:28 +08:00
|
|
|
return 0;
|
|
|
|
} else if (e7[0] == 0x73 && e7[1] == 0x03 && e7[2] == 0x50 &&
|
|
|
|
ec[0] == 0x73 && ec[1] == 0x01) {
|
|
|
|
priv->proto_version = ALPS_PROTO_V5;
|
|
|
|
alps_set_defaults(priv);
|
|
|
|
|
2013-02-14 14:27:08 +08:00
|
|
|
return 0;
|
|
|
|
} else if (ec[0] == 0x88 && ec[1] == 0x08) {
|
|
|
|
priv->proto_version = ALPS_PROTO_V3;
|
|
|
|
alps_set_defaults(priv);
|
|
|
|
|
|
|
|
priv->hw_init = alps_hw_init_rushmore_v3;
|
|
|
|
priv->decode_fields = alps_decode_rushmore;
|
|
|
|
priv->x_bits = 16;
|
|
|
|
priv->y_bits = 12;
|
|
|
|
|
2013-02-14 14:28:07 +08:00
|
|
|
/* hack to make addr_command, nibble_command available */
|
|
|
|
psmouse->private = priv;
|
|
|
|
|
|
|
|
if (alps_probe_trackstick_v3(psmouse, ALPS_REG_BASE_RUSHMORE))
|
|
|
|
priv->flags &= ~ALPS_DUALPOINT;
|
|
|
|
|
2013-02-14 14:23:34 +08:00
|
|
|
return 0;
|
|
|
|
} else if (ec[0] == 0x88 && ec[1] == 0x07 &&
|
|
|
|
ec[2] >= 0x90 && ec[2] <= 0x9d) {
|
|
|
|
priv->proto_version = ALPS_PROTO_V3;
|
|
|
|
alps_set_defaults(priv);
|
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
return 0;
|
2013-02-14 14:23:34 +08:00
|
|
|
}
|
2013-02-14 12:57:04 +08:00
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
psmouse_info(psmouse,
|
2013-02-15 01:04:24 +08:00
|
|
|
"Unknown ALPS touchpad: E7=%3ph, EC=%3ph\n", e7, ec);
|
2013-02-14 12:57:04 +08:00
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
return -EINVAL;
|
2013-02-14 12:57:04 +08:00
|
|
|
}
|
|
|
|
|
2007-05-15 11:51:54 +08:00
|
|
|
static int alps_reconnect(struct psmouse *psmouse)
|
|
|
|
{
|
2013-02-14 14:19:59 +08:00
|
|
|
struct alps_data *priv = psmouse->private;
|
2009-11-17 14:12:22 +08:00
|
|
|
|
2007-05-15 11:51:54 +08:00
|
|
|
psmouse_reset(psmouse);
|
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
if (alps_identify(psmouse, priv) < 0)
|
2007-05-15 11:51:54 +08:00
|
|
|
return -1;
|
|
|
|
|
2013-02-14 14:22:08 +08:00
|
|
|
return priv->hw_init(psmouse);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_disconnect(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct alps_data *priv = psmouse->private;
|
2005-09-15 15:01:44 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
psmouse_reset(psmouse);
|
2009-12-16 00:39:50 +08:00
|
|
|
del_timer_sync(&priv->timer);
|
2005-09-15 15:01:44 +08:00
|
|
|
input_unregister_device(priv->dev2);
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(priv);
|
|
|
|
}
|
|
|
|
|
2013-02-14 14:22:08 +08:00
|
|
|
static void alps_set_abs_params_st(struct alps_data *priv,
|
|
|
|
struct input_dev *dev1)
|
|
|
|
{
|
|
|
|
input_set_abs_params(dev1, ABS_X, 0, 1023, 0, 0);
|
|
|
|
input_set_abs_params(dev1, ABS_Y, 0, 767, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void alps_set_abs_params_mt(struct alps_data *priv,
|
|
|
|
struct input_dev *dev1)
|
|
|
|
{
|
|
|
|
set_bit(INPUT_PROP_SEMI_MT, dev1->propbit);
|
|
|
|
input_mt_init_slots(dev1, 2, 0);
|
2013-02-14 14:24:55 +08:00
|
|
|
input_set_abs_params(dev1, ABS_MT_POSITION_X, 0, priv->x_max, 0, 0);
|
|
|
|
input_set_abs_params(dev1, ABS_MT_POSITION_Y, 0, priv->y_max, 0, 0);
|
2013-02-14 14:22:08 +08:00
|
|
|
|
|
|
|
set_bit(BTN_TOOL_DOUBLETAP, dev1->keybit);
|
|
|
|
set_bit(BTN_TOOL_TRIPLETAP, dev1->keybit);
|
|
|
|
set_bit(BTN_TOOL_QUADTAP, dev1->keybit);
|
|
|
|
|
2013-02-14 14:24:55 +08:00
|
|
|
input_set_abs_params(dev1, ABS_X, 0, priv->x_max, 0, 0);
|
|
|
|
input_set_abs_params(dev1, ABS_Y, 0, priv->y_max, 0, 0);
|
2013-02-14 14:22:08 +08:00
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
int alps_init(struct psmouse *psmouse)
|
|
|
|
{
|
|
|
|
struct alps_data *priv;
|
2005-09-15 15:01:44 +08:00
|
|
|
struct input_dev *dev1 = psmouse->dev, *dev2;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-04-12 13:31:13 +08:00
|
|
|
priv = kzalloc(sizeof(struct alps_data), GFP_KERNEL);
|
2005-09-15 15:01:44 +08:00
|
|
|
dev2 = input_allocate_device();
|
|
|
|
if (!priv || !dev2)
|
2005-04-17 06:20:36 +08:00
|
|
|
goto init_fail;
|
2005-09-15 15:01:44 +08:00
|
|
|
|
|
|
|
priv->dev2 = dev2;
|
2009-12-16 00:39:50 +08:00
|
|
|
setup_timer(&priv->timer, alps_flush_packet, (unsigned long)psmouse);
|
|
|
|
|
2007-05-15 11:51:54 +08:00
|
|
|
psmouse->private = priv;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2011-11-08 11:53:36 +08:00
|
|
|
psmouse_reset(psmouse);
|
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
if (alps_identify(psmouse, priv) < 0)
|
2009-11-17 14:12:22 +08:00
|
|
|
goto init_fail;
|
|
|
|
|
2013-02-14 14:22:08 +08:00
|
|
|
if (priv->hw_init(psmouse))
|
2005-04-17 06:20:36 +08:00
|
|
|
goto init_fail;
|
|
|
|
|
2009-12-12 15:54:54 +08:00
|
|
|
/*
|
|
|
|
* Undo part of setup done for us by psmouse core since touchpad
|
|
|
|
* is not a relative device.
|
|
|
|
*/
|
|
|
|
__clear_bit(EV_REL, dev1->evbit);
|
|
|
|
__clear_bit(REL_X, dev1->relbit);
|
|
|
|
__clear_bit(REL_Y, dev1->relbit);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now set up our capabilities.
|
|
|
|
*/
|
2007-10-19 14:40:32 +08:00
|
|
|
dev1->evbit[BIT_WORD(EV_KEY)] |= BIT_MASK(EV_KEY);
|
|
|
|
dev1->keybit[BIT_WORD(BTN_TOUCH)] |= BIT_MASK(BTN_TOUCH);
|
|
|
|
dev1->keybit[BIT_WORD(BTN_TOOL_FINGER)] |= BIT_MASK(BTN_TOOL_FINGER);
|
2009-11-17 14:12:22 +08:00
|
|
|
dev1->keybit[BIT_WORD(BTN_LEFT)] |=
|
|
|
|
BIT_MASK(BTN_LEFT) | BIT_MASK(BTN_RIGHT);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-10-19 14:40:32 +08:00
|
|
|
dev1->evbit[BIT_WORD(EV_ABS)] |= BIT_MASK(EV_ABS);
|
2011-11-08 11:53:36 +08:00
|
|
|
|
2013-02-14 14:22:08 +08:00
|
|
|
priv->set_abs_params(priv, dev1);
|
2005-09-15 15:01:44 +08:00
|
|
|
input_set_abs_params(dev1, ABS_PRESSURE, 0, 127, 0, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_WHEEL) {
|
2007-10-19 14:40:32 +08:00
|
|
|
dev1->evbit[BIT_WORD(EV_REL)] |= BIT_MASK(EV_REL);
|
|
|
|
dev1->relbit[BIT_WORD(REL_WHEEL)] |= BIT_MASK(REL_WHEEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & (ALPS_FW_BK_1 | ALPS_FW_BK_2)) {
|
2007-10-19 14:40:32 +08:00
|
|
|
dev1->keybit[BIT_WORD(BTN_FORWARD)] |= BIT_MASK(BTN_FORWARD);
|
|
|
|
dev1->keybit[BIT_WORD(BTN_BACK)] |= BIT_MASK(BTN_BACK);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2013-02-14 12:56:33 +08:00
|
|
|
if (priv->flags & ALPS_FOUR_BUTTONS) {
|
2009-11-17 14:12:22 +08:00
|
|
|
dev1->keybit[BIT_WORD(BTN_0)] |= BIT_MASK(BTN_0);
|
|
|
|
dev1->keybit[BIT_WORD(BTN_1)] |= BIT_MASK(BTN_1);
|
|
|
|
dev1->keybit[BIT_WORD(BTN_2)] |= BIT_MASK(BTN_2);
|
|
|
|
dev1->keybit[BIT_WORD(BTN_3)] |= BIT_MASK(BTN_3);
|
|
|
|
} else {
|
|
|
|
dev1->keybit[BIT_WORD(BTN_MIDDLE)] |= BIT_MASK(BTN_MIDDLE);
|
|
|
|
}
|
|
|
|
|
2006-06-26 13:45:10 +08:00
|
|
|
snprintf(priv->phys, sizeof(priv->phys), "%s/input1", psmouse->ps2dev.serio->phys);
|
2005-09-15 15:01:44 +08:00
|
|
|
dev2->phys = priv->phys;
|
2013-02-14 12:56:33 +08:00
|
|
|
dev2->name = (priv->flags & ALPS_DUALPOINT) ?
|
|
|
|
"DualPoint Stick" : "PS/2 Mouse";
|
2005-09-15 15:01:44 +08:00
|
|
|
dev2->id.bustype = BUS_I8042;
|
|
|
|
dev2->id.vendor = 0x0002;
|
|
|
|
dev2->id.product = PSMOUSE_ALPS;
|
|
|
|
dev2->id.version = 0x0000;
|
2008-03-18 12:29:18 +08:00
|
|
|
dev2->dev.parent = &psmouse->ps2dev.serio->dev;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-10-19 14:40:32 +08:00
|
|
|
dev2->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REL);
|
2009-11-17 14:12:22 +08:00
|
|
|
dev2->relbit[BIT_WORD(REL_X)] = BIT_MASK(REL_X) | BIT_MASK(REL_Y);
|
|
|
|
dev2->keybit[BIT_WORD(BTN_LEFT)] =
|
|
|
|
BIT_MASK(BTN_LEFT) | BIT_MASK(BTN_MIDDLE) | BIT_MASK(BTN_RIGHT);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-04-12 13:31:13 +08:00
|
|
|
if (input_register_device(priv->dev2))
|
|
|
|
goto init_fail;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
psmouse->protocol_handler = alps_process_byte;
|
2006-01-14 13:27:37 +08:00
|
|
|
psmouse->poll = alps_poll;
|
2005-04-17 06:20:36 +08:00
|
|
|
psmouse->disconnect = alps_disconnect;
|
|
|
|
psmouse->reconnect = alps_reconnect;
|
2013-02-14 12:56:33 +08:00
|
|
|
psmouse->pktsize = priv->proto_version == ALPS_PROTO_V4 ? 8 : 6;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-01-14 13:27:37 +08:00
|
|
|
/* We are having trouble resyncing ALPS touchpads so disable it for now */
|
|
|
|
psmouse->resync_time = 0;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
init_fail:
|
2007-04-12 13:31:13 +08:00
|
|
|
psmouse_reset(psmouse);
|
2005-09-15 15:01:44 +08:00
|
|
|
input_free_device(dev2);
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(priv);
|
2007-05-15 11:51:54 +08:00
|
|
|
psmouse->private = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-09-10 10:13:20 +08:00
|
|
|
int alps_detect(struct psmouse *psmouse, bool set_properties)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2013-02-14 14:19:59 +08:00
|
|
|
struct alps_data dummy;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-02-14 14:19:59 +08:00
|
|
|
if (alps_identify(psmouse, &dummy) < 0)
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (set_properties) {
|
|
|
|
psmouse->vendor = "ALPS";
|
2013-02-14 14:19:59 +08:00
|
|
|
psmouse->name = dummy.flags & ALPS_DUALPOINT ?
|
2005-05-29 15:28:29 +08:00
|
|
|
"DualPoint TouchPad" : "GlidePoint";
|
2013-02-14 14:19:59 +08:00
|
|
|
psmouse->model = dummy.proto_version << 8;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|