2023-07-14 00:52:50 +02:00
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "board/io4.h"
|
|
|
|
|
2024-01-16 17:56:24 +01:00
|
|
|
#include "util/dprintf.h"
|
|
|
|
|
2023-07-14 00:52:50 +02:00
|
|
|
#include "swdchook/swdc-dll.h"
|
|
|
|
|
2024-01-16 17:56:24 +01:00
|
|
|
static HANDLE mmf;
|
|
|
|
static HRESULT init_mmf(void);
|
|
|
|
static void swdc_set_gamebtns(uint16_t value);
|
2023-07-14 00:52:50 +02:00
|
|
|
|
|
|
|
static HRESULT swdc_io4_poll(void *ctx, struct io4_state *state);
|
|
|
|
static uint16_t coins;
|
|
|
|
|
|
|
|
static const struct io4_ops swdc_io4_ops = {
|
|
|
|
.poll = swdc_io4_poll,
|
|
|
|
};
|
|
|
|
|
2024-01-16 17:56:24 +01:00
|
|
|
HRESULT swdc_io4_hook_init(const struct io4_config *cfg) {
|
2023-07-14 00:52:50 +02:00
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
assert(swdc_dll.init != NULL);
|
|
|
|
|
|
|
|
hr = io4_hook_init(cfg, &swdc_io4_ops, NULL);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
2024-01-16 17:56:24 +01:00
|
|
|
hr = init_mmf();
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
2023-07-14 00:52:50 +02:00
|
|
|
return swdc_dll.init();
|
|
|
|
}
|
|
|
|
|
2024-01-16 17:56:24 +01:00
|
|
|
// Function to initialize the memory-mapped file
|
|
|
|
static HRESULT init_mmf(void) {
|
|
|
|
mmf = CreateFileMappingA(INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE, 0, 2, "SWDCButton");
|
|
|
|
if (mmf == NULL) {
|
|
|
|
return S_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
swdc_set_gamebtns(0);
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void swdc_set_gamebtns(uint16_t value) {
|
|
|
|
// WaitForSingleObject(mutex, INFINITE);
|
|
|
|
|
|
|
|
// Update the memory-mapped file
|
|
|
|
LPVOID mmf_view = MapViewOfFile(mmf, FILE_MAP_ALL_ACCESS, 0, 0, 2);
|
|
|
|
if (mmf_view != NULL) {
|
|
|
|
uint16_t* ptr = (uint16_t*)mmf_view;
|
|
|
|
*ptr = value;
|
|
|
|
|
|
|
|
UnmapViewOfFile(mmf_view);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReleaseMutex(mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT swdc_io4_poll(void *ctx, struct io4_state *state) {
|
2023-07-14 00:52:50 +02:00
|
|
|
uint8_t opbtn;
|
|
|
|
uint16_t gamebtn;
|
|
|
|
struct swdc_io_analog_state analog_state;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
assert(swdc_dll.get_opbtns != NULL);
|
|
|
|
assert(swdc_dll.get_gamebtns != NULL);
|
|
|
|
assert(swdc_dll.get_analogs != NULL);
|
|
|
|
|
|
|
|
memset(state, 0, sizeof(*state));
|
|
|
|
memset(&analog_state, 0, sizeof(analog_state));
|
|
|
|
opbtn = 0;
|
|
|
|
gamebtn = 0;
|
|
|
|
|
|
|
|
swdc_dll.get_opbtns(&opbtn);
|
|
|
|
swdc_dll.get_gamebtns(&gamebtn);
|
|
|
|
swdc_dll.get_analogs(&analog_state);
|
|
|
|
|
|
|
|
if (opbtn & SWDC_IO_OPBTN_TEST) {
|
|
|
|
state->buttons[0] |= IO4_BUTTON_TEST;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opbtn & SWDC_IO_OPBTN_SERVICE) {
|
|
|
|
state->buttons[0] |= IO4_BUTTON_SERVICE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opbtn & SWDC_IO_OPBTN_COIN) {
|
|
|
|
coins++;
|
|
|
|
}
|
|
|
|
state->chutes[0] = coins << 8;
|
|
|
|
|
|
|
|
/* Update Cabinet buttons */
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_START) {
|
|
|
|
state->buttons[0] |= 1 << 7;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_VIEW_CHANGE) {
|
|
|
|
state->buttons[0] |= 1 << 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_UP) {
|
|
|
|
state->buttons[0] |= 1 << 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_DOWN) {
|
|
|
|
state->buttons[0] |= 1 << 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_LEFT) {
|
|
|
|
state->buttons[0] |= 1 << 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_RIGHT) {
|
|
|
|
state->buttons[0] |= 1 << 2;
|
|
|
|
}
|
|
|
|
|
2024-01-16 17:56:24 +01:00
|
|
|
/*
|
|
|
|
Update steering wheel buttons
|
|
|
|
|
2024-01-17 15:52:41 +01:00
|
|
|
Those are connected to the SEGA 838-15415 INDICATOR BD MAIN
|
2024-01-16 17:56:24 +01:00
|
|
|
USB board which is not emulated for now. So those buttons
|
|
|
|
are hooked to the built-in XInput support.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Instead update gamebtns for the file mapping */
|
|
|
|
|
|
|
|
swdc_set_gamebtns(gamebtn);
|
2023-07-14 00:52:50 +02:00
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_STEERING_BLUE) {
|
|
|
|
state->buttons[1] |= 1 << 15;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_STEERING_GREEN) {
|
|
|
|
state->buttons[1] |= 1 << 14;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_STEERING_RED) {
|
|
|
|
state->buttons[1] |= 1 << 13;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_STEERING_YELLOW) {
|
|
|
|
state->buttons[1] |= 1 << 12;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_STEERING_PADDLE_LEFT) {
|
|
|
|
state->buttons[1] |= 1 << 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamebtn & SWDC_IO_GAMEBTN_STEERING_PADDLE_RIGHT) {
|
|
|
|
state->buttons[1] |= 1 << 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Steering wheel increases left-to-right.
|
|
|
|
|
|
|
|
Use 0x8000 as the center point. */
|
|
|
|
|
|
|
|
state->adcs[0] = 0x8000 + analog_state.wheel;
|
|
|
|
state->adcs[1] = analog_state.accel;
|
|
|
|
state->adcs[2] = analog_state.brake;
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|