Subject: port-i386/1916: add a joystick driver to i386 port
To: None <gnats-bugs@gnats.netbsd.org>
From: Matthieu Herrb <matthieu@abel.laas.fr>
List: netbsd-bugs
Date: 01/08/1996 10:17:30
>Number:         1916
>Category:       port-i386
>Synopsis:       add a joystick driver to the i386 port
>Confidential:   no
>Severity:       non-critical
>Priority:       high
>Responsible:    gnats-admin (GNATS administrator)
>State:          open
>Class:          change-request
>Submitter-Id:   net
>Arrival-Date:   Mon Jan  8 04:50:01 1996
>Last-Modified:
>Originator:     Matthieu Herrb
>Organization:
Matthieu Herrb                          | CNRS/LAAS
E-mail: matthieu@laas.fr                | 7, avenue du Colonel Roche
   URL: http://www.laas.fr/~matthieu    | 31077 Toulouse Cedex - France
-----------------------------------------------------------------------
	War, what is it good for ? Absolutely nothing !
>Release:        1.1
>Environment:
		N/A

>Description:
	This is a port of the FreeBSD joystick port driver. There is
	an XInput extention in progress in XFree86 that can make use
	of it.

>How-To-Repeat:
	N/A
>Fix:
This patch is relative to /usr/src in a 1.1 source tree:

*** /dev/null	Sun Jan  7 19:24:37 1996
--- sys/arch/i386/isa/joy.c	Sun Jan  7 19:53:39 1996
***************
*** 0 ****
--- 1,253 ----
+ /*-
+  * Copyright (c) 1995 Jean-Marc Zucconi
+  * All rights reserved.
+  *
+  * Ported to NetBSD by Matthieu Herrb <matthieu@laas.fr>
+  *
+  * Redistribution and use in source and binary forms, with or without
+  * modification, are permitted provided that the following conditions
+  * are met:
+  * 1. Redistributions of source code must retain the above copyright
+  *    notice, this list of conditions and the following disclaimer
+  *    in this position and unchanged.
+  * 2. Redistributions in binary form must reproduce the above copyright
+  *    notice, this list of conditions and the following disclaimer in the
+  *    documentation and/or other materials provided with the distribution.
+  * 3. The name of the author may not be used to endorse or promote products
+  *    derived from this software withough specific prior written permission
+  *
+  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+  *
+  */
+ 
+ #include <errno.h>
+ 
+ #include <sys/param.h>
+ #include <sys/systm.h>
+ #include <sys/kernel.h>
+ #include <sys/device.h>
+ 
+ #include <machine/cpu.h>
+ #include <machine/pio.h>
+ #include <machine/cpufunc.h>
+ 
+ #include <machine/joystick.h>
+ 
+ #include <dev/isa/isavar.h>
+ #include <dev/isa/isareg.h>
+ #include <i386/isa/timerreg.h>
+ 
+ /* The game port can manage 4 buttons and 4 variable resistors (usually 2
+  * joysticks, each with 2 buttons and 2 pots.) via the port at address 0x201.
+  * Getting the state of the buttons is done by reading the game port:
+  * buttons 1-4 correspond to bits 4-7 and resistors 1-4 (X1, Y1, X2, Y2)
+  * to bits 0-3.
+  * if button 1 (resp 2, 3, 4) is pressed, the bit 4 (resp 5, 6, 7) is set to 0
+  * to get the value of a resistor, write the value 0xff at port and
+  * wait until the corresponding bit returns to 0.
+  */
+ 
+ 
+ /* the formulae below only work if u is  ``not too large''. See also
+  * the discussion in microtime.s */
+ #define usec2ticks(u) 	(((u) * 19549)>>14)
+ #define ticks2usec(u) 	(((u) * 3433)>>12)
+ 
+ 
+ #define joypart(d) minor(d)&1
+ #define JOYUNIT(d) minor(d)>>1&3
+ 
+ #ifndef JOY_TIMEOUT
+ #define JOY_TIMEOUT   2000 /* 2 milliseconds */
+ #endif
+ 
+ struct joy_softc {
+     struct device sc_dev;
+     int port;
+     int x_off[2], y_off[2];
+     int timeout[2];
+ };
+ 
+ 
+ int joyprobe __P((struct device *, void *, void *));
+ void joyattach __P((struct device *, struct device *, void *));
+ int joyopen __P((dev_t, int, int, struct proc *));
+ int joyclose __P((dev_t, int, int, struct proc *));
+ static int get_tick __P((void));
+ 
+ struct cfdriver joycd = {
+     NULL, "joy", joyprobe, joyattach, DV_DULL, sizeof(struct joy_softc)
+ };
+ 
+ 
+ int
+ joyprobe (parent, match, aux)
+     struct device *parent;
+     void *match, *aux;
+ 
+ {
+ #ifdef WANT_JOYSTICK_CONNECTED
+     outb (dev->id_iobase, 0xff);
+     DELAY (10000); /*  10 ms delay */
+     return (inb (dev->id_iobase) & 0x0f) != 0x0f;
+ #else
+     return 1;
+ #endif
+ }
+ 
+ void
+ joyattach (parent, self, aux)
+     struct device *parent, *self;
+     void *aux;
+ {
+     struct joy_softc *sc = (void *)self;
+     struct isa_attach_args *ia = aux;
+     int	unit = sc->sc_dev.dv_unit;
+     
+     sc->port = ia->ia_iobase;
+     sc->timeout[0] = sc->timeout[1] = 0;
+     printf(": joystick\n", unit);
+ }
+ 
+ int
+ joyopen (dev, flag, mode, p)
+     dev_t dev;
+     int flag, mode;
+     struct proc *p;
+ {
+     int unit = JOYUNIT (dev);
+     int i = joypart (dev);
+     struct joy_softc *sc;
+ 
+     if (unit >= joycd.cd_ndevs) {
+ 	return(ENXIO);
+     }
+     sc = joycd.cd_devs[unit];
+ 
+     if (sc->timeout[i]) {
+ 	return EBUSY;
+     }
+     sc->x_off[i] = sc->y_off[i] = 0;
+     sc->timeout[i] = JOY_TIMEOUT;
+     return 0;
+ }
+ 
+ int
+ joyclose(dev, flag, mode, p)
+ 	dev_t dev;
+ 	int flag, mode;
+ 	struct proc *p;
+ {
+     int unit = JOYUNIT (dev);
+     int i = joypart (dev);
+     struct joy_softc *sc = joycd.cd_devs[unit];
+ 
+     sc->timeout[i] = 0;
+     return 0;
+ }
+ 
+ int
+ joyread(dev, uio, flag)
+ 	dev_t dev;
+ 	struct uio *uio;
+ 	int flag;
+ {
+     int unit = JOYUNIT(dev);
+     struct joy_softc *sc = joycd.cd_devs[unit];
+     int port = sc->port;
+     int i, t0, t1;
+     int state = 0, x = 0, y = 0;
+     struct joystick c;
+ 
+     disable_intr ();
+     outb (port, 0xff);
+     t0 = get_tick ();
+     t1 = t0;
+     i = usec2ticks(sc->timeout[joypart(dev)]);
+     while (t0-t1 < i) {
+ 	state = inb (port);
+ 	if (joypart(dev) == 1)
+ 	    state >>= 2;
+ 	t1 = get_tick ();
+ 	if (t1 > t0)
+ 	    t1 -= TIMER_FREQ/hz;
+ 	if (!x && !(state & 0x01))
+ 	    x = t1;
+ 	if (!y && !(state & 0x02))
+ 	    y =  t1;
+ 	if (x && y)
+ 	    break;
+     }
+     enable_intr ();
+     c.x = x ? sc->x_off[joypart(dev)] + ticks2usec(t0-x) : 0x80000000;
+     c.y = y ? sc->y_off[joypart(dev)] + ticks2usec(t0-y) : 0x80000000;
+     state >>= 4;
+     c.b1 = ~state & 1;
+     c.b2 = ~(state >> 1) & 1;
+     return uiomove ((caddr_t)&c, sizeof(struct joystick), uio);
+ }
+ 
+ int
+ joyioctl(dev, cmd, data, flag, p)
+     dev_t dev;
+     u_long cmd;
+     caddr_t data;
+     int flag;
+     struct proc *p;
+ {
+     int unit = JOYUNIT (dev);
+     int i = joypart (dev);
+     struct joy_softc *sc = joycd.cd_devs[unit];
+     int x;
+     
+     switch (cmd) {
+       case JOY_SETTIMEOUT:
+ 	x = *(int *) data;
+ 	if (x < 1 || x > 10000) /* 10ms maximum! */
+ 	    return EINVAL;
+ 	sc->timeout[i] = x;
+ 	break;
+       case JOY_GETTIMEOUT:
+ 	*(int *) data = sc->timeout[i];
+       break;
+       case JOY_SET_X_OFFSET:
+ 	sc->x_off[i] = *(int *) data;
+ 	break;
+       case JOY_SET_Y_OFFSET:
+ 	sc->y_off[i] = *(int *) data;
+ 	break;
+       case JOY_GET_X_OFFSET:
+ 	*(int *) data = sc->x_off[i];
+       break;
+       case JOY_GET_Y_OFFSET:
+ 	*(int *) data = sc->y_off[i];
+       break;
+       default:
+ 	return ENXIO;
+     }
+     return 0;
+ }
+ 
+ static int
+ get_tick ()
+ {
+     int low, high;
+ 
+     outb (TIMER_MODE, TIMER_SEL0);
+     low = inb (TIMER_CNTR0);
+     high = inb (TIMER_CNTR0);
+ 
+     return (high << 8) | low;
+ }
+ 
+ 
+ 
*** /dev/null	Sun Jan  7 19:24:37 1996
--- sys/arch/i386/include/joystick.h	Sun Jan  7 14:12:03 1996
***************
*** 0 ****
--- 1,21 ----
+ #ifndef _JOY_IOCTL_H_
+ #define _JOY_IOCTL_H_
+ 
+ #include <sys/types.h>
+ #include <sys/ioctl.h>
+ 
+ struct joystick {
+     int x;
+     int y;
+     int b1;
+     int b2;
+ };
+ 
+ #define JOY_SETTIMEOUT    _IOW('J', 1, int)    /* set timeout */
+ #define JOY_GETTIMEOUT    _IOR('J', 2, int)    /* get timeout */
+ #define JOY_SET_X_OFFSET  _IOW('J', 3, int)    /* set offset on X-axis */
+ #define JOY_SET_Y_OFFSET  _IOW('J', 4, int)    /* set offset on X-axis */
+ #define JOY_GET_X_OFFSET  _IOR('J', 5, int)    /* get offset on X-axis */
+ #define JOY_GET_Y_OFFSET  _IOR('J', 6, int)    /* get offset on Y-axis */
+ 
+ #endif /* _JOY_IOCTL_H_ */
*** sys/arch/i386/conf/files.i386~	Sat Oct 14 02:55:39 1995
--- sys/arch/i386/conf/files.i386	Sun Jan  7 19:53:10 1996
***************
*** 107,112 ****
--- 107,116 ----
  device	le at isa, pci: ether, ifnet, isadma
  file	dev/isa/if_le.c			le
  
+ # Game adapter (joystick)
+ device	joy at isa
+ file	arch/i386/isa/joy.c			joy needs-flag
+ 
  #
  # EISA-only drivers
  #
*** sys/arch/i386/i386/conf.c~	Sat Oct 14 02:56:36 1995
--- sys/arch/i386/i386/conf.c	Sun Jan  7 18:18:06 1996
***************
*** 111,116 ****
--- 111,122 ----
  	dev_init(c,n,write), dev_init(c,n,ioctl), (dev_type_stop((*))) enodev, \
  	0, seltrue, (dev_type_mmap((*))) enodev }
  
+ /* open, close, read, ioctl */
+ #define cdev_joy_init(c,n) { \
+ 	dev_init(c,n,open), dev_init(c,n,close), dev_init(c,n,read), \
+ 	(dev_type_write((*))) enodev, dev_init(c,n,ioctl), \
+ 	(dev_type_stop((*))) enodev, 0, seltrue, (dev_type_mmap((*))) enodev }
+ 		 
  cdev_decl(cn);
  cdev_decl(ctty);
  #define	mmread	mmrw
***************
*** 170,175 ****
--- 176,183 ----
  cdev_decl(audio);
  cdev_decl(svr4_net);
  cdev_decl(ccd);
+ #include "joy.h"
+ cdev_decl(joy);
  
  struct cdevsw	cdevsw[] =
  {
***************
*** 201,207 ****
  	cdev_bpftun_init(NBPFILTER,bpf),/* 23: Berkeley packet filter */
  	cdev_notdef(),			/* 24 */
  	cdev_notdef(),			/* 25 */
! 	cdev_notdef(),			/* 26 */
  	cdev_spkr_init(NSPEAKER,spkr),	/* 27: PC speaker */
  	cdev_lkm_init(NLKM,lkm),	/* 28: loadable module driver */
  	cdev_lkm_dummy(),		/* 29 */
--- 209,215 ----
  	cdev_bpftun_init(NBPFILTER,bpf),/* 23: Berkeley packet filter */
  	cdev_notdef(),			/* 24 */
  	cdev_notdef(),			/* 25 */
! 	cdev_joy_init(NJOY,joy),	/* 26: Game adapter */
  	cdev_spkr_init(NSPEAKER,spkr),	/* 27: PC speaker */
  	cdev_lkm_init(NLKM,lkm),	/* 28: loadable module driver */
  	cdev_lkm_dummy(),		/* 29 */
*** etc/etc.i386/MAKEDEV~	Wed Oct 11 12:30:10 1995
--- etc/etc.i386/MAKEDEV	Sun Jan  7 20:04:59 1996
***************
*** 338,343 ****
--- 338,351 ----
  	chown root.wheel audio sound mixer
  	chmod 666 audio sound mixer
  	;;
+ 
+ joy*)
+ 	unit=`expr $i : 'joy\(.*\)'`
+ 	rm -f joy$unit
+ 	mknod joy$unit c 26 $unit
+ 	chown root.wheel joy$unit
+ 	chmod 666 joy$unit
+ 	;;
  	
  local)
  	umask 0
*** /dev/null	Sun Jan  7 19:24:37 1996
--- share/man/man4/man4.i386/joy.4	Sun Jan  7 21:05:45 1996
***************
*** 0 ****
--- 1,114 ----
+ .\"
+ .\" Copyright (c) 1996 Matthieu Herrb
+ .\" All rights reserved.
+ .\"
+ .\" Redistribution and use in source and binary forms, with or without
+ .\" modification, are permitted provided that the following conditions
+ .\" are met:
+ .\" 1. Redistributions of source code must retain the above copyright
+ .\"    notice, this list of conditions and the following disclaimer.
+ .\" 2. Redistributions in binary form must reproduce the above copyright
+ .\"    notice, this list of conditions and the following disclaimer in the
+ .\"    documentation and/or other materials provided with the distribution.
+ .\" 3. All advertising materials mentioning features or use of this software
+ .\"    must display the following acknowledgement:
+ .\"      This product includes software developed by Christopher G. Demetriou.
+ .\" 3. The name of the author may not be used to endorse or promote products
+ .\"    derived from this software without specific prior written permission
+ .\"
+ .\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ .\" IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ .\" OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ .\" IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ .\" INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ .\" NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ .\" DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ .\" THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ .\" (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ .\" THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ .\"
+ .\"     $Id$
+ .\"
+ .Dd January 7, 1996
+ .Dt JOY 4 i386
+ .Os "NetBSD 1.1A"
+ .Sh NAME
+ .Nm joy
+ .Nd
+ Games adapter driver
+ .Sh SYNOPSIS
+ .Cd "joy0 at isa? port 0x201"
+ .Sh DESCRIPTION
+ This driver provides access to the games adapter. The lower bit in the
+ minor device number selects the joystick: 0 is the first joystick and
+ 1 is the second.
+ .Pp
+ The game control adapter allows up to two joysticks to be attached to
+ the system. The adapter plus the driver convert the present resistive
+ value to a relative joystick position. On receipt of an output signal,
+ four timing circuits are started. By determining the time required for
+ the circuit to time-out (a function of the resistance), the paddle
+ position can be determined. The adapter could be used as a general
+ purpose I/O card with four analog (resistive) inputs plus four digital
+ input points. 
+ .Pp
+ Applications may call ioctl() on a game adapter driver file descriptor
+ to set and get the offets of the two potentiometers and the maximum
+ time-out value for the circuit. The
+ ioctl() commands are listed in
+ .Pa Aq machine/joystick.h
+ and currently are:
+ .Pp
+ .Bl -tag -width JOY_GET_X_OFFSET -compact
+ .It JOY_SETTIMEOUT
+ Sets the maximum time-out for the adapater.
+ .It JOY_GETTIMEOUT
+ Returns the current maximun time-out.
+ .It JOY_SET_X_OFFSET
+ Sets an offset on X value.
+ .It JOY_GET_X_OFFSET
+ Returns the current X offset.
+ .It JOY_SET_Y_OFFSET
+ Sets an offset on Y value.
+ .It JOY_GET_Y_OFFSET
+ Returns the current Y offset.
+ .El
+ .Pp
+ All this commands take an integer parameter.
+ .Pp
+ Read() on the file descriptor returns a 
+ .Fa joystick 
+ structure:
+ .Bd -literal -offset indent
+ struct joystick {
+ 	int x;
+ 	int y;
+ 	int b1;
+ 	int b2;
+ };
+ .Ed
+ .Pp
+ The fields have the following functions:
+ .Bl -tag -width b1
+ .It Fa x
+ The current X coordinate of the joystick (or position of paddle 1)
+ .It Fa y
+ The current Y coordinate of the joystick (or position of paddle 2)
+ .It Fa b1
+ The current state of button 1
+ .It Fa b2
+ The current state of button 2
+ .El
+ .Sh FILES
+ .Bl -tag -width Pa -compact
+ .It Pa /dev/joy0
+ first joystick
+ .br
+ .It Pa /dev/joy1
+ second joystick
+ .El
+ .Sh AUTHORS
+ .Pp
+ Jean-Marc Zucconi wrote the FreeBSD driver. Matthieu Herrb ported it
+ to NetBSD and wrote this manual page.
+ 
>Audit-Trail:
>Unformatted: