1
0
mirror of https://gitea.tendokyu.moe/Dniel97/segatools.git synced 2024-12-18 06:45:53 +01:00

idzio: Break out Initial D Zero IO DLL

This commit is contained in:
Tau 2019-05-03 20:36:23 -04:00
parent b172c22322
commit 9603a528c6
8 changed files with 261 additions and 89 deletions

View File

@ -1,11 +1,7 @@
#include <windows.h> #include <windows.h>
#include <xinput.h>
#include <math.h>
#include <stdbool.h> #include <stdbool.h>
#include <stddef.h>
#include <stdint.h> #include <stdint.h>
#include <stdlib.h>
#include "amex/jvs.h" #include "amex/jvs.h"
@ -13,6 +9,8 @@
#include "idzhook/jvs.h" #include "idzhook/jvs.h"
#include "idzio/idzio.h"
#include "jvs/jvs-bus.h" #include "jvs/jvs-bus.h"
#include "util/dprintf.h" #include "util/dprintf.h"
@ -30,12 +28,6 @@ static const struct io3_ops idz_jvs_io3_ops = {
.read_coin_counter = idz_jvs_read_coin_counter, .read_coin_counter = idz_jvs_read_coin_counter,
}; };
static struct io3 idz_jvs_io3;
static bool idz_jvs_coin;
static uint16_t idz_jvs_coins;
static bool idz_jvs_shifting;
static uint8_t idz_jvs_gear;
static const uint16_t idz_jvs_gear_signals[] = { static const uint16_t idz_jvs_gear_signals[] = {
/* Neutral */ /* Neutral */
0x0000, 0x0000,
@ -53,79 +45,82 @@ static const uint16_t idz_jvs_gear_signals[] = {
0x1400, 0x1400,
}; };
void idz_jvs_init(void) static struct io3 idz_jvs_io3;
HRESULT idz_jvs_init(void)
{ {
HRESULT hr;
hr = idz_io_init();
if (FAILED(hr)) {
return hr;
}
io3_init(&idz_jvs_io3, NULL, &idz_jvs_io3_ops, NULL); io3_init(&idz_jvs_io3, NULL, &idz_jvs_io3_ops, NULL);
jvs_attach(&idz_jvs_io3.jvs); jvs_attach(&idz_jvs_io3.jvs);
return S_OK;
} }
static void idz_jvs_read_switches(void *ctx, struct io3_switch_state *out) static void idz_jvs_read_switches(void *ctx, struct io3_switch_state *out)
{ {
bool shift_inc; uint8_t opbtn;
bool shift_dec; uint8_t gamebtn;
XINPUT_STATE xi; uint8_t gear;
WORD xb;
assert(out != NULL); assert(out != NULL);
memset(&xi, 0, sizeof(xi)); opbtn = 0;
XInputGetState(0, &xi); gamebtn = 0;
xb = xi.Gamepad.wButtons; gear = 0;
idz_io_jvs_read_buttons(&opbtn, &gamebtn);
idz_io_jvs_read_shifter(&gear);
/* Update gameplay buttons */ /* Update gameplay buttons */
if (xb & XINPUT_GAMEPAD_START) { if (gamebtn & IDZ_IO_GAMEBTN_UP) {
out->p1 |= 1 << 15;
idz_jvs_gear = 0; /* Reset to Neutral when start is pressed */
}
if (xb & XINPUT_GAMEPAD_DPAD_UP) {
out->p1 |= 1 << 13; out->p1 |= 1 << 13;
} }
if (xb & XINPUT_GAMEPAD_DPAD_DOWN) { if (gamebtn & IDZ_IO_GAMEBTN_DOWN) {
out->p1 |= 1 << 12; out->p1 |= 1 << 12;
} }
if (xb & XINPUT_GAMEPAD_DPAD_LEFT) { if (gamebtn & IDZ_IO_GAMEBTN_LEFT) {
out->p1 |= 1 << 11; out->p1 |= 1 << 11;
} }
if (xb & XINPUT_GAMEPAD_DPAD_RIGHT) { if (gamebtn & IDZ_IO_GAMEBTN_RIGHT) {
out->p1 |= 1 << 10; out->p1 |= 1 << 10;
} }
if (xb & XINPUT_GAMEPAD_BACK) { if (gamebtn & IDZ_IO_GAMEBTN_START) {
out->p1 |= 1 << 15;
}
if (gamebtn & IDZ_IO_GAMEBTN_VIEW_CHANGE) {
out->p1 |= 1 << 9; out->p1 |= 1 << 9;
} }
/* Update simulated six-speed shifter */ /* Update simulated six-speed shifter */
shift_inc = xb & (XINPUT_GAMEPAD_X | XINPUT_GAMEPAD_RIGHT_SHOULDER); if (gear > 6) {
shift_dec = xb & (XINPUT_GAMEPAD_Y | XINPUT_GAMEPAD_LEFT_SHOULDER); gear = 6;
if (!idz_jvs_shifting) {
if (shift_inc && idz_jvs_gear < 6) {
idz_jvs_gear++;
} }
if (shift_dec && idz_jvs_gear > 0) { out->p2 = idz_jvs_gear_signals[gear];
idz_jvs_gear--;
}
}
idz_jvs_shifting = shift_inc || shift_dec;
out->p2 = idz_jvs_gear_signals[idz_jvs_gear];
/* Update test/service buttons */ /* Update test/service buttons */
if (GetAsyncKeyState('1')) { if (opbtn & IDZ_IO_OPBTN_TEST) {
out->system = 0x80; out->system = 0x80;
} else { } else {
out->system = 0; out->system = 0;
} }
if (GetAsyncKeyState('2')) { if (opbtn & IDZ_IO_OPBTN_SERVICE) {
out->p1 |= 1 << 14; out->p1 |= 1 << 14;
} }
} }
@ -135,51 +130,23 @@ static void idz_jvs_read_analogs(
uint16_t *analogs, uint16_t *analogs,
uint8_t nanalogs) uint8_t nanalogs)
{ {
XINPUT_STATE xi; struct idz_io_analog_state state;
int left;
int right;
assert(analogs != NULL); assert(analogs != NULL);
memset(&xi, 0, sizeof(xi)); memset(&state, 0, sizeof(state));
XInputGetState(0, &xi); idz_io_jvs_read_analogs(&state);
/* Wheel */
if (nanalogs > 0) { if (nanalogs > 0) {
left = xi.Gamepad.sThumbLX; analogs[0] = 0x8000 + state.wheel;
if (left < -XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE) {
left += XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE;
} else if (left > XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE) {
left -= XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE;
} else {
left = 0;
} }
right = xi.Gamepad.sThumbRX;
if (right < -XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE) {
right += XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE;
} else if (right > XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE) {
right -= XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE;
} else {
right = 0;
}
analogs[0] = 0x8000 + (left + right) / 2;
}
/* Accel */
if (nanalogs > 1) { if (nanalogs > 1) {
analogs[1] = xi.Gamepad.bRightTrigger << 8; analogs[1] = state.accel;
} }
/* Brake */
if (nanalogs > 2) { if (nanalogs > 2) {
analogs[2] = xi.Gamepad.bLeftTrigger << 8; analogs[2] = state.brake;
} }
} }
@ -189,15 +156,5 @@ static uint16_t idz_jvs_read_coin_counter(void *ctx, uint8_t slot_no)
return 0; return 0;
} }
if (GetAsyncKeyState('3')) { return idz_io_jvs_read_coin_counter();
if (!idz_jvs_coin) {
dprintf("IDZero JVS: Coin drop\n");
idz_jvs_coin = true;
idz_jvs_coins++;
}
} else {
idz_jvs_coin = false;
}
return idz_jvs_coins;
} }

View File

@ -1,3 +1,5 @@
#pragma once #pragma once
void idz_jvs_init(void); #include <windows.h>
HRESULT idz_jvs_init(void);

View File

@ -14,6 +14,7 @@ shared_library(
aimeio_dll, aimeio_dll,
amex_lib, amex_lib,
board_lib, board_lib,
idzio_dll,
jvs_lib, jvs_lib,
platform_lib, platform_lib,
util_lib, util_lib,

151
idzio/idzio.c Normal file
View File

@ -0,0 +1,151 @@
#include <windows.h>
#include <xinput.h>
#include <stdbool.h>
#include <stdint.h>
#include "idzio/idzio.h"
static bool idz_io_coin;
static uint16_t idz_io_coins;
static bool idz_io_shifting;
static uint8_t idz_io_gear;
HRESULT idz_io_init(void)
{
return S_OK;
}
void idz_io_jvs_read_buttons(uint8_t *opbtn_out, uint8_t *gamebtn_out)
{
uint8_t opbtn;
uint8_t gamebtn;
XINPUT_STATE xi;
WORD xb;
opbtn = 0;
gamebtn = 0;
/* Update test/service buttons */
if (GetAsyncKeyState('1')) {
opbtn |= IDZ_IO_OPBTN_TEST;
}
if (GetAsyncKeyState('2')) {
opbtn |= IDZ_IO_OPBTN_SERVICE;
}
/* Update gameplay buttons */
memset(&xi, 0, sizeof(xi));
XInputGetState(0, &xi);
xb = xi.Gamepad.wButtons;
if (xb & XINPUT_GAMEPAD_DPAD_UP) {
gamebtn |= IDZ_IO_GAMEBTN_UP;
}
if (xb & XINPUT_GAMEPAD_DPAD_DOWN) {
gamebtn |= IDZ_IO_GAMEBTN_DOWN;
}
if (xb & XINPUT_GAMEPAD_DPAD_LEFT) {
gamebtn |= IDZ_IO_GAMEBTN_LEFT;
}
if (xb & XINPUT_GAMEPAD_DPAD_RIGHT) {
gamebtn |= IDZ_IO_GAMEBTN_RIGHT;
}
if (xb & XINPUT_GAMEPAD_START) {
gamebtn |= IDZ_IO_GAMEBTN_START;
}
if (xb & XINPUT_GAMEPAD_BACK) {
gamebtn |= IDZ_IO_GAMEBTN_VIEW_CHANGE;
}
*opbtn_out = opbtn;
*gamebtn_out = gamebtn;
}
void idz_io_jvs_read_shifter(uint8_t *gear)
{
bool shift_inc;
bool shift_dec;
XINPUT_STATE xi;
WORD xb;
memset(&xi, 0, sizeof(xi));
XInputGetState(0, &xi);
xb = xi.Gamepad.wButtons;
if (xb & XINPUT_GAMEPAD_START) {
idz_io_gear = 0; /* Reset to Neutral when start is pressed */
}
shift_inc = xb & (XINPUT_GAMEPAD_X | XINPUT_GAMEPAD_RIGHT_SHOULDER);
shift_dec = xb & (XINPUT_GAMEPAD_Y | XINPUT_GAMEPAD_LEFT_SHOULDER);
if (!idz_io_shifting) {
if (shift_inc && idz_io_gear < 6) {
idz_io_gear++;
}
if (shift_dec && idz_io_gear > 0) {
idz_io_gear--;
}
}
idz_io_shifting = shift_inc || shift_dec;
*gear = idz_io_gear;
}
void idz_io_jvs_read_analogs(struct idz_io_analog_state *out)
{
XINPUT_STATE xi;
int left;
int right;
memset(&xi, 0, sizeof(xi));
XInputGetState(0, &xi);
left = xi.Gamepad.sThumbLX;
if (left < -XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE) {
left += XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE;
} else if (left > XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE) {
left -= XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE;
} else {
left = 0;
}
right = xi.Gamepad.sThumbRX;
if (right < -XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE) {
right += XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE;
} else if (right > XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE) {
right -= XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE;
} else {
right = 0;
}
out->wheel = (left + right) / 2;
out->accel = xi.Gamepad.bRightTrigger << 8;
out->brake = xi.Gamepad.bLeftTrigger << 8;
}
uint16_t idz_io_jvs_read_coin_counter(void)
{
if (GetAsyncKeyState('3')) {
if (!idz_io_coin) {
idz_io_coin = true;
idz_io_coins++;
}
} else {
idz_io_coin = false;
}
return idz_io_coins;
}

8
idzio/idzio.def Normal file
View File

@ -0,0 +1,8 @@
LIBRARY idzio
EXPORTS
idz_io_init
idz_io_jvs_read_analogs
idz_io_jvs_read_buttons
idz_io_jvs_read_coin_counter
idz_io_jvs_read_shifter

37
idzio/idzio.h Normal file
View File

@ -0,0 +1,37 @@
#pragma once
#include <windows.h>
#include <stdint.h>
enum {
IDZ_IO_OPBTN_TEST = 0x01,
IDZ_IO_OPBTN_SERVICE = 0x02,
};
enum {
IDZ_IO_GAMEBTN_UP = 0x01,
IDZ_IO_GAMEBTN_DOWN = 0x02,
IDZ_IO_GAMEBTN_LEFT = 0x04,
IDZ_IO_GAMEBTN_RIGHT = 0x08,
IDZ_IO_GAMEBTN_START = 0x10,
IDZ_IO_GAMEBTN_VIEW_CHANGE = 0x20,
};
struct idz_io_analog_state {
int16_t wheel;
uint16_t accel;
uint16_t brake;
};
HRESULT idz_io_init(void);
void idz_io_jvs_read_analogs(struct idz_io_analog_state *out);
void idz_io_jvs_read_buttons(uint8_t *opbtn, uint8_t *gamebtn);
void idz_io_jvs_read_shifter(uint8_t *gear);
uint16_t idz_io_jvs_read_coin_counter(void);
// TODO force feedback once that gets reverse engineered

15
idzio/meson.build Normal file
View File

@ -0,0 +1,15 @@
idzio_dll = shared_library(
'idzio',
name_prefix : '',
include_directories : inc,
implicit_include_directories : false,
vs_module_defs : 'idzio.def',
c_pch : '../precompiled.h',
dependencies : [
xinput_lib,
],
sources : [
'idzio.c',
'idzio.h',
],
)

View File

@ -39,6 +39,7 @@ subdir('util')
subdir('aimeio') subdir('aimeio')
subdir('chuniio') subdir('chuniio')
subdir('divaio') subdir('divaio')
subdir('idzio')
subdir('cardhook') subdir('cardhook')
subdir('chunihook') subdir('chunihook')