1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Roccat Kone[+] driver for Linux
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (c) 2010 Stefan Achatz <erazor_de@users.sourceforge.net>
6*4882a593Smuzhiyun */
7*4882a593Smuzhiyun
8*4882a593Smuzhiyun /*
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun /*
12*4882a593Smuzhiyun * Roccat Kone[+] is an updated/improved version of the Kone with more memory
13*4882a593Smuzhiyun * and functionality and without the non-standard behaviours the Kone had.
14*4882a593Smuzhiyun * KoneXTD has same capabilities but updated sensor.
15*4882a593Smuzhiyun */
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun #include <linux/device.h>
18*4882a593Smuzhiyun #include <linux/input.h>
19*4882a593Smuzhiyun #include <linux/hid.h>
20*4882a593Smuzhiyun #include <linux/module.h>
21*4882a593Smuzhiyun #include <linux/slab.h>
22*4882a593Smuzhiyun #include <linux/hid-roccat.h>
23*4882a593Smuzhiyun #include "hid-ids.h"
24*4882a593Smuzhiyun #include "hid-roccat-common.h"
25*4882a593Smuzhiyun #include "hid-roccat-koneplus.h"
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun static uint profile_numbers[5] = {0, 1, 2, 3, 4};
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun static struct class *koneplus_class;
30*4882a593Smuzhiyun
koneplus_profile_activated(struct koneplus_device * koneplus,uint new_profile)31*4882a593Smuzhiyun static void koneplus_profile_activated(struct koneplus_device *koneplus,
32*4882a593Smuzhiyun uint new_profile)
33*4882a593Smuzhiyun {
34*4882a593Smuzhiyun koneplus->actual_profile = new_profile;
35*4882a593Smuzhiyun }
36*4882a593Smuzhiyun
koneplus_send_control(struct usb_device * usb_dev,uint value,enum koneplus_control_requests request)37*4882a593Smuzhiyun static int koneplus_send_control(struct usb_device *usb_dev, uint value,
38*4882a593Smuzhiyun enum koneplus_control_requests request)
39*4882a593Smuzhiyun {
40*4882a593Smuzhiyun struct roccat_common2_control control;
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun if ((request == KONEPLUS_CONTROL_REQUEST_PROFILE_SETTINGS ||
43*4882a593Smuzhiyun request == KONEPLUS_CONTROL_REQUEST_PROFILE_BUTTONS) &&
44*4882a593Smuzhiyun value > 4)
45*4882a593Smuzhiyun return -EINVAL;
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun control.command = ROCCAT_COMMON_COMMAND_CONTROL;
48*4882a593Smuzhiyun control.value = value;
49*4882a593Smuzhiyun control.request = request;
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun return roccat_common2_send_with_status(usb_dev,
52*4882a593Smuzhiyun ROCCAT_COMMON_COMMAND_CONTROL,
53*4882a593Smuzhiyun &control, sizeof(struct roccat_common2_control));
54*4882a593Smuzhiyun }
55*4882a593Smuzhiyun
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun /* retval is 0-4 on success, < 0 on error */
koneplus_get_actual_profile(struct usb_device * usb_dev)58*4882a593Smuzhiyun static int koneplus_get_actual_profile(struct usb_device *usb_dev)
59*4882a593Smuzhiyun {
60*4882a593Smuzhiyun struct koneplus_actual_profile buf;
61*4882a593Smuzhiyun int retval;
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun retval = roccat_common2_receive(usb_dev, KONEPLUS_COMMAND_ACTUAL_PROFILE,
64*4882a593Smuzhiyun &buf, KONEPLUS_SIZE_ACTUAL_PROFILE);
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun return retval ? retval : buf.actual_profile;
67*4882a593Smuzhiyun }
68*4882a593Smuzhiyun
koneplus_set_actual_profile(struct usb_device * usb_dev,int new_profile)69*4882a593Smuzhiyun static int koneplus_set_actual_profile(struct usb_device *usb_dev,
70*4882a593Smuzhiyun int new_profile)
71*4882a593Smuzhiyun {
72*4882a593Smuzhiyun struct koneplus_actual_profile buf;
73*4882a593Smuzhiyun
74*4882a593Smuzhiyun buf.command = KONEPLUS_COMMAND_ACTUAL_PROFILE;
75*4882a593Smuzhiyun buf.size = KONEPLUS_SIZE_ACTUAL_PROFILE;
76*4882a593Smuzhiyun buf.actual_profile = new_profile;
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun return roccat_common2_send_with_status(usb_dev,
79*4882a593Smuzhiyun KONEPLUS_COMMAND_ACTUAL_PROFILE,
80*4882a593Smuzhiyun &buf, KONEPLUS_SIZE_ACTUAL_PROFILE);
81*4882a593Smuzhiyun }
82*4882a593Smuzhiyun
koneplus_sysfs_read(struct file * fp,struct kobject * kobj,char * buf,loff_t off,size_t count,size_t real_size,uint command)83*4882a593Smuzhiyun static ssize_t koneplus_sysfs_read(struct file *fp, struct kobject *kobj,
84*4882a593Smuzhiyun char *buf, loff_t off, size_t count,
85*4882a593Smuzhiyun size_t real_size, uint command)
86*4882a593Smuzhiyun {
87*4882a593Smuzhiyun struct device *dev = kobj_to_dev(kobj)->parent->parent;
88*4882a593Smuzhiyun struct koneplus_device *koneplus = hid_get_drvdata(dev_get_drvdata(dev));
89*4882a593Smuzhiyun struct usb_device *usb_dev = interface_to_usbdev(to_usb_interface(dev));
90*4882a593Smuzhiyun int retval;
91*4882a593Smuzhiyun
92*4882a593Smuzhiyun if (off >= real_size)
93*4882a593Smuzhiyun return 0;
94*4882a593Smuzhiyun
95*4882a593Smuzhiyun if (off != 0 || count != real_size)
96*4882a593Smuzhiyun return -EINVAL;
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun mutex_lock(&koneplus->koneplus_lock);
99*4882a593Smuzhiyun retval = roccat_common2_receive(usb_dev, command, buf, real_size);
100*4882a593Smuzhiyun mutex_unlock(&koneplus->koneplus_lock);
101*4882a593Smuzhiyun
102*4882a593Smuzhiyun if (retval)
103*4882a593Smuzhiyun return retval;
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun return real_size;
106*4882a593Smuzhiyun }
107*4882a593Smuzhiyun
koneplus_sysfs_write(struct file * fp,struct kobject * kobj,void const * buf,loff_t off,size_t count,size_t real_size,uint command)108*4882a593Smuzhiyun static ssize_t koneplus_sysfs_write(struct file *fp, struct kobject *kobj,
109*4882a593Smuzhiyun void const *buf, loff_t off, size_t count,
110*4882a593Smuzhiyun size_t real_size, uint command)
111*4882a593Smuzhiyun {
112*4882a593Smuzhiyun struct device *dev = kobj_to_dev(kobj)->parent->parent;
113*4882a593Smuzhiyun struct koneplus_device *koneplus = hid_get_drvdata(dev_get_drvdata(dev));
114*4882a593Smuzhiyun struct usb_device *usb_dev = interface_to_usbdev(to_usb_interface(dev));
115*4882a593Smuzhiyun int retval;
116*4882a593Smuzhiyun
117*4882a593Smuzhiyun if (off != 0 || count != real_size)
118*4882a593Smuzhiyun return -EINVAL;
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun mutex_lock(&koneplus->koneplus_lock);
121*4882a593Smuzhiyun retval = roccat_common2_send_with_status(usb_dev, command,
122*4882a593Smuzhiyun buf, real_size);
123*4882a593Smuzhiyun mutex_unlock(&koneplus->koneplus_lock);
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun if (retval)
126*4882a593Smuzhiyun return retval;
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun return real_size;
129*4882a593Smuzhiyun }
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun #define KONEPLUS_SYSFS_W(thingy, THINGY) \
132*4882a593Smuzhiyun static ssize_t koneplus_sysfs_write_ ## thingy(struct file *fp, \
133*4882a593Smuzhiyun struct kobject *kobj, struct bin_attribute *attr, char *buf, \
134*4882a593Smuzhiyun loff_t off, size_t count) \
135*4882a593Smuzhiyun { \
136*4882a593Smuzhiyun return koneplus_sysfs_write(fp, kobj, buf, off, count, \
137*4882a593Smuzhiyun KONEPLUS_SIZE_ ## THINGY, KONEPLUS_COMMAND_ ## THINGY); \
138*4882a593Smuzhiyun }
139*4882a593Smuzhiyun
140*4882a593Smuzhiyun #define KONEPLUS_SYSFS_R(thingy, THINGY) \
141*4882a593Smuzhiyun static ssize_t koneplus_sysfs_read_ ## thingy(struct file *fp, \
142*4882a593Smuzhiyun struct kobject *kobj, struct bin_attribute *attr, char *buf, \
143*4882a593Smuzhiyun loff_t off, size_t count) \
144*4882a593Smuzhiyun { \
145*4882a593Smuzhiyun return koneplus_sysfs_read(fp, kobj, buf, off, count, \
146*4882a593Smuzhiyun KONEPLUS_SIZE_ ## THINGY, KONEPLUS_COMMAND_ ## THINGY); \
147*4882a593Smuzhiyun }
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun #define KONEPLUS_SYSFS_RW(thingy, THINGY) \
150*4882a593Smuzhiyun KONEPLUS_SYSFS_W(thingy, THINGY) \
151*4882a593Smuzhiyun KONEPLUS_SYSFS_R(thingy, THINGY)
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun #define KONEPLUS_BIN_ATTRIBUTE_RW(thingy, THINGY) \
154*4882a593Smuzhiyun KONEPLUS_SYSFS_RW(thingy, THINGY); \
155*4882a593Smuzhiyun static struct bin_attribute bin_attr_##thingy = { \
156*4882a593Smuzhiyun .attr = { .name = #thingy, .mode = 0660 }, \
157*4882a593Smuzhiyun .size = KONEPLUS_SIZE_ ## THINGY, \
158*4882a593Smuzhiyun .read = koneplus_sysfs_read_ ## thingy, \
159*4882a593Smuzhiyun .write = koneplus_sysfs_write_ ## thingy \
160*4882a593Smuzhiyun }
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun #define KONEPLUS_BIN_ATTRIBUTE_R(thingy, THINGY) \
163*4882a593Smuzhiyun KONEPLUS_SYSFS_R(thingy, THINGY); \
164*4882a593Smuzhiyun static struct bin_attribute bin_attr_##thingy = { \
165*4882a593Smuzhiyun .attr = { .name = #thingy, .mode = 0440 }, \
166*4882a593Smuzhiyun .size = KONEPLUS_SIZE_ ## THINGY, \
167*4882a593Smuzhiyun .read = koneplus_sysfs_read_ ## thingy, \
168*4882a593Smuzhiyun }
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun #define KONEPLUS_BIN_ATTRIBUTE_W(thingy, THINGY) \
171*4882a593Smuzhiyun KONEPLUS_SYSFS_W(thingy, THINGY); \
172*4882a593Smuzhiyun static struct bin_attribute bin_attr_##thingy = { \
173*4882a593Smuzhiyun .attr = { .name = #thingy, .mode = 0220 }, \
174*4882a593Smuzhiyun .size = KONEPLUS_SIZE_ ## THINGY, \
175*4882a593Smuzhiyun .write = koneplus_sysfs_write_ ## thingy \
176*4882a593Smuzhiyun }
177*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_W(control, CONTROL);
178*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_W(talk, TALK);
179*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_W(macro, MACRO);
180*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_R(tcu_image, TCU_IMAGE);
181*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_RW(info, INFO);
182*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_RW(sensor, SENSOR);
183*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_RW(tcu, TCU);
184*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_RW(profile_settings, PROFILE_SETTINGS);
185*4882a593Smuzhiyun KONEPLUS_BIN_ATTRIBUTE_RW(profile_buttons, PROFILE_BUTTONS);
186*4882a593Smuzhiyun
koneplus_sysfs_read_profilex_settings(struct file * fp,struct kobject * kobj,struct bin_attribute * attr,char * buf,loff_t off,size_t count)187*4882a593Smuzhiyun static ssize_t koneplus_sysfs_read_profilex_settings(struct file *fp,
188*4882a593Smuzhiyun struct kobject *kobj, struct bin_attribute *attr, char *buf,
189*4882a593Smuzhiyun loff_t off, size_t count)
190*4882a593Smuzhiyun {
191*4882a593Smuzhiyun struct device *dev = kobj_to_dev(kobj)->parent->parent;
192*4882a593Smuzhiyun struct usb_device *usb_dev = interface_to_usbdev(to_usb_interface(dev));
193*4882a593Smuzhiyun ssize_t retval;
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun retval = koneplus_send_control(usb_dev, *(uint *)(attr->private),
196*4882a593Smuzhiyun KONEPLUS_CONTROL_REQUEST_PROFILE_SETTINGS);
197*4882a593Smuzhiyun if (retval)
198*4882a593Smuzhiyun return retval;
199*4882a593Smuzhiyun
200*4882a593Smuzhiyun return koneplus_sysfs_read(fp, kobj, buf, off, count,
201*4882a593Smuzhiyun KONEPLUS_SIZE_PROFILE_SETTINGS,
202*4882a593Smuzhiyun KONEPLUS_COMMAND_PROFILE_SETTINGS);
203*4882a593Smuzhiyun }
204*4882a593Smuzhiyun
koneplus_sysfs_read_profilex_buttons(struct file * fp,struct kobject * kobj,struct bin_attribute * attr,char * buf,loff_t off,size_t count)205*4882a593Smuzhiyun static ssize_t koneplus_sysfs_read_profilex_buttons(struct file *fp,
206*4882a593Smuzhiyun struct kobject *kobj, struct bin_attribute *attr, char *buf,
207*4882a593Smuzhiyun loff_t off, size_t count)
208*4882a593Smuzhiyun {
209*4882a593Smuzhiyun struct device *dev = kobj_to_dev(kobj)->parent->parent;
210*4882a593Smuzhiyun struct usb_device *usb_dev = interface_to_usbdev(to_usb_interface(dev));
211*4882a593Smuzhiyun ssize_t retval;
212*4882a593Smuzhiyun
213*4882a593Smuzhiyun retval = koneplus_send_control(usb_dev, *(uint *)(attr->private),
214*4882a593Smuzhiyun KONEPLUS_CONTROL_REQUEST_PROFILE_BUTTONS);
215*4882a593Smuzhiyun if (retval)
216*4882a593Smuzhiyun return retval;
217*4882a593Smuzhiyun
218*4882a593Smuzhiyun return koneplus_sysfs_read(fp, kobj, buf, off, count,
219*4882a593Smuzhiyun KONEPLUS_SIZE_PROFILE_BUTTONS,
220*4882a593Smuzhiyun KONEPLUS_COMMAND_PROFILE_BUTTONS);
221*4882a593Smuzhiyun }
222*4882a593Smuzhiyun
223*4882a593Smuzhiyun #define PROFILE_ATTR(number) \
224*4882a593Smuzhiyun static struct bin_attribute bin_attr_profile##number##_settings = { \
225*4882a593Smuzhiyun .attr = { .name = "profile" #number "_settings", .mode = 0440 }, \
226*4882a593Smuzhiyun .size = KONEPLUS_SIZE_PROFILE_SETTINGS, \
227*4882a593Smuzhiyun .read = koneplus_sysfs_read_profilex_settings, \
228*4882a593Smuzhiyun .private = &profile_numbers[number-1], \
229*4882a593Smuzhiyun }; \
230*4882a593Smuzhiyun static struct bin_attribute bin_attr_profile##number##_buttons = { \
231*4882a593Smuzhiyun .attr = { .name = "profile" #number "_buttons", .mode = 0440 }, \
232*4882a593Smuzhiyun .size = KONEPLUS_SIZE_PROFILE_BUTTONS, \
233*4882a593Smuzhiyun .read = koneplus_sysfs_read_profilex_buttons, \
234*4882a593Smuzhiyun .private = &profile_numbers[number-1], \
235*4882a593Smuzhiyun };
236*4882a593Smuzhiyun PROFILE_ATTR(1);
237*4882a593Smuzhiyun PROFILE_ATTR(2);
238*4882a593Smuzhiyun PROFILE_ATTR(3);
239*4882a593Smuzhiyun PROFILE_ATTR(4);
240*4882a593Smuzhiyun PROFILE_ATTR(5);
241*4882a593Smuzhiyun
koneplus_sysfs_show_actual_profile(struct device * dev,struct device_attribute * attr,char * buf)242*4882a593Smuzhiyun static ssize_t koneplus_sysfs_show_actual_profile(struct device *dev,
243*4882a593Smuzhiyun struct device_attribute *attr, char *buf)
244*4882a593Smuzhiyun {
245*4882a593Smuzhiyun struct koneplus_device *koneplus =
246*4882a593Smuzhiyun hid_get_drvdata(dev_get_drvdata(dev->parent->parent));
247*4882a593Smuzhiyun return snprintf(buf, PAGE_SIZE, "%d\n", koneplus->actual_profile);
248*4882a593Smuzhiyun }
249*4882a593Smuzhiyun
koneplus_sysfs_set_actual_profile(struct device * dev,struct device_attribute * attr,char const * buf,size_t size)250*4882a593Smuzhiyun static ssize_t koneplus_sysfs_set_actual_profile(struct device *dev,
251*4882a593Smuzhiyun struct device_attribute *attr, char const *buf, size_t size)
252*4882a593Smuzhiyun {
253*4882a593Smuzhiyun struct koneplus_device *koneplus;
254*4882a593Smuzhiyun struct usb_device *usb_dev;
255*4882a593Smuzhiyun unsigned long profile;
256*4882a593Smuzhiyun int retval;
257*4882a593Smuzhiyun struct koneplus_roccat_report roccat_report;
258*4882a593Smuzhiyun
259*4882a593Smuzhiyun dev = dev->parent->parent;
260*4882a593Smuzhiyun koneplus = hid_get_drvdata(dev_get_drvdata(dev));
261*4882a593Smuzhiyun usb_dev = interface_to_usbdev(to_usb_interface(dev));
262*4882a593Smuzhiyun
263*4882a593Smuzhiyun retval = kstrtoul(buf, 10, &profile);
264*4882a593Smuzhiyun if (retval)
265*4882a593Smuzhiyun return retval;
266*4882a593Smuzhiyun
267*4882a593Smuzhiyun if (profile > 4)
268*4882a593Smuzhiyun return -EINVAL;
269*4882a593Smuzhiyun
270*4882a593Smuzhiyun mutex_lock(&koneplus->koneplus_lock);
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun retval = koneplus_set_actual_profile(usb_dev, profile);
273*4882a593Smuzhiyun if (retval) {
274*4882a593Smuzhiyun mutex_unlock(&koneplus->koneplus_lock);
275*4882a593Smuzhiyun return retval;
276*4882a593Smuzhiyun }
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun koneplus_profile_activated(koneplus, profile);
279*4882a593Smuzhiyun
280*4882a593Smuzhiyun roccat_report.type = KONEPLUS_MOUSE_REPORT_BUTTON_TYPE_PROFILE;
281*4882a593Smuzhiyun roccat_report.data1 = profile + 1;
282*4882a593Smuzhiyun roccat_report.data2 = 0;
283*4882a593Smuzhiyun roccat_report.profile = profile + 1;
284*4882a593Smuzhiyun roccat_report_event(koneplus->chrdev_minor,
285*4882a593Smuzhiyun (uint8_t const *)&roccat_report);
286*4882a593Smuzhiyun
287*4882a593Smuzhiyun mutex_unlock(&koneplus->koneplus_lock);
288*4882a593Smuzhiyun
289*4882a593Smuzhiyun return size;
290*4882a593Smuzhiyun }
291*4882a593Smuzhiyun static DEVICE_ATTR(actual_profile, 0660,
292*4882a593Smuzhiyun koneplus_sysfs_show_actual_profile,
293*4882a593Smuzhiyun koneplus_sysfs_set_actual_profile);
294*4882a593Smuzhiyun static DEVICE_ATTR(startup_profile, 0660,
295*4882a593Smuzhiyun koneplus_sysfs_show_actual_profile,
296*4882a593Smuzhiyun koneplus_sysfs_set_actual_profile);
297*4882a593Smuzhiyun
koneplus_sysfs_show_firmware_version(struct device * dev,struct device_attribute * attr,char * buf)298*4882a593Smuzhiyun static ssize_t koneplus_sysfs_show_firmware_version(struct device *dev,
299*4882a593Smuzhiyun struct device_attribute *attr, char *buf)
300*4882a593Smuzhiyun {
301*4882a593Smuzhiyun struct koneplus_device *koneplus;
302*4882a593Smuzhiyun struct usb_device *usb_dev;
303*4882a593Smuzhiyun struct koneplus_info info;
304*4882a593Smuzhiyun
305*4882a593Smuzhiyun dev = dev->parent->parent;
306*4882a593Smuzhiyun koneplus = hid_get_drvdata(dev_get_drvdata(dev));
307*4882a593Smuzhiyun usb_dev = interface_to_usbdev(to_usb_interface(dev));
308*4882a593Smuzhiyun
309*4882a593Smuzhiyun mutex_lock(&koneplus->koneplus_lock);
310*4882a593Smuzhiyun roccat_common2_receive(usb_dev, KONEPLUS_COMMAND_INFO,
311*4882a593Smuzhiyun &info, KONEPLUS_SIZE_INFO);
312*4882a593Smuzhiyun mutex_unlock(&koneplus->koneplus_lock);
313*4882a593Smuzhiyun
314*4882a593Smuzhiyun return snprintf(buf, PAGE_SIZE, "%d\n", info.firmware_version);
315*4882a593Smuzhiyun }
316*4882a593Smuzhiyun static DEVICE_ATTR(firmware_version, 0440,
317*4882a593Smuzhiyun koneplus_sysfs_show_firmware_version, NULL);
318*4882a593Smuzhiyun
319*4882a593Smuzhiyun static struct attribute *koneplus_attrs[] = {
320*4882a593Smuzhiyun &dev_attr_actual_profile.attr,
321*4882a593Smuzhiyun &dev_attr_startup_profile.attr,
322*4882a593Smuzhiyun &dev_attr_firmware_version.attr,
323*4882a593Smuzhiyun NULL,
324*4882a593Smuzhiyun };
325*4882a593Smuzhiyun
326*4882a593Smuzhiyun static struct bin_attribute *koneplus_bin_attributes[] = {
327*4882a593Smuzhiyun &bin_attr_control,
328*4882a593Smuzhiyun &bin_attr_talk,
329*4882a593Smuzhiyun &bin_attr_macro,
330*4882a593Smuzhiyun &bin_attr_tcu_image,
331*4882a593Smuzhiyun &bin_attr_info,
332*4882a593Smuzhiyun &bin_attr_sensor,
333*4882a593Smuzhiyun &bin_attr_tcu,
334*4882a593Smuzhiyun &bin_attr_profile_settings,
335*4882a593Smuzhiyun &bin_attr_profile_buttons,
336*4882a593Smuzhiyun &bin_attr_profile1_settings,
337*4882a593Smuzhiyun &bin_attr_profile2_settings,
338*4882a593Smuzhiyun &bin_attr_profile3_settings,
339*4882a593Smuzhiyun &bin_attr_profile4_settings,
340*4882a593Smuzhiyun &bin_attr_profile5_settings,
341*4882a593Smuzhiyun &bin_attr_profile1_buttons,
342*4882a593Smuzhiyun &bin_attr_profile2_buttons,
343*4882a593Smuzhiyun &bin_attr_profile3_buttons,
344*4882a593Smuzhiyun &bin_attr_profile4_buttons,
345*4882a593Smuzhiyun &bin_attr_profile5_buttons,
346*4882a593Smuzhiyun NULL,
347*4882a593Smuzhiyun };
348*4882a593Smuzhiyun
349*4882a593Smuzhiyun static const struct attribute_group koneplus_group = {
350*4882a593Smuzhiyun .attrs = koneplus_attrs,
351*4882a593Smuzhiyun .bin_attrs = koneplus_bin_attributes,
352*4882a593Smuzhiyun };
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun static const struct attribute_group *koneplus_groups[] = {
355*4882a593Smuzhiyun &koneplus_group,
356*4882a593Smuzhiyun NULL,
357*4882a593Smuzhiyun };
358*4882a593Smuzhiyun
koneplus_init_koneplus_device_struct(struct usb_device * usb_dev,struct koneplus_device * koneplus)359*4882a593Smuzhiyun static int koneplus_init_koneplus_device_struct(struct usb_device *usb_dev,
360*4882a593Smuzhiyun struct koneplus_device *koneplus)
361*4882a593Smuzhiyun {
362*4882a593Smuzhiyun int retval;
363*4882a593Smuzhiyun
364*4882a593Smuzhiyun mutex_init(&koneplus->koneplus_lock);
365*4882a593Smuzhiyun
366*4882a593Smuzhiyun retval = koneplus_get_actual_profile(usb_dev);
367*4882a593Smuzhiyun if (retval < 0)
368*4882a593Smuzhiyun return retval;
369*4882a593Smuzhiyun koneplus_profile_activated(koneplus, retval);
370*4882a593Smuzhiyun
371*4882a593Smuzhiyun return 0;
372*4882a593Smuzhiyun }
373*4882a593Smuzhiyun
koneplus_init_specials(struct hid_device * hdev)374*4882a593Smuzhiyun static int koneplus_init_specials(struct hid_device *hdev)
375*4882a593Smuzhiyun {
376*4882a593Smuzhiyun struct usb_interface *intf = to_usb_interface(hdev->dev.parent);
377*4882a593Smuzhiyun struct usb_device *usb_dev = interface_to_usbdev(intf);
378*4882a593Smuzhiyun struct koneplus_device *koneplus;
379*4882a593Smuzhiyun int retval;
380*4882a593Smuzhiyun
381*4882a593Smuzhiyun if (intf->cur_altsetting->desc.bInterfaceProtocol
382*4882a593Smuzhiyun == USB_INTERFACE_PROTOCOL_MOUSE) {
383*4882a593Smuzhiyun
384*4882a593Smuzhiyun koneplus = kzalloc(sizeof(*koneplus), GFP_KERNEL);
385*4882a593Smuzhiyun if (!koneplus) {
386*4882a593Smuzhiyun hid_err(hdev, "can't alloc device descriptor\n");
387*4882a593Smuzhiyun return -ENOMEM;
388*4882a593Smuzhiyun }
389*4882a593Smuzhiyun hid_set_drvdata(hdev, koneplus);
390*4882a593Smuzhiyun
391*4882a593Smuzhiyun retval = koneplus_init_koneplus_device_struct(usb_dev, koneplus);
392*4882a593Smuzhiyun if (retval) {
393*4882a593Smuzhiyun hid_err(hdev, "couldn't init struct koneplus_device\n");
394*4882a593Smuzhiyun goto exit_free;
395*4882a593Smuzhiyun }
396*4882a593Smuzhiyun
397*4882a593Smuzhiyun retval = roccat_connect(koneplus_class, hdev,
398*4882a593Smuzhiyun sizeof(struct koneplus_roccat_report));
399*4882a593Smuzhiyun if (retval < 0) {
400*4882a593Smuzhiyun hid_err(hdev, "couldn't init char dev\n");
401*4882a593Smuzhiyun } else {
402*4882a593Smuzhiyun koneplus->chrdev_minor = retval;
403*4882a593Smuzhiyun koneplus->roccat_claimed = 1;
404*4882a593Smuzhiyun }
405*4882a593Smuzhiyun } else {
406*4882a593Smuzhiyun hid_set_drvdata(hdev, NULL);
407*4882a593Smuzhiyun }
408*4882a593Smuzhiyun
409*4882a593Smuzhiyun return 0;
410*4882a593Smuzhiyun exit_free:
411*4882a593Smuzhiyun kfree(koneplus);
412*4882a593Smuzhiyun return retval;
413*4882a593Smuzhiyun }
414*4882a593Smuzhiyun
koneplus_remove_specials(struct hid_device * hdev)415*4882a593Smuzhiyun static void koneplus_remove_specials(struct hid_device *hdev)
416*4882a593Smuzhiyun {
417*4882a593Smuzhiyun struct usb_interface *intf = to_usb_interface(hdev->dev.parent);
418*4882a593Smuzhiyun struct koneplus_device *koneplus;
419*4882a593Smuzhiyun
420*4882a593Smuzhiyun if (intf->cur_altsetting->desc.bInterfaceProtocol
421*4882a593Smuzhiyun == USB_INTERFACE_PROTOCOL_MOUSE) {
422*4882a593Smuzhiyun koneplus = hid_get_drvdata(hdev);
423*4882a593Smuzhiyun if (koneplus->roccat_claimed)
424*4882a593Smuzhiyun roccat_disconnect(koneplus->chrdev_minor);
425*4882a593Smuzhiyun kfree(koneplus);
426*4882a593Smuzhiyun }
427*4882a593Smuzhiyun }
428*4882a593Smuzhiyun
koneplus_probe(struct hid_device * hdev,const struct hid_device_id * id)429*4882a593Smuzhiyun static int koneplus_probe(struct hid_device *hdev,
430*4882a593Smuzhiyun const struct hid_device_id *id)
431*4882a593Smuzhiyun {
432*4882a593Smuzhiyun int retval;
433*4882a593Smuzhiyun
434*4882a593Smuzhiyun if (!hid_is_usb(hdev))
435*4882a593Smuzhiyun return -EINVAL;
436*4882a593Smuzhiyun
437*4882a593Smuzhiyun retval = hid_parse(hdev);
438*4882a593Smuzhiyun if (retval) {
439*4882a593Smuzhiyun hid_err(hdev, "parse failed\n");
440*4882a593Smuzhiyun goto exit;
441*4882a593Smuzhiyun }
442*4882a593Smuzhiyun
443*4882a593Smuzhiyun retval = hid_hw_start(hdev, HID_CONNECT_DEFAULT);
444*4882a593Smuzhiyun if (retval) {
445*4882a593Smuzhiyun hid_err(hdev, "hw start failed\n");
446*4882a593Smuzhiyun goto exit;
447*4882a593Smuzhiyun }
448*4882a593Smuzhiyun
449*4882a593Smuzhiyun retval = koneplus_init_specials(hdev);
450*4882a593Smuzhiyun if (retval) {
451*4882a593Smuzhiyun hid_err(hdev, "couldn't install mouse\n");
452*4882a593Smuzhiyun goto exit_stop;
453*4882a593Smuzhiyun }
454*4882a593Smuzhiyun
455*4882a593Smuzhiyun return 0;
456*4882a593Smuzhiyun
457*4882a593Smuzhiyun exit_stop:
458*4882a593Smuzhiyun hid_hw_stop(hdev);
459*4882a593Smuzhiyun exit:
460*4882a593Smuzhiyun return retval;
461*4882a593Smuzhiyun }
462*4882a593Smuzhiyun
koneplus_remove(struct hid_device * hdev)463*4882a593Smuzhiyun static void koneplus_remove(struct hid_device *hdev)
464*4882a593Smuzhiyun {
465*4882a593Smuzhiyun koneplus_remove_specials(hdev);
466*4882a593Smuzhiyun hid_hw_stop(hdev);
467*4882a593Smuzhiyun }
468*4882a593Smuzhiyun
koneplus_keep_values_up_to_date(struct koneplus_device * koneplus,u8 const * data)469*4882a593Smuzhiyun static void koneplus_keep_values_up_to_date(struct koneplus_device *koneplus,
470*4882a593Smuzhiyun u8 const *data)
471*4882a593Smuzhiyun {
472*4882a593Smuzhiyun struct koneplus_mouse_report_button const *button_report;
473*4882a593Smuzhiyun
474*4882a593Smuzhiyun switch (data[0]) {
475*4882a593Smuzhiyun case KONEPLUS_MOUSE_REPORT_NUMBER_BUTTON:
476*4882a593Smuzhiyun button_report = (struct koneplus_mouse_report_button const *)data;
477*4882a593Smuzhiyun switch (button_report->type) {
478*4882a593Smuzhiyun case KONEPLUS_MOUSE_REPORT_BUTTON_TYPE_PROFILE:
479*4882a593Smuzhiyun koneplus_profile_activated(koneplus, button_report->data1 - 1);
480*4882a593Smuzhiyun break;
481*4882a593Smuzhiyun }
482*4882a593Smuzhiyun break;
483*4882a593Smuzhiyun }
484*4882a593Smuzhiyun }
485*4882a593Smuzhiyun
koneplus_report_to_chrdev(struct koneplus_device const * koneplus,u8 const * data)486*4882a593Smuzhiyun static void koneplus_report_to_chrdev(struct koneplus_device const *koneplus,
487*4882a593Smuzhiyun u8 const *data)
488*4882a593Smuzhiyun {
489*4882a593Smuzhiyun struct koneplus_roccat_report roccat_report;
490*4882a593Smuzhiyun struct koneplus_mouse_report_button const *button_report;
491*4882a593Smuzhiyun
492*4882a593Smuzhiyun if (data[0] != KONEPLUS_MOUSE_REPORT_NUMBER_BUTTON)
493*4882a593Smuzhiyun return;
494*4882a593Smuzhiyun
495*4882a593Smuzhiyun button_report = (struct koneplus_mouse_report_button const *)data;
496*4882a593Smuzhiyun
497*4882a593Smuzhiyun if ((button_report->type == KONEPLUS_MOUSE_REPORT_BUTTON_TYPE_QUICKLAUNCH ||
498*4882a593Smuzhiyun button_report->type == KONEPLUS_MOUSE_REPORT_BUTTON_TYPE_TIMER) &&
499*4882a593Smuzhiyun button_report->data2 != KONEPLUS_MOUSE_REPORT_BUTTON_ACTION_PRESS)
500*4882a593Smuzhiyun return;
501*4882a593Smuzhiyun
502*4882a593Smuzhiyun roccat_report.type = button_report->type;
503*4882a593Smuzhiyun roccat_report.data1 = button_report->data1;
504*4882a593Smuzhiyun roccat_report.data2 = button_report->data2;
505*4882a593Smuzhiyun roccat_report.profile = koneplus->actual_profile + 1;
506*4882a593Smuzhiyun roccat_report_event(koneplus->chrdev_minor,
507*4882a593Smuzhiyun (uint8_t const *)&roccat_report);
508*4882a593Smuzhiyun }
509*4882a593Smuzhiyun
koneplus_raw_event(struct hid_device * hdev,struct hid_report * report,u8 * data,int size)510*4882a593Smuzhiyun static int koneplus_raw_event(struct hid_device *hdev,
511*4882a593Smuzhiyun struct hid_report *report, u8 *data, int size)
512*4882a593Smuzhiyun {
513*4882a593Smuzhiyun struct usb_interface *intf = to_usb_interface(hdev->dev.parent);
514*4882a593Smuzhiyun struct koneplus_device *koneplus = hid_get_drvdata(hdev);
515*4882a593Smuzhiyun
516*4882a593Smuzhiyun if (intf->cur_altsetting->desc.bInterfaceProtocol
517*4882a593Smuzhiyun != USB_INTERFACE_PROTOCOL_MOUSE)
518*4882a593Smuzhiyun return 0;
519*4882a593Smuzhiyun
520*4882a593Smuzhiyun if (koneplus == NULL)
521*4882a593Smuzhiyun return 0;
522*4882a593Smuzhiyun
523*4882a593Smuzhiyun koneplus_keep_values_up_to_date(koneplus, data);
524*4882a593Smuzhiyun
525*4882a593Smuzhiyun if (koneplus->roccat_claimed)
526*4882a593Smuzhiyun koneplus_report_to_chrdev(koneplus, data);
527*4882a593Smuzhiyun
528*4882a593Smuzhiyun return 0;
529*4882a593Smuzhiyun }
530*4882a593Smuzhiyun
531*4882a593Smuzhiyun static const struct hid_device_id koneplus_devices[] = {
532*4882a593Smuzhiyun { HID_USB_DEVICE(USB_VENDOR_ID_ROCCAT, USB_DEVICE_ID_ROCCAT_KONEPLUS) },
533*4882a593Smuzhiyun { HID_USB_DEVICE(USB_VENDOR_ID_ROCCAT, USB_DEVICE_ID_ROCCAT_KONEXTD) },
534*4882a593Smuzhiyun { }
535*4882a593Smuzhiyun };
536*4882a593Smuzhiyun
537*4882a593Smuzhiyun MODULE_DEVICE_TABLE(hid, koneplus_devices);
538*4882a593Smuzhiyun
539*4882a593Smuzhiyun static struct hid_driver koneplus_driver = {
540*4882a593Smuzhiyun .name = "koneplus",
541*4882a593Smuzhiyun .id_table = koneplus_devices,
542*4882a593Smuzhiyun .probe = koneplus_probe,
543*4882a593Smuzhiyun .remove = koneplus_remove,
544*4882a593Smuzhiyun .raw_event = koneplus_raw_event
545*4882a593Smuzhiyun };
546*4882a593Smuzhiyun
koneplus_init(void)547*4882a593Smuzhiyun static int __init koneplus_init(void)
548*4882a593Smuzhiyun {
549*4882a593Smuzhiyun int retval;
550*4882a593Smuzhiyun
551*4882a593Smuzhiyun /* class name has to be same as driver name */
552*4882a593Smuzhiyun koneplus_class = class_create(THIS_MODULE, "koneplus");
553*4882a593Smuzhiyun if (IS_ERR(koneplus_class))
554*4882a593Smuzhiyun return PTR_ERR(koneplus_class);
555*4882a593Smuzhiyun koneplus_class->dev_groups = koneplus_groups;
556*4882a593Smuzhiyun
557*4882a593Smuzhiyun retval = hid_register_driver(&koneplus_driver);
558*4882a593Smuzhiyun if (retval)
559*4882a593Smuzhiyun class_destroy(koneplus_class);
560*4882a593Smuzhiyun return retval;
561*4882a593Smuzhiyun }
562*4882a593Smuzhiyun
koneplus_exit(void)563*4882a593Smuzhiyun static void __exit koneplus_exit(void)
564*4882a593Smuzhiyun {
565*4882a593Smuzhiyun hid_unregister_driver(&koneplus_driver);
566*4882a593Smuzhiyun class_destroy(koneplus_class);
567*4882a593Smuzhiyun }
568*4882a593Smuzhiyun
569*4882a593Smuzhiyun module_init(koneplus_init);
570*4882a593Smuzhiyun module_exit(koneplus_exit);
571*4882a593Smuzhiyun
572*4882a593Smuzhiyun MODULE_AUTHOR("Stefan Achatz");
573*4882a593Smuzhiyun MODULE_DESCRIPTION("USB Roccat Kone[+]/XTD driver");
574*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
575