/* * 86Box A hypervisor and IBM PC system emulator that specializes in * running old operating systems and software designed for IBM * PC systems and compatibles from 1981 through fairly recent * system designs based on the PCI bus. * * This file is part of the 86Box distribution. * * Implementation of Serial Mouse devices. * * Based on the 86Box Serial Mouse driver as a framework. * * Version: @(#)mouse_serial.c 1.0.4 2017/06/11 * * Author: Fred N. van Kempen, */ #include #include "ibm.h" #include "timer.h" #include "serial.h" #include "mouse.h" #include "mouse_serial.h" #define SERMOUSE_TYPE_MSYSTEMS 1 /* Mouse Systems */ #define SERMOUSE_TYPE_MICROSOFT 2 /* Microsoft */ #define SERMOUSE_TYPE_LOGITECH 3 /* Logitech */ typedef struct mouse_serial_t { int8_t port, type; int pos, delay; int oldb; SERIAL *serial; } mouse_serial_t; /* Callback from serial driver: RTS was toggled. */ static void sermouse_callback(void *priv) { mouse_serial_t *ms = (mouse_serial_t *)priv; /* Start a timer to wake us up in a little while. */ ms->pos = -1; ms->delay = 5000 * (1 << TIMER_SHIFT); } /* Callback timer expired, now send our "mouse ID" to the serial port. */ static void sermouse_timer(void *priv) { mouse_serial_t *ms = (mouse_serial_t *)priv; ms->delay = 0; switch(ms->type) { case SERMOUSE_TYPE_MICROSOFT: /* This identifies a two-button Microsoft Serial mouse. */ serial_write_fifo(ms->serial, 'M', 1); break; default: /* No action needed. */ break; } } static uint8_t sermouse_poll(int x, int y, int z, int b, void *priv) { mouse_serial_t *ms = (mouse_serial_t *)priv; uint8_t buff[16]; int len; if (!x && !y && b == ms->oldb) return(1); ms->oldb = b; if (ms->type == SERMOUSE_TYPE_MSYSTEMS) y = -y; if (x>127) x = 127; if (y>127) y = 127; if (x<-128) x = -128; if (y<-128) y = -128; len = 0; switch(ms->type) { case SERMOUSE_TYPE_MSYSTEMS: buff[0] = 0x80; buff[0] |= (b&0x01) ? 0x00 : 0x04; /* left button */ buff[0] |= (b&0x02) ? 0x00 : 0x01; /* middle button */ buff[0] |= (b&0x04) ? 0x00 : 0x02; /* right button */ buff[1] = x; buff[2] = y; buff[3] = x; /* same as byte 1 */ buff[4] = y; /* same as byte 2 */ len = 5; break; case SERMOUSE_TYPE_MICROSOFT: buff[0] = 0x40; buff[0] |= (((y>>6)&03)<<2); buff[0] |= ((x>>6)&03); if (b&0x01) buff[0] |= 0x20; if (b&0x02) buff[0] |= 0x10; buff[1] = x & 0x3F; buff[2] = y & 0x3F; len = 3; break; case SERMOUSE_TYPE_LOGITECH: break; } #if 0 pclog("Mouse_Serial(%d): [", ms->type); for (b=0; bserial, buff[b], 1); return(0); } static void sermouse_close(void *priv) { mouse_serial_t *ms = (mouse_serial_t *)priv; /* Detach serial port from the mouse. */ serial_attach(ms->port, NULL, NULL); free(ms); } static void * sermouse_init(int type) { mouse_serial_t *ms = (mouse_serial_t *)malloc(sizeof(mouse_serial_t)); memset(ms, 0x00, sizeof(mouse_serial_t)); ms->port = SERMOUSE_PORT; ms->type = type; /* Attach a serial port to the mouse. */ ms->serial = serial_attach(ms->port, sermouse_callback, ms); timer_add(sermouse_timer, &ms->delay, &ms->delay, ms); return(ms); } static void * sermouse_init_microsoft(void) { return(sermouse_init(SERMOUSE_TYPE_MICROSOFT)); } static void * sermouse_init_msystems(void) { return(sermouse_init(SERMOUSE_TYPE_MSYSTEMS)); } mouse_t mouse_msystems = { "Mouse Systems Mouse (serial)", "mssystems", MOUSE_TYPE_MSYSTEMS, sermouse_init_msystems, sermouse_close, sermouse_poll }; mouse_t mouse_serial_microsoft = { "Microsoft 2-button mouse (serial)", "msserial", MOUSE_TYPE_SERIAL, sermouse_init_microsoft, sermouse_close, sermouse_poll };