## diffname alphapc/mouse.c 1999/0415
## diff -e /dev/null /n/emeliedump/1999/0415/sys/src/brazil/alphapc/mouse.c
0a
#include "u.h"
#include "../port/lib.h"
#include "mem.h"
#include "dat.h"
#include "fns.h"
#include "../port/error.h"
#include "io.h"
#define Image IMAGE
#include <draw.h>
#include <memdraw.h>
#include <cursor.h>
#include "screen.h"
/*
* mouse types
*/
enum
{
Mouseother= 0,
Mouseserial= 1,
MousePS2= 2,
};
static int mousetype;
/*
* setup a serial mouse
*/
static void
serialmouse(int port, char *type, int setspeed)
{
if(mousetype == Mouseserial)
error(Emouseset);
if(port >= 3 || port < 0)
error(Ebadarg);
/* set up /dev/eia? as the mouse */
if(setspeed)
setspeed = 1200;
if(type && *type == 'M')
ns16552special(port, setspeed, 0, 0, m3mouseputc);
else
ns16552special(port, setspeed, 0, 0, mouseputc);
mousetype = Mouseserial;
}
/*
* ps/2 mouse message is three bytes
*
* byte 0 - 0 0 SDY SDX 1 M R L
* byte 1 - DX
* byte 2 - DY
*
* shift & left button is the same as middle button
*/
static void
ps2mouseputc(int c, int shift)
{
static short msg[3];
static int nb;
static uchar b[] = {0, 1, 4, 5, 2, 3, 6, 7, 0, 1, 2, 5, 2, 3, 6, 7 };
int buttons, dx, dy;
/*
* check byte 0 for consistency
*/
if(nb==0 && (c&0xc8)!=0x08)
return;
msg[nb] = c;
if(++nb == 3){
nb = 0;
if(msg[0] & 0x10)
msg[1] |= 0xFF00;
if(msg[0] & 0x20)
msg[2] |= 0xFF00;
buttons = b[(msg[0]&7) | (shift ? 8 : 0)];
dx = msg[1];
dy = -msg[2];
mousetrack(buttons, dx, dy);
}
return;
}
/*
* set up a ps2 mouse
*/
static void
ps2mouse(void)
{
if(mousetype == MousePS2)
return;
i8042auxenable(ps2mouseputc);
/* make mouse streaming, enabled */
i8042auxcmd(0xEA);
i8042auxcmd(0xF4);
mousetype = MousePS2;
}
static int intellimouse;
static int resolution;
static int accelerated;
static void
setaccelerated(int x)
{
accelerated = x;
switch(mousetype){
case MousePS2:
i8042auxcmd(0xE7);
break;
default:
mouseaccelerate(x);
break;
}
}
static void
setlinear(void)
{
accelerated = 0;
switch(mousetype){
case MousePS2:
i8042auxcmd(0xE6);
break;
default:
mouseaccelerate(0);
break;
}
}
static void
setres(int n)
{
resolution = n;
switch(mousetype){
case MousePS2:
i8042auxcmd(0xE8);
i8042auxcmd(n);
break;
}
}
static void
setintellimouse(void)
{
intellimouse = 1;
switch(mousetype){
case MousePS2:
i8042auxcmd(0xF3); /* set sample */
i8042auxcmd(0xC8);
i8042auxcmd(0xF3); /* set sample */
i8042auxcmd(0x64);
i8042auxcmd(0xF3); /* set sample */
i8042auxcmd(0x50);
break;
}
}
static void
resetmouse(void)
{
switch(mousetype){
case MousePS2:
i8042auxcmd(0xF6);
i8042auxcmd(0xEA); /* streaming */
i8042auxcmd(0xE8); /* set resolution */
i8042auxcmd(3);
i8042auxcmd(0xF4); /* enabled */
break;
}
}
void
mousectl(char* field[], int n)
{
if(strncmp(field[0], "serial", 6) == 0){
switch(n){
case 1:
serialmouse(atoi(field[0]+6), 0, 1);
break;
case 2:
serialmouse(atoi(field[1]), 0, 0);
break;
case 3:
default:
serialmouse(atoi(field[1]), field[2], 0);
break;
}
} else if(strcmp(field[0], "ps2") == 0){
ps2mouse();
} else if(strcmp(field[0], "ps2intellimouse") == 0){
ps2mouse();
setintellimouse();
} else if(strcmp(field[0], "accelerated") == 0){
setaccelerated(n == 1 ? 1 : atoi(field[1]));
} else if(strcmp(field[0], "linear") == 0){
setlinear();
} else if(strcmp(field[0], "res") == 0){
if(n >= 2)
n = atoi(field[1]);
setres(n);
} else if(strcmp(field[0], "reset") == 0){
resetmouse();
if(accelerated)
setaccelerated(accelerated);
if(resolution)
setres(resolution);
if(intellimouse)
setintellimouse();
} else if(strcmp(field[0], "intellimouse") == 0){
setintellimouse();
}
else
error(Ebadctl);
}
.
## diffname alphapc/mouse.c 2001/0727
## diff -e /n/emeliedump/1999/0415/sys/src/brazil/alphapc/mouse.c /n/emeliedump/2001/0727/sys/src/9/alphapc/mouse.c
82c
mousetrack(buttons, dx, dy, TK2MS(MACHP(0)->ticks));
.
45a
packetsize = 3;
#else
error("serial mouse not supported yet");
USED(port, type, setspeed);
#endif /* notdef */
.
44c
uartspecial(port, setspeed, 0, 0, mouseputc);
.
42c
uartspecial(port, setspeed, 0, 0, m3mouseputc);
.
31a
#ifdef notdef
.
## diffname alphapc/mouse.c 2001/1121
## diff -e /n/emeliedump/2001/0727/sys/src/9/alphapc/mouse.c /n/emeliedump/2001/1121/sys/src/9/alphapc/mouse.c
224,225d
221,222c
break;
case CMserial:
switch(cb->nf){
case 1:
serialmouse(atoi(cb->f[0]+6), 0, 1);
break;
case 2:
serialmouse(atoi(cb->f[1]), 0, 0);
break;
case 3:
default:
serialmouse(atoi(cb->f[1]), cb->f[2], 0);
break;
}
break;
.
205,213c
break;
case CMres:
if(cb->nf >= 2)
setres(atoi(cb->f[1]));
else
setres(1);
break;
case CMreset:
.
202c
break;
case CMps2intellimouse:
.
187,200c
Cmdtab *ct;
ct = lookupcmd(cb, mousectlmsg, nelem(mousectlmsg));
switch(ct->index){
case CMaccelerated:
setaccelerated(cb->nf == 1 ? 1 : atoi(cb->f[1]));
break;
case CMintellimouse:
setintellimouse();
break;
case CMlinear:
setlinear();
break;
case CMps2:
.
185c
mousectl(Cmdbuf *cb)
.
172a
packetsize = 3;
.
157a
packetsize = 4;
.
110,113d
107a
packetsize = 3;
.
88c
mousetrack(dx, dy, buttons, TK2MS(MACHP(0)->ticks));
.
85a
if(intellimouse && packetsize==4){
if((msg[3]&0xc8) == 0x08){
/* first byte of 3-byte packet */
packetsize = 3;
msg[0] = msg[3];
nb = 1;
/* fall through to emit previous packet */
}else{
/* the AccuPoint on the Toshiba 34[48]0CT encodes extra buttons as 4 and 5 */
/* they repeat and don't release, however, so user-level timing code is required */
if(msg[3] == 0xFF)
buttons |= 1<<3;
if(msg[3] == 0x01)
buttons |= 1<<4;
}
}
.
78c
if(++nb == packetsize){
.
75c
if(intellimouse && (c==0x00 || c==0x01 || c==0xFF)){
/* last byte of 4-byte packet */
packetsize = 4;
return;
}
.
70a
/*
* Resynchronize in stream with timing; see comment above.
*/
m = MACHP(0)->ticks;
if(TK2SEC(m - lasttick) > 2)
nb = 0;
lasttick = m;
.
68c
static uchar b[] = {0, 1, 4, 5, 2, 3, 6, 7, 0, 1, 2, 3, 2, 3, 6, 7 };
static ulong lasttick;
ulong m;
.
66c
static short msg[4];
.
61c
* shift & right button is the same as middle button
*
* Intellimouse and AccuPoint with extra buttons deliver
* byte 3 - 00 or 01 or FF according to extra button state.
* extra buttons are mapped in this code to buttons 4 and 5.
* AccuPoint generates repeated events for these buttons;
* it and Intellimouse generate 'down' events only, so
* user-level code is required to generate button 'up' events
* if they are needed by the application.
* Also on laptops with AccuPoint AND external mouse, the
* controller may deliver 3 or 4 bytes according to the type
* of the external mouse; code must adapt.
*
* On the NEC Versa series (and perhaps others?) we seem to
* lose a byte from the packet every once in a while, which
* means we lose where we are in the instruction stream.
* To resynchronize, if we get a byte more than two seconds
* after the previous byte, we assume it's the first in a packet.
.
25a
enum
{
CMaccelerated,
CMintellimouse,
CMlinear,
CMps2,
CMps2intellimouse,
CMres,
CMreset,
CMserial,
};
static Cmdtab mousectlmsg[] =
{
CMaccelerated, "accelerated", 0,
CMintellimouse, "intellimouse", 1,
CMlinear, "linear", 1,
CMps2, "ps2", 1,
CMps2intellimouse, "ps2intellimouse", 1,
CMres, "res", 0,
CMreset, "reset", 1,
CMserial, "serial", 0,
};
.
24a
static int intellimouse;
static int packetsize;
static int resolution;
static int accelerated;
.
## diffname alphapc/mouse.c 2001/1211
## diff -e /n/emeliedump/2001/1121/sys/src/9/alphapc/mouse.c /n/emeliedump/2001/1211/sys/src/9/alphapc/mouse.c
1,311c
#include "../pc/mouse.c"
.
|