1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Copyright (c) 2001 Arndt Schoenewald
4*4882a593Smuzhiyun * Copyright (c) 2000-2001 Vojtech Pavlik
5*4882a593Smuzhiyun * Copyright (c) 2000 Mark Fletcher
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Sponsored by Quelltext AG (http://www.quelltext-ag.de), Dortmund, Germany
8*4882a593Smuzhiyun */
9*4882a593Smuzhiyun
10*4882a593Smuzhiyun /*
11*4882a593Smuzhiyun * Driver to use Handykey's Twiddler (the first edition, i.e. the one with
12*4882a593Smuzhiyun * the RS232 interface) as a joystick under Linux
13*4882a593Smuzhiyun *
14*4882a593Smuzhiyun * The Twiddler is a one-handed chording keyboard featuring twelve buttons on
15*4882a593Smuzhiyun * the front, six buttons on the top, and a built-in tilt sensor. The buttons
16*4882a593Smuzhiyun * on the front, which are grouped as four rows of three buttons, are pressed
17*4882a593Smuzhiyun * by the four fingers (this implies only one button per row can be held down
18*4882a593Smuzhiyun * at the same time) and the buttons on the top are for the thumb. The tilt
19*4882a593Smuzhiyun * sensor delivers X and Y axis data depending on how the Twiddler is held.
20*4882a593Smuzhiyun * Additional information can be found at http://www.handykey.com.
21*4882a593Smuzhiyun *
22*4882a593Smuzhiyun * This driver does not use the Twiddler for its intended purpose, i.e. as
23*4882a593Smuzhiyun * a chording keyboard, but as a joystick: pressing and releasing a button
24*4882a593Smuzhiyun * immediately sends a corresponding button event, and tilting it generates
25*4882a593Smuzhiyun * corresponding ABS_X and ABS_Y events. This turns the Twiddler into a game
26*4882a593Smuzhiyun * controller with amazing 18 buttons :-)
27*4882a593Smuzhiyun *
28*4882a593Smuzhiyun * Note: The Twiddler2 (the successor of the Twiddler that connects directly
29*4882a593Smuzhiyun * to the PS/2 keyboard and mouse ports) is NOT supported by this driver!
30*4882a593Smuzhiyun *
31*4882a593Smuzhiyun * For questions or feedback regarding this driver module please contact:
32*4882a593Smuzhiyun * Arndt Schoenewald <arndt@quelltext.com>
33*4882a593Smuzhiyun */
34*4882a593Smuzhiyun
35*4882a593Smuzhiyun /*
36*4882a593Smuzhiyun */
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun #include <linux/kernel.h>
39*4882a593Smuzhiyun #include <linux/module.h>
40*4882a593Smuzhiyun #include <linux/slab.h>
41*4882a593Smuzhiyun #include <linux/input.h>
42*4882a593Smuzhiyun #include <linux/serio.h>
43*4882a593Smuzhiyun
44*4882a593Smuzhiyun #define DRIVER_DESC "Handykey Twiddler keyboard as a joystick driver"
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun MODULE_DESCRIPTION(DRIVER_DESC);
47*4882a593Smuzhiyun MODULE_LICENSE("GPL");
48*4882a593Smuzhiyun
49*4882a593Smuzhiyun /*
50*4882a593Smuzhiyun * Constants.
51*4882a593Smuzhiyun */
52*4882a593Smuzhiyun
53*4882a593Smuzhiyun #define TWIDJOY_MAX_LENGTH 5
54*4882a593Smuzhiyun
55*4882a593Smuzhiyun static struct twidjoy_button_spec {
56*4882a593Smuzhiyun int bitshift;
57*4882a593Smuzhiyun int bitmask;
58*4882a593Smuzhiyun int buttons[3];
59*4882a593Smuzhiyun }
60*4882a593Smuzhiyun twidjoy_buttons[] = {
61*4882a593Smuzhiyun { 0, 3, { BTN_A, BTN_B, BTN_C } },
62*4882a593Smuzhiyun { 2, 3, { BTN_X, BTN_Y, BTN_Z } },
63*4882a593Smuzhiyun { 4, 3, { BTN_TL, BTN_TR, BTN_TR2 } },
64*4882a593Smuzhiyun { 6, 3, { BTN_SELECT, BTN_START, BTN_MODE } },
65*4882a593Smuzhiyun { 8, 1, { BTN_BASE5 } },
66*4882a593Smuzhiyun { 9, 1, { BTN_BASE } },
67*4882a593Smuzhiyun { 10, 1, { BTN_BASE3 } },
68*4882a593Smuzhiyun { 11, 1, { BTN_BASE4 } },
69*4882a593Smuzhiyun { 12, 1, { BTN_BASE2 } },
70*4882a593Smuzhiyun { 13, 1, { BTN_BASE6 } },
71*4882a593Smuzhiyun { 0, 0, { 0 } }
72*4882a593Smuzhiyun };
73*4882a593Smuzhiyun
74*4882a593Smuzhiyun /*
75*4882a593Smuzhiyun * Per-Twiddler data.
76*4882a593Smuzhiyun */
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun struct twidjoy {
79*4882a593Smuzhiyun struct input_dev *dev;
80*4882a593Smuzhiyun int idx;
81*4882a593Smuzhiyun unsigned char data[TWIDJOY_MAX_LENGTH];
82*4882a593Smuzhiyun char phys[32];
83*4882a593Smuzhiyun };
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun /*
86*4882a593Smuzhiyun * twidjoy_process_packet() decodes packets the driver receives from the
87*4882a593Smuzhiyun * Twiddler. It updates the data accordingly.
88*4882a593Smuzhiyun */
89*4882a593Smuzhiyun
twidjoy_process_packet(struct twidjoy * twidjoy)90*4882a593Smuzhiyun static void twidjoy_process_packet(struct twidjoy *twidjoy)
91*4882a593Smuzhiyun {
92*4882a593Smuzhiyun struct input_dev *dev = twidjoy->dev;
93*4882a593Smuzhiyun unsigned char *data = twidjoy->data;
94*4882a593Smuzhiyun struct twidjoy_button_spec *bp;
95*4882a593Smuzhiyun int button_bits, abs_x, abs_y;
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun button_bits = ((data[1] & 0x7f) << 7) | (data[0] & 0x7f);
98*4882a593Smuzhiyun
99*4882a593Smuzhiyun for (bp = twidjoy_buttons; bp->bitmask; bp++) {
100*4882a593Smuzhiyun int value = (button_bits & (bp->bitmask << bp->bitshift)) >> bp->bitshift;
101*4882a593Smuzhiyun int i;
102*4882a593Smuzhiyun
103*4882a593Smuzhiyun for (i = 0; i < bp->bitmask; i++)
104*4882a593Smuzhiyun input_report_key(dev, bp->buttons[i], i+1 == value);
105*4882a593Smuzhiyun }
106*4882a593Smuzhiyun
107*4882a593Smuzhiyun abs_x = ((data[4] & 0x07) << 5) | ((data[3] & 0x7C) >> 2);
108*4882a593Smuzhiyun if (data[4] & 0x08) abs_x -= 256;
109*4882a593Smuzhiyun
110*4882a593Smuzhiyun abs_y = ((data[3] & 0x01) << 7) | ((data[2] & 0x7F) >> 0);
111*4882a593Smuzhiyun if (data[3] & 0x02) abs_y -= 256;
112*4882a593Smuzhiyun
113*4882a593Smuzhiyun input_report_abs(dev, ABS_X, -abs_x);
114*4882a593Smuzhiyun input_report_abs(dev, ABS_Y, +abs_y);
115*4882a593Smuzhiyun
116*4882a593Smuzhiyun input_sync(dev);
117*4882a593Smuzhiyun }
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun /*
120*4882a593Smuzhiyun * twidjoy_interrupt() is called by the low level driver when characters
121*4882a593Smuzhiyun * are ready for us. We then buffer them for further processing, or call the
122*4882a593Smuzhiyun * packet processing routine.
123*4882a593Smuzhiyun */
124*4882a593Smuzhiyun
twidjoy_interrupt(struct serio * serio,unsigned char data,unsigned int flags)125*4882a593Smuzhiyun static irqreturn_t twidjoy_interrupt(struct serio *serio, unsigned char data, unsigned int flags)
126*4882a593Smuzhiyun {
127*4882a593Smuzhiyun struct twidjoy *twidjoy = serio_get_drvdata(serio);
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun /* All Twiddler packets are 5 bytes. The fact that the first byte
130*4882a593Smuzhiyun * has a MSB of 0 and all other bytes have a MSB of 1 can be used
131*4882a593Smuzhiyun * to check and regain sync. */
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun if ((data & 0x80) == 0)
134*4882a593Smuzhiyun twidjoy->idx = 0; /* this byte starts a new packet */
135*4882a593Smuzhiyun else if (twidjoy->idx == 0)
136*4882a593Smuzhiyun return IRQ_HANDLED; /* wrong MSB -- ignore this byte */
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun if (twidjoy->idx < TWIDJOY_MAX_LENGTH)
139*4882a593Smuzhiyun twidjoy->data[twidjoy->idx++] = data;
140*4882a593Smuzhiyun
141*4882a593Smuzhiyun if (twidjoy->idx == TWIDJOY_MAX_LENGTH) {
142*4882a593Smuzhiyun twidjoy_process_packet(twidjoy);
143*4882a593Smuzhiyun twidjoy->idx = 0;
144*4882a593Smuzhiyun }
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun return IRQ_HANDLED;
147*4882a593Smuzhiyun }
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun /*
150*4882a593Smuzhiyun * twidjoy_disconnect() is the opposite of twidjoy_connect()
151*4882a593Smuzhiyun */
152*4882a593Smuzhiyun
twidjoy_disconnect(struct serio * serio)153*4882a593Smuzhiyun static void twidjoy_disconnect(struct serio *serio)
154*4882a593Smuzhiyun {
155*4882a593Smuzhiyun struct twidjoy *twidjoy = serio_get_drvdata(serio);
156*4882a593Smuzhiyun
157*4882a593Smuzhiyun serio_close(serio);
158*4882a593Smuzhiyun serio_set_drvdata(serio, NULL);
159*4882a593Smuzhiyun input_unregister_device(twidjoy->dev);
160*4882a593Smuzhiyun kfree(twidjoy);
161*4882a593Smuzhiyun }
162*4882a593Smuzhiyun
163*4882a593Smuzhiyun /*
164*4882a593Smuzhiyun * twidjoy_connect() is the routine that is called when someone adds a
165*4882a593Smuzhiyun * new serio device. It looks for the Twiddler, and if found, registers
166*4882a593Smuzhiyun * it as an input device.
167*4882a593Smuzhiyun */
168*4882a593Smuzhiyun
twidjoy_connect(struct serio * serio,struct serio_driver * drv)169*4882a593Smuzhiyun static int twidjoy_connect(struct serio *serio, struct serio_driver *drv)
170*4882a593Smuzhiyun {
171*4882a593Smuzhiyun struct twidjoy_button_spec *bp;
172*4882a593Smuzhiyun struct twidjoy *twidjoy;
173*4882a593Smuzhiyun struct input_dev *input_dev;
174*4882a593Smuzhiyun int err = -ENOMEM;
175*4882a593Smuzhiyun int i;
176*4882a593Smuzhiyun
177*4882a593Smuzhiyun twidjoy = kzalloc(sizeof(struct twidjoy), GFP_KERNEL);
178*4882a593Smuzhiyun input_dev = input_allocate_device();
179*4882a593Smuzhiyun if (!twidjoy || !input_dev)
180*4882a593Smuzhiyun goto fail1;
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun twidjoy->dev = input_dev;
183*4882a593Smuzhiyun snprintf(twidjoy->phys, sizeof(twidjoy->phys), "%s/input0", serio->phys);
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun input_dev->name = "Handykey Twiddler";
186*4882a593Smuzhiyun input_dev->phys = twidjoy->phys;
187*4882a593Smuzhiyun input_dev->id.bustype = BUS_RS232;
188*4882a593Smuzhiyun input_dev->id.vendor = SERIO_TWIDJOY;
189*4882a593Smuzhiyun input_dev->id.product = 0x0001;
190*4882a593Smuzhiyun input_dev->id.version = 0x0100;
191*4882a593Smuzhiyun input_dev->dev.parent = &serio->dev;
192*4882a593Smuzhiyun
193*4882a593Smuzhiyun input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS);
194*4882a593Smuzhiyun input_set_abs_params(input_dev, ABS_X, -50, 50, 4, 4);
195*4882a593Smuzhiyun input_set_abs_params(input_dev, ABS_Y, -50, 50, 4, 4);
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun for (bp = twidjoy_buttons; bp->bitmask; bp++)
198*4882a593Smuzhiyun for (i = 0; i < bp->bitmask; i++)
199*4882a593Smuzhiyun set_bit(bp->buttons[i], input_dev->keybit);
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun serio_set_drvdata(serio, twidjoy);
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun err = serio_open(serio, drv);
204*4882a593Smuzhiyun if (err)
205*4882a593Smuzhiyun goto fail2;
206*4882a593Smuzhiyun
207*4882a593Smuzhiyun err = input_register_device(twidjoy->dev);
208*4882a593Smuzhiyun if (err)
209*4882a593Smuzhiyun goto fail3;
210*4882a593Smuzhiyun
211*4882a593Smuzhiyun return 0;
212*4882a593Smuzhiyun
213*4882a593Smuzhiyun fail3: serio_close(serio);
214*4882a593Smuzhiyun fail2: serio_set_drvdata(serio, NULL);
215*4882a593Smuzhiyun fail1: input_free_device(input_dev);
216*4882a593Smuzhiyun kfree(twidjoy);
217*4882a593Smuzhiyun return err;
218*4882a593Smuzhiyun }
219*4882a593Smuzhiyun
220*4882a593Smuzhiyun /*
221*4882a593Smuzhiyun * The serio driver structure.
222*4882a593Smuzhiyun */
223*4882a593Smuzhiyun
224*4882a593Smuzhiyun static const struct serio_device_id twidjoy_serio_ids[] = {
225*4882a593Smuzhiyun {
226*4882a593Smuzhiyun .type = SERIO_RS232,
227*4882a593Smuzhiyun .proto = SERIO_TWIDJOY,
228*4882a593Smuzhiyun .id = SERIO_ANY,
229*4882a593Smuzhiyun .extra = SERIO_ANY,
230*4882a593Smuzhiyun },
231*4882a593Smuzhiyun { 0 }
232*4882a593Smuzhiyun };
233*4882a593Smuzhiyun
234*4882a593Smuzhiyun MODULE_DEVICE_TABLE(serio, twidjoy_serio_ids);
235*4882a593Smuzhiyun
236*4882a593Smuzhiyun static struct serio_driver twidjoy_drv = {
237*4882a593Smuzhiyun .driver = {
238*4882a593Smuzhiyun .name = "twidjoy",
239*4882a593Smuzhiyun },
240*4882a593Smuzhiyun .description = DRIVER_DESC,
241*4882a593Smuzhiyun .id_table = twidjoy_serio_ids,
242*4882a593Smuzhiyun .interrupt = twidjoy_interrupt,
243*4882a593Smuzhiyun .connect = twidjoy_connect,
244*4882a593Smuzhiyun .disconnect = twidjoy_disconnect,
245*4882a593Smuzhiyun };
246*4882a593Smuzhiyun
247*4882a593Smuzhiyun module_serio_driver(twidjoy_drv);
248