]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
HID: playstation: Center initial joystick axes to prevent spurious events
authorSiarhei Vishniakou <svv@google.com>
Tue, 11 Nov 2025 23:45:19 +0000 (15:45 -0800)
committerBenjamin Tissoires <bentiss@kernel.org>
Fri, 19 Dec 2025 11:07:30 +0000 (12:07 +0100)
When a new PlayStation gamepad (DualShock 4 or DualSense) is initialized,
the input subsystem sets the default value for its absolute axes (e.g.,
ABS_X, ABS_Y) to 0.

However, the hardware's actual neutral/resting state for these joysticks
is 128 (0x80). This creates a mismatch.

When the first HID report arrives from the device, the driver sees the
resting value of 128. The kernel compares this to its initial state of 0
and incorrectly interprets this as a delta (0 -> 128). Consequently, it
generates EV_ABS events for this initial, non-existent movement.

This behavior can fail userspace 'sanity check' tests (e.g., in
Android CTS) that correctly assert no motion events should be generated
from a device that is already at rest.

This patch fixes the issue by explicitly setting the initial value of the
main joystick axes (e.g., ABS_X, ABS_Y, ABS_RX, ABS_RY) to 128 (0x80)
in the common ps_gamepad_create() function.

This aligns the kernel's initial state with the hardware's expected
neutral state, ensuring that the first report (at 128) produces no
delta and thus, no spurious event.

Signed-off-by: Siarhei Vishniakou <svv@google.com>
Reviewed-by: Benjamin Tissoires <bentiss@kernel.org>
Signed-off-by: Benjamin Tissoires <bentiss@kernel.org>
drivers/hid/hid-playstation.c

index 128aa6abd10bed7de96f7ddc324f57d243d0b66b..e4dfcf26b04e74ce9e730679094ba14417462228 100644 (file)
@@ -753,11 +753,16 @@ ps_gamepad_create(struct hid_device *hdev,
        if (IS_ERR(gamepad))
                return ERR_CAST(gamepad);
 
+       /* Set initial resting state for joysticks to 128 (center) */
        input_set_abs_params(gamepad, ABS_X, 0, 255, 0, 0);
+       gamepad->absinfo[ABS_X].value = 128;
        input_set_abs_params(gamepad, ABS_Y, 0, 255, 0, 0);
+       gamepad->absinfo[ABS_Y].value = 128;
        input_set_abs_params(gamepad, ABS_Z, 0, 255, 0, 0);
        input_set_abs_params(gamepad, ABS_RX, 0, 255, 0, 0);
+       gamepad->absinfo[ABS_RX].value = 128;
        input_set_abs_params(gamepad, ABS_RY, 0, 255, 0, 0);
+       gamepad->absinfo[ABS_RY].value = 128;
        input_set_abs_params(gamepad, ABS_RZ, 0, 255, 0, 0);
 
        input_set_abs_params(gamepad, ABS_HAT0X, -1, 1, 0, 0);