2023-04-23 16:13:51 +02:00
|
|
|
#include <windows.h>
|
|
|
|
#include <dinput.h>
|
2024-09-30 18:50:46 +02:00
|
|
|
#include <stdbool.h>
|
2023-04-23 16:13:51 +02:00
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "idacio/di-dev.h"
|
|
|
|
|
|
|
|
#include "util/dprintf.h"
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
const struct idac_di_config *idac_di_cfg;
|
|
|
|
static HWND idac_di_wnd;
|
|
|
|
static IDirectInputDevice8W *idac_di_dev;
|
|
|
|
|
|
|
|
/* Individual DI Effects */
|
|
|
|
static IDirectInputEffect *idac_di_fx;
|
|
|
|
static IDirectInputEffect *idac_di_fx_rumble;
|
|
|
|
static IDirectInputEffect *idac_di_fx_damper;
|
|
|
|
|
|
|
|
/* Max FFB Board value is 127 */
|
|
|
|
static const double idac_di_ffb_scale = 127.0;
|
|
|
|
|
|
|
|
HRESULT idac_di_dev_init(
|
|
|
|
const struct idac_di_config *cfg,
|
|
|
|
IDirectInputDevice8W *dev,
|
|
|
|
HWND wnd)
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
assert(dev != NULL);
|
|
|
|
assert(wnd != NULL);
|
|
|
|
|
|
|
|
idac_di_cfg = cfg;
|
|
|
|
idac_di_dev = dev;
|
|
|
|
idac_di_wnd = wnd;
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT idac_di_dev_poll(
|
|
|
|
IDirectInputDevice8W *dev,
|
|
|
|
HWND wnd,
|
|
|
|
union idac_di_state *out)
|
2023-04-23 16:13:51 +02:00
|
|
|
{
|
|
|
|
HRESULT hr;
|
2024-09-30 18:50:46 +02:00
|
|
|
MSG msg;
|
|
|
|
|
|
|
|
assert(dev != NULL);
|
|
|
|
assert(wnd != NULL);
|
|
|
|
assert(out != NULL);
|
|
|
|
|
|
|
|
memset(out, 0, sizeof(*out));
|
|
|
|
|
|
|
|
/* Pump our dummy window's message queue just in case DirectInput or an
|
|
|
|
IHV DirectInput driver somehow relies on it */
|
|
|
|
|
|
|
|
while (PeekMessageW(&msg, wnd, 0, 0, PM_REMOVE)) {
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessage(&msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IDirectInputDevice8_GetDeviceState(
|
|
|
|
dev,
|
|
|
|
sizeof(out->st),
|
|
|
|
&out->st);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
dprintf("DirectInput: GetDeviceState error: %08x\n", (int) hr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* JVS lacks a protocol for reporting hardware errors from poll command
|
|
|
|
responses, so this ends up returning zeroed input state instead. */
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT idac_di_dev_start(IDirectInputDevice8W *dev, HWND wnd) {
|
|
|
|
HRESULT hr;
|
2023-04-23 16:13:51 +02:00
|
|
|
|
|
|
|
assert(dev != NULL);
|
|
|
|
assert(wnd != NULL);
|
|
|
|
|
|
|
|
hr = IDirectInputDevice8_SetCooperativeLevel(
|
|
|
|
dev,
|
|
|
|
wnd,
|
|
|
|
DISCL_BACKGROUND | DISCL_EXCLUSIVE);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
dprintf("DirectInput: SetCooperativeLevel failed: %08x\n", (int) hr);
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IDirectInputDevice8_SetDataFormat(dev, &c_dfDIJoystick);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
dprintf("DirectInput: SetDataFormat failed: %08x\n", (int) hr);
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IDirectInputDevice8_Acquire(dev);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
dprintf("DirectInput: Acquire failed: %08x\n", (int) hr);
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
HRESULT idac_di_ffb_init(void)
|
2023-04-23 16:13:51 +02:00
|
|
|
{
|
2024-09-30 18:50:46 +02:00
|
|
|
HRESULT hr;
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
hr = idac_di_dev_start(idac_di_dev, idac_di_wnd);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
return hr;
|
|
|
|
}
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
return S_OK;
|
|
|
|
}
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
void idac_di_ffb_toggle(bool active)
|
|
|
|
{
|
|
|
|
if (active) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Stop and release all effects */
|
|
|
|
/* I never programmed DirectInput Effects, so this might be bad practice. */
|
|
|
|
if (idac_di_fx != NULL) {
|
|
|
|
IDirectInputEffect_Stop(idac_di_fx);
|
|
|
|
IDirectInputEffect_Release(idac_di_fx);
|
|
|
|
idac_di_fx = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (idac_di_fx_rumble != NULL) {
|
|
|
|
IDirectInputEffect_Stop(idac_di_fx_rumble);
|
|
|
|
IDirectInputEffect_Release(idac_di_fx_rumble);
|
|
|
|
idac_di_fx_rumble = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (idac_di_fx_damper != NULL) {
|
|
|
|
IDirectInputEffect_Stop(idac_di_fx_damper);
|
|
|
|
IDirectInputEffect_Release(idac_di_fx_damper);
|
|
|
|
idac_di_fx_damper = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void idac_di_ffb_constant_force(uint8_t direction_ffb, uint8_t force)
|
|
|
|
{
|
|
|
|
/* DI expects a magnitude in the range of -10.000 to 10.000 */
|
|
|
|
uint16_t ffb_strength = idac_di_cfg->ffb_constant_force_strength * 100;
|
|
|
|
if (ffb_strength == 0) {
|
|
|
|
return;
|
|
|
|
}
|
2023-04-23 16:13:51 +02:00
|
|
|
|
|
|
|
DWORD axis;
|
|
|
|
LONG direction;
|
|
|
|
DIEFFECT fx;
|
2024-09-30 18:50:46 +02:00
|
|
|
DICONSTANTFORCE cf;
|
2023-04-23 16:13:51 +02:00
|
|
|
HRESULT hr;
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
/* Direction 0: move to the right, 1: move to the left */
|
|
|
|
LONG magnitude = (LONG)(((double)force / idac_di_ffb_scale) * ffb_strength);
|
|
|
|
cf.lMagnitude = (direction_ffb == 0) ? -magnitude : magnitude;
|
2023-04-23 16:13:51 +02:00
|
|
|
|
|
|
|
axis = DIJOFS_X;
|
2024-09-30 18:50:46 +02:00
|
|
|
/* Irrelevant as magnitude descripbes the direction */
|
2023-04-23 16:13:51 +02:00
|
|
|
direction = 0;
|
|
|
|
|
|
|
|
memset(&fx, 0, sizeof(fx));
|
|
|
|
fx.dwSize = sizeof(fx);
|
|
|
|
fx.dwFlags = DIEFF_CARTESIAN | DIEFF_OBJECTOFFSETS;
|
|
|
|
fx.dwDuration = INFINITE;
|
|
|
|
fx.dwGain = DI_FFNOMINALMAX;
|
|
|
|
fx.dwTriggerButton = DIEB_NOTRIGGER;
|
|
|
|
fx.dwTriggerRepeatInterval = INFINITE;
|
|
|
|
fx.cAxes = 1;
|
|
|
|
fx.rgdwAxes = &axis;
|
|
|
|
fx.rglDirection = &direction;
|
2024-09-30 18:50:46 +02:00
|
|
|
fx.cbTypeSpecificParams = sizeof(cf);
|
|
|
|
fx.lpvTypeSpecificParams = &cf;
|
|
|
|
|
|
|
|
if (idac_di_fx != NULL) {
|
|
|
|
// Try to update the existing effect
|
|
|
|
hr = IDirectInputEffect_SetParameters(idac_di_fx, &fx, DIEP_TYPESPECIFICPARAMS);
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
dprintf("DirectInput: Failed to update constant force feedback, recreating effect: %08x\n", (int)hr);
|
|
|
|
// Stop and release the current effect if updating fails
|
|
|
|
IDirectInputEffect_Stop(idac_di_fx);
|
|
|
|
IDirectInputEffect_Release(idac_di_fx);
|
|
|
|
idac_di_fx = NULL;
|
|
|
|
}
|
|
|
|
}
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
// Create a new constant force effect
|
|
|
|
IDirectInputEffect *obj;
|
2023-04-23 16:13:51 +02:00
|
|
|
hr = IDirectInputDevice8_CreateEffect(
|
2024-09-30 18:50:46 +02:00
|
|
|
idac_di_dev,
|
|
|
|
&GUID_ConstantForce,
|
2023-04-23 16:13:51 +02:00
|
|
|
&fx,
|
|
|
|
&obj,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
2024-09-30 18:50:46 +02:00
|
|
|
dprintf("DirectInput: Constant force feedback creation failed: %08x\n", (int) hr);
|
2023-04-23 16:13:51 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IDirectInputEffect_Start(obj, INFINITE, 0);
|
|
|
|
if (FAILED(hr)) {
|
2024-09-30 18:50:46 +02:00
|
|
|
dprintf("DirectInput: Constant force feedback start failed: %08x\n", (int) hr);
|
2023-04-23 16:13:51 +02:00
|
|
|
IDirectInputEffect_Release(obj);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
idac_di_fx = obj;
|
|
|
|
}
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
void idac_di_ffb_rumble(uint8_t force, uint8_t period)
|
|
|
|
{
|
|
|
|
/* DI expects a magnitude in the range of -10.000 to 10.000 */
|
|
|
|
uint16_t ffb_strength = idac_di_cfg->ffb_rumble_strength * 100;
|
|
|
|
if (ffb_strength == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t ffb_duration = idac_di_cfg->ffb_rumble_duration;
|
|
|
|
|
|
|
|
DWORD axis;
|
|
|
|
LONG direction;
|
|
|
|
DIEFFECT fx;
|
|
|
|
DIPERIODIC pe;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
/* Duration in microseconds,
|
|
|
|
Might be totally wrong as especially on FANATEC wheels as this code will
|
|
|
|
crash the game. TODO: Figure out why this effect will crash on FANATEC! */
|
|
|
|
DWORD duration = (DWORD)((double)force * ffb_duration);
|
|
|
|
|
|
|
|
memset(&pe, 0, sizeof(pe));
|
|
|
|
pe.dwMagnitude = (DWORD)(((double)force / idac_di_ffb_scale) * ffb_strength);
|
|
|
|
pe.lOffset = 0;
|
|
|
|
pe.dwPhase = 0;
|
|
|
|
pe.dwPeriod = duration;
|
|
|
|
|
|
|
|
axis = DIJOFS_X;
|
|
|
|
direction = 0;
|
|
|
|
|
|
|
|
memset(&fx, 0, sizeof(fx));
|
|
|
|
fx.dwSize = sizeof(fx);
|
|
|
|
fx.dwFlags = DIEFF_CARTESIAN | DIEFF_OBJECTOFFSETS;
|
|
|
|
fx.dwDuration = duration;
|
|
|
|
fx.dwGain = DI_FFNOMINALMAX;
|
|
|
|
fx.dwTriggerButton = DIEB_NOTRIGGER;
|
|
|
|
fx.dwTriggerRepeatInterval = INFINITE;
|
|
|
|
fx.cAxes = 1;
|
|
|
|
fx.rgdwAxes = &axis;
|
|
|
|
fx.rglDirection = &direction;
|
|
|
|
fx.cbTypeSpecificParams = sizeof(pe);
|
|
|
|
fx.lpvTypeSpecificParams = &pe;
|
|
|
|
|
|
|
|
if (idac_di_fx_rumble != NULL) {
|
|
|
|
// Try to update the existing effect
|
|
|
|
hr = IDirectInputEffect_SetParameters(idac_di_fx_rumble, &fx, DIEP_TYPESPECIFICPARAMS);
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
dprintf("DirectInput: Failed to update periodic force feedback, recreating effect: %08x\n", (int)hr);
|
|
|
|
// Stop and release the current effect if updating fails
|
|
|
|
IDirectInputEffect_Stop(idac_di_fx_rumble);
|
|
|
|
IDirectInputEffect_Release(idac_di_fx_rumble);
|
|
|
|
idac_di_fx_rumble = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
IDirectInputEffect *obj;
|
|
|
|
hr = IDirectInputDevice8_CreateEffect(
|
|
|
|
idac_di_dev,
|
|
|
|
&GUID_Sine,
|
|
|
|
&fx,
|
|
|
|
&obj,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
dprintf("DirectInput: Periodic force feedback creation failed: %08x\n", (int) hr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IDirectInputEffect_Start(obj, INFINITE, 0);
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
dprintf("DirectInput: Periodic force feedback start failed: %08x\n", (int) hr);
|
|
|
|
IDirectInputEffect_Release(obj);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
idac_di_fx_rumble = obj;
|
2023-04-23 16:13:51 +02:00
|
|
|
}
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
void idac_di_ffb_damper(uint8_t force)
|
2023-04-23 16:13:51 +02:00
|
|
|
{
|
2024-09-30 18:50:46 +02:00
|
|
|
/* DI expects a coefficient in the range of -10.000 to 10.000 */
|
|
|
|
uint16_t ffb_strength = idac_di_cfg->ffb_damper_strength * 100;
|
|
|
|
if (ffb_strength == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
DWORD axis;
|
|
|
|
LONG direction;
|
|
|
|
DIEFFECT fx;
|
|
|
|
DICONDITION cond;
|
2023-04-23 16:13:51 +02:00
|
|
|
HRESULT hr;
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
memset(&cond, 0, sizeof(cond));
|
|
|
|
cond.lOffset = 0;
|
|
|
|
cond.lPositiveCoefficient = (LONG)(((double)force / idac_di_ffb_scale) * ffb_strength);
|
|
|
|
cond.lNegativeCoefficient = (LONG)(((double)force / idac_di_ffb_scale) * ffb_strength);
|
|
|
|
/* Not sure on this one */
|
|
|
|
cond.dwPositiveSaturation = DI_FFNOMINALMAX;
|
|
|
|
cond.dwNegativeSaturation = DI_FFNOMINALMAX;
|
|
|
|
cond.lDeadBand = 0;
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
axis = DIJOFS_X;
|
|
|
|
direction = 0;
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
memset(&fx, 0, sizeof(fx));
|
|
|
|
fx.dwSize = sizeof(fx);
|
|
|
|
fx.dwFlags = DIEFF_CARTESIAN | DIEFF_OBJECTOFFSETS;
|
|
|
|
fx.dwDuration = INFINITE;
|
|
|
|
fx.dwGain = DI_FFNOMINALMAX;
|
|
|
|
fx.dwTriggerButton = DIEB_NOTRIGGER;
|
|
|
|
fx.dwTriggerRepeatInterval = INFINITE;
|
|
|
|
fx.cAxes = 1;
|
|
|
|
fx.rgdwAxes = &axis;
|
|
|
|
fx.rglDirection = &direction;
|
|
|
|
fx.cbTypeSpecificParams = sizeof(cond);
|
|
|
|
fx.lpvTypeSpecificParams = &cond;
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
if (idac_di_fx_damper != NULL) {
|
|
|
|
// Try to update the existing effect
|
|
|
|
hr = IDirectInputEffect_SetParameters(idac_di_fx_damper, &fx, DIEP_TYPESPECIFICPARAMS);
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
dprintf("DirectInput: Failed to update damper force feedback, recreating effect: %08x\n", (int)hr);
|
|
|
|
// Stop and release the current effect if updating fails
|
|
|
|
IDirectInputEffect_Stop(idac_di_fx_damper);
|
|
|
|
IDirectInputEffect_Release(idac_di_fx_damper);
|
|
|
|
idac_di_fx_damper = NULL;
|
|
|
|
}
|
2023-04-23 16:13:51 +02:00
|
|
|
}
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
// Create a new damper force effect
|
|
|
|
IDirectInputEffect *obj;
|
|
|
|
hr = IDirectInputDevice8_CreateEffect(
|
|
|
|
idac_di_dev,
|
|
|
|
&GUID_Damper,
|
|
|
|
&fx,
|
|
|
|
&obj,
|
|
|
|
NULL);
|
2023-04-23 16:13:51 +02:00
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
2024-09-30 18:50:46 +02:00
|
|
|
dprintf("DirectInput: Damper force feedback creation failed: %08x\n", (int) hr);
|
|
|
|
return;
|
2023-04-23 16:13:51 +02:00
|
|
|
}
|
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
hr = IDirectInputEffect_Start(obj, INFINITE, 0);
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
dprintf("DirectInput: Damper force feedback start failed: %08x\n", (int) hr);
|
|
|
|
IDirectInputEffect_Release(obj);
|
|
|
|
return;
|
|
|
|
}
|
2023-04-23 16:13:51 +02:00
|
|
|
|
2024-09-30 18:50:46 +02:00
|
|
|
idac_di_fx_damper = obj;
|
2023-04-23 16:13:51 +02:00
|
|
|
}
|