I'm in the process of creating a multiplatform (but OpenGL favoring) game engine on Java for Android/OUYA and PC, and the PC platform adapter is giving me some issues when rapidly checking gamepad state.
I'm currently using LWJGL/JInput to write the PC adapter, and whenever I poll gamepad state faster than about 30 times a second (giving about 33ms to do the poll and update the state), the values obtained from the gamepad are false. False in the sense that the stick can be halfway to the right, and yet getAxisValue returns 0 instead of something near 0.5f out of 1f. On top of that, it seems to take longer than the 33ms that would be fine otherwise when asked to do it more often. What gives?
In a nutshell, what happens on a typical update call is that the engine scans through the active players' controllers for the state of specific buttons that are actually used in the game, all set and registered ahead of time. At the moment, in my test app, that consists of two buttons and an axis. So all it's doing is checking the current state of three inputs, no more. After being passed through a couple layers of interfaces and into a switch block, eventually this is called:
return controller.isButtonPressed(map.A);
or
return controller.getAxisValue(map.LS_H);
where map contains final ints that link names to specific index values for the given controller.
What I've found through my own testing:
- This isn't a hardware issue, as it stays pretty accurate when below 30 to 35 updates per second.
- When ramping the speed up to about 60 updates a second to match the 60fps rate of the graphics thread, it causes massive lag on the input thread only. The graphics thread is not affected, which makes me think that it isn't a general performance issue either.
- Changing isButtonPressed or getAxisValue to arbitrary preset values solves the update lag issues, so it isn't my code causing the intermittent stops, definitely isButtonPressed and getAxisValue.
Is there any way to improve the speed of gamepad checking, or some setting somewhere that I missed that disables unnecessary input routines through LWJGL/JInput?
Is matching 60fps too much to ask in terms of performance?
Polling isn't the best way, it's a relatively expensive operation. Take a look at the event interface on JInput, you should be able to check the contents of the event queue quickly.