This is an attempt at creating an "open source Steam Input". Specifically the Gamepad APIs translating part.
This project aims to hook and re-implement various gamepad APIs such as XInput over SDL3.
The goal is to allow old and new games to support any gamepad controller supported by SDL3 (over 200+) out of the box with zero configuration.
If you are a game developper and you'd like your game to support a variety of gamepad controllers without rewriting your game, this project might be of interest to you.
XInput API calls ("Debug mode")
XInput
- XInputGetState ✔️
- XInputGetStateEx ✔️
- XInputSetState ✔️
- XInputSetStateEx¹ ✔️
- XInputGetCapabilities ✔️
- XInputGetCapabilitiesEx ✔️
- XInputGetBatteryInformation ✔️
- XInputGetKeystroke ❌️
- XInputWaitForGuideButton ❌️
- XInputCancelGuideButtonWait ❌
- XInputPowerOffController ❌
- XInputGetBaseBusInformation ❌
- XInputEnable 🚫
- XInputGetAudioDeviceIds 🚫
- XInputGetDSoundAudioDeviceGuids 🚫
💡XInputGetCapabilitiesEx() will report the real VID/PID of the controller.
¹ NB: XInputSetStateEx() from GDK (XInputOnGameInput) is implemented and has been arbitrarily set to ordinal 1000. It does not exist in XInput.
DInput8
- DirectInput8Create ✔️
- IDirectInput8::ConfigureDevices ❌
- IDirectInput8::CreateDevice ⚠
- IDirectInputDevice8::Acquire ⚠
- IDirectInputDevice8::BuildActionMap ❌
- IDirectInputDevice8::CreateEffect ❌
- IDirectInputDevice8::EnumCreatedEffectObjects ❌
- IDirectInputDevice8::EnumEffects ❌
- IDirectInputDevice8::EnumEffectsInFile ❌
- IDirectInputDevice8::EnumObjects ⚠
- IDirectInputDevice8::Escape ❌
- IDirectInputDevice8::GetCapabilities ✔
- IDirectInputDevice8::GetDeviceData ❌
- IDirectInputDevice8::GetDeviceInfo ❌
- IDirectInputDevice8::GetDeviceState ⚠
- IDirectInputDevice8::GetEffectInfo ❌
- IDirectInputDevice8::GetForceFeedbackState ❌
- IDirectInputDevice8::GetImageInfo ❌
- IDirectInputDevice8::GetObjectInfo ❌
- IDirectInputDevice8::GetProperty ❌
- IDirectInputDevice8::Initialize ⚠
- IDirectInputDevice8::Poll ✔
- IDirectInputDevice8::RunControlPanel ❌
- IDirectInputDevice8::SendDeviceData ❌
- IDirectInputDevice8::SendForceFeedbackCommand ❌
- IDirectInputDevice8::SetActionMap ❌
- IDirectInputDevice8::SetCooperativeLevel ⚠
- IDirectInputDevice8::SetDataFormat ⚠
- IDirectInputDevice8::SetEventNotification ❌
- IDirectInputDevice8::SetProperty ❌
- IDirectInputDevice8::Unacquire ⚠
- IDirectInputDevice8::WriteEffectToFile ❌
- IDirectInput8::EnumDevices ⚠
- IDirectInput8::EnumDevicesBySemantics ❌
- IDirectInput8::FindDevice ❌
- IDirectInput8::GetDeviceStatus ❌
- IDirectInput8::Initialize ⚠
- IDirectInput8::RunControlPanel ❌
DInput (1-7)
_To Do_WinMM
- joyConfigChanged ⚠
- joyGetDevCapsA ⚠
- joyGetDevCapsW ⚠
- joyGetNumDevs ✔
- joyGetPos ✔
- joyGetPosEx ✔
- joyGetThreshold ⚠
- joyReleaseCapture ⚠
- joySetCapture ⚠
- joySetThreshold ⚠
✔ | ⚠ | ❌ | 🚫 |
---|---|---|---|
Implemented | Work in progress | To do | Won't (Deprecated) |
Steam Input API
Steam Input API only games. You need an action set to translate input. Hooking these API is going down the Steam Enulator rabbit hole.
Not sure yet how best to handle this. Meanwhile when using a Steam Emulator that translates Steam Input to XInput you can leverage InputFusion to then translate XInput to SDL.
HID / Raw
These APIs aren't really like the standardised Gamepad APIs like XInput. They are much akin to low level access.
SDL mostly uses these low level APIs.
If a game uses these APIs to add support for a specific Gamepad; The game devs probably have a certain experience in mind and we shouldn't interfere with it.
Many mods and other 3rd party "fix" rely on these low level access to do their job. And they often complain about the new Steam Input capabilities of hooking system wide all relevant APIs for gamepad while Steam is running.
As such, I do no think these API are relevant for my project.
WGI (Windows Gaming Input)
This API is new and specifically designed to allow support for Gamepad others than Xbox controllers in a standardised way. Doesn't seem pertinent to this project for now.
The DLL can be used either as
- A) a drop-in replacement (DLL side-loading) or
- B) by being injected into a target process.
Unless you know what you are doing stick with the first approach.
Replace xinput*.dll
in the game dir if present, otherwise place it next to the game's executable.
This might require a little guess work to find which version of XInput a game is using: xinput1_4.dll, xinput1_3.dll, xinput9_1_0.dll.
The DLL exports every documented XInput functions as well as "hidden" XInput___Ex functions.
NB: If necessary, you can opt-in to also hook / detour XInput functions to force the game to use the ones from the DLL. See env var section below.
Set the env var to enable function hooking / detouring before you execute and inject into your target process. See env var section below. Otherwise once injected it won't do much more than just init. SDL.
Here is a simple example in Node.js using xan105/node-remote-thread:
import { env } from "node:process";
import { spawn } from "node:child_process";
import { dirname } from "node:path";
import { createRemoteThread } from "@xan105/create-remote-thread";
const EXECUTABLE = "G:\\METAPHOR\\METAPHOR.exe";
const ADDON = "G:\\METAPHOR\\InputFusion.dll";
const ARGS = [];
const binary = spawn(EXECUTABLE, ARGS, {
cwd: dirname(EXECUTABLE),
stdio:[ "ignore", "ignore", "ignore" ],
detached: true,
env: {
...env,
"GAMEPAD_API_XINPUT": "HOOK"
}
});
binary.once("spawn", () => {
binary.unref();
createRemoteThread(binary.pid, ADDON);
});
🧪 Mostly experimental features are behind env var flags.
When enabled the LED light of the controller is used to show the battery level of the controller:
- 100% / Green
- 75% / Yellow
- 50% / Orange
- 25% / Red
Currently only available for PS4/PS5 controller in wireless.
Enable XInput functions hooking / detouring.
This forces the use of the XInput functions from the DLL when calling XInput functions.
Enable DInput8 functions hooking / detouring.
Warning
The current implementation is very barebone and is based on a Xbox 360 controller, therefore it has the same limitations as a real Xbox 360 controller with DInput such as no force feedback and no individual trigger axis. Games have different DInput layout expectation depending on their era and/or gamepads they support.
Enable WinMM Joystick functions hooking / detouring.
Note
After considering the games from that era that could be played with a gamepad. I somewhat deviated from the usual mapping for this implementation.
The D-Pad and the left joystick both map the X and Y axis usually used for movement. The right joystick maps to the POV since back then it was used to move the camera point of view around which is traditionally the role of the right joystick on a gamepad. The right and left trigger are mapped to the Z axis meaning no individual trigger axis, similar to a Xbox360 in DInput.
SDL has the ability to change the mapping of an existing gamepad or add support for gamepads that SDL is unaware of.
To manually do so the end user can, for example, use the env var SDL_GAMECONTROLLERCONFIG
or SDL_GAMECONTROLLERCONFIG_FILE
.
The mapping string has the format: GUID,name,mapping
. Buttons can be used as a gamepad axes and vice versa.
Example of a valid mapping for a gamepad:
"341a3608000000000000504944564944,Afterglow PS3 Controller,a:b1,b:b2,y:b3,x:b0,start:b9,guide:b12,back:b8,dpup:h0.1,dpleft:h0.8,dpdown:h0.4,dpright:h0.2,leftshoulder:b4,rightshoulder:b5,leftstick:b10,rightstick:b11,leftx:a0,lefty:a1,rightx:a2,righty:a3,lefttrigger:b6,righttrigger:b7"
📄 For more details, please kindly see the SDL documentation.
-
SDL might still be initializing when the game does it's first Gamepad API call (on startup).
-
Games that support more than one input API usually, but not always, do a lot of Win32 APIs sniffing behind the scenes to determine which input API to use. Therefore even tho an input API is translated to SDL, your gamepad may still not work due to how a game engine is programmed and how it decides to handle input. The focus of this project is for now on API translation and not on poorly engineered games compatibility.
🆚 Visual Studio 2022
📦 Vendor dependencies:
- Microsoft Detours library
- SDL3 Library recompiled as a static lib.
Solution: ./vc/InputFusion.sln
The 4 projects inside the solution control the build output:
- InputFusion ->
InputFusion.dll
(DLL injection) - XInput ->
xinput.dll
(DLL sideloading / injection) - DInput8 ->
dinput8.dll
(DLL sideloading / injection) - WinMM ->
winmm.dll
(DLL sideloading / injection)
Output: ./build/${project}/output/${platform}/${config}