2003-06-21 16:55:19 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2011-02-25 01:05:47 +03:00
|
|
|
// $Id$
|
2003-06-21 16:55:19 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2021-01-23 15:06:11 +03:00
|
|
|
// Copyright (C) 2003-2021 The Bochs Project
|
2003-06-21 16:55:19 +04:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
2009-02-08 12:05:52 +03:00
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2003-06-21 16:55:19 +04:00
|
|
|
|
|
|
|
//
|
|
|
|
// Standard PC gameport
|
|
|
|
//
|
|
|
|
|
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
2008-01-27 01:24:03 +03:00
|
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
2003-06-21 16:55:19 +04:00
|
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
|
|
#define BX_PLUGGABLE
|
|
|
|
|
2004-06-19 19:20:15 +04:00
|
|
|
#include "iodev.h"
|
2008-12-29 23:16:08 +03:00
|
|
|
#include "gameport.h"
|
2003-06-21 16:55:19 +04:00
|
|
|
|
|
|
|
#ifdef __linux__
|
|
|
|
|
2024-04-07 00:18:01 +03:00
|
|
|
#ifndef __ANDROID__
|
2003-06-21 16:55:19 +04:00
|
|
|
#include <linux/joystick.h>
|
2016-08-12 20:06:14 +03:00
|
|
|
#endif
|
2003-06-21 16:55:19 +04:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2003-12-21 20:24:45 +03:00
|
|
|
#elif defined(WIN32)
|
|
|
|
|
2014-02-15 04:23:36 +04:00
|
|
|
#include <mmsystem.h>
|
2003-12-30 00:48:56 +03:00
|
|
|
#ifndef JOY_BUTTON1
|
2003-12-21 20:24:45 +03:00
|
|
|
#define JOY_BUTTON1 1
|
|
|
|
#define JOY_BUTTON2 2
|
|
|
|
UINT STDCALL joyGetPos(UINT, LPJOYINFO);
|
2003-12-30 00:48:56 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define JOYSTICKID1 0
|
2003-12-21 20:24:45 +03:00
|
|
|
|
2003-06-21 16:55:19 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define LOG_THIS theGameport->
|
|
|
|
|
|
|
|
bx_gameport_c *theGameport = NULL;
|
|
|
|
|
2021-01-23 15:06:11 +03:00
|
|
|
PLUGIN_ENTRY_FOR_MODULE(gameport)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
2021-02-07 19:16:06 +03:00
|
|
|
if (mode == PLUGIN_INIT) {
|
2021-01-23 15:06:11 +03:00
|
|
|
theGameport = new bx_gameport_c();
|
|
|
|
bx_devices.pluginGameport = theGameport;
|
|
|
|
BX_REGISTER_DEVICE_DEVMODEL(plugin, type, theGameport, BX_PLUGIN_GAMEPORT);
|
2021-02-07 19:16:06 +03:00
|
|
|
} else if (mode == PLUGIN_FINI) {
|
2021-01-23 15:06:11 +03:00
|
|
|
bx_devices.pluginGameport = &bx_devices.stubGameport;
|
|
|
|
delete theGameport;
|
2021-02-26 23:37:49 +03:00
|
|
|
} else if (mode == PLUGIN_PROBE) {
|
2021-02-07 19:16:06 +03:00
|
|
|
return (int)PLUGTYPE_OPTIONAL;
|
2021-01-23 15:06:11 +03:00
|
|
|
}
|
2003-06-21 16:55:19 +04:00
|
|
|
return(0); // Success
|
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
bx_gameport_c::bx_gameport_c()
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
2011-12-30 15:13:37 +04:00
|
|
|
put("gameport", "GAME");
|
2006-09-12 17:05:07 +04:00
|
|
|
joyfd = -1;
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
bx_gameport_c::~bx_gameport_c()
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
|
|
|
if (joyfd >= 0) close(joyfd);
|
2012-08-19 12:16:20 +04:00
|
|
|
SIM->get_bochs_root()->remove("gameport");
|
2006-09-10 21:18:44 +04:00
|
|
|
BX_DEBUG(("Exit"));
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
void bx_gameport_c::init(void)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
|
|
|
// Allocate the gameport IO address range 0x200..0x207
|
|
|
|
for (unsigned addr=0x200; addr<0x208; addr++) {
|
2003-07-31 16:04:48 +04:00
|
|
|
DEV_register_ioread_handler(this, read_handler, addr, "Gameport", 1);
|
|
|
|
DEV_register_iowrite_handler(this, write_handler, addr, "Gameport", 1);
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
|
|
|
|
2011-03-31 20:54:06 +04:00
|
|
|
// always enabled unless controlled by external device
|
|
|
|
BX_GAMEPORT_THIS enabled = 1;
|
2003-06-21 16:55:19 +04:00
|
|
|
BX_GAMEPORT_THIS port = 0xf0;
|
|
|
|
BX_GAMEPORT_THIS write_usec = 0;
|
|
|
|
BX_GAMEPORT_THIS timer_x = 0;
|
|
|
|
BX_GAMEPORT_THIS timer_y = 0;
|
|
|
|
|
|
|
|
#ifdef __linux__
|
|
|
|
BX_GAMEPORT_THIS joyfd = open("/dev/input/js0", O_RDONLY);
|
|
|
|
if (BX_GAMEPORT_THIS joyfd >= 0) {
|
|
|
|
for (unsigned i=0; i<4; i++) poll_joydev();
|
|
|
|
}
|
2003-12-21 20:24:45 +03:00
|
|
|
#elif defined(WIN32)
|
|
|
|
JOYINFO joypos;
|
|
|
|
if (joyGetPos(JOYSTICKID1, &joypos) == JOYERR_NOERROR) {
|
|
|
|
BX_GAMEPORT_THIS joyfd = 1;
|
|
|
|
} else {
|
|
|
|
BX_GAMEPORT_THIS joyfd = -1;
|
|
|
|
}
|
2003-06-21 16:55:19 +04:00
|
|
|
#else
|
|
|
|
BX_GAMEPORT_THIS joyfd = -1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
void bx_gameport_c::reset(unsigned type)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
|
|
|
// nothing for now
|
|
|
|
}
|
|
|
|
|
2006-05-27 19:54:49 +04:00
|
|
|
void bx_gameport_c::register_state(void)
|
|
|
|
{
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *list = new bx_list_c(SIM->get_bochs_root(), "gameport", "Gameport State");
|
2011-03-31 20:54:06 +04:00
|
|
|
BXRS_PARAM_BOOL(list, enabled, BX_GAMEPORT_THIS enabled);
|
2006-05-28 22:14:05 +04:00
|
|
|
BXRS_HEX_PARAM_FIELD(list, port, BX_GAMEPORT_THIS port);
|
|
|
|
BXRS_DEC_PARAM_FIELD(list, delay_x, BX_GAMEPORT_THIS delay_x);
|
|
|
|
BXRS_DEC_PARAM_FIELD(list, delay_y, BX_GAMEPORT_THIS delay_y);
|
|
|
|
BXRS_PARAM_BOOL(list, timer_x, BX_GAMEPORT_THIS timer_x);
|
|
|
|
BXRS_PARAM_BOOL(list, timer_y, BX_GAMEPORT_THIS timer_y);
|
|
|
|
BXRS_DEC_PARAM_FIELD(list, write_usec, BX_GAMEPORT_THIS write_usec);
|
2006-05-27 19:54:49 +04:00
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
void bx_gameport_c::poll_joydev(void)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
2024-04-07 00:18:01 +03:00
|
|
|
#ifndef __ANDROID__
|
2003-06-21 16:55:19 +04:00
|
|
|
#ifdef __linux__
|
|
|
|
struct js_event e;
|
|
|
|
fd_set joyfds;
|
|
|
|
struct timeval tv;
|
|
|
|
|
|
|
|
memset(&tv, 0, sizeof(tv));
|
|
|
|
FD_ZERO(&joyfds);
|
|
|
|
FD_SET(BX_GAMEPORT_THIS joyfd, &joyfds);
|
|
|
|
e.type = 0;
|
|
|
|
if (select(BX_GAMEPORT_THIS joyfd+1, &joyfds, NULL, NULL, &tv)) {
|
|
|
|
read(BX_GAMEPORT_THIS joyfd, &e, sizeof(struct js_event));
|
|
|
|
if (e.type & JS_EVENT_BUTTON) {
|
|
|
|
if (e.value == 1) {
|
|
|
|
BX_GAMEPORT_THIS port &= ~(0x10 << e.number);
|
|
|
|
} else {
|
|
|
|
BX_GAMEPORT_THIS port |= (0x10 << e.number);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (e.type & JS_EVENT_AXIS) {
|
|
|
|
if (e.number == 0) {
|
|
|
|
BX_GAMEPORT_THIS delay_x = 25 + ((e.value + 0x8000) / 60);
|
|
|
|
}
|
|
|
|
if (e.number == 1) {
|
|
|
|
BX_GAMEPORT_THIS delay_y = 25 + ((e.value + 0x8000) / 62);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2003-12-21 20:24:45 +03:00
|
|
|
#elif defined(WIN32)
|
|
|
|
JOYINFO joypos;
|
|
|
|
if (joyGetPos(JOYSTICKID1, &joypos) == JOYERR_NOERROR) {
|
|
|
|
if (joypos.wButtons & JOY_BUTTON1) {
|
|
|
|
BX_GAMEPORT_THIS port &= ~0x10;
|
|
|
|
} else {
|
|
|
|
BX_GAMEPORT_THIS port |= 0x10;
|
|
|
|
}
|
|
|
|
if (joypos.wButtons & JOY_BUTTON2) {
|
|
|
|
BX_GAMEPORT_THIS port &= ~0x20;
|
|
|
|
} else {
|
|
|
|
BX_GAMEPORT_THIS port |= 0x20;
|
|
|
|
}
|
|
|
|
BX_GAMEPORT_THIS delay_x = 25 + (joypos.wXpos / 60);
|
|
|
|
BX_GAMEPORT_THIS delay_y = 25 + (joypos.wYpos / 60);
|
|
|
|
}
|
2003-06-21 16:55:19 +04:00
|
|
|
#endif
|
2024-04-07 00:18:01 +03:00
|
|
|
#endif //__ANDROID__
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
// static IO port read callback handler
|
|
|
|
// redirects to non-static class handler to avoid virtual functions
|
2003-06-21 16:55:19 +04:00
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
Bit32u bx_gameport_c::read_handler(void *this_ptr, Bit32u address, unsigned io_len)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
2006-04-05 22:49:35 +04:00
|
|
|
#if !BX_USE_GAMEPORT_SMF
|
2003-06-21 16:55:19 +04:00
|
|
|
bx_gameport_c *class_ptr = (bx_gameport_c *) this_ptr;
|
2006-03-07 21:16:41 +03:00
|
|
|
return class_ptr->read(address, io_len);
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
Bit32u bx_gameport_c::read(Bit32u address, unsigned io_len)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
|
|
|
#else
|
|
|
|
UNUSED(this_ptr);
|
2006-04-05 22:49:35 +04:00
|
|
|
#endif // !BX_USE_GAMEPORT_SMF
|
2003-06-21 16:55:19 +04:00
|
|
|
Bit64u usec;
|
|
|
|
|
2011-03-31 20:54:06 +04:00
|
|
|
if (BX_GAMEPORT_THIS enabled) {
|
|
|
|
if (BX_GAMEPORT_THIS joyfd >= 0) {
|
|
|
|
poll_joydev();
|
|
|
|
usec = bx_pc_system.time_usec();
|
|
|
|
if (BX_GAMEPORT_THIS timer_x) {
|
|
|
|
if ((usec - BX_GAMEPORT_THIS write_usec) >= BX_GAMEPORT_THIS delay_x) {
|
|
|
|
BX_GAMEPORT_THIS port &= 0xfe;
|
|
|
|
BX_GAMEPORT_THIS timer_x = 0;
|
|
|
|
}
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
2011-03-31 20:54:06 +04:00
|
|
|
if (BX_GAMEPORT_THIS timer_y) {
|
|
|
|
if ((usec - BX_GAMEPORT_THIS write_usec) >= BX_GAMEPORT_THIS delay_y) {
|
|
|
|
BX_GAMEPORT_THIS port &= 0xfd;
|
|
|
|
BX_GAMEPORT_THIS timer_y = 0;
|
|
|
|
}
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
2011-03-31 20:54:06 +04:00
|
|
|
} else {
|
|
|
|
BX_DEBUG(("read: joystick not present"));
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
2011-03-31 20:54:06 +04:00
|
|
|
return BX_GAMEPORT_THIS port;
|
2003-06-21 16:55:19 +04:00
|
|
|
} else {
|
2011-03-31 20:54:06 +04:00
|
|
|
BX_DEBUG(("read: gameport disabled"));
|
|
|
|
return 0xff;
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
// static IO port write callback handler
|
|
|
|
// redirects to non-static class handler to avoid virtual functions
|
2003-06-21 16:55:19 +04:00
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
void bx_gameport_c::write_handler(void *this_ptr, Bit32u address, Bit32u value, unsigned io_len)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
2006-04-05 22:49:35 +04:00
|
|
|
#if !BX_USE_GAMEPORT_SMF
|
2003-06-21 16:55:19 +04:00
|
|
|
bx_gameport_c *class_ptr = (bx_gameport_c *) this_ptr;
|
|
|
|
class_ptr->write(address, value, io_len);
|
|
|
|
}
|
|
|
|
|
2006-03-07 21:16:41 +03:00
|
|
|
void bx_gameport_c::write(Bit32u address, Bit32u value, unsigned io_len)
|
2003-06-21 16:55:19 +04:00
|
|
|
{
|
|
|
|
#else
|
|
|
|
UNUSED(this_ptr);
|
2006-04-05 22:49:35 +04:00
|
|
|
#endif // !BX_USE_GAMEPORT_SMF
|
2003-06-21 16:55:19 +04:00
|
|
|
|
2011-03-31 20:54:06 +04:00
|
|
|
if (BX_GAMEPORT_THIS enabled) {
|
|
|
|
BX_GAMEPORT_THIS write_usec = bx_pc_system.time_usec();
|
|
|
|
BX_GAMEPORT_THIS timer_x = 1;
|
|
|
|
BX_GAMEPORT_THIS timer_y = 1;
|
|
|
|
BX_GAMEPORT_THIS port |= 0x0f;
|
|
|
|
} else {
|
|
|
|
BX_DEBUG(("write: gameport disabled"));
|
|
|
|
}
|
2003-06-21 16:55:19 +04:00
|
|
|
}
|