summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--usr.sbin/moused/Makefile9
-rw-r--r--usr.sbin/moused/moused.c562
2 files changed, 571 insertions, 0 deletions
diff --git a/usr.sbin/moused/Makefile b/usr.sbin/moused/Makefile
new file mode 100644
index 0000000..35e25a7
--- /dev/null
+++ b/usr.sbin/moused/Makefile
@@ -0,0 +1,9 @@
+PROG= moused
+SRCS= moused.c
+NOMAN= moused
+
+BINOWN= root
+BINMODE=4555
+INSTALLFLAGS=-fschg
+
+.include <bsd.prog.mk>
diff --git a/usr.sbin/moused/moused.c b/usr.sbin/moused/moused.c
new file mode 100644
index 0000000..342182c
--- /dev/null
+++ b/usr.sbin/moused/moused.c
@@ -0,0 +1,562 @@
+/**
+ ** Copyright (c) 1995 Michael Smith, 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 as
+ ** the first lines of this file unmodified.
+ ** 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 acknowledgment:
+ ** This product includes software developed by Michael Smith.
+ ** 4. 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 Michael Smith ``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 Michael Smith 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$
+ **/
+
+/**
+ ** MOUSED.C
+ **
+ ** Mouse daemon : listens to serial port for mouse data stream,
+ ** interprets same and passes ioctls off to the console driver.
+ **
+ ** The mouse interface functions are derived closely from the mouse
+ ** handler in the XFree86 X server. Many thanks to the XFree86 people
+ ** for their great work!
+ **
+ **/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <limits.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <termios.h>
+#include <machine/console.h>
+
+#define debug(fmt,args...) \
+ if (debug&&nodaemon) fprintf(stderr,"%s: " fmt "\n", progname, ##args)
+
+char *progname;
+int debug = 0;
+int nodaemon = 0;
+
+void usage(void);
+
+#define R_UNKNOWN 0
+#define R_MICROSOFT 1
+#define R_MOUSESYS 2
+#define R_MMSERIES 3
+#define R_LOGITECH 4
+#define R_BUSMOUSE 5
+#define R_LOGIMAN 6
+#define R_PS_2 7
+
+char *rnames[] = {
+ "xxx",
+ "microsoft",
+ "mousesystems",
+ "mmseries",
+ "logitech",
+ "busmouse",
+ "mouseman",
+ "ps/2",
+ NULL
+};
+
+unsigned short rodentcflags[] =
+{
+ 0, /* nomouse */
+ (CS7 | CREAD | CLOCAL | HUPCL ), /* MicroSoft */
+ (CS8 | CSTOPB | CREAD | CLOCAL | HUPCL ), /* MouseSystems */
+ (CS8 | PARENB | PARODD | CREAD | CLOCAL | HUPCL ), /* MMSeries */
+ (CS8 | CSTOPB | CREAD | CLOCAL | HUPCL ), /* Logitech */
+ 0, /* BusMouse */
+ (CS7 | CREAD | CLOCAL | HUPCL ), /* MouseMan */
+ 0 /* PS/2 */
+};
+
+
+typedef struct
+{
+ int
+ dx,dy,
+ buttons;
+} ACTIVITY;
+
+
+struct rodentparam
+{
+ int
+ baudrate,
+ samplerate,
+ flags,
+ rtype,
+ lastbuttons,
+ buttons,
+ mfd;
+
+ char
+ *portname;
+
+} rodent = { baudrate : 1200,
+ samplerate : 0,
+ flags : 0,
+ rtype : R_UNKNOWN,
+ lastbuttons : 0,
+ buttons : 0,
+ mfd : -1,
+ portname : NULL };
+
+#define ChordMiddle 1
+
+void r_init(void);
+ACTIVITY *r_protocol(u_char b);
+
+void
+main(int argc, char *argv[])
+{
+ int c,i,cfd;
+ u_char b;
+ ACTIVITY *act;
+ struct termios t;
+ struct mouse_info mouse;
+ int saved_buttons = 0;
+
+ progname = argv[0];
+
+ while((c = getopt(argc,argv,"cdfsp:t:h?")) != EOF)
+ switch(c)
+ {
+ case 'c':
+ rodent.flags |= ChordMiddle;
+ break;
+
+ case 'd':
+ debug = 1;
+ break;
+
+ case 'f':
+ nodaemon = 1;
+ break;
+
+ case 'p':
+ rodent.portname = optarg;
+ break;
+
+ case 's':
+ rodent.baudrate = 9600;
+ usage();
+
+ case 't':
+ for (i = 0; rnames[i]; i++)
+ if (!strcmp(optarg,rnames[i]))
+ {
+ debug("rodent is %s",rnames[i]);
+ rodent.rtype = i;
+ break;
+ }
+ if (rnames[i])
+ break;
+ fprintf(stderr,"no such mouse type `%s'\n",optarg);
+ usage();
+
+ case 'h':
+ case '?':
+ default:
+ usage();
+ }
+
+ switch(rodent.rtype)
+ {
+ case R_BUSMOUSE:
+ if (!rodent.portname)
+ rodent.portname = "/dev/mse0";
+ break;
+ case R_PS_2:
+ if (!rodent.portname)
+ rodent.portname = "/dev/psm0";
+ break;
+ default:
+ if (rodent.portname)
+ break;
+ fprintf(stderr,"No port name specified\n");
+ usage();
+ }
+
+ if ((rodent.mfd = open(rodent.portname, O_RDWR, 0)) == -1)
+ {
+ fprintf(stderr,"Can't open %s : %s\n",rodent.portname,strerror(errno));
+ usage();
+ }
+ if ((rodent.rtype != R_PS_2) && (rodent.rtype != R_BUSMOUSE))
+ {
+ tcgetattr(rodent.mfd,&t);
+ cfsetspeed(&t,rodent.baudrate);
+ t.c_cflag = rodentcflags[rodent.rtype];
+ if (tcsetattr(rodent.mfd,TCSAFLUSH,&t))
+ {
+ fprintf(stderr,
+ "%s: tcsetattr() failed : %s\n",progname,strerror(errno));
+ exit(1);
+ }
+ }
+ r_init(); /* call init function */
+
+ if ((cfd = open("/dev/console", O_RDWR, 0)) == -1)
+ fprintf(stderr, "error on /dev/console\n");
+
+ if (!nodaemon)
+ if (daemon(0,0))
+ {
+ fprintf(stderr,
+ "%s: daemon() failed : %s\n",progname,strerror(errno));
+ exit(1);
+ }
+
+ for(;;)
+ {
+ i = read(rodent.mfd,&b,1); /* get a byte */
+ if (i != 1) /* read returned or error; goodbye */
+ {
+ debug("read returned %d : %s exiting",i,strerror(errno));
+ close(rodent.mfd);
+ exit(1);
+ }
+ act = r_protocol(b); /* pass byte to handler */
+ if (act) /* handler detected action */
+ {
+ if (act->buttons != saved_buttons) {
+ if (act->buttons == 0x04) {
+ mouse.operation = MOUSE_CUT_START;
+ ioctl(cfd, CONS_MOUSECTL, &mouse);
+ }
+ if (act->buttons == 0x00) {
+ mouse.operation = MOUSE_CUT_END;
+ ioctl(cfd, CONS_MOUSECTL, &mouse);
+ }
+ if (act->buttons == 0x01) {
+ mouse.operation = MOUSE_RETURN_CUTBUFFER;
+ ioctl(cfd, CONS_MOUSECTL, &mouse);
+ }
+ saved_buttons = act->buttons;
+ }
+ mouse.operation = MOUSE_MOVEREL;
+ mouse.x = act->dx;
+ mouse.y = act->dy;
+ ioctl(cfd, CONS_MOUSECTL, &mouse);
+ debug("Activity : buttons 0x%02x dx %d dy %d",
+ act->buttons,act->dx,act->dy);
+ }
+ }
+}
+
+
+/**
+ ** usage
+ **
+ ** Complain, and free the CPU for more worthy tasks
+ **/
+void
+usage(void)
+{
+ fprintf(stderr,
+ " Usage is %s [options] -p <port> -t <mousetype>\n"
+ " Options are -s Select 9600 baud mouse.\n"
+ " -f Don't become a daemon\n"
+ " -d Enable debugging messages\n"
+ " -c Enable ChordMiddle option\n"
+ " <mousetype> should be one of :\n"
+ " microsoft\n"
+ " mousesystems\n"
+ " mmseries\n"
+ " logitech\n"
+ " busmouse\n"
+ " mouseman\n"
+ " ps/2\n"
+ " mmhittab\n"
+ ,progname);
+
+ exit(1);
+}
+
+
+/**
+ ** Mouse interface code, courtesy of XFree86 3.1.2.
+ **
+ ** Note: Various bits have been trimmed, and in my shortsighted enthusiasm
+ ** to clean, reformat and rationalise naming, it's quite possible that
+ ** some things in here have been broken.
+ **
+ ** I hope not 8)
+ **
+ ** The following code is derived from a module marked :
+ **/
+
+/* $XConsortium: xf86_Mouse.c,v 1.2 94/10/12 20:33:21 kaleb Exp $ */
+/* $XFree86: xc/programs/Xserver/hw/xfree86/common/xf86_Mouse.c,v 3.2 1995/01/28
+ 17:03:40 dawes Exp $ */
+/*
+ *
+ * Copyright 1990,91 by Thomas Roell, Dinkelscherben, Germany.
+ * Copyright 1993 by David Dawes <dawes@physics.su.oz.au>
+ *
+ * Permission to use, copy, modify, distribute, and sell this software and its
+ * documentation for any purpose is hereby granted without fee, provided that
+ * the above copyright notice appear in all copies and that both that
+ * copyright notice and this permission notice appear in supporting
+ * documentation, and that the names of Thomas Roell and David Dawes not be
+ * used in advertising or publicity pertaining to distribution of the
+ * software without specific, written prior permission. Thomas Roell
+ * and David Dawes makes no representations about the suitability of this
+ * software for any purpose. It is provided "as is" without express or
+ * implied warranty.
+ *
+ * THOMAS ROELL AND DAVID DAWES DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
+ * SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ * FITNESS, IN NO EVENT SHALL THOMAS ROELL OR DAVID DAWES BE LIABLE FOR ANY
+ * SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
+ * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
+ * CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
+ * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ *
+ */
+
+
+void
+r_init(void)
+{
+ /**
+ ** This comment is a little out of context here, but it contains
+ ** some useful information...
+ ********************************************************************
+ **
+ ** The following lines take care of the Logitech MouseMan protocols.
+ **
+ ** NOTE: There are different versions of both MouseMan and TrackMan!
+ ** Hence I add another protocol P_LOGIMAN, which the user can
+ ** specify as MouseMan in his XF86Config file. This entry was
+ ** formerly handled as a special case of P_MS. However, people
+ ** who don't have the middle button problem, can still specify
+ ** Microsoft and use P_MS.
+ **
+ ** By default, these mice should use a 3 byte Microsoft protocol
+ ** plus a 4th byte for the middle button. However, the mouse might
+ ** have switched to a different protocol before we use it, so I send
+ ** the proper sequence just in case.
+ **
+ ** NOTE: - all commands to (at least the European) MouseMan have to
+ ** be sent at 1200 Baud.
+ ** - each command starts with a '*'.
+ ** - whenever the MouseMan receives a '*', it will switch back
+ ** to 1200 Baud. Hence I have to select the desired protocol
+ ** first, then select the baud rate.
+ **
+ ** The protocols supported by the (European) MouseMan are:
+ ** - 5 byte packed binary protocol, as with the Mouse Systems
+ ** mouse. Selected by sequence "*U".
+ ** - 2 button 3 byte MicroSoft compatible protocol. Selected
+ ** by sequence "*V".
+ ** - 3 button 3+1 byte MicroSoft compatible protocol (default).
+ ** Selected by sequence "*X".
+ **
+ ** The following baud rates are supported:
+ ** - 1200 Baud (default). Selected by sequence "*n".
+ ** - 9600 Baud. Selected by sequence "*q".
+ **
+ ** Selecting a sample rate is no longer supported with the MouseMan!
+ ** Some additional lines in xf86Config.c take care of ill configured
+ ** baud rates and sample rates. (The user will get an error.)
+ */
+
+
+ if (rodent.rtype == R_LOGIMAN)
+ {
+ write(rodent.mfd, "*X", 2);
+ }
+ if ((rodent.rtype != R_BUSMOUSE) && (rodent.rtype != R_PS_2))
+ {
+ if (rodent.rtype == R_LOGITECH)
+ write(rodent.mfd, "S", 1);
+ /** Support for the Hitachi PUMA Plus tablet not brought through */
+ if (rodent.samplerate <= 0) write(rodent.mfd, "O", 1);
+ else if (rodent.samplerate <= 15) write(rodent.mfd, "J", 1);
+ else if (rodent.samplerate <= 27) write(rodent.mfd, "K", 1);
+ else if (rodent.samplerate <= 42) write(rodent.mfd, "L", 1);
+ else if (rodent.samplerate <= 60) write(rodent.mfd, "R", 1);
+ else if (rodent.samplerate <= 85) write(rodent.mfd, "M", 1);
+ else if (rodent.samplerate <= 125) write(rodent.mfd, "Q", 1);
+ else write(rodent.mfd, "N", 1);
+ }
+
+#ifdef CLEARDTR_SUPPORT /* XXX find someone to tell me about this */
+ if (xf86Info.mseType == P_MSC && (xf86Info.mouseFlags & MF_CLEAR_DTR))
+ {
+ int val = TIOCM_DTR;
+ ioctl(xf86Info.mseFd, TIOCMBIC, &val);
+ }
+ if (xf86Info.mseType == P_MSC && (xf86Info.mouseFlags & MF_CLEAR_RTS))
+ {
+ int val = TIOCM_RTS;
+ ioctl(xf86Info.mseFd, TIOCMBIC, &val);
+ }
+#endif
+}
+
+ACTIVITY *
+r_protocol(u_char rBuf)
+{
+ static int pBufP = 0;
+ static unsigned char pBuf[8];
+ static ACTIVITY act;
+
+ static unsigned char proto[9][5] = {
+ /* hd_mask hd_id dp_mask dp_id nobytes */
+ { 0, 0, 0, 0, 0 }, /* nomouse */
+ { 0x40, 0x40, 0x40, 0x00, 3 }, /* MicroSoft */
+ { 0xf8, 0x80, 0x00, 0x00, 5 }, /* MouseSystems */
+ { 0xe0, 0x80, 0x80, 0x00, 3 }, /* MMSeries */
+ { 0xe0, 0x80, 0x80, 0x00, 3 }, /* Logitech */
+ { 0xf8, 0x80, 0x00, 0x00, 5 }, /* BusMouse */
+ { 0x40, 0x40, 0x40, 0x00, 3 }, /* MouseMan */
+ { 0xc0, 0x00, 0x00, 0x00, 3 }, /* PS/2 mouse */
+ };
+
+ debug("received char 0x%x",(int)rBuf);
+
+ /*
+ * Hack for resyncing: We check here for a package that is:
+ * a) illegal (detected by wrong data-package header)
+ * b) invalid (0x80 == -128 and that might be wrong for MouseSystems)
+ * c) bad header-package
+ *
+ * NOTE: b) is a voilation of the MouseSystems-Protocol, since values of
+ * -128 are allowed, but since they are very seldom we can easily
+ * use them as package-header with no button pressed.
+ * NOTE/2: On a PS/2 mouse any byte is valid as a data byte. Furthermore,
+ * 0x80 is not valid as a header byte. For a PS/2 mouse we skip
+ * checking data bytes.
+ * For resyncing a PS/2 mouse we require the two most significant
+ * bits in the header byte to be 0. These are the overflow bits,
+ * and in case of an overflow we actually lose sync. Overflows
+ * are very rare, however, and we quickly gain sync again after
+ * an overflow condition. This is the best we can do. (Actually,
+ * we could use bit 0x08 in the header byte for resyncing, since
+ * that bit is supposed to be always on, but nobody told
+ * Microsoft...)
+ */
+
+ if (pBufP != 0 && rodent.rtype != R_PS_2 &&
+ ((rBuf & proto[rodent.rtype][2]) != proto[rodent.rtype][3]
+ || rBuf == 0x80))
+ {
+ pBufP = 0; /* skip package */
+ }
+
+ if (pBufP == 0 &&
+ (rBuf & proto[rodent.rtype][0]) != proto[rodent.rtype][1])
+ {
+ /*
+ * Hack for Logitech MouseMan Mouse - Middle button
+ *
+ * Unfortunately this mouse has variable length packets: the standard
+ * Microsoft 3 byte packet plus an optional 4th byte whenever the
+ * middle button status changes.
+ *
+ * We have already processed the standard packet with the movement
+ * and button info. Now post an event message with the old status
+ * of the left and right buttons and the updated middle button.
+ */
+
+ /*
+ * Even worse, different MouseMen and TrackMen differ in the 4th
+ * byte: some will send 0x00/0x20, others 0x01/0x21, or even
+ * 0x02/0x22, so I have to strip off the lower bits.
+ */
+ if ((rodent.rtype == R_MICROSOFT || rodent.rtype == R_LOGIMAN)
+ && (char)(rBuf & ~0x23) == 0)
+ {
+ act.buttons = ((int)(rBuf & 0x20) >> 4)
+ | (rodent.lastbuttons & 0x05);
+ rodent.lastbuttons = act.buttons; /* save new button state */
+ return(&act);
+ }
+
+ return(NULL); /* skip package */
+ }
+
+ pBuf[pBufP++] = rBuf;
+ if (pBufP != proto[rodent.rtype][4]) return(NULL);
+
+ /*
+ * assembly full package
+ */
+
+ debug("Assembled full packet (len %d) %x,%x,%x,%x,%x",
+ proto[rodent.rtype][4], pBuf[0],pBuf[1],pBuf[2],pBuf[3],pBuf[4]);
+
+ switch(rodent.rtype)
+ {
+ case R_LOGIMAN: /* MouseMan / TrackMan */
+ case R_MICROSOFT: /* Microsoft */
+ if (rodent.flags & ChordMiddle)
+ act.buttons = (((int) pBuf[0] & 0x30) == 0x30) ? 2 :
+ ((int)(pBuf[0]&0x20)>>3) | ((int)(pBuf[0]&0x10)>>4);
+ else
+ act.buttons = (rodent.lastbuttons & 2)
+ | ((int)(pBuf[0] & 0x20) >> 3)
+ | ((int)(pBuf[0] & 0x10) >> 4);
+ act.dx = (char)(((pBuf[0] & 0x03) << 6) | (pBuf[1] & 0x3F));
+ act.dy = (char)(((pBuf[0] & 0x0C) << 4) | (pBuf[2] & 0x3F));
+ break;
+
+ case R_MOUSESYS: /* Mouse Systems Corp */
+ act.buttons = (~pBuf[0]) & 0x07;
+ act.dx = (char)(pBuf[1]) + (char)(pBuf[3]);
+ act.dy = - ((char)(pBuf[2]) + (char)(pBuf[4]));
+ break;
+
+ case R_MMSERIES: /* MM Series */
+ case R_LOGITECH: /* Logitech Mice */
+ act.buttons = pBuf[0] & 0x07;
+ act.dx = (pBuf[0] & 0x10) ? pBuf[1] : - pBuf[1];
+ act.dy = (pBuf[0] & 0x08) ? - pBuf[2] : pBuf[2];
+ break;
+
+ case R_BUSMOUSE: /* BusMouse */
+ act.buttons = (~pBuf[0]) & 0x07;
+ act.dx = (char)pBuf[1];
+ act.dy = - (char)pBuf[2];
+ break;
+
+ case R_PS_2: /* PS/2 mouse */
+ act.buttons = (pBuf[0] & 0x04) >> 1 | /* Middle */
+ (pBuf[0] & 0x02) >> 1 | /* Right */
+ (pBuf[0] & 0x01) << 2; /* Left */
+ act.dx = (pBuf[0] & 0x10) ? pBuf[1]-256 : pBuf[1];
+ act.dy = (pBuf[0] & 0x20) ? -(pBuf[2]-256) : -pBuf[2];
+ break;
+ }
+ pBufP = 0;
+ return(&act);
+}
OpenPOWER on IntegriCloud