2017-06-21 00:41:34 -04:00
|
|
|
/*
|
2022-11-13 16:37:58 -05:00
|
|
|
* 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.
|
2017-06-21 00:41:34 -04:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* This file is part of the 86Box distribution.
|
2017-06-21 00:41:34 -04:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* Common driver module for MOUSE devices.
|
2017-06-21 00:41:34 -04:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* TODO: Add the Genius bus- and serial mouse.
|
|
|
|
|
* Remove the '3-button' flag from mouse types.
|
2017-06-21 00:41:34 -04:00
|
|
|
*
|
2020-03-25 00:46:02 +02:00
|
|
|
*
|
2017-12-04 11:59:26 -05:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* Authors: Miran Grca, <mgrca8@gmail.com>
|
|
|
|
|
* Fred N. van Kempen, <decwiz@yahoo.com>
|
2017-10-10 03:07:29 -04:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* Copyright 2016-2018 Miran Grca.
|
|
|
|
|
* Copyright 2017-2018 Fred N. van Kempen.
|
2017-06-21 00:41:34 -04:00
|
|
|
*/
|
2023-08-07 18:49:58 +02:00
|
|
|
#include <math.h>
|
2018-05-21 19:04:05 +02:00
|
|
|
#include <stdarg.h>
|
2023-08-07 18:49:58 +02:00
|
|
|
#include <stdatomic.h>
|
2017-09-25 04:31:20 -04:00
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <wchar.h>
|
2018-05-21 19:04:05 +02:00
|
|
|
#define HAVE_STDARG_H
|
2020-03-29 14:24:42 +02:00
|
|
|
#include <86box/86box.h>
|
|
|
|
|
#include <86box/device.h>
|
2023-03-30 01:39:00 +02:00
|
|
|
#include <86box/timer.h>
|
|
|
|
|
#include <86box/gdbstub.h>
|
2020-03-29 14:24:42 +02:00
|
|
|
#include <86box/mouse.h>
|
2023-08-11 04:45:32 +02:00
|
|
|
#include <86box/video.h>
|
|
|
|
|
#include <86box/plat.h>
|
2023-06-26 12:47:04 -04:00
|
|
|
#include <86box/plat_unused.h>
|
2017-12-04 11:59:26 -05:00
|
|
|
|
2023-06-26 22:31:03 -04:00
|
|
|
typedef struct mouse_t {
|
2022-09-18 17:13:28 -04:00
|
|
|
const device_t *device;
|
2017-12-04 11:59:26 -05:00
|
|
|
} mouse_t;
|
2017-06-21 00:41:34 -04:00
|
|
|
|
2022-09-18 17:13:28 -04:00
|
|
|
int mouse_type = 0;
|
2023-10-23 01:21:28 +02:00
|
|
|
int mouse_input_mode;
|
2023-08-07 18:49:58 +02:00
|
|
|
int mouse_timed = 1;
|
2023-05-11 03:02:36 -04:00
|
|
|
int mouse_tablet_in_proximity = 0;
|
|
|
|
|
int tablet_tool_type = 1; /* 0 = Puck/Cursor, 1 = Pen */
|
2023-01-03 15:42:57 +06:00
|
|
|
|
2023-05-11 03:02:36 -04:00
|
|
|
double mouse_x_abs;
|
|
|
|
|
double mouse_y_abs;
|
2017-10-23 05:20:18 -04:00
|
|
|
|
2023-08-07 18:49:58 +02:00
|
|
|
double mouse_sensitivity = 1.0;
|
|
|
|
|
|
2023-03-30 01:39:00 +02:00
|
|
|
pc_timer_t mouse_timer; /* mouse event timer */
|
|
|
|
|
|
2018-03-19 01:02:04 +01:00
|
|
|
static const device_t mouse_none_device = {
|
2022-09-18 17:13:28 -04:00
|
|
|
.name = "None",
|
2022-03-13 09:28:28 -04:00
|
|
|
.internal_name = "none",
|
2022-09-18 17:13:28 -04:00
|
|
|
.flags = 0,
|
|
|
|
|
.local = MOUSE_TYPE_NONE,
|
|
|
|
|
.init = NULL,
|
|
|
|
|
.close = NULL,
|
|
|
|
|
.reset = NULL,
|
2022-03-13 09:28:28 -04:00
|
|
|
{ .poll = NULL },
|
|
|
|
|
.speed_changed = NULL,
|
2022-09-18 17:13:28 -04:00
|
|
|
.force_redraw = NULL,
|
|
|
|
|
.config = NULL
|
2017-06-21 00:41:34 -04:00
|
|
|
};
|
2022-03-13 09:28:28 -04:00
|
|
|
|
2018-03-19 01:02:04 +01:00
|
|
|
static const device_t mouse_internal_device = {
|
2022-09-18 17:13:28 -04:00
|
|
|
.name = "Internal",
|
2022-03-13 09:28:28 -04:00
|
|
|
.internal_name = "internal",
|
2022-09-18 17:13:28 -04:00
|
|
|
.flags = 0,
|
|
|
|
|
.local = MOUSE_TYPE_INTERNAL,
|
|
|
|
|
.init = NULL,
|
|
|
|
|
.close = NULL,
|
|
|
|
|
.reset = NULL,
|
2022-03-13 09:28:28 -04:00
|
|
|
{ .poll = NULL },
|
|
|
|
|
.speed_changed = NULL,
|
2022-09-18 17:13:28 -04:00
|
|
|
.force_redraw = NULL,
|
|
|
|
|
.config = NULL
|
2017-11-05 01:57:04 -05:00
|
|
|
};
|
|
|
|
|
|
2017-12-04 11:59:26 -05:00
|
|
|
static mouse_t mouse_devices[] = {
|
2022-09-18 17:13:28 -04:00
|
|
|
// clang-format off
|
2023-03-04 11:35:10 +06:00
|
|
|
{ &mouse_none_device },
|
|
|
|
|
{ &mouse_internal_device },
|
|
|
|
|
{ &mouse_logibus_device },
|
|
|
|
|
{ &mouse_msinport_device },
|
2023-08-11 22:29:53 +02:00
|
|
|
#ifdef USE_GENIBUS
|
2023-03-04 11:35:10 +06:00
|
|
|
{ &mouse_genibus_device },
|
2017-05-05 01:49:42 +02:00
|
|
|
#endif
|
2023-03-04 11:35:10 +06:00
|
|
|
{ &mouse_mssystems_device },
|
|
|
|
|
{ &mouse_msserial_device },
|
|
|
|
|
{ &mouse_ltserial_device },
|
|
|
|
|
{ &mouse_ps2_device },
|
2023-08-24 23:49:06 +02:00
|
|
|
#ifdef USE_WACOM
|
2023-03-04 11:35:10 +06:00
|
|
|
{ &mouse_wacom_device },
|
|
|
|
|
{ &mouse_wacom_artpad_device },
|
2023-08-24 23:49:06 +02:00
|
|
|
#endif
|
2023-03-04 11:35:10 +06:00
|
|
|
{ NULL }
|
2022-09-18 17:13:28 -04:00
|
|
|
// clang-format on
|
2016-12-23 03:16:24 +01:00
|
|
|
};
|
2017-12-04 11:59:26 -05:00
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
static _Atomic double mouse_x;
|
|
|
|
|
static _Atomic double mouse_y;
|
|
|
|
|
static atomic_int mouse_z;
|
|
|
|
|
static atomic_int mouse_buttons;
|
|
|
|
|
|
|
|
|
|
static int mouse_delta_b;
|
|
|
|
|
static int mouse_old_b;
|
|
|
|
|
|
2022-09-18 17:13:28 -04:00
|
|
|
static const device_t *mouse_curr;
|
|
|
|
|
static void *mouse_priv;
|
|
|
|
|
static int mouse_nbut;
|
2023-08-12 00:17:01 +02:00
|
|
|
static int mouse_raw;
|
2023-08-11 22:29:53 +02:00
|
|
|
static int (*mouse_dev_poll)(void *priv);
|
2022-12-30 06:05:57 +01:00
|
|
|
static void (*mouse_poll_ex)(void) = NULL;
|
2017-05-05 01:49:42 +02:00
|
|
|
|
2023-04-26 01:42:23 +02:00
|
|
|
static double sample_rate = 200.0;
|
|
|
|
|
|
2018-05-21 19:04:05 +02:00
|
|
|
#ifdef ENABLE_MOUSE_LOG
|
|
|
|
|
int mouse_do_log = ENABLE_MOUSE_LOG;
|
|
|
|
|
|
|
|
|
|
static void
|
2018-10-19 00:39:32 +02:00
|
|
|
mouse_log(const char *fmt, ...)
|
2018-05-21 19:04:05 +02:00
|
|
|
{
|
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
|
|
if (mouse_do_log) {
|
2022-09-18 17:13:28 -04:00
|
|
|
va_start(ap, fmt);
|
|
|
|
|
pclog_ex(fmt, ap);
|
|
|
|
|
va_end(ap);
|
2018-05-21 19:04:05 +02:00
|
|
|
}
|
|
|
|
|
}
|
2018-10-19 00:39:32 +02:00
|
|
|
#else
|
2022-09-18 17:13:28 -04:00
|
|
|
# define mouse_log(fmt, ...)
|
2018-10-19 00:39:32 +02:00
|
|
|
#endif
|
2018-05-21 19:04:05 +02:00
|
|
|
|
2023-08-11 04:45:32 +02:00
|
|
|
void
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_clear_x(void)
|
2023-08-11 04:45:32 +02:00
|
|
|
{
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_x = 0.0;
|
2023-08-11 04:45:32 +02:00
|
|
|
}
|
|
|
|
|
|
2017-05-06 17:48:33 +02:00
|
|
|
void
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_clear_y(void)
|
2016-12-23 03:16:24 +01:00
|
|
|
{
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_y = 0.0;
|
2016-12-23 03:16:24 +01:00
|
|
|
}
|
|
|
|
|
|
2017-05-06 17:48:33 +02:00
|
|
|
void
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_clear_coords(void)
|
2016-12-23 03:16:24 +01:00
|
|
|
{
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_clear_x();
|
|
|
|
|
mouse_clear_y();
|
2017-06-21 00:41:34 -04:00
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_z = 0;
|
2023-03-30 01:39:00 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-13 18:25:56 +02:00
|
|
|
void
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_clear_buttons(void)
|
2023-08-11 04:45:32 +02:00
|
|
|
{
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_buttons = 0x00;
|
|
|
|
|
mouse_old_b = 0x00;
|
2023-08-11 04:45:32 +02:00
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_delta_b = 0x00;
|
2023-08-11 04:45:32 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-12 15:59:29 +02:00
|
|
|
static double
|
|
|
|
|
mouse_scale_coord_x(double x, int mul)
|
|
|
|
|
{
|
2023-08-12 17:56:44 +02:00
|
|
|
double ratio = 1.0;
|
|
|
|
|
|
|
|
|
|
if (!mouse_raw)
|
2023-08-13 17:25:45 +02:00
|
|
|
ratio = ((double) monitors[0].mon_unscaled_size_x) / monitors[0].mon_res_x;
|
2023-08-12 15:59:29 +02:00
|
|
|
|
|
|
|
|
if (mul)
|
|
|
|
|
x *= ratio;
|
|
|
|
|
else
|
|
|
|
|
x /= ratio;
|
|
|
|
|
|
|
|
|
|
return x;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static double
|
|
|
|
|
mouse_scale_coord_y(double y, int mul)
|
|
|
|
|
{
|
2023-08-12 17:56:44 +02:00
|
|
|
double ratio = 1.0;
|
|
|
|
|
|
|
|
|
|
if (!mouse_raw)
|
2023-08-13 17:25:45 +02:00
|
|
|
ratio = ((double) monitors[0].mon_efscrnsz_y) / monitors[0].mon_res_y;
|
2023-08-12 15:59:29 +02:00
|
|
|
|
|
|
|
|
if (mul)
|
|
|
|
|
y *= ratio;
|
|
|
|
|
else
|
|
|
|
|
y /= ratio;
|
|
|
|
|
|
|
|
|
|
return y;
|
|
|
|
|
}
|
2023-08-13 17:25:45 +02:00
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
void
|
|
|
|
|
mouse_subtract_x(int *delta_x, int *o_x, int min, int max, int abs)
|
2023-08-11 04:45:32 +02:00
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
double real_x = atomic_load(&mouse_x);
|
2023-08-11 22:29:53 +02:00
|
|
|
double smax_x;
|
|
|
|
|
double rsmin_x;
|
|
|
|
|
double smin_x;
|
2023-08-13 17:25:45 +02:00
|
|
|
int ds_x;
|
|
|
|
|
int scaled_x;
|
2023-08-11 22:29:53 +02:00
|
|
|
|
2023-08-12 15:59:29 +02:00
|
|
|
rsmin_x = mouse_scale_coord_x(min, 0);
|
2023-08-11 22:29:53 +02:00
|
|
|
if (abs) {
|
2023-08-12 15:59:29 +02:00
|
|
|
smax_x = mouse_scale_coord_x(max, 0) + ABS(rsmin_x);
|
2023-08-13 18:03:12 +02:00
|
|
|
max += ABSD(min);
|
2023-08-11 22:29:53 +02:00
|
|
|
real_x += rsmin_x;
|
|
|
|
|
smin_x = 0;
|
|
|
|
|
} else {
|
2023-08-12 15:59:29 +02:00
|
|
|
smax_x = mouse_scale_coord_x(max, 0);
|
2023-08-11 22:29:53 +02:00
|
|
|
smin_x = rsmin_x;
|
|
|
|
|
}
|
2023-08-11 04:45:32 +02:00
|
|
|
|
2023-08-13 17:25:45 +02:00
|
|
|
smax_x = floor(smax_x);
|
|
|
|
|
smin_x = ceil(smin_x);
|
|
|
|
|
|
|
|
|
|
/* Default the X overflow to 1. */
|
2023-08-11 22:29:53 +02:00
|
|
|
if (o_x != NULL)
|
|
|
|
|
*o_x = 1;
|
2023-08-11 04:45:32 +02:00
|
|
|
|
2023-08-13 17:25:45 +02:00
|
|
|
ds_x = mouse_scale_coord_x(real_x, 1);
|
|
|
|
|
|
|
|
|
|
if (ds_x >= 0.0)
|
|
|
|
|
scaled_x = (int) floor(mouse_scale_coord_x(real_x, 1));
|
|
|
|
|
else
|
|
|
|
|
scaled_x = (int) ceil(mouse_scale_coord_x(real_x, 1));
|
|
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
if (real_x > smax_x) {
|
2023-08-13 17:25:45 +02:00
|
|
|
if (abs) {
|
|
|
|
|
*delta_x = scaled_x;
|
|
|
|
|
real_x -= mouse_scale_coord_x((double) scaled_x, 0);
|
|
|
|
|
} else {
|
2023-08-12 15:59:29 +02:00
|
|
|
*delta_x = max;
|
2023-08-13 17:25:45 +02:00
|
|
|
real_x -= smax_x;
|
|
|
|
|
}
|
2023-08-11 22:29:53 +02:00
|
|
|
} else if (real_x < smin_x) {
|
2023-08-13 17:25:45 +02:00
|
|
|
if (abs) {
|
|
|
|
|
*delta_x = scaled_x;
|
|
|
|
|
real_x -= mouse_scale_coord_x((double) scaled_x, 0);
|
|
|
|
|
} else {
|
2023-08-12 15:59:29 +02:00
|
|
|
*delta_x = min;
|
2023-08-13 18:03:12 +02:00
|
|
|
real_x += ABSD(smin_x);
|
2023-08-13 17:25:45 +02:00
|
|
|
}
|
2023-08-11 22:29:53 +02:00
|
|
|
} else {
|
2023-08-13 17:25:45 +02:00
|
|
|
*delta_x = scaled_x;
|
|
|
|
|
real_x -= mouse_scale_coord_x((double) scaled_x, 0);
|
2023-08-11 22:29:53 +02:00
|
|
|
if (o_x != NULL)
|
|
|
|
|
*o_x = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (abs)
|
|
|
|
|
real_x -= rsmin_x;
|
|
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
atomic_store(&mouse_x, real_x);
|
2023-08-11 04:45:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* It appears all host platforms give us y in the Microsoft format
|
|
|
|
|
(positive to the south), so for all non-Microsoft report formsts,
|
2023-08-11 22:29:53 +02:00
|
|
|
we have to invert that. */
|
2023-08-11 04:45:32 +02:00
|
|
|
void
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_subtract_y(int *delta_y, int *o_y, int min, int max, int invert, int abs)
|
2023-08-11 04:45:32 +02:00
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
double real_y = atomic_load(&mouse_y);
|
2023-08-11 22:29:53 +02:00
|
|
|
double smax_y;
|
|
|
|
|
double rsmin_y;
|
|
|
|
|
double smin_y;
|
2023-08-13 17:25:45 +02:00
|
|
|
int ds_y;
|
|
|
|
|
int scaled_y;
|
2023-08-11 04:45:32 +02:00
|
|
|
|
|
|
|
|
if (invert)
|
|
|
|
|
real_y = -real_y;
|
|
|
|
|
|
2023-08-12 15:59:29 +02:00
|
|
|
rsmin_y = mouse_scale_coord_y(min, 0);
|
2023-08-11 04:45:32 +02:00
|
|
|
if (abs) {
|
2023-08-12 15:59:29 +02:00
|
|
|
smax_y = mouse_scale_coord_y(max, 0) + ABS(rsmin_y);
|
2023-08-13 18:03:12 +02:00
|
|
|
max += ABSD(min);
|
2023-08-11 04:45:32 +02:00
|
|
|
real_y += rsmin_y;
|
|
|
|
|
smin_y = 0;
|
|
|
|
|
} else {
|
2023-08-12 15:59:29 +02:00
|
|
|
smax_y = mouse_scale_coord_y(max, 0);
|
2023-08-11 04:45:32 +02:00
|
|
|
smin_y = rsmin_y;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-13 17:25:45 +02:00
|
|
|
smax_y = floor(smax_y);
|
|
|
|
|
smin_y = ceil(smin_y);
|
|
|
|
|
|
|
|
|
|
/* Default Y overflow to 1. */
|
2023-08-11 04:45:32 +02:00
|
|
|
if (o_y != NULL)
|
|
|
|
|
*o_y = 1;
|
|
|
|
|
|
2023-08-13 17:25:45 +02:00
|
|
|
ds_y = mouse_scale_coord_x(real_y, 1);
|
|
|
|
|
|
|
|
|
|
if (ds_y >= 0.0)
|
|
|
|
|
scaled_y = (int) floor(mouse_scale_coord_x(real_y, 1));
|
|
|
|
|
else
|
|
|
|
|
scaled_y = (int) ceil(mouse_scale_coord_x(real_y, 1));
|
|
|
|
|
|
2023-08-11 04:45:32 +02:00
|
|
|
if (real_y > smax_y) {
|
2023-08-13 17:25:45 +02:00
|
|
|
if (abs) {
|
|
|
|
|
*delta_y = scaled_y;
|
|
|
|
|
real_y -= mouse_scale_coord_y((double) scaled_y, 0);
|
|
|
|
|
} else {
|
2023-08-12 15:59:29 +02:00
|
|
|
*delta_y = max;
|
2023-08-13 17:25:45 +02:00
|
|
|
real_y -= smax_y;
|
|
|
|
|
}
|
2023-08-11 04:45:32 +02:00
|
|
|
} else if (real_y < smin_y) {
|
2023-08-13 17:25:45 +02:00
|
|
|
if (abs) {
|
|
|
|
|
*delta_y = scaled_y;
|
|
|
|
|
real_y -= mouse_scale_coord_y((double) scaled_y, 0);
|
|
|
|
|
} else {
|
2023-08-12 15:59:29 +02:00
|
|
|
*delta_y = min;
|
2023-08-13 18:03:12 +02:00
|
|
|
real_y += ABSD(smin_y);
|
2023-08-13 17:25:45 +02:00
|
|
|
}
|
2023-08-11 04:45:32 +02:00
|
|
|
} else {
|
2023-08-13 17:25:45 +02:00
|
|
|
*delta_y = scaled_y;
|
|
|
|
|
real_y -= mouse_scale_coord_y((double) scaled_y, 0);
|
2023-08-11 04:45:32 +02:00
|
|
|
if (o_y != NULL)
|
|
|
|
|
*o_y = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
if (abs)
|
2023-08-11 04:45:32 +02:00
|
|
|
real_y -= rsmin_y;
|
|
|
|
|
|
|
|
|
|
if (invert)
|
|
|
|
|
real_y = -real_y;
|
|
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
atomic_store(&mouse_y, real_y);
|
2023-08-11 04:45:32 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
/* It appears all host platforms give us y in the Microsoft format
|
|
|
|
|
(positive to the south), so for all non-Microsoft report formsts,
|
|
|
|
|
we have to invenrt that. */
|
|
|
|
|
void
|
|
|
|
|
mouse_subtract_coords(int *delta_x, int *delta_y, int *o_x, int *o_y,
|
|
|
|
|
int min, int max, int invert, int abs)
|
|
|
|
|
{
|
|
|
|
|
mouse_subtract_x(delta_x, o_x, min, max, abs);
|
|
|
|
|
mouse_subtract_y(delta_y, o_y, min, max, invert, abs);
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-26 23:16:50 +02:00
|
|
|
int
|
|
|
|
|
mouse_wheel_moved(void)
|
|
|
|
|
{
|
|
|
|
|
int ret = !!(atomic_load(&mouse_z));
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
int
|
|
|
|
|
mouse_moved(void)
|
|
|
|
|
{
|
2023-08-13 18:03:12 +02:00
|
|
|
int moved_x = !!((int) floor(ABSD(mouse_scale_coord_x(atomic_load(&mouse_x), 1))));
|
|
|
|
|
int moved_y = !!((int) floor(ABSD(mouse_scale_coord_y(atomic_load(&mouse_y), 1))));
|
2023-08-13 17:59:19 +02:00
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
/* Convert them to integer so we treat < 1.0 and > -1.0 as 0. */
|
2023-08-13 18:03:12 +02:00
|
|
|
int ret = (moved_x || moved_y);
|
2023-08-11 22:29:53 +02:00
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
mouse_state_changed(void)
|
|
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
int b;
|
2023-08-11 22:29:53 +02:00
|
|
|
int b_mask = (1 << mouse_nbut) - 1;
|
|
|
|
|
int wheel = (mouse_nbut >= 4);
|
|
|
|
|
int ret;
|
|
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
b = atomic_load(&mouse_buttons);
|
|
|
|
|
mouse_delta_b = (b ^ mouse_old_b);
|
|
|
|
|
mouse_old_b = b;
|
2023-08-11 22:29:53 +02:00
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
ret = mouse_moved() || ((atomic_load(&mouse_z) != 0) && wheel) || (mouse_delta_b & b_mask);
|
2023-08-11 22:29:53 +02:00
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
mouse_mbut_changed(void)
|
|
|
|
|
{
|
|
|
|
|
return !!(mouse_delta_b & 0x04);
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-30 01:39:00 +02:00
|
|
|
static void
|
2023-06-26 12:47:04 -04:00
|
|
|
mouse_timer_poll(UNUSED(void *priv))
|
2023-03-30 01:39:00 +02:00
|
|
|
{
|
2023-08-07 18:49:58 +02:00
|
|
|
/* Poll at the specified sample rate. */
|
2023-04-26 01:42:23 +02:00
|
|
|
timer_on_auto(&mouse_timer, 1000000.0 / sample_rate);
|
2023-03-30 01:39:00 +02:00
|
|
|
|
|
|
|
|
#ifdef USE_GDBSTUB /* avoid a KBC FIFO overflow when CPU emulation is stalled */
|
2023-08-07 18:49:58 +02:00
|
|
|
if (gdbstub_step == GDBSTUB_EXEC) {
|
2023-03-30 01:39:00 +02:00
|
|
|
#endif
|
2023-08-07 18:49:58 +02:00
|
|
|
if (mouse_timed)
|
|
|
|
|
mouse_process();
|
|
|
|
|
#ifdef USE_GDBSTUB /* avoid a KBC FIFO overflow when CPU emulation is stalled */
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
static void
|
|
|
|
|
atomic_double_add(_Atomic double *var, double val)
|
2023-08-07 18:49:58 +02:00
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
double temp = atomic_load(var);
|
|
|
|
|
|
|
|
|
|
temp += val;
|
2023-08-07 18:49:58 +02:00
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
atomic_store(var, temp);
|
2023-08-07 18:49:58 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-12 18:48:25 +02:00
|
|
|
void
|
|
|
|
|
mouse_scale_fx(double x)
|
|
|
|
|
{
|
|
|
|
|
atomic_double_add(&mouse_x, ((double) x) * mouse_sensitivity);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
mouse_scale_fy(double y)
|
|
|
|
|
{
|
|
|
|
|
atomic_double_add(&mouse_y, ((double) y) * mouse_sensitivity);
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-07 18:49:58 +02:00
|
|
|
void
|
|
|
|
|
mouse_scale_x(int x)
|
|
|
|
|
{
|
2023-08-12 15:59:29 +02:00
|
|
|
atomic_double_add(&mouse_x, ((double) x) * mouse_sensitivity);
|
2023-08-07 18:49:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
mouse_scale_y(int y)
|
|
|
|
|
{
|
2023-08-12 15:59:29 +02:00
|
|
|
atomic_double_add(&mouse_y, ((double) y) * mouse_sensitivity);
|
2023-08-12 00:17:01 +02:00
|
|
|
}
|
2023-08-07 18:49:58 +02:00
|
|
|
|
2023-08-12 18:48:25 +02:00
|
|
|
void
|
|
|
|
|
mouse_scalef(double x, double y)
|
|
|
|
|
{
|
|
|
|
|
mouse_scale_fx(x);
|
|
|
|
|
mouse_scale_fy(y);
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
void
|
|
|
|
|
mouse_scale(int x, int y)
|
|
|
|
|
{
|
|
|
|
|
mouse_scale_x(x);
|
|
|
|
|
mouse_scale_y(y);
|
2023-08-07 18:49:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
mouse_set_z(int z)
|
|
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
atomic_fetch_add(&mouse_z, z);
|
2023-08-07 18:49:58 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
void
|
|
|
|
|
mouse_clear_z(void)
|
|
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
atomic_store(&mouse_z, 0);
|
2023-08-11 22:29:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
mouse_subtract_z(int *delta_z, int min, int max, int invert)
|
|
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
int z = atomic_load(&mouse_z);
|
|
|
|
|
int real_z = invert ? -z : z;
|
2023-08-11 22:29:53 +02:00
|
|
|
|
|
|
|
|
if (mouse_z > max) {
|
|
|
|
|
*delta_z = max;
|
|
|
|
|
real_z -= max;
|
|
|
|
|
} else if (mouse_z < min) {
|
|
|
|
|
*delta_z = min;
|
|
|
|
|
real_z += ABS(min);
|
|
|
|
|
} else {
|
2023-08-26 22:47:32 +02:00
|
|
|
*delta_z = real_z;
|
2023-08-11 22:29:53 +02:00
|
|
|
real_z = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
atomic_store(&mouse_z, invert ? -real_z : real_z);
|
2023-08-11 22:29:53 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-07 18:49:58 +02:00
|
|
|
void
|
|
|
|
|
mouse_set_buttons_ex(int b)
|
|
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
atomic_store(&mouse_buttons, b);
|
2023-08-07 18:49:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
mouse_get_buttons_ex(void)
|
|
|
|
|
{
|
2023-08-12 00:17:01 +02:00
|
|
|
return atomic_load(&mouse_buttons);
|
2016-12-23 03:16:24 +01:00
|
|
|
}
|
|
|
|
|
|
2023-04-26 01:42:23 +02:00
|
|
|
void
|
|
|
|
|
mouse_set_sample_rate(double new_rate)
|
|
|
|
|
{
|
2023-08-07 18:49:58 +02:00
|
|
|
mouse_timed = (new_rate > 0.0);
|
|
|
|
|
|
2023-04-26 01:42:23 +02:00
|
|
|
timer_stop(&mouse_timer);
|
|
|
|
|
|
|
|
|
|
sample_rate = new_rate;
|
2023-08-07 18:49:58 +02:00
|
|
|
if (mouse_timed)
|
|
|
|
|
timer_on_auto(&mouse_timer, 1000000.0 / sample_rate);
|
2023-04-26 01:42:23 +02:00
|
|
|
}
|
|
|
|
|
|
2017-12-10 02:08:37 -05:00
|
|
|
/* Callback from the hardware driver. */
|
|
|
|
|
void
|
|
|
|
|
mouse_set_buttons(int buttons)
|
|
|
|
|
{
|
|
|
|
|
mouse_nbut = buttons;
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-30 06:05:57 +01:00
|
|
|
void
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_get_abs_coords(double *x_abs, double *y_abs)
|
2022-12-30 06:05:57 +01:00
|
|
|
{
|
2023-08-12 01:15:06 +02:00
|
|
|
*x_abs = mouse_x_abs;
|
|
|
|
|
*y_abs = mouse_y_abs;
|
2022-12-30 06:05:57 +01:00
|
|
|
}
|
|
|
|
|
|
2017-10-19 04:27:04 -04:00
|
|
|
void
|
|
|
|
|
mouse_process(void)
|
|
|
|
|
{
|
Added the IBM 5161 ISA expansion for PC and XT;
Cleaned up the parallel port emulation, added IRQ support, and made enabling/disabling per port;
Added the Award 430NX and the Intel Classic/PCI (Alfredo, 420TX);
Finished the 586MC1;
Added 8087 emulation;
Moved Cyrix 6x86'es to the Dev branch;
Sanitized/cleaned up memregs.c/h and intel.c/h;
Split the chipsets from machines and sanitized Port 92 emulation;
Added support for the 15bpp mode to the Compaq ATI 28800;
Moved the MR 386DX and 486 machines to the Dev branch;
Ported the new dynamic recompiler from PCem, but it remains in Dev branch until after v2.00;
Ported the new timer code from PCem;
Cleaned up the CPU table of unused stuff and better optimized its structure;
Ported the Open-XT and Open-AT from VARCem, the Open-AT is in the Dev branch;
Ported the XT MFM controller rewrite and adding of more controllers (incl. two RLL ones), from VARCem;
Added the AHA-1540A and the BusTek BT-542B;
Moved the Sumo SCSI-AT to the Dev branch;
Minor IDE, FDC, and floppy drive code clean-ups;
Made NCR 5380/53C400-based cards' BIOS address configurable;
Got rid of the legacy romset variable;
Unified (video) buffer and buffer32 into one and make the unified buffer 32-bit;
Added the Amstead PPC512 per PCem patch by John Elliott;
Switched memory mapping granularity from 16k to 4k (less than 1k not possible due to internal pages);
Rewrote the CL-GD 54xx blitter, fixes Win-OS/2 on the 54x6 among other thing;
Added the Image Manager 1024 and Professional Graphics Controller per PCem patch by John Elliott and work done on VARCem;
Added Headland HT-216, GC-205 and Video 7 VGA 1024i emulation based on PCem commit;
Implemented the fuction keys for the Toshiba T1000/T1200/T3100 enhancement;
Amstrad MegaPC does now works correctly with non-internal graphics card;
The SLiRP code no longer casts a packed struct type to a non-packed struct type;
The Xi8088 and PB410a no longer hang on 86Box when PS/2 mouse is not present;
The S3 Virge on BeOS is no longer broken (was broken by build #1591);
OS/2 2.0 build 6.167 now sees key presses again;
Xi8088 now work on CGA again;
86F images converted from either the old or new variants of the HxC MFM format now work correctly;
Hardware interrupts with a vector of 0xFF are now handled correctly;
OPTi 495SX boards no longer incorrectly have 64 MB maximum RAM when 32 MB is correct;
Fixed VNC keyboard input bugs;
Fixed AT RTC periodic interrupt - Chicago 58s / 73f / 73g / 81 MIDI play no longer hangs with the build's own VTD driver;
Fixed mouse polling with internal mice - Amstrad and Olivetti mice now work correctly;
Triones ATAPI DMA driver now correctly reads a file at the end of a CD image with a sectors number not divisible by 4;
Compaq Portable now works with all graphics cards;
Fixed various MDSI Genius bugs;
Added segment limit checks and improved page fault checks for several CPU instructions - Memphis 15xx WINSETUP and Chicago 58s WINDISK.CPL no longer issue a GPF, and some S3 drivers that used to have glitches, now work correctly;
Further improved the 808x emulation, also fixes the noticably choppy sound when using 808x CPU's, also fixes #355;
OS/2 installer no logner locks up on splash screen on PS/2 Model 70 and 80, fixes #400.
Fixed several Amstead bugs, GEM no longer crashes on the Amstrad 1640, fixes #391.
Ported John Elliott's Amstrad fixes and improvement from PCem, and fixed the default language so it's correctly Engliish, fixes #278, fixes #389.
Fixed a minor IDE timing bug, fixes #388.
Fixed Toshiba T1000 RAM issues, fixes #379.
Fixed EGA/(S)VGA overscan border handling, fixes #378;
Got rid of the now long useless IDE channel 2 auto-removal, fixes #370;
Fixed the BIOS files used by the AMSTRAD PC1512, fixes #366;
Ported the Unicode CD image file name fix from VARCem, fixes #365;
Fixed high density floppy disks on the Xi8088, fixes #359;
Fixed some bugs in the Hercules emulation, fixes #346, fixes #358;
Fixed the SCSI hard disk mode sense pages, fixes #356;
Removed the AMI Unknown 386SX because of impossibility to identify the chipset, closes #349;
Fixed bugs in the serial mouse emulation, fixes #344;
Compiled 86Box binaries now include all the required .DLL's, fixes #341;
Made some combo boxes in the Settings dialog slightly wider, fixes #276.
2019-09-20 14:02:30 +02:00
|
|
|
if (mouse_curr == NULL)
|
2022-09-18 17:13:28 -04:00
|
|
|
return;
|
2017-12-04 11:59:26 -05:00
|
|
|
|
2023-10-23 01:21:28 +02:00
|
|
|
if ((mouse_input_mode >= 1) && mouse_poll_ex)
|
2022-12-30 06:05:57 +01:00
|
|
|
mouse_poll_ex();
|
2023-10-23 01:21:28 +02:00
|
|
|
else if ((mouse_input_mode == 0) && ((mouse_dev_poll != NULL) || (mouse_curr->poll != NULL))) {
|
2022-09-18 17:13:28 -04:00
|
|
|
if (mouse_curr->poll != NULL)
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_curr->poll(mouse_priv);
|
2022-09-18 17:13:28 -04:00
|
|
|
else
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_dev_poll(mouse_priv);
|
2017-12-04 11:59:26 -05:00
|
|
|
}
|
2017-10-19 04:27:04 -04:00
|
|
|
}
|
|
|
|
|
|
2017-11-05 01:57:04 -05:00
|
|
|
void
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_set_poll_ex(void (*poll_ex)(void))
|
|
|
|
|
{
|
|
|
|
|
mouse_poll_ex = poll_ex;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
mouse_set_poll(int (*func)(void *), void *arg)
|
2017-11-05 01:57:04 -05:00
|
|
|
{
|
2022-09-18 17:13:28 -04:00
|
|
|
if (mouse_type != MOUSE_TYPE_INTERNAL)
|
|
|
|
|
return;
|
2017-11-05 01:57:04 -05:00
|
|
|
|
2018-03-18 20:48:10 +01:00
|
|
|
mouse_dev_poll = func;
|
2022-09-18 17:13:28 -04:00
|
|
|
mouse_priv = arg;
|
2017-11-05 01:57:04 -05:00
|
|
|
}
|
|
|
|
|
|
2023-08-21 20:22:39 -04:00
|
|
|
const char *
|
2017-05-06 17:48:33 +02:00
|
|
|
mouse_get_name(int mouse)
|
2016-12-23 03:16:24 +01:00
|
|
|
{
|
2023-08-21 20:22:39 -04:00
|
|
|
return (mouse_devices[mouse].device->name);
|
2016-12-23 03:16:24 +01:00
|
|
|
}
|
2017-05-05 01:49:42 +02:00
|
|
|
|
2023-08-21 20:22:39 -04:00
|
|
|
const char *
|
2017-05-06 17:48:33 +02:00
|
|
|
mouse_get_internal_name(int mouse)
|
2017-05-05 01:49:42 +02:00
|
|
|
{
|
2022-02-03 03:10:06 +01:00
|
|
|
return device_get_internal_name(mouse_devices[mouse].device);
|
2017-05-05 01:49:42 +02:00
|
|
|
}
|
|
|
|
|
|
2017-05-06 17:48:33 +02:00
|
|
|
int
|
|
|
|
|
mouse_get_from_internal_name(char *s)
|
2017-05-05 01:49:42 +02:00
|
|
|
{
|
|
|
|
|
int c = 0;
|
|
|
|
|
|
2022-02-03 03:10:06 +01:00
|
|
|
while (mouse_devices[c].device != NULL) {
|
2022-09-18 17:13:28 -04:00
|
|
|
if (!strcmp((char *) mouse_devices[c].device->internal_name, s))
|
2023-05-11 03:02:36 -04:00
|
|
|
return c;
|
2022-09-18 17:13:28 -04:00
|
|
|
c++;
|
2017-05-05 01:49:42 +02:00
|
|
|
}
|
|
|
|
|
|
2023-05-11 03:02:36 -04:00
|
|
|
return 0;
|
2017-05-05 01:49:42 +02:00
|
|
|
}
|
|
|
|
|
|
2018-01-26 13:35:50 +01:00
|
|
|
int
|
|
|
|
|
mouse_has_config(int mouse)
|
|
|
|
|
{
|
2022-09-18 17:13:28 -04:00
|
|
|
if (mouse_devices[mouse].device == NULL)
|
2023-05-11 03:02:36 -04:00
|
|
|
return 0;
|
2018-01-26 13:35:50 +01:00
|
|
|
|
2022-09-18 17:13:28 -04:00
|
|
|
return (mouse_devices[mouse].device->config ? 1 : 0);
|
2018-01-26 13:35:50 +01:00
|
|
|
}
|
|
|
|
|
|
2018-03-19 01:02:04 +01:00
|
|
|
const device_t *
|
2017-12-10 02:08:37 -05:00
|
|
|
mouse_get_device(int mouse)
|
|
|
|
|
{
|
2022-09-18 17:13:28 -04:00
|
|
|
return (mouse_devices[mouse].device);
|
2017-12-10 02:08:37 -05:00
|
|
|
}
|
|
|
|
|
|
2017-05-06 17:48:33 +02:00
|
|
|
int
|
2017-12-10 02:08:37 -05:00
|
|
|
mouse_get_buttons(void)
|
2016-12-23 03:16:24 +01:00
|
|
|
{
|
2023-05-11 03:02:36 -04:00
|
|
|
return mouse_nbut;
|
2017-05-05 01:49:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Return number of MOUSE types we know about. */
|
2017-05-06 17:48:33 +02:00
|
|
|
int
|
|
|
|
|
mouse_get_ndev(void)
|
2017-05-05 01:49:42 +02:00
|
|
|
{
|
2022-09-18 17:13:28 -04:00
|
|
|
return ((sizeof(mouse_devices) / sizeof(mouse_t)) - 1);
|
2016-12-23 03:16:24 +01:00
|
|
|
}
|
2023-08-11 22:29:53 +02:00
|
|
|
|
2023-08-12 00:17:01 +02:00
|
|
|
void
|
|
|
|
|
mouse_set_raw(int raw)
|
|
|
|
|
{
|
|
|
|
|
mouse_raw = raw;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-11 22:29:53 +02:00
|
|
|
void
|
|
|
|
|
mouse_reset(void)
|
|
|
|
|
{
|
|
|
|
|
if (mouse_curr != NULL)
|
|
|
|
|
return; /* Mouse already initialized. */
|
|
|
|
|
|
|
|
|
|
mouse_log("MOUSE: reset(type=%d, '%s')\n",
|
|
|
|
|
mouse_type, mouse_devices[mouse_type].device->name);
|
|
|
|
|
|
|
|
|
|
/* Clear local data. */
|
|
|
|
|
mouse_clear_coords();
|
|
|
|
|
mouse_clear_buttons();
|
2023-10-23 01:21:28 +02:00
|
|
|
mouse_input_mode = 0;
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_timed = 1;
|
|
|
|
|
|
|
|
|
|
/* If no mouse configured, we're done. */
|
|
|
|
|
if (mouse_type == 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
timer_add(&mouse_timer, mouse_timer_poll, NULL, 0);
|
|
|
|
|
|
|
|
|
|
/* Poll at 100 Hz, the default of a PS/2 mouse. */
|
|
|
|
|
sample_rate = 100.0;
|
|
|
|
|
timer_on_auto(&mouse_timer, 1000000.0 / sample_rate);
|
|
|
|
|
|
|
|
|
|
mouse_curr = mouse_devices[mouse_type].device;
|
|
|
|
|
|
2023-10-23 00:36:19 +02:00
|
|
|
if ((mouse_type > 1) && (mouse_curr != NULL))
|
2023-08-11 22:29:53 +02:00
|
|
|
mouse_priv = device_add(mouse_curr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
mouse_close(void)
|
|
|
|
|
{
|
|
|
|
|
if (mouse_curr == NULL)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
mouse_curr = NULL;
|
|
|
|
|
mouse_priv = NULL;
|
|
|
|
|
mouse_nbut = 0;
|
|
|
|
|
mouse_dev_poll = NULL;
|
|
|
|
|
|
|
|
|
|
timer_stop(&mouse_timer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Initialize the mouse module. */
|
|
|
|
|
void
|
|
|
|
|
mouse_init(void)
|
|
|
|
|
{
|
|
|
|
|
/* Initialize local data. */
|
|
|
|
|
mouse_clear_coords();
|
|
|
|
|
mouse_clear_buttons();
|
|
|
|
|
|
|
|
|
|
mouse_type = MOUSE_TYPE_NONE;
|
|
|
|
|
mouse_curr = NULL;
|
|
|
|
|
mouse_priv = NULL;
|
|
|
|
|
mouse_nbut = 0;
|
|
|
|
|
mouse_dev_poll = NULL;
|
|
|
|
|
}
|