aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/hid/hid-rakk.c
blob: 1140058a76ca4b57a649fa5bccf54029c59b35f8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
// SPDX-License-Identifier: GPL-2.0
/*
 *  HID driver for Rakk devices
 *
 *  Copyright (c) 2026 Karl Cayme
 *
 *  The Rakk Dasig X gaming mouse has a faulty HID report descriptor that
 *  declares USAGE_MAXIMUM = 3 (buttons 1-3) while actually sending 5 button
 *  bits (REPORT_COUNT = 5). This causes the kernel to ignore side buttons
 *  (buttons 4 and 5). This driver fixes the descriptor so all 5 buttons
 *  are properly recognized across 3 modes (wired, dongle, and Bluetooth).
 */

#include <linux/device.h>
#include <linux/hid.h>
#include <linux/module.h>
#include "hid-ids.h"

/*
 * The faulty byte is at offset 17 in the report descriptor for all three
 * connection modes (USB direct, wireless dongle, and Bluetooth).
 *
 * Bytes 16-17 are: 0x29 0x03 (USAGE_MAXIMUM = 3)
 * The fix changes byte 17 to 0x05 (USAGE_MAXIMUM = 5).
 *
 * Original descriptor bytes 0-17:
 *   05 01 09 02 a1 01 85 01 09 01 a1 00 05 09 19 01 29 03
 *                                                       ^^
 *   Should be 0x05 to declare 5 buttons instead of 3.
 */
#define RAKK_RDESC_USAGE_MAX_OFFSET	17
#define RAKK_RDESC_USAGE_MAX_ORIG	0x03
#define RAKK_RDESC_USAGE_MAX_FIXED	0x05
#define RAKK_RDESC_USB_SIZE		193
#define RAKK_RDESC_DONGLE_SIZE		150
#define RAKK_RDESC_BT_SIZE		89

static const __u8 *rakk_report_fixup(struct hid_device *hdev, __u8 *rdesc,
				     unsigned int *rsize)
{
	if (((*rsize == RAKK_RDESC_USB_SIZE &&
	      hdev->product == USB_DEVICE_ID_TELINK_RAKK_DASIG_X) ||
	     (*rsize == RAKK_RDESC_DONGLE_SIZE &&
	      hdev->product == USB_DEVICE_ID_TELINK_RAKK_DASIG_X_DONGLE) ||
	     (*rsize == RAKK_RDESC_BT_SIZE &&
	      hdev->product == USB_DEVICE_ID_TELINK_RAKK_DASIG_X_BT)) &&
	    rdesc[RAKK_RDESC_USAGE_MAX_OFFSET] == RAKK_RDESC_USAGE_MAX_ORIG) {
		hid_info(hdev, "fixing Rakk Dasig X button count (3 -> 5)\n");
		rdesc[RAKK_RDESC_USAGE_MAX_OFFSET] = RAKK_RDESC_USAGE_MAX_FIXED;
	}

	return rdesc;
}

static const struct hid_device_id rakk_devices[] = {
	{ HID_USB_DEVICE(USB_VENDOR_ID_TELINK,
			 USB_DEVICE_ID_TELINK_RAKK_DASIG_X) },
	{ HID_USB_DEVICE(USB_VENDOR_ID_TELINK,
			 USB_DEVICE_ID_TELINK_RAKK_DASIG_X_DONGLE) },
	{ HID_BLUETOOTH_DEVICE(USB_VENDOR_ID_TELINK,
			       USB_DEVICE_ID_TELINK_RAKK_DASIG_X_BT) },
	{ }
};
MODULE_DEVICE_TABLE(hid, rakk_devices);

static struct hid_driver rakk_driver = {
	.name = "rakk",
	.id_table = rakk_devices,
	.report_fixup = rakk_report_fixup,
};
module_hid_driver(rakk_driver);

MODULE_DESCRIPTION("HID driver for Rakk Dasig X mouse - fix side button support");
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Karl Cayme");