Skip to content

Commit 68878f1

Browse files
Roderick ColenbranderKishor Krishna Bhat
Roderick Colenbrander
authored and
Kishor Krishna Bhat
committed
UPSTREAM: HID: playstation: Initial DualSense USB support
Implement the support for PlayStation DualSense gamepad in USB mode. Support features include buttons and sticks,which adhere to the Linux gamepad spec. Bug: 167947264 CRs-fixed: 2971837 Change-Id: I7cf496f9b6f721cdd3e79387caa86b2ccc6378fb Signed-off-by: Roderick Colenbrander <[email protected]> Reviewed-by: Barnabás Pőcze <[email protected]> Signed-off-by: Benjamin Tissoires <[email protected]> Signed-off-by: Farid Chahla <[email protected]> Git-repo: https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git Git-commit: bc2e15a9a0228b10fece576d4f6a974c002ff07b Signed-off-by: Kishor Krishna Bhat <[email protected]>
1 parent 9093207 commit 68878f1

File tree

5 files changed

+339
-0
lines changed

5 files changed

+339
-0
lines changed

MAINTAINERS

+6
Original file line numberDiff line numberDiff line change
@@ -6611,6 +6611,12 @@ F: drivers/hid/
66116611
F: include/linux/hid*
66126612
F: include/uapi/linux/hid*
66136613

6614+
HID PLAYSTATION DRIVER
6615+
M: Roderick Colenbrander <[email protected]>
6616+
6617+
S: Supported
6618+
F: drivers/hid/hid-playstation.c
6619+
66146620
HID SENSOR HUB DRIVERS
66156621
M: Jiri Kosina <[email protected]>
66166622
M: Jonathan Cameron <[email protected]>

drivers/hid/Kconfig

+8
Original file line numberDiff line numberDiff line change
@@ -797,6 +797,14 @@ config HID_PLANTRONICS
797797

798798
Say M here if you may ever plug in a Plantronics USB audio device.
799799

800+
config HID_PLAYSTATION
801+
tristate "PlayStation HID Driver"
802+
depends on HID
803+
---help---
804+
Provides support for Sony PS5 controllers including support for
805+
its special functionalities e.g. touchpad, lights and motion
806+
sensors.
807+
800808
config HID_PRIMAX
801809
tristate "Primax non-fully HID-compliant devices"
802810
depends on HID

drivers/hid/Makefile

+1
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,7 @@ hid-picolcd-$(CONFIG_HID_PICOLCD_CIR) += hid-picolcd_cir.o
8787
hid-picolcd-$(CONFIG_DEBUG_FS) += hid-picolcd_debugfs.o
8888

8989
obj-$(CONFIG_HID_PLANTRONICS) += hid-plantronics.o
90+
obj-$(CONFIG_HID_PLAYSTATION) += hid-playstation.o
9091
obj-$(CONFIG_HID_PRIMAX) += hid-primax.o
9192
obj-$(CONFIG_HID_REDRAGON) += hid-redragon.o
9293
obj-$(CONFIG_HID_RETRODE) += hid-retrode.o

drivers/hid/hid-ids.h

+1
Original file line numberDiff line numberDiff line change
@@ -1031,6 +1031,7 @@
10311031
#define USB_DEVICE_ID_SONY_PS4_CONTROLLER 0x05c4
10321032
#define USB_DEVICE_ID_SONY_PS4_CONTROLLER_2 0x09cc
10331033
#define USB_DEVICE_ID_SONY_PS4_CONTROLLER_DONGLE 0x0ba0
1034+
#define USB_DEVICE_ID_SONY_PS5_CONTROLLER 0x0ce6
10341035
#define USB_DEVICE_ID_SONY_MOTION_CONTROLLER 0x03d5
10351036
#define USB_DEVICE_ID_SONY_NAVIGATION_CONTROLLER 0x042f
10361037
#define USB_DEVICE_ID_SONY_BUZZ_CONTROLLER 0x0002

drivers/hid/hid-playstation.c

+323
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,323 @@
1+
// SPDX-License-Identifier: GPL-2.0-or-later
2+
/*
3+
* HID driver for Sony DualSense(TM) controller.
4+
*
5+
* Copyright (c) 2020 Sony Interactive Entertainment
6+
*/
7+
#include <linux/bits.h>
8+
#include <linux/device.h>
9+
#include <linux/hid.h>
10+
#include <linux/input/mt.h>
11+
#include <linux/module.h>
12+
13+
#include <asm/unaligned.h>
14+
15+
#include "hid-ids.h"
16+
17+
#define HID_PLAYSTATION_VERSION_PATCH 0x8000
18+
19+
/* Base class for playstation devices. */
20+
struct ps_device {
21+
struct hid_device *hdev;
22+
int (*parse_report)(struct ps_device *dev, struct hid_report *report, u8 *data, int size);
23+
};
24+
25+
#define DS_INPUT_REPORT_USB 0x01
26+
#define DS_INPUT_REPORT_USB_SIZE 64
27+
28+
/* Button masks for DualSense input report. */
29+
#define DS_BUTTONS0_HAT_SWITCH GENMASK(3, 0)
30+
#define DS_BUTTONS0_SQUARE BIT(4)
31+
#define DS_BUTTONS0_CROSS BIT(5)
32+
#define DS_BUTTONS0_CIRCLE BIT(6)
33+
#define DS_BUTTONS0_TRIANGLE BIT(7)
34+
#define DS_BUTTONS1_L1 BIT(0)
35+
#define DS_BUTTONS1_R1 BIT(1)
36+
#define DS_BUTTONS1_L2 BIT(2)
37+
#define DS_BUTTONS1_R2 BIT(3)
38+
#define DS_BUTTONS1_CREATE BIT(4)
39+
#define DS_BUTTONS1_OPTIONS BIT(5)
40+
#define DS_BUTTONS1_L3 BIT(6)
41+
#define DS_BUTTONS1_R3 BIT(7)
42+
#define DS_BUTTONS2_PS_HOME BIT(0)
43+
#define DS_BUTTONS2_TOUCHPAD BIT(1)
44+
45+
struct dualsense {
46+
struct ps_device base;
47+
struct input_dev *gamepad;
48+
};
49+
50+
struct dualsense_touch_point {
51+
uint8_t contact;
52+
uint8_t x_lo;
53+
uint8_t x_hi:4, y_lo:4;
54+
uint8_t y_hi;
55+
} __packed;
56+
57+
/* Main DualSense input report excluding any BT/USB specific headers. */
58+
struct dualsense_input_report {
59+
uint8_t x, y;
60+
uint8_t rx, ry;
61+
uint8_t z, rz;
62+
uint8_t seq_number;
63+
uint8_t buttons[4];
64+
uint8_t reserved[4];
65+
66+
/* Motion sensors */
67+
__le16 gyro[3]; /* x, y, z */
68+
__le16 accel[3]; /* x, y, z */
69+
__le32 sensor_timestamp;
70+
uint8_t reserved2;
71+
72+
/* Touchpad */
73+
struct dualsense_touch_point points[2];
74+
75+
uint8_t reserved3[12];
76+
uint8_t status;
77+
uint8_t reserved4[10];
78+
} __packed;
79+
80+
/*
81+
* Common gamepad buttons across DualShock 3 / 4 and DualSense.
82+
* Note: for device with a touchpad, touchpad button is not included
83+
* as it will be part of the touchpad device.
84+
*/
85+
static const int ps_gamepad_buttons[] = {
86+
BTN_WEST, /* Square */
87+
BTN_NORTH, /* Triangle */
88+
BTN_EAST, /* Circle */
89+
BTN_SOUTH, /* Cross */
90+
BTN_TL, /* L1 */
91+
BTN_TR, /* R1 */
92+
BTN_TL2, /* L2 */
93+
BTN_TR2, /* R2 */
94+
BTN_SELECT, /* Create (PS5) / Share (PS4) */
95+
BTN_START, /* Option */
96+
BTN_THUMBL, /* L3 */
97+
BTN_THUMBR, /* R3 */
98+
BTN_MODE, /* PS Home */
99+
};
100+
101+
static const struct {int x; int y; } ps_gamepad_hat_mapping[] = {
102+
{0, -1}, {1, -1}, {1, 0}, {1, 1}, {0, 1}, {-1, 1}, {-1, 0}, {-1, -1},
103+
{0, 0},
104+
};
105+
106+
static struct input_dev *ps_allocate_input_dev(struct hid_device *hdev, const char *name_suffix)
107+
{
108+
struct input_dev *input_dev;
109+
110+
input_dev = devm_input_allocate_device(&hdev->dev);
111+
if (!input_dev)
112+
return ERR_PTR(-ENOMEM);
113+
114+
input_dev->id.bustype = hdev->bus;
115+
input_dev->id.vendor = hdev->vendor;
116+
input_dev->id.product = hdev->product;
117+
input_dev->id.version = hdev->version;
118+
input_dev->uniq = hdev->uniq;
119+
120+
if (name_suffix) {
121+
input_dev->name = devm_kasprintf(&hdev->dev, GFP_KERNEL, "%s %s", hdev->name,
122+
name_suffix);
123+
if (!input_dev->name)
124+
return ERR_PTR(-ENOMEM);
125+
} else {
126+
input_dev->name = hdev->name;
127+
}
128+
129+
input_set_drvdata(input_dev, hdev);
130+
131+
return input_dev;
132+
}
133+
134+
static struct input_dev *ps_gamepad_create(struct hid_device *hdev)
135+
{
136+
struct input_dev *gamepad;
137+
unsigned int i;
138+
int ret;
139+
140+
gamepad = ps_allocate_input_dev(hdev, NULL);
141+
if (IS_ERR(gamepad))
142+
return ERR_CAST(gamepad);
143+
144+
input_set_abs_params(gamepad, ABS_X, 0, 255, 0, 0);
145+
input_set_abs_params(gamepad, ABS_Y, 0, 255, 0, 0);
146+
input_set_abs_params(gamepad, ABS_Z, 0, 255, 0, 0);
147+
input_set_abs_params(gamepad, ABS_RX, 0, 255, 0, 0);
148+
input_set_abs_params(gamepad, ABS_RY, 0, 255, 0, 0);
149+
input_set_abs_params(gamepad, ABS_RZ, 0, 255, 0, 0);
150+
151+
input_set_abs_params(gamepad, ABS_HAT0X, -1, 1, 0, 0);
152+
input_set_abs_params(gamepad, ABS_HAT0Y, -1, 1, 0, 0);
153+
154+
for (i = 0; i < ARRAY_SIZE(ps_gamepad_buttons); i++)
155+
input_set_capability(gamepad, EV_KEY, ps_gamepad_buttons[i]);
156+
157+
ret = input_register_device(gamepad);
158+
if (ret)
159+
return ERR_PTR(ret);
160+
161+
return gamepad;
162+
}
163+
164+
static int dualsense_parse_report(struct ps_device *ps_dev, struct hid_report *report,
165+
u8 *data, int size)
166+
{
167+
struct hid_device *hdev = ps_dev->hdev;
168+
struct dualsense *ds = container_of(ps_dev, struct dualsense, base);
169+
struct dualsense_input_report *ds_report;
170+
uint8_t value;
171+
172+
/*
173+
* DualSense in USB uses the full HID report for reportID 1, but
174+
* Bluetooth uses a minimal HID report for reportID 1 and reports
175+
* the full report using reportID 49.
176+
*/
177+
if (hdev->bus == BUS_USB && report->id == DS_INPUT_REPORT_USB &&
178+
size == DS_INPUT_REPORT_USB_SIZE) {
179+
ds_report = (struct dualsense_input_report *)&data[1];
180+
} else {
181+
hid_err(hdev, "Unhandled reportID=%d\n", report->id);
182+
return -1;
183+
}
184+
185+
input_report_abs(ds->gamepad, ABS_X, ds_report->x);
186+
input_report_abs(ds->gamepad, ABS_Y, ds_report->y);
187+
input_report_abs(ds->gamepad, ABS_RX, ds_report->rx);
188+
input_report_abs(ds->gamepad, ABS_RY, ds_report->ry);
189+
input_report_abs(ds->gamepad, ABS_Z, ds_report->z);
190+
input_report_abs(ds->gamepad, ABS_RZ, ds_report->rz);
191+
192+
value = ds_report->buttons[0] & DS_BUTTONS0_HAT_SWITCH;
193+
if (value > ARRAY_SIZE(ps_gamepad_hat_mapping))
194+
value = 8; /* center */
195+
input_report_abs(ds->gamepad, ABS_HAT0X, ps_gamepad_hat_mapping[value].x);
196+
input_report_abs(ds->gamepad, ABS_HAT0Y, ps_gamepad_hat_mapping[value].y);
197+
198+
input_report_key(ds->gamepad, BTN_WEST, ds_report->buttons[0] & DS_BUTTONS0_SQUARE);
199+
input_report_key(ds->gamepad, BTN_SOUTH, ds_report->buttons[0] & DS_BUTTONS0_CROSS);
200+
input_report_key(ds->gamepad, BTN_EAST, ds_report->buttons[0] & DS_BUTTONS0_CIRCLE);
201+
input_report_key(ds->gamepad, BTN_NORTH, ds_report->buttons[0] & DS_BUTTONS0_TRIANGLE);
202+
input_report_key(ds->gamepad, BTN_TL, ds_report->buttons[1] & DS_BUTTONS1_L1);
203+
input_report_key(ds->gamepad, BTN_TR, ds_report->buttons[1] & DS_BUTTONS1_R1);
204+
input_report_key(ds->gamepad, BTN_TL2, ds_report->buttons[1] & DS_BUTTONS1_L2);
205+
input_report_key(ds->gamepad, BTN_TR2, ds_report->buttons[1] & DS_BUTTONS1_R2);
206+
input_report_key(ds->gamepad, BTN_SELECT, ds_report->buttons[1] & DS_BUTTONS1_CREATE);
207+
input_report_key(ds->gamepad, BTN_START, ds_report->buttons[1] & DS_BUTTONS1_OPTIONS);
208+
input_report_key(ds->gamepad, BTN_THUMBL, ds_report->buttons[1] & DS_BUTTONS1_L3);
209+
input_report_key(ds->gamepad, BTN_THUMBR, ds_report->buttons[1] & DS_BUTTONS1_R3);
210+
input_report_key(ds->gamepad, BTN_MODE, ds_report->buttons[2] & DS_BUTTONS2_PS_HOME);
211+
input_sync(ds->gamepad);
212+
213+
return 0;
214+
}
215+
216+
static struct ps_device *dualsense_create(struct hid_device *hdev)
217+
{
218+
struct dualsense *ds;
219+
int ret;
220+
221+
ds = devm_kzalloc(&hdev->dev, sizeof(*ds), GFP_KERNEL);
222+
if (!ds)
223+
return ERR_PTR(-ENOMEM);
224+
225+
/*
226+
* Patch version to allow userspace to distinguish between
227+
* hid-generic vs hid-playstation axis and button mapping.
228+
*/
229+
hdev->version |= HID_PLAYSTATION_VERSION_PATCH;
230+
231+
ds->base.hdev = hdev;
232+
ds->base.parse_report = dualsense_parse_report;
233+
hid_set_drvdata(hdev, ds);
234+
235+
ds->gamepad = ps_gamepad_create(hdev);
236+
if (IS_ERR(ds->gamepad)) {
237+
ret = PTR_ERR(ds->gamepad);
238+
goto err;
239+
}
240+
241+
return &ds->base;
242+
243+
err:
244+
return ERR_PTR(ret);
245+
}
246+
247+
static int ps_raw_event(struct hid_device *hdev, struct hid_report *report,
248+
u8 *data, int size)
249+
{
250+
struct ps_device *dev = hid_get_drvdata(hdev);
251+
252+
if (dev && dev->parse_report)
253+
return dev->parse_report(dev, report, data, size);
254+
255+
return 0;
256+
}
257+
258+
static int ps_probe(struct hid_device *hdev, const struct hid_device_id *id)
259+
{
260+
struct ps_device *dev;
261+
int ret;
262+
263+
ret = hid_parse(hdev);
264+
if (ret) {
265+
hid_err(hdev, "Parse failed\n");
266+
return ret;
267+
}
268+
269+
ret = hid_hw_start(hdev, HID_CONNECT_HIDRAW);
270+
if (ret) {
271+
hid_err(hdev, "Failed to start HID device\n");
272+
return ret;
273+
}
274+
275+
ret = hid_hw_open(hdev);
276+
if (ret) {
277+
hid_err(hdev, "Failed to open HID device\n");
278+
goto err_stop;
279+
}
280+
281+
if (hdev->product == USB_DEVICE_ID_SONY_PS5_CONTROLLER) {
282+
dev = dualsense_create(hdev);
283+
if (IS_ERR(dev)) {
284+
hid_err(hdev, "Failed to create dualsense.\n");
285+
ret = PTR_ERR(dev);
286+
goto err_close;
287+
}
288+
}
289+
290+
return ret;
291+
292+
err_close:
293+
hid_hw_close(hdev);
294+
err_stop:
295+
hid_hw_stop(hdev);
296+
return ret;
297+
}
298+
299+
static void ps_remove(struct hid_device *hdev)
300+
{
301+
hid_hw_close(hdev);
302+
hid_hw_stop(hdev);
303+
}
304+
305+
static const struct hid_device_id ps_devices[] = {
306+
{ HID_USB_DEVICE(USB_VENDOR_ID_SONY, USB_DEVICE_ID_SONY_PS5_CONTROLLER) },
307+
{ }
308+
};
309+
MODULE_DEVICE_TABLE(hid, ps_devices);
310+
311+
static struct hid_driver ps_driver = {
312+
.name = "playstation",
313+
.id_table = ps_devices,
314+
.probe = ps_probe,
315+
.remove = ps_remove,
316+
.raw_event = ps_raw_event,
317+
};
318+
319+
module_hid_driver(ps_driver);
320+
321+
MODULE_AUTHOR("Sony Interactive Entertainment");
322+
MODULE_DESCRIPTION("HID Driver for PlayStation peripherals.");
323+
MODULE_LICENSE("GPL v2");

0 commit comments

Comments
 (0)