11a59d1b8SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
21da177e4SLinus Torvalds /*
31da177e4SLinus Torvalds * NES, SNES, N64, MultiSystem, PSX gamepad driver for Linux
41da177e4SLinus Torvalds *
51da177e4SLinus Torvalds * Copyright (c) 1999-2004 Vojtech Pavlik <vojtech@suse.cz>
61da177e4SLinus Torvalds * Copyright (c) 2004 Peter Nelson <rufus-kernel@hackish.org>
71da177e4SLinus Torvalds *
81da177e4SLinus Torvalds * Based on the work of:
91da177e4SLinus Torvalds * Andree Borrmann John Dahlstrom
101da177e4SLinus Torvalds * David Kuder Nathan Hand
11b157d55eSRaphael Assenat * Raphael Assenat
121da177e4SLinus Torvalds */
131da177e4SLinus Torvalds
14a1e12747SDmitry Torokhov #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
15a1e12747SDmitry Torokhov
161da177e4SLinus Torvalds #include <linux/kernel.h>
171da177e4SLinus Torvalds #include <linux/delay.h>
181da177e4SLinus Torvalds #include <linux/module.h>
191da177e4SLinus Torvalds #include <linux/init.h>
201da177e4SLinus Torvalds #include <linux/parport.h>
211da177e4SLinus Torvalds #include <linux/input.h>
2272ba9f0cSIngo Molnar #include <linux/mutex.h>
235a0e3ad6STejun Heo #include <linux/slab.h>
241da177e4SLinus Torvalds
251da177e4SLinus Torvalds MODULE_AUTHOR("Vojtech Pavlik <vojtech@ucw.cz>");
261da177e4SLinus Torvalds MODULE_DESCRIPTION("NES, SNES, N64, MultiSystem, PSX gamepad driver");
271da177e4SLinus Torvalds MODULE_LICENSE("GPL");
281da177e4SLinus Torvalds
2917dd3f0fSDmitry Torokhov #define GC_MAX_PORTS 3
3017dd3f0fSDmitry Torokhov #define GC_MAX_DEVICES 5
311da177e4SLinus Torvalds
3217dd3f0fSDmitry Torokhov struct gc_config {
3317dd3f0fSDmitry Torokhov int args[GC_MAX_DEVICES + 1];
3478167236SDmitry Torokhov unsigned int nargs;
3517dd3f0fSDmitry Torokhov };
361da177e4SLinus Torvalds
37a517e87cSSudip Mukherjee static struct gc_config gc_cfg[GC_MAX_PORTS];
3817dd3f0fSDmitry Torokhov
3978167236SDmitry Torokhov module_param_array_named(map, gc_cfg[0].args, int, &gc_cfg[0].nargs, 0);
4017dd3f0fSDmitry Torokhov MODULE_PARM_DESC(map, "Describes first set of devices (<parport#>,<pad1>,<pad2>,..<pad5>)");
4178167236SDmitry Torokhov module_param_array_named(map2, gc_cfg[1].args, int, &gc_cfg[1].nargs, 0);
4217dd3f0fSDmitry Torokhov MODULE_PARM_DESC(map2, "Describes second set of devices");
4378167236SDmitry Torokhov module_param_array_named(map3, gc_cfg[2].args, int, &gc_cfg[2].nargs, 0);
4417dd3f0fSDmitry Torokhov MODULE_PARM_DESC(map3, "Describes third set of devices");
451da177e4SLinus Torvalds
461da177e4SLinus Torvalds /* see also gs_psx_delay parameter in PSX support section */
471da177e4SLinus Torvalds
480995174dSDmitry Torokhov enum gc_type {
490995174dSDmitry Torokhov GC_NONE = 0,
500995174dSDmitry Torokhov GC_SNES,
510995174dSDmitry Torokhov GC_NES,
520995174dSDmitry Torokhov GC_NES4,
530995174dSDmitry Torokhov GC_MULTI,
540995174dSDmitry Torokhov GC_MULTI2,
550995174dSDmitry Torokhov GC_N64,
560995174dSDmitry Torokhov GC_PSX,
570995174dSDmitry Torokhov GC_DDR,
580995174dSDmitry Torokhov GC_SNESMOUSE,
590995174dSDmitry Torokhov GC_MAX
600995174dSDmitry Torokhov };
611da177e4SLinus Torvalds
621da177e4SLinus Torvalds #define GC_REFRESH_TIME HZ/100
631da177e4SLinus Torvalds
640995174dSDmitry Torokhov struct gc_pad {
650995174dSDmitry Torokhov struct input_dev *dev;
660995174dSDmitry Torokhov enum gc_type type;
670995174dSDmitry Torokhov char phys[32];
680995174dSDmitry Torokhov };
690995174dSDmitry Torokhov
701da177e4SLinus Torvalds struct gc {
711da177e4SLinus Torvalds struct pardevice *pd;
720995174dSDmitry Torokhov struct gc_pad pads[GC_MAX_DEVICES];
731da177e4SLinus Torvalds struct timer_list timer;
740995174dSDmitry Torokhov int pad_count[GC_MAX];
751da177e4SLinus Torvalds int used;
76a517e87cSSudip Mukherjee int parportno;
7772ba9f0cSIngo Molnar struct mutex mutex;
781da177e4SLinus Torvalds };
791da177e4SLinus Torvalds
807aa9e0e8SScott Moreau struct gc_subdev {
817aa9e0e8SScott Moreau unsigned int idx;
827aa9e0e8SScott Moreau };
837aa9e0e8SScott Moreau
841da177e4SLinus Torvalds static struct gc *gc_base[3];
851da177e4SLinus Torvalds
86af930d64SDmitry Torokhov static const int gc_status_bit[] = { 0x40, 0x80, 0x20, 0x10, 0x08 };
871da177e4SLinus Torvalds
88af930d64SDmitry Torokhov static const char *gc_names[] = {
89d38fcb96SDmitry Torokhov NULL, "SNES pad", "NES pad", "NES FourPort", "Multisystem joystick",
901da177e4SLinus Torvalds "Multisystem 2-button joystick", "N64 controller", "PSX controller",
91d38fcb96SDmitry Torokhov "PSX DDR controller", "SNES mouse"
92d38fcb96SDmitry Torokhov };
93d38fcb96SDmitry Torokhov
941da177e4SLinus Torvalds /*
951da177e4SLinus Torvalds * N64 support.
961da177e4SLinus Torvalds */
971da177e4SLinus Torvalds
98af930d64SDmitry Torokhov static const unsigned char gc_n64_bytes[] = { 0, 1, 13, 15, 14, 12, 10, 11, 2, 3 };
99af930d64SDmitry Torokhov static const short gc_n64_btn[] = {
100d38fcb96SDmitry Torokhov BTN_A, BTN_B, BTN_C, BTN_X, BTN_Y, BTN_Z,
101d38fcb96SDmitry Torokhov BTN_TL, BTN_TR, BTN_TRIGGER, BTN_START
102d38fcb96SDmitry Torokhov };
1031da177e4SLinus Torvalds
1041da177e4SLinus Torvalds #define GC_N64_LENGTH 32 /* N64 bit length, not including stop bit */
1057aa9e0e8SScott Moreau #define GC_N64_STOP_LENGTH 5 /* Length of encoded stop bit */
1067aa9e0e8SScott Moreau #define GC_N64_CMD_00 0x11111111UL
1077aa9e0e8SScott Moreau #define GC_N64_CMD_01 0xd1111111UL
1087aa9e0e8SScott Moreau #define GC_N64_CMD_03 0xdd111111UL
1097aa9e0e8SScott Moreau #define GC_N64_CMD_1b 0xdd1dd111UL
1107aa9e0e8SScott Moreau #define GC_N64_CMD_c0 0x111111ddUL
1117aa9e0e8SScott Moreau #define GC_N64_CMD_80 0x1111111dUL
1127aa9e0e8SScott Moreau #define GC_N64_STOP_BIT 0x1d /* Encoded stop bit */
1137aa9e0e8SScott Moreau #define GC_N64_REQUEST_DATA GC_N64_CMD_01 /* the request data command */
1141da177e4SLinus Torvalds #define GC_N64_DELAY 133 /* delay between transmit request, and response ready (us) */
1151da177e4SLinus Torvalds #define GC_N64_DWS 3 /* delay between write segments (required for sound playback because of ISA DMA) */
1161da177e4SLinus Torvalds /* GC_N64_DWS > 24 is known to fail */
1171da177e4SLinus Torvalds #define GC_N64_POWER_W 0xe2 /* power during write (transmit request) */
1181da177e4SLinus Torvalds #define GC_N64_POWER_R 0xfd /* power during read */
1191da177e4SLinus Torvalds #define GC_N64_OUT 0x1d /* output bits to the 4 pads */
1201da177e4SLinus Torvalds /* Reading the main axes of any N64 pad is known to fail if the corresponding bit */
1211da177e4SLinus Torvalds /* in GC_N64_OUT is pulled low on the output port (by any routine) for more */
1221da177e4SLinus Torvalds /* than 123 us */
1231da177e4SLinus Torvalds #define GC_N64_CLOCK 0x02 /* clock bits for read */
1241da177e4SLinus Torvalds
1251da177e4SLinus Torvalds /*
1267aa9e0e8SScott Moreau * Used for rumble code.
1277aa9e0e8SScott Moreau */
1287aa9e0e8SScott Moreau
1297aa9e0e8SScott Moreau /* Send encoded command */
gc_n64_send_command(struct gc * gc,unsigned long cmd,unsigned char target)1307aa9e0e8SScott Moreau static void gc_n64_send_command(struct gc *gc, unsigned long cmd,
1317aa9e0e8SScott Moreau unsigned char target)
1327aa9e0e8SScott Moreau {
1337aa9e0e8SScott Moreau struct parport *port = gc->pd->port;
1347aa9e0e8SScott Moreau int i;
1357aa9e0e8SScott Moreau
1367aa9e0e8SScott Moreau for (i = 0; i < GC_N64_LENGTH; i++) {
1377aa9e0e8SScott Moreau unsigned char data = (cmd >> i) & 1 ? target : 0;
1387aa9e0e8SScott Moreau parport_write_data(port, GC_N64_POWER_W | data);
1397aa9e0e8SScott Moreau udelay(GC_N64_DWS);
1407aa9e0e8SScott Moreau }
1417aa9e0e8SScott Moreau }
1427aa9e0e8SScott Moreau
1437aa9e0e8SScott Moreau /* Send stop bit */
gc_n64_send_stop_bit(struct gc * gc,unsigned char target)1447aa9e0e8SScott Moreau static void gc_n64_send_stop_bit(struct gc *gc, unsigned char target)
1457aa9e0e8SScott Moreau {
1467aa9e0e8SScott Moreau struct parport *port = gc->pd->port;
1477aa9e0e8SScott Moreau int i;
1487aa9e0e8SScott Moreau
1497aa9e0e8SScott Moreau for (i = 0; i < GC_N64_STOP_LENGTH; i++) {
1507aa9e0e8SScott Moreau unsigned char data = (GC_N64_STOP_BIT >> i) & 1 ? target : 0;
1517aa9e0e8SScott Moreau parport_write_data(port, GC_N64_POWER_W | data);
1527aa9e0e8SScott Moreau udelay(GC_N64_DWS);
1537aa9e0e8SScott Moreau }
1547aa9e0e8SScott Moreau }
1557aa9e0e8SScott Moreau
1567aa9e0e8SScott Moreau /*
1571da177e4SLinus Torvalds * gc_n64_read_packet() reads an N64 packet.
158d38fcb96SDmitry Torokhov * Each pad uses one bit per byte. So all pads connected to this port
159d38fcb96SDmitry Torokhov * are read in parallel.
1601da177e4SLinus Torvalds */
1611da177e4SLinus Torvalds
gc_n64_read_packet(struct gc * gc,unsigned char * data)1621da177e4SLinus Torvalds static void gc_n64_read_packet(struct gc *gc, unsigned char *data)
1631da177e4SLinus Torvalds {
1641da177e4SLinus Torvalds int i;
1651da177e4SLinus Torvalds unsigned long flags;
1661da177e4SLinus Torvalds
1671da177e4SLinus Torvalds /*
1681da177e4SLinus Torvalds * Request the pad to transmit data
1691da177e4SLinus Torvalds */
1701da177e4SLinus Torvalds
1711da177e4SLinus Torvalds local_irq_save(flags);
1727aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_REQUEST_DATA, GC_N64_OUT);
1737aa9e0e8SScott Moreau gc_n64_send_stop_bit(gc, GC_N64_OUT);
1741da177e4SLinus Torvalds local_irq_restore(flags);
1751da177e4SLinus Torvalds
1761da177e4SLinus Torvalds /*
177d38fcb96SDmitry Torokhov * Wait for the pad response to be loaded into the 33-bit register
178d38fcb96SDmitry Torokhov * of the adapter.
1791da177e4SLinus Torvalds */
1801da177e4SLinus Torvalds
1811da177e4SLinus Torvalds udelay(GC_N64_DELAY);
1821da177e4SLinus Torvalds
1831da177e4SLinus Torvalds /*
1841da177e4SLinus Torvalds * Grab data (ignoring the last bit, which is a stop bit)
1851da177e4SLinus Torvalds */
1861da177e4SLinus Torvalds
1871da177e4SLinus Torvalds for (i = 0; i < GC_N64_LENGTH; i++) {
1881da177e4SLinus Torvalds parport_write_data(gc->pd->port, GC_N64_POWER_R);
1897aa9e0e8SScott Moreau udelay(2);
1901da177e4SLinus Torvalds data[i] = parport_read_status(gc->pd->port);
1911da177e4SLinus Torvalds parport_write_data(gc->pd->port, GC_N64_POWER_R | GC_N64_CLOCK);
1921da177e4SLinus Torvalds }
1931da177e4SLinus Torvalds
1941da177e4SLinus Torvalds /*
195d38fcb96SDmitry Torokhov * We must wait 200 ms here for the controller to reinitialize before
196d38fcb96SDmitry Torokhov * the next read request. No worries as long as gc_read is polled less
197d38fcb96SDmitry Torokhov * frequently than this.
1981da177e4SLinus Torvalds */
1991da177e4SLinus Torvalds
2001da177e4SLinus Torvalds }
2011da177e4SLinus Torvalds
gc_n64_process_packet(struct gc * gc)202c7fd018dSDmitry Torokhov static void gc_n64_process_packet(struct gc *gc)
203c7fd018dSDmitry Torokhov {
204c7fd018dSDmitry Torokhov unsigned char data[GC_N64_LENGTH];
205c7fd018dSDmitry Torokhov struct input_dev *dev;
206c7fd018dSDmitry Torokhov int i, j, s;
207d38fcb96SDmitry Torokhov signed char x, y;
208c7fd018dSDmitry Torokhov
209c7fd018dSDmitry Torokhov gc_n64_read_packet(gc, data);
210c7fd018dSDmitry Torokhov
211c7fd018dSDmitry Torokhov for (i = 0; i < GC_MAX_DEVICES; i++) {
212c7fd018dSDmitry Torokhov
2130995174dSDmitry Torokhov if (gc->pads[i].type != GC_N64)
214c7fd018dSDmitry Torokhov continue;
215c7fd018dSDmitry Torokhov
2160995174dSDmitry Torokhov dev = gc->pads[i].dev;
217c7fd018dSDmitry Torokhov s = gc_status_bit[i];
218c7fd018dSDmitry Torokhov
2190995174dSDmitry Torokhov if (s & ~(data[8] | data[9])) {
220c7fd018dSDmitry Torokhov
221d38fcb96SDmitry Torokhov x = y = 0;
222c7fd018dSDmitry Torokhov
223c7fd018dSDmitry Torokhov for (j = 0; j < 8; j++) {
224c7fd018dSDmitry Torokhov if (data[23 - j] & s)
225d38fcb96SDmitry Torokhov x |= 1 << j;
226c7fd018dSDmitry Torokhov if (data[31 - j] & s)
227d38fcb96SDmitry Torokhov y |= 1 << j;
228c7fd018dSDmitry Torokhov }
229c7fd018dSDmitry Torokhov
230d38fcb96SDmitry Torokhov input_report_abs(dev, ABS_X, x);
231d38fcb96SDmitry Torokhov input_report_abs(dev, ABS_Y, -y);
232c7fd018dSDmitry Torokhov
233d38fcb96SDmitry Torokhov input_report_abs(dev, ABS_HAT0X,
234d38fcb96SDmitry Torokhov !(s & data[6]) - !(s & data[7]));
235d38fcb96SDmitry Torokhov input_report_abs(dev, ABS_HAT0Y,
236d38fcb96SDmitry Torokhov !(s & data[4]) - !(s & data[5]));
237c7fd018dSDmitry Torokhov
238c7fd018dSDmitry Torokhov for (j = 0; j < 10; j++)
239d38fcb96SDmitry Torokhov input_report_key(dev, gc_n64_btn[j],
240d38fcb96SDmitry Torokhov s & data[gc_n64_bytes[j]]);
241c7fd018dSDmitry Torokhov
242c7fd018dSDmitry Torokhov input_sync(dev);
243c7fd018dSDmitry Torokhov }
244c7fd018dSDmitry Torokhov }
245c7fd018dSDmitry Torokhov }
246c7fd018dSDmitry Torokhov
gc_n64_play_effect(struct input_dev * dev,void * data,struct ff_effect * effect)2477aa9e0e8SScott Moreau static int gc_n64_play_effect(struct input_dev *dev, void *data,
2487aa9e0e8SScott Moreau struct ff_effect *effect)
2497aa9e0e8SScott Moreau {
2507aa9e0e8SScott Moreau int i;
2517aa9e0e8SScott Moreau unsigned long flags;
2527aa9e0e8SScott Moreau struct gc *gc = input_get_drvdata(dev);
2537aa9e0e8SScott Moreau struct gc_subdev *sdev = data;
2547aa9e0e8SScott Moreau unsigned char target = 1 << sdev->idx; /* select desired pin */
2557aa9e0e8SScott Moreau
2567aa9e0e8SScott Moreau if (effect->type == FF_RUMBLE) {
2577aa9e0e8SScott Moreau struct ff_rumble_effect *rumble = &effect->u.rumble;
2587aa9e0e8SScott Moreau unsigned int cmd =
2597aa9e0e8SScott Moreau rumble->strong_magnitude || rumble->weak_magnitude ?
2607aa9e0e8SScott Moreau GC_N64_CMD_01 : GC_N64_CMD_00;
2617aa9e0e8SScott Moreau
2627aa9e0e8SScott Moreau local_irq_save(flags);
2637aa9e0e8SScott Moreau
2647aa9e0e8SScott Moreau /* Init Rumble - 0x03, 0x80, 0x01, (34)0x80 */
2657aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_CMD_03, target);
2667aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_CMD_80, target);
2677aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_CMD_01, target);
2687aa9e0e8SScott Moreau for (i = 0; i < 32; i++)
2697aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_CMD_80, target);
2707aa9e0e8SScott Moreau gc_n64_send_stop_bit(gc, target);
2717aa9e0e8SScott Moreau
2727aa9e0e8SScott Moreau udelay(GC_N64_DELAY);
2737aa9e0e8SScott Moreau
2747aa9e0e8SScott Moreau /* Now start or stop it - 0x03, 0xc0, 0zx1b, (32)0x01/0x00 */
2757aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_CMD_03, target);
2767aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_CMD_c0, target);
2777aa9e0e8SScott Moreau gc_n64_send_command(gc, GC_N64_CMD_1b, target);
2787aa9e0e8SScott Moreau for (i = 0; i < 32; i++)
2797aa9e0e8SScott Moreau gc_n64_send_command(gc, cmd, target);
2807aa9e0e8SScott Moreau gc_n64_send_stop_bit(gc, target);
2817aa9e0e8SScott Moreau
2827aa9e0e8SScott Moreau local_irq_restore(flags);
2837aa9e0e8SScott Moreau
2847aa9e0e8SScott Moreau }
2857aa9e0e8SScott Moreau
2867aa9e0e8SScott Moreau return 0;
2877aa9e0e8SScott Moreau }
2887aa9e0e8SScott Moreau
gc_n64_init_ff(struct input_dev * dev,int i)289a517e87cSSudip Mukherjee static int gc_n64_init_ff(struct input_dev *dev, int i)
2907aa9e0e8SScott Moreau {
2917aa9e0e8SScott Moreau struct gc_subdev *sdev;
2927aa9e0e8SScott Moreau int err;
2937aa9e0e8SScott Moreau
2947aa9e0e8SScott Moreau sdev = kmalloc(sizeof(*sdev), GFP_KERNEL);
2957aa9e0e8SScott Moreau if (!sdev)
2967aa9e0e8SScott Moreau return -ENOMEM;
2977aa9e0e8SScott Moreau
2987aa9e0e8SScott Moreau sdev->idx = i;
2997aa9e0e8SScott Moreau
3007aa9e0e8SScott Moreau input_set_capability(dev, EV_FF, FF_RUMBLE);
3017aa9e0e8SScott Moreau
3027aa9e0e8SScott Moreau err = input_ff_create_memless(dev, sdev, gc_n64_play_effect);
3037aa9e0e8SScott Moreau if (err) {
3047aa9e0e8SScott Moreau kfree(sdev);
3057aa9e0e8SScott Moreau return err;
3067aa9e0e8SScott Moreau }
3077aa9e0e8SScott Moreau
3087aa9e0e8SScott Moreau return 0;
3097aa9e0e8SScott Moreau }
3107aa9e0e8SScott Moreau
3111da177e4SLinus Torvalds /*
3121da177e4SLinus Torvalds * NES/SNES support.
3131da177e4SLinus Torvalds */
3141da177e4SLinus Torvalds
3151da177e4SLinus Torvalds #define GC_NES_DELAY 6 /* Delay between bits - 6us */
3161da177e4SLinus Torvalds #define GC_NES_LENGTH 8 /* The NES pads use 8 bits of data */
317b157d55eSRaphael Assenat #define GC_SNES_LENGTH 12 /* The SNES true length is 16, but the
318b157d55eSRaphael Assenat last 4 bits are unused */
319b157d55eSRaphael Assenat #define GC_SNESMOUSE_LENGTH 32 /* The SNES mouse uses 32 bits, the first
320b157d55eSRaphael Assenat 16 bits are equivalent to a gamepad */
3211da177e4SLinus Torvalds
3221da177e4SLinus Torvalds #define GC_NES_POWER 0xfc
3231da177e4SLinus Torvalds #define GC_NES_CLOCK 0x01
3241da177e4SLinus Torvalds #define GC_NES_LATCH 0x02
3251da177e4SLinus Torvalds
326af930d64SDmitry Torokhov static const unsigned char gc_nes_bytes[] = { 0, 1, 2, 3 };
327af930d64SDmitry Torokhov static const unsigned char gc_snes_bytes[] = { 8, 0, 2, 3, 9, 1, 10, 11 };
328af930d64SDmitry Torokhov static const short gc_snes_btn[] = {
329d38fcb96SDmitry Torokhov BTN_A, BTN_B, BTN_SELECT, BTN_START, BTN_X, BTN_Y, BTN_TL, BTN_TR
330d38fcb96SDmitry Torokhov };
3311da177e4SLinus Torvalds
3321da177e4SLinus Torvalds /*
3331da177e4SLinus Torvalds * gc_nes_read_packet() reads a NES/SNES packet.
3341da177e4SLinus Torvalds * Each pad uses one bit per byte. So all pads connected to
3351da177e4SLinus Torvalds * this port are read in parallel.
3361da177e4SLinus Torvalds */
3371da177e4SLinus Torvalds
gc_nes_read_packet(struct gc * gc,int length,unsigned char * data)3381da177e4SLinus Torvalds static void gc_nes_read_packet(struct gc *gc, int length, unsigned char *data)
3391da177e4SLinus Torvalds {
3401da177e4SLinus Torvalds int i;
3411da177e4SLinus Torvalds
3421da177e4SLinus Torvalds parport_write_data(gc->pd->port, GC_NES_POWER | GC_NES_CLOCK | GC_NES_LATCH);
3431da177e4SLinus Torvalds udelay(GC_NES_DELAY * 2);
3441da177e4SLinus Torvalds parport_write_data(gc->pd->port, GC_NES_POWER | GC_NES_CLOCK);
3451da177e4SLinus Torvalds
3461da177e4SLinus Torvalds for (i = 0; i < length; i++) {
3471da177e4SLinus Torvalds udelay(GC_NES_DELAY);
3481da177e4SLinus Torvalds parport_write_data(gc->pd->port, GC_NES_POWER);
3491da177e4SLinus Torvalds data[i] = parport_read_status(gc->pd->port) ^ 0x7f;
3501da177e4SLinus Torvalds udelay(GC_NES_DELAY);
3511da177e4SLinus Torvalds parport_write_data(gc->pd->port, GC_NES_POWER | GC_NES_CLOCK);
3521da177e4SLinus Torvalds }
3531da177e4SLinus Torvalds }
3541da177e4SLinus Torvalds
gc_nes_process_packet(struct gc * gc)355c7fd018dSDmitry Torokhov static void gc_nes_process_packet(struct gc *gc)
356c7fd018dSDmitry Torokhov {
357b157d55eSRaphael Assenat unsigned char data[GC_SNESMOUSE_LENGTH];
3580995174dSDmitry Torokhov struct gc_pad *pad;
359c7fd018dSDmitry Torokhov struct input_dev *dev;
360b157d55eSRaphael Assenat int i, j, s, len;
361b157d55eSRaphael Assenat char x_rel, y_rel;
362c7fd018dSDmitry Torokhov
3630995174dSDmitry Torokhov len = gc->pad_count[GC_SNESMOUSE] ? GC_SNESMOUSE_LENGTH :
3640995174dSDmitry Torokhov (gc->pad_count[GC_SNES] ? GC_SNES_LENGTH : GC_NES_LENGTH);
365b157d55eSRaphael Assenat
366b157d55eSRaphael Assenat gc_nes_read_packet(gc, len, data);
367c7fd018dSDmitry Torokhov
368c7fd018dSDmitry Torokhov for (i = 0; i < GC_MAX_DEVICES; i++) {
369c7fd018dSDmitry Torokhov
3700995174dSDmitry Torokhov pad = &gc->pads[i];
3717b5d3312SDmitry Torokhov dev = pad->dev;
372c7fd018dSDmitry Torokhov s = gc_status_bit[i];
373c7fd018dSDmitry Torokhov
3740995174dSDmitry Torokhov switch (pad->type) {
3750995174dSDmitry Torokhov
3760995174dSDmitry Torokhov case GC_NES:
3770995174dSDmitry Torokhov
378c7fd018dSDmitry Torokhov input_report_abs(dev, ABS_X, !(s & data[6]) - !(s & data[7]));
379c7fd018dSDmitry Torokhov input_report_abs(dev, ABS_Y, !(s & data[4]) - !(s & data[5]));
380c7fd018dSDmitry Torokhov
381c7fd018dSDmitry Torokhov for (j = 0; j < 4; j++)
382d38fcb96SDmitry Torokhov input_report_key(dev, gc_snes_btn[j],
383d38fcb96SDmitry Torokhov s & data[gc_nes_bytes[j]]);
3840995174dSDmitry Torokhov input_sync(dev);
3850995174dSDmitry Torokhov break;
386c7fd018dSDmitry Torokhov
3870995174dSDmitry Torokhov case GC_SNES:
3880995174dSDmitry Torokhov
3890995174dSDmitry Torokhov input_report_abs(dev, ABS_X, !(s & data[6]) - !(s & data[7]));
3900995174dSDmitry Torokhov input_report_abs(dev, ABS_Y, !(s & data[4]) - !(s & data[5]));
3910995174dSDmitry Torokhov
392c7fd018dSDmitry Torokhov for (j = 0; j < 8; j++)
393d38fcb96SDmitry Torokhov input_report_key(dev, gc_snes_btn[j],
394d38fcb96SDmitry Torokhov s & data[gc_snes_bytes[j]]);
3950995174dSDmitry Torokhov input_sync(dev);
3960995174dSDmitry Torokhov break;
397c7fd018dSDmitry Torokhov
3980995174dSDmitry Torokhov case GC_SNESMOUSE:
399b157d55eSRaphael Assenat /*
400d38fcb96SDmitry Torokhov * The 4 unused bits from SNES controllers appear
401d38fcb96SDmitry Torokhov * to be ID bits so use them to make sure we are
402d38fcb96SDmitry Torokhov * dealing with a mouse.
403b157d55eSRaphael Assenat * gamepad is connected. This is important since
404b157d55eSRaphael Assenat * my SNES gamepad sends 1's for bits 16-31, which
405b157d55eSRaphael Assenat * cause the mouse pointer to quickly move to the
406b157d55eSRaphael Assenat * upper left corner of the screen.
407b157d55eSRaphael Assenat */
408b157d55eSRaphael Assenat if (!(s & data[12]) && !(s & data[13]) &&
409b157d55eSRaphael Assenat !(s & data[14]) && (s & data[15])) {
410b157d55eSRaphael Assenat input_report_key(dev, BTN_LEFT, s & data[9]);
411b157d55eSRaphael Assenat input_report_key(dev, BTN_RIGHT, s & data[8]);
412b157d55eSRaphael Assenat
413b157d55eSRaphael Assenat x_rel = y_rel = 0;
414b157d55eSRaphael Assenat for (j = 0; j < 7; j++) {
415b157d55eSRaphael Assenat x_rel <<= 1;
416b157d55eSRaphael Assenat if (data[25 + j] & s)
417b157d55eSRaphael Assenat x_rel |= 1;
418b157d55eSRaphael Assenat
419b157d55eSRaphael Assenat y_rel <<= 1;
420b157d55eSRaphael Assenat if (data[17 + j] & s)
421b157d55eSRaphael Assenat y_rel |= 1;
422b157d55eSRaphael Assenat }
423b157d55eSRaphael Assenat
424b157d55eSRaphael Assenat if (x_rel) {
425b157d55eSRaphael Assenat if (data[24] & s)
426b157d55eSRaphael Assenat x_rel = -x_rel;
427b157d55eSRaphael Assenat input_report_rel(dev, REL_X, x_rel);
428b157d55eSRaphael Assenat }
429b157d55eSRaphael Assenat
430b157d55eSRaphael Assenat if (y_rel) {
431b157d55eSRaphael Assenat if (data[16] & s)
432b157d55eSRaphael Assenat y_rel = -y_rel;
433b157d55eSRaphael Assenat input_report_rel(dev, REL_Y, y_rel);
434b157d55eSRaphael Assenat }
4350995174dSDmitry Torokhov
436c7fd018dSDmitry Torokhov input_sync(dev);
437c7fd018dSDmitry Torokhov }
4380995174dSDmitry Torokhov break;
4390995174dSDmitry Torokhov
4400995174dSDmitry Torokhov default:
4410995174dSDmitry Torokhov break;
4420995174dSDmitry Torokhov }
4430995174dSDmitry Torokhov }
444c7fd018dSDmitry Torokhov }
445c7fd018dSDmitry Torokhov
4461da177e4SLinus Torvalds /*
4471da177e4SLinus Torvalds * Multisystem joystick support
4481da177e4SLinus Torvalds */
4491da177e4SLinus Torvalds
4501da177e4SLinus Torvalds #define GC_MULTI_LENGTH 5 /* Multi system joystick packet length is 5 */
4511da177e4SLinus Torvalds #define GC_MULTI2_LENGTH 6 /* One more bit for one more button */
4521da177e4SLinus Torvalds
4531da177e4SLinus Torvalds /*
4541da177e4SLinus Torvalds * gc_multi_read_packet() reads a Multisystem joystick packet.
4551da177e4SLinus Torvalds */
4561da177e4SLinus Torvalds
gc_multi_read_packet(struct gc * gc,int length,unsigned char * data)4571da177e4SLinus Torvalds static void gc_multi_read_packet(struct gc *gc, int length, unsigned char *data)
4581da177e4SLinus Torvalds {
4591da177e4SLinus Torvalds int i;
4601da177e4SLinus Torvalds
4611da177e4SLinus Torvalds for (i = 0; i < length; i++) {
4621da177e4SLinus Torvalds parport_write_data(gc->pd->port, ~(1 << i));
4631da177e4SLinus Torvalds data[i] = parport_read_status(gc->pd->port) ^ 0x7f;
4641da177e4SLinus Torvalds }
4651da177e4SLinus Torvalds }
4661da177e4SLinus Torvalds
gc_multi_process_packet(struct gc * gc)467c7fd018dSDmitry Torokhov static void gc_multi_process_packet(struct gc *gc)
468c7fd018dSDmitry Torokhov {
469c7fd018dSDmitry Torokhov unsigned char data[GC_MULTI2_LENGTH];
4700995174dSDmitry Torokhov int data_len = gc->pad_count[GC_MULTI2] ? GC_MULTI2_LENGTH : GC_MULTI_LENGTH;
4710995174dSDmitry Torokhov struct gc_pad *pad;
472c7fd018dSDmitry Torokhov struct input_dev *dev;
473c7fd018dSDmitry Torokhov int i, s;
474c7fd018dSDmitry Torokhov
475d38fcb96SDmitry Torokhov gc_multi_read_packet(gc, data_len, data);
476c7fd018dSDmitry Torokhov
477c7fd018dSDmitry Torokhov for (i = 0; i < GC_MAX_DEVICES; i++) {
4780995174dSDmitry Torokhov pad = &gc->pads[i];
4790995174dSDmitry Torokhov dev = pad->dev;
480c7fd018dSDmitry Torokhov s = gc_status_bit[i];
481c7fd018dSDmitry Torokhov
4820995174dSDmitry Torokhov switch (pad->type) {
4830995174dSDmitry Torokhov case GC_MULTI2:
4840995174dSDmitry Torokhov input_report_key(dev, BTN_THUMB, s & data[5]);
485*6f49c4f5SGustavo A. R. Silva fallthrough;
4860995174dSDmitry Torokhov
4870995174dSDmitry Torokhov case GC_MULTI:
488d38fcb96SDmitry Torokhov input_report_abs(dev, ABS_X,
489d38fcb96SDmitry Torokhov !(s & data[2]) - !(s & data[3]));
490d38fcb96SDmitry Torokhov input_report_abs(dev, ABS_Y,
491d38fcb96SDmitry Torokhov !(s & data[0]) - !(s & data[1]));
492c7fd018dSDmitry Torokhov input_report_key(dev, BTN_TRIGGER, s & data[4]);
493c7fd018dSDmitry Torokhov input_sync(dev);
4940995174dSDmitry Torokhov break;
4950995174dSDmitry Torokhov
4960995174dSDmitry Torokhov default:
4970995174dSDmitry Torokhov break;
4980995174dSDmitry Torokhov }
499c7fd018dSDmitry Torokhov }
500c7fd018dSDmitry Torokhov }
501c7fd018dSDmitry Torokhov
5021da177e4SLinus Torvalds /*
5031da177e4SLinus Torvalds * PSX support
5041da177e4SLinus Torvalds *
5051da177e4SLinus Torvalds * See documentation at:
506631dd1a8SJustin P. Mattock * http://www.geocities.co.jp/Playtown/2004/psx/ps_eng.txt
5071da177e4SLinus Torvalds * http://www.gamesx.com/controldata/psxcont/psxcont.htm
5081da177e4SLinus Torvalds *
5091da177e4SLinus Torvalds */
5101da177e4SLinus Torvalds
5111da177e4SLinus Torvalds #define GC_PSX_DELAY 25 /* 25 usec */
5121da177e4SLinus Torvalds #define GC_PSX_LENGTH 8 /* talk to the controller in bits */
5131da177e4SLinus Torvalds #define GC_PSX_BYTES 6 /* the maximum number of bytes to read off the controller */
5141da177e4SLinus Torvalds
5151da177e4SLinus Torvalds #define GC_PSX_MOUSE 1 /* Mouse */
5161da177e4SLinus Torvalds #define GC_PSX_NEGCON 2 /* NegCon */
5171da177e4SLinus Torvalds #define GC_PSX_NORMAL 4 /* Digital / Analog or Rumble in Digital mode */
5181da177e4SLinus Torvalds #define GC_PSX_ANALOG 5 /* Analog in Analog mode / Rumble in Green mode */
5191da177e4SLinus Torvalds #define GC_PSX_RUMBLE 7 /* Rumble in Red mode */
5201da177e4SLinus Torvalds
5211da177e4SLinus Torvalds #define GC_PSX_CLOCK 0x04 /* Pin 4 */
5221da177e4SLinus Torvalds #define GC_PSX_COMMAND 0x01 /* Pin 2 */
5231da177e4SLinus Torvalds #define GC_PSX_POWER 0xf8 /* Pins 5-9 */
5241da177e4SLinus Torvalds #define GC_PSX_SELECT 0x02 /* Pin 3 */
5251da177e4SLinus Torvalds
5261da177e4SLinus Torvalds #define GC_PSX_ID(x) ((x) >> 4) /* High nibble is device type */
5271da177e4SLinus Torvalds #define GC_PSX_LEN(x) (((x) & 0xf) << 1) /* Low nibble is length in bytes/2 */
5281da177e4SLinus Torvalds
5291da177e4SLinus Torvalds static int gc_psx_delay = GC_PSX_DELAY;
5301da177e4SLinus Torvalds module_param_named(psx_delay, gc_psx_delay, uint, 0);
5311da177e4SLinus Torvalds MODULE_PARM_DESC(psx_delay, "Delay when accessing Sony PSX controller (usecs)");
5321da177e4SLinus Torvalds
533af930d64SDmitry Torokhov static const short gc_psx_abs[] = {
534d38fcb96SDmitry Torokhov ABS_X, ABS_Y, ABS_RX, ABS_RY, ABS_HAT0X, ABS_HAT0Y
535d38fcb96SDmitry Torokhov };
536af930d64SDmitry Torokhov static const short gc_psx_btn[] = {
537d38fcb96SDmitry Torokhov BTN_TL, BTN_TR, BTN_TL2, BTN_TR2, BTN_A, BTN_B, BTN_X, BTN_Y,
538d38fcb96SDmitry Torokhov BTN_START, BTN_SELECT, BTN_THUMBL, BTN_THUMBR
539d38fcb96SDmitry Torokhov };
540af930d64SDmitry Torokhov static const short gc_psx_ddr_btn[] = { BTN_0, BTN_1, BTN_2, BTN_3 };
5411da177e4SLinus Torvalds
5421da177e4SLinus Torvalds /*
5431da177e4SLinus Torvalds * gc_psx_command() writes 8bit command and reads 8bit data from
5441da177e4SLinus Torvalds * the psx pad.
5451da177e4SLinus Torvalds */
5461da177e4SLinus Torvalds
gc_psx_command(struct gc * gc,int b,unsigned char * data)547d38fcb96SDmitry Torokhov static void gc_psx_command(struct gc *gc, int b, unsigned char *data)
5481da177e4SLinus Torvalds {
549d38fcb96SDmitry Torokhov struct parport *port = gc->pd->port;
5501da177e4SLinus Torvalds int i, j, cmd, read;
551c7fd018dSDmitry Torokhov
552d38fcb96SDmitry Torokhov memset(data, 0, GC_MAX_DEVICES);
5531da177e4SLinus Torvalds
5541da177e4SLinus Torvalds for (i = 0; i < GC_PSX_LENGTH; i++, b >>= 1) {
5551da177e4SLinus Torvalds cmd = (b & 1) ? GC_PSX_COMMAND : 0;
556d38fcb96SDmitry Torokhov parport_write_data(port, cmd | GC_PSX_POWER);
5571da177e4SLinus Torvalds udelay(gc_psx_delay);
5580995174dSDmitry Torokhov
559d38fcb96SDmitry Torokhov read = parport_read_status(port) ^ 0x80;
5600995174dSDmitry Torokhov
5610995174dSDmitry Torokhov for (j = 0; j < GC_MAX_DEVICES; j++) {
562c25f7b76SDmitry Torokhov struct gc_pad *pad = &gc->pads[j];
5630995174dSDmitry Torokhov
5640995174dSDmitry Torokhov if (pad->type == GC_PSX || pad->type == GC_DDR)
5650995174dSDmitry Torokhov data[j] |= (read & gc_status_bit[j]) ? (1 << i) : 0;
5660995174dSDmitry Torokhov }
5670995174dSDmitry Torokhov
5681da177e4SLinus Torvalds parport_write_data(gc->pd->port, cmd | GC_PSX_CLOCK | GC_PSX_POWER);
5691da177e4SLinus Torvalds udelay(gc_psx_delay);
5701da177e4SLinus Torvalds }
5711da177e4SLinus Torvalds }
5721da177e4SLinus Torvalds
5731da177e4SLinus Torvalds /*
5741da177e4SLinus Torvalds * gc_psx_read_packet() reads a whole psx packet and returns
5751da177e4SLinus Torvalds * device identifier code.
5761da177e4SLinus Torvalds */
5771da177e4SLinus Torvalds
gc_psx_read_packet(struct gc * gc,unsigned char data[GC_MAX_DEVICES][GC_PSX_BYTES],unsigned char id[GC_MAX_DEVICES])5780995174dSDmitry Torokhov static void gc_psx_read_packet(struct gc *gc,
5790995174dSDmitry Torokhov unsigned char data[GC_MAX_DEVICES][GC_PSX_BYTES],
580c7fd018dSDmitry Torokhov unsigned char id[GC_MAX_DEVICES])
5811da177e4SLinus Torvalds {
5821da177e4SLinus Torvalds int i, j, max_len = 0;
5831da177e4SLinus Torvalds unsigned long flags;
584c7fd018dSDmitry Torokhov unsigned char data2[GC_MAX_DEVICES];
5851da177e4SLinus Torvalds
586d38fcb96SDmitry Torokhov /* Select pad */
587d38fcb96SDmitry Torokhov parport_write_data(gc->pd->port, GC_PSX_CLOCK | GC_PSX_SELECT | GC_PSX_POWER);
5881da177e4SLinus Torvalds udelay(gc_psx_delay);
589d38fcb96SDmitry Torokhov /* Deselect, begin command */
590d38fcb96SDmitry Torokhov parport_write_data(gc->pd->port, GC_PSX_CLOCK | GC_PSX_POWER);
5911da177e4SLinus Torvalds udelay(gc_psx_delay);
5921da177e4SLinus Torvalds
5931da177e4SLinus Torvalds local_irq_save(flags);
5941da177e4SLinus Torvalds
5951da177e4SLinus Torvalds gc_psx_command(gc, 0x01, data2); /* Access pad */
5961da177e4SLinus Torvalds gc_psx_command(gc, 0x42, id); /* Get device ids */
5971da177e4SLinus Torvalds gc_psx_command(gc, 0, data2); /* Dump status */
5981da177e4SLinus Torvalds
599d38fcb96SDmitry Torokhov /* Find the longest pad */
6000995174dSDmitry Torokhov for (i = 0; i < GC_MAX_DEVICES; i++) {
6010995174dSDmitry Torokhov struct gc_pad *pad = &gc->pads[i];
6020995174dSDmitry Torokhov
6030995174dSDmitry Torokhov if ((pad->type == GC_PSX || pad->type == GC_DDR) &&
604d38fcb96SDmitry Torokhov GC_PSX_LEN(id[i]) > max_len &&
605d38fcb96SDmitry Torokhov GC_PSX_LEN(id[i]) <= GC_PSX_BYTES) {
6061da177e4SLinus Torvalds max_len = GC_PSX_LEN(id[i]);
607d38fcb96SDmitry Torokhov }
6080995174dSDmitry Torokhov }
6091da177e4SLinus Torvalds
610d38fcb96SDmitry Torokhov /* Read in all the data */
611d38fcb96SDmitry Torokhov for (i = 0; i < max_len; i++) {
6121da177e4SLinus Torvalds gc_psx_command(gc, 0, data2);
613c7fd018dSDmitry Torokhov for (j = 0; j < GC_MAX_DEVICES; j++)
6141da177e4SLinus Torvalds data[j][i] = data2[j];
6151da177e4SLinus Torvalds }
6161da177e4SLinus Torvalds
6171da177e4SLinus Torvalds local_irq_restore(flags);
6181da177e4SLinus Torvalds
6191da177e4SLinus Torvalds parport_write_data(gc->pd->port, GC_PSX_CLOCK | GC_PSX_SELECT | GC_PSX_POWER);
6201da177e4SLinus Torvalds
621d38fcb96SDmitry Torokhov /* Set id's to the real value */
622d38fcb96SDmitry Torokhov for (i = 0; i < GC_MAX_DEVICES; i++)
6231da177e4SLinus Torvalds id[i] = GC_PSX_ID(id[i]);
6241da177e4SLinus Torvalds }
6251da177e4SLinus Torvalds
gc_psx_report_one(struct gc_pad * pad,unsigned char psx_type,unsigned char * data)6260995174dSDmitry Torokhov static void gc_psx_report_one(struct gc_pad *pad, unsigned char psx_type,
627d38fcb96SDmitry Torokhov unsigned char *data)
6281da177e4SLinus Torvalds {
6290995174dSDmitry Torokhov struct input_dev *dev = pad->dev;
630d38fcb96SDmitry Torokhov int i;
6311da177e4SLinus Torvalds
6320995174dSDmitry Torokhov switch (psx_type) {
6331da177e4SLinus Torvalds
6341da177e4SLinus Torvalds case GC_PSX_RUMBLE:
6351da177e4SLinus Torvalds
636d38fcb96SDmitry Torokhov input_report_key(dev, BTN_THUMBL, ~data[0] & 0x04);
637d38fcb96SDmitry Torokhov input_report_key(dev, BTN_THUMBR, ~data[0] & 0x02);
638*6f49c4f5SGustavo A. R. Silva fallthrough;
6391da177e4SLinus Torvalds
6401da177e4SLinus Torvalds case GC_PSX_NEGCON:
6411da177e4SLinus Torvalds case GC_PSX_ANALOG:
6421da177e4SLinus Torvalds
6430995174dSDmitry Torokhov if (pad->type == GC_DDR) {
644d38fcb96SDmitry Torokhov for (i = 0; i < 4; i++)
645d38fcb96SDmitry Torokhov input_report_key(dev, gc_psx_ddr_btn[i],
646d38fcb96SDmitry Torokhov ~data[0] & (0x10 << i));
6471da177e4SLinus Torvalds } else {
648d38fcb96SDmitry Torokhov for (i = 0; i < 4; i++)
649d38fcb96SDmitry Torokhov input_report_abs(dev, gc_psx_abs[i + 2],
650d38fcb96SDmitry Torokhov data[i + 2]);
6511da177e4SLinus Torvalds
652315543fdSDmitry Torokhov input_report_abs(dev, ABS_X,
653315543fdSDmitry Torokhov !!(data[0] & 0x80) * 128 + !(data[0] & 0x20) * 127);
654315543fdSDmitry Torokhov input_report_abs(dev, ABS_Y,
655315543fdSDmitry Torokhov !!(data[0] & 0x10) * 128 + !(data[0] & 0x40) * 127);
6561da177e4SLinus Torvalds }
6571da177e4SLinus Torvalds
658d38fcb96SDmitry Torokhov for (i = 0; i < 8; i++)
659d38fcb96SDmitry Torokhov input_report_key(dev, gc_psx_btn[i], ~data[1] & (1 << i));
6601da177e4SLinus Torvalds
661d38fcb96SDmitry Torokhov input_report_key(dev, BTN_START, ~data[0] & 0x08);
662d38fcb96SDmitry Torokhov input_report_key(dev, BTN_SELECT, ~data[0] & 0x01);
6631da177e4SLinus Torvalds
664c7fd018dSDmitry Torokhov input_sync(dev);
6651da177e4SLinus Torvalds
6661da177e4SLinus Torvalds break;
6671da177e4SLinus Torvalds
6681da177e4SLinus Torvalds case GC_PSX_NORMAL:
6690995174dSDmitry Torokhov
6700995174dSDmitry Torokhov if (pad->type == GC_DDR) {
671d38fcb96SDmitry Torokhov for (i = 0; i < 4; i++)
672d38fcb96SDmitry Torokhov input_report_key(dev, gc_psx_ddr_btn[i],
673d38fcb96SDmitry Torokhov ~data[0] & (0x10 << i));
6741da177e4SLinus Torvalds } else {
675315543fdSDmitry Torokhov input_report_abs(dev, ABS_X,
676315543fdSDmitry Torokhov !!(data[0] & 0x80) * 128 + !(data[0] & 0x20) * 127);
677315543fdSDmitry Torokhov input_report_abs(dev, ABS_Y,
678315543fdSDmitry Torokhov !!(data[0] & 0x10) * 128 + !(data[0] & 0x40) * 127);
6791da177e4SLinus Torvalds
680d38fcb96SDmitry Torokhov /*
681d38fcb96SDmitry Torokhov * For some reason if the extra axes are left unset
682d38fcb96SDmitry Torokhov * they drift.
683d38fcb96SDmitry Torokhov * for (i = 0; i < 4; i++)
684d38fcb96SDmitry Torokhov input_report_abs(dev, gc_psx_abs[i + 2], 128);
6851da177e4SLinus Torvalds * This needs to be debugged properly,
686d38fcb96SDmitry Torokhov * maybe fuzz processing needs to be done
687d38fcb96SDmitry Torokhov * in input_sync()
6881da177e4SLinus Torvalds * --vojtech
6891da177e4SLinus Torvalds */
6901da177e4SLinus Torvalds }
6911da177e4SLinus Torvalds
692d38fcb96SDmitry Torokhov for (i = 0; i < 8; i++)
693d38fcb96SDmitry Torokhov input_report_key(dev, gc_psx_btn[i], ~data[1] & (1 << i));
6941da177e4SLinus Torvalds
695d38fcb96SDmitry Torokhov input_report_key(dev, BTN_START, ~data[0] & 0x08);
696d38fcb96SDmitry Torokhov input_report_key(dev, BTN_SELECT, ~data[0] & 0x01);
6971da177e4SLinus Torvalds
698c7fd018dSDmitry Torokhov input_sync(dev);
6991da177e4SLinus Torvalds
7001da177e4SLinus Torvalds break;
7011da177e4SLinus Torvalds
7020995174dSDmitry Torokhov default: /* not a pad, ignore */
7031da177e4SLinus Torvalds break;
7041da177e4SLinus Torvalds }
7051da177e4SLinus Torvalds }
706d38fcb96SDmitry Torokhov
gc_psx_process_packet(struct gc * gc)707d38fcb96SDmitry Torokhov static void gc_psx_process_packet(struct gc *gc)
708d38fcb96SDmitry Torokhov {
709d38fcb96SDmitry Torokhov unsigned char data[GC_MAX_DEVICES][GC_PSX_BYTES];
710d38fcb96SDmitry Torokhov unsigned char id[GC_MAX_DEVICES];
7110995174dSDmitry Torokhov struct gc_pad *pad;
712d38fcb96SDmitry Torokhov int i;
713d38fcb96SDmitry Torokhov
714d38fcb96SDmitry Torokhov gc_psx_read_packet(gc, data, id);
715d38fcb96SDmitry Torokhov
716d38fcb96SDmitry Torokhov for (i = 0; i < GC_MAX_DEVICES; i++) {
7170995174dSDmitry Torokhov pad = &gc->pads[i];
7180995174dSDmitry Torokhov if (pad->type == GC_PSX || pad->type == GC_DDR)
7190995174dSDmitry Torokhov gc_psx_report_one(pad, id[i], data[i]);
720d38fcb96SDmitry Torokhov }
7211da177e4SLinus Torvalds }
7221da177e4SLinus Torvalds
723c7fd018dSDmitry Torokhov /*
724c7fd018dSDmitry Torokhov * gc_timer() initiates reads of console pads data.
725c7fd018dSDmitry Torokhov */
726c7fd018dSDmitry Torokhov
gc_timer(struct timer_list * t)727e99e88a9SKees Cook static void gc_timer(struct timer_list *t)
728c7fd018dSDmitry Torokhov {
729e99e88a9SKees Cook struct gc *gc = from_timer(gc, t, timer);
730c7fd018dSDmitry Torokhov
731c7fd018dSDmitry Torokhov /*
732c7fd018dSDmitry Torokhov * N64 pads - must be read first, any read confuses them for 200 us
733c7fd018dSDmitry Torokhov */
734c7fd018dSDmitry Torokhov
7350995174dSDmitry Torokhov if (gc->pad_count[GC_N64])
736c7fd018dSDmitry Torokhov gc_n64_process_packet(gc);
737c7fd018dSDmitry Torokhov
738c7fd018dSDmitry Torokhov /*
739b157d55eSRaphael Assenat * NES and SNES pads or mouse
740c7fd018dSDmitry Torokhov */
741c7fd018dSDmitry Torokhov
7420995174dSDmitry Torokhov if (gc->pad_count[GC_NES] ||
7430995174dSDmitry Torokhov gc->pad_count[GC_SNES] ||
7440995174dSDmitry Torokhov gc->pad_count[GC_SNESMOUSE]) {
745c7fd018dSDmitry Torokhov gc_nes_process_packet(gc);
7460995174dSDmitry Torokhov }
747c7fd018dSDmitry Torokhov
748c7fd018dSDmitry Torokhov /*
749c7fd018dSDmitry Torokhov * Multi and Multi2 joysticks
750c7fd018dSDmitry Torokhov */
751c7fd018dSDmitry Torokhov
7520995174dSDmitry Torokhov if (gc->pad_count[GC_MULTI] || gc->pad_count[GC_MULTI2])
753c7fd018dSDmitry Torokhov gc_multi_process_packet(gc);
754c7fd018dSDmitry Torokhov
755c7fd018dSDmitry Torokhov /*
756c7fd018dSDmitry Torokhov * PSX controllers
757c7fd018dSDmitry Torokhov */
758c7fd018dSDmitry Torokhov
7590995174dSDmitry Torokhov if (gc->pad_count[GC_PSX] || gc->pad_count[GC_DDR])
760c7fd018dSDmitry Torokhov gc_psx_process_packet(gc);
761c7fd018dSDmitry Torokhov
7621da177e4SLinus Torvalds mod_timer(&gc->timer, jiffies + GC_REFRESH_TIME);
7631da177e4SLinus Torvalds }
7641da177e4SLinus Torvalds
gc_open(struct input_dev * dev)7651da177e4SLinus Torvalds static int gc_open(struct input_dev *dev)
7661da177e4SLinus Torvalds {
7678715c1cfSDmitry Torokhov struct gc *gc = input_get_drvdata(dev);
7688b1a198bSDmitry Torokhov int err;
7698b1a198bSDmitry Torokhov
77072ba9f0cSIngo Molnar err = mutex_lock_interruptible(&gc->mutex);
7718b1a198bSDmitry Torokhov if (err)
7728b1a198bSDmitry Torokhov return err;
7738b1a198bSDmitry Torokhov
7741da177e4SLinus Torvalds if (!gc->used++) {
7751da177e4SLinus Torvalds parport_claim(gc->pd);
7761da177e4SLinus Torvalds parport_write_control(gc->pd->port, 0x04);
7771da177e4SLinus Torvalds mod_timer(&gc->timer, jiffies + GC_REFRESH_TIME);
7781da177e4SLinus Torvalds }
7798b1a198bSDmitry Torokhov
78072ba9f0cSIngo Molnar mutex_unlock(&gc->mutex);
7811da177e4SLinus Torvalds return 0;
7821da177e4SLinus Torvalds }
7831da177e4SLinus Torvalds
gc_close(struct input_dev * dev)7841da177e4SLinus Torvalds static void gc_close(struct input_dev *dev)
7851da177e4SLinus Torvalds {
7868715c1cfSDmitry Torokhov struct gc *gc = input_get_drvdata(dev);
7878b1a198bSDmitry Torokhov
78872ba9f0cSIngo Molnar mutex_lock(&gc->mutex);
7891da177e4SLinus Torvalds if (!--gc->used) {
7908b1a198bSDmitry Torokhov del_timer_sync(&gc->timer);
7911da177e4SLinus Torvalds parport_write_control(gc->pd->port, 0x00);
7921da177e4SLinus Torvalds parport_release(gc->pd);
7931da177e4SLinus Torvalds }
79472ba9f0cSIngo Molnar mutex_unlock(&gc->mutex);
7951da177e4SLinus Torvalds }
7961da177e4SLinus Torvalds
gc_setup_pad(struct gc * gc,int idx,int pad_type)797a517e87cSSudip Mukherjee static int gc_setup_pad(struct gc *gc, int idx, int pad_type)
7981da177e4SLinus Torvalds {
7990995174dSDmitry Torokhov struct gc_pad *pad = &gc->pads[idx];
80017dd3f0fSDmitry Torokhov struct input_dev *input_dev;
80117dd3f0fSDmitry Torokhov int i;
8027aa9e0e8SScott Moreau int err;
8031da177e4SLinus Torvalds
8045bc923c5SDan Carpenter if (pad_type < 1 || pad_type >= GC_MAX) {
805a1e12747SDmitry Torokhov pr_err("Pad type %d unknown\n", pad_type);
80617dd3f0fSDmitry Torokhov return -EINVAL;
8071da177e4SLinus Torvalds }
8081da177e4SLinus Torvalds
8090995174dSDmitry Torokhov pad->dev = input_dev = input_allocate_device();
81017dd3f0fSDmitry Torokhov if (!input_dev) {
811a1e12747SDmitry Torokhov pr_err("Not enough memory for input device\n");
81217dd3f0fSDmitry Torokhov return -ENOMEM;
8131da177e4SLinus Torvalds }
8141da177e4SLinus Torvalds
8150995174dSDmitry Torokhov pad->type = pad_type;
8160995174dSDmitry Torokhov
8170995174dSDmitry Torokhov snprintf(pad->phys, sizeof(pad->phys),
8180995174dSDmitry Torokhov "%s/input%d", gc->pd->port->name, idx);
8190995174dSDmitry Torokhov
82017dd3f0fSDmitry Torokhov input_dev->name = gc_names[pad_type];
8210995174dSDmitry Torokhov input_dev->phys = pad->phys;
82217dd3f0fSDmitry Torokhov input_dev->id.bustype = BUS_PARPORT;
82317dd3f0fSDmitry Torokhov input_dev->id.vendor = 0x0001;
82417dd3f0fSDmitry Torokhov input_dev->id.product = pad_type;
82517dd3f0fSDmitry Torokhov input_dev->id.version = 0x0100;
8268715c1cfSDmitry Torokhov
8278715c1cfSDmitry Torokhov input_set_drvdata(input_dev, gc);
8288b1a198bSDmitry Torokhov
82917dd3f0fSDmitry Torokhov input_dev->open = gc_open;
83017dd3f0fSDmitry Torokhov input_dev->close = gc_close;
8311da177e4SLinus Torvalds
832b157d55eSRaphael Assenat if (pad_type != GC_SNESMOUSE) {
8337b19ada2SJiri Slaby input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS);
8341da177e4SLinus Torvalds
83517dd3f0fSDmitry Torokhov for (i = 0; i < 2; i++)
83617dd3f0fSDmitry Torokhov input_set_abs_params(input_dev, ABS_X + i, -1, 1, 0, 0);
837b157d55eSRaphael Assenat } else
8387b19ada2SJiri Slaby input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REL);
8391da177e4SLinus Torvalds
8400995174dSDmitry Torokhov gc->pad_count[pad_type]++;
8411da177e4SLinus Torvalds
84217dd3f0fSDmitry Torokhov switch (pad_type) {
8431da177e4SLinus Torvalds
8441da177e4SLinus Torvalds case GC_N64:
84517dd3f0fSDmitry Torokhov for (i = 0; i < 10; i++)
846a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, gc_n64_btn[i]);
8471da177e4SLinus Torvalds
84817dd3f0fSDmitry Torokhov for (i = 0; i < 2; i++) {
84917dd3f0fSDmitry Torokhov input_set_abs_params(input_dev, ABS_X + i, -127, 126, 0, 2);
85017dd3f0fSDmitry Torokhov input_set_abs_params(input_dev, ABS_HAT0X + i, -1, 1, 0, 0);
8511da177e4SLinus Torvalds }
8521da177e4SLinus Torvalds
8537aa9e0e8SScott Moreau err = gc_n64_init_ff(input_dev, idx);
8547aa9e0e8SScott Moreau if (err) {
855fef5f569SJoe Perches pr_warn("Failed to initiate rumble for N64 device %d\n",
856fef5f569SJoe Perches idx);
8570995174dSDmitry Torokhov goto err_free_dev;
8587aa9e0e8SScott Moreau }
8597aa9e0e8SScott Moreau
8601da177e4SLinus Torvalds break;
8611da177e4SLinus Torvalds
862b157d55eSRaphael Assenat case GC_SNESMOUSE:
863a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, BTN_LEFT);
864a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, BTN_RIGHT);
865a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_REL, REL_X);
866a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_REL, REL_Y);
867b157d55eSRaphael Assenat break;
868b157d55eSRaphael Assenat
8691da177e4SLinus Torvalds case GC_SNES:
87017dd3f0fSDmitry Torokhov for (i = 4; i < 8; i++)
871a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, gc_snes_btn[i]);
872*6f49c4f5SGustavo A. R. Silva fallthrough;
873*6f49c4f5SGustavo A. R. Silva
8741da177e4SLinus Torvalds case GC_NES:
87517dd3f0fSDmitry Torokhov for (i = 0; i < 4; i++)
876a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, gc_snes_btn[i]);
8771da177e4SLinus Torvalds break;
8781da177e4SLinus Torvalds
8791da177e4SLinus Torvalds case GC_MULTI2:
880a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, BTN_THUMB);
881*6f49c4f5SGustavo A. R. Silva fallthrough;
882*6f49c4f5SGustavo A. R. Silva
8831da177e4SLinus Torvalds case GC_MULTI:
884a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, BTN_TRIGGER);
8851da177e4SLinus Torvalds break;
8861da177e4SLinus Torvalds
8871da177e4SLinus Torvalds case GC_PSX:
88817dd3f0fSDmitry Torokhov for (i = 0; i < 6; i++)
889d38fcb96SDmitry Torokhov input_set_abs_params(input_dev,
890d38fcb96SDmitry Torokhov gc_psx_abs[i], 4, 252, 0, 2);
89117dd3f0fSDmitry Torokhov for (i = 0; i < 12; i++)
892a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, gc_psx_btn[i]);
893a5a45b7fSMarcus Folkesson break;
8941da177e4SLinus Torvalds
89517dd3f0fSDmitry Torokhov break;
89617dd3f0fSDmitry Torokhov
89717dd3f0fSDmitry Torokhov case GC_DDR:
89817dd3f0fSDmitry Torokhov for (i = 0; i < 4; i++)
899a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY,
900a5a45b7fSMarcus Folkesson gc_psx_ddr_btn[i]);
90117dd3f0fSDmitry Torokhov for (i = 0; i < 12; i++)
902a5a45b7fSMarcus Folkesson input_set_capability(input_dev, EV_KEY, gc_psx_btn[i]);
9031da177e4SLinus Torvalds
9041da177e4SLinus Torvalds break;
9051da177e4SLinus Torvalds }
9061da177e4SLinus Torvalds
9070995174dSDmitry Torokhov err = input_register_device(pad->dev);
9080995174dSDmitry Torokhov if (err)
9090995174dSDmitry Torokhov goto err_free_dev;
9100995174dSDmitry Torokhov
91117dd3f0fSDmitry Torokhov return 0;
9120995174dSDmitry Torokhov
9130995174dSDmitry Torokhov err_free_dev:
9140995174dSDmitry Torokhov input_free_device(pad->dev);
9150995174dSDmitry Torokhov pad->dev = NULL;
9160995174dSDmitry Torokhov return err;
9171da177e4SLinus Torvalds }
9181da177e4SLinus Torvalds
gc_attach(struct parport * pp)919a517e87cSSudip Mukherjee static void gc_attach(struct parport *pp)
92017dd3f0fSDmitry Torokhov {
92117dd3f0fSDmitry Torokhov struct gc *gc;
92217dd3f0fSDmitry Torokhov struct pardevice *pd;
923c36906c5SSudip Mukherjee int i, port_idx;
9240995174dSDmitry Torokhov int count = 0;
925a517e87cSSudip Mukherjee int *pads, n_pads;
926a517e87cSSudip Mukherjee struct pardev_cb gc_parport_cb;
92717dd3f0fSDmitry Torokhov
928c36906c5SSudip Mukherjee for (port_idx = 0; port_idx < GC_MAX_PORTS; port_idx++) {
929c36906c5SSudip Mukherjee if (gc_cfg[port_idx].nargs == 0 || gc_cfg[port_idx].args[0] < 0)
930a517e87cSSudip Mukherjee continue;
931a517e87cSSudip Mukherjee
932c36906c5SSudip Mukherjee if (gc_cfg[port_idx].args[0] == pp->number)
933a517e87cSSudip Mukherjee break;
93417dd3f0fSDmitry Torokhov }
93517dd3f0fSDmitry Torokhov
936c36906c5SSudip Mukherjee if (port_idx == GC_MAX_PORTS) {
937a517e87cSSudip Mukherjee pr_debug("Not using parport%d.\n", pp->number);
938a517e87cSSudip Mukherjee return;
939a517e87cSSudip Mukherjee }
940c36906c5SSudip Mukherjee pads = gc_cfg[port_idx].args + 1;
941c36906c5SSudip Mukherjee n_pads = gc_cfg[port_idx].nargs - 1;
942a517e87cSSudip Mukherjee
943eb12a5f5SSudip Mukherjee memset(&gc_parport_cb, 0, sizeof(gc_parport_cb));
944a517e87cSSudip Mukherjee gc_parport_cb.flags = PARPORT_FLAG_EXCL;
945a517e87cSSudip Mukherjee
946c36906c5SSudip Mukherjee pd = parport_register_dev_model(pp, "gamecon", &gc_parport_cb,
947c36906c5SSudip Mukherjee port_idx);
94817dd3f0fSDmitry Torokhov if (!pd) {
949a1e12747SDmitry Torokhov pr_err("parport busy already - lp.o loaded?\n");
950a517e87cSSudip Mukherjee return;
95117dd3f0fSDmitry Torokhov }
95217dd3f0fSDmitry Torokhov
95317dd3f0fSDmitry Torokhov gc = kzalloc(sizeof(struct gc), GFP_KERNEL);
95417dd3f0fSDmitry Torokhov if (!gc) {
955a1e12747SDmitry Torokhov pr_err("Not enough memory\n");
95617dd3f0fSDmitry Torokhov goto err_unreg_pardev;
95717dd3f0fSDmitry Torokhov }
95817dd3f0fSDmitry Torokhov
95972ba9f0cSIngo Molnar mutex_init(&gc->mutex);
96017dd3f0fSDmitry Torokhov gc->pd = pd;
961a517e87cSSudip Mukherjee gc->parportno = pp->number;
962e99e88a9SKees Cook timer_setup(&gc->timer, gc_timer, 0);
96317dd3f0fSDmitry Torokhov
964c7fd018dSDmitry Torokhov for (i = 0; i < n_pads && i < GC_MAX_DEVICES; i++) {
96517dd3f0fSDmitry Torokhov if (!pads[i])
96617dd3f0fSDmitry Torokhov continue;
96717dd3f0fSDmitry Torokhov
968a517e87cSSudip Mukherjee if (gc_setup_pad(gc, i, pads[i]))
96977fc46caSDmitry Torokhov goto err_unreg_devs;
97017dd3f0fSDmitry Torokhov
9710995174dSDmitry Torokhov count++;
97217dd3f0fSDmitry Torokhov }
9731da177e4SLinus Torvalds
9740995174dSDmitry Torokhov if (count == 0) {
975a1e12747SDmitry Torokhov pr_err("No valid devices specified\n");
97617dd3f0fSDmitry Torokhov goto err_free_gc;
97717dd3f0fSDmitry Torokhov }
97817dd3f0fSDmitry Torokhov
979c36906c5SSudip Mukherjee gc_base[port_idx] = gc;
980a517e87cSSudip Mukherjee return;
98117dd3f0fSDmitry Torokhov
98277fc46caSDmitry Torokhov err_unreg_devs:
98317dd3f0fSDmitry Torokhov while (--i >= 0)
9840995174dSDmitry Torokhov if (gc->pads[i].dev)
9850995174dSDmitry Torokhov input_unregister_device(gc->pads[i].dev);
98617dd3f0fSDmitry Torokhov err_free_gc:
98717dd3f0fSDmitry Torokhov kfree(gc);
98817dd3f0fSDmitry Torokhov err_unreg_pardev:
98917dd3f0fSDmitry Torokhov parport_unregister_device(pd);
99017dd3f0fSDmitry Torokhov }
99117dd3f0fSDmitry Torokhov
gc_detach(struct parport * port)992a517e87cSSudip Mukherjee static void gc_detach(struct parport *port)
99317dd3f0fSDmitry Torokhov {
99417dd3f0fSDmitry Torokhov int i;
995a517e87cSSudip Mukherjee struct gc *gc;
996a517e87cSSudip Mukherjee
997a517e87cSSudip Mukherjee for (i = 0; i < GC_MAX_PORTS; i++) {
998a517e87cSSudip Mukherjee if (gc_base[i] && gc_base[i]->parportno == port->number)
999a517e87cSSudip Mukherjee break;
1000a517e87cSSudip Mukherjee }
1001a517e87cSSudip Mukherjee
1002a517e87cSSudip Mukherjee if (i == GC_MAX_PORTS)
1003a517e87cSSudip Mukherjee return;
1004a517e87cSSudip Mukherjee
1005a517e87cSSudip Mukherjee gc = gc_base[i];
1006a517e87cSSudip Mukherjee gc_base[i] = NULL;
100717dd3f0fSDmitry Torokhov
100817dd3f0fSDmitry Torokhov for (i = 0; i < GC_MAX_DEVICES; i++)
10090995174dSDmitry Torokhov if (gc->pads[i].dev)
10100995174dSDmitry Torokhov input_unregister_device(gc->pads[i].dev);
10111da177e4SLinus Torvalds parport_unregister_device(gc->pd);
10121da177e4SLinus Torvalds kfree(gc);
10131da177e4SLinus Torvalds }
10141da177e4SLinus Torvalds
1015a517e87cSSudip Mukherjee static struct parport_driver gc_parport_driver = {
1016a517e87cSSudip Mukherjee .name = "gamecon",
1017a517e87cSSudip Mukherjee .match_port = gc_attach,
1018a517e87cSSudip Mukherjee .detach = gc_detach,
1019a517e87cSSudip Mukherjee .devmodel = true,
1020a517e87cSSudip Mukherjee };
1021a517e87cSSudip Mukherjee
gc_init(void)10221da177e4SLinus Torvalds static int __init gc_init(void)
10231da177e4SLinus Torvalds {
102417dd3f0fSDmitry Torokhov int i;
102517dd3f0fSDmitry Torokhov int have_dev = 0;
10261da177e4SLinus Torvalds
102717dd3f0fSDmitry Torokhov for (i = 0; i < GC_MAX_PORTS; i++) {
102878167236SDmitry Torokhov if (gc_cfg[i].nargs == 0 || gc_cfg[i].args[0] < 0)
102917dd3f0fSDmitry Torokhov continue;
10301da177e4SLinus Torvalds
103178167236SDmitry Torokhov if (gc_cfg[i].nargs < 2) {
1032a1e12747SDmitry Torokhov pr_err("at least one device must be specified\n");
1033a517e87cSSudip Mukherjee return -EINVAL;
103417dd3f0fSDmitry Torokhov }
103517dd3f0fSDmitry Torokhov
103617dd3f0fSDmitry Torokhov have_dev = 1;
103717dd3f0fSDmitry Torokhov }
103817dd3f0fSDmitry Torokhov
1039a517e87cSSudip Mukherjee if (!have_dev)
1040a517e87cSSudip Mukherjee return -ENODEV;
104117dd3f0fSDmitry Torokhov
1042a517e87cSSudip Mukherjee return parport_register_driver(&gc_parport_driver);
10431da177e4SLinus Torvalds }
10441da177e4SLinus Torvalds
gc_exit(void)10451da177e4SLinus Torvalds static void __exit gc_exit(void)
10461da177e4SLinus Torvalds {
1047a517e87cSSudip Mukherjee parport_unregister_driver(&gc_parport_driver);
10481da177e4SLinus Torvalds }
10491da177e4SLinus Torvalds
10501da177e4SLinus Torvalds module_init(gc_init);
10511da177e4SLinus Torvalds module_exit(gc_exit);
1052