xref: /openbmc/linux/drivers/input/joystick/gamecon.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
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