OSDN Git Service

Add inputattach for ThinkPad X200 tablet
authorChih-Wei Huang <cwhuang@linux.org.tw>
Mon, 12 Jan 2015 16:19:59 +0000 (00:19 +0800)
committerChih-Wei Huang <cwhuang@linux.org.tw>
Mon, 19 Jan 2015 03:49:15 +0000 (11:49 +0800)
The patch adds a new service inputattach and starts it
if ThinkPad X200 is detected.

Contributed by Greg McGee <gjmcgee@gmail.com>.

init.sh
init.x86.rc
inputattach/Android.mk [new file with mode: 0644]
inputattach/README [new file with mode: 0644]
inputattach/inputattach.c [new file with mode: 0644]
inputattach/serio-ids.h [new file with mode: 0644]
packages.mk

diff --git a/init.sh b/init.sh
index 6be42c4..da6b9dd 100644 (file)
--- a/init.sh
+++ b/init.sh
@@ -230,7 +230,7 @@ function create_pointercal()
 function init_tscal()
 {
        case "$PRODUCT" in
-               T91|T101|ET2002|74499FU|945GSE-ITE8712|CF-19[CDYFGKLP]*)
+               T91|T101|ET2002|945GSE-ITE8712|CF-19[CDYFGKLP]*)
                        create_pointercal
                        return
                        ;;
@@ -341,7 +341,7 @@ function do_bootcomplete()
                        ;;
                7448???|7449???|7450???|7453???) # ThinkPad X200 Tablet
                        start tablet-mode
-                       start wacom-input
+                       start inputattach
                        setkeycodes 0xe012 158
                        setkeycodes 0x66 172
                        setkeycodes 0x6b 127
index 4a3001c..fff5207 100644 (file)
@@ -159,6 +159,10 @@ service tablet-mode /system/bin/tablet-mode
     disabled
     oneshot
 
+service inputattach /system/bin/inputattach --baud 19200 --daemon -w8001 /dev/ttyS0
+    disabled
+    oneshot
+
 on property:init.svc.wpa_supplicant=stopped
     stop dhcpcd
 
diff --git a/inputattach/Android.mk b/inputattach/Android.mk
new file mode 100644 (file)
index 0000000..6d0f91b
--- /dev/null
@@ -0,0 +1,13 @@
+# Copyright (C) 2015 The Android-x86 Open Source Project
+
+LOCAL_PATH := $(my-dir)
+
+include $(CLEAR_VARS)
+
+LOCAL_SRC_FILES := inputattach.c
+LOCAL_CFLAGS := -Werror
+
+LOCAL_MODULE := inputattach
+LOCAL_MODULE_TAGS := optional
+
+include $(BUILD_EXECUTABLE)
diff --git a/inputattach/README b/inputattach/README
new file mode 100644 (file)
index 0000000..fdd6abc
--- /dev/null
@@ -0,0 +1,50 @@
+This version of inputattach and serio.ids.h s from the linux console project v 1.4.7
+The following is the README file included from the original author/version
+******************************
+
+2010 - 2011 Ping Cheng, Wacom <pingc@wacom.com>
+
+inputattach is to provide /dev/input/event, an USB-like logical port, for Wacom
+serial ISDv4 digitizers on Linux. It requires wacom_w8001.ko being loaded in your
+kernel and the following:
+
+1.     inputattach.c
+2.     serio-ids.h: a header file for inputattach.c to compile on different platforms.
+3.     compile the code:  gcc inputattach.c -o inputattach
+
+Procedures to test the driver and device:
+1.     login to yourself and switch to root then issue:
+       xxd /dev/ttyS0 and move Wacom pen on the tablet.
+If you see data displays on the terminal where you launched xxd, this is the
+port the device is mapped to. If not, try ttyS1, 2,.... I see ttyS4 sometime.
+We assume it is on /dev/ttyS0 from now on.
+
+2.     ls /dev/input and keep a note on how many "event"s you see. We need the
+largest number later.
+
+3.     Connect your serial kernel driver to an input event by:
+       ./inputattach --wacom /dev/ttyS0  (if your device is at baud rate 19200)
+or
+       ./inputattach --wacom-384 /dev/ttyS0  (if your device is at baud rate 38400)
+
+4.     Check which port it is mapped to by:
+       ls /dev/input
+
+You should see a new /dev/input/event# added in the above output, that is the port
+your device is mapped to. If you have X server running and the latest Wacom X server
+driver is installed, you should see cursor movement when you move the pen on the
+tablet now.
+
+If you want to view the data directly from the port by evtest or some other kernel
+programs, you need to disable Wacom X driver or launch the program from a console
+before X server starts. Otherwise, the Wacom X driver or some other X input device
+drivcers, such as xf86-input-evdev, may grab the events before your program is launched.
+
+In order to let Wacom X driver communicate with the wacom_w8001.ko kernel driver
+instead of to the device directly, you need to install the newly built inputattach
+to a path your env is aware of, such as /usr/bin or /usr/sbin. Then add
+
+inputattach --wacom /dev/ttyS0 (or inputattach --wacom-384 /dev/ttyS0, whichever applies)
+
+to a system start script, such as /etc/rc.local, so the device will be mapped to a
+/dev/input/event# before X driver starts.
diff --git a/inputattach/inputattach.c b/inputattach/inputattach.c
new file mode 100644 (file)
index 0000000..bbadee7
--- /dev/null
@@ -0,0 +1,899 @@
+/*
+ * $Id$
+ *
+ *  Copyright (c) 1999-2000 Vojtech Pavlik
+ *
+ *  Sponsored by SuSE
+ *
+ *  Twiddler support Copyright (c) 2001 Arndt Schoenewald
+ *  Sponsored by Quelltext AG (http://www.quelltext-ag.de), Dortmund, Germany
+ *
+ *  Sahara Touchit-213 mode added by Claudio Nieder 2008-05-01.
+ */
+
+/*
+ * Input line discipline attach program
+ */
+
+/*
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+ * 02110-1301 USA.
+ */
+
+#include <ctype.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <linux/serio.h>
+#include "serio-ids.h"
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/ioctl.h>
+#include <termios.h>
+#include <unistd.h>
+
+static int readchar(int fd, unsigned char *c, int timeout)
+{
+       struct timeval tv;
+       fd_set set;
+
+       tv.tv_sec = 0;
+       tv.tv_usec = timeout * 1000;
+
+       FD_ZERO(&set);
+       FD_SET(fd, &set);
+
+       if (!select(fd + 1, &set, NULL, NULL, &tv))
+               return -1;
+
+       if (read(fd, c, 1) != 1)
+               return -1;
+
+       return 0;
+}
+
+static void setline(int fd, int flags, int speed)
+{
+       struct termios t;
+
+       tcgetattr(fd, &t);
+
+       t.c_cflag = flags | CREAD | HUPCL | CLOCAL;
+       t.c_iflag = IGNBRK | IGNPAR;
+       t.c_oflag = 0;
+       t.c_lflag = 0;
+       t.c_cc[VMIN ] = 1;
+       t.c_cc[VTIME] = 0;
+
+       cfsetispeed(&t, speed);
+       cfsetospeed(&t, speed);
+
+       tcsetattr(fd, TCSANOW, &t);
+}
+
+static int logitech_command(int fd, char *c)
+{
+       int i;
+       unsigned char d;
+
+       for (i = 0; c[i]; i++) {
+               if (write(fd, c + i, 1) != 1)
+                       return -1;
+               if (readchar(fd, &d, 1000))
+                       return -1;
+               if (c[i] != d)
+                       return -1;
+       }
+       return 0;
+}
+
+static int magellan_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       if (write(fd, "m3\rpBB\rz\r", 9) != 9)
+               return -1;
+       return 0;
+}
+
+static int warrior_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       if (logitech_command(fd, "*S"))
+               return -1;
+
+       setline(fd, CS8, B4800);
+       return 0;
+}
+
+static int spaceball_waitchar(int fd, unsigned char c, char *d,
+                               int timeout)
+{
+       unsigned char b = 0;
+
+       while (!readchar(fd, &b, timeout)) {
+               if (b == 0x0a)
+                       continue;
+               *d++ = b;
+               if (b == c)
+                       break;
+       }
+
+       *d = 0;
+
+       return -(b != c);
+}
+
+static int spaceball_waitcmd(int fd, char c, char *d)
+{
+       int i;
+
+       for (i = 0; i < 8; i++) {
+               if (spaceball_waitchar(fd, 0x0d, d, 1000))
+                       return -1;
+               if (d[0] == c)
+                       return 0;
+       }
+
+       return -1;
+}
+
+static int spaceball_cmd(int fd, char *c, char *d)
+{
+       int i;
+
+       for (i = 0; c[i]; i++)
+               if (write(fd, c + i, 1) != 1)
+                       return -1;
+       if (write(fd, "\r", 1) != 1)
+               return -1;
+
+       i = spaceball_waitcmd(fd, toupper(c[0]), d);
+
+       return i;
+}
+
+#define SPACEBALL_1003         1
+#define SPACEBALL_2003B                3
+#define SPACEBALL_2003C                4
+#define SPACEBALL_3003C                7
+#define SPACEBALL_4000FLX      8
+#define SPACEBALL_4000FLX_L    9
+
+static int spaceball_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       char r[64];
+
+       if (spaceball_waitchar(fd, 0x11, r, 4000) ||
+           spaceball_waitchar(fd, 0x0d, r, 1000))
+               return -1;
+
+       if (spaceball_waitcmd(fd, '@', r))
+               return -1;
+
+       if (strncmp("@1 Spaceball alive", r, 18))
+               return -1;
+
+       if (spaceball_waitcmd(fd, '@', r))
+               return -1;
+
+       if (spaceball_cmd(fd, "hm", r))
+               return -1;
+
+       if (!strncmp("Hm2003B", r, 7))
+               *id = SPACEBALL_2003B;
+       if (!strncmp("Hm2003C", r, 7))
+               *id = SPACEBALL_2003C;
+       if (!strncmp("Hm3003C", r, 7))
+               *id = SPACEBALL_3003C;
+
+       /* spaceball 4000 returns 'HVFirmware' with v2.4.3 */
+       if (!strncasecmp("HvFirmware", r, 10)) {
+
+               if (spaceball_cmd(fd, "\"", r))
+                       return -1;
+
+               if (strncmp("\"1 Spaceball 4000 FLX", r, 21))
+                       return -1;
+
+               if (spaceball_waitcmd(fd, '"', r))
+                       return -1;
+
+               if (strstr(r, " L "))
+                       *id = SPACEBALL_4000FLX_L;
+               else
+                       *id = SPACEBALL_4000FLX;
+
+               if (spaceball_waitcmd(fd, '"', r))
+                       return -1;
+
+               if (spaceball_cmd(fd, "YS", r))
+                       return -1;
+
+               if (spaceball_cmd(fd, "M", r))
+                       return -1;
+
+               return 0;
+       }
+
+       if (spaceball_cmd(fd, "P@A@A", r) ||
+           spaceball_cmd(fd, "FT@", r)   ||
+           spaceball_cmd(fd, "MSS", r))
+               return -1;
+
+       return 0;
+}
+
+static int stinger_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       int i;
+       unsigned char c;
+       unsigned char *response = (unsigned char *)"\r\n0600520058C272";
+
+       if (write(fd, " E5E5", 5) != 5)         /* Enable command */
+               return -1;
+
+       for (i = 0; i < 16; i++)                /* Check for Stinger */
+               if (readchar(fd, &c, 200) || c != response[i])
+                       return -1;
+
+       return 0;
+}
+
+static int mzp_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       if (logitech_command(fd, "*X*q"))
+               return -1;
+
+       setline(fd, CS8, B9600);
+       return 0;
+}
+
+static int newton_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       unsigned int i;
+       unsigned char c;
+       unsigned char response[35] = {
+               0x16, 0x10, 0x02, 0x64, 0x5f, 0x69, 0x64, 0x00,
+               0x00, 0x00, 0x0c, 0x6b, 0x79, 0x62, 0x64, 0x61,
+               0x70, 0x70, 0x6c, 0x00, 0x00, 0x00, 0x01, 0x6e,
+               0x6f, 0x66, 0x6d, 0x00, 0x00, 0x00, 0x00, 0x10,
+               0x03, 0xdd, 0xe7
+       };
+
+       for (i = 0; i < sizeof(response); i++)
+               if (readchar(fd, &c, 400) || c != response[i])
+                       return -1;
+
+       return 0;
+}
+
+static int twiddler_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       unsigned char c[10];
+       int count, line;
+
+       /* Turn DTR off, otherwise the Twiddler won't send any data. */
+       if (ioctl(fd, TIOCMGET, &line) < 0)
+               return -1;
+       line &= ~TIOCM_DTR;
+       if (ioctl(fd, TIOCMSET, &line) < 0)
+               return -1;
+
+       /*
+        * Check whether the device on the serial line is the Twiddler.
+        *
+        * The Twiddler sends data packets of 5 bytes which have the following
+        * properties: the MSB is 0 on the first and 1 on all other bytes, and
+        * the high order nibble of the last byte is always 0x8.
+        *
+        * We read and check two of those 5 byte packets to be sure that we
+        * are indeed talking to a Twiddler.
+        */
+
+       /* Read at most 5 bytes until we find one with the MSB set to 0 */
+       for (count = 0; count < 5; count++) {
+               if (readchar(fd, c, 500))
+                       return -1;
+               if ((c[0] & 0x80) == 0)
+                       break;
+       }
+
+       if (count == 5) {
+               /* Could not find header byte in data stream */
+               return -1;
+       }
+
+       /* Read remaining 4 bytes plus the full next data packet */
+       for (count = 1; count < 10; count++)
+               if (readchar(fd, c + count, 500))
+                       return -1;
+
+       /* Check whether the bytes of both data packets obey the rules */
+       for (count = 1; count < 10; count++) {
+               if ((count % 5 == 0 && (c[count] & 0x80) != 0x00) ||
+                   (count % 5 == 4 && (c[count] & 0xF0) != 0x80) ||
+                   (count % 5 != 0 && (c[count] & 0x80) != 0x80)) {
+                       /* Invalid byte in data packet */
+                       return -1;
+               }
+       }
+
+       return 0;
+}
+
+static int pm6k_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       unsigned int i;
+       unsigned char cmd[6] = {0xF1, 0x00, 0x00, 0x00, 0x00, 0x0E};
+       unsigned char data[6] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
+
+       /* Enable the touchscreen */
+       if (write(fd, cmd, sizeof(cmd)) != sizeof(cmd))
+               return -1;
+
+       /* Read ACK */
+       for(i = 0; i < sizeof(data); i++)
+               if (readchar(fd, &data[i], 100) < 0)
+                       break ;
+
+       return 0;
+}
+
+static int fujitsu_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       unsigned char cmd, data;
+
+       /* Wake up the touchscreen */
+       cmd = 0xff; /* Dummy data */;
+       if (write(fd, &cmd, 1) != 1)
+               return -1;
+
+       /* Wait to settle down */
+       usleep(100 * 1000); /* 100 ms */
+
+       /* Reset the touchscreen */
+       cmd = 0x81; /* Cold reset */
+       if (write(fd, &cmd, 1) != 1)
+               return -1;
+
+       /* Read ACK */
+       if (readchar(fd, &data, 100) || (data & 0xbf) != 0x90)
+               return -1;
+
+       /* Read status */
+       if (readchar(fd, &data, 100) || data != 0x00)
+               return -1;
+
+       return 0;
+}
+
+static int tsc40_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       unsigned char cmd[2], data;
+       unsigned int eeprom;
+
+       /* Datasheet can be found here:
+        * http://www.distec.de/PDF/Drivers/DMC/TSC40_Protocol_Description.pdf
+        */
+
+#define TSC40_CMD_DATA1        0x01
+#define TSC40_CMD_RATE 0x05
+#define TSC40_CMD_ID   0x15
+#define TSC40_CMD_RESET        0x55
+
+#define TSC40_RATE_150 0x45
+#define TSC40_NACK     0x15
+
+       /* trigger a software reset to get into a well known state */
+       cmd[0] = TSC40_CMD_RESET;
+       if (write(fd, cmd, 1) != 1)
+               return -1;
+
+       /* wait to settle down */
+       usleep(15 * 1000); /* 15 ms */
+
+       /* read panel ID to check if an EEPROM is used */
+       cmd[0] = TSC40_CMD_ID;
+       if (write(fd, cmd, 1) != 1)
+               return -1;
+
+       if (readchar(fd, &data, 100))
+               return -1;
+
+       /* if bit7 is not set --> EEPROM is used */
+       eeprom = !((data & 0x80) >> 7);
+
+       /* ignore 2nd byte of ID cmd */
+       if (readchar(fd, &data, 100))
+               return -1;
+
+       /* set coordinate oupt rate setting */
+       cmd[0] = TSC40_CMD_RATE;
+       cmd[1] = TSC40_RATE_150;
+       if (write(fd, cmd, 2) != 2)
+               return -1;
+
+       /* read response */
+       if (readchar(fd, &data, 100))
+               return -1;
+
+       if ((data == TSC40_NACK) && (eeprom == 1)) {
+               /* get detailed failure information */
+               if (readchar(fd, &data, 100))
+                       return -1;
+
+               switch (data) {
+               case 0x02:      /* EEPROM data abnormal */
+               case 0x04:      /* EEPROM write error */
+               case 0x08:      /* Touch screen not connected */
+                       return -1;
+                       break;
+
+               default:
+                       /* 0x01: EEPROM data empty */
+                       break;
+               }
+       }
+
+       /* start sending coordinate informations */
+       cmd[0] = TSC40_CMD_DATA1;
+       if (write(fd, cmd, 1) != 1)
+               return -1;
+
+       return 0;
+}
+
+static int t213_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       char cmd[] = { 0x0a, 1, 'A' };
+       int count = 10;
+       int state = 0;
+       unsigned char data;
+
+       /*
+        * In case the controller is in "ELO-mode" send a few times
+        * the check active packet to force it into the documented
+        * touchkit mode.
+        */
+       while (count > 0) {
+               if (write(fd, &cmd, 3) != 3)
+                       return -1;
+               while (!readchar(fd, &data, 100)) {
+                       switch (state) {
+                       case 0:
+                               if (data == 0x0a) {
+                                       state = 1;
+                               }
+                               break;
+                       case 1:
+                               if (data == 1) {
+                                       state = 2;
+                               } else if (data != 0x0a) {
+                                       state = 0;
+                               }
+                               break;
+                       case 2:
+                               if (data == 'A') {
+                                       return 0;
+                               } else if (data == 0x0a) {
+                                       state = 1;
+                               } else {
+                                       state = 0;
+                               }
+                               break;
+                       }
+
+               }
+               count--;
+       }
+       return -1;
+}
+
+static int zhenhua_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       /* Zhen Hua 5 byte protocol: first (synchronization) byte allways
+        * contain 0xF7, next four bytes are axis of controller with values
+        * between 50-200.
+        * Incoming data (each byte) have reversed bits (lowest bit is
+        * highest bit) - something like little-endian but on bit level.
+        * Synchronization byte without reversing bits have (raw) value:
+        * 0xEF
+        *
+        * Initialization is almost same as twiddler_init */
+
+       unsigned char c[10];
+       int count;
+
+       for (count = 0; count < 5 ; count++) {
+               if (readchar(fd, c+0, 500)) return -1;
+               if (c[0] == 0xef) break;
+       }
+
+       if (count == 5) {
+               /* Could not find header byte in data stream */
+               return -1;
+       }
+
+       /* Read remaining 4 bytes plus the full next data packet */
+       for (count = 1; count < 10; count++) {
+               if (readchar(fd, c+count, 500)) return -1;
+       }
+
+       /* check if next sync byte exists */
+       if (c[5] != 0xef)
+               return -1;
+
+       return 0;
+
+}
+
+#define EP_PROMPT_MODE  "B"     /* Prompt mode */
+#define EP_ABSOLUTE     "F"     /* Absolute Mode */
+#define EP_UPPER_ORIGIN "b"     /* Origin upper left */
+#define EP_STREAM_MODE  "@"     /* Stream mode */
+
+static int easypen_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       char buf[256];
+
+       /* reset */
+       write(fd, 0, 1);
+       usleep(400000);
+
+       /* set prompt mode */
+       if (write(fd, EP_PROMPT_MODE, 1) == -1)
+               return -1;
+
+       /* clear buffer */
+       while (read(fd, buf, sizeof(buf)) == sizeof(buf));
+
+       /* set options */
+       if (write(fd, EP_ABSOLUTE EP_STREAM_MODE EP_UPPER_ORIGIN, 3) == -1)
+               return -1;
+
+       return 0;
+}
+
+static int dump_init(int fd, unsigned long *id, unsigned long *extra)
+{
+       unsigned char c = 0x80, o = 0;
+
+       if (write(fd, &c, 1) != 1)         /* Enable command */
+                return -1;
+
+       while (1)
+               if (!readchar(fd, &c, 1)) {
+                       printf("%02x (%c) ", c, ((c > 32) && (c < 127)) ? c : 'x');
+                       o = 1;
+               } else {
+                       if (o) {
+                               printf("\n");
+                               o = 0;
+                       }
+               }
+}
+
+struct input_types {
+       const char *name;
+       const char *name2;
+       const char *desc;
+       int speed;
+       int flags;
+       unsigned long type;
+       unsigned long id;
+       unsigned long extra;
+       int flush;
+       int (*init)(int fd, unsigned long *id, unsigned long *extra);
+};
+
+static struct input_types input_types[] = {
+{ "--sunkbd",          "-skb",         "Sun Type 4 and Type 5 keyboards",
+       B1200, CS8,
+       SERIO_SUNKBD,           0x00,   0x00,   1,      NULL },
+{ "--lkkbd",           "-lk",          "DEC LK201 / LK401 keyboards",
+       B4800, CS8|CSTOPB,
+       SERIO_LKKBD,            0x00,   0x00,   1,      NULL },
+{ "--vsxxx-aa",                "-vs",
+                       "DEC VSXXX-AA / VSXXX-GA mouse and VSXXX-A tablet",
+       B4800, CS8|CSTOPB|PARENB|PARODD,
+       SERIO_VSXXXAA,          0x00,   0x00,   1,      NULL },
+{ "--spaceorb",                "-orb",         "SpaceOrb 360 / SpaceBall Avenger",
+       B9600, CS8,
+       SERIO_SPACEORB,         0x00,   0x00,   1,      NULL },
+{ "--spaceball",       "-sbl",         "SpaceBall 2003 / 3003 / 4000 FLX",
+       B9600, CS8,
+       SERIO_SPACEBALL,        0x00,   0x00,   0,      spaceball_init },
+{ "--magellan",                "-mag",         "Magellan / SpaceMouse",
+       B9600, CS8 | CSTOPB | CRTSCTS,
+       SERIO_MAGELLAN,         0x00,   0x00,   1,      magellan_init },
+{ "--warrior",         "-war",         "WingMan Warrior",
+       B1200, CS7 | CSTOPB,
+       SERIO_WARRIOR,          0x00,   0x00,   1,      warrior_init },
+{ "--stinger",         "-sting",       "Gravis Stinger",
+       B1200, CS8,
+       SERIO_STINGER,          0x00,   0x00,   1,      stinger_init },
+{ "--mousesystems",    "-msc",         "3-button Mouse Systems mouse",
+       B1200, CS8,
+       SERIO_MSC,              0x00,   0x01,   1,      NULL },
+{ "--sunmouse",                "-sun",         "3-button Sun mouse",
+       B1200, CS8,
+       SERIO_SUN,              0x00,   0x01,   1,      NULL },
+{ "--microsoft",       "-bare",        "2-button Microsoft mouse",
+       B1200, CS7,
+       SERIO_MS,               0x00,   0x00,   1,      NULL },
+{ "--mshack",          "-ms",          "3-button mouse in Microsoft mode",
+       B1200, CS7,
+       SERIO_MS,               0x00,   0x01,   1,      NULL },
+{ "--mouseman",                "-mman",        "3-button Logitech / Genius mouse",
+       B1200, CS7,
+       SERIO_MP,               0x00,   0x01,   1,      NULL },
+{ "--intellimouse",    "-ms3",         "Microsoft IntelliMouse",
+       B1200, CS7,
+       SERIO_MZ,               0x00,   0x11,   1,      NULL },
+{ "--mmwheel",         "-mmw",
+                       "Logitech mouse with 4-5 buttons or a wheel",
+       B1200, CS7 | CSTOPB,
+       SERIO_MZP,              0x00,   0x13,   1,      mzp_init },
+{ "--iforce",          "-ifor",        "I-Force joystick or wheel",
+       B38400, CS8,
+       SERIO_IFORCE,           0x00,   0x00,   0,      NULL },
+{ "--newtonkbd",       "-newt",        "Newton keyboard",
+       B9600, CS8,
+       SERIO_NEWTON,           0x00,   0x00,   1,      newton_init },
+{ "--h3600ts",         "-ipaq",        "Ipaq h3600 touchscreen",
+       B115200, CS8,
+       SERIO_H3600,            0x00,   0x00,   0,      NULL },
+{ "--stowawaykbd",     "-ipaqkbd",     "Stowaway keyboard",
+       B115200, CS8,
+       SERIO_STOWAWAY,         0x00,   0x00,   1,      NULL },
+{ "--ps2serkbd",       "-ps2ser",      "PS/2 via serial keyboard",
+       B1200, CS8,
+       SERIO_PS2SER,           0x00,   0x00,   1,      NULL },
+{ "--twiddler",                "-twid",        "Handykey Twiddler chording keyboard",
+       B2400, CS8,
+       SERIO_TWIDKBD,          0x00,   0x00,   0,      twiddler_init },
+{ "--twiddler-joy",    "-twidjoy",     "Handykey Twiddler used as a joystick",
+       B2400, CS8,
+       SERIO_TWIDJOY,          0x00,   0x00,   0,      twiddler_init },
+{ "--elotouch",                "-elo",         "ELO touchscreen, 10-byte mode",
+       B9600, CS8,
+       SERIO_ELO,              0x00,   0x00,   0,      NULL },
+{ "--elo4002",         "-elo6b",       "ELO touchscreen, 6-byte mode",
+       B9600, CS8 | CRTSCTS,
+       SERIO_ELO,              0x01,   0x00,   0,      NULL },
+{ "--elo271-140",      "-elo4b",       "ELO touchscreen, 4-byte mode",
+       B9600, CS8 | CRTSCTS,
+       SERIO_ELO,              0x02,   0x00,   0,      NULL },
+{ "--elo261-280",      "-elo3b",       "ELO Touchscreen, 3-byte mode",
+       B9600, CS8 | CRTSCTS,
+       SERIO_ELO,              0x03,   0x00,   0,      NULL },
+{ "--mtouch",          "-mtouch",      "MicroTouch (3M) touchscreen",
+       B9600, CS8 | CRTSCTS,
+       SERIO_MICROTOUCH,       0x00,   0x00,   0,      NULL },
+#ifdef SERIO_TSC40
+{ "--tsc",             "-tsc",         "TSC-10/25/40 serial touchscreen",
+       B9600, CS8,
+       SERIO_TSC40,            0x00,   0x00,   0,      tsc40_init },
+#endif
+{ "--touchit213",      "-t213",        "Sahara Touch-iT213 Tablet PC",
+       B9600, CS8,
+       SERIO_TOUCHIT213,       0x00,   0x00,   0,      t213_init },
+{ "--touchright",      "-tr",  "Touchright serial touchscreen",
+       B9600, CS8 | CRTSCTS,
+       SERIO_TOUCHRIGHT,       0x00,   0x00,   0,      NULL },
+{ "--touchwin",                "-tw",  "Touchwindow serial touchscreen",
+       B4800, CS8 | CRTSCTS,
+       SERIO_TOUCHWIN,         0x00,   0x00,   0,      NULL },
+{ "--penmount9000",            "-pm9k",        "PenMount 9000 touchscreen",
+       B19200, CS8,
+       SERIO_PENMOUNT,         0x00,   0x00,   0,      NULL },
+{ "--penmount6000",            "-pm6k",        "PenMount 6000 touchscreen",
+       B19200, CS8,
+       SERIO_PENMOUNT,         0x01,   0x00,   0,      pm6k_init },
+{ "--penmount3000",            "-pm3k",        "PenMount 3000 touchscreen",
+       B38400, CS8,
+       SERIO_PENMOUNT,         0x02,   0x00,   0,      NULL },
+{ "--penmount6250",            "-pmm1",        "PenMount 6250 touchscreen",
+       B19200, CS8,
+       SERIO_PENMOUNT,         0x03,   0x00,   0,      NULL },
+{ "--fujitsu",         "-fjt", "Fujitsu serial touchscreen",
+       B9600, CS8,
+       SERIO_FUJITSU,          0x00,   0x00,   1,      fujitsu_init },
+{ "--ps2mult", "-ps2m",        "PS/2 serial multiplexer",
+       B57600, CS8,
+       SERIO_PS2MULT,          0x00,   0x00,   1,      NULL },
+{ "--zhen-hua",                "-zhen",        "Zhen Hua 5-byte protocol",
+       B19200, CS8,
+       SERIO_ZHENHUA,          0x00,   0x00,   0,      zhenhua_init },
+{ "--easypen",         "-ep",          "Genius EasyPen 3x4 tablet",
+       B9600, CS8|CREAD|CLOCAL|HUPCL|PARENB|PARODD,
+       SERIO_EASYPEN,          0x00,   0x00,   0,      easypen_init },
+#ifdef SERIO_TAOSEVM
+{ "--taos-evm",                "-taos",        "TAOS evaluation module",
+       B1200, CS8,
+       SERIO_TAOSEVM,          0,      0,      0,      NULL },
+#endif
+{ "--dump",            "-dump",        "Just enable device",
+       B2400, CS8,
+       0,                      0x00,   0x00,   0,      dump_init },
+{ "--w8001",           "-w8001",       "Wacom W8001",
+       B38400, CS8,
+       SERIO_W8001,            0x00,   0x00,   0,      NULL },
+{ NULL, NULL, NULL, 0, 0, 0, 0, 0, 0, NULL }
+};
+
+static void show_help(void)
+{
+       struct input_types *type;
+
+       puts("");
+       puts("Usage: inputattach [--daemon] [--baud <baud>] [--always] [--noinit] <mode> <device>");
+       puts("");
+       puts("Modes:");
+
+       for (type = input_types; type->name; type++)
+               printf("  %-16s %-8s  %s\n",
+                       type->name, type->name2, type->desc);
+
+       puts("");
+}
+
+/* palmed wisdom from http://stackoverflow.com/questions/1674162/ */
+#define RETRY_ERROR(x) (x == EAGAIN || x == EWOULDBLOCK || x == EINTR)
+
+int main(int argc, char **argv)
+{
+       unsigned long devt;
+       int ldisc;
+       struct input_types *type = NULL;
+       const char *device = NULL;
+       int daemon_mode = 0;
+       int need_device = 0;
+       unsigned long id, extra;
+       int fd;
+       int i;
+       unsigned char c;
+       int retval;
+       int baud = -1;
+       int ignore_init_res = 0;
+       int no_init = 0;
+
+       for (i = 1; i < argc; i++) {
+               if (!strcasecmp(argv[i], "--help")) {
+                       show_help();
+                       return EXIT_SUCCESS;
+               } else if (!strcasecmp(argv[i], "--daemon")) {
+                       daemon_mode = 1;
+               } else if (!strcasecmp(argv[i], "--always")) {
+                       ignore_init_res = 1;
+               } else if (!strcasecmp(argv[i], "--noinit")) {
+                       no_init = 1;
+               } else if (need_device) {
+                       device = argv[i];
+                       need_device = 0;
+               } else if (!strcasecmp(argv[i], "--baud")) {
+                       if (argc <= i + 1) {
+                               show_help();
+                               fprintf(stderr,
+                                       "inputattach: require baud rate\n");
+                               return EXIT_FAILURE;
+                       }
+
+                       baud = atoi(argv[++i]);
+               } else {
+                       if (type && type->name) {
+                               fprintf(stderr,
+                                       "inputattach: '%s' - "
+                                       "only one mode allowed\n", argv[i]);
+                               return EXIT_FAILURE;
+                       }
+                       for (type = input_types; type->name; type++) {
+                               if (!strcasecmp(argv[i], type->name) ||
+                                   !strcasecmp(argv[i], type->name2)) {
+                                       break;
+                               }
+                       }
+                       if (!type->name) {
+                               fprintf(stderr,
+                                       "inputattach: invalid mode '%s'\n",
+                                       argv[i]);
+                               return EXIT_FAILURE;
+                       }
+                       need_device = 1;
+               }
+       }
+
+       if (!type || !type->name) {
+               fprintf(stderr, "inputattach: must specify mode\n");
+               return EXIT_FAILURE;
+        }
+
+       if (need_device) {
+               fprintf(stderr, "inputattach: must specify device\n");
+               return EXIT_FAILURE;
+       }
+
+       fd = open(device, O_RDWR | O_NOCTTY | O_NONBLOCK);
+       if (fd < 0) {
+               fprintf(stderr, "inputattach: '%s' - %s\n",
+                       device, strerror(errno));
+               return 1;
+       }
+
+       switch(baud) {
+       case -1: break;
+       case 2400: type->speed = B2400; break;
+       case 4800: type->speed = B4800; break;
+       case 9600: type->speed = B9600; break;
+       case 19200: type->speed = B19200; break;
+       case 38400: type->speed = B38400; break;
+       default:
+               fprintf(stderr, "inputattach: invalid baud rate '%d'\n",
+                               baud);
+               return EXIT_FAILURE;
+       }
+
+       setline(fd, type->flags, type->speed);
+
+       if (type->flush)
+               while (!readchar(fd, &c, 100))
+                       /* empty */;
+
+       id = type->id;
+       extra = type->extra;
+
+       if (type->init && !no_init) {
+               if (type->init(fd, &id, &extra)) {
+                       if (ignore_init_res) {
+                               fprintf(stderr, "inputattach: ignored device initialization failure\n");
+                       } else {
+                               fprintf(stderr, "inputattach: device initialization failed\n");
+                               return EXIT_FAILURE;
+                       }
+               }
+       }
+
+       ldisc = N_MOUSE;
+       if (ioctl(fd, TIOCSETD, &ldisc) < 0) {
+               fprintf(stderr, "inputattach: can't set line discipline\n");
+               return EXIT_FAILURE;
+       }
+
+       devt = type->type | (id << 8) | (extra << 16);
+
+       if (ioctl(fd, SPIOCSTYPE, &devt) < 0) {
+               fprintf(stderr, "inputattach: can't set device type\n");
+               return EXIT_FAILURE;
+       }
+
+       retval = EXIT_SUCCESS;
+       if (daemon_mode && daemon(0, 0) < 0) {
+               perror("inputattach");
+               retval = EXIT_FAILURE;
+       }
+
+       errno = 0;
+       do {
+               i = read(fd, NULL, 0);
+       } while (RETRY_ERROR(errno));
+
+       ldisc = 0;
+       if (errno == 0) {
+               // If we've never managed to read, avoid resetting the line
+               // discipline - another inputattach is probably running
+               ioctl(fd, TIOCSETD, &ldisc);
+       }
+       close(fd);
+
+       return retval;
+}
diff --git a/inputattach/serio-ids.h b/inputattach/serio-ids.h
new file mode 100644 (file)
index 0000000..0f2223b
--- /dev/null
@@ -0,0 +1,132 @@
+#ifndef _SERIO_IDS_H
+#define _SERIO_IDS_H
+
+#ifndef SERIO_RS232
+# define SERIO_RS232   0x02
+#endif
+
+/*
+ * Serio types
+ */
+#ifndef SERIO_UNKNOWN
+# define SERIO_UNKNOWN         0x00
+#endif
+#ifndef SERIO_MSC
+# define SERIO_MSC             0x01
+#endif
+#ifndef SERIO_SUN
+# define SERIO_SUN             0x02
+#endif
+#ifndef SERIO_MS
+# define SERIO_MS              0x03
+#endif
+#ifndef SERIO_MP
+# define SERIO_MP              0x04
+#endif
+#ifndef SERIO_MZ
+# define SERIO_MZ              0x05
+#endif
+#ifndef SERIO_MZP
+# define SERIO_MZP             0x06
+#endif
+#ifndef SERIO_MZPP
+# define SERIO_MZPP            0x07
+#endif
+#ifndef SERIO_VSXXXAA
+# define SERIO_VSXXXAA         0x08
+#endif
+#ifndef SERIO_SUNKBD
+# define SERIO_SUNKBD          0x10
+#endif
+#ifndef SERIO_WARRIOR
+# define SERIO_WARRIOR         0x18
+#endif
+#ifndef SERIO_SPACEORB
+# define SERIO_SPACEORB                0x19
+#endif
+#ifndef SERIO_MAGELLAN
+# define SERIO_MAGELLAN                0x1a
+#endif
+#ifndef SERIO_SPACEBALL
+# define SERIO_SPACEBALL       0x1b
+#endif
+#ifndef SERIO_GUNZE
+# define SERIO_GUNZE           0x1c
+#endif
+#ifndef SERIO_IFORCE
+# define SERIO_IFORCE          0x1d
+#endif
+#ifndef SERIO_STINGER
+# define SERIO_STINGER         0x1e
+#endif
+#ifndef SERIO_NEWTON
+# define SERIO_NEWTON          0x1f
+#endif
+#ifndef SERIO_STOWAWAY
+# define SERIO_STOWAWAY                0x20
+#endif
+#ifndef SERIO_H3600
+# define SERIO_H3600           0x21
+#endif
+#ifndef SERIO_PS2SER
+# define SERIO_PS2SER          0x22
+#endif
+#ifndef SERIO_TWIDKBD
+# define SERIO_TWIDKBD         0x23
+#endif
+#ifndef SERIO_TWIDJOY
+# define SERIO_TWIDJOY         0x24
+#endif
+#ifndef SERIO_HIL
+# define SERIO_HIL             0x25
+#endif
+#ifndef SERIO_SNES232
+# define SERIO_SNES232         0x26
+#endif
+#ifndef SERIO_SEMTECH
+# define SERIO_SEMTECH         0x27
+#endif
+#ifndef SERIO_LKKBD
+# define SERIO_LKKBD           0x28
+#endif
+#ifndef SERIO_ELO
+# define SERIO_ELO             0x29
+#endif
+#ifndef SERIO_MICROTOUCH
+# define SERIO_MICROTOUCH      0x30
+#endif
+#ifndef SERIO_PENMOUNT
+# define SERIO_PENMOUNT                0x31
+#endif
+#ifndef SERIO_TOUCHRIGHT
+# define SERIO_TOUCHRIGHT      0x32
+#endif
+#ifndef SERIO_TOUCHWIN
+# define SERIO_TOUCHWIN                0x33
+#endif
+#ifndef SERIO_TAOSEVM
+# define SERIO_TAOSEVM         0x34
+#endif
+#ifndef SERIO_FUJITSU
+# define SERIO_FUJITSU         0x35
+#endif
+#ifndef SERIO_ZHENHUA
+# define SERIO_ZHENHUA         0x36
+#endif
+#ifndef SERIO_INEXIO
+# define SERIO_INEXIO          0x37
+#endif
+#ifndef SERIO_TOUCHIT213
+# define SERIO_TOUCHIT213      0x38
+#endif
+#ifndef SERIO_W8001
+# define SERIO_W8001           0x39
+#endif
+#ifndef SERIO_PSMULT
+# define SERIO_PS2MULT         0x3c
+#endif
+#ifndef SERIO_EASYPEN
+# define SERIO_EASYPEN         0x3e
+#endif
+
+#endif
index db2f0d9..82cea09 100644 (file)
@@ -39,6 +39,7 @@ PRODUCT_PACKAGES := \
     gps.default \
     hwcomposer.x86 \
     icu.dat \
+    inputattach \
     io_switch \
     libFFmpegExtractor \
     libhuaweigeneric-ril \